2023-10-29 06:02:13 +00:00
|
|
|
//! All the tests in this file are based on [richtext.md]
|
|
|
|
|
|
|
|
use std::ops::Range;
|
|
|
|
|
2023-11-02 06:20:34 +00:00
|
|
|
use loro_common::LoroValue;
|
2023-10-29 06:02:13 +00:00
|
|
|
use loro_internal::{container::richtext::TextStyleInfoFlag, LoroDoc, ToJson};
|
2023-11-01 12:02:05 +00:00
|
|
|
use serde_json::json;
|
2023-10-29 06:02:13 +00:00
|
|
|
|
|
|
|
fn init(s: &str) -> LoroDoc {
|
|
|
|
let doc = LoroDoc::default();
|
2023-10-31 09:59:24 +00:00
|
|
|
doc.set_peer_id(1).unwrap();
|
2023-10-29 06:02:13 +00:00
|
|
|
let richtext = doc.get_text("r");
|
2023-11-28 08:22:43 +00:00
|
|
|
doc.with_txn(|txn| richtext.insert_with_txn(txn, 0, s))
|
|
|
|
.unwrap();
|
2023-10-29 06:02:13 +00:00
|
|
|
doc
|
|
|
|
}
|
|
|
|
|
|
|
|
fn clone(doc: &LoroDoc, peer_id: u64) -> LoroDoc {
|
|
|
|
let doc2 = LoroDoc::default();
|
2023-10-31 09:59:24 +00:00
|
|
|
doc2.set_peer_id(peer_id).unwrap();
|
2023-10-29 06:02:13 +00:00
|
|
|
doc2.import(&doc.export_from(&Default::default())).unwrap();
|
|
|
|
doc2
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Debug)]
|
|
|
|
enum Kind {
|
|
|
|
Bold,
|
|
|
|
Italic,
|
|
|
|
Link,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Kind {
|
|
|
|
fn key(&self) -> &str {
|
|
|
|
match self {
|
|
|
|
Kind::Bold => "bold",
|
|
|
|
Kind::Link => "link",
|
|
|
|
Kind::Italic => "italic",
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn flag(&self) -> TextStyleInfoFlag {
|
|
|
|
match self {
|
|
|
|
Kind::Bold => TextStyleInfoFlag::BOLD,
|
|
|
|
Kind::Link => TextStyleInfoFlag::LINK,
|
|
|
|
Kind::Italic => TextStyleInfoFlag::BOLD,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn insert(doc: &LoroDoc, pos: usize, s: &str) {
|
|
|
|
let richtext = doc.get_text("r");
|
2023-11-28 08:22:43 +00:00
|
|
|
doc.with_txn(|txn| richtext.insert_with_txn(txn, pos, s))
|
|
|
|
.unwrap();
|
2023-10-29 06:02:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
fn delete(doc: &LoroDoc, pos: usize, len: usize) {
|
|
|
|
let richtext = doc.get_text("r");
|
2023-11-28 08:22:43 +00:00
|
|
|
doc.with_txn(|txn| richtext.delete_with_txn(txn, pos, len))
|
|
|
|
.unwrap();
|
2023-10-29 06:02:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
fn mark(doc: &LoroDoc, range: Range<usize>, kind: Kind) {
|
|
|
|
let richtext = doc.get_text("r");
|
2023-11-02 06:20:34 +00:00
|
|
|
doc.with_txn(|txn| {
|
2023-11-28 08:22:43 +00:00
|
|
|
richtext.mark_with_txn(
|
2023-11-02 06:20:34 +00:00
|
|
|
txn,
|
|
|
|
range.start,
|
|
|
|
range.end,
|
|
|
|
kind.key(),
|
|
|
|
if kind.flag().is_delete() {
|
|
|
|
LoroValue::Null
|
|
|
|
} else {
|
|
|
|
true.into()
|
|
|
|
},
|
|
|
|
kind.flag(),
|
|
|
|
)
|
|
|
|
})
|
|
|
|
.unwrap();
|
2023-10-29 06:02:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
fn unmark(doc: &LoroDoc, range: Range<usize>, kind: Kind) {
|
|
|
|
let richtext = doc.get_text("r");
|
|
|
|
doc.with_txn(|txn| {
|
2023-11-28 08:22:43 +00:00
|
|
|
richtext.mark_with_txn(
|
2023-10-29 06:02:13 +00:00
|
|
|
txn,
|
|
|
|
range.start,
|
|
|
|
range.end,
|
|
|
|
kind.key(),
|
2023-11-02 06:20:34 +00:00
|
|
|
false.into(),
|
2023-10-29 06:02:13 +00:00
|
|
|
kind.flag().to_delete(),
|
|
|
|
)
|
|
|
|
})
|
|
|
|
.unwrap();
|
|
|
|
}
|
|
|
|
|
|
|
|
fn merge(a: &LoroDoc, b: &LoroDoc) {
|
|
|
|
a.import(&b.export_from(&a.oplog_vv())).unwrap();
|
|
|
|
b.import(&a.export_from(&b.oplog_vv())).unwrap();
|
|
|
|
}
|
|
|
|
|
2023-11-01 12:02:05 +00:00
|
|
|
fn expect_result(doc: &LoroDoc, json: serde_json::Value) {
|
2023-10-29 06:02:13 +00:00
|
|
|
let richtext = doc.get_text("r");
|
|
|
|
let s = richtext.get_richtext_value().to_json_value();
|
2023-11-01 12:02:05 +00:00
|
|
|
assert_eq!(
|
|
|
|
&s,
|
|
|
|
&json,
|
|
|
|
"expect: {}, got: {}",
|
|
|
|
serde_json::to_string_pretty(&json).unwrap(),
|
|
|
|
serde_json::to_string_pretty(&s).unwrap()
|
|
|
|
);
|
2023-10-29 06:02:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn case0() {
|
|
|
|
let doc_a = init("Hello World");
|
|
|
|
let doc_b = clone(&doc_a, 2);
|
|
|
|
mark(&doc_a, 0..11, Kind::Bold);
|
|
|
|
insert(&doc_b, 6, "New ");
|
|
|
|
merge(&doc_a, &doc_b);
|
|
|
|
expect_result(
|
|
|
|
&doc_a,
|
2023-11-01 12:02:05 +00:00
|
|
|
json!([{"insert":"Hello New World","attributes":{"bold":true}}]),
|
2023-10-29 06:02:13 +00:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn case1() {
|
|
|
|
let doc_a = init("Hello World");
|
|
|
|
let doc_b = clone(&doc_a, 2);
|
|
|
|
mark(&doc_a, 0..5, Kind::Bold);
|
|
|
|
mark(&doc_b, 3..11, Kind::Bold);
|
|
|
|
merge(&doc_a, &doc_b);
|
|
|
|
expect_result(
|
|
|
|
&doc_a,
|
2023-11-01 12:02:05 +00:00
|
|
|
json!([{"insert":"Hello World","attributes":{"bold":true}}]),
|
2023-10-29 06:02:13 +00:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn case2() {
|
|
|
|
let doc_a = init("Hello World");
|
|
|
|
mark(&doc_a, 0..11, Kind::Bold);
|
|
|
|
let doc_b = clone(&doc_a, 2);
|
|
|
|
unmark(&doc_a, 0..6, Kind::Bold);
|
|
|
|
insert(&doc_b, 5, " a");
|
|
|
|
merge(&doc_a, &doc_b);
|
|
|
|
expect_result(
|
|
|
|
&doc_a,
|
2023-11-01 12:02:05 +00:00
|
|
|
json!([{"insert":"Hello a ","attributes":{"bold":false}},{"insert":"World","attributes":{"bold":true}}]),
|
2023-10-29 06:02:13 +00:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
/// | Name | Text |
|
|
|
|
/// |:----------------|:-----------------------|
|
|
|
|
/// | Origin | `Hello World` |
|
|
|
|
/// | Concurrent A | `Hello <b>World</b>` |
|
|
|
|
/// | Concurrent B | `Hello a World` |
|
|
|
|
/// | Expected Result | `Hello a <b>World</b>` |
|
|
|
|
///
|
|
|
|
#[test]
|
|
|
|
fn case3() {
|
|
|
|
let doc_a = init("Hello World");
|
|
|
|
let doc_b = clone(&doc_a, 2);
|
|
|
|
mark(&doc_a, 6..11, Kind::Bold);
|
|
|
|
insert(&doc_b, 5, " a");
|
|
|
|
merge(&doc_a, &doc_b);
|
|
|
|
expect_result(
|
|
|
|
&doc_a,
|
2023-11-01 12:02:05 +00:00
|
|
|
json!([{"insert":"Hello a "},{"insert":"World","attributes":{"bold":true}}]),
|
2023-10-29 06:02:13 +00:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
/// | Name | Text |
|
|
|
|
/// |:----------------|:---------------------------|
|
|
|
|
/// | Origin | `Hello World` |
|
|
|
|
/// | Concurrent A | `<link>Hello</link> World` |
|
|
|
|
/// | Concurrent B | `Hey World` |
|
|
|
|
/// | Expected Result | `<link>Hey</link> World` |
|
|
|
|
#[test]
|
|
|
|
fn case5() {
|
|
|
|
let doc_a = init("Hello World");
|
|
|
|
let doc_b = clone(&doc_a, 2);
|
|
|
|
mark(&doc_a, 0..5, Kind::Link);
|
|
|
|
delete(&doc_b, 2, 3);
|
2023-11-01 12:02:05 +00:00
|
|
|
expect_result(&doc_b, json!([{"insert":"He World"}]));
|
2023-10-29 06:02:13 +00:00
|
|
|
insert(&doc_b, 2, "y");
|
2023-11-01 12:02:05 +00:00
|
|
|
expect_result(&doc_b, json!([{"insert":"Hey World"}]));
|
2023-10-29 06:02:13 +00:00
|
|
|
merge(&doc_a, &doc_b);
|
|
|
|
expect_result(
|
|
|
|
&doc_b,
|
2023-11-01 12:02:05 +00:00
|
|
|
json!([{"insert":"Hey","attributes":{"link":true}},{"insert":" World"}]),
|
2023-10-29 06:02:13 +00:00
|
|
|
);
|
|
|
|
expect_result(
|
|
|
|
&doc_a,
|
2023-11-01 12:02:05 +00:00
|
|
|
json!([{"insert":"Hey","attributes":{"link":true}},{"insert":" World"}]),
|
2023-10-29 06:02:13 +00:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
/// When insert a new character after "Hello", the new char should be bold but not link
|
|
|
|
///
|
|
|
|
/// | Name | Text |
|
|
|
|
/// |:----------------|:----------------------------------|
|
|
|
|
/// | Origin | `<b><link>Hello</link><b> World` |
|
|
|
|
/// | Expected Result | `<b><link>Hello</link>t<b> World` |
|
|
|
|
#[test]
|
|
|
|
fn case6() {
|
|
|
|
let doc = init("Hello World");
|
|
|
|
mark(&doc, 0..5, Kind::Bold);
|
2023-11-01 12:02:05 +00:00
|
|
|
expect_result(
|
2023-10-29 06:02:13 +00:00
|
|
|
&doc,
|
|
|
|
serde_json::json!([
|
|
|
|
{"insert": "Hello", "attributes": {"bold": true}},
|
|
|
|
{"insert": " World"}
|
|
|
|
]),
|
|
|
|
);
|
|
|
|
mark(&doc, 0..5, Kind::Link);
|
2023-11-01 12:02:05 +00:00
|
|
|
expect_result(
|
2023-10-29 06:02:13 +00:00
|
|
|
&doc,
|
|
|
|
serde_json::json!([
|
|
|
|
{"insert": "Hello", "attributes": {"bold": true, "link": true}},
|
|
|
|
{"insert": " World"}
|
|
|
|
]),
|
|
|
|
);
|
|
|
|
insert(&doc, 5, "t");
|
2023-11-01 12:02:05 +00:00
|
|
|
expect_result(
|
2023-10-29 06:02:13 +00:00
|
|
|
&doc,
|
|
|
|
serde_json::json!([
|
|
|
|
{"insert": "Hello", "attributes": {"bold": true, "link": true}},
|
|
|
|
{"insert": "t", "attributes": {"bold": true}},
|
|
|
|
{"insert": " World"}
|
|
|
|
]),
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
///
|
|
|
|
/// | Name | Text |
|
|
|
|
/// |:----------------|:---------------------------------------------|
|
|
|
|
/// | Origin | `<b>The fox jumped</b> over the dog.` |
|
|
|
|
/// | Concurrent A | `The fox jumped over the dog.` |
|
|
|
|
/// | Concurrent B | `<b>The </b>fox<b> jumped</b> over the dog.` |
|
|
|
|
/// | Expected Result | `The fox jumped over the dog.` |
|
|
|
|
#[test]
|
|
|
|
fn case7() {
|
|
|
|
let doc_a = init("The fox jumped over the dog.");
|
|
|
|
mark(&doc_a, 0..3, Kind::Bold);
|
|
|
|
let doc_b = clone(&doc_a, 2);
|
|
|
|
unmark(&doc_a, 0..3, Kind::Bold);
|
|
|
|
unmark(&doc_b, 4..7, Kind::Bold);
|
|
|
|
merge(&doc_a, &doc_b);
|
2023-11-01 12:02:05 +00:00
|
|
|
expect_result(
|
|
|
|
&doc_a,
|
|
|
|
json!([
|
|
|
|
{"insert":"The", "attributes": {"bold": false}},
|
|
|
|
{"insert":" ",},
|
|
|
|
{"insert":"fox", "attributes": {"bold": false}},
|
|
|
|
{"insert":" jumped over the dog."}
|
|
|
|
]),
|
|
|
|
);
|
2023-10-29 06:02:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/// | Name | Text |
|
|
|
|
/// |:----------------|:---------------------------------------------|
|
|
|
|
/// | Origin | `<b>The fox jumped</b> over the dog.` |
|
|
|
|
/// | Concurrent A | `<b>The fox</b> jumped over the dog.` |
|
|
|
|
/// | Concurrent B | `<b>The</b> fox jumped over the <b>dog</b>.` |
|
|
|
|
/// | Expected Result | `<b>The</b> fox jumped over the <b>dog</b>.` |
|
|
|
|
#[test]
|
|
|
|
fn case8() {
|
|
|
|
let doc_a = init("The fox jumped over the dog.");
|
|
|
|
mark(&doc_a, 0..14, Kind::Bold);
|
|
|
|
let doc_b = clone(&doc_a, 2);
|
|
|
|
unmark(&doc_a, 7..14, Kind::Bold);
|
|
|
|
unmark(&doc_b, 3..14, Kind::Bold);
|
|
|
|
mark(&doc_b, 24..27, Kind::Bold);
|
|
|
|
merge(&doc_a, &doc_b);
|
2023-11-01 12:02:05 +00:00
|
|
|
expect_result(
|
2023-10-29 06:02:13 +00:00
|
|
|
&doc_a,
|
|
|
|
serde_json::json!([
|
|
|
|
{"insert": "The", "attributes": {"bold": true}},
|
2023-11-01 12:02:05 +00:00
|
|
|
{"insert": " fox jumped", "attributes": {"bold": false}},
|
|
|
|
{"insert": " over the "},
|
2023-10-29 06:02:13 +00:00
|
|
|
{"insert": "dog", "attributes": {"bold": true}},
|
|
|
|
{"insert": "."}
|
|
|
|
]),
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
/// | Name | Text |
|
|
|
|
/// |:----------------|:-----------------------------|
|
|
|
|
/// | Origin | The fox jumped. |
|
|
|
|
/// | Concurrent A | **The fox** jumped. |
|
|
|
|
/// | Concurrent B | The *fox jumped*. |
|
|
|
|
/// | Expected Result | **The _fox_**<i> jumped</i>. |
|
|
|
|
#[test]
|
|
|
|
fn case9() {
|
|
|
|
let doc_a = init("The fox jumped.");
|
|
|
|
let doc_b = clone(&doc_a, 2);
|
|
|
|
mark(&doc_a, 0..7, Kind::Bold);
|
|
|
|
mark(&doc_a, 4..14, Kind::Italic);
|
|
|
|
merge(&doc_a, &doc_b);
|
2023-11-01 12:02:05 +00:00
|
|
|
expect_result(
|
2023-10-29 06:02:13 +00:00
|
|
|
&doc_a,
|
|
|
|
serde_json::json!([
|
|
|
|
{"insert": "The ", "attributes": {"bold": true}},
|
|
|
|
{"insert": "fox", "attributes": {"bold": true, "italic": true}},
|
|
|
|
{"insert": " jumped", "attributes": {"italic": true}},
|
|
|
|
{"insert": "."}
|
|
|
|
]),
|
|
|
|
);
|
|
|
|
}
|
2023-11-04 12:03:15 +00:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn insert_after_link() {
|
|
|
|
let doc_a = init("The fox jumped.");
|
|
|
|
let doc_b = clone(&doc_a, 2);
|
|
|
|
mark(&doc_a, 0..3, Kind::Link);
|
|
|
|
merge(&doc_a, &doc_b);
|
|
|
|
insert(&doc_a, 3, "a");
|
|
|
|
merge(&doc_a, &doc_b);
|
|
|
|
expect_result(
|
|
|
|
&doc_a,
|
|
|
|
serde_json::json!([
|
|
|
|
{"insert": "The", "attributes": {"link": true}},
|
|
|
|
{"insert": "a fox jumped."},
|
|
|
|
]),
|
|
|
|
);
|
|
|
|
expect_result(
|
|
|
|
&doc_b,
|
|
|
|
serde_json::json!([
|
|
|
|
{"insert": "The", "attributes": {"link": true}},
|
|
|
|
{"insert": "a fox jumped."},
|
|
|
|
]),
|
|
|
|
);
|
|
|
|
}
|