|
|
@ -4,6 +4,7 @@ import 'package:html2md/html2md.dart' as html2md; |
|
|
|
import 'package:markdown/markdown.dart' as md; |
|
|
|
import 'package:markdown/markdown.dart' as md; |
|
|
|
|
|
|
|
|
|
|
|
import '../../../markdown_quill.dart'; |
|
|
|
import '../../../markdown_quill.dart'; |
|
|
|
|
|
|
|
|
|
|
|
import '../../../quill_delta.dart'; |
|
|
|
import '../../../quill_delta.dart'; |
|
|
|
import '../../widgets/quill/embeds.dart'; |
|
|
|
import '../../widgets/quill/embeds.dart'; |
|
|
|
import '../rules/rule.dart'; |
|
|
|
import '../rules/rule.dart'; |
|
|
@ -57,7 +58,8 @@ class Document { |
|
|
|
_rules.setCustomRules(customRules); |
|
|
|
_rules.setCustomRules(customRules); |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
final StreamController<DocChange> documentChangeObserver = StreamController.broadcast(); |
|
|
|
final StreamController<DocChange> documentChangeObserver = |
|
|
|
|
|
|
|
StreamController.broadcast(); |
|
|
|
|
|
|
|
|
|
|
|
final History history = History(); |
|
|
|
final History history = History(); |
|
|
|
|
|
|
|
|
|
|
@ -82,7 +84,8 @@ class Document { |
|
|
|
return Delta(); |
|
|
|
return Delta(); |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
final delta = _rules.apply(RuleType.insert, this, index, data: data, len: replaceLength); |
|
|
|
final delta = _rules.apply(RuleType.insert, this, index, |
|
|
|
|
|
|
|
data: data, len: replaceLength); |
|
|
|
compose(delta, ChangeSource.local); |
|
|
|
compose(delta, ChangeSource.local); |
|
|
|
return delta; |
|
|
|
return delta; |
|
|
|
} |
|
|
|
} |
|
|
@ -145,7 +148,8 @@ class Document { |
|
|
|
|
|
|
|
|
|
|
|
var delta = Delta(); |
|
|
|
var delta = Delta(); |
|
|
|
|
|
|
|
|
|
|
|
final formatDelta = _rules.apply(RuleType.format, this, index, len: len, attribute: attribute); |
|
|
|
final formatDelta = _rules.apply(RuleType.format, this, index, |
|
|
|
|
|
|
|
len: len, attribute: attribute); |
|
|
|
if (formatDelta.isNotEmpty) { |
|
|
|
if (formatDelta.isNotEmpty) { |
|
|
|
compose(formatDelta, ChangeSource.local); |
|
|
|
compose(formatDelta, ChangeSource.local); |
|
|
|
delta = delta.compose(formatDelta); |
|
|
|
delta = delta.compose(formatDelta); |
|
|
@ -185,7 +189,8 @@ class Document { |
|
|
|
/// Returns all styles and Embed for each node within selection |
|
|
|
/// Returns all styles and Embed for each node within selection |
|
|
|
List<OffsetValue> collectAllIndividualStyleAndEmbed(int index, int len) { |
|
|
|
List<OffsetValue> collectAllIndividualStyleAndEmbed(int index, int len) { |
|
|
|
final res = queryChild(index); |
|
|
|
final res = queryChild(index); |
|
|
|
return (res.node as Line).collectAllIndividualStylesAndEmbed(res.offset, len); |
|
|
|
return (res.node as Line) |
|
|
|
|
|
|
|
.collectAllIndividualStylesAndEmbed(res.offset, len); |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
/// Returns all styles for any character within the specified text range. |
|
|
|
/// Returns all styles for any character within the specified text range. |
|
|
@ -294,7 +299,8 @@ class Document { |
|
|
|
delta = _transform(delta); |
|
|
|
delta = _transform(delta); |
|
|
|
final originalDelta = toDelta(); |
|
|
|
final originalDelta = toDelta(); |
|
|
|
for (final op in delta.toList()) { |
|
|
|
for (final op in delta.toList()) { |
|
|
|
final style = op.attributes != null ? Style.fromJson(op.attributes) : null; |
|
|
|
final style = |
|
|
|
|
|
|
|
op.attributes != null ? Style.fromJson(op.attributes) : null; |
|
|
|
|
|
|
|
|
|
|
|
if (op.isInsert) { |
|
|
|
if (op.isInsert) { |
|
|
|
// Must normalize data before inserting into the document, makes sure |
|
|
|
// Must normalize data before inserting into the document, makes sure |
|
|
@ -360,7 +366,8 @@ class Document { |
|
|
|
res.push(Operation.insert('\n')); |
|
|
|
res.push(Operation.insert('\n')); |
|
|
|
} |
|
|
|
} |
|
|
|
// embed could be image or video |
|
|
|
// embed could be image or video |
|
|
|
final opInsertEmbed = op.isInsert && op.data is Map && (op.data as Map).containsKey(type); |
|
|
|
final opInsertEmbed = |
|
|
|
|
|
|
|
op.isInsert && op.data is Map && (op.data as Map).containsKey(type); |
|
|
|
final nextOpIsLineBreak = i + 1 < ops.length && |
|
|
|
final nextOpIsLineBreak = i + 1 < ops.length && |
|
|
|
ops[i + 1].isInsert && |
|
|
|
ops[i + 1].isInsert && |
|
|
|
ops[i + 1].data is String && |
|
|
|
ops[i + 1].data is String && |
|
|
@ -392,7 +399,9 @@ class Document { |
|
|
|
Iterable<EmbedBuilder>? embedBuilders, |
|
|
|
Iterable<EmbedBuilder>? embedBuilders, |
|
|
|
EmbedBuilder? unknownEmbedBuilder, |
|
|
|
EmbedBuilder? unknownEmbedBuilder, |
|
|
|
]) => |
|
|
|
]) => |
|
|
|
_root.children.map((e) => e.toPlainText(embedBuilders, unknownEmbedBuilder)).join(); |
|
|
|
_root.children |
|
|
|
|
|
|
|
.map((e) => e.toPlainText(embedBuilders, unknownEmbedBuilder)) |
|
|
|
|
|
|
|
.join(); |
|
|
|
|
|
|
|
|
|
|
|
void _loadDocument(Delta doc) { |
|
|
|
void _loadDocument(Delta doc) { |
|
|
|
if (doc.isEmpty) { |
|
|
|
if (doc.isEmpty) { |
|
|
@ -405,16 +414,20 @@ class Document { |
|
|
|
var offset = 0; |
|
|
|
var offset = 0; |
|
|
|
for (final op in doc.toList()) { |
|
|
|
for (final op in doc.toList()) { |
|
|
|
if (!op.isInsert) { |
|
|
|
if (!op.isInsert) { |
|
|
|
throw ArgumentError.value( |
|
|
|
throw ArgumentError.value(doc, |
|
|
|
doc, 'Document can only contain insert operations but ${op.key} found.'); |
|
|
|
'Document can only contain insert operations but ${op.key} found.'); |
|
|
|
} |
|
|
|
} |
|
|
|
final style = op.attributes != null ? Style.fromJson(op.attributes) : null; |
|
|
|
final style = |
|
|
|
|
|
|
|
op.attributes != null ? Style.fromJson(op.attributes) : null; |
|
|
|
final data = _normalize(op.data); |
|
|
|
final data = _normalize(op.data); |
|
|
|
_root.insert(offset, data, style); |
|
|
|
_root.insert(offset, data, style); |
|
|
|
offset += op.length!; |
|
|
|
offset += op.length!; |
|
|
|
} |
|
|
|
} |
|
|
|
final node = _root.last; |
|
|
|
final node = _root.last; |
|
|
|
if (node is Line && node.parent is! Block && node.style.isEmpty && _root.childCount > 1) { |
|
|
|
if (node is Line && |
|
|
|
|
|
|
|
node.parent is! Block && |
|
|
|
|
|
|
|
node.style.isEmpty && |
|
|
|
|
|
|
|
_root.childCount > 1) { |
|
|
|
_root.remove(node); |
|
|
|
_root.remove(node); |
|
|
|
} |
|
|
|
} |
|
|
|
} |
|
|
|
} |
|
|
@ -430,11 +443,11 @@ class Document { |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
final delta = node.toDelta(); |
|
|
|
final delta = node.toDelta(); |
|
|
|
return delta.length == 1 && delta.first.data == '\n' && delta.first.key == 'insert'; |
|
|
|
return delta.length == 1 && |
|
|
|
|
|
|
|
delta.first.data == '\n' && |
|
|
|
|
|
|
|
delta.first.key == 'insert'; |
|
|
|
} |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class DeltaX { |
|
|
|
|
|
|
|
/// Convert the HTML Raw string to [Delta] |
|
|
|
/// Convert the HTML Raw string to [Delta] |
|
|
|
/// |
|
|
|
/// |
|
|
|
/// It will run using the following steps: |
|
|
|
/// It will run using the following steps: |
|
|
@ -445,26 +458,28 @@ class DeltaX { |
|
|
|
/// |
|
|
|
/// |
|
|
|
/// for more [info](https://github.com/singerdmx/flutter-quill/issues/1100) |
|
|
|
/// for more [info](https://github.com/singerdmx/flutter-quill/issues/1100) |
|
|
|
static Delta fromHtml(String html) { |
|
|
|
static Delta fromHtml(String html) { |
|
|
|
var rules = [ |
|
|
|
final markdown = html2md |
|
|
|
html2md.Rule('image', filters: ['img'], replacement: (content, node) { |
|
|
|
.convert( |
|
|
|
node.asElement()?.attributes.remove('class'); |
|
|
|
html, |
|
|
|
//Later we can convert this to delta along with the attributes by GoodInlineHtmlSyntax |
|
|
|
) |
|
|
|
return node.outerHTML; |
|
|
|
.replaceAll('unsafe:', ''); |
|
|
|
}), |
|
|
|
|
|
|
|
]; |
|
|
|
final mdDocument = md.Document(encodeHtml: false); |
|
|
|
|
|
|
|
|
|
|
|
final markdown = html2md.convert(html, rules: rules).replaceAll('unsafe:', ''); |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
final mdDocument = md.Document( |
|
|
|
|
|
|
|
encodeHtml: false, |
|
|
|
|
|
|
|
inlineSyntaxes: [ |
|
|
|
|
|
|
|
GoodInlineHtmlSyntax(), |
|
|
|
|
|
|
|
], |
|
|
|
|
|
|
|
); |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
final mdToDelta = MarkdownToDelta(markdownDocument: mdDocument); |
|
|
|
final mdToDelta = MarkdownToDelta(markdownDocument: mdDocument); |
|
|
|
|
|
|
|
|
|
|
|
return mdToDelta.convert(markdown); |
|
|
|
return mdToDelta.convert(markdown); |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// final deltaJsonString = markdownToDelta(markdown); |
|
|
|
|
|
|
|
// final deltaJson = jsonDecode(deltaJsonString); |
|
|
|
|
|
|
|
// if (deltaJson is! List) { |
|
|
|
|
|
|
|
// throw ArgumentError( |
|
|
|
|
|
|
|
// 'The delta json string should be of type list when jsonDecode() it', |
|
|
|
|
|
|
|
// ); |
|
|
|
|
|
|
|
// } |
|
|
|
|
|
|
|
// return Delta.fromJson( |
|
|
|
|
|
|
|
// deltaJson, |
|
|
|
|
|
|
|
// ); |
|
|
|
} |
|
|
|
} |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
@ -479,24 +494,3 @@ enum ChangeSource { |
|
|
|
/// Silent change. |
|
|
|
/// Silent change. |
|
|
|
silent; |
|
|
|
silent; |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
/// Convert the html to Element, not Text |
|
|
|
|
|
|
|
class GoodInlineHtmlSyntax extends md.InlineHtmlSyntax { |
|
|
|
|
|
|
|
@override |
|
|
|
|
|
|
|
onMatch(parser, match) { |
|
|
|
|
|
|
|
if (super.onMatch(parser, match)) { |
|
|
|
|
|
|
|
return true; |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
var root = html2md.Node.root(match.group(0)!); |
|
|
|
|
|
|
|
root = root.childNodes().last.firstChild!; |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
var node = md.Element.empty(root.nodeName); |
|
|
|
|
|
|
|
var attrs = root.asElement()?.attributes.map((key, value) => MapEntry(key.toString(), value)); |
|
|
|
|
|
|
|
if (attrs != null) node.attributes.addAll(attrs); |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
parser.addNode(node); |
|
|
|
|
|
|
|
parser.start = parser.pos; |
|
|
|
|
|
|
|
return false; |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|