static_assert( __cplusplus >= 2020'02 ); #ifndef __DW_TEXTBLOCK_HH__ #define __DW_TEXTBLOCK_HH__ #include #include #include "regardingborder.hh" #include "../lout/misc.hh" // These were used when improved line breaking and hyphenation were implemented. // Should be, bit by bit, replaced by RTFL (see ../lout/debug.hh). #define PRINTF(...) #define PUTCHAR(ch) #ifdef DBG_RTFL # define DEBUG #endif namespace dw { /** * \brief A Widget for rendering text blocks, i.e. paragraphs or sequences * of paragraphs. * *
Info: Some (not so) recent changes, line breaking * and hyphenation, have not yet been incorporated into this * documentation. See \ref dw-line-breaking.
* *

Signals

* * dw::Textblock uses the signals defined in * dw::core::Layout::LinkReceiver, related to links. The coordinates are * always -1. * * *

Collapsing Spaces

* *
Info: Collapsing spaces are deprecated, in favor of * collapsing margins (see below).
* * The idea behind this is that every paragraph has a specific vertical * space around and that they are combined to one space, according to * rules stated below. A paragraph consists either of the lines between * two paragraph breaks within a dw::Textblock, or of a dw::Textblock * within a dw::Textblock, in a single line; the latter is used for * indented boxes and list items. * * The rules: * *
    *
  1. If a paragraph is following by another, the space between them is the * maximum of both box spaces: * * \image html dw-textblock-collapsing-spaces-1-1.png * * are combined like this: * * \image html dw-textblock-collapsing-spaces-1-2.png * *
  2. a) If one paragraph is the first paragraph within another, the upper * space of these paragraphs collapse. b) The analogue is the case for the * last box: * * \image html dw-textblock-collapsing-spaces-2-1.png * * If B and C are put into A, the result is: * * \image html dw-textblock-collapsing-spaces-2-2.png *
* * For achieving this, there are some features of dw::Textblock: * * * * *

Collapsing Margins

* * Collapsing margins, as defined in the CSS2 specification, are, * supported in addition to collapsing spaces. Also, spaces and * margins collapse themselves. I. e., the space between two * paragraphs is the maximum of the space calculated as described in * "Collapsing Spaces" and the space calculated according to the rules * for collapsing margins. * * (This is an intermediate hybrid state, collapsing spaces are used in * the current version of dillo, while I implemented collapsing margins * for the CSS prototype and integrated it already into the main trunk. For * a pure CSS-based dillo, collapsing spaces will not be needed anymore, and * may be removed for simplicity.) * * Currently implemented cases: * * - The top margin of of the textblock widget and the top margin of * the first line box (based on widgets in the first line) collapse. * * - The bottom margin of of the textblock widget and the bottom * margin of the last line box collapse. * * - The bottom margin of a line box and the top margin of the * following line collapse. Here, the break space is regarded, too. * * Open issues: * * - Only the value of Style::margin is regarded, not the result of * the collapsing itself. For the widgets A, B (child of A), and C * (child of B), the effective margin of A is the maximum of the * *style* margins of A and B, while the effective margin of B (the * collapsed margin of B and C) is ignored here. This could be * solved by introducing an additional "effective" ("calculated", * "collapsed") margin as an attribute of Widget. * * - For similar reasons, backgrounds to not work exactly. Usage of * Widget::extraSpace should fix this, but it is only fully working * in the GROWS branch (). * * Update: This needs to be re-evaluated. * * - Do margins of inline blocks and tables collapse? Check CSS * spec. (They do currently; if not, ignoring them is simple.) * * - Lines which only contain a BREAK should be skipped for collapsing * margins, or at least all three should collapse: the previous * margin, the break, and the following margin. (Compare this with * the CSS spec.) * * - Related to this: adding breaks should be revised. * Textblock::addLinebreak and Textblock::addParbreak work quite * differently, and Textblock::addParbreak seems much to complex for * our needs, even when spaces of lines are kept. * * *

Some Internals

