@exodus/react-native-webview 9.4.0-no-android.0 → 11.26.1-exodus.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 (49) hide show
  1. package/README.md +21 -18
  2. package/android/.editorconfig +6 -0
  3. package/android/build.gradle +137 -0
  4. package/android/gradle.properties +6 -0
  5. package/android/src/main/AndroidManifest.xml +15 -0
  6. package/android/src/main/java/com/reactnativecommunity/webview/RNCWebViewFileProvider.java +14 -0
  7. package/android/src/main/java/com/reactnativecommunity/webview/RNCWebViewManager.java +1650 -0
  8. package/android/src/main/java/com/reactnativecommunity/webview/RNCWebViewModule.java +550 -0
  9. package/android/src/main/java/com/reactnativecommunity/webview/RNCWebViewPackage.kt +15 -0
  10. package/android/src/main/java/com/reactnativecommunity/webview/WebViewConfig.java +12 -0
  11. package/android/src/main/java/com/reactnativecommunity/webview/events/TopHttpErrorEvent.kt +25 -0
  12. package/android/src/main/java/com/reactnativecommunity/webview/events/TopLoadingErrorEvent.kt +25 -0
  13. package/android/src/main/java/com/reactnativecommunity/webview/events/TopLoadingFinishEvent.kt +24 -0
  14. package/android/src/main/java/com/reactnativecommunity/webview/events/TopLoadingProgressEvent.kt +24 -0
  15. package/android/src/main/java/com/reactnativecommunity/webview/events/TopLoadingStartEvent.kt +25 -0
  16. package/android/src/main/java/com/reactnativecommunity/webview/events/TopMessageEvent.kt +24 -0
  17. package/android/src/main/java/com/reactnativecommunity/webview/events/TopRenderProcessGoneEvent.kt +26 -0
  18. package/android/src/main/java/com/reactnativecommunity/webview/events/TopShouldStartLoadWithRequestEvent.kt +29 -0
  19. package/android/src/main/res/xml/file_provider_paths.xml +6 -0
  20. package/apple/RNCWKProcessPoolManager.h +15 -0
  21. package/apple/RNCWKProcessPoolManager.m +36 -0
  22. package/apple/RNCWebView.h +117 -0
  23. package/apple/RNCWebView.m +1532 -0
  24. package/apple/RNCWebViewManager.h +13 -0
  25. package/apple/RNCWebViewManager.m +288 -0
  26. package/index.d.ts +65 -0
  27. package/index.js +4 -0
  28. package/ios/RNCWebView.xcodeproj/project.pbxproj +2 -0
  29. package/lib/WebView.android.d.ts +7 -0
  30. package/lib/WebView.android.js +125 -1
  31. package/lib/WebView.d.ts +7 -0
  32. package/lib/WebView.ios.d.ts +7 -0
  33. package/lib/WebView.ios.js +148 -202
  34. package/lib/WebView.js +9 -2
  35. package/lib/WebView.styles.d.ts +12 -0
  36. package/lib/WebView.styles.js +7 -7
  37. package/lib/WebViewNativeComponent.android.d.ts +4 -0
  38. package/lib/WebViewNativeComponent.android.js +3 -0
  39. package/lib/WebViewNativeComponent.ios.d.ts +4 -0
  40. package/lib/WebViewNativeComponent.ios.js +3 -0
  41. package/lib/WebViewShared.d.ts +37 -0
  42. package/lib/WebViewShared.js +121 -24
  43. package/lib/WebViewTypes.d.ts +873 -0
  44. package/lib/WebViewTypes.js +31 -16
  45. package/lib/index.d.ts +4 -0
  46. package/lib/index.js +3 -0
  47. package/package.json +83 -87
  48. package/react-native-webview.podspec +4 -4
  49. package/react-native.config.js +37 -0
