From 039e3193d92bc7fad818f13cfda41ad9c2019137 Mon Sep 17 00:00:00 2001 From: Ellet <73608287+freshtechtips@users.noreply.github.com> Date: Tue, 5 Dec 2023 07:06:30 +0300 Subject: [PATCH] Improving flutter quill extensions (#1564) * organizations for flutter quill extensions --- .gitignore | 5 +- .../lib/presentation/quill/quill_editor.dart | 4 +- .../quill/samples/quill_images_sample.dart | 18 +- flutter_quill_extensions/CHANGELOG.md | 8 + flutter_quill_extensions/README.md | 25 +- .../embeds/embed_types.dart | 0 .../formula/editor/formula_embed.dart} | 5 +- .../formula}/toolbar/formula_button.dart | 2 +- .../lib/embeds/image/editor/image_embed.dart | 86 +++++ .../image/editor/image_embed_types.dart} | 4 +- .../image/editor}/image_menu.dart | 12 +- .../image/editor/image_web_embed.dart} | 8 +- .../image/toolbar}/image_button.dart | 8 +- .../image/toolbar}/select_image_source.dart | 30 +- .../others}/camera_button/camera_button.dart | 6 +- .../others/camera_button/camera_types.dart} | 4 +- .../camera_button/select_camera_action.dart | 2 +- .../others}/image_video_utils.dart | 0 .../others}/media_button/media_button.dart | 0 .../unknown/editor/unknown_embed.dart} | 0 .../video/editor/video_embed.dart} | 4 +- .../video/editor/video_web_embed.dart} | 6 +- .../video/toolbar}/select_video_source.dart | 22 +- .../video/toolbar}/video_button.dart | 9 +- .../embed_types => embeds/video}/video.dart | 4 +- .../embeds/widgets/image.dart | 2 +- .../embeds/widgets/image_resizer.dart | 0 .../embeds/widgets/video_app.dart | 2 +- .../embeds/widgets/youtube_video_app.dart | 0 .../lib/extensions/attribute.dart | 32 ++ .../{logic => }/extensions/controller.dart | 23 -- .../lib/flutter_quill_extensions.dart | 87 +++-- .../lib/logic/extensions/attribute.dart | 30 -- .../models/config/editor/image/image.dart | 2 +- .../models/config/editor/image/image_web.dart | 0 .../models/config/editor/video/video.dart | 0 .../models/config/editor/video/video_web.dart | 0 .../models/config/editor/webview.dart | 0 .../models/config/shared_configurations.dart | 0 .../models/config/toolbar/buttons/camera.dart | 2 +- .../config/toolbar/buttons/formula.dart | 0 .../models/config/toolbar/buttons/image.dart | 2 +- .../config/toolbar/buttons/media_button.dart | 0 .../models/config/toolbar/buttons/video.dart | 2 +- .../embeds/editor/image/image.dart | 150 --------- .../presentation/embeds/editor/webview.dart | 58 ---- .../lib/presentation/utils/utils.dart | 201 ------------ .../services/image_picker/image_options.dart | 0 .../services/image_picker/image_picker.dart | 0 .../image_picker/packages/image_picker.dart | 0 .../services/image_picker/s_image_picker.dart | 0 .../services/image_saver/exceptions.dart | 0 .../services/image_saver/image_saver.dart | 0 .../services/image_saver/packages/gal.dart | 0 .../services/image_saver/s_image_saver.dart | 0 .../shims => utils/dart_ui}/dart_ui_fake.dart | 0 .../shims => utils/dart_ui}/dart_ui_real.dart | 0 .../element_utils/element_shared_utils.dart | 29 ++ .../utils/element_utils/element_utils.dart | 98 ++++++ .../element_utils/element_web_utils.dart} | 24 +- .../{logic => }/utils/quill_image_utils.dart | 2 +- .../lib/{logic => }/utils/string.dart | 12 +- flutter_quill_extensions/lib/utils/utils.dart | 89 ++++++ flutter_quill_extensions/pubspec.yaml | 2 +- lib/flutter_quill.dart | 1 - .../l10n/generated/quill_localizations.dart | 36 +++ .../generated/quill_localizations_ar.dart | 20 ++ .../generated/quill_localizations_bg.dart | 20 ++ .../generated/quill_localizations_bn.dart | 20 ++ .../generated/quill_localizations_cs.dart | 20 ++ .../generated/quill_localizations_da.dart | 20 ++ .../generated/quill_localizations_de.dart | 20 ++ .../generated/quill_localizations_en.dart | 20 ++ .../generated/quill_localizations_es.dart | 20 ++ .../generated/quill_localizations_fa.dart | 20 ++ .../generated/quill_localizations_fr.dart | 20 ++ .../generated/quill_localizations_he.dart | 20 ++ .../generated/quill_localizations_hi.dart | 20 ++ .../generated/quill_localizations_id.dart | 20 ++ .../generated/quill_localizations_it.dart | 20 ++ .../generated/quill_localizations_ja.dart | 20 ++ .../generated/quill_localizations_ko.dart | 20 ++ .../generated/quill_localizations_ms.dart | 20 ++ .../generated/quill_localizations_nl.dart | 20 ++ .../generated/quill_localizations_no.dart | 20 ++ .../generated/quill_localizations_pl.dart | 20 ++ .../generated/quill_localizations_pt.dart | 20 ++ .../generated/quill_localizations_ru.dart | 20 ++ .../generated/quill_localizations_sr.dart | 20 ++ .../generated/quill_localizations_sw.dart | 20 ++ .../generated/quill_localizations_tk.dart | 20 ++ .../generated/quill_localizations_tr.dart | 20 ++ .../generated/quill_localizations_uk.dart | 20 ++ .../generated/quill_localizations_ur.dart | 20 ++ .../generated/quill_localizations_vi.dart | 20 ++ .../generated/quill_localizations_zh.dart | 20 ++ lib/src/l10n/quill_en.arb | 8 +- lib/src/l10n/untranslated.json | 299 +++++++++++++++++- lib/translations.dart | 1 + 99 files changed, 1441 insertions(+), 625 deletions(-) rename flutter_quill_extensions/lib/{presentation => }/embeds/embed_types.dart (100%) rename flutter_quill_extensions/lib/{presentation/embeds/editor/formula.dart => embeds/formula/editor/formula_embed.dart} (90%) rename flutter_quill_extensions/lib/{presentation/embeds => embeds/formula}/toolbar/formula_button.dart (98%) create mode 100644 flutter_quill_extensions/lib/embeds/image/editor/image_embed.dart rename flutter_quill_extensions/lib/{presentation/embeds/embed_types/image.dart => embeds/image/editor/image_embed_types.dart} (95%) rename flutter_quill_extensions/lib/{presentation/embeds/editor/image => embeds/image/editor}/image_menu.dart (94%) rename flutter_quill_extensions/lib/{presentation/embeds/editor/image/image_web.dart => embeds/image/editor/image_web_embed.dart} (88%) rename flutter_quill_extensions/lib/{presentation/embeds/toolbar/image_button => embeds/image/toolbar}/image_button.dart (96%) rename flutter_quill_extensions/lib/{presentation/embeds/toolbar/image_button => embeds/image/toolbar}/select_image_source.dart (65%) rename flutter_quill_extensions/lib/{presentation/embeds/toolbar => embeds/others}/camera_button/camera_button.dart (96%) rename flutter_quill_extensions/lib/{presentation/embeds/embed_types/camera.dart => embeds/others/camera_button/camera_types.dart} (94%) rename flutter_quill_extensions/lib/{presentation/embeds/toolbar => embeds/others}/camera_button/select_camera_action.dart (95%) rename flutter_quill_extensions/lib/{presentation/embeds/toolbar/utils => embeds/others}/image_video_utils.dart (100%) rename flutter_quill_extensions/lib/{presentation/embeds/toolbar => embeds/others}/media_button/media_button.dart (100%) rename flutter_quill_extensions/lib/{presentation/embeds/editor/unknown.dart => embeds/unknown/editor/unknown_embed.dart} (100%) rename flutter_quill_extensions/lib/{presentation/embeds/editor/video/video.dart => embeds/video/editor/video_embed.dart} (95%) rename flutter_quill_extensions/lib/{presentation/embeds/editor/video/video_web.dart => embeds/video/editor/video_web_embed.dart} (89%) rename flutter_quill_extensions/lib/{presentation/embeds/toolbar/video_button => embeds/video/toolbar}/select_video_source.dart (74%) rename flutter_quill_extensions/lib/{presentation/embeds/toolbar/video_button => embeds/video/toolbar}/video_button.dart (96%) rename flutter_quill_extensions/lib/{presentation/embeds/embed_types => embeds/video}/video.dart (94%) rename flutter_quill_extensions/lib/{presentation => }/embeds/widgets/image.dart (99%) rename flutter_quill_extensions/lib/{presentation => }/embeds/widgets/image_resizer.dart (100%) rename flutter_quill_extensions/lib/{presentation => }/embeds/widgets/video_app.dart (98%) rename flutter_quill_extensions/lib/{presentation => }/embeds/widgets/youtube_video_app.dart (100%) create mode 100644 flutter_quill_extensions/lib/extensions/attribute.dart rename flutter_quill_extensions/lib/{logic => }/extensions/controller.dart (70%) delete mode 100644 flutter_quill_extensions/lib/logic/extensions/attribute.dart rename flutter_quill_extensions/lib/{presentation => }/models/config/editor/image/image.dart (98%) rename flutter_quill_extensions/lib/{presentation => }/models/config/editor/image/image_web.dart (100%) rename flutter_quill_extensions/lib/{presentation => }/models/config/editor/video/video.dart (100%) rename flutter_quill_extensions/lib/{presentation => }/models/config/editor/video/video_web.dart (100%) rename flutter_quill_extensions/lib/{presentation => }/models/config/editor/webview.dart (100%) rename flutter_quill_extensions/lib/{logic => }/models/config/shared_configurations.dart (100%) rename flutter_quill_extensions/lib/{presentation => }/models/config/toolbar/buttons/camera.dart (93%) rename flutter_quill_extensions/lib/{presentation => }/models/config/toolbar/buttons/formula.dart (100%) rename flutter_quill_extensions/lib/{presentation => }/models/config/toolbar/buttons/image.dart (95%) rename flutter_quill_extensions/lib/{presentation => }/models/config/toolbar/buttons/media_button.dart (100%) rename flutter_quill_extensions/lib/{presentation => }/models/config/toolbar/buttons/video.dart (95%) delete mode 100644 flutter_quill_extensions/lib/presentation/embeds/editor/image/image.dart delete mode 100644 flutter_quill_extensions/lib/presentation/embeds/editor/webview.dart delete mode 100644 flutter_quill_extensions/lib/presentation/utils/utils.dart rename flutter_quill_extensions/lib/{logic => }/services/image_picker/image_options.dart (100%) rename flutter_quill_extensions/lib/{logic => }/services/image_picker/image_picker.dart (100%) rename flutter_quill_extensions/lib/{logic => }/services/image_picker/packages/image_picker.dart (100%) rename flutter_quill_extensions/lib/{logic => }/services/image_picker/s_image_picker.dart (100%) rename flutter_quill_extensions/lib/{logic => }/services/image_saver/exceptions.dart (100%) rename flutter_quill_extensions/lib/{logic => }/services/image_saver/image_saver.dart (100%) rename flutter_quill_extensions/lib/{logic => }/services/image_saver/packages/gal.dart (100%) rename flutter_quill_extensions/lib/{logic => }/services/image_saver/s_image_saver.dart (100%) rename flutter_quill_extensions/lib/{presentation/embeds/editor/shims => utils/dart_ui}/dart_ui_fake.dart (100%) rename flutter_quill_extensions/lib/{presentation/embeds/editor/shims => utils/dart_ui}/dart_ui_real.dart (100%) create mode 100644 flutter_quill_extensions/lib/utils/element_utils/element_shared_utils.dart create mode 100644 flutter_quill_extensions/lib/utils/element_utils/element_utils.dart rename flutter_quill_extensions/lib/{presentation/utils/web_utils.dart => utils/element_utils/element_web_utils.dart} (77%) rename flutter_quill_extensions/lib/{logic => }/utils/quill_image_utils.dart (99%) rename flutter_quill_extensions/lib/{logic => }/utils/string.dart (66%) create mode 100644 flutter_quill_extensions/lib/utils/utils.dart diff --git a/.gitignore b/.gitignore index cee99ed04..fc03c2767 100644 --- a/.gitignore +++ b/.gitignore @@ -80,4 +80,7 @@ pubspec.lock # For local development pubspec_overrides.yaml -old_example \ No newline at end of file +old_example + +# A directory where you put all of your local things that you don't want to push +.flutter-quill \ No newline at end of file diff --git a/example/lib/presentation/quill/quill_editor.dart b/example/lib/presentation/quill/quill_editor.dart index 133f9997f..06f7eb766 100644 --- a/example/lib/presentation/quill/quill_editor.dart +++ b/example/lib/presentation/quill/quill_editor.dart @@ -7,9 +7,9 @@ import 'package:desktop_drop/desktop_drop.dart' show DropTarget; import 'package:flutter/material.dart'; import 'package:flutter_quill/extensions.dart' show isAndroid, isIOS, isWeb; import 'package:flutter_quill/flutter_quill.dart'; -import 'package:flutter_quill_extensions/flutter_quill_extensions.dart'; -import 'package:flutter_quill_extensions/presentation/embeds/widgets/image.dart' +import 'package:flutter_quill_extensions/embeds/widgets/image.dart' show getImageProviderByImageSource, imageFileExtensions; +import 'package:flutter_quill_extensions/flutter_quill_extensions.dart'; import 'package:path/path.dart' as path; import '../extensions/scaffold_messenger.dart'; diff --git a/example/lib/presentation/quill/samples/quill_images_sample.dart b/example/lib/presentation/quill/samples/quill_images_sample.dart index 849abc326..9dadcd2ca 100644 --- a/example/lib/presentation/quill/samples/quill_images_sample.dart +++ b/example/lib/presentation/quill/samples/quill_images_sample.dart @@ -6,9 +6,9 @@ final quillImagesSample = [ { 'insert': {'image': Assets.images.screenshot1.path}, 'attributes': { - 'width': '100', - 'height': '100', - 'style': 'width:500px; height:350px; deletable: false;' + 'width': '200', + 'height': '500', + 'style': 'width:500px; height:350px; margin: 20px;' } }, {'insert': '\n'}, @@ -20,8 +20,8 @@ final quillImagesSample = [ 'https://helpx.adobe.com/content/dam/help/en/photoshop/using/convert-color-image-black-white/jcr_content/main-pars/before_and_after/image-before/Landscape-Color.jpg' }, 'attributes': { - 'width': '100', - 'height': '100', + 'width': '250', + 'height': '250', 'style': 'width:250px; height:250px;' } }, @@ -38,8 +38,8 @@ final quillImagesSample = [ '/9j/4AAQSkZJRgABAQAAAQABAAD/2wCEAAoHCBYWFRgWFRUYGRgaGBgYGhoYGBgaGhgYGBgaGRoaGBwcIS4lHB4rIRgYJjgmKy8xNTU1GiQ7QDszPy40NTEBDAwMEA8QHhISHjYkJCs0NDY2NDQ0NDQ0NDQ0NDQ0NDQ0NDQ0NDQ0NDY0NDQ0NDQ0NDQ0NDQ0NDQ0NDQ0NDQ0NP/AABEIALEBHQMBIgACEQEDEQH/xAAbAAACAwEBAQAAAAAAAAAAAAAAAwECBAUGB//EADkQAAEDAgMECAUEAgEFAAAAAAEAAhEDIQQxQRJRYXEFIlKBkaHR8BMUMpKxBmLB4ULxFSNTcoKy/8QAGQEAAwEBAQAAAAAAAAAAAAAAAAECAwQF/8QALBEAAgIBAwMDAwMFAAAAAAAAAAECESEDEjEEE0EiUWEUkaEFQnEVIzJSgf/aAAwDAQACEQMRAD8A+TseQpN10MRhHDNnG2ixERmFommaz05RdMvRiCD3KGUpMKWmcldpJysfymNU6GMoltyYniPwpczkfJUBn6s+Ofih7SDYqcmuEsLBDsONx9FmfSIW2lWj6itDgHCzxyMeR1RbXIu1GStcnKa4qWt1WmtRM6dxSmkixVowcWnTGPgwY4JlPDzeQBvKVKlgMyJQ0aJq8qxjxIIGnmsTwuiynORvySH0N+9JBOLeTLTNwVsfRsHaFIDIK30q7Q0sdkbtO47uRTeBacU7TwZDVgRqqHepqNl3imsp9XvhMVNuh+HEgjh/az1KUStNFhkAZgyr1GmTIzHkp4Zu47omN4BaL3iElqa9u5KGapGD5HFshIfktFHcqOYgJK1ZmhS0qXNVqQumjKslnsS015S4TG+SrwqkWTHBVcEEsoxspxsFNNh0HsqXsjNAVixGzK6WFbsQcoafNYmG40T31paecD1UtWaabUXYrEPkczKzhsprm2VC+ECbt2x9LE/+XirVagP0kcQshaQpISoruSqmSQRl4KzHb57lDHxvWho3T5ICKvgWDJzPemlh1mVX4aY185oLS9wps0KH0IyunhivVZZKzXZjJla+NEuJOa0ihItmqtokFO0Q4SxfAksK14XDTmmGNVJdIAGYRZpGEYu3kz4inExoYSmui575/K6TMPInyVquGaGzpqlfgrtN+pYOa9sxAPelOYVpa0tJbeFpp0AU7M9jkIwuGm58/wCEVeo4gQW5iOK14l+xZu6RwScMzbhuWg/vzQvcval6VyWwDuuHbuYCdiiCTExMzqTu5KcXR2C1g+odZ3hYK1KkC3NTjk3jFpPTOa+ne2SVVp3nRa8SyDAV6TAWOnfPkndKznencnEwMtdMYJlWNEjkmUYVMhRd0zDVaq0QnYkJTbBNGE1UijyrsbaUuJTn5QmSsiiqgKxCEyRzK2zlnP8ApJe6VRXSHbeC7W2JPD1VaTSfeQV3mx7o8FLGZjKw8SgpRyD2zkqupN3+Cs1si2iz1M1DLapcFmmc0MbHJUa5NaUyU7LVGRqDxCbQeB3pb3znf8qgJQXuUXaNNip+GlNctm1LRlMaT5pPBrGpcht2j+FQ3VXlRRdcfyihuWaY6ky8LUynNvBZ3G9vfehrzndQzaLSdF67IKZQbbcrE7Weau2mRCVmih6rXAtry0hVc8kxock0gG0KatMti3cFSYOLr4LYbCbTgHeJSq31m0AQO4WT2VpETBV6lPa628AH3vU27yabIyjUTDUp6xPFUw745rc2j1SLpDMMJmVUWZS0pJpo07W20PIuLHkq0KRBIBV8M/YJ/B13wtjmgwW5HPepbrB0RimrfPk5lTDEyTnuSmN2XQ+wIWikNp0HMT7CdicM0kck7rDMdm71ROdiARYXAy4hKczKE7EsLbbsuSJ6seCpcGMo3J2Z6tKVkqsW90xByWZzLxvKqJhqRRFLDHYL9xhKcF38RhtimG74PkuI5iIy3WVraHapPmhQbOSoWLbhmXuDAH+lmqG9lV5OZxxZnIVg1XayUx0JkJFWiZPJOpM2p4m/IKNieq3P8J9PqsIzvpGSiXBvpxzngQx+zI0IM/wsVXNNe8pClImc/CJhXaoa9MseCohEICtsHNDRGiodFgzJaGUjGnvcq0HgWcJB8uKY5kQJnOD7ySZrFJZK1WkC4SV12bIA2htNIAHWFjF1hq4W20zLmD+OSlM0np+UUY7Ja7EW9lYGv0hOpm9j4puIQnWDWam7vn0XQwtTaAkXGXsLl0XdbetzCBkBM2I3e96zlE7dGebbFOeWPuLTcLazZc4DaAERfPu3rFin3mb/AJVsMWky4kEZIatDjOpNeLNTsKAZE9xVWYgNgeI0jwVn4oCwOf53FVGH22yCJGlvGNVOa9Rs6v8At8j8RTBBdOYELNgLkx9Q0WukzZGzYgg7hHOLarDQGy8jI70ReGg1LUk6/kfVwpcXEC41Rh2FtnfTIGc+UBbMMyoTf6SDCc9jS2TbllPCMlDl4NVpJ+rhnCxDYeDMZ+/Na6WKcXfTtTYmMgq9I4cm4GXvNasBScGEbTTlkQfwSVo2nGzlhGS1XFY8nMxjNB7CzPqWjculjKUaQf4K5uzzTi8GWtBqTIgkC5KTk9vAhdLDt/aO9Z6jOsTG5WpGUtNqmdjpj6BG4fhee2F3cY8vptPALAGQMllo4R1dat8017IyYt4b1Wm035rCQn12GVWmxdEcI8qduVAynZVcxPcICq5ylyK7eBVNhBBORMKcTVnLkqPfKoQir5E5bVSEuUFqYWqibMKDZUwmQiEytpUEpnxJzUbKkNQNWXsck5rtPZWcNTBKdFpjHAxY23KaWKIscp00VFXYRVj3NO0aq8ZjXXLvWZ7NZlWAUtQo0De7kii4jVa2YrxWZwUBiGkxxlKOEbmO2hI8FRzt/wDrks7SU6m8ze6W00WpYmo4gxK34DFlptnHvvWasySltsbEEd/8hDimhRnKErTPSYfFMcQ2QHHKJA7wd6XisBBDjkezc9wWDBuBERff6LrPxQazYMHOSc44ELCUalg9OOvGcPUZamIa1ogm3+JPmpoVGPHWcZM2E7swuZiCCbJDHlpkEhadtNHM+rallWjqVq5HUfPppfgtuBwzXMlj+twtuXBq1HO6xJJOafgK7wYEj3qlKHpwEOoT1Latfk71TCy2XZ3EmMxoVi/407JLBN96s/H7DSJknfYepSMN0wWG3LK0ctVioS5R2S19JtKRjfIsQQQhhkR5r0FSm2o0HYAcdRMELmPwhHIajK/EJqSarhkvRae5O0Q1+0wNjXyUPa0y2chbitD6OwwkmIsD/iZ4ysNNhngBJPD3CItU6HqJ2k1yYK4lZnG601DJnjlqstYrTdg86cKbZDnKuatTpklMfSjNTuyPZJqxGyquTnMUFkK9xk4GchV2E5xVSEnJkbUP+Ej4S6LaKn5dPca9o5opKfhLpDDI+WT3i7LOcKasKa3/AC6t8snvH2mYRTU/CW8UEwUEbhrRZzhSR8FdIYZT8sjeHZZzfgqfhLpjDKlemGNLjojegek0rOd8NKfVa0wXCd2qw4npVzpDRsjK1z4rnc/evcpcznlJXg7jukGCZJJ4DNYqnSTjk0AeJWAqQJUObE5SkaW9JVBdro5AKW9JVZJ2iZEXgjwKQynKsKShyZSUvcsMbU7R7wPRWHSD5m3KLJfw1QsvmmpP3JcWb6PSejx3j0W3D9INNtqOdvNcEqQfeatTYlJxZ6Vjw+4O15pnwDuXmaFdzCHMMHf/AAdF6DorpX4jg14AJ1mJ7j/Ce9m+nOMnUuTp4LFuZbMbjuXZovY8yI2gL6G3fuuuY7DQhjSDr5rKUVLKPT0taWl6XlD+mLsERIzkZnS29c6oNmmLHrCSSD/AXcovEbTjJk2tPvJJxzWEWgxcczosVujivJ2PZO3aujy7i3Qgc80gtJPuy246BaPZSGUH5xA3n3daqR509P1Vz/A6kNgTqlCk5xlXpgzJPvlmnGTvjgIQmW43FLwZ30g3VZajwtb6RSvg3vbvVJnPOLeEqMoBOiNkrWSBkElz1VmLhR320kwUk8NRsrKzuUEJFJT8JODVcNRuL2Iz/CV20eCeGqwCW4agjP8ABV20Vpa1MYxTvKUEIZRVxh1paxPa0KXMtQRiGGXD/Vb9ilA2ZcYubxrsjU+q9UTC8r07+n31qwe02Ih0kAN2cgLE35FOEs5OfqYvY1FW2eNoYfavIA4mL896vTY3Xa2p1sLTY7jluyK19J4H4VQU3OB+klrZJAOgtnF0us9pAYGNadr6y51hFg4kx6Qt7PF206fJb5Zjg0M+oi7bkzJmd1t/DmodgjNhJuSGyYA4opS36QQf8ocC0gXaWunPPImy6FDFA5ghwjZAygznnfLzUSdG2nC2ZMPhSYjzGROXkt2G6P3tJMGwtaM76T70XQ6OwpfoLNbmf2mBnHGN67rOjn/WwbO1YBs2EDjf6fErmnq0enp9NFrJ4x+AOYBjfGmsCbhY34QlxGZgk7pE2nJe2xXRRaNghoJlwcbRAiAcrxPcRmVwK5DXdYDZl4dsF2RIkC/LXTxuGpZlraCStHIbgJbtSIAkxnESRB15bwlVmssG7rkTv1Ha4X5rdisS4kta3ZEnZJMGAZG1Ag6Wyk8Vz3tEFwIEQeses46xw9890zz5Roq2m1xMdUR/kbcp1N7WUYd5p1GmQNk65RN5ibJlaoHAkMa3KNkERGcifMDRJYxzyGtaS4mBxJ0GgVGfnB9FwzNtoMgjOQU8U4sAk9A9H/BotaSScyJEBxzA4LqFgsYWDlk9tJuKbVMzUuj3OvkE44BoF3SeY/2ukKzS2AALa71y6uHe6esO4yO9ZT1ZfwdWho6by2c/E4SnMwCdy5+JuYtbdl3+i67sEG5unjB9IhZ3YZgyI8D/AAs4ydndJQapHFdTJMD+/wCldlB+QHeuoKbNxPcU1j4yYfABa9w5Hp5wmckdHPO9NHQx1Hius2sRp5kqj8Q85DwB/JR3fYT6a8tHJqdFgZx4eqyuwrRu8l0q7ah/xPiFz6uHfOg7wtFqfJy6ug1wjrhymV4L5l/bf97vVW+Zf23/AHO9Vv2/k4F1vwe8EK0rwYxT+2/73eqn5h/bf9zvVLt/I/rfg94HKwevBDEP7b/ud6qRWf23fcfVHavyV9b8H0BhT2L50Kr+277irio/tu+4pdj5KXW/H5PpLGpwYvmjaj+077irNc/tO8Sk+mfuUusT8fk+h1EMavBsL+0fErSwP7R8SjsNeTWOvu8HqsZ0NSqElzGkkESBDoIjMXWEfpyiA1oaeqLHaJ33M2Jv7Fly2U37ynMpO3+f9JKDXkfbhJ24m6p+m2Cm9rHPbtCdluwZ2bgbJEG4GoJymMuTU/TdZjC8kFgkkWaQDHW2chraf8e5b2MO8+K00id6mUZe5UemjdrAdC9E1s2gggTcObIIiJI1BI7l9K/S9Sm1kPADg2OsADHhy8F4rCv4rH+puk3sbT2XES5wnhAXJKLUk0aa+lu06vB3+n8EatQ/CBa2HXuGxNxuPJeLxH6frPf1Wk3glwLQBlMuF9fpnK0r1tWuSIk2EBc6s528o0oy5NFoeja2cnD/AKJkzVqzlIa0HIW6zgbAzpuXQH6Pw4BGzY2zvG7az80qpUf2neJ9VlqV6vbf9x9V07ZPyYPpoR8Wdan+ksPLf+m3qgtGZEHtXvzK3N6Kp07Ma1utgBc8l5N9at23/e71SH4mt23/AHu9Udmb/cJbIu1Gv+HtPhDgoNMLwdTFVe2/7nLM/F1f+4/73Jrp5e5M+ogvDPoL3BuqyPxf7o7gvBvxVQ5vf9zvVKdXf23/AHFKXSSlywj1+nD9r+57l2Jn/I++QSDif3eX9rxJrv7bvuKoaz+277il9G15Kf6rDxF/c9yMUO049/oFLsUB7v5rwhxD+2/7iqnEP7bvuPqj6V+4f1WNf4v7nvDjkt+P9yvCmu/tu+4qhrO7bvuKpdJ8mcv1Vf6/k9nVxnELI7FDteS8r8V3bd4lRtu7R8SqXT15MpfqSf7SAphRKkLrPJRIClQFIKQ7LhWASw5WBTKTGNV2lKDlcP4popMe0+7JjZ9grMH8kxruCClI2Mf79haqT/c3XPY/mnMfzPc0pNG0NSjqNefYTw/n3gj+FzG1QNR9npATWVRw32MeAlZuJ0w1WdFtRWZVg5rnVcaxou4bjtTPJcut02BIYO/RZSRu+ohHlntaFfiPfJcL9Y4jq094c7/5/pcB3T1XQgeKyYjGvfG07ai91nsd2yNXrISg4xu2fTqWJlufvuUPfx/C+d0um6rRAdI4ick6n+pKwN4I7wlGDRquv0ms2e0quWJ9T37C49L9SNdZwLcr3PO6eMe1ws6Z4wDP/tc8FrFe4pdTCS9LNb38/NJqP3z3gDzssz6w48QAe7MpT38PJoWyic09Zk1X8vH+1me/3f0Uvefdx+El7/crRI5ZTshx95Jbioc/kluf7ugyciSqFBcqEoJskqpCC5VlImwIUEIJUFAmyFBUyhBJEoUBSEASrBVlUNYBJtIBwUrK6uVRzydVLkh2bC8DVHzLd/ksKEbmG42nFN4o+cHZKxIRuYbmbh0h+3zTG9KftP3f0uagJbmNSaOoelj2fP0ASK2Pe60wNwkf7WRCTbZW6T8kucTmSUSoQkKywcjaVUJUO2X21UuUIQFsJQHRkhCYjSzHPFptxAV/+SfrB8fVY1Up2wcn7m49IHsjzVfnj2QsaE9zFuZr+c/aj5vh5rIhG5itmv5kcVYVWnVYkJ7mFm7aCgrGHEZFXbWKFJBY8oS21AVeVadiAqJQoQBQ1FBqJaFnuYElxKhCFIAhCEACEIQAIQhAAhCEATKFCAgdlkKFKCgRKEJUAIUKJTE2ShQhArJJUIQgQIQhAAhCEACEIQAIQhAApBUIQBcPKn4iWhPcwBCEJACEIQAIQhAAhCEACEIQAIQhAAhCEASFKEIKQIQhAyCoQhBDBCEIAEIQgAQhCABCEIAEIQgAQhCABCEIAEIQgD//2Q==' }, 'attributes': { - 'width': '100', - 'height': '100', + 'width': '300', + 'height': '500', 'style': 'width:250px; height:250px;' } }, @@ -57,8 +57,8 @@ final quillImagesSample = [ '' }, 'attributes': { - 'width': '100', - 'height': '100', + 'width': '150', + 'height': '250', 'style': 'width:250px; height:250px;' } }, diff --git a/flutter_quill_extensions/CHANGELOG.md b/flutter_quill_extensions/CHANGELOG.md index cd3ec636f..5cf7f2802 100644 --- a/flutter_quill_extensions/CHANGELOG.md +++ b/flutter_quill_extensions/CHANGELOG.md @@ -2,6 +2,14 @@ All notable changes to this project will be documented in this file. +## 0.8.0-dev +* **Breaking Change**: Completly change the way how the source code structured to more basic and simple way, organize folders and file names, if you use the library +from `flutter_quill_extensions.dart` then there is nothing you need to do, but if you are using any other import then you need to re-imports +embed, this won't affect how quill js work +* Improvemenets to the image embed +* Add support for `margin` for web +* Add untranslated strings to the `quill_en.arb` + ## 0.7.2 * Fix a bug when opening the link dialog for both video and image buttons * Update `README.md` diff --git a/flutter_quill_extensions/README.md b/flutter_quill_extensions/README.md index 8368999b8..400d1e8af 100644 --- a/flutter_quill_extensions/README.md +++ b/flutter_quill_extensions/README.md @@ -14,8 +14,8 @@ to support embedding widgets like images, formulas, videos, and more. - [Platform Specific Configurations](#platform-specific-configurations) - [Usage](#usage) - [Embed Blocks](#embed-blocks) - - [Custom Size Image for Mobile](#custom-size-image-for-mobile) - - [Custom Size Image for other platforms](#custom-size-image-for-other-platforms) + - [Element properties](#element-properties) + - [Custom Element properties](#custom-element-properties) - [Drag and drop feature](#drag-and-drop-feature) - [Features](#features) - [Contributing](#contributing) @@ -130,37 +130,40 @@ As of version [flutter_quill](https://pub.dev/packages/flutter_quill) 6.0, embed The instructions for using the embed blocks are in the [Usage](#usage) section -### Custom Size Image for Mobile +### Element properties -Define `mobileWidth`, `mobileHeight`, `mobileMargin`, `mobileAlignment` as follows: +Currently the library has limitied support for the image and video properties +and it supports only `width`, `height`, `margin` ```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" + "attributes": { + "style":"width: 50px; height: 50px; margin: 10px;" } } ``` -### Custom Size Image for other platforms +### Custom Element properties + +Doesn't apply to official Quill JS -Define `width`, `height`, `margin`, `alignment` as follows: +Define flutterAlignment` 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" + "attributes":{ + "style":"flutterAlignment: topLeft" } } ``` - +This works for all platforms except Web ### Drag and drop feature Currently, the drag-and-drop feature is not officially supported, but you can achieve this very easily in the following steps: diff --git a/flutter_quill_extensions/lib/presentation/embeds/embed_types.dart b/flutter_quill_extensions/lib/embeds/embed_types.dart similarity index 100% rename from flutter_quill_extensions/lib/presentation/embeds/embed_types.dart rename to flutter_quill_extensions/lib/embeds/embed_types.dart diff --git a/flutter_quill_extensions/lib/presentation/embeds/editor/formula.dart b/flutter_quill_extensions/lib/embeds/formula/editor/formula_embed.dart similarity index 90% rename from flutter_quill_extensions/lib/presentation/embeds/editor/formula.dart rename to flutter_quill_extensions/lib/embeds/formula/editor/formula_embed.dart index 41b61e098..baee3c50e 100644 --- a/flutter_quill_extensions/lib/presentation/embeds/editor/formula.dart +++ b/flutter_quill_extensions/lib/embeds/formula/editor/formula_embed.dart @@ -1,7 +1,6 @@ import 'package:flutter/widgets.dart'; -import 'package:flutter_quill/extensions.dart' as base; import 'package:flutter_quill/flutter_quill.dart' - show BlockEmbed, EmbedBuilder, QuillController; + show BlockEmbed, Embed, EmbedBuilder, QuillController; class QuillEditorFormulaEmbedBuilder extends EmbedBuilder { const QuillEditorFormulaEmbedBuilder(); @@ -15,7 +14,7 @@ class QuillEditorFormulaEmbedBuilder extends EmbedBuilder { Widget build( BuildContext context, QuillController controller, - base.Embed node, + Embed node, bool readOnly, bool inline, TextStyle textStyle, diff --git a/flutter_quill_extensions/lib/presentation/embeds/toolbar/formula_button.dart b/flutter_quill_extensions/lib/embeds/formula/toolbar/formula_button.dart similarity index 98% rename from flutter_quill_extensions/lib/presentation/embeds/toolbar/formula_button.dart rename to flutter_quill_extensions/lib/embeds/formula/toolbar/formula_button.dart index e821d689d..935389f6b 100644 --- a/flutter_quill_extensions/lib/presentation/embeds/toolbar/formula_button.dart +++ b/flutter_quill_extensions/lib/embeds/formula/toolbar/formula_button.dart @@ -1,7 +1,7 @@ import 'package:flutter/material.dart'; import 'package:flutter_quill/flutter_quill.dart'; -import '../../models/config/toolbar/buttons/formula.dart'; +import '../../../models/config/toolbar/buttons/formula.dart'; class QuillToolbarFormulaButton extends StatelessWidget { const QuillToolbarFormulaButton({ diff --git a/flutter_quill_extensions/lib/embeds/image/editor/image_embed.dart b/flutter_quill_extensions/lib/embeds/image/editor/image_embed.dart new file mode 100644 index 000000000..2ddc38dbf --- /dev/null +++ b/flutter_quill_extensions/lib/embeds/image/editor/image_embed.dart @@ -0,0 +1,86 @@ +import 'package:flutter/foundation.dart' show kIsWeb; +import 'package:flutter/material.dart'; +import 'package:flutter_quill/flutter_quill.dart' hide OptionalSize; +import 'package:flutter_quill/translations.dart'; + +import '../../../models/config/editor/image/image.dart'; +import '../../../models/config/shared_configurations.dart'; +import '../../../utils/element_utils/element_utils.dart'; +import '../../widgets/image.dart'; +import 'image_menu.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, + Embed node, + bool readOnly, + bool inline, + TextStyle textStyle, + ) { + assert(!kIsWeb, 'Please provide image EmbedBuilder for Web'); + + final imageSource = standardizeImageUrl(node.value.data); + final ((imageSize), margin, alignment) = getElementAttributes(node); + + final width = imageSize.width; + final height = imageSize.height; + + final image = getImageWidgetByImageSource( + imageSource, + imageProviderBuilder: configurations.imageProviderBuilder, + imageErrorWidgetBuilder: configurations.imageErrorWidgetBuilder, + alignment: alignment, + height: height, + width: width, + assetsPrefix: QuillSharedExtensionsConfigurations.get(context: context) + .assetsPrefix, + ); + + final imageSaverService = + QuillSharedExtensionsConfigurations.get(context: context) + .imageSaverService; + return GestureDetector( + onTap: configurations.onImageClicked ?? + () => showDialog( + context: context, + builder: (_) => QuillProvider.value( + value: context.requireQuillProvider, + child: FlutterQuillLocalizationsWidget( + child: ImageOptionsMenu( + controller: controller, + configurations: configurations, + imageSource: imageSource, + imageSize: imageSize, + isReadOnly: readOnly, + imageSaverService: imageSaverService, + ), + ), + ), + ), + child: Builder( + builder: (context) { + if (margin != null) { + return Padding( + padding: EdgeInsets.all(margin), + child: image, + ); + } + return image; + }, + ), + ); + } +} diff --git a/flutter_quill_extensions/lib/presentation/embeds/embed_types/image.dart b/flutter_quill_extensions/lib/embeds/image/editor/image_embed_types.dart similarity index 95% rename from flutter_quill_extensions/lib/presentation/embeds/embed_types/image.dart rename to flutter_quill_extensions/lib/embeds/image/editor/image_embed_types.dart index c53796efa..44d2dfaa1 100644 --- a/flutter_quill_extensions/lib/presentation/embeds/embed_types/image.dart +++ b/flutter_quill_extensions/lib/embeds/image/editor/image_embed_types.dart @@ -4,8 +4,8 @@ import 'package:flutter/widgets.dart' show BuildContext; import 'package:flutter_quill/flutter_quill.dart'; import 'package:meta/meta.dart' show immutable; -import '../../../logic/extensions/controller.dart'; -import '../../../logic/services/image_picker/s_image_picker.dart'; +import '../../../extensions/controller.dart'; +import '../../../services/image_picker/s_image_picker.dart'; /// When request picking an image, for example when the image button toolbar /// clicked, it should be null in case the user didn't choose any image or diff --git a/flutter_quill_extensions/lib/presentation/embeds/editor/image/image_menu.dart b/flutter_quill_extensions/lib/embeds/image/editor/image_menu.dart similarity index 94% rename from flutter_quill_extensions/lib/presentation/embeds/editor/image/image_menu.dart rename to flutter_quill_extensions/lib/embeds/image/editor/image_menu.dart index 1ee3201ad..27c83ecef 100644 --- a/flutter_quill_extensions/lib/presentation/embeds/editor/image/image_menu.dart +++ b/flutter_quill_extensions/lib/embeds/image/editor/image_menu.dart @@ -1,9 +1,7 @@ import 'package:flutter/cupertino.dart' show showCupertinoModalPopup; import 'package:flutter/material.dart'; -import 'package:flutter_quill/extensions.dart' show isMobile; import 'package:flutter_quill/flutter_quill.dart' show - FlutterQuillLocalizationsWidget, ImageUrl, QuillController, QuillProvider, @@ -12,10 +10,11 @@ import 'package:flutter_quill/flutter_quill.dart' getEmbedNode; import 'package:flutter_quill/translations.dart'; -import '../../../../logic/models/config/shared_configurations.dart'; -import '../../../../logic/services/image_saver/s_image_saver.dart'; -import '../../../../logic/utils/string.dart'; import '../../../models/config/editor/image/image.dart'; +import '../../../models/config/shared_configurations.dart'; +import '../../../services/image_saver/s_image_saver.dart'; +import '../../../utils/element_utils/element_utils.dart'; +import '../../../utils/string.dart'; import '../../../utils/utils.dart'; import '../../widgets/image.dart' show ImageTapWrapper, getImageStyleString; import '../../widgets/image_resizer.dart' show ImageResizer; @@ -34,7 +33,7 @@ class ImageOptionsMenu extends StatelessWidget { final QuillController controller; final QuillEditorImageEmbedConfigurations configurations; final String imageSource; - final OptionalSize imageSize; + final ElementSize imageSize; final bool isReadOnly; final ImageSaverService imageSaverService; @@ -70,7 +69,6 @@ class ImageOptionsMenu extends StatelessWidget { getImageStyleString(controller), width: width, height: height, - isMobile: isMobile(supportWeb: false), ); controller ..skipRequestKeyboard = true diff --git a/flutter_quill_extensions/lib/presentation/embeds/editor/image/image_web.dart b/flutter_quill_extensions/lib/embeds/image/editor/image_web_embed.dart similarity index 88% rename from flutter_quill_extensions/lib/presentation/embeds/editor/image/image_web.dart rename to flutter_quill_extensions/lib/embeds/image/editor/image_web_embed.dart index 2ed5b967e..96c95ee76 100644 --- a/flutter_quill_extensions/lib/presentation/embeds/editor/image/image_web.dart +++ b/flutter_quill_extensions/lib/embeds/image/editor/image_web_embed.dart @@ -4,10 +4,10 @@ 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_ui/dart_ui_fake.dart' + if (dart.library.html) '../../../utils/dart_ui/dart_ui_real.dart' as ui; +import '../../../utils/element_utils/element_web_utils.dart'; import '../../../utils/utils.dart'; -import '../../../utils/web_utils.dart'; -import '../shims/dart_ui_fake.dart' - if (dart.library.html) '../shims/dart_ui_real.dart' as ui; class QuillEditorWebImageEmbedBuilder extends EmbedBuilder { const QuillEditorWebImageEmbedBuilder({ @@ -43,7 +43,7 @@ class QuillEditorWebImageEmbedBuilder extends EmbedBuilder { // 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 + // doesn't displayed with the 'data:image/png;base64' if (!(imageSource.startsWith('data:image/') && imageSource.contains('base64'))) { imageSource = 'data:image/png;base64, $imageSource'; diff --git a/flutter_quill_extensions/lib/presentation/embeds/toolbar/image_button/image_button.dart b/flutter_quill_extensions/lib/embeds/image/toolbar/image_button.dart similarity index 96% rename from flutter_quill_extensions/lib/presentation/embeds/toolbar/image_button/image_button.dart rename to flutter_quill_extensions/lib/embeds/image/toolbar/image_button.dart index 552211b19..fb00a6564 100644 --- a/flutter_quill_extensions/lib/presentation/embeds/toolbar/image_button/image_button.dart +++ b/flutter_quill_extensions/lib/embeds/image/toolbar/image_button.dart @@ -4,11 +4,11 @@ import 'package:flutter/material.dart'; import 'package:flutter_quill/flutter_quill.dart'; import 'package:flutter_quill/translations.dart'; -import '../../../../logic/models/config/shared_configurations.dart'; -import '../../../../logic/services/image_picker/image_picker.dart'; +import '../../../models/config/shared_configurations.dart'; import '../../../models/config/toolbar/buttons/image.dart'; -import '../../embed_types/image.dart'; -import '../utils/image_video_utils.dart'; +import '../../../services/image_picker/image_picker.dart'; +import '../../others/image_video_utils.dart'; +import '../editor/image_embed_types.dart'; import 'select_image_source.dart'; class QuillToolbarImageButton extends StatelessWidget { diff --git a/flutter_quill_extensions/lib/presentation/embeds/toolbar/image_button/select_image_source.dart b/flutter_quill_extensions/lib/embeds/image/toolbar/select_image_source.dart similarity index 65% rename from flutter_quill_extensions/lib/presentation/embeds/toolbar/image_button/select_image_source.dart rename to flutter_quill_extensions/lib/embeds/image/toolbar/select_image_source.dart index d40fd8f14..74de2c819 100644 --- a/flutter_quill_extensions/lib/presentation/embeds/toolbar/image_button/select_image_source.dart +++ b/flutter_quill_extensions/lib/embeds/image/toolbar/select_image_source.dart @@ -1,7 +1,9 @@ import 'package:flutter/material.dart'; import 'package:flutter_quill/extensions.dart' show isDesktop; +import 'package:flutter_quill/flutter_quill.dart'; +import 'package:flutter_quill/translations.dart'; -import '../../embed_types/image.dart'; +import '../editor/image_embed_types.dart'; class SelectImageSourceDialog extends StatelessWidget { const SelectImageSourceDialog({super.key}); @@ -14,28 +16,27 @@ class SelectImageSourceDialog extends StatelessWidget { child: SingleChildScrollView( child: Column( children: [ - // TODO: Needs to be translated ListTile( - title: const Text('Gallery'), - subtitle: const Text( - 'Pick a photo from your gallery', + title: Text(context.loc.gallery), + subtitle: Text( + context.loc.pickAPhotoFromYourGallery, ), leading: const Icon(Icons.photo_sharp), onTap: () => Navigator.of(context).pop(InsertImageSource.gallery), ), ListTile( - title: const Text('Camera'), - subtitle: const Text( - 'Take a photo using your phone camera', + title: Text(context.loc.camera), + subtitle: Text( + context.loc.takeAPhotoUsingYourCamera, ), leading: const Icon(Icons.camera), enabled: !isDesktop(supportWeb: false), onTap: () => Navigator.of(context).pop(InsertImageSource.camera), ), ListTile( - title: const Text('Link'), - subtitle: const Text( - 'Paste a photo using a link', + title: Text(context.loc.link), + subtitle: Text( + context.loc.pasteAPhotoUsingALink, ), leading: const Icon(Icons.link), onTap: () => Navigator.of(context).pop(InsertImageSource.link), @@ -54,7 +55,12 @@ Future showSelectImageSourceDialog({ showDragHandle: true, context: context, constraints: const BoxConstraints(maxWidth: 640), - builder: (_) => const SelectImageSourceDialog(), + builder: (_) => QuillProvider.value( + value: context.requireQuillProvider, + child: const FlutterQuillLocalizationsWidget( + child: SelectImageSourceDialog(), + ), + ), ); return imageSource; } diff --git a/flutter_quill_extensions/lib/presentation/embeds/toolbar/camera_button/camera_button.dart b/flutter_quill_extensions/lib/embeds/others/camera_button/camera_button.dart similarity index 96% rename from flutter_quill_extensions/lib/presentation/embeds/toolbar/camera_button/camera_button.dart rename to flutter_quill_extensions/lib/embeds/others/camera_button/camera_button.dart index 937d64435..c62976cb4 100644 --- a/flutter_quill_extensions/lib/presentation/embeds/toolbar/camera_button/camera_button.dart +++ b/flutter_quill_extensions/lib/embeds/others/camera_button/camera_button.dart @@ -8,10 +8,10 @@ import 'package:flutter_quill/flutter_quill.dart' QuillToolbarIconButton; import 'package:flutter_quill/translations.dart'; -import '../../../../logic/models/config/shared_configurations.dart'; -import '../../../../logic/services/image_picker/image_options.dart'; +import '../../../models/config/shared_configurations.dart'; import '../../../models/config/toolbar/buttons/camera.dart'; -import '../../embed_types/camera.dart'; +import '../../../services/image_picker/image_options.dart'; +import 'camera_types.dart'; import 'select_camera_action.dart'; class QuillToolbarCameraButton extends StatelessWidget { diff --git a/flutter_quill_extensions/lib/presentation/embeds/embed_types/camera.dart b/flutter_quill_extensions/lib/embeds/others/camera_button/camera_types.dart similarity index 94% rename from flutter_quill_extensions/lib/presentation/embeds/embed_types/camera.dart rename to flutter_quill_extensions/lib/embeds/others/camera_button/camera_types.dart index 1e2931230..28f7c5aac 100644 --- a/flutter_quill_extensions/lib/presentation/embeds/embed_types/camera.dart +++ b/flutter_quill_extensions/lib/embeds/others/camera_button/camera_types.dart @@ -1,8 +1,8 @@ import 'package:flutter/widgets.dart' show BuildContext; import 'package:meta/meta.dart' show immutable; -import 'image.dart'; -import 'video.dart'; +import '../../image/editor/image_embed_types.dart'; +import '../../video/video.dart'; enum CameraAction { video, diff --git a/flutter_quill_extensions/lib/presentation/embeds/toolbar/camera_button/select_camera_action.dart b/flutter_quill_extensions/lib/embeds/others/camera_button/select_camera_action.dart similarity index 95% rename from flutter_quill_extensions/lib/presentation/embeds/toolbar/camera_button/select_camera_action.dart rename to flutter_quill_extensions/lib/embeds/others/camera_button/select_camera_action.dart index f8823cf13..bd6527db7 100644 --- a/flutter_quill_extensions/lib/presentation/embeds/toolbar/camera_button/select_camera_action.dart +++ b/flutter_quill_extensions/lib/embeds/others/camera_button/select_camera_action.dart @@ -1,7 +1,7 @@ import 'package:flutter/material.dart'; import 'package:flutter_quill/translations.dart'; -import '../../embed_types/camera.dart'; +import 'camera_types.dart'; class SelectCameraActionDialog extends StatelessWidget { const SelectCameraActionDialog({super.key}); diff --git a/flutter_quill_extensions/lib/presentation/embeds/toolbar/utils/image_video_utils.dart b/flutter_quill_extensions/lib/embeds/others/image_video_utils.dart similarity index 100% rename from flutter_quill_extensions/lib/presentation/embeds/toolbar/utils/image_video_utils.dart rename to flutter_quill_extensions/lib/embeds/others/image_video_utils.dart diff --git a/flutter_quill_extensions/lib/presentation/embeds/toolbar/media_button/media_button.dart b/flutter_quill_extensions/lib/embeds/others/media_button/media_button.dart similarity index 100% rename from flutter_quill_extensions/lib/presentation/embeds/toolbar/media_button/media_button.dart rename to flutter_quill_extensions/lib/embeds/others/media_button/media_button.dart diff --git a/flutter_quill_extensions/lib/presentation/embeds/editor/unknown.dart b/flutter_quill_extensions/lib/embeds/unknown/editor/unknown_embed.dart similarity index 100% rename from flutter_quill_extensions/lib/presentation/embeds/editor/unknown.dart rename to flutter_quill_extensions/lib/embeds/unknown/editor/unknown_embed.dart diff --git a/flutter_quill_extensions/lib/presentation/embeds/editor/video/video.dart b/flutter_quill_extensions/lib/embeds/video/editor/video_embed.dart similarity index 95% rename from flutter_quill_extensions/lib/presentation/embeds/editor/video/video.dart rename to flutter_quill_extensions/lib/embeds/video/editor/video_embed.dart index a4b2a9b80..831da538b 100644 --- a/flutter_quill_extensions/lib/presentation/embeds/editor/video/video.dart +++ b/flutter_quill_extensions/lib/embeds/video/editor/video_embed.dart @@ -1,9 +1,9 @@ import 'package:flutter/foundation.dart' show kIsWeb; import 'package:flutter/material.dart'; -import 'package:flutter_quill/extensions.dart' as base; import 'package:flutter_quill/flutter_quill.dart'; import '../../../models/config/editor/video/video.dart'; +import '../../../utils/element_utils/element_utils.dart'; import '../../../utils/utils.dart'; import '../../widgets/video_app.dart'; import '../../widgets/youtube_video_app.dart'; @@ -22,7 +22,7 @@ class QuillEditorVideoEmbedBuilder extends EmbedBuilder { Widget build( BuildContext context, QuillController controller, - base.Embed node, + Embed node, bool readOnly, bool inline, TextStyle textStyle, diff --git a/flutter_quill_extensions/lib/presentation/embeds/editor/video/video_web.dart b/flutter_quill_extensions/lib/embeds/video/editor/video_web_embed.dart similarity index 89% rename from flutter_quill_extensions/lib/presentation/embeds/editor/video/video_web.dart rename to flutter_quill_extensions/lib/embeds/video/editor/video_web_embed.dart index abc821d08..650cd5431 100644 --- a/flutter_quill_extensions/lib/presentation/embeds/editor/video/video_web.dart +++ b/flutter_quill_extensions/lib/embeds/video/editor/video_web_embed.dart @@ -5,10 +5,10 @@ import 'package:youtube_player_flutter/youtube_player_flutter.dart' show YoutubePlayer; import '../../../models/config/editor/video/video_web.dart'; +import '../../../utils/dart_ui/dart_ui_fake.dart' + if (dart.library.html) '../../../utils/dart_ui/dart_ui_real.dart' as ui; +import '../../../utils/element_utils/element_web_utils.dart'; import '../../../utils/utils.dart'; -import '../../../utils/web_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({ diff --git a/flutter_quill_extensions/lib/presentation/embeds/toolbar/video_button/select_video_source.dart b/flutter_quill_extensions/lib/embeds/video/toolbar/select_video_source.dart similarity index 74% rename from flutter_quill_extensions/lib/presentation/embeds/toolbar/video_button/select_video_source.dart rename to flutter_quill_extensions/lib/embeds/video/toolbar/select_video_source.dart index 78fe6177b..f4c030c9f 100644 --- a/flutter_quill_extensions/lib/presentation/embeds/toolbar/video_button/select_video_source.dart +++ b/flutter_quill_extensions/lib/embeds/video/toolbar/select_video_source.dart @@ -1,7 +1,8 @@ import 'package:flutter/material.dart'; import 'package:flutter_quill/extensions.dart' show isDesktop; +import 'package:flutter_quill/translations.dart'; -import '../../embed_types/video.dart'; +import '../video.dart'; class SelectVideoSourceDialog extends StatelessWidget { const SelectVideoSourceDialog({super.key}); @@ -14,28 +15,25 @@ class SelectVideoSourceDialog extends StatelessWidget { child: SingleChildScrollView( child: Column( children: [ - // TODO: Needs to be translated ListTile( - title: const Text('Gallery'), - subtitle: const Text( - 'Pick a video from your gallery', + title: Text(context.loc.gallery), + subtitle: Text( + context.loc.pickAVideoFromYourGallery, ), leading: const Icon(Icons.photo_sharp), onTap: () => Navigator.of(context).pop(InsertVideoSource.gallery), ), ListTile( - title: const Text('Camera'), - subtitle: const Text( - 'Record a video using your phone camera', - ), + title: Text(context.loc.camera), + subtitle: Text(context.loc.recordAVideoUsingYourCamera), leading: const Icon(Icons.camera), enabled: !isDesktop(supportWeb: false), onTap: () => Navigator.of(context).pop(InsertVideoSource.camera), ), ListTile( - title: const Text('Link'), - subtitle: const Text( - 'Paste a video using a link', + title: Text(context.loc.link), + subtitle: Text( + context.loc.pasteAVideoUsingALink, ), leading: const Icon(Icons.link), onTap: () => Navigator.of(context).pop(InsertVideoSource.link), diff --git a/flutter_quill_extensions/lib/presentation/embeds/toolbar/video_button/video_button.dart b/flutter_quill_extensions/lib/embeds/video/toolbar/video_button.dart similarity index 96% rename from flutter_quill_extensions/lib/presentation/embeds/toolbar/video_button/video_button.dart rename to flutter_quill_extensions/lib/embeds/video/toolbar/video_button.dart index dd2fb8f58..f8b854b3c 100644 --- a/flutter_quill_extensions/lib/presentation/embeds/toolbar/video_button/video_button.dart +++ b/flutter_quill_extensions/lib/embeds/video/toolbar/video_button.dart @@ -2,12 +2,13 @@ import 'package:flutter/material.dart'; import 'package:flutter_quill/flutter_quill.dart'; +import 'package:flutter_quill/translations.dart'; -import '../../../../logic/models/config/shared_configurations.dart'; -import '../../../../logic/services/image_picker/image_options.dart'; +import '../../../models/config/shared_configurations.dart'; import '../../../models/config/toolbar/buttons/video.dart'; -import '../../embed_types/video.dart'; -import '../utils/image_video_utils.dart'; +import '../../../services/image_picker/image_options.dart'; +import '../../others/image_video_utils.dart'; +import '../video.dart'; import 'select_video_source.dart'; class QuillToolbarVideoButton extends StatelessWidget { diff --git a/flutter_quill_extensions/lib/presentation/embeds/embed_types/video.dart b/flutter_quill_extensions/lib/embeds/video/video.dart similarity index 94% rename from flutter_quill_extensions/lib/presentation/embeds/embed_types/video.dart rename to flutter_quill_extensions/lib/embeds/video/video.dart index d69ab5e05..b2e1ca24e 100644 --- a/flutter_quill_extensions/lib/presentation/embeds/embed_types/video.dart +++ b/flutter_quill_extensions/lib/embeds/video/video.dart @@ -2,8 +2,8 @@ import 'package:flutter/widgets.dart' show BuildContext; import 'package:flutter_quill/flutter_quill.dart'; import 'package:meta/meta.dart' show immutable; -import '../../../logic/extensions/controller.dart'; -import '../../../logic/services/image_picker/s_image_picker.dart'; +import '../../extensions/controller.dart'; +import '../../services/image_picker/s_image_picker.dart'; /// When request picking an video, for example when the video button toolbar /// clicked, it should be null in case the user didn't choose any video or diff --git a/flutter_quill_extensions/lib/presentation/embeds/widgets/image.dart b/flutter_quill_extensions/lib/embeds/widgets/image.dart similarity index 99% rename from flutter_quill_extensions/lib/presentation/embeds/widgets/image.dart rename to flutter_quill_extensions/lib/embeds/widgets/image.dart index 13c77eb9d..d072f202b 100644 --- a/flutter_quill_extensions/lib/presentation/embeds/widgets/image.dart +++ b/flutter_quill_extensions/lib/embeds/widgets/image.dart @@ -7,7 +7,7 @@ import 'package:photo_view/photo_view.dart'; import '../../models/config/editor/image/image.dart'; import '../../utils/utils.dart'; -import '../embed_types/image.dart'; +import '../image/editor/image_embed_types.dart'; const List imageFileExtensions = [ '.jpeg', diff --git a/flutter_quill_extensions/lib/presentation/embeds/widgets/image_resizer.dart b/flutter_quill_extensions/lib/embeds/widgets/image_resizer.dart similarity index 100% rename from flutter_quill_extensions/lib/presentation/embeds/widgets/image_resizer.dart rename to flutter_quill_extensions/lib/embeds/widgets/image_resizer.dart diff --git a/flutter_quill_extensions/lib/presentation/embeds/widgets/video_app.dart b/flutter_quill_extensions/lib/embeds/widgets/video_app.dart similarity index 98% rename from flutter_quill_extensions/lib/presentation/embeds/widgets/video_app.dart rename to flutter_quill_extensions/lib/embeds/widgets/video_app.dart index 5b794394b..334f64ca1 100644 --- a/flutter_quill_extensions/lib/presentation/embeds/widgets/video_app.dart +++ b/flutter_quill_extensions/lib/embeds/widgets/video_app.dart @@ -6,7 +6,7 @@ import 'package:flutter_quill/flutter_quill.dart'; import 'package:url_launcher/url_launcher.dart'; import 'package:video_player/video_player.dart'; -import '../../../flutter_quill_extensions.dart'; +import '../../flutter_quill_extensions.dart'; /// Widget for playing back video /// Refer to https://github.com/flutter/plugins/tree/master/packages/video_player/video_player diff --git a/flutter_quill_extensions/lib/presentation/embeds/widgets/youtube_video_app.dart b/flutter_quill_extensions/lib/embeds/widgets/youtube_video_app.dart similarity index 100% rename from flutter_quill_extensions/lib/presentation/embeds/widgets/youtube_video_app.dart rename to flutter_quill_extensions/lib/embeds/widgets/youtube_video_app.dart diff --git a/flutter_quill_extensions/lib/extensions/attribute.dart b/flutter_quill_extensions/lib/extensions/attribute.dart new file mode 100644 index 000000000..2467be3f2 --- /dev/null +++ b/flutter_quill_extensions/lib/extensions/attribute.dart @@ -0,0 +1,32 @@ +import 'package:flutter_quill/flutter_quill.dart' + show Attribute, AttributeScope; + +// class FlutterWidthAttribute extends Attribute { +// const FlutterWidthAttribute(String? val) +// : super('flutterWidth', AttributeScope.ignore, val); +// } + +// class FlutterHeightAttribute extends Attribute { +// const FlutterHeightAttribute(String? val) +// : super('flutterHeight', AttributeScope.ignore, val); +// } + +// class FlutterMarginAttribute extends Attribute { +// const FlutterMarginAttribute(String? val) +// : super('flutterMargin', AttributeScope.ignore, val); +// } + +class FlutterAlignmentAttribute extends Attribute { + const FlutterAlignmentAttribute(String? val) + : super('flutterAlignment', AttributeScope.ignore, val); +} + +extension AttributeExt on Attribute { + // static const FlutterWidthAttribute flutterWidth = FlutterWidthAttribute(null); + // static const FlutterHeightAttribute flutterHeight = + // FlutterHeightAttribute(null); + // static const FlutterMarginAttribute flutterMargin = + // FlutterMarginAttribute(null); + static const FlutterAlignmentAttribute flutterAlignment = + FlutterAlignmentAttribute(null); +} diff --git a/flutter_quill_extensions/lib/logic/extensions/controller.dart b/flutter_quill_extensions/lib/extensions/controller.dart similarity index 70% rename from flutter_quill_extensions/lib/logic/extensions/controller.dart rename to flutter_quill_extensions/lib/extensions/controller.dart index 474f3cf44..d1687ea7c 100644 --- a/flutter_quill_extensions/lib/logic/extensions/controller.dart +++ b/flutter_quill_extensions/lib/extensions/controller.dart @@ -1,7 +1,5 @@ import 'package:flutter_quill/flutter_quill.dart'; -// ignore: unused_import -import '../../presentation/embeds/editor/webview.dart'; import '../utils/quill_image_utils.dart'; /// Extension functions on [QuillController] @@ -12,27 +10,6 @@ extension QuillControllerExt on QuillController { int get index => selection.baseOffset; int get length => selection.extentOffset - index; - /// Insert webview embed block, it requires [initialUrl] to load - /// the initial page - // void insertWebViewBlock({ - // required String initialUrl, - // }) { - // final block = BlockEmbed.custom( - // QuillEditorWebViewBlockEmbed( - // initialUrl, - // ), - // ); - - // this - // ..skipRequestKeyboard = true - // ..replaceText( - // index, - // length, - // block, - // null, - // ); - // } - /// Insert image embed block, it requires the [imageSource] /// /// it could be local image on the system file diff --git a/flutter_quill_extensions/lib/flutter_quill_extensions.dart b/flutter_quill_extensions/lib/flutter_quill_extensions.dart index 22a69a4fe..b3f7f3bd0 100644 --- a/flutter_quill_extensions/lib/flutter_quill_extensions.dart +++ b/flutter_quill_extensions/lib/flutter_quill_extensions.dart @@ -6,52 +6,49 @@ import 'package:flutter/foundation.dart' show kIsWeb; import 'package:flutter_quill/flutter_quill.dart'; import 'package:meta/meta.dart' show immutable; -import 'presentation/embeds/editor/image/image.dart'; -import 'presentation/embeds/editor/image/image_web.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/image_button/image_button.dart'; -import 'presentation/embeds/toolbar/video_button/video_button.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/image.dart'; -import 'presentation/models/config/toolbar/buttons/media_button.dart'; -import 'presentation/models/config/toolbar/buttons/video.dart'; +import 'embeds/image/editor/image_embed.dart'; +import 'embeds/image/editor/image_web_embed.dart'; +import 'embeds/image/toolbar/image_button.dart'; +import 'embeds/others/camera_button/camera_button.dart'; +import 'embeds/video/editor/video_embed.dart'; +import 'embeds/video/editor/video_web_embed.dart'; +import 'embeds/video/toolbar/video_button.dart'; +import 'models/config/editor/image/image.dart'; +import 'models/config/editor/image/image_web.dart'; +import 'models/config/editor/video/video.dart'; +import 'models/config/editor/video/video_web.dart'; +import 'models/config/editor/webview.dart'; +import 'models/config/toolbar/buttons/camera.dart'; +import 'models/config/toolbar/buttons/image.dart'; +import 'models/config/toolbar/buttons/media_button.dart'; +import 'models/config/toolbar/buttons/video.dart'; -export '/logic/extensions/controller.dart'; -export '/presentation/models/config/editor/webview.dart'; -export 'logic/models/config/shared_configurations.dart'; -export 'presentation/embeds/editor/image/image.dart'; -export 'presentation/embeds/editor/image/image_web.dart'; -export 'presentation/embeds/editor/unknown.dart'; -export 'presentation/embeds/editor/video/video.dart'; -export 'presentation/embeds/editor/video/video_web.dart'; -export 'presentation/embeds/editor/webview.dart'; -export 'presentation/embeds/embed_types.dart'; -export 'presentation/embeds/embed_types/image.dart'; -export 'presentation/embeds/embed_types/video.dart'; -export 'presentation/embeds/toolbar/camera_button/camera_button.dart'; -export 'presentation/embeds/toolbar/formula_button.dart'; -export 'presentation/embeds/toolbar/image_button/image_button.dart'; -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/models/config/editor/image/image.dart'; -export 'presentation/models/config/editor/image/image_web.dart'; -export 'presentation/models/config/editor/video/video.dart'; -export 'presentation/models/config/editor/video/video_web.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'; -export 'presentation/models/config/toolbar/buttons/media_button.dart'; -export 'presentation/models/config/toolbar/buttons/video.dart'; -export 'presentation/utils/utils.dart'; +export 'embeds/embed_types.dart'; +export 'embeds/formula/toolbar/formula_button.dart'; +export 'embeds/image/editor/image_embed.dart'; +export 'embeds/image/editor/image_embed_types.dart'; +export 'embeds/image/editor/image_web_embed.dart'; +export 'embeds/image/toolbar/image_button.dart'; +export 'embeds/others/camera_button/camera_button.dart'; +export 'embeds/others/media_button/media_button.dart'; +export 'embeds/unknown/editor/unknown_embed.dart'; +export 'embeds/video/editor/video_embed.dart'; +export 'embeds/video/editor/video_web_embed.dart'; +export 'embeds/video/toolbar/video_button.dart'; +export 'embeds/video/video.dart'; +export 'extensions/controller.dart'; +export 'models/config/editor/image/image.dart'; +export 'models/config/editor/image/image_web.dart'; +export 'models/config/editor/video/video.dart'; +export 'models/config/editor/video/video_web.dart'; +export 'models/config/editor/webview.dart'; +export 'models/config/shared_configurations.dart'; +export 'models/config/toolbar/buttons/camera.dart'; +export 'models/config/toolbar/buttons/formula.dart'; +export 'models/config/toolbar/buttons/image.dart'; +export 'models/config/toolbar/buttons/media_button.dart'; +export 'models/config/toolbar/buttons/video.dart'; +export 'utils/utils.dart'; @immutable class FlutterQuillEmbeds { diff --git a/flutter_quill_extensions/lib/logic/extensions/attribute.dart b/flutter_quill_extensions/lib/logic/extensions/attribute.dart deleted file mode 100644 index fd939e774..000000000 --- a/flutter_quill_extensions/lib/logic/extensions/attribute.dart +++ /dev/null @@ -1,30 +0,0 @@ -import 'package:flutter_quill/flutter_quill.dart' - show Attribute, AttributeScope; - -class MobileWidthAttribute extends Attribute { - const MobileWidthAttribute(String? val) - : super('mobileWidth', AttributeScope.ignore, val); -} - -class MobileHeightAttribute extends Attribute { - const MobileHeightAttribute(String? val) - : super('mobileHeight', AttributeScope.ignore, val); -} - -class MobileMarginAttribute extends Attribute { - const MobileMarginAttribute(String? val) - : super('mobileMargin', AttributeScope.ignore, val); -} - -class MobileAlignmentAttribute extends Attribute { - const MobileAlignmentAttribute(String? val) - : super('mobileAlignment', AttributeScope.ignore, val); -} - -extension AttributeExt on Attribute { - static const MobileWidthAttribute mobileWidth = MobileWidthAttribute(null); - static const MobileHeightAttribute mobileHeight = MobileHeightAttribute(null); - static const MobileMarginAttribute mobileMargin = MobileMarginAttribute(null); - static const MobileAlignmentAttribute mobileAlignment = - MobileAlignmentAttribute(null); -} diff --git a/flutter_quill_extensions/lib/presentation/models/config/editor/image/image.dart b/flutter_quill_extensions/lib/models/config/editor/image/image.dart similarity index 98% rename from flutter_quill_extensions/lib/presentation/models/config/editor/image/image.dart rename to flutter_quill_extensions/lib/models/config/editor/image/image.dart index 8d6646f0a..208568bd1 100644 --- a/flutter_quill_extensions/lib/presentation/models/config/editor/image/image.dart +++ b/flutter_quill_extensions/lib/models/config/editor/image/image.dart @@ -4,7 +4,7 @@ import 'package:flutter/foundation.dart' show VoidCallback; import 'package:flutter_quill/extensions.dart'; import 'package:meta/meta.dart' show immutable; -import '../../../../embeds/embed_types/image.dart'; +import '../../../../embeds/image/editor/image_embed_types.dart'; /// [QuillEditorImageEmbedConfigurations] for desktop, mobile and /// other platforms diff --git a/flutter_quill_extensions/lib/presentation/models/config/editor/image/image_web.dart b/flutter_quill_extensions/lib/models/config/editor/image/image_web.dart similarity index 100% rename from flutter_quill_extensions/lib/presentation/models/config/editor/image/image_web.dart rename to flutter_quill_extensions/lib/models/config/editor/image/image_web.dart diff --git a/flutter_quill_extensions/lib/presentation/models/config/editor/video/video.dart b/flutter_quill_extensions/lib/models/config/editor/video/video.dart similarity index 100% rename from flutter_quill_extensions/lib/presentation/models/config/editor/video/video.dart rename to flutter_quill_extensions/lib/models/config/editor/video/video.dart diff --git a/flutter_quill_extensions/lib/presentation/models/config/editor/video/video_web.dart b/flutter_quill_extensions/lib/models/config/editor/video/video_web.dart similarity index 100% rename from flutter_quill_extensions/lib/presentation/models/config/editor/video/video_web.dart rename to flutter_quill_extensions/lib/models/config/editor/video/video_web.dart diff --git a/flutter_quill_extensions/lib/presentation/models/config/editor/webview.dart b/flutter_quill_extensions/lib/models/config/editor/webview.dart similarity index 100% rename from flutter_quill_extensions/lib/presentation/models/config/editor/webview.dart rename to flutter_quill_extensions/lib/models/config/editor/webview.dart diff --git a/flutter_quill_extensions/lib/logic/models/config/shared_configurations.dart b/flutter_quill_extensions/lib/models/config/shared_configurations.dart similarity index 100% rename from flutter_quill_extensions/lib/logic/models/config/shared_configurations.dart rename to flutter_quill_extensions/lib/models/config/shared_configurations.dart diff --git a/flutter_quill_extensions/lib/presentation/models/config/toolbar/buttons/camera.dart b/flutter_quill_extensions/lib/models/config/toolbar/buttons/camera.dart similarity index 93% rename from flutter_quill_extensions/lib/presentation/models/config/toolbar/buttons/camera.dart rename to flutter_quill_extensions/lib/models/config/toolbar/buttons/camera.dart index dafdf1abe..32e2f7549 100644 --- a/flutter_quill_extensions/lib/presentation/models/config/toolbar/buttons/camera.dart +++ b/flutter_quill_extensions/lib/models/config/toolbar/buttons/camera.dart @@ -1,7 +1,7 @@ import 'package:flutter/widgets.dart' show Color; import 'package:flutter_quill/flutter_quill.dart'; -import '../../../../embeds/embed_types/camera.dart'; +import '../../../../embeds/others/camera_button/camera_types.dart'; class QuillToolbarCameraButtonExtraOptions extends QuillToolbarBaseButtonExtraOptions { diff --git a/flutter_quill_extensions/lib/presentation/models/config/toolbar/buttons/formula.dart b/flutter_quill_extensions/lib/models/config/toolbar/buttons/formula.dart similarity index 100% rename from flutter_quill_extensions/lib/presentation/models/config/toolbar/buttons/formula.dart rename to flutter_quill_extensions/lib/models/config/toolbar/buttons/formula.dart diff --git a/flutter_quill_extensions/lib/presentation/models/config/toolbar/buttons/image.dart b/flutter_quill_extensions/lib/models/config/toolbar/buttons/image.dart similarity index 95% rename from flutter_quill_extensions/lib/presentation/models/config/toolbar/buttons/image.dart rename to flutter_quill_extensions/lib/models/config/toolbar/buttons/image.dart index d1e5efbd1..61e8901a0 100644 --- a/flutter_quill_extensions/lib/presentation/models/config/toolbar/buttons/image.dart +++ b/flutter_quill_extensions/lib/models/config/toolbar/buttons/image.dart @@ -2,7 +2,7 @@ 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/image.dart'; +import '../../../../embeds/image/editor/image_embed_types.dart'; class QuillToolbarImageButtonExtraOptions extends QuillToolbarBaseButtonExtraOptions { diff --git a/flutter_quill_extensions/lib/presentation/models/config/toolbar/buttons/media_button.dart b/flutter_quill_extensions/lib/models/config/toolbar/buttons/media_button.dart similarity index 100% rename from flutter_quill_extensions/lib/presentation/models/config/toolbar/buttons/media_button.dart rename to flutter_quill_extensions/lib/models/config/toolbar/buttons/media_button.dart diff --git a/flutter_quill_extensions/lib/presentation/models/config/toolbar/buttons/video.dart b/flutter_quill_extensions/lib/models/config/toolbar/buttons/video.dart similarity index 95% rename from flutter_quill_extensions/lib/presentation/models/config/toolbar/buttons/video.dart rename to flutter_quill_extensions/lib/models/config/toolbar/buttons/video.dart index e7efba515..e1f86a761 100644 --- a/flutter_quill_extensions/lib/presentation/models/config/toolbar/buttons/video.dart +++ b/flutter_quill_extensions/lib/models/config/toolbar/buttons/video.dart @@ -1,7 +1,7 @@ import 'package:flutter/widgets.dart' show Color; import 'package:flutter_quill/flutter_quill.dart'; -import '../../../../embeds/embed_types/video.dart'; +import '../../../../embeds/video/video.dart'; class QuillToolbarVideoButtonExtraOptions extends QuillToolbarBaseButtonExtraOptions { diff --git a/flutter_quill_extensions/lib/presentation/embeds/editor/image/image.dart b/flutter_quill_extensions/lib/presentation/embeds/editor/image/image.dart deleted file mode 100644 index e95cf9aa7..000000000 --- a/flutter_quill_extensions/lib/presentation/embeds/editor/image/image.dart +++ /dev/null @@ -1,150 +0,0 @@ -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' hide OptionalSize; - -import '../../../../logic/models/config/shared_configurations.dart'; -import '../../../models/config/editor/image/image.dart'; -import '../../../utils/utils.dart'; -import '../../widgets/image.dart'; -import 'image_menu.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'); - - final imageSource = standardizeImageUrl(node.value.data); - final ((imageSize), margin, alignment) = getElementAttributes(node); - - final width = imageSize.width; - final height = imageSize.height; - - final image = getImageWidgetByImageSource( - imageSource, - imageProviderBuilder: configurations.imageProviderBuilder, - imageErrorWidgetBuilder: configurations.imageErrorWidgetBuilder, - alignment: alignment, - height: height, - width: width, - assetsPrefix: QuillSharedExtensionsConfigurations.get(context: context) - .assetsPrefix, - ); - - // OptionalSize? imageSize; - // final style = node.style.attributes['style']; - - // if (style != null) { - // final attrs = base.isMobile(supportWeb: false) - // ? 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(supportWeb: false) - // ? attrs[Attribute.mobileWidth] - // : attrs[Attribute.width.key]) ?? - // '', - // ); - // final height = double.tryParse( - // (base.isMobile(supportWeb: false) - // ? attrs[Attribute.mobileHeight] - // : attrs[Attribute.height.key]) ?? - // '', - // ); - // final alignment = base.getAlignment(base.isMobile(supportWeb: false) - // ? attrs[Attribute.mobileAlignment] - // : attrs[Attribute.alignment]); - // final margin = (base.isMobile(supportWeb: false) - // ? double.tryParse(Attribute.mobileMargin) - // : double.tryParse(Attribute.margin)) ?? - // 0.0; - - // imageSize = OptionalSize(width, height); - // image = Padding( - // padding: EdgeInsets.all(margin), - // child: getImageWidgetByImageSource( - // imageSource, - // width: width, - // height: height, - // alignment: alignment, - // imageProviderBuilder: configurations.imageProviderBuilder, - // imageErrorWidgetBuilder: configurations.imageErrorWidgetBuilder, - // ), - // ); - // } - // } - - // if (imageSize == null) { - // image = getImageWidgetByImageSource( - // imageSource, - // imageProviderBuilder: configurations.imageProviderBuilder, - // imageErrorWidgetBuilder: configurations.imageErrorWidgetBuilder, - // ); - // imageSize = OptionalSize((image as Image).width, image.height); - // } - - final imageSaverService = - QuillSharedExtensionsConfigurations.get(context: context) - .imageSaverService; - return GestureDetector( - onTap: configurations.onImageClicked ?? - () => showDialog( - context: context, - builder: (_) { - return QuillProvider.value( - value: context.requireQuillProvider, - child: FlutterQuillLocalizationsWidget( - child: ImageOptionsMenu( - controller: controller, - configurations: configurations, - imageSource: imageSource, - imageSize: imageSize, - isReadOnly: readOnly, - imageSaverService: imageSaverService, - ), - ), - ); - }, - ), - child: Builder( - builder: (context) { - if (margin != null) { - return Padding( - padding: EdgeInsets.all(margin), - child: image, - ); - } - return image; - }, - ), - ); - } -} diff --git a/flutter_quill_extensions/lib/presentation/embeds/editor/webview.dart b/flutter_quill_extensions/lib/presentation/embeds/editor/webview.dart deleted file mode 100644 index 388b39b48..000000000 --- a/flutter_quill_extensions/lib/presentation/embeds/editor/webview.dart +++ /dev/null @@ -1,58 +0,0 @@ -// 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'; -// } diff --git a/flutter_quill_extensions/lib/presentation/utils/utils.dart b/flutter_quill_extensions/lib/presentation/utils/utils.dart deleted file mode 100644 index 00e986fd2..000000000 --- a/flutter_quill_extensions/lib/presentation/utils/utils.dart +++ /dev/null @@ -1,201 +0,0 @@ -import 'dart:io' show File; - -import 'package:flutter/foundation.dart' show immutable; -import 'package:flutter/widgets.dart' show Alignment; -import 'package:flutter_quill/extensions.dart' as base; -import 'package:flutter_quill/flutter_quill.dart' show Attribute, Node; -import '../../logic/extensions/attribute.dart'; -import '../../logic/services/image_saver/s_image_saver.dart'; -import '../embeds/widgets/image.dart'; - -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 isImageBase64(String imageUrl) { - return !isHttpBasedUrl(imageUrl) && isBase64(imageUrl); -} - -bool isYouTubeUrl(String videoUrl) { - try { - final uri = Uri.parse(videoUrl); - return uri.host == 'www.youtube.com' || - uri.host == 'youtube.com' || - uri.host == 'youtu.be' || - uri.host == 'www.youtu.be'; - } catch (_) { - return false; - } -} - -enum SaveImageResultMethod { network, localStorage } - -@immutable -class SaveImageResult { - const SaveImageResult({required this.error, required this.method}); - - final String? error; - final SaveImageResultMethod method; -} - -Future saveImage({ - required String imageUrl, - required ImageSaverService imageSaverService, -}) async { - final imageFile = File(imageUrl); - final hasPermission = await imageSaverService.hasAccess(); - if (!hasPermission) { - await imageSaverService.requestAccess(); - } - final imageExistsLocally = await imageFile.exists(); - if (!imageExistsLocally) { - try { - await imageSaverService.saveImageFromNetwork( - Uri.parse(appendFileExtensionToImageUrl(imageUrl)), - ); - return const SaveImageResult( - error: null, - method: SaveImageResultMethod.network, - ); - } catch (e) { - return SaveImageResult( - error: e.toString(), - method: SaveImageResultMethod.network, - ); - } - } - try { - await imageSaverService.saveLocalImage(imageUrl); - return const SaveImageResult( - error: null, - method: SaveImageResultMethod.localStorage, - ); - } catch (e) { - return SaveImageResult( - error: e.toString(), - method: SaveImageResultMethod.localStorage, - ); - } -} - -( - OptionalSize elementSize, - double? margin, - Alignment alignment, -) getElementAttributes( - Node node, -) { - var elementSize = const OptionalSize(null, null); - var elementAlignment = Alignment.center; - double? elementMargin; - - // Usually double value - final heightValue = double.tryParse( - node.style.attributes[Attribute.height.key]?.value.toString() ?? ''); - final widthValue = double.tryParse( - node.style.attributes[Attribute.width.key]?.value.toString() ?? ''); - - if (heightValue != null) { - elementSize = elementSize.copyWith( - height: heightValue, - ); - } - if (widthValue != null) { - elementSize = elementSize.copyWith( - width: widthValue, - ); - } - - final cssStyle = node.style.attributes['style']; - - if (cssStyle != null) { - final attrs = base.isMobile(supportWeb: false) - ? base.parseKeyValuePairs(cssStyle.value.toString(), { - AttributeExt.mobileWidth.key, - AttributeExt.mobileHeight.key, - AttributeExt.mobileMargin.key, - AttributeExt.mobileAlignment.key, - }) - : base.parseKeyValuePairs(cssStyle.value.toString(), { - Attribute.width.key, - Attribute.height.key, - 'margin', - 'alignment', - }); - if (attrs.isEmpty) { - return (elementSize, elementMargin, elementAlignment); - } - - // It css value as string but we will try to support it anyway - - // TODO: This could be improved much better - final cssHeightValue = double.tryParse(((base.isMobile(supportWeb: false) - ? attrs[AttributeExt.mobileHeight.key] - : attrs[Attribute.height.key]) ?? - '') - .replaceFirst('px', '')); - final cssWidthValue = double.tryParse(((!base.isMobile(supportWeb: false) - ? attrs[Attribute.width.key] - : attrs[AttributeExt.mobileWidth.key]) ?? - '') - .replaceFirst('px', '')); - - if (cssHeightValue != null) { - elementSize = elementSize.copyWith(height: cssHeightValue); - } - if (cssWidthValue != null) { - elementSize = elementSize.copyWith(width: cssWidthValue); - } - - elementAlignment = base.getAlignment(base.isMobile(supportWeb: false) - ? attrs[AttributeExt.mobileAlignment.key] - : attrs['alignment']); - final margin = (base.isMobile(supportWeb: false) - ? double.tryParse(AttributeExt.mobileMargin.key) - : double.tryParse('margin')); - if (margin != null) { - elementMargin = margin; - } - } - - return (elementSize, elementMargin, elementAlignment); -} - -@immutable -class OptionalSize { - const OptionalSize( - this.width, - this.height, - ); - - /// If non-null, requires the child to have exactly this width. - /// If null, the child is free to choose its own width. - final double? width; - - /// If non-null, requires the child to have exactly this height. - /// If null, the child is free to choose its own height. - final double? height; - - OptionalSize copyWith({ - double? width, - double? height, - }) { - return OptionalSize( - width ?? this.width, - height ?? this.height, - ); - } -} diff --git a/flutter_quill_extensions/lib/logic/services/image_picker/image_options.dart b/flutter_quill_extensions/lib/services/image_picker/image_options.dart similarity index 100% rename from flutter_quill_extensions/lib/logic/services/image_picker/image_options.dart rename to flutter_quill_extensions/lib/services/image_picker/image_options.dart diff --git a/flutter_quill_extensions/lib/logic/services/image_picker/image_picker.dart b/flutter_quill_extensions/lib/services/image_picker/image_picker.dart similarity index 100% rename from flutter_quill_extensions/lib/logic/services/image_picker/image_picker.dart rename to flutter_quill_extensions/lib/services/image_picker/image_picker.dart diff --git a/flutter_quill_extensions/lib/logic/services/image_picker/packages/image_picker.dart b/flutter_quill_extensions/lib/services/image_picker/packages/image_picker.dart similarity index 100% rename from flutter_quill_extensions/lib/logic/services/image_picker/packages/image_picker.dart rename to flutter_quill_extensions/lib/services/image_picker/packages/image_picker.dart diff --git a/flutter_quill_extensions/lib/logic/services/image_picker/s_image_picker.dart b/flutter_quill_extensions/lib/services/image_picker/s_image_picker.dart similarity index 100% rename from flutter_quill_extensions/lib/logic/services/image_picker/s_image_picker.dart rename to flutter_quill_extensions/lib/services/image_picker/s_image_picker.dart diff --git a/flutter_quill_extensions/lib/logic/services/image_saver/exceptions.dart b/flutter_quill_extensions/lib/services/image_saver/exceptions.dart similarity index 100% rename from flutter_quill_extensions/lib/logic/services/image_saver/exceptions.dart rename to flutter_quill_extensions/lib/services/image_saver/exceptions.dart diff --git a/flutter_quill_extensions/lib/logic/services/image_saver/image_saver.dart b/flutter_quill_extensions/lib/services/image_saver/image_saver.dart similarity index 100% rename from flutter_quill_extensions/lib/logic/services/image_saver/image_saver.dart rename to flutter_quill_extensions/lib/services/image_saver/image_saver.dart diff --git a/flutter_quill_extensions/lib/logic/services/image_saver/packages/gal.dart b/flutter_quill_extensions/lib/services/image_saver/packages/gal.dart similarity index 100% rename from flutter_quill_extensions/lib/logic/services/image_saver/packages/gal.dart rename to flutter_quill_extensions/lib/services/image_saver/packages/gal.dart diff --git a/flutter_quill_extensions/lib/logic/services/image_saver/s_image_saver.dart b/flutter_quill_extensions/lib/services/image_saver/s_image_saver.dart similarity index 100% rename from flutter_quill_extensions/lib/logic/services/image_saver/s_image_saver.dart rename to flutter_quill_extensions/lib/services/image_saver/s_image_saver.dart diff --git a/flutter_quill_extensions/lib/presentation/embeds/editor/shims/dart_ui_fake.dart b/flutter_quill_extensions/lib/utils/dart_ui/dart_ui_fake.dart similarity index 100% rename from flutter_quill_extensions/lib/presentation/embeds/editor/shims/dart_ui_fake.dart rename to flutter_quill_extensions/lib/utils/dart_ui/dart_ui_fake.dart diff --git a/flutter_quill_extensions/lib/presentation/embeds/editor/shims/dart_ui_real.dart b/flutter_quill_extensions/lib/utils/dart_ui/dart_ui_real.dart similarity index 100% rename from flutter_quill_extensions/lib/presentation/embeds/editor/shims/dart_ui_real.dart rename to flutter_quill_extensions/lib/utils/dart_ui/dart_ui_real.dart diff --git a/flutter_quill_extensions/lib/utils/element_utils/element_shared_utils.dart b/flutter_quill_extensions/lib/utils/element_utils/element_shared_utils.dart new file mode 100644 index 000000000..927a97e18 --- /dev/null +++ b/flutter_quill_extensions/lib/utils/element_utils/element_shared_utils.dart @@ -0,0 +1,29 @@ +Map parseCssString(String cssString) { + final result = {}; + final declarations = cssString.split(';'); + + for (final declaration in declarations) { + final parts = declaration.split(':'); + if (parts.length == 2) { + final property = parts[0].trim(); + final value = parts[1].trim(); + result[property] = value; + } + } + + return result; +} + +double? parseCssPropertyAsDouble(String value) { + if (value.trim().isEmpty) { + return null; + } + final list = [ + 'px', + // '%', 'vw', 'vh', 'em', 'rem' + ]; + for (final element in list) { + value = value.replaceFirst(element, ''); + } + return double.tryParse(value); +} diff --git a/flutter_quill_extensions/lib/utils/element_utils/element_utils.dart b/flutter_quill_extensions/lib/utils/element_utils/element_utils.dart new file mode 100644 index 000000000..24bd166e7 --- /dev/null +++ b/flutter_quill_extensions/lib/utils/element_utils/element_utils.dart @@ -0,0 +1,98 @@ +import 'package:flutter/foundation.dart' show immutable; +import 'package:flutter/widgets.dart' show Alignment; +import 'package:flutter_quill/extensions.dart'; +import 'package:flutter_quill/flutter_quill.dart' show Attribute, Node; + +import 'element_shared_utils.dart'; + +/// Theses properties are not officialy supported by quill js +/// but they are only used in all platforms other than web +/// and they will be stored in css style property so quill js ignore them +enum ExtraElementProperties { + deletable, +} + +( + ElementSize elementSize, + double? margin, + Alignment alignment, +) getElementAttributes( + Node node, +) { + var elementSize = const ElementSize(null, null); + var elementAlignment = Alignment.center; + double? elementMargin; + + final heightValue = parseCssPropertyAsDouble( + node.style.attributes[Attribute.height.key]?.value.toString() ?? ''); + final widthValue = parseCssPropertyAsDouble( + node.style.attributes[Attribute.width.key]?.value.toString() ?? ''); + + if (heightValue != null) { + elementSize = elementSize.copyWith( + height: heightValue, + ); + } + if (widthValue != null) { + elementSize = elementSize.copyWith( + width: widthValue, + ); + } + + final cssStyle = node.style.attributes['style']; + + if (cssStyle != null) { + // It css value as string but we will try to support it anyway + + final cssAttrs = parseCssString(cssStyle.value.toString()); + + // todo: This could be improved much better + final cssHeightValue = + parseCssPropertyAsDouble((cssAttrs[Attribute.height.key]) ?? ''); + final cssWidthValue = + parseCssPropertyAsDouble((cssAttrs[Attribute.width.key]) ?? ''); + + // cssHeightValue != null && elementSize.height == null + if (cssHeightValue != null) { + elementSize = elementSize.copyWith(height: cssHeightValue); + } + if (cssWidthValue != null) { + elementSize = elementSize.copyWith(width: cssWidthValue); + } + + elementAlignment = getAlignment(cssAttrs['alignment']); + + final margin = double.tryParse('margin'); + if (margin != null) { + elementMargin = margin; + } + } + + return (elementSize, elementMargin, elementAlignment); +} + +@immutable +class ElementSize { + const ElementSize( + this.width, + this.height, + ); + + /// If non-null, requires the child to have exactly this width. + /// If null, the child is free to choose its own width. + final double? width; + + /// If non-null, requires the child to have exactly this height. + /// If null, the child is free to choose its own height. + final double? height; + + ElementSize copyWith({ + double? width, + double? height, + }) { + return ElementSize( + width ?? this.width, + height ?? this.height, + ); + } +} diff --git a/flutter_quill_extensions/lib/presentation/utils/web_utils.dart b/flutter_quill_extensions/lib/utils/element_utils/element_web_utils.dart similarity index 77% rename from flutter_quill_extensions/lib/presentation/utils/web_utils.dart rename to flutter_quill_extensions/lib/utils/element_utils/element_web_utils.dart index b34d308fb..091f1c319 100644 --- a/flutter_quill_extensions/lib/presentation/utils/web_utils.dart +++ b/flutter_quill_extensions/lib/utils/element_utils/element_web_utils.dart @@ -1,6 +1,7 @@ -import 'package:flutter_quill/extensions.dart' as base; import 'package:flutter_quill/flutter_quill.dart' show Attribute, Node; +import 'element_shared_utils.dart'; + /// 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 @@ -14,26 +15,20 @@ import 'package:flutter_quill/flutter_quill.dart' show Attribute, Node; ) { var height = 'auto'; var width = 'auto'; - // TODO: Add support for margin and alignment - const margin = 'auto'; + // TODO(): Add support for margin and alignment + var margin = 'auto'; const alignment = 'center'; - // return (height, width, margin, alignment); - 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, - 'margin', - 'alignment', - }); + final attrs = parseCssString(cssStyle.value.toString()); + final cssHeightValue = attrs[Attribute.height.key]; + if (cssHeightValue != null) { height = cssHeightValue; } else { @@ -46,6 +41,11 @@ import 'package:flutter_quill/flutter_quill.dart' show Attribute, Node; width = '${widthValue}px'; } + final cssMarginValue = attrs['margin']; + if (cssMarginValue != null) { + margin = cssMarginValue; + } + return (height, width, margin, alignment); } diff --git a/flutter_quill_extensions/lib/logic/utils/quill_image_utils.dart b/flutter_quill_extensions/lib/utils/quill_image_utils.dart similarity index 99% rename from flutter_quill_extensions/lib/logic/utils/quill_image_utils.dart rename to flutter_quill_extensions/lib/utils/quill_image_utils.dart index 30eca1286..4a366c092 100644 --- a/flutter_quill_extensions/lib/logic/utils/quill_image_utils.dart +++ b/flutter_quill_extensions/lib/utils/quill_image_utils.dart @@ -4,7 +4,7 @@ import 'package:flutter/foundation.dart' show kIsWeb; import 'package:flutter_quill/flutter_quill.dart' as quill; import 'package:path/path.dart' as path; -import '../../presentation/utils/utils.dart'; +import 'utils.dart'; typedef OnGenerateNewFileNameCallback = String Function( String currentFileName, diff --git a/flutter_quill_extensions/lib/logic/utils/string.dart b/flutter_quill_extensions/lib/utils/string.dart similarity index 66% rename from flutter_quill_extensions/lib/logic/utils/string.dart rename to flutter_quill_extensions/lib/utils/string.dart index aca39f68b..a13cd3445 100644 --- a/flutter_quill_extensions/lib/logic/utils/string.dart +++ b/flutter_quill_extensions/lib/utils/string.dart @@ -1,12 +1,9 @@ import 'package:flutter_quill/flutter_quill.dart' show Attribute; -import '../extensions/attribute.dart'; - String replaceStyleStringWithSize( String cssStyle, { required double width, required double height, - required bool isMobile, }) { final result = {}; final pairs = cssStyle.split(';'); @@ -19,13 +16,8 @@ String replaceStyleStringWithSize( result[key] = pair.substring(index + 1).trim(); } - if (isMobile) { - result[AttributeExt.mobileWidth.key] = width.toString(); - result[AttributeExt.mobileHeight.key] = height.toString(); - } else { - result[Attribute.width.key] = width.toString(); - result[Attribute.height.key] = height.toString(); - } + result[Attribute.width.key] = width.toString(); + result[Attribute.height.key] = height.toString(); final sb = StringBuffer(); for (final pair in result.entries) { sb diff --git a/flutter_quill_extensions/lib/utils/utils.dart b/flutter_quill_extensions/lib/utils/utils.dart new file mode 100644 index 000000000..12c802a11 --- /dev/null +++ b/flutter_quill_extensions/lib/utils/utils.dart @@ -0,0 +1,89 @@ +import 'dart:io' show File; + +import 'package:flutter/foundation.dart' show immutable; + +import '../embeds/widgets/image.dart'; +import '../services/image_saver/s_image_saver.dart'; + +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 isImageBase64(String imageUrl) { + return !isHttpBasedUrl(imageUrl) && isBase64(imageUrl); +} + +bool isYouTubeUrl(String videoUrl) { + try { + final uri = Uri.parse(videoUrl); + return uri.host == 'www.youtube.com' || + uri.host == 'youtube.com' || + uri.host == 'youtu.be' || + uri.host == 'www.youtu.be'; + } catch (_) { + return false; + } +} + +enum SaveImageResultMethod { network, localStorage } + +@immutable +class SaveImageResult { + const SaveImageResult({required this.error, required this.method}); + + final String? error; + final SaveImageResultMethod method; +} + +Future saveImage({ + required String imageUrl, + required ImageSaverService imageSaverService, +}) async { + final imageFile = File(imageUrl); + final hasPermission = await imageSaverService.hasAccess(); + if (!hasPermission) { + await imageSaverService.requestAccess(); + } + final imageExistsLocally = await imageFile.exists(); + if (!imageExistsLocally) { + try { + await imageSaverService.saveImageFromNetwork( + Uri.parse(appendFileExtensionToImageUrl(imageUrl)), + ); + return const SaveImageResult( + error: null, + method: SaveImageResultMethod.network, + ); + } catch (e) { + return SaveImageResult( + error: e.toString(), + method: SaveImageResultMethod.network, + ); + } + } + try { + await imageSaverService.saveLocalImage(imageUrl); + return const SaveImageResult( + error: null, + method: SaveImageResultMethod.localStorage, + ); + } catch (e) { + return SaveImageResult( + error: e.toString(), + method: SaveImageResultMethod.localStorage, + ); + } +} diff --git a/flutter_quill_extensions/pubspec.yaml b/flutter_quill_extensions/pubspec.yaml index da3c46d87..254d628a2 100644 --- a/flutter_quill_extensions/pubspec.yaml +++ b/flutter_quill_extensions/pubspec.yaml @@ -1,6 +1,6 @@ name: flutter_quill_extensions description: Embed extensions for flutter_quill including image, video, formula and etc. -version: 0.7.2 +version: 0.8.0-dev homepage: https://github.com/singerdmx/flutter-quill/tree/master/flutter_quill_extensions/ repository: https://github.com/singerdmx/flutter-quill/tree/master/flutter_quill_extensions/ issue_tracker: https://github.com/singerdmx/flutter-quill/issues/ diff --git a/lib/flutter_quill.dart b/lib/flutter_quill.dart index 640571cc1..2c3170615 100644 --- a/lib/flutter_quill.dart +++ b/lib/flutter_quill.dart @@ -1,7 +1,6 @@ library flutter_quill; export 'src/extensions/quill_provider.dart'; -export 'src/l10n/widgets/localizations.dart'; export 'src/models/config/quill_configurations.dart'; export 'src/models/config/raw_editor/configurations.dart'; export 'src/models/config/toolbar/base_toolbar_configurations.dart'; diff --git a/lib/src/l10n/generated/quill_localizations.dart b/lib/src/l10n/generated/quill_localizations.dart index 756ac2c96..f4425f811 100644 --- a/lib/src/l10n/generated/quill_localizations.dart +++ b/lib/src/l10n/generated/quill_localizations.dart @@ -600,6 +600,42 @@ abstract class FlutterQuillLocalizations { /// In en, this message translates to: /// **'Insert image'** String get insertImage; + + /// No description provided for @pickAPhotoFromYourGallery. + /// + /// In en, this message translates to: + /// **'Pick a photo from your gallery'** + String get pickAPhotoFromYourGallery; + + /// No description provided for @takeAPhotoUsingYourCamera. + /// + /// In en, this message translates to: + /// **'Take a photo using your phone camera'** + String get takeAPhotoUsingYourCamera; + + /// No description provided for @pasteAPhotoUsingALink. + /// + /// In en, this message translates to: + /// **'Paste a photo using a link'** + String get pasteAPhotoUsingALink; + + /// No description provided for @pickAVideoFromYourGallery. + /// + /// In en, this message translates to: + /// **'Pick a video from your gallery'** + String get pickAVideoFromYourGallery; + + /// No description provided for @recordAVideoUsingYourCamera. + /// + /// In en, this message translates to: + /// **'Record a video using your phone camera'** + String get recordAVideoUsingYourCamera; + + /// No description provided for @pasteAVideoUsingALink. + /// + /// In en, this message translates to: + /// **'Paste a video using a link'** + String get pasteAVideoUsingALink; } class _FlutterQuillLocalizationsDelegate diff --git a/lib/src/l10n/generated/quill_localizations_ar.dart b/lib/src/l10n/generated/quill_localizations_ar.dart index d78d815a6..fc7b2dfc4 100644 --- a/lib/src/l10n/generated/quill_localizations_ar.dart +++ b/lib/src/l10n/generated/quill_localizations_ar.dart @@ -226,4 +226,24 @@ class FlutterQuillLocalizationsAr extends FlutterQuillLocalizations { @override String get insertImage => 'إدراج صورة'; + + @override + String get pickAPhotoFromYourGallery => 'Pick a photo from your gallery'; + + @override + String get takeAPhotoUsingYourCamera => + 'Take a photo using your phone camera'; + + @override + String get pasteAPhotoUsingALink => 'Paste a photo using a link'; + + @override + String get pickAVideoFromYourGallery => 'Pick a video from your gallery'; + + @override + String get recordAVideoUsingYourCamera => + 'Record a video using your phone camera'; + + @override + String get pasteAVideoUsingALink => 'Paste a video using a link'; } diff --git a/lib/src/l10n/generated/quill_localizations_bg.dart b/lib/src/l10n/generated/quill_localizations_bg.dart index b0fc467ca..a75b6ffab 100644 --- a/lib/src/l10n/generated/quill_localizations_bg.dart +++ b/lib/src/l10n/generated/quill_localizations_bg.dart @@ -228,4 +228,24 @@ class FlutterQuillLocalizationsBg extends FlutterQuillLocalizations { @override String get insertImage => 'Вмъкване на изображение'; + + @override + String get pickAPhotoFromYourGallery => 'Pick a photo from your gallery'; + + @override + String get takeAPhotoUsingYourCamera => + 'Take a photo using your phone camera'; + + @override + String get pasteAPhotoUsingALink => 'Paste a photo using a link'; + + @override + String get pickAVideoFromYourGallery => 'Pick a video from your gallery'; + + @override + String get recordAVideoUsingYourCamera => + 'Record a video using your phone camera'; + + @override + String get pasteAVideoUsingALink => 'Paste a video using a link'; } diff --git a/lib/src/l10n/generated/quill_localizations_bn.dart b/lib/src/l10n/generated/quill_localizations_bn.dart index eb8b0f564..7ef340ef6 100644 --- a/lib/src/l10n/generated/quill_localizations_bn.dart +++ b/lib/src/l10n/generated/quill_localizations_bn.dart @@ -228,4 +228,24 @@ class FlutterQuillLocalizationsBn extends FlutterQuillLocalizations { @override String get insertImage => 'চিত্র সন্নিবেশ'; + + @override + String get pickAPhotoFromYourGallery => 'Pick a photo from your gallery'; + + @override + String get takeAPhotoUsingYourCamera => + 'Take a photo using your phone camera'; + + @override + String get pasteAPhotoUsingALink => 'Paste a photo using a link'; + + @override + String get pickAVideoFromYourGallery => 'Pick a video from your gallery'; + + @override + String get recordAVideoUsingYourCamera => + 'Record a video using your phone camera'; + + @override + String get pasteAVideoUsingALink => 'Paste a video using a link'; } diff --git a/lib/src/l10n/generated/quill_localizations_cs.dart b/lib/src/l10n/generated/quill_localizations_cs.dart index 2c37d03fe..a74f67564 100644 --- a/lib/src/l10n/generated/quill_localizations_cs.dart +++ b/lib/src/l10n/generated/quill_localizations_cs.dart @@ -228,4 +228,24 @@ class FlutterQuillLocalizationsCs extends FlutterQuillLocalizations { @override String get insertImage => 'Vložit obrázek'; + + @override + String get pickAPhotoFromYourGallery => 'Pick a photo from your gallery'; + + @override + String get takeAPhotoUsingYourCamera => + 'Take a photo using your phone camera'; + + @override + String get pasteAPhotoUsingALink => 'Paste a photo using a link'; + + @override + String get pickAVideoFromYourGallery => 'Pick a video from your gallery'; + + @override + String get recordAVideoUsingYourCamera => + 'Record a video using your phone camera'; + + @override + String get pasteAVideoUsingALink => 'Paste a video using a link'; } diff --git a/lib/src/l10n/generated/quill_localizations_da.dart b/lib/src/l10n/generated/quill_localizations_da.dart index 4a136af45..6e8825fc5 100644 --- a/lib/src/l10n/generated/quill_localizations_da.dart +++ b/lib/src/l10n/generated/quill_localizations_da.dart @@ -226,4 +226,24 @@ class FlutterQuillLocalizationsDa extends FlutterQuillLocalizations { @override String get insertImage => 'Indsæt billede'; + + @override + String get pickAPhotoFromYourGallery => 'Pick a photo from your gallery'; + + @override + String get takeAPhotoUsingYourCamera => + 'Take a photo using your phone camera'; + + @override + String get pasteAPhotoUsingALink => 'Paste a photo using a link'; + + @override + String get pickAVideoFromYourGallery => 'Pick a video from your gallery'; + + @override + String get recordAVideoUsingYourCamera => + 'Record a video using your phone camera'; + + @override + String get pasteAVideoUsingALink => 'Paste a video using a link'; } diff --git a/lib/src/l10n/generated/quill_localizations_de.dart b/lib/src/l10n/generated/quill_localizations_de.dart index 79ae25bea..384b8ea33 100644 --- a/lib/src/l10n/generated/quill_localizations_de.dart +++ b/lib/src/l10n/generated/quill_localizations_de.dart @@ -227,4 +227,24 @@ class FlutterQuillLocalizationsDe extends FlutterQuillLocalizations { @override String get insertImage => 'Bild einfügen'; + + @override + String get pickAPhotoFromYourGallery => 'Pick a photo from your gallery'; + + @override + String get takeAPhotoUsingYourCamera => + 'Take a photo using your phone camera'; + + @override + String get pasteAPhotoUsingALink => 'Paste a photo using a link'; + + @override + String get pickAVideoFromYourGallery => 'Pick a video from your gallery'; + + @override + String get recordAVideoUsingYourCamera => + 'Record a video using your phone camera'; + + @override + String get pasteAVideoUsingALink => 'Paste a video using a link'; } diff --git a/lib/src/l10n/generated/quill_localizations_en.dart b/lib/src/l10n/generated/quill_localizations_en.dart index c69876d96..1233f1cf3 100644 --- a/lib/src/l10n/generated/quill_localizations_en.dart +++ b/lib/src/l10n/generated/quill_localizations_en.dart @@ -228,6 +228,26 @@ class FlutterQuillLocalizationsEn extends FlutterQuillLocalizations { @override String get insertImage => 'Insert image'; + + @override + String get pickAPhotoFromYourGallery => 'Pick a photo from your gallery'; + + @override + String get takeAPhotoUsingYourCamera => + 'Take a photo using your phone camera'; + + @override + String get pasteAPhotoUsingALink => 'Paste a photo using a link'; + + @override + String get pickAVideoFromYourGallery => 'Pick a video from your gallery'; + + @override + String get recordAVideoUsingYourCamera => + 'Record a video using your phone camera'; + + @override + String get pasteAVideoUsingALink => 'Paste a video using a link'; } /// The translations for English, as used in the United States (`en_US`). diff --git a/lib/src/l10n/generated/quill_localizations_es.dart b/lib/src/l10n/generated/quill_localizations_es.dart index 85c895f32..3c63865d1 100644 --- a/lib/src/l10n/generated/quill_localizations_es.dart +++ b/lib/src/l10n/generated/quill_localizations_es.dart @@ -227,4 +227,24 @@ class FlutterQuillLocalizationsEs extends FlutterQuillLocalizations { @override String get insertImage => 'Insertar imagen'; + + @override + String get pickAPhotoFromYourGallery => 'Pick a photo from your gallery'; + + @override + String get takeAPhotoUsingYourCamera => + 'Take a photo using your phone camera'; + + @override + String get pasteAPhotoUsingALink => 'Paste a photo using a link'; + + @override + String get pickAVideoFromYourGallery => 'Pick a video from your gallery'; + + @override + String get recordAVideoUsingYourCamera => + 'Record a video using your phone camera'; + + @override + String get pasteAVideoUsingALink => 'Paste a video using a link'; } diff --git a/lib/src/l10n/generated/quill_localizations_fa.dart b/lib/src/l10n/generated/quill_localizations_fa.dart index e649354f8..32040119a 100644 --- a/lib/src/l10n/generated/quill_localizations_fa.dart +++ b/lib/src/l10n/generated/quill_localizations_fa.dart @@ -229,4 +229,24 @@ class FlutterQuillLocalizationsFa extends FlutterQuillLocalizations { @override String get insertImage => 'وارد کردن تصویر'; + + @override + String get pickAPhotoFromYourGallery => 'Pick a photo from your gallery'; + + @override + String get takeAPhotoUsingYourCamera => + 'Take a photo using your phone camera'; + + @override + String get pasteAPhotoUsingALink => 'Paste a photo using a link'; + + @override + String get pickAVideoFromYourGallery => 'Pick a video from your gallery'; + + @override + String get recordAVideoUsingYourCamera => + 'Record a video using your phone camera'; + + @override + String get pasteAVideoUsingALink => 'Paste a video using a link'; } diff --git a/lib/src/l10n/generated/quill_localizations_fr.dart b/lib/src/l10n/generated/quill_localizations_fr.dart index 7bd59ee48..b94451ff1 100644 --- a/lib/src/l10n/generated/quill_localizations_fr.dart +++ b/lib/src/l10n/generated/quill_localizations_fr.dart @@ -230,4 +230,24 @@ class FlutterQuillLocalizationsFr extends FlutterQuillLocalizations { @override String get insertImage => 'Insérer une image'; + + @override + String get pickAPhotoFromYourGallery => 'Pick a photo from your gallery'; + + @override + String get takeAPhotoUsingYourCamera => + 'Take a photo using your phone camera'; + + @override + String get pasteAPhotoUsingALink => 'Paste a photo using a link'; + + @override + String get pickAVideoFromYourGallery => 'Pick a video from your gallery'; + + @override + String get recordAVideoUsingYourCamera => + 'Record a video using your phone camera'; + + @override + String get pasteAVideoUsingALink => 'Paste a video using a link'; } diff --git a/lib/src/l10n/generated/quill_localizations_he.dart b/lib/src/l10n/generated/quill_localizations_he.dart index a0d94ce0f..956c10f99 100644 --- a/lib/src/l10n/generated/quill_localizations_he.dart +++ b/lib/src/l10n/generated/quill_localizations_he.dart @@ -228,4 +228,24 @@ class FlutterQuillLocalizationsHe extends FlutterQuillLocalizations { @override String get insertImage => 'הכנס תמונה'; + + @override + String get pickAPhotoFromYourGallery => 'Pick a photo from your gallery'; + + @override + String get takeAPhotoUsingYourCamera => + 'Take a photo using your phone camera'; + + @override + String get pasteAPhotoUsingALink => 'Paste a photo using a link'; + + @override + String get pickAVideoFromYourGallery => 'Pick a video from your gallery'; + + @override + String get recordAVideoUsingYourCamera => + 'Record a video using your phone camera'; + + @override + String get pasteAVideoUsingALink => 'Paste a video using a link'; } diff --git a/lib/src/l10n/generated/quill_localizations_hi.dart b/lib/src/l10n/generated/quill_localizations_hi.dart index 4e2d2d5ed..3e2e738aa 100644 --- a/lib/src/l10n/generated/quill_localizations_hi.dart +++ b/lib/src/l10n/generated/quill_localizations_hi.dart @@ -229,4 +229,24 @@ class FlutterQuillLocalizationsHi extends FlutterQuillLocalizations { @override String get insertImage => 'छवि डालें'; + + @override + String get pickAPhotoFromYourGallery => 'Pick a photo from your gallery'; + + @override + String get takeAPhotoUsingYourCamera => + 'Take a photo using your phone camera'; + + @override + String get pasteAPhotoUsingALink => 'Paste a photo using a link'; + + @override + String get pickAVideoFromYourGallery => 'Pick a video from your gallery'; + + @override + String get recordAVideoUsingYourCamera => + 'Record a video using your phone camera'; + + @override + String get pasteAVideoUsingALink => 'Paste a video using a link'; } diff --git a/lib/src/l10n/generated/quill_localizations_id.dart b/lib/src/l10n/generated/quill_localizations_id.dart index 26eb61cb0..4d4d38c53 100644 --- a/lib/src/l10n/generated/quill_localizations_id.dart +++ b/lib/src/l10n/generated/quill_localizations_id.dart @@ -230,4 +230,24 @@ class FlutterQuillLocalizationsId extends FlutterQuillLocalizations { @override String get insertImage => 'Sisipkan Gambar'; + + @override + String get pickAPhotoFromYourGallery => 'Pick a photo from your gallery'; + + @override + String get takeAPhotoUsingYourCamera => + 'Take a photo using your phone camera'; + + @override + String get pasteAPhotoUsingALink => 'Paste a photo using a link'; + + @override + String get pickAVideoFromYourGallery => 'Pick a video from your gallery'; + + @override + String get recordAVideoUsingYourCamera => + 'Record a video using your phone camera'; + + @override + String get pasteAVideoUsingALink => 'Paste a video using a link'; } diff --git a/lib/src/l10n/generated/quill_localizations_it.dart b/lib/src/l10n/generated/quill_localizations_it.dart index 9d6c18eea..2b8fe0703 100644 --- a/lib/src/l10n/generated/quill_localizations_it.dart +++ b/lib/src/l10n/generated/quill_localizations_it.dart @@ -230,4 +230,24 @@ class FlutterQuillLocalizationsIt extends FlutterQuillLocalizations { @override String get insertImage => 'Inserisci immagine'; + + @override + String get pickAPhotoFromYourGallery => 'Pick a photo from your gallery'; + + @override + String get takeAPhotoUsingYourCamera => + 'Take a photo using your phone camera'; + + @override + String get pasteAPhotoUsingALink => 'Paste a photo using a link'; + + @override + String get pickAVideoFromYourGallery => 'Pick a video from your gallery'; + + @override + String get recordAVideoUsingYourCamera => + 'Record a video using your phone camera'; + + @override + String get pasteAVideoUsingALink => 'Paste a video using a link'; } diff --git a/lib/src/l10n/generated/quill_localizations_ja.dart b/lib/src/l10n/generated/quill_localizations_ja.dart index 94927b835..7a55d6b53 100644 --- a/lib/src/l10n/generated/quill_localizations_ja.dart +++ b/lib/src/l10n/generated/quill_localizations_ja.dart @@ -225,4 +225,24 @@ class FlutterQuillLocalizationsJa extends FlutterQuillLocalizations { @override String get insertImage => '画像を挿入'; + + @override + String get pickAPhotoFromYourGallery => 'Pick a photo from your gallery'; + + @override + String get takeAPhotoUsingYourCamera => + 'Take a photo using your phone camera'; + + @override + String get pasteAPhotoUsingALink => 'Paste a photo using a link'; + + @override + String get pickAVideoFromYourGallery => 'Pick a video from your gallery'; + + @override + String get recordAVideoUsingYourCamera => + 'Record a video using your phone camera'; + + @override + String get pasteAVideoUsingALink => 'Paste a video using a link'; } diff --git a/lib/src/l10n/generated/quill_localizations_ko.dart b/lib/src/l10n/generated/quill_localizations_ko.dart index bb6d87de0..c8a6b5669 100644 --- a/lib/src/l10n/generated/quill_localizations_ko.dart +++ b/lib/src/l10n/generated/quill_localizations_ko.dart @@ -225,4 +225,24 @@ class FlutterQuillLocalizationsKo extends FlutterQuillLocalizations { @override String get insertImage => '이미지 삽입'; + + @override + String get pickAPhotoFromYourGallery => 'Pick a photo from your gallery'; + + @override + String get takeAPhotoUsingYourCamera => + 'Take a photo using your phone camera'; + + @override + String get pasteAPhotoUsingALink => 'Paste a photo using a link'; + + @override + String get pickAVideoFromYourGallery => 'Pick a video from your gallery'; + + @override + String get recordAVideoUsingYourCamera => + 'Record a video using your phone camera'; + + @override + String get pasteAVideoUsingALink => 'Paste a video using a link'; } diff --git a/lib/src/l10n/generated/quill_localizations_ms.dart b/lib/src/l10n/generated/quill_localizations_ms.dart index b46f2fd90..c443712f5 100644 --- a/lib/src/l10n/generated/quill_localizations_ms.dart +++ b/lib/src/l10n/generated/quill_localizations_ms.dart @@ -228,4 +228,24 @@ class FlutterQuillLocalizationsMs extends FlutterQuillLocalizations { @override String get insertImage => 'Masukkan imej'; + + @override + String get pickAPhotoFromYourGallery => 'Pick a photo from your gallery'; + + @override + String get takeAPhotoUsingYourCamera => + 'Take a photo using your phone camera'; + + @override + String get pasteAPhotoUsingALink => 'Paste a photo using a link'; + + @override + String get pickAVideoFromYourGallery => 'Pick a video from your gallery'; + + @override + String get recordAVideoUsingYourCamera => + 'Record a video using your phone camera'; + + @override + String get pasteAVideoUsingALink => 'Paste a video using a link'; } diff --git a/lib/src/l10n/generated/quill_localizations_nl.dart b/lib/src/l10n/generated/quill_localizations_nl.dart index 4b8b444c6..65bc5b47c 100644 --- a/lib/src/l10n/generated/quill_localizations_nl.dart +++ b/lib/src/l10n/generated/quill_localizations_nl.dart @@ -230,4 +230,24 @@ class FlutterQuillLocalizationsNl extends FlutterQuillLocalizations { @override String get insertImage => 'Afbeelding invoegen'; + + @override + String get pickAPhotoFromYourGallery => 'Pick a photo from your gallery'; + + @override + String get takeAPhotoUsingYourCamera => + 'Take a photo using your phone camera'; + + @override + String get pasteAPhotoUsingALink => 'Paste a photo using a link'; + + @override + String get pickAVideoFromYourGallery => 'Pick a video from your gallery'; + + @override + String get recordAVideoUsingYourCamera => + 'Record a video using your phone camera'; + + @override + String get pasteAVideoUsingALink => 'Paste a video using a link'; } diff --git a/lib/src/l10n/generated/quill_localizations_no.dart b/lib/src/l10n/generated/quill_localizations_no.dart index 37bf4cf0f..087c69971 100644 --- a/lib/src/l10n/generated/quill_localizations_no.dart +++ b/lib/src/l10n/generated/quill_localizations_no.dart @@ -230,4 +230,24 @@ class FlutterQuillLocalizationsNo extends FlutterQuillLocalizations { @override String get insertImage => 'Sett inn bilde'; + + @override + String get pickAPhotoFromYourGallery => 'Pick a photo from your gallery'; + + @override + String get takeAPhotoUsingYourCamera => + 'Take a photo using your phone camera'; + + @override + String get pasteAPhotoUsingALink => 'Paste a photo using a link'; + + @override + String get pickAVideoFromYourGallery => 'Pick a video from your gallery'; + + @override + String get recordAVideoUsingYourCamera => + 'Record a video using your phone camera'; + + @override + String get pasteAVideoUsingALink => 'Paste a video using a link'; } diff --git a/lib/src/l10n/generated/quill_localizations_pl.dart b/lib/src/l10n/generated/quill_localizations_pl.dart index b983356bb..677ff5c82 100644 --- a/lib/src/l10n/generated/quill_localizations_pl.dart +++ b/lib/src/l10n/generated/quill_localizations_pl.dart @@ -227,4 +227,24 @@ class FlutterQuillLocalizationsPl extends FlutterQuillLocalizations { @override String get insertImage => 'Wstaw obraz'; + + @override + String get pickAPhotoFromYourGallery => 'Pick a photo from your gallery'; + + @override + String get takeAPhotoUsingYourCamera => + 'Take a photo using your phone camera'; + + @override + String get pasteAPhotoUsingALink => 'Paste a photo using a link'; + + @override + String get pickAVideoFromYourGallery => 'Pick a video from your gallery'; + + @override + String get recordAVideoUsingYourCamera => + 'Record a video using your phone camera'; + + @override + String get pasteAVideoUsingALink => 'Paste a video using a link'; } diff --git a/lib/src/l10n/generated/quill_localizations_pt.dart b/lib/src/l10n/generated/quill_localizations_pt.dart index a48362b1b..bec9b366c 100644 --- a/lib/src/l10n/generated/quill_localizations_pt.dart +++ b/lib/src/l10n/generated/quill_localizations_pt.dart @@ -228,6 +228,26 @@ class FlutterQuillLocalizationsPt extends FlutterQuillLocalizations { @override String get insertImage => 'Inserir imagem'; + + @override + String get pickAPhotoFromYourGallery => 'Pick a photo from your gallery'; + + @override + String get takeAPhotoUsingYourCamera => + 'Take a photo using your phone camera'; + + @override + String get pasteAPhotoUsingALink => 'Paste a photo using a link'; + + @override + String get pickAVideoFromYourGallery => 'Pick a video from your gallery'; + + @override + String get recordAVideoUsingYourCamera => + 'Record a video using your phone camera'; + + @override + String get pasteAVideoUsingALink => 'Paste a video using a link'; } /// The translations for Portuguese, as used in Brazil (`pt_BR`). diff --git a/lib/src/l10n/generated/quill_localizations_ru.dart b/lib/src/l10n/generated/quill_localizations_ru.dart index 376e3c12f..96a9314b8 100644 --- a/lib/src/l10n/generated/quill_localizations_ru.dart +++ b/lib/src/l10n/generated/quill_localizations_ru.dart @@ -227,4 +227,24 @@ class FlutterQuillLocalizationsRu extends FlutterQuillLocalizations { @override String get insertImage => 'Вставить изображение'; + + @override + String get pickAPhotoFromYourGallery => 'Pick a photo from your gallery'; + + @override + String get takeAPhotoUsingYourCamera => + 'Take a photo using your phone camera'; + + @override + String get pasteAPhotoUsingALink => 'Paste a photo using a link'; + + @override + String get pickAVideoFromYourGallery => 'Pick a video from your gallery'; + + @override + String get recordAVideoUsingYourCamera => + 'Record a video using your phone camera'; + + @override + String get pasteAVideoUsingALink => 'Paste a video using a link'; } diff --git a/lib/src/l10n/generated/quill_localizations_sr.dart b/lib/src/l10n/generated/quill_localizations_sr.dart index e49dcdb71..1e89073e7 100644 --- a/lib/src/l10n/generated/quill_localizations_sr.dart +++ b/lib/src/l10n/generated/quill_localizations_sr.dart @@ -229,4 +229,24 @@ class FlutterQuillLocalizationsSr extends FlutterQuillLocalizations { @override String get insertImage => 'Umetni sliku'; + + @override + String get pickAPhotoFromYourGallery => 'Pick a photo from your gallery'; + + @override + String get takeAPhotoUsingYourCamera => + 'Take a photo using your phone camera'; + + @override + String get pasteAPhotoUsingALink => 'Paste a photo using a link'; + + @override + String get pickAVideoFromYourGallery => 'Pick a video from your gallery'; + + @override + String get recordAVideoUsingYourCamera => + 'Record a video using your phone camera'; + + @override + String get pasteAVideoUsingALink => 'Paste a video using a link'; } diff --git a/lib/src/l10n/generated/quill_localizations_sw.dart b/lib/src/l10n/generated/quill_localizations_sw.dart index 5928e04ee..640403607 100644 --- a/lib/src/l10n/generated/quill_localizations_sw.dart +++ b/lib/src/l10n/generated/quill_localizations_sw.dart @@ -227,4 +227,24 @@ class FlutterQuillLocalizationsSw extends FlutterQuillLocalizations { @override String get insertImage => 'Weka Picha'; + + @override + String get pickAPhotoFromYourGallery => 'Pick a photo from your gallery'; + + @override + String get takeAPhotoUsingYourCamera => + 'Take a photo using your phone camera'; + + @override + String get pasteAPhotoUsingALink => 'Paste a photo using a link'; + + @override + String get pickAVideoFromYourGallery => 'Pick a video from your gallery'; + + @override + String get recordAVideoUsingYourCamera => + 'Record a video using your phone camera'; + + @override + String get pasteAVideoUsingALink => 'Paste a video using a link'; } diff --git a/lib/src/l10n/generated/quill_localizations_tk.dart b/lib/src/l10n/generated/quill_localizations_tk.dart index 566313280..13e7b1d3c 100644 --- a/lib/src/l10n/generated/quill_localizations_tk.dart +++ b/lib/src/l10n/generated/quill_localizations_tk.dart @@ -226,4 +226,24 @@ class FlutterQuillLocalizationsTk extends FlutterQuillLocalizations { @override String get insertImage => 'Surat goş'; + + @override + String get pickAPhotoFromYourGallery => 'Pick a photo from your gallery'; + + @override + String get takeAPhotoUsingYourCamera => + 'Take a photo using your phone camera'; + + @override + String get pasteAPhotoUsingALink => 'Paste a photo using a link'; + + @override + String get pickAVideoFromYourGallery => 'Pick a video from your gallery'; + + @override + String get recordAVideoUsingYourCamera => + 'Record a video using your phone camera'; + + @override + String get pasteAVideoUsingALink => 'Paste a video using a link'; } diff --git a/lib/src/l10n/generated/quill_localizations_tr.dart b/lib/src/l10n/generated/quill_localizations_tr.dart index dbd5d3ded..e6c52e9a5 100644 --- a/lib/src/l10n/generated/quill_localizations_tr.dart +++ b/lib/src/l10n/generated/quill_localizations_tr.dart @@ -227,4 +227,24 @@ class FlutterQuillLocalizationsTr extends FlutterQuillLocalizations { @override String get insertImage => 'Görüntü ekle'; + + @override + String get pickAPhotoFromYourGallery => 'Pick a photo from your gallery'; + + @override + String get takeAPhotoUsingYourCamera => + 'Take a photo using your phone camera'; + + @override + String get pasteAPhotoUsingALink => 'Paste a photo using a link'; + + @override + String get pickAVideoFromYourGallery => 'Pick a video from your gallery'; + + @override + String get recordAVideoUsingYourCamera => + 'Record a video using your phone camera'; + + @override + String get pasteAVideoUsingALink => 'Paste a video using a link'; } diff --git a/lib/src/l10n/generated/quill_localizations_uk.dart b/lib/src/l10n/generated/quill_localizations_uk.dart index 163c299fc..f7a273a5a 100644 --- a/lib/src/l10n/generated/quill_localizations_uk.dart +++ b/lib/src/l10n/generated/quill_localizations_uk.dart @@ -229,4 +229,24 @@ class FlutterQuillLocalizationsUk extends FlutterQuillLocalizations { @override String get insertImage => 'Вставити зображення'; + + @override + String get pickAPhotoFromYourGallery => 'Pick a photo from your gallery'; + + @override + String get takeAPhotoUsingYourCamera => + 'Take a photo using your phone camera'; + + @override + String get pasteAPhotoUsingALink => 'Paste a photo using a link'; + + @override + String get pickAVideoFromYourGallery => 'Pick a video from your gallery'; + + @override + String get recordAVideoUsingYourCamera => + 'Record a video using your phone camera'; + + @override + String get pasteAVideoUsingALink => 'Paste a video using a link'; } diff --git a/lib/src/l10n/generated/quill_localizations_ur.dart b/lib/src/l10n/generated/quill_localizations_ur.dart index 2f4b14fdb..79c665f56 100644 --- a/lib/src/l10n/generated/quill_localizations_ur.dart +++ b/lib/src/l10n/generated/quill_localizations_ur.dart @@ -231,4 +231,24 @@ class FlutterQuillLocalizationsUr extends FlutterQuillLocalizations { @override String get insertImage => 'تصویر داخل کریں'; + + @override + String get pickAPhotoFromYourGallery => 'Pick a photo from your gallery'; + + @override + String get takeAPhotoUsingYourCamera => + 'Take a photo using your phone camera'; + + @override + String get pasteAPhotoUsingALink => 'Paste a photo using a link'; + + @override + String get pickAVideoFromYourGallery => 'Pick a video from your gallery'; + + @override + String get recordAVideoUsingYourCamera => + 'Record a video using your phone camera'; + + @override + String get pasteAVideoUsingALink => 'Paste a video using a link'; } diff --git a/lib/src/l10n/generated/quill_localizations_vi.dart b/lib/src/l10n/generated/quill_localizations_vi.dart index 439d1de54..6f26695b0 100644 --- a/lib/src/l10n/generated/quill_localizations_vi.dart +++ b/lib/src/l10n/generated/quill_localizations_vi.dart @@ -228,4 +228,24 @@ class FlutterQuillLocalizationsVi extends FlutterQuillLocalizations { @override String get insertImage => 'Chèn hình ảnh'; + + @override + String get pickAPhotoFromYourGallery => 'Pick a photo from your gallery'; + + @override + String get takeAPhotoUsingYourCamera => + 'Take a photo using your phone camera'; + + @override + String get pasteAPhotoUsingALink => 'Paste a photo using a link'; + + @override + String get pickAVideoFromYourGallery => 'Pick a video from your gallery'; + + @override + String get recordAVideoUsingYourCamera => + 'Record a video using your phone camera'; + + @override + String get pasteAVideoUsingALink => 'Paste a video using a link'; } diff --git a/lib/src/l10n/generated/quill_localizations_zh.dart b/lib/src/l10n/generated/quill_localizations_zh.dart index 5c7a42327..59ba0ce1e 100644 --- a/lib/src/l10n/generated/quill_localizations_zh.dart +++ b/lib/src/l10n/generated/quill_localizations_zh.dart @@ -225,6 +225,26 @@ class FlutterQuillLocalizationsZh extends FlutterQuillLocalizations { @override String get insertImage => '插入图像'; + + @override + String get pickAPhotoFromYourGallery => 'Pick a photo from your gallery'; + + @override + String get takeAPhotoUsingYourCamera => + 'Take a photo using your phone camera'; + + @override + String get pasteAPhotoUsingALink => 'Paste a photo using a link'; + + @override + String get pickAVideoFromYourGallery => 'Pick a video from your gallery'; + + @override + String get recordAVideoUsingYourCamera => + 'Record a video using your phone camera'; + + @override + String get pasteAVideoUsingALink => 'Paste a video using a link'; } /// The translations for Chinese, as used in China (`zh_CN`). diff --git a/lib/src/l10n/quill_en.arb b/lib/src/l10n/quill_en.arb index c0ec79673..0d4d737f0 100644 --- a/lib/src/l10n/quill_en.arb +++ b/lib/src/l10n/quill_en.arb @@ -73,5 +73,11 @@ "photo": "Photo", "image": "Image", "caseSensitivityAndWholeWordSearch": "Case sensitivity and whole word search", - "insertImage": "Insert image" + "insertImage": "Insert image", + "pickAPhotoFromYourGallery": "Pick a photo from your gallery", + "takeAPhotoUsingYourCamera": "Take a photo using your phone camera", + "pasteAPhotoUsingALink": "Paste a photo using a link", + "pickAVideoFromYourGallery": "Pick a video from your gallery", + "recordAVideoUsingYourCamera": "Record a video using your phone camera", + "pasteAVideoUsingALink": "Paste a video using a link" } diff --git a/lib/src/l10n/untranslated.json b/lib/src/l10n/untranslated.json index 9e26dfeeb..d6c2ff3d2 100644 --- a/lib/src/l10n/untranslated.json +++ b/lib/src/l10n/untranslated.json @@ -1 +1,298 @@ -{} \ No newline at end of file +{ + "ar": [ + "pickAPhotoFromYourGallery", + "takeAPhotoUsingYourCamera", + "pasteAPhotoUsingALink", + "pickAVideoFromYourGallery", + "recordAVideoUsingYourCamera", + "pasteAVideoUsingALink" + ], + + "bg": [ + "pickAPhotoFromYourGallery", + "takeAPhotoUsingYourCamera", + "pasteAPhotoUsingALink", + "pickAVideoFromYourGallery", + "recordAVideoUsingYourCamera", + "pasteAVideoUsingALink" + ], + + "bn": [ + "pickAPhotoFromYourGallery", + "takeAPhotoUsingYourCamera", + "pasteAPhotoUsingALink", + "pickAVideoFromYourGallery", + "recordAVideoUsingYourCamera", + "pasteAVideoUsingALink" + ], + + "cs": [ + "pickAPhotoFromYourGallery", + "takeAPhotoUsingYourCamera", + "pasteAPhotoUsingALink", + "pickAVideoFromYourGallery", + "recordAVideoUsingYourCamera", + "pasteAVideoUsingALink" + ], + + "da": [ + "pickAPhotoFromYourGallery", + "takeAPhotoUsingYourCamera", + "pasteAPhotoUsingALink", + "pickAVideoFromYourGallery", + "recordAVideoUsingYourCamera", + "pasteAVideoUsingALink" + ], + + "de": [ + "pickAPhotoFromYourGallery", + "takeAPhotoUsingYourCamera", + "pasteAPhotoUsingALink", + "pickAVideoFromYourGallery", + "recordAVideoUsingYourCamera", + "pasteAVideoUsingALink" + ], + + "en_US": [ + "pickAPhotoFromYourGallery", + "takeAPhotoUsingYourCamera", + "pasteAPhotoUsingALink", + "pickAVideoFromYourGallery", + "recordAVideoUsingYourCamera", + "pasteAVideoUsingALink" + ], + + "es": [ + "pickAPhotoFromYourGallery", + "takeAPhotoUsingYourCamera", + "pasteAPhotoUsingALink", + "pickAVideoFromYourGallery", + "recordAVideoUsingYourCamera", + "pasteAVideoUsingALink" + ], + + "fa": [ + "pickAPhotoFromYourGallery", + "takeAPhotoUsingYourCamera", + "pasteAPhotoUsingALink", + "pickAVideoFromYourGallery", + "recordAVideoUsingYourCamera", + "pasteAVideoUsingALink" + ], + + "fr": [ + "pickAPhotoFromYourGallery", + "takeAPhotoUsingYourCamera", + "pasteAPhotoUsingALink", + "pickAVideoFromYourGallery", + "recordAVideoUsingYourCamera", + "pasteAVideoUsingALink" + ], + + "he": [ + "pickAPhotoFromYourGallery", + "takeAPhotoUsingYourCamera", + "pasteAPhotoUsingALink", + "pickAVideoFromYourGallery", + "recordAVideoUsingYourCamera", + "pasteAVideoUsingALink" + ], + + "hi": [ + "pickAPhotoFromYourGallery", + "takeAPhotoUsingYourCamera", + "pasteAPhotoUsingALink", + "pickAVideoFromYourGallery", + "recordAVideoUsingYourCamera", + "pasteAVideoUsingALink" + ], + + "id": [ + "pickAPhotoFromYourGallery", + "takeAPhotoUsingYourCamera", + "pasteAPhotoUsingALink", + "pickAVideoFromYourGallery", + "recordAVideoUsingYourCamera", + "pasteAVideoUsingALink" + ], + + "it": [ + "pickAPhotoFromYourGallery", + "takeAPhotoUsingYourCamera", + "pasteAPhotoUsingALink", + "pickAVideoFromYourGallery", + "recordAVideoUsingYourCamera", + "pasteAVideoUsingALink" + ], + + "ja": [ + "pickAPhotoFromYourGallery", + "takeAPhotoUsingYourCamera", + "pasteAPhotoUsingALink", + "pickAVideoFromYourGallery", + "recordAVideoUsingYourCamera", + "pasteAVideoUsingALink" + ], + + "ko": [ + "pickAPhotoFromYourGallery", + "takeAPhotoUsingYourCamera", + "pasteAPhotoUsingALink", + "pickAVideoFromYourGallery", + "recordAVideoUsingYourCamera", + "pasteAVideoUsingALink" + ], + + "ms": [ + "pickAPhotoFromYourGallery", + "takeAPhotoUsingYourCamera", + "pasteAPhotoUsingALink", + "pickAVideoFromYourGallery", + "recordAVideoUsingYourCamera", + "pasteAVideoUsingALink" + ], + + "nl": [ + "pickAPhotoFromYourGallery", + "takeAPhotoUsingYourCamera", + "pasteAPhotoUsingALink", + "pickAVideoFromYourGallery", + "recordAVideoUsingYourCamera", + "pasteAVideoUsingALink" + ], + + "no": [ + "pickAPhotoFromYourGallery", + "takeAPhotoUsingYourCamera", + "pasteAPhotoUsingALink", + "pickAVideoFromYourGallery", + "recordAVideoUsingYourCamera", + "pasteAVideoUsingALink" + ], + + "pl": [ + "pickAPhotoFromYourGallery", + "takeAPhotoUsingYourCamera", + "pasteAPhotoUsingALink", + "pickAVideoFromYourGallery", + "recordAVideoUsingYourCamera", + "pasteAVideoUsingALink" + ], + + "pt": [ + "pickAPhotoFromYourGallery", + "takeAPhotoUsingYourCamera", + "pasteAPhotoUsingALink", + "pickAVideoFromYourGallery", + "recordAVideoUsingYourCamera", + "pasteAVideoUsingALink" + ], + + "pt_BR": [ + "pickAPhotoFromYourGallery", + "takeAPhotoUsingYourCamera", + "pasteAPhotoUsingALink", + "pickAVideoFromYourGallery", + "recordAVideoUsingYourCamera", + "pasteAVideoUsingALink" + ], + + "ru": [ + "pickAPhotoFromYourGallery", + "takeAPhotoUsingYourCamera", + "pasteAPhotoUsingALink", + "pickAVideoFromYourGallery", + "recordAVideoUsingYourCamera", + "pasteAVideoUsingALink" + ], + + "sr": [ + "pickAPhotoFromYourGallery", + "takeAPhotoUsingYourCamera", + "pasteAPhotoUsingALink", + "pickAVideoFromYourGallery", + "recordAVideoUsingYourCamera", + "pasteAVideoUsingALink" + ], + + "sw": [ + "pickAPhotoFromYourGallery", + "takeAPhotoUsingYourCamera", + "pasteAPhotoUsingALink", + "pickAVideoFromYourGallery", + "recordAVideoUsingYourCamera", + "pasteAVideoUsingALink" + ], + + "tk": [ + "pickAPhotoFromYourGallery", + "takeAPhotoUsingYourCamera", + "pasteAPhotoUsingALink", + "pickAVideoFromYourGallery", + "recordAVideoUsingYourCamera", + "pasteAVideoUsingALink" + ], + + "tr": [ + "pickAPhotoFromYourGallery", + "takeAPhotoUsingYourCamera", + "pasteAPhotoUsingALink", + "pickAVideoFromYourGallery", + "recordAVideoUsingYourCamera", + "pasteAVideoUsingALink" + ], + + "uk": [ + "pickAPhotoFromYourGallery", + "takeAPhotoUsingYourCamera", + "pasteAPhotoUsingALink", + "pickAVideoFromYourGallery", + "recordAVideoUsingYourCamera", + "pasteAVideoUsingALink" + ], + + "ur": [ + "pickAPhotoFromYourGallery", + "takeAPhotoUsingYourCamera", + "pasteAPhotoUsingALink", + "pickAVideoFromYourGallery", + "recordAVideoUsingYourCamera", + "pasteAVideoUsingALink" + ], + + "vi": [ + "pickAPhotoFromYourGallery", + "takeAPhotoUsingYourCamera", + "pasteAPhotoUsingALink", + "pickAVideoFromYourGallery", + "recordAVideoUsingYourCamera", + "pasteAVideoUsingALink" + ], + + "zh": [ + "pickAPhotoFromYourGallery", + "takeAPhotoUsingYourCamera", + "pasteAPhotoUsingALink", + "pickAVideoFromYourGallery", + "recordAVideoUsingYourCamera", + "pasteAVideoUsingALink" + ], + + "zh_CN": [ + "pickAPhotoFromYourGallery", + "takeAPhotoUsingYourCamera", + "pasteAPhotoUsingALink", + "pickAVideoFromYourGallery", + "recordAVideoUsingYourCamera", + "pasteAVideoUsingALink" + ], + + "zh_HK": [ + "pickAPhotoFromYourGallery", + "takeAPhotoUsingYourCamera", + "pasteAPhotoUsingALink", + "pickAVideoFromYourGallery", + "recordAVideoUsingYourCamera", + "pasteAVideoUsingALink" + ] +} diff --git a/lib/translations.dart b/lib/translations.dart index d54eb374b..002a538f3 100644 --- a/lib/translations.dart +++ b/lib/translations.dart @@ -1,3 +1,4 @@ library flutter_quill.translations; export 'src/l10n/extensions/localizations.dart'; +export 'src/l10n/widgets/localizations.dart';