Bring all the changes from fresh_quill_extensions (#1495)
* Bring all the changes from fresh_quill_extensions * Update pubspec.yaml * Update pubspec.yaml and export new file * Update pubspec.yaml * Update CHANGELOG.md * Update README.mdpull/1506/head
parent
ba4a416f83
commit
bffae92b50
50 changed files with 2573 additions and 1548 deletions
@ -1,22 +1,154 @@ |
||||
# Flutter Quill Extensions |
||||
|
||||
Helpers to support embed widgets in flutter_quill. See [Flutter Quill](https://pub.dev/packages/flutter_quill) for details of use. |
||||
A extensions for [flutter_quill](https://pub.dev/packages/flutter_quill) |
||||
to support embed widgets like image, formula, video and more. |
||||
|
||||
Currently the support for **Web** is limitied. |
||||
|
||||
Check [Flutter Quill](https://github.com/singerdmx/flutter-quill) for details of use. |
||||
|
||||
## Table of Contents |
||||
|
||||
- [Flutter Quill Extensions](#flutter-quill-extensions) |
||||
- [Table of Contents](#table-of-contents) |
||||
- [About](#about) |
||||
- [Installation](#installation) |
||||
- [Platform Spesefic Configurations](#platform-spesefic-configurations) |
||||
- [Usage](#usage) |
||||
- [Features](#features) |
||||
- [Contributing](#contributing) |
||||
- [License](#license) |
||||
- [Acknowledgments](#acknowledgments) |
||||
|
||||
|
||||
## About |
||||
Flutter quill is a rich editor text and it's allow you to customize a lot of things, it has custom embed builders which allow you to render custom widgets in the editor <br> |
||||
this is a extensions to extends it functionallities by adding more features like images, videos, and more |
||||
|
||||
## Installation |
||||
|
||||
Before start using this package, please make sure to install |
||||
[flutter_quill](https://github.com/singerdmx/flutter-quill) package first and follow it's usage instructions. |
||||
|
||||
```yaml |
||||
dependencies: |
||||
flutter_quill_extensions: ^<latest-version-here> |
||||
``` |
||||
|
||||
## Platform Spesefic Configurations |
||||
|
||||
> |
||||
> 1. We are using [`gal`](https://github.com/natsuk4ze/) plugin to save images. |
||||
> For this to work, you need to add the appropriate permissions |
||||
> to your `Info.plist` and `AndroidManifest.xml` files. |
||||
> See <https://github.com/natsuk4ze/gal#-get-started> to add the needed lines. |
||||
> |
||||
> 2. We also use [`image_picker`](https://pub.dev/packages/image_picker) plugin for picking images so please make sure follow the instructions |
||||
> |
||||
> 3. For loading the image from the internet we need internet permission |
||||
> 1. For Android, you need to add some permissions in `AndroidManifest.xml`, Please follow this [link](https://developer.android.com/training/basics/network-ops/connecting) for more info, the internet permission included by default only for debugging so you need to follow this link to add it in the release version too. you should allow loading images and videos only for the `https` protocol but if you want http too then you need to configure your android application to accept `http` in the release mode, follow this [link](https://stackoverflow.com/questions/45940861/android-8-cleartext-http-traffic-not-permitted) for more info. |
||||
> 2. for macOS you also need to include a key in your `Info.plist`, please follow this [link](https://stackoverflow.com/a/61201081/18519412) to add the required configurations |
||||
> |
||||
> The extensions package also use [image_picker](https://pub.dev/packages/image_picker) which also require some configurations, follow this [link](https://pub.dev/packages/image_picker#installation). It's needed for Android, iOS, macOS, we must inform you that you can't pick photo using camera in desktop so make sure to handle that if you plan on add support for desktop, this might changed in the future and for more info follow this [link](https://pub.dev/packages/image_picker#windows-macos-and-linux) <br> |
||||
> |
||||
|
||||
## Usage |
||||
|
||||
Set the `embedBuilders` and `embedToolbar` params in `QuillEditor` and `QuillToolbar` with the |
||||
Before starting using this package you must follow the setup |
||||
|
||||
Set the `embedBuilders` and `embedToolbar` params in configurations of `QuillEditor` and `QuillToolbar` with the |
||||
values provided by this repository. |
||||
|
||||
**Quill toolbar**: |
||||
```dart |
||||
QuillToolbar( |
||||
configurations: QuillToolbarConfigurations( |
||||
embedButtons: FlutterQuillEmbeds.toolbarButtons( |
||||
imageButtonOptions: QuillToolbarImageButtonOptions( |
||||
onImagePickCallback: (file) async { |
||||
return file.path; |
||||
}, |
||||
), |
||||
), |
||||
), |
||||
), |
||||
``` |
||||
QuillEditor.basic( |
||||
controller: controller, |
||||
embedBuilders: FlutterQuillEmbeds.builders(), |
||||
); |
||||
|
||||
**Quill Editor** |
||||
```dart |
||||
Expanded( |
||||
child: QuillEditor.basic( |
||||
configurations: QuillEditorConfigurations( |
||||
readOnly: true, |
||||
embedBuilders: FlutterQuillEmbeds.editorBuilders( |
||||
imageEmbedConfigurations: |
||||
const QuillEditorImageEmbedConfigurations( |
||||
forceUseMobileOptionMenuForImageClick: true, |
||||
), |
||||
), |
||||
), |
||||
), |
||||
) |
||||
``` |
||||
|
||||
They both should be have a parent `QuillProvider` in the widget tree and setup properly <br> |
||||
Example: |
||||
|
||||
```dart |
||||
QuillProvider( |
||||
configurations: QuillConfigurations( |
||||
controller: _controller, |
||||
sharedConfigurations: const QuillSharedConfigurations(), |
||||
), |
||||
child: Column( |
||||
children: [ |
||||
QuillToolbar( |
||||
configurations: QuillToolbarConfigurations( |
||||
embedButtons: FlutterQuillEmbeds.toolbarButtons( |
||||
imageButtonOptions: QuillToolbarImageButtonOptions(), |
||||
), |
||||
), |
||||
), |
||||
Expanded( |
||||
child: QuillEditor.basic( |
||||
configurations: QuillEditorConfigurations( |
||||
padding: const EdgeInsets.all(16), |
||||
embedBuilders: FlutterQuillEmbeds.editorBuilders(), |
||||
), |
||||
), |
||||
) |
||||
], |
||||
), |
||||
) |
||||
``` |
||||
QuillToolbar.basic( |
||||
controller: controller, |
||||
embedButtons: FlutterQuillEmbeds.buttons(), |
||||
); |
||||
|
||||
For web, use: |
||||
```dart |
||||
FlutterQuillEmbeds.editorsWebBuilders() |
||||
``` |
||||
|
||||
## Features |
||||
|
||||
```markdown |
||||
## Features |
||||
|
||||
- Easy to use and customizable |
||||
- Has the option to use custom image provider for the images |
||||
- Usefull utilities and widgets |
||||
- Handle different errors |
||||
``` |
||||
|
||||
## Contributing |
||||
|
||||
We welcome contributions! |
||||
|
||||
Please follow these guidelines when contributing to our project. See [CONTRIBUTING.md](CONTRIBUTING.md) for more details. |
||||
|
||||
## License |
||||
|
||||
This project is licensed under the [MIT License](LICENSE) - see the [LICENSE](LICENSE) file for details. |
||||
|
||||
## Acknowledgments |
||||
|
||||
- Thanks to the [Flutter Team](https://flutter.dev/) |
||||
- Thanks to [flutter_quill](https://pub.dev/packages/flutter_quill) |
@ -0,0 +1,12 @@ |
||||
// import 'package:meta/meta.dart'; |
||||
|
||||
// @immutable |
||||
// class NetworkException implements Exception { |
||||
// const NetworkException({required this.message}); |
||||
|
||||
// final String message; |
||||
|
||||
// @override |
||||
// String toString() => |
||||
// 'Error while loading something from the network: $message'; |
||||
// } |
@ -1,476 +0,0 @@ |
||||
import 'dart:io' show File; |
||||
|
||||
import 'package:flutter/cupertino.dart'; |
||||
import 'package:flutter/foundation.dart'; |
||||
import 'package:flutter/material.dart'; |
||||
import 'package:flutter/services.dart'; |
||||
import 'package:flutter_quill/extensions.dart' as base; |
||||
import 'package:flutter_quill/flutter_quill.dart'; |
||||
import 'package:flutter_quill/translations.dart'; |
||||
import 'package:math_keyboard/math_keyboard.dart'; |
||||
import 'package:universal_html/html.dart' as html; |
||||
|
||||
import '../shims/dart_ui_fake.dart' |
||||
if (dart.library.html) '../shims/dart_ui_real.dart' as ui; |
||||
import 'embed_types.dart'; |
||||
import 'utils.dart'; |
||||
import 'widgets/image.dart'; |
||||
import 'widgets/image_resizer.dart'; |
||||
import 'widgets/video_app.dart'; |
||||
import 'widgets/youtube_video_app.dart'; |
||||
|
||||
class ImageEmbedBuilder extends EmbedBuilder { |
||||
ImageEmbedBuilder({ |
||||
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; |
||||
|
||||
@override |
||||
bool get expanded => false; |
||||
|
||||
@override |
||||
Widget build( |
||||
BuildContext context, |
||||
QuillController controller, |
||||
base.Embed node, |
||||
bool readOnly, |
||||
bool inline, |
||||
TextStyle textStyle, |
||||
) { |
||||
assert(!kIsWeb, 'Please provide image EmbedBuilder for Web'); |
||||
|
||||
Widget image = const SizedBox.shrink(); |
||||
final imageUrl = standardizeImageUrl(node.value.data); |
||||
OptionalSize? imageSize; |
||||
final style = node.style.attributes['style']; |
||||
|
||||
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, |
||||
Attribute.margin, |
||||
Attribute.alignment, |
||||
}); |
||||
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[Attribute.alignment]); |
||||
final margin = (base.isMobile() |
||||
? double.tryParse(Attribute.mobileMargin) |
||||
: double.tryParse(Attribute.margin)) ?? |
||||
0.0; |
||||
|
||||
assert( |
||||
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(margin), |
||||
child: getQuillImageByUrl( |
||||
imageUrl, |
||||
width: width, |
||||
height: height, |
||||
alignment: alignment, |
||||
imageProviderBuilder: imageProviderBuilder, |
||||
imageErrorWidgetBuilder: imageErrorWidgetBuilder, |
||||
), |
||||
); |
||||
} |
||||
} |
||||
|
||||
if (imageSize == null) { |
||||
image = getQuillImageByUrl( |
||||
imageUrl, |
||||
imageProviderBuilder: imageProviderBuilder, |
||||
imageErrorWidgetBuilder: imageErrorWidgetBuilder, |
||||
); |
||||
imageSize = OptionalSize((image as Image).width, image.height); |
||||
} |
||||
|
||||
if (!readOnly && (base.isMobile() || forceUseMobileOptionMenu)) { |
||||
return GestureDetector( |
||||
onTap: () { |
||||
showDialog( |
||||
context: context, |
||||
builder: (context) { |
||||
final copyOption = _SimpleDialogItem( |
||||
icon: Icons.copy_all_outlined, |
||||
color: Colors.cyanAccent, |
||||
text: 'Copy'.i18n, |
||||
onPressed: () { |
||||
final imageNode = |
||||
getEmbedNode(controller, controller.selection.start) |
||||
.value; |
||||
final imageUrl = imageNode.value.data; |
||||
controller.copiedImageUrl = |
||||
ImageUrl(imageUrl, getImageStyleString(controller)); |
||||
Navigator.pop(context); |
||||
}, |
||||
); |
||||
final removeOption = _SimpleDialogItem( |
||||
icon: Icons.delete_forever_outlined, |
||||
color: Colors.red.shade200, |
||||
text: 'Remove'.i18n, |
||||
onPressed: () async { |
||||
Navigator.of(context).pop(); |
||||
|
||||
final imageFile = File(imageUrl); |
||||
|
||||
// Call the remove check callback if set |
||||
if (await shouldRemoveImageCallback?.call(imageFile) == |
||||
false) { |
||||
return; |
||||
} |
||||
|
||||
final offset = getEmbedNode( |
||||
controller, |
||||
controller.selection.start, |
||||
).offset; |
||||
controller.replaceText( |
||||
offset, |
||||
1, |
||||
'', |
||||
TextSelection.collapsed(offset: offset), |
||||
); |
||||
// Call the post remove callback if set |
||||
await onImageRemovedCallback?.call(imageFile); |
||||
}, |
||||
); |
||||
return Padding( |
||||
padding: const EdgeInsets.fromLTRB(50, 0, 50, 0), |
||||
child: SimpleDialog( |
||||
shape: const RoundedRectangleBorder( |
||||
borderRadius: BorderRadius.all( |
||||
Radius.circular(10), |
||||
), |
||||
), |
||||
children: [ |
||||
_SimpleDialogItem( |
||||
icon: Icons.settings_outlined, |
||||
color: Colors.lightBlueAccent, |
||||
text: 'Resize'.i18n, |
||||
onPressed: () { |
||||
Navigator.pop(context); |
||||
showCupertinoModalPopup<void>( |
||||
context: context, |
||||
builder: (context) { |
||||
final screenSize = MediaQuery.sizeOf(context); |
||||
return ImageResizer( |
||||
onImageResize: (w, h) { |
||||
final res = getEmbedNode( |
||||
controller, |
||||
controller.selection.start, |
||||
); |
||||
|
||||
final attr = |
||||
base.replaceStyleStringWithSize( |
||||
getImageStyleString(controller), |
||||
width: w, |
||||
height: h, |
||||
isMobile: base.isMobile(), |
||||
); |
||||
controller |
||||
..skipRequestKeyboard = true |
||||
..formatText( |
||||
res.offset, |
||||
1, |
||||
StyleAttribute(attr), |
||||
); |
||||
}, |
||||
imageWidth: imageSize?.width, |
||||
imageHeight: imageSize?.height, |
||||
maxWidth: screenSize.width, |
||||
maxHeight: screenSize.height, |
||||
); |
||||
}, |
||||
); |
||||
}, |
||||
), |
||||
copyOption, |
||||
removeOption, |
||||
]), |
||||
); |
||||
}); |
||||
}, |
||||
child: image, |
||||
); |
||||
} |
||||
|
||||
if (!readOnly || isImageBase64(imageUrl)) { |
||||
// To enforce using it on the web, desktop and other platforms |
||||
// and that is up to the developer |
||||
if (!base.isMobile() && forceUseMobileOptionMenu) { |
||||
return _menuOptionsForReadonlyImage( |
||||
context: context, |
||||
imageUrl: imageUrl, |
||||
image: image, |
||||
imageProviderBuilder: imageProviderBuilder, |
||||
imageErrorWidgetBuilder: imageErrorWidgetBuilder, |
||||
); |
||||
} |
||||
return image; |
||||
} |
||||
|
||||
// We provide option menu for mobile platform excluding base64 image |
||||
return _menuOptionsForReadonlyImage( |
||||
context: context, |
||||
imageUrl: imageUrl, |
||||
image: image, |
||||
imageProviderBuilder: imageProviderBuilder, |
||||
imageErrorWidgetBuilder: imageErrorWidgetBuilder, |
||||
); |
||||
} |
||||
} |
||||
|
||||
class ImageEmbedBuilderWeb extends EmbedBuilder { |
||||
ImageEmbedBuilderWeb({this.constraints}) |
||||
: assert(kIsWeb, 'ImageEmbedBuilderWeb is only for web platform'); |
||||
|
||||
final BoxConstraints? constraints; |
||||
|
||||
@override |
||||
String get key => BlockEmbed.imageType; |
||||
|
||||
@override |
||||
Widget build( |
||||
BuildContext context, |
||||
QuillController controller, |
||||
Embed node, |
||||
bool readOnly, |
||||
bool inline, |
||||
TextStyle textStyle, |
||||
) { |
||||
final imageUrl = node.value.data; |
||||
|
||||
ui.platformViewRegistry.registerViewFactory(imageUrl, (viewId) { |
||||
return html.ImageElement() |
||||
..src = imageUrl |
||||
..style.height = 'auto' |
||||
..style.width = 'auto'; |
||||
}); |
||||
|
||||
return ConstrainedBox( |
||||
constraints: constraints ?? BoxConstraints.loose(const Size(200, 200)), |
||||
child: HtmlElementView( |
||||
viewType: imageUrl, |
||||
), |
||||
); |
||||
} |
||||
} |
||||
|
||||
class VideoEmbedBuilder extends EmbedBuilder { |
||||
VideoEmbedBuilder({this.onVideoInit}); |
||||
|
||||
final void Function(GlobalKey videoContainerKey)? onVideoInit; |
||||
|
||||
@override |
||||
String get key => BlockEmbed.videoType; |
||||
|
||||
@override |
||||
Widget build( |
||||
BuildContext context, |
||||
QuillController controller, |
||||
base.Embed node, |
||||
bool readOnly, |
||||
bool inline, |
||||
TextStyle textStyle, |
||||
) { |
||||
assert(!kIsWeb, 'Please provide video EmbedBuilder for Web'); |
||||
|
||||
final videoUrl = node.value.data; |
||||
if (isYouTubeUrl(videoUrl)) { |
||||
return YoutubeVideoApp( |
||||
videoUrl: videoUrl, context: context, readOnly: readOnly); |
||||
} |
||||
return VideoApp( |
||||
videoUrl: videoUrl, |
||||
context: context, |
||||
readOnly: readOnly, |
||||
onVideoInit: onVideoInit, |
||||
); |
||||
} |
||||
} |
||||
|
||||
class FormulaEmbedBuilder extends EmbedBuilder { |
||||
@override |
||||
String get key => BlockEmbed.formulaType; |
||||
|
||||
@override |
||||
Widget build( |
||||
BuildContext context, |
||||
QuillController controller, |
||||
base.Embed node, |
||||
bool readOnly, |
||||
bool inline, |
||||
TextStyle textStyle, |
||||
) { |
||||
assert(!kIsWeb, 'Please provide formula EmbedBuilder for Web'); |
||||
|
||||
final mathController = MathFieldEditingController(); |
||||
return Focus( |
||||
onFocusChange: (hasFocus) { |
||||
if (hasFocus) { |
||||
// If the MathField is tapped, hides the built in keyboard |
||||
SystemChannels.textInput.invokeMethod('TextInput.hide'); |
||||
debugPrint(mathController.currentEditingValue()); |
||||
} |
||||
}, |
||||
child: MathField( |
||||
controller: mathController, |
||||
variables: const ['x', 'y', 'z'], |
||||
onChanged: (value) {}, |
||||
onSubmitted: (value) {}, |
||||
), |
||||
); |
||||
} |
||||
} |
||||
|
||||
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), |
||||
), |
||||
); |
||||
}, |
||||
); |
||||
final zoomOption = _SimpleDialogItem( |
||||
icon: Icons.zoom_in, |
||||
color: Colors.cyanAccent, |
||||
text: 'Zoom'.i18n, |
||||
onPressed: () { |
||||
Navigator.pushReplacement( |
||||
context, |
||||
MaterialPageRoute( |
||||
builder: (context) => ImageTapWrapper( |
||||
imageUrl: imageUrl, |
||||
imageProviderBuilder: imageProviderBuilder, |
||||
imageErrorWidgetBuilder: imageErrorWidgetBuilder, |
||||
), |
||||
), |
||||
); |
||||
}, |
||||
); |
||||
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); |
||||
} |
||||
|
||||
class _SimpleDialogItem extends StatelessWidget { |
||||
const _SimpleDialogItem( |
||||
{required this.icon, |
||||
required this.color, |
||||
required this.text, |
||||
required this.onPressed, |
||||
Key? key}) |
||||
: super(key: key); |
||||
|
||||
final IconData icon; |
||||
final Color color; |
||||
final String text; |
||||
final VoidCallback onPressed; |
||||
|
||||
@override |
||||
Widget build(BuildContext context) { |
||||
return SimpleDialogOption( |
||||
onPressed: onPressed, |
||||
child: Row( |
||||
children: [ |
||||
Icon(icon, size: 36, color: color), |
||||
Padding( |
||||
padding: const EdgeInsetsDirectional.only(start: 16), |
||||
child: |
||||
Text(text, style: const TextStyle(fontWeight: FontWeight.bold)), |
||||
), |
||||
], |
||||
), |
||||
); |
||||
} |
||||
} |
@ -1,159 +0,0 @@ |
||||
import 'package:flutter/material.dart'; |
||||
import 'package:flutter_quill/flutter_quill.dart'; |
||||
import 'package:flutter_quill/translations.dart'; |
||||
import 'package:image_picker/image_picker.dart'; |
||||
|
||||
import '../embed_types.dart'; |
||||
import 'image_video_utils.dart'; |
||||
|
||||
class CameraButton extends StatelessWidget { |
||||
const CameraButton({ |
||||
required this.icon, |
||||
required this.controller, |
||||
this.iconSize = kDefaultIconSize, |
||||
this.fillColor, |
||||
this.onImagePickCallback, |
||||
this.onVideoPickCallback, |
||||
this.filePickImpl, |
||||
this.webImagePickImpl, |
||||
this.webVideoPickImpl, |
||||
this.cameraPickSettingSelector, |
||||
this.iconTheme, |
||||
this.tooltip, |
||||
Key? key, |
||||
}) : super(key: key); |
||||
|
||||
final IconData icon; |
||||
final double iconSize; |
||||
|
||||
final Color? fillColor; |
||||
|
||||
final QuillController controller; |
||||
|
||||
final OnImagePickCallback? onImagePickCallback; |
||||
|
||||
final OnVideoPickCallback? onVideoPickCallback; |
||||
|
||||
final WebImagePickImpl? webImagePickImpl; |
||||
|
||||
final WebVideoPickImpl? webVideoPickImpl; |
||||
|
||||
final FilePickImpl? filePickImpl; |
||||
|
||||
final MediaPickSettingSelector? cameraPickSettingSelector; |
||||
|
||||
final QuillIconTheme? iconTheme; |
||||
final String? tooltip; |
||||
|
||||
@override |
||||
Widget build(BuildContext context) { |
||||
final theme = Theme.of(context); |
||||
|
||||
final iconColor = iconTheme?.iconUnselectedColor ?? theme.iconTheme.color; |
||||
final iconFillColor = |
||||
iconTheme?.iconUnselectedFillColor ?? (fillColor ?? theme.canvasColor); |
||||
|
||||
return QuillToolbarIconButton( |
||||
icon: Icon(icon, size: iconSize, color: iconColor), |
||||
tooltip: tooltip, |
||||
highlightElevation: 0, |
||||
hoverElevation: 0, |
||||
size: iconSize * 1.77, |
||||
fillColor: iconFillColor, |
||||
borderRadius: iconTheme?.borderRadius ?? 2, |
||||
onPressed: () => _handleCameraButtonTap( |
||||
context, |
||||
controller, |
||||
onImagePickCallback: onImagePickCallback, |
||||
onVideoPickCallback: onVideoPickCallback, |
||||
filePickImpl: filePickImpl, |
||||
webImagePickImpl: webImagePickImpl, |
||||
), |
||||
); |
||||
} |
||||
|
||||
Future<void> _handleCameraButtonTap( |
||||
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<MediaPickSetting>( |
||||
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, |
||||
color: Colors.orangeAccent, |
||||
), |
||||
label: Text('Camera'.i18n), |
||||
onPressed: () => |
||||
Navigator.pop(ctx, MediaPickSetting.Camera), |
||||
), |
||||
if (onVideoPickCallback != null) |
||||
TextButton.icon( |
||||
icon: const Icon( |
||||
Icons.video_call, |
||||
color: Colors.cyanAccent, |
||||
), |
||||
label: Text('Video'.i18n), |
||||
onPressed: () => |
||||
Navigator.pop(ctx, MediaPickSetting.Video), |
||||
) |
||||
], |
||||
), |
||||
), |
||||
); |
||||
|
||||
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.\n' |
||||
'gallery is not related to camera button', |
||||
); |
||||
case MediaPickSetting.Link: |
||||
throw ArgumentError( |
||||
'Invalid MediaSetting for the camera button.\n' |
||||
'link is not related to camera button', |
||||
); |
||||
} |
||||
} |
||||
} |
@ -1,55 +0,0 @@ |
||||
import 'package:flutter/material.dart'; |
||||
import 'package:flutter_quill/flutter_quill.dart'; |
||||
|
||||
class FormulaButton extends StatelessWidget { |
||||
const FormulaButton({ |
||||
required this.icon, |
||||
required this.controller, |
||||
this.iconSize = kDefaultIconSize, |
||||
this.fillColor, |
||||
this.iconTheme, |
||||
this.dialogTheme, |
||||
this.tooltip, |
||||
Key? key, |
||||
}) : super(key: key); |
||||
|
||||
final IconData icon; |
||||
|
||||
final double iconSize; |
||||
|
||||
final Color? fillColor; |
||||
|
||||
final QuillController controller; |
||||
|
||||
final QuillIconTheme? iconTheme; |
||||
|
||||
final QuillDialogTheme? dialogTheme; |
||||
final String? tooltip; |
||||
|
||||
@override |
||||
Widget build(BuildContext context) { |
||||
final theme = Theme.of(context); |
||||
|
||||
final iconColor = iconTheme?.iconUnselectedColor ?? theme.iconTheme.color; |
||||
final iconFillColor = |
||||
iconTheme?.iconUnselectedFillColor ?? (fillColor ?? theme.canvasColor); |
||||
|
||||
return QuillToolbarIconButton( |
||||
icon: Icon(icon, size: iconSize, color: iconColor), |
||||
tooltip: tooltip, |
||||
highlightElevation: 0, |
||||
hoverElevation: 0, |
||||
size: iconSize * 1.77, |
||||
fillColor: iconFillColor, |
||||
borderRadius: iconTheme?.borderRadius ?? 2, |
||||
onPressed: () => _onPressedHandler(context), |
||||
); |
||||
} |
||||
|
||||
Future<void> _onPressedHandler(BuildContext context) async { |
||||
final index = controller.selection.baseOffset; |
||||
final length = controller.selection.extentOffset - index; |
||||
|
||||
controller.replaceText(index, length, BlockEmbed.formula(''), null); |
||||
} |
||||
} |
@ -1,138 +0,0 @@ |
||||
import 'package:flutter/material.dart'; |
||||
import 'package:flutter_quill/flutter_quill.dart'; |
||||
import 'package:image_picker/image_picker.dart'; |
||||
|
||||
import '../embed_types.dart'; |
||||
import 'image_video_utils.dart'; |
||||
|
||||
class ImageButton extends StatelessWidget { |
||||
const ImageButton({ |
||||
required this.icon, |
||||
required this.controller, |
||||
this.iconSize = kDefaultIconSize, |
||||
this.onImagePickCallback, |
||||
this.fillColor, |
||||
this.filePickImpl, |
||||
this.webImagePickImpl, |
||||
this.mediaPickSettingSelector, |
||||
this.iconTheme, |
||||
this.dialogTheme, |
||||
this.tooltip, |
||||
this.linkRegExp, |
||||
Key? key, |
||||
}) : super(key: key); |
||||
|
||||
final IconData icon; |
||||
final double iconSize; |
||||
|
||||
final Color? fillColor; |
||||
|
||||
final QuillController controller; |
||||
|
||||
final OnImagePickCallback? onImagePickCallback; |
||||
|
||||
final WebImagePickImpl? webImagePickImpl; |
||||
|
||||
final FilePickImpl? filePickImpl; |
||||
|
||||
final MediaPickSettingSelector? mediaPickSettingSelector; |
||||
|
||||
final QuillIconTheme? iconTheme; |
||||
|
||||
final QuillDialogTheme? dialogTheme; |
||||
final String? tooltip; |
||||
final RegExp? linkRegExp; |
||||
|
||||
@override |
||||
Widget build(BuildContext context) { |
||||
final theme = Theme.of(context); |
||||
|
||||
final iconColor = iconTheme?.iconUnselectedColor ?? theme.iconTheme.color; |
||||
final iconFillColor = |
||||
iconTheme?.iconUnselectedFillColor ?? (fillColor ?? theme.canvasColor); |
||||
|
||||
return QuillToolbarIconButton( |
||||
icon: Icon(icon, size: iconSize, color: iconColor), |
||||
tooltip: tooltip, |
||||
highlightElevation: 0, |
||||
hoverElevation: 0, |
||||
size: iconSize * 1.77, |
||||
fillColor: iconFillColor, |
||||
borderRadius: iconTheme?.borderRadius ?? 2, |
||||
onPressed: () => _onPressedHandler(context), |
||||
); |
||||
} |
||||
|
||||
Future<void> _onPressedHandler(BuildContext context) async { |
||||
final onImagePickCallbackRef = onImagePickCallback; |
||||
if (onImagePickCallbackRef == null) { |
||||
await _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: |
||||
await _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( |
||||
context, |
||||
controller, |
||||
ImageSource.gallery, |
||||
onImagePickCallback!, |
||||
filePickImpl: filePickImpl, |
||||
webImagePickImpl: webImagePickImpl, |
||||
); |
||||
|
||||
Future<void> _typeLink(BuildContext context) async { |
||||
final value = await showDialog<String>( |
||||
context: context, |
||||
builder: (_) => LinkDialog( |
||||
dialogTheme: dialogTheme, |
||||
linkRegExp: linkRegExp, |
||||
), |
||||
); |
||||
_linkSubmitted(value); |
||||
} |
||||
|
||||
void _linkSubmitted(String? value) { |
||||
if (value != null && value.isNotEmpty) { |
||||
final index = controller.selection.baseOffset; |
||||
final length = controller.selection.extentOffset - index; |
||||
|
||||
controller.replaceText(index, length, BlockEmbed.image(value), null); |
||||
} |
||||
} |
||||
} |
@ -1,110 +0,0 @@ |
||||
import 'package:flutter/material.dart'; |
||||
import 'package:flutter_quill/flutter_quill.dart'; |
||||
import 'package:image_picker/image_picker.dart'; |
||||
|
||||
import '../embed_types.dart'; |
||||
import 'image_video_utils.dart'; |
||||
|
||||
class VideoButton extends StatelessWidget { |
||||
const VideoButton({ |
||||
required this.icon, |
||||
required this.controller, |
||||
this.iconSize = kDefaultIconSize, |
||||
this.onVideoPickCallback, |
||||
this.fillColor, |
||||
this.filePickImpl, |
||||
this.webVideoPickImpl, |
||||
this.mediaPickSettingSelector, |
||||
this.iconTheme, |
||||
this.dialogTheme, |
||||
this.tooltip, |
||||
this.linkRegExp, |
||||
Key? key, |
||||
}) : super(key: key); |
||||
|
||||
final IconData icon; |
||||
final double iconSize; |
||||
|
||||
final Color? fillColor; |
||||
|
||||
final QuillController controller; |
||||
|
||||
final OnVideoPickCallback? onVideoPickCallback; |
||||
|
||||
final WebVideoPickImpl? webVideoPickImpl; |
||||
|
||||
final FilePickImpl? filePickImpl; |
||||
|
||||
final MediaPickSettingSelector? mediaPickSettingSelector; |
||||
|
||||
final QuillIconTheme? iconTheme; |
||||
|
||||
final QuillDialogTheme? dialogTheme; |
||||
|
||||
final String? tooltip; |
||||
|
||||
final RegExp? linkRegExp; |
||||
|
||||
@override |
||||
Widget build(BuildContext context) { |
||||
final theme = Theme.of(context); |
||||
|
||||
final iconColor = iconTheme?.iconUnselectedColor ?? theme.iconTheme.color; |
||||
final iconFillColor = |
||||
iconTheme?.iconUnselectedFillColor ?? (fillColor ?? theme.canvasColor); |
||||
|
||||
return QuillToolbarIconButton( |
||||
icon: Icon(icon, size: iconSize, color: iconColor), |
||||
tooltip: tooltip, |
||||
highlightElevation: 0, |
||||
hoverElevation: 0, |
||||
size: iconSize * 1.77, |
||||
fillColor: iconFillColor, |
||||
borderRadius: iconTheme?.borderRadius ?? 2, |
||||
onPressed: () => _onPressedHandler(context), |
||||
); |
||||
} |
||||
|
||||
Future<void> _onPressedHandler(BuildContext context) async { |
||||
if (onVideoPickCallback != null) { |
||||
final selector = |
||||
mediaPickSettingSelector ?? ImageVideoUtils.selectMediaPickSetting; |
||||
final source = await selector(context); |
||||
if (source != null) { |
||||
if (source == MediaPickSetting.Gallery) { |
||||
_pickVideo(context); |
||||
} else { |
||||
await _typeLink(context); |
||||
} |
||||
} |
||||
} else { |
||||
await _typeLink(context); |
||||
} |
||||
} |
||||
|
||||
void _pickVideo(BuildContext context) => ImageVideoUtils.handleVideoButtonTap( |
||||
context, |
||||
controller, |
||||
ImageSource.gallery, |
||||
onVideoPickCallback!, |
||||
filePickImpl: filePickImpl, |
||||
webVideoPickImpl: webVideoPickImpl, |
||||
); |
||||
|
||||
Future<void> _typeLink(BuildContext context) async { |
||||
final value = await showDialog<String>( |
||||
context: context, |
||||
builder: (_) => LinkDialog(dialogTheme: dialogTheme), |
||||
); |
||||
_linkSubmitted(value); |
||||
} |
||||
|
||||
void _linkSubmitted(String? value) { |
||||
if (value != null && value.isNotEmpty) { |
||||
final index = controller.selection.baseOffset; |
||||
final length = controller.selection.extentOffset - index; |
||||
|
||||
controller.replaceText(index, length, BlockEmbed.video(value), null); |
||||
} |
||||
} |
||||
} |
@ -1,106 +0,0 @@ |
||||
import 'dart:io' show File; |
||||
|
||||
import 'package:flutter/foundation.dart' show Uint8List, immutable; |
||||
import 'package:gal/gal.dart'; |
||||
import 'package:http/http.dart' as http; |
||||
|
||||
// I would like to orgnize the project structure and the code more |
||||
// but here I don't want to change too much since that is a community project |
||||
|
||||
RegExp _base64 = RegExp( |
||||
r'^(?:[A-Za-z0-9+\/]{4})*(?:[A-Za-z0-9+\/]{2}==|[A-Za-z0-9+\/]{3}=|[A-Za-z0-9+\/]{4})$', |
||||
); |
||||
|
||||
bool isBase64(String str) { |
||||
return _base64.hasMatch(str); |
||||
} |
||||
|
||||
bool isHttpBasedUrl(String url) { |
||||
try { |
||||
final uri = Uri.parse(url.trim()); |
||||
return uri.isScheme('HTTP') || uri.isScheme('HTTPS'); |
||||
} catch (_) { |
||||
return false; |
||||
} |
||||
} |
||||
|
||||
bool isYouTubeUrl(String videoUrl) { |
||||
try { |
||||
final uri = Uri.parse(videoUrl); |
||||
return uri.host == 'www.youtube.com' || |
||||
uri.host == 'youtube.com' || |
||||
uri.host == 'youtu.be'; |
||||
} catch (_) { |
||||
return false; |
||||
} |
||||
} |
||||
|
||||
bool isImageBase64(String imageUrl) { |
||||
return !isHttpBasedUrl(imageUrl) && isBase64(imageUrl); |
||||
} |
||||
|
||||
enum SaveImageResultMethod { network, localStorage } |
||||
|
||||
@immutable |
||||
class _SaveImageResult { |
||||
const _SaveImageResult({required this.isSuccess, required this.method}); |
||||
|
||||
final bool isSuccess; |
||||
final SaveImageResultMethod method; |
||||
} |
||||
|
||||
Future<_SaveImageResult> saveImage(String imageUrl) async { |
||||
final imageFile = File(imageUrl); |
||||
final hasPermission = await Gal.hasAccess(); |
||||
final imageExistsLocally = await imageFile.exists(); |
||||
if (!hasPermission) { |
||||
await Gal.requestAccess(); |
||||
} |
||||
if (!imageExistsLocally) { |
||||
final success = await _saveNetworkImageToLocal(imageUrl); |
||||
return _SaveImageResult( |
||||
isSuccess: success, |
||||
method: SaveImageResultMethod.network, |
||||
); |
||||
} |
||||
final success = await _saveImageLocally(imageFile); |
||||
return _SaveImageResult( |
||||
isSuccess: success, |
||||
method: SaveImageResultMethod.localStorage, |
||||
); |
||||
} |
||||
|
||||
Future<bool> _saveNetworkImageToLocal(String imageUrl) async { |
||||
try { |
||||
final response = await http.get( |
||||
Uri.parse(imageUrl), |
||||
); |
||||
if (response.statusCode != 200) { |
||||
return false; |
||||
} |
||||
final imageBytes = response.bodyBytes; |
||||
await Gal.putImageBytes(imageBytes); |
||||
return true; |
||||
} catch (e) { |
||||
return false; |
||||
} |
||||
} |
||||
|
||||
Future<Uint8List> _convertFileToUint8List(File file) async { |
||||
try { |
||||
final uint8list = await file.readAsBytes(); |
||||
return uint8list; |
||||
} catch (e) { |
||||
return Uint8List(0); |
||||
} |
||||
} |
||||
|
||||
Future<bool> _saveImageLocally(File imageFile) async { |
||||
try { |
||||
final imageBytes = await _convertFileToUint8List(imageFile); |
||||
await Gal.putImageBytes(imageBytes); |
||||
return true; |
||||
} catch (e) { |
||||
return false; |
||||
} |
||||
} |
@ -0,0 +1,39 @@ |
||||
import 'package:flutter_quill/flutter_quill.dart'; |
||||
|
||||
import '../../presentation/embeds/editor/webview.dart'; |
||||
|
||||
extension QuillControllerExt on QuillController { |
||||
int get index => selection.baseOffset; |
||||
int get length => selection.extentOffset - index; |
||||
void insertWebViewBlock({ |
||||
required String initialUrl, |
||||
}) { |
||||
final block = BlockEmbed.custom( |
||||
QuillEditorWebViewBlockEmbed( |
||||
initialUrl, |
||||
), |
||||
); |
||||
|
||||
this |
||||
..skipRequestKeyboard = true |
||||
..replaceText( |
||||
index, |
||||
length, |
||||
block, |
||||
null, |
||||
); |
||||
} |
||||
|
||||
void insertImageBlock({ |
||||
required String imageUrl, |
||||
}) { |
||||
this |
||||
..skipRequestKeyboard = true |
||||
..replaceText( |
||||
index, |
||||
length, |
||||
BlockEmbed.image(imageUrl), |
||||
null, |
||||
); |
||||
} |
||||
} |
@ -0,0 +1,23 @@ |
||||
import 'package:meta/meta.dart' show immutable; |
||||
|
||||
enum ImageSaverExceptionType { |
||||
accessDenied, |
||||
notEnoughSpace, |
||||
notSupportedFormat, |
||||
unexpected, |
||||
unknown; |
||||
} |
||||
|
||||
@immutable |
||||
class ImageSaverException implements Exception { |
||||
const ImageSaverException({ |
||||
required this.message, |
||||
required this.type, |
||||
}); |
||||
|
||||
final String message; |
||||
final ImageSaverExceptionType type; |
||||
|
||||
@override |
||||
String toString() => 'Error while saving image, error type: ${type.name}'; |
||||
} |
@ -0,0 +1,7 @@ |
||||
abstract class ImageSaverInterface { |
||||
const ImageSaverInterface(); |
||||
Future<void> saveLocalImage(String imageUrl); |
||||
Future<void> saveImageFromNetwork(Uri imageUrl); |
||||
Future<bool> hasAccess({required bool toAlbum}); |
||||
Future<bool> requestAccess({required bool toAlbum}); |
||||
} |
@ -0,0 +1,77 @@ |
||||
import 'package:flutter/widgets.dart' show NetworkImageLoadException; |
||||
import 'package:gal/gal.dart' show Gal, GalException, GalExceptionType; |
||||
import 'package:http/http.dart' as http; |
||||
|
||||
import '../exceptions.dart'; |
||||
import '../image_saver.dart'; |
||||
|
||||
class ImageSaverGalImpl extends ImageSaverInterface { |
||||
@override |
||||
Future<void> saveImageFromNetwork(Uri imageUrl) async { |
||||
try { |
||||
final response = await http.get( |
||||
imageUrl, |
||||
); |
||||
if (response.statusCode != 200) { |
||||
throw NetworkImageLoadException( |
||||
statusCode: response.statusCode, |
||||
uri: imageUrl, |
||||
); |
||||
} |
||||
final imageBytes = response.bodyBytes; |
||||
await Gal.putImageBytes(imageBytes); |
||||
} on GalException catch (e) { |
||||
throw ImageSaverException( |
||||
message: e.toString(), |
||||
type: e.type.toImageSaverExceptionType(), |
||||
); |
||||
} catch (e) { |
||||
throw ImageSaverException( |
||||
message: e.toString(), |
||||
type: ImageSaverExceptionType.unknown, |
||||
); |
||||
} |
||||
} |
||||
|
||||
@override |
||||
Future<void> saveLocalImage(String imageUrl) async { |
||||
try { |
||||
await Gal.putImage(imageUrl); |
||||
} on GalException catch (e) { |
||||
throw ImageSaverException( |
||||
message: e.toString(), |
||||
type: e.type.toImageSaverExceptionType(), |
||||
); |
||||
} catch (e) { |
||||
throw ImageSaverException( |
||||
message: e.toString(), |
||||
type: ImageSaverExceptionType.unknown, |
||||
); |
||||
} |
||||
} |
||||
|
||||
@override |
||||
Future<bool> hasAccess({required bool toAlbum}) { |
||||
return Gal.hasAccess(toAlbum: toAlbum); |
||||
} |
||||
|
||||
@override |
||||
Future<bool> requestAccess({required bool toAlbum}) { |
||||
return Gal.requestAccess(toAlbum: toAlbum); |
||||
} |
||||
} |
||||
|
||||
extension GalExceptionTypeExt on GalExceptionType { |
||||
ImageSaverExceptionType toImageSaverExceptionType() { |
||||
switch (this) { |
||||
case GalExceptionType.accessDenied: |
||||
return ImageSaverExceptionType.accessDenied; |
||||
case GalExceptionType.notEnoughSpace: |
||||
return ImageSaverExceptionType.notEnoughSpace; |
||||
case GalExceptionType.notSupportedFormat: |
||||
return ImageSaverExceptionType.notSupportedFormat; |
||||
case GalExceptionType.unexpected: |
||||
return ImageSaverExceptionType.unexpected; |
||||
} |
||||
} |
||||
} |
@ -0,0 +1,33 @@ |
||||
// ignore_for_file: public_member_api_docs, sort_constructors_first |
||||
import 'image_saver.dart'; |
||||
import 'packages/gal.dart' show ImageSaverGalImpl; |
||||
|
||||
class ImageSaverService extends ImageSaverInterface { |
||||
final ImageSaverInterface _provider; |
||||
const ImageSaverService({ |
||||
required ImageSaverInterface impl, |
||||
}) : _provider = impl; |
||||
|
||||
factory ImageSaverService.gal() => ImageSaverService( |
||||
impl: ImageSaverGalImpl(), |
||||
); |
||||
|
||||
static final _instance = ImageSaverService.gal(); |
||||
factory ImageSaverService.getInstance() => _instance; |
||||
|
||||
@override |
||||
Future<bool> hasAccess({bool toAlbum = false}) => |
||||
_provider.hasAccess(toAlbum: toAlbum); |
||||
|
||||
@override |
||||
Future<bool> requestAccess({bool toAlbum = false}) => |
||||
_provider.requestAccess(toAlbum: toAlbum); |
||||
|
||||
@override |
||||
Future<void> saveImageFromNetwork(Uri imageUrl) => |
||||
_provider.saveImageFromNetwork(imageUrl); |
||||
|
||||
@override |
||||
Future<void> saveLocalImage(String imageUrl) => |
||||
_provider.saveLocalImage(imageUrl); |
||||
} |
@ -0,0 +1,41 @@ |
||||
import 'package:flutter/foundation.dart' show kIsWeb; |
||||
import 'package:flutter/services.dart'; |
||||
import 'package:flutter/widgets.dart'; |
||||
import 'package:flutter_quill/extensions.dart' as base; |
||||
import 'package:flutter_quill/flutter_quill.dart'; |
||||
import 'package:math_keyboard/math_keyboard.dart'; |
||||
|
||||
class FormulaEmbedBuilder extends EmbedBuilder { |
||||
const FormulaEmbedBuilder(); |
||||
@override |
||||
String get key => BlockEmbed.formulaType; |
||||
|
||||
@override |
||||
Widget build( |
||||
BuildContext context, |
||||
QuillController controller, |
||||
base.Embed node, |
||||
bool readOnly, |
||||
bool inline, |
||||
TextStyle textStyle, |
||||
) { |
||||
assert(!kIsWeb, 'Please provide formula EmbedBuilder for Web'); |
||||
|
||||
final mathController = MathFieldEditingController(); |
||||
return Focus( |
||||
onFocusChange: (hasFocus) { |
||||
if (hasFocus) { |
||||
// If the MathField is tapped, hides the built in keyboard |
||||
SystemChannels.textInput.invokeMethod('TextInput.hide'); |
||||
debugPrint(mathController.currentEditingValue()); |
||||
} |
||||
}, |
||||
child: MathField( |
||||
controller: mathController, |
||||
variables: const ['x', 'y', 'z'], |
||||
onChanged: (value) {}, |
||||
onSubmitted: (value) {}, |
||||
), |
||||
); |
||||
} |
||||
} |
@ -0,0 +1,336 @@ |
||||
import 'dart:io' show File; |
||||
|
||||
import 'package:flutter/cupertino.dart'; |
||||
import 'package:flutter/foundation.dart'; |
||||
import 'package:flutter/material.dart'; |
||||
import 'package:flutter_quill/extensions.dart' as base; |
||||
import 'package:flutter_quill/flutter_quill.dart'; |
||||
import 'package:flutter_quill/translations.dart'; |
||||
|
||||
import '../../../models/config/editor/image.dart'; |
||||
import '../../embed_types.dart'; |
||||
import '../../utils.dart'; |
||||
import '../../widgets/image.dart'; |
||||
import '../../widgets/image_resizer.dart'; |
||||
import '../../widgets/simple_dialog_item.dart'; |
||||
|
||||
class QuillEditorImageEmbedBuilder extends EmbedBuilder { |
||||
QuillEditorImageEmbedBuilder({ |
||||
required this.configurations, |
||||
}); |
||||
final QuillEditorImageEmbedConfigurations configurations; |
||||
|
||||
@override |
||||
String get key => BlockEmbed.imageType; |
||||
|
||||
@override |
||||
bool get expanded => false; |
||||
|
||||
@override |
||||
Widget build( |
||||
BuildContext context, |
||||
QuillController controller, |
||||
base.Embed node, |
||||
bool readOnly, |
||||
bool inline, |
||||
TextStyle textStyle, |
||||
) { |
||||
assert(!kIsWeb, 'Please provide image EmbedBuilder for Web'); |
||||
|
||||
Widget image = const SizedBox.shrink(); |
||||
final imageUrl = standardizeImageUrl(node.value.data); |
||||
OptionalSize? imageSize; |
||||
final style = node.style.attributes['style']; |
||||
|
||||
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, |
||||
Attribute.margin, |
||||
Attribute.alignment, |
||||
}); |
||||
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[Attribute.alignment]); |
||||
final margin = (base.isMobile() |
||||
? double.tryParse(Attribute.mobileMargin) |
||||
: double.tryParse(Attribute.margin)) ?? |
||||
0.0; |
||||
|
||||
assert( |
||||
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(margin), |
||||
child: getQuillImageByUrl( |
||||
imageUrl, |
||||
width: width, |
||||
height: height, |
||||
alignment: alignment, |
||||
imageProviderBuilder: configurations.imageProviderBuilder, |
||||
imageErrorWidgetBuilder: configurations.imageErrorWidgetBuilder, |
||||
), |
||||
); |
||||
} |
||||
} |
||||
|
||||
if (imageSize == null) { |
||||
image = getQuillImageByUrl( |
||||
imageUrl, |
||||
imageProviderBuilder: configurations.imageProviderBuilder, |
||||
imageErrorWidgetBuilder: configurations.imageErrorWidgetBuilder, |
||||
); |
||||
imageSize = OptionalSize((image as Image).width, image.height); |
||||
} |
||||
|
||||
if (!readOnly && |
||||
(base.isMobile() || |
||||
configurations.forceUseMobileOptionMenuForImageClick)) { |
||||
return GestureDetector( |
||||
onTap: () { |
||||
showDialog( |
||||
context: context, |
||||
builder: (context) { |
||||
final copyOption = SimpleDialogItem( |
||||
icon: Icons.copy_all_outlined, |
||||
color: Colors.cyanAccent, |
||||
text: 'Copy'.i18n, |
||||
onPressed: () { |
||||
final imageNode = |
||||
getEmbedNode(controller, controller.selection.start) |
||||
.value; |
||||
final imageUrl = imageNode.value.data; |
||||
controller.copiedImageUrl = ImageUrl( |
||||
imageUrl, |
||||
getImageStyleString(controller), |
||||
); |
||||
Navigator.pop(context); |
||||
}, |
||||
); |
||||
final removeOption = SimpleDialogItem( |
||||
icon: Icons.delete_forever_outlined, |
||||
color: Colors.red.shade200, |
||||
text: 'Remove'.i18n, |
||||
onPressed: () async { |
||||
Navigator.of(context).pop(); |
||||
|
||||
final imageFile = File(imageUrl); |
||||
|
||||
// Call the remove check callback if set |
||||
if (await configurations.shouldRemoveImageCallback |
||||
?.call(imageFile) == |
||||
false) { |
||||
return; |
||||
} |
||||
|
||||
final offset = getEmbedNode( |
||||
controller, |
||||
controller.selection.start, |
||||
).offset; |
||||
controller.replaceText( |
||||
offset, |
||||
1, |
||||
'', |
||||
TextSelection.collapsed(offset: offset), |
||||
); |
||||
// Call the post remove callback if set |
||||
await configurations.onImageRemovedCallback |
||||
?.call(imageFile); |
||||
}, |
||||
); |
||||
return Padding( |
||||
padding: const EdgeInsets.fromLTRB(50, 0, 50, 0), |
||||
child: SimpleDialog( |
||||
shape: const RoundedRectangleBorder( |
||||
borderRadius: BorderRadius.all( |
||||
Radius.circular(10), |
||||
), |
||||
), |
||||
children: [ |
||||
SimpleDialogItem( |
||||
icon: Icons.settings_outlined, |
||||
color: Colors.lightBlueAccent, |
||||
text: 'Resize'.i18n, |
||||
onPressed: () { |
||||
Navigator.pop(context); |
||||
showCupertinoModalPopup<void>( |
||||
context: context, |
||||
builder: (context) { |
||||
final screenSize = MediaQuery.sizeOf(context); |
||||
return ImageResizer( |
||||
onImageResize: (w, h) { |
||||
final res = getEmbedNode( |
||||
controller, |
||||
controller.selection.start, |
||||
); |
||||
|
||||
final attr = |
||||
base.replaceStyleStringWithSize( |
||||
getImageStyleString(controller), |
||||
width: w, |
||||
height: h, |
||||
isMobile: base.isMobile(), |
||||
); |
||||
controller |
||||
..skipRequestKeyboard = true |
||||
..formatText( |
||||
res.offset, |
||||
1, |
||||
StyleAttribute(attr), |
||||
); |
||||
}, |
||||
imageWidth: imageSize?.width, |
||||
imageHeight: imageSize?.height, |
||||
maxWidth: screenSize.width, |
||||
maxHeight: screenSize.height, |
||||
); |
||||
}, |
||||
); |
||||
}, |
||||
), |
||||
copyOption, |
||||
removeOption, |
||||
]), |
||||
); |
||||
}); |
||||
}, |
||||
child: image, |
||||
); |
||||
} |
||||
|
||||
if (!readOnly || isImageBase64(imageUrl)) { |
||||
// To enforce using it on the web, desktop and other platforms |
||||
// and that is up to the developer |
||||
if (!base.isMobile() && |
||||
configurations.forceUseMobileOptionMenuForImageClick) { |
||||
return _menuOptionsForReadonlyImage( |
||||
context: context, |
||||
imageUrl: imageUrl, |
||||
image: image, |
||||
imageProviderBuilder: configurations.imageProviderBuilder, |
||||
imageErrorWidgetBuilder: configurations.imageErrorWidgetBuilder, |
||||
); |
||||
} |
||||
return image; |
||||
} |
||||
|
||||
// We provide option menu for mobile platform excluding base64 image |
||||
return _menuOptionsForReadonlyImage( |
||||
context: context, |
||||
imageUrl: imageUrl, |
||||
image: image, |
||||
imageProviderBuilder: configurations.imageProviderBuilder, |
||||
imageErrorWidgetBuilder: configurations.imageErrorWidgetBuilder, |
||||
); |
||||
} |
||||
} |
||||
|
||||
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; |
||||
} |
||||
|
||||
String 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, |
||||
MaterialPageRoute( |
||||
builder: (context) => ImageTapWrapper( |
||||
imageUrl: imageUrl, |
||||
imageProviderBuilder: imageProviderBuilder, |
||||
imageErrorWidgetBuilder: imageErrorWidgetBuilder, |
||||
), |
||||
), |
||||
); |
||||
}, |
||||
); |
||||
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, |
||||
); |
||||
} |
@ -0,0 +1,45 @@ |
||||
import 'package:flutter/foundation.dart' show kIsWeb; |
||||
import 'package:flutter/widgets.dart'; |
||||
import 'package:flutter_quill/flutter_quill.dart'; |
||||
import 'package:universal_html/html.dart' as html; |
||||
|
||||
import 'shims/dart_ui_fake.dart' |
||||
if (dart.library.html) 'shims/dart_ui_real.dart' as ui; |
||||
|
||||
class ImageEmbedBuilderWeb extends EmbedBuilder { |
||||
const ImageEmbedBuilderWeb({ |
||||
this.constraints, |
||||
}); |
||||
|
||||
final BoxConstraints? constraints; |
||||
|
||||
@override |
||||
String get key => BlockEmbed.imageType; |
||||
|
||||
@override |
||||
Widget build( |
||||
BuildContext context, |
||||
QuillController controller, |
||||
Embed node, |
||||
bool readOnly, |
||||
bool inline, |
||||
TextStyle textStyle, |
||||
) { |
||||
assert(kIsWeb, 'ImageEmbedBuilderWeb is only for web platform'); |
||||
final imageUrl = node.value.data; |
||||
|
||||
ui.PlatformViewRegistry().registerViewFactory(imageUrl, (viewId) { |
||||
return html.ImageElement() |
||||
..src = imageUrl |
||||
..style.height = 'auto' |
||||
..style.width = 'auto'; |
||||
}); |
||||
|
||||
return ConstrainedBox( |
||||
constraints: constraints ?? BoxConstraints.loose(const Size(200, 200)), |
||||
child: HtmlElementView( |
||||
viewType: imageUrl, |
||||
), |
||||
); |
||||
} |
||||
} |
@ -0,0 +1,43 @@ |
||||
// import 'package:universal_html/html.dart' as html; |
||||
|
||||
// Fake interface for the logic that this package needs from (web-only) dart:ui. |
||||
// This is conditionally exported so the analyzer sees these methods as |
||||
// available. |
||||
|
||||
// typedef PlatroformViewFactory = html.Element Function(int viewId); |
||||
|
||||
// /// Shim for web_ui engine.PlatformViewRegistry |
||||
// /// https://github.com/flutter/engine/blob/master/lib/web_ui/lib/ui.dart#L62 |
||||
// class PlatformViewRegistry { |
||||
// /// Shim for registerViewFactory |
||||
// /// https://github.com/flutter/engine/blob/master/lib/web_ui/lib/ui.dart#L72 |
||||
// static dynamic registerViewFactory( |
||||
// String viewTypeId, PlatroformViewFactory viewFactory) {} |
||||
// } |
||||
|
||||
// /// Shim for web_ui engine.AssetManager |
||||
// /// https://github.com/flutter/engine/blob/master/lib/web_ui/lib/src/engine/assets.dart#L12 |
||||
// class WebOnlyAssetManager { |
||||
// static dynamic getAssetUrl(String asset) {} |
||||
// } |
||||
|
||||
class PlatformViewRegistry { |
||||
/// Register [viewType] as being created by the given [viewFactory]. |
||||
/// |
||||
/// [viewFactory] can be any function that takes an integer and optional |
||||
/// `params` and returns an `HTMLElement` DOM object. |
||||
bool registerViewFactory( |
||||
String viewType, |
||||
Function viewFactory, { |
||||
bool isVisible = true, |
||||
}) { |
||||
return false; |
||||
} |
||||
|
||||
/// Returns the view previously created for [viewId]. |
||||
/// |
||||
/// Throws if no view has been created for [viewId]. |
||||
Object getViewById(int viewId) { |
||||
return ''; |
||||
} |
||||
} |
@ -0,0 +1,19 @@ |
||||
import 'package:flutter/widgets.dart'; |
||||
import 'package:flutter_quill/flutter_quill.dart'; |
||||
|
||||
class QuillEditorUnknownEmbedBuilder extends EmbedBuilder { |
||||
@override |
||||
Widget build( |
||||
BuildContext context, |
||||
QuillController controller, |
||||
Embed node, |
||||
bool readOnly, |
||||
bool inline, |
||||
TextStyle textStyle, |
||||
) { |
||||
return const Text('Unknown embed builder'); |
||||
} |
||||
|
||||
@override |
||||
String get key => 'unknown'; |
||||
} |
@ -0,0 +1,81 @@ |
||||
import 'package:flutter/foundation.dart'; |
||||
import 'package:flutter/material.dart'; |
||||
import 'package:flutter/services.dart'; |
||||
import 'package:flutter_quill/extensions.dart' as base; |
||||
import 'package:flutter_quill/flutter_quill.dart'; |
||||
import 'package:math_keyboard/math_keyboard.dart'; |
||||
|
||||
import '../../models/config/editor/video.dart'; |
||||
import '../utils.dart'; |
||||
import '../widgets/video_app.dart'; |
||||
import '../widgets/youtube_video_app.dart'; |
||||
|
||||
class QuillEditorVideoEmbedBuilder extends EmbedBuilder { |
||||
const QuillEditorVideoEmbedBuilder({ |
||||
required this.configurations, |
||||
}); |
||||
|
||||
final QuillEditorVideoEmbedConfigurations configurations; |
||||
|
||||
@override |
||||
String get key => BlockEmbed.videoType; |
||||
|
||||
@override |
||||
Widget build( |
||||
BuildContext context, |
||||
QuillController controller, |
||||
base.Embed node, |
||||
bool readOnly, |
||||
bool inline, |
||||
TextStyle textStyle, |
||||
) { |
||||
assert(!kIsWeb, 'Please provide video EmbedBuilder for Web'); |
||||
|
||||
final videoUrl = node.value.data; |
||||
if (isYouTubeUrl(videoUrl)) { |
||||
return YoutubeVideoApp( |
||||
videoUrl: videoUrl, context: context, readOnly: readOnly); |
||||
} |
||||
return VideoApp( |
||||
videoUrl: videoUrl, |
||||
context: context, |
||||
readOnly: readOnly, |
||||
onVideoInit: configurations.onVideoInit, |
||||
); |
||||
} |
||||
} |
||||
|
||||
class QuillEditorFormulaEmbedBuilder extends EmbedBuilder { |
||||
const QuillEditorFormulaEmbedBuilder(); |
||||
@override |
||||
String get key => BlockEmbed.formulaType; |
||||
|
||||
@override |
||||
Widget build( |
||||
BuildContext context, |
||||
QuillController controller, |
||||
base.Embed node, |
||||
bool readOnly, |
||||
bool inline, |
||||
TextStyle textStyle, |
||||
) { |
||||
assert(!kIsWeb, 'Please provide formula EmbedBuilder for Web'); |
||||
|
||||
final mathController = MathFieldEditingController(); |
||||
return Focus( |
||||
onFocusChange: (hasFocus) { |
||||
if (hasFocus) { |
||||
// If the MathField is tapped, hides the built in keyboard |
||||
SystemChannels.textInput.invokeMethod('TextInput.hide'); |
||||
debugPrint(mathController.currentEditingValue()); |
||||
} |
||||
}, |
||||
child: MathField( |
||||
controller: mathController, |
||||
variables: const ['x', 'y', 'z'], |
||||
onChanged: (value) {}, |
||||
onSubmitted: (value) {}, |
||||
), |
||||
); |
||||
} |
||||
} |
@ -0,0 +1,58 @@ |
||||
import 'dart:convert' show jsonDecode, jsonEncode; |
||||
|
||||
import 'package:flutter/widgets.dart'; |
||||
import 'package:flutter_inappwebview/flutter_inappwebview.dart'; |
||||
import 'package:flutter_quill/flutter_quill.dart'; |
||||
import 'package:meta/meta.dart' show experimental; |
||||
|
||||
import '../../models/config/editor/webview.dart'; |
||||
|
||||
@experimental |
||||
class QuillEditorWebViewBlockEmbed extends CustomBlockEmbed { |
||||
const QuillEditorWebViewBlockEmbed( |
||||
String value, |
||||
) : super(webViewType, value); |
||||
|
||||
factory QuillEditorWebViewBlockEmbed.fromDocument(Document document) => |
||||
QuillEditorWebViewBlockEmbed(jsonEncode(document.toDelta().toJson())); |
||||
|
||||
static const String webViewType = 'webview'; |
||||
|
||||
Document get document => Document.fromJson(jsonDecode(data)); |
||||
} |
||||
|
||||
@experimental |
||||
class QuillEditorWebViewEmbedBuilder extends EmbedBuilder { |
||||
const QuillEditorWebViewEmbedBuilder({ |
||||
required this.configurations, |
||||
}); |
||||
|
||||
@override |
||||
bool get expanded => false; |
||||
|
||||
final QuillEditorWebViewEmbedConfigurations configurations; |
||||
@override |
||||
Widget build( |
||||
BuildContext context, |
||||
QuillController controller, |
||||
Embed node, |
||||
bool readOnly, |
||||
bool inline, |
||||
TextStyle textStyle, |
||||
) { |
||||
final url = node.value.data as String; |
||||
|
||||
return SizedBox( |
||||
width: double.infinity, |
||||
height: 200, |
||||
child: InAppWebView( |
||||
initialUrlRequest: URLRequest( |
||||
url: Uri.parse(url), |
||||
), |
||||
), |
||||
); |
||||
} |
||||
|
||||
@override |
||||
String get key => 'webview'; |
||||
} |
@ -0,0 +1,202 @@ |
||||
// ignore_for_file: use_build_context_synchronously |
||||
|
||||
import 'package:flutter/material.dart'; |
||||
import 'package:flutter_quill/flutter_quill.dart'; |
||||
import 'package:flutter_quill/translations.dart'; |
||||
import 'package:image_picker/image_picker.dart'; |
||||
|
||||
import '../../models/config/toolbar/buttons/camera.dart'; |
||||
import '../embed_types.dart'; |
||||
import 'utils/image_video_utils.dart'; |
||||
|
||||
class QuillToolbarCameraButton extends StatelessWidget { |
||||
const QuillToolbarCameraButton({ |
||||
required this.controller, |
||||
required this.options, |
||||
super.key, |
||||
}); |
||||
|
||||
final QuillController controller; |
||||
final QuillToolbarCameraButtonOptions options; |
||||
|
||||
double _iconSize(BuildContext context) { |
||||
final baseFontSize = baseButtonExtraOptions(context).globalIconSize; |
||||
final iconSize = options.iconSize; |
||||
return iconSize ?? baseFontSize; |
||||
} |
||||
|
||||
VoidCallback? _afterButtonPressed(BuildContext context) { |
||||
return options.afterButtonPressed ?? |
||||
baseButtonExtraOptions(context).afterButtonPressed; |
||||
} |
||||
|
||||
QuillIconTheme? _iconTheme(BuildContext context) { |
||||
return options.iconTheme ?? baseButtonExtraOptions(context).iconTheme; |
||||
} |
||||
|
||||
QuillToolbarBaseButtonOptions baseButtonExtraOptions(BuildContext context) { |
||||
return context.requireQuillToolbarBaseButtonOptions; |
||||
} |
||||
|
||||
IconData _iconData(BuildContext context) { |
||||
return options.iconData ?? |
||||
baseButtonExtraOptions(context).iconData ?? |
||||
Icons.photo_camera; |
||||
} |
||||
|
||||
String _tooltip(BuildContext context) { |
||||
return options.tooltip ?? |
||||
baseButtonExtraOptions(context).tooltip ?? |
||||
'Camera'.i18n; |
||||
// ('Camera'.i18n); |
||||
} |
||||
|
||||
void _sharedOnPressed(BuildContext context) { |
||||
_onPressedHandler( |
||||
context, |
||||
controller, |
||||
onImagePickCallback: options.onImagePickCallback, |
||||
onVideoPickCallback: options.onVideoPickCallback, |
||||
filePickImpl: options.filePickImpl, |
||||
webImagePickImpl: options.webImagePickImpl, |
||||
); |
||||
_afterButtonPressed(context); |
||||
} |
||||
|
||||
@override |
||||
Widget build(BuildContext context) { |
||||
final iconTheme = _iconTheme(context); |
||||
final tooltip = _tooltip(context); |
||||
final iconSize = _iconSize(context); |
||||
final iconData = _iconData(context); |
||||
|
||||
final childBuilder = |
||||
options.childBuilder ?? baseButtonExtraOptions(context).childBuilder; |
||||
|
||||
if (childBuilder != null) { |
||||
childBuilder( |
||||
QuillToolbarCameraButtonOptions( |
||||
onImagePickCallback: options.onImagePickCallback, |
||||
onVideoPickCallback: options.onVideoPickCallback, |
||||
afterButtonPressed: _afterButtonPressed(context), |
||||
cameraPickSettingSelector: options.cameraPickSettingSelector, |
||||
filePickImpl: options.filePickImpl, |
||||
iconData: options.iconData, |
||||
fillColor: options.fillColor, |
||||
iconSize: options.iconSize, |
||||
iconTheme: options.iconTheme, |
||||
tooltip: options.tooltip, |
||||
webImagePickImpl: options.webImagePickImpl, |
||||
webVideoPickImpl: options.webVideoPickImpl, |
||||
), |
||||
QuillToolbarCameraButtonExtraOptions( |
||||
controller: controller, |
||||
context: context, |
||||
onPressed: () => _sharedOnPressed(context), |
||||
), |
||||
); |
||||
} |
||||
|
||||
final theme = Theme.of(context); |
||||
|
||||
final iconColor = iconTheme?.iconUnselectedColor ?? theme.iconTheme.color; |
||||
final iconFillColor = iconTheme?.iconUnselectedFillColor ?? |
||||
(options.fillColor ?? theme.canvasColor); |
||||
|
||||
return QuillToolbarIconButton( |
||||
icon: Icon(iconData, size: iconSize, color: iconColor), |
||||
tooltip: tooltip, |
||||
highlightElevation: 0, |
||||
hoverElevation: 0, |
||||
size: iconSize * 1.77, |
||||
fillColor: iconFillColor, |
||||
borderRadius: iconTheme?.borderRadius ?? 2, |
||||
onPressed: () => _sharedOnPressed(context), |
||||
); |
||||
} |
||||
|
||||
Future<void> _onPressedHandler( |
||||
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 = options.cameraPickSettingSelector ?? |
||||
(context) => showDialog<MediaPickSetting>( |
||||
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, |
||||
color: Colors.orangeAccent, |
||||
), |
||||
label: Text('Camera'.i18n), |
||||
onPressed: () => |
||||
Navigator.pop(ctx, MediaPickSetting.camera), |
||||
), |
||||
if (onVideoPickCallback != null) |
||||
TextButton.icon( |
||||
icon: const Icon( |
||||
Icons.video_call, |
||||
color: Colors.cyanAccent, |
||||
), |
||||
label: Text('Video'.i18n), |
||||
onPressed: () => |
||||
Navigator.pop(ctx, MediaPickSetting.video), |
||||
) |
||||
], |
||||
), |
||||
), |
||||
); |
||||
|
||||
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: options.webVideoPickImpl, |
||||
); |
||||
break; |
||||
case MediaPickSetting.gallery: |
||||
throw ArgumentError( |
||||
'Invalid MediaSetting for the camera button.\n' |
||||
'gallery is not related to camera button', |
||||
); |
||||
case MediaPickSetting.link: |
||||
throw ArgumentError( |
||||
'Invalid MediaSetting for the camera button.\n' |
||||
'link is not related to camera button', |
||||
); |
||||
} |
||||
} |
||||
} |
@ -0,0 +1,105 @@ |
||||
import 'package:flutter/material.dart'; |
||||
import 'package:flutter_quill/flutter_quill.dart'; |
||||
|
||||
import '../../models/config/toolbar/buttons/formula.dart'; |
||||
|
||||
class QuillToolbarFormulaButton extends StatelessWidget { |
||||
const QuillToolbarFormulaButton({ |
||||
required this.controller, |
||||
required this.options, |
||||
super.key, |
||||
}); |
||||
|
||||
final QuillController controller; |
||||
final QuillToolbarFormulaButtonOptions options; |
||||
|
||||
double _iconSize(BuildContext context) { |
||||
final baseFontSize = baseButtonExtraOptions(context).globalIconSize; |
||||
final iconSize = options.iconSize; |
||||
return iconSize ?? baseFontSize; |
||||
} |
||||
|
||||
VoidCallback? _afterButtonPressed(BuildContext context) { |
||||
return options.afterButtonPressed ?? |
||||
baseButtonExtraOptions(context).afterButtonPressed; |
||||
} |
||||
|
||||
QuillIconTheme? _iconTheme(BuildContext context) { |
||||
return options.iconTheme ?? baseButtonExtraOptions(context).iconTheme; |
||||
} |
||||
|
||||
QuillToolbarBaseButtonOptions baseButtonExtraOptions(BuildContext context) { |
||||
return context.requireQuillToolbarBaseButtonOptions; |
||||
} |
||||
|
||||
IconData _iconData(BuildContext context) { |
||||
return options.iconData ?? |
||||
baseButtonExtraOptions(context).iconData ?? |
||||
Icons.functions; |
||||
} |
||||
|
||||
String _tooltip(BuildContext context) { |
||||
return options.tooltip ?? |
||||
baseButtonExtraOptions(context).tooltip ?? |
||||
'Insert formula'; |
||||
// ('Insert formula'.i18n); |
||||
} |
||||
|
||||
void _sharedOnPressed(BuildContext context) { |
||||
_onPressedHandler(context); |
||||
_afterButtonPressed(context); |
||||
} |
||||
|
||||
@override |
||||
Widget build(BuildContext context) { |
||||
final theme = Theme.of(context); |
||||
|
||||
final iconTheme = _iconTheme(context); |
||||
|
||||
final tooltip = _tooltip(context); |
||||
final iconSize = _iconSize(context); |
||||
final iconData = _iconData(context); |
||||
final childBuilder = |
||||
options.childBuilder ?? baseButtonExtraOptions(context).childBuilder; |
||||
|
||||
final iconColor = iconTheme?.iconUnselectedColor ?? theme.iconTheme.color; |
||||
final iconFillColor = iconTheme?.iconUnselectedFillColor ?? |
||||
(options.fillColor ?? theme.canvasColor); |
||||
|
||||
if (childBuilder != null) { |
||||
return childBuilder( |
||||
QuillToolbarFormulaButtonOptions( |
||||
afterButtonPressed: _afterButtonPressed(context), |
||||
fillColor: iconFillColor, |
||||
iconData: iconData, |
||||
iconSize: iconSize, |
||||
iconTheme: iconTheme, |
||||
tooltip: tooltip, |
||||
), |
||||
QuillToolbarFormulaButtonExtraOptions( |
||||
context: context, |
||||
controller: controller, |
||||
onPressed: () => _sharedOnPressed(context), |
||||
), |
||||
); |
||||
} |
||||
|
||||
return QuillToolbarIconButton( |
||||
icon: Icon(iconData, size: iconSize, color: iconColor), |
||||
tooltip: tooltip, |
||||
highlightElevation: 0, |
||||
hoverElevation: 0, |
||||
size: iconSize * 1.77, |
||||
fillColor: iconFillColor, |
||||
borderRadius: iconTheme?.borderRadius ?? 2, |
||||
onPressed: () => _sharedOnPressed(context), |
||||
); |
||||
} |
||||
|
||||
Future<void> _onPressedHandler(BuildContext context) async { |
||||
final index = controller.selection.baseOffset; |
||||
final length = controller.selection.extentOffset - index; |
||||
|
||||
controller.replaceText(index, length, BlockEmbed.formula(''), null); |
||||
} |
||||
} |
@ -0,0 +1,182 @@ |
||||
// ignore_for_file: use_build_context_synchronously |
||||
|
||||
import 'package:flutter/material.dart'; |
||||
import 'package:flutter_quill/flutter_quill.dart'; |
||||
import 'package:image_picker/image_picker.dart'; |
||||
|
||||
import '../../models/config/toolbar/buttons/image.dart'; |
||||
import '../embed_types.dart'; |
||||
import 'utils/image_video_utils.dart'; |
||||
|
||||
class QuillToolbarImageButton extends StatelessWidget { |
||||
const QuillToolbarImageButton({ |
||||
required this.controller, |
||||
required this.options, |
||||
super.key, |
||||
}); |
||||
|
||||
final QuillController controller; |
||||
|
||||
final QuillToolbarImageButtonOptions options; |
||||
|
||||
double _iconSize(BuildContext context) { |
||||
final baseFontSize = baseButtonExtraOptions(context).globalIconSize; |
||||
final iconSize = options.iconSize; |
||||
return iconSize ?? baseFontSize; |
||||
} |
||||
|
||||
VoidCallback? _afterButtonPressed(BuildContext context) { |
||||
return options.afterButtonPressed ?? |
||||
baseButtonExtraOptions(context).afterButtonPressed; |
||||
} |
||||
|
||||
QuillIconTheme? _iconTheme(BuildContext context) { |
||||
return options.iconTheme ?? baseButtonExtraOptions(context).iconTheme; |
||||
} |
||||
|
||||
QuillToolbarBaseButtonOptions baseButtonExtraOptions(BuildContext context) { |
||||
return context.requireQuillToolbarBaseButtonOptions; |
||||
} |
||||
|
||||
IconData _iconData(BuildContext context) { |
||||
return options.iconData ?? |
||||
baseButtonExtraOptions(context).iconData ?? |
||||
Icons.image; |
||||
} |
||||
|
||||
String _tooltip(BuildContext context) { |
||||
return options.tooltip ?? |
||||
baseButtonExtraOptions(context).tooltip ?? |
||||
'Insert image'; |
||||
// ('Insert Image'.i18n); |
||||
} |
||||
|
||||
void _sharedOnPressed(BuildContext context) { |
||||
_onPressedHandler(context); |
||||
_afterButtonPressed(context); |
||||
} |
||||
|
||||
@override |
||||
Widget build(BuildContext context) { |
||||
final tooltip = _tooltip(context); |
||||
final iconSize = _iconSize(context); |
||||
final iconData = _iconData(context); |
||||
final childBuilder = |
||||
options.childBuilder ?? baseButtonExtraOptions(context).childBuilder; |
||||
|
||||
if (childBuilder != null) { |
||||
return childBuilder( |
||||
QuillToolbarImageButtonOptions( |
||||
afterButtonPressed: _afterButtonPressed(context), |
||||
iconData: iconData, |
||||
iconSize: iconSize, |
||||
dialogTheme: options.dialogTheme, |
||||
filePickImpl: options.filePickImpl, |
||||
webImagePickImpl: options.webImagePickImpl, |
||||
fillColor: options.fillColor, |
||||
iconTheme: options.iconTheme, |
||||
linkRegExp: options.linkRegExp, |
||||
mediaPickSettingSelector: options.mediaPickSettingSelector, |
||||
onImagePickCallback: options.onImagePickCallback, |
||||
tooltip: options.tooltip, |
||||
), |
||||
QuillToolbarImageButtonExtraOptions( |
||||
context: context, |
||||
controller: controller, |
||||
onPressed: () => _sharedOnPressed(context), |
||||
), |
||||
); |
||||
} |
||||
|
||||
final theme = Theme.of(context); |
||||
|
||||
final iconTheme = _iconTheme(context); |
||||
|
||||
final iconColor = iconTheme?.iconUnselectedColor ?? theme.iconTheme.color; |
||||
final iconFillColor = iconTheme?.iconUnselectedFillColor ?? |
||||
(options.fillColor ?? theme.canvasColor); |
||||
|
||||
return QuillToolbarIconButton( |
||||
icon: Icon( |
||||
iconData, |
||||
size: iconSize, |
||||
color: iconColor, |
||||
), |
||||
tooltip: tooltip, |
||||
highlightElevation: 0, |
||||
hoverElevation: 0, |
||||
size: iconSize * 1.77, |
||||
fillColor: iconFillColor, |
||||
borderRadius: iconTheme?.borderRadius ?? 2, |
||||
onPressed: () => _sharedOnPressed(context), |
||||
); |
||||
} |
||||
|
||||
Future<void> _onPressedHandler(BuildContext context) async { |
||||
final onImagePickCallbackRef = options.onImagePickCallback; |
||||
if (onImagePickCallbackRef == null) { |
||||
await _typeLink(context); |
||||
return; |
||||
} |
||||
final selector = options.mediaPickSettingSelector ?? |
||||
ImageVideoUtils.selectMediaPickSetting; |
||||
final source = await selector(context); |
||||
if (source == null) { |
||||
return; |
||||
} |
||||
switch (source) { |
||||
case MediaPickSetting.gallery: |
||||
_pickImage(context); |
||||
break; |
||||
case MediaPickSetting.link: |
||||
await _typeLink(context); |
||||
break; |
||||
case MediaPickSetting.camera: |
||||
await ImageVideoUtils.handleImageButtonTap( |
||||
context, |
||||
controller, |
||||
ImageSource.camera, |
||||
onImagePickCallbackRef, |
||||
filePickImpl: options.filePickImpl, |
||||
webImagePickImpl: options.webImagePickImpl, |
||||
); |
||||
break; |
||||
case MediaPickSetting.video: |
||||
throw ArgumentError( |
||||
'Sorry but this is the Image button and not the video one', |
||||
); |
||||
} |
||||
} |
||||
|
||||
void _pickImage(BuildContext context) => ImageVideoUtils.handleImageButtonTap( |
||||
context, |
||||
controller, |
||||
ImageSource.gallery, |
||||
options.onImagePickCallback ?? |
||||
(throw ArgumentError( |
||||
'onImagePickCallback should not be null', |
||||
)), |
||||
filePickImpl: options.filePickImpl, |
||||
webImagePickImpl: options.webImagePickImpl, |
||||
); |
||||
|
||||
Future<void> _typeLink(BuildContext context) async { |
||||
final value = await showDialog<String>( |
||||
context: context, |
||||
builder: (_) => LinkDialog( |
||||
dialogTheme: options.dialogTheme, |
||||
linkRegExp: options.linkRegExp, |
||||
), |
||||
); |
||||
_linkSubmitted(value); |
||||
} |
||||
|
||||
void _linkSubmitted(String? value) { |
||||
if (value != null && value.isNotEmpty) { |
||||
final index = controller.selection.baseOffset; |
||||
final length = controller.selection.extentOffset - index; |
||||
|
||||
controller.replaceText(index, length, BlockEmbed.image(value), null); |
||||
} |
||||
} |
||||
} |
@ -0,0 +1,155 @@ |
||||
// ignore_for_file: use_build_context_synchronously |
||||
|
||||
import 'package:flutter/material.dart'; |
||||
import 'package:flutter_quill/flutter_quill.dart'; |
||||
import 'package:image_picker/image_picker.dart'; |
||||
|
||||
import '../../models/config/toolbar/buttons/video.dart'; |
||||
import '../embed_types.dart'; |
||||
import 'utils/image_video_utils.dart'; |
||||
|
||||
class QuillToolbarVideoButton extends StatelessWidget { |
||||
const QuillToolbarVideoButton({ |
||||
required this.options, |
||||
required this.controller, |
||||
super.key, |
||||
}); |
||||
|
||||
final QuillController controller; |
||||
|
||||
final QuillToolbarVideoButtonOptions options; |
||||
|
||||
double _iconSize(BuildContext context) { |
||||
final baseFontSize = baseButtonExtraOptions(context).globalIconSize; |
||||
final iconSize = options.iconSize; |
||||
return iconSize ?? baseFontSize; |
||||
} |
||||
|
||||
VoidCallback? _afterButtonPressed(BuildContext context) { |
||||
return options.afterButtonPressed ?? |
||||
baseButtonExtraOptions(context).afterButtonPressed; |
||||
} |
||||
|
||||
QuillIconTheme? _iconTheme(BuildContext context) { |
||||
return options.iconTheme ?? baseButtonExtraOptions(context).iconTheme; |
||||
} |
||||
|
||||
QuillToolbarBaseButtonOptions baseButtonExtraOptions(BuildContext context) { |
||||
return context.requireQuillToolbarBaseButtonOptions; |
||||
} |
||||
|
||||
IconData _iconData(BuildContext context) { |
||||
return options.iconData ?? |
||||
baseButtonExtraOptions(context).iconData ?? |
||||
Icons.movie_creation; |
||||
} |
||||
|
||||
String _tooltip(BuildContext context) { |
||||
return options.tooltip ?? |
||||
baseButtonExtraOptions(context).tooltip ?? |
||||
'Insert video'; |
||||
// ('Insert video'.i18n); |
||||
} |
||||
|
||||
void _sharedOnPressed(BuildContext context) { |
||||
_onPressedHandler(context); |
||||
_afterButtonPressed(context); |
||||
} |
||||
|
||||
@override |
||||
Widget build(BuildContext context) { |
||||
final theme = Theme.of(context); |
||||
|
||||
final iconTheme = _iconTheme(context); |
||||
|
||||
final tooltip = _tooltip(context); |
||||
final iconSize = _iconSize(context); |
||||
final iconData = _iconData(context); |
||||
final childBuilder = |
||||
options.childBuilder ?? baseButtonExtraOptions(context).childBuilder; |
||||
|
||||
final iconColor = iconTheme?.iconUnselectedColor ?? theme.iconTheme.color; |
||||
final iconFillColor = iconTheme?.iconUnselectedFillColor ?? |
||||
(options.fillColor ?? theme.canvasColor); |
||||
|
||||
if (childBuilder != null) { |
||||
return childBuilder( |
||||
QuillToolbarVideoButtonOptions( |
||||
afterButtonPressed: _afterButtonPressed(context), |
||||
iconData: iconData, |
||||
dialogTheme: options.dialogTheme, |
||||
filePickImpl: options.filePickImpl, |
||||
fillColor: iconFillColor, |
||||
iconSize: options.iconSize, |
||||
linkRegExp: options.linkRegExp, |
||||
tooltip: options.tooltip, |
||||
mediaPickSettingSelector: options.mediaPickSettingSelector, |
||||
iconTheme: options.iconTheme, |
||||
onVideoPickCallback: options.onVideoPickCallback, |
||||
webVideoPickImpl: options.webVideoPickImpl, |
||||
), |
||||
QuillToolbarVideoButtonExtraOptions( |
||||
context: context, |
||||
controller: controller, |
||||
onPressed: () => _sharedOnPressed(context), |
||||
), |
||||
); |
||||
} |
||||
|
||||
return QuillToolbarIconButton( |
||||
icon: Icon(iconData, size: iconSize, color: iconColor), |
||||
tooltip: tooltip, |
||||
highlightElevation: 0, |
||||
hoverElevation: 0, |
||||
size: iconSize * 1.77, |
||||
fillColor: iconFillColor, |
||||
borderRadius: iconTheme?.borderRadius ?? 2, |
||||
onPressed: () => _sharedOnPressed(context), |
||||
); |
||||
} |
||||
|
||||
Future<void> _onPressedHandler(BuildContext context) async { |
||||
if (options.onVideoPickCallback != null) { |
||||
final selector = options.mediaPickSettingSelector ?? |
||||
ImageVideoUtils.selectMediaPickSetting; |
||||
final source = await selector(context); |
||||
if (source != null) { |
||||
if (source == MediaPickSetting.gallery) { |
||||
_pickVideo(context); |
||||
} else { |
||||
await _typeLink(context); |
||||
} |
||||
} |
||||
} else { |
||||
await _typeLink(context); |
||||
} |
||||
} |
||||
|
||||
void _pickVideo(BuildContext context) => ImageVideoUtils.handleVideoButtonTap( |
||||
context, |
||||
controller, |
||||
ImageSource.gallery, |
||||
options.onVideoPickCallback!, |
||||
filePickImpl: options.filePickImpl, |
||||
webVideoPickImpl: options.webVideoPickImpl, |
||||
); |
||||
|
||||
Future<void> _typeLink(BuildContext context) async { |
||||
final value = await showDialog<String>( |
||||
context: context, |
||||
builder: (_) => LinkDialog( |
||||
dialogTheme: options.dialogTheme, |
||||
), |
||||
); |
||||
_linkSubmitted(value); |
||||
} |
||||
|
||||
void _linkSubmitted(String? value) { |
||||
if (value != null && value.isNotEmpty) { |
||||
final index = controller.selection.baseOffset; |
||||
final length = controller.selection.extentOffset - index; |
||||
|
||||
controller.replaceText(index, length, BlockEmbed.video(value), null); |
||||
} |
||||
} |
||||
} |
@ -0,0 +1,87 @@ |
||||
import 'dart:io' show File; |
||||
|
||||
import 'package:flutter/foundation.dart' show immutable; |
||||
import '../../logic/services/s_image_saver.dart'; |
||||
|
||||
// I would like to orgnize the project structure and the code more |
||||
// but here I don't want to change too much since that is a community project |
||||
|
||||
RegExp _base64 = RegExp( |
||||
r'^(?:[A-Za-z0-9+\/]{4})*(?:[A-Za-z0-9+\/]{2}==|[A-Za-z0-9+\/]{3}=|[A-Za-z0-9+\/]{4})$', |
||||
); |
||||
|
||||
bool isBase64(String str) { |
||||
return _base64.hasMatch(str); |
||||
} |
||||
|
||||
bool isHttpBasedUrl(String url) { |
||||
try { |
||||
final uri = Uri.parse(url.trim()); |
||||
return uri.isScheme('HTTP') || uri.isScheme('HTTPS'); |
||||
} catch (_) { |
||||
return false; |
||||
} |
||||
} |
||||
|
||||
bool isYouTubeUrl(String videoUrl) { |
||||
try { |
||||
final uri = Uri.parse(videoUrl); |
||||
return uri.host == 'www.youtube.com' || |
||||
uri.host == 'youtube.com' || |
||||
uri.host == 'youtu.be'; |
||||
} catch (_) { |
||||
return false; |
||||
} |
||||
} |
||||
|
||||
bool isImageBase64(String imageUrl) { |
||||
return !isHttpBasedUrl(imageUrl) && isBase64(imageUrl); |
||||
} |
||||
|
||||
enum SaveImageResultMethod { network, localStorage } |
||||
|
||||
@immutable |
||||
class SaveImageResult { |
||||
const SaveImageResult({required this.isSuccess, required this.method}); |
||||
|
||||
final bool isSuccess; |
||||
final SaveImageResultMethod method; |
||||
} |
||||
|
||||
Future<SaveImageResult> saveImage(String imageUrl) async { |
||||
final imageSaverService = ImageSaverService.getInstance(); |
||||
final imageFile = File(imageUrl); |
||||
final hasPermission = await imageSaverService.hasAccess(); |
||||
final imageExistsLocally = await imageFile.exists(); |
||||
if (!hasPermission) { |
||||
await imageSaverService.requestAccess(); |
||||
} |
||||
if (!imageExistsLocally) { |
||||
try { |
||||
await imageSaverService.saveImageFromNetwork( |
||||
Uri.parse(imageUrl), |
||||
); |
||||
return const SaveImageResult( |
||||
isSuccess: true, |
||||
method: SaveImageResultMethod.network, |
||||
); |
||||
} catch (e) { |
||||
return const SaveImageResult( |
||||
isSuccess: false, |
||||
method: SaveImageResultMethod.network, |
||||
); |
||||
} |
||||
} |
||||
try { |
||||
await imageSaverService.saveLocalImage(imageUrl); |
||||
return const SaveImageResult( |
||||
isSuccess: true, |
||||
method: SaveImageResultMethod.localStorage, |
||||
); |
||||
} catch (e) { |
||||
return const SaveImageResult( |
||||
isSuccess: false, |
||||
method: SaveImageResultMethod.localStorage, |
||||
); |
||||
} |
||||
} |
@ -0,0 +1,33 @@ |
||||
import 'package:flutter/material.dart'; |
||||
|
||||
class SimpleDialogItem extends StatelessWidget { |
||||
const SimpleDialogItem({ |
||||
required this.icon, |
||||
required this.color, |
||||
required this.text, |
||||
required this.onPressed, |
||||
super.key, |
||||
}); |
||||
|
||||
final IconData icon; |
||||
final Color color; |
||||
final String text; |
||||
final VoidCallback onPressed; |
||||
|
||||
@override |
||||
Widget build(BuildContext context) { |
||||
return SimpleDialogOption( |
||||
onPressed: onPressed, |
||||
child: Row( |
||||
children: [ |
||||
Icon(icon, size: 36, color: color), |
||||
Padding( |
||||
padding: const EdgeInsetsDirectional.only(start: 16), |
||||
child: |
||||
Text(text, style: const TextStyle(fontWeight: FontWeight.bold)), |
||||
), |
||||
], |
||||
), |
||||
); |
||||
} |
||||
} |
@ -0,0 +1,136 @@ |
||||
import 'package:flutter_quill/extensions.dart'; |
||||
import 'package:meta/meta.dart' show immutable; |
||||
|
||||
import '../../../embeds/embed_types.dart'; |
||||
|
||||
/// [QuillEditorImageEmbedConfigurations] for desktop, mobile and |
||||
/// other platforms |
||||
/// excluding web, it's configurations that is needed for the editor |
||||
/// |
||||
@immutable |
||||
class QuillEditorImageEmbedConfigurations { |
||||
const QuillEditorImageEmbedConfigurations({ |
||||
this.forceUseMobileOptionMenuForImageClick = false, |
||||
this.onImageRemovedCallback, |
||||
this.shouldRemoveImageCallback, |
||||
this.imageProviderBuilder, |
||||
this.imageErrorWidgetBuilder, |
||||
}); |
||||
|
||||
/// [onImageRemovedCallback] is called when an image is |
||||
/// removed from the editor. |
||||
/// By default, [onImageRemovedCallback] deletes the |
||||
/// temporary image file if |
||||
/// the platform is mobile and if it still exists. You |
||||
/// can customize this behavior |
||||
/// by passing your own function that handles the removal process. |
||||
/// |
||||
/// Example of [onImageRemovedCallback] customization: |
||||
/// ```dart |
||||
/// afterRemoveImageFromEditor: (imageFile) async { |
||||
/// // Your custom logic here |
||||
/// // or leave it empty to do nothing |
||||
/// } |
||||
/// ``` |
||||
/// |
||||
final ImageEmbedBuilderOnRemovedCallback? onImageRemovedCallback; |
||||
|
||||
/// [shouldRemoveImageCallback] is a callback |
||||
/// function that is invoked when the |
||||
/// user attempts to remove an image from the editor. It allows you to control |
||||
/// whether the image should be removed based on your custom logic. |
||||
/// |
||||
/// Example of [shouldRemoveImageCallback] customization: |
||||
/// ```dart |
||||
/// shouldRemoveImageFromEditor: (imageFile) async { |
||||
/// // Show a confirmation dialog before removing the image |
||||
/// final isShouldRemove = await showYesCancelDialog( |
||||
/// context: context, |
||||
/// options: const YesOrCancelDialogOptions( |
||||
/// title: 'Deleting an image', |
||||
/// message: 'Are you sure you want' ' to delete this |
||||
/// image from the editor?', |
||||
/// ), |
||||
/// ); |
||||
/// |
||||
/// // Return `true` to allow image removal if the user confirms, otherwise |
||||
/// `false` |
||||
/// return isShouldRemove; |
||||
/// } |
||||
/// ``` |
||||
/// |
||||
final ImageEmbedBuilderWillRemoveCallback? shouldRemoveImageCallback; |
||||
|
||||
/// [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); |
||||
/// } |
||||
/// ``` |
||||
/// |
||||
final ImageEmbedBuilderProviderBuilder? imageProviderBuilder; |
||||
|
||||
/// [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 |
||||
/// |
||||
final ImageEmbedBuilderErrorWidgetBuilder? imageErrorWidgetBuilder; |
||||
|
||||
/// [forceUseMobileOptionMenuForImageClick] is a boolean |
||||
/// flag that, when set to `true`, |
||||
/// enforces the use of the mobile-specific option menu for image clicks in |
||||
/// other platforms like desktop, this option doesn't affect mobile. it will |
||||
/// not affect web |
||||
/// This option |
||||
/// can be used to override the default behavior based on the platform. |
||||
/// |
||||
final bool forceUseMobileOptionMenuForImageClick; |
||||
|
||||
static ImageEmbedBuilderOnRemovedCallback get defaultOnImageRemovedCallback { |
||||
return (imageFile) async { |
||||
final mobile = isMobile(); |
||||
// If the platform is not mobile, return void; |
||||
// Since the mobile OS gives us a copy of the image |
||||
|
||||
// Note: We should remove the image on Flutter web |
||||
// since the behavior is similar to how it is on mobile, |
||||
// but since this builder is not for web, we will ignore it |
||||
if (!mobile) { |
||||
return; |
||||
} |
||||
|
||||
// On mobile OS (Android, iOS), the system will not give us |
||||
// direct access to the image; instead, |
||||
// it will give us the image |
||||
// in the temp directory of the application. So, we want to |
||||
// remove it when we no longer need it. |
||||
|
||||
// but on desktop we don't want to touch user files |
||||
// especially on macOS, where we can't even delete |
||||
// it without |
||||
// permission |
||||
|
||||
final isFileExists = await imageFile.exists(); |
||||
if (isFileExists) { |
||||
await imageFile.delete(); |
||||
} |
||||
}; |
||||
} |
||||
} |
||||
|
||||
@immutable |
||||
class QuillEditorWebImageEmbedConfigurations { |
||||
const QuillEditorWebImageEmbedConfigurations(); |
||||
} |
@ -0,0 +1,24 @@ |
||||
import 'package:flutter/widgets.dart' show GlobalKey; |
||||
import 'package:meta/meta.dart' show immutable; |
||||
|
||||
@immutable |
||||
class QuillEditorVideoEmbedConfigurations { |
||||
const QuillEditorVideoEmbedConfigurations({ |
||||
this.onVideoInit, |
||||
}); |
||||
|
||||
/// [onVideoInit] is a callback function that gets triggered when |
||||
/// a video is initialized. |
||||
/// You can use this to perform actions or setup configurations related |
||||
/// to video embedding. |
||||
/// |
||||
/// |
||||
/// Example usage: |
||||
/// ```dart |
||||
/// onVideoInit: (videoContainerKey) { |
||||
/// // Custom video initialization logic |
||||
/// }, |
||||
/// // Customize other callback functions as needed |
||||
/// ``` |
||||
final void Function(GlobalKey videoContainerKey)? onVideoInit; |
||||
} |
@ -0,0 +1,6 @@ |
||||
import 'package:meta/meta.dart' show immutable; |
||||
|
||||
@immutable |
||||
class QuillEditorWebViewEmbedConfigurations { |
||||
const QuillEditorWebViewEmbedConfigurations(); |
||||
} |
@ -0,0 +1,49 @@ |
||||
import 'package:flutter/widgets.dart' show Color; |
||||
import 'package:flutter_quill/flutter_quill.dart'; |
||||
|
||||
import '../../../../embeds/embed_types.dart'; |
||||
|
||||
class QuillToolbarCameraButtonExtraOptions |
||||
extends QuillToolbarBaseButtonExtraOptions { |
||||
const QuillToolbarCameraButtonExtraOptions({ |
||||
required super.controller, |
||||
required super.context, |
||||
required super.onPressed, |
||||
}); |
||||
} |
||||
|
||||
class QuillToolbarCameraButtonOptions extends QuillToolbarBaseButtonOptions< |
||||
QuillToolbarCameraButtonOptions, QuillToolbarCameraButtonExtraOptions> { |
||||
const QuillToolbarCameraButtonOptions({ |
||||
required this.onImagePickCallback, |
||||
required this.onVideoPickCallback, |
||||
this.webImagePickImpl, |
||||
this.webVideoPickImpl, |
||||
this.filePickImpl, |
||||
this.cameraPickSettingSelector, |
||||
this.iconSize, |
||||
this.fillColor, |
||||
super.iconData, |
||||
super.afterButtonPressed, |
||||
super.tooltip, |
||||
super.iconTheme, |
||||
super.childBuilder, |
||||
super.controller, |
||||
}); |
||||
|
||||
final OnImagePickCallback onImagePickCallback; |
||||
|
||||
final OnVideoPickCallback onVideoPickCallback; |
||||
|
||||
final WebImagePickImpl? webImagePickImpl; |
||||
|
||||
final WebVideoPickImpl? webVideoPickImpl; |
||||
|
||||
final FilePickImpl? filePickImpl; |
||||
|
||||
final MediaPickSettingSelector? cameraPickSettingSelector; |
||||
|
||||
final double? iconSize; |
||||
|
||||
final Color? fillColor; |
||||
} |
@ -0,0 +1,28 @@ |
||||
import 'package:flutter/widgets.dart' show Color; |
||||
import 'package:flutter_quill/flutter_quill.dart'; |
||||
|
||||
class QuillToolbarFormulaButtonExtraOptions |
||||
extends QuillToolbarBaseButtonExtraOptions { |
||||
const QuillToolbarFormulaButtonExtraOptions({ |
||||
required super.controller, |
||||
required super.context, |
||||
required super.onPressed, |
||||
}); |
||||
} |
||||
|
||||
class QuillToolbarFormulaButtonOptions extends QuillToolbarBaseButtonOptions< |
||||
QuillToolbarFormulaButtonOptions, QuillToolbarFormulaButtonExtraOptions> { |
||||
const QuillToolbarFormulaButtonOptions({ |
||||
super.tooltip, |
||||
super.iconData, |
||||
super.iconTheme, |
||||
super.afterButtonPressed, |
||||
super.childBuilder, |
||||
this.fillColor, |
||||
this.iconSize, |
||||
}); |
||||
|
||||
final Color? fillColor; |
||||
|
||||
final double? iconSize; |
||||
} |
@ -0,0 +1,53 @@ |
||||
import 'package:flutter/widgets.dart' show Color; |
||||
import 'package:flutter_quill/flutter_quill.dart'; |
||||
import 'package:meta/meta.dart' show immutable; |
||||
|
||||
import '../../../../embeds/embed_types.dart'; |
||||
|
||||
class QuillToolbarImageButtonExtraOptions |
||||
extends QuillToolbarBaseButtonExtraOptions { |
||||
const QuillToolbarImageButtonExtraOptions({ |
||||
required super.controller, |
||||
required super.context, |
||||
required super.onPressed, |
||||
}); |
||||
} |
||||
|
||||
@immutable |
||||
class QuillToolbarImageButtonOptions extends QuillToolbarBaseButtonOptions< |
||||
QuillToolbarImageButtonOptions, QuillToolbarImageButtonExtraOptions> { |
||||
const QuillToolbarImageButtonOptions({ |
||||
super.iconData, |
||||
super.controller, |
||||
this.iconSize, |
||||
|
||||
/// specifies the tooltip text for the image button. |
||||
super.tooltip, |
||||
super.afterButtonPressed, |
||||
super.childBuilder, |
||||
super.iconTheme, |
||||
this.fillColor, |
||||
this.onImagePickCallback, |
||||
this.filePickImpl, |
||||
this.webImagePickImpl, |
||||
this.mediaPickSettingSelector, |
||||
this.dialogTheme, |
||||
this.linkRegExp, |
||||
}); |
||||
|
||||
final double? iconSize; |
||||
final Color? fillColor; |
||||
|
||||
final OnImagePickCallback? onImagePickCallback; |
||||
|
||||
final WebImagePickImpl? webImagePickImpl; |
||||
|
||||
final FilePickImpl? filePickImpl; |
||||
|
||||
final MediaPickSettingSelector? mediaPickSettingSelector; |
||||
|
||||
final QuillDialogTheme? dialogTheme; |
||||
|
||||
/// [imageLinkRegExp] is a regular expression to identify image links. |
||||
final RegExp? linkRegExp; |
||||
} |
@ -0,0 +1,84 @@ |
||||
import 'package:flutter/widgets.dart' show AutovalidateMode; |
||||
import 'package:flutter/widgets.dart' show Color, Size; |
||||
import 'package:flutter_quill/flutter_quill.dart'; |
||||
|
||||
import '../../../../embeds/embed_types.dart'; |
||||
|
||||
class QuillToolbarMediaButtonExtraOptions |
||||
extends QuillToolbarBaseButtonExtraOptions { |
||||
const QuillToolbarMediaButtonExtraOptions({ |
||||
required super.controller, |
||||
required super.context, |
||||
required super.onPressed, |
||||
}); |
||||
} |
||||
|
||||
class QuillToolbarMediaButtonOptions extends QuillToolbarBaseButtonOptions< |
||||
QuillToolbarMediaButtonOptions, QuillToolbarMediaButtonExtraOptions> { |
||||
const QuillToolbarMediaButtonOptions({ |
||||
required this.type, |
||||
required this.onMediaPickedCallback, |
||||
required this.onImagePickCallback, |
||||
required this.onVideoPickCallback, |
||||
this.dialogBarrierColor, |
||||
this.mediaFilePicker, |
||||
this.childrenSpacing = 16.0, |
||||
this.autovalidateMode = AutovalidateMode.disabled, |
||||
this.iconSize, |
||||
this.fillColor, |
||||
this.dialogTheme, |
||||
this.labelText, |
||||
this.hintText, |
||||
this.submitButtonText, |
||||
this.submitButtonSize, |
||||
this.galleryButtonText, |
||||
this.linkButtonText, |
||||
this.validationMessage, |
||||
this.filePickImpl, |
||||
this.webImagePickImpl, |
||||
this.webVideoPickImpl, |
||||
super.iconData, |
||||
super.afterButtonPressed, |
||||
super.tooltip, |
||||
super.iconTheme, |
||||
super.childBuilder, |
||||
super.controller, |
||||
}); |
||||
|
||||
final double? iconSize; |
||||
final Color? fillColor; |
||||
final QuillMediaType type; |
||||
final QuillDialogTheme? dialogTheme; |
||||
final MediaFilePicker? mediaFilePicker; |
||||
final MediaPickedCallback? onMediaPickedCallback; |
||||
final Color? dialogBarrierColor; |
||||
|
||||
/// The margin between child widgets in the dialog. |
||||
final double childrenSpacing; |
||||
|
||||
/// The text of label in link add mode. |
||||
final String? labelText; |
||||
|
||||
/// The hint text for link [TextField]. |
||||
final String? hintText; |
||||
|
||||
/// The text of the submit button. |
||||
final String? submitButtonText; |
||||
|
||||
/// The size of dialog buttons. |
||||
final Size? submitButtonSize; |
||||
|
||||
/// The text of the gallery button [MediaSourceSelectorDialog]. |
||||
final String? galleryButtonText; |
||||
|
||||
/// The text of the link button [MediaSourceSelectorDialog]. |
||||
final String? linkButtonText; |
||||
|
||||
final AutovalidateMode autovalidateMode; |
||||
final String? validationMessage; |
||||
final OnImagePickCallback onImagePickCallback; |
||||
final FilePickImpl? filePickImpl; |
||||
final WebImagePickImpl? webImagePickImpl; |
||||
final OnVideoPickCallback onVideoPickCallback; |
||||
final WebVideoPickImpl? webVideoPickImpl; |
||||
} |
@ -0,0 +1,47 @@ |
||||
import 'package:flutter/widgets.dart' show Color; |
||||
import 'package:flutter_quill/flutter_quill.dart'; |
||||
|
||||
import '../../../../embeds/embed_types.dart'; |
||||
|
||||
class QuillToolbarVideoButtonExtraOptions |
||||
extends QuillToolbarBaseButtonExtraOptions { |
||||
const QuillToolbarVideoButtonExtraOptions({ |
||||
required super.controller, |
||||
required super.context, |
||||
required super.onPressed, |
||||
}); |
||||
} |
||||
|
||||
class QuillToolbarVideoButtonOptions extends QuillToolbarBaseButtonOptions< |
||||
QuillToolbarVideoButtonOptions, QuillToolbarVideoButtonExtraOptions> { |
||||
const QuillToolbarVideoButtonOptions({ |
||||
this.linkRegExp, |
||||
this.dialogTheme, |
||||
this.onVideoPickCallback, |
||||
this.webVideoPickImpl, |
||||
this.filePickImpl, |
||||
this.mediaPickSettingSelector, |
||||
this.fillColor, |
||||
this.iconSize, |
||||
super.iconData, |
||||
super.afterButtonPressed, |
||||
super.tooltip, |
||||
super.iconTheme, |
||||
super.childBuilder, |
||||
super.controller, |
||||
}); |
||||
|
||||
final RegExp? linkRegExp; |
||||
final QuillDialogTheme? dialogTheme; |
||||
final OnVideoPickCallback? onVideoPickCallback; |
||||
|
||||
final WebVideoPickImpl? webVideoPickImpl; |
||||
|
||||
final FilePickImpl? filePickImpl; |
||||
|
||||
final MediaPickSettingSelector? mediaPickSettingSelector; |
||||
|
||||
final Color? fillColor; |
||||
|
||||
final double? iconSize; |
||||
} |
@ -1,23 +0,0 @@ |
||||
// ignore_for_file: avoid_classes_with_only_static_members, camel_case_types, lines_longer_than_80_chars |
||||
|
||||
import 'package:universal_html/html.dart' as html; |
||||
|
||||
// Fake interface for the logic that this package needs from (web-only) dart:ui. |
||||
// This is conditionally exported so the analyzer sees these methods as available. |
||||
|
||||
typedef PlatroformViewFactory = html.Element Function(int viewId); |
||||
|
||||
/// Shim for web_ui engine.PlatformViewRegistry |
||||
/// https://github.com/flutter/engine/blob/master/lib/web_ui/lib/ui.dart#L62 |
||||
class platformViewRegistry { |
||||
/// Shim for registerViewFactory |
||||
/// https://github.com/flutter/engine/blob/master/lib/web_ui/lib/ui.dart#L72 |
||||
static dynamic registerViewFactory( |
||||
String viewTypeId, PlatroformViewFactory viewFactory) {} |
||||
} |
||||
|
||||
/// Shim for web_ui engine.AssetManager |
||||
/// https://github.com/flutter/engine/blob/master/lib/web_ui/lib/src/engine/assets.dart#L12 |
||||
class webOnlyAssetManager { |
||||
static dynamic getAssetUrl(String asset) {} |
||||
} |
Loading…
Reference in new issue