1use std::{collections::VecDeque, sync::Arc};
2
3use fractional_index::FractionalIndex;
4use fxhash::FxHashMap;
5use loro_common::{
6 ContainerID, ContainerType, Counter, IdLp, LoroError, LoroResult, LoroTreeError, LoroValue,
7 PeerID, TreeID, ID,
8};
9use smallvec::smallvec;
10
11use crate::{
12 container::tree::tree_op::TreeOp,
13 delta::{TreeDiffItem, TreeExternalDiff},
14 state::{
15 FiIfNotConfigured, FractionalIndexGenResult, NodePosition, TreeNode, TreeNodeWithChildren,
16 TreeParentId,
17 },
18 txn::{EventHint, Transaction},
19 BasicHandler, HandlerTrait, MapHandler,
20};
21
22use super::{create_handler, Handler, MaybeDetached};
23
24#[derive(Clone)]
25pub struct TreeHandler {
26 pub(super) inner: MaybeDetached<TreeInner>,
27}
28
29#[derive(Clone)]
30pub(super) struct TreeInner {
31 next_counter: Counter,
32 map: FxHashMap<TreeID, MapHandler>,
33 parent_links: FxHashMap<TreeID, Option<TreeID>>,
34 children_links: FxHashMap<Option<TreeID>, Vec<TreeID>>,
35}
36
37impl TreeInner {
38 fn new() -> Self {
39 TreeInner {
40 next_counter: 0,
41 map: FxHashMap::default(),
42 parent_links: FxHashMap::default(),
43 children_links: FxHashMap::default(),
44 }
45 }
46
47 fn create(&mut self, parent: Option<TreeID>, index: usize) -> TreeID {
48 let id = TreeID::new(PeerID::MAX, self.next_counter);
49 self.next_counter += 1;
50 self.map.insert(id, MapHandler::new_detached());
51 self.parent_links.insert(id, parent);
52 let children = self.children_links.entry(parent).or_default();
53 children.insert(index, id);
54 id
55 }
56
57 fn mov(&mut self, target: TreeID, new_parent: Option<TreeID>, index: usize) -> LoroResult<()> {
58 let old_parent = self
59 .parent_links
60 .get_mut(&target)
61 .ok_or(LoroTreeError::TreeNodeNotExist(target))?;
62 let children = self.children_links.get_mut(old_parent).unwrap();
63 children.retain(|x| x != &target);
64 self.parent_links.insert(target, new_parent);
65 let children = self.children_links.entry(new_parent).or_default();
66 children.insert(index, target);
67 Ok(())
68 }
69
70 fn delete(&mut self, id: TreeID) -> LoroResult<()> {
71 self.map.remove(&id);
72 let parent = self
73 .parent_links
74 .remove(&id)
75 .ok_or(LoroTreeError::TreeNodeNotExist(id))?;
76 let children = self.children_links.get_mut(&parent).unwrap();
77 children.retain(|x| x != &id);
78 self.children_links.remove(&Some(id));
79 Ok(())
80 }
81
82 fn get_id_by_index(&self, parent: &Option<TreeID>, index: usize) -> Option<TreeID> {
83 self.children_links
84 .get(parent)
85 .and_then(|x| x.get(index).cloned())
86 }
87
88 fn get_parent(&self, id: &TreeID) -> Option<Option<TreeID>> {
89 self.parent_links.get(id).cloned()
90 }
91
92 fn get_children(&self, parent: Option<TreeID>) -> Option<Vec<TreeID>> {
93 self.children_links.get(&parent).cloned()
94 }
95
96 fn children_num(&self, parent: Option<TreeID>) -> Option<usize> {
97 self.children_links.get(&parent).map(|x| x.len())
98 }
99
100 fn is_parent(&self, target: &TreeID, parent: &Option<TreeID>) -> bool {
101 self.parent_links.get(target) == Some(parent)
102 }
103
104 fn get_index_by_tree_id(&self, target: &TreeID) -> Option<usize> {
105 self.parent_links
106 .get(target)
107 .and_then(|parent| self.children_links.get(parent))
108 .and_then(|children| children.iter().position(|x| x == target))
109 }
110
111 fn get_value(&self, deep: bool) -> LoroValue {
112 let mut ans = vec![];
113
114 let empty_vec = vec![];
115 let mut q = VecDeque::from_iter(
116 self.children_links
117 .get(&None)
118 .unwrap_or(&empty_vec)
119 .iter()
120 .enumerate()
121 .zip(std::iter::repeat(None::<TreeID>)),
122 );
123
124 while let Some(((idx, target), parent)) = q.pop_front() {
125 let map = self.map.get(target).unwrap();
126 let mut loro_map_value = FxHashMap::default();
127 loro_map_value.insert("id".to_string(), target.to_string().into());
128 let parent = parent
129 .map(|p| p.to_string().into())
130 .unwrap_or(LoroValue::Null);
131 loro_map_value.insert("parent".to_string(), parent);
132 loro_map_value.insert(
133 "meta".to_string(),
134 if deep {
135 map.get_deep_value()
136 } else {
137 String::from("UnResolved").into()
138 },
139 );
140 loro_map_value.insert("index".to_string(), (idx as i64).into());
141 ans.push(loro_map_value);
142 if let Some(children) = self.children_links.get(&Some(*target)) {
143 for (idx, child) in children.iter().enumerate() {
144 q.push_back(((idx, child), Some(*target)));
145 }
146 }
147 }
148 ans.into()
149 }
150}
151
152impl HandlerTrait for TreeHandler {
153 fn to_handler(&self) -> Handler {
154 Handler::Tree(self.clone())
155 }
156
157 fn attach(
158 &self,
159 txn: &mut Transaction,
160 parent: &BasicHandler,
161 self_id: ContainerID,
162 ) -> LoroResult<Self> {
163 match &self.inner {
164 MaybeDetached::Detached(t) => {
165 let t = t.lock().unwrap();
166 let inner = create_handler(parent, self_id);
167 let tree = inner.into_tree().unwrap();
168
169 let children = t.value.children_links.get(&None);
170 let mut q = children
171 .map(|c| {
172 VecDeque::from_iter(
173 c.iter()
174 .enumerate()
175 .zip(std::iter::repeat(TreeParentId::Root)),
176 )
177 })
178 .unwrap_or_default();
179 while let Some(((idx, target), parent)) = q.pop_front() {
180 let real_id =
181 tree.create_with_txn(txn, parent, idx, FiIfNotConfigured::UseJitterZero)?;
182 let map = t.value.map.get(target).unwrap();
183 map.attach(
184 txn,
185 tree.inner.try_attached_state()?,
186 real_id.associated_meta_container(),
187 )?;
188
189 if let Some(children) = t.value.children_links.get(&Some(*target)) {
190 for (idx, child) in children.iter().enumerate() {
191 q.push_back(((idx, child), TreeParentId::Node(real_id)));
192 }
193 }
194 }
195 Ok(tree)
196 }
197 MaybeDetached::Attached(a) => {
198 let new_inner = create_handler(a, self_id);
199 let ans = new_inner.into_tree().unwrap();
200 let tree_nodes = ans.with_state(|s| Ok(s.as_tree_state().unwrap().tree_nodes()))?;
201 for node in tree_nodes {
202 let parent = node.parent;
203 let index = node.index;
204 let target = node.id;
205 let real_id =
206 ans.create_with_txn(txn, parent, index, FiIfNotConfigured::UseJitterZero)?;
207 ans.get_meta(target)?
208 .attach(txn, a, real_id.associated_meta_container())?;
209 }
210 Ok(ans)
211 }
212 }
213 }
214
215 fn is_attached(&self) -> bool {
216 self.inner.is_attached()
217 }
218
219 fn attached_handler(&self) -> Option<&BasicHandler> {
220 self.inner.attached_handler()
221 }
222
223 fn get_value(&self) -> LoroValue {
224 match &self.inner {
225 MaybeDetached::Detached(t) => {
226 let t = t.lock().unwrap();
227 t.value.get_value(false)
228 }
229 MaybeDetached::Attached(a) => a.get_value(),
230 }
231 }
232
233 fn get_deep_value(&self) -> LoroValue {
234 match &self.inner {
235 MaybeDetached::Detached(t) => {
236 let t = t.lock().unwrap();
237 t.value.get_value(true)
238 }
239 MaybeDetached::Attached(a) => a.get_deep_value(),
240 }
241 }
242
243 fn kind(&self) -> ContainerType {
244 ContainerType::Tree
245 }
246
247 fn get_attached(&self) -> Option<Self> {
248 match &self.inner {
249 MaybeDetached::Detached(d) => d.lock().unwrap().attached.clone().map(|x| Self {
250 inner: MaybeDetached::Attached(x),
251 }),
252 MaybeDetached::Attached(_a) => Some(self.clone()),
253 }
254 }
255
256 fn from_handler(h: Handler) -> Option<Self> {
257 match h {
258 Handler::Tree(x) => Some(x),
259 _ => None,
260 }
261 }
262
263 fn doc(&self) -> Option<crate::LoroDoc> {
264 match &self.inner {
265 MaybeDetached::Detached(_) => None,
266 MaybeDetached::Attached(a) => Some(a.doc()),
267 }
268 }
269}
270
271impl std::fmt::Debug for TreeHandler {
272 fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
273 match &self.inner {
274 MaybeDetached::Detached(_) => write!(f, "TreeHandler Detached"),
275 MaybeDetached::Attached(a) => write!(f, "TreeHandler {}", a.id),
276 }
277 }
278}
279
280impl TreeHandler {
281 pub fn new_detached() -> Self {
287 Self {
288 inner: MaybeDetached::new_detached(TreeInner::new()),
289 }
290 }
291
292 pub fn delete(&self, target: TreeID) -> LoroResult<()> {
293 match &self.inner {
294 MaybeDetached::Detached(t) => {
295 let mut t = t.lock().unwrap();
296 t.value.delete(target)?;
297 Ok(())
298 }
299 MaybeDetached::Attached(a) => a.with_txn(|txn| self.delete_with_txn(txn, target)),
300 }
301 }
302
303 pub(crate) fn delete_with_txn(&self, txn: &mut Transaction, target: TreeID) -> LoroResult<()> {
304 let inner = self.inner.try_attached_state()?;
305 let index = match self.get_index_by_tree_id(&target) {
306 Some(i) => i,
307 None => {
308 return Err(LoroTreeError::TreeNodeDeletedOrNotExist(target).into());
309 }
310 };
311 txn.apply_local_op(
312 inner.container_idx,
313 crate::op::RawOpContent::Tree(Arc::new(TreeOp::Delete { target })),
314 EventHint::Tree(smallvec![TreeDiffItem {
315 target,
316 action: TreeExternalDiff::Delete {
317 old_parent: self.get_node_parent(&target).unwrap(),
318 old_index: index
319 },
320 }]),
321 &inner.doc,
322 )
323 }
324
325 pub fn create(&self, parent: TreeParentId) -> LoroResult<TreeID> {
326 match parent {
327 TreeParentId::Deleted | TreeParentId::Unexist => {
328 return Err(LoroTreeError::InvalidParent.into());
329 }
330 _ => {}
331 }
332 let index: usize = self.children_num(&parent).unwrap_or(0);
333 match &self.inner {
334 MaybeDetached::Detached(t) => {
335 let t = &mut t.lock().unwrap().value;
336 Ok(t.create(parent.tree_id(), index))
337 }
338 MaybeDetached::Attached(a) => {
339 a.with_txn(|txn| self.create_with_txn(txn, parent, index, FiIfNotConfigured::Zero))
340 }
341 }
342 }
343
344 pub fn create_at(&self, parent: TreeParentId, index: usize) -> LoroResult<TreeID> {
345 match &self.inner {
346 MaybeDetached::Detached(t) => {
347 let t = &mut t.lock().unwrap().value;
348 Ok(t.create(parent.tree_id(), index))
349 }
350 MaybeDetached::Attached(a) => {
351 a.with_txn(|txn| self.create_with_txn(txn, parent, index, FiIfNotConfigured::Throw))
352 }
353 }
354 }
355
356 pub(crate) fn create_at_with_target_for_apply_diff(
358 &self,
359 parent: TreeParentId,
360 position: FractionalIndex,
361 target: TreeID,
362 ) -> LoroResult<bool> {
363 let MaybeDetached::Attached(a) = &self.inner else {
364 unreachable!();
365 };
366
367 if let Some(p) = self.get_node_parent(&target) {
368 if p == parent {
369 return Ok(false);
370 }
372 match p {
373 TreeParentId::Node(p) => {
374 if !self.is_node_unexist(&target) && !self.is_node_deleted(&p)? {
375 return self.move_at_with_target_for_apply_diff(parent, position, target);
376 }
377 }
378 TreeParentId::Root => {
379 return self.move_at_with_target_for_apply_diff(parent, position, target);
380 }
381 TreeParentId::Deleted | TreeParentId::Unexist => {}
382 }
383 }
384
385 let with_event = !parent
386 .tree_id()
387 .is_some_and(|p| self.is_node_deleted(&p).unwrap());
388 if !with_event {
389 return Ok(false);
390 }
391
392 let index = self
398 .get_index_by_fractional_index(
399 &parent,
400 &NodePosition {
401 position: position.clone(),
402 idlp: self.next_idlp(),
403 },
404 )
405 .unwrap_or(0);
407
408 let children = a.with_txn(|txn| {
409 let inner = self.inner.try_attached_state()?;
410
411 txn.apply_local_op(
412 inner.container_idx,
413 crate::op::RawOpContent::Tree(Arc::new(TreeOp::Create {
414 target,
415 parent: parent.tree_id(),
416 position: position.clone(),
417 })),
418 EventHint::Tree(smallvec![TreeDiffItem {
419 target,
420 action: TreeExternalDiff::Create {
421 parent,
422 index,
423 position: position.clone(),
424 },
425 }]),
426 &inner.doc,
427 )?;
428
429 Ok(self
430 .children(&TreeParentId::Node(target))
431 .unwrap_or_default())
432 })?;
433 for child in children {
434 let position = self.get_position_by_tree_id(&child).unwrap();
435 self.create_at_with_target_for_apply_diff(TreeParentId::Node(target), position, child)?;
436 }
437 Ok(true)
438 }
439
440 pub(crate) fn move_at_with_target_for_apply_diff(
442 &self,
443 parent: TreeParentId,
444 position: FractionalIndex,
445 target: TreeID,
446 ) -> LoroResult<bool> {
447 let MaybeDetached::Attached(a) = &self.inner else {
448 unreachable!();
449 };
450
451 if let Some(p) = self.get_node_parent(&target) {
457 if p == parent {
458 return Ok(false);
459 }
460 }
461
462 let index = self
463 .get_index_by_fractional_index(
464 &parent,
465 &NodePosition {
466 position: position.clone(),
467 idlp: self.next_idlp(),
468 },
469 )
470 .unwrap_or(0);
471 let with_event = !parent
472 .tree_id()
473 .is_some_and(|p| self.is_node_deleted(&p).unwrap());
474
475 if !with_event {
476 return Ok(false);
477 }
478
479 a.with_txn(|txn| {
485 let inner = self.inner.try_attached_state()?;
486 txn.apply_local_op(
487 inner.container_idx,
488 crate::op::RawOpContent::Tree(Arc::new(TreeOp::Move {
489 target,
490 parent: parent.tree_id(),
491 position: position.clone(),
492 })),
493 EventHint::Tree(smallvec![TreeDiffItem {
494 target,
495 action: TreeExternalDiff::Move {
496 parent,
497 index,
498 position: position.clone(),
499 old_parent: self.get_node_parent(&target).unwrap(),
501 old_index: self.get_index_by_tree_id(&target).unwrap(),
502 },
503 }]),
504 &inner.doc,
505 )
506 })?;
507 Ok(true)
508 }
509
510 pub(crate) fn create_with_txn(
511 &self,
512 txn: &mut Transaction,
513 parent: TreeParentId,
514 index: usize,
515 cfg: FiIfNotConfigured,
516 ) -> LoroResult<TreeID> {
517 let inner = self.inner.try_attached_state()?;
518 let target = TreeID::from_id(txn.next_id());
519
520 match self.generate_position_at(&target, &parent, index, cfg) {
521 FractionalIndexGenResult::Ok(position) => {
522 self.create_with_position(inner, txn, target, parent, index, position)
523 }
524 FractionalIndexGenResult::Rearrange(ids) => {
525 for (i, (id, position)) in ids.into_iter().enumerate() {
526 if i == 0 {
527 self.create_with_position(inner, txn, id, parent, index, position)?;
528 continue;
529 }
530 self.mov_with_position(inner, txn, id, parent, index + i, position, index + i)?;
531 }
532 Ok(target)
533 }
534 FractionalIndexGenResult::NotConfigured => {
535 Err(LoroTreeError::FractionalIndexNotEnabled.into())
536 }
537 }
538 }
539
540 pub fn mov(&self, target: TreeID, parent: TreeParentId) -> LoroResult<()> {
541 match &self.inner {
542 MaybeDetached::Detached(_) => {
543 let mut index: usize = self.children_num(&parent).unwrap_or(0);
544 if self.is_parent(&target, &parent) {
545 index -= 1;
546 }
547 self.move_to(target, parent, index)
548 }
549 MaybeDetached::Attached(a) => {
550 let mut index: usize = self.children_num(&parent).unwrap_or(0);
551 if self.is_parent(&target, &parent) {
552 index -= 1;
553 }
554 a.with_txn(|txn| {
555 self.mov_with_txn(txn, target, parent, index, FiIfNotConfigured::Zero)
556 })
557 }
558 }
559 }
560
561 pub fn mov_after(&self, target: TreeID, other: TreeID) -> LoroResult<()> {
562 let parent = self
563 .get_node_parent(&other)
564 .ok_or(LoroTreeError::TreeNodeNotExist(other))?;
565 let mut index = self.get_index_by_tree_id(&other).unwrap() + 1;
566 if self.is_parent(&target, &parent) && self.get_index_by_tree_id(&target).unwrap() < index {
567 index -= 1;
568 }
569 self.move_to(target, parent, index)
570 }
571
572 pub fn mov_before(&self, target: TreeID, other: TreeID) -> LoroResult<()> {
573 let parent = self
574 .get_node_parent(&other)
575 .ok_or(LoroTreeError::TreeNodeNotExist(other))?;
576 let mut index = self.get_index_by_tree_id(&other).unwrap();
577 if self.is_parent(&target, &parent)
578 && index >= 1
579 && self.get_index_by_tree_id(&target).unwrap() < index
580 {
581 index -= 1;
582 }
583 self.move_to(target, parent, index)
584 }
585
586 pub fn move_to(&self, target: TreeID, parent: TreeParentId, index: usize) -> LoroResult<()> {
587 match &self.inner {
588 MaybeDetached::Detached(t) => {
589 let mut t = t.lock().unwrap();
590 t.value.mov(target, parent.tree_id(), index)
591 }
592 MaybeDetached::Attached(a) => a.with_txn(|txn| {
593 self.mov_with_txn(txn, target, parent, index, FiIfNotConfigured::Throw)
594 }),
595 }
596 }
597
598 pub(crate) fn mov_with_txn(
599 &self,
600 txn: &mut Transaction,
601 target: TreeID,
602 parent: TreeParentId,
603 index: usize,
604 cfg: FiIfNotConfigured,
605 ) -> LoroResult<()> {
606 let inner = self.inner.try_attached_state()?;
607 let mut children_len = self.children_num(&parent).unwrap_or(0);
608 let mut already_in_parent = false;
609 if self.is_parent(&target, &parent) {
611 if let Some(current_index) = self.get_index_by_tree_id(&target) {
613 if current_index == index {
614 return Ok(());
615 }
616 children_len -= 1;
619 already_in_parent = true;
620 }
621 };
622 if index > children_len {
623 return Err(LoroTreeError::IndexOutOfBound {
624 len: children_len,
625 index,
626 }
627 .into());
628 }
629 let Some(old_index) = self.get_index_by_tree_id(&target) else {
630 return Err(LoroError::TreeError(
631 LoroTreeError::TreeNodeDeletedOrNotExist(target),
632 ));
633 };
634
635 if already_in_parent {
636 self.delete_position(&parent, &target);
637 }
638
639 match self.generate_position_at(&target, &parent, index, cfg) {
640 FractionalIndexGenResult::Ok(position) => {
641 self.mov_with_position(inner, txn, target, parent, index, position, old_index)
642 }
643 FractionalIndexGenResult::Rearrange(ids) => {
644 for (i, (id, position)) in ids.into_iter().enumerate() {
645 self.mov_with_position(inner, txn, id, parent, index + i, position, old_index)?;
646 }
647 Ok(())
648 }
649 FractionalIndexGenResult::NotConfigured => {
650 Err(LoroTreeError::FractionalIndexNotEnabled.into())
651 }
652 }
653 }
654
655 #[allow(clippy::too_many_arguments)]
656 fn create_with_position(
657 &self,
658 inner: &BasicHandler,
659 txn: &mut Transaction,
660 tree_id: TreeID,
661 parent: TreeParentId,
662 index: usize,
663 position: FractionalIndex,
664 ) -> LoroResult<TreeID> {
665 txn.apply_local_op(
666 inner.container_idx,
667 crate::op::RawOpContent::Tree(Arc::new(TreeOp::Create {
668 target: tree_id,
669 parent: parent.tree_id(),
670 position: position.clone(),
671 })),
672 EventHint::Tree(smallvec![TreeDiffItem {
673 target: tree_id,
674 action: TreeExternalDiff::Create {
675 parent,
676 index,
677 position,
678 },
679 }]),
680 &inner.doc,
681 )?;
682 Ok(tree_id)
683 }
684
685 #[allow(clippy::too_many_arguments)]
686 fn mov_with_position(
687 &self,
688 inner: &BasicHandler,
689 txn: &mut Transaction,
690 target: TreeID,
691 parent: TreeParentId,
692 index: usize,
693 position: FractionalIndex,
694 old_index: usize,
695 ) -> LoroResult<()> {
696 txn.apply_local_op(
697 inner.container_idx,
698 crate::op::RawOpContent::Tree(Arc::new(TreeOp::Move {
699 target,
700 parent: parent.tree_id(),
701 position: position.clone(),
702 })),
703 EventHint::Tree(smallvec![TreeDiffItem {
704 target,
705 action: TreeExternalDiff::Move {
706 parent,
707 index,
708 position,
709 old_parent: self.get_node_parent(&target).unwrap(),
710 old_index,
711 },
712 }]),
713 &inner.doc,
714 )
715 }
716
717 pub fn get_meta(&self, target: TreeID) -> LoroResult<MapHandler> {
718 match &self.inner {
719 MaybeDetached::Detached(d) => {
720 let d = d.lock().unwrap();
721 d.value
722 .map
723 .get(&target)
724 .cloned()
725 .ok_or(LoroTreeError::TreeNodeNotExist(target).into())
726 }
727 MaybeDetached::Attached(a) => {
728 if self.is_node_unexist(&target) {
729 return Err(LoroTreeError::TreeNodeNotExist(target).into());
730 }
731 let map_container_id = target.associated_meta_container();
732 let handler = create_handler(a, map_container_id);
733 Ok(handler.into_map().unwrap())
734 }
735 }
736 }
737
738 pub fn is_node_unexist(&self, target: &TreeID) -> bool {
739 match &self.inner {
740 MaybeDetached::Detached(d) => {
741 let d = d.lock().unwrap();
742 !d.value.map.contains_key(target)
743 }
744 MaybeDetached::Attached(a) => a.with_state(|state| {
745 let a = state.as_tree_state().unwrap();
746 a.is_node_unexist(target)
747 }),
748 }
749 }
750
751 pub fn is_node_deleted(&self, target: &TreeID) -> LoroResult<bool> {
752 match &self.inner {
753 MaybeDetached::Detached(t) => {
754 let t = t.lock().unwrap();
755 t.value
756 .map
757 .get(target)
758 .and(Some(true))
759 .ok_or(LoroTreeError::TreeNodeNotExist(*target).into())
760 }
761 MaybeDetached::Attached(a) => a.with_state(|state| {
762 let a = state.as_tree_state().unwrap();
763 a.is_node_deleted(target)
764 .ok_or(LoroTreeError::TreeNodeNotExist(*target).into())
765 }),
766 }
767 }
768
769 pub fn get_node_parent(&self, target: &TreeID) -> Option<TreeParentId> {
771 match &self.inner {
772 MaybeDetached::Detached(t) => {
773 let t = t.lock().unwrap();
774 t.value.get_parent(target).map(TreeParentId::from)
775 }
776 MaybeDetached::Attached(a) => a.with_state(|state| {
777 let a = state.as_tree_state().unwrap();
778 a.parent(target)
779 }),
780 }
781 }
782
783 pub fn children(&self, parent: &TreeParentId) -> Option<Vec<TreeID>> {
785 match &self.inner {
786 MaybeDetached::Detached(t) => {
787 let t = t.lock().unwrap();
788 t.value.get_children(parent.tree_id())
789 }
790 MaybeDetached::Attached(a) => a.with_state(|state| {
791 let a = state.as_tree_state().unwrap();
792 a.get_children(parent).map(|x| x.collect())
793 }),
794 }
795 }
796
797 pub fn children_num(&self, parent: &TreeParentId) -> Option<usize> {
798 match &self.inner {
799 MaybeDetached::Detached(t) => {
800 let t = t.lock().unwrap();
801 t.value.children_num(parent.tree_id())
802 }
803 MaybeDetached::Attached(a) => a.with_state(|state| {
804 let a = state.as_tree_state().unwrap();
805 a.children_num(parent)
806 }),
807 }
808 }
809
810 pub fn contains(&self, target: TreeID) -> bool {
812 match &self.inner {
813 MaybeDetached::Detached(t) => {
814 let t = t.lock().unwrap();
815 t.value.map.contains_key(&target)
816 }
817 MaybeDetached::Attached(a) => a.with_state(|state| {
818 let a = state.as_tree_state().unwrap();
819 !a.is_node_unexist(&target)
820 }),
821 }
822 }
823
824 pub fn get_child_at(&self, parent: &TreeParentId, index: usize) -> Option<TreeID> {
825 match &self.inner {
826 MaybeDetached::Detached(t) => {
827 let t = t.lock().unwrap();
828 t.value.get_id_by_index(&parent.tree_id(), index)
829 }
830 MaybeDetached::Attached(a) => a.with_state(|state| {
831 let a = state.as_tree_state().unwrap();
832 a.get_id_by_index(parent, index)
833 }),
834 }
835 }
836
837 pub fn is_parent(&self, target: &TreeID, parent: &TreeParentId) -> bool {
838 match &self.inner {
839 MaybeDetached::Detached(t) => {
840 let t = t.lock().unwrap();
841 t.value.is_parent(target, &parent.tree_id())
842 }
843 MaybeDetached::Attached(a) => a.with_state(|state| {
844 let a = state.as_tree_state().unwrap();
845 a.is_parent(target, parent)
846 }),
847 }
848 }
849
850 pub fn nodes(&self) -> Vec<TreeID> {
852 match &self.inner {
853 MaybeDetached::Detached(t) => {
854 let t = t.lock().unwrap();
855 t.value.map.keys().cloned().collect()
856 }
857 MaybeDetached::Attached(a) => a.with_state(|state| {
858 let a = state.as_tree_state().unwrap();
859 a.nodes()
860 }),
861 }
862 }
863
864 pub fn get_nodes_under(&self, parent: TreeParentId) -> Vec<TreeNode> {
865 match &self.inner {
866 MaybeDetached::Detached(_t) => {
867 unreachable!()
868 }
869 MaybeDetached::Attached(a) => a.with_state(|state| {
870 let a = state.as_tree_state().unwrap();
871 a.get_all_tree_nodes_under(parent)
872 }),
873 }
874 }
875 pub fn roots(&self) -> Vec<TreeID> {
876 self.children(&TreeParentId::Root).unwrap_or_default()
877 }
878
879 pub fn get_all_hierarchy_nodes_under(&self, parent: TreeParentId) -> Vec<TreeNodeWithChildren> {
880 match &self.inner {
881 MaybeDetached::Detached(_t) => {
882 unimplemented!()
884 }
885 MaybeDetached::Attached(a) => a.with_state(|state| {
886 let a = state.as_tree_state().unwrap();
887 a.get_all_hierarchy_nodes_under(parent)
888 }),
889 }
890 }
891
892 #[allow(non_snake_case)]
893 pub fn __internal__next_tree_id(&self) -> TreeID {
894 match &self.inner {
895 MaybeDetached::Detached(d) => {
896 let d = d.lock().unwrap();
897 TreeID::new(PeerID::MAX, d.value.next_counter)
898 }
899 MaybeDetached::Attached(a) => a
900 .with_txn(|txn| Ok(TreeID::from_id(txn.next_id())))
901 .unwrap(),
902 }
903 }
904
905 fn generate_position_at(
906 &self,
907 target: &TreeID,
908 parent: &TreeParentId,
909 index: usize,
910 cfg: FiIfNotConfigured,
911 ) -> FractionalIndexGenResult {
912 let MaybeDetached::Attached(a) = &self.inner else {
913 unreachable!()
914 };
915 a.with_state(|state| {
916 let a = state.as_tree_state_mut().unwrap();
917 a.generate_position_at(target, parent, index, cfg)
918 })
919 }
920
921 pub fn get_index_by_tree_id(&self, target: &TreeID) -> Option<usize> {
925 match &self.inner {
926 MaybeDetached::Detached(t) => {
927 let t = t.lock().unwrap();
928 t.value.get_index_by_tree_id(target)
929 }
930 MaybeDetached::Attached(a) => a.with_state(|state| {
931 let a = state.as_tree_state().unwrap();
932 a.get_index_by_tree_id(target)
933 }),
934 }
935 }
936
937 pub fn get_position_by_tree_id(&self, target: &TreeID) -> Option<FractionalIndex> {
938 match &self.inner {
939 MaybeDetached::Detached(_) => unreachable!(),
940 MaybeDetached::Attached(a) => a.with_state(|state| {
941 let a = state.as_tree_state().unwrap();
942 a.get_position(target)
943 }),
944 }
945 }
946
947 fn delete_position(&self, parent: &TreeParentId, target: &TreeID) {
948 let MaybeDetached::Attached(a) = &self.inner else {
949 unreachable!()
950 };
951 a.with_state(|state| {
952 let a = state.as_tree_state_mut().unwrap();
953 a.try_delete_position_cache(parent, target)
954 })
955 }
956
957 pub(crate) fn get_index_by_fractional_index(
959 &self,
960 parent: &TreeParentId,
961 node_position: &NodePosition,
962 ) -> Option<usize> {
963 match &self.inner {
964 MaybeDetached::Detached(_) => {
965 unreachable!();
966 }
967 MaybeDetached::Attached(a) => a.with_state(|state| {
968 let a = state.as_tree_state().unwrap();
969 a.get_index_by_position(parent, node_position)
970 }),
971 }
972 }
973
974 pub(crate) fn next_idlp(&self) -> IdLp {
975 match &self.inner {
976 MaybeDetached::Detached(_) => {
977 unreachable!()
978 }
979 MaybeDetached::Attached(a) => a.with_txn(|txn| Ok(txn.next_idlp())).unwrap(),
980 }
981 }
982
983 pub fn is_fractional_index_enabled(&self) -> bool {
984 match &self.inner {
985 MaybeDetached::Detached(_) => {
986 unreachable!()
987 }
988 MaybeDetached::Attached(a) => a.with_state(|state| {
989 let a = state.as_tree_state().unwrap();
990 a.is_fractional_index_enabled()
991 }),
992 }
993 }
994
995 pub fn enable_fractional_index(&self, jitter: u8) {
1003 match &self.inner {
1004 MaybeDetached::Detached(_) => {
1005 unreachable!()
1006 }
1007 MaybeDetached::Attached(a) => a.with_state(|state| {
1008 let a = state.as_tree_state_mut().unwrap();
1009 a.enable_generate_fractional_index(jitter);
1010 }),
1011 }
1012 }
1013
1014 pub fn disable_fractional_index(&self) {
1019 match &self.inner {
1020 MaybeDetached::Detached(_) => {
1021 unreachable!()
1022 }
1023 MaybeDetached::Attached(a) => a.with_state(|state| {
1024 let a = state.as_tree_state_mut().unwrap();
1025 a.disable_generate_fractional_index();
1026 }),
1027 }
1028 }
1029
1030 pub fn is_deleted(&self) -> bool {
1031 match &self.inner {
1032 MaybeDetached::Detached(_) => false,
1033 MaybeDetached::Attached(a) => a.is_deleted(),
1034 }
1035 }
1036
1037 pub fn is_empty(&self) -> bool {
1038 match &self.inner {
1039 MaybeDetached::Detached(t) => {
1040 let t = t.lock().unwrap();
1041 t.value.map.is_empty()
1042 }
1043 MaybeDetached::Attached(a) => a.with_state(|state| {
1044 let a = state.as_tree_state().unwrap();
1045 a.is_empty()
1046 }),
1047 }
1048 }
1049
1050 pub fn get_last_move_id(&self, target: &TreeID) -> Option<ID> {
1051 match &self.inner {
1052 MaybeDetached::Detached(_) => None,
1053 MaybeDetached::Attached(a) => a.with_state(|state| {
1054 let a = state.as_tree_state().unwrap();
1055 a.get_last_move_id(target)
1056 }),
1057 }
1058 }
1059
1060 pub fn clear(&self) -> LoroResult<()> {
1061 match &self.inner {
1062 MaybeDetached::Detached(t) => {
1063 let mut t = t.lock().unwrap();
1064 t.value.map.clear();
1065 t.value.children_links.clear();
1066 t.value.parent_links.clear();
1067 Ok(())
1068 }
1069 MaybeDetached::Attached(_) => {
1070 let nodes = self.get_nodes_under(TreeParentId::Root);
1071 for node in nodes {
1072 self.delete(node.id)?;
1073 }
1074 Ok(())
1075 }
1076 }
1077 }
1078}