annotate_snippets/renderer/
margin.rs1use std::cmp::{max, min};
2
3const ELLIPSIS_PASSING: usize = 6;
4const LONG_WHITESPACE: usize = 20;
5const LONG_WHITESPACE_PADDING: usize = 4;
6
7#[derive(Clone, Copy, Debug, PartialEq)]
8pub(crate) struct Margin {
9 whitespace_left: usize,
11 span_left: usize,
13 span_right: usize,
15 computed_left: usize,
17 computed_right: usize,
19 term_width: usize,
21 label_right: usize,
24}
25
26impl Margin {
27 pub(crate) fn new(
28 whitespace_left: usize,
29 span_left: usize,
30 span_right: usize,
31 label_right: usize,
32 term_width: usize,
33 max_line_len: usize,
34 ) -> Self {
35 let mut m = Margin {
45 whitespace_left: whitespace_left.saturating_sub(ELLIPSIS_PASSING),
46 span_left: span_left.saturating_sub(ELLIPSIS_PASSING),
47 span_right: span_right + ELLIPSIS_PASSING,
48 computed_left: 0,
49 computed_right: 0,
50 term_width,
51 label_right: label_right + ELLIPSIS_PASSING,
52 };
53 m.compute(max_line_len);
54 m
55 }
56
57 pub(crate) fn was_cut_left(&self) -> bool {
58 self.computed_left > 0
59 }
60
61 pub(crate) fn was_cut_right(&self, line_len: usize) -> bool {
62 let right =
63 if self.computed_right == self.span_right || self.computed_right == self.label_right {
64 self.computed_right - ELLIPSIS_PASSING
67 } else {
68 self.computed_right
69 };
70 right < line_len && self.computed_left + self.term_width < line_len
71 }
72
73 fn compute(&mut self, max_line_len: usize) {
74 self.computed_left = if self.whitespace_left > LONG_WHITESPACE {
76 self.whitespace_left - (LONG_WHITESPACE - LONG_WHITESPACE_PADDING) } else {
78 0
79 };
80 self.computed_right = max(max_line_len, self.computed_left);
83
84 if self.computed_right - self.computed_left > self.term_width {
85 if self.label_right - self.whitespace_left <= self.term_width {
87 self.computed_left = self.whitespace_left;
89 self.computed_right = self.computed_left + self.term_width;
90 } else if self.label_right - self.span_left <= self.term_width {
91 let padding_left = (self.term_width - (self.label_right - self.span_left)) / 2;
93 self.computed_left = self.span_left.saturating_sub(padding_left);
94 self.computed_right = self.computed_left + self.term_width;
95 } else if self.span_right - self.span_left <= self.term_width {
96 let padding_left = (self.term_width - (self.span_right - self.span_left)) / 5 * 2;
98 self.computed_left = self.span_left.saturating_sub(padding_left);
99 self.computed_right = self.computed_left + self.term_width;
100 } else {
101 self.computed_left = self.span_left;
103 self.computed_right = self.span_right;
104 }
105 }
106 }
107
108 pub(crate) fn left(&self, line_len: usize) -> usize {
109 min(self.computed_left, line_len)
110 }
111
112 pub(crate) fn right(&self, line_len: usize) -> usize {
113 if line_len.saturating_sub(self.computed_left) <= self.term_width {
114 line_len
115 } else {
116 min(line_len, self.computed_right)
117 }
118 }
119}