1use crate::parser::LangId;
8
9#[derive(Debug, Clone, Copy, PartialEq, Eq)]
11pub enum IndentStyle {
12 Tabs,
13 Spaces(u8),
14}
15
16impl IndentStyle {
17 pub fn as_str(&self) -> &'static str {
19 match self {
20 IndentStyle::Tabs => "\t",
21 IndentStyle::Spaces(2) => " ",
22 IndentStyle::Spaces(4) => " ",
23 IndentStyle::Spaces(8) => " ",
24 IndentStyle::Spaces(n) => {
25 let s: String = " ".repeat(*n as usize);
28 Box::leak(s.into_boxed_str())
29 }
30 }
31 }
32
33 pub fn default_for(lang: LangId) -> Self {
35 match lang {
36 LangId::Python => IndentStyle::Spaces(4),
37 LangId::TypeScript
38 | LangId::Tsx
39 | LangId::JavaScript
40 | LangId::Vue
41 | LangId::Json
42 | LangId::Scala
43 | LangId::Ruby
44 | LangId::Lua
45 | LangId::Yaml => IndentStyle::Spaces(2),
46 LangId::Rust => IndentStyle::Spaces(4),
47 LangId::Go => IndentStyle::Tabs,
48 LangId::C | LangId::Cpp | LangId::Zig | LangId::CSharp | LangId::Bash => {
49 IndentStyle::Spaces(4)
50 }
51 LangId::Solidity
52 | LangId::Java
53 | LangId::Kotlin
54 | LangId::Swift
55 | LangId::Php
56 | LangId::Perl => IndentStyle::Spaces(4),
57 LangId::Html => IndentStyle::Spaces(2),
58 LangId::Markdown => IndentStyle::Spaces(4),
59 }
60 }
61}
62
63pub fn detect_indent(source: &str, lang: LangId) -> IndentStyle {
72 let mut tab_count: u32 = 0;
73 let mut space_count: u32 = 0;
74 let mut indent_widths: [u32; 9] = [0; 9]; for line in source.lines() {
77 if line.is_empty() {
78 continue;
79 }
80 let first = line.as_bytes()[0];
81 if first == b'\t' {
82 tab_count += 1;
83 } else if first == b' ' {
84 space_count += 1;
85 let leading = line.len() - line.trim_start_matches(' ').len();
87 if leading > 0 && leading <= 8 {
88 indent_widths[leading] += 1;
89 }
90 }
91 }
92
93 let total = tab_count + space_count;
94 if total == 0 {
95 return IndentStyle::default_for(lang);
96 }
97
98 if tab_count > total / 2 {
100 return IndentStyle::Tabs;
101 }
102
103 if space_count > total / 2 {
105 let width = determine_space_width(&indent_widths);
109 return IndentStyle::Spaces(width);
110 }
111
112 IndentStyle::default_for(lang)
114}
115
116fn determine_space_width(widths: &[u32; 9]) -> u8 {
122 let smallest = (1..=8usize).find(|&i| widths[i] > 0);
124 let smallest = match smallest {
125 Some(s) => s,
126 None => return 4,
127 };
128
129 let all_multiples = (1..=8).all(|i| widths[i] == 0 || i % smallest == 0);
131
132 if all_multiples && smallest >= 2 {
133 return smallest as u8;
134 }
135
136 for &candidate in &[4u8, 2, 8] {
138 let c = candidate as usize;
139 let mut matching: u32 = 0;
140 let mut non_matching: u32 = 0;
141 for i in 1..=8 {
142 if widths[i] > 0 {
143 if i % c == 0 {
144 matching += widths[i];
145 } else {
146 non_matching += widths[i];
147 }
148 }
149 }
150 if matching > 0 && non_matching == 0 {
151 return candidate;
152 }
153 }
154
155 smallest as u8
156}
157
158#[cfg(test)]
159mod tests {
160 use super::*;
161
162 #[test]
163 fn detect_indent_tabs() {
164 let source = "fn main() {\n\tlet x = 1;\n\tlet y = 2;\n}\n";
165 assert_eq!(detect_indent(source, LangId::Rust), IndentStyle::Tabs);
166 }
167
168 #[test]
169 fn detect_indent_two_spaces() {
170 let source = "class Foo {\n bar() {}\n baz() {}\n}\n";
171 assert_eq!(
172 detect_indent(source, LangId::TypeScript),
173 IndentStyle::Spaces(2)
174 );
175 }
176
177 #[test]
178 fn detect_indent_four_spaces() {
179 let source =
180 "class Foo:\n def bar(self):\n pass\n def baz(self):\n pass\n";
181 assert_eq!(
182 detect_indent(source, LangId::Python),
183 IndentStyle::Spaces(4)
184 );
185 }
186
187 #[test]
188 fn detect_indent_empty_source_uses_default() {
189 assert_eq!(detect_indent("", LangId::Python), IndentStyle::Spaces(4));
190 assert_eq!(
191 detect_indent("", LangId::TypeScript),
192 IndentStyle::Spaces(2)
193 );
194 assert_eq!(detect_indent("", LangId::Go), IndentStyle::Tabs);
195 }
196
197 #[test]
198 fn detect_indent_no_indented_lines_uses_default() {
199 let source = "x = 1\ny = 2\n";
200 assert_eq!(
201 detect_indent(source, LangId::Python),
202 IndentStyle::Spaces(4)
203 );
204 }
205
206 #[test]
207 fn indent_style_as_str() {
208 assert_eq!(IndentStyle::Tabs.as_str(), "\t");
209 assert_eq!(IndentStyle::Spaces(2).as_str(), " ");
210 assert_eq!(IndentStyle::Spaces(4).as_str(), " ");
211 }
212
213 #[test]
214 fn detect_indent_four_spaces_with_nested() {
215 let source = "impl Foo {\n fn bar() {\n let x = 1;\n }\n}\n";
217 assert_eq!(detect_indent(source, LangId::Rust), IndentStyle::Spaces(4));
218 }
219}