mirror of
https://github.com/davatorium/rofi.git
synced 2024-11-18 13:54:36 -05:00
Add sidebar modi to ROFI.
This commit is contained in:
parent
03f919a0fb
commit
ab82941b5e
6 changed files with 104 additions and 24 deletions
|
@ -80,6 +80,7 @@ Settings config = {
|
||||||
.disable_history = FALSE,
|
.disable_history = FALSE,
|
||||||
.levenshtein_sort = FALSE,
|
.levenshtein_sort = FALSE,
|
||||||
.separator = '\n',
|
.separator = '\n',
|
||||||
.element_height = 1
|
.element_height = 1,
|
||||||
|
.sidebar_mode = FALSE
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
|
@ -234,6 +234,13 @@ The default key combinations are:
|
||||||
|
|
||||||
rofi -hmode -padding 0
|
rofi -hmode -padding 0
|
||||||
|
|
||||||
|
`-sidebar-mode`
|
||||||
|
|
||||||
|
Go into side-bar mode, it will show list of modi at the bottom.
|
||||||
|
To show sidebar use:
|
||||||
|
|
||||||
|
rofi -rnow -sidebar-mode -lines 0
|
||||||
|
|
||||||
### Pattern setting
|
### Pattern setting
|
||||||
|
|
||||||
`-terminal`
|
`-terminal`
|
||||||
|
|
10
doc/rofi.1
10
doc/rofi.1
|
@ -305,6 +305,16 @@ To make rofi look like dmenu:
|
||||||
rofi \-hmode \-padding 0
|
rofi \-hmode \-padding 0
|
||||||
.fi
|
.fi
|
||||||
.RE
|
.RE
|
||||||
|
.PP
|
||||||
|
\fB\fC\-sidebar\-mode\fR
|
||||||
|
.PP
|
||||||
|
.RS
|
||||||
|
.nf
|
||||||
|
Go into side\-bar mode, it will show list of modi at the bottom.
|
||||||
|
To show sidebar use:
|
||||||
|
rofi \-rnow \-sidebar\-mode \-lines 0
|
||||||
|
.fi
|
||||||
|
.RE
|
||||||
.SS Pattern setting
|
.SS Pattern setting
|
||||||
.PP
|
.PP
|
||||||
\fB\fC\-terminal\fR
|
\fB\fC\-terminal\fR
|
||||||
|
|
|
@ -131,6 +131,7 @@ typedef struct _Settings
|
||||||
unsigned int levenshtein_sort;
|
unsigned int levenshtein_sort;
|
||||||
char separator;
|
char separator;
|
||||||
int element_height;
|
int element_height;
|
||||||
|
int sidebar_mode;
|
||||||
} Settings;
|
} Settings;
|
||||||
|
|
||||||
extern Settings config;
|
extern Settings config;
|
||||||
|
|
102
source/rofi.c
102
source/rofi.c
|
@ -65,6 +65,7 @@
|
||||||
#include "script-dialog.h"
|
#include "script-dialog.h"
|
||||||
|
|
||||||
#include "xrmoptions.h"
|
#include "xrmoptions.h"
|
||||||
|
#include "textbox.h"
|
||||||
|
|
||||||
#define LINE_MARGIN 3
|
#define LINE_MARGIN 3
|
||||||
|
|
||||||
|
@ -86,10 +87,12 @@ typedef struct _Switcher
|
||||||
char name[32];
|
char name[32];
|
||||||
switcher_callback cb;
|
switcher_callback cb;
|
||||||
void *cb_data;
|
void *cb_data;
|
||||||
|
textbox *tb;
|
||||||
} Switcher;
|
} Switcher;
|
||||||
|
|
||||||
Switcher *switchers = NULL;
|
Switcher *switchers = NULL;
|
||||||
unsigned int num_switchers = 0;
|
unsigned int num_switchers = 0;
|
||||||
|
unsigned int curr_switcher = 0;
|
||||||
|
|
||||||
|
|
||||||
void window_set_opacity ( Display *display, Window box, unsigned int opacity );
|
void window_set_opacity ( Display *display, Window box, unsigned int opacity );
|
||||||
|
@ -774,7 +777,6 @@ KeySym sshdialog_keysym;
|
||||||
Window main_window = None;
|
Window main_window = None;
|
||||||
GC gc = NULL;
|
GC gc = NULL;
|
||||||
|
|
||||||
#include "textbox.h"
|
|
||||||
|
|
||||||
void menu_hide_arrow_text ( int filtered_lines, int selected, int max_elements,
|
void menu_hide_arrow_text ( int filtered_lines, int selected, int max_elements,
|
||||||
textbox *arrowbox_top, textbox *arrowbox_bottom )
|
textbox *arrowbox_top, textbox *arrowbox_bottom )
|
||||||
|
@ -956,6 +958,7 @@ Window create_window ( Display *display )
|
||||||
|
|
||||||
typedef struct MenuState
|
typedef struct MenuState
|
||||||
{
|
{
|
||||||
|
unsigned int menu_lines;
|
||||||
unsigned int max_elements;
|
unsigned int max_elements;
|
||||||
unsigned int max_rows;
|
unsigned int max_rows;
|
||||||
unsigned int columns;
|
unsigned int columns;
|
||||||
|
@ -1084,24 +1087,24 @@ static void calculate_window_position ( MenuState *state, const workarea *mon )
|
||||||
static void menu_calculate_rows_columns ( MenuState *state )
|
static void menu_calculate_rows_columns ( MenuState *state )
|
||||||
{
|
{
|
||||||
state->columns = config.menu_columns;
|
state->columns = config.menu_columns;
|
||||||
state->max_elements = MIN ( config.menu_lines * state->columns, state->num_lines );
|
state->max_elements = MIN ( state->menu_lines * state->columns, state->num_lines );
|
||||||
|
|
||||||
// Calculate the number or rows. We do this by getting the num_lines rounded up to X columns
|
// Calculate the number or rows. We do this by getting the num_lines rounded up to X columns
|
||||||
// (num elements is better name) then dividing by columns.
|
// (num elements is better name) then dividing by columns.
|
||||||
state->max_rows = MIN ( config.menu_lines,
|
state->max_rows = MIN ( state->menu_lines,
|
||||||
(unsigned int) (
|
(unsigned int) (
|
||||||
( state->num_lines + ( state->columns - state->num_lines % state->columns ) %
|
( state->num_lines + ( state->columns - state->num_lines % state->columns ) %
|
||||||
state->columns ) / ( state->columns )
|
state->columns ) / ( state->columns )
|
||||||
) );
|
) );
|
||||||
|
|
||||||
if ( config.fixed_num_lines == TRUE ) {
|
if ( config.fixed_num_lines == TRUE ) {
|
||||||
state->max_elements = config.menu_lines * state->columns;
|
state->max_elements = state->menu_lines * state->columns;
|
||||||
state->max_rows = config.menu_lines;
|
state->max_rows = state->menu_lines;
|
||||||
// If it would fit in one column, only use one column.
|
// If it would fit in one column, only use one column.
|
||||||
if ( state->num_lines < state->max_elements ) {
|
if ( state->num_lines < state->max_elements ) {
|
||||||
state->columns = ( state->num_lines + ( state->max_rows - state->num_lines % state->max_rows ) %
|
state->columns = ( state->num_lines + ( state->max_rows - state->num_lines % state->max_rows ) %
|
||||||
state->max_rows ) / state->max_rows;
|
state->max_rows ) / state->max_rows;
|
||||||
state->max_elements = config.menu_lines * state->columns;
|
state->max_elements = state->menu_lines * state->columns;
|
||||||
}
|
}
|
||||||
// Sanitize.
|
// Sanitize.
|
||||||
if ( state->columns == 0 ) {
|
if ( state->columns == 0 ) {
|
||||||
|
@ -1418,10 +1421,26 @@ static void menu_update ( MenuState *state )
|
||||||
if ( config.hmode == FALSE ) {
|
if ( config.hmode == FALSE ) {
|
||||||
int line_height = textbox_get_height ( state->text );
|
int line_height = textbox_get_height ( state->text );
|
||||||
XDrawLine ( display, main_window, gc, ( config.padding ),
|
XDrawLine ( display, main_window, gc, ( config.padding ),
|
||||||
line_height + ( config.padding ) + ( LINE_MARGIN - 2 ) / 2,
|
line_height + ( config.padding ) + ( LINE_MARGIN ) / 2,
|
||||||
state->w - ( ( config.padding ) ) - 1,
|
state->w - ( ( config.padding ) ) - 1,
|
||||||
line_height + ( config.padding ) + ( LINE_MARGIN - 2 ) / 2 );
|
line_height + ( config.padding ) + ( LINE_MARGIN ) / 2 );
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if ( config.sidebar_mode == TRUE ) {
|
||||||
|
int line_height = textbox_get_height ( state->text );
|
||||||
|
XDrawLine ( display, main_window, gc,
|
||||||
|
( config.padding ),
|
||||||
|
state->h - line_height - ( config.padding ) + LINE_MARGIN - 1,
|
||||||
|
state->w - ( ( config.padding ) ) - 1,
|
||||||
|
state->h - line_height - ( config.padding ) + LINE_MARGIN - 1 );
|
||||||
|
if ( config.sidebar_mode == TRUE ) {
|
||||||
|
for ( int j = 0; j < num_switchers; j++ ) {
|
||||||
|
textbox_draw ( switchers[j].tb );
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
state->update = FALSE;
|
state->update = FALSE;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1496,21 +1515,16 @@ MenuReturn menu ( char **lines, unsigned int num_lines, char **input, char *prom
|
||||||
unsigned int i;
|
unsigned int i;
|
||||||
workarea mon;
|
workarea mon;
|
||||||
|
|
||||||
menu_calculate_rows_columns ( &state );
|
|
||||||
|
|
||||||
// Get active monitor size.
|
|
||||||
monitor_active ( &mon );
|
|
||||||
|
|
||||||
// main window isn't explicitly destroyed in case we switch modes. Reusing it prevents flicker
|
// main window isn't explicitly destroyed in case we switch modes. Reusing it prevents flicker
|
||||||
XWindowAttributes attr;
|
XWindowAttributes attr;
|
||||||
if ( main_window == None || XGetWindowAttributes ( display, main_window, &attr ) == 0 ) {
|
if ( main_window == None || XGetWindowAttributes ( display, main_window, &attr ) == 0 ) {
|
||||||
main_window = create_window ( display );
|
main_window = create_window ( display );
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
menu_calculate_window_and_element_width ( &state, &mon );
|
menu_calculate_window_and_element_width ( &state, &mon );
|
||||||
|
|
||||||
// search text input
|
// search text input
|
||||||
|
// we need this at this point so we can get height.
|
||||||
|
|
||||||
state.prompt_tb = textbox_create ( main_window, TB_AUTOHEIGHT | TB_AUTOWIDTH,
|
state.prompt_tb = textbox_create ( main_window, TB_AUTOHEIGHT | TB_AUTOWIDTH,
|
||||||
( config.padding ), ( config.padding ),
|
( config.padding ), ( config.padding ),
|
||||||
|
@ -1529,8 +1543,27 @@ MenuReturn menu ( char **lines, unsigned int num_lines, char **input, char *prom
|
||||||
textbox_show ( state.text );
|
textbox_show ( state.text );
|
||||||
textbox_show ( state.prompt_tb );
|
textbox_show ( state.prompt_tb );
|
||||||
|
|
||||||
|
// Height of a row.
|
||||||
int line_height = textbox_get_height ( state.text );
|
int line_height = textbox_get_height ( state.text );
|
||||||
|
// Get active monitor size.
|
||||||
|
monitor_active ( &mon );
|
||||||
|
if ( config.menu_lines == 0 ) {
|
||||||
|
// Autosize it.
|
||||||
|
int h = mon.h - config.padding * 2 - LINE_MARGIN;
|
||||||
|
int r = ( h ) / ( line_height * config.element_height ) - 1 - config.sidebar_mode;
|
||||||
|
// HACK todo fix this.
|
||||||
|
state.menu_lines = r;
|
||||||
|
menu_calculate_rows_columns ( &state );
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
state.menu_lines = config.menu_lines;
|
||||||
|
menu_calculate_rows_columns ( &state );
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
menu_calculate_window_and_element_width ( &state, &mon );
|
||||||
|
|
||||||
int element_height = line_height * config.element_height;
|
int element_height = line_height * config.element_height;
|
||||||
// filtered list display
|
// filtered list display
|
||||||
state.boxes = g_malloc0_n ( state.max_elements, sizeof ( textbox* ) );
|
state.boxes = g_malloc0_n ( state.max_elements, sizeof ( textbox* ) );
|
||||||
|
@ -1595,12 +1628,32 @@ MenuReturn menu ( char **lines, unsigned int num_lines, char **input, char *prom
|
||||||
if ( config.hmode == TRUE ) {
|
if ( config.hmode == TRUE ) {
|
||||||
state.h = line_height + ( config.padding ) * 2;
|
state.h = line_height + ( config.padding ) * 2;
|
||||||
}
|
}
|
||||||
|
// Add entry
|
||||||
|
if ( config.sidebar_mode == TRUE ) {
|
||||||
|
state.h += line_height + LINE_MARGIN;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Sidebar mode.
|
||||||
|
if ( config.menu_lines == 0 ) {
|
||||||
|
state.h = mon.h - config.menu_bw * 2;
|
||||||
|
}
|
||||||
|
|
||||||
// Move the window to the correct x,y position.
|
// Move the window to the correct x,y position.
|
||||||
calculate_window_position ( &state, &mon );
|
calculate_window_position ( &state, &mon );
|
||||||
XMoveResizeWindow ( display, main_window, state.x, state.y, state.w, state.h );
|
|
||||||
|
if ( config.sidebar_mode == TRUE ) {
|
||||||
|
int line_height = textbox_get_height ( state.text );
|
||||||
|
int width = ( state.w - ( 2 * ( config.padding ) ) ) / num_switchers;
|
||||||
|
for ( int j = 0; j < num_switchers; j++ ) {
|
||||||
|
switchers[j].tb = textbox_create ( main_window, TB_CENTER,
|
||||||
|
config.padding + j * width, state.h - line_height - config.padding + LINE_MARGIN,
|
||||||
|
width, line_height, ( j == curr_switcher ) ? HIGHLIGHT : NORMAL, switchers[j].name );
|
||||||
|
textbox_show ( switchers[j].tb );
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// Display it.
|
// Display it.
|
||||||
|
XMoveResizeWindow ( display, main_window, state.x, state.y, state.w, state.h );
|
||||||
XMapRaised ( display, main_window );
|
XMapRaised ( display, main_window );
|
||||||
|
|
||||||
// if grabbing keyboard failed, fall through
|
// if grabbing keyboard failed, fall through
|
||||||
|
@ -1792,7 +1845,6 @@ void error_dialog ( char *msg )
|
||||||
|
|
||||||
// Move the window to the correct x,y position.
|
// Move the window to the correct x,y position.
|
||||||
calculate_window_position ( &state, &mon );
|
calculate_window_position ( &state, &mon );
|
||||||
XMoveResizeWindow ( display, main_window, state.x, state.y, state.w, state.h );
|
|
||||||
|
|
||||||
// Display it.
|
// Display it.
|
||||||
XMapRaised ( display, main_window );
|
XMapRaised ( display, main_window );
|
||||||
|
@ -2023,6 +2075,7 @@ static void run_switcher ( int do_fork, SwitcherMode mode )
|
||||||
do {
|
do {
|
||||||
SwitcherMode retv;
|
SwitcherMode retv;
|
||||||
|
|
||||||
|
curr_switcher = mode;
|
||||||
retv = switchers[mode].cb ( &input, switchers[mode].cb_data );
|
retv = switchers[mode].cb ( &input, switchers[mode].cb_data );
|
||||||
|
|
||||||
// Find next enabled
|
// Find next enabled
|
||||||
|
@ -2259,6 +2312,10 @@ static void parse_cmd_options ( int argc, char ** argv )
|
||||||
|
|
||||||
|
|
||||||
find_arg_int ( argc, argv, "-eh", &( config.element_height ) );
|
find_arg_int ( argc, argv, "-eh", &( config.element_height ) );
|
||||||
|
|
||||||
|
if ( find_arg ( argc, argv, "-sidebar-mode" ) >= 0 ) {
|
||||||
|
config.sidebar_mode = TRUE;
|
||||||
|
}
|
||||||
// Dump.
|
// Dump.
|
||||||
if ( find_arg ( argc, argv, "-dump-xresources" ) >= 0 ) {
|
if ( find_arg ( argc, argv, "-dump-xresources" ) >= 0 ) {
|
||||||
xresource_dump ();
|
xresource_dump ();
|
||||||
|
@ -2315,10 +2372,10 @@ static void cleanup ()
|
||||||
*/
|
*/
|
||||||
static void config_sanity_check ( void )
|
static void config_sanity_check ( void )
|
||||||
{
|
{
|
||||||
if ( config.menu_lines == 0 ) {
|
// if ( config.menu_lines == 0 ) {
|
||||||
fprintf ( stderr, "config.menu_lines is invalid. You need at least one visible line.\n" );
|
// fprintf ( stderr, "config.menu_lines is invalid. You need at least one visible line.\n" );
|
||||||
exit ( 1 );
|
// exit ( 1 );
|
||||||
}
|
// }
|
||||||
if ( config.element_height < 1 ) {
|
if ( config.element_height < 1 ) {
|
||||||
fprintf ( stderr, "config.element_height is invalid. It needs to be atleast 1 line high.\n" );
|
fprintf ( stderr, "config.element_height is invalid. It needs to be atleast 1 line high.\n" );
|
||||||
exit ( 1 );
|
exit ( 1 );
|
||||||
|
@ -2514,6 +2571,9 @@ int main ( int argc, char *argv[] )
|
||||||
// flags to run immediately and exit
|
// flags to run immediately and exit
|
||||||
char *sname = NULL;
|
char *sname = NULL;
|
||||||
if ( find_arg ( argc, argv, "-dmenu" ) >= 0 || strcmp ( argv[0], "dmenu" ) == 0 ) {
|
if ( find_arg ( argc, argv, "-dmenu" ) >= 0 || strcmp ( argv[0], "dmenu" ) == 0 ) {
|
||||||
|
// force off sidebar mode:
|
||||||
|
config.sidebar_mode = FALSE;
|
||||||
|
// Check prompt
|
||||||
find_arg_str ( argc, argv, "-p", &dmenu_prompt );
|
find_arg_str ( argc, argv, "-p", &dmenu_prompt );
|
||||||
int retv = run_dmenu ();
|
int retv = run_dmenu ();
|
||||||
// User cancelled the operation.
|
// User cancelled the operation.
|
||||||
|
|
|
@ -108,7 +108,8 @@ static XrmOption xrmOptions[] = {
|
||||||
/* Key bindings */
|
/* Key bindings */
|
||||||
{ xrm_String, "key", { .str = &config.window_key }, NULL },
|
{ xrm_String, "key", { .str = &config.window_key }, NULL },
|
||||||
{ xrm_String, "rkey", { .str = &config.run_key }, NULL },
|
{ xrm_String, "rkey", { .str = &config.run_key }, NULL },
|
||||||
{ xrm_String, "skey", { .str = &config.ssh_key }, NULL }
|
{ xrm_String, "skey", { .str = &config.ssh_key }, NULL },
|
||||||
|
{ xrm_Boolean, "sidebar-mode", { .num = &config.sidebar_mode }, NULL }
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue