aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorGravatar Arun Prakash Jana <engineerarun@gmail.com>2017-08-23 00:46:58 +0530
committerGravatar Arun Prakash Jana <engineerarun@gmail.com>2017-08-23 00:51:59 +0530
commit0b3bec329feeab2cec2738228a397de6ca46344b (patch)
treeccee0a4719aec0600b3436e48dbd3d498261d2a1
parent5404cee9ee974020ca3203b874c91c286658f3c8 (diff)
downloadnnn-0b3bec329feeab2cec2738228a397de6ca46344b.tar.gz
Support in-place file rename
-rw-r--r--README.md1
-rw-r--r--config.h3
-rw-r--r--nnn.12
-rw-r--r--nnn.c63
4 files changed, 62 insertions, 7 deletions
diff --git a/README.md b/README.md
index 53605c9..325dd98 100644
--- a/README.md
+++ b/README.md
@@ -210,6 +210,7 @@ To cook yourself, download the [latest stable release](https://github.com/jarun/
D | Show current file details
m | Show concise media info
M | Show full media info
+ ^R | Rename selected entry
s | Toggle sort by file size
S | Toggle disk usage mode
t | Toggle sort by mtime
diff --git a/config.h b/config.h
index 275de03..31ea672 100644
--- a/config.h
+++ b/config.h
@@ -34,6 +34,7 @@ enum action {
SEL_MTIME,
SEL_REDRAW,
SEL_COPY,
+ SEL_RENAME,
SEL_HELP,
SEL_RUN,
SEL_RUNARG,
@@ -140,6 +141,8 @@ static struct key bindings[] = {
{ KEY_F(2), SEL_REDRAW, "", "" },
/* Copy currently selected file path */
{ CONTROL('K'), SEL_COPY, "", "" },
+ /* Show rename prompt */
+ { CONTROL('R'), SEL_RENAME, "", "" },
/* Show help */
{ '?', SEL_HELP, "", "" },
/* Run command */
diff --git a/nnn.1 b/nnn.1
index 7cece0d..7e91970 100644
--- a/nnn.1
+++ b/nnn.1
@@ -75,6 +75,8 @@ Show current file details screen
Show concise media info
.It Ic M
Show full media info
+.It Ic ^R
+Rename selected entry
.It Ic s
Toggle sort by file size
.It Ic S
diff --git a/nnn.c b/nnn.c
index 816bbfb..c8de442 100644
--- a/nnn.c
+++ b/nnn.c
@@ -1541,6 +1541,7 @@ show_help(char *path)
"eD | Show current file details\n"
"em | Show concise media info\n"
"eM | Show full media info\n"
+ "d^R | Rename selected entry\n"
"es | Toggle sort by file size\n"
"eS | Toggle disk usage mode\n"
"et | Toggle sort by mtime\n"
@@ -2322,7 +2323,7 @@ nochange:
tmp = readinput();
clearprompt();
if (tmp == NULL)
- goto nochange;
+ break;
for (r = 0; bookmark[r].key && r < MAX_BM; ++r) {
if (xstrcmp(bookmark[r].key, tmp) == -1)
@@ -2389,9 +2390,6 @@ nochange:
mkpath(path, dents[cur].name, oldpath, PATH_MAX);
goto begin;
case SEL_STATS:
- {
- struct stat sb;
-
if (ndents > 0) {
mkpath(path, dents[cur].name, oldpath, PATH_MAX);
@@ -2403,14 +2401,12 @@ nochange:
} else {
r = show_stats(oldpath, dents[cur].name, &sb);
if (r < 0) {
- printmsg(strerror(errno));
+ printwarn();
goto nochange;
}
}
}
-
break;
- }
case SEL_MEDIA: // fallthrough
case SEL_FMEDIA:
if (ndents > 0) {
@@ -2475,6 +2471,59 @@ nochange:
} else if (!copier)
printmsg("NNN_COPIER is not set");
goto nochange;
+ case SEL_RENAME:
+ if (ndents <= 0)
+ break;
+
+ printprompt("rename to: ");
+ tmp = readinput();
+ clearprompt();
+ if (tmp == NULL)
+ break;
+
+ /* Allow only relative paths */
+ if (tmp[0] == '/' || basename(tmp) != tmp) {
+ printmsg("relative paths only");
+ goto nochange;
+ }
+
+ /* Skip renaming to same name */
+ if (xstrcmp(tmp, dents[cur].name) == 0)
+ break;
+
+ /* Open the descriptor to currently open directory */
+ fd = open(path, O_RDONLY | O_DIRECTORY | O_NOATIME);
+ if (fd == -1) {
+ printwarn();
+ goto nochange;
+ }
+
+ /* Check if another file with same name exists */
+ if (faccessat(fd, tmp, F_OK, AT_SYMLINK_NOFOLLOW) != -1) {
+ /* File with the same name exists */
+ xstrlcpy(g_buf, tmp, NAME_MAX);
+
+ printprompt("overwrite? (y): ");
+ tmp = readinput();
+ if (tmp == NULL || tmp[0] != 'y' || tmp[1] != '\0') {
+ close(fd);
+ break;
+ }
+
+ tmp = g_buf;
+ }
+
+ /* Rename the file */
+ r = renameat(fd, dents[cur].name, fd, tmp);
+ if (r != 0) {
+ printwarn();
+ close(fd);
+ goto nochange;
+ }
+
+ close(fd);
+ mkpath(path, tmp, oldpath, PATH_MAX);
+ goto begin;
case SEL_HELP:
show_help(path);
break;