
resolves #24655 resolves #23945 I haven't yet added a default binding for the new command. #27797 added `:ls` and `:buffers` which in my opinion should use the global searchable version given that that matches the vim semantics of those commands better than just showing the tabs in the local pane. There's also a question of what to do when you select a tab from another pane, should the focus jump to that pane or should that tab move to the currently focused pane? For now I've implemented the former. Release Notes: - Added `tab_switcher::ToggleAll` to search open tabs from all panes and focus the selected one. --------- Co-authored-by: Conrad Irwin <conrad.irwin@gmail.com>
1606 lines
60 KiB
Rust
1606 lines
60 KiB
Rust
use collections::{HashMap, VecDeque};
|
|
use copilot::Copilot;
|
|
use editor::{Editor, EditorEvent, actions::MoveToEnd, scroll::Autoscroll};
|
|
use futures::{StreamExt, channel::mpsc};
|
|
use gpui::{
|
|
AnyView, App, Context, Corner, Entity, EventEmitter, FocusHandle, Focusable, IntoElement,
|
|
ParentElement, Render, Styled, Subscription, WeakEntity, Window, actions, div,
|
|
};
|
|
use language::{LanguageServerId, language_settings::SoftWrap};
|
|
use lsp::{
|
|
IoKind, LanguageServer, LanguageServerName, MessageType, SetTraceParams, TraceValue,
|
|
notification::SetTrace,
|
|
};
|
|
use project::{Project, WorktreeId, search::SearchQuery};
|
|
use std::{any::TypeId, borrow::Cow, sync::Arc};
|
|
use ui::{Button, Checkbox, ContextMenu, Label, PopoverMenu, ToggleState, prelude::*};
|
|
use workspace::{
|
|
SplitDirection, ToolbarItemEvent, ToolbarItemLocation, ToolbarItemView, Workspace, WorkspaceId,
|
|
item::{Item, ItemHandle},
|
|
searchable::{Direction, SearchEvent, SearchableItem, SearchableItemHandle},
|
|
};
|
|
|
|
const SEND_LINE: &str = "// Send:";
|
|
const RECEIVE_LINE: &str = "// Receive:";
|
|
const MAX_STORED_LOG_ENTRIES: usize = 2000;
|
|
|
|
pub struct LogStore {
|
|
projects: HashMap<WeakEntity<Project>, ProjectState>,
|
|
language_servers: HashMap<LanguageServerId, LanguageServerState>,
|
|
copilot_log_subscription: Option<lsp::Subscription>,
|
|
_copilot_subscription: Option<gpui::Subscription>,
|
|
io_tx: mpsc::UnboundedSender<(LanguageServerId, IoKind, String)>,
|
|
}
|
|
|
|
struct ProjectState {
|
|
_subscriptions: [gpui::Subscription; 2],
|
|
}
|
|
|
|
trait Message: AsRef<str> {
|
|
type Level: Copy + std::fmt::Debug;
|
|
fn should_include(&self, _: Self::Level) -> bool {
|
|
true
|
|
}
|
|
}
|
|
|
|
struct LogMessage {
|
|
message: String,
|
|
typ: MessageType,
|
|
}
|
|
|
|
impl AsRef<str> for LogMessage {
|
|
fn as_ref(&self) -> &str {
|
|
&self.message
|
|
}
|
|
}
|
|
|
|
impl Message for LogMessage {
|
|
type Level = MessageType;
|
|
|
|
fn should_include(&self, level: Self::Level) -> bool {
|
|
match (self.typ, level) {
|
|
(MessageType::ERROR, _) => true,
|
|
(_, MessageType::ERROR) => false,
|
|
(MessageType::WARNING, _) => true,
|
|
(_, MessageType::WARNING) => false,
|
|
(MessageType::INFO, _) => true,
|
|
(_, MessageType::INFO) => false,
|
|
_ => true,
|
|
}
|
|
}
|
|
}
|
|
|
|
struct TraceMessage {
|
|
message: String,
|
|
}
|
|
|
|
impl AsRef<str> for TraceMessage {
|
|
fn as_ref(&self) -> &str {
|
|
&self.message
|
|
}
|
|
}
|
|
|
|
impl Message for TraceMessage {
|
|
type Level = ();
|
|
}
|
|
|
|
struct RpcMessage {
|
|
message: String,
|
|
}
|
|
|
|
impl AsRef<str> for RpcMessage {
|
|
fn as_ref(&self) -> &str {
|
|
&self.message
|
|
}
|
|
}
|
|
|
|
impl Message for RpcMessage {
|
|
type Level = ();
|
|
}
|
|
|
|
struct LanguageServerState {
|
|
name: Option<LanguageServerName>,
|
|
worktree_id: Option<WorktreeId>,
|
|
kind: LanguageServerKind,
|
|
log_messages: VecDeque<LogMessage>,
|
|
trace_messages: VecDeque<TraceMessage>,
|
|
rpc_state: Option<LanguageServerRpcState>,
|
|
trace_level: TraceValue,
|
|
log_level: MessageType,
|
|
io_logs_subscription: Option<lsp::Subscription>,
|
|
}
|
|
|
|
#[derive(PartialEq, Clone)]
|
|
pub enum LanguageServerKind {
|
|
Local { project: WeakEntity<Project> },
|
|
Remote { project: WeakEntity<Project> },
|
|
Global,
|
|
}
|
|
|
|
impl LanguageServerKind {
|
|
fn is_remote(&self) -> bool {
|
|
matches!(self, LanguageServerKind::Remote { .. })
|
|
}
|
|
}
|
|
|
|
impl std::fmt::Debug for LanguageServerKind {
|
|
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
|
|
match self {
|
|
LanguageServerKind::Local { .. } => write!(f, "LanguageServerKind::Local"),
|
|
LanguageServerKind::Remote { .. } => write!(f, "LanguageServerKind::Remote"),
|
|
LanguageServerKind::Global => write!(f, "LanguageServerKind::Global"),
|
|
}
|
|
}
|
|
}
|
|
|
|
impl LanguageServerKind {
|
|
fn project(&self) -> Option<&WeakEntity<Project>> {
|
|
match self {
|
|
Self::Local { project } => Some(project),
|
|
Self::Remote { project } => Some(project),
|
|
Self::Global { .. } => None,
|
|
}
|
|
}
|
|
}
|
|
|
|
struct LanguageServerRpcState {
|
|
rpc_messages: VecDeque<RpcMessage>,
|
|
last_message_kind: Option<MessageKind>,
|
|
}
|
|
|
|
pub struct LspLogView {
|
|
pub(crate) editor: Entity<Editor>,
|
|
editor_subscriptions: Vec<Subscription>,
|
|
log_store: Entity<LogStore>,
|
|
current_server_id: Option<LanguageServerId>,
|
|
active_entry_kind: LogKind,
|
|
project: Entity<Project>,
|
|
focus_handle: FocusHandle,
|
|
_log_store_subscriptions: Vec<Subscription>,
|
|
}
|
|
|
|
pub struct LspLogToolbarItemView {
|
|
log_view: Option<Entity<LspLogView>>,
|
|
_log_view_subscription: Option<Subscription>,
|
|
}
|
|
|
|
#[derive(Copy, Clone, PartialEq, Eq)]
|
|
enum MessageKind {
|
|
Send,
|
|
Receive,
|
|
}
|
|
|
|
#[derive(Clone, Copy, Debug, Default, PartialEq)]
|
|
pub enum LogKind {
|
|
Rpc,
|
|
Trace,
|
|
#[default]
|
|
Logs,
|
|
ServerInfo,
|
|
}
|
|
|
|
impl LogKind {
|
|
fn label(&self) -> &'static str {
|
|
match self {
|
|
LogKind::Rpc => RPC_MESSAGES,
|
|
LogKind::Trace => SERVER_TRACE,
|
|
LogKind::Logs => SERVER_LOGS,
|
|
LogKind::ServerInfo => SERVER_INFO,
|
|
}
|
|
}
|
|
}
|
|
|
|
#[derive(Clone, Debug, PartialEq)]
|
|
pub(crate) struct LogMenuItem {
|
|
pub server_id: LanguageServerId,
|
|
pub server_name: LanguageServerName,
|
|
pub worktree_root_name: String,
|
|
pub rpc_trace_enabled: bool,
|
|
pub selected_entry: LogKind,
|
|
pub trace_level: lsp::TraceValue,
|
|
pub server_kind: LanguageServerKind,
|
|
}
|
|
|
|
actions!(debug, [OpenLanguageServerLogs]);
|
|
|
|
pub fn init(cx: &mut App) {
|
|
let log_store = cx.new(LogStore::new);
|
|
|
|
cx.observe_new(move |workspace: &mut Workspace, _, cx| {
|
|
let project = workspace.project();
|
|
if project.read(cx).is_local() || project.read(cx).is_via_ssh() {
|
|
log_store.update(cx, |store, cx| {
|
|
store.add_project(project, cx);
|
|
});
|
|
}
|
|
|
|
let log_store = log_store.clone();
|
|
workspace.register_action(move |workspace, _: &OpenLanguageServerLogs, window, cx| {
|
|
let project = workspace.project().read(cx);
|
|
if project.is_local() || project.is_via_ssh() {
|
|
workspace.split_item(
|
|
SplitDirection::Right,
|
|
Box::new(cx.new(|cx| {
|
|
LspLogView::new(workspace.project().clone(), log_store.clone(), window, cx)
|
|
})),
|
|
window,
|
|
cx,
|
|
);
|
|
}
|
|
});
|
|
})
|
|
.detach();
|
|
}
|
|
|
|
impl LogStore {
|
|
pub fn new(cx: &mut Context<Self>) -> Self {
|
|
let (io_tx, mut io_rx) = mpsc::unbounded();
|
|
|
|
let copilot_subscription = Copilot::global(cx).map(|copilot| {
|
|
let copilot = &copilot;
|
|
cx.subscribe(copilot, |this, copilot, inline_completion_event, cx| {
|
|
if let copilot::Event::CopilotLanguageServerStarted = inline_completion_event {
|
|
if let Some(server) = copilot.read(cx).language_server() {
|
|
let server_id = server.server_id();
|
|
let weak_this = cx.weak_entity();
|
|
this.copilot_log_subscription =
|
|
Some(server.on_notification::<copilot::request::LogMessage, _>(
|
|
move |params, cx| {
|
|
weak_this
|
|
.update(cx, |this, cx| {
|
|
this.add_language_server_log(
|
|
server_id,
|
|
MessageType::LOG,
|
|
¶ms.message,
|
|
cx,
|
|
);
|
|
})
|
|
.ok();
|
|
},
|
|
));
|
|
let name = LanguageServerName::new_static("copilot");
|
|
this.add_language_server(
|
|
LanguageServerKind::Global,
|
|
server.server_id(),
|
|
Some(name),
|
|
None,
|
|
Some(server.clone()),
|
|
cx,
|
|
);
|
|
}
|
|
}
|
|
})
|
|
});
|
|
|
|
let this = Self {
|
|
copilot_log_subscription: None,
|
|
_copilot_subscription: copilot_subscription,
|
|
projects: HashMap::default(),
|
|
language_servers: HashMap::default(),
|
|
io_tx,
|
|
};
|
|
|
|
cx.spawn(async move |this, cx| {
|
|
while let Some((server_id, io_kind, message)) = io_rx.next().await {
|
|
if let Some(this) = this.upgrade() {
|
|
this.update(cx, |this, cx| {
|
|
this.on_io(server_id, io_kind, &message, cx);
|
|
})?;
|
|
}
|
|
}
|
|
anyhow::Ok(())
|
|
})
|
|
.detach_and_log_err(cx);
|
|
this
|
|
}
|
|
|
|
pub fn add_project(&mut self, project: &Entity<Project>, cx: &mut Context<Self>) {
|
|
let weak_project = project.downgrade();
|
|
self.projects.insert(
|
|
project.downgrade(),
|
|
ProjectState {
|
|
_subscriptions: [
|
|
cx.observe_release(project, move |this, _, _| {
|
|
this.projects.remove(&weak_project);
|
|
this.language_servers
|
|
.retain(|_, state| state.kind.project() != Some(&weak_project));
|
|
}),
|
|
cx.subscribe(project, |this, project, event, cx| {
|
|
let server_kind = if project.read(cx).is_via_ssh() {
|
|
LanguageServerKind::Remote {
|
|
project: project.downgrade(),
|
|
}
|
|
} else {
|
|
LanguageServerKind::Local {
|
|
project: project.downgrade(),
|
|
}
|
|
};
|
|
|
|
match event {
|
|
project::Event::LanguageServerAdded(id, name, worktree_id) => {
|
|
this.add_language_server(
|
|
server_kind,
|
|
*id,
|
|
Some(name.clone()),
|
|
*worktree_id,
|
|
project
|
|
.read(cx)
|
|
.lsp_store()
|
|
.read(cx)
|
|
.language_server_for_id(*id),
|
|
cx,
|
|
);
|
|
}
|
|
project::Event::LanguageServerRemoved(id) => {
|
|
this.remove_language_server(*id, cx);
|
|
}
|
|
project::Event::LanguageServerLog(id, typ, message) => {
|
|
this.add_language_server(server_kind, *id, None, None, None, cx);
|
|
match typ {
|
|
project::LanguageServerLogType::Log(typ) => {
|
|
this.add_language_server_log(*id, *typ, message, cx);
|
|
}
|
|
project::LanguageServerLogType::Trace(_) => {
|
|
this.add_language_server_trace(*id, message, cx);
|
|
}
|
|
}
|
|
}
|
|
_ => {}
|
|
}
|
|
}),
|
|
],
|
|
},
|
|
);
|
|
}
|
|
|
|
fn get_language_server_state(
|
|
&mut self,
|
|
id: LanguageServerId,
|
|
) -> Option<&mut LanguageServerState> {
|
|
self.language_servers.get_mut(&id)
|
|
}
|
|
|
|
fn add_language_server(
|
|
&mut self,
|
|
kind: LanguageServerKind,
|
|
server_id: LanguageServerId,
|
|
name: Option<LanguageServerName>,
|
|
worktree_id: Option<WorktreeId>,
|
|
server: Option<Arc<LanguageServer>>,
|
|
cx: &mut Context<Self>,
|
|
) -> Option<&mut LanguageServerState> {
|
|
let server_state = self.language_servers.entry(server_id).or_insert_with(|| {
|
|
cx.notify();
|
|
LanguageServerState {
|
|
name: None,
|
|
worktree_id: None,
|
|
kind,
|
|
rpc_state: None,
|
|
log_messages: VecDeque::with_capacity(MAX_STORED_LOG_ENTRIES),
|
|
trace_messages: VecDeque::with_capacity(MAX_STORED_LOG_ENTRIES),
|
|
trace_level: TraceValue::Off,
|
|
log_level: MessageType::LOG,
|
|
io_logs_subscription: None,
|
|
}
|
|
});
|
|
|
|
if let Some(name) = name {
|
|
server_state.name = Some(name);
|
|
}
|
|
if let Some(worktree_id) = worktree_id {
|
|
server_state.worktree_id = Some(worktree_id);
|
|
}
|
|
|
|
if let Some(server) = server
|
|
.clone()
|
|
.filter(|_| server_state.io_logs_subscription.is_none())
|
|
{
|
|
let io_tx = self.io_tx.clone();
|
|
let server_id = server.server_id();
|
|
server_state.io_logs_subscription = Some(server.on_io(move |io_kind, message| {
|
|
io_tx
|
|
.unbounded_send((server_id, io_kind, message.to_string()))
|
|
.ok();
|
|
}));
|
|
}
|
|
|
|
Some(server_state)
|
|
}
|
|
|
|
fn add_language_server_log(
|
|
&mut self,
|
|
id: LanguageServerId,
|
|
typ: MessageType,
|
|
message: &str,
|
|
cx: &mut Context<Self>,
|
|
) -> Option<()> {
|
|
let language_server_state = self.get_language_server_state(id)?;
|
|
|
|
let log_lines = &mut language_server_state.log_messages;
|
|
Self::add_language_server_message(
|
|
log_lines,
|
|
id,
|
|
LogMessage {
|
|
message: message.trim_end().to_string(),
|
|
typ,
|
|
},
|
|
language_server_state.log_level,
|
|
LogKind::Logs,
|
|
cx,
|
|
);
|
|
Some(())
|
|
}
|
|
|
|
fn add_language_server_trace(
|
|
&mut self,
|
|
id: LanguageServerId,
|
|
message: &str,
|
|
cx: &mut Context<Self>,
|
|
) -> Option<()> {
|
|
let language_server_state = self.get_language_server_state(id)?;
|
|
|
|
let log_lines = &mut language_server_state.trace_messages;
|
|
Self::add_language_server_message(
|
|
log_lines,
|
|
id,
|
|
TraceMessage {
|
|
message: message.trim_end().to_string(),
|
|
},
|
|
(),
|
|
LogKind::Trace,
|
|
cx,
|
|
);
|
|
Some(())
|
|
}
|
|
|
|
fn add_language_server_message<T: Message>(
|
|
log_lines: &mut VecDeque<T>,
|
|
id: LanguageServerId,
|
|
message: T,
|
|
current_severity: <T as Message>::Level,
|
|
kind: LogKind,
|
|
cx: &mut Context<Self>,
|
|
) {
|
|
while log_lines.len() >= MAX_STORED_LOG_ENTRIES {
|
|
log_lines.pop_front();
|
|
}
|
|
let entry: &str = message.as_ref();
|
|
let entry = entry.to_string();
|
|
let visible = message.should_include(current_severity);
|
|
log_lines.push_back(message);
|
|
|
|
if visible {
|
|
cx.emit(Event::NewServerLogEntry { id, entry, kind });
|
|
cx.notify();
|
|
}
|
|
}
|
|
|
|
fn remove_language_server(&mut self, id: LanguageServerId, cx: &mut Context<Self>) {
|
|
self.language_servers.remove(&id);
|
|
cx.notify();
|
|
}
|
|
|
|
fn server_logs(&self, server_id: LanguageServerId) -> Option<&VecDeque<LogMessage>> {
|
|
Some(&self.language_servers.get(&server_id)?.log_messages)
|
|
}
|
|
|
|
fn server_trace(&self, server_id: LanguageServerId) -> Option<&VecDeque<TraceMessage>> {
|
|
Some(&self.language_servers.get(&server_id)?.trace_messages)
|
|
}
|
|
|
|
fn server_ids_for_project<'a>(
|
|
&'a self,
|
|
lookup_project: &'a WeakEntity<Project>,
|
|
) -> impl Iterator<Item = LanguageServerId> + 'a {
|
|
self.language_servers
|
|
.iter()
|
|
.filter_map(move |(id, state)| match &state.kind {
|
|
LanguageServerKind::Local { project } | LanguageServerKind::Remote { project } => {
|
|
if project == lookup_project {
|
|
Some(*id)
|
|
} else {
|
|
None
|
|
}
|
|
}
|
|
LanguageServerKind::Global => Some(*id),
|
|
})
|
|
}
|
|
|
|
fn enable_rpc_trace_for_language_server(
|
|
&mut self,
|
|
server_id: LanguageServerId,
|
|
) -> Option<&mut LanguageServerRpcState> {
|
|
let rpc_state = self
|
|
.language_servers
|
|
.get_mut(&server_id)?
|
|
.rpc_state
|
|
.get_or_insert_with(|| LanguageServerRpcState {
|
|
rpc_messages: VecDeque::with_capacity(MAX_STORED_LOG_ENTRIES),
|
|
last_message_kind: None,
|
|
});
|
|
Some(rpc_state)
|
|
}
|
|
|
|
pub fn disable_rpc_trace_for_language_server(
|
|
&mut self,
|
|
server_id: LanguageServerId,
|
|
) -> Option<()> {
|
|
self.language_servers.get_mut(&server_id)?.rpc_state.take();
|
|
Some(())
|
|
}
|
|
|
|
fn on_io(
|
|
&mut self,
|
|
language_server_id: LanguageServerId,
|
|
io_kind: IoKind,
|
|
message: &str,
|
|
cx: &mut Context<Self>,
|
|
) -> Option<()> {
|
|
let is_received = match io_kind {
|
|
IoKind::StdOut => true,
|
|
IoKind::StdIn => false,
|
|
IoKind::StdErr => {
|
|
self.add_language_server_log(language_server_id, MessageType::LOG, &message, cx);
|
|
return Some(());
|
|
}
|
|
};
|
|
|
|
let state = self
|
|
.get_language_server_state(language_server_id)?
|
|
.rpc_state
|
|
.as_mut()?;
|
|
let kind = if is_received {
|
|
MessageKind::Receive
|
|
} else {
|
|
MessageKind::Send
|
|
};
|
|
|
|
let rpc_log_lines = &mut state.rpc_messages;
|
|
if state.last_message_kind != Some(kind) {
|
|
let line_before_message = match kind {
|
|
MessageKind::Send => SEND_LINE,
|
|
MessageKind::Receive => RECEIVE_LINE,
|
|
};
|
|
rpc_log_lines.push_back(RpcMessage {
|
|
message: line_before_message.to_string(),
|
|
});
|
|
cx.emit(Event::NewServerLogEntry {
|
|
id: language_server_id,
|
|
entry: line_before_message.to_string(),
|
|
kind: LogKind::Rpc,
|
|
});
|
|
}
|
|
|
|
while rpc_log_lines.len() >= MAX_STORED_LOG_ENTRIES {
|
|
rpc_log_lines.pop_front();
|
|
}
|
|
let message = message.trim();
|
|
rpc_log_lines.push_back(RpcMessage {
|
|
message: message.to_string(),
|
|
});
|
|
cx.emit(Event::NewServerLogEntry {
|
|
id: language_server_id,
|
|
entry: message.to_string(),
|
|
kind: LogKind::Rpc,
|
|
});
|
|
cx.notify();
|
|
Some(())
|
|
}
|
|
}
|
|
|
|
impl LspLogView {
|
|
pub fn new(
|
|
project: Entity<Project>,
|
|
log_store: Entity<LogStore>,
|
|
window: &mut Window,
|
|
cx: &mut Context<Self>,
|
|
) -> Self {
|
|
let server_id = log_store
|
|
.read(cx)
|
|
.language_servers
|
|
.iter()
|
|
.find(|(_, server)| server.kind.project() == Some(&project.downgrade()))
|
|
.map(|(id, _)| *id);
|
|
|
|
let weak_project = project.downgrade();
|
|
let model_changes_subscription =
|
|
cx.observe_in(&log_store, window, move |this, store, window, cx| {
|
|
let first_server_id_for_project =
|
|
store.read(cx).server_ids_for_project(&weak_project).next();
|
|
if let Some(current_lsp) = this.current_server_id {
|
|
if !store.read(cx).language_servers.contains_key(¤t_lsp) {
|
|
if let Some(server_id) = first_server_id_for_project {
|
|
match this.active_entry_kind {
|
|
LogKind::Rpc => {
|
|
this.show_rpc_trace_for_server(server_id, window, cx)
|
|
}
|
|
LogKind::Trace => this.show_trace_for_server(server_id, window, cx),
|
|
LogKind::Logs => this.show_logs_for_server(server_id, window, cx),
|
|
LogKind::ServerInfo => this.show_server_info(server_id, window, cx),
|
|
}
|
|
}
|
|
}
|
|
} else if let Some(server_id) = first_server_id_for_project {
|
|
match this.active_entry_kind {
|
|
LogKind::Rpc => this.show_rpc_trace_for_server(server_id, window, cx),
|
|
LogKind::Trace => this.show_trace_for_server(server_id, window, cx),
|
|
LogKind::Logs => this.show_logs_for_server(server_id, window, cx),
|
|
LogKind::ServerInfo => this.show_server_info(server_id, window, cx),
|
|
}
|
|
}
|
|
|
|
cx.notify();
|
|
});
|
|
let events_subscriptions = cx.subscribe_in(
|
|
&log_store,
|
|
window,
|
|
move |log_view, _, e, window, cx| match e {
|
|
Event::NewServerLogEntry { id, entry, kind } => {
|
|
if log_view.current_server_id == Some(*id)
|
|
&& *kind == log_view.active_entry_kind
|
|
{
|
|
log_view.editor.update(cx, |editor, cx| {
|
|
editor.set_read_only(false);
|
|
let last_point = editor.buffer().read(cx).len(cx);
|
|
let newest_cursor_is_at_end =
|
|
editor.selections.newest::<usize>(cx).start >= last_point;
|
|
editor.edit(
|
|
vec![
|
|
(last_point..last_point, entry.trim()),
|
|
(last_point..last_point, "\n"),
|
|
],
|
|
cx,
|
|
);
|
|
let entry_length = entry.len();
|
|
if entry_length > 1024 {
|
|
editor.fold_ranges(
|
|
vec![last_point + 1024..last_point + entry_length],
|
|
false,
|
|
window,
|
|
cx,
|
|
);
|
|
}
|
|
|
|
if newest_cursor_is_at_end {
|
|
editor.request_autoscroll(Autoscroll::bottom(), cx);
|
|
}
|
|
editor.set_read_only(true);
|
|
});
|
|
}
|
|
}
|
|
},
|
|
);
|
|
let (editor, editor_subscriptions) = Self::editor_for_logs(String::new(), window, cx);
|
|
|
|
let focus_handle = cx.focus_handle();
|
|
let focus_subscription = cx.on_focus(&focus_handle, window, |log_view, window, cx| {
|
|
window.focus(&log_view.editor.focus_handle(cx));
|
|
});
|
|
|
|
let mut this = Self {
|
|
focus_handle,
|
|
editor,
|
|
editor_subscriptions,
|
|
project,
|
|
log_store,
|
|
current_server_id: None,
|
|
active_entry_kind: LogKind::Logs,
|
|
_log_store_subscriptions: vec![
|
|
model_changes_subscription,
|
|
events_subscriptions,
|
|
focus_subscription,
|
|
],
|
|
};
|
|
if let Some(server_id) = server_id {
|
|
this.show_logs_for_server(server_id, window, cx);
|
|
}
|
|
this
|
|
}
|
|
|
|
fn editor_for_logs(
|
|
log_contents: String,
|
|
window: &mut Window,
|
|
cx: &mut Context<Self>,
|
|
) -> (Entity<Editor>, Vec<Subscription>) {
|
|
let editor = cx.new(|cx| {
|
|
let mut editor = Editor::multi_line(window, cx);
|
|
editor.set_text(log_contents, window, cx);
|
|
editor.move_to_end(&MoveToEnd, window, cx);
|
|
editor.set_read_only(true);
|
|
editor.set_show_edit_predictions(Some(false), window, cx);
|
|
editor.set_soft_wrap_mode(SoftWrap::EditorWidth, cx);
|
|
editor
|
|
});
|
|
let editor_subscription = cx.subscribe(
|
|
&editor,
|
|
|_, _, event: &EditorEvent, cx: &mut Context<LspLogView>| cx.emit(event.clone()),
|
|
);
|
|
let search_subscription = cx.subscribe(
|
|
&editor,
|
|
|_, _, event: &SearchEvent, cx: &mut Context<LspLogView>| cx.emit(event.clone()),
|
|
);
|
|
(editor, vec![editor_subscription, search_subscription])
|
|
}
|
|
|
|
fn editor_for_server_info(
|
|
server: &LanguageServer,
|
|
window: &mut Window,
|
|
cx: &mut Context<Self>,
|
|
) -> (Entity<Editor>, Vec<Subscription>) {
|
|
let editor = cx.new(|cx| {
|
|
let mut editor = Editor::multi_line(window, cx);
|
|
let server_info = format!(
|
|
"* Server: {NAME} (id {ID})
|
|
|
|
* Binary: {BINARY:#?}
|
|
|
|
* Registered workspace folders:
|
|
{WORKSPACE_FOLDERS}
|
|
|
|
* Capabilities: {CAPABILITIES}
|
|
|
|
* Configuration: {CONFIGURATION}",
|
|
NAME = server.name(),
|
|
ID = server.server_id(),
|
|
BINARY = server.binary(),
|
|
WORKSPACE_FOLDERS = server
|
|
.workspace_folders()
|
|
.iter()
|
|
.filter_map(|path| path
|
|
.to_file_path()
|
|
.ok()
|
|
.map(|path| path.to_string_lossy().into_owned()))
|
|
.collect::<Vec<_>>()
|
|
.join(", "),
|
|
CAPABILITIES = serde_json::to_string_pretty(&server.capabilities())
|
|
.unwrap_or_else(|e| format!("Failed to serialize capabilities: {e}")),
|
|
CONFIGURATION = serde_json::to_string_pretty(server.configuration())
|
|
.unwrap_or_else(|e| format!("Failed to serialize configuration: {e}")),
|
|
);
|
|
editor.set_text(server_info, window, cx);
|
|
editor.set_read_only(true);
|
|
editor.set_show_edit_predictions(Some(false), window, cx);
|
|
editor.set_soft_wrap_mode(SoftWrap::EditorWidth, cx);
|
|
editor
|
|
});
|
|
let editor_subscription = cx.subscribe(
|
|
&editor,
|
|
|_, _, event: &EditorEvent, cx: &mut Context<LspLogView>| cx.emit(event.clone()),
|
|
);
|
|
let search_subscription = cx.subscribe(
|
|
&editor,
|
|
|_, _, event: &SearchEvent, cx: &mut Context<LspLogView>| cx.emit(event.clone()),
|
|
);
|
|
(editor, vec![editor_subscription, search_subscription])
|
|
}
|
|
|
|
pub(crate) fn menu_items<'a>(&'a self, cx: &'a App) -> Option<Vec<LogMenuItem>> {
|
|
let log_store = self.log_store.read(cx);
|
|
|
|
let unknown_server = LanguageServerName::new_static("unknown server");
|
|
|
|
let mut rows = log_store
|
|
.language_servers
|
|
.iter()
|
|
.map(|(server_id, state)| match &state.kind {
|
|
LanguageServerKind::Local { .. } | LanguageServerKind::Remote { .. } => {
|
|
let worktree_root_name = state
|
|
.worktree_id
|
|
.and_then(|id| self.project.read(cx).worktree_for_id(id, cx))
|
|
.map(|worktree| worktree.read(cx).root_name().to_string())
|
|
.unwrap_or_else(|| "Unknown worktree".to_string());
|
|
|
|
LogMenuItem {
|
|
server_id: *server_id,
|
|
server_name: state.name.clone().unwrap_or(unknown_server.clone()),
|
|
server_kind: state.kind.clone(),
|
|
worktree_root_name,
|
|
rpc_trace_enabled: state.rpc_state.is_some(),
|
|
selected_entry: self.active_entry_kind,
|
|
trace_level: lsp::TraceValue::Off,
|
|
}
|
|
}
|
|
|
|
LanguageServerKind::Global => LogMenuItem {
|
|
server_id: *server_id,
|
|
server_name: state.name.clone().unwrap_or(unknown_server.clone()),
|
|
server_kind: state.kind.clone(),
|
|
worktree_root_name: "supplementary".to_string(),
|
|
rpc_trace_enabled: state.rpc_state.is_some(),
|
|
selected_entry: self.active_entry_kind,
|
|
trace_level: lsp::TraceValue::Off,
|
|
},
|
|
})
|
|
.chain(
|
|
self.project
|
|
.read(cx)
|
|
.supplementary_language_servers(cx)
|
|
.filter_map(|(server_id, name)| {
|
|
let state = log_store.language_servers.get(&server_id)?;
|
|
Some(LogMenuItem {
|
|
server_id,
|
|
server_name: name.clone(),
|
|
server_kind: state.kind.clone(),
|
|
worktree_root_name: "supplementary".to_string(),
|
|
rpc_trace_enabled: state.rpc_state.is_some(),
|
|
selected_entry: self.active_entry_kind,
|
|
trace_level: lsp::TraceValue::Off,
|
|
})
|
|
}),
|
|
)
|
|
.collect::<Vec<_>>();
|
|
rows.sort_by_key(|row| row.server_id);
|
|
rows.dedup_by_key(|row| row.server_id);
|
|
Some(rows)
|
|
}
|
|
|
|
fn show_logs_for_server(
|
|
&mut self,
|
|
server_id: LanguageServerId,
|
|
window: &mut Window,
|
|
cx: &mut Context<Self>,
|
|
) {
|
|
let typ = self
|
|
.log_store
|
|
.read_with(cx, |v, _| {
|
|
v.language_servers.get(&server_id).map(|v| v.log_level)
|
|
})
|
|
.unwrap_or(MessageType::LOG);
|
|
let log_contents = self
|
|
.log_store
|
|
.read(cx)
|
|
.server_logs(server_id)
|
|
.map(|v| log_contents(v, typ));
|
|
if let Some(log_contents) = log_contents {
|
|
self.current_server_id = Some(server_id);
|
|
self.active_entry_kind = LogKind::Logs;
|
|
let (editor, editor_subscriptions) = Self::editor_for_logs(log_contents, window, cx);
|
|
self.editor = editor;
|
|
self.editor_subscriptions = editor_subscriptions;
|
|
cx.notify();
|
|
}
|
|
window.focus(&self.focus_handle);
|
|
}
|
|
|
|
fn update_log_level(
|
|
&self,
|
|
server_id: LanguageServerId,
|
|
level: MessageType,
|
|
window: &mut Window,
|
|
cx: &mut Context<Self>,
|
|
) {
|
|
let log_contents = self.log_store.update(cx, |this, _| {
|
|
if let Some(state) = this.get_language_server_state(server_id) {
|
|
state.log_level = level;
|
|
}
|
|
|
|
this.server_logs(server_id).map(|v| log_contents(v, level))
|
|
});
|
|
|
|
if let Some(log_contents) = log_contents {
|
|
self.editor.update(cx, |editor, cx| {
|
|
editor.set_text(log_contents, window, cx);
|
|
editor.move_to_end(&MoveToEnd, window, cx);
|
|
});
|
|
cx.notify();
|
|
}
|
|
|
|
window.focus(&self.focus_handle);
|
|
}
|
|
|
|
fn show_trace_for_server(
|
|
&mut self,
|
|
server_id: LanguageServerId,
|
|
window: &mut Window,
|
|
cx: &mut Context<Self>,
|
|
) {
|
|
let log_contents = self
|
|
.log_store
|
|
.read(cx)
|
|
.server_trace(server_id)
|
|
.map(|v| log_contents(v, ()));
|
|
if let Some(log_contents) = log_contents {
|
|
self.current_server_id = Some(server_id);
|
|
self.active_entry_kind = LogKind::Trace;
|
|
let (editor, editor_subscriptions) = Self::editor_for_logs(log_contents, window, cx);
|
|
self.editor = editor;
|
|
self.editor_subscriptions = editor_subscriptions;
|
|
cx.notify();
|
|
}
|
|
window.focus(&self.focus_handle);
|
|
}
|
|
|
|
fn show_rpc_trace_for_server(
|
|
&mut self,
|
|
server_id: LanguageServerId,
|
|
window: &mut Window,
|
|
cx: &mut Context<Self>,
|
|
) {
|
|
let rpc_log = self.log_store.update(cx, |log_store, _| {
|
|
log_store
|
|
.enable_rpc_trace_for_language_server(server_id)
|
|
.map(|state| log_contents(&state.rpc_messages, ()))
|
|
});
|
|
if let Some(rpc_log) = rpc_log {
|
|
self.current_server_id = Some(server_id);
|
|
self.active_entry_kind = LogKind::Rpc;
|
|
let (editor, editor_subscriptions) = Self::editor_for_logs(rpc_log, window, cx);
|
|
let language = self.project.read(cx).languages().language_for_name("JSON");
|
|
editor
|
|
.read(cx)
|
|
.buffer()
|
|
.read(cx)
|
|
.as_singleton()
|
|
.expect("log buffer should be a singleton")
|
|
.update(cx, |_, cx| {
|
|
cx.spawn({
|
|
let buffer = cx.entity();
|
|
async move |_, cx| {
|
|
let language = language.await.ok();
|
|
buffer.update(cx, |buffer, cx| {
|
|
buffer.set_language(language, cx);
|
|
})
|
|
}
|
|
})
|
|
.detach_and_log_err(cx);
|
|
});
|
|
|
|
self.editor = editor;
|
|
self.editor_subscriptions = editor_subscriptions;
|
|
cx.notify();
|
|
}
|
|
|
|
window.focus(&self.focus_handle);
|
|
}
|
|
|
|
fn toggle_rpc_trace_for_server(
|
|
&mut self,
|
|
server_id: LanguageServerId,
|
|
enabled: bool,
|
|
window: &mut Window,
|
|
cx: &mut Context<Self>,
|
|
) {
|
|
self.log_store.update(cx, |log_store, _| {
|
|
if enabled {
|
|
log_store.enable_rpc_trace_for_language_server(server_id);
|
|
} else {
|
|
log_store.disable_rpc_trace_for_language_server(server_id);
|
|
}
|
|
});
|
|
if !enabled && Some(server_id) == self.current_server_id {
|
|
self.show_logs_for_server(server_id, window, cx);
|
|
cx.notify();
|
|
}
|
|
}
|
|
|
|
fn update_trace_level(
|
|
&self,
|
|
server_id: LanguageServerId,
|
|
level: TraceValue,
|
|
cx: &mut Context<Self>,
|
|
) {
|
|
if let Some(server) = self
|
|
.project
|
|
.read(cx)
|
|
.lsp_store()
|
|
.read(cx)
|
|
.language_server_for_id(server_id)
|
|
{
|
|
self.log_store.update(cx, |this, _| {
|
|
if let Some(state) = this.get_language_server_state(server_id) {
|
|
state.trace_level = level;
|
|
}
|
|
});
|
|
|
|
server
|
|
.notify::<SetTrace>(&SetTraceParams { value: level })
|
|
.ok();
|
|
}
|
|
}
|
|
|
|
fn show_server_info(
|
|
&mut self,
|
|
server_id: LanguageServerId,
|
|
window: &mut Window,
|
|
cx: &mut Context<Self>,
|
|
) {
|
|
let lsp_store = self.project.read(cx).lsp_store();
|
|
let Some(server) = lsp_store.read(cx).language_server_for_id(server_id) else {
|
|
return;
|
|
};
|
|
self.current_server_id = Some(server_id);
|
|
self.active_entry_kind = LogKind::ServerInfo;
|
|
let (editor, editor_subscriptions) = Self::editor_for_server_info(&server, window, cx);
|
|
self.editor = editor;
|
|
self.editor_subscriptions = editor_subscriptions;
|
|
cx.notify();
|
|
window.focus(&self.focus_handle);
|
|
}
|
|
}
|
|
|
|
fn log_filter<T: Message>(line: &T, cmp: <T as Message>::Level) -> Option<&str> {
|
|
if line.should_include(cmp) {
|
|
Some(line.as_ref())
|
|
} else {
|
|
None
|
|
}
|
|
}
|
|
|
|
fn log_contents<T: Message>(lines: &VecDeque<T>, cmp: <T as Message>::Level) -> String {
|
|
let (a, b) = lines.as_slices();
|
|
let a = a.iter().filter_map(move |v| log_filter(v, cmp));
|
|
let b = b.iter().filter_map(move |v| log_filter(v, cmp));
|
|
a.chain(b).fold(String::new(), |mut acc, el| {
|
|
acc.push_str(el);
|
|
acc.push('\n');
|
|
acc
|
|
})
|
|
}
|
|
|
|
impl Render for LspLogView {
|
|
fn render(&mut self, window: &mut Window, cx: &mut Context<Self>) -> impl IntoElement {
|
|
self.editor.update(cx, |editor, cx| {
|
|
editor.render(window, cx).into_any_element()
|
|
})
|
|
}
|
|
}
|
|
|
|
impl Focusable for LspLogView {
|
|
fn focus_handle(&self, _: &App) -> FocusHandle {
|
|
self.focus_handle.clone()
|
|
}
|
|
}
|
|
|
|
impl Item for LspLogView {
|
|
type Event = EditorEvent;
|
|
|
|
fn to_item_events(event: &Self::Event, f: impl FnMut(workspace::item::ItemEvent)) {
|
|
Editor::to_item_events(event, f)
|
|
}
|
|
|
|
fn tab_content_text(&self, _detail: usize, _cx: &App) -> SharedString {
|
|
"LSP Logs".into()
|
|
}
|
|
|
|
fn telemetry_event_text(&self) -> Option<&'static str> {
|
|
None
|
|
}
|
|
|
|
fn as_searchable(&self, handle: &Entity<Self>) -> Option<Box<dyn SearchableItemHandle>> {
|
|
Some(Box::new(handle.clone()))
|
|
}
|
|
|
|
fn act_as_type<'a>(
|
|
&'a self,
|
|
type_id: TypeId,
|
|
self_handle: &'a Entity<Self>,
|
|
_: &'a App,
|
|
) -> Option<AnyView> {
|
|
if type_id == TypeId::of::<Self>() {
|
|
Some(self_handle.to_any())
|
|
} else if type_id == TypeId::of::<Editor>() {
|
|
Some(self.editor.to_any())
|
|
} else {
|
|
None
|
|
}
|
|
}
|
|
|
|
fn clone_on_split(
|
|
&self,
|
|
_workspace_id: Option<WorkspaceId>,
|
|
window: &mut Window,
|
|
cx: &mut Context<Self>,
|
|
) -> Option<Entity<Self>>
|
|
where
|
|
Self: Sized,
|
|
{
|
|
Some(cx.new(|cx| {
|
|
let mut new_view = Self::new(self.project.clone(), self.log_store.clone(), window, cx);
|
|
if let Some(server_id) = self.current_server_id {
|
|
match self.active_entry_kind {
|
|
LogKind::Rpc => new_view.show_rpc_trace_for_server(server_id, window, cx),
|
|
LogKind::Trace => new_view.show_trace_for_server(server_id, window, cx),
|
|
LogKind::Logs => new_view.show_logs_for_server(server_id, window, cx),
|
|
LogKind::ServerInfo => new_view.show_server_info(server_id, window, cx),
|
|
}
|
|
}
|
|
new_view
|
|
}))
|
|
}
|
|
}
|
|
|
|
impl SearchableItem for LspLogView {
|
|
type Match = <Editor as SearchableItem>::Match;
|
|
|
|
fn clear_matches(&mut self, window: &mut Window, cx: &mut Context<Self>) {
|
|
self.editor.update(cx, |e, cx| e.clear_matches(window, cx))
|
|
}
|
|
|
|
fn update_matches(
|
|
&mut self,
|
|
matches: &[Self::Match],
|
|
window: &mut Window,
|
|
cx: &mut Context<Self>,
|
|
) {
|
|
self.editor
|
|
.update(cx, |e, cx| e.update_matches(matches, window, cx))
|
|
}
|
|
|
|
fn query_suggestion(&mut self, window: &mut Window, cx: &mut Context<Self>) -> String {
|
|
self.editor
|
|
.update(cx, |e, cx| e.query_suggestion(window, cx))
|
|
}
|
|
|
|
fn activate_match(
|
|
&mut self,
|
|
index: usize,
|
|
matches: &[Self::Match],
|
|
window: &mut Window,
|
|
cx: &mut Context<Self>,
|
|
) {
|
|
self.editor
|
|
.update(cx, |e, cx| e.activate_match(index, matches, window, cx))
|
|
}
|
|
|
|
fn select_matches(
|
|
&mut self,
|
|
matches: &[Self::Match],
|
|
window: &mut Window,
|
|
cx: &mut Context<Self>,
|
|
) {
|
|
self.editor
|
|
.update(cx, |e, cx| e.select_matches(matches, window, cx))
|
|
}
|
|
|
|
fn find_matches(
|
|
&mut self,
|
|
query: Arc<project::search::SearchQuery>,
|
|
window: &mut Window,
|
|
cx: &mut Context<Self>,
|
|
) -> gpui::Task<Vec<Self::Match>> {
|
|
self.editor
|
|
.update(cx, |e, cx| e.find_matches(query, window, cx))
|
|
}
|
|
|
|
fn replace(
|
|
&mut self,
|
|
_: &Self::Match,
|
|
_: &SearchQuery,
|
|
_window: &mut Window,
|
|
_: &mut Context<Self>,
|
|
) {
|
|
// Since LSP Log is read-only, it doesn't make sense to support replace operation.
|
|
}
|
|
fn supported_options(&self) -> workspace::searchable::SearchOptions {
|
|
workspace::searchable::SearchOptions {
|
|
case: true,
|
|
word: true,
|
|
regex: true,
|
|
find_in_results: false,
|
|
// LSP log is read-only.
|
|
replacement: false,
|
|
selection: false,
|
|
}
|
|
}
|
|
fn active_match_index(
|
|
&mut self,
|
|
direction: Direction,
|
|
matches: &[Self::Match],
|
|
window: &mut Window,
|
|
cx: &mut Context<Self>,
|
|
) -> Option<usize> {
|
|
self.editor.update(cx, |e, cx| {
|
|
e.active_match_index(direction, matches, window, cx)
|
|
})
|
|
}
|
|
}
|
|
|
|
impl EventEmitter<ToolbarItemEvent> for LspLogToolbarItemView {}
|
|
|
|
impl ToolbarItemView for LspLogToolbarItemView {
|
|
fn set_active_pane_item(
|
|
&mut self,
|
|
active_pane_item: Option<&dyn ItemHandle>,
|
|
_: &mut Window,
|
|
cx: &mut Context<Self>,
|
|
) -> workspace::ToolbarItemLocation {
|
|
if let Some(item) = active_pane_item {
|
|
if let Some(log_view) = item.downcast::<LspLogView>() {
|
|
self.log_view = Some(log_view.clone());
|
|
self._log_view_subscription = Some(cx.observe(&log_view, |_, _, cx| {
|
|
cx.notify();
|
|
}));
|
|
return ToolbarItemLocation::PrimaryLeft;
|
|
}
|
|
}
|
|
self.log_view = None;
|
|
self._log_view_subscription = None;
|
|
ToolbarItemLocation::Hidden
|
|
}
|
|
}
|
|
|
|
impl Render for LspLogToolbarItemView {
|
|
fn render(&mut self, _: &mut Window, cx: &mut Context<Self>) -> impl IntoElement {
|
|
let Some(log_view) = self.log_view.clone() else {
|
|
return div();
|
|
};
|
|
let (menu_rows, current_server_id) = log_view.update(cx, |log_view, cx| {
|
|
let menu_rows = log_view.menu_items(cx).unwrap_or_default();
|
|
let current_server_id = log_view.current_server_id;
|
|
(menu_rows, current_server_id)
|
|
});
|
|
|
|
let current_server = current_server_id.and_then(|current_server_id| {
|
|
if let Ok(ix) = menu_rows.binary_search_by_key(¤t_server_id, |e| e.server_id) {
|
|
Some(menu_rows[ix].clone())
|
|
} else {
|
|
None
|
|
}
|
|
});
|
|
let available_language_servers: Vec<_> = menu_rows
|
|
.iter()
|
|
.map(|row| {
|
|
(
|
|
row.server_id,
|
|
row.server_name.clone(),
|
|
row.worktree_root_name.clone(),
|
|
row.selected_entry,
|
|
)
|
|
})
|
|
.collect();
|
|
let log_toolbar_view = cx.entity().clone();
|
|
let lsp_menu = PopoverMenu::new("LspLogView")
|
|
.anchor(Corner::TopLeft)
|
|
.trigger(Button::new(
|
|
"language_server_menu_header",
|
|
current_server
|
|
.as_ref()
|
|
.map(|row| {
|
|
Cow::Owned(format!(
|
|
"{} ({})",
|
|
row.server_name.0, row.worktree_root_name,
|
|
))
|
|
})
|
|
.unwrap_or_else(|| "No server selected".into()),
|
|
))
|
|
.menu({
|
|
let log_view = log_view.clone();
|
|
move |window, cx| {
|
|
let log_view = log_view.clone();
|
|
ContextMenu::build(window, cx, |mut menu, window, _| {
|
|
for (server_id, name, worktree_root, active_entry_kind) in
|
|
available_language_servers.iter()
|
|
{
|
|
let label = format!("{} ({})", name, worktree_root);
|
|
let server_id = *server_id;
|
|
let active_entry_kind = *active_entry_kind;
|
|
menu = menu.entry(
|
|
label,
|
|
None,
|
|
window.handler_for(&log_view, move |view, window, cx| {
|
|
view.current_server_id = Some(server_id);
|
|
view.active_entry_kind = active_entry_kind;
|
|
match view.active_entry_kind {
|
|
LogKind::Rpc => {
|
|
view.toggle_rpc_trace_for_server(
|
|
server_id, true, window, cx,
|
|
);
|
|
view.show_rpc_trace_for_server(server_id, window, cx);
|
|
}
|
|
LogKind::Trace => {
|
|
view.show_trace_for_server(server_id, window, cx)
|
|
}
|
|
LogKind::Logs => {
|
|
view.show_logs_for_server(server_id, window, cx)
|
|
}
|
|
LogKind::ServerInfo => {
|
|
view.show_server_info(server_id, window, cx)
|
|
}
|
|
}
|
|
cx.notify();
|
|
}),
|
|
);
|
|
}
|
|
menu
|
|
})
|
|
.into()
|
|
}
|
|
});
|
|
let view_selector = current_server.map(|server| {
|
|
let server_id = server.server_id;
|
|
let is_remote = server.server_kind.is_remote();
|
|
let rpc_trace_enabled = server.rpc_trace_enabled;
|
|
let log_view = log_view.clone();
|
|
PopoverMenu::new("LspViewSelector")
|
|
.anchor(Corner::TopLeft)
|
|
.trigger(Button::new(
|
|
"language_server_menu_header",
|
|
server.selected_entry.label(),
|
|
))
|
|
.menu(move |window, cx| {
|
|
let log_toolbar_view = log_toolbar_view.clone();
|
|
let log_view = log_view.clone();
|
|
Some(ContextMenu::build(window, cx, move |this, window, _| {
|
|
this.entry(
|
|
SERVER_LOGS,
|
|
None,
|
|
window.handler_for(&log_view, move |view, window, cx| {
|
|
view.show_logs_for_server(server_id, window, cx);
|
|
}),
|
|
)
|
|
.when(!is_remote, |this| {
|
|
this.entry(
|
|
SERVER_TRACE,
|
|
None,
|
|
window.handler_for(&log_view, move |view, window, cx| {
|
|
view.show_trace_for_server(server_id, window, cx);
|
|
}),
|
|
)
|
|
.custom_entry(
|
|
{
|
|
let log_toolbar_view = log_toolbar_view.clone();
|
|
move |window, _| {
|
|
h_flex()
|
|
.w_full()
|
|
.justify_between()
|
|
.child(Label::new(RPC_MESSAGES))
|
|
.child(
|
|
div().child(
|
|
Checkbox::new(
|
|
"LspLogEnableRpcTrace",
|
|
if rpc_trace_enabled {
|
|
ToggleState::Selected
|
|
} else {
|
|
ToggleState::Unselected
|
|
},
|
|
)
|
|
.on_click(window.listener_for(
|
|
&log_toolbar_view,
|
|
move |view, selection, window, cx| {
|
|
let enabled = matches!(
|
|
selection,
|
|
ToggleState::Selected
|
|
);
|
|
view.toggle_rpc_logging_for_server(
|
|
server_id, enabled, window, cx,
|
|
);
|
|
cx.stop_propagation();
|
|
},
|
|
)),
|
|
),
|
|
)
|
|
.into_any_element()
|
|
}
|
|
},
|
|
window.handler_for(&log_view, move |view, window, cx| {
|
|
view.show_rpc_trace_for_server(server_id, window, cx);
|
|
}),
|
|
)
|
|
})
|
|
.entry(
|
|
SERVER_INFO,
|
|
None,
|
|
window.handler_for(&log_view, move |view, window, cx| {
|
|
view.show_server_info(server_id, window, cx);
|
|
}),
|
|
)
|
|
}))
|
|
})
|
|
});
|
|
h_flex()
|
|
.size_full()
|
|
.justify_between()
|
|
.child(
|
|
h_flex()
|
|
.child(lsp_menu)
|
|
.children(view_selector)
|
|
.child(
|
|
log_view.update(cx, |this, _cx| match this.active_entry_kind {
|
|
LogKind::Trace => {
|
|
let log_view = log_view.clone();
|
|
div().child(
|
|
PopoverMenu::new("lsp-trace-level-menu")
|
|
.anchor(Corner::TopLeft)
|
|
.trigger(Button::new(
|
|
"language_server_trace_level_selector",
|
|
"Trace level",
|
|
))
|
|
.menu({
|
|
let log_view = log_view.clone();
|
|
|
|
move |window, cx| {
|
|
let id = log_view.read(cx).current_server_id?;
|
|
|
|
let trace_level =
|
|
log_view.update(cx, |this, cx| {
|
|
this.log_store.update(cx, |this, _| {
|
|
Some(
|
|
this.get_language_server_state(id)?
|
|
.trace_level,
|
|
)
|
|
})
|
|
})?;
|
|
|
|
ContextMenu::build(
|
|
window,
|
|
cx,
|
|
|mut menu, window, cx| {
|
|
let log_view = log_view.clone();
|
|
|
|
for (option, label) in [
|
|
(TraceValue::Off, "Off"),
|
|
(TraceValue::Messages, "Messages"),
|
|
(TraceValue::Verbose, "Verbose"),
|
|
] {
|
|
menu = menu.entry(label, None, {
|
|
let log_view = log_view.clone();
|
|
move |_, cx| {
|
|
log_view.update(cx, |this, cx| {
|
|
if let Some(id) =
|
|
this.current_server_id
|
|
{
|
|
this.update_trace_level(
|
|
id, option, cx,
|
|
);
|
|
}
|
|
});
|
|
}
|
|
});
|
|
if option == trace_level {
|
|
menu.select_last(window, cx);
|
|
}
|
|
}
|
|
|
|
menu
|
|
},
|
|
)
|
|
.into()
|
|
}
|
|
}),
|
|
)
|
|
}
|
|
LogKind::Logs => {
|
|
let log_view = log_view.clone();
|
|
div().child(
|
|
PopoverMenu::new("lsp-log-level-menu")
|
|
.anchor(Corner::TopLeft)
|
|
.trigger(Button::new(
|
|
"language_server_log_level_selector",
|
|
"Log level",
|
|
))
|
|
.menu({
|
|
let log_view = log_view.clone();
|
|
|
|
move |window, cx| {
|
|
let id = log_view.read(cx).current_server_id?;
|
|
|
|
let log_level =
|
|
log_view.update(cx, |this, cx| {
|
|
this.log_store.update(cx, |this, _| {
|
|
Some(
|
|
this.get_language_server_state(id)?
|
|
.log_level,
|
|
)
|
|
})
|
|
})?;
|
|
|
|
ContextMenu::build(
|
|
window,
|
|
cx,
|
|
|mut menu, window, cx| {
|
|
let log_view = log_view.clone();
|
|
|
|
for (option, label) in [
|
|
(MessageType::LOG, "Log"),
|
|
(MessageType::INFO, "Info"),
|
|
(MessageType::WARNING, "Warning"),
|
|
(MessageType::ERROR, "Error"),
|
|
] {
|
|
menu = menu.entry(label, None, {
|
|
let log_view = log_view.clone();
|
|
move |window, cx| {
|
|
log_view.update(cx, |this, cx| {
|
|
if let Some(id) =
|
|
this.current_server_id
|
|
{
|
|
this.update_log_level(
|
|
id, option, window, cx,
|
|
);
|
|
}
|
|
});
|
|
}
|
|
});
|
|
if option == log_level {
|
|
menu.select_last(window, cx);
|
|
}
|
|
}
|
|
|
|
menu
|
|
},
|
|
)
|
|
.into()
|
|
}
|
|
}),
|
|
)
|
|
}
|
|
_ => div(),
|
|
}),
|
|
),
|
|
)
|
|
.child(
|
|
div()
|
|
.child(
|
|
Button::new("clear_log_button", "Clear").on_click(cx.listener(
|
|
|this, _, window, cx| {
|
|
if let Some(log_view) = this.log_view.as_ref() {
|
|
log_view.update(cx, |log_view, cx| {
|
|
log_view.editor.update(cx, |editor, cx| {
|
|
editor.set_read_only(false);
|
|
editor.clear(window, cx);
|
|
editor.set_read_only(true);
|
|
});
|
|
})
|
|
}
|
|
},
|
|
)),
|
|
)
|
|
.ml_2(),
|
|
)
|
|
}
|
|
}
|
|
|
|
const RPC_MESSAGES: &str = "RPC Messages";
|
|
const SERVER_LOGS: &str = "Server Logs";
|
|
const SERVER_TRACE: &str = "Server Trace";
|
|
const SERVER_INFO: &str = "Server Info";
|
|
|
|
impl Default for LspLogToolbarItemView {
|
|
fn default() -> Self {
|
|
Self::new()
|
|
}
|
|
}
|
|
|
|
impl LspLogToolbarItemView {
|
|
pub fn new() -> Self {
|
|
Self {
|
|
log_view: None,
|
|
_log_view_subscription: None,
|
|
}
|
|
}
|
|
|
|
fn toggle_rpc_logging_for_server(
|
|
&mut self,
|
|
id: LanguageServerId,
|
|
enabled: bool,
|
|
window: &mut Window,
|
|
cx: &mut Context<Self>,
|
|
) {
|
|
if let Some(log_view) = &self.log_view {
|
|
log_view.update(cx, |log_view, cx| {
|
|
log_view.toggle_rpc_trace_for_server(id, enabled, window, cx);
|
|
if !enabled && Some(id) == log_view.current_server_id {
|
|
log_view.show_logs_for_server(id, window, cx);
|
|
cx.notify();
|
|
} else if enabled {
|
|
log_view.show_rpc_trace_for_server(id, window, cx);
|
|
cx.notify();
|
|
}
|
|
window.focus(&log_view.focus_handle);
|
|
});
|
|
}
|
|
cx.notify();
|
|
}
|
|
}
|
|
|
|
pub enum Event {
|
|
NewServerLogEntry {
|
|
id: LanguageServerId,
|
|
entry: String,
|
|
kind: LogKind,
|
|
},
|
|
}
|
|
|
|
impl EventEmitter<Event> for LogStore {}
|
|
impl EventEmitter<Event> for LspLogView {}
|
|
impl EventEmitter<EditorEvent> for LspLogView {}
|
|
impl EventEmitter<SearchEvent> for LspLogView {}
|