QSkinny 0.8.0
C++/Qt UI toolkit
Loading...
Searching...
No Matches
QskPlainTextRenderer.cpp
1/******************************************************************************
2 * QSkinny - Copyright (C) The authors
3 * SPDX-License-Identifier: BSD-3-Clause
4 *****************************************************************************/
5
6#include "QskPlainTextRenderer.h"
7#include "QskTextColors.h"
8#include "QskTextOptions.h"
9#include "QskInternalMacros.h"
10
11#include <qfontmetrics.h>
12#include <qmath.h>
13#include <qsgnode.h>
14
15QSK_QT_PRIVATE_BEGIN
16#include <private/qquickitem_p.h>
17QSK_QT_PRIVATE_END
18
19#define GlyphFlag static_cast< QSGNode::Flag >( 0x800 )
20
21QSizeF QskPlainTextRenderer::textSize(
22 const QString& text, const QFont& font, const QskTextOptions& options )
23{
24 // result differs from QQuickText::implicitSizeHint ???
25 return textRect( text, font, options, QSizeF( 10e6, 10e6 ) ).size();
26}
27
28QRectF QskPlainTextRenderer::textRect(
29 const QString& text, const QFont& font, const QskTextOptions& options,
30 const QSizeF& size )
31{
32 const QFontMetricsF fm( font );
33 const QRectF r( 0, 0, size.width(), size.height() );
34
35 return fm.boundingRect( r, options.textFlags(), text );
36}
37
38static qreal qskLayoutText( QTextLayout* layout,
39 qreal lineWidth, const QskTextOptions& options )
40{
41 if ( layout->text().isEmpty() )
42 return 0.0;
43
44 qreal y = 0;
45
46 const auto elideMode = options.effectiveElideMode();
47
48 if ( elideMode == Qt::ElideNone )
49 {
50 for ( int i = 0; i < options.maximumLineCount(); i++ )
51 {
52 auto line = layout->createLine();
53 if ( !line.isValid() )
54 break;
55
56 line.setPosition( QPointF( 0, y ) );
57 line.setLineWidth( lineWidth );
58
59 y += line.leading() + line.height();
60 }
61 }
62 else
63 {
64 const auto engine = layout->engine();
65
66 auto text = engine->elidedText(
67 elideMode, QFixed::fromReal( lineWidth ),
68 Qt::TextShowMnemonic, 0 );
69
70 // why do we need this padding ???
71 text = text.leftJustified( engine->text.length() );
72 engine->text = text;
73
74 auto line = layout->createLine();
75
76 if ( line.isValid() )
77 {
78 /*
79 For some reason the position of the text is wrong,
80 with QTextOption::NoWrap - even if word wrapping
81 for elided text does not make any sense.
82 Needs some debugging of QTextLine::layout_helper, TODO ...
83 */
84 auto option = layout->textOption();
85 option.setWrapMode( QTextOption::WrapAnywhere );
86 layout->setTextOption( option );
87
88 line.setPosition( QPointF( 0, y ) );
89 line.setLineWidth( lineWidth );
90
91 y += line.leading() + line.height();
92 }
93 }
94
95 return y;
96}
97
98static void qskRenderText(
99 QQuickItem* item, QSGNode* parentNode, const QTextLayout& layout, qreal baseLine,
100 const QColor& color, QQuickText::TextStyle style, const QColor& styleColor )
101{
102 auto renderContext = QQuickItemPrivate::get(item)->sceneGraphRenderContext();
103 auto sgContext = renderContext->sceneGraphContext();
104
105 // Clear out foreign nodes (e.g. from QskRichTextRenderer)
106 QSGNode* node = parentNode->firstChild();
107 while ( node )
108 {
109 auto sibling = node->nextSibling();
110 if ( !( node->flags() & GlyphFlag ) )
111 {
112 parentNode->removeChildNode( node );
113 delete node;
114 }
115 node = sibling;
116 }
117
118 auto glyphNode = static_cast< QSGGlyphNode* >( parentNode->firstChild() );
119
120 const QPointF position( 0, baseLine );
121
122 for ( int i = 0; i < layout.lineCount(); ++i )
123 {
124 const auto glyphRuns = layout.lineAt( i ).glyphRuns();
125
126 for ( const auto& glyphRun : glyphRuns )
127 {
128 if ( glyphNode == nullptr )
129 {
130 const bool preferNativeGlyphNode = false; // QskTextOptions?
131 constexpr int renderQuality = -1; // QQuickText::DefaultRenderTypeQuality
132
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 );
141#else
142 Q_UNUSED( renderQuality );
143 glyphNode = sgContext->createGlyphNode(
144 renderContext, preferNativeGlyphNode );
145#endif
146
147#if QT_VERSION < QT_VERSION_CHECK( 6, 7, 0 )
148 glyphNode->setOwnerElement( item );
149#endif
150
151 glyphNode->setFlags( QSGNode::OwnedByParent | GlyphFlag );
152 }
153
154 glyphNode->setStyle( style );
155 glyphNode->setColor( color );
156 glyphNode->setStyleColor( styleColor );
157 glyphNode->setGlyphs( position, glyphRun );
158 glyphNode->update();
159
160 if ( glyphNode->parent() != parentNode )
161 parentNode->appendChildNode( glyphNode );
162
163 glyphNode = static_cast< QSGGlyphNode* >( glyphNode->nextSibling() );
164 }
165 }
166
167 // Remove leftover glyphs
168 while ( glyphNode )
169 {
170 auto sibling = glyphNode->nextSibling();
171 if ( glyphNode->flags() & GlyphFlag )
172 {
173 parentNode->removeChildNode( glyphNode );
174 delete glyphNode;
175 }
176 glyphNode = static_cast< QSGGlyphNode* >( sibling );
177 }
178}
179
180void QskPlainTextRenderer::updateNode( const QString& text,
181 const QFont& font, const QskTextOptions& options,
182 Qsk::TextStyle style, const QskTextColors& colors,
183 Qt::Alignment alignment, const QRectF& rect,
184 const QQuickItem* item, QSGTransformNode* node )
185{
186 QTextOption textOption( alignment );
187 textOption.setWrapMode( static_cast< QTextOption::WrapMode >( options.wrapMode() ) );
188
189 QString tmp = text;
190
191#if 0
192 const int pos = tmp.indexOf( QLatin1Char( '\x9c' ) );
193 if ( pos != -1 )
194 {
195 // ST: string termination
196
197 tmp = tmp.mid( 0, pos );
198 tmp.replace( QLatin1Char( '\n' ), QChar::LineSeparator );
199 }
200 else
201#endif
202 if ( tmp.contains( QLatin1Char( '\n' ) ) )
203 {
204 tmp.replace( QLatin1Char('\n'), QChar::LineSeparator );
205 }
206
207
208 QTextLayout layout;
209 layout.setFont( font );
210 layout.setTextOption( textOption );
211 layout.setText( tmp );
212
213 layout.beginLayout();
214 const qreal textHeight = qskLayoutText( &layout, rect.width(), options );
215 layout.endLayout();
216
217 const qreal y0 = QFontMetricsF( font ).ascent();
218
219 qreal yBaseline = y0;
220
221 if ( alignment & Qt::AlignVCenter )
222 {
223 yBaseline += ( rect.height() - textHeight ) * 0.5;
224 }
225 else if ( alignment & Qt::AlignBottom )
226 {
227 yBaseline += rect.height() - textHeight;
228 }
229
230 if ( yBaseline != y0 )
231 {
232 /*
233 We need to have a stable algo for rounding the text base line,
234 so that texts don't start wobbling, when processing transitions
235 between margins/paddings.
236 */
237
238 const int bh = int( layout.boundingRect().height() );
239 yBaseline = ( bh % 2 ) ? qFloor( yBaseline ) : qCeil( yBaseline );
240 }
241
242 qskRenderText(
243 const_cast< QQuickItem* >( item ), node, layout, yBaseline,
244 colors.textColor(), static_cast< QQuickText::TextStyle >( style ),
245 colors.styleColor() );
246}
247
248void QskPlainTextRenderer::updateNodeColor(
249 QSGNode* parentNode, const QColor& textColor,
250 Qsk::TextStyle style, const QColor& styleColor )
251{
252 auto glyphNode = static_cast< QSGGlyphNode* >( parentNode->firstChild() );
253 while ( glyphNode )
254 {
255 if ( glyphNode->flags() & GlyphFlag )
256 {
257 glyphNode->setColor( textColor );
258 glyphNode->setStyle( static_cast< QQuickText::TextStyle >( style ) );
259 glyphNode->setStyleColor( styleColor );
260 glyphNode->update();
261 }
262 glyphNode = static_cast< QSGGlyphNode* >( glyphNode->nextSibling() );
263 }
264}