| /* |
| * RichEdit - functions working on paragraphs of text (diParagraph). |
| * |
| * Copyright 2004 by Krzysztof Foltman |
| * Copyright 2006 by Phil Krylov |
| * |
| * This library is free software; you can redistribute it and/or |
| * modify it under the terms of the GNU Lesser General Public |
| * License as published by the Free Software Foundation; either |
| * version 2.1 of the License, or (at your option) any later version. |
| * |
| * This library is distributed in the hope that it will be useful, |
| * but WITHOUT ANY WARRANTY; without even the implied warranty of |
| * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU |
| * Lesser General Public License for more details. |
| * |
| * You should have received a copy of the GNU Lesser General Public |
| * License along with this library; if not, write to the Free Software |
| * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA |
| */ |
| |
| #include "editor.h" |
| |
| WINE_DEFAULT_DEBUG_CHANNEL(richedit); |
| |
| static const WCHAR wszParagraphSign[] = {0xB6, 0}; |
| |
| void ME_MakeFirstParagraph(ME_TextEditor *editor) |
| { |
| ME_Context c; |
| HDC hDC; |
| PARAFORMAT2 fmt; |
| CHARFORMAT2W cf; |
| LOGFONTW lf; |
| HFONT hf; |
| ME_TextBuffer *text = editor->pBuffer; |
| ME_DisplayItem *para = ME_MakeDI(diParagraph); |
| ME_DisplayItem *run; |
| ME_Style *style; |
| |
| hDC = GetDC(editor->hWnd); |
| |
| ME_InitContext(&c, editor, hDC); |
| hf = (HFONT)GetStockObject(SYSTEM_FONT); |
| assert(hf); |
| GetObjectW(hf, sizeof(LOGFONTW), &lf); |
| ZeroMemory(&cf, sizeof(cf)); |
| cf.cbSize = sizeof(cf); |
| cf.dwMask = CFM_BACKCOLOR|CFM_COLOR|CFM_FACE|CFM_SIZE|CFM_CHARSET; |
| cf.dwMask |= CFM_ALLCAPS|CFM_BOLD|CFM_DISABLED|CFM_EMBOSS|CFM_HIDDEN; |
| cf.dwMask |= CFM_IMPRINT|CFM_ITALIC|CFM_LINK|CFM_OUTLINE|CFM_PROTECTED; |
| cf.dwMask |= CFM_REVISED|CFM_SHADOW|CFM_SMALLCAPS|CFM_STRIKEOUT; |
| cf.dwMask |= CFM_SUBSCRIPT|CFM_UNDERLINETYPE|CFM_WEIGHT; |
| |
| cf.dwEffects = CFE_AUTOCOLOR | CFE_AUTOBACKCOLOR; |
| lstrcpyW(cf.szFaceName, lf.lfFaceName); |
| cf.yHeight = ME_twips2pointsY(&c, lf.lfHeight); |
| if (lf.lfWeight >= 700) cf.dwEffects |= CFE_BOLD; |
| cf.wWeight = lf.lfWeight; |
| if (lf.lfItalic) cf.dwEffects |= CFE_ITALIC; |
| cf.bUnderlineType = (lf.lfUnderline) ? CFU_CF1UNDERLINE : CFU_UNDERLINENONE; |
| if (lf.lfStrikeOut) cf.dwEffects |= CFE_STRIKEOUT; |
| cf.bPitchAndFamily = lf.lfPitchAndFamily; |
| cf.bCharSet = lf.lfCharSet; |
| |
| ZeroMemory(&fmt, sizeof(fmt)); |
| fmt.cbSize = sizeof(fmt); |
| fmt.dwMask = PFM_ALIGNMENT | PFM_OFFSET | PFM_STARTINDENT | PFM_RIGHTINDENT | PFM_TABSTOPS; |
| fmt.wAlignment = PFA_LEFT; |
| |
| CopyMemory(para->member.para.pFmt, &fmt, sizeof(PARAFORMAT2)); |
| |
| style = ME_MakeStyle(&cf); |
| text->pDefaultStyle = style; |
| |
| run = ME_MakeRun(style, ME_MakeString(wszParagraphSign), MERF_ENDPARA); |
| run->member.run.nCharOfs = 0; |
| |
| ME_InsertBefore(text->pLast, para); |
| ME_InsertBefore(text->pLast, run); |
| para->member.para.prev_para = text->pFirst; |
| para->member.para.next_para = text->pLast; |
| text->pFirst->member.para.next_para = para; |
| text->pLast->member.para.prev_para = para; |
| |
| text->pLast->member.para.nCharOfs = 1; |
| |
| ME_DestroyContext(&c); |
| ReleaseDC(editor->hWnd, hDC); |
| } |
| |
| void ME_MarkAllForWrapping(ME_TextEditor *editor) |
| { |
| ME_MarkForWrapping(editor, editor->pBuffer->pFirst->member.para.next_para, editor->pBuffer->pLast); |
| } |
| |
| void ME_MarkForWrapping(ME_TextEditor *editor, ME_DisplayItem *first, const ME_DisplayItem *last) |
| { |
| while(first != last) |
| { |
| first->member.para.nFlags |= MEPF_REWRAP; |
| first = first->member.para.next_para; |
| } |
| } |
| |
| void ME_MarkForPainting(ME_TextEditor *editor, ME_DisplayItem *first, const ME_DisplayItem *last) |
| { |
| while(first != last) |
| { |
| first->member.para.nFlags |= MEPF_REPAINT; |
| first = first->member.para.next_para; |
| } |
| } |
| |
| /* split paragraph at the beginning of the run */ |
| ME_DisplayItem *ME_SplitParagraph(ME_TextEditor *editor, ME_DisplayItem *run, ME_Style *style) |
| { |
| ME_DisplayItem *next_para = NULL; |
| ME_DisplayItem *run_para = NULL; |
| ME_DisplayItem *new_para = ME_MakeDI(diParagraph); |
| ME_DisplayItem *end_run = ME_MakeRun(style,ME_MakeString(wszParagraphSign), MERF_ENDPARA); |
| ME_UndoItem *undo = NULL; |
| int ofs; |
| ME_DisplayItem *pp; |
| int end_len = (editor->bEmulateVersion10 ? 2 : 1); |
| |
| assert(run->type == diRun); |
| |
| run_para = ME_GetParagraph(run); |
| assert(run_para->member.para.pFmt->cbSize == sizeof(PARAFORMAT2)); |
| |
| ofs = end_run->member.run.nCharOfs = run->member.run.nCharOfs; |
| next_para = run_para->member.para.next_para; |
| assert(next_para == ME_FindItemFwd(run_para, diParagraphOrEnd)); |
| |
| undo = ME_AddUndoItem(editor, diUndoJoinParagraphs, NULL); |
| if (undo) |
| undo->nStart = run_para->member.para.nCharOfs + ofs; |
| |
| /* the new paragraph will have a different starting offset, so let's update its runs */ |
| pp = run; |
| while(pp->type == diRun) { |
| pp->member.run.nCharOfs -= ofs; |
| pp = ME_FindItemFwd(pp, diRunOrParagraphOrEnd); |
| } |
| new_para->member.para.nCharOfs = ME_GetParagraph(run)->member.para.nCharOfs+ofs; |
| new_para->member.para.nCharOfs += end_len; |
| |
| new_para->member.para.nFlags = MEPF_REWRAP; /* FIXME copy flags (if applicable) */ |
| /* FIXME initialize format style and call ME_SetParaFormat blah blah */ |
| CopyMemory(new_para->member.para.pFmt, run_para->member.para.pFmt, sizeof(PARAFORMAT2)); |
| |
| new_para->member.para.bTable = run_para->member.para.bTable; |
| |
| /* Inherit previous cell definitions if any */ |
| new_para->member.para.pCells = NULL; |
| if (run_para->member.para.pCells) |
| { |
| ME_TableCell *pCell, *pNewCell; |
| |
| for (pCell = run_para->member.para.pCells; pCell; pCell = pCell->next) |
| { |
| pNewCell = ALLOC_OBJ(ME_TableCell); |
| pNewCell->nRightBoundary = pCell->nRightBoundary; |
| pNewCell->next = NULL; |
| if (new_para->member.para.pCells) |
| new_para->member.para.pLastCell->next = pNewCell; |
| else |
| new_para->member.para.pCells = pNewCell; |
| new_para->member.para.pLastCell = pNewCell; |
| } |
| } |
| |
| /* fix paragraph properties. FIXME only needed when called from RTF reader */ |
| if (run_para->member.para.pCells && !run_para->member.para.bTable) |
| { |
| /* Paragraph does not have an \intbl keyword, so any table definition |
| * stored is invalid */ |
| ME_DestroyTableCellList(run_para); |
| } |
| |
| /* insert paragraph into paragraph double linked list */ |
| new_para->member.para.prev_para = run_para; |
| new_para->member.para.next_para = next_para; |
| run_para->member.para.next_para = new_para; |
| next_para->member.para.prev_para = new_para; |
| |
| /* insert end run of the old paragraph, and new paragraph, into DI double linked list */ |
| ME_InsertBefore(run, new_para); |
| ME_InsertBefore(new_para, end_run); |
| |
| /* force rewrap of the */ |
| run_para->member.para.prev_para->member.para.nFlags |= MEPF_REWRAP; |
| new_para->member.para.prev_para->member.para.nFlags |= MEPF_REWRAP; |
| |
| /* we've added the end run, so we need to modify nCharOfs in the next paragraphs */ |
| ME_PropagateCharOffset(next_para, end_len); |
| editor->nParagraphs++; |
| |
| return new_para; |
| } |
| |
| /* join tp with tp->member.para.next_para, keeping tp's style; this |
| * is consistent with the original */ |
| ME_DisplayItem *ME_JoinParagraphs(ME_TextEditor *editor, ME_DisplayItem *tp) |
| { |
| ME_DisplayItem *pNext, *pFirstRunInNext, *pRun, *pTmp; |
| int i, shift; |
| ME_UndoItem *undo = NULL; |
| int end_len = (editor->bEmulateVersion10 ? 2 : 1); |
| |
| assert(tp->type == diParagraph); |
| assert(tp->member.para.next_para); |
| assert(tp->member.para.next_para->type == diParagraph); |
| |
| pNext = tp->member.para.next_para; |
| |
| { |
| /* null char format operation to store the original char format for the ENDPARA run */ |
| CHARFORMAT2W fmt; |
| ME_InitCharFormat2W(&fmt); |
| ME_SetCharFormat(editor, pNext->member.para.nCharOfs - end_len, end_len, &fmt); |
| } |
| undo = ME_AddUndoItem(editor, diUndoSplitParagraph, NULL); |
| if (undo) |
| { |
| undo->nStart = pNext->member.para.nCharOfs - end_len; |
| assert(pNext->member.para.pFmt->cbSize == sizeof(PARAFORMAT2)); |
| CopyMemory(undo->di.member.para.pFmt, pNext->member.para.pFmt, sizeof(PARAFORMAT2)); |
| } |
| |
| shift = pNext->member.para.nCharOfs - tp->member.para.nCharOfs - end_len; |
| |
| pRun = ME_FindItemBack(pNext, diRunOrParagraph); |
| pFirstRunInNext = ME_FindItemFwd(pNext, diRunOrParagraph); |
| |
| assert(pRun); |
| assert(pRun->type == diRun); |
| assert(pRun->member.run.nFlags & MERF_ENDPARA); |
| assert(pFirstRunInNext->type == diRun); |
| |
| /* if some cursor points at end of paragraph, make it point to the first |
| run of the next joined paragraph */ |
| for (i=0; i<editor->nCursors; i++) { |
| if (editor->pCursors[i].pRun == pRun) { |
| editor->pCursors[i].pRun = pFirstRunInNext; |
| editor->pCursors[i].nOffset = 0; |
| } |
| } |
| |
| pTmp = pNext; |
| do { |
| pTmp = ME_FindItemFwd(pTmp, diRunOrParagraphOrEnd); |
| if (pTmp->type != diRun) |
| break; |
| TRACE("shifting \"%s\" by %d (previous %d)\n", debugstr_w(pTmp->member.run.strText->szData), shift, pTmp->member.run.nCharOfs); |
| pTmp->member.run.nCharOfs += shift; |
| } while(1); |
| |
| ME_Remove(pRun); |
| ME_DestroyDisplayItem(pRun); |
| |
| if (editor->pLastSelStartPara == pNext) |
| editor->pLastSelStartPara = tp; |
| if (editor->pLastSelEndPara == pNext) |
| editor->pLastSelEndPara = tp; |
| |
| tp->member.para.next_para = pNext->member.para.next_para; |
| pNext->member.para.next_para->member.para.prev_para = tp; |
| ME_Remove(pNext); |
| ME_DestroyDisplayItem(pNext); |
| |
| ME_PropagateCharOffset(tp->member.para.next_para, -end_len); |
| |
| ME_CheckCharOffsets(editor); |
| |
| editor->nParagraphs--; |
| tp->member.para.nFlags |= MEPF_REWRAP; |
| return tp; |
| } |
| |
| ME_DisplayItem *ME_GetParagraph(ME_DisplayItem *item) { |
| return ME_FindItemBackOrHere(item, diParagraph); |
| } |
| |
| void ME_DumpParaStyleToBuf(const PARAFORMAT2 *pFmt, char buf[2048]) |
| { |
| char *p; |
| p = buf; |
| |
| #define DUMP(mask, name, fmt, field) \ |
| if (pFmt->dwMask & (mask)) p += sprintf(p, "%-22s" fmt "\n", name, pFmt->field); \ |
| else p += sprintf(p, "%-22sN/A\n", name); |
| |
| /* we take for granted that PFE_xxx is the hiword of the corresponding PFM_xxx */ |
| #define DUMP_EFFECT(mask, name) \ |
| p += sprintf(p, "%-22s%s\n", name, (pFmt->dwMask & (mask)) ? ((pFmt->wEffects & ((mask) >> 8)) ? "yes" : "no") : "N/A"); |
| |
| DUMP(PFM_NUMBERING, "Numbering:", "%u", wNumbering); |
| DUMP_EFFECT(PFM_DONOTHYPHEN, "Disable auto-hyphen:"); |
| DUMP_EFFECT(PFM_KEEP, "No page break in para:"); |
| DUMP_EFFECT(PFM_KEEPNEXT, "No page break in para & next:"); |
| DUMP_EFFECT(PFM_NOLINENUMBER, "No line number:"); |
| DUMP_EFFECT(PFM_NOWIDOWCONTROL, "No widow & orphan:"); |
| DUMP_EFFECT(PFM_PAGEBREAKBEFORE, "Page break before:"); |
| DUMP_EFFECT(PFM_RTLPARA, "RTL para:"); |
| DUMP_EFFECT(PFM_SIDEBYSIDE, "Side by side:"); |
| DUMP_EFFECT(PFM_TABLE, "Table:"); |
| DUMP(PFM_OFFSETINDENT, "Offset indent:", "%d", dxStartIndent); |
| DUMP(PFM_STARTINDENT, "Start indent:", "%d", dxStartIndent); |
| DUMP(PFM_RIGHTINDENT, "Right indent:", "%d", dxRightIndent); |
| DUMP(PFM_OFFSET, "Offset:", "%d", dxOffset); |
| if (pFmt->dwMask & PFM_ALIGNMENT) { |
| switch (pFmt->wAlignment) { |
| case PFA_LEFT : p += sprintf(p, "Alignment: left\n"); break; |
| case PFA_RIGHT : p += sprintf(p, "Alignment: right\n"); break; |
| case PFA_CENTER : p += sprintf(p, "Alignment: center\n"); break; |
| case PFA_JUSTIFY: p += sprintf(p, "Alignment: justify\n"); break; |
| default : p += sprintf(p, "Alignment: incorrect %d\n", pFmt->wAlignment); break; |
| } |
| } |
| else p += sprintf(p, "Alignment: N/A\n"); |
| DUMP(PFM_TABSTOPS, "Tab Stops:", "%d", cTabCount); |
| if (pFmt->dwMask & PFM_TABSTOPS) { |
| int i; |
| p += sprintf(p, "\t"); |
| for (i = 0; i < pFmt->cTabCount; i++) p += sprintf(p, "%x ", pFmt->rgxTabs[i]); |
| p += sprintf(p, "\n"); |
| } |
| DUMP(PFM_SPACEBEFORE, "Space Before:", "%d", dySpaceBefore); |
| DUMP(PFM_SPACEAFTER, "Space After:", "%d", dySpaceAfter); |
| DUMP(PFM_LINESPACING, "Line spacing:", "%d", dyLineSpacing); |
| DUMP(PFM_STYLE, "Text style:", "%d", sStyle); |
| DUMP(PFM_LINESPACING, "Line spacing rule:", "%u", bLineSpacingRule); |
| /* bOutlineLevel should be 0 */ |
| DUMP(PFM_SHADING, "Shading Weigth:", "%u", wShadingWeight); |
| DUMP(PFM_SHADING, "Shading Style:", "%u", wShadingStyle); |
| DUMP(PFM_NUMBERINGSTART, "Numbering Start:", "%u", wNumberingStart); |
| DUMP(PFM_NUMBERINGSTYLE, "Numbering Style:", "0x%x", wNumberingStyle); |
| DUMP(PFM_NUMBERINGTAB, "Numbering Tab:", "%u", wNumberingStyle); |
| DUMP(PFM_BORDER, "Border Space:", "%u", wBorderSpace); |
| DUMP(PFM_BORDER, "Border Width:", "%u", wBorderWidth); |
| DUMP(PFM_BORDER, "Borders:", "%u", wBorders); |
| |
| #undef DUMP |
| #undef DUMP_EFFECT |
| } |
| |
| void ME_SetParaFormat(ME_TextEditor *editor, ME_DisplayItem *para, const PARAFORMAT2 *pFmt) |
| { |
| PARAFORMAT2 copy; |
| assert(sizeof(*para->member.para.pFmt) == sizeof(PARAFORMAT2)); |
| ME_AddUndoItem(editor, diUndoSetParagraphFormat, para); |
| |
| CopyMemory(©, para->member.para.pFmt, sizeof(PARAFORMAT2)); |
| |
| #define COPY_FIELD(m, f) \ |
| if (pFmt->dwMask & (m)) { \ |
| para->member.para.pFmt->dwMask |= m; \ |
| para->member.para.pFmt->f = pFmt->f; \ |
| } |
| |
| COPY_FIELD(PFM_NUMBERING, wNumbering); |
| #define EFFECTS_MASK (PFM_RTLPARA|PFM_KEEP|PFM_KEEPNEXT|PFM_PAGEBREAKBEFORE| \ |
| PFM_NOLINENUMBER|PFM_NOWIDOWCONTROL|PFM_DONOTHYPHEN|PFM_SIDEBYSIDE| \ |
| PFM_TABLE) |
| /* we take for granted that PFE_xxx is the hiword of the corresponding PFM_xxx */ |
| if (pFmt->dwMask & EFFECTS_MASK) { |
| para->member.para.pFmt->dwMask &= ~(pFmt->dwMask & EFFECTS_MASK); |
| para->member.para.pFmt->wEffects |= pFmt->wEffects & HIWORD(pFmt->dwMask); |
| } |
| #undef EFFECTS_MASK |
| |
| COPY_FIELD(PFM_STARTINDENT, dxStartIndent); |
| if (pFmt->dwMask & PFM_OFFSETINDENT) |
| para->member.para.pFmt->dxStartIndent += pFmt->dxStartIndent; |
| COPY_FIELD(PFM_RIGHTINDENT, dxRightIndent); |
| COPY_FIELD(PFM_OFFSET, dxOffset); |
| COPY_FIELD(PFM_ALIGNMENT, wAlignment); |
| |
| if (pFmt->dwMask & PFM_TABSTOPS) |
| { |
| para->member.para.pFmt->cTabCount = pFmt->cTabCount; |
| memcpy(para->member.para.pFmt->rgxTabs, pFmt->rgxTabs, pFmt->cTabCount*sizeof(LONG)); |
| } |
| COPY_FIELD(PFM_SPACEBEFORE, dySpaceBefore); |
| COPY_FIELD(PFM_SPACEAFTER, dySpaceAfter); |
| COPY_FIELD(PFM_LINESPACING, dyLineSpacing); |
| COPY_FIELD(PFM_STYLE, sStyle); |
| COPY_FIELD(PFM_LINESPACING, bLineSpacingRule); |
| COPY_FIELD(PFM_SHADING, wShadingWeight); |
| COPY_FIELD(PFM_SHADING, wShadingStyle); |
| COPY_FIELD(PFM_NUMBERINGSTART, wNumberingStart); |
| COPY_FIELD(PFM_NUMBERINGSTYLE, wNumberingStyle); |
| COPY_FIELD(PFM_NUMBERINGTAB, wNumberingTab); |
| COPY_FIELD(PFM_BORDER, wBorderSpace); |
| COPY_FIELD(PFM_BORDER, wBorderWidth); |
| COPY_FIELD(PFM_BORDER, wBorders); |
| |
| para->member.para.pFmt->dwMask |= pFmt->dwMask; |
| #undef COPY_FIELD |
| |
| if (memcmp(©, para->member.para.pFmt, sizeof(PARAFORMAT2))) |
| para->member.para.nFlags |= MEPF_REWRAP; |
| } |
| |
| |
| void |
| ME_GetSelectionParas(ME_TextEditor *editor, ME_DisplayItem **para, ME_DisplayItem **para_end) |
| { |
| ME_Cursor *pEndCursor = &editor->pCursors[1]; |
| |
| *para = ME_GetParagraph(editor->pCursors[0].pRun); |
| *para_end = ME_GetParagraph(editor->pCursors[1].pRun); |
| if ((*para_end)->member.para.nCharOfs < (*para)->member.para.nCharOfs) { |
| ME_DisplayItem *tmp = *para; |
| |
| *para = *para_end; |
| *para_end = tmp; |
| pEndCursor = &editor->pCursors[0]; |
| } |
| |
| /* selection consists of chars from nFrom up to nTo-1 */ |
| if ((*para_end)->member.para.nCharOfs > (*para)->member.para.nCharOfs) { |
| if (!pEndCursor->nOffset) { |
| *para_end = ME_GetParagraph(ME_FindItemBack(pEndCursor->pRun, diRun)); |
| } |
| } |
| } |
| |
| |
| void ME_SetSelectionParaFormat(ME_TextEditor *editor, const PARAFORMAT2 *pFmt) |
| { |
| ME_DisplayItem *para, *para_end; |
| |
| ME_GetSelectionParas(editor, ¶, ¶_end); |
| |
| do { |
| ME_SetParaFormat(editor, para, pFmt); |
| if (para == para_end) |
| break; |
| para = para->member.para.next_para; |
| } while(1); |
| } |
| |
| void ME_GetParaFormat(ME_TextEditor *editor, const ME_DisplayItem *para, PARAFORMAT2 *pFmt) |
| { |
| if (pFmt->cbSize >= sizeof(PARAFORMAT2)) |
| { |
| CopyMemory(pFmt, para->member.para.pFmt, sizeof(PARAFORMAT2)); |
| return; |
| } |
| CopyMemory(pFmt, para->member.para.pFmt, pFmt->cbSize); |
| } |
| |
| void ME_GetSelectionParaFormat(ME_TextEditor *editor, PARAFORMAT2 *pFmt) |
| { |
| ME_DisplayItem *para, *para_end; |
| PARAFORMAT2 tmp; |
| |
| ME_GetSelectionParas(editor, ¶, ¶_end); |
| |
| ME_GetParaFormat(editor, para, pFmt); |
| if (para == para_end) return; |
| |
| do { |
| ZeroMemory(&tmp, sizeof(tmp)); |
| tmp.cbSize = sizeof(tmp); |
| ME_GetParaFormat(editor, para, &tmp); |
| |
| #define CHECK_FIELD(m, f) \ |
| if (pFmt->f != tmp.f) pFmt->dwMask &= ~(m); |
| |
| CHECK_FIELD(PFM_NUMBERING, wNumbering); |
| /* para->member.para.pFmt->wEffects = pFmt->wEffects; */ |
| assert(tmp.dwMask & PFM_ALIGNMENT); |
| CHECK_FIELD(PFM_NUMBERING, wNumbering); |
| assert(tmp.dwMask & PFM_STARTINDENT); |
| CHECK_FIELD(PFM_STARTINDENT, dxStartIndent); |
| assert(tmp.dwMask & PFM_RIGHTINDENT); |
| CHECK_FIELD(PFM_RIGHTINDENT, dxRightIndent); |
| assert(tmp.dwMask & PFM_OFFSET); |
| CHECK_FIELD(PFM_OFFSET, dxOffset); |
| CHECK_FIELD(PFM_ALIGNMENT, wAlignment); |
| |
| assert(tmp.dwMask & PFM_TABSTOPS); |
| if (pFmt->dwMask & PFM_TABSTOPS) { |
| if (pFmt->cTabCount != tmp.cTabCount || |
| memcmp(pFmt->rgxTabs, tmp.rgxTabs, tmp.cTabCount*sizeof(int))) |
| pFmt->dwMask &= ~PFM_TABSTOPS; |
| } |
| |
| CHECK_FIELD(PFM_SPACEBEFORE, dySpaceBefore); |
| CHECK_FIELD(PFM_SPACEAFTER, dySpaceAfter); |
| CHECK_FIELD(PFM_LINESPACING, dyLineSpacing); |
| CHECK_FIELD(PFM_STYLE, sStyle); |
| CHECK_FIELD(PFM_SPACEAFTER, bLineSpacingRule); |
| CHECK_FIELD(PFM_SHADING, wShadingWeight); |
| CHECK_FIELD(PFM_SHADING, wShadingStyle); |
| CHECK_FIELD(PFM_NUMBERINGSTART, wNumberingStart); |
| CHECK_FIELD(PFM_NUMBERINGSTYLE, wNumberingStyle); |
| CHECK_FIELD(PFM_NUMBERINGTAB, wNumberingTab); |
| CHECK_FIELD(PFM_BORDER, wBorderSpace); |
| CHECK_FIELD(PFM_BORDER, wBorderWidth); |
| CHECK_FIELD(PFM_BORDER, wBorders); |
| |
| #undef CHECK_FIELD |
| |
| if (para == para_end) |
| return; |
| para = para->member.para.next_para; |
| } while(1); |
| } |