2019-08-17 21:19:32 -04:00
|
|
|
/*
|
2020-01-06 19:42:52 -05:00
|
|
|
* Copyright (C) 2019-2020 Scoopta
|
2019-08-17 21:19:32 -04:00
|
|
|
* 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>
|
|
|
|
|
2019-09-27 02:24:48 -04:00
|
|
|
static const char* terminals[] = {"kitty", "termite", "gnome-terminal", "weston-terminal"};
|
|
|
|
|
2019-11-21 04:01:27 -05:00
|
|
|
enum matching_mode {
|
|
|
|
MATCHING_MODE_CONTAINS,
|
|
|
|
MATCHING_MODE_FUZZY
|
|
|
|
};
|
|
|
|
|
2020-01-06 03:28:57 -05:00
|
|
|
enum locations {
|
|
|
|
LOCATION_CENTER,
|
|
|
|
LOCATION_TOP_LEFT,
|
|
|
|
LOCATION_TOP,
|
|
|
|
LOCATION_TOP_RIGHT,
|
|
|
|
LOCATION_RIGHT,
|
|
|
|
LOCATION_BOTTOM_RIGHT,
|
|
|
|
LOCATION_BOTTOM,
|
|
|
|
LOCATION_BOTTOM_LEFT,
|
|
|
|
LOCATION_LEFT
|
|
|
|
};
|
|
|
|
|
2019-08-17 21:19:32 -04:00
|
|
|
static uint64_t width, height;
|
2020-01-06 03:28:57 -05:00
|
|
|
static char* x, *y;
|
2019-08-17 21:19:32 -04:00
|
|
|
static struct zwlr_layer_shell_v1* shell;
|
2019-08-19 20:34:12 -04:00
|
|
|
static GtkWidget* window, *outer_box, *scroll, *entry, *inner_box, *previous_selection = NULL;
|
2019-08-17 21:19:32 -04:00
|
|
|
static const gchar* filter;
|
2019-11-04 23:56:40 -05:00
|
|
|
static char* mode = NULL;
|
2019-08-27 00:19:59 -04:00
|
|
|
static time_t filter_time;
|
|
|
|
static int64_t filter_rate;
|
2019-08-28 03:58:45 -04:00
|
|
|
static bool allow_images, allow_markup;
|
2019-08-28 20:36:20 -04:00
|
|
|
static uint64_t image_size;
|
2019-08-28 23:14:38 -04:00
|
|
|
static char* cache_file = NULL;
|
2019-09-13 03:25:25 -04:00
|
|
|
static char* config_dir;
|
2019-11-19 01:16:06 -05:00
|
|
|
static bool mod_shift;
|
2019-11-19 01:13:14 -05:00
|
|
|
static bool mod_ctrl;
|
2019-09-27 02:24:48 -04:00
|
|
|
static char* terminal;
|
2019-10-13 01:24:35 -04:00
|
|
|
static GtkOrientation outer_orientation;
|
2019-11-04 20:46:39 -05:00
|
|
|
static bool exec_search;
|
2019-11-04 23:56:40 -05:00
|
|
|
static struct map* modes;
|
2019-11-21 04:01:27 -05:00
|
|
|
static enum matching_mode matching;
|
2019-11-25 17:56:46 -05:00
|
|
|
static bool insensitive;
|
2019-12-16 21:36:21 -05:00
|
|
|
static bool parse_search;
|
2020-01-07 21:56:57 -05:00
|
|
|
static GtkAlign content_halign;
|
2019-12-02 01:58:30 -05:00
|
|
|
static struct map* config;
|
2020-01-06 03:28:57 -05:00
|
|
|
static enum locations location;
|
2020-01-10 17:34:49 -05:00
|
|
|
static bool no_actions;
|
2019-08-28 00:36:44 -04:00
|
|
|
|
2019-11-04 23:56:40 -05:00
|
|
|
struct mode {
|
|
|
|
void (*mode_exec)(const gchar* cmd);
|
2020-01-23 23:26:33 -05:00
|
|
|
struct widget* (*mode_get_widget)(void);
|
2020-01-20 23:09:40 -05:00
|
|
|
char* name;
|
2020-01-16 19:35:51 -05:00
|
|
|
};
|
|
|
|
|
|
|
|
struct widget {
|
|
|
|
size_t action_count;
|
|
|
|
char* mode, **text, *search_text, **actions;
|
2019-08-18 23:49:12 -04:00
|
|
|
};
|
|
|
|
|
2019-08-17 21:19:32 -04:00
|
|
|
static void nop() {}
|
|
|
|
|
|
|
|
static void add_interface(void* data, struct wl_registry* registry, uint32_t name, const char* interface, uint32_t version) {
|
|
|
|
(void) data;
|
|
|
|
if(strcmp(interface, zwlr_layer_shell_v1_interface.name) == 0) {
|
|
|
|
shell = wl_registry_bind(registry, name, &zwlr_layer_shell_v1_interface, version);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void config_surface(void* data, struct zwlr_layer_surface_v1* surface, uint32_t serial, uint32_t _width, uint32_t _height) {
|
|
|
|
(void) data;
|
|
|
|
(void) _width;
|
|
|
|
(void) _height;
|
|
|
|
zwlr_layer_surface_v1_ack_configure(surface, serial);
|
|
|
|
zwlr_layer_surface_v1_set_size(surface, width, height);
|
|
|
|
zwlr_layer_surface_v1_set_keyboard_interactivity(surface, true);
|
2020-01-06 03:28:57 -05:00
|
|
|
|
|
|
|
if(location > 8) {
|
|
|
|
location -= 9;
|
|
|
|
}
|
|
|
|
|
|
|
|
if(x != NULL || y != NULL) {
|
|
|
|
int64_t ix = x == NULL ? 0 : strtol(x, NULL, 10);
|
|
|
|
int64_t iy = y == NULL ? 0 : strtol(y, NULL, 10);
|
|
|
|
|
|
|
|
if(location == LOCATION_CENTER) {
|
|
|
|
location = LOCATION_TOP_LEFT;
|
|
|
|
}
|
|
|
|
|
|
|
|
zwlr_layer_surface_v1_set_margin(surface, iy, -ix, -iy, ix);
|
|
|
|
}
|
|
|
|
|
|
|
|
if(location > 0) {
|
2020-01-06 03:40:56 -05:00
|
|
|
enum zwlr_layer_surface_v1_anchor anchor = ZWLR_LAYER_SURFACE_V1_ANCHOR_TOP | ZWLR_LAYER_SURFACE_V1_ANCHOR_LEFT;
|
2020-01-06 03:28:57 -05:00
|
|
|
|
|
|
|
switch(location) {
|
|
|
|
case LOCATION_CENTER:
|
|
|
|
break;
|
|
|
|
case LOCATION_TOP_LEFT:
|
|
|
|
anchor = ZWLR_LAYER_SURFACE_V1_ANCHOR_TOP | ZWLR_LAYER_SURFACE_V1_ANCHOR_LEFT;
|
|
|
|
break;
|
|
|
|
case LOCATION_TOP:
|
|
|
|
anchor = ZWLR_LAYER_SURFACE_V1_ANCHOR_TOP;
|
|
|
|
break;
|
|
|
|
case LOCATION_TOP_RIGHT:
|
|
|
|
anchor = ZWLR_LAYER_SURFACE_V1_ANCHOR_TOP | ZWLR_LAYER_SURFACE_V1_ANCHOR_RIGHT;
|
|
|
|
break;
|
|
|
|
case LOCATION_LEFT:
|
|
|
|
anchor = ZWLR_LAYER_SURFACE_V1_ANCHOR_LEFT;
|
|
|
|
break;
|
|
|
|
case LOCATION_RIGHT:
|
|
|
|
anchor = ZWLR_LAYER_SURFACE_V1_ANCHOR_RIGHT;
|
|
|
|
break;
|
|
|
|
case LOCATION_BOTTOM_LEFT:
|
|
|
|
anchor = ZWLR_LAYER_SURFACE_V1_ANCHOR_BOTTOM | ZWLR_LAYER_SURFACE_V1_ANCHOR_LEFT;
|
|
|
|
break;
|
|
|
|
case LOCATION_BOTTOM:
|
|
|
|
anchor = ZWLR_LAYER_SURFACE_V1_ANCHOR_BOTTOM;
|
|
|
|
break;
|
|
|
|
case LOCATION_BOTTOM_RIGHT:
|
|
|
|
anchor = ZWLR_LAYER_SURFACE_V1_ANCHOR_BOTTOM | ZWLR_LAYER_SURFACE_V1_ANCHOR_RIGHT;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
zwlr_layer_surface_v1_set_anchor(surface, anchor);
|
2019-08-17 21:19:32 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void get_input(GtkSearchEntry* entry, gpointer data) {
|
2019-08-27 00:19:59 -04:00
|
|
|
(void) data;
|
|
|
|
if(utils_get_time_millis() - filter_time > filter_rate) {
|
|
|
|
filter = gtk_entry_get_text(GTK_ENTRY(entry));
|
|
|
|
filter_time = utils_get_time_millis();
|
2019-08-28 22:33:08 -04:00
|
|
|
gtk_flow_box_invalidate_filter(GTK_FLOW_BOX(inner_box));
|
2019-11-21 04:01:27 -05:00
|
|
|
gtk_flow_box_invalidate_sort(GTK_FLOW_BOX(inner_box));
|
2019-08-27 00:19:59 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void get_search(GtkSearchEntry* entry, gpointer data) {
|
2019-08-19 20:34:12 -04:00
|
|
|
(void) data;
|
2019-08-17 21:19:32 -04:00
|
|
|
filter = gtk_entry_get_text(GTK_ENTRY(entry));
|
2019-08-28 22:33:08 -04:00
|
|
|
gtk_flow_box_invalidate_filter(GTK_FLOW_BOX(inner_box));
|
2019-11-21 04:01:27 -05:00
|
|
|
gtk_flow_box_invalidate_sort(GTK_FLOW_BOX(inner_box));
|
2019-08-17 21:19:32 -04:00
|
|
|
}
|
|
|
|
|
2019-12-16 22:15:16 -05:00
|
|
|
static char* parse_images(WofiPropertyBox* box, const char* text, bool create_widgets) {
|
2019-12-16 21:36:21 -05:00
|
|
|
char* ret = strdup("");
|
|
|
|
struct map* mode_map = map_init();
|
|
|
|
map_put(mode_map, "img", "true");
|
|
|
|
map_put(mode_map, "text", "true");
|
2019-11-21 01:05:28 -05:00
|
|
|
|
2019-12-16 21:36:21 -05:00
|
|
|
char* tmp = strdup(text);
|
2019-08-28 02:15:18 -04:00
|
|
|
|
2019-12-16 21:36:21 -05:00
|
|
|
struct wl_list modes;
|
|
|
|
struct node {
|
|
|
|
char* str;
|
|
|
|
struct wl_list link;
|
|
|
|
};
|
2019-12-16 20:30:29 -05:00
|
|
|
|
2019-12-16 21:36:21 -05:00
|
|
|
wl_list_init(&modes);
|
2019-12-16 20:30:29 -05:00
|
|
|
|
2019-12-16 21:36:21 -05:00
|
|
|
bool data = false;
|
2019-12-16 20:30:29 -05:00
|
|
|
|
2019-12-16 21:36:21 -05:00
|
|
|
char* save_ptr;
|
|
|
|
char* str = strtok_r(tmp, ":", &save_ptr);
|
|
|
|
do {
|
2019-12-17 03:15:28 -05:00
|
|
|
if(str == NULL) {
|
|
|
|
break;
|
|
|
|
}
|
2019-12-16 21:36:21 -05:00
|
|
|
if(map_contains(mode_map, str) || data) {
|
|
|
|
struct node* node = malloc(sizeof(struct node));
|
|
|
|
node->str = str;
|
|
|
|
wl_list_insert(&modes, &node->link);
|
|
|
|
data = !data;
|
|
|
|
}
|
|
|
|
} while((str = strtok_r(NULL, ":", &save_ptr)) != NULL);
|
2019-12-16 20:30:29 -05:00
|
|
|
|
2019-12-16 21:36:21 -05:00
|
|
|
char* tmp2 = strdup(text);
|
|
|
|
char* start = tmp2;
|
2019-12-16 20:30:29 -05:00
|
|
|
|
2019-12-16 21:36:21 -05:00
|
|
|
char* mode = NULL;
|
2019-12-16 20:30:29 -05:00
|
|
|
|
2019-12-16 21:36:21 -05:00
|
|
|
struct node* node = wl_container_of(modes.prev, node, link);
|
|
|
|
while(true) {
|
|
|
|
if(mode == NULL) {
|
|
|
|
if(start == NULL) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
char* tmp_start = (start - tmp2) + tmp;
|
|
|
|
if(!wl_list_empty(&modes) && tmp_start == node->str) {
|
|
|
|
if(node->link.prev == &modes) {
|
2019-12-16 20:30:29 -05:00
|
|
|
break;
|
|
|
|
}
|
2019-12-16 21:36:21 -05:00
|
|
|
mode = node->str;
|
|
|
|
node = wl_container_of(node->link.prev, node, link);
|
|
|
|
str = node->str;
|
|
|
|
start = ((str + strlen(str) + 1) - tmp) + tmp2;
|
|
|
|
if(((start - tmp2) + text) > (text + strlen(text))) {
|
|
|
|
start = NULL;
|
|
|
|
}
|
|
|
|
if(node->link.prev != &modes) {
|
2019-12-16 20:30:29 -05:00
|
|
|
node = wl_container_of(node->link.prev, node, link);
|
2019-11-21 01:05:28 -05:00
|
|
|
}
|
2019-08-28 02:15:18 -04:00
|
|
|
} else {
|
2019-12-16 21:36:21 -05:00
|
|
|
mode = "text";
|
|
|
|
str = start;
|
|
|
|
if(!wl_list_empty(&modes)) {
|
|
|
|
start = (node->str - tmp - 1) + tmp2;
|
|
|
|
*start = 0;
|
|
|
|
++start;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if(strcmp(mode, "img") == 0 && create_widgets) {
|
|
|
|
GdkPixbuf* buf = gdk_pixbuf_new_from_file(str, NULL);
|
|
|
|
int width = gdk_pixbuf_get_width(buf);
|
|
|
|
int height = gdk_pixbuf_get_height(buf);
|
|
|
|
if(height > width) {
|
|
|
|
float percent = (float) image_size / height;
|
|
|
|
GdkPixbuf* tmp = gdk_pixbuf_scale_simple(buf, width * percent, image_size, GDK_INTERP_BILINEAR);
|
|
|
|
g_object_unref(buf);
|
|
|
|
buf = tmp;
|
|
|
|
} else {
|
|
|
|
float percent = (float) image_size / width;
|
|
|
|
GdkPixbuf* tmp = gdk_pixbuf_scale_simple(buf, image_size, height * percent, GDK_INTERP_BILINEAR);
|
|
|
|
g_object_unref(buf);
|
|
|
|
buf = tmp;
|
|
|
|
}
|
|
|
|
GtkWidget* img = gtk_image_new_from_pixbuf(buf);
|
|
|
|
gtk_widget_set_name(img, "img");
|
|
|
|
gtk_container_add(GTK_CONTAINER(box), img);
|
|
|
|
} else if(strcmp(mode, "text") == 0) {
|
|
|
|
if(create_widgets) {
|
2019-11-05 01:41:53 -05:00
|
|
|
GtkWidget* label = gtk_label_new(str);
|
2019-09-01 23:07:15 -04:00
|
|
|
gtk_widget_set_name(label, "text");
|
2019-08-28 03:58:45 -04:00
|
|
|
gtk_label_set_use_markup(GTK_LABEL(label), allow_markup);
|
2019-08-28 02:15:18 -04:00
|
|
|
gtk_label_set_xalign(GTK_LABEL(label), 0);
|
|
|
|
gtk_container_add(GTK_CONTAINER(box), label);
|
2019-12-16 21:36:21 -05:00
|
|
|
} else {
|
|
|
|
char* tmp = ret;
|
|
|
|
ret = utils_concat(2, ret, str);
|
|
|
|
free(tmp);
|
2019-12-16 20:30:29 -05:00
|
|
|
}
|
2019-08-28 02:15:18 -04:00
|
|
|
}
|
2019-12-16 21:36:21 -05:00
|
|
|
mode = NULL;
|
|
|
|
if(wl_list_empty(&modes)) {
|
|
|
|
break;
|
|
|
|
}
|
2019-12-16 20:30:29 -05:00
|
|
|
}
|
2019-12-16 21:36:21 -05:00
|
|
|
}
|
|
|
|
free(tmp);
|
|
|
|
free(tmp2);
|
|
|
|
map_free(mode_map);
|
|
|
|
|
|
|
|
struct node* tmp_node;
|
|
|
|
wl_list_for_each_safe(node, tmp_node, &modes, link) {
|
|
|
|
wl_list_remove(&node->link);
|
|
|
|
free(node);
|
|
|
|
}
|
|
|
|
if(create_widgets) {
|
|
|
|
free(ret);
|
|
|
|
return NULL;
|
|
|
|
} else {
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-12-16 22:15:16 -05:00
|
|
|
char* wofi_parse_image_escapes(const char* text) {
|
|
|
|
return parse_images(NULL, text, false);
|
|
|
|
}
|
|
|
|
|
2019-12-16 21:36:21 -05:00
|
|
|
static GtkWidget* create_label(char* mode, char* text, char* search_text, char* action) {
|
|
|
|
GtkWidget* box = wofi_property_box_new(GTK_ORIENTATION_HORIZONTAL, 0);
|
2019-12-23 04:27:49 -05:00
|
|
|
gtk_widget_set_halign(box, content_halign);
|
2019-12-16 21:36:21 -05:00
|
|
|
gtk_widget_set_name(box, "unselected");
|
|
|
|
GtkStyleContext* style = gtk_widget_get_style_context(box);
|
|
|
|
gtk_style_context_add_class(style, "entry");
|
|
|
|
wofi_property_box_add_property(WOFI_PROPERTY_BOX(box), "mode", mode);
|
|
|
|
wofi_property_box_add_property(WOFI_PROPERTY_BOX(box), "action", action);
|
|
|
|
if(allow_images) {
|
|
|
|
parse_images(WOFI_PROPERTY_BOX(box), text, true);
|
2019-08-28 02:15:18 -04:00
|
|
|
} else {
|
|
|
|
GtkWidget* label = gtk_label_new(text);
|
2019-09-15 19:29:15 -04:00
|
|
|
gtk_widget_set_name(label, "text");
|
2019-08-28 03:58:45 -04:00
|
|
|
gtk_label_set_use_markup(GTK_LABEL(label), allow_markup);
|
2019-08-28 02:15:18 -04:00
|
|
|
gtk_label_set_xalign(GTK_LABEL(label), 0);
|
|
|
|
gtk_container_add(GTK_CONTAINER(box), label);
|
|
|
|
}
|
2019-12-16 21:36:21 -05:00
|
|
|
if(parse_search) {
|
2019-12-16 22:38:12 -05:00
|
|
|
search_text = strdup(search_text);
|
|
|
|
if(allow_images) {
|
|
|
|
char* tmp = search_text;
|
|
|
|
search_text = parse_images(WOFI_PROPERTY_BOX(box), search_text, false);
|
|
|
|
free(tmp);
|
|
|
|
}
|
|
|
|
if(allow_markup) {
|
|
|
|
char* out;
|
|
|
|
pango_parse_markup(search_text, -1, 0, NULL, &out, NULL, NULL);
|
|
|
|
free(search_text);
|
|
|
|
search_text = out;
|
|
|
|
}
|
2019-12-16 21:36:21 -05:00
|
|
|
}
|
2019-10-28 17:17:06 -04:00
|
|
|
wofi_property_box_add_property(WOFI_PROPERTY_BOX(box), "filter", search_text);
|
2019-12-16 21:36:21 -05:00
|
|
|
if(parse_search) {
|
|
|
|
free(search_text);
|
|
|
|
}
|
2019-08-28 02:15:18 -04:00
|
|
|
return box;
|
2019-08-18 23:49:12 -04:00
|
|
|
}
|
|
|
|
|
2019-11-05 01:41:53 -05:00
|
|
|
static char* get_cache_path(const gchar* mode) {
|
2019-08-28 23:14:38 -04:00
|
|
|
if(cache_file != NULL) {
|
2019-09-12 14:06:57 -04:00
|
|
|
return strdup(cache_file);
|
2019-08-28 23:14:38 -04:00
|
|
|
}
|
2019-08-18 23:49:12 -04:00
|
|
|
char* cache_path = getenv("XDG_CACHE_HOME");
|
|
|
|
if(cache_path == NULL) {
|
|
|
|
cache_path = utils_concat(3, getenv("HOME"), "/.cache/wofi-", mode);
|
|
|
|
} else {
|
|
|
|
cache_path = utils_concat(3, cache_path, "/wofi-", mode);
|
|
|
|
}
|
|
|
|
return cache_path;
|
|
|
|
}
|
|
|
|
|
2019-11-16 17:52:20 -05:00
|
|
|
static void execute_action(const gchar* mode, const gchar* cmd) {
|
2019-11-13 04:17:45 -05:00
|
|
|
struct mode* mode_ptr = map_get(modes, mode);
|
|
|
|
mode_ptr->mode_exec(cmd);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void activate_item(GtkFlowBox* flow_box, GtkFlowBoxChild* row, gpointer data) {
|
|
|
|
(void) flow_box;
|
|
|
|
(void) data;
|
|
|
|
GtkWidget* box = gtk_bin_get_child(GTK_BIN(row));
|
|
|
|
bool primary_action = GTK_IS_EXPANDER(box);
|
|
|
|
if(primary_action) {
|
|
|
|
box = gtk_expander_get_label_widget(GTK_EXPANDER(box));
|
|
|
|
}
|
2019-11-16 17:52:20 -05:00
|
|
|
execute_action(wofi_property_box_get_property(WOFI_PROPERTY_BOX(box), "mode"), wofi_property_box_get_property(WOFI_PROPERTY_BOX(box), "action"));
|
2019-11-13 04:17:45 -05:00
|
|
|
}
|
|
|
|
|
2019-11-19 22:14:30 -05:00
|
|
|
static void expand(GtkExpander* expander, gpointer data) {
|
|
|
|
(void) data;
|
|
|
|
GtkWidget* box = gtk_bin_get_child(GTK_BIN(expander));
|
|
|
|
gtk_widget_set_visible(box, !gtk_expander_get_expanded(expander));
|
|
|
|
}
|
|
|
|
|
2019-11-13 04:17:45 -05:00
|
|
|
static gboolean _insert_widget(gpointer data) {
|
2020-01-16 19:35:51 -05:00
|
|
|
struct mode* mode = data;
|
|
|
|
struct widget* node;
|
|
|
|
if(mode->mode_get_widget == NULL) {
|
|
|
|
return FALSE;
|
|
|
|
} else {
|
|
|
|
node = mode->mode_get_widget();
|
|
|
|
}
|
|
|
|
if(node == NULL) {
|
|
|
|
return FALSE;
|
|
|
|
}
|
2019-11-13 04:17:45 -05:00
|
|
|
GtkWidget* parent;
|
2020-01-10 17:34:49 -05:00
|
|
|
if(node->action_count > 1 && !no_actions) {
|
2019-11-13 04:17:45 -05:00
|
|
|
parent = gtk_expander_new("");
|
2019-11-19 22:14:30 -05:00
|
|
|
g_signal_connect(parent, "activate", G_CALLBACK(expand), NULL);
|
2019-11-13 04:17:45 -05:00
|
|
|
GtkWidget* box = create_label(node->mode, node->text[0], node->search_text, node->actions[0]);
|
|
|
|
gtk_expander_set_label_widget(GTK_EXPANDER(parent), box);
|
2019-11-19 22:14:30 -05:00
|
|
|
|
2019-11-13 04:17:45 -05:00
|
|
|
GtkWidget* exp_box = gtk_list_box_new();
|
2019-11-13 04:42:27 -05:00
|
|
|
gtk_list_box_set_activate_on_single_click(GTK_LIST_BOX(exp_box), FALSE);
|
2019-11-13 04:17:45 -05:00
|
|
|
g_signal_connect(exp_box, "row-activated", G_CALLBACK(activate_item), NULL);
|
|
|
|
gtk_container_add(GTK_CONTAINER(parent), exp_box);
|
|
|
|
for(size_t count = 1; count < node->action_count; ++count) {
|
|
|
|
box = create_label(node->mode, node->text[count], node->search_text, node->actions[count]);
|
2020-01-18 17:54:52 -05:00
|
|
|
|
|
|
|
GtkWidget* row = gtk_list_box_row_new();
|
|
|
|
gtk_widget_set_name(row, "entry");
|
|
|
|
|
|
|
|
gtk_container_add(GTK_CONTAINER(row), box);
|
|
|
|
gtk_container_add(GTK_CONTAINER(exp_box), row);
|
2019-11-13 04:17:45 -05:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
parent = create_label(node->mode, node->text[0], node->search_text, node->actions[0]);
|
|
|
|
}
|
2020-01-18 17:54:52 -05:00
|
|
|
GtkWidget* child = gtk_flow_box_child_new();
|
|
|
|
gtk_widget_set_name(child, "entry");
|
|
|
|
|
|
|
|
gtk_container_add(GTK_CONTAINER(child), parent);
|
|
|
|
gtk_container_add(GTK_CONTAINER(inner_box), child);
|
|
|
|
gtk_widget_show_all(child);
|
2019-11-13 04:17:45 -05:00
|
|
|
|
2019-11-19 22:14:30 -05:00
|
|
|
if(GTK_IS_EXPANDER(parent)) {
|
|
|
|
GtkWidget* box = gtk_bin_get_child(GTK_BIN(parent));
|
|
|
|
gtk_widget_set_visible(box, FALSE);
|
|
|
|
}
|
|
|
|
|
2019-11-13 04:17:45 -05:00
|
|
|
free(node->mode);
|
|
|
|
for(size_t count = 0; count < node->action_count; ++count) {
|
|
|
|
free(node->text[count]);
|
|
|
|
}
|
|
|
|
free(node->text);
|
|
|
|
free(node->search_text);
|
|
|
|
for(size_t count = 0; count < node->action_count; ++count) {
|
|
|
|
free(node->actions[count]);
|
|
|
|
}
|
|
|
|
free(node->actions);
|
|
|
|
free(node);
|
2020-01-16 19:35:51 -05:00
|
|
|
return TRUE;
|
2019-11-13 04:17:45 -05:00
|
|
|
}
|
|
|
|
|
2020-01-30 00:49:16 -05:00
|
|
|
static char* escape_lf(const char* cmd) {
|
|
|
|
size_t len = strlen(cmd);
|
|
|
|
char* buffer = calloc(1, (len + 1) * 2);
|
|
|
|
|
|
|
|
size_t buf_count = 0;
|
|
|
|
for(size_t count = 0; count < len; ++count) {
|
|
|
|
char chr = cmd[count];
|
|
|
|
if(chr == '\n') {
|
|
|
|
buffer[buf_count++] = '\\';
|
|
|
|
buffer[buf_count++] = 'n';
|
|
|
|
} else if(chr == '\\') {
|
|
|
|
buffer[buf_count++] = '\\';
|
|
|
|
buffer[buf_count++] = '\\';
|
|
|
|
} else {
|
|
|
|
buffer[buf_count++] = chr;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return buffer;
|
|
|
|
}
|
|
|
|
|
|
|
|
static char* remove_escapes(const char* cmd) {
|
|
|
|
size_t len = strlen(cmd);
|
|
|
|
char* buffer = calloc(1, len + 1);
|
|
|
|
|
|
|
|
size_t buf_count = 0;
|
|
|
|
for(size_t count = 0; count < len; ++count) {
|
|
|
|
char chr = cmd[count];
|
|
|
|
if(chr == '\\') {
|
|
|
|
chr = cmd[++count];
|
|
|
|
if(chr == 'n') {
|
|
|
|
buffer[buf_count++] = '\n';
|
|
|
|
} else if(chr == '\\') {
|
|
|
|
buffer[buf_count++] = '\\';
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
buffer[buf_count++] = chr;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return buffer;
|
|
|
|
}
|
|
|
|
|
|
|
|
void wofi_write_cache(struct mode* mode, const char* _cmd) {
|
|
|
|
char* cmd = escape_lf(_cmd);
|
|
|
|
|
2020-01-20 23:09:40 -05:00
|
|
|
char* cache_path = get_cache_path(mode->name);
|
2019-11-16 17:52:20 -05:00
|
|
|
struct wl_list lines;
|
|
|
|
wl_list_init(&lines);
|
|
|
|
bool inc_count = false;
|
2020-01-16 19:49:57 -05:00
|
|
|
if(access(cache_path, R_OK) == 0) {
|
2019-11-16 17:52:20 -05:00
|
|
|
FILE* file = fopen(cache_path, "r");
|
|
|
|
char* line = NULL;
|
|
|
|
size_t size = 0;
|
|
|
|
while(getline(&line, &size, file) != -1) {
|
2019-12-17 02:38:25 -05:00
|
|
|
char* space = strchr(line, ' ');
|
2020-01-02 17:36:54 -05:00
|
|
|
char* lf = strchr(line, '\n');
|
|
|
|
if(lf != NULL) {
|
|
|
|
*lf = 0;
|
2019-12-17 02:38:25 -05:00
|
|
|
}
|
|
|
|
if(space != NULL && strcmp(cmd, space + 1) == 0) {
|
|
|
|
struct cache_line* node = malloc(sizeof(struct cache_line));
|
2020-01-30 01:01:29 -05:00
|
|
|
uint64_t count = strtol(line, NULL, 10) + 1;
|
2019-11-16 17:52:20 -05:00
|
|
|
char num[6];
|
|
|
|
snprintf(num, 5, "%" PRIu64, count);
|
|
|
|
node->line = utils_concat(4, num, " ", cmd, "\n");
|
|
|
|
inc_count = true;
|
2019-12-17 02:38:25 -05:00
|
|
|
wl_list_insert(&lines, &node->link);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
free(line);
|
|
|
|
line = NULL;
|
|
|
|
size = 0;
|
|
|
|
fseek(file, 0, SEEK_SET);
|
|
|
|
while(getline(&line, &size, file) != -1) {
|
|
|
|
char* space = strchr(line, ' ');
|
|
|
|
char* nl = strchr(line, '\n');
|
|
|
|
if(nl != NULL) {
|
|
|
|
*nl = 0;
|
|
|
|
}
|
|
|
|
if(space == NULL || strcmp(cmd, space + 1) != 0) {
|
|
|
|
struct cache_line* node = malloc(sizeof(struct cache_line));
|
2020-01-30 01:01:29 -05:00
|
|
|
node->line = utils_concat(2, line, "\n");
|
2019-12-17 02:38:25 -05:00
|
|
|
wl_list_insert(&lines, &node->link);
|
2019-11-16 17:52:20 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
free(line);
|
|
|
|
fclose(file);
|
2020-01-16 19:49:57 -05:00
|
|
|
}
|
|
|
|
char* tmp_path = strdup(cache_path);
|
|
|
|
char* dir = dirname(tmp_path);
|
2019-11-16 17:52:20 -05:00
|
|
|
|
2020-01-16 19:49:57 -05:00
|
|
|
if(access(dir, W_OK) == 0) {
|
2020-01-16 19:56:08 -05:00
|
|
|
if(!inc_count) {
|
|
|
|
struct cache_line* node = malloc(sizeof(struct cache_line));
|
|
|
|
node->line = utils_concat(3, "1 ", cmd, "\n");
|
|
|
|
wl_list_insert(&lines, &node->link);
|
|
|
|
}
|
2020-01-16 19:49:57 -05:00
|
|
|
FILE* file = fopen(cache_path, "w");
|
2020-01-02 18:24:30 -05:00
|
|
|
struct cache_line* node, *tmp;
|
|
|
|
wl_list_for_each_safe(node, tmp, &lines, link) {
|
|
|
|
fwrite(node->line, 1, strlen(node->line), file);
|
|
|
|
free(node->line);
|
|
|
|
wl_list_remove(&node->link);
|
|
|
|
free(node);
|
|
|
|
}
|
2019-11-16 17:52:20 -05:00
|
|
|
|
2020-01-02 18:24:30 -05:00
|
|
|
fclose(file);
|
|
|
|
}
|
2019-11-16 17:52:20 -05:00
|
|
|
free(cache_path);
|
2020-01-16 19:49:57 -05:00
|
|
|
free(tmp_path);
|
2020-01-30 00:49:16 -05:00
|
|
|
free(cmd);
|
2019-11-16 17:52:20 -05:00
|
|
|
}
|
|
|
|
|
2020-01-30 00:49:16 -05:00
|
|
|
void wofi_remove_cache(struct mode* mode, const char* _cmd) {
|
|
|
|
char* cmd = escape_lf(_cmd);
|
|
|
|
|
2020-01-20 23:09:40 -05:00
|
|
|
char* cache_path = get_cache_path(mode->name);
|
2020-01-02 18:24:30 -05:00
|
|
|
if(access(cache_path, R_OK | W_OK) == 0) {
|
2020-01-02 17:36:54 -05:00
|
|
|
struct wl_list lines;
|
|
|
|
wl_list_init(&lines);
|
|
|
|
|
|
|
|
FILE* file = fopen(cache_path, "r");
|
|
|
|
char* line = NULL;
|
|
|
|
size_t size = 0;
|
|
|
|
while(getline(&line, &size, file) != -1) {
|
|
|
|
char* space = strchr(line, ' ');
|
|
|
|
char* lf = strchr(line, '\n');
|
|
|
|
if(lf != NULL) {
|
|
|
|
*lf = 0;
|
|
|
|
}
|
|
|
|
if(space == NULL || strcmp(cmd, space + 1) != 0) {
|
|
|
|
struct cache_line* node = malloc(sizeof(struct cache_line));
|
|
|
|
node->line = utils_concat(2, line, "\n");
|
|
|
|
wl_list_insert(&lines, &node->link);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
free(line);
|
|
|
|
fclose(file);
|
|
|
|
|
|
|
|
file = fopen(cache_path, "w");
|
|
|
|
struct cache_line* node, *tmp;
|
|
|
|
wl_list_for_each_safe(node, tmp, &lines, link) {
|
|
|
|
fwrite(node->line, 1, strlen(node->line), file);
|
|
|
|
free(node->line);
|
|
|
|
wl_list_remove(&node->link);
|
|
|
|
free(node);
|
|
|
|
}
|
|
|
|
fclose(file);
|
|
|
|
}
|
|
|
|
free(cache_path);
|
2020-01-30 00:51:25 -05:00
|
|
|
free(cmd);
|
2020-01-02 17:36:54 -05:00
|
|
|
}
|
|
|
|
|
2020-01-20 23:09:40 -05:00
|
|
|
struct wl_list* wofi_read_cache(struct mode* mode) {
|
|
|
|
char* cache_path = get_cache_path(mode->name);
|
2019-08-18 23:49:12 -04:00
|
|
|
struct wl_list* cache = malloc(sizeof(struct wl_list));
|
|
|
|
wl_list_init(cache);
|
|
|
|
struct wl_list lines;
|
|
|
|
wl_list_init(&lines);
|
|
|
|
if(access(cache_path, R_OK) == 0) {
|
|
|
|
FILE* file = fopen(cache_path, "r");
|
|
|
|
char* line = NULL;
|
|
|
|
size_t size = 0;
|
|
|
|
while(getline(&line, &size, file) != -1) {
|
|
|
|
struct cache_line* node = malloc(sizeof(struct cache_line));
|
|
|
|
char* lf = strchr(line, '\n');
|
|
|
|
if(lf != NULL) {
|
|
|
|
*lf = 0;
|
|
|
|
}
|
2020-01-30 00:49:16 -05:00
|
|
|
node->line = remove_escapes(line);
|
2019-08-18 23:49:12 -04:00
|
|
|
wl_list_insert(&lines, &node->link);
|
|
|
|
}
|
|
|
|
free(line);
|
|
|
|
fclose(file);
|
|
|
|
}
|
|
|
|
while(wl_list_length(&lines) > 0) {
|
2019-09-13 03:25:25 -04:00
|
|
|
uint64_t smallest = UINT64_MAX;
|
|
|
|
struct cache_line* node, *smallest_node = NULL;
|
2019-08-18 23:49:12 -04:00
|
|
|
wl_list_for_each(node, &lines, link) {
|
|
|
|
uint64_t num = strtol(node->line, NULL, 10);
|
2019-09-13 03:25:25 -04:00
|
|
|
if(num < smallest) {
|
|
|
|
smallest = num;
|
|
|
|
smallest_node = node;
|
2019-08-18 23:49:12 -04:00
|
|
|
}
|
|
|
|
}
|
2019-09-13 03:25:25 -04:00
|
|
|
char* tmp = strdup(strchr(smallest_node->line, ' ') + 1);
|
|
|
|
free(smallest_node->line);
|
|
|
|
smallest_node->line = tmp;
|
|
|
|
wl_list_remove(&smallest_node->link);
|
|
|
|
wl_list_insert(cache, &smallest_node->link);
|
2019-08-18 23:49:12 -04:00
|
|
|
}
|
|
|
|
free(cache_path);
|
|
|
|
return cache;
|
|
|
|
}
|
|
|
|
|
2020-01-20 23:09:40 -05:00
|
|
|
struct widget* wofi_create_widget(struct mode* mode, char** text, char* search_text, char** actions, size_t action_count) {
|
2020-01-16 19:35:51 -05:00
|
|
|
struct widget* widget = malloc(sizeof(struct widget));
|
2020-01-20 23:09:40 -05:00
|
|
|
widget->mode = strdup(mode->name);
|
2019-11-13 04:17:45 -05:00
|
|
|
widget->text = malloc(action_count * sizeof(char*));
|
|
|
|
for(size_t count = 0; count < action_count; ++count) {
|
|
|
|
widget->text[count] = strdup(text[count]);
|
|
|
|
}
|
2019-10-28 17:17:06 -04:00
|
|
|
widget->search_text = strdup(search_text);
|
2019-11-13 04:17:45 -05:00
|
|
|
widget->action_count = action_count;
|
|
|
|
widget->actions = malloc(action_count * sizeof(char*));
|
|
|
|
for(size_t count = 0; count < action_count; ++count) {
|
|
|
|
widget->actions[count] = strdup(actions[count]);
|
|
|
|
}
|
2020-01-16 19:35:51 -05:00
|
|
|
return widget;
|
2019-08-28 23:43:47 -04:00
|
|
|
}
|
|
|
|
|
2020-01-06 03:39:47 -05:00
|
|
|
bool wofi_allow_images(void) {
|
2019-09-13 03:25:25 -04:00
|
|
|
return allow_images;
|
2019-08-17 23:21:52 -04:00
|
|
|
}
|
|
|
|
|
2020-01-06 03:39:47 -05:00
|
|
|
bool wofi_allow_markup(void) {
|
2019-12-16 22:38:12 -05:00
|
|
|
return allow_markup;
|
|
|
|
}
|
|
|
|
|
2020-01-06 03:39:47 -05:00
|
|
|
uint64_t wofi_get_image_size(void) {
|
2019-09-13 03:25:25 -04:00
|
|
|
return image_size;
|
2019-09-12 00:00:26 -04:00
|
|
|
}
|
|
|
|
|
2020-01-06 03:39:47 -05:00
|
|
|
bool wofi_mod_shift(void) {
|
2019-11-19 01:16:06 -05:00
|
|
|
return mod_shift;
|
2019-09-27 02:24:48 -04:00
|
|
|
}
|
|
|
|
|
2020-01-06 03:39:47 -05:00
|
|
|
bool wofi_mod_control(void) {
|
2019-11-19 01:13:14 -05:00
|
|
|
return mod_ctrl;
|
|
|
|
}
|
|
|
|
|
2019-09-27 02:24:48 -04:00
|
|
|
void wofi_term_run(const char* cmd) {
|
|
|
|
if(terminal != NULL) {
|
|
|
|
execlp(terminal, terminal, "--", cmd, NULL);
|
|
|
|
}
|
|
|
|
size_t term_count = sizeof(terminals) / sizeof(char*);
|
|
|
|
for(size_t count = 0; count < term_count; ++count) {
|
|
|
|
execlp(terminals[count], terminals[count], "--", cmd, NULL);
|
|
|
|
}
|
|
|
|
fprintf(stderr, "No terminal emulator found please set term in config or use --term\n");
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
|
2019-08-28 22:33:08 -04:00
|
|
|
static void select_item(GtkFlowBox* flow_box, gpointer data) {
|
2019-08-17 21:34:56 -04:00
|
|
|
(void) data;
|
|
|
|
if(previous_selection != NULL) {
|
|
|
|
gtk_widget_set_name(previous_selection, "unselected");
|
|
|
|
}
|
2019-08-28 22:33:08 -04:00
|
|
|
GList* selected_children = gtk_flow_box_get_selected_children(flow_box);
|
|
|
|
GtkWidget* box = gtk_bin_get_child(GTK_BIN(selected_children->data));
|
|
|
|
g_list_free(selected_children);
|
2019-08-28 02:15:18 -04:00
|
|
|
gtk_widget_set_name(box, "selected");
|
|
|
|
previous_selection = box;
|
2019-08-17 21:34:56 -04:00
|
|
|
}
|
|
|
|
|
2019-10-13 15:42:05 -04:00
|
|
|
static GtkWidget* get_first_child(GtkContainer* container) {
|
|
|
|
GList* children = gtk_container_get_children(container);
|
|
|
|
GList* list = children;
|
|
|
|
GtkWidget* min_child = NULL;
|
|
|
|
int64_t x = INT64_MAX;
|
|
|
|
int64_t y = INT64_MAX;
|
2019-11-04 16:55:35 -05:00
|
|
|
for(; list != NULL; list = list->next) {
|
2019-10-13 15:42:05 -04:00
|
|
|
GtkWidget* child = list->data;
|
|
|
|
GtkAllocation alloc;
|
|
|
|
gtk_widget_get_allocation(child, &alloc);
|
2019-11-04 20:46:39 -05:00
|
|
|
if(alloc.x <= x && alloc.y <= y && alloc.x != -1 && alloc.y != -1 && alloc.width != 1 && alloc.height != 1 && gtk_widget_get_child_visible(child)) {
|
2019-10-13 15:42:05 -04:00
|
|
|
x = alloc.x;
|
|
|
|
y = alloc.y;
|
|
|
|
min_child = child;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
g_list_free(children);
|
|
|
|
return min_child;
|
|
|
|
}
|
|
|
|
|
2019-08-17 21:19:32 -04:00
|
|
|
static void activate_search(GtkEntry* entry, gpointer data) {
|
2019-08-19 20:34:12 -04:00
|
|
|
(void) data;
|
2019-11-04 20:46:39 -05:00
|
|
|
GtkWidget* child = get_first_child(GTK_CONTAINER(inner_box));
|
2020-01-23 23:26:33 -05:00
|
|
|
if(mode != NULL && (exec_search || child == NULL)) {
|
2019-11-16 17:52:20 -05:00
|
|
|
execute_action(mode, gtk_entry_get_text(entry));
|
2020-01-23 23:26:33 -05:00
|
|
|
} else if(child != NULL) {
|
2019-10-13 15:42:05 -04:00
|
|
|
GtkWidget* box = gtk_bin_get_child(GTK_BIN(child));
|
2019-11-13 04:17:45 -05:00
|
|
|
bool primary_action = GTK_IS_EXPANDER(box);
|
|
|
|
if(primary_action) {
|
|
|
|
box = gtk_expander_get_label_widget(GTK_EXPANDER(box));
|
|
|
|
}
|
2019-11-16 17:52:20 -05:00
|
|
|
execute_action(wofi_property_box_get_property(WOFI_PROPERTY_BOX(box), "mode"), wofi_property_box_get_property(WOFI_PROPERTY_BOX(box), "action"));
|
2019-08-19 20:34:12 -04:00
|
|
|
}
|
2019-08-17 21:19:32 -04:00
|
|
|
}
|
|
|
|
|
2019-08-28 22:33:08 -04:00
|
|
|
static gboolean do_filter(GtkFlowBoxChild* row, gpointer data) {
|
2019-08-17 21:19:32 -04:00
|
|
|
(void) data;
|
2019-08-28 02:15:18 -04:00
|
|
|
GtkWidget* box = gtk_bin_get_child(GTK_BIN(row));
|
2019-11-13 04:17:45 -05:00
|
|
|
if(GTK_IS_EXPANDER(box)) {
|
|
|
|
box = gtk_expander_get_label_widget(GTK_EXPANDER(box));
|
|
|
|
}
|
2019-08-28 02:15:18 -04:00
|
|
|
const gchar* text = wofi_property_box_get_property(WOFI_PROPERTY_BOX(box), "filter");
|
2019-08-17 21:19:32 -04:00
|
|
|
if(filter == NULL || strcmp(filter, "") == 0) {
|
|
|
|
return TRUE;
|
|
|
|
}
|
2019-08-29 00:15:29 -04:00
|
|
|
if(text == NULL) {
|
|
|
|
return FALSE;
|
|
|
|
}
|
2019-11-25 17:56:46 -05:00
|
|
|
if(insensitive) {
|
|
|
|
return strcasestr(text, filter) != NULL;
|
|
|
|
} else {
|
|
|
|
return strstr(text, filter) != NULL;
|
|
|
|
}
|
2019-11-21 04:01:27 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
static gint do_sort(GtkFlowBoxChild* child1, GtkFlowBoxChild* child2, gpointer data) {
|
|
|
|
(void) data;
|
|
|
|
GtkWidget* box1 = gtk_bin_get_child(GTK_BIN(child1));
|
|
|
|
GtkWidget* box2 = gtk_bin_get_child(GTK_BIN(child2));
|
|
|
|
if(GTK_IS_EXPANDER(box1)) {
|
|
|
|
box1 = gtk_expander_get_label_widget(GTK_EXPANDER(box1));
|
|
|
|
}
|
|
|
|
if(GTK_IS_EXPANDER(box2)) {
|
|
|
|
box2 = gtk_expander_get_label_widget(GTK_EXPANDER(box2));
|
|
|
|
}
|
|
|
|
|
|
|
|
const gchar* text1 = wofi_property_box_get_property(WOFI_PROPERTY_BOX(box1), "filter");
|
|
|
|
const gchar* text2 = wofi_property_box_get_property(WOFI_PROPERTY_BOX(box2), "filter");
|
|
|
|
if(filter == NULL || strcmp(filter, "") == 0) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
if(text1 == NULL || text2 == NULL) {
|
|
|
|
return 0;
|
|
|
|
}
|
2020-01-26 14:45:56 -05:00
|
|
|
|
|
|
|
char* _filter = strdup(filter);
|
|
|
|
size_t len = strlen(_filter);
|
|
|
|
|
|
|
|
char* t1 = strdup(text1);
|
|
|
|
size_t t1l = strlen(t1);
|
|
|
|
|
|
|
|
char* t2 = strdup(text2);
|
|
|
|
size_t t2l = strlen(t2);
|
|
|
|
|
|
|
|
if(insensitive) {
|
|
|
|
for(size_t count = 0; count < len; ++count) {
|
|
|
|
char chr = _filter[count];
|
|
|
|
if(isalpha(chr)) {
|
|
|
|
_filter[count] = tolower(chr);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
for(size_t count = 0; count < t1l; ++count) {
|
|
|
|
char chr = t1[count];
|
|
|
|
if(isalpha(chr)) {
|
|
|
|
t1[count] = tolower(chr);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
for(size_t count = 0; count < t2l; ++count) {
|
|
|
|
char chr = t2[count];
|
|
|
|
if(isalpha(chr)) {
|
|
|
|
t2[count] = tolower(chr);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
size_t dist1 = utils_distance(t1, _filter);
|
|
|
|
size_t dist2 = utils_distance(t2, _filter);
|
|
|
|
free(_filter);
|
|
|
|
free(t1);
|
|
|
|
free(t2);
|
2020-01-26 15:03:08 -05:00
|
|
|
return dist1 - dist2;
|
2019-08-17 21:19:32 -04:00
|
|
|
}
|
|
|
|
|
2019-08-18 21:23:45 -04:00
|
|
|
static gboolean key_press(GtkWidget* widget, GdkEvent* event, gpointer data) {
|
2019-08-17 21:19:32 -04:00
|
|
|
(void) widget;
|
|
|
|
(void) data;
|
2019-09-27 02:24:48 -04:00
|
|
|
switch(event->key.keyval) {
|
2019-08-18 21:23:45 -04:00
|
|
|
case GDK_KEY_Escape:
|
2020-01-26 14:22:29 -05:00
|
|
|
exit(1);
|
2019-08-18 21:23:45 -04:00
|
|
|
break;
|
|
|
|
case GDK_KEY_Return:
|
2019-11-19 01:16:06 -05:00
|
|
|
mod_shift = (event->key.state & GDK_SHIFT_MASK) == GDK_SHIFT_MASK;
|
2019-11-19 01:13:14 -05:00
|
|
|
mod_ctrl = (event->key.state & GDK_CONTROL_MASK) == GDK_CONTROL_MASK;
|
2019-11-19 01:16:06 -05:00
|
|
|
if(mod_shift) {
|
2019-09-27 02:24:48 -04:00
|
|
|
event->key.state &= ~GDK_SHIFT_MASK;
|
|
|
|
}
|
2019-11-19 01:13:14 -05:00
|
|
|
if(mod_ctrl) {
|
|
|
|
event->key.state &= ~GDK_CONTROL_MASK;
|
|
|
|
}
|
2019-09-19 03:45:57 -04:00
|
|
|
if(gtk_widget_has_focus(scroll)) {
|
|
|
|
gtk_entry_grab_focus_without_selecting(GTK_ENTRY(entry));
|
|
|
|
}
|
2019-08-18 21:23:45 -04:00
|
|
|
break;
|
2019-10-07 05:50:26 -04:00
|
|
|
case GDK_KEY_Tab:
|
2019-08-28 22:33:08 -04:00
|
|
|
case GDK_KEY_Down:
|
2019-10-13 01:24:35 -04:00
|
|
|
case GDK_KEY_Right:
|
|
|
|
if(event->key.keyval == GDK_KEY_Down && outer_orientation == GTK_ORIENTATION_HORIZONTAL) {
|
|
|
|
return FALSE;
|
|
|
|
} else if(event->key.keyval == GDK_KEY_Right && outer_orientation == GTK_ORIENTATION_VERTICAL) {
|
|
|
|
return FALSE;
|
|
|
|
}
|
2019-09-19 03:45:57 -04:00
|
|
|
if(gtk_widget_has_focus(entry) || gtk_widget_has_focus(scroll)) {
|
2019-10-13 15:42:05 -04:00
|
|
|
GtkWidget* child = get_first_child(GTK_CONTAINER(inner_box));
|
|
|
|
gtk_widget_grab_focus(child);
|
|
|
|
gtk_flow_box_select_child(GTK_FLOW_BOX(inner_box), GTK_FLOW_BOX_CHILD(child));
|
2019-08-29 00:55:30 -04:00
|
|
|
return TRUE;
|
2019-08-28 22:33:08 -04:00
|
|
|
}
|
|
|
|
break;
|
2020-01-03 22:54:29 -05:00
|
|
|
case GDK_KEY_Up:
|
|
|
|
case GDK_KEY_Left:
|
|
|
|
case GDK_KEY_Shift_L:
|
|
|
|
case GDK_KEY_Shift_R:
|
|
|
|
case GDK_KEY_Control_L:
|
|
|
|
case GDK_KEY_Control_R:
|
|
|
|
case GDK_KEY_ISO_Left_Tab:
|
2019-10-01 00:38:56 -04:00
|
|
|
break;
|
2019-08-18 21:23:45 -04:00
|
|
|
default:
|
|
|
|
if(!gtk_widget_has_focus(entry)) {
|
2019-08-18 21:31:34 -04:00
|
|
|
gtk_entry_grab_focus_without_selecting(GTK_ENTRY(entry));
|
2019-08-18 21:23:45 -04:00
|
|
|
}
|
|
|
|
break;
|
2019-08-17 21:19:32 -04:00
|
|
|
}
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
2019-09-13 03:25:25 -04:00
|
|
|
static void* get_plugin_proc(const char* prefix, const char* suffix) {
|
2019-11-04 16:25:44 -05:00
|
|
|
char* proc_name = utils_concat(3, "wofi_", prefix, suffix);
|
2019-09-13 03:25:25 -04:00
|
|
|
void* proc = dlsym(RTLD_DEFAULT, proc_name);
|
|
|
|
free(proc_name);
|
|
|
|
return proc;
|
|
|
|
}
|
|
|
|
|
2020-01-23 23:26:33 -05:00
|
|
|
static void add_mode(char* _mode) {
|
|
|
|
char* dso = strstr(_mode, ".so");
|
2019-11-04 23:56:40 -05:00
|
|
|
struct mode* mode_ptr = calloc(1, sizeof(struct mode));
|
2020-01-23 23:26:33 -05:00
|
|
|
map_put_void(modes, _mode, mode_ptr);
|
2019-09-13 03:25:25 -04:00
|
|
|
|
2020-01-23 23:26:33 -05:00
|
|
|
mode_ptr->name = strdup(_mode);
|
2020-01-20 23:09:40 -05:00
|
|
|
|
2020-01-23 23:26:33 -05:00
|
|
|
void (*init)(struct mode* _mode, struct map* props);
|
|
|
|
const char** (*get_arg_names)(void);
|
|
|
|
size_t (*get_arg_count)(void);
|
|
|
|
bool (*no_entry)(void);
|
2019-09-13 03:25:25 -04:00
|
|
|
if(dso == NULL) {
|
2020-01-23 23:26:33 -05:00
|
|
|
init = get_plugin_proc(_mode, "_init");
|
|
|
|
get_arg_names = get_plugin_proc(_mode, "_get_arg_names");
|
|
|
|
get_arg_count = get_plugin_proc(_mode, "_get_arg_count");
|
|
|
|
mode_ptr->mode_exec = get_plugin_proc(_mode, "_exec");
|
|
|
|
mode_ptr->mode_get_widget = get_plugin_proc(_mode, "_get_widget");
|
|
|
|
no_entry = get_plugin_proc(_mode, "_no_entry");
|
2019-09-13 03:25:25 -04:00
|
|
|
} else {
|
|
|
|
char* plugins_dir = utils_concat(2, config_dir, "/plugins/");
|
2020-01-23 23:26:33 -05:00
|
|
|
char* full_name = utils_concat(2, plugins_dir, _mode);
|
2019-11-04 16:29:53 -05:00
|
|
|
void* plugin = dlopen(full_name, RTLD_LAZY | RTLD_LOCAL);
|
2019-09-13 03:25:25 -04:00
|
|
|
free(full_name);
|
|
|
|
free(plugins_dir);
|
|
|
|
init = dlsym(plugin, "init");
|
2019-12-02 01:58:30 -05:00
|
|
|
get_arg_names = dlsym(plugin, "get_arg_names");
|
|
|
|
get_arg_count = dlsym(plugin, "get_arg_count");
|
2019-11-04 23:56:40 -05:00
|
|
|
mode_ptr->mode_exec = dlsym(plugin, "exec");
|
2020-01-16 19:35:51 -05:00
|
|
|
mode_ptr->mode_get_widget = dlsym(plugin, "get_widget");
|
2020-01-23 23:26:33 -05:00
|
|
|
no_entry = dlsym(plugin, "no_entry");
|
2019-09-13 03:25:25 -04:00
|
|
|
}
|
|
|
|
|
2019-12-02 01:58:30 -05:00
|
|
|
const char** arg_names = NULL;
|
|
|
|
size_t arg_count = 0;
|
|
|
|
if(get_arg_names != NULL && get_arg_count != NULL) {
|
|
|
|
arg_names = get_arg_names();
|
|
|
|
arg_count = get_arg_count();
|
|
|
|
}
|
|
|
|
|
2020-01-23 23:26:33 -05:00
|
|
|
if(mode == NULL && no_entry != NULL && no_entry()) {
|
|
|
|
mode = _mode;
|
|
|
|
}
|
|
|
|
|
2019-12-02 01:58:30 -05:00
|
|
|
struct map* props = map_init();
|
|
|
|
for(size_t count = 0; count < arg_count; ++count) {
|
|
|
|
const char* arg = arg_names[count];
|
2020-01-23 23:26:33 -05:00
|
|
|
char* full_name = utils_concat(3, _mode, "-", arg);
|
2019-12-02 01:58:30 -05:00
|
|
|
map_put(props, arg, config_get(config, full_name, NULL));
|
|
|
|
free(full_name);
|
|
|
|
}
|
|
|
|
|
2019-09-13 03:25:25 -04:00
|
|
|
if(init != NULL) {
|
2020-01-20 23:09:40 -05:00
|
|
|
init(mode_ptr, props);
|
2020-01-16 19:35:51 -05:00
|
|
|
gdk_threads_add_idle(_insert_widget, mode_ptr);
|
2019-09-13 03:25:25 -04:00
|
|
|
} else {
|
2020-01-23 23:26:33 -05:00
|
|
|
fprintf(stderr, "I would love to show %s but Idk what it is\n", _mode);
|
2019-09-13 03:25:25 -04:00
|
|
|
exit(1);
|
|
|
|
}
|
2019-12-02 01:58:30 -05:00
|
|
|
|
|
|
|
map_free(props);
|
2019-11-04 23:56:40 -05:00
|
|
|
}
|
2019-09-13 03:25:25 -04:00
|
|
|
|
2019-11-04 23:56:40 -05:00
|
|
|
static void* start_thread(void* data) {
|
2020-01-23 23:26:33 -05:00
|
|
|
char* mode = data;
|
2019-11-05 01:41:53 -05:00
|
|
|
if(strchr(mode, ',') != NULL) {
|
2019-11-04 23:56:40 -05:00
|
|
|
char* save_ptr;
|
2019-11-05 01:41:53 -05:00
|
|
|
char* str = strtok_r(mode, ",", &save_ptr);
|
2019-11-04 23:56:40 -05:00
|
|
|
do {
|
2019-11-05 01:41:53 -05:00
|
|
|
add_mode(str);
|
|
|
|
} while((str = strtok_r(NULL, ",", &save_ptr)) != NULL);
|
2019-11-04 23:56:40 -05:00
|
|
|
} else {
|
2019-11-05 01:41:53 -05:00
|
|
|
add_mode(mode);
|
2019-11-04 23:56:40 -05:00
|
|
|
}
|
2019-09-13 03:25:25 -04:00
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2019-12-02 01:58:30 -05:00
|
|
|
void wofi_init(struct map* _config) {
|
|
|
|
config = _config;
|
2019-08-17 21:19:32 -04:00
|
|
|
width = strtol(config_get(config, "width", "1000"), NULL, 10);
|
|
|
|
height = strtol(config_get(config, "height", "400"), NULL, 10);
|
2020-01-06 03:28:57 -05:00
|
|
|
x = map_get(config, "x");
|
|
|
|
y = map_get(config, "y");
|
2019-08-17 22:22:09 -04:00
|
|
|
bool normal_window = strcmp(config_get(config, "normal_window", "false"), "true") == 0;
|
2020-01-23 23:26:33 -05:00
|
|
|
char* mode = map_get(config, "mode");
|
2020-01-07 21:56:57 -05:00
|
|
|
GtkOrientation orientation = config_get_mnemonic(config, "orientation", "vertical", 2, "vertical", "horizontal");
|
2019-10-13 01:24:35 -04:00
|
|
|
outer_orientation = config_get_mnemonic(config, "orientation", "vertical", 2, "horizontal", "vertical");
|
2020-01-07 21:56:57 -05:00
|
|
|
GtkAlign halign = config_get_mnemonic(config, "halign", "fill", 4, "fill", "start", "end", "center");
|
2019-12-23 04:27:49 -05:00
|
|
|
content_halign = config_get_mnemonic(config, "content_halign", "fill", 4, "fill", "start", "end", "center");
|
2019-10-13 01:24:35 -04:00
|
|
|
char* default_valign = "start";
|
|
|
|
if(outer_orientation == GTK_ORIENTATION_HORIZONTAL) {
|
|
|
|
default_valign = "center";
|
|
|
|
}
|
2020-01-07 21:56:57 -05:00
|
|
|
GtkAlign valign = config_get_mnemonic(config, "valign", default_valign, 4, "fill", "start", "end", "center");
|
2019-08-17 21:19:32 -04:00
|
|
|
char* prompt = config_get(config, "prompt", mode);
|
2019-08-27 00:19:59 -04:00
|
|
|
filter_rate = strtol(config_get(config, "filter_rate", "100"), NULL, 10);
|
|
|
|
filter_time = utils_get_time_millis();
|
2019-08-28 02:15:18 -04:00
|
|
|
allow_images = strcmp(config_get(config, "allow_images", "false"), "true") == 0;
|
2019-08-28 03:58:45 -04:00
|
|
|
allow_markup = strcmp(config_get(config, "allow_markup", "false"), "true") == 0;
|
2019-08-28 20:36:20 -04:00
|
|
|
image_size = strtol(config_get(config, "image_size", "32"), NULL, 10);
|
2019-08-28 23:14:38 -04:00
|
|
|
cache_file = map_get(config, "cache_file");
|
2019-09-13 03:25:25 -04:00
|
|
|
config_dir = map_get(config, "config_dir");
|
2019-09-27 02:24:48 -04:00
|
|
|
terminal = map_get(config, "term");
|
2019-10-23 00:19:57 -04:00
|
|
|
char* password_char = map_get(config, "password_char");
|
2019-11-04 20:46:39 -05:00
|
|
|
exec_search = strcmp(config_get(config, "exec_search", "false"), "true") == 0;
|
2019-11-18 16:42:40 -05:00
|
|
|
bool hide_scroll = strcmp(config_get(config, "hide_scroll", "false"), "true") == 0;
|
2019-11-21 04:01:27 -05:00
|
|
|
matching = config_get_mnemonic(config, "matching", "contains", 2, "contains", "fuzzy");
|
2019-11-25 17:56:46 -05:00
|
|
|
insensitive = strcmp(config_get(config, "insensitive", "false"), "true") == 0;
|
2019-12-16 21:36:21 -05:00
|
|
|
parse_search = strcmp(config_get(config, "parse_search", "false"), "true") == 0;
|
2020-01-07 21:30:34 -05:00
|
|
|
location = config_get_mnemonic(config, "location", "center", 18,
|
|
|
|
"center", "top_left", "top", "top_right", "right", "bottom_right", "bottom", "bottom_left", "left",
|
|
|
|
"0", "1", "2", "3", "4", "5", "6", "7", "8");
|
2020-01-10 17:34:49 -05:00
|
|
|
no_actions = strcmp(config_get(config, "no_actions", "false"), "true") == 0;
|
2019-11-04 23:56:40 -05:00
|
|
|
modes = map_init_void();
|
2019-08-17 21:19:32 -04:00
|
|
|
|
|
|
|
window = gtk_window_new(GTK_WINDOW_TOPLEVEL);
|
|
|
|
gtk_widget_realize(window);
|
|
|
|
gtk_widget_set_name(window, "window");
|
2019-08-17 22:22:09 -04:00
|
|
|
gtk_window_set_default_size(GTK_WINDOW(window), width, height);
|
|
|
|
gtk_window_resize(GTK_WINDOW(window), width, height);
|
|
|
|
gtk_window_set_resizable(GTK_WINDOW(window), FALSE);
|
|
|
|
gtk_window_set_decorated(GTK_WINDOW(window), FALSE);
|
|
|
|
if(!normal_window) {
|
|
|
|
GdkDisplay* disp = gdk_display_get_default();
|
|
|
|
struct wl_display* wl = gdk_wayland_display_get_wl_display(disp);
|
|
|
|
struct wl_registry* registry = wl_display_get_registry(wl);
|
|
|
|
struct wl_registry_listener listener = {
|
|
|
|
.global = add_interface,
|
|
|
|
.global_remove = nop
|
|
|
|
};
|
|
|
|
wl_registry_add_listener(registry, &listener, NULL);
|
|
|
|
wl_display_roundtrip(wl);
|
|
|
|
GdkWindow* gdk_win = gtk_widget_get_window(window);
|
|
|
|
gdk_wayland_window_set_use_custom_surface(gdk_win);
|
|
|
|
struct wl_surface* wl_surface = gdk_wayland_window_get_wl_surface(gdk_win);
|
|
|
|
struct zwlr_layer_surface_v1* surface = zwlr_layer_shell_v1_get_layer_surface(shell, wl_surface, NULL, ZWLR_LAYER_SHELL_V1_LAYER_TOP, "wofi");
|
|
|
|
struct zwlr_layer_surface_v1_listener* surface_listener = malloc(sizeof(struct zwlr_layer_surface_v1_listener));
|
|
|
|
surface_listener->configure = config_surface;
|
|
|
|
surface_listener->closed = nop;
|
|
|
|
zwlr_layer_surface_v1_add_listener(surface, surface_listener, NULL);
|
|
|
|
wl_surface_commit(wl_surface);
|
|
|
|
wl_display_roundtrip(wl);
|
|
|
|
}
|
2019-08-17 21:19:32 -04:00
|
|
|
|
2019-10-13 01:24:35 -04:00
|
|
|
outer_box = gtk_box_new(outer_orientation, 0);
|
2019-08-19 20:34:12 -04:00
|
|
|
gtk_widget_set_name(outer_box, "outer-box");
|
|
|
|
gtk_container_add(GTK_CONTAINER(window), outer_box);
|
|
|
|
entry = gtk_search_entry_new();
|
2019-08-17 21:19:32 -04:00
|
|
|
gtk_widget_set_name(entry, "input");
|
|
|
|
gtk_entry_set_placeholder_text(GTK_ENTRY(entry), prompt);
|
2019-08-19 20:34:12 -04:00
|
|
|
gtk_container_add(GTK_CONTAINER(outer_box), entry);
|
2019-10-23 00:19:57 -04:00
|
|
|
if(password_char != NULL) {
|
|
|
|
gtk_entry_set_visibility(GTK_ENTRY(entry), FALSE);
|
|
|
|
gtk_entry_set_invisible_char(GTK_ENTRY(entry), password_char[0]);
|
|
|
|
}
|
2019-08-17 21:19:32 -04:00
|
|
|
|
2019-08-19 20:34:12 -04:00
|
|
|
scroll = gtk_scrolled_window_new(NULL, NULL);
|
2019-08-17 21:19:32 -04:00
|
|
|
gtk_widget_set_name(scroll, "scroll");
|
2019-08-19 20:34:12 -04:00
|
|
|
gtk_container_add(GTK_CONTAINER(outer_box), scroll);
|
2019-08-17 21:19:32 -04:00
|
|
|
gtk_widget_set_size_request(scroll, width, height);
|
2019-11-18 16:42:40 -05:00
|
|
|
if(hide_scroll) {
|
|
|
|
gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(scroll), GTK_POLICY_EXTERNAL, GTK_POLICY_EXTERNAL);
|
|
|
|
}
|
2019-08-17 21:19:32 -04:00
|
|
|
|
2019-08-28 22:33:08 -04:00
|
|
|
inner_box = gtk_flow_box_new();
|
|
|
|
gtk_flow_box_set_max_children_per_line(GTK_FLOW_BOX(inner_box), 1);
|
|
|
|
gtk_orientable_set_orientation(GTK_ORIENTABLE(inner_box), orientation);
|
|
|
|
gtk_widget_set_halign(inner_box, halign);
|
|
|
|
gtk_widget_set_valign(inner_box, valign);
|
|
|
|
|
2019-08-17 21:19:32 -04:00
|
|
|
gtk_widget_set_name(inner_box, "inner-box");
|
2019-08-28 22:33:08 -04:00
|
|
|
gtk_flow_box_set_activate_on_single_click(GTK_FLOW_BOX(inner_box), FALSE);
|
2019-10-13 02:56:27 -04:00
|
|
|
|
|
|
|
GtkWidget* wrapper_box = gtk_box_new(GTK_ORIENTATION_VERTICAL, 0);
|
|
|
|
gtk_box_set_homogeneous(GTK_BOX(wrapper_box), TRUE);
|
|
|
|
gtk_container_add(GTK_CONTAINER(wrapper_box), inner_box);
|
|
|
|
gtk_container_add(GTK_CONTAINER(scroll), wrapper_box);
|
2019-08-17 21:19:32 -04:00
|
|
|
|
2019-11-21 04:01:27 -05:00
|
|
|
switch(matching) {
|
|
|
|
case MATCHING_MODE_CONTAINS:
|
|
|
|
gtk_flow_box_set_filter_func(GTK_FLOW_BOX(inner_box), do_filter, NULL, NULL);
|
|
|
|
break;
|
|
|
|
case MATCHING_MODE_FUZZY:
|
|
|
|
gtk_flow_box_set_sort_func(GTK_FLOW_BOX(inner_box), do_sort, NULL, NULL);
|
|
|
|
break;
|
|
|
|
}
|
2019-08-17 21:19:32 -04:00
|
|
|
|
2019-08-27 00:19:59 -04:00
|
|
|
g_signal_connect(entry, "changed", G_CALLBACK(get_input), NULL);
|
|
|
|
g_signal_connect(entry, "search-changed", G_CALLBACK(get_search), NULL);
|
2019-08-28 22:33:08 -04:00
|
|
|
g_signal_connect(inner_box, "child-activated", G_CALLBACK(activate_item), NULL);
|
|
|
|
g_signal_connect(inner_box, "selected-children-changed", G_CALLBACK(select_item), NULL);
|
2019-08-19 20:34:12 -04:00
|
|
|
g_signal_connect(entry, "activate", G_CALLBACK(activate_search), NULL);
|
|
|
|
g_signal_connect(window, "key-press-event", G_CALLBACK(key_press), NULL);
|
2019-08-17 21:19:32 -04:00
|
|
|
|
2019-08-18 21:06:13 -04:00
|
|
|
pthread_t thread;
|
2020-01-23 23:26:33 -05:00
|
|
|
pthread_create(&thread, NULL, start_thread, mode);
|
2019-09-19 02:35:51 -04:00
|
|
|
gtk_widget_grab_focus(scroll);
|
2019-08-18 01:00:23 -04:00
|
|
|
gtk_window_set_title(GTK_WINDOW(window), prompt);
|
2019-08-17 22:22:09 -04:00
|
|
|
gtk_widget_show_all(window);
|
2019-08-17 21:19:32 -04:00
|
|
|
}
|