您最多选择25个主题
主题必须以中文或者字母或数字开头,可以包含连字符 (-),并且长度不得超过35个字符
917 行
36 KiB
917 行
36 KiB
using System;
|
|
using System.Collections.Generic;
|
|
using com.unity.uiwidgets.Runtime.rendering;
|
|
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 Color = Unity.UIWidgets.ui.Color;
|
|
using TextStyle = Unity.UIWidgets.painting.TextStyle;
|
|
|
|
namespace Unity.UIWidgets.material {
|
|
public partial class material_ {
|
|
public static readonly TimeSpan _monthScrollDuration = new TimeSpan(0, 0, 0, 0, 200);
|
|
|
|
public const float _dayPickerRowHeight = 42.0f;
|
|
|
|
public const int _maxDayPickerRowCount = 6; // A 31 day month that starts on Saturday.
|
|
|
|
public const float _maxDayPickerHeight = _dayPickerRowHeight * (_maxDayPickerRowCount + 1);
|
|
public const float _monthPickerHorizontalPadding = 8.0f;
|
|
|
|
public const int _yearPickerColumnCount = 3;
|
|
public const float _yearPickerPadding = 16.0f;
|
|
public const float _yearPickerRowHeight = 52.0f;
|
|
public const float _yearPickerRowSpacing = 8.0f;
|
|
|
|
public const float _subHeaderHeight = 52.0f;
|
|
public const float _monthNavButtonsWidth = 108.0f;
|
|
}
|
|
|
|
public class CalendarDatePicker : StatefulWidget {
|
|
public CalendarDatePicker(
|
|
Key key = null,
|
|
DateTime? initialDate = null,
|
|
DateTime? firstDate = null,
|
|
DateTime? lastDate = null,
|
|
ValueChanged<DateTime> onDateChanged = null,
|
|
ValueChanged<DateTime> onDisplayedMonthChanged = null,
|
|
DatePickerMode initialCalendarMode = DatePickerMode.day,
|
|
material_.SelectableDayPredicate selectableDayPredicate = null
|
|
) : base(key: key) {
|
|
D.assert(initialDate != null);
|
|
D.assert(firstDate != null);
|
|
D.assert(lastDate != null);
|
|
this.initialDate = utils.dateOnly(initialDate.Value);
|
|
this.firstDate = utils.dateOnly(firstDate.Value);
|
|
this.lastDate = utils.dateOnly(lastDate.Value);
|
|
D.assert(onDateChanged != null);
|
|
|
|
this.onDateChanged = onDateChanged;
|
|
this.onDisplayedMonthChanged = onDisplayedMonthChanged;
|
|
this.initialCalendarMode = initialCalendarMode;
|
|
this.selectableDayPredicate = selectableDayPredicate;
|
|
D.assert(
|
|
!(this.lastDate < this.firstDate),
|
|
() => $"lastDate {lastDate} must be on or after firstDate {firstDate}."
|
|
);
|
|
D.assert(
|
|
!(this.initialDate < this.firstDate),
|
|
() => $"initialDate {initialDate} must be on or after firstDate {firstDate}."
|
|
);
|
|
D.assert(
|
|
!(this.initialDate > this.lastDate),
|
|
() => $"initialDate {initialDate} must be on or before lastDate {lastDate}."
|
|
);
|
|
D.assert(
|
|
selectableDayPredicate == null || selectableDayPredicate(this.initialDate),
|
|
() => $"Provided initialDate {initialDate} must satisfy provided selectableDayPredicate."
|
|
);
|
|
}
|
|
|
|
public readonly DateTime initialDate;
|
|
|
|
public readonly DateTime firstDate;
|
|
|
|
public readonly DateTime lastDate;
|
|
|
|
public readonly ValueChanged<DateTime> onDateChanged;
|
|
|
|
public readonly ValueChanged<DateTime> onDisplayedMonthChanged;
|
|
|
|
public readonly DatePickerMode initialCalendarMode;
|
|
|
|
public readonly material_.SelectableDayPredicate selectableDayPredicate;
|
|
|
|
public override State createState() => new UIWidgets.material._CalendarDatePickerState();
|
|
}
|
|
|
|
internal class _CalendarDatePickerState : State<CalendarDatePicker> {
|
|
bool _announcedInitialDate = false;
|
|
DatePickerMode _mode;
|
|
DateTime _currentDisplayedMonthDate;
|
|
DateTime _selectedDate;
|
|
public readonly GlobalKey _monthPickerKey = GlobalKey.key();
|
|
public readonly GlobalKey _yearPickerKey = GlobalKey.key();
|
|
MaterialLocalizations _localizations;
|
|
TextDirection _textDirection;
|
|
|
|
public override void initState() {
|
|
base.initState();
|
|
_mode = widget.initialCalendarMode;
|
|
_currentDisplayedMonthDate = new DateTime(widget.initialDate.Year, widget.initialDate.Month, 1);
|
|
_selectedDate = widget.initialDate;
|
|
}
|
|
|
|
public override void didChangeDependencies() {
|
|
base.didChangeDependencies();
|
|
_localizations = MaterialLocalizations.of(context);
|
|
_textDirection = Directionality.of(context);
|
|
if (!_announcedInitialDate) {
|
|
_announcedInitialDate = true;
|
|
// SemanticsService.announce(
|
|
// _localizations.formatFullDate(_selectedDate),
|
|
// _textDirection,
|
|
// );
|
|
}
|
|
}
|
|
|
|
void _vibrate() {
|
|
// switch (Theme.of(context).platform) {
|
|
// case TargetPlatform.android:
|
|
// case TargetPlatform.fuchsia:
|
|
// case TargetPlatform.linux:
|
|
// case TargetPlatform.windows:
|
|
// // HapticFeedback.vibrate();
|
|
// break;
|
|
// case TargetPlatform.iOS:
|
|
// case TargetPlatform.macOS:
|
|
// break;
|
|
// }
|
|
}
|
|
|
|
void _handleModeChanged(DatePickerMode mode) {
|
|
_vibrate();
|
|
setState(() => {
|
|
_mode = mode;
|
|
// if (_mode == DatePickerMode.day) {
|
|
// SemanticsService.announce(
|
|
// _localizations.formatMonthYear(_selectedDate),
|
|
// _textDirection,
|
|
// );
|
|
// } else {
|
|
// SemanticsService.announce(
|
|
// _localizations.formatYear(_selectedDate),
|
|
// _textDirection,
|
|
// );
|
|
// }
|
|
});
|
|
}
|
|
|
|
void _handleMonthChanged(DateTime date) {
|
|
setState(() => {
|
|
if (_currentDisplayedMonthDate.Year != date.Year || _currentDisplayedMonthDate.Month != date.Month) {
|
|
_currentDisplayedMonthDate = new DateTime(date.Year, date.Month, 1);
|
|
widget.onDisplayedMonthChanged?.Invoke(_currentDisplayedMonthDate);
|
|
}
|
|
});
|
|
}
|
|
|
|
void _handleYearChanged(DateTime value) {
|
|
_vibrate();
|
|
|
|
if (value < widget.firstDate) {
|
|
value = widget.firstDate;
|
|
}
|
|
else if (value > widget.lastDate) {
|
|
value = widget.lastDate;
|
|
}
|
|
|
|
setState(() => {
|
|
_mode = DatePickerMode.day;
|
|
_handleMonthChanged(value);
|
|
});
|
|
}
|
|
|
|
void _handleDayChanged(DateTime value) {
|
|
_vibrate();
|
|
setState(() => {
|
|
_selectedDate = value;
|
|
widget.onDateChanged?.Invoke(_selectedDate);
|
|
});
|
|
}
|
|
|
|
Widget _buildPicker() {
|
|
switch (_mode) {
|
|
case DatePickerMode.day:
|
|
return new _MonthPicker(
|
|
key: _monthPickerKey,
|
|
initialMonth: _currentDisplayedMonthDate,
|
|
currentDate: DateTime.Now,
|
|
firstDate: widget.firstDate,
|
|
lastDate: widget.lastDate,
|
|
selectedDate: _selectedDate,
|
|
onChanged: _handleDayChanged,
|
|
onDisplayedMonthChanged: _handleMonthChanged,
|
|
selectableDayPredicate: widget.selectableDayPredicate
|
|
);
|
|
case DatePickerMode.year:
|
|
return new Padding(
|
|
padding: EdgeInsets.only(top: material_._subHeaderHeight),
|
|
child: new _YearPicker(
|
|
key: _yearPickerKey,
|
|
currentDate: DateTime.Now,
|
|
firstDate: widget.firstDate,
|
|
lastDate: widget.lastDate,
|
|
initialDate: _currentDisplayedMonthDate,
|
|
selectedDate: _selectedDate,
|
|
onChanged: _handleYearChanged
|
|
)
|
|
);
|
|
}
|
|
|
|
return null;
|
|
}
|
|
|
|
public override Widget build(BuildContext context) {
|
|
return new Stack(
|
|
children: new List<Widget> {
|
|
new SingleChildScrollView(
|
|
child: new SizedBox(
|
|
height: material_._maxDayPickerHeight,
|
|
child: _buildPicker()
|
|
)
|
|
),
|
|
// Put the mode toggle button on top so that it won"t be covered up by the _MonthPicker
|
|
new _DatePickerModeToggleButton(
|
|
mode: _mode,
|
|
title: _localizations.formatMonthYear(_currentDisplayedMonthDate),
|
|
onTitlePressed: () => {
|
|
// Toggle the day/year mode.
|
|
_handleModeChanged(_mode == DatePickerMode.day ? DatePickerMode.year : DatePickerMode.day);
|
|
}
|
|
)
|
|
}
|
|
);
|
|
}
|
|
}
|
|
|
|
class _DatePickerModeToggleButton : StatefulWidget {
|
|
internal _DatePickerModeToggleButton(
|
|
DatePickerMode mode,
|
|
string title,
|
|
VoidCallback onTitlePressed
|
|
) {
|
|
this.mode = mode;
|
|
this.title = title;
|
|
this.onTitlePressed = onTitlePressed;
|
|
}
|
|
|
|
public readonly DatePickerMode mode;
|
|
|
|
public readonly string title;
|
|
|
|
public readonly VoidCallback onTitlePressed;
|
|
|
|
public override State createState() => new _DatePickerModeToggleButtonState();
|
|
}
|
|
|
|
class _DatePickerModeToggleButtonState : SingleTickerProviderStateMixin<_DatePickerModeToggleButton> {
|
|
AnimationController _controller;
|
|
|
|
public override void initState() {
|
|
base.initState();
|
|
_controller = new AnimationController(
|
|
value: widget.mode == DatePickerMode.year ? 0.5f : 0,
|
|
upperBound: 0.5f,
|
|
duration: new TimeSpan(0, 0, 0, 0, 200),
|
|
vsync: this
|
|
);
|
|
}
|
|
|
|
public override void didUpdateWidget(StatefulWidget statefulWidget) {
|
|
base.didUpdateWidget(statefulWidget);
|
|
if (statefulWidget is _DatePickerModeToggleButton oldWidget) {
|
|
if (oldWidget.mode == widget.mode) {
|
|
return;
|
|
}
|
|
|
|
if (widget.mode == DatePickerMode.year) {
|
|
_controller.forward();
|
|
}
|
|
else {
|
|
_controller.reverse();
|
|
}
|
|
}
|
|
}
|
|
|
|
public override Widget build(BuildContext context) {
|
|
ColorScheme colorScheme = Theme.of(context).colorScheme;
|
|
TextTheme textTheme = Theme.of(context).textTheme;
|
|
Color controlColor = colorScheme.onSurface.withOpacity(0.60f);
|
|
|
|
var rowChildren = new List<Widget> {
|
|
new Flexible(
|
|
child: new Container(
|
|
height: material_._subHeaderHeight,
|
|
child: new InkWell(
|
|
onTap: () => widget.onTitlePressed(),
|
|
child: new Padding(
|
|
padding: EdgeInsets.symmetric(horizontal: 8),
|
|
child: new Row(
|
|
children: new List<Widget> {
|
|
new Flexible(
|
|
child: new Text(
|
|
widget.title,
|
|
overflow: TextOverflow.ellipsis,
|
|
style: textTheme.subtitle2?.copyWith(
|
|
color: controlColor
|
|
)
|
|
)
|
|
),
|
|
new RotationTransition(
|
|
turns: _controller,
|
|
child: new Icon(
|
|
Icons.arrow_drop_down,
|
|
color: controlColor
|
|
)
|
|
),
|
|
}
|
|
)
|
|
)
|
|
)
|
|
)
|
|
)
|
|
};
|
|
if (widget.mode == DatePickerMode.day) {
|
|
// Give space for the prev/next month buttons that are underneath this row
|
|
rowChildren.Add(new SizedBox(width: material_._monthNavButtonsWidth));
|
|
}
|
|
|
|
return new Container(
|
|
padding: EdgeInsetsDirectional.only(start: 16, end: 4),
|
|
height: material_._subHeaderHeight,
|
|
child: new Row(
|
|
children: rowChildren
|
|
)
|
|
);
|
|
}
|
|
|
|
public override void dispose() {
|
|
_controller.dispose();
|
|
base.dispose();
|
|
}
|
|
}
|
|
|
|
class _MonthPicker : StatefulWidget {
|
|
internal _MonthPicker(
|
|
Key key = null,
|
|
DateTime? initialMonth = null,
|
|
DateTime? currentDate = null,
|
|
DateTime? firstDate = null,
|
|
DateTime? lastDate = null,
|
|
DateTime? selectedDate = null,
|
|
ValueChanged<DateTime> onChanged = null,
|
|
ValueChanged<DateTime> onDisplayedMonthChanged = null,
|
|
material_.SelectableDayPredicate selectableDayPredicate = null
|
|
) : base(key: key) {
|
|
D.assert(selectedDate != null);
|
|
D.assert(currentDate != null);
|
|
D.assert(onChanged != null);
|
|
D.assert(firstDate != null);
|
|
D.assert(lastDate != null);
|
|
D.assert(!(firstDate > lastDate));
|
|
D.assert(!(selectedDate < firstDate));
|
|
D.assert(!(selectedDate > lastDate));
|
|
this.initialMonth = initialMonth.Value;
|
|
this.currentDate = currentDate.Value;
|
|
this.firstDate = firstDate.Value;
|
|
this.lastDate = lastDate.Value;
|
|
this.selectedDate = selectedDate.Value;
|
|
this.onChanged = onChanged;
|
|
this.onDisplayedMonthChanged = onDisplayedMonthChanged;
|
|
this.selectableDayPredicate = selectableDayPredicate;
|
|
}
|
|
|
|
public readonly DateTime initialMonth;
|
|
|
|
public readonly DateTime currentDate;
|
|
|
|
public readonly DateTime firstDate;
|
|
|
|
public readonly DateTime lastDate;
|
|
|
|
public readonly DateTime selectedDate;
|
|
|
|
public readonly ValueChanged<DateTime> onChanged;
|
|
|
|
public readonly ValueChanged<DateTime> onDisplayedMonthChanged;
|
|
|
|
public readonly material_.SelectableDayPredicate selectableDayPredicate;
|
|
|
|
public override State createState() => new _MonthPickerState();
|
|
}
|
|
|
|
class _MonthPickerState : State<_MonthPicker> {
|
|
DateTime _currentMonth;
|
|
DateTime _nextMonthDate;
|
|
DateTime _previousMonthDate;
|
|
PageController _pageController;
|
|
MaterialLocalizations _localizations;
|
|
TextDirection _textDirection;
|
|
|
|
public override void initState() {
|
|
base.initState();
|
|
_currentMonth = widget.initialMonth;
|
|
_previousMonthDate = utils.addMonthsToMonthDate(_currentMonth, -1);
|
|
_nextMonthDate = utils.addMonthsToMonthDate(_currentMonth, 1);
|
|
_pageController = new PageController(initialPage: utils.monthDelta(widget.firstDate, _currentMonth));
|
|
}
|
|
|
|
public override void didChangeDependencies() {
|
|
base.didChangeDependencies();
|
|
_localizations = MaterialLocalizations.of(context);
|
|
_textDirection = Directionality.of(context);
|
|
}
|
|
|
|
public override void dispose() {
|
|
_pageController?.dispose();
|
|
base.dispose();
|
|
}
|
|
|
|
void _handleMonthPageChanged(int monthPage) {
|
|
DateTime monthDate = utils.addMonthsToMonthDate(widget.firstDate, monthPage);
|
|
if (_currentMonth.Year != monthDate.Year || _currentMonth.Month != monthDate.Month) {
|
|
_currentMonth = new DateTime(monthDate.Year, monthDate.Month, 1);
|
|
_previousMonthDate = utils.addMonthsToMonthDate(_currentMonth, -1);
|
|
_nextMonthDate = utils.addMonthsToMonthDate(_currentMonth, 1);
|
|
widget.onDisplayedMonthChanged?.Invoke(_currentMonth);
|
|
}
|
|
}
|
|
|
|
void _handleNextMonth() {
|
|
if (!_isDisplayingLastMonth) {
|
|
// SemanticsService.announce(
|
|
// _localizations.formatMonthYear(_nextMonthDate),
|
|
// _textDirection,
|
|
// );
|
|
_pageController.nextPage(
|
|
duration: material_._monthScrollDuration,
|
|
curve: Curves.ease
|
|
);
|
|
}
|
|
}
|
|
|
|
void _handlePreviousMonth() {
|
|
if (!_isDisplayingFirstMonth) {
|
|
// SemanticsService.announce(
|
|
// _localizations.formatMonthYear(_previousMonthDate),
|
|
// _textDirection,
|
|
// );
|
|
_pageController.previousPage(
|
|
duration: material_._monthScrollDuration,
|
|
curve: Curves.ease
|
|
);
|
|
}
|
|
}
|
|
|
|
bool _isDisplayingFirstMonth {
|
|
get { return !(_currentMonth > new DateTime(widget.firstDate.Year, widget.firstDate.Month, 1)); }
|
|
}
|
|
|
|
bool _isDisplayingLastMonth {
|
|
get {
|
|
return !(_currentMonth < new DateTime(widget.lastDate.Year, widget.lastDate.Month, 1)
|
|
);
|
|
}
|
|
}
|
|
|
|
Widget _buildItems(BuildContext context, int index) {
|
|
DateTime month = utils.addMonthsToMonthDate(widget.firstDate, index);
|
|
return new _DayPicker(
|
|
key: new ValueKey<DateTime>(month),
|
|
selectedDate: widget.selectedDate,
|
|
currentDate: widget.currentDate,
|
|
onChanged: widget.onChanged,
|
|
firstDate: widget.firstDate,
|
|
lastDate: widget.lastDate,
|
|
displayedMonth: month,
|
|
selectableDayPredicate: widget.selectableDayPredicate
|
|
);
|
|
}
|
|
|
|
public override Widget build(BuildContext context) {
|
|
string previousTooltipText =
|
|
$"{_localizations.previousMonthTooltip} {_localizations.formatMonthYear(_previousMonthDate)}";
|
|
string nextTooltipText =
|
|
$"{_localizations.nextMonthTooltip} {_localizations.formatMonthYear(_nextMonthDate)}";
|
|
Color controlColor = Theme.of(context).colorScheme.onSurface.withOpacity(0.60f);
|
|
|
|
return new Column(
|
|
children: new List<Widget> {
|
|
new Container(
|
|
padding: EdgeInsetsDirectional.only(start: 16, end: 4),
|
|
height: material_._subHeaderHeight,
|
|
child:
|
|
new Row(
|
|
children: new List<Widget> {
|
|
new Spacer(),
|
|
new IconButton(
|
|
icon: new Icon(Icons.chevron_left),
|
|
color: controlColor,
|
|
tooltip: _isDisplayingFirstMonth ? null : previousTooltipText,
|
|
onPressed: _isDisplayingFirstMonth ? (VoidCallback) null : _handlePreviousMonth
|
|
),
|
|
new IconButton(icon: new Icon(Icons.chevron_right),
|
|
color: controlColor,
|
|
tooltip: _isDisplayingLastMonth ? null : nextTooltipText,
|
|
onPressed: _isDisplayingLastMonth ? (VoidCallback) null : _handleNextMonth
|
|
)
|
|
}
|
|
)
|
|
),
|
|
new _DayHeaders(),
|
|
new Expanded(
|
|
child: PageView.builder(
|
|
controller: _pageController,
|
|
itemBuilder: _buildItems,
|
|
itemCount: utils.monthDelta(widget.firstDate, widget.lastDate) + 1,
|
|
scrollDirection: Axis.horizontal,
|
|
onPageChanged: _handleMonthPageChanged
|
|
)
|
|
)
|
|
}
|
|
);
|
|
}
|
|
}
|
|
|
|
class _DayPicker : StatelessWidget {
|
|
internal _DayPicker(
|
|
Key key = null,
|
|
DateTime? currentDate = null,
|
|
DateTime? displayedMonth = null,
|
|
DateTime? firstDate = null,
|
|
DateTime? lastDate = null,
|
|
DateTime? selectedDate = null,
|
|
ValueChanged<DateTime> onChanged = null,
|
|
material_.SelectableDayPredicate selectableDayPredicate = null
|
|
) : base(key: key) {
|
|
D.assert(currentDate != null);
|
|
D.assert(displayedMonth != null);
|
|
D.assert(firstDate != null);
|
|
D.assert(lastDate != null);
|
|
D.assert(selectedDate != null);
|
|
D.assert(onChanged != null);
|
|
D.assert(!(firstDate > lastDate));
|
|
D.assert(!(selectedDate < firstDate));
|
|
D.assert(!(selectedDate > lastDate));
|
|
this.currentDate = currentDate.Value;
|
|
this.displayedMonth = displayedMonth.Value;
|
|
this.firstDate = firstDate.Value;
|
|
this.lastDate = lastDate.Value;
|
|
this.selectedDate = selectedDate.Value;
|
|
this.onChanged = onChanged;
|
|
this.selectableDayPredicate = selectableDayPredicate;
|
|
}
|
|
|
|
public readonly DateTime selectedDate;
|
|
|
|
public readonly DateTime currentDate;
|
|
|
|
public readonly ValueChanged<DateTime> onChanged;
|
|
|
|
public readonly DateTime firstDate;
|
|
|
|
public readonly DateTime lastDate;
|
|
|
|
public readonly DateTime displayedMonth;
|
|
|
|
public readonly material_.SelectableDayPredicate selectableDayPredicate;
|
|
|
|
public override Widget build(BuildContext context) {
|
|
ColorScheme colorScheme = Theme.of(context).colorScheme;
|
|
MaterialLocalizations localizations = MaterialLocalizations.of(context);
|
|
TextTheme textTheme = Theme.of(context).textTheme;
|
|
TextStyle dayStyle = textTheme.caption;
|
|
Color enabledDayColor = colorScheme.onSurface.withOpacity(0.87f);
|
|
Color disabledDayColor = colorScheme.onSurface.withOpacity(0.38f);
|
|
Color selectedDayColor = colorScheme.onPrimary;
|
|
Color selectedDayBackground = colorScheme.primary;
|
|
Color todayColor = colorScheme.primary;
|
|
|
|
int year = displayedMonth.Year;
|
|
int month = displayedMonth.Month;
|
|
|
|
int daysInMonth = utils.getDaysInMonth(year, month);
|
|
int dayOffset = utils.firstDayOffset(year, month, localizations);
|
|
|
|
List<Widget> dayItems = new List<Widget>();
|
|
// 1-based day of month, e.g. 1-31 for January, and 1-29 for February on
|
|
// a leap year.
|
|
int day = -dayOffset;
|
|
while (day < daysInMonth) {
|
|
day++;
|
|
if (day < 1) {
|
|
dayItems.Add(new Container());
|
|
}
|
|
else {
|
|
DateTime dayToBuild = new DateTime(year, month, day);
|
|
bool isDisabled = dayToBuild > lastDate ||
|
|
dayToBuild < firstDate ||
|
|
(selectableDayPredicate != null && !selectableDayPredicate(dayToBuild));
|
|
|
|
BoxDecoration decoration = null;
|
|
Color dayColor = enabledDayColor;
|
|
bool isSelectedDay = utils.isSameDay(selectedDate, dayToBuild);
|
|
if (isSelectedDay) {
|
|
// The selected day gets a circle background highlight, and a
|
|
// contrasting text color.
|
|
dayColor = selectedDayColor;
|
|
decoration = new BoxDecoration(
|
|
color: selectedDayBackground,
|
|
shape: BoxShape.circle
|
|
);
|
|
}
|
|
else if (isDisabled) {
|
|
dayColor = disabledDayColor;
|
|
}
|
|
else if (utils.isSameDay(currentDate, dayToBuild)) {
|
|
// The current day gets a different text color and a circle stroke
|
|
// border.
|
|
dayColor = todayColor;
|
|
decoration = new BoxDecoration(
|
|
border: Border.all(color: todayColor, width: 1),
|
|
shape: BoxShape.circle
|
|
);
|
|
}
|
|
|
|
Widget dayWidget = new Container(
|
|
decoration: decoration,
|
|
child: new Center(
|
|
child: new Text(localizations.formatDecimal(day), style: dayStyle.apply(color: dayColor))
|
|
)
|
|
);
|
|
|
|
if (!isDisabled) {
|
|
dayWidget = new GestureDetector(
|
|
behavior: HitTestBehavior.opaque,
|
|
onTap: () => onChanged(dayToBuild),
|
|
child: dayWidget
|
|
);
|
|
}
|
|
|
|
dayItems.Add(dayWidget);
|
|
}
|
|
}
|
|
|
|
return new Padding(
|
|
padding: EdgeInsets.symmetric(
|
|
horizontal: material_._monthPickerHorizontalPadding
|
|
),
|
|
child: GridView.custom(
|
|
physics: new ClampingScrollPhysics(),
|
|
gridDelegate: material_._dayPickerGridDelegate,
|
|
childrenDelegate: new SliverChildListDelegate(
|
|
dayItems,
|
|
addRepaintBoundaries: false
|
|
)
|
|
)
|
|
);
|
|
}
|
|
}
|
|
|
|
class _DayPickerGridDelegate : SliverGridDelegate {
|
|
internal _DayPickerGridDelegate() {
|
|
}
|
|
|
|
const int daysPerWeek = 7;
|
|
|
|
public override SliverGridLayout getLayout(SliverConstraints constraints) {
|
|
const int columnCount = daysPerWeek;
|
|
float tileWidth = constraints.crossAxisExtent / columnCount;
|
|
float tileHeight = Mathf.Min(material_._dayPickerRowHeight,
|
|
constraints.viewportMainAxisExtent / material_._maxDayPickerRowCount);
|
|
return new SliverGridRegularTileLayout(
|
|
childCrossAxisExtent: tileWidth,
|
|
childMainAxisExtent: tileHeight,
|
|
crossAxisCount: columnCount,
|
|
crossAxisStride: tileWidth,
|
|
mainAxisStride: tileHeight,
|
|
reverseCrossAxis: AxisUtils.axisDirectionIsReversed(constraints.crossAxisDirection)
|
|
);
|
|
}
|
|
|
|
public override bool shouldRelayout(SliverGridDelegate sliverGridDelegate) => false;
|
|
}
|
|
|
|
public partial class material_ {
|
|
internal static readonly _DayPickerGridDelegate _dayPickerGridDelegate = new _DayPickerGridDelegate();
|
|
}
|
|
|
|
class _DayHeaders : StatelessWidget {
|
|
List<Widget> _getDayHeaders(TextStyle headerStyle, MaterialLocalizations localizations) {
|
|
List<Widget> result = new List<Widget>();
|
|
for (int i = localizations.firstDayOfWeekIndex; true; i = (i + 1) % 7) {
|
|
string weekday = localizations.narrowWeekdays[i];
|
|
result.Add(new Center(child: new Text(weekday, style: headerStyle)));
|
|
if (i == (localizations.firstDayOfWeekIndex - 1 + 7) % 7)
|
|
break;
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
public override Widget build(BuildContext context) {
|
|
ThemeData theme = Theme.of(context);
|
|
ColorScheme colorScheme = theme.colorScheme;
|
|
TextStyle dayHeaderStyle = theme.textTheme.caption?.apply(
|
|
color: colorScheme.onSurface.withOpacity(0.60f)
|
|
);
|
|
MaterialLocalizations localizations = MaterialLocalizations.of(context);
|
|
List<Widget>
|
|
labels = _getDayHeaders(dayHeaderStyle, localizations);
|
|
|
|
return new Padding(
|
|
padding: EdgeInsets.symmetric(horizontal: material_._monthPickerHorizontalPadding),
|
|
child:
|
|
GridView.custom(
|
|
shrinkWrap: true,
|
|
gridDelegate: material_._dayPickerGridDelegate,
|
|
childrenDelegate: new SliverChildListDelegate(
|
|
labels,
|
|
addRepaintBoundaries: false
|
|
)
|
|
)
|
|
);
|
|
}
|
|
}
|
|
|
|
class _YearPicker : StatefulWidget {
|
|
internal _YearPicker(
|
|
Key key = null,
|
|
DateTime? currentDate = null,
|
|
DateTime? firstDate = null,
|
|
DateTime? lastDate = null,
|
|
DateTime? initialDate = null,
|
|
DateTime? selectedDate = null,
|
|
ValueChanged<DateTime> onChanged = null
|
|
) : base(key: key) {
|
|
D.assert(currentDate != null);
|
|
D.assert(firstDate != null);
|
|
D.assert(lastDate != null);
|
|
D.assert(initialDate != null);
|
|
D.assert(selectedDate != null);
|
|
D.assert(onChanged != null);
|
|
D.assert(!(firstDate > lastDate));
|
|
this.currentDate = currentDate.Value;
|
|
this.firstDate = firstDate.Value;
|
|
this.lastDate = lastDate.Value;
|
|
this.initialDate = initialDate.Value;
|
|
this.selectedDate = selectedDate.Value;
|
|
this.onChanged = onChanged;
|
|
}
|
|
|
|
public readonly DateTime currentDate;
|
|
|
|
public readonly DateTime firstDate;
|
|
|
|
public readonly DateTime lastDate;
|
|
|
|
public readonly DateTime initialDate;
|
|
|
|
public readonly DateTime selectedDate;
|
|
|
|
public readonly ValueChanged<DateTime> onChanged;
|
|
|
|
public override
|
|
State createState() => new _YearPickerState();
|
|
}
|
|
|
|
class _YearPickerState : State<_YearPicker> {
|
|
ScrollController scrollController;
|
|
|
|
// The approximate number of years necessary to fill the available space.
|
|
public const int minYears = 18;
|
|
|
|
public override void initState() {
|
|
base.initState();
|
|
|
|
// Set the scroll position to approximately center the initial year.
|
|
int initialYearIndex = widget.selectedDate.Year - widget.firstDate.Year;
|
|
int initialYearRow = (int) (1.0f * initialYearIndex / material_._yearPickerColumnCount);
|
|
// Move the offset down by 2 rows to approximately center it.
|
|
int centeredYearRow = initialYearRow - 2;
|
|
float scrollOffset = _itemCount < minYears ? 0 : centeredYearRow * material_._yearPickerRowHeight;
|
|
scrollController = new ScrollController(initialScrollOffset: scrollOffset);
|
|
}
|
|
|
|
Widget _buildYearItem(BuildContext context, int index) {
|
|
ColorScheme colorScheme = Theme.of(context).colorScheme;
|
|
TextTheme textTheme = Theme.of(context).textTheme;
|
|
|
|
// Backfill the _YearPicker with disabled years if necessary.
|
|
int offset = _itemCount < minYears ? (int) (1.0f * (minYears - _itemCount) / 2) : 0;
|
|
int year = widget.firstDate.Year + index - offset;
|
|
bool isSelected = year == widget.selectedDate.Year;
|
|
bool isCurrentYear = year == widget.currentDate.Year;
|
|
bool isDisabled = year < widget.firstDate.Year || year > widget.lastDate.Year;
|
|
const float decorationHeight = 36.0f;
|
|
const float decorationWidth = 72.0f;
|
|
|
|
Color textColor;
|
|
if (isSelected) {
|
|
textColor = colorScheme.onPrimary;
|
|
}
|
|
else if (isDisabled) {
|
|
textColor = colorScheme.onSurface.withOpacity(0.38f);
|
|
}
|
|
else if (isCurrentYear) {
|
|
textColor = colorScheme.primary;
|
|
}
|
|
else {
|
|
textColor = colorScheme.onSurface.withOpacity(0.87f);
|
|
}
|
|
|
|
TextStyle itemStyle = textTheme.bodyText1?.apply(color: textColor);
|
|
|
|
BoxDecoration decoration = null;
|
|
if (isSelected) {
|
|
decoration = new BoxDecoration(
|
|
color: colorScheme.primary,
|
|
borderRadius: BorderRadius.circular(decorationHeight / 2),
|
|
shape: BoxShape.rectangle
|
|
);
|
|
}
|
|
else if (isCurrentYear && !isDisabled) {
|
|
decoration = new BoxDecoration(
|
|
border: Border.all(
|
|
color: colorScheme.primary,
|
|
width: 1
|
|
),
|
|
borderRadius: BorderRadius.circular(decorationHeight / 2),
|
|
shape: BoxShape.rectangle
|
|
);
|
|
}
|
|
|
|
Widget yearItem = new Center(
|
|
child: new Container(
|
|
decoration: decoration,
|
|
height: decorationHeight,
|
|
width: decorationWidth,
|
|
child: new Center(
|
|
child: new Text(year.ToString(), style: itemStyle)
|
|
)
|
|
)
|
|
);
|
|
|
|
if (!isDisabled) {
|
|
yearItem = new InkWell(
|
|
key: new ValueKey<int>(year),
|
|
onTap: () => {
|
|
widget.onChanged(
|
|
new DateTime(
|
|
year,
|
|
widget.initialDate.Month,
|
|
widget.initialDate.Day
|
|
)
|
|
);
|
|
},
|
|
child: yearItem
|
|
);
|
|
}
|
|
|
|
return yearItem;
|
|
}
|
|
|
|
int _itemCount {
|
|
get { return widget.lastDate.Year - widget.firstDate.Year + 1; }
|
|
}
|
|
|
|
public override Widget build(BuildContext context) {
|
|
return new Column(
|
|
children: new List<Widget> {
|
|
new Divider(),
|
|
new Expanded(
|
|
child: GridView.builder(
|
|
controller: scrollController,
|
|
gridDelegate: material_._yearPickerGridDelegate,
|
|
itemBuilder: _buildYearItem,
|
|
itemCount: Math.Max(_itemCount, minYears),
|
|
padding: EdgeInsets.symmetric(horizontal: material_._yearPickerPadding)
|
|
)
|
|
),
|
|
new Divider(),
|
|
}
|
|
);
|
|
}
|
|
}
|
|
|
|
class _YearPickerGridDelegate : SliverGridDelegate {
|
|
internal _YearPickerGridDelegate() {
|
|
}
|
|
|
|
public override SliverGridLayout getLayout(SliverConstraints constraints) {
|
|
float tileWidth =
|
|
(constraints.crossAxisExtent -
|
|
(material_._yearPickerColumnCount - 1) * material_._yearPickerRowSpacing) /
|
|
material_._yearPickerColumnCount;
|
|
return new SliverGridRegularTileLayout(
|
|
childCrossAxisExtent: tileWidth,
|
|
childMainAxisExtent: material_._yearPickerRowHeight,
|
|
crossAxisCount: material_._yearPickerColumnCount,
|
|
crossAxisStride: tileWidth + material_._yearPickerRowSpacing,
|
|
mainAxisStride: material_._yearPickerRowHeight,
|
|
reverseCrossAxis: AxisUtils.axisDirectionIsReversed(constraints.crossAxisDirection)
|
|
);
|
|
}
|
|
|
|
public override bool shouldRelayout(SliverGridDelegate sliverGridDelegate) => false;
|
|
}
|
|
|
|
public partial class material_ {
|
|
internal static readonly _YearPickerGridDelegate _yearPickerGridDelegate = new _YearPickerGridDelegate();
|
|
}
|
|
}
|