花了半天时间研究下了自定义锁屏,发现其实实现并不是很神秘。不过有些地方还是值得注意。

首先说流程,锁屏界面一般是在关闭屏幕时启用,打开屏幕时展现在我们面前,所以我们知道了锁屏的时机,恰好屏幕开关会发出相应的广播,所以我们也可以像系统一样捕获到屏幕开关的事件。

广播对应的两个action

 

android.intent.action.SCREEN_ON
 android.intent.action.SCREEN_OFF
  
  
 有趣的是我在我的me860上还看到了两个广播
  
 引用
  
 android.intent.action.batteryprofile.SCR_OFF
 android.intent.action.batteryprofile.SCR_ON
  
  
 这可能是moto自己定义的广播,看大家需要了。
 如果单单只截获到广播,锁屏的界面怎么替换系统锁屏呢?其实我们这里并没有替换掉系统锁屏,只是我们自己的锁屏开启了系统锁屏,从而达到了一个替换的效果。
 这里还是要注意点,上述action不能在AndroidManifest.xml中注册,不能触发。所以这里我采用的方法是放在一个service中动态注册,截获广播正常。
 这里还顺带提下service防杀的功能,因为service的进程如果不存在了,那么屏幕开关的广播是无论如何也接收不到的。
 如果是系统回收掉的service,系统会在过段时间,资源充足的情况下再启动起来,不过我们可不想自己的service坐冷板凳,所以对付系统关闭service我们采取如下方法
 在service生命周期中
 Java代码 
 Intent startIntent=null;  
 @Override 
     public void onStart(Intent intent, int startId) {  
         startIntent=intent;  
         IntentFilter filter=new IntentFilter();  
         filter.addAction(Intent.ACTION_SCREEN_ON);  
         filter.addAction(Intent.ACTION_SCREEN_OFF);  
         receiver=new RelativeBroadcastReceiver();  
         registerReceiver(receiver, filter);  
         System.out.println("service onStart and action is "+intent.getAction());  
         System.out.println("service onStart and startId is "+startId);  
     };  
       
     @Override 
     public void onDestroy() {  
         // TODO Auto-generated method stub  
         System.out.println("service onDestroy");  
         unregisterReceiver(receiver);  
         if(startIntent!=null){  
             System.out.println("serviceIntent not null");  
             startService(startIntent);  
         } 
  
 Intent startIntent=null;
 @Override
        public void onStart(Intent intent, int startId) {
               startIntent=intent;
               IntentFilter filter=new IntentFilter();
               filter.addAction(Intent.ACTION_SCREEN_ON);
               filter.addAction(Intent.ACTION_SCREEN_OFF);
               receiver=new RelativeBroadcastReceiver();
               registerReceiver(receiver, filter);
               System.out.println("service onStart and action is "+intent.getAction());
               System.out.println("service onStart and startId is "+startId);
        };
       
        @Override
        public void onDestroy() {
               // TODO Auto-generated method stub
               System.out.println("service onDestroy");
               unregisterReceiver(receiver);
               if(startIntent!=null){
                      System.out.println("serviceIntent not null");
                      startService(startIntent);
               }
  
  
 我们保留了开启service的intent,当进入ondestroy周期中时再启动一次自己,系统看到你这劲头也只好答应你常驻内存了。
  
 不过如果是一些内存管理软件杀掉了程序进程,以上方法就没用了,我们还是可以通过截获一些关键广播来启动自己的service,类似91助手等就是这样。可以监听wifi连接,电池电量发生变化等广播来启动自己的service。
  
 保证自己的service常驻,这时候就该我们自己的广播接收器发挥作用了。
 Java代码 
 @Override 
     public void onReceive(Context context, Intent intent) {  
         // TODO Auto-generated method stub  
         String action=intent.getAction();  
         System.out.println("action is "+action);  
         Intent lockIntent=new Intent(context,MyLockScreen.class);   
         lockIntent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);  
         context.startActivity(lockIntent);  
 } 
  
 @Override
        public void onReceive(Context context, Intent intent) {
               // TODO Auto-generated method stub
               String action=intent.getAction();
               System.out.println("action is "+action);
               Intent lockIntent=new Intent(context,MyLockScreen.class);
               lockIntent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
               context.startActivity(lockIntent);
 }
  
 需要加上lockIntent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);没有加的时候系统会报错,但是加上以后也有问题,这会导致多次退出才能退出自定义的锁屏界面。
 其实可以自定义一个stack来管理这些activity,有其他办法的同学请提示我改正。
  
 启动了一个activity以后我们发现还是原来的锁屏界面,这也是前面提到的,我们的锁屏需要打开系统锁屏。
 打开系统锁屏:
 Java代码 
 super.onCreate(savedInstanceState);   
         getWindow().addFlags(WindowManager.LayoutParams.FLAG_SHOW_WHEN_LOCKED);  
         getWindow().addFlags(WindowManager.LayoutParams.FLAG_DISMISS_KEYGUARD);  
         setContentView(R.layout.main); 
  
 super.onCreate(savedInstanceState);
               getWindow().addFlags(WindowManager.LayoutParams.FLAG_SHOW_WHEN_LOCKED);
               getWindow().addFlags(WindowManager.LayoutParams.FLAG_DISMISS_KEYGUARD);
               setContentView(R.layout.main);

添加了这两个flag以后,系统锁屏就被替换成了我们自己的锁屏界面。

锁屏替换功能也就完成了。其实还可以修改系统锁屏,不过设计到框架层的修改,推广也很麻烦,所以掠过不说了

 

作者wiseideal