6#include "QskPlainTextRenderer.h"
7#include "QskTextColors.h"
8#include "QskTextOptions.h"
9#include "QskInternalMacros.h"
11#include <qfontmetrics.h>
16#include <private/qquickitem_p.h>
19#define GlyphFlag static_cast< QSGNode::Flag >( 0x800 )
21QSizeF QskPlainTextRenderer::textSize(
22 const QString& text,
const QFont& font,
const QskTextOptions& options )
25 return textRect( text, font, options, QSizeF( 10e6, 10e6 ) ).size();
28QRectF QskPlainTextRenderer::textRect(
29 const QString& text,
const QFont& font,
const QskTextOptions& options,
32 const QFontMetricsF fm( font );
33 const QRectF r( 0, 0, size.width(), size.height() );
35 return fm.boundingRect( r, options.textFlags(), text );
38static qreal qskLayoutText( QTextLayout* layout,
41 if ( layout->text().isEmpty() )
46 const auto elideMode = options.effectiveElideMode();
48 if ( elideMode == Qt::ElideNone )
50 for (
int i = 0; i < options.maximumLineCount(); i++ )
52 auto line = layout->createLine();
53 if ( !line.isValid() )
56 line.setPosition( QPointF( 0, y ) );
57 line.setLineWidth( lineWidth );
59 y += line.leading() + line.height();
64 const auto engine = layout->engine();
66 auto text = engine->elidedText(
67 elideMode, QFixed::fromReal( lineWidth ),
68 Qt::TextShowMnemonic, 0 );
71 text = text.leftJustified( engine->text.length() );
74 auto line = layout->createLine();
84 auto option = layout->textOption();
85 option.setWrapMode( QTextOption::WrapAnywhere );
86 layout->setTextOption( option );
88 line.setPosition( QPointF( 0, y ) );
89 line.setLineWidth( lineWidth );
91 y += line.leading() + line.height();
98static void qskRenderText(
99 QQuickItem* item, QSGNode* parentNode,
const QTextLayout& layout, qreal baseLine,
100 const QColor& color, QQuickText::TextStyle style,
const QColor& styleColor )
102 auto renderContext = QQuickItemPrivate::get(item)->sceneGraphRenderContext();
103 auto sgContext = renderContext->sceneGraphContext();
106 QSGNode* node = parentNode->firstChild();
109 auto sibling = node->nextSibling();
110 if ( !( node->flags() & GlyphFlag ) )
112 parentNode->removeChildNode( node );
118 auto glyphNode =
static_cast< QSGGlyphNode*
>( parentNode->firstChild() );
120 const QPointF position( 0, baseLine );
122 for (
int i = 0; i < layout.lineCount(); ++i )
124 const auto glyphRuns = layout.lineAt( i ).glyphRuns();
126 for (
const auto& glyphRun : glyphRuns )
128 if ( glyphNode ==
nullptr )
130 const bool preferNativeGlyphNode =
false;
131 constexpr int renderQuality = -1;
133#if QT_VERSION >= QT_VERSION_CHECK( 6, 7, 0 )
134 const auto renderType = preferNativeGlyphNode
135 ? QSGTextNode::QtRendering : QSGTextNode::NativeRendering;
136 glyphNode = sgContext->createGlyphNode(
137 renderContext, renderType, renderQuality );
138#elif QT_VERSION >= QT_VERSION_CHECK( 6, 0, 0 )
139 glyphNode = sgContext->createGlyphNode(
140 renderContext, preferNativeGlyphNode, renderQuality );
142 Q_UNUSED( renderQuality );
143 glyphNode = sgContext->createGlyphNode(
144 renderContext, preferNativeGlyphNode );
147#if QT_VERSION < QT_VERSION_CHECK( 6, 7, 0 )
148 glyphNode->setOwnerElement( item );
151 glyphNode->setFlags( QSGNode::OwnedByParent | GlyphFlag );
154 glyphNode->setStyle( style );
155 glyphNode->setColor( color );
156 glyphNode->setStyleColor( styleColor );
157 glyphNode->setGlyphs( position, glyphRun );
160 if ( glyphNode->parent() != parentNode )
161 parentNode->appendChildNode( glyphNode );
163 glyphNode =
static_cast< QSGGlyphNode*
>( glyphNode->nextSibling() );
170 auto sibling = glyphNode->nextSibling();
171 if ( glyphNode->flags() & GlyphFlag )
173 parentNode->removeChildNode( glyphNode );
176 glyphNode =
static_cast< QSGGlyphNode*
>( sibling );
180void QskPlainTextRenderer::updateNode(
const QString& text,
183 Qt::Alignment alignment,
const QRectF& rect,
184 const QQuickItem* item, QSGTransformNode* node )
186 QTextOption textOption( alignment );
187 textOption.setWrapMode(
static_cast< QTextOption::WrapMode
>( options.wrapMode() ) );
192 const int pos = tmp.indexOf( QLatin1Char(
'\x9c' ) );
197 tmp = tmp.mid( 0, pos );
198 tmp.replace( QLatin1Char(
'\n' ), QChar::LineSeparator );
202 if ( tmp.contains( QLatin1Char(
'\n' ) ) )
204 tmp.replace( QLatin1Char(
'\n'), QChar::LineSeparator );
209 layout.setFont( font );
210 layout.setTextOption( textOption );
211 layout.setText( tmp );
213 layout.beginLayout();
214 const qreal textHeight = qskLayoutText( &layout, rect.width(), options );
217 const qreal y0 = QFontMetricsF( font ).ascent();
219 qreal yBaseline = y0;
221 if ( alignment & Qt::AlignVCenter )
223 yBaseline += ( rect.height() - textHeight ) * 0.5;
225 else if ( alignment & Qt::AlignBottom )
227 yBaseline += rect.height() - textHeight;
230 if ( yBaseline != y0 )
238 const int bh = int( layout.boundingRect().height() );
239 yBaseline = ( bh % 2 ) ? qFloor( yBaseline ) : qCeil( yBaseline );
243 const_cast< QQuickItem*
>( item ), node, layout, yBaseline,
244 colors.textColor(),
static_cast< QQuickText::TextStyle
>( style ),
245 colors.styleColor() );
248void QskPlainTextRenderer::updateNodeColor(
249 QSGNode* parentNode,
const QColor& textColor,
250 Qsk::TextStyle style,
const QColor& styleColor )
252 auto glyphNode =
static_cast< QSGGlyphNode*
>( parentNode->firstChild() );
255 if ( glyphNode->flags() & GlyphFlag )
257 glyphNode->setColor( textColor );
258 glyphNode->setStyle(
static_cast< QQuickText::TextStyle
>( style ) );
259 glyphNode->setStyleColor( styleColor );
262 glyphNode =
static_cast< QSGGlyphNode*
>( glyphNode->nextSibling() );