mirror of
https://github.com/davatorium/rofi.git
synced 2024-11-18 13:54:36 -05:00
75ae2a41f0
* implemented xdg thumbnails fetching with fallback on mimetype icons for menu entries in filebrowser mode * included original license text * added md5 header and source file * implemented xdg compatible thumbnail's creation * added -preview-cmd string option to program settings * support custom command to create images for entries with thumbnail:// prefix * fix custom thumbnailer command crash caused by null uri when entry is not a valid filename * check entry_name is not NULL or empty when generating thumbnails; use snprintf to avoid static analyzer complains * avoid using gstrvbuilder to build thumbnailer command args * fixed static analyzer complain about always wrong condition * use g_spawn_check_exit_status to avoid bump to glib 2.70 * removed md5-c dependency and use glib checksum implementation * fixed meson build after md5-c library removal * support thumbnail generation in recursivebrowser mode * restored check rofi_icon_fetcher_file_is_image * create thumbnail directories if not existing * use g_malloc0, g_strdup and g_strdup_printf * fixed formatting with clang-format * don't wait for jobs in execution when finalizing the icon fetcher worker threadpool * destroy and rebuild the icon fetcher worker threadpool when the current page is changed * added query_started boolean member to IconFetcherEntry; check if an icon fetcher query was started on an IconFetcherEntry and submit the query again otherwise * force icon cache lookup even if the item has a valid icon_fetch_uid (the fetching job could have been discarded before starting) * search binaries in PATH when executing thumbnailer command * mark icon query as not started in threadpool item free_func * added listview page_changed_callback; rebuild icon fetcher threadpool in page_changed_callback * [listview] Add missing code documentation param * Create rofi-thumbnails.5.markdown * Updated documentation with apparmor issues and workaround * [Doc] Ship rofi-thumbnails.5 With some formatting fixes * use a more compact thumbnailer example --------- Co-authored-by: giomatfois62 <giomatfois62@yahoo.it> Co-authored-by: Dave Davenport <DaveDavenport@users.noreply.github.com> Co-authored-by: lbonn <github@lbonnans.net>
582 lines
18 KiB
C
582 lines
18 KiB
C
/**
|
|
* rofi-recursive_browser
|
|
*
|
|
* MIT/X11 License
|
|
* Copyright (c) 2017 Qball Cow <qball@gmpclient.org>
|
|
*
|
|
* Permission is hereby granted, free of charge, to any person obtaining
|
|
* a copy of this software and associated documentation files (the
|
|
* "Software"), to deal in the Software without restriction, including
|
|
* without limitation the rights to use, copy, modify, merge, publish,
|
|
* distribute, sublicense, and/or sell copies of the Software, and to
|
|
* permit persons to whom the Software is furnished to do so, subject to
|
|
* the following conditions:
|
|
*
|
|
* The above copyright notice and this permission notice shall be
|
|
* included in all copies or substantial portions of the Software.
|
|
*
|
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
|
|
* OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
|
* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.
|
|
* IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY
|
|
* CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT,
|
|
* TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE
|
|
* SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
|
*/
|
|
#define G_LOG_DOMAIN "Modes.RecursiveBrowser"
|
|
|
|
#include "config.h"
|
|
#include <errno.h>
|
|
#include <gio/gio.h>
|
|
#include <gmodule.h>
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
#include <unistd.h>
|
|
|
|
#include <dirent.h>
|
|
#include <glib-unix.h>
|
|
#include <glib/gstdio.h>
|
|
#include <sys/stat.h>
|
|
#include <sys/types.h>
|
|
|
|
#include "helper.h"
|
|
#include "history.h"
|
|
#include "mode-private.h"
|
|
#include "mode.h"
|
|
#include "modes/recursivebrowser.h"
|
|
#include "rofi.h"
|
|
#include "theme.h"
|
|
|
|
#include <stdint.h>
|
|
|
|
#include "rofi-icon-fetcher.h"
|
|
|
|
/** The default program used to open the file. */
|
|
#define DEFAULT_OPEN "xdg-open"
|
|
|
|
/**
|
|
* The internal data structure holding the private data of the TEST Mode.
|
|
*/
|
|
enum FBFileType {
|
|
UP,
|
|
DIRECTORY,
|
|
RFILE,
|
|
NUM_FILE_TYPES,
|
|
};
|
|
|
|
/** Icons to use for the file type */
|
|
const char *rb_icon_name[NUM_FILE_TYPES] = {"go-up", "folder", "gtk-file"};
|
|
typedef struct {
|
|
char *name;
|
|
char *path;
|
|
enum FBFileType type;
|
|
uint32_t icon_fetch_uid;
|
|
uint32_t icon_fetch_size;
|
|
gboolean link;
|
|
time_t time;
|
|
} FBFile;
|
|
|
|
typedef struct {
|
|
char *command;
|
|
GFile *current_dir;
|
|
FBFile *array;
|
|
unsigned int array_length;
|
|
unsigned int array_length_real;
|
|
|
|
GThread *reading_thread;
|
|
GAsyncQueue *async_queue;
|
|
guint wake_source;
|
|
guint end_thread;
|
|
gboolean loading;
|
|
int pipefd2[2];
|
|
GRegex *filter_regex;
|
|
} FileBrowserModePrivateData;
|
|
|
|
static void free_list(FileBrowserModePrivateData *pd) {
|
|
for (unsigned int i = 0; i < pd->array_length; i++) {
|
|
FBFile *fb = &(pd->array[i]);
|
|
g_free(fb->name);
|
|
g_free(fb->path);
|
|
}
|
|
g_free(pd->array);
|
|
pd->array = NULL;
|
|
pd->array_length = 0;
|
|
pd->array_length_real = 0;
|
|
}
|
|
#include <dirent.h>
|
|
#include <sys/types.h>
|
|
|
|
inline static void fb_resize_array(FileBrowserModePrivateData *pd) {
|
|
if ((pd->array_length + 1) > pd->array_length_real) {
|
|
pd->array_length_real += 10240;
|
|
pd->array =
|
|
g_realloc(pd->array, (pd->array_length_real + 1) * sizeof(FBFile));
|
|
}
|
|
}
|
|
|
|
static void recursive_browser_mode_init_config(Mode *sw) {
|
|
FileBrowserModePrivateData *pd =
|
|
(FileBrowserModePrivateData *)mode_get_private_data(sw);
|
|
char *msg = NULL;
|
|
gboolean found_error = FALSE;
|
|
|
|
ThemeWidget *wid = rofi_config_find_widget(sw->name, NULL, TRUE);
|
|
Property *p =
|
|
rofi_theme_find_property(wid, P_BOOLEAN, "cancel-returns-1", TRUE);
|
|
if (p && p->type == P_BOOLEAN && p->value.b == TRUE) {
|
|
rofi_set_return_code(1);
|
|
}
|
|
|
|
p = rofi_theme_find_property(wid, P_STRING, "filter-regex", TRUE);
|
|
if (p != NULL && p->type == P_STRING) {
|
|
GError *error = NULL;
|
|
g_debug("compile regex: %s\n", p->value.s);
|
|
pd->filter_regex = g_regex_new(p->value.s, G_REGEX_OPTIMIZE, 0, &error);
|
|
if (error) {
|
|
msg = g_strdup_printf("\"%s\" is not a valid regex for filtering: %s",
|
|
p->value.s, error->message);
|
|
found_error = TRUE;
|
|
g_error_free(error);
|
|
}
|
|
}
|
|
if (pd->filter_regex == NULL) {
|
|
g_debug("compile default regex\n");
|
|
pd->filter_regex = g_regex_new("^(\\..*)", G_REGEX_OPTIMIZE, 0, NULL);
|
|
}
|
|
p = rofi_theme_find_property(wid, P_STRING, "command", TRUE);
|
|
if (p != NULL && p->type == P_STRING) {
|
|
pd->command = g_strdup(p->value.s);
|
|
} else {
|
|
pd->command = g_strdup(DEFAULT_OPEN);
|
|
}
|
|
|
|
if (found_error) {
|
|
rofi_view_error_dialog(msg, FALSE);
|
|
|
|
g_free(msg);
|
|
}
|
|
}
|
|
|
|
static void recursive_browser_mode_init_current_dir(Mode *sw) {
|
|
FileBrowserModePrivateData *pd =
|
|
(FileBrowserModePrivateData *)mode_get_private_data(sw);
|
|
|
|
ThemeWidget *wid = rofi_config_find_widget(sw->name, NULL, TRUE);
|
|
|
|
Property *p = rofi_theme_find_property(wid, P_STRING, "directory", TRUE);
|
|
|
|
gboolean config_has_valid_dir = p != NULL && p->type == P_STRING &&
|
|
g_file_test(p->value.s, G_FILE_TEST_IS_DIR);
|
|
|
|
if (config_has_valid_dir) {
|
|
pd->current_dir = g_file_new_for_path(p->value.s);
|
|
}
|
|
if (pd->current_dir == NULL) {
|
|
pd->current_dir = g_file_new_for_path(g_get_home_dir());
|
|
}
|
|
}
|
|
|
|
static void scan_dir(FileBrowserModePrivateData *pd, GFile *path) {
|
|
GQueue *dirs_to_scan = g_queue_new();
|
|
g_queue_push_tail(dirs_to_scan, g_object_ref(path));
|
|
GFile *dir_to_scan = NULL;
|
|
while ( (dir_to_scan = g_queue_pop_head(dirs_to_scan)) != NULL) {
|
|
char *cdir = g_file_get_path(dir_to_scan);
|
|
DIR *dir = opendir(cdir);
|
|
g_object_unref(dir_to_scan);
|
|
if (dir) {
|
|
struct dirent *rd = NULL;
|
|
while (pd->end_thread == FALSE && (rd = readdir(dir)) != NULL) {
|
|
if (g_strcmp0(rd->d_name, "..") == 0) {
|
|
continue;
|
|
}
|
|
if (g_strcmp0(rd->d_name, ".") == 0) {
|
|
continue;
|
|
}
|
|
if (pd->filter_regex &&
|
|
g_regex_match(pd->filter_regex, rd->d_name, 0, NULL)) {
|
|
continue;
|
|
}
|
|
switch (rd->d_type) {
|
|
case DT_BLK:
|
|
case DT_CHR:
|
|
case DT_FIFO:
|
|
case DT_SOCK:
|
|
default:
|
|
break;
|
|
case DT_REG: {
|
|
FBFile *f = g_malloc0(sizeof(FBFile));
|
|
// Rofi expects utf-8, so lets convert the filename.
|
|
f->path = g_build_filename(cdir, rd->d_name, NULL);
|
|
f->name = g_filename_to_utf8(f->path, -1, NULL, NULL, NULL);
|
|
if (f->name == NULL) {
|
|
f->name = rofi_force_utf8(rd->d_name, -1);
|
|
}
|
|
if (f->name == NULL) {
|
|
f->name = g_strdup("n/a");
|
|
}
|
|
f->type = (rd->d_type == DT_DIR) ? DIRECTORY : RFILE;
|
|
f->icon_fetch_uid = 0;
|
|
f->icon_fetch_size = 0;
|
|
f->link = FALSE;
|
|
|
|
g_async_queue_push(pd->async_queue, f);
|
|
if (g_async_queue_length(pd->async_queue) > 10000) {
|
|
write(pd->pipefd2[1], "r", 1);
|
|
}
|
|
break;
|
|
}
|
|
case DT_DIR: {
|
|
char *d = g_build_filename(cdir, rd->d_name, NULL);
|
|
GFile *dirp = g_file_new_for_path(d);
|
|
g_queue_push_tail(dirs_to_scan, dirp);
|
|
g_free(d);
|
|
break;
|
|
}
|
|
case DT_UNKNOWN:
|
|
case DT_LNK: {
|
|
FBFile *f = g_malloc0(sizeof(FBFile));
|
|
// Rofi expects utf-8, so lets convert the filename.
|
|
f->path = g_build_filename(cdir, rd->d_name, NULL);
|
|
f->name = g_filename_to_utf8(f->path, -1, NULL, NULL, NULL);
|
|
if (f->name == NULL) {
|
|
f->name = rofi_force_utf8(rd->d_name, -1);
|
|
}
|
|
if (f->name == NULL) {
|
|
f->name = g_strdup("n/a");
|
|
}
|
|
f->icon_fetch_uid = 0;
|
|
f->icon_fetch_size = 0;
|
|
// Default to file.
|
|
f->type = RFILE;
|
|
if (rd->d_type == DT_LNK) {
|
|
f->link = TRUE;
|
|
} else {
|
|
f->link = FALSE;
|
|
}
|
|
{
|
|
// If we have link, use a stat to fine out what it is, if we fail, we
|
|
// mark it as file.
|
|
// TODO have a 'broken link' mode?
|
|
// Convert full path to right encoding.
|
|
// DD: Path should be in file encoding, not utf-8
|
|
// char *file =
|
|
// g_filename_from_utf8(pd->array[pd->array_length].path,
|
|
// -1, NULL, NULL, NULL);
|
|
// TODO: How to handle loops in links.
|
|
if (f->path) {
|
|
GStatBuf statbuf;
|
|
if (g_stat(f->path, &statbuf) == 0) {
|
|
if (S_ISDIR(statbuf.st_mode)) {
|
|
char *new_full_path = g_build_filename(cdir, rd->d_name, NULL);
|
|
g_free(f->path);
|
|
g_free(f->name);
|
|
g_free(f);
|
|
f = NULL;
|
|
GFile *dirp = g_file_new_for_path(new_full_path);
|
|
g_queue_push_tail(dirs_to_scan, dirp);
|
|
g_free(new_full_path);
|
|
break;
|
|
} else if (S_ISREG(statbuf.st_mode)) {
|
|
f->type = RFILE;
|
|
}
|
|
|
|
} else {
|
|
g_warning("Failed to stat file: %s, %s", f->path,
|
|
strerror(errno));
|
|
}
|
|
|
|
// g_free(file);
|
|
}
|
|
}
|
|
if (f != NULL) {
|
|
g_async_queue_push(pd->async_queue, f);
|
|
if (g_async_queue_length(pd->async_queue) > 10000) {
|
|
write(pd->pipefd2[1], "r", 1);
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
closedir(dir);
|
|
}
|
|
g_free(cdir);
|
|
}
|
|
|
|
g_queue_free(dirs_to_scan);
|
|
}
|
|
static gpointer recursive_browser_input_thread(gpointer userdata) {
|
|
FileBrowserModePrivateData *pd = (FileBrowserModePrivateData *)userdata;
|
|
GTimer *t = g_timer_new();
|
|
g_debug("Start scan.\n");
|
|
scan_dir(pd, pd->current_dir);
|
|
write(pd->pipefd2[1], "r", 1);
|
|
write(pd->pipefd2[1], "q", 1);
|
|
double f = g_timer_elapsed(t, NULL);
|
|
g_debug("End scan: %f\n", f);
|
|
g_timer_destroy(t);
|
|
return NULL;
|
|
}
|
|
static gboolean recursive_browser_async_read_proc(gint fd,
|
|
GIOCondition condition,
|
|
gpointer user_data) {
|
|
FileBrowserModePrivateData *pd = (FileBrowserModePrivateData *)user_data;
|
|
char command;
|
|
// Only interrested in read events.
|
|
if ((condition & G_IO_IN) != G_IO_IN) {
|
|
return G_SOURCE_CONTINUE;
|
|
}
|
|
// Read the entry from the pipe that was used to signal this action.
|
|
if (read(fd, &command, 1) == 1) {
|
|
if (command == 'r') {
|
|
FBFile *block = NULL;
|
|
gboolean changed = FALSE;
|
|
// Empty out the AsyncQueue (that is thread safe) from all blocks pushed
|
|
// into it.
|
|
while ((block = g_async_queue_try_pop(pd->async_queue)) != NULL) {
|
|
|
|
fb_resize_array(pd);
|
|
pd->array[pd->array_length] = *block;
|
|
pd->array_length++;
|
|
g_free(block);
|
|
changed = TRUE;
|
|
}
|
|
if (changed) {
|
|
rofi_view_reload();
|
|
}
|
|
} else if (command == 'q') {
|
|
if (pd->loading) {
|
|
rofi_view_set_overlay(rofi_view_get_active(), NULL);
|
|
}
|
|
}
|
|
}
|
|
return G_SOURCE_CONTINUE;
|
|
}
|
|
|
|
static int recursive_browser_mode_init(Mode *sw) {
|
|
/**
|
|
* Called on startup when enabled (in modes list)
|
|
*/
|
|
if (mode_get_private_data(sw) == NULL) {
|
|
FileBrowserModePrivateData *pd = g_malloc0(sizeof(*pd));
|
|
mode_set_private_data(sw, (void *)pd);
|
|
|
|
recursive_browser_mode_init_config(sw);
|
|
recursive_browser_mode_init_current_dir(sw);
|
|
|
|
// Load content.
|
|
if (pipe(pd->pipefd2) == -1) {
|
|
g_error("Failed to create pipe");
|
|
}
|
|
pd->wake_source = g_unix_fd_add(pd->pipefd2[0], G_IO_IN,
|
|
recursive_browser_async_read_proc, pd);
|
|
|
|
// Create the message passing queue to the UI thread.
|
|
pd->async_queue = g_async_queue_new();
|
|
pd->end_thread = FALSE;
|
|
pd->reading_thread = g_thread_new(
|
|
"dmenu-read", (GThreadFunc)recursive_browser_input_thread, pd);
|
|
pd->loading = TRUE;
|
|
}
|
|
return TRUE;
|
|
}
|
|
static unsigned int recursive_browser_mode_get_num_entries(const Mode *sw) {
|
|
const FileBrowserModePrivateData *pd =
|
|
(const FileBrowserModePrivateData *)mode_get_private_data(sw);
|
|
return pd->array_length;
|
|
}
|
|
|
|
static ModeMode recursive_browser_mode_result(Mode *sw, int mretv,
|
|
G_GNUC_UNUSED char **input,
|
|
unsigned int selected_line) {
|
|
ModeMode retv = MODE_EXIT;
|
|
FileBrowserModePrivateData *pd =
|
|
(FileBrowserModePrivateData *)mode_get_private_data(sw);
|
|
|
|
if ((mretv & MENU_CANCEL) == MENU_CANCEL) {
|
|
ThemeWidget *wid = rofi_config_find_widget(sw->name, NULL, TRUE);
|
|
Property *p =
|
|
rofi_theme_find_property(wid, P_BOOLEAN, "cancel-returns-1", TRUE);
|
|
if (p && p->type == P_BOOLEAN && p->value.b == TRUE) {
|
|
rofi_set_return_code(1);
|
|
}
|
|
|
|
return MODE_EXIT;
|
|
}
|
|
if (mretv & MENU_CUSTOM_COMMAND) {
|
|
retv = (mretv & MENU_LOWER_MASK);
|
|
} else if ((mretv & MENU_OK)) {
|
|
if (selected_line < pd->array_length) {
|
|
if (pd->array[selected_line].type == RFILE) {
|
|
char *d_esc = g_shell_quote(pd->array[selected_line].path);
|
|
char *cmd = g_strdup_printf("%s %s", pd->command, d_esc);
|
|
g_free(d_esc);
|
|
char *cdir = g_file_get_path(pd->current_dir);
|
|
helper_execute_command(cdir, cmd, FALSE, NULL);
|
|
g_free(cdir);
|
|
g_free(cmd);
|
|
return MODE_EXIT;
|
|
}
|
|
}
|
|
retv = RELOAD_DIALOG;
|
|
} else if ((mretv & MENU_CUSTOM_INPUT)) {
|
|
retv = RELOAD_DIALOG;
|
|
} else if ((mretv & MENU_ENTRY_DELETE) == MENU_ENTRY_DELETE) {
|
|
retv = RELOAD_DIALOG;
|
|
}
|
|
return retv;
|
|
}
|
|
|
|
static void recursive_browser_mode_destroy(Mode *sw) {
|
|
FileBrowserModePrivateData *pd =
|
|
(FileBrowserModePrivateData *)mode_get_private_data(sw);
|
|
if (pd != NULL) {
|
|
if (pd->reading_thread) {
|
|
pd->end_thread = TRUE;
|
|
g_thread_join(pd->reading_thread);
|
|
}
|
|
if (pd->filter_regex) {
|
|
g_regex_unref(pd->filter_regex);
|
|
}
|
|
g_object_unref(pd->current_dir);
|
|
g_free(pd->command);
|
|
free_list(pd);
|
|
g_free(pd);
|
|
mode_set_private_data(sw, NULL);
|
|
}
|
|
}
|
|
|
|
static char *_get_display_value(const Mode *sw, unsigned int selected_line,
|
|
G_GNUC_UNUSED int *state,
|
|
G_GNUC_UNUSED GList **attr_list,
|
|
int get_entry) {
|
|
FileBrowserModePrivateData *pd =
|
|
(FileBrowserModePrivateData *)mode_get_private_data(sw);
|
|
|
|
// Only return the string if requested, otherwise only set state.
|
|
if (!get_entry) {
|
|
return NULL;
|
|
}
|
|
if (pd->array[selected_line].type == UP) {
|
|
return g_strdup(" ..");
|
|
}
|
|
if (pd->array[selected_line].link) {
|
|
return g_strconcat("@", pd->array[selected_line].name, NULL);
|
|
}
|
|
return g_strdup(pd->array[selected_line].name);
|
|
}
|
|
|
|
/**
|
|
* @param sw The mode object.
|
|
* @param tokens The tokens to match against.
|
|
* @param index The index in this plugin to match against.
|
|
*
|
|
* Match the entry.
|
|
*
|
|
* @returns try when a match.
|
|
*/
|
|
static int recursive_browser_token_match(const Mode *sw,
|
|
rofi_int_matcher **tokens,
|
|
unsigned int index) {
|
|
FileBrowserModePrivateData *pd =
|
|
(FileBrowserModePrivateData *)mode_get_private_data(sw);
|
|
|
|
// Call default matching function.
|
|
return helper_token_match(tokens, pd->array[index].name);
|
|
}
|
|
|
|
static cairo_surface_t *_get_icon(const Mode *sw, unsigned int selected_line,
|
|
unsigned int height) {
|
|
FileBrowserModePrivateData *pd =
|
|
(FileBrowserModePrivateData *)mode_get_private_data(sw);
|
|
g_return_val_if_fail(pd->array != NULL, NULL);
|
|
FBFile *dr = &(pd->array[selected_line]);
|
|
if (rofi_icon_fetcher_file_is_image(dr->path)) {
|
|
dr->icon_fetch_uid = rofi_icon_fetcher_query(dr->path, height);
|
|
} else if (dr->type == RFILE) {
|
|
gchar* _path = g_strconcat("thumbnail://", dr->path, NULL);
|
|
dr->icon_fetch_uid = rofi_icon_fetcher_query(_path, height);
|
|
g_free(_path);
|
|
} else {
|
|
dr->icon_fetch_uid = rofi_icon_fetcher_query(rb_icon_name[dr->type], height);
|
|
}
|
|
dr->icon_fetch_size = height;
|
|
return rofi_icon_fetcher_get(dr->icon_fetch_uid);
|
|
}
|
|
|
|
static char *_get_message(const Mode *sw) {
|
|
FileBrowserModePrivateData *pd =
|
|
(FileBrowserModePrivateData *)mode_get_private_data(sw);
|
|
if (pd->current_dir) {
|
|
char *dirname = g_file_get_parse_name(pd->current_dir);
|
|
char *str =
|
|
g_markup_printf_escaped("<b>Current directory:</b> %s", dirname);
|
|
g_free(dirname);
|
|
return str;
|
|
}
|
|
return "n/a";
|
|
}
|
|
|
|
static char *_get_completion(const Mode *sw, unsigned int index) {
|
|
FileBrowserModePrivateData *pd =
|
|
(FileBrowserModePrivateData *)mode_get_private_data(sw);
|
|
|
|
char *d = g_strescape(pd->array[index].path, NULL);
|
|
return d;
|
|
}
|
|
|
|
Mode *create_new_recursive_browser(void) {
|
|
Mode *sw = g_malloc0(sizeof(Mode));
|
|
|
|
*sw = recursive_browser_mode;
|
|
|
|
sw->private_data = NULL;
|
|
return sw;
|
|
}
|
|
|
|
#if 1
|
|
ModeMode recursive_browser_mode_completer(Mode *sw, int mretv, char **input,
|
|
unsigned int selected_line,
|
|
char **path) {
|
|
ModeMode retv = MODE_EXIT;
|
|
FileBrowserModePrivateData *pd =
|
|
(FileBrowserModePrivateData *)mode_get_private_data(sw);
|
|
if ((mretv & MENU_OK)) {
|
|
if (selected_line < pd->array_length) {
|
|
if (pd->array[selected_line].type == RFILE) {
|
|
*path = g_strescape(pd->array[selected_line].path, NULL);
|
|
return MODE_EXIT;
|
|
}
|
|
}
|
|
retv = RELOAD_DIALOG;
|
|
} else if ((mretv & MENU_CUSTOM_INPUT) && *input) {
|
|
retv = RELOAD_DIALOG;
|
|
} else if ((mretv & MENU_ENTRY_DELETE) == MENU_ENTRY_DELETE) {
|
|
retv = RELOAD_DIALOG;
|
|
}
|
|
return retv;
|
|
}
|
|
#endif
|
|
|
|
Mode recursive_browser_mode = {
|
|
.display_name = NULL,
|
|
.abi_version = ABI_VERSION,
|
|
.name = "recursivebrowser",
|
|
.cfg_name_key = "display-recursivebrowser",
|
|
._init = recursive_browser_mode_init,
|
|
._get_num_entries = recursive_browser_mode_get_num_entries,
|
|
._result = recursive_browser_mode_result,
|
|
._destroy = recursive_browser_mode_destroy,
|
|
._token_match = recursive_browser_token_match,
|
|
._get_display_value = _get_display_value,
|
|
._get_icon = _get_icon,
|
|
._get_message = _get_message,
|
|
._get_completion = _get_completion,
|
|
._preprocess_input = NULL,
|
|
._create = create_new_recursive_browser,
|
|
._completer_result = recursive_browser_mode_completer,
|
|
.private_data = NULL,
|
|
.free = NULL,
|
|
.type = MODE_TYPE_SWITCHER | MODE_TYPE_COMPLETER};
|