ksAndroid.Helpers.pas 23 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654
  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>Check HasAssocApp and Call a Java Activity</summary>
  132. class function CheckAndStartActivity(Intent: JIntent; const Code: Integer = -1): Boolean; static;
  133. /// <summary>Returns Application package name</summary>
  134. class function GetPackageName: string; static;
  135. /// <summary>Returns MimeType from filename</summary>
  136. class function GetMimeType(AFileName: string): JString; static;
  137. /// <summary>Returns Primary SDCard path</summary>
  138. class function GetSDCardPath: string; static;
  139. /// <summary>Checks if there is at least one application capable of receiving the intent.</summary>
  140. class function HasAssocApp(const URI: string): Boolean; overload; static;
  141. /// <summary>Checks if there is at least one application capable of receiving the intent.</summary>
  142. class function HasAssocApp(const Intent: JIntent): Boolean; overload; static;
  143. /// <summary>Install an android package: xxx.apk</summary>
  144. /// <remarks>
  145. /// <uses-permission android:name="android.permission.REQUEST_INSTALL_PACKAGES"/>
  146. /// </remarks>
  147. class function InstallPackage(const AFileName: string; const AAuthority: string = ''): Boolean; static;
  148. /// <summary>Check if an application is installed</summary>
  149. class function IsAppInstalled(const APackage: string): Boolean; static;
  150. /// <summary>Add/Cear FLAG_KEEP_SCREEN_ON</summary>
  151. class function KeepScreen(AOn: Boolean): Boolean; static;
  152. end;
  153. implementation
  154. uses
  155. {$IFDEF DEBUG}FMX.Types,{$ENDIF}
  156. Androidapi.JNI.Os, Androidapi.JNI.Support,
  157. Androidapi.JNI.Media, Androidapi.JNI.Provider, Androidapi.JNI.Webkit,
  158. Androidapi.Helpers, FMX.Helpers.Android, System.IOUtils;
  159. { TAndroidHelperEx }
  160. class procedure TAndroidHelperEx.BringAppToFront;
  161. begin
  162. ActivityManager.moveTaskToFront(Activity.getTaskId, TJIntent.JavaClass.FLAG_ACTIVITY_NEW_TASK);
  163. end;
  164. class function TAndroidHelperEx.CheckAndStartActivity(Intent: JIntent;
  165. const Code: Integer): Boolean;
  166. begin
  167. Result := (Intent <> nil) and HasAssocApp(Intent);
  168. if not Result then
  169. Exit;
  170. Result := TAndroidHelperEx.StartActivity(Intent, Code);
  171. end;
  172. class function TAndroidHelperEx.CheckBuildAndTarget(const AValue: Integer): Boolean;
  173. begin
  174. Result := (GetBuildSdkVersion >= AValue) and (GetTargetSdkVersion >= AValue);
  175. end;
  176. (*
  177. * >=4.4
  178. * uri=content://com.android.providers.media.documents/document/image%3A293502
  179. * uri=file:///storage/emulated/0/temp_photo.jpg
  180. * uri=content://media/external/images/media/193968
  181. * <4.4
  182. * uri=content://media/external/images/media/13
  183. * third party
  184. * content://com.speedsoftware.explorer.fileprovider/root/storage/emulated/0/Android/data/com.lifan.qspsy/files/cache/thumb/F6AB021A6BBCEFC3B942625FBA2E6ADE/7.jpg
  185. * content://com.tencent.mtt.fileprovider/QQBrowser/Movies/BVR_2019_10_14_10_52_29_trimq.mp4
  186. * 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
  187. *
  188. * 参考文献
  189. * https://github.com/coltoscosmin/FileUtils/blob/master/FileUtils.java
  190. * https://github.com/DB-BOY/FileChoose/blob/master/app/src/main/java/cn/dbboy/filechoose/FileUtil.java
  191. * https://www.jianshu.com/p/c5f207f8cce6
  192. * https://blog.csdn.net/chengfu116/article/details/74923161
  193. * https://www.cnblogs.com/epmouse/p/5421048.html
  194. *
  195. * 暂未实现
  196. * 安卓10好像完全没有获取真实路径的机会
  197. * 安卓7/8/9貌似有些机型还是不行,只能是获取内容后在自己可读写的区域存一份儿
  198. * 如果是获取内容而不是path的话,就不需要用这里了,单独搞个api更好
  199. *)
  200. class function TAndroidHelperEx.FileFromUri(const AUri: Jnet_Uri): string;
  201. function GetUriByType(S: string): Jnet_Uri;
  202. begin
  203. if SameText(S, 'image') then
  204. Result := TJImages_Media.JavaClass.EXTERNAL_CONTENT_URI
  205. else if SameText(S, 'video') then
  206. Result := TJVideo_Media.JavaClass.EXTERNAL_CONTENT_URI
  207. else if SameText(S, 'audio') then
  208. Result := TJAudio_Media.JavaClass.EXTERNAL_CONTENT_URI
  209. else
  210. Result := nil;
  211. end;
  212. function GetDocId(var ADocId, AType, AId: string): Boolean;
  213. var
  214. LJString: JString;
  215. LArr: TArray<string>;
  216. begin
  217. Result := False;
  218. LJString := TJDocumentsContract.JavaClass.getDocumentId(AUri);
  219. if LJString = nil then
  220. Exit;
  221. ADocId := JStringToString(LJString);
  222. if ADocId = '' then
  223. Exit;
  224. LArr := ADocId.Split([':']);
  225. if Length(LArr) < 2 then begin
  226. AType := '';
  227. AId := ADocId;
  228. end
  229. else begin
  230. AType := LArr[0];
  231. AId := LArr[1];
  232. end;
  233. Result := AId <> '';
  234. end;
  235. var
  236. LSdCard, LPath: string;
  237. LAuthority, LScheme, LDocId, LType, LId: string;
  238. LSelection: JString;
  239. LUri: Jnet_Uri;
  240. LArr: TArray<string>;
  241. I: Integer;
  242. iId: Int64;
  243. begin
  244. Result := '';
  245. if AUri = nil then
  246. Exit;
  247. LSdCard := GetSDCardPath;
  248. LAuthority := JStringToString(AUri.getAuthority().toString);
  249. LScheme := JStringToString(AUri.getScheme());
  250. {$IFDEF DEBUG}
  251. Log.d('---TAndroidHelperEx.FileFromUri-AUri:%s', [JStringToString(AUri.toString)]);
  252. Log.d('---TAndroidHelperEx.FileFromUri-AUri.Authority:%s-LScheme:%s', [LAuthority, LScheme]);
  253. {$ENDIF}
  254. if CheckBuildAndTarget(KITKAT) and TJDocumentsContract.JavaClass.isDocumentUri(Context, AUri) then begin
  255. if not GetDocId(LDocId, LType, LId) then
  256. Exit;
  257. {$IFDEF DEBUG}
  258. Log.d('---TAndroidHelperEx.FileFromUri-LDocId:%s-LType:%s-LId:%s', [LDocId, LType, LId]);
  259. {$ENDIF}
  260. if SameText(LAuthority, 'com.android.externalstorage.documents') then begin
  261. if SameText(LType, 'primary') then
  262. Result := TPath.Combine(LSdCard, LId)
  263. else if SameText(LType, 'home') then
  264. Result := TPath.Combine(TPath.GetSharedDocumentsPath, LId)
  265. {$IFDEF DEBUG}
  266. else
  267. Log.d('---TAndroidHelperEx.FileFromUri-Unkown externalstorage-LDocId:%s-LType:%s-LId:%s', [LDocId, LType, LId]);
  268. {$ENDIF}
  269. end
  270. else if SameText(LAuthority, 'com.android.providers.media.documents') then begin
  271. LSelection := StringToJString('_id=' + LId);
  272. LUri := GetUriByType(LType);
  273. if LUri <> nil then
  274. Result := GetDataColumnAsString(LUri, LSelection);
  275. end else if SameText(LAuthority, 'com.android.providers.downloads.documents') then begin
  276. if SameText(LType, 'raw') then
  277. Result := Copy(LDocId, 5)
  278. //else if GetBuildSdkVersion < OREO then begin // 有人说O的时候不需要自己解析,但测试锤子手机7.1,不行
  279. else begin
  280. SetLength(LArr, 2);
  281. LArr[0] := 'content://downloads/public_downloads';
  282. LArr[1] := 'content://downloads/my_downloads';
  283. //LArr[2] := 'content://downloads/all_downloads'; // 这个貌似没权限
  284. iId := StrToInt64Def(LId, -1);
  285. for I := 0 to Length(LArr) - 1 do begin
  286. LUri := TJContentUris.JavaClass.withAppendedId(UriParse(LArr[I]), iId);
  287. if LUri <> nil then
  288. try
  289. Result := GetDataColumnAsString(LUri);
  290. if Result <> '' then
  291. Break;
  292. except
  293. on E: Exception do begin
  294. {$IFDEF DEBUG}
  295. Log.d('---TAndroidHelperEx.FileFromUri Error: [%s]%s', [E.ClassName, E.Message]);
  296. {$ENDIF}
  297. end;
  298. end;
  299. end;
  300. end;
  301. //else
  302. if Result = '' then
  303. Result := GetDataColumnAsString(AUri);
  304. end
  305. {$IFDEF DEBUG}
  306. else
  307. Log.d('---TAndroidHelperEx.FileFromUri-Unkown DocumentUri-');
  308. {$ENDIF}
  309. end else if SameText(LScheme, 'content') then begin
  310. // Return the remote address
  311. if SameText(LAuthority, 'com.google.android.apps.photos.content') then
  312. Result := JStringToString(AUri.getLastPathSegment())
  313. else if SameText(LAuthority, 'com.tencent.mtt.fileprovider') then begin
  314. LPath := JStringToString(AUri.getPath());
  315. if Pos('/QQBrowser/', LPath) = 1 then begin // /QQBrowser/XXX
  316. LPath := TPath.Combine(LSdCard, Copy(LPath, 12));
  317. if FileExists(LPath) then
  318. Result := LPath;
  319. end;
  320. end
  321. else if SameText(LAuthority, 'com.speedsoftware.explorer.fileprovider') then begin
  322. LPath := JStringToString(AUri.getPath());
  323. if Pos('/root/', LPath) = 1 then
  324. Result := Copy(LPath, 6);
  325. end
  326. else if SameText(LAuthority, 'com.estrongs.files') then
  327. Result := JStringToString(AUri.getPath())
  328. else
  329. Result := GetDataColumnAsString(AUri);
  330. end
  331. else if SameText(LScheme, 'file') then
  332. Result := JStringToString(AUri.getPath())
  333. {$IFDEF DEBUG}
  334. else
  335. Log.d('---TAndroidHelperEx.FileFromUri-Unkown Uri-');
  336. {$ENDIF}
  337. end;
  338. class function TAndroidHelperEx.GetBuildSdkVersion: Integer;
  339. begin
  340. Result := TJBuild_VERSION.JavaClass.SDK_INT;
  341. end;
  342. class function TAndroidHelperEx.GetClass(const APackageClassName: string): Jlang_Class;
  343. begin
  344. Result := TJLang_Class.JavaClass.forName(StringToJString(APackageClassName), True, Context.getClassLoader);
  345. end;
  346. class function TAndroidHelperEx.GetColumnAsString(const AUri: Jnet_Uri; const AColumn: JString;
  347. ASelection: JString; ASelectionArgs: TJavaObjectArray<JString>; ASortOrder: JString): string;
  348. var
  349. LCursor: JCursor;
  350. LIndex: Integer;
  351. LPojection: TJavaObjectArray<JString>;
  352. begin
  353. Result := '';
  354. if AUri = nil then
  355. Exit;
  356. LPojection := TJavaObjectArray<JString>.Create(1);
  357. LPojection.Items[0] := AColumn;
  358. LCursor := Context.getContentResolver().query(AUri, LPojection, ASelection, ASelectionArgs, ASortOrder);
  359. if LCursor = nil then begin
  360. {$IFDEF DEBUG}
  361. Log.d('---TAndroidHelperEx.GetColumnAsString-LCursor = nil-');
  362. {$ENDIF}
  363. Exit;
  364. end;
  365. try
  366. LIndex := LCursor.getColumnIndex(AColumn);
  367. if (LIndex > -1) and LCursor.moveToFirst then
  368. Result := JStringToString(LCursor.getString(LIndex))
  369. {$IFDEF DEBUG}
  370. else
  371. Log.d('---TAndroidHelperEx.GetColumnAsString-LIndex:%d-', [LIndex]);
  372. {$ENDIF}
  373. finally
  374. LCursor.close;
  375. end;
  376. end;
  377. class function TAndroidHelperEx.GetDataColumnAsString(const AUri: Jnet_Uri;
  378. ASelection: JString; ASelectionArgs: TJavaObjectArray<JString>;
  379. ASortOrder: JString): string;
  380. begin
  381. Result := GetColumnAsString(AUri, StringToJString('_data'), ASelection, ASelectionArgs, ASortOrder);
  382. end;
  383. class function TAndroidHelperEx.GetDefaultIconID: Integer;
  384. begin
  385. Result := Context.getApplicationInfo.icon;
  386. end;
  387. class function TAndroidHelperEx.GetDefaultNotificationSound: Jnet_Uri;
  388. begin
  389. Result := TJRingtoneManager.JavaClass.getDefaultUri(TJRingtoneManager.JavaClass.TYPE_NOTIFICATION);
  390. end;
  391. class function TAndroidHelperEx.GetJActivity: JActivity;
  392. begin
  393. Result :=
  394. {$IF CompilerVersion > 27}
  395. TAndroidHelper.Activity
  396. {$ELSE}
  397. SharedActivity
  398. {$ENDIF}
  399. end;
  400. class function TAndroidHelperEx.GetJActivityManager: JActivityManager;
  401. var
  402. AService: JObject;
  403. begin
  404. AService := Context.getSystemService(TJContext.JavaClass.ACTIVITY_SERVICE);
  405. Result := TJActivityManager.Wrap((AService as ILocalObject).GetObjectID);
  406. end;
  407. class function TAndroidHelperEx.GetJContext: JContext;
  408. begin
  409. Result :=
  410. {$IF CompilerVersion > 27}
  411. TAndroidHelper.Context;
  412. {$ELSE}
  413. SharedActivityContext;
  414. {$ENDIF}
  415. end;
  416. class function TAndroidHelperEx.GetMimeType(AFileName: string): JString;
  417. var
  418. LExt: string;
  419. LTypeMap: JMimeTypeMap;
  420. begin
  421. Result := nil;
  422. if AFileName = '' then
  423. Exit;
  424. LExt := LowerCase(ExtractFileExt(AFileName));
  425. if Length(LExt) < 2 then
  426. Exit;
  427. LExt := Copy(LExt, 1);
  428. LTypeMap := TJMimeTypeMap.JavaClass.getSingleton();
  429. if LTypeMap = nil then
  430. Exit;
  431. Result := LTypeMap.getMimeTypeFromExtension(StringToJString(LExt));
  432. if Result <> nil then
  433. Result := Result.trim();
  434. end;
  435. class function TAndroidHelperEx.GetPackageName: string;
  436. begin
  437. Result := JStringToString(Context.getPackageName);
  438. end;
  439. class function TAndroidHelperEx.GetSDCardPath: string;
  440. var
  441. sPath: string;
  442. begin
  443. Result := '';
  444. sPath := System.IOUtils.TPath.GetSharedDocumentsPath;
  445. if Pos(PathDelim, sPath) = 0 then
  446. Exit;
  447. Result := ExtractFilePath(ExcludeTrailingPathDelimiter(sPath));
  448. end;
  449. class function TAndroidHelperEx.GetTargetSdkVersion: Integer;
  450. var
  451. LApplicationInfo: JApplicationInfo;
  452. begin
  453. LApplicationInfo := Context.getPackageManager.getApplicationInfo(Context.getPackageName, 0);
  454. Result := LApplicationInfo.targetSdkVersion;
  455. end;
  456. class function TAndroidHelperEx.HasAssocApp(const Intent: JIntent): Boolean;
  457. var
  458. LList: JList;
  459. begin
  460. Result := False;
  461. if Intent = nil then
  462. Exit;
  463. Result := Intent.resolveActivityInfo(Activity.getPackageManager(),
  464. TJPackageManager.JavaClass.MATCH_DEFAULT_ONLY) <> nil;
  465. // Android 6+ APP LINK closed Will cause list is null.
  466. // Activity with <action android:name="android.intent.action.VIEW" />
  467. // 被第三方软件套用的时候,比如应用隐藏大师,会导致这里size不正确
  468. // LList := Activity.getPackageManager.queryIntentActivities(Intent,
  469. // TJPackageManager.JavaClass.MATCH_DEFAULT_ONLY);
  470. // Result := (LList = nil) or (LList.size > 0);
  471. end;
  472. class function TAndroidHelperEx.HasAssocApp(const URI: string): Boolean;
  473. var
  474. Intent: JIntent;
  475. begin
  476. Result := False;
  477. if URI = '' then
  478. Exit;
  479. Intent := TJIntent.Create;
  480. Intent.setData(UriParse(URI));
  481. Intent.setAction(StringToJString('android.intent.action.VIEW'));
  482. Result := HasAssocApp(Intent);
  483. end;
  484. class function TAndroidHelperEx.InstallPackage(const AFileName, AAuthority: string): Boolean;
  485. var
  486. LIntent: JIntent;
  487. begin
  488. Result := False;
  489. if Trim(AFileName) = '' then
  490. Exit;
  491. LIntent := TJIntent.Create;
  492. if CheckBuildAndTarget(OREO) then
  493. LIntent.setAction(TJIntent.JavaClass.ACTION_INSTALL_PACKAGE)
  494. else
  495. LIntent.setAction(TJIntent.JavaClass.ACTION_VIEW);
  496. // 没有这个也可以安装成功,但是安装成功后的成功页面,也就是,完成/打开 会无法显示
  497. LIntent.addFlags(TJIntent.JavaClass.FLAG_ACTIVITY_NEW_TASK);
  498. LIntent.setDataAndType(SharedUriFromFile(AFileName, AAuthority),
  499. StringToJString('application/vnd.android.package-archive'));
  500. LIntent.addFlags(TJIntent.JavaClass.FLAG_GRANT_READ_URI_PERMISSION);
  501. Result := StartActivity(LIntent);
  502. end;
  503. class function TAndroidHelperEx.IsActivityForeground: Boolean;
  504. var
  505. LService: JObject;
  506. LRunningApps: JList;
  507. LAppInfo: JActivityManager_RunningAppProcessInfo;
  508. I: Integer;
  509. begin
  510. Result := False;
  511. LService := Context.getSystemService(TJContext.JavaClass.ACTIVITY_SERVICE);
  512. LRunningApps := TJActivityManager.Wrap(TAndroidHelper.JObjectToID(LService)).getRunningAppProcesses;
  513. for I := 0 to LRunningApps.size - 1 do begin
  514. LAppInfo := TJActivityManager_RunningAppProcessInfo.Wrap(TAndroidHelper.JObjectToID(LRunningApps.get(I)));
  515. if LAppInfo.importance = 100 then begin
  516. if LAppInfo.importanceReasonComponent <> nil then begin
  517. if LAppInfo.importanceReasonComponent.getPackageName.equals(Context.getPackageName) then
  518. Exit(True);
  519. end
  520. else if LRunningApps.size = 1 then
  521. Exit(True);
  522. end;
  523. end;
  524. end;
  525. class function TAndroidHelperEx.IsAppInstalled(const APackage: string): Boolean;
  526. begin
  527. Result := False;
  528. try
  529. //只有异常是可靠的,返回值判定不对
  530. Result := Context.getPackageManager.getPackageInfo(StringToJString(APackage),
  531. TJPackageManager.JavaClass.GET_ACTIVITIES) = nil;
  532. Result := True;
  533. except
  534. end;
  535. end;
  536. class function TAndroidHelperEx.KeepScreen(AOn: Boolean): Boolean;
  537. begin
  538. CallInUIThreadAndWaitFinishing(
  539. procedure begin
  540. if AOn then
  541. SharedActivity.getWindow.addFlags(
  542. TJWindowManager_LayoutParams.JavaClass.FLAG_KEEP_SCREEN_ON)
  543. else
  544. SharedActivity.getWindow.clearFlags(
  545. TJWindowManager_LayoutParams.JavaClass.FLAG_KEEP_SCREEN_ON);
  546. end);
  547. end;
  548. class procedure TAndroidHelperEx.SendAppToBack;
  549. begin
  550. //SharedActivityManager.moveTaskToBack(Activity.getTaskId, TJIntent.JavaClass.FLAG_ACTIVITY_NEW_TASK);
  551. Activity.moveTaskToBack(True);
  552. end;
  553. class function TAndroidHelperEx.StartActivity(Intent: JIntent; const Code: Integer): Boolean;
  554. begin
  555. Result := False;
  556. if Code = -1 then
  557. Activity.startActivity(Intent)
  558. else
  559. Activity.startActivityForResult(Intent, Code);
  560. Result := True;
  561. end;
  562. class function TAndroidHelperEx.UriParse(const S: JString): Jnet_Uri;
  563. begin
  564. Result := TJnet_Uri.JavaClass.parse(S);
  565. end;
  566. class function TAndroidHelperEx.UriParse(const S: string): Jnet_Uri;
  567. begin
  568. Result := UriParse(StringToJString(S));
  569. end;
  570. class function TAndroidHelperEx.UriFromFile(const AFileName: string): Jnet_Uri;
  571. begin
  572. Result := UriFromFile(TJFile.JavaClass.init(StringToJString(AFileName)));
  573. end;
  574. class function TAndroidHelperEx.UriFromFile(const AFile: JFile): Jnet_Uri;
  575. begin
  576. Result := TJnet_uri.JavaClass.fromFile(AFile);
  577. end;
  578. class function TAndroidHelperEx.SharedUriFromFile(const AFile: JFile; const AAuthority: string): Jnet_Uri;
  579. var
  580. LAuthority: JString;
  581. begin
  582. if CheckBuildAndTarget(NOUGAT) then begin
  583. if AAuthority <> '' then
  584. LAuthority := StringToJString(AAuthority)
  585. else
  586. LAuthority := Context.getApplicationContext.getPackageName.concat(StringToJString('.fileprovider'));
  587. Result := TJFileProvider.JavaClass.getUriForFile(Context, LAuthority, AFile);
  588. end
  589. else
  590. Result := TJnet_uri.JavaClass.fromFile(AFile);
  591. end;
  592. class function TAndroidHelperEx.SharedUriFromFile(const AFileName, AAuthority: string): Jnet_Uri;
  593. begin
  594. Result := SharedUriFromFile(TJFile.JavaClass.init(StringToJString(AFileName)), AAuthority);
  595. end;
  596. end.