2021-12-10 11:21:25 +08:00

81 lines
1.8 KiB
Rust

use lib_ot::rich_text::RichTextDelta;
const MAX_UNDOS: usize = 20;
#[derive(Debug, Clone)]
pub struct UndoResult {
#[allow(dead_code)]
success: bool,
#[allow(dead_code)]
len: usize,
}
impl UndoResult {
pub fn fail() -> Self { UndoResult { success: false, len: 0 } }
pub fn success(len: usize) -> Self { UndoResult { success: true, len } }
}
#[derive(Debug, Clone)]
pub struct History {
#[allow(dead_code)]
cur_undo: usize,
undos: Vec<RichTextDelta>,
redoes: Vec<RichTextDelta>,
capacity: usize,
}
impl std::default::Default for History {
fn default() -> Self {
History {
cur_undo: 1,
undos: Vec::new(),
redoes: Vec::new(),
capacity: MAX_UNDOS,
}
}
}
impl History {
pub fn new() -> Self { History::default() }
pub fn can_undo(&self) -> bool { !self.undos.is_empty() }
pub fn can_redo(&self) -> bool { !self.redoes.is_empty() }
pub fn add_undo(&mut self, delta: RichTextDelta) { self.undos.push(delta); }
pub fn add_redo(&mut self, delta: RichTextDelta) { self.redoes.push(delta); }
pub fn record(&mut self, delta: RichTextDelta) {
if delta.ops.is_empty() {
return;
}
self.redoes.clear();
self.add_undo(delta);
if self.undos.len() > self.capacity {
self.undos.remove(0);
}
}
pub fn undo(&mut self) -> Option<RichTextDelta> {
if !self.can_undo() {
return None;
}
let delta = self.undos.pop().unwrap();
Some(delta)
}
pub fn redo(&mut self) -> Option<RichTextDelta> {
if !self.can_redo() {
return None;
}
let delta = self.redoes.pop().unwrap();
Some(delta)
}
}