You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
97 lines
3.2 KiB
97 lines
3.2 KiB
import 'package:flutter/material.dart'; |
|
|
|
|
|
|
|
///参考UnderlineTabIndicator,仅仅修改画笔StrokeCap.square 为 StrokeCap.round |
|
class CustomUnderlineTabIndicator extends Decoration { |
|
/// Create an underline style selected tab indicator. |
|
/// |
|
/// The [borderSide] and [insets] arguments must not be null. |
|
const CustomUnderlineTabIndicator({ |
|
this.borderSide = const BorderSide(width: 2.0, color: Colors.white), |
|
this.insets = EdgeInsets.zero, |
|
}) : assert(borderSide != null), |
|
assert(insets != null); |
|
|
|
/// The color and weight of the horizontal line drawn below the selected tab. |
|
final BorderSide borderSide; |
|
|
|
/// Locates the selected tab's underline relative to the tab's boundary. |
|
/// |
|
/// The [TabBar.indicatorSize] property can be used to define the tab |
|
/// indicator's bounds in terms of its (centered) tab widget with |
|
/// [TabBarIndicatorSize.label], or the entire tab with |
|
/// [TabBarIndicatorSize.tab]. |
|
final EdgeInsetsGeometry insets; |
|
|
|
@override |
|
Decoration? lerpFrom(Decoration? a, double t) { |
|
if (a is UnderlineTabIndicator) { |
|
return UnderlineTabIndicator( |
|
borderSide: BorderSide.lerp(a.borderSide, borderSide, t), |
|
insets: EdgeInsetsGeometry.lerp(a.insets, insets, t) ?? EdgeInsets.zero, |
|
); |
|
} |
|
return super.lerpFrom(a, t); |
|
} |
|
|
|
@override |
|
Decoration? lerpTo(Decoration? b, double t) { |
|
if (b is UnderlineTabIndicator) { |
|
return UnderlineTabIndicator( |
|
borderSide: BorderSide.lerp(borderSide, b.borderSide, t), |
|
insets: EdgeInsetsGeometry.lerp(insets, b.insets, t) ?? EdgeInsets.zero, |
|
); |
|
} |
|
return super.lerpTo(b, t); |
|
} |
|
|
|
@override |
|
_UnderlinePainter createBoxPainter([VoidCallback? onChanged]) { |
|
assert(onChanged != null); |
|
return _UnderlinePainter(this, onChanged!); |
|
} |
|
|
|
Rect _indicatorRectFor(Rect rect, TextDirection? textDirection) { |
|
assert(rect != null); |
|
assert(textDirection != null); |
|
final Rect indicator = insets.resolve(textDirection).deflateRect(rect); |
|
return Rect.fromLTWH( |
|
indicator.left, |
|
indicator.bottom - borderSide.width, |
|
indicator.width, |
|
borderSide.width, |
|
); |
|
} |
|
|
|
@override |
|
Path getClipPath(Rect rect, TextDirection textDirection) { |
|
return Path()..addRect(_indicatorRectFor(rect, textDirection)); |
|
} |
|
} |
|
|
|
class _UnderlinePainter extends BoxPainter { |
|
_UnderlinePainter(this.decoration, VoidCallback onChanged) |
|
: assert(decoration != null), |
|
super(onChanged); |
|
|
|
final CustomUnderlineTabIndicator decoration; |
|
|
|
BorderSide? get borderSide => decoration?.borderSide; |
|
|
|
EdgeInsetsGeometry? get insets => decoration?.insets; |
|
|
|
@override |
|
void paint(Canvas canvas, Offset offset, ImageConfiguration? configuration) { |
|
assert(configuration != null); |
|
assert(configuration!.size != null); |
|
final Rect rect = offset & configuration!.size!; |
|
final TextDirection? textDirection = configuration.textDirection; |
|
final Rect indicator = decoration._indicatorRectFor(rect, textDirection) |
|
.deflate(decoration.borderSide.width / 2.0); |
|
final Paint paint = decoration.borderSide.toPaint(); |
|
paint.strokeWidth = 4; |
|
paint.strokeCap = StrokeCap.round; //主要是修改此处 圆角 |
|
canvas.drawLine(indicator.bottomLeft, indicator.bottomRight, paint); |
|
} |
|
} |