@@ -0,0 +1,550 @@
1
+ package com.reactnativecommunity.webview;
2
+
3
+ import android.Manifest;
4
+ import android.app.Activity;
5
+ import android.app.DownloadManager;
6
+ import android.content.Context;
7
+ import android.content.Intent;
8
+ import android.content.pm.PackageManager;
9
+ import android.net.Uri;
10
+ import android.os.Build;
11
+ import android.os.Environment;
12
+ import android.os.Parcelable;
13
+ import android.provider.MediaStore;
14
+
15
+ import androidx.annotation.Nullable;
16
+ import androidx.annotation.RequiresApi;
17
+ import androidx.core.content.ContextCompat;
18
+ import androidx.core.content.FileProvider;
19
+ import androidx.core.util.Pair;
20
+
21
+ import android.util.Log;
22
+ import android.webkit.MimeTypeMap;
23
+ import android.webkit.ValueCallback;
24
+ import android.webkit.WebChromeClient;
25
+ import android.widget.Toast;
26
+
27
+ import com.facebook.react.bridge.ActivityEventListener;
28
+ import com.facebook.react.bridge.Promise;
29
+ import com.facebook.react.bridge.ReactApplicationContext;
30
+ import com.facebook.react.bridge.ReactContextBaseJavaModule;
31
+ import com.facebook.react.bridge.ReactMethod;
32
+ import com.facebook.react.module.annotations.ReactModule;
33
+ import com.facebook.react.modules.core.PermissionAwareActivity;
34
+ import com.facebook.react.modules.core.PermissionListener;
35
+
36
+ import java.io.File;
37
+ import java.io.IOException;
38
+ import java.util.ArrayList;
39
+ import java.util.Arrays;
40
+ import java.util.HashMap;
41
+ import java.util.concurrent.atomic.AtomicReference;
42
+
43
+ import static android.app.Activity.RESULT_OK;
44
+
45
+ @ReactModule(name = RNCWebViewModule.MODULE_NAME)
46
+ public class RNCWebViewModule extends ReactContextBaseJavaModule implements ActivityEventListener {
47
+ public static final String MODULE_NAME = "RNCWebView";
48
+ private static final int PICKER = 1;
49
+ private static final int PICKER_LEGACY = 3;
50
+ private static final int FILE_DOWNLOAD_PERMISSION_REQUEST = 1;
51
+ private ValueCallback<Uri> filePathCallbackLegacy;
52
+ private ValueCallback<Uri[]> filePathCallback;
53
+ private File outputImage;
54
+ private File outputVideo;
55
+ private DownloadManager.Request downloadRequest;
56
+
57
+ protected static class ShouldOverrideUrlLoadingLock {
58
+ protected enum ShouldOverrideCallbackState {
59
+ UNDECIDED,
60
+ SHOULD_OVERRIDE,
61
+ DO_NOT_OVERRIDE,
62
+ }
63
+
64
+ private int nextLockIdentifier = 1;
65
+ private final HashMap<Integer, AtomicReference<ShouldOverrideCallbackState>> shouldOverrideLocks = new HashMap<>();
66
+
67
+ public synchronized Pair<Integer, AtomicReference<ShouldOverrideCallbackState>> getNewLock() {
68
+ final int lockIdentifier = nextLockIdentifier++;
69
+ final AtomicReference<ShouldOverrideCallbackState> shouldOverride = new AtomicReference<>(ShouldOverrideCallbackState.UNDECIDED);
70
+ shouldOverrideLocks.put(lockIdentifier, shouldOverride);
71
+ return new Pair<>(lockIdentifier, shouldOverride);
72
+ }
73
+
74
+ @Nullable
75
+ public synchronized AtomicReference<ShouldOverrideCallbackState> getLock(Integer lockIdentifier) {
76
+ return shouldOverrideLocks.get(lockIdentifier);
77
+ }
78
+
79
+ public synchronized void removeLock(Integer lockIdentifier) {
80
+ shouldOverrideLocks.remove(lockIdentifier);
81
+ }
82
+ }
83
+
84
+ protected static final ShouldOverrideUrlLoadingLock shouldOverrideUrlLoadingLock = new ShouldOverrideUrlLoadingLock();
85
+
86
+ private enum MimeType {
87
+ DEFAULT("*/*"),
88
+ IMAGE("image"),
89
+ VIDEO("video");
90
+
91
+ private final String value;
92
+
93
+ MimeType(String value) {
94
+ this.value = value;
95
+ }
96
+ }
97
+
98
+ private PermissionListener getWebviewFileDownloaderPermissionListener(String downloadingMessage, String lackPermissionToDownloadMessage) {
99
+ return new PermissionListener() {
100
+ @Override
101
+ public boolean onRequestPermissionsResult(int requestCode, String[] permissions, int[] grantResults) {
102
+ switch (requestCode) {
103
+ case FILE_DOWNLOAD_PERMISSION_REQUEST: {
104
+ // If request is cancelled, the result arrays are empty.
105
+ if (grantResults.length > 0 && grantResults[0] == PackageManager.PERMISSION_GRANTED) {
106
+ if (downloadRequest != null) {
107
+ downloadFile(downloadingMessage);
108
+ }
109
+ } else {
110
+ Toast.makeText(getCurrentActivity().getApplicationContext(), lackPermissionToDownloadMessage, Toast.LENGTH_LONG).show();
111
+ }
112
+ return true;
113
+ }
114
+ }
115
+ return false;
116
+ }
117
+ };
118
+ }
119
+
120
+ public RNCWebViewModule(ReactApplicationContext reactContext) {
121
+ super(reactContext);
122
+ reactContext.addActivityEventListener(this);
123
+ }
124
+
125
+ @Override
126
+ public String getName() {
127
+ return MODULE_NAME;
128
+ }
129
+
130
+ @ReactMethod
131
+ public void isFileUploadSupported(final Promise promise) {
132
+ Boolean result = false;
133
+ int current = Build.VERSION.SDK_INT;
134
+ if (current >= Build.VERSION_CODES.LOLLIPOP) {
135
+ result = true;
136
+ }
137
+ if (current >= Build.VERSION_CODES.JELLY_BEAN && current <= Build.VERSION_CODES.JELLY_BEAN_MR2) {
138
+ result = true;
139
+ }
140
+ promise.resolve(result);
141
+ }
142
+
143
+ @ReactMethod(isBlockingSynchronousMethod = true)
144
+ public void onShouldStartLoadWithRequestCallback(final boolean shouldStart, final int lockIdentifier) {
145
+ final AtomicReference<ShouldOverrideUrlLoadingLock.ShouldOverrideCallbackState> lockObject = shouldOverrideUrlLoadingLock.getLock(lockIdentifier);
146
+ if (lockObject != null) {
147
+ synchronized (lockObject) {
148
+ lockObject.set(shouldStart ? ShouldOverrideUrlLoadingLock.ShouldOverrideCallbackState.DO_NOT_OVERRIDE : ShouldOverrideUrlLoadingLock.ShouldOverrideCallbackState.SHOULD_OVERRIDE);
149
+ lockObject.notify();
150
+ }
151
+ }
152
+ }
153
+
154
+ public void onActivityResult(Activity activity, int requestCode, int resultCode, Intent data) {
155
+
156
+ if (filePathCallback == null && filePathCallbackLegacy == null) {
157
+ return;
158
+ }
159
+
160
+ boolean imageTaken = false;
161
+ boolean videoTaken = false;
162
+
163
+ if (outputImage != null && outputImage.length() > 0) {
164
+ imageTaken = true;
165
+ }
166
+ if (outputVideo != null && outputVideo.length() > 0) {
167
+ videoTaken = true;
168
+ }
169
+
170
+ // based off of which button was pressed, we get an activity result and a file
171
+ // the camera activity doesn't properly return the filename* (I think?) so we use
172
+ // this filename instead
173
+ switch (requestCode) {
174
+ case PICKER:
175
+ if (resultCode != RESULT_OK) {
176
+ if (filePathCallback != null) {
177
+ filePathCallback.onReceiveValue(null);
178
+ }
179
+ } else {
180
+ if (imageTaken) {
181
+ filePathCallback.onReceiveValue(new Uri[]{getOutputUri(outputImage)});
182
+ } else if (videoTaken) {
183
+ filePathCallback.onReceiveValue(new Uri[]{getOutputUri(outputVideo)});
184
+ } else {
185
+ filePathCallback.onReceiveValue(this.getSelectedFiles(data, resultCode));
186
+ }
187
+ }
188
+ break;
189
+ case PICKER_LEGACY:
190
+ if (resultCode != RESULT_OK) {
191
+ filePathCallbackLegacy.onReceiveValue(null);
192
+ } else {
193
+ if (imageTaken) {
194
+ filePathCallbackLegacy.onReceiveValue(getOutputUri(outputImage));
195
+ } else if (videoTaken) {
196
+ filePathCallbackLegacy.onReceiveValue(getOutputUri(outputVideo));
197
+ } else {
198
+ filePathCallbackLegacy.onReceiveValue(data.getData());
199
+ }
200
+ }
201
+ break;
202
+
203
+ }
204
+
205
+ if (outputImage != null && !imageTaken) {
206
+ outputImage.delete();
207
+ }
208
+ if (outputVideo != null && !videoTaken) {
209
+ outputVideo.delete();
210
+ }
211
+
212
+ filePathCallback = null;
213
+ filePathCallbackLegacy = null;
214
+ outputImage = null;
215
+ outputVideo = null;
216
+ }
217
+
218
+ public void onNewIntent(Intent intent) {
219
+ }
220
+
221
+ private Uri[] getSelectedFiles(Intent data, int resultCode) {
222
+ if (data == null) {
223
+ return null;
224
+ }
225
+
226
+ // we have multiple files selected
227
+ if (data.getClipData() != null) {
228
+ final int numSelectedFiles = data.getClipData().getItemCount();
229
+ Uri[] result = new Uri[numSelectedFiles];
230
+ for (int i = 0; i < numSelectedFiles; i++) {
231
+ result[i] = data.getClipData().getItemAt(i).getUri();
232
+ }
233
+ return result;
234
+ }
235
+
236
+ // we have one file selected
237
+ if (data.getData() != null && resultCode == RESULT_OK && Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP) {
238
+ return WebChromeClient.FileChooserParams.parseResult(resultCode, data);
239
+ }
240
+
241
+ return null;
242
+ }
243
+
244
+ public void startPhotoPickerIntent(ValueCallback<Uri> filePathCallback, String acceptType) {
245
+ filePathCallbackLegacy = filePathCallback;
246
+
247
+ Intent fileChooserIntent = getFileChooserIntent(acceptType);
248
+ Intent chooserIntent = Intent.createChooser(fileChooserIntent, "");
249
+
250
+ ArrayList<Parcelable> extraIntents = new ArrayList<>();
251
+ if (acceptsImages(acceptType)) {
252
+ Intent photoIntent = getPhotoIntent();
253
+ if (photoIntent != null) {
254
+ extraIntents.add(photoIntent);
255
+ }
256
+ }
257
+ if (acceptsVideo(acceptType)) {
258
+ Intent videoIntent = getVideoIntent();
259
+ if (videoIntent != null) {
260
+ extraIntents.add(videoIntent);
261
+ }
262
+ }
263
+ chooserIntent.putExtra(Intent.EXTRA_INITIAL_INTENTS, extraIntents.toArray(new Parcelable[]{}));
264
+
265
+ if (chooserIntent.resolveActivity(getCurrentActivity().getPackageManager()) != null) {
266
+ getCurrentActivity().startActivityForResult(chooserIntent, PICKER_LEGACY);
267
+ } else {
268
+ Log.w("RNCWebViewModule", "there is no Activity to handle this Intent");
269
+ }
270
+ }
271
+
272
+ @RequiresApi(api = Build.VERSION_CODES.LOLLIPOP)
273
+ public boolean startPhotoPickerIntent(final ValueCallback<Uri[]> callback, final String[] acceptTypes, final boolean allowMultiple) {
274
+ filePathCallback = callback;
275
+
276
+ ArrayList<Parcelable> extraIntents = new ArrayList<>();
277
+ if (!needsCameraPermission()) {
278
+ if (acceptsImages(acceptTypes)) {
279
+ Intent photoIntent = getPhotoIntent();
280
+ if (photoIntent != null) {
281
+ extraIntents.add(photoIntent);
282
+ }
283
+ }
284
+ if (acceptsVideo(acceptTypes)) {
285
+ Intent videoIntent = getVideoIntent();
286
+ if (videoIntent != null) {
287
+ extraIntents.add(videoIntent);
288
+ }
289
+ }
290
+ }
291
+
292
+ Intent fileSelectionIntent = getFileChooserIntent(acceptTypes, allowMultiple);
293
+
294
+ Intent chooserIntent = new Intent(Intent.ACTION_CHOOSER);
295
+ chooserIntent.putExtra(Intent.EXTRA_INTENT, fileSelectionIntent);
296
+ chooserIntent.putExtra(Intent.EXTRA_INITIAL_INTENTS, extraIntents.toArray(new Parcelable[]{}));
297
+
298
+ if (chooserIntent.resolveActivity(getCurrentActivity().getPackageManager()) != null) {
299
+ getCurrentActivity().startActivityForResult(chooserIntent, PICKER);
300
+ } else {
301
+ Log.w("RNCWebViewModule", "there is no Activity to handle this Intent");
302
+ }
303
+
304
+ return true;
305
+ }
306
+
307
+ public void setDownloadRequest(DownloadManager.Request request) {
308
+ this.downloadRequest = request;
309
+ }
310
+
311
+ public void downloadFile(String downloadingMessage) {
312
+ DownloadManager dm = (DownloadManager) getCurrentActivity().getBaseContext().getSystemService(Context.DOWNLOAD_SERVICE);
313
+
314
+ try {
315
+ dm.enqueue(this.downloadRequest);
316
+ } catch (IllegalArgumentException e) {
317
+ Log.w("RNCWebViewModule", "Unsupported URI, aborting download", e);
318
+ return;
319
+ }
320
+
321
+ Toast.makeText(getCurrentActivity().getApplicationContext(), downloadingMessage, Toast.LENGTH_LONG).show();
322
+ }
323
+
324
+ public boolean grantFileDownloaderPermissions(String downloadingMessage, String lackPermissionToDownloadMessage) {
325
+ // Permission not required for Android Q and above
326
+ if (Build.VERSION.SDK_INT > Build.VERSION_CODES.P) {
327
+ return true;
328
+ }
329
+
330
+ boolean result = ContextCompat.checkSelfPermission(getCurrentActivity(), Manifest.permission.WRITE_EXTERNAL_STORAGE) == PackageManager.PERMISSION_GRANTED;
331
+ if (!result && Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {
332
+ PermissionAwareActivity activity = getPermissionAwareActivity();
333
+ activity.requestPermissions(new String[]{Manifest.permission.WRITE_EXTERNAL_STORAGE}, FILE_DOWNLOAD_PERMISSION_REQUEST, getWebviewFileDownloaderPermissionListener(downloadingMessage, lackPermissionToDownloadMessage));
334
+ }
335
+
336
+ return result;
337
+ }
338
+
339
+ protected boolean needsCameraPermission() {
340
+ boolean needed = false;
341
+
342
+ PackageManager packageManager = getCurrentActivity().getPackageManager();
343
+ try {
344
+ String[] requestedPermissions = packageManager.getPackageInfo(getReactApplicationContext().getPackageName(), PackageManager.GET_PERMISSIONS).requestedPermissions;
345
+ if (Arrays.asList(requestedPermissions).contains(Manifest.permission.CAMERA)
346
+ && ContextCompat.checkSelfPermission(getCurrentActivity(), Manifest.permission.CAMERA) != PackageManager.PERMISSION_GRANTED) {
347
+ needed = true;
348
+ }
349
+ } catch (PackageManager.NameNotFoundException e) {
350
+ needed = true;
351
+ }
352
+
353
+ return needed;
354
+ }
355
+
356
+ private Intent getPhotoIntent() {
357
+ Intent intent = null;
358
+
359
+ try {
360
+ outputImage = getCapturedFile(MimeType.IMAGE);
361
+ Uri outputImageUri = getOutputUri(outputImage);
362
+ intent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);
363
+ intent.putExtra(MediaStore.EXTRA_OUTPUT, outputImageUri);
364
+ } catch (IOException | IllegalArgumentException e) {
365
+ Log.e("CREATE FILE", "Error occurred while creating the File", e);
366
+ e.printStackTrace();
367
+ }
368
+
369
+ return intent;
370
+ }
371
+
372
+ private Intent getVideoIntent() {
373
+ Intent intent = null;
374
+
375
+ try {
376
+ outputVideo = getCapturedFile(MimeType.VIDEO);
377
+ Uri outputVideoUri = getOutputUri(outputVideo);
378
+ intent = new Intent(MediaStore.ACTION_VIDEO_CAPTURE);
379
+ intent.putExtra(MediaStore.EXTRA_OUTPUT, outputVideoUri);
380
+ } catch (IOException | IllegalArgumentException e) {
381
+ Log.e("CREATE FILE", "Error occurred while creating the File", e);
382
+ e.printStackTrace();
383
+ }
384
+
385
+ return intent;
386
+ }
387
+
388
+ private Intent getFileChooserIntent(String acceptTypes) {
389
+ String _acceptTypes = acceptTypes;
390
+ if (acceptTypes.isEmpty()) {
391
+ _acceptTypes = MimeType.DEFAULT.value;
392
+ }
393
+ if (acceptTypes.matches("\\.\\w+")) {
394
+ _acceptTypes = getMimeTypeFromExtension(acceptTypes.replace(".", ""));
395
+ }
396
+ Intent intent = new Intent(Intent.ACTION_GET_CONTENT);
397
+ intent.addCategory(Intent.CATEGORY_OPENABLE);
398
+ intent.setType(_acceptTypes);
399
+ return intent;
400
+ }
401
+
402
+ private Intent getFileChooserIntent(String[] acceptTypes, boolean allowMultiple) {
403
+ Intent intent = new Intent(Intent.ACTION_GET_CONTENT);
404
+ intent.addCategory(Intent.CATEGORY_OPENABLE);
405
+ intent.setType(MimeType.DEFAULT.value);
406
+ intent.putExtra(Intent.EXTRA_MIME_TYPES, getAcceptedMimeType(acceptTypes));
407
+ intent.putExtra(Intent.EXTRA_ALLOW_MULTIPLE, allowMultiple);
408
+ return intent;
409
+ }
410
+
411
+ private Boolean acceptsImages(String types) {
412
+ String mimeType = types;
413
+ if (types.matches("\\.\\w+")) {
414
+ mimeType = getMimeTypeFromExtension(types.replace(".", ""));
415
+ }
416
+ return mimeType.isEmpty() || mimeType.toLowerCase().contains(MimeType.IMAGE.value);
417
+ }
418
+
419
+ private Boolean acceptsImages(String[] types) {
420
+ String[] mimeTypes = getAcceptedMimeType(types);
421
+ return arrayContainsString(mimeTypes, MimeType.DEFAULT.value) || arrayContainsString(mimeTypes, MimeType.IMAGE.value);
422
+ }
423
+
424
+ private Boolean acceptsVideo(String types) {
425
+ if (Build.VERSION.SDK_INT < Build.VERSION_CODES.M) {
426
+ return false;
427
+ }
428
+
429
+ String mimeType = types;
430
+ if (types.matches("\\.\\w+")) {
431
+ mimeType = getMimeTypeFromExtension(types.replace(".", ""));
432
+ }
433
+ return mimeType.isEmpty() || mimeType.toLowerCase().contains(MimeType.VIDEO.value);
434
+ }
435
+
436
+ private Boolean acceptsVideo(String[] types) {
437
+ if (Build.VERSION.SDK_INT < Build.VERSION_CODES.M) {
438
+ return false;
439
+ }
440
+
441
+ String[] mimeTypes = getAcceptedMimeType(types);
442
+ return arrayContainsString(mimeTypes, MimeType.DEFAULT.value) || arrayContainsString(mimeTypes, MimeType.VIDEO.value);
443
+ }
444
+
445
+ private Boolean arrayContainsString(String[] array, String pattern) {
446
+ for (String content : array) {
447
+ if (content.contains(pattern)) {
448
+ return true;
449
+ }
450
+ }
451
+ return false;
452
+ }
453
+
454
+ private String[] getAcceptedMimeType(String[] types) {
455
+ if (noAcceptTypesSet(types)) {
456
+ return new String[]{MimeType.DEFAULT.value};
457
+ }
458
+ String[] mimeTypes = new String[types.length];
459
+ for (int i = 0; i < types.length; i++) {
460
+ String t = types[i];
461
+ // convert file extensions to mime types
462
+ if (t.matches("\\.\\w+")) {
463
+ String mimeType = getMimeTypeFromExtension(t.replace(".", ""));
464
+ if(mimeType != null) {
465
+ mimeTypes[i] = mimeType;
466
+ } else {
467
+ mimeTypes[i] = t;
468
+ }
469
+ } else {
470
+ mimeTypes[i] = t;
471
+ }
472
+ }
473
+ return mimeTypes;
474
+ }
475
+
476
+ private String getMimeTypeFromExtension(String extension) {
477
+ String type = null;
478
+ if (extension != null) {
479
+ type = MimeTypeMap.getSingleton().getMimeTypeFromExtension(extension);
480
+ }
481
+ return type;
482
+ }
483
+
484
+ private Uri getOutputUri(File capturedFile) {
485
+ // for versions below 6.0 (23) we use the old File creation & permissions model
486
+ if (Build.VERSION.SDK_INT < Build.VERSION_CODES.M) {
487
+ return Uri.fromFile(capturedFile);
488
+ }
489
+
490
+ // for versions 6.0+ (23) we use the FileProvider to avoid runtime permissions
491
+ String packageName = getReactApplicationContext().getPackageName();
492
+ return FileProvider.getUriForFile(getReactApplicationContext(), packageName + ".fileprovider", capturedFile);
493
+ }
494
+
495
+ private File getCapturedFile(MimeType mimeType) throws IOException {
496
+ String prefix = "";
497
+ String suffix = "";
498
+ String dir = "";
499
+
500
+ switch (mimeType) {
501
+ case IMAGE:
502
+ prefix = "image-";
503
+ suffix = ".jpg";
504
+ dir = Environment.DIRECTORY_PICTURES;
505
+ break;
506
+ case VIDEO:
507
+ prefix = "video-";
508
+ suffix = ".mp4";
509
+ dir = Environment.DIRECTORY_MOVIES;
510
+ break;
511
+
512
+ default:
513
+ break;
514
+ }
515
+
516
+ String filename = prefix + String.valueOf(System.currentTimeMillis()) + suffix;
517
+ File outputFile = null;
518
+
519
+ // for versions below 6.0 (23) we use the old File creation & permissions model
520
+ if (Build.VERSION.SDK_INT < Build.VERSION_CODES.M) {
521
+ // only this Directory works on all tested Android versions
522
+ // ctx.getExternalFilesDir(dir) was failing on Android 5.0 (sdk 21)
523
+ File storageDir = Environment.getExternalStoragePublicDirectory(dir);
524
+ outputFile = new File(storageDir, filename);
525
+ } else {
526
+ File storageDir = getReactApplicationContext().getExternalFilesDir(null);
527
+ outputFile = File.createTempFile(prefix, suffix, storageDir);
528
+ }
529
+
530
+ return outputFile;
531
+ }
532
+
533
+ private Boolean noAcceptTypesSet(String[] types) {
534
+ // when our array returned from getAcceptTypes() has no values set from the webview
535
+ // i.e. <input type="file" />, without any "accept" attr
536
+ // will be an array with one empty string element, afaik
537
+
538
+ return types.length == 0 || (types.length == 1 && types[0] != null && types[0].length() == 0);
539
+ }
540
+
541
+ private PermissionAwareActivity getPermissionAwareActivity() {
542
+ Activity activity = getCurrentActivity();
543
+ if (activity == null) {
544
+ throw new IllegalStateException("Tried to use permissions API while not attached to an Activity.");
545
+ } else if (!(activity instanceof PermissionAwareActivity)) {
546
+ throw new IllegalStateException("Tried to use permissions API but the host Activity doesn't implement PermissionAwareActivity.");
547
+ }
548
+ return (PermissionAwareActivity) activity;
549
+ }
550
+ }
@@ -0,0 +1,15 @@
1
+ package com.reactnativecommunity.webview
2
+
3
+ import com.facebook.react.ReactPackage
4
+ import com.facebook.react.bridge.ReactApplicationContext
5
+
6
+
7
+ class RNCWebViewPackage: ReactPackage {
8
+ override fun createNativeModules(reactContext: ReactApplicationContext) = listOf(
9
+ RNCWebViewModule(reactContext)
10
+ )
11
+
12
+ override fun createViewManagers(reactContext: ReactApplicationContext) = listOf(
13
+ RNCWebViewManager()
14
+ )
15
+ }
@@ -0,0 +1,12 @@
1
+ package com.reactnativecommunity.webview;
2
+
3
+ import android.webkit.WebView;
4
+
5
+ /**
6
+ * Implement this interface in order to config your {@link WebView}. An instance of that
7
+ * implementation will have to be given as a constructor argument to {@link RNCWebViewManager}.
8
+ */
9
+ public interface WebViewConfig {
10
+
11
+ void configWebView(WebView webView);
12
+ }
@@ -0,0 +1,25 @@
1
+ package com.reactnativecommunity.webview.events
2
+
3
+ import com.facebook.react.bridge.WritableMap
4
+ import com.facebook.react.uimanager.events.Event
5
+ import com.facebook.react.uimanager.events.RCTEventEmitter
6
+
7
+ /**
8
+ * Event emitted when a http error is received from the server.
9
+ */
10
+ class TopHttpErrorEvent(viewId: Int, private val mEventData: WritableMap) :
11
+ Event<TopHttpErrorEvent>(viewId) {
12
+ companion object {
13
+ const val EVENT_NAME = "topHttpError"
14
+ }
15
+
16
+ override fun getEventName(): String = EVENT_NAME
17
+
18
+ override fun canCoalesce(): Boolean = false
19
+
20
+ override fun getCoalescingKey(): Short = 0
21
+
22
+ override fun dispatch(rctEventEmitter: RCTEventEmitter) =
23
+ rctEventEmitter.receiveEvent(viewTag, eventName, mEventData)
24
+
25
+ }
@@ -0,0 +1,25 @@
1
+ package com.reactnativecommunity.webview.events
2
+
3
+ import com.facebook.react.bridge.WritableMap
4
+ import com.facebook.react.uimanager.events.Event
5
+ import com.facebook.react.uimanager.events.RCTEventEmitter
6
+
7
+ /**
8
+ * Event emitted when there is an error in loading.
9
+ */
10
+ class TopLoadingErrorEvent(viewId: Int, private val mEventData: WritableMap) :
11
+ Event<TopLoadingErrorEvent>(viewId) {
12
+ companion object {
13
+ const val EVENT_NAME = "topLoadingError"
14
+ }
15
+
16
+ override fun getEventName(): String = EVENT_NAME
17
+
18
+ override fun canCoalesce(): Boolean = false
19
+
20
+ override fun getCoalescingKey(): Short = 0
21
+
22
+ override fun dispatch(rctEventEmitter: RCTEventEmitter) =
23
+ rctEventEmitter.receiveEvent(viewTag, eventName, mEventData)
24
+
25
+ }
@@ -0,0 +1,24 @@
1
+ package com.reactnativecommunity.webview.events
2
+
3
+ import com.facebook.react.bridge.WritableMap
4
+ import com.facebook.react.uimanager.events.Event
5
+ import com.facebook.react.uimanager.events.RCTEventEmitter
6
+
7
+ /**
8
+ * Event emitted when loading is completed.
9
+ */
10
+ class TopLoadingFinishEvent(viewId: Int, private val mEventData: WritableMap) :
11
+ Event<TopLoadingFinishEvent>(viewId) {
12
+ companion object {
13
+ const val EVENT_NAME = "topLoadingFinish"
14
+ }
15
+
16
+ override fun getEventName(): String = EVENT_NAME
17
+
18
+ override fun canCoalesce(): Boolean = false
19
+
20
+ override fun getCoalescingKey(): Short = 0
21
+
22
+ override fun dispatch(rctEventEmitter: RCTEventEmitter) =
23
+ rctEventEmitter.receiveEvent(viewTag, eventName, mEventData)
24
+ }
@@ -0,0 +1,24 @@
1
+ package com.reactnativecommunity.webview.events
2
+
3
+ import com.facebook.react.bridge.WritableMap
4
+ import com.facebook.react.uimanager.events.Event
5
+ import com.facebook.react.uimanager.events.RCTEventEmitter
6
+
7
+ /**
8
+ * Event emitted when there is a loading progress event.
9
+ */
10
+ class TopLoadingProgressEvent(viewId: Int, private val mEventData: WritableMap) :
11
+ Event<TopLoadingProgressEvent>(viewId) {
12
+ companion object {
13
+ const val EVENT_NAME = "topLoadingProgress"
14
+ }
15
+
16
+ override fun getEventName(): String = EVENT_NAME
17
+
18
+ override fun canCoalesce(): Boolean = false
19
+
20
+ override fun getCoalescingKey(): Short = 0
21
+
22
+ override fun dispatch(rctEventEmitter: RCTEventEmitter) =
23
+ rctEventEmitter.receiveEvent(viewTag, eventName, mEventData)
24
+ }