Modularized modes and separated them into their own source files
This commit is contained in:
58
modes/dmenu.c
Normal file
58
modes/dmenu.c
Normal file
@@ -0,0 +1,58 @@
|
||||
/*
|
||||
* Copyright (C) 2019 Scoopta
|
||||
* This file is part of Wofi
|
||||
* Wofi is free software: you can redistribute it and/or modify
|
||||
it under the terms of the GNU General Public License as published by
|
||||
the Free Software Foundation, either version 3 of the License, or
|
||||
(at your option) any later version.
|
||||
|
||||
Wofi is distributed in the hope that it will be useful,
|
||||
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
GNU General Public License for more details.
|
||||
|
||||
You should have received a copy of the GNU General Public License
|
||||
along with Wofi. If not, see <http://www.gnu.org/licenses/>.
|
||||
*/
|
||||
|
||||
#include <wofi.h>
|
||||
|
||||
void dmenu_init() {
|
||||
struct map* cached = map_init();
|
||||
struct wl_list* cache = wofi_read_cache("dmenu");
|
||||
|
||||
struct cache_line* node, *tmp;
|
||||
wl_list_for_each_safe(node, tmp, cache, link) {
|
||||
wofi_insert_widget(node->line, node->line);
|
||||
map_put(cached, node->line, "true");
|
||||
free(node->line);
|
||||
wl_list_remove(&node->link);
|
||||
free(node);
|
||||
}
|
||||
|
||||
free(cache);
|
||||
|
||||
char* line;
|
||||
size_t size = 0;
|
||||
while(getline(&line, &size, stdin) != -1) {
|
||||
char* lf = strchr(line, '\n');
|
||||
if(lf != NULL) {
|
||||
*lf = 0;
|
||||
}
|
||||
if(map_contains(cached, line)) {
|
||||
continue;
|
||||
}
|
||||
wofi_insert_widget(line, line);
|
||||
}
|
||||
free(line);
|
||||
map_free(cached);
|
||||
}
|
||||
|
||||
void dmenu_exec(const gchar* cmd) {
|
||||
printf("%s\n", cmd);
|
||||
exit(0);
|
||||
}
|
||||
|
||||
bool dmenu_exec_search() {
|
||||
return true;
|
||||
}
|
138
modes/drun.c
Normal file
138
modes/drun.c
Normal file
@@ -0,0 +1,138 @@
|
||||
/*
|
||||
* Copyright (C) 2019 Scoopta
|
||||
* This file is part of Wofi
|
||||
* Wofi is free software: you can redistribute it and/or modify
|
||||
it under the terms of the GNU General Public License as published by
|
||||
the Free Software Foundation, either version 3 of the License, or
|
||||
(at your option) any later version.
|
||||
|
||||
Wofi is distributed in the hope that it will be useful,
|
||||
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
GNU General Public License for more details.
|
||||
|
||||
You should have received a copy of the GNU General Public License
|
||||
along with Wofi. If not, see <http://www.gnu.org/licenses/>.
|
||||
*/
|
||||
|
||||
#include <wofi.h>
|
||||
|
||||
static char* get_text(char* file) {
|
||||
GDesktopAppInfo* info = g_desktop_app_info_new_from_filename(file);
|
||||
if(info == NULL || g_desktop_app_info_get_is_hidden(info) || g_desktop_app_info_get_nodisplay(info)) {
|
||||
return NULL;
|
||||
}
|
||||
const char* name = g_app_info_get_display_name(G_APP_INFO(info));
|
||||
if(name == NULL) {
|
||||
return NULL;
|
||||
}
|
||||
if(wofi_allow_images()) {
|
||||
GIcon* icon = g_app_info_get_icon(G_APP_INFO(info));
|
||||
if(G_IS_THEMED_ICON(icon)) {
|
||||
GtkIconTheme* theme = gtk_icon_theme_get_default();
|
||||
const gchar* const* icon_names = g_themed_icon_get_names(G_THEMED_ICON(icon));
|
||||
GtkIconInfo* info = gtk_icon_theme_choose_icon(theme, (const gchar**) icon_names, wofi_get_image_size(), 0);
|
||||
if(info == NULL) {
|
||||
return utils_concat(2, "text:", name);
|
||||
} else {
|
||||
const gchar* icon_path = gtk_icon_info_get_filename(info);
|
||||
return utils_concat(4, "img:", icon_path, ":text:", name);
|
||||
}
|
||||
} else {
|
||||
return utils_concat(2, "text:", name);
|
||||
}
|
||||
} else {
|
||||
return strdup(name);
|
||||
}
|
||||
}
|
||||
|
||||
void drun_init() {
|
||||
struct map* cached = map_init();
|
||||
struct wl_list* cache = wofi_read_cache("drun");
|
||||
|
||||
struct cache_line* node, *tmp;
|
||||
wl_list_for_each_safe(node, tmp, cache, link) {
|
||||
char* text = get_text(node->line);
|
||||
if(text == NULL) {
|
||||
goto cache_cont;
|
||||
}
|
||||
wofi_insert_widget(text, node->line);
|
||||
map_put(cached, node->line, "true");
|
||||
free(text);
|
||||
|
||||
cache_cont:
|
||||
free(node->line);
|
||||
wl_list_remove(&node->link);
|
||||
free(node);
|
||||
}
|
||||
|
||||
free(cache);
|
||||
|
||||
char* data_home = getenv("XDG_DATA_HOME");
|
||||
if(data_home == NULL) {
|
||||
data_home = utils_concat(2, getenv("HOME"), "/.local/share");
|
||||
} else {
|
||||
data_home = strdup(data_home);
|
||||
}
|
||||
char* data_dirs = getenv("XDG_DATA_DIRS");
|
||||
if(data_dirs == NULL) {
|
||||
data_dirs = "/usr/local/share:/usr/share";
|
||||
}
|
||||
char* dirs = utils_concat(3, data_home, ":", data_dirs);
|
||||
char* original_dirs = dirs;
|
||||
free(data_home);
|
||||
|
||||
size_t colon_count = utils_split(dirs, ':');
|
||||
for(size_t count = 0; count < colon_count; ++count) {
|
||||
char* app_dir = utils_concat(2, dirs, "/applications");
|
||||
DIR* dir = opendir(app_dir);
|
||||
if(dir == NULL) {
|
||||
goto cont;
|
||||
}
|
||||
struct dirent* entry;
|
||||
while((entry = readdir(dir)) != NULL) {
|
||||
if(strcmp(entry->d_name, ".") == 0 || strcmp(entry->d_name, "..") == 0) {
|
||||
continue;
|
||||
}
|
||||
char* full_path = utils_concat(3, app_dir, "/", entry->d_name);
|
||||
if(map_contains(cached, full_path)) {
|
||||
free(full_path);
|
||||
continue;
|
||||
}
|
||||
char* text = get_text(full_path);
|
||||
if(text == NULL) {
|
||||
free(full_path);
|
||||
continue;
|
||||
}
|
||||
wofi_insert_widget(text, full_path);
|
||||
free(text);
|
||||
free(full_path);
|
||||
}
|
||||
closedir(dir);
|
||||
cont:
|
||||
dirs += strlen(dirs) + 1;
|
||||
free(app_dir);
|
||||
}
|
||||
free(original_dirs);
|
||||
map_free(cached);
|
||||
}
|
||||
|
||||
static void launch_done(GObject* obj, GAsyncResult* result, gpointer data) {
|
||||
if(g_app_info_launch_uris_finish(G_APP_INFO(obj), result, NULL)) {
|
||||
exit(0);
|
||||
} else {
|
||||
char* cmd = data;
|
||||
fprintf(stderr, "%s cannot be executed\n", cmd);
|
||||
exit(1);
|
||||
}
|
||||
}
|
||||
|
||||
void drun_exec(const gchar* cmd) {
|
||||
GDesktopAppInfo* info = g_desktop_app_info_new_from_filename(cmd);
|
||||
if(G_IS_DESKTOP_APP_INFO(info)) {
|
||||
g_app_info_launch_uris_async(G_APP_INFO(info), NULL, NULL, NULL, launch_done, (gchar*) cmd);
|
||||
} else {
|
||||
fprintf(stderr, "%s cannot be executed\n", cmd);
|
||||
exit(1);
|
||||
}
|
||||
}
|
69
modes/run.c
Normal file
69
modes/run.c
Normal file
@@ -0,0 +1,69 @@
|
||||
/*
|
||||
* Copyright (C) 2019 Scoopta
|
||||
* This file is part of Wofi
|
||||
* Wofi is free software: you can redistribute it and/or modify
|
||||
it under the terms of the GNU General Public License as published by
|
||||
the Free Software Foundation, either version 3 of the License, or
|
||||
(at your option) any later version.
|
||||
|
||||
Wofi is distributed in the hope that it will be useful,
|
||||
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
GNU General Public License for more details.
|
||||
|
||||
You should have received a copy of the GNU General Public License
|
||||
along with Wofi. If not, see <http://www.gnu.org/licenses/>.
|
||||
*/
|
||||
|
||||
#include <wofi.h>
|
||||
|
||||
void run_init() {
|
||||
struct map* cached = map_init();
|
||||
struct wl_list* cache = wofi_read_cache("run");
|
||||
|
||||
struct cache_line* node, *tmp;
|
||||
wl_list_for_each_safe(node, tmp, cache, link) {
|
||||
char* text = strrchr(node->line, '/') + 1;
|
||||
wofi_insert_widget(text, node->line);
|
||||
map_put(cached, node->line, "true");
|
||||
free(node->line);
|
||||
wl_list_remove(&node->link);
|
||||
free(node);
|
||||
}
|
||||
|
||||
free(cache);
|
||||
|
||||
char* path = strdup(getenv("PATH"));
|
||||
char* original_path = path;
|
||||
size_t colon_count = utils_split(path, ':');
|
||||
for(size_t count = 0; count < colon_count; ++count) {
|
||||
DIR* dir = opendir(path);
|
||||
if(dir == NULL) {
|
||||
goto cont;
|
||||
}
|
||||
struct dirent* entry;
|
||||
while((entry = readdir(dir)) != NULL) {
|
||||
if(strcmp(entry->d_name, ".") == 0 || strcmp(entry->d_name, "..") == 0) {
|
||||
continue;
|
||||
}
|
||||
char* full_path = utils_concat(3, path, "/", entry->d_name);
|
||||
struct stat info;
|
||||
stat(full_path, &info);
|
||||
if(access(full_path, X_OK) == 0 && S_ISREG(info.st_mode) && !map_contains(cached, full_path)) {
|
||||
wofi_insert_widget(entry->d_name, full_path);
|
||||
}
|
||||
free(full_path);
|
||||
}
|
||||
closedir(dir);
|
||||
cont:
|
||||
path += strlen(path) + 1;
|
||||
}
|
||||
free(original_path);
|
||||
map_free(cached);
|
||||
}
|
||||
|
||||
void run_exec(const gchar* cmd) {
|
||||
execl(cmd, cmd, NULL);
|
||||
fprintf(stderr, "%s cannot be executed\n", cmd);
|
||||
exit(errno);
|
||||
}
|
Reference in New Issue
Block a user