@ -1,4 +1,4 @@
import ' dart:io ' ;
import ' dart:io ' show File ;
import ' package:flutter/cupertino.dart ' ;
import ' package:flutter/foundation.dart ' ;
@ -7,13 +7,13 @@ import 'package:flutter/services.dart';
import ' package:flutter_quill/extensions.dart ' as base ;
import ' package:flutter_quill/flutter_quill.dart ' hide Text ;
import ' package:flutter_quill/translations.dart ' ;
import ' package:gal/gal.dart ' ;
import ' package:http/http.dart ' as http ;
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 ;
if ( dart . library . html ) ' package:flutter_quill_extensions/shims/dart_ui_real.dart '
as ui ;
import ' embed_types.dart ' ;
import ' utils.dart ' ;
import ' widgets/image.dart ' ;
import ' widgets/image_resizer.dart ' ;
@ -21,6 +21,9 @@ import 'widgets/video_app.dart';
import ' widgets/youtube_video_app.dart ' ;
class ImageEmbedBuilder extends EmbedBuilder {
ImageEmbedBuilder ( { required this . afterRemoveImageFromEditor } ) ;
final ImageEmbedBuilderAfterRemoveImageFromEditor afterRemoveImageFromEditor ;
@ override
String get key = > BlockEmbed . imageType ;
@ -38,112 +41,118 @@ class ImageEmbedBuilder extends EmbedBuilder {
) {
assert ( ! kIsWeb , ' Please provide image EmbedBuilder for Web ' ) ;
var image ;
Widget image = const SizedBox . shrink ( ) ;
final imageUrl = standardizeImageUrl ( node . value . data ) ;
OptionalSize ? _ imageSize;
OptionalSize ? imageSize ;
final style = node . style . attributes [ ' style ' ] ;
if ( base . isMobile ( ) & & style ! = null ) {
final _ attrs = base . parseKeyValuePairs ( style . value . toString ( ) , {
final attrs = base . parseKeyValuePairs ( style . value . toString ( ) , {
Attribute . mobileWidth ,
Attribute . mobileHeight ,
Attribute . mobileMargin ,
Attribute . mobileAlignment
} ) ;
if ( _ attrs. isNotEmpty ) {
if ( attrs . isNotEmpty ) {
assert (
_ attrs[ Attribute . mobileWidth ] ! = null & &
_ attrs[ Attribute . mobileHeight ] ! = null ,
attrs [ Attribute . mobileWidth ] ! = null & &
attrs [ Attribute . mobileHeight ] ! = null ,
' mobileWidth and mobileHeight must be specified ' ) ;
final w = double . parse ( _ attrs[ Attribute . mobileWidth ] ! ) ;
final h = double . parse ( _ attrs[ Attribute . mobileHeight ] ! ) ;
_ imageSize = OptionalSize ( w , h ) ;
final m = _ attrs[ Attribute . mobileMargin ] = = null
final w = double . parse ( attrs [ Attribute . mobileWidth ] ! ) ;
final h = double . parse ( attrs [ Attribute . mobileHeight ] ! ) ;
imageSize = OptionalSize ( w , h ) ;
final m = attrs [ Attribute . mobileMargin ] = = null
? 0.0
: double . parse ( _ attrs[ Attribute . mobileMargin ] ! ) ;
final a = base . getAlignment ( _ attrs[ Attribute . mobileAlignment ] ) ;
: double . parse ( attrs [ Attribute . mobileMargin ] ! ) ;
final a = base . getAlignment ( attrs [ Attribute . mobileAlignment ] ) ;
image = Padding (
padding: EdgeInsets . all ( m ) ,
child: imageByUrl ( imageUrl , width: w , height: h , alignment: a ) ) ;
}
}
if ( _ imageSize = = null ) {
if ( imageSize = = null ) {
image = imageByUrl ( imageUrl ) ;
_ imageSize = OptionalSize ( ( image as Image ) . width , image . height ) ;
imageSize = OptionalSize ( ( image as Image ) . width , image . height ) ;
}
if ( ! readOnly & & base . isMobile ( ) ) {
return GestureDetector (
onTap: ( ) {
showDialog (
context: context ,
builder: ( context ) {
final resizeOption = _SimpleDialogItem (
icon: Icons . settings_outlined ,
color: Colors . lightBlueAccent ,
text: ' Resize ' . i18n ,
onPressed: ( ) {
Navigator . pop ( context ) ;
showCupertinoModalPopup < void > (
context: context ,
builder: ( context ) {
final _screenSize = MediaQuery . of ( context ) . size ;
return ImageResizer (
onImageResize: ( w , h ) {
final res = getEmbedNode (
controller , controller . selection . start ) ;
final attr = base . replaceStyleString (
getImageStyleString ( controller ) , w , h ) ;
controller
. . skipRequestKeyboard = true
. . formatText (
res . offset , 1 , StyleAttribute ( attr ) ) ;
} ,
imageWidth: _imageSize ? . width ,
imageHeight: _imageSize ? . height ,
maxWidth: _screenSize . width ,
maxHeight: _screenSize . height ) ;
} ) ;
} ,
) ;
final copyOption = _SimpleDialogItem (
icon: Icons . copy_all_outlined ,
color: Colors . cyanAccent ,
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: ( ) {
final offset =
getEmbedNode ( controller , controller . selection . start )
. offset ;
controller . replaceText ( offset , 1 , ' ' ,
TextSelection . collapsed ( offset: offset ) ) ;
Navigator . pop ( context ) ;
} ,
) ;
return Padding (
padding: const EdgeInsets . fromLTRB ( 50 , 0 , 50 , 0 ) ,
child: SimpleDialog (
shape: const RoundedRectangleBorder (
borderRadius:
BorderRadius . all ( Radius . circular ( 10 ) ) ) ,
children: [ resizeOption , copyOption , removeOption ] ) ,
) ;
} ) ;
} ,
child: image ) ;
onTap: ( ) {
showDialog (
context: context ,
builder: ( context ) {
final resizeOption = _SimpleDialogItem (
icon: Icons . settings_outlined ,
color: Colors . lightBlueAccent ,
text: ' Resize ' . i18n ,
onPressed: ( ) {
Navigator . pop ( context ) ;
showCupertinoModalPopup < void > (
context: context ,
builder: ( context ) {
final screenSize = MediaQuery . of ( context ) . size ;
return ImageResizer (
onImageResize: ( w , h ) {
final res = getEmbedNode (
controller , controller . selection . start ) ;
final attr = base . replaceStyleString (
getImageStyleString ( controller ) , w , h ) ;
controller
. . skipRequestKeyboard = true
. . formatText (
res . offset , 1 , StyleAttribute ( attr ) ) ;
} ,
imageWidth: imageSize ? . width ,
imageHeight: imageSize ? . height ,
maxWidth: screenSize . width ,
maxHeight: screenSize . height ) ;
} ) ;
} ,
) ;
final copyOption = _SimpleDialogItem (
icon: Icons . copy_all_outlined ,
color: Colors . cyanAccent ,
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 {
final navigator = Navigator . of ( context ) ;
final offset =
getEmbedNode ( controller , controller . selection . start )
. offset ;
controller . replaceText (
offset ,
1 ,
' ' ,
TextSelection . collapsed ( offset: offset ) ,
) ;
navigator . pop ( ) ;
await afterRemoveImageFromEditor ( File ( imageUrl ) ) ;
} ,
) ;
return Padding (
padding: const EdgeInsets . fromLTRB ( 50 , 0 , 50 , 0 ) ,
child: SimpleDialog (
shape: const RoundedRectangleBorder (
borderRadius: BorderRadius . all ( Radius . circular ( 10 ) ) ) ,
children: [ resizeOption , copyOption , removeOption ] ) ,
) ;
} ) ;
} ,
child: image ,
) ;
}
if ( ! readOnly | | ! base . isMobile ( ) | | isImageBase64 ( imageUrl ) ) {
@ -151,7 +160,11 @@ class ImageEmbedBuilder extends EmbedBuilder {
}
/ / We provide option menu for mobile platform excluding base64 image
return _menuOptionsForReadonlyImage ( context , imageUrl , image ) ;
return _menuOptionsForReadonlyImage (
context ,
imageUrl ,
image ,
) ;
}
}
@ -271,29 +284,39 @@ Widget _menuOptionsForReadonlyImage(
text: ' Save ' . i18n ,
onPressed: ( ) async {
imageUrl = appendFileExtensionToImageUrl ( imageUrl ) ;
final messenger = ScaffoldMessenger . of ( context ) ;
Navigator . of ( context ) . pop ( ) ;
/ / Download image
final uri = Uri . parse ( imageUrl ) ;
final response = await http . get ( uri ) ;
if ( response . statusCode ! = 200 ) {
throw Exception (
' failed to download image: ${ response . statusCode } ' ,
) ;
}
final saveImageResult = await saveImage ( imageUrl ) ;
final imageSavedSuccessfully = saveImageResult . isSuccess ;
messenger . clearSnackBars ( ) ;
/ / Save image to a temporary path
final fileName = uri . pathSegments . isEmpty ? ' image.jpg '
: uri . pathSegments . last ;
final imagePath = ' ${ Directory . systemTemp . path } /menu-opt- $ fileName ' ;
final imageFile = File ( imagePath ) ;
await imageFile . writeAsBytes ( response . bodyBytes ) ;
if ( ! imageSavedSuccessfully ) {
/ / TODO: Please translate this
messenger . showSnackBar ( const SnackBar (
content: Text (
' Error while saveing the image ' ,
) ) ) ;
return ;
}
/ / Save image to gallery
await Gal . putImage ( imagePath ) ;
var message = ' Saved ' . i18n ;
switch ( saveImageResult . method ) {
/ / TODO: Please translate this too
case SaveImageResultMethod . network:
message + = ' using the network. ' ;
break ;
case SaveImageResultMethod . localStorage:
message + = ' using the local storage. ' ;
break ;
}
ScaffoldMessenger . of ( context )
. showSnackBar ( SnackBar ( content: Text ( ' Saved ' . i18n ) ) ) ;
Navigator . pop ( context ) ;
messenger . showSnackBar (
SnackBar (
content: Text ( message ) ,
) ,
) ;
} ,
) ;
final zoomOption = _SimpleDialogItem (