oxc/napi/parser/src/magic_string.rs

170 lines
4.4 KiB
Rust

#![allow(clippy::cast_possible_truncation)]
// use std::sync::Arc;
use napi_derive::napi;
use oxc_data_structures::rope::{get_line_column, Rope};
// use oxc_sourcemap::napi::SourceMap;
use self_cell::self_cell;
use string_wizard::MagicString as MS;
#[napi]
pub struct MagicString {
cell: MagicStringImpl,
rope: Option<Rope>,
}
self_cell!(
struct MagicStringImpl {
owner: String,
#[covariant]
dependent: MS,
}
);
impl MagicString {
pub fn new(source_text: String) -> Self {
Self {
cell: MagicStringImpl::new(source_text, |s| string_wizard::MagicString::new(s)),
rope: None,
}
}
}
#[napi(object)]
pub struct LineColumn {
pub line: u32,
pub column: u32,
}
#[napi(object)]
pub struct OverwriteOptions {
pub content_only: bool,
}
#[napi(object)]
pub struct SourceMapOptions {
pub include_content: Option<bool>,
pub source: Option<String>,
pub hires: Option<bool>,
}
#[napi]
impl MagicString {
/// Get source text from utf8 offset.
#[napi]
pub fn get_source_text(&self, start: u32, end: u32) -> &str {
&self.cell.borrow_owner()[start as usize..end as usize]
}
/// Get 0-based line and column number from utf8 offset.
#[napi]
pub fn get_line_column_number(&mut self, offset: u32) -> LineColumn {
let source_text = self.cell.borrow_owner();
let rope = self.rope.get_or_insert_with(|| Rope::from_str(source_text));
let (line, column) = get_line_column(rope, offset, source_text);
LineColumn { line, column }
}
/// Get UTF16 byte offset from UTF8 byte offset.
#[napi]
pub fn get_utf16_byte_offset(&mut self, offset: u32) -> u32 {
let source_text = self.cell.borrow_owner();
// TODO(perf): this is obviously slow ...
source_text[..offset as usize].encode_utf16().count() as u32
}
#[napi]
pub fn length(&self) -> u32 {
self.cell.borrow_dependent().len() as u32
}
#[napi]
#[allow(clippy::inherent_to_string)]
pub fn to_string(&self) -> String {
self.cell.borrow_dependent().to_string()
}
// #[napi]
// pub fn source_map(&self, options: Option<SourceMapOptions>) -> SourceMap {
// let options = options.map(|o| string_wizard::SourceMapOptions {
// include_content: o.include_content.unwrap_or_default(),
// source: o.source.map(Arc::from).unwrap_or_default(),
// hires: o.hires.unwrap_or_default(),
// });
// let map = self.cell.borrow_dependent().source_map(options.unwrap_or_default());
// oxc_sourcemap::napi::SourceMap::from(map)
// }
#[napi]
pub fn append(&mut self, input: String) -> &Self {
self.cell.with_dependent_mut(|_, ms| {
ms.append(input);
});
self
}
#[napi]
pub fn append_left(&mut self, index: u32, input: String) -> &Self {
self.cell.with_dependent_mut(|_, ms| {
ms.append_left(index as usize, input);
});
self
}
#[napi]
pub fn append_right(&mut self, index: u32, input: String) -> &Self {
self.cell.with_dependent_mut(|_, ms| {
ms.append_right(index as usize, input);
});
self
}
#[napi]
pub fn indent(&mut self) -> &Self {
self.cell.with_dependent_mut(|_, ms| {
ms.indent();
});
self
}
#[napi]
pub fn prepend(&mut self, input: String) -> &Self {
self.cell.with_dependent_mut(|_, ms| {
ms.prepend(input);
});
self
}
#[napi]
pub fn prepend_left(&mut self, index: u32, input: String) -> &Self {
self.cell.with_dependent_mut(|_, ms| {
ms.prepend_left(index as usize, input);
});
self
}
#[napi]
pub fn prepend_right(&mut self, index: u32, input: String) -> &Self {
self.cell.with_dependent_mut(|_, ms| {
ms.prepend_right(index as usize, input);
});
self
}
#[napi]
pub fn relocate(&mut self, start: u32, end: u32, to: u32) -> &Self {
self.cell.with_dependent_mut(|_, ms| {
ms.relocate(start as usize, end as usize, to as usize);
});
self
}
#[napi]
pub fn remove(&mut self, start: u32, end: u32) -> &Self {
self.cell.with_dependent_mut(|_, ms| {
ms.remove(start as usize, end as usize);
});
self
}
}