Android开发中适配多种 ROM 的快捷方式是如何实现的?

安卓开发中,要提高开发效率,掌握一些快捷方式是必不可少的,特别是对于android入门阶段的童鞋而言,非常重要。今天小编在安卓开发教程网站上,搜罗了一些常用的Android 适配多种 ROM 的快捷方式,分享给大家,一起来看看他们是怎么实现的吧。

首先,需要获取权限:

<!-- 添加快捷方式 -->

<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" />

然后,定义一下我们要发起的action:

//删除快捷方式的action

public static final String ACTION_REMOVE_SHORTCUT = "com.android.launcher.action.UNINSTALL_SHORTCUT";

//添加快捷方式的action

public static final String ACTION_ADD_SHORTCUT = "com.android.launcher.action.INSTALL_SHORTCUT";

然后写2个方法即可:

//删除快捷方式

public static void removeShortcut(Context context, Intent actionIntent, String name) {

Intent intent = new Intent(ACTION_REMOVE_SHORTCUT);

intent.putExtra(Intent.EXTRA_SHORTCUT_NAME, name);

intent.putExtra("duplicate", false);

intent.putExtra(Intent.EXTRA_SHORTCUT_INTENT, actionIntent);

context.sendBroadcast(intent);

}

//增加快捷方式

public static void addShortcut(Context context, Intent actionIntent, String name,

boolean allowRepeat, Bitmap iconBitmap) {

Intent addShortcutIntent = new Intent(ACTION_ADD_SHORTCUT);

// 是否允许重复创建

addShortcutIntent.putExtra("duplicate", allowRepeat);

// 快捷方式的标题

addShortcutIntent.putExtra(Intent.EXTRA_SHORTCUT_NAME, name);

// 快捷方式的图标

addShortcutIntent.putExtra(Intent.EXTRA_SHORTCUT_ICON, iconBitmap);

// 快捷方式的动作

addShortcutIntent.putExtra(Intent.EXTRA_SHORTCUT_INTENT, actionIntent);

context.sendBroadcast(addShortcutIntent);

}

这种方法,在大部分的手机都能正常使用。但是在某些国产手机上会有各种各样的问题。比如hw的手机这样做就完全无效、小米的手机 无法重复创建快捷方式等。那现在看看有没有什么方法 能解决这些问题。

首先我们来看一张图:

这张图很清晰的告诉我们launcher这个应用下面 是有一个数据库的。

我们可以打开这个数据库 看看到底是什么?

你看这个就一目了然了,快捷方式都是存在于这个表里面的!所以 ,我们除了发送广播可以创建快捷方式以外,我们直接操作数据库也应该是可以创建快捷方式的。

但是我们要注意啊,各家的lanucher 都有很大不同的,上面的截图 是用的模拟器,所以是官方的rom,那路径肯定就是常规的,但是深度定制的android系统,大家都知道各家lanucher都是自己在做,所以favorites 这张表的位置到底在哪里,这就是个问题了。我们需要先查找出来这张表到底在哪个位置才能好操作这个表,要想知道这个表在哪个位置,我们首先要知道对于这个rom来说,是哪个lanucher在启作用?包名是什么?

//此函数返回当前rom下的lanucher的包名

private String getCurrentLanucherPackageName(Context context)

{

//这个intent很好理解 就是启动lanucher的intent

Intent intent=new Intent(Intent.ACTION_MAIN);

intent.addCategory(Intent.CATEGORY_HOME);

//getPackageManager().resolveActivity 这个函数就是查询是否有符合条件的activity的

ResolveInfo res=context.getPackageManager().resolveActivity(intent,0);

//为避免空指针 我们要判定下空,虽然你我都知道这种情况不会发生

if(res==null||res.activityInfo==null)

{

return "";

}

return res.activityInfo.packageName;

}

有些人可能对于4-5行还是理解不了,为什么这个相当于是启动lanucher的intent 其实很简单,我们在android studio里 run一个app的时候,我们通常都能看到下面这个界面:

你看android studio 实际上安装完我们的应用以后 也是通过 这个intent来启动手机里的lanucher 然后让lanucher来运行我们的app,因为studio也不知道你手机里的lanucher到底是哪个,所以只能用intent来完成,这也就是上述代码能必定成功运行的原因!否则的话 android studio 就无法把你的app run进去了~~

当然了你也可以进入shell ,打ps 命令看看到底是不是有这个 包名的 进程在 运行中~~

