Add option -0 to use null separator for file paths in picker mode

This commit is contained in:
Arun Prakash Jana 2024-08-25 09:30:59 +05:30
parent a2deaf57f3
commit 7def65fc02
No known key found for this signature in database
GPG key ID: 4A865183AF6C5631
5 changed files with 25 additions and 10 deletions

View file

@ -44,6 +44,7 @@ _nnn ()
-U -U
-V -V
-x -x
-0
-h -h
) )
if [[ $prev == -b ]]; then if [[ $prev == -b ]]; then

View file

@ -43,4 +43,5 @@ complete -c nnn -s u -d 'use selection (no prompt)'
complete -c nnn -s U -d 'show user and group' complete -c nnn -s U -d 'show user and group'
complete -c nnn -s V -d 'show program version and exit' complete -c nnn -s V -d 'show program version and exit'
complete -c nnn -s x -d 'notis, sel to system clipboard, xterm title' complete -c nnn -s x -d 'notis, sel to system clipboard, xterm title'
complete -c nnn -s 0 -d 'use null separator in picker mode'
complete -c nnn -s h -d 'show program help' complete -c nnn -s h -d 'show program help'

View file

@ -42,6 +42,7 @@ args=(
'(-U)-U[show user and group]' '(-U)-U[show user and group]'
'(-V)-V[show program version and exit]' '(-V)-V[show program version and exit]'
'(-x)-x[notis, sel to system clipboard, xterm title]' '(-x)-x[notis, sel to system clipboard, xterm title]'
'(-0)-0[use null separator in picker mode]'
'(-h)-h[show program help]' '(-h)-h[show program help]'
'*:filename:_files' '*:filename:_files'
) )

3
nnn.1
View file

@ -165,6 +165,9 @@ supports the following options:
copy path to system clipboard on selection (requires \fI.cbcp\fR plugin) copy path to system clipboard on selection (requires \fI.cbcp\fR plugin)
show xterm title (if non-picker mode) show xterm title (if non-picker mode)
.Pp .Pp
.Fl 0
use null separator (instead of newline) to separate file paths in picker mode outout
.Pp
.Fl h .Fl h
show program help and exit show program help and exit
.Sh CONFIGURATION .Sh CONFIGURATION

View file

