ksAndroid.Helpers.pas 22 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634
  1. {*******************************************************}
  2. { }
  3. { Helpers for Android }
  4. { }
  5. { Copyright (C) 2020 KngStr }
  6. { }
  7. { Some Code from }
  8. { Kastri Free of DelphiWorlds }
  9. { QDAC of swish }
  10. { Thanks }
  11. { }
  12. {*******************************************************}
  13. unit ksAndroid.Helpers;
  14. interface
  15. uses
  16. System.SysUtils, Androidapi.JNIBridge,
  17. Androidapi.JNI.GraphicsContentViewText, Androidapi.JNI.App,
  18. Androidapi.JNI.JavaTypes, Androidapi.JNI.Net;
  19. type
  20. TAndroidHelperEx = record
  21. private
  22. class function GetJActivity: JActivity; static;
  23. class function GetJContext: JContext; static;
  24. class function GetJActivityManager: JActivityManager; static;
  25. public
  26. const
  27. ICE_CREAM_SANDWICH = 14;
  28. ICE_CREAM_SANDWICH_MR1 = 15;
  29. JELLY_BEAN = 16;
  30. JELLY_BEAN_MR1 = 17;
  31. JELLY_BEAN_MR2 = 18;
  32. KITKAT = 19;
  33. KITKAT_MR1 = 20;
  34. LOLLIPOP = 21;
  35. LOLLIPOP_MR1 = 22;
  36. MARSHMALLOW = 23;
  37. NOUGAT = 24;
  38. NOUGAT_MR1 = 25;
  39. OREO = 26;
  40. OREO_MR1 = 27;
  41. PIE = 28;
  42. Q = 29;
  43. /// <summary>
  44. /// Checks if both build and target are greater or equal to the tested value
  45. /// </summary>
  46. class function CheckBuildAndTarget(const AValue: Integer): Boolean; static;
  47. /// <summary>
  48. /// Returns the equivalent of "AndroidClass.class"
  49. /// </summary>
  50. class function GetClass(const APackageClassName: string): Jlang_Class; static;
  51. /// <summary>
  52. /// Returns the application default icon ID
  53. /// </summary>
  54. class function GetDefaultIconID: Integer; static;
  55. /// <summary>
  56. /// Returns a URI to the notification sound
  57. /// </summary>
  58. class function GetDefaultNotificationSound: Jnet_Uri; static;
  59. /// <summary>
  60. /// Returns target Sdk version
  61. /// </summary>
  62. class function GetTargetSdkVersion: Integer; static;
  63. /// <summary>
  64. /// Returns installed Sdk version
  65. /// </summary>
  66. class function GetBuildSdkVersion: Integer; static;
  67. /// <summary>
  68. /// Returns whether the activity is running foreground
  69. /// </summary>
  70. /// <remarks>
  71. /// Useful from within a service to determine whether or not the service needs to run in foreground mode
  72. /// </remarks>
  73. class function IsActivityForeground: Boolean; static;
  74. /// <summary>
  75. /// Get column data from uri
  76. /// </summary>
  77. class function GetColumnAsString(const AUri: Jnet_Uri; const AColumn: JString; ASelection: JString = nil;
  78. ASelectionArgs: TJavaObjectArray<JString> = nil; ASortOrder: JString = nil): string; static;
  79. /// <summary>
  80. /// Get data column data from uri
  81. /// </summary>
  82. class function GetDataColumnAsString(const AUri: Jnet_Uri; ASelection: JString = nil;
  83. ASelectionArgs: TJavaObjectArray<JString> = nil; ASortOrder: JString = nil): string; static;
  84. /// <summary>
  85. /// Converts uri to file path
  86. /// </summary>
  87. class function FileFromUri(const AUri: Jnet_Uri): string; overload; static;
  88. /// <summary>
  89. /// Converts file to uri, without FileProvider
  90. /// </summary>
  91. class function UriFromFile(const AFile: JFile): Jnet_Uri; overload; static;
  92. /// <summary>
  93. /// Converts file to uri, without FileProvider
  94. /// </summary>
  95. class function UriFromFile(const AFileName: string): Jnet_Uri; overload; static;
  96. /// <summary>
  97. /// Converts file to uri, using FileProvider if target API >= 24
  98. /// </summary>
  99. /// <remarks>
  100. /// Use this only when accessing files with an "external" URI
  101. /// </remarks>
  102. class function SharedUriFromFile(const AFile: JFile; const AAuthority: string = ''): Jnet_Uri; overload; static;
  103. /// <summary>
  104. /// Converts filename to uri, using FileProvider if target API >= 24
  105. /// </summary>
  106. /// <remarks>
  107. /// Use this only when accessing files with an "external" URI
  108. /// </remarks>
  109. class function SharedUriFromFile(const AFileName: string; const AAuthority: string = ''): Jnet_Uri; overload; static;
  110. /// <summary>
  111. /// TJnet_Uri.JavaClass.parse
  112. /// </summary>
  113. class function UriParse(const S: string): Jnet_Uri; overload; static;
  114. /// <summary>
  115. /// TJnet_Uri.JavaClass.parse
  116. /// </summary>
  117. class function UriParse(const S: JString): Jnet_Uri; overload; static;
  118. /// <summary>Returns Java Application Context</summary>
  119. class property Context: JContext read GetJContext;
  120. /// <summary>Returns Java Application Activity</summary>
  121. /// <remarks>An exception will be launched if there is no activity, for example a Service</remarks>
  122. class property Activity: JActivity read GetJActivity;
  123. /// <summary>Returns Java Application Activity Manager</summary>
  124. class property ActivityManager: JActivityManager read GetJActivityManager;
  125. /// <remarks>Need reorder tasks permission</remarks>
  126. class procedure BringAppToFront; static;
  127. /// <remarks>Need reorder tasks permission</remarks>
  128. class procedure SendAppToBack; static;
  129. /// <summary>Call a Java Activity</summary>
  130. class function StartActivity(Intent: JIntent; const Code: Integer = -1): Boolean; static;
  131. /// <summary>Returns Application package name</summary>
  132. class function GetPackageName: string; static;
  133. /// <summary>Returns MimeType from filename</summary>
  134. class function GetMimeType(AFileName: string): JString; static;
  135. /// <summary>Returns Primary SDCard path</summary>
  136. class function GetSDCardPath: string; static;
  137. /// <summary>Checks if there is at least one application capable of receiving the intent.</summary>
  138. class function HasAssocApp(const URI: string): Boolean; overload; static;
  139. /// <summary>Checks if there is at least one application capable of receiving the intent.</summary>
  140. class function HasAssocApp(const Intent: JIntent): Boolean; overload; static;
  141. /// <summary>Install an android package: xxx.apk</summary>
  142. /// <remarks>
  143. /// <uses-permission android:name="android.permission.REQUEST_INSTALL_PACKAGES"/>
  144. /// </remarks>
  145. class function InstallPackage(const AFileName: string; const AAuthority: string = ''): Boolean; static;
  146. /// <summary>Check if an application is installed</summary>
  147. class function IsAppInstalled(const APackage: string): Boolean; static;
  148. /// <summary>Add/Cear FLAG_KEEP_SCREEN_ON</summary>
  149. class function KeepScreen(AOn: Boolean): Boolean; static;
  150. end;
  151. implementation
  152. uses
  153. {$IFDEF DEBUG}FMX.Types,{$ENDIF}
  154. Androidapi.JNI.Os, Androidapi.JNI.Support,
  155. Androidapi.JNI.Media, Androidapi.JNI.Provider, Androidapi.JNI.Webkit,
  156. Androidapi.Helpers, FMX.Helpers.Android, System.IOUtils;
  157. { TAndroidHelperEx }
  158. class procedure TAndroidHelperEx.BringAppToFront;
  159. begin
  160. ActivityManager.moveTaskToFront(Activity.getTaskId, TJIntent.JavaClass.FLAG_ACTIVITY_NEW_TASK);
  161. end;
  162. class function TAndroidHelperEx.CheckBuildAndTarget(const AValue: Integer): Boolean;
  163. begin
  164. Result := (GetBuildSdkVersion >= AValue) and (GetTargetSdkVersion >= AValue);
  165. end;
  166. (*
  167. * >=4.4
  168. * uri=content://com.android.providers.media.documents/document/image%3A293502
  169. * uri=file:///storage/emulated/0/temp_photo.jpg
  170. * uri=content://media/external/images/media/193968
  171. * <4.4
  172. * uri=content://media/external/images/media/13
  173. * third party
  174. * content://com.speedsoftware.explorer.fileprovider/root/storage/emulated/0/Android/data/com.lifan.qspsy/files/cache/thumb/F6AB021A6BBCEFC3B942625FBA2E6ADE/7.jpg
  175. * content://com.tencent.mtt.fileprovider/QQBrowser/Movies/BVR_2019_10_14_10_52_29_trimq.mp4
  176. * content://com.estrongs.files/storage/emulated/0/DCIM/360%E8%A1%8C%E8%BD%A6%E8%AE%B0%E5%BD%95%E4%BB%AA/2018_02_11_14_16_40_ABAC1BB6.mp4
  177. *
  178. * 参考文献
  179. * https://github.com/coltoscosmin/FileUtils/blob/master/FileUtils.java
  180. * https://github.com/DB-BOY/FileChoose/blob/master/app/src/main/java/cn/dbboy/filechoose/FileUtil.java
  181. * https://www.jianshu.com/p/c5f207f8cce6
  182. * https://blog.csdn.net/chengfu116/article/details/74923161
  183. * https://www.cnblogs.com/epmouse/p/5421048.html
  184. *
  185. * 暂未实现
  186. * 安卓10好像完全没有获取真实路径的机会
  187. * 安卓7/8/9貌似有些机型还是不行,只能是获取内容后在自己可读写的区域存一份儿
  188. * 如果是获取内容而不是path的话,就不需要用这里了,单独搞个api更好
  189. *)
  190. class function TAndroidHelperEx.FileFromUri(const AUri: Jnet_Uri): string;
  191. function GetUriByType(S: string): Jnet_Uri;
  192. begin
  193. if SameText(S, 'image') then
  194. Result := TJImages_Media.JavaClass.EXTERNAL_CONTENT_URI
  195. else if SameText(S, 'video') then
  196. Result := TJVideo_Media.JavaClass.EXTERNAL_CONTENT_URI
  197. else if SameText(S, 'audio') then
  198. Result := TJAudio_Media.JavaClass.EXTERNAL_CONTENT_URI
  199. else
  200. Result := nil;
  201. end;
  202. function GetDocId(var ADocId, AType, AId: string): Boolean;
  203. var
  204. LJString: JString;
  205. LArr: TArray<string>;
  206. begin
  207. Result := False;
  208. LJString := TJDocumentsContract.JavaClass.getDocumentId(AUri);
  209. if LJString = nil then
  210. Exit;
  211. ADocId := JStringToString(LJString);
  212. if ADocId = '' then
  213. Exit;
  214. LArr := ADocId.Split([':']);
  215. if Length(LArr) < 2 then begin
  216. AType := '';
  217. AId := ADocId;
  218. end
  219. else begin
  220. AType := LArr[0];
  221. AId := LArr[1];
  222. end;
  223. Result := AId <> '';
  224. end;
  225. var
  226. LSdCard, LPath: string;
  227. LAuthority, LScheme, LDocId, LType, LId: string;
  228. LSelection: JString;
  229. LUri: Jnet_Uri;
  230. LArr: TArray<string>;
  231. I: Integer;
  232. iId: Int64;
  233. begin
  234. Result := '';
  235. if AUri = nil then
  236. Exit;
  237. LSdCard := GetSDCardPath;
  238. LAuthority := JStringToString(AUri.getAuthority().toString);
  239. LScheme := JStringToString(AUri.getScheme());
  240. {$IFDEF DEBUG}
  241. Log.d('---TAndroidHelperEx.FileFromUri-AUri:%s', [JStringToString(AUri.toString)]);
  242. Log.d('---TAndroidHelperEx.FileFromUri-AUri.Authority:%s-LScheme:%s', [LAuthority, LScheme]);
  243. {$ENDIF}
  244. if CheckBuildAndTarget(KITKAT) and TJDocumentsContract.JavaClass.isDocumentUri(Context, AUri) then begin
  245. if not GetDocId(LDocId, LType, LId) then
  246. Exit;
  247. {$IFDEF DEBUG}
  248. Log.d('---TAndroidHelperEx.FileFromUri-LDocId:%s-LType:%s-LId:%s', [LDocId, LType, LId]);
  249. {$ENDIF}
  250. if SameText(LAuthority, 'com.android.externalstorage.documents') then begin
  251. if SameText(LType, 'primary') then
  252. Result := TPath.Combine(LSdCard, LId)
  253. else if SameText(LType, 'home') then
  254. Result := TPath.Combine(TPath.GetSharedDocumentsPath, LId)
  255. {$IFDEF DEBUG}
  256. else
  257. Log.d('---TAndroidHelperEx.FileFromUri-Unkown externalstorage-LDocId:%s-LType:%s-LId:%s', [LDocId, LType, LId]);
  258. {$ENDIF}
  259. end
  260. else if SameText(LAuthority, 'com.android.providers.media.documents') then begin
  261. LSelection := StringToJString('_id=' + LId);
  262. LUri := GetUriByType(LType);
  263. if LUri <> nil then
  264. Result := GetDataColumnAsString(LUri, LSelection);
  265. end else if SameText(LAuthority, 'com.android.providers.downloads.documents') then begin
  266. if SameText(LType, 'raw') then
  267. Result := Copy(LDocId, 5)
  268. //else if GetBuildSdkVersion < OREO then begin // 有人说O的时候不需要自己解析,但测试锤子手机7.1,不行
  269. else begin
  270. SetLength(LArr, 2);
  271. LArr[0] := 'content://downloads/public_downloads';
  272. LArr[1] := 'content://downloads/my_downloads';
  273. //LArr[2] := 'content://downloads/all_downloads'; // 这个貌似没权限
  274. iId := StrToInt64Def(LId, -1);
  275. for I := 0 to Length(LArr) - 1 do begin
  276. LUri := TJContentUris.JavaClass.withAppendedId(UriParse(LArr[I]), iId);
  277. if LUri <> nil then
  278. try
  279. Result := GetDataColumnAsString(LUri);
  280. if Result <> '' then
  281. Break;
  282. except
  283. on E: Exception do begin
  284. {$IFDEF DEBUG}
  285. Log.d('---TAndroidHelperEx.FileFromUri Error: [%s]%s', [E.ClassName, E.Message]);
  286. {$ENDIF}
  287. end;
  288. end;
  289. end;
  290. end;
  291. //else
  292. if Result = '' then
  293. Result := GetDataColumnAsString(AUri);
  294. end
  295. {$IFDEF DEBUG}
  296. else
  297. Log.d('---TAndroidHelperEx.FileFromUri-Unkown DocumentUri-');
  298. {$ENDIF}
  299. end else if SameText(LScheme, 'content') then begin
  300. // Return the remote address
  301. if SameText(LAuthority, 'com.google.android.apps.photos.content') then
  302. Result := JStringToString(AUri.getLastPathSegment())
  303. else if SameText(LAuthority, 'com.tencent.mtt.fileprovider') then begin
  304. LPath := JStringToString(AUri.getPath());
  305. if Pos('/QQBrowser/', LPath) = 1 then begin // /QQBrowser/XXX
  306. LPath := TPath.Combine(LSdCard, Copy(LPath, 12));
  307. if FileExists(LPath) then
  308. Result := LPath;
  309. end;
  310. end
  311. else if SameText(LAuthority, 'com.speedsoftware.explorer.fileprovider') then begin
  312. LPath := JStringToString(AUri.getPath());
  313. if Pos('/root/', LPath) = 1 then
  314. Result := Copy(LPath, 6);
  315. end
  316. else if SameText(LAuthority, 'com.estrongs.files') then
  317. Result := JStringToString(AUri.getPath())
  318. else
  319. Result := GetDataColumnAsString(AUri);
  320. end
  321. else if SameText(LScheme, 'file') then
  322. Result := JStringToString(AUri.getPath())
  323. {$IFDEF DEBUG}
  324. else
  325. Log.d('---TAndroidHelperEx.FileFromUri-Unkown Uri-');
  326. {$ENDIF}
  327. end;
  328. class function TAndroidHelperEx.GetBuildSdkVersion: Integer;
  329. begin
  330. Result := TJBuild_VERSION.JavaClass.SDK_INT;
  331. end;
  332. class function TAndroidHelperEx.GetClass(const APackageClassName: string): Jlang_Class;
  333. begin
  334. Result := TJLang_Class.JavaClass.forName(StringToJString(APackageClassName), True, Context.getClassLoader);
  335. end;
  336. class function TAndroidHelperEx.GetColumnAsString(const AUri: Jnet_Uri; const AColumn: JString;
  337. ASelection: JString; ASelectionArgs: TJavaObjectArray<JString>; ASortOrder: JString): string;
  338. var
  339. LCursor: JCursor;
  340. LIndex: Integer;
  341. LPojection: TJavaObjectArray<JString>;
  342. begin
  343. Result := '';
  344. if AUri = nil then
  345. Exit;
  346. LPojection := TJavaObjectArray<JString>.Create(1);
  347. LPojection.Items[0] := AColumn;
  348. LCursor := Context.getContentResolver().query(AUri, LPojection, ASelection, ASelectionArgs, ASortOrder);
  349. if LCursor = nil then begin
  350. {$IFDEF DEBUG}
  351. Log.d('---TAndroidHelperEx.GetColumnAsString-LCursor = nil-');
  352. {$ENDIF}
  353. Exit;
  354. end;
  355. try
  356. LIndex := LCursor.getColumnIndex(AColumn);
  357. if (LIndex > -1) and LCursor.moveToFirst then
  358. Result := JStringToString(LCursor.getString(LIndex))
  359. {$IFDEF DEBUG}
  360. else
  361. Log.d('---TAndroidHelperEx.GetColumnAsString-LIndex:%d-', [LIndex]);
  362. {$ENDIF}
  363. finally
  364. LCursor.close;
  365. end;
  366. end;
  367. class function TAndroidHelperEx.GetDataColumnAsString(const AUri: Jnet_Uri;
  368. ASelection: JString; ASelectionArgs: TJavaObjectArray<JString>;
  369. ASortOrder: JString): string;
  370. begin
  371. Result := GetColumnAsString(AUri, StringToJString('_data'), ASelection, ASelectionArgs, ASortOrder);
  372. end;
  373. class function TAndroidHelperEx.GetDefaultIconID: Integer;
  374. begin
  375. Result := Context.getApplicationInfo.icon;
  376. end;
  377. class function TAndroidHelperEx.GetDefaultNotificationSound: Jnet_Uri;
  378. begin
  379. Result := TJRingtoneManager.JavaClass.getDefaultUri(TJRingtoneManager.JavaClass.TYPE_NOTIFICATION);
  380. end;
  381. class function TAndroidHelperEx.GetJActivity: JActivity;
  382. begin
  383. Result :=
  384. {$IF CompilerVersion > 27}
  385. TAndroidHelper.Activity
  386. {$ELSE}
  387. SharedActivity
  388. {$ENDIF}
  389. end;
  390. class function TAndroidHelperEx.GetJActivityManager: JActivityManager;
  391. var
  392. AService: JObject;
  393. begin
  394. AService := Context.getSystemService(TJContext.JavaClass.ACTIVITY_SERVICE);
  395. Result := TJActivityManager.Wrap((AService as ILocalObject).GetObjectID);
  396. end;
  397. class function TAndroidHelperEx.GetJContext: JContext;
  398. begin
  399. Result :=
  400. {$IF CompilerVersion > 27}
  401. TAndroidHelper.Context;
  402. {$ELSE}
  403. SharedActivityContext;
  404. {$ENDIF}
  405. end;
  406. class function TAndroidHelperEx.GetMimeType(AFileName: string): JString;
  407. var
  408. LExt: string;
  409. LTypeMap: JMimeTypeMap;
  410. begin
  411. Result := nil;
  412. if AFileName = '' then
  413. Exit;
  414. LExt := LowerCase(ExtractFileExt(AFileName));
  415. if Length(LExt) < 2 then
  416. Exit;
  417. LExt := Copy(LExt, 1);
  418. LTypeMap := TJMimeTypeMap.JavaClass.getSingleton();
  419. if LTypeMap = nil then
  420. Exit;
  421. Result := LTypeMap.getMimeTypeFromExtension(StringToJString(LExt));
  422. if Result <> nil then
  423. Result := Result.trim();
  424. end;
  425. class function TAndroidHelperEx.GetPackageName: string;
  426. begin
  427. Result := JStringToString(Context.getPackageName);
  428. end;
  429. class function TAndroidHelperEx.GetSDCardPath: string;
  430. var
  431. sPath: string;
  432. begin
  433. Result := '';
  434. sPath := System.IOUtils.TPath.GetSharedDocumentsPath;
  435. if Pos(PathDelim, sPath) = 0 then
  436. Exit;
  437. Result := ExtractFilePath(ExcludeTrailingPathDelimiter(sPath));
  438. end;
  439. class function TAndroidHelperEx.GetTargetSdkVersion: Integer;
  440. var
  441. LApplicationInfo: JApplicationInfo;
  442. begin
  443. LApplicationInfo := Context.getPackageManager.getApplicationInfo(Context.getPackageName, 0);
  444. Result := LApplicationInfo.targetSdkVersion;
  445. end;
  446. class function TAndroidHelperEx.HasAssocApp(const Intent: JIntent): Boolean;
  447. var
  448. LList: JList;
  449. begin
  450. // Android 6+ APP LINK closed Will cause list is null.
  451. // Activity with <action android:name="android.intent.action.VIEW" />
  452. LList := Activity.getPackageManager.queryIntentActivities(Intent,
  453. TJPackageManager.JavaClass.MATCH_DEFAULT_ONLY);
  454. Result := (LList <> nil) and (LList.size > 0);
  455. end;
  456. class function TAndroidHelperEx.HasAssocApp(const URI: string): Boolean;
  457. var
  458. Intent: JIntent;
  459. begin
  460. Result := False;
  461. Intent := TJIntent.Create;
  462. Intent.setData(UriParse(URI));
  463. Intent.setAction(StringToJString('android.intent.action.VIEW'));
  464. Result := HasAssocApp(Intent);
  465. end;
  466. class function TAndroidHelperEx.InstallPackage(const AFileName, AAuthority: string): Boolean;
  467. var
  468. LIntent: JIntent;
  469. begin
  470. Result := False;
  471. if Trim(AFileName) = '' then
  472. Exit;
  473. LIntent := TJIntent.Create;
  474. if CheckBuildAndTarget(OREO) then
  475. LIntent.setAction(TJIntent.JavaClass.ACTION_INSTALL_PACKAGE)
  476. else
  477. LIntent.setAction(TJIntent.JavaClass.ACTION_VIEW);
  478. // 没有这个也可以安装成功,但是安装成功后的成功页面,也就是,完成/打开 会无法显示
  479. LIntent.addFlags(TJIntent.JavaClass.FLAG_ACTIVITY_NEW_TASK);
  480. LIntent.setDataAndType(SharedUriFromFile(AFileName, AAuthority),
  481. StringToJString('application/vnd.android.package-archive'));
  482. LIntent.addFlags(TJIntent.JavaClass.FLAG_GRANT_READ_URI_PERMISSION);
  483. Result := StartActivity(LIntent);
  484. end;
  485. class function TAndroidHelperEx.IsActivityForeground: Boolean;
  486. var
  487. LService: JObject;
  488. LRunningApps: JList;
  489. LAppInfo: JActivityManager_RunningAppProcessInfo;
  490. I: Integer;
  491. begin
  492. Result := False;
  493. LService := Context.getSystemService(TJContext.JavaClass.ACTIVITY_SERVICE);
  494. LRunningApps := TJActivityManager.Wrap(TAndroidHelper.JObjectToID(LService)).getRunningAppProcesses;
  495. for I := 0 to LRunningApps.size - 1 do begin
  496. LAppInfo := TJActivityManager_RunningAppProcessInfo.Wrap(TAndroidHelper.JObjectToID(LRunningApps.get(I)));
  497. if LAppInfo.importance = 100 then begin
  498. if LAppInfo.importanceReasonComponent <> nil then begin
  499. if LAppInfo.importanceReasonComponent.getPackageName.equals(Context.getPackageName) then
  500. Exit(True);
  501. end
  502. else if LRunningApps.size = 1 then
  503. Exit(True);
  504. end;
  505. end;
  506. end;
  507. class function TAndroidHelperEx.IsAppInstalled(const APackage: string): Boolean;
  508. begin
  509. Result := False;
  510. try
  511. //只有异常是可靠的,返回值判定不对
  512. Result := Context.getPackageManager.getPackageInfo(StringToJString(APackage),
  513. TJPackageManager.JavaClass.GET_ACTIVITIES) = nil;
  514. Result := True;
  515. except
  516. end;
  517. end;
  518. class function TAndroidHelperEx.KeepScreen(AOn: Boolean): Boolean;
  519. begin
  520. CallInUIThreadAndWaitFinishing(
  521. procedure begin
  522. if AOn then
  523. SharedActivity.getWindow.addFlags(
  524. TJWindowManager_LayoutParams.JavaClass.FLAG_KEEP_SCREEN_ON)
  525. else
  526. SharedActivity.getWindow.clearFlags(
  527. TJWindowManager_LayoutParams.JavaClass.FLAG_KEEP_SCREEN_ON);
  528. end);
  529. end;
  530. class procedure TAndroidHelperEx.SendAppToBack;
  531. begin
  532. //SharedActivityManager.moveTaskToBack(Activity.getTaskId, TJIntent.JavaClass.FLAG_ACTIVITY_NEW_TASK);
  533. Activity.moveTaskToBack(True);
  534. end;
  535. class function TAndroidHelperEx.StartActivity(Intent: JIntent; const Code: Integer): Boolean;
  536. begin
  537. Result := False;
  538. if HasAssocApp(Intent) then begin
  539. if Code = -1 then
  540. Activity.startActivity(Intent)
  541. else
  542. Activity.startActivityForResult(Intent, Code);
  543. Result := True;
  544. end;
  545. end;
  546. class function TAndroidHelperEx.UriParse(const S: JString): Jnet_Uri;
  547. begin
  548. Result := TJnet_Uri.JavaClass.parse(S);
  549. end;
  550. class function TAndroidHelperEx.UriParse(const S: string): Jnet_Uri;
  551. begin
  552. Result := UriParse(StringToJString(S));
  553. end;
  554. class function TAndroidHelperEx.UriFromFile(const AFileName: string): Jnet_Uri;
  555. begin
  556. Result := UriFromFile(TJFile.JavaClass.init(StringToJString(AFileName)));
  557. end;
  558. class function TAndroidHelperEx.UriFromFile(const AFile: JFile): Jnet_Uri;
  559. begin
  560. Result := TJnet_uri.JavaClass.fromFile(AFile);
  561. end;
  562. class function TAndroidHelperEx.SharedUriFromFile(const AFile: JFile; const AAuthority: string): Jnet_Uri;
  563. var
  564. LAuthority: JString;
  565. begin
  566. if CheckBuildAndTarget(NOUGAT) then begin
  567. if AAuthority <> '' then
  568. LAuthority := StringToJString(AAuthority)
  569. else
  570. LAuthority := Context.getApplicationContext.getPackageName.concat(StringToJString('.fileprovider'));
  571. Result := TJFileProvider.JavaClass.getUriForFile(Context, LAuthority, AFile);
  572. end
  573. else
  574. Result := TJnet_uri.JavaClass.fromFile(AFile);
  575. end;
  576. class function TAndroidHelperEx.SharedUriFromFile(const AFileName, AAuthority: string): Jnet_Uri;
  577. begin
  578. Result := SharedUriFromFile(TJFile.JavaClass.init(StringToJString(AFileName)), AAuthority);
  579. end;
  580. end.