From 09a199127fb4e43b0d31af5cf3fc85d6000576e6 Mon Sep 17 00:00:00 2001 From: kleuter Date: Mon, 18 Nov 2024 11:31:57 +0100 Subject: [PATCH] qt 6.8.0 support --- README.md | 8 +- designer.png | Bin 77464 -> 73746 bytes qtbase/src/corelib/kernel/qfunctions_win.cpp | 1 + qtbase/src/corelib/thread/qthread_win.cpp | 648 ++++++++++++++ qtbase/src/gui/rhi/qrhid3d11.cpp | 95 +- qtbase/src/gui/rhi/qrhid3d11_p.h | 2 + qtbase/src/gui/rhi/qrhid3d12.cpp | 101 ++- .../text/windows/qwindowsfontdatabasebase.cpp | 27 +- .../platforms/windows/qwindowscontext.cpp | 92 +- .../platforms/windows/qwindowscontext.h | 6 +- .../platforms/windows/qwindowsdrag.cpp | 12 +- .../platforms/windows/qwindowskeymapper.cpp | 26 +- .../windows/qwindowspointerhandler.cpp | 31 +- .../platforms/windows/qwindowstheme.cpp | 338 ++++--- .../platforms/windows/qwindowswindow.cpp | 27 +- .../uiautomation/qwindowsuiaaccessibility.cpp | 170 ++++ .../uiautomation/qwindowsuiamainprovider.cpp | 843 ++++++++++++++++++ .../uiautomation/qwindowsuiawrapper.cpp | 89 ++ .../uiautomation/qwindowsuiawrapper_p.h | 57 ++ qtbase/src/widgets/styles/qwindowsstyle.cpp | 45 - 20 files changed, 2264 insertions(+), 354 deletions(-) create mode 100644 qtbase/src/corelib/thread/qthread_win.cpp create mode 100644 qtbase/src/plugins/platforms/windows/uiautomation/qwindowsuiaaccessibility.cpp create mode 100644 qtbase/src/plugins/platforms/windows/uiautomation/qwindowsuiamainprovider.cpp create mode 100644 qtbase/src/plugins/platforms/windows/uiautomation/qwindowsuiawrapper.cpp create mode 100644 qtbase/src/plugins/platforms/windows/uiautomation/qwindowsuiawrapper_p.h diff --git a/README.md b/README.md index a59e6ac8..1b34b45e 100644 --- a/README.md +++ b/README.md @@ -1,11 +1,11 @@ -This is Qt 6.7.3 **qtbase module** backport that runs on Windows 7/8. The repository contains patched source files from the qtbase module. +This is Qt 6.8.0 **qtbase module** backport that runs on Windows 7/8. The repository contains patched source files from the qtbase module. Approach is based on this [forum thread](https://forum.qt.io/topic/133002/qt-creator-6-0-1-and-qt-6-2-2-running-on-windows-7/60) but better: many improvements amongst important fallbacks to default Qt 6 behaviour when running on newer Windows. - After replacing qtbase source files with the patched ones from this repository, you can compile Qt yourself using compiler and build options you need. -- You can use our [compile_win.pl](https://github.com/crystalidea/qt-build-tools/tree/master/6.7.3) build script (uses Visual C++ 2022 with OpenSSL 3.0.13 statically linked) +- You can use our [compile_win.pl](https://github.com/crystalidea/qt-build-tools/tree/master/6.8.0) build script (uses Visual C++ 2022 with OpenSSL 3.0.13 statically linked) - You can download our [prebuild Qt dlls](https://github.com/crystalidea/qt6windows7/releases) which also have Qt Designer binary for demonstration -**Qt 6.7.3 designer running on Windows 7**: +**Qt 6.8.0 designer running on Windows 7**: ![Qt Designer](designer.png) @@ -15,6 +15,8 @@ Approach is based on this [forum thread](https://forum.qt.io/topic/133002/qt-cre ### Older versions: +- [Qt 6.7.3](https://github.com/crystalidea/qt6windows7/releases/tag/v6.7.3) +- [Qt 6.7.2](https://github.com/crystalidea/qt6windows7/releases/tag/v6.7.2) - [Qt 6.6.3](https://github.com/crystalidea/qt6windows7/releases/tag/v6.6.3) - [Qt 6.6.2](https://github.com/crystalidea/qt6windows7/releases/tag/v6.6.2) - [Qt 6.6.1](https://github.com/crystalidea/qt6windows7/releases/tag/v6.6.1) diff --git a/designer.png b/designer.png index 95cf566479fbf745ba9c1de1d1a267bceded10fb..06024de1963fbdc760a6754f825d908ce246eadc 100644 GIT binary patch literal 73746 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 literal 77464 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

RS!q?_ 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 NvM_-g*BjnV_#Z+84>AA% diff --git a/qtbase/src/corelib/kernel/qfunctions_win.cpp b/qtbase/src/corelib/kernel/qfunctions_win.cpp index a74d3979..e715093d 100644 --- a/qtbase/src/corelib/kernel/qfunctions_win.cpp +++ b/qtbase/src/corelib/kernel/qfunctions_win.cpp @@ -28,6 +28,7 @@ QComHelper::QComHelper(COINIT concurrencyModel) QComHelper::~QComHelper() { + Q_ASSERT(m_threadId == GetCurrentThreadId()); if (SUCCEEDED(m_initResult)) CoUninitialize(); } diff --git a/qtbase/src/corelib/thread/qthread_win.cpp b/qtbase/src/corelib/thread/qthread_win.cpp new file mode 100644 index 00000000..3fa2fba6 --- /dev/null +++ b/qtbase/src/corelib/thread/qthread_win.cpp @@ -0,0 +1,648 @@ +// Copyright (C) 2016 The Qt Company Ltd. +// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR LGPL-3.0-only OR GPL-2.0-only OR GPL-3.0-only + +#include "qthread.h" +#include "qthread_p.h" +#include "qthreadstorage.h" +#include "qmutex.h" + +#include +#include + +#include +#include + +#include + +#ifndef _MT +# define _MT +#endif // _MT +#include + +extern "C" { +// MinGW is missing the declaration of SetThreadDescription: +WINBASEAPI +HRESULT +WINAPI +SetThreadDescription( + _In_ HANDLE hThread, + _In_ PCWSTR lpThreadDescription + ); +} + +QT_BEGIN_NAMESPACE + +#if QT_CONFIG(thread) + +void qt_watch_adopted_thread(const HANDLE adoptedThreadHandle, QThread *qthread); +DWORD WINAPI qt_adopted_thread_watcher_function(LPVOID); + +static DWORD qt_current_thread_data_tls_index = TLS_OUT_OF_INDEXES; +void qt_create_tls() +{ + if (qt_current_thread_data_tls_index != TLS_OUT_OF_INDEXES) + return; + Q_CONSTINIT static QBasicMutex mutex; + QMutexLocker locker(&mutex); + if (qt_current_thread_data_tls_index != TLS_OUT_OF_INDEXES) + return; + qt_current_thread_data_tls_index = TlsAlloc(); +} + +static void qt_free_tls() +{ + if (qt_current_thread_data_tls_index != TLS_OUT_OF_INDEXES) { + TlsFree(qt_current_thread_data_tls_index); + qt_current_thread_data_tls_index = TLS_OUT_OF_INDEXES; + } +} +Q_DESTRUCTOR_FUNCTION(qt_free_tls) + +/* + QThreadData +*/ +void QThreadData::clearCurrentThreadData() +{ + TlsSetValue(qt_current_thread_data_tls_index, 0); +} + +QThreadData *QThreadData::current(bool createIfNecessary) +{ + qt_create_tls(); + QThreadData *threadData = reinterpret_cast(TlsGetValue(qt_current_thread_data_tls_index)); + if (!threadData && createIfNecessary) { + threadData = new QThreadData; + // This needs to be called prior to new AdoptedThread() to + // avoid recursion. + TlsSetValue(qt_current_thread_data_tls_index, threadData); + QT_TRY { + threadData->thread = new QAdoptedThread(threadData); + } QT_CATCH(...) { + TlsSetValue(qt_current_thread_data_tls_index, 0); + threadData->deref(); + threadData = 0; + QT_RETHROW; + } + threadData->deref(); + threadData->isAdopted = true; + threadData->threadId.storeRelaxed(reinterpret_cast(quintptr(GetCurrentThreadId()))); + + if (!QCoreApplicationPrivate::theMainThreadId) { + auto *mainThread = threadData->thread.loadRelaxed(); + mainThread->setObjectName("Qt mainThread"); + QCoreApplicationPrivate::theMainThread.storeRelease(mainThread); + QCoreApplicationPrivate::theMainThreadId.storeRelaxed(threadData->threadId.loadRelaxed()); + } else { + HANDLE realHandle = INVALID_HANDLE_VALUE; + DuplicateHandle(GetCurrentProcess(), + GetCurrentThread(), + GetCurrentProcess(), + &realHandle, + 0, + FALSE, + DUPLICATE_SAME_ACCESS); + qt_watch_adopted_thread(realHandle, threadData->thread); + } + } + return threadData; +} + +void QAdoptedThread::init() +{ + d_func()->handle = GetCurrentThread(); + d_func()->id = GetCurrentThreadId(); +} + +static QList qt_adopted_thread_handles; +static QList qt_adopted_qthreads; +Q_CONSTINIT static QBasicMutex qt_adopted_thread_watcher_mutex; +static DWORD qt_adopted_thread_watcher_id = 0; +static HANDLE qt_adopted_thread_wakeup = 0; + +/*! + \internal + Adds an adopted thread to the list of threads that Qt watches to make sure + the thread data is properly cleaned up. This function starts the watcher + thread if necessary. +*/ +void qt_watch_adopted_thread(const HANDLE adoptedThreadHandle, QThread *qthread) +{ + QMutexLocker lock(&qt_adopted_thread_watcher_mutex); + + if (GetCurrentThreadId() == qt_adopted_thread_watcher_id) { + CloseHandle(adoptedThreadHandle); + return; + } + + qt_adopted_thread_handles.append(adoptedThreadHandle); + qt_adopted_qthreads.append(qthread); + + // Start watcher thread if it is not already running. + if (qt_adopted_thread_watcher_id == 0) { + if (qt_adopted_thread_wakeup == 0) { + qt_adopted_thread_wakeup = CreateEvent(0, false, false, 0); + qt_adopted_thread_handles.prepend(qt_adopted_thread_wakeup); + } + + CloseHandle(CreateThread(0, 0, qt_adopted_thread_watcher_function, 0, 0, &qt_adopted_thread_watcher_id)); + } else { + SetEvent(qt_adopted_thread_wakeup); + } +} + +/* + This function loops and waits for native adopted threads to finish. + When this happens it derefs the QThreadData for the adopted thread + to make sure it gets cleaned up properly. +*/ +DWORD WINAPI qt_adopted_thread_watcher_function(LPVOID) +{ + forever { + qt_adopted_thread_watcher_mutex.lock(); + + if (qt_adopted_thread_handles.count() == 1) { + qt_adopted_thread_watcher_id = 0; + qt_adopted_thread_watcher_mutex.unlock(); + break; + } + + QList handlesCopy = qt_adopted_thread_handles; + qt_adopted_thread_watcher_mutex.unlock(); + + DWORD ret = WAIT_TIMEOUT; + int count; + int offset; + int loops = handlesCopy.size() / MAXIMUM_WAIT_OBJECTS; + if (handlesCopy.size() % MAXIMUM_WAIT_OBJECTS) + ++loops; + if (loops == 1) { + // no need to loop, no timeout + offset = 0; + count = handlesCopy.count(); + ret = WaitForMultipleObjects(handlesCopy.count(), handlesCopy.constData(), false, INFINITE); + } else { + int loop = 0; + do { + offset = loop * MAXIMUM_WAIT_OBJECTS; + count = qMin(handlesCopy.count() - offset, MAXIMUM_WAIT_OBJECTS); + ret = WaitForMultipleObjects(count, handlesCopy.constData() + offset, false, 100); + loop = (loop + 1) % loops; + } while (ret == WAIT_TIMEOUT); + } + + if (ret == WAIT_FAILED || ret >= WAIT_OBJECT_0 + uint(count)) { + qWarning("QThread internal error while waiting for adopted threads: %d", int(GetLastError())); + continue; + } + + const int handleIndex = offset + ret - WAIT_OBJECT_0; + if (handleIndex == 0) // New handle to watch was added. + continue; + const int qthreadIndex = handleIndex - 1; + + qt_adopted_thread_watcher_mutex.lock(); + QThreadData *data = QThreadData::get2(qt_adopted_qthreads.at(qthreadIndex)); + qt_adopted_thread_watcher_mutex.unlock(); + if (data->isAdopted) { + QThread *thread = data->thread; + Q_ASSERT(thread); + auto thread_p = static_cast(QObjectPrivate::get(thread)); + Q_UNUSED(thread_p); + Q_ASSERT(!thread_p->finished); + QThreadPrivate::finish(thread); + } + data->deref(); + + QMutexLocker lock(&qt_adopted_thread_watcher_mutex); + CloseHandle(qt_adopted_thread_handles.at(handleIndex)); + qt_adopted_thread_handles.remove(handleIndex); + qt_adopted_qthreads.remove(qthreadIndex); + } + + QThreadData *threadData = reinterpret_cast(TlsGetValue(qt_current_thread_data_tls_index)); + if (threadData) + threadData->deref(); + + return 0; +} + +#ifndef Q_OS_WIN64 +# define ULONG_PTR DWORD +#endif + +typedef struct tagTHREADNAME_INFO +{ + DWORD dwType; // must be 0x1000 + LPCSTR szName; // pointer to name (in user addr space) + HANDLE dwThreadID; // thread ID (-1=caller thread) + DWORD dwFlags; // reserved for future use, must be zero +} THREADNAME_INFO; + +typedef HRESULT(WINAPI* SetThreadDescriptionFunc)(HANDLE, PCWSTR); + +// Helper function to set the thread name using RaiseException and __try +static void setThreadNameUsingException(HANDLE threadId, LPCSTR threadName) { + THREADNAME_INFO info; + info.dwType = 0x1000; + info.szName = threadName; + info.dwThreadID = threadId; + info.dwFlags = 0; + + __try { + RaiseException(0x406D1388, 0, sizeof(info) / sizeof(DWORD), + reinterpret_cast(&info)); + } + __except (EXCEPTION_CONTINUE_EXECUTION) { + } +} + +void qt_set_thread_name(HANDLE threadId, const QString &name) +{ + HMODULE hKernel32 = GetModuleHandleW(L"Kernel32.dll"); + + SetThreadDescriptionFunc pSetThreadDescription = + reinterpret_cast(GetProcAddress(hKernel32, "SetThreadDescription")); + + if (pSetThreadDescription) { + pSetThreadDescription(threadId, reinterpret_cast(name.utf16()) ); + } + else { + + #if defined(Q_CC_MSVC) + + std::string stdStr = name.toStdString(); + LPCSTR threadName = stdStr.c_str(); + setThreadNameUsingException(threadId, threadName); + + #endif // Q_CC_MSVC + } +} + + +/************************************************************************** + ** QThreadPrivate + *************************************************************************/ + +#endif // QT_CONFIG(thread) + +QAbstractEventDispatcher *QThreadPrivate::createEventDispatcher(QThreadData *data) +{ + Q_UNUSED(data); + return new QEventDispatcherWin32; +} + +#if QT_CONFIG(thread) + +unsigned int __stdcall QT_ENSURE_STACK_ALIGNED_FOR_SSE QThreadPrivate::start(void *arg) noexcept +{ + QThread *thr = reinterpret_cast(arg); + QThreadData *data = QThreadData::get2(thr); + + qt_create_tls(); + TlsSetValue(qt_current_thread_data_tls_index, data); + data->threadId.storeRelaxed(reinterpret_cast(quintptr(GetCurrentThreadId()))); + + QThread::setTerminationEnabled(false); + + { + QMutexLocker locker(&thr->d_func()->mutex); + data->quitNow = thr->d_func()->exited; + } + + data->ensureEventDispatcher(); + data->eventDispatcher.loadRelaxed()->startingUp(); + + // sets the name of the current thread. + QString threadName = std::exchange(thr->d_func()->objectName, {}); + if (Q_LIKELY(threadName.isEmpty())) + threadName = QString::fromUtf8(thr->metaObject()->className()); + qt_set_thread_name(GetCurrentThread(), threadName); + + emit thr->started(QThread::QPrivateSignal()); + QThread::setTerminationEnabled(true); + thr->run(); + + finish(arg); + return 0; +} + +/* + For regularly terminating threads, this will be called and executed by the thread as the + last code before the thread exits. In that case, \a arg is the current QThread. + + However, this function will also be called by QThread::terminate (as well as wait() and + setTerminationEnabled) to give Qt a chance to update the terminated thread's state and + process pending DeleteLater events for objects that live in the terminated thread. And for + adopted thread, this method is called by the thread watcher. + + In those cases, \a arg will not be the current thread. +*/ +void QThreadPrivate::finish(void *arg, bool lockAnyway) noexcept +{ + QThread *thr = reinterpret_cast(arg); + QThreadPrivate *d = thr->d_func(); + + QMutexLocker locker(lockAnyway ? &d->mutex : nullptr); + d->isInFinish = true; + d->priority = QThread::InheritPriority; + void **tls_data = reinterpret_cast(&d->data->tls); + if (lockAnyway) + locker.unlock(); + emit thr->finished(QThread::QPrivateSignal()); + qCDebug(lcDeleteLater) << "Sending deferred delete events as part of finishing thread" << thr; + QCoreApplicationPrivate::sendPostedEvents(nullptr, QEvent::DeferredDelete, d->data); + QThreadStorageData::finish(tls_data); + if (lockAnyway) + locker.relock(); + + QAbstractEventDispatcher *eventDispatcher = d->data->eventDispatcher.loadRelaxed(); + if (eventDispatcher) { + d->data->eventDispatcher = 0; + if (lockAnyway) + locker.unlock(); + eventDispatcher->closingDown(); + delete eventDispatcher; + if (lockAnyway) + locker.relock(); + } + + d->running = false; + d->finished = true; + d->isInFinish = false; + d->interruptionRequested.store(false, std::memory_order_relaxed); + + if (!d->waiters) { + CloseHandle(d->handle); + d->handle = 0; + } + + d->id = 0; +} + +/************************************************************************** + ** QThread + *************************************************************************/ + +Qt::HANDLE QThread::currentThreadIdImpl() noexcept +{ + return reinterpret_cast(quintptr(GetCurrentThreadId())); +} + +int QThread::idealThreadCount() noexcept +{ + SYSTEM_INFO sysinfo; + GetSystemInfo(&sysinfo); + return sysinfo.dwNumberOfProcessors; +} + +void QThread::yieldCurrentThread() +{ + SwitchToThread(); +} + +#endif // QT_CONFIG(thread) + +void QThread::sleep(std::chrono::nanoseconds nsecs) +{ + using namespace std::chrono; + ::Sleep(DWORD(duration_cast(nsecs).count())); +} + +void QThread::sleep(unsigned long secs) +{ + ::Sleep(secs * 1000); +} + +void QThread::msleep(unsigned long msecs) +{ + ::Sleep(msecs); +} + +void QThread::usleep(unsigned long usecs) +{ + ::Sleep((usecs / 1000) + 1); +} + +#if QT_CONFIG(thread) + +void QThread::start(Priority priority) +{ + Q_D(QThread); + QMutexLocker locker(&d->mutex); + + if (d->isInFinish) { + locker.unlock(); + wait(); + locker.relock(); + } + + if (d->running) + return; + + // avoid interacting with the binding system + d->objectName = d->extraData ? d->extraData->objectName.valueBypassingBindings() + : QString(); + d->running = true; + d->finished = false; + d->exited = false; + d->returnCode = 0; + d->interruptionRequested.store(false, std::memory_order_relaxed); + + /* + NOTE: we create the thread in the suspended state, set the + priority and then resume the thread. + + since threads are created with normal priority by default, we + could get into a case where a thread (with priority less than + NormalPriority) tries to create a new thread (also with priority + less than NormalPriority), but the newly created thread preempts + its 'parent' and runs at normal priority. + */ +#if defined(Q_CC_MSVC) && !defined(_DLL) + // MSVC -MT or -MTd build + d->handle = (Qt::HANDLE) _beginthreadex(NULL, d->stackSize, QThreadPrivate::start, + this, CREATE_SUSPENDED, &(d->id)); +#else + // MSVC -MD or -MDd or MinGW build + d->handle = CreateThread(nullptr, d->stackSize, + reinterpret_cast(QThreadPrivate::start), + this, CREATE_SUSPENDED, reinterpret_cast(&d->id)); +#endif + + if (!d->handle) { + qErrnoWarning("QThread::start: Failed to create thread"); + d->running = false; + d->finished = true; + return; + } + + int prio; + d->priority = priority; + switch (priority) { + case IdlePriority: + prio = THREAD_PRIORITY_IDLE; + break; + + case LowestPriority: + prio = THREAD_PRIORITY_LOWEST; + break; + + case LowPriority: + prio = THREAD_PRIORITY_BELOW_NORMAL; + break; + + case NormalPriority: + prio = THREAD_PRIORITY_NORMAL; + break; + + case HighPriority: + prio = THREAD_PRIORITY_ABOVE_NORMAL; + break; + + case HighestPriority: + prio = THREAD_PRIORITY_HIGHEST; + break; + + case TimeCriticalPriority: + prio = THREAD_PRIORITY_TIME_CRITICAL; + break; + + case InheritPriority: + default: + prio = GetThreadPriority(GetCurrentThread()); + break; + } + + if (!SetThreadPriority(d->handle, prio)) { + qErrnoWarning("QThread::start: Failed to set thread priority"); + } + + if (ResumeThread(d->handle) == (DWORD) -1) { + qErrnoWarning("QThread::start: Failed to resume new thread"); + } +} + +void QThread::terminate() +{ + Q_D(QThread); + QMutexLocker locker(&d->mutex); + if (!d->running) + return; + if (!d->terminationEnabled) { + d->terminatePending = true; + return; + } + + TerminateThread(d->handle, 0); + QThreadPrivate::finish(this, false); +} + +bool QThread::wait(QDeadlineTimer deadline) +{ + Q_D(QThread); + QMutexLocker locker(&d->mutex); + + if (d->id == GetCurrentThreadId()) { + qWarning("QThread::wait: Thread tried to wait on itself"); + return false; + } + if (d->finished || !d->running) + return true; + + ++d->waiters; + locker.mutex()->unlock(); + + bool ret = false; + switch (WaitForSingleObject(d->handle, deadline.remainingTime())) { + case WAIT_OBJECT_0: + ret = true; + break; + case WAIT_FAILED: + qErrnoWarning("QThread::wait: Thread wait failure"); + break; + case WAIT_ABANDONED: + case WAIT_TIMEOUT: + default: + break; + } + + locker.mutex()->lock(); + --d->waiters; + + if (ret && !d->finished) { + // thread was terminated by someone else + + QThreadPrivate::finish(this, false); + } + + if (d->finished && !d->waiters) { + CloseHandle(d->handle); + d->handle = 0; + } + + return ret; +} + +void QThread::setTerminationEnabled(bool enabled) +{ + QThread *thr = currentThread(); + Q_ASSERT_X(thr != 0, "QThread::setTerminationEnabled()", + "Current thread was not started with QThread."); + QThreadPrivate *d = thr->d_func(); + QMutexLocker locker(&d->mutex); + d->terminationEnabled = enabled; + if (enabled && d->terminatePending) { + QThreadPrivate::finish(thr, false); + locker.unlock(); // don't leave the mutex locked! + _endthreadex(0); + } +} + +// Caller must hold the mutex +void QThreadPrivate::setPriority(QThread::Priority threadPriority) +{ + // copied from start() with a few modifications: + + int prio; + priority = threadPriority; + switch (threadPriority) { + case QThread::IdlePriority: + prio = THREAD_PRIORITY_IDLE; + break; + + case QThread::LowestPriority: + prio = THREAD_PRIORITY_LOWEST; + break; + + case QThread::LowPriority: + prio = THREAD_PRIORITY_BELOW_NORMAL; + break; + + case QThread::NormalPriority: + prio = THREAD_PRIORITY_NORMAL; + break; + + case QThread::HighPriority: + prio = THREAD_PRIORITY_ABOVE_NORMAL; + break; + + case QThread::HighestPriority: + prio = THREAD_PRIORITY_HIGHEST; + break; + + case QThread::TimeCriticalPriority: + prio = THREAD_PRIORITY_TIME_CRITICAL; + break; + + default: + return; + } + + if (!SetThreadPriority(handle, prio)) { + qErrnoWarning("QThread::setPriority: Failed to set thread priority"); + } +} + +#endif // QT_CONFIG(thread) + +QT_END_NAMESPACE diff --git a/qtbase/src/gui/rhi/qrhid3d11.cpp b/qtbase/src/gui/rhi/qrhid3d11.cpp index 4f015668..2a4e0666 100644 --- a/qtbase/src/gui/rhi/qrhid3d11.cpp +++ b/qtbase/src/gui/rhi/qrhid3d11.cpp @@ -10,6 +10,8 @@ #include #include "qrhid3dhelpers_p.h" +#include + #include QT_BEGIN_NAMESPACE @@ -28,7 +30,8 @@ using namespace Qt::StringLiterals; /*! \class QRhiD3D11InitParams - \inmodule QtGui + \inmodule QtGuiPrivate + \inheaderfile rhi/qrhi.h \since 6.6 \brief Direct3D 11 specific initialization parameters. @@ -81,7 +84,8 @@ using namespace Qt::StringLiterals; /*! \class QRhiD3D11NativeHandles - \inmodule QtGui + \inmodule QtGuiPrivate + \inheaderfile rhi/qrhi.h \since 6.6 \brief Holds the D3D device and device context used by the QRhi. @@ -233,9 +237,19 @@ bool QRhiD3D11::create(QRhi::Flags flags) // there. (some features are not supported then, however) useLegacySwapchainModel = qEnvironmentVariableIntValue("QT_D3D_NO_FLIP"); - qCDebug(QRHI_LOG_INFO, "FLIP_* swapchain supported = true, ALLOW_TEARING supported = %s, use legacy (non-FLIP) model = %s", + if (!useLegacySwapchainModel) { + if (qEnvironmentVariableIsSet("QT_D3D_MAX_FRAME_LATENCY")) + maxFrameLatency = UINT(qMax(0, qEnvironmentVariableIntValue("QT_D3D_MAX_FRAME_LATENCY"))); + } else { + maxFrameLatency = 0; + } + + qCDebug(QRHI_LOG_INFO, "FLIP_* swapchain supported = true, ALLOW_TEARING supported = %s, use legacy (non-FLIP) model = %s, max frame latency = %u", supportsAllowTearing ? "true" : "false", - useLegacySwapchainModel ? "true" : "false"); + useLegacySwapchainModel ? "true" : "false", + maxFrameLatency); + if (maxFrameLatency == 0) + qCDebug(QRHI_LOG_INFO, "Disabling FRAME_LATENCY_WAITABLE_OBJECT usage"); if (!importedDeviceAndContext) { IDXGIAdapter1 *adapter; @@ -634,6 +648,10 @@ bool QRhiD3D11::isFeatureSupported(QRhi::Feature feature) const return true; case QRhi::MultiView: return false; + case QRhi::TextureViewFormat: + return false; // because we use fully typed formats for textures and relaxed casting is a D3D12 thing + case QRhi::ResolveDepthStencil: + return false; default: Q_UNREACHABLE(); return false; @@ -1347,6 +1365,10 @@ QRhi::FrameOpResult QRhiD3D11::beginFrame(QRhiSwapChain *swapChain, QRhi::BeginF contextState.currentSwapChain = swapChainD; const int currentFrameSlot = swapChainD->currentFrameSlot; + // if we have a waitable object, now is the time to wait on it + if (swapChainD->frameLatencyWaitableObject) + WaitForSingleObjectEx(swapChainD->frameLatencyWaitableObject, 1000, true); + swapChainD->cb.resetState(); swapChainD->rt.d.rtv[0] = swapChainD->sampleDesc.Count > 1 ? @@ -2153,6 +2175,8 @@ void QRhiD3D11::endPass(QRhiCommandBuffer *cb, QRhiResourceUpdateBatch *resource cmd.args.resolveSubRes.srcSubRes = D3D11CalcSubresource(0, UINT(colorAtt.layer()), 1); cmd.args.resolveSubRes.format = dstTexD->dxgiFormat; } + if (rtTex->m_desc.depthResolveTexture()) + qWarning("Resolving multisample depth-stencil buffers is not supported with D3D"); } cbD->recordingPass = QD3D11CommandBuffer::NoPass; @@ -4250,6 +4274,22 @@ static inline DXGI_FORMAT toD3DAttributeFormat(QRhiVertexInputAttribute::Format return DXGI_FORMAT_R16G16_FLOAT; case QRhiVertexInputAttribute::Half: return DXGI_FORMAT_R16_FLOAT; + case QRhiVertexInputAttribute::UShort4: + // Note: D3D does not support UShort3. Pass through UShort3 as UShort4. + case QRhiVertexInputAttribute::UShort3: + return DXGI_FORMAT_R16G16B16A16_UINT; + case QRhiVertexInputAttribute::UShort2: + return DXGI_FORMAT_R16G16_UINT; + case QRhiVertexInputAttribute::UShort: + return DXGI_FORMAT_R16_UINT; + case QRhiVertexInputAttribute::SShort4: + // Note: D3D does not support SShort3. Pass through SShort3 as SShort4. + case QRhiVertexInputAttribute::SShort3: + return DXGI_FORMAT_R16G16B16A16_SINT; + case QRhiVertexInputAttribute::SShort2: + return DXGI_FORMAT_R16G16_SINT; + case QRhiVertexInputAttribute::SShort: + return DXGI_FORMAT_R16_SINT; default: Q_UNREACHABLE(); return DXGI_FORMAT_R32G32B32A32_FLOAT; @@ -4672,7 +4712,7 @@ bool QD3D11GraphicsPipeline::create() } else { QByteArray sem; sem.resize(16); - qsnprintf(sem.data(), sem.size(), "TEXCOORD%d_", it->location() - matrixSlice); + std::snprintf(sem.data(), sem.size(), "TEXCOORD%d_", it->location() - matrixSlice); matrixSliceSemantics.append(sem); desc.SemanticName = matrixSliceSemantics.last().constData(); desc.SemanticIndex = UINT(matrixSlice); @@ -4938,6 +4978,11 @@ void QD3D11SwapChain::destroy() dcompTarget = nullptr; } + if (frameLatencyWaitableObject) { + CloseHandle(frameLatencyWaitableObject); + frameLatencyWaitableObject = nullptr; + } + QRHI_RES_RHI(QRhiD3D11); if (rhiD) { rhiD->unregisterResource(this); @@ -5100,7 +5145,7 @@ bool QD3D11SwapChain::createOrResize() if (m_flags.testFlag(SurfaceHasPreMulAlpha) || m_flags.testFlag(SurfaceHasNonPreMulAlpha)) { if (!rhiD->useLegacySwapchainModel && rhiD->ensureDirectCompositionDevice()) { if (!dcompTarget) { - hr = rhiD->dcompDevice->CreateTargetForHwnd(hwnd, true, &dcompTarget); + hr = rhiD->dcompDevice->CreateTargetForHwnd(hwnd, false, &dcompTarget); if (FAILED(hr)) { qWarning("Failed to create Direct Compsition target for the window: %s", qPrintable(QSystemError::windowsComString(hr))); @@ -5130,6 +5175,17 @@ bool QD3D11SwapChain::createOrResize() if (swapInterval == 0 && rhiD->supportsAllowTearing) swapChainFlags |= DXGI_SWAP_CHAIN_FLAG_ALLOW_TEARING; + // maxFrameLatency 0 means no waitable object usage. + // Ignore it also when NoVSync is on, and when using WARP. + const bool useFrameLatencyWaitableObject = rhiD->maxFrameLatency != 0 + && swapInterval != 0 + && rhiD->driverInfoStruct.deviceType != QRhiDriverInfo::CpuDevice; + + if (useFrameLatencyWaitableObject) { + // the flag is not supported in real fullscreen on D3D11, but perhaps that's fine since we only do borderless + swapChainFlags |= DXGI_SWAP_CHAIN_FLAG_FRAME_LATENCY_WAITABLE_OBJECT; + } + if (!swapChain) { sampleDesc = rhiD->effectiveSampleDesc(m_sampleCount); colorFormat = DEFAULT_FORMAT; @@ -5215,16 +5271,31 @@ bool QD3D11SwapChain::createOrResize() if (SUCCEEDED(hr)) { swapChain = sc1; - if (m_format != SDR) { - IDXGISwapChain3 *sc3 = nullptr; - if (SUCCEEDED(sc1->QueryInterface(__uuidof(IDXGISwapChain3), reinterpret_cast(&sc3)))) { + IDXGISwapChain3 *sc3 = nullptr; + if (SUCCEEDED(sc1->QueryInterface(__uuidof(IDXGISwapChain3), reinterpret_cast(&sc3)))) { + if (m_format != SDR) { hr = sc3->SetColorSpace1(hdrColorSpace); if (FAILED(hr)) qWarning("Failed to set color space on swapchain: %s", - qPrintable(QSystemError::windowsComString(hr))); - sc3->Release(); - } else { + qPrintable(QSystemError::windowsComString(hr))); + } + if (useFrameLatencyWaitableObject) { + sc3->SetMaximumFrameLatency(rhiD->maxFrameLatency); + frameLatencyWaitableObject = sc3->GetFrameLatencyWaitableObject(); + } + sc3->Release(); + } else { + if (m_format != SDR) qWarning("IDXGISwapChain3 not available, HDR swapchain will not work as expected"); + if (useFrameLatencyWaitableObject) { + IDXGISwapChain2 *sc2 = nullptr; + if (SUCCEEDED(sc1->QueryInterface(__uuidof(IDXGISwapChain2), reinterpret_cast(&sc2)))) { + sc2->SetMaximumFrameLatency(rhiD->maxFrameLatency); + frameLatencyWaitableObject = sc2->GetFrameLatencyWaitableObject(); + sc2->Release(); + } else { // this cannot really happen since we require DXGIFactory2 + qWarning("IDXGISwapChain2 not available, FrameLatencyWaitableObject cannot be used"); + } } } if (dcompVisual) { diff --git a/qtbase/src/gui/rhi/qrhid3d11_p.h b/qtbase/src/gui/rhi/qrhid3d11_p.h index 86b810cc..7f232f1a 100644 --- a/qtbase/src/gui/rhi/qrhid3d11_p.h +++ b/qtbase/src/gui/rhi/qrhid3d11_p.h @@ -626,6 +626,7 @@ struct QD3D11SwapChain : public QRhiSwapChain IDCompositionVisual *dcompVisual = nullptr; QD3D11SwapChainTimestamps timestamps; int currentTimestampPairIndex = 0; + HANDLE frameLatencyWaitableObject = nullptr; }; class QRhiD3D11 : public QRhiImplementation @@ -761,6 +762,7 @@ public: QRhi::Flags rhiFlags; bool debugLayer = false; + UINT maxFrameLatency = 2; // 1-3, use 2 to keep CPU-GPU parallelism while reducing lag compared to tripple buffering bool importedDeviceAndContext = false; ID3D11Device *dev = nullptr; ID3D11DeviceContext1 *context = nullptr; diff --git a/qtbase/src/gui/rhi/qrhid3d12.cpp b/qtbase/src/gui/rhi/qrhid3d12.cpp index 4e8dba9c..d7595c73 100644 --- a/qtbase/src/gui/rhi/qrhid3d12.cpp +++ b/qtbase/src/gui/rhi/qrhid3d12.cpp @@ -23,7 +23,8 @@ QT_BEGIN_NAMESPACE /*! \class QRhiD3D12InitParams - \inmodule QtGui + \inmodule QtGuiPrivate + \inheaderfile rhi/qrhi.h \brief Direct3D 12 specific initialization parameters. \note This is a RHI API with limited compatibility guarantees, see \l QRhi @@ -70,7 +71,8 @@ QT_BEGIN_NAMESPACE /*! \class QRhiD3D12NativeHandles - \inmodule QtGui + \inmodule QtGuiPrivate + \inheaderfile rhi/qrhi.h \brief Holds the D3D12 device used by the QRhi. \note The class uses \c{void *} as the type since including the COM-based @@ -129,7 +131,8 @@ QT_BEGIN_NAMESPACE /*! \class QRhiD3D12CommandBufferNativeHandles - \inmodule QtGui + \inmodule QtGuiPrivate + \inheaderfile rhi/qrhi.h \brief Holds the ID3D12GraphicsCommandList1 object that is backing a QRhiCommandBuffer. \note The command list object is only guaranteed to be valid, and @@ -239,6 +242,11 @@ bool QRhiD3D12::create(QRhi::Flags flags) } } + if (qEnvironmentVariableIsSet("QT_D3D_MAX_FRAME_LATENCY")) + maxFrameLatency = UINT(qMax(0, qEnvironmentVariableIntValue("QT_D3D_MAX_FRAME_LATENCY"))); + if (maxFrameLatency != 0) + qCDebug(QRHI_LOG_INFO, "Using frame latency waitable object with max frame latency %u", maxFrameLatency); + supportsAllowTearing = false; IDXGIFactory5 *factory5 = nullptr; if (SUCCEEDED(dxgiFactory->QueryInterface(__uuidof(IDXGIFactory5), reinterpret_cast(&factory5)))) { @@ -501,9 +509,13 @@ bool QRhiD3D12::create(QRhi::Flags flags) memset(timestampReadbackArea.mem.p, 0, readbackBufSize); } + caps = {}; D3D12_FEATURE_DATA_D3D12_OPTIONS3 options3 = {}; - if (SUCCEEDED(dev->CheckFeatureSupport(D3D12_FEATURE_D3D12_OPTIONS3, &options3, sizeof(options3)))) + if (SUCCEEDED(dev->CheckFeatureSupport(D3D12_FEATURE_D3D12_OPTIONS3, &options3, sizeof(options3)))) { caps.multiView = options3.ViewInstancingTier != D3D12_VIEW_INSTANCING_TIER_NOT_SUPPORTED; + // https://microsoft.github.io/DirectX-Specs/d3d/RelaxedCasting.html + caps.textureViewFormat = options3.CastingFullyTypedFormatSupported; + } deviceLost = false; offscreenActive = false; @@ -748,6 +760,12 @@ bool QRhiD3D12::isFeatureSupported(QRhi::Feature feature) const return false; // we generate mipmaps ourselves with compute and this is not implemented case QRhi::MultiView: return caps.multiView; + case QRhi::TextureViewFormat: + return caps.textureViewFormat; + case QRhi::ResolveDepthStencil: + // there is no Multisample Resolve support for depth/stencil formats + // https://learn.microsoft.com/en-us/windows/win32/direct3ddxgi/hardware-support-for-direct3d-12-1-formats + return false; } return false; } @@ -974,7 +992,7 @@ void QD3D12CommandBuffer::visitStorageImage(QD3D12Stage s, const bool isArray = texD->m_flags.testFlag(QRhiTexture::TextureArray); const bool is3D = texD->m_flags.testFlag(QRhiTexture::ThreeDimensional); D3D12_UNORDERED_ACCESS_VIEW_DESC uavDesc = {}; - uavDesc.Format = texD->dxgiFormat; + uavDesc.Format = texD->rtFormat; if (isCube) { uavDesc.ViewDimension = D3D12_UAV_DIMENSION_TEXTURE2DARRAY; uavDesc.Texture2DArray.MipSlice = UINT(d.level); @@ -1533,6 +1551,9 @@ QRhi::FrameOpResult QRhiD3D12::beginFrame(QRhiSwapChain *swapChain, QRhi::BeginF for (QD3D12SwapChain *sc : std::as_const(swapchains)) sc->waitCommandCompletionForFrameSlot(currentFrameSlot); // note: swapChainD->currentFrameSlot, not sc's + if (swapChainD->frameLatencyWaitableObject) + WaitForSingleObjectEx(swapChainD->frameLatencyWaitableObject, 1000, true); + HRESULT hr = cmdAllocators[currentFrameSlot]->Reset(); if (FAILED(hr)) { qWarning("Failed to reset command allocator: %s", @@ -1996,7 +2017,8 @@ void QRhiD3D12::endPass(QRhiCommandBuffer *cb, QRhiResourceUpdateBatch *resource dstTexD->dxgiFormat); } } - + if (rtTex->m_desc.depthResolveTexture()) + qWarning("Resolving multisample depth-stencil buffers is not supported with D3D"); } cbD->recordingPass = QD3D12CommandBuffer::NoPass; @@ -4198,8 +4220,27 @@ bool QD3D12Texture::prepareCreate(QSize *adjustedSize) const QSize size = is1D ? QSize(qMax(1, m_pixelSize.width()), 1) : (m_pixelSize.isEmpty() ? QSize(1, 1) : m_pixelSize); - QRHI_RES_RHI(QRhiD3D12); dxgiFormat = toD3DTextureFormat(m_format, m_flags); + if (isDepth) { + srvFormat = toD3DDepthTextureSRVFormat(m_format); + rtFormat = toD3DDepthTextureDSVFormat(m_format); + } else { + srvFormat = dxgiFormat; + rtFormat = dxgiFormat; + } + if (m_writeViewFormat.format != UnknownFormat) { + if (isDepth) + rtFormat = toD3DDepthTextureDSVFormat(m_writeViewFormat.format); + else + rtFormat = toD3DTextureFormat(m_writeViewFormat.format, m_writeViewFormat.srgb ? sRGB : Flags()); + } + if (m_readViewFormat.format != UnknownFormat) { + if (isDepth) + srvFormat = toD3DDepthTextureSRVFormat(m_readViewFormat.format); + else + srvFormat = toD3DTextureFormat(m_readViewFormat.format, m_readViewFormat.srgb ? sRGB : Flags()); + } + QRHI_RES_RHI(QRhiD3D12); mipLevelCount = uint(hasMipMaps ? rhiD->q->mipLevelsForSize(size) : 1); sampleDesc = rhiD->effectiveSampleDesc(m_sampleCount, dxgiFormat); if (sampleDesc.Count > 1) { @@ -4258,14 +4299,13 @@ bool QD3D12Texture::prepareCreate(QSize *adjustedSize) bool QD3D12Texture::finishCreate() { QRHI_RES_RHI(QRhiD3D12); - const bool isDepth = isDepthTextureFormat(m_format); const bool isCube = m_flags.testFlag(CubeMap); const bool is3D = m_flags.testFlag(ThreeDimensional); const bool isArray = m_flags.testFlag(TextureArray); const bool is1D = m_flags.testFlag(OneDimensional); D3D12_SHADER_RESOURCE_VIEW_DESC srvDesc = {}; - srvDesc.Format = isDepth ? toD3DDepthTextureSRVFormat(m_format) : dxgiFormat; + srvDesc.Format = srvFormat; srvDesc.Shader4ComponentMapping = D3D12_DEFAULT_SHADER_4_COMPONENT_MAPPING; if (isCube) { @@ -4624,7 +4664,7 @@ QRhiRenderPassDescriptor *QD3D12TextureRenderTarget::newCompatibleRenderPassDesc QD3D12Texture *texD = QRHI_RES(QD3D12Texture, it->texture()); QD3D12RenderBuffer *rbD = QRHI_RES(QD3D12RenderBuffer, it->renderBuffer()); if (texD) - rpD->colorFormat[rpD->colorAttachmentCount] = texD->dxgiFormat; + rpD->colorFormat[rpD->colorAttachmentCount] = texD->rtFormat; else if (rbD) rpD->colorFormat[rpD->colorAttachmentCount] = rbD->dxgiFormat; rpD->colorAttachmentCount += 1; @@ -4675,7 +4715,7 @@ bool QD3D12TextureRenderTarget::create() const bool isMultiView = it->multiViewCount() >= 2; UINT layerCount = isMultiView ? UINT(it->multiViewCount()) : 1; D3D12_RENDER_TARGET_VIEW_DESC rtvDesc = {}; - rtvDesc.Format = toD3DTextureFormat(texD->format(), texD->flags()); + rtvDesc.Format = texD->rtFormat; if (texD->flags().testFlag(QRhiTexture::CubeMap)) { rtvDesc.ViewDimension = D3D12_RTV_DIMENSION_TEXTURE2DARRAY; rtvDesc.Texture2DArray.MipSlice = UINT(colorAtt.level()); @@ -4749,7 +4789,7 @@ bool QD3D12TextureRenderTarget::create() return false; } D3D12_DEPTH_STENCIL_VIEW_DESC dsvDesc = {}; - dsvDesc.Format = toD3DDepthTextureDSVFormat(depthTexD->format()); + dsvDesc.Format = depthTexD->rtFormat; dsvDesc.ViewDimension = depthTexD->sampleDesc.Count > 1 ? D3D12_DSV_DIMENSION_TEXTURE2DMS : D3D12_DSV_DIMENSION_TEXTURE2D; if (depthTexD->flags().testFlag(QRhiTexture::TextureArray)) { @@ -5545,6 +5585,22 @@ static inline DXGI_FORMAT toD3DAttributeFormat(QRhiVertexInputAttribute::Format return DXGI_FORMAT_R16G16_FLOAT; case QRhiVertexInputAttribute::Half: return DXGI_FORMAT_R16_FLOAT; + case QRhiVertexInputAttribute::UShort4: + // Note: D3D does not support UShort3. Pass through UShort3 as UShort4. + case QRhiVertexInputAttribute::UShort3: + return DXGI_FORMAT_R16G16B16A16_UINT; + case QRhiVertexInputAttribute::UShort2: + return DXGI_FORMAT_R16G16_UINT; + case QRhiVertexInputAttribute::UShort: + return DXGI_FORMAT_R16_UINT; + case QRhiVertexInputAttribute::SShort4: + // Note: D3D does not support SShort3. Pass through SShort3 as SShort4. + case QRhiVertexInputAttribute::SShort3: + return DXGI_FORMAT_R16G16B16A16_SINT; + case QRhiVertexInputAttribute::SShort2: + return DXGI_FORMAT_R16G16_SINT; + case QRhiVertexInputAttribute::SShort: + return DXGI_FORMAT_R16_SINT; } Q_UNREACHABLE_RETURN(DXGI_FORMAT_R32G32B32A32_FLOAT); } @@ -5678,7 +5734,7 @@ bool QD3D12GraphicsPipeline::create() } else { QByteArray sem; sem.resize(16); - qsnprintf(sem.data(), sem.size(), "TEXCOORD%d_", it->location() - matrixSlice); + std::snprintf(sem.data(), sem.size(), "TEXCOORD%d_", it->location() - matrixSlice); matrixSliceSemantics.append(sem); desc.SemanticName = matrixSliceSemantics.last().constData(); desc.SemanticIndex = UINT(matrixSlice); @@ -6105,6 +6161,11 @@ void QD3D12SwapChain::destroy() dcompTarget = nullptr; } + if (frameLatencyWaitableObject) { + CloseHandle(frameLatencyWaitableObject); + frameLatencyWaitableObject = nullptr; + } + QRHI_RES_RHI(QRhiD3D12); if (rhiD) { rhiD->swapchains.remove(this); @@ -6296,7 +6357,7 @@ bool QD3D12SwapChain::createOrResize() if (m_flags.testFlag(SurfaceHasPreMulAlpha) || m_flags.testFlag(SurfaceHasNonPreMulAlpha)) { if (rhiD->ensureDirectCompositionDevice()) { if (!dcompTarget) { - hr = rhiD->dcompDevice->CreateTargetForHwnd(hwnd, true, &dcompTarget); + hr = rhiD->dcompDevice->CreateTargetForHwnd(hwnd, false, &dcompTarget); if (FAILED(hr)) { qWarning("Failed to create Direct Composition target for the window: %s", qPrintable(QSystemError::windowsComString(hr))); @@ -6321,6 +6382,14 @@ bool QD3D12SwapChain::createOrResize() if (swapInterval == 0 && rhiD->supportsAllowTearing) swapChainFlags |= DXGI_SWAP_CHAIN_FLAG_ALLOW_TEARING; + // maxFrameLatency 0 means no waitable object usage. + // Ignore it also when NoVSync is on, and when using WARP. + const bool useFrameLatencyWaitableObject = rhiD->maxFrameLatency != 0 + && swapInterval != 0 + && rhiD->driverInfoStruct.deviceType != QRhiDriverInfo::CpuDevice; + if (useFrameLatencyWaitableObject) + swapChainFlags |= DXGI_SWAP_CHAIN_FLAG_FRAME_LATENCY_WAITABLE_OBJECT; + if (!swapChain) { chooseFormats(); @@ -6377,6 +6446,10 @@ bool QD3D12SwapChain::createOrResize() qPrintable(QSystemError::windowsComString(hr))); } } + if (useFrameLatencyWaitableObject) { + swapChain->SetMaximumFrameLatency(rhiD->maxFrameLatency); + frameLatencyWaitableObject = swapChain->GetFrameLatencyWaitableObject(); + } if (dcompVisual) { hr = dcompVisual->SetContent(swapChain); if (SUCCEEDED(hr)) { diff --git a/qtbase/src/gui/text/windows/qwindowsfontdatabasebase.cpp b/qtbase/src/gui/text/windows/qwindowsfontdatabasebase.cpp index fe2070a0..e751d4b1 100644 --- a/qtbase/src/gui/text/windows/qwindowsfontdatabasebase.cpp +++ b/qtbase/src/gui/text/windows/qwindowsfontdatabasebase.cpp @@ -565,16 +565,27 @@ void QWindowsFontDatabaseBase::createDirectWriteFactory(IDWriteFactory **factory IUnknown *result = nullptr; # if QT_CONFIG(directwrite3) - DWriteCreateFactory(DWRITE_FACTORY_TYPE_SHARED, __uuidof(IDWriteFactory5), &result); - - if (result == nullptr) - DWriteCreateFactory(DWRITE_FACTORY_TYPE_SHARED, __uuidof(IDWriteFactory3), &result); -# endif - - if (result == nullptr) - DWriteCreateFactory(DWRITE_FACTORY_TYPE_SHARED, __uuidof(IDWriteFactory2), &result); + qCDebug(lcQpaFonts) << "Trying to create IDWriteFactory6"; + DWriteCreateFactory(DWRITE_FACTORY_TYPE_SHARED, __uuidof(IDWriteFactory6), &result); if (result == nullptr) { + qCDebug(lcQpaFonts) << "Trying to create IDWriteFactory5"; + DWriteCreateFactory(DWRITE_FACTORY_TYPE_SHARED, __uuidof(IDWriteFactory5), &result); + } + + if (result == nullptr) { + qCDebug(lcQpaFonts) << "Trying to create IDWriteFactory3"; + DWriteCreateFactory(DWRITE_FACTORY_TYPE_SHARED, __uuidof(IDWriteFactory3), &result); + } +# endif + + if (result == nullptr) { + qCDebug(lcQpaFonts) << "Trying to create IDWriteFactory2"; + DWriteCreateFactory(DWRITE_FACTORY_TYPE_SHARED, __uuidof(IDWriteFactory2), &result); + } + + if (result == nullptr) { + qCDebug(lcQpaFonts) << "Trying to create plain IDWriteFactory"; if (FAILED(DWriteCreateFactory(DWRITE_FACTORY_TYPE_SHARED, __uuidof(IDWriteFactory), &result))) { qErrnoWarning("DWriteCreateFactory failed"); return; diff --git a/qtbase/src/plugins/platforms/windows/qwindowscontext.cpp b/qtbase/src/plugins/platforms/windows/qwindowscontext.cpp index 1d2b8eb9..aa9d6065 100644 --- a/qtbase/src/plugins/platforms/windows/qwindowscontext.cpp +++ b/qtbase/src/plugins/platforms/windows/qwindowscontext.cpp @@ -7,7 +7,6 @@ #include "qwindowswindow.h" #include "qwindowskeymapper.h" #include "qwindowsnativeinterface.h" -#include "qwindowsmousehandler.h" #include "qwindowspointerhandler.h" #include "qtwindowsglobal.h" #include "qwindowsmenu.h" @@ -215,7 +214,6 @@ struct QWindowsContextPrivate { HDC m_displayContext = nullptr; int m_defaultDPI = 96; QWindowsKeyMapper m_keyMapper; - QWindowsMouseHandler m_mouseHandler; QWindowsPointerHandler m_pointerHandler; QWindowsMimeRegistry m_mimeConverter; QWindowsScreenManager m_screenManager; @@ -228,11 +226,9 @@ struct QWindowsContextPrivate { bool m_asyncExpose = false; HPOWERNOTIFY m_powerNotification = nullptr; HWND m_powerDummyWindow = nullptr; - static bool m_darkMode; static bool m_v2DpiAware; }; -bool QWindowsContextPrivate::m_darkMode = false; bool QWindowsContextPrivate::m_v2DpiAware = false; QWindowsContextPrivate::QWindowsContextPrivate() @@ -241,7 +237,7 @@ QWindowsContextPrivate::QWindowsContextPrivate() QWindowsContext::user32dll.init(); QWindowsContext::shcoredll.init(); - if (m_pointerHandler.touchDevice() || m_mouseHandler.touchDevice()) + if (m_pointerHandler.touchDevice()) m_systemInfo |= QWindowsContext::SI_SupportsTouch; m_displayContext = GetDC(nullptr); m_defaultDPI = GetDeviceCaps(m_displayContext, LOGPIXELSY); @@ -249,7 +245,6 @@ QWindowsContextPrivate::QWindowsContextPrivate() m_systemInfo |= QWindowsContext::SI_RTL_Extensions; m_keyMapper.setUseRTLExtensions(true); } - m_darkMode = QWindowsTheme::queryDarkMode(); if (FAILED(m_oleInitializeResult)) { qWarning() << "QWindowsContext: OleInitialize() failed: " << QSystemError::windowsComString(m_oleInitializeResult); @@ -306,8 +301,7 @@ bool QWindowsContext::initTouch(unsigned integrationOptions) { if (d->m_systemInfo & QWindowsContext::SI_SupportsTouch) return true; - const bool usePointerHandler = (d->m_systemInfo & QWindowsContext::SI_SupportsPointer) != 0; - auto touchDevice = usePointerHandler ? d->m_pointerHandler.touchDevice() : d->m_mouseHandler.touchDevice(); + auto touchDevice = d->m_pointerHandler.touchDevice(); if (touchDevice.isNull()) { const bool mouseEmulation = (integrationOptions & QWindowsIntegration::DontPassOsMouseEventsSynthesizedFromTouch) == 0; @@ -316,7 +310,6 @@ bool QWindowsContext::initTouch(unsigned integrationOptions) if (touchDevice.isNull()) return false; d->m_pointerHandler.setTouchDevice(touchDevice); - d->m_mouseHandler.setTouchDevice(touchDevice); QWindowSystemInterface::registerInputDevice(touchDevice.data()); d->m_systemInfo |= QWindowsContext::SI_SupportsTouch; @@ -356,21 +349,6 @@ bool QWindowsContext::disposeTablet() #endif } -bool QWindowsContext::initPointer(unsigned integrationOptions) -{ - if (integrationOptions & QWindowsIntegration::DontUseWMPointer) - return false; - - if (QOperatingSystemVersion::current() < QOperatingSystemVersion::Windows8) - return false; - - if (!QWindowsContext::user32dll.supportsPointerApi()) - return false; - - d->m_systemInfo |= QWindowsContext::SI_SupportsPointer; - return true; -} - extern "C" LRESULT QT_WIN_CALLBACK qWindowsPowerWindowProc(HWND hwnd, UINT message, WPARAM wParam, LPARAM lParam) { if (message != WM_POWERBROADCAST || wParam != PBT_POWERSETTINGCHANGE) @@ -585,11 +563,6 @@ bool QWindowsContext::setProcessDpiAwareness(QtWindows::DpiAwareness dpiAwarenes return true; } -bool QWindowsContext::isDarkMode() -{ - return QWindowsContextPrivate::m_darkMode; -} - QWindowsContext *QWindowsContext::instance() { return m_instance; @@ -857,16 +830,12 @@ QWindow *QWindowsContext::findWindow(HWND hwnd) const QWindow *QWindowsContext::windowUnderMouse() const { - return (d->m_systemInfo & QWindowsContext::SI_SupportsPointer) ? - d->m_pointerHandler.windowUnderMouse() : d->m_mouseHandler.windowUnderMouse(); + return d->m_pointerHandler.windowUnderMouse(); } void QWindowsContext::clearWindowUnderMouse() { - if (d->m_systemInfo & QWindowsContext::SI_SupportsPointer) - d->m_pointerHandler.clearWindowUnderMouse(); - else - d->m_mouseHandler.clearWindowUnderMouse(); + d->m_pointerHandler.clearWindowUnderMouse(); } /*! @@ -1130,9 +1099,10 @@ bool QWindowsContext::windowsProc(HWND hwnd, UINT message, MSG msg; msg.hwnd = hwnd; // re-create MSG structure - msg.message = message; // time and pt fields ignored + msg.message = message; msg.wParam = wParam; msg.lParam = lParam; + msg.time = GetMessageTime(); msg.pt.x = msg.pt.y = 0; if (et != QtWindows::CursorEvent && (et & (QtWindows::MouseEventFlag | QtWindows::NonClientEventFlag))) { msg.pt.x = GET_X_LPARAM(lParam); @@ -1181,8 +1151,7 @@ bool QWindowsContext::windowsProc(HWND hwnd, UINT message, switch (et) { case QtWindows::GestureEvent: - if (!(d->m_systemInfo & QWindowsContext::SI_SupportsPointer)) - return sessionManagerInteractionBlocked() || d->m_mouseHandler.translateGestureEvent(platformWindow->window(), hwnd, et, msg, result); + // TODO??? break; case QtWindows::InputMethodOpenCandidateWindowEvent: case QtWindows::InputMethodCloseCandidateWindowEvent: @@ -1216,21 +1185,7 @@ bool QWindowsContext::windowsProc(HWND hwnd, UINT message, // Only refresh the window theme if the user changes the personalize settings. if ((wParam == 0) && (lParam != 0) // lParam sometimes may be NULL. && (wcscmp(reinterpret_cast(lParam), L"ImmersiveColorSet") == 0)) { - const bool darkMode = QWindowsTheme::queryDarkMode(); - const bool darkModeChanged = darkMode != QWindowsContextPrivate::m_darkMode; - QWindowsContextPrivate::m_darkMode = darkMode; - auto integration = QWindowsIntegration::instance(); - integration->updateApplicationBadge(); - if (integration->darkModeHandling().testFlag(QWindowsApplication::DarkModeStyle)) { - QWindowsTheme::instance()->refresh(); - QWindowSystemInterface::handleThemeChange(); - } - if (darkModeChanged) { - if (integration->darkModeHandling().testFlag(QWindowsApplication::DarkModeWindowFrames)) { - for (QWindowsWindow *w : d->m_windows) - w->setDarkBorder(QWindowsContextPrivate::m_darkMode); - } - } + QWindowsTheme::handleSettingsChanged(); } return d->m_screenManager.handleScreenChanges(); } @@ -1336,16 +1291,11 @@ bool QWindowsContext::windowsProc(HWND hwnd, UINT message, case QtWindows::NonClientMouseEvent: if (!platformWindow->frameStrutEventsEnabled()) break; - if ((d->m_systemInfo & QWindowsContext::SI_SupportsPointer)) - return sessionManagerInteractionBlocked() || d->m_pointerHandler.translateMouseEvent(platformWindow->window(), hwnd, et, msg, result); - else - return sessionManagerInteractionBlocked() || d->m_mouseHandler.translateMouseEvent(platformWindow->window(), hwnd, et, msg, result); + return sessionManagerInteractionBlocked() || d->m_pointerHandler.translateMouseEvent(platformWindow->window(), hwnd, et, msg, result); case QtWindows::NonClientPointerEvent: if (!platformWindow->frameStrutEventsEnabled()) break; - if ((d->m_systemInfo & QWindowsContext::SI_SupportsPointer)) - return sessionManagerInteractionBlocked() || d->m_pointerHandler.translatePointerEvent(platformWindow->window(), hwnd, et, msg, result); - break; + return sessionManagerInteractionBlocked() || d->m_pointerHandler.translatePointerEvent(platformWindow->window(), hwnd, et, msg, result); case QtWindows::EnterSizeMoveEvent: platformWindow->setFlag(QWindowsWindow::ResizeMoveActive); if (!IsZoomed(hwnd)) @@ -1359,8 +1309,7 @@ bool QWindowsContext::windowsProc(HWND hwnd, UINT message, platformWindow->updateRestoreGeometry(); return true; case QtWindows::ScrollEvent: - if (!(d->m_systemInfo & QWindowsContext::SI_SupportsPointer)) - return sessionManagerInteractionBlocked() || d->m_mouseHandler.translateScrollEvent(platformWindow->window(), hwnd, msg, result); + // TODO??? break; case QtWindows::MouseWheelEvent: case QtWindows::MouseEvent: @@ -1371,20 +1320,14 @@ bool QWindowsContext::windowsProc(HWND hwnd, UINT message, window = window->parent(); if (!window) return false; - if (d->m_systemInfo & QWindowsContext::SI_SupportsPointer) - return sessionManagerInteractionBlocked() || d->m_pointerHandler.translateMouseEvent(window, hwnd, et, msg, result); - else - return sessionManagerInteractionBlocked() || d->m_mouseHandler.translateMouseEvent(window, hwnd, et, msg, result); + return sessionManagerInteractionBlocked() || d->m_pointerHandler.translateMouseEvent(window, hwnd, et, msg, result); } break; case QtWindows::TouchEvent: - if (!(d->m_systemInfo & QWindowsContext::SI_SupportsPointer)) - return sessionManagerInteractionBlocked() || d->m_mouseHandler.translateTouchEvent(platformWindow->window(), hwnd, et, msg, result); + // TODO??? break; case QtWindows::PointerEvent: - if (d->m_systemInfo & QWindowsContext::SI_SupportsPointer) - return sessionManagerInteractionBlocked() || d->m_pointerHandler.translatePointerEvent(platformWindow->window(), hwnd, et, msg, result); - break; + return sessionManagerInteractionBlocked() || d->m_pointerHandler.translatePointerEvent(platformWindow->window(), hwnd, et, msg, result); case QtWindows::FocusInEvent: // see QWindowsWindow::requestActivateWindow(). case QtWindows::FocusOutEvent: handleFocusEvent(et, platformWindow); @@ -1588,7 +1531,7 @@ void QWindowsContext::handleExitSizeMove(QWindow *window) // Mouse is left in pressed state after press on size grip (inside window), // no further mouse events are received // For cases 1,3, intercept WM_EXITSIZEMOVE to sync the buttons. - const Qt::MouseButtons currentButtons = QWindowsMouseHandler::queryMouseButtons(); + const Qt::MouseButtons currentButtons = QWindowsPointerHandler::queryMouseButtons(); const Qt::MouseButtons appButtons = QGuiApplication::mouseButtons(); if (currentButtons == appButtons) return; @@ -1604,10 +1547,7 @@ void QWindowsContext::handleExitSizeMove(QWindow *window) currentButtons, button, type, keyboardModifiers); } } - if (d->m_systemInfo & QWindowsContext::SI_SupportsPointer) - d->m_pointerHandler.clearEvents(); - else - d->m_mouseHandler.clearEvents(); + d->m_pointerHandler.clearEvents(); } bool QWindowsContext::asyncExpose() const diff --git a/qtbase/src/plugins/platforms/windows/qwindowscontext.h b/qtbase/src/plugins/platforms/windows/qwindowscontext.h index b0c16593..357f7839 100644 --- a/qtbase/src/plugins/platforms/windows/qwindowscontext.h +++ b/qtbase/src/plugins/platforms/windows/qwindowscontext.h @@ -143,8 +143,7 @@ public: enum SystemInfoFlags { SI_RTL_Extensions = 0x1, - SI_SupportsTouch = 0x2, - SI_SupportsPointer = 0x4, + SI_SupportsTouch = 0x2 }; // Verbose flag set by environment variable QT_QPA_VERBOSE @@ -157,7 +156,6 @@ public: bool initTouch(unsigned integrationOptions); // For calls from QWindowsIntegration::QWindowsIntegration() only. void registerTouchWindows(); bool initTablet(); - bool initPointer(unsigned integrationOptions); bool disposeTablet(); bool initPowerNotificationHandler(); @@ -210,8 +208,6 @@ public: static QtWindows::DpiAwareness processDpiAwareness(); static QtWindows::DpiAwareness windowDpiAwareness(HWND hwnd); - static bool isDarkMode(); - void setDetectAltGrModifier(bool a); // Returns a combination of SystemInfoFlags diff --git a/qtbase/src/plugins/platforms/windows/qwindowsdrag.cpp b/qtbase/src/plugins/platforms/windows/qwindowsdrag.cpp index b4439b46..563b22a8 100644 --- a/qtbase/src/plugins/platforms/windows/qwindowsdrag.cpp +++ b/qtbase/src/plugins/platforms/windows/qwindowsdrag.cpp @@ -11,7 +11,7 @@ #include "qwindowsintegration.h" #include "qwindowsdropdataobject.h" #include "qwindowswindow.h" -#include "qwindowsmousehandler.h" +#include "qwindowspointerhandler.h" #include "qwindowscursor.h" #include "qwindowskeymapper.h" @@ -344,7 +344,7 @@ QWindowsOleDropSource::QueryContinueDrag(BOOL fEscapePressed, DWORD grfKeyState) // In some rare cases, when a mouse button is released but the mouse is static, // grfKeyState will not be updated with these released buttons until the mouse // is moved. So we use the async key state given by queryMouseButtons() instead. - Qt::MouseButtons buttons = QWindowsMouseHandler::queryMouseButtons(); + Qt::MouseButtons buttons = QWindowsPointerHandler::queryMouseButtons(); SCODE result = S_OK; if (fEscapePressed || QWindowsDrag::isCanceled()) { @@ -369,7 +369,7 @@ QWindowsOleDropSource::QueryContinueDrag(BOOL fEscapePressed, DWORD grfKeyState) const QPoint localPos = m_windowUnderMouse->handle()->mapFromGlobal(globalPos); QWindowSystemInterface::handleMouseEvent(m_windowUnderMouse.data(), QPointF(localPos), QPointF(globalPos), - QWindowsMouseHandler::queryMouseButtons(), + QWindowsPointerHandler::queryMouseButtons(), Qt::LeftButton, QEvent::MouseButtonRelease); } m_currentButtons = Qt::NoButton; @@ -463,7 +463,7 @@ void QWindowsOleDropTarget::handleDrag(QWindow *window, DWORD grfKeyState, const Qt::DropActions actions = translateToQDragDropActions(*pdwEffect); lastModifiers = toQtKeyboardModifiers(grfKeyState); - lastButtons = QWindowsMouseHandler::queryMouseButtons(); + lastButtons = QWindowsPointerHandler::queryMouseButtons(); const QPlatformDragQtResponse response = QWindowSystemInterface::handleDrag(window, windowsDrag->dropData(), @@ -533,7 +533,7 @@ QWindowsOleDropTarget::DragLeave() const auto *keyMapper = QWindowsContext::instance()->keyMapper(); lastModifiers = keyMapper->queryKeyboardModifiers(); - lastButtons = QWindowsMouseHandler::queryMouseButtons(); + lastButtons = QWindowsPointerHandler::queryMouseButtons(); QWindowSystemInterface::handleDrag(m_window, nullptr, QPoint(), Qt::IgnoreAction, Qt::NoButton, Qt::NoModifier); @@ -562,7 +562,7 @@ QWindowsOleDropTarget::Drop(LPDATAOBJECT pDataObj, DWORD grfKeyState, QWindowsDrag *windowsDrag = QWindowsDrag::instance(); lastModifiers = toQtKeyboardModifiers(grfKeyState); - lastButtons = QWindowsMouseHandler::queryMouseButtons(); + lastButtons = QWindowsPointerHandler::queryMouseButtons(); const QPlatformDropQtResponse response = QWindowSystemInterface::handleDrop(m_window, windowsDrag->dropData(), diff --git a/qtbase/src/plugins/platforms/windows/qwindowskeymapper.cpp b/qtbase/src/plugins/platforms/windows/qwindowskeymapper.cpp index 622a062f..80a193d1 100644 --- a/qtbase/src/plugins/platforms/windows/qwindowskeymapper.cpp +++ b/qtbase/src/plugins/platforms/windows/qwindowskeymapper.cpp @@ -806,7 +806,7 @@ static void showSystemMenu(QWindow* w) qWindowsWndProc(topLevelHwnd, WM_SYSCOMMAND, WPARAM(ret), 0); } -static inline void sendExtendedPressRelease(QWindow *w, int k, +static inline void sendExtendedPressRelease(QWindow *w, unsigned long timestamp, int k, Qt::KeyboardModifiers mods, quint32 nativeScanCode, quint32 nativeVirtualKey, @@ -815,8 +815,8 @@ static inline void sendExtendedPressRelease(QWindow *w, int k, bool autorep = false, ushort count = 1) { - QWindowSystemInterface::handleExtendedKeyEvent(w, QEvent::KeyPress, k, mods, nativeScanCode, nativeVirtualKey, nativeModifiers, text, autorep, count); - QWindowSystemInterface::handleExtendedKeyEvent(w, QEvent::KeyRelease, k, mods, nativeScanCode, nativeVirtualKey, nativeModifiers, text, autorep, count); + QWindowSystemInterface::handleExtendedKeyEvent(w, timestamp, QEvent::KeyPress, k, mods, nativeScanCode, nativeVirtualKey, nativeModifiers, text, autorep, count); + QWindowSystemInterface::handleExtendedKeyEvent(w, timestamp, QEvent::KeyRelease, k, mods, nativeScanCode, nativeVirtualKey, nativeModifiers, text, autorep, count); } /*! @@ -883,7 +883,7 @@ bool QWindowsKeyMapper::translateMultimediaKeyEventInternal(QWindow *window, con const int qtKey = int(CmdTbl[cmd]); if (!skipPressRelease) - sendExtendedPressRelease(receiver, qtKey, Qt::KeyboardModifier(state), 0, 0, 0); + sendExtendedPressRelease(receiver, msg.time, qtKey, Qt::KeyboardModifier(state), 0, 0, 0); // QTBUG-43343: Make sure to return false if Qt does not handle the key, otherwise, // the keys are not passed to the active media player. # if QT_CONFIG(shortcut) @@ -963,7 +963,7 @@ bool QWindowsKeyMapper::translateKeyEventInternal(QWindow *window, MSG msg, // A multi-character key or a Input method character // not found by our look-ahead if (msgType == WM_CHAR || msgType == WM_IME_CHAR) { - sendExtendedPressRelease(receiver, 0, Qt::KeyboardModifier(state), scancode, 0, nModifiers, messageKeyText(msg), false); + sendExtendedPressRelease(receiver, msg.time, 0, Qt::KeyboardModifier(state), scancode, 0, nModifiers, messageKeyText(msg), false); return true; } @@ -998,14 +998,14 @@ bool QWindowsKeyMapper::translateKeyEventInternal(QWindow *window, MSG msg, if (dirStatus == VK_LSHIFT && ((msg.wParam == VK_SHIFT && GetKeyState(VK_LCONTROL)) || (msg.wParam == VK_CONTROL && GetKeyState(VK_LSHIFT)))) { - sendExtendedPressRelease(receiver, Qt::Key_Direction_L, {}, + sendExtendedPressRelease(receiver, msg.time, Qt::Key_Direction_L, {}, scancode, vk_key, nModifiers, QString(), false); result = true; dirStatus = 0; } else if (dirStatus == VK_RSHIFT && ( (msg.wParam == VK_SHIFT && GetKeyState(VK_RCONTROL)) || (msg.wParam == VK_CONTROL && GetKeyState(VK_RSHIFT)))) { - sendExtendedPressRelease(receiver, Qt::Key_Direction_R, {}, + sendExtendedPressRelease(receiver, msg.time, Qt::Key_Direction_R, {}, scancode, vk_key, nModifiers, QString(), false); result = true; dirStatus = 0; @@ -1218,9 +1218,9 @@ bool QWindowsKeyMapper::translateKeyEventInternal(QWindow *window, MSG msg, // so, we have an auto-repeating key if (rec) { if (code < Qt::Key_Shift || code > Qt::Key_ScrollLock) { - QWindowSystemInterface::handleExtendedKeyEvent(receiver, QEvent::KeyRelease, code, + QWindowSystemInterface::handleExtendedKeyEvent(receiver, msg.time, QEvent::KeyRelease, code, Qt::KeyboardModifier(state), scancode, quint32(msg.wParam), nModifiers, rec->text, true); - QWindowSystemInterface::handleExtendedKeyEvent(receiver, QEvent::KeyPress, code, + QWindowSystemInterface::handleExtendedKeyEvent(receiver, msg.time, QEvent::KeyPress, code, Qt::KeyboardModifier(state), scancode, quint32(msg.wParam), nModifiers, rec->text, true); result = true; } @@ -1246,7 +1246,7 @@ bool QWindowsKeyMapper::translateKeyEventInternal(QWindow *window, MSG msg, if (msg.wParam == VK_PACKET) code = asciiToKeycode(char(uch.cell()), state); - QWindowSystemInterface::handleExtendedKeyEvent(receiver, QEvent::KeyPress, code, + QWindowSystemInterface::handleExtendedKeyEvent(receiver, msg.time, QEvent::KeyPress, code, modifiers, scancode, quint32(msg.wParam), nModifiers, text, false); result =true; bool store = true; @@ -1288,10 +1288,10 @@ bool QWindowsKeyMapper::translateKeyEventInternal(QWindow *window, MSG msg, if ((msg.lParam & 0x40000000) == 0 && Qt::KeyboardModifier(state) == Qt::NoModifier && ((code == Qt::Key_F18) || (code == Qt::Key_F19) || (code == Qt::Key_F20))) { - QWindowSystemInterface::handleExtendedKeyEvent(receiver, QEvent::KeyPress, code, + QWindowSystemInterface::handleExtendedKeyEvent(receiver, msg.time, QEvent::KeyPress, code, Qt::MetaModifier, scancode, quint32(msg.wParam), MetaLeft); - QWindowSystemInterface::handleExtendedKeyEvent(receiver, QEvent::KeyRelease, code, + QWindowSystemInterface::handleExtendedKeyEvent(receiver, msg.time, QEvent::KeyRelease, code, Qt::NoModifier, scancode, quint32(msg.wParam), 0); result = true; @@ -1303,7 +1303,7 @@ bool QWindowsKeyMapper::translateKeyEventInternal(QWindow *window, MSG msg, // Map SHIFT + Tab to SHIFT + BackTab, QShortcutMap knows about this translation if (code == Qt::Key_Tab && (state & Qt::ShiftModifier) == Qt::ShiftModifier) code = Qt::Key_Backtab; - QWindowSystemInterface::handleExtendedKeyEvent(receiver, QEvent::KeyRelease, code, + QWindowSystemInterface::handleExtendedKeyEvent(receiver, msg.time, QEvent::KeyRelease, code, Qt::KeyboardModifier(state), scancode, quint32(msg.wParam), nModifiers, (rec ? rec->text : QString()), false); diff --git a/qtbase/src/plugins/platforms/windows/qwindowspointerhandler.cpp b/qtbase/src/plugins/platforms/windows/qwindowspointerhandler.cpp index f7bf7578..fab22b39 100644 --- a/qtbase/src/plugins/platforms/windows/qwindowspointerhandler.cpp +++ b/qtbase/src/plugins/platforms/windows/qwindowspointerhandler.cpp @@ -4,7 +4,6 @@ #include #include "qwindowspointerhandler.h" -#include "qwindowsmousehandler.h" #if QT_CONFIG(tabletevent) # include "qwindowstabletsupport.h" #endif @@ -41,6 +40,14 @@ enum { qint64 QWindowsPointerHandler::m_nextInputDeviceId = 1; +const QPointingDevice *primaryMouse() +{ + static QPointer result; + if (!result) + result = QPointingDevice::primaryPointingDevice(); + return result; +} + QWindowsPointerHandler::~QWindowsPointerHandler() { } @@ -240,7 +247,7 @@ static Qt::MouseButtons mouseButtonsFromKeyState(WPARAM keyState) return result; } -static Qt::MouseButtons queryMouseButtons() +Qt::MouseButtons QWindowsPointerHandler::queryMouseButtons() { Qt::MouseButtons result = Qt::NoButton; const bool mouseSwapped = GetSystemMetrics(SM_SWAPBUTTON); @@ -454,7 +461,7 @@ bool QWindowsPointerHandler::translateTouchEvent(QWindow *window, HWND hwnd, if (msg.message == WM_POINTERCAPTURECHANGED) { const auto *keyMapper = QWindowsContext::instance()->keyMapper(); - QWindowSystemInterface::handleTouchCancelEvent(window, m_touchDevice.data(), + QWindowSystemInterface::handleTouchCancelEvent(window, msg.time, m_touchDevice.data(), keyMapper->queryKeyboardModifiers()); m_lastTouchPoints.clear(); return true; @@ -569,7 +576,7 @@ bool QWindowsPointerHandler::translateTouchEvent(QWindow *window, HWND hwnd, m_touchInputIDToTouchPointID.clear(); const auto *keyMapper = QWindowsContext::instance()->keyMapper(); - QWindowSystemInterface::handleTouchEvent(window, m_touchDevice.data(), touchPoints, + QWindowSystemInterface::handleTouchEvent(window, msg.time, m_touchDevice.data(), touchPoints, keyMapper->queryKeyboardModifiers()); return false; // Allow mouse messages to be generated. } @@ -672,7 +679,7 @@ bool QWindowsPointerHandler::translatePenEvent(QWindow *window, HWND hwnd, QtWin switch (msg.message) { case WM_POINTERENTER: { - QWindowSystemInterface::handleTabletEnterLeaveProximityEvent(window, device.data(), true); + QWindowSystemInterface::handleTabletEnterLeaveProximityEvent(window, msg.time, device.data(), true); m_windowUnderPointer = window; // The local coordinates may fall outside the window. // Wait until the next update to send the enter event. @@ -685,7 +692,7 @@ bool QWindowsPointerHandler::translatePenEvent(QWindow *window, HWND hwnd, QtWin m_windowUnderPointer = nullptr; m_currentWindow = nullptr; } - QWindowSystemInterface::handleTabletEnterLeaveProximityEvent(window, device.data(), false); + QWindowSystemInterface::handleTabletEnterLeaveProximityEvent(window, msg.time, device.data(), false); break; case WM_POINTERDOWN: case WM_POINTERUP: @@ -711,7 +718,7 @@ bool QWindowsPointerHandler::translatePenEvent(QWindow *window, HWND hwnd, QtWin const auto *keyMapper = QWindowsContext::instance()->keyMapper(); const Qt::KeyboardModifiers keyModifiers = keyMapper->queryKeyboardModifiers(); - QWindowSystemInterface::handleTabletEvent(target, device.data(), + QWindowSystemInterface::handleTabletEvent(target, msg.time, device.data(), localPos, hiResGlobalPos, mouseButtons, pressure, xTilt, yTilt, tangentialPressure, rotation, z, keyModifiers); @@ -762,7 +769,7 @@ bool QWindowsPointerHandler::translateMouseWheelEvent(QWindow *window, QPoint localPos = QWindowsGeometryHint::mapFromGlobal(receiver, globalPos); - QWindowSystemInterface::handleWheelEvent(receiver, localPos, globalPos, QPoint(), angleDelta, keyModifiers); + QWindowSystemInterface::handleWheelEvent(receiver, msg.time, localPos, globalPos, QPoint(), angleDelta, keyModifiers); return true; } @@ -818,7 +825,7 @@ bool QWindowsPointerHandler::translateMouseEvent(QWindow *window, } Qt::MouseEventSource source = Qt::MouseEventNotSynthesized; - const QPointingDevice *device = QWindowsMouseHandler::primaryMouse(); + const QPointingDevice *device = primaryMouse(); // Following the logic of the old mouse handler, only events synthesized // for touch screen are marked as such. On some systems, using the bit 7 of @@ -861,14 +868,14 @@ bool QWindowsPointerHandler::translateMouseEvent(QWindow *window, && (m_lastEventButton & mouseButtons) == 0) { auto releaseType = mouseEvent.type == QEvent::NonClientAreaMouseMove ? QEvent::NonClientAreaMouseButtonRelease : QEvent::MouseButtonRelease; - QWindowSystemInterface::handleMouseEvent(window, device, localPos, globalPos, mouseButtons, m_lastEventButton, + QWindowSystemInterface::handleMouseEvent(window, msg.time, device, localPos, globalPos, mouseButtons, m_lastEventButton, releaseType, keyModifiers, source); } m_lastEventType = mouseEvent.type; m_lastEventButton = mouseEvent.button; if (mouseEvent.type >= QEvent::NonClientAreaMouseMove && mouseEvent.type <= QEvent::NonClientAreaMouseButtonDblClick) { - QWindowSystemInterface::handleMouseEvent(window, device, localPos, globalPos, mouseButtons, + QWindowSystemInterface::handleMouseEvent(window, msg.time, device, localPos, globalPos, mouseButtons, mouseEvent.button, mouseEvent.type, keyModifiers, source); return false; // Allow further event processing } @@ -888,7 +895,7 @@ bool QWindowsPointerHandler::translateMouseEvent(QWindow *window, handleEnterLeave(window, currentWindowUnderPointer, globalPos); if (!discardEvent && mouseEvent.type != QEvent::None) { - QWindowSystemInterface::handleMouseEvent(window, device, localPos, globalPos, mouseButtons, + QWindowSystemInterface::handleMouseEvent(window, msg.time, device, localPos, globalPos, mouseButtons, mouseEvent.button, mouseEvent.type, keyModifiers, source); } diff --git a/qtbase/src/plugins/platforms/windows/qwindowstheme.cpp b/qtbase/src/plugins/platforms/windows/qwindowstheme.cpp index 2873f7b4..94766cfc 100644 --- a/qtbase/src/plugins/platforms/windows/qwindowstheme.cpp +++ b/qtbase/src/plugins/platforms/windows/qwindowstheme.cpp @@ -13,19 +13,20 @@ # include "qwindowssystemtrayicon.h" #endif #include "qwindowsscreen.h" +#include "qwindowswindow.h" #include #include -#ifndef Q_CC_MINGW -# include -#endif +#include #include +#include #include #include #include #include #include #include +#include #include #include #include @@ -52,10 +53,6 @@ # include #endif // QT_CONFIG(cpp_winrt) -#if defined(__IImageList_INTERFACE_DEFINED__) && defined(__IID_DEFINED__) -# define USE_IIMAGELIST -#endif - QT_BEGIN_NAMESPACE using namespace Qt::StringLiterals; @@ -85,7 +82,11 @@ static inline QColor mixColors(const QColor &c1, const QColor &c2) enum AccentColorLevel { AccentColorDarkest, + AccentColorDarker, + AccentColorDark, AccentColorNormal, + AccentColorLight, + AccentColorLighter, AccentColorLightest }; @@ -100,6 +101,10 @@ static constexpr QColor getSysColor(winrt::Windows::UI::Color &&color) { QColor accent; QColor accentLight; + QColor accentLighter; + QColor accentLightest; + QColor accentDark; + QColor accentDarker; QColor accentDarkest; #if QT_CONFIG(cpp_winrt) @@ -109,6 +114,10 @@ static constexpr QColor getSysColor(winrt::Windows::UI::Color &&color) const auto settings = UISettings(); accent = getSysColor(settings.GetColorValue(UIColorType::Accent)); accentLight = getSysColor(settings.GetColorValue(UIColorType::AccentLight1)); + accentLighter = getSysColor(settings.GetColorValue(UIColorType::AccentLight2)); + accentLightest = getSysColor(settings.GetColorValue(UIColorType::AccentLight3)); + accentDark = getSysColor(settings.GetColorValue(UIColorType::AccentDark1)); + accentDarker = getSysColor(settings.GetColorValue(UIColorType::AccentDark2)); accentDarkest = getSysColor(settings.GetColorValue(UIColorType::AccentDark3)); } #endif @@ -128,14 +137,30 @@ static constexpr QColor getSysColor(winrt::Windows::UI::Color &&color) return {}; accent = QColor::fromRgb(abgr.blue(), abgr.green(), abgr.red(), abgr.alpha()); accentLight = accent.lighter(120); + accentLighter = accentLight.lighter(120); + accentLightest = accentLighter.lighter(120); accentDarkest = accent.darker(120 * 120 * 120); + accentDark = accent.darker(120); + accentDarker = accentDark.darker(120); + accentDarkest = accentDarker.darker(120); } - if (level == AccentColorDarkest) + switch (level) { + case AccentColorDarkest: return accentDarkest; - else if (level == AccentColorLightest) + case AccentColorDarker: + return accentDarker; + case AccentColorDark: + return accentDark; + case AccentColorLight: return accentLight; - return accent; + case AccentColorLighter: + return accentLighter; + case AccentColorLightest: + return accentLightest; + default: + return accent; + } } static inline QColor getSysColor(int index) @@ -148,107 +173,106 @@ static inline QColor getSysColor(int index) // models has been observed to trigger a WM_PAINT on the mainwindow. Suppress the // behavior by running it in a thread. -struct QShGetFileInfoParams -{ - QShGetFileInfoParams(const QString &fn, DWORD a, SHFILEINFO *i, UINT f, bool *r) - : fileName(fn), attributes(a), flags(f), info(i), result(r) - { } - - const QString &fileName; - const DWORD attributes; - const UINT flags; - SHFILEINFO *const info; - bool *const result; -}; - class QShGetFileInfoThread : public QThread { public: - explicit QShGetFileInfoThread() - : QThread(), m_params(nullptr) + struct Task { - connect(this, &QThread::finished, this, &QObject::deleteLater); + Task(const QString &fn, DWORD a, UINT f) + : fileName(fn), attributes(a), flags(f) + {} + Q_DISABLE_COPY(Task) + ~Task() + { + DestroyIcon(hIcon); + hIcon = 0; + } + // Request + const QString fileName; + const DWORD attributes; + const UINT flags; + // Result + HICON hIcon = 0; + int iIcon = -1; + bool finished = false; + bool resultValid() const { return hIcon != 0 && iIcon >= 0 && finished; } + }; + + QShGetFileInfoThread() + : QThread() + { + start(); + } + + ~QShGetFileInfoThread() + { + cancel(); + wait(); + } + + QSharedPointer getNextTask() + { + QMutexLocker l(&m_waitForTaskMutex); + while (!isInterruptionRequested()) { + if (!m_taskQueue.isEmpty()) + return m_taskQueue.dequeue(); + m_waitForTaskCondition.wait(&m_waitForTaskMutex); + } + return nullptr; } void run() override { QComHelper comHelper(COINIT_MULTITHREADED); - QMutexLocker readyLocker(&m_readyMutex); - while (!m_cancelled.loadRelaxed()) { - if (!m_params && !m_cancelled.loadRelaxed() - && !m_readyCondition.wait(&m_readyMutex, QDeadlineTimer(1000ll))) - continue; - - if (m_params) { - const QString fileName = m_params->fileName; + while (!isInterruptionRequested()) { + auto task = getNextTask(); + if (task) { SHFILEINFO info; - const bool result = SHGetFileInfo(reinterpret_cast(fileName.utf16()), - m_params->attributes, &info, sizeof(SHFILEINFO), - m_params->flags); - m_doneMutex.lock(); - if (!m_cancelled.loadRelaxed()) { - *m_params->result = result; - memcpy(m_params->info, &info, sizeof(SHFILEINFO)); + const bool result = SHGetFileInfo(reinterpret_cast(task->fileName.utf16()), + task->attributes, &info, sizeof(SHFILEINFO), + task->flags); + if (result) { + task->hIcon = info.hIcon; + task->iIcon = info.iIcon; } - m_params = nullptr; - + task->finished = true; m_doneCondition.wakeAll(); - m_doneMutex.unlock(); } } } - bool runWithParams(QShGetFileInfoParams *params, qint64 timeOutMSecs) + void runWithParams(const QSharedPointer &task, + std::chrono::milliseconds timeout = std::chrono::milliseconds(5000)) { + { + QMutexLocker l(&m_waitForTaskMutex); + m_taskQueue.enqueue(task); + m_waitForTaskCondition.wakeAll(); + } + QMutexLocker doneLocker(&m_doneMutex); - - m_readyMutex.lock(); - m_params = params; - m_readyCondition.wakeAll(); - m_readyMutex.unlock(); - - return m_doneCondition.wait(&m_doneMutex, QDeadlineTimer(timeOutMSecs)); + while (!task->finished && !isInterruptionRequested()) { + if (!m_doneCondition.wait(&m_doneMutex, QDeadlineTimer(timeout))) + return; + } } void cancel() { - QMutexLocker doneLocker(&m_doneMutex); - m_cancelled.storeRelaxed(1); - m_readyCondition.wakeAll(); + requestInterruption(); + m_doneCondition.wakeAll(); + m_waitForTaskCondition.wakeAll(); } private: - QShGetFileInfoParams *m_params; - QAtomicInt m_cancelled; - QWaitCondition m_readyCondition; + QQueue> m_taskQueue; QWaitCondition m_doneCondition; - QMutex m_readyMutex; + QWaitCondition m_waitForTaskCondition; QMutex m_doneMutex; + QMutex m_waitForTaskMutex; }; - -static bool shGetFileInfoBackground(const QString &fileName, DWORD attributes, - SHFILEINFO *info, UINT flags, - qint64 timeOutMSecs = 5000) -{ - static QShGetFileInfoThread *getFileInfoThread = nullptr; - if (!getFileInfoThread) { - getFileInfoThread = new QShGetFileInfoThread; - getFileInfoThread->start(); - } - - bool result = false; - QShGetFileInfoParams params(fileName, attributes, info, flags, &result); - if (!getFileInfoThread->runWithParams(¶ms, timeOutMSecs)) { - // Cancel and reset getFileInfoThread. It'll - // be reinitialized the next time we get called. - getFileInfoThread->cancel(); - getFileInfoThread = nullptr; - qWarning().noquote() << "SHGetFileInfo() timed out for " << fileName; - return false; - } - return result; -} +Q_APPLICATION_STATIC(QShGetFileInfoThread, s_shGetFileInfoThread) // from QStyle::standardPalette static inline QPalette standardPalette() @@ -277,15 +301,16 @@ static QColor placeHolderColor(QColor textColor) This is used when the theme is light mode, and when the theme is dark but the application doesn't support dark mode. In the latter case, we need to check. */ -static void populateLightSystemBasePalette(QPalette &result) +void QWindowsTheme::populateLightSystemBasePalette(QPalette &result) { const QColor background = getSysColor(COLOR_BTNFACE); const QColor textColor = getSysColor(COLOR_WINDOWTEXT); - const QColor accent = qt_accentColor(AccentColorNormal); + const QColor accentDark = qt_accentColor(AccentColorDark); + const QColor accentDarker = qt_accentColor(AccentColorDarker); const QColor accentDarkest = qt_accentColor(AccentColorDarkest); - const QColor linkColor = accent; + const QColor linkColor = accentDarker; const QColor btnFace = background; const QColor btnHighlight = getSysColor(COLOR_BTNHIGHLIGHT); @@ -304,7 +329,7 @@ static void populateLightSystemBasePalette(QPalette &result) result.setColor(QPalette::Midlight, getSysColor(COLOR_3DLIGHT)); result.setColor(QPalette::Shadow, getSysColor(COLOR_3DDKSHADOW)); result.setColor(QPalette::HighlightedText, getSysColor(COLOR_HIGHLIGHTTEXT)); - result.setColor(QPalette::Accent, accent); + result.setColor(QPalette::Accent, accentDark); // default accent color for controls on Light mode is AccentDark1 result.setColor(QPalette::Link, linkColor); result.setColor(QPalette::LinkVisited, accentDarkest); @@ -317,7 +342,7 @@ static void populateLightSystemBasePalette(QPalette &result) result.setColor(QPalette::Midlight, result.button().color().lighter(110)); } -static void populateDarkSystemBasePalette(QPalette &result) +void QWindowsTheme::populateDarkSystemBasePalette(QPalette &result) { QColor foreground = Qt::white; QColor background = QColor(0x1E, 0x1E, 0x1E); @@ -338,25 +363,27 @@ static void populateDarkSystemBasePalette(QPalette &result) // We have to craft a palette from these colors. The settings.UIElementColor(UIElementType) API // returns the old system colors, not the dark mode colors. If the background is black (which it // usually), then override it with a dark gray instead so that we can go up and down the lightness. - foreground = getSysColor(settings.GetColorValue(UIColorType::Foreground)); - background = [&settings]() -> QColor { - auto systemBackground = getSysColor(settings.GetColorValue(UIColorType::Background)); - if (systemBackground == Qt::black) - systemBackground = QColor(0x1E, 0x1E, 0x1E); - return systemBackground; - }(); - - accent = getSysColor(settings.GetColorValue(UIColorType::Accent)); - accentDark = getSysColor(settings.GetColorValue(UIColorType::AccentDark1)); - accentDarker = getSysColor(settings.GetColorValue(UIColorType::AccentDark2)); - accentDarkest = getSysColor(settings.GetColorValue(UIColorType::AccentDark3)); - accentLight = getSysColor(settings.GetColorValue(UIColorType::AccentLight1)); - accentLighter = getSysColor(settings.GetColorValue(UIColorType::AccentLight2)); - accentLightest = getSysColor(settings.GetColorValue(UIColorType::AccentLight3)); + if (QWindowsTheme::queryColorScheme() == Qt::ColorScheme::Dark) { + // the system is actually running in dark mode, so UISettings will give us dark colors + foreground = getSysColor(settings.GetColorValue(UIColorType::Foreground)); + background = [&settings]() -> QColor { + auto systemBackground = getSysColor(settings.GetColorValue(UIColorType::Background)); + if (systemBackground == Qt::black) + systemBackground = QColor(0x1E, 0x1E, 0x1E); + return systemBackground; + }(); + accent = qt_accentColor(AccentColorNormal); + accentDark = qt_accentColor(AccentColorDark); + accentDarker = qt_accentColor(AccentColorDarker); + accentDarkest = qt_accentColor(AccentColorDarkest); + accentLight = qt_accentColor(AccentColorLight); + accentLighter = qt_accentColor(AccentColorLighter); + accentLightest = qt_accentColor(AccentColorLightest); + } } #endif - const QColor linkColor = accent; + const QColor linkColor = accentLightest; const QColor buttonColor = background.lighter(200); result.setColor(QPalette::All, QPalette::WindowText, foreground); @@ -377,12 +404,12 @@ static void populateDarkSystemBasePalette(QPalette &result) result.setColor(QPalette::All, QPalette::Highlight, accent); result.setColor(QPalette::All, QPalette::HighlightedText, accent.lightness() > 128 ? Qt::black : Qt::white); result.setColor(QPalette::All, QPalette::Link, linkColor); - result.setColor(QPalette::All, QPalette::LinkVisited, accentDarkest); + result.setColor(QPalette::All, QPalette::LinkVisited, accentLighter); result.setColor(QPalette::All, QPalette::AlternateBase, accentDarkest); result.setColor(QPalette::All, QPalette::ToolTipBase, buttonColor); result.setColor(QPalette::All, QPalette::ToolTipText, foreground.darker(120)); result.setColor(QPalette::All, QPalette::PlaceholderText, placeHolderColor(foreground)); - result.setColor(QPalette::All, QPalette::Accent, accent); + result.setColor(QPalette::All, QPalette::Accent, accentLighter); } static inline QPalette toolTipPalette(const QPalette &systemPalette, bool light) @@ -474,6 +501,7 @@ QWindowsTheme *QWindowsTheme::m_instance = nullptr; QWindowsTheme::QWindowsTheme() { m_instance = this; + s_colorScheme = QWindowsTheme::queryColorScheme(); std::fill(m_fonts, m_fonts + NFonts, nullptr); std::fill(m_palettes, m_palettes + NPalettes, nullptr); refresh(); @@ -562,10 +590,44 @@ QVariant QWindowsTheme::themeHint(ThemeHint hint) const } Qt::ColorScheme QWindowsTheme::colorScheme() const +{ + return QWindowsTheme::effectiveColorScheme(); +} + +Qt::ColorScheme QWindowsTheme::effectiveColorScheme() { if (queryHighContrast()) return Qt::ColorScheme::Unknown; - return QWindowsContext::isDarkMode() ? Qt::ColorScheme::Dark : Qt::ColorScheme::Light; + if (s_colorSchemeOverride != Qt::ColorScheme::Unknown) + return s_colorSchemeOverride; + if (s_colorScheme != Qt::ColorScheme::Unknown) + return s_colorScheme; + return queryColorScheme(); +} + +void QWindowsTheme::requestColorScheme(Qt::ColorScheme scheme) +{ + s_colorSchemeOverride = scheme; + handleSettingsChanged(); +} + +void QWindowsTheme::handleSettingsChanged() +{ + const auto oldColorScheme = s_colorScheme; + s_colorScheme = Qt::ColorScheme::Unknown; // make effectiveColorScheme() query registry + const auto newColorScheme = effectiveColorScheme(); + const bool colorSchemeChanged = newColorScheme != oldColorScheme; + s_colorScheme = newColorScheme; + auto integration = QWindowsIntegration::instance(); + integration->updateApplicationBadge(); + if (integration->darkModeHandling().testFlag(QWindowsApplication::DarkModeStyle)) { + QWindowsTheme::instance()->refresh(); + QWindowSystemInterface::handleThemeChange(); + } + if (colorSchemeChanged) { + for (QWindowsWindow *w : std::as_const(QWindowsContext::instance()->windows())) + w->setDarkBorder(s_colorScheme == Qt::ColorScheme::Dark); + } } void QWindowsTheme::clearPalettes() @@ -579,17 +641,17 @@ void QWindowsTheme::refreshPalettes() if (!QGuiApplication::desktopSettingsAware()) return; const bool light = - !QWindowsContext::isDarkMode() + effectiveColorScheme() != Qt::ColorScheme::Dark || !QWindowsIntegration::instance()->darkModeHandling().testFlag(QWindowsApplication::DarkModeStyle); clearPalettes(); - m_palettes[SystemPalette] = new QPalette(QWindowsTheme::systemPalette(light ? Qt::ColorScheme::Light : Qt::ColorScheme::Dark)); + m_palettes[SystemPalette] = new QPalette(QWindowsTheme::systemPalette(s_colorScheme)); m_palettes[ToolTipPalette] = new QPalette(toolTipPalette(*m_palettes[SystemPalette], light)); m_palettes[MenuPalette] = new QPalette(menuPalette(*m_palettes[SystemPalette], light)); m_palettes[MenuBarPalette] = menuBarPalette(*m_palettes[MenuPalette], light); if (!light) { m_palettes[CheckBoxPalette] = new QPalette(*m_palettes[SystemPalette]); m_palettes[CheckBoxPalette]->setColor(QPalette::Active, QPalette::Base, qt_accentColor(AccentColorNormal)); - m_palettes[CheckBoxPalette]->setColor(QPalette::Active, QPalette::Button, qt_accentColor(AccentColorLightest)); + m_palettes[CheckBoxPalette]->setColor(QPalette::Active, QPalette::Button, qt_accentColor(AccentColorLighter)); m_palettes[CheckBoxPalette]->setColor(QPalette::Inactive, QPalette::Base, qt_accentColor(AccentColorDarkest)); m_palettes[RadioButtonPalette] = new QPalette(*m_palettes[CheckBoxPalette]); } @@ -600,15 +662,15 @@ QPalette QWindowsTheme::systemPalette(Qt::ColorScheme colorScheme) QPalette result = standardPalette(); switch (colorScheme) { + case Qt::ColorScheme::Unknown: + // when a high-contrast theme is active or when we fail to read, assume light + Q_FALLTHROUGH(); case Qt::ColorScheme::Light: populateLightSystemBasePalette(result); break; case Qt::ColorScheme::Dark: populateDarkSystemBasePalette(result); break; - default: - qFatal("Unknown color scheme"); - break; } if (result.window() != result.base()) { @@ -750,11 +812,7 @@ void QWindowsTheme::refreshIconPixmapSizes() fileIconSizes[LargeFileIcon] + fileIconSizes[LargeFileIcon] / 2; fileIconSizes[JumboFileIcon] = 8 * fileIconSizes[LargeFileIcon]; // empirical, has not been observed to work -#ifdef USE_IIMAGELIST int *availEnd = fileIconSizes + JumboFileIcon + 1; -#else - int *availEnd = fileIconSizes + LargeFileIcon + 1; -#endif // USE_IIMAGELIST m_fileIconSizes = QAbstractFileIconEngine::toSizeList(fileIconSizes, availEnd); qCDebug(lcQpaWindow) << __FUNCTION__ << m_fileIconSizes; } @@ -948,10 +1006,10 @@ public: // Shell image list helper functions. -static QPixmap pixmapFromShellImageList(int iImageList, const SHFILEINFO &info) +static QPixmap pixmapFromShellImageList(int iImageList, int iIcon) { QPixmap result; -#ifdef USE_IIMAGELIST + // For MinGW: static const IID iID_IImageList = {0x46eb5926, 0x582e, 0x4017, {0x9f, 0xdf, 0xe8, 0x99, 0x8d, 0xaa, 0x9, 0x50}}; @@ -960,16 +1018,13 @@ static QPixmap pixmapFromShellImageList(int iImageList, const SHFILEINFO &info) if (hr != S_OK) return result; HICON hIcon; - hr = imageList->GetIcon(info.iIcon, ILD_TRANSPARENT, &hIcon); + hr = imageList->GetIcon(iIcon, ILD_TRANSPARENT, &hIcon); if (hr == S_OK) { result = qt_pixmapFromWinHICON(hIcon); DestroyIcon(hIcon); } imageList->Release(); -#else - Q_UNUSED(iImageList); - Q_UNUSED(info); -#endif // USE_IIMAGELIST + return result; } @@ -1021,13 +1076,9 @@ QPixmap QWindowsFileIconEngine::filePixmap(const QSize &size, QIcon::Mode, QIcon const int width = int(size.width()); const int iconSize = width > fileIconSizes[SmallFileIcon] ? SHGFI_LARGEICON : SHGFI_SMALLICON; const int requestedImageListSize = -#ifdef USE_IIMAGELIST width > fileIconSizes[ExtraLargeFileIcon] ? sHIL_JUMBO : (width > fileIconSizes[LargeFileIcon] ? sHIL_EXTRALARGE : 0); -#else - 0; -#endif // !USE_IIMAGELIST bool cacheableDirIcon = fileInfo().isDir() && !fileInfo().isRoot(); if (cacheableDirIcon) { QMutexLocker locker(&mx); @@ -1043,7 +1094,6 @@ QPixmap QWindowsFileIconEngine::filePixmap(const QSize &size, QIcon::Mode, QIcon } } - SHFILEINFO info; unsigned int flags = SHGFI_ICON | iconSize | SHGFI_SYSICONINDEX | SHGFI_ADDOVERLAYS | SHGFI_OVERLAYINDEX; DWORD attributes = 0; QString path = filePath; @@ -1055,43 +1105,43 @@ QPixmap QWindowsFileIconEngine::filePixmap(const QSize &size, QIcon::Mode, QIcon flags |= SHGFI_USEFILEATTRIBUTES; attributes |= FILE_ATTRIBUTE_NORMAL; } - const bool val = shGetFileInfoBackground(path, attributes, &info, flags); - + auto task = QSharedPointer( + new QShGetFileInfoThread::Task(path, attributes, flags)); + s_shGetFileInfoThread()->runWithParams(task); // Even if GetFileInfo returns a valid result, hIcon can be empty in some cases - if (val && info.hIcon) { + if (task->resultValid()) { QString key; if (cacheableDirIcon) { if (useDefaultFolderIcon && defaultFolderIIcon < 0) - defaultFolderIIcon = info.iIcon; + defaultFolderIIcon = task->iIcon; //using the unique icon index provided by windows save us from duplicate keys - key = dirIconPixmapCacheKey(info.iIcon, iconSize, requestedImageListSize); + key = dirIconPixmapCacheKey(task->iIcon, iconSize, requestedImageListSize); QPixmapCache::find(key, &pixmap); if (!pixmap.isNull()) { QMutexLocker locker(&mx); - dirIconEntryCache.insert(filePath, FakePointer::create(info.iIcon)); + dirIconEntryCache.insert(filePath, FakePointer::create(task->iIcon)); } } if (pixmap.isNull()) { if (requestedImageListSize) { - pixmap = pixmapFromShellImageList(requestedImageListSize, info); + pixmap = pixmapFromShellImageList(requestedImageListSize, task->iIcon); if (pixmap.isNull() && requestedImageListSize == sHIL_JUMBO) - pixmap = pixmapFromShellImageList(sHIL_EXTRALARGE, info); + pixmap = pixmapFromShellImageList(sHIL_EXTRALARGE, task->iIcon); } if (pixmap.isNull()) - pixmap = qt_pixmapFromWinHICON(info.hIcon); + pixmap = qt_pixmapFromWinHICON(task->hIcon); if (!pixmap.isNull()) { if (cacheableDirIcon) { QMutexLocker locker(&mx); QPixmapCache::insert(key, pixmap); - dirIconEntryCache.insert(filePath, FakePointer::create(info.iIcon)); + dirIconEntryCache.insert(filePath, FakePointer::create(task->iIcon)); } } else { qWarning("QWindowsTheme::fileIconPixmap() no icon found"); } } - DestroyIcon(info.hIcon); } return pixmap; @@ -1131,14 +1181,14 @@ bool QWindowsTheme::useNativeMenus() return result; } -bool QWindowsTheme::queryDarkMode() +Qt::ColorScheme QWindowsTheme::queryColorScheme() { - if (queryHighContrast()) { - return false; - } + if (queryHighContrast()) + return Qt::ColorScheme::Unknown; + const auto setting = QWinRegistryKey(HKEY_CURRENT_USER, LR"(Software\Microsoft\Windows\CurrentVersion\Themes\Personalize)") .dwordValue(L"AppsUseLightTheme"); - return setting.second && setting.first == 0; + return setting.second && setting.first == 0 ? Qt::ColorScheme::Dark : Qt::ColorScheme::Light; } bool QWindowsTheme::queryHighContrast() diff --git a/qtbase/src/plugins/platforms/windows/qwindowswindow.cpp b/qtbase/src/plugins/platforms/windows/qwindowswindow.cpp index b4cd8a8e..c1c0554b 100644 --- a/qtbase/src/plugins/platforms/windows/qwindowswindow.cpp +++ b/qtbase/src/plugins/platforms/windows/qwindowswindow.cpp @@ -5,6 +5,7 @@ #include "qwindowswindow.h" #include "qwindowscontext.h" +#include "qwindowstheme.h" #if QT_CONFIG(draganddrop) # include "qwindowsdrag.h" #endif @@ -867,7 +868,8 @@ static inline bool shouldApplyDarkFrame(const QWindow *w) { if (!w->isTopLevel() || w->flags().testFlag(Qt::FramelessWindowHint)) return false; - // the application has explicitly opted out of dark frames + + // the user of the application has explicitly opted out of dark frames if (!QWindowsIntegration::instance()->darkModeHandling().testFlag(QWindowsApplication::DarkModeWindowFrames)) return false; @@ -947,7 +949,7 @@ QWindowsWindowData return result; } - if (QWindowsContext::isDarkMode() && shouldApplyDarkFrame(w)) + if (QWindowsTheme::instance()->colorScheme() == Qt::ColorScheme::Dark && shouldApplyDarkFrame(w)) QWindowsWindow::setDarkBorderToWindow(result.hwnd, true); if (mirrorParentWidth != 0) { @@ -1387,6 +1389,12 @@ QWindowsForeignWindow::QWindowsForeignWindow(QWindow *window, HWND hwnd) setParent(QPlatformWindow::parent()); } +QWindowsForeignWindow::~QWindowsForeignWindow() +{ + if (QPlatformWindow::parent()) + setParent(nullptr); +} + void QWindowsForeignWindow::setParent(const QPlatformWindow *newParentWindow) { const bool wasTopLevel = isTopLevel_sys(); @@ -2739,7 +2747,7 @@ bool QWindowsWindow::windowEvent(QEvent *event) { switch (event->type()) { case QEvent::ApplicationPaletteChange: - setDarkBorder(QWindowsContext::isDarkMode()); + setDarkBorder(QWindowsTheme::instance()->colorScheme() == Qt::ColorScheme::Dark); break; case QEvent::WindowBlocked: // Blocked by another modal window. setEnabled(false); @@ -3335,17 +3343,6 @@ enum : WORD { DwmwaUseImmersiveDarkModeBefore20h1 = 19 }; -static bool queryDarkBorder(HWND hwnd) -{ - BOOL result = FALSE; - const bool ok = - SUCCEEDED(DwmGetWindowAttribute(hwnd, DwmwaUseImmersiveDarkMode, &result, sizeof(result))) - || SUCCEEDED(DwmGetWindowAttribute(hwnd, DwmwaUseImmersiveDarkModeBefore20h1, &result, sizeof(result))); - if (!ok) - qCWarning(lcQpaWindow, "%s: Unable to retrieve dark window border setting.", __FUNCTION__); - return result == TRUE; -} - bool QWindowsWindow::setDarkBorderToWindow(HWND hwnd, bool d) { const BOOL darkBorder = d ? TRUE : FALSE; @@ -3361,8 +3358,6 @@ void QWindowsWindow::setDarkBorder(bool d) { // respect explicit opt-out and incompatible palettes or styles d = d && shouldApplyDarkFrame(window()); - if (queryDarkBorder(m_data.hwnd) == d) - return; setDarkBorderToWindow(m_data.hwnd, d); } diff --git a/qtbase/src/plugins/platforms/windows/uiautomation/qwindowsuiaaccessibility.cpp b/qtbase/src/plugins/platforms/windows/uiautomation/qwindowsuiaaccessibility.cpp new file mode 100644 index 00000000..1c89fd87 --- /dev/null +++ b/qtbase/src/plugins/platforms/windows/uiautomation/qwindowsuiaaccessibility.cpp @@ -0,0 +1,170 @@ +// Copyright (C) 2017 The Qt Company Ltd. +// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR LGPL-3.0-only OR GPL-2.0-only OR GPL-3.0-only + +#include +#if QT_CONFIG(accessibility) + +#include "qwindowsuiaaccessibility.h" +#include "qwindowsuiautomation.h" +#include "qwindowsuiamainprovider.h" +#include "qwindowsuiautils.h" + +#include +#include +#include +#include +#include +#include +#include "qwindowsuiawrapper_p.h" +#include "qwindowsuiawrapper.cpp" + +#include + +QT_BEGIN_NAMESPACE + +using namespace QWindowsUiAutomation; +using namespace Qt::Literals::StringLiterals; + +bool QWindowsUiaAccessibility::m_accessibleActive = false; + +QWindowsUiaAccessibility::QWindowsUiaAccessibility() +{ +} + +QWindowsUiaAccessibility::~QWindowsUiaAccessibility() +{ +} + +// Handles UI Automation window messages. +bool QWindowsUiaAccessibility::handleWmGetObject(HWND hwnd, WPARAM wParam, LPARAM lParam, LRESULT *lResult) +{ + // Start handling accessibility internally + QGuiApplicationPrivate::platformIntegration()->accessibility()->setActive(true); + m_accessibleActive = true; + + // Ignoring all requests while starting up / shutting down + if (QCoreApplication::startingUp() || QCoreApplication::closingDown()) + return false; + + if (QWindow *window = QWindowsContext::instance()->findWindow(hwnd)) { + if (QAccessibleInterface *accessible = window->accessibleRoot()) { + auto provider = QWindowsUiaMainProvider::providerForAccessible(accessible); + *lResult = QWindowsUiaWrapper::instance()->returnRawElementProvider(hwnd, wParam, lParam, provider.Get()); + return true; + } + } + return false; +} + +// Retrieve sound name by checking the icon property of a message box +// should it be the event object. +static QString alertSound(const QObject *object) +{ + if (object->inherits("QMessageBox")) { + enum MessageBoxIcon { // Keep in sync with QMessageBox::Icon + Information = 1, + Warning = 2, + Critical = 3 + }; + switch (object->property("icon").toInt()) { + case Information: + return QStringLiteral("SystemAsterisk"); + case Warning: + return QStringLiteral("SystemExclamation"); + case Critical: + return QStringLiteral("SystemHand"); + } + return QString(); + } + return QStringLiteral("SystemAsterisk"); +} + +static QString soundFileName(const QString &soundName) +{ + const QString key = "AppEvents\\Schemes\\Apps\\.Default\\"_L1 + + soundName + "\\.Current"_L1; + return QWinRegistryKey(HKEY_CURRENT_USER, key).stringValue(L""); +} + +static void playSystemSound(const QString &soundName) +{ + if (!soundName.isEmpty() && !soundFileName(soundName).isEmpty()) { + PlaySound(reinterpret_cast(soundName.utf16()), nullptr, + SND_ALIAS | SND_ASYNC | SND_NODEFAULT | SND_NOWAIT); + } +} + +// Handles accessibility update notifications. +void QWindowsUiaAccessibility::notifyAccessibilityUpdate(QAccessibleEvent *event) +{ + if (!event) + return; + + // Always handle system sound events + switch (event->type()) { + case QAccessible::PopupMenuStart: + playSystemSound(QStringLiteral("MenuPopup")); + break; + case QAccessible::MenuCommand: + playSystemSound(QStringLiteral("MenuCommand")); + break; + case QAccessible::Alert: + playSystemSound(alertSound(event->object())); + break; + default: + break; + } + + // Ignore events sent before the first UI Automation + // request or while QAccessible is being activated. + if (!m_accessibleActive) + return; + + QAccessibleInterface *accessible = event->accessibleInterface(); + if (!isActive() || !accessible || !accessible->isValid()) + return; + + // Ensures QWindowsUiaWrapper is properly initialized. + if (!QWindowsUiaWrapper::instance()->ready()) + return; + + // No need to do anything when nobody is listening. + if (!QWindowsUiaWrapper::instance()->clientsAreListening()) + return; + + switch (event->type()) { + case QAccessible::Announcement: + QWindowsUiaMainProvider::raiseNotification(static_cast(event)); + break; + case QAccessible::Focus: + QWindowsUiaMainProvider::notifyFocusChange(event); + break; + case QAccessible::StateChanged: + QWindowsUiaMainProvider::notifyStateChange(static_cast(event)); + break; + case QAccessible::ValueChanged: + QWindowsUiaMainProvider::notifyValueChange(static_cast(event)); + break; + case QAccessible::NameChanged: + QWindowsUiaMainProvider::notifyNameChange(event); + break; + case QAccessible::SelectionAdd: + QWindowsUiaMainProvider::notifySelectionChange(event); + break; + case QAccessible::TextAttributeChanged: + case QAccessible::TextColumnChanged: + case QAccessible::TextInserted: + case QAccessible::TextRemoved: + case QAccessible::TextUpdated: + case QAccessible::TextSelectionChanged: + case QAccessible::TextCaretMoved: + QWindowsUiaMainProvider::notifyTextChange(event); + break; + default: + break; + } +} + +QT_END_NAMESPACE + +#endif // QT_CONFIG(accessibility) diff --git a/qtbase/src/plugins/platforms/windows/uiautomation/qwindowsuiamainprovider.cpp b/qtbase/src/plugins/platforms/windows/uiautomation/qwindowsuiamainprovider.cpp new file mode 100644 index 00000000..b1e43639 --- /dev/null +++ b/qtbase/src/plugins/platforms/windows/uiautomation/qwindowsuiamainprovider.cpp @@ -0,0 +1,843 @@ +// Copyright (C) 2017 The Qt Company Ltd. +// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR LGPL-3.0-only OR GPL-2.0-only OR GPL-3.0-only + +#include +#if QT_CONFIG(accessibility) + +#include "qwindowsuiamainprovider.h" +#include "qwindowsuiavalueprovider.h" +#include "qwindowsuiarangevalueprovider.h" +#include "qwindowsuiatextprovider.h" +#include "qwindowsuiatoggleprovider.h" +#include "qwindowsuiainvokeprovider.h" +#include "qwindowsuiaselectionprovider.h" +#include "qwindowsuiaselectionitemprovider.h" +#include "qwindowsuiatableprovider.h" +#include "qwindowsuiatableitemprovider.h" +#include "qwindowsuiagridprovider.h" +#include "qwindowsuiagriditemprovider.h" +#include "qwindowsuiawindowprovider.h" +#include "qwindowsuiaexpandcollapseprovider.h" +#include "qwindowscontext.h" +#include "qwindowsuiautils.h" +#include "qwindowsuiaprovidercache.h" +#include "qwindowsuiawrapper_p.h" + +#include +#include +#include +#include +#include +#include +#include + +#if !defined(Q_CC_BOR) && !defined (Q_CC_GNU) +#include +#endif + +#include + +QT_BEGIN_NAMESPACE + +using namespace QWindowsUiAutomation; + +// Returns a cached instance of the provider for a specific accessible interface. +ComPtr QWindowsUiaMainProvider::providerForAccessible(QAccessibleInterface *accessible) +{ + if (!accessible) + return nullptr; + + QAccessible::Id id = QAccessible::uniqueId(accessible); + QWindowsUiaProviderCache *providerCache = QWindowsUiaProviderCache::instance(); + ComPtr provider = providerCache->providerForId(id); + + if (!provider) { + provider = makeComObject(accessible); + providerCache->insert(id, provider.Get()); // Cache holds weak references + } + return provider; +} + +QWindowsUiaMainProvider::QWindowsUiaMainProvider(QAccessibleInterface *a) + : QWindowsUiaBaseProvider(QAccessible::uniqueId(a)) +{ +} + +QWindowsUiaMainProvider::~QWindowsUiaMainProvider() +{ +} + +void QWindowsUiaMainProvider::notifyFocusChange(QAccessibleEvent *event) +{ + if (QAccessibleInterface *accessible = event->accessibleInterface()) { + // If this is a complex element, raise event for the focused child instead. + if (accessible->childCount()) { + if (QAccessibleInterface *child = accessible->focusChild()) + accessible = child; + } + if (auto provider = providerForAccessible(accessible)) + QWindowsUiaWrapper::instance()->raiseAutomationEvent(provider.Get(), UIA_AutomationFocusChangedEventId); + } +} + +void QWindowsUiaMainProvider::notifyStateChange(QAccessibleStateChangeEvent *event) +{ + if (QAccessibleInterface *accessible = event->accessibleInterface()) { + if (event->changedStates().checked || event->changedStates().checkStateMixed) { + // Notifies states changes in checkboxes. + if (accessible->role() == QAccessible::CheckBox) { + if (auto provider = providerForAccessible(accessible)) { + long toggleState = ToggleState_Off; + if (accessible->state().checked) + toggleState = accessible->state().checkStateMixed ? ToggleState_Indeterminate : ToggleState_On; + + QComVariant oldVal; + QComVariant newVal{toggleState}; + QWindowsUiaWrapper::instance()->raiseAutomationPropertyChangedEvent( + provider.Get(), UIA_ToggleToggleStatePropertyId, oldVal.get(), newVal.get()); + } + } + } + if (event->changedStates().active) { + if (accessible->role() == QAccessible::Window) { + // Notifies window opened/closed. + if (auto provider = providerForAccessible(accessible)) { + if (accessible->state().active) { + QWindowsUiaWrapper::instance()->raiseAutomationEvent(provider.Get(), UIA_Window_WindowOpenedEventId); + if (QAccessibleInterface *focused = accessible->focusChild()) { + if (auto focusedProvider = providerForAccessible(focused)) { + QWindowsUiaWrapper::instance()->raiseAutomationEvent(focusedProvider.Get(), + UIA_AutomationFocusChangedEventId); + } + } + } else { + QWindowsUiaWrapper::instance()->raiseAutomationEvent(provider.Get(), UIA_Window_WindowClosedEventId); + } + } + } + } + } +} + +void QWindowsUiaMainProvider::notifyValueChange(QAccessibleValueChangeEvent *event) +{ + if (QAccessibleInterface *accessible = event->accessibleInterface()) { + if (accessible->role() == QAccessible::ComboBox && accessible->childCount() > 0) { + QAccessibleInterface *listacc = accessible->child(0); + if (listacc && listacc->role() == QAccessible::List) { + int count = listacc->childCount(); + for (int i = 0; i < count; ++i) { + QAccessibleInterface *item = listacc->child(i); + if (item && item->isValid() && item->text(QAccessible::Name) == event->value()) { + if (!item->state().selected) { + if (QAccessibleActionInterface *actionInterface = item->actionInterface()) + actionInterface->doAction(QAccessibleActionInterface::toggleAction()); + } + break; + } + } + } + } + if (event->value().typeId() == QMetaType::QString) { + if (auto provider = providerForAccessible(accessible)) { + // Notifies changes in string values. + const QComVariant oldVal; + const QComVariant newVal{ event->value().toString() }; + QWindowsUiaWrapper::instance()->raiseAutomationPropertyChangedEvent(provider.Get(), UIA_ValueValuePropertyId, + oldVal.get(), newVal.get()); + } + } else if (QAccessibleValueInterface *valueInterface = accessible->valueInterface()) { + if (auto provider = providerForAccessible(accessible)) { + // Notifies changes in values of controls supporting the value interface. + const QComVariant oldVal; + const QComVariant newVal{ valueInterface->currentValue().toDouble() }; + QWindowsUiaWrapper::instance()->raiseAutomationPropertyChangedEvent( + provider.Get(), UIA_RangeValueValuePropertyId, oldVal.get(), newVal.get()); + } + } + } +} + +void QWindowsUiaMainProvider::notifyNameChange(QAccessibleEvent *event) +{ + if (QAccessibleInterface *accessible = event->accessibleInterface()) { + // Restrict notification to combo boxes, which need it for accessibility, + // in order to avoid slowdowns with unnecessary notifications. + if (accessible->role() == QAccessible::ComboBox) { + if (auto provider = providerForAccessible(accessible)) { + QComVariant oldVal; + QComVariant newVal{ accessible->text(QAccessible::Name) }; + QWindowsUiaWrapper::instance()->raiseAutomationPropertyChangedEvent(provider.Get(), UIA_NamePropertyId, + oldVal.get(), newVal.get()); + } + } + } +} + +void QWindowsUiaMainProvider::notifySelectionChange(QAccessibleEvent *event) +{ + if (QAccessibleInterface *accessible = event->accessibleInterface()) { + if (auto provider = providerForAccessible(accessible)) + QWindowsUiaWrapper::instance()->raiseAutomationEvent(provider.Get(), UIA_SelectionItem_ElementSelectedEventId); + } +} + +// Notifies changes in text content and selection state of text controls. +void QWindowsUiaMainProvider::notifyTextChange(QAccessibleEvent *event) +{ + if (QAccessibleInterface *accessible = event->accessibleInterface()) { + if (accessible->textInterface()) { + if (auto provider = providerForAccessible(accessible)) { + if (event->type() == QAccessible::TextSelectionChanged) { + QWindowsUiaWrapper::instance()->raiseAutomationEvent(provider.Get(), UIA_Text_TextSelectionChangedEventId); + } else if (event->type() == QAccessible::TextCaretMoved) { + if (!accessible->state().readOnly) { + QWindowsUiaWrapper::instance()->raiseAutomationEvent(provider.Get(), + UIA_Text_TextSelectionChangedEventId); + } + } else { + QWindowsUiaWrapper::instance()->raiseAutomationEvent(provider.Get(), UIA_Text_TextChangedEventId); + } + } + } + } +} + +void QWindowsUiaMainProvider::raiseNotification(QAccessibleAnnouncementEvent *event) +{ + if (QAccessibleInterface *accessible = event->accessibleInterface()) { + if (auto provider = providerForAccessible(accessible)) { + QBStr message{ event->message() }; + QAccessible::AnnouncementPoliteness prio = event->politeness(); + NotificationProcessing processing = (prio == QAccessible::AnnouncementPoliteness::Assertive) + ? NotificationProcessing_ImportantAll + : NotificationProcessing_All; + QBStr activityId{ QString::fromLatin1("") }; + QWindowsUiaWrapper::instance()->raiseNotificationEvent(provider.Get(), NotificationKind_Other, processing, message.bstr(), + activityId.bstr()); + + } + } +} + +HRESULT STDMETHODCALLTYPE QWindowsUiaMainProvider::QueryInterface(REFIID iid, LPVOID *iface) +{ + HRESULT result = QComObject::QueryInterface(iid, iface); + + if (SUCCEEDED(result) && iid == __uuidof(IRawElementProviderFragmentRoot)) { + QAccessibleInterface *accessible = accessibleInterface(); + if (accessible && hwndForAccessible(accessible)) { + result = S_OK; + } else { + Release(); + result = E_NOINTERFACE; + *iface = nullptr; + } + } + + return result; +} + +HRESULT QWindowsUiaMainProvider::get_ProviderOptions(ProviderOptions *pRetVal) +{ + if (!pRetVal) + return E_INVALIDARG; + // We are STA, (OleInitialize()). + *pRetVal = static_cast(ProviderOptions_ServerSideProvider | ProviderOptions_UseComThreading); + return S_OK; +} + +// Return providers for specific control patterns +HRESULT QWindowsUiaMainProvider::GetPatternProvider(PATTERNID idPattern, IUnknown **pRetVal) +{ + qCDebug(lcQpaUiAutomation) << __FUNCTION__ << idPattern; + + if (!pRetVal) + return E_INVALIDARG; + *pRetVal = nullptr; + + QAccessibleInterface *accessible = accessibleInterface(); + if (!accessible) + return UIA_E_ELEMENTNOTAVAILABLE; + + switch (idPattern) { + case UIA_WindowPatternId: + if (accessible->parent() && (accessible->parent()->role() == QAccessible::Application)) { + *pRetVal = makeComObject(id()).Detach(); + } + break; + case UIA_TextPatternId: + case UIA_TextPattern2Id: + // All text controls. + if (accessible->textInterface()) { + *pRetVal = makeComObject(id()).Detach(); + } + break; + case UIA_ValuePatternId: + // All non-static controls support the Value pattern. + if (accessible->role() != QAccessible::StaticText) + *pRetVal = makeComObject(id()).Detach(); + break; + case UIA_RangeValuePatternId: + // Controls providing a numeric value within a range (e.g., sliders, scroll bars, dials). + if (accessible->valueInterface()) { + *pRetVal = makeComObject(id()).Detach(); + } + break; + case UIA_TogglePatternId: + // Checkboxes and other checkable controls. + if (accessible->state().checkable) + *pRetVal = makeComObject(id()).Detach(); + break; + case UIA_SelectionPatternId: + case UIA_SelectionPattern2Id: + // Selections via QAccessibleSelectionInterface or lists of items. + if (accessible->selectionInterface() + || accessible->role() == QAccessible::List + || accessible->role() == QAccessible::PageTabList) { + *pRetVal = makeComObject(id()).Detach(); + } + break; + case UIA_SelectionItemPatternId: + // Parent supports selection interface or items within a list and radio buttons. + if ((accessible->parent() && accessible->parent()->selectionInterface()) + || (accessible->role() == QAccessible::RadioButton) + || (accessible->role() == QAccessible::ListItem) + || (accessible->role() == QAccessible::PageTab)) { + *pRetVal = makeComObject(id()).Detach(); + } + break; + case UIA_TablePatternId: + // Table/tree. + if (accessible->tableInterface() + && ((accessible->role() == QAccessible::Table) || (accessible->role() == QAccessible::Tree))) { + *pRetVal = makeComObject(id()).Detach(); + } + break; + case UIA_TableItemPatternId: + // Item within a table/tree. + if (accessible->tableCellInterface() + && ((accessible->role() == QAccessible::Cell) || (accessible->role() == QAccessible::TreeItem))) { + *pRetVal = makeComObject(id()).Detach(); + } + break; + case UIA_GridPatternId: + // Table/tree. + if (accessible->tableInterface() + && ((accessible->role() == QAccessible::Table) || (accessible->role() == QAccessible::Tree))) { + *pRetVal = makeComObject(id()).Detach(); + } + break; + case UIA_GridItemPatternId: + // Item within a table/tree. + if (accessible->tableCellInterface() + && ((accessible->role() == QAccessible::Cell) || (accessible->role() == QAccessible::TreeItem))) { + *pRetVal = makeComObject(id()).Detach(); + } + break; + case UIA_InvokePatternId: + // Things that have an invokable action (e.g., simple buttons). + if (accessible->actionInterface()) { + *pRetVal = makeComObject(id()).Detach(); + } + break; + case UIA_ExpandCollapsePatternId: + // Menu items with submenus. + if ((accessible->role() == QAccessible::MenuItem + && accessible->childCount() > 0 + && accessible->child(0)->role() == QAccessible::PopupMenu) + || accessible->role() == QAccessible::ComboBox + || (accessible->role() == QAccessible::TreeItem && accessible->state().expandable)) { + *pRetVal = makeComObject(id()).Detach(); + } + break; + default: + break; + } + + return S_OK; +} + +void QWindowsUiaMainProvider::fillVariantArrayForRelation(QAccessibleInterface* accessible, + QAccessible::Relation relation, VARIANT *pRetVal) +{ + Q_ASSERT(accessible); + + typedef QPair RelationPair; + const QList relationInterfaces = accessible->relations(relation); + if (relationInterfaces.empty()) + return; + + SAFEARRAY *elements = SafeArrayCreateVector(VT_UNKNOWN, 0, relationInterfaces.size()); + for (LONG i = 0; i < relationInterfaces.size(); ++i) { + if (ComPtr provider = + providerForAccessible(relationInterfaces.at(i).first)) { + SafeArrayPutElement(elements, &i, provider.Get()); + } + } + + pRetVal->vt = VT_UNKNOWN | VT_ARRAY; + pRetVal->parray = elements; +} + +void QWindowsUiaMainProvider::setAriaProperties(QAccessibleInterface *accessible, VARIANT *pRetVal) +{ + Q_ASSERT(accessible); + + QAccessibleAttributesInterface *attributesIface = accessible->attributesInterface(); + if (!attributesIface) + return; + + QString ariaString; + const QList attrKeys = attributesIface->attributeKeys(); + for (qsizetype i = 0; i < attrKeys.size(); ++i) { + if (i != 0) + ariaString += QStringLiteral(";"); + const QAccessible::Attribute key = attrKeys.at(i); + const QVariant value = attributesIface->attributeValue(key); + // see "Core Accessibility API Mappings" spec: https://www.w3.org/TR/core-aam-1.2/ + switch (key) { + case QAccessible::Attribute::Custom: + { + // forward custom attributes as-is + Q_ASSERT((value.canConvert>())); + const QHash attrMap = value.value>(); + for (auto [name, val] : attrMap.asKeyValueRange()) { + if (name != *attrMap.keyBegin()) + ariaString += QStringLiteral(";"); + ariaString += name + QStringLiteral("=") + val; + } + break; + } + case QAccessible::Attribute::Level: + Q_ASSERT(value.canConvert()); + ariaString += QStringLiteral("level=") + QString::number(value.toInt()); + break; + default: + break; + } + } + + *pRetVal = QComVariant{ ariaString }.release(); +} + +void QWindowsUiaMainProvider::setStyle(QAccessibleInterface *accessible, VARIANT *pRetVal) +{ + Q_ASSERT(accessible); + + QAccessibleAttributesInterface *attributesIface = accessible->attributesInterface(); + if (!attributesIface) + return; + + // currently, only heading styles are implemented here + if (accessible->role() != QAccessible::Role::Heading) + return; + + const QVariant levelVariant = attributesIface->attributeValue(QAccessible::Attribute::Level); + if (!levelVariant.isValid()) + return; + + Q_ASSERT(levelVariant.canConvert()); + // UIA only has styles for heading levels 1-9 + const int level = levelVariant.toInt(); + if (level < 1 || level > 9) + return; + + const long styleId = styleIdForHeadingLevel(level); + *pRetVal = QComVariant{ styleId }.release(); +} + +int QWindowsUiaMainProvider::styleIdForHeadingLevel(int headingLevel) +{ + // only heading levels 1-9 have a corresponding UIA style ID + Q_ASSERT(headingLevel > 0 && headingLevel <= 9); + + static constexpr int styles[] = { + StyleId_Heading1, + StyleId_Heading2, + StyleId_Heading3, + StyleId_Heading4, + StyleId_Heading5, + StyleId_Heading6, + StyleId_Heading7, + StyleId_Heading8, + StyleId_Heading9, + }; + + return styles[headingLevel - 1]; +} + +HRESULT QWindowsUiaMainProvider::GetPropertyValue(PROPERTYID idProp, VARIANT *pRetVal) +{ + qCDebug(lcQpaUiAutomation) << __FUNCTION__ << idProp; + + if (!pRetVal) + return E_INVALIDARG; + clearVariant(pRetVal); + + QAccessibleInterface *accessible = accessibleInterface(); + if (!accessible) + return UIA_E_ELEMENTNOTAVAILABLE; + + bool topLevelWindow = accessible->parent() && (accessible->parent()->role() == QAccessible::Application); + + switch (idProp) { + case UIA_ProcessIdPropertyId: + // PID + *pRetVal = QComVariant{ static_cast(GetCurrentProcessId()) }.release(); + break; + case UIA_AccessKeyPropertyId: + // Accelerator key. + *pRetVal = QComVariant{ accessible->text(QAccessible::Accelerator) }.release(); + break; + case UIA_AriaPropertiesPropertyId: + setAriaProperties(accessible, pRetVal); + break; + case UIA_AutomationIdPropertyId: + // Automation ID, which can be used by tools to select a specific control in the UI. + *pRetVal = QComVariant{ QAccessibleBridgeUtils::accessibleId(accessible) }.release(); + break; + case UIA_ClassNamePropertyId: + // Class name. + if (QObject *o = accessible->object()) { + QString className = QLatin1StringView(o->metaObject()->className()); + *pRetVal = QComVariant{ className }.release(); + } + break; + case UIA_DescribedByPropertyId: + fillVariantArrayForRelation(accessible, QAccessible::DescriptionFor, pRetVal); + break; + case UIA_FlowsFromPropertyId: + fillVariantArrayForRelation(accessible, QAccessible::FlowsTo, pRetVal); + break; + case UIA_FlowsToPropertyId: + fillVariantArrayForRelation(accessible, QAccessible::FlowsFrom, pRetVal); + break; + case UIA_FrameworkIdPropertyId: + *pRetVal = QComVariant{ QStringLiteral("Qt") }.release(); + break; + case UIA_ControlTypePropertyId: + if (topLevelWindow) { + // Reports a top-level widget as a window, instead of "custom". + *pRetVal = QComVariant{ UIA_WindowControlTypeId }.release(); + } else { + // Control type converted from role. + auto controlType = roleToControlTypeId(accessible->role()); + + // The native OSK should be disabled if the Qt OSK is in use, + // or if disabled via application attribute. + static bool imModuleEmpty = QPlatformInputContextFactory::requested().isEmpty(); + bool nativeVKDisabled = QCoreApplication::testAttribute(Qt::AA_DisableNativeVirtualKeyboard); + + // If we want to disable the native OSK auto-showing + // we have to report text fields as non-editable. + if (controlType == UIA_EditControlTypeId && (!imModuleEmpty || nativeVKDisabled)) + controlType = UIA_TextControlTypeId; + + *pRetVal = QComVariant{ controlType }.release(); + } + break; + case UIA_HelpTextPropertyId: + *pRetVal = QComVariant{ accessible->text(QAccessible::Help) }.release(); + break; + case UIA_HasKeyboardFocusPropertyId: + if (topLevelWindow) { + // Windows set the active state to true when they are focused + *pRetVal = QComVariant{ accessible->state().active ? true : false }.release(); + } else { + *pRetVal = QComVariant{ accessible->state().focused ? true : false }.release(); + } + break; + case UIA_IsKeyboardFocusablePropertyId: + if (topLevelWindow) { + // Windows should always be focusable + *pRetVal = QComVariant{ true }.release(); + } else { + *pRetVal = QComVariant{ accessible->state().focusable ? true : false }.release(); + } + break; + case UIA_IsOffscreenPropertyId: + *pRetVal = QComVariant{ accessible->state().offscreen ? true : false }.release(); + break; + case UIA_IsContentElementPropertyId: + *pRetVal = QComVariant{ true }.release(); + break; + case UIA_IsControlElementPropertyId: + *pRetVal = QComVariant{ true }.release(); + break; + case UIA_IsEnabledPropertyId: + *pRetVal = QComVariant{ !accessible->state().disabled }.release(); + break; + case UIA_IsPasswordPropertyId: + *pRetVal = QComVariant{ accessible->role() == QAccessible::EditableText + && accessible->state().passwordEdit } + .release(); + break; + case UIA_IsPeripheralPropertyId: + // True for peripheral UIs. + if (QWindow *window = windowForAccessible(accessible)) { + const Qt::WindowType wt = window->type(); + *pRetVal = QComVariant{ wt == Qt::Popup || wt == Qt::ToolTip || wt == Qt::SplashScreen } + .release(); + } + break; + case UIA_IsDialogPropertyId: + *pRetVal = QComVariant{ accessible->role() == QAccessible::Dialog + || accessible->role() == QAccessible::AlertMessage } + .release(); + break; + case UIA_FullDescriptionPropertyId: + *pRetVal = QComVariant{ accessible->text(QAccessible::Description) }.release(); + break; + case UIA_NamePropertyId: { + QString name = accessible->text(QAccessible::Name); + if (name.isEmpty() && topLevelWindow) + name = QCoreApplication::applicationName(); + *pRetVal = QComVariant{ name }.release(); + break; + } + case UIA_StyleIdAttributeId: + setStyle(accessible, pRetVal); + break; + default: + break; + } + return S_OK; +} + +HRESULT QWindowsUiaMainProvider::get_HostRawElementProvider(IRawElementProviderSimple **pRetVal) +{ + qCDebug(lcQpaUiAutomation) << __FUNCTION__ << this; + + if (!pRetVal) + return E_INVALIDARG; + *pRetVal = nullptr; + + // Returns a host provider only for controls associated with a native window handle. Others should return NULL. + if (QAccessibleInterface *accessible = accessibleInterface()) { + if (HWND hwnd = hwndForAccessible(accessible)) { + return QWindowsUiaWrapper::instance()->hostProviderFromHwnd(hwnd, pRetVal); + } + } + return S_OK; +} + +// Navigates within the tree of accessible controls. +HRESULT QWindowsUiaMainProvider::Navigate(NavigateDirection direction, IRawElementProviderFragment **pRetVal) +{ + qCDebug(lcQpaUiAutomation) << __FUNCTION__ << direction << " this: " << this; + + if (!pRetVal) + return E_INVALIDARG; + *pRetVal = nullptr; + + QAccessibleInterface *accessible = accessibleInterface(); + if (!accessible) + return UIA_E_ELEMENTNOTAVAILABLE; + + QAccessibleInterface *targetacc = nullptr; + + if (direction == NavigateDirection_Parent) { + if (QAccessibleInterface *parent = accessible->parent()) { + // The Application's children are considered top level objects. + if (parent->isValid() && parent->role() != QAccessible::Application) { + targetacc = parent; + } + } + } else { + QAccessibleInterface *parent = nullptr; + int index = 0; + int incr = 1; + switch (direction) { + case NavigateDirection_FirstChild: + parent = accessible; + index = 0; + incr = 1; + break; + case NavigateDirection_LastChild: + parent = accessible; + index = accessible->childCount() - 1; + incr = -1; + break; + case NavigateDirection_NextSibling: + if ((parent = accessible->parent())) + index = parent->indexOfChild(accessible) + 1; + incr = 1; + break; + case NavigateDirection_PreviousSibling: + if ((parent = accessible->parent())) + index = parent->indexOfChild(accessible) - 1; + incr = -1; + break; + default: + Q_UNREACHABLE(); + break; + } + + if (parent && parent->isValid()) { + for (int count = parent->childCount(); index >= 0 && index < count; index += incr) { + if (QAccessibleInterface *child = parent->child(index)) { + if (child->isValid() && !child->state().invisible) { + targetacc = child; + break; + } + } + } + } + } + + if (targetacc) + *pRetVal = providerForAccessible(targetacc).Detach(); + return S_OK; +} + +// Returns a unique id assigned to the UI element, used as key by the UI Automation framework. +HRESULT QWindowsUiaMainProvider::GetRuntimeId(SAFEARRAY **pRetVal) +{ + qCDebug(lcQpaUiAutomation) << __FUNCTION__ << this; + + if (!pRetVal) + return E_INVALIDARG; + *pRetVal = nullptr; + + QAccessibleInterface *accessible = accessibleInterface(); + if (!accessible) + return UIA_E_ELEMENTNOTAVAILABLE; + + // The UiaAppendRuntimeId constant is used to make then ID unique + // among multiple instances running on the system. + int rtId[] = { UiaAppendRuntimeId, int(id()) }; + + if ((*pRetVal = SafeArrayCreateVector(VT_I4, 0, 2))) { + for (LONG i = 0; i < 2; ++i) + SafeArrayPutElement(*pRetVal, &i, &rtId[i]); + } + return S_OK; +} + +// Returns the bounding rectangle for the accessible control. +HRESULT QWindowsUiaMainProvider::get_BoundingRectangle(UiaRect *pRetVal) +{ + qCDebug(lcQpaUiAutomation) << __FUNCTION__ << this; + + if (!pRetVal) + return E_INVALIDARG; + + QAccessibleInterface *accessible = accessibleInterface(); + if (!accessible) + return UIA_E_ELEMENTNOTAVAILABLE; + + QWindow *window = windowForAccessible(accessible); + if (!window) + return UIA_E_ELEMENTNOTAVAILABLE; + + rectToNativeUiaRect(accessible->rect(), window, pRetVal); + return S_OK; +} + +HRESULT QWindowsUiaMainProvider::GetEmbeddedFragmentRoots(SAFEARRAY **pRetVal) +{ + qCDebug(lcQpaUiAutomation) << __FUNCTION__ << this; + + if (!pRetVal) + return E_INVALIDARG; + *pRetVal = nullptr; + // No embedded roots. + return S_OK; +} + +// Sets focus to the control. +HRESULT QWindowsUiaMainProvider::SetFocus() +{ + qCDebug(lcQpaUiAutomation) << __FUNCTION__ << this; + + QAccessibleInterface *accessible = accessibleInterface(); + if (!accessible) + return UIA_E_ELEMENTNOTAVAILABLE; + + QAccessibleActionInterface *actionInterface = accessible->actionInterface(); + if (!actionInterface) + return UIA_E_ELEMENTNOTAVAILABLE; + + actionInterface->doAction(QAccessibleActionInterface::setFocusAction()); + return S_OK; +} + +HRESULT QWindowsUiaMainProvider::get_FragmentRoot(IRawElementProviderFragmentRoot **pRetVal) +{ + qCDebug(lcQpaUiAutomation) << __FUNCTION__ << this; + + if (!pRetVal) + return E_INVALIDARG; + *pRetVal = nullptr; + + // Our UI Automation implementation considers the window as the root for + // non-native controls/fragments. + if (QAccessibleInterface *accessible = accessibleInterface()) { + if (QWindow *window = windowForAccessible(accessible)) { + if (QAccessibleInterface *rootacc = window->accessibleRoot()) + *pRetVal = providerForAccessible(rootacc).Detach(); + } + } + return S_OK; +} + +// Returns a provider for the UI element present at the specified screen coordinates. +HRESULT QWindowsUiaMainProvider::ElementProviderFromPoint(double x, double y, IRawElementProviderFragment **pRetVal) +{ + qCDebug(lcQpaUiAutomation) << __FUNCTION__ << x << y; + + if (!pRetVal) { + return E_INVALIDARG; + } + *pRetVal = nullptr; + + QAccessibleInterface *accessible = accessibleInterface(); + if (!accessible) + return UIA_E_ELEMENTNOTAVAILABLE; + + QWindow *window = windowForAccessible(accessible); + if (!window) + return UIA_E_ELEMENTNOTAVAILABLE; + + // Scales coordinates from High DPI screens. + UiaPoint uiaPoint = {x, y}; + QPoint point; + nativeUiaPointToPoint(uiaPoint, window, &point); + + QAccessibleInterface *targetacc = accessible->childAt(point.x(), point.y()); + + if (targetacc) { + QAccessibleInterface *acc = targetacc; + // Controls can be embedded within grouping elements. By default returns the innermost control. + while (acc) { + targetacc = acc; + // For accessibility tools it may be better to return the text element instead of its subcomponents. + if (targetacc->textInterface()) break; + acc = acc->childAt(point.x(), point.y()); + } + *pRetVal = providerForAccessible(targetacc).Detach(); + } + return S_OK; +} + +// Returns the fragment with focus. +HRESULT QWindowsUiaMainProvider::GetFocus(IRawElementProviderFragment **pRetVal) +{ + qCDebug(lcQpaUiAutomation) << __FUNCTION__ << this; + + if (!pRetVal) + return E_INVALIDARG; + *pRetVal = nullptr; + + if (QAccessibleInterface *accessible = accessibleInterface()) { + if (QAccessibleInterface *focusacc = accessible->focusChild()) { + *pRetVal = providerForAccessible(focusacc).Detach(); + } + } + return S_OK; +} + +QT_END_NAMESPACE + +#endif // QT_CONFIG(accessibility) diff --git a/qtbase/src/plugins/platforms/windows/uiautomation/qwindowsuiawrapper.cpp b/qtbase/src/plugins/platforms/windows/uiautomation/qwindowsuiawrapper.cpp new file mode 100644 index 00000000..caeb8e1d --- /dev/null +++ b/qtbase/src/plugins/platforms/windows/uiautomation/qwindowsuiawrapper.cpp @@ -0,0 +1,89 @@ +// Copyright (C) 2017 The Qt Company Ltd. +// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR LGPL-3.0-only OR GPL-2.0-only OR GPL-3.0-only + +#include + +#include "qwindowsuiawrapper_p.h" +#include + +QT_BEGIN_NAMESPACE + +// private constructor +QWindowsUiaWrapper::QWindowsUiaWrapper() +{ + QSystemLibrary uiaLib(QStringLiteral("UIAutomationCore")); + if (uiaLib.load()) { + m_pUiaReturnRawElementProvider = reinterpret_cast(uiaLib.resolve("UiaReturnRawElementProvider")); + m_pUiaHostProviderFromHwnd = reinterpret_cast(uiaLib.resolve("UiaHostProviderFromHwnd")); + m_pUiaRaiseAutomationPropertyChangedEvent = reinterpret_cast(uiaLib.resolve("UiaRaiseAutomationPropertyChangedEvent")); + m_pUiaRaiseAutomationEvent = reinterpret_cast(uiaLib.resolve("UiaRaiseAutomationEvent")); + m_pUiaRaiseNotificationEvent = reinterpret_cast(uiaLib.resolve("UiaRaiseNotificationEvent")); + m_pUiaClientsAreListening = reinterpret_cast(uiaLib.resolve("UiaClientsAreListening")); + } +} + +QWindowsUiaWrapper::~QWindowsUiaWrapper() +{ +} + +// shared instance +QWindowsUiaWrapper *QWindowsUiaWrapper::instance() +{ + static QWindowsUiaWrapper wrapper; + return &wrapper; +} + +// True if all symbols resolved. +BOOL QWindowsUiaWrapper::ready() +{ + return m_pUiaReturnRawElementProvider + && m_pUiaHostProviderFromHwnd + && m_pUiaRaiseAutomationPropertyChangedEvent + && m_pUiaRaiseAutomationEvent + && m_pUiaClientsAreListening; +} + +BOOL QWindowsUiaWrapper::clientsAreListening() +{ + if (!m_pUiaClientsAreListening) + return FALSE; + return m_pUiaClientsAreListening(); +} + +LRESULT QWindowsUiaWrapper::returnRawElementProvider(HWND hwnd, WPARAM wParam, LPARAM lParam, IRawElementProviderSimple *el) +{ + if (!m_pUiaReturnRawElementProvider) + return static_cast(NULL); + return m_pUiaReturnRawElementProvider(hwnd, wParam, lParam, el); +} + +HRESULT QWindowsUiaWrapper::hostProviderFromHwnd(HWND hwnd, IRawElementProviderSimple **ppProvider) +{ + if (!m_pUiaHostProviderFromHwnd) + return UIA_E_NOTSUPPORTED; + return m_pUiaHostProviderFromHwnd(hwnd, ppProvider); +} + +HRESULT QWindowsUiaWrapper::raiseAutomationPropertyChangedEvent(IRawElementProviderSimple *pProvider, PROPERTYID id, VARIANT oldValue, VARIANT newValue) +{ + if (!m_pUiaRaiseAutomationPropertyChangedEvent) + return UIA_E_NOTSUPPORTED; + return m_pUiaRaiseAutomationPropertyChangedEvent(pProvider, id, oldValue, newValue); +} + +HRESULT QWindowsUiaWrapper::raiseAutomationEvent(IRawElementProviderSimple *pProvider, EVENTID id) +{ + if (!m_pUiaRaiseAutomationEvent) + return UIA_E_NOTSUPPORTED; + return m_pUiaRaiseAutomationEvent(pProvider, id); +} + +HRESULT QWindowsUiaWrapper::raiseNotificationEvent(IRawElementProviderSimple *pProvider, NotificationKind notificationKind, NotificationProcessing notificationProcessing, BSTR displayString, BSTR activityId) +{ + if (!m_pUiaRaiseNotificationEvent) + return UIA_E_NOTSUPPORTED; + return m_pUiaRaiseNotificationEvent(pProvider, notificationKind, notificationProcessing, displayString, activityId); +} + +QT_END_NAMESPACE + diff --git a/qtbase/src/plugins/platforms/windows/uiautomation/qwindowsuiawrapper_p.h b/qtbase/src/plugins/platforms/windows/uiautomation/qwindowsuiawrapper_p.h new file mode 100644 index 00000000..e0053181 --- /dev/null +++ b/qtbase/src/plugins/platforms/windows/uiautomation/qwindowsuiawrapper_p.h @@ -0,0 +1,57 @@ +// Copyright (C) 2017 The Qt Company Ltd. +// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR LGPL-3.0-only OR GPL-2.0-only OR GPL-3.0-only + +#ifndef QWINDOWSUIAWRAPPER_H +#define QWINDOWSUIAWRAPPER_H + +// +// W A R N I N G +// ------------- +// +// This file is not part of the Qt API. It exists for the convenience +// of other Qt classes. This header file may change from version to +// version without notice, or even be removed. +// +// We mean it. +// + +#include + +QT_REQUIRE_CONFIG(accessibility); + +QT_BEGIN_NAMESPACE + +class QWindowsUiaWrapper +{ + QWindowsUiaWrapper(); + virtual ~QWindowsUiaWrapper(); +public: + static QWindowsUiaWrapper *instance(); + BOOL ready(); + BOOL clientsAreListening(); + LRESULT returnRawElementProvider(HWND hwnd, WPARAM wParam, LPARAM lParam, IRawElementProviderSimple *el); + HRESULT hostProviderFromHwnd(HWND hwnd, IRawElementProviderSimple **ppProvider); + HRESULT raiseAutomationPropertyChangedEvent(IRawElementProviderSimple *pProvider, PROPERTYID id, VARIANT oldValue, VARIANT newValue); + HRESULT raiseAutomationEvent(IRawElementProviderSimple *pProvider, EVENTID id); + HRESULT raiseNotificationEvent(IRawElementProviderSimple *pProvider, NotificationKind notificationKind, NotificationProcessing notificationProcessing, BSTR displayString, BSTR activityId); + +private: + typedef LRESULT (WINAPI *PtrUiaReturnRawElementProvider)(HWND, WPARAM, LPARAM, IRawElementProviderSimple *); + typedef HRESULT (WINAPI *PtrUiaHostProviderFromHwnd)(HWND, IRawElementProviderSimple **); + typedef HRESULT (WINAPI *PtrUiaRaiseAutomationPropertyChangedEvent)(IRawElementProviderSimple *, PROPERTYID, VARIANT, VARIANT); + typedef HRESULT (WINAPI *PtrUiaRaiseAutomationEvent)(IRawElementProviderSimple *, EVENTID); + typedef HRESULT (WINAPI *PtrUiaRaiseNotificationEvent)(IRawElementProviderSimple *, NotificationKind, NotificationProcessing, BSTR, BSTR); + + typedef BOOL (WINAPI *PtrUiaClientsAreListening)(); + PtrUiaReturnRawElementProvider m_pUiaReturnRawElementProvider = nullptr; + PtrUiaHostProviderFromHwnd m_pUiaHostProviderFromHwnd = nullptr; + PtrUiaRaiseAutomationPropertyChangedEvent m_pUiaRaiseAutomationPropertyChangedEvent = nullptr; + PtrUiaRaiseAutomationEvent m_pUiaRaiseAutomationEvent = nullptr; + PtrUiaRaiseNotificationEvent m_pUiaRaiseNotificationEvent = nullptr; + PtrUiaClientsAreListening m_pUiaClientsAreListening = nullptr; +}; + +QT_END_NAMESPACE + +#endif //QWINDOWSUIAWRAPPER_H + diff --git a/qtbase/src/widgets/styles/qwindowsstyle.cpp b/qtbase/src/widgets/styles/qwindowsstyle.cpp index 76ced203..f4e0e7e8 100644 --- a/qtbase/src/widgets/styles/qwindowsstyle.cpp +++ b/qtbase/src/widgets/styles/qwindowsstyle.cpp @@ -90,11 +90,6 @@ enum QSliderDirection { SlUp, SlDown, SlLeft, SlRight }; QWindowsStylePrivate::QWindowsStylePrivate() = default; -qreal QWindowsStylePrivate::appDevicePixelRatio() -{ - return qApp->devicePixelRatio(); -} - // Returns \c true if the toplevel parent of \a widget has seen the Alt-key bool QWindowsStylePrivate::hasSeenAlt(const QWidget *widget) const { @@ -459,46 +454,6 @@ int QWindowsStyle::pixelMetric(PixelMetric pm, const QStyleOption *opt, const QW QPixmap QWindowsStyle::standardPixmap(StandardPixmap standardPixmap, const QStyleOption *opt, const QWidget *widget) const { -#if defined(Q_OS_WIN) - QPixmap desktopIcon; - switch(standardPixmap) { - case SP_DriveCDIcon: - case SP_DriveDVDIcon: - case SP_DriveNetIcon: - case SP_DriveHDIcon: - case SP_DriveFDIcon: - case SP_FileIcon: - case SP_FileLinkIcon: - case SP_DirLinkIcon: - case SP_DirClosedIcon: - case SP_DesktopIcon: - case SP_ComputerIcon: - case SP_DirOpenIcon: - case SP_FileDialogNewFolder: - case SP_DirHomeIcon: - case SP_TrashIcon: - case SP_VistaShield: - if (const QPlatformTheme *theme = QGuiApplicationPrivate::platformTheme()) { - QPlatformTheme::StandardPixmap sp = static_cast(standardPixmap); - desktopIcon = theme->standardPixmap(sp, QSizeF(16, 16)); - } - break; - case SP_MessageBoxInformation: - case SP_MessageBoxWarning: - case SP_MessageBoxCritical: - case SP_MessageBoxQuestion: - if (const QPlatformTheme *theme = QGuiApplicationPrivate::platformTheme()) { - QPlatformTheme::StandardPixmap sp = static_cast(standardPixmap); - desktopIcon = theme->standardPixmap(sp, QSizeF()); - } - break; - default: - break; - } - if (!desktopIcon.isNull()) { - return desktopIcon; - } -#endif // Q_OS_WIN return QCommonStyle::standardPixmap(standardPixmap, opt, widget); }