🔥 應用場景
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提供的方法:

有這幾個方法,咱們實踐一下,保險起見看看原始碼
💥 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

Android 5.1

💥 工具類
如果覺得設定麻煩可以看看下面這邊文章,然后整合一套適合自己的,
https://www.jianshu.com/p/f4ba05d7bbda
轉載請註明出處,本文鏈接:https://www.uj5u.com/yidong/356141.html
標籤:其他
上一篇:Swift 函式與閉包
