pub mod running; use crate::{StackTraceView, debugger_panel::DebugPanel, persistence::SerializedLayout}; use dap::client::SessionId; use gpui::{ App, Axis, Entity, EventEmitter, FocusHandle, Focusable, Subscription, Task, WeakEntity, }; use project::Project; use project::debugger::session::Session; use project::worktree_store::WorktreeStore; use rpc::proto; use running::RunningState; use std::{cell::OnceCell, sync::OnceLock}; use ui::{Indicator, prelude::*}; use workspace::{ CollaboratorId, FollowableItem, ViewId, Workspace, item::{self, Item}, }; pub struct DebugSession { remote_id: Option, running_state: Entity, label: OnceLock, stack_trace_view: OnceCell>, _debug_panel: WeakEntity, _worktree_store: WeakEntity, workspace: WeakEntity, _subscriptions: [Subscription; 1], } #[derive(Debug)] pub enum DebugPanelItemEvent { Close, Stopped { go_to_stack_frame: bool }, } impl DebugSession { pub(crate) fn running( project: Entity, workspace: WeakEntity, session: Entity, _debug_panel: WeakEntity, serialized_layout: Option, dock_axis: Axis, window: &mut Window, cx: &mut App, ) -> Entity { let running_state = cx.new(|cx| { RunningState::new( session.clone(), project.clone(), workspace.clone(), serialized_layout, dock_axis, window, cx, ) }); cx.new(|cx| Self { _subscriptions: [cx.subscribe(&running_state, |_, _, _, cx| { cx.notify(); })], remote_id: None, running_state, label: OnceLock::new(), _debug_panel, stack_trace_view: OnceCell::new(), _worktree_store: project.read(cx).worktree_store().downgrade(), workspace, }) } pub(crate) fn session_id(&self, cx: &App) -> SessionId { self.running_state.read(cx).session_id() } pub(crate) fn stack_trace_view( &mut self, project: &Entity, window: &mut Window, cx: &mut Context, ) -> &Entity { let workspace = self.workspace.clone(); let running_state = self.running_state.clone(); self.stack_trace_view.get_or_init(|| { let stackframe_list = running_state.read(cx).stack_frame_list().clone(); let stack_frame_view = cx.new(|cx| { StackTraceView::new( workspace.clone(), project.clone(), stackframe_list, window, cx, ) }); stack_frame_view }) } pub fn session(&self, cx: &App) -> Entity { self.running_state.read(cx).session().clone() } pub(crate) fn shutdown(&mut self, cx: &mut Context) { self.running_state .update(cx, |state, cx| state.shutdown(cx)); } pub(crate) fn label(&self, cx: &App) -> SharedString { if let Some(label) = self.label.get() { return label.clone(); } let session = self.running_state.read(cx).session(); self.label .get_or_init(|| session.read(cx).label()) .to_owned() } pub(crate) fn running_state(&self) -> &Entity { &self.running_state } pub(crate) fn label_element(&self, cx: &App) -> AnyElement { let label = self.label(cx); let is_terminated = self .running_state .read(cx) .session() .read(cx) .is_terminated(); let icon = { if is_terminated { Some(Indicator::dot().color(Color::Error)) } else { match self .running_state .read(cx) .thread_status(cx) .unwrap_or_default() { project::debugger::session::ThreadStatus::Stopped => { Some(Indicator::dot().color(Color::Conflict)) } _ => Some(Indicator::dot().color(Color::Success)), } } }; h_flex() .gap_2() .when_some(icon, |this, indicator| this.child(indicator)) .justify_between() .child( Label::new(label) .size(LabelSize::Small) .when(is_terminated, |this| this.strikethrough()), ) .into_any_element() } } impl EventEmitter for DebugSession {} impl Focusable for DebugSession { fn focus_handle(&self, cx: &App) -> FocusHandle { self.running_state.focus_handle(cx) } } impl Item for DebugSession { type Event = DebugPanelItemEvent; fn tab_content_text(&self, _detail: usize, _cx: &App) -> SharedString { "Debugger".into() } } impl FollowableItem for DebugSession { fn remote_id(&self) -> Option { self.remote_id } fn to_state_proto(&self, _window: &Window, _cx: &App) -> Option { None } fn from_state_proto( _workspace: Entity, _remote_id: ViewId, _state: &mut Option, _window: &mut Window, _cx: &mut App, ) -> Option>>> { None } fn add_event_to_update_proto( &self, _event: &Self::Event, _update: &mut Option, _window: &Window, _cx: &App, ) -> bool { // update.get_or_insert_with(|| proto::update_view::Variant::DebugPanel(Default::default())); true } fn apply_update_proto( &mut self, _project: &Entity, _message: proto::update_view::Variant, _window: &mut Window, _cx: &mut Context, ) -> gpui::Task> { Task::ready(Ok(())) } fn set_leader_id( &mut self, _leader_id: Option, _window: &mut Window, _cx: &mut Context, ) { } fn to_follow_event(_event: &Self::Event) -> Option { None } fn dedup(&self, existing: &Self, _window: &Window, cx: &App) -> Option { if existing.session_id(cx) == self.session_id(cx) { Some(item::Dedup::KeepExisting) } else { None } } fn is_project_item(&self, _window: &Window, _cx: &App) -> bool { true } } impl Render for DebugSession { fn render(&mut self, window: &mut Window, cx: &mut Context) -> impl IntoElement { self.running_state .update(cx, |this, cx| this.render(window, cx).into_any_element()) } }