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
qqmlthread_p.h
Go to the documentation of this file.
1// Copyright (C) 2016 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
4#ifndef QQMLTHREAD_P_H
5#define QQMLTHREAD_P_H
6
7//
8// W A R N I N G
9// -------------
10//
11// This file is not part of the Qt API. It exists purely as an
12// implementation detail. This header file may change from version to
13// version without notice, or even be removed.
14//
15// We mean it.
16//
17
18
19#include <QtCore/qglobal.h>
20
21#include <private/qintrusivelist_p.h>
22
23QT_BEGIN_NAMESPACE
24
25class QThread;
26class QMutex;
27
28class QQmlThreadPrivate;
30{
31public:
32 QQmlThread();
33 virtual ~QQmlThread();
34
35 void lock();
36 void unlock();
37 void wakeOne();
38 void wait();
39
40 QThread *thread() const;
41 QObject *threadObject() const;
42 bool isThisThread() const;
43
44 // Synchronously invoke a method in the thread
45 template<typename Method, typename ...Args>
46 void callMethodInThread(Method &&method, Args &&...args);
47
48 // Synchronously invoke a method in the main thread. If the main thread is
49 // blocked in a callMethodInThread() call, the call is made from within that
50 // call.
51 template<typename Method, typename ...Args>
52 void callMethodInMain(Method &&method, Args &&...args);
53
54 // Asynchronously invoke a method in the thread.
55 template<typename Method, typename ...Args>
56 void postMethodToThread(Method &&method, Args &&...args);
57
58 // Asynchronously invoke a method in the main thread.
59 template<typename Method, typename ...Args>
60 void postMethodToMain(Method &&method, Args &&...args);
61
62 void waitForNextMessage();
63 void discardMessages();
64
65protected:
66 void startup();
67 void shutdown();
68
69private:
70 friend class QQmlThreadPrivate;
71
72 struct Message {
73 Message() : next(nullptr) {}
74 virtual ~Message() {}
75 Message *next;
76 virtual void call(QQmlThread *) = 0;
77 };
78 template<typename Method, typename ...Args>
79 Message *createMessageFromMethod(Method &&method, Args &&...args);
80 void internalCallMethodInThread(Message *);
81 void internalCallMethodInMain(Message *);
82 void internalPostMethodToThread(Message *);
83 void internalPostMethodToMain(Message *);
84 QQmlThreadPrivate *d;
85};
86
87namespace QtPrivate {
88template <typename> struct member_function_traits;
89
90template <typename Return, typename Object, typename... Args>
91struct member_function_traits<Return (Object::*)(Args...)>
92{
93 using class_type = Object;
94};
95}
96
97template<typename Method, typename ...Args>
98QQmlThread::Message *QQmlThread::createMessageFromMethod(Method &&method, Args &&...args)
99{
100 struct I : public Message {
101 Method m;
102 std::tuple<std::decay_t<Args>...> arguments;
103 I(Method &&method, Args&& ...args) : m(std::forward<Method>(method)), arguments(std::forward<Args>(args)...) {}
104 void call(QQmlThread *thread) override {
105 using class_type = typename QtPrivate::member_function_traits<Method>::class_type;
106 class_type *me = static_cast<class_type *>(thread);
107 std::apply(m, std::tuple_cat(std::make_tuple(me), arguments));
108 }
109 };
110 return new I(std::forward<Method>(method), std::forward<Args>(args)...);
111}
112
113template<typename Method, typename ...Args>
114void QQmlThread::callMethodInMain(Method &&method, Args&& ...args)
115{
116 Message *m = createMessageFromMethod(std::forward<Method>(method), std::forward<Args>(args)...);
117 internalCallMethodInMain(m);
118}
119
120template<typename Method, typename ...Args>
121void QQmlThread::callMethodInThread(Method &&method, Args&& ...args)
122{
123 Message *m = createMessageFromMethod(std::forward<Method>(method), std::forward<Args>(args)...);
124 internalCallMethodInThread(m);
125}
126
127template<typename Method, typename ...Args>
128void QQmlThread::postMethodToThread(Method &&method, Args&& ...args)
129{
130 Message *m = createMessageFromMethod(std::forward<Method>(method), std::forward<Args>(args)...);
131 internalPostMethodToThread(m);
132}
133
134template<typename Method, typename ...Args>
135void QQmlThread::postMethodToMain(Method &&method, Args&& ...args)
136{
137 Message *m = createMessageFromMethod(std::forward<Method>(method), std::forward<Args>(args)...);
138 internalPostMethodToMain(m);
139}
140
141QT_END_NAMESPACE
142
143#endif // QQMLTHREAD_P_H
QQmlThreadPrivate(QQmlThread *q)
bool event(QEvent *e) override
This virtual function receives events to an object and should return true if the event e was recogniz...
void postMethodToThread(Method &&method, Args &&...args)
QThread * thread() const
bool isThisThread() const
void waitForNextMessage()
void postMethodToMain(Method &&method, Args &&...args)
void callMethodInMain(Method &&method, Args &&...args)
virtual ~QQmlThread()
void callMethodInThread(Method &&method, Args &&...args)
QObject * threadObject() const
And object living in the QML thread, in case you want to parent other objects to it.
QT_REQUIRE_CONFIG(itemmodel)