ksAndroid.Helpers.pas 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356
  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,
  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. /// Converts file to uri, using FileProvider if target API >= 24
  76. /// </summary>
  77. /// <remarks>
  78. /// Use this only when accessing files with an "external" URI
  79. /// </remarks>
  80. class function UriFromFile(const AFile: JFile; const AAuthority: string = ''): Jnet_Uri; overload; static;
  81. /// <summary>
  82. /// Converts filename to uri, using FileProvider if target API >= 24
  83. /// </summary>
  84. /// <remarks>
  85. /// Use this only when accessing files with an "external" URI
  86. /// </remarks>
  87. class function UriFromFile(const AFileName: string; const AAuthority: string = ''): Jnet_Uri; overload; static;
  88. /// <summary>Returns Java Application Context</summary>
  89. class property Context: JContext read GetJContext;
  90. /// <summary>Returns Java Application Activity</summary>
  91. /// <remarks>An exception will be launched if there is no activity, for example a Service</remarks>
  92. class property Activity: JActivity read GetJActivity;
  93. /// <summary>Returns Java Application Activity Manager</summary>
  94. class property ActivityManager: JActivityManager read GetJActivityManager;
  95. /// <remarks>Need reorder tasks permission</remarks>
  96. class procedure BringAppToFront; static;
  97. /// <remarks>Need reorder tasks permission</remarks>
  98. class procedure SendAppToBack; static;
  99. /// <summary>Call a Java Activity</summary>
  100. class function StartActivity(Intent: JIntent; const Code: Integer = -1): Boolean; static;
  101. /// <summary>Returns Application package name</summary>
  102. class function GetPackageName: string; static;
  103. /// <summary>Returns MimeType from filename</summary>
  104. class function GetMimeType(AFileName: string): JString; static;
  105. /// <summary>Checks if there is at least one application capable of receiving the intent.</summary>
  106. class function HasAssocApp(const URI: string): Boolean; overload; static;
  107. /// <summary>Checks if there is at least one application capable of receiving the intent.</summary>
  108. class function HasAssocApp(const Intent: JIntent): Boolean; overload; static;
  109. /// <summary>Install an android package: xxx.apk</summary>
  110. /// <remarks>
  111. /// <uses-permission android:name="android.permission.REQUEST_INSTALL_PACKAGES"/>
  112. /// </remarks>
  113. class function InstallPackage(const AFileName: string; const AAuthority: string = ''): Boolean; static;
  114. /// <summary>Check if an application is installed</summary>
  115. class function IsAppInstalled(const APackage: string): Boolean; static;
  116. /// <summary>Add/Cear FLAG_KEEP_SCREEN_ON</summary>
  117. class function KeepScreen(AOn: Boolean): Boolean; static;
  118. end;
  119. implementation
  120. uses
  121. Androidapi.JNIBridge, Androidapi.JNI.Os, Androidapi.JNI.Support,
  122. Androidapi.JNI.Media, Androidapi.JNI.Provider, Androidapi.JNI.Webkit,
  123. FMX.Helpers.Android, Androidapi.Helpers;
  124. { TAndroidHelperEx }
  125. class procedure TAndroidHelperEx.BringAppToFront;
  126. begin
  127. ActivityManager.moveTaskToFront(Activity.getTaskId, TJIntent.JavaClass.FLAG_ACTIVITY_NEW_TASK);
  128. end;
  129. class function TAndroidHelperEx.CheckBuildAndTarget(const AValue: Integer): Boolean;
  130. begin
  131. Result := (GetBuildSdkVersion >= AValue) and (GetTargetSdkVersion >= AValue);
  132. end;
  133. class function TAndroidHelperEx.GetBuildSdkVersion: Integer;
  134. begin
  135. Result := TJBuild_VERSION.JavaClass.SDK_INT;
  136. end;
  137. class function TAndroidHelperEx.GetClass(const APackageClassName: string): Jlang_Class;
  138. begin
  139. Result := TJLang_Class.JavaClass.forName(StringToJString(APackageClassName), True, Context.getClassLoader);
  140. end;
  141. class function TAndroidHelperEx.GetDefaultIconID: Integer;
  142. begin
  143. Result := Context.getApplicationInfo.icon;
  144. end;
  145. class function TAndroidHelperEx.GetDefaultNotificationSound: Jnet_Uri;
  146. begin
  147. Result := TJRingtoneManager.JavaClass.getDefaultUri(TJRingtoneManager.JavaClass.TYPE_NOTIFICATION);
  148. end;
  149. class function TAndroidHelperEx.GetJActivity: JActivity;
  150. begin
  151. Result :=
  152. {$IF CompilerVersion > 27}
  153. TAndroidHelper.Activity
  154. {$ELSE}
  155. SharedActivity
  156. {$ENDIF}
  157. end;
  158. class function TAndroidHelperEx.GetJActivityManager: JActivityManager;
  159. var
  160. AService: JObject;
  161. begin
  162. AService := Context.getSystemService(TJContext.JavaClass.ACTIVITY_SERVICE);
  163. Result := TJActivityManager.Wrap((AService as ILocalObject).GetObjectID);
  164. end;
  165. class function TAndroidHelperEx.GetJContext: JContext;
  166. begin
  167. Result :=
  168. {$IF CompilerVersion > 27}
  169. TAndroidHelper.Context;
  170. {$ELSE}
  171. SharedActivityContext;
  172. {$ENDIF}
  173. end;
  174. class function TAndroidHelperEx.GetMimeType(AFileName: string): JString;
  175. var
  176. LExt: string;
  177. LTypeMap: JMimeTypeMap;
  178. begin
  179. Result := nil;
  180. if AFileName = '' then
  181. Exit;
  182. LExt := LowerCase(ExtractFileExt(AFileName));
  183. if Length(LExt) < 2 then
  184. Exit;
  185. LExt := Copy(LExt, 1);
  186. LTypeMap := TJMimeTypeMap.JavaClass.getSingleton();
  187. if LTypeMap = nil then
  188. Exit;
  189. Result := LTypeMap.getMimeTypeFromExtension(StringToJString(LExt));
  190. end;
  191. class function TAndroidHelperEx.GetPackageName: string;
  192. begin
  193. Result := JStringToString(Context.getPackageName);
  194. end;
  195. class function TAndroidHelperEx.GetTargetSdkVersion: Integer;
  196. var
  197. LApplicationInfo: JApplicationInfo;
  198. begin
  199. LApplicationInfo := Context.getPackageManager.getApplicationInfo(Context.getPackageName, 0);
  200. Result := LApplicationInfo.targetSdkVersion;
  201. end;
  202. class function TAndroidHelperEx.HasAssocApp(const Intent: JIntent): Boolean;
  203. var
  204. LList: JList;
  205. begin
  206. // Android 6+ APP LINK closed Will cause list is null.
  207. // Activity with <action android:name="android.intent.action.VIEW" />
  208. LList := Activity.getPackageManager.queryIntentActivities(Intent,
  209. TJPackageManager.JavaClass.MATCH_DEFAULT_ONLY);
  210. Result := (LList <> nil) and (LList.size > 0);
  211. end;
  212. class function TAndroidHelperEx.HasAssocApp(const URI: string): Boolean;
  213. var
  214. Intent: JIntent;
  215. begin
  216. Result := False;
  217. Intent := TJIntent.Create;
  218. Intent.setData(TJnet_Uri.JavaClass.parse(StringToJString(URI)));
  219. Intent.setAction(StringToJString('android.intent.action.VIEW'));
  220. Result := HasAssocApp(Intent);
  221. end;
  222. class function TAndroidHelperEx.InstallPackage(const AFileName, AAuthority: string): Boolean;
  223. var
  224. LIntent: JIntent;
  225. begin
  226. Result := False;
  227. if Trim(AFileName) = '' then
  228. Exit;
  229. LIntent := TJIntent.Create;
  230. if CheckBuildAndTarget(OREO) then
  231. LIntent.setAction(TJIntent.JavaClass.ACTION_INSTALL_PACKAGE)
  232. else
  233. LIntent.setAction(TJIntent.JavaClass.ACTION_VIEW);
  234. // 没有这个也可以安装成功,但是安装成功后的成功页面,也就是,完成/打开 会无法显示
  235. LIntent.addFlags(TJIntent.JavaClass.FLAG_ACTIVITY_NEW_TASK);
  236. LIntent.setDataAndType(UriFromFile(AFileName, AAuthority),
  237. StringToJString('application/vnd.android.package-archive'));
  238. LIntent.addFlags(TJIntent.JavaClass.FLAG_GRANT_READ_URI_PERMISSION);
  239. Result := StartActivity(LIntent);
  240. end;
  241. class function TAndroidHelperEx.IsActivityForeground: Boolean;
  242. var
  243. LService: JObject;
  244. LRunningApps: JList;
  245. LAppInfo: JActivityManager_RunningAppProcessInfo;
  246. I: Integer;
  247. begin
  248. Result := False;
  249. LService := Context.getSystemService(TJContext.JavaClass.ACTIVITY_SERVICE);
  250. LRunningApps := TJActivityManager.Wrap(TAndroidHelper.JObjectToID(LService)).getRunningAppProcesses;
  251. for I := 0 to LRunningApps.size - 1 do begin
  252. LAppInfo := TJActivityManager_RunningAppProcessInfo.Wrap(TAndroidHelper.JObjectToID(LRunningApps.get(I)));
  253. if LAppInfo.importance = 100 then begin
  254. if LAppInfo.importanceReasonComponent <> nil then begin
  255. if LAppInfo.importanceReasonComponent.getPackageName.equals(Context.getPackageName) then
  256. Exit(True);
  257. end
  258. else if LRunningApps.size = 1 then
  259. Exit(True);
  260. end;
  261. end;
  262. end;
  263. class function TAndroidHelperEx.IsAppInstalled(const APackage: string): Boolean;
  264. begin
  265. Result := False;
  266. try
  267. //只有异常是可靠的,返回值判定不对
  268. Result := Context.getPackageManager.getPackageInfo(StringToJString(APackage),
  269. TJPackageManager.JavaClass.GET_ACTIVITIES) = nil;
  270. Result := True;
  271. except
  272. end;
  273. end;
  274. class function TAndroidHelperEx.KeepScreen(AOn: Boolean): Boolean;
  275. begin
  276. CallInUIThreadAndWaitFinishing(
  277. procedure begin
  278. if AOn then
  279. SharedActivity.getWindow.addFlags(
  280. TJWindowManager_LayoutParams.JavaClass.FLAG_KEEP_SCREEN_ON)
  281. else
  282. SharedActivity.getWindow.clearFlags(
  283. TJWindowManager_LayoutParams.JavaClass.FLAG_KEEP_SCREEN_ON);
  284. end);
  285. end;
  286. class procedure TAndroidHelperEx.SendAppToBack;
  287. begin
  288. //SharedActivityManager.moveTaskToBack(Activity.getTaskId, TJIntent.JavaClass.FLAG_ACTIVITY_NEW_TASK);
  289. Activity.moveTaskToBack(True);
  290. end;
  291. class function TAndroidHelperEx.StartActivity(Intent: JIntent; const Code: Integer): Boolean;
  292. begin
  293. Result := False;
  294. if HasAssocApp(Intent) then begin
  295. if Code = -1 then
  296. Activity.startActivity(Intent)
  297. else
  298. Activity.startActivityForResult(Intent, Code);
  299. Result := True;
  300. end;
  301. end;
  302. class function TAndroidHelperEx.UriFromFile(const AFile: JFile; const AAuthority: string): Jnet_Uri;
  303. var
  304. LAuthority: JString;
  305. begin
  306. if CheckBuildAndTarget(NOUGAT) then begin
  307. if AAuthority <> '' then
  308. LAuthority := StringToJString(AAuthority)
  309. else
  310. LAuthority := Context.getApplicationContext.getPackageName.concat(StringToJString('.fileprovider'));
  311. Result := TJFileProvider.JavaClass.getUriForFile(Context, LAuthority, AFile);
  312. end
  313. else
  314. Result := TJnet_uri.JavaClass.fromFile(AFile);
  315. end;
  316. class function TAndroidHelperEx.UriFromFile(const AFileName, AAuthority: string): Jnet_Uri;
  317. begin
  318. Result := UriFromFile(TJFile.JavaClass.init(StringToJString(AFileName)), AAuthority);
  319. end;
  320. end.