* git mv src/wasm-lib rust Signed-off-by: Jess Frazelle <github@jessfraz.com> * mv wasm-lib to workspace Signed-off-by: Jess Frazelle <github@jessfraz.com> * mv kcl-lib Signed-off-by: Jess Frazelle <github@jessfraz.com> * mv derive docs Signed-off-by: Jess Frazelle <github@jessfraz.com> * resolve file paths Signed-off-by: Jess Frazelle <github@jessfraz.com> * clippy Signed-off-by: Jess Frazelle <github@jessfraz.com> * move more shit Signed-off-by: Jess Frazelle <github@jessfraz.com> * fix more paths Signed-off-by: Jess Frazelle <github@jessfraz.com> * make yarn build:wasm work Signed-off-by: Jess Frazelle <github@jessfraz.com> * fix scripts Signed-off-by: Jess Frazelle <github@jessfraz.com> * fixups Signed-off-by: Jess Frazelle <github@jessfraz.com> * better references Signed-off-by: Jess Frazelle <github@jessfraz.com> * fix cargo ci Signed-off-by: Jess Frazelle <github@jessfraz.com> * fix reference Signed-off-by: Jess Frazelle <github@jessfraz.com> * fix more ci Signed-off-by: Jess Frazelle <github@jessfraz.com> * fix tests Signed-off-by: Jess Frazelle <github@jessfraz.com> * cargo sort Signed-off-by: Jess Frazelle <github@jessfraz.com> * fix script Signed-off-by: Jess Frazelle <github@jessfraz.com> * fix Signed-off-by: Jess Frazelle <github@jessfraz.com> * fmt Signed-off-by: Jess Frazelle <github@jessfraz.com> * fix a dep Signed-off-by: Jess Frazelle <github@jessfraz.com> * sort Signed-off-by: Jess Frazelle <github@jessfraz.com> * remove unused deps Signed-off-by: Jess Frazelle <github@jessfraz.com> * Revert "remove unused deps" This reverts commit fbabdb062e275fd5cbc1476f8480a1afee15d972. * updates Signed-off-by: Jess Frazelle <github@jessfraz.com> * deps; Signed-off-by: Jess Frazelle <github@jessfraz.com> * fixes Signed-off-by: Jess Frazelle <github@jessfraz.com> * updates Signed-off-by: Jess Frazelle <github@jessfraz.com> --------- Signed-off-by: Jess Frazelle <github@jessfraz.com>
119 lines
3.8 KiB
Rust
119 lines
3.8 KiB
Rust
use schemars::JsonSchema;
|
|
use serde::{Deserialize, Serialize};
|
|
use tower_lsp::lsp_types::{Position as LspPosition, Range as LspRange};
|
|
|
|
use crate::modules::ModuleId;
|
|
|
|
/// The first two items are the start and end points (byte offsets from the start of the file).
|
|
/// The third item is whether the source range belongs to the 'main' file, i.e., the file currently
|
|
/// being rendered/displayed in the editor.
|
|
//
|
|
// Don't use a doc comment for the below since the above goes in the website docs.
|
|
// @see isTopLevelModule() in wasm.ts.
|
|
// TODO we need to handle modules better in the frontend.
|
|
#[derive(Debug, Default, Deserialize, Serialize, PartialEq, Copy, Clone, ts_rs::TS, JsonSchema, Hash, Eq)]
|
|
#[ts(export, type = "[number, number, number]")]
|
|
pub struct SourceRange([usize; 3]);
|
|
|
|
impl From<[usize; 3]> for SourceRange {
|
|
fn from(value: [usize; 3]) -> Self {
|
|
Self(value)
|
|
}
|
|
}
|
|
|
|
impl From<&SourceRange> for miette::SourceSpan {
|
|
fn from(source_range: &SourceRange) -> Self {
|
|
let length = source_range.end() - source_range.start();
|
|
let start = miette::SourceOffset::from(source_range.start());
|
|
Self::new(start, length)
|
|
}
|
|
}
|
|
|
|
impl From<SourceRange> for miette::SourceSpan {
|
|
fn from(source_range: SourceRange) -> Self {
|
|
Self::from(&source_range)
|
|
}
|
|
}
|
|
|
|
impl SourceRange {
|
|
/// Create a new source range.
|
|
pub fn new(start: usize, end: usize, module_id: ModuleId) -> Self {
|
|
Self([start, end, module_id.as_usize()])
|
|
}
|
|
|
|
/// A source range that doesn't correspond to any source code.
|
|
pub fn synthetic() -> Self {
|
|
Self::default()
|
|
}
|
|
|
|
/// True if this is a source range that doesn't correspond to any source
|
|
/// code.
|
|
pub fn is_synthetic(&self) -> bool {
|
|
self.start() == 0 && self.end() == 0
|
|
}
|
|
|
|
/// Get the start of the range.
|
|
pub fn start(&self) -> usize {
|
|
self.0[0]
|
|
}
|
|
|
|
/// Get the start of the range as a zero-length SourceRange, effectively collapse `self` to it's
|
|
/// start.
|
|
pub fn start_as_range(&self) -> Self {
|
|
Self([self.0[0], self.0[0], self.0[2]])
|
|
}
|
|
|
|
/// Get the end of the range.
|
|
pub fn end(&self) -> usize {
|
|
self.0[1]
|
|
}
|
|
|
|
/// Get the module ID of the range.
|
|
pub fn module_id(&self) -> ModuleId {
|
|
ModuleId::from_usize(self.0[2])
|
|
}
|
|
|
|
/// Check if the range contains a position.
|
|
pub fn contains(&self, pos: usize) -> bool {
|
|
pos >= self.start() && pos <= self.end()
|
|
}
|
|
|
|
pub fn start_to_lsp_position(&self, code: &str) -> LspPosition {
|
|
// Calculate the line and column of the error from the source range.
|
|
// Lines are zero indexed in vscode so we need to subtract 1.
|
|
let mut line = code.get(..self.start()).unwrap_or_default().lines().count();
|
|
if line > 0 {
|
|
line = line.saturating_sub(1);
|
|
}
|
|
let column = code[..self.start()].lines().last().map(|l| l.len()).unwrap_or_default();
|
|
|
|
LspPosition {
|
|
line: line as u32,
|
|
character: column as u32,
|
|
}
|
|
}
|
|
|
|
pub fn end_to_lsp_position(&self, code: &str) -> LspPosition {
|
|
let lines = code.get(..self.end()).unwrap_or_default().lines();
|
|
if lines.clone().count() == 0 {
|
|
return LspPosition { line: 0, character: 0 };
|
|
}
|
|
|
|
// Calculate the line and column of the error from the source range.
|
|
// Lines are zero indexed in vscode so we need to subtract 1.
|
|
let line = lines.clone().count() - 1;
|
|
let column = lines.last().map(|l| l.len()).unwrap_or_default();
|
|
|
|
LspPosition {
|
|
line: line as u32,
|
|
character: column as u32,
|
|
}
|
|
}
|
|
|
|
pub fn to_lsp_range(&self, code: &str) -> LspRange {
|
|
let start = self.start_to_lsp_position(code);
|
|
let end = self.end_to_lsp_position(code);
|
|
LspRange { start, end }
|
|
}
|
|
}
|