您最多选择25个主题
主题必须以中文或者字母或数字开头,可以包含连字符 (-),并且长度不得超过35个字符
1012 行
43 KiB
1012 行
43 KiB
using Unity.UIWidgets.animation;
|
|
using Unity.UIWidgets.foundation;
|
|
using Unity.UIWidgets.painting;
|
|
using Unity.UIWidgets.rendering;
|
|
using Unity.UIWidgets.ui;
|
|
using Unity.UIWidgets.widgets;
|
|
using UnityEngine;
|
|
using Canvas = Unity.UIWidgets.ui.Canvas;
|
|
using Color = Unity.UIWidgets.ui.Color;
|
|
using Rect = Unity.UIWidgets.ui.Rect;
|
|
using TextStyle = Unity.UIWidgets.painting.TextStyle;
|
|
|
|
namespace Unity.UIWidgets.material {
|
|
public class SliderTheme : InheritedWidget {
|
|
public SliderTheme(
|
|
Key key = null,
|
|
SliderThemeData data = null,
|
|
Widget child = null)
|
|
: base(key: key, child: child) {
|
|
D.assert(child != null);
|
|
D.assert(data != null);
|
|
this.data = data;
|
|
}
|
|
|
|
public readonly SliderThemeData data;
|
|
|
|
public static SliderThemeData of(BuildContext context) {
|
|
SliderTheme inheritedTheme = (SliderTheme) context.inheritFromWidgetOfExactType(typeof(SliderTheme));
|
|
return inheritedTheme != null ? inheritedTheme.data : Theme.of(context).sliderTheme;
|
|
}
|
|
|
|
public override bool updateShouldNotify(InheritedWidget oldWidget) {
|
|
SliderTheme _oldWidget = (SliderTheme) oldWidget;
|
|
return data != _oldWidget.data;
|
|
}
|
|
}
|
|
|
|
|
|
public enum ShowValueIndicator {
|
|
onlyForDiscrete,
|
|
onlyForContinuous,
|
|
always,
|
|
never
|
|
}
|
|
|
|
public class SliderThemeData : Diagnosticable {
|
|
public SliderThemeData(
|
|
float? trackHeight = null,
|
|
Color activeTrackColor = null,
|
|
Color inactiveTrackColor = null,
|
|
Color disabledActiveTrackColor = null,
|
|
Color disabledInactiveTrackColor = null,
|
|
Color activeTickMarkColor = null,
|
|
Color inactiveTickMarkColor = null,
|
|
Color disabledActiveTickMarkColor = null,
|
|
Color disabledInactiveTickMarkColor = null,
|
|
Color thumbColor = null,
|
|
Color disabledThumbColor = null,
|
|
Color overlayColor = null,
|
|
Color valueIndicatorColor = null,
|
|
SliderTrackShape trackShape = null,
|
|
SliderTickMarkShape tickMarkShape = null,
|
|
SliderComponentShape thumbShape = null,
|
|
SliderComponentShape overlayShape = null,
|
|
SliderComponentShape valueIndicatorShape = null,
|
|
ShowValueIndicator? showValueIndicator = null,
|
|
TextStyle valueIndicatorTextStyle = null
|
|
) {
|
|
D.assert(trackHeight != null);
|
|
D.assert(activeTrackColor != null);
|
|
D.assert(inactiveTrackColor != null);
|
|
D.assert(disabledActiveTrackColor != null);
|
|
D.assert(disabledInactiveTrackColor != null);
|
|
D.assert(activeTickMarkColor != null);
|
|
D.assert(inactiveTickMarkColor != null);
|
|
D.assert(disabledActiveTickMarkColor != null);
|
|
D.assert(disabledInactiveTickMarkColor != null);
|
|
D.assert(thumbColor != null);
|
|
D.assert(disabledThumbColor != null);
|
|
D.assert(overlayColor != null);
|
|
D.assert(valueIndicatorColor != null);
|
|
D.assert(trackShape != null);
|
|
D.assert(tickMarkShape != null);
|
|
D.assert(thumbShape != null);
|
|
D.assert(overlayShape != null);
|
|
D.assert(valueIndicatorShape != null);
|
|
D.assert(valueIndicatorTextStyle != null);
|
|
D.assert(showValueIndicator != null);
|
|
this.trackHeight = trackHeight.Value;
|
|
this.activeTrackColor = activeTrackColor;
|
|
this.inactiveTrackColor = inactiveTrackColor;
|
|
this.disabledActiveTrackColor = disabledActiveTrackColor;
|
|
this.disabledInactiveTrackColor = disabledInactiveTrackColor;
|
|
this.activeTickMarkColor = activeTickMarkColor;
|
|
this.inactiveTickMarkColor = inactiveTickMarkColor;
|
|
this.disabledActiveTickMarkColor = disabledActiveTickMarkColor;
|
|
this.disabledInactiveTickMarkColor = disabledInactiveTickMarkColor;
|
|
this.thumbColor = thumbColor;
|
|
this.disabledThumbColor = disabledThumbColor;
|
|
this.overlayColor = overlayColor;
|
|
this.valueIndicatorColor = valueIndicatorColor;
|
|
this.trackShape = trackShape;
|
|
this.tickMarkShape = tickMarkShape;
|
|
this.thumbShape = thumbShape;
|
|
this.overlayShape = overlayShape;
|
|
this.valueIndicatorShape = valueIndicatorShape;
|
|
this.showValueIndicator = showValueIndicator.Value;
|
|
this.valueIndicatorTextStyle = valueIndicatorTextStyle;
|
|
}
|
|
|
|
public static SliderThemeData fromPrimaryColors(
|
|
Color primaryColor = null,
|
|
Color primaryColorDark = null,
|
|
Color primaryColorLight = null,
|
|
TextStyle valueIndicatorTextStyle = null) {
|
|
D.assert(primaryColor != null);
|
|
D.assert(primaryColorDark != null);
|
|
D.assert(primaryColorLight != null);
|
|
D.assert(valueIndicatorTextStyle != null);
|
|
|
|
const int activeTrackAlpha = 0xff;
|
|
const int inactiveTrackAlpha = 0x3d; // 24% opacity
|
|
const int disabledActiveTrackAlpha = 0x52; // 32% opacity
|
|
const int disabledInactiveTrackAlpha = 0x1f; // 12% opacity
|
|
const int activeTickMarkAlpha = 0x8a; // 54% opacity
|
|
const int inactiveTickMarkAlpha = 0x8a; // 54% opacity
|
|
const int disabledActiveTickMarkAlpha = 0x1f; // 12% opacity
|
|
const int disabledInactiveTickMarkAlpha = 0x1f; // 12% opacity
|
|
const int thumbAlpha = 0xff;
|
|
const int disabledThumbAlpha = 0x52; // 32% opacity
|
|
const int valueIndicatorAlpha = 0xff;
|
|
|
|
const int overlayLightAlpha = 0x29;
|
|
|
|
return new SliderThemeData(
|
|
trackHeight: 2.0f,
|
|
activeTrackColor: primaryColor.withAlpha(activeTrackAlpha),
|
|
inactiveTrackColor: primaryColor.withAlpha(inactiveTrackAlpha),
|
|
disabledActiveTrackColor: primaryColorDark.withAlpha(disabledActiveTrackAlpha),
|
|
disabledInactiveTrackColor: primaryColorDark.withAlpha(disabledInactiveTrackAlpha),
|
|
activeTickMarkColor: primaryColorLight.withAlpha(activeTickMarkAlpha),
|
|
inactiveTickMarkColor: primaryColor.withAlpha(inactiveTickMarkAlpha),
|
|
disabledActiveTickMarkColor: primaryColorLight.withAlpha(disabledActiveTickMarkAlpha),
|
|
disabledInactiveTickMarkColor: primaryColorDark.withAlpha(disabledInactiveTickMarkAlpha),
|
|
thumbColor: primaryColor.withAlpha(thumbAlpha),
|
|
disabledThumbColor: primaryColorDark.withAlpha(disabledThumbAlpha),
|
|
overlayColor: primaryColor.withAlpha(overlayLightAlpha),
|
|
valueIndicatorColor: primaryColor.withAlpha(valueIndicatorAlpha),
|
|
trackShape: new RectangularSliderTrackShape(),
|
|
tickMarkShape: new RoundSliderTickMarkShape(),
|
|
thumbShape: new RoundSliderThumbShape(),
|
|
overlayShape: new RoundSliderOverlayShape(),
|
|
valueIndicatorShape: new PaddleSliderValueIndicatorShape(),
|
|
valueIndicatorTextStyle: valueIndicatorTextStyle,
|
|
showValueIndicator: ShowValueIndicator.onlyForDiscrete
|
|
);
|
|
}
|
|
|
|
public readonly float trackHeight;
|
|
|
|
public readonly Color activeTrackColor;
|
|
|
|
public readonly Color inactiveTrackColor;
|
|
|
|
public readonly Color disabledActiveTrackColor;
|
|
|
|
public readonly Color disabledInactiveTrackColor;
|
|
|
|
public readonly Color activeTickMarkColor;
|
|
|
|
public readonly Color inactiveTickMarkColor;
|
|
|
|
public readonly Color disabledActiveTickMarkColor;
|
|
|
|
public readonly Color disabledInactiveTickMarkColor;
|
|
|
|
public readonly Color thumbColor;
|
|
|
|
public readonly Color disabledThumbColor;
|
|
|
|
public readonly Color overlayColor;
|
|
|
|
public readonly Color valueIndicatorColor;
|
|
|
|
public readonly SliderTrackShape trackShape;
|
|
|
|
public readonly SliderTickMarkShape tickMarkShape;
|
|
|
|
public readonly SliderComponentShape overlayShape;
|
|
|
|
public readonly SliderComponentShape thumbShape;
|
|
|
|
public readonly SliderComponentShape valueIndicatorShape;
|
|
|
|
public readonly ShowValueIndicator showValueIndicator;
|
|
|
|
public readonly TextStyle valueIndicatorTextStyle;
|
|
|
|
public SliderThemeData copyWith(
|
|
float? trackHeight = null,
|
|
Color activeTrackColor = null,
|
|
Color inactiveTrackColor = null,
|
|
Color disabledActiveTrackColor = null,
|
|
Color disabledInactiveTrackColor = null,
|
|
Color activeTickMarkColor = null,
|
|
Color inactiveTickMarkColor = null,
|
|
Color disabledActiveTickMarkColor = null,
|
|
Color disabledInactiveTickMarkColor = null,
|
|
Color thumbColor = null,
|
|
Color disabledThumbColor = null,
|
|
Color overlayColor = null,
|
|
Color valueIndicatorColor = null,
|
|
SliderTrackShape trackShape = null,
|
|
SliderTickMarkShape tickMarkShape = null,
|
|
SliderComponentShape thumbShape = null,
|
|
SliderComponentShape overlayShape = null,
|
|
SliderComponentShape valueIndicatorShape = null,
|
|
ShowValueIndicator? showValueIndicator = null,
|
|
TextStyle valueIndicatorTextStyle = null
|
|
) {
|
|
return new SliderThemeData(
|
|
trackHeight: trackHeight ?? this.trackHeight,
|
|
activeTrackColor: activeTrackColor ?? this.activeTrackColor,
|
|
inactiveTrackColor: inactiveTrackColor ?? this.inactiveTrackColor,
|
|
disabledActiveTrackColor: disabledActiveTrackColor ?? this.disabledActiveTrackColor,
|
|
disabledInactiveTrackColor: disabledInactiveTrackColor ?? this.disabledInactiveTrackColor,
|
|
activeTickMarkColor: activeTickMarkColor ?? this.activeTickMarkColor,
|
|
inactiveTickMarkColor: inactiveTickMarkColor ?? this.inactiveTickMarkColor,
|
|
disabledActiveTickMarkColor: disabledActiveTickMarkColor ?? this.disabledActiveTickMarkColor,
|
|
disabledInactiveTickMarkColor: disabledInactiveTickMarkColor ?? this.disabledInactiveTickMarkColor,
|
|
thumbColor: thumbColor ?? this.thumbColor,
|
|
disabledThumbColor: disabledThumbColor ?? this.disabledThumbColor,
|
|
overlayColor: overlayColor ?? this.overlayColor,
|
|
valueIndicatorColor: valueIndicatorColor ?? this.valueIndicatorColor,
|
|
trackShape: trackShape ?? this.trackShape,
|
|
tickMarkShape: tickMarkShape ?? this.tickMarkShape,
|
|
thumbShape: thumbShape ?? this.thumbShape,
|
|
overlayShape: overlayShape ?? this.overlayShape,
|
|
valueIndicatorShape: valueIndicatorShape ?? this.valueIndicatorShape,
|
|
showValueIndicator: showValueIndicator ?? this.showValueIndicator,
|
|
valueIndicatorTextStyle: valueIndicatorTextStyle ?? this.valueIndicatorTextStyle
|
|
);
|
|
}
|
|
|
|
public static SliderThemeData lerp(SliderThemeData a, SliderThemeData b, float t) {
|
|
D.assert(a != null);
|
|
D.assert(b != null);
|
|
return new SliderThemeData(
|
|
trackHeight: Mathf.Lerp(a.trackHeight, b.trackHeight, t),
|
|
activeTrackColor: Color.lerp(a.activeTrackColor, b.activeTrackColor, t),
|
|
inactiveTrackColor: Color.lerp(a.inactiveTrackColor, b.inactiveTrackColor, t),
|
|
disabledActiveTrackColor: Color.lerp(a.disabledActiveTrackColor, b.disabledActiveTrackColor, t),
|
|
disabledInactiveTrackColor: Color.lerp(a.disabledInactiveTrackColor, b.disabledInactiveTrackColor, t),
|
|
activeTickMarkColor: Color.lerp(a.activeTickMarkColor, b.activeTickMarkColor, t),
|
|
inactiveTickMarkColor: Color.lerp(a.inactiveTickMarkColor, b.inactiveTickMarkColor, t),
|
|
disabledActiveTickMarkColor: Color.lerp(a.disabledActiveTickMarkColor, b.disabledActiveTickMarkColor,
|
|
t),
|
|
disabledInactiveTickMarkColor: Color.lerp(a.disabledInactiveTickMarkColor,
|
|
b.disabledInactiveTickMarkColor, t),
|
|
thumbColor: Color.lerp(a.thumbColor, b.thumbColor, t),
|
|
disabledThumbColor: Color.lerp(a.disabledThumbColor, b.disabledThumbColor, t),
|
|
overlayColor: Color.lerp(a.overlayColor, b.overlayColor, t),
|
|
valueIndicatorColor: Color.lerp(a.valueIndicatorColor, b.valueIndicatorColor, t),
|
|
trackShape: t < 0.5 ? a.trackShape : b.trackShape,
|
|
tickMarkShape: t < 0.5 ? a.tickMarkShape : b.tickMarkShape,
|
|
thumbShape: t < 0.5 ? a.thumbShape : b.thumbShape,
|
|
overlayShape: t < 0.5 ? a.overlayShape : b.overlayShape,
|
|
valueIndicatorShape: t < 0.5 ? a.valueIndicatorShape : b.valueIndicatorShape,
|
|
showValueIndicator: t < 0.5 ? a.showValueIndicator : b.showValueIndicator,
|
|
valueIndicatorTextStyle: TextStyle.lerp(a.valueIndicatorTextStyle, b.valueIndicatorTextStyle, t)
|
|
);
|
|
}
|
|
|
|
public bool Equals(SliderThemeData other) {
|
|
if (ReferenceEquals(null, other)) {
|
|
return false;
|
|
}
|
|
|
|
if (ReferenceEquals(this, other)) {
|
|
return true;
|
|
}
|
|
|
|
return other.trackHeight == trackHeight
|
|
&& other.activeTrackColor == activeTrackColor
|
|
&& other.inactiveTrackColor == inactiveTrackColor
|
|
&& other.disabledActiveTrackColor == disabledActiveTrackColor
|
|
&& other.disabledInactiveTrackColor == disabledInactiveTrackColor
|
|
&& other.activeTickMarkColor == activeTickMarkColor
|
|
&& other.inactiveTickMarkColor == inactiveTickMarkColor
|
|
&& other.disabledActiveTickMarkColor == disabledActiveTickMarkColor
|
|
&& other.disabledInactiveTickMarkColor == disabledInactiveTickMarkColor
|
|
&& other.thumbColor == thumbColor
|
|
&& other.disabledThumbColor == disabledThumbColor
|
|
&& other.overlayColor == overlayColor
|
|
&& other.valueIndicatorColor == valueIndicatorColor
|
|
&& other.trackShape == trackShape
|
|
&& other.tickMarkShape == tickMarkShape
|
|
&& other.thumbShape == thumbShape
|
|
&& other.overlayShape == overlayShape
|
|
&& other.valueIndicatorShape == valueIndicatorShape
|
|
&& other.showValueIndicator == showValueIndicator
|
|
&& other.valueIndicatorTextStyle == valueIndicatorTextStyle;
|
|
}
|
|
|
|
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((SliderThemeData) obj);
|
|
}
|
|
|
|
public static bool operator ==(SliderThemeData left, SliderThemeData right) {
|
|
return Equals(left, right);
|
|
}
|
|
|
|
public static bool operator !=(SliderThemeData left, SliderThemeData right) {
|
|
return !Equals(left, right);
|
|
}
|
|
|
|
int? _cachedHashCode = null;
|
|
|
|
public override int GetHashCode() {
|
|
if (_cachedHashCode != null) {
|
|
return _cachedHashCode.Value;
|
|
}
|
|
|
|
unchecked {
|
|
var hashCode = trackHeight.GetHashCode();
|
|
hashCode = (hashCode * 397) ^ activeTrackColor.GetHashCode();
|
|
hashCode = (hashCode * 397) ^ inactiveTrackColor.GetHashCode();
|
|
hashCode = (hashCode * 397) ^ disabledActiveTrackColor.GetHashCode();
|
|
hashCode = (hashCode * 397) ^ disabledInactiveTrackColor.GetHashCode();
|
|
hashCode = (hashCode * 397) ^ activeTickMarkColor.GetHashCode();
|
|
hashCode = (hashCode * 397) ^ inactiveTickMarkColor.GetHashCode();
|
|
hashCode = (hashCode * 397) ^ disabledActiveTickMarkColor.GetHashCode();
|
|
hashCode = (hashCode * 397) ^ disabledInactiveTickMarkColor.GetHashCode();
|
|
hashCode = (hashCode * 397) ^ thumbColor.GetHashCode();
|
|
hashCode = (hashCode * 397) ^ disabledThumbColor.GetHashCode();
|
|
hashCode = (hashCode * 397) ^ overlayColor.GetHashCode();
|
|
hashCode = (hashCode * 397) ^ valueIndicatorColor.GetHashCode();
|
|
hashCode = (hashCode * 397) ^ trackShape.GetHashCode();
|
|
hashCode = (hashCode * 397) ^ tickMarkShape.GetHashCode();
|
|
hashCode = (hashCode * 397) ^ thumbShape.GetHashCode();
|
|
hashCode = (hashCode * 397) ^ overlayShape.GetHashCode();
|
|
hashCode = (hashCode * 397) ^ valueIndicatorShape.GetHashCode();
|
|
hashCode = (hashCode * 397) ^ showValueIndicator.GetHashCode();
|
|
hashCode = (hashCode * 397) ^ valueIndicatorTextStyle.GetHashCode();
|
|
|
|
_cachedHashCode = hashCode;
|
|
return hashCode;
|
|
}
|
|
}
|
|
|
|
public override void debugFillProperties(DiagnosticPropertiesBuilder properties) {
|
|
base.debugFillProperties(properties);
|
|
ThemeData defaultTheme = ThemeData.fallback();
|
|
SliderThemeData defaultData = fromPrimaryColors(
|
|
primaryColor: defaultTheme.primaryColor,
|
|
primaryColorDark: defaultTheme.primaryColorDark,
|
|
primaryColorLight: defaultTheme.primaryColorLight,
|
|
valueIndicatorTextStyle: defaultTheme.accentTextTheme.body2
|
|
);
|
|
properties.add(new DiagnosticsProperty<Color>("activeTrackColor", activeTrackColor,
|
|
defaultValue: defaultData.activeTrackColor));
|
|
properties.add(new DiagnosticsProperty<Color>("activeTrackColor", activeTrackColor,
|
|
defaultValue: defaultData.activeTrackColor));
|
|
properties.add(new DiagnosticsProperty<Color>("inactiveTrackColor", inactiveTrackColor,
|
|
defaultValue: defaultData.inactiveTrackColor));
|
|
properties.add(new DiagnosticsProperty<Color>("disabledActiveTrackColor", disabledActiveTrackColor,
|
|
defaultValue: defaultData.disabledActiveTrackColor, level: DiagnosticLevel.debug));
|
|
properties.add(new DiagnosticsProperty<Color>("disabledInactiveTrackColor", disabledInactiveTrackColor,
|
|
defaultValue: defaultData.disabledInactiveTrackColor, level: DiagnosticLevel.debug));
|
|
properties.add(new DiagnosticsProperty<Color>("activeTickMarkColor", activeTickMarkColor,
|
|
defaultValue: defaultData.activeTickMarkColor, level: DiagnosticLevel.debug));
|
|
properties.add(new DiagnosticsProperty<Color>("inactiveTickMarkColor", inactiveTickMarkColor,
|
|
defaultValue: defaultData.inactiveTickMarkColor, level: DiagnosticLevel.debug));
|
|
properties.add(new DiagnosticsProperty<Color>("disabledActiveTickMarkColor",
|
|
disabledActiveTickMarkColor, defaultValue: defaultData.disabledActiveTickMarkColor,
|
|
level: DiagnosticLevel.debug));
|
|
properties.add(new DiagnosticsProperty<Color>("disabledInactiveTickMarkColor",
|
|
disabledInactiveTickMarkColor, defaultValue: defaultData.disabledInactiveTickMarkColor,
|
|
level: DiagnosticLevel.debug));
|
|
properties.add(new DiagnosticsProperty<Color>("thumbColor", thumbColor,
|
|
defaultValue: defaultData.thumbColor));
|
|
properties.add(new DiagnosticsProperty<Color>("disabledThumbColor", disabledThumbColor,
|
|
defaultValue: defaultData.disabledThumbColor, level: DiagnosticLevel.debug));
|
|
properties.add(new DiagnosticsProperty<Color>("overlayColor", overlayColor,
|
|
defaultValue: defaultData.overlayColor, level: DiagnosticLevel.debug));
|
|
properties.add(new DiagnosticsProperty<Color>("valueIndicatorColor", valueIndicatorColor,
|
|
defaultValue: defaultData.valueIndicatorColor));
|
|
properties.add(new DiagnosticsProperty<SliderTrackShape>("trackShape", trackShape,
|
|
defaultValue: defaultData.trackShape, level: DiagnosticLevel.debug));
|
|
properties.add(new DiagnosticsProperty<SliderTickMarkShape>("tickMarkShape", tickMarkShape,
|
|
defaultValue: defaultData.tickMarkShape, level: DiagnosticLevel.debug));
|
|
properties.add(new DiagnosticsProperty<SliderComponentShape>("thumbShape", thumbShape,
|
|
defaultValue: defaultData.thumbShape, level: DiagnosticLevel.debug));
|
|
properties.add(new DiagnosticsProperty<SliderComponentShape>("overlayShape", overlayShape,
|
|
defaultValue: defaultData.overlayShape, level: DiagnosticLevel.debug));
|
|
properties.add(new DiagnosticsProperty<SliderComponentShape>("valueIndicatorShape",
|
|
valueIndicatorShape, defaultValue: defaultData.valueIndicatorShape, level: DiagnosticLevel.debug));
|
|
properties.add(new EnumProperty<ShowValueIndicator>("showValueIndicator", showValueIndicator,
|
|
defaultValue: defaultData.showValueIndicator));
|
|
properties.add(new DiagnosticsProperty<TextStyle>("valueIndicatorTextStyle", valueIndicatorTextStyle,
|
|
defaultValue: defaultData.valueIndicatorTextStyle));
|
|
}
|
|
}
|
|
|
|
public abstract class SliderTrackShape {
|
|
public SliderTrackShape() {
|
|
}
|
|
|
|
public abstract Rect getPreferredRect(
|
|
RenderBox parentBox = null,
|
|
Offset offset = null,
|
|
SliderThemeData sliderTheme = null,
|
|
bool? isEnabled = null,
|
|
bool? isDiscrete = null);
|
|
|
|
public abstract void paint(
|
|
PaintingContext context,
|
|
Offset offset,
|
|
RenderBox parentBox = null,
|
|
SliderThemeData sliderTheme = null,
|
|
Animation<float> enableAnimation = null,
|
|
Offset thumbCenter = null,
|
|
bool? isEnabled = null,
|
|
bool? isDiscrete = null
|
|
);
|
|
}
|
|
|
|
public abstract class SliderTickMarkShape {
|
|
public SliderTickMarkShape() {
|
|
}
|
|
|
|
public abstract Size getPreferredSize(
|
|
SliderThemeData sliderTheme = null,
|
|
bool? isEnabled = null);
|
|
|
|
public abstract void paint(
|
|
PaintingContext context,
|
|
Offset offset,
|
|
RenderBox parentBox = null,
|
|
SliderThemeData sliderTheme = null,
|
|
Animation<float> enableAnimation = null,
|
|
Offset thumbCenter = null,
|
|
bool? isEnabled = null);
|
|
|
|
public static readonly SliderTickMarkShape noTickMark = new _EmptySliderTickMarkShape();
|
|
}
|
|
|
|
|
|
class _EmptySliderTickMarkShape : SliderTickMarkShape {
|
|
public override Size getPreferredSize(
|
|
SliderThemeData sliderTheme = null,
|
|
bool? isEnabled = null) {
|
|
return Size.zero;
|
|
}
|
|
|
|
public override void paint(
|
|
PaintingContext context,
|
|
Offset offset,
|
|
RenderBox parentBox = null,
|
|
SliderThemeData sliderTheme = null,
|
|
Animation<float> enableAnimation = null,
|
|
Offset thumbCenter = null,
|
|
bool? isEnabled = null) {
|
|
}
|
|
}
|
|
|
|
public abstract class SliderComponentShape {
|
|
public SliderComponentShape() {
|
|
}
|
|
|
|
public abstract Size getPreferredSize(
|
|
bool? isEnabled,
|
|
bool? isDiscrete);
|
|
|
|
public abstract void paint(
|
|
PaintingContext context,
|
|
Offset center,
|
|
Animation<float> activationAnimation = null,
|
|
Animation<float> enableAnimation = null,
|
|
bool? isDiscrete = null,
|
|
TextPainter labelPainter = null,
|
|
RenderBox parentBox = null,
|
|
SliderThemeData sliderTheme = null,
|
|
float? value = null);
|
|
|
|
public static readonly SliderComponentShape noThumb = new _EmptySliderComponentShape();
|
|
|
|
public static readonly SliderComponentShape noOverlay = new _EmptySliderComponentShape();
|
|
}
|
|
|
|
class _EmptySliderComponentShape : SliderComponentShape {
|
|
public override Size getPreferredSize(
|
|
bool? isEnabled,
|
|
bool? isDiscrete) {
|
|
return Size.zero;
|
|
}
|
|
|
|
public override void paint(
|
|
PaintingContext context,
|
|
Offset center,
|
|
Animation<float> activationAnimation = null,
|
|
Animation<float> enableAnimation = null,
|
|
bool? isDiscrete = null,
|
|
TextPainter labelPainter = null,
|
|
RenderBox parentBox = null,
|
|
SliderThemeData sliderTheme = null,
|
|
float? value = null) {
|
|
}
|
|
}
|
|
|
|
public class RectangularSliderTrackShape : SliderTrackShape {
|
|
public RectangularSliderTrackShape(
|
|
float disabledThumbGapWidth = 2.0f) {
|
|
this.disabledThumbGapWidth = disabledThumbGapWidth;
|
|
}
|
|
|
|
public readonly float disabledThumbGapWidth;
|
|
|
|
|
|
public override Rect getPreferredRect(
|
|
RenderBox parentBox = null,
|
|
Offset offset = null,
|
|
SliderThemeData sliderTheme = null,
|
|
bool? isEnabled = null,
|
|
bool? isDiscrete = null) {
|
|
float overlayWidth = sliderTheme.overlayShape.getPreferredSize(isEnabled, isDiscrete).width;
|
|
float trackHeight = sliderTheme.trackHeight;
|
|
D.assert(overlayWidth >= 0);
|
|
D.assert(trackHeight >= 0);
|
|
D.assert(parentBox.size.width >= overlayWidth);
|
|
D.assert(parentBox.size.height >= trackHeight);
|
|
|
|
float trackLeft = offset.dx + overlayWidth / 2f;
|
|
float trackTop = offset.dy + (parentBox.size.height - trackHeight) / 2f;
|
|
float trackWidth = parentBox.size.width - overlayWidth;
|
|
return Rect.fromLTWH(trackLeft, trackTop, trackWidth, trackHeight);
|
|
}
|
|
|
|
public override void paint(
|
|
PaintingContext context,
|
|
Offset offset,
|
|
RenderBox parentBox = null,
|
|
SliderThemeData sliderTheme = null,
|
|
Animation<float> enableAnimation = null,
|
|
Offset thumbCenter = null,
|
|
bool? isEnabled = null,
|
|
bool? isDiscrete = null) {
|
|
if (sliderTheme.trackHeight == 0) {
|
|
return;
|
|
}
|
|
|
|
ColorTween activeTrackColorTween = new ColorTween(begin: sliderTheme.disabledActiveTrackColor,
|
|
end: sliderTheme.activeTrackColor);
|
|
ColorTween inactiveTrackColorTween = new ColorTween(begin: sliderTheme.disabledInactiveTrackColor,
|
|
end: sliderTheme.inactiveTrackColor);
|
|
Paint activePaint = new Paint {color = activeTrackColorTween.evaluate(enableAnimation)};
|
|
Paint inactivePaint = new Paint {color = inactiveTrackColorTween.evaluate(enableAnimation)};
|
|
Paint leftTrackPaint = activePaint;
|
|
Paint rightTrackPaint = inactivePaint;
|
|
|
|
float horizontalAdjustment = 0.0f;
|
|
if (!isEnabled.Value) {
|
|
float disabledThumbRadius = sliderTheme.thumbShape.getPreferredSize(false, isDiscrete).width / 2.0f;
|
|
float gap = disabledThumbGapWidth * (1.0f - enableAnimation.value);
|
|
horizontalAdjustment = disabledThumbRadius + gap;
|
|
}
|
|
|
|
Rect trackRect = getPreferredRect(
|
|
parentBox: parentBox,
|
|
offset: offset,
|
|
sliderTheme: sliderTheme,
|
|
isEnabled: isEnabled,
|
|
isDiscrete: isDiscrete
|
|
);
|
|
|
|
Rect leftTrackSegment = Rect.fromLTRB(trackRect.left, trackRect.top, thumbCenter.dx - horizontalAdjustment,
|
|
trackRect.bottom);
|
|
context.canvas.drawRect(leftTrackSegment, leftTrackPaint);
|
|
|
|
Rect rightTrackSegment = Rect.fromLTRB(thumbCenter.dx + horizontalAdjustment, trackRect.top,
|
|
trackRect.right, trackRect.bottom);
|
|
context.canvas.drawRect(rightTrackSegment, rightTrackPaint);
|
|
}
|
|
}
|
|
|
|
public class RoundSliderTickMarkShape : SliderTickMarkShape {
|
|
public RoundSliderTickMarkShape(
|
|
float? tickMarkRadius = null) {
|
|
this.tickMarkRadius = tickMarkRadius;
|
|
}
|
|
|
|
public readonly float? tickMarkRadius;
|
|
|
|
|
|
public override Size getPreferredSize(
|
|
SliderThemeData sliderTheme = null,
|
|
bool? isEnabled = null
|
|
) {
|
|
return Size.fromRadius(tickMarkRadius ?? sliderTheme.trackHeight / 2f);
|
|
}
|
|
|
|
|
|
public override void paint(
|
|
PaintingContext context,
|
|
Offset center,
|
|
RenderBox parentBox = null,
|
|
SliderThemeData sliderTheme = null,
|
|
Animation<float> enableAnimation = null,
|
|
Offset thumbCenter = null,
|
|
bool? isEnabled = null
|
|
) {
|
|
Color begin;
|
|
Color end;
|
|
|
|
bool isTickMarkRightOfThumb = center.dx > thumbCenter.dx;
|
|
begin = isTickMarkRightOfThumb
|
|
? sliderTheme.disabledInactiveTickMarkColor
|
|
: sliderTheme.disabledActiveTickMarkColor;
|
|
end = isTickMarkRightOfThumb ? sliderTheme.inactiveTickMarkColor : sliderTheme.activeTickMarkColor;
|
|
|
|
Paint paint = new Paint {color = new ColorTween(begin: begin, end: end).evaluate(enableAnimation)};
|
|
|
|
float tickMarkRadius = getPreferredSize(
|
|
isEnabled: isEnabled,
|
|
sliderTheme: sliderTheme
|
|
).width / 2f;
|
|
context.canvas.drawCircle(center, tickMarkRadius, paint);
|
|
}
|
|
}
|
|
|
|
public class RoundSliderThumbShape : SliderComponentShape {
|
|
public RoundSliderThumbShape(
|
|
float enabledThumbRadius = 6.0f,
|
|
float? disabledThumbRadius = null
|
|
) {
|
|
this.enabledThumbRadius = enabledThumbRadius;
|
|
this.disabledThumbRadius = disabledThumbRadius;
|
|
}
|
|
|
|
public readonly float enabledThumbRadius;
|
|
|
|
public readonly float? disabledThumbRadius;
|
|
|
|
float _disabledThumbRadius {
|
|
get { return disabledThumbRadius ?? enabledThumbRadius * 2f / 3f; }
|
|
}
|
|
|
|
|
|
public override Size getPreferredSize(bool? isEnabled, bool? isDiscrete) {
|
|
return Size.fromRadius(isEnabled.Value ? enabledThumbRadius : _disabledThumbRadius);
|
|
}
|
|
|
|
|
|
public override void paint(
|
|
PaintingContext context,
|
|
Offset center,
|
|
Animation<float> activationAnimation = null,
|
|
Animation<float> enableAnimation = null,
|
|
bool? isDiscrete = null,
|
|
TextPainter labelPainter = null,
|
|
RenderBox parentBox = null,
|
|
SliderThemeData sliderTheme = null,
|
|
float? value = null
|
|
) {
|
|
Canvas canvas = context.canvas;
|
|
FloatTween radiusTween = new FloatTween(
|
|
begin: _disabledThumbRadius,
|
|
end: enabledThumbRadius
|
|
);
|
|
ColorTween colorTween = new ColorTween(
|
|
begin: sliderTheme.disabledThumbColor,
|
|
end: sliderTheme.thumbColor
|
|
);
|
|
canvas.drawCircle(
|
|
center,
|
|
radiusTween.evaluate(enableAnimation),
|
|
new Paint {color = colorTween.evaluate(enableAnimation)}
|
|
);
|
|
}
|
|
}
|
|
|
|
public class RoundSliderOverlayShape : SliderComponentShape {
|
|
public RoundSliderOverlayShape(
|
|
float overlayRadius = 16.0f) {
|
|
this.overlayRadius = overlayRadius;
|
|
}
|
|
|
|
public readonly float overlayRadius;
|
|
|
|
public override Size getPreferredSize(bool? isEnabled, bool? isDiscrete) {
|
|
return Size.fromRadius(overlayRadius);
|
|
}
|
|
|
|
public override void paint(
|
|
PaintingContext context,
|
|
Offset center,
|
|
Animation<float> activationAnimation = null,
|
|
Animation<float> enableAnimation = null,
|
|
bool? isDiscrete = null,
|
|
TextPainter labelPainter = null,
|
|
RenderBox parentBox = null,
|
|
SliderThemeData sliderTheme = null,
|
|
float? value = null
|
|
) {
|
|
Canvas canvas = context.canvas;
|
|
FloatTween radiusTween = new FloatTween(
|
|
begin: 0.0f,
|
|
end: overlayRadius
|
|
);
|
|
canvas.drawCircle(
|
|
center,
|
|
radiusTween.evaluate(activationAnimation),
|
|
new Paint {color = sliderTheme.overlayColor}
|
|
);
|
|
}
|
|
}
|
|
|
|
public class PaddleSliderValueIndicatorShape : SliderComponentShape {
|
|
public PaddleSliderValueIndicatorShape() {
|
|
}
|
|
|
|
const float _topLobeRadius = 16.0f;
|
|
const float _labelTextDesignSize = 14.0f;
|
|
const float _bottomLobeRadius = 6.0f;
|
|
const float _bottomLobeStartAngle = -1.1f * Mathf.PI / 4.0f;
|
|
const float _bottomLobeEndAngle = 1.1f * 5f * Mathf.PI / 4.0f;
|
|
const float _labelPadding = 8.0f;
|
|
const float _distanceBetweenTopBottomCenters = 40.0f;
|
|
static readonly Offset _topLobeCenter = new Offset(0.0f, -_distanceBetweenTopBottomCenters);
|
|
const float _topNeckRadius = 14.0f;
|
|
|
|
const float _neckTriangleHypotenuse = _topLobeRadius + _topNeckRadius;
|
|
|
|
const float _twoSeventyDegrees = 3.0f * Mathf.PI / 2.0f;
|
|
const float _ninetyDegrees = Mathf.PI / 2.0f;
|
|
const float _thirtyDegrees = Mathf.PI / 6.0f;
|
|
|
|
static readonly Size _preferredSize =
|
|
Size.fromHeight(_distanceBetweenTopBottomCenters + _topLobeRadius + _bottomLobeRadius);
|
|
|
|
const bool _debuggingLabelLocation = false;
|
|
|
|
static Path _bottomLobePath;
|
|
static Offset _bottomLobeEnd;
|
|
|
|
|
|
public override Size getPreferredSize(
|
|
bool? isEnabled,
|
|
bool? isDiscrete) {
|
|
return _preferredSize;
|
|
}
|
|
|
|
static void _addArc(Path path, Offset center, float radius, float startAngle, float endAngle) {
|
|
Rect arcRect = Rect.fromCircle(center: center, radius: radius);
|
|
path.arcTo(arcRect, startAngle, endAngle - startAngle, false);
|
|
}
|
|
|
|
static void _generateBottomLobe() {
|
|
const float bottomNeckRadius = 4.5f;
|
|
const float bottomNeckStartAngle = _bottomLobeEndAngle - Mathf.PI;
|
|
const float bottomNeckEndAngle = 0.0f;
|
|
|
|
Path path = new Path();
|
|
Offset bottomKnobStart = new Offset(
|
|
_bottomLobeRadius * Mathf.Cos(_bottomLobeStartAngle),
|
|
_bottomLobeRadius * Mathf.Sin(_bottomLobeStartAngle)
|
|
);
|
|
Offset bottomNeckRightCenter = bottomKnobStart +
|
|
new Offset(
|
|
bottomNeckRadius * Mathf.Cos(bottomNeckStartAngle),
|
|
-bottomNeckRadius * Mathf.Sin(bottomNeckStartAngle)
|
|
);
|
|
Offset bottomNeckLeftCenter = new Offset(
|
|
-bottomNeckRightCenter.dx,
|
|
bottomNeckRightCenter.dy
|
|
);
|
|
|
|
Offset bottomNeckStartRight = new Offset(
|
|
bottomNeckRightCenter.dx - bottomNeckRadius,
|
|
bottomNeckRightCenter.dy
|
|
);
|
|
|
|
path.moveTo(bottomNeckStartRight.dx, bottomNeckStartRight.dy);
|
|
_addArc(
|
|
path,
|
|
bottomNeckRightCenter,
|
|
bottomNeckRadius,
|
|
Mathf.PI - bottomNeckEndAngle,
|
|
Mathf.PI - bottomNeckStartAngle
|
|
);
|
|
_addArc(
|
|
path,
|
|
Offset.zero,
|
|
_bottomLobeRadius,
|
|
_bottomLobeStartAngle,
|
|
_bottomLobeEndAngle
|
|
);
|
|
_addArc(
|
|
path,
|
|
bottomNeckLeftCenter,
|
|
bottomNeckRadius,
|
|
bottomNeckStartAngle,
|
|
bottomNeckEndAngle
|
|
);
|
|
|
|
_bottomLobeEnd = new Offset(
|
|
-bottomNeckStartRight.dx,
|
|
bottomNeckStartRight.dy
|
|
);
|
|
|
|
_bottomLobePath = path;
|
|
}
|
|
|
|
|
|
Offset _addBottomLobe(Path path) {
|
|
if (_bottomLobePath == null || _bottomLobeEnd == null) {
|
|
_generateBottomLobe();
|
|
}
|
|
|
|
path.addPath(_bottomLobePath, Offset.zero);
|
|
return _bottomLobeEnd;
|
|
}
|
|
|
|
float _getIdealOffset(
|
|
RenderBox parentBox,
|
|
float halfWidthNeeded,
|
|
float scale,
|
|
Offset center
|
|
) {
|
|
const float edgeMargin = 4.0f;
|
|
Rect topLobeRect = Rect.fromLTWH(
|
|
-_topLobeRadius - halfWidthNeeded,
|
|
-_topLobeRadius - _distanceBetweenTopBottomCenters,
|
|
2.0f * (_topLobeRadius + halfWidthNeeded),
|
|
2.0f * _topLobeRadius
|
|
);
|
|
|
|
Offset topLeft = (topLobeRect.topLeft * scale) + center;
|
|
Offset bottomRight = (topLobeRect.bottomRight * scale) + center;
|
|
float shift = 0.0f;
|
|
if (topLeft.dx < edgeMargin) {
|
|
shift = edgeMargin - topLeft.dx;
|
|
}
|
|
|
|
if (bottomRight.dx > parentBox.size.width - edgeMargin) {
|
|
shift = parentBox.size.width - bottomRight.dx - edgeMargin;
|
|
}
|
|
|
|
shift = scale == 0.0f ? 0.0f : shift / scale;
|
|
return shift;
|
|
}
|
|
|
|
void _drawValueIndicator(
|
|
RenderBox parentBox,
|
|
Canvas canvas,
|
|
Offset center,
|
|
Paint paint,
|
|
float scale,
|
|
TextPainter labelPainter
|
|
) {
|
|
canvas.save();
|
|
canvas.translate(center.dx, center.dy);
|
|
|
|
float textScaleFactor = labelPainter.height / _labelTextDesignSize;
|
|
float overallScale = scale * textScaleFactor;
|
|
canvas.scale(overallScale, overallScale);
|
|
float inverseTextScale = textScaleFactor != 0 ? 1.0f / textScaleFactor : 0.0f;
|
|
float labelHalfWidth = labelPainter.width / 2.0f;
|
|
|
|
float halfWidthNeeded = Mathf.Max(
|
|
0.0f,
|
|
inverseTextScale * labelHalfWidth - (_topLobeRadius - _labelPadding)
|
|
);
|
|
|
|
float shift = _getIdealOffset(parentBox, halfWidthNeeded, overallScale, center);
|
|
float leftWidthNeeded;
|
|
float rightWidthNeeded;
|
|
if (shift < 0.0) {
|
|
shift = Mathf.Max(shift, -halfWidthNeeded);
|
|
}
|
|
else {
|
|
shift = Mathf.Min(shift, halfWidthNeeded);
|
|
}
|
|
|
|
rightWidthNeeded = halfWidthNeeded + shift;
|
|
leftWidthNeeded = halfWidthNeeded - shift;
|
|
|
|
Path path = new Path();
|
|
Offset bottomLobeEnd = _addBottomLobe(path);
|
|
|
|
float neckTriangleBase = _topNeckRadius - bottomLobeEnd.dx;
|
|
|
|
float leftAmount = Mathf.Max(0.0f, Mathf.Min(1.0f, leftWidthNeeded / neckTriangleBase));
|
|
float rightAmount = Mathf.Max(0.0f, Mathf.Min(1.0f, rightWidthNeeded / neckTriangleBase));
|
|
|
|
float leftTheta = (1.0f - leftAmount) * _thirtyDegrees;
|
|
float rightTheta = (1.0f - rightAmount) * _thirtyDegrees;
|
|
Offset neckLeftCenter = new Offset(
|
|
-neckTriangleBase,
|
|
_topLobeCenter.dy + Mathf.Cos(leftTheta) * _neckTriangleHypotenuse
|
|
);
|
|
Offset neckRightCenter = new Offset(
|
|
neckTriangleBase,
|
|
_topLobeCenter.dy + Mathf.Cos(rightTheta) * _neckTriangleHypotenuse
|
|
);
|
|
|
|
float leftNeckArcAngle = _ninetyDegrees - leftTheta;
|
|
float rightNeckArcAngle = Mathf.PI + _ninetyDegrees - rightTheta;
|
|
|
|
float neckStretchBaseline = bottomLobeEnd.dy - Mathf.Max(neckLeftCenter.dy, neckRightCenter.dy);
|
|
float t = Mathf.Pow(inverseTextScale, 3.0f);
|
|
float stretch = (neckStretchBaseline * t).clamp(0.0f, 10.0f * neckStretchBaseline);
|
|
Offset neckStretch = new Offset(0.0f, neckStretchBaseline - stretch);
|
|
|
|
D.assert(() => {
|
|
if (!_debuggingLabelLocation) {
|
|
return true;
|
|
}
|
|
|
|
#pragma warning disable 0162
|
|
Offset leftCenter = _topLobeCenter - new Offset(leftWidthNeeded, 0.0f) + neckStretch;
|
|
Offset rightCenter = _topLobeCenter + new Offset(rightWidthNeeded, 0.0f) + neckStretch;
|
|
Rect valueRect = Rect.fromLTRB(
|
|
leftCenter.dx - _topLobeRadius,
|
|
leftCenter.dy - _topLobeRadius,
|
|
rightCenter.dx + _topLobeRadius,
|
|
rightCenter.dy + _topLobeRadius
|
|
);
|
|
Paint outlinePaint = new Paint();
|
|
outlinePaint.color = new Color(0xffff0000);
|
|
outlinePaint.style = PaintingStyle.stroke;
|
|
outlinePaint.strokeWidth = 1.0f;
|
|
canvas.drawRect(valueRect, outlinePaint);
|
|
return true;
|
|
#pragma warning restore 0162
|
|
});
|
|
|
|
_addArc(
|
|
path,
|
|
neckLeftCenter + neckStretch,
|
|
_topNeckRadius,
|
|
0.0f,
|
|
-leftNeckArcAngle
|
|
);
|
|
_addArc(
|
|
path,
|
|
_topLobeCenter - new Offset(leftWidthNeeded, 0.0f) + neckStretch,
|
|
_topLobeRadius,
|
|
_ninetyDegrees + leftTheta,
|
|
_twoSeventyDegrees
|
|
);
|
|
_addArc(
|
|
path,
|
|
_topLobeCenter + new Offset(rightWidthNeeded, 0.0f) + neckStretch,
|
|
_topLobeRadius,
|
|
_twoSeventyDegrees,
|
|
_twoSeventyDegrees + Mathf.PI - rightTheta
|
|
);
|
|
_addArc(
|
|
path,
|
|
neckRightCenter + neckStretch,
|
|
_topNeckRadius,
|
|
rightNeckArcAngle,
|
|
Mathf.PI
|
|
);
|
|
canvas.drawPath(path, paint);
|
|
|
|
canvas.save();
|
|
canvas.translate(shift, -_distanceBetweenTopBottomCenters + neckStretch.dy);
|
|
canvas.scale(inverseTextScale, inverseTextScale);
|
|
labelPainter.paint(canvas, Offset.zero - new Offset(labelHalfWidth, labelPainter.height / 2.0f));
|
|
canvas.restore();
|
|
canvas.restore();
|
|
}
|
|
|
|
public override void paint(
|
|
PaintingContext context,
|
|
Offset center,
|
|
Animation<float> activationAnimation = null,
|
|
Animation<float> enableAnimation = null,
|
|
bool? isDiscrete = null,
|
|
TextPainter labelPainter = null,
|
|
RenderBox parentBox = null,
|
|
SliderThemeData sliderTheme = null,
|
|
float? value = null) {
|
|
ColorTween enableColor = new ColorTween(
|
|
begin: sliderTheme.disabledThumbColor,
|
|
end: sliderTheme.valueIndicatorColor
|
|
);
|
|
_drawValueIndicator(
|
|
parentBox,
|
|
context.canvas,
|
|
center,
|
|
new Paint {color = enableColor.evaluate(enableAnimation)},
|
|
activationAnimation.value,
|
|
labelPainter
|
|
);
|
|
}
|
|
}
|
|
}
|