Update extensions and example

pull/1508/head
Ellet 1 year ago
parent b9b1ecb2bd
commit f0d011007e
No known key found for this signature in database
GPG Key ID: C488CC70BBCEF0D1
  1. 3
      CHANGELOG.md
  2. 32
      README.md
  3. 2
      example/assets/sample_data.json
  4. 10
      example/assets/sample_data_nomedia.json
  5. 97
      example/assets/sample_data_testing.json
  6. 11
      example/lib/pages/home_page.dart
  7. 195
      example/lib/universal_ui/universal_ui.dart
  8. 4
      flutter_quill_extensions/CHANGELOG.md
  9. 37
      flutter_quill_extensions/README.md
  10. 32
      flutter_quill_extensions/lib/flutter_quill_extensions.dart
  11. 6
      flutter_quill_extensions/lib/presentation/embeds/editor/image/image.dart
  12. 111
      flutter_quill_extensions/lib/presentation/embeds/editor/image/image_web.dart
  13. 0
      flutter_quill_extensions/lib/presentation/embeds/editor/shims/dart_ui_fake.dart
  14. 0
      flutter_quill_extensions/lib/presentation/embeds/editor/shims/dart_ui_real.dart
  15. 8
      flutter_quill_extensions/lib/presentation/embeds/editor/video/video.dart
  16. 57
      flutter_quill_extensions/lib/presentation/embeds/editor/video/video_web.dart
  17. 3
      flutter_quill_extensions/lib/presentation/embeds/utils.dart
  18. 22
      flutter_quill_extensions/lib/presentation/embeds/widgets/image.dart
  19. 7
      flutter_quill_extensions/lib/presentation/models/config/editor/image/image.dart
  20. 11
      flutter_quill_extensions/lib/presentation/models/config/editor/image/image_web.dart
  21. 0
      flutter_quill_extensions/lib/presentation/models/config/editor/video/video.dart
  22. 11
      flutter_quill_extensions/lib/presentation/models/config/editor/video/video_web.dart
  23. 8
      flutter_quill_extensions/pubspec.yaml
  24. 14
      lib/src/models/documents/nodes/node.dart
  25. 2
      pubspec.yaml

