我们都知道Toast的几种用法,像什么居中啊,添加图片显示啊等等。不过这里我要说的是能够全屏显示和永远不会消失的另类Toast。全屏显示和之前大家了解的居中有点类似,只要修改setGravity(int gravity, int xOffset,int yOffset)中的第一个参数值为Gravity.FILL就ok了。麻烦的是要让Toast永远不消失,这里我们要用到反射机制。
我们要了解一件事,那就是我们在弹出Toast的时候,必须加上.show()。顾名思义,大家可能都会以为,这里的.show()是去显示这个Toast。不过遗憾的是,事实并非如此。我们可以看看如下show()的源代码:
public void show() { if (mNextView == null) { throw new RuntimeException("setView must have been called"); } INotificationManager service = getService(); String pkg = mContext.getPackageName(); TN tn = mTN; try { // 将当前Toast加入到Toast队列 service.enqueueToast(pkg, tn, mDuration); } catch (RemoteException e) { // Empty } }正如service.enqueueToast(pkg, tn, mDuration);这一句所说,我们只是通过.show()来把当前Toast加入到Toast队列中,再由系统根据Toast队列来显示Toast信息提示框,并不是去显示当前Toast。因为我们不能去直接控制系统中要做的事,所以我们不能再去调用show()方法了。如果不调用show()方法,我们知道,Toast是不会显示出来的。所以,这里我们就要去自己写一个方法让我们当前的Toast显示出来。我们是用反射来解决这个问题的。关键代码如下:
private Object mTN; private Method mShow; private Method mHide; private Field mViewFeild; private View mLayout; private HomeKeyEventBroadCastReceiver receiver; // 监听HOME键 /** * 开启永不消失的Toast */ private void showForeverToast() { mLayout = getLayoutInflater().inflate(R.layout.newtoast, null); Toast toast = new Toast(getApplicationContext()); toast.setGravity(Gravity.FILL, 0, 0); toast.setDuration(0); Field field = null; try { field = toast.getClass().getDeclaredField("mTN"); field.setAccessible(true); mTN = field.get(toast); mShow = mTN.getClass().getDeclaredMethod("show"); mHide = mTN.getClass().getDeclaredMethod("hide"); mViewFeild = mTN.getClass().getDeclaredField("mNextView"); mViewFeild.setAccessible(true); } catch (NoSuchFieldException e) { e.printStackTrace(); } catch (IllegalAccessException e) { e.printStackTrace(); } catch (IllegalArgumentException e) { e.printStackTrace(); } catch (NoSuchMethodException e1) { e1.printStackTrace(); } show(mLayout); // 显示Toast }显示Toast的代码如下:
private void show(View layout) { try { mViewFeild.set(mTN, layout); mShow.invoke(mTN, new Object[] {}); } catch (IllegalAccessException e) { e.printStackTrace(); } catch (IllegalArgumentException e) { e.printStackTrace(); } catch (InvocationTargetException e) { e.printStackTrace(); } }隐藏Toast的代码如下:
private void hide(View layout) { try { mHide.invoke(mTN, new Object[] {}); } catch (IllegalAccessException e) { e.printStackTrace(); } catch (IllegalArgumentException e) { e.printStackTrace(); } catch (InvocationTargetException e) { e.printStackTrace(); } }通过以上的代码就可以让我们的Toast全屏显示和永不消失了。
下面我就来说说关于拦截Home键的方法。之前我也是找了很多关于拦截和屏蔽Home键的方法,自己都试过,根本行不通(当然也可能是我自己的问题,如果大家有什么好的方法,可以给我留言)。所以,这里我就取了个巧(其实是一个假的拦截和屏蔽),我是采用了当我们点击Home键的时候,退回桌面,同时程序转入后台,一定时间后程序再从后台转入前台。关键代码正下:
class HomeKeyEventBroadCastReceiver extends BroadcastReceiver { static final String SYSTEM_REASON = "reason"; static final String SYSTEM_HOME_KEY = "homekey"; // home key static final String SYSTEM_RECENT_APPS = "recentapps"; // long home key @Override public void onReceive(Context context, Intent intent) { String action = intent.getAction(); if (action.equals(Intent.ACTION_CLOSE_SYSTEM_DIALOGS)) { String reason = intent.getStringExtra(SYSTEM_REASON); if (reason != null) { if (reason.equals(SYSTEM_HOME_KEY)) { // 从桌面启动app(利用RestartService过渡) Intent newIntent = new Intent(); newIntent.setClass(context, RestartService.class); context.startService(newIntent); showToastByHome(); } else if (reason.equals(SYSTEM_RECENT_APPS)) { hideToastByHome(); } } } } }点击Home键后,弹出全屏的Toast.5秒后又自动消失:
private void showToastByHome() { showForeverToast(); // 显示全屏的Toast if (mThread == null || !mThread.isAlive() || mThread.isInterrupted()) { mThread = new Thread() { @Override public void run() { try { sleep(5000); hide(mLayout); } catch (InterruptedException e) { e.printStackTrace(); } } }; mThread.start(); } }看上面的第一段代码,可以很容易地发现我是在退回桌面的同时,跳转到一个RestartService,这个Service会在5秒后再进行一次跳转(从Service跳转程序的前台Activity),这样就实现了假的拦截Home了,如果亲们对这个拦截Home键的功能比较急,要求又不是很苛刻的话,可以先用这样的方法顶一下。而Service中的代码也很简单,几句话,如下:
public void onStart(Intent intent, int startId) { Intent newIntent = new Intent(getApplicationContext(), MainActivity.class); newIntent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK); startActivity(newIntent); super.onStart(intent, startId); }这里是本篇博客的工程连接:http://download.csdn.net/detail/u013761665/8015397
本博客参考博文:
Android开发技巧:永不关闭的Toast信息框
Android开发技巧:永不关闭的Toast信息框(修正版)android4.x