前言
相信每位Android开发者都用过
Toast
,都知道是弹出消息的。类似于js里面的alert
,C#里面的MesageBox
。当然android里面也有dialog
,dialog
是有焦点的,可与用户交互。而toast
是没有焦点的,时间到了自动消失,不能回应用户的交互,下面就跟大家分享下Android中Toast
提示框的优化方法。
先看下源码:
public class Toast {
public static final int LENGTH_SHORT = 0;
public static final int LENGTH_LONG = 1;
public Toast(Context context) {
...
//获取系统内置的toast_y_offset常量值
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);
}
public void show() {
//如果mNextView为空,即没有设置view
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
}
}
public void cancel() {
mTN.hide();
try {
getService().cancelToast(mContext.getPackageName(), mTN);
} catch (RemoteException e) {
// Empty
}
}
public void setView(View view) {
mNextView = view;
}
public void setDuration(@Duration int duration) {
mDuration = duration;
}
public void setMargin(float horizontalMargin, float verticalMargin) {
mTN.mHorizontalMargin = horizontalMargin;
mTN.mVerticalMargin = verticalMargin;
}
public void setGravity(int gravity, int xOffset, int yOffset) {
mTN.mGravity = gravity;
mTN.mX = xOffset;
mTN.mY = yOffset;
}
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);
//包含了一个默认的TextView,这个textview的布局位置在
com.android.internal.R.layout.transient_notification,可以去查看下内容
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;
}
public void setText(CharSequence s) {
if (mNextView == null) {
throw new RuntimeException("This Toast was not created with Toast.makeText()");
}
TextView tv = (TextView) mNextView.findViewById(com.android.internal.R.id.message);
if (tv == null) {
throw new RuntimeException("This Toast was not created with Toast.makeText()");
}
tv.setText(s);
}
static private INotificationManager getService() {
if (sService != null) {
return sService;
}
//获取远程的通知服务
sService = INotificationManager.Stub.asInterface(ServiceManager.getService("notification"));
return sService;
}
//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;
}
};
//出现Handler了哦
final Handler mHandler = new Handler();
@Override
public void show() {
if (localLOGV) Log.v(TAG, "SHOW: " + this);
//handler发送异步任务了
mHandler.post(mShow);
}
@Override
public void hide() {
if (localLOGV) Log.v(TAG, "HIDE: " + this);
mHandler.post(mHide);
}
//...
}
}
通过上面的源码解读,了解到有远程通知,
handler
异步任务等信息,不多说,自己看。重点是toast的用法:
1、直接调用makeText静态方法即可,返回的是Toast对象,最后别忘了调用show方法显示:
Toast.makeText(context, text, duration).show();
或
Toast toast = Toast.makeText(context, text, duration);
pre name="code" class="html"> toast.setView(view);
toast.setText(s);
toast.setGravity(gravity, xOffset, yOffset);
toast.setDuration(duration);
toast.show();
2、还可以使用new的方式创建,别忘了setView()方法:
Toast toast = new Toast();
toast.setView(view);
toast.setText(s);
toast.setGravity(gravity, xOffset, yOffset);
toast.setDuration(duration);
toast.show();
以上这些都不值得一提,很简单。
在开发过程中,有这样的需求:在项目总,我们偷懒,想连串
toast
出多个变量的值或者其他任务,可在操作手机时直观可见。问题来了,弹出是无论我们的操作有多快,这些toast
内容都是一个跟着一个显示,没办法快进。哪怕我们玩完了,退出了app,它还在弹。怎么办?有没有办法让toast
的内容与我们的操作同步,快速反应?
public class T {
private static Toast toast;
public static void show(Context context, String msg) {
if (toast == null) {
toast = Toast.makeText(context, msg, Toast.LENGTH_SHORT);
} else {
toast.setText(msg);
}
toast.show();
}
}
单例模式,每次创建
toast
都调用这个类的show
方法,Toast
的生命周期从show
开始,到自己消失或者cancel
为止。如果正在显示,则修改显示的内容,你持有这个对象的引用,当然可以修改显示的内容了。若每次你都makeText
或者new
一个toast
对象,即每次通过handler
发送异步任务,调用远程通知服务显示通知,当然是排队等待显示了。
结束语
以上就是Android中Toast提示框优化的全部内容,希望对大家开发Android能有所帮助,如果有大家有疑问可以留言交流。
您可能感兴趣的文章:Android实现简单的popupwindow提示框Android中仿IOS提示框的实现方法Android使用Toast显示消息提示框IOS 仿Android吐司提示框的实例(分享)Android 自定义一套 Dialog通用提示框 (代码库)Android仿IOS自定义AlertDialog提示框Android仿QQ、微信聊天界面长按提示框效果Android仿百度谷歌搜索自动提示框AutoCompleteTextView简单应用示例Android实现Toast提示框图文并存的方法Android编程之自定义AlertDialog(退出提示框)用法实例flutter Toast实现消息提示框