14#include <QtQuick/private/qsgrenderer_p.h>
15#include <QtQuick/private/qsgplaintexture_p.h>
16#include <QtQuick/private/qquickpointerhandler_p.h>
17#include <QtQuick/private/qquickpointerhandler_p_p.h>
18#include <private/qsgrenderloop_p.h>
19#include <private/qsgrhisupport_p.h>
20#include <private/qquickrendercontrol_p.h>
21#include <private/qquickanimatorcontroller_p.h>
22#include <private/qquickprofiler_p.h>
23#include <private/qquicktextinterface_p.h>
25#include <private/qguiapplication_p.h>
27#include <private/qabstractanimation_p.h>
29#include <QtGui/qpainter.h>
30#include <QtGui/qevent.h>
31#include <QtGui/qmatrix4x4.h>
32#include <QtGui/private/qevent_p.h>
33#include <QtGui/private/qpointingdevice_p.h>
34#include <QtCore/qvarlengtharray.h>
35#include <QtCore/qabstractanimation.h>
36#include <QtCore/QLibraryInfo>
37#include <QtCore/QRunnable>
38#include <QtQml/qqmlincubator.h>
39#include <QtQml/qqmlinfo.h>
40#include <QtQml/private/qqmlmetatype_p.h>
42#include <QtQuick/private/qquickpixmap_p.h>
44#include <private/qqmldebugserviceinterfaces_p.h>
45#include <private/qqmldebugconnector_p.h>
46#include <private/qsgdefaultrendercontext_p.h>
47#include <private/qsgsoftwarerenderer_p.h>
49#include <private/qopengl_p.h>
50#include <QOpenGLContext>
52#ifndef QT_NO_DEBUG_STREAM
53#include <private/qdebug_p.h>
55#include <QtCore/qpointer.h>
65Q_LOGGING_CATEGORY(lcQuickWindow,
"qt.quick.window")
67bool QQuickWindowPrivate::defaultAlphaBuffer =
false;
69#if defined(QT_QUICK_DEFAULT_TEXT_RENDER_TYPE)
70QQuickWindow::TextRenderType QQuickWindowPrivate::textRenderType = QQuickWindow::QT_QUICK_DEFAULT_TEXT_RENDER_TYPE;
72QQuickWindow::TextRenderType QQuickWindowPrivate::textRenderType = QQuickWindow::QtTextRendering;
133 int m_incubation_time;
137#if QT_CONFIG(accessibility)
139
140
141
142QAccessibleInterface *QQuickWindow::accessibleRoot()
const
144 return QAccessible::queryAccessibleInterface(
const_cast<QQuickWindow*>(
this));
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
168QQuickRootItem::QQuickRootItem()
172 setFlag(ItemIsViewport);
176void QQuickWindow::exposeEvent(QExposeEvent *)
179 if (d->windowManager)
180 d->windowManager->exposureChanged(
this);
184void QQuickWindow::resizeEvent(QResizeEvent *ev)
188 d->contentItem->setSize(ev->size());
189 if (d->windowManager)
190 d->windowManager->resize(
this);
194void QQuickWindow::showEvent(QShowEvent *)
197 if (d->windowManager)
198 d->windowManager->show(
this);
202void QQuickWindow::hideEvent(QHideEvent *)
205 if (
auto da = d->deliveryAgentPrivate())
206 da->handleWindowHidden(
this);
207 if (d->windowManager)
208 d->windowManager->hide(
this);
212void QQuickWindow::closeEvent(QCloseEvent *e)
214 QQuickCloseEvent qev;
215 qev.setAccepted(e->isAccepted());
217 e->setAccepted(qev.isAccepted());
221void QQuickWindow::focusOutEvent(QFocusEvent *ev)
225 d->contentItem->setFocus(
false, ev->reason());
229void QQuickWindow::focusInEvent(QFocusEvent *ev)
235 d->contentItem->setFocus(
true, ev->reason());
236 if (
auto da = d->deliveryAgentPrivate())
237 da->updateFocusItemTransform();
241static bool transformDirtyOnItemOrAncestor(
const QQuickItem *item)
244 if (QQuickItemPrivate::get(item)->dirtyAttributes & (
245 QQuickItemPrivate::TransformOrigin |
246 QQuickItemPrivate::Transform |
247 QQuickItemPrivate::BasicTransform |
248 QQuickItemPrivate::Position |
249 QQuickItemPrivate::Size |
250 QQuickItemPrivate::ParentChanged |
251 QQuickItemPrivate::Clip)) {
254 item = item->parentItem();
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
284
285
286
287 bool check(QQuickItem *item,
int itemsRemainingBeforeUpdatePolish)
289 if (itemsToPolish.size() > itemsRemainingBeforeUpdatePolish) {
304 QQuickItem *guiltyItem = itemsToPolish.last();
305 qmlWarning(item) <<
"possible QQuickItem::polish() loop";
307 auto typeAndObjectName = [](QQuickItem *item) {
308 QString typeName = QQmlMetaType::prettyTypeName(item);
309 QString objName = item->objectName();
310 if (!objName.isNull())
311 return QLatin1String(
"%1(%2)").arg(typeName, objName);
315 qmlWarning(guiltyItem) << typeAndObjectName(guiltyItem)
316 <<
" called polish() inside updatePolish() of " << typeAndObjectName(item);
327void QQuickWindowPrivate::polishItems()
337 PolishLoopDetector polishLoopDetector(itemsToPolish);
338 while (!itemsToPolish.isEmpty()) {
339 QQuickItem *item = itemsToPolish.takeLast();
340 QQuickItemPrivate *itemPrivate = QQuickItemPrivate::get(item);
341 itemPrivate->polishScheduled =
false;
342 const int itemsRemaining = itemsToPolish.size();
343 itemPrivate->updatePolish();
344 item->updatePolish();
345 if (polishLoopDetector.check(item, itemsRemaining) ==
true)
350 if (QQuickItem *focusItem = q_func()->activeFocusItem()) {
354 const bool isActiveFocusItem = (focusItem == QGuiApplication::focusObject());
355 const bool hasImEnabled = focusItem->inputMethodQuery(Qt::ImEnabled).toBool();
356 if (isActiveFocusItem && hasImEnabled && transformDirtyOnItemOrAncestor(focusItem))
357 deliveryAgentPrivate()->updateFocusItemTransform();
361 if (needsChildWindowStackingOrderUpdate) {
362 updateChildWindowStackingOrder();
363 needsChildWindowStackingOrderUpdate =
false;
368
369
370
371
372
373
374void QQuickWindow::update()
377 if (d->windowManager)
378 d->windowManager->update(
this);
379 else if (d->renderControl)
380 QQuickRenderControlPrivate::get(d->renderControl)->update();
385 if (item->flags() & QQuickItem::ItemHasContents) {
386 QQuickItemPrivate *itemPrivate = QQuickItemPrivate::get(item);
387 itemPrivate->itemChange(QQuickItem::ItemDevicePixelRatioHasChanged, pixelRatio);
390 QList <QQuickItem *> items = item->childItems();
391 for (
int i = 0; i < items.size(); ++i)
392 updatePixelRatioHelper(items.at(i), pixelRatio);
395void QQuickWindow::physicalDpiChanged()
398 const qreal newPixelRatio = effectiveDevicePixelRatio();
399 if (qFuzzyCompare(newPixelRatio, d->lastReportedItemDevicePixelRatio))
401 d->lastReportedItemDevicePixelRatio = newPixelRatio;
403 updatePixelRatioHelper(d->contentItem, newPixelRatio);
407void QQuickWindow::handleFontDatabaseChanged()
410 d->pendingFontUpdate =
true;
415 if (item->flags() & QQuickItem::ItemHasContents) {
419 QList <QQuickItem *> items = item->childItems();
420 for (
int i=0; i<items.size(); ++i)
421 forcePolishHelper(items.at(i));
424void QQuickWindow::handleScreenChanged(QScreen *screen)
432
433
434void QQuickWindowPrivate::forcePolish()
439 forcePolishHelper(contentItem);
444 if (item->flags() & QQuickItem::ItemHasContents)
446 QQuickItemPrivate::get(item)->dirty(QQuickItemPrivate::ChildrenUpdateMask);
448 QList <QQuickItem *> items = item->childItems();
449 for (
int i=0; i<items.size(); ++i)
450 forceUpdate(items.at(i));
457 delete rt.renderTarget;
471 delete sw.paintDevice;
486void QQuickWindowPrivate::invalidateFontData(QQuickItem *item)
488 QQuickTextInterface *textItem = qobject_cast<QQuickTextInterface *>(item);
489 if (textItem !=
nullptr)
490 textItem->invalidate();
492 QList<QQuickItem *> children = item->childItems();
493 for (QQuickItem *child : children)
494 invalidateFontData(child);
497void QQuickWindowPrivate::ensureCustomRenderTarget()
501 if (!redirect.renderTargetDirty)
504 redirect.renderTargetDirty =
false;
506 redirect.rt.reset(rhi, QQuickWindowRenderTarget::ResetFlag::KeepImplicitBuffers);
508 if (!QQuickRenderTargetPrivate::get(&customRenderTarget)->resolve(rhi, &redirect.rt)) {
509 qWarning(
"Failed to set up render target redirection for QQuickWindow");
510 redirect.rt.reset(rhi);
514void QQuickWindowPrivate::setCustomCommandBuffer(QRhiCommandBuffer *cb)
517 redirect.commandBuffer = cb;
520void QQuickWindowPrivate::syncSceneGraph()
524 const bool wasRtDirty = redirect.renderTargetDirty;
525 ensureCustomRenderTarget();
527 QRhiCommandBuffer *cb =
nullptr;
529 if (redirect.commandBuffer)
530 cb = redirect.commandBuffer;
532 cb = swapchain->currentFrameCommandBuffer();
534 context->prepareSync(q->effectiveDevicePixelRatio(), cb, graphicsConfig);
536 animationController->beforeNodeSync();
538 emit q->beforeSynchronizing();
539 runAndClearJobs(&beforeSynchronizingJobs);
541 if (pendingFontUpdate) {
543 invalidateFontData(contentItem);
546 if (Q_UNLIKELY(!renderer)) {
547 forceUpdate(contentItem);
549 QSGRootNode *rootNode =
new QSGRootNode;
550 rootNode->appendChildNode(QQuickItemPrivate::get(contentItem)->itemNode());
551 const bool useDepth = graphicsConfig.isDepthBufferEnabledFor2D();
552 const QSGRendererInterface::RenderMode renderMode = useDepth ? QSGRendererInterface::RenderMode2D
553 : QSGRendererInterface::RenderMode2DNoDepthBuffer;
554 renderer = context->createRenderer(renderMode);
555 renderer->setRootNode(rootNode);
556 }
else if (Q_UNLIKELY(wasRtDirty)
557 && q->rendererInterface()->graphicsApi() == QSGRendererInterface::Software) {
558 auto softwareRenderer =
static_cast<QSGSoftwareRenderer *>(renderer);
559 softwareRenderer->markDirty();
564 animationController->afterNodeSync();
566 renderer->setClearColor(clearColor);
568 renderer->setVisualizationMode(visualizationMode);
570 if (pendingFontUpdate) {
571 context->invalidateGlyphCaches();
572 pendingFontUpdate =
false;
575 emit q->afterSynchronizing();
576 runAndClearJobs(&afterSynchronizingJobs);
579void QQuickWindowPrivate::emitBeforeRenderPassRecording(
void *ud)
581 QQuickWindow *w =
reinterpret_cast<QQuickWindow *>(ud);
582 emit w->beforeRenderPassRecording();
585void QQuickWindowPrivate::emitAfterRenderPassRecording(
void *ud)
587 QQuickWindow *w =
reinterpret_cast<QQuickWindow *>(ud);
588 emit w->afterRenderPassRecording();
591int QQuickWindowPrivate::multiViewCount()
594 ensureCustomRenderTarget();
595 if (redirect.rt.rt.renderTarget)
596 return redirect.rt.rt.multiViewCount;
606QRhiRenderTarget *QQuickWindowPrivate::activeCustomRhiRenderTarget()
609 ensureCustomRenderTarget();
610 return redirect.rt.rt.renderTarget;
615void QQuickWindowPrivate::renderSceneGraph()
621 ensureCustomRenderTarget();
623 QSGRenderTarget sgRenderTarget;
625 QRhiRenderTarget *rt;
626 QRhiRenderPassDescriptor *rp;
627 QRhiCommandBuffer *cb;
628 if (redirect.rt.rt.renderTarget) {
629 rt = redirect.rt.rt.renderTarget;
630 rp = rt->renderPassDescriptor();
632 qWarning(
"Custom render target is set but no renderpass descriptor has been provided.");
635 cb = redirect.commandBuffer;
637 qWarning(
"Custom render target is set but no command buffer has been provided.");
642 qWarning(
"QQuickWindow: No render target (neither swapchain nor custom target was provided)");
645 rt = swapchain->currentFrameRenderTarget();
646 rp = rpDescForSwapchain;
647 cb = swapchain->currentFrameCommandBuffer();
649 sgRenderTarget = QSGRenderTarget(rt, rp, cb);
650 sgRenderTarget.multiViewCount = multiViewCount();
652 sgRenderTarget = QSGRenderTarget(redirect.rt.sw.paintDevice);
655 context->beginNextFrame(renderer,
657 emitBeforeRenderPassRecording,
658 emitAfterRenderPassRecording,
661 animationController->advance();
662 emit q->beforeRendering();
663 runAndClearJobs(&beforeRenderingJobs);
665 const qreal devicePixelRatio = q->effectiveDevicePixelRatio();
667 if (redirect.rt.rt.renderTarget)
668 pixelSize = redirect.rt.rt.renderTarget->pixelSize();
669 else if (redirect.rt.sw.paintDevice)
670 pixelSize = QSize(redirect.rt.sw.paintDevice->width(), redirect.rt.sw.paintDevice->height());
672 pixelSize = swapchain->currentPixelSize();
674 pixelSize = q->size() * devicePixelRatio;
676 renderer->setDevicePixelRatio(devicePixelRatio);
677 renderer->setDeviceRect(QRect(QPoint(0, 0), pixelSize));
678 renderer->setViewportRect(QRect(QPoint(0, 0), pixelSize));
680 QSGAbstractRenderer::MatrixTransformFlags matrixFlags;
681 bool flipY = rhi ? !rhi->isYUpInNDC() :
false;
682 if (!customRenderTarget.isNull() && customRenderTarget.mirrorVertically())
685 matrixFlags |= QSGAbstractRenderer::MatrixTransformFlipY;
687 const QRectF rect(QPointF(0, 0), pixelSize / devicePixelRatio);
688 renderer->setProjectionMatrixToRect(rect, matrixFlags, rhi && !rhi->isYUpInNDC());
690 context->renderNextFrame(renderer);
692 emit q->afterRendering();
693 runAndClearJobs(&afterRenderingJobs);
695 context->endNextFrame(renderer);
697 if (renderer && renderer->hasVisualizationModeWithContinuousUpdate()) {
701 QCoreApplication::postEvent(q,
new QEvent(QEvent::Type(FullUpdateRequest)));
705QQuickWindowPrivate::QQuickWindowPrivate()
706 : contentItem(
nullptr)
707 , dirtyItemList(
nullptr)
708 , lastReportedItemDevicePixelRatio(0)
711 , windowManager(
nullptr)
712 , renderControl(
nullptr)
713 , clearColor(Qt::white)
714 , persistentGraphics(
true)
715 , persistentSceneGraph(
true)
716 , inDestructor(
false)
717 , incubationController(
nullptr)
718 , hasActiveSwapchain(
false)
719 , hasRenderableSwapchain(
false)
720 , swapchainJustBecameRenderable(
false)
721 , updatesEnabled(
true)
725QQuickWindowPrivate::~QQuickWindowPrivate()
728 redirect.rt.reset(rhi);
729 if (QQmlInspectorService *service = QQmlDebugConnector::service<QQmlInspectorService>())
730 service->removeWindow(q_func());
731 deliveryAgent =
nullptr;
734void QQuickWindowPrivate::setPalette(QQuickPalette* palette)
736 if (windowPaletteRef == palette)
739 if (windowPaletteRef)
740 disconnect(windowPaletteRef, &QQuickPalette::changed,
this, &QQuickWindowPrivate::updateWindowPalette);
741 windowPaletteRef = palette;
742 updateWindowPalette();
743 if (windowPaletteRef)
744 connect(windowPaletteRef, &QQuickPalette::changed,
this, &QQuickWindowPrivate::updateWindowPalette);
747void QQuickWindowPrivate::updateWindowPalette()
749 QQuickPaletteProviderPrivateBase::setPalette(windowPaletteRef);
752void QQuickWindowPrivate::updateChildrenPalettes(
const QPalette &parentPalette)
755 if (
auto root = q->contentItem()) {
756 for (
auto &&child: root->childItems()) {
757 QQuickItemPrivate::get(child)->inheritPalette(parentPalette);
762void QQuickWindowPrivate::init(QQuickWindow *c, QQuickRenderControl *control)
769 contentItem =
new QQuickRootItem;
770 contentItem->setObjectName(q->objectName());
771 QQml_setParent_noEvent(contentItem, c);
772 QQmlEngine::setObjectOwnership(contentItem, QQmlEngine::CppOwnership);
773 QQuickItemPrivate *contentItemPrivate = QQuickItemPrivate::get(contentItem);
774 contentItemPrivate->window = q;
775 contentItemPrivate->windowRefCount = 1;
776 contentItemPrivate->flags |= QQuickItem::ItemIsFocusScope;
777 contentItem->setSize(q->size());
778 deliveryAgent =
new QQuickDeliveryAgent(contentItem);
780 visualizationMode = qgetenv(
"QSG_VISUALIZE");
781 renderControl = control;
783 QQuickRenderControlPrivate::get(renderControl)->window = q;
786 windowManager = QSGRenderLoop::instance();
788 Q_ASSERT(windowManager || renderControl);
790 QObject::connect(
static_cast<QGuiApplication *>(QGuiApplication::instance()),
791 &QGuiApplication::fontDatabaseChanged,
793 &QQuickWindow::handleFontDatabaseChanged);
796 lastReportedItemDevicePixelRatio = q->effectiveDevicePixelRatio();
801 QQuickRenderControlPrivate *renderControlPriv = QQuickRenderControlPrivate::get(renderControl);
802 sg = renderControlPriv->sg;
803 context = renderControlPriv->rc;
805 windowManager->addWindow(q);
806 sg = windowManager->sceneGraphContext();
807 context = windowManager->createRenderContext(sg);
810 q->setSurfaceType(windowManager ? windowManager->windowSurfaceType() : QSurface::OpenGLSurface);
811 q->setFormat(sg->defaultSurfaceFormat());
817 animationController.reset(
new QQuickAnimatorController(q));
819 QObject::connect(context, &QSGRenderContext::initialized, q, &QQuickWindow::sceneGraphInitialized, Qt::DirectConnection);
820 QObject::connect(context, &QSGRenderContext::invalidated, q, &QQuickWindow::sceneGraphInvalidated, Qt::DirectConnection);
821 QObject::connect(context, &QSGRenderContext::invalidated, q, &QQuickWindow::cleanupSceneGraph, Qt::DirectConnection);
823 QObject::connect(q, &QQuickWindow::focusObjectChanged, q, &QQuickWindow::activeFocusItemChanged);
824 QObject::connect(q, &QQuickWindow::screenChanged, q, &QQuickWindow::handleScreenChanged);
825 QObject::connect(qApp, &QGuiApplication::applicationStateChanged, q, &QQuickWindow::handleApplicationStateChanged);
826 QObject::connect(q, &QQuickWindow::frameSwapped, q, &QQuickWindow::runJobsAfterSwap, Qt::DirectConnection);
828 if (QQmlInspectorService *service = QQmlDebugConnector::service<QQmlInspectorService>())
829 service->addWindow(q);
832void QQuickWindow::handleApplicationStateChanged(Qt::ApplicationState state)
835 if (state != Qt::ApplicationActive && d->contentItem) {
836 auto da = d->deliveryAgentPrivate();
838 da->handleWindowDeactivate(
this);
843
844
845
847QQmlListProperty<QObject> QQuickWindowPrivate::data()
849 QQmlListProperty<QObject> ret;
851 ret.object = q_func();
852 ret.append = QQuickWindowPrivate::data_append;
853 ret.count = QQuickWindowPrivate::data_count;
854 ret.at = QQuickWindowPrivate::data_at;
855 ret.clear = QQuickWindowPrivate::data_clear;
857 ret.removeLast = QQuickWindowPrivate::data_removeLast;
862void QQuickWindowPrivate::dirtyItem(QQuickItem *item)
866 QQuickItemPrivate *itemPriv = QQuickItemPrivate::get(item);
867 if (itemPriv->dirtyAttributes & QQuickItemPrivate::ChildrenStackingChanged)
868 needsChildWindowStackingOrderUpdate =
true;
874
875
876
877QQuickItem *QQuickWindow::mouseGrabberItem()
const
879 Q_D(
const QQuickWindow);
880 auto da =
const_cast<QQuickWindowPrivate *>(d)->deliveryAgentPrivate();
884 if (
auto epd = da->mousePointData())
885 return qmlobject_cast<QQuickItem *>(epd->exclusiveGrabber);
887 if (Q_LIKELY(d->deliveryAgentPrivate()->eventsInDelivery.isEmpty()))
889 qCDebug(lcMouse,
"mouse grabber ambiguous: no event is currently being delivered");
893 return qmlobject_cast<QQuickItem *>(QPointingDevicePrivate::get(QPointingDevice::primaryPointingDevice())->
894 firstPointExclusiveGrabber());
897void QQuickWindowPrivate::cleanup(QSGNode *n)
901 Q_ASSERT(!cleanupNodeList.contains(n));
902 cleanupNodeList.append(n);
907
908
909
910
911
912
913
914
915
916
917
918
919
920
921
922
923
924
925
926
927
928
929
930
931
932
933
934
935
936
937
938
939
940
941
942
943
944
945
946
947
948
949
950
951
952
953
954
955
956
957
958
959
960
961
962
963
964
967
968
969
970
971
972
973
974
975
976
977
978
979
980
981
982
983
984
985
986
987
988
989
990
991
992
993
994
995
996
997
998
999
1000
1001
1002
1003
1004
1005
1006
1007
1008
1009
1010
1011
1012
1013
1014
1015
1016
1017
1018
1019
1020
1021
1022
1023
1024
1025
1026
1027
1028
1029
1030
1031
1032
1033
1034
1035
1036
1037
1038
1039
1040
1041
1042
1043
1044
1045
1046
1047
1048
1049
1050
1051
1052
1053
1054
1055
1056
1057
1058
1059
1060
1061
1062
1063
1064
1065
1066
1067
1068
1069
1070
1071
1072
1073
1074
1075
1076
1077
1078
1079
1080
1081
1082
1083
1084
1085
1086
1087
1088
1089
1090
1091
1092
1093
1094
1095
1096
1097
1098
1099
1100
1101
1102
1103
1104
1105
1106
1107
1108
1109
1110
1111
1112
1113
1114
1115
1116
1117
1118
1119
1120
1121
1122
1123
1124
1125
1126
1127
1130
1131
1132QQuickWindow::QQuickWindow(QWindow *parent)
1133 : QQuickWindow(*
new QQuickWindowPrivate, parent)
1140
1141
1142QQuickWindow::QQuickWindow(QQuickWindowPrivate &dd, QWindow *parent)
1143 : QWindow(dd, parent)
1150
1151
1152
1153
1154
1155
1156QQuickWindow::QQuickWindow(QQuickRenderControl *control)
1157 : QWindow(*(
new QQuickWindowPrivate),
nullptr)
1160 d->init(
this, control);
1164
1165
1166QQuickWindow::QQuickWindow(QQuickWindowPrivate &dd, QQuickRenderControl *control)
1167 : QWindow(dd,
nullptr)
1170 d->init(
this, control);
1174
1175
1176QQuickWindow::~QQuickWindow()
1179 d->inDestructor =
true;
1180 if (d->renderControl) {
1181 QQuickRenderControlPrivate::get(d->renderControl)->windowDestroyed();
1182 }
else if (d->windowManager) {
1183 d->windowManager->removeWindow(
this);
1184 d->windowManager->windowDestroyed(
this);
1187 disconnect(
this, &QQuickWindow::focusObjectChanged,
this, &QQuickWindow::activeFocusItemChanged);
1188 disconnect(
this, &QQuickWindow::screenChanged,
this, &QQuickWindow::handleScreenChanged);
1189 disconnect(qApp, &QGuiApplication::applicationStateChanged,
this, &QQuickWindow::handleApplicationStateChanged);
1190 disconnect(
this, &QQuickWindow::frameSwapped,
this, &QQuickWindow::runJobsAfterSwap);
1192 delete d->incubationController; d->incubationController =
nullptr;
1193 QQuickRootItem *root = d->contentItem;
1194 d->contentItem =
nullptr;
1195 root->setParent(
nullptr);
1197 d->deliveryAgent =
nullptr;
1201 const std::lock_guard locker(d->renderJobMutex);
1202 qDeleteAll(std::exchange(d->beforeSynchronizingJobs, {}));
1203 qDeleteAll(std::exchange(d->afterSynchronizingJobs, {}));
1204 qDeleteAll(std::exchange(d->beforeRenderingJobs, {}));
1205 qDeleteAll(std::exchange(d->afterRenderingJobs, {}));;
1206 qDeleteAll(std::exchange(d->afterSwapJobs, {}));
1214 QQuickPixmap::purgeCache();
1217#if QT_CONFIG(quick_shadereffect)
1218void qtquick_shadereffect_purge_gui_thread_shader_cache();
1222
1223
1224
1225
1226
1227
1228
1229
1230
1231
1232
1233
1234
1235
1236
1237
1238
1239
1240
1241
1242
1243
1244
1245
1247void QQuickWindow::releaseResources()
1250 if (d->windowManager)
1251 d->windowManager->releaseResources(
this);
1252 QQuickPixmap::purgeCache();
1253#if QT_CONFIG(quick_shadereffect)
1254 qtquick_shadereffect_purge_gui_thread_shader_cache();
1261
1262
1263
1264
1265
1266
1267
1268
1269
1270
1271
1272
1273
1274
1275
1276
1277
1278
1279
1280
1281
1282
1283
1284
1285
1286
1288void QQuickWindow::setPersistentGraphics(
bool persistent)
1291 d->persistentGraphics = persistent;
1297
1298
1299
1300
1301
1302
1303
1305bool QQuickWindow::isPersistentGraphics()
const
1307 Q_D(
const QQuickWindow);
1308 return d->persistentGraphics;
1314
1315
1316
1317
1318
1319
1320
1321
1322
1323
1324
1325
1326
1327
1328
1329
1330
1331
1332
1333
1335void QQuickWindow::setPersistentSceneGraph(
bool persistent)
1338 d->persistentSceneGraph = persistent;
1344
1345
1346
1347
1348
1349
1351bool QQuickWindow::isPersistentSceneGraph()
const
1353 Q_D(
const QQuickWindow);
1354 return d->persistentSceneGraph;
1358
1359
1360
1361
1362
1363
1364
1367
1368
1369
1370
1371
1372
1373
1374QQuickItem *QQuickWindow::contentItem()
const
1376 Q_D(
const QQuickWindow);
1378 return d->contentItem;
1382
1383
1384
1385
1386
1387
1388
1389QQuickItem *QQuickWindow::activeFocusItem()
const
1391 Q_D(
const QQuickWindow);
1392 auto da = d->deliveryAgentPrivate();
1394 return da->activeFocusItem;
1398
1399
1400
1401QObject *QQuickWindow::focusObject()
const
1403 Q_D(
const QQuickWindow);
1404 auto da = d->deliveryAgentPrivate();
1406 if (!d->inDestructor && da->activeFocusItem)
1407 return da->activeFocusItem;
1408 return const_cast<QQuickWindow*>(
this);
1412bool QQuickWindow::event(QEvent *event)
1417 QQuickDeliveryAgent *da = d->deliveryAgent;
1418 if (event->isPointerEvent()) {
1420
1421
1422
1423
1424
1425
1426
1427
1428
1429 if (d->windowEventDispatch)
1432 const bool wasAccepted = event->isAccepted();
1433 QScopedValueRollback windowEventDispatchGuard(d->windowEventDispatch,
true);
1434 qCDebug(lcPtr) <<
"dispatching to window functions in case of override" << event;
1435 QWindow::event(event);
1436 if (event->isAccepted() && !wasAccepted)
1440
1441
1442
1443
1444
1445
1446 auto pe =
static_cast<QPointerEvent *>(event);
1447 if (QQuickDeliveryAgentPrivate::isTouchEvent(pe))
1452
1453
1454
1455
1456
1457
1458
1459
1460
1461
1462
1463
1464 if (pe->pointCount()) {
1465 const bool synthMouse = QQuickDeliveryAgentPrivate::isSynthMouse(pe);
1466 if (QQuickDeliveryAgentPrivate::subsceneAgentsExist) {
1470 QFlatMap<QQuickDeliveryAgent*, QList<QEventPoint>> deliveryAgentsNeedingPoints;
1471 QEventPoint::States eventStates;
1473 auto insert = [&](QQuickDeliveryAgent *ptda,
const QEventPoint &pt) {
1474 if (pt.state() == QEventPoint::Pressed && !synthMouse)
1475 pe->clearPassiveGrabbers(pt);
1476 auto &ptList = deliveryAgentsNeedingPoints[ptda];
1477 auto idEquals = [](
auto id) {
return [id] (
const auto &e) {
return e.id() == id; }; };
1478 if (std::none_of(ptList.cbegin(), ptList.cend(), idEquals(pt.id())))
1482 for (
const auto &pt : pe->points()) {
1483 eventStates |= pt.state();
1484 auto epd = QPointingDevicePrivate::get(
const_cast<QPointingDevice*>(pe->pointingDevice()))->queryPointById(pt.id());
1486 bool foundAgent =
false;
1487 if (!epd->exclusiveGrabber.isNull() && !epd->exclusiveGrabberContext.isNull()) {
1488 if (
auto ptda = qobject_cast<QQuickDeliveryAgent *>(epd->exclusiveGrabberContext.data())) {
1490 qCDebug(lcPtr) << pe->type() <<
"point" << pt.id() << pt.state()
1491 <<
"@" << pt.scenePosition() <<
"will be re-delivered via known grabbing agent" << ptda <<
"to" << epd->exclusiveGrabber.data();
1495 for (
auto pgda : epd->passiveGrabbersContext) {
1496 if (
auto ptda = qobject_cast<QQuickDeliveryAgent *>(pgda.data())) {
1498 qCDebug(lcPtr) << pe->type() <<
"point" << pt.id() << pt.state()
1499 <<
"@" << pt.scenePosition() <<
"will be re-delivered via known passive-grabbing agent" << ptda;
1507 for (
auto daAndPoints : deliveryAgentsNeedingPoints) {
1508 if (pe->pointCount() > 1) {
1509 Q_ASSERT(QQuickDeliveryAgentPrivate::isTouchEvent(pe));
1511 QEvent::Type eventType = pe->type();
1512 switch (eventStates) {
1513 case QEventPoint::State::Pressed:
1514 eventType = QEvent::TouchBegin;
1516 case QEventPoint::State::Released:
1517 eventType = QEvent::TouchEnd;
1520 eventType = QEvent::TouchUpdate;
1524 QMutableTouchEvent te(eventType, pe->pointingDevice(), pe->modifiers(), daAndPoints.second);
1525 te.setTimestamp(pe->timestamp());
1527 qCDebug(lcTouch) << daAndPoints.first <<
"shall now receive" << &te;
1528 ret = daAndPoints.first->event(&te) || ret;
1530 qCDebug(lcPtr) << daAndPoints.first <<
"shall now receive" << pe;
1531 ret = daAndPoints.first->event(pe) || ret;
1536 d->deliveryAgentPrivate()->clearGrabbers(pe);
1539 }
else if (!synthMouse) {
1542 for (
const auto &pt : pe->points()) {
1543 if (pt.state() == QEventPoint::Pressed)
1544 pe->clearPassiveGrabbers(pt);
1553 qCDebug(lcHoverTrace) <<
this <<
"some sort of event" << event;
1554 bool ret = (da && da->event(event));
1556 d->deliveryAgentPrivate()->clearGrabbers(pe);
1558 if (pe->type() == QEvent::MouseButtonPress || pe->type() == QEvent::MouseButtonRelease) {
1561 d->maybeSynthesizeContextMenuEvent(
static_cast<QMouseEvent *>(pe));
1566 }
else if (event->isInputEvent()) {
1567 if (da && da->event(event))
1571 switch (event->type()) {
1573 case QEvent::FocusAboutToChange:
1576 case QEvent::InputMethod:
1577 case QEvent::InputMethodQuery:
1578#if QT_CONFIG(quick_draganddrop)
1579 case QEvent::DragEnter:
1580 case QEvent::DragLeave:
1581 case QEvent::DragMove:
1584 if (d->inDestructor)
1586 if (da && da->event(event))
1589 case QEvent::LanguageChange:
1590 case QEvent::LocaleChange:
1592 QCoreApplication::sendEvent(d->contentItem, event);
1594 case QEvent::UpdateRequest:
1595 if (d->windowManager)
1596 d->windowManager->handleUpdateRequest(
this);
1598 case QEvent::PlatformSurface:
1599 if ((
static_cast<QPlatformSurfaceEvent *>(event))->surfaceEventType() == QPlatformSurfaceEvent::SurfaceAboutToBeDestroyed) {
1602 if (d->windowManager)
1603 d->windowManager->hide(
this);
1606 case QEvent::WindowDeactivate:
1607 if (
auto da = d->deliveryAgentPrivate())
1608 da->handleWindowDeactivate(
this);
1610 case QEvent::WindowActivate:
1612 QCoreApplication::sendEvent(d->contentItem, event);
1614 case QEvent::ApplicationPaletteChange:
1615 d->inheritPalette(QGuiApplication::palette());
1617 QCoreApplication::sendEvent(d->contentItem, event);
1619 case QEvent::DevicePixelRatioChange:
1620 physicalDpiChanged();
1622 case QEvent::SafeAreaMarginsChange:
1623 QQuickSafeArea::updateSafeAreasRecursively(d->contentItem);
1625 case QEvent::ChildWindowAdded: {
1626 auto *childEvent =
static_cast<QChildWindowEvent*>(event);
1627 auto *childWindow = childEvent->child();
1628 qCDebug(lcQuickWindow) <<
"Child window" << childWindow <<
"added to" <<
this;
1629 if (childWindow->handle()) {
1635 d->updateChildWindowStackingOrder();
1637 qCDebug(lcQuickWindow) <<
"No platform window yet."
1638 <<
"Deferring child window stacking until surface creation";
1646 if (event->type() == QEvent::Type(QQuickWindowPrivate::FullUpdateRequest))
1648 else if (event->type() == QEvent::Type(QQuickWindowPrivate::TriggerContextCreationFailure))
1649 d->windowManager->handleContextCreationFailure(
this);
1651 if (event->isPointerEvent())
1654 return QWindow::event(event);
1657void QQuickWindowPrivate::maybeSynthesizeContextMenuEvent(QMouseEvent *event)
1666 if (windowEventDispatch || !rmbContextMenuEventEnabled)
1669 QWindowPrivate::maybeSynthesizeContextMenuEvent(event);
1672void QQuickWindowPrivate::updateChildWindowStackingOrder(QQuickItem *item)
1677 qCDebug(lcQuickWindow) <<
"Updating child window stacking order for" << q;
1680 auto *itemPrivate = QQuickItemPrivate::get(item);
1681 const auto paintOrderChildItems = itemPrivate->paintOrderChildItems();
1682 for (
auto *child : paintOrderChildItems) {
1683 if (
auto *windowContainer = qobject_cast<QQuickWindowContainer*>(child)) {
1684 auto *window = windowContainer->containedWindow();
1686 qCDebug(lcQuickWindow) << windowContainer <<
"has no contained window yet";
1689 if (window->parent() != q) {
1690 qCDebug(lcQuickWindow) << window <<
"is not yet child of this window";
1693 qCDebug(lcQuickWindow) <<
"Raising" << window <<
"owned by" << windowContainer;
1697 updateChildWindowStackingOrder(child);
1702void QQuickWindow::keyPressEvent(QKeyEvent *e)
1705 if (d->windowEventDispatch)
1707 auto da = d->deliveryAgentPrivate();
1709 da->deliverKeyEvent(e);
1713void QQuickWindow::keyReleaseEvent(QKeyEvent *e)
1716 if (d->windowEventDispatch)
1718 auto da = d->deliveryAgentPrivate();
1720 da->deliverKeyEvent(e);
1723#if QT_CONFIG(wheelevent)
1725void QQuickWindow::wheelEvent(QWheelEvent *event)
1728 if (d->windowEventDispatch)
1730 auto da = d->deliveryAgentPrivate();
1732 da->deliverSinglePointEventUntilAccepted(event);
1736#if QT_CONFIG(tabletevent)
1738void QQuickWindow::tabletEvent(QTabletEvent *event)
1741 if (d->windowEventDispatch)
1743 auto da = d->deliveryAgentPrivate();
1745 da->deliverPointerEvent(event);
1750void QQuickWindow::mousePressEvent(QMouseEvent *event)
1753 if (d->windowEventDispatch)
1755 auto da = d->deliveryAgentPrivate();
1757 da->handleMouseEvent(event);
1760void QQuickWindow::mouseMoveEvent(QMouseEvent *event)
1763 if (d->windowEventDispatch)
1765 auto da = d->deliveryAgentPrivate();
1767 da->handleMouseEvent(event);
1770void QQuickWindow::mouseDoubleClickEvent(QMouseEvent *event)
1773 if (d->windowEventDispatch)
1775 auto da = d->deliveryAgentPrivate();
1777 da->handleMouseEvent(event);
1780void QQuickWindow::mouseReleaseEvent(QMouseEvent *event)
1783 if (d->windowEventDispatch)
1785 auto da = d->deliveryAgentPrivate();
1787 da->handleMouseEvent(event);
1790#if QT_CONFIG(cursor)
1791void QQuickWindowPrivate::updateCursor(
const QPointF &scenePos, QQuickItem *rootItem)
1795 rootItem = contentItem;
1796 auto cursorItemAndHandler = findCursorItemAndHandler(rootItem, scenePos);
1797 if (cursorItem != cursorItemAndHandler.first || cursorHandler != cursorItemAndHandler.second ||
1798 (cursorItemAndHandler.second && QQuickPointerHandlerPrivate::get(cursorItemAndHandler.second)->cursorDirty)) {
1799 QWindow *renderWindow = QQuickRenderControl::renderWindowFor(q);
1800 QWindow *window = renderWindow ? renderWindow : q;
1801 cursorItem = cursorItemAndHandler.first;
1802 cursorHandler = cursorItemAndHandler.second;
1804 QQuickPointerHandlerPrivate::get(cursorItemAndHandler.second)->cursorDirty =
false;
1806 const auto cursor = QQuickItemPrivate::get(cursorItem)->effectiveCursor(cursorHandler);
1807 qCDebug(lcHoverTrace) <<
"setting cursor" << cursor <<
"from" << cursorHandler <<
"or" << cursorItem;
1808 window->setCursor(cursor);
1810 qCDebug(lcHoverTrace) <<
"unsetting cursor";
1811 window->unsetCursor();
1816QPair<QQuickItem*, QQuickPointerHandler*> QQuickWindowPrivate::findCursorItemAndHandler(QQuickItem *item,
const QPointF &scenePos)
const
1818 QQuickItemPrivate *itemPrivate = QQuickItemPrivate::get(item);
1819 if (itemPrivate->flags & QQuickItem::ItemClipsChildrenToShape) {
1820 QPointF p = item->mapFromScene(scenePos);
1821 if (!item->contains(p))
1822 return {
nullptr,
nullptr};
1825 if (itemPrivate->subtreeCursorEnabled) {
1826 QList<QQuickItem *> children = itemPrivate->paintOrderChildItems();
1827 for (
int ii = children.size() - 1; ii >= 0; --ii) {
1828 QQuickItem *child = children.at(ii);
1829 if (!child->isVisible() || !child->isEnabled() || QQuickItemPrivate::get(child)->culled)
1831 auto ret = findCursorItemAndHandler(child, scenePos);
1835 if (itemPrivate->hasCursorHandler) {
1836 if (
auto handler = itemPrivate->effectiveCursorHandler()) {
1837 if (handler->parentContains(scenePos))
1838 return {item, handler};
1841 if (itemPrivate->hasCursor) {
1842 QPointF p = item->mapFromScene(scenePos);
1843 if (item->contains(p))
1844 return {item,
nullptr};
1848 return {
nullptr,
nullptr};
1852void QQuickWindowPrivate::clearFocusObject()
1854 if (
auto da = deliveryAgentPrivate())
1855 da->clearFocusObject();
1858void QQuickWindowPrivate::setFocusToTarget(FocusTarget target, Qt::FocusReason reason)
1863 QQuickItem *newFocusItem =
nullptr;
1865 case FocusTarget::First:
1866 case FocusTarget::Last: {
1867 const bool forward = (target == FocusTarget::First);
1868 newFocusItem = QQuickItemPrivate::nextPrevItemInTabFocusChain(contentItem, forward);
1870 const auto *itemPriv = QQuickItemPrivate::get(newFocusItem);
1871 if (itemPriv->subFocusItem && itemPriv->flags & QQuickItem::ItemIsFocusScope)
1872 deliveryAgentPrivate()->clearFocusInScope(newFocusItem, itemPriv->subFocusItem, reason);
1876 case FocusTarget::Next:
1877 case FocusTarget::Prev: {
1878 const auto da = deliveryAgentPrivate();
1880 QQuickItem *focusItem = da->focusTargetItem() ? da->focusTargetItem() : contentItem;
1881 bool forward = (target == FocusTarget::Next);
1882 newFocusItem = QQuickItemPrivate::nextPrevItemInTabFocusChain(focusItem, forward);
1890 newFocusItem->forceActiveFocus(reason);
1894
1895
1896
1897
1898
1899
1900
1901
1902
1903
1904
1905
1906
1907
1908
1909
1910
1911
1912
1913
1914
1915
1917void QQuickWindowPrivate::data_append(QQmlListProperty<QObject> *property, QObject *o)
1921 QQuickWindow *that =
static_cast<QQuickWindow *>(property->object);
1922 QQmlListProperty<QObject> itemProperty = QQuickItemPrivate::get(that->contentItem())->data();
1923 itemProperty.append(&itemProperty, o);
1926qsizetype QQuickWindowPrivate::data_count(QQmlListProperty<QObject> *property)
1928 QQuickWindow *win =
static_cast<QQuickWindow*>(property->object);
1929 if (!win || !win->contentItem() || !QQuickItemPrivate::get(win->contentItem())->data().count)
1931 QQmlListProperty<QObject> itemProperty = QQuickItemPrivate::get(win->contentItem())->data();
1932 return itemProperty.count(&itemProperty);
1935QObject *QQuickWindowPrivate::data_at(QQmlListProperty<QObject> *property, qsizetype i)
1937 QQuickWindow *win =
static_cast<QQuickWindow*>(property->object);
1938 QQmlListProperty<QObject> itemProperty = QQuickItemPrivate::get(win->contentItem())->data();
1939 return itemProperty.at(&itemProperty, i);
1942void QQuickWindowPrivate::data_clear(QQmlListProperty<QObject> *property)
1944 QQuickWindow *win =
static_cast<QQuickWindow*>(property->object);
1945 QQmlListProperty<QObject> itemProperty = QQuickItemPrivate::get(win->contentItem())->data();
1946 itemProperty.clear(&itemProperty);
1949void QQuickWindowPrivate::data_removeLast(QQmlListProperty<QObject> *property)
1951 QQuickWindow *win =
static_cast<QQuickWindow*>(property->object);
1952 QQmlListProperty<QObject> itemProperty = QQuickItemPrivate::get(win->contentItem())->data();
1953 itemProperty.removeLast(&itemProperty);
1956bool QQuickWindowPrivate::isRenderable()
const
1958 Q_Q(
const QQuickWindow);
1959 return ((q->isExposed() && q->isVisible())) && q->geometry().isValid();
1962void QQuickWindowPrivate::rhiCreationFailureMessage(
const QString &backendName,
1963 QString *translatedMessage,
1964 QString *untranslatedMessage)
1966 const char msg[] = QT_TRANSLATE_NOOP(
"QQuickWindow",
1967 "Failed to initialize graphics backend for %1.");
1968 *translatedMessage = QQuickWindow::tr(msg).arg(backendName);
1969 *untranslatedMessage = QString::fromLatin1(msg).arg(backendName);
1972void QQuickWindowPrivate::cleanupNodes()
1974 qDeleteAll(std::exchange(cleanupNodeList, {}));
1977void QQuickWindowPrivate::cleanupNodesOnShutdown(QQuickItem *item)
1979 QQuickItemPrivate *p = QQuickItemPrivate::get(item);
1980 if (p->itemNodeInstance) {
1981 delete p->itemNodeInstance;
1982 p->itemNodeInstance =
nullptr;
1984 if (p->extra.isAllocated()) {
1985 p->extra->opacityNode =
nullptr;
1986 p->extra->clipNode =
nullptr;
1987 p->extra->rootNode =
nullptr;
1990 p->paintNode =
nullptr;
1992 p->dirty(QQuickItemPrivate::Window);
1996 if (p->flags & QQuickItem::ItemHasContents) {
1997 const QMetaObject *mo = item->metaObject();
1998 int index = mo->indexOfSlot(
"invalidateSceneGraph()");
2000 const QMetaMethod &method = mo->method(index);
2002 if (strstr(method.enclosingMetaObject()->className(),
"_QML_") ==
nullptr)
2003 method.invoke(item, Qt::DirectConnection);
2007 for (
int ii = 0; ii < p->childItems.size(); ++ii)
2008 cleanupNodesOnShutdown(p->childItems.at(ii));
2012void QQuickWindowPrivate::cleanupNodesOnShutdown()
2016 cleanupNodesOnShutdown(contentItem);
2017 for (QSet<QQuickItem *>::const_iterator it = parentlessItems.begin(), cend = parentlessItems.end(); it != cend; ++it)
2018 cleanupNodesOnShutdown(*it);
2019 animationController->windowNodesDestroyed();
2020 q->cleanupSceneGraph();
2023void QQuickWindowPrivate::updateDirtyNodes()
2025 qCDebug(lcDirty) <<
"QQuickWindowPrivate::updateDirtyNodes():";
2029 QQuickItem *updateList = dirtyItemList;
2030 dirtyItemList =
nullptr;
2031 if (updateList) QQuickItemPrivate::get(updateList)->prevDirtyItem = &updateList;
2033 while (updateList) {
2034 QQuickItem *item = updateList;
2035 QQuickItemPrivate *itemPriv = QQuickItemPrivate::get(item);
2036 itemPriv->removeFromDirtyList();
2038 qCDebug(lcDirty) <<
" QSGNode:" << item << qPrintable(itemPriv->dirtyToString());
2039 updateDirtyNode(item);
2045 const QList<QQuickItem *> childItems = d->paintOrderChildItems();
2046 QQuickItem *before =
nullptr;
2047 for (
int i=0; i<childItems.size(); ++i) {
2048 QQuickItemPrivate *dd = QQuickItemPrivate::get(childItems.at(i));
2050 if (dd->z() < 0 && (dd->explicitVisible || (dd->extra.isAllocated() && dd->extra->effectRefCount)))
2051 before = childItems.at(i);
2055 return Q_UNLIKELY(before) ? QQuickItemPrivate::get(before)->itemNode() :
nullptr;
2058static QSGNode *
fetchNextNode(QQuickItemPrivate *itemPriv,
int &ii,
bool &returnedPaintNode)
2060 QList<QQuickItem *> orderedChildren = itemPriv->paintOrderChildItems();
2062 for (; ii < orderedChildren.size() && orderedChildren.at(ii)->z() < 0; ++ii) {
2063 QQuickItemPrivate *childPrivate = QQuickItemPrivate::get(orderedChildren.at(ii));
2064 if (!childPrivate->explicitVisible &&
2065 (!childPrivate->extra.isAllocated() || !childPrivate->extra->effectRefCount))
2069 return childPrivate->itemNode();
2072 if (itemPriv->paintNode && !returnedPaintNode) {
2073 returnedPaintNode =
true;
2074 return itemPriv->paintNode;
2077 for (; ii < orderedChildren.size(); ++ii) {
2078 QQuickItemPrivate *childPrivate = QQuickItemPrivate::get(orderedChildren.at(ii));
2079 if (!childPrivate->explicitVisible &&
2080 (!childPrivate->extra.isAllocated() || !childPrivate->extra->effectRefCount))
2084 return childPrivate->itemNode();
2090void QQuickWindowPrivate::updateDirtyNode(QQuickItem *item)
2092 QQuickItemPrivate *itemPriv = QQuickItemPrivate::get(item);
2093 quint32 dirty = itemPriv->dirtyAttributes;
2094 itemPriv->dirtyAttributes = 0;
2096 if ((dirty & QQuickItemPrivate::TransformUpdateMask) ||
2097 (dirty & QQuickItemPrivate::Size && itemPriv->origin() != QQuickItem::TopLeft &&
2098 (itemPriv->scale() != 1. || itemPriv->rotation() != 0.))) {
2102 if (itemPriv->x != 0. || itemPriv->y != 0.)
2103 matrix.translate(itemPriv->x, itemPriv->y);
2105 for (
int ii = itemPriv->transforms.size() - 1; ii >= 0; --ii)
2106 itemPriv->transforms.at(ii)->applyTo(&matrix);
2108 if (itemPriv->scale() != 1. || itemPriv->rotation() != 0.) {
2109 QPointF origin = item->transformOriginPoint();
2110 matrix.translate(origin.x(), origin.y());
2111 if (itemPriv->scale() != 1.)
2112 matrix.scale(itemPriv->scale(), itemPriv->scale());
2113 if (itemPriv->rotation() != 0.)
2114 matrix.rotate(itemPriv->rotation(), 0, 0, 1);
2115 matrix.translate(-origin.x(), -origin.y());
2118 itemPriv->itemNode()->setMatrix(matrix);
2121 const bool clipEffectivelyChanged = dirty & (QQuickItemPrivate::Clip | QQuickItemPrivate::Window);
2122 if (clipEffectivelyChanged) {
2123 QSGNode *parent = itemPriv->opacityNode() ? (QSGNode *)itemPriv->opacityNode()
2124 : (QSGNode *)itemPriv->itemNode();
2125 QSGNode *child = itemPriv->rootNode();
2127 if (
bool initializeClipNode = item->clip() && itemPriv->clipNode() ==
nullptr;
2128 initializeClipNode) {
2129 QQuickDefaultClipNode *clip =
new QQuickDefaultClipNode(item->clipRect());
2130 itemPriv->extra.value().clipNode = clip;
2134 parent->reparentChildNodesTo(clip);
2135 parent->appendChildNode(clip);
2137 parent->removeChildNode(child);
2138 clip->appendChildNode(child);
2139 parent->appendChildNode(clip);
2142 }
else if (
bool updateClipNode = item->clip() && itemPriv->clipNode() !=
nullptr;
2144 QQuickDefaultClipNode *clip = itemPriv->clipNode();
2145 clip->setClipRect(item->clipRect());
2147 }
else if (
bool removeClipNode = !item->clip() && itemPriv->clipNode() !=
nullptr;
2149 QQuickDefaultClipNode *clip = itemPriv->clipNode();
2150 parent->removeChildNode(clip);
2152 clip->removeChildNode(child);
2153 parent->appendChildNode(child);
2155 clip->reparentChildNodesTo(parent);
2158 delete itemPriv->clipNode();
2159 itemPriv->extra->clipNode =
nullptr;
2163 const int effectRefCount = itemPriv->extra.isAllocated() ? itemPriv->extra->effectRefCount : 0;
2164 const bool effectRefEffectivelyChanged =
2165 (dirty & (QQuickItemPrivate::EffectReference | QQuickItemPrivate::Window))
2166 && ((effectRefCount == 0) != (itemPriv->rootNode() ==
nullptr));
2167 if (effectRefEffectivelyChanged) {
2168 if (dirty & QQuickItemPrivate::ChildrenUpdateMask)
2169 itemPriv->childContainerNode()->removeAllChildNodes();
2171 QSGNode *parent = itemPriv->clipNode();
2173 parent = itemPriv->opacityNode();
2175 parent = itemPriv->itemNode();
2177 if (itemPriv->extra.isAllocated() && itemPriv->extra->effectRefCount) {
2178 Q_ASSERT(itemPriv->rootNode() ==
nullptr);
2179 QSGRootNode *root =
new QSGRootNode();
2180 itemPriv->extra->rootNode = root;
2181 parent->reparentChildNodesTo(root);
2182 parent->appendChildNode(root);
2184 Q_ASSERT(itemPriv->rootNode() !=
nullptr);
2185 QSGRootNode *root = itemPriv->rootNode();
2186 parent->removeChildNode(root);
2187 root->reparentChildNodesTo(parent);
2188 delete itemPriv->rootNode();
2189 itemPriv->extra->rootNode =
nullptr;
2193 if (dirty & QQuickItemPrivate::ChildrenUpdateMask) {
2195 bool fetchedPaintNode =
false;
2196 QList<QQuickItem *> orderedChildren = itemPriv->paintOrderChildItems();
2197 int desiredNodesSize = orderedChildren.size() + (itemPriv->paintNode ? 1 : 0);
2205 int desiredNodesProcessed = 0;
2210 QSGNode *groupNode = itemPriv->childContainerNode();
2211 QSGNode *currentNode = groupNode->firstChild();
2212 QSGNode *desiredNode =
nullptr;
2214 while (currentNode && (desiredNode = fetchNextNode(itemPriv, ii, fetchedPaintNode))) {
2215 if (currentNode != desiredNode) {
2218 if (currentNode->nextSibling() == desiredNode) {
2220 groupNode->removeChildNode(currentNode);
2225 if (desiredNode->parent()) {
2226 desiredNode->parent()->removeChildNode(desiredNode);
2228 groupNode->insertChildNodeBefore(desiredNode, currentNode);
2232 currentNode = desiredNode;
2235 currentNode = currentNode->nextSibling();
2236 desiredNodesProcessed++;
2242 if (desiredNodesProcessed < desiredNodesSize) {
2243 while ((desiredNode = fetchNextNode(itemPriv, ii, fetchedPaintNode))) {
2244 if (desiredNode->parent())
2245 desiredNode->parent()->removeChildNode(desiredNode);
2246 groupNode->appendChildNode(desiredNode);
2248 }
else if (currentNode) {
2252 while (currentNode) {
2253 QSGNode *node = currentNode->nextSibling();
2254 groupNode->removeChildNode(currentNode);
2260 if ((dirty & QQuickItemPrivate::Size) && itemPriv->clipNode()) {
2261 itemPriv->clipNode()->setRect(item->clipRect());
2262 itemPriv->clipNode()->update();
2265 if (dirty & (QQuickItemPrivate::OpacityValue | QQuickItemPrivate::Visible
2266 | QQuickItemPrivate::HideReference | QQuickItemPrivate::Window))
2268 qreal opacity = itemPriv->explicitVisible && (!itemPriv->extra.isAllocated() || itemPriv->extra->hideRefCount == 0)
2269 ? itemPriv->opacity() : qreal(0);
2271 if (opacity != 1 && !itemPriv->opacityNode()) {
2272 QSGOpacityNode *node =
new QSGOpacityNode;
2273 itemPriv->extra.value().opacityNode = node;
2275 QSGNode *parent = itemPriv->itemNode();
2276 QSGNode *child = itemPriv->clipNode();
2278 child = itemPriv->rootNode();
2281 parent->removeChildNode(child);
2282 node->appendChildNode(child);
2283 parent->appendChildNode(node);
2285 parent->reparentChildNodesTo(node);
2286 parent->appendChildNode(node);
2289 if (itemPriv->opacityNode())
2290 itemPriv->opacityNode()->setOpacity(opacity);
2293 if (dirty & QQuickItemPrivate::ContentUpdateMask) {
2295 if (itemPriv->flags & QQuickItem::ItemHasContents) {
2296 updatePaintNodeData.transformNode = itemPriv->itemNode();
2297 itemPriv->paintNode = item->updatePaintNode(itemPriv->paintNode, &updatePaintNodeData);
2299 Q_ASSERT(itemPriv->paintNode ==
nullptr ||
2300 itemPriv->paintNode->parent() ==
nullptr ||
2301 itemPriv->paintNode->parent() == itemPriv->childContainerNode());
2303 if (itemPriv->paintNode && itemPriv->paintNode->parent() ==
nullptr) {
2304 QSGNode *before = qquickitem_before_paintNode(itemPriv);
2305 if (before && before->parent()) {
2306 Q_ASSERT(before->parent() == itemPriv->childContainerNode());
2307 itemPriv->childContainerNode()->insertChildNodeAfter(itemPriv->paintNode, before);
2309 itemPriv->childContainerNode()->prependChildNode(itemPriv->paintNode);
2312 }
else if (itemPriv->paintNode) {
2313 delete itemPriv->paintNode;
2314 itemPriv->paintNode =
nullptr;
2321 QList<QSGNode *> nodes;
2322 nodes << itemPriv->itemNodeInstance
2323 << itemPriv->opacityNode()
2324 << itemPriv->clipNode()
2325 << itemPriv->rootNode()
2326 << itemPriv->paintNode;
2327 nodes.removeAll(
nullptr);
2329 Q_ASSERT(nodes.constFirst() == itemPriv->itemNodeInstance);
2330 for (
int i=1; i<nodes.size(); ++i) {
2331 QSGNode *n = nodes.at(i);
2333 Q_ASSERT(n->parent() == nodes.at(i-1));
2335 Q_ASSERT(n == itemPriv->paintNode || n == itemPriv->childContainerNode() || n->childCount() == 1);
2341bool QQuickWindowPrivate::emitError(QQuickWindow::SceneGraphError error,
const QString &msg)
2344 static const QMetaMethod errorSignal = QMetaMethod::fromSignal(&QQuickWindow::sceneGraphError);
2345 if (q->isSignalConnected(errorSignal)) {
2346 emit q->sceneGraphError(error, msg);
2352void QQuickWindow::maybeUpdate()
2355 if (d->renderControl)
2356 QQuickRenderControlPrivate::get(d->renderControl)->maybeUpdate();
2357 else if (d->windowManager)
2358 d->windowManager->maybeUpdate(
this);
2361void QQuickWindow::cleanupSceneGraph()
2367 delete d->renderer->rootNode();
2369 d->renderer =
nullptr;
2371 d->runAndClearJobs(&d->beforeSynchronizingJobs);
2372 d->runAndClearJobs(&d->afterSynchronizingJobs);
2373 d->runAndClearJobs(&d->beforeRenderingJobs);
2374 d->runAndClearJobs(&d->afterRenderingJobs);
2375 d->runAndClearJobs(&d->afterSwapJobs);
2378QOpenGLContext *QQuickWindowPrivate::openglContext()
2380#if QT_CONFIG(opengl)
2381 if (context && context->isValid()) {
2382 QSGRendererInterface *rif = context->sceneGraphContext()->rendererInterface(context);
2385 return reinterpret_cast<QOpenGLContext *>(rif->getResource(q, QSGRendererInterface::OpenGLContextResource));
2393
2394
2395bool QQuickWindow::isSceneGraphInitialized()
const
2397 Q_D(
const QQuickWindow);
2398 return d->context !=
nullptr && d->context->isValid();
2402
2403
2404
2405
2406
2407
2408
2409
2412
2413
2414
2415
2416
2417
2420
2421
2422
2423
2424
2425
2428
2429
2430
2433
2434
2435
2436
2437
2438
2439
2440
2441
2442
2443
2444
2445
2446
2447
2450
2451
2452
2455
2456
2457
2458
2459
2460
2461
2462
2463
2464
2465
2466
2467
2470
2471
2472
2473
2474
2475
2476
2477
2478
2479
2480
2483
2484
2485
2486
2487
2488
2489
2490
2492
2493
2494
2495
2496
2497
2498
2499
2500
2501
2502
2505
2506
2507
2508
2509
2512
2513
2514
2515
2516
2517
2518
2519
2520
2521
2522
2523
2524
2527
2528
2529
2530
2531
2532
2533
2534
2535
2536
2539
2540
2541
2542
2543
2544
2545
2546
2547
2548
2549
2550
2551
2552
2553
2554
2555
2556
2557
2558
2559
2560
2561
2562
2563
2564
2565
2566
2567
2568
2569
2570
2571
2572
2573
2574
2575
2576
2577
2578
2579
2580
2581
2582
2583
2584
2585
2586
2587
2588
2589
2590
2591
2592
2593
2594
2595
2596
2597
2598
2599
2600void QQuickWindow::setRenderTarget(
const QQuickRenderTarget &target)
2603 if (target != d->customRenderTarget) {
2604 d->customRenderTarget = target;
2605 d->redirect.renderTargetDirty =
true;
2610
2611
2612
2613
2614
2615
2616
2617QQuickRenderTarget QQuickWindow::renderTarget()
const
2619 Q_D(
const QQuickWindow);
2620 return d->customRenderTarget;
2624class GrabWindowForProtectedContent :
public QRunnable
2627 GrabWindowForProtectedContent(QQuickWindow *window, QImage *image, QWaitCondition *condition)
2630 , m_condition(condition)
2634 bool checkGrabbable()
2640 if (!QQuickWindowPrivate::get(m_window))
2648 if (!checkGrabbable())
2651 *m_image = QSGRhiSupport::instance()->grabOffscreenForProtectedContent(m_window);
2653 m_condition->wakeOne();
2658 QQuickWindow *m_window;
2660 QWaitCondition *m_condition;
2666
2667
2668
2669
2670
2671
2672
2673
2674
2675
2676
2677
2678
2679
2680
2681
2682
2683
2684
2685QImage QQuickWindow::grabWindow()
2689 if (!d->isRenderable() && !d->renderControl) {
2691 if (d->windowManager && (d->windowManager->flags() & QSGRenderLoop::SupportsGrabWithoutExpose))
2692 return d->windowManager->grab(
this);
2694 if (!isSceneGraphInitialized()) {
2701 return QSGRhiSupport::instance()->grabOffscreen(
this);
2706 if (requestedFormat().testOption(QSurfaceFormat::ProtectedContent)) {
2709 QWaitCondition condition;
2711 GrabWindowForProtectedContent *job =
new GrabWindowForProtectedContent(
this, &image, &condition);
2713 qWarning(
"QQuickWindow::grabWindow: Failed to create a job for capturing protected content");
2717 scheduleRenderJob(job, QQuickWindow::NoStage);
2718 condition.wait(&mutex);
2727 if (d->renderControl)
2728 return QQuickRenderControlPrivate::get(d->renderControl)->grab();
2729 else if (d->windowManager)
2730 return d->windowManager->grab(
this);
2736
2737
2738
2739
2740
2741
2742
2743QQmlIncubationController *QQuickWindow::incubationController()
const
2745 Q_D(
const QQuickWindow);
2747 if (!d->windowManager)
2750 if (!d->incubationController)
2751 d->incubationController =
new QQuickWindowIncubationController(d->windowManager);
2752 return d->incubationController;
2758
2759
2760
2761
2762
2763
2764
2765
2766
2767
2768
2769
2770
2771
2772
2773
2774
2775
2776
2777
2778
2779
2782
2783
2784
2785
2786
2787
2788
2789
2790
2791
2792
2793
2796
2797
2798
2799
2800
2801
2802
2803
2804
2805
2806
2807
2808
2809
2810
2811
2812
2813
2814
2815
2816
2817
2820
2821
2822
2823
2824
2825
2826
2827
2828
2829
2830
2831
2832
2833
2834
2835
2836
2837
2838
2839
2840
2841
2842
2843
2846
2847
2848
2851
2852
2853
2854
2855
2856
2857
2858
2859
2860
2861
2862
2863
2864
2865
2866
2867
2868
2869
2870
2873
2874
2875
2876
2879
2880
2881
2882
2883
2884
2885
2886
2887
2888
2889
2890
2891
2892
2893
2894
2895
2896
2897
2898
2899
2900
2901
2902
2903
2904
2905
2906
2909
2910
2911
2914
2915
2916
2917
2918
2919
2920
2921
2922
2923
2924
2925
2926
2927
2928
2929
2930
2931
2932
2933
2934
2935
2936
2937
2938
2939
2940
2941
2944
2945
2946
2949
2950
2951
2952
2953
2954
2955
2956
2957
2958
2959
2960
2961
2962
2963
2964
2965
2966
2967
2968
2969
2970
2971
2972
2973
2974
2975
2976
2979
2980
2981
2982
2985
2986
2987
2988
2989
2990
2991
2992
2993
2994
2995
2996
2997
2998
2999
3000
3001
3002
3003
3004
3005
3006
3007
3008
3009
3010
3011
3014
3015
3016
3017
3018
3019
3020
3021
3022
3023
3024
3025
3026
3027
3028
3029
3030
3031
3032
3033
3036
3037
3038
3041
3042
3043
3044
3045
3046
3047
3048
3049
3050
3051
3052
3053
3054
3055
3056
3057
3058
3061
3062
3063
3066
3067
3068
3069
3072
3073
3074
3075
3076
3077
3078
3079
3080
3081
3082
3083
3086
3087
3088
3089
3090
3091
3092
3093
3094
3097
3098
3099
3100
3101
3102
3103
3104
3105
3106
3107
3108
3109
3110
3111
3112
3113
3114
3115
3116
3117
3120
3121
3122
3123
3126
3127
3129QSGTexture *QQuickWindow::createTextureFromImage(
const QImage &image)
const
3131 return createTextureFromImage(image, {});
3136
3137
3138
3139
3140
3141
3142
3143
3144
3145
3146
3147
3148
3149
3150
3151
3152
3153
3154
3155
3156
3157
3158
3159
3160
3161
3162
3163
3164
3165
3166
3167
3168
3169
3170
3171
3172
3173
3174
3175
3176
3177
3178
3180QSGTexture *QQuickWindow::createTextureFromImage(
const QImage &image, CreateTextureOptions options)
const
3182 Q_D(
const QQuickWindow);
3183 if (!isSceneGraphInitialized())
3186 if (options & TextureCanUseAtlas) flags |= QSGRenderContext::CreateTexture_Atlas;
3187 if (options & TextureHasMipmaps) flags |= QSGRenderContext::CreateTexture_Mipmap;
3188 if (!(options & TextureIsOpaque)) flags |= QSGRenderContext::CreateTexture_Alpha;
3189 return d->context->createTexture(image, flags);
3193
3194
3195
3196
3197
3198
3199
3200
3201
3202
3203
3204
3205
3206
3207
3208
3209
3210
3211
3212
3213
3214
3215
3216
3217
3218
3219
3220
3221
3222
3223
3224
3225QSGTexture *QQuickWindow::createTextureFromRhiTexture(QRhiTexture *texture, CreateTextureOptions options)
const
3227 Q_D(
const QQuickWindow);
3231 QSGPlainTexture *t =
new QSGPlainTexture;
3232 t->setOwnsTexture(
true);
3233 t->setTexture(texture);
3234 t->setHasAlphaChannel(options & QQuickWindow::TextureHasAlphaChannel);
3235 t->setTextureSize(texture->pixelSize());
3242QSGTexture *QQuickWindowPrivate::createTextureFromNativeTexture(quint64 nativeObjectHandle,
3243 int nativeLayoutOrState,
3246 QQuickWindow::CreateTextureOptions options,
3247 TextureFromNativeTextureFlags flags)
const
3252 QSGPlainTexture *texture =
new QSGPlainTexture;
3253 texture->setTextureFromNativeTexture(rhi, nativeObjectHandle, nativeLayoutOrState, nativeFormat,
3254 size, options, flags);
3255 texture->setHasAlphaChannel(options & QQuickWindow::TextureHasAlphaChannel);
3257 texture->setOwnsTexture(
true);
3258 texture->setTextureSize(size);
3263
3264
3265
3266
3267
3268
3269
3270
3271
3272
3275
3276
3277
3278
3279
3280
3281
3283void QQuickWindow::setColor(
const QColor &color)
3286 if (color == d->clearColor)
3289 if (color.alpha() != d->clearColor.alpha()) {
3290 QSurfaceFormat fmt = requestedFormat();
3291 if (color.alpha() < 255)
3292 fmt.setAlphaBufferSize(8);
3294 fmt.setAlphaBufferSize(-1);
3297 d->clearColor = color;
3298 emit colorChanged(color);
3302QColor QQuickWindow::color()
const
3304 return d_func()->clearColor;
3308
3309
3310
3311
3312
3313bool QQuickWindow::hasDefaultAlphaBuffer()
3315 return QQuickWindowPrivate::defaultAlphaBuffer;
3319
3320
3321
3322
3323
3324
3325
3326
3327void QQuickWindow::setDefaultAlphaBuffer(
bool useAlpha)
3329 QQuickWindowPrivate::defaultAlphaBuffer = useAlpha;
3333
3334
3335
3336
3337
3338
3339
3342
3343
3344
3345
3346
3347
3348
3349
3350
3351
3352
3353
3354
3355
3356
3357
3358
3359
3360
3361
3362
3363
3364
3365
3366
3367
3368
3369
3370
3371
3372
3373
3374
3375
3378
3379
3380
3381
3382
3383
3384
3387
3388
3389
3390
3391
3392
3393
3394
3395const QQuickWindow::GraphicsStateInfo &QQuickWindow::graphicsStateInfo()
3399 d->rhiStateInfo.currentFrameSlot = d->rhi->currentFrameSlot();
3400 d->rhiStateInfo.framesInFlight = d->rhi->resourceLimit(QRhi::FramesInFlight);
3402 return d->rhiStateInfo;
3406
3407
3408
3409
3410
3411
3412
3413
3414
3415
3416
3417
3418
3419
3420
3421
3422
3423
3424
3425
3426
3427
3428
3429
3430
3431
3432
3433
3434
3435
3436
3437
3438
3439
3440
3441
3442
3443
3444
3445
3446void QQuickWindow::beginExternalCommands()
3449 if (d->rhi && d->context && d->context->isValid()) {
3450 QSGDefaultRenderContext *rc =
static_cast<QSGDefaultRenderContext *>(d->context);
3451 QRhiCommandBuffer *cb = rc->currentFrameCommandBuffer();
3453 cb->beginExternal();
3458
3459
3460
3461
3462
3463
3464
3465
3466
3467
3468
3469
3470
3471
3472
3473
3474
3475
3476
3477
3478
3479
3480
3481
3482
3483void QQuickWindow::endExternalCommands()
3486 if (d->rhi && d->context && d->context->isValid()) {
3487 QSGDefaultRenderContext *rc =
static_cast<QSGDefaultRenderContext *>(d->context);
3488 QRhiCommandBuffer *cb = rc->currentFrameCommandBuffer();
3495
3496
3497
3498
3499
3500
3501
3502
3503
3506
3507
3508
3509
3510
3511
3512
3513
3516
3517
3518
3519
3520
3521
3522
3523
3524
3525
3526
3527
3528
3529
3530
3533
3534
3535
3536
3537
3538
3541
3542
3543
3544
3545
3546
3547
3550
3551
3552
3553
3554
3555
3556
3557
3558
3559
3560
3561
3562
3563
3564
3565
3566
3567
3568
3569
3570
3573
3574
3575
3576
3577
3578
3579
3580
3581
3584
3585
3586
3587
3588
3589
3590
3591
3592
3595
3596
3597
3598
3599
3600
3601
3602
3603
3604
3607
3608
3609
3610
3611
3612
3613
3614
3615
3616
3617
3618
3619
3620
3621
3622
3623
3624
3625
3626
3627
3628
3629
3630
3633
3634
3635
3636
3637
3638
3639
3640
3641
3642
3643
3646
3647
3648
3649
3652
3653
3654
3655
3656
3657
3658
3659
3660
3661
3662
3663
3664
3665
3666
3667
3668
3669
3670
3673
3674
3675
3676
3677
3678
3679
3680
3681
3682
3683
3684
3685
3686
3687
3690
3691
3692
3693
3694
3695
3696
3697
3698
3699
3700
3701
3702
3703
3704
3705
3706
3707
3708
3709
3712
3713
3714
3715
3716
3717
3718
3719
3720
3721
3722
3723
3724
3725
3726
3727
3728
3729
3730
3731
3732
3733
3734
3735
3736
3737
3738
3739
3740
3741
3742
3743
3744
3745
3746
3747
3748
3749
3752
3753
3754
3755
3756
3757
3758
3759
3760
3761
3762
3763
3764
3767
3768
3769
3770
3771
3772
3775
3776
3777
3778
3779
3780
3781
3784
3785
3786
3787
3788
3789
3790
3791
3792
3793
3796
3797
3798
3799
3800
3801
3802
3803
3804
3805
3806
3809
3810
3811
3812
3813
3816
3817
3818
3819
3820
3821
3822
3823
3824
3825
3828
3829
3830
3831
3832
3833
3834
3835
3836
3837
3838
3841
3842
3843
3844
3845
3846
3849
3850
3851
3852
3853
3854
3857
3858
3859
3860
3861
3862
3863
3864
3865
3868
3869
3870
3871
3872
3873
3874
3875
3878
3879
3880
3881
3882
3883
3886
3887
3888
3889
3890
3891
3894
3895
3896
3897
3898
3899
3902
3903
3904
3905
3906
3907
3910
3911
3912
3913
3914
3915
3916
3917
3918
3919
3920
3921
3924
3925
3926
3927
3928
3929
3930
3931
3932
3933
3934
3935
3936
3937
3938
3939
3940
3941
3942
3943
3944
3945
3946
3947
3948
3949
3950
3951
3952
3953
3954
3955
3956
3957
3958
3960void QQuickWindow::scheduleRenderJob(QRunnable *job, RenderStage stage)
3964 d->renderJobMutex.lock();
3965 if (stage == BeforeSynchronizingStage) {
3966 d->beforeSynchronizingJobs << job;
3967 }
else if (stage == AfterSynchronizingStage) {
3968 d->afterSynchronizingJobs << job;
3969 }
else if (stage == BeforeRenderingStage) {
3970 d->beforeRenderingJobs << job;
3971 }
else if (stage == AfterRenderingStage) {
3972 d->afterRenderingJobs << job;
3973 }
else if (stage == AfterSwapStage) {
3974 d->afterSwapJobs << job;
3975 }
else if (stage == NoStage) {
3976 if (d->renderControl && d->rhi && d->rhi->thread() == QThread::currentThread()) {
3979 }
else if (isExposed()) {
3980 d->windowManager->postJob(
this, job);
3985 d->renderJobMutex.unlock();
3988void QQuickWindowPrivate::runAndClearJobs(QList<QRunnable *> *jobs)
3990 renderJobMutex.lock();
3991 QList<QRunnable *> jobList = *jobs;
3993 renderJobMutex.unlock();
3995 for (QRunnable *r : std::as_const(jobList)) {
4001void QQuickWindow::runJobsAfterSwap()
4004 d->runAndClearJobs(&d->afterSwapJobs);
4008
4009
4010
4011
4012
4013
4014
4015
4016
4017
4018
4019
4020
4021
4022qreal QQuickWindow::effectiveDevicePixelRatio()
const
4024 Q_D(
const QQuickWindow);
4025 QWindow *w = QQuickRenderControl::renderWindowFor(
const_cast<QQuickWindow *>(
this));
4027 return w->devicePixelRatio();
4029 if (!d->customRenderTarget.isNull())
4030 return d->customRenderTarget.devicePixelRatio();
4032 return devicePixelRatio();
4036
4037
4038
4039
4040
4041
4042
4043
4044
4045
4046
4047
4048
4049
4050
4051
4052
4053
4054
4055
4056QSGRendererInterface *QQuickWindow::rendererInterface()
const
4058 Q_D(
const QQuickWindow);
4066 return d->context->sceneGraphContext()->rendererInterface(d->context);
4070
4071
4072
4073
4074
4075
4076
4077
4078
4079
4080
4081
4082
4083
4084
4085
4086
4087
4088
4089QRhi *QQuickWindow::rhi()
const
4091 Q_D(
const QQuickWindow);
4096
4097
4098
4099
4100
4101
4102
4103
4104
4105QRhiSwapChain *QQuickWindow::swapChain()
const
4107 Q_D(
const QQuickWindow);
4108 return d->swapchain;
4112
4113
4114
4115
4116
4117
4118
4119
4120
4121
4122
4123
4124
4125
4126
4127
4128
4129
4130
4131
4132
4133
4134
4135
4136
4137
4138
4139
4140
4141
4142
4143
4144
4145
4146
4147
4148
4149
4150
4151
4152
4153
4154
4155
4156
4157void QQuickWindow::setGraphicsApi(QSGRendererInterface::GraphicsApi api)
4161 case QSGRendererInterface::Software:
4162 setSceneGraphBackend(QStringLiteral(
"software"));
4164 case QSGRendererInterface::OpenVG:
4165 setSceneGraphBackend(QStringLiteral(
"openvg"));
4173 if (QSGRendererInterface::isApiRhiBased(api) || api == QSGRendererInterface::Unknown)
4174 QSGRhiSupport::instance_internal()->configure(api);
4178
4179
4180
4181
4182
4183
4184
4185
4186
4187
4188
4189
4190
4191
4192
4193
4194
4195
4196
4197
4198
4199
4200
4201
4202
4203
4204
4205
4206QSGRendererInterface::GraphicsApi QQuickWindow::graphicsApi()
4212 return QSGRhiSupport::instance()->graphicsApi();
4216
4217
4218
4219
4220
4221
4222
4223
4224
4225
4226
4227
4228
4229
4230
4231
4232
4233
4234
4235void QQuickWindow::setSceneGraphBackend(
const QString &backend)
4237 QSGContext::setBackend(backend);
4241
4242
4243
4244
4245
4246
4247
4248
4249
4250
4251
4252QString QQuickWindow::sceneGraphBackend()
4254 return QSGContext::backend();
4258
4259
4260
4261
4262
4263
4264
4265
4266
4267
4268
4269
4270
4271
4272
4273
4274
4275
4276
4277
4278
4279
4280
4281
4282
4283
4284
4285
4286
4287
4288
4289
4290
4291
4292
4293
4294
4295
4296
4297
4298
4299
4300
4301
4302
4303
4304
4305
4306
4307
4308
4309
4310
4311
4312
4313
4314
4315
4316
4317
4318
4319
4320void QQuickWindow::setGraphicsDevice(
const QQuickGraphicsDevice &device)
4323 d->customDeviceObjects = device;
4327
4328
4329
4330
4331
4332
4333
4334QQuickGraphicsDevice QQuickWindow::graphicsDevice()
const
4336 Q_D(
const QQuickWindow);
4337 return d->customDeviceObjects;
4341
4342
4343
4344
4345
4346
4347
4348
4349
4350
4351
4352
4353
4354
4355
4356
4357
4358
4359
4360
4361
4362
4363
4364
4365
4366
4367void QQuickWindow::setGraphicsConfiguration(
const QQuickGraphicsConfiguration &config)
4370 d->graphicsConfig = config;
4374
4375
4376
4377
4378
4379
4380
4381QQuickGraphicsConfiguration QQuickWindow::graphicsConfiguration()
const
4383 Q_D(
const QQuickWindow);
4384 return d->graphicsConfig;
4388
4389
4390
4391
4392
4393QSGTextNode *QQuickWindow::createTextNode()
const
4395 Q_D(
const QQuickWindow);
4396 return isSceneGraphInitialized() ? d->context->sceneGraphContext()->createTextNode(d->context) :
nullptr;
4400
4401
4402
4403
4404
4405
4406
4407QSGRectangleNode *QQuickWindow::createRectangleNode()
const
4409 Q_D(
const QQuickWindow);
4410 return isSceneGraphInitialized() ? d->context->sceneGraphContext()->createRectangleNode() :
nullptr;
4414
4415
4416
4417
4418
4419
4420
4421QSGImageNode *QQuickWindow::createImageNode()
const
4423 Q_D(
const QQuickWindow);
4424 return isSceneGraphInitialized() ? d->context->sceneGraphContext()->createImageNode() :
nullptr;
4428
4429
4430
4431
4432QSGNinePatchNode *QQuickWindow::createNinePatchNode()
const
4434 Q_D(
const QQuickWindow);
4435 return isSceneGraphInitialized() ? d->context->sceneGraphContext()->createNinePatchNode() :
nullptr;
4439
4440
4441
4442
4443
4444
4445
4446QQuickWindow::TextRenderType QQuickWindow::textRenderType()
4448 return QQuickWindowPrivate::textRenderType;
4452
4453
4454
4455
4456
4457
4458
4459
4460
4461void QQuickWindow::setTextRenderType(QQuickWindow::TextRenderType renderType)
4463 QQuickWindowPrivate::textRenderType = renderType;
4468
4469
4470
4471
4472
4473
4474
4475
4476
4477
4478
4479
4480
4481
4482
4483
4484
4486#ifndef QT_NO_DEBUG_STREAM
4489 QDebugStateSaver saver(debug);
4492 debug <<
"QQuickWindow(nullptr)";
4496 debug << win->metaObject()->className() <<
'(' <<
static_cast<
const void *>(win);
4497 if (win->isActive())
4499 if (win->isExposed())
4500 debug <<
" exposed";
4501 debug <<
", visibility=" << win->visibility() <<
", flags=" << win->flags();
4502 if (!win->title().isEmpty())
4503 debug <<
", title=" << win->title();
4504 if (!win->objectName().isEmpty())
4505 debug <<
", name=" << win->objectName();
4507 debug <<
", parent=" <<
static_cast<
const void *>(win->parent());
4508 if (win->transientParent())
4509 debug <<
", transientParent=" <<
static_cast<
const void *>(win->transientParent());
4510 debug <<
", geometry=";
4511 QtDebugUtils::formatQRect(debug, win->geometry());
4519#include "qquickwindow.moc"
4520#include "moc_qquickwindow_p.cpp"
4521#include "moc_qquickwindow.cpp"
Q_STATIC_LOGGING_CATEGORY(lcAccessibilityCore, "qt.accessibility.core")
static void updatePixelRatioHelper(QQuickItem *item, float pixelRatio)
void forcePolishHelper(QQuickItem *item)
void forceUpdate(QQuickItem *item)
QDebug operator<<(QDebug debug, const QQuickWindow *win)
static QSGNode * qquickitem_before_paintNode(QQuickItemPrivate *d)
static QSGNode * fetchNextNode(QQuickItemPrivate *itemPriv, int &ii, bool &returnedPaintNode)
bool check(QQuickItem *item, int itemsRemainingBeforeUpdatePolish)
const QVector< QQuickItem * > & itemsToPolish
int numPolishLoopsInSequence
PolishLoopDetector(const QVector< QQuickItem * > &itemsToPolish)
QRhiRenderBuffer * depthStencil
QRhiTexture * multisampleTexture
QRhiTexture * depthStencilTexture
QRhiRenderPassDescriptor * rpDesc
QRhiRenderBuffer * renderBuffer