@@ -24,17 +24,20 @@ const Duration _kDropdownMenuDuration = Duration(milliseconds: 300);
24
24
const double _kMenuItemHeight = kMinInteractiveDimension;
25
25
const double _kDenseButtonHeight = 24.0 ;
26
26
const EdgeInsets _kMenuItemPadding = EdgeInsets .symmetric (horizontal: 16.0 );
27
- const EdgeInsetsGeometry _kAlignedButtonPadding = EdgeInsetsDirectional .only (start: 16.0 , end: 4.0 );
27
+ const EdgeInsetsGeometry _kAlignedButtonPadding =
28
+ EdgeInsetsDirectional .only (start: 16.0 , end: 4.0 );
28
29
const EdgeInsets _kUnalignedButtonPadding = EdgeInsets .zero;
29
30
30
31
/// A builder to customize the selected menu item.
31
- typedef SelectedMenuItemBuilder = Widget Function (BuildContext context, Widget child);
32
+ typedef SelectedMenuItemBuilder = Widget Function (
33
+ BuildContext context, Widget child);
32
34
33
35
/// Signature for the callback that's called when when the dropdown menu opens or closes.
34
36
typedef OnMenuStateChangeFn = void Function (bool isOpen);
35
37
36
38
/// Signature for the callback for the match function used for searchable dropdowns.
37
- typedef SearchMatchFn <T > = bool Function (DropdownItem <T > item, String searchValue);
39
+ typedef SearchMatchFn <T > = bool Function (
40
+ DropdownItem <T > item, String searchValue);
38
41
39
42
/// A Material Design button for selecting from a list of items.
40
43
///
@@ -361,7 +364,8 @@ class DropdownButton2<T> extends StatefulWidget {
361
364
}
362
365
363
366
// ignore: public_member_api_docs
364
- class DropdownButton2State <T > extends State <DropdownButton2 <T >> with WidgetsBindingObserver {
367
+ class DropdownButton2State <T > extends State <DropdownButton2 <T >>
368
+ with WidgetsBindingObserver {
365
369
int ? _selectedIndex;
366
370
_DropdownRoute <T >? _dropdownRoute;
367
371
Orientation ? _lastOrientation;
@@ -439,13 +443,17 @@ class DropdownButton2State<T> extends State<DropdownButton2<T>> with WidgetsBind
439
443
widget.items! .isEmpty ||
440
444
(widget.value == null &&
441
445
widget.items!
442
- .where ((DropdownItem <T > item) => item.enabled && item.value == widget.value)
446
+ .where ((DropdownItem <T > item) =>
447
+ item.enabled && item.value == widget.value)
443
448
.isEmpty)) {
444
449
_selectedIndex = null ;
445
450
return ;
446
451
}
447
452
448
- assert (widget.items! .where ((DropdownItem <T > item) => item.value == widget.value).length == 1 );
453
+ assert (widget.items!
454
+ .where ((DropdownItem <T > item) => item.value == widget.value)
455
+ .length ==
456
+ 1 );
449
457
for (int itemIndex = 0 ; itemIndex < widget.items! .length; itemIndex++ ) {
450
458
if (widget.items! [itemIndex].value == widget.value) {
451
459
_selectedIndex = itemIndex;
@@ -468,18 +476,20 @@ class DropdownButton2State<T> extends State<DropdownButton2<T>> with WidgetsBind
468
476
_rect.value = newRect;
469
477
}
470
478
471
- TextStyle ? get _textStyle => widget.style ?? Theme .of (context).textTheme.titleMedium;
479
+ TextStyle ? get _textStyle =>
480
+ widget.style ?? Theme .of (context).textTheme.titleMedium;
472
481
473
482
Rect _getRect () {
474
483
final TextDirection ? textDirection = Directionality .maybeOf (context);
475
484
const EdgeInsetsGeometry menuMargin = EdgeInsets .zero;
476
485
final NavigatorState navigator = Navigator .of (context,
477
- rootNavigator: _dropdownStyle.isFullScreen ?? _dropdownStyle.useRootNavigator);
486
+ rootNavigator:
487
+ _dropdownStyle.isFullScreen ?? _dropdownStyle.useRootNavigator);
478
488
479
489
final RenderBox itemBox = context.findRenderObject ()! as RenderBox ;
480
- final Rect itemRect =
481
- itemBox. localToGlobal ( Offset .zero, ancestor: navigator.context.findRenderObject ()) &
482
- itemBox.size;
490
+ final Rect itemRect = itemBox. localToGlobal ( Offset .zero,
491
+ ancestor: navigator.context.findRenderObject ()) &
492
+ itemBox.size;
483
493
484
494
return menuMargin.resolve (textDirection).inflateRect (itemRect);
485
495
}
@@ -494,7 +504,8 @@ class DropdownButton2State<T> extends State<DropdownButton2<T>> with WidgetsBind
494
504
495
505
void _handleTap () {
496
506
final NavigatorState navigator = Navigator .of (context,
497
- rootNavigator: _dropdownStyle.isFullScreen ?? _dropdownStyle.useRootNavigator);
507
+ rootNavigator:
508
+ _dropdownStyle.isFullScreen ?? _dropdownStyle.useRootNavigator);
498
509
499
510
final items = widget.items! ;
500
511
final separator = widget.dropdownSeparator;
@@ -506,12 +517,13 @@ class DropdownButton2State<T> extends State<DropdownButton2<T>> with WidgetsBind
506
517
buttonRect: _rect,
507
518
selectedIndex: _selectedIndex ?? 0 ,
508
519
isNoSelectedItem: _selectedIndex == null ,
509
- capturedThemes: InheritedTheme .capture (from: context, to: navigator.context),
520
+ capturedThemes:
521
+ InheritedTheme .capture (from: context, to: navigator.context),
510
522
style: _textStyle! ,
511
523
barrierDismissible: widget.barrierDismissible,
512
524
barrierColor: widget.barrierColor,
513
- barrierLabel:
514
- widget.barrierLabel ?? MaterialLocalizations .of (context).modalBarrierDismissLabel,
525
+ barrierLabel: widget.barrierLabel ??
526
+ MaterialLocalizations .of (context).modalBarrierDismissLabel,
515
527
parentFocusNode: _focusNode,
516
528
enableFeedback: widget.enableFeedback ?? true ,
517
529
dropdownStyle: _dropdownStyle,
@@ -527,7 +539,9 @@ class DropdownButton2State<T> extends State<DropdownButton2<T>> with WidgetsBind
527
539
WidgetsBinding .instance.addPostFrameCallback ((_) {
528
540
_dropdownRoute? ._childNode.requestFocus ();
529
541
});
530
- navigator.push (_dropdownRoute! ).then <void >((_DropdownRouteResult <T >? newValue) {
542
+ navigator
543
+ .push (_dropdownRoute! )
544
+ .then <void >((_DropdownRouteResult <T >? newValue) {
531
545
_removeDropdownRoute ();
532
546
_isMenuOpen.value = false ;
533
547
widget.onMenuStateChange? .call (false );
@@ -550,10 +564,11 @@ class DropdownButton2State<T> extends State<DropdownButton2<T>> with WidgetsBind
550
564
// would be clipped.
551
565
double get _denseButtonHeight {
552
566
final double textScaleFactor = MediaQuery .textScaleFactorOf (context);
553
- final double fontSize =
554
- _textStyle ! .fontSize ?? Theme .of (context).textTheme.titleMedium! .fontSize! ;
567
+ final double fontSize = _textStyle ! .fontSize ??
568
+ Theme .of (context).textTheme.titleMedium! .fontSize! ;
555
569
final double scaledFontSize = textScaleFactor * fontSize;
556
- return math.max (scaledFontSize, math.max (_iconStyle.iconSize, _kDenseButtonHeight));
570
+ return math.max (
571
+ scaledFontSize, math.max (_iconStyle.iconSize, _kDenseButtonHeight));
557
572
}
558
573
559
574
Color get _iconColor {
@@ -583,7 +598,10 @@ class DropdownButton2State<T> extends State<DropdownButton2<T>> with WidgetsBind
583
598
}
584
599
}
585
600
586
- bool get _enabled => widget.items != null && widget.items! .isNotEmpty && widget.onChanged != null ;
601
+ bool get _enabled =>
602
+ widget.items != null &&
603
+ widget.items! .isNotEmpty &&
604
+ widget.onChanged != null ;
587
605
588
606
Orientation _getOrientation (BuildContext context) {
589
607
// TODO(Ahmed): use maybeOrientationOf [flutter>=v3.10.0].
@@ -594,7 +612,9 @@ class DropdownButton2State<T> extends State<DropdownButton2<T>> with WidgetsBind
594
612
// TODO(Ahmed): use View.of(context) and update the comment [flutter>=v3.10.0].
595
613
// ignore: deprecated_member_use
596
614
final Size size = WidgetsBinding .instance.window.physicalSize;
597
- result = size.width > size.height ? Orientation .landscape : Orientation .portrait;
615
+ result = size.width > size.height
616
+ ? Orientation .landscape
617
+ : Orientation .portrait;
598
618
}
599
619
return result;
600
620
}
@@ -635,7 +655,8 @@ class DropdownButton2State<T> extends State<DropdownButton2<T>> with WidgetsBind
635
655
636
656
int ? hintIndex;
637
657
if (widget.hint != null || (! _enabled && widget.disabledHint != null )) {
638
- final Widget displayedHint = _enabled ? widget.hint! : widget.disabledHint ?? widget.hint! ;
658
+ final Widget displayedHint =
659
+ _enabled ? widget.hint! : widget.disabledHint ?? widget.hint! ;
639
660
640
661
hintIndex = buttonItems.length;
641
662
buttonItems.add (DefaultTextStyle (
@@ -649,8 +670,9 @@ class DropdownButton2State<T> extends State<DropdownButton2<T>> with WidgetsBind
649
670
));
650
671
}
651
672
652
- final EdgeInsetsGeometry padding =
653
- ButtonTheme .of (context).alignedDropdown ? _kAlignedButtonPadding : _kUnalignedButtonPadding;
673
+ final EdgeInsetsGeometry padding = ButtonTheme .of (context).alignedDropdown
674
+ ? _kAlignedButtonPadding
675
+ : _kUnalignedButtonPadding;
654
676
655
677
// If value is null (then _selectedIndex is null) then we
656
678
// display the hint or nothing at all.
@@ -663,9 +685,10 @@ class DropdownButton2State<T> extends State<DropdownButton2<T>> with WidgetsBind
663
685
//from the maximum width of menu items or the hint text (width of IndexedStack).
664
686
//We need to add MenuHorizontalPadding so menu width adapts to max items width with padding properly
665
687
padding: EdgeInsets .symmetric (
666
- horizontal: _buttonStyle? .width == null && _dropdownStyle.width == null
667
- ? _getMenuHorizontalPadding ()
668
- : 0.0 ,
688
+ horizontal:
689
+ _buttonStyle? .width == null && _dropdownStyle.width == null
690
+ ? _getMenuHorizontalPadding ()
691
+ : 0.0 ,
669
692
),
670
693
child: IndexedStack (
671
694
index: _selectedIndex ?? hintIndex,
@@ -676,7 +699,9 @@ class DropdownButton2State<T> extends State<DropdownButton2<T>> with WidgetsBind
676
699
: buttonItems.mapIndexed ((item, index) {
677
700
return SizedBox (
678
701
// hintIndex is the last item in buttonItems.
679
- height: index == hintIndex ? _kMenuItemHeight : widget.items! [index].height,
702
+ height: index == hintIndex
703
+ ? _kMenuItemHeight
704
+ : widget.items! [index].height,
680
705
child: item,
681
706
);
682
707
}).toList (),
@@ -685,21 +710,28 @@ class DropdownButton2State<T> extends State<DropdownButton2<T>> with WidgetsBind
685
710
}
686
711
687
712
Widget result = DefaultTextStyle (
688
- style: _enabled ? _textStyle! : _textStyle! .copyWith (color: Theme .of (context).disabledColor),
713
+ style: _enabled
714
+ ? _textStyle!
715
+ : _textStyle! .copyWith (color: Theme .of (context).disabledColor),
689
716
child: widget.customButton ??
690
717
Container (
691
718
decoration: _buttonStyle? .decoration? .copyWith (
692
719
boxShadow: _buttonStyle! .decoration! .boxShadow ??
693
720
kElevationToShadow[_buttonStyle! .elevation ?? 0 ],
694
721
),
695
- padding: _buttonStyle? .padding ?? padding.resolve (Directionality .of (context)),
696
- height: _buttonStyle? .height ?? (widget.isDense ? _denseButtonHeight : null ),
722
+ padding: _buttonStyle? .padding ??
723
+ padding.resolve (Directionality .of (context)),
724
+ height: _buttonStyle? .height ??
725
+ (widget.isDense ? _denseButtonHeight : null ),
697
726
width: _buttonStyle? .width,
698
727
child: Row (
699
728
mainAxisAlignment: MainAxisAlignment .spaceBetween,
700
729
mainAxisSize: MainAxisSize .min,
701
730
children: < Widget > [
702
- if (widget.isExpanded) Expanded (child: innerItemsWidget) else innerItemsWidget,
731
+ if (widget.isExpanded)
732
+ Expanded (child: innerItemsWidget)
733
+ else
734
+ innerItemsWidget,
703
735
IconTheme (
704
736
data: IconThemeData (
705
737
color: _iconColor,
@@ -747,7 +779,8 @@ class DropdownButton2State<T> extends State<DropdownButton2<T>> with WidgetsBind
747
779
);
748
780
}
749
781
750
- final MouseCursor effectiveMouseCursor = MaterialStateProperty .resolveAs <MouseCursor >(
782
+ final MouseCursor effectiveMouseCursor =
783
+ MaterialStateProperty .resolveAs <MouseCursor >(
751
784
MaterialStateMouseCursor .clickable,
752
785
< MaterialState > {
753
786
if (! _enabled) MaterialState .disabled,
@@ -901,23 +934,29 @@ class DropdownButtonFormField2<T> extends FormField<T> {
901
934
builder: (FormFieldState <T > field) {
902
935
final _DropdownButtonFormFieldState <T > state =
903
936
field as _DropdownButtonFormFieldState <T >;
904
- final InputDecoration decorationArg = _getInputDecoration (decoration, buttonStyleData);
905
- final InputDecoration effectiveDecoration = decorationArg.applyDefaults (
937
+ final InputDecoration decorationArg =
938
+ _getInputDecoration (decoration, buttonStyleData);
939
+ final InputDecoration effectiveDecoration =
940
+ decorationArg.applyDefaults (
906
941
Theme .of (field.context).inputDecorationTheme,
907
942
);
908
943
909
944
final bool showSelectedItem = items != null &&
910
- items.where ((DropdownItem <T > item) => item.value == state.value).isNotEmpty;
945
+ items
946
+ .where ((DropdownItem <T > item) => item.value == state.value)
947
+ .isNotEmpty;
911
948
bool isHintOrDisabledHintAvailable () {
912
- final bool isDropdownDisabled = onChanged == null || (items == null || items.isEmpty);
949
+ final bool isDropdownDisabled =
950
+ onChanged == null || (items == null || items.isEmpty);
913
951
if (isDropdownDisabled) {
914
952
return hint != null || disabledHint != null ;
915
953
} else {
916
954
return hint != null ;
917
955
}
918
956
}
919
957
920
- final bool isEmpty = ! showSelectedItem && ! isHintOrDisabledHintAvailable ();
958
+ final bool isEmpty =
959
+ ! showSelectedItem && ! isHintOrDisabledHintAvailable ();
921
960
922
961
// An unFocusable Focus widget so that this widget can detect if its
923
962
// descendants have focus or not.
@@ -954,7 +993,8 @@ class DropdownButtonFormField2<T> extends FormField<T> {
954
993
barrierDismissible: barrierDismissible,
955
994
barrierColor: barrierColor,
956
995
barrierLabel: barrierLabel,
957
- inputDecoration: effectiveDecoration.copyWith (errorText: field.errorText),
996
+ inputDecoration: effectiveDecoration.copyWith (
997
+ errorText: field.errorText),
958
998
isEmpty: isEmpty,
959
999
isFocused: Focus .of (context).hasFocus,
960
1000
),
@@ -987,10 +1027,10 @@ class DropdownButtonFormField2<T> extends FormField<T> {
987
1027
InputDecoration ? decoration, ButtonStyleData ? buttonStyleData) {
988
1028
return decoration ??
989
1029
InputDecoration (
990
- focusColor:
991
- buttonStyleData ? .overlayColor ? .resolve (< MaterialState > {MaterialState .focused}),
992
- hoverColor:
993
- buttonStyleData ? .overlayColor ? .resolve (< MaterialState > {MaterialState .hovered}),
1030
+ focusColor: buttonStyleData ? .overlayColor
1031
+ ? .resolve (< MaterialState > {MaterialState .focused}),
1032
+ hoverColor: buttonStyleData ? .overlayColor
1033
+ ? .resolve (< MaterialState > {MaterialState .hovered}),
994
1034
);
995
1035
}
996
1036
0 commit comments