import 'package:flutter/foundation.dart'; import 'package:flutter/material.dart'; import 'package:flutter_slidable/flutter_slidable.dart'; import '../theme.dart'; class SliderTileAction { const SliderTileAction({ required this.actionScale, required this.onPressed, this.key, this.icon, this.label, }); final Key? key; final ScaleKind actionScale; final String? label; final IconData? icon; final SlidableActionCallback? onPressed; } class SliderTile extends StatelessWidget { const SliderTile( {required this.disabled, required this.selected, required this.tileScale, required this.title, this.subtitle = '', this.endActions = const [], this.startActions = const [], this.onTap, this.icon, super.key}); final bool disabled; final bool selected; final ScaleKind tileScale; final List endActions; final List startActions; final GestureTapCallback? onTap; final IconData? icon; final String title; final String subtitle; @override void debugFillProperties(DiagnosticPropertiesBuilder properties) { super.debugFillProperties(properties); properties ..add(DiagnosticsProperty('disabled', disabled)) ..add(DiagnosticsProperty('selected', selected)) ..add(DiagnosticsProperty('tileScale', tileScale)) ..add(IterableProperty('endActions', endActions)) ..add(IterableProperty('startActions', startActions)) ..add(ObjectFlagProperty.has('onTap', onTap)) ..add(DiagnosticsProperty('icon', icon)) ..add(StringProperty('title', title)) ..add(StringProperty('subtitle', subtitle)); } @override // ignore: prefer_expression_function_bodies Widget build(BuildContext context) { final theme = Theme.of(context); final scale = theme.extension()!; final tileColor = scale.scale(!disabled ? tileScale : ScaleKind.gray); final scaleConfig = theme.extension()!; final borderColor = selected ? tileColor.hoverBorder : tileColor.border; final backgroundColor = scaleConfig.useVisualIndicators && !selected ? tileColor.borderText : borderColor; final textColor = scaleConfig.useVisualIndicators && !selected ? borderColor : tileColor.borderText; return Container( clipBehavior: Clip.antiAlias, decoration: ShapeDecoration( color: backgroundColor, shape: RoundedRectangleBorder( side: scaleConfig.useVisualIndicators ? BorderSide(width: 2, color: borderColor, strokeAlign: 0) : BorderSide.none, borderRadius: BorderRadius.circular(8 * scaleConfig.borderRadiusScale), )), child: Slidable( // Specify a key if the Slidable is dismissible. key: key, endActionPane: endActions.isEmpty ? null : ActionPane( motion: const DrawerMotion(), children: endActions .map( (a) => SlidableAction( onPressed: disabled ? null : a.onPressed, backgroundColor: scaleConfig.useVisualIndicators ? (selected ? tileColor.borderText : tileColor.border) : scale.scale(a.actionScale).primary, foregroundColor: scaleConfig.useVisualIndicators ? (selected ? tileColor.border : tileColor.borderText) : scale.scale(a.actionScale).primaryText, icon: subtitle.isNotEmpty ? a.icon : null, label: a.label, padding: const EdgeInsets.all(2)), ) .toList()), startActionPane: startActions.isEmpty ? null : ActionPane( motion: const DrawerMotion(), children: startActions .map( (a) => SlidableAction( onPressed: disabled ? null : a.onPressed, backgroundColor: scaleConfig.useVisualIndicators ? (selected ? tileColor.borderText : tileColor.border) : scale.scale(a.actionScale).primary, foregroundColor: scaleConfig.useVisualIndicators ? (selected ? tileColor.border : tileColor.borderText) : scale.scale(a.actionScale).primaryText, icon: subtitle.isNotEmpty ? a.icon : null, label: a.label, padding: const EdgeInsets.all(2)), ) .toList()), child: Padding( padding: scaleConfig.useVisualIndicators ? EdgeInsets.zero : const EdgeInsets.fromLTRB(0, 2, 0, 2), child: ListTile( onTap: onTap, dense: true, visualDensity: const VisualDensity(vertical: -4), title: Text( title, overflow: TextOverflow.fade, softWrap: false, ), subtitle: subtitle.isNotEmpty ? Text(subtitle) : null, iconColor: textColor, textColor: textColor, leading: icon == null ? null : Icon(icon))))); } }