ruff/crates/ruff_text_size/src/traits.rs

99 lines
2 KiB
Rust

use std::sync::Arc;
use {crate::TextRange, crate::TextSize, std::convert::TryInto};
use priv_in_pub::Sealed;
mod priv_in_pub {
pub trait Sealed {}
}
/// Primitives with a textual length that can be passed to [`TextSize::of`].
pub trait TextLen: Copy + Sealed {
/// The textual length of this primitive.
fn text_len(self) -> TextSize;
}
impl Sealed for &'_ str {}
impl TextLen for &'_ str {
#[inline]
fn text_len(self) -> TextSize {
self.len().try_into().unwrap()
}
}
impl Sealed for &'_ String {}
impl TextLen for &'_ String {
#[inline]
fn text_len(self) -> TextSize {
self.as_str().text_len()
}
}
impl Sealed for char {}
impl TextLen for char {
#[inline]
#[allow(clippy::cast_possible_truncation)]
fn text_len(self) -> TextSize {
(self.len_utf8() as u32).into()
}
}
/// A ranged item in the source text.
pub trait Ranged {
/// The range of this item in the source text.
fn range(&self) -> TextRange;
/// The start offset of this item in the source text.
fn start(&self) -> TextSize {
self.range().start()
}
/// The end offset of this item in the source text.
fn end(&self) -> TextSize {
self.range().end()
}
}
impl Ranged for TextRange {
fn range(&self) -> TextRange {
*self
}
}
impl<T> Ranged for &T
where
T: Ranged,
{
fn range(&self) -> TextRange {
T::range(self)
}
}
impl<T> Ranged for Arc<T>
where
T: Ranged,
{
fn range(&self) -> TextRange {
T::range(self)
}
}
/// A slice of the source text.
pub trait TextSlice: Sealed {
/// Returns the slice of the text within the given `range`.
///
/// ## Note
///
/// This is the same as `&self[range]` if `self` is a `str` and `range` a `TextRange`.
///
/// ## Panics
/// If the range is out of bounds.
fn slice(&self, range: impl Ranged) -> &str;
}
impl Sealed for str {}
impl TextSlice for str {
fn slice(&self, ranged: impl Ranged) -> &str {
&self[ranged.range()]
}
}