Android 快别用Toast了,来试试Snackbar

简介: 🔥 应用场景🔥 源码💥 Toast.setGravity()💥 Toast.isSystemRenderedTextToast()🔥 Toast 提供的方法💥 Toast.setView() 源码🔥 Snackbar💥 代码实现💥 效果💥 工具类

🔥 应用场景


       Toast提示默认显示在界面底部,使用Toast.setGravity()将提示显示在中间,如下:


        Toast toast = Toast.makeText(this, str, Toast.LENGTH_SHORT);
        toast.setGravity(Gravity.CENTER, 0, 0);
        toast.show();


   运行在在Android 12上无法显示,查看Logcat提示如下:


Toast: setGravity() shouldn't be called on text toasts, the values won't be used


意思就是:你不能使用toast调用setGravity,调用无效。哎呀,看给牛气的,咱看看源码找找原因


🔥 源码


💥 Toast.setGravity()


    /**
     * 设置Toast出现在屏幕上的位置。
     *
     * 警告:从 Android R 开始,对于面向 API 级别 R 或更高级别的应用程序,此方法在文本 toast 上调用时无效。
     */
    public void setGravity(int gravity, int xOffset, int yOffset) {
        if (isSystemRenderedTextToast()) {
            Log.e(TAG, "setGravity() shouldn't be called on text toasts, the values won't be used");
        }
        mTN.mGravity = gravity;
        mTN.mX = xOffset;
        mTN.mY = yOffset;
    }


妥了,人家就告诉你了 版本>=Android R(30),调用该方法无效。无效就无效呗,还不给显示了,过分。


Logcat的提示居然是在这里提示的,来都来了,咱们看看isSystemRenderedTextToast()方法。


💥 Toast.isSystemRenderedTextToast()


    /**
     *Text Toast 将由 SystemUI 呈现,而不是在应用程序内呈现,因此应用程序无法绕过后台自定义 Toast 限制。
     */
    @ChangeId
    @EnabledAfter(targetSdkVersion = Build.VERSION_CODES.Q)
    private static final long CHANGE_TEXT_TOASTS_IN_THE_SYSTEM = 147798919L;
    private boolean isSystemRenderedTextToast() {
        return Compatibility.isChangeEnabled(CHANGE_TEXT_TOASTS_IN_THE_SYSTEM) && mNextView == null;
    }


 重点了。Text Toast 将由 SystemUI 呈现,而不是在应用程序内呈现。


       清晰明了,可以这样玩,但是你级别不够,不给你玩。


       事情整明白了,再想想解决解决方案。他说了Text Toast 将由 SystemUI 呈现,那我不用 Text 不就行了。


🔥 Toast 提供的方法


先看看Tast提供的方法:

微信图片_20220524124956.png


       有这几个方法。咱们实践一下。保险起见看看源码


💥 Toast.setView() 源码


    /**
     * 设置显示的View
     * @deprecated 自定义 Toast 视图已弃用。 应用程序可以使用 makeText 方法创建标准文本 toast,
     * 或使用 Snackbar
     */
    @Deprecated
    public void setView(View view) {
        mNextView = view;
    }


这个更狠,直接弃用。


  • 要么老老实实的用默认的Toast。


  • 要么使用 Snackbar。


🔥 Snackbar


Snackbar 就是一个类似Toast的快速弹出消息提示的控件。


与Toast相比:


  • 一次只能显示一个


  • 与用户交互


  • 在右侧设置按钮来添加事件,根据 Material Design 的设计原则,只显示 1 个按钮 (添加多个,以最后的为准)


  • 提供Snackbar显示和关闭的监听事件


  • BaseTransientBottomBar.addCallback(BaseCallback)


