Android 实现双Launcher的无缝切换

第一种方式 :  清理Launcher的默认设置的 默认值相当于在设置->应用程序->管理应用程序->所有应用程序列表,找到之前设置的

默认Launcher,并取消了默认值,系统会重新弹出launcher选择框。

private void clearDefaultValues(String packageName) {

getPackageManager().clearPackagePreferredActivities(packageName);

}

第二种方式: 通过pm进行 清理 后添加 或者是替换当前的 ,这里显示的是两个Launcher的替换

private void setDefaultHome(String packeageName ,String className,String oldPackage,String oldName) {

PackageManager pm = getPackageManager();

IntentFilter f = new IntentFilter();

f.addAction(Intent.ACTION_MAIN);

f.addCategory(Intent.CATEGORY_HOME);

f.addCategory(Intent.CATEGORY_DEFAULT);

ComponentName component = new ComponentName(packeageName,className );

ComponentName[] components = new ComponentName[] {new ComponentName(oldPackage,oldName),component};

pm.clearPackagePreferredActivities(oldPackage);// 清理配置的默认信息

pm.addPreferredActivity(f, IntentFilter.MATCH_CATEGORY_EMPTY, components, component);//添加

//pm.replacePreferredActivity(f, IntentFilter.MATCH_CATEGORY_EMPTY, components, component); //替换

}

//存在多个以上的Launcher的时候

private void setDefaultLauncher() {

// remove this activity from the package manager.

PackageManager pm = getPackageManager();

String examplePackageName = “com.jeejen.family”; //请修改为需要设置的 package name

String exampleActivityName = “com.jeejen.home.launcher.Launcher”; //请修改为需要设置的 launcher activity name

ComponentName defaultLauncher = null;

Intent intent = new Intent(Intent.ACTION_MAIN);

intent.addCategory(Intent.CATEGORY_HOME);

List resolveInfoList =

pm.queryIntentActivities(intent, 0);

if (resolveInfoList != null) {

int size = resolveInfoList.size();

for (int j = 0; j < size; ) {

final ResolveInfo r = resolveInfoList.get(j);

if (!r.activityInfo.packageName.equals(examplePackageName)) {

resolveInfoList.remove(j);

size -= 1;

} else {

j++;

}

}

ComponentName[] set = new ComponentName[size];

defaultLauncher = new ComponentName(examplePackageName, exampleActivityName);

int defaultMatch = 0;

for (int i = 0; i < size; i++) {

final ResolveInfo resolveInfo =

resolveInfoList.get(i);

set[i] = new ComponentName(resolveInfo.activityInfo.packageName, resolveInfo.activityInfo.name);

if (defaultLauncher.getClassName().equals(resolveInfo.activityInfo.name)) {

defaultMatch = resolveInfo.match;

}

}

IntentFilter filter = new IntentFilter();

filter.addAction(Intent.ACTION_MAIN);

filter.addCategory(Intent.CATEGORY_HOME);

filter.addCategory(Intent.CATEGORY_DEFAULT);

pm.clearPackagePreferredActivities(defaultLauncher.getPackageName());

pm.addPreferredActivity(filter, defaultMatch, set, defaultLauncher);

}

}

切换后启动当前的Launcher,

public void startMainLauncher(){

Intent startMain = new Intent(Intent.ACTION_MAIN);

startMain.addCategory(Intent.CATEGORY_HOME);

startMain.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);

startActivity(startMain);

}

注意,如果直接切换的后 finish() 当前Activity 会比较突兀,最后做一个延时的加载动画去

animdraw2.xml

<?xml version="1.0" encoding="utf-8"?>

<rotate xmlns:android=“http://schemas.android.com/apk/res/android”

android:drawable=“@drawable/ll_loading_outside”

android:fromDegrees=“0.0”

android:pivotX=“50.0%”

android:pivotY=“50.0%”

android:toDegrees=“360.0” />

activity_main.xml

<LinearLayout xmlns:android=“http://schemas.android.com/apk/res/android”

xmlns:tools=“http://schemas.android.com/tools”

android:layout_width=“match_parent”

android:layout_height=“match_parent”

android:orientation=“vertical”

android:gravity=“center”

android:layout_gravity=“center”

tools:context=“.MainActivity” >

<ProgressBar

android:layout_width=“wrap_content”

android:layout_height=“wrap_content”

android:layout_gravity=“center”

android:layout_marginTop=“10dip”

android:indeterminateDrawable=“@drawable/animdraw2”

android:indeterminateDuration=“1500” />

package com.rtkj.switchlauncher;

import java.util.List;

import java.util.Timer;

import java.util.TimerTask;

import android.app.Activity;

import android.content.ComponentName;

import android.content.Intent;

import android.content.IntentFilter;

import android.content.SharedPreferences;

import android.content.SharedPreferences.Editor;

import android.content.pm.PackageManager;

import android.content.pm.ResolveInfo;

import android.os.Bundle;

import android.view.KeyEvent;

import android.view.View;

import android.view.View.OnClickListener;

import android.widget.Button;

import android.widget.Toast;

