Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

File explorer and tree helper (v3) #5768

Open
wants to merge 118 commits into
base: master
Choose a base branch
from
Open
Changes from 1 commit
Commits
Show all changes
118 commits
Select commit Hold shift + click to select a range
b652f96
tree helper and file explorer
cossonleo Apr 3, 2022
d9d4daa
feat(ui/explore): implement "focus current file"
wongjiahau Feb 1, 2023
c446c39
feat(explorer/position): right
wongjiahau Feb 6, 2023
d04a1ce
refactor(tree): change internal implementation
wongjiahau Feb 11, 2023
aa397ef
feat(explore): reveal current file
wongjiahau Feb 12, 2023
bdab93e
feat(explore): search
wongjiahau Feb 12, 2023
82fe4a3
test(ui/tree): find
wongjiahau Feb 12, 2023
0f8b641
feat(tree): filter
wongjiahau Feb 12, 2023
458fa1c
feat(explore): add folder/file
wongjiahau Feb 13, 2023
2af8b41
feat(explore): remove files/folder
wongjiahau Feb 13, 2023
44b46dd
feat(explore): rename file/folder
wongjiahau Feb 13, 2023
5a5a1de
fix(explore/rename): should regenarate index
wongjiahau Feb 13, 2023
52a26ff
feat(explore): refresh
wongjiahau Feb 13, 2023
ec2059b
style(ui/tree): highlight ancestor
wongjiahau Feb 13, 2023
ddb7564
feat(explore): add help
wongjiahau Feb 13, 2023
2bafac0
feat(explore): go to previous root
wongjiahau Feb 13, 2023
35ffc60
feat(explore): increase/decrease explorer size
wongjiahau Feb 13, 2023
2c221f0
fix(explore): help page overflow
wongjiahau Feb 13, 2023
790192d
doc(explorer): up to date
wongjiahau Feb 13, 2023
b38a941
feat(explore): close without clearing previous state
wongjiahau Feb 13, 2023
56056e8
fix(explore): increase size will cause panic
wongjiahau Feb 14, 2023
a079477
fix(compile): warnings
wongjiahau Feb 14, 2023
85fa1c5
feat(explore):
wongjiahau Feb 14, 2023
9bd534b
fix(explore): filter
wongjiahau Feb 14, 2023
7249536
fix(explore): 'h' does not realign preview properly
wongjiahau Feb 14, 2023
94e2c29
fix(command): space e does not focus explorer when no files are opened
wongjiahau Feb 15, 2023
374b8dd
style(explore): make Right the default position
wongjiahau Feb 15, 2023
c8578ba
fix: warnings
wongjiahau Feb 15, 2023
30bac64
Revert "style(explore): make Right the default position"
wongjiahau Feb 15, 2023
ef73559
fix(explore): cannot focus explorer if no opened document
wongjiahau Feb 15, 2023
0f8e0a5
fix(tree): deleting last file causes panic
wongjiahau Feb 15, 2023
70984fd
Merge branch 'master' of https://github.com/helix-editor/helix into r…
wongjiahau Feb 15, 2023
f0a4b10
Merge branch 'refactor-tree-explorer' of github.com:pinelang/helix-tr…
wongjiahau Feb 15, 2023
4dfa869
style(tree): increase indentation
wongjiahau Feb 16, 2023
c88164f
feat(tree-view): add unit tests
wongjiahau Feb 16, 2023
64059fb
feat(tree): move left/right
wongjiahau Feb 19, 2023
2a60662
feat(explore): add focus indicator
wongjiahau Feb 20, 2023
2e654a0
refactor(explore): move search function to Tree
wongjiahau Feb 20, 2023
2e7709e
MULTI
wongjiahau Feb 21, 2023
a259c20
fix(explore): help overflow
wongjiahau Feb 22, 2023
bcb1672
fix(explore):
wongjiahau Feb 22, 2023
78bb297
Merge branch 'master' of https://github.com/helix-editor/helix into a…
wongjiahau Feb 22, 2023
6321dc9
chore: rename explore to explorer
wongjiahau Feb 22, 2023
7b63fda
test(explorer): add integration tests
wongjiahau Feb 23, 2023
f9ff01d
chore(ui/tree): bind 'o' to Toggle
wongjiahau Feb 23, 2023
899491b
feat(tree): add C-n/C-p keybinding
wongjiahau Feb 23, 2023
6af9a06
feat(explorer): bind "="/"_" to "Zoom in"/"Zoom out"
wongjiahau Feb 23, 2023
9205117
fix: failing tests
wongjiahau Feb 23, 2023
cf9b60a
feat(tree): sticky ancestors
wongjiahau Feb 25, 2023
dffbc15
refactor(explorer,tree): remove unwrap to avoid panics
wongjiahau Feb 25, 2023
36769cb
fix(explorer/keymap): change 'b' to 'B'
wongjiahau Feb 25, 2023
b5d92ac
chore: fix clippy warnings
wongjiahau Feb 25, 2023
38ef079
feat(tree): jump forward
wongjiahau Feb 25, 2023
24b50bb
feat(explorer): toggle preview
wongjiahau Feb 25, 2023
72b845d
Merge branch 'master' of https://github.com/helix-editor/helix into t…
wongjiahau Feb 25, 2023
ba00a80
fix(tree): shouldn't use patched font
wongjiahau Feb 25, 2023
601f2c4
chore(ui/tree): remove useless comments
wongjiahau Feb 25, 2023
ef18502
chore: remove temp file
wongjiahau Feb 25, 2023
5d600fe
doc(helix-term/.gitignore): document purpose of test-explorer
wongjiahau Feb 25, 2023
d578f8a
chore: fix clippy warning
wongjiahau Feb 26, 2023
c3b8be9
fix(ci): clippy + failure on Windows
wongjiahau Feb 26, 2023
c0073ed
Merge branch 'tree_explore' of github.com:pinelang/helix-tree-explore…
wongjiahau Feb 26, 2023
4a0c620
fix(explorer/filter): not working for newly opened folder
wongjiahau Feb 26, 2023
7e4feb0
fix(explore): search using previous search word after filter does not…
wongjiahau Feb 27, 2023
fae4990
test(tree): search prompt and filter prompt
wongjiahau Feb 28, 2023
8379669
Merge branch 'master' of https://github.com/helix-editor/helix into t…
wongjiahau Feb 28, 2023
b18a974
fix(explorer): go to previous root does not update state.current_root
wongjiahau Feb 28, 2023
19d436e
Merge branch 'master' of https://github.com/helix-editor/helix into t…
wongjiahau Mar 1, 2023
a2cb28d
chore(keymap): merge with the correct version
wongjiahau Mar 1, 2023
43b226a
feat(explorer/keymap): combine 'a' with 'A'
wongjiahau Mar 2, 2023
c2e2f05
feat(explorer/delete): no need to press Enter, just press y
wongjiahau Mar 2, 2023
a4943a7
fix(explorer/overlay): prompt overflow
wongjiahau Mar 2, 2023
8ef95ee
Merge branch 'master' of https://github.com/helix-editor/helix into t…
wongjiahau Mar 6, 2023
d3db1b6
style(tree): improve ancestor contrast
wongjiahau Mar 6, 2023
31c0e84
fix(ci): failing windows test & clippy
wongjiahau Mar 6, 2023
bc62b76
fix(ci): failing windows test & clippy
wongjiahau Mar 6, 2023
aa6780e
feat(ui/tree): tree-based movements
wongjiahau Mar 7, 2023
80a2f86
Merge branch 'tree_explore' of github.com:pinelang/helix-tree-explore…
wongjiahau Mar 7, 2023
d62b487
feat(ui/tree): undo breaking changes
wongjiahau Mar 7, 2023
d1e6a21
Merge branch 'master' of https://github.com/helix-editor/helix into t…
wongjiahau Mar 7, 2023
e991ed9
refactor(runtime/themes): revert changes to theme files
wongjiahau Mar 7, 2023
9726ae7
fix(ci/test): failing on Windows
wongjiahau Mar 7, 2023
7ccee10
chore: correction of e991ed9
wongjiahau Mar 7, 2023
10032eb
fix(ci): cargo fmt
wongjiahau Mar 8, 2023
d043ea4
Merge branch 'master' of https://github.com/helix-editor/helix into t…
wongjiahau Mar 8, 2023
1108c88
Merge branch 'master' into tree_explore
wongjiahau Mar 9, 2023
eb9287d
fix(ci): cargo fmt and windows test
wongjiahau Mar 9, 2023
20241fb
Merge branch 'master' of https://github.com/helix-editor/helix into t…
wongjiahau Mar 13, 2023
54b1693
Merge branch 'master' of https://github.com/helix-editor/helix into t…
wongjiahau Mar 14, 2023
c4c3e80
style(explorer/delete): capitalize default choice
wongjiahau Mar 14, 2023
1780867
refactor(ui/explorer/handle_prompt_event): remove unnecessary function
wongjiahau Mar 14, 2023
9a1aff2
refactor(ui/explorer/close_documents): concise code
wongjiahau Mar 14, 2023
8b561e2
fix: type error
wongjiahau Mar 14, 2023
41ebc30
fix(ui/tree/clone): `is_openend` should not be false
wongjiahau Mar 14, 2023
52be2e0
refactor(ui/tree): remove filter
wongjiahau Mar 16, 2023
f5af209
refactor(explorer): remove preview
wongjiahau Mar 17, 2023
afda68a
chore: cargo fmt
wongjiahau Mar 17, 2023
e5dfde2
refactor(explorer): remove overlay option
wongjiahau Mar 17, 2023
1be2ac2
fix(ui/explorer): tree search cursor not rendered
wongjiahau Mar 18, 2023
ee34720
style(explorer): move title to statusline
wongjiahau Mar 18, 2023
404f950
fix(tests/explorer/new_folder): failing on Windows
wongjiahau Mar 22, 2023
898c167
fix(integration-test/test_goto_file_impl): failing due to untested ch…
wongjiahau Mar 22, 2023
33542e9
refactor: remove unnecessary dev-dependencies
wongjiahau Mar 22, 2023
f5aec54
chore(commands): revert accidental typo
wongjiahau Mar 22, 2023
a331e52
chore(keymap): remove "<space>E"
wongjiahau Mar 22, 2023
eebff62
chore(doc/configuration/explorer/position): remove `overlay` option
wongjiahau Mar 22, 2023
f37c795
chore(ui/prompt): use &str instead of Cow<str>
wongjiahau Mar 23, 2023
88ac941
Merge branch 'master' of https://github.com/helix-editor/helix into t…
wongjiahau Mar 25, 2023
cf9669f
fix(ci): clippy error
wongjiahau Mar 27, 2023
e2c3757
Merge branch 'master' of https://github.com/helix-editor/helix into t…
wongjiahau May 12, 2023
2d1ca23
refactor(explorer/tests): use tempfile instead
wongjiahau May 12, 2023
4fe9896
chore(book/src/configuration.md): remove accidental addition
wongjiahau May 12, 2023
d86abf1
chore(explorer): minor code changes
wongjiahau May 12, 2023
0885057
perf(tree/reveal-item): remove over-conservative refresh
wongjiahau May 16, 2023
99e3db4
Merge branch 'master' of https://github.com/helix-editor/helix into t…
wongjiahau Jun 16, 2023
10f302d
Merge branch 'master' of https://github.com/helix-editor/helix into t…
wongjiahau Jul 25, 2023
fd80660
Merge branch 'master' of https://github.com/helix-editor/helix into t…
wongjiahau Jul 31, 2023
2c27974
Merge branch 'master' of https://github.com/helix-editor/helix into t…
wongjiahau Aug 22, 2023
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Prev Previous commit
Next Next commit
feat(tree): filter
wongjiahau committed Feb 12, 2023
commit 0f8b641a5d67027cfc69aaeeb22e46f7fbd26eaf
7 changes: 3 additions & 4 deletions helix-term/src/ui/explore.rs
Original file line number Diff line number Diff line change
@@ -618,10 +618,9 @@ impl Explorer {
.handle_event(Event::Key(event), cx, &mut self.state);
}
key!(Enter) => {
self.tree.clean_recycle();
return self
.tree
.handle_event(Event::Key(event), cx, &mut self.state);
if let EventResult::Consumed(_) = prompt.handle_event(Event::Key(event), cx) {
self.tree.filter(prompt.line(), cx, &mut self.state);
}
}
key!(Esc) | ctrl!('c') => self.tree.restore_recycle(),
_ => {
175 changes: 93 additions & 82 deletions helix-term/src/ui/tree.rs
Original file line number Diff line number Diff line change
@@ -165,6 +165,32 @@ impl<'a, T> DoubleEndedIterator for TreeIter<'a, T> {

impl<'a, T> ExactSizeIterator for TreeIter<'a, T> {}

impl<T: Clone> Tree<T> {
pub fn filter<P>(tree: &Tree<T>, predicate: &P) -> Option<Tree<T>>
where
P: Fn(&T) -> bool,
{
let children = tree
.children
.iter()
.filter_map(|tree| Self::filter(tree, predicate))
.collect::<Vec<_>>();
if predicate(&tree.item) || !children.is_empty() {
let mut tree = Tree {
item: tree.item.clone(),
parent_index: tree.parent_index,
index: tree.index,
is_opened: tree.is_opened,
children,
};
tree.regenerate_index();
Some(tree)
} else {
None
}
}
}

impl<T> Tree<T> {
pub fn new(item: T, children: Vec<Tree<T>>) -> Self {
Self {
@@ -234,7 +260,7 @@ impl<T> Tree<T> {

pub struct TreeView<T: TreeItem> {
tree: Tree<T>,
recycle: Option<(String, Vec<Tree<T>>)>,
recycle: Option<(String, Tree<T>)>,
/// Selected item idex
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Suggested change
/// Selected item idex
/// Selected item index

selected: usize,

@@ -878,96 +904,42 @@ impl<T: TreeItem> TreeView<T> {

impl<T: TreeItem + Clone> TreeView<T> {
pub fn filter(&mut self, s: &str, cx: &mut Context, params: &mut T::Params) {
todo!()
// fn filter_recursion<T>(
// elems: &Vec<Tree<T>>,
// mut index: usize,
// s: &str,
// cx: &mut Context,
// params: &mut T::Params,
// ) -> (Vec<Tree<T>>, usize)
// where
// T: TreeItem + Clone,
// {
// let mut retain = vec![];
// let elem = &elems[index];
// loop {
// let child = match elems.get(index + 1) {
// Some(child) if child.item.is_child(&elem.item) => child,
// _ => break,
// };
// index += 1;
// let next = elems.get(index + 1);
// if next.map_or(false, |n| n.item.is_child(&child.item)) {
// let (sub_retain, current_index) = filter_recursion(elems, index, s, cx, params);
// retain.extend(sub_retain);
// index = current_index;
// } else if child.item.filter(s) {
// retain.push(child.clone());
// }
// }
// if !retain.is_empty() || elem.item.filter(s) {
// retain.insert(0, elem.clone());
// }
// (retain, index)
// }

// if s.is_empty() {
// if let Some((_, recycle)) = self.recycle.take() {
// // self.tree = recycle;
// self.restore_view();
// return;
// }
// }

// let mut retain = vec![];
// let mut index = 0;
// let items = match &self.recycle {
// Some((pre, _)) if pre == s => return,
// Some((pre, recycle)) if pre.contains(s) => recycle,
// _ => &self.tree,
// };
// while let Some(elem) = items.get(index) {
// let next = items.get(index + 1);
// if next.map_or(false, |n| n.item.is_child(&elem.item)) {
// let (sub_items, current_index) = filter_recursion(items, index, s, cx, params);
// index = current_index;
// retain.extend(sub_items);
// } else if elem.item.filter(s) {
// retain.push(elem.clone())
// }
// index += 1;
// }

// if retain.is_empty() {
// if let Some((_, recycle)) = self.recycle.take() {
// self.tree = recycle;
// self.restore_view();
// }
// return;
// }
if s.is_empty() {
self.restore_recycle();
return;
}

// let recycle = std::mem::replace(&mut self.tree, retain);
// if let Some(r) = self.recycle.as_mut() {
// r.0 = s.into()
// } else {
// self.recycle = Some((s.into(), recycle));
// self.save_view();
// }
let new_tree = Tree::filter(&self.tree, &|item: &T| {
item.text_string()
.to_lowercase()
.contains(&s.to_lowercase())
})
.unwrap_or_else(|| Tree {
item: self.tree.item.clone(),
children: vec![],
..self.tree.clone()
});
let recycle = std::mem::replace(&mut self.tree, new_tree);
if let Some(r) = self.recycle.as_mut() {
r.0 = s.into()
} else {
self.recycle = Some((s.into(), recycle));
self.save_view();
}

// self.selected = self.find(0, false, |elem| elem.item.filter(s)).unwrap_or(0);
// self.winline = self.selected;
self.selected = 0;
self.winline = 0
}

pub fn clean_recycle(&mut self) {
self.recycle = None;
}

pub fn restore_recycle(&mut self) {
todo!();
// if let Some((_, recycle)) = self.recycle.take() {
// self.tree = recycle;
// }
if let Some((_, recycle)) = self.recycle.take() {
self.tree = recycle;
}
self.restore_view();
}
}

@@ -1196,4 +1168,43 @@ mod test_tree {

assert_eq!(result, Some(3));
}

#[test]
fn test_filter() {
let tree = Tree::new(
".cargo",
vec![
Tree::new("spam", vec![Tree::new("Cargo.toml", vec![])]),
Tree::new("Cargo.toml", vec![Tree::new("pam", vec![])]),
Tree::new("hello", vec![]),
],
);

let result = Tree::filter(&tree, &|item| item.to_lowercase().contains("cargo"));
assert_eq!(
result,
Some(Tree::new(
".cargo",
vec![
Tree::new("spam", vec![Tree::new("Cargo.toml", vec![])]),
Tree::new("Cargo.toml", vec![]),
],
))
);

let result = Tree::filter(&tree, &|item| item.to_lowercase().contains("pam"));
assert_eq!(
result,
Some(Tree::new(
".cargo",
vec![
Tree::new("spam", vec![]),
Tree::new("Cargo.toml", vec![Tree::new("pam", vec![])]),
],
))
);

let result = Tree::filter(&tree, &|item| item.to_lowercase().contains("helix"));
assert_eq!(result, None)
}
}