FSelectMenuTile<T> constructor

FSelectMenuTile<T>({
  1. required FSelectGroupController<T> groupController,
  2. required Widget title,
  3. required List<FSelectTile<T>> menu,
  4. FPopoverController? popoverController,
  5. ScrollController? scrollController,
  6. FSelectMenuTileStyle? style,
  7. double? cacheExtent,
  8. double maxHeight = double.infinity,
  9. DragStartBehavior dragStartBehavior = DragStartBehavior.start,
  10. FTileDivider divider = FTileDivider.full,
  11. Alignment menuAnchor = Alignment.topRight,
  12. Alignment tileAnchor = Alignment.bottomRight,
  13. Offset shift(
    1. Size,
    2. FPortalTarget,
    3. FPortalFollower
    ) = FPortalFollowerShift.flip,
  14. bool hideOnTapOutside = true,
  15. bool directionPadding = false,
  16. bool autoHide = false,
  17. Widget? label,
  18. Widget? description,
  19. Widget errorBuilder(
    1. BuildContext,
    2. String
    ) = _errorBuilder,
  20. String? semanticLabel,
  21. bool autofocus = false,
  22. FocusNode? focusNode,
  23. ValueChanged<bool>? onFocusChange,
  24. Widget? prefixIcon,
  25. Widget? subtitle,
  26. Widget? details,
  27. Widget? suffixIcon,
  28. FormFieldSetter<Set<T>>? onSaved,
  29. FormFieldValidator<Set<T>>? validator,
  30. Set<T>? initialValue,
  31. String? forceErrorText,
  32. bool enabled = true,
  33. AutovalidateMode? autovalidateMode,
  34. String? restorationId,
  35. Key? key,
})

Creates a FSelectMenuTile.

Implementation

FSelectMenuTile({
  required this.groupController,
  required this.title,
  required List<FSelectTile<T>> menu,
  this.popoverController,
  this.scrollController,
  this.style,
  this.cacheExtent,
  this.maxHeight = double.infinity,
  this.dragStartBehavior = DragStartBehavior.start,
  this.divider = FTileDivider.full,
  this.menuAnchor = Alignment.topRight,
  this.tileAnchor = Alignment.bottomRight,
  this.shift = FPortalFollowerShift.flip,
  this.hideOnTapOutside = true,
  this.directionPadding = false,
  this.autoHide = false,
  this.label,
  this.description,
  this.errorBuilder = _errorBuilder,
  this.semanticLabel,
  this.autofocus = false,
  this.focusNode,
  this.onFocusChange,
  this.prefixIcon,
  this.subtitle,
  this.details,
  this.suffixIcon,
  super.onSaved,
  super.validator,
  super.initialValue,
  super.forceErrorText,
  super.enabled = true,
  super.autovalidateMode,
  super.restorationId,
  super.key,
}) : super(
        builder: (field) {
          final state = field as _State<T>;
          final groupData = FTileGroupData.maybeOf(state.context);
          final tileData = FTileData.maybeOf(state.context);

          final global = state.context.theme.selectMenuTileStyle;
          final labelStyle = style?.labelStyle ?? global.labelStyle;
          final menuStyle = style?.menuStyle ?? global.menuStyle;
          final tileStyle = style?.tileStyle ?? tileData?.style ?? groupData?.style.tileStyle ?? global.tileStyle;

          final (labelState, error) = switch (state.errorText) {
            _ when !enabled => (FLabelState.disabled, null),
            final text? => (FLabelState.error, errorBuilder(state.context, text)),
            null => (FLabelState.enabled, null),
          };

          Widget tile = FPopover(
            // A GlobalObjectKey is used to workaround Flutter not recognizing how widgets move inside the widget tree.
            //
            // OverlayPortalControllers are tied to a single _OverlayPortalState, and conditional rebuilds introduced
            // by FLabel and its internals can cause a new parent to be inserted above FPopover. This leads to the
            // entire widget subtree being rebuilt and losing their state. Consequently, the controller is assigned
            // another _OverlayPortalState, causing an assertion to be thrown.
            //
            // See https://stackoverflow.com/a/59410824/4189771
            key: GlobalObjectKey(state._controller._popover),
            controller: state._controller._popover,
            style: menuStyle,
            followerAnchor: menuAnchor,
            targetAnchor: tileAnchor,
            shift: shift,
            hideOnTapOutside: hideOnTapOutside,
            directionPadding: directionPadding,
            autofocus: autofocus,
            focusNode: focusNode,
            onFocusChange: onFocusChange,
            followerBuilder: (context, _, __) => ConstrainedBox(
              constraints: BoxConstraints(maxWidth: menuStyle.maxWidth),
              child: FSelectTileGroup<T>(
                groupController: state._controller,
                scrollController: scrollController,
                cacheExtent: cacheExtent,
                maxHeight: maxHeight,
                dragStartBehavior: dragStartBehavior,
                style: menuStyle.tileGroupStyle,
                semanticLabel: semanticLabel,
                divider: divider,
                children: menu,
              ),
            ),
            target: FTile(
              style: tileStyle,
              prefixIcon: prefixIcon,
              enabled: enabled,
              title: title,
              subtitle: subtitle,
              details: details,
              suffixIcon: suffixIcon ?? FIcon(FAssets.icons.chevronsUpDown),
              onPress: state._controller._popover.toggle,
            ),
          );

          if (groupData == null && tileData == null && (label != null || description != null || error != null)) {
            tile = FLabel(
              axis: Axis.vertical,
              style: labelStyle,
              state: labelState,
              label: label,
              description: description,
              error: error,
              child: tile,
            );
          }

          return tile;
        },
      );