Qt
Internal/Contributor docs for the Qt SDK. Note: These are NOT official API docs; those are found at https://doc.qt.io/
Loading...
Searching...
No Matches
qquickstackelement.cpp
Go to the documentation of this file.
1// Copyright (C) 2017 The Qt Company Ltd.
2// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR LGPL-3.0-only OR GPL-2.0-only OR GPL-3.0-only
3// Qt-Security score:significant reason:default
4
8
9#include <QtQml/qqmlinfo.h>
10#include <QtQml/qqmlengine.h>
11#include <QtQml/qqmlcomponent.h>
12#include <QtQml/qqmlincubator.h>
13#include <QtQml/private/qv4qobjectwrapper_p.h>
14#include <QtQml/private/qqmlcomponent_p.h>
15#include <QtQml/private/qqmlengine_p.h>
16#include <QtQml/private/qqmlincubator_p.h>
17
19
20#if QT_CONFIG(quick_viewtransitions)
21static QQuickStackViewAttached *attachedStackObject(QQuickStackElement *element)
22{
23 QQuickStackViewAttached *attached = qobject_cast<QQuickStackViewAttached *>(qmlAttachedPropertiesObject<QQuickStackView>(element->item, false));
24 if (attached)
25 QQuickStackViewAttachedPrivate::get(attached)->element = element;
26 return attached;
27}
28#endif
29
31{
32public:
35 element(element)
36 {
37 }
38
39protected:
40 void setInitialState(QObject *object) override
41 {
42 auto privIncubator = QQmlIncubatorPrivate::get(this);
43 if (QQmlEnginePrivate *enginePriv = privIncubator->enginePriv) {
44 element->incubate(enginePriv->v4Engine.get(), object,
45 privIncubator->requiredProperties());
46 }
47 }
48
49private:
50 QQuickStackElement *element;
51};
52
53QQuickStackElement::QQuickStackElement()
54#if QT_CONFIG(quick_viewtransitions)
55 : QQuickItemViewTransitionableItem(nullptr)
56#endif
57{
58}
59
61{
62#if QT_CONFIG(quick_viewtransitions)
63 if (item)
64 QQuickItemPrivate::get(item)->removeItemChangeListener(this, QQuickItemPrivate::Destroyed);
65#endif
66
67 if (ownComponent)
68 delete component;
69
70#if QT_CONFIG(quick_viewtransitions)
71 QQuickStackViewAttached *attached = attachedStackObject(this);
72 if (item) {
73 if (ownItem) {
74 item->setParentItem(nullptr);
75 item->deleteLater();
76 item = nullptr;
77 } else {
78 setVisible(false);
79 if (!widthValid)
80 item->resetWidth();
81 if (!heightValid)
82 item->resetHeight();
83 if (item->parentItem() != originalParent) {
84 item->setParentItem(originalParent);
85 } else {
86 if (attached)
87 QQuickStackViewAttachedPrivate::get(attached)->itemParentChanged(item, nullptr);
88 }
89 }
90 }
91
92 if (attached)
93 emit attached->removed();
94#endif
95}
96
98 QQmlEngine *engine, const QString &str, QQuickStackView *view, QString *error)
99{
100 QUrl url(str);
101 if (!url.isValid()) {
102 *error = QStringLiteral("invalid url: ") + str;
103 return nullptr;
104 }
105
106 if (url.isRelative())
107 url = qmlContext(view)->resolvedUrl(url);
108
110 element->component = new QQmlComponent(engine, url, view);
111 element->ownComponent = true;
112 return element;
113}
114
115QQuickStackElement *QQuickStackElement::fromObject(QObject *object, QQuickStackView *view, QString *error)
116{
117 Q_UNUSED(view);
118 QQmlComponent *component = qobject_cast<QQmlComponent *>(object);
119 QQuickItem *item = qobject_cast<QQuickItem *>(object);
120 if (!component && !item) {
121 *error = QQmlMetaType::prettyTypeName(object) + QStringLiteral(" is not supported. Must be Item or Component.");
122 return nullptr;
123 }
124
125 QQuickStackElement *element = new QQuickStackElement;
126 element->component = qobject_cast<QQmlComponent *>(object);
127#if QT_CONFIG(quick_viewtransitions)
128 element->item = qobject_cast<QQuickItem *>(object);
129 if (element->item)
130 element->originalParent = element->item->parentItem();
131#endif
132 return element;
133}
134
136 QQmlEngine *engine, QQuickStackView *view, QQuickStackViewArg arg)
137{
138 QQuickStackElement *element = new QQuickStackElement;
139#if QT_CONFIG(quick_viewtransitions)
140 element->item = arg.mItem;
141 if (element->item) {
142 element->originalParent = element->item->parentItem();
143
144 Q_ASSERT(!arg.mComponent);
145 Q_ASSERT(!arg.mUrl.isValid());
146 } else
147#endif
148 if (arg.mComponent) {
149 element->component = arg.mComponent;
150
151 Q_ASSERT(!arg.mUrl.isValid());
152 } else if (arg.mUrl.isValid()) {
153 element->component = new QQmlComponent(engine, arg.mUrl, view);
154 element->ownComponent = true;
155 } else {
156 qFatal("No Item, Component or URL set on arg passed to fromStrictArg");
157 }
158 return element;
159}
160
161bool QQuickStackElement::load(QV4::ExecutionEngine *v4, QQuickStackView *parent)
162{
163 setView(parent);
164 if (!item) {
165 ownItem = true;
166
167 if (component->isLoading()) {
168 QObject::connect(component, &QQmlComponent::statusChanged, [this](QQmlComponent::Status status) {
169 if (status == QQmlComponent::Ready)
170 load(component->engine()->handle(), view);
171 else if (status == QQmlComponent::Error)
172 QQuickStackViewPrivate::get(view)->warn(component->errorString().trimmed());
173 });
174 return true;
175 }
176
177 QQmlContext *context = component->creationContext();
178 if (!context)
179 context = qmlContext(parent);
180
181 QQuickStackIncubator incubator(this);
182 component->create(incubator, context);
183 if (component->isError())
184 QQuickStackViewPrivate::get(parent)->warn(component->errorString().trimmed());
185 } else {
186 initialize(v4, /*required properties=*/nullptr);
187 }
188 return item;
189}
190
192 QV4::ExecutionEngine *v4, QObject *object, RequiredProperties *requiredProperties)
193{
194 item = qmlobject_cast<QQuickItem *>(object);
195 if (item) {
196 QQmlEngine::setObjectOwnership(item, QQmlEngine::CppOwnership);
197 item->setParent(view);
198 initialize(v4, requiredProperties);
199 }
200}
201
203 QV4::ExecutionEngine *v4, RequiredProperties *requiredProperties)
204{
205 if (!item || init)
206 return;
207
208 QQuickItemPrivate *p = QQuickItemPrivate::get(item);
209 if (!(widthValid = p->widthValid()))
210 item->setWidth(view->width());
211 if (!(heightValid = p->heightValid()))
212 item->setHeight(view->height());
213 item->setParentItem(view);
214
215 if (!properties.isUndefined()) {
216 QV4::Scope scope(v4);
217 Q_ASSERT(scope.engine);
218 QV4::ScopedValue ipv(scope, properties.value());
219 QV4::Scoped<QV4::QmlContext> qmlContext(scope, qmlCallingContext.value());
220 QV4::ScopedValue qmlObject(scope, QV4::QObjectWrapper::wrap(scope.engine, item));
221 QQmlComponentPrivate::setInitialProperties(
222 scope.engine, qmlContext, qmlObject, ipv, requiredProperties, item,
223 component ? QQmlComponentPrivate::get(component)->creator() : nullptr);
224 properties.clear();
225 }
226
227 if (requiredProperties && !requiredProperties->empty()) {
228 QString error;
229 for (const auto &property: *requiredProperties) {
230 error += QLatin1String("Property %1 was marked as required but not set.\n")
231 .arg(property.propertyName);
232 }
233 QQuickStackViewPrivate::get(view)->warn(error);
234 item = nullptr;
235 } else {
236 p->addItemChangeListener(this, QQuickItemPrivate::Destroyed);
237 }
238
239 init = true;
240}
241
243{
244 if (index == value)
245 return;
246
247 index = value;
248#if QT_CONFIG(quick_viewtransitions)
249 QQuickStackViewAttached *attached = attachedStackObject(this);
250 if (attached)
251 emit attached->indexChanged();
252#endif
253}
254
255void QQuickStackElement::setView(QQuickStackView *value)
256{
257 if (view == value)
258 return;
259
260 view = value;
261#if QT_CONFIG(quick_viewtransitions)
262 QQuickStackViewAttached *attached = attachedStackObject(this);
263 if (attached)
264 emit attached->viewChanged();
265#endif
266}
267
268void QQuickStackElement::setStatus(QQuickStackView::Status value)
269{
270 if (status == value)
271 return;
272
273 status = value;
274#if QT_CONFIG(quick_viewtransitions)
275 QQuickStackViewAttached *attached = attachedStackObject(this);
276 if (!attached)
277 return;
278
279 switch (value) {
280 case QQuickStackView::Inactive:
281 emit attached->deactivated();
282 break;
283 case QQuickStackView::Deactivating:
284 emit attached->deactivating();
285 break;
286 case QQuickStackView::Activating:
287 emit attached->activating();
288 break;
289 case QQuickStackView::Active:
290 emit attached->activated();
291 break;
292 default:
293 Q_UNREACHABLE();
294 break;
295 }
296
297 emit attached->statusChanged();
298#endif
299}
300
302{
303#if QT_CONFIG(quick_viewtransitions)
304 QQuickStackViewAttached *attached = attachedStackObject(this);
305#endif
306 if (!item
307#if QT_CONFIG(quick_viewtransitions)
308 || (attached && QQuickStackViewAttachedPrivate::get(attached)->explicitVisible)
309#endif
310 )
311 return;
312
313 item->setVisible(visible);
314}
315
316#if QT_CONFIG(quick_viewtransitions)
317void QQuickStackElement::transitionNextReposition(QQuickItemViewTransitioner *transitioner, QQuickItemViewTransitioner::TransitionType type, bool asTarget)
318{
319 if (transitioner)
320 transitioner->transitionNextReposition(this, type, asTarget);
321}
322
323bool QQuickStackElement::prepareTransition(QQuickItemViewTransitioner *transitioner, const QRectF &viewBounds)
324{
325 if (transitioner) {
326 if (item) {
327 QQuickAnchors *anchors = QQuickItemPrivate::get(item)->_anchors;
328 // TODO: expose QQuickAnchorLine so we can test for other conflicting anchors
329 if (anchors && (anchors->fill() || anchors->centerIn()))
330 qmlWarning(item) << "StackView has detected conflicting anchors. Transitions may not execute properly.";
331 }
332
333 // TODO: add force argument to QQuickItemViewTransitionableItem::prepareTransition()?
334 nextTransitionToSet = true;
335 nextTransitionFromSet = true;
336 nextTransitionFrom += QPointF(1, 1);
337 return QQuickItemViewTransitionableItem::prepareTransition(transitioner, index, viewBounds);
338 }
339 return false;
340}
341
342void QQuickStackElement::startTransition(QQuickItemViewTransitioner *transitioner, QQuickStackView::Status status)
343{
344 setStatus(status);
345 if (transitioner)
346 QQuickItemViewTransitionableItem::startTransition(transitioner, index);
347}
348
349void QQuickStackElement::completeTransition(QQuickTransition *quickTransition)
350{
351 QQuickItemViewTransitionableItem::completeTransition(quickTransition);
352}
353#endif
354
356{
357#if QT_CONFIG(quick_viewtransitions)
358 item = nullptr;
359#endif
360}
361
362QT_END_NAMESPACE
void itemDestroyed(QQuickItem *item) override
void initialize(QV4::ExecutionEngine *v4, RequiredProperties *requiredProperties)
void setVisible(bool visible)
void setStatus(QQuickStackView::Status status)
void incubate(QV4::ExecutionEngine *v4, QObject *object, RequiredProperties *requiredProperties)
QQuickStackIncubator(QQuickStackElement *element)
void setInitialState(QObject *object) override
Called after the object is first created, but before complex property bindings are evaluated and,...