Commit 475e7ce9 authored by xster's avatar xster Committed by GitHub

Optional general date predicate mechanism (#7858)

parent be7be2b8
...@@ -172,7 +172,8 @@ class DayPicker extends StatelessWidget { ...@@ -172,7 +172,8 @@ class DayPicker extends StatelessWidget {
@required this.onChanged, @required this.onChanged,
@required this.firstDate, @required this.firstDate,
@required this.lastDate, @required this.lastDate,
@required this.displayedMonth @required this.displayedMonth,
this.selectableDayPredicate
}) : super(key: key) { }) : super(key: key) {
assert(selectedDate != null); assert(selectedDate != null);
assert(currentDate != null); assert(currentDate != null);
...@@ -202,6 +203,9 @@ class DayPicker extends StatelessWidget { ...@@ -202,6 +203,9 @@ class DayPicker extends StatelessWidget {
/// The month whose days are displayed by this picker. /// The month whose days are displayed by this picker.
final DateTime displayedMonth; final DateTime displayedMonth;
/// Optional user supplied predicate function to customize selectable days.
final SelectableDayPredicate selectableDayPredicate;
List<Widget> _getDayHeaders(TextStyle headerStyle) { List<Widget> _getDayHeaders(TextStyle headerStyle) {
final DateFormat dateFormat = new DateFormat(); final DateFormat dateFormat = new DateFormat();
final DateSymbols symbols = dateFormat.dateSymbols; final DateSymbols symbols = dateFormat.dateSymbols;
...@@ -230,7 +234,9 @@ class DayPicker extends StatelessWidget { ...@@ -230,7 +234,9 @@ class DayPicker extends StatelessWidget {
labels.add(new Container()); labels.add(new Container());
} else { } else {
final DateTime dayToBuild = new DateTime(year, month, day); final DateTime dayToBuild = new DateTime(year, month, day);
final bool disabled = dayToBuild.isAfter(lastDate) || dayToBuild.isBefore(firstDate); final bool disabled = dayToBuild.isAfter(lastDate)
|| dayToBuild.isBefore(firstDate)
|| (selectableDayPredicate != null && !selectableDayPredicate(dayToBuild));
BoxDecoration decoration; BoxDecoration decoration;
TextStyle itemStyle = themeData.textTheme.body1; TextStyle itemStyle = themeData.textTheme.body1;
...@@ -315,7 +321,8 @@ class MonthPicker extends StatefulWidget { ...@@ -315,7 +321,8 @@ class MonthPicker extends StatefulWidget {
@required this.selectedDate, @required this.selectedDate,
@required this.onChanged, @required this.onChanged,
@required this.firstDate, @required this.firstDate,
@required this.lastDate @required this.lastDate,
this.selectableDayPredicate
}) : super(key: key) { }) : super(key: key) {
assert(selectedDate != null); assert(selectedDate != null);
assert(onChanged != null); assert(onChanged != null);
...@@ -337,6 +344,9 @@ class MonthPicker extends StatefulWidget { ...@@ -337,6 +344,9 @@ class MonthPicker extends StatefulWidget {
/// The latest date the user is permitted to pick. /// The latest date the user is permitted to pick.
final DateTime lastDate; final DateTime lastDate;
/// Optional user supplied predicate function to customize selectable days.
final SelectableDayPredicate selectableDayPredicate;
@override @override
_MonthPickerState createState() => new _MonthPickerState(); _MonthPickerState createState() => new _MonthPickerState();
} }
...@@ -399,7 +409,8 @@ class _MonthPickerState extends State<MonthPicker> { ...@@ -399,7 +409,8 @@ class _MonthPickerState extends State<MonthPicker> {
onChanged: config.onChanged, onChanged: config.onChanged,
firstDate: config.firstDate, firstDate: config.firstDate,
lastDate: config.lastDate, lastDate: config.lastDate,
displayedMonth: monthToBuild displayedMonth: monthToBuild,
selectableDayPredicate: config.selectableDayPredicate
)); ));
} }
return result; return result;
...@@ -570,12 +581,14 @@ class _DatePickerDialog extends StatefulWidget { ...@@ -570,12 +581,14 @@ class _DatePickerDialog extends StatefulWidget {
Key key, Key key,
this.initialDate, this.initialDate,
this.firstDate, this.firstDate,
this.lastDate this.lastDate,
this.selectableDayPredicate
}) : super(key: key); }) : super(key: key);
final DateTime initialDate; final DateTime initialDate;
final DateTime firstDate; final DateTime firstDate;
final DateTime lastDate; final DateTime lastDate;
final SelectableDayPredicate selectableDayPredicate;
@override @override
_DatePickerDialogState createState() => new _DatePickerDialogState(); _DatePickerDialogState createState() => new _DatePickerDialogState();
...@@ -631,7 +644,8 @@ class _DatePickerDialogState extends State<_DatePickerDialog> { ...@@ -631,7 +644,8 @@ class _DatePickerDialogState extends State<_DatePickerDialog> {
selectedDate: _selectedDate, selectedDate: _selectedDate,
onChanged: _handleDayChanged, onChanged: _handleDayChanged,
firstDate: config.firstDate, firstDate: config.firstDate,
lastDate: config.lastDate lastDate: config.lastDate,
selectableDayPredicate: config.selectableDayPredicate
); );
case _DatePickerMode.year: case _DatePickerMode.year:
return new YearPicker( return new YearPicker(
...@@ -717,11 +731,20 @@ class _DatePickerDialogState extends State<_DatePickerDialog> { ...@@ -717,11 +731,20 @@ class _DatePickerDialogState extends State<_DatePickerDialog> {
} }
} }
/// Signature for predicating dates for enabled date selections.
///
/// See [showDatePicker].
typedef bool SelectableDayPredicate(DateTime day);
/// Shows a dialog containing a material design date picker. /// Shows a dialog containing a material design date picker.
/// ///
/// The returned [Future] resolves to the date selected by the user when the /// The returned [Future] resolves to the date selected by the user when the
/// user closes the dialog. If the user cancels the dialog, null is returned. /// user closes the dialog. If the user cancels the dialog, null is returned.
/// ///
/// An optional [selectableDayPredicate] function can be passed in to customize
/// the days to enable for selection. If provided, only the days that
/// [selectableDayPredicate] returned true for will be selectable.
///
/// See also: /// See also:
/// ///
/// * [showTimePicker] /// * [showTimePicker]
...@@ -730,14 +753,23 @@ Future<DateTime> showDatePicker({ ...@@ -730,14 +753,23 @@ Future<DateTime> showDatePicker({
@required BuildContext context, @required BuildContext context,
@required DateTime initialDate, @required DateTime initialDate,
@required DateTime firstDate, @required DateTime firstDate,
@required DateTime lastDate @required DateTime lastDate,
SelectableDayPredicate selectableDayPredicate
}) async { }) async {
assert(!initialDate.isBefore(firstDate), 'initialDate must be on or after firstDate');
assert(!initialDate.isAfter(lastDate), 'initialDate must be on or before lastDate');
assert(!firstDate.isAfter(lastDate), 'lastDate must be on or after firstDate');
assert(
selectableDayPredicate == null || selectableDayPredicate(initialDate),
'Provided initialDate must satisfy provided selectableDayPredicate'
);
return await showDialog( return await showDialog(
context: context, context: context,
child: new _DatePickerDialog( child: new _DatePickerDialog(
initialDate: initialDate, initialDate: initialDate,
firstDate: firstDate, firstDate: firstDate,
lastDate: lastDate lastDate: lastDate,
selectableDayPredicate: selectableDayPredicate
) )
); );
} }
...@@ -10,6 +10,7 @@ void main() { ...@@ -10,6 +10,7 @@ void main() {
DateTime firstDate; DateTime firstDate;
DateTime lastDate; DateTime lastDate;
DateTime initialDate; DateTime initialDate;
SelectableDayPredicate selectableDayPredicate;
setUp(() { setUp(() {
firstDate = new DateTime(2001, DateTime.JANUARY, 1); firstDate = new DateTime(2001, DateTime.JANUARY, 1);
...@@ -144,6 +145,7 @@ void main() { ...@@ -144,6 +145,7 @@ void main() {
initialDate: initialDate, initialDate: initialDate,
firstDate: firstDate, firstDate: firstDate,
lastDate: lastDate, lastDate: lastDate,
selectableDayPredicate: selectableDayPredicate
); );
await tester.pumpUntilNoTransientCallbacks(const Duration(seconds: 1)); await tester.pumpUntilNoTransientCallbacks(const Duration(seconds: 1));
...@@ -257,4 +259,18 @@ void main() { ...@@ -257,4 +259,18 @@ void main() {
expect(await date, equals(new DateTime(2016, DateTime.DECEMBER, 10))); expect(await date, equals(new DateTime(2016, DateTime.DECEMBER, 10)));
}); });
}); });
testWidgets('Only predicate days are selectable', (WidgetTester tester) async {
initialDate = new DateTime(2017, DateTime.JANUARY, 16);
firstDate = new DateTime(2017, DateTime.JANUARY, 10);
lastDate = new DateTime(2017, DateTime.JANUARY, 20);
selectableDayPredicate = (DateTime day) => day.day.isEven;
await preparePicker(tester, (Future<DateTime> date) async {
await tester.tap(find.text('10')); // Even, works.
await tester.tap(find.text('13')); // Odd, doesn't work.
await tester.tap(find.text('17')); // Odd, doesn't work.
await tester.tap(find.text('OK'));
expect(await date, equals(new DateTime(2017, DateTime.JANUARY, 10)));
});
});
} }
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment