// Copyright 2014 PDFium Authors. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. // Original code copyright 2014 Foxit Software Inc. http://www.foxitsoftware.com #include "../../include/reflow/reflowengine.h" #include "reflowedpage.h" #include "layoutprovider_taggedpdf.h" IPDF_LayoutProcessor* IPDF_LayoutProcessor::Create_LayoutProcessor_Reflow(FX_FLOAT TopIndent, FX_FLOAT fWidth, FX_FLOAT fHeight, void* pReflowedPage, int flags, FX_FLOAT lineSpace ) { if(pReflowedPage == NULL || fWidth <= 20) { return NULL; } CPDF_LayoutProcessor_Reflow* pReflowEngine = FX_NEW CPDF_LayoutProcessor_Reflow(); if (NULL == pReflowEngine) { return NULL; } pReflowEngine->Init(TopIndent, fWidth, fHeight, (CPDF_ReflowedPage*)pReflowedPage, flags, lineSpace); return pReflowEngine; } CPDF_LayoutProcessor_Reflow::CPDF_LayoutProcessor_Reflow() { m_pPause = NULL; m_pLayoutElement = NULL; m_fRefWidth = 0; m_fRefWidth = 0; m_fCurrLineWidth = 0; m_fCurrLineHeight = 0; m_bIllustration = FALSE; m_pPreObj = NULL; m_pCurrLine = FX_NEW CRF_DataPtrArray(50); m_pTempLine = FX_NEW CRF_DataPtrArray(50); m_StartIndent = 0; m_PausePosition = 0; } CPDF_LayoutProcessor_Reflow::~CPDF_LayoutProcessor_Reflow() { if (m_pCurrLine) { m_pCurrLine->RemoveAll(); delete m_pCurrLine; } m_pCurrLine = NULL; if (m_pTempLine) { m_pTempLine->RemoveAll(); delete m_pTempLine; } m_pTempLine = NULL; } void CPDF_LayoutProcessor_Reflow::Init(FX_FLOAT TopIndent, FX_FLOAT fWidth, FX_FLOAT fHeight, CPDF_ReflowedPage* pReflowedPage, int flags, FX_FLOAT lineSpace) { m_pLayoutElement = NULL; m_TopIndent = TopIndent; m_Status = LayoutReady; m_flags = flags; m_pReflowedPage = pReflowedPage; m_fScreenHeight = fHeight; m_fRefWidth = fWidth; m_fCurrLineHeight = 0; m_fCurrLineWidth = 0; m_fLineSpace = lineSpace; pReflowedPage->m_PageWidth = fWidth; pReflowedPage->m_PageHeight = TopIndent; } void CPDF_LayoutProcessor_Reflow::FitPageMode() { if(m_flags & RF_PARSER_PAGEMODE && m_fScreenHeight > 20) { float fitPageHeight = m_fScreenHeight; CPDF_ReflowedPage* pRFPage = m_pReflowedPage; int count = pRFPage->m_pReflowed->GetSize(); CFX_WordArray dy; dy.Add(0); int pos = 0; int screenCount = 1; FX_FLOAT h = pRFPage->GetPageHeight(); while (h > screenCount * fitPageHeight) { FX_FLOAT tempPageHeight = screenCount * fitPageHeight; int j = 0; FX_FLOAT tempDy = 0; for(int i = 0; i < count; i++) { CRF_Data* pData = (*pRFPage->m_pReflowed)[i]; FX_FLOAT posY; posY = pData->m_PosY; if(FXSYS_fabs(posY) > tempPageHeight && FXSYS_fabs(posY + pData->m_Height) < tempPageHeight) { if(j == 0) { j = i; } if(pData->m_Height > fitPageHeight) { FX_FLOAT zoom; FX_FLOAT spaceh = screenCount * fitPageHeight + posY + pData->m_Height; if(spaceh < fitPageHeight / 3 * 2) { spaceh = fitPageHeight; } zoom = spaceh / pData->m_Height; tempDy = spaceh - pData->m_Height; pData->m_Height = spaceh; pData->m_Width *= zoom; break; } FX_FLOAT dy = pData->m_PosY + pData->m_Height + tempPageHeight; if(dy > tempDy) { tempDy = dy; } } else if(FXSYS_fabs(posY + pData->m_Height) > tempPageHeight) { break; } } for(; j < count; j++) { CRF_Data* pData = (*pRFPage->m_pReflowed)[j]; FX_FLOAT posY; posY = pData->m_PosY; if(FXSYS_fabs(posY) > tempPageHeight ) { pData->m_PosY -= tempDy; } if(pData->m_Height >= fitPageHeight) { pData->m_Height = fitPageHeight - 1; if(pData->GetType() == CRF_Data::Text) { CRF_CharData* pCharData = (CRF_CharData*)pData; pCharData->m_pCharState->m_fFontSize = pData->m_Height; } } } pRFPage->m_PageHeight += tempDy; h += tempDy; screenCount++; } } } LayoutStatus CPDF_LayoutProcessor_Reflow::StartProcess(IPDF_LayoutElement* pElement, IFX_Pause* pPause, const CFX_AffineMatrix* pPDFMatrix) { if(!pElement) { return LayoutError; } m_pPause = pPause; m_PDFMatrix = *pPDFMatrix; m_pRootElement = pElement; ProcessElement(m_pRootElement, m_fRefWidth); if(m_Status == LayoutToBeContinued) { return LayoutToBeContinued; } m_Status = LayoutFinished; FitPageMode(); return LayoutFinished; } LayoutStatus CPDF_LayoutProcessor_Reflow::Continue() { int size = m_pReflowedPage->m_pReflowed->GetSize(); ProcessElement(m_pRootElement, m_CurrRefWidth); size = m_pReflowedPage->m_pReflowed->GetSize(); if(m_Status == LayoutReady) { m_Status = LayoutFinished; FitPageMode(); } return m_Status; } int CPDF_LayoutProcessor_Reflow::GetPosition() { return m_PausePosition; } FX_BOOL CPDF_LayoutProcessor_Reflow::IsCanBreakAfter(FX_DWORD unicode) { if(unicode == -1) { return FALSE; } switch(unicode) { case 40: case 91: case 123: return FALSE; } if(unicode >= 256) { return TRUE; } else if(unicode >= 48 && unicode <= 57) { return FALSE; } else if(unicode >= 64 && unicode <= 90) { return FALSE; } else if(unicode >= 97 && unicode <= 122) { return FALSE; } return TRUE; } FX_BOOL CPDF_LayoutProcessor_Reflow::IsCanBreakBefore(FX_DWORD unicode) { if(unicode == -1) { return FALSE; } switch(unicode) { case 33: case 41: case 44: case 46: case 59: case 63: case 93: case 125: return FALSE; } if(unicode >= 256) { return TRUE; } else if(unicode >= 48 && unicode <= 57) { return FALSE; } else if(unicode >= 64 && unicode <= 90) { return FALSE; } else if(unicode >= 97 && unicode <= 122) { return FALSE; } return TRUE; } void CPDF_LayoutProcessor_Reflow::ProcessTable(FX_FLOAT dx) { if(m_pReflowedPage->m_pReflowed->GetSize() == 0) { return; } CRF_Table* pTable = m_TableArray.GetAt(m_TableArray.GetSize() - 1); int rowCount = pTable->m_nCell.GetSize(); int n = 0; FX_FLOAT* dyRow = FX_Alloc(FX_FLOAT, rowCount + 1); FXSYS_memset32(dyRow, 0, sizeof(FX_FLOAT) * (rowCount + 1)); dyRow[0] = 0 ; dyRow[0] = - pTable->m_ReflowPageHeight; int tableColCount = 0; int i; for(i = 0; i < rowCount; i++) { int colCount = pTable->m_nCell.GetAt(i); if(colCount > tableColCount) { tableColCount = colCount; } } int cellCount = tableColCount * rowCount; RF_TableCell** pVirtualTable = FX_Alloc(RF_TableCell*, cellCount); FXSYS_memset32(pVirtualTable, 0, sizeof(RF_TableCell*) * cellCount); for(i = 0; i < rowCount; i++) { int colCount = pTable->m_nCell.GetAt(i); FX_FLOAT rowWidth = 0; int j = 0; int s = pTable->m_pCellArray.GetSize(); for(j = 0; j < colCount; j++) { RF_TableCell* pCell = (RF_TableCell*)pTable->m_pCellArray.GetAt(n++); if(pCell->m_EndPos < pCell->m_BeginPos) { continue; } int pos = i * tableColCount; while(pos < cellCount && pVirtualTable[pos] != NULL) { pos++; } if(pos >= (i + 1) * tableColCount) { pos = i * tableColCount + j; } int RowSpan = pCell->m_RowSpan; int ColSpan = pCell->m_ColSpan; if(RowSpan + i > rowCount) { RowSpan = rowCount - i; } if(ColSpan + j > colCount) { ColSpan = colCount - j; } for(int m = 0; m < RowSpan; m++) { for(int nn = 0; nn < ColSpan; nn++) { if(pos + nn >= cellCount) { break; } pVirtualTable[pos + nn] = pCell; } pos += tableColCount; } FX_FLOAT dxCell = dx; for(pos = i * tableColCount; pVirtualTable[pos] != pCell && pos < cellCount; pos++) { dxCell += (pVirtualTable[pos])->m_MaxWidth; } CRF_Data* pData = (*m_pReflowedPage->m_pReflowed)[pCell->m_BeginPos]; FX_FLOAT dy = dyRow[i] - pData->m_Height - pData->m_PosY; CFX_AffineMatrix matrix(1, 0, 0, 1, dxCell, dy); Transform(&matrix, m_pReflowedPage->m_pReflowed, pCell->m_BeginPos, pCell->m_EndPos - pCell->m_BeginPos + 1); if(pCell->m_RowSpan + i <= rowCount) { if(FXSYS_fabs(dyRow[pCell->m_RowSpan + i]) < FXSYS_fabs(dyRow[i] - pCell->m_CellHeight)) { dyRow[pCell->m_RowSpan + i] = dyRow[i] - pCell->m_CellHeight; } } } } n = 0; for(i = 0; i < rowCount; i++) { int colCount = pTable->m_nCell.GetAt(i); for(int j = 0; j < colCount; j++) { RF_TableCell* pCell = (RF_TableCell*)pTable->m_pCellArray.GetAt(n++); switch(pCell->m_BlockAlign) { case LayoutAfter: { FX_FLOAT dy = dyRow[i + pCell->m_RowSpan] - pCell->m_CellHeight - dyRow[i]; CFX_AffineMatrix matrix(1, 0, 0, 1, 0, dy); Transform(&matrix, m_pReflowedPage->m_pReflowed, pCell->m_BeginPos, pCell->m_EndPos - pCell->m_BeginPos + 1); } break; case LayoutMiddle: case LayoutJustify: { FX_FLOAT dy = (dyRow[i + pCell->m_RowSpan] + pCell->m_CellHeight - dyRow[i]) / 2; CFX_AffineMatrix matrix(1, 0, 0, 1, 0, dy); Transform(&matrix, m_pReflowedPage->m_pReflowed, pCell->m_BeginPos, pCell->m_EndPos - pCell->m_BeginPos + 1); break; } default: break; } } } CRF_Data* pData = (*m_pReflowedPage->m_pReflowed)[m_pReflowedPage->m_pReflowed->GetSize() - 1]; m_pReflowedPage->m_PageHeight = - dyRow[rowCount] + pData->m_Height; FX_Free(pVirtualTable); FX_Free(dyRow); int size = pTable->m_pCellArray.GetSize(); for(i = 0; i < size; i++) { RF_TableCell* pCell = pTable->m_pCellArray.GetAt(i); FX_Free(pCell); } pTable->m_pCellArray.RemoveAll(); pTable->m_nCell.RemoveAll(); int s = sizeof(CRF_Table); delete pTable; m_TableArray.RemoveAt(m_TableArray.GetSize() - 1); } CFX_FloatRect CPDF_LayoutProcessor_Reflow::GetElmBBox(IPDF_LayoutElement* pElement) { CFX_FloatRect rect; int objCount = pElement->CountObjects(); int count = pElement->CountChildren(); if(objCount == 0 && count == 0) { return rect; } CFX_AffineMatrix matrix; int i; for(i = 0; i < objCount; i++) { CPDF_PageObject* pObj = pElement->GetObject(0); if(!pObj) { continue; } if( rect.Height() == 0 ) { rect = pObj->GetBBox(&matrix); } else { rect.Union(pObj->GetBBox(&matrix)); } } for(i = 0; i < count; i++) { IPDF_LayoutElement* pChildElement = pElement->GetChild(i); if( rect.Height() == 0 ) { rect = GetElmBBox(pChildElement); } else { rect.Union(GetElmBBox(pChildElement)); } } return rect; } FX_FLOAT CPDF_LayoutProcessor_Reflow::GetElmWidth(IPDF_LayoutElement* pElement) { if(!pElement) { return 0; } LayoutType layoutType = pElement->GetType(); FX_FLOAT width = 0; if(layoutType == LayoutTable || layoutType == LayoutTableDataCell || layoutType == LayoutTableHeaderCell) { width = pElement->GetNumberAttr(LayoutWidth); if(width > 0) { return width; } } else if( layoutType == LayoutTableRow) { int count = pElement->CountChildren(); for(int i = 0; i < count; i++) { IPDF_LayoutElement* pElm = pElement->GetChild(i); width += pElm->GetNumberAttr(LayoutWidth); } if(width > 0) { return width; } } CFX_FloatRect rect = GetElmBBox(pElement); return rect.Width(); } FX_BOOL GetIntersection(FX_FLOAT low1, FX_FLOAT high1, FX_FLOAT low2, FX_FLOAT high2, FX_FLOAT& interlow, FX_FLOAT& interhigh); FX_BOOL IsSameLine(FX_BOOL bHorizontal, CFX_FloatRect Rect1, CFX_FloatRect Rect2) { if(bHorizontal) { FX_FLOAT inter_top, inter_bottom; if (!GetIntersection(Rect1.bottom, Rect1.top, Rect2.bottom, Rect2.top, inter_bottom, inter_top)) { return FALSE; } FX_FLOAT lineHeight = Rect1.top - Rect1.bottom; if(lineHeight > 20 && lineHeight > Rect2.Height() * 2) { return FALSE; } if(lineHeight > 5 && Rect2.Height() / 2 > lineHeight) { return FALSE; } FX_FLOAT inter_h = inter_top - inter_bottom; if (inter_h < (lineHeight) / 2 && inter_h < Rect2.Height() / 2) { return FALSE; } } else { FX_FLOAT inter_left, inter_right; if(!GetIntersection(Rect1.left, Rect1.right, Rect2.left, Rect2.right, inter_left, inter_right)) { return FALSE; } FX_FLOAT inter_w = inter_right - inter_left; if (inter_w < (Rect1.right - Rect1.left) / 2 && inter_w < (Rect2.right - Rect2.left) / 2) { return FALSE; } } return TRUE; } FX_INT32 IsCanMergeParagraph(IPDF_LayoutElement* pPrevElement, IPDF_LayoutElement* pNextElement) { FX_INT32 analogial = 100; FX_INT32 nPrevObj = pPrevElement->CountObjects(), i; CPDF_PageObject* pPrevObj = NULL; CFX_FloatRect prevRect, rect; CFX_PtrArray prevLine, line; FX_BOOL bParagraphStart = FALSE; for(i = 0; i < nPrevObj; i++) { CPDF_PageObject* pObj = pPrevElement->GetObject(i); if(!pPrevObj) { pPrevObj = pObj; rect = CFX_FloatRect(pObj->m_Left, pObj->m_Bottom, pObj->m_Right, pObj->m_Top); line.Add(pObj); continue; } CFX_FloatRect objRect = CFX_FloatRect(pObj->m_Left, pObj->m_Bottom, pObj->m_Right, pObj->m_Top); if(IsSameLine(TRUE, rect, objRect)) { line.Add(pObj); rect.Union(objRect); } else { prevLine.RemoveAll(); prevLine.Append(line); prevRect = rect; line.RemoveAll(); line.Add(pObj); rect = objRect; if(!bParagraphStart) { if (prevRect.left > rect.left + rect.Height() * 1.5) { bParagraphStart = TRUE; } } } } if(prevLine.GetSize()) { if(FXSYS_fabs(rect.right - prevRect.right) > rect.Height()) { analogial -= 50; } } CPDF_PageObject* pObj = pPrevElement->GetObject(nPrevObj - 1); if(pObj->m_Type == PDFPAGE_TEXT) { CPDF_TextObject* pText = (CPDF_TextObject*)pObj; FX_INT32 nItem = pText->CountItems(); CPDF_TextObjectItem item; pText->GetItemInfo(nItem - 1, &item); CFX_WideString wStr = pText->GetFont()->UnicodeFromCharCode(item.m_CharCode); if(wStr.IsEmpty()) { wStr = (FX_WCHAR)item.m_CharCode; } FX_WCHAR wch = wStr.GetAt(wStr.GetLength() - 1); switch(wch) { case '.': case 12290: case 65311: case 63: case 33: case 65281: analogial -= 50; break; } } prevLine.RemoveAll(); prevLine.Append(line); line.RemoveAll(); FX_INT32 nNextObj = pNextElement->CountObjects(); pPrevObj = NULL; FX_BOOL bFirst = TRUE; for(i = 0; i < nNextObj; i++) { CPDF_PageObject* pObj = pNextElement->GetObject(i); if(!pPrevObj) { pPrevObj = pObj; rect = CFX_FloatRect(pObj->m_Left, pObj->m_Bottom, pObj->m_Right, pObj->m_Top); line.Add(pObj); continue; } CFX_FloatRect objRect = CFX_FloatRect(pObj->m_Left, pObj->m_Bottom, pObj->m_Right, pObj->m_Top); if(IsSameLine(TRUE, rect, objRect)) { line.Add(pObj); rect.Union(objRect); } else { if(FXSYS_fabs(rect.right - prevRect.right) < rect.Height() && FXSYS_fabs(rect.left - prevRect.left) < rect.Height()) { analogial += 50; } prevLine.RemoveAll(); prevLine.Append(line); prevRect = rect; line.RemoveAll(); line.Add(pObj); rect = objRect; if(!bFirst) { break; } bFirst = FALSE; } } if(prevLine.GetSize()) { if(bParagraphStart) { if(prevRect.left - rect.left > rect.Height() && prevRect.left - rect.left < rect.Height() * 3) { analogial -= 50; } } else { if(FXSYS_fabs(prevRect.left - rect.left) < rect.Height()) { analogial -= 50; } } } return analogial; } void CPDF_LayoutProcessor_Reflow::ProcessElement(IPDF_LayoutElement* pElement, FX_FLOAT reflowWidth) { if(pElement == NULL) { return; } if(m_Status == LayoutReady) { LayoutType layoutType = pElement->GetType(); FX_INT32 ElementType = GetElementTypes(layoutType); switch(ElementType) { case SST_IE: m_bIllustration = TRUE; break; case SST_BLSE: FinishedCurrLine(); FX_FLOAT StartIndent = 0; if(IPDF_LayoutElement* pParent = pElement->GetParent()) { StartIndent = pParent->GetNumberAttr(LayoutStartIndent); } FX_FLOAT currStartIndent = pElement->GetNumberAttr(LayoutStartIndent); m_StartIndent = ConverWidth(currStartIndent); FX_FLOAT width = reflowWidth; if(StartIndent != currStartIndent) { reflowWidth -= m_StartIndent; } FX_FLOAT spaceBefore = pElement->GetNumberAttr(LayoutSpaceBefore); m_pReflowedPage->m_PageHeight += spaceBefore; m_TextAlign = pElement->GetEnumAttr(LayoutTextAlign); if(IPDF_LayoutElement* pParent = pElement->GetParent()) { StartIndent = pParent->GetNumberAttr(LayoutEndIndent); FX_FLOAT currEndIndent = pElement->GetNumberAttr(LayoutEndIndent); if(StartIndent != currStartIndent) { reflowWidth -= ConverWidth(currEndIndent); } } if(reflowWidth * 2 < width) { reflowWidth = width; m_StartIndent = 0; } break; } switch(layoutType) { case LayoutTable: { CRF_Table* pTable = FX_NEW CRF_Table; if (NULL == pTable) { break; } m_TableArray.Add(pTable); pTable->m_ReflowPageHeight = m_pReflowedPage->m_PageHeight; pTable->m_TableWidth = GetElmWidth(pElement); break; } case LayoutTableRow: { if(!m_TableArray.GetSize()) { break; } int count = pElement->CountChildren(); CRF_Table* pTable = m_TableArray.GetAt(m_TableArray.GetSize() - 1); int f = 0; for(int i = 0; i < count; i++) { IPDF_LayoutElement* pChildElement = pElement->GetChild(i); LayoutType type = pChildElement->GetType(); if(type == LayoutTableDataCell || type == LayoutTableHeaderCell) { f++; } } pTable->m_nCell.Add(f); break; } case LayoutTableDataCell: case LayoutTableHeaderCell: { if(!m_TableArray.GetSize()) { break; } RF_TableCell* pCell = FX_Alloc(RF_TableCell, 1); FXSYS_memset32(pCell, 0 , sizeof(RF_TableCell)); CRF_Table* pTable = m_TableArray.GetAt(m_TableArray.GetSize() - 1); int pos = pTable->m_nCell.GetSize() - 1; pCell->m_BeginPos = m_pReflowedPage->m_pReflowed->GetSize(); FX_FLOAT cellWidth = pElement->GetNumberAttr(LayoutWidth); if(cellWidth == 0 || pCell->m_MaxWidth > pTable->m_TableWidth) { CRF_Table* pTable = m_TableArray.GetAt(m_TableArray.GetSize() - 1); pCell->m_MaxWidth = reflowWidth / pTable->m_nCell.GetAt(pTable->m_nCell.GetSize() - 1); } else { pCell->m_MaxWidth = pElement->GetNumberAttr(LayoutWidth) * reflowWidth / pTable->m_TableWidth; } pCell->m_ColSpan = (int)(pElement->GetNumberAttr(LayoutColSpan)); pCell->m_RowSpan = (int)(pElement->GetNumberAttr(LayoutRowSpan)); if(!pCell->m_ColSpan) { pCell->m_ColSpan = 1; } if(!pCell->m_RowSpan ) { pCell->m_RowSpan = 1; } pCell->m_BlockAlign = pElement->GetEnumAttr(LayoutBlockAlign); m_TextAlign = pElement->GetEnumAttr(LayoutInlineAlign); pCell->m_PosX = 0; pCell->m_PosY = 0; reflowWidth = pCell->m_MaxWidth; pTable->m_pCellArray.Add(pCell); break; } default: break; } m_fLineHeight = pElement->GetNumberAttr(LayoutLineHeight); int ReflowedSize = m_pReflowedPage->m_pReflowed->GetSize(); if(pElement->CountObjects()) { ProcessObjs(pElement, reflowWidth); } } int count = pElement->CountChildren(); for(int i = 0; i < count; i++) { IPDF_LayoutElement* pChildElement = pElement->GetChild(i); ProcessElement(pChildElement, reflowWidth); if(m_pPause && m_pRootElement == pElement && m_Status != LayoutToBeContinued ) { if(m_pPause->NeedToPauseNow()) { m_pLayoutElement = pChildElement; m_Status = LayoutToBeContinued; m_CurrRefWidth = reflowWidth; m_PausePosition = (i + 1) * 100 / (count + 1); return ; } } if(m_Status == LayoutToBeContinued && m_pLayoutElement == pChildElement) { m_Status = LayoutReady; } } if(m_Status == LayoutReady) { FX_FLOAT dx = 0; LayoutType layoutType = pElement->GetType(); FX_INT32 ElementType = GetElementTypes(layoutType); switch(ElementType) { case SST_IE: m_bIllustration = FALSE; FinishedCurrLine(); break; case SST_BLSE: FinishedCurrLine(); FX_FLOAT StartIndent = 0; if(IPDF_LayoutElement* pParent = pElement->GetParent()) { StartIndent = pParent->GetNumberAttr(LayoutStartIndent); } FX_FLOAT currStartIndent = pElement->GetNumberAttr(LayoutStartIndent); if(StartIndent != currStartIndent) { reflowWidth += ConverWidth(currStartIndent); dx += ConverWidth(currStartIndent); } FX_FLOAT spaceAfter = pElement->GetNumberAttr(LayoutSpaceAfter); m_pReflowedPage->m_PageHeight += spaceAfter; break; } switch(layoutType) { case LayoutTableDataCell: case LayoutTableHeaderCell: { if(!m_TableArray.GetSize()) { break; } CRF_Table* pTable = m_TableArray.GetAt(m_TableArray.GetSize() - 1); RF_TableCell* pCell = pTable->m_pCellArray.GetAt(pTable->m_pCellArray.GetSize() - 1); pCell->m_EndPos = m_pReflowedPage->m_pReflowed->GetSize() - 1; if(pCell->m_EndPos < pCell->m_BeginPos) { pCell->m_CellHeight = 0; } else { CRF_Data* pBeginData = (*m_pReflowedPage->m_pReflowed)[pCell->m_BeginPos]; CRF_Data* pEndData = (*m_pReflowedPage->m_pReflowed)[pCell->m_EndPos]; pCell->m_CellHeight = pBeginData->m_Height > pEndData->m_Height ? pBeginData->m_Height : pEndData->m_Height; pCell->m_CellHeight -= pEndData->m_PosY - pBeginData->m_PosY; } break; } case LayoutTableRow: { if(!m_TableArray.GetSize()) { break; } CRF_Table* pTable = m_TableArray.GetAt(m_TableArray.GetSize() - 1); if(pTable->m_nCol == 0) { pTable->m_nCol = pTable->m_pCellArray.GetSize(); } break; } case LayoutTable: { ProcessTable(dx); break; } default: if(dx) { CFX_AffineMatrix matrix(1, 0, 0, 1, dx, 0); int ReflowedSize = m_pReflowedPage->m_pReflowed->GetSize(); Transform(&matrix, m_pReflowedPage->m_pReflowed, ReflowedSize, m_pReflowedPage->m_pReflowed->GetSize() - ReflowedSize); } } } if(m_pRootElement == pElement) { m_PausePosition = 100; } } FX_INT32 CPDF_LayoutProcessor_Reflow::GetElementTypes(LayoutType layoutType) { switch(layoutType) { case LayoutParagraph: case LayoutHeading: case LayoutHeading1: case LayoutHeading2: case LayoutHeading3: case LayoutHeading4: case LayoutHeading5: case LayoutHeading6: case LayoutList: case LayoutListItem: case LayoutListLabel: case LayoutListBody: case LayoutTable: case LayoutTableHeaderCell: case LayoutTableDataCell: case LayoutTableRow: case LayoutTableHeaderGroup: case LayoutTableBodyGroup: case LayoutTableFootGroup: case LayoutTOCI: case LayoutCaption: return SST_BLSE; case LayoutFigure: case LayoutFormula: case LayoutForm: return SST_IE; case LayoutSpan: case LayoutQuote: case LayoutNote: case LayoutReference: case LayoutBibEntry: case LayoutCode: case LayoutLink: case LayoutAnnot: case LayoutRuby: case LayoutWarichu: return SST_ILSE; default: return SST_GE; } return FALSE; } FX_FLOAT CPDF_LayoutProcessor_Reflow::ConverWidth(FX_FLOAT width) { return width; } void CPDF_LayoutProcessor_Reflow::ProcessObject(CPDF_PageObject* pObj, FX_FLOAT reflowWidth, CFX_AffineMatrix objMatrix) { if(!pObj) { return; } if(pObj->m_Type == PDFPAGE_TEXT) { ProcessTextObject( (CPDF_TextObject *)pObj, reflowWidth, objMatrix); } else if(pObj->m_Type == PDFPAGE_IMAGE) { if(!(m_flags & RF_PARSER_IMAGE)) { return; } CPDF_PageObjects* pObjs = FX_NEW CPDF_PageObjects(FALSE); if (NULL == pObjs) { return; } FX_POSITION pos = pObjs->GetLastObjectPosition(); pos = pObjs->InsertObject(pos, pObj); CFX_AffineMatrix matrix; FX_RECT rect = pObj->GetBBox(&matrix); CPDF_ImageObject* ImageObj = (CPDF_ImageObject*)pObj; ProcessUnitaryObjs(pObjs, reflowWidth, objMatrix); delete pObjs; } else if(pObj->m_Type == PDFPAGE_PATH) { } else if(pObj->m_Type == PDFPAGE_FORM) { CPDF_FormObject* pForm = (CPDF_FormObject*)pObj; FX_POSITION pos = pForm->m_pForm->GetFirstObjectPosition(); objMatrix.Concat(pForm->m_FormMatrix); while (pos) { CPDF_PageObject* pObj1 = pForm->m_pForm->GetNextObject(pos); ProcessObject(pObj1, reflowWidth, objMatrix); } } } void CPDF_LayoutProcessor_Reflow::ProcessObjs(IPDF_LayoutElement* pElement, FX_FLOAT reflowWidth) { m_fCurrMaxWidth = reflowWidth; int ObjCount = pElement->CountObjects(); for(int i = 0; i < ObjCount; i++) { CPDF_PageObject* pObj = pElement->GetObject(i); ProcessObject(pObj, reflowWidth, m_PDFMatrix); continue; } } void CPDF_LayoutProcessor_Reflow::AddTemp2CurrLine(int begin, int count) { if(begin < 0 || count <= 0 || !m_pReflowedPage || !m_pReflowedPage->m_pReflowed || !m_pTempLine) { return; } else { count += begin; } int size = m_pReflowedPage->m_pReflowed->GetSize(); int temps = m_pTempLine->GetSize(); for(int i = begin; i < count; i++) { CRF_Data* pData = (*m_pTempLine)[i]; AddData2CurrLine(pData); } } void CPDF_LayoutProcessor_Reflow::AddData2CurrLine(CRF_Data* pData) { if(pData == NULL || m_pCurrLine == NULL) { return; } m_pCurrLine->Add(pData); m_fCurrLineWidth = pData->m_PosX + pData->m_Width; if(pData->m_Height > m_fCurrLineHeight) { m_fCurrLineHeight = pData->m_Height; } } void CPDF_LayoutProcessor_Reflow::UpdateCurrLine() { } void CPDF_LayoutProcessor_Reflow::Transform(const CFX_AffineMatrix* pMatrix, CRF_DataPtrArray* pDataArray, int beginPos, int count) { if (!pDataArray) { return; } if(count == 0) { count = pDataArray->GetSize(); } else { count += beginPos; } for(int i = beginPos; i < count; i++) { CRF_Data* pData = (*pDataArray)[i]; Transform(pMatrix, pData); } } void CPDF_LayoutProcessor_Reflow::Transform(const CFX_AffineMatrix* pMatrix, CRF_Data* pData) { if(pData->GetType() == CRF_Data::Path) { CRF_PathData* pPathData = (CRF_PathData*)pData; pPathData->m_pPath2Device.Concat(*pMatrix); } pMatrix->Transform(pData->m_PosX, pData->m_PosY, pData->m_PosX, pData->m_PosY); } FX_BOOL CPDF_LayoutProcessor_Reflow::FinishedCurrLine() { if (NULL == m_pCurrLine) { return FALSE; } int count = m_pCurrLine->GetSize(); if(count == 0) { return FALSE; } if(m_fLineHeight > m_fCurrLineHeight) { m_fCurrLineHeight = m_fLineHeight; } else { m_fCurrLineHeight += 2; } if(m_pReflowedPage->m_pReflowed->GetSize() > 0) { m_fCurrLineHeight += m_fLineSpace; } FX_FLOAT height = m_pReflowedPage->m_PageHeight + m_fCurrLineHeight; FX_FLOAT lineHeight = m_fLineHeight; if(lineHeight == 0) { lineHeight = m_fCurrLineHeight; } FX_FLOAT dx = 0; switch(m_TextAlign) { case LayoutCenter: dx = (m_fCurrMaxWidth - m_fCurrLineWidth) / 2; break; case LayoutEnd: dx = m_fCurrMaxWidth - m_fCurrLineWidth; break; case LayoutJustify: break; default: break; } FX_FLOAT dy = - height; int refedSize = m_pReflowedPage->m_pReflowed->GetSize(); if(count == 13) { int a = 0; } for(int i = 0; i < count; i++) { CRF_Data* pData = (*m_pCurrLine)[i]; m_pReflowedPage->m_pReflowed->Add(pData); FX_FLOAT x = m_StartIndent + dx * (m_TextAlign == LayoutJustify ? i + 1 : 1); CFX_AffineMatrix matrix(1, 0, 0, 1, x, dy); Transform(&matrix, pData); } m_pCurrLine->RemoveAll(); m_fCurrLineWidth = 0; m_pReflowedPage->m_PageHeight += m_fCurrLineHeight; m_fCurrLineHeight = 0; return TRUE; } CRF_CharState* CPDF_LayoutProcessor_Reflow::GetCharState(CPDF_TextObject* pObj, CPDF_Font* pFont, FX_FLOAT fHeight, FX_ARGB color) { if (NULL == m_pReflowedPage->m_pCharState) { return NULL; } int count = m_pReflowedPage->m_pCharState->GetSize(); for(int i = count - 1; i >= 0; i--) { CRF_CharState* pState = (CRF_CharState*)m_pReflowedPage->m_pCharState->GetAt(i); if(pState->m_Color == color && pState->m_fFontSize == fHeight && pState->m_pFont == pFont && pState->m_pTextObj == pObj) { return pState; } } CRF_CharState pState; pState.m_pTextObj = pObj; pState.m_Color = color; pState.m_pFont = pFont; pState.m_fFontSize = fHeight; int ascent = pFont->GetTypeAscent(); int descent = pFont->GetTypeDescent(); pState.m_fAscent = ascent * fHeight / (ascent - descent); if(descent == 0) { pState.m_fDescent = 0; } else { pState.m_fDescent = descent * fHeight / (ascent - descent); } pState.m_bVert = FALSE; CPDF_CIDFont *pCIDFont = pFont->GetCIDFont(); if(pCIDFont) { pState.m_bVert = pCIDFont->IsVertWriting(); } m_pReflowedPage->m_pCharState->Add(pState); return (CRF_CharState*)m_pReflowedPage->m_pCharState->GetAt(count); } int CPDF_LayoutProcessor_Reflow::GetCharWidth(FX_DWORD charCode, CPDF_Font* pFont) const { if(charCode == -1) { return 0; } int w = pFont->GetCharWidthF(charCode); if(w == 0) { CFX_ByteString str; pFont->AppendChar(str, charCode); w = pFont->GetStringWidth(str, 1); if(w == 0) { FX_RECT BBox; pFont->GetCharBBox(charCode, BBox); w = BBox.right - BBox.left; } } return w; } void CPDF_LayoutProcessor_Reflow::CreateRFData(CPDF_PageObject* pObj, CFX_AffineMatrix* pObjMatrix) { if (NULL == m_pReflowedPage->m_pMemoryPool) { return; } if(pObj->m_Type == PDFPAGE_TEXT) { CPDF_TextObject* pTextObj = (CPDF_TextObject* )pObj; int count = pTextObj->CountItems(); if(!count) { return; } if(count == 1) { CPDF_TextObjectItem Item; pTextObj->GetItemInfo(0, &Item); if(Item.m_CharCode == 49) { int a = 0; } } CPDF_Font * pFont = pTextObj->GetFont(); FX_FLOAT fs = pTextObj->GetFontSize(); FX_FLOAT* pmatrix = pTextObj->m_TextState.GetMatrix(); FX_FLOAT matrix1 = pmatrix[1]; if(pmatrix[2] == 0) { matrix1 = 0; } CFX_AffineMatrix textMatrix(pmatrix[0], matrix1, pmatrix[2], pmatrix[3], 0, 0); FX_FLOAT height = FXSYS_fabs(textMatrix.TransformDistance(fs)); if(pObjMatrix) { height = FXSYS_fabs(pObjMatrix->TransformDistance(height)); } int r = 0, g = 0, b = 0; pTextObj->m_ColorState.GetFillColor()->GetRGB(r, g, b); FX_ARGB col = r * 0x10000; col += g * 0x100; col += b; CRF_CharState* pState = GetCharState(pTextObj, pFont, height, col); FX_FLOAT dx = 0, dy = 0; FX_RECT ObjBBox; if(pObjMatrix) { ObjBBox = pTextObj->GetBBox(pObjMatrix); dx = (float)ObjBBox.left; dy = (float)ObjBBox.bottom; } else { CFX_AffineMatrix matrix; ObjBBox = pTextObj->GetBBox(&matrix); } FX_FLOAT objWidth = 0; CFX_ByteString str; FX_BOOL bOrder = TRUE; CFX_PtrArray tempArray; int i = 0; CPDF_TextObjectItem Item; pTextObj->GetItemInfo(i, &Item); dx = Item.m_OriginX; dy = Item.m_OriginY; textMatrix.Transform(Item.m_OriginX, Item.m_OriginY, dx, dy); CRF_CharData* pLastData = NULL; FX_FLOAT horzScale = pTextObj->m_TextState.GetFontSizeV() / pTextObj->m_TextState.GetFontSizeH(); while(i < count) { pTextObj->GetItemInfo(i, &Item); if(Item.m_CharCode == -1) { i++; continue; } FX_FLOAT OriginX, OriginY; textMatrix.Transform(Item.m_OriginX, Item.m_OriginY, OriginX, OriginY); CRF_CharData* pData = (CRF_CharData*)m_pReflowedPage->m_pMemoryPool->Alloc(sizeof(CRF_CharData)); if (NULL == pData) { continue; } pData->m_Type = CRF_Data::Text; if(FXSYS_fabs(OriginY - dy) > FXSYS_fabs(OriginX - dx)) { pData->m_PosY = dy; pData->m_PosX = pLastData->m_PosX + pLastData->m_Width + textMatrix.TransformDistance(pTextObj->m_TextState.GetObject()->m_CharSpace); } else { pData->m_PosY = OriginY; pData->m_PosX = OriginX; } int size = tempArray.GetSize(); if(size && pData->m_PosX < pLastData->m_PosX ) { for (int j = 0; j < size; j++) { CRF_CharData* pData1 = (CRF_CharData*)tempArray.GetAt(j); if(pData1->m_PosX > pData->m_PosX) { tempArray.InsertAt(j, pData); break; } } } else { tempArray.Add(pData); } pLastData = pData; pData->m_CharCode = Item.m_CharCode; pData->m_Height = FXSYS_fabs(height); int w = GetCharWidth(Item.m_CharCode, pFont); pData->m_Width = FXSYS_fabs(fs * textMatrix.TransformDistance((FX_FLOAT)w) / 1000); if(horzScale) { pData->m_Width /= horzScale; } pData->m_pCharState = pState; i++; } count = tempArray.GetSize(); for (int j = 0; j < count; j++) { CRF_CharData* pData = (CRF_CharData*)tempArray.GetAt(j); if (m_pTempLine) { m_pTempLine->Add(pData); } } tempArray.RemoveAll(); } else if(pObj->m_Type == PDFPAGE_IMAGE) { CPDF_ImageObject* pImageObj = (CPDF_ImageObject* )pObj; CRF_ImageData* pRFImage = (CRF_ImageData*)m_pReflowedPage->m_pMemoryPool->Alloc(sizeof(CRF_ImageData)); if (NULL == pRFImage) { return; } pRFImage->m_pBitmap = NULL; pRFImage->m_Type = CRF_Data::Image; if (m_pTempLine) { m_pTempLine->Add(pRFImage); } CPDF_Image *pImage = pImageObj->m_pImage; if (!pImage->m_pDIBSource || !pImage->m_pMask) { if(pImage->StartLoadDIBSource(m_pReflowedPage->GetFormResDict(pImageObj), m_pReflowedPage->m_pPDFPage->m_pResources, 0, 0, TRUE)) { pImage->Continue(NULL); } } CFX_DIBSource* pDibSource = pImage->DetachBitmap(); if (pDibSource) { pRFImage->m_pBitmap = pDibSource->Clone(); delete pDibSource; } CFX_DIBSource* pMask = pImage->DetachMask(); if (pMask) { if (!pMask->IsAlphaMask()) { CFX_DIBitmap* pMaskBmp = pMask->Clone(); pMaskBmp->ConvertFormat(FXDIB_8bppMask); pRFImage->m_pBitmap->MultiplyAlpha(pMaskBmp); delete pMaskBmp; } else { pRFImage->m_pBitmap->MultiplyAlpha(pMask); } delete pMask; } CFX_FloatRect ObjBBox; if(pObjMatrix) { ObjBBox = pImageObj->GetBBox(pObjMatrix); } else { CFX_AffineMatrix matrix; ObjBBox = pImageObj->GetBBox(&matrix); } pRFImage->m_Width = ObjBBox.Width(); pRFImage->m_Height = ObjBBox.Height(); pRFImage->m_PosX = 0; pRFImage->m_PosY = 0; CFX_AffineMatrix matrix(1, 0, 0, -1, 0, 0); matrix.Concat(pImageObj->m_Matrix); matrix.Concat(*pObjMatrix); pRFImage->m_Matrix.Set(matrix.a == 0 ? 0 : matrix.a / FXSYS_fabs(matrix.a), matrix.b == 0 ? 0 : matrix.b / FXSYS_fabs(matrix.b), matrix.c == 0 ? 0 : matrix.c / FXSYS_fabs(matrix.c), matrix.d == 0 ? 0 : matrix.d / FXSYS_fabs(matrix.d), 0, 0); } else if(pObj->m_Type == PDFPAGE_PATH) { } } FX_FLOAT CPDF_LayoutProcessor_Reflow:: GetDatasWidth(int beginPos, int endpos) { if(endpos < beginPos || !m_pTempLine) { return 0; } if(endpos > m_pTempLine->GetSize() - 1) { endpos = m_pTempLine->GetSize() - 1; } CRF_Data* pBeginData = (*m_pTempLine)[beginPos]; CRF_Data* pEndData = (*m_pTempLine)[endpos]; return pEndData->m_PosX - pBeginData->m_PosX + pEndData->m_Width; } FX_WCHAR CPDF_LayoutProcessor_Reflow::GetPreChar() { if (NULL == m_pCurrLine) { return -1; } int index = m_pCurrLine->GetSize() - 1; CRF_CharData* pCharData = NULL; while (index >= 0 && !pCharData) { CRF_Data* pData = (*m_pCurrLine)[index]; if(pData->GetType() == CRF_Data::Text) { pCharData = (CRF_CharData*)pData; } else { return -1; } index --; } if(m_pReflowedPage) { index = m_pReflowedPage->m_pReflowed->GetSize() - 1; } while(!pCharData && index >= 0) { CRF_Data* pData = (*m_pReflowedPage->m_pReflowed)[index]; if(pData->GetType() == CRF_Data::Text) { pCharData = (CRF_CharData*)pData; } else { return -1; } index --; } if(pCharData) { CFX_WideString str = pCharData->m_pCharState->m_pFont->UnicodeFromCharCode(pCharData->m_CharCode); return str.GetAt(0); } return -1; } int CPDF_LayoutProcessor_Reflow::ProcessInsertObject(CPDF_TextObject* pObj, CFX_AffineMatrix formMatrix) { if(!pObj || !m_pPreObj || !m_pCurrLine) { return 0; } if(m_pCurrLine->GetSize() == 0) { return 0; } CPDF_TextObjectItem item; int nItem = m_pPreObj->CountItems(); m_pPreObj->GetItemInfo(nItem - 1, &item); FX_FLOAT last_pos = item.m_OriginX; FX_FLOAT last_width = GetCharWidth(item.m_CharCode, m_pPreObj->GetFont()) * m_pPreObj->GetFontSize() / 1000; last_width = FXSYS_fabs(last_width); pObj->GetItemInfo(0, &item); FX_FLOAT this_width = GetCharWidth(item.m_CharCode, pObj->GetFont()) * pObj->GetFontSize() / 1000; this_width = FXSYS_fabs(this_width); FX_FLOAT threshold = last_width > this_width ? last_width / 4 : this_width / 4; CFX_AffineMatrix prev_matrix, prev_reverse; m_pPreObj->GetTextMatrix(&prev_matrix); prev_matrix.Concat(m_perMatrix); prev_reverse.SetReverse(prev_matrix); FX_FLOAT x = pObj->GetPosX(), y = pObj->GetPosY(); formMatrix.Transform(x, y); prev_reverse.Transform(x, y); FX_WCHAR preChar = GetPreChar(); CFX_WideString wstrItem = pObj->GetFont()->UnicodeFromCharCode(item.m_CharCode); FX_WCHAR curChar = wstrItem.GetAt(0); if (FXSYS_fabs(y) > threshold * 2) { if (preChar == L'-') { return 3; } if (preChar != L' ') { return 1; } return 2; } if ((x - last_pos - last_width) > threshold && curChar != L' ' && preChar != L' ') { return 1; } return 0; } FX_INT32 CPDF_LayoutProcessor_Reflow::LogicPreObj(CPDF_TextObject* pObj) { CPDF_TextObject* pPreObj = m_pPreObj; m_pPreObj = pObj; if(!pObj || !pPreObj) { return 0; } CPDF_TextObjectItem item; pPreObj->GetItemInfo(pPreObj->CountItems() - 1, &item); FX_FLOAT last_pos = item.m_OriginX; FX_FLOAT last_width = pPreObj->GetFont()->GetCharWidthF(item.m_CharCode) * pPreObj->GetFontSize() / 1000; last_width = FXSYS_fabs(last_width); pObj->GetItemInfo(0, &item); FX_FLOAT this_width = pObj->GetFont()->GetCharWidthF(item.m_CharCode) * pObj->GetFontSize() / 1000; this_width = FXSYS_fabs(this_width); FX_FLOAT threshold = last_width > this_width ? last_width / 4 : this_width / 4; CFX_AffineMatrix prev_matrix, prev_reverse; pPreObj->GetTextMatrix(&prev_matrix); prev_reverse.SetReverse(prev_matrix); FX_FLOAT x = pObj->GetPosX(), y = pObj->GetPosY(); prev_reverse.Transform(x, y); CFX_WideString wstrItem = pObj->GetFont()->UnicodeFromCharCode(item.m_CharCode); FX_WCHAR curChar = wstrItem.GetAt(0); if (FXSYS_fabs(y) > threshold * 2) { return 2; } FX_WCHAR preChar = 0; if (FXSYS_fabs(last_pos + last_width - x) > threshold && curChar != L' ') { return 1; } return 0; m_pPreObj = pObj; if(!pPreObj) { return 0; } if(pPreObj->m_Type != pObj->m_Type) { return 0; } CFX_FloatRect rcCurObj(pObj->m_Left, pObj->m_Bottom, pObj->m_Right, pObj->m_Top); CFX_FloatRect rcPreObj(pPreObj->m_Left, pPreObj->m_Bottom, pPreObj->m_Right, pPreObj->m_Top); if(pObj->m_Type == PDFPAGE_IMAGE) { if(rcPreObj.Contains(rcCurObj)) { return 2; } if(rcCurObj.Contains(rcPreObj)) { return 2; } return 0; } if(pObj->m_Type == PDFPAGE_TEXT) { if(!((rcPreObj.bottom > rcCurObj.top) || (rcPreObj.top < rcCurObj.bottom))) { FX_FLOAT height = FX_MIN(rcPreObj.Height(), rcCurObj.Height()); if((rcCurObj.left - rcPreObj.right) > height / 3) { return 3; } } if(FXSYS_fabs(rcPreObj.Width() - rcCurObj.Width()) >= 2 || FXSYS_fabs(rcPreObj.Height() - rcCurObj.Height()) >= 2 ) { return 0; } CPDF_TextObject* pPreTextObj = (CPDF_TextObject*)pPreObj; CPDF_TextObject* pCurTextObj = (CPDF_TextObject*)pObj; int nPreCount = pPreTextObj->CountItems(); int nCurCount = pCurTextObj->CountItems(); if (nPreCount != nCurCount) { return 0; } FX_BOOL bSame = TRUE; for (int i = 0; i < nPreCount; i++) { CPDF_TextObjectItem itemPer, itemCur; pPreTextObj->GetItemInfo(i, &itemPer); pCurTextObj->GetItemInfo(i, &itemCur); if (itemCur.m_CharCode != itemPer.m_CharCode) { return 0; } if (itemCur.m_OriginX != itemPer.m_OriginX) { bSame = FALSE; } if (itemCur.m_OriginY != itemPer.m_OriginY) { bSame = FALSE; } } if(rcPreObj.left == rcCurObj.left && rcPreObj.top == rcCurObj.top) { return 1; } if(FXSYS_fabs(rcPreObj.left - rcCurObj.left) < rcPreObj.Width() / 3 && FXSYS_fabs(rcPreObj.top - rcCurObj.top) < rcPreObj.Height() / 3) { return 2; } } return 0; } FX_BOOL CPDF_LayoutProcessor_Reflow::IsSameTextObject(CPDF_TextObject* pTextObj1, CPDF_TextObject* pTextObj2) { if (!pTextObj1 || !pTextObj2) { return FALSE; } CFX_FloatRect rcPreObj(pTextObj2->m_Left, pTextObj2->m_Bottom, pTextObj2->m_Right, pTextObj2->m_Top); CFX_FloatRect rcCurObj(pTextObj1->m_Left, pTextObj1->m_Bottom, pTextObj1->m_Right, pTextObj1->m_Top); if (rcPreObj.IsEmpty() && rcCurObj.IsEmpty()) { return FALSE; } if (!rcPreObj.IsEmpty() || !rcCurObj.IsEmpty()) { rcPreObj.Intersect(rcCurObj); if (rcPreObj.IsEmpty()) { return FALSE; } if (FXSYS_fabs(rcPreObj.Width() - rcCurObj.Width()) > rcCurObj.Width() / 2) { return FALSE; } if (pTextObj2->GetFontSize() != pTextObj1->GetFontSize()) { return FALSE; } } int nPreCount = pTextObj2->CountItems(); int nCurCount = pTextObj1->CountItems(); if (nPreCount != nCurCount) { return FALSE; } for (int i = 0; i < nPreCount; i++) { CPDF_TextObjectItem itemPer, itemCur; pTextObj2->GetItemInfo(i, &itemPer); pTextObj1->GetItemInfo(i, &itemCur); if (itemCur.m_CharCode != itemPer.m_CharCode) { return FALSE; } } return TRUE; } void CPDF_LayoutProcessor_Reflow::ProcessTextObject(CPDF_TextObject *pTextObj, FX_FLOAT reflowWidth, CFX_AffineMatrix objMatrix) { if(reflowWidth < 0 || !m_pCurrLine || !m_pTempLine) { return; } if(IsSameTextObject(pTextObj, m_pPreObj)) { return; } CPDF_PageObject* pPreObj = m_pPreObj; FX_INT32 logic = ProcessInsertObject(pTextObj, objMatrix); m_pPreObj = pTextObj; m_perMatrix.Copy(objMatrix); int size = m_pTempLine->GetSize(); int curs = m_pCurrLine->GetSize(); CreateRFData(pTextObj); size = m_pTempLine->GetSize(); int reds = m_pReflowedPage->m_pReflowed->GetSize(); if(size == 0) { return; } if(logic == 1) { m_fCurrLineWidth += pTextObj->GetBBox(&objMatrix).Height() / 3; } else if(logic == 3 && curs) { m_fCurrLineWidth -= (*m_pCurrLine)[curs - 1]->m_Width; m_pCurrLine->Delete(curs - 1); } int beginPos = 0, endPos = m_pTempLine->GetSize() - 1; while(beginPos <= endPos) { int tempBeginPos = beginPos; int tempEndPos = endPos; FX_FLOAT all_width = GetDatasWidth( beginPos, endPos); if(all_width < reflowWidth - m_fCurrLineWidth) { CRF_CharData* pBeginData = (CRF_CharData*)(*m_pTempLine)[beginPos]; CFX_AffineMatrix matrix(1, 0, 0, 1, -pBeginData->m_PosX + m_fCurrLineWidth, -pBeginData->m_PosY); Transform(&matrix, m_pTempLine, beginPos, endPos - beginPos + 1); AddTemp2CurrLine(beginPos, endPos - beginPos + 1); m_pTempLine->RemoveAll(); return; } int midPos ; if(tempBeginPos >= tempEndPos && tempEndPos != 0) { midPos = tempEndPos; } else { while (tempBeginPos < tempEndPos ) { midPos = (tempEndPos - tempBeginPos) / 2 + tempBeginPos; if(midPos == tempBeginPos || midPos == tempEndPos) { break; } FX_FLOAT w = GetDatasWidth( beginPos, midPos); if(w < reflowWidth - m_fCurrLineWidth) { tempBeginPos = midPos; } else { tempEndPos = midPos; } } midPos = tempBeginPos; if(midPos == 0) { FX_FLOAT w = GetDatasWidth( beginPos, 1); if(w > reflowWidth - m_fCurrLineWidth) { midPos = -1; } } } if(midPos == -1) { int count = m_pCurrLine->GetSize(); if(count == 0) { midPos = 0; } } int f = -1; int i = 0; for(i = midPos; i >= beginPos; i--) { CRF_CharData* pData = (CRF_CharData*)(*m_pTempLine)[i]; CFX_WideString Wstr = pData->m_pCharState->m_pFont->UnicodeFromCharCode(pData->m_CharCode); FX_WCHAR cha = Wstr.GetAt(0); if(i < m_pTempLine->GetSize() - 1) { CRF_CharData* pNextData = (CRF_CharData*)(*m_pTempLine)[i + 1]; if(pNextData->m_PosX - (pData->m_PosX + pData->m_Width) >= pData->m_Height / 4) { f = i; i++; } } if(f == -1) { if(IsCanBreakAfter((FX_DWORD)cha)) { f = i; i++; } else if(IsCanBreakBefore((FX_DWORD)cha)) { f = i - 1; if(f < beginPos) { f = -1; } } } if(f != -1) { CRF_CharData* pBeginData = (CRF_CharData*)(*m_pTempLine)[beginPos]; CFX_AffineMatrix matrix(1, 0, 0, 1, -pBeginData->m_PosX + m_fCurrLineWidth, -pBeginData->m_PosY); Transform(&matrix, m_pTempLine, beginPos, f - beginPos + 1); CRF_Data* pData = (*m_pTempLine)[0]; AddTemp2CurrLine(beginPos, f - beginPos + 1); beginPos = i; FinishedCurrLine(); f = 1; break; } } if(f == -1 && i < beginPos) { if( m_pCurrLine->GetSize()) { int count = m_pCurrLine->GetSize(); f = -1; for(int i = count - 1; i >= 0; i--) { CRF_Data* pData = (*m_pCurrLine)[i]; if(pData->GetType() != CRF_Data::Text) { f = i + 1; } else { CRF_CharData* pCharData = (CRF_CharData*)pData; CFX_WideString Wstr = pCharData->m_pCharState->m_pFont->UnicodeFromCharCode(pCharData->m_CharCode); FX_WCHAR cha = Wstr.GetAt(0); if(IsCanBreakAfter(cha)) { f = i + 1; i++; } else if(IsCanBreakBefore(cha)) { f = i; } if(f == 0) { f = -1; } } if(f != -1) { FinishedCurrLine(); if(f < count) { int reflowdCount = m_pReflowedPage->m_pReflowed->GetSize(); int pos = reflowdCount + f - count; CRF_CharData* pData = (CRF_CharData*)(*m_pReflowedPage->m_pReflowed)[pos]; CFX_AffineMatrix matrix(1, 0, 0, 1, -pData->m_PosX + m_fCurrLineWidth, -pData->m_PosY); Transform(&matrix, m_pReflowedPage->m_pReflowed, pos, reflowdCount - pos); for(int j = pos; j < reflowdCount; j++) { AddData2CurrLine((*m_pReflowedPage->m_pReflowed)[j]); } m_pReflowedPage->m_pReflowed->Delete(pos, count - f); if(logic == 3) { m_fCurrLineWidth += pTextObj->GetBBox(&objMatrix).Height() / 3; } } break; } } } if(f == -1) { CRF_CharData* pData = (CRF_CharData*)(*m_pTempLine)[beginPos]; CFX_AffineMatrix matrix(1, 0, 0, 1, -pData->m_PosX + m_fCurrLineWidth, -pData->m_PosY); if(beginPos == midPos) { Transform(&matrix, pData); FX_RECT rect; pData->m_pCharState->m_pFont->GetFontBBox(rect); FX_FLOAT* pmatrix = pTextObj->m_TextState.GetMatrix(); CFX_AffineMatrix textMatrix(pmatrix[0], pmatrix[1], pmatrix[2], pmatrix[3], 0, 0); FX_FLOAT width = pData->m_Height * (rect.right - rect.left) / 1000; FX_FLOAT f = (reflowWidth - m_fCurrLineWidth) / width; pData->m_PosY *= f; pData->m_Width *= f; pData->m_Height *= f; pData->m_pCharState = GetCharState(pData->m_pCharState->m_pTextObj, pData->m_pCharState->m_pFont, pData->m_Height, pData->m_pCharState->m_Color); AddData2CurrLine(pData); } else { for(int m = beginPos; m <= midPos; m++) { CRF_CharData* pData = (CRF_CharData*)(*m_pTempLine)[m]; Transform(&matrix, pData); AddData2CurrLine(pData); } } FinishedCurrLine(); beginPos = midPos + 1; } } } m_pTempLine->RemoveAll(); return; } void CPDF_LayoutProcessor_Reflow::ProcessUnitaryObjs(CPDF_PageObjects *pObjs, FX_FLOAT reflowWidth, CFX_AffineMatrix objMatrix) { if(!pObjs) { return; } CFX_FloatRect ObjBBox = pObjs->CalcBoundingBox(); objMatrix.TransformRect(ObjBBox); FX_FLOAT ObjWidth = ObjBBox.Width(); FX_FLOAT ObjHeight = ObjBBox.Height(); CFX_AffineMatrix matrix; if(ObjWidth <= reflowWidth - m_fCurrLineWidth) { matrix.Set(1, 0, 0, 1, m_fCurrLineWidth , 0); } else if(ObjWidth <= reflowWidth) { FinishedCurrLine(); matrix.Set(1, 0, 0, 1, 0, 0); } else { FinishedCurrLine(); FX_FLOAT f = reflowWidth / ObjWidth ; matrix.Set(f, 0, 0, f, 0, 0); } CFX_AffineMatrix tempMatrix = matrix; matrix.Concat(objMatrix); FX_POSITION pos = pObjs->GetFirstObjectPosition(); while(pos) { CPDF_PageObject* pObj = pObjs->GetNextObject(pos); if(pObj->m_Type == PDFPAGE_TEXT) { FX_INT32 ret = LogicPreObj((CPDF_TextObject*)pObj); if(ret == 1 || ret == 2) { continue; } } CreateRFData(pObj, &matrix); } if (m_pTempLine) { Transform(&tempMatrix, m_pTempLine, 0, m_pTempLine->GetSize()); AddTemp2CurrLine(0, m_pTempLine->GetSize()); m_pTempLine->RemoveAll(); } } void CPDF_LayoutProcessor_Reflow::ProcessPathObject(CPDF_PathObject *pObj, FX_FLOAT reflowWidth) { }