Android系统里面有个东西很好用,也很常用,那就是Toast,但是长期使用也会发现,Toast有他的不足之处:形式单一,只有文字,风格不变等等,那么要如何自定义一个Toast呢,我们可以先从分析Android定义Toast的代码着手:
Toast的makeText方法:
这里实际上Android所做的工作是将Toast显示的文本和持续时间设置了一下,然后返回了Toast对象,用以执行show()方法。这里核心的地方是要弄明白
这一句Android做了一些什么工作,下面我们继续看:
这里实际上是new 了一个TN()对象,然后对TN对象设置了一些关于位置和偏移的参数。那么那个TN是何方神圣呢?我们可以继续往下看:
我们可以发现,TN的构造函数里面执行了一个handleShow的方法
我们可以发现handleshow这个方法执行了一大堆的语句,但是实际上我们需要重点关注的只有一句,也就是上面选中的一句:mWM.addView(mView,mParams);
实际上这个mWM是一个WindowManager,mView是要显示的组件的view,而mParams应该是参数的设置一类,但是具体是何方神圣呢?我们可以在Toast类中找到一个成员变量:
我们发现mParams实际上是WindowManager的一个布局参数。但是如果单从handleShow方法来看mParams的设置感觉还是太乱,我们可以在Toast中找到下面这一段:
在这里我们可以比较清楚看到mParams的一部分主要参数设置,其中包含了长宽都是wrap_content,然后是半透明的(TransLucent),然后还设置了一个动画效果(Animation_Toast),和一个标题(setTitle("Toast"))最后设置了Toast一些标识比如让屏幕亮起,不可触摸,不可获取焦点等。
到这里,我们基本上清楚了Toast是如何显示出来的,核心的在于使用一个WindowManager的addView方法,将要显示的view和参数mParams传进来,但是在此之前还需要设置mParams的一些参数;
那么这里是显示,Toast是如何让显示消失的呢?我们可以在handleShow方法下面找到一个handHide的方法:
同样核心语句也是选中的一句,mWM.removeView(mView),这样就让Toast消失掉了
那么有了上面的一些分析,我们不难写出我们自己的Toast类,里面提供一些静态方法,让我们来自己定义Toast的显示风格和显示内容,也可以控制Toast的显示开始和结束的时机。
下面的自定义Toast类,提供了几种形式的show方法,可以像传统Toast一样,传入文本来显示一个testview,也可以完全自己定义传入view对象和mParams参数,这样可以定义出了一个完全自己定义的Toast
package com.alexchen.mobilesafeexercise.ui;
import android.content.Context;
import android.graphics.PixelFormat;
import android.view.View;
import android.view.WindowManager;
import android.widget.TextView;
import android.widget.Toast;
/**
* 自定义吐司
*
* @author Alex
*
*/
public class MyToast {
/**
* 窗体管理者
*/
private static WindowManager wm;
private static WindowManager.LayoutParams mParams = new WindowManager.LayoutParams();
private static View mView;
private static TextView tv;
/**
* 显示自定义吐司
*
* @param info
* @param context
*/
public static void show(String message, Context context) {
wm = (WindowManager) context.getSystemService(Context.WINDOW_SERVICE);
tv = new TextView(context);
tv.setText(message);
tv.setTextSize(20);
// 原来TN所做的工作
WindowManager.LayoutParams params = mParams;
params.height = WindowManager.LayoutParams.WRAP_CONTENT;
params.width = WindowManager.LayoutParams.WRAP_CONTENT;
params.format = PixelFormat.TRANSLUCENT;
params.type = WindowManager.LayoutParams.TYPE_TOAST;
params.setTitle("Toast");
params.flags = WindowManager.LayoutParams.FLAG_KEEP_SCREEN_ON
| WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE
| WindowManager.LayoutParams.FLAG_NOT_TOUCHABLE;
wm.addView(tv, params);
}
/**
* 自定义文本吐司的显示方法,这个方法提供了更多自定义的内容,比如textview可以应用一个传入的style的id
*
* @param message
* @param context
* @param textViewResid
* @param params
*/
public static void show(String message, Context context, int textViewResid,
WindowManager.LayoutParams params) {
wm = (WindowManager) context.getSystemService(Context.WINDOW_SERVICE);
tv = new TextView(context);
// 多加了一个风格
tv.setTextAppearance(context, textViewResid);
// 原来TN所做的工作
// WindowManager.LayoutParams params = mParams;
// params.height = WindowManager.LayoutParams.WRAP_CONTENT;
// params.width = WindowManager.LayoutParams.WRAP_CONTENT;
// params.format = PixelFormat.TRANSLUCENT;
// params.type = WindowManager.LayoutParams.TYPE_TOAST;
// params.setTitle("Toast");
// params.flags = WindowManager.LayoutParams.FLAG_KEEP_SCREEN_ON
// | WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE
// | WindowManager.LayoutParams.FLAG_NOT_TOUCHABLE;
wm.addView(tv, params);
}
/**
* 更多自定义的形式,可以直接传入任一个自己定义好的view,自己设置wm的参数
*
* @param view
* @param context
* @param params
* WindowManager.LayoutParams类型的参数, WindowManager.LayoutParams
* mParams = new WindowManager.LayoutParams(); params.height =
* WindowManager.LayoutParams.WRAP_CONTENT; params.width =
* WindowManager.LayoutParams.WRAP_CONTENT; params.format =
* PixelFormat.TRANSLUCENT; params.type =
* WindowManager.LayoutParams.TYPE_TOAST;
* params.setTitle("Toast"); params.flags =
* WindowManager.LayoutParams.FLAG_KEEP_SCREEN_ON |
* WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE |
* WindowManager.LayoutParams.FLAG_NOT_TOUCHABLE;
*/
public static void show(View view, Context context,
WindowManager.LayoutParams params) {
wm = (WindowManager) context.getSystemService(Context.WINDOW_SERVICE);
MyToast.mView = view;
wm.addView(mView, params);
}
/**
* 隐藏自定义吐司 这里一定要记得判空一下,因为平时没有打电话时,这两个量应该都是空的
*/
public static void hide() {
if (wm != null) {
if (tv != null) {
wm.removeView(tv);
tv = null;
}
if (mView != null) {
wm.removeView(mView);
mView = null;
}
if (tv == null && mView == null) {
wm = null;
}
}
}
}