feat: bulk create (#3793)
Some checks failed
Cachix / Publish Flake (push) Has been cancelled
Cachix / Publish Flake-1 (push) Has been cancelled
Check / clippy (push) Has been cancelled
Check / rustfmt (push) Has been cancelled
Check / stylua (push) Has been cancelled
Draft / build-unix (gcc-aarch64-linux-gnu, ubuntu-latest, aarch64-unknown-linux-gnu) (push) Has been cancelled
Draft / build-unix (gcc-i686-linux-gnu, ubuntu-latest, i686-unknown-linux-gnu) (push) Has been cancelled
Draft / build-unix (gcc-riscv64-linux-gnu, ubuntu-latest, riscv64gc-unknown-linux-gnu) (push) Has been cancelled
Draft / build-unix (gcc-sparc64-linux-gnu, ubuntu-latest, sparc64-unknown-linux-gnu) (push) Has been cancelled
Draft / build-unix (macos-latest, aarch64-apple-darwin) (push) Has been cancelled
Draft / build-unix (macos-latest, x86_64-apple-darwin) (push) Has been cancelled
Draft / build-unix (ubuntu-latest, x86_64-unknown-linux-gnu) (push) Has been cancelled
Draft / build-windows (windows-latest, aarch64-pc-windows-msvc) (push) Has been cancelled
Draft / build-windows (windows-latest, x86_64-pc-windows-msvc) (push) Has been cancelled
Draft / build-musl (aarch64-unknown-linux-musl) (push) Has been cancelled
Draft / build-musl (x86_64-unknown-linux-musl) (push) Has been cancelled
Draft / build-snap (amd64, ubuntu-latest) (push) Has been cancelled
Draft / build-snap (arm64, ubuntu-24.04-arm) (push) Has been cancelled
Test / test (macos-latest) (push) Has been cancelled
Test / test (ubuntu-latest) (push) Has been cancelled
Test / test (windows-latest) (push) Has been cancelled
Draft / snap (push) Has been cancelled
Draft / draft (push) Has been cancelled
Draft / nightly (push) Has been cancelled

Co-authored-by: sxyazi <sxyazi@gmail.com>
This commit is contained in:
Daniel Vincent 2026-05-11 23:42:08 +05:30 committed by GitHub
parent 247f925e53
commit fde563380b
No known key found for this signature in database
GPG key ID: B5690EEEBB952194
22 changed files with 243 additions and 42 deletions

View file

@ -12,6 +12,10 @@ The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.1.0/):
## [Unreleased]
### Added
- Bulk create ([#3793])
## [v26.5.6]
### Added
@ -1704,6 +1708,7 @@ The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.1.0/):
[#3780]: https://github.com/sxyazi/yazi/pull/3780
[#3781]: https://github.com/sxyazi/yazi/pull/3781
[#3792]: https://github.com/sxyazi/yazi/pull/3792
[#3793]: https://github.com/sxyazi/yazi/pull/3793
[#3804]: https://github.com/sxyazi/yazi/pull/3804
[#3813]: https://github.com/sxyazi/yazi/pull/3813
[#3846]: https://github.com/sxyazi/yazi/pull/3846

View file

@ -23,7 +23,7 @@ Yazi (means "duck") is a terminal file manager written in Rust, based on non-blo
- 🧰 Integration with ripgrep, fd, fzf, zoxide
- 💫 Vim-like input/pick/confirm/which/notify component, auto-completion for cd paths
- 🏷️ Multi-Tab Support, Cross-directory selection, Scrollable Preview (for videos, PDFs, archives, code, directories, etc.)
- 🔄 Bulk Renaming, Archive Extraction, Visual Mode, File Chooser, [Git Integration](https://github.com/yazi-rs/plugins/tree/main/git.yazi), [Mount Manager](https://github.com/yazi-rs/plugins/tree/main/mount.yazi)
- 🔄 Bulk Rename/Create, Archive Extraction, Visual Mode, File Chooser, [Git Integration](https://github.com/yazi-rs/plugins/tree/main/git.yazi), [Mount Manager](https://github.com/yazi-rs/plugins/tree/main/mount.yazi)
- 🎨 Theme System, Mouse Support, Trash Bin, Custom Layouts, CSI u, OSC 52
- ... and more!

View file

@ -0,0 +1,176 @@
use std::{fmt::{self, Display}, io::{self, Read, Write}, path::{MAIN_SEPARATOR, Path}, sync::Arc};
use anyhow::{Result, anyhow};
use scopeguard::defer;
use yazi_binding::Permit;
use yazi_config::{YAZI, opener::OpenerRule};
use yazi_fs::{File, FilesOp, Splatter, provider::{Provider, local::Local}};
use yazi_macro::succ;
use yazi_parser::VoidForm;
use yazi_proxy::TasksProxy;
use yazi_scheduler::{AppProxy, NotifyProxy};
use yazi_shared::{data::Data, strand::Strand, terminal_clear, url::{AsUrl, UrlBuf, UrlCow, UrlLike}};
use yazi_shim::path::CROSS_SEPARATOR;
use yazi_term::YIELD_TO_SUBPROCESS;
use yazi_tty::TTY;
use yazi_vfs::{VfsFile, provider};
use yazi_watcher::WATCHER;
use crate::{Actor, Ctx};
pub struct BulkCreate;
impl Actor for BulkCreate {
type Form = VoidForm;
const NAME: &str = "bulk_create";
fn act(cx: &mut Ctx, _: Self::Form) -> Result<Data> {
let Some(opener) = Self::opener() else {
succ!(NotifyProxy::push_warn("Bulk create", "No text opener found"));
};
let cwd = cx.cwd().clone();
tokio::spawn(async move {
let tmp = YAZI.preview.tmpfile("bulk-create");
provider::create_new(&tmp).await?;
defer! {
let tmp = tmp.clone();
tokio::spawn(async move {
Local::regular(&tmp).remove_file().await
});
}
TasksProxy::process_exec(
cwd.clone(),
Splatter::new(&[UrlCow::default(), tmp.as_url().into()]).splat(&opener.run),
vec![UrlCow::default(), UrlBuf::from(&tmp).into()],
opener.block,
opener.orphan,
)
.await;
let _permit = Permit::new(YIELD_TO_SUBPROCESS.acquire().await.unwrap(), AppProxy::resume());
AppProxy::stop().await;
let content = Local::regular(&tmp).read_to_string().await?;
Self::r#do(cwd, content.lines().filter_map(Entry::parse).collect()).await
});
succ!()
}
}
impl BulkCreate {
async fn r#do(cwd: UrlBuf, todo: Vec<Entry<'_>>) -> Result<()> {
terminal_clear(TTY.writer())?;
if todo.is_empty() {
return Ok(());
} else if !Self::ask_continue(&todo, None)? {
return Ok(()); // TODO: support `bulk_exit`?
}
let _permit = WATCHER.acquire().await.unwrap();
let (mut failed, mut succeeded) = (vec![], Vec::with_capacity(todo.len()));
for entry in todo {
let Ok(dist) = cwd.try_join(entry.path) else {
failed.push((entry, anyhow!("Invalid path")));
continue;
};
let result: io::Result<()> = if entry.is_dir {
provider::create_dir_all(&dist).await
} else if let Some(parent) = dist.parent() {
provider::create_dir_all(parent).await.ok();
provider::create_new(&dist).await.map(|_| ())
} else {
Err(io::Error::other("No parent directory"))
};
if let Err(e) = result {
failed.push((entry, e.into()));
} else if let Ok(f) = File::new(dist).await {
succeeded.push(f);
} else {
failed.push((entry, anyhow!("Failed to retrieve file info")));
}
}
if !succeeded.is_empty() {
// err!(Pubsub::pub_after_bulk_create(it)); // FIXME
FilesOp::create(succeeded);
}
drop(_permit);
if !failed.is_empty() {
Self::output_failed(failed).await?;
}
Ok(())
}
fn opener() -> Option<Arc<OpenerRule>> {
YAZI
.open
.match_dummy(Path::new("bulk-create.txt"), "text/plain")
.and_then(|r| YAZI.opener.block(&r))
}
fn ask_continue(todo: &[Entry], decision: Option<bool>) -> Result<bool> {
if let Some(decision) = decision {
return Ok(decision);
}
{
let mut w = TTY.lockout();
for entry in todo {
writeln!(w, "{entry}")?;
}
write!(w, "Continue to create? (y/N): ")?;
w.flush()?;
}
let mut buf = [0; 10];
_ = TTY.reader().read(&mut buf)?;
Ok(buf[0] == b'y' || buf[0] == b'Y')
}
async fn output_failed(failed: Vec<(Entry<'_>, anyhow::Error)>) -> Result<()> {
let mut stdout = TTY.lockout();
terminal_clear(&mut *stdout)?;
writeln!(stdout, "Failed to create:")?;
for (entry, err) in failed {
writeln!(stdout, "{entry}: {err}")?;
}
writeln!(stdout, "\nPress ENTER to exit")?;
stdout.flush()?;
TTY.reader().read_exact(&mut [0])?;
Ok(())
}
}
// --- Entry
struct Entry<'a> {
path: Strand<'a>,
is_dir: bool,
}
impl<'a> Entry<'a> {
fn parse(s: &'a str) -> Option<Self> {
let (path, is_dir) = match s.strip_suffix(CROSS_SEPARATOR) {
Some(p) => (p, true),
None => (s, false),
};
Some(Self { path: path.into(), is_dir }).filter(|_| !path.is_empty())
}
}
impl Display for Entry<'_> {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
if self.is_dir {
write!(f, "{}{MAIN_SEPARATOR}", self.path.display())
} else {
self.path.display().fmt(f)
}
}
}

View file

@ -45,7 +45,7 @@ impl Actor for BulkRename {
let cwd = cx.cwd().clone();
let batcher = cx.core.mgr.batcher.clone();
tokio::spawn(async move {
let tmp = YAZI.preview.tmpfile("bulk");
let tmp = YAZI.preview.tmpfile("bulk-rename");
Gate::default()
.write(true)
@ -143,7 +143,7 @@ impl BulkRename {
if !succeeded.is_empty() {
let it = succeeded.iter().map(|(o, n)| (o.as_url(), n.url.as_url()));
err!(Pubsub::pub_after_bulk(it));
err!(Pubsub::pub_after_bulk_rename(it));
FilesOp::rename(succeeded);
}
drop(permit);

View file

@ -1,6 +1,7 @@
yazi_macro::mod_flat!(
arrow
back
bulk_create
bulk_exit
bulk_rename
cd

View file

@ -1,5 +1,5 @@
# A TOML linter such as https://github.com/tombi-toml/tombi can use this schema to validate your config.
# If you encounter any problems, please make an issue at https://github.com/yazi-rs/schemas.
# A TOML linter such as Tombi can use this schema to validate your config.
# If you encounter any problems, please file an issue at https://github.com/yazi-rs/schemas.
#:schema https://yazi-rs.github.io/schemas/keymap.json
@ -76,6 +76,7 @@ keymap = [
{ on = "d", run = "remove", desc = "Trash selected files" },
{ on = "D", run = "remove --permanently", desc = "Permanently delete selected files" },
{ on = "a", run = "create", desc = "Create a file (ends with / for directories)" },
{ on = "A", run = "bulk_create", desc = "Bulk create files" },
{ on = "r", run = "rename --cursor=before_ext", desc = "Rename selected file(s)" },
{ on = ";", run = "shell --interactive", desc = "Run a shell command" },
{ on = ":", run = "shell --block --interactive", desc = "Run a shell command (block until finishes)" },

View file

@ -1,5 +1,5 @@
# A TOML linter such as https://github.com/tombi-toml/tombi can use this schema to validate your config.
# If you encounter any issues, please make an issue at https://github.com/yazi-rs/schemas.
# A TOML linter such as Tombi can use this schema to validate your config.
# If you encounter any problems, please file an issue at https://github.com/yazi-rs/schemas.
#:schema https://yazi-rs.github.io/schemas/yazi.json

View file

@ -6,11 +6,11 @@ use yazi_shared::url::{Url, UrlCow};
use super::Ember;
#[derive(Clone, Debug, Deserialize, Serialize)]
pub struct EmberBulk<'a> {
pub struct EmberBulkRename<'a> {
pub changes: HashMap<UrlCow<'a>, UrlCow<'a>>,
}
impl<'a> EmberBulk<'a> {
impl<'a> EmberBulkRename<'a> {
pub fn borrowed<I>(changes: I) -> Ember<'a>
where
I: Iterator<Item = (Url<'a>, Url<'a>)>,
@ -19,7 +19,7 @@ impl<'a> EmberBulk<'a> {
}
}
impl EmberBulk<'static> {
impl EmberBulkRename<'static> {
pub fn owned<'a, I>(changes: I) -> Ember<'static>
where
I: Iterator<Item = (Url<'a>, Url<'a>)>,
@ -31,11 +31,11 @@ impl EmberBulk<'static> {
}
}
impl<'a> From<EmberBulk<'a>> for Ember<'a> {
fn from(value: EmberBulk<'a>) -> Self { Self::Bulk(value) }
impl<'a> From<EmberBulkRename<'a>> for Ember<'a> {
fn from(value: EmberBulkRename<'a>) -> Self { Self::BulkRename(value) }
}
impl IntoLua for EmberBulk<'_> {
impl IntoLua for EmberBulkRename<'_> {
fn into_lua(self, lua: &Lua) -> mlua::Result<Value> {
lua
.create_table_from(

View file

@ -2,7 +2,7 @@ use anyhow::{Result, bail};
use mlua::{ExternalResult, IntoLua, Lua, Value};
use yazi_shared::Id;
use super::{EmberBulk, EmberBye, EmberCd, EmberCustom, EmberDelete, EmberDownload, EmberDuplicate, EmberHey, EmberHi, EmberHover, EmberLoad, EmberMount, EmberMove, EmberRename, EmberTab, EmberTrash, EmberYank};
use super::{EmberBulkRename, EmberBye, EmberCd, EmberCustom, EmberDelete, EmberDownload, EmberDuplicate, EmberHey, EmberHi, EmberHover, EmberLoad, EmberMount, EmberMove, EmberRename, EmberTab, EmberTrash, EmberYank};
use crate::Payload;
#[derive(Clone, Debug)]
@ -15,7 +15,7 @@ pub enum Ember<'a> {
Load(EmberLoad<'a>),
Hover(EmberHover<'a>),
Rename(EmberRename<'a>),
Bulk(EmberBulk<'a>),
BulkRename(EmberBulkRename<'a>),
Yank(EmberYank<'a>),
Duplicate(EmberDuplicate<'a>),
Move(EmberMove<'a>),
@ -37,7 +37,7 @@ impl Ember<'static> {
"load" => Self::Load(serde_json::from_str(body)?),
"hover" => Self::Hover(serde_json::from_str(body)?),
"rename" => Self::Rename(serde_json::from_str(body)?),
"bulk" => Self::Bulk(serde_json::from_str(body)?),
"bulk-rename" => Self::BulkRename(serde_json::from_str(body)?),
"@yank" => Self::Yank(serde_json::from_str(body)?),
"duplicate" => Self::Duplicate(serde_json::from_str(body)?),
"move" => Self::Move(serde_json::from_str(body)?),
@ -65,7 +65,7 @@ impl Ember<'static> {
| "load"
| "hover"
| "rename"
| "bulk"
| "bulk-rename"
| "@yank"
| "duplicate"
| "move"
@ -104,7 +104,7 @@ impl<'a> Ember<'a> {
Self::Load(_) => "load",
Self::Hover(_) => "hover",
Self::Rename(_) => "rename",
Self::Bulk(_) => "bulk",
Self::BulkRename(_) => "bulk-rename",
Self::Yank(_) => "@yank",
Self::Duplicate(_) => "duplicate",
Self::Move(_) => "move",
@ -132,7 +132,7 @@ impl<'a> IntoLua for Ember<'a> {
Self::Hover(b) => b.into_lua(lua),
Self::Tab(b) => b.into_lua(lua),
Self::Rename(b) => b.into_lua(lua),
Self::Bulk(b) => b.into_lua(lua),
Self::BulkRename(b) => b.into_lua(lua),
Self::Yank(b) => b.into_lua(lua),
Self::Duplicate(b) => b.into_lua(lua),
Self::Move(b) => b.into_lua(lua),

View file

@ -1,3 +1,3 @@
yazi_macro::mod_flat!(
bulk bye cd custom delete download duplicate ember hey hi hover load mount r#move rename tab trash yank
bulk_rename bye cd custom delete download duplicate ember hey hi hover load mount r#move rename tab trash yank
);

View file

@ -94,7 +94,7 @@ impl Display for Payload<'_> {
Ember::Hover(b) => serde_json::to_string(b),
Ember::Tab(b) => serde_json::to_string(b),
Ember::Rename(b) => serde_json::to_string(b),
Ember::Bulk(b) => serde_json::to_string(b),
Ember::BulkRename(b) => serde_json::to_string(b),
Ember::Yank(b) => serde_json::to_string(b),
Ember::Duplicate(b) => serde_json::to_string(b),
Ember::Move(b) => serde_json::to_string(b),

View file

@ -8,7 +8,7 @@ use yazi_fs::FolderStage;
use yazi_shared::{Id, url::{Url, UrlBuf, UrlBufCov}};
use yazi_shim::cell::RoCell;
use crate::{Client, ID, PEERS, ember::{Ember, EmberBulk, EmberDuplicateItem, EmberHi, EmberMoveItem}};
use crate::{Client, ID, PEERS, ember::{Ember, EmberBulkRename, EmberDuplicateItem, EmberHi, EmberMoveItem}};
pub static LOCAL: RoCell<RwLock<HashMap<String, HashMap<String, Function>>>> = RoCell::new();
@ -128,18 +128,18 @@ impl Pubsub {
true
}
pub fn pub_after_bulk<'a, I>(changes: I) -> Result<()>
pub fn pub_after_bulk_rename<'a, I>(changes: I) -> Result<()>
where
I: Iterator<Item = (Url<'a>, Url<'a>)> + Clone,
{
if BOOT.local_events.contains("bulk") {
EmberBulk::borrowed(changes.clone()).with_receiver(*ID).flush()?;
if BOOT.local_events.contains("bulk-rename") {
EmberBulkRename::borrowed(changes.clone()).with_receiver(*ID).flush()?;
}
if PEERS.read().values().any(|p| p.able("bulk")) {
Client::push(EmberBulk::borrowed(changes.clone()))?;
if PEERS.read().values().any(|p| p.able("bulk-rename")) {
Client::push(EmberBulkRename::borrowed(changes.clone()))?;
}
if LOCAL.read().contains_key("bulk") {
Self::r#pub(EmberBulk::owned(changes))?;
if LOCAL.read().contains_key("bulk-rename") {
Self::r#pub(EmberBulkRename::owned(changes))?;
}
Ok(())
}

View file

@ -132,6 +132,7 @@ impl<'a> Executor<'a> {
on!(search_do);
on!(bulk_exit);
on!(bulk_rename);
on!(bulk_create);
// Filter
on!(filter);

View file

@ -51,6 +51,18 @@ impl FilesOp {
ticket
}
pub fn create(files: Vec<File>) {
let mut parents: HashMap<UrlBuf, Vec<_>> = Default::default();
for file in files {
if let Some(p) = file.url.parent() {
parents.get_or_insert_default(p).push(file);
}
}
for (p, files) in parents {
Self::Creating(p, files).emit();
}
}
pub fn rename(map: HashMap<UrlBuf, File>) {
let mut parents: HashMap<UrlBuf, (HashSet<_>, HashMap<_, _>)> = Default::default();
for (o, n) in map {

View file

@ -28,6 +28,7 @@ pub enum Spark<'a> {
// Mgr
Arrow(crate::ArrowForm),
Back(crate::VoidForm),
BulkCreate(crate::VoidForm),
BulkExit(crate::mgr::BulkExitForm),
BulkRename(crate::VoidForm),
Cd(crate::mgr::CdForm),
@ -215,6 +216,7 @@ impl<'a> IntoLua for Spark<'a> {
// Mgr
Self::Arrow(b) => b.into_lua(lua),
Self::Back(b) => b.into_lua(lua),
Self::BulkCreate(b) => b.into_lua(lua),
Self::BulkExit(b) => b.into_lua(lua),
Self::BulkRename(b) => b.into_lua(lua),
Self::Cd(b) => b.into_lua(lua),

View file

@ -1,3 +1,3 @@
yazi_macro::mod_pub!(arc_swap cell crossterm mlua ratatui serde strum toml vec);
yazi_macro::mod_pub!(arc_swap cell crossterm mlua path ratatui serde strum toml vec);
yazi_macro::mod_flat!(twox);

View file

@ -0,0 +1 @@
yazi_macro::mod_flat!(separator);

View file

@ -0,0 +1,5 @@
#[cfg(windows)]
pub const CROSS_SEPARATOR: [char; 2] = ['/', '\\'];
#[cfg(not(windows))]
pub const CROSS_SEPARATOR: char = std::path::MAIN_SEPARATOR;

View file

@ -3,16 +3,17 @@ use std::path::MAIN_SEPARATOR_STR;
use anyhow::Result;
use yazi_macro::{act, render, succ};
use yazi_shared::data::Data;
use yazi_shim::path::CROSS_SEPARATOR;
use crate::input::{Input, SEPARATOR, parser::CompleteOpt};
use crate::input::{Input, parser::CompleteOpt};
impl Input {
pub fn complete(&mut self, opt: CompleteOpt) -> Result<Data> {
let (before, after) = self.partition();
let new = if let Some((prefix, _)) = before.rsplit_once(SEPARATOR) {
format!("{prefix}/{}{after}", opt.completable()).replace(SEPARATOR, MAIN_SEPARATOR_STR)
let new = if let Some((prefix, _)) = before.rsplit_once(CROSS_SEPARATOR) {
format!("{prefix}/{}{after}", opt.completable()).replace(CROSS_SEPARATOR, MAIN_SEPARATOR_STR)
} else {
format!("{}{after}", opt.completable()).replace(SEPARATOR, MAIN_SEPARATOR_STR)
format!("{}{after}", opt.completable()).replace(CROSS_SEPARATOR, MAIN_SEPARATOR_STR)
};
let snap = self.snap_mut();

View file

@ -6,9 +6,10 @@ use tokio::sync::mpsc;
use yazi_config::YAZI;
use yazi_macro::act;
use yazi_shared::Ids;
use yazi_shim::path::CROSS_SEPARATOR;
use super::{InputSnap, InputSnaps, mode::InputMode, op::InputOp};
use crate::{CLIPBOARD, input::{InputEvent, InputOpt, SEPARATOR}};
use crate::{CLIPBOARD, input::{InputEvent, InputOpt}};
#[derive(Default)]
pub struct Input {
@ -152,7 +153,7 @@ impl Input {
let snap = self.snap();
let idx = snap.idx(snap.cursor).unwrap();
if let Some(sep) = snap.value[idx..].find(SEPARATOR).map(|i| idx + i) {
if let Some(sep) = snap.value[idx..].find(CROSS_SEPARATOR).map(|i| idx + i) {
(&snap.value[..sep], &snap.value[sep + 1..])
} else {
(&snap.value, "")

View file

@ -1,3 +1,3 @@
yazi_macro::mod_pub!(actor parser);
yazi_macro::mod_flat!(event input mode op option separator snap snaps widget);
yazi_macro::mod_flat!(event input mode op option snap snaps widget);

View file

@ -1,5 +0,0 @@
#[cfg(windows)]
pub(super) const SEPARATOR: [char; 2] = ['/', '\\'];
#[cfg(not(windows))]
pub(super) const SEPARATOR: char = std::path::MAIN_SEPARATOR;