@react-native-ohos/clipboard 1.2.0-beta.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (33) hide show
  1. package/LICENSE +21 -0
  2. package/README.OpenSource +11 -0
  3. package/README.md +13 -0
  4. package/RNCClipboard.podspec +37 -0
  5. package/dist/Clipboard.d.ts +159 -0
  6. package/dist/Clipboard.js +232 -0
  7. package/dist/index.d.ts +3 -0
  8. package/dist/index.js +7 -0
  9. package/harmony/clipboard/LICENSE +21 -0
  10. package/harmony/clipboard/NOTICE +33 -0
  11. package/harmony/clipboard/OAT.xml +38 -0
  12. package/harmony/clipboard/README.OpenSource +11 -0
  13. package/harmony/clipboard/README.md +307 -0
  14. package/harmony/clipboard/build-profile.json5 +8 -0
  15. package/harmony/clipboard/hvigorfile.ts +1 -0
  16. package/harmony/clipboard/index.ets +27 -0
  17. package/harmony/clipboard/oh-package.json5 +12 -0
  18. package/harmony/clipboard/src/main/cpp/CMakeLists.txt +7 -0
  19. package/harmony/clipboard/src/main/cpp/ClipboardPackage.h +58 -0
  20. package/harmony/clipboard/src/main/cpp/RNCClipboardTurboModule.cpp +198 -0
  21. package/harmony/clipboard/src/main/cpp/RNCClipboardTurboModule.h +39 -0
  22. package/harmony/clipboard/src/main/ets/ClipboardPackage.ets +46 -0
  23. package/harmony/clipboard/src/main/ets/Logger.ts +64 -0
  24. package/harmony/clipboard/src/main/ets/RNCClipboardTurboModule.ts +759 -0
  25. package/harmony/clipboard/src/main/module.json5 +7 -0
  26. package/harmony/clipboard/src/main/resources/base/element/string.json +8 -0
  27. package/harmony/clipboard/src/main/resources/en_US/element/string.json +8 -0
  28. package/harmony/clipboard/src/main/resources/zh_CN/element/string.json +8 -0
  29. package/harmony/clipboard/ts.ets +26 -0
  30. package/harmony/clipboard.har +0 -0
  31. package/package.json +96 -0
  32. package/src/Clipboard.ts +211 -0
  33. package/src/index.ts +3 -0