@ -1,3 +1,6 @@
## [8.2.3]
- Update `README.md`
## [8.2.2]
- Move the `flutter_quill_test` to seperated package [flutter_quill_test](https://pub.dev/packages/flutter_quill_test)

@ -44,8 +44,6 @@ Pub: [FlutterQuill]
- [Custom Buttons](#custom-buttons)
- [Embed Blocks](#embed-blocks)
- [Using the embed blocks from `flutter_quill_extensions`](#using-the-embed-blocks-from-flutter_quill_extensions)
- [Custom Size Image for Mobile](#custom-size-image-for-mobile)
- [Custom Size Image for other platforms (excluding web)](#custom-size-image-for-other-platforms-excluding-web)
- [Custom Embed Blocks](#custom-embed-blocks)
- [Custom Toolbar](#custom-toolbar)
- [Translation](#translation)
@ -253,36 +251,6 @@ Provide a list of embed
To see how to use the extensions package, please take a look at the [README](./flutter_quill_extensions/README.md) of [FlutterQuill Extensions]
### Custom Size Image for Mobile
Define `mobileWidth`, `mobileHeight`, `mobileMargin`, `mobileAlignment` as follows:
```json
{
"insert": {
"image": "https://user-images.githubusercontent.com/122956/72955931-ccc07900-3d52-11ea-89b1-d468a6e2aa2b.png"
},
"attributes":{
"style":"mobileWidth: 50; mobileHeight: 50; mobileMargin: 10; mobileAlignment: topLeft"
}
}
```
### Custom Size Image for other platforms (excluding web)
Define `width`, `height`, `margin`, `alignment` as follows:
```json
{
"insert": {
"image": "https://user-images.githubusercontent.com/122956/72955931-ccc07900-3d52-11ea-89b1-d468a6e2aa2b.png"
},
"attributes":{
"style":"width: 50; height: 50; margin: 10; alignment: topLeft"
}
}
```
### Custom Embed Blocks
Sometimes you want to add some custom content inside your text, custom widgets inside of them. An example is adding notes to the text, or anything custom that you want to add in your text editor.

@ -531,7 +531,7 @@
},
"attributes": {
"width": "230",
"style": "display: block; margin: auto;"
"style": "display: block; margin: auto; width: 500px;"
}
},
{

@ -63,16 +63,6 @@
},
"insert": " and "
},
{
"attributes": {
"strike": true,
"color": "black"
},
"insert": "web"
},
{
"insert": " is not supported.\nYou are welcome to use "
},
{
"attributes": {
"link": "https://bulletjournal.us/home/index.html"

File diff suppressed because one or more lines are too long

@ -53,16 +53,19 @@ class _HomePageState extends State<HomePage> {
Future<void> _loadFromAssets() async {
try {
final result = await rootBundle.loadString(isDesktop()
? 'assets/sample_data_nomedia.json'
: 'assets/sample_data.json');
final result =
await rootBundle.loadString('assets/sample_data_testing.json');
// final result = await rootBundle.loadString(isDesktop()
// ? 'assets/sample_data_nomedia.json'
// : 'assets/sample_data.json');
final doc = Document.fromJson(jsonDecode(result));
_controller = QuillController(
document: doc,
selection: const TextSelection.collapsed(offset: 0),
);
} catch (error) {
final doc = Document()..insert(0, 'Empty asset');
final doc = Document()
..insert(0, 'Error while loading the document: ${error.toString()}');
_controller = QuillController(
document: doc,
selection: const TextSelection.collapsed(offset: 0),

@ -1,118 +1,115 @@
library universal_ui;
import 'package:flutter/cupertino.dart';
import 'package:flutter/foundation.dart';
import 'package:flutter_quill/flutter_quill.dart';
import 'package:flutter_quill_extensions/flutter_quill_extensions.dart';
import 'package:universal_html/html.dart' as html;
import 'package:youtube_player_flutter/youtube_player_flutter.dart';
import '../widgets/responsive_widget.dart';
import 'fake_ui.dart' if (dart.library.html) 'real_ui.dart' as ui_instance;
// import '../widgets/responsive_widget.dart';
// import 'fake_ui.dart' if (dart.library.html) 'real_ui.dart' as ui_instance;
class PlatformViewRegistryFix {
void registerViewFactory(dynamic x, dynamic y) {
if (kIsWeb) {
ui_instance.PlatformViewRegistry.registerViewFactory(
x,
y,
);
}
}
}
// class PlatformViewRegistryFix {
// void registerViewFactory(dynamic x, dynamic y) {
// if (kIsWeb) {
// ui_instance.PlatformViewRegistry.registerViewFactory(
// x,
// y,
// );
// }
// }
// }
class UniversalUI {
PlatformViewRegistryFix platformViewRegistry = PlatformViewRegistryFix();
}
// class UniversalUI {
// PlatformViewRegistryFix platformViewRegistry = PlatformViewRegistryFix();
// }
var ui = UniversalUI();
// var ui = UniversalUI();
class ImageEmbedBuilderWeb extends EmbedBuilder {
@override
String get key => BlockEmbed.imageType;
// class ImageEmbedBuilderWeb extends EmbedBuilder {
// @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;
if (isImageBase64(imageUrl)) {
// TODO: handle imageUrl of base64
return const SizedBox();
}
final size = MediaQuery.sizeOf(context);
UniversalUI().platformViewRegistry.registerViewFactory(imageUrl, (viewId) {
return html.ImageElement()
..src = imageUrl
..style.height = 'auto'
..style.width = 'auto';
});
return Padding(
padding: EdgeInsets.only(
right: ResponsiveWidget.isMediumScreen(context)
? size.width * 0.5
: (ResponsiveWidget.isLargeScreen(context))
? size.width * 0.75
: size.width * 0.2,
),
child: SizedBox(
height: size.height * 0.45,
child: HtmlElementView(
viewType: imageUrl,
),
),
);
}
}
// @override
// Widget build(
// BuildContext context,
// QuillController controller,
// Embed node,
// bool readOnly,
// bool inline,
// TextStyle textStyle,
// ) {
// final imageUrl = node.value.data;
// if (isImageBase64(imageUrl)) {
// // TODO: handle imageUrl of base64
// return const SizedBox();
// }
// final size = MediaQuery.sizeOf(context);
// UniversalUI().platformViewRegistry.registerViewFactory(imageUrl,
//(viewId) {
// return html.ImageElement()
// ..src = imageUrl
// ..style.height = 'auto'
// ..style.width = 'auto';
// });
// return Padding(
// padding: EdgeInsets.only(
// right: ResponsiveWidget.isMediumScreen(context)
// ? size.width * 0.5
// : (ResponsiveWidget.isLargeScreen(context))
// ? size.width * 0.75
// : size.width * 0.2,
// ),
// child: SizedBox(
// height: size.height * 0.45,
// child: HtmlElementView(
// viewType: imageUrl,
// ),
// ),
// );
// }
// }
class VideoEmbedBuilderWeb extends EmbedBuilder {
@override
String get key => BlockEmbed.videoType;
// class VideoEmbedBuilderWeb extends EmbedBuilder {
// @override
// String get key => BlockEmbed.videoType;
@override
Widget build(
BuildContext context,
QuillController controller,
Embed node,
bool readOnly,
bool inline,
TextStyle textStyle,
) {
var videoUrl = node.value.data;
if (videoUrl.contains('youtube.com') || videoUrl.contains('youtu.be')) {
final youtubeID = YoutubePlayer.convertUrlToId(videoUrl);
if (youtubeID != null) {
videoUrl = 'https://www.youtube.com/embed/$youtubeID';
}
}
// @override
// Widget build(
// BuildContext context,
// QuillController controller,
// Embed node,
// bool readOnly,
// bool inline,
// TextStyle textStyle,
// ) {
// var videoUrl = node.value.data;
// if (videoUrl.contains('youtube.com') || videoUrl.contains('youtu.be')) {
// final youtubeID = YoutubePlayer.convertUrlToId(videoUrl);
// if (youtubeID != null) {
// videoUrl = 'https://www.youtube.com/embed/$youtubeID';
// }
// }
final size = MediaQuery.sizeOf(context);
// final size = MediaQuery.sizeOf(context);
UniversalUI().platformViewRegistry.registerViewFactory(
videoUrl,
(id) => html.IFrameElement()
..width = size.width.toString()
..height = size.height.toString()
..src = videoUrl
..style.border = 'none',
);
// UniversalUI().platformViewRegistry.registerViewFactory(
// videoUrl,
// (id) => html.IFrameElement()
// ..width = size.width.toString()
// ..height = size.height.toString()
// ..src = videoUrl
// ..style.border = 'none',
// );
return SizedBox(
height: 500,
child: HtmlElementView(
viewType: videoUrl,
),
);
}
}
// return SizedBox(
// height: 500,
// child: HtmlElementView(
// viewType: videoUrl,
// ),
// );
// }
// }
List<EmbedBuilder> get defaultEmbedBuildersWeb => [
...FlutterQuillEmbeds.editorsWebBuilders(),
// ImageEmbedBuilderWeb(),
const QuillEditorWebImageEmbedBuilder(),
VideoEmbedBuilderWeb(),
// VideoEmbedBuilderWeb(),
];

@ -1,3 +1,7 @@
## 0.6.0-dev.6
- Better support for web
- Smal fixes and updates
## 0.6.0-dev.5
- Update the camera button

@ -15,6 +15,9 @@ Currently the support for **Web** is limitied.
- [Installation](#installation)
- [Platform Spesefic Configurations](#platform-spesefic-configurations)
- [Usage](#usage)
- [Embed Blocks](#embed-blocks)
- [Custom Size Image for Mobile](#custom-size-image-for-mobile)
- [Custom Size Image for other platforms (excluding web)](#custom-size-image-for-other-platforms-excluding-web)
- [Features](#features)
- [Contributing](#contributing)
- [License](#license)
@ -110,6 +113,40 @@ QuillProvider(
)
```
## Embed Blocks
As of version [flutter_quill](https://pub.dev/packages/flutter_quill) 6.0, embed blocks are not provided by default as part of Flutter quill. Instead, it provides an interface to all the user to provide there own implementations for embed blocks. Implementations for image, video and formula embed blocks is proved in this package
### Custom Size Image for Mobile
Define `mobileWidth`, `mobileHeight`, `mobileMargin`, `mobileAlignment` as follows:
```json
{
"insert": {
"image": "https://user-images.githubusercontent.com/122956/72955931-ccc07900-3d52-11ea-89b1-d468a6e2aa2b.png"
},
"attributes":{
"style":"mobileWidth: 50; mobileHeight: 50; mobileMargin: 10; mobileAlignment: topLeft"
}
}
```
### Custom Size Image for other platforms (excluding web)
Define `width`, `height`, `margin`, `alignment` as follows:
```json
{
"insert": {
"image": "https://user-images.githubusercontent.com/122956/72955931-ccc07900-3d52-11ea-89b1-d468a6e2aa2b.png"
},
"attributes": {
"style":"width: 50; height: 50; margin: 10; alignment: topLeft"
}
}
```
## Features
```markdown

@ -4,10 +4,10 @@ import 'package:flutter/foundation.dart' show kIsWeb;
import 'package:flutter_quill/flutter_quill.dart';
import 'package:meta/meta.dart' show immutable;
import 'logic/models/config/shared_configurations.dart';
import 'presentation/embeds/editor/image/image.dart';
import 'presentation/embeds/editor/image/image_web.dart';
import 'presentation/embeds/editor/video.dart';
import 'presentation/embeds/editor/video/video.dart';
import 'presentation/embeds/editor/video/video_web.dart';
import 'presentation/embeds/editor/webview.dart';
import 'presentation/embeds/toolbar/camera_button/camera_button.dart';
import 'presentation/embeds/toolbar/formula_button.dart';
@ -16,8 +16,10 @@ import 'presentation/embeds/toolbar/image_button/image_button.dart';
// ignore: unused_import
import 'presentation/embeds/toolbar/media_button/media_button.dart';
import 'presentation/embeds/toolbar/video_button/video_button.dart';
import 'presentation/models/config/editor/image.dart';
import 'presentation/models/config/editor/video.dart';
import 'presentation/models/config/editor/image/image.dart';
import 'presentation/models/config/editor/image/image_web.dart';
import 'presentation/models/config/editor/video/video.dart';
import 'presentation/models/config/editor/video/video_web.dart';
import 'presentation/models/config/editor/webview.dart';
import 'presentation/models/config/toolbar/buttons/camera.dart';
import 'presentation/models/config/toolbar/buttons/formula.dart';
@ -39,7 +41,7 @@ export 'presentation/embeds/toolbar/media_button/media_button.dart';
export 'presentation/embeds/toolbar/utils/image_video_utils.dart';
export 'presentation/embeds/toolbar/video_button/video_button.dart';
export 'presentation/embeds/utils.dart';
export 'presentation/models/config/editor/image.dart';
export 'presentation/models/config/editor/image/image.dart';
export 'presentation/models/config/toolbar/buttons/camera.dart';
export 'presentation/models/config/toolbar/buttons/formula.dart';
export 'presentation/models/config/toolbar/buttons/image.dart';
@ -112,12 +114,11 @@ class FlutterQuillEmbeds {
/// [QuillEditorImageEmbedBuilderWeb] is the embed builder for handling
/// images on the web.
///
static List<EmbedBuilder> editorsWebBuilders({
QuillSharedExtensionsConfigurations sharedExtensionsConfigurations =
const QuillSharedExtensionsConfigurations(),
QuillEditorWebImageEmbedConfigurations? imageEmbedConfigurations =
const QuillEditorWebImageEmbedConfigurations(),
}) {
static List<EmbedBuilder> editorsWebBuilders(
{QuillEditorWebImageEmbedConfigurations? imageEmbedConfigurations =
const QuillEditorWebImageEmbedConfigurations(),
QuillEditorWebVideoEmbedConfigurations? videoEmbedConfigurations =
const QuillEditorWebVideoEmbedConfigurations()}) {
if (!kIsWeb) {
throw UnsupportedError(
'The editorsWebBuilders() is only for web, please use editorBuilders() '
@ -126,7 +127,13 @@ class FlutterQuillEmbeds {
}
return [
if (imageEmbedConfigurations != null)
const QuillEditorWebImageEmbedBuilder(),
QuillEditorWebImageEmbedBuilder(
configurations: imageEmbedConfigurations,
),
if (videoEmbedConfigurations != null)
QuillEditorWebVideoEmbedBuilder(
configurations: videoEmbedConfigurations,
),
];
}
@ -183,6 +190,7 @@ class FlutterQuillEmbeds {
controller: cameraButtonOptions.controller ?? controller,
options: cameraButtonOptions,
),
// TODO: We will return the support for this later
// if (mediaButtonOptions != null)
// (controller, toolbarIconSize, iconTheme, dialogTheme) =>
// QuillToolbarMediaButton(

@ -5,7 +5,7 @@ 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 '../../../models/config/editor/image/image.dart';
import '../../embed_types/image.dart';
import '../../utils.dart';
import '../../widgets/image.dart';
@ -84,7 +84,7 @@ class QuillEditorImageEmbedBuilder extends EmbedBuilder {
imageSize = OptionalSize(width, height);
image = Padding(
padding: EdgeInsets.all(margin),
child: getQuillImageByUrl(
child: getQuillImageByImageSource(
imageUrl,
width: width,
height: height,
@ -97,7 +97,7 @@ class QuillEditorImageEmbedBuilder extends EmbedBuilder {
}
if (imageSize == null) {
image = getQuillImageByUrl(
image = getQuillImageByImageSource(
imageUrl,
imageProviderBuilder: configurations.imageProviderBuilder,
imageErrorWidgetBuilder: configurations.imageErrorWidgetBuilder,

@ -4,16 +4,17 @@ import 'package:flutter_quill/extensions.dart' as base;
import 'package:flutter_quill/flutter_quill.dart';
import 'package:universal_html/html.dart' as html;
import '../../../models/config/editor/image/image_web.dart';
import '../../utils.dart';
import 'shims/dart_ui_fake.dart'
import '../shims/dart_ui_fake.dart'
if (dart.library.html) 'shims/dart_ui_real.dart' as ui;
class QuillEditorWebImageEmbedBuilder extends EmbedBuilder {
const QuillEditorWebImageEmbedBuilder({
this.constraints,
required this.configurations,
});
final BoxConstraints? constraints;
final QuillEditorWebImageEmbedConfigurations configurations;
@override
String get key => BlockEmbed.imageType;
@ -28,46 +29,94 @@ class QuillEditorWebImageEmbedBuilder extends EmbedBuilder {
TextStyle textStyle,
) {
assert(kIsWeb, 'ImageEmbedBuilderWeb is only for web platform');
final imageUrl = node.value.data;
if (isImageBase64(imageUrl)) {
// TODO: handle imageUrl of base64
return const Text('Image base 64 is not supported yet.');
}
final (height, width, margin, alignment) = _getImageSizeForWeb(node);
var height = 'auto';
var width = 'auto';
final style = node.style.attributes['style'];
if (style != null) {
final attrs = base.parseKeyValuePairs(style.value.toString(), {
Attribute.width.key,
Attribute.height.key,
Attribute.margin,
Attribute.alignment,
});
final heightValue = attrs[Attribute.height.key];
if (heightValue != null) {
height = heightValue;
}
final widthValue = attrs[Attribute.width.key];
if (widthValue != null) {
width = widthValue;
var imageSource = node.value.data.toString();
// This logic make sure if the image is imageBase64 then
// it make sure if the pattern is like
// data:image/png;base64, [base64 encoded image string here]
// if not then it will add the data:image/png;base64, at the first
if (isImageBase64(imageSource)) {
// Sometimes the image base 64 for some reasons
// doesn't displayed with the
if (!(imageSource.startsWith('data:image/') &&
imageSource.contains('base64'))) {
imageSource = 'data:image/png;base64, $imageSource';
}
}
ui.PlatformViewRegistry().registerViewFactory(imageUrl, (viewId) {
ui.PlatformViewRegistry().registerViewFactory(imageSource, (viewId) {
return html.ImageElement()
..src = imageUrl
..src = imageSource
..style.height = height
..style.width = width;
..style.width = width
..style.margin = margin
..style.alignSelf = alignment;
});
return ConstrainedBox(
constraints: constraints ?? BoxConstraints.loose(const Size(200, 200)),
constraints: configurations.constraints ??
BoxConstraints.loose(const Size(200, 200)),
child: HtmlElementView(
viewType: imageUrl,
viewType: imageSource,
),
);
}
}
/// Prefer the width, and height from the css style attribute if exits
/// it can be `auto` or `100px` so it's specific to HTML && CSS
/// if not, we will use the one from attributes which is usually just an double
(
String height,
String width,
String margin,
String alignment,
) _getImageSizeForWeb(
Node node,
) {
var height = 'auto';
var width = 'auto';
const margin = 'auto';
const alignment = 'center';
final cssStyle = node.style.attributes['style'];
// Usually double value
final heightValue = node.style.attributes[Attribute.height.key]?.value;
final widthValue = node.style.attributes[Attribute.width.key]?.value;
if (cssStyle != null) {
final attrs = base.parseKeyValuePairs(cssStyle.value.toString(), {
Attribute.width.key,
Attribute.height.key,
Attribute.margin,
Attribute.alignment,
});
final cssHeightValue = attrs[Attribute.height.key];
if (cssHeightValue != null) {
height = cssHeightValue;
} else {
height = '${heightValue}px';
}
final cssWidthValue = attrs[Attribute.width.key];
if (cssWidthValue != null) {
width = cssWidthValue;
} else if (widthValue != null) {
width = '${widthValue}px';
}
return (height, width, margin, alignment);
}
if (heightValue != null) {
height = '${heightValue}px';
}
if (widthValue != null) {
width = '${widthValue}px';
}
return (height, width, margin, alignment);
}

@ -5,10 +5,10 @@ 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';
import '../../../models/config/editor/video/video.dart';
import '../../utils.dart';
import '../../widgets/video_app.dart';
import '../../widgets/youtube_video_app.dart';
class QuillEditorVideoEmbedBuilder extends EmbedBuilder {
const QuillEditorVideoEmbedBuilder({

@ -0,0 +1,57 @@
import 'package:flutter/widgets.dart';
import 'package:flutter_quill/flutter_quill.dart';
import 'package:universal_html/html.dart' as html;
import 'package:youtube_player_flutter/youtube_player_flutter.dart'
show YoutubePlayer;
import '../../../models/config/editor/video/video_web.dart';
import '../../utils.dart';
import '../shims/dart_ui_fake.dart'
if (dart.library.html) 'shims/dart_ui_real.dart' as ui;
class QuillEditorWebVideoEmbedBuilder extends EmbedBuilder {
const QuillEditorWebVideoEmbedBuilder({
required this.configurations,
});
final QuillEditorWebVideoEmbedConfigurations configurations;
@override
String get key => BlockEmbed.videoType;
@override
Widget build(
BuildContext context,
QuillController controller,
Embed node,
bool readOnly,
bool inline,
TextStyle textStyle,
) {
var videoUrl = node.value.data;
if (isYouTubeUrl(videoUrl)) {
final youtubeID = YoutubePlayer.convertUrlToId(videoUrl);
if (youtubeID != null) {
videoUrl = 'https://www.youtube.com/embed/$youtubeID';
}
}
final size = MediaQuery.sizeOf(context);
ui.PlatformViewRegistry().registerViewFactory(
videoUrl,
(id) => html.IFrameElement()
..width = size.width.toString()
..height = size.height.toString()
..src = videoUrl
..style.border = 'none',
);
return SizedBox(
height: 500,
child: HtmlElementView(
viewType: videoUrl,
),
);
}
}

@ -30,7 +30,8 @@ bool isYouTubeUrl(String videoUrl) {
final uri = Uri.parse(videoUrl);
return uri.host == 'www.youtube.com' ||
uri.host == 'youtube.com' ||
uri.host == 'youtu.be';
uri.host == 'youtu.be' ||
uri.host == 'www.youtu.be';
} catch (_) {
return false;
}

@ -28,31 +28,35 @@ String getImageStyleString(QuillController controller) {
return s ?? '';
}
Image getQuillImageByUrl(
String imageUrl, {
Image getQuillImageByImageSource(
String imageSource, {
required ImageEmbedBuilderProviderBuilder? imageProviderBuilder,
required ImageErrorWidgetBuilder? imageErrorWidgetBuilder,
double? width,
double? height,
AlignmentGeometry alignment = Alignment.center,
}) {
if (isImageBase64(imageUrl)) {
return Image.memory(base64.decode(imageUrl),
width: width, height: height, alignment: alignment);
if (isImageBase64(imageSource)) {
return Image.memory(
base64.decode(imageSource),
width: width,
height: height,
alignment: alignment,
);
}
if (imageProviderBuilder != null) {
return Image(
image: imageProviderBuilder(imageUrl),
image: imageProviderBuilder(imageSource),
width: width,
height: height,
alignment: alignment,
errorBuilder: imageErrorWidgetBuilder,
);
}
if (isHttpBasedUrl(imageUrl)) {
if (isHttpBasedUrl(imageSource)) {
return Image.network(
imageUrl,
imageSource,
width: width,
height: height,
alignment: alignment,
@ -60,7 +64,7 @@ Image getQuillImageByUrl(
);
}
return Image.file(
File(imageUrl),
File(imageSource),
width: width,
height: height,
alignment: alignment,

@ -3,7 +3,7 @@ import 'dart:io' show File;
import 'package:flutter_quill/extensions.dart';
import 'package:meta/meta.dart' show immutable;
import '../../../embeds/embed_types/image.dart';
import '../../../../embeds/embed_types/image.dart';
/// [QuillEditorImageEmbedConfigurations] for desktop, mobile and
/// other platforms
@ -167,8 +167,3 @@ class QuillEditorImageEmbedConfigurations {
);
}
}
@immutable
class QuillEditorWebImageEmbedConfigurations {
const QuillEditorWebImageEmbedConfigurations();
}

@ -0,0 +1,11 @@
import 'package:flutter/widgets.dart' show BoxConstraints;
import 'package:meta/meta.dart' show immutable;
@immutable
class QuillEditorWebImageEmbedConfigurations {
const QuillEditorWebImageEmbedConfigurations({
this.constraints,
});
final BoxConstraints? constraints;
}

@ -0,0 +1,11 @@
import 'package:flutter/widgets.dart' show BoxConstraints;
import 'package:meta/meta.dart' show immutable;
@immutable
class QuillEditorWebVideoEmbedConfigurations {
const QuillEditorWebVideoEmbedConfigurations({
this.constraints,
});
final BoxConstraints? constraints;
}

@ -1,6 +1,6 @@
name: flutter_quill_extensions
description: Embed extensions for flutter_quill including image, video, formula and etc.
version: 0.6.0-dev.5
version: 0.6.0-dev.6
homepage: https://github.com/singerdmx/flutter-quill/tree/master/flutter_quill_extensions
repository: https://github.com/singerdmx/flutter-quill/tree/master/flutter_quill_extensions
@ -45,9 +45,9 @@ dependencies:
# In case you are working on changes for both libraries
# Comment the dependency_overrides section when publishing the package,
# then uncomment it back, this will be automated later
# dependency_overrides:
# flutter_quill:
# path: ../
dependency_overrides:
flutter_quill:
path: ../
dev_dependencies:
flutter_test:

@ -21,6 +21,20 @@ abstract base class Node extends LinkedListEntry<Node> {
/// Current parent of this node. May be null if this node is not mounted.
Container? parent;
/// The style attributes
/// Note: This is not the same as style attribute of css
///
/// Example:
///
/// {
/// "insert": {
/// "image": "https://user-images.githubusercontent.com/122956/72955931-ccc07900-3d52-11ea-89b1-d468a6e2aa2b.png"
/// },
/// "attributes": { // this one
/// "width": "230",
/// "style": "display: block; margin: auto; width: 500px;" // Not this
/// }
/// },
Style get style => _style;
Style _style = const Style();

@ -1,6 +1,6 @@
name: flutter_quill
description: A rich text editor built for the modern Android, iOS, web and desktop platforms. It is the WYSIWYG editor and a Quill component for Flutter.
version: 8.2.2
version: 8.2.3
homepage: https://1o24bbs.com/c/bulletjournal/108
repository: https://github.com/singerdmx/flutter-quill

Loading…
Cancel
Save