mirror of
https://github.com/helix-editor/helix.git
synced 2025-01-19 13:37:06 +04:00
Move Tree nodes on view swap
Instead of moving the Node contents on view swap if they have the same parent reorder them to keep traversal order otherwise re-parent them.
This commit is contained in:
parent
3f10473d30
commit
6bd8924436
@ -538,20 +538,72 @@ pub fn transpose(&mut self) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn swap_split_in_direction(&mut self, direction: Direction) {
|
pub fn swap_split_in_direction(&mut self, direction: Direction) -> Option<()> {
|
||||||
if let Some(id) = self.find_split_in_direction(self.focus, direction) {
|
let focus = self.focus;
|
||||||
if let Some([focused, target]) = self.nodes.get_disjoint_mut([self.focus, id]) {
|
let target = self.find_split_in_direction(focus, direction)?;
|
||||||
match (&mut focused.content, &mut target.content) {
|
let focus_parent = self.nodes[focus].parent;
|
||||||
(Content::View(focused), Content::View(target)) => {
|
let target_parent = self.nodes[target].parent;
|
||||||
std::mem::swap(&mut focused.doc, &mut target.doc);
|
|
||||||
std::mem::swap(&mut focused.id, &mut target.id);
|
if focus_parent == target_parent {
|
||||||
self.focus = id;
|
let parent = focus_parent;
|
||||||
}
|
let [parent, focus, target] = self.nodes.get_disjoint_mut([parent, focus, target])?;
|
||||||
// self.focus always points to a view which has a content of Content::View
|
match (&mut parent.content, &mut focus.content, &mut target.content) {
|
||||||
// and find_split_in_direction() only returns a view which has content of
|
(
|
||||||
// Content::View.
|
Content::Container(parent),
|
||||||
_ => unreachable!(),
|
Content::View(focus_view),
|
||||||
|
Content::View(target_view),
|
||||||
|
) => {
|
||||||
|
let focus_pos = parent.children.iter().position(|id| focus_view.id == *id)?;
|
||||||
|
let target_pos = parent
|
||||||
|
.children
|
||||||
|
.iter()
|
||||||
|
.position(|id| target_view.id == *id)?;
|
||||||
|
// swap node positions so that traversal order is kept
|
||||||
|
parent.children[focus_pos] = target_view.id;
|
||||||
|
parent.children[target_pos] = focus_view.id;
|
||||||
|
// swap area so that views rendered at the correct location
|
||||||
|
std::mem::swap(&mut focus_view.area, &mut target_view.area);
|
||||||
|
|
||||||
|
Some(())
|
||||||
}
|
}
|
||||||
|
_ => unreachable!(),
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
let [focus_parent, target_parent, focus, target] =
|
||||||
|
self.nodes
|
||||||
|
.get_disjoint_mut([focus_parent, target_parent, focus, target])?;
|
||||||
|
match (
|
||||||
|
&mut focus_parent.content,
|
||||||
|
&mut target_parent.content,
|
||||||
|
&mut focus.content,
|
||||||
|
&mut target.content,
|
||||||
|
) {
|
||||||
|
(
|
||||||
|
Content::Container(focus_parent),
|
||||||
|
Content::Container(target_parent),
|
||||||
|
Content::View(focus_view),
|
||||||
|
Content::View(target_view),
|
||||||
|
) => {
|
||||||
|
let focus_pos = focus_parent
|
||||||
|
.children
|
||||||
|
.iter()
|
||||||
|
.position(|id| focus_view.id == *id)?;
|
||||||
|
let target_pos = target_parent
|
||||||
|
.children
|
||||||
|
.iter()
|
||||||
|
.position(|id| target_view.id == *id)?;
|
||||||
|
// re-parent target and focus nodes
|
||||||
|
std::mem::swap(
|
||||||
|
&mut focus_parent.children[focus_pos],
|
||||||
|
&mut target_parent.children[target_pos],
|
||||||
|
);
|
||||||
|
std::mem::swap(&mut focus.parent, &mut target.parent);
|
||||||
|
// swap area so that views rendered at the correct location
|
||||||
|
std::mem::swap(&mut focus_view.area, &mut target_view.area);
|
||||||
|
|
||||||
|
Some(())
|
||||||
|
}
|
||||||
|
_ => unreachable!(),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -738,7 +790,7 @@ fn doc_id(tree: &Tree, view_id: ViewId) -> Option<DocumentId> {
|
|||||||
tree.swap_split_in_direction(Direction::Down);
|
tree.swap_split_in_direction(Direction::Down);
|
||||||
// | l1 | l2 | |
|
// | l1 | l2 | |
|
||||||
// | l0* | r0 |
|
// | l0* | r0 |
|
||||||
assert_eq!(tree.focus, l1);
|
assert_eq!(tree.focus, l0);
|
||||||
assert_eq!(doc_id(&tree, l0), Some(doc_l1));
|
assert_eq!(doc_id(&tree, l0), Some(doc_l1));
|
||||||
assert_eq!(doc_id(&tree, l1), Some(doc_l0));
|
assert_eq!(doc_id(&tree, l1), Some(doc_l0));
|
||||||
assert_eq!(doc_id(&tree, l2), Some(doc_l2));
|
assert_eq!(doc_id(&tree, l2), Some(doc_l2));
|
||||||
@ -748,7 +800,7 @@ fn doc_id(tree: &Tree, view_id: ViewId) -> Option<DocumentId> {
|
|||||||
|
|
||||||
// | l1 | l2 | |
|
// | l1 | l2 | |
|
||||||
// | r0 | l0* |
|
// | r0 | l0* |
|
||||||
assert_eq!(tree.focus, r0);
|
assert_eq!(tree.focus, l0);
|
||||||
assert_eq!(doc_id(&tree, l0), Some(doc_l1));
|
assert_eq!(doc_id(&tree, l0), Some(doc_l1));
|
||||||
assert_eq!(doc_id(&tree, l1), Some(doc_r0));
|
assert_eq!(doc_id(&tree, l1), Some(doc_r0));
|
||||||
assert_eq!(doc_id(&tree, l2), Some(doc_l2));
|
assert_eq!(doc_id(&tree, l2), Some(doc_l2));
|
||||||
@ -758,7 +810,7 @@ fn doc_id(tree: &Tree, view_id: ViewId) -> Option<DocumentId> {
|
|||||||
tree.swap_split_in_direction(Direction::Up);
|
tree.swap_split_in_direction(Direction::Up);
|
||||||
// | l1 | l2 | |
|
// | l1 | l2 | |
|
||||||
// | r0 | l0* |
|
// | r0 | l0* |
|
||||||
assert_eq!(tree.focus, r0);
|
assert_eq!(tree.focus, l0);
|
||||||
assert_eq!(doc_id(&tree, l0), Some(doc_l1));
|
assert_eq!(doc_id(&tree, l0), Some(doc_l1));
|
||||||
assert_eq!(doc_id(&tree, l1), Some(doc_r0));
|
assert_eq!(doc_id(&tree, l1), Some(doc_r0));
|
||||||
assert_eq!(doc_id(&tree, l2), Some(doc_l2));
|
assert_eq!(doc_id(&tree, l2), Some(doc_l2));
|
||||||
@ -768,7 +820,7 @@ fn doc_id(tree: &Tree, view_id: ViewId) -> Option<DocumentId> {
|
|||||||
tree.swap_split_in_direction(Direction::Down);
|
tree.swap_split_in_direction(Direction::Down);
|
||||||
// | l1 | l2 | |
|
// | l1 | l2 | |
|
||||||
// | r0 | l0* |
|
// | r0 | l0* |
|
||||||
assert_eq!(tree.focus, r0);
|
assert_eq!(tree.focus, l0);
|
||||||
assert_eq!(doc_id(&tree, l0), Some(doc_l1));
|
assert_eq!(doc_id(&tree, l0), Some(doc_l1));
|
||||||
assert_eq!(doc_id(&tree, l1), Some(doc_r0));
|
assert_eq!(doc_id(&tree, l1), Some(doc_r0));
|
||||||
assert_eq!(doc_id(&tree, l2), Some(doc_l2));
|
assert_eq!(doc_id(&tree, l2), Some(doc_l2));
|
||||||
@ -781,7 +833,7 @@ fn doc_id(tree: &Tree, view_id: ViewId) -> Option<DocumentId> {
|
|||||||
tree.swap_split_in_direction(Direction::Down);
|
tree.swap_split_in_direction(Direction::Down);
|
||||||
// | l1 | r0 | |
|
// | l1 | r0 | |
|
||||||
// | l2* | l0 |
|
// | l2* | l0 |
|
||||||
assert_eq!(tree.focus, l1);
|
assert_eq!(tree.focus, l2);
|
||||||
assert_eq!(doc_id(&tree, l0), Some(doc_l1));
|
assert_eq!(doc_id(&tree, l0), Some(doc_l1));
|
||||||
assert_eq!(doc_id(&tree, l1), Some(doc_l2));
|
assert_eq!(doc_id(&tree, l1), Some(doc_l2));
|
||||||
assert_eq!(doc_id(&tree, l2), Some(doc_r0));
|
assert_eq!(doc_id(&tree, l2), Some(doc_r0));
|
||||||
@ -790,7 +842,7 @@ fn doc_id(tree: &Tree, view_id: ViewId) -> Option<DocumentId> {
|
|||||||
tree.swap_split_in_direction(Direction::Up);
|
tree.swap_split_in_direction(Direction::Up);
|
||||||
// | l2* | r0 | |
|
// | l2* | r0 | |
|
||||||
// | l1 | l0 |
|
// | l1 | l0 |
|
||||||
assert_eq!(tree.focus, l0);
|
assert_eq!(tree.focus, l2);
|
||||||
assert_eq!(doc_id(&tree, l0), Some(doc_l2));
|
assert_eq!(doc_id(&tree, l0), Some(doc_l2));
|
||||||
assert_eq!(doc_id(&tree, l1), Some(doc_l1));
|
assert_eq!(doc_id(&tree, l1), Some(doc_l1));
|
||||||
assert_eq!(doc_id(&tree, l2), Some(doc_r0));
|
assert_eq!(doc_id(&tree, l2), Some(doc_r0));
|
||||||
|
Loading…
Reference in New Issue
Block a user