Add ability to paste images (#950)

pull/951/head
Till Friebe 3 years ago committed by GitHub
parent ffbd50cd04
commit cacea0074d
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
  1. 4
      example/linux/flutter/generated_plugin_registrant.cc
  2. 1
      example/linux/flutter/generated_plugins.cmake
  3. 2
      example/macos/Flutter/GeneratedPluginRegistrant.swift
  4. 3
      example/windows/flutter/generated_plugin_registrant.cc
  5. 1
      example/windows/flutter/generated_plugins.cmake
  6. 8
      lib/src/widgets/editor.dart
  7. 55
      lib/src/widgets/raw_editor.dart
  8. 1
      pubspec.yaml

@ -6,9 +6,13 @@
#include "generated_plugin_registrant.h" #include "generated_plugin_registrant.h"
#include <pasteboard/pasteboard_plugin.h>
#include <url_launcher_linux/url_launcher_plugin.h> #include <url_launcher_linux/url_launcher_plugin.h>
void fl_register_plugins(FlPluginRegistry* registry) { void fl_register_plugins(FlPluginRegistry* registry) {
g_autoptr(FlPluginRegistrar) pasteboard_registrar =
fl_plugin_registry_get_registrar_for_plugin(registry, "PasteboardPlugin");
pasteboard_plugin_register_with_registrar(pasteboard_registrar);
g_autoptr(FlPluginRegistrar) url_launcher_linux_registrar = g_autoptr(FlPluginRegistrar) url_launcher_linux_registrar =
fl_plugin_registry_get_registrar_for_plugin(registry, "UrlLauncherPlugin"); fl_plugin_registry_get_registrar_for_plugin(registry, "UrlLauncherPlugin");
url_launcher_plugin_register_with_registrar(url_launcher_linux_registrar); url_launcher_plugin_register_with_registrar(url_launcher_linux_registrar);

@ -3,6 +3,7 @@
# #
list(APPEND FLUTTER_PLUGIN_LIST list(APPEND FLUTTER_PLUGIN_LIST
pasteboard
url_launcher_linux url_launcher_linux
) )

@ -6,11 +6,13 @@ import FlutterMacOS
import Foundation import Foundation
import device_info_plus_macos import device_info_plus_macos
import pasteboard
import path_provider_macos import path_provider_macos
import url_launcher_macos import url_launcher_macos
func RegisterGeneratedPlugins(registry: FlutterPluginRegistry) { func RegisterGeneratedPlugins(registry: FlutterPluginRegistry) {
DeviceInfoPlusMacosPlugin.register(with: registry.registrar(forPlugin: "DeviceInfoPlusMacosPlugin")) DeviceInfoPlusMacosPlugin.register(with: registry.registrar(forPlugin: "DeviceInfoPlusMacosPlugin"))
PasteboardPlugin.register(with: registry.registrar(forPlugin: "PasteboardPlugin"))
PathProviderPlugin.register(with: registry.registrar(forPlugin: "PathProviderPlugin")) PathProviderPlugin.register(with: registry.registrar(forPlugin: "PathProviderPlugin"))
UrlLauncherPlugin.register(with: registry.registrar(forPlugin: "UrlLauncherPlugin")) UrlLauncherPlugin.register(with: registry.registrar(forPlugin: "UrlLauncherPlugin"))
} }

@ -6,9 +6,12 @@
#include "generated_plugin_registrant.h" #include "generated_plugin_registrant.h"
#include <pasteboard/pasteboard_plugin.h>
#include <url_launcher_windows/url_launcher_windows.h> #include <url_launcher_windows/url_launcher_windows.h>
void RegisterPlugins(flutter::PluginRegistry* registry) { void RegisterPlugins(flutter::PluginRegistry* registry) {
PasteboardPluginRegisterWithRegistrar(
registry->GetRegistrarForPlugin("PasteboardPlugin"));
UrlLauncherWindowsRegisterWithRegistrar( UrlLauncherWindowsRegisterWithRegistrar(
registry->GetRegistrarForPlugin("UrlLauncherWindows")); registry->GetRegistrarForPlugin("UrlLauncherWindows"));
} }

@ -3,6 +3,7 @@
# #
list(APPEND FLUTTER_PLUGIN_LIST list(APPEND FLUTTER_PLUGIN_LIST
pasteboard
url_launcher_windows url_launcher_windows
) )

@ -1,4 +1,5 @@
import 'dart:math' as math; import 'dart:math' as math;
import 'dart:typed_data';
import 'package:flutter/cupertino.dart'; import 'package:flutter/cupertino.dart';
import 'package:flutter/foundation.dart'; import 'package:flutter/foundation.dart';
@ -175,6 +176,7 @@ class QuillEditor extends StatefulWidget {
this.locale, this.locale,
this.floatingCursorDisabled = false, this.floatingCursorDisabled = false,
this.textSelectionControls, this.textSelectionControls,
this.onImagePaste,
Key? key}) Key? key})
: super(key: key); : super(key: key);
@ -377,6 +379,11 @@ class QuillEditor extends StatefulWidget {
/// will be used /// will be used
final TextSelectionControls? textSelectionControls; final TextSelectionControls? textSelectionControls;
/// Callback when the user pastes the given image.
///
/// Returns the url of the image if the image should be inserted.
final Future<String?> Function(Uint8List imageBytes)? onImagePaste;
@override @override
QuillEditorState createState() => QuillEditorState(); QuillEditorState createState() => QuillEditorState();
} }
@ -499,6 +506,7 @@ class QuillEditorState extends State<QuillEditor>
linkActionPickerDelegate: widget.linkActionPickerDelegate, linkActionPickerDelegate: widget.linkActionPickerDelegate,
customStyleBuilder: widget.customStyleBuilder, customStyleBuilder: widget.customStyleBuilder,
floatingCursorDisabled: widget.floatingCursorDisabled, floatingCursorDisabled: widget.floatingCursorDisabled,
onImagePaste: widget.onImagePaste,
); );
final editor = I18n( final editor = I18n(

@ -1,6 +1,7 @@
import 'dart:async'; import 'dart:async';
import 'dart:convert'; import 'dart:convert';
import 'dart:math' as math; import 'dart:math' as math;
import 'dart:typed_data';
import 'package:flutter/cupertino.dart'; import 'package:flutter/cupertino.dart';
import 'package:flutter/foundation.dart'; import 'package:flutter/foundation.dart';
@ -9,6 +10,7 @@ import 'package:flutter/rendering.dart';
import 'package:flutter/scheduler.dart'; import 'package:flutter/scheduler.dart';
import 'package:flutter/services.dart'; import 'package:flutter/services.dart';
import 'package:flutter_keyboard_visibility/flutter_keyboard_visibility.dart'; import 'package:flutter_keyboard_visibility/flutter_keyboard_visibility.dart';
import 'package:pasteboard/pasteboard.dart';
import 'package:tuple/tuple.dart'; import 'package:tuple/tuple.dart';
import '../models/documents/attribute.dart'; import '../models/documents/attribute.dart';
@ -72,7 +74,8 @@ class RawEditor extends StatefulWidget {
this.scrollPhysics, this.scrollPhysics,
this.linkActionPickerDelegate = defaultLinkActionPickerDelegate, this.linkActionPickerDelegate = defaultLinkActionPickerDelegate,
this.customStyleBuilder, this.customStyleBuilder,
this.floatingCursorDisabled = false}) this.floatingCursorDisabled = false,
this.onImagePaste})
: assert(maxHeight == null || maxHeight > 0, 'maxHeight cannot be null'), : assert(maxHeight == null || maxHeight > 0, 'maxHeight cannot be null'),
assert(minHeight == null || minHeight >= 0, 'minHeight cannot be null'), assert(minHeight == null || minHeight >= 0, 'minHeight cannot be null'),
assert(maxHeight == null || minHeight == null || maxHeight >= minHeight, assert(maxHeight == null || minHeight == null || maxHeight >= minHeight,
@ -219,6 +222,8 @@ class RawEditor extends StatefulWidget {
/// See [Scrollable.physics]. /// See [Scrollable.physics].
final ScrollPhysics? scrollPhysics; final ScrollPhysics? scrollPhysics;
final Future<String?> Function(Uint8List imageBytes)? onImagePaste;
/// Builder function for embeddable objects. /// Builder function for embeddable objects.
final EmbedsBuilder embedBuilder; final EmbedsBuilder embedBuilder;
final LinkActionPickerDelegate linkActionPickerDelegate; final LinkActionPickerDelegate linkActionPickerDelegate;
@ -1022,25 +1027,45 @@ class RawEditorState extends EditorState
} }
// Snapshot the input before using `await`. // Snapshot the input before using `await`.
// See https://github.com/flutter/flutter/issues/11427 // See https://github.com/flutter/flutter/issues/11427
final data = await Clipboard.getData(Clipboard.kTextPlain); final text = await Clipboard.getData(Clipboard.kTextPlain);
if (data == null) { if (text != null) {
_replaceText(
ReplaceTextIntent(textEditingValue, text.text!, selection, cause));
bringIntoView(textEditingValue.selection.extent);
// Collapse the selection and hide the toolbar and handles.
userUpdateTextEditingValue(
TextEditingValue(
text: textEditingValue.text,
selection:
TextSelection.collapsed(offset: textEditingValue.selection.end),
),
cause,
);
return; return;
} }
_replaceText( if (widget.onImagePaste != null) {
ReplaceTextIntent(textEditingValue, data.text!, selection, cause)); final image = await Pasteboard.image;
bringIntoView(textEditingValue.selection.extent); if (image == null) {
return;
}
// Collapse the selection and hide the toolbar and handles. final imageUrl = await widget.onImagePaste!(image);
userUpdateTextEditingValue( if (imageUrl == null) {
TextEditingValue( return;
text: textEditingValue.text, }
selection:
TextSelection.collapsed(offset: textEditingValue.selection.end), controller.replaceText(
), textEditingValue.selection.end,
cause, 0,
); BlockEmbed.image(imageUrl),
null,
);
}
} }
/// Select the entire text value. /// Select the entire text value.

@ -24,6 +24,7 @@ dependencies:
i18n_extension: ^5.0.1 i18n_extension: ^5.0.1
device_info_plus: ^4.0.0 device_info_plus: ^4.0.0
platform: ^3.1.0 platform: ^3.1.0
pasteboard: ^0.2.0
dev_dependencies: dev_dependencies:
flutter_test: flutter_test:

Loading…
Cancel
Save