fix: FontFamily and FontSize toolbars track the text selected in the editor. (#1829)

* toggle_style_button : calls to options.afterButtonPressed replaced by call to class function afterButtonPressed to allow default call to base button settings
quill_icon_button: L26 build for isSelected updated to call afterButtonPressed = same as if not selected
QuillController _updateSelection removed param=source because not used; added new param insertNewline when true set tog to style of preceding char (last entered); updated replaceText to call _updateSelection for NL
document collectStyle:  Selecting the start of a line, user expects the style to be the visible style of the line including inline styles

* color_button calls afterButtonPressed
insert at start of line uses style for line

* Remove comments

* Fix formatting issue

* Fix FontFamily and Size button actions

* Fix FontFamily and Size button actions

* Value setting Stateful toolbar buttons derive from base class

* Rename base class as QuillToolbarBaseValueButton

* Fixes for before_push script

* Removed deprecated functions

---------

Co-authored-by: Douglas Ward <dward@scied.com>
pull/1855/head
AtlasAutocode 11 months ago committed by GitHub
parent 9044e2ad49
commit 08d2a2e4f2
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
  1. 6
      example/lib/screens/quill/my_quill_toolbar.dart
  2. 35
      lib/src/widgets/quill/quill_controller.dart
  3. 98
      lib/src/widgets/toolbar/base_button/base_value_button.dart
  4. 121
      lib/src/widgets/toolbar/buttons/font_family_button.dart
  5. 92
      lib/src/widgets/toolbar/buttons/font_size_button.dart
  6. 96
      lib/src/widgets/toolbar/buttons/toggle_check_list_button.dart
  7. 96
      lib/src/widgets/toolbar/buttons/toggle_style_button.dart

@ -225,7 +225,7 @@ class MyQuillToolbar extends StatelessWidget {
// headerStyleType: HeaderStyleType.buttons, // headerStyleType: HeaderStyleType.buttons,
// buttonOptions: QuillSimpleToolbarButtonOptions( // buttonOptions: QuillSimpleToolbarButtonOptions(
// base: QuillToolbarBaseButtonOptions( // base: QuillToolbarBaseButtonOptions(
// afterButtonPressed: focusNode.requestFocus, // afterButtonPressed: focusNode.requestFocus,
// // iconSize: 20, // // iconSize: 20,
// iconTheme: QuillIconTheme( // iconTheme: QuillIconTheme(
// iconButtonSelectedData: IconButtonData( // iconButtonSelectedData: IconButtonData(
@ -239,8 +239,8 @@ class MyQuillToolbar extends StatelessWidget {
// ), // ),
// ), // ),
// ), // ),
// ), // ),
// ), //),
customButtons: [ customButtons: [
QuillToolbarCustomButtonOptions( QuillToolbarCustomButtonOptions(
icon: const Icon(Icons.add_alarm_rounded), icon: const Icon(Icons.add_alarm_rounded),

@ -14,7 +14,6 @@ import '../../models/structs/doc_change.dart';
import '../../models/structs/image_url.dart'; import '../../models/structs/image_url.dart';
import '../../models/structs/offset_value.dart'; import '../../models/structs/offset_value.dart';
import '../../utils/delta.dart'; import '../../utils/delta.dart';
import '../toolbar/buttons/toggle_style_button.dart';
typedef ReplaceTextCallback = bool Function(int index, int len, Object? data); typedef ReplaceTextCallback = bool Function(int index, int len, Object? data);
typedef DeleteCallback = void Function(int cursorPosition, bool forward); typedef DeleteCallback = void Function(int cursorPosition, bool forward);
@ -67,40 +66,6 @@ class QuillController extends ChangeNotifier {
notifyListeners(); notifyListeners();
} }
// Thoses are the values that the user selects and not the one
// from the current line
/// The current font family, null to use the default one
MapEntry<String, String>? _selectedFontFamily;
/// The current font family, null to use the default one
MapEntry<String, String>? get selectedFontFamily => _selectedFontFamily;
void selectFontFamily(MapEntry<String, String>? newFontFamily) {
_selectedFontFamily = newFontFamily;
}
/// The current font size, null to use the default one
MapEntry<String, String>? _selectedFontSize;
/// The current font size, null to use the default one
MapEntry<String, String>? get selectedFontSize => _selectedFontSize;
void selectFontSize(MapEntry<String, String>? newFontSize) {
_selectedFontSize = newFontSize;
}
/// For the [QuillToolbarToggleStyleButton]
final Map<Attribute, bool?> _selectedStyles = {};
/// For the [QuillToolbarToggleStyleButton]
Map<Attribute, bool?> get selectedStyles => _selectedStyles;
/// For the [QuillToolbarToggleStyleButton]
void selectStyle(Attribute attribute, bool value) {
_selectedStyles[attribute] = value;
}
/// Tells whether to keep or reset the [toggledStyle] /// Tells whether to keep or reset the [toggledStyle]
/// when user adds a new line. /// when user adds a new line.
final bool keepStyleOnNewLine; final bool keepStyleOnNewLine;

@ -0,0 +1,98 @@
import 'package:flutter/material.dart';
import '../../../../flutter_quill.dart';
/// The [T] is the options for the button
/// The [E] is the extra options for the button
abstract class QuillToolbarBaseValueButton<
T extends QuillToolbarBaseButtonOptions<T, E>,
E extends QuillToolbarBaseButtonExtraOptions> extends StatefulWidget {
const QuillToolbarBaseValueButton(
{required this.controller, required this.options, super.key});
final T options;
final QuillController controller;
}
/// The [W] is the widget that creates this State
/// The [V] is the type of the currentValue
abstract class QuillToolbarBaseValueButtonState<
W extends QuillToolbarBaseValueButton<T, E>,
T extends QuillToolbarBaseButtonOptions<T, E>,
E extends QuillToolbarBaseButtonExtraOptions,
V> extends State<W> {
T get options => widget.options;
QuillController get controller => widget.controller;
late V currentValue;
/// Callback to query the widget's state for the value to be assigned to currentState
V get currentStateValue;
@override
void initState() {
super.initState();
controller.addListener(didChangeEditingValue);
}
@override
void didChangeDependencies() {
super.didChangeDependencies();
currentValue = currentStateValue;
}
void didChangeEditingValue() {
setState(() => currentValue = currentStateValue);
}
@override
void dispose() {
controller.removeListener(didChangeEditingValue);
super.dispose();
}
@override
void didUpdateWidget(covariant W oldWidget) {
super.didUpdateWidget(oldWidget);
if (oldWidget.controller != controller) {
oldWidget.controller.removeListener(didChangeEditingValue);
controller.addListener(didChangeEditingValue);
currentValue = currentStateValue;
}
}
String get defaultTooltip;
String get tooltip {
return options.tooltip ??
context.quillToolbarBaseButtonOptions?.tooltip ??
defaultTooltip;
}
double get iconSize {
final baseFontSize = baseButtonExtraOptions?.iconSize;
final iconSize = options.iconSize;
return iconSize ?? baseFontSize ?? kDefaultIconSize;
}
double get iconButtonFactor {
final baseIconFactor = baseButtonExtraOptions?.iconButtonFactor;
final iconButtonFactor = options.iconButtonFactor;
return iconButtonFactor ?? baseIconFactor ?? kDefaultIconButtonFactor;
}
QuillIconTheme? get iconTheme {
return options.iconTheme ?? baseButtonExtraOptions?.iconTheme;
}
QuillToolbarBaseButtonOptions? get baseButtonExtraOptions {
return context.quillToolbarBaseButtonOptions;
}
VoidCallback? get afterButtonPressed {
return options.afterButtonPressed ??
baseButtonExtraOptions?.afterButtonPressed;
}
}

@ -4,55 +4,43 @@ import '../../../../extensions.dart';
import '../../../extensions/quill_configurations_ext.dart'; import '../../../extensions/quill_configurations_ext.dart';
import '../../../l10n/extensions/localizations.dart'; import '../../../l10n/extensions/localizations.dart';
import '../../../models/documents/attribute.dart'; import '../../../models/documents/attribute.dart';
import '../../../models/themes/quill_icon_theme.dart'; import '../base_button/base_value_button.dart';
import '../../quill/quill_controller.dart';
import '../base_toolbar.dart'; import '../base_toolbar.dart';
class QuillToolbarFontFamilyButton extends StatefulWidget { class QuillToolbarFontFamilyButton extends QuillToolbarBaseValueButton<
QuillToolbarFontFamilyButtonOptions,
QuillToolbarFontFamilyButtonExtraOptions> {
QuillToolbarFontFamilyButton({ QuillToolbarFontFamilyButton({
required this.controller, required super.controller,
@Deprecated('Please use the default display text from the options') @Deprecated('Please use the default display text from the options')
this.defaultDisplayText, this.defaultDisplayText,
this.options = const QuillToolbarFontFamilyButtonOptions(), super.options = const QuillToolbarFontFamilyButtonOptions(),
super.key, super.key,
}) : assert(options.rawItemsMap?.isNotEmpty ?? (true)), }) : assert(options.rawItemsMap?.isNotEmpty ?? (true)),
assert( assert(
options.initialValue == null || options.initialValue!.isNotEmpty, options.initialValue == null || options.initialValue!.isNotEmpty,
); );
final QuillToolbarFontFamilyButtonOptions options;
final String? defaultDisplayText; final String? defaultDisplayText;
/// Since we can't get the state from the instace of the widget for comparing
/// in [didUpdateWidget] then we will have to store reference here
final QuillController controller;
@override @override
QuillToolbarFontFamilyButtonState createState() => QuillToolbarFontFamilyButtonState createState() =>
QuillToolbarFontFamilyButtonState(); QuillToolbarFontFamilyButtonState();
} }
class QuillToolbarFontFamilyButtonState class QuillToolbarFontFamilyButtonState
extends State<QuillToolbarFontFamilyButton> { extends QuillToolbarBaseValueButtonState<
var _currentValue = ''; QuillToolbarFontFamilyButton,
QuillToolbarFontFamilyButtonOptions,
QuillToolbarFontFamilyButtonOptions get options { QuillToolbarFontFamilyButtonExtraOptions,
return widget.options; String> {
}
@override
void initState() {
super.initState();
_initState();
}
void _initState() {}
@override @override
void didChangeDependencies() { String get currentStateValue {
super.didChangeDependencies(); final attribute =
_currentValue = _defaultDisplayText; controller.getSelectionStyle().attributes[options.attribute.key];
return attribute == null
? _defaultDisplayText
: (_getKeyName(attribute.value) ?? _defaultDisplayText);
} }
String get _defaultDisplayText { String get _defaultDisplayText {
@ -62,27 +50,6 @@ class QuillToolbarFontFamilyButtonState
context.loc.font; context.loc.font;
} }
// @override
// void didUpdateWidget(covariant QuillToolbarFontFamilyButton oldWidget) {
// super.didUpdateWidget(oldWidget);
// if (oldWidget.controller == controller) {
// return;
// }
// controller
// ..removeListener(_didChangeEditingValue)
// ..addListener(_didChangeEditingValue);
// }
// void _didChangeEditingValue() {
// final attribute = _selectionStyle.attributes[options.attribute.key];
// if (attribute == null) {
// setState(() => _currentValue = _defaultDisplayText);
// return;
// }
// final keyName = _getKeyName(attribute.value);
// setState(() => _currentValue = keyName ?? _defaultDisplayText);
// }
Map<String, String> get rawItemsMap { Map<String, String> get rawItemsMap {
final rawItemsMap = final rawItemsMap =
context.quillSimpleToolbarConfigurations?.fontFamilyValues ?? context.quillSimpleToolbarConfigurations?.fontFamilyValues ??
@ -110,38 +77,8 @@ class QuillToolbarFontFamilyButtonState
return null; return null;
} }
QuillController get controller { @override
return widget.controller; String get defaultTooltip => context.loc.fontFamily;
}
double get iconSize {
final baseFontSize = context.quillToolbarBaseButtonOptions?.iconSize;
final iconSize = options.iconSize;
return iconSize ?? baseFontSize ?? kDefaultIconSize;
}
double get iconButtonFactor {
final baseIconFactor =
context.quillToolbarBaseButtonOptions?.iconButtonFactor;
final iconButtonFactor = widget.options.iconButtonFactor;
return iconButtonFactor ?? baseIconFactor ?? kDefaultIconButtonFactor;
}
VoidCallback? get afterButtonPressed {
return options.afterButtonPressed ??
context.quillToolbarBaseButtonOptions?.afterButtonPressed;
}
QuillIconTheme? get iconTheme {
return options.iconTheme ??
context.quillToolbarBaseButtonOptions?.iconTheme;
}
String get tooltip {
return options.tooltip ??
context.quillToolbarBaseButtonOptions?.tooltip ??
context.loc.fontFamily;
}
void _onPressed() { void _onPressed() {
if (_menuController.isOpen) { if (_menuController.isOpen) {
@ -149,7 +86,7 @@ class QuillToolbarFontFamilyButtonState
} else { } else {
_menuController.open(); _menuController.open();
} }
options.afterButtonPressed?.call(); afterButtonPressed?.call();
} }
final _menuController = MenuController(); final _menuController = MenuController();
@ -163,7 +100,7 @@ class QuillToolbarFontFamilyButtonState
return childBuilder( return childBuilder(
options, options,
QuillToolbarFontFamilyButtonExtraOptions( QuillToolbarFontFamilyButtonExtraOptions(
currentValue: _currentValue, currentValue: currentValue,
defaultDisplayText: _defaultDisplayText, defaultDisplayText: _defaultDisplayText,
controller: controller, controller: controller,
context: context, context: context,
@ -177,8 +114,8 @@ class QuillToolbarFontFamilyButtonState
var effectiveTooltip = tooltip; var effectiveTooltip = tooltip;
if (options.overrideTooltipByFontFamily) { if (options.overrideTooltipByFontFamily) {
effectiveTooltip = effectiveTooltip.isNotEmpty effectiveTooltip = effectiveTooltip.isNotEmpty
? '$effectiveTooltip: $_currentValue' ? '$effectiveTooltip: $currentValue'
: '${context.loc.font}: $_currentValue'; : '${context.loc.font}: $currentValue';
} }
return Tooltip(message: effectiveTooltip, child: child); return Tooltip(message: effectiveTooltip, child: child);
}, },
@ -196,9 +133,9 @@ class QuillToolbarFontFamilyButtonState
final keyName = _getKeyName(newValue); final keyName = _getKeyName(newValue);
setState(() { setState(() {
if (keyName != 'Clear') { if (keyName != 'Clear') {
_currentValue = keyName ?? _defaultDisplayText; currentValue = keyName ?? _defaultDisplayText;
} else { } else {
_currentValue = _defaultDisplayText; currentValue = _defaultDisplayText;
} }
if (keyName != null) { if (keyName != null) {
controller.formatSelection( controller.formatSelection(
@ -210,12 +147,6 @@ class QuillToolbarFontFamilyButtonState
options.onSelected?.call(newValue); options.onSelected?.call(newValue);
} }
}); });
if (fontFamily.value == 'Clear') {
controller.selectFontFamily(null);
return;
}
controller.selectFontFamily(fontFamily);
}, },
child: Text( child: Text(
fontFamily.key.toString(), fontFamily.key.toString(),
@ -262,7 +193,7 @@ class QuillToolbarFontFamilyButtonState
enabled: hasFinalWidth, enabled: hasFinalWidth,
wrapper: (child) => Expanded(child: child), wrapper: (child) => Expanded(child: child),
child: Text( child: Text(
widget.controller.selectedFontFamily?.key ?? _currentValue, currentValue,
maxLines: 1, maxLines: 1,
overflow: options.labelOverflow, overflow: options.labelOverflow,
style: options.style ?? style: options.style ??

@ -5,43 +5,35 @@ import '../../../../extensions.dart';
import '../../../extensions/quill_configurations_ext.dart'; import '../../../extensions/quill_configurations_ext.dart';
import '../../../l10n/extensions/localizations.dart'; import '../../../l10n/extensions/localizations.dart';
import '../../../models/documents/attribute.dart'; import '../../../models/documents/attribute.dart';
import '../../../models/themes/quill_icon_theme.dart';
import '../../../utils/font.dart'; import '../../../utils/font.dart';
import '../../quill/quill_controller.dart'; import '../base_button/base_value_button.dart';
import '../base_toolbar.dart'; import '../base_toolbar.dart';
class QuillToolbarFontSizeButton extends StatefulWidget { class QuillToolbarFontSizeButton extends QuillToolbarBaseValueButton<
QuillToolbarFontSizeButtonOptions, QuillToolbarFontSizeButtonExtraOptions> {
QuillToolbarFontSizeButton({ QuillToolbarFontSizeButton({
required this.controller, required super.controller,
@Deprecated('Please use the default display text from the options') @Deprecated('Please use the default display text from the options')
this.defaultDisplayText, this.defaultDisplayText,
this.options = const QuillToolbarFontSizeButtonOptions(), super.options = const QuillToolbarFontSizeButtonOptions(),
super.key, super.key,
}) : assert(options.rawItemsMap?.isNotEmpty ?? true), }) : assert(options.rawItemsMap?.isNotEmpty ?? true),
assert(options.initialValue == null || assert(options.initialValue == null ||
(options.initialValue?.isNotEmpty ?? true)); (options.initialValue?.isNotEmpty ?? true));
final QuillToolbarFontSizeButtonOptions options;
final String? defaultDisplayText; final String? defaultDisplayText;
/// Since we can't get the state from the instace of the widget for comparing
/// in [didUpdateWidget] then we will have to store reference here
final QuillController controller;
@override @override
QuillToolbarFontSizeButtonState createState() => QuillToolbarFontSizeButtonState createState() =>
QuillToolbarFontSizeButtonState(); QuillToolbarFontSizeButtonState();
} }
class QuillToolbarFontSizeButtonState class QuillToolbarFontSizeButtonState extends QuillToolbarBaseValueButtonState<
extends State<QuillToolbarFontSizeButton> { QuillToolbarFontSizeButton,
QuillToolbarFontSizeButtonOptions,
QuillToolbarFontSizeButtonExtraOptions,
String> {
final _menuController = MenuController(); final _menuController = MenuController();
String _currentValue = '';
QuillToolbarFontSizeButtonOptions get options {
return widget.options;
}
Map<String, String> get rawItemsMap { Map<String, String> get rawItemsMap {
final fontSizes = options.rawItemsMap ?? final fontSizes = options.rawItemsMap ??
@ -73,14 +65,12 @@ class QuillToolbarFontSizeButtonState
} }
@override @override
void didChangeDependencies() { String get currentStateValue {
super.didChangeDependencies(); final attribute =
_currentValue = _defaultDisplayText; controller.getSelectionStyle().attributes[options.attribute.key];
} return attribute == null
? _defaultDisplayText
@override : (_getKeyName(attribute.value) ?? _defaultDisplayText);
void dispose() {
super.dispose();
} }
String? _getKeyName(dynamic value) { String? _getKeyName(dynamic value) {
@ -92,38 +82,8 @@ class QuillToolbarFontSizeButtonState
return null; return null;
} }
QuillController get controller { @override
return widget.controller; String get defaultTooltip => context.loc.fontSize;
}
double get iconSize {
final baseFontSize = context.quillToolbarBaseButtonOptions?.iconSize;
final iconSize = options.iconSize;
return iconSize ?? baseFontSize ?? kDefaultIconSize;
}
double get iconButtonFactor {
final baseIconFactor =
context.quillToolbarBaseButtonOptions?.iconButtonFactor;
final iconButtonFactor = options.iconButtonFactor;
return iconButtonFactor ?? baseIconFactor ?? kDefaultIconButtonFactor;
}
VoidCallback? get afterButtonPressed {
return options.afterButtonPressed ??
context.quillToolbarBaseButtonOptions?.afterButtonPressed;
}
QuillIconTheme? get iconTheme {
return options.iconTheme ??
context.quillToolbarBaseButtonOptions?.iconTheme;
}
String get tooltip {
return options.tooltip ??
context.quillToolbarBaseButtonOptions?.tooltip ??
context.loc.fontSize;
}
void _onDropdownButtonPressed() { void _onDropdownButtonPressed() {
if (_menuController.isOpen) { if (_menuController.isOpen) {
@ -144,7 +104,7 @@ class QuillToolbarFontSizeButtonState
options, options,
QuillToolbarFontSizeButtonExtraOptions( QuillToolbarFontSizeButtonExtraOptions(
controller: controller, controller: controller,
currentValue: _currentValue, currentValue: currentValue,
defaultDisplayText: _defaultDisplayText, defaultDisplayText: _defaultDisplayText,
context: context, context: context,
onPressed: _onDropdownButtonPressed, onPressed: _onDropdownButtonPressed,
@ -162,9 +122,9 @@ class QuillToolbarFontSizeButtonState
final keyName = _getKeyName(newValue); final keyName = _getKeyName(newValue);
setState(() { setState(() {
if (keyName != context.loc.clear) { if (keyName != context.loc.clear) {
_currentValue = keyName ?? _defaultDisplayText; currentValue = keyName ?? _defaultDisplayText;
} else { } else {
_currentValue = _defaultDisplayText; currentValue = _defaultDisplayText;
} }
if (keyName != null) { if (keyName != null) {
controller.formatSelection( controller.formatSelection(
@ -176,12 +136,6 @@ class QuillToolbarFontSizeButtonState
options.onSelected?.call(newValue); options.onSelected?.call(newValue);
} }
}); });
if (fontSize.value == '0') {
controller.selectFontSize(null);
return;
}
controller.selectFontSize(fontSize);
}, },
child: Text( child: Text(
fontSize.key.toString(), fontSize.key.toString(),
@ -224,9 +178,7 @@ class QuillToolbarFontSizeButtonState
enabled: hasFinalWidth, enabled: hasFinalWidth,
wrapper: (child) => Expanded(child: child), wrapper: (child) => Expanded(child: child),
child: Text( child: Text(
getLabel(widget.controller.selectedFontSize?.key) ?? getLabel(currentValue) ?? '',
getLabel(_currentValue) ??
'',
overflow: options.labelOverflow, overflow: options.labelOverflow,
style: options.style ?? style: options.style ??
TextStyle( TextStyle(

@ -1,48 +1,36 @@
import 'package:flutter/material.dart'; import 'package:flutter/material.dart';
import '../../../extensions/quill_configurations_ext.dart';
import '../../../l10n/extensions/localizations.dart'; import '../../../l10n/extensions/localizations.dart';
import '../../../models/documents/attribute.dart'; import '../../../models/documents/attribute.dart';
import '../../../models/documents/style.dart'; import '../../../models/documents/style.dart';
import '../../../models/themes/quill_icon_theme.dart';
import '../../../utils/widgets.dart'; import '../../../utils/widgets.dart';
import '../../quill/quill_controller.dart'; import '../base_button/base_value_button.dart';
import '../base_toolbar.dart'; import '../base_toolbar.dart';
class QuillToolbarToggleCheckListButton extends StatefulWidget { class QuillToolbarToggleCheckListButton extends QuillToolbarBaseValueButton<
QuillToolbarToggleCheckListButtonOptions,
QuillToolbarToggleCheckListButtonExtraOptions> {
const QuillToolbarToggleCheckListButton({ const QuillToolbarToggleCheckListButton({
required this.controller, required super.controller,
this.options = const QuillToolbarToggleCheckListButtonOptions(), super.options = const QuillToolbarToggleCheckListButtonOptions(),
super.key, super.key,
}); });
final QuillToolbarToggleCheckListButtonOptions options;
final QuillController controller;
@override @override
QuillToolbarToggleCheckListButtonState createState() => QuillToolbarToggleCheckListButtonState createState() =>
QuillToolbarToggleCheckListButtonState(); QuillToolbarToggleCheckListButtonState();
} }
class QuillToolbarToggleCheckListButtonState class QuillToolbarToggleCheckListButtonState
extends State<QuillToolbarToggleCheckListButton> { extends QuillToolbarBaseValueButtonState<
bool? _isToggled; QuillToolbarToggleCheckListButton,
QuillToolbarToggleCheckListButtonOptions,
QuillToolbarToggleCheckListButtonExtraOptions,
bool> {
Style get _selectionStyle => controller.getSelectionStyle(); Style get _selectionStyle => controller.getSelectionStyle();
void _didChangeEditingValue() {
setState(() {
_isToggled = _getIsToggled(controller.getSelectionStyle().attributes);
});
}
@override @override
void initState() { bool get currentStateValue => _getIsToggled(_selectionStyle.attributes);
super.initState();
_isToggled = _getIsToggled(_selectionStyle.attributes);
controller.addListener(_didChangeEditingValue);
}
bool _getIsToggled(Map<String, Attribute> attrs) { bool _getIsToggled(Map<String, Attribute> attrs) {
var attribute = controller.toolbarButtonToggler[Attribute.list.key]; var attribute = controller.toolbarButtonToggler[Attribute.list.key];
@ -62,53 +50,7 @@ class QuillToolbarToggleCheckListButtonState
} }
@override @override
void didUpdateWidget(covariant QuillToolbarToggleCheckListButton oldWidget) { String get defaultTooltip => context.loc.checkedList;
super.didUpdateWidget(oldWidget);
if (oldWidget.controller != controller) {
oldWidget.controller.removeListener(_didChangeEditingValue);
controller.addListener(_didChangeEditingValue);
_isToggled = _getIsToggled(_selectionStyle.attributes);
}
}
@override
void dispose() {
controller.removeListener(_didChangeEditingValue);
super.dispose();
}
QuillToolbarToggleCheckListButtonOptions get options {
return widget.options;
}
QuillController get controller {
return widget.controller;
}
double get iconSize {
final baseFontSize = baseButtonExtraOptions?.iconSize;
final iconSize = options.iconSize;
return iconSize ?? baseFontSize ?? kDefaultIconSize;
}
double get iconButtonFactor {
final baseIconFactor = baseButtonExtraOptions?.iconButtonFactor;
final iconButtonFactor = options.iconButtonFactor;
return iconButtonFactor ?? baseIconFactor ?? kDefaultIconButtonFactor;
}
VoidCallback? get afterButtonPressed {
return options.afterButtonPressed ??
baseButtonExtraOptions?.afterButtonPressed;
}
QuillIconTheme? get iconTheme {
return options.iconTheme ?? baseButtonExtraOptions?.iconTheme;
}
QuillToolbarBaseButtonOptions? get baseButtonExtraOptions {
return context.quillToolbarBaseButtonOptions;
}
IconData get iconData { IconData get iconData {
return options.iconData ?? return options.iconData ??
@ -116,12 +58,6 @@ class QuillToolbarToggleCheckListButtonState
Icons.check_box; Icons.check_box;
} }
String get tooltip {
return options.tooltip ??
baseButtonExtraOptions?.tooltip ??
context.loc.checkedList;
}
@override @override
Widget build(BuildContext context) { Widget build(BuildContext context) {
final childBuilder = final childBuilder =
@ -136,7 +72,7 @@ class QuillToolbarToggleCheckListButtonState
_toggleAttribute(); _toggleAttribute();
afterButtonPressed?.call(); afterButtonPressed?.call();
}, },
isToggled: _isToggled ?? false, isToggled: currentValue,
), ),
); );
} }
@ -146,7 +82,7 @@ class QuillToolbarToggleCheckListButtonState
context, context,
Attribute.unchecked, Attribute.unchecked,
iconData, iconData,
_isToggled, currentValue,
_toggleAttribute, _toggleAttribute,
afterButtonPressed, afterButtonPressed,
iconSize, iconSize,
@ -160,7 +96,7 @@ class QuillToolbarToggleCheckListButtonState
controller controller
..skipRequestKeyboard = !options.isShouldRequestKeyboard ..skipRequestKeyboard = !options.isShouldRequestKeyboard
..formatSelection( ..formatSelection(
_isToggled! currentValue
? Attribute.clone(Attribute.unchecked, null) ? Attribute.clone(Attribute.unchecked, null)
: Attribute.unchecked, : Attribute.unchecked,
); );

@ -6,7 +6,7 @@ import '../../../models/documents/attribute.dart';
import '../../../models/documents/style.dart'; import '../../../models/documents/style.dart';
import '../../../models/themes/quill_icon_theme.dart'; import '../../../models/themes/quill_icon_theme.dart';
import '../../../utils/widgets.dart'; import '../../../utils/widgets.dart';
import '../../quill/quill_controller.dart'; import '../base_button/base_value_button.dart';
import '../base_toolbar.dart'; import '../base_toolbar.dart';
typedef ToggleStyleButtonBuilder = Widget Function( typedef ToggleStyleButtonBuilder = Widget Function(
@ -20,68 +20,33 @@ typedef ToggleStyleButtonBuilder = Widget Function(
QuillIconTheme? iconTheme, QuillIconTheme? iconTheme,
]); ]);
class QuillToolbarToggleStyleButton extends StatefulWidget { class QuillToolbarToggleStyleButton extends QuillToolbarBaseValueButton<
QuillToolbarToggleStyleButtonOptions,
QuillToolbarToggleStyleButtonExtraOptions> {
const QuillToolbarToggleStyleButton({ const QuillToolbarToggleStyleButton({
required this.controller, required super.controller,
required this.attribute, required this.attribute,
this.options = const QuillToolbarToggleStyleButtonOptions(), super.options = const QuillToolbarToggleStyleButtonOptions(),
super.key, super.key,
}); });
final Attribute attribute; final Attribute attribute;
final QuillToolbarToggleStyleButtonOptions options;
final QuillController controller;
@override @override
QuillToolbarToggleStyleButtonState createState() => QuillToolbarToggleStyleButtonState createState() =>
QuillToolbarToggleStyleButtonState(); QuillToolbarToggleStyleButtonState();
} }
class QuillToolbarToggleStyleButtonState class QuillToolbarToggleStyleButtonState
extends State<QuillToolbarToggleStyleButton> { extends QuillToolbarBaseValueButtonState<
bool? _isToggled; QuillToolbarToggleStyleButton,
QuillToolbarToggleStyleButtonOptions,
QuillToolbarToggleStyleButtonExtraOptions,
bool> {
Style get _selectionStyle => controller.getSelectionStyle(); Style get _selectionStyle => controller.getSelectionStyle();
QuillToolbarToggleStyleButtonOptions get options {
return widget.options;
}
@override @override
void initState() { bool get currentStateValue => _getIsToggled(_selectionStyle.attributes);
super.initState();
_isToggled = _getIsToggled(_selectionStyle.attributes);
controller.addListener(_didChangeEditingValue);
}
QuillController get controller {
return widget.controller;
}
double get iconSize {
final baseFontSize = context.quillToolbarBaseButtonOptions?.iconSize;
final iconSize = options.iconSize;
return iconSize ?? baseFontSize ?? kDefaultIconSize;
}
double get iconButtonFactor {
final baseIconFactor =
context.quillToolbarBaseButtonOptions?.iconButtonFactor;
final iconButtonFactor = options.iconButtonFactor;
return iconButtonFactor ?? baseIconFactor ?? kDefaultIconButtonFactor;
}
VoidCallback? get afterButtonPressed {
return options.afterButtonPressed ??
context.quillToolbarBaseButtonOptions?.afterButtonPressed;
}
QuillIconTheme? get iconTheme {
return options.iconTheme ??
context.quillToolbarBaseButtonOptions?.iconTheme;
}
(String, IconData) get _defaultTooltipAndIconData { (String, IconData) get _defaultTooltipAndIconData {
switch (widget.attribute.key) { switch (widget.attribute.key) {
@ -133,11 +98,8 @@ class QuillToolbarToggleStyleButtonState
} }
} }
String? get tooltip { @override
return options.tooltip ?? String get defaultTooltip => _defaultTooltipAndIconData.$1;
context.quillToolbarBaseButtonOptions?.tooltip ??
_defaultTooltipAndIconData.$1;
}
IconData get iconData { IconData get iconData {
return options.iconData ?? return options.iconData ??
@ -161,7 +123,7 @@ class QuillToolbarToggleStyleButtonState
context: context, context: context,
controller: controller, controller: controller,
onPressed: _onPressed, onPressed: _onPressed,
isToggled: _isToggled ?? false, isToggled: currentValue,
), ),
); );
} }
@ -171,7 +133,7 @@ class QuillToolbarToggleStyleButtonState
context, context,
widget.attribute, widget.attribute,
iconData, iconData,
_isToggled, currentValue,
_toggleAttribute, _toggleAttribute,
afterButtonPressed, afterButtonPressed,
iconSize, iconSize,
@ -181,26 +143,6 @@ class QuillToolbarToggleStyleButtonState
); );
} }
@override
void didUpdateWidget(covariant QuillToolbarToggleStyleButton oldWidget) {
super.didUpdateWidget(oldWidget);
if (oldWidget.controller != controller) {
oldWidget.controller.removeListener(_didChangeEditingValue);
controller.addListener(_didChangeEditingValue);
_isToggled = _getIsToggled(_selectionStyle.attributes);
}
}
@override
void dispose() {
controller.removeListener(_didChangeEditingValue);
super.dispose();
}
void _didChangeEditingValue() {
setState(() => _isToggled = _getIsToggled(_selectionStyle.attributes));
}
bool _getIsToggled(Map<String, Attribute> attrs) { bool _getIsToggled(Map<String, Attribute> attrs) {
if (widget.attribute.key == Attribute.list.key || if (widget.attribute.key == Attribute.list.key ||
widget.attribute.key == Attribute.script.key || widget.attribute.key == Attribute.script.key ||
@ -216,12 +158,12 @@ class QuillToolbarToggleStyleButtonState
void _toggleAttribute() { void _toggleAttribute() {
controller controller
..skipRequestKeyboard = !widget.attribute.isInline
..formatSelection( ..formatSelection(
(_isToggled ?? false) currentValue
? Attribute.clone(widget.attribute, null) ? Attribute.clone(widget.attribute, null)
: widget.attribute, : widget.attribute,
) );
..selectStyle(widget.attribute, _isToggled ?? false);
} }
} }

Loading…
Cancel
Save