1
0
Fork 0
mirror of https://github.com/davatorium/rofi.git synced 2024-11-18 13:54:36 -05:00
rofi/source/widgets/listview.c
giomatfois62 75ae2a41f0
[WIP] xdg thumbnails fetching with fallback on mimetype icons (#1939)
* 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>
2024-06-21 18:47:38 +02:00

1207 lines
36 KiB
C

/*
* rofi
*
* MIT/X11 License
* Copyright © 2013-2023 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.
*
*/
#include "config.h"
#include <glib.h>
#include <widgets/box.h>
#include <widgets/icon.h>
#include <widgets/listview.h>
#include <widgets/scrollbar.h>
#include <widgets/textbox.h>
#include <widgets/widget.h>
#include "settings.h"
#include "theme.h"
#include "view.h"
#include "timings.h"
/** Default spacing between the elements in the listview. */
#define DEFAULT_SPACING 2
/**
* Orientation of the listview
*/
/** Vertical (classical) list */
#define LISTVIEW ROFI_ORIENTATION_VERTICAL
/** Horizontal list. (barview) */
#define BARVIEW ROFI_ORIENTATION_HORIZONTAL
/**
* The moving direction of the selection, this (in barview) affects the
* scrolling.
*/
typedef enum { LEFT_TO_RIGHT = 0, RIGHT_TO_LEFT = 1 } MoveDirection;
typedef struct {
box *box;
textbox *textbox;
textbox *index;
icon *icon;
} _listview_row;
struct _listview {
widget widget;
RofiOrientation type;
// RChanged
// Text needs to be repainted.
unsigned int rchanged;
// The direction we pack the widgets.
RofiOrientation pack_direction;
// Administration
unsigned int cur_page;
unsigned int last_offset;
unsigned int selected;
unsigned int element_height;
unsigned int max_rows;
unsigned int max_elements;
//
gboolean fixed_columns;
unsigned int cur_columns;
unsigned int req_elements;
unsigned int cur_elements;
RofiDistance spacing;
unsigned int menu_lines;
unsigned int max_displayed_lines;
unsigned int menu_columns;
unsigned int fixed_num_lines;
unsigned int dynamic;
unsigned int eh;
unsigned int reverse;
gboolean require_input;
gboolean filtered;
gboolean cycle;
ScrollType scroll_type;
_listview_row *boxes;
scrollbar *scrollbar;
listview_update_callback callback;
void *udata;
listview_selection_changed_callback sc_callback;
void *sc_udata;
gboolean scrollbar_scroll;
xcb_timestamp_t last_click;
listview_mouse_activated_cb mouse_activated;
void *mouse_activated_data;
listview_page_changed_cb page_callback;
char *listview_name;
PangoEllipsizeMode emode;
/** Barview */
struct {
MoveDirection direction;
unsigned int cur_visible;
} barview;
};
/**
* Names used for theming the elements in the listview.
* Each row can have 3 modes, normal, selected and alternate.
* Each row can have 3 states, normal, urgent and active.
*/
const char *const listview_theme_prop_names[][3] = {
/** Normal row */
{"normal.normal", "selected.normal", "alternate.normal"},
/** Urgent row */
{"normal.urgent", "selected.urgent", "alternate.urgent"},
/** Active row */
{"normal.active", "selected.active", "alternate.active"},
};
static void listview_set_state(_listview_row r, TextBoxFontType tbft) {
widget *w = WIDGET(r.box);
TextBoxFontType t = tbft & STATE_MASK;
if (w == NULL) {
return;
}
// ACTIVE has priority over URGENT if both set.
if (t == (URGENT | ACTIVE)) {
t = ACTIVE;
}
switch ((tbft & FMOD_MASK)) {
case HIGHLIGHT:
widget_set_state(w, listview_theme_prop_names[t][1]);
break;
case ALT:
widget_set_state(w, listview_theme_prop_names[t][2]);
break;
default:
widget_set_state(w, listview_theme_prop_names[t][0]);
break;
}
}
static void listview_add_widget(listview *lv, _listview_row *row, widget *wid,
const char *label) {
if (strcasecmp(label, "element-icon") == 0) {
row->icon = icon_create(WIDGET(wid), "element-icon");
box_add((box *)wid, WIDGET(row->icon), FALSE);
} else if (strcasecmp(label, "element-text") == 0) {
row->textbox =
textbox_create(WIDGET(wid), WIDGET_TYPE_TEXTBOX_TEXT, "element-text",
TB_AUTOHEIGHT, NORMAL, "DDD", 0, 0);
textbox_set_ellipsize(row->textbox, lv->emode);
box_add((box *)wid, WIDGET(row->textbox), TRUE);
} else if (strcasecmp(label, "element-index") == 0) {
row->index =
textbox_create(WIDGET(wid), WIDGET_TYPE_TEXTBOX_TEXT, "element-index",
TB_AUTOHEIGHT, NORMAL, " ", 0, 0);
box_add((box *)wid, WIDGET(row->index), FALSE);
} else if (strncasecmp(label, "textbox", 7) == 0) {
textbox *textbox_custom =
textbox_create(wid, WIDGET_TYPE_TEXTBOX_TEXT, label,
TB_AUTOHEIGHT | TB_WRAP, NORMAL, "", 0, 0);
box_add((box *)wid, WIDGET(textbox_custom), TRUE);
} else if (strncasecmp(label, "button", 6) == 0) {
textbox *button_custom =
textbox_create(wid, WIDGET_TYPE_EDITBOX, label, TB_AUTOHEIGHT | TB_WRAP,
NORMAL, "", 0, 0);
box_add((box *)wid, WIDGET(button_custom), TRUE);
widget_set_trigger_action_handler(WIDGET(button_custom),
textbox_button_trigger_action, lv->udata);
} else if (strncasecmp(label, "icon", 4) == 0) {
icon *icon_custom = icon_create(wid, label);
/* small hack to make it clickable */
const char *type =
rofi_theme_get_string(WIDGET(icon_custom), "action", NULL);
if (type) {
WIDGET(icon_custom)->type = WIDGET_TYPE_EDITBOX;
}
box_add((box *)wid, WIDGET(icon_custom), TRUE);
widget_set_trigger_action_handler(WIDGET(icon_custom),
textbox_button_trigger_action, lv->udata);
} else {
widget *wid2 = (widget *)box_create(wid, label, ROFI_ORIENTATION_VERTICAL);
box_add((box *)wid, WIDGET(wid2), TRUE);
GList *list = rofi_theme_get_list_strings(
WIDGET(wid2),
"children"); // rofi_theme_get_list(WIDGET(wid2), "children", "");
for (GList *iter = g_list_first(list); iter != NULL;
iter = g_list_next(iter)) {
listview_add_widget(lv, row, wid2, (const char *)iter->data);
}
}
}
static void listview_create_row(listview *lv, _listview_row *row) {
row->box = box_create(WIDGET(lv), "element", ROFI_ORIENTATION_HORIZONTAL);
widget_set_type(WIDGET(row->box), WIDGET_TYPE_LISTVIEW_ELEMENT);
GList *list = NULL;
list = rofi_theme_get_list_strings(WIDGET(row->box), "children");
if (list == NULL) {
if (config.show_icons) {
list = g_list_append(list, g_strdup("element-icon"));
list = g_list_append(list, g_strdup("element-text"));
} else {
list = g_list_append(list, g_strdup("element-text"));
}
}
row->textbox = NULL;
row->icon = NULL;
row->index = NULL;
for (GList *iter = g_list_first(list); iter != NULL;
iter = g_list_next(iter)) {
listview_add_widget(lv, row, WIDGET(row->box), (const char *)iter->data);
}
g_list_free_full(list, g_free);
}
static int listview_get_desired_height(widget *wid, const int width);
static void listview_free(widget *wid) {
listview *lv = (listview *)wid;
for (unsigned int i = 0; i < lv->cur_elements; i++) {
widget_free(WIDGET(lv->boxes[i].box));
}
g_free(lv->boxes);
g_free(lv->listview_name);
widget_free(WIDGET(lv->scrollbar));
g_free(lv);
}
static unsigned int scroll_per_page_barview(listview *lv) {
unsigned int offset = lv->last_offset;
// selected row is always visible.
// If selected is visible do not scroll.
if (lv->selected < lv->last_offset) {
offset = lv->selected;
lv->rchanged = TRUE;
} else if (lv->selected >= (lv->last_offset + lv->barview.cur_visible)) {
offset = lv->selected;
lv->rchanged = TRUE;
}
return offset;
}
static unsigned int scroll_per_page(listview *lv) {
int offset = 0;
// selected row is always visible.
// If selected is visible do not scroll.
if (((lv->selected - (lv->last_offset)) < (lv->max_elements)) &&
(lv->selected >= (lv->last_offset))) {
offset = lv->last_offset;
} else {
// Do paginating
unsigned int page =
(lv->max_elements > 0) ? (lv->selected / lv->max_elements) : 0;
offset = page * lv->max_elements;
if (page != lv->cur_page) {
if (lv->page_callback)
lv->page_callback();
lv->cur_page = page;
lv->rchanged = TRUE;
}
// Set the position
// scrollbar_set_handle ( lv->scrollbar, page * lv->max_elements );
}
return offset;
}
// For vertical packing flow
static unsigned int scroll_continious_elements(listview *lv) {
unsigned int vmid = (lv->max_rows - 1) / 2;
unsigned int hmid = (lv->menu_columns - 1) / 2;
unsigned int middle = (lv->max_rows * hmid) + vmid;
unsigned int offset = 0;
if (lv->selected > middle) {
if (lv->selected < (lv->req_elements - (lv->max_elements - middle))) {
offset = lv->selected - middle;
}
// Don't go below zero.
else if (lv->req_elements > lv->max_elements) {
offset = lv->req_elements - lv->max_elements;
}
}
if (offset != lv->cur_page) {
// scrollbar_set_handle ( lv->scrollbar, offset );
lv->cur_page = offset;
lv->rchanged = TRUE;
}
return offset;
}
// For horizontal packing flow
static unsigned int scroll_continious_rows(listview *lv) {
unsigned int middle, selected, req_rows, offset;
middle = (lv->max_rows - 1) / 2;
selected = lv->selected / lv->menu_columns;
req_rows = (lv->req_elements + lv->menu_columns - 1) / lv->menu_columns;
offset = 0;
if (selected > middle) {
if (selected < (req_rows - (lv->max_rows - middle))) {
offset = selected - middle;
}
// Don't go below zero.
else if (req_rows > lv->max_rows) {
offset = req_rows - lv->max_rows;
}
}
offset *= lv->menu_columns;
if (offset != lv->cur_page) {
// scrollbar_set_handle ( lv->scrollbar, offset );
lv->cur_page = offset;
lv->rchanged = TRUE;
}
return offset;
}
static void update_element(listview *lv, unsigned int tb, unsigned int index,
gboolean full) {
// Select drawing mode
TextBoxFontType type = (index & 1) == 0 ? NORMAL : ALT;
type = (index) == lv->selected ? HIGHLIGHT : type;
if (lv->boxes[tb].index) {
if (index < 10) {
char str[2] = {((index + 1) % 10) + '0', '\0'};
textbox_text(lv->boxes[tb].index, str);
} else {
textbox_text(lv->boxes[tb].index, " ");
}
}
if (lv->callback) {
lv->callback(lv->boxes[tb].textbox, lv->boxes[tb].icon, index, lv->udata,
&type, full);
listview_set_state(lv->boxes[tb], type);
}
}
static void barview_draw(widget *wid, cairo_t *draw) {
unsigned int offset = 0;
listview *lv = (listview *)wid;
offset = scroll_per_page_barview(lv);
lv->last_offset = offset;
int spacing_hori =
distance_get_pixel(lv->spacing, ROFI_ORIENTATION_HORIZONTAL);
int left_offset = widget_padding_get_left(wid);
int right_offset = lv->widget.w - widget_padding_get_right(wid);
int top_offset = widget_padding_get_top(wid);
if (lv->cur_elements > 0) {
// Set new x/y position.
unsigned int max = MIN(lv->cur_elements, lv->req_elements - offset);
if (lv->rchanged) {
int first = TRUE;
int width = lv->widget.w;
lv->barview.cur_visible = 0;
width -= widget_padding_get_padding_width(wid);
if (lv->barview.direction == LEFT_TO_RIGHT) {
for (unsigned int i = 0; i < max && width > 0; i++) {
update_element(lv, i, i + offset, TRUE);
int twidth = widget_get_desired_width(WIDGET(lv->boxes[i].box),
lv->element_height);
if (twidth >= width) {
if (!first) {
break;
}
twidth = width;
}
widget_move(WIDGET(lv->boxes[i].box), left_offset, top_offset);
widget_resize(WIDGET(lv->boxes[i].box), twidth, lv->element_height);
widget_draw(WIDGET(lv->boxes[i].box), draw);
width -= twidth + spacing_hori;
left_offset += twidth + spacing_hori;
first = FALSE;
lv->barview.cur_visible++;
}
} else {
for (unsigned int i = 0;
i < lv->cur_elements && width > 0 && i <= offset; i++) {
update_element(lv, i, offset - i, TRUE);
int twidth = widget_get_desired_width(WIDGET(lv->boxes[i].box),
lv->element_height);
if (twidth >= width) {
if (!first) {
break;
}
twidth = width;
}
right_offset -= twidth;
widget_move(WIDGET(lv->boxes[i].box), right_offset, top_offset);
widget_resize(WIDGET(lv->boxes[i].box), twidth, lv->element_height);
widget_draw(WIDGET(lv->boxes[i].box), draw);
width -= twidth + spacing_hori;
right_offset -= spacing_hori;
first = FALSE;
lv->barview.cur_visible++;
}
offset -= lv->barview.cur_visible - 1;
lv->last_offset = offset;
for (unsigned int i = 0; i < (lv->barview.cur_visible / 2); i++) {
_listview_row temp = lv->boxes[i];
int sw = lv->barview.cur_visible - i - 1;
lv->boxes[i] = lv->boxes[sw];
lv->boxes[sw] = temp;
}
}
lv->rchanged = FALSE;
} else {
for (unsigned int i = 0; i < lv->barview.cur_visible; i++) {
update_element(lv, i, i + offset, TRUE);
widget_draw(WIDGET(lv->boxes[i].box), draw);
}
}
}
}
static void listview_draw(widget *wid, cairo_t *draw) {
unsigned int offset = 0;
listview *lv = (listview *)wid;
if (lv->scroll_type == LISTVIEW_SCROLL_PER_PAGE) {
offset = scroll_per_page(lv);
} else if (lv->pack_direction == ROFI_ORIENTATION_VERTICAL) {
offset = scroll_continious_elements(lv);
} else {
offset = scroll_continious_rows(lv);
}
// Set these all together to make sure they update consistently.
scrollbar_set_max_value(lv->scrollbar, lv->req_elements);
scrollbar_set_handle_length(lv->scrollbar, lv->cur_columns * lv->max_rows);
if (lv->reverse) {
scrollbar_set_handle(lv->scrollbar, lv->req_elements - lv->selected - 1);
} else {
scrollbar_set_handle(lv->scrollbar, lv->selected);
}
lv->last_offset = offset;
int spacing_vert = distance_get_pixel(lv->spacing, ROFI_ORIENTATION_VERTICAL);
int spacing_hori =
distance_get_pixel(lv->spacing, ROFI_ORIENTATION_HORIZONTAL);
int left_offset = widget_padding_get_left(wid);
int top_offset = widget_padding_get_top(wid);
/*
if ( lv->scrollbar->widget.index == 0 ) {
left_offset += spacing_hori + lv->scrollbar->widget.w;
}
*/
if (lv->cur_elements > 0 && lv->max_rows > 0) {
// Set new x/y position.
unsigned int max = MIN(lv->cur_elements, lv->req_elements - offset);
if (lv->rchanged) {
unsigned int width = lv->widget.w;
width -= widget_padding_get_padding_width(wid);
if (widget_enabled(WIDGET(lv->scrollbar))) {
width -= spacing_hori;
width -= widget_get_width(WIDGET(lv->scrollbar));
}
unsigned int element_width =
(width - spacing_hori * (lv->cur_columns - 1)) / lv->cur_columns;
int d = width - (element_width + spacing_hori) * (lv->cur_columns - 1) -
element_width;
if (lv->cur_columns > 1) {
int diff = d / (lv->cur_columns - 1);
if (diff >= 1) {
spacing_hori += 1;
d -= lv->cur_columns - 1;
}
}
for (unsigned int i = 0; i < max; i++) {
if (lv->pack_direction == ROFI_ORIENTATION_HORIZONTAL) {
unsigned int ex = left_offset + ((i) % lv->cur_columns) *
(element_width + spacing_hori);
unsigned int ey = 0;
if (lv->reverse) {
ey = wid->h -
(widget_padding_get_bottom(wid) +
((i) / lv->cur_columns) *
(lv->element_height + spacing_vert)) -
lv->element_height;
if ((i) / lv->cur_columns == (lv->cur_columns - 1)) {
ex += d;
}
} else {
ey = top_offset +
((i) / lv->cur_columns) * (lv->element_height + spacing_vert);
if ((i) / lv->cur_columns == (lv->cur_columns - 1)) {
ex += d;
}
}
widget_move(WIDGET(lv->boxes[i].box), ex, ey);
widget_resize(WIDGET(lv->boxes[i].box), element_width,
lv->element_height);
} else {
unsigned int ex = left_offset + ((i) / lv->max_rows) *
(element_width + spacing_hori);
if ((i) / lv->max_rows == (lv->cur_columns - 1)) {
ex += d;
}
unsigned int ey = 0;
if (lv->reverse) {
ey = wid->h -
(widget_padding_get_bottom(wid) +
((i) % lv->max_rows) * (lv->element_height + spacing_vert)) -
lv->element_height;
} else {
ey = top_offset +
((i) % lv->max_rows) * (lv->element_height + spacing_vert);
}
widget_move(WIDGET(lv->boxes[i].box), ex, ey);
widget_resize(WIDGET(lv->boxes[i].box), element_width,
lv->element_height);
}
update_element(lv, i, i + offset, TRUE);
widget_draw(WIDGET(lv->boxes[i].box), draw);
}
lv->rchanged = FALSE;
} else {
for (unsigned int i = 0; i < max; i++) {
update_element(lv, i, i + offset, TRUE);
widget_draw(WIDGET(lv->boxes[i].box), draw);
}
}
}
widget_draw(WIDGET(lv->scrollbar), draw);
}
static WidgetTriggerActionResult
listview_element_trigger_action(widget *wid,
MouseBindingListviewElementAction action,
gint x, gint y, void *user_data);
static gboolean listview_element_motion_notify(widget *wid, gint x, gint y);
static void _listview_draw(widget *wid, cairo_t *draw) {
listview *lv = (listview *)wid;
if (lv->type == LISTVIEW) {
listview_draw(wid, draw);
} else {
barview_draw(wid, draw);
}
}
/**
* State names used for theming.
*/
static void listview_recompute_elements(listview *lv) {
unsigned int newne = 0;
if (lv->max_rows == 0) {
return;
}
if (!(lv->fixed_columns) && lv->req_elements < lv->max_elements) {
newne = lv->req_elements;
if (lv->pack_direction == ROFI_ORIENTATION_VERTICAL) {
lv->cur_columns = (lv->req_elements + (lv->max_rows - 1)) / lv->max_rows;
} else {
lv->cur_columns = lv->menu_columns;
if (lv->req_elements < lv->menu_columns) {
lv->cur_columns = lv->req_elements;
}
}
} else {
newne = MIN(lv->req_elements, lv->max_elements);
lv->cur_columns = lv->menu_columns;
}
for (unsigned int i = newne; i < lv->cur_elements; i++) {
widget_free(WIDGET(lv->boxes[i].box));
}
lv->boxes = g_realloc(lv->boxes, newne * sizeof(_listview_row));
if (newne > 0) {
for (unsigned int i = lv->cur_elements; i < newne; i++) {
listview_create_row(lv, &(lv->boxes[i]));
widget *wid = WIDGET(lv->boxes[i].box);
widget_set_trigger_action_handler(wid, listview_element_trigger_action,
lv);
if (wid != NULL) {
wid->motion_notify = listview_element_motion_notify;
}
listview_set_state(lv->boxes[i], NORMAL);
}
}
lv->rchanged = TRUE;
lv->cur_elements = newne;
}
void listview_set_num_elements(listview *lv, unsigned int rows) {
if (lv == NULL) {
return;
}
TICK_N("listview_set_num_elements");
lv->req_elements = rows;
if (lv->require_input && !lv->filtered) {
lv->req_elements = 0;
}
listview_set_selected(lv, lv->selected);
TICK_N("Set selected");
listview_recompute_elements(lv);
TICK_N("recompute elements");
widget_queue_redraw(WIDGET(lv));
TICK_N("queue redraw");
}
unsigned int listview_get_selected(listview *lv) {
if (lv != NULL) {
return lv->selected;
}
return 0;
}
void listview_set_selected(listview *lv, unsigned int selected) {
if (lv == NULL) {
return;
}
if (lv->req_elements > 0) {
lv->selected = MIN(selected, lv->req_elements - 1);
lv->barview.direction = LEFT_TO_RIGHT;
widget_queue_redraw(WIDGET(lv));
if (lv->sc_callback) {
lv->sc_callback(lv, lv->selected, lv->sc_udata);
}
} else {
if (lv->sc_callback) {
lv->sc_callback(lv, UINT32_MAX, lv->sc_udata);
}
}
}
static void listview_resize(widget *wid, short w, short h) {
listview *lv = (listview *)wid;
lv->widget.w = MAX(0, w);
lv->widget.h = MAX(0, h);
int height = lv->widget.h - widget_padding_get_padding_height(WIDGET(lv));
int spacing_vert = distance_get_pixel(lv->spacing, ROFI_ORIENTATION_VERTICAL);
if (lv->widget.h == 0) {
lv->max_rows = lv->menu_lines;
} else {
lv->max_rows =
(spacing_vert + height) / (lv->element_height + spacing_vert);
}
lv->max_elements = lv->max_rows * lv->menu_columns;
widget_move(WIDGET(lv->scrollbar),
lv->widget.w - widget_padding_get_right(WIDGET(lv)) -
widget_get_width(WIDGET(lv->scrollbar)),
widget_padding_get_top(WIDGET(lv)));
widget_resize(WIDGET(lv->scrollbar), widget_get_width(WIDGET(lv->scrollbar)),
height);
if (lv->type == BARVIEW) {
lv->max_elements = lv->menu_lines;
}
listview_recompute_elements(lv);
widget_queue_redraw(wid);
}
static widget *listview_find_mouse_target(widget *wid, WidgetType type, gint x,
gint y) {
widget *target = NULL;
gint rx, ry;
listview *lv = (listview *)wid;
if (widget_enabled(WIDGET(lv->scrollbar)) &&
widget_intersect(WIDGET(lv->scrollbar), x, y)) {
rx = x - widget_get_x_pos(WIDGET(lv->scrollbar));
ry = y - widget_get_y_pos(WIDGET(lv->scrollbar));
target = widget_find_mouse_target(WIDGET(lv->scrollbar), type, rx, ry);
}
unsigned int max = MIN(lv->cur_elements, lv->req_elements - lv->last_offset);
unsigned int i;
for (i = 0; i < max && target == NULL; i++) {
widget *w = WIDGET(lv->boxes[i].box);
if (widget_intersect(w, x, y)) {
rx = x - widget_get_x_pos(w);
ry = y - widget_get_y_pos(w);
target = widget_find_mouse_target(w, type, rx, ry);
}
}
return target;
}
static WidgetTriggerActionResult
listview_trigger_action(widget *wid, MouseBindingListviewAction action,
G_GNUC_UNUSED gint x, G_GNUC_UNUSED gint y,
G_GNUC_UNUSED void *user_data) {
listview *lv = (listview *)wid;
switch (action) {
case SCROLL_LEFT:
listview_nav_left(lv);
break;
case SCROLL_RIGHT:
listview_nav_right(lv);
break;
case SCROLL_DOWN:
listview_nav_down(lv);
break;
case SCROLL_UP:
listview_nav_up(lv);
break;
}
return WIDGET_TRIGGER_ACTION_RESULT_HANDLED;
}
static WidgetTriggerActionResult listview_element_trigger_action(
widget *wid, MouseBindingListviewElementAction action, G_GNUC_UNUSED gint x,
G_GNUC_UNUSED gint y, void *user_data) {
listview *lv = (listview *)user_data;
unsigned int max = MIN(lv->cur_elements, lv->req_elements - lv->last_offset);
unsigned int i;
for (i = 0; i < max && WIDGET(lv->boxes[i].box) != wid; i++) {
}
if (i == max) {
return WIDGET_TRIGGER_ACTION_RESULT_IGNORED;
}
gboolean custom = FALSE;
switch (action) {
case SELECT_HOVERED_ENTRY:
listview_set_selected(lv, lv->last_offset + i);
break;
case ACCEPT_HOVERED_CUSTOM:
custom = TRUE;
/* FALLTHRU */
case ACCEPT_HOVERED_ENTRY:
listview_set_selected(lv, lv->last_offset + i);
lv->mouse_activated(lv, custom, lv->mouse_activated_data);
break;
}
return WIDGET_TRIGGER_ACTION_RESULT_HANDLED;
}
static gboolean listview_element_motion_notify(widget *wid,
G_GNUC_UNUSED gint x,
G_GNUC_UNUSED gint y) {
listview *lv = (listview *)wid->parent;
unsigned int max = MIN(lv->cur_elements, lv->req_elements - lv->last_offset);
unsigned int i;
for (i = 0; i < max && WIDGET(lv->boxes[i].box) != wid; i++) {
}
if (i < max && (lv->last_offset + i) != listview_get_selected(lv)) {
listview_set_selected(lv, lv->last_offset + i);
}
return TRUE;
}
listview *listview_create(widget *parent, const char *name,
listview_update_callback cb,
listview_page_changed_cb page_cb, void *udata,
unsigned int eh, gboolean reverse) {
listview *lv = g_malloc0(sizeof(listview));
widget_init(WIDGET(lv), parent, WIDGET_TYPE_LISTVIEW, name);
lv->listview_name = g_strdup(name);
lv->widget.free = listview_free;
lv->widget.resize = listview_resize;
lv->widget.draw = _listview_draw;
lv->widget.find_mouse_target = listview_find_mouse_target;
lv->widget.trigger_action = listview_trigger_action;
lv->widget.get_desired_height = listview_get_desired_height;
lv->eh = eh;
lv->emode = PANGO_ELLIPSIZE_END;
lv->scrollbar = scrollbar_create(WIDGET(lv), "scrollbar");
// Calculate height of an element.
//
_listview_row row;
listview_create_row(lv, &row);
// FIXME: hack to scale hight correctly.
if (lv->eh > 1 && row.textbox) {
char buff[lv->eh * 2 + 1];
memset(buff, '\0', lv->eh * 2 + 1);
for (unsigned int i = 0; i < (lv->eh - 1); i++) {
buff[i * 2] = 'a';
buff[i * 2 + 1] = '\n';
};
textbox_moveresize(row.textbox, 0, 0, 100000000, -1);
textbox_text(row.textbox, buff);
}
// Make textbox very wide.
lv->element_height = widget_get_desired_height(WIDGET(row.box), 100000);
widget_free(WIDGET(row.box));
lv->callback = cb;
lv->udata = udata;
lv->page_callback = page_cb;
// Some settings.
lv->spacing = rofi_theme_get_distance(WIDGET(lv), "spacing", DEFAULT_SPACING);
lv->menu_columns =
rofi_theme_get_integer(WIDGET(lv), "columns", DEFAULT_MENU_COLUMNS);
lv->menu_lines =
rofi_theme_get_integer(WIDGET(lv), "lines", DEFAULT_MENU_LINES);
lv->fixed_num_lines = rofi_theme_get_boolean(WIDGET(lv), "fixed-height",
config.fixed_num_lines);
lv->dynamic = rofi_theme_get_boolean(WIDGET(lv), "dynamic", TRUE);
lv->reverse = rofi_theme_get_boolean(WIDGET(lv), "reverse", reverse);
lv->pack_direction =
rofi_theme_get_orientation(WIDGET(lv), "flow", ROFI_ORIENTATION_VERTICAL);
lv->cycle = rofi_theme_get_boolean(WIDGET(lv), "cycle", config.cycle);
lv->fixed_columns =
rofi_theme_get_boolean(WIDGET(lv), "fixed-columns", FALSE);
lv->require_input =
rofi_theme_get_boolean(WIDGET(lv), "require-input", FALSE);
lv->type = rofi_theme_get_orientation(WIDGET(lv), "layout",
ROFI_ORIENTATION_VERTICAL);
if (lv->type == LISTVIEW) {
listview_set_show_scrollbar(
lv, rofi_theme_get_boolean(WIDGET(lv), "scrollbar", FALSE));
} else {
listview_set_show_scrollbar(lv, FALSE);
}
return lv;
}
/**
* Navigation commands.
*/
static void listview_nav_up_int(listview *lv) {
if (lv == NULL) {
return;
}
if (lv->req_elements == 0 || (lv->selected == 0 && !lv->cycle)) {
return;
}
if (lv->selected == 0) {
lv->selected = lv->req_elements;
}
lv->selected--;
lv->barview.direction = RIGHT_TO_LEFT;
if (lv->sc_callback) {
lv->sc_callback(lv, lv->selected, lv->sc_udata);
}
widget_queue_redraw(WIDGET(lv));
}
static void listview_nav_down_int(listview *lv) {
if (lv == NULL) {
return;
}
if (lv->req_elements == 0 ||
(lv->selected == (lv->req_elements - 1) && !lv->cycle)) {
return;
}
lv->selected = lv->selected < lv->req_elements - 1
? MIN(lv->req_elements - 1, lv->selected + 1)
: 0;
lv->barview.direction = LEFT_TO_RIGHT;
if (lv->sc_callback) {
lv->sc_callback(lv, lv->selected, lv->sc_udata);
}
widget_queue_redraw(WIDGET(lv));
}
void listview_nav_next(listview *lv) {
if (lv == NULL) {
return;
}
listview_nav_down_int(lv);
}
void listview_nav_prev(listview *lv) {
if (lv == NULL) {
return;
}
listview_nav_up_int(lv);
}
static void listview_nav_column_left_int(listview *lv) {
if (lv->selected >= lv->cur_columns) {
lv->selected -= lv->cur_columns;
if (lv->sc_callback) {
lv->sc_callback(lv, lv->selected, lv->sc_udata);
}
widget_queue_redraw(WIDGET(lv));
}
}
static void listview_nav_column_right_int(listview *lv) {
if ((lv->selected + lv->cur_columns) < lv->req_elements) {
lv->selected += lv->cur_columns;
if (lv->sc_callback) {
lv->sc_callback(lv, lv->selected, lv->sc_udata);
}
widget_queue_redraw(WIDGET(lv));
}
}
void listview_nav_up(listview *lv) {
if (lv == NULL) {
return;
}
if (lv->pack_direction == ROFI_ORIENTATION_HORIZONTAL) {
if (lv->reverse) {
listview_nav_column_right_int(lv);
} else {
listview_nav_column_left_int(lv);
}
return;
}
if (lv->reverse) {
listview_nav_down_int(lv);
} else {
listview_nav_up_int(lv);
}
}
void listview_nav_down(listview *lv) {
if (lv == NULL) {
return;
}
if (lv->pack_direction == ROFI_ORIENTATION_HORIZONTAL) {
if (lv->reverse) {
listview_nav_column_left_int(lv);
} else {
listview_nav_column_right_int(lv);
}
return;
}
if (lv->reverse) {
listview_nav_up_int(lv);
} else {
listview_nav_down_int(lv);
}
}
void listview_nav_left(listview *lv) {
if (lv == NULL) {
return;
}
if (lv->max_rows == 0) {
return;
}
if (lv->pack_direction == ROFI_ORIENTATION_HORIZONTAL) {
listview_nav_up_int(lv);
return;
}
if (lv->type == BARVIEW) {
listview_nav_up_int(lv);
return;
}
if (lv->selected >= lv->max_rows) {
lv->selected -= lv->max_rows;
if (lv->sc_callback) {
lv->sc_callback(lv, lv->selected, lv->sc_udata);
}
widget_queue_redraw(WIDGET(lv));
}
}
void listview_nav_right(listview *lv) {
if (lv == NULL) {
return;
}
if (lv->max_rows == 0) {
return;
}
if (lv->pack_direction == ROFI_ORIENTATION_HORIZONTAL) {
listview_nav_down_int(lv);
return;
}
if (lv->type == BARVIEW) {
listview_nav_down_int(lv);
return;
}
if ((lv->selected + lv->max_rows) < lv->req_elements) {
lv->selected += lv->max_rows;
if (lv->sc_callback) {
lv->sc_callback(lv, lv->selected, lv->sc_udata);
}
widget_queue_redraw(WIDGET(lv));
} else if (lv->selected < (lv->req_elements - 1)) {
// We do not want to move to last item, UNLESS the last column is only
// partially filled, then we still want to move column and select last
// entry. First check the column we are currently in.
int col = lv->selected / lv->max_rows;
// Check total number of columns.
int ncol = lv->req_elements / lv->max_rows;
// If there is an extra column, move.
if (col != ncol) {
lv->selected = lv->req_elements - 1;
if (lv->sc_callback) {
lv->sc_callback(lv, lv->selected, lv->sc_udata);
}
widget_queue_redraw(WIDGET(lv));
}
}
}
static void listview_nav_page_prev_int(listview *lv) {
if (lv == NULL) {
return;
}
if (lv->type == BARVIEW) {
if (lv->last_offset == 0) {
lv->selected = 0;
} else {
lv->selected = lv->last_offset - 1;
}
lv->barview.direction = RIGHT_TO_LEFT;
widget_queue_redraw(WIDGET(lv));
return;
}
if (lv->selected < lv->max_elements) {
lv->selected = 0;
} else {
lv->selected -= (lv->max_elements);
}
if (lv->sc_callback) {
lv->sc_callback(lv, lv->selected, lv->sc_udata);
}
widget_queue_redraw(WIDGET(lv));
}
static void listview_nav_page_next_int(listview *lv) {
if (lv == NULL) {
return;
}
if (lv->req_elements == 0) {
return;
}
if (lv->type == BARVIEW) {
unsigned int new = lv->last_offset + lv->barview.cur_visible;
lv->selected = MIN(new, lv->req_elements - 1);
lv->barview.direction = LEFT_TO_RIGHT;
if (lv->sc_callback) {
lv->sc_callback(lv, lv->selected, lv->sc_udata);
}
widget_queue_redraw(WIDGET(lv));
return;
}
lv->selected += (lv->max_elements);
if (lv->selected >= lv->req_elements) {
lv->selected = lv->req_elements - 1;
}
if (lv->sc_callback) {
lv->sc_callback(lv, lv->selected, lv->sc_udata);
}
widget_queue_redraw(WIDGET(lv));
}
void listview_nav_page_prev(listview *lv) {
if (lv == NULL) {
return;
}
if (lv->reverse) {
listview_nav_page_next_int(lv);
} else {
listview_nav_page_prev_int(lv);
}
}
void listview_nav_page_next(listview *lv) {
if (lv == NULL) {
return;
}
if (lv->reverse) {
listview_nav_page_prev_int(lv);
} else {
listview_nav_page_next_int(lv);
}
}
static int listview_get_desired_height(widget *wid,
G_GNUC_UNUSED const int width) {
listview *lv = (listview *)wid;
if (lv == NULL || lv->widget.enabled == FALSE) {
return 0;
}
int spacing = distance_get_pixel(lv->spacing, ROFI_ORIENTATION_VERTICAL);
int h = lv->menu_lines;
if (!(lv->fixed_num_lines)) {
if (lv->dynamic) {
h = MIN(lv->menu_lines, lv->req_elements);
} else {
h = MIN(lv->menu_lines, lv->max_displayed_lines);
}
}
if (lv->type == BARVIEW) {
h = MIN(h, 1);
}
if (h == 0) {
if (lv->dynamic && !lv->fixed_num_lines) {
// Hide widget fully.
return 0;
}
return widget_padding_get_padding_height(WIDGET(lv));
}
int height = widget_padding_get_padding_height(WIDGET(lv));
height += h * (lv->element_height + spacing) - spacing;
return height;
}
void listview_set_show_scrollbar(listview *lv, gboolean enabled) {
if (lv) {
if (enabled) {
widget_enable(WIDGET(lv->scrollbar));
} else {
widget_disable(WIDGET(lv->scrollbar));
}
listview_recompute_elements(lv);
}
}
void listview_set_scroll_type(listview *lv, ScrollType type) {
if (lv) {
lv->scroll_type = type;
}
}
void listview_set_mouse_activated_cb(listview *lv,
listview_mouse_activated_cb cb,
void *udata) {
if (lv) {
lv->mouse_activated = cb;
lv->mouse_activated_data = udata;
}
}
void listview_set_max_lines(listview *lv, unsigned int max_lines) {
if (lv) {
lv->max_displayed_lines = max_lines;
}
}
gboolean listview_get_fixed_num_lines(listview *lv) {
if (lv) {
return lv->fixed_num_lines;
}
return FALSE;
}
void listview_set_fixed_num_lines(listview *lv) {
if (lv) {
lv->fixed_num_lines = TRUE;
}
}
void listview_set_ellipsize(listview *lv, PangoEllipsizeMode mode) {
if (lv) {
lv->emode = mode;
for (unsigned int i = 0; i < lv->cur_elements; i++) {
textbox_set_ellipsize(lv->boxes[i].textbox, lv->emode);
}
}
}
void listview_toggle_ellipsizing(listview *lv) {
if (lv) {
PangoEllipsizeMode mode = lv->emode;
if (mode == PANGO_ELLIPSIZE_START) {
mode = PANGO_ELLIPSIZE_MIDDLE;
} else if (mode == PANGO_ELLIPSIZE_MIDDLE) {
mode = PANGO_ELLIPSIZE_END;
} else if (mode == PANGO_ELLIPSIZE_END) {
mode = PANGO_ELLIPSIZE_START;
}
lv->emode = mode;
for (unsigned int i = 0; i < lv->cur_elements; i++) {
textbox_set_ellipsize(lv->boxes[i].textbox, mode);
}
}
}
void listview_set_filtered(listview *lv, gboolean filtered) {
if (lv) {
lv->filtered = filtered;
}
}
void listview_set_selection_changed_callback(
listview *lv, listview_selection_changed_callback cb, void *udata) {
lv->sc_callback = cb;
lv->sc_udata = udata;
}