123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337 |
- use crate::{
- core::{FlowyStr, Interval, OpBuilder, OperationTransformable},
- rich_text::{RichTextAttribute, RichTextAttributes},
- };
- use serde::__private::Formatter;
- use std::{
- cmp::min,
- fmt,
- fmt::Debug,
- ops::{Deref, DerefMut},
- };
- pub trait Attributes: fmt::Display + Eq + PartialEq + Default + Clone + Debug + OperationTransformable {
- fn is_empty(&self) -> bool;
- // Remove the empty attribute which value is None.
- fn remove_empty(&mut self);
- fn extend_other(&mut self, other: Self);
- }
- pub type RichTextOperation = Operation<RichTextAttributes>;
- impl RichTextOperation {
- pub fn contain_attribute(&self, attribute: &RichTextAttribute) -> bool {
- self.get_attributes().contains_key(&attribute.key)
- }
- }
- #[derive(Debug, Clone, Eq, PartialEq)]
- pub enum Operation<T: Attributes> {
- Delete(usize),
- Retain(Retain<T>),
- Insert(Insert<T>),
- }
- impl<T> Operation<T>
- where
- T: Attributes,
- {
- pub fn get_data(&self) -> &str {
- match self {
- Operation::Delete(_) => "",
- Operation::Retain(_) => "",
- Operation::Insert(insert) => &insert.s,
- }
- }
- pub fn get_attributes(&self) -> T {
- match self {
- Operation::Delete(_) => T::default(),
- Operation::Retain(retain) => retain.attributes.clone(),
- Operation::Insert(insert) => insert.attributes.clone(),
- }
- }
- pub fn set_attributes(&mut self, attributes: T) {
- match self {
- Operation::Delete(_) => log::error!("Delete should not contains attributes"),
- Operation::Retain(retain) => retain.attributes = attributes,
- Operation::Insert(insert) => insert.attributes = attributes,
- }
- }
- pub fn has_attribute(&self) -> bool { !self.get_attributes().is_empty() }
- pub fn len(&self) -> usize {
- match self {
- Operation::Delete(n) => *n,
- Operation::Retain(r) => r.n,
- Operation::Insert(i) => i.count_of_utf16_code_units(),
- }
- }
- pub fn is_empty(&self) -> bool { self.len() == 0 }
- #[allow(dead_code)]
- pub fn split(&self, index: usize) -> (Option<Operation<T>>, Option<Operation<T>>) {
- debug_assert!(index < self.len());
- let left;
- let right;
- match self {
- Operation::Delete(n) => {
- left = Some(OpBuilder::<T>::delete(index).build());
- right = Some(OpBuilder::<T>::delete(*n - index).build());
- },
- Operation::Retain(retain) => {
- left = Some(OpBuilder::<T>::delete(index).build());
- right = Some(OpBuilder::<T>::delete(retain.n - index).build());
- },
- Operation::Insert(insert) => {
- let attributes = self.get_attributes();
- left = Some(
- OpBuilder::<T>::insert(&insert.s[0..index])
- .attributes(attributes.clone())
- .build(),
- );
- right = Some(
- OpBuilder::<T>::insert(&insert.s[index..insert.count_of_utf16_code_units()])
- .attributes(attributes)
- .build(),
- );
- },
- }
- (left, right)
- }
- pub fn shrink(&self, interval: Interval) -> Option<Operation<T>> {
- let op = match self {
- Operation::Delete(n) => OpBuilder::delete(min(*n, interval.size())).build(),
- Operation::Retain(retain) => OpBuilder::retain(min(retain.n, interval.size()))
- .attributes(retain.attributes.clone())
- .build(),
- Operation::Insert(insert) => {
- if interval.start > insert.count_of_utf16_code_units() {
- OpBuilder::insert("").build()
- } else {
- // let s = &insert
- // .s
- // .chars()
- // .skip(interval.start)
- // .take(min(interval.size(), insert.count_of_code_units()))
- // .collect::<String>();
- let s = insert.s.sub_str(interval);
- OpBuilder::insert(&s).attributes(insert.attributes.clone()).build()
- }
- },
- };
- match op.is_empty() {
- true => None,
- false => Some(op),
- }
- }
- pub fn is_delete(&self) -> bool {
- if let Operation::Delete(_) = self {
- return true;
- }
- false
- }
- pub fn is_insert(&self) -> bool {
- if let Operation::Insert(_) = self {
- return true;
- }
- false
- }
- pub fn is_retain(&self) -> bool {
- if let Operation::Retain(_) = self {
- return true;
- }
- false
- }
- pub fn is_plain(&self) -> bool {
- match self {
- Operation::Delete(_) => true,
- Operation::Retain(retain) => retain.is_plain(),
- Operation::Insert(insert) => insert.is_plain(),
- }
- }
- }
- impl<T> fmt::Display for Operation<T>
- where
- T: Attributes,
- {
- fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
- f.write_str("{")?;
- match self {
- Operation::Delete(n) => {
- f.write_fmt(format_args!("delete: {}", n))?;
- },
- Operation::Retain(r) => {
- f.write_fmt(format_args!("{}", r))?;
- },
- Operation::Insert(i) => {
- f.write_fmt(format_args!("{}", i))?;
- },
- }
- f.write_str("}")?;
- Ok(())
- }
- }
- #[derive(Clone, Debug, Eq, PartialEq)]
- pub struct Retain<T: Attributes> {
- // #[serde(rename(serialize = "retain", deserialize = "retain"))]
- pub n: usize,
- // #[serde(skip_serializing_if = "is_empty")]
- pub attributes: T,
- }
- impl<T> fmt::Display for Retain<T>
- where
- T: Attributes,
- {
- fn fmt(&self, f: &mut Formatter<'_>) -> fmt::Result {
- if self.attributes.is_empty() {
- f.write_fmt(format_args!("retain: {}", self.n))
- } else {
- f.write_fmt(format_args!("retain: {}, attributes: {}", self.n, self.attributes))
- }
- }
- }
- impl<T> Retain<T>
- where
- T: Attributes,
- {
- pub fn merge_or_new(&mut self, n: usize, attributes: T) -> Option<Operation<T>> {
- tracing::trace!(
- "merge_retain_or_new_op: len: {:?}, l: {} - r: {}",
- n,
- self.attributes,
- attributes
- );
- if self.attributes == attributes {
- self.n += n;
- None
- } else {
- Some(OpBuilder::retain(n).attributes(attributes).build())
- }
- }
- pub fn is_plain(&self) -> bool { self.attributes.is_empty() }
- }
- impl<T> std::convert::From<usize> for Retain<T>
- where
- T: Attributes,
- {
- fn from(n: usize) -> Self {
- Retain {
- n,
- attributes: T::default(),
- }
- }
- }
- impl<T> Deref for Retain<T>
- where
- T: Attributes,
- {
- type Target = usize;
- fn deref(&self) -> &Self::Target { &self.n }
- }
- impl<T> DerefMut for Retain<T>
- where
- T: Attributes,
- {
- fn deref_mut(&mut self) -> &mut Self::Target { &mut self.n }
- }
- #[derive(Clone, Debug, Eq, PartialEq)]
- pub struct Insert<T: Attributes> {
- // #[serde(rename(serialize = "insert", deserialize = "insert"))]
- pub s: FlowyStr,
- // #[serde(skip_serializing_if = "is_empty")]
- pub attributes: T,
- }
- impl<T> fmt::Display for Insert<T>
- where
- T: Attributes,
- {
- fn fmt(&self, f: &mut Formatter<'_>) -> fmt::Result {
- let mut s = self.s.clone();
- if s.ends_with('\n') {
- s.pop();
- if s.is_empty() {
- s = "new_line".into();
- }
- }
- if self.attributes.is_empty() {
- f.write_fmt(format_args!("insert: {}", s))
- } else {
- f.write_fmt(format_args!("insert: {}, attributes: {}", s, self.attributes))
- }
- }
- }
- impl<T> Insert<T>
- where
- T: Attributes,
- {
- pub fn count_of_utf16_code_units(&self) -> usize { self.s.utf16_size() }
- pub fn merge_or_new_op(&mut self, s: &str, attributes: T) -> Option<Operation<T>> {
- if self.attributes == attributes {
- self.s += s;
- None
- } else {
- Some(OpBuilder::<T>::insert(s).attributes(attributes).build())
- }
- }
- pub fn is_plain(&self) -> bool { self.attributes.is_empty() }
- }
- impl<T> std::convert::From<String> for Insert<T>
- where
- T: Attributes,
- {
- fn from(s: String) -> Self {
- Insert {
- s: s.into(),
- attributes: T::default(),
- }
- }
- }
- impl<T> std::convert::From<&str> for Insert<T>
- where
- T: Attributes,
- {
- fn from(s: &str) -> Self { Insert::from(s.to_owned()) }
- }
- impl<T> std::convert::From<FlowyStr> for Insert<T>
- where
- T: Attributes,
- {
- fn from(s: FlowyStr) -> Self {
- Insert {
- s,
- attributes: T::default(),
- }
- }
- }
|