Skip to content

Commit e11e7df

Browse files
Restore editor state on reopen (#27672)
Closes #11626 Part of #12853 `"restore_on_file_reopen": true` in workspace settings can now be used to enable and disable editor data between file reopens in the same pane: https://github.com/user-attachments/assets/8d938ee1-d854-42a8-bbc3-2a4e4d7d5933 The settings are generic and panes' data store can be extended for further entities, beyond editors. --------------- Impl details: Currently, the project entry IDs seem to be stable across file reopens, unlike BufferIds, so those were used. Originally, the DB data was considered over in-memory one as editors serialize their state anyway, but managing and exposing PaneIds out of the DB is quite tedious and joining the DB data otherwise is not possible. Release Notes: - Started to restore editor state on reopen
1 parent bbd1e62 commit e11e7df

File tree

13 files changed

+611
-48
lines changed

13 files changed

+611
-48
lines changed

assets/settings/default.json

+9
Original file line numberDiff line numberDiff line change
@@ -115,6 +115,15 @@
115115
"confirm_quit": false,
116116
// Whether to restore last closed project when fresh Zed instance is opened.
117117
"restore_on_startup": "last_session",
118+
// Whether to attempt to restore previous file's state when opening it again.
119+
// The state is stored per pane.
120+
// When disabled, defaults are applied instead of the state restoration.
121+
//
122+
// E.g. for editors, selections, folds and scroll positions are restored, if the same file is closed and, later, opened again in the same pane.
123+
// When disabled, a single selection in the very beginning of the file, zero scroll position and no folds state is used as a default.
124+
//
125+
// Default: true
126+
"restore_on_file_reopen": true,
118127
// Size of the drop target in the editor.
119128
"drop_target_size": 0.2,
120129
// Whether the window should be closed when using 'close active item' on a window with no tabs.

crates/editor/src/editor.rs

+57-28
Original file line numberDiff line numberDiff line change
@@ -1597,6 +1597,17 @@ impl Editor {
15971597
}
15981598
this.tasks_update_task = Some(this.refresh_runnables(window, cx));
15991599
this._subscriptions.extend(project_subscriptions);
1600+
this._subscriptions
1601+
.push(cx.subscribe_self(|editor, e: &EditorEvent, cx| {
1602+
if let EditorEvent::SelectionsChanged { local } = e {
1603+
if *local {
1604+
let new_anchor = editor.scroll_manager.anchor();
1605+
editor.update_restoration_data(cx, move |data| {
1606+
data.scroll_anchor = new_anchor;
1607+
});
1608+
}
1609+
}
1610+
}));
16001611

16011612
this.end_selection(window, cx);
16021613
this.scroll_manager.show_scrollbars(window, cx);
@@ -2317,18 +2328,24 @@ impl Editor {
23172328
if selections.len() == 1 {
23182329
cx.emit(SearchEvent::ActiveMatchChanged)
23192330
}
2320-
if local
2321-
&& self.is_singleton(cx)
2322-
&& WorkspaceSettings::get(None, cx).restore_on_startup != RestoreOnStartupBehavior::None
2323-
{
2324-
if let Some(workspace_id) = self.workspace.as_ref().and_then(|workspace| workspace.1) {
2325-
let background_executor = cx.background_executor().clone();
2326-
let editor_id = cx.entity().entity_id().as_u64() as ItemId;
2327-
let snapshot = self.buffer().read(cx).snapshot(cx);
2328-
let selections = selections.clone();
2329-
self.serialize_selections = cx.background_spawn(async move {
2331+
if local && self.is_singleton(cx) {
2332+
let inmemory_selections = selections.iter().map(|s| s.range()).collect();
2333+
self.update_restoration_data(cx, |data| {
2334+
data.selections = inmemory_selections;
2335+
});
2336+
2337+
if WorkspaceSettings::get(None, cx).restore_on_startup != RestoreOnStartupBehavior::None
2338+
{
2339+
if let Some(workspace_id) =
2340+
self.workspace.as_ref().and_then(|workspace| workspace.1)
2341+
{
2342+
let snapshot = self.buffer().read(cx).snapshot(cx);
2343+
let selections = selections.clone();
2344+
let background_executor = cx.background_executor().clone();
2345+
let editor_id = cx.entity().entity_id().as_u64() as ItemId;
2346+
self.serialize_selections = cx.background_spawn(async move {
23302347
background_executor.timer(SERIALIZATION_THROTTLE_TIME).await;
2331-
let selections = selections
2348+
let db_selections = selections
23322349
.iter()
23332350
.map(|selection| {
23342351
(
@@ -2338,11 +2355,12 @@ impl Editor {
23382355
})
23392356
.collect();
23402357

2341-
DB.save_editor_selections(editor_id, workspace_id, selections)
2358+
DB.save_editor_selections(editor_id, workspace_id, db_selections)
23422359
.await
23432360
.with_context(|| format!("persisting editor selections for editor {editor_id}, workspace {workspace_id:?}"))
23442361
.log_err();
23452362
});
2363+
}
23462364
}
23472365
}
23482366

@@ -2356,13 +2374,24 @@ impl Editor {
23562374
return;
23572375
}
23582376

2377+
let snapshot = self.buffer().read(cx).snapshot(cx);
2378+
let inmemory_folds = self.display_map.update(cx, |display_map, cx| {
2379+
display_map
2380+
.snapshot(cx)
2381+
.folds_in_range(0..snapshot.len())
2382+
.map(|fold| fold.range.deref().clone())
2383+
.collect()
2384+
});
2385+
self.update_restoration_data(cx, |data| {
2386+
data.folds = inmemory_folds;
2387+
});
2388+
23592389
let Some(workspace_id) = self.workspace.as_ref().and_then(|workspace| workspace.1) else {
23602390
return;
23612391
};
23622392
let background_executor = cx.background_executor().clone();
23632393
let editor_id = cx.entity().entity_id().as_u64() as ItemId;
2364-
let snapshot = self.buffer().read(cx).snapshot(cx);
2365-
let folds = self.display_map.update(cx, |display_map, cx| {
2394+
let db_folds = self.display_map.update(cx, |display_map, cx| {
23662395
display_map
23672396
.snapshot(cx)
23682397
.folds_in_range(0..snapshot.len())
@@ -2376,7 +2405,7 @@ impl Editor {
23762405
});
23772406
self.serialize_folds = cx.background_spawn(async move {
23782407
background_executor.timer(SERIALIZATION_THROTTLE_TIME).await;
2379-
DB.save_editor_folds(editor_id, workspace_id, folds)
2408+
DB.save_editor_folds(editor_id, workspace_id, db_folds)
23802409
.await
23812410
.with_context(|| format!("persisting editor folds for editor {editor_id}, workspace {workspace_id:?}"))
23822411
.log_err();
@@ -17454,19 +17483,6 @@ impl Editor {
1745417483
{
1745517484
let buffer_snapshot = OnceCell::new();
1745617485

17457-
if let Some(selections) = DB.get_editor_selections(item_id, workspace_id).log_err() {
17458-
if !selections.is_empty() {
17459-
let snapshot =
17460-
buffer_snapshot.get_or_init(|| self.buffer.read(cx).snapshot(cx));
17461-
self.change_selections(None, window, cx, |s| {
17462-
s.select_ranges(selections.into_iter().map(|(start, end)| {
17463-
snapshot.clip_offset(start, Bias::Left)
17464-
..snapshot.clip_offset(end, Bias::Right)
17465-
}));
17466-
});
17467-
}
17468-
};
17469-
1747017486
if let Some(folds) = DB.get_editor_folds(item_id, workspace_id).log_err() {
1747117487
if !folds.is_empty() {
1747217488
let snapshot =
@@ -17485,6 +17501,19 @@ impl Editor {
1748517501
);
1748617502
}
1748717503
}
17504+
17505+
if let Some(selections) = DB.get_editor_selections(item_id, workspace_id).log_err() {
17506+
if !selections.is_empty() {
17507+
let snapshot =
17508+
buffer_snapshot.get_or_init(|| self.buffer.read(cx).snapshot(cx));
17509+
self.change_selections(None, window, cx, |s| {
17510+
s.select_ranges(selections.into_iter().map(|(start, end)| {
17511+
snapshot.clip_offset(start, Bias::Left)
17512+
..snapshot.clip_offset(end, Bias::Right)
17513+
}));
17514+
});
17515+
}
17516+
};
1748817517
}
1748917518

1749017519
self.read_scroll_position_from_db(item_id, workspace_id, window, cx);

0 commit comments

Comments
 (0)