拿到了我们的lanucher的包名,下面一步就是去我们的包名下操作那个数据库即可了。那显然这个步骤 最合适的就是contentprovider来操作了。

//此函数返回 要查找的permission的 provider的 authority

private String getAuthorityFromPermission(Context context, String permission) {

//返回安装的app的 provider的信息

List<PackageInfo> packs = context.getPackageManager().getInstalledPackages(PackageManager.GET_PROVIDERS);

//遍历获取到的安装包的信息

for (PackageInfo pack : packs) {

//每个安装包提供的provider 都在这个数组里面

ProviderInfo[] providers = pack.providers;

if (providers != null) {

//遍历每个provider 看需要的权限是否与我们传进来的权限参数相等

for (ProviderInfo providerInfo : providers) {

if (permission.equals(providerInfo.readPermission) || permission.equals(providerInfo.writePermission)) {

return providerInfo.authority;

}

}

}

}

return "";

}

private String getAuthorityFromPermissionDefault(Context context) {

return getAuthorityFromPermission(context, "com.android.launcher.permission.READ_SETTINGS");

}

private  Uri getUriFromLauncher(Context context) {

StringBuilder uriStrBuilder = new StringBuilder();

//为了速度考虑,这里我们先查找默认的 看是否能查找到 因为多数手机的rom还是用的默认的lanucher

String authority = getAuthorityFromPermissionDefault(context);

//如果找不到的话 就说明这个rom一定是用的其他的自定义的lanucher。那就拼一下 这个自定义的lanucher的permission再去查找一次

if (authority == null || authority.trim().equals("")) {

authority = getAuthorityFromPermission(context,getCurrentLanucherPackageName(context) + ".permission.READ_SETTINGS");

}

uriStrBuilder.append("content://");

//如果连上面的方法都查找不到这个authority的话 那下面的方法 就肯定查找到了 但是很少有情况会是如下这种

//多数都是else里面的逻辑

if (TextUtils.isEmpty(authority)) {

int sdkInt = android.os.Build.VERSION.SDK_INT;

if (sdkInt < 8) { // Android 2.1.x(API 7)以及以下的

uriStrBuilder.append("com.android.launcher.settings");

} else if (sdkInt < 19) {// Android 4.4以下

uriStrBuilder.append("com.android.launcher2.settings");

} else {// 4.4以及以上

uriStrBuilder.append("com.android.launcher3.settings");

}

} else {

uriStrBuilder.append(authority);

}

uriStrBuilder.append("/favorites?notify=true");

return Uri.parse(uriStrBuilder.toString());

}

好,到这里 我们就能拿到所有rom 任意一款手机的 快捷方式 那张数据库表对应的uri了。

到这里应该来说问题就基本解决了,你所有对快捷方式的操作都可以直接通过这个uri来进行,无非就是一些crud的拼装。

再也不需要经过广播 那一道程序了,但是这里要注意的是这种方法通常耗时都比较久,根据手机的性能的不同 200ms-600ms 才能完成uri的查找。所以记得做一下异步处理。

并且所有你对lanucher的 需求 都可以这么来做,比方说有些功能 在使用原生lanucher的手机上使用正常,在小米 华为 oppo使用不正常了,你就去找出来那个不正常的rom的lanucher的 包名,然后找出他provider需要的权限,然后在你的manifest里 直接加权限 应该就可以正常使用了。第三方lanucher 引起的bug 基本上都可以通过这个方案来解决。

以上就是Android 适配多种 ROM 的快捷方式的实现方法,大家不妨在自己的android环境下试一下伤面子这些操作,看看有没有什么问题。

时间: 2024-08-04 09:34:24

Android开发中适配多种 ROM 的快捷方式是如何实现的?的相关文章

Android 适配多种ROM的快捷方式

快捷方式 应该来说 很多人都做过,我们就来看一下基本的快捷方式 是怎么实现的,会有什么问题? 首先 肯定要获取权限: 1 <!-- 添加快捷方式 --> 2 <uses-permission android:name="com.android.launcher.permission.INSTALL_SHORTCUT" /> 3 <!-- 移除快捷方式 --> 4 <uses-permission android:name="com.a

Android开发中的问题及相应解决(持续更新)

