1use zed::lsp::CompletionKind;
2use zed::settings::LspSettings;
3use zed::{CodeLabel, CodeLabelSpan};
4use zed_extension_api::{self as zed, serde_json, Result};
5
6struct DartBinary {
7 pub path: String,
8 pub args: Option<Vec<String>>,
9}
10
11struct DartExtension;
12
13impl DartExtension {
14 fn language_server_binary(
15 &mut self,
16 _language_server_id: &zed::LanguageServerId,
17 worktree: &zed::Worktree,
18 ) -> Result<DartBinary> {
19 let binary_settings = LspSettings::for_worktree("dart", worktree)
20 .ok()
21 .and_then(|lsp_settings| lsp_settings.binary);
22 let binary_args = binary_settings
23 .as_ref()
24 .and_then(|binary_settings| binary_settings.arguments.clone());
25
26 if let Some(path) = binary_settings.and_then(|binary_settings| binary_settings.path) {
27 return Ok(DartBinary {
28 path,
29 args: binary_args,
30 });
31 }
32
33 if let Some(path) = worktree.which("dart") {
34 return Ok(DartBinary {
35 path,
36 args: binary_args,
37 });
38 }
39
40 Err(
41 "dart must be installed from dart.dev/get-dart or pointed to by the LSP binary settings"
42 .to_string(),
43 )
44 }
45}
46
47impl zed::Extension for DartExtension {
48 fn new() -> Self {
49 Self
50 }
51
52 fn language_server_command(
53 &mut self,
54 language_server_id: &zed::LanguageServerId,
55 worktree: &zed::Worktree,
56 ) -> Result<zed::Command> {
57 let dart_binary = self.language_server_binary(language_server_id, worktree)?;
58
59 Ok(zed::Command {
60 command: dart_binary.path,
61 args: dart_binary.args.unwrap_or_else(|| {
62 vec!["language-server".to_string(), "--protocol=lsp".to_string()]
63 }),
64 env: Default::default(),
65 })
66 }
67
68 fn language_server_workspace_configuration(
69 &mut self,
70 _language_server_id: &zed::LanguageServerId,
71 worktree: &zed::Worktree,
72 ) -> Result<Option<serde_json::Value>> {
73 let settings = LspSettings::for_worktree("dart", worktree)
74 .ok()
75 .and_then(|lsp_settings| lsp_settings.settings.clone())
76 .unwrap_or_default();
77
78 Ok(Some(serde_json::json!({
79 "dart": settings
80 })))
81 }
82
83 fn label_for_completion(
84 &self,
85 _language_server_id: &zed::LanguageServerId,
86 completion: zed::lsp::Completion,
87 ) -> Option<CodeLabel> {
88 let arrow = " → ";
89
90 match completion.kind? {
91 CompletionKind::Class => Some(CodeLabel {
92 filter_range: (0..completion.label.len()).into(),
93 spans: vec![CodeLabelSpan::literal(
94 completion.label,
95 Some("type".into()),
96 )],
97 code: String::new(),
98 }),
99 CompletionKind::Function | CompletionKind::Constructor | CompletionKind::Method => {
100 let mut parts = completion.detail.as_ref()?.split(arrow);
101 let (name, _) = completion.label.split_once('(')?;
102 let parameter_list = parts.next()?;
103 let return_type = parts.next()?;
104 let fn_name = " a";
105 let fat_arrow = " => ";
106 let call_expr = "();";
107
108 let code =
109 format!("{return_type}{fn_name}{parameter_list}{fat_arrow}{name}{call_expr}");
110
111 let parameter_list_start = return_type.len() + fn_name.len();
112
113 Some(CodeLabel {
114 spans: vec![
115 CodeLabelSpan::code_range(
116 code.len() - call_expr.len() - name.len()..code.len() - call_expr.len(),
117 ),
118 CodeLabelSpan::code_range(
119 parameter_list_start..parameter_list_start + parameter_list.len(),
120 ),
121 CodeLabelSpan::literal(arrow, None),
122 CodeLabelSpan::code_range(0..return_type.len()),
123 ],
124 filter_range: (0..name.len()).into(),
125 code,
126 })
127 }
128 CompletionKind::Property => {
129 let class_start = "class A {";
130 let get = " get ";
131 let property_end = " => a; }";
132 let ty = completion.detail?;
133 let name = completion.label;
134
135 let code = format!("{class_start}{ty}{get}{name}{property_end}");
136 let name_start = class_start.len() + ty.len() + get.len();
137
138 Some(CodeLabel {
139 spans: vec![
140 CodeLabelSpan::code_range(name_start..name_start + name.len()),
141 CodeLabelSpan::literal(arrow, None),
142 CodeLabelSpan::code_range(class_start.len()..class_start.len() + ty.len()),
143 ],
144 filter_range: (0..name.len()).into(),
145 code,
146 })
147 }
148 CompletionKind::Variable => {
149 let name = completion.label;
150
151 Some(CodeLabel {
152 filter_range: (0..name.len()).into(),
153 spans: vec![CodeLabelSpan::literal(name, Some("variable".into()))],
154 code: String::new(),
155 })
156 }
157 _ => None,
158 }
159 }
160}
161
162zed::register_extension!(DartExtension);