Toast是Android中使用频率较高的弹窗提示手段,使用起来简单、方便。常规使用方法这里不做说明,继前一篇博客《Android中Toast全屏显示》 ,其中抛砖引玉的给出一个简单的实现Toast全屏显示的方法后,发现无法控制Toast的显示时长。虽然Toast中有setDuration(int duration)接口,但是跟踪代码发现,设置的时间没起作用,只有系统默认的两个时间LENGTH_DURATION = 3500毫秒,SHORT_DURATION = 2000毫秒。也就是说,无论我们设置多长时间,最终影响Toast弹窗时间的只有Toast.LENGTH_LONG和Toast.LENGTH_SHORT两个参数。
目前解决该问题的方法主要有两个:
1、利用反射原理,通过控制Toast的show()和hide()接口来控制显示时间,可参见博客《利用反射机制控制Toast的显示时间》。不过该方法只对Android4.0以下的系统有效,通过模拟器实测,也是如此。当前系统基本都在Android4.0以上,该方法过于老旧。
2、利用WindowManager的addView()方法动态刷屏,可看见博客《Android自定义Toast,可设定显示时间》 。该方法被很多软件用来显示浮动窗口和图片的动态悬浮效果,如360手机软件和一些手游软件。在Android4.0上是一种不错的选择。当然,对于遇到系统默认把悬浮窗口功能关闭的手机,这招可能就不灵了。
通过分析Toast的显示原理和弹窗控制逻辑,本人借助Handler和Runnable机制,也成功实现了对Toast显示任意自定义时长。代码是在Toast全屏显示的基础上修改而来,贴出如下:
package com.dls.nltest; import android.content.Context; import android.os.Handler; import android.util.DisplayMetrics; import android.util.Log; import android.view.Gravity; import android.view.WindowManager; import android.widget.LinearLayout; import android.widget.TextView; import android.widget.Toast; import android.widget.LinearLayout.LayoutParams; public class GenericToast{ private static final String TAG = "GenericToast"; private static final int TOAST_TEXTSIZE = 20; /** {@link Toast#LENGTH_SHORT} default time is 3500ms */ private static final int LENGTH_SHORT_TIME = 2000; private static Context mContext = null; private static Toast mToast = null; private static TextView mTextView = null; private static int mDuration = 0; private static CharSequence mText = null; private Handler mHandler = new Handler(); private GenericToast(Context context) { mContext = context; } public static GenericToast makeText(Context context, CharSequence text, int duration){ GenericToast instance = new GenericToast(context); mContext = context; mDuration = duration; mText = text; return instance; } private static void getToast(Context context, CharSequence text){ mToast = Toast.makeText(context, null, Toast.LENGTH_LONG); mToast.setGravity(Gravity.CENTER, 0, 0); LinearLayout toastView = (LinearLayout)mToast.getView(); // Get the screen size with unit pixels. WindowManager wm = (WindowManager)context.getSystemService(Context.WINDOW_SERVICE); DisplayMetrics outMetrics = new DisplayMetrics(); wm.getDefaultDisplay().getMetrics(outMetrics); mTextView = new TextView(context); LayoutParams vlp = new LayoutParams(outMetrics.widthPixels, outMetrics.heightPixels); vlp.setMargins(0, 0, 0, 0); mTextView.setLayoutParams(vlp); mTextView.setTextSize(TOAST_TEXTSIZE); mTextView.setText(text); mTextView.setGravity(Gravity.CENTER); toastView.addView(mTextView); } /** * Before call this method, you should call {@link makeText}. * * @return Toast display duration. */ public int getDuration(){ return mDuration; } public void show(){ Log.d(TAG, "Show custom toast"); mHandler.post(showRunnable); } public void hide(){ Log.d(TAG, "Hide custom toast"); mDuration = 0; if(mToast != null){ mToast.cancel(); } } private Runnable showRunnable = new Runnable(){ @Override public void run() { if(mToast != null){ mTextView.setText(mText); }else{ getToast(mContext, mText); } if(mDuration != 0){ mToast.show(); }else{ Log.d(TAG, "Hide custom toast in runnable"); hide(); return; } if(mDuration > LENGTH_SHORT_TIME){ mHandler.postDelayed(showRunnable, LENGTH_SHORT_TIME); mDuration -= LENGTH_SHORT_TIME; }else{ mHandler.postDelayed(showRunnable, mDuration); mDuration = 0; } } }; }
Toast弹窗10s,测试代码如下:
GenericToast mGToast = GenericToast.makeText(this, "I am generic toast", 10 * 1000); mGToast.show();
如果需要终止弹窗,只要在需要的地方调用hide()即可。
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持自由互联。