Compare commits

..

1 commit

Author SHA1 Message Date
Conrad Irwin
c985789d84 acp-native-rewind 2025-08-25 20:58:03 -06:00
95 changed files with 3936 additions and 6910 deletions

View file

@ -14,7 +14,7 @@ body:
### Description ### Description
<!-- Describe with sufficient detail to reproduce from a clean Zed install. <!-- Describe with sufficient detail to reproduce from a clean Zed install.
- Any code must be sufficient to reproduce (include context!) - Any code must be sufficient to reproduce (include context!)
- Include code as text, not just as a screenshot. - Code must as text, not just as a screenshot.
- Issues with insufficient detail may be summarily closed. - Issues with insufficient detail may be summarily closed.
--> -->

4
Cargo.lock generated
View file

@ -191,9 +191,7 @@ dependencies = [
[[package]] [[package]]
name = "agent-client-protocol" name = "agent-client-protocol"
version = "0.0.31" version = "0.0.32"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "289eb34ee17213dadcca47eedadd386a5e7678094095414e475965d1bcca2860"
dependencies = [ dependencies = [
"anyhow", "anyhow",
"async-broadcast", "async-broadcast",

View file

@ -426,7 +426,7 @@ zlog_settings = { path = "crates/zlog_settings" }
# External crates # External crates
# #
agent-client-protocol = "0.0.31" agent-client-protocol = { path = "../agent-client-protocol"}
aho-corasick = "1.1" aho-corasick = "1.1"
alacritty_terminal = { git = "https://github.com/zed-industries/alacritty.git", branch = "add-hush-login-flag" } alacritty_terminal = { git = "https://github.com/zed-industries/alacritty.git", branch = "add-hush-login-flag" }
any_vec = "0.14" any_vec = "0.14"

View file

@ -1,4 +0,0 @@
<svg width="16" height="16" viewBox="0 0 16 16" fill="none" xmlns="http://www.w3.org/2000/svg">
<path d="M8 12.375H13" stroke="black" stroke-width="1.2" stroke-linecap="round" stroke-linejoin="round"/>
<path d="M3 11.125L6.75003 7.375L3 3.62497" stroke="black" stroke-width="1.2" stroke-linecap="round" stroke-linejoin="round"/>
</svg>

Before

Width:  |  Height:  |  Size: 336 B

File diff suppressed because it is too large Load diff

Before

Width:  |  Height:  |  Size: 176 KiB

View file

@ -1 +0,0 @@
<svg xmlns="http://www.w3.org/2000/svg" width="160" height="61" fill="none"><g clip-path="url(#a)"><path fill="#000" d="M130.75.385c5.428 0 10.297 2.81 13.011 7.511l14.214 24.618-.013-.005c2.599 4.504 2.707 9.932.28 14.513-2.618 4.944-7.862 8.015-13.679 8.015h-31.811c-.452 0-.873-.242-1.103-.637a1.268 1.268 0 0 1 0-1.274l3.919-6.78c.223-.394.65-.636 1.102-.636h28.288a5.622 5.622 0 0 0 4.925-2.849 5.615 5.615 0 0 0 0-5.69l-14.214-24.617a5.621 5.621 0 0 0-4.925-2.848 5.621 5.621 0 0 0-4.925 2.848l-14.214 24.618a6.267 6.267 0 0 0-.319.643.998.998 0 0 1-.069.14L101.724 54.4l-.823 1.313-2.529 4.39a1.27 1.27 0 0 1-1.103.636h-7.83c-.452 0-.873-.242-1.102-.637-.23-.394-.23-.879 0-1.274l2.188-3.791H66.803c-3.32 0-6.454-1.122-8.818-3.167a17.141 17.141 0 0 1-3.394-3.96 1.261 1.261 0 0 1-.091-.137L34.2 12.573a5.622 5.622 0 0 0-4.925-2.849 5.621 5.621 0 0 0-4.924 2.85L10.137 37.19a5.615 5.615 0 0 0 0 5.69 5.63 5.63 0 0 0 4.925 2.841h29.862a1.276 1.276 0 0 1 1.102 1.912l-3.912 6.778a1.27 1.27 0 0 1-1.102.638H14.495c-3.32 0-6.454-1.128-8.817-3.173-5.906-5.104-7.36-12.883-3.62-19.363L16.267 7.89C18.872 3.385 23.517.583 28.697.39c.184-.006.356-.006.534-.006 5.378 0 10.45 3.007 13.246 7.85l12.986 22.372L68.58 7.891C71.186 3.385 75.83.582 81.01.39c.185-.006.358-.006.536-.006 4.453 0 8.71 2.039 11.672 5.588.337.407.388.98.127 1.446l-3.765 6.6a1.268 1.268 0 0 1-2.205.006l-.847-1.465a5.623 5.623 0 0 0-4.926-2.848 5.622 5.622 0 0 0-4.924 2.848L62.464 37.18a5.614 5.614 0 0 0 0 5.689 5.628 5.628 0 0 0 4.925 2.842H95.91L117.76 7.87c2.714-4.683 7.575-7.486 12.99-7.486Z"/></g><defs><clipPath id="a"><path fill="#fff" d="M0 .385h160v60.36H0z"/></clipPath></defs></svg>

Before

Width:  |  Height:  |  Size: 1.6 KiB

File diff suppressed because one or more lines are too long

Before

Width:  |  Height:  |  Size: 14 KiB

View file

@ -40,7 +40,7 @@
"shift-f11": "debugger::StepOut", "shift-f11": "debugger::StepOut",
"f11": "zed::ToggleFullScreen", "f11": "zed::ToggleFullScreen",
"ctrl-alt-z": "edit_prediction::RateCompletions", "ctrl-alt-z": "edit_prediction::RateCompletions",
"ctrl-alt-shift-i": "edit_prediction::ToggleMenu", "ctrl-shift-i": "edit_prediction::ToggleMenu",
"ctrl-alt-l": "lsp_tool::ToggleMenu" "ctrl-alt-l": "lsp_tool::ToggleMenu"
} }
}, },
@ -120,7 +120,7 @@
"alt-g m": "git::OpenModifiedFiles", "alt-g m": "git::OpenModifiedFiles",
"menu": "editor::OpenContextMenu", "menu": "editor::OpenContextMenu",
"shift-f10": "editor::OpenContextMenu", "shift-f10": "editor::OpenContextMenu",
"ctrl-alt-shift-e": "editor::ToggleEditPrediction", "ctrl-shift-e": "editor::ToggleEditPrediction",
"f9": "editor::ToggleBreakpoint", "f9": "editor::ToggleBreakpoint",
"shift-f9": "editor::EditLogBreakpoint" "shift-f9": "editor::EditLogBreakpoint"
} }

File diff suppressed because it is too large Load diff

View file

@ -38,7 +38,6 @@
"alt-;": ["editor::ToggleComments", { "advance_downwards": false }], "alt-;": ["editor::ToggleComments", { "advance_downwards": false }],
"ctrl-x ctrl-;": "editor::ToggleComments", "ctrl-x ctrl-;": "editor::ToggleComments",
"alt-.": "editor::GoToDefinition", // xref-find-definitions "alt-.": "editor::GoToDefinition", // xref-find-definitions
"alt-?": "editor::FindAllReferences", // xref-find-references
"alt-,": "pane::GoBack", // xref-pop-marker-stack "alt-,": "pane::GoBack", // xref-pop-marker-stack
"ctrl-x h": "editor::SelectAll", // mark-whole-buffer "ctrl-x h": "editor::SelectAll", // mark-whole-buffer
"ctrl-d": "editor::Delete", // delete-char "ctrl-d": "editor::Delete", // delete-char

View file

@ -38,7 +38,6 @@
"alt-;": ["editor::ToggleComments", { "advance_downwards": false }], "alt-;": ["editor::ToggleComments", { "advance_downwards": false }],
"ctrl-x ctrl-;": "editor::ToggleComments", "ctrl-x ctrl-;": "editor::ToggleComments",
"alt-.": "editor::GoToDefinition", // xref-find-definitions "alt-.": "editor::GoToDefinition", // xref-find-definitions
"alt-?": "editor::FindAllReferences", // xref-find-references
"alt-,": "pane::GoBack", // xref-pop-marker-stack "alt-,": "pane::GoBack", // xref-pop-marker-stack
"ctrl-x h": "editor::SelectAll", // mark-whole-buffer "ctrl-x h": "editor::SelectAll", // mark-whole-buffer
"ctrl-d": "editor::Delete", // delete-char "ctrl-d": "editor::Delete", // delete-char

View file

@ -428,13 +428,11 @@
"g h": "vim::StartOfLine", "g h": "vim::StartOfLine",
"g s": "vim::FirstNonWhitespace", // "g s" default behavior is "space s" "g s": "vim::FirstNonWhitespace", // "g s" default behavior is "space s"
"g e": "vim::EndOfDocument", "g e": "vim::EndOfDocument",
"g .": "vim::HelixGotoLastModification", // go to last modification
"g r": "editor::FindAllReferences", // zed specific "g r": "editor::FindAllReferences", // zed specific
"g t": "vim::WindowTop", "g t": "vim::WindowTop",
"g c": "vim::WindowMiddle", "g c": "vim::WindowMiddle",
"g b": "vim::WindowBottom", "g b": "vim::WindowBottom",
"shift-r": "editor::Paste",
"x": "editor::SelectLine", "x": "editor::SelectLine",
"shift-x": "editor::SelectLine", "shift-x": "editor::SelectLine",
"%": "editor::SelectAll", "%": "editor::SelectAll",

View file

@ -653,8 +653,6 @@
// "never" // "never"
"show": "always" "show": "always"
}, },
// Whether to enable drag-and-drop operations in the project panel.
"drag_and_drop": true,
// Whether to hide the root entry when only one folder is open in the window. // Whether to hide the root entry when only one folder is open in the window.
"hide_root": false "hide_root": false
}, },

View file

@ -43,8 +43,8 @@
// "args": ["--login"] // "args": ["--login"]
// } // }
// } // }
"shell": "system" "shell": "system",
// Represents the tags for inline runnable indicators, or spawning multiple tasks at once. // Represents the tags for inline runnable indicators, or spawning multiple tasks at once.
// "tags": [] "tags": []
} }
] ]

View file

@ -32,10 +32,15 @@ use std::time::{Duration, Instant};
use std::{fmt::Display, mem, path::PathBuf, sync::Arc}; use std::{fmt::Display, mem, path::PathBuf, sync::Arc};
use ui::App; use ui::App;
use util::ResultExt; use util::ResultExt;
use uuid::Uuid;
pub fn new_prompt_id() -> acp::PromptId {
acp::PromptId(Uuid::new_v4().to_string().into())
}
#[derive(Debug)] #[derive(Debug)]
pub struct UserMessage { pub struct UserMessage {
pub id: Option<UserMessageId>, pub prompt_id: Option<acp::PromptId>,
pub content: ContentBlock, pub content: ContentBlock,
pub chunks: Vec<acp::ContentBlock>, pub chunks: Vec<acp::ContentBlock>,
pub checkpoint: Option<Checkpoint>, pub checkpoint: Option<Checkpoint>,
@ -962,7 +967,7 @@ impl AcpThread {
pub fn push_user_content_block( pub fn push_user_content_block(
&mut self, &mut self,
message_id: Option<UserMessageId>, prompt_id: Option<acp::PromptId>,
chunk: acp::ContentBlock, chunk: acp::ContentBlock,
cx: &mut Context<Self>, cx: &mut Context<Self>,
) { ) {
@ -971,13 +976,13 @@ impl AcpThread {
if let Some(last_entry) = self.entries.last_mut() if let Some(last_entry) = self.entries.last_mut()
&& let AgentThreadEntry::UserMessage(UserMessage { && let AgentThreadEntry::UserMessage(UserMessage {
id, prompt_id: id,
content, content,
chunks, chunks,
.. ..
}) = last_entry }) = last_entry
{ {
*id = message_id.or(id.take()); *id = prompt_id.or(id.take());
content.append(chunk.clone(), &language_registry, cx); content.append(chunk.clone(), &language_registry, cx);
chunks.push(chunk); chunks.push(chunk);
let idx = entries_len - 1; let idx = entries_len - 1;
@ -986,7 +991,7 @@ impl AcpThread {
let content = ContentBlock::new(chunk.clone(), &language_registry, cx); let content = ContentBlock::new(chunk.clone(), &language_registry, cx);
self.push_entry( self.push_entry(
AgentThreadEntry::UserMessage(UserMessage { AgentThreadEntry::UserMessage(UserMessage {
id: message_id, prompt_id,
content, content,
chunks: vec![chunk], chunks: vec![chunk],
checkpoint: None, checkpoint: None,
@ -1336,6 +1341,7 @@ impl AcpThread {
cx: &mut Context<Self>, cx: &mut Context<Self>,
) -> BoxFuture<'static, Result<()>> { ) -> BoxFuture<'static, Result<()>> {
self.send( self.send(
new_prompt_id(),
vec![acp::ContentBlock::Text(acp::TextContent { vec![acp::ContentBlock::Text(acp::TextContent {
text: message.to_string(), text: message.to_string(),
annotations: None, annotations: None,
@ -1346,6 +1352,7 @@ impl AcpThread {
pub fn send( pub fn send(
&mut self, &mut self,
prompt_id: acp::PromptId,
message: Vec<acp::ContentBlock>, message: Vec<acp::ContentBlock>,
cx: &mut Context<Self>, cx: &mut Context<Self>,
) -> BoxFuture<'static, Result<()>> { ) -> BoxFuture<'static, Result<()>> {
@ -1355,22 +1362,17 @@ impl AcpThread {
cx, cx,
); );
let request = acp::PromptRequest { let request = acp::PromptRequest {
prompt_id: Some(prompt_id.clone()),
prompt: message.clone(), prompt: message.clone(),
session_id: self.session_id.clone(), session_id: self.session_id.clone(),
}; };
let git_store = self.project.read(cx).git_store().clone(); let git_store = self.project.read(cx).git_store().clone();
let message_id = if self.connection.truncate(&self.session_id, cx).is_some() {
Some(UserMessageId::new())
} else {
None
};
self.run_turn(cx, async move |this, cx| { self.run_turn(cx, async move |this, cx| {
this.update(cx, |this, cx| { this.update(cx, |this, cx| {
this.push_entry( this.push_entry(
AgentThreadEntry::UserMessage(UserMessage { AgentThreadEntry::UserMessage(UserMessage {
id: message_id.clone(), prompt_id: Some(prompt_id),
content: block, content: block,
chunks: message, chunks: message,
checkpoint: None, checkpoint: None,
@ -1392,7 +1394,7 @@ impl AcpThread {
show: false, show: false,
}); });
} }
this.connection.prompt(message_id, request, cx) this.connection.prompt(request, cx)
})? })?
.await .await
}) })
@ -1509,8 +1511,8 @@ impl AcpThread {
/// Rewinds this thread to before the entry at `index`, removing it and all /// Rewinds this thread to before the entry at `index`, removing it and all
/// subsequent entries while reverting any changes made from that point. /// subsequent entries while reverting any changes made from that point.
pub fn rewind(&mut self, id: UserMessageId, cx: &mut Context<Self>) -> Task<Result<()>> { pub fn rewind(&mut self, id: acp::PromptId, cx: &mut Context<Self>) -> Task<Result<()>> {
let Some(truncate) = self.connection.truncate(&self.session_id, cx) else { let Some(rewind) = self.connection.rewind(&self.session_id, cx) else {
return Task::ready(Err(anyhow!("not supported"))); return Task::ready(Err(anyhow!("not supported")));
}; };
let Some(message) = self.user_message(&id) else { let Some(message) = self.user_message(&id) else {
@ -1530,7 +1532,7 @@ impl AcpThread {
.await?; .await?;
} }
cx.update(|cx| truncate.run(id.clone(), cx))?.await?; cx.update(|cx| rewind.rewind(id.clone(), cx))?.await?;
this.update(cx, |this, cx| { this.update(cx, |this, cx| {
if let Some((ix, _)) = this.user_message_mut(&id) { if let Some((ix, _)) = this.user_message_mut(&id) {
let range = ix..this.entries.len(); let range = ix..this.entries.len();
@ -1594,10 +1596,10 @@ impl AcpThread {
}) })
} }
fn user_message(&self, id: &UserMessageId) -> Option<&UserMessage> { fn user_message(&self, id: &acp::PromptId) -> Option<&UserMessage> {
self.entries.iter().find_map(|entry| { self.entries.iter().find_map(|entry| {
if let AgentThreadEntry::UserMessage(message) = entry { if let AgentThreadEntry::UserMessage(message) = entry {
if message.id.as_ref() == Some(id) { if message.prompt_id.as_ref() == Some(id) {
Some(message) Some(message)
} else { } else {
None None
@ -1608,10 +1610,10 @@ impl AcpThread {
}) })
} }
fn user_message_mut(&mut self, id: &UserMessageId) -> Option<(usize, &mut UserMessage)> { fn user_message_mut(&mut self, id: &acp::PromptId) -> Option<(usize, &mut UserMessage)> {
self.entries.iter_mut().enumerate().find_map(|(ix, entry)| { self.entries.iter_mut().enumerate().find_map(|(ix, entry)| {
if let AgentThreadEntry::UserMessage(message) = entry { if let AgentThreadEntry::UserMessage(message) = entry {
if message.id.as_ref() == Some(id) { if message.prompt_id.as_ref() == Some(id) {
Some((ix, message)) Some((ix, message))
} else { } else {
None None
@ -1905,7 +1907,7 @@ mod tests {
thread.update(cx, |thread, cx| { thread.update(cx, |thread, cx| {
assert_eq!(thread.entries.len(), 1); assert_eq!(thread.entries.len(), 1);
if let AgentThreadEntry::UserMessage(user_msg) = &thread.entries[0] { if let AgentThreadEntry::UserMessage(user_msg) = &thread.entries[0] {
assert_eq!(user_msg.id, None); assert_eq!(user_msg.prompt_id, None);
assert_eq!(user_msg.content.to_markdown(cx), "Hello, "); assert_eq!(user_msg.content.to_markdown(cx), "Hello, ");
} else { } else {
panic!("Expected UserMessage"); panic!("Expected UserMessage");
@ -1913,7 +1915,7 @@ mod tests {
}); });
// Test appending to existing user message // Test appending to existing user message
let message_1_id = UserMessageId::new(); let message_1_id = new_prompt_id();
thread.update(cx, |thread, cx| { thread.update(cx, |thread, cx| {
thread.push_user_content_block( thread.push_user_content_block(
Some(message_1_id.clone()), Some(message_1_id.clone()),
@ -1928,7 +1930,7 @@ mod tests {
thread.update(cx, |thread, cx| { thread.update(cx, |thread, cx| {
assert_eq!(thread.entries.len(), 1); assert_eq!(thread.entries.len(), 1);
if let AgentThreadEntry::UserMessage(user_msg) = &thread.entries[0] { if let AgentThreadEntry::UserMessage(user_msg) = &thread.entries[0] {
assert_eq!(user_msg.id, Some(message_1_id)); assert_eq!(user_msg.prompt_id, Some(message_1_id));
assert_eq!(user_msg.content.to_markdown(cx), "Hello, world!"); assert_eq!(user_msg.content.to_markdown(cx), "Hello, world!");
} else { } else {
panic!("Expected UserMessage"); panic!("Expected UserMessage");
@ -1947,7 +1949,7 @@ mod tests {
); );
}); });
let message_2_id = UserMessageId::new(); let message_2_id = new_prompt_id();
thread.update(cx, |thread, cx| { thread.update(cx, |thread, cx| {
thread.push_user_content_block( thread.push_user_content_block(
Some(message_2_id.clone()), Some(message_2_id.clone()),
@ -1962,7 +1964,7 @@ mod tests {
thread.update(cx, |thread, cx| { thread.update(cx, |thread, cx| {
assert_eq!(thread.entries.len(), 3); assert_eq!(thread.entries.len(), 3);
if let AgentThreadEntry::UserMessage(user_msg) = &thread.entries[2] { if let AgentThreadEntry::UserMessage(user_msg) = &thread.entries[2] {
assert_eq!(user_msg.id, Some(message_2_id)); assert_eq!(user_msg.prompt_id, Some(message_2_id));
assert_eq!(user_msg.content.to_markdown(cx), "New user message"); assert_eq!(user_msg.content.to_markdown(cx), "New user message");
} else { } else {
panic!("Expected UserMessage at index 2"); panic!("Expected UserMessage at index 2");
@ -2259,9 +2261,13 @@ mod tests {
.await .await
.unwrap(); .unwrap();
cx.update(|cx| thread.update(cx, |thread, cx| thread.send(vec!["Hi".into()], cx))) cx.update(|cx| {
.await thread.update(cx, |thread, cx| {
.unwrap(); thread.send(new_prompt_id(), vec!["Hi".into()], cx)
})
})
.await
.unwrap();
assert!(cx.read(|cx| !thread.read(cx).has_pending_edit_tool_calls())); assert!(cx.read(|cx| !thread.read(cx).has_pending_edit_tool_calls()));
} }
@ -2320,9 +2326,13 @@ mod tests {
.await .await
.unwrap(); .unwrap();
cx.update(|cx| thread.update(cx, |thread, cx| thread.send(vec!["Lorem".into()], cx))) cx.update(|cx| {
.await thread.update(cx, |thread, cx| {
.unwrap(); thread.send(new_prompt_id(), vec!["Lorem".into()], cx)
})
})
.await
.unwrap();
thread.read_with(cx, |thread, cx| { thread.read_with(cx, |thread, cx| {
assert_eq!( assert_eq!(
thread.to_markdown(cx), thread.to_markdown(cx),
@ -2340,9 +2350,13 @@ mod tests {
}); });
assert_eq!(fs.files(), vec![Path::new(path!("/test/file-0"))]); assert_eq!(fs.files(), vec![Path::new(path!("/test/file-0"))]);
cx.update(|cx| thread.update(cx, |thread, cx| thread.send(vec!["ipsum".into()], cx))) cx.update(|cx| {
.await thread.update(cx, |thread, cx| {
.unwrap(); thread.send(new_prompt_id(), vec!["ipsum".into()], cx)
})
})
.await
.unwrap();
thread.read_with(cx, |thread, cx| { thread.read_with(cx, |thread, cx| {
assert_eq!( assert_eq!(
thread.to_markdown(cx), thread.to_markdown(cx),
@ -2376,9 +2390,13 @@ mod tests {
// Checkpoint isn't stored when there are no changes. // Checkpoint isn't stored when there are no changes.
simulate_changes.store(false, SeqCst); simulate_changes.store(false, SeqCst);
cx.update(|cx| thread.update(cx, |thread, cx| thread.send(vec!["dolor".into()], cx))) cx.update(|cx| {
.await thread.update(cx, |thread, cx| {
.unwrap(); thread.send(new_prompt_id(), vec!["dolor".into()], cx)
})
})
.await
.unwrap();
thread.read_with(cx, |thread, cx| { thread.read_with(cx, |thread, cx| {
assert_eq!( assert_eq!(
thread.to_markdown(cx), thread.to_markdown(cx),
@ -2424,7 +2442,7 @@ mod tests {
let AgentThreadEntry::UserMessage(message) = &thread.entries[2] else { let AgentThreadEntry::UserMessage(message) = &thread.entries[2] else {
panic!("unexpected entries {:?}", thread.entries) panic!("unexpected entries {:?}", thread.entries)
}; };
thread.rewind(message.id.clone().unwrap(), cx) thread.rewind(message.prompt_id.clone().unwrap(), cx)
}) })
.await .await
.unwrap(); .unwrap();
@ -2490,9 +2508,13 @@ mod tests {
.await .await
.unwrap(); .unwrap();
cx.update(|cx| thread.update(cx, |thread, cx| thread.send(vec!["hello".into()], cx))) cx.update(|cx| {
.await thread.update(cx, |thread, cx| {
.unwrap(); thread.send(new_prompt_id(), vec!["hello".into()], cx)
})
})
.await
.unwrap();
thread.read_with(cx, |thread, cx| { thread.read_with(cx, |thread, cx| {
assert_eq!( assert_eq!(
thread.to_markdown(cx), thread.to_markdown(cx),
@ -2512,9 +2534,13 @@ mod tests {
// Simulate refusing the second message, ensuring the conversation gets // Simulate refusing the second message, ensuring the conversation gets
// truncated to before sending it. // truncated to before sending it.
refuse_next.store(true, SeqCst); refuse_next.store(true, SeqCst);
cx.update(|cx| thread.update(cx, |thread, cx| thread.send(vec!["world".into()], cx))) cx.update(|cx| {
.await thread.update(cx, |thread, cx| {
.unwrap(); thread.send(new_prompt_id(), vec!["world".into()], cx)
})
})
.await
.unwrap();
thread.read_with(cx, |thread, cx| { thread.read_with(cx, |thread, cx| {
assert_eq!( assert_eq!(
thread.to_markdown(cx), thread.to_markdown(cx),
@ -2653,7 +2679,6 @@ mod tests {
fn prompt( fn prompt(
&self, &self,
_id: Option<UserMessageId>,
params: acp::PromptRequest, params: acp::PromptRequest,
cx: &mut App, cx: &mut App,
) -> Task<gpui::Result<acp::PromptResponse>> { ) -> Task<gpui::Result<acp::PromptResponse>> {
@ -2683,11 +2708,11 @@ mod tests {
.detach(); .detach();
} }
fn truncate( fn rewind(
&self, &self,
session_id: &acp::SessionId, session_id: &acp::SessionId,
_cx: &App, _cx: &App,
) -> Option<Rc<dyn AgentSessionTruncate>> { ) -> Option<Rc<dyn AgentSessionRewind>> {
Some(Rc::new(FakeAgentSessionEditor { Some(Rc::new(FakeAgentSessionEditor {
_session_id: session_id.clone(), _session_id: session_id.clone(),
})) }))
@ -2702,8 +2727,8 @@ mod tests {
_session_id: acp::SessionId, _session_id: acp::SessionId,
} }
impl AgentSessionTruncate for FakeAgentSessionEditor { impl AgentSessionRewind for FakeAgentSessionEditor {
fn run(&self, _message_id: UserMessageId, _cx: &mut App) -> Task<Result<()>> { fn rewind(&self, _message_id: acp::PromptId, _cx: &mut App) -> Task<Result<()>> {
Task::ready(Ok(())) Task::ready(Ok(()))
} }
} }

View file

@ -5,19 +5,8 @@ use collections::IndexMap;
use gpui::{Entity, SharedString, Task}; use gpui::{Entity, SharedString, Task};
use language_model::LanguageModelProviderId; use language_model::LanguageModelProviderId;
use project::Project; use project::Project;
use serde::{Deserialize, Serialize}; use std::{any::Any, error::Error, fmt, path::Path, rc::Rc};
use std::{any::Any, error::Error, fmt, path::Path, rc::Rc, sync::Arc};
use ui::{App, IconName}; use ui::{App, IconName};
use uuid::Uuid;
#[derive(Debug, Clone, PartialEq, Eq, Serialize, Deserialize, Hash)]
pub struct UserMessageId(Arc<str>);
impl UserMessageId {
pub fn new() -> Self {
Self(Uuid::new_v4().to_string().into())
}
}
pub trait AgentConnection { pub trait AgentConnection {
fn new_thread( fn new_thread(
@ -31,12 +20,8 @@ pub trait AgentConnection {
fn authenticate(&self, method: acp::AuthMethodId, cx: &mut App) -> Task<Result<()>>; fn authenticate(&self, method: acp::AuthMethodId, cx: &mut App) -> Task<Result<()>>;
fn prompt( fn prompt(&self, params: acp::PromptRequest, cx: &mut App)
&self, -> Task<Result<acp::PromptResponse>>;
user_message_id: Option<UserMessageId>,
params: acp::PromptRequest,
cx: &mut App,
) -> Task<Result<acp::PromptResponse>>;
fn resume( fn resume(
&self, &self,
@ -48,11 +33,11 @@ pub trait AgentConnection {
fn cancel(&self, session_id: &acp::SessionId, cx: &mut App); fn cancel(&self, session_id: &acp::SessionId, cx: &mut App);
fn truncate( fn rewind(
&self, &self,
_session_id: &acp::SessionId, _session_id: &acp::SessionId,
_cx: &App, _cx: &App,
) -> Option<Rc<dyn AgentSessionTruncate>> { ) -> Option<Rc<dyn AgentSessionRewind>> {
None None
} }
@ -85,8 +70,8 @@ impl dyn AgentConnection {
} }
} }
pub trait AgentSessionTruncate { pub trait AgentSessionRewind {
fn run(&self, message_id: UserMessageId, cx: &mut App) -> Task<Result<()>>; fn rewind(&self, message_id: acp::PromptId, cx: &mut App) -> Task<Result<()>>;
} }
pub trait AgentSessionResume { pub trait AgentSessionResume {
@ -362,7 +347,6 @@ mod test_support {
fn prompt( fn prompt(
&self, &self,
_id: Option<UserMessageId>,
params: acp::PromptRequest, params: acp::PromptRequest,
cx: &mut App, cx: &mut App,
) -> Task<gpui::Result<acp::PromptResponse>> { ) -> Task<gpui::Result<acp::PromptResponse>> {
@ -432,11 +416,11 @@ mod test_support {
} }
} }
fn truncate( fn rewind(
&self, &self,
_session_id: &agent_client_protocol::SessionId, _session_id: &agent_client_protocol::SessionId,
_cx: &App, _cx: &App,
) -> Option<Rc<dyn AgentSessionTruncate>> { ) -> Option<Rc<dyn AgentSessionRewind>> {
Some(Rc::new(StubAgentSessionEditor)) Some(Rc::new(StubAgentSessionEditor))
} }
@ -447,8 +431,8 @@ mod test_support {
struct StubAgentSessionEditor; struct StubAgentSessionEditor;
impl AgentSessionTruncate for StubAgentSessionEditor { impl AgentSessionRewind for StubAgentSessionEditor {
fn run(&self, _: UserMessageId, _: &mut App) -> Task<Result<()>> { fn rewind(&self, _: acp::PromptId, _: &mut App) -> Task<Result<()>> {
Task::ready(Ok(())) Task::ready(Ok(()))
} }
} }

View file

@ -664,7 +664,7 @@ impl Thread {
} }
pub fn get_or_init_configured_model(&mut self, cx: &App) -> Option<ConfiguredModel> { pub fn get_or_init_configured_model(&mut self, cx: &App) -> Option<ConfiguredModel> {
if self.configured_model.is_none() { if self.configured_model.is_none() || self.messages.is_empty() {
self.configured_model = LanguageModelRegistry::read_global(cx).default_model(); self.configured_model = LanguageModelRegistry::read_global(cx).default_model();
} }
self.configured_model.clone() self.configured_model.clone()
@ -2097,7 +2097,7 @@ impl Thread {
} }
pub fn summarize(&mut self, cx: &mut Context<Self>) { pub fn summarize(&mut self, cx: &mut Context<Self>) {
let Some(model) = LanguageModelRegistry::read_global(cx).thread_summary_model() else { let Some(model) = LanguageModelRegistry::read_global(cx).thread_summary_model(cx) else {
println!("No thread summary model"); println!("No thread summary model");
return; return;
}; };
@ -2416,7 +2416,7 @@ impl Thread {
} }
let Some(ConfiguredModel { model, provider }) = let Some(ConfiguredModel { model, provider }) =
LanguageModelRegistry::read_global(cx).thread_summary_model() LanguageModelRegistry::read_global(cx).thread_summary_model(cx)
else { else {
return; return;
}; };
@ -5410,13 +5410,10 @@ fn main() {{
}), }),
cx, cx,
); );
registry.set_thread_summary_model( registry.set_thread_summary_model(Some(ConfiguredModel {
Some(ConfiguredModel { provider,
provider, model: model.clone(),
model: model.clone(), }));
}),
cx,
);
}) })
}); });

View file

@ -228,7 +228,7 @@ impl NativeAgent {
) -> Entity<AcpThread> { ) -> Entity<AcpThread> {
let connection = Rc::new(NativeAgentConnection(cx.entity())); let connection = Rc::new(NativeAgentConnection(cx.entity()));
let registry = LanguageModelRegistry::read_global(cx); let registry = LanguageModelRegistry::read_global(cx);
let summarization_model = registry.thread_summary_model().map(|c| c.model); let summarization_model = registry.thread_summary_model(cx).map(|c| c.model);
thread_handle.update(cx, |thread, cx| { thread_handle.update(cx, |thread, cx| {
thread.set_summarization_model(summarization_model, cx); thread.set_summarization_model(summarization_model, cx);
@ -524,7 +524,7 @@ impl NativeAgent {
let registry = LanguageModelRegistry::read_global(cx); let registry = LanguageModelRegistry::read_global(cx);
let default_model = registry.default_model().map(|m| m.model); let default_model = registry.default_model().map(|m| m.model);
let summarization_model = registry.thread_summary_model().map(|m| m.model); let summarization_model = registry.thread_summary_model(cx).map(|m| m.model);
for session in self.sessions.values_mut() { for session in self.sessions.values_mut() {
session.thread.update(cx, |thread, cx| { session.thread.update(cx, |thread, cx| {
@ -905,11 +905,10 @@ impl acp_thread::AgentConnection for NativeAgentConnection {
fn prompt( fn prompt(
&self, &self,
id: Option<acp_thread::UserMessageId>,
params: acp::PromptRequest, params: acp::PromptRequest,
cx: &mut App, cx: &mut App,
) -> Task<Result<acp::PromptResponse>> { ) -> Task<Result<acp::PromptResponse>> {
let id = id.expect("UserMessageId is required"); let id = params.prompt_id.expect("UserMessageId is required");
let session_id = params.session_id.clone(); let session_id = params.session_id.clone();
log::info!("Received prompt request for session: {}", session_id); log::info!("Received prompt request for session: {}", session_id);
log::debug!("Prompt blocks count: {}", params.prompt.len()); log::debug!("Prompt blocks count: {}", params.prompt.len());
@ -948,11 +947,11 @@ impl acp_thread::AgentConnection for NativeAgentConnection {
}); });
} }
fn truncate( fn rewind(
&self, &self,
session_id: &agent_client_protocol::SessionId, session_id: &agent_client_protocol::SessionId,
cx: &App, cx: &App,
) -> Option<Rc<dyn acp_thread::AgentSessionTruncate>> { ) -> Option<Rc<dyn acp_thread::AgentSessionRewind>> {
self.0.read_with(cx, |agent, _cx| { self.0.read_with(cx, |agent, _cx| {
agent.sessions.get(session_id).map(|session| { agent.sessions.get(session_id).map(|session| {
Rc::new(NativeAgentSessionEditor { Rc::new(NativeAgentSessionEditor {
@ -1009,10 +1008,10 @@ struct NativeAgentSessionEditor {
acp_thread: WeakEntity<AcpThread>, acp_thread: WeakEntity<AcpThread>,
} }
impl acp_thread::AgentSessionTruncate for NativeAgentSessionEditor { impl acp_thread::AgentSessionRewind for NativeAgentSessionEditor {
fn run(&self, message_id: acp_thread::UserMessageId, cx: &mut App) -> Task<Result<()>> { fn rewind(&self, message_id: acp::PromptId, cx: &mut App) -> Task<Result<()>> {
match self.thread.update(cx, |thread, cx| { match self.thread.update(cx, |thread, cx| {
thread.truncate(message_id.clone(), cx)?; thread.rewind(message_id.clone(), cx)?;
Ok(thread.latest_token_usage()) Ok(thread.latest_token_usage())
}) { }) {
Ok(usage) => { Ok(usage) => {
@ -1065,6 +1064,7 @@ mod tests {
use super::*; use super::*;
use acp_thread::{ use acp_thread::{
AgentConnection, AgentModelGroupName, AgentModelId, AgentModelInfo, MentionUri, AgentConnection, AgentModelGroupName, AgentModelId, AgentModelInfo, MentionUri,
new_prompt_id,
}; };
use fs::FakeFs; use fs::FakeFs;
use gpui::TestAppContext; use gpui::TestAppContext;
@ -1311,6 +1311,7 @@ mod tests {
let send = acp_thread.update(cx, |thread, cx| { let send = acp_thread.update(cx, |thread, cx| {
thread.send( thread.send(
new_prompt_id(),
vec![ vec![
"What does ".into(), "What does ".into(),
acp::ContentBlock::ResourceLink(acp::ResourceLink { acp::ContentBlock::ResourceLink(acp::ResourceLink {

View file

@ -1,5 +1,5 @@
use crate::{AgentMessage, AgentMessageContent, UserMessage, UserMessageContent}; use crate::{AgentMessage, AgentMessageContent, UserMessage, UserMessageContent};
use acp_thread::UserMessageId; use acp_thread::new_prompt_id;
use agent::{thread::DetailedSummaryState, thread_store}; use agent::{thread::DetailedSummaryState, thread_store};
use agent_client_protocol as acp; use agent_client_protocol as acp;
use agent_settings::{AgentProfileId, CompletionMode}; use agent_settings::{AgentProfileId, CompletionMode};
@ -43,7 +43,7 @@ pub struct DbThread {
#[serde(default)] #[serde(default)]
pub cumulative_token_usage: language_model::TokenUsage, pub cumulative_token_usage: language_model::TokenUsage,
#[serde(default)] #[serde(default)]
pub request_token_usage: HashMap<acp_thread::UserMessageId, language_model::TokenUsage>, pub request_token_usage: HashMap<acp::PromptId, language_model::TokenUsage>,
#[serde(default)] #[serde(default)]
pub model: Option<DbLanguageModel>, pub model: Option<DbLanguageModel>,
#[serde(default)] #[serde(default)]
@ -97,7 +97,7 @@ impl DbThread {
content.push(UserMessageContent::Text(msg.context)); content.push(UserMessageContent::Text(msg.context));
} }
let id = UserMessageId::new(); let id = new_prompt_id();
last_user_message_id = Some(id.clone()); last_user_message_id = Some(id.clone());
crate::Message::User(UserMessage { crate::Message::User(UserMessage {

View file

@ -1,5 +1,5 @@
use super::*; use super::*;
use acp_thread::{AgentConnection, AgentModelGroupName, AgentModelList, UserMessageId}; use acp_thread::{AgentConnection, AgentModelGroupName, AgentModelList, new_prompt_id};
use agent_client_protocol::{self as acp}; use agent_client_protocol::{self as acp};
use agent_settings::AgentProfileId; use agent_settings::AgentProfileId;
use anyhow::Result; use anyhow::Result;
@ -48,7 +48,7 @@ async fn test_echo(cx: &mut TestAppContext) {
let events = thread let events = thread
.update(cx, |thread, cx| { .update(cx, |thread, cx| {
thread.send(UserMessageId::new(), ["Testing: Reply with 'Hello'"], cx) thread.send(new_prompt_id(), ["Testing: Reply with 'Hello'"], cx)
}) })
.unwrap(); .unwrap();
cx.run_until_parked(); cx.run_until_parked();
@ -72,7 +72,6 @@ async fn test_echo(cx: &mut TestAppContext) {
} }
#[gpui::test] #[gpui::test]
#[cfg_attr(target_os = "windows", ignore)] // TODO: Fix this test on Windows
async fn test_thinking(cx: &mut TestAppContext) { async fn test_thinking(cx: &mut TestAppContext) {
let ThreadTest { model, thread, .. } = setup(cx, TestModel::Fake).await; let ThreadTest { model, thread, .. } = setup(cx, TestModel::Fake).await;
let fake_model = model.as_fake(); let fake_model = model.as_fake();
@ -80,7 +79,7 @@ async fn test_thinking(cx: &mut TestAppContext) {
let events = thread let events = thread
.update(cx, |thread, cx| { .update(cx, |thread, cx| {
thread.send( thread.send(
UserMessageId::new(), new_prompt_id(),
[indoc! {" [indoc! {"
Testing: Testing:
@ -131,9 +130,7 @@ async fn test_system_prompt(cx: &mut TestAppContext) {
}); });
thread.update(cx, |thread, _| thread.add_tool(EchoTool)); thread.update(cx, |thread, _| thread.add_tool(EchoTool));
thread thread
.update(cx, |thread, cx| { .update(cx, |thread, cx| thread.send(new_prompt_id(), ["abc"], cx))
thread.send(UserMessageId::new(), ["abc"], cx)
})
.unwrap(); .unwrap();
cx.run_until_parked(); cx.run_until_parked();
let mut pending_completions = fake_model.pending_completions(); let mut pending_completions = fake_model.pending_completions();
@ -169,7 +166,7 @@ async fn test_prompt_caching(cx: &mut TestAppContext) {
// Send initial user message and verify it's cached // Send initial user message and verify it's cached
thread thread
.update(cx, |thread, cx| { .update(cx, |thread, cx| {
thread.send(UserMessageId::new(), ["Message 1"], cx) thread.send(new_prompt_id(), ["Message 1"], cx)
}) })
.unwrap(); .unwrap();
cx.run_until_parked(); cx.run_until_parked();
@ -192,7 +189,7 @@ async fn test_prompt_caching(cx: &mut TestAppContext) {
// Send another user message and verify only the latest is cached // Send another user message and verify only the latest is cached
thread thread
.update(cx, |thread, cx| { .update(cx, |thread, cx| {
thread.send(UserMessageId::new(), ["Message 2"], cx) thread.send(new_prompt_id(), ["Message 2"], cx)
}) })
.unwrap(); .unwrap();
cx.run_until_parked(); cx.run_until_parked();
@ -228,7 +225,7 @@ async fn test_prompt_caching(cx: &mut TestAppContext) {
thread.update(cx, |thread, _| thread.add_tool(EchoTool)); thread.update(cx, |thread, _| thread.add_tool(EchoTool));
thread thread
.update(cx, |thread, cx| { .update(cx, |thread, cx| {
thread.send(UserMessageId::new(), ["Use the echo tool"], cx) thread.send(new_prompt_id(), ["Use the echo tool"], cx)
}) })
.unwrap(); .unwrap();
cx.run_until_parked(); cx.run_until_parked();
@ -305,7 +302,7 @@ async fn test_basic_tool_calls(cx: &mut TestAppContext) {
.update(cx, |thread, cx| { .update(cx, |thread, cx| {
thread.add_tool(EchoTool); thread.add_tool(EchoTool);
thread.send( thread.send(
UserMessageId::new(), new_prompt_id(),
["Now test the echo tool with 'Hello'. Does it work? Say 'Yes' or 'No'."], ["Now test the echo tool with 'Hello'. Does it work? Say 'Yes' or 'No'."],
cx, cx,
) )
@ -321,7 +318,7 @@ async fn test_basic_tool_calls(cx: &mut TestAppContext) {
thread.remove_tool(&EchoTool::name()); thread.remove_tool(&EchoTool::name());
thread.add_tool(DelayTool); thread.add_tool(DelayTool);
thread.send( thread.send(
UserMessageId::new(), new_prompt_id(),
[ [
"Now call the delay tool with 200ms.", "Now call the delay tool with 200ms.",
"When the timer goes off, then you echo the output of the tool.", "When the timer goes off, then you echo the output of the tool.",
@ -364,7 +361,7 @@ async fn test_streaming_tool_calls(cx: &mut TestAppContext) {
let mut events = thread let mut events = thread
.update(cx, |thread, cx| { .update(cx, |thread, cx| {
thread.add_tool(WordListTool); thread.add_tool(WordListTool);
thread.send(UserMessageId::new(), ["Test the word_list tool."], cx) thread.send(new_prompt_id(), ["Test the word_list tool."], cx)
}) })
.unwrap(); .unwrap();
@ -415,7 +412,7 @@ async fn test_tool_authorization(cx: &mut TestAppContext) {
let mut events = thread let mut events = thread
.update(cx, |thread, cx| { .update(cx, |thread, cx| {
thread.add_tool(ToolRequiringPermission); thread.add_tool(ToolRequiringPermission);
thread.send(UserMessageId::new(), ["abc"], cx) thread.send(new_prompt_id(), ["abc"], cx)
}) })
.unwrap(); .unwrap();
cx.run_until_parked(); cx.run_until_parked();
@ -472,7 +469,7 @@ async fn test_tool_authorization(cx: &mut TestAppContext) {
tool_name: ToolRequiringPermission::name().into(), tool_name: ToolRequiringPermission::name().into(),
is_error: true, is_error: true,
content: "Permission to run tool denied by user".into(), content: "Permission to run tool denied by user".into(),
output: Some("Permission to run tool denied by user".into()) output: None
}) })
] ]
); );
@ -545,9 +542,7 @@ async fn test_tool_hallucination(cx: &mut TestAppContext) {
let fake_model = model.as_fake(); let fake_model = model.as_fake();
let mut events = thread let mut events = thread
.update(cx, |thread, cx| { .update(cx, |thread, cx| thread.send(new_prompt_id(), ["abc"], cx))
thread.send(UserMessageId::new(), ["abc"], cx)
})
.unwrap(); .unwrap();
cx.run_until_parked(); cx.run_until_parked();
fake_model.send_last_completion_stream_event(LanguageModelCompletionEvent::ToolUse( fake_model.send_last_completion_stream_event(LanguageModelCompletionEvent::ToolUse(
@ -576,7 +571,7 @@ async fn test_resume_after_tool_use_limit(cx: &mut TestAppContext) {
let events = thread let events = thread
.update(cx, |thread, cx| { .update(cx, |thread, cx| {
thread.add_tool(EchoTool); thread.add_tool(EchoTool);
thread.send(UserMessageId::new(), ["abc"], cx) thread.send(new_prompt_id(), ["abc"], cx)
}) })
.unwrap(); .unwrap();
cx.run_until_parked(); cx.run_until_parked();
@ -685,7 +680,7 @@ async fn test_send_after_tool_use_limit(cx: &mut TestAppContext) {
let events = thread let events = thread
.update(cx, |thread, cx| { .update(cx, |thread, cx| {
thread.add_tool(EchoTool); thread.add_tool(EchoTool);
thread.send(UserMessageId::new(), ["abc"], cx) thread.send(new_prompt_id(), ["abc"], cx)
}) })
.unwrap(); .unwrap();
cx.run_until_parked(); cx.run_until_parked();
@ -719,7 +714,7 @@ async fn test_send_after_tool_use_limit(cx: &mut TestAppContext) {
thread thread
.update(cx, |thread, cx| { .update(cx, |thread, cx| {
thread.send(UserMessageId::new(), vec!["ghi"], cx) thread.send(new_prompt_id(), vec!["ghi"], cx)
}) })
.unwrap(); .unwrap();
cx.run_until_parked(); cx.run_until_parked();
@ -819,7 +814,7 @@ async fn test_concurrent_tool_calls(cx: &mut TestAppContext) {
.update(cx, |thread, cx| { .update(cx, |thread, cx| {
thread.add_tool(DelayTool); thread.add_tool(DelayTool);
thread.send( thread.send(
UserMessageId::new(), new_prompt_id(),
[ [
"Call the delay tool twice in the same message.", "Call the delay tool twice in the same message.",
"Once with 100ms. Once with 300ms.", "Once with 100ms. Once with 300ms.",
@ -899,7 +894,7 @@ async fn test_profiles(cx: &mut TestAppContext) {
thread thread
.update(cx, |thread, cx| { .update(cx, |thread, cx| {
thread.set_profile(AgentProfileId("test-1".into())); thread.set_profile(AgentProfileId("test-1".into()));
thread.send(UserMessageId::new(), ["test"], cx) thread.send(new_prompt_id(), ["test"], cx)
}) })
.unwrap(); .unwrap();
cx.run_until_parked(); cx.run_until_parked();
@ -919,7 +914,7 @@ async fn test_profiles(cx: &mut TestAppContext) {
thread thread
.update(cx, |thread, cx| { .update(cx, |thread, cx| {
thread.set_profile(AgentProfileId("test-2".into())); thread.set_profile(AgentProfileId("test-2".into()));
thread.send(UserMessageId::new(), ["test2"], cx) thread.send(new_prompt_id(), ["test2"], cx)
}) })
.unwrap(); .unwrap();
cx.run_until_parked(); cx.run_until_parked();
@ -987,7 +982,7 @@ async fn test_mcp_tools(cx: &mut TestAppContext) {
); );
let events = thread.update(cx, |thread, cx| { let events = thread.update(cx, |thread, cx| {
thread.send(UserMessageId::new(), ["Hey"], cx).unwrap() thread.send(new_prompt_id(), ["Hey"], cx).unwrap()
}); });
cx.run_until_parked(); cx.run_until_parked();
@ -1029,7 +1024,7 @@ async fn test_mcp_tools(cx: &mut TestAppContext) {
// Send again after adding the echo tool, ensuring the name collision is resolved. // Send again after adding the echo tool, ensuring the name collision is resolved.
let events = thread.update(cx, |thread, cx| { let events = thread.update(cx, |thread, cx| {
thread.add_tool(EchoTool); thread.add_tool(EchoTool);
thread.send(UserMessageId::new(), ["Go"], cx).unwrap() thread.send(new_prompt_id(), ["Go"], cx).unwrap()
}); });
cx.run_until_parked(); cx.run_until_parked();
let completion = fake_model.pending_completions().pop().unwrap(); let completion = fake_model.pending_completions().pop().unwrap();
@ -1236,9 +1231,7 @@ async fn test_mcp_tool_truncation(cx: &mut TestAppContext) {
); );
thread thread
.update(cx, |thread, cx| { .update(cx, |thread, cx| thread.send(new_prompt_id(), ["Go"], cx))
thread.send(UserMessageId::new(), ["Go"], cx)
})
.unwrap(); .unwrap();
cx.run_until_parked(); cx.run_until_parked();
let completion = fake_model.pending_completions().pop().unwrap(); let completion = fake_model.pending_completions().pop().unwrap();
@ -1272,7 +1265,7 @@ async fn test_cancellation(cx: &mut TestAppContext) {
thread.add_tool(InfiniteTool); thread.add_tool(InfiniteTool);
thread.add_tool(EchoTool); thread.add_tool(EchoTool);
thread.send( thread.send(
UserMessageId::new(), new_prompt_id(),
["Call the echo tool, then call the infinite tool, then explain their output"], ["Call the echo tool, then call the infinite tool, then explain their output"],
cx, cx,
) )
@ -1328,7 +1321,7 @@ async fn test_cancellation(cx: &mut TestAppContext) {
let events = thread let events = thread
.update(cx, |thread, cx| { .update(cx, |thread, cx| {
thread.send( thread.send(
UserMessageId::new(), new_prompt_id(),
["Testing: reply with 'Hello' then stop."], ["Testing: reply with 'Hello' then stop."],
cx, cx,
) )
@ -1348,14 +1341,13 @@ async fn test_cancellation(cx: &mut TestAppContext) {
} }
#[gpui::test] #[gpui::test]
#[cfg_attr(target_os = "windows", ignore)] // TODO: Fix this test on Windows
async fn test_in_progress_send_canceled_by_next_send(cx: &mut TestAppContext) { async fn test_in_progress_send_canceled_by_next_send(cx: &mut TestAppContext) {
let ThreadTest { model, thread, .. } = setup(cx, TestModel::Fake).await; let ThreadTest { model, thread, .. } = setup(cx, TestModel::Fake).await;
let fake_model = model.as_fake(); let fake_model = model.as_fake();
let events_1 = thread let events_1 = thread
.update(cx, |thread, cx| { .update(cx, |thread, cx| {
thread.send(UserMessageId::new(), ["Hello 1"], cx) thread.send(new_prompt_id(), ["Hello 1"], cx)
}) })
.unwrap(); .unwrap();
cx.run_until_parked(); cx.run_until_parked();
@ -1364,7 +1356,7 @@ async fn test_in_progress_send_canceled_by_next_send(cx: &mut TestAppContext) {
let events_2 = thread let events_2 = thread
.update(cx, |thread, cx| { .update(cx, |thread, cx| {
thread.send(UserMessageId::new(), ["Hello 2"], cx) thread.send(new_prompt_id(), ["Hello 2"], cx)
}) })
.unwrap(); .unwrap();
cx.run_until_parked(); cx.run_until_parked();
@ -1386,7 +1378,7 @@ async fn test_subsequent_successful_sends_dont_cancel(cx: &mut TestAppContext) {
let events_1 = thread let events_1 = thread
.update(cx, |thread, cx| { .update(cx, |thread, cx| {
thread.send(UserMessageId::new(), ["Hello 1"], cx) thread.send(new_prompt_id(), ["Hello 1"], cx)
}) })
.unwrap(); .unwrap();
cx.run_until_parked(); cx.run_until_parked();
@ -1398,7 +1390,7 @@ async fn test_subsequent_successful_sends_dont_cancel(cx: &mut TestAppContext) {
let events_2 = thread let events_2 = thread
.update(cx, |thread, cx| { .update(cx, |thread, cx| {
thread.send(UserMessageId::new(), ["Hello 2"], cx) thread.send(new_prompt_id(), ["Hello 2"], cx)
}) })
.unwrap(); .unwrap();
cx.run_until_parked(); cx.run_until_parked();
@ -1418,9 +1410,7 @@ async fn test_refusal(cx: &mut TestAppContext) {
let fake_model = model.as_fake(); let fake_model = model.as_fake();
let events = thread let events = thread
.update(cx, |thread, cx| { .update(cx, |thread, cx| thread.send(new_prompt_id(), ["Hello"], cx))
thread.send(UserMessageId::new(), ["Hello"], cx)
})
.unwrap(); .unwrap();
cx.run_until_parked(); cx.run_until_parked();
thread.read_with(cx, |thread, _| { thread.read_with(cx, |thread, _| {
@ -1466,7 +1456,7 @@ async fn test_truncate_first_message(cx: &mut TestAppContext) {
let ThreadTest { model, thread, .. } = setup(cx, TestModel::Fake).await; let ThreadTest { model, thread, .. } = setup(cx, TestModel::Fake).await;
let fake_model = model.as_fake(); let fake_model = model.as_fake();
let message_id = UserMessageId::new(); let message_id = new_prompt_id();
thread thread
.update(cx, |thread, cx| { .update(cx, |thread, cx| {
thread.send(message_id.clone(), ["Hello"], cx) thread.send(message_id.clone(), ["Hello"], cx)
@ -1518,7 +1508,7 @@ async fn test_truncate_first_message(cx: &mut TestAppContext) {
}); });
thread thread
.update(cx, |thread, cx| thread.truncate(message_id, cx)) .update(cx, |thread, cx| thread.rewind(message_id, cx))
.unwrap(); .unwrap();
cx.run_until_parked(); cx.run_until_parked();
thread.read_with(cx, |thread, _| { thread.read_with(cx, |thread, _| {
@ -1528,9 +1518,7 @@ async fn test_truncate_first_message(cx: &mut TestAppContext) {
// Ensure we can still send a new message after truncation. // Ensure we can still send a new message after truncation.
thread thread
.update(cx, |thread, cx| { .update(cx, |thread, cx| thread.send(new_prompt_id(), ["Hi"], cx))
thread.send(UserMessageId::new(), ["Hi"], cx)
})
.unwrap(); .unwrap();
thread.update(cx, |thread, _cx| { thread.update(cx, |thread, _cx| {
assert_eq!( assert_eq!(
@ -1584,7 +1572,7 @@ async fn test_truncate_second_message(cx: &mut TestAppContext) {
thread thread
.update(cx, |thread, cx| { .update(cx, |thread, cx| {
thread.send(UserMessageId::new(), ["Message 1"], cx) thread.send(new_prompt_id(), ["Message 1"], cx)
}) })
.unwrap(); .unwrap();
cx.run_until_parked(); cx.run_until_parked();
@ -1627,7 +1615,7 @@ async fn test_truncate_second_message(cx: &mut TestAppContext) {
assert_first_message_state(cx); assert_first_message_state(cx);
let second_message_id = UserMessageId::new(); let second_message_id = new_prompt_id();
thread thread
.update(cx, |thread, cx| { .update(cx, |thread, cx| {
thread.send(second_message_id.clone(), ["Message 2"], cx) thread.send(second_message_id.clone(), ["Message 2"], cx)
@ -1679,7 +1667,7 @@ async fn test_truncate_second_message(cx: &mut TestAppContext) {
}); });
thread thread
.update(cx, |thread, cx| thread.truncate(second_message_id, cx)) .update(cx, |thread, cx| thread.rewind(second_message_id, cx))
.unwrap(); .unwrap();
cx.run_until_parked(); cx.run_until_parked();
@ -1698,9 +1686,7 @@ async fn test_title_generation(cx: &mut TestAppContext) {
}); });
let send = thread let send = thread
.update(cx, |thread, cx| { .update(cx, |thread, cx| thread.send(new_prompt_id(), ["Hello"], cx))
thread.send(UserMessageId::new(), ["Hello"], cx)
})
.unwrap(); .unwrap();
cx.run_until_parked(); cx.run_until_parked();
@ -1721,7 +1707,7 @@ async fn test_title_generation(cx: &mut TestAppContext) {
// Send another message, ensuring no title is generated this time. // Send another message, ensuring no title is generated this time.
let send = thread let send = thread
.update(cx, |thread, cx| { .update(cx, |thread, cx| {
thread.send(UserMessageId::new(), ["Hello again"], cx) thread.send(new_prompt_id(), ["Hello again"], cx)
}) })
.unwrap(); .unwrap();
cx.run_until_parked(); cx.run_until_parked();
@ -1742,7 +1728,7 @@ async fn test_building_request_with_pending_tools(cx: &mut TestAppContext) {
.update(cx, |thread, cx| { .update(cx, |thread, cx| {
thread.add_tool(ToolRequiringPermission); thread.add_tool(ToolRequiringPermission);
thread.add_tool(EchoTool); thread.add_tool(EchoTool);
thread.send(UserMessageId::new(), ["Hey!"], cx) thread.send(new_prompt_id(), ["Hey!"], cx)
}) })
.unwrap(); .unwrap();
cx.run_until_parked(); cx.run_until_parked();
@ -1822,11 +1808,11 @@ async fn test_agent_connection(cx: &mut TestAppContext) {
let clock = Arc::new(clock::FakeSystemClock::new()); let clock = Arc::new(clock::FakeSystemClock::new());
let client = Client::new(clock, http_client, cx); let client = Client::new(clock, http_client, cx);
let user_store = cx.new(|cx| UserStore::new(client.clone(), cx)); let user_store = cx.new(|cx| UserStore::new(client.clone(), cx));
Project::init_settings(cx);
agent_settings::init(cx);
language_model::init(client.clone(), cx); language_model::init(client.clone(), cx);
language_models::init(user_store, client.clone(), cx); language_models::init(user_store, client.clone(), cx);
Project::init_settings(cx);
LanguageModelRegistry::test(cx); LanguageModelRegistry::test(cx);
agent_settings::init(cx);
}); });
cx.executor().forbid_parking(); cx.executor().forbid_parking();
@ -1894,7 +1880,9 @@ async fn test_agent_connection(cx: &mut TestAppContext) {
let model = model.as_fake(); let model = model.as_fake();
assert_eq!(model.id().0, "fake", "should return default model"); assert_eq!(model.id().0, "fake", "should return default model");
let request = acp_thread.update(cx, |thread, cx| thread.send(vec!["abc".into()], cx)); let request = acp_thread.update(cx, |thread, cx| {
thread.send(new_prompt_id(), vec!["abc".into()], cx)
});
cx.run_until_parked(); cx.run_until_parked();
model.send_last_completion_stream_text_chunk("def"); model.send_last_completion_stream_text_chunk("def");
cx.run_until_parked(); cx.run_until_parked();
@ -1924,8 +1912,8 @@ async fn test_agent_connection(cx: &mut TestAppContext) {
let result = cx let result = cx
.update(|cx| { .update(|cx| {
connection.prompt( connection.prompt(
Some(acp_thread::UserMessageId::new()),
acp::PromptRequest { acp::PromptRequest {
prompt_id: Some(new_prompt_id()),
session_id: session_id.clone(), session_id: session_id.clone(),
prompt: vec!["ghi".into()], prompt: vec!["ghi".into()],
}, },
@ -1948,9 +1936,7 @@ async fn test_tool_updates_to_completion(cx: &mut TestAppContext) {
let fake_model = model.as_fake(); let fake_model = model.as_fake();
let mut events = thread let mut events = thread
.update(cx, |thread, cx| { .update(cx, |thread, cx| thread.send(new_prompt_id(), ["Think"], cx))
thread.send(UserMessageId::new(), ["Think"], cx)
})
.unwrap(); .unwrap();
cx.run_until_parked(); cx.run_until_parked();
@ -2051,7 +2037,7 @@ async fn test_send_no_retry_on_success(cx: &mut TestAppContext) {
let mut events = thread let mut events = thread
.update(cx, |thread, cx| { .update(cx, |thread, cx| {
thread.set_completion_mode(agent_settings::CompletionMode::Burn, cx); thread.set_completion_mode(agent_settings::CompletionMode::Burn, cx);
thread.send(UserMessageId::new(), ["Hello!"], cx) thread.send(new_prompt_id(), ["Hello!"], cx)
}) })
.unwrap(); .unwrap();
cx.run_until_parked(); cx.run_until_parked();
@ -2095,7 +2081,7 @@ async fn test_send_retry_on_error(cx: &mut TestAppContext) {
let mut events = thread let mut events = thread
.update(cx, |thread, cx| { .update(cx, |thread, cx| {
thread.set_completion_mode(agent_settings::CompletionMode::Burn, cx); thread.set_completion_mode(agent_settings::CompletionMode::Burn, cx);
thread.send(UserMessageId::new(), ["Hello!"], cx) thread.send(new_prompt_id(), ["Hello!"], cx)
}) })
.unwrap(); .unwrap();
cx.run_until_parked(); cx.run_until_parked();
@ -2161,7 +2147,7 @@ async fn test_send_retry_finishes_tool_calls_on_error(cx: &mut TestAppContext) {
.update(cx, |thread, cx| { .update(cx, |thread, cx| {
thread.set_completion_mode(agent_settings::CompletionMode::Burn, cx); thread.set_completion_mode(agent_settings::CompletionMode::Burn, cx);
thread.add_tool(EchoTool); thread.add_tool(EchoTool);
thread.send(UserMessageId::new(), ["Call the echo tool!"], cx) thread.send(new_prompt_id(), ["Call the echo tool!"], cx)
}) })
.unwrap(); .unwrap();
cx.run_until_parked(); cx.run_until_parked();
@ -2236,7 +2222,7 @@ async fn test_send_max_retries_exceeded(cx: &mut TestAppContext) {
let mut events = thread let mut events = thread
.update(cx, |thread, cx| { .update(cx, |thread, cx| {
thread.set_completion_mode(agent_settings::CompletionMode::Burn, cx); thread.set_completion_mode(agent_settings::CompletionMode::Burn, cx);
thread.send(UserMessageId::new(), ["Hello!"], cx) thread.send(new_prompt_id(), ["Hello!"], cx)
}) })
.unwrap(); .unwrap();
cx.run_until_parked(); cx.run_until_parked();

View file

@ -4,7 +4,7 @@ use crate::{
ListDirectoryTool, MovePathTool, NowTool, OpenTool, ReadFileTool, SystemPromptTemplate, ListDirectoryTool, MovePathTool, NowTool, OpenTool, ReadFileTool, SystemPromptTemplate,
Template, Templates, TerminalTool, ThinkingTool, WebSearchTool, Template, Templates, TerminalTool, ThinkingTool, WebSearchTool,
}; };
use acp_thread::{MentionUri, UserMessageId}; use acp_thread::MentionUri;
use action_log::ActionLog; use action_log::ActionLog;
use agent::thread::{GitState, ProjectSnapshot, WorktreeSnapshot}; use agent::thread::{GitState, ProjectSnapshot, WorktreeSnapshot};
use agent_client_protocol as acp; use agent_client_protocol as acp;
@ -137,7 +137,7 @@ impl Message {
#[derive(Debug, Clone, PartialEq, Eq, Serialize, Deserialize)] #[derive(Debug, Clone, PartialEq, Eq, Serialize, Deserialize)]
pub struct UserMessage { pub struct UserMessage {
pub id: UserMessageId, pub id: acp::PromptId,
pub content: Vec<UserMessageContent>, pub content: Vec<UserMessageContent>,
} }
@ -564,7 +564,7 @@ pub struct Thread {
pending_message: Option<AgentMessage>, pending_message: Option<AgentMessage>,
tools: BTreeMap<SharedString, Arc<dyn AnyAgentTool>>, tools: BTreeMap<SharedString, Arc<dyn AnyAgentTool>>,
tool_use_limit_reached: bool, tool_use_limit_reached: bool,
request_token_usage: HashMap<UserMessageId, language_model::TokenUsage>, request_token_usage: HashMap<acp::PromptId, language_model::TokenUsage>,
#[allow(unused)] #[allow(unused)]
cumulative_token_usage: TokenUsage, cumulative_token_usage: TokenUsage,
#[allow(unused)] #[allow(unused)]
@ -732,17 +732,7 @@ impl Thread {
stream.update_tool_call_fields( stream.update_tool_call_fields(
&tool_use.id, &tool_use.id,
acp::ToolCallUpdateFields { acp::ToolCallUpdateFields {
status: Some( status: Some(acp::ToolCallStatus::Completed),
tool_result
.as_ref()
.map_or(acp::ToolCallStatus::Failed, |result| {
if result.is_error {
acp::ToolCallStatus::Failed
} else {
acp::ToolCallStatus::Completed
}
}),
),
raw_output: output, raw_output: output,
..Default::default() ..Default::default()
}, },
@ -1080,7 +1070,7 @@ impl Thread {
cx.notify(); cx.notify();
} }
pub fn truncate(&mut self, message_id: UserMessageId, cx: &mut Context<Self>) -> Result<()> { pub fn rewind(&mut self, message_id: acp::PromptId, cx: &mut Context<Self>) -> Result<()> {
self.cancel(cx); self.cancel(cx);
let Some(position) = self.messages.iter().position( let Some(position) = self.messages.iter().position(
|msg| matches!(msg, Message::User(UserMessage { id, .. }) if id == &message_id), |msg| matches!(msg, Message::User(UserMessage { id, .. }) if id == &message_id),
@ -1128,7 +1118,7 @@ impl Thread {
/// The returned channel will report all the occurrences in which the model stops before erroring or ending its turn. /// The returned channel will report all the occurrences in which the model stops before erroring or ending its turn.
pub fn send<T>( pub fn send<T>(
&mut self, &mut self,
id: UserMessageId, id: acp::PromptId,
content: impl IntoIterator<Item = T>, content: impl IntoIterator<Item = T>,
cx: &mut Context<Self>, cx: &mut Context<Self>,
) -> Result<mpsc::UnboundedReceiver<Result<ThreadEvent>>> ) -> Result<mpsc::UnboundedReceiver<Result<ThreadEvent>>>
@ -1567,7 +1557,7 @@ impl Thread {
tool_name: tool_use.name, tool_name: tool_use.name,
is_error: true, is_error: true,
content: LanguageModelToolResultContent::Text(Arc::from(error.to_string())), content: LanguageModelToolResultContent::Text(Arc::from(error.to_string())),
output: Some(error.to_string().into()), output: None,
}, },
} }
})) }))
@ -2469,30 +2459,6 @@ impl ToolCallEventStreamReceiver {
} }
} }
pub async fn expect_update_fields(&mut self) -> acp::ToolCallUpdateFields {
let event = self.0.next().await;
if let Some(Ok(ThreadEvent::ToolCallUpdate(acp_thread::ToolCallUpdate::UpdateFields(
update,
)))) = event
{
update.fields
} else {
panic!("Expected update fields but got: {:?}", event);
}
}
pub async fn expect_diff(&mut self) -> Entity<acp_thread::Diff> {
let event = self.0.next().await;
if let Some(Ok(ThreadEvent::ToolCallUpdate(acp_thread::ToolCallUpdate::UpdateDiff(
update,
)))) = event
{
update.diff
} else {
panic!("Expected diff but got: {:?}", event);
}
}
pub async fn expect_terminal(&mut self) -> Entity<acp_thread::Terminal> { pub async fn expect_terminal(&mut self) -> Entity<acp_thread::Terminal> {
let event = self.0.next().await; let event = self.0.next().await;
if let Some(Ok(ThreadEvent::ToolCallUpdate(acp_thread::ToolCallUpdate::UpdateTerminal( if let Some(Ok(ThreadEvent::ToolCallUpdate(acp_thread::ToolCallUpdate::UpdateTerminal(

View file

@ -273,13 +273,6 @@ impl AgentTool for EditFileTool {
let diff = cx.new(|cx| Diff::new(buffer.clone(), cx))?; let diff = cx.new(|cx| Diff::new(buffer.clone(), cx))?;
event_stream.update_diff(diff.clone()); event_stream.update_diff(diff.clone());
let _finalize_diff = util::defer({
let diff = diff.downgrade();
let mut cx = cx.clone();
move || {
diff.update(&mut cx, |diff, cx| diff.finalize(cx)).ok();
}
});
let old_snapshot = buffer.read_with(cx, |buffer, _cx| buffer.snapshot())?; let old_snapshot = buffer.read_with(cx, |buffer, _cx| buffer.snapshot())?;
let old_text = cx let old_text = cx
@ -396,6 +389,8 @@ impl AgentTool for EditFileTool {
}) })
.await; .await;
diff.update(cx, |diff, cx| diff.finalize(cx)).ok();
let input_path = input.path.display(); let input_path = input.path.display();
if unified_diff.is_empty() { if unified_diff.is_empty() {
anyhow::ensure!( anyhow::ensure!(
@ -1550,100 +1545,6 @@ mod tests {
); );
} }
#[gpui::test]
async fn test_diff_finalization(cx: &mut TestAppContext) {
init_test(cx);
let fs = project::FakeFs::new(cx.executor());
fs.insert_tree("/", json!({"main.rs": ""})).await;
let project = Project::test(fs.clone(), [path!("/").as_ref()], cx).await;
let languages = project.read_with(cx, |project, _cx| project.languages().clone());
let context_server_registry =
cx.new(|cx| ContextServerRegistry::new(project.read(cx).context_server_store(), cx));
let model = Arc::new(FakeLanguageModel::default());
let thread = cx.new(|cx| {
Thread::new(
project.clone(),
cx.new(|_cx| ProjectContext::default()),
context_server_registry.clone(),
Templates::new(),
Some(model.clone()),
cx,
)
});
// Ensure the diff is finalized after the edit completes.
{
let tool = Arc::new(EditFileTool::new(thread.downgrade(), languages.clone()));
let (stream_tx, mut stream_rx) = ToolCallEventStream::test();
let edit = cx.update(|cx| {
tool.run(
EditFileToolInput {
display_description: "Edit file".into(),
path: path!("/main.rs").into(),
mode: EditFileMode::Edit,
},
stream_tx,
cx,
)
});
stream_rx.expect_update_fields().await;
let diff = stream_rx.expect_diff().await;
diff.read_with(cx, |diff, _| assert!(matches!(diff, Diff::Pending(_))));
cx.run_until_parked();
model.end_last_completion_stream();
edit.await.unwrap();
diff.read_with(cx, |diff, _| assert!(matches!(diff, Diff::Finalized(_))));
}
// Ensure the diff is finalized if an error occurs while editing.
{
model.forbid_requests();
let tool = Arc::new(EditFileTool::new(thread.downgrade(), languages.clone()));
let (stream_tx, mut stream_rx) = ToolCallEventStream::test();
let edit = cx.update(|cx| {
tool.run(
EditFileToolInput {
display_description: "Edit file".into(),
path: path!("/main.rs").into(),
mode: EditFileMode::Edit,
},
stream_tx,
cx,
)
});
stream_rx.expect_update_fields().await;
let diff = stream_rx.expect_diff().await;
diff.read_with(cx, |diff, _| assert!(matches!(diff, Diff::Pending(_))));
edit.await.unwrap_err();
diff.read_with(cx, |diff, _| assert!(matches!(diff, Diff::Finalized(_))));
model.allow_requests();
}
// Ensure the diff is finalized if the tool call gets dropped.
{
let tool = Arc::new(EditFileTool::new(thread.downgrade(), languages.clone()));
let (stream_tx, mut stream_rx) = ToolCallEventStream::test();
let edit = cx.update(|cx| {
tool.run(
EditFileToolInput {
display_description: "Edit file".into(),
path: path!("/main.rs").into(),
mode: EditFileMode::Edit,
},
stream_tx,
cx,
)
});
stream_rx.expect_update_fields().await;
let diff = stream_rx.expect_diff().await;
diff.read_with(cx, |diff, _| assert!(matches!(diff, Diff::Pending(_))));
drop(edit);
cx.run_until_parked();
diff.read_with(cx, |diff, _| assert!(matches!(diff, Diff::Finalized(_))));
}
}
fn init_test(cx: &mut TestAppContext) { fn init_test(cx: &mut TestAppContext) {
cx.update(|cx| { cx.update(|cx| {
let settings_store = SettingsStore::test(cx); let settings_store = SettingsStore::test(cx);

View file

@ -29,6 +29,7 @@ pub struct AcpConnection {
sessions: Rc<RefCell<HashMap<acp::SessionId, AcpSession>>>, sessions: Rc<RefCell<HashMap<acp::SessionId, AcpSession>>>,
auth_methods: Vec<acp::AuthMethod>, auth_methods: Vec<acp::AuthMethod>,
prompt_capabilities: acp::PromptCapabilities, prompt_capabilities: acp::PromptCapabilities,
supports_rewind: bool,
_io_task: Task<Result<()>>, _io_task: Task<Result<()>>,
} }
@ -147,6 +148,7 @@ impl AcpConnection {
server_name, server_name,
sessions, sessions,
prompt_capabilities: response.agent_capabilities.prompt_capabilities, prompt_capabilities: response.agent_capabilities.prompt_capabilities,
supports_rewind: response.agent_capabilities.rewind_session,
_io_task: io_task, _io_task: io_task,
}) })
} }
@ -162,34 +164,12 @@ impl AgentConnection for AcpConnection {
let conn = self.connection.clone(); let conn = self.connection.clone();
let sessions = self.sessions.clone(); let sessions = self.sessions.clone();
let cwd = cwd.to_path_buf(); let cwd = cwd.to_path_buf();
let context_server_store = project.read(cx).context_server_store().read(cx);
let mcp_servers = context_server_store
.configured_server_ids()
.iter()
.filter_map(|id| {
let configuration = context_server_store.configuration_for_server(id)?;
let command = configuration.command();
Some(acp::McpServer {
name: id.0.to_string(),
command: command.path.clone(),
args: command.args.clone(),
env: if let Some(env) = command.env.as_ref() {
env.iter()
.map(|(name, value)| acp::EnvVariable {
name: name.clone(),
value: value.clone(),
})
.collect()
} else {
vec![]
},
})
})
.collect();
cx.spawn(async move |cx| { cx.spawn(async move |cx| {
let response = conn let response = conn
.new_session(acp::NewSessionRequest { mcp_servers, cwd }) .new_session(acp::NewSessionRequest {
mcp_servers: vec![],
cwd,
})
.await .await
.map_err(|err| { .map_err(|err| {
if err.code == acp::ErrorCode::AUTH_REQUIRED.code { if err.code == acp::ErrorCode::AUTH_REQUIRED.code {
@ -247,9 +227,22 @@ impl AgentConnection for AcpConnection {
}) })
} }
fn rewind(
&self,
session_id: &agent_client_protocol::SessionId,
_cx: &App,
) -> Option<Rc<dyn acp_thread::AgentSessionRewind>> {
if !self.supports_rewind {
return None;
}
Some(Rc::new(AcpRewinder {
connection: self.connection.clone(),
session_id: session_id.clone(),
}) as _)
}
fn prompt( fn prompt(
&self, &self,
_id: Option<acp_thread::UserMessageId>,
params: acp::PromptRequest, params: acp::PromptRequest,
cx: &mut App, cx: &mut App,
) -> Task<Result<acp::PromptResponse>> { ) -> Task<Result<acp::PromptResponse>> {
@ -324,6 +317,25 @@ impl AgentConnection for AcpConnection {
} }
} }
struct AcpRewinder {
connection: Rc<acp::ClientSideConnection>,
session_id: acp::SessionId,
}
impl acp_thread::AgentSessionRewind for AcpRewinder {
fn rewind(&self, prompt_id: acp::PromptId, cx: &mut App) -> Task<Result<()>> {
let conn = self.connection.clone();
let params = acp::RewindRequest {
session_id: self.session_id.clone(),
prompt_id,
};
cx.foreground_executor().spawn(async move {
conn.rewind(params).await?;
Ok(())
})
}
}
struct ClientDelegate { struct ClientDelegate {
sessions: Rc<RefCell<HashMap<acp::SessionId, AcpSession>>>, sessions: Rc<RefCell<HashMap<acp::SessionId, AcpSession>>>,
cx: AsyncApp, cx: AsyncApp,

View file

@ -0,0 +1,524 @@
// Translates old acp agents into the new schema
use action_log::ActionLog;
use agent_client_protocol as acp;
use agentic_coding_protocol::{self as acp_old, AgentRequest as _};
use anyhow::{Context as _, Result, anyhow};
use futures::channel::oneshot;
use gpui::{AppContext as _, AsyncApp, Entity, Task, WeakEntity};
use project::Project;
use std::{any::Any, cell::RefCell, path::Path, rc::Rc};
use ui::App;
use util::ResultExt as _;
use crate::AgentServerCommand;
use acp_thread::{AcpThread, AgentConnection, AuthRequired};
#[derive(Clone)]
struct OldAcpClientDelegate {
thread: Rc<RefCell<WeakEntity<AcpThread>>>,
cx: AsyncApp,
next_tool_call_id: Rc<RefCell<u64>>,
// sent_buffer_versions: HashMap<Entity<Buffer>, HashMap<u64, BufferSnapshot>>,
}
impl OldAcpClientDelegate {
fn new(thread: Rc<RefCell<WeakEntity<AcpThread>>>, cx: AsyncApp) -> Self {
Self {
thread,
cx,
next_tool_call_id: Rc::new(RefCell::new(0)),
}
}
}
impl acp_old::Client for OldAcpClientDelegate {
async fn stream_assistant_message_chunk(
&self,
params: acp_old::StreamAssistantMessageChunkParams,
) -> Result<(), acp_old::Error> {
let cx = &mut self.cx.clone();
cx.update(|cx| {
self.thread
.borrow()
.update(cx, |thread, cx| match params.chunk {
acp_old::AssistantMessageChunk::Text { text } => {
thread.push_assistant_content_block(text.into(), false, cx)
}
acp_old::AssistantMessageChunk::Thought { thought } => {
thread.push_assistant_content_block(thought.into(), true, cx)
}
})
.log_err();
})?;
Ok(())
}
async fn request_tool_call_confirmation(
&self,
request: acp_old::RequestToolCallConfirmationParams,
) -> Result<acp_old::RequestToolCallConfirmationResponse, acp_old::Error> {
let cx = &mut self.cx.clone();
let old_acp_id = *self.next_tool_call_id.borrow() + 1;
self.next_tool_call_id.replace(old_acp_id);
let tool_call = into_new_tool_call(
acp::ToolCallId(old_acp_id.to_string().into()),
request.tool_call,
);
let mut options = match request.confirmation {
acp_old::ToolCallConfirmation::Edit { .. } => vec![(
acp_old::ToolCallConfirmationOutcome::AlwaysAllow,
acp::PermissionOptionKind::AllowAlways,
"Always Allow Edits".to_string(),
)],
acp_old::ToolCallConfirmation::Execute { root_command, .. } => vec![(
acp_old::ToolCallConfirmationOutcome::AlwaysAllow,
acp::PermissionOptionKind::AllowAlways,
format!("Always Allow {}", root_command),
)],
acp_old::ToolCallConfirmation::Mcp {
server_name,
tool_name,
..
} => vec![
(
acp_old::ToolCallConfirmationOutcome::AlwaysAllowMcpServer,
acp::PermissionOptionKind::AllowAlways,
format!("Always Allow {}", server_name),
),
(
acp_old::ToolCallConfirmationOutcome::AlwaysAllowTool,
acp::PermissionOptionKind::AllowAlways,
format!("Always Allow {}", tool_name),
),
],
acp_old::ToolCallConfirmation::Fetch { .. } => vec![(
acp_old::ToolCallConfirmationOutcome::AlwaysAllow,
acp::PermissionOptionKind::AllowAlways,
"Always Allow".to_string(),
)],
acp_old::ToolCallConfirmation::Other { .. } => vec![(
acp_old::ToolCallConfirmationOutcome::AlwaysAllow,
acp::PermissionOptionKind::AllowAlways,
"Always Allow".to_string(),
)],
};
options.extend([
(
acp_old::ToolCallConfirmationOutcome::Allow,
acp::PermissionOptionKind::AllowOnce,
"Allow".to_string(),
),
(
acp_old::ToolCallConfirmationOutcome::Reject,
acp::PermissionOptionKind::RejectOnce,
"Reject".to_string(),
),
]);
let mut outcomes = Vec::with_capacity(options.len());
let mut acp_options = Vec::with_capacity(options.len());
for (index, (outcome, kind, label)) in options.into_iter().enumerate() {
outcomes.push(outcome);
acp_options.push(acp::PermissionOption {
id: acp::PermissionOptionId(index.to_string().into()),
name: label,
kind,
})
}
let response = cx
.update(|cx| {
self.thread.borrow().update(cx, |thread, cx| {
thread.request_tool_call_authorization(tool_call.into(), acp_options, cx)
})
})??
.context("Failed to update thread")?
.await;
let outcome = match response {
Ok(option_id) => outcomes[option_id.0.parse::<usize>().unwrap_or(0)],
Err(oneshot::Canceled) => acp_old::ToolCallConfirmationOutcome::Cancel,
};
Ok(acp_old::RequestToolCallConfirmationResponse {
id: acp_old::ToolCallId(old_acp_id),
outcome,
})
}
async fn push_tool_call(
&self,
request: acp_old::PushToolCallParams,
) -> Result<acp_old::PushToolCallResponse, acp_old::Error> {
let cx = &mut self.cx.clone();
let old_acp_id = *self.next_tool_call_id.borrow() + 1;
self.next_tool_call_id.replace(old_acp_id);
cx.update(|cx| {
self.thread.borrow().update(cx, |thread, cx| {
thread.upsert_tool_call(
into_new_tool_call(acp::ToolCallId(old_acp_id.to_string().into()), request),
cx,
)
})
})??
.context("Failed to update thread")?;
Ok(acp_old::PushToolCallResponse {
id: acp_old::ToolCallId(old_acp_id),
})
}
async fn update_tool_call(
&self,
request: acp_old::UpdateToolCallParams,
) -> Result<(), acp_old::Error> {
let cx = &mut self.cx.clone();
cx.update(|cx| {
self.thread.borrow().update(cx, |thread, cx| {
thread.update_tool_call(
acp::ToolCallUpdate {
id: acp::ToolCallId(request.tool_call_id.0.to_string().into()),
fields: acp::ToolCallUpdateFields {
status: Some(into_new_tool_call_status(request.status)),
content: Some(
request
.content
.into_iter()
.map(into_new_tool_call_content)
.collect::<Vec<_>>(),
),
..Default::default()
},
},
cx,
)
})
})?
.context("Failed to update thread")??;
Ok(())
}
async fn update_plan(&self, request: acp_old::UpdatePlanParams) -> Result<(), acp_old::Error> {
let cx = &mut self.cx.clone();
cx.update(|cx| {
self.thread.borrow().update(cx, |thread, cx| {
thread.update_plan(
acp::Plan {
entries: request
.entries
.into_iter()
.map(into_new_plan_entry)
.collect(),
},
cx,
)
})
})?
.context("Failed to update thread")?;
Ok(())
}
async fn read_text_file(
&self,
acp_old::ReadTextFileParams { path, line, limit }: acp_old::ReadTextFileParams,
) -> Result<acp_old::ReadTextFileResponse, acp_old::Error> {
let content = self
.cx
.update(|cx| {
self.thread.borrow().update(cx, |thread, cx| {
thread.read_text_file(path, line, limit, false, cx)
})
})?
.context("Failed to update thread")?
.await?;
Ok(acp_old::ReadTextFileResponse { content })
}
async fn write_text_file(
&self,
acp_old::WriteTextFileParams { path, content }: acp_old::WriteTextFileParams,
) -> Result<(), acp_old::Error> {
self.cx
.update(|cx| {
self.thread
.borrow()
.update(cx, |thread, cx| thread.write_text_file(path, content, cx))
})?
.context("Failed to update thread")?
.await?;
Ok(())
}
}
fn into_new_tool_call(id: acp::ToolCallId, request: acp_old::PushToolCallParams) -> acp::ToolCall {
acp::ToolCall {
id,
title: request.label,
kind: acp_kind_from_old_icon(request.icon),
status: acp::ToolCallStatus::InProgress,
content: request
.content
.into_iter()
.map(into_new_tool_call_content)
.collect(),
locations: request
.locations
.into_iter()
.map(into_new_tool_call_location)
.collect(),
raw_input: None,
raw_output: None,
}
}
fn acp_kind_from_old_icon(icon: acp_old::Icon) -> acp::ToolKind {
match icon {
acp_old::Icon::FileSearch => acp::ToolKind::Search,
acp_old::Icon::Folder => acp::ToolKind::Search,
acp_old::Icon::Globe => acp::ToolKind::Search,
acp_old::Icon::Hammer => acp::ToolKind::Other,
acp_old::Icon::LightBulb => acp::ToolKind::Think,
acp_old::Icon::Pencil => acp::ToolKind::Edit,
acp_old::Icon::Regex => acp::ToolKind::Search,
acp_old::Icon::Terminal => acp::ToolKind::Execute,
}
}
fn into_new_tool_call_status(status: acp_old::ToolCallStatus) -> acp::ToolCallStatus {
match status {
acp_old::ToolCallStatus::Running => acp::ToolCallStatus::InProgress,
acp_old::ToolCallStatus::Finished => acp::ToolCallStatus::Completed,
acp_old::ToolCallStatus::Error => acp::ToolCallStatus::Failed,
}
}
fn into_new_tool_call_content(content: acp_old::ToolCallContent) -> acp::ToolCallContent {
match content {
acp_old::ToolCallContent::Markdown { markdown } => markdown.into(),
acp_old::ToolCallContent::Diff { diff } => acp::ToolCallContent::Diff {
diff: into_new_diff(diff),
},
}
}
fn into_new_diff(diff: acp_old::Diff) -> acp::Diff {
acp::Diff {
path: diff.path,
old_text: diff.old_text,
new_text: diff.new_text,
}
}
fn into_new_tool_call_location(location: acp_old::ToolCallLocation) -> acp::ToolCallLocation {
acp::ToolCallLocation {
path: location.path,
line: location.line,
}
}
fn into_new_plan_entry(entry: acp_old::PlanEntry) -> acp::PlanEntry {
acp::PlanEntry {
content: entry.content,
priority: into_new_plan_priority(entry.priority),
status: into_new_plan_status(entry.status),
}
}
fn into_new_plan_priority(priority: acp_old::PlanEntryPriority) -> acp::PlanEntryPriority {
match priority {
acp_old::PlanEntryPriority::Low => acp::PlanEntryPriority::Low,
acp_old::PlanEntryPriority::Medium => acp::PlanEntryPriority::Medium,
acp_old::PlanEntryPriority::High => acp::PlanEntryPriority::High,
}
}
fn into_new_plan_status(status: acp_old::PlanEntryStatus) -> acp::PlanEntryStatus {
match status {
acp_old::PlanEntryStatus::Pending => acp::PlanEntryStatus::Pending,
acp_old::PlanEntryStatus::InProgress => acp::PlanEntryStatus::InProgress,
acp_old::PlanEntryStatus::Completed => acp::PlanEntryStatus::Completed,
}
}
pub struct AcpConnection {
pub name: &'static str,
pub connection: acp_old::AgentConnection,
pub _child_status: Task<Result<()>>,
pub current_thread: Rc<RefCell<WeakEntity<AcpThread>>>,
}
impl AcpConnection {
pub fn stdio(
name: &'static str,
command: AgentServerCommand,
root_dir: &Path,
cx: &mut AsyncApp,
) -> Task<Result<Self>> {
let root_dir = root_dir.to_path_buf();
cx.spawn(async move |cx| {
let mut child = util::command::new_smol_command(&command.path)
.args(command.args.iter())
.current_dir(root_dir)
.stdin(std::process::Stdio::piped())
.stdout(std::process::Stdio::piped())
.stderr(std::process::Stdio::inherit())
.kill_on_drop(true)
.spawn()?;
let stdin = child.stdin.take().unwrap();
let stdout = child.stdout.take().unwrap();
log::trace!("Spawned (pid: {})", child.id());
let foreground_executor = cx.foreground_executor().clone();
let thread_rc = Rc::new(RefCell::new(WeakEntity::new_invalid()));
let (connection, io_fut) = acp_old::AgentConnection::connect_to_agent(
OldAcpClientDelegate::new(thread_rc.clone(), cx.clone()),
stdin,
stdout,
move |fut| foreground_executor.spawn(fut).detach(),
);
let io_task = cx.background_spawn(async move {
io_fut.await.log_err();
});
let child_status = cx.background_spawn(async move {
let result = match child.status().await {
Err(e) => Err(anyhow!(e)),
Ok(result) if result.success() => Ok(()),
Ok(result) => Err(anyhow!(result)),
};
drop(io_task);
result
});
Ok(Self {
name,
connection,
_child_status: child_status,
current_thread: thread_rc,
})
})
}
}
impl AgentConnection for AcpConnection {
fn new_thread(
self: Rc<Self>,
project: Entity<Project>,
_cwd: &Path,
cx: &mut App,
) -> Task<Result<Entity<AcpThread>>> {
let task = self.connection.request_any(
acp_old::InitializeParams {
protocol_version: acp_old::ProtocolVersion::latest(),
}
.into_any(),
);
let current_thread = self.current_thread.clone();
cx.spawn(async move |cx| {
let result = task.await?;
let result = acp_old::InitializeParams::response_from_any(result)?;
if !result.is_authenticated {
anyhow::bail!(AuthRequired::new())
}
cx.update(|cx| {
let thread = cx.new(|cx| {
let session_id = acp::SessionId("acp-old-no-id".into());
let action_log = cx.new(|_| ActionLog::new(project.clone()));
AcpThread::new(self.name, self.clone(), project, action_log, session_id)
});
current_thread.replace(thread.downgrade());
thread
})
})
}
fn auth_methods(&self) -> &[acp::AuthMethod] {
&[]
}
fn authenticate(&self, _method_id: acp::AuthMethodId, cx: &mut App) -> Task<Result<()>> {
let task = self
.connection
.request_any(acp_old::AuthenticateParams.into_any());
cx.foreground_executor().spawn(async move {
task.await?;
Ok(())
})
}
fn prompt(
&self,
_id: Option<acp_thread::UserMessageId>,
params: acp::PromptRequest,
cx: &mut App,
) -> Task<Result<acp::PromptResponse>> {
let chunks = params
.prompt
.into_iter()
.filter_map(|block| match block {
acp::ContentBlock::Text(text) => {
Some(acp_old::UserMessageChunk::Text { text: text.text })
}
acp::ContentBlock::ResourceLink(link) => Some(acp_old::UserMessageChunk::Path {
path: link.uri.into(),
}),
_ => None,
})
.collect();
let task = self
.connection
.request_any(acp_old::SendUserMessageParams { chunks }.into_any());
cx.foreground_executor().spawn(async move {
task.await?;
anyhow::Ok(acp::PromptResponse {
stop_reason: acp::StopReason::EndTurn,
})
})
}
fn prompt_capabilities(&self) -> acp::PromptCapabilities {
acp::PromptCapabilities {
image: false,
audio: false,
embedded_context: false,
}
}
fn cancel(&self, _session_id: &acp::SessionId, cx: &mut App) {
let task = self
.connection
.request_any(acp_old::CancelSendMessageParams.into_any());
cx.foreground_executor()
.spawn(async move {
task.await?;
anyhow::Ok(())
})
.detach_and_log_err(cx)
}
fn into_any(self: Rc<Self>) -> Rc<dyn Any> {
self
}
}

View file

@ -0,0 +1,376 @@
use acp_tools::AcpConnectionRegistry;
use action_log::ActionLog;
use agent_client_protocol::{self as acp, Agent as _, ErrorCode};
use anyhow::anyhow;
use collections::HashMap;
use futures::AsyncBufReadExt as _;
use futures::channel::oneshot;
use futures::io::BufReader;
use project::Project;
use serde::Deserialize;
use std::path::Path;
use std::rc::Rc;
use std::{any::Any, cell::RefCell};
use anyhow::{Context as _, Result};
use gpui::{App, AppContext as _, AsyncApp, Entity, Task, WeakEntity};
use crate::{AgentServerCommand, acp::UnsupportedVersion};
use acp_thread::{AcpThread, AgentConnection, AuthRequired, LoadError};
pub struct AcpConnection {
server_name: &'static str,
connection: Rc<acp::ClientSideConnection>,
sessions: Rc<RefCell<HashMap<acp::SessionId, AcpSession>>>,
auth_methods: Vec<acp::AuthMethod>,
prompt_capabilities: acp::PromptCapabilities,
_io_task: Task<Result<()>>,
}
pub struct AcpSession {
thread: WeakEntity<AcpThread>,
suppress_abort_err: bool,
}
const MINIMUM_SUPPORTED_VERSION: acp::ProtocolVersion = acp::V1;
impl AcpConnection {
pub async fn stdio(
server_name: &'static str,
command: AgentServerCommand,
root_dir: &Path,
cx: &mut AsyncApp,
) -> Result<Self> {
let mut child = util::command::new_smol_command(&command.path)
.args(command.args.iter().map(|arg| arg.as_str()))
.envs(command.env.iter().flatten())
.current_dir(root_dir)
.stdin(std::process::Stdio::piped())
.stdout(std::process::Stdio::piped())
.stderr(std::process::Stdio::piped())
.kill_on_drop(true)
.spawn()?;
let stdout = child.stdout.take().context("Failed to take stdout")?;
let stdin = child.stdin.take().context("Failed to take stdin")?;
let stderr = child.stderr.take().context("Failed to take stderr")?;
log::trace!("Spawned (pid: {})", child.id());
let sessions = Rc::new(RefCell::new(HashMap::default()));
let client = ClientDelegate {
sessions: sessions.clone(),
cx: cx.clone(),
};
let (connection, io_task) = acp::ClientSideConnection::new(client, stdin, stdout, {
let foreground_executor = cx.foreground_executor().clone();
move |fut| {
foreground_executor.spawn(fut).detach();
}
});
let io_task = cx.background_spawn(io_task);
cx.background_spawn(async move {
let mut stderr = BufReader::new(stderr);
let mut line = String::new();
while let Ok(n) = stderr.read_line(&mut line).await
&& n > 0
{
log::warn!("agent stderr: {}", &line);
line.clear();
}
})
.detach();
cx.spawn({
let sessions = sessions.clone();
async move |cx| {
let status = child.status().await?;
for session in sessions.borrow().values() {
session
.thread
.update(cx, |thread, cx| {
thread.emit_load_error(LoadError::Exited { status }, cx)
})
.ok();
}
anyhow::Ok(())
}
})
.detach();
let connection = Rc::new(connection);
cx.update(|cx| {
AcpConnectionRegistry::default_global(cx).update(cx, |registry, cx| {
registry.set_active_connection(server_name, &connection, cx)
});
})?;
let response = connection
.initialize(acp::InitializeRequest {
protocol_version: acp::VERSION,
client_capabilities: acp::ClientCapabilities {
fs: acp::FileSystemCapability {
read_text_file: true,
write_text_file: true,
},
},
})
.await?;
if response.protocol_version < MINIMUM_SUPPORTED_VERSION {
return Err(UnsupportedVersion.into());
}
Ok(Self {
auth_methods: response.auth_methods,
connection,
server_name,
sessions,
prompt_capabilities: response.agent_capabilities.prompt_capabilities,
_io_task: io_task,
})
}
}
impl AgentConnection for AcpConnection {
fn new_thread(
self: Rc<Self>,
project: Entity<Project>,
cwd: &Path,
cx: &mut App,
) -> Task<Result<Entity<AcpThread>>> {
let conn = self.connection.clone();
let sessions = self.sessions.clone();
let cwd = cwd.to_path_buf();
cx.spawn(async move |cx| {
let response = conn
.new_session(acp::NewSessionRequest {
mcp_servers: vec![],
cwd,
})
.await
.map_err(|err| {
if err.code == acp::ErrorCode::AUTH_REQUIRED.code {
let mut error = AuthRequired::new();
if err.message != acp::ErrorCode::AUTH_REQUIRED.message {
error = error.with_description(err.message);
}
anyhow!(error)
} else {
anyhow!(err)
}
})?;
let session_id = response.session_id;
let action_log = cx.new(|_| ActionLog::new(project.clone()))?;
let thread = cx.new(|_cx| {
AcpThread::new(
self.server_name,
self.clone(),
project,
action_log,
session_id.clone(),
)
})?;
let session = AcpSession {
thread: thread.downgrade(),
suppress_abort_err: false,
};
sessions.borrow_mut().insert(session_id, session);
Ok(thread)
})
}
fn auth_methods(&self) -> &[acp::AuthMethod] {
&self.auth_methods
}
fn authenticate(&self, method_id: acp::AuthMethodId, cx: &mut App) -> Task<Result<()>> {
let conn = self.connection.clone();
cx.foreground_executor().spawn(async move {
let result = conn
.authenticate(acp::AuthenticateRequest {
method_id: method_id.clone(),
})
.await?;
Ok(result)
})
}
fn prompt(
&self,
_id: Option<acp_thread::UserMessageId>,
params: acp::PromptRequest,
cx: &mut App,
) -> Task<Result<acp::PromptResponse>> {
let conn = self.connection.clone();
let sessions = self.sessions.clone();
let session_id = params.session_id.clone();
cx.foreground_executor().spawn(async move {
let result = conn.prompt(params).await;
let mut suppress_abort_err = false;
if let Some(session) = sessions.borrow_mut().get_mut(&session_id) {
suppress_abort_err = session.suppress_abort_err;
session.suppress_abort_err = false;
}
match result {
Ok(response) => Ok(response),
Err(err) => {
if err.code != ErrorCode::INTERNAL_ERROR.code {
anyhow::bail!(err)
}
let Some(data) = &err.data else {
anyhow::bail!(err)
};
// Temporary workaround until the following PR is generally available:
// https://github.com/google-gemini/gemini-cli/pull/6656
#[derive(Deserialize)]
#[serde(deny_unknown_fields)]
struct ErrorDetails {
details: Box<str>,
}
match serde_json::from_value(data.clone()) {
Ok(ErrorDetails { details }) => {
if suppress_abort_err && details.contains("This operation was aborted")
{
Ok(acp::PromptResponse {
stop_reason: acp::StopReason::Cancelled,
})
} else {
Err(anyhow!(details))
}
}
Err(_) => Err(anyhow!(err)),
}
}
}
})
}
fn prompt_capabilities(&self) -> acp::PromptCapabilities {
self.prompt_capabilities
}
fn cancel(&self, session_id: &acp::SessionId, cx: &mut App) {
if let Some(session) = self.sessions.borrow_mut().get_mut(session_id) {
session.suppress_abort_err = true;
}
let conn = self.connection.clone();
let params = acp::CancelNotification {
session_id: session_id.clone(),
};
cx.foreground_executor()
.spawn(async move { conn.cancel(params).await })
.detach();
}
fn into_any(self: Rc<Self>) -> Rc<dyn Any> {
self
}
}
struct ClientDelegate {
sessions: Rc<RefCell<HashMap<acp::SessionId, AcpSession>>>,
cx: AsyncApp,
}
impl acp::Client for ClientDelegate {
async fn request_permission(
&self,
arguments: acp::RequestPermissionRequest,
) -> Result<acp::RequestPermissionResponse, acp::Error> {
let cx = &mut self.cx.clone();
let rx = self
.sessions
.borrow()
.get(&arguments.session_id)
.context("Failed to get session")?
.thread
.update(cx, |thread, cx| {
thread.request_tool_call_authorization(arguments.tool_call, arguments.options, cx)
})?;
let result = rx?.await;
let outcome = match result {
Ok(option) => acp::RequestPermissionOutcome::Selected { option_id: option },
Err(oneshot::Canceled) => acp::RequestPermissionOutcome::Cancelled,
};
Ok(acp::RequestPermissionResponse { outcome })
}
async fn write_text_file(
&self,
arguments: acp::WriteTextFileRequest,
) -> Result<(), acp::Error> {
let cx = &mut self.cx.clone();
let task = self
.sessions
.borrow()
.get(&arguments.session_id)
.context("Failed to get session")?
.thread
.update(cx, |thread, cx| {
thread.write_text_file(arguments.path, arguments.content, cx)
})?;
task.await?;
Ok(())
}
async fn read_text_file(
&self,
arguments: acp::ReadTextFileRequest,
) -> Result<acp::ReadTextFileResponse, acp::Error> {
let cx = &mut self.cx.clone();
let task = self
.sessions
.borrow()
.get(&arguments.session_id)
.context("Failed to get session")?
.thread
.update(cx, |thread, cx| {
thread.read_text_file(arguments.path, arguments.line, arguments.limit, false, cx)
})?;
let content = task.await?;
Ok(acp::ReadTextFileResponse { content })
}
async fn session_notification(
&self,
notification: acp::SessionNotification,
) -> Result<(), acp::Error> {
let cx = &mut self.cx.clone();
let sessions = self.sessions.borrow();
let session = sessions
.get(&notification.session_id)
.context("Failed to get session")?;
session.thread.update(cx, |thread, cx| {
thread.handle_session_update(notification.update, cx)
})??;
Ok(())
}
}

View file

@ -294,7 +294,6 @@ impl AgentConnection for ClaudeAgentConnection {
fn prompt( fn prompt(
&self, &self,
_id: Option<acp_thread::UserMessageId>,
params: acp::PromptRequest, params: acp::PromptRequest,
cx: &mut App, cx: &mut App,
) -> Task<Result<acp::PromptResponse>> { ) -> Task<Result<acp::PromptResponse>> {

View file

@ -1,5 +1,5 @@
use crate::AgentServer; use crate::AgentServer;
use acp_thread::{AcpThread, AgentThreadEntry, ToolCall, ToolCallStatus}; use acp_thread::{AcpThread, AgentThreadEntry, ToolCall, ToolCallStatus, new_prompt_id};
use agent_client_protocol as acp; use agent_client_protocol as acp;
use futures::{FutureExt, StreamExt, channel::mpsc, select}; use futures::{FutureExt, StreamExt, channel::mpsc, select};
use gpui::{AppContext, Entity, TestAppContext}; use gpui::{AppContext, Entity, TestAppContext};
@ -77,6 +77,7 @@ where
thread thread
.update(cx, |thread, cx| { .update(cx, |thread, cx| {
thread.send( thread.send(
new_prompt_id(),
vec![ vec![
acp::ContentBlock::Text(acp::TextContent { acp::ContentBlock::Text(acp::TextContent {
text: "Read the file ".into(), text: "Read the file ".into(),

View file

@ -6,7 +6,7 @@ use agent2::HistoryStore;
use collections::HashMap; use collections::HashMap;
use editor::{Editor, EditorMode, MinimapVisibility}; use editor::{Editor, EditorMode, MinimapVisibility};
use gpui::{ use gpui::{
AnyEntity, App, AppContext as _, Entity, EntityId, EventEmitter, Focusable, ScrollHandle, AnyEntity, App, AppContext as _, Entity, EntityId, EventEmitter, Focusable,
TextStyleRefinement, WeakEntity, Window, TextStyleRefinement, WeakEntity, Window,
}; };
use language::language_settings::SoftWrap; use language::language_settings::SoftWrap;
@ -67,7 +67,7 @@ impl EntryViewState {
match thread_entry { match thread_entry {
AgentThreadEntry::UserMessage(message) => { AgentThreadEntry::UserMessage(message) => {
let has_id = message.id.is_some(); let has_id = message.prompt_id.is_some();
let chunks = message.chunks.clone(); let chunks = message.chunks.clone();
if let Some(Entry::UserMessage(editor)) = self.entries.get_mut(index) { if let Some(Entry::UserMessage(editor)) = self.entries.get_mut(index) {
if !editor.focus_handle(cx).is_focused(window) { if !editor.focus_handle(cx).is_focused(window) {
@ -154,22 +154,10 @@ impl EntryViewState {
}); });
} }
} }
AgentThreadEntry::AssistantMessage(message) => { AgentThreadEntry::AssistantMessage(_) => {
let entry = if let Some(Entry::AssistantMessage(entry)) = if index == self.entries.len() {
self.entries.get_mut(index) self.entries.push(Entry::empty())
{ }
entry
} else {
self.set_entry(
index,
Entry::AssistantMessage(AssistantMessageEntry::default()),
);
let Some(Entry::AssistantMessage(entry)) = self.entries.get_mut(index) else {
unreachable!()
};
entry
};
entry.sync(message);
} }
}; };
} }
@ -189,7 +177,7 @@ impl EntryViewState {
pub fn settings_changed(&mut self, cx: &mut App) { pub fn settings_changed(&mut self, cx: &mut App) {
for entry in self.entries.iter() { for entry in self.entries.iter() {
match entry { match entry {
Entry::UserMessage { .. } | Entry::AssistantMessage { .. } => {} Entry::UserMessage { .. } => {}
Entry::Content(response_views) => { Entry::Content(response_views) => {
for view in response_views.values() { for view in response_views.values() {
if let Ok(diff_editor) = view.clone().downcast::<Editor>() { if let Ok(diff_editor) = view.clone().downcast::<Editor>() {
@ -220,29 +208,9 @@ pub enum ViewEvent {
MessageEditorEvent(Entity<MessageEditor>, MessageEditorEvent), MessageEditorEvent(Entity<MessageEditor>, MessageEditorEvent),
} }
#[derive(Default, Debug)]
pub struct AssistantMessageEntry {
scroll_handles_by_chunk_index: HashMap<usize, ScrollHandle>,
}
impl AssistantMessageEntry {
pub fn scroll_handle_for_chunk(&self, ix: usize) -> Option<ScrollHandle> {
self.scroll_handles_by_chunk_index.get(&ix).cloned()
}
pub fn sync(&mut self, message: &acp_thread::AssistantMessage) {
if let Some(acp_thread::AssistantMessageChunk::Thought { .. }) = message.chunks.last() {
let ix = message.chunks.len() - 1;
let handle = self.scroll_handles_by_chunk_index.entry(ix).or_default();
handle.scroll_to_bottom();
}
}
}
#[derive(Debug)] #[derive(Debug)]
pub enum Entry { pub enum Entry {
UserMessage(Entity<MessageEditor>), UserMessage(Entity<MessageEditor>),
AssistantMessage(AssistantMessageEntry),
Content(HashMap<EntityId, AnyEntity>), Content(HashMap<EntityId, AnyEntity>),
} }
@ -250,7 +218,7 @@ impl Entry {
pub fn message_editor(&self) -> Option<&Entity<MessageEditor>> { pub fn message_editor(&self) -> Option<&Entity<MessageEditor>> {
match self { match self {
Self::UserMessage(editor) => Some(editor), Self::UserMessage(editor) => Some(editor),
Self::AssistantMessage(_) | Self::Content(_) => None, Entry::Content(_) => None,
} }
} }
@ -271,16 +239,6 @@ impl Entry {
.map(|entity| entity.downcast::<TerminalView>().unwrap()) .map(|entity| entity.downcast::<TerminalView>().unwrap())
} }
pub fn scroll_handle_for_assistant_message_chunk(
&self,
chunk_ix: usize,
) -> Option<ScrollHandle> {
match self {
Self::AssistantMessage(message) => message.scroll_handle_for_chunk(chunk_ix),
Self::UserMessage(_) | Self::Content(_) => None,
}
}
fn content_map(&self) -> Option<&HashMap<EntityId, AnyEntity>> { fn content_map(&self) -> Option<&HashMap<EntityId, AnyEntity>> {
match self { match self {
Self::Content(map) => Some(map), Self::Content(map) => Some(map),
@ -296,7 +254,7 @@ impl Entry {
pub fn has_content(&self) -> bool { pub fn has_content(&self) -> bool {
match self { match self {
Self::Content(map) => !map.is_empty(), Self::Content(map) => !map.is_empty(),
Self::UserMessage(_) | Self::AssistantMessage(_) => false, Self::UserMessage(_) => false,
} }
} }
} }

File diff suppressed because it is too large Load diff

View file

@ -3,23 +3,20 @@ mod configure_context_server_modal;
mod manage_profiles_modal; mod manage_profiles_modal;
mod tool_picker; mod tool_picker;
use std::{ops::Range, sync::Arc, time::Duration}; use std::{sync::Arc, time::Duration};
use agent_servers::{AgentServerCommand, AgentServerSettings, AllAgentServersSettings, Gemini}; use agent_servers::{AgentServerCommand, AllAgentServersSettings, Gemini};
use agent_settings::AgentSettings; use agent_settings::AgentSettings;
use anyhow::Result;
use assistant_tool::{ToolSource, ToolWorkingSet}; use assistant_tool::{ToolSource, ToolWorkingSet};
use cloud_llm_client::Plan; use cloud_llm_client::Plan;
use collections::HashMap; use collections::HashMap;
use context_server::ContextServerId; use context_server::ContextServerId;
use editor::{Editor, SelectionEffects, scroll::Autoscroll};
use extension::ExtensionManifest; use extension::ExtensionManifest;
use extension_host::ExtensionStore; use extension_host::ExtensionStore;
use fs::Fs; use fs::Fs;
use gpui::{ use gpui::{
Action, Animation, AnimationExt as _, AnyView, App, AsyncWindowContext, Corner, Entity, Action, Animation, AnimationExt as _, AnyView, App, Corner, Entity, EventEmitter, FocusHandle,
EventEmitter, FocusHandle, Focusable, Hsla, ScrollHandle, Subscription, Task, Transformation, Focusable, Hsla, ScrollHandle, Subscription, Task, Transformation, WeakEntity, percentage,
WeakEntity, percentage,
}; };
use language::LanguageRegistry; use language::LanguageRegistry;
use language_model::{ use language_model::{
@ -37,7 +34,7 @@ use ui::{
Scrollbar, ScrollbarState, Switch, SwitchColor, SwitchField, Tooltip, prelude::*, Scrollbar, ScrollbarState, Switch, SwitchColor, SwitchField, Tooltip, prelude::*,
}; };
use util::ResultExt as _; use util::ResultExt as _;
use workspace::{Workspace, create_and_open_local_file}; use workspace::Workspace;
use zed_actions::ExtensionCategoryFilter; use zed_actions::ExtensionCategoryFilter;
pub(crate) use configure_context_server_modal::ConfigureContextServerModal; pub(crate) use configure_context_server_modal::ConfigureContextServerModal;
@ -1061,39 +1058,10 @@ impl AgentConfiguration {
.child( .child(
v_flex() v_flex()
.gap_0p5() .gap_0p5()
.child( .child(Headline::new("External Agents"))
h_flex()
.w_full()
.gap_2()
.justify_between()
.child(Headline::new("External Agents"))
.child(
Button::new("add-agent", "Add Agent")
.icon_position(IconPosition::Start)
.icon(IconName::Plus)
.icon_size(IconSize::Small)
.icon_color(Color::Muted)
.label_size(LabelSize::Small)
.on_click(
move |_, window, cx| {
if let Some(workspace) = window.root().flatten() {
let workspace = workspace.downgrade();
window
.spawn(cx, async |cx| {
open_new_agent_servers_entry_in_settings_editor(
workspace,
cx,
).await
})
.detach_and_log_err(cx);
}
}
),
)
)
.child( .child(
Label::new( Label::new(
"Bring the agent of your choice to Zed via our new Agent Client Protocol.", "Use the full power of Zed's UI with your favorite agent, connected via the Agent Client Protocol.",
) )
.color(Color::Muted), .color(Color::Muted),
), ),
@ -1356,109 +1324,3 @@ fn show_unable_to_uninstall_extension_with_context_server(
workspace.toggle_status_toast(status_toast, cx); workspace.toggle_status_toast(status_toast, cx);
} }
async fn open_new_agent_servers_entry_in_settings_editor(
workspace: WeakEntity<Workspace>,
cx: &mut AsyncWindowContext,
) -> Result<()> {
let settings_editor = workspace
.update_in(cx, |_, window, cx| {
create_and_open_local_file(paths::settings_file(), window, cx, || {
settings::initial_user_settings_content().as_ref().into()
})
})?
.await?
.downcast::<Editor>()
.unwrap();
settings_editor
.downgrade()
.update_in(cx, |item, window, cx| {
let text = item.buffer().read(cx).snapshot(cx).text();
let settings = cx.global::<SettingsStore>();
let mut unique_server_name = None;
let edits = settings.edits_for_update::<AllAgentServersSettings>(&text, |file| {
let server_name: Option<SharedString> = (0..u8::MAX)
.map(|i| {
if i == 0 {
"your_agent".into()
} else {
format!("your_agent_{}", i).into()
}
})
.find(|name| !file.custom.contains_key(name));
if let Some(server_name) = server_name {
unique_server_name = Some(server_name.clone());
file.custom.insert(
server_name,
AgentServerSettings {
command: AgentServerCommand {
path: "path_to_executable".into(),
args: vec![],
env: Some(HashMap::default()),
},
},
);
}
});
if edits.is_empty() {
return;
}
let ranges = edits
.iter()
.map(|(range, _)| range.clone())
.collect::<Vec<_>>();
item.edit(edits, cx);
if let Some((unique_server_name, buffer)) =
unique_server_name.zip(item.buffer().read(cx).as_singleton())
{
let snapshot = buffer.read(cx).snapshot();
if let Some(range) =
find_text_in_buffer(&unique_server_name, ranges[0].start, &snapshot)
{
item.change_selections(
SelectionEffects::scroll(Autoscroll::newest()),
window,
cx,
|selections| {
selections.select_ranges(vec![range]);
},
);
}
}
})
}
fn find_text_in_buffer(
text: &str,
start: usize,
snapshot: &language::BufferSnapshot,
) -> Option<Range<usize>> {
let chars = text.chars().collect::<Vec<char>>();
let mut offset = start;
let mut char_offset = 0;
for c in snapshot.chars_at(start) {
if char_offset >= chars.len() {
break;
}
offset += 1;
if c == chars[char_offset] {
char_offset += 1;
} else {
char_offset = 0;
}
}
if char_offset == chars.len() {
Some(offset.saturating_sub(chars.len())..offset)
} else {
None
}
}

View file

@ -14,7 +14,6 @@ use zed_actions::agent::ReauthenticateAgent;
use crate::acp::{AcpThreadHistory, ThreadHistoryEvent}; use crate::acp::{AcpThreadHistory, ThreadHistoryEvent};
use crate::agent_diff::AgentDiffThread; use crate::agent_diff::AgentDiffThread;
use crate::ui::AcpOnboardingModal;
use crate::{ use crate::{
AddContextServer, AgentDiffPane, ContinueThread, ContinueWithBurnMode, AddContextServer, AgentDiffPane, ContinueThread, ContinueWithBurnMode,
DeleteRecentlyOpenThread, ExpandMessageEditor, Follow, InlineAssistant, NewTextThread, DeleteRecentlyOpenThread, ExpandMessageEditor, Follow, InlineAssistant, NewTextThread,
@ -78,10 +77,7 @@ use workspace::{
}; };
use zed_actions::{ use zed_actions::{
DecreaseBufferFontSize, IncreaseBufferFontSize, ResetBufferFontSize, DecreaseBufferFontSize, IncreaseBufferFontSize, ResetBufferFontSize,
agent::{ agent::{OpenOnboardingModal, OpenSettings, ResetOnboarding, ToggleModelSelector},
OpenAcpOnboardingModal, OpenOnboardingModal, OpenSettings, ResetOnboarding,
ToggleModelSelector,
},
assistant::{OpenRulesLibrary, ToggleFocus}, assistant::{OpenRulesLibrary, ToggleFocus},
}; };
@ -205,9 +201,6 @@ pub fn init(cx: &mut App) {
.register_action(|workspace, _: &OpenOnboardingModal, window, cx| { .register_action(|workspace, _: &OpenOnboardingModal, window, cx| {
AgentOnboardingModal::toggle(workspace, window, cx) AgentOnboardingModal::toggle(workspace, window, cx)
}) })
.register_action(|workspace, _: &OpenAcpOnboardingModal, window, cx| {
AcpOnboardingModal::toggle(workspace, window, cx)
})
.register_action(|_workspace, _: &ResetOnboarding, window, cx| { .register_action(|_workspace, _: &ResetOnboarding, window, cx| {
window.dispatch_action(workspace::RestoreBanner.boxed_clone(), cx); window.dispatch_action(workspace::RestoreBanner.boxed_clone(), cx);
window.refresh(); window.refresh();
@ -598,6 +591,17 @@ impl AgentPanel {
None None
}; };
// Wait for the Gemini/Native feature flag to be available.
let client = workspace.read_with(cx, |workspace, _| workspace.client().clone())?;
if !client.status().borrow().is_signed_out() {
cx.update(|_, cx| {
cx.wait_for_flag_or_timeout::<feature_flags::GeminiAndNativeFeatureFlag>(
Duration::from_secs(2),
)
})?
.await;
}
let panel = workspace.update_in(cx, |workspace, window, cx| { let panel = workspace.update_in(cx, |workspace, window, cx| {
let panel = cx.new(|cx| { let panel = cx.new(|cx| {
Self::new( Self::new(
@ -1848,6 +1852,19 @@ impl AgentPanel {
menu menu
} }
pub fn set_selected_agent(
&mut self,
agent: AgentType,
window: &mut Window,
cx: &mut Context<Self>,
) {
if self.selected_agent != agent {
self.selected_agent = agent.clone();
self.serialize(cx);
}
self.new_agent_thread(agent, window, cx);
}
pub fn selected_agent(&self) -> AgentType { pub fn selected_agent(&self) -> AgentType {
self.selected_agent.clone() self.selected_agent.clone()
} }
@ -1858,11 +1875,6 @@ impl AgentPanel {
window: &mut Window, window: &mut Window,
cx: &mut Context<Self>, cx: &mut Context<Self>,
) { ) {
if self.selected_agent != agent {
self.selected_agent = agent.clone();
self.serialize(cx);
}
match agent { match agent {
AgentType::Zed => { AgentType::Zed => {
window.dispatch_action( window.dispatch_action(
@ -2543,7 +2555,7 @@ impl AgentPanel {
workspace.panel::<AgentPanel>(cx) workspace.panel::<AgentPanel>(cx)
{ {
panel.update(cx, |panel, cx| { panel.update(cx, |panel, cx| {
panel.new_agent_thread( panel.set_selected_agent(
AgentType::NativeAgent, AgentType::NativeAgent,
window, window,
cx, cx,
@ -2569,7 +2581,7 @@ impl AgentPanel {
workspace.panel::<AgentPanel>(cx) workspace.panel::<AgentPanel>(cx)
{ {
panel.update(cx, |panel, cx| { panel.update(cx, |panel, cx| {
panel.new_agent_thread( panel.set_selected_agent(
AgentType::TextThread, AgentType::TextThread,
window, window,
cx, cx,
@ -2597,7 +2609,7 @@ impl AgentPanel {
workspace.panel::<AgentPanel>(cx) workspace.panel::<AgentPanel>(cx)
{ {
panel.update(cx, |panel, cx| { panel.update(cx, |panel, cx| {
panel.new_agent_thread( panel.set_selected_agent(
AgentType::Gemini, AgentType::Gemini,
window, window,
cx, cx,
@ -2624,7 +2636,7 @@ impl AgentPanel {
workspace.panel::<AgentPanel>(cx) workspace.panel::<AgentPanel>(cx)
{ {
panel.update(cx, |panel, cx| { panel.update(cx, |panel, cx| {
panel.new_agent_thread( panel.set_selected_agent(
AgentType::ClaudeCode, AgentType::ClaudeCode,
window, window,
cx, cx,
@ -2657,7 +2669,7 @@ impl AgentPanel {
workspace.panel::<AgentPanel>(cx) workspace.panel::<AgentPanel>(cx)
{ {
panel.update(cx, |panel, cx| { panel.update(cx, |panel, cx| {
panel.new_agent_thread( panel.set_selected_agent(
AgentType::Custom { AgentType::Custom {
name: agent_name name: agent_name
.clone(), .clone(),
@ -2681,9 +2693,9 @@ impl AgentPanel {
}) })
.when(cx.has_flag::<GeminiAndNativeFeatureFlag>(), |menu| { .when(cx.has_flag::<GeminiAndNativeFeatureFlag>(), |menu| {
menu.separator().link( menu.separator().link(
"Add Other Agents", "Add Your Own Agent",
OpenBrowser { OpenBrowser {
url: zed_urls::external_agents_docs(cx), url: "https://agentclientprotocol.com/".into(),
} }
.boxed_clone(), .boxed_clone(),
) )

View file

@ -6,8 +6,7 @@ use feature_flags::ZedProFeatureFlag;
use fuzzy::{StringMatch, StringMatchCandidate, match_strings}; use fuzzy::{StringMatch, StringMatchCandidate, match_strings};
use gpui::{Action, AnyElement, App, BackgroundExecutor, DismissEvent, Subscription, Task}; use gpui::{Action, AnyElement, App, BackgroundExecutor, DismissEvent, Subscription, Task};
use language_model::{ use language_model::{
AuthenticateError, ConfiguredModel, LanguageModel, LanguageModelProviderId, ConfiguredModel, LanguageModel, LanguageModelProviderId, LanguageModelRegistry,
LanguageModelRegistry,
}; };
use ordered_float::OrderedFloat; use ordered_float::OrderedFloat;
use picker::{Picker, PickerDelegate}; use picker::{Picker, PickerDelegate};
@ -77,7 +76,6 @@ pub struct LanguageModelPickerDelegate {
all_models: Arc<GroupedModels>, all_models: Arc<GroupedModels>,
filtered_entries: Vec<LanguageModelPickerEntry>, filtered_entries: Vec<LanguageModelPickerEntry>,
selected_index: usize, selected_index: usize,
_authenticate_all_providers_task: Task<()>,
_subscriptions: Vec<Subscription>, _subscriptions: Vec<Subscription>,
} }
@ -98,7 +96,6 @@ impl LanguageModelPickerDelegate {
selected_index: Self::get_active_model_index(&entries, get_active_model(cx)), selected_index: Self::get_active_model_index(&entries, get_active_model(cx)),
filtered_entries: entries, filtered_entries: entries,
get_active_model: Arc::new(get_active_model), get_active_model: Arc::new(get_active_model),
_authenticate_all_providers_task: Self::authenticate_all_providers(cx),
_subscriptions: vec![cx.subscribe_in( _subscriptions: vec![cx.subscribe_in(
&LanguageModelRegistry::global(cx), &LanguageModelRegistry::global(cx),
window, window,
@ -142,56 +139,6 @@ impl LanguageModelPickerDelegate {
.unwrap_or(0) .unwrap_or(0)
} }
/// Authenticates all providers in the [`LanguageModelRegistry`].
///
/// We do this so that we can populate the language selector with all of the
/// models from the configured providers.
fn authenticate_all_providers(cx: &mut App) -> Task<()> {
let authenticate_all_providers = LanguageModelRegistry::global(cx)
.read(cx)
.providers()
.iter()
.map(|provider| (provider.id(), provider.name(), provider.authenticate(cx)))
.collect::<Vec<_>>();
cx.spawn(async move |_cx| {
for (provider_id, provider_name, authenticate_task) in authenticate_all_providers {
if let Err(err) = authenticate_task.await {
if matches!(err, AuthenticateError::CredentialsNotFound) {
// Since we're authenticating these providers in the
// background for the purposes of populating the
// language selector, we don't care about providers
// where the credentials are not found.
} else {
// Some providers have noisy failure states that we
// don't want to spam the logs with every time the
// language model selector is initialized.
//
// Ideally these should have more clear failure modes
// that we know are safe to ignore here, like what we do
// with `CredentialsNotFound` above.
match provider_id.0.as_ref() {
"lmstudio" | "ollama" => {
// LM Studio and Ollama both make fetch requests to the local APIs to determine if they are "authenticated".
//
// These fail noisily, so we don't log them.
}
"copilot_chat" => {
// Copilot Chat returns an error if Copilot is not enabled, so we don't log those errors.
}
_ => {
log::error!(
"Failed to authenticate provider: {}: {err}",
provider_name.0
);
}
}
}
}
}
})
}
pub fn active_model(&self, cx: &App) -> Option<ConfiguredModel> { pub fn active_model(&self, cx: &App) -> Option<ConfiguredModel> {
(self.get_active_model)(cx) (self.get_active_model)(cx)
} }

View file

@ -1,4 +1,3 @@
mod acp_onboarding_modal;
mod agent_notification; mod agent_notification;
mod burn_mode_tooltip; mod burn_mode_tooltip;
mod context_pill; mod context_pill;
@ -7,7 +6,6 @@ mod onboarding_modal;
pub mod preview; pub mod preview;
mod unavailable_editing_tooltip; mod unavailable_editing_tooltip;
pub use acp_onboarding_modal::*;
pub use agent_notification::*; pub use agent_notification::*;
pub use burn_mode_tooltip::*; pub use burn_mode_tooltip::*;
pub use context_pill::*; pub use context_pill::*;

View file

@ -1,254 +0,0 @@
use client::zed_urls;
use gpui::{
ClickEvent, DismissEvent, Entity, EventEmitter, FocusHandle, Focusable, MouseDownEvent, Render,
linear_color_stop, linear_gradient,
};
use ui::{TintColor, Vector, VectorName, prelude::*};
use workspace::{ModalView, Workspace};
use crate::agent_panel::{AgentPanel, AgentType};
macro_rules! acp_onboarding_event {
($name:expr) => {
telemetry::event!($name, source = "ACP Onboarding");
};
($name:expr, $($key:ident $(= $value:expr)?),+ $(,)?) => {
telemetry::event!($name, source = "ACP Onboarding", $($key $(= $value)?),+);
};
}
pub struct AcpOnboardingModal {
focus_handle: FocusHandle,
workspace: Entity<Workspace>,
}
impl AcpOnboardingModal {
pub fn toggle(workspace: &mut Workspace, window: &mut Window, cx: &mut Context<Workspace>) {
let workspace_entity = cx.entity();
workspace.toggle_modal(window, cx, |_window, cx| Self {
workspace: workspace_entity,
focus_handle: cx.focus_handle(),
});
}
fn open_panel(&mut self, _: &ClickEvent, window: &mut Window, cx: &mut Context<Self>) {
self.workspace.update(cx, |workspace, cx| {
workspace.focus_panel::<AgentPanel>(window, cx);
if let Some(panel) = workspace.panel::<AgentPanel>(cx) {
panel.update(cx, |panel, cx| {
panel.new_agent_thread(AgentType::Gemini, window, cx);
});
}
});
cx.emit(DismissEvent);
acp_onboarding_event!("Open Panel Clicked");
}
fn view_docs(&mut self, _: &ClickEvent, _: &mut Window, cx: &mut Context<Self>) {
cx.open_url(&zed_urls::external_agents_docs(cx));
cx.notify();
acp_onboarding_event!("Documentation Link Clicked");
}
fn cancel(&mut self, _: &menu::Cancel, _: &mut Window, cx: &mut Context<Self>) {
cx.emit(DismissEvent);
}
}
impl EventEmitter<DismissEvent> for AcpOnboardingModal {}
impl Focusable for AcpOnboardingModal {
fn focus_handle(&self, _cx: &App) -> FocusHandle {
self.focus_handle.clone()
}
}
impl ModalView for AcpOnboardingModal {}
impl Render for AcpOnboardingModal {
fn render(&mut self, _: &mut Window, cx: &mut Context<Self>) -> impl IntoElement {
let illustration_element = |label: bool, opacity: f32| {
h_flex()
.px_1()
.py_0p5()
.gap_1()
.rounded_sm()
.bg(cx.theme().colors().element_active.opacity(0.05))
.border_1()
.border_color(cx.theme().colors().border)
.border_dashed()
.child(
Icon::new(IconName::Stop)
.size(IconSize::Small)
.color(Color::Custom(cx.theme().colors().text_muted.opacity(0.15))),
)
.map(|this| {
if label {
this.child(
Label::new("Your Agent Here")
.size(LabelSize::Small)
.color(Color::Muted),
)
} else {
this.child(
div().w_16().h_1().rounded_full().bg(cx
.theme()
.colors()
.element_active
.opacity(0.6)),
)
}
})
.opacity(opacity)
};
let illustration = h_flex()
.relative()
.h(rems_from_px(126.))
.bg(cx.theme().colors().editor_background)
.border_b_1()
.border_color(cx.theme().colors().border_variant)
.justify_center()
.gap_8()
.rounded_t_md()
.overflow_hidden()
.child(
div().absolute().inset_0().w(px(515.)).h(px(126.)).child(
Vector::new(VectorName::AcpGrid, rems_from_px(515.), rems_from_px(126.))
.color(ui::Color::Custom(cx.theme().colors().text.opacity(0.02))),
),
)
.child(div().absolute().inset_0().size_full().bg(linear_gradient(
0.,
linear_color_stop(
cx.theme().colors().elevated_surface_background.opacity(0.1),
0.9,
),
linear_color_stop(
cx.theme().colors().elevated_surface_background.opacity(0.),
0.,
),
)))
.child(
div()
.absolute()
.inset_0()
.size_full()
.bg(gpui::black().opacity(0.15)),
)
.child(
h_flex()
.gap_4()
.child(
Vector::new(VectorName::AcpLogo, rems_from_px(106.), rems_from_px(40.))
.color(ui::Color::Custom(cx.theme().colors().text.opacity(0.8))),
)
.child(
Vector::new(
VectorName::AcpLogoSerif,
rems_from_px(111.),
rems_from_px(41.),
)
.color(ui::Color::Custom(cx.theme().colors().text.opacity(0.8))),
),
)
.child(
v_flex()
.gap_1p5()
.child(illustration_element(false, 0.15))
.child(illustration_element(true, 0.3))
.child(
h_flex()
.pl_1()
.pr_2()
.py_0p5()
.gap_1()
.rounded_sm()
.bg(cx.theme().colors().element_active.opacity(0.2))
.border_1()
.border_color(cx.theme().colors().border)
.child(
Icon::new(IconName::AiGemini)
.size(IconSize::Small)
.color(Color::Muted),
)
.child(Label::new("New Gemini CLI Thread").size(LabelSize::Small)),
)
.child(illustration_element(true, 0.3))
.child(illustration_element(false, 0.15)),
);
let heading = v_flex()
.w_full()
.gap_1()
.child(
Label::new("Now Available")
.size(LabelSize::Small)
.color(Color::Muted),
)
.child(Headline::new("Bring Your Own Agent to Zed").size(HeadlineSize::Large));
let copy = "Bring the agent of your choice to Zed via our new Agent Client Protocol (ACP), starting with Google's Gemini CLI integration.";
let open_panel_button = Button::new("open-panel", "Start with Gemini CLI")
.icon_size(IconSize::Indicator)
.style(ButtonStyle::Tinted(TintColor::Accent))
.full_width()
.on_click(cx.listener(Self::open_panel));
let docs_button = Button::new("add-other-agents", "Add Other Agents")
.icon(IconName::ArrowUpRight)
.icon_size(IconSize::Indicator)
.icon_color(Color::Muted)
.full_width()
.on_click(cx.listener(Self::view_docs));
let close_button = h_flex().absolute().top_2().right_2().child(
IconButton::new("cancel", IconName::Close).on_click(cx.listener(
|_, _: &ClickEvent, _window, cx| {
acp_onboarding_event!("Canceled", trigger = "X click");
cx.emit(DismissEvent);
},
)),
);
v_flex()
.id("acp-onboarding")
.key_context("AcpOnboardingModal")
.relative()
.w(rems(34.))
.h_full()
.elevation_3(cx)
.track_focus(&self.focus_handle(cx))
.overflow_hidden()
.on_action(cx.listener(Self::cancel))
.on_action(cx.listener(|_, _: &menu::Cancel, _window, cx| {
acp_onboarding_event!("Canceled", trigger = "Action");
cx.emit(DismissEvent);
}))
.on_any_mouse_down(cx.listener(|this, _: &MouseDownEvent, window, _cx| {
this.focus_handle.focus(window);
}))
.child(illustration)
.child(
v_flex()
.p_4()
.gap_2()
.child(heading)
.child(Label::new(copy).color(Color::Muted))
.child(
v_flex()
.w_full()
.mt_2()
.gap_1()
.child(open_panel_button)
.child(docs_button),
),
)
.child(close_button)
}
}

View file

@ -43,11 +43,3 @@ pub fn ai_privacy_and_security(cx: &App) -> String {
server_url = server_url(cx) server_url = server_url(cx)
) )
} }
/// Returns the URL to Zed AI's external agents documentation.
pub fn external_agents_docs(cx: &App) -> String {
format!(
"{server_url}/docs/ai/external-agents",
server_url = server_url(cx)
)
}

View file

@ -1,10 +1,7 @@
use anyhow::Result; use anyhow::Result;
use db::{ use db::{
query, define_connection, query,
sqlez::{ sqlez::{bindable::Column, statement::Statement},
bindable::Column, domain::Domain, statement::Statement,
thread_safe_connection::ThreadSafeConnection,
},
sqlez_macros::sql, sqlez_macros::sql,
}; };
use serde::{Deserialize, Serialize}; use serde::{Deserialize, Serialize};
@ -53,11 +50,8 @@ impl Column for SerializedCommandInvocation {
} }
} }
pub struct CommandPaletteDB(ThreadSafeConnection); define_connection!(pub static ref COMMAND_PALETTE_HISTORY: CommandPaletteDB<()> =
&[sql!(
impl Domain for CommandPaletteDB {
const NAME: &str = stringify!(CommandPaletteDB);
const MIGRATIONS: &[&str] = &[sql!(
CREATE TABLE IF NOT EXISTS command_invocations( CREATE TABLE IF NOT EXISTS command_invocations(
id INTEGER PRIMARY KEY AUTOINCREMENT, id INTEGER PRIMARY KEY AUTOINCREMENT,
command_name TEXT NOT NULL, command_name TEXT NOT NULL,
@ -65,9 +59,7 @@ impl Domain for CommandPaletteDB {
last_invoked INTEGER DEFAULT (unixepoch()) NOT NULL last_invoked INTEGER DEFAULT (unixepoch()) NOT NULL
) STRICT; ) STRICT;
)]; )];
} );
db::static_connection!(COMMAND_PALETTE_HISTORY, CommandPaletteDB, []);
impl CommandPaletteDB { impl CommandPaletteDB {
pub async fn write_command_invocation( pub async fn write_command_invocation(

View file

@ -110,14 +110,11 @@ pub async fn open_test_db<M: Migrator>(db_name: &str) -> ThreadSafeConnection {
} }
/// Implements a basic DB wrapper for a given domain /// Implements a basic DB wrapper for a given domain
///
/// Arguments:
/// - static variable name for connection
/// - type of connection wrapper
/// - dependencies, whose migrations should be run prior to this domain's migrations
#[macro_export] #[macro_export]
macro_rules! static_connection { macro_rules! define_connection {
($id:ident, $t:ident, [ $($d:ty),* ] $(, $global:ident)?) => { (pub static ref $id:ident: $t:ident<()> = $migrations:expr; $($global:ident)?) => {
pub struct $t($crate::sqlez::thread_safe_connection::ThreadSafeConnection);
impl ::std::ops::Deref for $t { impl ::std::ops::Deref for $t {
type Target = $crate::sqlez::thread_safe_connection::ThreadSafeConnection; type Target = $crate::sqlez::thread_safe_connection::ThreadSafeConnection;
@ -126,6 +123,16 @@ macro_rules! static_connection {
} }
} }
impl $crate::sqlez::domain::Domain for $t {
fn name() -> &'static str {
stringify!($t)
}
fn migrations() -> &'static [&'static str] {
$migrations
}
}
impl $t { impl $t {
#[cfg(any(test, feature = "test-support"))] #[cfg(any(test, feature = "test-support"))]
pub async fn open_test_db(name: &'static str) -> Self { pub async fn open_test_db(name: &'static str) -> Self {
@ -135,8 +142,7 @@ macro_rules! static_connection {
#[cfg(any(test, feature = "test-support"))] #[cfg(any(test, feature = "test-support"))]
pub static $id: std::sync::LazyLock<$t> = std::sync::LazyLock::new(|| { pub static $id: std::sync::LazyLock<$t> = std::sync::LazyLock::new(|| {
#[allow(unused_parens)] $t($crate::smol::block_on($crate::open_test_db::<$t>(stringify!($id))))
$t($crate::smol::block_on($crate::open_test_db::<($($d,)* $t)>(stringify!($id))))
}); });
#[cfg(not(any(test, feature = "test-support")))] #[cfg(not(any(test, feature = "test-support")))]
@ -147,10 +153,46 @@ macro_rules! static_connection {
} else { } else {
$crate::RELEASE_CHANNEL.dev_name() $crate::RELEASE_CHANNEL.dev_name()
}; };
#[allow(unused_parens)] $t($crate::smol::block_on($crate::open_db::<$t>(db_dir, scope)))
$t($crate::smol::block_on($crate::open_db::<($($d,)* $t)>(db_dir, scope)))
}); });
} };
(pub static ref $id:ident: $t:ident<$($d:ty),+> = $migrations:expr; $($global:ident)?) => {
pub struct $t($crate::sqlez::thread_safe_connection::ThreadSafeConnection);
impl ::std::ops::Deref for $t {
type Target = $crate::sqlez::thread_safe_connection::ThreadSafeConnection;
fn deref(&self) -> &Self::Target {
&self.0
}
}
impl $crate::sqlez::domain::Domain for $t {
fn name() -> &'static str {
stringify!($t)
}
fn migrations() -> &'static [&'static str] {
$migrations
}
}
#[cfg(any(test, feature = "test-support"))]
pub static $id: std::sync::LazyLock<$t> = std::sync::LazyLock::new(|| {
$t($crate::smol::block_on($crate::open_test_db::<($($d),+, $t)>(stringify!($id))))
});
#[cfg(not(any(test, feature = "test-support")))]
pub static $id: std::sync::LazyLock<$t> = std::sync::LazyLock::new(|| {
let db_dir = $crate::database_dir();
let scope = if false $(|| stringify!($global) == "global")? {
"global"
} else {
$crate::RELEASE_CHANNEL.dev_name()
};
$t($crate::smol::block_on($crate::open_db::<($($d),+, $t)>(db_dir, scope)))
});
};
} }
pub fn write_and_log<F>(cx: &App, db_write: impl FnOnce() -> F + Send + 'static) pub fn write_and_log<F>(cx: &App, db_write: impl FnOnce() -> F + Send + 'static)
@ -177,12 +219,17 @@ mod tests {
enum BadDB {} enum BadDB {}
impl Domain for BadDB { impl Domain for BadDB {
const NAME: &str = "db_tests"; fn name() -> &'static str {
const MIGRATIONS: &[&str] = &[ "db_tests"
sql!(CREATE TABLE test(value);), }
// failure because test already exists
sql!(CREATE TABLE test(value);), fn migrations() -> &'static [&'static str] {
]; &[
sql!(CREATE TABLE test(value);),
// failure because test already exists
sql!(CREATE TABLE test(value);),
]
}
} }
let tempdir = tempfile::Builder::new() let tempdir = tempfile::Builder::new()
@ -204,15 +251,25 @@ mod tests {
enum CorruptedDB {} enum CorruptedDB {}
impl Domain for CorruptedDB { impl Domain for CorruptedDB {
const NAME: &str = "db_tests"; fn name() -> &'static str {
const MIGRATIONS: &[&str] = &[sql!(CREATE TABLE test(value);)]; "db_tests"
}
fn migrations() -> &'static [&'static str] {
&[sql!(CREATE TABLE test(value);)]
}
} }
enum GoodDB {} enum GoodDB {}
impl Domain for GoodDB { impl Domain for GoodDB {
const NAME: &str = "db_tests"; //Notice same name fn name() -> &'static str {
const MIGRATIONS: &[&str] = &[sql!(CREATE TABLE test2(value);)]; "db_tests" //Notice same name
}
fn migrations() -> &'static [&'static str] {
&[sql!(CREATE TABLE test2(value);)] //But different migration
}
} }
let tempdir = tempfile::Builder::new() let tempdir = tempfile::Builder::new()
@ -248,16 +305,25 @@ mod tests {
enum CorruptedDB {} enum CorruptedDB {}
impl Domain for CorruptedDB { impl Domain for CorruptedDB {
const NAME: &str = "db_tests"; fn name() -> &'static str {
"db_tests"
}
const MIGRATIONS: &[&str] = &[sql!(CREATE TABLE test(value);)]; fn migrations() -> &'static [&'static str] {
&[sql!(CREATE TABLE test(value);)]
}
} }
enum GoodDB {} enum GoodDB {}
impl Domain for GoodDB { impl Domain for GoodDB {
const NAME: &str = "db_tests"; //Notice same name fn name() -> &'static str {
const MIGRATIONS: &[&str] = &[sql!(CREATE TABLE test2(value);)]; // But different migration "db_tests" //Notice same name
}
fn migrations() -> &'static [&'static str] {
&[sql!(CREATE TABLE test2(value);)] //But different migration
}
} }
let tempdir = tempfile::Builder::new() let tempdir = tempfile::Builder::new()

View file

@ -2,26 +2,16 @@ use gpui::App;
use sqlez_macros::sql; use sqlez_macros::sql;
use util::ResultExt as _; use util::ResultExt as _;
use crate::{ use crate::{define_connection, query, write_and_log};
query,
sqlez::{domain::Domain, thread_safe_connection::ThreadSafeConnection},
write_and_log,
};
pub struct KeyValueStore(crate::sqlez::thread_safe_connection::ThreadSafeConnection); define_connection!(pub static ref KEY_VALUE_STORE: KeyValueStore<()> =
&[sql!(
impl Domain for KeyValueStore {
const NAME: &str = stringify!(KeyValueStore);
const MIGRATIONS: &[&str] = &[sql!(
CREATE TABLE IF NOT EXISTS kv_store( CREATE TABLE IF NOT EXISTS kv_store(
key TEXT PRIMARY KEY, key TEXT PRIMARY KEY,
value TEXT NOT NULL value TEXT NOT NULL
) STRICT; ) STRICT;
)]; )];
} );
crate::static_connection!(KEY_VALUE_STORE, KeyValueStore, []);
pub trait Dismissable { pub trait Dismissable {
const KEY: &'static str; const KEY: &'static str;
@ -101,19 +91,15 @@ mod tests {
} }
} }
pub struct GlobalKeyValueStore(ThreadSafeConnection); define_connection!(pub static ref GLOBAL_KEY_VALUE_STORE: GlobalKeyValueStore<()> =
&[sql!(
impl Domain for GlobalKeyValueStore {
const NAME: &str = stringify!(GlobalKeyValueStore);
const MIGRATIONS: &[&str] = &[sql!(
CREATE TABLE IF NOT EXISTS kv_store( CREATE TABLE IF NOT EXISTS kv_store(
key TEXT PRIMARY KEY, key TEXT PRIMARY KEY,
value TEXT NOT NULL value TEXT NOT NULL
) STRICT; ) STRICT;
)]; )];
} global
);
crate::static_connection!(GLOBAL_KEY_VALUE_STORE, GlobalKeyValueStore, [], global);
impl GlobalKeyValueStore { impl GlobalKeyValueStore {
query! { query! {

View file

@ -19,10 +19,6 @@ static KEYMAP_LINUX: LazyLock<KeymapFile> = LazyLock::new(|| {
load_keymap("keymaps/default-linux.json").expect("Failed to load Linux keymap") load_keymap("keymaps/default-linux.json").expect("Failed to load Linux keymap")
}); });
static KEYMAP_WINDOWS: LazyLock<KeymapFile> = LazyLock::new(|| {
load_keymap("keymaps/default-windows.json").expect("Failed to load Windows keymap")
});
static ALL_ACTIONS: LazyLock<Vec<ActionDef>> = LazyLock::new(dump_all_gpui_actions); static ALL_ACTIONS: LazyLock<Vec<ActionDef>> = LazyLock::new(dump_all_gpui_actions);
const FRONT_MATTER_COMMENT: &str = "<!-- ZED_META {} -->"; const FRONT_MATTER_COMMENT: &str = "<!-- ZED_META {} -->";
@ -220,7 +216,6 @@ fn find_binding(os: &str, action: &str) -> Option<String> {
let keymap = match os { let keymap = match os {
"macos" => &KEYMAP_MACOS, "macos" => &KEYMAP_MACOS,
"linux" | "freebsd" => &KEYMAP_LINUX, "linux" | "freebsd" => &KEYMAP_LINUX,
"windows" => &KEYMAP_WINDOWS,
_ => unreachable!("Not a valid OS: {}", os), _ => unreachable!("Not a valid OS: {}", os),
}; };

View file

@ -2588,7 +2588,7 @@ impl Editor {
|| binding || binding
.keystrokes() .keystrokes()
.first() .first()
.is_some_and(|keystroke| keystroke.display_modifiers.modified()) .is_some_and(|keystroke| keystroke.modifiers.modified())
})) }))
} }
@ -7686,16 +7686,16 @@ impl Editor {
.keystroke() .keystroke()
{ {
modifiers_held = modifiers_held modifiers_held = modifiers_held
|| (&accept_keystroke.display_modifiers == modifiers || (&accept_keystroke.modifiers == modifiers
&& accept_keystroke.display_modifiers.modified()); && accept_keystroke.modifiers.modified());
}; };
if let Some(accept_partial_keystroke) = self if let Some(accept_partial_keystroke) = self
.accept_edit_prediction_keybind(true, window, cx) .accept_edit_prediction_keybind(true, window, cx)
.keystroke() .keystroke()
{ {
modifiers_held = modifiers_held modifiers_held = modifiers_held
|| (&accept_partial_keystroke.display_modifiers == modifiers || (&accept_partial_keystroke.modifiers == modifiers
&& accept_partial_keystroke.display_modifiers.modified()); && accept_partial_keystroke.modifiers.modified());
} }
if modifiers_held { if modifiers_held {
@ -9044,7 +9044,7 @@ impl Editor {
let is_platform_style_mac = PlatformStyle::platform() == PlatformStyle::Mac; let is_platform_style_mac = PlatformStyle::platform() == PlatformStyle::Mac;
let modifiers_color = if accept_keystroke.display_modifiers == window.modifiers() { let modifiers_color = if accept_keystroke.modifiers == window.modifiers() {
Color::Accent Color::Accent
} else { } else {
Color::Muted Color::Muted
@ -9056,19 +9056,19 @@ impl Editor {
.font(theme::ThemeSettings::get_global(cx).buffer_font.clone()) .font(theme::ThemeSettings::get_global(cx).buffer_font.clone())
.text_size(TextSize::XSmall.rems(cx)) .text_size(TextSize::XSmall.rems(cx))
.child(h_flex().children(ui::render_modifiers( .child(h_flex().children(ui::render_modifiers(
&accept_keystroke.display_modifiers, &accept_keystroke.modifiers,
PlatformStyle::platform(), PlatformStyle::platform(),
Some(modifiers_color), Some(modifiers_color),
Some(IconSize::XSmall.rems().into()), Some(IconSize::XSmall.rems().into()),
true, true,
))) )))
.when(is_platform_style_mac, |parent| { .when(is_platform_style_mac, |parent| {
parent.child(accept_keystroke.display_key.clone()) parent.child(accept_keystroke.key.clone())
}) })
.when(!is_platform_style_mac, |parent| { .when(!is_platform_style_mac, |parent| {
parent.child( parent.child(
Key::new( Key::new(
util::capitalize(&accept_keystroke.display_key), util::capitalize(&accept_keystroke.key),
Some(Color::Default), Some(Color::Default),
) )
.size(Some(IconSize::XSmall.rems().into())), .size(Some(IconSize::XSmall.rems().into())),
@ -9171,7 +9171,7 @@ impl Editor {
max_width: Pixels, max_width: Pixels,
cursor_point: Point, cursor_point: Point,
style: &EditorStyle, style: &EditorStyle,
accept_keystroke: Option<&gpui::KeybindingKeystroke>, accept_keystroke: Option<&gpui::Keystroke>,
_window: &Window, _window: &Window,
cx: &mut Context<Editor>, cx: &mut Context<Editor>,
) -> Option<AnyElement> { ) -> Option<AnyElement> {
@ -9249,7 +9249,7 @@ impl Editor {
accept_keystroke.as_ref(), accept_keystroke.as_ref(),
|el, accept_keystroke| { |el, accept_keystroke| {
el.child(h_flex().children(ui::render_modifiers( el.child(h_flex().children(ui::render_modifiers(
&accept_keystroke.display_modifiers, &accept_keystroke.modifiers,
PlatformStyle::platform(), PlatformStyle::platform(),
Some(Color::Default), Some(Color::Default),
Some(IconSize::XSmall.rems().into()), Some(IconSize::XSmall.rems().into()),
@ -9319,7 +9319,7 @@ impl Editor {
.child(completion), .child(completion),
) )
.when_some(accept_keystroke, |el, accept_keystroke| { .when_some(accept_keystroke, |el, accept_keystroke| {
if !accept_keystroke.display_modifiers.modified() { if !accept_keystroke.modifiers.modified() {
return el; return el;
} }
@ -9338,7 +9338,7 @@ impl Editor {
.font(theme::ThemeSettings::get_global(cx).buffer_font.clone()) .font(theme::ThemeSettings::get_global(cx).buffer_font.clone())
.when(is_platform_style_mac, |parent| parent.gap_1()) .when(is_platform_style_mac, |parent| parent.gap_1())
.child(h_flex().children(ui::render_modifiers( .child(h_flex().children(ui::render_modifiers(
&accept_keystroke.display_modifiers, &accept_keystroke.modifiers,
PlatformStyle::platform(), PlatformStyle::platform(),
Some(if !has_completion { Some(if !has_completion {
Color::Muted Color::Muted

View file

@ -43,10 +43,10 @@ use gpui::{
Bounds, ClickEvent, ClipboardItem, ContentMask, Context, Corner, Corners, CursorStyle, Bounds, ClickEvent, ClipboardItem, ContentMask, Context, Corner, Corners, CursorStyle,
DispatchPhase, Edges, Element, ElementInputHandler, Entity, Focusable as _, FontId, DispatchPhase, Edges, Element, ElementInputHandler, Entity, Focusable as _, FontId,
GlobalElementId, Hitbox, HitboxBehavior, Hsla, InteractiveElement, IntoElement, IsZero, GlobalElementId, Hitbox, HitboxBehavior, Hsla, InteractiveElement, IntoElement, IsZero,
KeybindingKeystroke, Length, ModifiersChangedEvent, MouseButton, MouseClickEvent, Keystroke, Length, ModifiersChangedEvent, MouseButton, MouseClickEvent, MouseDownEvent,
MouseDownEvent, MouseMoveEvent, MouseUpEvent, PaintQuad, ParentElement, Pixels, ScrollDelta, MouseMoveEvent, MouseUpEvent, PaintQuad, ParentElement, Pixels, ScrollDelta, ScrollHandle,
ScrollHandle, ScrollWheelEvent, ShapedLine, SharedString, Size, StatefulInteractiveElement, ScrollWheelEvent, ShapedLine, SharedString, Size, StatefulInteractiveElement, Style, Styled,
Style, Styled, TextRun, TextStyleRefinement, WeakEntity, Window, anchored, deferred, div, fill, TextRun, TextStyleRefinement, WeakEntity, Window, anchored, deferred, div, fill,
linear_color_stop, linear_gradient, outline, point, px, quad, relative, size, solid_background, linear_color_stop, linear_gradient, outline, point, px, quad, relative, size, solid_background,
transparent_black, transparent_black,
}; };
@ -7150,7 +7150,7 @@ fn header_jump_data(
pub struct AcceptEditPredictionBinding(pub(crate) Option<gpui::KeyBinding>); pub struct AcceptEditPredictionBinding(pub(crate) Option<gpui::KeyBinding>);
impl AcceptEditPredictionBinding { impl AcceptEditPredictionBinding {
pub fn keystroke(&self) -> Option<&KeybindingKeystroke> { pub fn keystroke(&self) -> Option<&Keystroke> {
if let Some(binding) = self.0.as_ref() { if let Some(binding) = self.0.as_ref() {
match &binding.keystrokes() { match &binding.keystrokes() {
[keystroke, ..] => Some(keystroke), [keystroke, ..] => Some(keystroke),

View file

@ -1,17 +1,13 @@
use anyhow::Result; use anyhow::Result;
use db::{ use db::sqlez::bindable::{Bind, Column, StaticColumnCount};
query, use db::sqlez::statement::Statement;
sqlez::{
bindable::{Bind, Column, StaticColumnCount},
domain::Domain,
statement::Statement,
},
sqlez_macros::sql,
};
use fs::MTime; use fs::MTime;
use itertools::Itertools as _; use itertools::Itertools as _;
use std::path::PathBuf; use std::path::PathBuf;
use db::sqlez_macros::sql;
use db::{define_connection, query};
use workspace::{ItemId, WorkspaceDb, WorkspaceId}; use workspace::{ItemId, WorkspaceDb, WorkspaceId};
#[derive(Clone, Debug, PartialEq, Default)] #[derive(Clone, Debug, PartialEq, Default)]
@ -87,11 +83,7 @@ impl Column for SerializedEditor {
} }
} }
pub struct EditorDb(db::sqlez::thread_safe_connection::ThreadSafeConnection); define_connection!(
impl Domain for EditorDb {
const NAME: &str = stringify!(EditorDb);
// Current schema shape using pseudo-rust syntax: // Current schema shape using pseudo-rust syntax:
// editors( // editors(
// item_id: usize, // item_id: usize,
@ -121,8 +113,7 @@ impl Domain for EditorDb {
// start: usize, // start: usize,
// end: usize, // end: usize,
// ) // )
pub static ref DB: EditorDb<WorkspaceDb> = &[
const MIGRATIONS: &[&str] = &[
sql! ( sql! (
CREATE TABLE editors( CREATE TABLE editors(
item_id INTEGER NOT NULL, item_id INTEGER NOT NULL,
@ -198,9 +189,7 @@ impl Domain for EditorDb {
) STRICT; ) STRICT;
), ),
]; ];
} );
db::static_connection!(DB, EditorDb, [WorkspaceDb]);
// https://www.sqlite.org/limits.html // https://www.sqlite.org/limits.html
// > <..> the maximum value of a host parameter number is SQLITE_MAX_VARIABLE_NUMBER, // > <..> the maximum value of a host parameter number is SQLITE_MAX_VARIABLE_NUMBER,

View file

@ -98,10 +98,6 @@ impl FeatureFlag for GeminiAndNativeFeatureFlag {
// integration too, and we'd like to turn Gemini/Native on in new builds // integration too, and we'd like to turn Gemini/Native on in new builds
// without enabling Claude Code in old builds. // without enabling Claude Code in old builds.
const NAME: &'static str = "gemini-and-native"; const NAME: &'static str = "gemini-and-native";
fn enabled_for_all() -> bool {
true
}
} }
pub struct ClaudeCodeFeatureFlag; pub struct ClaudeCodeFeatureFlag;
@ -205,7 +201,7 @@ impl FeatureFlagAppExt for App {
fn has_flag<T: FeatureFlag>(&self) -> bool { fn has_flag<T: FeatureFlag>(&self) -> bool {
self.try_global::<FeatureFlags>() self.try_global::<FeatureFlags>()
.map(|flags| flags.has_flag::<T>()) .map(|flags| flags.has_flag::<T>())
.unwrap_or(T::enabled_for_all()) .unwrap_or(false)
} }
fn is_staff(&self) -> bool { fn is_staff(&self) -> bool {

View file

@ -4466,7 +4466,7 @@ fn current_language_model(cx: &Context<'_, GitPanel>) -> Option<Arc<dyn Language
is_enabled is_enabled
.then(|| { .then(|| {
let ConfiguredModel { provider, model } = let ConfiguredModel { provider, model } =
LanguageModelRegistry::read_global(cx).commit_message_model()?; LanguageModelRegistry::read_global(cx).commit_message_model(cx)?;
provider.is_authenticated(cx).then(|| model) provider.is_authenticated(cx).then(|| model)
}) })

View file

@ -37,10 +37,10 @@ use crate::{
AssetSource, BackgroundExecutor, Bounds, ClipboardItem, CursorStyle, DispatchPhase, DisplayId, AssetSource, BackgroundExecutor, Bounds, ClipboardItem, CursorStyle, DispatchPhase, DisplayId,
EventEmitter, FocusHandle, FocusMap, ForegroundExecutor, Global, KeyBinding, KeyContext, EventEmitter, FocusHandle, FocusMap, ForegroundExecutor, Global, KeyBinding, KeyContext,
Keymap, Keystroke, LayoutId, Menu, MenuItem, OwnedMenu, PathPromptOptions, Pixels, Platform, Keymap, Keystroke, LayoutId, Menu, MenuItem, OwnedMenu, PathPromptOptions, Pixels, Platform,
PlatformDisplay, PlatformKeyboardLayout, PlatformKeyboardMapper, Point, PromptBuilder, PlatformDisplay, PlatformKeyboardLayout, Point, PromptBuilder, PromptButton, PromptHandle,
PromptButton, PromptHandle, PromptLevel, Render, RenderImage, RenderablePromptHandle, PromptLevel, Render, RenderImage, RenderablePromptHandle, Reservation, ScreenCaptureSource,
Reservation, ScreenCaptureSource, SubscriberSet, Subscription, SvgRenderer, Task, TextSystem, SubscriberSet, Subscription, SvgRenderer, Task, TextSystem, Window, WindowAppearance,
Window, WindowAppearance, WindowHandle, WindowId, WindowInvalidator, WindowHandle, WindowId, WindowInvalidator,
colors::{Colors, GlobalColors}, colors::{Colors, GlobalColors},
current_platform, hash, init_app_menus, current_platform, hash, init_app_menus,
}; };
@ -263,7 +263,6 @@ pub struct App {
pub(crate) focus_handles: Arc<FocusMap>, pub(crate) focus_handles: Arc<FocusMap>,
pub(crate) keymap: Rc<RefCell<Keymap>>, pub(crate) keymap: Rc<RefCell<Keymap>>,
pub(crate) keyboard_layout: Box<dyn PlatformKeyboardLayout>, pub(crate) keyboard_layout: Box<dyn PlatformKeyboardLayout>,
pub(crate) keyboard_mapper: Rc<dyn PlatformKeyboardMapper>,
pub(crate) global_action_listeners: pub(crate) global_action_listeners:
FxHashMap<TypeId, Vec<Rc<dyn Fn(&dyn Any, DispatchPhase, &mut Self)>>>, FxHashMap<TypeId, Vec<Rc<dyn Fn(&dyn Any, DispatchPhase, &mut Self)>>>,
pending_effects: VecDeque<Effect>, pending_effects: VecDeque<Effect>,
@ -313,7 +312,6 @@ impl App {
let text_system = Arc::new(TextSystem::new(platform.text_system())); let text_system = Arc::new(TextSystem::new(platform.text_system()));
let entities = EntityMap::new(); let entities = EntityMap::new();
let keyboard_layout = platform.keyboard_layout(); let keyboard_layout = platform.keyboard_layout();
let keyboard_mapper = platform.keyboard_mapper();
let app = Rc::new_cyclic(|this| AppCell { let app = Rc::new_cyclic(|this| AppCell {
app: RefCell::new(App { app: RefCell::new(App {
@ -339,7 +337,6 @@ impl App {
focus_handles: Arc::new(RwLock::new(SlotMap::with_key())), focus_handles: Arc::new(RwLock::new(SlotMap::with_key())),
keymap: Rc::new(RefCell::new(Keymap::default())), keymap: Rc::new(RefCell::new(Keymap::default())),
keyboard_layout, keyboard_layout,
keyboard_mapper,
global_action_listeners: FxHashMap::default(), global_action_listeners: FxHashMap::default(),
pending_effects: VecDeque::new(), pending_effects: VecDeque::new(),
pending_notifications: FxHashSet::default(), pending_notifications: FxHashSet::default(),
@ -379,7 +376,6 @@ impl App {
if let Some(app) = app.upgrade() { if let Some(app) = app.upgrade() {
let cx = &mut app.borrow_mut(); let cx = &mut app.borrow_mut();
cx.keyboard_layout = cx.platform.keyboard_layout(); cx.keyboard_layout = cx.platform.keyboard_layout();
cx.keyboard_mapper = cx.platform.keyboard_mapper();
cx.keyboard_layout_observers cx.keyboard_layout_observers
.clone() .clone()
.retain(&(), move |callback| (callback)(cx)); .retain(&(), move |callback| (callback)(cx));
@ -428,11 +424,6 @@ impl App {
self.keyboard_layout.as_ref() self.keyboard_layout.as_ref()
} }
/// Get the current keyboard mapper.
pub fn keyboard_mapper(&self) -> &Rc<dyn PlatformKeyboardMapper> {
&self.keyboard_mapper
}
/// Invokes a handler when the current keyboard layout changes /// Invokes a handler when the current keyboard layout changes
pub fn on_keyboard_layout_change<F>(&self, mut callback: F) -> Subscription pub fn on_keyboard_layout_change<F>(&self, mut callback: F) -> Subscription
where where

View file

@ -4,7 +4,7 @@ mod context;
pub use binding::*; pub use binding::*;
pub use context::*; pub use context::*;
use crate::{Action, AsKeystroke, Keystroke, is_no_action}; use crate::{Action, Keystroke, is_no_action};
use collections::{HashMap, HashSet}; use collections::{HashMap, HashSet};
use smallvec::SmallVec; use smallvec::SmallVec;
use std::any::TypeId; use std::any::TypeId;
@ -141,7 +141,7 @@ impl Keymap {
/// only. /// only.
pub fn bindings_for_input( pub fn bindings_for_input(
&self, &self,
input: &[impl AsKeystroke], input: &[Keystroke],
context_stack: &[KeyContext], context_stack: &[KeyContext],
) -> (SmallVec<[KeyBinding; 1]>, bool) { ) -> (SmallVec<[KeyBinding; 1]>, bool) {
let mut matched_bindings = SmallVec::<[(usize, BindingIndex, &KeyBinding); 1]>::new(); let mut matched_bindings = SmallVec::<[(usize, BindingIndex, &KeyBinding); 1]>::new();
@ -192,6 +192,7 @@ impl Keymap {
(bindings, !pending.is_empty()) (bindings, !pending.is_empty())
} }
/// Check if the given binding is enabled, given a certain key context. /// Check if the given binding is enabled, given a certain key context.
/// Returns the deepest depth at which the binding matches, or None if it doesn't match. /// Returns the deepest depth at which the binding matches, or None if it doesn't match.
fn binding_enabled(&self, binding: &KeyBinding, contexts: &[KeyContext]) -> Option<usize> { fn binding_enabled(&self, binding: &KeyBinding, contexts: &[KeyContext]) -> Option<usize> {
@ -638,7 +639,7 @@ mod tests {
fn assert_bindings(keymap: &Keymap, action: &dyn Action, expected: &[&str]) { fn assert_bindings(keymap: &Keymap, action: &dyn Action, expected: &[&str]) {
let actual = keymap let actual = keymap
.bindings_for_action(action) .bindings_for_action(action)
.map(|binding| binding.keystrokes[0].inner.unparse()) .map(|binding| binding.keystrokes[0].unparse())
.collect::<Vec<_>>(); .collect::<Vec<_>>();
assert_eq!(actual, expected, "{:?}", action); assert_eq!(actual, expected, "{:?}", action);
} }

View file

@ -1,15 +1,14 @@
use std::rc::Rc; use std::rc::Rc;
use crate::{ use collections::HashMap;
Action, AsKeystroke, DummyKeyboardMapper, InvalidKeystrokeError, KeyBindingContextPredicate,
KeybindingKeystroke, Keystroke, PlatformKeyboardMapper, SharedString, use crate::{Action, InvalidKeystrokeError, KeyBindingContextPredicate, Keystroke, SharedString};
};
use smallvec::SmallVec; use smallvec::SmallVec;
/// A keybinding and its associated metadata, from the keymap. /// A keybinding and its associated metadata, from the keymap.
pub struct KeyBinding { pub struct KeyBinding {
pub(crate) action: Box<dyn Action>, pub(crate) action: Box<dyn Action>,
pub(crate) keystrokes: SmallVec<[KeybindingKeystroke; 2]>, pub(crate) keystrokes: SmallVec<[Keystroke; 2]>,
pub(crate) context_predicate: Option<Rc<KeyBindingContextPredicate>>, pub(crate) context_predicate: Option<Rc<KeyBindingContextPredicate>>,
pub(crate) meta: Option<KeyBindingMetaIndex>, pub(crate) meta: Option<KeyBindingMetaIndex>,
/// The json input string used when building the keybinding, if any /// The json input string used when building the keybinding, if any
@ -33,15 +32,7 @@ impl KeyBinding {
pub fn new<A: Action>(keystrokes: &str, action: A, context: Option<&str>) -> Self { pub fn new<A: Action>(keystrokes: &str, action: A, context: Option<&str>) -> Self {
let context_predicate = let context_predicate =
context.map(|context| KeyBindingContextPredicate::parse(context).unwrap().into()); context.map(|context| KeyBindingContextPredicate::parse(context).unwrap().into());
Self::load( Self::load(keystrokes, Box::new(action), context_predicate, None, None).unwrap()
keystrokes,
Box::new(action),
context_predicate,
false,
None,
&DummyKeyboardMapper,
)
.unwrap()
} }
/// Load a keybinding from the given raw data. /// Load a keybinding from the given raw data.
@ -49,22 +40,24 @@ impl KeyBinding {
keystrokes: &str, keystrokes: &str,
action: Box<dyn Action>, action: Box<dyn Action>,
context_predicate: Option<Rc<KeyBindingContextPredicate>>, context_predicate: Option<Rc<KeyBindingContextPredicate>>,
use_key_equivalents: bool, key_equivalents: Option<&HashMap<char, char>>,
action_input: Option<SharedString>, action_input: Option<SharedString>,
keyboard_mapper: &dyn PlatformKeyboardMapper,
) -> std::result::Result<Self, InvalidKeystrokeError> { ) -> std::result::Result<Self, InvalidKeystrokeError> {
let keystrokes: SmallVec<[KeybindingKeystroke; 2]> = keystrokes let mut keystrokes: SmallVec<[Keystroke; 2]> = keystrokes
.split_whitespace() .split_whitespace()
.map(|source| { .map(Keystroke::parse)
let keystroke = Keystroke::parse(source)?;
Ok(KeybindingKeystroke::new(
keystroke,
use_key_equivalents,
keyboard_mapper,
))
})
.collect::<std::result::Result<_, _>>()?; .collect::<std::result::Result<_, _>>()?;
if let Some(equivalents) = key_equivalents {
for keystroke in keystrokes.iter_mut() {
if keystroke.key.chars().count() == 1
&& let Some(key) = equivalents.get(&keystroke.key.chars().next().unwrap())
{
keystroke.key = key.to_string();
}
}
}
Ok(Self { Ok(Self {
keystrokes, keystrokes,
action, action,
@ -86,13 +79,13 @@ impl KeyBinding {
} }
/// Check if the given keystrokes match this binding. /// Check if the given keystrokes match this binding.
pub fn match_keystrokes(&self, typed: &[impl AsKeystroke]) -> Option<bool> { pub fn match_keystrokes(&self, typed: &[Keystroke]) -> Option<bool> {
if self.keystrokes.len() < typed.len() { if self.keystrokes.len() < typed.len() {
return None; return None;
} }
for (target, typed) in self.keystrokes.iter().zip(typed.iter()) { for (target, typed) in self.keystrokes.iter().zip(typed.iter()) {
if !typed.as_keystroke().should_match(target) { if !typed.should_match(target) {
return None; return None;
} }
} }
@ -101,7 +94,7 @@ impl KeyBinding {
} }
/// Get the keystrokes associated with this binding /// Get the keystrokes associated with this binding
pub fn keystrokes(&self) -> &[KeybindingKeystroke] { pub fn keystrokes(&self) -> &[Keystroke] {
self.keystrokes.as_slice() self.keystrokes.as_slice()
} }

View file

@ -231,6 +231,7 @@ pub(crate) trait Platform: 'static {
fn on_quit(&self, callback: Box<dyn FnMut()>); fn on_quit(&self, callback: Box<dyn FnMut()>);
fn on_reopen(&self, callback: Box<dyn FnMut()>); fn on_reopen(&self, callback: Box<dyn FnMut()>);
fn on_keyboard_layout_change(&self, callback: Box<dyn FnMut()>);
fn set_menus(&self, menus: Vec<Menu>, keymap: &Keymap); fn set_menus(&self, menus: Vec<Menu>, keymap: &Keymap);
fn get_menus(&self) -> Option<Vec<OwnedMenu>> { fn get_menus(&self) -> Option<Vec<OwnedMenu>> {
@ -250,6 +251,7 @@ pub(crate) trait Platform: 'static {
fn on_app_menu_action(&self, callback: Box<dyn FnMut(&dyn Action)>); fn on_app_menu_action(&self, callback: Box<dyn FnMut(&dyn Action)>);
fn on_will_open_app_menu(&self, callback: Box<dyn FnMut()>); fn on_will_open_app_menu(&self, callback: Box<dyn FnMut()>);
fn on_validate_app_menu_command(&self, callback: Box<dyn FnMut(&dyn Action) -> bool>); fn on_validate_app_menu_command(&self, callback: Box<dyn FnMut(&dyn Action) -> bool>);
fn keyboard_layout(&self) -> Box<dyn PlatformKeyboardLayout>;
fn compositor_name(&self) -> &'static str { fn compositor_name(&self) -> &'static str {
"" ""
@ -270,10 +272,6 @@ pub(crate) trait Platform: 'static {
fn write_credentials(&self, url: &str, username: &str, password: &[u8]) -> Task<Result<()>>; fn write_credentials(&self, url: &str, username: &str, password: &[u8]) -> Task<Result<()>>;
fn read_credentials(&self, url: &str) -> Task<Result<Option<(String, Vec<u8>)>>>; fn read_credentials(&self, url: &str) -> Task<Result<Option<(String, Vec<u8>)>>>;
fn delete_credentials(&self, url: &str) -> Task<Result<()>>; fn delete_credentials(&self, url: &str) -> Task<Result<()>>;
fn keyboard_layout(&self) -> Box<dyn PlatformKeyboardLayout>;
fn keyboard_mapper(&self) -> Rc<dyn PlatformKeyboardMapper>;
fn on_keyboard_layout_change(&self, callback: Box<dyn FnMut()>);
} }
/// A handle to a platform's display, e.g. a monitor or laptop screen. /// A handle to a platform's display, e.g. a monitor or laptop screen.

View file

@ -1,7 +1,3 @@
use collections::HashMap;
use crate::{KeybindingKeystroke, Keystroke};
/// A trait for platform-specific keyboard layouts /// A trait for platform-specific keyboard layouts
pub trait PlatformKeyboardLayout { pub trait PlatformKeyboardLayout {
/// Get the keyboard layout ID, which should be unique to the layout /// Get the keyboard layout ID, which should be unique to the layout
@ -9,33 +5,3 @@ pub trait PlatformKeyboardLayout {
/// Get the keyboard layout display name /// Get the keyboard layout display name
fn name(&self) -> &str; fn name(&self) -> &str;
} }
/// A trait for platform-specific keyboard mappings
pub trait PlatformKeyboardMapper {
/// Map a key equivalent to its platform-specific representation
fn map_key_equivalent(
&self,
keystroke: Keystroke,
use_key_equivalents: bool,
) -> KeybindingKeystroke;
/// Get the key equivalents for the current keyboard layout,
/// only used on macOS
fn get_key_equivalents(&self) -> Option<&HashMap<char, char>>;
}
/// A dummy implementation of the platform keyboard mapper
pub struct DummyKeyboardMapper;
impl PlatformKeyboardMapper for DummyKeyboardMapper {
fn map_key_equivalent(
&self,
keystroke: Keystroke,
_use_key_equivalents: bool,
) -> KeybindingKeystroke {
KeybindingKeystroke::from_keystroke(keystroke)
}
fn get_key_equivalents(&self) -> Option<&HashMap<char, char>> {
None
}
}

View file

@ -5,14 +5,6 @@ use std::{
fmt::{Display, Write}, fmt::{Display, Write},
}; };
use crate::PlatformKeyboardMapper;
/// This is a helper trait so that we can simplify the implementation of some functions
pub trait AsKeystroke {
/// Returns the GPUI representation of the keystroke.
fn as_keystroke(&self) -> &Keystroke;
}
/// A keystroke and associated metadata generated by the platform /// A keystroke and associated metadata generated by the platform
#[derive(Clone, Debug, Eq, PartialEq, Default, Deserialize, Hash)] #[derive(Clone, Debug, Eq, PartialEq, Default, Deserialize, Hash)]
pub struct Keystroke { pub struct Keystroke {
@ -32,17 +24,6 @@ pub struct Keystroke {
pub key_char: Option<String>, pub key_char: Option<String>,
} }
/// Represents a keystroke that can be used in keybindings and displayed to the user.
#[derive(Debug, Clone, Eq, PartialEq, Hash)]
pub struct KeybindingKeystroke {
/// The GPUI representation of the keystroke.
pub inner: Keystroke,
/// The modifiers to display.
pub display_modifiers: Modifiers,
/// The key to display.
pub display_key: String,
}
/// Error type for `Keystroke::parse`. This is used instead of `anyhow::Error` so that Zed can use /// Error type for `Keystroke::parse`. This is used instead of `anyhow::Error` so that Zed can use
/// markdown to display it. /// markdown to display it.
#[derive(Debug)] #[derive(Debug)]
@ -77,7 +58,7 @@ impl Keystroke {
/// ///
/// This method assumes that `self` was typed and `target' is in the keymap, and checks /// This method assumes that `self` was typed and `target' is in the keymap, and checks
/// both possibilities for self against the target. /// both possibilities for self against the target.
pub fn should_match(&self, target: &KeybindingKeystroke) -> bool { pub fn should_match(&self, target: &Keystroke) -> bool {
#[cfg(not(target_os = "windows"))] #[cfg(not(target_os = "windows"))]
if let Some(key_char) = self if let Some(key_char) = self
.key_char .key_char
@ -90,7 +71,7 @@ impl Keystroke {
..Default::default() ..Default::default()
}; };
if &target.inner.key == key_char && target.inner.modifiers == ime_modifiers { if &target.key == key_char && target.modifiers == ime_modifiers {
return true; return true;
} }
} }
@ -102,12 +83,12 @@ impl Keystroke {
.filter(|key_char| key_char != &&self.key) .filter(|key_char| key_char != &&self.key)
{ {
// On Windows, if key_char is set, then the typed keystroke produced the key_char // On Windows, if key_char is set, then the typed keystroke produced the key_char
if &target.inner.key == key_char && target.inner.modifiers == Modifiers::none() { if &target.key == key_char && target.modifiers == Modifiers::none() {
return true; return true;
} }
} }
target.inner.modifiers == self.modifiers && target.inner.key == self.key target.modifiers == self.modifiers && target.key == self.key
} }
/// key syntax is: /// key syntax is:
@ -219,7 +200,31 @@ impl Keystroke {
/// Produces a representation of this key that Parse can understand. /// Produces a representation of this key that Parse can understand.
pub fn unparse(&self) -> String { pub fn unparse(&self) -> String {
unparse(&self.modifiers, &self.key) let mut str = String::new();
if self.modifiers.function {
str.push_str("fn-");
}
if self.modifiers.control {
str.push_str("ctrl-");
}
if self.modifiers.alt {
str.push_str("alt-");
}
if self.modifiers.platform {
#[cfg(target_os = "macos")]
str.push_str("cmd-");
#[cfg(any(target_os = "linux", target_os = "freebsd"))]
str.push_str("super-");
#[cfg(target_os = "windows")]
str.push_str("win-");
}
if self.modifiers.shift {
str.push_str("shift-");
}
str.push_str(&self.key);
str
} }
/// Returns true if this keystroke left /// Returns true if this keystroke left
@ -261,32 +266,6 @@ impl Keystroke {
} }
} }
impl KeybindingKeystroke {
/// Create a new keybinding keystroke from the given keystroke
pub fn new(
inner: Keystroke,
use_key_equivalents: bool,
keyboard_mapper: &dyn PlatformKeyboardMapper,
) -> Self {
keyboard_mapper.map_key_equivalent(inner, use_key_equivalents)
}
pub(crate) fn from_keystroke(keystroke: Keystroke) -> Self {
let key = keystroke.key.clone();
let modifiers = keystroke.modifiers;
KeybindingKeystroke {
inner: keystroke,
display_modifiers: modifiers,
display_key: key,
}
}
/// Produces a representation of this key that Parse can understand.
pub fn unparse(&self) -> String {
unparse(&self.display_modifiers, &self.display_key)
}
}
fn is_printable_key(key: &str) -> bool { fn is_printable_key(key: &str) -> bool {
!matches!( !matches!(
key, key,
@ -343,15 +322,65 @@ fn is_printable_key(key: &str) -> bool {
impl std::fmt::Display for Keystroke { impl std::fmt::Display for Keystroke {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result { fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
display_modifiers(&self.modifiers, f)?; if self.modifiers.control {
display_key(&self.key, f) #[cfg(target_os = "macos")]
} f.write_char('^')?;
}
impl std::fmt::Display for KeybindingKeystroke { #[cfg(not(target_os = "macos"))]
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result { write!(f, "ctrl-")?;
display_modifiers(&self.display_modifiers, f)?; }
display_key(&self.display_key, f) if self.modifiers.alt {
#[cfg(target_os = "macos")]
f.write_char('⌥')?;
#[cfg(not(target_os = "macos"))]
write!(f, "alt-")?;
}
if self.modifiers.platform {
#[cfg(target_os = "macos")]
f.write_char('⌘')?;
#[cfg(any(target_os = "linux", target_os = "freebsd"))]
f.write_char('❖')?;
#[cfg(target_os = "windows")]
f.write_char('⊞')?;
}
if self.modifiers.shift {
#[cfg(target_os = "macos")]
f.write_char('⇧')?;
#[cfg(not(target_os = "macos"))]
write!(f, "shift-")?;
}
let key = match self.key.as_str() {
#[cfg(target_os = "macos")]
"backspace" => '⌫',
#[cfg(target_os = "macos")]
"up" => '↑',
#[cfg(target_os = "macos")]
"down" => '↓',
#[cfg(target_os = "macos")]
"left" => '←',
#[cfg(target_os = "macos")]
"right" => '→',
#[cfg(target_os = "macos")]
"tab" => '⇥',
#[cfg(target_os = "macos")]
"escape" => '⎋',
#[cfg(target_os = "macos")]
"shift" => '⇧',
#[cfg(target_os = "macos")]
"control" => '⌃',
#[cfg(target_os = "macos")]
"alt" => '⌥',
#[cfg(target_os = "macos")]
"platform" => '⌘',
key if key.len() == 1 => key.chars().next().unwrap().to_ascii_uppercase(),
key => return f.write_str(key),
};
f.write_char(key)
} }
} }
@ -571,110 +600,3 @@ pub struct Capslock {
#[serde(default)] #[serde(default)]
pub on: bool, pub on: bool,
} }
impl AsKeystroke for Keystroke {
fn as_keystroke(&self) -> &Keystroke {
self
}
}
impl AsKeystroke for KeybindingKeystroke {
fn as_keystroke(&self) -> &Keystroke {
&self.inner
}
}
fn display_modifiers(modifiers: &Modifiers, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
if modifiers.control {
#[cfg(target_os = "macos")]
f.write_char('^')?;
#[cfg(not(target_os = "macos"))]
write!(f, "ctrl-")?;
}
if modifiers.alt {
#[cfg(target_os = "macos")]
f.write_char('⌥')?;
#[cfg(not(target_os = "macos"))]
write!(f, "alt-")?;
}
if modifiers.platform {
#[cfg(target_os = "macos")]
f.write_char('⌘')?;
#[cfg(any(target_os = "linux", target_os = "freebsd"))]
f.write_char('❖')?;
#[cfg(target_os = "windows")]
f.write_char('⊞')?;
}
if modifiers.shift {
#[cfg(target_os = "macos")]
f.write_char('⇧')?;
#[cfg(not(target_os = "macos"))]
write!(f, "shift-")?;
}
Ok(())
}
fn display_key(key: &str, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
let key = match key {
#[cfg(target_os = "macos")]
"backspace" => '⌫',
#[cfg(target_os = "macos")]
"up" => '↑',
#[cfg(target_os = "macos")]
"down" => '↓',
#[cfg(target_os = "macos")]
"left" => '←',
#[cfg(target_os = "macos")]
"right" => '→',
#[cfg(target_os = "macos")]
"tab" => '⇥',
#[cfg(target_os = "macos")]
"escape" => '⎋',
#[cfg(target_os = "macos")]
"shift" => '⇧',
#[cfg(target_os = "macos")]
"control" => '⌃',
#[cfg(target_os = "macos")]
"alt" => '⌥',
#[cfg(target_os = "macos")]
"platform" => '⌘',
key if key.len() == 1 => key.chars().next().unwrap().to_ascii_uppercase(),
key => return f.write_str(key),
};
f.write_char(key)
}
#[inline]
fn unparse(modifiers: &Modifiers, key: &str) -> String {
let mut result = String::new();
if modifiers.function {
result.push_str("fn-");
}
if modifiers.control {
result.push_str("ctrl-");
}
if modifiers.alt {
result.push_str("alt-");
}
if modifiers.platform {
#[cfg(target_os = "macos")]
result.push_str("cmd-");
#[cfg(any(target_os = "linux", target_os = "freebsd"))]
result.push_str("super-");
#[cfg(target_os = "windows")]
result.push_str("win-");
}
if modifiers.shift {
result.push_str("shift-");
}
result.push_str(&key);
result
}

View file

@ -25,8 +25,8 @@ use xkbcommon::xkb::{self, Keycode, Keysym, State};
use crate::{ use crate::{
Action, AnyWindowHandle, BackgroundExecutor, ClipboardItem, CursorStyle, DisplayId, Action, AnyWindowHandle, BackgroundExecutor, ClipboardItem, CursorStyle, DisplayId,
ForegroundExecutor, Keymap, LinuxDispatcher, Menu, MenuItem, OwnedMenu, PathPromptOptions, ForegroundExecutor, Keymap, LinuxDispatcher, Menu, MenuItem, OwnedMenu, PathPromptOptions,
Pixels, Platform, PlatformDisplay, PlatformKeyboardLayout, PlatformKeyboardMapper, Pixels, Platform, PlatformDisplay, PlatformKeyboardLayout, PlatformTextSystem, PlatformWindow,
PlatformTextSystem, PlatformWindow, Point, Result, Task, WindowAppearance, WindowParams, px, Point, Result, Task, WindowAppearance, WindowParams, px,
}; };
#[cfg(any(feature = "wayland", feature = "x11"))] #[cfg(any(feature = "wayland", feature = "x11"))]
@ -144,10 +144,6 @@ impl<P: LinuxClient + 'static> Platform for P {
self.keyboard_layout() self.keyboard_layout()
} }
fn keyboard_mapper(&self) -> Rc<dyn PlatformKeyboardMapper> {
Rc::new(crate::DummyKeyboardMapper)
}
fn on_keyboard_layout_change(&self, callback: Box<dyn FnMut()>) { fn on_keyboard_layout_change(&self, callback: Box<dyn FnMut()>) {
self.with_common(|common| common.callbacks.keyboard_layout_change = Some(callback)); self.with_common(|common| common.callbacks.keyboard_layout_change = Some(callback));
} }

File diff suppressed because it is too large Load diff

View file

@ -1,5 +1,5 @@
use super::{ use super::{
BoolExt, MacKeyboardLayout, MacKeyboardMapper, BoolExt, MacKeyboardLayout,
attributed_string::{NSAttributedString, NSMutableAttributedString}, attributed_string::{NSAttributedString, NSMutableAttributedString},
events::key_to_native, events::key_to_native,
renderer, renderer,
@ -8,9 +8,8 @@ use crate::{
Action, AnyWindowHandle, BackgroundExecutor, ClipboardEntry, ClipboardItem, ClipboardString, Action, AnyWindowHandle, BackgroundExecutor, ClipboardEntry, ClipboardItem, ClipboardString,
CursorStyle, ForegroundExecutor, Image, ImageFormat, KeyContext, Keymap, MacDispatcher, CursorStyle, ForegroundExecutor, Image, ImageFormat, KeyContext, Keymap, MacDispatcher,
MacDisplay, MacWindow, Menu, MenuItem, OsMenu, OwnedMenu, PathPromptOptions, Platform, MacDisplay, MacWindow, Menu, MenuItem, OsMenu, OwnedMenu, PathPromptOptions, Platform,
PlatformDisplay, PlatformKeyboardLayout, PlatformKeyboardMapper, PlatformTextSystem, PlatformDisplay, PlatformKeyboardLayout, PlatformTextSystem, PlatformWindow, Result,
PlatformWindow, Result, SemanticVersion, SystemMenuType, Task, WindowAppearance, WindowParams, SemanticVersion, SystemMenuType, Task, WindowAppearance, WindowParams, hash,
hash,
}; };
use anyhow::{Context as _, anyhow}; use anyhow::{Context as _, anyhow};
use block::ConcreteBlock; use block::ConcreteBlock;
@ -172,7 +171,6 @@ pub(crate) struct MacPlatformState {
finish_launching: Option<Box<dyn FnOnce()>>, finish_launching: Option<Box<dyn FnOnce()>>,
dock_menu: Option<id>, dock_menu: Option<id>,
menus: Option<Vec<OwnedMenu>>, menus: Option<Vec<OwnedMenu>>,
keyboard_mapper: Rc<MacKeyboardMapper>,
} }
impl Default for MacPlatform { impl Default for MacPlatform {
@ -191,9 +189,6 @@ impl MacPlatform {
#[cfg(not(feature = "font-kit"))] #[cfg(not(feature = "font-kit"))]
let text_system = Arc::new(crate::NoopTextSystem::new()); let text_system = Arc::new(crate::NoopTextSystem::new());
let keyboard_layout = MacKeyboardLayout::new();
let keyboard_mapper = Rc::new(MacKeyboardMapper::new(keyboard_layout.id()));
Self(Mutex::new(MacPlatformState { Self(Mutex::new(MacPlatformState {
headless, headless,
text_system, text_system,
@ -214,7 +209,6 @@ impl MacPlatform {
dock_menu: None, dock_menu: None,
on_keyboard_layout_change: None, on_keyboard_layout_change: None,
menus: None, menus: None,
keyboard_mapper,
})) }))
} }
@ -354,19 +348,19 @@ impl MacPlatform {
let mut mask = NSEventModifierFlags::empty(); let mut mask = NSEventModifierFlags::empty();
for (modifier, flag) in &[ for (modifier, flag) in &[
( (
keystroke.display_modifiers.platform, keystroke.modifiers.platform,
NSEventModifierFlags::NSCommandKeyMask, NSEventModifierFlags::NSCommandKeyMask,
), ),
( (
keystroke.display_modifiers.control, keystroke.modifiers.control,
NSEventModifierFlags::NSControlKeyMask, NSEventModifierFlags::NSControlKeyMask,
), ),
( (
keystroke.display_modifiers.alt, keystroke.modifiers.alt,
NSEventModifierFlags::NSAlternateKeyMask, NSEventModifierFlags::NSAlternateKeyMask,
), ),
( (
keystroke.display_modifiers.shift, keystroke.modifiers.shift,
NSEventModifierFlags::NSShiftKeyMask, NSEventModifierFlags::NSShiftKeyMask,
), ),
] { ] {
@ -379,7 +373,7 @@ impl MacPlatform {
.initWithTitle_action_keyEquivalent_( .initWithTitle_action_keyEquivalent_(
ns_string(name), ns_string(name),
selector, selector,
ns_string(key_to_native(&keystroke.display_key).as_ref()), ns_string(key_to_native(&keystroke.key).as_ref()),
) )
.autorelease(); .autorelease();
if Self::os_version() >= SemanticVersion::new(12, 0, 0) { if Self::os_version() >= SemanticVersion::new(12, 0, 0) {
@ -888,10 +882,6 @@ impl Platform for MacPlatform {
Box::new(MacKeyboardLayout::new()) Box::new(MacKeyboardLayout::new())
} }
fn keyboard_mapper(&self) -> Rc<dyn PlatformKeyboardMapper> {
self.0.lock().keyboard_mapper.clone()
}
fn app_path(&self) -> Result<PathBuf> { fn app_path(&self) -> Result<PathBuf> {
unsafe { unsafe {
let bundle: id = NSBundle::mainBundle(); let bundle: id = NSBundle::mainBundle();
@ -1403,8 +1393,6 @@ extern "C" fn will_terminate(this: &mut Object, _: Sel, _: id) {
extern "C" fn on_keyboard_layout_change(this: &mut Object, _: Sel, _: id) { extern "C" fn on_keyboard_layout_change(this: &mut Object, _: Sel, _: id) {
let platform = unsafe { get_mac_platform(this) }; let platform = unsafe { get_mac_platform(this) };
let mut lock = platform.0.lock(); let mut lock = platform.0.lock();
let keyboard_layout = MacKeyboardLayout::new();
lock.keyboard_mapper = Rc::new(MacKeyboardMapper::new(keyboard_layout.id()));
if let Some(mut callback) = lock.on_keyboard_layout_change.take() { if let Some(mut callback) = lock.on_keyboard_layout_change.take() {
drop(lock); drop(lock);
callback(); callback();

View file

@ -1,9 +1,8 @@
use crate::{ use crate::{
AnyWindowHandle, BackgroundExecutor, ClipboardItem, CursorStyle, DevicePixels, AnyWindowHandle, BackgroundExecutor, ClipboardItem, CursorStyle, DevicePixels,
DummyKeyboardMapper, ForegroundExecutor, Keymap, NoopTextSystem, Platform, PlatformDisplay, ForegroundExecutor, Keymap, NoopTextSystem, Platform, PlatformDisplay, PlatformKeyboardLayout,
PlatformKeyboardLayout, PlatformKeyboardMapper, PlatformTextSystem, PromptButton, PlatformTextSystem, PromptButton, ScreenCaptureFrame, ScreenCaptureSource, ScreenCaptureStream,
ScreenCaptureFrame, ScreenCaptureSource, ScreenCaptureStream, SourceMetadata, Task, SourceMetadata, Task, TestDisplay, TestWindow, WindowAppearance, WindowParams, size,
TestDisplay, TestWindow, WindowAppearance, WindowParams, size,
}; };
use anyhow::Result; use anyhow::Result;
use collections::VecDeque; use collections::VecDeque;
@ -238,10 +237,6 @@ impl Platform for TestPlatform {
Box::new(TestKeyboardLayout) Box::new(TestKeyboardLayout)
} }
fn keyboard_mapper(&self) -> Rc<dyn PlatformKeyboardMapper> {
Rc::new(DummyKeyboardMapper)
}
fn on_keyboard_layout_change(&self, _: Box<dyn FnMut()>) {} fn on_keyboard_layout_change(&self, _: Box<dyn FnMut()>) {}
fn run(&self, _on_finish_launching: Box<dyn FnOnce()>) { fn run(&self, _on_finish_launching: Box<dyn FnOnce()>) {

View file

@ -1,31 +1,22 @@
use anyhow::Result; use anyhow::Result;
use collections::HashMap;
use windows::Win32::UI::{ use windows::Win32::UI::{
Input::KeyboardAndMouse::{ Input::KeyboardAndMouse::{
GetKeyboardLayoutNameW, MAPVK_VK_TO_CHAR, MAPVK_VK_TO_VSC, MapVirtualKeyW, ToUnicode, GetKeyboardLayoutNameW, MAPVK_VK_TO_CHAR, MapVirtualKeyW, ToUnicode, VIRTUAL_KEY, VK_0,
VIRTUAL_KEY, VK_0, VK_1, VK_2, VK_3, VK_4, VK_5, VK_6, VK_7, VK_8, VK_9, VK_ABNT_C1, VK_1, VK_2, VK_3, VK_4, VK_5, VK_6, VK_7, VK_8, VK_9, VK_ABNT_C1, VK_CONTROL, VK_MENU,
VK_CONTROL, VK_MENU, VK_OEM_1, VK_OEM_2, VK_OEM_3, VK_OEM_4, VK_OEM_5, VK_OEM_6, VK_OEM_7, VK_OEM_1, VK_OEM_2, VK_OEM_3, VK_OEM_4, VK_OEM_5, VK_OEM_6, VK_OEM_7, VK_OEM_8, VK_OEM_102,
VK_OEM_8, VK_OEM_102, VK_OEM_COMMA, VK_OEM_MINUS, VK_OEM_PERIOD, VK_OEM_PLUS, VK_SHIFT, VK_OEM_COMMA, VK_OEM_MINUS, VK_OEM_PERIOD, VK_OEM_PLUS, VK_SHIFT,
}, },
WindowsAndMessaging::KL_NAMELENGTH, WindowsAndMessaging::KL_NAMELENGTH,
}; };
use windows_core::HSTRING; use windows_core::HSTRING;
use crate::{ use crate::{Modifiers, PlatformKeyboardLayout};
KeybindingKeystroke, Keystroke, Modifiers, PlatformKeyboardLayout, PlatformKeyboardMapper,
};
pub(crate) struct WindowsKeyboardLayout { pub(crate) struct WindowsKeyboardLayout {
id: String, id: String,
name: String, name: String,
} }
pub(crate) struct WindowsKeyboardMapper {
key_to_vkey: HashMap<String, (u16, bool)>,
vkey_to_key: HashMap<u16, String>,
vkey_to_shifted: HashMap<u16, String>,
}
impl PlatformKeyboardLayout for WindowsKeyboardLayout { impl PlatformKeyboardLayout for WindowsKeyboardLayout {
fn id(&self) -> &str { fn id(&self) -> &str {
&self.id &self.id
@ -36,65 +27,6 @@ impl PlatformKeyboardLayout for WindowsKeyboardLayout {
} }
} }
impl PlatformKeyboardMapper for WindowsKeyboardMapper {
fn map_key_equivalent(
&self,
mut keystroke: Keystroke,
use_key_equivalents: bool,
) -> KeybindingKeystroke {
let Some((vkey, shifted_key)) = self.get_vkey_from_key(&keystroke.key, use_key_equivalents)
else {
return KeybindingKeystroke::from_keystroke(keystroke);
};
if shifted_key && keystroke.modifiers.shift {
log::warn!(
"Keystroke '{}' has both shift and a shifted key, this is likely a bug",
keystroke.key
);
}
let shift = shifted_key || keystroke.modifiers.shift;
keystroke.modifiers.shift = false;
let Some(key) = self.vkey_to_key.get(&vkey).cloned() else {
log::error!(
"Failed to map key equivalent '{:?}' to a valid key",
keystroke
);
return KeybindingKeystroke::from_keystroke(keystroke);
};
keystroke.key = if shift {
let Some(shifted_key) = self.vkey_to_shifted.get(&vkey).cloned() else {
log::error!(
"Failed to map keystroke {:?} with virtual key '{:?}' to a shifted key",
keystroke,
vkey
);
return KeybindingKeystroke::from_keystroke(keystroke);
};
shifted_key
} else {
key.clone()
};
let modifiers = Modifiers {
shift,
..keystroke.modifiers
};
KeybindingKeystroke {
inner: keystroke,
display_modifiers: modifiers,
display_key: key,
}
}
fn get_key_equivalents(&self) -> Option<&HashMap<char, char>> {
None
}
}
impl WindowsKeyboardLayout { impl WindowsKeyboardLayout {
pub(crate) fn new() -> Result<Self> { pub(crate) fn new() -> Result<Self> {
let mut buffer = [0u16; KL_NAMELENGTH as usize]; let mut buffer = [0u16; KL_NAMELENGTH as usize];
@ -116,41 +48,6 @@ impl WindowsKeyboardLayout {
} }
} }
impl WindowsKeyboardMapper {
pub(crate) fn new() -> Self {
let mut key_to_vkey = HashMap::default();
let mut vkey_to_key = HashMap::default();
let mut vkey_to_shifted = HashMap::default();
for vkey in CANDIDATE_VKEYS {
if let Some(key) = get_key_from_vkey(*vkey) {
key_to_vkey.insert(key.clone(), (vkey.0, false));
vkey_to_key.insert(vkey.0, key);
}
let scan_code = unsafe { MapVirtualKeyW(vkey.0 as u32, MAPVK_VK_TO_VSC) };
if scan_code == 0 {
continue;
}
if let Some(shifted_key) = get_shifted_key(*vkey, scan_code) {
key_to_vkey.insert(shifted_key.clone(), (vkey.0, true));
vkey_to_shifted.insert(vkey.0, shifted_key);
}
}
Self {
key_to_vkey,
vkey_to_key,
vkey_to_shifted,
}
}
fn get_vkey_from_key(&self, key: &str, use_key_equivalents: bool) -> Option<(u16, bool)> {
if use_key_equivalents {
get_vkey_from_key_with_us_layout(key)
} else {
self.key_to_vkey.get(key).cloned()
}
}
}
pub(crate) fn get_keystroke_key( pub(crate) fn get_keystroke_key(
vkey: VIRTUAL_KEY, vkey: VIRTUAL_KEY,
scan_code: u32, scan_code: u32,
@ -243,134 +140,3 @@ pub(crate) fn generate_key_char(
_ => None, _ => None,
} }
} }
fn get_vkey_from_key_with_us_layout(key: &str) -> Option<(u16, bool)> {
match key {
// ` => VK_OEM_3
"`" => Some((VK_OEM_3.0, false)),
"~" => Some((VK_OEM_3.0, true)),
"1" => Some((VK_1.0, false)),
"!" => Some((VK_1.0, true)),
"2" => Some((VK_2.0, false)),
"@" => Some((VK_2.0, true)),
"3" => Some((VK_3.0, false)),
"#" => Some((VK_3.0, true)),
"4" => Some((VK_4.0, false)),
"$" => Some((VK_4.0, true)),
"5" => Some((VK_5.0, false)),
"%" => Some((VK_5.0, true)),
"6" => Some((VK_6.0, false)),
"^" => Some((VK_6.0, true)),
"7" => Some((VK_7.0, false)),
"&" => Some((VK_7.0, true)),
"8" => Some((VK_8.0, false)),
"*" => Some((VK_8.0, true)),
"9" => Some((VK_9.0, false)),
"(" => Some((VK_9.0, true)),
"0" => Some((VK_0.0, false)),
")" => Some((VK_0.0, true)),
"-" => Some((VK_OEM_MINUS.0, false)),
"_" => Some((VK_OEM_MINUS.0, true)),
"=" => Some((VK_OEM_PLUS.0, false)),
"+" => Some((VK_OEM_PLUS.0, true)),
"[" => Some((VK_OEM_4.0, false)),
"{" => Some((VK_OEM_4.0, true)),
"]" => Some((VK_OEM_6.0, false)),
"}" => Some((VK_OEM_6.0, true)),
"\\" => Some((VK_OEM_5.0, false)),
"|" => Some((VK_OEM_5.0, true)),
";" => Some((VK_OEM_1.0, false)),
":" => Some((VK_OEM_1.0, true)),
"'" => Some((VK_OEM_7.0, false)),
"\"" => Some((VK_OEM_7.0, true)),
"," => Some((VK_OEM_COMMA.0, false)),
"<" => Some((VK_OEM_COMMA.0, true)),
"." => Some((VK_OEM_PERIOD.0, false)),
">" => Some((VK_OEM_PERIOD.0, true)),
"/" => Some((VK_OEM_2.0, false)),
"?" => Some((VK_OEM_2.0, true)),
_ => None,
}
}
const CANDIDATE_VKEYS: &[VIRTUAL_KEY] = &[
VK_OEM_3,
VK_OEM_MINUS,
VK_OEM_PLUS,
VK_OEM_4,
VK_OEM_5,
VK_OEM_6,
VK_OEM_1,
VK_OEM_7,
VK_OEM_COMMA,
VK_OEM_PERIOD,
VK_OEM_2,
VK_OEM_102,
VK_OEM_8,
VK_ABNT_C1,
VK_0,
VK_1,
VK_2,
VK_3,
VK_4,
VK_5,
VK_6,
VK_7,
VK_8,
VK_9,
];
#[cfg(test)]
mod tests {
use crate::{Keystroke, Modifiers, PlatformKeyboardMapper, WindowsKeyboardMapper};
#[test]
fn test_keyboard_mapper() {
let mapper = WindowsKeyboardMapper::new();
// Normal case
let keystroke = Keystroke {
modifiers: Modifiers::control(),
key: "a".to_string(),
key_char: None,
};
let mapped = mapper.map_key_equivalent(keystroke.clone(), true);
assert_eq!(mapped.inner, keystroke);
assert_eq!(mapped.display_key, "a");
assert_eq!(mapped.display_modifiers, Modifiers::control());
// Shifted case, ctrl-$
let keystroke = Keystroke {
modifiers: Modifiers::control(),
key: "$".to_string(),
key_char: None,
};
let mapped = mapper.map_key_equivalent(keystroke.clone(), true);
assert_eq!(mapped.inner, keystroke);
assert_eq!(mapped.display_key, "4");
assert_eq!(mapped.display_modifiers, Modifiers::control_shift());
// Shifted case, but shift is true
let keystroke = Keystroke {
modifiers: Modifiers::control_shift(),
key: "$".to_string(),
key_char: None,
};
let mapped = mapper.map_key_equivalent(keystroke, true);
assert_eq!(mapped.inner.modifiers, Modifiers::control());
assert_eq!(mapped.display_key, "4");
assert_eq!(mapped.display_modifiers, Modifiers::control_shift());
// Windows style
let keystroke = Keystroke {
modifiers: Modifiers::control_shift(),
key: "4".to_string(),
key_char: None,
};
let mapped = mapper.map_key_equivalent(keystroke, true);
assert_eq!(mapped.inner.modifiers, Modifiers::control());
assert_eq!(mapped.inner.key, "$");
assert_eq!(mapped.display_key, "4");
assert_eq!(mapped.display_modifiers, Modifiers::control_shift());
}
}

View file

@ -351,10 +351,6 @@ impl Platform for WindowsPlatform {
) )
} }
fn keyboard_mapper(&self) -> Rc<dyn PlatformKeyboardMapper> {
Rc::new(WindowsKeyboardMapper::new())
}
fn on_keyboard_layout_change(&self, callback: Box<dyn FnMut()>) { fn on_keyboard_layout_change(&self, callback: Box<dyn FnMut()>) {
self.state.borrow_mut().callbacks.keyboard_layout_change = Some(callback); self.state.borrow_mut().callbacks.keyboard_layout_change = Some(callback);
} }

View file

@ -215,7 +215,6 @@ pub enum IconName {
Tab, Tab,
Terminal, Terminal,
TerminalAlt, TerminalAlt,
TerminalGhost,
TextSnippet, TextSnippet,
TextThread, TextThread,
Thread, Thread,

View file

@ -401,19 +401,12 @@ pub fn init(cx: &mut App) {
mod persistence { mod persistence {
use std::path::PathBuf; use std::path::PathBuf;
use db::{ use db::{define_connection, query, sqlez_macros::sql};
query,
sqlez::{domain::Domain, thread_safe_connection::ThreadSafeConnection},
sqlez_macros::sql,
};
use workspace::{ItemId, WorkspaceDb, WorkspaceId}; use workspace::{ItemId, WorkspaceDb, WorkspaceId};
pub struct ImageViewerDb(ThreadSafeConnection); define_connection! {
pub static ref IMAGE_VIEWER: ImageViewerDb<WorkspaceDb> =
impl Domain for ImageViewerDb { &[sql!(
const NAME: &str = stringify!(ImageViewerDb);
const MIGRATIONS: &[&str] = &[sql!(
CREATE TABLE image_viewers ( CREATE TABLE image_viewers (
workspace_id INTEGER, workspace_id INTEGER,
item_id INTEGER UNIQUE, item_id INTEGER UNIQUE,
@ -424,11 +417,9 @@ mod persistence {
FOREIGN KEY(workspace_id) REFERENCES workspaces(workspace_id) FOREIGN KEY(workspace_id) REFERENCES workspaces(workspace_id)
ON DELETE CASCADE ON DELETE CASCADE
) STRICT; ) STRICT;
)]; )];
} }
db::static_connection!(IMAGE_VIEWER, ImageViewerDb, [WorkspaceDb]);
impl ImageViewerDb { impl ImageViewerDb {
query! { query! {
pub async fn save_image_path( pub async fn save_image_path(

View file

@ -4,16 +4,12 @@ use crate::{
LanguageModelProviderId, LanguageModelProviderName, LanguageModelProviderState, LanguageModelProviderId, LanguageModelProviderName, LanguageModelProviderState,
LanguageModelRequest, LanguageModelToolChoice, LanguageModelRequest, LanguageModelToolChoice,
}; };
use anyhow::anyhow;
use futures::{FutureExt, channel::mpsc, future::BoxFuture, stream::BoxStream}; use futures::{FutureExt, channel::mpsc, future::BoxFuture, stream::BoxStream};
use gpui::{AnyView, App, AsyncApp, Entity, Task, Window}; use gpui::{AnyView, App, AsyncApp, Entity, Task, Window};
use http_client::Result; use http_client::Result;
use parking_lot::Mutex; use parking_lot::Mutex;
use smol::stream::StreamExt; use smol::stream::StreamExt;
use std::sync::{ use std::sync::Arc;
Arc,
atomic::{AtomicBool, Ordering::SeqCst},
};
#[derive(Clone)] #[derive(Clone)]
pub struct FakeLanguageModelProvider { pub struct FakeLanguageModelProvider {
@ -110,7 +106,6 @@ pub struct FakeLanguageModel {
>, >,
)>, )>,
>, >,
forbid_requests: AtomicBool,
} }
impl Default for FakeLanguageModel { impl Default for FakeLanguageModel {
@ -119,20 +114,11 @@ impl Default for FakeLanguageModel {
provider_id: LanguageModelProviderId::from("fake".to_string()), provider_id: LanguageModelProviderId::from("fake".to_string()),
provider_name: LanguageModelProviderName::from("Fake".to_string()), provider_name: LanguageModelProviderName::from("Fake".to_string()),
current_completion_txs: Mutex::new(Vec::new()), current_completion_txs: Mutex::new(Vec::new()),
forbid_requests: AtomicBool::new(false),
} }
} }
} }
impl FakeLanguageModel { impl FakeLanguageModel {
pub fn allow_requests(&self) {
self.forbid_requests.store(false, SeqCst);
}
pub fn forbid_requests(&self) {
self.forbid_requests.store(true, SeqCst);
}
pub fn pending_completions(&self) -> Vec<LanguageModelRequest> { pub fn pending_completions(&self) -> Vec<LanguageModelRequest> {
self.current_completion_txs self.current_completion_txs
.lock() .lock()
@ -265,18 +251,9 @@ impl LanguageModel for FakeLanguageModel {
LanguageModelCompletionError, LanguageModelCompletionError,
>, >,
> { > {
if self.forbid_requests.load(SeqCst) { let (tx, rx) = mpsc::unbounded();
async move { self.current_completion_txs.lock().push((request, tx));
Err(LanguageModelCompletionError::Other(anyhow!( async move { Ok(rx.boxed()) }.boxed()
"requests are forbidden"
)))
}
.boxed()
} else {
let (tx, rx) = mpsc::unbounded();
self.current_completion_txs.lock().push((request, tx));
async move { Ok(rx.boxed()) }.boxed()
}
} }
fn as_fake(&self) -> &Self { fn as_fake(&self) -> &Self {

View file

@ -6,7 +6,6 @@ use collections::BTreeMap;
use gpui::{App, Context, Entity, EventEmitter, Global, prelude::*}; use gpui::{App, Context, Entity, EventEmitter, Global, prelude::*};
use std::{str::FromStr, sync::Arc}; use std::{str::FromStr, sync::Arc};
use thiserror::Error; use thiserror::Error;
use util::maybe;
pub fn init(cx: &mut App) { pub fn init(cx: &mut App) {
let registry = cx.new(|_cx| LanguageModelRegistry::default()); let registry = cx.new(|_cx| LanguageModelRegistry::default());
@ -42,7 +41,9 @@ impl std::fmt::Debug for ConfigurationError {
#[derive(Default)] #[derive(Default)]
pub struct LanguageModelRegistry { pub struct LanguageModelRegistry {
default_model: Option<ConfiguredModel>, default_model: Option<ConfiguredModel>,
default_fast_model: Option<ConfiguredModel>, /// This model is automatically configured by a user's environment after
/// authenticating all providers. It's only used when default_model is not available.
environment_fallback_model: Option<ConfiguredModel>,
inline_assistant_model: Option<ConfiguredModel>, inline_assistant_model: Option<ConfiguredModel>,
commit_message_model: Option<ConfiguredModel>, commit_message_model: Option<ConfiguredModel>,
thread_summary_model: Option<ConfiguredModel>, thread_summary_model: Option<ConfiguredModel>,
@ -98,9 +99,6 @@ impl ConfiguredModel {
pub enum Event { pub enum Event {
DefaultModelChanged, DefaultModelChanged,
InlineAssistantModelChanged,
CommitMessageModelChanged,
ThreadSummaryModelChanged,
ProviderStateChanged(LanguageModelProviderId), ProviderStateChanged(LanguageModelProviderId),
AddedProvider(LanguageModelProviderId), AddedProvider(LanguageModelProviderId),
RemovedProvider(LanguageModelProviderId), RemovedProvider(LanguageModelProviderId),
@ -226,7 +224,7 @@ impl LanguageModelRegistry {
cx: &mut Context<Self>, cx: &mut Context<Self>,
) { ) {
let configured_model = model.and_then(|model| self.select_model(model, cx)); let configured_model = model.and_then(|model| self.select_model(model, cx));
self.set_inline_assistant_model(configured_model, cx); self.set_inline_assistant_model(configured_model);
} }
pub fn select_commit_message_model( pub fn select_commit_message_model(
@ -235,7 +233,7 @@ impl LanguageModelRegistry {
cx: &mut Context<Self>, cx: &mut Context<Self>,
) { ) {
let configured_model = model.and_then(|model| self.select_model(model, cx)); let configured_model = model.and_then(|model| self.select_model(model, cx));
self.set_commit_message_model(configured_model, cx); self.set_commit_message_model(configured_model);
} }
pub fn select_thread_summary_model( pub fn select_thread_summary_model(
@ -244,7 +242,7 @@ impl LanguageModelRegistry {
cx: &mut Context<Self>, cx: &mut Context<Self>,
) { ) {
let configured_model = model.and_then(|model| self.select_model(model, cx)); let configured_model = model.and_then(|model| self.select_model(model, cx));
self.set_thread_summary_model(configured_model, cx); self.set_thread_summary_model(configured_model);
} }
/// Selects and sets the inline alternatives for language models based on /// Selects and sets the inline alternatives for language models based on
@ -278,68 +276,60 @@ impl LanguageModelRegistry {
} }
pub fn set_default_model(&mut self, model: Option<ConfiguredModel>, cx: &mut Context<Self>) { pub fn set_default_model(&mut self, model: Option<ConfiguredModel>, cx: &mut Context<Self>) {
match (self.default_model.as_ref(), model.as_ref()) { match (self.default_model(), model.as_ref()) {
(Some(old), Some(new)) if old.is_same_as(new) => {} (Some(old), Some(new)) if old.is_same_as(new) => {}
(None, None) => {} (None, None) => {}
_ => cx.emit(Event::DefaultModelChanged), _ => cx.emit(Event::DefaultModelChanged),
} }
self.default_fast_model = maybe!({
let provider = &model.as_ref()?.provider;
let fast_model = provider.default_fast_model(cx)?;
Some(ConfiguredModel {
provider: provider.clone(),
model: fast_model,
})
});
self.default_model = model; self.default_model = model;
} }
pub fn set_inline_assistant_model( pub fn set_environment_fallback_model(
&mut self, &mut self,
model: Option<ConfiguredModel>, model: Option<ConfiguredModel>,
cx: &mut Context<Self>, cx: &mut Context<Self>,
) { ) {
match (self.inline_assistant_model.as_ref(), model.as_ref()) { if self.default_model.is_none() {
(Some(old), Some(new)) if old.is_same_as(new) => {} match (self.environment_fallback_model.as_ref(), model.as_ref()) {
(None, None) => {} (Some(old), Some(new)) if old.is_same_as(new) => {}
_ => cx.emit(Event::InlineAssistantModelChanged), (None, None) => {}
_ => cx.emit(Event::DefaultModelChanged),
}
} }
self.environment_fallback_model = model;
}
pub fn set_inline_assistant_model(&mut self, model: Option<ConfiguredModel>) {
self.inline_assistant_model = model; self.inline_assistant_model = model;
} }
pub fn set_commit_message_model( pub fn set_commit_message_model(&mut self, model: Option<ConfiguredModel>) {
&mut self,
model: Option<ConfiguredModel>,
cx: &mut Context<Self>,
) {
match (self.commit_message_model.as_ref(), model.as_ref()) {
(Some(old), Some(new)) if old.is_same_as(new) => {}
(None, None) => {}
_ => cx.emit(Event::CommitMessageModelChanged),
}
self.commit_message_model = model; self.commit_message_model = model;
} }
pub fn set_thread_summary_model( pub fn set_thread_summary_model(&mut self, model: Option<ConfiguredModel>) {
&mut self,
model: Option<ConfiguredModel>,
cx: &mut Context<Self>,
) {
match (self.thread_summary_model.as_ref(), model.as_ref()) {
(Some(old), Some(new)) if old.is_same_as(new) => {}
(None, None) => {}
_ => cx.emit(Event::ThreadSummaryModelChanged),
}
self.thread_summary_model = model; self.thread_summary_model = model;
} }
#[track_caller]
pub fn default_model(&self) -> Option<ConfiguredModel> { pub fn default_model(&self) -> Option<ConfiguredModel> {
#[cfg(debug_assertions)] #[cfg(debug_assertions)]
if std::env::var("ZED_SIMULATE_NO_LLM_PROVIDER").is_ok() { if std::env::var("ZED_SIMULATE_NO_LLM_PROVIDER").is_ok() {
return None; return None;
} }
self.default_model.clone() self.default_model
.clone()
.or_else(|| self.environment_fallback_model.clone())
}
pub fn default_fast_model(&self, cx: &App) -> Option<ConfiguredModel> {
let provider = self.default_model()?.provider;
let fast_model = provider.default_fast_model(cx)?;
Some(ConfiguredModel {
provider,
model: fast_model,
})
} }
pub fn inline_assistant_model(&self) -> Option<ConfiguredModel> { pub fn inline_assistant_model(&self) -> Option<ConfiguredModel> {
@ -353,7 +343,7 @@ impl LanguageModelRegistry {
.or_else(|| self.default_model.clone()) .or_else(|| self.default_model.clone())
} }
pub fn commit_message_model(&self) -> Option<ConfiguredModel> { pub fn commit_message_model(&self, cx: &App) -> Option<ConfiguredModel> {
#[cfg(debug_assertions)] #[cfg(debug_assertions)]
if std::env::var("ZED_SIMULATE_NO_LLM_PROVIDER").is_ok() { if std::env::var("ZED_SIMULATE_NO_LLM_PROVIDER").is_ok() {
return None; return None;
@ -361,11 +351,11 @@ impl LanguageModelRegistry {
self.commit_message_model self.commit_message_model
.clone() .clone()
.or_else(|| self.default_fast_model.clone()) .or_else(|| self.default_fast_model(cx))
.or_else(|| self.default_model.clone()) .or_else(|| self.default_model.clone())
} }
pub fn thread_summary_model(&self) -> Option<ConfiguredModel> { pub fn thread_summary_model(&self, cx: &App) -> Option<ConfiguredModel> {
#[cfg(debug_assertions)] #[cfg(debug_assertions)]
if std::env::var("ZED_SIMULATE_NO_LLM_PROVIDER").is_ok() { if std::env::var("ZED_SIMULATE_NO_LLM_PROVIDER").is_ok() {
return None; return None;
@ -373,7 +363,7 @@ impl LanguageModelRegistry {
self.thread_summary_model self.thread_summary_model
.clone() .clone()
.or_else(|| self.default_fast_model.clone()) .or_else(|| self.default_fast_model(cx))
.or_else(|| self.default_model.clone()) .or_else(|| self.default_model.clone())
} }
@ -410,4 +400,34 @@ mod tests {
let providers = registry.read(cx).providers(); let providers = registry.read(cx).providers();
assert!(providers.is_empty()); assert!(providers.is_empty());
} }
#[gpui::test]
async fn test_configure_environment_fallback_model(cx: &mut gpui::TestAppContext) {
let registry = cx.new(|_| LanguageModelRegistry::default());
let provider = FakeLanguageModelProvider::default();
registry.update(cx, |registry, cx| {
registry.register_provider(provider.clone(), cx);
});
cx.update(|cx| provider.authenticate(cx)).await.unwrap();
registry.update(cx, |registry, cx| {
let provider = registry.provider(&provider.id()).unwrap();
registry.set_environment_fallback_model(
Some(ConfiguredModel {
provider: provider.clone(),
model: provider.default_model(cx).unwrap(),
}),
cx,
);
let default_model = registry.default_model().unwrap();
let fallback_model = registry.environment_fallback_model.clone().unwrap();
assert_eq!(default_model.model.id(), fallback_model.model.id());
assert_eq!(default_model.provider.id(), fallback_model.provider.id());
});
}
} }

View file

@ -44,6 +44,7 @@ ollama = { workspace = true, features = ["schemars"] }
open_ai = { workspace = true, features = ["schemars"] } open_ai = { workspace = true, features = ["schemars"] }
open_router = { workspace = true, features = ["schemars"] } open_router = { workspace = true, features = ["schemars"] }
partial-json-fixer.workspace = true partial-json-fixer.workspace = true
project.workspace = true
release_channel.workspace = true release_channel.workspace = true
schemars.workspace = true schemars.workspace = true
serde.workspace = true serde.workspace = true

View file

@ -3,8 +3,12 @@ use std::sync::Arc;
use ::settings::{Settings, SettingsStore}; use ::settings::{Settings, SettingsStore};
use client::{Client, UserStore}; use client::{Client, UserStore};
use collections::HashSet; use collections::HashSet;
use gpui::{App, Context, Entity}; use futures::future;
use language_model::{LanguageModelProviderId, LanguageModelRegistry}; use gpui::{App, AppContext as _, Context, Entity};
use language_model::{
AuthenticateError, ConfiguredModel, LanguageModelProviderId, LanguageModelRegistry,
};
use project::DisableAiSettings;
use provider::deepseek::DeepSeekLanguageModelProvider; use provider::deepseek::DeepSeekLanguageModelProvider;
pub mod provider; pub mod provider;
@ -13,7 +17,7 @@ pub mod ui;
use crate::provider::anthropic::AnthropicLanguageModelProvider; use crate::provider::anthropic::AnthropicLanguageModelProvider;
use crate::provider::bedrock::BedrockLanguageModelProvider; use crate::provider::bedrock::BedrockLanguageModelProvider;
use crate::provider::cloud::CloudLanguageModelProvider; use crate::provider::cloud::{self, CloudLanguageModelProvider};
use crate::provider::copilot_chat::CopilotChatLanguageModelProvider; use crate::provider::copilot_chat::CopilotChatLanguageModelProvider;
use crate::provider::google::GoogleLanguageModelProvider; use crate::provider::google::GoogleLanguageModelProvider;
use crate::provider::lmstudio::LmStudioLanguageModelProvider; use crate::provider::lmstudio::LmStudioLanguageModelProvider;
@ -48,6 +52,13 @@ pub fn init(user_store: Entity<UserStore>, client: Arc<Client>, cx: &mut App) {
cx, cx,
); );
}); });
let mut already_authenticated = false;
if !DisableAiSettings::get_global(cx).disable_ai {
authenticate_all_providers(registry.clone(), cx);
already_authenticated = true;
}
cx.observe_global::<SettingsStore>(move |cx| { cx.observe_global::<SettingsStore>(move |cx| {
let openai_compatible_providers_new = AllLanguageModelSettings::get_global(cx) let openai_compatible_providers_new = AllLanguageModelSettings::get_global(cx)
.openai_compatible .openai_compatible
@ -65,6 +76,12 @@ pub fn init(user_store: Entity<UserStore>, client: Arc<Client>, cx: &mut App) {
); );
}); });
openai_compatible_providers = openai_compatible_providers_new; openai_compatible_providers = openai_compatible_providers_new;
already_authenticated = false;
}
if !DisableAiSettings::get_global(cx).disable_ai && !already_authenticated {
authenticate_all_providers(registry.clone(), cx);
already_authenticated = true;
} }
}) })
.detach(); .detach();
@ -151,3 +168,83 @@ fn register_language_model_providers(
registry.register_provider(XAiLanguageModelProvider::new(client.http_client(), cx), cx); registry.register_provider(XAiLanguageModelProvider::new(client.http_client(), cx), cx);
registry.register_provider(CopilotChatLanguageModelProvider::new(cx), cx); registry.register_provider(CopilotChatLanguageModelProvider::new(cx), cx);
} }
/// Authenticates all providers in the [`LanguageModelRegistry`].
///
/// We do this so that we can populate the language selector with all of the
/// models from the configured providers.
///
/// This function won't do anything if AI is disabled.
fn authenticate_all_providers(registry: Entity<LanguageModelRegistry>, cx: &mut App) {
let providers_to_authenticate = registry
.read(cx)
.providers()
.iter()
.map(|provider| (provider.id(), provider.name(), provider.authenticate(cx)))
.collect::<Vec<_>>();
let mut tasks = Vec::with_capacity(providers_to_authenticate.len());
for (provider_id, provider_name, authenticate_task) in providers_to_authenticate {
tasks.push(cx.background_spawn(async move {
if let Err(err) = authenticate_task.await {
if matches!(err, AuthenticateError::CredentialsNotFound) {
// Since we're authenticating these providers in the
// background for the purposes of populating the
// language selector, we don't care about providers
// where the credentials are not found.
} else {
// Some providers have noisy failure states that we
// don't want to spam the logs with every time the
// language model selector is initialized.
//
// Ideally these should have more clear failure modes
// that we know are safe to ignore here, like what we do
// with `CredentialsNotFound` above.
match provider_id.0.as_ref() {
"lmstudio" | "ollama" => {
// LM Studio and Ollama both make fetch requests to the local APIs to determine if they are "authenticated".
//
// These fail noisily, so we don't log them.
}
"copilot_chat" => {
// Copilot Chat returns an error if Copilot is not enabled, so we don't log those errors.
}
_ => {
log::error!(
"Failed to authenticate provider: {}: {err}",
provider_name.0
);
}
}
}
}
}));
}
let all_authenticated_future = future::join_all(tasks);
cx.spawn(async move |cx| {
all_authenticated_future.await;
registry
.update(cx, |registry, cx| {
let cloud_provider = registry.provider(&cloud::PROVIDER_ID);
let fallback_model = cloud_provider
.iter()
.chain(registry.providers().iter())
.find(|provider| provider.is_authenticated(cx))
.and_then(|provider| {
Some(ConfiguredModel {
provider: provider.clone(),
model: provider
.default_model(cx)
.or_else(|| provider.recommended_models(cx).first().cloned())?,
})
});
registry.set_environment_fallback_model(fallback_model, cx);
})
.ok();
})
.detach();
}

View file

@ -44,8 +44,8 @@ use crate::provider::anthropic::{AnthropicEventMapper, count_anthropic_tokens, i
use crate::provider::google::{GoogleEventMapper, into_google}; use crate::provider::google::{GoogleEventMapper, into_google};
use crate::provider::open_ai::{OpenAiEventMapper, count_open_ai_tokens, into_open_ai}; use crate::provider::open_ai::{OpenAiEventMapper, count_open_ai_tokens, into_open_ai};
const PROVIDER_ID: LanguageModelProviderId = language_model::ZED_CLOUD_PROVIDER_ID; pub const PROVIDER_ID: LanguageModelProviderId = language_model::ZED_CLOUD_PROVIDER_ID;
const PROVIDER_NAME: LanguageModelProviderName = language_model::ZED_CLOUD_PROVIDER_NAME; pub const PROVIDER_NAME: LanguageModelProviderName = language_model::ZED_CLOUD_PROVIDER_NAME;
#[derive(Default, Clone, Debug, PartialEq)] #[derive(Default, Clone, Debug, PartialEq)]
pub struct ZedDotDevSettings { pub struct ZedDotDevSettings {
@ -146,7 +146,7 @@ impl State {
default_fast_model: None, default_fast_model: None,
recommended_models: Vec::new(), recommended_models: Vec::new(),
_fetch_models_task: cx.spawn(async move |this, cx| { _fetch_models_task: cx.spawn(async move |this, cx| {
maybe!(async move { maybe!(async {
let (client, llm_api_token) = this let (client, llm_api_token) = this
.read_with(cx, |this, _cx| (client.clone(), this.llm_api_token.clone()))?; .read_with(cx, |this, _cx| (client.clone(), this.llm_api_token.clone()))?;

View file

@ -4,6 +4,7 @@ use gpui::{
}; };
use itertools::Itertools; use itertools::Itertools;
use serde_json::json; use serde_json::json;
use settings::get_key_equivalents;
use ui::{Button, ButtonStyle}; use ui::{Button, ButtonStyle};
use ui::{ use ui::{
ButtonCommon, Clickable, Context, FluentBuilder, InteractiveElement, Label, LabelCommon, ButtonCommon, Clickable, Context, FluentBuilder, InteractiveElement, Label, LabelCommon,
@ -168,8 +169,7 @@ impl Item for KeyContextView {
impl Render for KeyContextView { impl Render for KeyContextView {
fn render(&mut self, window: &mut Window, cx: &mut Context<Self>) -> impl ui::IntoElement { fn render(&mut self, window: &mut Window, cx: &mut Context<Self>) -> impl ui::IntoElement {
use itertools::Itertools; use itertools::Itertools;
let key_equivalents = get_key_equivalents(cx.keyboard_layout().id());
let key_equivalents = cx.keyboard_mapper().get_key_equivalents();
v_flex() v_flex()
.id("key-context-view") .id("key-context-view")
.overflow_scroll() .overflow_scroll()

View file

@ -1323,7 +1323,7 @@ fn render_copy_code_block_button(
.icon_size(IconSize::Small) .icon_size(IconSize::Small)
.style(ButtonStyle::Filled) .style(ButtonStyle::Filled)
.shape(ui::IconButtonShape::Square) .shape(ui::IconButtonShape::Square)
.tooltip(Tooltip::text("Copy")) .tooltip(Tooltip::text("Copy Code"))
.on_click({ .on_click({
let markdown = markdown; let markdown = markdown;
move |_event, _window, cx| { move |_event, _window, cx| {

View file

@ -850,19 +850,13 @@ impl workspace::SerializableItem for Onboarding {
} }
mod persistence { mod persistence {
use db::{ use db::{define_connection, query, sqlez_macros::sql};
query,
sqlez::{domain::Domain, thread_safe_connection::ThreadSafeConnection},
sqlez_macros::sql,
};
use workspace::WorkspaceDb; use workspace::WorkspaceDb;
pub struct OnboardingPagesDb(ThreadSafeConnection); define_connection! {
pub static ref ONBOARDING_PAGES: OnboardingPagesDb<WorkspaceDb> =
impl Domain for OnboardingPagesDb { &[
const NAME: &str = stringify!(OnboardingPagesDb); sql!(
const MIGRATIONS: &[&str] = &[sql!(
CREATE TABLE onboarding_pages ( CREATE TABLE onboarding_pages (
workspace_id INTEGER, workspace_id INTEGER,
item_id INTEGER UNIQUE, item_id INTEGER UNIQUE,
@ -872,11 +866,10 @@ mod persistence {
FOREIGN KEY(workspace_id) REFERENCES workspaces(workspace_id) FOREIGN KEY(workspace_id) REFERENCES workspaces(workspace_id)
ON DELETE CASCADE ON DELETE CASCADE
) STRICT; ) STRICT;
)]; ),
];
} }
db::static_connection!(ONBOARDING_PAGES, OnboardingPagesDb, [WorkspaceDb]);
impl OnboardingPagesDb { impl OnboardingPagesDb {
query! { query! {
pub async fn save_onboarding_page( pub async fn save_onboarding_page(

View file

@ -414,19 +414,13 @@ impl workspace::SerializableItem for WelcomePage {
} }
mod persistence { mod persistence {
use db::{ use db::{define_connection, query, sqlez_macros::sql};
query,
sqlez::{domain::Domain, thread_safe_connection::ThreadSafeConnection},
sqlez_macros::sql,
};
use workspace::WorkspaceDb; use workspace::WorkspaceDb;
pub struct WelcomePagesDb(ThreadSafeConnection); define_connection! {
pub static ref WELCOME_PAGES: WelcomePagesDb<WorkspaceDb> =
impl Domain for WelcomePagesDb { &[
const NAME: &str = stringify!(WelcomePagesDb); sql!(
const MIGRATIONS: &[&str] = (&[sql!(
CREATE TABLE welcome_pages ( CREATE TABLE welcome_pages (
workspace_id INTEGER, workspace_id INTEGER,
item_id INTEGER UNIQUE, item_id INTEGER UNIQUE,
@ -436,11 +430,10 @@ mod persistence {
FOREIGN KEY(workspace_id) REFERENCES workspaces(workspace_id) FOREIGN KEY(workspace_id) REFERENCES workspaces(workspace_id)
ON DELETE CASCADE ON DELETE CASCADE
) STRICT; ) STRICT;
)]); ),
];
} }
db::static_connection!(WELCOME_PAGES, WelcomePagesDb, [WorkspaceDb]);
impl WelcomePagesDb { impl WelcomePagesDb {
query! { query! {
pub async fn save_welcome_page( pub async fn save_welcome_page(

View file

@ -4089,7 +4089,6 @@ impl ProjectPanel {
.when(!is_sticky, |this| { .when(!is_sticky, |this| {
this this
.when(is_highlighted && folded_directory_drag_target.is_none(), |this| this.border_color(transparent_white()).bg(item_colors.drag_over)) .when(is_highlighted && folded_directory_drag_target.is_none(), |this| this.border_color(transparent_white()).bg(item_colors.drag_over))
.when(settings.drag_and_drop, |this| this
.on_drag_move::<ExternalPaths>(cx.listener( .on_drag_move::<ExternalPaths>(cx.listener(
move |this, event: &DragMoveEvent<ExternalPaths>, _, cx| { move |this, event: &DragMoveEvent<ExternalPaths>, _, cx| {
let is_current_target = this.drag_target_entry.as_ref() let is_current_target = this.drag_target_entry.as_ref()
@ -4223,7 +4222,7 @@ impl ProjectPanel {
} }
this.drag_onto(selections, entry_id, kind.is_file(), window, cx); this.drag_onto(selections, entry_id, kind.is_file(), window, cx);
}), }),
)) )
}) })
.on_mouse_down( .on_mouse_down(
MouseButton::Left, MouseButton::Left,
@ -4434,7 +4433,6 @@ impl ProjectPanel {
div() div()
.when(!is_sticky, |div| { .when(!is_sticky, |div| {
div div
.when(settings.drag_and_drop, |div| div
.on_drop(cx.listener(move |this, selections: &DraggedSelection, window, cx| { .on_drop(cx.listener(move |this, selections: &DraggedSelection, window, cx| {
this.hover_scroll_task.take(); this.hover_scroll_task.take();
this.drag_target_entry = None; this.drag_target_entry = None;
@ -4466,7 +4464,7 @@ impl ProjectPanel {
} }
}, },
))) ))
}) })
.child( .child(
Label::new(DELIMITER.clone()) Label::new(DELIMITER.clone())
@ -4486,7 +4484,6 @@ impl ProjectPanel {
.when(index != components_len - 1, |div|{ .when(index != components_len - 1, |div|{
let target_entry_id = folded_ancestors.ancestors.get(components_len - 1 - index).cloned(); let target_entry_id = folded_ancestors.ancestors.get(components_len - 1 - index).cloned();
div div
.when(settings.drag_and_drop, |div| div
.on_drag_move(cx.listener( .on_drag_move(cx.listener(
move |this, event: &DragMoveEvent<DraggedSelection>, _, _| { move |this, event: &DragMoveEvent<DraggedSelection>, _, _| {
if event.bounds.contains(&event.event.position) { if event.bounds.contains(&event.event.position) {
@ -4524,7 +4521,7 @@ impl ProjectPanel {
target.index == index target.index == index
), |this| { ), |this| {
this.bg(item_colors.drag_over) this.bg(item_colors.drag_over)
})) })
}) })
}) })
.on_click(cx.listener(move |this, _, _, cx| { .on_click(cx.listener(move |this, _, _, cx| {
@ -5032,8 +5029,7 @@ impl ProjectPanel {
sticky_parents.reverse(); sticky_parents.reverse();
let panel_settings = ProjectPanelSettings::get_global(cx); let git_status_enabled = ProjectPanelSettings::get_global(cx).git_status;
let git_status_enabled = panel_settings.git_status;
let root_name = OsStr::new(worktree.root_name()); let root_name = OsStr::new(worktree.root_name());
let git_summaries_by_id = if git_status_enabled { let git_summaries_by_id = if git_status_enabled {
@ -5117,11 +5113,11 @@ impl Render for ProjectPanel {
fn render(&mut self, window: &mut Window, cx: &mut Context<Self>) -> impl IntoElement { fn render(&mut self, window: &mut Window, cx: &mut Context<Self>) -> impl IntoElement {
let has_worktree = !self.visible_entries.is_empty(); let has_worktree = !self.visible_entries.is_empty();
let project = self.project.read(cx); let project = self.project.read(cx);
let panel_settings = ProjectPanelSettings::get_global(cx); let indent_size = ProjectPanelSettings::get_global(cx).indent_size;
let indent_size = panel_settings.indent_size; let show_indent_guides =
let show_indent_guides = panel_settings.indent_guides.show == ShowIndentGuides::Always; ProjectPanelSettings::get_global(cx).indent_guides.show == ShowIndentGuides::Always;
let show_sticky_entries = { let show_sticky_entries = {
if panel_settings.sticky_scroll { if ProjectPanelSettings::get_global(cx).sticky_scroll {
let is_scrollable = self.scroll_handle.is_scrollable(); let is_scrollable = self.scroll_handle.is_scrollable();
let is_scrolled = self.scroll_handle.offset().y < px(0.); let is_scrolled = self.scroll_handle.offset().y < px(0.);
is_scrollable && is_scrolled is_scrollable && is_scrolled
@ -5209,10 +5205,8 @@ impl Render for ProjectPanel {
h_flex() h_flex()
.id("project-panel") .id("project-panel")
.group("project-panel") .group("project-panel")
.when(panel_settings.drag_and_drop, |this| { .on_drag_move(cx.listener(handle_drag_move::<ExternalPaths>))
this.on_drag_move(cx.listener(handle_drag_move::<ExternalPaths>)) .on_drag_move(cx.listener(handle_drag_move::<DraggedSelection>))
.on_drag_move(cx.listener(handle_drag_move::<DraggedSelection>))
})
.size_full() .size_full()
.relative() .relative()
.on_modifiers_changed(cx.listener( .on_modifiers_changed(cx.listener(
@ -5550,32 +5544,30 @@ impl Render for ProjectPanel {
})), })),
) )
.when(is_local, |div| { .when(is_local, |div| {
div.when(panel_settings.drag_and_drop, |div| { div.drag_over::<ExternalPaths>(|style, _, _, cx| {
div.drag_over::<ExternalPaths>(|style, _, _, cx| { style.bg(cx.theme().colors().drop_target_background)
style.bg(cx.theme().colors().drop_target_background)
})
.on_drop(cx.listener(
move |this, external_paths: &ExternalPaths, window, cx| {
this.drag_target_entry = None;
this.hover_scroll_task.take();
if let Some(task) = this
.workspace
.update(cx, |workspace, cx| {
workspace.open_workspace_for_paths(
true,
external_paths.paths().to_owned(),
window,
cx,
)
})
.log_err()
{
task.detach_and_log_err(cx);
}
cx.stop_propagation();
},
))
}) })
.on_drop(cx.listener(
move |this, external_paths: &ExternalPaths, window, cx| {
this.drag_target_entry = None;
this.hover_scroll_task.take();
if let Some(task) = this
.workspace
.update(cx, |workspace, cx| {
workspace.open_workspace_for_paths(
true,
external_paths.paths().to_owned(),
window,
cx,
)
})
.log_err()
{
task.detach_and_log_err(cx);
}
cx.stop_propagation();
},
))
}) })
} }
} }

View file

@ -47,7 +47,6 @@ pub struct ProjectPanelSettings {
pub scrollbar: ScrollbarSettings, pub scrollbar: ScrollbarSettings,
pub show_diagnostics: ShowDiagnostics, pub show_diagnostics: ShowDiagnostics,
pub hide_root: bool, pub hide_root: bool,
pub drag_and_drop: bool,
} }
#[derive(Copy, Clone, Debug, Serialize, Deserialize, JsonSchema, PartialEq, Eq)] #[derive(Copy, Clone, Debug, Serialize, Deserialize, JsonSchema, PartialEq, Eq)]
@ -161,10 +160,6 @@ pub struct ProjectPanelSettingsContent {
/// ///
/// Default: true /// Default: true
pub sticky_scroll: Option<bool>, pub sticky_scroll: Option<bool>,
/// Whether to enable drag-and-drop operations in the project panel.
///
/// Default: true
pub drag_and_drop: Option<bool>,
} }
impl Settings for ProjectPanelSettings { impl Settings for ProjectPanelSettings {

View file

@ -445,7 +445,7 @@ impl SshSocket {
} }
async fn platform(&self) -> Result<SshPlatform> { async fn platform(&self) -> Result<SshPlatform> {
let uname = self.run_command("sh", &["-lc", "uname -sm"]).await?; let uname = self.run_command("sh", &["-c", "uname -sm"]).await?;
let Some((os, arch)) = uname.split_once(" ") else { let Some((os, arch)) = uname.split_once(" ") else {
anyhow::bail!("unknown uname: {uname:?}") anyhow::bail!("unknown uname: {uname:?}")
}; };
@ -476,7 +476,7 @@ impl SshSocket {
} }
async fn shell(&self) -> String { async fn shell(&self) -> String {
match self.run_command("sh", &["-lc", "echo $SHELL"]).await { match self.run_command("sh", &["-c", "echo $SHELL"]).await {
Ok(shell) => shell.trim().to_owned(), Ok(shell) => shell.trim().to_owned(),
Err(e) => { Err(e) => {
log::error!("Failed to get shell: {e}"); log::error!("Failed to get shell: {e}");
@ -1533,7 +1533,7 @@ impl RemoteConnection for SshRemoteConnection {
let ssh_proxy_process = match self let ssh_proxy_process = match self
.socket .socket
.ssh_command("sh", &["-lc", &start_proxy_command]) .ssh_command("sh", &["-c", &start_proxy_command])
// IMPORTANT: we kill this process when we drop the task that uses it. // IMPORTANT: we kill this process when we drop the task that uses it.
.kill_on_drop(true) .kill_on_drop(true)
.spawn() .spawn()
@ -1910,7 +1910,7 @@ impl SshRemoteConnection {
.run_command( .run_command(
"sh", "sh",
&[ &[
"-lc", "-c",
&shell_script!("mkdir -p {parent}", parent = parent.to_string().as_ref()), &shell_script!("mkdir -p {parent}", parent = parent.to_string().as_ref()),
], ],
) )
@ -1988,7 +1988,7 @@ impl SshRemoteConnection {
.run_command( .run_command(
"sh", "sh",
&[ &[
"-lc", "-c",
&shell_script!("mkdir -p {parent}", parent = parent.to_string().as_ref()), &shell_script!("mkdir -p {parent}", parent = parent.to_string().as_ref()),
], ],
) )
@ -2036,7 +2036,7 @@ impl SshRemoteConnection {
dst_path = &dst_path.to_string() dst_path = &dst_path.to_string()
) )
}; };
self.socket.run_command("sh", &["-lc", &script]).await?; self.socket.run_command("sh", &["-c", &script]).await?;
Ok(()) Ok(())
} }

File diff suppressed because it is too large Load diff

View file

@ -3,8 +3,7 @@ use collections::{BTreeMap, HashMap, IndexMap};
use fs::Fs; use fs::Fs;
use gpui::{ use gpui::{
Action, ActionBuildError, App, InvalidKeystrokeError, KEYSTROKE_PARSE_EXPECTED_MESSAGE, Action, ActionBuildError, App, InvalidKeystrokeError, KEYSTROKE_PARSE_EXPECTED_MESSAGE,
KeyBinding, KeyBindingContextPredicate, KeyBindingMetaIndex, KeybindingKeystroke, Keystroke, KeyBinding, KeyBindingContextPredicate, KeyBindingMetaIndex, Keystroke, NoAction, SharedString,
NoAction, SharedString,
}; };
use schemars::{JsonSchema, json_schema}; use schemars::{JsonSchema, json_schema};
use serde::Deserialize; use serde::Deserialize;
@ -212,6 +211,9 @@ impl KeymapFile {
} }
pub fn load(content: &str, cx: &App) -> KeymapFileLoadResult { pub fn load(content: &str, cx: &App) -> KeymapFileLoadResult {
let key_equivalents =
crate::key_equivalents::get_key_equivalents(cx.keyboard_layout().id());
if content.is_empty() { if content.is_empty() {
return KeymapFileLoadResult::Success { return KeymapFileLoadResult::Success {
key_bindings: Vec::new(), key_bindings: Vec::new(),
@ -253,6 +255,12 @@ impl KeymapFile {
} }
}; };
let key_equivalents = if *use_key_equivalents {
key_equivalents.as_ref()
} else {
None
};
let mut section_errors = String::new(); let mut section_errors = String::new();
if !unrecognized_fields.is_empty() { if !unrecognized_fields.is_empty() {
@ -270,7 +278,7 @@ impl KeymapFile {
keystrokes, keystrokes,
action, action,
context_predicate.clone(), context_predicate.clone(),
*use_key_equivalents, key_equivalents,
cx, cx,
); );
match result { match result {
@ -328,7 +336,7 @@ impl KeymapFile {
keystrokes: &str, keystrokes: &str,
action: &KeymapAction, action: &KeymapAction,
context: Option<Rc<KeyBindingContextPredicate>>, context: Option<Rc<KeyBindingContextPredicate>>,
use_key_equivalents: bool, key_equivalents: Option<&HashMap<char, char>>,
cx: &App, cx: &App,
) -> std::result::Result<KeyBinding, String> { ) -> std::result::Result<KeyBinding, String> {
let (build_result, action_input_string) = match &action.0 { let (build_result, action_input_string) = match &action.0 {
@ -396,9 +404,8 @@ impl KeymapFile {
keystrokes, keystrokes,
action, action,
context, context,
use_key_equivalents, key_equivalents,
action_input_string.map(SharedString::from), action_input_string.map(SharedString::from),
cx.keyboard_mapper().as_ref(),
) { ) {
Ok(key_binding) => key_binding, Ok(key_binding) => key_binding,
Err(InvalidKeystrokeError { keystroke }) => { Err(InvalidKeystrokeError { keystroke }) => {
@ -600,7 +607,6 @@ impl KeymapFile {
mut operation: KeybindUpdateOperation<'a>, mut operation: KeybindUpdateOperation<'a>,
mut keymap_contents: String, mut keymap_contents: String,
tab_size: usize, tab_size: usize,
keyboard_mapper: &dyn gpui::PlatformKeyboardMapper,
) -> Result<String> { ) -> Result<String> {
match operation { match operation {
// if trying to replace a keybinding that is not user-defined, treat it as an add operation // if trying to replace a keybinding that is not user-defined, treat it as an add operation
@ -640,7 +646,7 @@ impl KeymapFile {
.action_value() .action_value()
.context("Failed to generate target action JSON value")?; .context("Failed to generate target action JSON value")?;
let Some((index, keystrokes_str)) = let Some((index, keystrokes_str)) =
find_binding(&keymap, &target, &target_action_value, keyboard_mapper) find_binding(&keymap, &target, &target_action_value)
else { else {
anyhow::bail!("Failed to find keybinding to remove"); anyhow::bail!("Failed to find keybinding to remove");
}; };
@ -675,7 +681,7 @@ impl KeymapFile {
.context("Failed to generate source action JSON value")?; .context("Failed to generate source action JSON value")?;
if let Some((index, keystrokes_str)) = if let Some((index, keystrokes_str)) =
find_binding(&keymap, &target, &target_action_value, keyboard_mapper) find_binding(&keymap, &target, &target_action_value)
{ {
if target.context == source.context { if target.context == source.context {
// if we are only changing the keybinding (common case) // if we are only changing the keybinding (common case)
@ -775,7 +781,7 @@ impl KeymapFile {
} }
let use_key_equivalents = from.and_then(|from| { let use_key_equivalents = from.and_then(|from| {
let action_value = from.action_value().context("Failed to serialize action value. `use_key_equivalents` on new keybinding may be incorrect.").log_err()?; let action_value = from.action_value().context("Failed to serialize action value. `use_key_equivalents` on new keybinding may be incorrect.").log_err()?;
let (index, _) = find_binding(&keymap, &from, &action_value, keyboard_mapper)?; let (index, _) = find_binding(&keymap, &from, &action_value)?;
Some(keymap.0[index].use_key_equivalents) Some(keymap.0[index].use_key_equivalents)
}).unwrap_or(false); }).unwrap_or(false);
if use_key_equivalents { if use_key_equivalents {
@ -802,7 +808,6 @@ impl KeymapFile {
keymap: &'b KeymapFile, keymap: &'b KeymapFile,
target: &KeybindUpdateTarget<'a>, target: &KeybindUpdateTarget<'a>,
target_action_value: &Value, target_action_value: &Value,
keyboard_mapper: &dyn gpui::PlatformKeyboardMapper,
) -> Option<(usize, &'b str)> { ) -> Option<(usize, &'b str)> {
let target_context_parsed = let target_context_parsed =
KeyBindingContextPredicate::parse(target.context.unwrap_or("")).ok(); KeyBindingContextPredicate::parse(target.context.unwrap_or("")).ok();
@ -818,11 +823,8 @@ impl KeymapFile {
for (keystrokes_str, action) in bindings { for (keystrokes_str, action) in bindings {
let Ok(keystrokes) = keystrokes_str let Ok(keystrokes) = keystrokes_str
.split_whitespace() .split_whitespace()
.map(|source| { .map(Keystroke::parse)
let keystroke = Keystroke::parse(source)?; .collect::<Result<Vec<_>, _>>()
Ok(KeybindingKeystroke::new(keystroke, false, keyboard_mapper))
})
.collect::<Result<Vec<_>, InvalidKeystrokeError>>()
else { else {
continue; continue;
}; };
@ -830,7 +832,7 @@ impl KeymapFile {
|| !keystrokes || !keystrokes
.iter() .iter()
.zip(target.keystrokes) .zip(target.keystrokes)
.all(|(a, b)| a.inner.should_match(b)) .all(|(a, b)| a.should_match(b))
{ {
continue; continue;
} }
@ -845,7 +847,7 @@ impl KeymapFile {
} }
} }
#[derive(Clone, Debug)] #[derive(Clone)]
pub enum KeybindUpdateOperation<'a> { pub enum KeybindUpdateOperation<'a> {
Replace { Replace {
/// Describes the keybind to create /// Describes the keybind to create
@ -914,7 +916,7 @@ impl<'a> KeybindUpdateOperation<'a> {
#[derive(Debug, Clone)] #[derive(Debug, Clone)]
pub struct KeybindUpdateTarget<'a> { pub struct KeybindUpdateTarget<'a> {
pub context: Option<&'a str>, pub context: Option<&'a str>,
pub keystrokes: &'a [KeybindingKeystroke], pub keystrokes: &'a [Keystroke],
pub action_name: &'a str, pub action_name: &'a str,
pub action_arguments: Option<&'a str>, pub action_arguments: Option<&'a str>,
} }
@ -939,9 +941,6 @@ impl<'a> KeybindUpdateTarget<'a> {
fn keystrokes_unparsed(&self) -> String { fn keystrokes_unparsed(&self) -> String {
let mut keystrokes = String::with_capacity(self.keystrokes.len() * 8); let mut keystrokes = String::with_capacity(self.keystrokes.len() * 8);
for keystroke in self.keystrokes { for keystroke in self.keystrokes {
// The reason use `keystroke.unparse()` instead of `keystroke.inner.unparse()`
// here is that, we want the user to use `ctrl-shift-4` instead of `ctrl-$`
// by default on Windows.
keystrokes.push_str(&keystroke.unparse()); keystrokes.push_str(&keystroke.unparse());
keystrokes.push(' '); keystrokes.push(' ');
} }
@ -960,7 +959,7 @@ impl<'a> KeybindUpdateTarget<'a> {
} }
} }
#[derive(Clone, Copy, Default, PartialEq, Eq, PartialOrd, Ord, Debug)] #[derive(Clone, Copy, Default, PartialEq, Eq, PartialOrd, Ord)]
pub enum KeybindSource { pub enum KeybindSource {
User, User,
Vim, Vim,
@ -1021,7 +1020,7 @@ impl From<KeybindSource> for KeyBindingMetaIndex {
#[cfg(test)] #[cfg(test)]
mod tests { mod tests {
use gpui::{DummyKeyboardMapper, KeybindingKeystroke, Keystroke}; use gpui::Keystroke;
use unindent::Unindent; use unindent::Unindent;
use crate::{ use crate::{
@ -1050,27 +1049,16 @@ mod tests {
operation: KeybindUpdateOperation, operation: KeybindUpdateOperation,
expected: impl ToString, expected: impl ToString,
) { ) {
let result = KeymapFile::update_keybinding( let result = KeymapFile::update_keybinding(operation, input.to_string(), 4)
operation, .expect("Update succeeded");
input.to_string(),
4,
&gpui::DummyKeyboardMapper,
)
.expect("Update succeeded");
pretty_assertions::assert_eq!(expected.to_string(), result); pretty_assertions::assert_eq!(expected.to_string(), result);
} }
#[track_caller] #[track_caller]
fn parse_keystrokes(keystrokes: &str) -> Vec<KeybindingKeystroke> { fn parse_keystrokes(keystrokes: &str) -> Vec<Keystroke> {
keystrokes keystrokes
.split(' ') .split(' ')
.map(|s| { .map(|s| Keystroke::parse(s).expect("Keystrokes valid"))
KeybindingKeystroke::new(
Keystroke::parse(s).expect("Keystrokes valid"),
false,
&DummyKeyboardMapper,
)
})
.collect() .collect()
} }

View file

@ -1,5 +1,6 @@
mod base_keymap_setting; mod base_keymap_setting;
mod editable_setting_control; mod editable_setting_control;
mod key_equivalents;
mod keymap_file; mod keymap_file;
mod settings_file; mod settings_file;
mod settings_json; mod settings_json;
@ -13,6 +14,7 @@ use util::asset_str;
pub use base_keymap_setting::*; pub use base_keymap_setting::*;
pub use editable_setting_control::*; pub use editable_setting_control::*;
pub use key_equivalents::*;
pub use keymap_file::{ pub use keymap_file::{
KeyBindingValidator, KeyBindingValidatorRegistration, KeybindSource, KeybindUpdateOperation, KeyBindingValidator, KeyBindingValidatorRegistration, KeybindSource, KeybindUpdateOperation,
KeybindUpdateTarget, KeymapFile, KeymapFileLoadResult, KeybindUpdateTarget, KeymapFile, KeymapFileLoadResult,
@ -87,10 +89,7 @@ pub fn default_settings() -> Cow<'static, str> {
#[cfg(target_os = "macos")] #[cfg(target_os = "macos")]
pub const DEFAULT_KEYMAP_PATH: &str = "keymaps/default-macos.json"; pub const DEFAULT_KEYMAP_PATH: &str = "keymaps/default-macos.json";
#[cfg(target_os = "windows")] #[cfg(not(target_os = "macos"))]
pub const DEFAULT_KEYMAP_PATH: &str = "keymaps/default-windows.json";
#[cfg(not(any(target_os = "macos", target_os = "windows")))]
pub const DEFAULT_KEYMAP_PATH: &str = "keymaps/default-linux.json"; pub const DEFAULT_KEYMAP_PATH: &str = "keymaps/default-linux.json";
pub fn default_keymap() -> Cow<'static, str> { pub fn default_keymap() -> Cow<'static, str> {

View file

@ -14,9 +14,9 @@ use gpui::{
Action, AppContext as _, AsyncApp, Axis, ClickEvent, Context, DismissEvent, Entity, Action, AppContext as _, AsyncApp, Axis, ClickEvent, Context, DismissEvent, Entity,
EventEmitter, FocusHandle, Focusable, Global, IsZero, EventEmitter, FocusHandle, Focusable, Global, IsZero,
KeyBindingContextPredicate::{And, Descendant, Equal, Identifier, Not, NotEqual, Or}, KeyBindingContextPredicate::{And, Descendant, Equal, Identifier, Not, NotEqual, Or},
KeyContext, KeybindingKeystroke, Keystroke, MouseButton, PlatformKeyboardMapper, Point, KeyContext, Keystroke, MouseButton, Point, ScrollStrategy, ScrollWheelEvent, Stateful,
ScrollStrategy, ScrollWheelEvent, Stateful, StyledText, Subscription, Task, StyledText, Subscription, Task, TextStyleRefinement, WeakEntity, actions, anchored, deferred,
TextStyleRefinement, WeakEntity, actions, anchored, deferred, div, div,
}; };
use language::{Language, LanguageConfig, ToOffset as _}; use language::{Language, LanguageConfig, ToOffset as _};
use notifications::status_toast::{StatusToast, ToastIcon}; use notifications::status_toast::{StatusToast, ToastIcon};
@ -174,7 +174,7 @@ impl FilterState {
#[derive(Debug, Default, PartialEq, Eq, Clone, Hash)] #[derive(Debug, Default, PartialEq, Eq, Clone, Hash)]
struct ActionMapping { struct ActionMapping {
keystrokes: Vec<KeybindingKeystroke>, keystrokes: Vec<Keystroke>,
context: Option<SharedString>, context: Option<SharedString>,
} }
@ -236,7 +236,7 @@ struct ConflictState {
} }
type ConflictKeybindMapping = HashMap< type ConflictKeybindMapping = HashMap<
Vec<KeybindingKeystroke>, Vec<Keystroke>,
Vec<( Vec<(
Option<gpui::KeyBindingContextPredicate>, Option<gpui::KeyBindingContextPredicate>,
Vec<ConflictOrigin>, Vec<ConflictOrigin>,
@ -414,14 +414,12 @@ impl Focusable for KeymapEditor {
} }
} }
/// Helper function to check if two keystroke sequences match exactly /// Helper function to check if two keystroke sequences match exactly
fn keystrokes_match_exactly( fn keystrokes_match_exactly(keystrokes1: &[Keystroke], keystrokes2: &[Keystroke]) -> bool {
keystrokes1: &[KeybindingKeystroke],
keystrokes2: &[KeybindingKeystroke],
) -> bool {
keystrokes1.len() == keystrokes2.len() keystrokes1.len() == keystrokes2.len()
&& keystrokes1.iter().zip(keystrokes2).all(|(k1, k2)| { && keystrokes1
k1.inner.key == k2.inner.key && k1.inner.modifiers == k2.inner.modifiers .iter()
}) .zip(keystrokes2)
.all(|(k1, k2)| k1.key == k2.key && k1.modifiers == k2.modifiers)
} }
impl KeymapEditor { impl KeymapEditor {
@ -511,7 +509,7 @@ impl KeymapEditor {
self.filter_editor.read(cx).text(cx) self.filter_editor.read(cx).text(cx)
} }
fn current_keystroke_query(&self, cx: &App) -> Vec<KeybindingKeystroke> { fn current_keystroke_query(&self, cx: &App) -> Vec<Keystroke> {
match self.search_mode { match self.search_mode {
SearchMode::KeyStroke { .. } => self.keystroke_editor.read(cx).keystrokes().to_vec(), SearchMode::KeyStroke { .. } => self.keystroke_editor.read(cx).keystrokes().to_vec(),
SearchMode::Normal => Default::default(), SearchMode::Normal => Default::default(),
@ -532,7 +530,7 @@ impl KeymapEditor {
let keystroke_query = keystroke_query let keystroke_query = keystroke_query
.into_iter() .into_iter()
.map(|keystroke| keystroke.inner.unparse()) .map(|keystroke| keystroke.unparse())
.collect::<Vec<String>>() .collect::<Vec<String>>()
.join(" "); .join(" ");
@ -556,7 +554,7 @@ impl KeymapEditor {
async fn update_matches( async fn update_matches(
this: WeakEntity<Self>, this: WeakEntity<Self>,
action_query: String, action_query: String,
keystroke_query: Vec<KeybindingKeystroke>, keystroke_query: Vec<Keystroke>,
cx: &mut AsyncApp, cx: &mut AsyncApp,
) -> anyhow::Result<()> { ) -> anyhow::Result<()> {
let action_query = command_palette::normalize_action_query(&action_query); let action_query = command_palette::normalize_action_query(&action_query);
@ -605,15 +603,13 @@ impl KeymapEditor {
{ {
let query = &keystroke_query[query_cursor]; let query = &keystroke_query[query_cursor];
let keystroke = &keystrokes[keystroke_cursor]; let keystroke = &keystrokes[keystroke_cursor];
let matches = query let matches =
.inner query.modifiers.is_subset_of(&keystroke.modifiers)
.modifiers && ((query.key.is_empty()
.is_subset_of(&keystroke.inner.modifiers) || query.key == keystroke.key)
&& ((query.inner.key.is_empty() && query.key_char.as_ref().is_none_or(
|| query.inner.key == keystroke.inner.key) |q_kc| q_kc == &keystroke.key,
&& query.inner.key_char.as_ref().is_none_or( ));
|q_kc| q_kc == &keystroke.inner.key,
));
if matches { if matches {
found_count += 1; found_count += 1;
query_cursor += 1; query_cursor += 1;
@ -682,7 +678,7 @@ impl KeymapEditor {
.map(KeybindSource::from_meta) .map(KeybindSource::from_meta)
.unwrap_or(KeybindSource::Unknown); .unwrap_or(KeybindSource::Unknown);
let keystroke_text = ui::text_for_keybinding_keystrokes(key_binding.keystrokes(), cx); let keystroke_text = ui::text_for_keystrokes(key_binding.keystrokes(), cx);
let ui_key_binding = ui::KeyBinding::new_from_gpui(key_binding.clone(), cx) let ui_key_binding = ui::KeyBinding::new_from_gpui(key_binding.clone(), cx)
.vim_mode(source == KeybindSource::Vim); .vim_mode(source == KeybindSource::Vim);
@ -1206,11 +1202,8 @@ impl KeymapEditor {
.read(cx) .read(cx)
.get_scrollbar_offset(Axis::Vertical), .get_scrollbar_offset(Axis::Vertical),
)); ));
let keyboard_mapper = cx.keyboard_mapper().clone(); cx.spawn(async move |_, _| remove_keybinding(to_remove, &fs, tab_size).await)
cx.spawn(async move |_, _| { .detach_and_notify_err(window, cx);
remove_keybinding(to_remove, &fs, tab_size, keyboard_mapper.as_ref()).await
})
.detach_and_notify_err(window, cx);
} }
fn copy_context_to_clipboard( fn copy_context_to_clipboard(
@ -1429,7 +1422,7 @@ impl ProcessedBinding {
.map(|keybind| keybind.get_action_mapping()) .map(|keybind| keybind.get_action_mapping())
} }
fn keystrokes(&self) -> Option<&[KeybindingKeystroke]> { fn keystrokes(&self) -> Option<&[Keystroke]> {
self.ui_key_binding() self.ui_key_binding()
.map(|binding| binding.keystrokes.as_slice()) .map(|binding| binding.keystrokes.as_slice())
} }
@ -2227,7 +2220,7 @@ impl KeybindingEditorModal {
Ok(action_arguments) Ok(action_arguments)
} }
fn validate_keystrokes(&self, cx: &App) -> anyhow::Result<Vec<KeybindingKeystroke>> { fn validate_keystrokes(&self, cx: &App) -> anyhow::Result<Vec<Keystroke>> {
let new_keystrokes = self let new_keystrokes = self
.keybind_editor .keybind_editor
.read_with(cx, |editor, _| editor.keystrokes().to_vec()); .read_with(cx, |editor, _| editor.keystrokes().to_vec());
@ -2323,7 +2316,6 @@ impl KeybindingEditorModal {
}).unwrap_or(Ok(()))?; }).unwrap_or(Ok(()))?;
let create = self.creating; let create = self.creating;
let keyboard_mapper = cx.keyboard_mapper().clone();
cx.spawn(async move |this, cx| { cx.spawn(async move |this, cx| {
let action_name = existing_keybind.action().name; let action_name = existing_keybind.action().name;
@ -2336,7 +2328,6 @@ impl KeybindingEditorModal {
new_action_args.as_deref(), new_action_args.as_deref(),
&fs, &fs,
tab_size, tab_size,
keyboard_mapper.as_ref(),
) )
.await .await
{ {
@ -2454,21 +2445,11 @@ impl KeybindingEditorModal {
} }
} }
fn remove_key_char( fn remove_key_char(Keystroke { modifiers, key, .. }: Keystroke) -> Keystroke {
KeybindingKeystroke { Keystroke {
inner, modifiers,
display_modifiers, key,
display_key, ..Default::default()
}: KeybindingKeystroke,
) -> KeybindingKeystroke {
KeybindingKeystroke {
inner: Keystroke {
modifiers: inner.modifiers,
key: inner.key,
key_char: None,
},
display_modifiers,
display_key,
} }
} }
@ -3011,7 +2992,6 @@ async fn save_keybinding_update(
new_args: Option<&str>, new_args: Option<&str>,
fs: &Arc<dyn Fs>, fs: &Arc<dyn Fs>,
tab_size: usize, tab_size: usize,
keyboard_mapper: &dyn PlatformKeyboardMapper,
) -> anyhow::Result<()> { ) -> anyhow::Result<()> {
let keymap_contents = settings::KeymapFile::load_keymap_file(fs) let keymap_contents = settings::KeymapFile::load_keymap_file(fs)
.await .await
@ -3054,13 +3034,9 @@ async fn save_keybinding_update(
let (new_keybinding, removed_keybinding, source) = operation.generate_telemetry(); let (new_keybinding, removed_keybinding, source) = operation.generate_telemetry();
let updated_keymap_contents = settings::KeymapFile::update_keybinding( let updated_keymap_contents =
operation, settings::KeymapFile::update_keybinding(operation, keymap_contents, tab_size)
keymap_contents, .map_err(|err| anyhow::anyhow!("Could not save updated keybinding: {}", err))?;
tab_size,
keyboard_mapper,
)
.map_err(|err| anyhow::anyhow!("Could not save updated keybinding: {}", err))?;
fs.write( fs.write(
paths::keymap_file().as_path(), paths::keymap_file().as_path(),
updated_keymap_contents.as_bytes(), updated_keymap_contents.as_bytes(),
@ -3081,7 +3057,6 @@ async fn remove_keybinding(
existing: ProcessedBinding, existing: ProcessedBinding,
fs: &Arc<dyn Fs>, fs: &Arc<dyn Fs>,
tab_size: usize, tab_size: usize,
keyboard_mapper: &dyn PlatformKeyboardMapper,
) -> anyhow::Result<()> { ) -> anyhow::Result<()> {
let Some(keystrokes) = existing.keystrokes() else { let Some(keystrokes) = existing.keystrokes() else {
anyhow::bail!("Cannot remove a keybinding that does not exist"); anyhow::bail!("Cannot remove a keybinding that does not exist");
@ -3105,13 +3080,9 @@ async fn remove_keybinding(
}; };
let (new_keybinding, removed_keybinding, source) = operation.generate_telemetry(); let (new_keybinding, removed_keybinding, source) = operation.generate_telemetry();
let updated_keymap_contents = settings::KeymapFile::update_keybinding( let updated_keymap_contents =
operation, settings::KeymapFile::update_keybinding(operation, keymap_contents, tab_size)
keymap_contents, .context("Failed to update keybinding")?;
tab_size,
keyboard_mapper,
)
.context("Failed to update keybinding")?;
fs.write( fs.write(
paths::keymap_file().as_path(), paths::keymap_file().as_path(),
updated_keymap_contents.as_bytes(), updated_keymap_contents.as_bytes(),
@ -3377,15 +3348,12 @@ impl SerializableItem for KeymapEditor {
} }
mod persistence { mod persistence {
use db::{query, sqlez::domain::Domain, sqlez_macros::sql}; use db::{define_connection, query, sqlez_macros::sql};
use workspace::WorkspaceDb; use workspace::WorkspaceDb;
pub struct KeybindingEditorDb(db::sqlez::thread_safe_connection::ThreadSafeConnection); define_connection! {
pub static ref KEYBINDING_EDITORS: KeybindingEditorDb<WorkspaceDb> =
impl Domain for KeybindingEditorDb { &[sql!(
const NAME: &str = stringify!(KeybindingEditorDb);
const MIGRATIONS: &[&str] = &[sql!(
CREATE TABLE keybinding_editors ( CREATE TABLE keybinding_editors (
workspace_id INTEGER, workspace_id INTEGER,
item_id INTEGER UNIQUE, item_id INTEGER UNIQUE,
@ -3394,11 +3362,9 @@ mod persistence {
FOREIGN KEY(workspace_id) REFERENCES workspaces(workspace_id) FOREIGN KEY(workspace_id) REFERENCES workspaces(workspace_id)
ON DELETE CASCADE ON DELETE CASCADE
) STRICT; ) STRICT;
)]; )];
} }
db::static_connection!(KEYBINDING_EDITORS, KeybindingEditorDb, [WorkspaceDb]);
impl KeybindingEditorDb { impl KeybindingEditorDb {
query! { query! {
pub async fn save_keybinding_editor( pub async fn save_keybinding_editor(

View file

@ -1,6 +1,6 @@
use gpui::{ use gpui::{
Animation, AnimationExt, Context, EventEmitter, FocusHandle, Focusable, FontWeight, KeyContext, Animation, AnimationExt, Context, EventEmitter, FocusHandle, Focusable, FontWeight, KeyContext,
KeybindingKeystroke, Keystroke, Modifiers, ModifiersChangedEvent, Subscription, Task, actions, Keystroke, Modifiers, ModifiersChangedEvent, Subscription, Task, actions,
}; };
use ui::{ use ui::{
ActiveTheme as _, Color, IconButton, IconButtonShape, IconName, IconSize, Label, LabelSize, ActiveTheme as _, Color, IconButton, IconButtonShape, IconName, IconSize, Label, LabelSize,
@ -42,8 +42,8 @@ impl PartialEq for CloseKeystrokeResult {
} }
pub struct KeystrokeInput { pub struct KeystrokeInput {
keystrokes: Vec<KeybindingKeystroke>, keystrokes: Vec<Keystroke>,
placeholder_keystrokes: Option<Vec<KeybindingKeystroke>>, placeholder_keystrokes: Option<Vec<Keystroke>>,
outer_focus_handle: FocusHandle, outer_focus_handle: FocusHandle,
inner_focus_handle: FocusHandle, inner_focus_handle: FocusHandle,
intercept_subscription: Option<Subscription>, intercept_subscription: Option<Subscription>,
@ -70,7 +70,7 @@ impl KeystrokeInput {
const KEYSTROKE_COUNT_MAX: usize = 3; const KEYSTROKE_COUNT_MAX: usize = 3;
pub fn new( pub fn new(
placeholder_keystrokes: Option<Vec<KeybindingKeystroke>>, placeholder_keystrokes: Option<Vec<Keystroke>>,
window: &mut Window, window: &mut Window,
cx: &mut Context<Self>, cx: &mut Context<Self>,
) -> Self { ) -> Self {
@ -97,7 +97,7 @@ impl KeystrokeInput {
} }
} }
pub fn set_keystrokes(&mut self, keystrokes: Vec<KeybindingKeystroke>, cx: &mut Context<Self>) { pub fn set_keystrokes(&mut self, keystrokes: Vec<Keystroke>, cx: &mut Context<Self>) {
self.keystrokes = keystrokes; self.keystrokes = keystrokes;
self.keystrokes_changed(cx); self.keystrokes_changed(cx);
} }
@ -106,7 +106,7 @@ impl KeystrokeInput {
self.search = search; self.search = search;
} }
pub fn keystrokes(&self) -> &[KeybindingKeystroke] { pub fn keystrokes(&self) -> &[Keystroke] {
if let Some(placeholders) = self.placeholder_keystrokes.as_ref() if let Some(placeholders) = self.placeholder_keystrokes.as_ref()
&& self.keystrokes.is_empty() && self.keystrokes.is_empty()
{ {
@ -116,22 +116,18 @@ impl KeystrokeInput {
&& self && self
.keystrokes .keystrokes
.last() .last()
.is_some_and(|last| last.display_key.is_empty()) .is_some_and(|last| last.key.is_empty())
{ {
return &self.keystrokes[..self.keystrokes.len() - 1]; return &self.keystrokes[..self.keystrokes.len() - 1];
} }
&self.keystrokes &self.keystrokes
} }
fn dummy(modifiers: Modifiers) -> KeybindingKeystroke { fn dummy(modifiers: Modifiers) -> Keystroke {
KeybindingKeystroke { Keystroke {
inner: Keystroke { modifiers,
modifiers, key: "".to_string(),
key: "".to_string(), key_char: None,
key_char: None,
},
display_modifiers: modifiers,
display_key: "".to_string(),
} }
} }
@ -258,7 +254,7 @@ impl KeystrokeInput {
self.keystrokes_changed(cx); self.keystrokes_changed(cx);
if let Some(last) = self.keystrokes.last_mut() if let Some(last) = self.keystrokes.last_mut()
&& last.display_key.is_empty() && last.key.is_empty()
&& keystrokes_len <= Self::KEYSTROKE_COUNT_MAX && keystrokes_len <= Self::KEYSTROKE_COUNT_MAX
{ {
if !self.search && !event.modifiers.modified() { if !self.search && !event.modifiers.modified() {
@ -267,15 +263,13 @@ impl KeystrokeInput {
} }
if self.search { if self.search {
if self.previous_modifiers.modified() { if self.previous_modifiers.modified() {
last.display_modifiers |= event.modifiers; last.modifiers |= event.modifiers;
last.inner.modifiers |= event.modifiers;
} else { } else {
self.keystrokes.push(Self::dummy(event.modifiers)); self.keystrokes.push(Self::dummy(event.modifiers));
} }
self.previous_modifiers |= event.modifiers; self.previous_modifiers |= event.modifiers;
} else { } else {
last.display_modifiers = event.modifiers; last.modifiers = event.modifiers;
last.inner.modifiers = event.modifiers;
return; return;
} }
} else if keystrokes_len < Self::KEYSTROKE_COUNT_MAX { } else if keystrokes_len < Self::KEYSTROKE_COUNT_MAX {
@ -303,17 +297,14 @@ impl KeystrokeInput {
return; return;
} }
let mut keystroke = let mut keystroke = keystroke.clone();
KeybindingKeystroke::new(keystroke.clone(), false, cx.keyboard_mapper().as_ref());
if let Some(last) = self.keystrokes.last() if let Some(last) = self.keystrokes.last()
&& last.display_key.is_empty() && last.key.is_empty()
&& (!self.search || self.previous_modifiers.modified()) && (!self.search || self.previous_modifiers.modified())
{ {
let display_key = keystroke.display_key.clone(); let key = keystroke.key.clone();
let inner_key = keystroke.inner.key.clone();
keystroke = last.clone(); keystroke = last.clone();
keystroke.display_key = display_key; keystroke.key = key;
keystroke.inner.key = inner_key;
self.keystrokes.pop(); self.keystrokes.pop();
} }
@ -333,14 +324,11 @@ impl KeystrokeInput {
self.keystrokes_changed(cx); self.keystrokes_changed(cx);
if self.search { if self.search {
self.previous_modifiers = keystroke.display_modifiers; self.previous_modifiers = keystroke.modifiers;
return; return;
} }
if self.keystrokes.len() < Self::KEYSTROKE_COUNT_MAX if self.keystrokes.len() < Self::KEYSTROKE_COUNT_MAX && keystroke.modifiers.modified() {
&& keystroke.display_modifiers.modified() self.keystrokes.push(Self::dummy(keystroke.modifiers));
{
self.keystrokes
.push(Self::dummy(keystroke.display_modifiers));
} }
} }
@ -376,7 +364,7 @@ impl KeystrokeInput {
&self.keystrokes &self.keystrokes
}; };
keystrokes.iter().map(move |keystroke| { keystrokes.iter().map(move |keystroke| {
h_flex().children(ui::render_keybinding_keystroke( h_flex().children(ui::render_keystroke(
keystroke, keystroke,
Some(Color::Default), Some(Color::Default),
Some(rems(0.875).into()), Some(rems(0.875).into()),
@ -821,13 +809,9 @@ mod tests {
/// Verifies that the keystrokes match the expected strings /// Verifies that the keystrokes match the expected strings
#[track_caller] #[track_caller]
pub fn expect_keystrokes(&mut self, expected: &[&str]) -> &mut Self { pub fn expect_keystrokes(&mut self, expected: &[&str]) -> &mut Self {
let actual: Vec<Keystroke> = self.input.read_with(&self.cx, |input, _| { let actual = self
input .input
.keystrokes .read_with(&self.cx, |input, _| input.keystrokes.clone());
.iter()
.map(|keystroke| keystroke.inner.clone())
.collect()
});
Self::expect_keystrokes_equal(&actual, expected); Self::expect_keystrokes_equal(&actual, expected);
self self
} }
@ -955,7 +939,7 @@ mod tests {
} }
struct KeystrokeUpdateTracker { struct KeystrokeUpdateTracker {
initial_keystrokes: Vec<KeybindingKeystroke>, initial_keystrokes: Vec<Keystroke>,
_subscription: Subscription, _subscription: Subscription,
input: Entity<KeystrokeInput>, input: Entity<KeystrokeInput>,
received_keystrokes_updated: bool, received_keystrokes_updated: bool,
@ -999,8 +983,8 @@ mod tests {
); );
} }
fn keystrokes_str(ks: &[KeybindingKeystroke]) -> String { fn keystrokes_str(ks: &[Keystroke]) -> String {
ks.iter().map(|ks| ks.inner.unparse()).join(" ") ks.iter().map(|ks| ks.unparse()).join(" ")
} }
} }
} }

View file

@ -1,12 +1,8 @@
use crate::connection::Connection; use crate::connection::Connection;
pub trait Domain: 'static { pub trait Domain: 'static {
const NAME: &str; fn name() -> &'static str;
const MIGRATIONS: &[&str]; fn migrations() -> &'static [&'static str];
fn should_allow_migration_change(_index: usize, _old: &str, _new: &str) -> bool {
false
}
} }
pub trait Migrator: 'static { pub trait Migrator: 'static {
@ -21,11 +17,7 @@ impl Migrator for () {
impl<D: Domain> Migrator for D { impl<D: Domain> Migrator for D {
fn migrate(connection: &Connection) -> anyhow::Result<()> { fn migrate(connection: &Connection) -> anyhow::Result<()> {
connection.migrate( connection.migrate(Self::name(), Self::migrations())
Self::NAME,
Self::MIGRATIONS,
Self::should_allow_migration_change,
)
} }
} }

View file

@ -34,12 +34,7 @@ impl Connection {
/// Note: Unlike everything else in SQLez, migrations are run eagerly, without first /// Note: Unlike everything else in SQLez, migrations are run eagerly, without first
/// preparing the SQL statements. This makes it possible to do multi-statement schema /// preparing the SQL statements. This makes it possible to do multi-statement schema
/// updates in a single string without running into prepare errors. /// updates in a single string without running into prepare errors.
pub fn migrate( pub fn migrate(&self, domain: &'static str, migrations: &[&'static str]) -> Result<()> {
&self,
domain: &'static str,
migrations: &[&'static str],
mut should_allow_migration_change: impl FnMut(usize, &str, &str) -> bool,
) -> Result<()> {
self.with_savepoint("migrating", || { self.with_savepoint("migrating", || {
// Setup the migrations table unconditionally // Setup the migrations table unconditionally
self.exec(indoc! {" self.exec(indoc! {"
@ -70,14 +65,9 @@ impl Connection {
&sqlformat::QueryParams::None, &sqlformat::QueryParams::None,
Default::default(), Default::default(),
); );
if completed_migration == migration if completed_migration == migration {
|| migration.trim().starts_with("-- ALLOW_MIGRATION_CHANGE")
{
// Migration already run. Continue // Migration already run. Continue
continue; continue;
} else if should_allow_migration_change(index, &completed_migration, &migration)
{
continue;
} else { } else {
anyhow::bail!(formatdoc! {" anyhow::bail!(formatdoc! {"
Migration changed for {domain} at step {index} Migration changed for {domain} at step {index}
@ -118,7 +108,6 @@ mod test {
a TEXT, a TEXT,
b TEXT b TEXT
)"}], )"}],
disallow_migration_change,
) )
.unwrap(); .unwrap();
@ -147,7 +136,6 @@ mod test {
d TEXT d TEXT
)"}, )"},
], ],
disallow_migration_change,
) )
.unwrap(); .unwrap();
@ -226,11 +214,7 @@ mod test {
// Run the migration verifying that the row got dropped // Run the migration verifying that the row got dropped
connection connection
.migrate( .migrate("test", &["DELETE FROM test_table"])
"test",
&["DELETE FROM test_table"],
disallow_migration_change,
)
.unwrap(); .unwrap();
assert_eq!( assert_eq!(
connection connection
@ -248,11 +232,7 @@ mod test {
// Run the same migration again and verify that the table was left unchanged // Run the same migration again and verify that the table was left unchanged
connection connection
.migrate( .migrate("test", &["DELETE FROM test_table"])
"test",
&["DELETE FROM test_table"],
disallow_migration_change,
)
.unwrap(); .unwrap();
assert_eq!( assert_eq!(
connection connection
@ -272,28 +252,27 @@ mod test {
.migrate( .migrate(
"test migration", "test migration",
&[ &[
"CREATE TABLE test (col INTEGER)", indoc! {"
"INSERT INTO test (col) VALUES (1)", CREATE TABLE test (
col INTEGER
)"},
indoc! {"
INSERT INTO test (col) VALUES (1)"},
], ],
disallow_migration_change,
) )
.unwrap(); .unwrap();
let mut migration_changed = false;
// Create another migration with the same domain but different steps // Create another migration with the same domain but different steps
let second_migration_result = connection.migrate( let second_migration_result = connection.migrate(
"test migration", "test migration",
&[ &[
"CREATE TABLE test (color INTEGER )", indoc! {"
"INSERT INTO test (color) VALUES (1)", CREATE TABLE test (
color INTEGER
)"},
indoc! {"
INSERT INTO test (color) VALUES (1)"},
], ],
|_, old, new| {
assert_eq!(old, "CREATE TABLE test (col INTEGER)");
assert_eq!(new, "CREATE TABLE test (color INTEGER)");
migration_changed = true;
false
},
); );
// Verify new migration returns error when run // Verify new migration returns error when run
@ -305,11 +284,7 @@ mod test {
let connection = Connection::open_memory(Some("test_create_alter_drop")); let connection = Connection::open_memory(Some("test_create_alter_drop"));
connection connection
.migrate( .migrate("first_migration", &["CREATE TABLE table1(a TEXT) STRICT;"])
"first_migration",
&["CREATE TABLE table1(a TEXT) STRICT;"],
disallow_migration_change,
)
.unwrap(); .unwrap();
connection connection
@ -330,7 +305,6 @@ mod test {
ALTER TABLE table2 RENAME TO table1; ALTER TABLE table2 RENAME TO table1;
"}], "}],
disallow_migration_change,
) )
.unwrap(); .unwrap();
@ -338,8 +312,4 @@ mod test {
assert_eq!(res, "test text"); assert_eq!(res, "test text");
} }
fn disallow_migration_change(_: usize, _: &str, _: &str) -> bool {
false
}
} }

View file

@ -278,8 +278,12 @@ mod test {
enum TestDomain {} enum TestDomain {}
impl Domain for TestDomain { impl Domain for TestDomain {
const NAME: &str = "test"; fn name() -> &'static str {
const MIGRATIONS: &[&str] = &["CREATE TABLE test(col1 TEXT, col2 TEXT) STRICT;"]; "test"
}
fn migrations() -> &'static [&'static str] {
&["CREATE TABLE test(col1 TEXT, col2 TEXT) STRICT;"]
}
} }
for _ in 0..100 { for _ in 0..100 {
@ -308,9 +312,12 @@ mod test {
fn wild_zed_lost_failure() { fn wild_zed_lost_failure() {
enum TestWorkspace {} enum TestWorkspace {}
impl Domain for TestWorkspace { impl Domain for TestWorkspace {
const NAME: &str = "workspace"; fn name() -> &'static str {
"workspace"
}
const MIGRATIONS: &[&str] = &[" fn migrations() -> &'static [&'static str] {
&["
CREATE TABLE workspaces( CREATE TABLE workspaces(
workspace_id INTEGER PRIMARY KEY, workspace_id INTEGER PRIMARY KEY,
dock_visible INTEGER, -- Boolean dock_visible INTEGER, -- Boolean
@ -329,7 +336,8 @@ mod test {
ON DELETE CASCADE ON DELETE CASCADE
ON UPDATE CASCADE ON UPDATE CASCADE
) STRICT; ) STRICT;
"]; "]
}
} }
let builder = let builder =

View file

@ -9,11 +9,7 @@ use std::path::{Path, PathBuf};
use ui::{App, Context, Pixels, Window}; use ui::{App, Context, Pixels, Window};
use util::ResultExt as _; use util::ResultExt as _;
use db::{ use db::{define_connection, query, sqlez::statement::Statement, sqlez_macros::sql};
query,
sqlez::{domain::Domain, statement::Statement, thread_safe_connection::ThreadSafeConnection},
sqlez_macros::sql,
};
use workspace::{ use workspace::{
ItemHandle, ItemId, Member, Pane, PaneAxis, PaneGroup, SerializableItem as _, Workspace, ItemHandle, ItemId, Member, Pane, PaneAxis, PaneGroup, SerializableItem as _, Workspace,
WorkspaceDb, WorkspaceId, WorkspaceDb, WorkspaceId,
@ -379,13 +375,9 @@ impl<'de> Deserialize<'de> for SerializedAxis {
} }
} }
pub struct TerminalDb(ThreadSafeConnection); define_connection! {
pub static ref TERMINAL_DB: TerminalDb<WorkspaceDb> =
impl Domain for TerminalDb { &[sql!(
const NAME: &str = stringify!(TerminalDb);
const MIGRATIONS: &[&str] = &[
sql!(
CREATE TABLE terminals ( CREATE TABLE terminals (
workspace_id INTEGER, workspace_id INTEGER,
item_id INTEGER UNIQUE, item_id INTEGER UNIQUE,
@ -422,8 +414,6 @@ impl Domain for TerminalDb {
]; ];
} }
db::static_connection!(TERMINAL_DB, TerminalDb, [WorkspaceDb]);
impl TerminalDb { impl TerminalDb {
query! { query! {
pub async fn update_workspace_id( pub async fn update_workspace_id(

View file

@ -119,7 +119,7 @@ impl Render for OnboardingBanner {
h_flex() h_flex()
.h_full() .h_full()
.gap_1() .gap_1()
.child(Icon::new(self.details.icon_name).size(IconSize::XSmall)) .child(Icon::new(self.details.icon_name).size(IconSize::Small))
.child( .child(
h_flex() h_flex()
.gap_0p5() .gap_0p5()

View file

@ -275,11 +275,11 @@ impl TitleBar {
let banner = cx.new(|cx| { let banner = cx.new(|cx| {
OnboardingBanner::new( OnboardingBanner::new(
"ACP Onboarding", "Debugger Onboarding",
IconName::Sparkle, IconName::Debug,
"Bring Your Own Agent", "The Debugger",
Some("Introducing:".into()), None,
zed_actions::agent::OpenAcpOnboardingModal.boxed_clone(), zed_actions::debugger::OpenOnboardingModal.boxed_clone(),
cx, cx,
) )
}); });

View file

@ -13,9 +13,6 @@ use crate::prelude::*;
)] )]
#[strum(serialize_all = "snake_case")] #[strum(serialize_all = "snake_case")]
pub enum VectorName { pub enum VectorName {
AcpGrid,
AcpLogo,
AcpLogoSerif,
AiGrid, AiGrid,
DebuggerGrid, DebuggerGrid,
Grid, Grid,

View file

@ -1,8 +1,8 @@
use crate::PlatformStyle; use crate::PlatformStyle;
use crate::{Icon, IconName, IconSize, h_flex, prelude::*}; use crate::{Icon, IconName, IconSize, h_flex, prelude::*};
use gpui::{ use gpui::{
Action, AnyElement, App, FocusHandle, Global, IntoElement, KeybindingKeystroke, Keystroke, Action, AnyElement, App, FocusHandle, Global, IntoElement, Keystroke, Modifiers, Window,
Modifiers, Window, relative, relative,
}; };
use itertools::Itertools; use itertools::Itertools;
@ -13,7 +13,7 @@ pub struct KeyBinding {
/// More than one keystroke produces a chord. /// More than one keystroke produces a chord.
/// ///
/// This should always contain at least one keystroke. /// This should always contain at least one keystroke.
pub keystrokes: Vec<KeybindingKeystroke>, pub keystrokes: Vec<Keystroke>,
/// The [`PlatformStyle`] to use when displaying this keybinding. /// The [`PlatformStyle`] to use when displaying this keybinding.
platform_style: PlatformStyle, platform_style: PlatformStyle,
@ -59,7 +59,7 @@ impl KeyBinding {
cx.try_global::<VimStyle>().is_some_and(|g| g.0) cx.try_global::<VimStyle>().is_some_and(|g| g.0)
} }
pub fn new(keystrokes: Vec<KeybindingKeystroke>, cx: &App) -> Self { pub fn new(keystrokes: Vec<Keystroke>, cx: &App) -> Self {
Self { Self {
keystrokes, keystrokes,
platform_style: PlatformStyle::platform(), platform_style: PlatformStyle::platform(),
@ -99,16 +99,16 @@ impl KeyBinding {
} }
fn render_key( fn render_key(
key: &str, keystroke: &Keystroke,
color: Option<Color>, color: Option<Color>,
platform_style: PlatformStyle, platform_style: PlatformStyle,
size: impl Into<Option<AbsoluteLength>>, size: impl Into<Option<AbsoluteLength>>,
) -> AnyElement { ) -> AnyElement {
let key_icon = icon_for_key(key, platform_style); let key_icon = icon_for_key(keystroke, platform_style);
match key_icon { match key_icon {
Some(icon) => KeyIcon::new(icon, color).size(size).into_any_element(), Some(icon) => KeyIcon::new(icon, color).size(size).into_any_element(),
None => { None => {
let key = util::capitalize(key); let key = util::capitalize(&keystroke.key);
Key::new(&key, color).size(size).into_any_element() Key::new(&key, color).size(size).into_any_element()
} }
} }
@ -124,7 +124,7 @@ impl RenderOnce for KeyBinding {
"KEY_BINDING-{}", "KEY_BINDING-{}",
self.keystrokes self.keystrokes
.iter() .iter()
.map(|k| k.display_key.to_string()) .map(|k| k.key.to_string())
.collect::<Vec<_>>() .collect::<Vec<_>>()
.join(" ") .join(" ")
) )
@ -137,7 +137,7 @@ impl RenderOnce for KeyBinding {
.py_0p5() .py_0p5()
.rounded_xs() .rounded_xs()
.text_color(cx.theme().colors().text_muted) .text_color(cx.theme().colors().text_muted)
.children(render_keybinding_keystroke( .children(render_keystroke(
keystroke, keystroke,
color, color,
self.size, self.size,
@ -148,8 +148,8 @@ impl RenderOnce for KeyBinding {
} }
} }
pub fn render_keybinding_keystroke( pub fn render_keystroke(
keystroke: &KeybindingKeystroke, keystroke: &Keystroke,
color: Option<Color>, color: Option<Color>,
size: impl Into<Option<AbsoluteLength>>, size: impl Into<Option<AbsoluteLength>>,
platform_style: PlatformStyle, platform_style: PlatformStyle,
@ -163,39 +163,26 @@ pub fn render_keybinding_keystroke(
let size = size.into(); let size = size.into();
if use_text { if use_text {
let element = Key::new( let element = Key::new(keystroke_text(keystroke, platform_style, vim_mode), color)
keystroke_text( .size(size)
&keystroke.display_modifiers, .into_any_element();
&keystroke.display_key,
platform_style,
vim_mode,
),
color,
)
.size(size)
.into_any_element();
vec![element] vec![element]
} else { } else {
let mut elements = Vec::new(); let mut elements = Vec::new();
elements.extend(render_modifiers( elements.extend(render_modifiers(
&keystroke.display_modifiers, &keystroke.modifiers,
platform_style, platform_style,
color, color,
size, size,
true, true,
)); ));
elements.push(render_key( elements.push(render_key(keystroke, color, platform_style, size));
&keystroke.display_key,
color,
platform_style,
size,
));
elements elements
} }
} }
fn icon_for_key(key: &str, platform_style: PlatformStyle) -> Option<IconName> { fn icon_for_key(keystroke: &Keystroke, platform_style: PlatformStyle) -> Option<IconName> {
match key { match keystroke.key.as_str() {
"left" => Some(IconName::ArrowLeft), "left" => Some(IconName::ArrowLeft),
"right" => Some(IconName::ArrowRight), "right" => Some(IconName::ArrowRight),
"up" => Some(IconName::ArrowUp), "up" => Some(IconName::ArrowUp),
@ -392,7 +379,7 @@ impl KeyIcon {
/// Returns a textual representation of the key binding for the given [`Action`]. /// Returns a textual representation of the key binding for the given [`Action`].
pub fn text_for_action(action: &dyn Action, window: &Window, cx: &App) -> Option<String> { pub fn text_for_action(action: &dyn Action, window: &Window, cx: &App) -> Option<String> {
let key_binding = window.highest_precedence_binding_for_action(action)?; let key_binding = window.highest_precedence_binding_for_action(action)?;
Some(text_for_keybinding_keystrokes(key_binding.keystrokes(), cx)) Some(text_for_keystrokes(key_binding.keystrokes(), cx))
} }
pub fn text_for_keystrokes(keystrokes: &[Keystroke], cx: &App) -> String { pub fn text_for_keystrokes(keystrokes: &[Keystroke], cx: &App) -> String {
@ -400,50 +387,22 @@ pub fn text_for_keystrokes(keystrokes: &[Keystroke], cx: &App) -> String {
let vim_enabled = cx.try_global::<VimStyle>().is_some(); let vim_enabled = cx.try_global::<VimStyle>().is_some();
keystrokes keystrokes
.iter() .iter()
.map(|keystroke| { .map(|keystroke| keystroke_text(keystroke, platform_style, vim_enabled))
keystroke_text(
&keystroke.modifiers,
&keystroke.key,
platform_style,
vim_enabled,
)
})
.join(" ") .join(" ")
} }
pub fn text_for_keybinding_keystrokes(keystrokes: &[KeybindingKeystroke], cx: &App) -> String { pub fn text_for_keystroke(keystroke: &Keystroke, cx: &App) -> String {
let platform_style = PlatformStyle::platform(); let platform_style = PlatformStyle::platform();
let vim_enabled = cx.try_global::<VimStyle>().is_some(); let vim_enabled = cx.try_global::<VimStyle>().is_some();
keystrokes keystroke_text(keystroke, platform_style, vim_enabled)
.iter()
.map(|keystroke| {
keystroke_text(
&keystroke.display_modifiers,
&keystroke.display_key,
platform_style,
vim_enabled,
)
})
.join(" ")
}
pub fn text_for_keystroke(modifiers: &Modifiers, key: &str, cx: &App) -> String {
let platform_style = PlatformStyle::platform();
let vim_enabled = cx.try_global::<VimStyle>().is_some();
keystroke_text(modifiers, key, platform_style, vim_enabled)
} }
/// Returns a textual representation of the given [`Keystroke`]. /// Returns a textual representation of the given [`Keystroke`].
fn keystroke_text( fn keystroke_text(keystroke: &Keystroke, platform_style: PlatformStyle, vim_mode: bool) -> String {
modifiers: &Modifiers,
key: &str,
platform_style: PlatformStyle,
vim_mode: bool,
) -> String {
let mut text = String::new(); let mut text = String::new();
let delimiter = '-'; let delimiter = '-';
if modifiers.function { if keystroke.modifiers.function {
match vim_mode { match vim_mode {
false => text.push_str("Fn"), false => text.push_str("Fn"),
true => text.push_str("fn"), true => text.push_str("fn"),
@ -452,7 +411,7 @@ fn keystroke_text(
text.push(delimiter); text.push(delimiter);
} }
if modifiers.control { if keystroke.modifiers.control {
match (platform_style, vim_mode) { match (platform_style, vim_mode) {
(PlatformStyle::Mac, false) => text.push_str("Control"), (PlatformStyle::Mac, false) => text.push_str("Control"),
(PlatformStyle::Linux | PlatformStyle::Windows, false) => text.push_str("Ctrl"), (PlatformStyle::Linux | PlatformStyle::Windows, false) => text.push_str("Ctrl"),
@ -462,7 +421,7 @@ fn keystroke_text(
text.push(delimiter); text.push(delimiter);
} }
if modifiers.platform { if keystroke.modifiers.platform {
match (platform_style, vim_mode) { match (platform_style, vim_mode) {
(PlatformStyle::Mac, false) => text.push_str("Command"), (PlatformStyle::Mac, false) => text.push_str("Command"),
(PlatformStyle::Mac, true) => text.push_str("cmd"), (PlatformStyle::Mac, true) => text.push_str("cmd"),
@ -475,7 +434,7 @@ fn keystroke_text(
text.push(delimiter); text.push(delimiter);
} }
if modifiers.alt { if keystroke.modifiers.alt {
match (platform_style, vim_mode) { match (platform_style, vim_mode) {
(PlatformStyle::Mac, false) => text.push_str("Option"), (PlatformStyle::Mac, false) => text.push_str("Option"),
(PlatformStyle::Linux | PlatformStyle::Windows, false) => text.push_str("Alt"), (PlatformStyle::Linux | PlatformStyle::Windows, false) => text.push_str("Alt"),
@ -485,7 +444,7 @@ fn keystroke_text(
text.push(delimiter); text.push(delimiter);
} }
if modifiers.shift { if keystroke.modifiers.shift {
match (platform_style, vim_mode) { match (platform_style, vim_mode) {
(_, false) => text.push_str("Shift"), (_, false) => text.push_str("Shift"),
(_, true) => text.push_str("shift"), (_, true) => text.push_str("shift"),
@ -494,9 +453,9 @@ fn keystroke_text(
} }
if vim_mode { if vim_mode {
text.push_str(key) text.push_str(&keystroke.key)
} else { } else {
let key = match key { let key = match keystroke.key.as_str() {
"pageup" => "PageUp", "pageup" => "PageUp",
"pagedown" => "PageDown", "pagedown" => "PageDown",
key => &util::capitalize(key), key => &util::capitalize(key),
@ -603,11 +562,9 @@ mod tests {
#[test] #[test]
fn test_text_for_keystroke() { fn test_text_for_keystroke() {
let keystroke = Keystroke::parse("cmd-c").unwrap();
assert_eq!( assert_eq!(
keystroke_text( keystroke_text(
&keystroke.modifiers, &Keystroke::parse("cmd-c").unwrap(),
&keystroke.key,
PlatformStyle::Mac, PlatformStyle::Mac,
false false
), ),
@ -615,8 +572,7 @@ mod tests {
); );
assert_eq!( assert_eq!(
keystroke_text( keystroke_text(
&keystroke.modifiers, &Keystroke::parse("cmd-c").unwrap(),
&keystroke.key,
PlatformStyle::Linux, PlatformStyle::Linux,
false false
), ),
@ -624,19 +580,16 @@ mod tests {
); );
assert_eq!( assert_eq!(
keystroke_text( keystroke_text(
&keystroke.modifiers, &Keystroke::parse("cmd-c").unwrap(),
&keystroke.key,
PlatformStyle::Windows, PlatformStyle::Windows,
false false
), ),
"Win-C".to_string() "Win-C".to_string()
); );
let keystroke = Keystroke::parse("ctrl-alt-delete").unwrap();
assert_eq!( assert_eq!(
keystroke_text( keystroke_text(
&keystroke.modifiers, &Keystroke::parse("ctrl-alt-delete").unwrap(),
&keystroke.key,
PlatformStyle::Mac, PlatformStyle::Mac,
false false
), ),
@ -644,8 +597,7 @@ mod tests {
); );
assert_eq!( assert_eq!(
keystroke_text( keystroke_text(
&keystroke.modifiers, &Keystroke::parse("ctrl-alt-delete").unwrap(),
&keystroke.key,
PlatformStyle::Linux, PlatformStyle::Linux,
false false
), ),
@ -653,19 +605,16 @@ mod tests {
); );
assert_eq!( assert_eq!(
keystroke_text( keystroke_text(
&keystroke.modifiers, &Keystroke::parse("ctrl-alt-delete").unwrap(),
&keystroke.key,
PlatformStyle::Windows, PlatformStyle::Windows,
false false
), ),
"Ctrl-Alt-Delete".to_string() "Ctrl-Alt-Delete".to_string()
); );
let keystroke = Keystroke::parse("shift-pageup").unwrap();
assert_eq!( assert_eq!(
keystroke_text( keystroke_text(
&keystroke.modifiers, &Keystroke::parse("shift-pageup").unwrap(),
&keystroke.key,
PlatformStyle::Mac, PlatformStyle::Mac,
false false
), ),
@ -673,8 +622,7 @@ mod tests {
); );
assert_eq!( assert_eq!(
keystroke_text( keystroke_text(
&keystroke.modifiers, &Keystroke::parse("shift-pageup").unwrap(),
&keystroke.key,
PlatformStyle::Linux, PlatformStyle::Linux,
false, false,
), ),
@ -682,8 +630,7 @@ mod tests {
); );
assert_eq!( assert_eq!(
keystroke_text( keystroke_text(
&keystroke.modifiers, &Keystroke::parse("shift-pageup").unwrap(),
&keystroke.key,
PlatformStyle::Windows, PlatformStyle::Windows,
false false
), ),

View file

@ -23,8 +23,6 @@ actions!(
HelixInsert, HelixInsert,
/// Appends at the end of the selection. /// Appends at the end of the selection.
HelixAppend, HelixAppend,
/// Goes to the location of the last modification.
HelixGotoLastModification,
] ]
); );
@ -33,7 +31,6 @@ pub fn register(editor: &mut Editor, cx: &mut Context<Vim>) {
Vim::action(editor, cx, Vim::helix_insert); Vim::action(editor, cx, Vim::helix_insert);
Vim::action(editor, cx, Vim::helix_append); Vim::action(editor, cx, Vim::helix_append);
Vim::action(editor, cx, Vim::helix_yank); Vim::action(editor, cx, Vim::helix_yank);
Vim::action(editor, cx, Vim::helix_goto_last_modification);
} }
impl Vim { impl Vim {
@ -433,15 +430,6 @@ impl Vim {
}); });
self.switch_mode(Mode::HelixNormal, true, window, cx); self.switch_mode(Mode::HelixNormal, true, window, cx);
} }
pub fn helix_goto_last_modification(
&mut self,
_: &HelixGotoLastModification,
window: &mut Window,
cx: &mut Context<Self>,
) {
self.jump(".".into(), false, false, window, cx);
}
} }
#[cfg(test)] #[cfg(test)]
@ -453,7 +441,6 @@ mod test {
#[gpui::test] #[gpui::test]
async fn test_word_motions(cx: &mut gpui::TestAppContext) { async fn test_word_motions(cx: &mut gpui::TestAppContext) {
let mut cx = VimTestContext::new(cx, true).await; let mut cx = VimTestContext::new(cx, true).await;
cx.enable_helix();
// « // «
// ˇ // ˇ
// » // »
@ -515,7 +502,6 @@ mod test {
#[gpui::test] #[gpui::test]
async fn test_delete(cx: &mut gpui::TestAppContext) { async fn test_delete(cx: &mut gpui::TestAppContext) {
let mut cx = VimTestContext::new(cx, true).await; let mut cx = VimTestContext::new(cx, true).await;
cx.enable_helix();
// test delete a selection // test delete a selection
cx.set_state( cx.set_state(
@ -596,7 +582,6 @@ mod test {
#[gpui::test] #[gpui::test]
async fn test_f_and_t(cx: &mut gpui::TestAppContext) { async fn test_f_and_t(cx: &mut gpui::TestAppContext) {
let mut cx = VimTestContext::new(cx, true).await; let mut cx = VimTestContext::new(cx, true).await;
cx.enable_helix();
cx.set_state( cx.set_state(
indoc! {" indoc! {"
@ -650,7 +635,6 @@ mod test {
#[gpui::test] #[gpui::test]
async fn test_newline_char(cx: &mut gpui::TestAppContext) { async fn test_newline_char(cx: &mut gpui::TestAppContext) {
let mut cx = VimTestContext::new(cx, true).await; let mut cx = VimTestContext::new(cx, true).await;
cx.enable_helix();
cx.set_state("aa«\nˇ»bb cc", Mode::HelixNormal); cx.set_state("aa«\nˇ»bb cc", Mode::HelixNormal);
@ -668,7 +652,6 @@ mod test {
#[gpui::test] #[gpui::test]
async fn test_insert_selected(cx: &mut gpui::TestAppContext) { async fn test_insert_selected(cx: &mut gpui::TestAppContext) {
let mut cx = VimTestContext::new(cx, true).await; let mut cx = VimTestContext::new(cx, true).await;
cx.enable_helix();
cx.set_state( cx.set_state(
indoc! {" indoc! {"
«The ˇ»quick brown «The ˇ»quick brown
@ -691,7 +674,6 @@ mod test {
#[gpui::test] #[gpui::test]
async fn test_append(cx: &mut gpui::TestAppContext) { async fn test_append(cx: &mut gpui::TestAppContext) {
let mut cx = VimTestContext::new(cx, true).await; let mut cx = VimTestContext::new(cx, true).await;
cx.enable_helix();
// test from the end of the selection // test from the end of the selection
cx.set_state( cx.set_state(
indoc! {" indoc! {"
@ -734,7 +716,6 @@ mod test {
#[gpui::test] #[gpui::test]
async fn test_replace(cx: &mut gpui::TestAppContext) { async fn test_replace(cx: &mut gpui::TestAppContext) {
let mut cx = VimTestContext::new(cx, true).await; let mut cx = VimTestContext::new(cx, true).await;
cx.enable_helix();
// No selection (single character) // No selection (single character)
cx.set_state("ˇaa", Mode::HelixNormal); cx.set_state("ˇaa", Mode::HelixNormal);
@ -782,72 +763,4 @@ mod test {
cx.shared_clipboard().assert_eq("worl"); cx.shared_clipboard().assert_eq("worl");
cx.assert_state("hello «worlˇ»d", Mode::HelixNormal); cx.assert_state("hello «worlˇ»d", Mode::HelixNormal);
} }
#[gpui::test]
async fn test_shift_r_paste(cx: &mut gpui::TestAppContext) {
let mut cx = VimTestContext::new(cx, true).await;
cx.enable_helix();
// First copy some text to clipboard
cx.set_state("«hello worldˇ»", Mode::HelixNormal);
cx.simulate_keystrokes("y");
// Test paste with shift-r on single cursor
cx.set_state("foo ˇbar", Mode::HelixNormal);
cx.simulate_keystrokes("shift-r");
cx.assert_state("foo hello worldˇbar", Mode::HelixNormal);
// Test paste with shift-r on selection
cx.set_state("foo «barˇ» baz", Mode::HelixNormal);
cx.simulate_keystrokes("shift-r");
cx.assert_state("foo hello worldˇ baz", Mode::HelixNormal);
}
#[gpui::test]
async fn test_insert_mode_stickiness(cx: &mut gpui::TestAppContext) {
let mut cx = VimTestContext::new(cx, true).await;
cx.enable_helix();
// Make a modification at a specific location
cx.set_state("ˇhello", Mode::HelixNormal);
assert_eq!(cx.mode(), Mode::HelixNormal);
cx.simulate_keystrokes("i");
assert_eq!(cx.mode(), Mode::Insert);
cx.simulate_keystrokes("escape");
assert_eq!(cx.mode(), Mode::HelixNormal);
}
#[gpui::test]
async fn test_goto_last_modification(cx: &mut gpui::TestAppContext) {
let mut cx = VimTestContext::new(cx, true).await;
cx.enable_helix();
// Make a modification at a specific location
cx.set_state("line one\nline ˇtwo\nline three", Mode::HelixNormal);
cx.assert_state("line one\nline ˇtwo\nline three", Mode::HelixNormal);
cx.simulate_keystrokes("i");
cx.simulate_keystrokes("escape");
cx.simulate_keystrokes("i");
cx.simulate_keystrokes("m o d i f i e d space");
cx.simulate_keystrokes("escape");
// TODO: this fails, because state is no longer helix
cx.assert_state(
"line one\nline modified ˇtwo\nline three",
Mode::HelixNormal,
);
// Move cursor away from the modification
cx.simulate_keystrokes("up");
// Use "g ." to go back to last modification
cx.simulate_keystrokes("g .");
// Verify we're back at the modification location and still in HelixNormal mode
cx.assert_state(
"line one\nline modifiedˇ two\nline three",
Mode::HelixNormal,
);
}
} }

View file

@ -203,10 +203,7 @@ impl Vim {
// hook into the existing to clear out any vim search state on cmd+f or edit -> find. // hook into the existing to clear out any vim search state on cmd+f or edit -> find.
fn search_deploy(&mut self, _: &buffer_search::Deploy, _: &mut Window, cx: &mut Context<Self>) { fn search_deploy(&mut self, _: &buffer_search::Deploy, _: &mut Window, cx: &mut Context<Self>) {
// Preserve the current mode when resetting search state
let current_mode = self.mode;
self.search = Default::default(); self.search = Default::default();
self.search.prior_mode = current_mode;
cx.propagate(); cx.propagate();
} }

View file

@ -7,10 +7,8 @@ use crate::{motion::Motion, object::Object};
use anyhow::Result; use anyhow::Result;
use collections::HashMap; use collections::HashMap;
use command_palette_hooks::{CommandPaletteFilter, CommandPaletteInterceptor}; use command_palette_hooks::{CommandPaletteFilter, CommandPaletteInterceptor};
use db::{ use db::define_connection;
sqlez::{domain::Domain, thread_safe_connection::ThreadSafeConnection}, use db::sqlez_macros::sql;
sqlez_macros::sql,
};
use editor::display_map::{is_invisible, replacement}; use editor::display_map::{is_invisible, replacement};
use editor::{Anchor, ClipboardSelection, Editor, MultiBuffer, ToPoint as EditorToPoint}; use editor::{Anchor, ClipboardSelection, Editor, MultiBuffer, ToPoint as EditorToPoint};
use gpui::{ use gpui::{
@ -1670,12 +1668,8 @@ impl MarksView {
} }
} }
pub struct VimDb(ThreadSafeConnection); define_connection! (
pub static ref DB: VimDb<WorkspaceDb> = &[
impl Domain for VimDb {
const NAME: &str = stringify!(VimDb);
const MIGRATIONS: &[&str] = &[
sql! ( sql! (
CREATE TABLE vim_marks ( CREATE TABLE vim_marks (
workspace_id INTEGER, workspace_id INTEGER,
@ -1695,9 +1689,7 @@ impl Domain for VimDb {
ON vim_global_marks_paths(workspace_id, mark_name); ON vim_global_marks_paths(workspace_id, mark_name);
), ),
]; ];
} );
db::static_connection!(DB, VimDb, [WorkspaceDb]);
struct SerializedMark { struct SerializedMark {
path: Arc<Path>, path: Arc<Path>,

View file

@ -58,7 +58,11 @@ impl PathList {
let mut paths: Vec<PathBuf> = if serialized.paths.is_empty() { let mut paths: Vec<PathBuf> = if serialized.paths.is_empty() {
Vec::new() Vec::new()
} else { } else {
serialized.paths.split('\n').map(PathBuf::from).collect() serde_json::from_str::<Vec<PathBuf>>(&serialized.paths)
.unwrap_or(Vec::new())
.into_iter()
.map(|s| SanitizedPath::from(s).into())
.collect()
}; };
let mut order: Vec<usize> = serialized let mut order: Vec<usize> = serialized
@ -81,13 +85,7 @@ impl PathList {
pub fn serialize(&self) -> SerializedPathList { pub fn serialize(&self) -> SerializedPathList {
use std::fmt::Write as _; use std::fmt::Write as _;
let mut paths = String::new(); let paths = serde_json::to_string(&self.paths).unwrap_or_default();
for path in self.paths.iter() {
if !paths.is_empty() {
paths.push('\n');
}
paths.push_str(&path.to_string_lossy());
}
let mut order = String::new(); let mut order = String::new();
for ix in self.order.iter() { for ix in self.order.iter() {

View file

@ -10,11 +10,7 @@ use std::{
use anyhow::{Context as _, Result, bail}; use anyhow::{Context as _, Result, bail};
use collections::HashMap; use collections::HashMap;
use db::{ use db::{define_connection, query, sqlez::connection::Connection, sqlez_macros::sql};
query,
sqlez::{connection::Connection, domain::Domain},
sqlez_macros::sql,
};
use gpui::{Axis, Bounds, Task, WindowBounds, WindowId, point, size}; use gpui::{Axis, Bounds, Task, WindowBounds, WindowId, point, size};
use project::debugger::breakpoint_store::{BreakpointState, SourceBreakpoint}; use project::debugger::breakpoint_store::{BreakpointState, SourceBreakpoint};
@ -279,189 +275,186 @@ impl sqlez::bindable::Bind for SerializedPixels {
} }
} }
pub struct WorkspaceDb(ThreadSafeConnection); define_connection! {
pub static ref DB: WorkspaceDb<()> =
&[
sql!(
CREATE TABLE workspaces(
workspace_id INTEGER PRIMARY KEY,
workspace_location BLOB UNIQUE,
dock_visible INTEGER, // Deprecated. Preserving so users can downgrade Zed.
dock_anchor TEXT, // Deprecated. Preserving so users can downgrade Zed.
dock_pane INTEGER, // Deprecated. Preserving so users can downgrade Zed.
left_sidebar_open INTEGER, // Boolean
timestamp TEXT DEFAULT CURRENT_TIMESTAMP NOT NULL,
FOREIGN KEY(dock_pane) REFERENCES panes(pane_id)
) STRICT;
impl Domain for WorkspaceDb { CREATE TABLE pane_groups(
const NAME: &str = stringify!(WorkspaceDb); group_id INTEGER PRIMARY KEY,
workspace_id INTEGER NOT NULL,
parent_group_id INTEGER, // NULL indicates that this is a root node
position INTEGER, // NULL indicates that this is a root node
axis TEXT NOT NULL, // Enum: 'Vertical' / 'Horizontal'
FOREIGN KEY(workspace_id) REFERENCES workspaces(workspace_id)
ON DELETE CASCADE
ON UPDATE CASCADE,
FOREIGN KEY(parent_group_id) REFERENCES pane_groups(group_id) ON DELETE CASCADE
) STRICT;
const MIGRATIONS: &[&str] = &[ CREATE TABLE panes(
sql!( pane_id INTEGER PRIMARY KEY,
CREATE TABLE workspaces( workspace_id INTEGER NOT NULL,
workspace_id INTEGER PRIMARY KEY, active INTEGER NOT NULL, // Boolean
workspace_location BLOB UNIQUE, FOREIGN KEY(workspace_id) REFERENCES workspaces(workspace_id)
dock_visible INTEGER, // Deprecated. Preserving so users can downgrade Zed. ON DELETE CASCADE
dock_anchor TEXT, // Deprecated. Preserving so users can downgrade Zed. ON UPDATE CASCADE
dock_pane INTEGER, // Deprecated. Preserving so users can downgrade Zed. ) STRICT;
left_sidebar_open INTEGER, // Boolean
timestamp TEXT DEFAULT CURRENT_TIMESTAMP NOT NULL,
FOREIGN KEY(dock_pane) REFERENCES panes(pane_id)
) STRICT;
CREATE TABLE pane_groups( CREATE TABLE center_panes(
group_id INTEGER PRIMARY KEY, pane_id INTEGER PRIMARY KEY,
workspace_id INTEGER NOT NULL, parent_group_id INTEGER, // NULL means that this is a root pane
parent_group_id INTEGER, // NULL indicates that this is a root node position INTEGER, // NULL means that this is a root pane
position INTEGER, // NULL indicates that this is a root node FOREIGN KEY(pane_id) REFERENCES panes(pane_id)
axis TEXT NOT NULL, // Enum: 'Vertical' / 'Horizontal' ON DELETE CASCADE,
FOREIGN KEY(workspace_id) REFERENCES workspaces(workspace_id) FOREIGN KEY(parent_group_id) REFERENCES pane_groups(group_id) ON DELETE CASCADE
ON DELETE CASCADE ) STRICT;
ON UPDATE CASCADE,
FOREIGN KEY(parent_group_id) REFERENCES pane_groups(group_id) ON DELETE CASCADE
) STRICT;
CREATE TABLE panes( CREATE TABLE items(
pane_id INTEGER PRIMARY KEY, item_id INTEGER NOT NULL, // This is the item's view id, so this is not unique
workspace_id INTEGER NOT NULL, workspace_id INTEGER NOT NULL,
active INTEGER NOT NULL, // Boolean pane_id INTEGER NOT NULL,
FOREIGN KEY(workspace_id) REFERENCES workspaces(workspace_id) kind TEXT NOT NULL,
ON DELETE CASCADE position INTEGER NOT NULL,
ON UPDATE CASCADE active INTEGER NOT NULL,
) STRICT; FOREIGN KEY(workspace_id) REFERENCES workspaces(workspace_id)
ON DELETE CASCADE
CREATE TABLE center_panes( ON UPDATE CASCADE,
pane_id INTEGER PRIMARY KEY, FOREIGN KEY(pane_id) REFERENCES panes(pane_id)
parent_group_id INTEGER, // NULL means that this is a root pane ON DELETE CASCADE,
position INTEGER, // NULL means that this is a root pane PRIMARY KEY(item_id, workspace_id)
FOREIGN KEY(pane_id) REFERENCES panes(pane_id) ) STRICT;
ON DELETE CASCADE, ),
FOREIGN KEY(parent_group_id) REFERENCES pane_groups(group_id) ON DELETE CASCADE sql!(
) STRICT; ALTER TABLE workspaces ADD COLUMN window_state TEXT;
ALTER TABLE workspaces ADD COLUMN window_x REAL;
CREATE TABLE items( ALTER TABLE workspaces ADD COLUMN window_y REAL;
item_id INTEGER NOT NULL, // This is the item's view id, so this is not unique ALTER TABLE workspaces ADD COLUMN window_width REAL;
workspace_id INTEGER NOT NULL, ALTER TABLE workspaces ADD COLUMN window_height REAL;
pane_id INTEGER NOT NULL, ALTER TABLE workspaces ADD COLUMN display BLOB;
kind TEXT NOT NULL, ),
position INTEGER NOT NULL, // Drop foreign key constraint from workspaces.dock_pane to panes table.
active INTEGER NOT NULL, sql!(
FOREIGN KEY(workspace_id) REFERENCES workspaces(workspace_id) CREATE TABLE workspaces_2(
ON DELETE CASCADE workspace_id INTEGER PRIMARY KEY,
ON UPDATE CASCADE, workspace_location BLOB UNIQUE,
FOREIGN KEY(pane_id) REFERENCES panes(pane_id) dock_visible INTEGER, // Deprecated. Preserving so users can downgrade Zed.
ON DELETE CASCADE, dock_anchor TEXT, // Deprecated. Preserving so users can downgrade Zed.
PRIMARY KEY(item_id, workspace_id) dock_pane INTEGER, // Deprecated. Preserving so users can downgrade Zed.
) STRICT; left_sidebar_open INTEGER, // Boolean
), timestamp TEXT DEFAULT CURRENT_TIMESTAMP NOT NULL,
sql!( window_state TEXT,
ALTER TABLE workspaces ADD COLUMN window_state TEXT; window_x REAL,
ALTER TABLE workspaces ADD COLUMN window_x REAL; window_y REAL,
ALTER TABLE workspaces ADD COLUMN window_y REAL; window_width REAL,
ALTER TABLE workspaces ADD COLUMN window_width REAL; window_height REAL,
ALTER TABLE workspaces ADD COLUMN window_height REAL; display BLOB
ALTER TABLE workspaces ADD COLUMN display BLOB; ) STRICT;
), INSERT INTO workspaces_2 SELECT * FROM workspaces;
// Drop foreign key constraint from workspaces.dock_pane to panes table. DROP TABLE workspaces;
sql!( ALTER TABLE workspaces_2 RENAME TO workspaces;
CREATE TABLE workspaces_2( ),
workspace_id INTEGER PRIMARY KEY, // Add panels related information
workspace_location BLOB UNIQUE, sql!(
dock_visible INTEGER, // Deprecated. Preserving so users can downgrade Zed. ALTER TABLE workspaces ADD COLUMN left_dock_visible INTEGER; //bool
dock_anchor TEXT, // Deprecated. Preserving so users can downgrade Zed. ALTER TABLE workspaces ADD COLUMN left_dock_active_panel TEXT;
dock_pane INTEGER, // Deprecated. Preserving so users can downgrade Zed. ALTER TABLE workspaces ADD COLUMN right_dock_visible INTEGER; //bool
left_sidebar_open INTEGER, // Boolean ALTER TABLE workspaces ADD COLUMN right_dock_active_panel TEXT;
timestamp TEXT DEFAULT CURRENT_TIMESTAMP NOT NULL, ALTER TABLE workspaces ADD COLUMN bottom_dock_visible INTEGER; //bool
window_state TEXT, ALTER TABLE workspaces ADD COLUMN bottom_dock_active_panel TEXT;
window_x REAL, ),
window_y REAL, // Add panel zoom persistence
window_width REAL, sql!(
window_height REAL, ALTER TABLE workspaces ADD COLUMN left_dock_zoom INTEGER; //bool
display BLOB ALTER TABLE workspaces ADD COLUMN right_dock_zoom INTEGER; //bool
) STRICT; ALTER TABLE workspaces ADD COLUMN bottom_dock_zoom INTEGER; //bool
INSERT INTO workspaces_2 SELECT * FROM workspaces; ),
DROP TABLE workspaces; // Add pane group flex data
ALTER TABLE workspaces_2 RENAME TO workspaces; sql!(
), ALTER TABLE pane_groups ADD COLUMN flexes TEXT;
// Add panels related information ),
sql!( // Add fullscreen field to workspace
ALTER TABLE workspaces ADD COLUMN left_dock_visible INTEGER; //bool // Deprecated, `WindowBounds` holds the fullscreen state now.
ALTER TABLE workspaces ADD COLUMN left_dock_active_panel TEXT; // Preserving so users can downgrade Zed.
ALTER TABLE workspaces ADD COLUMN right_dock_visible INTEGER; //bool sql!(
ALTER TABLE workspaces ADD COLUMN right_dock_active_panel TEXT; ALTER TABLE workspaces ADD COLUMN fullscreen INTEGER; //bool
ALTER TABLE workspaces ADD COLUMN bottom_dock_visible INTEGER; //bool ),
ALTER TABLE workspaces ADD COLUMN bottom_dock_active_panel TEXT; // Add preview field to items
), sql!(
// Add panel zoom persistence ALTER TABLE items ADD COLUMN preview INTEGER; //bool
sql!( ),
ALTER TABLE workspaces ADD COLUMN left_dock_zoom INTEGER; //bool // Add centered_layout field to workspace
ALTER TABLE workspaces ADD COLUMN right_dock_zoom INTEGER; //bool sql!(
ALTER TABLE workspaces ADD COLUMN bottom_dock_zoom INTEGER; //bool ALTER TABLE workspaces ADD COLUMN centered_layout INTEGER; //bool
), ),
// Add pane group flex data sql!(
sql!( CREATE TABLE remote_projects (
ALTER TABLE pane_groups ADD COLUMN flexes TEXT; remote_project_id INTEGER NOT NULL UNIQUE,
), path TEXT,
// Add fullscreen field to workspace dev_server_name TEXT
// Deprecated, `WindowBounds` holds the fullscreen state now. );
// Preserving so users can downgrade Zed. ALTER TABLE workspaces ADD COLUMN remote_project_id INTEGER;
sql!( ALTER TABLE workspaces RENAME COLUMN workspace_location TO local_paths;
ALTER TABLE workspaces ADD COLUMN fullscreen INTEGER; //bool ),
), sql!(
// Add preview field to items DROP TABLE remote_projects;
sql!( CREATE TABLE dev_server_projects (
ALTER TABLE items ADD COLUMN preview INTEGER; //bool id INTEGER NOT NULL UNIQUE,
), path TEXT,
// Add centered_layout field to workspace dev_server_name TEXT
sql!( );
ALTER TABLE workspaces ADD COLUMN centered_layout INTEGER; //bool ALTER TABLE workspaces DROP COLUMN remote_project_id;
), ALTER TABLE workspaces ADD COLUMN dev_server_project_id INTEGER;
sql!( ),
CREATE TABLE remote_projects ( sql!(
remote_project_id INTEGER NOT NULL UNIQUE, ALTER TABLE workspaces ADD COLUMN local_paths_order BLOB;
path TEXT, ),
dev_server_name TEXT sql!(
); ALTER TABLE workspaces ADD COLUMN session_id TEXT DEFAULT NULL;
ALTER TABLE workspaces ADD COLUMN remote_project_id INTEGER; ),
ALTER TABLE workspaces RENAME COLUMN workspace_location TO local_paths; sql!(
), ALTER TABLE workspaces ADD COLUMN window_id INTEGER DEFAULT NULL;
sql!( ),
DROP TABLE remote_projects; sql!(
CREATE TABLE dev_server_projects ( ALTER TABLE panes ADD COLUMN pinned_count INTEGER DEFAULT 0;
id INTEGER NOT NULL UNIQUE, ),
path TEXT, sql!(
dev_server_name TEXT CREATE TABLE ssh_projects (
); id INTEGER PRIMARY KEY,
ALTER TABLE workspaces DROP COLUMN remote_project_id; host TEXT NOT NULL,
ALTER TABLE workspaces ADD COLUMN dev_server_project_id INTEGER; port INTEGER,
), path TEXT NOT NULL,
sql!( user TEXT
ALTER TABLE workspaces ADD COLUMN local_paths_order BLOB; );
), ALTER TABLE workspaces ADD COLUMN ssh_project_id INTEGER REFERENCES ssh_projects(id) ON DELETE CASCADE;
sql!( ),
ALTER TABLE workspaces ADD COLUMN session_id TEXT DEFAULT NULL; sql!(
), ALTER TABLE ssh_projects RENAME COLUMN path TO paths;
sql!( ),
ALTER TABLE workspaces ADD COLUMN window_id INTEGER DEFAULT NULL; sql!(
), CREATE TABLE toolchains (
sql!( workspace_id INTEGER,
ALTER TABLE panes ADD COLUMN pinned_count INTEGER DEFAULT 0; worktree_id INTEGER,
), language_name TEXT NOT NULL,
sql!( name TEXT NOT NULL,
CREATE TABLE ssh_projects ( path TEXT NOT NULL,
id INTEGER PRIMARY KEY, PRIMARY KEY (workspace_id, worktree_id, language_name)
host TEXT NOT NULL, );
port INTEGER, ),
path TEXT NOT NULL, sql!(
user TEXT ALTER TABLE toolchains ADD COLUMN raw_json TEXT DEFAULT "{}";
); ),
ALTER TABLE workspaces ADD COLUMN ssh_project_id INTEGER REFERENCES ssh_projects(id) ON DELETE CASCADE; sql!(
),
sql!(
ALTER TABLE ssh_projects RENAME COLUMN path TO paths;
),
sql!(
CREATE TABLE toolchains (
workspace_id INTEGER,
worktree_id INTEGER,
language_name TEXT NOT NULL,
name TEXT NOT NULL,
path TEXT NOT NULL,
PRIMARY KEY (workspace_id, worktree_id, language_name)
);
),
sql!(
ALTER TABLE toolchains ADD COLUMN raw_json TEXT DEFAULT "{}";
),
sql!(
CREATE TABLE breakpoints ( CREATE TABLE breakpoints (
workspace_id INTEGER NOT NULL, workspace_id INTEGER NOT NULL,
path TEXT NOT NULL, path TEXT NOT NULL,
@ -473,172 +466,141 @@ impl Domain for WorkspaceDb {
ON UPDATE CASCADE ON UPDATE CASCADE
); );
), ),
sql!( sql!(
ALTER TABLE workspaces ADD COLUMN local_paths_array TEXT; ALTER TABLE workspaces ADD COLUMN local_paths_array TEXT;
CREATE UNIQUE INDEX local_paths_array_uq ON workspaces(local_paths_array); CREATE UNIQUE INDEX local_paths_array_uq ON workspaces(local_paths_array);
ALTER TABLE workspaces ADD COLUMN local_paths_order_array TEXT; ALTER TABLE workspaces ADD COLUMN local_paths_order_array TEXT;
), ),
sql!( sql!(
ALTER TABLE breakpoints ADD COLUMN state INTEGER DEFAULT(0) NOT NULL ALTER TABLE breakpoints ADD COLUMN state INTEGER DEFAULT(0) NOT NULL
), ),
sql!( sql!(
ALTER TABLE breakpoints DROP COLUMN kind ALTER TABLE breakpoints DROP COLUMN kind
), ),
sql!(ALTER TABLE toolchains ADD COLUMN relative_worktree_path TEXT DEFAULT "" NOT NULL), sql!(ALTER TABLE toolchains ADD COLUMN relative_worktree_path TEXT DEFAULT "" NOT NULL),
sql!( sql!(
ALTER TABLE breakpoints ADD COLUMN condition TEXT; ALTER TABLE breakpoints ADD COLUMN condition TEXT;
ALTER TABLE breakpoints ADD COLUMN hit_condition TEXT; ALTER TABLE breakpoints ADD COLUMN hit_condition TEXT;
), ),
sql!(CREATE TABLE toolchains2 ( sql!(CREATE TABLE toolchains2 (
workspace_id INTEGER, workspace_id INTEGER,
worktree_id INTEGER, worktree_id INTEGER,
language_name TEXT NOT NULL, language_name TEXT NOT NULL,
name TEXT NOT NULL, name TEXT NOT NULL,
path TEXT NOT NULL, path TEXT NOT NULL,
raw_json TEXT NOT NULL, raw_json TEXT NOT NULL,
relative_worktree_path TEXT NOT NULL, relative_worktree_path TEXT NOT NULL,
PRIMARY KEY (workspace_id, worktree_id, language_name, relative_worktree_path)) STRICT; PRIMARY KEY (workspace_id, worktree_id, language_name, relative_worktree_path)) STRICT;
INSERT INTO toolchains2 INSERT INTO toolchains2
SELECT * FROM toolchains; SELECT * FROM toolchains;
DROP TABLE toolchains; DROP TABLE toolchains;
ALTER TABLE toolchains2 RENAME TO toolchains; ALTER TABLE toolchains2 RENAME TO toolchains;
), ),
sql!( sql!(
CREATE TABLE ssh_connections ( CREATE TABLE ssh_connections (
id INTEGER PRIMARY KEY, id INTEGER PRIMARY KEY,
host TEXT NOT NULL, host TEXT NOT NULL,
port INTEGER, port INTEGER,
user TEXT user TEXT
); );
INSERT INTO ssh_connections (host, port, user) INSERT INTO ssh_connections (host, port, user)
SELECT DISTINCT host, port, user SELECT DISTINCT host, port, user
FROM ssh_projects; FROM ssh_projects;
CREATE TABLE workspaces_2( CREATE TABLE workspaces_2(
workspace_id INTEGER PRIMARY KEY, workspace_id INTEGER PRIMARY KEY,
paths TEXT, paths TEXT,
paths_order TEXT, paths_order TEXT,
ssh_connection_id INTEGER REFERENCES ssh_connections(id), ssh_connection_id INTEGER REFERENCES ssh_connections(id),
timestamp TEXT DEFAULT CURRENT_TIMESTAMP NOT NULL, timestamp TEXT DEFAULT CURRENT_TIMESTAMP NOT NULL,
window_state TEXT, window_state TEXT,
window_x REAL, window_x REAL,
window_y REAL, window_y REAL,
window_width REAL, window_width REAL,
window_height REAL, window_height REAL,
display BLOB, display BLOB,
left_dock_visible INTEGER, left_dock_visible INTEGER,
left_dock_active_panel TEXT, left_dock_active_panel TEXT,
right_dock_visible INTEGER, right_dock_visible INTEGER,
right_dock_active_panel TEXT, right_dock_active_panel TEXT,
bottom_dock_visible INTEGER, bottom_dock_visible INTEGER,
bottom_dock_active_panel TEXT, bottom_dock_active_panel TEXT,
left_dock_zoom INTEGER, left_dock_zoom INTEGER,
right_dock_zoom INTEGER, right_dock_zoom INTEGER,
bottom_dock_zoom INTEGER, bottom_dock_zoom INTEGER,
fullscreen INTEGER, fullscreen INTEGER,
centered_layout INTEGER, centered_layout INTEGER,
session_id TEXT, session_id TEXT,
window_id INTEGER window_id INTEGER
) STRICT; ) STRICT;
INSERT INSERT
INTO workspaces_2 INTO workspaces_2
SELECT SELECT
workspaces.workspace_id, workspaces.workspace_id,
CASE CASE
WHEN ssh_projects.id IS NOT NULL THEN ssh_projects.paths WHEN ssh_projects.id IS NOT NULL THEN ssh_projects.paths
ELSE
CASE
WHEN workspaces.local_paths_array IS NULL OR workspaces.local_paths_array = "" THEN
NULL
ELSE
replace(workspaces.local_paths_array, ',', CHAR(10))
END
END as paths,
CASE
WHEN ssh_projects.id IS NOT NULL THEN ""
ELSE workspaces.local_paths_order_array
END as paths_order,
CASE
WHEN ssh_projects.id IS NOT NULL THEN (
SELECT ssh_connections.id
FROM ssh_connections
WHERE
ssh_connections.host IS ssh_projects.host AND
ssh_connections.port IS ssh_projects.port AND
ssh_connections.user IS ssh_projects.user
)
ELSE NULL
END as ssh_connection_id,
workspaces.timestamp,
workspaces.window_state,
workspaces.window_x,
workspaces.window_y,
workspaces.window_width,
workspaces.window_height,
workspaces.display,
workspaces.left_dock_visible,
workspaces.left_dock_active_panel,
workspaces.right_dock_visible,
workspaces.right_dock_active_panel,
workspaces.bottom_dock_visible,
workspaces.bottom_dock_active_panel,
workspaces.left_dock_zoom,
workspaces.right_dock_zoom,
workspaces.bottom_dock_zoom,
workspaces.fullscreen,
workspaces.centered_layout,
workspaces.session_id,
workspaces.window_id
FROM
workspaces LEFT JOIN
ssh_projects ON
workspaces.ssh_project_id = ssh_projects.id;
DELETE FROM workspaces_2
WHERE workspace_id NOT IN (
SELECT MAX(workspace_id)
FROM workspaces_2
GROUP BY ssh_connection_id, paths
);
DROP TABLE ssh_projects;
DROP TABLE workspaces;
ALTER TABLE workspaces_2 RENAME TO workspaces;
CREATE UNIQUE INDEX ix_workspaces_location ON workspaces(ssh_connection_id, paths);
),
// Fix any data from when workspaces.paths were briefly encoded as JSON arrays
sql!(
UPDATE workspaces
SET paths = CASE
WHEN substr(paths, 1, 2) = '[' || '"' AND substr(paths, -2, 2) = '"' || ']' THEN
replace(
substr(paths, 3, length(paths) - 4),
'"' || ',' || '"',
CHAR(10)
)
ELSE ELSE
replace(paths, ',', CHAR(10)) CASE
END WHEN workspaces.local_paths_array IS NULL OR workspaces.local_paths_array = "" THEN
WHERE paths IS NOT NULL NULL
), ELSE
json('[' || '"' || replace(workspaces.local_paths_array, ',', '"' || "," || '"') || '"' || ']')
END
END as paths,
CASE
WHEN ssh_projects.id IS NOT NULL THEN ""
ELSE workspaces.local_paths_order_array
END as paths_order,
CASE
WHEN ssh_projects.id IS NOT NULL THEN (
SELECT ssh_connections.id
FROM ssh_connections
WHERE
ssh_connections.host IS ssh_projects.host AND
ssh_connections.port IS ssh_projects.port AND
ssh_connections.user IS ssh_projects.user
)
ELSE NULL
END as ssh_connection_id,
workspaces.timestamp,
workspaces.window_state,
workspaces.window_x,
workspaces.window_y,
workspaces.window_width,
workspaces.window_height,
workspaces.display,
workspaces.left_dock_visible,
workspaces.left_dock_active_panel,
workspaces.right_dock_visible,
workspaces.right_dock_active_panel,
workspaces.bottom_dock_visible,
workspaces.bottom_dock_active_panel,
workspaces.left_dock_zoom,
workspaces.right_dock_zoom,
workspaces.bottom_dock_zoom,
workspaces.fullscreen,
workspaces.centered_layout,
workspaces.session_id,
workspaces.window_id
FROM
workspaces LEFT JOIN
ssh_projects ON
workspaces.ssh_project_id = ssh_projects.id;
DROP TABLE ssh_projects;
DROP TABLE workspaces;
ALTER TABLE workspaces_2 RENAME TO workspaces;
CREATE UNIQUE INDEX ix_workspaces_location ON workspaces(ssh_connection_id, paths);
),
]; ];
// Allow recovering from bad migration that was initially shipped to nightly
// when introducing the ssh_connections table.
fn should_allow_migration_change(_index: usize, old: &str, new: &str) -> bool {
old.starts_with("CREATE TABLE ssh_connections")
&& new.starts_with("CREATE TABLE ssh_connections")
}
} }
db::static_connection!(DB, WorkspaceDb, []);
impl WorkspaceDb { impl WorkspaceDb {
/// Returns a serialized workspace for the given worktree_roots. If the passed array /// Returns a serialized workspace for the given worktree_roots. If the passed array
/// is empty, the most recent workspace is returned instead. If no workspace for the /// is empty, the most recent workspace is returned instead. If no workspace for the
@ -1841,7 +1803,6 @@ mod tests {
ON DELETE CASCADE ON DELETE CASCADE
) STRICT; ) STRICT;
)], )],
|_, _, _| false,
) )
.unwrap(); .unwrap();
}) })
@ -1890,7 +1851,6 @@ mod tests {
REFERENCES workspaces(workspace_id) REFERENCES workspaces(workspace_id)
ON DELETE CASCADE ON DELETE CASCADE
) STRICT;)], ) STRICT;)],
|_, _, _| false,
) )
}) })
.await .await

View file

@ -1308,11 +1308,11 @@ pub fn handle_keymap_file_changes(
}) })
.detach(); .detach();
let mut current_layout_id = cx.keyboard_layout().id().to_string(); let mut current_mapping = settings::get_key_equivalents(cx.keyboard_layout().id());
cx.on_keyboard_layout_change(move |cx| { cx.on_keyboard_layout_change(move |cx| {
let next_layout_id = cx.keyboard_layout().id(); let next_mapping = settings::get_key_equivalents(cx.keyboard_layout().id());
if next_layout_id != current_layout_id { if next_mapping != current_mapping {
current_layout_id = next_layout_id.to_string(); current_mapping = next_mapping;
keyboard_layout_tx.unbounded_send(()).ok(); keyboard_layout_tx.unbounded_send(()).ok();
} }
}) })
@ -4729,7 +4729,7 @@ mod tests {
// and key strokes contain the given key // and key strokes contain the given key
bindings bindings
.into_iter() .into_iter()
.any(|binding| binding.keystrokes().iter().any(|k| k.display_key == key)), .any(|binding| binding.keystrokes().iter().any(|k| k.key == key)),
"On {} Failed to find {} with key binding {}", "On {} Failed to find {} with key binding {}",
line, line,
action.name(), action.name(),

View file

@ -1,17 +1,10 @@
use anyhow::Result; use anyhow::Result;
use db::{ use db::{define_connection, query, sqlez::statement::Statement, sqlez_macros::sql};
query,
sqlez::{domain::Domain, statement::Statement, thread_safe_connection::ThreadSafeConnection},
sqlez_macros::sql,
};
use workspace::{ItemId, WorkspaceDb, WorkspaceId}; use workspace::{ItemId, WorkspaceDb, WorkspaceId};
pub struct ComponentPreviewDb(ThreadSafeConnection); define_connection! {
pub static ref COMPONENT_PREVIEW_DB: ComponentPreviewDb<WorkspaceDb> =
impl Domain for ComponentPreviewDb { &[sql!(
const NAME: &str = stringify!(ComponentPreviewDb);
const MIGRATIONS: &[&str] = &[sql!(
CREATE TABLE component_previews ( CREATE TABLE component_previews (
workspace_id INTEGER, workspace_id INTEGER,
item_id INTEGER UNIQUE, item_id INTEGER UNIQUE,
@ -20,11 +13,9 @@ impl Domain for ComponentPreviewDb {
FOREIGN KEY(workspace_id) REFERENCES workspaces(workspace_id) FOREIGN KEY(workspace_id) REFERENCES workspaces(workspace_id)
ON DELETE CASCADE ON DELETE CASCADE
) STRICT; ) STRICT;
)]; )];
} }
db::static_connection!(COMPONENT_PREVIEW_DB, ComponentPreviewDb, [WorkspaceDb]);
impl ComponentPreviewDb { impl ComponentPreviewDb {
pub async fn save_active_page( pub async fn save_active_page(
&self, &self,

View file

@ -72,10 +72,7 @@ impl QuickActionBar {
Tooltip::with_meta( Tooltip::with_meta(
tooltip_text, tooltip_text,
Some(open_action_for_tooltip), Some(open_action_for_tooltip),
format!( format!("{} to open in a split", text_for_keystroke(&alt_click, cx)),
"{} to open in a split",
text_for_keystroke(&alt_click.modifiers, &alt_click.key, cx)
),
window, window,
cx, cx,
) )

View file

@ -284,8 +284,6 @@ pub mod agent {
OpenSettings, OpenSettings,
/// Opens the agent onboarding modal. /// Opens the agent onboarding modal.
OpenOnboardingModal, OpenOnboardingModal,
/// Opens the ACP onboarding modal.
OpenAcpOnboardingModal,
/// Resets the agent onboarding state. /// Resets the agent onboarding state.
ResetOnboarding, ResetOnboarding,
/// Starts a chat conversation with the agent. /// Starts a chat conversation with the agent.

View file

@ -3243,7 +3243,6 @@ Run the `theme selector: toggle` action in the command palette to see a current
"indent_size": 20, "indent_size": 20,
"auto_reveal_entries": true, "auto_reveal_entries": true,
"auto_fold_dirs": true, "auto_fold_dirs": true,
"drag_and_drop": true,
"scrollbar": { "scrollbar": {
"show": null "show": null
}, },

View file

@ -45,9 +45,9 @@ Zed supports ways to spawn (and rerun) commands using its integrated terminal to
// Whether to show the task line in the output of the spawned task, defaults to `true`. // Whether to show the task line in the output of the spawned task, defaults to `true`.
"show_summary": true, "show_summary": true,
// Whether to show the command line in the output of the spawned task, defaults to `true`. // Whether to show the command line in the output of the spawned task, defaults to `true`.
"show_output": true "show_output": true,
// Represents the tags for inline runnable indicators, or spawning multiple tasks at once. // Represents the tags for inline runnable indicators, or spawning multiple tasks at once.
// "tags": [] "tags": []
} }
] ]
``` ```

View file

@ -431,7 +431,6 @@ Project panel can be shown/hidden with {#action project_panel::ToggleFocus} ({#k
"auto_reveal_entries": true, // Show file in panel when activating its buffer "auto_reveal_entries": true, // Show file in panel when activating its buffer
"auto_fold_dirs": true, // Fold dirs with single subdir "auto_fold_dirs": true, // Fold dirs with single subdir
"sticky_scroll": true, // Stick parent directories at top of the project panel. "sticky_scroll": true, // Stick parent directories at top of the project panel.
"drag_and_drop": true, // Whether drag and drop is enabled
"scrollbar": { // Project panel scrollbar settings "scrollbar": { // Project panel scrollbar settings
"show": null // Show/hide: (auto, system, always, never) "show": null // Show/hide: (auto, system, always, never)
}, },