ソースを参照

Removed dependency on image_picker plugin

Anatoly Pulyaevskiy 5 年 前
コミット
94859a3b3b

+ 82
- 22
doc/images.md ファイルの表示

@@ -1,40 +1,54 @@
1 1
 ## Images
2 2
 
3
-> Note that described API is considered experimental and is likely to be
3
+> Note that Image API is considered experimental and is likely to be
4 4
 > changed in backward incompatible ways. If this happens all changes will be
5 5
 > described in detail in the changelog to simplify upgrading.
6 6
 
7
-Zefyr (and Notus) supports embedded images. In order to handle images in
7
+Zefyr supports embedding images. In order to handle images in
8 8
 your application you need to implement `ZefyrImageDelegate` interface which
9 9
 looks like this:
10 10
 
11 11
 ```dart
12 12
 abstract class ZefyrImageDelegate<S> {
13
-  /// Builds image widget for specified [imageSource] and [context].
14
-  Widget buildImage(BuildContext context, String imageSource);
13
+  /// Unique key to identify camera source.
14
+  S get cameraSource;
15
+
16
+  /// Unique key to identify gallery source.
17
+  S get gallerySource;
18
+
19
+  /// Builds image widget for specified image [key].
20
+  ///
21
+  /// The [key] argument contains value which was previously returned from
22
+  /// [pickImage].
23
+  Widget buildImage(BuildContext context, String key);
15 24
 
16 25
   /// Picks an image from specified [source].
17 26
   ///
18 27
   /// Returns unique string key for the selected image. Returned key is stored
19 28
   /// in the document.
29
+  ///
30
+  /// Depending on your application returned key may represent a path to
31
+  /// an image file on user's device, an HTTP link, or an identifier generated
32
+  /// by a file hosting service like AWS S3 or Google Drive.
20 33
   Future<String> pickImage(S source);
21 34
 }
22 35
 ```
23 36
 
24
-Zefyr comes with default implementation which exists mostly to provide an
25
-example and a starting point for your own version.
37
+There is no default implementation of this interface since resolving image
38
+sources is always application-specific.
26 39
 
27
-It is recommended to always have your own implementation specific to your
28
-application.
40
+> Note that prior to 0.7.0 Zefyr did provide simple default implementation of
41
+> `ZefyrImageDelegate` however it was removed as it introduced unnecessary
42
+> dependency on `image_picker` plugin.
29 43
 
30 44
 ### Implementing ZefyrImageDelegate
31 45
 
