diff --git a/src/record.rs b/src/record.rs index bb39ffe..da74ca7 100644 --- a/src/record.rs +++ b/src/record.rs @@ -162,6 +162,13 @@ impl Record { pub fn checkpoint(&mut self) -> Checkpoint { Checkpoint::from(self) } + + /// Remove all elements after the index. + fn rm_tail(&mut self) -> VecDeque> { + // Check if the saved edit will be removed. + self.saved = self.saved.filter(|&saved| saved <= self.index); + self.entries.split_off(self.index) + } } impl Record { @@ -178,14 +185,12 @@ impl Record { ) -> (E::Output, bool, VecDeque>) { let output = edit.edit(target); // We store the state of the stack before adding the entry. - let index = self.index; + let old_index = self.index(); let could_undo = self.can_undo(); let could_redo = self.can_redo(); let was_saved = self.is_saved(); - // Pop off all elements after len from entries. - let tail = self.entries.split_off(index); - // Check if the saved state was popped off. - self.saved = self.saved.filter(|&saved| saved <= index); + + let tail = self.rm_tail(); // Try to merge edits unless the target is in a saved state. let merged = match self.entries.back_mut() { Some(last) if !was_saved => last.edit.merge(edit), @@ -216,6 +221,8 @@ impl Record { self.socket.emit_if(could_redo, || Signal::Redo(false)); self.socket.emit_if(!could_undo, || Signal::Undo(true)); self.socket.emit_if(was_saved, || Signal::Saved(false)); + self.socket + .emit_if(old_index != self.index, || Signal::Index(self.index)); (output, merged_or_annulled, tail) } @@ -224,15 +231,16 @@ impl Record { pub fn undo(&mut self, target: &mut E::Target) -> Option { self.can_undo().then(|| { let was_saved = self.is_saved(); - let old = self.index; + let old_index = self.index; let output = self.entries[self.index - 1].undo(target); self.index -= 1; let is_saved = self.is_saved(); - self.socket.emit_if(old == 1, || Signal::Undo(false)); + self.socket.emit_if(old_index == 1, || Signal::Undo(false)); self.socket - .emit_if(old == self.entries.len(), || Signal::Redo(true)); + .emit_if(old_index == self.entries.len(), || Signal::Redo(true)); self.socket .emit_if(was_saved != is_saved, || Signal::Saved(is_saved)); + self.socket.emit(|| Signal::Index(self.index)); output }) } @@ -242,15 +250,16 @@ impl Record { pub fn redo(&mut self, target: &mut E::Target) -> Option { self.can_redo().then(|| { let was_saved = self.is_saved(); - let old = self.index; + let old_index = self.index; let output = self.entries[self.index].redo(target); self.index += 1; let is_saved = self.is_saved(); - self.socket.emit_if(old == 0, || Signal::Undo(true)); + self.socket.emit_if(old_index == 0, || Signal::Undo(true)); self.socket - .emit_if(old == self.len() - 1, || Signal::Redo(false)); + .emit_if(old_index == self.len() - 1, || Signal::Redo(false)); self.socket .emit_if(was_saved != is_saved, || Signal::Saved(is_saved)); + self.socket.emit(|| Signal::Index(self.index)); output }) } @@ -269,6 +278,7 @@ impl Record { /// Removes all edits from the record without undoing them. pub fn clear(&mut self) { + let old_index = self.index; let could_undo = self.can_undo(); let could_redo = self.can_redo(); self.entries.clear(); @@ -276,6 +286,7 @@ impl Record { self.index = 0; self.socket.emit_if(could_undo, || Signal::Undo(false)); self.socket.emit_if(could_redo, || Signal::Redo(false)); + self.socket.emit_if(old_index != 0, || Signal::Index(0)); } /// Revert the changes done to the target since the saved state. @@ -290,6 +301,7 @@ impl Record { return Vec::new(); } + let old_index = self.index(); let could_undo = self.can_undo(); let could_redo = self.can_redo(); let was_saved = self.is_saved(); @@ -319,6 +331,8 @@ impl Record { .emit_if(could_redo != can_redo, || Signal::Redo(can_redo)); self.socket .emit_if(was_saved != is_saved, || Signal::Saved(is_saved)); + self.socket + .emit_if(old_index != self.index, || Signal::Index(self.index)); outputs } diff --git a/src/socket.rs b/src/socket.rs index 59f67f9..070e218 100644 --- a/src/socket.rs +++ b/src/socket.rs @@ -51,12 +51,14 @@ impl Socket { #[derive(Clone, Debug, PartialEq)] #[non_exhaustive] pub enum Signal { - /// Says if the structures can undo. + /// Emitted when the structures ability to undo has changed. Undo(bool), - /// Says if the structures can redo. + /// Emitted when the structures ability to redo has changed. Redo(bool), - /// Says if the target is in a saved state. + /// Emitted when the saved state has changed. Saved(bool), + /// Emitted when the index has changed. + Index(usize), } /// Use this to handle signals emitted. @@ -80,12 +82,14 @@ pub enum Signal { /// record.edit(&mut target, Add('a')); /// assert_eq!(iter.next(), Some(Signal::Undo(true))); /// assert_eq!(iter.next(), Some(Signal::Saved(false))); +/// assert_eq!(iter.next(), Some(Signal::Index(1))); /// assert_eq!(iter.next(), None); /// /// record.undo(&mut target); /// assert_eq!(iter.next(), Some(Signal::Undo(false))); /// assert_eq!(iter.next(), Some(Signal::Redo(true))); /// assert_eq!(iter.next(), Some(Signal::Saved(true))); +/// assert_eq!(iter.next(), Some(Signal::Index(0))); /// assert_eq!(iter.next(), None); /// # } /// ```