xref: /aosp_15_r20/external/pdfium/xfa/fde/cfde_textout.cpp (revision 3ac0a46f773bac49fa9476ec2b1cf3f8da5ec3a4)
1*3ac0a46fSAndroid Build Coastguard Worker // Copyright 2014 The PDFium Authors
2*3ac0a46fSAndroid Build Coastguard Worker // Use of this source code is governed by a BSD-style license that can be
3*3ac0a46fSAndroid Build Coastguard Worker // found in the LICENSE file.
4*3ac0a46fSAndroid Build Coastguard Worker 
5*3ac0a46fSAndroid Build Coastguard Worker // Original code copyright 2014 Foxit Software Inc. http://www.foxitsoftware.com
6*3ac0a46fSAndroid Build Coastguard Worker 
7*3ac0a46fSAndroid Build Coastguard Worker #include "xfa/fde/cfde_textout.h"
8*3ac0a46fSAndroid Build Coastguard Worker 
9*3ac0a46fSAndroid Build Coastguard Worker #include <algorithm>
10*3ac0a46fSAndroid Build Coastguard Worker #include <utility>
11*3ac0a46fSAndroid Build Coastguard Worker 
12*3ac0a46fSAndroid Build Coastguard Worker #include "build/build_config.h"
13*3ac0a46fSAndroid Build Coastguard Worker #include "core/fxcrt/fx_coordinates.h"
14*3ac0a46fSAndroid Build Coastguard Worker #include "core/fxcrt/fx_extension.h"
15*3ac0a46fSAndroid Build Coastguard Worker #include "core/fxcrt/fx_system.h"
16*3ac0a46fSAndroid Build Coastguard Worker #include "core/fxcrt/stl_util.h"
17*3ac0a46fSAndroid Build Coastguard Worker #include "core/fxge/cfx_font.h"
18*3ac0a46fSAndroid Build Coastguard Worker #include "core/fxge/cfx_path.h"
19*3ac0a46fSAndroid Build Coastguard Worker #include "core/fxge/cfx_renderdevice.h"
20*3ac0a46fSAndroid Build Coastguard Worker #include "core/fxge/cfx_substfont.h"
21*3ac0a46fSAndroid Build Coastguard Worker #include "core/fxge/cfx_textrenderoptions.h"
22*3ac0a46fSAndroid Build Coastguard Worker #include "core/fxge/fx_font.h"
23*3ac0a46fSAndroid Build Coastguard Worker #include "core/fxge/text_char_pos.h"
24*3ac0a46fSAndroid Build Coastguard Worker #include "third_party/base/check.h"
25*3ac0a46fSAndroid Build Coastguard Worker #include "third_party/base/check_op.h"
26*3ac0a46fSAndroid Build Coastguard Worker #include "third_party/base/numerics/safe_conversions.h"
27*3ac0a46fSAndroid Build Coastguard Worker #include "xfa/fgas/font/cfgas_gefont.h"
28*3ac0a46fSAndroid Build Coastguard Worker #include "xfa/fgas/layout/cfgas_txtbreak.h"
29*3ac0a46fSAndroid Build Coastguard Worker 
30*3ac0a46fSAndroid Build Coastguard Worker namespace {
31*3ac0a46fSAndroid Build Coastguard Worker 
TextAlignmentVerticallyCentered(const FDE_TextAlignment align)32*3ac0a46fSAndroid Build Coastguard Worker bool TextAlignmentVerticallyCentered(const FDE_TextAlignment align) {
33*3ac0a46fSAndroid Build Coastguard Worker   return align == FDE_TextAlignment::kCenterLeft ||
34*3ac0a46fSAndroid Build Coastguard Worker          align == FDE_TextAlignment::kCenter ||
35*3ac0a46fSAndroid Build Coastguard Worker          align == FDE_TextAlignment::kCenterRight;
36*3ac0a46fSAndroid Build Coastguard Worker }
37*3ac0a46fSAndroid Build Coastguard Worker 
IsTextAlignmentTop(const FDE_TextAlignment align)38*3ac0a46fSAndroid Build Coastguard Worker bool IsTextAlignmentTop(const FDE_TextAlignment align) {
39*3ac0a46fSAndroid Build Coastguard Worker   return align == FDE_TextAlignment::kTopLeft;
40*3ac0a46fSAndroid Build Coastguard Worker }
41*3ac0a46fSAndroid Build Coastguard Worker 
42*3ac0a46fSAndroid Build Coastguard Worker }  // namespace
43*3ac0a46fSAndroid Build Coastguard Worker 
44*3ac0a46fSAndroid Build Coastguard Worker // static
DrawString(CFX_RenderDevice * device,FX_ARGB color,const RetainPtr<CFGAS_GEFont> & pFont,pdfium::span<TextCharPos> pCharPos,float fFontSize,const CFX_Matrix & matrix)45*3ac0a46fSAndroid Build Coastguard Worker bool CFDE_TextOut::DrawString(CFX_RenderDevice* device,
46*3ac0a46fSAndroid Build Coastguard Worker                               FX_ARGB color,
47*3ac0a46fSAndroid Build Coastguard Worker                               const RetainPtr<CFGAS_GEFont>& pFont,
48*3ac0a46fSAndroid Build Coastguard Worker                               pdfium::span<TextCharPos> pCharPos,
49*3ac0a46fSAndroid Build Coastguard Worker                               float fFontSize,
50*3ac0a46fSAndroid Build Coastguard Worker                               const CFX_Matrix& matrix) {
51*3ac0a46fSAndroid Build Coastguard Worker   DCHECK(pFont);
52*3ac0a46fSAndroid Build Coastguard Worker   DCHECK(!pCharPos.empty());
53*3ac0a46fSAndroid Build Coastguard Worker 
54*3ac0a46fSAndroid Build Coastguard Worker   CFX_Font* pFxFont = pFont->GetDevFont();
55*3ac0a46fSAndroid Build Coastguard Worker   if (FontStyleIsItalic(pFont->GetFontStyles()) && !pFxFont->IsItalic()) {
56*3ac0a46fSAndroid Build Coastguard Worker     for (auto& pos : pCharPos) {
57*3ac0a46fSAndroid Build Coastguard Worker       static constexpr float mc = 0.267949f;
58*3ac0a46fSAndroid Build Coastguard Worker       pos.m_AdjustMatrix[2] += mc * pos.m_AdjustMatrix[0];
59*3ac0a46fSAndroid Build Coastguard Worker       pos.m_AdjustMatrix[3] += mc * pos.m_AdjustMatrix[1];
60*3ac0a46fSAndroid Build Coastguard Worker     }
61*3ac0a46fSAndroid Build Coastguard Worker   }
62*3ac0a46fSAndroid Build Coastguard Worker 
63*3ac0a46fSAndroid Build Coastguard Worker #if !BUILDFLAG(IS_WIN)
64*3ac0a46fSAndroid Build Coastguard Worker   uint32_t dwFontStyle = pFont->GetFontStyles();
65*3ac0a46fSAndroid Build Coastguard Worker   CFX_Font FxFont;
66*3ac0a46fSAndroid Build Coastguard Worker   auto SubstFxFont = std::make_unique<CFX_SubstFont>();
67*3ac0a46fSAndroid Build Coastguard Worker   SubstFxFont->m_Weight = FontStyleIsForceBold(dwFontStyle) ? 700 : 400;
68*3ac0a46fSAndroid Build Coastguard Worker   SubstFxFont->m_ItalicAngle = FontStyleIsItalic(dwFontStyle) ? -12 : 0;
69*3ac0a46fSAndroid Build Coastguard Worker   SubstFxFont->m_WeightCJK = SubstFxFont->m_Weight;
70*3ac0a46fSAndroid Build Coastguard Worker   SubstFxFont->m_bItalicCJK = FontStyleIsItalic(dwFontStyle);
71*3ac0a46fSAndroid Build Coastguard Worker   FxFont.SetSubstFont(std::move(SubstFxFont));
72*3ac0a46fSAndroid Build Coastguard Worker #endif
73*3ac0a46fSAndroid Build Coastguard Worker 
74*3ac0a46fSAndroid Build Coastguard Worker   RetainPtr<CFGAS_GEFont> pCurFont;
75*3ac0a46fSAndroid Build Coastguard Worker   TextCharPos* pCurCP = nullptr;
76*3ac0a46fSAndroid Build Coastguard Worker   int32_t iCurCount = 0;
77*3ac0a46fSAndroid Build Coastguard Worker   static constexpr CFX_TextRenderOptions kOptions(CFX_TextRenderOptions::kLcd);
78*3ac0a46fSAndroid Build Coastguard Worker   for (auto& pos : pCharPos) {
79*3ac0a46fSAndroid Build Coastguard Worker     RetainPtr<CFGAS_GEFont> pSTFont =
80*3ac0a46fSAndroid Build Coastguard Worker         pFont->GetSubstFont(static_cast<int32_t>(pos.m_GlyphIndex));
81*3ac0a46fSAndroid Build Coastguard Worker     pos.m_GlyphIndex &= 0x00FFFFFF;
82*3ac0a46fSAndroid Build Coastguard Worker     pos.m_bFontStyle = false;
83*3ac0a46fSAndroid Build Coastguard Worker     if (pCurFont != pSTFont) {
84*3ac0a46fSAndroid Build Coastguard Worker       if (pCurFont) {
85*3ac0a46fSAndroid Build Coastguard Worker         pFxFont = pCurFont->GetDevFont();
86*3ac0a46fSAndroid Build Coastguard Worker 
87*3ac0a46fSAndroid Build Coastguard Worker         CFX_Font* font;
88*3ac0a46fSAndroid Build Coastguard Worker #if !BUILDFLAG(IS_WIN)
89*3ac0a46fSAndroid Build Coastguard Worker         FxFont.SetFace(pFxFont->GetFace());
90*3ac0a46fSAndroid Build Coastguard Worker         FxFont.SetFontSpan(pFxFont->GetFontSpan());
91*3ac0a46fSAndroid Build Coastguard Worker         font = &FxFont;
92*3ac0a46fSAndroid Build Coastguard Worker #else
93*3ac0a46fSAndroid Build Coastguard Worker         font = pFxFont;
94*3ac0a46fSAndroid Build Coastguard Worker #endif
95*3ac0a46fSAndroid Build Coastguard Worker 
96*3ac0a46fSAndroid Build Coastguard Worker         device->DrawNormalText(pdfium::make_span(pCurCP, iCurCount), font,
97*3ac0a46fSAndroid Build Coastguard Worker                                -fFontSize, matrix, color, kOptions);
98*3ac0a46fSAndroid Build Coastguard Worker       }
99*3ac0a46fSAndroid Build Coastguard Worker       pCurFont = pSTFont;
100*3ac0a46fSAndroid Build Coastguard Worker       pCurCP = &pos;
101*3ac0a46fSAndroid Build Coastguard Worker       iCurCount = 1;
102*3ac0a46fSAndroid Build Coastguard Worker     } else {
103*3ac0a46fSAndroid Build Coastguard Worker       ++iCurCount;
104*3ac0a46fSAndroid Build Coastguard Worker     }
105*3ac0a46fSAndroid Build Coastguard Worker   }
106*3ac0a46fSAndroid Build Coastguard Worker 
107*3ac0a46fSAndroid Build Coastguard Worker   bool bRet = true;
108*3ac0a46fSAndroid Build Coastguard Worker   if (pCurFont && iCurCount) {
109*3ac0a46fSAndroid Build Coastguard Worker     pFxFont = pCurFont->GetDevFont();
110*3ac0a46fSAndroid Build Coastguard Worker     CFX_Font* font;
111*3ac0a46fSAndroid Build Coastguard Worker #if !BUILDFLAG(IS_WIN)
112*3ac0a46fSAndroid Build Coastguard Worker     FxFont.SetFace(pFxFont->GetFace());
113*3ac0a46fSAndroid Build Coastguard Worker     FxFont.SetFontSpan(pFxFont->GetFontSpan());
114*3ac0a46fSAndroid Build Coastguard Worker     font = &FxFont;
115*3ac0a46fSAndroid Build Coastguard Worker #else
116*3ac0a46fSAndroid Build Coastguard Worker     font = pFxFont;
117*3ac0a46fSAndroid Build Coastguard Worker #endif
118*3ac0a46fSAndroid Build Coastguard Worker 
119*3ac0a46fSAndroid Build Coastguard Worker     bRet = device->DrawNormalText(pdfium::make_span(pCurCP, iCurCount), font,
120*3ac0a46fSAndroid Build Coastguard Worker                                   -fFontSize, matrix, color, kOptions);
121*3ac0a46fSAndroid Build Coastguard Worker   }
122*3ac0a46fSAndroid Build Coastguard Worker 
123*3ac0a46fSAndroid Build Coastguard Worker   return bRet;
124*3ac0a46fSAndroid Build Coastguard Worker }
125*3ac0a46fSAndroid Build Coastguard Worker 
126*3ac0a46fSAndroid Build Coastguard Worker CFDE_TextOut::Piece::Piece() = default;
127*3ac0a46fSAndroid Build Coastguard Worker 
128*3ac0a46fSAndroid Build Coastguard Worker CFDE_TextOut::Piece::Piece(const Piece& that) = default;
129*3ac0a46fSAndroid Build Coastguard Worker 
130*3ac0a46fSAndroid Build Coastguard Worker CFDE_TextOut::Piece::~Piece() = default;
131*3ac0a46fSAndroid Build Coastguard Worker 
CFDE_TextOut()132*3ac0a46fSAndroid Build Coastguard Worker CFDE_TextOut::CFDE_TextOut()
133*3ac0a46fSAndroid Build Coastguard Worker     : m_pTxtBreak(std::make_unique<CFGAS_TxtBreak>()) {}
134*3ac0a46fSAndroid Build Coastguard Worker 
135*3ac0a46fSAndroid Build Coastguard Worker CFDE_TextOut::~CFDE_TextOut() = default;
136*3ac0a46fSAndroid Build Coastguard Worker 
SetFont(RetainPtr<CFGAS_GEFont> pFont)137*3ac0a46fSAndroid Build Coastguard Worker void CFDE_TextOut::SetFont(RetainPtr<CFGAS_GEFont> pFont) {
138*3ac0a46fSAndroid Build Coastguard Worker   DCHECK(pFont);
139*3ac0a46fSAndroid Build Coastguard Worker   m_pFont = std::move(pFont);
140*3ac0a46fSAndroid Build Coastguard Worker   m_pTxtBreak->SetFont(m_pFont);
141*3ac0a46fSAndroid Build Coastguard Worker }
142*3ac0a46fSAndroid Build Coastguard Worker 
SetFontSize(float fFontSize)143*3ac0a46fSAndroid Build Coastguard Worker void CFDE_TextOut::SetFontSize(float fFontSize) {
144*3ac0a46fSAndroid Build Coastguard Worker   DCHECK(fFontSize > 0);
145*3ac0a46fSAndroid Build Coastguard Worker   m_fFontSize = fFontSize;
146*3ac0a46fSAndroid Build Coastguard Worker   m_pTxtBreak->SetFontSize(fFontSize);
147*3ac0a46fSAndroid Build Coastguard Worker }
148*3ac0a46fSAndroid Build Coastguard Worker 
SetStyles(const FDE_TextStyle & dwStyles)149*3ac0a46fSAndroid Build Coastguard Worker void CFDE_TextOut::SetStyles(const FDE_TextStyle& dwStyles) {
150*3ac0a46fSAndroid Build Coastguard Worker   m_Styles = dwStyles;
151*3ac0a46fSAndroid Build Coastguard Worker   m_dwTxtBkStyles = m_Styles.single_line_
152*3ac0a46fSAndroid Build Coastguard Worker                         ? CFGAS_Break::LayoutStyle::kSingleLine
153*3ac0a46fSAndroid Build Coastguard Worker                         : CFGAS_Break::LayoutStyle::kNone;
154*3ac0a46fSAndroid Build Coastguard Worker 
155*3ac0a46fSAndroid Build Coastguard Worker   m_pTxtBreak->SetLayoutStyles(m_dwTxtBkStyles);
156*3ac0a46fSAndroid Build Coastguard Worker }
157*3ac0a46fSAndroid Build Coastguard Worker 
SetAlignment(FDE_TextAlignment iAlignment)158*3ac0a46fSAndroid Build Coastguard Worker void CFDE_TextOut::SetAlignment(FDE_TextAlignment iAlignment) {
159*3ac0a46fSAndroid Build Coastguard Worker   m_iAlignment = iAlignment;
160*3ac0a46fSAndroid Build Coastguard Worker 
161*3ac0a46fSAndroid Build Coastguard Worker   int32_t txtBreakAlignment = 0;
162*3ac0a46fSAndroid Build Coastguard Worker   switch (m_iAlignment) {
163*3ac0a46fSAndroid Build Coastguard Worker     case FDE_TextAlignment::kCenter:
164*3ac0a46fSAndroid Build Coastguard Worker       txtBreakAlignment = CFX_TxtLineAlignment_Center;
165*3ac0a46fSAndroid Build Coastguard Worker       break;
166*3ac0a46fSAndroid Build Coastguard Worker     case FDE_TextAlignment::kCenterRight:
167*3ac0a46fSAndroid Build Coastguard Worker       txtBreakAlignment = CFX_TxtLineAlignment_Right;
168*3ac0a46fSAndroid Build Coastguard Worker       break;
169*3ac0a46fSAndroid Build Coastguard Worker     case FDE_TextAlignment::kCenterLeft:
170*3ac0a46fSAndroid Build Coastguard Worker     case FDE_TextAlignment::kTopLeft:
171*3ac0a46fSAndroid Build Coastguard Worker       txtBreakAlignment = CFX_TxtLineAlignment_Left;
172*3ac0a46fSAndroid Build Coastguard Worker       break;
173*3ac0a46fSAndroid Build Coastguard Worker   }
174*3ac0a46fSAndroid Build Coastguard Worker   m_pTxtBreak->SetAlignment(txtBreakAlignment);
175*3ac0a46fSAndroid Build Coastguard Worker }
176*3ac0a46fSAndroid Build Coastguard Worker 
SetLineSpace(float fLineSpace)177*3ac0a46fSAndroid Build Coastguard Worker void CFDE_TextOut::SetLineSpace(float fLineSpace) {
178*3ac0a46fSAndroid Build Coastguard Worker   DCHECK(fLineSpace > 1.0f);
179*3ac0a46fSAndroid Build Coastguard Worker   m_fLineSpace = fLineSpace;
180*3ac0a46fSAndroid Build Coastguard Worker }
181*3ac0a46fSAndroid Build Coastguard Worker 
SetLineBreakTolerance(float fTolerance)182*3ac0a46fSAndroid Build Coastguard Worker void CFDE_TextOut::SetLineBreakTolerance(float fTolerance) {
183*3ac0a46fSAndroid Build Coastguard Worker   m_fTolerance = fTolerance;
184*3ac0a46fSAndroid Build Coastguard Worker   m_pTxtBreak->SetLineBreakTolerance(m_fTolerance);
185*3ac0a46fSAndroid Build Coastguard Worker }
186*3ac0a46fSAndroid Build Coastguard Worker 
CalcLogicSize(WideStringView str,CFX_SizeF * pSize)187*3ac0a46fSAndroid Build Coastguard Worker void CFDE_TextOut::CalcLogicSize(WideStringView str, CFX_SizeF* pSize) {
188*3ac0a46fSAndroid Build Coastguard Worker   CFX_RectF rtText(0.0f, 0.0f, pSize->width, pSize->height);
189*3ac0a46fSAndroid Build Coastguard Worker   CalcLogicSize(str, &rtText);
190*3ac0a46fSAndroid Build Coastguard Worker   *pSize = rtText.Size();
191*3ac0a46fSAndroid Build Coastguard Worker }
192*3ac0a46fSAndroid Build Coastguard Worker 
CalcLogicSize(WideStringView str,CFX_RectF * pRect)193*3ac0a46fSAndroid Build Coastguard Worker void CFDE_TextOut::CalcLogicSize(WideStringView str, CFX_RectF* pRect) {
194*3ac0a46fSAndroid Build Coastguard Worker   if (str.IsEmpty()) {
195*3ac0a46fSAndroid Build Coastguard Worker     pRect->width = 0.0f;
196*3ac0a46fSAndroid Build Coastguard Worker     pRect->height = 0.0f;
197*3ac0a46fSAndroid Build Coastguard Worker     return;
198*3ac0a46fSAndroid Build Coastguard Worker   }
199*3ac0a46fSAndroid Build Coastguard Worker 
200*3ac0a46fSAndroid Build Coastguard Worker   DCHECK(m_pFont);
201*3ac0a46fSAndroid Build Coastguard Worker   DCHECK(m_fFontSize >= 1.0f);
202*3ac0a46fSAndroid Build Coastguard Worker 
203*3ac0a46fSAndroid Build Coastguard Worker   if (!m_Styles.single_line_) {
204*3ac0a46fSAndroid Build Coastguard Worker     if (pRect->Width() < 1.0f)
205*3ac0a46fSAndroid Build Coastguard Worker       pRect->width = m_fFontSize * 1000.0f;
206*3ac0a46fSAndroid Build Coastguard Worker 
207*3ac0a46fSAndroid Build Coastguard Worker     m_pTxtBreak->SetLineWidth(pRect->Width());
208*3ac0a46fSAndroid Build Coastguard Worker   }
209*3ac0a46fSAndroid Build Coastguard Worker 
210*3ac0a46fSAndroid Build Coastguard Worker   m_iTotalLines = 0;
211*3ac0a46fSAndroid Build Coastguard Worker   float fWidth = 0.0f;
212*3ac0a46fSAndroid Build Coastguard Worker   float fHeight = 0.0f;
213*3ac0a46fSAndroid Build Coastguard Worker   float fStartPos = pRect->right();
214*3ac0a46fSAndroid Build Coastguard Worker   CFGAS_Char::BreakType dwBreakStatus = CFGAS_Char::BreakType::kNone;
215*3ac0a46fSAndroid Build Coastguard Worker   bool break_char_is_set = false;
216*3ac0a46fSAndroid Build Coastguard Worker   for (const wchar_t& wch : str) {
217*3ac0a46fSAndroid Build Coastguard Worker     if (!break_char_is_set && (wch == L'\n' || wch == L'\r')) {
218*3ac0a46fSAndroid Build Coastguard Worker       break_char_is_set = true;
219*3ac0a46fSAndroid Build Coastguard Worker       m_pTxtBreak->SetParagraphBreakChar(wch);
220*3ac0a46fSAndroid Build Coastguard Worker     }
221*3ac0a46fSAndroid Build Coastguard Worker     dwBreakStatus = m_pTxtBreak->AppendChar(wch);
222*3ac0a46fSAndroid Build Coastguard Worker     if (!CFX_BreakTypeNoneOrPiece(dwBreakStatus))
223*3ac0a46fSAndroid Build Coastguard Worker       RetrieveLineWidth(dwBreakStatus, &fStartPos, &fWidth, &fHeight);
224*3ac0a46fSAndroid Build Coastguard Worker   }
225*3ac0a46fSAndroid Build Coastguard Worker 
226*3ac0a46fSAndroid Build Coastguard Worker   dwBreakStatus = m_pTxtBreak->EndBreak(CFGAS_Char::BreakType::kParagraph);
227*3ac0a46fSAndroid Build Coastguard Worker   if (!CFX_BreakTypeNoneOrPiece(dwBreakStatus))
228*3ac0a46fSAndroid Build Coastguard Worker     RetrieveLineWidth(dwBreakStatus, &fStartPos, &fWidth, &fHeight);
229*3ac0a46fSAndroid Build Coastguard Worker 
230*3ac0a46fSAndroid Build Coastguard Worker   m_pTxtBreak->Reset();
231*3ac0a46fSAndroid Build Coastguard Worker   float fInc = pRect->Height() - fHeight;
232*3ac0a46fSAndroid Build Coastguard Worker   if (TextAlignmentVerticallyCentered(m_iAlignment))
233*3ac0a46fSAndroid Build Coastguard Worker     fInc /= 2.0f;
234*3ac0a46fSAndroid Build Coastguard Worker   else if (IsTextAlignmentTop(m_iAlignment))
235*3ac0a46fSAndroid Build Coastguard Worker     fInc = 0.0f;
236*3ac0a46fSAndroid Build Coastguard Worker 
237*3ac0a46fSAndroid Build Coastguard Worker   pRect->left += fStartPos;
238*3ac0a46fSAndroid Build Coastguard Worker   pRect->top += fInc;
239*3ac0a46fSAndroid Build Coastguard Worker   pRect->width = std::min(fWidth, pRect->Width());
240*3ac0a46fSAndroid Build Coastguard Worker   pRect->height = fHeight;
241*3ac0a46fSAndroid Build Coastguard Worker   if (m_Styles.last_line_height_)
242*3ac0a46fSAndroid Build Coastguard Worker     pRect->height -= m_fLineSpace - m_fFontSize;
243*3ac0a46fSAndroid Build Coastguard Worker }
244*3ac0a46fSAndroid Build Coastguard Worker 
RetrieveLineWidth(CFGAS_Char::BreakType dwBreakStatus,float * pStartPos,float * pWidth,float * pHeight)245*3ac0a46fSAndroid Build Coastguard Worker bool CFDE_TextOut::RetrieveLineWidth(CFGAS_Char::BreakType dwBreakStatus,
246*3ac0a46fSAndroid Build Coastguard Worker                                      float* pStartPos,
247*3ac0a46fSAndroid Build Coastguard Worker                                      float* pWidth,
248*3ac0a46fSAndroid Build Coastguard Worker                                      float* pHeight) {
249*3ac0a46fSAndroid Build Coastguard Worker   if (CFX_BreakTypeNoneOrPiece(dwBreakStatus))
250*3ac0a46fSAndroid Build Coastguard Worker     return false;
251*3ac0a46fSAndroid Build Coastguard Worker 
252*3ac0a46fSAndroid Build Coastguard Worker   float fLineStep = std::max(m_fLineSpace, m_fFontSize);
253*3ac0a46fSAndroid Build Coastguard Worker   float fLineWidth = 0.0f;
254*3ac0a46fSAndroid Build Coastguard Worker   for (int32_t i = 0; i < m_pTxtBreak->CountBreakPieces(); i++) {
255*3ac0a46fSAndroid Build Coastguard Worker     const CFGAS_BreakPiece* pPiece = m_pTxtBreak->GetBreakPieceUnstable(i);
256*3ac0a46fSAndroid Build Coastguard Worker     fLineWidth += static_cast<float>(pPiece->GetWidth()) / 20000.0f;
257*3ac0a46fSAndroid Build Coastguard Worker     *pStartPos = std::min(*pStartPos,
258*3ac0a46fSAndroid Build Coastguard Worker                           static_cast<float>(pPiece->GetStartPos()) / 20000.0f);
259*3ac0a46fSAndroid Build Coastguard Worker   }
260*3ac0a46fSAndroid Build Coastguard Worker   m_pTxtBreak->ClearBreakPieces();
261*3ac0a46fSAndroid Build Coastguard Worker 
262*3ac0a46fSAndroid Build Coastguard Worker   if (dwBreakStatus == CFGAS_Char::BreakType::kParagraph)
263*3ac0a46fSAndroid Build Coastguard Worker     m_pTxtBreak->Reset();
264*3ac0a46fSAndroid Build Coastguard Worker   if (!m_Styles.line_wrap_ && dwBreakStatus == CFGAS_Char::BreakType::kLine) {
265*3ac0a46fSAndroid Build Coastguard Worker     *pWidth += fLineWidth;
266*3ac0a46fSAndroid Build Coastguard Worker   } else {
267*3ac0a46fSAndroid Build Coastguard Worker     *pWidth = std::max(*pWidth, fLineWidth);
268*3ac0a46fSAndroid Build Coastguard Worker     *pHeight += fLineStep;
269*3ac0a46fSAndroid Build Coastguard Worker   }
270*3ac0a46fSAndroid Build Coastguard Worker   ++m_iTotalLines;
271*3ac0a46fSAndroid Build Coastguard Worker   return true;
272*3ac0a46fSAndroid Build Coastguard Worker }
273*3ac0a46fSAndroid Build Coastguard Worker 
DrawLogicText(CFX_RenderDevice * device,const WideString & str,const CFX_RectF & rect)274*3ac0a46fSAndroid Build Coastguard Worker void CFDE_TextOut::DrawLogicText(CFX_RenderDevice* device,
275*3ac0a46fSAndroid Build Coastguard Worker                                  const WideString& str,
276*3ac0a46fSAndroid Build Coastguard Worker                                  const CFX_RectF& rect) {
277*3ac0a46fSAndroid Build Coastguard Worker   DCHECK(m_pFont);
278*3ac0a46fSAndroid Build Coastguard Worker   DCHECK(m_fFontSize >= 1.0f);
279*3ac0a46fSAndroid Build Coastguard Worker 
280*3ac0a46fSAndroid Build Coastguard Worker   if (str.IsEmpty())
281*3ac0a46fSAndroid Build Coastguard Worker     return;
282*3ac0a46fSAndroid Build Coastguard Worker   if (rect.width < m_fFontSize || rect.height < m_fFontSize)
283*3ac0a46fSAndroid Build Coastguard Worker     return;
284*3ac0a46fSAndroid Build Coastguard Worker 
285*3ac0a46fSAndroid Build Coastguard Worker   float fLineWidth = rect.width;
286*3ac0a46fSAndroid Build Coastguard Worker   m_pTxtBreak->SetLineWidth(fLineWidth);
287*3ac0a46fSAndroid Build Coastguard Worker   m_ttoLines.clear();
288*3ac0a46fSAndroid Build Coastguard Worker   m_wsText.clear();
289*3ac0a46fSAndroid Build Coastguard Worker 
290*3ac0a46fSAndroid Build Coastguard Worker   LoadText(str, rect);
291*3ac0a46fSAndroid Build Coastguard Worker   Reload(rect);
292*3ac0a46fSAndroid Build Coastguard Worker   DoAlignment(rect);
293*3ac0a46fSAndroid Build Coastguard Worker 
294*3ac0a46fSAndroid Build Coastguard Worker   if (!device || m_ttoLines.empty())
295*3ac0a46fSAndroid Build Coastguard Worker     return;
296*3ac0a46fSAndroid Build Coastguard Worker 
297*3ac0a46fSAndroid Build Coastguard Worker   CFX_RectF rtClip = m_Matrix.TransformRect(CFX_RectF());
298*3ac0a46fSAndroid Build Coastguard Worker   device->SaveState();
299*3ac0a46fSAndroid Build Coastguard Worker   if (rtClip.Width() > 0.0f && rtClip.Height() > 0.0f)
300*3ac0a46fSAndroid Build Coastguard Worker     device->SetClip_Rect(rtClip.GetOuterRect());
301*3ac0a46fSAndroid Build Coastguard Worker 
302*3ac0a46fSAndroid Build Coastguard Worker   for (auto& line : m_ttoLines) {
303*3ac0a46fSAndroid Build Coastguard Worker     for (size_t i = 0; i < line.GetSize(); ++i) {
304*3ac0a46fSAndroid Build Coastguard Worker       const Piece* pPiece = line.GetPieceAtIndex(i);
305*3ac0a46fSAndroid Build Coastguard Worker       size_t szCount = GetDisplayPos(pPiece);
306*3ac0a46fSAndroid Build Coastguard Worker       if (szCount == 0)
307*3ac0a46fSAndroid Build Coastguard Worker         continue;
308*3ac0a46fSAndroid Build Coastguard Worker 
309*3ac0a46fSAndroid Build Coastguard Worker       CFDE_TextOut::DrawString(device, m_TxtColor, m_pFont,
310*3ac0a46fSAndroid Build Coastguard Worker                                {m_CharPos.data(), szCount}, m_fFontSize,
311*3ac0a46fSAndroid Build Coastguard Worker                                m_Matrix);
312*3ac0a46fSAndroid Build Coastguard Worker     }
313*3ac0a46fSAndroid Build Coastguard Worker   }
314*3ac0a46fSAndroid Build Coastguard Worker   device->RestoreState(false);
315*3ac0a46fSAndroid Build Coastguard Worker }
316*3ac0a46fSAndroid Build Coastguard Worker 
LoadText(const WideString & str,const CFX_RectF & rect)317*3ac0a46fSAndroid Build Coastguard Worker void CFDE_TextOut::LoadText(const WideString& str, const CFX_RectF& rect) {
318*3ac0a46fSAndroid Build Coastguard Worker   DCHECK(!str.IsEmpty());
319*3ac0a46fSAndroid Build Coastguard Worker 
320*3ac0a46fSAndroid Build Coastguard Worker   m_wsText = str;
321*3ac0a46fSAndroid Build Coastguard Worker 
322*3ac0a46fSAndroid Build Coastguard Worker   if (m_CharWidths.size() < str.GetLength())
323*3ac0a46fSAndroid Build Coastguard Worker     m_CharWidths.resize(str.GetLength(), 0);
324*3ac0a46fSAndroid Build Coastguard Worker 
325*3ac0a46fSAndroid Build Coastguard Worker   float fLineStep = std::max(m_fLineSpace, m_fFontSize);
326*3ac0a46fSAndroid Build Coastguard Worker   float fLineStop = rect.bottom();
327*3ac0a46fSAndroid Build Coastguard Worker   m_fLinePos = rect.top;
328*3ac0a46fSAndroid Build Coastguard Worker   size_t start_char = 0;
329*3ac0a46fSAndroid Build Coastguard Worker   int32_t iPieceWidths = 0;
330*3ac0a46fSAndroid Build Coastguard Worker   CFGAS_Char::BreakType dwBreakStatus;
331*3ac0a46fSAndroid Build Coastguard Worker   bool bRet = false;
332*3ac0a46fSAndroid Build Coastguard Worker   for (const auto& wch : str) {
333*3ac0a46fSAndroid Build Coastguard Worker     dwBreakStatus = m_pTxtBreak->AppendChar(wch);
334*3ac0a46fSAndroid Build Coastguard Worker     if (CFX_BreakTypeNoneOrPiece(dwBreakStatus))
335*3ac0a46fSAndroid Build Coastguard Worker       continue;
336*3ac0a46fSAndroid Build Coastguard Worker 
337*3ac0a46fSAndroid Build Coastguard Worker     bool bEndofLine =
338*3ac0a46fSAndroid Build Coastguard Worker         RetrievePieces(dwBreakStatus, false, rect, &start_char, &iPieceWidths);
339*3ac0a46fSAndroid Build Coastguard Worker     if (bEndofLine && (m_Styles.line_wrap_ ||
340*3ac0a46fSAndroid Build Coastguard Worker                        dwBreakStatus == CFGAS_Char::BreakType::kParagraph ||
341*3ac0a46fSAndroid Build Coastguard Worker                        dwBreakStatus == CFGAS_Char::BreakType::kPage)) {
342*3ac0a46fSAndroid Build Coastguard Worker       iPieceWidths = 0;
343*3ac0a46fSAndroid Build Coastguard Worker       ++m_iCurLine;
344*3ac0a46fSAndroid Build Coastguard Worker       m_fLinePos += fLineStep;
345*3ac0a46fSAndroid Build Coastguard Worker     }
346*3ac0a46fSAndroid Build Coastguard Worker     if (m_fLinePos + fLineStep > fLineStop) {
347*3ac0a46fSAndroid Build Coastguard Worker       size_t iCurLine = bEndofLine ? m_iCurLine - 1 : m_iCurLine;
348*3ac0a46fSAndroid Build Coastguard Worker       CHECK_LT(m_iCurLine, m_ttoLines.size());
349*3ac0a46fSAndroid Build Coastguard Worker       m_ttoLines[iCurLine].set_new_reload(true);
350*3ac0a46fSAndroid Build Coastguard Worker       bRet = true;
351*3ac0a46fSAndroid Build Coastguard Worker       break;
352*3ac0a46fSAndroid Build Coastguard Worker     }
353*3ac0a46fSAndroid Build Coastguard Worker   }
354*3ac0a46fSAndroid Build Coastguard Worker 
355*3ac0a46fSAndroid Build Coastguard Worker   dwBreakStatus = m_pTxtBreak->EndBreak(CFGAS_Char::BreakType::kParagraph);
356*3ac0a46fSAndroid Build Coastguard Worker   if (!CFX_BreakTypeNoneOrPiece(dwBreakStatus) && !bRet)
357*3ac0a46fSAndroid Build Coastguard Worker     RetrievePieces(dwBreakStatus, false, rect, &start_char, &iPieceWidths);
358*3ac0a46fSAndroid Build Coastguard Worker 
359*3ac0a46fSAndroid Build Coastguard Worker   m_pTxtBreak->ClearBreakPieces();
360*3ac0a46fSAndroid Build Coastguard Worker   m_pTxtBreak->Reset();
361*3ac0a46fSAndroid Build Coastguard Worker }
362*3ac0a46fSAndroid Build Coastguard Worker 
RetrievePieces(CFGAS_Char::BreakType dwBreakStatus,bool bReload,const CFX_RectF & rect,size_t * pStartChar,int32_t * pPieceWidths)363*3ac0a46fSAndroid Build Coastguard Worker bool CFDE_TextOut::RetrievePieces(CFGAS_Char::BreakType dwBreakStatus,
364*3ac0a46fSAndroid Build Coastguard Worker                                   bool bReload,
365*3ac0a46fSAndroid Build Coastguard Worker                                   const CFX_RectF& rect,
366*3ac0a46fSAndroid Build Coastguard Worker                                   size_t* pStartChar,
367*3ac0a46fSAndroid Build Coastguard Worker                                   int32_t* pPieceWidths) {
368*3ac0a46fSAndroid Build Coastguard Worker   float fLineStep = std::max(m_fLineSpace, m_fFontSize);
369*3ac0a46fSAndroid Build Coastguard Worker   bool bNeedReload = false;
370*3ac0a46fSAndroid Build Coastguard Worker   int32_t iLineWidth = FXSYS_roundf(rect.Width() * 20000.0f);
371*3ac0a46fSAndroid Build Coastguard Worker   int32_t iCount = m_pTxtBreak->CountBreakPieces();
372*3ac0a46fSAndroid Build Coastguard Worker 
373*3ac0a46fSAndroid Build Coastguard Worker   size_t chars_to_skip = *pStartChar;
374*3ac0a46fSAndroid Build Coastguard Worker   for (int32_t i = 0; i < iCount; i++) {
375*3ac0a46fSAndroid Build Coastguard Worker     const CFGAS_BreakPiece* pPiece = m_pTxtBreak->GetBreakPieceUnstable(i);
376*3ac0a46fSAndroid Build Coastguard Worker     size_t iPieceChars = pPiece->GetLength();
377*3ac0a46fSAndroid Build Coastguard Worker     if (chars_to_skip > iPieceChars) {
378*3ac0a46fSAndroid Build Coastguard Worker       chars_to_skip -= iPieceChars;
379*3ac0a46fSAndroid Build Coastguard Worker       continue;
380*3ac0a46fSAndroid Build Coastguard Worker     }
381*3ac0a46fSAndroid Build Coastguard Worker 
382*3ac0a46fSAndroid Build Coastguard Worker     size_t iChar = *pStartChar;
383*3ac0a46fSAndroid Build Coastguard Worker     int32_t iWidth = 0;
384*3ac0a46fSAndroid Build Coastguard Worker     size_t j = chars_to_skip;
385*3ac0a46fSAndroid Build Coastguard Worker     for (; j < iPieceChars; j++) {
386*3ac0a46fSAndroid Build Coastguard Worker       const CFGAS_Char* pTC = pPiece->GetChar(j);
387*3ac0a46fSAndroid Build Coastguard Worker       int32_t iCurCharWidth = std::max(pTC->m_iCharWidth, 0);
388*3ac0a46fSAndroid Build Coastguard Worker       if (m_Styles.single_line_ || !m_Styles.line_wrap_) {
389*3ac0a46fSAndroid Build Coastguard Worker         if (iLineWidth - *pPieceWidths - iWidth < iCurCharWidth) {
390*3ac0a46fSAndroid Build Coastguard Worker           bNeedReload = true;
391*3ac0a46fSAndroid Build Coastguard Worker           break;
392*3ac0a46fSAndroid Build Coastguard Worker         }
393*3ac0a46fSAndroid Build Coastguard Worker       }
394*3ac0a46fSAndroid Build Coastguard Worker       iWidth += iCurCharWidth;
395*3ac0a46fSAndroid Build Coastguard Worker       m_CharWidths[iChar++] = iCurCharWidth;
396*3ac0a46fSAndroid Build Coastguard Worker     }
397*3ac0a46fSAndroid Build Coastguard Worker 
398*3ac0a46fSAndroid Build Coastguard Worker     if (j == chars_to_skip && !bReload) {
399*3ac0a46fSAndroid Build Coastguard Worker       CHECK_LT(m_iCurLine, m_ttoLines.size());
400*3ac0a46fSAndroid Build Coastguard Worker       m_ttoLines[m_iCurLine].set_new_reload(true);
401*3ac0a46fSAndroid Build Coastguard Worker     } else if (j > chars_to_skip) {
402*3ac0a46fSAndroid Build Coastguard Worker       Piece piece;
403*3ac0a46fSAndroid Build Coastguard Worker       piece.start_char = *pStartChar;
404*3ac0a46fSAndroid Build Coastguard Worker       piece.char_count = j - chars_to_skip;
405*3ac0a46fSAndroid Build Coastguard Worker       piece.char_styles = pPiece->GetCharStyles();
406*3ac0a46fSAndroid Build Coastguard Worker       piece.bounds = CFX_RectF(
407*3ac0a46fSAndroid Build Coastguard Worker           rect.left + static_cast<float>(pPiece->GetStartPos()) / 20000.0f,
408*3ac0a46fSAndroid Build Coastguard Worker           m_fLinePos, iWidth / 20000.0f, fLineStep);
409*3ac0a46fSAndroid Build Coastguard Worker 
410*3ac0a46fSAndroid Build Coastguard Worker       if (FX_IsOdd(pPiece->GetBidiLevel()))
411*3ac0a46fSAndroid Build Coastguard Worker         piece.char_styles |= FX_TXTCHARSTYLE_OddBidiLevel;
412*3ac0a46fSAndroid Build Coastguard Worker 
413*3ac0a46fSAndroid Build Coastguard Worker       AppendPiece(piece, bNeedReload, (bReload && i == iCount - 1));
414*3ac0a46fSAndroid Build Coastguard Worker     }
415*3ac0a46fSAndroid Build Coastguard Worker     *pStartChar += iPieceChars;
416*3ac0a46fSAndroid Build Coastguard Worker     *pPieceWidths += iWidth;
417*3ac0a46fSAndroid Build Coastguard Worker   }
418*3ac0a46fSAndroid Build Coastguard Worker   m_pTxtBreak->ClearBreakPieces();
419*3ac0a46fSAndroid Build Coastguard Worker 
420*3ac0a46fSAndroid Build Coastguard Worker   return m_Styles.single_line_ || m_Styles.line_wrap_ || bNeedReload ||
421*3ac0a46fSAndroid Build Coastguard Worker          dwBreakStatus == CFGAS_Char::BreakType::kParagraph;
422*3ac0a46fSAndroid Build Coastguard Worker }
423*3ac0a46fSAndroid Build Coastguard Worker 
AppendPiece(const Piece & piece,bool bNeedReload,bool bEnd)424*3ac0a46fSAndroid Build Coastguard Worker void CFDE_TextOut::AppendPiece(const Piece& piece,
425*3ac0a46fSAndroid Build Coastguard Worker                                bool bNeedReload,
426*3ac0a46fSAndroid Build Coastguard Worker                                bool bEnd) {
427*3ac0a46fSAndroid Build Coastguard Worker   if (m_iCurLine >= m_ttoLines.size()) {
428*3ac0a46fSAndroid Build Coastguard Worker     Line ttoLine;
429*3ac0a46fSAndroid Build Coastguard Worker     ttoLine.set_new_reload(bNeedReload);
430*3ac0a46fSAndroid Build Coastguard Worker 
431*3ac0a46fSAndroid Build Coastguard Worker     m_iCurPiece = ttoLine.AddPiece(m_iCurPiece, piece);
432*3ac0a46fSAndroid Build Coastguard Worker     m_ttoLines.push_back(ttoLine);
433*3ac0a46fSAndroid Build Coastguard Worker     m_iCurLine = m_ttoLines.size() - 1;
434*3ac0a46fSAndroid Build Coastguard Worker   } else {
435*3ac0a46fSAndroid Build Coastguard Worker     Line* pLine = &m_ttoLines[m_iCurLine];
436*3ac0a46fSAndroid Build Coastguard Worker     pLine->set_new_reload(bNeedReload);
437*3ac0a46fSAndroid Build Coastguard Worker 
438*3ac0a46fSAndroid Build Coastguard Worker     m_iCurPiece = pLine->AddPiece(m_iCurPiece, piece);
439*3ac0a46fSAndroid Build Coastguard Worker     if (bEnd) {
440*3ac0a46fSAndroid Build Coastguard Worker       size_t iPieces = pLine->GetSize();
441*3ac0a46fSAndroid Build Coastguard Worker       if (m_iCurPiece < iPieces)
442*3ac0a46fSAndroid Build Coastguard Worker         pLine->RemoveLast(iPieces - m_iCurPiece - 1);
443*3ac0a46fSAndroid Build Coastguard Worker     }
444*3ac0a46fSAndroid Build Coastguard Worker   }
445*3ac0a46fSAndroid Build Coastguard Worker   if (!bEnd && bNeedReload)
446*3ac0a46fSAndroid Build Coastguard Worker     m_iCurPiece = 0;
447*3ac0a46fSAndroid Build Coastguard Worker }
448*3ac0a46fSAndroid Build Coastguard Worker 
Reload(const CFX_RectF & rect)449*3ac0a46fSAndroid Build Coastguard Worker void CFDE_TextOut::Reload(const CFX_RectF& rect) {
450*3ac0a46fSAndroid Build Coastguard Worker   size_t i = 0;
451*3ac0a46fSAndroid Build Coastguard Worker   for (auto& line : m_ttoLines) {
452*3ac0a46fSAndroid Build Coastguard Worker     if (line.new_reload()) {
453*3ac0a46fSAndroid Build Coastguard Worker       m_iCurLine = i;
454*3ac0a46fSAndroid Build Coastguard Worker       m_iCurPiece = 0;
455*3ac0a46fSAndroid Build Coastguard Worker       ReloadLinePiece(&line, rect);
456*3ac0a46fSAndroid Build Coastguard Worker     }
457*3ac0a46fSAndroid Build Coastguard Worker     ++i;
458*3ac0a46fSAndroid Build Coastguard Worker   }
459*3ac0a46fSAndroid Build Coastguard Worker }
460*3ac0a46fSAndroid Build Coastguard Worker 
ReloadLinePiece(Line * line,const CFX_RectF & rect)461*3ac0a46fSAndroid Build Coastguard Worker void CFDE_TextOut::ReloadLinePiece(Line* line, const CFX_RectF& rect) {
462*3ac0a46fSAndroid Build Coastguard Worker   pdfium::span<const wchar_t> text_span = m_wsText.span();
463*3ac0a46fSAndroid Build Coastguard Worker   size_t start_char = 0;
464*3ac0a46fSAndroid Build Coastguard Worker   size_t piece_count = line->GetSize();
465*3ac0a46fSAndroid Build Coastguard Worker   int32_t piece_widths = 0;
466*3ac0a46fSAndroid Build Coastguard Worker   CFGAS_Char::BreakType break_status = CFGAS_Char::BreakType::kNone;
467*3ac0a46fSAndroid Build Coastguard Worker   for (size_t piece_index = 0; piece_index < piece_count; ++piece_index) {
468*3ac0a46fSAndroid Build Coastguard Worker     const Piece* piece = line->GetPieceAtIndex(piece_index);
469*3ac0a46fSAndroid Build Coastguard Worker     if (piece_index == 0)
470*3ac0a46fSAndroid Build Coastguard Worker       m_fLinePos = piece->bounds.top;
471*3ac0a46fSAndroid Build Coastguard Worker 
472*3ac0a46fSAndroid Build Coastguard Worker     start_char = piece->start_char;
473*3ac0a46fSAndroid Build Coastguard Worker     const size_t end = piece->start_char + piece->char_count;
474*3ac0a46fSAndroid Build Coastguard Worker     for (size_t char_index = start_char; char_index < end; ++char_index) {
475*3ac0a46fSAndroid Build Coastguard Worker       break_status = m_pTxtBreak->AppendChar(text_span[char_index]);
476*3ac0a46fSAndroid Build Coastguard Worker       if (!CFX_BreakTypeNoneOrPiece(break_status))
477*3ac0a46fSAndroid Build Coastguard Worker         RetrievePieces(break_status, true, rect, &start_char, &piece_widths);
478*3ac0a46fSAndroid Build Coastguard Worker     }
479*3ac0a46fSAndroid Build Coastguard Worker   }
480*3ac0a46fSAndroid Build Coastguard Worker 
481*3ac0a46fSAndroid Build Coastguard Worker   break_status = m_pTxtBreak->EndBreak(CFGAS_Char::BreakType::kParagraph);
482*3ac0a46fSAndroid Build Coastguard Worker   if (!CFX_BreakTypeNoneOrPiece(break_status))
483*3ac0a46fSAndroid Build Coastguard Worker     RetrievePieces(break_status, true, rect, &start_char, &piece_widths);
484*3ac0a46fSAndroid Build Coastguard Worker 
485*3ac0a46fSAndroid Build Coastguard Worker   m_pTxtBreak->Reset();
486*3ac0a46fSAndroid Build Coastguard Worker }
487*3ac0a46fSAndroid Build Coastguard Worker 
DoAlignment(const CFX_RectF & rect)488*3ac0a46fSAndroid Build Coastguard Worker void CFDE_TextOut::DoAlignment(const CFX_RectF& rect) {
489*3ac0a46fSAndroid Build Coastguard Worker   if (m_ttoLines.empty())
490*3ac0a46fSAndroid Build Coastguard Worker     return;
491*3ac0a46fSAndroid Build Coastguard Worker 
492*3ac0a46fSAndroid Build Coastguard Worker   const Piece* pFirstPiece = m_ttoLines.back().GetPieceAtIndex(0);
493*3ac0a46fSAndroid Build Coastguard Worker   if (!pFirstPiece)
494*3ac0a46fSAndroid Build Coastguard Worker     return;
495*3ac0a46fSAndroid Build Coastguard Worker 
496*3ac0a46fSAndroid Build Coastguard Worker   float fInc = rect.bottom() - pFirstPiece->bounds.bottom();
497*3ac0a46fSAndroid Build Coastguard Worker   if (TextAlignmentVerticallyCentered(m_iAlignment))
498*3ac0a46fSAndroid Build Coastguard Worker     fInc /= 2.0f;
499*3ac0a46fSAndroid Build Coastguard Worker   else if (IsTextAlignmentTop(m_iAlignment))
500*3ac0a46fSAndroid Build Coastguard Worker     fInc = 0.0f;
501*3ac0a46fSAndroid Build Coastguard Worker 
502*3ac0a46fSAndroid Build Coastguard Worker   if (fInc < 1.0f)
503*3ac0a46fSAndroid Build Coastguard Worker     return;
504*3ac0a46fSAndroid Build Coastguard Worker 
505*3ac0a46fSAndroid Build Coastguard Worker   for (auto& line : m_ttoLines) {
506*3ac0a46fSAndroid Build Coastguard Worker     for (size_t i = 0; i < line.GetSize(); ++i)
507*3ac0a46fSAndroid Build Coastguard Worker       line.GetPieceAtIndex(i)->bounds.top += fInc;
508*3ac0a46fSAndroid Build Coastguard Worker   }
509*3ac0a46fSAndroid Build Coastguard Worker }
510*3ac0a46fSAndroid Build Coastguard Worker 
GetDisplayPos(const Piece * pPiece)511*3ac0a46fSAndroid Build Coastguard Worker size_t CFDE_TextOut::GetDisplayPos(const Piece* pPiece) {
512*3ac0a46fSAndroid Build Coastguard Worker   if (m_CharPos.size() < pPiece->char_count)
513*3ac0a46fSAndroid Build Coastguard Worker     m_CharPos.resize(pPiece->char_count, TextCharPos());
514*3ac0a46fSAndroid Build Coastguard Worker 
515*3ac0a46fSAndroid Build Coastguard Worker   CFGAS_TxtBreak::Run tr;
516*3ac0a46fSAndroid Build Coastguard Worker   tr.wsStr = m_wsText.Substr(pPiece->start_char);
517*3ac0a46fSAndroid Build Coastguard Worker   tr.pWidths = &m_CharWidths[pPiece->start_char];
518*3ac0a46fSAndroid Build Coastguard Worker   tr.iLength = pdfium::base::checked_cast<int32_t>(pPiece->char_count);
519*3ac0a46fSAndroid Build Coastguard Worker   tr.pFont = m_pFont;
520*3ac0a46fSAndroid Build Coastguard Worker   tr.fFontSize = m_fFontSize;
521*3ac0a46fSAndroid Build Coastguard Worker   tr.dwStyles = m_dwTxtBkStyles;
522*3ac0a46fSAndroid Build Coastguard Worker   tr.dwCharStyles = pPiece->char_styles;
523*3ac0a46fSAndroid Build Coastguard Worker   tr.pRect = &pPiece->bounds;
524*3ac0a46fSAndroid Build Coastguard Worker 
525*3ac0a46fSAndroid Build Coastguard Worker   return m_pTxtBreak->GetDisplayPos(tr, m_CharPos.data());
526*3ac0a46fSAndroid Build Coastguard Worker }
527*3ac0a46fSAndroid Build Coastguard Worker 
528*3ac0a46fSAndroid Build Coastguard Worker CFDE_TextOut::Line::Line() = default;
529*3ac0a46fSAndroid Build Coastguard Worker 
Line(const Line & that)530*3ac0a46fSAndroid Build Coastguard Worker CFDE_TextOut::Line::Line(const Line& that)
531*3ac0a46fSAndroid Build Coastguard Worker     : new_reload_(that.new_reload_), pieces_(that.pieces_) {}
532*3ac0a46fSAndroid Build Coastguard Worker 
533*3ac0a46fSAndroid Build Coastguard Worker CFDE_TextOut::Line::~Line() = default;
534*3ac0a46fSAndroid Build Coastguard Worker 
AddPiece(size_t index,const Piece & piece)535*3ac0a46fSAndroid Build Coastguard Worker size_t CFDE_TextOut::Line::AddPiece(size_t index, const Piece& piece) {
536*3ac0a46fSAndroid Build Coastguard Worker   if (index >= pieces_.size()) {
537*3ac0a46fSAndroid Build Coastguard Worker     pieces_.push_back(piece);
538*3ac0a46fSAndroid Build Coastguard Worker     return pieces_.size();
539*3ac0a46fSAndroid Build Coastguard Worker   }
540*3ac0a46fSAndroid Build Coastguard Worker   pieces_[index] = piece;
541*3ac0a46fSAndroid Build Coastguard Worker   return index;
542*3ac0a46fSAndroid Build Coastguard Worker }
543*3ac0a46fSAndroid Build Coastguard Worker 
GetSize() const544*3ac0a46fSAndroid Build Coastguard Worker size_t CFDE_TextOut::Line::GetSize() const {
545*3ac0a46fSAndroid Build Coastguard Worker   return pieces_.size();
546*3ac0a46fSAndroid Build Coastguard Worker }
547*3ac0a46fSAndroid Build Coastguard Worker 
GetPieceAtIndex(size_t index) const548*3ac0a46fSAndroid Build Coastguard Worker const CFDE_TextOut::Piece* CFDE_TextOut::Line::GetPieceAtIndex(
549*3ac0a46fSAndroid Build Coastguard Worker     size_t index) const {
550*3ac0a46fSAndroid Build Coastguard Worker   CHECK(fxcrt::IndexInBounds(pieces_, index));
551*3ac0a46fSAndroid Build Coastguard Worker   return &pieces_[index];
552*3ac0a46fSAndroid Build Coastguard Worker }
553*3ac0a46fSAndroid Build Coastguard Worker 
GetPieceAtIndex(size_t index)554*3ac0a46fSAndroid Build Coastguard Worker CFDE_TextOut::Piece* CFDE_TextOut::Line::GetPieceAtIndex(size_t index) {
555*3ac0a46fSAndroid Build Coastguard Worker   CHECK(fxcrt::IndexInBounds(pieces_, index));
556*3ac0a46fSAndroid Build Coastguard Worker   return &pieces_[index];
557*3ac0a46fSAndroid Build Coastguard Worker }
558*3ac0a46fSAndroid Build Coastguard Worker 
RemoveLast(size_t count)559*3ac0a46fSAndroid Build Coastguard Worker void CFDE_TextOut::Line::RemoveLast(size_t count) {
560*3ac0a46fSAndroid Build Coastguard Worker   pieces_.erase(pieces_.end() - std::min(count, pieces_.size()), pieces_.end());
561*3ac0a46fSAndroid Build Coastguard Worker }
562