工做中遇到这样一个需求,须要为按键添加一个亲情号,提供一个接口启动内置的APK,思考再三决定更改Framework,利用广播机制去实现。android
1、代码动态自主启动内置APKweb
咱们都知道Android系统为咱们提供了不少服务管理类,PackageManager主要是管理应用程序包,经过它就能够获取应用程序信息并构建Intent,启动对应的应用。除此以外Android还未咱们提供了一些对应的类来管理相关的xml文件,好比说能够经过PackageInfo来获取AndroidManifest.xml文件的信息;PackageItemInfo——AndroidManifest.xml文件中全部节点的基类,提供了这些节点的基本信息好比label、icon、 meta-data。但它并不直接使用,而是由子类继承而后调用相应方法;ApplicationInfo继承自 PackageItemInfo用于获取一个特定程序中节点的信息,好比常见的字段:flags字段: **FLAG_SYSTEM 系统应用程序、FLAG_EXTERNAL_STORAGE 表示该应用安装在sdcard中;ActivityInfo也继承自 PackageItemInfo用于得到应用程序中或者 节点的信息 。咱们能够经过它来获取咱们设置的任何属性好比:theme 、launchMode、launchmode等,还有ServiceInfo与ActivityInfo相似,只不过是用于获取节点的信息;最后一个ResolveInfo是基于节点来获取其上一层目录的信息,好比说、、节点信息。app
一、PackageManager简介ide
PackageManager主要用于管理应用程序包,获取程序中所安装的应用程序包的相关信息,并且是一个抽象类,可是咱们能够经过getPackageManager()方法来获取PackageManager对象。svg
二、代码实现code
包名和mainActivity未知时:xml
/*
** @param packageName 将要去启动第三方app的package
*/
private void runAppByPackage(@NonNull String packageName) {
PackageInfo packageInfo = null;
PackageManager packageManager=getPackageManager();//获取PackageManager对象
try {
packageInfo = packageManager.getPackageInfo(packageName, 0);//经过包名获取PackInfo
} catch (PackageManager.NameNotFoundException e) {
e.printStackTrace();
}
Intent resolveIntent = new Intent(Intent.ACTION_MAIN, null);
resolveIntent.addCategory(Intent.CATEGORY_LAUNCHER);
resolveIntent.setPackage(packageInfo.packageName);
List apps = packageManager.queryIntentActivities(resolveIntent, 0);
ResolveInfo resolveInfo = apps.iterator().next();
if (resolveInfo != null ) {
String pkgeName = resolveInfo.activityInfo.packageName;
String className = resolveInfo.activityInfo.name;
Intent intent = new Intent(Intent.ACTION_MAIN);
intent.addCategory(Intent.CATEGORY_LAUNCHER);
ComponentName cn = new ComponentName(pkgeName, className);
intent.setComponent(cn);
startActivity(intent);
}
}
包名和mainActivity已知时:对象
private void startLenovePlan(){
ComponentName componetName = new ComponentName(
"com.lenovo.ue.service",
"com.lenovo.ue.service.LenovoUEServiceActivity");
try {
Intent intent = new Intent();
intent.setComponent(componetName);
startActivity(intent);
} catch (Exception e) {
}
}
**2、动态发送和接收广播
一、发送自定义广播**blog
Intent intent = new Intent("android.intent.action.CART_BROADCAST_SILENT");
mContext.sendBroadcast(intent);
二、定义广播接收器并处理 继承
private static String BROADCAST_FILTER="android.intent.action.CART_BROADCAST_SILENT";
private AudioManager mAudioManager=(AudioManager) (getActivity().getApplicationContext().getSystemService(Context.AUDIO_SERVICE));
private BroadcastReceiver zenmodeReceiver = new BroadcastReceiver() {
//定义接收广播接收器
@Override
public void onReceive(Context contex, Intent intent) {
String action = intent.getAction();
PreferenceScreen sound = (PreferenceScreen) findPreference("hq_ringtone_volume_settings");
if (action.equals(BROADCAST_FILTER)) {
int ringMode = mAudioManager.getRingerMode();
if (ringMode == AudioManager.RINGER_MODE_SILENT) {
if (findPreference(KEY_RING_VOLUME) != null && findPreference(KEY_NOTIFICATION_VOLUME) != null) {
((VolumeSeekBarPreference) findPreference(KEY_RING_VOLUME)).setSeekBarStatus(false, 0);
}
} else {
if (findPreference(KEY_RING_VOLUME) != null && findPreference(KEY_NOTIFICATION_VOLUME) != null) {
findPreference(KEY_RING_VOLUME).setEnabled(false);
}
}
}
}
};
三、注册广播接收器
@Override
public void onCreate(Bundle icicle) {
super.onCreate(icicle);
addPreferencesFromResource(R.xml.hq_volume_fragment_prefs);
initVolume(getPreferenceScreen());
IntentFilter intent = new IntentFilter();
intent.addAction(BROADCAST_FILTER);
getActivity().getApplicationContext().registerReceiver(zenmodeReceiver,intent);// 注册广播接收器
}
四、使用完毕以后记得释放掉广播,不然会有OOM的隐患
unregisterReceiver(receiver);
这是最简单的内置APK和发送广播方式,有什么问题,请关注个人公众号,咱们再进行技术交流!