权限
要在手机桌面上添加快捷方式,首先需要在manifest中添加权限。
<!-- 添加快捷方式 --> <uses-permission android:name="com.android.launcher.permission.INSTALL_SHORTCUT" /> <!-- 移除快捷方式 --> <uses-permission android:name="com.android.launcher.permission.UNINSTALL_SHORTCUT" /> <!-- 查询快捷方式 --> <uses-permission android:name="com.android.launcher.permission.READ_SETTINGS" />
添加快捷方式
添加快捷方式,是向桌面应用(launcher)发送相关action的广播,相关的action如下:
public static final String ACTION_ADD_SHORTCUT = "com.android.launcher.action.INSTALL_SHORTCUT";
添加快捷方式:
private void addShortcut(String name) { Intent addShortcutIntent = new Intent(ACTION_ADD_SHORTCUT); // 不允许重复创建 addShortcutIntent.putExtra("duplicate", false);// 经测试不是根据快捷方式的名字判断重复的 // 应该是根据快链的Intent来判断是否重复的,即Intent.EXTRA_SHORTCUT_INTENT字段的value // 但是名称不同时,虽然有的手机系统会显示Toast提示重复,仍然会建立快链 // 屏幕上没有空间时会提示 // 注意:重复创建的行为MIUI和三星手机上不太一样,小米上似乎不能重复创建快捷方式 // 名字 addShortcutIntent.putExtra(Intent.EXTRA_SHORTCUT_NAME, name); // 图标 addShortcutIntent.putExtra(Intent.EXTRA_SHORTCUT_ICON_RESOURCE, Intent.ShortcutIconResource.fromContext(MainActivity.this, R.drawable.ic_launcher)); // 设置关联程序 Intent launcherIntent = new Intent(Intent.ACTION_MAIN); launcherIntent.setClass(MainActivity.this, MainActivity.class); launcherIntent.addCategory(Intent.CATEGORY_LAUNCHER); addShortcutIntent .putExtra(Intent.EXTRA_SHORTCUT_INTENT, launcherIntent); // 发送广播 sendBroadcast(addShortcutIntent); }
移除快捷方式
移除快捷方式的action:
public static final String ACTION_REMOVE_SHORTCUT = "com.android.launcher.action.UNINSTALL_SHORTCUT";
移除快捷方式的方法:
private void removeShortcut(String name) { // remove shortcut的方法在小米系统上不管用,在三星上可以移除 Intent intent = new Intent(ACTION_REMOVE_SHORTCUT); // 名字 intent.putExtra(Intent.EXTRA_SHORTCUT_NAME, name); // 设置关联程序 Intent launcherIntent = new Intent(MainActivity.this, MainActivity.class).setAction(Intent.ACTION_MAIN); intent.putExtra(Intent.EXTRA_SHORTCUT_INTENT, launcherIntent); // 发送广播 sendBroadcast(intent); }
在两个手机上测试,发现小米手机上添加了快捷方式后不能移除,三星手机可以。
查询快捷方式
相关的玩法:安装并从Root Explorer这个文件管理器的/system/app目录下,查找Launcher相关的apk,反编译它,获取相关provider及uri的信息。
查询的流程:
1 根据权限获取uri,由于碎片化严重,uri格式不固定,现在收录的只是大部分机型;
2根据uri获取provider提供的内容;
注意:由于部分机型provider不对外开放,故暂不可查,例如:一加手机 华为荣耀v8,华为p9等。
以下是相关代码
public static String getAuthorityFromPermission(Activity activity) {
// 先得到默认的Launcher
Intent intent = new Intent(Intent.ACTION_MAIN);
intent.addCategory(Intent.CATEGORY_HOME);
PackageManager mPackageManager = activity.getPackageManager();
ResolveInfo resolveInfo = mPackageManager.resolveActivity(intent, 0);
if (resolveInfo == null) {
return null;
}
List<ProviderInfo> info = mPackageManager.queryContentProviders(resolveInfo.activityInfo.packageName, resolveInfo.activityInfo.applicationInfo.uid, PackageManager.GET_PROVIDERS);
if (info != null) {
for (int j = 0; j < info.size(); j++) {
ProviderInfo provider = info.get(j);
if (provider.readPermission == null) {
continue;
}
if (Pattern.matches(".*launcher.*READ_SETTINGS", provider.readPermission)) {
return provider.authority;
}
}
}
return null;
}
public static boolean hasShortcut(String appName,Activity activity) {
long start = System.currentTimeMillis();
String authority = getAuthorityFromPermission(activity);
if (authority == null) {
return false;
}
long end = System.currentTimeMillis() - start;
String url = "content://" + authority + "/favorites?notify=true";
try {
Uri CONTENT_URI = Uri.parse(url);
Cursor c = activity.getContentResolver().query(CONTENT_URI, null, " title= ? ", new String[]{appName}, null);
if (c != null && c.moveToNext()) {
c.close();
return true;
}
} catch (Exception e) {
}
return false;
}
参考资料
http://blog.csdn.net/ldj299/article/details/6298452
http://www.xmumu.com/post/2012-04-01/17357119
http://www.cnblogs.com/CoolPigs/p/3317234.html
<activity
android:name=".activity.alert.ShortCutActivity"
android:excludeFromRecents="true"
android:exported="true"
android:launchMode="singleInstance"
android:screenOrientation="portrait"
android:theme="@style/DialogBigNoActionBar">
<intent-filter>
<action android:name="jinjinzhege" />
</intent-filter>
</activity>
<style name="DialogBigNoActionBar" parent="Theme.AppCompat.Light.NoActionBar">
<item name="android:windowIsTranslucent">true</item>
<item name="colorPrimary">@color/colorPrimary</item>
<item name="colorPrimaryDark">@android:color/black</item>
<item name="android:windowBackground">@color/tran_black</item>
</style>