Fix jump_backwards behaviour when jumplist is at capacity (#10968)

* Fix jump_backwards behaviour when jumplist is at capacity

* Decrement self.current while popping from front

* Fix issue with conflicting updates to self.current

* Realised that truncate is intentional

* Use saturating_sub when decrementing current

* Fix naming of previous jump, and remove unneeded comment change

* Remove unnecessary changes in push

* Return num elements removed from front, and use in backward method

* Hide num_removed from public interface and tidy up jump location check
This commit is contained in:
Thomas Schafer 2024-06-18 09:42:46 +01:00 committed by GitHub
parent 69acf66cd8
commit 668f1239a9
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194

View File

@ -38,18 +38,25 @@ pub fn new(initial: Jump) -> Self {
Self { jumps, current: 0 }
}
pub fn push(&mut self, jump: Jump) {
fn push_impl(&mut self, jump: Jump) -> usize {
let mut num_removed_from_front = 0;
self.jumps.truncate(self.current);
// don't push duplicates
if self.jumps.back() != Some(&jump) {
// If the jumplist is full, drop the oldest item.
while self.jumps.len() >= JUMP_LIST_CAPACITY {
self.jumps.pop_front();
num_removed_from_front += 1;
}
self.jumps.push_back(jump);
self.current = self.jumps.len();
}
num_removed_from_front
}
pub fn push(&mut self, jump: Jump) {
self.push_impl(jump);
}
pub fn forward(&mut self, count: usize) -> Option<&Jump> {
@ -63,13 +70,22 @@ pub fn forward(&mut self, count: usize) -> Option<&Jump> {
// Taking view and doc to prevent unnecessary cloning when jump is not required.
pub fn backward(&mut self, view_id: ViewId, doc: &mut Document, count: usize) -> Option<&Jump> {
if let Some(current) = self.current.checked_sub(count) {
if let Some(mut current) = self.current.checked_sub(count) {
if self.current == self.jumps.len() {
let jump = (doc.id(), doc.selection(view_id).clone());
self.push(jump);
let num_removed = self.push_impl(jump);
current = current.saturating_sub(num_removed);
}
self.current = current;
// Avoid jumping to the current location.
let jump @ (doc_id, selection) = self.jumps.get(self.current)?;
if doc.id() == *doc_id && doc.selection(view_id) == selection {
self.current = self.current.checked_sub(1)?;
self.jumps.get(self.current)
} else {
Some(jump)
}
} else {
None
}