From db143c95561bb2299db302e79fccf9b4e1d0c48c Mon Sep 17 00:00:00 2001 From: Ahmed Hnewa <73608287+freshtechtips@users.noreply.github.com> Date: Thu, 19 Oct 2023 08:49:36 +0300 Subject: [PATCH] Bug fixes + Provide a way to handle the image errors and use custom image provider (#1428) --- CHANGELOG.md | 4 + FETCH_HEAD | 0 example/lib/pages/home_page.dart | 4 +- example/lib/universal_ui/universal_ui.dart | 8 +- example/lib/widgets/responsive_widget.dart | 8 +- flutter_quill_extensions/CHANGELOG.md | 7 + .../lib/embeds/builders.dart | 365 ++++++++++++------ .../lib/embeds/embed_types.dart | 10 +- .../lib/embeds/toolbar/camera_button.dart | 109 ++++-- .../lib/embeds/toolbar/image_button.dart | 51 ++- .../lib/embeds/toolbar/image_video_utils.dart | 28 +- .../lib/embeds/toolbar/media_button.dart | 22 +- .../lib/embeds/utils.dart | 3 +- .../lib/embeds/widgets/image.dart | 66 +++- .../lib/embeds/widgets/image_resizer.dart | 11 +- .../lib/flutter_quill_extensions.dart | 30 +- flutter_quill_extensions/pubspec.yaml | 2 +- lib/src/models/documents/attribute.dart | 8 + lib/src/models/rules/rule.dart | 14 +- lib/src/translations/toolbar.i18n.dart | 7 +- lib/src/utils/platform.dart | 6 +- lib/src/utils/string.dart | 30 +- lib/src/widgets/editor.dart | 2 +- lib/src/widgets/raw_editor.dart | 2 +- lib/src/widgets/text_block.dart | 2 +- .../widgets/toolbar/link_style_button2.dart | 5 +- pubspec.yaml | 2 +- 27 files changed, 568 insertions(+), 238 deletions(-) create mode 100644 FETCH_HEAD diff --git a/CHANGELOG.md b/CHANGELOG.md index 934e3513..99fef3b1 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -1,3 +1,7 @@ +# [7.4.14] +- Custom style attrbuites for platforms other than mobile (alignment, margin, width, height) +- Improve performance by reducing the number of widgets rebuilt by listening to media query for only the needed things, for example instead of using `MediaQuery.of(context).size`, now we are using `MediaQuery.sizeOf(context)` + # [7.4.13] - Fixed tab editing when in readOnly mode. diff --git a/FETCH_HEAD b/FETCH_HEAD new file mode 100644 index 00000000..e69de29b diff --git a/example/lib/pages/home_page.dart b/example/lib/pages/home_page.dart index 1f764cbc..2367719d 100644 --- a/example/lib/pages/home_page.dart +++ b/example/lib/pages/home_page.dart @@ -104,7 +104,7 @@ class _HomePageState extends State { ), drawer: Container( constraints: - BoxConstraints(maxWidth: MediaQuery.of(context).size.width * 0.7), + BoxConstraints(maxWidth: MediaQuery.sizeOf(context).width * 0.7), color: Colors.grey.shade800, child: _buildMenuBar(context), ), @@ -408,7 +408,7 @@ class _HomePageState extends State { ); Widget _buildMenuBar(BuildContext context) { - final size = MediaQuery.of(context).size; + final size = MediaQuery.sizeOf(context); const itemStyle = TextStyle( color: Colors.white, fontSize: 18, diff --git a/example/lib/universal_ui/universal_ui.dart b/example/lib/universal_ui/universal_ui.dart index 91344afb..421725e2 100644 --- a/example/lib/universal_ui/universal_ui.dart +++ b/example/lib/universal_ui/universal_ui.dart @@ -45,7 +45,7 @@ class ImageEmbedBuilderWeb extends EmbedBuilder { // TODO: handle imageUrl of base64 return const SizedBox(); } - final size = MediaQuery.of(context).size; + final size = MediaQuery.sizeOf(context); UniversalUI().platformViewRegistry.registerViewFactory(imageUrl, (viewId) { return html.ImageElement() ..src = imageUrl @@ -61,7 +61,7 @@ class ImageEmbedBuilderWeb extends EmbedBuilder { : size.width * 0.2, ), child: SizedBox( - height: MediaQuery.of(context).size.height * 0.45, + height: MediaQuery.sizeOf(context).height * 0.45, child: HtmlElementView( viewType: imageUrl, ), @@ -94,8 +94,8 @@ class VideoEmbedBuilderWeb extends EmbedBuilder { UniversalUI().platformViewRegistry.registerViewFactory( videoUrl, (id) => html.IFrameElement() - ..width = MediaQuery.of(context).size.width.toString() - ..height = MediaQuery.of(context).size.height.toString() + ..width = MediaQuery.sizeOf(context).width.toString() + ..height = MediaQuery.sizeOf(context).height.toString() ..src = videoUrl ..style.border = 'none'); diff --git a/example/lib/widgets/responsive_widget.dart b/example/lib/widgets/responsive_widget.dart index 3829565c..f9de4027 100644 --- a/example/lib/widgets/responsive_widget.dart +++ b/example/lib/widgets/responsive_widget.dart @@ -13,16 +13,16 @@ class ResponsiveWidget extends StatelessWidget { final Widget? smallScreen; static bool isSmallScreen(BuildContext context) { - return MediaQuery.of(context).size.width < 800; + return MediaQuery.sizeOf(context).width < 800; } static bool isLargeScreen(BuildContext context) { - return MediaQuery.of(context).size.width > 1200; + return MediaQuery.sizeOf(context).width > 1200; } static bool isMediumScreen(BuildContext context) { - return MediaQuery.of(context).size.width >= 800 && - MediaQuery.of(context).size.width <= 1200; + return MediaQuery.sizeOf(context).width >= 800 && + MediaQuery.sizeOf(context).width <= 1200; } @override diff --git a/flutter_quill_extensions/CHANGELOG.md b/flutter_quill_extensions/CHANGELOG.md index f8a309ec..a6155cb5 100644 --- a/flutter_quill_extensions/CHANGELOG.md +++ b/flutter_quill_extensions/CHANGELOG.md @@ -1,8 +1,15 @@ ## 0.5.1 +- Provide a way to use custom image provider for the image widgets +- Provide a way to handle different errors in image widgets +- Two bug fixes related to pick the image and capture it using the camera +- Add support for image resizing on desktop platforms when forced using the mobile context menu +- Improve performance by reducing the number of widgets rebuilt by listening to media query for only the needed things, for example instead of using `MediaQuery.of(context).size`, now we are using `MediaQuery.sizeOf(context)` - Fix warrning "The platformViewRegistry getter is deprecated and will be removed in a future release. Please import it from dart:ui_web instead." - Add QuillImageUtilities class - Small improvemenets - Allow to use the mobile context menu on desktop by force using it +- Add the resizing option to the forced mobile context menu +- Add new custom style attrbuite for desktop and other platforms ## 0.5.0 - Migrated from `gallery_saver` to `gal` for saving images diff --git a/flutter_quill_extensions/lib/embeds/builders.dart b/flutter_quill_extensions/lib/embeds/builders.dart index e9f8d753..5d9e0236 100644 --- a/flutter_quill_extensions/lib/embeds/builders.dart +++ b/flutter_quill_extensions/lib/embeds/builders.dart @@ -21,13 +21,17 @@ import 'widgets/youtube_video_app.dart'; class ImageEmbedBuilder extends EmbedBuilder { ImageEmbedBuilder({ - this.onImageRemovedCallback, - this.shouldRemoveImageCallback, + required this.imageProviderBuilder, + required this.imageErrorWidgetBuilder, + required this.onImageRemovedCallback, + required this.shouldRemoveImageCallback, this.forceUseMobileOptionMenu = false, }); final ImageEmbedBuilderOnRemovedCallback? onImageRemovedCallback; final ImageEmbedBuilderWillRemoveCallback? shouldRemoveImageCallback; final bool forceUseMobileOptionMenu; + final ImageEmbedBuilderProviderBuilder? imageProviderBuilder; + final ImageEmbedBuilderErrorWidgetBuilder? imageErrorWidgetBuilder; @override String get key => BlockEmbed.imageType; @@ -50,33 +54,73 @@ class ImageEmbedBuilder extends EmbedBuilder { final imageUrl = standardizeImageUrl(node.value.data); OptionalSize? imageSize; final style = node.style.attributes['style']; - if (base.isMobile() && style != null) { - final attrs = base.parseKeyValuePairs(style.value.toString(), { - Attribute.mobileWidth, - Attribute.mobileHeight, - Attribute.mobileMargin, - Attribute.mobileAlignment - }); + + // TODO: Please use the one from [Attribute.margin] + const marginKey = 'margin'; + // TODO: Please use the one from [Attribute.alignment] + const alignmentKey = 'alignment'; + if (style != null) { + final attrs = base.isMobile() + ? base.parseKeyValuePairs(style.value.toString(), { + Attribute.mobileWidth, + Attribute.mobileHeight, + Attribute.mobileMargin, + Attribute.mobileAlignment, + }) + : base.parseKeyValuePairs(style.value.toString(), { + Attribute.width.key, + Attribute.height.key, + marginKey, + alignmentKey, + }); if (attrs.isNotEmpty) { + final width = double.tryParse( + (base.isMobile() + ? attrs[Attribute.mobileWidth] + : attrs[Attribute.width.key]) ?? + '', + ); + final height = double.tryParse( + (base.isMobile() + ? attrs[Attribute.mobileHeight] + : attrs[Attribute.height.key]) ?? + '', + ); + final alignment = base.getAlignment(base.isMobile() + ? attrs[Attribute.mobileAlignment] + : attrs[alignmentKey]); + final margin = (base.isMobile() + ? double.tryParse(Attribute.mobileMargin) + : double.tryParse(marginKey)) ?? + 0.0; + assert( - attrs[Attribute.mobileWidth] != null && - attrs[Attribute.mobileHeight] != null, - 'mobileWidth and mobileHeight must be specified'); - final w = double.parse(attrs[Attribute.mobileWidth]!); - final h = double.parse(attrs[Attribute.mobileHeight]!); - imageSize = OptionalSize(w, h); - final m = attrs[Attribute.mobileMargin] == null - ? 0.0 - : double.parse(attrs[Attribute.mobileMargin]!); - final a = base.getAlignment(attrs[Attribute.mobileAlignment]); + width != null && height != null, + base.isMobile() + ? 'mobileWidth and mobileHeight must be specified' + : 'width and height must be specified', + ); + imageSize = OptionalSize(width, height); image = Padding( - padding: EdgeInsets.all(m), - child: imageByUrl(imageUrl, width: w, height: h, alignment: a)); + padding: EdgeInsets.all(margin), + child: getQuillImageByUrl( + imageUrl, + width: width, + height: height, + alignment: alignment, + imageProviderBuilder: imageProviderBuilder, + imageErrorWidgetBuilder: imageErrorWidgetBuilder, + ), + ); } } if (imageSize == null) { - image = imageByUrl(imageUrl); + image = getQuillImageByUrl( + imageUrl, + imageProviderBuilder: imageProviderBuilder, + imageErrorWidgetBuilder: imageErrorWidgetBuilder, + ); imageSize = OptionalSize((image as Image).width, image.height); } @@ -86,34 +130,6 @@ class ImageEmbedBuilder extends EmbedBuilder { showDialog( context: context, builder: (context) { - final resizeOption = _SimpleDialogItem( - icon: Icons.settings_outlined, - color: Colors.lightBlueAccent, - text: 'Resize'.i18n, - onPressed: () { - Navigator.pop(context); - showCupertinoModalPopup( - context: context, - builder: (context) { - final screenSize = MediaQuery.of(context).size; - return ImageResizer( - onImageResize: (w, h) { - final res = getEmbedNode( - controller, controller.selection.start); - final attr = base.replaceStyleString( - getImageStyleString(controller), w, h); - controller - ..skipRequestKeyboard = true - ..formatText( - res.offset, 1, StyleAttribute(attr)); - }, - imageWidth: imageSize?.width, - imageHeight: imageSize?.height, - maxWidth: screenSize.width, - maxHeight: screenSize.height); - }); - }, - ); final copyOption = _SimpleDialogItem( icon: Icons.copy_all_outlined, color: Colors.cyanAccent, @@ -161,9 +177,95 @@ class ImageEmbedBuilder extends EmbedBuilder { padding: const EdgeInsets.fromLTRB(50, 0, 50, 0), child: SimpleDialog( shape: const RoundedRectangleBorder( - borderRadius: BorderRadius.all(Radius.circular(10))), + borderRadius: BorderRadius.all( + Radius.circular(10), + ), + ), children: [ - if (base.isMobile()) resizeOption, + _SimpleDialogItem( + icon: Icons.settings_outlined, + color: Colors.lightBlueAccent, + text: 'Resize'.i18n, + onPressed: () { + Navigator.pop(context); + showCupertinoModalPopup( + context: context, + builder: (context) { + final screenSize = MediaQuery.sizeOf(context); + return ImageResizer( + onImageResize: (w, h) { + final res = getEmbedNode( + controller, + controller.selection.start, + ); + // For desktop + String _replaceStyleStringWithSize( + String s, + double width, + double height, + ) { + final result = {}; + final pairs = s.split(';'); + for (final pair in pairs) { + final _index = pair.indexOf(':'); + if (_index < 0) { + continue; + } + final _key = + pair.substring(0, _index).trim(); + result[_key] = + pair.substring(_index + 1).trim(); + } + + result[Attribute.width.key] = + width.toString(); + result[Attribute.height.key] = + height.toString(); + final sb = StringBuffer(); + for (final pair in result.entries) { + sb + ..write(pair.key) + ..write(': ') + ..write(pair.value) + ..write('; '); + } + return sb.toString(); + } + + // TODO: When update flutter_quill + // we should update flutter_quill_extensions + // to use the latest version and use + // base.replaceStyleStringWithSize() + // instead of replaceStyleString + + final attr = base.isMobile() + ? base.replaceStyleString( + getImageStyleString(controller), + w, + h, + ) + : _replaceStyleStringWithSize( + getImageStyleString(controller), + w, + h, + ); + controller + ..skipRequestKeyboard = true + ..formatText( + res.offset, + 1, + StyleAttribute(attr), + ); + }, + imageWidth: imageSize?.width, + imageHeight: imageSize?.height, + maxWidth: screenSize.width, + maxHeight: screenSize.height, + ); + }, + ); + }, + ), copyOption, removeOption, ]), @@ -179,9 +281,11 @@ class ImageEmbedBuilder extends EmbedBuilder { // and that is up to the developer if (!base.isMobile() && forceUseMobileOptionMenu) { return _menuOptionsForReadonlyImage( - context, - imageUrl, - image, + context: context, + imageUrl: imageUrl, + image: image, + imageProviderBuilder: imageProviderBuilder, + imageErrorWidgetBuilder: imageErrorWidgetBuilder, ); } return image; @@ -189,9 +293,11 @@ class ImageEmbedBuilder extends EmbedBuilder { // We provide option menu for mobile platform excluding base64 image return _menuOptionsForReadonlyImage( - context, - imageUrl, - image, + context: context, + imageUrl: imageUrl, + image: image, + imageProviderBuilder: imageProviderBuilder, + imageErrorWidgetBuilder: imageErrorWidgetBuilder, ); } } @@ -299,72 +405,91 @@ class FormulaEmbedBuilder extends EmbedBuilder { } } -Widget _menuOptionsForReadonlyImage( - BuildContext context, String imageUrl, Widget image) { +Widget _menuOptionsForReadonlyImage({ + required BuildContext context, + required String imageUrl, + required Widget image, + required ImageEmbedBuilderProviderBuilder? imageProviderBuilder, + required ImageEmbedBuilderErrorWidgetBuilder? imageErrorWidgetBuilder, +}) { return GestureDetector( onTap: () { showDialog( - context: context, - builder: (context) { - final saveOption = _SimpleDialogItem( - icon: Icons.save, - color: Colors.greenAccent, - text: 'Save'.i18n, - onPressed: () async { - imageUrl = appendFileExtensionToImageUrl(imageUrl); - final messenger = ScaffoldMessenger.of(context); - Navigator.of(context).pop(); - - final saveImageResult = await saveImage(imageUrl); - final imageSavedSuccessfully = saveImageResult.isSuccess; - - messenger.clearSnackBars(); - - if (!imageSavedSuccessfully) { - messenger.showSnackBar(SnackBar( - content: Text( - 'Error while saving image'.i18n, - ))); - return; - } - - var message; - switch (saveImageResult.method) { - case SaveImageResultMethod.network: - message = 'Saved using the network'.i18n; - break; - case SaveImageResultMethod.localStorage: - message = 'Saved using the local storage'.i18n; - break; - } - - messenger.showSnackBar( - SnackBar( - content: Text(message), + context: context, + builder: (context) { + final saveOption = _SimpleDialogItem( + icon: Icons.save, + color: Colors.greenAccent, + text: 'Save'.i18n, + onPressed: () async { + imageUrl = appendFileExtensionToImageUrl(imageUrl); + final messenger = ScaffoldMessenger.of(context); + Navigator.of(context).pop(); + + final saveImageResult = await saveImage(imageUrl); + final imageSavedSuccessfully = saveImageResult.isSuccess; + + messenger.clearSnackBars(); + + if (!imageSavedSuccessfully) { + messenger.showSnackBar(SnackBar( + content: Text( + 'Error while saving image'.i18n, + ))); + return; + } + + var message; + switch (saveImageResult.method) { + case SaveImageResultMethod.network: + message = 'Saved using the network'.i18n; + break; + case SaveImageResultMethod.localStorage: + message = 'Saved using the local storage'.i18n; + break; + } + + messenger.showSnackBar( + SnackBar( + content: Text(message), + ), + ); + }, + ); + final zoomOption = _SimpleDialogItem( + icon: Icons.zoom_in, + color: Colors.cyanAccent, + text: 'Zoom'.i18n, + onPressed: () { + Navigator.pushReplacement( + context, + // TODO: Consider add support for other theme system + // like Cupertino or at least add the option to by + // by using PageRoute as option so dev can ovveride this + // this change should be done in all places if you want to + MaterialPageRoute( + builder: (context) => ImageTapWrapper( + imageUrl: imageUrl, + imageProviderBuilder: imageProviderBuilder, + imageErrorWidgetBuilder: imageErrorWidgetBuilder, ), - ); - }, - ); - final zoomOption = _SimpleDialogItem( - icon: Icons.zoom_in, - color: Colors.cyanAccent, - text: 'Zoom'.i18n, - onPressed: () { - Navigator.pushReplacement( - context, - MaterialPageRoute( - builder: (context) => - ImageTapWrapper(imageUrl: imageUrl))); - }, - ); - return Padding( - padding: const EdgeInsets.fromLTRB(50, 0, 50, 0), - child: SimpleDialog( - shape: const RoundedRectangleBorder( - borderRadius: BorderRadius.all(Radius.circular(10))), - children: [saveOption, zoomOption]), - ); - }); + ), + ); + }, + ); + return Padding( + padding: const EdgeInsets.fromLTRB(50, 0, 50, 0), + child: SimpleDialog( + shape: const RoundedRectangleBorder( + borderRadius: BorderRadius.all( + Radius.circular(10), + ), + ), + children: [saveOption, zoomOption], + ), + ); + }, + ); }, child: image); } diff --git a/flutter_quill_extensions/lib/embeds/embed_types.dart b/flutter_quill_extensions/lib/embeds/embed_types.dart index 46cc1563..9fa4184a 100644 --- a/flutter_quill_extensions/lib/embeds/embed_types.dart +++ b/flutter_quill_extensions/lib/embeds/embed_types.dart @@ -1,7 +1,8 @@ import 'dart:io' show File; import 'dart:typed_data'; -import 'package:flutter/material.dart'; +import 'package:flutter/material.dart' + show ImageErrorWidgetBuilder, BuildContext, ImageProvider; typedef OnImagePickCallback = Future Function(File file); typedef OnVideoPickCallback = Future Function(File file); @@ -52,3 +53,10 @@ typedef ImageEmbedBuilderWillRemoveCallback = Future Function( typedef ImageEmbedBuilderOnRemovedCallback = Future Function( File imageFile, ); + +typedef ImageEmbedBuilderProviderBuilder = ImageProvider Function( + String imageUrl, + // {required bool isLocalImage} +); + +typedef ImageEmbedBuilderErrorWidgetBuilder = ImageErrorWidgetBuilder; diff --git a/flutter_quill_extensions/lib/embeds/toolbar/camera_button.dart b/flutter_quill_extensions/lib/embeds/toolbar/camera_button.dart index 5ec8e28e..b9cb4458 100644 --- a/flutter_quill_extensions/lib/embeds/toolbar/camera_button.dart +++ b/flutter_quill_extensions/lib/embeds/toolbar/camera_button.dart @@ -61,30 +61,40 @@ class CameraButton extends StatelessWidget { size: iconSize * 1.77, fillColor: iconFillColor, borderRadius: iconTheme?.borderRadius ?? 2, - onPressed: () => _handleCameraButtonTap(context, controller, - onImagePickCallback: onImagePickCallback, - onVideoPickCallback: onVideoPickCallback, - filePickImpl: filePickImpl, - webImagePickImpl: webImagePickImpl), + onPressed: () => _handleCameraButtonTap( + context, + controller, + onImagePickCallback: onImagePickCallback, + onVideoPickCallback: onVideoPickCallback, + filePickImpl: filePickImpl, + webImagePickImpl: webImagePickImpl, + ), ); } Future _handleCameraButtonTap( - BuildContext context, QuillController controller, - {OnImagePickCallback? onImagePickCallback, - OnVideoPickCallback? onVideoPickCallback, - FilePickImpl? filePickImpl, - WebImagePickImpl? webImagePickImpl}) async { - if (onImagePickCallback != null && onVideoPickCallback != null) { - final selector = cameraPickSettingSelector ?? - (context) => showDialog( - context: context, - builder: (ctx) => AlertDialog( - contentPadding: EdgeInsets.zero, - backgroundColor: Colors.transparent, - content: Column( - mainAxisSize: MainAxisSize.min, - children: [ + BuildContext context, + QuillController controller, { + OnImagePickCallback? onImagePickCallback, + OnVideoPickCallback? onVideoPickCallback, + FilePickImpl? filePickImpl, + WebImagePickImpl? webImagePickImpl, + }) async { + if (onVideoPickCallback == null && onImagePickCallback == null) { + throw ArgumentError( + 'onImagePickCallback and onVideoPickCallback are both null', + ); + } + final selector = cameraPickSettingSelector ?? + (context) => showDialog( + context: context, + builder: (ctx) => AlertDialog( + contentPadding: EdgeInsets.zero, + backgroundColor: Colors.transparent, + content: Column( + mainAxisSize: MainAxisSize.min, + children: [ + if (onImagePickCallback != null) TextButton.icon( icon: const Icon( Icons.camera, @@ -94,6 +104,7 @@ class CameraButton extends StatelessWidget { onPressed: () => Navigator.pop(ctx, MediaPickSetting.Camera), ), + if (onVideoPickCallback != null) TextButton.icon( icon: const Icon( Icons.video_call, @@ -103,28 +114,44 @@ class CameraButton extends StatelessWidget { onPressed: () => Navigator.pop(ctx, MediaPickSetting.Video), ) - ], - ), + ], ), - ); - - final source = await selector(context); - if (source != null) { - switch (source) { - case MediaPickSetting.Camera: - await ImageVideoUtils.handleImageButtonTap( - context, controller, ImageSource.camera, onImagePickCallback, - filePickImpl: filePickImpl, webImagePickImpl: webImagePickImpl); - break; - case MediaPickSetting.Video: - await ImageVideoUtils.handleVideoButtonTap( - context, controller, ImageSource.camera, onVideoPickCallback, - filePickImpl: filePickImpl, webVideoPickImpl: webVideoPickImpl); - break; - default: - throw ArgumentError('Invalid MediaSetting'); - } - } + ), + ); + + final source = await selector(context); + if (source == null) { + return; + } + switch (source) { + case MediaPickSetting.Camera: + await ImageVideoUtils.handleImageButtonTap( + context, + controller, + ImageSource.camera, + onImagePickCallback!, + filePickImpl: filePickImpl, + webImagePickImpl: webImagePickImpl, + ); + break; + case MediaPickSetting.Video: + await ImageVideoUtils.handleVideoButtonTap( + context, + controller, + ImageSource.camera, + onVideoPickCallback!, + filePickImpl: filePickImpl, + webVideoPickImpl: webVideoPickImpl, + ); + break; + case MediaPickSetting.Gallery: + throw ArgumentError( + 'Invalid MediaSetting for the camera button', + ); + case MediaPickSetting.Link: + throw ArgumentError( + 'Invalid MediaSetting for the camera button', + ); } } } diff --git a/flutter_quill_extensions/lib/embeds/toolbar/image_button.dart b/flutter_quill_extensions/lib/embeds/toolbar/image_button.dart index dffbb387..2cc4781a 100644 --- a/flutter_quill_extensions/lib/embeds/toolbar/image_button.dart +++ b/flutter_quill_extensions/lib/embeds/toolbar/image_button.dart @@ -64,20 +64,47 @@ class ImageButton extends StatelessWidget { } Future _onPressedHandler(BuildContext context) async { - if (onImagePickCallback != null) { - final selector = - mediaPickSettingSelector ?? ImageVideoUtils.selectMediaPickSetting; - final source = await selector(context); - if (source != null) { - if (source == MediaPickSetting.Gallery) { - _pickImage(context); - } else { - _typeLink(context); - } - } - } else { + final onImagePickCallbackRef = onImagePickCallback; + if (onImagePickCallbackRef == null) { _typeLink(context); + return; } + final selector = + mediaPickSettingSelector ?? ImageVideoUtils.selectMediaPickSetting; + final source = await selector(context); + if (source == null) { + return; + } + switch (source) { + case MediaPickSetting.Gallery: + _pickImage(context); + break; + case MediaPickSetting.Link: + _typeLink(context); + break; + case MediaPickSetting.Camera: + await ImageVideoUtils.handleImageButtonTap( + context, + controller, + ImageSource.camera, + onImagePickCallbackRef, + filePickImpl: filePickImpl, + webImagePickImpl: webImagePickImpl, + ); + break; + case MediaPickSetting.Video: + throw ArgumentError( + 'Sorry but this is the Image button and not the video one', + ); + } + + // This will not work for the pick image using camera (bug fix) + // if (source != null) { + // if (source == MediaPickSetting.Gallery) { + + // } else { + // _typeLink(context); + // } } void _pickImage(BuildContext context) => ImageVideoUtils.handleImageButtonTap( diff --git a/flutter_quill_extensions/lib/embeds/toolbar/image_video_utils.dart b/flutter_quill_extensions/lib/embeds/toolbar/image_video_utils.dart index beb68ce3..f7739ce9 100644 --- a/flutter_quill_extensions/lib/embeds/toolbar/image_video_utils.dart +++ b/flutter_quill_extensions/lib/embeds/toolbar/image_video_utils.dart @@ -38,6 +38,10 @@ class LinkDialogState extends State { // TODO: Consider replace the default Regex with this one // Since that is not the reason I sent the changes then I will not edit it + // TODO: Consider use one of those as default or provide a + // way to custmize the check, that are not based on RegExp, + // I already implemented one so tell me if you are interested + // final defaultLinkNonSecureRegExp = RegExp(r'https?://.*?\.(?:png|jpe?g|gif|bmp|webp|tiff?)'); // Not secure // final defaultLinkRegExp = RegExp(r'https://.*?\.(?:png|jpe?g|gif|bmp|webp|tiff?)'); // Secure // _linkRegExp = widget.linkRegExp ?? defaultLinkRegExp; @@ -53,9 +57,10 @@ class LinkDialogState extends State { maxLines: null, style: widget.dialogTheme?.inputTextStyle, decoration: InputDecoration( - labelText: 'Paste a link'.i18n, - labelStyle: widget.dialogTheme?.labelTextStyle, - floatingLabelStyle: widget.dialogTheme?.labelTextStyle), + labelText: 'Paste a link'.i18n, + labelStyle: widget.dialogTheme?.labelTextStyle, + floatingLabelStyle: widget.dialogTheme?.labelTextStyle, + ), autofocus: true, onChanged: _linkChanged, controller: _controller, @@ -119,12 +124,13 @@ class ImageVideoUtils { /// For image picking logic static Future handleImageButtonTap( - BuildContext context, - QuillController controller, - ImageSource imageSource, - OnImagePickCallback onImagePickCallback, - {FilePickImpl? filePickImpl, - WebImagePickImpl? webImagePickImpl}) async { + BuildContext context, + QuillController controller, + ImageSource imageSource, + OnImagePickCallback onImagePickCallback, { + FilePickImpl? filePickImpl, + WebImagePickImpl? webImagePickImpl, + }) async { final index = controller.selection.baseOffset; final length = controller.selection.extentOffset - index; @@ -149,7 +155,9 @@ class ImageVideoUtils { } static Future _pickImage( - ImageSource source, OnImagePickCallback onImagePickCallback) async { + ImageSource source, + OnImagePickCallback onImagePickCallback, + ) async { final pickedFile = await ImagePicker().pickImage(source: source); if (pickedFile == null) { return null; diff --git a/flutter_quill_extensions/lib/embeds/toolbar/media_button.dart b/flutter_quill_extensions/lib/embeds/toolbar/media_button.dart index 2eaeec60..1bb1673f 100644 --- a/flutter_quill_extensions/lib/embeds/toolbar/media_button.dart +++ b/flutter_quill_extensions/lib/embeds/toolbar/media_button.dart @@ -118,7 +118,8 @@ class MediaButton extends StatelessWidget { Future _pickImage() async { if (!(kIsWeb || isMobile() || isDesktop())) { throw UnsupportedError( - 'Unsupported target platform: ${defaultTargetPlatform.name}'); + 'Unsupported target platform: ${defaultTargetPlatform.name}', + ); } final mediaFileUrl = await _pickMediaFileUrl(); @@ -127,7 +128,11 @@ class MediaButton extends StatelessWidget { final index = controller.selection.baseOffset; final length = controller.selection.extentOffset - index; controller.replaceText( - index, length, BlockEmbed.image(mediaFileUrl), null); + index, + length, + BlockEmbed.image(mediaFileUrl), + null, + ); } } @@ -219,9 +224,8 @@ class _MediaLinkDialogState extends State { Widget build(BuildContext context) { final constraints = widget.dialogTheme?.linkDialogConstraints ?? () { - final mediaQuery = MediaQuery.of(context); - final maxWidth = - kIsWeb ? mediaQuery.size.width / 4 : mediaQuery.size.width - 80; + final size = MediaQuery.sizeOf(context); + final maxWidth = kIsWeb ? size.width / 4 : size.width - 80; return BoxConstraints(maxWidth: maxWidth, maxHeight: 80); }(); @@ -333,13 +337,13 @@ class MediaSourceSelectorDialog extends StatelessWidget { Widget build(BuildContext context) { final constraints = dialogTheme?.mediaSelectorDialogConstraints ?? () { - final mediaQuery = MediaQuery.of(context); + final size = MediaQuery.sizeOf(context); double maxWidth, maxHeight; if (kIsWeb) { - maxWidth = mediaQuery.size.width / 7; - maxHeight = mediaQuery.size.height / 7; + maxWidth = size.width / 7; + maxHeight = size.height / 7; } else { - maxWidth = mediaQuery.size.width - 80; + maxWidth = size.width - 80; maxHeight = maxWidth / 2; } return BoxConstraints(maxWidth: maxWidth, maxHeight: maxHeight); diff --git a/flutter_quill_extensions/lib/embeds/utils.dart b/flutter_quill_extensions/lib/embeds/utils.dart index aea72e15..9d92f149 100644 --- a/flutter_quill_extensions/lib/embeds/utils.dart +++ b/flutter_quill_extensions/lib/embeds/utils.dart @@ -1,6 +1,6 @@ import 'dart:io' show File; -import 'package:flutter/foundation.dart' show Uint8List; +import 'package:flutter/foundation.dart' show Uint8List, immutable; import 'package:gal/gal.dart'; import 'package:http/http.dart' as http; @@ -41,6 +41,7 @@ bool isImageBase64(String imageUrl) { enum SaveImageResultMethod { network, localStorage } +@immutable class _SaveImageResult { const _SaveImageResult({required this.isSuccess, required this.method}); diff --git a/flutter_quill_extensions/lib/embeds/widgets/image.dart b/flutter_quill_extensions/lib/embeds/widgets/image.dart index 2679dac5..fe698a43 100644 --- a/flutter_quill_extensions/lib/embeds/widgets/image.dart +++ b/flutter_quill_extensions/lib/embeds/widgets/image.dart @@ -5,6 +5,7 @@ import 'package:flutter/material.dart'; import 'package:flutter_quill/flutter_quill.dart'; import 'package:photo_view/photo_view.dart'; +import '../embed_types.dart'; import '../utils.dart'; const List imageFileExtensions = [ @@ -27,21 +28,44 @@ String getImageStyleString(QuillController controller) { return s ?? ''; } -Image imageByUrl(String imageUrl, - {double? width, - double? height, - AlignmentGeometry alignment = Alignment.center}) { +Image getQuillImageByUrl( + String imageUrl, { + required ImageEmbedBuilderProviderBuilder? imageProviderBuilder, + required ImageErrorWidgetBuilder? imageErrorWidgetBuilder, + double? width, + double? height, + AlignmentGeometry alignment = Alignment.center, +}) { if (isImageBase64(imageUrl)) { return Image.memory(base64.decode(imageUrl), width: width, height: height, alignment: alignment); } - if (imageUrl.startsWith('http')) { - return Image.network(imageUrl, - width: width, height: height, alignment: alignment); + if (imageProviderBuilder != null) { + return Image( + image: imageProviderBuilder(imageUrl), + width: width, + height: height, + alignment: alignment, + errorBuilder: imageErrorWidgetBuilder, + ); + } + if (isHttpBasedUrl(imageUrl)) { + return Image.network( + imageUrl, + width: width, + height: height, + alignment: alignment, + errorBuilder: imageErrorWidgetBuilder, + ); } - return Image.file(File(imageUrl), - width: width, height: height, alignment: alignment); + return Image.file( + File(imageUrl), + width: width, + height: height, + alignment: alignment, + errorBuilder: imageErrorWidgetBuilder, + ); } String standardizeImageUrl(String url) { @@ -73,12 +97,22 @@ String appendFileExtensionToImageUrl(String url) { class ImageTapWrapper extends StatelessWidget { const ImageTapWrapper({ required this.imageUrl, + required this.imageProviderBuilder, + required this.imageErrorWidgetBuilder, }); final String imageUrl; + final ImageEmbedBuilderProviderBuilder? imageProviderBuilder; + final ImageEmbedBuilderErrorWidgetBuilder? imageErrorWidgetBuilder; - ImageProvider _imageProviderByUrl(String imageUrl) { - if (imageUrl.startsWith('http')) { + ImageProvider _imageProviderByUrl( + String imageUrl, { + required ImageEmbedBuilderProviderBuilder? customImageProviderBuilder, + }) { + if (customImageProviderBuilder != null) { + return customImageProviderBuilder(imageUrl); + } + if (isHttpBasedUrl(imageUrl)) { return NetworkImage(imageUrl); } @@ -90,12 +124,16 @@ class ImageTapWrapper extends StatelessWidget { return Scaffold( body: Container( constraints: BoxConstraints.expand( - height: MediaQuery.of(context).size.height, + height: MediaQuery.sizeOf(context).height, ), child: Stack( children: [ PhotoView( - imageProvider: _imageProviderByUrl(imageUrl), + imageProvider: _imageProviderByUrl( + imageUrl, + customImageProviderBuilder: imageProviderBuilder, + ), + errorBuilder: imageErrorWidgetBuilder, loadingBuilder: (context, event) { return Container( color: Colors.black, @@ -107,7 +145,7 @@ class ImageTapWrapper extends StatelessWidget { ), Positioned( right: 10, - top: MediaQuery.of(context).padding.top + 10.0, + top: MediaQuery.paddingOf(context).top + 10.0, child: InkWell( onTap: () { Navigator.pop(context); diff --git a/flutter_quill_extensions/lib/embeds/widgets/image_resizer.dart b/flutter_quill_extensions/lib/embeds/widgets/image_resizer.dart index 19552296..9f84ad17 100644 --- a/flutter_quill_extensions/lib/embeds/widgets/image_resizer.dart +++ b/flutter_quill_extensions/lib/embeds/widgets/image_resizer.dart @@ -42,6 +42,11 @@ class _ImageResizerState extends State { return _showCupertinoMenu(); case TargetPlatform.android: return _showMaterialMenu(); + case TargetPlatform.macOS: + case TargetPlatform.windows: + case TargetPlatform.linux: + case TargetPlatform.fuchsia: + return _showMaterialMenu(); default: throw 'Not supposed to be invoked for $defaultTargetPlatform'; } @@ -68,7 +73,11 @@ class _ImageResizerState extends State { } Widget _slider( - double value, double max, String label, ValueChanged onChanged) { + double value, + double max, + String label, + ValueChanged onChanged, + ) { return Padding( padding: const EdgeInsets.symmetric(horizontal: 8), child: Card( diff --git a/flutter_quill_extensions/lib/flutter_quill_extensions.dart b/flutter_quill_extensions/lib/flutter_quill_extensions.dart index afb49cce..e4c77396 100644 --- a/flutter_quill_extensions/lib/flutter_quill_extensions.dart +++ b/flutter_quill_extensions/lib/flutter_quill_extensions.dart @@ -72,11 +72,35 @@ class FlutterQuillEmbeds { /// ), /// ); /// - /// // Return `true` to allow image removal if the user confirms, otherwise `false` + /// // Return `true` to allow image removal if the user confirms, otherwise + /// `false` /// return isShouldRemove; /// } /// ``` /// + /// [imageProviderBuilder] if you want to use custom image provider, please + /// pass a value to this property + /// By default we will use [NetworkImage] provider if the image url/path + /// is using http/https, if not then we will use [FileImage] provider + /// If you ovveride this make sure to handle the case where if the [imageUrl] + /// is in the local storage or it does exists in the system file + /// or use the same way we did it + /// + /// Example of [imageProviderBuilder] customization: + /// ```dart + /// imageProviderBuilder: (imageUrl) async { + /// // Example of using cached_network_image package + /// // Don't forgot to check if that image is local or network one + /// return CachedNetworkImageProvider(imageUrl); + /// } + /// ``` + /// + /// [imageErrorWidgetBuilder] if you want to show a custom widget based on the + /// exception that happen while loading the image, if it network image or + /// local one, and it will get called on all the images even in the photo + /// preview widget and not just in the quill editor + /// by default the default error from flutter framework will thrown + /// /// [forceUseMobileOptionMenuForImageClick] is a boolean /// flag that, when set to `true`, /// enforces the use of the mobile-specific option menu for image clicks in @@ -107,10 +131,14 @@ class FlutterQuillEmbeds { void Function(GlobalKey videoContainerKey)? onVideoInit, ImageEmbedBuilderOnRemovedCallback? onImageRemovedCallback, ImageEmbedBuilderWillRemoveCallback? shouldRemoveImageCallback, + ImageEmbedBuilderProviderBuilder? imageProviderBuilder, + ImageEmbedBuilderErrorWidgetBuilder? imageErrorWidgetBuilder, bool forceUseMobileOptionMenuForImageClick = false, }) => [ ImageEmbedBuilder( + imageErrorWidgetBuilder: imageErrorWidgetBuilder, + imageProviderBuilder: imageProviderBuilder, forceUseMobileOptionMenu: forceUseMobileOptionMenuForImageClick, onImageRemovedCallback: onImageRemovedCallback ?? (imageFile) async { diff --git a/flutter_quill_extensions/pubspec.yaml b/flutter_quill_extensions/pubspec.yaml index 206cc0c7..c557c735 100644 --- a/flutter_quill_extensions/pubspec.yaml +++ b/flutter_quill_extensions/pubspec.yaml @@ -12,7 +12,7 @@ dependencies: flutter: sdk: flutter - flutter_quill: ^7.4.9 + flutter_quill: ^7.4.13 http: ^1.1.0 image_picker: ">=1.0.4" diff --git a/lib/src/models/documents/attribute.dart b/lib/src/models/documents/attribute.dart index c8d40360..d0dd2cfc 100644 --- a/lib/src/models/documents/attribute.dart +++ b/lib/src/models/documents/attribute.dart @@ -102,6 +102,8 @@ class Attribute { static final ScriptAttribute script = ScriptAttribute(null); + // TODO: You might want to mark those as key (mobileWidthKey) + // because it was not very clear to a developer that is new to this project static const String mobileWidth = 'mobileWidth'; static const String mobileHeight = 'mobileHeight'; @@ -110,6 +112,12 @@ class Attribute { static const String mobileAlignment = 'mobileAlignment'; + /// For other platforms, for mobile use [mobileAlignment] + static const String alignment = 'alignment'; + + /// For other platforms, for mobile use [mobileMargin] + static const String margin = 'margin'; + static const ImageAttribute image = ImageAttribute(null); static const VideoAttribute video = VideoAttribute(null); diff --git a/lib/src/models/rules/rule.dart b/lib/src/models/rules/rule.dart index de9db513..96a6d413 100644 --- a/lib/src/models/rules/rule.dart +++ b/lib/src/models/rules/rule.dart @@ -59,8 +59,14 @@ class Rules { _customRules = customRules; } - Delta apply(RuleType ruleType, Document document, int index, - {int? len, Object? data, Attribute? attribute}) { + Delta apply( + RuleType ruleType, + Document document, + int index, { + int? len, + Object? data, + Attribute? attribute, + }) { final delta = document.toDelta(); for (final rule in _customRules + _rules) { if (rule.type != ruleType) { @@ -76,6 +82,8 @@ class Rules { rethrow; } } - throw 'Apply rules failed'; + throw FormatException( + 'Apply delta rules failed. No matching rule found for type: $ruleType', + ); } } diff --git a/lib/src/translations/toolbar.i18n.dart b/lib/src/translations/toolbar.i18n.dart index a7e0eb49..6ad15ec9 100644 --- a/lib/src/translations/toolbar.i18n.dart +++ b/lib/src/translations/toolbar.i18n.dart @@ -206,9 +206,10 @@ extension Localization on String { 'Find text': 'بحث عن نص', 'Move to previous occurrence': 'الانتقال إلى الحدث السابق', 'Move to next occurrence': 'الانتقال إلى الحدث التالي', - 'Saved using the network': 'Saved using the network', - 'Saved using the local storage': 'Saved using the local storage', - 'Error while saving image': 'Error while saving image', + 'Saved using the network': 'تم الحفظ باستخدام الشبكة', + 'Saved using the local storage': + 'تم الحفظ باستخدام وحدة التخزين المحلية', + 'Error while saving image': 'حدث خطأ أثناء حفظ الصورة', }, 'da': { 'Paste a link': 'Indsæt link', diff --git a/lib/src/utils/platform.dart b/lib/src/utils/platform.dart index 96cb866b..75965bdd 100644 --- a/lib/src/utils/platform.dart +++ b/lib/src/utils/platform.dart @@ -1,12 +1,15 @@ import 'package:device_info_plus/device_info_plus.dart'; -import 'package:flutter/foundation.dart'; +import 'package:flutter/foundation.dart' + show kIsWeb, TargetPlatform, defaultTargetPlatform; bool isMobile([TargetPlatform? targetPlatform]) { + if (kIsWeb) return false; targetPlatform ??= defaultTargetPlatform; return {TargetPlatform.iOS, TargetPlatform.android}.contains(targetPlatform); } bool isDesktop([TargetPlatform? targetPlatform]) { + if (kIsWeb) return false; targetPlatform ??= defaultTargetPlatform; return {TargetPlatform.macOS, TargetPlatform.linux, TargetPlatform.windows} .contains(targetPlatform); @@ -18,6 +21,7 @@ bool isKeyboardOS([TargetPlatform? targetPlatform]) { } bool isAppleOS([TargetPlatform? targetPlatform]) { + if (kIsWeb) return false; targetPlatform ??= defaultTargetPlatform; return { TargetPlatform.macOS, diff --git a/lib/src/utils/string.dart b/lib/src/utils/string.dart index 5acadb9b..f639c716 100644 --- a/lib/src/utils/string.dart +++ b/lib/src/utils/string.dart @@ -19,7 +19,26 @@ Map parseKeyValuePairs(String s, Set targetKeys) { return result; } -String replaceStyleString(String s, double width, double height) { +@Deprecated('Use replaceStyleStringWithSize instead') +String replaceStyleString( + String s, + double width, + double height, +) { + return replaceStyleStringWithSize( + s, + width: width, + height: height, + isMobile: true, + ); +} + +String replaceStyleStringWithSize( + String s, { + required double width, + required double height, + required bool isMobile, +}) { final result = {}; final pairs = s.split(';'); for (final pair in pairs) { @@ -31,8 +50,13 @@ String replaceStyleString(String s, double width, double height) { result[_key] = pair.substring(_index + 1).trim(); } - result[Attribute.mobileWidth] = width.toString(); - result[Attribute.mobileHeight] = height.toString(); + if (isMobile) { + result[Attribute.mobileWidth] = width.toString(); + result[Attribute.mobileHeight] = height.toString(); + } else { + result[Attribute.width.key] = width.toString(); + result[Attribute.height.key] = height.toString(); + } final sb = StringBuffer(); for (final pair in result.entries) { sb diff --git a/lib/src/widgets/editor.dart b/lib/src/widgets/editor.dart index 896b6ae4..11fd530d 100644 --- a/lib/src/widgets/editor.dart +++ b/lib/src/widgets/editor.dart @@ -497,7 +497,7 @@ class QuillEditorState extends State cupertinoTheme.primaryColor.withOpacity(0.40); cursorRadius ??= const Radius.circular(2); cursorOffset = Offset( - iOSHorizontalOffset / MediaQuery.of(context).devicePixelRatio, 0); + iOSHorizontalOffset / MediaQuery.devicePixelRatioOf(context), 0); } else { textSelectionControls = materialTextSelectionControls; paintCursorAboveText = false; diff --git a/lib/src/widgets/raw_editor.dart b/lib/src/widgets/raw_editor.dart index 5c68111b..15483c88 100644 --- a/lib/src/widgets/raw_editor.dart +++ b/lib/src/widgets/raw_editor.dart @@ -977,7 +977,7 @@ class RawEditorState extends EditorState widget.selectionColor, widget.enableInteractiveSelection, _hasFocus, - MediaQuery.of(context).devicePixelRatio, + MediaQuery.devicePixelRatioOf(context), _cursorCont); return editableTextLine; } diff --git a/lib/src/widgets/text_block.dart b/lib/src/widgets/text_block.dart index 16bebeec..64dacb4f 100644 --- a/lib/src/widgets/text_block.dart +++ b/lib/src/widgets/text_block.dart @@ -159,7 +159,7 @@ class EditableTextBlock extends StatelessWidget { color, enableInteractiveSelection, hasFocus, - MediaQuery.of(context).devicePixelRatio, + MediaQuery.devicePixelRatioOf(context), cursorCont); final nodeTextDirection = getDirectionOfNode(line); children.add(Directionality( diff --git a/lib/src/widgets/toolbar/link_style_button2.dart b/lib/src/widgets/toolbar/link_style_button2.dart index 42aaff11..98140312 100644 --- a/lib/src/widgets/toolbar/link_style_button2.dart +++ b/lib/src/widgets/toolbar/link_style_button2.dart @@ -245,9 +245,8 @@ class _LinkStyleDialogState extends State { final constraints = widget.constraints ?? widget.dialogTheme?.linkDialogConstraints ?? () { - final mediaQuery = MediaQuery.of(context); - final maxWidth = - kIsWeb ? mediaQuery.size.width / 4 : mediaQuery.size.width - 80; + final size = MediaQuery.sizeOf(context); + final maxWidth = kIsWeb ? size.width / 4 : size.width - 80; return BoxConstraints(maxWidth: maxWidth, maxHeight: 80); }(); diff --git a/pubspec.yaml b/pubspec.yaml index 459d9b92..24457151 100644 --- a/pubspec.yaml +++ b/pubspec.yaml @@ -1,6 +1,6 @@ name: flutter_quill description: A rich text editor built for the modern Android, iOS, web and desktop platforms. It is the WYSIWYG editor and a Quill component for Flutter. -version: 7.4.13 +version: 7.4.14 homepage: https://1o24bbs.com/c/bulletjournal/108 repository: https://github.com/singerdmx/flutter-quill