1use anyhow::Context as _;
2use gpui::{App, Context, Entity, Window};
3use language::Language;
4use project::lsp_store::lsp_ext_command::SwitchSourceHeaderResult;
5use rpc::proto;
6use url::Url;
7use util::paths::{PathStyle, UrlExt as _};
8use workspace::{OpenOptions, OpenVisible};
9
10use crate::lsp_ext::find_specific_language_server_in_selection;
11
12use crate::{Editor, SwitchSourceHeader, element::register_action};
13
14use project::lsp_store::clangd_ext::CLANGD_SERVER_NAME;
15
16fn is_c_language(language: &Language) -> bool {
17 language.name() == "C++".into() || language.name() == "C".into()
18}
19
20pub fn switch_source_header(
21 editor: &mut Editor,
22 _: &SwitchSourceHeader,
23 window: &mut Window,
24 cx: &mut Context<Editor>,
25) {
26 let Some(project) = &editor.project else {
27 return;
28 };
29 let Some(workspace) = editor.workspace() else {
30 return;
31 };
32
33 let Some((_, _, server_to_query, buffer)) =
34 find_specific_language_server_in_selection(editor, cx, is_c_language, CLANGD_SERVER_NAME)
35 else {
36 return;
37 };
38 let project = project.clone();
39 let upstream_client = project.read(cx).lsp_store().read(cx).upstream_client();
40 cx.spawn_in(window, async move |_editor, cx| {
41 let source_file = buffer.read_with(cx, |buffer, _| {
42 buffer
43 .file()
44 .map(|file| file.path())
45 .map(|path| path.display(PathStyle::local()).to_string())
46 .unwrap_or_else(|| "Unknown".to_string())
47 });
48
49 let switch_source_header = if let Some((client, project_id)) = upstream_client {
50 let buffer_id = buffer.read_with(cx, |buffer, _| buffer.remote_id());
51 let request = proto::LspExtSwitchSourceHeader {
52 project_id,
53 buffer_id: buffer_id.to_proto(),
54 };
55 let response = client
56 .request(request)
57 .await
58 .context("lsp ext switch source header proto request")?;
59 SwitchSourceHeaderResult(response.target_file)
60 } else {
61 project
62 .update(cx, |project, cx| {
63 project.request_lsp(
64 buffer,
65 project::LanguageServerToQuery::Other(server_to_query),
66 project::lsp_store::lsp_ext_command::SwitchSourceHeader,
67 cx,
68 )
69 })
70 .await
71 .with_context(|| {
72 format!("Switch source/header LSP request for path \"{source_file}\" failed")
73 })?
74 };
75
76 if switch_source_header.0.is_empty() {
77 return Ok(());
78 }
79 let path_style = workspace.update(cx, |ws, cx| ws.path_style(cx));
80 let path = Url::parse(&switch_source_header.0).with_context(|| {
81 format!(
82 "Parsing URL \"{}\" returned from switch source/header failed",
83 switch_source_header.0
84 )
85 })?;
86 let path = path.to_file_path_ext(path_style).map_err(|()| {
87 anyhow::anyhow!(
88 "URL conversion to file path failed for \"{}\"",
89 switch_source_header.0
90 )
91 })?;
92
93 workspace
94 .update_in(cx, |workspace, window, cx| {
95 workspace.open_abs_path(
96 path,
97 OpenOptions {
98 visible: Some(OpenVisible::None),
99 ..Default::default()
100 },
101 window,
102 cx,
103 )
104 })
105 .with_context(|| {
106 format!(
107 "Switch source/header could not open \"{}\" in workspace",
108 switch_source_header.0
109 )
110 })?
111 .await
112 .map(|_| ())
113 })
114 .detach_and_log_err(cx);
115}
116
117pub fn apply_related_actions(editor: &Entity<Editor>, window: &mut Window, cx: &mut App) {
118 if editor
119 .read(cx)
120 .buffer()
121 .read(cx)
122 .all_buffers()
123 .into_iter()
124 .filter_map(|buffer| buffer.read(cx).language())
125 .any(|language| is_c_language(language))
126 {
127 register_action(editor, window, switch_source_header);
128 }
129}