之前在自定义Toast的过程中走了不少弯路,不是去折腾其子布局,就是通过别人的方法设置了却不知道是什么原理。具体可参照一下之前记录的文档。
今天看了下源码,重新梳理了一下。
目前我需要自定义Toast的使用场景:
需要一个横向内容充满屏幕的Toast(默认的Toast一般出现在屏幕的中间靠下方的位置,并且只是一小块区域,文字多了的时候会换行,实际希望是不换行,单行显示充满屏幕),连续点击button的时候,希望只保持一个Toast实例(如果不做特殊处理,每次点击button都会重新创建一个Toast,快速连续点击之后,就会出现虽然停止点击了,但是taost一直显示着,并且每个toast接替显示会产生闪烁的效果)。
先解决全屏的问题。
首先我们先来看看正常使用toast的示例:
Toast.makeText(this,"abc",Toast.LENGTH_LONG).show();跟进源码看下makeText做了什么:
public static Toast makeText(Context context, CharSequence text, @Duration int duration) {
Toast result = new Toast(context);
LayoutInflater inflate = (LayoutInflater)
context.getSystemService(Context.LAYOUT_INFLATER_SERVICE);
View v = inflate.inflate(com.android.internal.R.layout.transient_notification, null);
TextView tv = (TextView)v.findViewById(com.android.internal.R.id.message);
tv.setText(text);
result.mNextView = v;
result.mDuration = duration;
return result;
}
这个静态方法初始化了一个toast示例,然后通过layoutInflater实例化了一个默认的布局文件,并设置了默认的显示时长。
public static void showToastSmall(Activity act, String str, int length) {
// TODO Auto-generated method stub
LayoutInflater inflater = act.getLayoutInflater();
View layout = inflater.inflate(R.layout.toast_submit_small, null);
TextView text = (TextView) layout.findViewById(R.id.tv_toast_small);
text.setText(str);
if (sToast == null) {
sToast = new Toast(act.getApplicationContext());
}
sToast.setGravity(Gravity.BOTTOM, 0, -75);
sToast.setDuration(length);
sToast.setView(layout);
sToast.show();
}
其实也就是把我们自己定义的布局文件,加载进来,通过toast的setView方法,把自定义的布局view设置给toast的变量mNextView。
在这里我想要实现全屏的效果,原来的做法是直接设置自定义布局为全屏的,但是实际效果不是这样的。原因是toast的mNextView实际上是附着在一个WindowManager上的,
这个WindowManager才是真正控制toast大小的窗口。
真正的启动一个toast是show方法:
public void show() {
if (mNextView == null) {
throw new RuntimeException("setView must have been called");
}
INotificationManager service = getService();
String pkg = mContext.getOpPackageName();
TN tn = mTN;
tn.mNextView = mNextView;
try {
service.enqueueToast(pkg, tn, mDuration);
} catch (RemoteException e) {
// Empty
}
}
show方法里面拿到了一个service示例,猜想这个是用来把toast加入到服务器队列的,这个队列里面还传入了一个TN对象,看TN对象是在Toast构造里面初始化的:
public Toast(Context context) {
mContext = context;
mTN = new TN();
mTN.mY = context.getResources().getDimensionPixelSize(
com.android.internal.R.dimen.toast_y_offset);
mTN.mGravity = context.getResources().getInteger(
com.android.internal.R.integer.config_toastDefaultGravity);
}
跟进这个TN对象中:
private static class TN extends ITransientNotification.Stub {
final Runnable mShow = new Runnable() {
@Override
public void run() {
handleShow();
}
};
final Runnable mHide = new Runnable() {
@Override
public void run() {
handleHide();
// Don't do this in handleHide() because it is also invoked by handleShow()
mNextView = null;
}
};
private final WindowManager.LayoutParams mParams = new WindowManager.LayoutParams();
final Handler mHandler = new Handler();
int mGravity;
int mX, mY;
float mHorizontalMargin;
float mVerticalMargin;
View mView;
View mNextView;
WindowManager mWM;
TN() {
// XXX This should be changed to use a Dialog, with a Theme.Toast
// defined that sets up the layout params appropriately.
final WindowManager.LayoutParams params = mParams;
params.height = WindowManager.LayoutParams.WRAP_CONTENT;
params.width = WindowManager.LayoutParams.WRAP_CONTENT;
params.format = PixelFormat.TRANSLUCENT;
params.windowAnimations = com.android.internal.R.style.Animation_Toast;
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;
}
/**
* schedule handleShow into the right thread
*/
@Override
public void show() {
if (localLOGV) Log.v(TAG, "SHOW: " + this);
mHandler.post(mShow);
}
/**
* schedule handleHide into the right thread
*/
@Override
public void hide() {
if (localLOGV) Log.v(TAG, "HIDE: " + this);
mHandler.post(mHide);
}
public void handleShow() {
if (localLOGV) Log.v(TAG, "HANDLE SHOW: " + this + " mView=" + mView
+ " mNextView=" + mNextView);
if (mView != mNextView) {
// remove the old view if necessary
handleHide();
mView = mNextView;
Context context = mView.getContext().getApplicationContext();
String packageName = mView.getContext().getOpPackageName();
if (context == null) {
context = mView.getContext();
}
mWM = (WindowManager)context.getSystemService(Context.WINDOW_SERVICE);
// We can resolve the Gravity here by using the Locale for getting
// the layout direction
final Configuration config = mView.getContext().getResources().getConfiguration();
final int gravity = Gravity.getAbsoluteGravity(mGravity, config.getLayoutDirection());
mParams.gravity = gravity;
if ((gravity & Gravity.HORIZONTAL_GRAVITY_MASK) == Gravity.FILL_HORIZONTAL) {
mParams.horizontalWeight = 1.0f;
}
if ((gravity & Gravity.VERTICAL_GRAVITY_MASK) == Gravity.FILL_VERTICAL) {
mParams.verticalWeight = 1.0f;
}
mParams.x = mX;
mParams.y = mY;
mParams.verticalMargin = mVerticalMargin;
mParams.horizontalMargin = mHorizontalMargin;
mParams.packageName = packageName;
if (mView.getParent() != null) {
if (localLOGV) Log.v(TAG, "REMOVE! " + mView + " in " + this);
mWM.removeView(mView);
}
if (localLOGV) Log.v(TAG, "ADD! " + mView + " in " + this);
mWM.addView(mView, mParams);
trySendAccessibilityEvent();
}
}
private void trySendAccessibilityEvent() {
AccessibilityManager accessibilityManager =
AccessibilityManager.getInstance(mView.getContext());
if (!accessibilityManager.isEnabled()) {
return;
}
// treat toasts as notifications since they are used to
// announce a transient piece of information to the user
AccessibilityEvent event = AccessibilityEvent.obtain(
AccessibilityEvent.TYPE_NOTIFICATION_STATE_CHANGED);
event.setClassName(getClass().getName());
event.setPackageName(mView.getContext().getPackageName());
mView.dispatchPopulateAccessibilityEvent(event);
accessibilityManager.sendAccessibilityEvent(event);
}
public void handleHide() {
if (localLOGV) Log.v(TAG, "HANDLE HIDE: " + this + " mView=" + mView);
if (mView != null) {
// note: checking parent() just to make sure the view has
// been added... i have seen cases where we get here when
// the view isn't yet added, so let's try not to crash.
if (mView.getParent() != null) {
if (localLOGV) Log.v(TAG, "REMOVE! " + mView + " in " + this);
mWM.removeView(mView);
}
mView = null;
}
}
}
实际上是一个aidl接口的实现类,保证与服务进程通信的。
这个类里面有个WindowManager示例,并且有个handler负责把我们自定义的view加入到WindowMagener中或者移除。
主要看它的成员变量mGrayvity,在toast初始化的时候得到了一个默认的值:
mTN.mGravity = context.getResources().getInteger(
com.android.internal.R.integer.config_toastDefaultGravity);
}
这个默认值是这样定义的:
<integer name="config_toastDefaultGravity">0x00000051integer>即效果是居中,底部显示
在TN的handleShow方法中有这么一个判断:
mParams.gravity = gravity;
if ((gravity & Gravity.HORIZONTAL_GRAVITY_MASK) == Gravity.FILL_HORIZONTAL) {
mParams.horizontalWeight = 1.0f;
}
if ((gravity & Gravity.VERTICAL_GRAVITY_MASK) == Gravity.FILL_VERTICAL) {
mParams.verticalWeight = 1.0f;
}
这是横向或纵向的权重,权重为1表示充满这个屏幕,所以我们在定义toast的时候,使用setGrayvity设置属性:
sToast.setGravity(Gravity.BOTTOM|Gravity.FILL_HORIZONTAL, 0,300);
这样WindowManager就是横向全屏的了,里面的text就不会换行了。
public static void showToast1(Activity act,CharSequence text, int length) {
// TODO Auto-generated method stub
if (sToast == null) {
LayoutInflater inflater = act.getLayoutInflater();
View layout = inflater.inflate(R.layout.toast_submit2, null);
TextView textView = (TextView) layout.findViewById(R.id.tv_toast);
textView.setText(text);
sToast = new Toast(act.getApplicationContext());
sToast.setView(layout);
sToast.setGravity(Gravity.BOTTOM|Gravity.FILL_HORIZONTAL, 0,(int) act.getResources().getDimension(R.dimen.video_fwd_toast_postion_to_bottom));
}
sToast.setDuration(length);
sToast.show();
}
保证多次点击,只会有一个toast实例存在,show只是把这个实例重新加入到队列中,并且重新设置显示时间。
关于这个显示时间,如何自定义?并没有好的解决办法,如下所述:
1: private static final int LONG_DELAY = 3500; // 3.5 seconds
2: private static final int SHORT_DELAY = 2000; // 2 seconds
3: private void scheduleTimeoutLocked(ToastRecord r, boolean immediate)
4: {
5: Message m = Message.obtain(mHandler, MESSAGE_TIMEOUT, r);
6: long delay = immediate ? 0 : (r.duration == Toast.LENGTH_LONG ? LONG_DELAY : SHORT_DELAY);
7: mHandler.removeCallbacksAndMessages(r);
8: mHandler.sendMessageDelayed(m, delay);
9: }
这一段源码位于\sources\Android-16\com\android\server包的NotificationManagerService类中,这里代码很明确的告诉我们,如果我们show一个Toast的时候,设置的时间为Toast.LENGTH_LONG,那么就按Toast.LENGTH_LONG来显示,如果设置了其他值则全部按Toast.LENGTH_SHORT来显示。而LONG_DELAY是3.5秒,SHORT_DELAY是2秒,也就是说一个Toast最长的显示时间为3.5秒。
sToast.setDuration(length);
sToast.show();
需要显示多久就不停的计时调用多少次。似乎也只能这样了。