如何在Flutter上禁用特定日期列表
问题内容:
我有一个代码调用showDatePicker,我的问题是如何禁用特定日期,一个,两个或多个json日期的列表?我尝试在禁用星期六和星期日的情况下使用selectableDayPredicate,但是如何使用api中的日期的json列表来禁用呢?
onTap: () {
showDatePicker(
context: context,
initialDate:
DateTime.now().add(Duration(days: 2)),
firstDate:
DateTime.now().add(Duration(days: 1)),
lastDate:
DateTime.now().add(Duration(days: 730)),
selectableDayPredicate: (DateTime val) =>
val.weekday == 5 || val.weekday == 6
? false
: true,
);
}
问题答案:
这很大程度上取决于您从API获得的数据。我无法帮助您,但我可以提供谓词。
假设您已经从服务器中检索了包含一串字符串的json列表,并将其解析为DateTime对象(可能使用类似DateTime.parse的东西),那么它就相对简单了。
基本上,您需要检查每个日期(使用selectableDayPredicate),如果不在列表中,则返回true。
首先,您需要清理DateTime对象,因为它们可能包含小时,秒,毫秒等。并且您需要将它们设置为可以快速搜索的格式-
可能不是DateTime。DateTimes从技术上来说是可行的,因为它们确实具有hashCode实现,但是无论如何,您都必须创建仅包含年,月和日的新DateTime对象(因为技术上+1毫秒的日期时间与没有它的日期时间不同)。如果您真的想这样做,则可以更改sanitizeDateTime
下面的函数以返回a
DateTime
和getDateSet
返回a
Set<DateTime>
,但是我建议将它们制作为Strings,因为它们在集合中可以很好地工作并且很简单。
因此,您需要一个类似以下的函数:
String sanitizeDateTime(DateTime dateTime) => "${dateTime.year}-${dateTime.month}-${dateTime.day}";
而且无论您在何处获得json,都希望使用该json对其进行格式化,然后将其插入到集合中。
Set<String> getDateSet(List<DateTime> dates) => dates.map(sanitizeDateTime).toSet();
重要的是要有一组字符串而不是一个列表,因为selectableDayPredicate将运行很多次,并且如果每次执行时都要遍历整个不可选择日期的列表(这就是List
contains
所做的),则可能因为它很慢,所以要检查的日期数和不可选择的日期数O(n*m)
在哪里-
而如果使用集合,则使用更有效的哈希计算,然后进行查找,结果为。这可以解释为过早的优化,但是在这种情况下,我认为这是一个值得的优化。n``m``contains``O(n)
一旦有了一组不可选择的日期,您只需要在谓词中对它们进行检查即可。
即
class .... {
Set<String> unselectableDates; // assuming this is set somewhere
....
onTap: () {
showDatePicker(
...
selectableDayPredicate: (DateTime val) {
String sanitized = sanitizeDateTime(val);
return !unselectableDates.contains(sanitized);
}
}
}