1
  2
  3
  4
  5
  6
  7
  8
  9
 10
 11
 12
 13
 14
 15
 16
 17
 18
 19
 20
 21
 22
 23
 24
 25
 26
 27
 28
 29
 30
 31
 32
 33
 34
 35
 36
 37
 38
 39
 40
 41
 42
 43
 44
 45
 46
 47
 48
 49
 50
 51
 52
 53
 54
 55
 56
 57
 58
 59
 60
 61
 62
 63
 64
 65
 66
 67
 68
 69
 70
 71
 72
 73
 74
 75
 76
 77
 78
 79
 80
 81
 82
 83
 84
 85
 86
 87
 88
 89
 90
 91
 92
 93
 94
 95
 96
 97
 98
 99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
use lsp_types::{Position, Range, TextDocumentContentChangeEvent, Url};
#[derive(Clone)]
pub struct FullTextDocument {
    pub uri: Url,

    /// The text document's language identifier.
    pub language_id: String,

    /// The version number of this document (it will strictly increase after each
    /// change, including undo/redo).
    pub version: i64,

    /// The content of the opened text document.
    pub text: String,

    line_offset: Option<Vec<usize>>,
}

impl FullTextDocument {
    pub fn new(uri: Url, language_id: String, version: i64, text: String) -> FullTextDocument {
        // let item = lsp_types::TextDocumentItem::new(uri, language_id, version, text);
        FullTextDocument {
            uri,
            language_id,
            version,
            text,
            line_offset: None,
        }
    }

    pub fn update(&mut self, changes: Vec<TextDocumentContentChangeEvent>, version: i64) {
        for change in changes {
            if Self::is_incremental(&change) {
                // makes sure start is before end
                let range = get_wellformed_range(change.range.unwrap());

                let start_offset = self.offset_at(range.start);
                let end_offset = self.offset_at(range.end);

                let (start_byte, end_byte) = self.transform_offset_to_byte_offset(start_offset, end_offset);
                self.text = self.text[0..start_byte].to_string()
                    + &change.text
                    + &self.text[end_byte..];
                // self.text =
                //     self.text.chars().take(start_offset).chain(change.text.chars()).chain(self.text.chars().skip(end_offset)).collect::<String>();
                let start_line = range.start.line as usize;
                let end_line = range.end.line as usize;
                let line_offsets = self.get_line_offsets();

                let mut add_line_offsets =
                    compute_line_offsets(&change.text, false, Some(start_offset));

                let add_line_offsets_len = add_line_offsets.len();
                // if line_offsets.len() <= end_line as usize {
                //     line_offsets.extend(vec![0; end_line as usize + 1 - line_offsets.len()]);
                // }

                if end_line - start_line == add_line_offsets.len() {
                    for (i, offset) in add_line_offsets.into_iter().enumerate() {
                        line_offsets[i + start_line + 1] = offset;
                    }
                } else {
                    *line_offsets = {
                        let mut res =
                            line_offsets[0..=start_line.min(line_offsets.len() - 1)].to_vec();
                        res.append(&mut add_line_offsets);
                        res.extend_from_slice(
                            &line_offsets[end_line.min(line_offsets.len() - 1) + 1..],
                        );
                        res
                    };
                }
                let diff: i32 = change.text.len() as i32 - (end_offset - start_offset) as i32;
                if diff != 0 {
                    for i in start_line + 1 + add_line_offsets_len..line_offsets.len() {
                        line_offsets[i] = (line_offsets[i] as i32 + diff) as usize;
                    }
                }
            } else if Self::is_full(&change) {
                self.text = change.text;
                self.line_offset = None;
            }
            self.version = version;
        }
    }

    pub fn transform_offset_to_byte_offset(&self, start_offset: usize, end_offset: usize) -> (usize, usize) {
        let start_byte = self.text
            .chars()
            .take(start_offset)
            .fold(0, |acc, cur| acc + cur.len_utf8());
        let end_byte = (&self.text[start_offset..end_offset])
            .chars()
            .take(end_offset)
            .fold(0, |acc, cur| acc + cur.len_utf8())
            + start_byte;
        (start_byte, end_byte)
    }
    pub fn position_at(&mut self, mut offset: u32) -> Position {
        offset = offset.min(self.text.len() as u32).max(0);

        let line_offsets = self.get_line_offsets();
        // let low = 0, high = lineOffsets.length;
        let mut low = 0usize;
        let mut high = line_offsets.len();
        if high == 0 {
            return Position {
                line: 0,
                character: offset,
            };
        }
        while low < high {
            let mid = low + (high - low) / 2;
            if line_offsets[mid] as u32 > offset {
                high = mid;
            } else {
                low = mid + 1;
            }
        }
        let line = low as u32 - 1;
        return Position {
            line,
            character: offset - line_offsets[line as usize] as u32,
        };
        // while (low < high) {
        // 	let mid = Math.floor((low + high) / 2);
        // 	if (lineOffsets[mid] > offset) {
        // 		high = mid;
        // 	} else {
        // 		low = mid + 1;
        // 	}
        // }
        // // low is the least x for which the line offset is larger than the current offset
        // // or array.length if no line offset is larger than the current offset
        // let line = low - 1;
        // return { line, character: offset - lineOffsets[line] };
    }

    pub fn line_count(&mut self) -> usize {
        self.get_line_offsets().len()
    }
    pub fn is_incremental(event: &TextDocumentContentChangeEvent) -> bool {
        event.range.is_some()
    }

    pub fn is_full(event: &TextDocumentContentChangeEvent) -> bool {
        !event.range_length.is_some() && !event.range.is_some()
    }

    pub fn get_line_offsets(&mut self) -> &mut Vec<usize> {
        if self.line_offset.is_none() {
            self.line_offset = Some(compute_line_offsets(&self.text, true, None));
        }
        self.line_offset.as_mut().unwrap()
    }
    pub fn offset_at(&mut self, position: Position) -> usize {
        let line_offsets = self.get_line_offsets();
        if position.line >= line_offsets.len() as u32 {
            return self.text.len();
        }
        let line_offset = line_offsets[position.line as usize];
        let next_line_offset = if position.line + 1 < line_offsets.len() as u32 {
            line_offsets[position.line as usize + 1]
        } else {
            self.text.len()
        };
        (line_offset + position.character as usize)
            .min(next_line_offset)
            .max(line_offset)
        // return Math.max(
        //     Math.min(line_offset + position.character, next_line_offset),
        //     line_offset,
        // );
    }
}

pub fn compute_line_offsets(
    text: &String,
    is_at_line_start: bool,
    text_offset: Option<usize>,
) -> Vec<usize> {
    let text_offset = if let Some(offset) = text_offset {
        offset
    } else {
        0
    };
    let mut result = if is_at_line_start {
        vec![text_offset]
    } else {
        vec![]
    };
    let char_array: Vec<char> = text.chars().collect();
    let mut i = 0;
    while i < char_array.len() {
        let &ch = unsafe { char_array.get_unchecked(i) };
        if ch == '\r' || ch == '\n' {
            if ch == '\r'
                && i + 1 < char_array.len()
                && unsafe { char_array.get_unchecked(i + 1) == &'\n' }
            {
                i += 1;
            }
            result.push(text_offset + i + 1);
        }
        i += 1;
    }
    result
}

fn get_wellformed_range(range: Range) -> Range {
    let start = range.start;
    let end = range.end;
    if start.line > end.line || (start.line == end.line && start.character > end.character) {
        Range::new(end, start)
    } else {
        range
    }
}