46
+For this example we will use [image_picker](https://pub.dev/packages/image_picker)
47
+plugin which allows us to select images from device's camera or photo gallery.
48
+
32 49
 Let's start from the `pickImage` method:
33 50
 
34 51
 ```dart
35
-// Currently Zefyr depends on image_picker plugin to show camera or image gallery.
36
-// (note that in future versions this may change so that users can choose their
37
-// own plugin and define custom sources)
38 52
 import 'package:image_picker/image_picker.dart';
39 53
 
40 54
 class MyAppZefyrImageDelegate implements ZefyrImageDelegate<ImageSource> {
@@ -53,17 +67,17 @@ camera or gallery), handling result of selection and returning a string value
53 67
 which essentially serves as an identifier for the image.
54 68
 
55 69
 Returned value is stored in the document Delta and later on used to build the
56
-appropriate `Widget`.
70
+appropriate image widget.
57 71
 
58 72
 It is up to the developer to define what this value represents.
59 73
 
60 74
 In the above example we simply return a full path to the file on user's device,
61 75
 e.g. `file:///Users/something/something/image.jpg`. Some other examples
62
-may include a web link, `https://myapp.com/images/some.jpg` or just some
63
-arbitrary string like an ID.
76
+may include a web link, `https://myapp.com/images/some.jpg` or an
77
+arbitrary string like an identifier of an image in a cloud storage like AWS S3.
64 78
 
65 79
 For instance, if you upload files to your server you can initiate this task
66
-in `pickImage`, for instance:
80
+in `pickImage` as follows:
67 81
 
68 82
 ```dart
69 83
 class MyAppZefyrImageDelegate implements ZefyrImageDelegate<ImageSource> {
@@ -84,7 +98,7 @@ class MyAppZefyrImageDelegate implements ZefyrImageDelegate<ImageSource> {
84 98
 
85 99
 Next we need to implement `buildImage`. This method takes `imageSource` argument
86 100
 which contains that same string you returned from `pickImage`. Here you can
87
-use this value to create a Flutter `Widget` which renders the image. Normally
101
+use this value to create a Flutter widget which renders the image. Normally
88 102
 you would return the standard `Image` widget from this method, but it is not
89 103
 a requirement. You are free to create a custom widget which, for instance,
90 104
 shows progress of upload operation that you initiated in the `pickImage` call.
@@ -97,12 +111,58 @@ class MyAppZefyrImageDelegate implements ZefyrImageDelegate<ImageSource> {
97 111
   // ...
98 112
 
99 113
   @override
100
-  Widget buildImage(BuildContext context, String imageSource) {
101
-    final file = new File.fromUri(Uri.parse(imageSource));
102
-    /// Create standard [FileImage] provider. If [imageSource] was an HTTP link
114
+  Widget buildImage(BuildContext context, String key) {
115
+    final file = File.fromUri(Uri.parse(key));
116
+    /// Create standard [FileImage] provider. If [key] was an HTTP link
103 117
     /// we could use [NetworkImage] instead.
104
-    final image = new FileImage(file);
105
-    return new Image(image: image);
118
+    final image = FileImage(file);
119
+    return Image(image: image);
120
+  }
121
+}
122
+```
123
+
124
+There is two more overrides we need to implement which configure source types
125
+used by Zefyr toolbar:
126
+
127
+```dart
128
+class MyAppZefyrImageDelegate implements ZefyrImageDelegate<ImageSource> {
129
+  // ...
130
+  @override
131
+  ImageSource get cameraSource => ImageSource.camera;
132
+
133
+  @override
134
+  ImageSource get gallerySource => ImageSource.gallery;
135
+}
136
+```
137
+
138
+Now our image delegate is ready to be used by Zefyr so the last step is to
139
+pass it to Zefyr editor:
140
+
141
+```dart
142
+import 'package:zefyr/zefyr.dart'
143
+
144
+class MyAppPageState extends State<MyAppPage> {
145
+  FocueNode _focusNode = FocusNode();
146
+  ZefyrController _controller;
147
+
148
+  // ...
149
+
150
+  @override
151
+  Widget build(BuildContext context) {
152
+    final editor = new ZefyrEditor(
153
+      focusNode: _focusNode,
154
+      controller: _controller,
155
+      imageDelegate: MyAppZefyrImageDelegate(),
156
+    );
157
+
158
+    // ... do more with this page's layout
159
+
160
+    return ZefyrScaffold(
161
+      child: Container(
162
+        // ... customize
163
+        child: editor,
164
+      )
165
+    );
106 166
   }
107 167
 }
108 168
 ```
@@ -111,4 +171,4 @@ class MyAppZefyrImageDelegate implements ZefyrImageDelegate<ImageSource> {
111 171
 
112 172
 * [Heuristics][heuristics]
113 173
 
114
-[heuristics]: /doc/heuristics.md
174
+[heuristics]: /doc/heuristics.md

+ 8
- 0
packages/zefyr/CHANGELOG.md ファイルの表示

@@ -10,6 +10,14 @@ This release contains breaking changes.
10 10
     - `ZefyrMode.view`: the same as `enabled: false`, read-only.
11 11
 * Added optional `selectionControls` field to `ZefyrEditor` and `ZefyrEditableText`. If not provided
12 12
   then by default uses platform-specific implementation.
13
+* Added support for "selectAll" action in selection toolbar.
14
+* Breaking change: removed `ZefyrDefaultImageDelegate` as well as dependency on
15
+  `image_picker` plugin. Users are required to provide their own implementation. If image delegate
16
+  is not provided then image toolbar button is disabled.
17
+* Breaking change: added `ZefyrImageDelegate.cameraSource` and `ZefyrImageDelegate.gallerySource`
18
+  fields. For users of `image_picker` plugin these should return `ImageSource.camera` and
19
+  `ImageSource.gallery` respectively. See documentation on implementing image support for more
20
+  details.
13 21
 
14 22
 ## 0.6.0
15 23
 

+ 5
- 0
packages/zefyr/example/lib/src/form.dart ファイルの表示

@@ -1,7 +1,12 @@
1
+// Copyright (c) 2018, the Zefyr project authors.  Please see the AUTHORS file
2
+// for details. All rights reserved. Use of this source code is governed by a
3
+// BSD-style license that can be found in the LICENSE file.
4
+
1 5
 import 'package:flutter/material.dart';
2 6
 import 'package:zefyr/zefyr.dart';
3 7
 
4 8
 import 'full_page.dart';
9
+import 'images.dart';
5 10
 
6 11
 class FormEmbeddedScreen extends StatefulWidget {
7 12
   @override

+ 6
- 17
packages/zefyr/example/lib/src/full_page.dart ファイルの表示

@@ -1,3 +1,7 @@
1
+// Copyright (c) 2018, the Zefyr project authors.  Please see the AUTHORS file
2
+// for details. All rights reserved. Use of this source code is governed by a
3
+// BSD-style license that can be found in the LICENSE file.
4
+
1 5
 import 'dart:async';
2 6
 import 'dart:convert';
3 7
 
@@ -5,6 +9,8 @@ import 'package:flutter/material.dart';
5 9
 import 'package:quill_delta/quill_delta.dart';
6 10
 import 'package:zefyr/zefyr.dart';
7 11
 
12
+import 'images.dart';
13
+
8 14
 class ZefyrLogo extends StatelessWidget {
9 15
   @override
10 16
   Widget build(BuildContext context) {
@@ -106,20 +112,3 @@ class _FullPageEditorScreenState extends State<FullPageEditorScreen> {
106 112
     });
107 113
   }
108 114
 }
109
-
110
-/// Custom image delegate used by this example to load image from application
111
-/// assets.
112
-///
113
-/// Default image delegate only supports [FileImage]s.
114
-class CustomImageDelegate extends ZefyrDefaultImageDelegate {
115
-  @override
116
-  Widget buildImage(BuildContext context, String imageSource) {
117
-    // We use custom "asset" scheme to distinguish asset images from other files.
118
-    if (imageSource.startsWith('asset://')) {
119
-      final asset = new AssetImage(imageSource.replaceFirst('asset://', ''));
120
-      return new Image(image: asset);
121
-    } else {
122
-      return super.buildImage(context, imageSource);
123
-    }
124
-  }
125
-}

+ 40
- 0
packages/zefyr/example/lib/src/images.dart ファイルの表示

@@ -0,0 +1,40 @@
1
+// Copyright (c) 2018, the Zefyr project authors.  Please see the AUTHORS file
2
+// for details. All rights reserved. Use of this source code is governed by a
3
+// BSD-style license that can be found in the LICENSE file.
4
+
5
+import 'dart:io';
6
+
7
+import 'package:flutter/widgets.dart';
8
+import 'package:image_picker/image_picker.dart';
9
+import 'package:zefyr/zefyr.dart';
10
+
11
+/// Custom image delegate used by this example to load image from application
12
+/// assets.
13
+class CustomImageDelegate implements ZefyrImageDelegate<ImageSource> {
14
+  @override
15
+  ImageSource get cameraSource => ImageSource.camera;
16
+
17
+  @override
18
+  ImageSource get gallerySource => ImageSource.gallery;
19
+
20
+  @override
21
+  Future<String> pickImage(ImageSource source) async {
22
+    final file = await ImagePicker.pickImage(source: source);
23
+    if (file == null) return null;
24
+    return file.uri.toString();
25
+  }
26
+
27
+  @override
28
+  Widget buildImage(BuildContext context, String key) {
29
+    // We use custom "asset" scheme to distinguish asset images from other files.
30
+    if (key.startsWith('asset://')) {
31
+      final asset = AssetImage(key.replaceFirst('asset://', ''));
32
+      return Image(image: asset);
33
+    } else {
34
+      // Otherwise assume this is a file stored locally on user's device.
35
+      final file = File.fromUri(Uri.parse(key));
36
+      final image = FileImage(file);
37
+      return new Image(image: image);
38
+    }
39
+  }
40
+}

+ 5
- 17
packages/zefyr/example/lib/src/view.dart ファイルの表示

@@ -1,3 +1,7 @@
1
+// Copyright (c) 2018, the Zefyr project authors.  Please see the AUTHORS file
2
+// for details. All rights reserved. Use of this source code is governed by a
3
+// BSD-style license that can be found in the LICENSE file.
4
+
1 5
 import 'dart:convert';
2 6
 
3 7
 import 'package:flutter/material.dart';
@@ -5,6 +9,7 @@ import 'package:quill_delta/quill_delta.dart';
5 9
 import 'package:zefyr/zefyr.dart';
6 10
 
7 11
 import 'full_page.dart';
12
+import 'images.dart';
8 13
 
9 14
 class ViewScreen extends StatefulWidget {
10 15
   @override
@@ -68,20 +73,3 @@ class _ViewScreen extends State<ViewScreen> {
68 73
     );
69 74
   }
70 75
 }
71
-
72
-/// Custom image delegate used by this example to load image from application
73
-/// assets.
74
-///
75
-/// Default image delegate only supports [FileImage]s.
76
-class CustomImageDelegate extends ZefyrDefaultImageDelegate {
77
-  @override
78
-  Widget buildImage(BuildContext context, String imageSource) {
79
-    // We use custom "asset" scheme to distinguish asset images from other files.
80
-    if (imageSource.startsWith('asset://')) {
81
-      final asset = new AssetImage(imageSource.replaceFirst('asset://', ''));
82
-      return new Image(image: asset);
83
-    } else {
84
-      return super.buildImage(context, imageSource);
85
-    }
86
-  }
87
-}

+ 1
- 0
packages/zefyr/example/pubspec.yaml ファイルの表示

@@ -16,6 +16,7 @@ dependencies:
16 16
   # The following adds the Cupertino Icons font to your application.
17 17
   # Use with the CupertinoIcons class for iOS style icons.
18 18
   cupertino_icons: ^0.1.2
19
+  image_picker: ^0.6.1
19 20
   zefyr:
20 21
     path: ../
21 22
 

+ 4
- 3
packages/zefyr/lib/src/widgets/buttons.dart ファイルの表示

@@ -3,7 +3,6 @@
3 3
 // BSD-style license that can be found in the LICENSE file.
4 4
 import 'package:flutter/material.dart';
5 5
 import 'package:flutter/services.dart';
6
-import 'package:image_picker/image_picker.dart';
7 6
 import 'package:notus/notus.dart';
8 7
 import 'package:url_launcher/url_launcher.dart';
9 8
 
@@ -279,14 +278,16 @@ class _ImageButtonState extends State<ImageButton> {
279 278
 
280 279
   void _pickFromCamera() async {
281 280
     final editor = ZefyrToolbar.of(context).editor;
282
-    final image = await editor.imageDelegate.pickImage(ImageSource.camera);
281
+    final image =
282
+        await editor.imageDelegate.pickImage(editor.imageDelegate.cameraSource);
283 283
     if (image != null)
284 284
       editor.formatSelection(NotusAttribute.embed.image(image));
285 285
   }
286 286
 
287 287
   void _pickFromGallery() async {
288 288
     final editor = ZefyrToolbar.of(context).editor;
289
-    final image = await editor.imageDelegate.pickImage(ImageSource.gallery);
289
+    final image = await editor.imageDelegate
290
+        .pickImage(editor.imageDelegate.gallerySource);
290 291
     if (image != null)
291 292
       editor.formatSelection(NotusAttribute.embed.image(image));
292 293
   }

+ 2
- 2
packages/zefyr/lib/src/widgets/editor.dart ファイルの表示

@@ -121,7 +121,7 @@ class _ZefyrEditorState extends State<ZefyrEditor> {
121 121
   @override
122 122
   void initState() {
123 123
     super.initState();
124
-    _imageDelegate = widget.imageDelegate ?? new ZefyrDefaultImageDelegate();
124
+    _imageDelegate = widget.imageDelegate;
125 125
   }
126 126
 
127 127
   @override
@@ -131,7 +131,7 @@ class _ZefyrEditorState extends State<ZefyrEditor> {
131 131
     _scope.controller = widget.controller;
132 132
     _scope.focusNode = widget.focusNode;
133 133
     if (widget.imageDelegate != oldWidget.imageDelegate) {
134
-      _imageDelegate = widget.imageDelegate ?? new ZefyrDefaultImageDelegate();
134
+      _imageDelegate = widget.imageDelegate;
135 135
       _scope.imageDelegate = _imageDelegate;
136 136
     }
137 137
   }

+ 19
- 20
packages/zefyr/lib/src/widgets/image.dart ファイルの表示

@@ -2,44 +2,43 @@
2 2
 // for details. All rights reserved. Use of this source code is governed by a
3 3
 // BSD-style license that can be found in the LICENSE file.
4 4
 import 'dart:async';
5
-import 'dart:io';
6 5
 import 'dart:math' as math;
7 6
 import 'dart:ui' as ui;
8 7
 
9 8
 import 'package:flutter/rendering.dart';
10 9
 import 'package:flutter/widgets.dart';
10
+import 'package:meta/meta.dart';
11 11
 import 'package:notus/notus.dart';
12
-import 'package:image_picker/image_picker.dart';
13 12
 
14 13
 import 'editable_box.dart';
15 14
 
15
+/// Provides interface for embedding images into Zefyr editor.
16
+// TODO: allow configuring image sources and related toolbar buttons.
17
+@experimental
16 18
 abstract class ZefyrImageDelegate<S> {
17
-  /// Builds image widget for specified [imageSource] and [context].
18
-  Widget buildImage(BuildContext context, String imageSource);
19
+  /// Unique key to identify camera source.
20
+  S get cameraSource;
21
+
22
+  /// Unique key to identify gallery source.
23
+  S get gallerySource;
24
+
25
+  /// Builds image widget for specified image [key].
26
+  ///
27
+  /// The [key] argument contains value which was previously returned from
28
+  /// [pickImage] method.
29
+  Widget buildImage(BuildContext context, String key);
19 30
 
20 31
   /// Picks an image from specified [source].
21 32
   ///
22 33
   /// Returns unique string key for the selected image. Returned key is stored
23 34
   /// in the document.
35
+  ///
36
+  /// Depending on your application returned key may represent a path to
37
+  /// an image file on user's device, an HTTP link, or an identifier generated
38
+  /// by a file hosting service like AWS S3 or Google Drive.
24 39
   Future<String> pickImage(S source);
25 40
 }
26 41
 
27
-class ZefyrDefaultImageDelegate implements ZefyrImageDelegate<ImageSource> {
28
-  @override
29
-  Widget buildImage(BuildContext context, String imageSource) {
30
-    final file = new File.fromUri(Uri.parse(imageSource));
31
-    final image = new FileImage(file);
32
-    return new Image(image: image);
33
-  }
34
-
35
-  @override
36
-  Future<String> pickImage(ImageSource source) async {
37
-    final file = await ImagePicker.pickImage(source: source);
38
-    if (file == null) return null;
39
-    return file.uri.toString();
40
-  }
41
-}
42
-
43 42
 class ZefyrImage extends StatefulWidget {
44 43
   const ZefyrImage({Key key, @required this.node, @required this.delegate})
45 44
       : super(key: key);

+ 4
- 8
packages/zefyr/lib/src/widgets/scope.dart ファイルの表示

@@ -25,11 +25,9 @@ class ZefyrScope extends ChangeNotifier {
25 25
   /// Creates a view-only scope.
26 26
   ///
27 27
   /// Normally used in [ZefyrView].
28
-  ZefyrScope.view({
29
-    @required ZefyrMode mode,
30
-    @required ZefyrImageDelegate imageDelegate,
31
-  })  : assert(imageDelegate != null),
32
-        isEditable = false,
28
+  ZefyrScope.view({ZefyrImageDelegate imageDelegate})
29
+      : isEditable = false,
30
+        _mode = ZefyrMode.view,
33 31
         _imageDelegate = imageDelegate;
34 32
 
35 33
   /// Creates editable scope.
@@ -38,12 +36,11 @@ class ZefyrScope extends ChangeNotifier {
38 36
   ZefyrScope.editable({
39 37
     @required ZefyrMode mode,
40 38
     @required ZefyrController controller,
41
-    @required ZefyrImageDelegate imageDelegate,
42 39
     @required FocusNode focusNode,
43 40
     @required FocusScopeNode focusScope,
41
+    ZefyrImageDelegate imageDelegate,
44 42
   })  : assert(mode != null),
45 43
         assert(controller != null),
46
-        assert(imageDelegate != null),
47 44
         assert(focusNode != null),
48 45
         assert(focusScope != null),
49 46
         isEditable = true,
@@ -69,7 +66,6 @@ class ZefyrScope extends ChangeNotifier {
69 66
   ZefyrImageDelegate _imageDelegate;
70 67
   ZefyrImageDelegate get imageDelegate => _imageDelegate;
71 68
   set imageDelegate(ZefyrImageDelegate value) {
72
-    assert(value != null);
73 69
     if (_imageDelegate != value) {
74 70
       _imageDelegate = value;
75 71
       notifyListeners();

+ 1
- 1
packages/zefyr/lib/src/widgets/toolbar.dart ファイルの表示

@@ -259,7 +259,7 @@ class ZefyrToolbarState extends State<ZefyrToolbar>
259 259
       buildButton(context, ZefyrToolbarAction.quote),
260 260
       buildButton(context, ZefyrToolbarAction.code),
261 261
       buildButton(context, ZefyrToolbarAction.horizontalRule),
262
-      ImageButton(),
262
+      if (editor.imageDelegate != null) ImageButton(),
263 263
     ];
264 264
     return buttons;
265 265
   }

+ 1
- 5
packages/zefyr/lib/src/widgets/view.dart ファイルの表示

@@ -9,7 +9,6 @@ import 'code.dart';
9 9
 import 'common.dart';
10 10
 import 'image.dart';
11 11
 import 'list.dart';
12
-import 'mode.dart';
13 12
 import 'paragraph.dart';
14 13
 import 'quote.dart';
15 14
 import 'scope.dart';
@@ -37,10 +36,7 @@ class ZefyrViewState extends State<ZefyrView> {
37 36
   @override
38 37
   void initState() {
39 38
     super.initState();
40
-    _scope = ZefyrScope.view(
41
-      mode: ZefyrMode.view,
42
-      imageDelegate: widget.imageDelegate,
43
-    );
39
+    _scope = ZefyrScope.view(imageDelegate: widget.imageDelegate);
44 40
   }
45 41
 
46 42
   @override

+ 1
- 2
packages/zefyr/pubspec.yaml ファイルの表示

@@ -5,14 +5,13 @@ author: Anatoly Pulyaevskiy <anatoly.pulyaevskiy@gmail.com>
5 5
 homepage: https://github.com/memspace/zefyr
6 6
 
7 7
 environment:
8
-  sdk: '>=2.1.0 <3.0.0'
8
+  sdk: '>=2.2.2 <3.0.0'
9 9
 
10 10
 dependencies:
11 11
   flutter:
12 12
     sdk: flutter
13 13
   collection: ^1.14.6
14 14
   url_launcher: ^5.0.0
15
-  image_picker: ^0.5.0
16 15
   quill_delta: ^1.0.0-dev.1.0
17 16
   notus: ^0.1.0
18 17
   meta: ^1.1.0

+ 11
- 3
packages/zefyr/test/testing.dart ファイルの表示

@@ -22,6 +22,7 @@ class EditorSandBox {
22 22
     NotusDocument document,
23 23
     ZefyrThemeData theme,
24 24
     bool autofocus: false,
25
+    ZefyrImageDelegate imageDelegate,
25 26
   }) {
26 27
     focusNode ??= FocusNode();
27 28
     document ??= NotusDocument.fromDelta(delta);
@@ -31,6 +32,7 @@ class EditorSandBox {
31 32
       controller: controller,
32 33
       focusNode: focusNode,
33 34
       autofocus: autofocus,
35
+      imageDelegate: imageDelegate,
34 36
     );
35 37
 
36 38
     if (theme != null) {
@@ -115,12 +117,17 @@ class EditorSandBox {
115 117
 }
116 118
 
117 119
 class _ZefyrSandbox extends StatefulWidget {
118
-  const _ZefyrSandbox(
119
-      {Key key, this.controller, this.focusNode, this.autofocus})
120
-      : super(key: key);
120
+  const _ZefyrSandbox({
121
+    Key key,
122
+    this.controller,
123
+    this.focusNode,
124
+    this.autofocus,
125
+    this.imageDelegate,
126
+  }) : super(key: key);
121 127
   final ZefyrController controller;
122 128
   final FocusNode focusNode;
123 129
   final bool autofocus;
130
+  final ZefyrImageDelegate imageDelegate;
124 131
 
125 132
   @override
126 133
   _ZefyrSandboxState createState() => _ZefyrSandboxState();
@@ -136,6 +143,7 @@ class _ZefyrSandboxState extends State<_ZefyrSandbox> {
136 143
       focusNode: widget.focusNode,
137 144
       mode: _enabled ? ZefyrMode.edit : ZefyrMode.view,
138 145
       autofocus: widget.autofocus,
146
+      imageDelegate: widget.imageDelegate,
139 147
     );
140 148
   }
141 149
 

+ 34
- 41
packages/zefyr/test/widgets/buttons_test.dart ファイルの表示

@@ -1,8 +1,9 @@
1 1
 // Copyright (c) 2018, the Zefyr project authors.  Please see the AUTHORS file
2 2
 // for details. All rights reserved. Use of this source code is governed by a
3 3
 // BSD-style license that can be found in the LICENSE file.
4
+import 'dart:io';
5
+
4 6
 import 'package:flutter/material.dart';
5
-import 'package:flutter/services.dart';
6 7
 import 'package:flutter_test/flutter_test.dart';
7 8
 import 'package:zefyr/src/widgets/buttons.dart';
8 9
 import 'package:zefyr/zefyr.dart';
@@ -145,21 +146,11 @@ void main() {
145 146
   });
146 147
 
147 148
   group('$ImageButton', () {
148
-    const MethodChannel channel =
149
-        const MethodChannel('plugins.flutter.io/image_picker');
150
-
151
-    final List<MethodCall> log = <MethodCall>[];
152
-
153
-    setUp(() {
154
-      channel.setMockMethodCallHandler((MethodCall methodCall) async {
155
-        log.add(methodCall);
156
-        return '/tmp/test.jpg';
157
-      });
158
-      log.clear();
159
-    });
160
-
161 149
     testWidgets('toggle overlay', (tester) async {
162
-      final editor = new EditorSandBox(tester: tester);
150
+      final editor = new EditorSandBox(
151
+        tester: tester,
152
+        imageDelegate: _TestImageDelegate(),
153
+      );
163 154
       await editor.pumpAndTap();
164 155
       await editor.tapButtonWithIcon(Icons.photo);
165 156
 
@@ -169,41 +160,43 @@ void main() {
169 160
     });
170 161
 
171 162
     testWidgets('pick from camera', (tester) async {
172
-      final editor = new EditorSandBox(tester: tester);
163
+      final editor = new EditorSandBox(
164
+        tester: tester,
165
+        imageDelegate: _TestImageDelegate(),
166
+      );
173 167
       await editor.pumpAndTap();
174 168
       await editor.tapButtonWithIcon(Icons.photo);
175 169
       await editor.tapButtonWithIcon(Icons.photo_camera);
176
-      expect(log, hasLength(1));
177
-      expect(
178
-        log.single,
179
-        isMethodCall(
180
-          'pickImage',
181
-          arguments: <String, dynamic>{
182
-            'source': 0,
183
-            'maxWidth': null,
184
-            'maxHeight': null,
185
-          },
186
-        ),
187
-      );
170
+      expect(find.byType(ZefyrImage), findsOneWidget);
188 171
     });
189 172
 
190 173
     testWidgets('pick from gallery', (tester) async {
191
-      final editor = new EditorSandBox(tester: tester);
174
+      final editor = new EditorSandBox(
175
+        tester: tester,
176
+        imageDelegate: _TestImageDelegate(),
177
+      );
192 178
       await editor.pumpAndTap();
193 179
       await editor.tapButtonWithIcon(Icons.photo);
194 180
       await editor.tapButtonWithIcon(Icons.photo_library);
195
-      expect(log, hasLength(1));
196
-      expect(
197
-        log.single,
198
-        isMethodCall(
199
-          'pickImage',
200
-          arguments: <String, dynamic>{
201
-            'source': 1,
202
-            'maxWidth': null,
203
-            'maxHeight': null,
204
-          },
205
-        ),
206
-      );
181
+      expect(find.byType(ZefyrImage), findsOneWidget);
207 182
     });
208 183
   });
209 184
 }
185
+
186
+class _TestImageDelegate implements ZefyrImageDelegate<String> {
187
+  @override
188
+  Widget buildImage(BuildContext context, String key) {
189
+    return Image.file(File(key));
190
+  }
191
+
192
+  @override
193
+  String get cameraSource => "camera";
194
+
195
+  @override
196
+  String get gallerySource => "gallery";
197
+
198
+  @override
199
+  Future<String> pickImage(String source) {
200
+    return Future.value("file:///tmp/test.jpg");
201
+  }
202
+}

+ 36
- 40
packages/zefyr/test/widgets/image_test.dart ファイルの表示

@@ -1,52 +1,21 @@
1 1
 // Copyright (c) 2018, the Zefyr project authors.  Please see the AUTHORS file
2 2
 // for details. All rights reserved. Use of this source code is governed by a
3 3
 // BSD-style license that can be found in the LICENSE file.
4
+import 'dart:io';
5
+
4 6
 import 'package:flutter/material.dart';
5
-import 'package:flutter/services.dart';
6 7
 import 'package:flutter_test/flutter_test.dart';
7
-import 'package:image_picker/image_picker.dart';
8 8
 import 'package:zefyr/zefyr.dart';
9 9
 
10 10
 import '../testing.dart';
11 11
 
12 12
 void main() {
13
-  group('$ZefyrDefaultImageDelegate', () {
14
-    const MethodChannel channel =
15
-        const MethodChannel('plugins.flutter.io/image_picker');
16
-
17
-    final List<MethodCall> log = <MethodCall>[];
18
-
19
-    setUp(() {
20
-      channel.setMockMethodCallHandler((MethodCall methodCall) async {
21
-        log.add(methodCall);
22
-        return '/tmp/test.jpg';
23
-      });
24
-      log.clear();
25
-    });
26
-
27
-    test('pick image', () async {
28
-      final delegate = new ZefyrDefaultImageDelegate();
29
-      final result = await delegate.pickImage(ImageSource.gallery);
30
-      expect(result, 'file:///tmp/test.jpg');
31
-    });
32
-  });
33
-
34 13
   group('$ZefyrImage', () {
35
-    const MethodChannel channel =
36
-        const MethodChannel('plugins.flutter.io/image_picker');
37
-
38
-    final List<MethodCall> log = <MethodCall>[];
39
-
40
-    setUp(() {
41
-      channel.setMockMethodCallHandler((MethodCall methodCall) async {
42
-        log.add(methodCall);
43
-        return '/tmp/test.jpg';
44
-      });
45
-      log.clear();
46
-    });
47
-
48 14
     testWidgets('embed image', (tester) async {
49
-      final editor = new EditorSandBox(tester: tester);
15
+      final editor = new EditorSandBox(
16
+        tester: tester,
17
+        imageDelegate: _TestImageDelegate(),
18
+      );
50 19
       await editor.pumpAndTap();
51 20
       await editor.tapButtonWithIcon(Icons.photo);
52 21
       await editor.tapButtonWithIcon(Icons.photo_camera);
@@ -62,7 +31,10 @@ void main() {
62 31
 
63 32
     testWidgets('tap on left side of image puts caret before it',
64 33
         (tester) async {
65
-      final editor = new EditorSandBox(tester: tester);
34
+      final editor = new EditorSandBox(
35
+        tester: tester,
36
+        imageDelegate: _TestImageDelegate(),
37
+      );
66 38
       await editor.pumpAndTap();
67 39
       await editor.tapButtonWithIcon(Icons.photo);
68 40
       await editor.tapButtonWithIcon(Icons.photo_camera);
@@ -76,7 +48,10 @@ void main() {
76 48
     });
77 49
 
78 50
     testWidgets('tap right side of image puts caret after it', (tester) async {
79
-      final editor = new EditorSandBox(tester: tester);
51
+      final editor = new EditorSandBox(
52
+        tester: tester,
53
+        imageDelegate: _TestImageDelegate(),
54
+      );
80 55
       await editor.pumpAndTap();
81 56
       await editor.tapButtonWithIcon(Icons.photo);
82 57
       await editor.tapButtonWithIcon(Icons.photo_camera);
@@ -92,7 +67,10 @@ void main() {
92 67
     });
93 68
 
94 69
     testWidgets('selects on long press', (tester) async {
95
-      final editor = new EditorSandBox(tester: tester);
70
+      final editor = new EditorSandBox(
71
+        tester: tester,
72
+        imageDelegate: _TestImageDelegate(),
73
+      );
96 74
       await editor.pumpAndTap();
97 75
       await editor.tapButtonWithIcon(Icons.photo);
98 76
       await editor.tapButtonWithIcon(Icons.photo_camera);
@@ -108,3 +86,21 @@ void main() {
108 86
     });
109 87
   });
110 88
 }
89
+
90
+class _TestImageDelegate implements ZefyrImageDelegate<String> {
91
+  @override
92
+  Widget buildImage(BuildContext context, String key) {
93
+    return Image.file(File(key));
94
+  }
95
+
96
+  @override
97
+  String get cameraSource => "camera";
98
+
99
+  @override
100
+  String get gallerySource => "gallery";
101
+
102
+  @override
103
+  Future<String> pickImage(String source) {
104
+    return Future.value("file:///tmp/test.jpg");
105
+  }
106
+}

+ 19
- 2
packages/zefyr/test/widgets/scope_test.dart ファイルの表示

@@ -16,7 +16,6 @@ void main() {
16 16
       scope = ZefyrScope.editable(
17 17
         mode: ZefyrMode.edit,
18 18
         controller: ZefyrController(doc),
19
-        imageDelegate: ZefyrDefaultImageDelegate(),
20 19
         focusNode: FocusNode(),
21 20
         focusScope: FocusScopeNode(),
22 21
       );
@@ -27,7 +26,7 @@ void main() {
27 26
       scope.addListener(() {
28 27
         notified = true;
29 28
       });
30
-      final delegate = ZefyrDefaultImageDelegate();
29
+      final delegate = _TestImageDelegate();
31 30
       scope.imageDelegate = delegate;
32 31
       expect(notified, isTrue);
33 32
       notified = false;
@@ -74,3 +73,21 @@ void main() {
74 73
     });
75 74
   });
76 75
 }
76
+
77
+class _TestImageDelegate implements ZefyrImageDelegate<String> {
78
+  @override
79
+  Widget buildImage(BuildContext context, String key) {
80
+    return null;
81
+  }
82
+
83
+  @override
84
+  String get cameraSource => null;
85
+
86
+  @override
87
+  String get gallerySource => null;
88
+
89
+  @override
90
+  Future<String> pickImage(String source) {
91
+    return null;
92
+  }
93
+}