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.

888 lines
40KB

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