Qt
Internal/Contributor docs for the Qt SDK. Note: These are NOT official API docs; those are found at https://doc.qt.io/
Loading...
Searching...
No Matches
qohosclipboardobject.cpp
Go to the documentation of this file.
1// Copyright (C) 2025 The Qt Company Ltd.
2// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR LGPL-3.0-only OR GPL-2.0-only OR GPL-3.0-only
3
4#include <qohosclipboardobject.h>
5
6#include <QtCore/private/qohoscommon_p.h>
7#include <QtCore/private/qohoslogger_p.h>
8#include <QtCore/qurl.h>
9#include <database/pasteboard/oh_pasteboard.h>
10#include <database/pasteboard/oh_pasteboard_err_code.h>
11#include <database/udmf/udmf.h>
12#include <database/udmf/uds.h>
13#include <memory>
14#include <multimedia/image_framework/image/image_common.h>
15#include <multimedia/image_framework/image/pixelmap_native.h>
16#include <qarkui/qarkuiutils.h>
17#include <qohosapppermissions_p.h>
18#include <qohosplugincore.h>
19#include <qohosudmfconversions.h>
20#include <qohosutils.h>
21
23
24namespace {
25
26const auto mimeTextPlain = QString::fromUtf8("text/plain");
27const auto mimeTextHtml = QString::fromUtf8("text/html");
28const auto mimeTextUriList = QString::fromUtf8("text/uri-list");
29const auto mimeAppXQtImage = QString::fromUtf8("application/x-qt-image");
30
35
36const char *getPasteboardNotifyTypeAsString(::Pasteboard_NotifyType notifyType)
37{
38 switch (notifyType) {
39 case ::Pasteboard_NotifyType::NOTIFY_LOCAL_DATA_CHANGE:
40 return "NOTIFY_LOCAL_DATA_CHANGE";
41 case ::Pasteboard_NotifyType::NOTIFY_REMOTE_DATA_CHANGE:
42 return "NOTIFY_REMOTE_DATA_CHANGE";
43 }
44 return "<illegal-Pasteboard_NotifyType>";
45}
46
48{
49 switch (dataSource) {
51 return "OurProcess";
53 return "OtherProcess";
54 }
55 return "<illegal-PasteboardDataSource>";
56}
57
59{
60 return std::shared_ptr<::OH_PasteboardObserver>(
61 QArkUi::callArkUiOrFailOnNullResult(
62 Q_OHOS_NAMED_FUNC(::OH_PasteboardObserver_Create)),
63 [](auto *pasteboardDataObserver) {
64 int observerDestroyRes = QArkUi::callArkUi(
65 Q_OHOS_NAMED_FUNC(::OH_PasteboardObserver_Destroy),
66 pasteboardDataObserver);
67 if (observerDestroyRes != ::PASTEBOARD_ErrCode::ERR_OK) {
68 qOhosPrintfError(
69 "%s: failed at destroying pasteboard observer.", Q_FUNC_INFO);
70 }
71 });
72}
73
75 std::shared_ptr<::OH_Pasteboard> pasteboard,
76 std::shared_ptr<::OH_PasteboardObserver> pasteboardObserver,
77 ::Pasteboard_NotifyType notifyType,
78 std::function<void()> dataChangedFunc)
79{
80 QArkUi::callArkUiOrFailOnErrorResult(
81 Q_OHOS_NAMED_FUNC(::OH_Pasteboard_Subscribe),
82 pasteboard.get(), notifyType, pasteboardObserver.get());
83
84 auto sharedDataChangedFunc = QtOhos::moveToSharedPtr(std::move(dataChangedFunc));
85
86 auto subscriptionHandle = QtOhos::makeDestroyNotifier(
87 [pasteboard, notifyType, pasteboardObserver, sharedDataChangedFunc]() {
88 int unsubscribeRes = QArkUi::callArkUi(
89 Q_OHOS_NAMED_FUNC(::OH_Pasteboard_Unsubscribe),
90 pasteboard.get(), notifyType, pasteboardObserver.get());
91 if (unsubscribeRes != ::PASTEBOARD_ErrCode::ERR_OK) {
92 qOhosPrintfError(
93 "%s: failed at unsubscribing from pasteboard %s.", Q_FUNC_INFO,
94 getPasteboardNotifyTypeAsString(notifyType));
95 }
96 });
97
98 QArkUi::callArkUiOrFailOnErrorResult(
99 Q_OHOS_NAMED_FUNC(::OH_PasteboardObserver_SetData),
100 pasteboardObserver.get(), sharedDataChangedFunc.get(),
101 [](void *ctx, ::Pasteboard_NotifyType) {
102 auto *dataChangedFunc = reinterpret_cast<std::function<void()> *>(ctx);
103 (*dataChangedFunc)();
104 },
105 [](void *) {});
106
107 return subscriptionHandle;
108}
109
111 std::shared_ptr<::OH_Pasteboard> pasteboard, std::function<void()> dataChangedListener,
112 std::vector<::Pasteboard_NotifyType> monitoredNotifyTypes)
113{
114 std::vector<std::shared_ptr<void>> subscriptionHandles;
115 auto sharedDataChangedFunc = QtOhos::moveToSharedPtr(std::move(dataChangedListener));
116
117 for (auto notifyType : monitoredNotifyTypes) {
118 auto observer = createPasteboardDataObserver();
119 subscriptionHandles.push_back(
120 subscribePasteboardObserver(
121 pasteboard, observer, notifyType,
122 [sharedDataChangedFunc]() {
123 (*sharedDataChangedFunc)();
124 }));
125 }
126
127 return QtOhos::moveToSharedPtr(std::move(subscriptionHandles));
128}
129
131{
132 if (!QArkUi::callArkUi(Q_OHOS_NAMED_FUNC(::OH_Pasteboard_HasData), pasteboard))
133 return nullptr;
134
135 int res;
136 auto *udmfDataPtr = QArkUi::callArkUi(
137 Q_OHOS_NAMED_FUNC(::OH_Pasteboard_GetData),
138 pasteboard, &res);
139
140 if (res != ::PASTEBOARD_ErrCode::ERR_OK || udmfDataPtr == nullptr) {
141 qOhosPrintfError("%s: error reading pasteboard data: %d / %p", Q_FUNC_INFO, res, udmfDataPtr);
142 return nullptr;
143 }
144
145 return std::make_unique<QOhosUdmfData>(QOhosUdmfData::takeOwnership(udmfDataPtr));
146}
147
148}
149
151 std::function<void(QOhosOptional<PasteboardDataSource>)> &&pasteboardUpdatesNotifier)
152{
154 [&](QtOhos::JsState &) {
155 m_pasteboard = std::shared_ptr<::OH_Pasteboard>(
156 QArkUi::callArkUiOrFailOnNullResult(
157 Q_OHOS_NAMED_FUNC(::OH_Pasteboard_Create)),
158 [](auto *pasteboard) {
159 QArkUi::callArkUi(Q_OHOS_NAMED_FUNC(::OH_Pasteboard_Destroy), pasteboard);
160 });
161 auto sharedPasteboardUpdatesNotifier = QtOhos::moveToSharedPtr(
162 std::move(pasteboardUpdatesNotifier));
163
164 auto pasteboardDataChangedJsThreadListener = std::make_shared<std::function<void()>>(
165 [this, weakPasteboardUpdatesNotifier = QtOhos::makeWeakPtr(sharedPasteboardUpdatesNotifier)]() {
166 auto optPasteboardUdmfData = tryGetUdmfDataFromPasteboard(m_pasteboard.get());
167 auto pasteboardDataSource =
168 optPasteboardUdmfData
169 ? makeQOhosOptional(
170 isQOhosUdmfDataConvertedFromThisProcessMimeData(*optPasteboardUdmfData)
174
175 qOhosPrintfDebug(
176 "%s: Pasteboard data source: %s",
177 Q_FUNC_INFO,
178 qTransform(pasteboardDataSource, &getPasteboardDataSourceAsString).value_or("<empty>"));
179
180 QtOhos::invokeInQtThread(
181 [weakPasteboardUpdatesNotifier, pasteboardDataSource]() {
182 auto sharedPasteboardUpdatesNotifier = weakPasteboardUpdatesNotifier.lock();
183 if (sharedPasteboardUpdatesNotifier)
184 (*sharedPasteboardUpdatesNotifier)(pasteboardDataSource);
185 });
186 });
187
188 m_pasteboardDataChangedListenerHandle = QtOhos::makeSharedPtrWithAttachedExtraData(
189 addPasteboardDataChangedListener(
190 m_pasteboard,
191 [pasteboardDataChangedJsThreadListener]() {
192 auto __dbg = make_QCScopedDebugJS(Q_FUNC_INFO);
193 QtOhos::invokeInJsThread(
194 [pasteboardDataChangedJsThreadListener](auto &) {
195 (*pasteboardDataChangedJsThreadListener)();
196 });
197 },
198 {
199 ::Pasteboard_NotifyType::NOTIFY_LOCAL_DATA_CHANGE,
200 ::Pasteboard_NotifyType::NOTIFY_REMOTE_DATA_CHANGE,
201 }),
202 sharedPasteboardUpdatesNotifier);
203 },
204 Q_FUNC_INFO);
205}
206
207std::unique_ptr<QOhosClipboardObject> QOhosClipboardObject::makeInstance(
208 std::function<void(QOhosOptional<QOhosClipboardObject::PasteboardDataSource>)> &&pasteboardUpdatesNotifier)
209{
210 return std::unique_ptr<QOhosClipboardObject>(
211 new QOhosClipboardObject(std::move(pasteboardUpdatesNotifier)));
212}
213
215{
216 QOhosOptional<PasteboardDataSource> dataSource;
217 QOhosSupplier<std::unique_ptr<QMimeData>> mimeDataFactory;
218 std::tie(dataSource, mimeDataFactory) = QtOhos::evalInJsThreadWithPromise<std::pair<QOhosOptional<PasteboardDataSource>, QOhosSupplier<std::unique_ptr<QMimeData>>>>(
219 [&](QtOhos::JsState &jsState, auto evalPromise) {
220 static constexpr const char *ohosGetPasteboardDataPermission = "ohos.permission.READ_PASTEBOARD";
221 auto sharedEvalPromise = QtOhos::moveToSharedPtr(std::move(evalPromise).makeChained(Q_FUNC_INFO));
222 QOhosAppPermissions::checkAppPermissionGrantedWithConsumer(
223 jsState, ohosGetPasteboardDataPermission,
224 [this, sharedEvalPromise](auto &, bool permissionGranted) {
225 if (!permissionGranted) {
226 qOhosPrintfError(
227 "%s: %s hasn't been granted by user. Cannot read pasteboard data.", Q_FUNC_INFO,
228 ohosGetPasteboardDataPermission);
229 (*sharedEvalPromise)({{}, std::make_unique<QMimeData>});
230 return;
231 }
232
233 auto optPasteboardUdmfData = tryGetUdmfDataFromPasteboard(m_pasteboard.get());
234 if (!optPasteboardUdmfData) {
235 (*sharedEvalPromise)({{}, std::make_unique<QMimeData>});
236 return;
237 }
238
239 (*sharedEvalPromise)(
240 {
241 makeQOhosOptional(
242 isQOhosUdmfDataConvertedFromThisProcessMimeData(*optPasteboardUdmfData)
243 ? PasteboardDataSource::OurProcess
244 : PasteboardDataSource::OtherProcess),
245 makeLazyFetchingQMimeDataFactoryFromUdmfData(std::move(*optPasteboardUdmfData))
246 });
247 });
248 },
249 Q_FUNC_INFO);
250
251 return PasteboardData{
252 .dataSource = dataSource,
253 .lazyFetchingData = mimeDataFactory(),
254 };
255}
256
258 std::shared_ptr<QMimeData> mimeData, const QOhosOptional<bool> &shareInAppOnly)
259{
260 if (!mimeData)
261 return;
262
263 auto udmfDataFactory = makeLazyProcessingUdmfDataFactoryFromQMimeData(mimeData, shareInAppOnly);
264
266 [&](QtOhos::JsState &) {
267 auto udmfData = udmfDataFactory();
268
269 int res = QArkUi::callArkUi(
270 Q_OHOS_NAMED_FUNC(::OH_Pasteboard_SetData),
271 m_pasteboard.get(), udmfData.nativePtr());
272 if (res != ::PASTEBOARD_ErrCode::ERR_OK) {
273 qOhosPrintfError("%s: cannot set data for pasteboard.", Q_FUNC_INFO);
274 return;
275 }
276 },
277 Q_FUNC_INFO);
278}
279
280QT_END_NAMESPACE
void setMimeDataSync(std::shared_ptr< QMimeData > mimeData, const QOhosOptional< bool > &shareInAppOnly)
QOhosClipboardObject(std::function< void(QOhosOptional< PasteboardDataSource >)> &&pasteboardUpdatesNotifier)
PasteboardData getPasteboardDataWithLazyFetch()
Combined button and popup list for selecting options.
std::shared_ptr<::OH_PasteboardObserver > createPasteboardDataObserver()
std::shared_ptr< void > subscribePasteboardObserver(std::shared_ptr<::OH_Pasteboard > pasteboard, std::shared_ptr<::OH_PasteboardObserver > pasteboardObserver, ::Pasteboard_NotifyType notifyType, std::function< void()> dataChangedFunc)
const char * getPasteboardDataSourceAsString(QOhosClipboardObject::PasteboardDataSource dataSource)
std::shared_ptr< void > addPasteboardDataChangedListener(std::shared_ptr<::OH_Pasteboard > pasteboard, std::function< void()> dataChangedListener, std::vector<::Pasteboard_NotifyType > monitoredNotifyTypes)
const char * getPasteboardNotifyTypeAsString(::Pasteboard_NotifyType notifyType)
std::unique_ptr< QOhosUdmfData > tryGetUdmfDataFromPasteboard(::OH_Pasteboard *pasteboard)
void runInJsThreadAndWait(const std::function< void(JsState &)> &task, std::string callerContextName={})
std::nullopt_t makeEmptyQOhosOptional()
std::weak_ptr< QOhosClipboardObject > weakThis