feat: scroll half/full page with arrow percentage supported, and new Vi-like <C-u>, <C-d>, <C-b>, and <C-f> keybindings added (#213)

This commit is contained in:
Collide 2023-09-24 17:27:10 +08:00 committed by GitHub
parent d3ed8e7cf8
commit f7fdda9d9b
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
6 changed files with 80 additions and 18 deletions

View File

@ -61,7 +61,7 @@ impl Executor {
// Navigation
"arrow" => {
let step = exec.args.get(0).and_then(|s| s.parse().ok()).unwrap_or(0);
let step = exec.args.get(0).and_then(|s| s.parse().ok()).unwrap_or_default();
cx.manager.active_mut().arrow(step)
}
"peek" => {

View File

@ -13,6 +13,11 @@ keymap = [
{ on = [ "K" ], exec = "arrow -5", desc = "Move cursor up 5 lines" },
{ on = [ "J" ], exec = "arrow 5", desc = "Move cursor down 5 lines" },
{ on = [ "<C-u>" ], exec = "arrow -50%", desc = "Move cursor up half page" },
{ on = [ "<C-d>" ], exec = "arrow 50%", desc = "Move cursor down half page" },
{ on = [ "<C-b>" ], exec = "arrow -100%", desc = "Move cursor up one page" },
{ on = [ "<C-f>" ], exec = "arrow 100%", desc = "Move cursor down one page" },
{ on = [ "h" ], exec = "leave", desc = "Go back to the parent directory" },
{ on = [ "l" ], exec = "enter", desc = "Enter the child directory" },

View File

@ -14,8 +14,9 @@ pub mod help;
mod highlighter;
pub mod input;
pub mod manager;
pub mod position;
mod position;
pub mod select;
mod step;
pub mod tasks;
pub mod which;
@ -23,5 +24,6 @@ pub use blocker::*;
pub use event::*;
pub use highlighter::*;
pub use position::*;
pub use step::*;
pub fn init() { init_blocker(); }

View File

@ -2,7 +2,7 @@ use config::MANAGER;
use ratatui::layout::Rect;
use shared::Url;
use crate::{emit, files::{File, Files, FilesOp}};
use crate::{emit, files::{File, Files, FilesOp}, Step};
#[derive(Default)]
pub struct Folder {
@ -58,18 +58,18 @@ impl Folder {
true
}
pub fn next(&mut self, step: usize) -> bool {
pub fn next(&mut self, step: Step) -> bool {
let len = self.files.len();
if len == 0 {
return false;
}
let old = self.cursor;
self.cursor = (self.cursor + step).min(len - 1);
let limit = MANAGER.layout.folder_height();
self.cursor = step.add(self.cursor, || limit).min(len - 1);
self.hovered = self.files.duplicate(self.cursor);
self.set_page(false);
let limit = MANAGER.layout.folder_height();
if self.cursor >= (self.offset + limit).min(len).saturating_sub(5) {
self.offset = len.saturating_sub(limit).min(self.offset + self.cursor - old);
}
@ -77,9 +77,9 @@ impl Folder {
old != self.cursor
}
pub fn prev(&mut self, step: usize) -> bool {
pub fn prev(&mut self, step: Step) -> bool {
let old = self.cursor;
self.cursor = self.cursor.saturating_sub(step);
self.cursor = step.add(self.cursor, || MANAGER.layout.folder_height());
self.hovered = self.files.duplicate(self.cursor);
self.set_page(false);
@ -105,7 +105,11 @@ impl Folder {
pub fn hover(&mut self, url: &Url) -> bool {
let new = self.files.position(url).unwrap_or(self.cursor);
if new > self.cursor { self.next(new - self.cursor) } else { self.prev(self.cursor - new) }
if new > self.cursor {
self.next(Step::from(new - self.cursor))
} else {
self.prev(Step::from(self.cursor - new))
}
}
#[inline]

View File

@ -7,7 +7,7 @@ use tokio::{pin, task::JoinHandle};
use tokio_stream::{wrappers::UnboundedReceiverStream, StreamExt};
use super::{Finder, Folder, Mode, Preview, PreviewLock};
use crate::{emit, external::{self, FzfOpt, ZoxideOpt}, files::{File, FilesOp, FilesSorter}, input::InputOpt, Event, BLOCKER};
use crate::{emit, external::{self, FzfOpt, ZoxideOpt}, files::{File, FilesOp, FilesSorter}, input::InputOpt, Event, Step, BLOCKER};
pub struct Tab {
pub(super) mode: Mode,
@ -67,12 +67,8 @@ impl Tab {
self.search_stop()
}
pub fn arrow(&mut self, step: isize) -> bool {
let ok = if step > 0 {
self.current.next(step as usize)
} else {
self.current.prev(step.unsigned_abs())
};
pub fn arrow(&mut self, step: Step) -> bool {
let ok = if step.is_positive() { self.current.next(step) } else { self.current.prev(step) };
if !ok {
return false;
}
@ -248,7 +244,7 @@ impl Tab {
};
if let Some(step) = finder.ring(&self.current.files, self.current.cursor(), prev) {
self.arrow(step);
self.arrow(step.into());
}
self.finder = Some(finder);
@ -280,7 +276,7 @@ impl Tab {
let mut b = finder.catchup(&self.current.files);
if let Some(step) = finder.arrow(&self.current.files, self.current.cursor(), prev) {
b |= self.arrow(step);
b |= self.arrow(step.into());
}
b

55
core/src/step.rs Normal file
View File

@ -0,0 +1,55 @@
use std::{num::ParseIntError, str::FromStr};
pub enum Step {
Fixed(isize),
Percent(i8),
}
impl Default for Step {
fn default() -> Self { Self::Fixed(0) }
}
impl FromStr for Step {
type Err = ParseIntError;
fn from_str(s: &str) -> Result<Self, Self::Err> {
Ok(if let Some(s) = s.strip_suffix('%') {
Self::Percent(s.parse()?)
} else {
Self::Fixed(s.parse()?)
})
}
}
impl From<isize> for Step {
fn from(n: isize) -> Self { Self::Fixed(n) }
}
impl From<usize> for Step {
fn from(n: usize) -> Self { Self::Fixed(n as isize) }
}
impl Step {
#[inline]
fn fixed<F: FnOnce() -> usize>(self, f: F) -> isize {
match self {
Self::Fixed(n) => n,
Self::Percent(0) => 0,
Self::Percent(n) => n as isize * f() as isize / 100,
}
}
#[inline]
pub fn add<F: FnOnce() -> usize>(self, pos: usize, f: F) -> usize {
let fixed = self.fixed(f);
if fixed > 0 { pos + fixed as usize } else { pos.saturating_sub(fixed.unsigned_abs()) }
}
#[inline]
pub fn is_positive(&self) -> bool {
match *self {
Self::Fixed(n) => n > 0,
Self::Percent(n) => n > 0,
}
}
}