@ -192,6 +192,10 @@
#define MSGWAIT '$' #define MSGWAIT '$'
#define SELECT ' ' #define SELECT ' '
#define PROMPT ">>> " #define PROMPT ">>> "
#undef NEWLINE
#define NEWLINE "\n"
#undef NUL
#define NUL "\0"
#define REGEX_MAX 48 #define REGEX_MAX 48
#define ENTRY_INCR 64 /* Number of dir 'entry' structures to allocate per shot */ #define ENTRY_INCR 64 /* Number of dir 'entry' structures to allocate per shot */
#define NAMEBUF_INCR 0x800 /* 64 dir entries at once, avg. 32 chars per file name = 64*32B = 2KB */ #define NAMEBUF_INCR 0x800 /* 64 dir entries at once, avg. 32 chars per file name = 64*32B = 2KB */
@ -1611,7 +1615,7 @@ static void selbufrealloc(const size_t alloclen)
} }
/* Write selected file paths to fd, linefeed separated */ /* Write selected file paths to fd, linefeed separated */
static size_t seltofile(int fd, uint_t *pcount) static size_t seltofile(int fd, uint_t *pcount, const char *separator)
{ {
uint_t lastpos, count = 0; uint_t lastpos, count = 0;
char *pbuf = pselbuf; char *pbuf = pselbuf;
@ -1647,7 +1651,7 @@ static size_t seltofile(int fd, uint_t *pcount)
pos += len; pos += len;
if (pos <= lastpos) { if (pos <= lastpos) {
if (write(fd, "\n", 1) != 1) if (write(fd, separator, 1) != 1)
return pos; return pos;
pbuf += len + 1; pbuf += len + 1;
} }
@ -1972,7 +1976,7 @@ static void endselection(bool endselmode)
return; return;
} }
seltofile(fd, NULL); seltofile(fd, NULL, NEWLINE);
if (close(fd)) { if (close(fd)) {
DPRINTF_S(strerror(errno)); DPRINTF_S(strerror(errno));
printwarn(NULL); printwarn(NULL);
@ -2036,7 +2040,7 @@ static int editselection(void)
return -1; return -1;
} }
seltofile(fd, NULL); seltofile(fd, NULL, NEWLINE);
if (close(fd)) { if (close(fd)) {
DPRINTF_S(strerror(errno)); DPRINTF_S(strerror(errno));
return -1; return -1;
@ -2644,7 +2648,7 @@ static bool cpmv_rename(int choice, const char *path)
if (!count) if (!count)
goto finish; goto finish;
} else } else
seltofile(fd, &count); seltofile(fd, &count, NEWLINE);
close(fd); close(fd);
@ -2764,8 +2768,8 @@ static bool batch_rename(void)
for (i = 0; i < ndents; ++i) for (i = 0; i < ndents; ++i)
appendfpath(pdents[i].name, NAME_MAX); appendfpath(pdents[i].name, NAME_MAX);
seltofile(fd1, &count); seltofile(fd1, &count, NEWLINE);
seltofile(fd2, NULL); seltofile(fd2, NULL, NEWLINE);
close(fd2); close(fd2);
if (dir) /* Don't retain dir entries in selection */ if (dir) /* Don't retain dir entries in selection */
@ -6074,7 +6078,7 @@ static void send_to_explorer(int *presel)
{ {
if (nselected) { if (nselected) {
int fd = open(fifopath, O_WRONLY|O_NONBLOCK|O_CLOEXEC, 0600); int fd = open(fifopath, O_WRONLY|O_NONBLOCK|O_CLOEXEC, 0600);
if ((fd == -1) || (seltofile(fd, NULL) != (size_t)(selbufpos))) if ((fd == -1) || (seltofile(fd, NULL, NEWLINE) != (size_t)(selbufpos)))
printwarn(presel); printwarn(presel);
else { else {
resetselind(); resetselind();
@ -8426,6 +8430,7 @@ static void usage(void)
#ifndef NOX11 #ifndef NOX11
" -x notis, selection sync, xterm title\n" " -x notis, selection sync, xterm title\n"
#endif #endif
" -0 null separator in picker mode\n"
" -h show help\n\n" " -h show help\n\n"
"v%s\n%s\n", __func__, VERSION, GENERAL_INFO); "v%s\n%s\n", __func__, VERSION, GENERAL_INFO);
} }
@ -8566,6 +8571,7 @@ int main(int argc, char *argv[])
char *arg = NULL; char *arg = NULL;
char *session = NULL; char *session = NULL;
int fd, opt, sort = 0, pkey = '\0'; /* Plugin key */ int fd, opt, sort = 0, pkey = '\0'; /* Plugin key */
bool sepnul = FALSE;
#ifndef NOMOUSE #ifndef NOMOUSE
mmask_t mask; mmask_t mask;
char *middle_click_env = xgetenv(env_cfg[NNN_MCLICK], "\0"); char *middle_click_env = xgetenv(env_cfg[NNN_MCLICK], "\0");
@ -8583,7 +8589,7 @@ int main(int argc, char *argv[])
while ((opt = (env_opts_id > 0 while ((opt = (env_opts_id > 0
? env_opts[--env_opts_id] ? env_opts[--env_opts_id]
: getopt(argc, argv, "aAb:BcCdDeEfF:gHiJKl:nNop:P:QrRs:St:T:uUVxh"))) != -1) { : getopt(argc, argv, "aAb:BcCdDeEfF:gHiJKl:nNop:P:QrRs:St:T:uUVx0h"))) != -1) {
switch (opt) { switch (opt) {
#ifndef NOFIFO #ifndef NOFIFO
case 'a': case 'a':
@ -8728,6 +8734,9 @@ int main(int argc, char *argv[])
case 'x': case 'x':
cfg.x11 = 1; cfg.x11 = 1;
break; break;
case '0':
sepnul = TRUE;
break;
case 'h': case 'h':
usage(); usage();
return EXIT_SUCCESS; return EXIT_SUCCESS;
@ -9045,7 +9054,7 @@ int main(int argc, char *argv[])
if (g_state.picker) { if (g_state.picker) {
if (selbufpos) { if (selbufpos) {
fd = selpath ? open(selpath, O_WRONLY | O_CREAT | O_TRUNC, 0600) : STDOUT_FILENO; fd = selpath ? open(selpath, O_WRONLY | O_CREAT | O_TRUNC, 0600) : STDOUT_FILENO;
if ((fd == -1) || (seltofile(fd, NULL) != (size_t)(selbufpos))) if ((fd == -1) || (seltofile(fd, NULL, sepnul ? NUL : NEWLINE) != (size_t)(selbufpos)))
xerror(); xerror();
if (fd > 1) if (fd > 1)