Compare commits

...

5 Commits

39 changed files with 1668 additions and 605 deletions

View File

@ -19,4 +19,7 @@ import 'package:wyatt_ui_components/src/features/features.dart';
extension ThemeComponentBuildContext on BuildContext { extension ThemeComponentBuildContext on BuildContext {
ComponentThemeData get components => ComponentTheme.of(this); ComponentThemeData get components => ComponentTheme.of(this);
TextTheme get textTheme => Theme.of(this).textTheme;
ColorScheme get colorScheme => Theme.of(this).colorScheme;
ButtonThemeData get buttonTheme => Theme.of(this).buttonTheme;
} }

View File

@ -42,4 +42,14 @@ class MultiColor {
bool get isGradient => bool get isGradient =>
(_colors?.isNotEmpty ?? false) && (_colors?.length ?? 0) > 1; (_colors?.isNotEmpty ?? false) && (_colors?.length ?? 0) > 1;
static MultiColor? lerp(MultiColor? a, MultiColor? b, double t) {
if (a == null && b == null) {
return null;
}
if (b == null) {
return a;
}
return b;
}
} }

View File

@ -30,25 +30,25 @@ abstract class ButtonComponent extends Component {
}); });
/// Style of this button in disabled state /// Style of this button in disabled state
final ButtonStyle? disabledStyle; final ButtonStyle<dynamic>? disabledStyle;
/// Style of this button in normal state /// Style of this button in normal state
final ButtonStyle? normalStyle; final ButtonStyle<dynamic>? normalStyle;
/// Style of this button in hovered state /// Style of this button in hovered state
final ButtonStyle? hoveredStyle; final ButtonStyle<dynamic>? hoveredStyle;
/// Style of this button in focused state /// Style of this button in focused state
final ButtonStyle? focusedStyle; final ButtonStyle<dynamic>? focusedStyle;
/// Style of this button in tapped state /// Style of this button in tapped state
final ButtonStyle? tappedStyle; final ButtonStyle<dynamic>? tappedStyle;
/// Style of this button in selected state /// Style of this button in selected state
final ButtonStyle? selectedStyle; final ButtonStyle<dynamic>? selectedStyle;
/// Style of this button in invalid state /// Style of this button in invalid state
final ButtonStyle? invalidStyle; final ButtonStyle<dynamic>? invalidStyle;
/// Callback on button press /// Callback on button press
final void Function(ControlState state)? onPressed; final void Function(ControlState state)? onPressed;

View File

@ -17,41 +17,49 @@
import 'package:flutter/widgets.dart'; import 'package:flutter/widgets.dart';
import 'package:wyatt_ui_components/src/core/utils/multi_color.dart'; import 'package:wyatt_ui_components/src/core/utils/multi_color.dart';
abstract class ButtonStyle { abstract class ButtonStyle<T> {
const ButtonStyle({ const ButtonStyle({
this.radius = 15, this.radius,
this.padding = 10, this.padding,
this.foregroundColors, this.foregroundColors,
this.backgroundColors, this.backgroundColors,
this.borderColors, this.borderColors,
this.stroke = 2, this.stroke,
this.shadow = const BoxShadow( this.shadow,
blurRadius: 30,
offset: Offset(0, 5),
color: Color.fromRGBO(0, 0, 0, 0.05),
),
}); });
/// Button radius /// Button radius
final double? radius; ///
/// Default to `BorderRadius.all(Radius.circular(4.0))`
final BorderRadiusGeometry? radius;
/// Padding and gaps of this card /// Padding and gaps of this button
final double? padding; ///
/// Default to `ButtonTheme.padding`
final EdgeInsetsGeometry? padding;
/// Button foreground gradient colors (from left to right) /// Button foreground gradient colors (from left to right)
///
/// Default to `Theme.colorScheme.onPrimary`
final MultiColor? foregroundColors; final MultiColor? foregroundColors;
/// Button background gradient colors (from left to right) /// Button background gradient colors (from left to right)
///
/// Default to `Theme.colorScheme.primary`
final MultiColor? backgroundColors; final MultiColor? backgroundColors;
/// Border colors (from left to right). /// Border colors (from left to right).
///
/// Default to `null`
final MultiColor? borderColors; final MultiColor? borderColors;
/// Stroke of the border /// Stroke of the border
///
/// Default to `null`
final double? stroke; final double? stroke;
/// Drop shadow /// Drop shadow
///
/// Default to `null`
final BoxShadow? shadow; final BoxShadow? shadow;
ButtonStyle copyWith();
} }

View File

@ -11,13 +11,15 @@ abstract class $FileSelectionButtonComponentCWProxy {
FileSelectionButtonComponent leading(Widget? leading); FileSelectionButtonComponent leading(Widget? leading);
FileSelectionButtonComponent title(TextWrapper? title); FileSelectionButtonComponent title(TextWrapper? title);
FileSelectionButtonComponent subTitle(TextWrapper? subTitle); FileSelectionButtonComponent subTitle(TextWrapper? subTitle);
FileSelectionButtonComponent disabledStyle(ButtonStyle? disabledStyle); FileSelectionButtonComponent disabledStyle(
FileSelectionButtonComponent normalStyle(ButtonStyle? normalStyle); ButtonStyle<dynamic>? disabledStyle);
FileSelectionButtonComponent hoveredStyle(ButtonStyle? hoveredStyle); FileSelectionButtonComponent normalStyle(ButtonStyle<dynamic>? normalStyle);
FileSelectionButtonComponent focusedStyle(ButtonStyle? focusedStyle); FileSelectionButtonComponent hoveredStyle(ButtonStyle<dynamic>? hoveredStyle);
FileSelectionButtonComponent tappedStyle(ButtonStyle? tappedStyle); FileSelectionButtonComponent focusedStyle(ButtonStyle<dynamic>? focusedStyle);
FileSelectionButtonComponent selectedStyle(ButtonStyle? selectedStyle); FileSelectionButtonComponent tappedStyle(ButtonStyle<dynamic>? tappedStyle);
FileSelectionButtonComponent invalidStyle(ButtonStyle? invalidStyle); FileSelectionButtonComponent selectedStyle(
ButtonStyle<dynamic>? selectedStyle);
FileSelectionButtonComponent invalidStyle(ButtonStyle<dynamic>? invalidStyle);
FileSelectionButtonComponent onPressed( FileSelectionButtonComponent onPressed(
void Function(ControlState)? onPressed); void Function(ControlState)? onPressed);
FileSelectionButtonComponent key(Key? key); FileSelectionButtonComponent key(Key? key);
@ -26,13 +28,13 @@ abstract class $FileSelectionButtonComponentCWProxy {
Widget? leading, Widget? leading,
TextWrapper? title, TextWrapper? title,
TextWrapper? subTitle, TextWrapper? subTitle,
ButtonStyle? disabledStyle, ButtonStyle<dynamic>? disabledStyle,
ButtonStyle? normalStyle, ButtonStyle<dynamic>? normalStyle,
ButtonStyle? hoveredStyle, ButtonStyle<dynamic>? hoveredStyle,
ButtonStyle? focusedStyle, ButtonStyle<dynamic>? focusedStyle,
ButtonStyle? tappedStyle, ButtonStyle<dynamic>? tappedStyle,
ButtonStyle? selectedStyle, ButtonStyle<dynamic>? selectedStyle,
ButtonStyle? invalidStyle, ButtonStyle<dynamic>? invalidStyle,
void Function(ControlState)? onPressed, void Function(ControlState)? onPressed,
Key? key, Key? key,
}); });

View File

@ -14,42 +14,85 @@
// You should have received a copy of the GNU General Public License // You should have received a copy of the GNU General Public License
// along with this program. If not, see <https://www.gnu.org/licenses/>. // along with this program. If not, see <https://www.gnu.org/licenses/>.
import 'dart:ui';
import 'package:copy_with_extension/copy_with_extension.dart';
import 'package:flutter/widgets.dart'; import 'package:flutter/widgets.dart';
import 'package:wyatt_ui_components/src/core/extensions/build_context_extensions.dart';
import 'package:wyatt_ui_components/src/core/utils/multi_color.dart'; import 'package:wyatt_ui_components/src/core/utils/multi_color.dart';
import 'package:wyatt_ui_components/src/domain/entities/buttons/button_style.dart'; import 'package:wyatt_ui_components/src/domain/entities/buttons/button_style.dart';
class FileSelectionButtonStyle extends ButtonStyle { part 'file_selection_button_style.g.dart';
@CopyWith()
class FileSelectionButtonStyle extends ButtonStyle<FileSelectionButtonStyle> {
const FileSelectionButtonStyle({ const FileSelectionButtonStyle({
super.radius = 12, this.title,
super.padding = 13, this.subTitle,
super.radius,
super.padding,
super.foregroundColors, super.foregroundColors,
super.backgroundColors, super.backgroundColors,
super.borderColors, super.borderColors,
super.stroke = 2, super.stroke,
super.shadow = const BoxShadow( super.shadow,
blurRadius: 30,
offset: Offset(0, 5),
color: Color.fromRGBO(0, 0, 0, 0.05),
),
}); });
@override /// Used in negociation to build a style from Flutter default values.
FileSelectionButtonStyle copyWith({ factory FileSelectionButtonStyle.fromFlutter(BuildContext context) =>
double? radius,
double? padding,
MultiColor? foregroundColors,
MultiColor? backgroundColors,
MultiColor? borderColors,
double? stroke,
BoxShadow? shadow,
}) =>
FileSelectionButtonStyle( FileSelectionButtonStyle(
radius: radius ?? this.radius, title: context.textTheme.labelLarge,
padding: padding ?? this.padding, subTitle: context.textTheme.labelSmall,
foregroundColors: foregroundColors ?? this.foregroundColors, radius: (context.buttonTheme.shape is RoundedRectangleBorder)
backgroundColors: backgroundColors ?? this.backgroundColors, ? (context.buttonTheme.shape as RoundedRectangleBorder).borderRadius
borderColors: borderColors ?? this.borderColors, : null,
stroke: stroke ?? this.stroke, padding: context.buttonTheme.padding,
shadow: shadow ?? this.shadow, foregroundColors: MultiColor.single(context.colorScheme.onPrimary),
backgroundColors: MultiColor.single(context.colorScheme.primary),
); );
/// Used for interpolation.
static FileSelectionButtonStyle? lerp(
FileSelectionButtonStyle? a,
FileSelectionButtonStyle? b,
double t,
) {
if (a == null || b == null) {
return null;
}
// b.copyWith to return b attributes even if they are not lerped
return b.copyWith(
title: TextStyle.lerp(a.title, b.title, t),
subTitle: TextStyle.lerp(a.title, b.title, t),
foregroundColors: MultiColor.lerp(
a.foregroundColors,
b.foregroundColors,
t,
),
backgroundColors: MultiColor.lerp(
a.backgroundColors,
b.backgroundColors,
t,
),
borderColors: MultiColor.lerp(
a.borderColors,
b.borderColors,
t,
),
radius: BorderRadiusGeometry.lerp(a.radius, b.radius, t),
padding: EdgeInsetsGeometry.lerp(a.padding, b.padding, t),
stroke: lerpDouble(a.stroke, b.stroke, t),
shadow: BoxShadow.lerp(a.shadow, b.shadow, t),
);
}
/// Title text style
///
/// Default to `TextTheme.labelLarge`
final TextStyle? title;
/// Sub title text style
///
/// Default to `TextTheme.labelSmall`
final TextStyle? subTitle;
} }

View File

@ -0,0 +1,152 @@
// GENERATED CODE - DO NOT MODIFY BY HAND
part of 'file_selection_button_style.dart';
// **************************************************************************
// CopyWithGenerator
// **************************************************************************
abstract class _$FileSelectionButtonStyleCWProxy {
FileSelectionButtonStyle title(TextStyle? title);
FileSelectionButtonStyle subTitle(TextStyle? subTitle);
FileSelectionButtonStyle radius(BorderRadiusGeometry? radius);
FileSelectionButtonStyle padding(EdgeInsetsGeometry? padding);
FileSelectionButtonStyle foregroundColors(MultiColor? foregroundColors);
FileSelectionButtonStyle backgroundColors(MultiColor? backgroundColors);
FileSelectionButtonStyle borderColors(MultiColor? borderColors);
FileSelectionButtonStyle stroke(double? stroke);
FileSelectionButtonStyle shadow(BoxShadow? shadow);
/// This function **does support** nullification of nullable fields. All `null` values passed to `non-nullable` fields will be ignored. You can also use `FileSelectionButtonStyle(...).copyWith.fieldName(...)` to override fields one at a time with nullification support.
///
/// Usage
/// ```dart
/// FileSelectionButtonStyle(...).copyWith(id: 12, name: "My name")
/// ````
FileSelectionButtonStyle call({
TextStyle? title,
TextStyle? subTitle,
BorderRadiusGeometry? radius,
EdgeInsetsGeometry? padding,
MultiColor? foregroundColors,
MultiColor? backgroundColors,
MultiColor? borderColors,
double? stroke,
BoxShadow? shadow,
});
}
/// Proxy class for `copyWith` functionality. This is a callable class and can be used as follows: `instanceOfFileSelectionButtonStyle.copyWith(...)`. Additionally contains functions for specific fields e.g. `instanceOfFileSelectionButtonStyle.copyWith.fieldName(...)`
class _$FileSelectionButtonStyleCWProxyImpl
implements _$FileSelectionButtonStyleCWProxy {
const _$FileSelectionButtonStyleCWProxyImpl(this._value);
final FileSelectionButtonStyle _value;
@override
FileSelectionButtonStyle title(TextStyle? title) => this(title: title);
@override
FileSelectionButtonStyle subTitle(TextStyle? subTitle) =>
this(subTitle: subTitle);
@override
FileSelectionButtonStyle radius(BorderRadiusGeometry? radius) =>
this(radius: radius);
@override
FileSelectionButtonStyle padding(EdgeInsetsGeometry? padding) =>
this(padding: padding);
@override
FileSelectionButtonStyle foregroundColors(MultiColor? foregroundColors) =>
this(foregroundColors: foregroundColors);
@override
FileSelectionButtonStyle backgroundColors(MultiColor? backgroundColors) =>
this(backgroundColors: backgroundColors);
@override
FileSelectionButtonStyle borderColors(MultiColor? borderColors) =>
this(borderColors: borderColors);
@override
FileSelectionButtonStyle stroke(double? stroke) => this(stroke: stroke);
@override
FileSelectionButtonStyle shadow(BoxShadow? shadow) => this(shadow: shadow);
@override
/// This function **does support** nullification of nullable fields. All `null` values passed to `non-nullable` fields will be ignored. You can also use `FileSelectionButtonStyle(...).copyWith.fieldName(...)` to override fields one at a time with nullification support.
///
/// Usage
/// ```dart
/// FileSelectionButtonStyle(...).copyWith(id: 12, name: "My name")
/// ````
FileSelectionButtonStyle call({
Object? title = const $CopyWithPlaceholder(),
Object? subTitle = const $CopyWithPlaceholder(),
Object? radius = const $CopyWithPlaceholder(),
Object? padding = const $CopyWithPlaceholder(),
Object? foregroundColors = const $CopyWithPlaceholder(),
Object? backgroundColors = const $CopyWithPlaceholder(),
Object? borderColors = const $CopyWithPlaceholder(),
Object? stroke = const $CopyWithPlaceholder(),
Object? shadow = const $CopyWithPlaceholder(),
}) {
return FileSelectionButtonStyle(
title: title == const $CopyWithPlaceholder()
? _value.title
// ignore: cast_nullable_to_non_nullable
: title as TextStyle?,
subTitle: subTitle == const $CopyWithPlaceholder()
? _value.subTitle
// ignore: cast_nullable_to_non_nullable
: subTitle as TextStyle?,
radius: radius == const $CopyWithPlaceholder()
? _value.radius
// ignore: cast_nullable_to_non_nullable
: radius as BorderRadiusGeometry?,
padding: padding == const $CopyWithPlaceholder()
? _value.padding
// ignore: cast_nullable_to_non_nullable
: padding as EdgeInsetsGeometry?,
foregroundColors: foregroundColors == const $CopyWithPlaceholder()
? _value.foregroundColors
// ignore: cast_nullable_to_non_nullable
: foregroundColors as MultiColor?,
backgroundColors: backgroundColors == const $CopyWithPlaceholder()
? _value.backgroundColors
// ignore: cast_nullable_to_non_nullable
: backgroundColors as MultiColor?,
borderColors: borderColors == const $CopyWithPlaceholder()
? _value.borderColors
// ignore: cast_nullable_to_non_nullable
: borderColors as MultiColor?,
stroke: stroke == const $CopyWithPlaceholder()
? _value.stroke
// ignore: cast_nullable_to_non_nullable
: stroke as double?,
shadow: shadow == const $CopyWithPlaceholder()
? _value.shadow
// ignore: cast_nullable_to_non_nullable
: shadow as BoxShadow?,
);
}
}
extension $FileSelectionButtonStyleCopyWith on FileSelectionButtonStyle {
/// Returns a callable class that can be used as follows: `instanceOfFileSelectionButtonStyle.copyWith(...)` or like so:`instanceOfFileSelectionButtonStyle.copyWith.fieldName(...)`.
// ignore: library_private_types_in_public_api
_$FileSelectionButtonStyleCWProxy get copyWith =>
_$FileSelectionButtonStyleCWProxyImpl(this);
}

