您最多选择25个主题
主题必须以中文或者字母或数字开头,可以包含连字符 (-),并且长度不得超过35个字符
360 行
13 KiB
360 行
13 KiB
using System;
|
|
using System.Collections.Generic;
|
|
using Unity.UIWidgets.foundation;
|
|
using Unity.UIWidgets.ui;
|
|
using UnityEngine;
|
|
using Canvas = Unity.UIWidgets.ui.Canvas;
|
|
using Color = Unity.UIWidgets.ui.Color;
|
|
using Rect = Unity.UIWidgets.ui.Rect;
|
|
|
|
namespace Unity.UIWidgets.painting {
|
|
public class BoxDecoration : Decoration, IEquatable<BoxDecoration> {
|
|
public BoxDecoration(
|
|
Color color = null,
|
|
DecorationImage image = null,
|
|
Border border = null,
|
|
BorderRadius borderRadius = null,
|
|
List<BoxShadow> boxShadow = null,
|
|
Gradient gradient = null,
|
|
BlendMode? backgroundBlendMode = null,
|
|
BoxShape shape = BoxShape.rectangle
|
|
) {
|
|
D.assert(
|
|
backgroundBlendMode == null || color != null || gradient != null,
|
|
() => "backgroundBlendMode applies to BoxDecoration\'s background color or " +
|
|
"gradient, but no color or gradient was provided."
|
|
);
|
|
|
|
this.color = color;
|
|
this.image = image;
|
|
this.border = border;
|
|
this.borderRadius = borderRadius;
|
|
this.boxShadow = boxShadow;
|
|
this.gradient = gradient;
|
|
this.backgroundBlendMode = backgroundBlendMode;
|
|
this.shape = shape;
|
|
}
|
|
|
|
public override bool debugAssertIsValid() {
|
|
D.assert(shape != BoxShape.circle || borderRadius == null);
|
|
return base.debugAssertIsValid();
|
|
}
|
|
|
|
public readonly Color color;
|
|
public readonly DecorationImage image;
|
|
public readonly Border border;
|
|
public readonly BorderRadius borderRadius;
|
|
public readonly List<BoxShadow> boxShadow;
|
|
public readonly Gradient gradient;
|
|
public readonly BlendMode? backgroundBlendMode;
|
|
public readonly BoxShape shape;
|
|
|
|
public override EdgeInsets padding {
|
|
get { return border?.dimensions; }
|
|
}
|
|
|
|
public BoxDecoration scale(float factor) {
|
|
return new BoxDecoration(
|
|
color: Color.lerp(null, color, factor),
|
|
image: image,
|
|
border: Border.lerp(null, border, factor),
|
|
borderRadius: BorderRadius.lerp(null, borderRadius, factor),
|
|
boxShadow: BoxShadow.lerpList(null, boxShadow, factor),
|
|
gradient: gradient?.scale(factor),
|
|
backgroundBlendMode: backgroundBlendMode,
|
|
shape: shape
|
|
);
|
|
}
|
|
|
|
public override bool isComplex {
|
|
get { return boxShadow != null; }
|
|
}
|
|
|
|
public override Decoration lerpFrom(Decoration a, float t) {
|
|
if (a == null) {
|
|
return scale(t);
|
|
}
|
|
|
|
if (a is BoxDecoration boxDecoration) {
|
|
return lerp(boxDecoration, this, t);
|
|
}
|
|
|
|
return base.lerpFrom(a, t);
|
|
}
|
|
|
|
public override Decoration lerpTo(Decoration b, float t) {
|
|
if (b == null) {
|
|
return scale(1.0f - t);
|
|
}
|
|
|
|
if (b is BoxDecoration boxDecoration) {
|
|
return lerp(this, boxDecoration, t);
|
|
}
|
|
|
|
return base.lerpTo(b, t);
|
|
}
|
|
|
|
public static BoxDecoration lerp(BoxDecoration a, BoxDecoration b, float t) {
|
|
if (a == null && b == null) {
|
|
return null;
|
|
}
|
|
|
|
if (a == null) {
|
|
return b.scale(t);
|
|
}
|
|
|
|
if (b == null) {
|
|
return a.scale(1.0f - t);
|
|
}
|
|
|
|
if (t == 0.0) {
|
|
return a;
|
|
}
|
|
|
|
if (t == 1.0) {
|
|
return b;
|
|
}
|
|
|
|
return new BoxDecoration(
|
|
color: Color.lerp(a.color, b.color, t),
|
|
image: t < 0.5 ? a.image : b.image,
|
|
border: Border.lerp(a.border, b.border, t),
|
|
borderRadius: BorderRadius.lerp(a.borderRadius, b.borderRadius, t),
|
|
boxShadow: BoxShadow.lerpList(a.boxShadow, b.boxShadow, t),
|
|
gradient: Gradient.lerp(a.gradient, b.gradient, t),
|
|
backgroundBlendMode: t < 0.5 ? a.backgroundBlendMode : b.backgroundBlendMode,
|
|
shape: t < 0.5 ? a.shape : b.shape
|
|
);
|
|
}
|
|
|
|
public bool Equals(BoxDecoration other) {
|
|
if (ReferenceEquals(null, other)) {
|
|
return false;
|
|
}
|
|
|
|
if (ReferenceEquals(this, other)) {
|
|
return true;
|
|
}
|
|
|
|
return Equals(color, other.color) && Equals(image, other.image) &&
|
|
Equals(border, other.border) && Equals(borderRadius, other.borderRadius) &&
|
|
Equals(boxShadow, other.boxShadow) && Equals(gradient, other.gradient) &&
|
|
backgroundBlendMode == other.backgroundBlendMode && shape == other.shape;
|
|
}
|
|
|
|
public override bool Equals(object obj) {
|
|
if (ReferenceEquals(null, obj)) {
|
|
return false;
|
|
}
|
|
|
|
if (ReferenceEquals(this, obj)) {
|
|
return true;
|
|
}
|
|
|
|
if (obj.GetType() != GetType()) {
|
|
return false;
|
|
}
|
|
|
|
return Equals((BoxDecoration) obj);
|
|
}
|
|
|
|
public override int GetHashCode() {
|
|
unchecked {
|
|
var hashCode = (color != null ? color.GetHashCode() : 0);
|
|
hashCode = (hashCode * 397) ^ (image != null ? image.GetHashCode() : 0);
|
|
hashCode = (hashCode * 397) ^ (border != null ? border.GetHashCode() : 0);
|
|
hashCode = (hashCode * 397) ^ (borderRadius != null ? borderRadius.GetHashCode() : 0);
|
|
hashCode = (hashCode * 397) ^ (boxShadow != null ? boxShadow.GetHashCode() : 0);
|
|
hashCode = (hashCode * 397) ^ (gradient != null ? gradient.GetHashCode() : 0);
|
|
hashCode = (hashCode * 397) ^ backgroundBlendMode.GetHashCode();
|
|
hashCode = (hashCode * 397) ^ (int) shape;
|
|
return hashCode;
|
|
}
|
|
}
|
|
|
|
public static bool operator ==(BoxDecoration left, BoxDecoration right) {
|
|
return Equals(left, right);
|
|
}
|
|
|
|
public static bool operator !=(BoxDecoration left, BoxDecoration right) {
|
|
return !Equals(left, right);
|
|
}
|
|
|
|
public override void debugFillProperties(DiagnosticPropertiesBuilder properties) {
|
|
base.debugFillProperties(properties);
|
|
properties.defaultDiagnosticsTreeStyle = DiagnosticsTreeStyle.whitespace;
|
|
properties.emptyBodyDescription = "<no decorations specified>";
|
|
properties.add(new DiagnosticsProperty<Color>("color", color,
|
|
defaultValue: foundation_.kNullDefaultValue));
|
|
properties.add(new DiagnosticsProperty<DecorationImage>("image", image,
|
|
defaultValue: foundation_.kNullDefaultValue));
|
|
properties.add(new DiagnosticsProperty<Border>("border", border,
|
|
defaultValue: foundation_.kNullDefaultValue));
|
|
properties.add(new DiagnosticsProperty<BorderRadius>("borderRadius", borderRadius,
|
|
defaultValue: foundation_.kNullDefaultValue));
|
|
properties.add(new EnumerableProperty<BoxShadow>("boxShadow", boxShadow,
|
|
defaultValue: foundation_.kNullDefaultValue, style: DiagnosticsTreeStyle.whitespace));
|
|
properties.add(new DiagnosticsProperty<Gradient>("gradient", gradient,
|
|
defaultValue: foundation_.kNullDefaultValue));
|
|
properties.add(new DiagnosticsProperty<BlendMode?>("backgroundBlendMode", backgroundBlendMode,
|
|
defaultValue: foundation_.kNullDefaultValue));
|
|
properties.add(new DiagnosticsProperty<BoxShape>("shape", shape, defaultValue: BoxShape.rectangle));
|
|
}
|
|
|
|
public override bool hitTest(Size size, Offset position) {
|
|
D.assert((Offset.zero & size).contains(position));
|
|
switch (shape) {
|
|
case BoxShape.rectangle:
|
|
if (borderRadius != null) {
|
|
RRect bounds = borderRadius.toRRect(Offset.zero & size);
|
|
return bounds.contains(position);
|
|
}
|
|
|
|
return true;
|
|
case BoxShape.circle:
|
|
Offset center = size.center(Offset.zero);
|
|
float distance = (position - center).distance;
|
|
return distance <= Mathf.Min(size.width, size.height) / 2.0;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
public override BoxPainter createBoxPainter(VoidCallback onChanged = null) {
|
|
D.assert(onChanged != null || image == null);
|
|
return new _BoxDecorationPainter(this, onChanged);
|
|
}
|
|
}
|
|
|
|
class _BoxDecorationPainter : BoxPainter {
|
|
public _BoxDecorationPainter(BoxDecoration decoration, VoidCallback onChanged)
|
|
: base(onChanged) {
|
|
D.assert(decoration != null);
|
|
_decoration = decoration;
|
|
}
|
|
|
|
readonly BoxDecoration _decoration;
|
|
|
|
Paint _cachedBackgroundPaint;
|
|
|
|
Rect _rectForCachedBackgroundPaint;
|
|
|
|
Paint _getBackgroundPaint(Rect rect, TextDirection textDirection) {
|
|
D.assert(rect != null);
|
|
D.assert(_decoration.gradient != null || _rectForCachedBackgroundPaint == null);
|
|
|
|
if (_cachedBackgroundPaint == null ||
|
|
(_decoration.gradient != null && _rectForCachedBackgroundPaint != rect)) {
|
|
var paint = new Paint();
|
|
if (_decoration.backgroundBlendMode != null) {
|
|
paint.blendMode = _decoration.backgroundBlendMode.Value;
|
|
}
|
|
|
|
if (_decoration.color != null) {
|
|
paint.color = _decoration.color;
|
|
}
|
|
|
|
if (_decoration.gradient != null) {
|
|
paint.shader = _decoration.gradient.createShader(rect, textDirection: textDirection);
|
|
_rectForCachedBackgroundPaint = rect;
|
|
}
|
|
|
|
_cachedBackgroundPaint = paint;
|
|
}
|
|
|
|
return _cachedBackgroundPaint;
|
|
}
|
|
|
|
void _paintBox(Canvas canvas, Rect rect, Paint paint, TextDirection textDirection) {
|
|
switch (_decoration.shape) {
|
|
case BoxShape.circle:
|
|
D.assert(_decoration.borderRadius == null);
|
|
Offset center = rect.center;
|
|
float radius = rect.shortestSide / 2.0f;
|
|
canvas.drawCircle(center, radius, paint);
|
|
break;
|
|
case BoxShape.rectangle:
|
|
if (_decoration.borderRadius == null) {
|
|
canvas.drawRect(rect, paint);
|
|
}
|
|
else {
|
|
canvas.drawRRect(_decoration.borderRadius.toRRect(rect), paint);
|
|
}
|
|
|
|
break;
|
|
}
|
|
}
|
|
|
|
void _paintShadows(Canvas canvas, Rect rect, TextDirection textDirection) {
|
|
if (_decoration.boxShadow == null) {
|
|
return;
|
|
}
|
|
|
|
foreach (BoxShadow boxShadow in _decoration.boxShadow) {
|
|
Paint paint = boxShadow.toPaint();
|
|
Rect bounds = rect.shift(boxShadow.offset).inflate(boxShadow.spreadRadius);
|
|
_paintBox(canvas, bounds, paint, textDirection);
|
|
}
|
|
}
|
|
|
|
void _paintBackgroundColor(Canvas canvas, Rect rect, TextDirection textDirection) {
|
|
if (_decoration.color != null || _decoration.gradient != null) {
|
|
_paintBox(canvas, rect, _getBackgroundPaint(rect, textDirection), textDirection);
|
|
}
|
|
}
|
|
|
|
DecorationImagePainter _imagePainter;
|
|
|
|
void _paintBackgroundImage(Canvas canvas, Rect rect, ImageConfiguration configuration) {
|
|
if (_decoration.image == null) {
|
|
return;
|
|
}
|
|
|
|
_imagePainter = _imagePainter ?? _decoration.image.createPainter(onChanged);
|
|
|
|
Path clipPath = null;
|
|
switch (_decoration.shape) {
|
|
case BoxShape.circle:
|
|
clipPath = new Path();
|
|
clipPath.addOval(rect);
|
|
break;
|
|
case BoxShape.rectangle:
|
|
if (_decoration.borderRadius != null) {
|
|
clipPath = new Path();
|
|
clipPath.addRRect(_decoration.borderRadius.toRRect(rect));
|
|
}
|
|
|
|
break;
|
|
}
|
|
|
|
_imagePainter.paint(canvas, rect, clipPath, configuration);
|
|
}
|
|
|
|
public override void Dispose() {
|
|
_imagePainter?.Dispose();
|
|
base.Dispose();
|
|
}
|
|
|
|
public override void paint(Canvas canvas, Offset offset, ImageConfiguration configuration) {
|
|
D.assert(configuration != null);
|
|
D.assert(configuration.size != null);
|
|
|
|
Rect rect = offset & configuration.size;
|
|
TextDirection textDirection = configuration.textDirection;
|
|
|
|
_paintShadows(canvas, rect, textDirection);
|
|
_paintBackgroundColor(canvas, rect, textDirection);
|
|
_paintBackgroundImage(canvas, rect, configuration);
|
|
_decoration.border?.paint(
|
|
canvas,
|
|
rect,
|
|
shape: _decoration.shape,
|
|
borderRadius: _decoration.borderRadius
|
|
);
|
|
}
|
|
|
|
public override string ToString() {
|
|
return $"BoxPainter for {_decoration}";
|
|
}
|
|
}
|
|
}
|