7#include "xfa/fde/cfde_textout.h"
12#include "build/build_config.h"
13#include "core/fxcrt/check.h"
14#include "core/fxcrt/check_op.h"
15#include "core/fxcrt/compiler_specific.h"
16#include "core/fxcrt/fx_coordinates.h"
17#include "core/fxcrt/fx_extension.h"
18#include "core/fxcrt/fx_system.h"
19#include "core/fxcrt/numerics/safe_conversions.h"
20#include "core/fxcrt/stl_util.h"
21#include "core/fxge/cfx_font.h"
22#include "core/fxge/cfx_path.h"
23#include "core/fxge/cfx_renderdevice.h"
24#include "core/fxge/cfx_substfont.h"
25#include "core/fxge/cfx_textrenderoptions.h"
26#include "core/fxge/fx_font.h"
27#include "core/fxge/text_char_pos.h"
28#include "xfa/fgas/font/cfgas_gefont.h"
29#include "xfa/fgas/layout/cfgas_txtbreak.h"
51 span<TextCharPos> pCharPos,
57 CFX_Font* pFxFont = pFont->GetDevFont();
58 if (FontStyleIsItalic(pFont->GetFontStyles()) && !pFxFont->IsItalic()) {
59 for (
auto& pos : pCharPos) {
60 static constexpr float mc = 0.267949f;
61 pos.m_AdjustMatrix[2] += mc * pos.m_AdjustMatrix[0];
62 pos.m_AdjustMatrix[3] += mc * pos.m_AdjustMatrix[1];
67 uint32_t dwFontStyle = pFont->GetFontStyles();
69 auto SubstFxFont = std::make_unique<CFX_SubstFont>();
70 SubstFxFont->m_Weight = FontStyleIsForceBold(dwFontStyle) ? 700 : 400;
71 SubstFxFont->m_ItalicAngle = FontStyleIsItalic(dwFontStyle) ? -12 : 0;
72 SubstFxFont->m_WeightCJK = SubstFxFont->m_Weight;
73 SubstFxFont->m_bItalicCJK = FontStyleIsItalic(dwFontStyle);
74 FxFont.SetSubstFont(std::move(SubstFxFont));
81 for (
auto& pos : pCharPos) {
82 RetainPtr<CFGAS_GEFont> pSTFont =
83 pFont->GetSubstFont(
static_cast<int32_t>(pos.m_GlyphIndex));
84 pos.m_GlyphIndex &= 0x00FFFFFF;
85 pos.m_bFontStyle =
false;
86 if (pCurFont != pSTFont) {
88 pFxFont = pCurFont->GetDevFont();
92 FxFont.SetFace(pFxFont->GetFace());
93 FxFont.SetFontSpan(pFxFont->GetFontSpan());
98 device->DrawNormalText(
UNSAFE_TODO(make_span(pCurCP, count)), font,
99 -fFontSize, matrix, color, kOptions);
108 if (pCurFont && count) {
109 pFxFont = pCurFont->GetDevFont();
111#if !BUILDFLAG(IS_WIN)
112 FxFont.SetFace(pFxFont->GetFace());
113 FxFont.SetFontSpan(pFxFont->GetFontSpan());
118 return device->DrawNormalText(
UNSAFE_TODO(make_span(pCurCP, count)), font,
119 -fFontSize, matrix, color, kOptions);
137 m_pFont = std::move(pFont);
138 m_pTxtBreak->SetFont(m_pFont);
143 m_fFontSize = fFontSize;
144 m_pTxtBreak->SetFontSize(fFontSize);
149 m_dwTxtBkStyles = m_Styles.single_line_
150 ? CFGAS_Break::LayoutStyle::kSingleLine
151 : CFGAS_Break::LayoutStyle::kNone;
153 m_pTxtBreak->SetLayoutStyles(m_dwTxtBkStyles);
157 m_iAlignment = iAlignment;
159 int32_t txtBreakAlignment = 0;
160 switch (m_iAlignment) {
172 m_pTxtBreak->SetAlignment(txtBreakAlignment);
176 DCHECK(fLineSpace > 1.0f);
177 m_fLineSpace = fLineSpace;
181 m_fTolerance = fTolerance;
182 m_pTxtBreak->SetLineBreakTolerance(m_fTolerance);
186 CFX_RectF rtText(0.0f, 0.0f, pSize->width, pSize->height);
187 CalcLogicSize(str, &rtText);
188 *pSize = rtText.Size();
199 DCHECK(m_fFontSize >= 1.0f);
203 pRect
->width = m_fFontSize * 1000.0f;
205 m_pTxtBreak->SetLineWidth(pRect->Width());
210 float fHeight = 0.0f;
213 bool break_char_is_set =
false;
214 for (
const wchar_t& wch : str) {
215 if (!break_char_is_set && (wch == L'\n' || wch == L'\r')) {
216 break_char_is_set =
true;
217 m_pTxtBreak->SetParagraphBreakChar(wch);
219 dwBreakStatus = m_pTxtBreak->AppendChar(wch);
220 if (!CFX_BreakTypeNoneOrPiece(dwBreakStatus))
221 RetrieveLineWidth(dwBreakStatus, &fStartPos, &fWidth, &fHeight);
224 dwBreakStatus = m_pTxtBreak->EndBreak(CFGAS_Char::BreakType::kParagraph);
226 RetrieveLineWidth(dwBreakStatus, &fStartPos, &fWidth, &fHeight);
228 m_pTxtBreak->Reset();
230 if (TextAlignmentVerticallyCentered(m_iAlignment))
232 else if (IsTextAlignmentTop(m_iAlignment))
240 pRect
->height -= m_fLineSpace - m_fFontSize;
250 float fLineStep =
std::max(m_fLineSpace, m_fFontSize);
251 float fLineWidth = 0.0f;
252 for (int32_t i = 0; i < m_pTxtBreak->CountBreakPieces(); i++) {
254 fLineWidth +=
static_cast<
float>(pPiece
->GetWidth()) / 20000.0f;
255 *pStartPos =
std::min(*pStartPos,
258 m_pTxtBreak->ClearBreakPieces();
260 if (dwBreakStatus == CFGAS_Char::BreakType::kParagraph)
261 m_pTxtBreak->Reset();
263 *pWidth += fLineWidth;
265 *pWidth =
std::max(*pWidth, fLineWidth);
266 *pHeight += fLineStep;
276 DCHECK(m_fFontSize >= 1.0f);
283 float fLineWidth = rect
.width;
284 m_pTxtBreak->SetLineWidth(fLineWidth);
292 if (!device || m_ttoLines.empty())
300 for (
auto& line : m_ttoLines) {
301 for (size_t i = 0; i < line.GetSize(); ++i) {
302 const Piece* pPiece = line.GetPieceAtIndex(i);
303 size_t szCount = GetDisplayPos(pPiece);
307 CFDE_TextOut::DrawString(device, m_TxtColor, m_pFont,
308 make_span(m_CharPos).first(szCount), m_fFontSize,
320 if (m_CharWidths.size() < str.GetLength())
321 m_CharWidths.resize(str.GetLength(), 0);
323 float fLineStep =
std::max(m_fLineSpace, m_fFontSize);
325 m_fLinePos = rect
.top;
326 size_t start_char = 0;
327 int32_t iPieceWidths = 0;
330 for (
const auto& wch : str) {
331 dwBreakStatus = m_pTxtBreak->AppendChar(wch);
332 if (CFX_BreakTypeNoneOrPiece(dwBreakStatus))
336 RetrievePieces(dwBreakStatus,
false, rect, &start_char, &iPieceWidths);
337 if (bEndofLine && (m_Styles.line_wrap_ ||
338 dwBreakStatus == CFGAS_Char::BreakType::kParagraph ||
339 dwBreakStatus == CFGAS_Char::BreakType::kPage)) {
342 m_fLinePos += fLineStep;
344 if (m_fLinePos + fLineStep > fLineStop) {
345 size_t iCurLine = bEndofLine ? m_iCurLine - 1 : m_iCurLine;
346 CHECK_LT(m_iCurLine, m_ttoLines.size());
347 m_ttoLines[iCurLine].set_new_reload(
true);
353 dwBreakStatus = m_pTxtBreak->EndBreak(CFGAS_Char::BreakType::kParagraph);
354 if (!CFX_BreakTypeNoneOrPiece(dwBreakStatus) && !bRet)
355 RetrievePieces(dwBreakStatus,
false, rect, &start_char, &iPieceWidths);
357 m_pTxtBreak->ClearBreakPieces();
358 m_pTxtBreak->Reset();
365 int32_t* pPieceWidths) {
366 float fLineStep =
std::max(m_fLineSpace, m_fFontSize);
367 bool bNeedReload =
false;
369 int32_t iCount = m_pTxtBreak->CountBreakPieces();
371 size_t chars_to_skip = *pStartChar;
372 for (int32_t i = 0; i < iCount; i++) {
374 size_t iPieceChars = pPiece->GetLength();
375 if (chars_to_skip > iPieceChars) {
376 chars_to_skip -= iPieceChars;
380 size_t iChar = *pStartChar;
382 size_t j = chars_to_skip;
383 for (; j < iPieceChars; j++) {
387 if (iLineWidth - *pPieceWidths - iWidth < iCurCharWidth) {
392 iWidth += iCurCharWidth;
393 m_CharWidths[iChar++] = iCurCharWidth;
396 if (j == chars_to_skip && !bReload) {
397 CHECK_LT(m_iCurLine, m_ttoLines.size());
398 m_ttoLines[m_iCurLine].set_new_reload(
true);
399 }
else if (j > chars_to_skip) {
401 piece.start_char = *pStartChar;
402 piece.char_count = j - chars_to_skip;
406 m_fLinePos
, iWidth / 20000.0f
, fLineStep
);
411 AppendPiece(piece, bNeedReload, (bReload && i == iCount - 1));
413 *pStartChar += iPieceChars;
414 *pPieceWidths += iWidth;
416 m_pTxtBreak->ClearBreakPieces();
425 if (m_iCurLine >= m_ttoLines.size()) {
427 ttoLine.set_new_reload(bNeedReload);
429 m_iCurPiece = ttoLine.AddPiece(m_iCurPiece, piece);
430 m_ttoLines.push_back(ttoLine);
431 m_iCurLine = m_ttoLines.size() - 1;
433 Line* pLine = &m_ttoLines[m_iCurLine];
434 pLine->set_new_reload(bNeedReload);
436 m_iCurPiece = pLine->AddPiece(m_iCurPiece, piece);
438 size_t iPieces = pLine->GetSize();
439 if (m_iCurPiece < iPieces)
440 pLine->RemoveLast(iPieces - m_iCurPiece - 1);
443 if (!bEnd && bNeedReload)
449 for (
auto& line : m_ttoLines) {
450 if (line.new_reload()) {
453 ReloadLinePiece(&line, rect);
460 span<
const wchar_t> text_span = m_wsText.span();
461 size_t start_char = 0;
462 size_t piece_count = line->GetSize();
463 int32_t piece_widths = 0;
465 for (size_t piece_index = 0; piece_index < piece_count; ++piece_index) {
466 const Piece* piece = line->GetPieceAtIndex(piece_index);
467 if (piece_index == 0)
468 m_fLinePos = piece->bounds
.top;
470 start_char = piece->start_char;
471 const size_t end = piece->start_char + piece->char_count;
472 for (size_t char_index = start_char; char_index < end; ++char_index) {
473 break_status = m_pTxtBreak->AppendChar(text_span[char_index]);
474 if (!CFX_BreakTypeNoneOrPiece(break_status))
475 RetrievePieces(break_status,
true, rect, &start_char, &piece_widths);
479 break_status = m_pTxtBreak->EndBreak(CFGAS_Char::BreakType::kParagraph);
480 if (!CFX_BreakTypeNoneOrPiece(break_status))
481 RetrievePieces(break_status,
true, rect, &start_char, &piece_widths);
483 m_pTxtBreak->Reset();
487 if (m_ttoLines.empty())
490 const Piece* pFirstPiece = m_ttoLines.back().GetPieceAtIndex(0);
495 if (TextAlignmentVerticallyCentered(m_iAlignment))
497 else if (IsTextAlignmentTop(m_iAlignment))
503 for (
auto& line : m_ttoLines) {
504 for (size_t i = 0; i < line.GetSize(); ++i)
505 line.GetPieceAtIndex(i)->bounds.top += fInc;
509size_t
CFDE_TextOut::GetDisplayPos(
const Piece* pPiece) {
510 if (m_CharPos.size() < pPiece->char_count)
511 m_CharPos.resize(pPiece->char_count, TextCharPos());
513 CFGAS_TxtBreak::
Run tr;
514 tr.wsStr = m_wsText.Substr(pPiece->start_char);
515 tr.pWidths = make_span(m_CharWidths).subspan(pPiece->start_char);
516 tr
.iLength = checked_cast<int32_t>(pPiece->char_count);
519 tr.dwStyles = m_dwTxtBkStyles;
521 tr.pRect = &pPiece->bounds;
523 return m_pTxtBreak->GetDisplayPos(tr, m_CharPos);
532size_t
CFDE_TextOut::Line::AddPiece(size_t index,
const Piece& piece) {
533 if (index >= pieces_.size()) {
534 pieces_.push_back(piece);
535 return pieces_.size();
537 pieces_[index] = piece;
542 return pieces_.size();
546 size_t index)
const {
547 CHECK(fxcrt::IndexInBounds(pieces_, index));
548 return &pieces_[index];
552 CHECK(fxcrt::IndexInBounds(pieces_, index));
553 return &pieces_[index];
557 pieces_.erase(pieces_.end() - std::min(count, pieces_.size()), pieces_.end());
@ CFX_TxtLineAlignment_Right
@ CFX_TxtLineAlignment_Center
@ CFX_TxtLineAlignment_Left
#define FX_TXTCHARSTYLE_OddBidiLevel
bool CFX_BreakTypeNoneOrPiece(CFGAS_Char::BreakType type)
int32_t GetStartPos() const
uint32_t GetCharStyles() const
int32_t GetBidiLevel() const
CFGAS_Char * GetChar(int32_t index) const
CFX_RectF TransformRect(const CFX_RectF &rect) const
constexpr CFX_RectF()=default
FX_RECT GetOuterRect() const
CFX_RectF & operator=(const CFX_RectF &other)=default
constexpr CFX_RectF(float dst_left, float dst_top, float dst_width, float dst_height)
bool SetClip_Rect(const FX_RECT &pRect)
void RestoreState(bool bKeepSaved)
void DrawLogicText(CFX_RenderDevice *device, const WideString &str, const CFX_RectF &rect)
void SetFont(RetainPtr< CFGAS_GEFont > pFont)
void CalcLogicSize(WideStringView str, CFX_RectF *pRect)
void SetStyles(const FDE_TextStyle &dwStyles)
void SetAlignment(FDE_TextAlignment iAlignment)
void SetLineSpace(float fLineSpace)
void CalcLogicSize(WideStringView str, CFX_SizeF *pSize)
void SetLineBreakTolerance(float fTolerance)
void SetFontSize(float fFontSize)
CFX_STemplate< float > CFX_SizeF
int FXSYS_roundf(float f)
fxcrt::WideStringView WideStringView
constexpr CFX_TextRenderOptions(AliasingType type)
fxcrt::WideString WideString