您最多选择25个主题
主题必须以中文或者字母或数字开头,可以包含连字符 (-),并且长度不得超过35个字符
691 行
26 KiB
691 行
26 KiB
using System;
|
|
using System.Collections.Generic;
|
|
using System.Linq;
|
|
using uiwidgets;
|
|
using Unity.UIWidgets.animation;
|
|
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_ {
|
|
public delegate void DataColumnSortCallback(int columnIndex, bool ascending);
|
|
}
|
|
|
|
public class DataColumn {
|
|
public DataColumn(
|
|
Widget label,
|
|
string tooltip = null,
|
|
bool numeric = false,
|
|
material_.DataColumnSortCallback onSort = null
|
|
) {
|
|
D.assert(label != null);
|
|
this.label = label;
|
|
this.tooltip = tooltip;
|
|
this.numeric = numeric;
|
|
this.onSort = onSort;
|
|
}
|
|
|
|
public readonly Widget label;
|
|
|
|
public readonly string tooltip;
|
|
|
|
public readonly bool numeric;
|
|
|
|
public readonly material_.DataColumnSortCallback onSort;
|
|
|
|
internal bool _debugInteractive {
|
|
get { return onSort != null; }
|
|
}
|
|
}
|
|
|
|
public class DataRow {
|
|
public DataRow(
|
|
LocalKey key = null,
|
|
bool selected = false,
|
|
ValueChanged<bool> onSelectChanged = null,
|
|
List<DataCell> cells = null
|
|
) {
|
|
D.assert(cells != null);
|
|
this.key = key;
|
|
this.selected = selected;
|
|
this.onSelectChanged = onSelectChanged;
|
|
this.cells = cells;
|
|
}
|
|
|
|
public static DataRow byIndex(
|
|
int index = 0,
|
|
bool selected = false,
|
|
ValueChanged<bool> onSelectChanged = null,
|
|
List<DataCell> cells = null
|
|
) {
|
|
D.assert(cells != null);
|
|
return new DataRow(
|
|
new ValueKey<int>(index),
|
|
selected,
|
|
onSelectChanged,
|
|
cells
|
|
);
|
|
}
|
|
|
|
|
|
public readonly LocalKey key;
|
|
|
|
public readonly ValueChanged<bool> onSelectChanged;
|
|
|
|
public readonly bool selected;
|
|
|
|
public readonly List<DataCell> cells;
|
|
|
|
internal bool _debugInteractive {
|
|
get { return onSelectChanged != null || cells.Any((DataCell cell) => cell._debugInteractive); }
|
|
}
|
|
}
|
|
|
|
|
|
public class DataCell {
|
|
public DataCell(
|
|
Widget child,
|
|
bool placeholder = false,
|
|
bool showEditIcon = false,
|
|
VoidCallback onTap = null
|
|
) {
|
|
D.assert(child != null);
|
|
this.child = child;
|
|
this.placeholder = placeholder;
|
|
this.showEditIcon = showEditIcon;
|
|
this.onTap = onTap;
|
|
}
|
|
|
|
public static readonly DataCell empty = new DataCell(new Container(width: 0.0f, height: 0.0f));
|
|
|
|
public readonly Widget child;
|
|
|
|
public readonly bool placeholder;
|
|
|
|
public readonly bool showEditIcon;
|
|
|
|
public readonly VoidCallback onTap;
|
|
|
|
internal bool _debugInteractive {
|
|
get { return onTap != null; }
|
|
}
|
|
}
|
|
|
|
public class DataTable : StatelessWidget {
|
|
public DataTable(
|
|
Key key = null,
|
|
List<DataColumn> columns = null,
|
|
int? sortColumnIndex = 0,
|
|
bool sortAscending = true,
|
|
ValueSetter<bool> onSelectAll = null,
|
|
float dataRowHeight = material_.kMinInteractiveDimension,
|
|
float headingRowHeight = 56.0f,
|
|
float horizontalMargin = 24.0f,
|
|
float columnSpacing = 56.0f,
|
|
bool showCheckboxColumn = true,
|
|
float dividerThickness = 1.0f,
|
|
List<DataRow> rows = null
|
|
) : base(key: key) {
|
|
D.assert(columns != null);
|
|
D.assert(columns.isNotEmpty);
|
|
D.assert(sortColumnIndex == null || (sortColumnIndex >= 0 && sortColumnIndex < columns.Count));
|
|
D.assert(rows != null);
|
|
D.assert(!rows.Any((DataRow row) => row.cells.Count != columns.Count));
|
|
D.assert(dividerThickness >= 0);
|
|
this.columns = columns;
|
|
this.sortColumnIndex = sortColumnIndex;
|
|
this.sortAscending = sortAscending;
|
|
this.onSelectAll = onSelectAll;
|
|
this.dataRowHeight = dataRowHeight;
|
|
this.headingRowHeight = headingRowHeight;
|
|
this.horizontalMargin = horizontalMargin;
|
|
this.columnSpacing = columnSpacing;
|
|
this.showCheckboxColumn = showCheckboxColumn;
|
|
this.dividerThickness = dividerThickness;
|
|
this.rows = rows;
|
|
_onlyTextColumn = _initOnlyTextColumn(columns);
|
|
}
|
|
|
|
public readonly List<DataColumn> columns;
|
|
public readonly int? sortColumnIndex;
|
|
public readonly bool sortAscending;
|
|
public readonly ValueSetter<bool> onSelectAll;
|
|
public readonly float dataRowHeight;
|
|
public readonly float headingRowHeight;
|
|
public readonly float horizontalMargin;
|
|
public readonly float columnSpacing;
|
|
public readonly bool showCheckboxColumn;
|
|
public readonly List<DataRow> rows;
|
|
|
|
public readonly int? _onlyTextColumn;
|
|
|
|
static int? _initOnlyTextColumn(List<DataColumn> columns) {
|
|
int? result = null;
|
|
for (int index = 0; index < columns.Count; index += 1) {
|
|
DataColumn column = columns[index];
|
|
if (!column.numeric) {
|
|
if (result != null) {
|
|
return null;
|
|
}
|
|
|
|
result = index;
|
|
}
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
bool _debugInteractive {
|
|
get {
|
|
return columns.Any((DataColumn column) => column._debugInteractive)
|
|
|| rows.Any((DataRow row) => row._debugInteractive);
|
|
}
|
|
}
|
|
|
|
static readonly LocalKey _headingRowKey = new UniqueKey();
|
|
|
|
void _handleSelectAll(bool isChecked) {
|
|
if (onSelectAll != null) {
|
|
onSelectAll(isChecked);
|
|
}
|
|
else {
|
|
foreach (DataRow row in rows) {
|
|
if ((row.onSelectChanged != null) && (row.selected != isChecked))
|
|
row.onSelectChanged(isChecked);
|
|
}
|
|
}
|
|
}
|
|
|
|
static readonly float _sortArrowPadding = 2.0f;
|
|
static readonly float _headingFontSize = 12.0f;
|
|
static readonly TimeSpan _sortArrowAnimationDuration = new TimeSpan(0, 0, 0, 0, 150);
|
|
static readonly Color _grey100Opacity = new Color(0x0A000000);
|
|
static readonly Color _grey300Opacity = new Color(0x1E000000);
|
|
public readonly float dividerThickness;
|
|
|
|
Widget _buildCheckbox(
|
|
Color color = null,
|
|
bool isChecked = false,
|
|
VoidCallback onRowTap = null,
|
|
ValueChanged<bool?> onCheckboxChanged = null
|
|
) {
|
|
Widget contents = new Padding(
|
|
padding: EdgeInsetsDirectional.only(start: horizontalMargin,
|
|
end: horizontalMargin / 2.0f),
|
|
child: new Center(
|
|
child: new Checkbox(
|
|
activeColor: color,
|
|
value: isChecked,
|
|
onChanged: onCheckboxChanged
|
|
)
|
|
)
|
|
);
|
|
if (onRowTap != null) {
|
|
contents = new TableRowInkWell(
|
|
onTap: () => onRowTap(),
|
|
child: contents
|
|
);
|
|
}
|
|
|
|
return new TableCell(
|
|
verticalAlignment: TableCellVerticalAlignment.fill,
|
|
child: contents
|
|
);
|
|
}
|
|
|
|
Widget _buildHeadingCell(
|
|
BuildContext context = null,
|
|
EdgeInsetsGeometry padding = null,
|
|
Widget label = null,
|
|
string tooltip = null,
|
|
bool? numeric = null,
|
|
VoidCallback onSort = null,
|
|
bool? sorted = null,
|
|
bool? ascending = null
|
|
) {
|
|
List<Widget> arrowWithPadding() {
|
|
return onSort == null
|
|
? new List<Widget>()
|
|
: new List<
|
|
Widget>() {
|
|
new _SortArrow(
|
|
visible: sorted,
|
|
down: sorted ?? false ? ascending : null,
|
|
duration: _sortArrowAnimationDuration
|
|
),
|
|
|
|
new SizedBox(width: _sortArrowPadding)
|
|
};
|
|
}
|
|
|
|
var rowChild = new List<Widget>();
|
|
rowChild.Add(label);
|
|
rowChild.AddRange(arrowWithPadding());
|
|
label = new Row(
|
|
textDirection: numeric ?? false ? TextDirection.rtl : (TextDirection?) null,
|
|
children: rowChild
|
|
);
|
|
label = new Container(
|
|
padding: padding,
|
|
height: headingRowHeight,
|
|
alignment: numeric ?? false
|
|
? Alignment.centerRight
|
|
: (AlignmentGeometry) AlignmentDirectional.centerStart,
|
|
child: new AnimatedDefaultTextStyle(
|
|
style: new TextStyle(
|
|
fontWeight: FontWeight.w500,
|
|
fontSize: _headingFontSize,
|
|
height: Mathf.Min(1.0f, headingRowHeight / _headingFontSize),
|
|
color: (Theme.of(context).brightness == Brightness.light)
|
|
? ((onSort != null && (sorted ?? false)) ? Colors.black87 : Colors.black54)
|
|
: ((onSort != null && (sorted ?? false)) ? Colors.white : Colors.white70)
|
|
),
|
|
softWrap: false,
|
|
duration: _sortArrowAnimationDuration,
|
|
child: label
|
|
)
|
|
);
|
|
if (tooltip != null) {
|
|
label = new Tooltip(
|
|
message: tooltip,
|
|
child: label
|
|
);
|
|
}
|
|
|
|
// TODO(dkwingsmt): Only wrap Inkwell if onSort != null. Blocked by
|
|
// https://github.com/flutter/flutter/issues/51152
|
|
label = new InkWell(
|
|
onTap: () => onSort?.Invoke(),
|
|
child: label
|
|
);
|
|
return label;
|
|
}
|
|
|
|
Widget _buildDataCell(
|
|
BuildContext context,
|
|
EdgeInsetsGeometry padding,
|
|
Widget label,
|
|
bool numeric,
|
|
bool placeholder,
|
|
bool showEditIcon,
|
|
VoidCallback onTap,
|
|
VoidCallback onSelectChanged
|
|
) {
|
|
bool isLightTheme = Theme.of(context).brightness == Brightness.light;
|
|
if (showEditIcon) {
|
|
Widget icon = new Icon(Icons.edit, size: 18.0f);
|
|
label = new Expanded(child: label);
|
|
label = new Row(
|
|
textDirection: numeric ? TextDirection.rtl : (TextDirection?) null,
|
|
children: new List<Widget> {label, icon}
|
|
);
|
|
}
|
|
|
|
label = new Container(
|
|
padding: padding,
|
|
height: dataRowHeight,
|
|
alignment: numeric ? Alignment.centerRight : (AlignmentGeometry) AlignmentDirectional.centerStart,
|
|
child: new DefaultTextStyle(
|
|
style: new TextStyle(
|
|
// TODO(ianh): font family should be Roboto; see https://github.com/flutter/flutter/issues/3116
|
|
fontSize: 13.0f,
|
|
color: isLightTheme
|
|
? (placeholder ? Colors.black38 : Colors.black87)
|
|
: (placeholder ? Colors.white38 : Colors.white70)
|
|
),
|
|
child: IconTheme.merge(
|
|
data: new IconThemeData(
|
|
color: isLightTheme ? Colors.black54 : Colors.white70
|
|
),
|
|
child: new DropdownButtonHideUnderline(child: label)
|
|
)
|
|
)
|
|
);
|
|
if (onTap != null) {
|
|
label = new InkWell(
|
|
onTap: () => onTap(),
|
|
child: label
|
|
);
|
|
}
|
|
else if (onSelectChanged != null) {
|
|
label = new TableRowInkWell(
|
|
onTap: () => onSelectChanged(),
|
|
child: label
|
|
);
|
|
}
|
|
|
|
return label;
|
|
}
|
|
|
|
public override Widget build(BuildContext context) {
|
|
D.assert(!_debugInteractive || material_.debugCheckHasMaterial(context));
|
|
|
|
ThemeData theme = Theme.of(context);
|
|
BoxDecoration _kSelectedDecoration = new BoxDecoration(
|
|
border: new Border(bottom: Divider.createBorderSide(context, width: dividerThickness)),
|
|
// The backgroundColor has to be transparent so you can see the ink on the material
|
|
color: (Theme.of(context).brightness == Brightness.light) ? _grey100Opacity : _grey300Opacity
|
|
);
|
|
BoxDecoration _kUnselectedDecoration = new BoxDecoration(
|
|
border: new Border(bottom: Divider.createBorderSide(context, width: dividerThickness))
|
|
);
|
|
|
|
bool displayCheckboxColumn =
|
|
showCheckboxColumn && rows.Any((DataRow row) => row.onSelectChanged != null);
|
|
bool allChecked = displayCheckboxColumn &&
|
|
!rows.Any((DataRow row) => row.onSelectChanged != null && !row.selected);
|
|
|
|
List<TableColumnWidth> tableColumns =
|
|
new List<TableColumnWidth>(new TableColumnWidth[columns.Count + (displayCheckboxColumn ? 1 : 0)]);
|
|
|
|
List<TableRow> tableRows = Enumerable.Range(0, rows.Count + 1).Select((index) => {
|
|
return new TableRow(
|
|
key: index == 0 ? _headingRowKey : rows[index - 1].key,
|
|
decoration: index > 0 && rows[index - 1].selected
|
|
? _kSelectedDecoration
|
|
: _kUnselectedDecoration,
|
|
children: new List<Widget>(new Widget[tableColumns.Count])
|
|
);
|
|
}).ToList();
|
|
|
|
int rowIndex;
|
|
|
|
int displayColumnIndex = 0;
|
|
if (displayCheckboxColumn) {
|
|
tableColumns[0] = new FixedColumnWidth(horizontalMargin + Checkbox.width + horizontalMargin / 2.0f);
|
|
tableRows[0].children[0] = _buildCheckbox(
|
|
color: theme.accentColor,
|
|
isChecked: allChecked,
|
|
onCheckboxChanged: _check => _handleSelectAll(_check ?? false)
|
|
);
|
|
rowIndex = 1;
|
|
foreach (DataRow row in rows) {
|
|
tableRows[rowIndex].children[0] = _buildCheckbox(
|
|
color: theme.accentColor,
|
|
isChecked: row.selected,
|
|
onRowTap: () => {
|
|
if (row.onSelectChanged != null) {
|
|
row.onSelectChanged(!row.selected);
|
|
}
|
|
},
|
|
onCheckboxChanged: _select => row.onSelectChanged(_select ?? false)
|
|
);
|
|
rowIndex += 1;
|
|
}
|
|
|
|
displayColumnIndex += 1;
|
|
}
|
|
|
|
for (int dataColumnIndex = 0; dataColumnIndex < columns.Count; dataColumnIndex += 1) {
|
|
DataColumn column = columns[dataColumnIndex];
|
|
|
|
float paddingStart;
|
|
if (dataColumnIndex == 0 && displayCheckboxColumn) {
|
|
paddingStart = horizontalMargin / 2.0f;
|
|
}
|
|
else if (dataColumnIndex == 0 && !displayCheckboxColumn) {
|
|
paddingStart = horizontalMargin;
|
|
}
|
|
else {
|
|
paddingStart = columnSpacing / 2.0f;
|
|
}
|
|
|
|
float paddingEnd;
|
|
if (dataColumnIndex == columns.Count - 1) {
|
|
paddingEnd = horizontalMargin;
|
|
}
|
|
else {
|
|
paddingEnd = columnSpacing / 2.0f;
|
|
}
|
|
|
|
EdgeInsetsDirectional padding = EdgeInsetsDirectional.only(
|
|
start: paddingStart,
|
|
end: paddingEnd
|
|
);
|
|
if (dataColumnIndex == _onlyTextColumn) {
|
|
tableColumns[displayColumnIndex] = new IntrinsicColumnWidth(flex: 1.0f);
|
|
}
|
|
else {
|
|
tableColumns[displayColumnIndex] = new IntrinsicColumnWidth();
|
|
}
|
|
|
|
var currentColumnIndex = dataColumnIndex;
|
|
tableRows[0].children[displayColumnIndex] = _buildHeadingCell(
|
|
context: context,
|
|
padding: padding,
|
|
label: column.label,
|
|
tooltip: column.tooltip,
|
|
numeric: column.numeric,
|
|
onSort: column.onSort != null
|
|
? () => column.onSort(currentColumnIndex, sortColumnIndex != currentColumnIndex || !sortAscending)
|
|
: (VoidCallback) null,
|
|
sorted: dataColumnIndex == sortColumnIndex,
|
|
ascending: sortAscending
|
|
);
|
|
rowIndex = 1;
|
|
foreach (DataRow row in rows) {
|
|
DataCell cell = row.cells[dataColumnIndex];
|
|
var curRow = row;
|
|
tableRows[rowIndex].children[displayColumnIndex] = _buildDataCell(
|
|
context: context,
|
|
padding: padding,
|
|
label: cell.child,
|
|
numeric: column.numeric,
|
|
placeholder: cell.placeholder,
|
|
showEditIcon: cell.showEditIcon,
|
|
onTap: cell.onTap,
|
|
onSelectChanged: () => {
|
|
if (curRow.onSelectChanged != null) {
|
|
curRow.onSelectChanged(!curRow.selected);
|
|
}
|
|
});
|
|
rowIndex += 1;
|
|
}
|
|
|
|
displayColumnIndex += 1;
|
|
}
|
|
|
|
return new Table(
|
|
columnWidths: tableColumns.Select((x, i) => new {x, i})
|
|
.ToDictionary(a => a.i, a => a.x),
|
|
children: tableRows
|
|
);
|
|
}
|
|
}
|
|
|
|
public class TableRowInkWell : InkResponse {
|
|
public TableRowInkWell(
|
|
Key key = null,
|
|
Widget child = null,
|
|
GestureTapCallback onTap = null,
|
|
GestureTapCallback onDoubleTap = null,
|
|
GestureLongPressCallback onLongPress = null,
|
|
ValueChanged<bool> onHighlightChanged = null
|
|
) : base(
|
|
key: key,
|
|
child: child,
|
|
onTap: onTap,
|
|
onDoubleTap: onDoubleTap,
|
|
onLongPress: onLongPress,
|
|
onHighlightChanged: onHighlightChanged,
|
|
containedInkWell: true,
|
|
highlightShape: BoxShape.rectangle
|
|
) {
|
|
}
|
|
|
|
public override RectCallback getRectCallback(RenderBox referenceBox) {
|
|
return () => {
|
|
RenderObject cell = referenceBox;
|
|
AbstractNodeMixinDiagnosticableTree table = cell.parent;
|
|
Matrix4 transform = Matrix4.identity();
|
|
while (table is RenderObject && !(table is RenderTable)) {
|
|
RenderObject parentBox = table as RenderObject;
|
|
parentBox.applyPaintTransform(cell, transform);
|
|
D.assert(table == cell.parent);
|
|
cell = parentBox;
|
|
table = table.parent;
|
|
}
|
|
|
|
if (table is RenderTable renderTable) {
|
|
TableCellParentData cellParentData = cell.parentData as TableCellParentData;
|
|
D.assert(cellParentData.y != null);
|
|
Rect rect = renderTable.getRowBox(cellParentData.y);
|
|
// The rect is in the table's coordinate space. We need to change it to the
|
|
// TableRowInkWell's coordinate space.
|
|
renderTable.applyPaintTransform(cell, transform);
|
|
Offset offset = MatrixUtils.getAsTranslation(transform);
|
|
if (offset != null)
|
|
return rect.shift(-offset);
|
|
}
|
|
|
|
return Rect.zero;
|
|
}
|
|
;
|
|
}
|
|
|
|
public override bool debugCheckContext(BuildContext context) {
|
|
D.assert(WidgetsD.debugCheckHasTable(context));
|
|
return base.debugCheckContext(context);
|
|
}
|
|
}
|
|
|
|
internal class _SortArrow : StatefulWidget {
|
|
internal _SortArrow(
|
|
Key key = null,
|
|
bool? visible = null,
|
|
bool? down = null,
|
|
TimeSpan? duration = null
|
|
) : base(key: key) {
|
|
this.visible = visible;
|
|
this.down = down;
|
|
this.duration = duration;
|
|
}
|
|
|
|
public readonly bool? visible;
|
|
public readonly bool? down;
|
|
public readonly TimeSpan? duration;
|
|
|
|
public override State createState() => new _SortArrowState();
|
|
}
|
|
|
|
class _SortArrowState : TickerProviderStateMixin<_SortArrow> {
|
|
AnimationController _opacityController;
|
|
Animation<float> _opacityAnimation;
|
|
AnimationController _orientationController;
|
|
Animation<float> _orientationAnimation;
|
|
float _orientationOffset = 0.0f;
|
|
bool _down;
|
|
|
|
static readonly Animatable<float> _turnTween =
|
|
new FloatTween(begin: 0.0f, end: Mathf.PI).chain(new CurveTween(curve: Curves.easeIn));
|
|
|
|
public override void initState() {
|
|
base.initState();
|
|
_opacityAnimation = new CurvedAnimation(
|
|
parent: _opacityController = new AnimationController(
|
|
duration: widget.duration,
|
|
vsync: this
|
|
),
|
|
curve: Curves.fastOutSlowIn
|
|
);
|
|
_opacityAnimation.addListener(_rebuild);
|
|
_opacityController.setValue(widget.visible ?? false ? 1.0f : 0.0f);
|
|
_orientationController = new AnimationController(
|
|
duration: widget.duration,
|
|
vsync: this
|
|
);
|
|
_orientationAnimation = _orientationController.drive(_turnTween);
|
|
_orientationAnimation.addListener(_rebuild);
|
|
_orientationAnimation.addStatusListener(_resetOrientationAnimation);
|
|
if (widget.visible ?? false) {
|
|
_orientationOffset = widget.down ?? false ? 0.0f : Mathf.PI;
|
|
}
|
|
}
|
|
|
|
void _rebuild() {
|
|
setState(() => {
|
|
// The animations changed, so we need to rebuild.
|
|
});
|
|
}
|
|
|
|
void _resetOrientationAnimation(AnimationStatus status) {
|
|
if (status == AnimationStatus.completed) {
|
|
D.assert(_orientationAnimation.value == Mathf.PI);
|
|
_orientationOffset += Mathf.PI;
|
|
_orientationController.setValue(0.0f); // TODO(ianh): This triggers a pointless rebuild.
|
|
}
|
|
}
|
|
|
|
public override void didUpdateWidget(StatefulWidget oldStatefullWidget) {
|
|
var oldWidget = oldStatefullWidget as _SortArrow;
|
|
if (oldWidget == null) {
|
|
return;
|
|
}
|
|
|
|
base.didUpdateWidget(oldWidget);
|
|
|
|
bool skipArrow = false;
|
|
bool newDown = widget.down ?? _down;
|
|
if (oldWidget.visible != widget.visible) {
|
|
if ((widget.visible ?? false) && (_opacityController.status == AnimationStatus.dismissed)) {
|
|
_orientationController.stop();
|
|
_orientationController.setValue(0.0f);
|
|
_orientationOffset = newDown ? 0.0f : Mathf.PI;
|
|
skipArrow = true;
|
|
}
|
|
|
|
if (widget.visible ?? false) {
|
|
_opacityController.forward();
|
|
}
|
|
else {
|
|
_opacityController.reverse();
|
|
}
|
|
}
|
|
|
|
if ((_down != newDown) && !skipArrow) {
|
|
if (_orientationController.status == AnimationStatus.dismissed) {
|
|
_orientationController.forward();
|
|
}
|
|
else {
|
|
_orientationController.reverse();
|
|
}
|
|
}
|
|
|
|
_down = newDown;
|
|
}
|
|
|
|
public override void dispose() {
|
|
_opacityController.dispose();
|
|
_orientationController.dispose();
|
|
base.dispose();
|
|
}
|
|
|
|
const float _arrowIconBaselineOffset = -1.5f;
|
|
const float _arrowIconSize = 16.0f;
|
|
|
|
public override Widget build(BuildContext context) {
|
|
var transform = Matrix4.rotationZ(_orientationOffset + _orientationAnimation.value);
|
|
transform.setTranslationRaw(0.0f, _arrowIconBaselineOffset, 0.0f);
|
|
return new Opacity(
|
|
opacity: _opacityAnimation.value,
|
|
child: new widgets.Transform(
|
|
transform: transform,
|
|
alignment: Alignment.center,
|
|
child: new Icon(
|
|
Icons.arrow_downward,
|
|
size: _arrowIconSize,
|
|
color: (Theme.of(context).brightness == Brightness.light) ? Colors.black87 : Colors.white70
|
|
)
|
|
)
|
|
);
|
|
}
|
|
}
|
|
}
|