summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--cmake/QtBaseGlobalTargets.cmake15
-rw-r--r--cmake/QtBuildHelpers.cmake1
-rw-r--r--cmake/QtBuildOptionsHelpers.cmake17
-rw-r--r--cmake/QtBuildRepoHelpers.cmake1
-rw-r--r--coin/instructions/prepare_building_env.yaml3
-rw-r--r--src/corelib/Qt6WasmMacros.cmake9
-rw-r--r--src/corelib/doc/src/cpp20-overview.qdoc10
-rw-r--r--src/corelib/itemmodels/qrangemodel.cpp103
-rw-r--r--src/corelib/itemmodels/qrangemodel.h20
-rw-r--r--src/corelib/itemmodels/qrangemodel_impl.h39
-rw-r--r--src/corelib/kernel/qabstracteventdispatcher.cpp6
-rw-r--r--src/corelib/kernel/qbasictimer.cpp9
-rw-r--r--src/corelib/platform/windows/qcomobject_p.h13
-rw-r--r--src/corelib/serialization/qtextstream.cpp47
-rw-r--r--src/corelib/serialization/qtextstream_p.h12
-rw-r--r--src/corelib/thread/qthread_p.h2
-rw-r--r--src/dbus/dbus_minimal_p.h1
-rw-r--r--src/dbus/qdbus_symbols_p.h1
-rw-r--r--src/dbus/qdbusabstractadaptor.cpp1
-rw-r--r--src/dbus/qdbusabstractadaptor.h1
-rw-r--r--src/dbus/qdbusabstractadaptor_p.h1
-rw-r--r--src/dbus/qdbusabstractinterface.cpp1
-rw-r--r--src/dbus/qdbusabstractinterface.h1
-rw-r--r--src/dbus/qdbusabstractinterface_p.h1
-rw-r--r--src/dbus/qdbusargument.h42
-rw-r--r--src/dbus/qdbusargument_p.h1
-rw-r--r--src/dbus/qdbusconnection.cpp1
-rw-r--r--src/dbus/qdbusconnection.h1
-rw-r--r--src/dbus/qdbusconnection_p.h1
-rw-r--r--src/dbus/qdbusconnectioninterface.cpp1
-rw-r--r--src/dbus/qdbusconnectioninterface.h1
-rw-r--r--src/dbus/qdbusconnectionmanager.cpp1
-rw-r--r--src/dbus/qdbusconnectionmanager_p.h1
-rw-r--r--src/dbus/qdbuscontext.cpp1
-rw-r--r--src/dbus/qdbuscontext.h1
-rw-r--r--src/dbus/qdbuscontext_p.h1
-rw-r--r--src/dbus/qdbuserror.cpp1
-rw-r--r--src/dbus/qdbuserror.h1
-rw-r--r--src/dbus/qdbusextratypes.cpp1
-rw-r--r--src/dbus/qdbusextratypes.h1
-rw-r--r--src/dbus/qdbusintegrator.cpp1
-rw-r--r--src/dbus/qdbusintegrator_p.h1
-rw-r--r--src/dbus/qdbusinterface.cpp1
-rw-r--r--src/dbus/qdbusinterface.h1
-rw-r--r--src/dbus/qdbusinterface_p.h1
-rw-r--r--src/dbus/qdbusinternalfilters.cpp1
-rw-r--r--src/dbus/qdbusintrospection.cpp1
-rw-r--r--src/dbus/qdbusintrospection_p.h1
-rw-r--r--src/dbus/qdbusmessage.cpp1
-rw-r--r--src/dbus/qdbusmessage.h1
-rw-r--r--src/dbus/qdbusmessage_p.h1
-rw-r--r--src/dbus/qdbusmetaobject.cpp1
-rw-r--r--src/dbus/qdbusmetaobject_p.h1
-rw-r--r--src/dbus/qdbusmetatype.cpp1
-rw-r--r--src/dbus/qdbusmetatype.h1
-rw-r--r--src/dbus/qdbusmetatype_p.h1
-rw-r--r--src/dbus/qdbusmisc.cpp1
-rw-r--r--src/dbus/qdbuspendingcall.cpp3
-rw-r--r--src/dbus/qdbuspendingcall.h1
-rw-r--r--src/dbus/qdbuspendingcall_p.h1
-rw-r--r--src/dbus/qdbuspendingreply.cpp1
-rw-r--r--src/dbus/qdbuspendingreply.h1
-rw-r--r--src/dbus/qdbusreply.cpp1
-rw-r--r--src/dbus/qdbusreply.h1
-rw-r--r--src/dbus/qdbusserver.cpp1
-rw-r--r--src/dbus/qdbusserver.h1
-rw-r--r--src/dbus/qdbusservicewatcher.cpp1
-rw-r--r--src/dbus/qdbusservicewatcher.h1
-rw-r--r--src/dbus/qdbusthreaddebug_p.h1
-rw-r--r--src/dbus/qdbusunixfiledescriptor.cpp1
-rw-r--r--src/dbus/qdbusunixfiledescriptor.h1
-rw-r--r--src/dbus/qdbusutil.cpp1
-rw-r--r--src/dbus/qdbusutil_p.h1
-rw-r--r--src/dbus/qdbusvirtualobject.cpp1
-rw-r--r--src/dbus/qdbusvirtualobject.h1
-rw-r--r--src/dbus/qdbusxmlgenerator.cpp1
-rw-r--r--src/dbus/qdbusxmlparser.cpp1
-rw-r--r--src/dbus/qdbusxmlparser_p.h1
-rw-r--r--src/dbus/qtdbusglobal.h1
-rw-r--r--src/dbus/qtdbusglobal_p.h1
-rw-r--r--src/gui/math3d/qmatrix4x4.cpp2
-rw-r--r--src/gui/math3d/qmatrix4x4.h4
-rw-r--r--src/network/access/qhttp2connection.cpp13
-rw-r--r--src/plugins/platforms/wasm/qtloader.js3
-rw-r--r--src/plugins/platforms/wasm/qwasmclipboard.cpp110
-rw-r--r--src/plugins/platforms/wasm/qwasminputcontext.cpp309
-rw-r--r--src/plugins/platforms/wasm/qwasminputcontext.h23
-rw-r--r--src/plugins/platforms/wasm/qwasmwindow.cpp181
-rw-r--r--src/plugins/platforms/wasm/qwasmwindow.h25
-rw-r--r--src/plugins/styles/modernwindows/qwindows11style.cpp3
-rw-r--r--src/testlib/3rdparty/catch2/qt_attribution.json1
-rw-r--r--src/tools/CMakeLists.txt4
-rw-r--r--src/tools/configure.cmake6
-rw-r--r--src/tools/wasmdeployqt/CMakeLists.txt19
-rw-r--r--src/tools/wasmdeployqt/common.h26
-rw-r--r--src/tools/wasmdeployqt/jsontools.cpp101
-rw-r--r--src/tools/wasmdeployqt/jsontools.h19
-rw-r--r--src/tools/wasmdeployqt/main.cpp417
-rw-r--r--src/tools/wasmdeployqt/wasmbinary.cpp91
-rw-r--r--src/tools/wasmdeployqt/wasmbinary.h24
-rw-r--r--tests/auto/corelib/io/qtemporaryfile/tst_qtemporaryfile.cpp9
-rw-r--r--tests/auto/corelib/serialization/qtextstream/tst_qtextstream.cpp65
-rw-r--r--tests/manual/wasm/eventloop/suspendresumecontrol_auto/main.cpp2
-rw-r--r--util/wasm/preload/generate_default_preloads.sh.in19
-rwxr-xr-xutil/wasm/preload/preload_qml_imports.py98
-rwxr-xr-xutil/wasm/preload/preload_qt_plugins.py57
106 files changed, 1326 insertions, 719 deletions
diff --git a/cmake/QtBaseGlobalTargets.cmake b/cmake/QtBaseGlobalTargets.cmake
index 405b5e22635..601592277fe 100644
--- a/cmake/QtBaseGlobalTargets.cmake
+++ b/cmake/QtBaseGlobalTargets.cmake
@@ -437,21 +437,6 @@ elseif(WASM)
"${QT_BUILD_DIR}/${INSTALL_LIBEXECDIR}/qt-wasmtestrunner.py" @ONLY)
qt_install(PROGRAMS "${QT_BUILD_DIR}/${INSTALL_LIBEXECDIR}/qt-wasmtestrunner.py"
DESTINATION "${INSTALL_LIBEXECDIR}")
-
- if(QT_FEATURE_shared)
- configure_file("${CMAKE_CURRENT_SOURCE_DIR}/util/wasm/preload/preload_qml_imports.py"
- "${QT_BUILD_DIR}/${INSTALL_LIBEXECDIR}/preload_qml_imports.py" COPYONLY)
- configure_file("${CMAKE_CURRENT_SOURCE_DIR}/util/wasm/preload/preload_qt_plugins.py"
- "${QT_BUILD_DIR}/${INSTALL_LIBEXECDIR}/preload_qt_plugins.py" COPYONLY)
- configure_file("${CMAKE_CURRENT_SOURCE_DIR}/util/wasm/preload/generate_default_preloads.sh.in"
- "${QT_BUILD_DIR}/${INSTALL_LIBEXECDIR}/generate_default_preloads.sh.in" COPYONLY)
- qt_install(PROGRAMS "${QT_BUILD_DIR}/${INSTALL_LIBEXECDIR}/preload_qml_imports.py"
- DESTINATION "${INSTALL_LIBEXECDIR}")
- qt_install(PROGRAMS "${QT_BUILD_DIR}/${INSTALL_LIBEXECDIR}/preload_qt_plugins.py"
- DESTINATION "${INSTALL_LIBEXECDIR}")
- qt_install(PROGRAMS "${QT_BUILD_DIR}/${INSTALL_LIBEXECDIR}/generate_default_preloads.sh.in"
- DESTINATION "${INSTALL_LIBEXECDIR}")
- endif()
endif()
# Install CI support files to libexec.
diff --git a/cmake/QtBuildHelpers.cmake b/cmake/QtBuildHelpers.cmake
index 1fb5c6350e1..df0aa33cda2 100644
--- a/cmake/QtBuildHelpers.cmake
+++ b/cmake/QtBuildHelpers.cmake
@@ -487,7 +487,6 @@ macro(qt_internal_setup_build_and_global_variables)
qt_set_language_standards()
qt_internal_set_use_ccache()
- qt_internal_set_unity_build()
qt_internal_set_allow_symlink_in_paths()
qt_internal_set_skip_setup_deployment()
qt_internal_set_qt_allow_download()
diff --git a/cmake/QtBuildOptionsHelpers.cmake b/cmake/QtBuildOptionsHelpers.cmake
index ccd054183d4..ae8cf51e016 100644
--- a/cmake/QtBuildOptionsHelpers.cmake
+++ b/cmake/QtBuildOptionsHelpers.cmake
@@ -444,10 +444,19 @@ endmacro()
macro(qt_internal_set_unity_build)
option(QT_UNITY_BUILD "Enable unity (jumbo) build")
set(QT_UNITY_BUILD_BATCH_SIZE "32" CACHE STRING "Unity build batch size")
- if(QT_UNITY_BUILD)
- set(CMAKE_UNITY_BUILD ON)
- set(CMAKE_UNITY_BUILD_BATCH_SIZE "${QT_UNITY_BUILD_BATCH_SIZE}")
- endif()
+
+ include(CMakeDependentOption)
+ string(TOLOWER "${PROJECT_NAME}" project_name_lower)
+ cmake_dependent_option(
+ "QT_UNITY_BUILD_PROJECT_${project_name_lower}"
+ "Enable unity builds for project ${PROJECT_NAME}"
+ TRUE
+ QT_UNITY_BUILD
+ FALSE
+ )
+
+ set(CMAKE_UNITY_BUILD "${QT_UNITY_BUILD_PROJECT_${project_name_lower}}")
+ set(CMAKE_UNITY_BUILD_BATCH_SIZE "${QT_UNITY_BUILD_BATCH_SIZE}")
endmacro()
macro(qt_internal_set_allow_symlink_in_paths)
diff --git a/cmake/QtBuildRepoHelpers.cmake b/cmake/QtBuildRepoHelpers.cmake
index 86107179475..70a066f11b2 100644
--- a/cmake/QtBuildRepoHelpers.cmake
+++ b/cmake/QtBuildRepoHelpers.cmake
@@ -355,6 +355,7 @@ macro(qt_build_repo_begin)
endif()
_qt_internal_sbom_auto_begin_qt_repo_project()
+ qt_internal_set_unity_build()
endmacro()
# Runs delayed actions on some of the Qt targets.
diff --git a/coin/instructions/prepare_building_env.yaml b/coin/instructions/prepare_building_env.yaml
index 0330369662d..ac2ad89027b 100644
--- a/coin/instructions/prepare_building_env.yaml
+++ b/coin/instructions/prepare_building_env.yaml
@@ -679,6 +679,9 @@ instructions:
- type: AppendToEnvironmentVariable
variableName: COMMON_NON_QTBASE_TARGET_CMAKE_ARGS
variableValue: " -DQT_USE_VCPKG=ON -DVCPKG_INSTALLED_DIR={{.Env.VCPKG_INSTALLED_DIR}} -DVCPKG_HOST_TRIPLET={{.Env.VCPKG_HOST_TRIPLET}} -DVCPKG_TARGET_TRIPLET={{.Env.VCPKG_TARGET_TRIPLET}}"
+ - type: AppendToEnvironmentVariable
+ variableName: COMMON_TARGET_TEST_CMAKE_ARGS
+ variableValue: " -DQT_USE_VCPKG=ON -DVCPKG_INSTALLED_DIR={{.Env.VCPKG_INSTALLED_DIR}} -DVCPKG_HOST_TRIPLET={{.Env.VCPKG_HOST_TRIPLET}} -DVCPKG_TARGET_TRIPLET={{.Env.VCPKG_TARGET_TRIPLET}}"
enable_if:
condition: runtime
env_var: USE_VCPKG
diff --git a/src/corelib/Qt6WasmMacros.cmake b/src/corelib/Qt6WasmMacros.cmake
index e185ef67490..eae356679bd 100644
--- a/src/corelib/Qt6WasmMacros.cmake
+++ b/src/corelib/Qt6WasmMacros.cmake
@@ -91,15 +91,6 @@ function(_qt_internal_wasm_add_target_helpers target)
${_target_directory}/qtloader.js COPYONLY)
configure_file("${WASM_BUILD_DIR}/plugins/platforms/qtlogo.svg"
${_target_directory}/qtlogo.svg COPYONLY)
- if(QT_FEATURE_shared)
- set(TARGET_DIR "${_target_directory}")
- set(SOURCE_DIR "${CMAKE_CURRENT_SOURCE_DIR}")
- set(QT_HOST_DIR "${QT_HOST_PATH}")
- set(QT_WASM_DIR "${WASM_BUILD_DIR}")
- set(QT_INSTALL_DIR "${QT6_INSTALL_PREFIX}")
- configure_file("${WASM_BUILD_DIR}/libexec/generate_default_preloads.sh.in"
- "${_target_directory}/generate_default_preloads_for_${target}.sh" @ONLY)
- endif()
endif()
endif()
endif()
diff --git a/src/corelib/doc/src/cpp20-overview.qdoc b/src/corelib/doc/src/cpp20-overview.qdoc
index ed48e10f94a..1231394c161 100644
--- a/src/corelib/doc/src/cpp20-overview.qdoc
+++ b/src/corelib/doc/src/cpp20-overview.qdoc
@@ -12,6 +12,16 @@
This page gives a brief overview of C++20 features available in Qt.
+ \section1 WebEngine and Module-Specific Requirements
+
+ While Qt 6 only requires a C++17-compatible compiler in general,
+ the \l{Qt WebEngine} module requires a C++20-compatible compiler
+ due to its dependencies (such as Chromium).
+
+ If you are building Qt WebEngine, ensure that your compiler fully supports C++20.
+
+ See \l{qtwebengine-platform-notes.html}{Qt WebEngine Platform Notes} for detailed requirements.
+
\section1 Support for \c{std::chrono}
Various classes related to date and time have support for \l
diff --git a/src/corelib/itemmodels/qrangemodel.cpp b/src/corelib/itemmodels/qrangemodel.cpp
index 51043eb0e24..8ced35a4ae0 100644
--- a/src/corelib/itemmodels/qrangemodel.cpp
+++ b/src/corelib/itemmodels/qrangemodel.cpp
@@ -5,8 +5,48 @@
#include "qrangemodel.h"
#include <QtCore/qsize.h>
+#include <QtCore/private/qabstractitemmodel_p.h>
+
QT_BEGIN_NAMESPACE
+class QRangeModelPrivate : QAbstractItemModelPrivate
+{
+ Q_DECLARE_PUBLIC(QRangeModel)
+
+ struct Deleter { void operator()(QRangeModelImplBase *that) { that->destroy(); } };
+
+public:
+ explicit QRangeModelPrivate(std::unique_ptr<QRangeModelImplBase, Deleter> impl)
+ : impl(std::move(impl))
+ {}
+
+ template <typename Ret, typename ...Args>
+ Ret call(QRangeModelImplBase::ConstOp op, const Args &...args) const
+ {
+ Ret ret = {};
+ const auto tuple = std::tie(args...);
+ impl->callConst_fn(op, impl.get(), &ret, &tuple);
+ return ret;
+ }
+
+ template <typename Ret, typename ...Args>
+ Ret call(QRangeModelImplBase::Op op, const Args &...args)
+ {
+ Ret ret = {};
+ const auto tuple = std::tie(args...);
+ impl->call_fn(op, impl.get(), &ret, &tuple);
+ return ret;
+ }
+
+private:
+ std::unique_ptr<QRangeModelImplBase, Deleter> impl;
+};
+
+QRangeModel::QRangeModel(QRangeModelImplBase *impl, QObject *parent)
+ : QAbstractItemModel(*new QRangeModelPrivate({impl, {}}), parent)
+{
+}
+
/*!
\class QRangeModel
\inmodule QtCore
@@ -503,7 +543,8 @@ QRangeModel::~QRangeModel() = default;
*/
QModelIndex QRangeModel::index(int row, int column, const QModelIndex &parent) const
{
- return impl->callConst<QModelIndex>(QRangeModelImplBase::Index, row, column, parent);
+ Q_D(const QRangeModel);
+ return d->call<QModelIndex>(QRangeModelImplBase::Index, row, column, parent);
}
/*!
@@ -520,7 +561,8 @@ QModelIndex QRangeModel::index(int row, int column, const QModelIndex &parent) c
*/
QModelIndex QRangeModel::parent(const QModelIndex &child) const
{
- return impl->callConst<QModelIndex>(QRangeModelImplBase::Parent, child);
+ Q_D(const QRangeModel);
+ return d->call<QModelIndex>(QRangeModelImplBase::Parent, child);
}
/*!
@@ -536,7 +578,8 @@ QModelIndex QRangeModel::parent(const QModelIndex &child) const
*/
QModelIndex QRangeModel::sibling(int row, int column, const QModelIndex &index) const
{
- return impl->callConst<QModelIndex>(QRangeModelImplBase::Sibling, row, column, index);
+ Q_D(const QRangeModel);
+ return d->call<QModelIndex>(QRangeModelImplBase::Sibling, row, column, index);
}
/*!
@@ -554,7 +597,8 @@ QModelIndex QRangeModel::sibling(int row, int column, const QModelIndex &index)
*/
int QRangeModel::rowCount(const QModelIndex &parent) const
{
- return impl->callConst<int>(QRangeModelImplBase::RowCount, parent);
+ Q_D(const QRangeModel);
+ return d->call<int>(QRangeModelImplBase::RowCount, parent);
}
/*!
@@ -572,7 +616,8 @@ int QRangeModel::rowCount(const QModelIndex &parent) const
*/
int QRangeModel::columnCount(const QModelIndex &parent) const
{
- return impl->callConst<int>(QRangeModelImplBase::ColumnCount, parent);
+ Q_D(const QRangeModel);
+ return d->call<int>(QRangeModelImplBase::ColumnCount, parent);
}
/*!
@@ -589,7 +634,8 @@ int QRangeModel::columnCount(const QModelIndex &parent) const
*/
Qt::ItemFlags QRangeModel::flags(const QModelIndex &index) const
{
- return impl->callConst<Qt::ItemFlags>(QRangeModelImplBase::Flags, index);
+ Q_D(const QRangeModel);
+ return d->call<Qt::ItemFlags>(QRangeModelImplBase::Flags, index);
}
/*!
@@ -606,8 +652,8 @@ Qt::ItemFlags QRangeModel::flags(const QModelIndex &index) const
*/
QVariant QRangeModel::headerData(int section, Qt::Orientation orientation, int role) const
{
- return impl->callConst<QVariant>(QRangeModelImplBase::HeaderData,
- section, orientation, role);
+ Q_D(const QRangeModel);
+ return d->call<QVariant>(QRangeModelImplBase::HeaderData, section, orientation, role);
}
/*!
@@ -629,7 +675,8 @@ QVariant QRangeModel::headerData(int section, Qt::Orientation orientation, int r
*/
QVariant QRangeModel::data(const QModelIndex &index, int role) const
{
- return impl->callConst<QVariant>(QRangeModelImplBase::Data, index, role);
+ Q_D(const QRangeModel);
+ return d->call<QVariant>(QRangeModelImplBase::Data, index, role);
}
/*!
@@ -654,7 +701,8 @@ QVariant QRangeModel::data(const QModelIndex &index, int role) const
*/
bool QRangeModel::setData(const QModelIndex &index, const QVariant &data, int role)
{
- return impl->call<bool>(QRangeModelImplBase::SetData, index, data, role);
+ Q_D(QRangeModel);
+ return d->call<bool>(QRangeModelImplBase::SetData, index, data, role);
}
/*!
@@ -677,7 +725,8 @@ bool QRangeModel::setData(const QModelIndex &index, const QVariant &data, int ro
*/
QMap<int, QVariant> QRangeModel::itemData(const QModelIndex &index) const
{
- return impl->callConst<QMap<int, QVariant>>(QRangeModelImplBase::ItemData, index);
+ Q_D(const QRangeModel);
+ return d->call<QMap<int, QVariant>>(QRangeModelImplBase::ItemData, index);
}
/*!
@@ -708,7 +757,8 @@ QMap<int, QVariant> QRangeModel::itemData(const QModelIndex &index) const
*/
bool QRangeModel::setItemData(const QModelIndex &index, const QMap<int, QVariant> &data)
{
- return impl->call<bool>(QRangeModelImplBase::SetItemData, index, data);
+ Q_D(QRangeModel);
+ return d->call<bool>(QRangeModelImplBase::SetItemData, index, data);
}
/*!
@@ -721,7 +771,8 @@ bool QRangeModel::setItemData(const QModelIndex &index, const QMap<int, QVariant
*/
bool QRangeModel::clearItemData(const QModelIndex &index)
{
- return impl->call<bool>(QRangeModelImplBase::ClearItemData, index);
+ Q_D(QRangeModel);
+ return d->call<bool>(QRangeModelImplBase::ClearItemData, index);
}
/*
@@ -746,7 +797,8 @@ bool QRangeModel::clearItemData(const QModelIndex &index)
*/
bool QRangeModel::insertColumns(int column, int count, const QModelIndex &parent)
{
- return impl->call<bool>(QRangeModelImplBase::InsertColumns, column, count, parent);
+ Q_D(QRangeModel);
+ return d->call<bool>(QRangeModelImplBase::InsertColumns, column, count, parent);
}
/*!
@@ -760,7 +812,8 @@ bool QRangeModel::insertColumns(int column, int count, const QModelIndex &parent
*/
bool QRangeModel::removeColumns(int column, int count, const QModelIndex &parent)
{
- return impl->call<bool>(QRangeModelImplBase::RemoveColumns, column, count, parent);
+ Q_D(QRangeModel);
+ return d->call<bool>(QRangeModelImplBase::RemoveColumns, column, count, parent);
}
/*!
@@ -775,9 +828,10 @@ bool QRangeModel::removeColumns(int column, int count, const QModelIndex &parent
bool QRangeModel::moveColumns(const QModelIndex &sourceParent, int sourceColumn, int count,
const QModelIndex &destinationParent, int destinationColumn)
{
- return impl->call<bool>(QRangeModelImplBase::MoveColumns,
- sourceParent, sourceColumn, count,
- destinationParent, destinationColumn);
+ Q_D(QRangeModel);
+ return d->call<bool>(QRangeModelImplBase::MoveColumns,
+ sourceParent, sourceColumn, count,
+ destinationParent, destinationColumn);
}
/*
@@ -804,7 +858,8 @@ bool QRangeModel::moveColumns(const QModelIndex &sourceParent, int sourceColumn,
*/
bool QRangeModel::insertRows(int row, int count, const QModelIndex &parent)
{
- return impl->call<bool>(QRangeModelImplBase::InsertRows, row, count, parent);
+ Q_D(QRangeModel);
+ return d->call<bool>(QRangeModelImplBase::InsertRows, row, count, parent);
}
/*!
@@ -817,7 +872,8 @@ bool QRangeModel::insertRows(int row, int count, const QModelIndex &parent)
*/
bool QRangeModel::removeRows(int row, int count, const QModelIndex &parent)
{
- return impl->call<bool>(QRangeModelImplBase::RemoveRows, row, count, parent);
+ Q_D(QRangeModel);
+ return d->call<bool>(QRangeModelImplBase::RemoveRows, row, count, parent);
}
/*!
@@ -832,9 +888,10 @@ bool QRangeModel::removeRows(int row, int count, const QModelIndex &parent)
bool QRangeModel::moveRows(const QModelIndex &sourceParent, int sourceRow, int count,
const QModelIndex &destinationParent, int destinationRow)
{
- return impl->call<bool>(QRangeModelImplBase::MoveRows,
- sourceParent, sourceRow, count,
- destinationParent, destinationRow);
+ Q_D(QRangeModel);
+ return d->call<bool>(QRangeModelImplBase::MoveRows,
+ sourceParent, sourceRow, count,
+ destinationParent, destinationRow);
}
/*!
diff --git a/src/corelib/itemmodels/qrangemodel.h b/src/corelib/itemmodels/qrangemodel.h
index 328966f26fd..0d9add5def3 100644
--- a/src/corelib/itemmodels/qrangemodel.h
+++ b/src/corelib/itemmodels/qrangemodel.h
@@ -9,6 +9,8 @@
QT_BEGIN_NAMESPACE
+class QRangeModelPrivate;
+
class Q_CORE_EXPORT QRangeModel : public QAbstractItemModel
{
Q_OBJECT
@@ -103,10 +105,10 @@ protected:
private:
Q_DISABLE_COPY_MOVE(QRangeModel)
+ Q_DECLARE_PRIVATE(QRangeModel)
+ explicit QRangeModel(QRangeModelImplBase *impl, QObject *parent);
friend class QRangeModelImplBase;
- struct Deleter { void operator()(QRangeModelImplBase *that) { that->destroy(); } };
- std::unique_ptr<QRangeModelImplBase, Deleter> impl;
};
// implementation of forwarders
@@ -193,21 +195,21 @@ const QAbstractItemModel &QRangeModelImplBase::itemModel() const
template <typename Range, QRangeModelDetails::if_table_range<Range>>
QRangeModel::QRangeModel(Range &&range, QObject *parent)
- : QAbstractItemModel(parent)
- , impl(new QGenericTableItemModelImpl<Range>(std::forward<Range>(range), this))
+ : QRangeModel(new QGenericTableItemModelImpl<Range>(std::forward<Range>(range), this), parent)
{}
template <typename Range, QRangeModelDetails::if_tree_range<Range>>
QRangeModel::QRangeModel(Range &&range, QObject *parent)
- : QRangeModel(std::forward<Range>(range),
- QRangeModelDetails::DefaultTreeProtocol<Range>{}, parent)
+ : QRangeModel(std::forward<Range>(range), QRangeModelDetails::DefaultTreeProtocol<Range>{},
+ parent)
{}
template <typename Range, typename Protocol, QRangeModelDetails::if_tree_range<Range, Protocol>>
QRangeModel::QRangeModel(Range &&range, Protocol &&protocol, QObject *parent)
- : QAbstractItemModel(parent)
- , impl(new QGenericTreeItemModelImpl<Range, Protocol>(std::forward<Range>(range),
- std::forward<Protocol>(protocol), this))
+ : QRangeModel(new QGenericTreeItemModelImpl<Range, Protocol>(std::forward<Range>(range),
+ std::forward<Protocol>(protocol),
+ this),
+ parent)
{}
QT_END_NAMESPACE
diff --git a/src/corelib/itemmodels/qrangemodel_impl.h b/src/corelib/itemmodels/qrangemodel_impl.h
index 38378fdcc64..34c9ba235a6 100644
--- a/src/corelib/itemmodels/qrangemodel_impl.h
+++ b/src/corelib/itemmodels/qrangemodel_impl.h
@@ -604,7 +604,7 @@ public:
void destroy()
{
- call<bool>(Destroy);
+ call_fn(Destroy, this, nullptr, nullptr);
}
private:
@@ -615,6 +615,7 @@ private:
using CallConstFN = decltype(callConst);
using CallTupleFN = decltype(call);
+ friend class QRangeModelPrivate;
CallConstFN *callConst_fn;
CallTupleFN *call_fn;
QRangeModel *m_rangeModel;
@@ -647,25 +648,6 @@ protected:
inline void endMoveRows();
inline QAbstractItemModel &itemModel();
inline const QAbstractItemModel &itemModel() const;
-
-public:
- template <typename Ret, typename ...Args>
- Ret callConst(ConstOp op, const Args &...args) const
- {
- Ret ret = {};
- const auto tuple = std::tie(args...);
- callConst_fn(op, this, &ret, &tuple);
- return ret;
- }
-
- template <typename Ret, typename ...Args>
- Ret call(Op op, const Args &...args)
- {
- Ret ret = {};
- const auto tuple = std::tie(args...);
- call_fn(op, this, &ret, &tuple);
- return ret;
- }
};
template <typename Structure, typename Range,
@@ -1357,7 +1339,7 @@ public:
bool removeRows(int row, int count, const QModelIndex &parent = {})
{
- if constexpr (Structure::canRemoveRows()) {
+ if constexpr (canRemoveRows()) {
const int prevRowCount = that().rowCount(parent);
if (row < 0 || row + count > prevRowCount)
return false;
@@ -1476,10 +1458,15 @@ protected:
// row elements.
return false;
} else {
- return Structure::canInsertRows();
+ return Structure::canInsertRowsImpl();
}
}
+ static constexpr bool canRemoveRows()
+ {
+ return Structure::canRemoveRowsImpl();
+ }
+
template <typename F>
bool writeAt(const QModelIndex &index, F&& writer)
{
@@ -1805,7 +1792,7 @@ protected:
return Qt::ItemIsEnabled | Qt::ItemIsSelectable;
}
- static constexpr bool canInsertRows()
+ static constexpr bool canInsertRowsImpl()
{
// We must not insert rows if we cannot adjust the parents of the
// children of the following rows. We don't have to do that if the
@@ -1814,7 +1801,7 @@ protected:
&& Base::dynamicRows() && range_features::has_insert;
}
- static constexpr bool canRemoveRows()
+ static constexpr bool canRemoveRowsImpl()
{
// We must not remove rows if we cannot adjust the parents of the
// children of the following rows. We don't have to do that if the
@@ -2090,12 +2077,12 @@ protected:
return Qt::ItemIsEnabled | Qt::ItemIsSelectable | Qt::ItemNeverHasChildren;
}
- static constexpr bool canInsertRows()
+ static constexpr bool canInsertRowsImpl()
{
return Base::dynamicRows() && range_features::has_insert;
}
- static constexpr bool canRemoveRows()
+ static constexpr bool canRemoveRowsImpl()
{
return Base::dynamicRows() && range_features::has_erase;
}
diff --git a/src/corelib/kernel/qabstracteventdispatcher.cpp b/src/corelib/kernel/qabstracteventdispatcher.cpp
index 997d6f0c98e..fde406a78d8 100644
--- a/src/corelib/kernel/qabstracteventdispatcher.cpp
+++ b/src/corelib/kernel/qabstracteventdispatcher.cpp
@@ -176,8 +176,9 @@ QAbstractEventDispatcher::QAbstractEventDispatcher(QAbstractEventDispatcherPriva
*/
QAbstractEventDispatcher::~QAbstractEventDispatcher()
{
- QThreadData *data = QThreadData::current();
- if (data->eventDispatcher.loadRelaxed() == this)
+ // don't recreate the QThreadData if it has already been destroyed
+ QThreadData *data = QThreadData::currentThreadData();
+ if (data && data->eventDispatcher.loadRelaxed() == this)
data->eventDispatcher.storeRelaxed(nullptr);
}
@@ -192,6 +193,7 @@ QAbstractEventDispatcher::~QAbstractEventDispatcher()
*/
QAbstractEventDispatcher *QAbstractEventDispatcher::instance(QThread *thread)
{
+ // do create a QThreadData, in case this is very early in an adopted thread
QThreadData *data = thread ? QThreadData::get2(thread) : QThreadData::current();
return data->eventDispatcher.loadRelaxed();
}
diff --git a/src/corelib/kernel/qbasictimer.cpp b/src/corelib/kernel/qbasictimer.cpp
index 60387381da6..1906457b13d 100644
--- a/src/corelib/kernel/qbasictimer.cpp
+++ b/src/corelib/kernel/qbasictimer.cpp
@@ -5,6 +5,8 @@
#include "qabstracteventdispatcher.h"
#include "qabstracteventdispatcher_p.h"
+#include <private/qthread_p.h>
+
using namespace std::chrono_literals;
QT_BEGIN_NAMESPACE
@@ -209,7 +211,12 @@ void QBasicTimer::start(Duration duration, Qt::TimerType timerType, QObject *obj
void QBasicTimer::stop()
{
if (isActive()) {
- QAbstractEventDispatcher *eventDispatcher = QAbstractEventDispatcher::instance();
+ QAbstractEventDispatcher *eventDispatcher = nullptr;
+
+ // don't create the current thread data if it's already been destroyed
+ if (QThreadData *data = QThreadData::currentThreadData())
+ eventDispatcher = data->eventDispatcher.loadRelaxed();
+
if (eventDispatcher && !eventDispatcher->unregisterTimer(m_id)) {
qWarning("QBasicTimer::stop: Failed. Possibly trying to stop from a different thread");
return;
diff --git a/src/corelib/platform/windows/qcomobject_p.h b/src/corelib/platform/windows/qcomobject_p.h
index 33c4584e278..bd6f81d1c28 100644
--- a/src/corelib/platform/windows/qcomobject_p.h
+++ b/src/corelib/platform/windows/qcomobject_p.h
@@ -80,18 +80,21 @@ public:
return tryQueryInterface<TFirstInterface, TAdditionalInterfaces...>(riid, ppvObject);
}
- // clang-format off
STDMETHODIMP_(ULONG) AddRef() override
{
- return ++m_referenceCount;
+ return m_referenceCount.fetch_add(1, std::memory_order_relaxed) + 1;
}
- // clang-format on
STDMETHODIMP_(ULONG) Release() override
{
- const LONG referenceCount = --m_referenceCount;
- if (referenceCount == 0)
+ const LONG referenceCount = m_referenceCount.fetch_sub(1, std::memory_order_release) - 1;
+ if (referenceCount == 0) {
+ // This acquire fence synchronizes with the release operation in other threads.
+ // It ensures that all memory writes made to this object by other threads
+ // are visible to this thread before we proceed to delete it.
+ std::atomic_thread_fence(std::memory_order_acquire);
delete this;
+ }
return referenceCount;
}
diff --git a/src/corelib/serialization/qtextstream.cpp b/src/corelib/serialization/qtextstream.cpp
index e4dc98af98b..c357d40c8f5 100644
--- a/src/corelib/serialization/qtextstream.cpp
+++ b/src/corelib/serialization/qtextstream.cpp
@@ -833,29 +833,29 @@ QTextStreamPrivate::PaddingResult QTextStreamPrivate::padding(qsizetype len) con
/*!
\internal
*/
-void QTextStreamPrivate::putString(const QChar *data, qsizetype len, bool number)
+template <typename StringView>
+void QTextStreamPrivate::putStringImpl(StringView data, bool number)
{
- if (Q_UNLIKELY(params.fieldWidth > len)) {
+ if (Q_UNLIKELY(params.fieldWidth > data.size())) {
// handle padding:
- const PaddingResult pad = padding(len);
+ const PaddingResult pad = padding(data.size());
if (params.fieldAlignment == QTextStream::AlignAccountingStyle && number) {
- const QChar sign = len > 0 ? data[0] : QChar();
+ const QChar sign = data.size() > 0 ? data.front() : QChar();
if (sign == locale.negativeSign() || sign == locale.positiveSign()) {
// write the sign before the padding, then skip it later
write(sign);
- ++data;
- --len;
+ data = data.sliced(1);
}
}
writePadding(pad.left);
- write(data, len);
+ write(data);
writePadding(pad.right);
} else {
- write(data, len);
+ write(data);
}
}
@@ -864,29 +864,20 @@ void QTextStreamPrivate::putString(const QChar *data, qsizetype len, bool number
*/
void QTextStreamPrivate::putString(QLatin1StringView data, bool number)
{
- if (Q_UNLIKELY(params.fieldWidth > data.size())) {
-
- // handle padding
-
- const PaddingResult pad = padding(data.size());
-
- if (params.fieldAlignment == QTextStream::AlignAccountingStyle && number) {
- const QChar sign = data.size() > 0 ? QLatin1Char(*data.data()) : QChar();
- if (sign == locale.negativeSign() || sign == locale.positiveSign()) {
- // write the sign before the padding, then skip it later
- write(sign);
- data = QLatin1StringView(data.data() + 1, data.size() - 1);
- }
- }
+ putStringImpl(data, number);
+}
- writePadding(pad.left);
- write(data);
- writePadding(pad.right);
- } else {
- write(data);
- }
+/*!
+ \internal
+*/
+void QTextStreamPrivate::putString(QStringView data, bool number)
+{
+ putStringImpl(data, number);
}
+/*!
+ \internal
+*/
void QTextStreamPrivate::putString(QUtf8StringView data, bool number)
{
putString(data.toString(), number);
diff --git a/src/corelib/serialization/qtextstream_p.h b/src/corelib/serialization/qtextstream_p.h
index bf3ce7b2ef7..2da7fe8e9d5 100644
--- a/src/corelib/serialization/qtextstream_p.h
+++ b/src/corelib/serialization/qtextstream_p.h
@@ -149,11 +149,9 @@ public:
void write(const QChar *data, qsizetype len);
void write(QLatin1StringView data);
void writePadding(qsizetype len);
- inline void putString(QStringView string, bool number = false)
- {
- putString(string.constData(), string.size(), number);
- }
- void putString(const QChar *data, qsizetype len, bool number = false);
+ void putString(QStringView string, bool number = false);
+ void putString(const QChar *data, qsizetype len, bool number = false)
+ { putString(QStringView{data, len}, number); }
void putString(QLatin1StringView data, bool number = false);
void putString(QUtf8StringView data, bool number = false);
inline void putChar(QChar ch);
@@ -168,6 +166,10 @@ public:
bool fillReadBuffer(qint64 maxBytes = -1);
void resetReadBuffer();
void flushWriteBuffer();
+
+private:
+ template <typename StringView>
+ void putStringImpl(StringView view, bool number);
};
QT_END_NAMESPACE
diff --git a/src/corelib/thread/qthread_p.h b/src/corelib/thread/qthread_p.h
index b34336d7e16..c23a50d158e 100644
--- a/src/corelib/thread/qthread_p.h
+++ b/src/corelib/thread/qthread_p.h
@@ -374,6 +374,8 @@ public:
bool requiresCoreApplication = true;
private:
+ friend class QAbstractEventDispatcher;
+ friend class QBasicTimer;
static Q_AUTOTEST_EXPORT QThreadData *currentThreadData() noexcept Q_DECL_PURE_FUNCTION;
static Q_AUTOTEST_EXPORT QThreadData *createCurrentThreadData();
};
diff --git a/src/dbus/dbus_minimal_p.h b/src/dbus/dbus_minimal_p.h
index 36e7fab55d9..a87c07defcb 100644
--- a/src/dbus/dbus_minimal_p.h
+++ b/src/dbus/dbus_minimal_p.h
@@ -2,6 +2,7 @@
// Copyright (C) 2002, 2003, 2004, 2005 Red Hat, Inc.
// Copyright (C) 2016 Intel Corporation.
// SPDX-License-Identifier: AFL-2.1 OR GPL-2.0-or-later
+// Qt-Security score:significant reason:default
#ifndef DBUS_MINIMAL_P_H
#define DBUS_MINIMAL_P_H
diff --git a/src/dbus/qdbus_symbols_p.h b/src/dbus/qdbus_symbols_p.h
index 78b7e049ef3..a3e3b202c12 100644
--- a/src/dbus/qdbus_symbols_p.h
+++ b/src/dbus/qdbus_symbols_p.h
@@ -1,6 +1,7 @@
// Copyright (C) 2016 The Qt Company Ltd.
// Copyright (C) 2016 Intel Corporation.
// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR LGPL-3.0-only OR GPL-2.0-only OR GPL-3.0-only
+// Qt-Security score:significant reason:default
//
// W A R N I N G
diff --git a/src/dbus/qdbusabstractadaptor.cpp b/src/dbus/qdbusabstractadaptor.cpp
index df83cc6f462..b9c87797846 100644
--- a/src/dbus/qdbusabstractadaptor.cpp
+++ b/src/dbus/qdbusabstractadaptor.cpp
@@ -1,6 +1,7 @@
// Copyright (C) 2016 The Qt Company Ltd.
// Copyright (C) 2016 Intel Corporation.
// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR LGPL-3.0-only OR GPL-2.0-only OR GPL-3.0-only
+// Qt-Security score:significant reason:default
#include "qdbusabstractadaptor.h"
#include "qdbusabstractadaptor_p.h"
diff --git a/src/dbus/qdbusabstractadaptor.h b/src/dbus/qdbusabstractadaptor.h
index f143056b43a..649dc4622d0 100644
--- a/src/dbus/qdbusabstractadaptor.h
+++ b/src/dbus/qdbusabstractadaptor.h
@@ -1,5 +1,6 @@
// 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
+// Qt-Security score:significant reason:default
#ifndef QDBUSABSTRACTADAPTOR_H
#define QDBUSABSTRACTADAPTOR_H
diff --git a/src/dbus/qdbusabstractadaptor_p.h b/src/dbus/qdbusabstractadaptor_p.h
index c047af18211..51dc727b2fb 100644
--- a/src/dbus/qdbusabstractadaptor_p.h
+++ b/src/dbus/qdbusabstractadaptor_p.h
@@ -1,5 +1,6 @@
// 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
+// Qt-Security score:significant reason:default
//
// W A R N I N G
diff --git a/src/dbus/qdbusabstractinterface.cpp b/src/dbus/qdbusabstractinterface.cpp
index 0c6dbf1b3b7..4e46fc8f743 100644
--- a/src/dbus/qdbusabstractinterface.cpp
+++ b/src/dbus/qdbusabstractinterface.cpp
@@ -1,6 +1,7 @@
// Copyright (C) 2016 The Qt Company Ltd.
// Copyright (C) 2016 Intel Corporation.
// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR LGPL-3.0-only OR GPL-2.0-only OR GPL-3.0-only
+// Qt-Security score:significant reason:default
#include "qdbusabstractinterface.h"
#include "qdbusabstractinterface_p.h"
diff --git a/src/dbus/qdbusabstractinterface.h b/src/dbus/qdbusabstractinterface.h
index 8d36fb37281..2bdbdc419db 100644
--- a/src/dbus/qdbusabstractinterface.h
+++ b/src/dbus/qdbusabstractinterface.h
@@ -1,5 +1,6 @@
// 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
+// Qt-Security score:significant reason:default
#ifndef QDBUSABSTRACTINTERFACE_H
#define QDBUSABSTRACTINTERFACE_H
diff --git a/src/dbus/qdbusabstractinterface_p.h b/src/dbus/qdbusabstractinterface_p.h
index 1bd5e96f2d4..aca8ed23253 100644
--- a/src/dbus/qdbusabstractinterface_p.h
+++ b/src/dbus/qdbusabstractinterface_p.h
@@ -1,6 +1,7 @@
// Copyright (C) 2016 The Qt Company Ltd.
// Copyright (C) 2016 Intel Corporation.
// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR LGPL-3.0-only OR GPL-2.0-only OR GPL-3.0-only
+// Qt-Security score:significant reason:default
//
// W A R N I N G
diff --git a/src/dbus/qdbusargument.h b/src/dbus/qdbusargument.h
index f04cbc1ed43..b79f675a9e4 100644
--- a/src/dbus/qdbusargument.h
+++ b/src/dbus/qdbusargument.h
@@ -1,5 +1,6 @@
// 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
+// Qt-Security score:significant reason:default
#ifndef QDBUSARGUMENT_H
#define QDBUSARGUMENT_H
@@ -119,6 +120,27 @@ protected:
QDBusArgument(QDBusArgumentPrivate *d);
friend class QDBusArgumentPrivate;
mutable QDBusArgumentPrivate *d;
+
+private:
+ template <typename... T>
+ friend QDBusArgument &operator<<(QDBusArgument &argument, const std::tuple<T...> &tuple)
+ {
+ static_assert(sizeof...(T) != 0, "D-Bus doesn't allow empty structs");
+ argument.beginStructure();
+ std::apply([&argument](const auto &...elements) { (argument << ... << elements); }, tuple);
+ argument.endStructure();
+ return argument;
+ }
+
+ template <typename... T>
+ friend const QDBusArgument &operator>>(const QDBusArgument &argument, std::tuple<T...> &tuple)
+ {
+ static_assert(sizeof...(T) != 0, "D-Bus doesn't allow empty structs");
+ argument.beginStructure();
+ std::apply([&argument](auto &...elements) { (argument >> ... >> elements); }, tuple);
+ argument.endStructure();
+ return argument;
+ }
};
Q_DECLARE_SHARED(QDBusArgument)
@@ -323,26 +345,6 @@ inline const QDBusArgument &operator>>(const QDBusArgument &arg, std::pair<T1, T
return arg;
}
-template <typename... T>
-QDBusArgument &operator<<(QDBusArgument &argument, const std::tuple<T...> &tuple)
-{
- static_assert(sizeof...(T) != 0, "D-Bus doesn't allow empty structs");
- argument.beginStructure();
- std::apply([&argument](const auto &...elements) { (argument << ... << elements); }, tuple);
- argument.endStructure();
- return argument;
-}
-
-template <typename... T>
-const QDBusArgument &operator>>(const QDBusArgument &argument, std::tuple<T...> &tuple)
-{
- static_assert(sizeof...(T) != 0, "D-Bus doesn't allow empty structs");
- argument.beginStructure();
- std::apply([&argument](auto &...elements) { (argument >> ... >> elements); }, tuple);
- argument.endStructure();
- return argument;
-}
-
QT_END_NAMESPACE
#endif // QT_NO_DBUS
diff --git a/src/dbus/qdbusargument_p.h b/src/dbus/qdbusargument_p.h
index d9a73827424..3d007ea2c7c 100644
--- a/src/dbus/qdbusargument_p.h
+++ b/src/dbus/qdbusargument_p.h
@@ -1,5 +1,6 @@
// 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
+// Qt-Security score:significant reason:default
#ifndef QDBUSARGUMENT_P_H
#define QDBUSARGUMENT_P_H
diff --git a/src/dbus/qdbusconnection.cpp b/src/dbus/qdbusconnection.cpp
index 08cec30afa3..aa2ba432fa5 100644
--- a/src/dbus/qdbusconnection.cpp
+++ b/src/dbus/qdbusconnection.cpp
@@ -1,6 +1,7 @@
// Copyright (C) 2016 The Qt Company Ltd.
// Copyright (C) 2016 Intel Corporation.
// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR LGPL-3.0-only OR GPL-2.0-only OR GPL-3.0-only
+// Qt-Security score:significant reason:default
#include "qdbusconnection.h"
#include "qdbusconnection_p.h"
diff --git a/src/dbus/qdbusconnection.h b/src/dbus/qdbusconnection.h
index be8acdc4ea5..45d3850faed 100644
--- a/src/dbus/qdbusconnection.h
+++ b/src/dbus/qdbusconnection.h
@@ -1,6 +1,7 @@
// Copyright (C) 2016 The Qt Company Ltd.
// Copyright (C) 2016 Intel Corporation.
// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR LGPL-3.0-only OR GPL-2.0-only OR GPL-3.0-only
+// Qt-Security score:significant reason:default
#ifndef QDBUSCONNECTION_H
#define QDBUSCONNECTION_H
diff --git a/src/dbus/qdbusconnection_p.h b/src/dbus/qdbusconnection_p.h
index 5cb175c1cdc..b553cd62a03 100644
--- a/src/dbus/qdbusconnection_p.h
+++ b/src/dbus/qdbusconnection_p.h
@@ -1,6 +1,7 @@
// Copyright (C) 2016 The Qt Company Ltd.
// Copyright (C) 2016 Intel Corporation.
// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR LGPL-3.0-only OR GPL-2.0-only OR GPL-3.0-only
+// Qt-Security score:significant reason:default
//
// W A R N I N G
diff --git a/src/dbus/qdbusconnectioninterface.cpp b/src/dbus/qdbusconnectioninterface.cpp
index f7dbbb01563..36636b355b0 100644
--- a/src/dbus/qdbusconnectioninterface.cpp
+++ b/src/dbus/qdbusconnectioninterface.cpp
@@ -1,5 +1,6 @@
// 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
+// Qt-Security score:significant reason:default
#include "qdbusconnectioninterface.h"
diff --git a/src/dbus/qdbusconnectioninterface.h b/src/dbus/qdbusconnectioninterface.h
index e728539a385..60742ea4043 100644
--- a/src/dbus/qdbusconnectioninterface.h
+++ b/src/dbus/qdbusconnectioninterface.h
@@ -1,5 +1,6 @@
// 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
+// Qt-Security score:significant reason:default
#ifndef QDBUSCONNECTIONINTERFACE_H
#define QDBUSCONNECTIONINTERFACE_H
diff --git a/src/dbus/qdbusconnectionmanager.cpp b/src/dbus/qdbusconnectionmanager.cpp
index d41178b5c30..ef96231e1cc 100644
--- a/src/dbus/qdbusconnectionmanager.cpp
+++ b/src/dbus/qdbusconnectionmanager.cpp
@@ -1,6 +1,7 @@
// Copyright (C) 2016 The Qt Company Ltd.
// Copyright (C) 2016 Intel Corporation.
// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR LGPL-3.0-only OR GPL-2.0-only OR GPL-3.0-only
+// Qt-Security score:significant reason:default
#include "qdbusconnectionmanager_p.h"
diff --git a/src/dbus/qdbusconnectionmanager_p.h b/src/dbus/qdbusconnectionmanager_p.h
index 644c3c8fb1c..1ef048e5d72 100644
--- a/src/dbus/qdbusconnectionmanager_p.h
+++ b/src/dbus/qdbusconnectionmanager_p.h
@@ -1,6 +1,7 @@
// Copyright (C) 2016 The Qt Company Ltd.
// Copyright (C) 2016 Intel Corporation.
// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR LGPL-3.0-only OR GPL-2.0-only OR GPL-3.0-only
+// Qt-Security score:significant reason:default
//
// W A R N I N G
diff --git a/src/dbus/qdbuscontext.cpp b/src/dbus/qdbuscontext.cpp
index b8cb1dc8b10..f788b398186 100644
--- a/src/dbus/qdbuscontext.cpp
+++ b/src/dbus/qdbuscontext.cpp
@@ -1,5 +1,6 @@
// 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
+// Qt-Security score:significant reason:default
#include "qdbusmessage.h"
#include "qdbusconnection.h"
diff --git a/src/dbus/qdbuscontext.h b/src/dbus/qdbuscontext.h
index 02620449e18..47ced171171 100644
--- a/src/dbus/qdbuscontext.h
+++ b/src/dbus/qdbuscontext.h
@@ -1,5 +1,6 @@
// 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
+// Qt-Security score:significant reason:default
#ifndef QDBUSCONTEXT_H
#define QDBUSCONTEXT_H
diff --git a/src/dbus/qdbuscontext_p.h b/src/dbus/qdbuscontext_p.h
index 1a544b8e44e..edd7aa068f0 100644
--- a/src/dbus/qdbuscontext_p.h
+++ b/src/dbus/qdbuscontext_p.h
@@ -1,5 +1,6 @@
// 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
+// Qt-Security score:significant reason:default
//
// W A R N I N G
diff --git a/src/dbus/qdbuserror.cpp b/src/dbus/qdbuserror.cpp
index 0a5ba30d3a3..c168d8d471c 100644
--- a/src/dbus/qdbuserror.cpp
+++ b/src/dbus/qdbuserror.cpp
@@ -1,5 +1,6 @@
// 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
+// Qt-Security score:significant reason:default
#include "qdbuserror.h"
diff --git a/src/dbus/qdbuserror.h b/src/dbus/qdbuserror.h
index 292f967acbe..dcb0c6666a0 100644
--- a/src/dbus/qdbuserror.h
+++ b/src/dbus/qdbuserror.h
@@ -1,5 +1,6 @@
// 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
+// Qt-Security score:significant reason:default
#ifndef QDBUSERROR_H
#define QDBUSERROR_H
diff --git a/src/dbus/qdbusextratypes.cpp b/src/dbus/qdbusextratypes.cpp
index 042014ceab0..c27c1cbd5db 100644
--- a/src/dbus/qdbusextratypes.cpp
+++ b/src/dbus/qdbusextratypes.cpp
@@ -1,5 +1,6 @@
// 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
+// Qt-Security score:significant reason:default
#include "qdbusextratypes.h"
#include "qdbusutil_p.h"
diff --git a/src/dbus/qdbusextratypes.h b/src/dbus/qdbusextratypes.h
index 3b44bfc2175..3bfd38486a2 100644
--- a/src/dbus/qdbusextratypes.h
+++ b/src/dbus/qdbusextratypes.h
@@ -1,5 +1,6 @@
// 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
+// Qt-Security score:significant reason:default
#ifndef QDBUSEXTRATYPES_H
#define QDBUSEXTRATYPES_H
diff --git a/src/dbus/qdbusintegrator.cpp b/src/dbus/qdbusintegrator.cpp
index 4c9736dea6d..bfd3a092d39 100644
--- a/src/dbus/qdbusintegrator.cpp
+++ b/src/dbus/qdbusintegrator.cpp
@@ -1,6 +1,7 @@
// Copyright (C) 2016 The Qt Company Ltd.
// Copyright (C) 2016 Intel Corporation.
// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR LGPL-3.0-only OR GPL-2.0-only OR GPL-3.0-only
+// Qt-Security score:significant reason:default
#include "qdbusintegrator_p.h"
diff --git a/src/dbus/qdbusintegrator_p.h b/src/dbus/qdbusintegrator_p.h
index ac9adf02100..f6486d87256 100644
--- a/src/dbus/qdbusintegrator_p.h
+++ b/src/dbus/qdbusintegrator_p.h
@@ -1,6 +1,7 @@
// Copyright (C) 2016 The Qt Company Ltd.
// Copyright (C) 2016 Intel Corporation.
// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR LGPL-3.0-only OR GPL-2.0-only OR GPL-3.0-only
+// Qt-Security score:significant reason:default
//
// W A R N I N G
diff --git a/src/dbus/qdbusinterface.cpp b/src/dbus/qdbusinterface.cpp
index d4da36b35c4..7481229dccf 100644
--- a/src/dbus/qdbusinterface.cpp
+++ b/src/dbus/qdbusinterface.cpp
@@ -1,5 +1,6 @@
// 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
+// Qt-Security score:significant reason:default
#include "qdbusinterface.h"
#include "qdbusinterface_p.h"
diff --git a/src/dbus/qdbusinterface.h b/src/dbus/qdbusinterface.h
index fc02463ac72..bff0336fd82 100644
--- a/src/dbus/qdbusinterface.h
+++ b/src/dbus/qdbusinterface.h
@@ -1,5 +1,6 @@
// 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
+// Qt-Security score:significant reason:default
#ifndef QDBUSINTERFACE_H
#define QDBUSINTERFACE_H
diff --git a/src/dbus/qdbusinterface_p.h b/src/dbus/qdbusinterface_p.h
index caf24e589d4..63b7196527f 100644
--- a/src/dbus/qdbusinterface_p.h
+++ b/src/dbus/qdbusinterface_p.h
@@ -1,5 +1,6 @@
// 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
+// Qt-Security score:significant reason:default
//
// W A R N I N G
diff --git a/src/dbus/qdbusinternalfilters.cpp b/src/dbus/qdbusinternalfilters.cpp
index 721564ed3cd..a46bc8e6c87 100644
--- a/src/dbus/qdbusinternalfilters.cpp
+++ b/src/dbus/qdbusinternalfilters.cpp
@@ -1,6 +1,7 @@
// Copyright (C) 2016 The Qt Company Ltd.
// Copyright (C) 2016 Intel Corporation.
// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR LGPL-3.0-only OR GPL-2.0-only OR GPL-3.0-only
+// Qt-Security score:significant reason:default
#include "qdbusconnection_p.h"
diff --git a/src/dbus/qdbusintrospection.cpp b/src/dbus/qdbusintrospection.cpp
index 04b5ab77519..4f6c40743a5 100644
--- a/src/dbus/qdbusintrospection.cpp
+++ b/src/dbus/qdbusintrospection.cpp
@@ -1,5 +1,6 @@
// 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
+// Qt-Security score:significant reason:default
#include "qdbusintrospection_p.h"
#include "qdbusxmlparser_p.h"
diff --git a/src/dbus/qdbusintrospection_p.h b/src/dbus/qdbusintrospection_p.h
index 766cdffb621..3eccef8b7b0 100644
--- a/src/dbus/qdbusintrospection_p.h
+++ b/src/dbus/qdbusintrospection_p.h
@@ -1,5 +1,6 @@
// 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
+// Qt-Security score:significant reason:default
#ifndef QDBUSINTROSPECTION_P_H
#define QDBUSINTROSPECTION_P_H
diff --git a/src/dbus/qdbusmessage.cpp b/src/dbus/qdbusmessage.cpp
index b10266ac8b7..f0a056e3127 100644
--- a/src/dbus/qdbusmessage.cpp
+++ b/src/dbus/qdbusmessage.cpp
@@ -1,6 +1,7 @@
// Copyright (C) 2016 The Qt Company Ltd.
// Copyright (C) 2016 Intel Corporation.
// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR LGPL-3.0-only OR GPL-2.0-only OR GPL-3.0-only
+// Qt-Security score:significant reason:default
#include "qdbusmessage.h"
#include "qdbusmessage_p.h"
diff --git a/src/dbus/qdbusmessage.h b/src/dbus/qdbusmessage.h
index 608b9779d22..5b631c2b847 100644
--- a/src/dbus/qdbusmessage.h
+++ b/src/dbus/qdbusmessage.h
@@ -1,5 +1,6 @@
// 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
+// Qt-Security score:significant reason:default
#ifndef QDBUSMESSAGE_H
#define QDBUSMESSAGE_H
diff --git a/src/dbus/qdbusmessage_p.h b/src/dbus/qdbusmessage_p.h
index 88ba78025e7..07cdeda4034 100644
--- a/src/dbus/qdbusmessage_p.h
+++ b/src/dbus/qdbusmessage_p.h
@@ -1,5 +1,6 @@
// 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
+// Qt-Security score:significant reason:default
#ifndef QDBUSMESSAGE_P_H
#define QDBUSMESSAGE_P_H
diff --git a/src/dbus/qdbusmetaobject.cpp b/src/dbus/qdbusmetaobject.cpp
index c3590dc51cd..149392f9c3c 100644
--- a/src/dbus/qdbusmetaobject.cpp
+++ b/src/dbus/qdbusmetaobject.cpp
@@ -1,6 +1,7 @@
// Copyright (C) 2016 The Qt Company Ltd.
// Copyright (C) 2016 Intel Corporation.
// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR LGPL-3.0-only OR GPL-2.0-only OR GPL-3.0-only
+// Qt-Security score:significant reason:default
#include "qdbusmetaobject_p.h"
diff --git a/src/dbus/qdbusmetaobject_p.h b/src/dbus/qdbusmetaobject_p.h
index 97d16d99d0e..473765a175c 100644
--- a/src/dbus/qdbusmetaobject_p.h
+++ b/src/dbus/qdbusmetaobject_p.h
@@ -1,5 +1,6 @@
// 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
+// Qt-Security score:significant reason:default
#ifndef QDBUSMETAOBJECT_P_H
#define QDBUSMETAOBJECT_P_H
diff --git a/src/dbus/qdbusmetatype.cpp b/src/dbus/qdbusmetatype.cpp
index 3ae7589480e..5a862528026 100644
--- a/src/dbus/qdbusmetatype.cpp
+++ b/src/dbus/qdbusmetatype.cpp
@@ -1,5 +1,6 @@
// 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
+// Qt-Security score:critical reason:data-parser
#include "qdbusmetatype.h"
#include "qdbusmetatype_p.h"
diff --git a/src/dbus/qdbusmetatype.h b/src/dbus/qdbusmetatype.h
index 3304a69e1ef..66b6b1b0456 100644
--- a/src/dbus/qdbusmetatype.h
+++ b/src/dbus/qdbusmetatype.h
@@ -1,5 +1,6 @@
// 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
+// Qt-Security score:significant reason:default
#ifndef QDBUSMETATYPE_H
#define QDBUSMETATYPE_H
diff --git a/src/dbus/qdbusmetatype_p.h b/src/dbus/qdbusmetatype_p.h
index 86a59f587dd..b72c0c431e5 100644
--- a/src/dbus/qdbusmetatype_p.h
+++ b/src/dbus/qdbusmetatype_p.h
@@ -1,5 +1,6 @@
// 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
+// Qt-Security score:significant reason:default
#ifndef QDBUSMETATYPE_P_H
#define QDBUSMETATYPE_P_H
diff --git a/src/dbus/qdbusmisc.cpp b/src/dbus/qdbusmisc.cpp
index 635258c86d2..bf18b00d1cc 100644
--- a/src/dbus/qdbusmisc.cpp
+++ b/src/dbus/qdbusmisc.cpp
@@ -1,5 +1,6 @@
// 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
+// Qt-Security score:significant reason:default
#include <string.h>
diff --git a/src/dbus/qdbuspendingcall.cpp b/src/dbus/qdbuspendingcall.cpp
index cd222c9ffa5..c0b382ca114 100644
--- a/src/dbus/qdbuspendingcall.cpp
+++ b/src/dbus/qdbuspendingcall.cpp
@@ -1,6 +1,7 @@
// Copyright (C) 2016 The Qt Company Ltd.
// Copyright (C) 2016 Intel Corporation.
// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR LGPL-3.0-only OR GPL-2.0-only OR GPL-3.0-only
+// Qt-Security score:significant reason:default
#include "qdbuspendingcall.h"
#include "qdbuspendingcall_p.h"
@@ -228,7 +229,7 @@ void QDBusPendingCallPrivate::waitForFinishedWithGui()
/*!
\fn QDBusPendingCall::QDBusPendingCall(QDBusPendingCall &&other)
- \since 6.11
+ \since 6.10
Moves \a other into this object.
diff --git a/src/dbus/qdbuspendingcall.h b/src/dbus/qdbuspendingcall.h
index 0db6667e5ec..1907fab78b9 100644
--- a/src/dbus/qdbuspendingcall.h
+++ b/src/dbus/qdbuspendingcall.h
@@ -1,5 +1,6 @@
// 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
+// Qt-Security score:significant reason:default
#ifndef QDBUSPENDINGCALL_H
#define QDBUSPENDINGCALL_H
diff --git a/src/dbus/qdbuspendingcall_p.h b/src/dbus/qdbuspendingcall_p.h
index 2795cc3ecf8..407537c9bb0 100644
--- a/src/dbus/qdbuspendingcall_p.h
+++ b/src/dbus/qdbuspendingcall_p.h
@@ -1,6 +1,7 @@
// Copyright (C) 2016 The Qt Company Ltd.
// Copyright (C) 2016 Intel Corporation.
// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR LGPL-3.0-only OR GPL-2.0-only OR GPL-3.0-only
+// Qt-Security score:significant reason:default
//
// W A R N I N G
diff --git a/src/dbus/qdbuspendingreply.cpp b/src/dbus/qdbuspendingreply.cpp
index 7bb23aa3e36..b0fb2d75af5 100644
--- a/src/dbus/qdbuspendingreply.cpp
+++ b/src/dbus/qdbuspendingreply.cpp
@@ -1,6 +1,7 @@
// Copyright (C) 2016 The Qt Company Ltd.
// Copyright (C) 2016 Intel Corporation.
// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR LGPL-3.0-only OR GPL-2.0-only OR GPL-3.0-only
+// Qt-Security score:significant reason:default
#include "qdbuspendingreply.h"
#include "qdbuspendingcall_p.h"
diff --git a/src/dbus/qdbuspendingreply.h b/src/dbus/qdbuspendingreply.h
index 72e1bbdc313..7b76bcd1b7c 100644
--- a/src/dbus/qdbuspendingreply.h
+++ b/src/dbus/qdbuspendingreply.h
@@ -1,5 +1,6 @@
// 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
+// Qt-Security score:significant reason:default
#ifndef QDBUSPENDINGREPLY_H
#define QDBUSPENDINGREPLY_H
diff --git a/src/dbus/qdbusreply.cpp b/src/dbus/qdbusreply.cpp
index 5b26250b10d..a90e7354ed2 100644
--- a/src/dbus/qdbusreply.cpp
+++ b/src/dbus/qdbusreply.cpp
@@ -1,5 +1,6 @@
// 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
+// Qt-Security score:significant reason:default
#include "qdbusreply.h"
#include "qdbusmetatype.h"
diff --git a/src/dbus/qdbusreply.h b/src/dbus/qdbusreply.h
index df488208c85..5dfb63f3b4b 100644
--- a/src/dbus/qdbusreply.h
+++ b/src/dbus/qdbusreply.h
@@ -1,5 +1,6 @@
// 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
+// Qt-Security score:significant reason:default
#ifndef QDBUSREPLY_H
#define QDBUSREPLY_H
diff --git a/src/dbus/qdbusserver.cpp b/src/dbus/qdbusserver.cpp
index e9131a14c4b..9e1063424a5 100644
--- a/src/dbus/qdbusserver.cpp
+++ b/src/dbus/qdbusserver.cpp
@@ -1,6 +1,7 @@
// Copyright (C) 2016 The Qt Company Ltd.
// Copyright (C) 2016 Intel Corporation.
// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR LGPL-3.0-only OR GPL-2.0-only OR GPL-3.0-only
+// Qt-Security score:significant reason:default
#include "qdbusserver.h"
#include "qdbusconnection_p.h"
diff --git a/src/dbus/qdbusserver.h b/src/dbus/qdbusserver.h
index 34985cc0554..04b7930d412 100644
--- a/src/dbus/qdbusserver.h
+++ b/src/dbus/qdbusserver.h
@@ -1,6 +1,7 @@
// Copyright (C) 2016 The Qt Company Ltd.
// Copyright (C) 2016 Intel Corporation.
// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR LGPL-3.0-only OR GPL-2.0-only OR GPL-3.0-only
+// Qt-Security score:significant reason:default
#ifndef QDBUSSERVER_H
#define QDBUSSERVER_H
diff --git a/src/dbus/qdbusservicewatcher.cpp b/src/dbus/qdbusservicewatcher.cpp
index bf94f54f564..6b9675c19ec 100644
--- a/src/dbus/qdbusservicewatcher.cpp
+++ b/src/dbus/qdbusservicewatcher.cpp
@@ -1,5 +1,6 @@
// 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
+// Qt-Security score:significant reason:default
#include "qdbusservicewatcher.h"
#include "qdbusconnection.h"
diff --git a/src/dbus/qdbusservicewatcher.h b/src/dbus/qdbusservicewatcher.h
index 71c63084b18..d07ae0fe5d2 100644
--- a/src/dbus/qdbusservicewatcher.h
+++ b/src/dbus/qdbusservicewatcher.h
@@ -1,5 +1,6 @@
// 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
+// Qt-Security score:significant reason:default
#ifndef QDBUSSERVICEWATCHER_H
#define QDBUSSERVICEWATCHER_H
diff --git a/src/dbus/qdbusthreaddebug_p.h b/src/dbus/qdbusthreaddebug_p.h
index bcbd1efb494..70fbf762d19 100644
--- a/src/dbus/qdbusthreaddebug_p.h
+++ b/src/dbus/qdbusthreaddebug_p.h
@@ -1,6 +1,7 @@
// Copyright (C) 2016 The Qt Company Ltd.
// Copyright (C) 2016 Intel Corporation.
// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR LGPL-3.0-only OR GPL-2.0-only OR GPL-3.0-only
+// Qt-Security score:significant reason:default
#ifndef QDBUSTHREADDEBUG_P_H
#define QDBUSTHREADDEBUG_P_H
diff --git a/src/dbus/qdbusunixfiledescriptor.cpp b/src/dbus/qdbusunixfiledescriptor.cpp
index 365f6cbc76c..bc38d142b95 100644
--- a/src/dbus/qdbusunixfiledescriptor.cpp
+++ b/src/dbus/qdbusunixfiledescriptor.cpp
@@ -1,5 +1,6 @@
// 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
+// Qt-Security score:significant reason:default
#include "qdbusunixfiledescriptor.h"
diff --git a/src/dbus/qdbusunixfiledescriptor.h b/src/dbus/qdbusunixfiledescriptor.h
index cbcd8b53a95..c946ec1761b 100644
--- a/src/dbus/qdbusunixfiledescriptor.h
+++ b/src/dbus/qdbusunixfiledescriptor.h
@@ -1,5 +1,6 @@
// 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
+// Qt-Security score:significant reason:default
#ifndef QDBUSUNIXFILEDESCRIPTOR_H
diff --git a/src/dbus/qdbusutil.cpp b/src/dbus/qdbusutil.cpp
index 84ce5ed78dc..9c1c7e8edf3 100644
--- a/src/dbus/qdbusutil.cpp
+++ b/src/dbus/qdbusutil.cpp
@@ -1,5 +1,6 @@
// 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
+// Qt-Security score:critical reason:data-parser
#include "qdbusutil_p.h"
diff --git a/src/dbus/qdbusutil_p.h b/src/dbus/qdbusutil_p.h
index 3db93849682..580a2b92c96 100644
--- a/src/dbus/qdbusutil_p.h
+++ b/src/dbus/qdbusutil_p.h
@@ -1,6 +1,7 @@
// Copyright (C) 2016 The Qt Company Ltd.
// Copyright (C) 2016 Intel Corporation.
// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR LGPL-3.0-only OR GPL-2.0-only OR GPL-3.0-only
+// Qt-Security score:critical reason:data-parser
//
// W A R N I N G
diff --git a/src/dbus/qdbusvirtualobject.cpp b/src/dbus/qdbusvirtualobject.cpp
index 2e5df125d25..878152ad32d 100644
--- a/src/dbus/qdbusvirtualobject.cpp
+++ b/src/dbus/qdbusvirtualobject.cpp
@@ -1,5 +1,6 @@
// 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
+// Qt-Security score:significant reason:default
#include "qdbusvirtualobject.h"
diff --git a/src/dbus/qdbusvirtualobject.h b/src/dbus/qdbusvirtualobject.h
index 573d731d201..451b0040069 100644
--- a/src/dbus/qdbusvirtualobject.h
+++ b/src/dbus/qdbusvirtualobject.h
@@ -1,5 +1,6 @@
// 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
+// Qt-Security score:significant reason:default
#ifndef QDBUSVIRTUALOBJECT_H
#define QDBUSVIRTUALOBJECT_H
diff --git a/src/dbus/qdbusxmlgenerator.cpp b/src/dbus/qdbusxmlgenerator.cpp
index 412ac180951..69d27a49f9f 100644
--- a/src/dbus/qdbusxmlgenerator.cpp
+++ b/src/dbus/qdbusxmlgenerator.cpp
@@ -1,5 +1,6 @@
// 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
+// Qt-Security score:significant reason:default
#include <QtCore/qmetaobject.h>
#include <QtCore/qstringlist.h>
diff --git a/src/dbus/qdbusxmlparser.cpp b/src/dbus/qdbusxmlparser.cpp
index c2e8df8be7e..b3e1b6b78b6 100644
--- a/src/dbus/qdbusxmlparser.cpp
+++ b/src/dbus/qdbusxmlparser.cpp
@@ -1,5 +1,6 @@
// 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
+// Qt-Security score:critical reason:data-parser
#include "qdbusxmlparser_p.h"
#include "qdbusutil_p.h"
diff --git a/src/dbus/qdbusxmlparser_p.h b/src/dbus/qdbusxmlparser_p.h
index 0476ba36281..6cee2da76cc 100644
--- a/src/dbus/qdbusxmlparser_p.h
+++ b/src/dbus/qdbusxmlparser_p.h
@@ -1,5 +1,6 @@
// 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
+// Qt-Security score:significant reason:default
#ifndef QDBUSXMLPARSER_P_H
#define QDBUSXMLPARSER_P_H
diff --git a/src/dbus/qtdbusglobal.h b/src/dbus/qtdbusglobal.h
index 9724c066f6a..07cfa16a258 100644
--- a/src/dbus/qtdbusglobal.h
+++ b/src/dbus/qtdbusglobal.h
@@ -1,5 +1,6 @@
// 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
+// Qt-Security score:significant reason:default
#ifndef QTDBUSGLOBAL_H
#define QTDBUSGLOBAL_H
diff --git a/src/dbus/qtdbusglobal_p.h b/src/dbus/qtdbusglobal_p.h
index 9d315de75c8..6e748c20ae3 100644
--- a/src/dbus/qtdbusglobal_p.h
+++ b/src/dbus/qtdbusglobal_p.h
@@ -1,5 +1,6 @@
// 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
+// Qt-Security score:significant reason:default
#ifndef QTDBUSGLOBAL_P_H
#define QTDBUSGLOBAL_P_H
diff --git a/src/gui/math3d/qmatrix4x4.cpp b/src/gui/math3d/qmatrix4x4.cpp
index 1703fa5b0fe..1e0f9295110 100644
--- a/src/gui/math3d/qmatrix4x4.cpp
+++ b/src/gui/math3d/qmatrix4x4.cpp
@@ -4,6 +4,8 @@
#include "qmatrix4x4.h"
#include <QtCore/qmath.h>
#include <QtCore/qvariant.h>
+
+#include <QtGui/qquaternion.h>
#include <QtGui/qtransform.h>
#include <cmath>
diff --git a/src/gui/math3d/qmatrix4x4.h b/src/gui/math3d/qmatrix4x4.h
index 80ba62ca36f..2ba274d4517 100644
--- a/src/gui/math3d/qmatrix4x4.h
+++ b/src/gui/math3d/qmatrix4x4.h
@@ -7,7 +7,6 @@
#include <QtGui/qtguiglobal.h>
#include <QtGui/qvector3d.h>
#include <QtGui/qvector4d.h>
-#include <QtGui/qquaternion.h>
#include <QtGui/qgenericmatrix.h>
#include <QtCore/qrect.h>
@@ -15,6 +14,9 @@ class tst_QMatrixNxN;
QT_BEGIN_NAMESPACE
+#ifndef QT_NO_QUATERNION
+class QQuaternion;
+#endif
#ifndef QT_NO_MATRIX4X4
diff --git a/src/network/access/qhttp2connection.cpp b/src/network/access/qhttp2connection.cpp
index 6f8a318dae0..9b339a19a24 100644
--- a/src/network/access/qhttp2connection.cpp
+++ b/src/network/access/qhttp2connection.cpp
@@ -663,8 +663,10 @@ void QHttp2Stream::handleDATA(const Frame &inboundFrame)
{
QHttp2Connection *connection = getConnection();
- qCDebug(qHttp2ConnectionLog, "[%p] stream %u, received DATA frame with payload of %u bytes",
- connection, m_streamID, inboundFrame.payloadSize());
+ qCDebug(qHttp2ConnectionLog,
+ "[%p] stream %u, received DATA frame with payload of %u bytes, closing stream? %s",
+ connection, m_streamID, inboundFrame.payloadSize(),
+ inboundFrame.flags().testFlag(Http2::FrameFlag::END_STREAM) ? "yes" : "no");
// RFC 9113, 6.1: If a DATA frame is received whose stream is not in the "open" or "half-closed
// (local)" state, the recipient MUST respond with a stream error (Section 5.4.2) of type
@@ -1426,7 +1428,8 @@ void QHttp2Connection::handleHEADERS()
Q_ASSERT(inboundFrame.type() == FrameType::HEADERS);
const auto streamID = inboundFrame.streamID();
- qCDebug(qHttp2ConnectionLog, "[%p] Received HEADERS frame on stream %d", this, streamID);
+ qCDebug(qHttp2ConnectionLog, "[%p] Received HEADERS frame on stream %d, end stream? %s", this,
+ streamID, inboundFrame.flags().testFlag(Http2::FrameFlag::END_STREAM) ? "yes" : "no");
// RFC 9113, 6.2: If a HEADERS frame is received whose Stream Identifier field is 0x00, the
// recipient MUST respond with a connection error.
@@ -1849,6 +1852,10 @@ void QHttp2Connection::handleWINDOW_UPDATE()
void QHttp2Connection::handleCONTINUATION()
{
Q_ASSERT(inboundFrame.type() == FrameType::CONTINUATION);
+ auto streamID = inboundFrame.streamID();
+ qCDebug(qHttp2ConnectionLog,
+ "[%p] Received CONTINUATION frame on stream %d, end stream? %s", this, streamID,
+ inboundFrame.flags().testFlag(Http2::FrameFlag::END_STREAM) ? "yes" : "no");
if (continuedFrames.empty())
return connectionError(PROTOCOL_ERROR,
"CONTINUATION without a preceding HEADERS or PUSH_PROMISE");
diff --git a/src/plugins/platforms/wasm/qtloader.js b/src/plugins/platforms/wasm/qtloader.js
index dc7f4583da8..909d8da8856 100644
--- a/src/plugins/platforms/wasm/qtloader.js
+++ b/src/plugins/platforms/wasm/qtloader.js
@@ -190,7 +190,8 @@ async function qtLoad(config)
const originalLocateFile = config.locateFile;
config.locateFile = filename => {
const originalLocatedFilename = originalLocateFile ? originalLocateFile(filename) : filename;
- if (originalLocatedFilename.startsWith('libQt6'))
+ if (originalLocatedFilename.startsWith(
+ 'libQt6')) // wasmqtdeploy rely on this behavior, update both in case of change
return `${config.qt.qtdir}/lib/${originalLocatedFilename}`;
return originalLocatedFilename;
}
diff --git a/src/plugins/platforms/wasm/qwasmclipboard.cpp b/src/plugins/platforms/wasm/qwasmclipboard.cpp
index e5392f33cd7..44db371ca4d 100644
--- a/src/plugins/platforms/wasm/qwasmclipboard.cpp
+++ b/src/plugins/platforms/wasm/qwasmclipboard.cpp
@@ -48,10 +48,6 @@ static void commonCopyEvent(val event)
void QWasmClipboard::cut(val event)
{
- QWasmInputContext *wasmInput = QWasmIntegration::get()->wasmInputContext();
- if (wasmInput && wasmInput->usingTextInput())
- return;
-
if (!QWasmIntegration::get()->getWasmClipboard()->hasClipboardApi()) {
// Send synthetic Ctrl+X to make the app cut data to Qt's clipboard
QWindowSystemInterface::handleKeyEvent(
@@ -63,10 +59,6 @@ void QWasmClipboard::cut(val event)
void QWasmClipboard::copy(val event)
{
- QWasmInputContext *wasmInput = QWasmIntegration::get()->wasmInputContext();
- if (wasmInput && wasmInput->usingTextInput())
- return;
-
if (!QWasmIntegration::get()->getWasmClipboard()->hasClipboardApi()) {
// Send synthetic Ctrl+C to make the app copy data to Qt's clipboard
QWindowSystemInterface::handleKeyEvent(
@@ -77,10 +69,6 @@ void QWasmClipboard::copy(val event)
void QWasmClipboard::paste(val event)
{
- QWasmInputContext *wasmInput = QWasmIntegration::get()->wasmInputContext();
- if (wasmInput && wasmInput->usingTextInput())
- return;
-
event.call<void>("preventDefault"); // prevent browser from handling drop event
QWasmIntegration::get()->getWasmClipboard()->sendClipboardData(event);
@@ -183,86 +171,42 @@ void QWasmClipboard::writeToClipboardApi()
{
Q_ASSERT(m_hasClipboardApi);
- // copy event
- // browser event handler detected ctrl c if clipboard API
- // or Qt call from keyboard event handler
-
- QMimeData *_mimes = mimeData(QClipboard::Clipboard);
- if (!_mimes)
+ QMimeData *mimeData = this->mimeData(QClipboard::Clipboard);
+ if (!mimeData)
return;
- emscripten::val clipboardWriteArray = emscripten::val::array();
- QByteArray ba;
-
- for (auto mimetype : _mimes->formats()) {
- // we need to treat binary and text differently, as the blob method below
- // fails for text mimetypes
- // ignore text types
-
- if (mimetype.contains("STRING", Qt::CaseSensitive) || mimetype.contains("TEXT", Qt::CaseSensitive))
- continue;
-
- if (_mimes->hasHtml()) { // prefer html over text
- ba = _mimes->html().toLocal8Bit();
- // force this mime
- mimetype = "text/html";
- } else if (mimetype.contains("text/plain")) {
- ba = _mimes->text().toLocal8Bit();
- } else if (mimetype.contains("image")) {
- QImage img = qvariant_cast<QImage>( _mimes->imageData());
+ // Support for plain text, html and images (png) are standardized,
+ // copy those to the clipboard data object.
+ emscripten::val clipboardData = emscripten::val::object();
+ for (const QString &mimetype: mimeData->formats()) {
+ if (mimetype == QLatin1String("text/plain")) {
+ emscripten::val text = mimeData->text().toEcmaString();
+ clipboardData.set(mimetype.toEcmaString(), text);
+ } else if (mimetype == QLatin1String("text/html")) {
+ emscripten::val html = mimeData->html().toEcmaString();
+ clipboardData.set(mimetype.toEcmaString(), html);
+ } else if (mimetype.contains(QLatin1String("image"))) {
+ // Serialize the Qt image data to browser supported png
+ QImage img = qvariant_cast<QImage>(mimeData->imageData());
+ QByteArray ba;
QBuffer buffer(&ba);
buffer.open(QIODevice::WriteOnly);
img.save(&buffer, "PNG");
- mimetype = "image/png"; // chrome only allows png
- // clipboard error "NotAllowedError" "Type application/x-qt-image not supported on write."
- // safari silently fails
- // so we use png internally for now
- } else {
- // DATA
- ba = _mimes->data(mimetype);
- }
- // Create file data Blob
- const char *content = ba.data();
- int dataLength = ba.length();
- if (dataLength < 1) {
- qDebug() << "no content found";
- return;
+ qstdweb::Blob blob = qstdweb::Blob::fromArrayBuffer(qstdweb::Uint8Array::copyFrom(ba).buffer());
+ clipboardData.set(std::string("image/png"), blob.val());
}
-
- emscripten::val document = emscripten::val::global("document");
- emscripten::val window = emscripten::val::global("window");
-
- emscripten::val fileContentView =
- emscripten::val(emscripten::typed_memory_view(dataLength, content));
- emscripten::val fileContentCopy = emscripten::val::global("ArrayBuffer").new_(dataLength);
- emscripten::val fileContentCopyView =
- emscripten::val::global("Uint8Array").new_(fileContentCopy);
- fileContentCopyView.call<void>("set", fileContentView);
-
- emscripten::val contentArray = emscripten::val::array();
- contentArray.call<void>("push", fileContentCopyView);
-
- // we have a blob, now create a ClipboardItem
- emscripten::val type = emscripten::val::array();
- type.set("type", mimetype.toEcmaString());
-
- emscripten::val contentBlob = emscripten::val::global("Blob").new_(contentArray, type);
-
- emscripten::val clipboardItemObject = emscripten::val::object();
- clipboardItemObject.set(mimetype.toEcmaString(), contentBlob);
-
- val clipboardItemData = val::global("ClipboardItem").new_(clipboardItemObject);
-
- clipboardWriteArray.call<void>("push", clipboardItemData);
-
- // Clipboard write is only supported with one ClipboardItem at the moment
- // but somehow this still works?
- // break;
}
- val navigator = val::global("navigator");
+ // Return if there is no data (creating an empty ClipboardItem is an error)
+ if (val::global("Object").call<val>("keys", clipboardData)["length"].as<int>() == 0)
+ return;
+ // Write a single clipboard item containing the data formats to the clipboard
+ emscripten::val clipboardItem = val::global("ClipboardItem").new_(clipboardData);
+ emscripten::val clipboardItemArray = emscripten::val::array();
+ clipboardItemArray.call<void>("push", clipboardItem);
+ val navigator = val::global("navigator");
qstdweb::Promise::make(
navigator["clipboard"], "write",
{
@@ -272,7 +216,7 @@ void QWasmClipboard::writeToClipboardApi()
<< QString::fromStdString(error["message"].as<std::string>());
}
},
- clipboardWriteArray);
+ clipboardItemArray);
}
void QWasmClipboard::writeToClipboard()
diff --git a/src/plugins/platforms/wasm/qwasminputcontext.cpp b/src/plugins/platforms/wasm/qwasminputcontext.cpp
index 2df7066b8e2..191e2947629 100644
--- a/src/plugins/platforms/wasm/qwasminputcontext.cpp
+++ b/src/plugins/platforms/wasm/qwasminputcontext.cpp
@@ -25,105 +25,88 @@ void QWasmInputContext::inputCallback(emscripten::val event)
{
qCDebug(qLcQpaWasmInputContext) << Q_FUNC_INFO << "isComposing : " << event["isComposing"].as<bool>();
- QString inputStr = (event["data"] != emscripten::val::null()
- && event["data"] != emscripten::val::undefined()) ?
- QString::fromStdString(event["data"].as<std::string>()) : QString();
-
- QWasmInputContext *wasmInput =
- reinterpret_cast<QWasmInputContext *>(event["target"]["data-qinputcontext"].as<quintptr>());
-
emscripten::val inputType = event["inputType"];
- if (inputType != emscripten::val::null()
- && inputType != emscripten::val::undefined()) {
- const auto inputTypeString = inputType.as<std::string>();
- // There are many inputTypes for InputEvent
- // https://www.w3.org/TR/input-events-1/
- // Some of them should be implemented here later.
- qCDebug(qLcQpaWasmInputContext) << Q_FUNC_INFO << "inputType : " << inputTypeString;
- if (!inputTypeString.compare("deleteContentBackward")) {
- QWindowSystemInterface::handleKeyEvent(0,
- QEvent::KeyPress,
- Qt::Key_Backspace,
- Qt::NoModifier);
- QWindowSystemInterface::handleKeyEvent(0,
- QEvent::KeyRelease,
- Qt::Key_Backspace,
- Qt::NoModifier);
- event.call<void>("stopImmediatePropagation");
- return;
- } else if (!inputTypeString.compare("deleteContentForward")) {
- QWindowSystemInterface::handleKeyEvent(0,
- QEvent::KeyPress,
- Qt::Key_Delete,
- Qt::NoModifier);
- QWindowSystemInterface::handleKeyEvent(0,
- QEvent::KeyRelease,
- Qt::Key_Delete,
- Qt::NoModifier);
- event.call<void>("stopImmediatePropagation");
- return;
- } else if (!inputTypeString.compare("insertCompositionText")) {
- qCDebug(qLcQpaWasmInputContext) << "inputString : " << inputStr;
- wasmInput->insertPreedit();
- event.call<void>("stopImmediatePropagation");
- return;
- } else if (!inputTypeString.compare("insertReplacementText")) {
- qCDebug(qLcQpaWasmInputContext) << "inputString : " << inputStr;
- //auto ranges = event.call<emscripten::val>("getTargetRanges");
- //qCDebug(qLcQpaWasmInputContext) << ranges["length"].as<int>();
- // WA For Korean IME
- // insertReplacementText should have targetRanges but
- // Safari cannot have it and just it seems to be supposed
- // to replace previous input.
- wasmInput->insertText(inputStr, true);
-
- event.call<void>("stopImmediatePropagation");
- return;
- } else if (!inputTypeString.compare("deleteCompositionText")) {
- wasmInput->setPreeditString("", 0);
- wasmInput->insertPreedit();
- event.call<void>("stopImmediatePropagation");
- return;
- } else if (!inputTypeString.compare("insertFromComposition")) {
- wasmInput->setPreeditString(inputStr, 0);
- wasmInput->insertPreedit();
- event.call<void>("stopImmediatePropagation");
- return;
- } else if (!inputTypeString.compare("insertText")) {
- wasmInput->insertText(inputStr);
- event.call<void>("stopImmediatePropagation");
+ if (inputType.isNull() || inputType.isUndefined())
+ return;
+ const auto inputTypeString = inputType.as<std::string>();
+
+ emscripten::val inputData = event["data"];
+ QString inputStr = (!inputData.isNull() && !inputData.isUndefined())
+ ? QString::fromEcmaString(inputData) : QString();
+
+ // There are many inputTypes for InputEvent
+ // https://www.w3.org/TR/input-events-1/
+ // Some of them should be implemented here later.
+ qCDebug(qLcQpaWasmInputContext) << Q_FUNC_INFO << "inputType : " << inputTypeString;
+ if (!inputTypeString.compare("deleteContentBackward")) {
+ QWindowSystemInterface::handleKeyEvent(0, QEvent::KeyPress, Qt::Key_Backspace, Qt::NoModifier);
+ QWindowSystemInterface::handleKeyEvent(0, QEvent::KeyRelease, Qt::Key_Backspace, Qt::NoModifier);
+ event.call<void>("stopImmediatePropagation");
+ return;
+ } else if (!inputTypeString.compare("deleteContentForward")) {
+ QWindowSystemInterface::handleKeyEvent(0, QEvent::KeyPress, Qt::Key_Delete, Qt::NoModifier);
+ QWindowSystemInterface::handleKeyEvent(0, QEvent::KeyRelease, Qt::Key_Delete, Qt::NoModifier);
+ event.call<void>("stopImmediatePropagation");
+ return;
+ } else if (!inputTypeString.compare("insertCompositionText")) {
+ qCDebug(qLcQpaWasmInputContext) << "inputString : " << inputStr;
+ insertPreedit();
+ event.call<void>("stopImmediatePropagation");
+ return;
+ } else if (!inputTypeString.compare("insertReplacementText")) {
+ qCDebug(qLcQpaWasmInputContext) << "inputString : " << inputStr;
+ //auto ranges = event.call<emscripten::val>("getTargetRanges");
+ //qCDebug(qLcQpaWasmInputContext) << ranges["length"].as<int>();
+ // WA For Korean IME
+ // insertReplacementText should have targetRanges but
+ // Safari cannot have it and just it seems to be supposed
+ // to replace previous input.
+ insertText(inputStr, true);
+
+ event.call<void>("stopImmediatePropagation");
+ return;
+ } else if (!inputTypeString.compare("deleteCompositionText")) {
+ setPreeditString("", 0);
+ insertPreedit();
+ event.call<void>("stopImmediatePropagation");
+ return;
+ } else if (!inputTypeString.compare("insertFromComposition")) {
+ setPreeditString(inputStr, 0);
+ insertPreedit();
+ event.call<void>("stopImmediatePropagation");
+ return;
+ } else if (!inputTypeString.compare("insertText")) {
+ insertText(inputStr);
+ event.call<void>("stopImmediatePropagation");
#if QT_CONFIG(clipboard)
- } else if (!inputTypeString.compare("insertFromPaste")) {
- wasmInput->insertText(QGuiApplication::clipboard()->text());
- event.call<void>("stopImmediatePropagation");
- // These can be supported here,
- // But now, keyCallback in QWasmWindow
- // will take them as exceptions.
- //} else if (!inputTypeString.compare("deleteByCut")) {
+ } else if (!inputTypeString.compare("insertFromPaste")) {
+ insertText(QGuiApplication::clipboard()->text());
+ event.call<void>("stopImmediatePropagation");
+ // These can be supported here,
+ // But now, keyCallback in QWasmWindow
+ // will take them as exceptions.
+ //} else if (!inputTypeString.compare("deleteByCut")) {
#endif
- } else {
- qCWarning(qLcQpaWasmInputContext) << Q_FUNC_INFO << "inputType \"" << inputType.as<std::string>() << "\" is not supported in Qt yet";
- }
+ } else {
+ qCWarning(qLcQpaWasmInputContext) << Q_FUNC_INFO << "inputType \"" <<
+ inputType.as<std::string>() << "\" is not supported in Qt yet";
}
}
void QWasmInputContext::compositionEndCallback(emscripten::val event)
{
- const auto inputStr = QString::fromStdString(event["data"].as<std::string>());
+ const auto inputStr = QString::fromEcmaString(event["data"]);
qCDebug(qLcQpaWasmInputContext) << Q_FUNC_INFO << inputStr;
- QWasmInputContext *wasmInput =
- reinterpret_cast<QWasmInputContext *>(event["target"]["data-qinputcontext"].as<quintptr>());
-
- if (wasmInput->preeditString().isEmpty())
+ if (preeditString().isEmpty())
return;
- if (inputStr != wasmInput->preeditString()) {
+ if (inputStr != preeditString()) {
qCWarning(qLcQpaWasmInputContext) << Q_FUNC_INFO
<< "Composition string" << inputStr
- << "is differ from" << wasmInput->preeditString();
+ << "is differ from" << preeditString();
}
- wasmInput->commitPreeditAndClear();
+ commitPreeditAndClear();
}
void QWasmInputContext::compositionStartCallback(emscripten::val event)
@@ -151,19 +134,15 @@ static void beforeInputCallback(emscripten::val event)
void QWasmInputContext::compositionUpdateCallback(emscripten::val event)
{
- const auto compositionStr = QString::fromStdString(event["data"].as<std::string>());
+ const auto compositionStr = QString::fromEcmaString(event["data"]);
qCDebug(qLcQpaWasmInputContext) << Q_FUNC_INFO << compositionStr;
- QWasmInputContext *wasmInput =
- reinterpret_cast<QWasmInputContext *>(event["target"]["data-qinputcontext"].as<quintptr>());
-
// WA for IOS.
// Not sure now because I cannot test it anymore.
// int replaceSize = 0;
// emscripten::val win = emscripten::val::global("window");
// emscripten::val sel = win.call<emscripten::val>("getSelection");
-// if (sel != emscripten::val::null()
-// && sel != emscripten::val::undefined()
+// if (!sel.isNull() && !sel.isUndefined()
// && sel["rangeCount"].as<int>() > 0) {
// QInputMethodQueryEvent queryEvent(Qt::ImQueryAll);
// QCoreApplication::sendEvent(QGuiApplication::focusObject(), &queryEvent);
@@ -172,8 +151,8 @@ void QWasmInputContext::compositionUpdateCallback(emscripten::val event)
// qCDebug(qLcQpaWasmInputContext) << "Qt text before cursor: " << queryEvent.value(Qt::ImTextBeforeCursor).toString();
// qCDebug(qLcQpaWasmInputContext) << "Qt text after cursor: " << queryEvent.value(Qt::ImTextAfterCursor).toString();
//
-// const QString &selectedStr = QString::fromUtf8(sel.call<emscripten::val>("toString").as<std::string>());
-// const auto &preeditStr = wasmInput->preeditString();
+// const QString &selectedStr = QString::fromEcmaString(sel.call<emscripten::val>("toString"));
+// const auto &preeditStr = preeditString();
// qCDebug(qLcQpaWasmInputContext) << "Selection.type : " << sel["type"].as<std::string>();
// qCDebug(qLcQpaWasmInputContext) << Q_FUNC_INFO << "Selected: " << selectedStr;
// qCDebug(qLcQpaWasmInputContext) << Q_FUNC_INFO << "PreeditString: " << preeditStr;
@@ -189,90 +168,13 @@ void QWasmInputContext::compositionUpdateCallback(emscripten::val event)
// qCDebug(qLcQpaWasmInputContext) << "Range.endOffset : " << range["endOffset"].as<int>();
// }
//
-// wasmInput->setPreeditString(compositionStr, replaceSize);
- wasmInput->setPreeditString(compositionStr, 0);
-}
-
-#if QT_CONFIG(clipboard)
-static void copyCallback(emscripten::val event)
-{
- qCDebug(qLcQpaWasmInputContext) << Q_FUNC_INFO;
-
- QClipboard *clipboard = QGuiApplication::clipboard();
- QString inputStr = clipboard->text();
- qCDebug(qLcQpaWasmInputContext) << "QClipboard : " << inputStr;
- event["clipboardData"].call<void>("setData",
- emscripten::val("text/plain"),
- inputStr.toStdString());
- event.call<void>("preventDefault");
- event.call<void>("stopImmediatePropagation");
-}
-
-static void cutCallback(emscripten::val event)
-{
- qCDebug(qLcQpaWasmInputContext) << Q_FUNC_INFO;
-
- QClipboard *clipboard = QGuiApplication::clipboard();
- QString inputStr = clipboard->text();
- qCDebug(qLcQpaWasmInputContext) << "QClipboard : " << inputStr;
- event["clipboardData"].call<void>("setData",
- emscripten::val("text/plain"),
- inputStr.toStdString());
- event.call<void>("preventDefault");
- event.call<void>("stopImmediatePropagation");
+// setPreeditString(compositionStr, replaceSize);
+ setPreeditString(compositionStr, 0);
}
-static void pasteCallback(emscripten::val event)
-{
- qCDebug(qLcQpaWasmInputContext) << Q_FUNC_INFO;
-
- emscripten::val clipboardData = event["clipboardData"].call<emscripten::val>("getData", emscripten::val("text/plain"));
- QString clipboardStr = QString::fromStdString(clipboardData.as<std::string>());
- qCDebug(qLcQpaWasmInputContext) << "wasm clipboard : " << clipboardStr;
- QClipboard *clipboard = QGuiApplication::clipboard();
- if (clipboard->text() != clipboardStr)
- clipboard->setText(clipboardStr);
-
- // propagate to input event (insertFromPaste)
-}
-#endif // QT_CONFIG(clipboard)
-
QWasmInputContext::QWasmInputContext()
{
qCDebug(qLcQpaWasmInputContext) << Q_FUNC_INFO;
- emscripten::val document = emscripten::val::global("document");
- // This 'input' can be an issue to handle multiple lines,
- // 'textarea' can be used instead.
- m_inputElement = document.call<emscripten::val>("createElement", std::string("input"));
- m_inputElement.set("type", "text");
- m_inputElement.set("contenteditable","true");
- m_inputElement.call<void>("setAttribute", std::string("aria-hidden"), std::string("true"));
-
- m_inputElement["style"].set("position", "absolute");
- m_inputElement["style"].set("left", 0);
- m_inputElement["style"].set("top", 0);
- m_inputElement["style"].set("opacity", 0);
- m_inputElement["style"].set("display", "");
- m_inputElement["style"].set("z-index", -2);
- m_inputElement["style"].set("width", "1px");
- m_inputElement["style"].set("height", "1px");
-
- m_inputElement.set("data-qinputcontext",
- emscripten::val(quintptr(reinterpret_cast<void *>(this))));
- emscripten::val body = document["body"];
- body.call<void>("appendChild", m_inputElement);
-
- m_inputCallback = QWasmEventHandler(m_inputElement, "input", QWasmInputContext::inputCallback);
- m_compositionEndCallback = QWasmEventHandler(m_inputElement, "compositionend", QWasmInputContext::compositionEndCallback);
- m_compositionStartCallback = QWasmEventHandler(m_inputElement, "compositionstart", QWasmInputContext::compositionStartCallback);
- m_compositionUpdateCallback = QWasmEventHandler(m_inputElement, "compositionupdate", QWasmInputContext::compositionUpdateCallback);
-
-#if QT_CONFIG(clipboard)
- // Clipboard for InputContext
- m_clipboardCut = QWasmEventHandler(m_inputElement, "cut", cutCallback);
- m_clipboardCopy = QWasmEventHandler(m_inputElement, "copy", copyCallback);
- m_clipboardPaste = QWasmEventHandler(m_inputElement, "paste", pasteCallback);
-#endif
}
QWasmInputContext::~QWasmInputContext()
@@ -303,6 +205,9 @@ void QWasmInputContext::showInputPanel()
void QWasmInputContext::updateGeometry()
{
+ if (m_inputElement.isNull())
+ return;
+
const QWindow *focusWindow = QGuiApplication::focusWindow();
if (!m_focusObject || !focusWindow || !m_inputMethodAccepted) {
m_inputElement["style"].set("left", "0px");
@@ -312,23 +217,12 @@ void QWasmInputContext::updateGeometry()
Q_ASSERT(m_focusObject);
Q_ASSERT(m_inputMethodAccepted);
- // Set the geometry
- QPoint globalPos;
- const QRect cursorRectangle = QPlatformInputContext::cursorRectangle().toRect();
- if (cursorRectangle.isValid()) {
- qCDebug(qLcQpaWasmInputContext)
- << Q_FUNC_INFO << "cursorRectangle: " << cursorRectangle;
- globalPos = focusWindow->mapToGlobal(cursorRectangle.topLeft());
- if (globalPos.x() > 0)
- globalPos.setX(globalPos.x() - 1);
- if (globalPos.y() > 0)
- globalPos.setY(globalPos.y() - 1);
- }
-
- const auto styleLeft = std::to_string(globalPos.x()) + "px";
- const auto styleTop = std::to_string(globalPos.y()) + "px";
- m_inputElement["style"].set("left", styleLeft);
- m_inputElement["style"].set("top", styleTop);
+ const QRect inputItemRectangle = QPlatformInputContext::inputItemRectangle().toRect();
+ qCDebug(qLcQpaWasmInputContext) << Q_FUNC_INFO << "propagating inputItemRectangle:" << inputItemRectangle;
+ m_inputElement["style"].set("left", std::to_string(inputItemRectangle.x()) + "px");
+ m_inputElement["style"].set("top", std::to_string(inputItemRectangle.y()) + "px");
+ m_inputElement["style"].set("width", std::to_string(inputItemRectangle.width()) + "px");
+ m_inputElement["style"].set("height", std::to_string(inputItemRectangle.height()) + "px");
}
}
@@ -341,16 +235,21 @@ void QWasmInputContext::updateInputElement()
updateGeometry();
// If there is no focus object, or no visible input panel, remove focus
- const QWindow *focusWindow = QGuiApplication::focusWindow();
+ QWasmWindow *focusWindow = QWasmWindow::fromWindow(QGuiApplication::focusWindow());
if (!m_focusObject || !focusWindow || !m_inputMethodAccepted) {
- m_inputElement.set("value", "");
+ if (!m_inputElement.isNull()) {
+ m_inputElement.set("value", "");
+ m_inputElement.set("inputMode", std::string("none"));
+ }
- if (QWasmWindow *wasmwindow = QWasmWindow::fromWindow(focusWindow))
- wasmwindow->focus();
- else
- m_inputElement.call<void>("blur");
+ if (focusWindow) {
+ focusWindow->focus();
+ } else {
+ if (!m_inputElement.isNull())
+ m_inputElement.call<void>("blur");
+ }
- m_inputElement.set("inputMode", std::string("none"));
+ m_inputElement = emscripten::val::null();
return;
}
@@ -358,6 +257,8 @@ void QWasmInputContext::updateInputElement()
Q_ASSERT(m_focusObject);
Q_ASSERT(m_inputMethodAccepted);
+ m_inputElement = focusWindow->inputElement();
+
qCDebug(qLcQpaWasmInputContext) << Q_FUNC_INFO << QRectF::fromDOMRect(m_inputElement.call<emscripten::val>("getBoundingClientRect"));
// Set the text input
@@ -375,7 +276,15 @@ void QWasmInputContext::updateInputElement()
m_inputElement.set("selectionStart", queryEvent.value(Qt::ImAnchorPosition).toUInt());
m_inputElement.set("selectionEnd", queryEvent.value(Qt::ImCursorPosition).toUInt());
+ QInputMethodQueryEvent query((Qt::InputMethodQueries(Qt::ImHints)));
+ QCoreApplication::sendEvent(m_focusObject, &query);
+ if (Qt::InputMethodHints(query.value(Qt::ImHints).toInt()).testFlag(Qt::ImhHiddenText))
+ m_inputElement.set("type", "password");
+ else
+ m_inputElement.set("type", "text");
+
m_inputElement.set("inputMode", std::string("text"));
+
m_inputElement.call<void>("focus");
}
@@ -383,16 +292,6 @@ void QWasmInputContext::setFocusObject(QObject *object)
{
qCDebug(qLcQpaWasmInputContext) << Q_FUNC_INFO << object << inputMethodAccepted();
- QInputMethodQueryEvent query(Qt::InputMethodQueries(Qt::ImEnabled | Qt::ImHints));
- QCoreApplication::sendEvent(object, &query);
- if (query.value(Qt::ImEnabled).toBool()
- && Qt::InputMethodHints(query.value(Qt::ImHints).toInt()).testFlag(Qt::ImhHiddenText)) {
- m_inputElement.set("type", "password");
- } else {
- if (m_inputElement["type"].as<std::string>() != std::string("text"))
- m_inputElement.set("type", "text");
- }
-
// Commit the previous composition before change m_focusObject
if (m_focusObject && !m_preeditString.isEmpty())
commitPreeditAndClear();
diff --git a/src/plugins/platforms/wasm/qwasminputcontext.h b/src/plugins/platforms/wasm/qwasminputcontext.h
index 72476adfe1b..6d24c7fea0d 100644
--- a/src/plugins/platforms/wasm/qwasminputcontext.h
+++ b/src/plugins/platforms/wasm/qwasminputcontext.h
@@ -35,38 +35,33 @@ public:
void setPreeditString(QString preeditStr, int replaceSize);
void insertPreedit();
void commitPreeditAndClear();
- emscripten::val m_inputElement = emscripten::val::null();
void insertText(QString inputStr, bool replace = false);
- QWasmEventHandler m_inputCallback;
- QWasmEventHandler m_compositionEndCallback;
- QWasmEventHandler m_compositionStartCallback;
- QWasmEventHandler m_compositionUpdateCallback;
-
bool usingTextInput() const { return m_inputMethodAccepted; }
void setFocusObject(QObject *object) override;
- static void inputCallback(emscripten::val event);
- static void compositionEndCallback(emscripten::val event);
- static void compositionStartCallback(emscripten::val event);
- static void compositionUpdateCallback(emscripten::val event);
+ void inputCallback(emscripten::val event);
+ void compositionEndCallback(emscripten::val event);
+ void compositionStartCallback(emscripten::val event);
+ void compositionUpdateCallback(emscripten::val event);
void updateGeometry();
+ bool isActive() const {
+ return m_focusObject && m_inputMethodAccepted;
+ }
+
private:
void updateInputElement();
private:
- QWasmEventHandler m_clipboardCut;
- QWasmEventHandler m_clipboardCopy;
- QWasmEventHandler m_clipboardPaste;
-
QString m_preeditString;
int m_replaceSize = 0;
bool m_inputMethodAccepted = false;
QObject *m_focusObject = nullptr;
+ emscripten::val m_inputElement = emscripten::val::null();
};
QT_END_NAMESPACE
diff --git a/src/plugins/platforms/wasm/qwasmwindow.cpp b/src/plugins/platforms/wasm/qwasmwindow.cpp
index d690bcfe10a..d27385be723 100644
--- a/src/plugins/platforms/wasm/qwasmwindow.cpp
+++ b/src/plugins/platforms/wasm/qwasmwindow.cpp
@@ -51,7 +51,10 @@ QWasmWindow::QWasmWindow(QWindow *w, QWasmDeadKeySupport *deadKeySupport,
m_decoratedWindow(m_document.call<emscripten::val>("createElement", emscripten::val("div"))),
m_window(m_document.call<emscripten::val>("createElement", emscripten::val("div"))),
m_a11yContainer(m_document.call<emscripten::val>("createElement", emscripten::val("div"))),
- m_canvas(m_document.call<emscripten::val>("createElement", emscripten::val("canvas")))
+ m_canvas(m_document.call<emscripten::val>("createElement", emscripten::val("canvas"))),
+ m_focusHelper(m_document.call<emscripten::val>("createElement", emscripten::val("div"))),
+ m_inputElement(m_document.call<emscripten::val>("createElement", emscripten::val("input")))
+
{
m_decoratedWindow.set("className", "qt-decorated-window");
m_decoratedWindow["style"].set("display", std::string("none"));
@@ -80,17 +83,6 @@ QWasmWindow::QWasmWindow(QWindow *w, QWasmDeadKeySupport *deadKeySupport,
m_canvas["classList"].call<void>("add", emscripten::val("qt-window-canvas"));
- // Set contentEditable for two reasons;
- // 1) so that the window gets clipboard events,
- // 2) For applications who will handle keyboard events, but without having inputMethodAccepted()
- //
- // Set inputMode to none to avoid keyboard popping up on push buttons
- // This is a tradeoff, we are not able to separate between a push button and
- // a widget that reads keyboard events.
- m_canvas.call<void>("setAttribute", std::string("inputmode"), std::string("none"));
- m_canvas.call<void>("setAttribute", std::string("contenteditable"), std::string("true"));
- m_canvas["style"].set("outline", std::string("none"));
-
#if QT_CONFIG(clipboard)
if (QWasmClipboard::shouldInstallWindowEventHandlers()) {
m_cutCallback = QWasmEventHandler(m_canvas, "cut", QWasmClipboard::cut);
@@ -99,9 +91,37 @@ QWasmWindow::QWasmWindow(QWindow *w, QWasmDeadKeySupport *deadKeySupport,
}
#endif
- // Set inputMode to none to stop the mobile keyboard from opening
- // when the user clicks on the window.
- m_window.set("inputMode", std::string("none"));
+ // Set up m_focusHelper, which is an invisible child element of the window which takes
+ // focus on behalf of the window any time the window has focus in general, but none
+ // of the special child elements such as the inputElment or a11y elements have focus.
+ // Set inputMode=none set to prevent the virtual keyboard from popping up.
+ m_focusHelper["classList"].call<void>("add", emscripten::val("qt-window-focus-helper"));
+ m_focusHelper.set("inputMode", std::string("none"));
+ m_focusHelper.call<void>("setAttribute", std::string("aria-hidden"), std::string("true"));
+ m_focusHelper.call<void>("setAttribute", std::string("contenteditable"), std::string("true"));
+ m_focusHelper["style"].set("position", "absolute");
+ m_focusHelper["style"].set("left", 0);
+ m_focusHelper["style"].set("top", 0);
+ m_focusHelper["style"].set("width", "1px");
+ m_focusHelper["style"].set("height", "1px");
+ m_focusHelper["style"].set("z-index", -2);
+ m_focusHelper["style"].set("opacity", 0);
+ m_window.call<void>("appendChild", m_focusHelper);
+
+ // Set up m_inputElement, which takes focus whenever a Qt text input UI element has
+ // foucus.
+ m_inputElement["classList"].call<void>("add", emscripten::val("qt-window-input-element"));
+ m_inputElement.set("type", "text");
+ m_inputElement.call<void>("setAttribute", std::string("aria-hidden"), std::string("true"));
+ m_inputElement["style"].set("position", "absolute");
+ m_inputElement["style"].set("left", 0);
+ m_inputElement["style"].set("top", 0);
+ m_inputElement["style"].set("width", "1px");
+ m_inputElement["style"].set("height", "1px");
+ m_inputElement["style"].set("z-index", -2);
+ m_inputElement["style"].set("opacity", 0);
+ m_inputElement["style"].set("display", "");
+ m_window.call<void>("appendChild", m_inputElement);
// Hide the canvas from screen readers.
m_canvas.call<void>("setAttribute", std::string("aria-hidden"), std::string("true"));
@@ -193,21 +213,20 @@ void QWasmWindow::registerEventHandlers()
m_wheelEventCallback = QWasmEventHandler(m_window, "wheel",
[this](emscripten::val event) { this->handleWheelEvent(event); });
- QWasmInputContext *wasmInput = QWasmIntegration::get()->wasmInputContext();
- if (wasmInput) {
- m_keyDownCallbackForInputContext =
- QWasmEventHandler(wasmInput->m_inputElement, "keydown",
- [this](emscripten::val event) { this->handleKeyForInputContextEvent(EventType::KeyDown, event); });
- m_keyUpCallbackForInputContext =
- QWasmEventHandler(wasmInput->m_inputElement, "keyup",
- [this](emscripten::val event) { this->handleKeyForInputContextEvent(EventType::KeyUp, event); });
- }
-
- m_keyDownCallback = QWasmEventHandler(m_canvas, "keydown",
+ m_keyDownCallback = QWasmEventHandler(m_window, "keydown",
[this](emscripten::val event) { this->handleKeyEvent(KeyEvent(EventType::KeyDown, event, m_deadKeySupport)); });
- m_keyUpCallback =QWasmEventHandler(m_canvas, "keyup",
+ m_keyUpCallback =QWasmEventHandler(m_window, "keyup",
[this](emscripten::val event) {this->handleKeyEvent(KeyEvent(EventType::KeyUp, event, m_deadKeySupport)); });
-}
+
+ m_inputCallback = QWasmEventHandler(m_window, "input",
+ [this](emscripten::val event){ handleInputEvent(event); });
+ m_compositionUpdateCallback = QWasmEventHandler(m_window, "compositionupdate",
+ [this](emscripten::val event){ handleCompositionUpdateEvent(event); });
+ m_compositionStartCallback = QWasmEventHandler(m_window, "compositionstart",
+ [this](emscripten::val event){ handleCompositionStartEvent(event); });
+ m_compositionEndCallback = QWasmEventHandler(m_window, "compositionend",
+ [this](emscripten::val event){ handleCompositionEndEvent(event); });
+ }
QWasmWindow::~QWasmWindow()
{
@@ -624,10 +643,15 @@ void QWasmWindow::commitParent(QWasmWindowTreeNode *parent)
void QWasmWindow::handleKeyEvent(const KeyEvent &event)
{
- qCDebug(qLcQpaWasmInputContext) << "processKey as KeyEvent";
- if (processKey(event)) {
- event.webEvent.call<void>("preventDefault");
- event.webEvent.call<void>("stopPropagation");
+ qCDebug(qLcQpaWasmInputContext) << "handleKeyEvent";
+
+ if (QWasmInputContext *inputContext = QWasmIntegration::get()->wasmInputContext(); inputContext->isActive()) {
+ handleKeyForInputContextEvent(event);
+ } else {
+ if (processKey(event)) {
+ event.webEvent.call<void>("preventDefault");
+ event.webEvent.call<void>("stopPropagation");
+ }
}
}
@@ -658,7 +682,7 @@ bool QWasmWindow::processKey(const KeyEvent &event)
#endif
}
-void QWasmWindow::handleKeyForInputContextEvent(EventType eventType, const emscripten::val &event)
+void QWasmWindow::handleKeyForInputContextEvent(const KeyEvent &keyEvent)
{
//
// Things to consider:
@@ -668,40 +692,43 @@ void QWasmWindow::handleKeyForInputContextEvent(EventType eventType, const emscr
// complex (i.e Chinese et al) input handling
// Multiline text edit backspace at start of line
//
- const QWasmInputContext *wasmInput = QWasmIntegration::get()->wasmInputContext();
- if (wasmInput) {
+ emscripten::val event = keyEvent.webEvent;
+ bool useInputContext = [event]() -> bool {
+ const QWasmInputContext *wasmInput = QWasmIntegration::get()->wasmInputContext();
+ if (!wasmInput)
+ return false;
+
const auto keyString = QString::fromStdString(event["key"].as<std::string>());
qCDebug(qLcQpaWasmInputContext) << "Key callback" << keyString << keyString.size();
- if (keyString == "Unidentified") {
- // Android makes a bunch of KeyEvents as "Unidentified"
- // They will be processed just in InputContext.
- return;
- } else if (event["isComposing"].as<bool>()) {
- // Handled by the input context
- return;
- } else if (event["ctrlKey"].as<bool>()
- || event["altKey"].as<bool>()
- || event["metaKey"].as<bool>()) {
- // Not all platforms use 'isComposing' for '~' + 'a', in this
- // case send the key with state ('ctrl', 'alt', or 'meta') to
- // processKeyForInputContext
-
- ; // fallthrough
- } else if (keyString.size() != 1) {
- // This is like; 'Shift','ArrowRight','AltGraph', ...
- // send all of these to processKeyForInputContext
-
- ; // fallthrough
- } else if (wasmInput->inputMethodAccepted()) {
- // processed in inputContext with skipping processKey
- return;
- }
- }
- qCDebug(qLcQpaWasmInputContext) << "processKey as KeyEvent";
- if (processKeyForInputContext(KeyEvent(eventType, event, m_deadKeySupport)))
- event.call<void>("preventDefault");
- event.call<void>("stopImmediatePropagation");
+ // Events with isComposing set are handled by the input context
+ bool composing = event["isComposing"].as<bool>();
+
+ // Android makes a bunch of KeyEvents as "Unidentified",
+ // make inputContext handle those.
+ bool androidUnidentified = (keyString == "Unidentified");
+
+ // Not all platforms use 'isComposing' for '~' + 'a', in this
+ // case send the key with state ('ctrl', 'alt', or 'meta') to
+ // processKeyForInputContext
+ bool hasModifiers = event["ctrlKey"].as<bool>()
+ || event["altKey"].as<bool>()
+ || event["metaKey"].as<bool>();
+
+ // This is like; 'Shift','ArrowRight','AltGraph', ...
+ // send all of these to processKeyForInputContext
+ bool hasNoncharacterKeyString = keyString.size() != 1;
+
+ bool overrideCompose = !hasModifiers && !hasNoncharacterKeyString && wasmInput->inputMethodAccepted();
+ return composing || androidUnidentified || overrideCompose;
+ }();
+
+ if (!useInputContext) {
+ qCDebug(qLcQpaWasmInputContext) << "processKey as KeyEvent";
+ if (processKeyForInputContext(keyEvent))
+ event.call<void>("preventDefault");
+ event.call<void>("stopImmediatePropagation");
+ }
}
bool QWasmWindow::processKeyForInputContext(const KeyEvent &event)
@@ -729,6 +756,30 @@ bool QWasmWindow::processKeyForInputContext(const KeyEvent &event)
return result;
}
+void QWasmWindow::handleInputEvent(emscripten::val event)
+{
+ if (QWasmInputContext *inputContext = QWasmIntegration::get()->wasmInputContext(); inputContext->isActive())
+ inputContext->inputCallback(event);
+}
+
+void QWasmWindow::handleCompositionStartEvent(emscripten::val event)
+{
+ if (QWasmInputContext *inputContext = QWasmIntegration::get()->wasmInputContext(); inputContext->isActive())
+ inputContext->compositionStartCallback(event);
+}
+
+void QWasmWindow::handleCompositionUpdateEvent(emscripten::val event)
+{
+ if (QWasmInputContext *inputContext = QWasmIntegration::get()->wasmInputContext(); inputContext->isActive())
+ inputContext->compositionUpdateCallback(event);
+}
+
+void QWasmWindow::handleCompositionEndEvent(emscripten::val event)
+{
+ if (QWasmInputContext *inputContext = QWasmIntegration::get()->wasmInputContext(); inputContext->isActive())
+ inputContext->compositionEndCallback(event);
+}
+
void QWasmWindow::handlePointerEnterLeaveEvent(const PointerEvent &event)
{
if (processPointerEnterLeave(event))
@@ -1040,7 +1091,7 @@ void QWasmWindow::requestActivateWindow()
void QWasmWindow::focus()
{
- m_canvas.call<void>("focus");
+ m_focusHelper.call<void>("focus");
}
bool QWasmWindow::setMouseGrabEnabled(bool grab)
diff --git a/src/plugins/platforms/wasm/qwasmwindow.h b/src/plugins/platforms/wasm/qwasmwindow.h
index 904e736a7e7..0c63ebdc16e 100644
--- a/src/plugins/platforms/wasm/qwasmwindow.h
+++ b/src/plugins/platforms/wasm/qwasmwindow.h
@@ -100,6 +100,7 @@ public:
emscripten::val context2d() const { return m_context2d; }
emscripten::val a11yContainer() const { return m_a11yContainer; }
emscripten::val inputHandlerElement() const { return m_window; }
+ emscripten::val inputElement() const { return m_inputElement; }
// QNativeInterface::Private::QWasmWindow
emscripten::val document() const override { return m_document; }
@@ -137,8 +138,13 @@ private:
void handleKeyEvent(const KeyEvent &event);
bool processKey(const KeyEvent &event);
- void handleKeyForInputContextEvent(EventType eventType, const emscripten::val &event);
+ void handleKeyForInputContextEvent(const KeyEvent &event);
bool processKeyForInputContext(const KeyEvent &event);
+ void handleInputEvent(emscripten::val event);
+ void handleCompositionStartEvent(emscripten::val event);
+ void handleCompositionUpdateEvent(emscripten::val event);
+ void handleCompositionEndEvent(emscripten::val event);
+
void handlePointerEnterLeaveEvent(const PointerEvent &event);
bool processPointerEnterLeave(const PointerEvent &event);
void processPointer(const PointerEvent &event);
@@ -154,11 +160,14 @@ private:
QWasmDeadKeySupport *m_deadKeySupport;
QRect m_normalGeometry {0, 0, 0 ,0};
- emscripten::val m_document = emscripten::val::undefined();
- emscripten::val m_decoratedWindow = emscripten::val::undefined();
- emscripten::val m_window = emscripten::val::undefined();
- emscripten::val m_a11yContainer = emscripten::val::undefined();
- emscripten::val m_canvas = emscripten::val::undefined();
+ emscripten::val m_document;
+ emscripten::val m_decoratedWindow;
+ emscripten::val m_window;
+ emscripten::val m_a11yContainer;
+ emscripten::val m_canvas;
+ emscripten::val m_focusHelper;
+ emscripten::val m_inputElement;
+
emscripten::val m_context2d = emscripten::val::undefined();
std::unique_ptr<NonClientArea> m_nonClientArea;
@@ -169,6 +178,10 @@ private:
QWasmEventHandler m_keyUpCallback;
QWasmEventHandler m_keyDownCallbackForInputContext;
QWasmEventHandler m_keyUpCallbackForInputContext;
+ QWasmEventHandler m_inputCallback;
+ QWasmEventHandler m_compositionStartCallback;
+ QWasmEventHandler m_compositionUpdateCallback;
+ QWasmEventHandler m_compositionEndCallback;
QWasmEventHandler m_pointerDownCallback;
QWasmEventHandler m_pointerMoveCallback;
diff --git a/src/plugins/styles/modernwindows/qwindows11style.cpp b/src/plugins/styles/modernwindows/qwindows11style.cpp
index 62269a21de1..84101b69e9f 100644
--- a/src/plugins/styles/modernwindows/qwindows11style.cpp
+++ b/src/plugins/styles/modernwindows/qwindows11style.cpp
@@ -2151,9 +2151,6 @@ void QWindows11Style::polish(QWidget* widget)
widget->setWindowFlag(Qt::NoDropShadowWindowHint);
widget->setAttribute(Qt::WA_RightToLeft, layoutDirection);
widget->setAttribute(Qt::WA_WState_Created, wasCreated);
- auto pal = widget->palette();
- pal.setColor(widget->backgroundRole(), Qt::transparent);
- widget->setPalette(pal);
if (!isScrollBar) {
bool inGraphicsView = widget->graphicsProxyWidget() != nullptr;
if (!inGraphicsView && comboBoxContainer && comboBoxContainer->parentWidget())
diff --git a/src/testlib/3rdparty/catch2/qt_attribution.json b/src/testlib/3rdparty/catch2/qt_attribution.json
index 1e4f2c5297c..ad231efd426 100644
--- a/src/testlib/3rdparty/catch2/qt_attribution.json
+++ b/src/testlib/3rdparty/catch2/qt_attribution.json
@@ -5,6 +5,7 @@
"QDocModule": "qttestlib",
"QtUsage": "Used for testing of the Qt Test module.",
"Comment": {
+ "OldBranch": "Stick to its v2.x branch; no suitable file in v3",
"UpstreamFile": "single_include/catch2/catch.hpp",
"Generator": "scripts/generateSingleHeader.py",
"General": "no relevant CPE found"
diff --git a/src/tools/CMakeLists.txt b/src/tools/CMakeLists.txt
index 1ff52f8a84f..2f06cbcf67e 100644
--- a/src/tools/CMakeLists.txt
+++ b/src/tools/CMakeLists.txt
@@ -28,6 +28,10 @@ if(QT_FEATURE_macdeployqt)
add_subdirectory(macdeployqt)
endif()
+if(QT_FEATURE_wasmdeployqt)
+ add_subdirectory(wasmdeployqt)
+endif()
+
if(QT_FEATURE_windeployqt)
add_subdirectory(windeployqt)
endif()
diff --git a/src/tools/configure.cmake b/src/tools/configure.cmake
index 6a9c1b8e3f3..27ea90b89ac 100644
--- a/src/tools/configure.cmake
+++ b/src/tools/configure.cmake
@@ -18,6 +18,12 @@ qt_feature("macdeployqt" PRIVATE
AUTODETECT CMAKE_HOST_APPLE
CONDITION MACOS AND QT_FEATURE_thread)
+qt_feature("wasmdeployqt" PRIVATE
+ SECTION "Deployment"
+ LABEL "WebAssembly deployment tool"
+ PURPOSE "The WebAssembly deployment tool is designed to automate the process of creating a deployable folder especially for dynamic linking case variant."
+ CONDITION QT_FEATURE_process)
+
qt_feature("windeployqt" PRIVATE
SECTION "Deployment"
LABEL "Windows deployment tool"
diff --git a/src/tools/wasmdeployqt/CMakeLists.txt b/src/tools/wasmdeployqt/CMakeLists.txt
new file mode 100644
index 00000000000..7305c14c269
--- /dev/null
+++ b/src/tools/wasmdeployqt/CMakeLists.txt
@@ -0,0 +1,19 @@
+# Copyright (C) 2025 The Qt Company Ltd.
+# SPDX-License-Identifier: BSD-3-Clause
+
+#####################################################################
+## wasmdeployqt Tool:
+#####################################################################
+
+qt_get_tool_target_name(target_name wasmdeployqt)
+qt_internal_add_tool(${target_name}
+ TOOLS_TARGET Core
+ USER_FACING
+ INSTALL_VERSIONED_LINK
+ TARGET_DESCRIPTION "Qt WebAssembly Deployment Tool"
+ SOURCES
+ main.cpp wasmbinary.cpp jsontools.cpp
+ LIBRARIES
+ Qt::CorePrivate
+)
+qt_internal_return_unless_building_tools()
diff --git a/src/tools/wasmdeployqt/common.h b/src/tools/wasmdeployqt/common.h
new file mode 100644
index 00000000000..258d6161e67
--- /dev/null
+++ b/src/tools/wasmdeployqt/common.h
@@ -0,0 +1,26 @@
+// Copyright (C) 2025 The Qt Company Ltd.
+// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR GPL-3.0-only WITH Qt-GPL-exception-1.0
+
+#ifndef COMMON_H
+#define COMMON_H
+
+#include <QHash>
+#include <QString>
+
+struct PreloadEntry
+{
+ QString source;
+ QString destination;
+
+ bool operator==(const PreloadEntry &other) const
+ {
+ return source == other.source && destination == other.destination;
+ }
+};
+
+inline uint qHash(const PreloadEntry &key, uint seed = 0)
+{
+ return qHash(key.source, seed) ^ qHash(key.destination, seed);
+}
+
+#endif
diff --git a/src/tools/wasmdeployqt/jsontools.cpp b/src/tools/wasmdeployqt/jsontools.cpp
new file mode 100644
index 00000000000..d76f9190b73
--- /dev/null
+++ b/src/tools/wasmdeployqt/jsontools.cpp
@@ -0,0 +1,101 @@
+// Copyright (C) 2025 The Qt Company Ltd.
+// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR GPL-3.0-only WITH Qt-GPL-exception-1.0
+
+#include <QDir>
+#include <QJsonArray>
+#include <QJsonObject>
+
+#include "jsontools.h"
+#include "common.h"
+
+#include <iostream>
+#include <optional>
+
+namespace JsonTools {
+
+bool savePreloadFile(QSet<PreloadEntry> preload, QString destFile)
+{
+
+ QJsonArray jsonArray;
+ for (const PreloadEntry &entry : preload) {
+ QJsonObject obj;
+ obj["source"] = entry.source;
+ obj["destination"] = entry.destination;
+ jsonArray.append(obj);
+ }
+ QJsonDocument doc(jsonArray);
+
+ QFile outFile(destFile);
+ if (outFile.exists()) {
+ if (!outFile.remove()) {
+ std::cout << "ERROR: Failed to delete old file: " << outFile.fileName().toStdString()
+ << std::endl;
+ return false;
+ }
+ }
+ if (!outFile.open(QIODevice::WriteOnly | QIODevice::Text)) {
+ std::cout << "ERROR: Failed to open file for writing:" << outFile.fileName().toStdString()
+ << std::endl;
+ return false;
+ }
+ if (outFile.write(doc.toJson(QJsonDocument::Indented)) == -1) {
+ std::cout << "ERROR: Failed writing into file :" << outFile.fileName().toStdString()
+ << std::endl;
+ return false;
+ }
+ if (!outFile.flush()) {
+ std::cout << "ERROR: Failed flushing the file :" << outFile.fileName().toStdString()
+ << std::endl;
+ return false;
+ }
+ outFile.close();
+ return true;
+}
+
+std::optional<QSet<PreloadEntry>> getPreloadsFromQmlImportScannerOutput(QString output)
+{
+ QString qtLibPath = "$QTDIR/lib";
+ QString qtQmlPath = "$QTDIR/qml";
+ QString qtDeployQmlPath = "/qt/qml";
+ QSet<PreloadEntry> res;
+ auto addImport = [&res](const PreloadEntry &entry) {
+ // qDebug() << "adding " << entry.source << "" << entry.destination;
+ res.insert(entry);
+ };
+
+ QJsonParseError parseError;
+ QJsonDocument doc = QJsonDocument::fromJson(output.toUtf8(), &parseError);
+
+ if (parseError.error != QJsonParseError::NoError) {
+ std::cout << "ERROR: QmlImport JSON parse error: " << parseError.errorString().toStdString()
+ << std::endl;
+ return std::nullopt;
+ }
+ if (!doc.isArray()) {
+ std::cout << "ERROR: QmlImport JSON is not an array." << std::endl;
+ return std::nullopt;
+ }
+
+ QJsonArray jsonArray = doc.array();
+ for (const QJsonValue &value : jsonArray) {
+ if (value.isObject()) {
+ QJsonObject obj = value.toObject();
+ auto relativePath = obj["relativePath"].toString();
+ auto plugin = obj["plugin"].toString();
+ if (plugin.isEmpty() || relativePath.isEmpty()) {
+ continue;
+ }
+ auto pluginFilename = "lib" + plugin + ".so";
+ addImport(PreloadEntry{
+ QDir::cleanPath(qtQmlPath + "/" + relativePath + "/" + pluginFilename),
+ QDir::cleanPath(qtDeployQmlPath + "/" + relativePath + "/" + pluginFilename) });
+ addImport(PreloadEntry{
+ QDir::cleanPath(qtQmlPath + "/" + relativePath + "/" + "qmldir"),
+ QDir::cleanPath(qtDeployQmlPath + "/" + relativePath + "/" + "qmldir") });
+ }
+ }
+
+ return res;
+}
+
+}; // namespace JsonTools
diff --git a/src/tools/wasmdeployqt/jsontools.h b/src/tools/wasmdeployqt/jsontools.h
new file mode 100644
index 00000000000..a1691a2be8d
--- /dev/null
+++ b/src/tools/wasmdeployqt/jsontools.h
@@ -0,0 +1,19 @@
+// Copyright (C) 2025 The Qt Company Ltd.
+// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR GPL-3.0-only WITH Qt-GPL-exception-1.0
+
+#ifndef JSONTOOLS_H
+#define JSONTOOLS_H
+
+#include <QFileInfo>
+#include <QSet>
+
+#include "common.h"
+
+#include <optional>
+
+namespace JsonTools {
+bool savePreloadFile(QSet<PreloadEntry> preload, QString destFile);
+std::optional<QSet<PreloadEntry>> getPreloadsFromQmlImportScannerOutput(QString output);
+}; // namespace JsonTools
+
+#endif
diff --git a/src/tools/wasmdeployqt/main.cpp b/src/tools/wasmdeployqt/main.cpp
new file mode 100644
index 00000000000..3bf2647cfaf
--- /dev/null
+++ b/src/tools/wasmdeployqt/main.cpp
@@ -0,0 +1,417 @@
+// Copyright (C) 2025 The Qt Company Ltd.
+// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR GPL-3.0-only WITH Qt-GPL-exception-1.0
+
+#include "common.h"
+#include "jsontools.h"
+#include "wasmbinary.h"
+
+#include <QCoreApplication>
+#include <QDir>
+#include <QDirListing>
+#include <QDirIterator>
+#include <QtGlobal>
+#include <QLibraryInfo>
+#include <QJsonDocument>
+#include <QStringList>
+#include <QtCore/QCommandLineOption>
+#include <QtCore/QCommandLineParser>
+#include <QtCore/QProcess>
+#include <QQueue>
+#include <QMap>
+#include <QSet>
+
+#include <optional>
+#include <iostream>
+#include <ostream>
+
+struct Parameters
+{
+ std::optional<QString> argAppPath;
+ QString appWasmPath;
+ std::optional<QDir> qtHostDir;
+ std::optional<QDir> qtWasmDir;
+ QList<QDir> libPaths;
+ std::optional<QDir> qmlRootPath;
+
+ QSet<QString> loadedQtLibraries;
+};
+
+bool parseArguments(Parameters &params)
+{
+ QCoreApplication::setApplicationName("wasmdeployqt");
+ QCoreApplication::setApplicationVersion("1.0");
+ QCommandLineParser parser;
+ parser.setApplicationDescription(
+ QStringLiteral("Qt for WebAssembly deployment tool \n\n"
+ "Example:\n"
+ "wasmdeployqt app.wasm --qml-root-path=repo/myapp "
+ "--qt-wasm-dir=/home/user/qt/shared-qt-wasm/bin"));
+ parser.addHelpOption();
+
+ QStringList args = QCoreApplication::arguments();
+
+ parser.addPositionalArgument("app", "Path to the application.");
+ QCommandLineOption libPathOption("lib-path", "Colon-separated list of library directories.",
+ "paths");
+ parser.addOption(libPathOption);
+ QCommandLineOption qtWasmDirOption("qt-wasm-dir", "Path to the Qt for WebAssembly directory.",
+ "dir");
+ parser.addOption(qtWasmDirOption);
+ QCommandLineOption qtHostDirOption("qt-host-dir", "Path to the Qt host directory.", "dir");
+ parser.addOption(qtHostDirOption);
+ QCommandLineOption qmlRootPathOption("qml-root-path", "Root directory for QML files.", "dir");
+ parser.addOption(qmlRootPathOption);
+ parser.process(args);
+
+ const QStringList positionalArgs = parser.positionalArguments();
+ if (positionalArgs.size() > 1) {
+ std::cout << "ERROR: Expected only one positional argument with path to the app. Received: "
+ << positionalArgs.join(" ").toStdString() << std::endl;
+ return false;
+ }
+ if (!positionalArgs.isEmpty()) {
+ params.argAppPath = positionalArgs.first();
+ }
+
+ if (parser.isSet(libPathOption)) {
+ QStringList paths = parser.value(libPathOption).split(';', Qt::SkipEmptyParts);
+ for (const QString &path : paths) {
+ QDir dir(path);
+ if (dir.exists()) {
+ params.libPaths.append(dir);
+ } else {
+ std::cout << "ERROR: Directory does not exist: " << path.toStdString() << std::endl;
+ return false;
+ }
+ }
+ }
+ if (parser.isSet(qtWasmDirOption)) {
+ QDir dir(parser.value(qtWasmDirOption));
+ if (dir.cdUp() && dir.exists())
+ params.qtWasmDir = dir;
+ else {
+ std::cout << "ERROR: Directory does not exist: " << dir.absolutePath().toStdString()
+ << std::endl;
+ return false;
+ }
+ }
+ if (parser.isSet(qtHostDirOption)) {
+ QDir dir(parser.value(qtHostDirOption));
+ if (dir.cdUp() && dir.exists())
+ params.qtHostDir = dir;
+ else {
+ std::cout << "ERROR: Directory does not exist: " << dir.absolutePath().toStdString()
+ << std::endl;
+ return false;
+ }
+ }
+ if (parser.isSet(qmlRootPathOption)) {
+ QDir dir(parser.value(qmlRootPathOption));
+ if (dir.exists()) {
+ params.qmlRootPath = dir;
+ } else {
+ std::cout << "ERROR: Directory specified for qml-root-path does not exist: "
+ << dir.absolutePath().toStdString() << std::endl;
+ return false;
+ }
+ }
+ return true;
+}
+
+std::optional<QString> detectAppName()
+{
+ QDirIterator it(QDir::currentPath(), QStringList() << "*.html" << "*.wasm" << "*.js",
+ QDir::NoFilter);
+ QMap<QString, QSet<QString>> fileGroups;
+ while (it.hasNext()) {
+ QFileInfo fileInfo(it.next());
+ QString baseName = fileInfo.completeBaseName();
+ QString suffix = fileInfo.suffix();
+ fileGroups[baseName].insert(suffix);
+ }
+ for (auto it = fileGroups.constBegin(); it != fileGroups.constEnd(); ++it) {
+ const QSet<QString> &extensions = it.value();
+ if (extensions.contains("html") && extensions.contains("js")
+ && extensions.contains("wasm")) {
+ return it.key();
+ }
+ }
+ return std::nullopt;
+}
+
+bool verifyPaths(Parameters &params)
+{
+ if (params.argAppPath) {
+ QFileInfo fileInfo(*params.argAppPath);
+ if (!fileInfo.exists()) {
+ std::cout << "ERROR: Cannot find " << params.argAppPath->toStdString() << std::endl;
+ std::cout << "Make sure that the path is valid." << std::endl;
+ return false;
+ }
+ params.appWasmPath = fileInfo.absoluteFilePath();
+ } else {
+ auto appName = detectAppName();
+ if (!appName) {
+ std::cout << "ERROR: Cannot find the application in current directory. Specify the "
+ "path as an argument:"
+ "wasmdeployqt <path-to-app-wasm-binary>"
+ << std::endl;
+ return false;
+ }
+ params.appWasmPath = QDir::current().filePath(*appName + ".wasm");
+ std::cout << "Automatically detected " << params.appWasmPath.toStdString() << std::endl;
+ }
+ if (!params.qtWasmDir) {
+ std::cout << "ERROR: Please set path to Qt WebAssembly installation as "
+ "--qt-wasm-dir=<path_to_qt_wasm_bin>"
+ << std::endl;
+ return false;
+ }
+ if (!params.qtHostDir) {
+ auto qtHostPath = QLibraryInfo::path(QLibraryInfo::BinariesPath);
+ if (qtHostPath.length() == 0) {
+ std::cout << "ERROR: Cannot read Qt host path or detect it from environment. Please "
+ "pass it explicitly with --qt-host-dir=<path>. "
+ << std::endl;
+ } else {
+ auto qtHostDir = QDir(qtHostPath);
+ if (!qtHostDir.cdUp()) {
+ std::cout << "ERROR: Invalid Qt host path: "
+ << qtHostDir.absolutePath().toStdString() << std::endl;
+ return false;
+ }
+ params.qtHostDir = qtHostDir;
+ }
+ }
+ params.libPaths.push_front(params.qtWasmDir->filePath("lib"));
+ params.libPaths.push_front(*params.qtWasmDir);
+ return true;
+}
+
+bool copyFile(QString srcPath, QString destPath)
+{
+ auto file = QFile(destPath);
+ if (file.exists()) {
+ file.remove();
+ }
+ QFileInfo destInfo(destPath);
+ if (!QDir().mkpath(destInfo.path())) {
+ std::cout << "ERROR: Cannot create path " << destInfo.path().toStdString() << std::endl;
+ return false;
+ }
+ if (!QFile::copy(srcPath, destPath)) {
+
+ std::cout << "ERROR: Failed to copy " << srcPath.toStdString() << " to "
+ << destPath.toStdString() << std::endl;
+
+ return false;
+ }
+ return true;
+}
+
+bool copyDirectDependencies(QList<QString> dependencies, const Parameters &params)
+{
+ for (auto &&depFilename : dependencies) {
+ if (params.loadedQtLibraries.contains(depFilename)) {
+ continue; // dont copy library that has been already copied
+ }
+
+ std::optional<QString> libPath;
+ for (auto &&libDir : params.libPaths) {
+ auto path = libDir.filePath(depFilename);
+ QFileInfo file(path);
+ if (file.exists()) {
+ libPath = path;
+ }
+ }
+ if (!libPath) {
+ std::cout << "ERROR: Cannot find required library " << depFilename.toStdString()
+ << std::endl;
+ return false;
+ }
+ if (!copyFile(*libPath, QDir::current().filePath(depFilename)))
+ return false;
+ }
+ std::cout << "INFO: Succesfully copied direct dependencies." << std::endl;
+ return true;
+}
+
+QStringList findSoFiles(const QString &directory)
+{
+ QStringList soFiles;
+ QDir baseDir(directory);
+ if (!baseDir.exists())
+ return soFiles;
+
+ QDirIterator it(directory, QStringList() << "*.so", QDir::Files, QDirIterator::Subdirectories);
+ while (it.hasNext()) {
+ it.next();
+ QString absPath = it.filePath();
+ QString filePath = baseDir.relativeFilePath(absPath);
+ soFiles.append(filePath);
+ }
+ return soFiles;
+}
+
+bool copyQtLibs(Parameters &params)
+{
+ Q_ASSERT(params.qtWasmDir);
+ auto qtLibDir = *params.qtWasmDir;
+ if (!qtLibDir.cd("lib")) {
+ std::cout << "ERROR: Cannot find lib directory in Qt installation." << std::endl;
+ return false;
+ }
+ auto qtLibTargetDir = QDir(QDir(QDir::current().filePath("qt")).filePath("lib"));
+
+ auto soFiles = findSoFiles(qtLibDir.absolutePath());
+ for (auto &&soFilePath : soFiles) {
+ auto relativeFilePath = QDir("lib").filePath(soFilePath);
+ auto srcPath = qtLibDir.absoluteFilePath(soFilePath);
+ auto destPath = qtLibTargetDir.absoluteFilePath(soFilePath);
+ if (!copyFile(srcPath, destPath))
+ return false;
+ params.loadedQtLibraries.insert(QFileInfo(srcPath).fileName());
+ }
+ std::cout << "INFO: Succesfully deployed qt lib shared objects." << std::endl;
+ return true;
+}
+
+bool copyPreloadPlugins(Parameters &params)
+{
+ Q_ASSERT(params.qtWasmDir);
+ auto qtPluginsDir = *params.qtWasmDir;
+ if (!qtPluginsDir.cd("plugins")) {
+ std::cout << "ERROR: Cannot find plugins directory in Qt installation." << std::endl;
+ return false;
+ }
+ auto qtPluginsTargetDir = QDir(QDir(QDir::current().filePath("qt")).filePath("plugins"));
+
+ // copy files
+ auto soFiles = findSoFiles(qtPluginsDir.absolutePath());
+ for (auto &&soFilePath : soFiles) {
+ auto relativeFilePath = QDir("plugins").filePath(soFilePath);
+ params.loadedQtLibraries.insert(QFileInfo(relativeFilePath).fileName());
+ auto srcPath = qtPluginsDir.absoluteFilePath(soFilePath);
+ auto destPath = qtPluginsTargetDir.absoluteFilePath(soFilePath);
+ if (!copyFile(srcPath, destPath))
+ return false;
+ }
+
+ // qt_plugins.json
+ QSet<PreloadEntry> preload{ { { "qt.conf" }, { "/qt.conf" } } };
+ for (auto &&plugin : soFiles) {
+ PreloadEntry entry;
+ entry.source = QDir("$QTDIR").filePath("plugins") + QDir::separator()
+ + QDir(qtPluginsDir).relativeFilePath(plugin);
+ entry.destination = "/qt/plugins/" + QDir(qtPluginsTargetDir).relativeFilePath(plugin);
+ preload.insert(entry);
+ }
+ JsonTools::savePreloadFile(preload, QDir::current().filePath("qt_plugins.json"));
+
+ QString qtconfContent = "[Paths]\nPrefix = /qt\n";
+ QString filePath = QDir::current().filePath("qt.conf");
+
+ QFile file(filePath);
+ if (file.open(QIODevice::WriteOnly | QIODevice::Text)) {
+ QTextStream out(&file);
+ out << qtconfContent;
+ if (!file.flush()) {
+ std::cout << "ERROR: Failed flushing the file :" << file.fileName().toStdString()
+ << std::endl;
+ return false;
+ }
+ file.close();
+ } else {
+ std::cout << "ERROR: Failed to write to qt.conf." << std::endl;
+ return false;
+ }
+ std::cout << "INFO: Succesfully deployed qt plugins." << std::endl;
+ return true;
+}
+
+bool copyPreloadQmlImports(Parameters &params)
+{
+ Q_ASSERT(params.qtWasmDir);
+ if (!params.qmlRootPath) {
+ std::cout << "WARNING: qml-root-path not specified. Skipping generating preloads for QML "
+ "imports."
+ << std::endl;
+ std::cout << "WARNING: This may lead to erronous behaviour if applications requires QML "
+ "imports."
+ << std::endl;
+ QSet<PreloadEntry> preload;
+ JsonTools::savePreloadFile(preload, QDir::current().filePath("qt_qml_imports.json"));
+ return true;
+ }
+ auto qmlImportScannerPath = params.qtHostDir
+ ? QDir(params.qtHostDir->filePath("libexec")).filePath("qmlimportscanner")
+ : "qmlimportscanner";
+ QProcess process;
+ auto qmlImportPath = *params.qtWasmDir;
+ qmlImportPath.cd("qml");
+ if (!qmlImportPath.exists()) {
+ std::cout << "ERROR: Cannot find qml import path: "
+ << qmlImportPath.absolutePath().toStdString() << std::endl;
+ return -1;
+ }
+
+ QStringList args{ "-rootPath", params.qmlRootPath->absolutePath(), "-importPath",
+ qmlImportPath.absolutePath() };
+ process.start(qmlImportScannerPath, args);
+ if (!process.waitForFinished()) {
+ std::cout << "ERROR: Failed to execute qmlImportScanner." << std::endl;
+ return false;
+ }
+
+ QString stdoutOutput = process.readAllStandardOutput();
+ auto qmlImports = JsonTools::getPreloadsFromQmlImportScannerOutput(stdoutOutput);
+ if (!qmlImports) {
+ return false;
+ }
+ JsonTools::savePreloadFile(*qmlImports, QDir::current().filePath("qt_qml_imports.json"));
+ for (const PreloadEntry &import : *qmlImports) {
+ auto relativePath = import.source;
+ relativePath.remove("$QTDIR/");
+
+ auto srcPath = params.qtWasmDir->absoluteFilePath(relativePath);
+ auto destPath = QDir(QDir::current().filePath("qt")).absoluteFilePath(relativePath);
+ if (!copyFile(srcPath, destPath))
+ return false;
+ }
+ std::cout << "INFO: Succesfully deployed qml imports." << std::endl;
+ return true;
+}
+
+int main(int argc, char **argv)
+{
+ QCoreApplication app(argc, argv);
+ Parameters params;
+ if (!parseArguments(params)) {
+ return -1;
+ }
+ if (!verifyPaths(params)) {
+ return -1;
+ }
+ std::cout << "INFO: Target: " << params.appWasmPath.toStdString() << std::endl;
+ WasmBinary wasmBinary(params.appWasmPath);
+ if (wasmBinary.type == WasmBinary::Type::INVALID) {
+ return -1;
+ } else if (wasmBinary.type == WasmBinary::Type::STATIC) {
+ std::cout << "INFO: This is statically linked WebAssembly binary." << std::endl;
+ std::cout << "INFO: No extra steps required!" << std::endl;
+ return 0;
+ }
+ std::cout << "INFO: Verified as shared module." << std::endl;
+
+ if (!copyQtLibs(params))
+ return -1;
+ if (!copyPreloadPlugins(params))
+ return -1;
+ if (!copyPreloadQmlImports(params))
+ return -1;
+ if (!copyDirectDependencies(wasmBinary.dependencies, params))
+ return -1;
+
+ std::cout << "INFO: Deployment done!" << std::endl;
+ return 0;
+}
diff --git a/src/tools/wasmdeployqt/wasmbinary.cpp b/src/tools/wasmdeployqt/wasmbinary.cpp
new file mode 100644
index 00000000000..1a041c94066
--- /dev/null
+++ b/src/tools/wasmdeployqt/wasmbinary.cpp
@@ -0,0 +1,91 @@
+// Copyright (C) 2025 The Qt Company Ltd.
+// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR GPL-3.0-only WITH Qt-GPL-exception-1.0
+
+#include "wasmbinary.h"
+
+#include <QFile>
+
+#include <iostream>
+
+WasmBinary::WasmBinary(QString filepath)
+{
+ QFile file(filepath);
+ if (!file.open(QIODevice::ReadOnly)) {
+ std::cout << "ERROR: Cannot open the file " << filepath.toStdString() << std::endl;
+ std::cout << file.errorString().toStdString() << std::endl;
+ type = WasmBinary::Type::INVALID;
+ return;
+ }
+ auto bytes = file.readAll();
+ if (!parsePreambule(bytes)) {
+ type = WasmBinary::Type::INVALID;
+ }
+}
+
+bool WasmBinary::parsePreambule(QByteArrayView data)
+{
+ const auto preambuleSize = 24;
+ if (data.size() < preambuleSize) {
+ std::cout << "ERROR: Preambule of binary shorter than expected!" << std::endl;
+ return false;
+ }
+ uint32_t int32View[6];
+ std::memcpy(int32View, data.data(), sizeof(int32View));
+ if (int32View[0] != 0x6d736100) {
+ std::cout << "ERROR: Magic WASM number not found in binary. Binary corrupted?" << std::endl;
+ return false;
+ }
+ if (data[8] != 0) {
+ type = WasmBinary::Type::STATIC;
+ return true;
+ } else {
+ type = WasmBinary::Type::SHARED;
+ }
+ const auto sectionStart = 9;
+ size_t offset = sectionStart;
+ auto sectionSize = getLeb(data, offset);
+ auto sectionEnd = sectionStart + sectionSize;
+ auto name = getString(data, offset);
+ if (name != "dylink.0") {
+ type = WasmBinary::Type::INVALID;
+ std::cout << "ERROR: dylink.0 was not found in supposedly dynamically linked module"
+ << std::endl;
+ return false;
+ }
+
+ const auto WASM_DYLINK_NEEDED = 0x2;
+ while (offset < sectionEnd) {
+ auto subsectionType = data[offset++];
+ auto subsectionSize = getLeb(data, offset);
+ if (subsectionType == WASM_DYLINK_NEEDED) {
+ auto neededDynlibsCount = getLeb(data, offset);
+ while (neededDynlibsCount--) {
+ dependencies.append(getString(data, offset));
+ }
+ } else {
+ offset += subsectionSize;
+ }
+ }
+ return true;
+}
+
+size_t WasmBinary::getLeb(QByteArrayView data, size_t &offset)
+{
+ auto ret = 0;
+ auto mul = 1;
+ while (true) {
+ auto byte = data[offset++];
+ ret += (byte & 0x7f) * mul;
+ mul *= 0x80;
+ if (!(byte & 0x80))
+ break;
+ }
+ return ret;
+}
+
+QString WasmBinary::getString(QByteArrayView data, size_t &offset)
+{
+ auto length = getLeb(data, offset);
+ offset += length;
+ return QString::fromUtf8(data.sliced(offset - length, length));
+}
diff --git a/src/tools/wasmdeployqt/wasmbinary.h b/src/tools/wasmdeployqt/wasmbinary.h
new file mode 100644
index 00000000000..c3bb3f0eaa4
--- /dev/null
+++ b/src/tools/wasmdeployqt/wasmbinary.h
@@ -0,0 +1,24 @@
+// Copyright (C) 2025 The Qt Company Ltd.
+// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR GPL-3.0-only WITH Qt-GPL-exception-1.0
+
+#ifndef WASMBINARY_H
+#define WASMBINARY_H
+
+#include <QString>
+#include <QList>
+
+class WasmBinary
+{
+public:
+ enum class Type { INVALID, STATIC, SHARED };
+ WasmBinary(QString filepath);
+ Type type;
+ QList<QString> dependencies;
+
+private:
+ bool parsePreambule(QByteArrayView data);
+ size_t getLeb(QByteArrayView data, size_t &offset);
+ QString getString(QByteArrayView data, size_t &offset);
+};
+
+#endif
diff --git a/tests/auto/corelib/io/qtemporaryfile/tst_qtemporaryfile.cpp b/tests/auto/corelib/io/qtemporaryfile/tst_qtemporaryfile.cpp
index 170c38d57c1..820a7b4bdb5 100644
--- a/tests/auto/corelib/io/qtemporaryfile/tst_qtemporaryfile.cpp
+++ b/tests/auto/corelib/io/qtemporaryfile/tst_qtemporaryfile.cpp
@@ -7,7 +7,6 @@
#include <qfile.h>
#include <qdatetime.h>
#include <qdir.h>
-#include <qset.h>
#include <qstandardpaths.h>
#include <qstring.h>
#include <qtemporarydir.h>
@@ -15,6 +14,7 @@
#include <QtTest/private/qtesthelpers_p.h>
+#include <QtCore/private/qduplicatetracker_p.h>
#include <QtCore/qscopeguard.h>
#if defined(Q_OS_WIN)
@@ -551,9 +551,9 @@ void tst_QTemporaryFile::openOnRootDrives()
void tst_QTemporaryFile::stressTest()
{
- const int iterations = 1000;
+ constexpr int iterations = 1000;
- QSet<QString> names;
+ QDuplicateTracker<QString, iterations> names;
const auto remover = qScopeGuard([&] {
for (const QString &s : std::as_const(names))
@@ -564,8 +564,7 @@ void tst_QTemporaryFile::stressTest()
QTemporaryFile file;
file.setAutoRemove(false);
QVERIFY2(file.open(), qPrintable(file.errorString()));
- QVERIFY(!names.contains(file.fileName()));
- names.insert(file.fileName());
+ QVERIFY(!names.hasSeen(file.fileName()));
}
}
diff --git a/tests/auto/corelib/serialization/qtextstream/tst_qtextstream.cpp b/tests/auto/corelib/serialization/qtextstream/tst_qtextstream.cpp
index 0254cbd1360..f03be6cc69e 100644
--- a/tests/auto/corelib/serialization/qtextstream/tst_qtextstream.cpp
+++ b/tests/auto/corelib/serialization/qtextstream/tst_qtextstream.cpp
@@ -23,6 +23,8 @@
#include "../../../network-settings.h"
#include <QtTest/private/qemulationdetector_p.h>
+using namespace Qt::StringLiterals;
+
QT_BEGIN_NAMESPACE
template<> struct QMetaTypeId<QIODevice::OpenModeFlag>
{ enum { Defined = 1 }; static inline int qt_metatype_id() { return QMetaType::Int; } };
@@ -1379,12 +1381,12 @@ void tst_QTextStream::pos2()
// ------------------------------------------------------------------------------
void tst_QTextStream::pos3LargeFile()
{
+ // NOTE: The unusual spacing is to ensure non-1-character whitespace.
+ constexpr auto lineString = " 0 1 2\t3 4\t \t5 6 7 8 9 \n"_L1;
{
QFile file(testFileName);
QVERIFY(file.open(QIODevice::WriteOnly | QIODevice::Text));
QTextStream out( &file );
- // NOTE: The unusual spacing is to ensure non-1-character whitespace.
- QString lineString = " 0 1 2\t3 4\t \t5 6 7 8 9 \n";
// Approximately 5kb text file (more is too slow (QTBUG-138435))
const int NbLines = (5 * 1024) / lineString.size() + 1;
for (int line = 0; line < NbLines; ++line)
@@ -1395,8 +1397,15 @@ void tst_QTextStream::pos3LargeFile()
QVERIFY(file.open(QIODevice::ReadOnly | QIODevice::Text));
QTextStream in( &file );
constexpr int testValues[] = { 0, 1, 2, 3, 4, 5, 6, 7, 8, 9 };
+ qint64 expectedLineEnd = 0;
+#ifdef Q_OS_WIN // CRLF platform
+ constexpr int crlfAdjustment = 1;
+#else
+ constexpr int crlfAdjustment = 0;
+#endif
+ const auto expectedLineLength = lineString.size() + crlfAdjustment;
+ QCOMPARE(in.pos(), 0);
while (true) {
- in.pos();
for (size_t i = 0; i < std::size(testValues); ++i) {
int value = -42;
if (!(in >> value)) {
@@ -1407,6 +1416,9 @@ void tst_QTextStream::pos3LargeFile()
}
QCOMPARE(value, testValues[i]);
}
+ expectedLineEnd += expectedLineLength;
+ // Final space and newline are not consumed until next read.
+ QCOMPARE(in.pos(), expectedLineEnd - 2 - crlfAdjustment);
}
}
@@ -2997,14 +3009,41 @@ void tst_QTextStream::int_write_with_locale_data()
QTest::addColumn<int>("numberFlags");
QTest::addColumn<int>("input");
QTest::addColumn<QString>("output");
+ QTest::addColumn<int>("fieldWidth");
+ QTest::addColumn<QTextStream::FieldAlignment>("fieldAlignment");
+
+ const auto alignDefault = QTextStream().fieldAlignment();
+ constexpr int forceSign = QTextStream::ForceSign;
+
+ QTest::newRow("C -123") << u"C"_s << 0 << -123 << u"-123"_s << 0 << alignDefault;
+ QTest::newRow("C +123") << u"C"_s << forceSign << 123 << u"+123"_s << 0 << alignDefault;
+ QTest::newRow("C 12345") << u"C"_s << 0 << 12345 << u"12345"_s << 0 << alignDefault;
- QTest::newRow("C -123") << QString("C") << 0 << -123 << QString("-123");
- QTest::newRow("C +123") << QString("C") << (int)QTextStream::ForceSign << 123 << QString("+123");
- QTest::newRow("C 12345") << QString("C") << 0 << 12345 << QString("12345");
+ QTest::newRow("de_DE -123") << u"de_DE"_s << 0 << -123 << u"-123"_s << 0 << alignDefault;
+ QTest::newRow("de_DE +123") << u"de_DE"_s << forceSign << 123 << u"+123"_s << 0 << alignDefault;
+ QTest::newRow("de_DE 12345") << u"de_DE"_s << 0 << 12345 << u"12.345"_s << 0 << alignDefault;
- QTest::newRow("de_DE -123") << QString("de_DE") << 0 << -123 << QString("-123");
- QTest::newRow("de_DE +123") << QString("de_DE") << (int)QTextStream::ForceSign << 123 << QString("+123");
- QTest::newRow("de_DE 12345") << QString("de_DE") << 0 << 12345 << QString("12.345");
+ constexpr auto alignAccountingStyle = QTextStream::FieldAlignment::AlignAccountingStyle;
+
+ {
+ const QLocale loc("ar_EG"_L1);
+ // Arabic as spoken in Egypt has a two-code-point negativeSign():
+ const auto minus = loc.negativeSign();
+ QCOMPARE(minus.size(), 2);
+ // ditto positiveSign():
+ const auto plus = loc.positiveSign();
+ QCOMPARE(plus.size(), 2);
+
+ QTest::addRow("ar_EG -123") << u"ar_EG"_s << 0 << -123
+ << (minus + u" ١٢٣")
+ << 10 << alignAccountingStyle;
+ QTest::newRow("ar_EG +123") << u"ar_EG"_s << forceSign << 123
+ << (plus + u" ١٢٣")
+ << 10 << alignAccountingStyle;
+ QTest::newRow("ar_EG 12345") << u"ar_EG"_s << 0 << 12345
+ << u" ١٢٬٣٤٥"_s
+ << 10 << alignAccountingStyle;
+ }
}
void tst_QTextStream::int_write_with_locale()
@@ -3013,13 +3052,21 @@ void tst_QTextStream::int_write_with_locale()
QFETCH(int, numberFlags);
QFETCH(int, input);
QFETCH(QString, output);
+ QFETCH(const int, fieldWidth);
+ QFETCH(const QTextStream::FieldAlignment, fieldAlignment);
QString result;
QTextStream stream(&result);
stream.setLocale(QLocale(locale));
+ stream.setFieldAlignment(fieldAlignment);
if (numberFlags)
stream.setNumberFlags(QTextStream::NumberFlags(numberFlags));
+ if (fieldWidth)
+ stream.setFieldWidth(fieldWidth);
+
QVERIFY(stream << input);
+ QEXPECT_FAIL("ar_EG -123", "QTBUG-138484", Continue);
+ QEXPECT_FAIL("ar_EG +123", "QTBUG-138484", Continue);
QCOMPARE(result, output);
}
diff --git a/tests/manual/wasm/eventloop/suspendresumecontrol_auto/main.cpp b/tests/manual/wasm/eventloop/suspendresumecontrol_auto/main.cpp
index 59f1f00cc59..1fd6e0a0c4f 100644
--- a/tests/manual/wasm/eventloop/suspendresumecontrol_auto/main.cpp
+++ b/tests/manual/wasm/eventloop/suspendresumecontrol_auto/main.cpp
@@ -7,7 +7,7 @@
using namespace emscripten;
-const int timerTimeout = 10;
+const std::chrono::milliseconds timerTimeout{10};
// Test QWasmSuspendResumeControl suspend/resume and event processing,
// via QWasmTimer native timer events.
diff --git a/util/wasm/preload/generate_default_preloads.sh.in b/util/wasm/preload/generate_default_preloads.sh.in
deleted file mode 100644
index b38d308fa45..00000000000
--- a/util/wasm/preload/generate_default_preloads.sh.in
+++ /dev/null
@@ -1,19 +0,0 @@
-#!/bin/bash
-
-TARGET_DIR="@TARGET_DIR@"
-SOURCE_DIR="@SOURCE_DIR@"
-QT_HOST_DIR="@QT_HOST_DIR@"
-QT_WASM_DIR="@QT_WASM_DIR@"
-QT_INSTALL_DIR="@QT_INSTALL_DIR@"
-
-python3 \
- "$QT_WASM_DIR/libexec/preload_qt_plugins.py" \
- "$QT_INSTALL_DIR" \
- "$TARGET_DIR"
-
-python3 \
- "$QT_WASM_DIR/libexec/preload_qml_imports.py" \
- "$SOURCE_DIR" \
- "$QT_HOST_DIR" \
- "$QT_INSTALL_DIR" \
- "$TARGET_DIR"
diff --git a/util/wasm/preload/preload_qml_imports.py b/util/wasm/preload/preload_qml_imports.py
deleted file mode 100755
index b78ef5ee744..00000000000
--- a/util/wasm/preload/preload_qml_imports.py
+++ /dev/null
@@ -1,98 +0,0 @@
-#!/usr/bin/env python3
-# Copyright (C) 2023 The Qt Company Ltd.
-# SPDX-License-Identifier: LicenseRef-Qt-Commercial OR GPL-3.0-only WITH Qt-GPL-exception-1.0
-
-import os
-import sys
-import subprocess
-import json
-
-# Paths to shared libraries and qml imports on the Qt installation on the web server.
-# "$QTDIR" is replaced by qtloader.js at load time (defaults to "qt"), and makes
-# possible to relocate the application build relative to the Qt build on the web server.
-qt_lib_path = "$QTDIR/lib"
-qt_qml_path = "$QTDIR/qml"
-
-# Path to QML imports on the in-memory file system provided by Emscripten. This script emits
-# preload commands which copies QML imports to this directory. In addition, preload_qt_plugins.py
-# creates (and preloads) a qt.conf file which makes Qt load QML plugins from this location.
-qt_deploy_qml_path = "/qt/qml"
-
-
-def preload_file(source, destination):
- preload_files.append({"source": source, "destination": destination})
-
-
-def extract_preload_files_from_imports(imports):
- libraries = []
- for qml_import in imports:
- try:
- relative_path = qml_import["relativePath"]
- plugin = qml_import["plugin"]
-
- # plugin .so
- plugin_filename = "lib" + plugin + ".so"
- so_plugin_source_path = os.path.join(
- qt_qml_path, relative_path, plugin_filename
- )
- so_plugin_destination_path = os.path.join(
- qt_deploy_qml_path, relative_path, plugin_filename
- )
-
- preload_file(so_plugin_source_path, so_plugin_destination_path)
- so_plugin_qt_install_path = os.path.join(
- qt_wasm_path, "qml", relative_path, plugin_filename
- )
-
- # qmldir file
- qmldir_source_path = os.path.join(qt_qml_path, relative_path, "qmldir")
- qmldir_destination_path = os.path.join(
- qt_deploy_qml_path, relative_path, "qmldir"
- )
- preload_file(qmldir_source_path, qmldir_destination_path)
- except Exception as e:
- continue
- return libraries
-
-
-if __name__ == "__main__":
- if len(sys.argv) != 5:
- print("Usage: python preload_qml_imports.py <qml-source-path> <qt-host-path> <qt-wasm-path> <output-dir>")
- sys.exit(1)
-
- qml_source_path = sys.argv[1]
- qt_host_path = sys.argv[2]
- qt_wasm_path = sys.argv[3]
- output_dir = sys.argv[4]
-
- qml_import_path = os.path.join(qt_wasm_path, "qml")
- qmlimportsscanner_path = os.path.join(qt_host_path, "libexec/qmlimportscanner")
-
- command = [qmlimportsscanner_path, "-rootPath", qml_source_path, "-importPath", qml_import_path]
- result = subprocess.run(command, stdout=subprocess.PIPE)
- imports = json.loads(result.stdout)
-
- preload_files = []
- libraries = extract_preload_files_from_imports(imports)
-
- # Deploy plugin dependencies, that is, shared libraries used by the plugins.
- # Skip some of the obvious libraries which will be
- skip_libraries = [
- "libQt6Core.so",
- "libQt6Gui.so",
- "libQt6Quick.so",
- "libQt6Qml.so" "libQt6Network.so",
- "libQt6OpenGL.so",
- ]
-
- libraries = set(libraries) - set(skip_libraries)
- for library in libraries:
- source = os.path.join(qt_lib_path, library)
- # Emscripten looks for shared libraries on "/", shared libraries
- # most be deployed there instead of at /qt/lib
- destination = os.path.join("/", library)
- preload_file(source, destination)
-
- with open(f"{output_dir}/qt_qml_imports.json", "w") as f:
- f.write(json.dumps(preload_files, indent=2))
-
diff --git a/util/wasm/preload/preload_qt_plugins.py b/util/wasm/preload/preload_qt_plugins.py
deleted file mode 100755
index 4b9b3683a70..00000000000
--- a/util/wasm/preload/preload_qt_plugins.py
+++ /dev/null
@@ -1,57 +0,0 @@
-#!/usr/bin/env python3
-# Copyright (C) 2023 The Qt Company Ltd.
-# SPDX-License-Identifier: LicenseRef-Qt-Commercial OR GPL-3.0-only WITH Qt-GPL-exception-1.0
-
-import os
-import sys
-import json
-
-# Path to plugins on the Qt installation on the web server. "$QTPATH" is replaced by qtloader.js
-# at load time (defaults to "qt"), which makes it possible to relocate the application build relative
-# to the Qt build on the web server.
-qt_plugins_path = "$QTDIR/plugins"
-
-# Path to plugins on the in-memory file system provided by Emscripten. This script emits
-# preload commands which copies plugins to this directory.
-qt_deploy_plugins_path = "/qt/plugins"
-
-
-def find_so_files(directory):
- so_files = []
- for root, dirs, files in os.walk(directory):
- for file in files:
- if file.endswith(".so"):
- relative_path = os.path.relpath(os.path.join(root, file), directory)
- so_files.append(relative_path)
- return so_files
-
-
-if __name__ == "__main__":
- if len(sys.argv) != 3:
- print("Usage: python preload_qt_plugins.py <qt-wasm-path> <output-dir>")
- sys.exit(1)
-
- qt_wasm_path = sys.argv[1]
- output_dir = sys.argv[2]
-
- # preload all plugins
- plugins = find_so_files(os.path.join(qt_wasm_path, "plugins"))
- preload = [
- {
- "source": os.path.join(qt_plugins_path, plugin),
- "destination": os.path.join(qt_deploy_plugins_path, plugin),
- }
- for plugin in plugins
- ]
-
- # Create and preload qt.conf which will tell Qt to look for plugins
- # and QML imports in /qt/plugins and /qt/qml. The qt.conf file is
- # written to the current directory.
- qtconf = "[Paths]\nPrefix = /qt\n"
- with open(f"{output_dir}/qt.conf", "w") as f:
- f.write(qtconf)
- preload.append({"source": "qt.conf", "destination": "/qt.conf"})
-
- with open(f"{output_dir}/qt_plugins.json", "w") as f:
- f.write(json.dumps(preload, indent=2))
-