1
1
use std:: ops:: Range ;
2
2
3
+ use append_only_bytes:: BytesSlice ;
3
4
use enum_as_inner:: EnumAsInner ;
4
5
use rle:: { HasLength , Mergable , Sliceable } ;
5
6
use serde:: { Deserialize , Serialize } ;
6
7
7
8
use crate :: {
8
9
container:: richtext:: TextStyleInfoFlag ,
9
10
op:: { ListSlice , SliceRange } ,
11
+ utils:: string_slice:: unicode_range_to_byte_range,
10
12
InternalString ,
11
13
} ;
12
14
@@ -37,6 +39,12 @@ pub enum InnerListOp {
37
39
slice : SliceRange ,
38
40
pos : usize ,
39
41
} ,
42
+ InsertText {
43
+ slice : BytesSlice ,
44
+ unicode_start : u32 ,
45
+ unicode_len : u32 ,
46
+ pos : u32 ,
47
+ } ,
40
48
Delete ( DeleteSpan ) ,
41
49
/// StyleStart and StyleEnd must be paired.
42
50
StyleStart {
@@ -297,45 +305,76 @@ impl<'a> Sliceable for ListOp<'a> {
297
305
}
298
306
299
307
impl Mergable for InnerListOp {
300
- fn is_mergable ( & self , _other : & Self , _conf : & ( ) ) -> bool
308
+ fn is_mergable ( & self , other : & Self , _conf : & ( ) ) -> bool
301
309
where
302
310
Self : Sized ,
303
311
{
304
- match self {
305
- InnerListOp :: Insert { pos, slice, .. } => match _other {
312
+ match ( self , other) {
313
+ (
314
+ InnerListOp :: Insert { pos, slice, .. } ,
306
315
InnerListOp :: Insert {
307
316
pos : other_pos,
308
317
slice : other_slice,
309
318
..
310
- } => pos + slice. content_len ( ) == * other_pos && slice. is_mergable ( other_slice, & ( ) ) ,
311
- _ => false ,
312
- } ,
313
- & InnerListOp :: Delete ( span) => match _other {
314
- InnerListOp :: Delete ( other_span) => span. is_mergable ( other_span, & ( ) ) ,
315
- _ => false ,
316
- } ,
317
- InnerListOp :: StyleStart { .. } | InnerListOp :: StyleEnd { .. } => false ,
319
+ } ,
320
+ ) => pos + slice. content_len ( ) == * other_pos && slice. is_mergable ( other_slice, & ( ) ) ,
321
+ ( InnerListOp :: Delete ( span) , InnerListOp :: Delete ( other_span) ) => {
322
+ span. is_mergable ( other_span, & ( ) )
323
+ }
324
+ (
325
+ InnerListOp :: InsertText {
326
+ unicode_start,
327
+ slice,
328
+ pos,
329
+ unicode_len : len,
330
+ } ,
331
+ InnerListOp :: InsertText {
332
+ slice : other_slice,
333
+ pos : other_pos,
334
+ unicode_start : other_unicode_start,
335
+ unicode_len : _,
336
+ } ,
337
+ ) => {
338
+ pos + len == * other_pos
339
+ && slice. can_merge ( other_slice)
340
+ && unicode_start + len == * other_unicode_start
341
+ }
342
+ _ => false ,
318
343
}
319
344
}
320
345
321
- fn merge ( & mut self , _other : & Self , _conf : & ( ) )
346
+ fn merge ( & mut self , other : & Self , _conf : & ( ) )
322
347
where
323
348
Self : Sized ,
324
349
{
325
- match self {
326
- InnerListOp :: Insert { slice, .. } => match _other {
350
+ match ( self , other) {
351
+ (
352
+ InnerListOp :: Insert { slice, .. } ,
327
353
InnerListOp :: Insert {
328
354
slice : other_slice, ..
329
- } => {
330
- slice. merge ( other_slice, & ( ) ) ;
331
- }
332
- _ => unreachable ! ( ) ,
333
- } ,
334
- InnerListOp :: Delete ( span) => match _other {
335
- InnerListOp :: Delete ( other_span) => span. merge ( other_span, & ( ) ) ,
336
- _ => unreachable ! ( ) ,
337
- } ,
338
- InnerListOp :: StyleStart { .. } | InnerListOp :: StyleEnd { .. } => unreachable ! ( ) ,
355
+ } ,
356
+ ) => {
357
+ slice. merge ( other_slice, & ( ) ) ;
358
+ }
359
+ ( InnerListOp :: Delete ( span) , InnerListOp :: Delete ( other_span) ) => {
360
+ span. merge ( other_span, & ( ) )
361
+ }
362
+ (
363
+ InnerListOp :: InsertText {
364
+ slice,
365
+ unicode_len : len,
366
+ ..
367
+ } ,
368
+ InnerListOp :: InsertText {
369
+ slice : other_slice,
370
+ unicode_len : other_len,
371
+ ..
372
+ } ,
373
+ ) => {
374
+ slice. merge ( other_slice, & ( ) ) ;
375
+ * len += * other_len;
376
+ }
377
+ _ => unreachable ! ( ) ,
339
378
}
340
379
}
341
380
}
@@ -344,6 +383,9 @@ impl HasLength for InnerListOp {
344
383
fn content_len ( & self ) -> usize {
345
384
match self {
346
385
InnerListOp :: Insert { slice, .. } => slice. content_len ( ) ,
386
+ InnerListOp :: InsertText {
387
+ unicode_len : len, ..
388
+ } => * len as usize ,
347
389
InnerListOp :: Delete ( span) => span. atom_len ( ) ,
348
390
InnerListOp :: StyleStart { .. } | InnerListOp :: StyleEnd { .. } => 1 ,
349
391
}
@@ -357,6 +399,20 @@ impl Sliceable for InnerListOp {
357
399
slice : slice. slice ( from, to) ,
358
400
pos : * pos + from,
359
401
} ,
402
+ InnerListOp :: InsertText {
403
+ slice,
404
+ unicode_start,
405
+ unicode_len : _,
406
+ pos,
407
+ } => InnerListOp :: InsertText {
408
+ slice : {
409
+ let ( a, b) = unicode_range_to_byte_range ( slice, from, to) ;
410
+ slice. slice ( a, b)
411
+ } ,
412
+ unicode_start : * unicode_start + from as u32 ,
413
+ unicode_len : ( to - from) as u32 ,
414
+ pos : * pos + from as u32 ,
415
+ } ,
360
416
InnerListOp :: Delete ( span) => InnerListOp :: Delete ( span. slice ( from, to) ) ,
361
417
InnerListOp :: StyleStart { .. } | InnerListOp :: StyleEnd { .. } => self . clone ( ) ,
362
418
}
0 commit comments