codespan_reporting/term/
config.rs

1use termcolor::{Color, ColorSpec};
2
3use crate::diagnostic::{LabelStyle, Severity};
4
5/// Configures how a diagnostic is rendered.
6#[derive(Clone, Debug)]
7pub struct Config {
8    /// The display style to use when rendering diagnostics.
9    /// Defaults to: [`DisplayStyle::Rich`].
10    ///
11    /// [`DisplayStyle::Rich`]: DisplayStyle::Rich
12    pub display_style: DisplayStyle,
13    /// Column width of tabs.
14    /// Defaults to: `4`.
15    pub tab_width: usize,
16    /// Styles to use when rendering the diagnostic.
17    pub styles: Styles,
18    /// Characters to use when rendering the diagnostic.
19    pub chars: Chars,
20    /// The minimum number of lines to be shown after the line on which a multiline [`Label`] begins.
21    ///
22    /// Defaults to: `3`.
23    ///
24    /// [`Label`]: crate::diagnostic::Label
25    pub start_context_lines: usize,
26    /// The minimum number of lines to be shown before the line on which a multiline [`Label`] ends.
27    ///
28    /// Defaults to: `1`.
29    ///
30    /// [`Label`]: crate::diagnostic::Label
31    pub end_context_lines: usize,
32}
33
34impl Default for Config {
35    fn default() -> Config {
36        Config {
37            display_style: DisplayStyle::Rich,
38            tab_width: 4,
39            styles: Styles::default(),
40            chars: Chars::default(),
41            start_context_lines: 3,
42            end_context_lines: 1,
43        }
44    }
45}
46
47/// The display style to use when rendering diagnostics.
48#[derive(Clone, Debug)]
49pub enum DisplayStyle {
50    /// Output a richly formatted diagnostic, with source code previews.
51    ///
52    /// ```text
53    /// error[E0001]: unexpected type in `+` application
54    ///   ┌─ test:2:9
55    ///   │
56    /// 2 │ (+ test "")
57    ///   │         ^^ expected `Int` but found `String`
58    ///   │
59    ///   = expected type `Int`
60    ///        found type `String`
61    ///
62    /// error[E0002]: Bad config found
63    ///
64    /// ```
65    Rich,
66    /// Output a condensed diagnostic, with a line number, severity, message and notes (if any).
67    ///
68    /// ```text
69    /// test:2:9: error[E0001]: unexpected type in `+` application
70    /// = expected type `Int`
71    ///      found type `String`
72    ///
73    /// error[E0002]: Bad config found
74    /// ```
75    Medium,
76    /// Output a short diagnostic, with a line number, severity, and message.
77    ///
78    /// ```text
79    /// test:2:9: error[E0001]: unexpected type in `+` application
80    /// error[E0002]: Bad config found
81    /// ```
82    Short,
83}
84
85/// Styles to use when rendering the diagnostic.
86#[derive(Clone, Debug)]
87pub struct Styles {
88    /// The style to use when rendering bug headers.
89    /// Defaults to `fg:red bold intense`.
90    pub header_bug: ColorSpec,
91    /// The style to use when rendering error headers.
92    /// Defaults to `fg:red bold intense`.
93    pub header_error: ColorSpec,
94    /// The style to use when rendering warning headers.
95    /// Defaults to `fg:yellow bold intense`.
96    pub header_warning: ColorSpec,
97    /// The style to use when rendering note headers.
98    /// Defaults to `fg:green bold intense`.
99    pub header_note: ColorSpec,
100    /// The style to use when rendering help headers.
101    /// Defaults to `fg:cyan bold intense`.
102    pub header_help: ColorSpec,
103    /// The style to use when the main diagnostic message.
104    /// Defaults to `bold intense`.
105    pub header_message: ColorSpec,
106
107    /// The style to use when rendering bug labels.
108    /// Defaults to `fg:red`.
109    pub primary_label_bug: ColorSpec,
110    /// The style to use when rendering error labels.
111    /// Defaults to `fg:red`.
112    pub primary_label_error: ColorSpec,
113    /// The style to use when rendering warning labels.
114    /// Defaults to `fg:yellow`.
115    pub primary_label_warning: ColorSpec,
116    /// The style to use when rendering note labels.
117    /// Defaults to `fg:green`.
118    pub primary_label_note: ColorSpec,
119    /// The style to use when rendering help labels.
120    /// Defaults to `fg:cyan`.
121    pub primary_label_help: ColorSpec,
122    /// The style to use when rendering secondary labels.
123    /// Defaults `fg:blue` (or `fg:cyan` on windows).
124    pub secondary_label: ColorSpec,
125
126    /// The style to use when rendering the line numbers.
127    /// Defaults `fg:blue` (or `fg:cyan` on windows).
128    pub line_number: ColorSpec,
129    /// The style to use when rendering the source code borders.
130    /// Defaults `fg:blue` (or `fg:cyan` on windows).
131    pub source_border: ColorSpec,
132    /// The style to use when rendering the note bullets.
133    /// Defaults `fg:blue` (or `fg:cyan` on windows).
134    pub note_bullet: ColorSpec,
135}
136
137impl Styles {
138    /// The style used to mark a header at a given severity.
139    pub fn header(&self, severity: Severity) -> &ColorSpec {
140        match severity {
141            Severity::Bug => &self.header_bug,
142            Severity::Error => &self.header_error,
143            Severity::Warning => &self.header_warning,
144            Severity::Note => &self.header_note,
145            Severity::Help => &self.header_help,
146        }
147    }
148
149    /// The style used to mark a primary or secondary label at a given severity.
150    pub fn label(&self, severity: Severity, label_style: LabelStyle) -> &ColorSpec {
151        match (label_style, severity) {
152            (LabelStyle::Primary, Severity::Bug) => &self.primary_label_bug,
153            (LabelStyle::Primary, Severity::Error) => &self.primary_label_error,
154            (LabelStyle::Primary, Severity::Warning) => &self.primary_label_warning,
155            (LabelStyle::Primary, Severity::Note) => &self.primary_label_note,
156            (LabelStyle::Primary, Severity::Help) => &self.primary_label_help,
157            (LabelStyle::Secondary, _) => &self.secondary_label,
158        }
159    }
160
161    #[doc(hidden)]
162    pub fn with_blue(blue: Color) -> Styles {
163        let header = ColorSpec::new().set_bold(true).set_intense(true).clone();
164
165        Styles {
166            header_bug: header.clone().set_fg(Some(Color::Red)).clone(),
167            header_error: header.clone().set_fg(Some(Color::Red)).clone(),
168            header_warning: header.clone().set_fg(Some(Color::Yellow)).clone(),
169            header_note: header.clone().set_fg(Some(Color::Green)).clone(),
170            header_help: header.clone().set_fg(Some(Color::Cyan)).clone(),
171            header_message: header,
172
173            primary_label_bug: ColorSpec::new().set_fg(Some(Color::Red)).clone(),
174            primary_label_error: ColorSpec::new().set_fg(Some(Color::Red)).clone(),
175            primary_label_warning: ColorSpec::new().set_fg(Some(Color::Yellow)).clone(),
176            primary_label_note: ColorSpec::new().set_fg(Some(Color::Green)).clone(),
177            primary_label_help: ColorSpec::new().set_fg(Some(Color::Cyan)).clone(),
178            secondary_label: ColorSpec::new().set_fg(Some(blue)).clone(),
179
180            line_number: ColorSpec::new().set_fg(Some(blue)).clone(),
181            source_border: ColorSpec::new().set_fg(Some(blue)).clone(),
182            note_bullet: ColorSpec::new().set_fg(Some(blue)).clone(),
183        }
184    }
185}
186
187impl Default for Styles {
188    fn default() -> Styles {
189        // Blue is really difficult to see on the standard windows command line
190        #[cfg(windows)]
191        const BLUE: Color = Color::Cyan;
192        #[cfg(not(windows))]
193        const BLUE: Color = Color::Blue;
194
195        Self::with_blue(BLUE)
196    }
197}
198
199/// Characters to use when rendering the diagnostic.
200///
201/// By using [`Chars::ascii()`] you can switch to an ASCII-only format suitable
202/// for rendering on terminals that do not support box drawing characters.
203#[derive(Clone, Debug)]
204pub struct Chars {
205    /// The characters to use for the top-left border of the snippet.
206    /// Defaults to: `"┌─"` or `"-->"` with [`Chars::ascii()`].
207    pub snippet_start: String,
208    /// The character to use for the left border of the source.
209    /// Defaults to: `'│'` or `'|'` with [`Chars::ascii()`].
210    pub source_border_left: char,
211    /// The character to use for the left border break of the source.
212    /// Defaults to: `'·'` or `'.'` with [`Chars::ascii()`].
213    pub source_border_left_break: char,
214
215    /// The character to use for the note bullet.
216    /// Defaults to: `'='`.
217    pub note_bullet: char,
218
219    /// The character to use for marking a single-line primary label.
220    /// Defaults to: `'^'`.
221    pub single_primary_caret: char,
222    /// The character to use for marking a single-line secondary label.
223    /// Defaults to: `'-'`.
224    pub single_secondary_caret: char,
225
226    /// The character to use for marking the start of a multi-line primary label.
227    /// Defaults to: `'^'`.
228    pub multi_primary_caret_start: char,
229    /// The character to use for marking the end of a multi-line primary label.
230    /// Defaults to: `'^'`.
231    pub multi_primary_caret_end: char,
232    /// The character to use for marking the start of a multi-line secondary label.
233    /// Defaults to: `'\''`.
234    pub multi_secondary_caret_start: char,
235    /// The character to use for marking the end of a multi-line secondary label.
236    /// Defaults to: `'\''`.
237    pub multi_secondary_caret_end: char,
238    /// The character to use for the top-left corner of a multi-line label.
239    /// Defaults to: `'╭'` or `'/'` with [`Chars::ascii()`].
240    pub multi_top_left: char,
241    /// The character to use for the top of a multi-line label.
242    /// Defaults to: `'─'` or `'-'` with [`Chars::ascii()`].
243    pub multi_top: char,
244    /// The character to use for the bottom-left corner of a multi-line label.
245    /// Defaults to: `'╰'` or `'\'` with [`Chars::ascii()`].
246    pub multi_bottom_left: char,
247    /// The character to use when marking the bottom of a multi-line label.
248    /// Defaults to: `'─'` or `'-'` with [`Chars::ascii()`].
249    pub multi_bottom: char,
250    /// The character to use for the left of a multi-line label.
251    /// Defaults to: `'│'` or `'|'` with [`Chars::ascii()`].
252    pub multi_left: char,
253
254    /// The character to use for the left of a pointer underneath a caret.
255    /// Defaults to: `'│'` or `'|'` with [`Chars::ascii()`].
256    pub pointer_left: char,
257}
258
259impl Default for Chars {
260    fn default() -> Chars {
261        Chars::box_drawing()
262    }
263}
264
265impl Chars {
266    /// A character set that uses Unicode box drawing characters.
267    pub fn box_drawing() -> Chars {
268        Chars {
269            snippet_start: "┌─".into(),
270            source_border_left: '│',
271            source_border_left_break: '·',
272
273            note_bullet: '=',
274
275            single_primary_caret: '^',
276            single_secondary_caret: '-',
277
278            multi_primary_caret_start: '^',
279            multi_primary_caret_end: '^',
280            multi_secondary_caret_start: '\'',
281            multi_secondary_caret_end: '\'',
282            multi_top_left: '╭',
283            multi_top: '─',
284            multi_bottom_left: '╰',
285            multi_bottom: '─',
286            multi_left: '│',
287
288            pointer_left: '│',
289        }
290    }
291
292    /// A character set that only uses ASCII characters.
293    ///
294    /// This is useful if your terminal's font does not support box drawing
295    /// characters well and results in output that looks similar to rustc's
296    /// diagnostic output.
297    pub fn ascii() -> Chars {
298        Chars {
299            snippet_start: "-->".into(),
300            source_border_left: '|',
301            source_border_left_break: '.',
302
303            note_bullet: '=',
304
305            single_primary_caret: '^',
306            single_secondary_caret: '-',
307
308            multi_primary_caret_start: '^',
309            multi_primary_caret_end: '^',
310            multi_secondary_caret_start: '\'',
311            multi_secondary_caret_end: '\'',
312            multi_top_left: '/',
313            multi_top: '-',
314            multi_bottom_left: '\\',
315            multi_bottom: '-',
316            multi_left: '|',
317
318            pointer_left: '|',
319        }
320    }
321}