QSkinny 0.8.0
C++/Qt UI toolkit
Loading...
Searching...
No Matches
QskItemAnchors.cpp
1/******************************************************************************
2 * QSkinny - Copyright (C) The authors
3 * SPDX-License-Identifier: BSD-3-Clause
4 *****************************************************************************/
5
6#include "QskItemAnchors.h"
7#include "QskMargins.h"
8#include "QskInternalMacros.h"
9
10QSK_QT_PRIVATE_BEGIN
11#include <private/qquickanchors_p.h>
12#include <private/qquickanchors_p_p.h>
13#include <private/qquickitem_p.h>
14QSK_QT_PRIVATE_END
15
16namespace
17{
18 struct AnchorLineOperators
19 {
20 QQuickAnchorLine ( QQuickAnchors::*line ) () const;
21 void ( QQuickAnchors::*setLine )( const QQuickAnchorLine& );
22 void ( QQuickAnchors::*resetLine )();
23 };
24
25 class UpdateBlocker
26 {
27 public:
28 UpdateBlocker( QQuickAnchors* anchors )
29 : m_anchors( anchors )
30 {
31 auto d = QQuickAnchorsPrivate::get( anchors );
32
33 m_wasComplete = d->componentComplete;
34 d->componentComplete = false;
35 }
36
37 ~UpdateBlocker()
38 {
39 reset();
40 }
41
42 void reset() const
43 {
44 auto d = QQuickAnchorsPrivate::get( m_anchors );
45 d->componentComplete = m_wasComplete;
46 }
47 private:
48 QQuickAnchors* m_anchors;
49 bool m_wasComplete;
50 };
51}
52
53static inline QQuickAnchors* qskGetOrCreateAnchors( QQuickItem* item )
54{
55 if ( item == nullptr )
56 return nullptr;
57
58 return QQuickItemPrivate::get( item )->anchors();
59}
60
61static inline QQuickAnchors* qskGetAnchors( QQuickItem* item )
62{
63 if ( item == nullptr )
64 return nullptr;
65
66 return QQuickItemPrivate::get( item )->_anchors;
67}
68
69static inline const QQuickAnchors* qskGetAnchors( const QQuickItem* item )
70{
71 return qskGetAnchors( const_cast< QQuickItem* >( item ) );
72}
73
74static inline QQuickAnchors::Anchor qskToQuickAnchor( Qt::AnchorPoint edge )
75{
76 using A = QQuickAnchors;
77
78 switch( edge )
79 {
80 case Qt::AnchorLeft: return A::LeftAnchor;
81 case Qt::AnchorHorizontalCenter: return A::HCenterAnchor;
82 case Qt::AnchorRight: return A::RightAnchor;
83 case Qt::AnchorTop: return A::TopAnchor;
84 case Qt::AnchorVerticalCenter: return A::VCenterAnchor;
85 case Qt::AnchorBottom: return A::BottomAnchor;
86 default: return A::InvalidAnchor;
87 }
88}
89
90static inline Qt::AnchorPoint qskToAnchorPoint( QQuickAnchors::Anchor anchor )
91{
92 using A = QQuickAnchors;
93
94 switch( anchor )
95 {
96 case A::LeftAnchor: return Qt::AnchorLeft;
97 case A::HCenterAnchor: return Qt::AnchorHorizontalCenter;
98 case A::RightAnchor: return Qt::AnchorRight;
99 case A::TopAnchor: return Qt::AnchorTop;
100 case A::VCenterAnchor: return Qt::AnchorVerticalCenter;
101 case A::BottomAnchor: return Qt::AnchorBottom;
102 case A::BaselineAnchor: return Qt::AnchorTop; // not supported
103 default: return Qt::AnchorLeft;
104 }
105}
106
107static inline bool qskIsCenterAnchorPoint( Qt::AnchorPoint edge )
108{
109 return ( edge == Qt::AnchorHorizontalCenter )
110 || ( edge == Qt::AnchorVerticalCenter );
111}
112
113static inline const AnchorLineOperators& qskAnchorLineOperators( Qt::AnchorPoint edge )
114{
115 using A = QQuickAnchors;
116
117 static constexpr AnchorLineOperators table[] =
118 {
119 // in order of Qt::AnchorPoint
120
121 { &A::left, &A::setLeft, &A::resetLeft },
122 { &A::horizontalCenter, &A::setHorizontalCenter, &A::resetHorizontalCenter },
123 { &A::right, &A::setRight, &A::resetRight },
124 { &A::top, &A::setTop, &A::resetTop },
125 { &A::verticalCenter, &A::setVerticalCenter, &A::resetVerticalCenter },
126 { &A::bottom, &A::setBottom, &A::resetBottom }
127 };
128
129 return table[edge];
130}
131
132QskItemAnchors::QskItemAnchors( QQuickItem* attachedItem )
133 : m_attachedItem( attachedItem )
134{
135}
136
137QskItemAnchors::~QskItemAnchors()
138{
139}
140
141bool QskItemAnchors::operator==( const QskItemAnchors& other ) const noexcept
142{
143 return m_attachedItem.data() == other.m_attachedItem.data();
144}
145
146QQuickItem* QskItemAnchors::attachedItem() const
147{
148 return m_attachedItem;
149}
150
151QMarginsF QskItemAnchors::margins() const
152{
153 const auto anchors = qskGetAnchors( m_attachedItem );
154 if ( anchors == nullptr )
155 return QMarginsF();
156
157 const auto d = QQuickAnchorsPrivate::get( anchors );
158
159 const auto left = d->leftMarginExplicit ? d->leftMargin : d->margins;
160 const auto right = d->rightMarginExplicit ? d->rightMargin : d->margins;
161 const auto top = d->rightMarginExplicit ? d->rightMargin : d->margins;
162 const auto bottom = d->bottomMarginExplicit ? d->bottomMargin : d->margins;
163
164 return QMarginsF( left, top, right, bottom );
165}
166
167void QskItemAnchors::setMargins( const QMarginsF& margins )
168{
169 auto anchors = qskGetOrCreateAnchors( m_attachedItem );
170 if ( anchors == nullptr )
171 return;
172
173 const auto oldMargins = this->margins();
174
175 Qt::Orientations changes;
176
177 if ( margins.left() != oldMargins.left() )
178 changes |= Qt::Horizontal;
179
180 if ( margins.right() != oldMargins.right() )
181 changes |= Qt::Horizontal;
182
183 if ( margins.top() != oldMargins.top() )
184 changes |= Qt::Vertical;
185
186 if ( margins.bottom() != oldMargins.bottom() )
187 changes |= Qt::Vertical;
188
189 if ( changes )
190 {
191 const auto left = margins.left();
192
193 UpdateBlocker blocker( anchors );
194
195 anchors->setLeftMargin( left + 1.0 );
196 anchors->setRightMargin( margins.right() );
197 anchors->setTopMargin( margins.top() );
198 anchors->setBottomMargin( margins.bottom() );
199
200 blocker.reset();
201
202 anchors->setLeftMargin( left ); // trigger final update
203 }
204}
205
206void QskItemAnchors::setCenterOffsets(
207 qreal horizontalOffset, qreal verticalOffset )
208{
209 if ( auto anchors = qskGetOrCreateAnchors( m_attachedItem ) )
210 {
211 anchors->setHorizontalCenterOffset( horizontalOffset );
212 anchors->setVerticalCenterOffset( verticalOffset );
213 }
214}
215
216void QskItemAnchors::setCenterOffset( Qt::Orientation orientation, qreal offset )
217{
218 auto anchors = qskGetOrCreateAnchors( m_attachedItem );
219 if ( anchors == nullptr )
220 return;
221
222 if ( orientation == Qt::Horizontal )
223 anchors->setHorizontalCenterOffset( offset );
224 else
225 anchors->setVerticalCenterOffset( offset );
226}
227
228qreal QskItemAnchors::centerOffset( Qt::Orientation orientation )
229{
230 if ( const auto anchors = qskGetAnchors( m_attachedItem ) )
231 {
232 if ( orientation == Qt::Horizontal )
233 return anchors->horizontalCenterOffset();
234 else
235 return anchors->verticalCenterOffset();
236 }
237
238 return 0.0;
239}
240
241QQuickItem* QskItemAnchors::settledItem( Qt::AnchorPoint edge ) const
242{
243 const auto anchors = qskGetAnchors( m_attachedItem );
244 if ( anchors == nullptr )
245 return nullptr;
246
247 if ( auto fill = anchors->fill() )
248 return !qskIsCenterAnchorPoint( edge ) ? fill : nullptr;
249
250 if ( auto centerIn = anchors->centerIn() )
251 return qskIsCenterAnchorPoint( edge ) ? centerIn : nullptr;
252
253 const auto& ops = qskAnchorLineOperators( edge );
254 return ( ( anchors->*ops.line ) () ).item;
255}
256
257void QskItemAnchors::addAnchors( Qt::Corner corner,
258 QQuickItem* settledItem, Qt::Corner settledItemCorner )
259{
260 auto anchorPoint =
261 []( Qt::Corner cn, Qt::Orientation orientation )
262 {
263 if ( orientation == Qt::Horizontal )
264 return ( cn & 0x1 ) ? Qt::AnchorRight : Qt::AnchorLeft;
265 else
266 return ( cn >= 0x2 ) ? Qt::AnchorBottom : Qt::AnchorTop;
267 };
268
269 addAnchor( anchorPoint( corner, Qt::Horizontal ),
270 settledItem, anchorPoint( settledItemCorner, Qt::Horizontal ) );
271
272 addAnchor( anchorPoint( corner, Qt::Vertical ),
273 settledItem, anchorPoint( settledItemCorner, Qt::Vertical ) );
274}
275
276void QskItemAnchors::addAnchor( Qt::AnchorPoint edge, QQuickItem* settledItem,
277 Qt::AnchorPoint settledItemEdge )
278{
279 if ( settledItem == nullptr )
280 return;
281
282 if ( const auto anchors = qskGetOrCreateAnchors( m_attachedItem ) )
283 {
284 const auto& ops = qskAnchorLineOperators( edge );
285 ( anchors->*ops.setLine )( { settledItem, qskToQuickAnchor( settledItemEdge ) } );
286 }
287}
288
289void QskItemAnchors::removeAnchor( Qt::AnchorPoint edge )
290{
291 const auto anchors = qskGetAnchors( m_attachedItem );
292 if ( anchors == nullptr )
293 return;
294
295 if ( auto fill = anchors->fill() )
296 {
297 if ( !qskIsCenterAnchorPoint( edge ) )
298 {
299 anchors->resetFill();
300
301 // setting the other borders as anchors
302 for ( auto anchorPoint : { Qt::AnchorLeft, Qt::AnchorRight,
303 Qt::AnchorTop, Qt::AnchorBottom } )
304 {
305 if ( edge != anchorPoint )
306 addAnchor( anchorPoint, fill, anchorPoint );
307 }
308 }
309
310 return;
311 }
312
313 if ( auto centerIn = anchors->centerIn() )
314 {
315 if ( qskIsCenterAnchorPoint( edge ) )
316 {
317 anchors->resetCenterIn();
318
319 // setting the other direction as anchor
320 const auto otherEdge = ( edge == Qt::AnchorHorizontalCenter )
321 ? Qt::AnchorVerticalCenter : Qt::AnchorHorizontalCenter;
322
323 addAnchor( otherEdge, centerIn, otherEdge );
324 }
325
326 return;
327 }
328
329 const auto& ops = qskAnchorLineOperators( edge );
330 ( anchors->*ops.resetLine ) ();
331}
332
333void QskItemAnchors::clearAnchors()
334{
335 const auto anchors = qskGetAnchors( m_attachedItem );
336 if ( anchors == nullptr )
337 return;
338
339 const UpdateBlocker blocker( anchors );
340
341 anchors->resetFill();
342 anchors->resetCenterIn();
343
344 for ( int i = 0; i < 6; i++ )
345 {
346 const auto& ops = qskAnchorLineOperators( static_cast< Qt::AnchorPoint >( i ) );
347 ( anchors->*ops.resetLine ) ();
348 }
349}
350
351void QskItemAnchors::setBorderAnchors(
352 QQuickItem* settledItem, Qt::Orientations orientations )
353{
354 if ( settledItem == nullptr || m_attachedItem == nullptr )
355 return;
356
357 auto anchors = qskGetOrCreateAnchors( m_attachedItem );
358
359 switch( orientations )
360 {
361 case Qt::Horizontal:
362 {
363 clearAnchors();
364
365 const UpdateBlocker blocker( anchors );
366 addAnchor( Qt::AnchorLeft, settledItem, Qt::AnchorLeft );
367 blocker.reset();
368
369 addAnchor( Qt::AnchorRight, settledItem, Qt::AnchorRight );
370
371
372 break;
373 }
374
375 case Qt::Vertical:
376 {
377 clearAnchors();
378
379 const UpdateBlocker blocker( anchors );
380 addAnchor( Qt::AnchorTop, settledItem, Qt::AnchorTop );
381 blocker.reset();
382
383 addAnchor( Qt::AnchorBottom, settledItem, Qt::AnchorBottom );
384
385 break;
386 }
387
388 case Qt::Horizontal | Qt::Vertical:
389 {
390 if ( settledItem != anchors->fill() )
391 {
392 clearAnchors();
393 anchors->setFill( settledItem );
394 }
395 break;
396 }
397 }
398}
399
400void QskItemAnchors::setCenterAnchors(
401 QQuickItem* settledItem, Qt::Orientations orientations )
402{
403 if ( settledItem == nullptr || m_attachedItem == nullptr )
404 return;
405
406 switch( orientations )
407 {
408 case Qt::Horizontal:
409 {
410 clearAnchors();
411 addAnchor( Qt::AnchorHorizontalCenter,
412 settledItem, Qt::AnchorHorizontalCenter );
413 break;
414 }
415
416 case Qt::Vertical:
417 {
418 clearAnchors();
419 addAnchor( Qt::AnchorVerticalCenter,
420 settledItem, Qt::AnchorVerticalCenter );
421 break;
422 }
423
424 case Qt::Horizontal | Qt::Vertical:
425 {
426 auto anchors = qskGetOrCreateAnchors( m_attachedItem );
427
428 if ( settledItem != anchors->centerIn() )
429 {
430 clearAnchors();
431 anchors->setCenterIn( settledItem );
432 }
433 break;
434 }
435 }
436}
437
438Qt::AnchorPoint QskItemAnchors::settledItemAnchorPoint( Qt::AnchorPoint edge ) const
439{
440 if ( const auto anchors = qskGetAnchors( m_attachedItem ) )
441 {
442 /*
443 Anchoring to the baseline of the attachedItem might have been
444 done in QML code. As Qt::AnchorPoint does not have a corresponding
445 value for it and QSkinny does not support the baseline concept at all
446 we are lying and report Qt::AnchorTop instead. Hm ...
447 */
448
449 const auto& ops = qskAnchorLineOperators( edge );
450 return qskToAnchorPoint( ( ( anchors->*ops.line ) () ).anchorLine );
451 }
452
453 return Qt::AnchorLeft; // something
454}