QSkinny 0.8.0
C++/Qt UI toolkit
Loading...
Searching...
No Matches
QskRichTextRenderer.cpp
1/******************************************************************************
2 * QSkinny - Copyright (C) The authors
3 * SPDX-License-Identifier: BSD-3-Clause
4 *****************************************************************************/
5
6#include "QskRichTextRenderer.h"
7#include "QskTextColors.h"
8#include "QskTextOptions.h"
9#include "QskInternalMacros.h"
10
11#include <qglobalstatic.h>
12#include <qmutex.h>
13#include <qthread.h>
14
15class QQuickWindow;
16
17QSK_QT_PRIVATE_BEGIN
18#include <private/qquicktext_p.h>
19#include <private/qquicktext_p_p.h>
20QSK_QT_PRIVATE_END
21
22// Since Qt 5.7 QQuickTextNode is public and could be used TODO ...
23
24namespace
25{
26 class TextItem final : public QQuickText
27 {
28 public:
29 TextItem()
30 {
31#if 1
32 /*
33 QQuickTextPrivate::ExtraData::ExtraData is not exported with MSVC, so we
34 preallocate it by setting/unsetting the bottom padding
35 */
36 setBottomPadding( 1 );
37 setBottomPadding( 0 );
38#endif
39
40 // fonts are supposed to be defined in the application skin and we
41 // probably don't want to have them scaled
42 setFontSizeMode( QQuickText::FixedSize );
43
44#if 0
45 setAntialiasing( true );
46 setRenderType( QQuickText::QtRendering );
47 setPadding( 0 );
48
49 setMinimumPixelSize();
50 setMinimumPointSize();
51
52 // also something, that should be defined in an application skin
53 setLineHeightMode( ... );
54 setLineHeight();
55#endif
56 }
57
58 inline void setGeometry( const QRectF& rect )
59 {
60 auto d = QQuickTextPrivate::get( this );
61
62#if QT_VERSION >= QT_VERSION_CHECK( 6, 2, 0 )
63 d->heightValidFlag = true;
64 d->widthValidFlag = true;
65#else
66 d->heightValid = true;
67 d->widthValid = true;
68#endif
69
70 if ( ( d->x != rect.x() ) || ( d->y != rect.y() ) )
71 {
72 d->x = rect.x();
73 d->y = rect.y();
74 d->dirty( QQuickItemPrivate::Position );
75 }
76
77 if ( ( d->width != rect.width() ) || ( d->height != rect.height() ) )
78 {
79 d->height = rect.height();
80 d->width = rect.width();
81 d->dirty( QQuickItemPrivate::Size );
82 }
83 }
84
85 inline void setAlignment( Qt::Alignment alignment )
86 {
87 setHAlign( static_cast< QQuickText::HAlignment >( int( alignment ) & 0x0f ) );
88 setVAlign( static_cast< QQuickText::VAlignment >( int( alignment ) & 0xf0 ) );
89 }
90
91 inline void setOptions( const QskTextOptions& options )
92 {
93 // what about Qt::TextShowMnemonic ???
94 setTextFormat( static_cast< QQuickText::TextFormat >( options.format() ) );
95 setElideMode( static_cast< QQuickText::TextElideMode >( options.elideMode() ) );
96 setMaximumLineCount( options.maximumLineCount() );
97 setWrapMode( static_cast< QQuickText::WrapMode >( options.wrapMode() ) );
98 }
99
100 inline void begin()
101 {
102 classBegin();
103 QQuickTextPrivate::get( this )->updateOnComponentComplete = true;
104 }
105
106 inline void end()
107 {
108 componentComplete();
109 }
110
111 inline void reset()
112 {
113 setText( QString() );
114 }
115
116 inline QRectF layedOutTextRect() const
117 {
118 auto that = const_cast< TextItem* >( this );
119 return QQuickTextPrivate::get( that )->layedOutTextRect;
120 }
121
122 void updateTextNode( QQuickWindow* window, QSGNode* parentNode )
123 {
124 QQuickItemPrivate::get( this )->refWindow( window );
125
126 while ( parentNode->firstChild() )
127 delete parentNode->firstChild();
128
129 auto node = QQuickText::updatePaintNode( nullptr, nullptr );
130 node->reparentChildNodesTo( parentNode );
131 delete node;
132
133 QQuickItemPrivate::get( this )->derefWindow();
134 }
135
136 protected:
137 QSGNode* updatePaintNode( QSGNode*, UpdatePaintNodeData* ) override
138 {
139 Q_ASSERT( false );
140 return nullptr;
141 }
142 };
143
144 class TextItemMap
145 {
146 public:
147 ~TextItemMap()
148 {
149 qDeleteAll( m_hash );
150 }
151
152 inline TextItem* item()
153 {
154 const auto thread = QThread::currentThread();
155
156 QMutexLocker locker( &m_mutex );
157
158 auto it = m_hash.constFind( thread );
159 if ( it == m_hash.constEnd() )
160 {
161 auto textItem = new TextItem();
162 QObject::connect( thread, &QThread::finished,
163 textItem, [ this, thread ] { removeItem( thread ); } );
164
165 m_hash.insert( thread, textItem );
166 return textItem;
167 }
168
169 return it.value();
170 }
171
172 private:
173 void removeItem( const QThread* thread )
174 {
175 auto textItem = m_hash.take( thread );
176 if ( textItem )
177 textItem->deleteLater();
178 }
179
180 QMutex m_mutex;
181 QHash< const QThread*, TextItem* > m_hash;
182 };
183}
184
185/*
186 size requests and rendering might be from different threads and we
187 better use different items as we might end up in events internally
188 being sent, that leads to crashes because of it
189 */
190Q_GLOBAL_STATIC( TextItemMap, qskTextItemMap )
191
192QSizeF QskRichTextRenderer::textSize(
193 const QString& text, const QFont& font, const QskTextOptions& options )
194{
195 auto& textItem = *qskTextItemMap->item();
196
197 textItem.begin();
198
199 textItem.setFont( font );
200 textItem.setOptions( options );
201
202 textItem.setWidth( -1 );
203 textItem.setText( text );
204
205 textItem.end();
206
207 const QSizeF sz( textItem.implicitWidth(), textItem.implicitHeight() );
208
209 textItem.reset();
210
211 return sz;
212}
213
214QRectF QskRichTextRenderer::textRect(
215 const QString& text, const QFont& font,
216 const QskTextOptions& options, const QSizeF& size )
217{
218 auto& textItem = *qskTextItemMap->item();
219
220 textItem.begin();
221
222 textItem.setFont( font );
223 textItem.setOptions( options );
224 textItem.setAlignment( Qt::Alignment() );
225
226 textItem.setWidth( size.width() );
227 textItem.setHeight( size.height() );
228
229 textItem.setText( text );
230
231 textItem.end();
232
233 const auto rect = textItem.layedOutTextRect();
234
235 textItem.reset();
236
237 return rect;
238}
239
240void QskRichTextRenderer::updateNode(
241 const QString& text, const QFont& font,
242 const QskTextOptions& options, Qsk::TextStyle style,
243 const QskTextColors& colors, Qt::Alignment alignment,
244 const QRectF& rect, const QQuickItem* item, QSGTransformNode* node )
245{
246 // are we killing internal caches of QQuickText, when always using
247 // the same item for the creation the text nodes. TODO ...
248
249 auto& textItem = *qskTextItemMap->item();
250
251 textItem.begin();
252
253 textItem.setGeometry( rect );
254
255 textItem.setBottomPadding( 0 );
256 textItem.setTopPadding( 0 );
257 textItem.setFont( font );
258 textItem.setOptions( options );
259 textItem.setAlignment( alignment );
260
261 textItem.setColor( colors.textColor() );
262 textItem.setStyle( static_cast< QQuickText::TextStyle >( style ) );
263 textItem.setStyleColor( colors.styleColor() );
264 textItem.setLinkColor( colors.linkColor() );
265
266 textItem.setText( text );
267
268 textItem.end();
269
270 if ( alignment & Qt::AlignVCenter )
271 {
272 /*
273 We need to have a stable algo for rounding the text base line,
274 so that texts don't start wobbling, when processing transitions
275 between margins/paddings. We manipulate the layout code
276 by adding some padding, so that the position of base line
277 gets always floored.
278 */
279 auto d = QQuickTextPrivate::get( &textItem );
280
281 const qreal h = d->layedOutTextRect.height() + d->lineHeightOffset();
282
283 if ( static_cast< int >( rect.height() - h ) % 2 )
284 {
285 if ( static_cast< int >( h ) % 2 )
286 d->extra->bottomPadding = 1;
287 else
288 d->extra->topPadding = 1;
289 }
290 }
291
292 textItem.updateTextNode( item->window(), node );
293 textItem.reset();
294}