The JUCE cross-platform C++ framework, with DISTRHO/KXStudio specific changes
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

843 lines
37KB

  1. /*
  2. ==============================================================================
  3. This file is part of the JUCE library.
  4. Copyright (c) 2017 - ROLI Ltd.
  5. JUCE is an open source library subject to commercial or open-source
  6. licensing.
  7. By using JUCE, you agree to the terms of both the JUCE 5 End-User License
  8. Agreement and JUCE 5 Privacy Policy (both updated and effective as of the
  9. 27th April 2017).
  10. End User License Agreement: www.juce.com/juce-5-licence
  11. Privacy Policy: www.juce.com/juce-5-privacy-policy
  12. Or: You may also use this code under the terms of the GPL v3 (see
  13. www.gnu.org/licenses).
  14. JUCE IS PROVIDED "AS IS" WITHOUT ANY WARRANTY, AND ALL WARRANTIES, WHETHER
  15. EXPRESSED OR IMPLIED, INCLUDING MERCHANTABILITY AND FITNESS FOR PURPOSE, ARE
  16. DISCLAIMED.
  17. ==============================================================================
  18. */
  19. namespace juce
  20. {
  21. //==============================================================================
  22. // This byte-code is generated from native/java/app/com/roli/juce/JuceContentProviderCursor.java with min sdk version 16
  23. // See juce_core/native/java/README.txt on how to generate this byte-code.
  24. static const uint8 javaJuceContentProviderCursor[] =
  25. { 31,139,8,8,138,122,94,94,0,3,74,117,99,101,67,111,110,116,101,110,116,80,114,111,118,105,100,101,114,67,117,114,115,111,114,46,100,101,120,0,117,147,177,111,211,64,20,198,223,157,157,148,150,52,184,144,1,6,32,160,46,29,138,43,209,178,132,34,80,144,42,
  26. 89,6,85,10,164,82,59,93,109,211,184,77,124,145,237,68,17,98,0,4,91,6,6,38,38,64,133,153,255,0,33,133,255,131,137,149,133,177,3,223,229,206,37,18,194,210,207,247,222,229,189,239,157,206,95,194,104,180,176,118,115,131,142,7,95,175,127,216,250,229,236,28,
  27. 61,169,85,222,124,219,126,234,140,155,159,157,142,184,107,17,245,137,104,212,94,63,79,230,121,207,137,28,210,251,37,240,5,216,96,2,24,248,109,246,79,76,190,140,215,9,184,15,222,129,143,224,19,248,14,126,130,37,232,213,192,69,112,25,92,3,183,192,38,240,
  28. 64,4,142,192,144,107,61,203,204,83,51,230,205,106,153,179,149,77,252,10,133,115,38,30,35,62,99,226,183,136,33,67,139,211,179,233,46,165,185,64,106,159,79,115,126,154,51,170,152,124,201,212,21,191,171,103,108,130,18,78,163,194,103,92,207,239,215,137,238,
  29. 209,46,238,171,138,9,122,202,107,174,207,42,29,134,218,22,42,250,107,232,98,213,127,116,237,105,110,83,249,118,156,196,249,29,98,30,93,241,6,65,212,148,73,30,37,249,118,42,135,113,24,165,205,65,154,201,244,198,161,24,10,186,234,139,36,76,101,28,186,161,
  30. 200,197,190,200,34,247,129,200,211,120,164,139,26,180,226,7,178,231,166,178,27,187,135,144,114,255,171,215,32,214,38,222,246,200,106,123,62,2,159,46,236,249,106,134,219,21,201,129,219,130,104,114,208,160,82,208,149,89,68,103,3,217,29,244,146,135,162,
  31. 23,101,116,41,208,130,173,142,72,11,185,166,42,11,201,238,200,44,167,121,245,126,36,31,163,209,206,59,113,70,21,126,174,50,87,221,92,165,26,97,217,197,69,112,92,14,127,241,220,158,112,254,146,147,197,126,240,69,115,55,108,102,45,188,199,103,252,87,120,
  32. 98,98,214,89,31,22,94,44,207,248,145,213,117,191,242,36,115,254,122,134,215,181,190,242,169,101,106,212,119,165,186,238,157,126,115,71,199,234,127,240,7,124,122,243,207,64,3,0,0,0,0 };
  33. // This byte-code is generated from native/java/app/com/roli/juce/JuceContentProviderFileObserver.java with min sdk version 16
  34. // See juce_core/native/java/README.txt on how to generate this byte-code.
  35. static const uint8 javaJuceContentProviderFileObserver[] =
  36. { 31,139,8,8,123,122,94,94,0,3,74,117,99,101,67,111,110,116,101,110,116,80,114,111,118,105,100,101,114,70,105,108,101,79,98,115,101,114,118,101,114,46,100,101,120,0,133,147,205,107,19,65,24,198,223,249,72,98,91,93,210,84,4,17,36,130,130,32,186,145,218,
  37. 131,196,10,197,47,18,22,44,166,6,41,120,216,38,75,119,107,186,19,118,215,144,131,5,45,61,148,158,188,244,164,71,65,241,212,163,40,138,136,224,213,91,253,3,60,250,7,120,211,103,118,166,116,61,25,248,205,251,206,251,206,60,59,59,251,164,31,140,39,27,179,
  38. 115,116,110,243,212,238,147,208,189,112,245,227,236,183,157,19,103,216,220,215,223,91,251,175,247,249,88,16,13,137,104,220,189,82,35,251,251,193,137,142,147,169,151,193,103,32,117,29,48,112,132,153,121,149,153,121,3,195,121,236,89,68,220,3,239,192,7,
  39. 240,9,252,4,127,64,3,253,155,224,14,240,192,61,240,0,60,4,27,96,11,236,128,93,240,2,188,2,111,185,209,231,246,249,37,123,158,138,62,131,157,111,99,193,132,205,159,35,159,180,249,75,228,83,54,127,131,92,64,201,201,181,68,254,142,34,87,53,245,163,121,36,
  40. 154,46,68,110,239,66,216,248,197,22,202,80,212,181,61,110,206,50,108,16,45,148,150,113,119,78,222,209,207,123,111,123,157,58,250,75,156,248,101,199,234,201,124,47,179,239,99,114,153,119,14,222,177,124,45,138,163,236,58,177,22,177,54,157,109,63,238,5,
  41. 55,84,156,5,113,182,152,168,81,212,15,146,219,209,32,184,187,146,6,201,40,72,46,173,249,35,159,78,122,126,220,79,84,212,119,85,234,22,219,77,114,189,158,90,119,19,53,136,220,53,72,185,255,209,107,82,205,211,146,238,192,143,87,221,78,150,68,241,106,147,
  42. 88,151,68,183,229,145,236,182,205,232,181,80,192,112,186,103,164,58,161,159,252,43,116,107,132,50,149,130,60,200,80,165,25,77,232,113,73,221,79,3,146,235,126,250,136,42,42,54,203,228,208,207,66,146,89,24,165,52,37,170,51,211,21,103,254,34,213,184,51,
  43. 83,113,150,113,69,156,17,103,207,158,202,239,156,109,226,251,177,95,184,172,99,246,187,176,66,60,240,44,47,248,86,20,188,43,11,254,45,209,161,135,203,116,232,99,86,55,251,181,151,89,213,172,209,30,227,117,163,175,253,45,236,26,237,1,170,155,189,185,63,
  44. 170,38,215,255,159,191,29,121,169,41,120,3,0,0,0,0 };
  45. //==============================================================================
  46. #define JNI_CLASS_MEMBERS(METHOD, STATICMETHOD, FIELD, STATICFIELD, CALLBACK) \
  47. FIELD (authority, "authority", "Ljava/lang/String;")
  48. DECLARE_JNI_CLASS (AndroidProviderInfo, "android/content/pm/ProviderInfo")
  49. #undef JNI_CLASS_MEMBERS
  50. #define JNI_CLASS_MEMBERS(METHOD, STATICMETHOD, FIELD, STATICFIELD, CALLBACK) \
  51. METHOD (constructor, "<init>", "(Landroid/os/ParcelFileDescriptor;JJ)V") \
  52. METHOD (createInputStream, "createInputStream", "()Ljava/io/FileInputStream;") \
  53. METHOD (getLength, "getLength", "()J")
  54. DECLARE_JNI_CLASS (AssetFileDescriptor, "android/content/res/AssetFileDescriptor")
  55. #undef JNI_CLASS_MEMBERS
  56. #define JNI_CLASS_MEMBERS(METHOD, STATICMETHOD, FIELD, STATICFIELD, CALLBACK) \
  57. METHOD (close, "close", "()V")
  58. DECLARE_JNI_CLASS (JavaCloseable, "java/io/Closeable")
  59. #undef JNI_CLASS_MEMBERS
  60. #define JNI_CLASS_MEMBERS(METHOD, STATICMETHOD, FIELD, STATICFIELD, CALLBACK) \
  61. STATICMETHOD (open, "open", "(Ljava/io/File;I)Landroid/os/ParcelFileDescriptor;")
  62. DECLARE_JNI_CLASS (ParcelFileDescriptor, "android/os/ParcelFileDescriptor")
  63. #undef JNI_CLASS_MEMBERS
  64. //==============================================================================
  65. class AndroidContentSharerCursor
  66. {
  67. public:
  68. class Owner
  69. {
  70. public:
  71. virtual ~Owner() {}
  72. virtual void cursorClosed (const AndroidContentSharerCursor&) = 0;
  73. };
  74. AndroidContentSharerCursor (Owner& ownerToUse, JNIEnv* env,
  75. const LocalRef<jobject>& contentProvider,
  76. const LocalRef<jobjectArray>& resultColumns)
  77. : owner (ownerToUse),
  78. cursor (GlobalRef (LocalRef<jobject> (env->NewObject (JuceContentProviderCursor,
  79. JuceContentProviderCursor.constructor,
  80. reinterpret_cast<jlong> (this),
  81. resultColumns.get()))))
  82. {
  83. // the content provider must be created first
  84. jassert (contentProvider.get() != nullptr);
  85. }
  86. jobject getNativeCursor() { return cursor.get(); }
  87. void cursorClosed()
  88. {
  89. MessageManager::callAsync ([this] { owner.cursorClosed (*this); });
  90. }
  91. void addRow (LocalRef<jobjectArray>& values)
  92. {
  93. auto* env = getEnv();
  94. env->CallVoidMethod (cursor.get(), JuceContentProviderCursor.addRow, values.get());
  95. }
  96. private:
  97. Owner& owner;
  98. GlobalRef cursor;
  99. //==============================================================================
  100. #define JNI_CLASS_MEMBERS(METHOD, STATICMETHOD, FIELD, STATICFIELD, CALLBACK) \
  101. METHOD (addRow, "addRow", "([Ljava/lang/Object;)V") \
  102. METHOD (constructor, "<init>", "(J[Ljava/lang/String;)V") \
  103. CALLBACK (contentSharerCursorClosed, "contentSharerCursorClosed", "(J)V") \
  104. DECLARE_JNI_CLASS_WITH_BYTECODE (JuceContentProviderCursor, "com/roli/juce/JuceContentProviderCursor", 16, javaJuceContentProviderCursor, sizeof (javaJuceContentProviderCursor))
  105. #undef JNI_CLASS_MEMBERS
  106. static void JNICALL contentSharerCursorClosed (JNIEnv*, jobject, jlong host)
  107. {
  108. if (auto* myself = reinterpret_cast<AndroidContentSharerCursor*> (host))
  109. myself->cursorClosed();
  110. }
  111. };
  112. AndroidContentSharerCursor::JuceContentProviderCursor_Class AndroidContentSharerCursor::JuceContentProviderCursor;
  113. //==============================================================================
  114. class AndroidContentSharerFileObserver
  115. {
  116. public:
  117. class Owner
  118. {
  119. public:
  120. virtual ~Owner() {}
  121. virtual void fileHandleClosed (const AndroidContentSharerFileObserver&) = 0;
  122. };
  123. AndroidContentSharerFileObserver (Owner& ownerToUse, JNIEnv* env,
  124. const LocalRef<jobject>& contentProvider,
  125. const String& filepathToUse)
  126. : owner (ownerToUse),
  127. filepath (filepathToUse),
  128. fileObserver (GlobalRef (LocalRef<jobject> (env->NewObject (JuceContentProviderFileObserver,
  129. JuceContentProviderFileObserver.constructor,
  130. reinterpret_cast<jlong> (this),
  131. javaString (filepath).get(),
  132. open | access | closeWrite | closeNoWrite))))
  133. {
  134. // the content provider must be created first
  135. jassert (contentProvider.get() != nullptr);
  136. env->CallVoidMethod (fileObserver, JuceContentProviderFileObserver.startWatching);
  137. }
  138. void onFileEvent (int event, const LocalRef<jstring>& path)
  139. {
  140. ignoreUnused (path);
  141. if (event == open)
  142. {
  143. ++numOpenedHandles;
  144. }
  145. else if (event == access)
  146. {
  147. fileWasRead = true;
  148. }
  149. else if (event == closeNoWrite || event == closeWrite)
  150. {
  151. --numOpenedHandles;
  152. // numOpenedHandles may get negative if we don't receive open handle event.
  153. if (fileWasRead && numOpenedHandles <= 0)
  154. {
  155. MessageManager::callAsync ([this]
  156. {
  157. getEnv()->CallVoidMethod (fileObserver, JuceContentProviderFileObserver.stopWatching);
  158. owner.fileHandleClosed (*this);
  159. });
  160. }
  161. }
  162. }
  163. private:
  164. static constexpr int open = 32;
  165. static constexpr int access = 1;
  166. static constexpr int closeWrite = 8;
  167. static constexpr int closeNoWrite = 16;
  168. bool fileWasRead = false;
  169. int numOpenedHandles = 0;
  170. Owner& owner;
  171. String filepath;
  172. GlobalRef fileObserver;
  173. //==============================================================================
  174. #define JNI_CLASS_MEMBERS(METHOD, STATICMETHOD, FIELD, STATICFIELD, CALLBACK) \
  175. METHOD (constructor, "<init>", "(JLjava/lang/String;I)V") \
  176. METHOD (startWatching, "startWatching", "()V") \
  177. METHOD (stopWatching, "stopWatching", "()V") \
  178. CALLBACK (contentSharerFileObserverEvent, "contentSharerFileObserverEvent", "(JILjava/lang/String;)V") \
  179. DECLARE_JNI_CLASS_WITH_BYTECODE (JuceContentProviderFileObserver, "com/roli/juce/JuceContentProviderFileObserver", 16, javaJuceContentProviderFileObserver, sizeof (javaJuceContentProviderFileObserver))
  180. #undef JNI_CLASS_MEMBERS
  181. static void JNICALL contentSharerFileObserverEvent (JNIEnv*, jobject /*fileObserver*/, jlong host, int event, jstring path)
  182. {
  183. if (auto* myself = reinterpret_cast<AndroidContentSharerFileObserver*> (host))
  184. myself->onFileEvent (event, LocalRef<jstring> (path));
  185. }
  186. };
  187. AndroidContentSharerFileObserver::JuceContentProviderFileObserver_Class AndroidContentSharerFileObserver::JuceContentProviderFileObserver;
  188. //==============================================================================
  189. class AndroidContentSharerPrepareFilesThread : private Thread
  190. {
  191. public:
  192. AndroidContentSharerPrepareFilesThread (AsyncUpdater& ownerToUse,
  193. const Array<URL>& fileUrlsToUse,
  194. const String& packageNameToUse,
  195. const String& uriBaseToUse)
  196. : Thread ("AndroidContentSharerPrepareFilesThread"),
  197. owner (ownerToUse),
  198. fileUrls (fileUrlsToUse),
  199. resultFileUris (GlobalRef (LocalRef<jobject> (getEnv()->NewObject (JavaArrayList,
  200. JavaArrayList.constructor,
  201. fileUrls.size())))),
  202. packageName (packageNameToUse),
  203. uriBase (uriBaseToUse)
  204. {
  205. startThread();
  206. }
  207. ~AndroidContentSharerPrepareFilesThread() override
  208. {
  209. signalThreadShouldExit();
  210. waitForThreadToExit (10000);
  211. for (auto& f : temporaryFilesFromAssetFiles)
  212. f.deleteFile();
  213. }
  214. jobject getResultFileUris() { return resultFileUris.get(); }
  215. const StringArray& getMimeTypes() const { return mimeTypes; }
  216. const StringArray& getFilePaths() const { return filePaths; }
  217. private:
  218. struct StreamCloser
  219. {
  220. StreamCloser (const LocalRef<jobject>& streamToUse)
  221. : stream (GlobalRef (streamToUse))
  222. {
  223. }
  224. ~StreamCloser()
  225. {
  226. if (stream.get() != nullptr)
  227. getEnv()->CallVoidMethod (stream, JavaCloseable.close);
  228. }
  229. GlobalRef stream;
  230. };
  231. void run() override
  232. {
  233. auto* env = getEnv();
  234. bool canSpecifyMimeTypes = true;
  235. for (auto f : fileUrls)
  236. {
  237. auto scheme = f.getScheme();
  238. // Only "file://" scheme or no scheme (for files in app bundle) are allowed!
  239. jassert (scheme.isEmpty() || scheme == "file");
  240. if (scheme.isEmpty())
  241. {
  242. // Raw resource names need to be all lower case
  243. jassert (f.toString (true).toLowerCase() == f.toString (true));
  244. // This will get us a file with file:// URI
  245. f = copyAssetFileToTemporaryFile (env, f.toString (true));
  246. if (f.isEmpty())
  247. continue;
  248. }
  249. if (threadShouldExit())
  250. return;
  251. auto filepath = URL::removeEscapeChars (f.toString (true).fromFirstOccurrenceOf ("file://", false, false));
  252. filePaths.add (filepath);
  253. auto filename = filepath.fromLastOccurrenceOf ("/", false, true);
  254. auto fileExtension = filename.fromLastOccurrenceOf (".", false, true);
  255. auto contentString = uriBase + String (filePaths.size() - 1) + "/" + filename;
  256. auto uri = LocalRef<jobject> (env->CallStaticObjectMethod (AndroidUri, AndroidUri.parse,
  257. javaString (contentString).get()));
  258. if (canSpecifyMimeTypes)
  259. canSpecifyMimeTypes = fileExtension.isNotEmpty();
  260. if (canSpecifyMimeTypes)
  261. mimeTypes.addArray (getMimeTypesForFileExtension (fileExtension));
  262. else
  263. mimeTypes.clear();
  264. env->CallBooleanMethod (resultFileUris, JavaArrayList.add, uri.get());
  265. }
  266. owner.triggerAsyncUpdate();
  267. }
  268. URL copyAssetFileToTemporaryFile (JNIEnv* env, const String& filename)
  269. {
  270. auto resources = LocalRef<jobject> (env->CallObjectMethod (getAppContext().get(), AndroidContext.getResources));
  271. int fileId = env->CallIntMethod (resources, AndroidResources.getIdentifier, javaString (filename).get(),
  272. javaString ("raw").get(), javaString (packageName).get());
  273. // Raw resource not found. Please make sure that you include your file as a raw resource
  274. // and that you specify just the file name, without an extension.
  275. jassert (fileId != 0);
  276. if (fileId == 0)
  277. return {};
  278. auto assetFd = LocalRef<jobject> (env->CallObjectMethod (resources,
  279. AndroidResources.openRawResourceFd,
  280. fileId));
  281. auto inputStream = StreamCloser (LocalRef<jobject> (env->CallObjectMethod (assetFd,
  282. AssetFileDescriptor.createInputStream)));
  283. if (jniCheckHasExceptionOccurredAndClear())
  284. {
  285. // Failed to open file stream for resource
  286. jassertfalse;
  287. return {};
  288. }
  289. auto tempFile = File::createTempFile ({});
  290. tempFile.createDirectory();
  291. tempFile = tempFile.getChildFile (filename);
  292. auto outputStream = StreamCloser (LocalRef<jobject> (env->NewObject (JavaFileOutputStream,
  293. JavaFileOutputStream.constructor,
  294. javaString (tempFile.getFullPathName()).get())));
  295. if (jniCheckHasExceptionOccurredAndClear())
  296. {
  297. // Failed to open file stream for temporary file
  298. jassertfalse;
  299. return {};
  300. }
  301. auto buffer = LocalRef<jbyteArray> (env->NewByteArray (1024));
  302. int bytesRead = 0;
  303. for (;;)
  304. {
  305. if (threadShouldExit())
  306. return {};
  307. bytesRead = env->CallIntMethod (inputStream.stream, JavaFileInputStream.read, buffer.get());
  308. if (jniCheckHasExceptionOccurredAndClear())
  309. {
  310. // Failed to read from resource file.
  311. jassertfalse;
  312. return {};
  313. }
  314. if (bytesRead < 0)
  315. break;
  316. env->CallVoidMethod (outputStream.stream, JavaFileOutputStream.write, buffer.get(), 0, bytesRead);
  317. if (jniCheckHasExceptionOccurredAndClear())
  318. {
  319. // Failed to write to temporary file.
  320. jassertfalse;
  321. return {};
  322. }
  323. }
  324. temporaryFilesFromAssetFiles.add (tempFile);
  325. return URL (tempFile);
  326. }
  327. AsyncUpdater& owner;
  328. Array<URL> fileUrls;
  329. GlobalRef resultFileUris;
  330. String packageName;
  331. String uriBase;
  332. StringArray filePaths;
  333. Array<File> temporaryFilesFromAssetFiles;
  334. StringArray mimeTypes;
  335. };
  336. //==============================================================================
  337. class ContentSharer::ContentSharerNativeImpl : public ContentSharer::Pimpl,
  338. public AndroidContentSharerFileObserver::Owner,
  339. public AndroidContentSharerCursor::Owner,
  340. public AsyncUpdater,
  341. private Timer
  342. {
  343. public:
  344. ContentSharerNativeImpl (ContentSharer& cs)
  345. : owner (cs),
  346. packageName (juceString (LocalRef<jstring> ((jstring) getEnv()->CallObjectMethod (getAppContext().get(), AndroidContext.getPackageName)))),
  347. uriBase ("content://" + packageName + ".sharingcontentprovider/")
  348. {
  349. }
  350. ~ContentSharerNativeImpl() override
  351. {
  352. masterReference.clear();
  353. }
  354. void shareFiles (const Array<URL>& files) override
  355. {
  356. if (! isContentSharingEnabled())
  357. {
  358. // You need to enable "Content Sharing" in Projucer's Android exporter.
  359. jassertfalse;
  360. owner.sharingFinished (false, {});
  361. }
  362. prepareFilesThread.reset (new AndroidContentSharerPrepareFilesThread (*this, files, packageName, uriBase));
  363. }
  364. void shareText (const String& text) override
  365. {
  366. if (! isContentSharingEnabled())
  367. {
  368. // You need to enable "Content Sharing" in Projucer's Android exporter.
  369. jassertfalse;
  370. owner.sharingFinished (false, {});
  371. }
  372. auto* env = getEnv();
  373. auto intent = LocalRef<jobject> (env->NewObject (AndroidIntent, AndroidIntent.constructor));
  374. env->CallObjectMethod (intent, AndroidIntent.setAction,
  375. javaString ("android.intent.action.SEND").get());
  376. env->CallObjectMethod (intent, AndroidIntent.putExtra,
  377. javaString ("android.intent.extra.TEXT").get(),
  378. javaString (text).get());
  379. env->CallObjectMethod (intent, AndroidIntent.setType, javaString ("text/plain").get());
  380. auto chooserIntent = LocalRef<jobject> (env->CallStaticObjectMethod (AndroidIntent, AndroidIntent.createChooser,
  381. intent.get(), javaString ("Choose share target").get()));
  382. WeakReference<ContentSharerNativeImpl> weakRef (this);
  383. startAndroidActivityForResult (chooserIntent, 1003,
  384. [weakRef] (int /*requestCode*/, int resultCode, LocalRef<jobject> /*intentData*/) mutable
  385. {
  386. if (weakRef != nullptr)
  387. weakRef->sharingFinished (resultCode);
  388. });
  389. }
  390. //==============================================================================
  391. void cursorClosed (const AndroidContentSharerCursor& cursor) override
  392. {
  393. cursors.removeObject (&cursor);
  394. }
  395. void fileHandleClosed (const AndroidContentSharerFileObserver&) override
  396. {
  397. decrementPendingFileCountAndNotifyOwnerIfReady();
  398. }
  399. //==============================================================================
  400. jobject openFile (const LocalRef<jobject>& contentProvider,
  401. const LocalRef<jobject>& uri, const LocalRef<jstring>& mode)
  402. {
  403. ignoreUnused (mode);
  404. WeakReference<ContentSharerNativeImpl> weakRef (this);
  405. if (weakRef == nullptr)
  406. return nullptr;
  407. auto* env = getEnv();
  408. auto uriElements = getContentUriElements (env, uri);
  409. if (uriElements.filepath.isEmpty())
  410. return nullptr;
  411. return getAssetFileDescriptor (env, contentProvider, uriElements.filepath);
  412. }
  413. jobject query (const LocalRef<jobject>& contentProvider, const LocalRef<jobject>& uri,
  414. const LocalRef<jobjectArray>& projection)
  415. {
  416. StringArray requestedColumns = javaStringArrayToJuce (projection);
  417. StringArray supportedColumns = getSupportedColumns();
  418. StringArray resultColumns;
  419. for (const auto& col : supportedColumns)
  420. {
  421. if (requestedColumns.contains (col))
  422. resultColumns.add (col);
  423. }
  424. // Unsupported columns were queried, file sharing may fail.
  425. if (resultColumns.isEmpty())
  426. return nullptr;
  427. auto resultJavaColumns = juceStringArrayToJava (resultColumns);
  428. auto* env = getEnv();
  429. auto cursor = cursors.add (new AndroidContentSharerCursor (*this, env, contentProvider,
  430. resultJavaColumns));
  431. auto uriElements = getContentUriElements (env, uri);
  432. if (uriElements.filepath.isEmpty())
  433. return cursor->getNativeCursor();
  434. auto values = LocalRef<jobjectArray> (env->NewObjectArray ((jsize) resultColumns.size(),
  435. JavaObject, nullptr));
  436. for (int i = 0; i < resultColumns.size(); ++i)
  437. {
  438. if (resultColumns.getReference (i) == "_display_name")
  439. {
  440. env->SetObjectArrayElement (values, i, javaString (uriElements.filename).get());
  441. }
  442. else if (resultColumns.getReference (i) == "_size")
  443. {
  444. auto javaFile = LocalRef<jobject> (env->NewObject (JavaFile, JavaFile.constructor,
  445. javaString (uriElements.filepath).get()));
  446. jlong fileLength = env->CallLongMethod (javaFile, JavaFile.length);
  447. env->SetObjectArrayElement (values, i, env->NewObject (JavaLong,
  448. JavaLong.constructor,
  449. fileLength));
  450. }
  451. }
  452. cursor->addRow (values);
  453. return cursor->getNativeCursor();
  454. }
  455. jobjectArray getStreamTypes (const LocalRef<jobject>& uri, const LocalRef<jstring>& mimeTypeFilter)
  456. {
  457. auto* env = getEnv();
  458. auto extension = getContentUriElements (env, uri).filename.fromLastOccurrenceOf (".", false, true);
  459. if (extension.isEmpty())
  460. return nullptr;
  461. return juceStringArrayToJava (filterMimeTypes (getMimeTypesForFileExtension (extension),
  462. juceString (mimeTypeFilter.get())));
  463. }
  464. void sharingFinished (int resultCode)
  465. {
  466. sharingActivityDidFinish = true;
  467. succeeded = resultCode == -1;
  468. // Give content sharer a chance to request file access.
  469. if (nonAssetFilesPendingShare.get() == 0)
  470. startTimer (2000);
  471. else
  472. notifyOwnerIfReady();
  473. }
  474. private:
  475. bool isContentSharingEnabled() const
  476. {
  477. auto* env = getEnv();
  478. LocalRef<jobject> packageManager (env->CallObjectMethod (getAppContext().get(), AndroidContext.getPackageManager));
  479. constexpr int getProviders = 8;
  480. auto packageInfo = LocalRef<jobject> (env->CallObjectMethod (packageManager,
  481. AndroidPackageManager.getPackageInfo,
  482. javaString (packageName).get(),
  483. getProviders));
  484. auto providers = LocalRef<jobjectArray> ((jobjectArray) env->GetObjectField (packageInfo,
  485. AndroidPackageInfo.providers));
  486. if (providers == nullptr)
  487. return false;
  488. auto sharingContentProviderAuthority = packageName + ".sharingcontentprovider";
  489. const int numProviders = env->GetArrayLength (providers.get());
  490. for (int i = 0; i < numProviders; ++i)
  491. {
  492. auto providerInfo = LocalRef<jobject> (env->GetObjectArrayElement (providers, i));
  493. auto authority = LocalRef<jstring> ((jstring) env->GetObjectField (providerInfo,
  494. AndroidProviderInfo.authority));
  495. if (juceString (authority) == sharingContentProviderAuthority)
  496. return true;
  497. }
  498. return false;
  499. }
  500. void handleAsyncUpdate() override
  501. {
  502. jassert (prepareFilesThread != nullptr);
  503. if (prepareFilesThread == nullptr)
  504. return;
  505. filesPrepared (prepareFilesThread->getResultFileUris(), prepareFilesThread->getMimeTypes());
  506. }
  507. void filesPrepared (jobject fileUris, const StringArray& mimeTypes)
  508. {
  509. auto* env = getEnv();
  510. auto intent = LocalRef<jobject> (env->NewObject (AndroidIntent, AndroidIntent.constructor));
  511. env->CallObjectMethod (intent, AndroidIntent.setAction,
  512. javaString ("android.intent.action.SEND_MULTIPLE").get());
  513. env->CallObjectMethod (intent, AndroidIntent.setType,
  514. javaString (getCommonMimeType (mimeTypes)).get());
  515. constexpr int grantReadPermission = 1;
  516. env->CallObjectMethod (intent, AndroidIntent.setFlags, grantReadPermission);
  517. env->CallObjectMethod (intent, AndroidIntent.putParcelableArrayListExtra,
  518. javaString ("android.intent.extra.STREAM").get(),
  519. fileUris);
  520. auto chooserIntent = LocalRef<jobject> (env->CallStaticObjectMethod (AndroidIntent,
  521. AndroidIntent.createChooser,
  522. intent.get(),
  523. javaString ("Choose share target").get()));
  524. WeakReference<ContentSharerNativeImpl> weakRef (this);
  525. startAndroidActivityForResult (chooserIntent, 1003,
  526. [weakRef] (int /*requestCode*/, int resultCode, LocalRef<jobject> /*intentData*/) mutable
  527. {
  528. if (weakRef != nullptr)
  529. weakRef->sharingFinished (resultCode);
  530. });
  531. }
  532. void decrementPendingFileCountAndNotifyOwnerIfReady()
  533. {
  534. --nonAssetFilesPendingShare;
  535. notifyOwnerIfReady();
  536. }
  537. void notifyOwnerIfReady()
  538. {
  539. if (sharingActivityDidFinish && nonAssetFilesPendingShare.get() == 0)
  540. owner.sharingFinished (succeeded, {});
  541. }
  542. void timerCallback() override
  543. {
  544. stopTimer();
  545. notifyOwnerIfReady();
  546. }
  547. //==============================================================================
  548. struct ContentUriElements
  549. {
  550. String index;
  551. String filename;
  552. String filepath;
  553. };
  554. ContentUriElements getContentUriElements (JNIEnv* env, const LocalRef<jobject>& uri) const
  555. {
  556. jassert (prepareFilesThread != nullptr);
  557. if (prepareFilesThread == nullptr)
  558. return {};
  559. auto fullUri = juceString ((jstring) env->CallObjectMethod (uri.get(), AndroidUri.toString));
  560. auto index = fullUri.fromFirstOccurrenceOf (uriBase, false, false)
  561. .upToFirstOccurrenceOf ("/", false, true);
  562. auto filename = fullUri.fromLastOccurrenceOf ("/", false, true);
  563. return { index, filename, prepareFilesThread->getFilePaths()[index.getIntValue()] };
  564. }
  565. static StringArray getSupportedColumns()
  566. {
  567. return StringArray ("_display_name", "_size");
  568. }
  569. jobject getAssetFileDescriptor (JNIEnv* env, const LocalRef<jobject>& contentProvider,
  570. const String& filepath)
  571. {
  572. // This function can be called from multiple threads.
  573. {
  574. const ScopedLock sl (nonAssetFileOpenLock);
  575. if (! nonAssetFilePathsPendingShare.contains (filepath))
  576. {
  577. nonAssetFilePathsPendingShare.add (filepath);
  578. ++nonAssetFilesPendingShare;
  579. nonAssetFileObservers.add (new AndroidContentSharerFileObserver (*this, env,
  580. contentProvider,
  581. filepath));
  582. }
  583. }
  584. auto javaFile = LocalRef<jobject> (env->NewObject (JavaFile, JavaFile.constructor,
  585. javaString (filepath).get()));
  586. constexpr int modeReadOnly = 268435456;
  587. auto parcelFileDescriptor = LocalRef<jobject> (env->CallStaticObjectMethod (ParcelFileDescriptor,
  588. ParcelFileDescriptor.open,
  589. javaFile.get(), modeReadOnly));
  590. if (jniCheckHasExceptionOccurredAndClear())
  591. {
  592. // Failed to create file descriptor. Have you provided a valid file path/resource name?
  593. jassertfalse;
  594. return nullptr;
  595. }
  596. jlong startOffset = 0;
  597. jlong unknownLength = -1;
  598. assetFileDescriptors.add (GlobalRef (LocalRef<jobject> (env->NewObject (AssetFileDescriptor,
  599. AssetFileDescriptor.constructor,
  600. parcelFileDescriptor.get(),
  601. startOffset, unknownLength))));
  602. return assetFileDescriptors.getReference (assetFileDescriptors.size() - 1).get();
  603. }
  604. ContentSharer& owner;
  605. String packageName;
  606. String uriBase;
  607. std::unique_ptr<AndroidContentSharerPrepareFilesThread> prepareFilesThread;
  608. bool succeeded = false;
  609. String errorDescription;
  610. bool sharingActivityDidFinish = false;
  611. OwnedArray<AndroidContentSharerCursor> cursors;
  612. Array<GlobalRef> assetFileDescriptors;
  613. CriticalSection nonAssetFileOpenLock;
  614. StringArray nonAssetFilePathsPendingShare;
  615. Atomic<int> nonAssetFilesPendingShare { 0 };
  616. OwnedArray<AndroidContentSharerFileObserver> nonAssetFileObservers;
  617. WeakReference<ContentSharerNativeImpl>::Master masterReference;
  618. friend class WeakReference<ContentSharerNativeImpl>;
  619. //==============================================================================
  620. #define JNI_CLASS_MEMBERS(METHOD, STATICMETHOD, FIELD, STATICFIELD, CALLBACK) \
  621. CALLBACK (contentSharerQuery, "contentSharerQuery", "(Landroid/net/Uri;[Ljava/lang/String;)Landroid/database/Cursor;") \
  622. CALLBACK (contentSharerOpenFile, "contentSharerOpenFile", "(Landroid/net/Uri;Ljava/lang/String;)Landroid/content/res/AssetFileDescriptor;") \
  623. CALLBACK (contentSharerGetStreamTypes, "contentSharerGetStreamTypes", "(Landroid/net/Uri;Ljava/lang/String;)[Ljava/lang/String;") \
  624. DECLARE_JNI_CLASS_WITH_MIN_SDK (JuceSharingContentProvider, "com/roli/juce/JuceSharingContentProvider", 16)
  625. #undef JNI_CLASS_MEMBERS
  626. static jobject JNICALL contentSharerQuery (JNIEnv*, jobject contentProvider, jobject uri, jobjectArray projection)
  627. {
  628. if (auto *pimpl = (ContentSharer::ContentSharerNativeImpl *) ContentSharer::getInstance ()->pimpl.get ())
  629. return pimpl->query (LocalRef<jobject> (static_cast<jobject> (contentProvider)),
  630. LocalRef<jobject> (static_cast<jobject> (uri)),
  631. LocalRef<jobjectArray> (static_cast<jobjectArray> (projection)));
  632. return nullptr;
  633. }
  634. static jobject JNICALL contentSharerOpenFile (JNIEnv*, jobject contentProvider, jobject uri, jstring mode)
  635. {
  636. if (auto* pimpl = (ContentSharer::ContentSharerNativeImpl*) ContentSharer::getInstance()->pimpl.get())
  637. return pimpl->openFile (LocalRef<jobject> (static_cast<jobject> (contentProvider)),
  638. LocalRef<jobject> (static_cast<jobject> (uri)),
  639. LocalRef<jstring> (static_cast<jstring> (mode)));
  640. return nullptr;
  641. }
  642. static jobjectArray JNICALL contentSharerGetStreamTypes (JNIEnv*, jobject /*contentProvider*/, jobject uri, jstring mimeTypeFilter)
  643. {
  644. if (auto* pimpl = (ContentSharer::ContentSharerNativeImpl*) ContentSharer::getInstance()->pimpl.get())
  645. return pimpl->getStreamTypes (LocalRef<jobject> (static_cast<jobject> (uri)),
  646. LocalRef<jstring> (static_cast<jstring> (mimeTypeFilter)));
  647. return nullptr;
  648. }
  649. };
  650. //==============================================================================
  651. ContentSharer::Pimpl* ContentSharer::createPimpl()
  652. {
  653. return new ContentSharerNativeImpl (*this);
  654. }
  655. ContentSharer::ContentSharerNativeImpl::JuceSharingContentProvider_Class ContentSharer::ContentSharerNativeImpl::JuceSharingContentProvider;
  656. } // namespace juce