Feat: Line height support (#1972)
* Support for Line height * fix(translations): line-height translation empty * chore: dart fixes * fix(test): blockSelectionStyle doesn't find line-height attribute * fix: increased _expectedTranslationKeysLength value for new line height translation * fix: restore h1 configurations on DefaultStyles * fix: better organization for line height attribute * fix: select_line_height_dropdown_button doesn't find LineHeightAttribute constants * chore: dart fixes * fix(test): controller_test doesn't find LineHeightAttribute * fix: changed MenuAnchor to showMenu method since this MenuAnchor appears below keyboard * added docs comments * set showLineHeightButton to false * added warnings about line height attribute could cause conflicts with original Quill API * chore: dart fixes --------- Co-authored-by: CatHood0 <santiagowmar@gmail.com>pull/1976/head^2 v9.5.4
parent
151e629ee3
commit
6598f3d74d
90 changed files with 784 additions and 199 deletions
@ -0,0 +1,78 @@ |
|||||||
|
import 'package:flutter/widgets.dart' |
||||||
|
show IconData, TextStyle, ValueChanged, VoidCallback; |
||||||
|
|
||||||
|
import '../../../../widgets/toolbar/base_toolbar.dart'; |
||||||
|
import '../../../documents/attribute.dart'; |
||||||
|
import '../../../themes/quill_icon_theme.dart'; |
||||||
|
|
||||||
|
class QuillToolbarSelectLineHeightStyleDropdownButtonExtraOptions |
||||||
|
extends QuillToolbarBaseButtonExtraOptions { |
||||||
|
const QuillToolbarSelectLineHeightStyleDropdownButtonExtraOptions({ |
||||||
|
required super.controller, |
||||||
|
required super.context, |
||||||
|
required super.onPressed, |
||||||
|
required this.currentValue, |
||||||
|
}); |
||||||
|
final Attribute currentValue; |
||||||
|
} |
||||||
|
|
||||||
|
class QuillToolbarSelectLineHeightStyleDropdownButtonOptions |
||||||
|
extends QuillToolbarBaseButtonOptions< |
||||||
|
QuillToolbarSelectLineHeightStyleDropdownButtonOptions, |
||||||
|
QuillToolbarSelectLineHeightStyleDropdownButtonExtraOptions> { |
||||||
|
const QuillToolbarSelectLineHeightStyleDropdownButtonOptions({ |
||||||
|
super.afterButtonPressed, |
||||||
|
super.childBuilder, |
||||||
|
super.iconTheme, |
||||||
|
super.tooltip, |
||||||
|
super.iconSize, |
||||||
|
super.iconButtonFactor, |
||||||
|
this.textStyle, |
||||||
|
super.iconData, |
||||||
|
this.attributes, |
||||||
|
this.defaultDisplayText, |
||||||
|
this.width, |
||||||
|
}); |
||||||
|
|
||||||
|
final TextStyle? textStyle; |
||||||
|
|
||||||
|
/// Line-height attributes, defaults to: |
||||||
|
/// ```dart |
||||||
|
/// [ |
||||||
|
/// Attribute.lineHeightNormal, |
||||||
|
/// Attribute.lineHeightTight, |
||||||
|
/// Attribute.lineHeightOneAndHalf, |
||||||
|
/// Attribute.lineHeightDouble, |
||||||
|
/// ] |
||||||
|
/// ``` |
||||||
|
final List<Attribute<double?>>? attributes; |
||||||
|
final double? width; |
||||||
|
|
||||||
|
final String? defaultDisplayText; |
||||||
|
|
||||||
|
QuillToolbarSelectLineHeightStyleDropdownButtonOptions copyWith({ |
||||||
|
ValueChanged<String>? onSelected, |
||||||
|
List<Attribute<double>>? attributes, |
||||||
|
TextStyle? style, |
||||||
|
double? iconSize, |
||||||
|
double? iconButtonFactor, |
||||||
|
IconData? iconData, |
||||||
|
VoidCallback? afterButtonPressed, |
||||||
|
String? tooltip, |
||||||
|
QuillIconTheme? iconTheme, |
||||||
|
String? defaultDisplayText, |
||||||
|
double? width, |
||||||
|
}) { |
||||||
|
return QuillToolbarSelectLineHeightStyleDropdownButtonOptions( |
||||||
|
attributes: attributes ?? this.attributes, |
||||||
|
iconData: iconData ?? this.iconData, |
||||||
|
afterButtonPressed: afterButtonPressed ?? this.afterButtonPressed, |
||||||
|
tooltip: tooltip ?? this.tooltip, |
||||||
|
iconTheme: iconTheme ?? this.iconTheme, |
||||||
|
iconSize: iconSize ?? this.iconSize, |
||||||
|
iconButtonFactor: iconButtonFactor ?? this.iconButtonFactor, |
||||||
|
defaultDisplayText: defaultDisplayText ?? this.defaultDisplayText, |
||||||
|
width: width ?? this.width, |
||||||
|
); |
||||||
|
} |
||||||
|
} |
@ -0,0 +1,234 @@ |
|||||||
|
import 'package:flutter/material.dart'; |
||||||
|
|
||||||
|
import '../../../../translations.dart'; |
||||||
|
import '../../../extensions/quill_configurations_ext.dart'; |
||||||
|
import '../../../models/config/toolbar/buttons/select_line_height_style_dropdown_button_configurations.dart'; |
||||||
|
import '../../../models/documents/attribute.dart'; |
||||||
|
import '../../../models/themes/quill_icon_theme.dart'; |
||||||
|
import '../base_button/base_value_button.dart'; |
||||||
|
import '../base_toolbar.dart'; |
||||||
|
|
||||||
|
typedef QuillToolbarSelectLineHeightStyleDropdownBaseButton |
||||||
|
= QuillToolbarBaseButton< |
||||||
|
QuillToolbarSelectLineHeightStyleDropdownButtonOptions, |
||||||
|
QuillToolbarSelectLineHeightStyleDropdownButtonExtraOptions>; |
||||||
|
|
||||||
|
typedef QuillToolbarSelectLineHeightStyleDropdownBaseButtonsState< |
||||||
|
W extends QuillToolbarSelectLineHeightStyleDropdownButton> |
||||||
|
= QuillToolbarCommonButtonState< |
||||||
|
W, |
||||||
|
QuillToolbarSelectLineHeightStyleDropdownButtonOptions, |
||||||
|
QuillToolbarSelectLineHeightStyleDropdownButtonExtraOptions>; |
||||||
|
|
||||||
|
class QuillToolbarSelectLineHeightStyleDropdownButton |
||||||
|
extends QuillToolbarSelectLineHeightStyleDropdownBaseButton { |
||||||
|
const QuillToolbarSelectLineHeightStyleDropdownButton({ |
||||||
|
required super.controller, |
||||||
|
super.options = |
||||||
|
const QuillToolbarSelectLineHeightStyleDropdownButtonOptions(), |
||||||
|
super.key, |
||||||
|
}); |
||||||
|
|
||||||
|
@override |
||||||
|
QuillToolbarSelectLineHeightStyleDropdownBaseButtonsState createState() => |
||||||
|
_QuillToolbarSelectLineHeightStyleDropdownButtonState(); |
||||||
|
} |
||||||
|
|
||||||
|
class _QuillToolbarSelectLineHeightStyleDropdownButtonState |
||||||
|
extends QuillToolbarSelectLineHeightStyleDropdownBaseButtonsState { |
||||||
|
@override |
||||||
|
String get defaultTooltip => context.loc.lineheight; |
||||||
|
|
||||||
|
@override |
||||||
|
IconData get defaultIconData => Icons.question_mark_outlined; |
||||||
|
|
||||||
|
Attribute<dynamic> _selectedItem = Attribute.lineHeight; |
||||||
|
|
||||||
|
@override |
||||||
|
void initState() { |
||||||
|
super.initState(); |
||||||
|
widget.controller.addListener(_didChangeEditingValue); |
||||||
|
} |
||||||
|
|
||||||
|
@override |
||||||
|
void dispose() { |
||||||
|
widget.controller.removeListener(_didChangeEditingValue); |
||||||
|
super.dispose(); |
||||||
|
} |
||||||
|
|
||||||
|
@override |
||||||
|
void didUpdateWidget( |
||||||
|
covariant QuillToolbarSelectLineHeightStyleDropdownButton oldWidget) { |
||||||
|
super.didUpdateWidget(oldWidget); |
||||||
|
if (oldWidget.controller == widget.controller) { |
||||||
|
return; |
||||||
|
} |
||||||
|
widget.controller |
||||||
|
..removeListener(_didChangeEditingValue) |
||||||
|
..addListener(_didChangeEditingValue); |
||||||
|
} |
||||||
|
|
||||||
|
void _didChangeEditingValue() { |
||||||
|
final newSelectedItem = _getLineHeightValue(); |
||||||
|
if (newSelectedItem == _selectedItem) { |
||||||
|
return; |
||||||
|
} |
||||||
|
setState(() { |
||||||
|
_selectedItem = newSelectedItem; |
||||||
|
}); |
||||||
|
} |
||||||
|
|
||||||
|
Attribute<dynamic> _getLineHeightValue() { |
||||||
|
final attr = |
||||||
|
widget.controller.toolbarButtonToggler[Attribute.lineHeight.key]; |
||||||
|
if (attr != null) { |
||||||
|
widget.controller.toolbarButtonToggler.remove(Attribute.lineHeight.key); |
||||||
|
return attr; |
||||||
|
} |
||||||
|
return widget.controller |
||||||
|
.getSelectionStyle() |
||||||
|
.attributes[Attribute.lineHeight.key] ?? |
||||||
|
Attribute.lineHeight; |
||||||
|
} |
||||||
|
|
||||||
|
String _label(Attribute<dynamic> attribute) { |
||||||
|
var label = LineHeightAttribute.lineHeightNormal.value.toString(); |
||||||
|
if (attribute.value != null) { |
||||||
|
label = attribute.value.toString(); |
||||||
|
} |
||||||
|
return label; |
||||||
|
} |
||||||
|
|
||||||
|
List<Attribute<dynamic>> get lineHeightAttributes { |
||||||
|
return widget.options.attributes ?? |
||||||
|
[ |
||||||
|
LineHeightAttribute.lineHeightNormal, |
||||||
|
LineHeightAttribute.lineHeightTight, |
||||||
|
LineHeightAttribute.lineHeightOneAndHalf, |
||||||
|
LineHeightAttribute.lineHeightDouble, |
||||||
|
]; |
||||||
|
} |
||||||
|
|
||||||
|
void _onPressed(Attribute<dynamic> e) { |
||||||
|
setState(() => _selectedItem = e); |
||||||
|
widget.controller.formatSelection(_selectedItem); |
||||||
|
} |
||||||
|
|
||||||
|
@override |
||||||
|
Widget build(BuildContext context) { |
||||||
|
final baseButtonConfigurations = context.quillToolbarBaseButtonOptions; |
||||||
|
final childBuilder = |
||||||
|
widget.options.childBuilder ?? baseButtonConfigurations?.childBuilder; |
||||||
|
if (childBuilder != null) { |
||||||
|
return childBuilder( |
||||||
|
widget.options, |
||||||
|
QuillToolbarSelectLineHeightStyleDropdownButtonExtraOptions( |
||||||
|
currentValue: _selectedItem, |
||||||
|
context: context, |
||||||
|
controller: widget.controller, |
||||||
|
onPressed: () { |
||||||
|
throw UnimplementedError('Not implemented yet.'); |
||||||
|
}, |
||||||
|
), |
||||||
|
); |
||||||
|
} |
||||||
|
|
||||||
|
return Builder( |
||||||
|
builder: (context) { |
||||||
|
final isMaterial3 = Theme.of(context).useMaterial3; |
||||||
|
final child = Row( |
||||||
|
mainAxisSize: MainAxisSize.min, |
||||||
|
mainAxisAlignment: MainAxisAlignment.spaceBetween, |
||||||
|
children: [ |
||||||
|
Text( |
||||||
|
_label(_selectedItem), |
||||||
|
style: widget.options.textStyle ?? |
||||||
|
TextStyle( |
||||||
|
fontSize: iconSize / 1.15, |
||||||
|
), |
||||||
|
), |
||||||
|
Icon( |
||||||
|
Icons.arrow_drop_down, |
||||||
|
size: iconSize * iconButtonFactor, |
||||||
|
), |
||||||
|
], |
||||||
|
); |
||||||
|
if (!isMaterial3) { |
||||||
|
return RawMaterialButton( |
||||||
|
onPressed: () => _onDropdownButtonPressed(context), |
||||||
|
child: child, |
||||||
|
); |
||||||
|
} |
||||||
|
return _QuillToolbarLineHeightIcon( |
||||||
|
iconTheme: iconTheme, |
||||||
|
tooltip: tooltip, |
||||||
|
onPressed: _onDropdownButtonPressed, |
||||||
|
child: child, |
||||||
|
); |
||||||
|
}, |
||||||
|
); |
||||||
|
} |
||||||
|
|
||||||
|
Future<void> _onDropdownButtonPressed(BuildContext context) async { |
||||||
|
final position = _renderPosition(context); |
||||||
|
await showMenu<Attribute<dynamic>>( |
||||||
|
context: context, |
||||||
|
position: position, |
||||||
|
items: lineHeightAttributes |
||||||
|
.map( |
||||||
|
(e) => PopupMenuItem( |
||||||
|
value: e, |
||||||
|
child: Text(_label(e)), |
||||||
|
), |
||||||
|
) |
||||||
|
.toList()) |
||||||
|
.then<void>( |
||||||
|
(value) { |
||||||
|
if (value != null) { |
||||||
|
_onPressed(value); |
||||||
|
} |
||||||
|
}, |
||||||
|
); |
||||||
|
} |
||||||
|
|
||||||
|
RelativeRect _renderPosition(BuildContext context) { |
||||||
|
final size = MediaQuery.sizeOf(context); |
||||||
|
final overlay = Overlay.of(context).context.findRenderObject() as RenderBox; |
||||||
|
final button = context.findRenderObject() as RenderBox; |
||||||
|
final position = RelativeRect.fromRect( |
||||||
|
Rect.fromPoints( |
||||||
|
button.localToGlobal(const Offset(0, -65), ancestor: overlay), |
||||||
|
button.localToGlobal( |
||||||
|
button.size.bottomRight(Offset.zero) + const Offset(-50, 0), |
||||||
|
ancestor: overlay), |
||||||
|
), |
||||||
|
Offset.zero & size * 0.40, |
||||||
|
); |
||||||
|
return position; |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
class _QuillToolbarLineHeightIcon extends StatelessWidget { |
||||||
|
const _QuillToolbarLineHeightIcon({ |
||||||
|
required this.tooltip, |
||||||
|
required this.iconTheme, |
||||||
|
required this.onPressed, |
||||||
|
required this.child, |
||||||
|
}); |
||||||
|
|
||||||
|
final Row child; |
||||||
|
final void Function(BuildContext) onPressed; |
||||||
|
final QuillIconTheme? iconTheme; |
||||||
|
final String tooltip; |
||||||
|
|
||||||
|
@override |
||||||
|
Widget build(BuildContext context) { |
||||||
|
return QuillToolbarIconButton( |
||||||
|
icon: child, |
||||||
|
isSelected: false, |
||||||
|
iconTheme: iconTheme, |
||||||
|
tooltip: tooltip, |
||||||
|
onPressed: () => onPressed(context), |
||||||
|
); |
||||||
|
} |
||||||
|
} |
Loading…
Reference in new issue