View File

@ -24,7 +24,7 @@ part 'flat_button_component.g.dart';
abstract class FlatButtonComponent extends ButtonComponent abstract class FlatButtonComponent extends ButtonComponent
with CopyWithMixin<$FlatButtonComponentCWProxy> { with CopyWithMixin<$FlatButtonComponentCWProxy> {
const FlatButtonComponent({ const FlatButtonComponent({
this.mainAxisSize = MainAxisSize.min, this.mainAxisSize,
this.prefix, this.prefix,
this.suffix, this.suffix,
this.label, this.label,

View File

@ -11,11 +11,11 @@ abstract class $FlatButtonComponentCWProxy {
FlatButtonComponent prefix(Widget? prefix); FlatButtonComponent prefix(Widget? prefix);
FlatButtonComponent suffix(Widget? suffix); FlatButtonComponent suffix(Widget? suffix);
FlatButtonComponent label(TextWrapper? label); FlatButtonComponent label(TextWrapper? label);
FlatButtonComponent disabledStyle(ButtonStyle? disabledStyle); FlatButtonComponent disabledStyle(ButtonStyle<dynamic>? disabledStyle);
FlatButtonComponent normalStyle(ButtonStyle? normalStyle); FlatButtonComponent normalStyle(ButtonStyle<dynamic>? normalStyle);
FlatButtonComponent hoveredStyle(ButtonStyle? hoveredStyle); FlatButtonComponent hoveredStyle(ButtonStyle<dynamic>? hoveredStyle);
FlatButtonComponent focusedStyle(ButtonStyle? focusedStyle); FlatButtonComponent focusedStyle(ButtonStyle<dynamic>? focusedStyle);
FlatButtonComponent tappedStyle(ButtonStyle? tappedStyle); FlatButtonComponent tappedStyle(ButtonStyle<dynamic>? tappedStyle);
FlatButtonComponent onPressed(void Function(ControlState)? onPressed); FlatButtonComponent onPressed(void Function(ControlState)? onPressed);
FlatButtonComponent key(Key? key); FlatButtonComponent key(Key? key);
FlatButtonComponent call({ FlatButtonComponent call({
@ -23,11 +23,11 @@ abstract class $FlatButtonComponentCWProxy {
Widget? prefix, Widget? prefix,
Widget? suffix, Widget? suffix,
TextWrapper? label, TextWrapper? label,
ButtonStyle? disabledStyle, ButtonStyle<dynamic>? disabledStyle,
ButtonStyle? normalStyle, ButtonStyle<dynamic>? normalStyle,
ButtonStyle? hoveredStyle, ButtonStyle<dynamic>? hoveredStyle,
ButtonStyle? focusedStyle, ButtonStyle<dynamic>? focusedStyle,
ButtonStyle? tappedStyle, ButtonStyle<dynamic>? tappedStyle,
void Function(ControlState)? onPressed, void Function(ControlState)? onPressed,
Key? key, Key? key,
}); });

View File

@ -14,42 +14,76 @@
// You should have received a copy of the GNU General Public License // You should have received a copy of the GNU General Public License
// along with this program. If not, see <https://www.gnu.org/licenses/>. // along with this program. If not, see <https://www.gnu.org/licenses/>.
import 'dart:ui';
import 'package:copy_with_extension/copy_with_extension.dart';
import 'package:flutter/widgets.dart'; import 'package:flutter/widgets.dart';
import 'package:wyatt_ui_components/src/core/extensions/build_context_extensions.dart';
import 'package:wyatt_ui_components/src/core/utils/multi_color.dart'; import 'package:wyatt_ui_components/src/core/utils/multi_color.dart';
import 'package:wyatt_ui_components/src/domain/entities/buttons/button_style.dart'; import 'package:wyatt_ui_components/src/domain/entities/buttons/button_style.dart';
class FlatButtonStyle extends ButtonStyle { part 'flat_button_style.g.dart';
@CopyWith()
class FlatButtonStyle extends ButtonStyle<FlatButtonStyle> {
const FlatButtonStyle({ const FlatButtonStyle({
super.radius = 15, this.label,
super.padding = 10, super.radius,
super.padding,
super.foregroundColors, super.foregroundColors,
super.backgroundColors, super.backgroundColors,
super.borderColors, super.borderColors,
super.stroke = 2, super.stroke,
super.shadow = const BoxShadow( super.shadow,
blurRadius: 30,
offset: Offset(0, 5),
color: Color.fromRGBO(0, 0, 0, 0.05),
),
}); });
@override /// Used in negociation to build a style from Flutter default values.
FlatButtonStyle copyWith({ factory FlatButtonStyle.fromFlutter(BuildContext context) => FlatButtonStyle(
double? radius, label: context.textTheme.labelLarge,
double? padding, radius: (context.buttonTheme.shape is RoundedRectangleBorder)
MultiColor? foregroundColors, ? (context.buttonTheme.shape as RoundedRectangleBorder).borderRadius
MultiColor? backgroundColors, : null,
MultiColor? borderColors, padding: context.buttonTheme.padding,
double? stroke, foregroundColors: MultiColor.single(context.colorScheme.onPrimary),
BoxShadow? shadow, backgroundColors: MultiColor.single(context.colorScheme.primary),
}) =>
FlatButtonStyle(
radius: radius ?? this.radius,
padding: padding ?? this.padding,
foregroundColors: foregroundColors ?? this.foregroundColors,
backgroundColors: backgroundColors ?? this.backgroundColors,
borderColors: borderColors ?? this.borderColors,
stroke: stroke ?? this.stroke,
shadow: shadow ?? this.shadow,
); );
/// Used for interpolation.
static FlatButtonStyle? lerp(
FlatButtonStyle? a,
FlatButtonStyle? b,
double t,
) {
if (a == null || b == null) {
return null;
}
// b.copyWith to return b attributes even if they are not lerped
return b.copyWith(
label: TextStyle.lerp(a.label, b.label, t),
foregroundColors: MultiColor.lerp(
a.foregroundColors,
b.foregroundColors,
t,
),
backgroundColors: MultiColor.lerp(
a.backgroundColors,
b.backgroundColors,
t,
),
borderColors: MultiColor.lerp(
a.borderColors,
b.borderColors,
t,
),
radius: BorderRadiusGeometry.lerp(a.radius, b.radius, t),
padding: EdgeInsetsGeometry.lerp(a.padding, b.padding, t),
stroke: lerpDouble(a.stroke, b.stroke, t),
shadow: BoxShadow.lerp(a.shadow, b.shadow, t),
);
}
/// Label text style
///
/// Default to `TextTheme.labelLarge`
final TextStyle? label;
} }

View File

@ -0,0 +1,137 @@
// GENERATED CODE - DO NOT MODIFY BY HAND
part of 'flat_button_style.dart';
// **************************************************************************
// CopyWithGenerator
// **************************************************************************
abstract class _$FlatButtonStyleCWProxy {
FlatButtonStyle label(TextStyle? label);
FlatButtonStyle radius(BorderRadiusGeometry? radius);
FlatButtonStyle padding(EdgeInsetsGeometry? padding);
FlatButtonStyle foregroundColors(MultiColor? foregroundColors);
FlatButtonStyle backgroundColors(MultiColor? backgroundColors);
FlatButtonStyle borderColors(MultiColor? borderColors);
FlatButtonStyle stroke(double? stroke);
FlatButtonStyle shadow(BoxShadow? shadow);
/// This function **does support** nullification of nullable fields. All `null` values passed to `non-nullable` fields will be ignored. You can also use `FlatButtonStyle(...).copyWith.fieldName(...)` to override fields one at a time with nullification support.
///
/// Usage
/// ```dart
/// FlatButtonStyle(...).copyWith(id: 12, name: "My name")
/// ````
FlatButtonStyle call({
TextStyle? label,
BorderRadiusGeometry? radius,
EdgeInsetsGeometry? padding,
MultiColor? foregroundColors,
MultiColor? backgroundColors,
MultiColor? borderColors,
double? stroke,
BoxShadow? shadow,
});
}
/// Proxy class for `copyWith` functionality. This is a callable class and can be used as follows: `instanceOfFlatButtonStyle.copyWith(...)`. Additionally contains functions for specific fields e.g. `instanceOfFlatButtonStyle.copyWith.fieldName(...)`
class _$FlatButtonStyleCWProxyImpl implements _$FlatButtonStyleCWProxy {
const _$FlatButtonStyleCWProxyImpl(this._value);
final FlatButtonStyle _value;
@override
FlatButtonStyle label(TextStyle? label) => this(label: label);
@override
FlatButtonStyle radius(BorderRadiusGeometry? radius) => this(radius: radius);
@override
FlatButtonStyle padding(EdgeInsetsGeometry? padding) =>
this(padding: padding);
@override
FlatButtonStyle foregroundColors(MultiColor? foregroundColors) =>
this(foregroundColors: foregroundColors);
@override
FlatButtonStyle backgroundColors(MultiColor? backgroundColors) =>
this(backgroundColors: backgroundColors);
@override
FlatButtonStyle borderColors(MultiColor? borderColors) =>
this(borderColors: borderColors);
@override
FlatButtonStyle stroke(double? stroke) => this(stroke: stroke);
@override
FlatButtonStyle shadow(BoxShadow? shadow) => this(shadow: shadow);
@override
/// This function **does support** nullification of nullable fields. All `null` values passed to `non-nullable` fields will be ignored. You can also use `FlatButtonStyle(...).copyWith.fieldName(...)` to override fields one at a time with nullification support.
///
/// Usage
/// ```dart
/// FlatButtonStyle(...).copyWith(id: 12, name: "My name")
/// ````
FlatButtonStyle call({
Object? label = const $CopyWithPlaceholder(),
Object? radius = const $CopyWithPlaceholder(),
Object? padding = const $CopyWithPlaceholder(),
Object? foregroundColors = const $CopyWithPlaceholder(),
Object? backgroundColors = const $CopyWithPlaceholder(),
Object? borderColors = const $CopyWithPlaceholder(),
Object? stroke = const $CopyWithPlaceholder(),
Object? shadow = const $CopyWithPlaceholder(),
}) {
return FlatButtonStyle(
label: label == const $CopyWithPlaceholder()
? _value.label
// ignore: cast_nullable_to_non_nullable
: label as TextStyle?,
radius: radius == const $CopyWithPlaceholder()
? _value.radius
// ignore: cast_nullable_to_non_nullable
: radius as BorderRadiusGeometry?,
padding: padding == const $CopyWithPlaceholder()
? _value.padding
// ignore: cast_nullable_to_non_nullable
: padding as EdgeInsetsGeometry?,
foregroundColors: foregroundColors == const $CopyWithPlaceholder()
? _value.foregroundColors
// ignore: cast_nullable_to_non_nullable
: foregroundColors as MultiColor?,
backgroundColors: backgroundColors == const $CopyWithPlaceholder()
? _value.backgroundColors
// ignore: cast_nullable_to_non_nullable
: backgroundColors as MultiColor?,
borderColors: borderColors == const $CopyWithPlaceholder()
? _value.borderColors
// ignore: cast_nullable_to_non_nullable
: borderColors as MultiColor?,
stroke: stroke == const $CopyWithPlaceholder()
? _value.stroke
// ignore: cast_nullable_to_non_nullable
: stroke as double?,
shadow: shadow == const $CopyWithPlaceholder()
? _value.shadow
// ignore: cast_nullable_to_non_nullable
: shadow as BoxShadow?,
);
}
}
extension $FlatButtonStyleCopyWith on FlatButtonStyle {
/// Returns a callable class that can be used as follows: `instanceOfFlatButtonStyle.copyWith(...)` or like so:`instanceOfFlatButtonStyle.copyWith.fieldName(...)`.
// ignore: library_private_types_in_public_api
_$FlatButtonStyleCWProxy get copyWith => _$FlatButtonStyleCWProxyImpl(this);
}

View File

@ -16,7 +16,6 @@
import 'package:flutter/widgets.dart'; import 'package:flutter/widgets.dart';
import 'package:wyatt_component_copy_with_extension/component_copy_with_extension.dart'; import 'package:wyatt_component_copy_with_extension/component_copy_with_extension.dart';
import 'package:wyatt_ui_components/src/domain/entities/buttons/simple_icon_button_style.dart';
import 'package:wyatt_ui_components/wyatt_wyatt_ui_components.dart'; import 'package:wyatt_ui_components/wyatt_wyatt_ui_components.dart';
part 'simple_icon_button_component.g.dart'; part 'simple_icon_button_component.g.dart';
@ -31,7 +30,6 @@ abstract class SimpleIconButtonComponent extends ButtonComponent
super.hoveredStyle, super.hoveredStyle,
super.focusedStyle, super.focusedStyle,
super.tappedStyle, super.tappedStyle,
super.selectedStyle,
super.onPressed, super.onPressed,
super.key, super.key,
}); });
@ -51,8 +49,5 @@ abstract class SimpleIconButtonComponent extends ButtonComponent
@override @override
SimpleIconButtonStyle? get tappedStyle; SimpleIconButtonStyle? get tappedStyle;
@override
SimpleIconButtonStyle? get selectedStyle;
final Icon? icon; final Icon? icon;
} }

View File

@ -8,22 +8,20 @@ part of 'simple_icon_button_component.dart';
abstract class $SimpleIconButtonComponentCWProxy { abstract class $SimpleIconButtonComponentCWProxy {
SimpleIconButtonComponent icon(Icon? icon); SimpleIconButtonComponent icon(Icon? icon);
SimpleIconButtonComponent disabledStyle(ButtonStyle? disabledStyle); SimpleIconButtonComponent disabledStyle(ButtonStyle<dynamic>? disabledStyle);
SimpleIconButtonComponent normalStyle(ButtonStyle? normalStyle); SimpleIconButtonComponent normalStyle(ButtonStyle<dynamic>? normalStyle);
SimpleIconButtonComponent hoveredStyle(ButtonStyle? hoveredStyle); SimpleIconButtonComponent hoveredStyle(ButtonStyle<dynamic>? hoveredStyle);
SimpleIconButtonComponent focusedStyle(ButtonStyle? focusedStyle); SimpleIconButtonComponent focusedStyle(ButtonStyle<dynamic>? focusedStyle);
SimpleIconButtonComponent tappedStyle(ButtonStyle? tappedStyle); SimpleIconButtonComponent tappedStyle(ButtonStyle<dynamic>? tappedStyle);
SimpleIconButtonComponent selectedStyle(ButtonStyle? selectedStyle);
SimpleIconButtonComponent onPressed(void Function(ControlState)? onPressed); SimpleIconButtonComponent onPressed(void Function(ControlState)? onPressed);
SimpleIconButtonComponent key(Key? key); SimpleIconButtonComponent key(Key? key);
SimpleIconButtonComponent call({ SimpleIconButtonComponent call({
Icon? icon, Icon? icon,
ButtonStyle? disabledStyle, ButtonStyle<dynamic>? disabledStyle,
ButtonStyle? normalStyle, ButtonStyle<dynamic>? normalStyle,
ButtonStyle? hoveredStyle, ButtonStyle<dynamic>? hoveredStyle,
ButtonStyle? focusedStyle, ButtonStyle<dynamic>? focusedStyle,
ButtonStyle? tappedStyle, ButtonStyle<dynamic>? tappedStyle,
ButtonStyle? selectedStyle,
void Function(ControlState)? onPressed, void Function(ControlState)? onPressed,
Key? key, Key? key,
}); });

View File

@ -14,47 +14,77 @@
// You should have received a copy of the GNU General Public License // You should have received a copy of the GNU General Public License
// along with this program. If not, see <https://www.gnu.org/licenses/>. // along with this program. If not, see <https://www.gnu.org/licenses/>.
import 'dart:ui';
import 'package:copy_with_extension/copy_with_extension.dart';
import 'package:flutter/widgets.dart'; import 'package:flutter/widgets.dart';
import 'package:wyatt_ui_components/src/core/extensions/build_context_extensions.dart';
import 'package:wyatt_ui_components/src/core/utils/multi_color.dart'; import 'package:wyatt_ui_components/src/core/utils/multi_color.dart';
import 'package:wyatt_ui_components/src/domain/entities/buttons/button_style.dart'; import 'package:wyatt_ui_components/src/domain/entities/buttons/button_style.dart';
class SimpleIconButtonStyle extends ButtonStyle { part 'simple_icon_button_style.g.dart';
@CopyWith()
class SimpleIconButtonStyle extends ButtonStyle<SimpleIconButtonStyle> {
const SimpleIconButtonStyle({ const SimpleIconButtonStyle({
this.dimension = 30, this.dimension,
super.radius = 5, super.radius,
super.padding = 5, super.padding,
super.foregroundColors, super.foregroundColors,
super.backgroundColors, super.backgroundColors,
super.borderColors, super.borderColors,
super.stroke = 2, super.stroke,
super.shadow = const BoxShadow( super.shadow,
blurRadius: 30,
offset: Offset(0, 5),
color: Color.fromRGBO(0, 0, 0, 0.05),
),
}); });
final double? dimension; /// Used in negociation to build a style from Flutter default values.
factory SimpleIconButtonStyle.fromFlutter(BuildContext context) =>
@override
SimpleIconButtonStyle copyWith({
double? dimension,
double? radius,
double? padding,
MultiColor? foregroundColors,
MultiColor? backgroundColors,
MultiColor? borderColors,
double? stroke,
BoxShadow? shadow,
}) =>
SimpleIconButtonStyle( SimpleIconButtonStyle(
dimension: dimension ?? this.dimension, dimension: context.buttonTheme.height,
radius: radius ?? radius, radius: (context.buttonTheme.shape is RoundedRectangleBorder)
padding: padding ?? padding, ? (context.buttonTheme.shape as RoundedRectangleBorder).borderRadius
foregroundColors: foregroundColors ?? foregroundColors, : null,
backgroundColors: backgroundColors ?? backgroundColors, padding: context.buttonTheme.padding,
borderColors: borderColors ?? borderColors, foregroundColors: MultiColor.single(context.colorScheme.onPrimary),
stroke: stroke ?? stroke, backgroundColors: MultiColor.single(context.colorScheme.primary),
shadow: shadow ?? shadow,
); );
/// Used for interpolation.
static SimpleIconButtonStyle? lerp(
SimpleIconButtonStyle? a,
SimpleIconButtonStyle? b,
double t,
) {
if (a == null || b == null) {
return null;
}
// b.copyWith to return b attributes even if they are not lerped
return b.copyWith(
dimension: lerpDouble(a.dimension, b.dimension, t),
foregroundColors: MultiColor.lerp(
a.foregroundColors,
b.foregroundColors,
t,
),
backgroundColors: MultiColor.lerp(
a.backgroundColors,
b.backgroundColors,
t,
),
borderColors: MultiColor.lerp(
a.borderColors,
b.borderColors,
t,
),
radius: BorderRadiusGeometry.lerp(a.radius, b.radius, t),
padding: EdgeInsetsGeometry.lerp(a.padding, b.padding, t),
stroke: lerpDouble(a.stroke, b.stroke, t),
shadow: BoxShadow.lerp(a.shadow, b.shadow, t),
);
}
/// Dimension of this button (as a square)
///
/// Default to `context.buttonTheme.height`
final double? dimension;
} }

View File

@ -0,0 +1,141 @@
// GENERATED CODE - DO NOT MODIFY BY HAND
part of 'simple_icon_button_style.dart';
// **************************************************************************
// CopyWithGenerator
// **************************************************************************
abstract class _$SimpleIconButtonStyleCWProxy {
SimpleIconButtonStyle dimension(double? dimension);
SimpleIconButtonStyle radius(BorderRadiusGeometry? radius);
SimpleIconButtonStyle padding(EdgeInsetsGeometry? padding);
SimpleIconButtonStyle foregroundColors(MultiColor? foregroundColors);
SimpleIconButtonStyle backgroundColors(MultiColor? backgroundColors);
SimpleIconButtonStyle borderColors(MultiColor? borderColors);
SimpleIconButtonStyle stroke(double? stroke);
SimpleIconButtonStyle shadow(BoxShadow? shadow);
/// This function **does support** nullification of nullable fields. All `null` values passed to `non-nullable` fields will be ignored. You can also use `SimpleIconButtonStyle(...).copyWith.fieldName(...)` to override fields one at a time with nullification support.
///
/// Usage
/// ```dart
/// SimpleIconButtonStyle(...).copyWith(id: 12, name: "My name")
/// ````
SimpleIconButtonStyle call({
double? dimension,
BorderRadiusGeometry? radius,
EdgeInsetsGeometry? padding,
MultiColor? foregroundColors,
MultiColor? backgroundColors,
MultiColor? borderColors,
double? stroke,
BoxShadow? shadow,
});
}
/// Proxy class for `copyWith` functionality. This is a callable class and can be used as follows: `instanceOfSimpleIconButtonStyle.copyWith(...)`. Additionally contains functions for specific fields e.g. `instanceOfSimpleIconButtonStyle.copyWith.fieldName(...)`
class _$SimpleIconButtonStyleCWProxyImpl
implements _$SimpleIconButtonStyleCWProxy {
const _$SimpleIconButtonStyleCWProxyImpl(this._value);
final SimpleIconButtonStyle _value;
@override
SimpleIconButtonStyle dimension(double? dimension) =>
this(dimension: dimension);
@override
SimpleIconButtonStyle radius(BorderRadiusGeometry? radius) =>
this(radius: radius);
@override
SimpleIconButtonStyle padding(EdgeInsetsGeometry? padding) =>
this(padding: padding);
@override
SimpleIconButtonStyle foregroundColors(MultiColor? foregroundColors) =>
this(foregroundColors: foregroundColors);
@override
SimpleIconButtonStyle backgroundColors(MultiColor? backgroundColors) =>
this(backgroundColors: backgroundColors);
@override
SimpleIconButtonStyle borderColors(MultiColor? borderColors) =>
this(borderColors: borderColors);
@override
SimpleIconButtonStyle stroke(double? stroke) => this(stroke: stroke);
@override
SimpleIconButtonStyle shadow(BoxShadow? shadow) => this(shadow: shadow);
@override
/// This function **does support** nullification of nullable fields. All `null` values passed to `non-nullable` fields will be ignored. You can also use `SimpleIconButtonStyle(...).copyWith.fieldName(...)` to override fields one at a time with nullification support.
///
/// Usage
/// ```dart
/// SimpleIconButtonStyle(...).copyWith(id: 12, name: "My name")
/// ````
SimpleIconButtonStyle call({
Object? dimension = const $CopyWithPlaceholder(),
Object? radius = const $CopyWithPlaceholder(),
Object? padding = const $CopyWithPlaceholder(),
Object? foregroundColors = const $CopyWithPlaceholder(),
Object? backgroundColors = const $CopyWithPlaceholder(),
Object? borderColors = const $CopyWithPlaceholder(),
Object? stroke = const $CopyWithPlaceholder(),
Object? shadow = const $CopyWithPlaceholder(),
}) {
return SimpleIconButtonStyle(
dimension: dimension == const $CopyWithPlaceholder()
? _value.dimension
// ignore: cast_nullable_to_non_nullable
: dimension as double?,
radius: radius == const $CopyWithPlaceholder()
? _value.radius
// ignore: cast_nullable_to_non_nullable
: radius as BorderRadiusGeometry?,
padding: padding == const $CopyWithPlaceholder()
? _value.padding
// ignore: cast_nullable_to_non_nullable
: padding as EdgeInsetsGeometry?,
foregroundColors: foregroundColors == const $CopyWithPlaceholder()
? _value.foregroundColors
// ignore: cast_nullable_to_non_nullable
: foregroundColors as MultiColor?,
backgroundColors: backgroundColors == const $CopyWithPlaceholder()
? _value.backgroundColors
// ignore: cast_nullable_to_non_nullable
: backgroundColors as MultiColor?,
borderColors: borderColors == const $CopyWithPlaceholder()
? _value.borderColors
// ignore: cast_nullable_to_non_nullable
: borderColors as MultiColor?,
stroke: stroke == const $CopyWithPlaceholder()
? _value.stroke
// ignore: cast_nullable_to_non_nullable
: stroke as double?,
shadow: shadow == const $CopyWithPlaceholder()
? _value.shadow
// ignore: cast_nullable_to_non_nullable
: shadow as BoxShadow?,
);
}
}
extension $SimpleIconButtonStyleCopyWith on SimpleIconButtonStyle {
/// Returns a callable class that can be used as follows: `instanceOfSimpleIconButtonStyle.copyWith(...)` or like so:`instanceOfSimpleIconButtonStyle.copyWith.fieldName(...)`.
// ignore: library_private_types_in_public_api
_$SimpleIconButtonStyleCWProxy get copyWith =>
_$SimpleIconButtonStyleCWProxyImpl(this);
}

View File

@ -10,24 +10,24 @@ abstract class $SymbolButtonComponentCWProxy {
SymbolButtonComponent mainAxisSize(MainAxisSize? mainAxisSize); SymbolButtonComponent mainAxisSize(MainAxisSize? mainAxisSize);
SymbolButtonComponent label(TextWrapper? label); SymbolButtonComponent label(TextWrapper? label);
SymbolButtonComponent icon(Widget? icon); SymbolButtonComponent icon(Widget? icon);
SymbolButtonComponent disabledStyle(ButtonStyle? disabledStyle); SymbolButtonComponent disabledStyle(ButtonStyle<dynamic>? disabledStyle);
SymbolButtonComponent normalStyle(ButtonStyle? normalStyle); SymbolButtonComponent normalStyle(ButtonStyle<dynamic>? normalStyle);
SymbolButtonComponent hoveredStyle(ButtonStyle? hoveredStyle); SymbolButtonComponent hoveredStyle(ButtonStyle<dynamic>? hoveredStyle);
SymbolButtonComponent focusedStyle(ButtonStyle? focusedStyle); SymbolButtonComponent focusedStyle(ButtonStyle<dynamic>? focusedStyle);
SymbolButtonComponent tappedStyle(ButtonStyle? tappedStyle); SymbolButtonComponent tappedStyle(ButtonStyle<dynamic>? tappedStyle);
SymbolButtonComponent selectedStyle(ButtonStyle? selectedStyle); SymbolButtonComponent selectedStyle(ButtonStyle<dynamic>? selectedStyle);
SymbolButtonComponent onPressed(void Function(ControlState)? onPressed); SymbolButtonComponent onPressed(void Function(ControlState)? onPressed);
SymbolButtonComponent key(Key? key); SymbolButtonComponent key(Key? key);
SymbolButtonComponent call({ SymbolButtonComponent call({
MainAxisSize? mainAxisSize, MainAxisSize? mainAxisSize,
TextWrapper? label, TextWrapper? label,
Widget? icon, Widget? icon,
ButtonStyle? disabledStyle, ButtonStyle<dynamic>? disabledStyle,
ButtonStyle? normalStyle, ButtonStyle<dynamic>? normalStyle,
ButtonStyle? hoveredStyle, ButtonStyle<dynamic>? hoveredStyle,
ButtonStyle? focusedStyle, ButtonStyle<dynamic>? focusedStyle,
ButtonStyle? tappedStyle, ButtonStyle<dynamic>? tappedStyle,
ButtonStyle? selectedStyle, ButtonStyle<dynamic>? selectedStyle,
void Function(ControlState)? onPressed, void Function(ControlState)? onPressed,
Key? key, Key? key,
}); });

View File

@ -14,47 +14,85 @@
// You should have received a copy of the GNU General Public License // You should have received a copy of the GNU General Public License
// along with this program. If not, see <https://www.gnu.org/licenses/>. // along with this program. If not, see <https://www.gnu.org/licenses/>.
import 'dart:ui';
import 'package:copy_with_extension/copy_with_extension.dart';
import 'package:flutter/widgets.dart'; import 'package:flutter/widgets.dart';
import 'package:wyatt_ui_components/src/core/extensions/build_context_extensions.dart';
import 'package:wyatt_ui_components/src/core/utils/multi_color.dart'; import 'package:wyatt_ui_components/src/core/utils/multi_color.dart';
import 'package:wyatt_ui_components/src/domain/entities/buttons/button_style.dart'; import 'package:wyatt_ui_components/src/domain/entities/buttons/button_style.dart';
class SymbolButtonStyle extends ButtonStyle { part 'symbol_button_style.g.dart';
@CopyWith()
class SymbolButtonStyle extends ButtonStyle<SymbolButtonStyle> {
const SymbolButtonStyle({ const SymbolButtonStyle({
this.dimension = 60, this.label,
super.radius = 15, this.dimension,
super.padding = 10, super.radius,
super.padding,
super.foregroundColors, super.foregroundColors,
super.backgroundColors, super.backgroundColors,
super.borderColors, super.borderColors,
super.stroke = 2, super.stroke,
super.shadow = const BoxShadow( super.shadow,
blurRadius: 30,
offset: Offset(0, 5),
color: Color.fromRGBO(0, 0, 0, 0.05),
),
}); });
final double? dimension; /// Used in negociation to build a style from Flutter default values.
factory SymbolButtonStyle.fromFlutter(BuildContext context) =>
@override
SymbolButtonStyle copyWith({
double? dimension,
double? radius,
double? padding,
MultiColor? foregroundColors,
MultiColor? backgroundColors,
MultiColor? borderColors,
double? stroke,
BoxShadow? shadow,
}) =>
SymbolButtonStyle( SymbolButtonStyle(
dimension: dimension ?? this.dimension, label: context.textTheme.labelLarge,
radius: radius ?? this.radius, dimension: context.buttonTheme.height,
padding: padding ?? this.padding, radius: (context.buttonTheme.shape is RoundedRectangleBorder)
foregroundColors: foregroundColors ?? this.foregroundColors, ? (context.buttonTheme.shape as RoundedRectangleBorder).borderRadius
backgroundColors: backgroundColors ?? this.backgroundColors, : null,
borderColors: borderColors ?? this.borderColors, padding: context.buttonTheme.padding,
stroke: stroke ?? this.stroke, foregroundColors: MultiColor.single(context.colorScheme.onPrimary),
shadow: shadow ?? this.shadow, backgroundColors: MultiColor.single(context.colorScheme.primary),
); );
/// Used for interpolation.
static SymbolButtonStyle? lerp(
SymbolButtonStyle? a,
SymbolButtonStyle? b,
double t,
) {
if (a == null || b == null) {
return null;
}
// b.copyWith to return b attributes even if they are not lerped
return b.copyWith(
label: TextStyle.lerp(a.label, b.label, t),
dimension: lerpDouble(a.dimension, b.dimension, t),
foregroundColors: MultiColor.lerp(
a.foregroundColors,
b.foregroundColors,
t,
),
backgroundColors: MultiColor.lerp(
a.backgroundColors,
b.backgroundColors,
t,
),
borderColors: MultiColor.lerp(
a.borderColors,
b.borderColors,
t,
),
radius: BorderRadiusGeometry.lerp(a.radius, b.radius, t),
padding: EdgeInsetsGeometry.lerp(a.padding, b.padding, t),
stroke: lerpDouble(a.stroke, b.stroke, t),
shadow: BoxShadow.lerp(a.shadow, b.shadow, t),
);
}
/// Label text style
///
/// Default to `TextTheme.labelLarge`
final TextStyle? label;
/// Dimension of this button (as a square)
///
/// Default to `context.buttonTheme.height`
final double? dimension;
} }

View File

@ -0,0 +1,150 @@
// GENERATED CODE - DO NOT MODIFY BY HAND
part of 'symbol_button_style.dart';
// **************************************************************************
// CopyWithGenerator
// **************************************************************************
abstract class _$SymbolButtonStyleCWProxy {
SymbolButtonStyle label(TextStyle? label);
SymbolButtonStyle dimension(double? dimension);
SymbolButtonStyle radius(BorderRadiusGeometry? radius);
SymbolButtonStyle padding(EdgeInsetsGeometry? padding);
SymbolButtonStyle foregroundColors(MultiColor? foregroundColors);
SymbolButtonStyle backgroundColors(MultiColor? backgroundColors);
SymbolButtonStyle borderColors(MultiColor? borderColors);
SymbolButtonStyle stroke(double? stroke);
SymbolButtonStyle shadow(BoxShadow? shadow);
/// This function **does support** nullification of nullable fields. All `null` values passed to `non-nullable` fields will be ignored. You can also use `SymbolButtonStyle(...).copyWith.fieldName(...)` to override fields one at a time with nullification support.
///
/// Usage
/// ```dart
/// SymbolButtonStyle(...).copyWith(id: 12, name: "My name")
/// ````
SymbolButtonStyle call({
TextStyle? label,
double? dimension,
BorderRadiusGeometry? radius,
EdgeInsetsGeometry? padding,
MultiColor? foregroundColors,
MultiColor? backgroundColors,
MultiColor? borderColors,
double? stroke,
BoxShadow? shadow,
});
}
/// Proxy class for `copyWith` functionality. This is a callable class and can be used as follows: `instanceOfSymbolButtonStyle.copyWith(...)`. Additionally contains functions for specific fields e.g. `instanceOfSymbolButtonStyle.copyWith.fieldName(...)`
class _$SymbolButtonStyleCWProxyImpl implements _$SymbolButtonStyleCWProxy {
const _$SymbolButtonStyleCWProxyImpl(this._value);
final SymbolButtonStyle _value;
@override
SymbolButtonStyle label(TextStyle? label) => this(label: label);
@override
SymbolButtonStyle dimension(double? dimension) => this(dimension: dimension);
@override
SymbolButtonStyle radius(BorderRadiusGeometry? radius) =>
this(radius: radius);
@override
SymbolButtonStyle padding(EdgeInsetsGeometry? padding) =>
this(padding: padding);
@override
SymbolButtonStyle foregroundColors(MultiColor? foregroundColors) =>
this(foregroundColors: foregroundColors);
@override
SymbolButtonStyle backgroundColors(MultiColor? backgroundColors) =>
this(backgroundColors: backgroundColors);
@override
SymbolButtonStyle borderColors(MultiColor? borderColors) =>
this(borderColors: borderColors);
@override
SymbolButtonStyle stroke(double? stroke) => this(stroke: stroke);
@override
SymbolButtonStyle shadow(BoxShadow? shadow) => this(shadow: shadow);
@override
/// This function **does support** nullification of nullable fields. All `null` values passed to `non-nullable` fields will be ignored. You can also use `SymbolButtonStyle(...).copyWith.fieldName(...)` to override fields one at a time with nullification support.
///
/// Usage
/// ```dart
/// SymbolButtonStyle(...).copyWith(id: 12, name: "My name")
/// ````
SymbolButtonStyle call({
Object? label = const $CopyWithPlaceholder(),
Object? dimension = const $CopyWithPlaceholder(),
Object? radius = const $CopyWithPlaceholder(),
Object? padding = const $CopyWithPlaceholder(),
Object? foregroundColors = const $CopyWithPlaceholder(),
Object? backgroundColors = const $CopyWithPlaceholder(),
Object? borderColors = const $CopyWithPlaceholder(),
Object? stroke = const $CopyWithPlaceholder(),
Object? shadow = const $CopyWithPlaceholder(),
}) {
return SymbolButtonStyle(
label: label == const $CopyWithPlaceholder()
? _value.label
// ignore: cast_nullable_to_non_nullable
: label as TextStyle?,
dimension: dimension == const $CopyWithPlaceholder()
? _value.dimension
// ignore: cast_nullable_to_non_nullable
: dimension as double?,
radius: radius == const $CopyWithPlaceholder()
? _value.radius
// ignore: cast_nullable_to_non_nullable
: radius as BorderRadiusGeometry?,
padding: padding == const $CopyWithPlaceholder()
? _value.padding
// ignore: cast_nullable_to_non_nullable
: padding as EdgeInsetsGeometry?,
foregroundColors: foregroundColors == const $CopyWithPlaceholder()
? _value.foregroundColors
// ignore: cast_nullable_to_non_nullable
: foregroundColors as MultiColor?,
backgroundColors: backgroundColors == const $CopyWithPlaceholder()
? _value.backgroundColors
// ignore: cast_nullable_to_non_nullable
: backgroundColors as MultiColor?,
borderColors: borderColors == const $CopyWithPlaceholder()
? _value.borderColors
// ignore: cast_nullable_to_non_nullable
: borderColors as MultiColor?,
stroke: stroke == const $CopyWithPlaceholder()
? _value.stroke
// ignore: cast_nullable_to_non_nullable
: stroke as double?,
shadow: shadow == const $CopyWithPlaceholder()
? _value.shadow
// ignore: cast_nullable_to_non_nullable
: shadow as BoxShadow?,
);
}
}
extension $SymbolButtonStyleCopyWith on SymbolButtonStyle {
/// Returns a callable class that can be used as follows: `instanceOfSymbolButtonStyle.copyWith(...)` or like so:`instanceOfSymbolButtonStyle.copyWith.fieldName(...)`.
// ignore: library_private_types_in_public_api
_$SymbolButtonStyleCWProxy get copyWith =>
_$SymbolButtonStyleCWProxyImpl(this);
}

View File

@ -13,6 +13,8 @@ abstract class $SkillCardComponentCWProxy {
SkillCardComponent description(TextWrapper? description); SkillCardComponent description(TextWrapper? description);
SkillCardComponent skills(List<TextWrapper>? skills); SkillCardComponent skills(List<TextWrapper>? skills);
SkillCardComponent leadingIcon(IconData? leadingIcon); SkillCardComponent leadingIcon(IconData? leadingIcon);
SkillCardComponent secondaryBackgroundColors(
Color? secondaryBackgroundColors);
SkillCardComponent radius(double? radius); SkillCardComponent radius(double? radius);
SkillCardComponent padding(double? padding); SkillCardComponent padding(double? padding);
SkillCardComponent borderColors(List<Color>? borderColors); SkillCardComponent borderColors(List<Color>? borderColors);
@ -21,8 +23,6 @@ abstract class $SkillCardComponentCWProxy {
SkillCardComponent maxSize(Size? maxSize); SkillCardComponent maxSize(Size? maxSize);
SkillCardComponent shadow(BoxShadow? shadow); SkillCardComponent shadow(BoxShadow? shadow);
SkillCardComponent background(Widget? background); SkillCardComponent background(Widget? background);
SkillCardComponent secondaryBackgroundColors(
Color? secondaryBackgroundColors);
SkillCardComponent key(Key? key); SkillCardComponent key(Key? key);
SkillCardComponent call({ SkillCardComponent call({
IconData? icon, IconData? icon,
@ -31,6 +31,7 @@ abstract class $SkillCardComponentCWProxy {
TextWrapper? description, TextWrapper? description,
List<TextWrapper>? skills, List<TextWrapper>? skills,
IconData? leadingIcon, IconData? leadingIcon,
Color? secondaryBackgroundColors,
double? radius, double? radius,
double? padding, double? padding,
List<Color>? borderColors, List<Color>? borderColors,
@ -39,7 +40,6 @@ abstract class $SkillCardComponentCWProxy {
Size? maxSize, Size? maxSize,
BoxShadow? shadow, BoxShadow? shadow,
Widget? background, Widget? background,
Color? secondaryBackgroundColors,
Key? key, Key? key,
}); });
} }

View File

@ -9,25 +9,21 @@ environment:
sdk: ">=2.17.0 <3.0.0" sdk: ">=2.17.0 <3.0.0"
dependencies: dependencies:
flutter: copy_with_extension: ^5.0.0
sdk: flutter flutter: { sdk: flutter }
wyatt_component_copy_with_extension: wyatt_component_copy_with_extension:
git: git:
url: ssh://git@git.wyatt-studio.fr:993/Wyatt-FOSS/wyatt-packages.git url: ssh://git@git.wyatt-studio.fr:993/Wyatt-FOSS/wyatt-packages.git
path: packages/wyatt_component_copy_with_extension path: packages/wyatt_component_copy_with_extension
dev_dependencies: dev_dependencies:
flutter_test:
sdk: flutter
build_runner: ^2.3.3 build_runner: ^2.3.3
copy_with_extension_gen: ^5.0.0
flutter_test: { sdk: flutter }
wyatt_analysis:
hosted: https://git.wyatt-studio.fr/api/packages/Wyatt-FOSS/pub
version: ^2.4.0
wyatt_component_copy_with_gen: wyatt_component_copy_with_gen:
git: git:
url: ssh://git@git.wyatt-studio.fr:993/Wyatt-FOSS/wyatt-packages.git url: ssh://git@git.wyatt-studio.fr:993/Wyatt-FOSS/wyatt-packages.git
path: packages/wyatt_component_copy_with_gen path: packages/wyatt_component_copy_with_gen
wyatt_analysis:
git:
url: https://git.wyatt-studio.fr/Wyatt-FOSS/wyatt-packages
ref: wyatt_analysis-v2.4.0
path: packages/wyatt_analysis

View File

@ -21,25 +21,25 @@ class $FileSelectionButtonCWProxyImpl
FileSelectionButton subTitle(TextWrapper? subTitle) => FileSelectionButton subTitle(TextWrapper? subTitle) =>
this(subTitle: subTitle); this(subTitle: subTitle);
@override @override
FileSelectionButton disabledStyle(ButtonStyle? disabledStyle) => FileSelectionButton disabledStyle(ButtonStyle<dynamic>? disabledStyle) =>
this(disabledStyle: disabledStyle); this(disabledStyle: disabledStyle);
@override @override
FileSelectionButton normalStyle(ButtonStyle? normalStyle) => FileSelectionButton normalStyle(ButtonStyle<dynamic>? normalStyle) =>
this(normalStyle: normalStyle); this(normalStyle: normalStyle);
@override @override
FileSelectionButton hoveredStyle(ButtonStyle? hoveredStyle) => FileSelectionButton hoveredStyle(ButtonStyle<dynamic>? hoveredStyle) =>
this(hoveredStyle: hoveredStyle); this(hoveredStyle: hoveredStyle);
@override @override
FileSelectionButton focusedStyle(ButtonStyle? focusedStyle) => FileSelectionButton focusedStyle(ButtonStyle<dynamic>? focusedStyle) =>
this(focusedStyle: focusedStyle); this(focusedStyle: focusedStyle);
@override @override
FileSelectionButton tappedStyle(ButtonStyle? tappedStyle) => FileSelectionButton tappedStyle(ButtonStyle<dynamic>? tappedStyle) =>
this(tappedStyle: tappedStyle); this(tappedStyle: tappedStyle);
@override @override
FileSelectionButton selectedStyle(ButtonStyle? selectedStyle) => FileSelectionButton selectedStyle(ButtonStyle<dynamic>? selectedStyle) =>
this(selectedStyle: selectedStyle); this(selectedStyle: selectedStyle);
@override @override
FileSelectionButton invalidStyle(ButtonStyle? invalidStyle) => FileSelectionButton invalidStyle(ButtonStyle<dynamic>? invalidStyle) =>
this(invalidStyle: invalidStyle); this(invalidStyle: invalidStyle);
@override @override
FileSelectionButton onPressed(void Function(ControlState)? onPressed) => FileSelectionButton onPressed(void Function(ControlState)? onPressed) =>
@ -52,13 +52,13 @@ class $FileSelectionButtonCWProxyImpl
Widget? leading, Widget? leading,
TextWrapper? title, TextWrapper? title,
TextWrapper? subTitle, TextWrapper? subTitle,
ButtonStyle? disabledStyle, ButtonStyle<dynamic>? disabledStyle,
ButtonStyle? normalStyle, ButtonStyle<dynamic>? normalStyle,
ButtonStyle? hoveredStyle, ButtonStyle<dynamic>? hoveredStyle,
ButtonStyle? focusedStyle, ButtonStyle<dynamic>? focusedStyle,
ButtonStyle? tappedStyle, ButtonStyle<dynamic>? tappedStyle,
ButtonStyle? selectedStyle, ButtonStyle<dynamic>? selectedStyle,
ButtonStyle? invalidStyle, ButtonStyle<dynamic>? invalidStyle,
void Function(ControlState)? onPressed, void Function(ControlState)? onPressed,
Key? key, Key? key,
}) => }) =>

View File

@ -25,6 +25,7 @@ import 'package:wyatt_ui_kit/src/components/buttons/cubit/invalid_button_cubit.d
import 'package:wyatt_ui_kit/src/components/gradients/gradient_text.dart'; import 'package:wyatt_ui_kit/src/components/gradients/gradient_text.dart';
import 'package:wyatt_ui_kit/src/core/extensions/theme_extensions.dart'; import 'package:wyatt_ui_kit/src/core/extensions/theme_extensions.dart';
import 'package:wyatt_ui_kit/src/core/helpers/linear_gradient_helper.dart'; import 'package:wyatt_ui_kit/src/core/helpers/linear_gradient_helper.dart';
import 'package:wyatt_ui_kit/src/domain/button_theme_extension/file_selection_button_theme_extension.dart';
class FileSelectionButtonScreen class FileSelectionButtonScreen
extends CubitScreen<InvalidButtonCubit, ButtonState> { extends CubitScreen<InvalidButtonCubit, ButtonState> {
@ -47,6 +48,7 @@ class FileSelectionButtonScreen
final Widget? leading; final Widget? leading;
final TextWrapper? title; final TextWrapper? title;
final TextWrapper? subTitle; final TextWrapper? subTitle;
final MainAxisSize? mainAxisSize;
final FileSelectionButtonStyle? disabledStyle; final FileSelectionButtonStyle? disabledStyle;
final FileSelectionButtonStyle? normalStyle; final FileSelectionButtonStyle? normalStyle;
@ -58,44 +60,164 @@ class FileSelectionButtonScreen
final void Function(ControlState state)? onPressed; final void Function(ControlState state)? onPressed;
final MainAxisSize? mainAxisSize;
@override @override
InvalidButtonCubit create(BuildContext context) => InvalidButtonCubit(); InvalidButtonCubit create(BuildContext context) => InvalidButtonCubit();
@override /// Negotiate the theme to get a complete style.
Widget onBuild(BuildContext context, ButtonState state) { FileSelectionButtonStyle negotiate(BuildContext context, ButtonState state) {
// Set a default style // Define default style from Flutter values.
FileSelectionButtonStyle? style = FileSelectionButtonStyle style =
normalStyle ?? const FileSelectionButtonStyle(); FileSelectionButtonStyle.fromFlutter(context);
// Try to retrieve custom theme extension
final fileSelectionButtonThemeExtension =
context.themeExtension<FileSelectionButtonThemeExtension>();
switch (state.state) { switch (state.state) {
case ControlState.disabled: case ControlState.disabled:
style = disabledStyle ?? style; style = disabledStyle ??
fileSelectionButtonThemeExtension?.disabledStyle ??
style.copyWith(
foregroundColors:
MultiColor.single(context.colorScheme.onSurface),
backgroundColors: MultiColor.single(context.colorScheme.surface),
);
break; break;
case ControlState.hovered: case ControlState.hovered:
style = hoveredStyle ?? style; style = hoveredStyle ??
fileSelectionButtonThemeExtension?.hoveredStyle ??
style;
break; break;
case ControlState.tapped: case ControlState.tapped:
style = tappedStyle ?? style; style = tappedStyle ??
fileSelectionButtonThemeExtension?.tappedStyle ??
style;
break; break;
case ControlState.focused: case ControlState.focused:
style = focusedStyle ?? style; style = focusedStyle ??
fileSelectionButtonThemeExtension?.focusedStyle ??
style;
break; break;
case ControlState.normal: case ControlState.normal:
// already done style = normalStyle ??
fileSelectionButtonThemeExtension?.normalStyle ??
style;
break; break;
} }
// Apply extra theme
if (state.isSelected) { if (state.isSelected) {
style = selectedStyle ?? style; // TODO(hpcl): enhance copyWith to copy only non-null attributes of an object
style = style.copyWith(
title: (selectedStyle ??
fileSelectionButtonThemeExtension?.selectedStyle ??
style)
.title,
subTitle: (selectedStyle ??
fileSelectionButtonThemeExtension?.selectedStyle ??
style)
.subTitle,
radius: (selectedStyle ??
fileSelectionButtonThemeExtension?.selectedStyle ??
style)
.radius,
padding: (selectedStyle ??
fileSelectionButtonThemeExtension?.selectedStyle ??
style)
.padding,
foregroundColors: (selectedStyle ??
fileSelectionButtonThemeExtension?.selectedStyle ??
style)
.foregroundColors,
backgroundColors: (selectedStyle ??
fileSelectionButtonThemeExtension?.selectedStyle ??
style)
.backgroundColors,
borderColors: (selectedStyle ??
fileSelectionButtonThemeExtension?.selectedStyle ??
style)
.borderColors,
stroke: (selectedStyle ??
fileSelectionButtonThemeExtension?.selectedStyle ??
style)
.stroke,
shadow: (selectedStyle ??
fileSelectionButtonThemeExtension?.selectedStyle ??
style)
.shadow,
);
} }
if (state.isInvalid) { if (state.isInvalid) {
style = invalidStyle ?? style; // TODO(hpcl): enhance copyWith to copy only non-null attributes of an object
style = style.copyWith(
title: (invalidStyle ??
fileSelectionButtonThemeExtension?.invalidStyle ??
style)
.title,
subTitle: (invalidStyle ??
fileSelectionButtonThemeExtension?.invalidStyle ??
style)
.subTitle,
radius: (invalidStyle ??
fileSelectionButtonThemeExtension?.invalidStyle ??
style)
.radius,
padding: (invalidStyle ??
fileSelectionButtonThemeExtension?.invalidStyle ??
style)
.padding,
foregroundColors: (invalidStyle ??
fileSelectionButtonThemeExtension?.invalidStyle ??
style)
.foregroundColors,
backgroundColors: (invalidStyle ??
fileSelectionButtonThemeExtension?.invalidStyle ??
style)
.backgroundColors,
borderColors: (invalidStyle ??
fileSelectionButtonThemeExtension?.invalidStyle ??
style)
.borderColors,
stroke: (invalidStyle ??
fileSelectionButtonThemeExtension?.invalidStyle ??
style)
.stroke,
shadow: (invalidStyle ??
fileSelectionButtonThemeExtension?.invalidStyle ??
style)
.shadow,
);
} }
print(state); return style;
}
Widget _border(
BuildContext context,
FileSelectionButtonStyle style,
Widget child,
) {
if (style.borderColors != null && style.stroke != null) {
return DottedBorder(
padding: EdgeInsets.zero,
dashPattern: const [5, 5],
strokeWidth: style.stroke!,
color: style.borderColors!.color,
borderType: BorderType.RRect,
radius:
style.radius?.resolve(TextDirection.ltr).bottomLeft ?? Radius.zero,
strokeCap: StrokeCap.square,
child: child,
);
} else {
return child;
}
}
@override
Widget onBuild(BuildContext context, ButtonState state) {
final style = negotiate(context, state);
return Focus( return Focus(
onFocusChange: (hasFocus) => onFocusChange: (hasFocus) =>
@ -128,18 +250,12 @@ class FileSelectionButtonScreen
onPressed?.call(state.state); onPressed?.call(state.state);
bloc(context).onClickUpOut(); bloc(context).onClickUpOut();
}, },
child: DottedBorder( child: _border(
padding: EdgeInsets.zero, context,
dashPattern: const [5, 5], style,
strokeWidth: style.stroke ?? 3, DecoratedBox(
color: style.borderColors?.color ?? context.colorScheme.primary,
borderType: BorderType.RRect,
radius: Radius.circular(style.radius ?? 0),
strokeCap: StrokeCap.square,
child: DecoratedBox(
decoration: BoxDecoration( decoration: BoxDecoration(
color: style.backgroundColors?.color ?? color: style.backgroundColors?.color,
context.colorScheme.primary,
// if no gradient colors => no default value // if no gradient colors => no default value
gradient: (style.backgroundColors?.isGradient ?? false) gradient: (style.backgroundColors?.isGradient ?? false)
? LinearGradient( ? LinearGradient(
@ -149,9 +265,7 @@ class FileSelectionButtonScreen
boxShadow: [ boxShadow: [
if (style.shadow != null) ...[style.shadow!] if (style.shadow != null) ...[style.shadow!]
], ],
borderRadius: BorderRadius.all( borderRadius: style.radius,
Radius.circular(style.radius ?? 0),
),
), ),
child: ConstrainedBox( child: ConstrainedBox(
constraints: const BoxConstraints( constraints: const BoxConstraints(
@ -159,81 +273,70 @@ class FileSelectionButtonScreen
minHeight: 50, minHeight: 50,
), // min sizes for Material buttons ), // min sizes for Material buttons
child: Padding( child: Padding(
padding: EdgeInsets.all(style.padding ?? 0), padding: style.padding ?? EdgeInsets.zero,
child: Row( child: Row(
mainAxisSize: mainAxisSize ?? MainAxisSize.min, mainAxisSize: mainAxisSize ?? MainAxisSize.min,
mainAxisAlignment: MainAxisAlignment.spaceBetween, mainAxisAlignment: MainAxisAlignment.spaceBetween,
children: [ children: [
if (leading != null) ...[ if (leading != null) ...[
leading ?? const SizedBox.shrink(), leading ?? const SizedBox.shrink(),
Gap(style.padding ?? 10), Gap(style.padding?.horizontal ?? 10),
], ],
// Choose color
// label.style.color ??
// buttonStyle.foregroundColor.color ??
// context.textTheme.titleLarge.color
//
// Choose gradient
// label.gradient ??
// buttonStyle.foregroundColor.colors ??
// null
Column( Column(
mainAxisAlignment: MainAxisAlignment.spaceBetween, mainAxisAlignment: MainAxisAlignment.spaceBetween,
crossAxisAlignment: CrossAxisAlignment.start, crossAxisAlignment: CrossAxisAlignment.start,
children: [ children: [
/// Choose color
/// title.style.color ??
/// buttonStyle.title.style.color ??
/// context.textTheme.titleLarge.color
///
/// Choose gradient
/// title.gradient ??
/// buttonStyle.foregroundColor.colors ??
/// null
///
/// More infos in `negociate()` method
if (title != null) ...[ if (title != null) ...[
Builder( Gap(style.padding?.horizontal ?? 10),
builder: (context) { Text(
final color = title?.style?.color ?? title!.text,
style?.foregroundColors?.color ?? style: title!.style ?? style.title,
context.textTheme.titleLarge?.color; ).toGradient(
final buttonStyleGradient = LinearGradientHelper.fromNullableColors(
(style?.foregroundColors?.isGradient ?? title?.gradient ??
((style.foregroundColors?.isGradient ??
false) false)
? style?.foregroundColors?.colors ? style.foregroundColors?.colors
: null; : null),
final gradient = ),
title?.gradient ?? buttonStyleGradient;
return Text(
title!.text,
style: (title!.style ??
context.textTheme.titleLarge)
?.copyWith(color: color),
).toGradient(
LinearGradientHelper.fromNullableColors(
gradient,
),
);
},
), ),
], ],
if (subTitle != null) ...[
Builder(
builder: (context) {
final color = subTitle?.style?.color ??
style?.foregroundColors?.color ??
context.textTheme.bodyMedium?.color;
final buttonStyleGradient =
(style?.foregroundColors?.isGradient ??
false)
? style?.foregroundColors?.colors
: null;
final gradient =
subTitle?.gradient ?? buttonStyleGradient;
return Text( /// Choose color
subTitle!.text, /// subTitle.style.color ??
style: (subTitle!.style ?? /// buttonStyle.subTitle.style.color ??
context.textTheme.bodyMedium) /// context.textTheme.subTitleLarge.color
?.copyWith(color: color), ///
).toGradient( /// Choose gradient
LinearGradientHelper.fromNullableColors( /// subTitle.gradient ??
gradient, /// buttonStyle.foregroundColor.colors ??
), /// null
); ///
}, /// More infos in `negociate()` method
if (subTitle != null) ...[
Gap(style.padding?.horizontal ?? 10),
Text(
subTitle!.text,
style: subTitle!.style ?? style.subTitle,
).toGradient(
LinearGradientHelper.fromNullableColors(
subTitle?.gradient ??
((style.foregroundColors?.isGradient ??
false)
? style.foregroundColors?.colors
: null),
),
), ),
], ],
], ],

View File

@ -19,19 +19,19 @@ class $FlatButtonCWProxyImpl implements $FlatButtonComponentCWProxy {
@override @override
FlatButton label(TextWrapper? label) => this(label: label); FlatButton label(TextWrapper? label) => this(label: label);
@override @override
FlatButton disabledStyle(ButtonStyle? disabledStyle) => FlatButton disabledStyle(ButtonStyle<dynamic>? disabledStyle) =>
this(disabledStyle: disabledStyle); this(disabledStyle: disabledStyle);
@override @override
FlatButton normalStyle(ButtonStyle? normalStyle) => FlatButton normalStyle(ButtonStyle<dynamic>? normalStyle) =>
this(normalStyle: normalStyle); this(normalStyle: normalStyle);
@override @override
FlatButton hoveredStyle(ButtonStyle? hoveredStyle) => FlatButton hoveredStyle(ButtonStyle<dynamic>? hoveredStyle) =>
this(hoveredStyle: hoveredStyle); this(hoveredStyle: hoveredStyle);
@override @override
FlatButton focusedStyle(ButtonStyle? focusedStyle) => FlatButton focusedStyle(ButtonStyle<dynamic>? focusedStyle) =>
this(focusedStyle: focusedStyle); this(focusedStyle: focusedStyle);
@override @override
FlatButton tappedStyle(ButtonStyle? tappedStyle) => FlatButton tappedStyle(ButtonStyle<dynamic>? tappedStyle) =>
this(tappedStyle: tappedStyle); this(tappedStyle: tappedStyle);
@override @override
FlatButton onPressed(void Function(ControlState)? onPressed) => FlatButton onPressed(void Function(ControlState)? onPressed) =>
@ -44,11 +44,11 @@ class $FlatButtonCWProxyImpl implements $FlatButtonComponentCWProxy {
Widget? prefix, Widget? prefix,
Widget? suffix, Widget? suffix,
TextWrapper? label, TextWrapper? label,
ButtonStyle? disabledStyle, ButtonStyle<dynamic>? disabledStyle,
ButtonStyle? normalStyle, ButtonStyle<dynamic>? normalStyle,
ButtonStyle? hoveredStyle, ButtonStyle<dynamic>? hoveredStyle,
ButtonStyle? focusedStyle, ButtonStyle<dynamic>? focusedStyle,
ButtonStyle? tappedStyle, ButtonStyle<dynamic>? tappedStyle,
void Function(ControlState)? onPressed, void Function(ControlState)? onPressed,
Key? key, Key? key,
}) => }) =>

View File

@ -24,6 +24,7 @@ import 'package:wyatt_ui_kit/src/components/gradients/gradient_box_border.dart';
import 'package:wyatt_ui_kit/src/components/gradients/gradient_text.dart'; import 'package:wyatt_ui_kit/src/components/gradients/gradient_text.dart';
import 'package:wyatt_ui_kit/src/core/extensions/theme_extensions.dart'; import 'package:wyatt_ui_kit/src/core/extensions/theme_extensions.dart';
import 'package:wyatt_ui_kit/src/core/helpers/linear_gradient_helper.dart'; import 'package:wyatt_ui_kit/src/core/helpers/linear_gradient_helper.dart';
import 'package:wyatt_ui_kit/src/domain/button_theme_extension/flat_button_theme_extension.dart';
class FlatButtonScreen extends CubitScreen<ButtonCubit, ButtonState> { class FlatButtonScreen extends CubitScreen<ButtonCubit, ButtonState> {
const FlatButtonScreen({ const FlatButtonScreen({
@ -43,6 +44,7 @@ class FlatButtonScreen extends CubitScreen<ButtonCubit, ButtonState> {
final Widget? prefix; final Widget? prefix;
final Widget? suffix; final Widget? suffix;
final TextWrapper? label; final TextWrapper? label;
final MainAxisSize? mainAxisSize;
final FlatButtonStyle? disabledStyle; final FlatButtonStyle? disabledStyle;
final FlatButtonStyle? normalStyle; final FlatButtonStyle? normalStyle;
@ -52,34 +54,49 @@ class FlatButtonScreen extends CubitScreen<ButtonCubit, ButtonState> {
final void Function(ControlState state)? onPressed; final void Function(ControlState state)? onPressed;
final MainAxisSize? mainAxisSize;
@override @override
ButtonCubit create(BuildContext context) => ButtonCubit(); ButtonCubit create(BuildContext context) => ButtonCubit();
@override /// Negotiate the theme to get a complete style.
Widget onBuild(BuildContext context, ButtonState state) { FlatButtonStyle negotiate(BuildContext context, ControlState state) {
// Set a default style // Define default style from Flutter values.
FlatButtonStyle? style = normalStyle ?? const FlatButtonStyle(); FlatButtonStyle style = FlatButtonStyle.fromFlutter(context);
switch (state.state) { // Try to retrieve custom theme extension
final flatButtonThemeExtension =
context.themeExtension<FlatButtonThemeExtension>();
switch (state) {
case ControlState.disabled: case ControlState.disabled:
style = disabledStyle ?? style; style = disabledStyle ??
flatButtonThemeExtension?.disabledStyle ??
style.copyWith(
foregroundColors:
MultiColor.single(context.colorScheme.onSurface),
backgroundColors: MultiColor.single(context.colorScheme.surface),
);
break; break;
case ControlState.hovered: case ControlState.hovered:
style = hoveredStyle ?? style; style = hoveredStyle ?? flatButtonThemeExtension?.hoveredStyle ?? style;
break; break;
case ControlState.tapped: case ControlState.tapped:
style = tappedStyle ?? style; style = tappedStyle ?? flatButtonThemeExtension?.tappedStyle ?? style;
break; break;
case ControlState.focused: case ControlState.focused:
style = focusedStyle ?? style; style = focusedStyle ?? flatButtonThemeExtension?.focusedStyle ?? style;
break; break;
case ControlState.normal: case ControlState.normal:
// already done style = normalStyle ?? flatButtonThemeExtension?.normalStyle ?? style;
break; break;
} }
return style;
}
@override
Widget onBuild(BuildContext context, ButtonState state) {
final style = negotiate(context, state.state);
return Focus( return Focus(
onFocusChange: (hasFocus) => onFocusChange: (hasFocus) =>
hasFocus ? bloc(context).onFocus() : bloc(context).onUnfocus(), hasFocus ? bloc(context).onFocus() : bloc(context).onUnfocus(),
@ -113,20 +130,19 @@ class FlatButtonScreen extends CubitScreen<ButtonCubit, ButtonState> {
}, },
child: DecoratedBox( child: DecoratedBox(
decoration: BoxDecoration( decoration: BoxDecoration(
color: style.backgroundColors?.color ?? color: style.backgroundColors?.color,
context.colorScheme.primary,
// If no border color => no default value // If no border color => no default value
border: (style.borderColors != null) border: (style.borderColors != null && style.stroke != null)
? (style.borderColors?.isGradient ?? false) ? (style.borderColors?.isGradient ?? false)
? GradientBoxBorder( ? GradientBoxBorder(
gradient: LinearGradient( gradient: LinearGradient(
colors: style.borderColors!.colors, colors: style.borderColors!.colors,
), ),
width: style.stroke ?? 2, width: style.stroke!,
) )
: Border.all( : Border.all(
color: style.borderColors!.color, color: style.borderColors!.color,
width: style.stroke ?? 2, width: style.stroke!,
) )
: null, : null,
// if no gradient colors => no default value // if no gradient colors => no default value
@ -138,9 +154,7 @@ class FlatButtonScreen extends CubitScreen<ButtonCubit, ButtonState> {
boxShadow: [ boxShadow: [
if (style.shadow != null) ...[style.shadow!] if (style.shadow != null) ...[style.shadow!]
], ],
borderRadius: BorderRadius.all( borderRadius: style.radius,
Radius.circular(style.radius ?? 0),
),
), ),
child: ConstrainedBox( child: ConstrainedBox(
constraints: const BoxConstraints( constraints: const BoxConstraints(
@ -148,76 +162,62 @@ class FlatButtonScreen extends CubitScreen<ButtonCubit, ButtonState> {
minHeight: 36, minHeight: 36,
), // min sizes for Material buttons ), // min sizes for Material buttons
child: Padding( child: Padding(
padding: EdgeInsets.all(style.padding ?? 0), padding: style.padding ?? EdgeInsets.zero,
child: Row( child: Row(
mainAxisSize: mainAxisSize ?? MainAxisSize.min, mainAxisSize: mainAxisSize ?? MainAxisSize.min,
mainAxisAlignment: MainAxisAlignment.spaceBetween, mainAxisAlignment: MainAxisAlignment.spaceBetween,
children: [ children: [
Builder( if (style.foregroundColors?.color != null &&
builder: (context) { prefix != null) ...[
final color = style?.foregroundColors?.color; ColorFiltered(
if (color != null) { colorFilter: ColorFilter.mode(
return ColorFiltered( style.foregroundColors!.color,
colorFilter: BlendMode.srcIn,
ColorFilter.mode(color, BlendMode.srcIn), ),
child: prefix ?? const SizedBox.shrink(), child: prefix,
); )
} else { ] else ...[
return prefix ?? const SizedBox.shrink(); prefix ?? const SizedBox.shrink()
} ],
}, Gap(style.padding?.vertical ?? 10),
),
Gap(style.padding ?? 10), /// Choose color
// Choose color /// label.style.color ??
// label.style.color ?? /// buttonStyle.label.style.color ??
// buttonStyle.foregroundColor.color ?? /// context.textTheme.labelLarge.color
// context.textTheme.titleLarge.color ///
// /// Choose gradient
// Choose gradient /// label.gradient ??
// label.gradient ?? /// buttonStyle.foregroundColor.colors ??
// buttonStyle.foregroundColor.colors ?? /// null
// null ///
if (label != null) ...[ /// More infos in `negociate()` method
Builder( if (label != null) ...[
builder: (context) { Text(
final color = label?.style?.color ?? label!.text,
style?.foregroundColors?.color ?? style: label!.style ?? style.label,
context.textTheme.titleLarge?.color; ).toGradient(
final buttonStyleGradient = LinearGradientHelper.fromNullableColors(
(style?.foregroundColors?.isGradient ?? false) label?.gradient ??
? style?.foregroundColors?.colors ((style.foregroundColors?.isGradient ?? false)
: null; ? style.foregroundColors?.colors
final gradient = : null),
label?.gradient ?? buttonStyleGradient; ),
)
return Text( ],
label!.text, Gap(style.padding?.vertical ?? 10),
style: if (style.foregroundColors?.color != null &&
(label!.style ?? context.textTheme.titleLarge) suffix != null) ...[
?.copyWith(color: color), ColorFiltered(
).toGradient( colorFilter: ColorFilter.mode(
LinearGradientHelper.fromNullableColors( style.foregroundColors!.color,
gradient, BlendMode.srcIn,
), ),
); child: suffix,
}, )
), ] else ...[
suffix ?? const SizedBox.shrink()
], ],
Gap(style.padding ?? 10),
Builder(
builder: (context) {
final color = style?.foregroundColors?.color;
if (color != null) {
return ColorFiltered(
colorFilter:
ColorFilter.mode(color, BlendMode.srcIn),
child: suffix ?? const SizedBox.shrink(),
);
} else {
return suffix ?? const SizedBox.shrink();
}
},
),
], ],
), ),
), ),

View File

@ -33,7 +33,6 @@ class SimpleIconButton extends SimpleIconButtonComponent
super.hoveredStyle, super.hoveredStyle,
super.focusedStyle, super.focusedStyle,
super.tappedStyle, super.tappedStyle,
super.selectedStyle,
super.onPressed, super.onPressed,
super.key, super.key,
}); });
@ -63,9 +62,6 @@ class SimpleIconButton extends SimpleIconButtonComponent
SimpleIconButtonStyle? get tappedStyle => SimpleIconButtonStyle? get tappedStyle =>
super.tappedStyle as SimpleIconButtonStyle?; super.tappedStyle as SimpleIconButtonStyle?;
@override
SimpleIconButtonStyle? get selectedStyle =>
super.selectedStyle as SimpleIconButtonStyle?;
@override @override
Widget build(BuildContext context) => exportBloc( Widget build(BuildContext context) => exportBloc(
@ -76,7 +72,6 @@ class SimpleIconButton extends SimpleIconButtonComponent
hoveredStyle: hoveredStyle, hoveredStyle: hoveredStyle,
focusedStyle: focusedStyle, focusedStyle: focusedStyle,
tappedStyle: tappedStyle, tappedStyle: tappedStyle,
selectedStyle: selectedStyle,
onPressed: onPressed, onPressed: onPressed,
key: key, key: key,
), ),

View File

@ -13,24 +13,21 @@ class $SimpleIconButtonCWProxyImpl
@override @override
SimpleIconButton icon(Icon? icon) => this(icon: icon); SimpleIconButton icon(Icon? icon) => this(icon: icon);
@override @override
SimpleIconButton disabledStyle(ButtonStyle? disabledStyle) => SimpleIconButton disabledStyle(ButtonStyle<dynamic>? disabledStyle) =>
this(disabledStyle: disabledStyle); this(disabledStyle: disabledStyle);
@override @override
SimpleIconButton normalStyle(ButtonStyle? normalStyle) => SimpleIconButton normalStyle(ButtonStyle<dynamic>? normalStyle) =>
this(normalStyle: normalStyle); this(normalStyle: normalStyle);
@override @override
SimpleIconButton hoveredStyle(ButtonStyle? hoveredStyle) => SimpleIconButton hoveredStyle(ButtonStyle<dynamic>? hoveredStyle) =>
this(hoveredStyle: hoveredStyle); this(hoveredStyle: hoveredStyle);
@override @override
SimpleIconButton focusedStyle(ButtonStyle? focusedStyle) => SimpleIconButton focusedStyle(ButtonStyle<dynamic>? focusedStyle) =>
this(focusedStyle: focusedStyle); this(focusedStyle: focusedStyle);
@override @override
SimpleIconButton tappedStyle(ButtonStyle? tappedStyle) => SimpleIconButton tappedStyle(ButtonStyle<dynamic>? tappedStyle) =>
this(tappedStyle: tappedStyle); this(tappedStyle: tappedStyle);
@override @override
SimpleIconButton selectedStyle(ButtonStyle? selectedStyle) =>
this(selectedStyle: selectedStyle);
@override
SimpleIconButton onPressed(void Function(ControlState)? onPressed) => SimpleIconButton onPressed(void Function(ControlState)? onPressed) =>
this(onPressed: onPressed); this(onPressed: onPressed);
@override @override
@ -38,12 +35,11 @@ class $SimpleIconButtonCWProxyImpl
@override @override
SimpleIconButton call({ SimpleIconButton call({
Icon? icon, Icon? icon,
ButtonStyle? disabledStyle, ButtonStyle<dynamic>? disabledStyle,
ButtonStyle? normalStyle, ButtonStyle<dynamic>? normalStyle,
ButtonStyle? hoveredStyle, ButtonStyle<dynamic>? hoveredStyle,
ButtonStyle? focusedStyle, ButtonStyle<dynamic>? focusedStyle,
ButtonStyle? tappedStyle, ButtonStyle<dynamic>? tappedStyle,
ButtonStyle? selectedStyle,
void Function(ControlState)? onPressed, void Function(ControlState)? onPressed,
Key? key, Key? key,
}) => }) =>
@ -54,7 +50,6 @@ class $SimpleIconButtonCWProxyImpl
hoveredStyle: hoveredStyle ?? _value.hoveredStyle, hoveredStyle: hoveredStyle ?? _value.hoveredStyle,
focusedStyle: focusedStyle ?? _value.focusedStyle, focusedStyle: focusedStyle ?? _value.focusedStyle,
tappedStyle: tappedStyle ?? _value.tappedStyle, tappedStyle: tappedStyle ?? _value.tappedStyle,
selectedStyle: selectedStyle ?? _value.selectedStyle,
onPressed: onPressed ?? _value.onPressed, onPressed: onPressed ?? _value.onPressed,
key: key ?? _value.key, key: key ?? _value.key,
); );

View File

@ -20,7 +20,10 @@ import 'package:wyatt_bloc_helper/wyatt_bloc_helper.dart';
import 'package:wyatt_ui_components/wyatt_wyatt_ui_components.dart'; import 'package:wyatt_ui_components/wyatt_wyatt_ui_components.dart';
import 'package:wyatt_ui_kit/src/components/buttons/cubit/button_cubit.dart'; import 'package:wyatt_ui_kit/src/components/buttons/cubit/button_cubit.dart';
import 'package:wyatt_ui_kit/src/components/gradients/gradient_box_border.dart'; import 'package:wyatt_ui_kit/src/components/gradients/gradient_box_border.dart';
import 'package:wyatt_ui_kit/src/components/gradients/gradient_icon.dart';
import 'package:wyatt_ui_kit/src/core/extensions/theme_extensions.dart'; import 'package:wyatt_ui_kit/src/core/extensions/theme_extensions.dart';
import 'package:wyatt_ui_kit/src/core/helpers/linear_gradient_helper.dart';
import 'package:wyatt_ui_kit/src/domain/button_theme_extension/simple_icon_button_theme_extension.dart';
class SimpleIconButtonScreen extends CubitScreen<ButtonCubit, ButtonState> { class SimpleIconButtonScreen extends CubitScreen<ButtonCubit, ButtonState> {
const SimpleIconButtonScreen({ const SimpleIconButtonScreen({
@ -30,7 +33,6 @@ class SimpleIconButtonScreen extends CubitScreen<ButtonCubit, ButtonState> {
this.hoveredStyle, this.hoveredStyle,
this.focusedStyle, this.focusedStyle,
this.tappedStyle, this.tappedStyle,
this.selectedStyle,
this.onPressed, this.onPressed,
super.key, super.key,
}); });
@ -42,35 +44,58 @@ class SimpleIconButtonScreen extends CubitScreen<ButtonCubit, ButtonState> {
final SimpleIconButtonStyle? hoveredStyle; final SimpleIconButtonStyle? hoveredStyle;
final SimpleIconButtonStyle? focusedStyle; final SimpleIconButtonStyle? focusedStyle;
final SimpleIconButtonStyle? tappedStyle; final SimpleIconButtonStyle? tappedStyle;
final SimpleIconButtonStyle? selectedStyle;
final void Function(ControlState state)? onPressed; final void Function(ControlState state)? onPressed;
@override @override
ButtonCubit create(BuildContext context) => ButtonCubit(); ButtonCubit create(BuildContext context) => ButtonCubit();
@override /// Negotiate the theme to get a complete style.
Widget onBuild(BuildContext context, ButtonState state) { SimpleIconButtonStyle negotiate(BuildContext context, ControlState state) {
// Set a default style // Define default style from Flutter values.
SimpleIconButtonStyle? style = normalStyle ?? const SimpleIconButtonStyle(); SimpleIconButtonStyle style = SimpleIconButtonStyle.fromFlutter(context);
switch (state.state) { // Try to retrieve custom theme extension
final simpleIconButtonThemeExtension =
context.themeExtension<SimpleIconButtonThemeExtension>();
switch (state) {
case ControlState.disabled: case ControlState.disabled:
style = disabledStyle ?? style; style = disabledStyle ??
simpleIconButtonThemeExtension?.disabledStyle ??
style.copyWith(
foregroundColors:
MultiColor.single(context.colorScheme.onSurface),
backgroundColors: MultiColor.single(context.colorScheme.surface),
);
break; break;
case ControlState.hovered: case ControlState.hovered:
style = hoveredStyle ?? style; style = hoveredStyle ??
simpleIconButtonThemeExtension?.hoveredStyle ??
style;
break; break;
case ControlState.tapped: case ControlState.tapped:
style = tappedStyle ?? style; style =
tappedStyle ?? simpleIconButtonThemeExtension?.tappedStyle ?? style;
break; break;
case ControlState.focused: case ControlState.focused:
style = focusedStyle ?? style; style = focusedStyle ??
simpleIconButtonThemeExtension?.focusedStyle ??
style;
break; break;
case ControlState.normal: case ControlState.normal:
style =
normalStyle ?? simpleIconButtonThemeExtension?.normalStyle ?? style;
break; break;
} }
return style;
}
@override
Widget onBuild(BuildContext context, ButtonState state) {
final style = negotiate(context, state.state);
return Focus( return Focus(
onFocusChange: (hasFocus) => onFocusChange: (hasFocus) =>
hasFocus ? bloc(context).onFocus() : bloc(context).onUnfocus(), hasFocus ? bloc(context).onFocus() : bloc(context).onUnfocus(),
@ -104,25 +129,24 @@ class SimpleIconButtonScreen extends CubitScreen<ButtonCubit, ButtonState> {
bloc(context).onClickUpOut(); bloc(context).onClickUpOut();
}, },
child: SizedBox.square( child: SizedBox.square(
dimension: style.dimension ?? 30, dimension: style.dimension,
child: AspectRatio( child: AspectRatio(
aspectRatio: 1, aspectRatio: 1,
child: DecoratedBox( child: DecoratedBox(
decoration: BoxDecoration( decoration: BoxDecoration(
color: style.backgroundColors?.color ?? color: style.backgroundColors?.color,
context.colorScheme.tertiary,
// If no border color => no default value // If no border color => no default value
border: (style.borderColors != null) border: (style.borderColors != null && style.stroke != null)
? (style.borderColors?.isGradient ?? false) ? (style.borderColors?.isGradient ?? false)
? GradientBoxBorder( ? GradientBoxBorder(
gradient: LinearGradient( gradient: LinearGradient(
colors: style.borderColors!.colors, colors: style.borderColors!.colors,
), ),
width: style.stroke ?? 2, width: style.stroke!,
) )
: Border.all( : Border.all(
color: style.borderColors!.color, color: style.borderColors!.color,
width: style.stroke ?? 2, width: style.stroke!,
) )
: null, : null,
// if no gradient colors => no default value // if no gradient colors => no default value
@ -134,52 +158,40 @@ class SimpleIconButtonScreen extends CubitScreen<ButtonCubit, ButtonState> {
boxShadow: [ boxShadow: [
if (style.shadow != null) ...[style.shadow!] if (style.shadow != null) ...[style.shadow!]
], ],
borderRadius: BorderRadius.all( borderRadius: style.radius,
Radius.circular(style.radius ?? 0),
),
), ),
child: ConstrainedBox( child: ConstrainedBox(
constraints: BoxConstraints( constraints: BoxConstraints(
minWidth: style.dimension ?? 30, minWidth: style.dimension ?? 30,
), // min sizes for Material buttons ),
child: Padding( child: Padding(
padding: EdgeInsets.all(style.padding ?? 0), padding: style.padding ?? EdgeInsets.zero,
child: Center( child: Center(
// Choose color /// Choose color
// button.foreground.colors (gradient) ?? /// icon.color ??
// buttonStyle.foregroundColor.color ?? /// button.foregroundColors.colors ??
// context.colorScheme.onTertiary /// buttonStyle.foregroundColors.colors ??
/// context.buttonTheme.onPrimary
child: Builder( child: Builder(
builder: (context) { builder: (context) {
if (icon?.color != null) { if (icon?.color != null) {
return icon!; return icon!;
} }
final gradient =
(style?.foregroundColors?.isGradient ?? false) if (!(style.foregroundColors?.isGradient ?? false)) {
? LinearGradient( return ColorFiltered(
colors: style!.foregroundColors!.colors, colorFilter: ColorFilter.mode(
) style.foregroundColors!.color,
: null; BlendMode.srcIn,
final color = style?.foregroundColors?.color ??
context.colorScheme.onTertiary;
if (gradient != null) {
return ShaderMask(
blendMode: BlendMode.srcIn,
shaderCallback: (bounds) => gradient.createShader(
Rect.fromLTWH(
0,
0,
bounds.width,
bounds.height,
),
), ),
child: icon, child: icon,
); );
} }
return ColorFiltered(
colorFilter: return icon!.toGradient(
ColorFilter.mode(color, BlendMode.srcIn), LinearGradientHelper.fromMultiColor(
child: icon, style.foregroundColors!,
),
); );
}, },
), ),

View File

@ -17,22 +17,22 @@ class $SymbolButtonCWProxyImpl implements $SymbolButtonComponentCWProxy {
@override @override
SymbolButton icon(Widget? icon) => this(icon: icon); SymbolButton icon(Widget? icon) => this(icon: icon);
@override @override
SymbolButton disabledStyle(ButtonStyle? disabledStyle) => SymbolButton disabledStyle(ButtonStyle<dynamic>? disabledStyle) =>
this(disabledStyle: disabledStyle); this(disabledStyle: disabledStyle);
@override @override
SymbolButton normalStyle(ButtonStyle? normalStyle) => SymbolButton normalStyle(ButtonStyle<dynamic>? normalStyle) =>
this(normalStyle: normalStyle); this(normalStyle: normalStyle);
@override @override
SymbolButton hoveredStyle(ButtonStyle? hoveredStyle) => SymbolButton hoveredStyle(ButtonStyle<dynamic>? hoveredStyle) =>
this(hoveredStyle: hoveredStyle); this(hoveredStyle: hoveredStyle);
@override @override
SymbolButton focusedStyle(ButtonStyle? focusedStyle) => SymbolButton focusedStyle(ButtonStyle<dynamic>? focusedStyle) =>
this(focusedStyle: focusedStyle); this(focusedStyle: focusedStyle);
@override @override
SymbolButton tappedStyle(ButtonStyle? tappedStyle) => SymbolButton tappedStyle(ButtonStyle<dynamic>? tappedStyle) =>
this(tappedStyle: tappedStyle); this(tappedStyle: tappedStyle);
@override @override
SymbolButton selectedStyle(ButtonStyle? selectedStyle) => SymbolButton selectedStyle(ButtonStyle<dynamic>? selectedStyle) =>
this(selectedStyle: selectedStyle); this(selectedStyle: selectedStyle);
@override @override
SymbolButton onPressed(void Function(ControlState)? onPressed) => SymbolButton onPressed(void Function(ControlState)? onPressed) =>
@ -44,12 +44,12 @@ class $SymbolButtonCWProxyImpl implements $SymbolButtonComponentCWProxy {
MainAxisSize? mainAxisSize, MainAxisSize? mainAxisSize,
TextWrapper? label, TextWrapper? label,
Widget? icon, Widget? icon,
ButtonStyle? disabledStyle, ButtonStyle<dynamic>? disabledStyle,
ButtonStyle? normalStyle, ButtonStyle<dynamic>? normalStyle,
ButtonStyle? hoveredStyle, ButtonStyle<dynamic>? hoveredStyle,
ButtonStyle? focusedStyle, ButtonStyle<dynamic>? focusedStyle,
ButtonStyle? tappedStyle, ButtonStyle<dynamic>? tappedStyle,
ButtonStyle? selectedStyle, ButtonStyle<dynamic>? selectedStyle,
void Function(ControlState)? onPressed, void Function(ControlState)? onPressed,
Key? key, Key? key,
}) => }) =>

View File

@ -25,6 +25,7 @@ import 'package:wyatt_ui_kit/src/components/gradients/gradient_box_border.dart';
import 'package:wyatt_ui_kit/src/components/gradients/gradient_text.dart'; import 'package:wyatt_ui_kit/src/components/gradients/gradient_text.dart';
import 'package:wyatt_ui_kit/src/core/extensions/theme_extensions.dart'; import 'package:wyatt_ui_kit/src/core/extensions/theme_extensions.dart';
import 'package:wyatt_ui_kit/src/core/helpers/linear_gradient_helper.dart'; import 'package:wyatt_ui_kit/src/core/helpers/linear_gradient_helper.dart';
import 'package:wyatt_ui_kit/src/domain/button_theme_extension/symbol_button_theme_extension.dart';
class SymbolButtonScreen class SymbolButtonScreen
extends CubitScreen<SelectableButtonCubit, ButtonState> { extends CubitScreen<SelectableButtonCubit, ButtonState> {
@ -44,6 +45,7 @@ class SymbolButtonScreen
final Widget? icon; final Widget? icon;
final TextWrapper? label; final TextWrapper? label;
final MainAxisSize? mainAxisSize;
final SymbolButtonStyle? disabledStyle; final SymbolButtonStyle? disabledStyle;
final SymbolButtonStyle? normalStyle; final SymbolButtonStyle? normalStyle;
@ -54,37 +56,94 @@ class SymbolButtonScreen
final void Function(ControlState state)? onPressed; final void Function(ControlState state)? onPressed;
final MainAxisSize? mainAxisSize;
@override @override
SelectableButtonCubit create(BuildContext context) => SelectableButtonCubit(); SelectableButtonCubit create(BuildContext context) => SelectableButtonCubit();
@override /// Negotiate the theme to get a complete style.
Widget onBuild(BuildContext context, ButtonState state) { SymbolButtonStyle negotiate(BuildContext context, ButtonState state) {
// Set a default style // Define default style from Flutter values.
SymbolButtonStyle? style = normalStyle ?? const SymbolButtonStyle(); SymbolButtonStyle style = SymbolButtonStyle.fromFlutter(context);
// Try to retrieve custom theme extension
final symbolButtonThemeExtension =
context.themeExtension<SymbolButtonThemeExtension>();
switch (state.state) { switch (state.state) {
case ControlState.disabled: case ControlState.disabled:
style = disabledStyle ?? style; style = disabledStyle ??
symbolButtonThemeExtension?.disabledStyle ??
style.copyWith(
foregroundColors:
MultiColor.single(context.colorScheme.onSurface),
backgroundColors: MultiColor.single(context.colorScheme.surface),
);
break; break;
case ControlState.hovered: case ControlState.hovered:
style = hoveredStyle ?? style; style =
hoveredStyle ?? symbolButtonThemeExtension?.hoveredStyle ?? style;
break; break;
case ControlState.tapped: case ControlState.tapped:
style = tappedStyle ?? style; style = tappedStyle ?? symbolButtonThemeExtension?.tappedStyle ?? style;
break; break;
case ControlState.focused: case ControlState.focused:
style = focusedStyle ?? style; style =
focusedStyle ?? symbolButtonThemeExtension?.focusedStyle ?? style;
break; break;
case ControlState.normal: case ControlState.normal:
style = normalStyle ?? symbolButtonThemeExtension?.normalStyle ?? style;
break; break;
} }
// Apply extra theme
if (state.isSelected) { if (state.isSelected) {
style = selectedStyle ?? style; // TODO(hpcl): enhance copyWith to copy only non-null attributes of an object
style = style.copyWith(
label: (selectedStyle ??
symbolButtonThemeExtension?.selectedStyle ??
style)
.label,
dimension: (selectedStyle ??
symbolButtonThemeExtension?.selectedStyle ??
style)
.dimension,
radius: (selectedStyle ??
symbolButtonThemeExtension?.selectedStyle ??
style)
.radius,
padding: (selectedStyle ??
symbolButtonThemeExtension?.selectedStyle ??
style)
.padding,
foregroundColors: (selectedStyle ??
symbolButtonThemeExtension?.selectedStyle ??
style)
.foregroundColors,
backgroundColors: (selectedStyle ??
symbolButtonThemeExtension?.selectedStyle ??
style)
.backgroundColors,
borderColors: (selectedStyle ??
symbolButtonThemeExtension?.selectedStyle ??
style)
.borderColors,
stroke: (selectedStyle ??
symbolButtonThemeExtension?.selectedStyle ??
style)
.stroke,
shadow: (selectedStyle ??
symbolButtonThemeExtension?.selectedStyle ??
style)
.shadow,
);
} }
return style;
}
@override
Widget onBuild(BuildContext context, ButtonState state) {
final style = negotiate(context, state);
return Focus( return Focus(
onFocusChange: (hasFocus) => onFocusChange: (hasFocus) =>
hasFocus ? bloc(context).onFocus() : bloc(context).onUnfocus(), hasFocus ? bloc(context).onFocus() : bloc(context).onUnfocus(),
@ -122,27 +181,27 @@ class SymbolButtonScreen
mainAxisAlignment: MainAxisAlignment.spaceBetween, mainAxisAlignment: MainAxisAlignment.spaceBetween,
children: [ children: [
SizedBox.square( SizedBox.square(
dimension: style.dimension ?? 60, dimension: style.dimension,
child: AspectRatio( child: AspectRatio(
aspectRatio: 1, aspectRatio: 1,
child: DecoratedBox( child: DecoratedBox(
decoration: BoxDecoration( decoration: BoxDecoration(
color: style.backgroundColors?.color ?? color: style.backgroundColors?.color,
context.colorScheme.primary,
// If no border color => no default value // If no border color => no default value
border: (style.borderColors != null) border:
? (style.borderColors?.isGradient ?? false) (style.borderColors != null && style.stroke != null)
? GradientBoxBorder( ? (style.borderColors?.isGradient ?? false)
gradient: LinearGradient( ? GradientBoxBorder(
colors: style.borderColors!.colors, gradient: LinearGradient(
), colors: style.borderColors!.colors,
width: style.stroke ?? 2, ),
) width: style.stroke!,
: Border.all( )
color: style.borderColors!.color, : Border.all(
width: style.stroke ?? 2, color: style.borderColors!.color,
) width: style.stroke!,
: null, )
: null,
// if no gradient colors => no default value // if no gradient colors => no default value
gradient: (style.backgroundColors?.isGradient ?? false) gradient: (style.backgroundColors?.isGradient ?? false)
? LinearGradient( ? LinearGradient(
@ -152,92 +211,44 @@ class SymbolButtonScreen
boxShadow: [ boxShadow: [
if (style.shadow != null) ...[style.shadow!] if (style.shadow != null) ...[style.shadow!]
], ],
borderRadius: BorderRadius.all( borderRadius: style.radius,
Radius.circular(style.radius ?? 0),
),
), ),
child: ConstrainedBox( child: ConstrainedBox(
constraints: BoxConstraints( constraints: BoxConstraints(
minWidth: style.dimension ?? 60, minWidth: style.dimension ?? 60,
), // min sizes for Material buttons ), // min sizes for Material buttons
child: Padding( child: Padding(
padding: EdgeInsets.all(style.padding ?? 0), padding: style.padding ?? EdgeInsets.zero,
child: Center( child: Center(child: icon),
// Choose color
// button.foreground.colors (gradient) ??
// buttonStyle.foregroundColor.color ??
// context.colorScheme.secondary
child: Builder(
builder: (context) {
final gradient =
(style?.foregroundColors?.isGradient ?? false)
? LinearGradient(
colors:
style!.foregroundColors!.colors,
)
: null;
final color = style?.foregroundColors?.color ??
context.colorScheme.secondary;
if (gradient != null) {
return ShaderMask(
blendMode: BlendMode.srcIn,
shaderCallback: (bounds) =>
gradient.createShader(
Rect.fromLTWH(
0,
0,
bounds.width,
bounds.height,
),
),
child: icon,
);
}
return ColorFiltered(
colorFilter:
ColorFilter.mode(color, BlendMode.srcIn),
child: icon,
);
},
),
),
), ),
), ),
), ),
), ),
), ),
// Choose color
// label.style.color ??
// buttonStyle.foregroundColor.color ??
// context.textTheme.titleLarge.color
//
// Choose gradient
// label.gradient ??
// buttonStyle.foregroundColor.colors ??
// null
if (label != null) ...[
const Gap(10),
Builder(
builder: (context) {
final color = label?.style?.color ??
style?.foregroundColors?.color ??
context.textTheme.titleLarge?.color;
final buttonStyleGradient =
(style?.foregroundColors?.isGradient ?? false)
? style?.foregroundColors?.colors
: null;
final gradient = label?.gradient ?? buttonStyleGradient;
return Text( /// Choose color
label!.text, /// label.style.color ??
style: (label!.style ?? context.textTheme.titleMedium) /// buttonStyle.label.style.color ??
?.copyWith(color: color), /// context.textTheme.labelLarge.color
).toGradient( ///
LinearGradientHelper.fromNullableColors( /// Choose gradient
gradient, /// label.gradient ??
), /// buttonStyle.foregroundColor.colors ??
); /// null
}, ///
/// More infos in `negociate()` method
if (label != null) ...[
Gap(style.padding?.horizontal ?? 10),
Text(
label!.text,
style: label!.style ?? style.label,
).toGradient(
LinearGradientHelper.fromNullableColors(
label?.gradient ??
((style.foregroundColors?.isGradient ?? false)
? style.foregroundColors?.colors
: null),
),
), ),
], ],
], ],

View File

@ -16,7 +16,7 @@
import 'package:flutter/widgets.dart'; import 'package:flutter/widgets.dart';
extension GradientTextExtension on Icon { extension GradientIconExtension on Icon {
GradientIcon toGradient(Gradient? gradient) => GradientIcon toGradient(Gradient? gradient) =>
GradientIcon.from(this, gradient); GradientIcon.from(this, gradient);
} }

View File

@ -16,8 +16,6 @@
import 'package:flutter/material.dart'; import 'package:flutter/material.dart';
extension ThemeExtension on BuildContext { extension BuildContextThemeExtension on BuildContext {
TextTheme get textTheme => Theme.of(this).textTheme; T? themeExtension<T>() => Theme.of(this).extension<T>();
ColorScheme get colorScheme => Theme.of(this).colorScheme;
ButtonThemeData get buttonTheme => Theme.of(this).buttonTheme;
} }

View File

@ -0,0 +1,20 @@
// Copyright (C) 2023 WYATT GROUP
// Please see the AUTHORS file for details.
//
// This program is free software: you can redistribute it and/or modify
// it under the terms of the GNU General Public License as published by
// the Free Software Foundation, either version 3 of the License, or
// any later version.
//
// This program is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
// GNU General Public License for more details.
//
// You should have received a copy of the GNU General Public License
// along with this program. If not, see <https://www.gnu.org/licenses/>.
export 'file_selection_button_theme_extension.dart';
export 'flat_button_theme_extension.dart';
export 'simple_icon_button_theme_extension.dart';
export 'symbol_button_theme_extension.dart';

View File

@ -0,0 +1,52 @@
// Copyright (C) 2023 WYATT GROUP
// Please see the AUTHORS file for details.
//
// This program is free software: you can redistribute it and/or modify
// it under the terms of the GNU General Public License as published by
// the Free Software Foundation, either version 3 of the License, or
// any later version.
//
// This program is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
// GNU General Public License for more details.
//
// You should have received a copy of the GNU General Public License
// along with this program. If not, see <https://www.gnu.org/licenses/>.
import 'package:flutter/material.dart';
import 'package:wyatt_ui_components/wyatt_wyatt_ui_components.dart';
abstract class FileSelectionButtonThemeExtension
extends ThemeExtension<FileSelectionButtonThemeExtension> {
const FileSelectionButtonThemeExtension({
this.disabledStyle,
this.focusedStyle,
this.hoveredStyle,
this.normalStyle,
this.tappedStyle,
this.selectedStyle,
this.invalidStyle,
});
/// Style of this button in disabled state
final FileSelectionButtonStyle? disabledStyle;
/// Style of this button in focused state
final FileSelectionButtonStyle? focusedStyle;
/// Style of this button in hovered state
final FileSelectionButtonStyle? hoveredStyle;
/// Style of this button in normal state
final FileSelectionButtonStyle? normalStyle;
/// Style of this button in tapped state
final FileSelectionButtonStyle? tappedStyle;
/// Style of this button in selected state
final FileSelectionButtonStyle? selectedStyle;
/// Style of this button in invalid state
final FileSelectionButtonStyle? invalidStyle;
}

View File

@ -0,0 +1,44 @@
// Copyright (C) 2023 WYATT GROUP
// Please see the AUTHORS file for details.
//
// This program is free software: you can redistribute it and/or modify
// it under the terms of the GNU General Public License as published by
// the Free Software Foundation, either version 3 of the License, or
// any later version.
//
// This program is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
// GNU General Public License for more details.
//
// You should have received a copy of the GNU General Public License
// along with this program. If not, see <https://www.gnu.org/licenses/>.
import 'package:flutter/material.dart';
import 'package:wyatt_ui_components/wyatt_wyatt_ui_components.dart';
abstract class FlatButtonThemeExtension
extends ThemeExtension<FlatButtonThemeExtension> {
const FlatButtonThemeExtension({
this.disabledStyle,
this.focusedStyle,
this.hoveredStyle,
this.normalStyle,
this.tappedStyle,
});
/// Style of this button in disabled state
final FlatButtonStyle? disabledStyle;
/// Style of this button in focused state
final FlatButtonStyle? focusedStyle;
/// Style of this button in hovered state
final FlatButtonStyle? hoveredStyle;
/// Style of this button in normal state
final FlatButtonStyle? normalStyle;
/// Style of this button in tapped state
final FlatButtonStyle? tappedStyle;
}

View File

@ -0,0 +1,44 @@
// Copyright (C) 2023 WYATT GROUP
// Please see the AUTHORS file for details.
//
// This program is free software: you can redistribute it and/or modify
// it under the terms of the GNU General Public License as published by
// the Free Software Foundation, either version 3 of the License, or
// any later version.
//
// This program is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
// GNU General Public License for more details.
//
// You should have received a copy of the GNU General Public License
// along with this program. If not, see <https://www.gnu.org/licenses/>.
import 'package:flutter/material.dart';
import 'package:wyatt_ui_components/wyatt_wyatt_ui_components.dart';
abstract class SimpleIconButtonThemeExtension
extends ThemeExtension<SimpleIconButtonThemeExtension> {
const SimpleIconButtonThemeExtension({
this.disabledStyle,
this.focusedStyle,
this.hoveredStyle,
this.normalStyle,
this.tappedStyle,
});
/// Style of this button in disabled state
final SimpleIconButtonStyle? disabledStyle;
/// Style of this button in focused state
final SimpleIconButtonStyle? focusedStyle;
/// Style of this button in hovered state
final SimpleIconButtonStyle? hoveredStyle;
/// Style of this button in normal state
final SimpleIconButtonStyle? normalStyle;
/// Style of this button in tapped state
final SimpleIconButtonStyle? tappedStyle;
}

View File

@ -0,0 +1,48 @@
// Copyright (C) 2023 WYATT GROUP
// Please see the AUTHORS file for details.
//
// This program is free software: you can redistribute it and/or modify
// it under the terms of the GNU General Public License as published by
// the Free Software Foundation, either version 3 of the License, or
// any later version.
//
// This program is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
// GNU General Public License for more details.
//
// You should have received a copy of the GNU General Public License
// along with this program. If not, see <https://www.gnu.org/licenses/>.
import 'package:flutter/material.dart';
import 'package:wyatt_ui_components/wyatt_wyatt_ui_components.dart';
abstract class SymbolButtonThemeExtension
extends ThemeExtension<SymbolButtonThemeExtension> {
const SymbolButtonThemeExtension({
this.disabledStyle,
this.focusedStyle,
this.hoveredStyle,
this.normalStyle,
this.tappedStyle,
this.selectedStyle,
});
/// Style of this button in disabled state
final SymbolButtonStyle? disabledStyle;
/// Style of this button in focused state
final SymbolButtonStyle? focusedStyle;
/// Style of this button in hovered state
final SymbolButtonStyle? hoveredStyle;
/// Style of this button in normal state
final SymbolButtonStyle? normalStyle;
/// Style of this button in tapped state
final SymbolButtonStyle? tappedStyle;
/// Style of this button in selected state
final SymbolButtonStyle? selectedStyle;
}

View File

@ -0,0 +1,18 @@
// Copyright (C) 2023 WYATT GROUP
// Please see the AUTHORS file for details.
//
// This program is free software: you can redistribute it and/or modify
// it under the terms of the GNU General Public License as published by
// the Free Software Foundation, either version 3 of the License, or
// any later version.
//
// This program is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
// GNU General Public License for more details.
//
// You should have received a copy of the GNU General Public License
// along with this program. If not, see <https://www.gnu.org/licenses/>.
export 'button_theme_extension/button_theme_extension.dart';
export 'card_theme_extension.dart';

View File

@ -14,22 +14,6 @@
// You should have received a copy of the GNU General Public License // You should have received a copy of the GNU General Public License
// along with this program. If not, see <https://www.gnu.org/licenses/>. // along with this program. If not, see <https://www.gnu.org/licenses/>.
// Copyright (C) 2023 WYATT GROUP
// Please see the AUTHORS file for details.
//
// This program is free software: you can redistribute it and/or modify
// it under the terms of the GNU General Public License as published by
// the Free Software Foundation, either version 3 of the License, or
// any later version.
//
// This program is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
// GNU General Public License for more details.
//
// You should have received a copy of the GNU General Public License
// along with this program. If not, see <https://www.gnu.org/licenses/>.
export './components/components.dart'; export './components/components.dart';
export './core/core.dart'; export './core/core.dart';
export './domain/card_theme_extension.dart'; export './domain/domain.dart';

View File

@ -17,4 +17,6 @@
/// UIKit and Design System used in Wyatt Studio. /// UIKit and Design System used in Wyatt Studio.
library wyatt_ui_kit; library wyatt_ui_kit;
export 'package:wyatt_ui_components/src/core/extensions/build_context_extensions.dart';
export './src/src.dart'; export './src/src.dart';