file_finder: Display duplicated file in file finder history (#41917)

Closes #41850

When digging into this I figured out that basically what was going on is
in the history of the file finder it doesn't update the name of the file
duplicated because when you duplicate a file it's named automatically
with `filename copy` and so this filename was added to the history but
not updated so once you wanted to go back into this file it was not part
of file finder displayed history anymore because this file doesn't exist
anymore but the entity id remains the same.
I was also to reproduce this bug when just renaming a file.

Release Notes:

- Fixed: Display duplicated file in file finder history

---------

Signed-off-by: Benjamin <5719034+bnjjj@users.noreply.github.com>
Co-authored-by: Conrad Irwin <conrad.irwin@gmail.com>
Co-authored-by: Lukas Wirth <me@lukaswirth.dev>
This commit is contained in:
Coenen Benjamin
2025-11-06 08:06:45 +01:00
committed by GitHub
parent 58cec41932
commit 2b6cf31ace
5 changed files with 431 additions and 5 deletions

View File

@@ -1905,7 +1905,7 @@ impl Editor {
}
}
project::Event::EntryRenamed(transaction) => {
project::Event::EntryRenamed(transaction, project_path, abs_path) => {
let Some(workspace) = editor.workspace() else {
return;
};
@@ -1913,7 +1913,23 @@ impl Editor {
else {
return;
};
if active_editor.entity_id() == cx.entity_id() {
let entity_id = cx.entity_id();
workspace.update(cx, |this, cx| {
this.panes_mut()
.iter_mut()
.filter(|pane| pane.entity_id() != entity_id)
.for_each(|p| {
p.update(cx, |pane, _| {
pane.nav_history_mut().rename_item(
entity_id,
project_path.clone(),
abs_path.clone().into(),
);
})
});
});
let edited_buffers_already_open = {
let other_editors: Vec<Entity<Editor>> = workspace
.read(cx)
@@ -1936,7 +1952,6 @@ impl Editor {
})
})
};
if !edited_buffers_already_open {
let workspace = workspace.downgrade();
let transaction = transaction.clone();

View File

@@ -344,7 +344,7 @@ pub enum Event {
RevealInProjectPanel(ProjectEntryId),
SnippetEdit(BufferId, Vec<(lsp::Range, Snippet)>),
ExpandedAllForEntry(WorktreeId, ProjectEntryId),
EntryRenamed(ProjectTransaction),
EntryRenamed(ProjectTransaction, ProjectPath, PathBuf),
AgentLocationChanged,
}
@@ -2248,7 +2248,11 @@ impl Project {
project
.update(cx, |_, cx| {
cx.emit(Event::EntryRenamed(transaction));
cx.emit(Event::EntryRenamed(
transaction,
new_path.clone(),
new_abs_path.clone(),
));
})
.ok();

View File

@@ -2199,6 +2199,396 @@ async fn test_create_duplicate_items(cx: &mut gpui::TestAppContext) {
);
}
// NOTE: This test is skipped on Windows, because on Windows,
// when it triggers the lsp store it converts `/src/test/first copy.txt` into an uri
// but it fails with message `"/src\\test\\first copy.txt" is not parseable as an URI`
#[gpui::test]
#[cfg_attr(target_os = "windows", ignore)]
async fn test_create_duplicate_items_and_check_history(cx: &mut gpui::TestAppContext) {
init_test_with_editor(cx);
let fs = FakeFs::new(cx.executor());
fs.insert_tree(
"/src",
json!({
"test": {
"first.txt": "// First Txt file",
"second.txt": "// Second Txt file",
"third.txt": "// Third Txt file",
}
}),
)
.await;
let project = Project::test(fs.clone(), ["/src".as_ref()], cx).await;
let workspace = cx.add_window(|window, cx| Workspace::test_new(project.clone(), window, cx));
let cx = &mut VisualTestContext::from_window(*workspace, cx);
let panel = workspace
.update(cx, |workspace, window, cx| {
let panel = ProjectPanel::new(workspace, window, cx);
workspace.add_panel(panel.clone(), window, cx);
panel
})
.unwrap();
cx.run_until_parked();
select_path(&panel, "src", cx);
panel.update_in(cx, |panel, window, cx| panel.confirm(&Confirm, window, cx));
cx.executor().run_until_parked();
assert_eq!(
visible_entries_as_strings(&panel, 0..10, cx),
&[
//
"v src <== selected",
" > test"
]
);
panel.update_in(cx, |panel, window, cx| {
panel.new_directory(&NewDirectory, window, cx)
});
cx.run_until_parked();
panel.update_in(cx, |panel, window, cx| {
assert!(panel.filename_editor.read(cx).is_focused(window));
});
cx.executor().run_until_parked();
assert_eq!(
visible_entries_as_strings(&panel, 0..10, cx),
&[
//
"v src",
" > [EDITOR: ''] <== selected",
" > test"
]
);
panel.update_in(cx, |panel, window, cx| {
panel
.filename_editor
.update(cx, |editor, cx| editor.set_text("test", window, cx));
assert!(
panel.confirm_edit(true, window, cx).is_none(),
"Should not allow to confirm on conflicting new directory name"
);
});
cx.executor().run_until_parked();
panel.update_in(cx, |panel, window, cx| {
assert!(
panel.state.edit_state.is_some(),
"Edit state should not be None after conflicting new directory name"
);
panel.cancel(&menu::Cancel, window, cx);
panel.update_visible_entries(None, false, false, window, cx);
});
cx.run_until_parked();
assert_eq!(
visible_entries_as_strings(&panel, 0..10, cx),
&[
//
"v src <== selected",
" > test"
],
"File list should be unchanged after failed folder create confirmation"
);
select_path(&panel, "src/test", cx);
panel.update_in(cx, |panel, window, cx| panel.confirm(&Confirm, window, cx));
cx.executor().run_until_parked();
assert_eq!(
visible_entries_as_strings(&panel, 0..10, cx),
&[
//
"v src",
" > test <== selected"
]
);
panel.update_in(cx, |panel, window, cx| panel.new_file(&NewFile, window, cx));
cx.run_until_parked();
panel.update_in(cx, |panel, window, cx| {
assert!(panel.filename_editor.read(cx).is_focused(window));
});
assert_eq!(
visible_entries_as_strings(&panel, 0..10, cx),
&[
"v src",
" v test",
" [EDITOR: ''] <== selected",
" first.txt",
" second.txt",
" third.txt"
]
);
panel.update_in(cx, |panel, window, cx| {
panel
.filename_editor
.update(cx, |editor, cx| editor.set_text("first.txt", window, cx));
assert!(
panel.confirm_edit(true, window, cx).is_none(),
"Should not allow to confirm on conflicting new file name"
);
});
cx.executor().run_until_parked();
panel.update_in(cx, |panel, window, cx| {
assert!(
panel.state.edit_state.is_some(),
"Edit state should not be None after conflicting new file name"
);
panel.cancel(&menu::Cancel, window, cx);
panel.update_visible_entries(None, false, false, window, cx);
});
cx.run_until_parked();
assert_eq!(
visible_entries_as_strings(&panel, 0..10, cx),
&[
"v src",
" v test <== selected",
" first.txt",
" second.txt",
" third.txt"
],
"File list should be unchanged after failed file create confirmation"
);
select_path(&panel, "src/test/first.txt", cx);
panel.update_in(cx, |panel, window, cx| panel.confirm(&Confirm, window, cx));
cx.executor().run_until_parked();
assert_eq!(
visible_entries_as_strings(&panel, 0..10, cx),
&[
"v src",
" v test",
" first.txt <== selected",
" second.txt",
" third.txt"
],
);
panel.update_in(cx, |panel, window, cx| panel.rename(&Rename, window, cx));
panel.update_in(cx, |panel, window, cx| {
assert!(panel.filename_editor.read(cx).is_focused(window));
});
assert_eq!(
visible_entries_as_strings(&panel, 0..10, cx),
&[
"v src",
" v test",
" [EDITOR: 'first.txt'] <== selected",
" second.txt",
" third.txt"
]
);
panel.update_in(cx, |panel, window, cx| {
panel
.filename_editor
.update(cx, |editor, cx| editor.set_text("second.txt", window, cx));
assert!(
panel.confirm_edit(true, window, cx).is_none(),
"Should not allow to confirm on conflicting file rename"
)
});
cx.executor().run_until_parked();
panel.update_in(cx, |panel, window, cx| {
assert!(
panel.state.edit_state.is_some(),
"Edit state should not be None after conflicting file rename"
);
panel.cancel(&menu::Cancel, window, cx);
});
assert_eq!(
visible_entries_as_strings(&panel, 0..10, cx),
&[
"v src",
" v test",
" first.txt <== selected",
" second.txt",
" third.txt"
],
"File list should be unchanged after failed rename confirmation"
);
panel.update_in(cx, |panel, window, cx| panel.open(&Open, window, cx));
cx.executor().run_until_parked();
// Try to duplicate and check history
panel.update_in(cx, |panel, window, cx| {
panel.duplicate(&Duplicate, window, cx)
});
cx.executor().run_until_parked();
assert_eq!(
visible_entries_as_strings(&panel, 0..10, cx),
&[
"v src",
" v test",
" first.txt",
" [EDITOR: 'first copy.txt'] <== selected <== marked",
" second.txt",
" third.txt"
],
);
let confirm = panel.update_in(cx, |panel, window, cx| {
panel
.filename_editor
.update(cx, |editor, cx| editor.set_text("fourth.txt", window, cx));
panel.confirm_edit(true, window, cx).unwrap()
});
confirm.await.unwrap();
cx.executor().run_until_parked();
assert_eq!(
visible_entries_as_strings(&panel, 0..10, cx),
&[
"v src",
" v test",
" first.txt",
" fourth.txt <== selected",
" second.txt",
" third.txt"
],
"File list should be different after rename confirmation"
);
panel.update_in(cx, |panel, window, cx| {
panel.update_visible_entries(None, false, false, window, cx);
});
cx.executor().run_until_parked();
select_path(&panel, "src/test/first.txt", cx);
panel.update_in(cx, |panel, window, cx| panel.open(&Open, window, cx));
cx.executor().run_until_parked();
workspace
.read_with(cx, |this, cx| {
assert!(
this.recent_navigation_history_iter(cx)
.any(|(project_path, abs_path)| {
project_path.path == Arc::from(rel_path("test/fourth.txt"))
&& abs_path == Some(PathBuf::from(path!("/src/test/fourth.txt")))
})
);
})
.unwrap();
}
// NOTE: This test is skipped on Windows, because on Windows,
// when it triggers the lsp store it converts `/src/test/first.txt` into an uri
// but it fails with message `"/src\\test\\first.txt" is not parseable as an URI`
#[gpui::test]
#[cfg_attr(target_os = "windows", ignore)]
async fn test_rename_item_and_check_history(cx: &mut gpui::TestAppContext) {
init_test_with_editor(cx);
let fs = FakeFs::new(cx.executor());
fs.insert_tree(
"/src",
json!({
"test": {
"first.txt": "// First Txt file",
"second.txt": "// Second Txt file",
"third.txt": "// Third Txt file",
}
}),
)
.await;
let project = Project::test(fs.clone(), ["/src".as_ref()], cx).await;
let workspace = cx.add_window(|window, cx| Workspace::test_new(project.clone(), window, cx));
let cx = &mut VisualTestContext::from_window(*workspace, cx);
let panel = workspace
.update(cx, |workspace, window, cx| {
let panel = ProjectPanel::new(workspace, window, cx);
workspace.add_panel(panel.clone(), window, cx);
panel
})
.unwrap();
cx.run_until_parked();
select_path(&panel, "src", cx);
panel.update_in(cx, |panel, window, cx| panel.confirm(&Confirm, window, cx));
cx.executor().run_until_parked();
assert_eq!(
visible_entries_as_strings(&panel, 0..10, cx),
&[
//
"v src <== selected",
" > test"
]
);
select_path(&panel, "src/test", cx);
panel.update_in(cx, |panel, window, cx| panel.confirm(&Confirm, window, cx));
cx.executor().run_until_parked();
assert_eq!(
visible_entries_as_strings(&panel, 0..10, cx),
&[
//
"v src",
" > test <== selected"
]
);
panel.update_in(cx, |panel, window, cx| panel.new_file(&NewFile, window, cx));
cx.run_until_parked();
panel.update_in(cx, |panel, window, cx| {
assert!(panel.filename_editor.read(cx).is_focused(window));
});
select_path(&panel, "src/test/first.txt", cx);
panel.update_in(cx, |panel, window, cx| panel.open(&Open, window, cx));
cx.executor().run_until_parked();
panel.update_in(cx, |panel, window, cx| panel.rename(&Rename, window, cx));
cx.executor().run_until_parked();
assert_eq!(
visible_entries_as_strings(&panel, 0..10, cx),
&[
"v src",
" v test",
" [EDITOR: 'first.txt'] <== selected <== marked",
" second.txt",
" third.txt"
],
);
let confirm = panel.update_in(cx, |panel, window, cx| {
panel
.filename_editor
.update(cx, |editor, cx| editor.set_text("fourth.txt", window, cx));
panel.confirm_edit(true, window, cx).unwrap()
});
confirm.await.unwrap();
cx.executor().run_until_parked();
assert_eq!(
visible_entries_as_strings(&panel, 0..10, cx),
&[
"v src",
" v test",
" fourth.txt <== selected",
" second.txt",
" third.txt"
],
"File list should be different after rename confirmation"
);
panel.update_in(cx, |panel, window, cx| {
panel.update_visible_entries(None, false, false, window, cx);
});
cx.executor().run_until_parked();
select_path(&panel, "src/test/second.txt", cx);
panel.update_in(cx, |panel, window, cx| panel.open(&Open, window, cx));
cx.executor().run_until_parked();
workspace
.read_with(cx, |this, cx| {
assert!(
this.recent_navigation_history_iter(cx)
.any(|(project_path, abs_path)| {
project_path.path == Arc::from(rel_path("test/fourth.txt"))
&& abs_path == Some(PathBuf::from(path!("/src/test/fourth.txt")))
})
);
})
.unwrap();
}
#[gpui::test]
async fn test_select_git_entry(cx: &mut gpui::TestAppContext) {
init_test_with_editor(cx);

View File

@@ -1123,7 +1123,6 @@ impl Pane {
false
}
});
if let Some(existing_item_index) = existing_item_index {
// If the item already exists, move it to the desired destination and activate it
@@ -4132,6 +4131,20 @@ impl NavHistory {
.retain(|entry| entry.item.id() != item_id);
}
pub fn rename_item(
&mut self,
item_id: EntityId,
project_path: ProjectPath,
abs_path: Option<PathBuf>,
) {
let mut state = self.0.lock();
let path_for_item = state.paths_by_item.get_mut(&item_id);
if let Some(path_for_item) = path_for_item {
path_for_item.0 = project_path;
path_for_item.1 = abs_path;
}
}
pub fn path_for_item(&self, item_id: EntityId) -> Option<(ProjectPath, Option<PathBuf>)> {
self.0.lock().paths_by_item.get(&item_id).cloned()
}

View File

@@ -4325,6 +4325,10 @@ impl Workspace {
cx.emit(Event::PaneRemoved);
}
pub fn panes_mut(&mut self) -> &mut [Entity<Pane>] {
&mut self.panes
}
pub fn panes(&self) -> &[Entity<Pane>] {
&self.panes
}