最近博客写的少了,以后还得经常更新才行. ------------------------------------------------------------ 1.try cath 异常不全导致的意外问题. 之所以将此问题放在第一位,是因为前阵子中项目中出现了此类情况,并且这种情况很容易不注意或遗忘,但其错误却是致命的. 在Java/Android开发中,当调用一个函数时,此函数抛出一个A类型的异常,很自然的,在调用的地方我们会try.. catch此异常,并且绝大多数情况下,捕获异常是由E

Android开发中的MVP架构(转)

写在前面,本博客来源于公众号文章:http://mp.weixin.qq.com/s?__biz=MzA3MDMyMjkzNg==&mid=402435540&idx=1&sn=1cd10bd9efaac7083575367a8b4af52f&scene=1&srcid=0910ARzPpBvVYPI1NDBZnixa#wechat_redirect 最近越来越多的人开始谈论架构.我周围的同事和工程师也是如此.尽管我还不是特别深入理解MVP和DDD,但是我们的新项目

Android开发中无处不在的设计模式——单例模式

对于开发人员来说,设计模式有时候就是一道坎,但是设计模式又非常有用,过了这道坎,它可以让你水平提高一个档次.而在android开发中,必要的了解一些设计模式又是非常有必要的.对于想系统的学习设计模式的同学,这里推荐2本书.一本是Head First系列的Head Hirst Design Pattern,英文好的可以看英文,可以多读几遍.另外一本是大话设计模式. 这篇文章介绍一个模式,就是单例模式,因为个人觉得这个模式理解起来最容易,而且不是太复杂. 首先了解一些什么是单例,从名字中就可以听出来

Android开发中Eclispe相关问题及相应解决(持续更新)

1.Eclipse项目中的Android Private Libraries没有自动生成. 一般而言,在Android开发中,项目中引用到的jar包会放到项目目录中的libs中,引入库会放到Android Dependencies中,对于放置于libs中的jar包,会自动置于项目中的Android Private Libraries下,如果没有自动生成,clean项目后一般在console下会发现有相应的错误提示,修正即可.如:项目中引用了不同的v4兼容包等会出现此类情况. 2.Eclipse中

MVP模式在Android开发中的最佳实践

这篇文章拖了好久了,一直存在草稿箱里没有继续写,趁今天有空,撸撸完. 回想一下,你刚刚学习Android的时候,总会看到一些书上写着,Android使用的是MVC模式,Activity就是一个Controller,或许那个时候,你没有什么深刻的体会.随着经验的积累.你发现,Activity既是Controller,掌管着许许多多的业务逻辑,同时它也作为View的一部分,控制着视图层的显示.久而久之,这个Controller便显得过于重,职责不再那么单一. 于是,再后来,为了使Activity的职

android开发中的5种存储数据方式

数据存储在开发中是使用最频繁的,根据不同的情况选择不同的存储数据方式对于提高开发效率很有帮助.下面笔者在主要介绍Android平台中实现数据存储的5种方式. 1.使用SharedPreferences存储数据 SharedPreferences是Android平台上一个轻量级的存储类,主要是保存一些常用的配置比如窗口状态,一般在Activity中 重载窗口状态onSaveInstance State保存一般使用SharedPreferences完成,它提供了Android平台常规的Long长 整

android开发分辨率适配总结

重要概念 什么是屏幕尺寸.屏幕分辨率.屏幕像素密度? 什么是dp.dip.dpi.sp.px?他们之间的关系是什么? 什么是mdpi.hdpi.xdpi.xxdpi?如何计算和区分? 在下面的内容中我们将介绍这些概念. 屏幕尺寸 屏幕尺寸指屏幕的对角线的长度,单位是英寸,1英寸=2.54厘米 比如常见的屏幕尺寸有2.4.2.8.3.5.3.7.4.2.5.0.5.5.6.0等 屏幕分辨率 屏幕分辨率是指在横纵向上的像素点数,单位是px,1px=1个像素点.一般以纵向像素*横向像素,如1960*1

Android开发中,那些让你觉得相见恨晚的方法、类或接口

本篇文章内容提取自知乎android开发中,有哪些让你觉得相见恨晚的方法.类或接口?,其实有一部是JAVA的,但是在android开发中也算常见.大多数的函数自己还是见过的,这里记录一下备忘.同时呢,也推荐一个github项目,里面记录了自己日常开发中见过的比较有用的东西开发中常用的工具.链接 Throwable类中的getStackTrace()方法,根据这个方法可以得到函数的逐层调用地址,其返回值为StackTraceElement[],而在StackTraceElement类中有四个方法g