* * There are 4 lists, dw::Textblock::words, dw::Textblock::paragraphs, * dw::Textblock::lines, and dw::Textblock::anchors. The word list is * quite static; only new words may be added. A word is either text, a * widget, or a break. * * Lines refer to the word list (first and last). They are completely * redundant, i.e., they can be rebuilt from the words. Lines can be * rewrapped either completely or partially (see "Incremental Resizing" * below). For the latter purpose, several values are accumulated in the * lines. See dw::Textblock::Line for details. * * A recent change was the introduction of the paragraphs list, which * works quite similar, is also redundant, but is used to calculate * the extremes, not the size. * * Anchors associate the anchor name with the index of the next word at * the point of the anchor. * *

Incremental Resizing

* * dw::Textblock makes use of incremental resizing as described in \ref * dw-widget-sizes. The parentRef is, for children of a dw::Textblock, simply * the number of the line. [Update: Incorrect; see \ref dw-out-of-flow.] * * Generally, there are three cases which may change the size of the * widget: * *
    *
  • The line break size of the widget has changed, e.g., because the * user has changed the size of the browser window. In this case, * it is necessary to rewrap all the lines. * *
  • A child widget has changed its size. In this case, only a rewrap * down from the line where this widget is located is necessary. * * (This case is very important for tables. Tables are quite at the * bottom, so that a partial rewrap is relevant. Otherwise, tables * change their size quite often, so that this is necessary for a * fast, non-blocking rendering) * *
  • A word (or widget, break etc.) is added to the text block. This * makes it possible to reuse the old size by simply adjusting the * current width and height, so no rewrapping is necessary. *
* * The state of the size calculation is stored in wrapRef within * dw::Textblock, which has the value -1 if no rewrapping of lines * necessary, or otherwise the line from which a rewrap is necessary. * *

Widgets Ouf Of Flow

