rust/crates/rust-analyzer/src/from_proto.rs

122 lines
4.4 KiB
Rust
Raw Normal View History

//! Conversion lsp_types types to rust-analyzer specific ones.
2020-06-11 11:04:09 +02:00
use std::convert::TryFrom;
2021-02-13 00:55:27 +03:00
use ide::{Annotation, AnnotationKind, AssistKind, LineCol, LineColUtf16};
2020-10-24 11:39:57 +03:00
use ide_db::base_db::{FileId, FilePosition, FileRange};
2020-08-12 18:26:51 +02:00
use syntax::{TextRange, TextSize};
2020-06-11 11:04:09 +02:00
use vfs::AbsPathBuf;
use crate::{
2021-02-13 00:55:27 +03:00
from_json,
global_state::GlobalStateSnapshot,
2021-02-13 01:28:48 +03:00
line_index::{LineIndex, OffsetEncoding},
lsp_ext, LspError, Result,
};
2020-06-11 11:04:09 +02:00
pub(crate) fn abs_path(url: &lsp_types::Url) -> Result<AbsPathBuf> {
let path = url.to_file_path().map_err(|()| "url is not a file")?;
Ok(AbsPathBuf::try_from(path).unwrap())
}
pub(crate) fn vfs_path(url: &lsp_types::Url) -> Result<vfs::VfsPath> {
abs_path(url).map(vfs::VfsPath::from)
}
pub(crate) fn offset(line_index: &LineIndex, position: lsp_types::Position) -> TextSize {
2021-02-13 00:55:27 +03:00
let line_col = match line_index.encoding {
OffsetEncoding::Utf8 => {
LineCol { line: position.line as u32, col: position.character as u32 }
}
OffsetEncoding::Utf16 => {
let line_col =
LineColUtf16 { line: position.line as u32, col: position.character as u32 };
line_index.index.to_utf8(line_col)
}
};
line_index.index.offset(line_col)
}
pub(crate) fn text_range(line_index: &LineIndex, range: lsp_types::Range) -> TextRange {
let start = offset(line_index, range.start);
let end = offset(line_index, range.end);
TextRange::new(start, end)
}
pub(crate) fn file_id(snap: &GlobalStateSnapshot, url: &lsp_types::Url) -> Result<FileId> {
snap.url_to_file_id(url)
}
pub(crate) fn file_position(
snap: &GlobalStateSnapshot,
tdpp: lsp_types::TextDocumentPositionParams,
) -> Result<FilePosition> {
let file_id = file_id(snap, &tdpp.text_document.uri)?;
let line_index = snap.file_line_index(file_id)?;
let offset = offset(&line_index, tdpp.position);
Ok(FilePosition { file_id, offset })
}
pub(crate) fn file_range(
snap: &GlobalStateSnapshot,
text_document_identifier: lsp_types::TextDocumentIdentifier,
range: lsp_types::Range,
) -> Result<FileRange> {
let file_id = file_id(snap, &text_document_identifier.uri)?;
let line_index = snap.file_line_index(file_id)?;
let range = text_range(&line_index, range);
Ok(FileRange { file_id, range })
}
2020-07-13 17:41:47 -04:00
pub(crate) fn assist_kind(kind: lsp_types::CodeActionKind) -> Option<AssistKind> {
let assist_kind = match &kind {
k if k == &lsp_types::CodeActionKind::EMPTY => AssistKind::None,
k if k == &lsp_types::CodeActionKind::QUICKFIX => AssistKind::QuickFix,
k if k == &lsp_types::CodeActionKind::REFACTOR => AssistKind::Refactor,
2020-07-15 08:17:32 -04:00
k if k == &lsp_types::CodeActionKind::REFACTOR_EXTRACT => AssistKind::RefactorExtract,
k if k == &lsp_types::CodeActionKind::REFACTOR_INLINE => AssistKind::RefactorInline,
k if k == &lsp_types::CodeActionKind::REFACTOR_REWRITE => AssistKind::RefactorRewrite,
2020-07-13 17:41:47 -04:00
_ => return None,
};
Some(assist_kind)
}
2021-02-13 13:07:47 +02:00
pub(crate) fn annotation(
snap: &GlobalStateSnapshot,
2021-02-13 13:07:47 +02:00
code_lens: lsp_types::CodeLens,
) -> Result<Annotation> {
let data = code_lens.data.ok_or_else(|| LspError {
code: lsp_server::ErrorCode::InvalidParams as i32,
message: "code lens without data".to_string(),
});
2021-02-13 13:07:47 +02:00
let resolve = from_json::<lsp_ext::CodeLensResolveData>("CodeLensResolveData", data)?;
match resolve {
lsp_ext::CodeLensResolveData::Impls(params) => {
let file_id =
snap.url_to_file_id(&params.text_document_position_params.text_document.uri)?;
let line_index = snap.file_line_index(file_id)?;
2021-02-13 13:07:47 +02:00
Ok(Annotation {
range: text_range(&line_index, code_lens.range),
kind: AnnotationKind::HasImpls {
position: file_position(snap, params.text_document_position_params)?,
2021-02-13 13:07:47 +02:00
data: None,
},
})
}
lsp_ext::CodeLensResolveData::References(params) => {
let file_id = snap.url_to_file_id(&params.text_document.uri)?;
let line_index = snap.file_line_index(file_id)?;
2021-02-13 13:07:47 +02:00
Ok(Annotation {
range: text_range(&line_index, code_lens.range),
kind: AnnotationKind::HasReferences {
position: file_position(snap, params)?,
2021-02-13 13:07:47 +02:00
data: None,
},
})
}
}
}