137 lines
4.1 KiB
Dart
137 lines
4.1 KiB
Dart
import 'package:flutter/material.dart';
|
||
|
||
//Toast 显示位置控制
|
||
enum ToastPostion {
|
||
top,
|
||
center,
|
||
bottom,
|
||
}
|
||
|
||
class Toast {
|
||
// toast靠它加到屏幕上
|
||
static OverlayEntry? _overlayEntry;
|
||
// toast是否正在showing
|
||
static bool _showing = false;
|
||
// 开启一个新toast的当前时间,用于对比是否已经展示了足够时间
|
||
static DateTime? _startedTime;
|
||
// 提示内容
|
||
static String? _msg;
|
||
// toast显示时间
|
||
static int _showTime = 10;
|
||
// 背景颜色
|
||
static Color? _bgColor;
|
||
// 文本颜色
|
||
static Color? _textColor;
|
||
// 文字大小
|
||
static double? _textSize;
|
||
// 显示位置
|
||
static ToastPostion? _toastPosition;
|
||
// 左右边距
|
||
static double? _pdHorizontal;
|
||
// 上下边距
|
||
static double? _pdVertical;
|
||
static void toast(
|
||
BuildContext context, {
|
||
//显示的文本
|
||
String? msg,
|
||
//显示的时间 单位毫秒
|
||
int showTime = 1000,
|
||
//显示的背景
|
||
Color bgColor = Colors.black,
|
||
//显示的文本颜色
|
||
Color textColor = Colors.white,
|
||
//显示的文字大小
|
||
double textSize = 14.0,
|
||
//显示的位置
|
||
ToastPostion position = ToastPostion.center,
|
||
//文字水平方向的内边距
|
||
double pdHorizontal = 20.0,
|
||
//文字垂直方向的内边距
|
||
double pdVertical = 10.0,
|
||
}) async {
|
||
assert(msg != null);
|
||
_msg = msg;
|
||
_startedTime = DateTime.now();
|
||
_showTime = showTime;
|
||
_bgColor = bgColor;
|
||
_textColor = textColor;
|
||
_textSize = textSize;
|
||
_toastPosition = position;
|
||
_pdHorizontal = pdHorizontal;
|
||
_pdVertical = pdVertical;
|
||
//获取OverlayState
|
||
OverlayState overlayState = Overlay.of(context)!;
|
||
_showing = true;
|
||
if (_overlayEntry == null) {
|
||
//OverlayEntry负责构建布局
|
||
//通过OverlayEntry将构建的布局插入到整个布局的最上层
|
||
_overlayEntry = OverlayEntry(
|
||
builder: (BuildContext context) => Positioned(
|
||
//top值,可以改变这个值来改变toast在屏幕中的位置
|
||
top: buildToastPosition(context),
|
||
child: Container(
|
||
alignment: Alignment.center,
|
||
width: MediaQuery.of(context).size.width,
|
||
child: Padding(
|
||
padding: EdgeInsets.symmetric(horizontal: 40.0),
|
||
child: AnimatedOpacity(
|
||
opacity: _showing ? 1.0 : 0.0, //目标透明度
|
||
duration: _showing
|
||
? Duration(milliseconds: 100)
|
||
: Duration(milliseconds: 400),
|
||
child: _buildToastWidget(),
|
||
),
|
||
)),
|
||
));
|
||
//插入到整个布局的最上层
|
||
overlayState.insert(_overlayEntry!);
|
||
} else {
|
||
//重新绘制UI,类似setState
|
||
_overlayEntry!.markNeedsBuild();
|
||
}
|
||
// 等待时间
|
||
await Future.delayed(Duration(milliseconds: _showTime));
|
||
//2秒后 到底消失不消失
|
||
if (DateTime.now().difference(_startedTime!).inMilliseconds >= _showTime) {
|
||
_showing = false;
|
||
_overlayEntry!.markNeedsBuild();
|
||
await Future.delayed(Duration(milliseconds: 400));
|
||
_overlayEntry!.remove();
|
||
_overlayEntry = null;
|
||
}
|
||
}
|
||
|
||
//toast绘制
|
||
static _buildToastWidget() {
|
||
return Center(
|
||
child: Card(
|
||
color: _bgColor,
|
||
child: Padding(
|
||
padding: EdgeInsets.symmetric(
|
||
horizontal: _pdHorizontal!, vertical: _pdVertical!),
|
||
child: Text(
|
||
_msg!,
|
||
style: TextStyle(
|
||
fontSize: _textSize,
|
||
color: _textColor,
|
||
),
|
||
),
|
||
),
|
||
),
|
||
);
|
||
}
|
||
|
||
// 设置toast位置
|
||
static buildToastPosition(context) {
|
||
var backResult;
|
||
if (_toastPosition == ToastPostion.top) {
|
||
backResult = MediaQuery.of(context).size.height * 1 / 4;
|
||
} else if (_toastPosition == ToastPostion.center) {
|
||
backResult = MediaQuery.of(context).size.height * 2 / 5;
|
||
} else {
|
||
backResult = MediaQuery.of(context).size.height * 3 / 4;
|
||
}
|
||
return backResult;
|
||
}
|
||
}
|