* * See * * - dw::oof::OOFAwareWidget (base class) and * - \ref dw-out-of-flow. */ class Textblock: public RegardingBorder { private: /** * This class encapsulates the badness/penalty calculation, and so * (i) makes changes (hopefully) simpler, and (ii) hides the * integer arithmetic (floating point arithmetic avoided for * performance reasons). Unfortunately, the value range of the * badness is not well defined, so fiddling with the penalties is a * bit difficult. */ enum { PENALTY_FORCE_BREAK = INT_MIN, PENALTY_PROHIBIT_BREAK = INT_MAX }; class BadnessAndPenalty { private: enum { NOT_STRETCHABLE, QUITE_LOOSE, BADNESS_VALUE, TOO_TIGHT } badnessState; int ratio; // ratio is only defined when badness is defined int badness, penalty[2]; // For debugging: define DEBUG for more information in print(). #ifdef DEBUG int totalWidth, idealWidth, totalStretchability, totalShrinkability; #endif // "Infinity levels" are used to represent very large numbers, // including "quasi-infinite" numbers. A couple of infinity // level and number can be mathematically represented as // // number * N ^ (infinity level) // // where N is a number which is large enough. Practically, // infinity levels are used to circumvent limited ranges for // integer numbers. // Here, all infinity levels have got special meanings. enum { INF_VALUE = 0, /* simple values */ INF_LARGE, /* large values, like QUITE_LOOSE */ INF_NOT_STRETCHABLE, /* reserved for NOT_STRECTHABLE */ INF_TOO_TIGHT, /* used for lines, which are too tight */ INF_PENALTIES, /* used for penalties */ INF_MAX = INF_PENALTIES // That INF_PENALTIES is the last value means that an // infinite penalty (breaking is prohibited) makes a break // not possible at all, so that pre-formatted text // etc. works. }; void setSinglePenalty (int index, int penalty); int badnessValue (int infLevel); int penaltyValue (int index, int infLevel); public: void calcBadness (int totalWidth, int idealWidth, int totalStretchability, int totalShrinkability); inline void setPenalty (int penalty) { setPenalties (penalty, penalty); } void setPenalties (int penalty1, int penalty2); // Rather for debugging: inline int getPenalty (int i) { return penalty[i]; } bool lineLoose (); bool lineTight (); bool lineTooTight (); bool lineMustBeBroken (int penaltyIndex); bool lineCanBeBroken (int penaltyIndex); int compareTo (int penaltyIndex, BadnessAndPenalty *other); void intoStringBuffer(lout::misc::StringBuffer *sb); }; enum { PENALTY_HYPHEN, PENALTY_EM_DASH_LEFT, PENALTY_EM_DASH_RIGHT, PENALTY_NUM }; enum { NUM_DIV_CHARS = 4 }; typedef struct { const char *s; bool charRemoved, canBeHyphenated, unbreakableForMinWidth; int penaltyIndexLeft, penaltyIndexRight; } DivChar; static DivChar divChars[NUM_DIV_CHARS]; static const char *hyphenDrawChar; protected: /** * \brief Implementation used for words. */ class WordImgRenderer: public core::style::StyleImage::ExternalWidgetImgRenderer { protected: Textblock *textblock; int wordNo, xWordWidget, lineNo; bool dataSet; public: WordImgRenderer (Textblock *textblock, int wordNo); ~WordImgRenderer (); void setData (int xWordWidget, int lineNo); bool readyToDraw (); void getBgArea (int *x, int *y, int *width, int *height); void getRefArea (int *xRef, int *yRef, int *widthRef, int *heightRef); core::style::Style *getStyle (); void draw (int x, int y, int width, int height); }; class SpaceImgRenderer: public WordImgRenderer { public: inline SpaceImgRenderer (Textblock *textblock, int wordNo) : WordImgRenderer (textblock, wordNo) { } void getBgArea (int *x, int *y, int *width, int *height); core::style::Style *getStyle (); }; struct Paragraph { int firstWord; /* first word's index in word vector */ int lastWord; /* last word's index in word vector */ /* * General remark: all values include the last hyphen width, but * not the last space; these values are, however corrected, when * another word is added. * * Also, as opposed to lines, paragraphs are created with the * first, not the last word, so these values change when new * words are added. */ int parMin; /* The sum of all word minima (plus spaces, hyphen width etc.) since the last possible break within this paragraph. */ int parMinIntrinsic; int parAdjustmentWidth; int parMax; /* The sum of all word maxima in this paragraph (plus spaces, hyphen width etc.). */ int parMaxIntrinsic; int maxParMin; /* Maximum of all paragraph minima (value of "parMin"), including this paragraph. */ int maxParMinIntrinsic; int maxParAdjustmentWidth; int maxParMax; /* Maximum of all paragraph maxima (value of "parMax""), including this paragraph. */ int maxParMaxIntrinsic; }; struct Line { int firstWord; /* first word's index in word vector */ int lastWord; /* last word's index in word vector */ int top; /* "top" is always relative to the top of the first line, i.e. page->lines[0].top is always 0. */ int marginAscent; /* Maximum of all total ascents (including margin: hence the name) of the words in this line. */ int marginDescent; /* Maximum of all total decents (including margin: hence the name) of the words in this line. */ int borderAscent; /* Maximum of all ascents minus margin (but including padding and border: hence the name) of the words in this line. */ int borderDescent; /* Maximum of all descents minus margin (but including padding and border: hence the name) of the words in this line. */ int contentAscent; /* ??? (deprecated?) */ int contentDescent; /* ??? (deprecated?) */ int breakSpace; /* Space between this line and the next one. */ int textOffset; /* Horizontal position of the first word of the line, in widget coordinates. */ /** * \brief Returns the difference between two vertical lines * positions: height of this line plus space below this * line. The margin of the next line (marginAscent - * borderAscent) must be passed separately. */ inline int totalHeight (int marginNextLine) { return borderAscent + borderDescent // Collapsing of the margins of adjacent lines is done here: + std::max( marginDescent - borderDescent, std::max( marginNextLine, breakSpace ) ); } /* Maximum of all line widths, including this line. Does not * include the last space, but the last hyphen width. Please * notice a change: until recently (before hyphenation and * changed line breaking), the values were accumulated up to the * last line, not this line.*/ int maxLineWidth; /* The word index of the last OOF reference (most importantly: * float) whic is positioned before this line, or -1, if there * is no OOF reference positioned before. * * **Important:** These references may still be part of this or * even a following line, when positioned before (this is the * reason this attribute exists); see \ref dw-out-of-flow. */ int lastOofRefPositionedBeforeThisLine; int leftOffset, rightOffset; enum { LEFT, RIGHT, CENTER } alignment; }; struct Word { enum { /** Can be hyphenated automatically. (Cleared after * hyphenation.) */ CAN_BE_HYPHENATED = 1 << 0, /** Must be drawn with a hyphen, when at the end of the line. */ DIV_CHAR_AT_EOL = 1 << 1, /** Is or ends with a "division character", which is part of * the word. */ PERM_DIV_CHAR = 1 << 2, /** This word must be drawn, together with the following * word(s), by only one call of View::drawText(), to get * kerning, ligatures etc. right. The last of the words drawn * as one text does *not* have this flag set. */ DRAW_AS_ONE_TEXT = 1 << 3, /* When calculating the minimal width (as part of extremes), * do not consider this word as breakable. This flag is * ignored when the line is actually broken. */ UNBREAKABLE_FOR_MIN_WIDTH = 1 << 4, /* If a word represents a "real" text word, or (after * hyphenation) the first part of a "real" text word, this * flag is set. Plays a role for text transformation. */ WORD_START = 1 << 5, /* If a word represents a "real" text word, or (after * hyphenation) the last part of a "real" text word, this * flag is set. Analogue to WORD_START. */ WORD_END = 1 << 6, /* This word is put at the top of the line, and at the * left. This is necessary if the size of a child widget * depends on the position, which, on the other hand, cannot * be determined before the whole line is broken. */ TOPLEFT_OF_LINE = 1 << 7 }; /* TODO: perhaps add a xLeft? */ core::Requisition size; /* Space after the word, only if it's not a break: */ short origSpace; /* from font, set by addSpace */ short effSpace; /* effective space, set by wordWrap, * used for drawing etc. */ short hyphenWidth; /* Additional width, when a word is part * (except the last part) of a hyphenationed * word. Has to be added to the width, when * this is the last word of the line, and * "hyphenWidth > 0" is also used to decide * whether to draw a hyphen. */ short flags; core::Content content; // accumulated values, relative to the beginning of the line int totalWidth; /* The sum of all word widths; plus all spaces, excluding the one of this word; plus the hyphen width of this word (but of course, no hyphen widths of previous words. In other words: the value compared to the ideal width of the line, if the line would be broken after this word. */ int maxAscent, maxDescent; int totalSpaceStretchability; // includes all *before* current word int totalSpaceShrinkability; // includes all *before* current word BadnessAndPenalty badnessAndPenalty; /* when line is broken after this * word */ core::style::Style *style; core::style::Style *spaceStyle; /* initially the same as of the word, later set by a_Dw_page_add_space */ // These two are used rarely, so there is perhaps a way to store // them which is consuming less memory. WordImgRenderer *wordImgRenderer; SpaceImgRenderer *spaceImgRenderer; }; struct Anchor { std::string_view name; int wordIndex; }; class TextblockIterator: public OOFAwareWidgetIterator { protected: int numContentsInFlow (); void getContentInFlow (int index, core::Content *content); public: TextblockIterator (Textblock *textblock, core::Content::Type mask, bool atEnd); static TextblockIterator *createWordIndexIterator (Textblock *textblock, core::Content::Type mask, int wordIndex); lout::object::Object *clone(); void highlight (int start, int end, core::HighlightLayer layer); void unhighlight (int direction, core::HighlightLayer layer); void getAllocation (int start, int end, core::Allocation *allocation); }; friend class TextblockIterator; /* These fields provide some ad-hoc-functionality, used by sub-classes. */ bool hasListitemValue; /* If true, the first word of the page is treated specially (search in source). */ int leftInnerPadding; /* This is an additional padding on the left side (used by ListItem). */ int line1Offset; /* This is an additional offset of the first line. May be negative (shift to left) or positive (shift to right). */ int line1OffsetEff; /* The "effective" value of line1_offset, may differ from line1_offset when ignoreLine1OffsetSometimes is set to true. */ /* The following is really hackish: It is used for DwTableCell (see * comment in dw_table_cell.c), to avoid too wide table columns. If * set to true, it has following effects: * * (i) line1_offset is ignored in calculating the minimal width * (which is used by DwTable!), and * (ii) line1_offset is ignored (line1_offset_eff is set to 0), * when line1_offset plus the width of the first word is * greater than the line break width. * * \todo Eliminate all these ad-hoc features by a new, simpler and * more elegant design. ;-) */ bool ignoreLine1OffsetSometimes; bool mustQueueResize; /** * The penalties for hyphens and other, multiplied by 100. So, 100 * means 1.0. INT_MAX and INT_MIN are also allowed. See * dw::Textblock::BadnessAndPenalty::setPenalty for more * details. Set from preferences. */ static int penalties[PENALTY_NUM][2]; /** * ... */ static int stretchabilityFactor; bool limitTextWidth; /* from preferences */ bool treatAsInline; int redrawY; int lastWordDrawn; core::SizeParams sizeRequestParams; /* Stores the value of getAvailWidth(). */ int lineBreakWidth; int wrapRefLines, wrapRefParagraphs; /* 0-based. Important: Both are the line numbers, not the value stored in parentRef. */ int wrapRefLinesFCX, wrapRefLinesFCY; // These four values are calculated by containingBlock->outOfFlowMgr // (when defined; otherwise, they are false, or 0, respectively), for // the newly constructed line, only when needed: when a new line is // added, or if something in the line currently constructed has // changed, e. g. a float has been added. bool newLineHasFloatLeft, newLineHasFloatRight; int newLineLeftBorder, newLineRightBorder; /* As returned by outOfFlowMgr->get...Border, or 0, if outOfFlowMgr is NULL */ int newLineLeftFloatHeight, newLineRightFloatHeight; // Ascent and descent of the newly constructed line, i. e. maximum // of all words ascent/descent since the end of the last line. Not // necessary the ascent and descent of the newly added line, since // not all words are added to it. int newLineAscent, newLineDescent; lout::misc::SimpleVector *lines; lout::misc::SimpleVector *paragraphs; int nonTemporaryLines; lout::misc::NotSoSimpleVector *words; std::vector< Anchor > anchors; struct { int index, nChar; } hlStart[core::HIGHLIGHT_NUM_LAYERS], hlEnd[core::HIGHLIGHT_NUM_LAYERS]; int hoverLink; /* The link under the mouse pointer */ int numSizeReferences; Widget *sizeReferences[NUM_OOFM]; void queueDrawRange (int index1, int index2); int calcVerticalBorder (int widgetPadding, int widgetBorder, int widgetMargin, int lineBorderTotal, int lineMarginTotal); void getWordExtremes (Word *word, core::Extremes *extremes); void justifyLine (Line *line, int diff); Line *addLine (int firstWord, int lastWord, int newLastOofPos, bool temporary, int minHeight); void rewrap (); void fillParagraphs (); void initNewLine (); void calcBorders (int lastOofRef, int height); void showMissingLines (); void removeTemporaryLines (); void decorateText (core::View *view, core::style::Style *style, core::style::Color::Shading shading, int x, int yBase, int width); void drawText (core::View *view, core::style::Style *style, core::style::Color::Shading shading, int x, int y, const char *text, int start, int len, bool isStart, bool isEnd); void drawWord (Line *line, int wordIndex1, int wordIndex2, core::View *view, core::Rectangle *area, int xWidget, int yWidgetBase); void drawWord0 (int wordIndex1, int wordIndex2, const char *text, int totalWidth, bool drawHyphen, core::style::Style *style, core::View *view, core::Rectangle *area, int xWidget, int yWidgetBase); void drawSpace (int wordIndex, core::View *view, core::Rectangle *area, int xWidget, int yWidgetBase); void drawLine (Line *line, core::View *view, core::Rectangle *area, core::DrawingContext *context); int findLineIndex (int y); int findLineIndexWhenNotAllocated (int y); int findLineIndexWhenAllocated (int y); int findLineIndex (int y, int ascent); int findLineOfWord (int wordIndex); int findParagraphOfWord (int wordIndex); Word *findWord (int x, int y, bool *inSpace); Word *addWord (int width, int ascent, int descent, short flags, core::style::Style *style); void breakAdded (); void initWord (int wordNo); void cleanupWord (int wordNo); void removeWordImgRenderer (int wordNo); void setWordImgRenderer (int wordNo); void removeSpaceImgRenderer (int wordNo); void setSpaceImgRenderer (int wordNo); void fillWord (int wordNo, int width, int ascent, int descent, short flags, core::style::Style *style); void fillSpace (int wordNo, core::style::Style *style); void setBreakOption (Word *word, core::style::Style *style, int breakPenalty1, int breakPenalty2, bool forceBreak); bool isBreakAllowedInWord (Word *word) { return isBreakAllowed (word->style); } bool isBreakAllowed (core::style::Style *style); int textWidth (const char *text, int start, int len, core::style::Style *style, bool isStart, bool isEnd); void calcTextSize (const char *text, size_t len, core::style::Style *style, core::Requisition *size, bool isStart, bool isEnd); bool calcSizeOfWidgetInFlow (int wordIndex, Widget *widget, core::Requisition *size); bool findSizeRequestReference (Widget *reference, int *xRef = NULL, int *yRef = NULL); bool findSizeRequestReference (int oofmIndex, int *xRef = NULL, int *yRef = NULL) { return findSizeRequestReference (oofContainer[oofmIndex], xRef, yRef); } /** * Of nested text blocks, only the most inner one must regard the * borders of floats. */ inline bool mustBorderBeRegarded (Line *line) { return getWidgetRegardingBorderForLine (line) == NULL; } inline bool mustBorderBeRegarded (int lineNo) { return getWidgetRegardingBorderForLine (lineNo) == NULL; } // The following methods return the y offset of a line, // - given as pointer or by index; // - either within the canvas, or within this widget; // - with allocation passed explicitly, or using the widget allocation // (important: this is set *after* sizeRequestImpl is returning. inline int lineYOffsetWidget (Line *line, core::Allocation *allocation) { return line->top + (allocation->ascent - lines->getRef(0)->borderAscent); } inline int lineYOffsetWidget (Line *line) { return lineYOffsetWidget (line, &allocation); } inline int lineYOffsetCanvas (Line *line, core::Allocation *allocation) { return allocation->y + lineYOffsetWidget (line, allocation); } inline int lineYOffsetCanvas (Line *line) { return lineYOffsetCanvas (line, &allocation); } inline int lineYOffsetWidget (int lineIndex) { return lineYOffsetWidget (lines->getRef (lineIndex)); } inline int lineYOffsetWidget (int lineIndex, core::Allocation *allocation) { return lineYOffsetWidget (lines->getRef (lineIndex), allocation); } inline int lineYOffsetCanvas (int lineIndex) { return lineYOffsetCanvas (lines->getRef (lineIndex)); } inline int calcPenaltyIndexForNewLine () { if (lines->size() == 0) return 0; else { Line *line = lines->getLastRef(); if (line->firstWord <= line->lastWord) return (words->getRef(line->lastWord)->flags & (Word::DIV_CHAR_AT_EOL | Word::PERM_DIV_CHAR)) ? 1 : 0; else // empty line return 0; } } RegardingBorder *getWidgetRegardingBorderForLine (Line *line); RegardingBorder *getWidgetRegardingBorderForLine (int lineNo); RegardingBorder *getWidgetRegardingBorderForLine (int firstWord, int lastWord); int yOffsetOfLineToBeCreated (int *lastMargin = NULL); int yOffsetOfLineCreated (Line *line); bool sendSelectionEvent (core::SelectionState::EventType eventType, core::MousePositionEvent *event); void processWord (int wordIndex); virtual int wordWrap (int wordIndex, bool wrapAll); int wrapWordInFlow (int wordIndex, bool wrapAll); int wrapWordOofRef (int wordIndex, bool wrapAll); void balanceBreakPosAndHeight (int wordIndex, int firstIndex, int *searchUntil, bool tempNewLine, int penaltyIndex, bool borderIsCalculated, bool *thereWillBeMoreSpace, bool wrapAll, int *diffWords, int *wordIndexEnd, int *lastFloatPos, bool regardBorder, int *height, int *breakPos); int searchBreakPos (int wordIndex, int firstIndex, int *searchUntil, bool tempNewLine, int penaltyIndex, bool thereWillBeMoreSpace, bool wrapAll, int *diffWords, int *wordIndexEnd, int *addIndex1 = NULL); int searchMinBap (int firstWord, int lastWordm, int penaltyIndex, bool thereWillBeMoreSpace, bool correctAtEnd); int considerHyphenation (int firstIndex, int breakPos); bool isHyphenationCandidate (Word *word); int calcLinePartHeight (int firstWord, int lastWord); void handleWordExtremes (int wordIndex); void correctLastWordExtremes (); static int getSpaceShrinkability(struct Word *word); static int getSpaceStretchability(struct Word *word); int getLineShrinkability(int lastWordIndex); int getLineStretchability(int lastWordIndex); int hyphenateWord (int wordIndex, int *addIndex1 = NULL); void moveWordIndices (int wordIndex, int num, int *addIndex1 = NULL); void accumulateWordForLine (int lineIndex, int wordIndex); void accumulateWordData (int wordIndex); int calcLineBreakWidth (int lineIndex); void initLine1Offset (int wordIndex); void alignLine (int lineIndex); void calcTextOffset (int lineIndex, int totalWidth); void drawLevel (core::View *view, core::Rectangle *area, int level, core::DrawingContext *context); Widget *getWidgetAtPointLevel (int x, int y, int level, core::GettingWidgetAtPointContext *context); void sizeRequestImpl (core::Requisition *requisition, int numPos, Widget **references, int *x, int *y); int numSizeRequestReferences (); Widget *sizeRequestReference (int index); void getExtremesSimpl (core::Extremes *extremes); int numGetExtremesReferences (); Widget *getExtremesReference (int index); void notifySetAsTopLevel (); void notifySetParent (); void sizeAllocateImpl (core::Allocation *allocation); void calcExtraSpaceImpl (int numPos, Widget **references, int *x, int *y); int getAvailWidthOfChild (core::Widget *child, bool forceValue); int getAvailHeightOfChild (core::Widget *child, bool forceValue); void containerSizeChangedForChildren (); bool affectsSizeChangeContainerChild (Widget *child); bool usesAvailWidth (); void resizeDrawImpl (); void markSizeChange (int ref); void markExtremesChange (int ref); bool isBlockLevel (); bool buttonPressImpl (core::EventButton *event); bool buttonReleaseImpl (core::EventButton *event); bool motionNotifyImpl (core::EventMotion *event); void enterNotifyImpl (core::EventCrossing *event); void leaveNotifyImpl (core::EventCrossing *event); void removeChild (Widget *child); void addText0 (const char *text, size_t len, short flags, core::style::Style *style, core::Requisition *size); void calcTextSizes (const char *text, size_t textLen, core::style::Style *style, int numBreaks, int *breakPos, core::Requisition *wordSize); int getGeneratorRest (int oofmIndex); public: static void setPenaltyHyphen (int penaltyHyphen); static void setPenaltyHyphen2 (int penaltyHyphen2); static void setPenaltyEmDashLeft (int penaltyLeftEmDash); static void setPenaltyEmDashRight (int penaltyRightEmDash); static void setPenaltyEmDashRight2 (int penaltyRightEmDash2); static void setStretchabilityFactor (int stretchabilityFactor); static inline bool mustAddBreaks (core::style::Style *style) { return !testStyleOutOfFlow (style) || testStyleRelativelyPositioned (style); } Textblock (bool limitTextWidth, bool treatAsInline = false); ~Textblock (); core::Iterator *iterator (core::Content::Type mask, bool atEnd); void flush (); void addText (const char *text, size_t len, core::style::Style *style); inline void addText (const char *text, core::style::Style *style) { addText (text, strlen(text), style); } void addWidget (core::Widget *widget, core::style::Style *style); bool addAnchor (const std::string &name, core::style::Style *style); void addSpace (core::style::Style *style); void addBreakOption (core::style::Style *style, bool forceBreak); void addParbreak (int space, core::style::Style *style); void addLinebreak (core::style::Style *style); void handOverBreak (core::style::Style *style); void changeLinkColor (int link, int newColor); void changeWordStyle (int from, int to, core::style::Style *style, bool includeFirstSpace, bool includeLastSpace); void updateReference (int ref); void widgetRefSizeChanged (int externalIndex); void oofSizeChanged (bool extremesChanged); int getGeneratorX (int oofmIndex); int getGeneratorY (int oofmIndex); int getGeneratorWidth (); int getMaxGeneratorWidth (); bool usesMaxGeneratorWidth (); bool isPossibleOOFContainer (int oofmIndex); bool isPossibleOOFContainerParent (int oofmIndex); }; #define DBG_SET_WORD_PENALTY(n, i, is) \ D_STMT_START { \ if (words->getRef(n)->badnessAndPenalty.getPenalty (i) == INT_MIN) \ DBG_OBJ_ARRATTRSET_SYM ("words", n, "penalty." is, "-inf"); \ else if (words->getRef(n)->badnessAndPenalty.getPenalty (i) == INT_MAX) \ DBG_OBJ_ARRATTRSET_SYM ("words", n, "penalty." is, "inf"); \ else \ DBG_OBJ_ARRATTRSET_NUM ("words", n, "penalty." is, \ words->getRef(n)->badnessAndPenalty \ .getPenalty (i)); \ } D_STMT_END #ifdef DBG_RTFL #define DBG_OBJ_ARRATTRSET_WREF(var, ind, attr, wref) \ RTFL_OBJ_PRINT ("set", "p:s.d.s:p (p, d)", this, var, ind, attr, wref, \ wref->widget, wref->parentRef) #else #define DBG_OBJ_ARRATTRSET_WREF(var, ind, attr, wref) STMT_NOP #endif #define DBG_SET_WORD(n) \ D_STMT_START { \ switch (words->getRef(n)->content.type) { \ case ::dw::core::Content::TEXT: \ DBG_OBJ_ARRATTRSET_SYM ("words", n, "type", "TEXT"); \ DBG_OBJ_ARRATTRSET_STR ("words", n, \ "text/widget/widgetReference/breakSpace", \ words->getRef(n)->content.text); \ break; \ case ::dw::core::Content::WIDGET_IN_FLOW: \ DBG_OBJ_ARRATTRSET_SYM ("words", n, "type", "WIDGET_IN_FLOW"); \ DBG_OBJ_ARRATTRSET_PTR ("words", n, \ "text/widget/widgetReference/breakSpace", \ words->getRef(n)->content.widget); \ break; \ case ::dw::core::Content::WIDGET_OOF_REF: \ DBG_OBJ_ARRATTRSET_SYM ("words", n, "type", "WIDGET_OOF_REF"); \ DBG_OBJ_ARRATTRSET_WREF ("words", n, \ "text/widget/widgetReference/breakSpace", \ words->getRef(n)->content.widgetReference); \ break; \ case ::dw::core::Content::BREAK: \ DBG_OBJ_ARRATTRSET_SYM ("words", n, "type", "BREAK"); \ DBG_OBJ_ARRATTRSET_NUM ("words", n, \ "text/widget/widgetReference/breakSpace", \ words->getRef(n)->content.breakSpace); \ break; \ default: \ DBG_OBJ_ARRATTRSET_SYM ("words", n, "type", "???"); \ DBG_OBJ_ARRATTRSET_SYM ("words", n, \ "text/widget/widgetReference/breakSpace", \ "???"); \ } \ DBG_SET_WORD_PENALTY (n, 0, "0"); \ DBG_SET_WORD_PENALTY (n, 1, "1"); \ } D_STMT_END #define DBG_SET_WORD_SIZE(n) \ D_STMT_START { \ DBG_OBJ_ARRATTRSET_NUM ("words", n, "size.width", \ words->getRef(n)->size.width); \ DBG_OBJ_ARRATTRSET_NUM ("words", n, "size.ascent", \ words->getRef(n)->size.ascent); \ DBG_OBJ_ARRATTRSET_NUM ("words", n, "size.descent", \ words->getRef(n)->size.descent); \ } D_STMT_END #define DBG_MSG_WORD(aspect, prio, prefix, n, suffix) \ D_STMT_START { \ if ((n) < 0 || (n) >= words->size ()) \ DBG_OBJ_MSG (aspect, prio, prefix "undefined (wrong index)" suffix); \ else { \ switch (words->getRef(n)->content.type) { \ case ::dw::core::Content::TEXT: \ DBG_OBJ_MSGF (aspect, prio, prefix "TEXT / \"%s\"" suffix, \ words->getRef(n)->content.text); \ break; \ case ::dw::core::Content::WIDGET_IN_FLOW: \ DBG_OBJ_MSGF (aspect, prio, prefix "WIDGET_IN_FLOW / %p" suffix, \ words->getRef(n)->content.widget); \ break; \ case ::dw::core::Content::WIDGET_OOF_REF: \ DBG_OBJ_MSGF (aspect, prio, \ prefix "WIDGET_OOF_REF / %p (%p, %d)" suffix,\ words->getRef(n)->content.widgetReference, \ words->getRef(n)->content.widgetReference->widget, \ words->getRef(n)->content.widgetReference \ ->parentRef); \ break; \ case ::dw::core::Content::BREAK: \ DBG_OBJ_MSGF (aspect, prio, prefix "BREAK / %d" suffix, \ words->getRef(n)->content.breakSpace); \ break; \ default: \ DBG_OBJ_MSG (aspect, prio, prefix "??? / ???" suffix); \ } \ } \ } D_STMT_END } // namespace dw #endif // __DW_TEXTBLOCK_HH__