Android超实用的Toast提示框优化分享
前言
相信每位Android开发者都用过Toast,都知道是弹出消息的。类似于js里面的alert,C#里面的MesageBox。当然android里面也有dialog,dialog是有焦点的,可与用户交互。而toast是没有焦点的,时间到了自动消失,不能回应用户的交互,下面就跟大家分享下Android中Toast提示框的优化方法。
先看下源码:
publicclassToast{
publicstaticfinalintLENGTH_SHORT=0;
publicstaticfinalintLENGTH_LONG=1;
/**
*构造一个空的toast。你必须在调动show()之前,线调用setView()
*@paramcontext参数,application或者activity都可以
*/
publicToast(Contextcontext){
...
//获取系统内置的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);
}
/**
*在指定的时长显示view视图
*/
publicvoidshow(){
//如果mNextView为空,即没有设置view
if(mNextView==null){
thrownewRuntimeException("setViewmusthavebeencalled");
}
//通过系统服务,获取通知管理器。看来这是使用系统通知?
INotificationManagerservice=getService();
Stringpkg=mContext.getOpPackageName();
TNtn=mTN;
tn.mNextView=mNextView;
try{
service.enqueueToast(pkg,tn,mDuration);
}catch(RemoteExceptione){
//Empty
}
}
/**
*关闭一个正在显示的toast,或者取消一个未显示的toast.
*通常你不必调用它,在适当的时长后它会自动消失的。
*/
publicvoidcancel(){
mTN.hide();
try{
getService().cancelToast(mContext.getPackageName(),mTN);
}catch(RemoteExceptione){
//Empty
}
}
/**
*设置toast显示的视图内容,不单单是黑色的界面,你可以自己决定显示什么
*@see#getView
*/
publicvoidsetView(Viewview){
mNextView=view;
}
/**
*设置时长,只能是下面这两个常量值,没什么卵用
*@see#LENGTH_SHORT2000毫秒
*@see#LENGTH_LONG3500毫秒
*/
publicvoidsetDuration(@Durationintduration){
mDuration=duration;
}
/**
*设置view的外间距,不用多说.
*
*@paramhorizontalMarginThehorizontalmargin,inpercentageofthe
*containerwidth,betweenthecontainer'sedgesandthe
*notification
*@paramverticalMarginTheverticalmargin,inpercentageofthe
*containerheight,betweenthecontainer'sedgesandthe
*notification
*/
publicvoidsetMargin(floathorizontalMargin,floatverticalMargin){
mTN.mHorizontalMargin=horizontalMargin;
mTN.mVerticalMargin=verticalMargin;
}
/**
*设置notification在屏幕中的方位,大家都知道.上中下左中右什么的都有
*@seeandroid.view.Gravity
*@see#getGravity
*/
publicvoidsetGravity(intgravity,intxOffset,intyOffset){
mTN.mGravity=gravity;
mTN.mX=xOffset;
mTN.mY=yOffset;
}
/**
*构造一个只包含一个TextView的标准toast对象
*
*@paramcontext通常是application或者activity对象
*@paramtext用于显示的文本,可以是formattedtext.
*@paramduration显示时长.LENGTH_SHORT或LENGTH_LONG
*
*/
publicstaticToastmakeText(Contextcontext,CharSequencetext,@Durationintduration){
Toastresult=newToast(context);
LayoutInflaterinflate=(LayoutInflater)
context.getSystemService(Context.LAYOUT_INFLATER_SERVICE);
//包含了一个默认的TextView,这个textview的布局位置在
com.android.internal.R.layout.transient_notification,可以去查看下内容
Viewv=inflate.inflate(com.android.internal.R.layout.transient_notification,null);
TextViewtv=(TextView)v.findViewById(com.android.internal.R.id.message);
tv.setText(text);
result.mNextView=v;
result.mDuration=duration;
returnresult;
}
/**
*更新通过makeText()方法创建出来的toast对象显示的文本内容
*@params待显示的新文本内容.
*/
publicvoidsetText(CharSequences){
if(mNextView==null){
thrownewRuntimeException("ThisToastwasnotcreatedwithToast.makeText()");
}
/**
*看来com.android.internal.R.layout.transient_notification布局里面的唯一的
*TextView标签的id是R.id.message。拿到这个textview,设置新文本内容
*/
TextViewtv=(TextView)mNextView.findViewById(com.android.internal.R.id.message);
if(tv==null){
thrownewRuntimeException("ThisToastwasnotcreatedwithToast.makeText()");
}
tv.setText(s);
}
staticprivateINotificationManagergetService(){
if(sService!=null){
returnsService;
}
//获取远程的通知服务
sService=INotificationManager.Stub.asInterface(ServiceManager.getService("notification"));
returnsService;
}
//TN是一个瞬态通知的子类,里面包含显示和隐藏两个任务对象
privatestaticclassTNextendsITransientNotification.Stub{
finalRunnablemShow=newRunnable(){
@Override
publicvoidrun(){
handleShow();
}
};
finalRunnablemHide=newRunnable(){
@Override
publicvoidrun(){
handleHide();
//Don'tdothisinhandleHide()becauseitisalsoinvokedbyhandleShow()
mNextView=null;
}
};
//出现Handler了哦
finalHandlermHandler=newHandler();
/**
*调度handleShow任务到执行线程中
*/
@Override
publicvoidshow(){
if(localLOGV)Log.v(TAG,"SHOW:"+this);
//handler发送异步任务了
mHandler.post(mShow);
}
/**
*同上
*/
@Override
publicvoidhide(){
if(localLOGV)Log.v(TAG,"HIDE:"+this);
mHandler.post(mHide);
}
//...
}
}
通过上面的源码解读,了解到有远程通知,handler异步任务等信息,不多说,自己看。
重点是toast的用法:
1、直接调用makeText静态方法即可,返回的是Toast对象,最后别忘了调用show方法显示:
Toast.makeText(context,text,duration).show();
或
Toasttoast=Toast.makeText(context,text,duration); prename="code"class="html">toast.setView(view); toast.setText(s); toast.setGravity(gravity,xOffset,yOffset); toast.setDuration(duration); toast.show();
2、还可以使用new的方式创建,别忘了setView()方法:
Toasttoast=newToast(); toast.setView(view); toast.setText(s); toast.setGravity(gravity,xOffset,yOffset); toast.setDuration(duration); toast.show();
以上这些都不值得一提,很简单。
在开发过程中,有这样的需求:在项目总,我们偷懒,想连串toast出多个变量的值或者其他任务,可在操作手机时直观可见。问题来了,弹出是无论我们的操作有多快,这些toast内容都是一个跟着一个显示,没办法快进。哪怕我们玩完了,退出了app,它还在弹。怎么办?有没有办法让toast的内容与我们的操作同步,快速反应?
publicclassT{
privatestaticToasttoast;
publicstaticvoidshow(Contextcontext,Stringmsg){
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能有所帮助,如果有大家有疑问可以留言交流。