您最多选择25个主题 主题必须以中文或者字母或数字开头,可以包含连字符 (-),并且长度不得超过35个字符
 
 
 
 
 
 

760 行
27 KiB

using System;
using System.Collections.Generic;
using Unity.UIWidgets.animation;
using Unity.UIWidgets.async;
using Unity.UIWidgets.foundation;
using Unity.UIWidgets.gestures;
using Unity.UIWidgets.painting;
using Unity.UIWidgets.rendering;
using Unity.UIWidgets.ui;
using Unity.UIWidgets.widgets;
using UnityEngine;
using Color = Unity.UIWidgets.ui.Color;
using Rect = Unity.UIWidgets.ui.Rect;
using TextStyle = Unity.UIWidgets.painting.TextStyle;
namespace Unity.UIWidgets.material {
public partial class material_ {
internal static readonly TimeSpan _kMenuDuration = new TimeSpan(0, 0, 0, 0, 300);
internal const float _kMenuCloseIntervalEnd = 2.0f / 3.0f;
internal const float _kMenuHorizontalPadding = 16.0f;
internal const float _kMenuDividerHeight = 16.0f;
internal const float _kMenuMaxWidth = 5.0f * _kMenuWidthStep;
internal const float _kMenuMinWidth = 2.0f * _kMenuWidthStep;
internal const float _kMenuVerticalPadding = 8.0f;
internal const float _kMenuWidthStep = 56.0f;
internal const float _kMenuScreenPadding = 8.0f;
}
public abstract class PopupMenuEntry<T> : StatefulWidget {
protected PopupMenuEntry(Key key = null) : base(key: key) {
}
public abstract float height { get; }
public abstract bool represents(T value);
}
public class PopupMenuDivider<T> : PopupMenuEntry<T> {
public PopupMenuDivider(Key key = null, float height = material_._kMenuDividerHeight) : base(key: key) {
_height = height;
}
readonly float _height;
public override float height {
get { return _height; }
}
public override bool represents(T value) {
return false;
}
public override State createState() {
return new _PopupMenuDividerState<T>();
}
}
class _PopupMenuDividerState<T> : State<PopupMenuDivider<T>> {
public override Widget build(BuildContext context) {
return new Divider(height: widget.height);
}
}
class _MenuItem : SingleChildRenderObjectWidget {
internal _MenuItem(
Key key = null,
ValueChanged<Size> onLayout = null,
Widget child = null
) : base(key: key, child: child) {
this.onLayout = onLayout;
}
public readonly ValueChanged<Size> onLayout;
public override RenderObject createRenderObject(BuildContext context) {
return new _RenderMenuItemDuplicated(onLayout);
}
public override void updateRenderObject(BuildContext context, RenderObject renderObject) {
if (renderObject is _RenderMenuItemDuplicated renderMenuItemDuplicated) {
renderMenuItemDuplicated.onLayout = onLayout;
}
}
}
class _RenderMenuItemDuplicated : RenderShiftedBox {
internal _RenderMenuItemDuplicated(
ValueChanged<Size> onLayout,
RenderBox child = null
) : base(child) {
D.assert(onLayout != null);
this.onLayout = onLayout;
}
public ValueChanged<Size> onLayout;
protected override void performLayout() {
if (child == null) {
size = Size.zero;
}
else {
child.layout(constraints, parentUsesSize: true);
size = constraints.constrain(child.size);
}
BoxParentData childParentData = child.parentData as BoxParentData;
childParentData.offset = Offset.zero;
onLayout(size);
}
}
public class PopupMenuItem<T> : PopupMenuEntry<T> {
public PopupMenuItem(
Key key = null,
T value = default,
bool enabled = true,
float height = material_.kMinInteractiveDimension,
TextStyle textStyle = null,
Widget child = null
) : base(key: key) {
this.value = value;
this.enabled = enabled;
_height = height;
this.textStyle = textStyle;
this.child = child;
}
public readonly T value;
public readonly bool enabled;
readonly float _height;
public readonly TextStyle textStyle;
public override float height {
get { return _height; }
}
public readonly Widget child;
public override bool represents(T value) {
return Equals(value, this.value);
}
public override State createState() {
return new PopupMenuItemState<T, PopupMenuItem<T>>();
}
}
public class PopupMenuItemState<T, W> : State<W> where W : PopupMenuItem<T> {
protected virtual Widget buildChild() {
return widget.child;
}
protected virtual void handleTap() {
Navigator.pop(context, widget.value);
}
public override Widget build(BuildContext context) {
ThemeData theme = Theme.of(context);
PopupMenuThemeData popupMenuTheme = PopupMenuTheme.of(context);
TextStyle style = widget.textStyle ?? popupMenuTheme.textStyle ?? theme.textTheme.subtitle1;
if (!widget.enabled) {
style = style.copyWith(color: theme.disabledColor);
}
Widget item = new AnimatedDefaultTextStyle(
style: style,
duration: material_.kThemeChangeDuration,
child: new Container(
alignment: AlignmentDirectional.centerStart,
constraints: new BoxConstraints(minHeight: widget.height),
padding: EdgeInsets.symmetric(horizontal: material_._kMenuHorizontalPadding),
child: buildChild()
)
);
if (!widget.enabled) {
bool isDark = theme.brightness == Brightness.dark;
item = IconTheme.merge(
data: new IconThemeData(opacity: isDark ? 0.5f : 0.38f),
child: item
);
}
return new InkWell(
onTap: widget.enabled ? handleTap : (GestureTapCallback) null,
canRequestFocus: widget.enabled,
child: item
);
}
}
public class PopupMenuItemSingleTickerProviderState<T, W> : SingleTickerProviderStateMixin<W>
where W : PopupMenuItem<T> {
protected virtual Widget buildChild() {
return widget.child;
}
protected virtual void handleTap() {
Navigator.pop(context, widget.value);
}
public override Widget build(BuildContext context) {
ThemeData theme = Theme.of(context);
PopupMenuThemeData popupMenuTheme = PopupMenuTheme.of(context);
TextStyle style = widget.textStyle ?? popupMenuTheme.textStyle ?? theme.textTheme.subtitle1;
if (!widget.enabled) {
style = style.copyWith(color: theme.disabledColor);
}
Widget item = new AnimatedDefaultTextStyle(
style: style,
duration: material_.kThemeChangeDuration,
child: new Container(
alignment: AlignmentDirectional.centerStart,
constraints: new BoxConstraints(minHeight: widget.height),
padding: EdgeInsets.symmetric(horizontal: material_._kMenuHorizontalPadding),
child: buildChild()
)
);
if (!widget.enabled) {
bool isDark = theme.brightness == Brightness.dark;
item = IconTheme.merge(
data: new IconThemeData(opacity: isDark ? 0.5f : 0.38f),
child: item
);
}
return new InkWell(
onTap: widget.enabled ? handleTap : (GestureTapCallback) null,
canRequestFocus: widget.enabled,
child: item
);
}
}
public class CheckedPopupMenuItem<T> : PopupMenuItem<T> {
public CheckedPopupMenuItem(
Key key = null,
T value = default,
bool isChecked = false,
bool enabled = true,
Widget child = null
) : base(
key: key,
value: value,
enabled: enabled,
child: child
) {
this.isChecked = isChecked;
}
public readonly bool isChecked;
public override State createState() {
return new _CheckedPopupMenuItemState<T>();
}
}
class _CheckedPopupMenuItemState<T> : PopupMenuItemSingleTickerProviderState<T, CheckedPopupMenuItem<T>> {
static readonly TimeSpan _fadeDuration = new TimeSpan(0, 0, 0, 0, 150);
AnimationController _controller;
Animation<float> _opacity {
get { return _controller.view; }
}
public override void initState() {
base.initState();
_controller = new AnimationController(duration: _fadeDuration, vsync: this);
_controller.setValue(widget.isChecked ? 1.0f : 0.0f);
_controller.addListener(() => setState(() => {
/* animation changed */
}));
}
protected override void handleTap() {
if (widget.isChecked) {
_controller.reverse();
}
else {
_controller.forward();
}
base.handleTap();
}
protected override Widget buildChild() {
return new ListTile(
enabled: widget.enabled,
leading: new FadeTransition(
opacity: _opacity,
child: new Icon(_controller.isDismissed ? null : Icons.done)
),
title: widget.child
);
}
}
class _PopupMenu<T> : StatelessWidget {
public _PopupMenu(
Key key = null,
_PopupMenuRoute<T> route = null
) : base(key: key) {
this.route = route;
}
public readonly _PopupMenuRoute<T> route;
public override Widget build(BuildContext context) {
float unit = 1.0f / (route.items.Count + 1.5f);
List<Widget> children = new List<Widget>();
PopupMenuThemeData popupMenuTheme = PopupMenuTheme.of(context);
for (int i = 0; i < route.items.Count; i += 1) {
int index = i;
float start = (index + 1) * unit;
float end = (start + 1.5f * unit).clamp(0.0f, 1.0f);
CurvedAnimation opacityCurvedAnimation = new CurvedAnimation(
parent: route.animation,
curve: new Interval(start, end)
);
Widget item = route.items[index];
if (route.initialValue != null && route.items[index].represents((T) route.initialValue)) {
item = new Container(
color: Theme.of(context).highlightColor,
child: item
);
}
children.Add(
new _MenuItem(
onLayout: (Size size) => { route.itemSizes[index] = size; },
child: new FadeTransition(
opacity: opacityCurvedAnimation,
child: item
)
)
);
}
CurveTween opacity = new CurveTween(curve: new Interval(0.0f, 1.0f / 3.0f));
CurveTween width = new CurveTween(curve: new Interval(0.0f, unit));
CurveTween height = new CurveTween(curve: new Interval(0.0f, unit * route.items.Count));
Widget child = new ConstrainedBox(
constraints: new BoxConstraints(
minWidth: material_._kMenuMinWidth,
maxWidth: material_._kMenuMaxWidth
),
child: new IntrinsicWidth(
stepWidth: material_._kMenuWidthStep,
child: new SingleChildScrollView(
padding: EdgeInsets.symmetric(
vertical: material_._kMenuVerticalPadding
),
child: new ListBody(children: children)
)
)
);
return new AnimatedBuilder(
animation: route.animation,
builder: (_, builderChild) => {
return new Opacity(
opacity: opacity.evaluate(route.animation),
child: new Material(
shape: route.shape ?? popupMenuTheme.shape,
color: route.color ?? popupMenuTheme.color,
type: MaterialType.card,
elevation: route.elevation ?? popupMenuTheme.elevation ?? 8.0f,
child: new Align(
alignment: Alignment.topRight,
widthFactor: width.evaluate(route.animation),
heightFactor: height.evaluate(route.animation),
child: builderChild
)
)
);
},
child: child
);
}
}
class _PopupMenuRouteLayout : SingleChildLayoutDelegate {
public _PopupMenuRouteLayout(RelativeRect position, List<Size> itemSizes, int selectedItemIndex,
TextDirection? textDirection) {
this.position = position;
this.itemSizes = itemSizes;
this.selectedItemIndex = selectedItemIndex;
this.textDirection = textDirection;
}
public readonly RelativeRect position;
public List<Size> itemSizes;
public readonly int selectedItemIndex;
public readonly TextDirection? textDirection;
public override BoxConstraints getConstraintsForChild(BoxConstraints constraints) {
return BoxConstraints.loose(
constraints.biggest - new Offset(material_._kMenuScreenPadding * 2.0f,
material_._kMenuScreenPadding * 2.0f)
);
}
public override Offset getPositionForChild(Size size, Size childSize) {
float y = position.top;
if (itemSizes != null) {
float selectedItemOffset = material_._kMenuVerticalPadding;
for (int index = 0; index < selectedItemIndex; index += 1)
selectedItemOffset += itemSizes[index].height;
selectedItemOffset += itemSizes[selectedItemIndex].height / 2;
y = position.top + (size.height - position.top - position.bottom) / 2.0f - selectedItemOffset;
}
float x = 0;
if (position.left > position.right) {
x = size.width - position.right - childSize.width;
}
else if (position.left < position.right) {
x = position.left;
}
else {
D.assert(textDirection != null);
switch (textDirection) {
case TextDirection.rtl:
x = size.width - position.right - childSize.width;
break;
case TextDirection.ltr:
x = position.left;
break;
}
}
if (x < material_._kMenuScreenPadding) {
x = material_._kMenuScreenPadding;
}
else if (x + childSize.width > size.width - material_._kMenuScreenPadding) {
x = size.width - childSize.width - material_._kMenuScreenPadding;
}
if (y < material_._kMenuScreenPadding) {
y = material_._kMenuScreenPadding;
}
else if (y + childSize.height > size.height - material_._kMenuScreenPadding) {
y = size.height - childSize.height - material_._kMenuScreenPadding;
}
return new Offset(x, y);
}
public override bool shouldRelayout(SingleChildLayoutDelegate oldDelegate) {
if (oldDelegate is _PopupMenuRouteLayout popupMenu) {
D.assert(itemSizes.Count == popupMenu.itemSizes.Count);
return position != popupMenu.position
|| selectedItemIndex != popupMenu.selectedItemIndex
|| textDirection != popupMenu.textDirection
|| !itemSizes.equalsList(popupMenu.itemSizes);
}
else {
return false;
}
}
}
class _PopupMenuRoute<T> : PopupRoute {
public _PopupMenuRoute(
RelativeRect position = null,
List<PopupMenuEntry<T>> items = null,
T initialValue = default,
float? elevation = 8.0f,
ThemeData theme = null,
PopupMenuThemeData popupMenuTheme = null,
string barrierLabel = null,
ShapeBorder shape = null,
Color color = null,
BuildContext showMenuContext = null,
bool? captureInheritedThemes = null
) {
this.position = position;
this.items = items;
this.initialValue = initialValue;
this.elevation = elevation;
this.theme = theme;
this.popupMenuTheme = popupMenuTheme;
this.barrierLabel = barrierLabel;
this.shape = shape;
this.color = color;
this.showMenuContext = showMenuContext;
this.captureInheritedThemes = captureInheritedThemes;
itemSizes = new List<Size>(new Size[items.Count]);
}
public readonly RelativeRect position;
public readonly List<PopupMenuEntry<T>> items;
public readonly List<Size> itemSizes;
public readonly T initialValue;
public readonly float? elevation;
public readonly ThemeData theme;
public readonly ShapeBorder shape;
public readonly Color color;
public readonly PopupMenuThemeData popupMenuTheme;
public readonly BuildContext showMenuContext;
public readonly bool? captureInheritedThemes;
public override Animation<float> createAnimation() {
return new CurvedAnimation(
parent: base.createAnimation(),
curve: Curves.linear,
reverseCurve: new Interval(0.0f, material_._kMenuCloseIntervalEnd)
);
}
public override TimeSpan transitionDuration {
get { return material_._kMenuDuration; }
}
public override bool barrierDismissible {
get { return true; }
}
public override Color barrierColor {
get { return null; }
}
public override string barrierLabel { get; }
public override Widget buildPage(BuildContext context, Animation<float> animation,
Animation<float> secondaryAnimation) {
int? selectedItemIndex = null;
if (initialValue != null) {
for (int index = 0; selectedItemIndex == null && index < items.Count; index += 1) {
if (items[index].represents(initialValue))
selectedItemIndex = index;
}
}
Widget menu = new _PopupMenu<T>(route: this);
if (captureInheritedThemes ?? false) {
menu = InheritedTheme.captureAll(showMenuContext, menu);
}
else {
if (theme != null)
menu = new Theme(data: theme, child: menu);
}
return MediaQuery.removePadding(
context: context,
removeTop: true,
removeBottom: true,
removeLeft: true,
removeRight: true,
child: new Builder(
builder: _ => new CustomSingleChildLayout(
layoutDelegate: new _PopupMenuRouteLayout(
position,
itemSizes,
selectedItemIndex ?? 0,
Directionality.of(context)
),
child: menu
))
);
}
}
public partial class material_ {
public static Future<T> showMenu<T>(
BuildContext context,
RelativeRect position,
List<PopupMenuEntry<T>> items,
T initialValue,
float? elevation,
ShapeBorder shape,
Color color,
bool captureInheritedThemes = true,
bool useRootNavigator = false
) {
D.assert(context != null);
D.assert(position != null);
D.assert(items != null && items.isNotEmpty());
D.assert(material_.debugCheckHasMaterialLocalizations(context));
return Navigator.of(context, rootNavigator: useRootNavigator).push(new _PopupMenuRoute<T>(
position: position,
items: items,
initialValue: initialValue,
elevation: elevation,
theme: Theme.of(context, shadowThemeOnly: true),
popupMenuTheme: PopupMenuTheme.of(context),
barrierLabel: MaterialLocalizations.of(context).modalBarrierDismissLabel,
shape: shape,
color: color,
showMenuContext: context,
captureInheritedThemes: captureInheritedThemes
)).to<T>();
}
}
public delegate void PopupMenuItemSelected<T>(T value);
public delegate void PopupMenuCanceled();
public delegate List<PopupMenuEntry<T>> PopupMenuItemBuilder<T>(BuildContext context);
public class PopupMenuButton<T> : StatefulWidget {
public PopupMenuButton(
Key key = null,
PopupMenuItemBuilder<T> itemBuilder = null,
T initialValue = default,
PopupMenuItemSelected<T> onSelected = null,
PopupMenuCanceled onCanceled = null,
string tooltip = null,
float? elevation = null,
EdgeInsetsGeometry padding = null,
Widget child = null,
Icon icon = null,
Offset offset = null,
bool enabled = true,
ShapeBorder shape = null,
Color color = null,
bool captureInheritedThemes = true
) : base(key: key) {
offset = offset ?? Offset.zero;
D.assert(itemBuilder != null);
D.assert(offset != null);
D.assert(!(child != null && icon != null), () => "You can only pass [child] or [icon], not both.");
this.itemBuilder = itemBuilder;
this.initialValue = initialValue;
this.onSelected = onSelected;
this.onCanceled = onCanceled;
this.tooltip = tooltip;
this.elevation = elevation;
this.padding = padding ?? EdgeInsets.all(8.0f);
this.child = child;
this.icon = icon;
this.offset = offset;
this.enabled = enabled;
this.shape = shape;
this.color = color;
this.captureInheritedThemes = captureInheritedThemes;
}
public readonly PopupMenuItemBuilder<T> itemBuilder;
public readonly T initialValue;
public readonly PopupMenuItemSelected<T> onSelected;
public readonly PopupMenuCanceled onCanceled;
public readonly string tooltip;
public readonly float? elevation;
public readonly EdgeInsetsGeometry padding;
public readonly Widget child;
public readonly Widget icon;
public readonly Offset offset;
public readonly bool enabled;
public readonly ShapeBorder shape;
public readonly Color color;
public readonly bool captureInheritedThemes;
public override State createState() {
return new PopupMenuButtonState<T>();
}
}
public class PopupMenuButtonState<T> : State<PopupMenuButton<T>> {
void showButtonMenu() {
PopupMenuThemeData popupMenuTheme = PopupMenuTheme.of(context);
RenderBox button = (RenderBox) context.findRenderObject();
RenderBox overlay = (RenderBox) Overlay.of(context).context.findRenderObject();
RelativeRect position = RelativeRect.fromRect(
Rect.fromPoints(
button.localToGlobal(widget.offset, ancestor: overlay),
button.localToGlobal(button.size.bottomRight(Offset.zero), ancestor: overlay)
),
Offset.zero & overlay.size
);
List<PopupMenuEntry<T>> items = widget.itemBuilder(context);
if (items.isNotEmpty()) {
material_.showMenu<T>(
context: context,
elevation: widget.elevation ?? popupMenuTheme.elevation,
items: items,
initialValue: widget.initialValue,
position: position,
shape: widget.shape ?? popupMenuTheme.shape,
color: widget.color ?? popupMenuTheme.color,
captureInheritedThemes: widget.captureInheritedThemes
)
.then(newValue => {
if (!mounted)
return;
if (newValue == null) {
if (widget.onCanceled != null)
widget.onCanceled();
return;
}
if (widget.onSelected != null)
widget.onSelected((T) newValue);
});
}
}
Icon _getIcon(RuntimePlatform platform) {
switch (platform) {
case RuntimePlatform.IPhonePlayer:
case RuntimePlatform.OSXEditor:
case RuntimePlatform.OSXPlayer:
return new Icon(Icons.more_horiz);
default:
return new Icon(Icons.more_vert);
}
}
public override Widget build(BuildContext context) {
D.assert(material_.debugCheckHasMaterialLocalizations(context));
if (widget.child != null)
return new Tooltip(
message: widget.tooltip ?? MaterialLocalizations.of(context).showMenuTooltip,
child: new InkWell(
onTap: widget.enabled ? showButtonMenu : (GestureTapCallback) null,
canRequestFocus: widget.enabled,
child: widget.child
)
);
return new IconButton(
icon: widget.icon ?? _getIcon(Theme.of(context).platform),
padding: widget.padding,
tooltip: widget.tooltip ?? MaterialLocalizations.of(context).showMenuTooltip,
onPressed: widget.enabled ? showButtonMenu : (VoidCallback) null
);
}
}
}