@@ -0,0 +1,759 @@
1
+ /**
2
+ * MIT License
3
+ *
4
+ * Copyright (C) 2025 Huawei Device Co., Ltd.
5
+ *
6
+ * Permission is hereby granted, free of charge, to any person obtaining a copy
7
+ * of this software and associated documentation files (the "Software"), to deal
8
+ * in the Software without restriction, including without limitation the rights
9
+ * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
10
+ * copies of the Software, and to permit persons to whom the Software is
11
+ * furnished to do so, subject to the following conditions:
12
+ *
13
+ * The above copyright notice and this permission notice shall be included in all
14
+ * copies or substantial portions of the Software.
15
+ *
16
+ * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
17
+ * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
18
+ * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
19
+ * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
20
+ * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
21
+ * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
22
+ * SOFTWARE.
23
+ */
24
+
25
+ import { TurboModule, TurboModuleContext } from '@rnoh/react-native-openharmony/ts';
26
+ import { pasteboard, BusinessError } from '@kit.BasicServicesKit';
27
+ import util from '@ohos.util';
28
+ import image from '@ohos.multimedia.image';
29
+ import logger from './Logger';
30
+ import abilityAccessCtrl, { Permissions } from '@ohos.abilityAccessCtrl';
31
+ import url from '@ohos.url';
32
+
33
+ const TAG = "RNCClipboardTurboModule"
34
+ const prefixPNG = "data:image/png;base64,"
35
+ const prefixJPG = "data:image/jpg;base64,"
36
+ const imageMimePrefix = "image/"
37
+ const defaultImageMimeType = "image/png"
38
+ const PERMISSIONS: Array<Permissions> = [
39
+ 'ohos.permission.READ_PASTEBOARD'
40
+ ]
41
+
42
+ export class RNCClipboardTurboModule extends TurboModule {
43
+ constructor(protected ctx: TurboModuleContext) {
44
+ super(ctx);
45
+ logger.debug(TAG, "RNCClipboardTurboModule constructor");
46
+ }
47
+
48
+ getConstants() {
49
+ logger.debug(TAG, "RNCClipboardTurboModule call getConstants");
50
+ return {};
51
+ }
52
+
53
+ getString(): Promise<string> {
54
+ return new Promise<string>((resolve, reject) => {
55
+ this.requestPermission().then(res => {
56
+ if (res) {
57
+ let systemPasteboard = pasteboard.getSystemPasteboard();
58
+ systemPasteboard.getData().then((pasteData) => {
59
+ let text = pasteData.getPrimaryText();
60
+ logger.debug(TAG, `getString,text out:${text}`);
61
+ resolve(text);
62
+ }).catch((err) => {
63
+ logger.error(TAG, `getString,Failed to get PasteData. Cause:${err.message}`);
64
+ reject(err);
65
+ })
66
+ } else {
67
+ reject({
68
+ code: 0,
69
+ message: "User refuses authorization"
70
+ })
71
+ }
72
+ })
73
+ });
74
+ }
75
+
76
+ getStrings(): Promise<string[]> {
77
+ return new Promise<string[]>((resolve, reject) => {
78
+ this.requestPermission().then(res => {
79
+ if (res) {
80
+ logger.debug(TAG, "call getStrings fun");
81
+ let systemPasteboard = pasteboard.getSystemPasteboard();
82
+ systemPasteboard.getData().then((pasteData) => {
83
+ let count = pasteData.getRecordCount();
84
+ let resultSet = []
85
+ logger.debug(TAG, `getStrings fun,getRecordCount :${count}`);
86
+ for (let index = 0; index < count; index++) {
87
+ let record = pasteData.getRecord(index)
88
+ if (record.mimeType == pasteboard.MIMETYPE_TEXT_PLAIN) {
89
+ resultSet.push(record.plainText)
90
+ }
91
+ }
92
+ resolve(resultSet)
93
+ }).catch((err) => {
94
+ logger.error(TAG, `getString fun,Failed to get PasteData. Cause:${err.message}`);
95
+ reject(err)
96
+ })
97
+ } else {
98
+ reject({
99
+ code: 0,
100
+ message: "User refuses authorization"
101
+ })
102
+ }
103
+ })
104
+ });
105
+ }
106
+
107
+ setString(content: string) {
108
+ logger.debug(TAG, "setString fun");
109
+ let systemPasteboard = pasteboard.getSystemPasteboard();
110
+ let dataText = content;
111
+ let pasteData = pasteboard.createData(pasteboard.MIMETYPE_TEXT_PLAIN, dataText);
112
+ systemPasteboard.setData(pasteData).then((data) => {
113
+ logger.debug(TAG, "setString fun,Succeeded PasteData");
114
+ }).catch((err) => {
115
+ logger.error(TAG, `setString fun,Failed to set PasteData.setString, Cause:${err.message}`);
116
+ });
117
+ return;
118
+ }
119
+
120
+ hasString(): Promise<boolean> {
121
+ return new Promise<boolean>((resolve, reject) => {
122
+ this.requestPermission().then(res => {
123
+ if (res) {
124
+ logger.debug(TAG, "RNCClipboardTurboModule call hasString fun");
125
+ let systemPasteboard = pasteboard.getSystemPasteboard();
126
+ systemPasteboard.getData().then((pasteData) => {
127
+ let count = pasteData.getRecordCount();
128
+ resolve(count > 0)
129
+ }).catch((err) => {
130
+ logger.error(TAG, `RNCClipboardTurboModule hasString,Failed to get PasteData. Cause:${err.message}`);
131
+ reject(err)
132
+ })
133
+ } else {
134
+ reject({
135
+ code: 0,
136
+ message: "User refuses authorization"
137
+ })
138
+ }
139
+ })
140
+ });
141
+ }
142
+
143
+ hasNumber(): Promise<boolean> {
144
+ return new Promise<boolean>((resolve, reject) => {
145
+ this.requestPermission().then(res => {
146
+ if (res) {
147
+ logger.debug(TAG, "[RNOH]:RNCClipboardTurboModule call hasNumber");
148
+ let systemPasteboard = pasteboard.getSystemPasteboard();
149
+ let reg = /^[\d]*$/;
150
+ systemPasteboard.getData().then((pasteData) => {
151
+ let count = pasteData.getRecordCount();
152
+ let result = false
153
+ for (let index = 0; index < count; index++) {
154
+ let record = pasteData.getRecord(index);
155
+ if (record.mimeType == pasteboard.MIMETYPE_TEXT_PLAIN && reg.test(record.plainText)) {
156
+ result = true
157
+ logger.debug(TAG, `[RNOH]:RNCClipboardTurboModule hasNumber,result out:${result}`);
158
+ break
159
+ }
160
+ }
161
+ resolve(result)
162
+ }).catch((err) => {
163
+ logger.error(TAG, `[RNOH]: hasNumber,Failed to get PasteData. Cause:${err.message}`);
164
+ reject(err)
165
+ })
166
+ } else {
167
+ reject({
168
+ code: 0,
169
+ message: "User refuses authorization"
170
+ })
171
+ }
172
+ })
173
+ });
174
+ }
175
+
176
+ // ios
177
+ getImagePNG(): Promise<string> {
178
+ return new Promise<string>((resolve, reject) => {
179
+ this.requestPermission().then(res => {
180
+ if (res) {
181
+ logger.debug(TAG, "[RNOH]:RNCClipboardTurboModule call getImagePNG");
182
+ let systemPasteboard = pasteboard.getSystemPasteboard();
183
+
184
+ systemPasteboard.getData().then((pasteData) => {
185
+ let pixMap = pasteData.getPrimaryPixelMap();
186
+ if (!pixMap) {
187
+ logger.error(TAG, "[RNOH]:RNCClipboardTurboModule call getImagePNG,pixMap is null or undefined");
188
+ reject({
189
+ code: 1,
190
+ message: "Clipboard does not contain an image"
191
+ });
192
+ return;
193
+ }
194
+ const imagePackerApi = image.createImagePacker();
195
+ let packOpt: image.PackingOption = {
196
+ format: "image/png",
197
+ quality: 96
198
+ }
199
+ imagePackerApi.packing(pixMap, packOpt).then(data => {
200
+ let uint8Array = new Uint8Array(data);
201
+ let base64Helper = new util.Base64Helper();
202
+ let base64Str = base64Helper.encodeToStringSync(uint8Array, util.Type.BASIC)
203
+ let finalStr = prefixPNG + base64Str
204
+ resolve(finalStr)
205
+ }).catch((err) => {
206
+ logger.error(TAG, "[RNOH]:RNCClipboardTurboModule call getImagePNG,failed to packing");
207
+ reject(err)
208
+ }).finally(() => {
209
+ imagePackerApi.release().then(() => {
210
+ logger.debug(TAG, "[RNOH]:RNCClipboardTurboModule call getImagePNG, releasing image packaging");
211
+ }).catch((error: BusinessError) => {
212
+ logger.error(TAG, "[RNOH]:RNCClipboardTurboModule call getImagePNG, releasing image packaging error");
213
+ })
214
+ })
215
+
216
+ }).catch((err) => {
217
+ logger.error(TAG, "[RNOH]:RNCClipboardTurboModule call getImagePNG,failed to getData");
218
+ reject(err)
219
+ })
220
+ } else {
221
+ reject({
222
+ code: 0,
223
+ message: "User refuses authorization"
224
+ })
225
+ }
226
+ }).catch((err) => {
227
+ logger.error(TAG, "[RNOH]:RNCClipboardTurboModule call getImagePNG,failed to request permission");
228
+ reject({
229
+ code: 2,
230
+ message: "Failed to request permission"
231
+ })
232
+ });
233
+ });
234
+ }
235
+
236
+ // ios
237
+ getImageJPG(): Promise<string> {
238
+ return new Promise<string>((resolve, reject) => {
239
+ this.requestPermission().then(res => {
240
+ if (res) {
241
+ logger.debug(TAG, "[RNOH]:RNCClipboardTurboModule call getImageJPG");
242
+ let systemPasteboard = pasteboard.getSystemPasteboard();
243
+
244
+ systemPasteboard.getData().then((pasteData) => {
245
+ let pixMap = pasteData.getPrimaryPixelMap();
246
+ if (!pixMap) {
247
+ logger.error(TAG, "[RNOH]:RNCClipboardTurboModule call getImageJPG,pixMap is null or undefined");
248
+ reject({
249
+ code: 1,
250
+ message: "Clipboard does not contain an image"
251
+ });
252
+ return;
253
+ }
254
+ const imagePackerApi = image.createImagePacker();
255
+ let packOpt: image.PackingOption = {
256
+ format: "image/jpeg",
257
+ quality: 96
258
+ }
259
+ imagePackerApi.packing(pixMap, packOpt).then(data => {
260
+ let uint8Array = new Uint8Array(data);
261
+ let base64Helper = new util.Base64Helper();
262
+ let base64Str = base64Helper.encodeToStringSync(uint8Array, util.Type.BASIC)
263
+ let finalStr = prefixJPG + base64Str
264
+ resolve(finalStr)
265
+ }).catch((err) => {
266
+ logger.error(TAG, "[RNOH]:RNCClipboardTurboModule call getImageJPG,failed to packing");
267
+ reject(err)
268
+ }).finally(() => {
269
+ imagePackerApi.release().then(() => {
270
+ logger.debug(TAG, "[RNOH]:RNCClipboardTurboModule call getImageJPG, releasing image packaging");
271
+ }).catch((error: BusinessError) => {
272
+ logger.error(TAG, "[RNOH]:RNCClipboardTurboModule call getImageJPG, releasing image packaging error");
273
+ })
274
+ })
275
+
276
+ }).catch((err) => {
277
+ logger.error(TAG, "[RNOH]:RNCClipboardTurboModule call getImageJPG,failed to getData");
278
+ reject(err)
279
+ })
280
+ } else {
281
+ reject({
282
+ code: 0,
283
+ message: "User refuses authorization"
284
+ })
285
+ }
286
+ }).catch((err) => {
287
+ logger.error(TAG, "[RNOH]:RNCClipboardTurboModule call getImageJPG,failed to request permission");
288
+ reject({
289
+ code: 2,
290
+ message: "Failed to request permission"
291
+ })
292
+ });
293
+ });
294
+ }
295
+
296
+ setImage(content: string) {
297
+ logger.debug(TAG, `[RNOH]:RNCClipboardTurboModule call setImage:${content}`);
298
+ let iconBase64 = content
299
+
300
+ let base64 = new util.Base64Helper();
301
+ let uint8 = base64.decodeSync(iconBase64, util.Type.BASIC)
302
+ let arrayBuffer = uint8.buffer.slice(uint8.byteOffset, uint8.byteLength + uint8.byteOffset)
303
+ let imageSource: image.ImageSource = image.createImageSource(arrayBuffer);
304
+ logger.debug(TAG, "[RNOH]:RNCClipboardTurboModule call setImage100");
305
+
306
+ // 获取图片格式信息
307
+ let imageFormat = this.getImageFormatFromArrayBuffer(arrayBuffer);
308
+ logger.debug(TAG, `[RNOH]:RNCClipboardTurboModule call setImage, detected format: ${imageFormat}`);
309
+
310
+ imageSource.getImageInfo().then(value => {
311
+ let hValue = Math.round(value.size.height);
312
+ let wValue = Math.round(value.size.width);
313
+ let defaultSize: image.Size = {
314
+ height: hValue,
315
+ width: wValue
316
+ };
317
+ let opts: image.DecodingOptions = {
318
+ editable: true,
319
+ desiredSize: defaultSize
320
+ };
321
+ logger.debug(TAG, "[RNOH]:RNCClipboardTurboModule call setImage200");
322
+ imageSource.createPixelMap(opts).then((pixMap) => {
323
+ let iconPixelMap = pixMap
324
+ logger.debug(TAG, "[RNOH]:RNCClipboardTurboModule call setImage300");
325
+ imageSource.release()
326
+
327
+ let systemPasteboard = pasteboard.getSystemPasteboard();
328
+ systemPasteboard.getData().then((pasteData) => {
329
+ // 使用 MIMETYPE_PIXELMAP 创建记录,确保兼容性
330
+ let record = pasteboard.createRecord(pasteboard.MIMETYPE_PIXELMAP, iconPixelMap)
331
+ // 在 record 的 data 中保存原始格式信息,供 getImage 读取
332
+ if (!record.data) {
333
+ record.data = {};
334
+ }
335
+ record.data[imageFormat] = arrayBuffer;
336
+ pasteData.addRecord(record)
337
+
338
+ systemPasteboard.setData(pasteData).then((data: void) => {
339
+ logger.debug(TAG, `[RNOH]:RNCClipboardTurboModule call setImage305,successed in setting pasteData with format: ${imageFormat}`);
340
+ }).catch((err) => {
341
+ logger.error(TAG, "[RNOH]:RNCClipboardTurboModule call setImage305,failed in setting pasteData");
342
+ })
343
+ }).catch((err) => {
344
+ logger.debug(TAG, `[RNOH]:RNCClipboardTurboModule call setImage,failed get pasteData.cause:${err.message}`);
345
+ })
346
+ })
347
+ })
348
+
349
+ }
350
+
351
+ // 从 ArrayBuffer 中检测图片格式
352
+ private getImageFormatFromArrayBuffer(buffer: ArrayBuffer): string {
353
+ let uint8Array = new Uint8Array(buffer);
354
+
355
+ // 检查 JPEG (FF D8 FF)
356
+ if (uint8Array.length >= 3 &&
357
+ uint8Array[0] === 0xFF &&
358
+ uint8Array[1] === 0xD8 &&
359
+ uint8Array[2] === 0xFF) {
360
+ return "image/jpeg";
361
+ }
362
+
363
+ // 检查 PNG (89 50 4E 47 0D 0A 1A 0A)
364
+ if (uint8Array.length >= 8 &&
365
+ uint8Array[0] === 0x89 &&
366
+ uint8Array[1] === 0x50 &&
367
+ uint8Array[2] === 0x4E &&
368
+ uint8Array[3] === 0x47 &&
369
+ uint8Array[4] === 0x0D &&
370
+ uint8Array[5] === 0x0A &&
371
+ uint8Array[6] === 0x1A &&
372
+ uint8Array[7] === 0x0A) {
373
+ return "image/png";
374
+ }
375
+
376
+ // 检查 WEBP (52 49 46 46 ... 57 45 42 50)
377
+ if (uint8Array.length >= 12 &&
378
+ uint8Array[0] === 0x52 &&
379
+ uint8Array[1] === 0x49 &&
380
+ uint8Array[2] === 0x46 &&
381
+ uint8Array[3] === 0x46 &&
382
+ uint8Array[8] === 0x57 &&
383
+ uint8Array[9] === 0x45 &&
384
+ uint8Array[10] === 0x42 &&
385
+ uint8Array[11] === 0x50) {
386
+ return "image/webp";
387
+ }
388
+
389
+ // 检查 GIF (47 49 46 38)
390
+ if (uint8Array.length >= 4 &&
391
+ uint8Array[0] === 0x47 &&
392
+ uint8Array[1] === 0x49 &&
393
+ uint8Array[2] === 0x46 &&
394
+ uint8Array[3] === 0x38) {
395
+ return "image/gif";
396
+ }
397
+
398
+ // 检查 BMP (42 4D)
399
+ if (uint8Array.length >= 2 &&
400
+ uint8Array[0] === 0x42 &&
401
+ uint8Array[1] === 0x4D) {
402
+ return "image/bmp";
403
+ }
404
+
405
+ // 默认返回 PNG
406
+ logger.debug(TAG, "[RNOH]:RNCClipboardTurboModule getImageFormatFromArrayBuffer, unknown format, defaulting to image/png");
407
+ return "image/png";
408
+ }
409
+
410
+ getImage(): Promise<string> {
411
+ return new Promise<string>((resolve, reject) => {
412
+ logger.debug(TAG, "[RNOH]:RNCClipboardTurboModule call getImage");
413
+ this.requestPermission().then(async res => {
414
+ if (!res) {
415
+ reject({
416
+ code: 0,
417
+ message: "User refuses authorization"
418
+ })
419
+ return
420
+ }
421
+
422
+ try {
423
+ let systemPasteboard = pasteboard.getSystemPasteboard();
424
+ let pasteData = await systemPasteboard.getData();
425
+
426
+ // 检查剪贴板中是否有 JPEG 或 PNG 格式的数据
427
+ // 如果有,使用与 getImageJPG/getImagePNG 相同的逻辑:通过 getPrimaryPixelMap 重新打包
428
+ let imageFormat = this.getImageFormatFromPasteData(pasteData);
429
+ if (imageFormat) {
430
+ logger.debug(TAG, `[RNOH]:RNCClipboardTurboModule getImage, detected format: ${imageFormat}, using getImageJPG/getImagePNG logic`);
431
+ let pixelMapObj = pasteData.getPrimaryPixelMap();
432
+ if (pixelMapObj) {
433
+ // 使用与 getImageJPG/getImagePNG 相同的参数重新打包
434
+ let imageDataUri = await this.packPixelMapAsDataUri(pixelMapObj, imageFormat);
435
+ resolve(imageDataUri);
436
+ return;
437
+ }
438
+ }
439
+
440
+ let imageDataUri = await this.getImageDataUriFromPasteData(pasteData);
441
+ resolve(imageDataUri)
442
+ } catch (err) {
443
+ logger.error(TAG, `[RNOH]:RNCClipboardTurboModule call getImage,failed to get image. Cause:${err.message}`);
444
+ reject(err)
445
+ }
446
+ }).catch((err) => {
447
+ logger.error(TAG, `[RNOH]:RNCClipboardTurboModule call getImage,failed to request permission. Cause:${err.message}`);
448
+ reject(err)
449
+ })
450
+ });
451
+ }
452
+
453
+ // 从剪贴板数据中检测图片格式
454
+ private getImageFormatFromPasteData(pasteData: pasteboard.PasteData): string | null {
455
+ let count = pasteData.getRecordCount();
456
+ for (let index = 0; index < count; index++) {
457
+ let record = pasteData.getRecord(index);
458
+ // 检查 record.mimeType
459
+ let mimeType = record.mimeType?.toLowerCase() || "";
460
+ if (mimeType === "image/jpeg" || mimeType === "image/jpg") {
461
+ return "image/jpeg";
462
+ }
463
+ if (mimeType === "image/png") {
464
+ return "image/png";
465
+ }
466
+ // 检查 record.data 中的 mimeType
467
+ if (record.data) {
468
+ for (let mt of Object.keys(record.data)) {
469
+ let lowerMt = mt.toLowerCase();
470
+ if (lowerMt === "image/jpeg" || lowerMt === "image/jpg") {
471
+ return "image/jpeg";
472
+ }
473
+ if (lowerMt === "image/png") {
474
+ return "image/png";
475
+ }
476
+ }
477
+ }
478
+ }
479
+ return null;
480
+ }
481
+
482
+ private async getImageDataUriFromPasteData(pasteData: pasteboard.PasteData): Promise<string> {
483
+ let count = pasteData.getRecordCount();
484
+ for (let index = 0; index < count; index++) {
485
+ let record = pasteData.getRecord(index);
486
+ let dataUri = await this.getImageDataUriFromRecord(record);
487
+ if (dataUri.length > 0) {
488
+ return dataUri
489
+ }
490
+ }
491
+
492
+ try {
493
+ let pixelMapObj = pasteData.getPrimaryPixelMap();
494
+ if (pixelMapObj) {
495
+ return await this.packPixelMapAsDataUri(pixelMapObj, defaultImageMimeType)
496
+ }
497
+ } catch (err) {
498
+ logger.debug(TAG, `[RNOH]:RNCClipboardTurboModule getImage,no primary PixelMap. Cause:${err.message}`);
499
+ }
500
+
501
+ return ""
502
+ }
503
+
504
+ private async getImageDataUriFromRecord(record: pasteboard.PasteDataRecord): Promise<string> {
505
+ if (this.isImageMimeType(record.mimeType)) {
506
+ let dataUri = await this.getImageDataUriByMimeType(record, record.mimeType);
507
+ if (dataUri.length > 0) {
508
+ return dataUri
509
+ }
510
+ }
511
+
512
+ if (record.data) {
513
+ let customMimeTypes = Object.keys(record.data);
514
+ for (let index = 0; index < customMimeTypes.length; index++) {
515
+ let mimeType = customMimeTypes[index];
516
+ if (this.isImageMimeType(mimeType)) {
517
+ return this.createDataUri(mimeType, record.data[mimeType])
518
+ }
519
+ }
520
+ }
521
+
522
+ if (record.pixelMap) {
523
+ let mimeType = this.isImageMimeType(record.mimeType) ? record.mimeType : defaultImageMimeType;
524
+ return await this.packPixelMapAsDataUri(record.pixelMap, mimeType)
525
+ }
526
+
527
+ return ""
528
+ }
529
+
530
+ private async getImageDataUriByMimeType(record: pasteboard.PasteDataRecord, mimeType: string): Promise<string> {
531
+ if (record.data && record.data[mimeType]) {
532
+ return this.createDataUri(mimeType, record.data[mimeType])
533
+ }
534
+
535
+ try {
536
+ let value = await record.getData(mimeType);
537
+ if (value instanceof ArrayBuffer) {
538
+ return this.createDataUri(mimeType, value)
539
+ }
540
+
541
+ if (this.isPixelMap(value)) {
542
+ return await this.packPixelMapAsDataUri(value as image.PixelMap, mimeType)
543
+ }
544
+ } catch (err) {
545
+ logger.debug(TAG, `[RNOH]:RNCClipboardTurboModule getImage,failed to read mimeType:${mimeType}. Cause:${err.message}`);
546
+ }
547
+
548
+ return ""
549
+ }
550
+
551
+ private async packPixelMapAsDataUri(pixelMapObj: image.PixelMap, mimeType: string): Promise<string> {
552
+ const imagePackerApi = image.createImagePacker();
553
+ let packingMimeType = this.getPackingMimeType(mimeType);
554
+ let packOpt: image.PackingOption = {
555
+ format: packingMimeType,
556
+ quality: 96
557
+ }
558
+
559
+ try {
560
+ let data = await imagePackerApi.packing(pixelMapObj, packOpt);
561
+ return this.createDataUri(packingMimeType, data)
562
+ } finally {
563
+ imagePackerApi.release().then(() => {
564
+ logger.debug(TAG, "[RNOH]:RNCClipboardTurboModule call getImage, releasing image packaging");
565
+ }).catch((error: BusinessError) => {
566
+ logger.error(TAG, `[RNOH]:RNCClipboardTurboModule call getImage, releasing image packaging error. Cause:${error.message}`);
567
+ })
568
+ }
569
+ }
570
+
571
+ private createDataUri(mimeType: string, data: ArrayBuffer): string {
572
+ let uint8Array = new Uint8Array(data);
573
+ let base64Helper = new util.Base64Helper();
574
+ let base64Str = base64Helper.encodeToStringSync(uint8Array, util.Type.BASIC);
575
+ return `data:${mimeType};base64,${base64Str}`
576
+ }
577
+
578
+ private getPackingMimeType(mimeType: string): string {
579
+ let normalizedMimeType = mimeType.toLowerCase();
580
+ if (normalizedMimeType == "image/jpeg" || normalizedMimeType == "image/jpg") {
581
+ return "image/jpeg"
582
+ }
583
+ if (normalizedMimeType == "image/webp") {
584
+ return "image/webp"
585
+ }
586
+ return defaultImageMimeType
587
+ }
588
+
589
+ private isImageMimeType(mimeType: string): boolean {
590
+ return mimeType != undefined && mimeType.length > 0 && mimeType.toLowerCase().indexOf(imageMimePrefix) == 0
591
+ }
592
+
593
+ private isPixelMap(value: pasteboard.ValueType): boolean {
594
+ return value != undefined && typeof value == "object" && "getImageInfo" in value
595
+ }
596
+
597
+ setStrings(content: string[]) {
598
+ logger.debug(TAG, "[RNOH]:RNCClipboardTurboModule call setStrings fun");
599
+ let systemPasteboard = pasteboard.getSystemPasteboard();
600
+ systemPasteboard.clear().then(() => {
601
+ systemPasteboard.getData().then((pasteData) => {
602
+ while (content.length > 0) {
603
+ pasteData.addRecord(pasteboard.MIMETYPE_TEXT_PLAIN, content.pop())
604
+ }
605
+
606
+ // setData
607
+ systemPasteboard.setData(pasteData).then((data: void) => {
608
+ logger.debug(TAG, "setStrings,Succeeded in setting PasteData.");
609
+ }).catch((err) => {
610
+ logger.error(TAG, `setStrings,Failed to set PasteData. Cause:${err.message}`);
611
+ });
612
+ }).catch((err) => {
613
+ logger.error(TAG, `setStrings,getData error,Cause:${err.message}`);
614
+ })
615
+ }).catch((err: BusinessError) => {
616
+ console.error(`Failed to clear the PasteData. Cause: ${err.message}`);
617
+ });
618
+ return;
619
+ }
620
+
621
+ hasImage(): Promise<boolean> {
622
+ return new Promise<boolean>((resolve, reject) => {
623
+ this.requestPermission().then(res => {
624
+ if (res) {
625
+ logger.debug(TAG, "RNCClipboardTurboModule call hasImage");
626
+ let systemPasteboard = pasteboard.getSystemPasteboard();
627
+ systemPasteboard.getData().then((pasteData) => {
628
+ let pixelMapObj = pasteData.getPrimaryPixelMap();
629
+ let result = false
630
+ if (pixelMapObj) {
631
+ logger.debug(TAG, "RNCClipboardTurboModule call hasImage,hasPixelobj");
632
+ result = true
633
+ }
634
+ resolve(result)
635
+ }).catch((err) => {
636
+ logger.error(TAG, `hasImage,Failed to get PasteData. Cause:${err.message}`);
637
+ reject(err)
638
+ })
639
+ } else {
640
+ reject({
641
+ code: 0,
642
+ message: "User refuses authorization"
643
+ })
644
+ }
645
+ });
646
+ });
647
+ }
648
+
649
+ hasURL(): Promise<boolean> {
650
+ return new Promise<boolean>((resolve, reject) => {
651
+ this.requestPermission().then(res => {
652
+ if (res) {
653
+ logger.debug(TAG, "RNCClipboardTurboModule call hasWebURL");
654
+ let systemPasteboard = pasteboard.getSystemPasteboard();
655
+ systemPasteboard.getData().then((pasteData) => {
656
+ let count = pasteData.getRecordCount();
657
+ let result = false;
658
+ let isValidUrl = (string) => {
659
+ try {
660
+ url.URL.parseURL(string);
661
+ return true;
662
+ } catch (err) {
663
+ return false;
664
+ }
665
+ };
666
+ for (let index = 0; index < count; index++) {
667
+ let record = pasteData.getRecord(index);
668
+ if (record.mimeType == pasteboard.MIMETYPE_TEXT_URI) {
669
+ if (isValidUrl(record.uri)) {
670
+ logger.debug(TAG, "hasURL,mimeType=MIMETYPE_TEXT_URI");
671
+ result = true
672
+ break
673
+ }
674
+ } else if (record.mimeType == pasteboard.MIMETYPE_TEXT_PLAIN) {
675
+ if (isValidUrl(record.plainText)) {
676
+ logger.debug(TAG, "hasURL,find URL in MIMETYPE_TEXT_PLAIN");
677
+ result = true
678
+ break
679
+ }
680
+ }
681
+ }
682
+ resolve(result)
683
+ }).catch((err) => {
684
+ logger.error(TAG, `[RNOH]: hasURL,Failed to get PasteData. Cause:${err.message}`);
685
+ reject(err)
686
+ });
687
+ } else {
688
+ reject({
689
+ code: 0,
690
+ message: "User refuses authorization"
691
+ })
692
+ }
693
+ });
694
+ });
695
+ }
696
+
697
+ hasWebURL(): Promise<boolean> {
698
+ return new Promise<boolean>((resolve, reject) => {
699
+ this.requestPermission().then(res => {
700
+ if (res) {
701
+ logger.debug(TAG, "RNCClipboardTurboModule call hasWebURL");
702
+ let systemPasteboard = pasteboard.getSystemPasteboard();
703
+ systemPasteboard.getData().then((pasteData) => {
704
+ let count = pasteData.getRecordCount();
705
+ let result = false
706
+ let reg = /(http|ftp|https):\/\/[\w\-_]+(\.[\w\-_]+)+([\w\-\.,@?^=%&:/~\+#]*[\w\-\@?^=%&/~\+#])?/;
707
+
708
+ for (let index = 0; index < count; index++) {
709
+ let record = pasteData.getRecord(index);
710
+ if (record.mimeType == pasteboard.MIMETYPE_TEXT_URI) {
711
+ if (reg.test(record.uri)) {
712
+ logger.debug(TAG, "hasWebURL,find webURL in MIMETYPE_TEXT_URI");
713
+ result = true
714
+ break
715
+ }
716
+ } else if (record.mimeType == pasteboard.MIMETYPE_TEXT_PLAIN) {
717
+ if (reg.test(record.plainText)) {
718
+ logger.debug(TAG, "hasWebURL,find webURL in MIMETYPE_TEXT_PLAIN");
719
+ result = true
720
+ break
721
+ }
722
+ }
723
+
724
+ }
725
+ resolve(result)
726
+ }).catch((err) => {
727
+ logger.error(TAG, `hasWebURL,Failed to get PasteData. Cause:${err.message}`);
728
+ reject(err)
729
+ });
730
+ } else {
731
+ reject({
732
+ code: 0,
733
+ message: "User refuses authorization"
734
+ })
735
+ }
736
+ });
737
+ });
738
+ }
739
+
740
+ requestPermission(): Promise<boolean> {
741
+ return new Promise<boolean>((resolve) => {
742
+ abilityAccessCtrl.createAtManager()
743
+ .requestPermissionsFromUser(this.ctx.uiAbilityContext, PERMISSIONS).then(result => {
744
+ if (result.authResults[0] == 0) {
745
+ resolve(true);
746
+ } else {
747
+ logger.debug(TAG, `getString,text out:用户拒绝授权`);
748
+ resolve(false);
749
+ }
750
+ }).catch(() => {
751
+ logger.debug(TAG, `getString,text out:用户拒绝授权`);
752
+ resolve(false);
753
+ })
754
+ });
755
+ }
756
+
757
+ addListener(eventName: string): void {}
758
+ removeListeners(count: number): void {}
759
+ }