您最多选择25个主题
主题必须以中文或者字母或数字开头,可以包含连字符 (-),并且长度不得超过35个字符
574 行
20 KiB
574 行
20 KiB
using System;
|
|
using System.Collections.Generic;
|
|
using Unity.UIWidgets.animation;
|
|
using Unity.UIWidgets.foundation;
|
|
using Unity.UIWidgets.painting;
|
|
using Unity.UIWidgets.rendering;
|
|
using Unity.UIWidgets.scheduler;
|
|
using Unity.UIWidgets.ui;
|
|
using Unity.UIWidgets.widgets;
|
|
using TextStyle = Unity.UIWidgets.painting.TextStyle;
|
|
|
|
namespace Unity.UIWidgets.material {
|
|
static class MaterialConstantsUtils {
|
|
public static readonly Dictionary<MaterialType, BorderRadius> kMaterialEdges =
|
|
new Dictionary<MaterialType, BorderRadius> {
|
|
{MaterialType.canvas, null},
|
|
{MaterialType.card, BorderRadius.circular(2.0f)},
|
|
{MaterialType.circle, null},
|
|
{MaterialType.button, BorderRadius.circular(2.0f)},
|
|
{MaterialType.transparency, null}
|
|
};
|
|
}
|
|
|
|
|
|
public delegate Rect RectCallback();
|
|
|
|
public enum MaterialType {
|
|
canvas,
|
|
card,
|
|
circle,
|
|
button,
|
|
transparency
|
|
}
|
|
|
|
|
|
public interface MaterialInkController {
|
|
Color color { get; set; }
|
|
|
|
TickerProvider vsync { get; }
|
|
|
|
void addInkFeature(InkFeature feature);
|
|
|
|
void markNeedsPaint();
|
|
}
|
|
|
|
|
|
public class Material : StatefulWidget {
|
|
public Material(
|
|
Key key = null,
|
|
MaterialType type = MaterialType.canvas,
|
|
float elevation = 0.0f,
|
|
Color color = null,
|
|
Color shadowColor = null,
|
|
TextStyle textStyle = null,
|
|
BorderRadius borderRadius = null,
|
|
ShapeBorder shape = null,
|
|
bool borderOnForeground = true,
|
|
Clip clipBehavior = Clip.none,
|
|
TimeSpan? animationDuration = null,
|
|
Widget child = null
|
|
) : base(key: key) {
|
|
D.assert(elevation >= 0.0f);
|
|
D.assert(!(shape != null && borderRadius != null));
|
|
D.assert(!(type == MaterialType.circle && (borderRadius != null || shape != null)));
|
|
|
|
this.type = type;
|
|
this.elevation = elevation;
|
|
this.color = color;
|
|
this.shadowColor = shadowColor ?? new Color(0xFF000000);
|
|
this.textStyle = textStyle;
|
|
this.borderRadius = borderRadius;
|
|
this.shape = shape;
|
|
this.borderOnForeground = borderOnForeground;
|
|
this.clipBehavior = clipBehavior;
|
|
this.animationDuration = animationDuration ?? material_.kThemeChangeDuration;
|
|
this.child = child;
|
|
}
|
|
|
|
public readonly Widget child;
|
|
|
|
public readonly MaterialType type;
|
|
|
|
public readonly float elevation;
|
|
|
|
public readonly Color color;
|
|
|
|
public readonly Color shadowColor;
|
|
|
|
public readonly TextStyle textStyle;
|
|
|
|
public readonly ShapeBorder shape;
|
|
|
|
public readonly bool borderOnForeground;
|
|
|
|
public readonly Clip clipBehavior;
|
|
|
|
public readonly TimeSpan animationDuration;
|
|
|
|
public readonly BorderRadius borderRadius;
|
|
|
|
|
|
public static MaterialInkController of(BuildContext context) {
|
|
_RenderInkFeatures result =
|
|
(_RenderInkFeatures) context.findAncestorRenderObjectOfType<_RenderInkFeatures>();
|
|
return result;
|
|
}
|
|
|
|
public override State createState() {
|
|
return new _MaterialState();
|
|
}
|
|
|
|
|
|
public override void debugFillProperties(DiagnosticPropertiesBuilder properties) {
|
|
base.debugFillProperties(properties);
|
|
properties.add(new EnumProperty<MaterialType>("type", type));
|
|
properties.add(new FloatProperty("elevation", elevation, defaultValue: 0.0f));
|
|
properties.add(new ColorProperty("color", color, defaultValue: null));
|
|
properties.add(new ColorProperty("shadowColor", shadowColor, defaultValue: new Color(0xFF000000)));
|
|
textStyle?.debugFillProperties(properties);
|
|
properties.add(new DiagnosticsProperty<ShapeBorder>("shape", shape, defaultValue: null));
|
|
properties.add(new DiagnosticsProperty<bool>("borderOnForeground", borderOnForeground,
|
|
defaultValue: null));
|
|
properties.add(
|
|
new DiagnosticsProperty<BorderRadius>("borderRadius", borderRadius, defaultValue: null));
|
|
}
|
|
|
|
public const float defaultSplashRadius = 35.0f;
|
|
}
|
|
|
|
|
|
class _MaterialState : TickerProviderStateMixin<Material> {
|
|
readonly GlobalKey _inkFeatureRenderer = GlobalKey.key(debugLabel: "ink renderer");
|
|
|
|
Color _getBackgroundColor(BuildContext context) {
|
|
ThemeData theme = Theme.of(context);
|
|
Color color = widget.color;
|
|
if (color == null) {
|
|
switch (widget.type) {
|
|
case MaterialType.canvas:
|
|
color = theme.canvasColor;
|
|
break;
|
|
case MaterialType.card:
|
|
color = theme.cardColor;
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
return color;
|
|
}
|
|
|
|
public override Widget build(BuildContext context) {
|
|
Color backgroundColor = _getBackgroundColor(context);
|
|
D.assert(backgroundColor != null || widget.type == MaterialType.transparency,
|
|
() => "If Material type is not MaterialType.transparency, a color must" +
|
|
"either be passed in through the 'color' property, or be defined " +
|
|
"in the theme (ex. canvasColor != null if type is set to " +
|
|
"MaterialType.canvas");
|
|
Widget contents = widget.child;
|
|
if (contents != null) {
|
|
contents = new AnimatedDefaultTextStyle(
|
|
style: widget.textStyle ?? Theme.of(context).textTheme.bodyText2,
|
|
duration: widget.animationDuration,
|
|
child: contents
|
|
);
|
|
}
|
|
|
|
contents = new NotificationListener<LayoutChangedNotification>(
|
|
onNotification: (LayoutChangedNotification notification) => {
|
|
_RenderInkFeatures renderer =
|
|
(_RenderInkFeatures) _inkFeatureRenderer.currentContext.findRenderObject();
|
|
renderer._didChangeLayout();
|
|
return false;
|
|
},
|
|
child: new _InkFeatures(
|
|
key: _inkFeatureRenderer,
|
|
color: backgroundColor,
|
|
child: contents,
|
|
vsync: this
|
|
)
|
|
);
|
|
|
|
if (widget.type == MaterialType.canvas && widget.shape == null &&
|
|
widget.borderRadius == null) {
|
|
return new AnimatedPhysicalModel(
|
|
curve: Curves.fastOutSlowIn,
|
|
duration: widget.animationDuration,
|
|
shape: BoxShape.rectangle,
|
|
clipBehavior: widget.clipBehavior,
|
|
borderRadius: BorderRadius.zero,
|
|
elevation: widget.elevation,
|
|
color: ElevationOverlay.applyOverlay(context, backgroundColor, widget.elevation),
|
|
shadowColor: widget.shadowColor,
|
|
animateColor: false,
|
|
child: contents
|
|
);
|
|
}
|
|
|
|
ShapeBorder shape = _getShape();
|
|
|
|
if (widget.type == MaterialType.transparency) {
|
|
return _transparentInterior(
|
|
context: context,
|
|
shape: shape,
|
|
clipBehavior: widget.clipBehavior,
|
|
contents: contents);
|
|
}
|
|
|
|
return new _MaterialInterior(
|
|
curve: Curves.fastOutSlowIn,
|
|
duration: widget.animationDuration,
|
|
shape: shape,
|
|
borderOnForeground: widget.borderOnForeground,
|
|
clipBehavior: widget.clipBehavior,
|
|
elevation: widget.elevation,
|
|
color: backgroundColor,
|
|
shadowColor: widget.shadowColor,
|
|
child: contents
|
|
);
|
|
}
|
|
|
|
|
|
static Widget _transparentInterior(
|
|
BuildContext context = null,
|
|
ShapeBorder shape = null,
|
|
Clip? clipBehavior = null,
|
|
Widget contents = null) {
|
|
_ShapeBorderPaint child = new _ShapeBorderPaint(
|
|
child: contents,
|
|
shape: shape);
|
|
|
|
if (clipBehavior == Clip.none) {
|
|
return child;
|
|
}
|
|
|
|
return new ClipPath(
|
|
child: child,
|
|
clipper: new ShapeBorderClipper(shape: shape),
|
|
clipBehavior: clipBehavior ?? Clip.none
|
|
);
|
|
}
|
|
|
|
|
|
ShapeBorder _getShape() {
|
|
if (widget.shape != null) {
|
|
return widget.shape;
|
|
}
|
|
|
|
if (widget.borderRadius != null) {
|
|
return new RoundedRectangleBorder(borderRadius: widget.borderRadius);
|
|
}
|
|
|
|
switch (widget.type) {
|
|
case MaterialType.canvas:
|
|
case MaterialType.transparency:
|
|
return new RoundedRectangleBorder();
|
|
case MaterialType.card:
|
|
case MaterialType.button:
|
|
return new RoundedRectangleBorder(
|
|
borderRadius: widget.borderRadius ??
|
|
MaterialConstantsUtils.kMaterialEdges[widget.type]);
|
|
case MaterialType.circle:
|
|
return new CircleBorder();
|
|
}
|
|
|
|
return new RoundedRectangleBorder();
|
|
}
|
|
}
|
|
|
|
|
|
public class _RenderInkFeatures : RenderProxyBox, MaterialInkController {
|
|
public _RenderInkFeatures(
|
|
RenderBox child = null,
|
|
TickerProvider vsync = null,
|
|
Color color = null) : base(child: child) {
|
|
D.assert(vsync != null);
|
|
_vsync = vsync;
|
|
_color = color;
|
|
}
|
|
|
|
public TickerProvider vsync {
|
|
get { return _vsync; }
|
|
}
|
|
|
|
readonly TickerProvider _vsync;
|
|
|
|
public Color color {
|
|
get { return _color; }
|
|
set { _color = value; }
|
|
}
|
|
|
|
Color _color;
|
|
|
|
List<InkFeature> _inkFeatures;
|
|
|
|
public void addInkFeature(InkFeature feature) {
|
|
D.assert(!feature._debugDisposed);
|
|
D.assert(feature._controller == this);
|
|
_inkFeatures = _inkFeatures ?? new List<InkFeature>();
|
|
D.assert(!_inkFeatures.Contains(feature));
|
|
_inkFeatures.Add(feature);
|
|
markNeedsPaint();
|
|
}
|
|
|
|
public void _removeFeature(InkFeature feature) {
|
|
D.assert(_inkFeatures != null);
|
|
_inkFeatures.Remove(feature);
|
|
markNeedsPaint();
|
|
}
|
|
|
|
public void _didChangeLayout() {
|
|
if (_inkFeatures != null && _inkFeatures.isNotEmpty()) {
|
|
markNeedsPaint();
|
|
}
|
|
}
|
|
|
|
protected override bool hitTestSelf(Offset position) {
|
|
return true;
|
|
}
|
|
|
|
public override void paint(PaintingContext context, Offset offset) {
|
|
if (_inkFeatures != null && _inkFeatures.isNotEmpty()) {
|
|
Canvas canvas = context.canvas;
|
|
canvas.save();
|
|
canvas.translate(offset.dx, offset.dy);
|
|
canvas.clipRect(Offset.zero & size);
|
|
foreach (InkFeature inkFeature in _inkFeatures) {
|
|
inkFeature._paint(canvas);
|
|
}
|
|
|
|
canvas.restore();
|
|
}
|
|
|
|
base.paint(context, offset);
|
|
}
|
|
}
|
|
|
|
|
|
public class _InkFeatures : SingleChildRenderObjectWidget {
|
|
public _InkFeatures(
|
|
Key key = null,
|
|
Color color = null,
|
|
TickerProvider vsync = null,
|
|
Widget child = null) : base(key: key, child: child) {
|
|
D.assert(vsync != null);
|
|
this.color = color;
|
|
this.vsync = vsync;
|
|
}
|
|
|
|
public readonly Color color;
|
|
|
|
public readonly TickerProvider vsync;
|
|
|
|
public override RenderObject createRenderObject(BuildContext context) {
|
|
return new _RenderInkFeatures(
|
|
color: color,
|
|
vsync: vsync);
|
|
}
|
|
|
|
public override void updateRenderObject(BuildContext context, RenderObject renderObject) {
|
|
_RenderInkFeatures _renderObject = (_RenderInkFeatures) renderObject;
|
|
_renderObject.color = color;
|
|
D.assert(vsync == _renderObject.vsync);
|
|
}
|
|
}
|
|
|
|
public abstract class InkFeature {
|
|
public InkFeature(
|
|
MaterialInkController controller = null,
|
|
RenderBox referenceBox = null,
|
|
VoidCallback onRemoved = null) {
|
|
D.assert(controller != null);
|
|
D.assert(referenceBox != null);
|
|
_controller = (_RenderInkFeatures) controller;
|
|
this.referenceBox = referenceBox;
|
|
this.onRemoved = onRemoved;
|
|
}
|
|
|
|
public MaterialInkController controller {
|
|
get { return _controller; }
|
|
}
|
|
|
|
public readonly _RenderInkFeatures _controller;
|
|
|
|
public readonly RenderBox referenceBox;
|
|
|
|
public readonly VoidCallback onRemoved;
|
|
|
|
public bool _debugDisposed = false;
|
|
|
|
public virtual void dispose() {
|
|
D.assert(!_debugDisposed);
|
|
D.assert(() => {
|
|
_debugDisposed = true;
|
|
return true;
|
|
});
|
|
_controller._removeFeature(this);
|
|
if (onRemoved != null) {
|
|
onRemoved();
|
|
}
|
|
}
|
|
|
|
public void _paint(Canvas canvas) {
|
|
D.assert(referenceBox.attached);
|
|
D.assert(!_debugDisposed);
|
|
|
|
List<RenderObject> descendants = new List<RenderObject> {referenceBox};
|
|
RenderObject node = referenceBox;
|
|
while (node != _controller) {
|
|
node = (RenderObject) node.parent;
|
|
D.assert(node != null);
|
|
descendants.Add(node);
|
|
}
|
|
|
|
Matrix4 transform = Matrix4.identity();
|
|
D.assert(descendants.Count >= 2);
|
|
for (int index = descendants.Count - 1; index > 0; index -= 1) {
|
|
descendants[index].applyPaintTransform(descendants[index - 1], transform);
|
|
}
|
|
|
|
paintFeature(canvas, transform);
|
|
}
|
|
|
|
protected abstract void paintFeature(Canvas canvas, Matrix4 transform);
|
|
|
|
public override string ToString() {
|
|
return GetType() + "";
|
|
}
|
|
}
|
|
|
|
public class ShapeBorderTween : Tween<ShapeBorder> {
|
|
public ShapeBorderTween(
|
|
ShapeBorder begin = null,
|
|
ShapeBorder end = null) : base(begin: begin, end: end) {
|
|
}
|
|
|
|
public override ShapeBorder lerp(float t) {
|
|
return ShapeBorder.lerp(begin, end, t);
|
|
}
|
|
}
|
|
|
|
public class _MaterialInterior : ImplicitlyAnimatedWidget {
|
|
public _MaterialInterior(
|
|
Key key = null,
|
|
Widget child = null,
|
|
ShapeBorder shape = null,
|
|
bool borderOnForeground = true,
|
|
Clip clipBehavior = Clip.none,
|
|
float? elevation = null,
|
|
Color color = null,
|
|
Color shadowColor = null,
|
|
Curve curve = null,
|
|
TimeSpan? duration = null
|
|
) : base(key: key, curve: curve ?? Curves.linear, duration: duration) {
|
|
D.assert(child != null);
|
|
D.assert(shape != null);
|
|
D.assert(elevation != null && elevation >= 0.0f);
|
|
D.assert(color != null);
|
|
D.assert(shadowColor != null);
|
|
D.assert(duration != null);
|
|
this.child = child;
|
|
this.shape = shape;
|
|
this.borderOnForeground = borderOnForeground;
|
|
this.clipBehavior = clipBehavior;
|
|
this.elevation = elevation ?? 0.0f;
|
|
this.color = color;
|
|
this.shadowColor = shadowColor;
|
|
}
|
|
|
|
public readonly Widget child;
|
|
|
|
public readonly ShapeBorder shape;
|
|
|
|
public readonly bool borderOnForeground;
|
|
|
|
public readonly Clip clipBehavior;
|
|
|
|
public readonly float elevation;
|
|
|
|
public readonly Color color;
|
|
|
|
public readonly Color shadowColor;
|
|
|
|
public override State createState() {
|
|
return new _MaterialInteriorState();
|
|
}
|
|
|
|
|
|
public override void debugFillProperties(DiagnosticPropertiesBuilder description) {
|
|
base.debugFillProperties(description);
|
|
description.add(new DiagnosticsProperty<ShapeBorder>("shape", shape));
|
|
description.add(new FloatProperty("elevation", elevation));
|
|
description.add(new ColorProperty("color", color));
|
|
description.add(new ColorProperty("shadowColor", shadowColor));
|
|
}
|
|
}
|
|
|
|
public class _MaterialInteriorState : AnimatedWidgetBaseState<_MaterialInterior> {
|
|
FloatTween _elevation;
|
|
ColorTween _shadowColor;
|
|
ShapeBorderTween _border;
|
|
|
|
protected override void forEachTween(TweenVisitor visitor) {
|
|
_elevation = (FloatTween) visitor.visit(this, _elevation, widget.elevation,
|
|
(float value) => new FloatTween(begin: value, end: value));
|
|
_shadowColor = (ColorTween) visitor.visit(this, _shadowColor, widget.shadowColor,
|
|
(Color value) => new ColorTween(begin: value));
|
|
_border = (ShapeBorderTween) visitor.visit(this, _border, widget.shape,
|
|
(ShapeBorder value) => new ShapeBorderTween(begin: value));
|
|
}
|
|
|
|
public override Widget build(BuildContext context) {
|
|
ShapeBorder shape = _border.evaluate(animation);
|
|
float elevation = _elevation.evaluate(animation);
|
|
return new PhysicalShape(
|
|
child: new _ShapeBorderPaint(
|
|
child: widget.child,
|
|
shape: shape,
|
|
borderOnForeground: widget.borderOnForeground),
|
|
clipper: new ShapeBorderClipper(
|
|
shape: shape),
|
|
clipBehavior: widget.clipBehavior,
|
|
elevation: _elevation.evaluate(animation),
|
|
color: ElevationOverlay.applyOverlay(context, widget.color, elevation),
|
|
shadowColor: _shadowColor.evaluate(animation)
|
|
);
|
|
}
|
|
}
|
|
|
|
class _ShapeBorderPaint : StatelessWidget {
|
|
public _ShapeBorderPaint(
|
|
Widget child = null,
|
|
ShapeBorder shape = null,
|
|
bool borderOnForeground = true) {
|
|
D.assert(child != null);
|
|
D.assert(shape != null);
|
|
this.child = child;
|
|
this.shape = shape;
|
|
this.borderOnForeground = borderOnForeground;
|
|
}
|
|
|
|
public readonly Widget child;
|
|
|
|
public readonly ShapeBorder shape;
|
|
|
|
public readonly bool borderOnForeground;
|
|
|
|
public override Widget build(BuildContext context) {
|
|
return new CustomPaint(
|
|
child: child,
|
|
painter: borderOnForeground ? null : new _ShapeBorderPainter(shape),
|
|
foregroundPainter: borderOnForeground ? new _ShapeBorderPainter(shape) : null);
|
|
}
|
|
}
|
|
|
|
class _ShapeBorderPainter : AbstractCustomPainter {
|
|
public _ShapeBorderPainter(ShapeBorder border = null) : base(null) {
|
|
this.border = border;
|
|
}
|
|
|
|
public readonly ShapeBorder border;
|
|
|
|
|
|
public override void paint(Canvas canvas, Size size) {
|
|
border.paint(canvas, Offset.zero & size, null);
|
|
}
|
|
|
|
public override bool shouldRepaint(CustomPainter oldDelegate) {
|
|
_ShapeBorderPainter _oldDelegate = (_ShapeBorderPainter) oldDelegate;
|
|
return _oldDelegate.border != border;
|
|
}
|
|
}
|
|
}
|