aboutsummaryrefslogtreecommitdiffstats
path: root/noice.c
diff options
context:
space:
mode:
Diffstat (limited to 'noice.c')
-rw-r--r--noice.c185
1 files changed, 106 insertions, 79 deletions
diff --git a/noice.c b/noice.c
index 6232489..ac965bd 100644
--- a/noice.c
+++ b/noice.c
@@ -75,6 +75,12 @@ struct entry {
time_t t;
};
+/* Global context */
+struct entry *dents;
+int n, cur;
+char *path, *oldpath;
+char *fltr;
+
/*
* Layout:
* .---------
@@ -565,36 +571,29 @@ dentfind(struct entry *dents, int n, char *cwd, char *path)
return 0;
}
-void
-browse(const char *ipath, const char *ifilter)
+int
+populate(void)
{
- struct entry *dents;
- int i, n, cur, r, fd;
- int nlines, odd;
- char *path = xstrdup(ipath);
- char *filter = xstrdup(ifilter);
- regex_t filter_re, re;
- char *cwd, *newpath, *oldpath = NULL;
- struct stat sb;
- char *name, *bin, *dir, *tmp, *run;
- int nowtyping = 0;
-
-begin:
- /* Path and filter should be malloc(3)-ed strings at all times */
- n = 0;
- dents = NULL;
+ regex_t re;
+ int r;
+ /* Can fail when permissions change while browsing */
if (canopendir(path) == 0) {
printwarn();
- goto nochange;
+ return -1;
}
/* Search filter */
- r = setfilter(&filter_re, filter);
+ r = setfilter(&re, fltr);
if (r != 0)
- goto nochange;
+ return -1;
+
+ dentfree(dents, n);
+
+ n = 0;
+ dents = NULL;
- n = dentfill(path, &dents, visible, &filter_re);
+ n = dentfill(path, &dents, visible, &re);
qsort(dents, n, sizeof(*dents), entrycmp);
@@ -603,52 +602,85 @@ begin:
free(oldpath);
oldpath = NULL;
- for (;;) {
- nlines = MIN(LINES - 4, n);
+ return 0;
+}
- /* Clean screen */
- erase();
+void
+redraw(void)
+{
+ int nlines, odd;
+ char *cwd;
+ int i;
- /* Strip trailing slashes */
- for (i = strlen(path) - 1; i > 0; i--)
- if (path[i] == '/')
- path[i] = '\0';
- else
- break;
+ nlines = MIN(LINES - 4, n);
- DPRINTF_D(cur);
- DPRINTF_S(path);
+ /* Clean screen */
+ erase();
- /* No text wrapping in cwd line */
- cwd = xmalloc(COLS * sizeof(char));
- strlcpy(cwd, path, COLS * sizeof(char));
- cwd[COLS - strlen(CWD) - 1] = '\0';
-
- printw(CWD "%s\n\n", cwd);
-
- /* Print listing */
- odd = ISODD(nlines);
- if (cur < nlines / 2) {
- for (i = 0; i < nlines; i++)
- printent(&dents[i], i == cur);
- } else if (cur >= n - nlines / 2) {
- for (i = n - nlines; i < n; i++)
- printent(&dents[i], i == cur);
- } else {
- for (i = cur - nlines / 2;
- i < cur + nlines / 2 + odd; i++)
- printent(&dents[i], i == cur);
- }
+ /* Strip trailing slashes */
+ for (i = strlen(path) - 1; i > 0; i--)
+ if (path[i] == '/')
+ path[i] = '\0';
+ else
+ break;
+
+ DPRINTF_D(cur);
+ DPRINTF_S(path);
+
+ /* No text wrapping in cwd line */
+ cwd = xmalloc(COLS * sizeof(char));
+ strlcpy(cwd, path, COLS * sizeof(char));
+ cwd[COLS - strlen(CWD) - 1] = '\0';
+
+ printw(CWD "%s\n\n", cwd);
+
+ /* Print listing */
+ odd = ISODD(nlines);
+ if (cur < nlines / 2) {
+ for (i = 0; i < nlines; i++)
+ printent(&dents[i], i == cur);
+ } else if (cur >= n - nlines / 2) {
+ for (i = n - nlines; i < n; i++)
+ printent(&dents[i], i == cur);
+ } else {
+ for (i = cur - nlines / 2;
+ i < cur + nlines / 2 + odd; i++)
+ printent(&dents[i], i == cur);
+ }
+}
+
+void
+browse(const char *ipath, const char *ifilter)
+{
+ int r, fd;
+ regex_t re;
+ char *newpath;
+ struct stat sb;
+ char *name, *bin, *dir, *tmp, *run;
+ int nowtyping = 0;
+
+ oldpath = NULL;
+ path = xstrdup(ipath);
+ fltr = xstrdup(ifilter);
+begin:
+ /* Path and filter should be malloc(3)-ed strings at all times */
+ r = populate();
+ if (r == -1) {
+ nowtyping = 0;
+ goto nochange;
+ }
+
+ for (;;) {
+ redraw();
/* Handle filter-as-you-type mode */
if (nowtyping)
goto moretyping;
-
nochange:
switch (nextsel(&run)) {
case SEL_QUIT:
free(path);
- free(filter);
+ free(fltr);
dentfree(dents, n);
return;
case SEL_BACK:
@@ -666,9 +698,9 @@ nochange:
oldpath = path;
path = dir;
/* Reset filter */
- free(filter);
- filter = xstrdup(ifilter);
- goto out;
+ free(fltr);
+ fltr = xstrdup(ifilter);
+ goto begin;
case SEL_GOIN:
/* Cannot descend in empty directories */
if (n == 0)
@@ -705,9 +737,9 @@ nochange:
free(path);
path = newpath;
/* Reset filter */
- free(filter);
- filter = xstrdup(ifilter);
- goto out;
+ free(fltr);
+ fltr = xstrdup(ifilter);
+ goto begin;
case S_IFREG:
bin = openwith(newpath);
if (bin == NULL) {
@@ -736,13 +768,13 @@ nochange:
free(tmp);
goto nochange;
}
- free(filter);
- filter = tmp;
- DPRINTF_S(filter);
+ free(fltr);
+ fltr = tmp;
+ DPRINTF_S(fltr);
/* Save current */
if (n > 0)
oldpath = makepath(path, dents[cur].name);
- goto out;
+ goto begin;
case SEL_TYPE:
nowtyping = 1;
tmp = NULL;
@@ -767,17 +799,17 @@ moretyping:
}
}
/* Copy or reset filter */
- free(filter);
+ free(fltr);
if (tmp != NULL)
- filter = xstrdup(tmp);
+ fltr = xstrdup(tmp);
else
- filter = xstrdup(ifilter);
+ fltr = xstrdup(ifilter);
/* Save current */
if (n > 0)
oldpath = makepath(path, dents[cur].name);
if (!nowtyping)
free(tmp);
- goto out;
+ goto begin;
case SEL_NEXT:
if (cur < n - 1)
cur++;
@@ -811,15 +843,15 @@ moretyping:
}
free(path);
path = newpath;
- free(filter);
- filter = xstrdup(ifilter); /* Reset filter */
+ free(fltr);
+ fltr = xstrdup(ifilter); /* Reset filter */
DPRINTF_S(path);
- goto out;
+ goto begin;
case SEL_MTIME:
mtimeorder = !mtimeorder;
- goto out;
+ goto begin;
case SEL_REDRAW:
- goto out;
+ goto begin;
case SEL_RUN:
exitcurses();
spawn(run, NULL, path);
@@ -833,11 +865,6 @@ moretyping:
break;
}
}
-
-out:
- dentfree(dents, n);
-
- goto begin;
}
int