Add tests for PreserveInlineStylesRule and fix link editing. Other minor fixes. (#2058)

* Value setting Stateful toolbar buttons derive from base class

* Removed deprecated functions

* Move clipboard actions to QuillController

* Add: Clipboard toolbar buttons

* Translation Justify

* Translation alignJustify

* Fix: Translation en-US

* Misc fixes

---------

Co-authored-by: Douglas Ward <dward@scied.com>
pull/2060/head v10.0.5
AtlasAutocode 8 months ago committed by GitHub
parent faf8f558a8
commit 42d830f037
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
  1. 35
      lib/src/controller/quill_controller.dart
  2. 28
      lib/src/editor/raw_editor/raw_editor_state.dart
  3. 13
      lib/src/editor/widgets/text/text_line.dart
  4. 73
      lib/src/rules/insert.dart
  5. 289
      test/rules/insert_test.dart

@ -8,6 +8,7 @@ import 'package:meta/meta.dart' show experimental;
import '../../quill_delta.dart';
import '../common/structs/image_url.dart';
import '../common/structs/offset_value.dart';
import '../common/utils/embeds.dart';
import '../delta/delta_diff.dart';
import '../delta/delta_x.dart';
import '../document/attribute.dart';
@ -515,6 +516,12 @@ class QuillController extends ChangeNotifier {
Future<bool> clipboardPaste({void Function()? updateEditor}) async {
if (readOnly || !selection.isValid) return true;
final pasteUsingInternalImageSuccess = await _pasteInternalImage();
if (pasteUsingInternalImageSuccess) {
updateEditor?.call();
return true;
}
final pasteUsingHtmlSuccess = await _pasteHTML();
if (pasteUsingHtmlSuccess) {
updateEditor?.call();
@ -574,6 +581,34 @@ class QuillController extends ChangeNotifier {
);
}
/// Return true if can paste internal image
Future<bool> _pasteInternalImage() async {
final copiedImageUrl = _copiedImageUrl;
if (copiedImageUrl != null) {
final index = selection.baseOffset;
final length = selection.extentOffset - index;
replaceText(
index,
length,
BlockEmbed.image(copiedImageUrl.url),
null,
);
if (copiedImageUrl.styleString.isNotEmpty) {
formatText(
getEmbedNode(this, index + 1).offset,
1,
StyleAttribute(copiedImageUrl.styleString),
);
}
_copiedImageUrl = null;
await Clipboard.setData(
const ClipboardData(text: ''),
);
return true;
}
return false;
}
/// Return true if can paste using HTML
Future<bool> _pasteHTML() async {
final clipboardService = ClipboardServiceProvider.instance;

@ -11,7 +11,6 @@ import 'package:flutter/scheduler.dart' show SchedulerBinding;
import 'package:flutter/services.dart'
show
Clipboard,
ClipboardData,
HardwareKeyboard,
KeyDownEvent,
LogicalKeyboardKey,
@ -24,7 +23,6 @@ import '../../common/structs/horizontal_spacing.dart';
import '../../common/structs/offset_value.dart';
import '../../common/structs/vertical_spacing.dart';
import '../../common/utils/cast.dart';
import '../../common/utils/embeds.dart';
import '../../common/utils/platform.dart';
import '../../controller/quill_controller.dart';
import '../../delta/delta_diff.dart';
@ -155,31 +153,6 @@ class QuillRawEditorState extends EditorState
return;
}
// When image copied internally in the editor
final copiedImageUrl = controller.copiedImageUrl;
if (copiedImageUrl != null) {
final index = textEditingValue.selection.baseOffset;
final length = textEditingValue.selection.extentOffset - index;
controller.replaceText(
index,
length,
BlockEmbed.image(copiedImageUrl.url),
null,
);
if (copiedImageUrl.styleString.isNotEmpty) {
controller.formatText(
getEmbedNode(controller, index + 1).offset,
1,
StyleAttribute(copiedImageUrl.styleString),
);
}
controller.copiedImageUrl = null;
await Clipboard.setData(
const ClipboardData(text: ''),
);
return;
}
if (await controller.clipboardPaste()) {
bringIntoView(textEditingValue.selection.extent);
return;
@ -909,6 +882,7 @@ class QuillRawEditorState extends EditorState
for (final attr in style.values) {
controller.formatSelection(attr);
}
controller.formatSelection(attribute);
}
}

@ -931,10 +931,15 @@ class RenderEditableTextLine extends RenderEditableBox {
@override
TextPosition? getPositionAbove(TextPosition position) {
/// Move up by fraction of the default font height, larger font sizes need larger offset
for (var offset = -0.5;; offset -= 0.25) {
final pos = _getPosition(position, offset);
if (pos != position || offset <= -2.0) {
double? maxOffset;
double limit() => maxOffset ??=
_body!.semanticBounds.height / preferredLineHeight(position) + 1;
bool checkLimit(double offset) => offset < 4.0 ? false : offset > limit();
/// Move up by fraction of the default font height, larger font sizes need larger offset, embed images need larger offset
for (var offset = 0.5;; offset += offset < 4 ? 0.25 : 1.0) {
final pos = _getPosition(position, -offset);
if (pos?.offset != position.offset || checkLimit(offset)) {
return pos;
}
}

@ -558,35 +558,43 @@ class PreserveInlineStylesRule extends InsertRule {
final documentDelta = document.toDelta();
final itr = DeltaIterator(documentDelta);
len ??= 0;
var prev = itr.skip(len == 0 ? index : index + 1);
var excludeLinkAtLineStart = false;
if (prev == null || prev.data is! String) return null;
/// Trap for simple insertions at start of line
/// Process simple insertions at start of line
if (len == 0) {
final prevData = prev.data as String;
if (prevData.endsWith('\n')) {
/// If current line is empty get attributes from a prior line
final currLine = itr.next();
final currData =
currLine.data is String ? currLine.data as String : null;
if (currData?.isEmpty == true || currData?.startsWith('\n') == true) {
if (prevData.trimRight().isEmpty) {
final back =
DeltaIterator(documentDelta).skip(index - prevData.length);
if (back != null && back.data is String) {
prev = back;
final currLine = itr.next();
/// Trap for previous is not text with attributes
if (prev?.data is! String) {
prev = currLine;
excludeLinkAtLineStart = true;
} else {
final prevData = prev!.data as String;
if (prevData.endsWith('\n')) {
/// If current line is empty get attributes from a prior line
final currData =
currLine.data is String ? currLine.data as String : null;
if (currData?.startsWith('\n') == true) {
if (prevData.trimRight().isEmpty) {
final back =
DeltaIterator(documentDelta).skip(index - prevData.length);
if (back != null && back.data is String) {
prev = back;
}
}
} else {
prev = currLine;
excludeLinkAtLineStart = true;
}
} else {
prev = currLine;
}
}
}
final attributes = <String, dynamic>{};
if (prev.attributes != null) {
for (final entry in prev.attributes!.entries) {
if (prev?.attributes != null) {
for (final entry in prev!.attributes!.entries) {
if (Attribute.inlineKeys.contains(entry.key)) {
attributes[entry.key] = entry.value;
}
@ -596,29 +604,12 @@ class PreserveInlineStylesRule extends InsertRule {
return null;
}
final text = data;
if (attributes.isEmpty || !attributes.containsKey(Attribute.link.key)) {
return Delta()
..retain(index + (len ?? 0))
..insert(text, attributes);
if (excludeLinkAtLineStart) {
attributes.remove(Attribute.link.key);
}
attributes.remove(Attribute.link.key);
final delta = Delta()
..retain(index + (len ?? 0))
..insert(text, attributes.isEmpty ? null : attributes);
final next = itr.next();
final nextAttributes = next.attributes ?? const <String, dynamic>{};
if (!nextAttributes.containsKey(Attribute.link.key)) {
return delta;
}
if (attributes[Attribute.link.key] == nextAttributes[Attribute.link.key]) {
return Delta()
..retain(index + (len ?? 0))
..insert(text, attributes);
}
return delta;
return Delta()
..retain(index + len)
..insert(data, attributes.isEmpty ? null : attributes);
}
}

@ -0,0 +1,289 @@
import 'package:flutter/services.dart';
import 'package:flutter_quill/flutter_quill.dart';
import 'package:flutter_quill/quill_delta.dart';
import 'package:flutter_quill/src/rules/insert.dart';
import 'package:test/test.dart';
void main() {
group('PreserveInlineStylesRule', () {
const rule = PreserveInlineStylesRule();
test('Data does not apply', () {
final delta = Delta()
..insert('data\n')
..insert('second\n', <String, dynamic>{'bold': true})
..insert('\n\nplain\n');
final document = Document.fromDelta(delta);
//
expect(rule.apply(document, 0, data: 1), null);
expect(rule.apply(document, 0, data: '\n'), null);
});
test('Insert in text', () {
final delta = Delta()
..insert('data\n')
..insert('second\n', <String, dynamic>{'bold': true})
..insert('\n\nplain\n');
final document = Document.fromDelta(delta);
//
expect(rule.apply(document, 1, data: 'X', len: 0), null);
expect(
rule.apply(document, 6, data: 'X', len: 0),
Delta()
..retain(6)
..insert('X', <String, dynamic>{'bold': true}));
expect(rule.apply(document, 16, data: 'X', len: 0), null,
reason: 'insertion with no attributes');
});
test('Insert at start of line', () {
final delta = Delta()
..insert('data\n')
..insert('second\n', <String, dynamic>{'bold': true})
..insert('\n\nplain\n');
final document = Document.fromDelta(delta);
//
expect(rule.apply(document, 0, data: 'a', len: 0), null);
expect(
rule.apply(document, 5, data: 'X', len: 0),
Delta()
..retain(5)
..insert('X', <String, dynamic>{'bold': true}));
expect(
rule.apply(document, 12, data: 'X', len: 0),
Delta()
..retain(12)
..insert('X', <String, dynamic>{'bold': true}));
expect(
rule.apply(document, 13, data: 'X', len: 0),
Delta()
..retain(13)
..insert('X', <String, dynamic>{'bold': true}));
expect(rule.apply(document, 14, data: 'X', len: 0), null,
reason: 'insertion before "plain" has no attributes');
});
test('Insert on first line of document with bold text', () {
final delta = Delta()..insert('data\n', <String, dynamic>{'bold': true});
final document = Document.fromDelta(delta);
//
expect(rule.apply(document, 0, data: 'X', len: 0),
Delta()..insert('X', <String, dynamic>{'bold': true}),
reason: 'Insert at document start must pickup style for the line');
expect(
rule.apply(document, 1, data: 'X', len: 0),
Delta()
..retain(1)
..insert('X', <String, dynamic>{'bold': true}));
});
test('Insert around image', () {
final delta = Delta()
..insert(<String, String>{'image': 'url'})
..insert('data\n');
final document = Document.fromDelta(delta);
//
expect(rule.apply(document, 0, data: 'X', len: 0), null);
expect(rule.apply(document, 1, data: 'X', len: 0), null);
});
test('Insert around image with bold text', () {
final delta = Delta()
..insert(<String, String>{'image': 'url'})
..insert('data\n', <String, dynamic>{'bold': true});
final document = Document.fromDelta(delta);
//
expect(rule.apply(document, 0, data: 'X', len: 0), null,
reason:
'Insert before image must pickup inline attribute for the image');
expect(
rule.apply(document, 1, data: 'X', len: 0),
Delta()
..retain(1)
..insert('X', <String, dynamic>{'bold': true}),
reason:
'Insert after image must pickup style for text following the image');
});
test('Insert around image with inline attribute', () {
final delta = Delta()
..insert(
<String, String>{'image': 'url'}, <String, dynamic>{'bold': true})
..insert('data\n', <String, dynamic>{'bold': true});
final document = Document.fromDelta(delta);
//
expect(rule.apply(document, 0, data: 'X', len: 0),
Delta()..insert('X', <String, dynamic>{'bold': true}),
reason: 'Insert before image must pickup style for the image');
expect(
rule.apply(document, 1, data: 'X', len: 0),
Delta()
..retain(1)
..insert('X', <String, dynamic>{'bold': true}));
});
test('Replace in text', () {
final delta = Delta()
..insert('data\n')
..insert('second\n', <String, dynamic>{'bold': true})
..insert('\n\nplain\n');
final document = Document.fromDelta(delta);
//
expect(rule.apply(document, 0, data: 'X', len: 1), null);
expect(
rule.apply(document, 5, data: 'X', len: 1),
Delta()
..retain(6)
..insert('X', <String, dynamic>{'bold': true}));
});
test('Insert around multiple images', () {
final delta = Delta()
..insert(
<String, String>{'image': 'url'}, <String, dynamic>{'bold': true})
..insert(<String, String>{'image': 'url2'},
<String, dynamic>{'italic': true})
..insert('data\n', <String, dynamic>{'underline': true});
final document = Document.fromDelta(delta);
//
expect(rule.apply(document, 0, data: 'X', len: 0),
Delta()..insert('X', <String, dynamic>{'bold': true}));
expect(
rule.apply(document, 1, data: 'X', len: 0),
Delta()
..retain(1)
..insert('X', <String, dynamic>{'italic': true}));
expect(
rule.apply(document, 2, data: 'X', len: 0),
Delta()
..retain(2)
..insert('X', <String, dynamic>{'underline': true}));
});
test('Insert around mix of text and images', () {
final delta = Delta()
..insert(
<String, String>{'image': 'url'}, <String, dynamic>{'bold': true})
..insert('p\n')
..insert(<String, String>{'image': 'url2'},
<String, dynamic>{'italic': true})
..insert('data\n', <String, dynamic>{'underline': true});
final document = Document.fromDelta(delta);
//
expect(
rule.apply(document, 3, data: 'X', len: 0),
Delta()
..retain(3)
..insert('X', <String, dynamic>{'italic': true}));
});
test('Insert around images with NL', () {
final delta = Delta()
..insert('\n\n\n', <String, dynamic>{'strike': true})
..insert(
<String, String>{'image': 'url'}, <String, dynamic>{'bold': true})
..insert('\n\n\n', <String, dynamic>{'strike': true})
..insert(<String, String>{'image': 'url2'},
<String, dynamic>{'italic': true})
..insert('data\n', <String, dynamic>{'underline': true});
final document = Document.fromDelta(delta);
//
expect(
rule.apply(document, 2, data: 'X', len: 0),
Delta()
..retain(2)
..insert('X', <String, dynamic>{'strike': true}));
expect(
rule.apply(document, 6, data: 'X', len: 0),
Delta()
..retain(6)
..insert('X', <String, dynamic>{'strike': true}));
expect(
rule.apply(document, 7, data: 'X', len: 0),
Delta()
..retain(7)
..insert('X', <String, dynamic>{'italic': true}));
});
test('Exclude non-inline styles', () {
final delta = Delta()..insert('\n', <String, dynamic>{'list': 'ordered'});
final document = Document.fromDelta(delta);
expect(rule.apply(document, 0, data: 'X', len: 0), null);
});
test('Insert around non-inline styles', () {
final delta = Delta()
..insert('data\n')
..insert('first')
..insert('\n', <String, dynamic>{'list': 'ordered'})
..insert('A', <String, dynamic>{'bold': true})
..insert('B')
..insert('C', <String, dynamic>{'italic': true})
..insert('\n\n', <String, dynamic>{'list': 'ordered'})
..insert('D', <String, dynamic>{'strike': true})
..insert('\n', <String, dynamic>{'list': 'ordered'})
..insert(
<String, String>{'image': 'url'}, <String, dynamic>{'bold': true})
..insert('\n', <String, dynamic>{'list': 'ordered'})
..insert(' plain\n');
final document = Document.fromDelta(delta);
//
expect(rule.apply(document, 0, data: 'X', len: 0), null);
expect(rule.apply(document, 5, data: 'X', len: 0), null,
reason: '1. plain text');
expect(
rule.apply(document, 11, data: 'X', len: 0),
Delta()
..retain(11)
..insert('X', <String, dynamic>{'bold': true}),
reason: '2. bold text at start');
expect(
rule.apply(document, 15, data: 'X', len: 0),
Delta()
..retain(15)
..insert('X', <String, dynamic>{'italic': true}),
reason: '3. blank entry gets style from end of previous line');
expect(
rule.apply(document, 16, data: 'X', len: 0),
Delta()
..retain(16)
..insert('X', <String, dynamic>{'strike': true}),
reason: '4. strike text');
expect(
rule.apply(document, 18, data: 'X', len: 0),
Delta()
..retain(18)
..insert('X', <String, dynamic>{'bold': true}),
reason: '5. bold image');
expect(rule.apply(document, 20, data: 'X', len: 0), null);
//
expect(rule.apply(document, 16, data: LogicalKeyboardKey.enter, len: 0),
null);
});
test('Insert around link, insert within link label', () {
final delta = Delta()
..insert(<String, String>{'image': 'imageUrl'},
<String, dynamic>{'link': 'linkURL'})
..insert('data\n')
..insert('link', <String, dynamic>{'link': 'linkURL', 'bold': true})
..insert('\n');
final document = Document.fromDelta(delta);
//
expect(rule.apply(document, 0, data: 'X', len: 0), Delta()..insert('X'));
expect(rule.apply(document, 1, data: 'X', len: 0), null);
expect(
rule.apply(document, 6, data: 'X', len: 0),
Delta()
..retain(6)
..insert('X', <String, dynamic>{'bold': true}));
expect(
rule.apply(document, 7, data: 'X', len: 0),
Delta()
..retain(7)
..insert('X', <String, dynamic>{'link': 'linkURL', 'bold': true}),
reason: 'Insertion within link label updates label');
});
});
}
Loading…
Cancel
Save