summaryrefslogtreecommitdiff
path: root/xfa/fde/cfde_textout.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'xfa/fde/cfde_textout.cpp')
-rw-r--r--xfa/fde/cfde_textout.cpp229
1 files changed, 105 insertions, 124 deletions
diff --git a/xfa/fde/cfde_textout.cpp b/xfa/fde/cfde_textout.cpp
index 9bbeadee58..2e4753a902 100644
--- a/xfa/fde/cfde_textout.cpp
+++ b/xfa/fde/cfde_textout.cpp
@@ -34,29 +34,23 @@ bool IsTextAlignmentTop(const FDE_TextAlignment align) {
bool CFDE_TextOut::DrawString(CFX_RenderDevice* device,
FX_ARGB color,
const CFX_RetainPtr<CFGAS_GEFont>& pFont,
- const FXTEXT_CHARPOS* pCharPos,
+ FXTEXT_CHARPOS* pCharPos,
int32_t iCount,
float fFontSize,
const CFX_Matrix* pMatrix) {
ASSERT(pFont && pCharPos && iCount > 0);
+
CFX_Font* pFxFont = pFont->GetDevFont();
if ((pFont->GetFontStyles() & FX_FONTSTYLE_Italic) != 0 &&
!pFxFont->IsItalic()) {
- FXTEXT_CHARPOS* pCP = (FXTEXT_CHARPOS*)pCharPos;
- float* pAM;
for (int32_t i = 0; i < iCount; ++i) {
static const float mc = 0.267949f;
- pAM = pCP->m_AdjustMatrix;
+ float* pAM = pCharPos->m_AdjustMatrix;
pAM[2] = mc * pAM[0] + pAM[2];
pAM[3] = mc * pAM[1] + pAM[3];
- pCP++;
+ ++pCharPos;
}
}
- FXTEXT_CHARPOS* pCP = (FXTEXT_CHARPOS*)pCharPos;
- CFX_RetainPtr<CFGAS_GEFont> pCurFont;
- CFX_RetainPtr<CFGAS_GEFont> pSTFont;
- FXTEXT_CHARPOS* pCurCP = nullptr;
- int32_t iCurCount = 0;
#if _FXM_PLATFORM_ != _FXM_PLATFORM_WINDOWS_
uint32_t dwFontStyle = pFont->GetFontStyles();
@@ -69,49 +63,58 @@ bool CFDE_TextOut::DrawString(CFX_RenderDevice* device,
FxFont.SetSubstFont(std::move(SubstFxFont));
#endif // _FXM_PLATFORM_ != _FXM_PLATFORM_WINDOWS_
+ CFX_RetainPtr<CFGAS_GEFont> pCurFont;
+ FXTEXT_CHARPOS* pCurCP = nullptr;
+ int32_t iCurCount = 0;
for (int32_t i = 0; i < iCount; ++i) {
- pSTFont = pFont->GetSubstFont((int32_t)pCP->m_GlyphIndex);
- pCP->m_GlyphIndex &= 0x00FFFFFF;
- pCP->m_bFontStyle = false;
+ CFX_RetainPtr<CFGAS_GEFont> pSTFont =
+ pFont->GetSubstFont(static_cast<int32_t>(pCharPos->m_GlyphIndex));
+ pCharPos->m_GlyphIndex &= 0x00FFFFFF;
+ pCharPos->m_bFontStyle = false;
if (pCurFont != pSTFont) {
if (pCurFont) {
pFxFont = pCurFont->GetDevFont();
+
+ CFX_Font* font;
#if _FXM_PLATFORM_ != _FXM_PLATFORM_WINDOWS_
FxFont.SetFace(pFxFont->GetFace());
- device->DrawNormalText(iCurCount, pCurCP, &FxFont, -fFontSize, pMatrix,
- color, FXTEXT_CLEARTYPE);
+ font = &FxFont;
#else
- device->DrawNormalText(iCurCount, pCurCP, pFxFont, -fFontSize, pMatrix,
- color, FXTEXT_CLEARTYPE);
+ font = pFxFont;
#endif // _FXM_PLATFORM_ != _FXM_PLATFORM_WINDOWS_
+
+ device->DrawNormalText(iCurCount, pCurCP, font, -fFontSize, pMatrix,
+ color, FXTEXT_CLEARTYPE);
}
pCurFont = pSTFont;
- pCurCP = pCP;
+ pCurCP = pCharPos;
iCurCount = 1;
} else {
- iCurCount++;
+ ++iCurCount;
}
- pCP++;
+ ++pCharPos;
}
+
+ bool bRet = true;
if (pCurFont && iCurCount) {
pFxFont = pCurFont->GetDevFont();
+ CFX_Font* font;
#if _FXM_PLATFORM_ != _FXM_PLATFORM_WINDOWS_
FxFont.SetFace(pFxFont->GetFace());
- bool bRet = device->DrawNormalText(iCurCount, pCurCP, &FxFont, -fFontSize,
- pMatrix, color, FXTEXT_CLEARTYPE);
- FxFont.SetFace(nullptr);
- return bRet;
+ font = &FxFont;
#else
- return device->DrawNormalText(iCurCount, pCurCP, pFxFont, -fFontSize,
- pMatrix, color, FXTEXT_CLEARTYPE);
+ font = pFxFont;
#endif // _FXM_PLATFORM_ != _FXM_PLATFORM_WINDOWS_
+
+ bRet = device->DrawNormalText(iCurCount, pCurCP, font, -fFontSize, pMatrix,
+ color, FXTEXT_CLEARTYPE);
}
#if _FXM_PLATFORM_ != _FXM_PLATFORM_WINDOWS_
FxFont.SetFace(nullptr);
#endif // _FXM_PLATFORM_ != _FXM_PLATFORM_WINDOWS_
- return true;
+ return bRet;
}
FDE_TTOPIECE::FDE_TTOPIECE() = default;
@@ -133,9 +136,7 @@ CFDE_TextOut::CFDE_TextOut()
m_ttoLines(5),
m_iCurLine(0),
m_iCurPiece(0),
- m_iTotalLines(0) {
- m_Matrix.SetIdentity();
-}
+ m_iTotalLines(0) {}
CFDE_TextOut::~CFDE_TextOut() {}
@@ -190,18 +191,14 @@ void CFDE_TextOut::SetLineBreakTolerance(float fTolerance) {
m_pTxtBreak->SetLineBreakTolerance(m_fTolerance);
}
-void CFDE_TextOut::CalcLogicSize(const wchar_t* pwsStr,
- int32_t iLength,
- CFX_SizeF& size) {
+void CFDE_TextOut::CalcLogicSize(const CFX_WideString& str, CFX_SizeF& size) {
CFX_RectF rtText(0.0f, 0.0f, size.width, size.height);
- CalcLogicSize(pwsStr, iLength, rtText);
+ CalcLogicSize(str, rtText);
size = rtText.Size();
}
-void CFDE_TextOut::CalcLogicSize(const wchar_t* pwsStr,
- int32_t iLength,
- CFX_RectF& rect) {
- if (!pwsStr || iLength < 1) {
+void CFDE_TextOut::CalcLogicSize(const CFX_WideString& str, CFX_RectF& rect) {
+ if (str.IsEmpty()) {
rect.width = 0.0f;
rect.height = 0.0f;
return;
@@ -217,26 +214,21 @@ void CFDE_TextOut::CalcLogicSize(const wchar_t* pwsStr,
}
m_iTotalLines = 0;
- const wchar_t* pStr = pwsStr;
float fWidth = 0.0f;
float fHeight = 0.0f;
float fStartPos = rect.right();
CFX_BreakType dwBreakStatus = CFX_BreakType::None;
- wchar_t wPreChar = 0;
- wchar_t wch;
- wchar_t wBreak = 0;
- while (iLength-- > 0) {
- wch = *pStr++;
- if (wBreak == 0 && (wch == L'\n' || wch == L'\r')) {
- wBreak = wch;
+ bool break_char_is_set = false;
+ for (const wchar_t& wch : str) {
+ if (!break_char_is_set && (wch == L'\n' || wch == L'\r')) {
+ break_char_is_set = true;
m_pTxtBreak->SetParagraphBreakChar(wch);
}
dwBreakStatus = m_pTxtBreak->AppendChar(wch);
if (!CFX_BreakTypeNoneOrPiece(dwBreakStatus))
RetrieveLineWidth(dwBreakStatus, fStartPos, fWidth, fHeight);
-
- wPreChar = 0;
}
+
dwBreakStatus = m_pTxtBreak->EndBreak(CFX_BreakType::Paragraph);
if (!CFX_BreakTypeNoneOrPiece(dwBreakStatus))
RetrieveLineWidth(dwBreakStatus, fStartPos, fWidth, fHeight);
@@ -265,33 +257,32 @@ bool CFDE_TextOut::RetrieveLineWidth(CFX_BreakType dwBreakStatus,
float fLineStep = (m_fLineSpace > m_fFontSize) ? m_fLineSpace : m_fFontSize;
float fLineWidth = 0.0f;
- int32_t iCount = m_pTxtBreak->CountBreakPieces();
- for (int32_t i = 0; i < iCount; i++) {
+ for (int32_t i = 0; i < m_pTxtBreak->CountBreakPieces(); i++) {
const CFX_BreakPiece* pPiece = m_pTxtBreak->GetBreakPieceUnstable(i);
fLineWidth += static_cast<float>(pPiece->m_iWidth) / 20000.0f;
fStartPos =
std::min(fStartPos, static_cast<float>(pPiece->m_iStartPos) / 20000.0f);
}
m_pTxtBreak->ClearBreakPieces();
- if (dwBreakStatus == CFX_BreakType::Paragraph) {
+
+ if (dwBreakStatus == CFX_BreakType::Paragraph)
m_pTxtBreak->Reset();
- }
if (!m_Styles.line_wrap_ && dwBreakStatus == CFX_BreakType::Line) {
fWidth += fLineWidth;
} else {
fWidth = std::max(fWidth, fLineWidth);
fHeight += fLineStep;
}
- m_iTotalLines++;
+ ++m_iTotalLines;
return true;
}
void CFDE_TextOut::DrawLogicText(CFX_RenderDevice* device,
- const wchar_t* pwsStr,
- int32_t iLength,
+ const CFX_WideStringC& str,
const CFX_RectF& rect) {
ASSERT(m_pFont && m_fFontSize >= 1.0f);
- if (!pwsStr || iLength < 1)
+
+ if (str.IsEmpty())
return;
if (rect.width < m_fFontSize || rect.height < m_fFontSize)
return;
@@ -301,7 +292,7 @@ void CFDE_TextOut::DrawLogicText(CFX_RenderDevice* device,
m_ttoLines.clear();
m_wsText.clear();
- LoadText(pwsStr, iLength, rect);
+ LoadText(CFX_WideString(str), rect);
Reload(rect);
DoAlignment(rect);
@@ -332,55 +323,49 @@ void CFDE_TextOut::DrawLogicText(CFX_RenderDevice* device,
device->RestoreState(false);
}
-void CFDE_TextOut::LoadText(const wchar_t* pwsStr,
- int32_t iLength,
- const CFX_RectF& rect) {
- wchar_t* pStr = m_wsText.GetBuffer(iLength);
- int32_t iTxtLength = iLength;
+void CFDE_TextOut::LoadText(const CFX_WideString& str, const CFX_RectF& rect) {
+ ASSERT(!str.IsEmpty());
- ASSERT(iTxtLength >= 0);
- if (pdfium::CollectionSize<int32_t>(m_CharWidths) < iTxtLength)
- m_CharWidths.resize(iTxtLength, 0);
+ m_wsText = str;
+
+ if (pdfium::CollectionSize<int32_t>(m_CharWidths) < str.GetLength())
+ m_CharWidths.resize(str.GetLength(), 0);
float fLineStep = (m_fLineSpace > m_fFontSize) ? m_fLineSpace : m_fFontSize;
float fLineStop = rect.bottom();
m_fLinePos = rect.top;
int32_t iStartChar = 0;
- int32_t iChars = 0;
int32_t iPieceWidths = 0;
CFX_BreakType dwBreakStatus;
- wchar_t wch;
bool bRet = false;
- while (iTxtLength-- > 0) {
- wch = *pwsStr++;
- *pStr++ = wch;
- iChars++;
+ for (const auto& wch : str) {
dwBreakStatus = m_pTxtBreak->AppendChar(wch);
- if (!CFX_BreakTypeNoneOrPiece(dwBreakStatus)) {
- bool bEndofLine =
- RetrievePieces(dwBreakStatus, iStartChar, iPieceWidths, false, rect);
- if (bEndofLine &&
- (m_Styles.line_wrap_ || dwBreakStatus == CFX_BreakType::Paragraph ||
- dwBreakStatus == CFX_BreakType::Page)) {
- iPieceWidths = 0;
- m_iCurLine++;
- m_fLinePos += fLineStep;
- }
- if (m_fLinePos + fLineStep > fLineStop) {
- int32_t iCurLine = bEndofLine ? m_iCurLine - 1 : m_iCurLine;
- m_ttoLines[iCurLine].SetNewReload(true);
- bRet = true;
- break;
- }
+ if (CFX_BreakTypeNoneOrPiece(dwBreakStatus))
+ continue;
+
+ bool bEndofLine =
+ RetrievePieces(dwBreakStatus, iStartChar, iPieceWidths, false, rect);
+ if (bEndofLine &&
+ (m_Styles.line_wrap_ || dwBreakStatus == CFX_BreakType::Paragraph ||
+ dwBreakStatus == CFX_BreakType::Page)) {
+ iPieceWidths = 0;
+ ++m_iCurLine;
+ m_fLinePos += fLineStep;
+ }
+ if (m_fLinePos + fLineStep > fLineStop) {
+ int32_t iCurLine = bEndofLine ? m_iCurLine - 1 : m_iCurLine;
+ m_ttoLines[iCurLine].SetNewReload(true);
+ bRet = true;
+ break;
}
}
+
dwBreakStatus = m_pTxtBreak->EndBreak(CFX_BreakType::Paragraph);
if (!CFX_BreakTypeNoneOrPiece(dwBreakStatus) && !bRet)
RetrievePieces(dwBreakStatus, iStartChar, iPieceWidths, false, rect);
m_pTxtBreak->ClearBreakPieces();
m_pTxtBreak->Reset();
- m_wsText.ReleaseBuffer(iLength);
}
bool CFDE_TextOut::RetrievePieces(CFX_BreakType dwBreakStatus,
@@ -390,8 +375,7 @@ bool CFDE_TextOut::RetrievePieces(CFX_BreakType dwBreakStatus,
const CFX_RectF& rect) {
float fLineStep = (m_fLineSpace > m_fFontSize) ? m_fLineSpace : m_fFontSize;
bool bNeedReload = false;
- float fLineWidth = rect.Width();
- int32_t iLineWidth = FXSYS_round(fLineWidth * 20000.0f);
+ int32_t iLineWidth = FXSYS_round(rect.Width() * 20000.0f);
int32_t iCount = m_pTxtBreak->CountBreakPieces();
for (int32_t i = 0; i < iCount; i++) {
const CFX_BreakPiece* pPiece = m_pTxtBreak->GetBreakPieceUnstable(i);
@@ -411,29 +395,28 @@ bool CFDE_TextOut::RetrievePieces(CFX_BreakType dwBreakStatus,
iWidth += iCurCharWidth;
m_CharWidths[iChar++] = iCurCharWidth;
}
+
if (j == 0 && !bReload) {
m_ttoLines[m_iCurLine].SetNewReload(true);
} else if (j > 0) {
- CFX_RectF rtPiece;
- rtPiece.left = rect.left + (float)pPiece->m_iStartPos / 20000.0f;
- rtPiece.top = m_fLinePos;
- rtPiece.width = iWidth / 20000.0f;
- rtPiece.height = fLineStep;
-
FDE_TTOPIECE ttoPiece;
ttoPiece.iStartChar = iStartChar;
ttoPiece.iChars = j;
- ttoPiece.rtPiece = rtPiece;
ttoPiece.dwCharStyles = pPiece->m_dwCharStyles;
- if (FX_IsOdd(pPiece->m_iBidiLevel)) {
+ ttoPiece.rtPiece = CFX_RectF(
+ rect.left + static_cast<float>(pPiece->m_iStartPos) / 20000.0f,
+ m_fLinePos, iWidth / 20000.0f, fLineStep);
+
+ if (FX_IsOdd(pPiece->m_iBidiLevel))
ttoPiece.dwCharStyles |= FX_TXTCHARSTYLE_OddBidiLevel;
- }
+
AppendPiece(ttoPiece, bNeedReload, (bReload && i == iCount - 1));
}
iStartChar += iPieceChars;
iPieceWidths += iWidth;
}
m_pTxtBreak->ClearBreakPieces();
+
return m_Styles.single_line_ || m_Styles.line_wrap_ || bNeedReload ||
dwBreakStatus == CFX_BreakType::Paragraph;
}
@@ -444,18 +427,19 @@ void CFDE_TextOut::AppendPiece(const FDE_TTOPIECE& ttoPiece,
if (m_iCurLine >= pdfium::CollectionSize<int32_t>(m_ttoLines)) {
CFDE_TTOLine ttoLine;
ttoLine.SetNewReload(bNeedReload);
+
m_iCurPiece = ttoLine.AddPiece(m_iCurPiece, ttoPiece);
m_ttoLines.push_back(ttoLine);
m_iCurLine = pdfium::CollectionSize<int32_t>(m_ttoLines) - 1;
} else {
CFDE_TTOLine* pLine = &m_ttoLines[m_iCurLine];
pLine->SetNewReload(bNeedReload);
+
m_iCurPiece = pLine->AddPiece(m_iCurPiece, ttoPiece);
if (bEnd) {
int32_t iPieces = pLine->GetSize();
- if (m_iCurPiece < iPieces) {
+ if (m_iCurPiece < iPieces)
pLine->RemoveLast(iPieces - m_iCurPiece - 1);
- }
}
}
if (!bEnd && bNeedReload)
@@ -477,26 +461,27 @@ void CFDE_TextOut::Reload(const CFX_RectF& rect) {
void CFDE_TextOut::ReloadLinePiece(CFDE_TTOLine* pLine, const CFX_RectF& rect) {
const wchar_t* pwsStr = m_wsText.c_str();
int32_t iPieceWidths = 0;
+
FDE_TTOPIECE* pPiece = pLine->GetPtrAt(0);
int32_t iStartChar = pPiece->iStartChar;
- m_fLinePos = pPiece->rtPiece.top;
int32_t iPieceCount = pLine->GetSize();
int32_t iPieceIndex = 0;
CFX_BreakType dwBreakStatus = CFX_BreakType::None;
- wchar_t wch;
+ m_fLinePos = pPiece->rtPiece.top;
while (iPieceIndex < iPieceCount) {
int32_t iStar = iStartChar;
int32_t iEnd = pPiece->iChars + iStar;
while (iStar < iEnd) {
- wch = *(pwsStr + iStar);
- dwBreakStatus = m_pTxtBreak->AppendChar(wch);
+ dwBreakStatus = m_pTxtBreak->AppendChar(*(pwsStr + iStar));
if (!CFX_BreakTypeNoneOrPiece(dwBreakStatus))
RetrievePieces(dwBreakStatus, iStartChar, iPieceWidths, true, rect);
- iStar++;
+
+ ++iStar;
}
- iPieceIndex++;
+ ++iPieceIndex;
pPiece = pLine->GetPtrAt(iPieceIndex);
}
+
dwBreakStatus = m_pTxtBreak->EndBreak(CFX_BreakType::Paragraph);
if (!CFX_BreakTypeNoneOrPiece(dwBreakStatus))
RetrievePieces(dwBreakStatus, iStartChar, iPieceWidths, true, rect);
@@ -508,13 +493,11 @@ void CFDE_TextOut::DoAlignment(const CFX_RectF& rect) {
if (m_ttoLines.empty())
return;
- float fLineStopS = rect.bottom();
FDE_TTOPIECE* pFirstPiece = m_ttoLines.back().GetPtrAt(0);
if (!pFirstPiece)
return;
- float fLineStopD = pFirstPiece->rtPiece.bottom();
- float fInc = fLineStopS - fLineStopD;
+ float fInc = rect.bottom() - pFirstPiece->rtPiece.bottom();
if (TextAlignmentVerticallyCentered(m_iAlignment))
fInc /= 2.0f;
else if (IsTextAlignmentTop(m_iAlignment))
@@ -525,22 +508,17 @@ void CFDE_TextOut::DoAlignment(const CFX_RectF& rect) {
for (auto& line : m_ttoLines) {
int32_t iPieces = line.GetSize();
- for (int32_t j = 0; j < iPieces; j++) {
- FDE_TTOPIECE* pPiece = line.GetPtrAt(j);
- pPiece->rtPiece.top += fInc;
- }
+ for (int32_t j = 0; j < iPieces; j++)
+ line.GetPtrAt(j)->rtPiece.top += fInc;
}
}
int32_t CFDE_TextOut::GetDisplayPos(FDE_TTOPIECE* pPiece) {
- FX_TXTRUN tr = ToTextRun(pPiece);
- ASSERT(tr.iLength >= 0);
- if (pdfium::CollectionSize<int32_t>(m_CharPos) < tr.iLength)
- m_CharPos.resize(tr.iLength, FXTEXT_CHARPOS());
- return m_pTxtBreak->GetDisplayPos(&tr, m_CharPos.data());
-}
+ ASSERT(pPiece->iChars >= 0);
+
+ if (pdfium::CollectionSize<int32_t>(m_CharPos) < pPiece->iChars)
+ m_CharPos.resize(pPiece->iChars, FXTEXT_CHARPOS());
-FX_TXTRUN CFDE_TextOut::ToTextRun(const FDE_TTOPIECE* pPiece) {
FX_TXTRUN tr;
tr.wsStr = m_wsText + pPiece->iStartChar;
tr.pWidths = &m_CharWidths[pPiece->iStartChar];
@@ -550,7 +528,8 @@ FX_TXTRUN CFDE_TextOut::ToTextRun(const FDE_TTOPIECE* pPiece) {
tr.dwStyles = m_dwTxtBkStyles;
tr.dwCharStyles = pPiece->dwCharStyles;
tr.pRect = &pPiece->rtPiece;
- return tr;
+
+ return m_pTxtBreak->GetDisplayPos(&tr, m_CharPos.data());
}
CFDE_TTOLine::CFDE_TTOLine() : m_bNewReload(false) {}
@@ -582,6 +561,8 @@ FDE_TTOPIECE* CFDE_TTOLine::GetPtrAt(int32_t index) {
void CFDE_TTOLine::RemoveLast(int32_t icount) {
if (icount < 0)
return;
- icount = std::min(icount, pdfium::CollectionSize<int32_t>(m_pieces));
- m_pieces.erase(m_pieces.end() - icount, m_pieces.end());
+ m_pieces.erase(
+ m_pieces.end() -
+ std::min(icount, pdfium::CollectionSize<int32_t>(m_pieces)),
+ m_pieces.end());
}