Flutter給控件實現(xiàn)鉆石般的微光特效
效果圖

使用方法
Shimmer(
baseColor: const Color(0x08ffffff), // 背景顏色
highlightColor: Colors.white, // 高光的顏色
loop: 2, // 閃爍循環(huán)次數(shù),不傳默認一直循環(huán)
child: Image.asset('assets/images/watermelon.png',width: 325, height: 260, fit: BoxFit.contain),
)
實現(xiàn)原理
① 特效控件分為兩層:底層顯示調用方傳入的控件;上層覆蓋一層漸變著色器。
② 啟動動畫,根據(jù)動畫的進度,對漸變著色器的區(qū)域進行繪制,當區(qū)域變大變小時,著色器高光的地方也在相應進行偏移。
③ 同時著色器不能超出底層控件的繪制范圍,底層控件的形狀是不規(guī)則的,漸變層不能超出底層控件的layer對象。這樣才能實現(xiàn)完全貼合 底層控件形狀 的微光閃爍。
控件分層顯示
@override
Widget build(BuildContext context) {
return Stack(
children: [
// 底層控件
widget.child,
// 覆蓋閃爍微光
AnimatedBuilder(
animation: _controller,
child: widget.child,
builder: (BuildContext context, Widget? child) => _Shimmer(
child: child,
percent: _controller.value,
direction: widget.direction,
gradient: widget.gradient,
),
)
],
);
開啟動畫
late AnimationController _controller;
int _count = 0;
@override
void initState() {
super.initState();
_controller = AnimationController(vsync: this, duration: widget.duration)
..addStatusListener((AnimationStatus status) {
if (status != AnimationStatus.completed) {
return;
}
_count++;
if (widget.loop != 0 && _count < widget.loop) {
_controller.forward(from: 0.0);
}
});
if (widget.loop == 0) {
_controller.repeat();
} else {
_controller.forward();
}
}
重點:著色器該如何繪制,又該如何通過AnimationController的進度進行偏移?由于著色器不能超出底層控件的繪制范圍,所以必須拿到底層控件的繪制上下文【即 PaintingContext】,調用其pushLayer方法,讓引擎把著色器繪制上去。
需要用到PaintingContext,自然就需要去管理RenderObject,所以著色器的編寫使用RenderProxyBox進行計算并繪制出layer對象,計算的過程根據(jù)上面的AnimationController的進度進行計算。
class _ShimmerFilter extends RenderProxyBox {
ShimmerDirection _direction;
Gradient _gradient;
double _percent;
_ShimmerFilter(this._percent, this._direction, this._gradient);
@override
ShaderMaskLayer? get layer => super.layer as ShaderMaskLayer?;
set percent(double newValue) {
if (newValue != _percent) {
_percent = newValue;
markNeedsPaint();
}
}
set gradient(Gradient newValue) {
if (newValue != _gradient) {
_gradient = newValue;
markNeedsPaint();
}
}
set direction(ShimmerDirection newDirection) {
if (newDirection != _direction) {
_direction = newDirection;
markNeedsLayout();
}
}
@override
void paint(PaintingContext context, Offset offset) {
if (child != null) {
final double width = child!.size.width;
final double height = child!.size.height;
Rect rect;
double dx, dy;
if (_direction == ShimmerDirection.rtl) {
dx = _offset(width, -width, _percent);
dy = 0.0;
rect = Rect.fromLTWH(dx - width, dy, 3 * width, height);
} else if (_direction == ShimmerDirection.ttb) {
dx = 0.0;
dy = _offset(-height, height, _percent);
rect = Rect.fromLTWH(dx, dy - height, width, 3 * height);
} else if (_direction == ShimmerDirection.btt) {
dx = 0.0;
dy = _offset(height, -height, _percent);
rect = Rect.fromLTWH(dx, dy - height, width, 3 * height);
} else {
dx = _offset(-width, width, _percent);
dy = 0.0;
rect = Rect.fromLTWH(dx - width, dy, 3 * width, height);
}
layer ??= ShaderMaskLayer();
layer!
..shader = _gradient.createShader(rect)
..maskRect = offset & size
..blendMode = BlendMode.srcIn;
context.pushLayer(layer!, super.paint, offset);
} else {
layer = null;
}
}
double _offset(double start, double end, double percent) {
return start + (end - start) * percent;
}
}
Render對象繪制出來后,需要封裝成widget使用,由于是單一組件,用SingleChildRenderObjectWidget即可。
class _Shimmer extends SingleChildRenderObjectWidget {
@override
_ShimmerFilter createRenderObject(BuildContext context) {
return _ShimmerFilter(percent, direction, gradient);
}
@override
void updateRenderObject(BuildContext context, _ShimmerFilter shimmer) {
shimmer.percent = percent;
shimmer.gradient = gradient;
shimmer.direction = direction;
}
}
寫在最后
這種閃爍動畫,應用場景多種多樣。可以作為對重要視圖的著重顯示,例如:勛章;也可以作為加載中骨架屏的加載動畫。自己靈活使用即可。
作為一個大前端開發(fā)者,我希望把UI盡善盡美的展現(xiàn)給用戶;此時你不僅需要一個集能力、審美、高標準于一體的設計師配合,更需要自己對所寫界面有著極高的追求。而Flutter作為一個UI框架,玩到最后其實就是特效動畫的高性能編寫,這勢必離不開其繪制原理,不要停留在widget、element的學習,Render、layer甚至再底層的C++才是我們學習路徑。
參考文檔:
總結
到此這篇關于Flutter給控件實現(xiàn)鉆石般的微光特效的文章就介紹到這了,更多相關Flutter控件微光特效內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關文章希望大家以后多多支持腳本之家!
相關文章
Android使用BottomNavigationBar實現(xiàn)底部導航欄
這篇文章主要為大家詳細介紹了Android使用BottomNavigationBar實現(xiàn)底部導航欄,具有一定的參考價值,感興趣的小伙伴們可以參考一下2019-02-02
Android動態(tài)人臉檢測的示例代碼(臉數(shù)可調)
本篇文章主要介紹了Android動態(tài)人臉檢測的示例代碼(臉數(shù)可調),具有一定的參考價值,有興趣的可以了解一下2017-08-08
Android Studio實現(xiàn)第三方QQ登錄操作代碼
這篇文章主要介紹了Android Studio實現(xiàn)第三方QQ登錄的操作方法,本文圖文并茂給大家介紹的非常詳細,具有參考借鑒價值,需要的朋友可以參考下2017-12-12
Android Canvas之drawBitmap方法案例詳解
這篇文章主要介紹了Android Canvas之drawBitmap方法案例詳解,本篇文章通過簡要的案例,講解了該項技術的了解與使用,以下就是詳細內(nèi)容,需要的朋友可以參考下2021-08-08
Android SQLite數(shù)據(jù)庫加密的操作方法
因為Android自帶的SQLite數(shù)據(jù)庫本身是沒有實現(xiàn)加密的,那我們?nèi)绾螌崿F(xiàn)對數(shù)據(jù)庫的加密呢?今天通過本文給大家介紹下Android SQLite數(shù)據(jù)庫加密的操作方法,一起看看吧2021-09-09