💥 代码实现


    showMessage(findViewById(android.R.id.content), str, Snackbar.LENGTH_INDEFINITE);
    public static void showMessage(View view, String str, int length) {
        Snackbar snackbar = Snackbar.make(view, str, length);
        View snackbarView = snackbar.getView();
        //设置布局居中
        FrameLayout.LayoutParams params = new FrameLayout.LayoutParams(snackbarView.getLayoutParams().width, snackbarView.getLayoutParams().height);
        params.gravity = Gravity.CENTER;
        snackbarView.setLayoutParams(params);
        //文字居中
        TextView message = (TextView) snackbarView.findViewById(R.id.snackbar_text);
        //View.setTextAlignment需要SDK>=17
        message.setTextAlignment(View.TEXT_ALIGNMENT_GRAVITY);
        message.setGravity(Gravity.CENTER);
        message.setMaxLines(1);
        snackbar.addCallback(new BaseTransientBottomBar.BaseCallback<Snackbar>() {
            @Override
            public void onDismissed(Snackbar transientBottomBar, int event) {
                super.onDismissed(transientBottomBar, event);
                //Snackbar关闭
            }
            @Override
            public void onShown(Snackbar transientBottomBar) {
                super.onShown(transientBottomBar);
                //Snackbar显示
            }
        });
        snackbar.setAction("取消", new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                //显示一个默认的Snackbar。
                Snackbar.make(view, "我先走", BaseTransientBottomBar.LENGTH_LONG).show();
                snackbar.dismiss();
            }
        });
        snackbar.show();
    }


Snackbar.make的三个参数:


  • View:从View中找出当前窗口最外层视图,然后在其底部显示。


  • 第二个参数(text):
  • CharSequence


  • StringRes


  • duration(显示时长)
  • Snackbar.LENGTH_INDEFINITE 从 show()开始显示,直到它被关闭或显示另一个 Snackbar****。

  • Snackbar.LENGTH_SHORT 短时间


  • Snackbar.LENGTH_LONG 长时间


  • 自定义持续时间 以毫秒为单位


💥 效果


Android 12

微信图片_20220524125305.gif


Android 5.1


微信图片_20220524125332.gif


💥 工具类


如果觉得设置麻烦可以看看下面这边文章,然后整合一套适合自己的。


https://www.jianshu.com/p/f4ba05d7bbda


相关文章
|
5月前
|
XML Java API
30. 【Android教程】吐司提示:Toast 的使用方法
30. 【Android教程】吐司提示:Toast 的使用方法
228 2
|
Android开发
android之Toast使用
android之Toast使用
101 0
|
Android开发 数据安全/隐私保护 开发者
ApeForms | C#WinForm弹出简易的消息提示框 (仿Android Toast消息提示)
在使用手机的时候经常会见到屏幕的中下方会弹出消息提示框,它就是Toast。 ApeForms中也实现了非常简洁易用Toast,与Android的Toast不同的是,ApeForms允许开发者设置不同的弹出模式。此外还针对PC端有鼠标的情况进行了改进,当鼠标悬停于消息弹出框之上时弹出框不会消失。
427 0
ApeForms | C#WinForm弹出简易的消息提示框 (仿Android Toast消息提示)
|
Android开发
android中的提示信息显示方法(toast应用)
android中的提示信息显示方法(toast应用)
261 1
|
API Android开发
干货|APP自动化Android特殊控件Toast识别
干货|APP自动化Android特殊控件Toast识别
|
Android开发
从Toast显示原理初窥Android窗口管理
从Toast显示原理初窥Android窗口管理
208 0
从Toast显示原理初窥Android窗口管理
|
架构师 API Android开发
干货|APP自动化Android特殊控件Toast识别
Toast 是 Android 系统中的一种消息框类型,它属于一种轻量级的消息提示,常常以小弹框的形式出现,一般出现 1 到 2 秒会自动消失,可以出现在屏幕上中下任意位置。它不同于 Dialog,它没有焦点。Toast 的设计思想是尽可能的不引人注意,同时还向用户显示信息希望他们看到。 测试 APP 下载地址: 首先将上面地址的 apk 包下载到本地,并安装到模拟器中;在模拟器中打开 API
|
架构师 API Android开发
干货|APP自动化Android特殊控件Toast识别
Toast 是 Android 系统中的一种消息框类型,它属于一种轻量级的消息提示,常常以小弹框的形式出现,一般出现 1 到 2 秒会自动消失,可以出现在屏幕上中下任意位置。它不同于 Dialog,它没有焦点。Toast 的设计思想是尽可能的不引人注意,同时还向用户显示信息希望他们看到。 测试 APP 下载地址: 首先将上面地址的 apk 包下载到本地,并安装到模拟器中;在模拟器中打开 API
|
Android开发
Android之Toast的高级使用
Android之Toast的高级使用
234 0
|
XML Android开发 数据格式
安卓 topic-通知 Toast
http://developer.android.youdaxue.com/guide/topics/ui/notifiers/toasts.html#Positioning 创建自定义Toast View To create a custom layout, define a View layout, in XML or in your application code, and pass the root View object to the setView(View) method. For example, you can create the layout for the toast
135 0