use anyhow::Result; use async_recursion::async_recursion; use collections::HashSet; use futures::{StreamExt as _, stream::FuturesUnordered}; use gpui::{AppContext as _, AsyncWindowContext, Axis, Entity, Task, WeakEntity}; use project::{Project, ProjectPath}; use serde::{Deserialize, Serialize}; use std::{ path::{Path, PathBuf}, sync::Arc, }; use ui::{App, Context, Pixels, Window}; use util::ResultExt as _; use db::{define_connection, query, sqlez::statement::Statement, sqlez_macros::sql}; use workspace::{ ItemHandle, ItemId, Member, Pane, PaneAxis, PaneGroup, SerializableItem as _, Workspace, WorkspaceDb, WorkspaceId, }; use crate::{ TerminalView, default_working_directory, terminal_panel::{TerminalPanel, new_terminal_pane}, }; pub(crate) fn serialize_pane_group( pane_group: &PaneGroup, active_pane: &Entity, cx: &mut App, ) -> SerializedPaneGroup { build_serialized_pane_group(&pane_group.root, active_pane, cx) } fn build_serialized_pane_group( pane_group: &Member, active_pane: &Entity, cx: &mut App, ) -> SerializedPaneGroup { match pane_group { Member::Axis(PaneAxis { axis, members, flexes, bounding_boxes: _, }) => SerializedPaneGroup::Group { axis: SerializedAxis(*axis), children: members .iter() .map(|member| build_serialized_pane_group(member, active_pane, cx)) .collect::>(), flexes: Some(flexes.lock().clone()), }, Member::Pane(pane_handle) => { SerializedPaneGroup::Pane(serialize_pane(pane_handle, pane_handle == active_pane, cx)) } } } fn serialize_pane(pane: &Entity, active: bool, cx: &mut App) -> SerializedPane { let mut items_to_serialize = HashSet::default(); let pane = pane.read(cx); let children = pane .items() .filter_map(|item| { let terminal_view = item.act_as::(cx)?; if terminal_view.read(cx).terminal().read(cx).task().is_some() { None } else { let id = item.item_id().as_u64(); items_to_serialize.insert(id); Some(id) } }) .collect::>(); let active_item = pane .active_item() .map(|item| item.item_id().as_u64()) .filter(|active_id| items_to_serialize.contains(active_id)); let pinned_count = pane.pinned_count(); SerializedPane { active, children, active_item, pinned_count, } } pub(crate) fn deserialize_terminal_panel( workspace: WeakEntity, project: Entity, database_id: WorkspaceId, serialized_panel: SerializedTerminalPanel, window: &mut Window, cx: &mut App, ) -> Task>> { window.spawn(cx, async move |cx| { let terminal_panel = workspace.update_in(cx, |workspace, window, cx| { cx.new(|cx| { let mut panel = TerminalPanel::new(workspace, window, cx); panel.height = serialized_panel.height.map(|h| h.round()); panel.width = serialized_panel.width.map(|w| w.round()); panel }) })?; match &serialized_panel.items { SerializedItems::NoSplits(item_ids) => { let items = deserialize_terminal_views( database_id, project, workspace, item_ids.as_slice(), cx, ) .await; let active_item = serialized_panel.active_item_id; terminal_panel.update_in(cx, |terminal_panel, window, cx| { terminal_panel.active_pane.update(cx, |pane, cx| { populate_pane_items(pane, items, active_item, window, cx); }); })?; } SerializedItems::WithSplits(serialized_pane_group) => { let center_pane = deserialize_pane_group( workspace, project, terminal_panel.clone(), database_id, serialized_pane_group, cx, ) .await; if let Some((center_group, active_pane)) = center_pane { terminal_panel.update(cx, |terminal_panel, _| { terminal_panel.center = PaneGroup::with_root(center_group); terminal_panel.active_pane = active_pane.unwrap_or_else(|| terminal_panel.center.first_pane()); })?; } } } Ok(terminal_panel) }) } fn populate_pane_items( pane: &mut Pane, items: Vec>, active_item: Option, window: &mut Window, cx: &mut Context, ) { let mut item_index = pane.items_len(); let mut active_item_index = None; for item in items { if Some(item.item_id().as_u64()) == active_item { active_item_index = Some(item_index); } pane.add_item(Box::new(item), false, false, None, window, cx); item_index += 1; } if let Some(index) = active_item_index { pane.activate_item(index, false, false, window, cx); } } #[async_recursion(?Send)] async fn deserialize_pane_group( workspace: WeakEntity, project: Entity, panel: Entity, workspace_id: WorkspaceId, serialized: &SerializedPaneGroup, cx: &mut AsyncWindowContext, ) -> Option<(Member, Option>)> { match serialized { SerializedPaneGroup::Group { axis, flexes, children, } => { let mut current_active_pane = None; let mut members = Vec::new(); for child in children { if let Some((new_member, active_pane)) = deserialize_pane_group( workspace.clone(), project.clone(), panel.clone(), workspace_id, child, cx, ) .await { members.push(new_member); current_active_pane = current_active_pane.or(active_pane); } } if members.is_empty() { return None; } if members.len() == 1 { return Some((members.remove(0), current_active_pane)); } Some(( Member::Axis(PaneAxis::load(axis.0, members, flexes.clone())), current_active_pane, )) } SerializedPaneGroup::Pane(serialized_pane) => { let active = serialized_pane.active; let new_items = deserialize_terminal_views( workspace_id, project.clone(), workspace.clone(), serialized_pane.children.as_slice(), cx, ) .await; let pane = panel .update_in(cx, |terminal_panel, window, cx| { new_terminal_pane( workspace.clone(), project.clone(), terminal_panel.active_pane.read(cx).is_zoomed(), window, cx, ) }) .log_err()?; let active_item = serialized_pane.active_item; let pinned_count = serialized_pane.pinned_count; let terminal = pane .update_in(cx, |pane, window, cx| { populate_pane_items(pane, new_items, active_item, window, cx); pane.set_pinned_count(pinned_count); // Avoid blank panes in splits if pane.items_len() == 0 { let working_directory = workspace .update(cx, |workspace, cx| default_working_directory(workspace, cx)) .ok() .flatten(); let p = workspace .update(cx, |workspace, cx| { let worktree = workspace.worktrees(cx).next()?.read(cx); worktree.root_dir()?; Some(ProjectPath { worktree_id: worktree.id(), path: Arc::from(Path::new("")), }) }) .ok() .flatten(); let terminal = project.update(cx, |project, cx| { project.create_terminal_shell(working_directory, cx, p) }); Some(Some(terminal)) } else { Some(None) } }) .ok() .flatten()?; if let Some(terminal) = terminal { let terminal = terminal.await.ok()?; pane.update_in(cx, |pane, window, cx| { let terminal_view = Box::new(cx.new(|cx| { TerminalView::new( terminal, workspace.clone(), Some(workspace_id), project.downgrade(), window, cx, ) })); pane.add_item(terminal_view, true, false, None, window, cx); }) .ok()?; } Some((Member::Pane(pane.clone()), active.then_some(pane))) } } } async fn deserialize_terminal_views( workspace_id: WorkspaceId, project: Entity, workspace: WeakEntity, item_ids: &[u64], cx: &mut AsyncWindowContext, ) -> Vec> { let mut items = Vec::with_capacity(item_ids.len()); let mut deserialized_items = item_ids .iter() .map(|item_id| { cx.update(|window, cx| { TerminalView::deserialize( project.clone(), workspace.clone(), workspace_id, *item_id, window, cx, ) }) .unwrap_or_else(|e| Task::ready(Err(e.context("no window present")))) }) .collect::>(); while let Some(item) = deserialized_items.next().await { if let Some(item) = item.log_err() { items.push(item); } } items } #[derive(Debug, Serialize, Deserialize)] pub(crate) struct SerializedTerminalPanel { pub items: SerializedItems, // A deprecated field, kept for backwards compatibility for the code before terminal splits were introduced. pub active_item_id: Option, pub width: Option, pub height: Option, } #[derive(Debug, Serialize, Deserialize)] #[serde(untagged)] pub(crate) enum SerializedItems { // The data stored before terminal splits were introduced. NoSplits(Vec), WithSplits(SerializedPaneGroup), } #[derive(Debug, Serialize, Deserialize)] pub(crate) enum SerializedPaneGroup { Pane(SerializedPane), Group { axis: SerializedAxis, flexes: Option>, children: Vec, }, } #[derive(Debug, Serialize, Deserialize)] pub(crate) struct SerializedPane { pub active: bool, pub children: Vec, pub active_item: Option, #[serde(default)] pub pinned_count: usize, } #[derive(Debug)] pub(crate) struct SerializedAxis(pub Axis); impl Serialize for SerializedAxis { fn serialize(&self, serializer: S) -> Result where S: serde::Serializer, { match self.0 { Axis::Horizontal => serializer.serialize_str("horizontal"), Axis::Vertical => serializer.serialize_str("vertical"), } } } impl<'de> Deserialize<'de> for SerializedAxis { fn deserialize(deserializer: D) -> Result where D: serde::Deserializer<'de>, { let s = String::deserialize(deserializer)?; match s.as_str() { "horizontal" => Ok(SerializedAxis(Axis::Horizontal)), "vertical" => Ok(SerializedAxis(Axis::Vertical)), invalid => Err(serde::de::Error::custom(format!( "Invalid axis value: '{invalid}'" ))), } } } define_connection! { pub static ref TERMINAL_DB: TerminalDb = &[sql!( CREATE TABLE terminals ( workspace_id INTEGER, item_id INTEGER UNIQUE, working_directory BLOB, PRIMARY KEY(workspace_id, item_id), FOREIGN KEY(workspace_id) REFERENCES workspaces(workspace_id) ON DELETE CASCADE ) STRICT; ), // Remove the unique constraint on the item_id table // SQLite doesn't have a way of doing this automatically, so // we have to do this silly copying. sql!( CREATE TABLE terminals2 ( workspace_id INTEGER, item_id INTEGER, working_directory BLOB, PRIMARY KEY(workspace_id, item_id), FOREIGN KEY(workspace_id) REFERENCES workspaces(workspace_id) ON DELETE CASCADE ) STRICT; INSERT INTO terminals2 (workspace_id, item_id, working_directory) SELECT workspace_id, item_id, working_directory FROM terminals; DROP TABLE terminals; ALTER TABLE terminals2 RENAME TO terminals; ), sql! ( ALTER TABLE terminals ADD COLUMN working_directory_path TEXT; UPDATE terminals SET working_directory_path = CAST(working_directory AS TEXT); ), ]; } impl TerminalDb { query! { pub async fn update_workspace_id( new_id: WorkspaceId, old_id: WorkspaceId, item_id: ItemId ) -> Result<()> { UPDATE terminals SET workspace_id = ? WHERE workspace_id = ? AND item_id = ? } } pub async fn save_working_directory( &self, item_id: ItemId, workspace_id: WorkspaceId, working_directory: PathBuf, ) -> Result<()> { log::debug!( "Saving working directory {working_directory:?} for item {item_id} in workspace {workspace_id:?}" ); let query = "INSERT INTO terminals(item_id, workspace_id, working_directory, working_directory_path) VALUES (?1, ?2, ?3, ?4) ON CONFLICT DO UPDATE SET item_id = ?1, workspace_id = ?2, working_directory = ?3, working_directory_path = ?4" ; self.write(move |conn| { let mut statement = Statement::prepare(conn, query)?; let mut next_index = statement.bind(&item_id, 1)?; next_index = statement.bind(&workspace_id, next_index)?; next_index = statement.bind(&working_directory, next_index)?; statement.bind(&working_directory.to_string_lossy().to_string(), next_index)?; statement.exec() }) .await } query! { pub fn get_working_directory(item_id: ItemId, workspace_id: WorkspaceId) -> Result> { SELECT working_directory FROM terminals WHERE item_id = ? AND workspace_id = ? } } }