public class MainActivity extends Activity{

private Editor edit;

private SharedPreferences sp;

private Button bt_lock,bt_unlock;

private static final String DEFAULT_LAUNCHER=“default_launcher”;

private static final String LAUNCHER_NAME=“com.snowfish.aios.launcher”;

private static final String LAUNCHER_CLASS_NAME=“com.snowfish.aios.launcher.MainActivity”;

private static final String OLD_LAUNCHER_NAME=“com.jeejen.family”;

private static final String OLD_LAUNCHER_CLASS_NAME=“com.jeejen.home.launcher.Launcher”;

private static final String DEFAULT_OLD_MMS =“set_default_launcer_first”;

private static final String DEFAULT_MMS=“set_default_launcer_second”;

public static final int FLAG_HOMEKEY_DISPATCHED = 0x80000000;//定义屏蔽参数

Timer timer = new Timer();

@Override

protected void onCreate(Bundle savedInstanceState) {

super.onCreate(savedInstanceState);

setContentView(R.layout.activity_main);

this.getWindow().setFlags(FLAG_HOMEKEY_DISPATCHED, FLAG_HOMEKEY_DISPATCHED);

sp=getSharedPreferences(“config_home”, 0);

edit=sp.edit();

timer.schedule(task, 1000);

}

TimerTask task = new TimerTask() {

@Override

public void run() {

runOnUiThread(new Runnable() { // UI thread

@Override

public void run() {

//Toast.makeText(getApplicationContext(), “this show”, 0).show();

switchHome();

finish();

}

});

}

};

public boolean onKeyDown(int keyCode, KeyEvent paramKeyEvent)

{

switch(keyCode){

case KeyEvent.KEYCODE_BACK:

case KeyEvent.KEYCODE_HOME:

case KeyEvent.KEYCODE_MENU:

//case KeyEvent.KEYCODE_VOLUME_DOWN:

//case KeyEvent.KEYCODE_VOLUME_UP:

//case KeyEvent.KEYCODE_VOLUME_MUTE:

return true;

default:

return false;

}

}

private void switchHome() {

if(sp.getBoolean(DEFAULT_LAUNCHER,true)){

edit.putBoolean(DEFAULT_LAUNCHER, false);

edit.commit();

setDefaultHome(OLD_LAUNCHER_NAME,OLD_LAUNCHER_CLASS_NAME,LAUNCHER_NAME,LAUNCHER_CLASS_NAME);

sendToBroad(DEFAULT_OLD_MMS);

killProgress(OLD_LAUNCHER_NAME,OLD_LAUNCHER_CLASS_NAME);

}else{

edit.putBoolean(DEFAULT_LAUNCHER, true);

edit.commit();

setDefaultHome(LAUNCHER_NAME,LAUNCHER_CLASS_NAME,OLD_LAUNCHER_NAME,OLD_LAUNCHER_CLASS_NAME);

sendToBroad(DEFAULT_MMS);

killProgress(LAUNCHER_NAME,LAUNCHER_CLASS_NAME);

}

}

public void sendToBroad(String msg){//发送广播通知系统替换默认的短信应用 其他文章有写

Intent intent=new Intent(msg);

sendBroadcast(intent);

}

public void killProgress(String packageName,String className){

自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。

深知大多数Android工程师,想要提升技能,往往是自己摸索成长或者是报班学习,但对于培训机构动则几千的学费,着实压力不小。自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!

因此收集整理了一份《2024年Android移动开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。
img
img
img
img
img
img
img

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Android开发知识点,真正体系化!

由于文件比较大,这里只是将部分目录大纲截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且后续会持续更新

如果你觉得这些内容对你有帮助,可以添加V获取:vip204888 (备注Android)
img

最后

今天关于面试的分享就到这里,还是那句话,有些东西你不仅要懂,而且要能够很好地表达出来,能够让面试官认可你的理解,例如Handler机制,这个是面试必问之题。有些晦涩的点,或许它只活在面试当中,实际工作当中你压根不会用到它,但是你要知道它是什么东西。

最后在这里小编分享一份自己收录整理上述技术体系图相关的几十套腾讯、头条、阿里、美团等公司19年的面试题,把技术点整理成了视频和PDF(实际上比预期多花了不少精力),包含知识脉络 + 诸多细节,由于篇幅有限,这里以图片的形式给大家展示一部分。

还有 高级架构技术进阶脑图、Android开发面试专题资料,高级进阶架构资料 帮助大家学习提升进阶,也节省大家在网上搜索资料的时间来学习,也可以分享给身边好友一起学习。

【Android核心高级技术PDF文档,BAT大厂面试真题解析】

【算法合集】

【延伸Android必备知识点】

【Android部分高级架构视频学习资源】

**Android精讲视频领取学习后更加是如虎添翼!**进军BATJ大厂等(备战)!现在都说互联网寒冬,其实无非就是你上错了车,且穿的少(技能),要是你上对车,自身技术能力够强,公司换掉的代价大,怎么可能会被裁掉,都是淘汰末端的业务Curd而已!现如今市场上初级程序员泛滥,这套教程针对Android开发工程师1-6年的人员、正处于瓶颈期,想要年后突破自己涨薪的,进阶Android中高级、架构师对你更是如鱼得水,赶快领取吧!

一个人可以走的很快,但一群人才能走的更远。不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎扫码加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!
img
【算法合集】*

[外链图片转存中…(img-W3jm8mc0-1712823557693)]

【延伸Android必备知识点】

[外链图片转存中…(img-pbRb0UQ1-1712823557693)]

【Android部分高级架构视频学习资源】

**Android精讲视频领取学习后更加是如虎添翼!**进军BATJ大厂等(备战)!现在都说互联网寒冬,其实无非就是你上错了车,且穿的少(技能),要是你上对车,自身技术能力够强,公司换掉的代价大,怎么可能会被裁掉,都是淘汰末端的业务Curd而已!现如今市场上初级程序员泛滥,这套教程针对Android开发工程师1-6年的人员、正处于瓶颈期,想要年后突破自己涨薪的,进阶Android中高级、架构师对你更是如鱼得水,赶快领取吧!

一个人可以走的很快,但一群人才能走的更远。不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎扫码加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!
[外链图片转存中…(img-DWYWA8ry-1712823557693)]

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值