Android O PackageInstaller 解析
Android O 8.0
1.src\com\android\packageinstaller\permission\mode\PermissionGroups.java
@Override
public List<PermissionGroup> loadInBackground() {
ArraySet<String> launcherPkgs = Utils.getLauncherPackages(getContext());
PermissionApps.PmCache pmCache = new PermissionApps.PmCache(
getContext().getPackageManager()); List<PermissionGroup> groups = new ArrayList<>();
Set<String> seenPermissions = new ArraySet<>(); PackageManager packageManager = getContext().getPackageManager();
List<PermissionGroupInfo> groupInfos = packageManager.getAllPermissionGroups(0); for (PermissionGroupInfo groupInfo : groupInfos) {
// Mare sure we respond to cancellation.
if (isLoadInBackgroundCanceled()) {
return Collections.emptyList();
} // 得到权限组里的所有权限
// Get the permissions in this group.
final List<PermissionInfo> groupPermissions;
try {
groupPermissions = packageManager.queryPermissionsByGroup(groupInfo.name, 0);
} catch (PackageManager.NameNotFoundException e) {
continue;
} boolean hasRuntimePermissions = false; // 查看某个权限组是否含有运行时权限
// Cache seen permissions and see if group has runtime permissions.
for (PermissionInfo groupPermission : groupPermissions) {
seenPermissions.add(groupPermission.name);
if ((groupPermission.protectionLevel & PermissionInfo.PROTECTION_MASK_BASE)
== PermissionInfo.PROTECTION_DANGEROUS
&& (groupPermission.flags & PermissionInfo.FLAG_INSTALLED) != 0
&& (groupPermission.flags & PermissionInfo.FLAG_REMOVED) == 0) {
hasRuntimePermissions = true;
}
} // 权限组里没有运行时权限,我们不感兴趣,继续循环查询
// No runtime permissions - not interesting for us.
if (!hasRuntimePermissions) {
continue;
} // 如果权限组里含有运行时权限,则继续往下执行 // 获取权限组的label
CharSequence label = loadItemInfoLabel(groupInfo);
Drawable icon = loadItemInfoIcon(groupInfo); // 根据权限组 新建 PermissionApps 对象
PermissionApps permApps = new PermissionApps(getContext(), groupInfo.name, null,
pmCache);
permApps.refreshSync(); // 创建 PermissionGroup 对象
// Create the group and add to the list.
PermissionGroup group = new PermissionGroup(groupInfo.name,
groupInfo.packageName, label, icon, permApps.getTotalCount(launcherPkgs),
permApps.getGrantedCount(launcherPkgs));
groups.add(group);
} // 获取所有的安装包权限
// Make sure we add groups for lone runtime permissions.
List<PackageInfo> installedPackages = getContext().getPackageManager()
.getInstalledPackages(PackageManager.GET_PERMISSIONS); // We will filter out permissions that no package requests.
Set<String> requestedPermissions = new ArraySet<>();
for (PackageInfo installedPackage : installedPackages) {
if (installedPackage.requestedPermissions == null) {
continue;
}
// 将安装包请求的权限加入到 列表中
for (String requestedPermission : installedPackage.requestedPermissions) {
requestedPermissions.add(requestedPermission);
}
} for (PackageInfo installedPackage : installedPackages) {
if (installedPackage.permissions == null) {
continue;
} // 自定义的权限
for (PermissionInfo permissionInfo : installedPackage.permissions) {
// If we have handled this permission, no more work to do.
if (!seenPermissions.add(permissionInfo.name)) {
continue;
} // 关心的是 运行时权限
// We care only about installed runtime permissions.
if ((permissionInfo.protectionLevel & PermissionInfo.PROTECTION_MASK_BASE)
!= PermissionInfo.PROTECTION_DANGEROUS
|| (permissionInfo.flags & PermissionInfo.FLAG_INSTALLED) == 0) {
continue;
} // 没有app使用那个自定义的权限
// If no app uses this permission,
if (!requestedPermissions.contains(permissionInfo.name)) {
continue;
} CharSequence label = loadItemInfoLabel(permissionInfo);
Drawable icon = loadItemInfoIcon(permissionInfo); PermissionApps permApps = new PermissionApps(getContext(), permissionInfo.name,
null, pmCache);
permApps.refreshSync(); // Create the group and add to the list.
PermissionGroup group = new PermissionGroup(permissionInfo.name,
permissionInfo.packageName, label, icon,
permApps.getTotalCount(launcherPkgs),
permApps.getGrantedCount(launcherPkgs));
groups.add(group);
}
} Collections.sort(groups);
return groups;
}
2.PermissionApps.java
/**
* Refresh the state and do not return until it finishes. Should not be called while an {@link
* #refresh async referesh} is in progress.
*/
public void refreshSync() {
mSkipUi = true;
createMap(loadPermissionApps());
}
3.
src\com\android\packageinstaller\permission\utils\EventLogger.java
/**
* For each permission there are four events. The events are in the order of
* #ALL_DANGEROUS_PERMISSIONS. The four events per permission are (in that order): "requested",
* "granted", "denied", and "revoked".
*/
public class EventLogger {
private static final String LOG_TAG = EventLogger.class.getSimpleName(); /** All dangerous permission names in the same order as the events in MetricsEvent */
private static final List<String> ALL_DANGEROUS_PERMISSIONS = Arrays.asList(
Manifest.permission.READ_CALENDAR,
Manifest.permission.WRITE_CALENDAR,
Manifest.permission.CAMERA,
Manifest.permission.READ_CONTACTS,
Manifest.permission.WRITE_CONTACTS,
Manifest.permission.GET_ACCOUNTS,
Manifest.permission.ACCESS_FINE_LOCATION,
Manifest.permission.ACCESS_COARSE_LOCATION,
Manifest.permission.RECORD_AUDIO,
Manifest.permission.READ_PHONE_STATE,
Manifest.permission.CALL_PHONE,
Manifest.permission.READ_CALL_LOG,
Manifest.permission.WRITE_CALL_LOG,
Manifest.permission.ADD_VOICEMAIL,
Manifest.permission.USE_SIP,
Manifest.permission.PROCESS_OUTGOING_CALLS,
Manifest.permission.READ_CELL_BROADCASTS,
Manifest.permission.BODY_SENSORS,
Manifest.permission.SEND_SMS,
Manifest.permission.RECEIVE_SMS,
Manifest.permission.READ_SMS,
Manifest.permission.RECEIVE_WAP_PUSH,
Manifest.permission.RECEIVE_MMS,
Manifest.permission.READ_EXTERNAL_STORAGE,
Manifest.permission.WRITE_EXTERNAL_STORAGE,
Manifest.permission.READ_PHONE_NUMBERS,
Manifest.permission.ANSWER_PHONE_CALLS,
Manifest.permission.ACCEPT_HANDOVER); private static final List<String> ALL_APPOP_PERMISSIONS = Arrays.asList(
Manifest.permission.ACCESS_NOTIFICATIONS,
Manifest.permission.SYSTEM_ALERT_WINDOW,
Manifest.permission.WRITE_SETTINGS,
Manifest.permission.REQUEST_INSTALL_PACKAGES); ...... }
4.
src\com\android\packageinstaller\permission\utils\Utils.java
public static final String OS_PKG = "android";
public static final String[] MODERN_PERMISSION_GROUPS = { // 目前平台支持的权限组
Manifest.permission_group.CALENDAR,
Manifest.permission_group.CAMERA,
Manifest.permission_group.CONTACTS,
Manifest.permission_group.LOCATION,
Manifest.permission_group.SENSORS,
Manifest.permission_group.SMS,
Manifest.permission_group.PHONE,
Manifest.permission_group.MICROPHONE,
Manifest.permission_group.STORAGE
};
private static final Intent LAUNCHER_INTENT = new Intent(Intent.ACTION_MAIN, null)
.addCategory(Intent.CATEGORY_LAUNCHER);
5.
public class DeviceUtils {
public static boolean isTelevision(Context context) { // 电视
int uiMode = context.getResources().getConfiguration().uiMode;
return (uiMode & Configuration.UI_MODE_TYPE_MASK) == Configuration.UI_MODE_TYPE_TELEVISION;
}
public static boolean isWear(final Context context) {// 穿戴设备
return context.getPackageManager().hasSystemFeature(PackageManager.FEATURE_WATCH);
}
public static boolean isAuto(Context context) {// 汽车、自动设备
return context.getPackageManager().hasSystemFeature(PackageManager.FEATURE_AUTOMOTIVE);
}
}
Android O PackageInstaller 解析的更多相关文章
- Android Service完全解析,关于服务你所需知道的一切(下)
转载请注册出处:http://blog.csdn.net/guolin_blog/article/details/9797169 在上一篇文章中,我们学习了Android Service相关的许多重要 ...
- Android Service完全解析,关于服务你所需知道的一切(上)
转载请注明出处:http://blog.csdn.net/guolin_blog/article/details/11952435 相信大多数朋友对Service这个名词都不会陌生,没错,一个老练的A ...
- [转] Android Volley完全解析(一),初识Volley的基本用法
版权声明:本文出自郭霖的博客,转载必须注明出处. 目录(?)[-] Volley简介 下载Volley StringRequest的用法 JsonRequest的用法 转载请注明出处:http ...
- Android OkHttp完全解析 --zz
参考文章 https://github.com/square/okhttp http://square.github.io/okhttp/ 泡网OkHttp使用教程 Android OkHttp完全解 ...
- Android IntentService完全解析 当Service遇到Handler
一 概述 大家都清楚,在Android的开发中,凡是遇到耗时的操作尽可能的会交给Service去做,比如我们上传多张图,上传的过程用户可能将应用置于后台,然后干别的去了,我们的Activity就很可能 ...
- Android Volley完全解析
1. Volley简介 我们平时在开发Android应用的时候不可避免地都需要用到网络技术,而多数情况下应用程序都会使用HTTP协议来发送和接收网络数据.Android系统中主要提供了两种方式来进行H ...
- Android OkHttp完全解析 是时候来了解OkHttp了
Android OkHttp完全解析 是时候来了解OkHttp了 标签: AndroidOkHttp 2015-08-24 15:36 316254人阅读 评论(306) 收藏 举报 分类: [an ...
- Android Bitmap 全面解析(四)图片处理效果对比 ...
对比对象: UIL Volley 官方教程中的方法(此系列教程一里介绍的,ImageLoader的处理方法和官方的差不多) -------------------------------------- ...
- android源码解析(十七)-->Activity布局加载流程
版权声明:本文为博主原创文章,未经博主允许不得转载. 好吧,终于要开始讲讲Activity的布局加载流程了,大家都知道在Android体系中Activity扮演了一个界面展示的角色,这也是它与andr ...
随机推荐
- 指定webapi 返回 json 格式 ; GlobalConfiguration.Configuration.Formatters.Clear()
因为 Internet Explorer 和 Firefox 发送了不同的 Accept 头,所以 web API 在响应里就发送了不同的内容类型. 解决方法,在 Global.asax的 App ...
- POI写docx文件table中的单元格水平、垂直对齐
核心示例代码 垂直对齐 XWPFTableCell cell = table.getRow(i).getCell(j); cell.setVerticalAlignment(XWPFTableCell ...
- vue加百度统计代码(亲测有效)
申请百度统计后,会得到一段JS代码,需要插入到每个网页中去,在Vue.js项目首先想到的可能就是,把统计代码插入到index.html入口文件中,这样就全局插入,每个页面就都有了;这样做就涉及到一个问 ...
- Vue.nextTick和Vue.$nextTick
`Vue.nextTick(callback)`,当数据发生变化,更新后执行回调. `Vue.$nextTick(callback)`,当dom发生变化,更新后执行的回调. 参考原文:http://w ...
- js堆栈与队列简单记忆
在面向对象的程序设计里,一般都提供了实现队列(queue)和堆栈(stack)的方法,而对于JS来说,我们可以实现数组的相关操作,来实现队列和堆栈的功能,看下面的相关介绍. 一 看一下它们的性质,这种 ...
- es 5 数组reduce方法记忆
reduce() 方法接收一个函数作为累加器(accumulator),数组中的每个值(从左到右)开始合并,最终为一个值. 概念:对数组中的所有元素调用指定的回调函数.该回调函数的返回值为累积结果,并 ...
- git branch 命令
1.git init 该命令执行之后并没有创建branch 2.git add 添加文件,这时branch 也还没生成.git branch name也没用 3.git commit 提交到git r ...
- C# winform pictureBox如何突出显示,放大并给pictureBox边框变色
PictureBox old = null; private void pictureBox2_Click(object sender, EventArgs e) { PictureBox p = ( ...
- Java使用泛型实现栈结构
泛型是Java SE5.0的重要特性,使用泛型编程可以使代码获得最大的重用.由于在使用泛型时要指明泛型的具体类型,这样就避免了类型转换.本实例将使用泛型来实现一个栈结构,并对其进行测试. 思路分析:既 ...
- css counter的使用方法
<!doctype html> <html lang="en"> <head> <meta charset="UTF-8&quo ...