Skip to content

Commit

Permalink
feat: more complete hovered file cursor tracking (#2218)
Browse files Browse the repository at this point in the history
  • Loading branch information
sxyazi authored Jan 18, 2025
1 parent e37d8d6 commit ef48b7f
Show file tree
Hide file tree
Showing 7 changed files with 42 additions and 48 deletions.
16 changes: 4 additions & 12 deletions yazi-core/src/manager/commands/hover.rs
Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
use std::{collections::HashSet, path::PathBuf};
use std::collections::HashSet;

use yazi_dds::Pubsub;
use yazi_macro::render;
Expand Down Expand Up @@ -26,7 +26,7 @@ impl Manager {
if let Some(u) = opt.url {
self.hover_do(u, opt.tab);
} else {
self.current_or_mut(opt.tab).repos(None);
self.current_or_mut(opt.tab).arrow(0);
}

// Repeek
Expand All @@ -50,16 +50,8 @@ impl Manager {
}

fn hover_do(&mut self, url: Url, tab: Option<Id>) {
// Hover on the file
if let Ok(p) = url.strip_prefix(&self.current_or(tab).url).map(PathBuf::from) {
render!(self.current_or_mut(tab).repos(Some(Urn::new(&p))));
}

// Turn on tracing
if self.current_or(tab).hovered().is_some_and(|f| url == f.url) {
// `hover(Some)` occurs after user actions, such as create, rename, reveal, etc.
// At this point, it's intuitive to track the location of this file regardless.
self.current_or_mut(tab).tracing = true;
if let Ok(p) = url.strip_prefix(&self.current_or(tab).url) {
render!(self.current_or_mut(tab).hover(Urn::new(p)));
}
}
}
18 changes: 7 additions & 11 deletions yazi-core/src/manager/commands/update_files.rs
Original file line number Diff line number Diff line change
Expand Up @@ -70,16 +70,12 @@ impl Manager {
}

fn update_current(tab: &mut Tab, op: Cow<FilesOp>, tasks: &Tasks) {
let hovered = tab.hovered().filter(|_| tab.current.tracing).map(|h| h.urn_owned());
let calc = !matches!(*op, FilesOp::Size(..) | FilesOp::Deleting(..));

let foreign = matches!(op, Cow::Borrowed(_));

if !tab.current.update_pub(tab.id, op.into_owned()) {
return;
}

tab.current.repos(hovered.as_ref().map(|u| u.as_urn()));
if foreign {
} else if foreign {
return;
}

Expand Down Expand Up @@ -109,11 +105,11 @@ impl Manager {
|(p, n)| matches!(*op, FilesOp::Deleting(ref parent, ref urns) if *parent == p && urns.contains(n)),
);

let folder = tab.history.entry(op.cwd().clone()).or_insert_with(|| Folder::from(op.cwd()));
let hovered = folder.hovered().filter(|_| folder.tracing).map(|h| h.urn_owned());
if folder.update_pub(tab.id, op.into_owned()) {
folder.repos(hovered.as_ref().map(|u| u.as_urn()));
}
tab
.history
.entry(op.cwd().clone())
.or_insert_with(|| Folder::from(op.cwd()))
.update_pub(tab.id, op.into_owned());

if leave {
tab.leave(());
Expand Down
2 changes: 1 addition & 1 deletion yazi-core/src/tab/commands/filter_do.rs
Original file line number Diff line number Diff line change
Expand Up @@ -25,7 +25,7 @@ impl Tab {
return;
}

self.current.repos(hovered.as_ref().map(|u| u.as_urn()));
self.current.repos(hovered.as_ref());
if self.hovered().map(|f| f.urn()) != hovered.as_ref().map(|u| u.as_urn()) {
ManagerProxy::hover(None, self.id);
}
Expand Down
37 changes: 19 additions & 18 deletions yazi-core/src/tab/folder.rs
Original file line number Diff line number Diff line change
Expand Up @@ -4,7 +4,7 @@ use yazi_config::{LAYOUT, MANAGER};
use yazi_dds::Pubsub;
use yazi_fs::{Cha, File, Files, FilesOp, FolderStage, Step};
use yazi_proxy::ManagerProxy;
use yazi_shared::{Id, url::{Url, Urn}};
use yazi_shared::{Id, url::{Url, Urn, UrnBuf}};

pub struct Folder {
pub url: Url,
Expand All @@ -15,21 +15,21 @@ pub struct Folder {
pub offset: usize,
pub cursor: usize,

pub page: usize,
pub tracing: bool,
pub page: usize,
pub trace: Option<UrnBuf>,
}

impl Default for Folder {
fn default() -> Self {
Self {
url: Default::default(),
cha: Default::default(),
files: Files::new(MANAGER.show_hidden),
stage: Default::default(),
offset: Default::default(),
cursor: Default::default(),
page: Default::default(),
tracing: Default::default(),
url: Default::default(),
cha: Default::default(),
files: Files::new(MANAGER.show_hidden),
stage: Default::default(),
offset: Default::default(),
cursor: Default::default(),
page: Default::default(),
trace: Default::default(),
}
}
}
Expand Down Expand Up @@ -73,7 +73,9 @@ impl Folder {
FilesOp::Upserting(_, files) => self.files.update_upserting(files),
}

self.arrow(0);
self.trace = self.trace.take_if(|_| !self.files.is_empty() || self.stage.is_loading());
self.repos(self.trace.clone());

(stage, revision) != (self.stage, self.files.revision)
}

Expand All @@ -90,15 +92,14 @@ impl Folder {
pub fn arrow(&mut self, step: impl Into<Step>) -> bool {
let step = step.into() as Step;
let mut b = if self.files.is_empty() {
(self.cursor, self.offset, self.tracing) = (0, 0, false);
false
(mem::take(&mut self.cursor), mem::take(&mut self.offset)) != (0, 0)
} else if step.is_positive() {
self.next(step)
} else {
self.prev(step)
};

self.tracing |= b;
self.trace = self.hovered().filter(|_| b).map(|h| h.urn_owned()).or(self.trace.take());
b |= self.squeeze_offset();

self.sync_page(false);
Expand All @@ -107,16 +108,16 @@ impl Folder {

pub fn hover(&mut self, urn: &Urn) -> bool {
if self.hovered().map(|h| h.urn()) == Some(urn) {
return false;
return self.arrow(0);
}

let new = self.files.position(urn).unwrap_or(self.cursor) as isize;
self.arrow(new - self.cursor as isize)
}

#[inline]
pub fn repos(&mut self, url: Option<&Urn>) -> bool {
if let Some(u) = url { self.hover(u) } else { self.arrow(0) }
pub fn repos(&mut self, urn: Option<impl AsRef<Urn>>) -> bool {
if let Some(u) = urn { self.hover(u.as_ref()) } else { self.arrow(0) }
}

pub fn sync_page(&mut self, force: bool) {
Expand Down
8 changes: 2 additions & 6 deletions yazi-core/src/tab/tab.rs
Original file line number Diff line number Diff line change
Expand Up @@ -123,22 +123,18 @@ impl Tab {
return render!();
}

let hovered = f.hovered().filter(|_| f.tracing).map(|h| h.urn_owned());
f.files.set_show_hidden(self.pref.show_hidden);
f.files.set_sorter(<_>::from(&self.pref));

render!(f.files.catchup_revision());
render!(f.repos(hovered.as_ref().map(|u| u.as_urn())));
render!(f.repos(f.trace.clone()));
};

apply(&mut self.current);

if let Some(parent) = &mut self.parent {
apply(parent);

// The parent should always track the CWD
parent.hover(self.current.url.urn());
parent.tracing = parent.hovered().map(|h| &h.url) == Some(&self.current.url);
parent.hover(self.current.url.urn()); // The parent should always track the CWD
}

self
Expand Down
5 changes: 5 additions & 0 deletions yazi-fs/src/stage.rs
Original file line number Diff line number Diff line change
Expand Up @@ -8,6 +8,11 @@ pub enum FolderStage {
Failed(std::io::ErrorKind),
}

impl FolderStage {
#[inline]
pub fn is_loading(self) -> bool { self == Self::Loading }
}

impl Serialize for FolderStage {
fn serialize<S: serde::Serializer>(&self, serializer: S) -> Result<S::Ok, S::Error> {
let mut map = serializer.serialize_map(Some(2))?;
Expand Down
4 changes: 4 additions & 0 deletions yazi-shared/src/url/urn.rs
Original file line number Diff line number Diff line change
Expand Up @@ -55,6 +55,10 @@ impl Borrow<Urn> for UrnBuf {
fn borrow(&self) -> &Urn { Urn::new(&self.0) }
}

impl AsRef<Urn> for UrnBuf {
fn as_ref(&self) -> &Urn { self.borrow() }
}

impl AsRef<Path> for UrnBuf {
fn as_ref(&self) -> &Path { &self.0 }
}
Expand Down

0 comments on commit ef48b7f

Please sign in to comment.