mirror of
https://github.com/lbonn/rofi
synced 2024-11-27 22:30:37 +00:00
2944 lines
94 KiB
C
2944 lines
94 KiB
C
/**
|
|
* rofi
|
|
*
|
|
* MIT/X11 License
|
|
* Copyright (c) 2012 Sean Pringle <sean.pringle@gmail.com>
|
|
* Modified 2013-2014 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 <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
#include <unistd.h>
|
|
#include <stdint.h>
|
|
#include <signal.h>
|
|
#include <fcntl.h>
|
|
#include <errno.h>
|
|
#include <time.h>
|
|
#include <X11/X.h>
|
|
#include <X11/Xatom.h>
|
|
#include <X11/Xlib.h>
|
|
#include <X11/Xmd.h>
|
|
#include <X11/Xutil.h>
|
|
#include <X11/Xproto.h>
|
|
#include <X11/keysym.h>
|
|
#include <X11/XKBlib.h>
|
|
#include <X11/extensions/Xinerama.h>
|
|
|
|
#include <sys/wait.h>
|
|
#include <sys/file.h>
|
|
|
|
#ifdef HAVE_I3_IPC_H
|
|
#include <sys/types.h>
|
|
#include <sys/socket.h>
|
|
#include <linux/un.h>
|
|
#include <i3/ipc.h>
|
|
#endif
|
|
|
|
#include "helper.h"
|
|
#include "rofi.h"
|
|
|
|
#include "run-dialog.h"
|
|
#include "ssh-dialog.h"
|
|
#include "dmenu-dialog.h"
|
|
#include "script-dialog.h"
|
|
|
|
#include "xrmoptions.h"
|
|
#include "textbox.h"
|
|
|
|
#define LINE_MARGIN 3
|
|
|
|
#ifdef HAVE_I3_IPC_H
|
|
// This setting is no longer user configurable, but partial to this file:
|
|
int config_i3_mode = 0;
|
|
// Path to HAVE_I3_IPC_H socket.
|
|
char *i3_socket_path = NULL;
|
|
#endif
|
|
|
|
char *pidfile = NULL;
|
|
|
|
const char *cache_dir = NULL;
|
|
unsigned int NumlockMask = 0;
|
|
Display *display = NULL;
|
|
char *display_str = NULL;
|
|
|
|
static int ( *xerror )( Display *, XErrorEvent * );
|
|
|
|
#define ATOM_ENUM( x ) x
|
|
#define ATOM_CHAR( x ) # x
|
|
|
|
#define EWMH_ATOMS( X ) \
|
|
X ( _NET_CLIENT_LIST_STACKING ), \
|
|
X ( _NET_NUMBER_OF_DESKTOPS ), \
|
|
X ( _NET_CURRENT_DESKTOP ), \
|
|
X ( _NET_ACTIVE_WINDOW ), \
|
|
X ( _NET_WM_NAME ), \
|
|
X ( _NET_WM_STATE ), \
|
|
X ( _NET_WM_STATE_SKIP_TASKBAR ), \
|
|
X ( _NET_WM_STATE_SKIP_PAGER ), \
|
|
X ( _NET_WM_STATE_ABOVE ), \
|
|
X ( _NET_WM_DESKTOP ), \
|
|
X ( I3_SOCKET_PATH ), \
|
|
X ( CLIPBOARD ), \
|
|
X ( UTF8_STRING ), \
|
|
X ( _NET_WM_WINDOW_OPACITY )
|
|
|
|
enum { EWMH_ATOMS ( ATOM_ENUM ), NUM_NETATOMS };
|
|
const char *netatom_names[] = { EWMH_ATOMS ( ATOM_CHAR ) };
|
|
Atom netatoms[NUM_NETATOMS];
|
|
|
|
/**
|
|
* Structure defining a switcher.
|
|
* It consists of a name, callback and if enabled
|
|
* a textbox for the sidebar-mode.
|
|
*/
|
|
typedef struct _Switcher
|
|
{
|
|
// Name (max 31 char long)
|
|
char name[32];
|
|
// Switcher callback.
|
|
switcher_callback cb;
|
|
// Callback data.
|
|
void *cb_data;
|
|
// Textbox used in the sidebar-mode.
|
|
textbox *tb;
|
|
} Switcher;
|
|
|
|
// Array of switchers.
|
|
Switcher *switchers = NULL;
|
|
// Number of switchers.
|
|
unsigned int num_switchers = 0;
|
|
// Current selected switcher.
|
|
unsigned int curr_switcher = 0;
|
|
|
|
|
|
/**
|
|
* @param name Name of the switcher to lookup.
|
|
*
|
|
* Find the index of the switcher with name.
|
|
*
|
|
* @returns index of the switcher in switchers, -1 if not found.
|
|
*/
|
|
static int switcher_get ( const char *name )
|
|
{
|
|
for ( unsigned int i = 0; i < num_switchers; i++ ) {
|
|
if ( strcmp ( switchers[i].name, name ) == 0 ) {
|
|
return i;
|
|
}
|
|
}
|
|
return -1;
|
|
}
|
|
|
|
|
|
#ifdef HAVE_I3_IPC_H
|
|
/**
|
|
* @param socket_path The I3 IPC socket.
|
|
* @param id The window to focus on.
|
|
*
|
|
* If we want to switch windows in I3, we use I3 IPC mode.
|
|
* This works more better then sending messages via X11.
|
|
* Hopefully at some point, I3 gets fixed and this is not needed.
|
|
* This function takes the path to the i3 IPC socket, and the XID of the window.
|
|
*/
|
|
static void focus_window_i3 ( const char *socket_path, Window id )
|
|
{
|
|
i3_ipc_header_t head;
|
|
char command[128];
|
|
int s, len;
|
|
ssize_t t;
|
|
struct sockaddr_un remote;
|
|
|
|
if ( strlen ( socket_path ) > UNIX_PATH_MAX ) {
|
|
fprintf ( stderr, "Socket path is to long. %zd > %d\n", strlen ( socket_path ), UNIX_PATH_MAX );
|
|
return;
|
|
}
|
|
|
|
if ( ( s = socket ( AF_UNIX, SOCK_STREAM, 0 ) ) == -1 ) {
|
|
fprintf ( stderr, "Failed to open connection to I3: %s\n", strerror ( errno ) );
|
|
return;
|
|
}
|
|
|
|
remote.sun_family = AF_UNIX;
|
|
strcpy ( remote.sun_path, socket_path );
|
|
len = strlen ( remote.sun_path ) + sizeof ( remote.sun_family );
|
|
|
|
if ( connect ( s, ( struct sockaddr * ) &remote, len ) == -1 ) {
|
|
fprintf ( stderr, "Failed to connect to I3 (%s): %s\n", socket_path, strerror ( errno ) );
|
|
close ( s );
|
|
return;
|
|
}
|
|
|
|
|
|
// Formulate command
|
|
snprintf ( command, 128, "[id=\"%lu\"] focus", id );
|
|
// Prepare header.
|
|
memcpy ( head.magic, I3_IPC_MAGIC, 6 );
|
|
head.size = strlen ( command );
|
|
head.type = I3_IPC_MESSAGE_TYPE_COMMAND;
|
|
// Send header.
|
|
t = send ( s, &head, sizeof ( i3_ipc_header_t ), 0 );
|
|
if ( t == -1 ) {
|
|
char *msg = g_strdup_printf ( "Failed to send message header to i3: %s\n", strerror ( errno ) );
|
|
error_dialog ( msg );
|
|
g_free ( msg );
|
|
close ( s );
|
|
return;
|
|
}
|
|
// Send message
|
|
t = send ( s, command, strlen ( command ), 0 );
|
|
if ( t == -1 ) {
|
|
char *msg = g_strdup_printf ( "Failed to send message body to i3: %s\n", strerror ( errno ) );
|
|
error_dialog ( msg );
|
|
g_free ( msg );
|
|
close ( s );
|
|
return;
|
|
}
|
|
// Receive header.
|
|
t = recv ( s, &head, sizeof ( head ), 0 );
|
|
|
|
if ( t == sizeof ( head ) ) {
|
|
t = recv ( s, command, head.size, 0 );
|
|
if ( t == head.size ) {
|
|
// Response.
|
|
}
|
|
}
|
|
|
|
close ( s );
|
|
}
|
|
#endif
|
|
|
|
void catch_exit ( __attribute__( ( unused ) ) int sig )
|
|
{
|
|
while ( 0 < waitpid ( -1, NULL, WNOHANG ) ) {
|
|
;
|
|
}
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
* @param d The display on witch the error occurred.
|
|
* @param ee The XErrorEvent
|
|
*
|
|
* X11 Error handler.
|
|
*/
|
|
static int display_oops ( Display *d, XErrorEvent *ee )
|
|
{
|
|
if ( ee->error_code == BadWindow
|
|
|| ( ee->request_code == X_GrabButton && ee->error_code == BadAccess )
|
|
|| ( ee->request_code == X_GrabKey && ee->error_code == BadAccess )
|
|
) {
|
|
return 0;
|
|
}
|
|
|
|
fprintf ( stderr, "error: request code=%d, error code=%d\n", ee->request_code, ee->error_code );
|
|
return xerror ( d, ee );
|
|
}
|
|
|
|
// usable space on a monitor
|
|
typedef struct
|
|
{
|
|
int x, y, w, h;
|
|
int l, r, t, b;
|
|
} workarea;
|
|
|
|
|
|
// window lists
|
|
typedef struct
|
|
{
|
|
Window *array;
|
|
void **data;
|
|
int len;
|
|
} winlist;
|
|
|
|
winlist *cache_client = NULL;
|
|
winlist *cache_xattr = NULL;
|
|
|
|
#define WINLIST 32
|
|
|
|
/**
|
|
* Create a window list, pre-seeded with WINLIST entries.
|
|
*
|
|
* @returns A new window list.
|
|
*/
|
|
winlist* winlist_new ()
|
|
{
|
|
winlist *l = g_malloc ( sizeof ( winlist ) );
|
|
l->len = 0;
|
|
l->array = g_malloc_n ( WINLIST + 1, sizeof ( Window ) );
|
|
l->data = g_malloc_n ( WINLIST + 1, sizeof ( void* ) );
|
|
return l;
|
|
}
|
|
|
|
/**
|
|
* @param l The winlist.
|
|
* @param w The window to add.
|
|
* @param d Data pointer.
|
|
*
|
|
* Add one entry. If Full, extend with WINLIST entries.
|
|
*
|
|
* @returns 0 if failed, 1 is successful.
|
|
*/
|
|
int winlist_append ( winlist *l, Window w, void *d )
|
|
{
|
|
if ( l->len > 0 && !( l->len % WINLIST ) ) {
|
|
l->array = g_realloc ( l->array, sizeof ( Window ) * ( l->len + WINLIST + 1 ) );
|
|
l->data = g_realloc ( l->data, sizeof ( void* ) * ( l->len + WINLIST + 1 ) );
|
|
}
|
|
// Make clang-check happy.
|
|
// TODO: make clang-check clear this should never be 0.
|
|
if ( l->data == NULL || l->array == NULL ) {
|
|
return 0;
|
|
}
|
|
|
|
l->data[l->len] = d;
|
|
l->array[l->len++] = w;
|
|
return l->len - 1;
|
|
}
|
|
|
|
/**
|
|
* @param l The winlist entry
|
|
*
|
|
* Empty winlist without free-ing
|
|
*/
|
|
void winlist_empty ( winlist *l )
|
|
{
|
|
while ( l->len > 0 ) {
|
|
g_free ( l->data[--( l->len )] );
|
|
}
|
|
}
|
|
|
|
/**
|
|
* @param l The winlist entry
|
|
*
|
|
* Free the winlist.
|
|
*/
|
|
void winlist_free ( winlist *l )
|
|
{
|
|
winlist_empty ( l );
|
|
g_free ( l->array );
|
|
g_free ( l->data );
|
|
g_free ( l );
|
|
}
|
|
|
|
/**
|
|
* @param l The winlist.
|
|
* @param w The window to find.
|
|
*
|
|
* Find the window in the list, and return the array entry.
|
|
*
|
|
* @returns -1 if failed, index is successful.
|
|
*/
|
|
int winlist_find ( winlist *l, Window w )
|
|
{
|
|
// iterate backwards. theory is: windows most often accessed will be
|
|
// nearer the end. testing with kcachegrind seems to support this...
|
|
int i;
|
|
|
|
for ( i = ( l->len - 1 ); i >= 0; i-- ) {
|
|
if ( l->array[i] == w ) {
|
|
return i;
|
|
}
|
|
}
|
|
|
|
return -1;
|
|
}
|
|
|
|
#define CLIENTTITLE 100
|
|
#define CLIENTCLASS 50
|
|
#define CLIENTNAME 50
|
|
#define CLIENTSTATE 10
|
|
#define CLIENTROLE 50
|
|
|
|
// a managable window
|
|
typedef struct
|
|
{
|
|
Window window, trans;
|
|
XWindowAttributes xattr;
|
|
char title[CLIENTTITLE];
|
|
char class[CLIENTCLASS];
|
|
char name[CLIENTNAME];
|
|
char role[CLIENTROLE];
|
|
int states;
|
|
Atom state[CLIENTSTATE];
|
|
workarea monitor;
|
|
int active;
|
|
} client;
|
|
|
|
|
|
unsigned int windows_modmask;
|
|
KeySym windows_keysym;
|
|
unsigned int rundialog_modmask;
|
|
KeySym rundialog_keysym;
|
|
unsigned int sshdialog_modmask;
|
|
KeySym sshdialog_keysym;
|
|
|
|
Window main_window = None;
|
|
GC gc = NULL;
|
|
|
|
|
|
Colormap map = None;
|
|
XVisualInfo vinfo;
|
|
int truecolor = FALSE;
|
|
|
|
static void create_visual_and_colormap ()
|
|
{
|
|
map = None;
|
|
// Try to create TrueColor map
|
|
if ( XMatchVisualInfo ( display, DefaultScreen ( display ), 32, TrueColor, &vinfo ) ) {
|
|
// Visual found, lets try to create map.
|
|
map = XCreateColormap ( display, DefaultRootWindow ( display ), vinfo.visual, AllocNone );
|
|
truecolor = TRUE;
|
|
}
|
|
// Failed to create map.
|
|
// Use the defaults then.
|
|
if ( map == None ) {
|
|
truecolor = FALSE;
|
|
// Two fields we use.
|
|
vinfo.visual = DefaultVisual ( display, DefaultScreen ( display ) );
|
|
vinfo.depth = DefaultDepth ( display, DefaultScreen ( display ) );
|
|
map = DefaultColormap ( display, DefaultScreen ( display ) );
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Allocate a pixel value for an X named color
|
|
*/
|
|
static unsigned int color_get ( Display *display, const char *const name )
|
|
{
|
|
XColor color = { 0, };
|
|
// Special format.
|
|
if ( strncmp ( name, "argb:", 5 ) == 0 ) {
|
|
color.pixel = strtoul ( &name[5], NULL, 16 );
|
|
color.red = ( ( color.pixel & 0x00FF0000 ) >> 16 ) * 255;
|
|
color.green = ( ( color.pixel & 0x0000FF00 ) >> 8 ) * 255;
|
|
color.blue = ( ( color.pixel & 0x000000FF ) ) * 255;
|
|
if ( !truecolor ) {
|
|
// This will drop alpha part.
|
|
return XAllocColor ( display, map, &color ) ? color.pixel : None;
|
|
}
|
|
return color.pixel;
|
|
}
|
|
else {
|
|
return XAllocNamedColor ( display, map, name, &color, &color ) ? color.pixel : None;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* @param x The x position of the mouse [out]
|
|
* @param y The y position of the mouse [out]
|
|
*
|
|
* find mouse pointer location
|
|
*
|
|
* @returns 1 when found
|
|
*/
|
|
static int pointer_get ( Display *display, Window root, int *x, int *y )
|
|
{
|
|
*x = 0;
|
|
*y = 0;
|
|
Window rr, cr;
|
|
int rxr, ryr, wxr, wyr;
|
|
unsigned int mr;
|
|
|
|
if ( XQueryPointer ( display, root, &rr, &cr, &rxr, &ryr, &wxr, &wyr, &mr ) ) {
|
|
*x = rxr;
|
|
*y = ryr;
|
|
return 1;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
typedef enum _MainLoopEvent
|
|
{
|
|
ML_XEVENT,
|
|
ML_TIMEOUT
|
|
} MainLoopEvent;
|
|
static inline MainLoopEvent wait_for_xevent_or_timeout ( Display *display, int x11_fd )
|
|
{
|
|
// Check if events are pending.
|
|
if ( XPending ( display ) ) {
|
|
return ML_XEVENT;
|
|
}
|
|
// If not, wait for timeout.
|
|
struct timeval tv;
|
|
fd_set in_fds;
|
|
// Create a File Description Set containing x11_fd
|
|
FD_ZERO ( &in_fds );
|
|
FD_SET ( x11_fd, &in_fds );
|
|
|
|
// Set our timer. 200ms is a decent delay
|
|
tv.tv_usec = 200000;
|
|
tv.tv_sec = 0;
|
|
// Wait for X Event or a Timer
|
|
if ( select ( x11_fd + 1, &in_fds, 0, 0, &tv ) == 0 ) {
|
|
return ML_TIMEOUT;
|
|
}
|
|
return ML_XEVENT;
|
|
}
|
|
|
|
/**
|
|
* @param display The display.
|
|
* @param w Window we want to grab keyboard on.
|
|
*
|
|
* Grab keyboard and mouse.
|
|
*
|
|
* @return 1 when keyboard is grabbed, 0 not.
|
|
*/
|
|
static int take_keyboard ( Display *display, Window w )
|
|
{
|
|
if ( XGrabKeyboard ( display, w, True, GrabModeAsync, GrabModeAsync, CurrentTime ) == GrabSuccess ) {
|
|
return 1;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* @param display The display.
|
|
*
|
|
* Release keyboard.
|
|
*/
|
|
static void release_keyboard ( Display *display )
|
|
{
|
|
XUngrabKeyboard ( display, CurrentTime );
|
|
}
|
|
|
|
// XGetWindowAttributes with caching
|
|
static XWindowAttributes* window_get_attributes ( Window w )
|
|
{
|
|
int idx = winlist_find ( cache_xattr, w );
|
|
|
|
if ( idx < 0 ) {
|
|
XWindowAttributes *cattr = g_malloc ( sizeof ( XWindowAttributes ) );
|
|
|
|
if ( XGetWindowAttributes ( display, w, cattr ) ) {
|
|
winlist_append ( cache_xattr, w, cattr );
|
|
return cattr;
|
|
}
|
|
|
|
g_free ( cattr );
|
|
return NULL;
|
|
}
|
|
|
|
return cache_xattr->data[idx];
|
|
}
|
|
|
|
// retrieve a property of any type from a window
|
|
static int window_get_prop ( Window w, Atom prop, Atom *type, int *items, void *buffer, unsigned int bytes ) __attribute__ ( ( nonnull ( 3, 4 ) ) );
|
|
static int window_get_prop ( Window w, Atom prop, Atom *type, int *items, void *buffer, unsigned int bytes )
|
|
{
|
|
int format;
|
|
unsigned long nitems, nbytes;
|
|
unsigned char *ret = NULL;
|
|
memset ( buffer, 0, bytes );
|
|
|
|
if ( XGetWindowProperty ( display, w, prop, 0, bytes / 4, False, AnyPropertyType, type,
|
|
&format, &nitems, &nbytes, &ret ) == Success && ret && *type != None && format ) {
|
|
if ( format == 8 ) {
|
|
memmove ( buffer, ret, MIN ( bytes, nitems ) );
|
|
}
|
|
|
|
if ( format == 16 ) {
|
|
memmove ( buffer, ret, MIN ( bytes, nitems * sizeof ( short ) ) );
|
|
}
|
|
|
|
if ( format == 32 ) {
|
|
memmove ( buffer, ret, MIN ( bytes, nitems * sizeof ( long ) ) );
|
|
}
|
|
|
|
*items = ( int ) nitems;
|
|
XFree ( ret );
|
|
return 1;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
// retrieve a text property from a window
|
|
// technically we could use window_get_prop(), but this is better for character set support
|
|
static char* window_get_text_prop ( Window w, Atom atom )
|
|
{
|
|
XTextProperty prop;
|
|
char *res = NULL;
|
|
char **list = NULL;
|
|
int count;
|
|
|
|
if ( XGetTextProperty ( display, w, &prop, atom ) && prop.value && prop.nitems ) {
|
|
if ( prop.encoding == XA_STRING ) {
|
|
res = g_malloc ( strlen ( ( char * ) prop.value ) + 1 );
|
|
// make clang-check happy.
|
|
if ( res ) {
|
|
strcpy ( res, ( char * ) prop.value );
|
|
}
|
|
}
|
|
else if ( Xutf8TextPropertyToTextList ( display, &prop, &list, &count ) >= Success && count > 0 && *list ) {
|
|
res = g_malloc ( strlen ( *list ) + 1 );
|
|
// make clang-check happy.
|
|
if ( res ) {
|
|
strcpy ( res, *list );
|
|
}
|
|
XFreeStringList ( list );
|
|
}
|
|
}
|
|
|
|
if ( prop.value ) {
|
|
XFree ( prop.value );
|
|
}
|
|
|
|
return res;
|
|
}
|
|
|
|
static int window_get_atom_prop ( Window w, Atom atom, Atom *list, int count )
|
|
{
|
|
Atom type;
|
|
int items;
|
|
return window_get_prop ( w, atom, &type, &items, list, count * sizeof ( Atom ) ) && type == XA_ATOM ? items : 0;
|
|
}
|
|
|
|
static void window_set_atom_prop ( Window w, Atom prop, Atom *atoms, int count )
|
|
{
|
|
XChangeProperty ( display, w, prop, XA_ATOM, 32, PropModeReplace, ( unsigned char * ) atoms, count );
|
|
}
|
|
|
|
static int window_get_cardinal_prop ( Window w, Atom atom, unsigned long *list, int count )
|
|
{
|
|
Atom type; int items;
|
|
return window_get_prop ( w, atom, &type, &items, list, count * sizeof ( unsigned long ) ) && type == XA_CARDINAL ? items : 0;
|
|
}
|
|
|
|
// a ClientMessage
|
|
static int window_send_message ( Window target, Window subject, Atom atom, unsigned long protocol, unsigned long mask, Time time )
|
|
{
|
|
XEvent e;
|
|
memset ( &e, 0, sizeof ( XEvent ) );
|
|
e.xclient.type = ClientMessage;
|
|
e.xclient.message_type = atom;
|
|
e.xclient.window = subject;
|
|
e.xclient.data.l[0] = protocol;
|
|
e.xclient.data.l[1] = time;
|
|
e.xclient.send_event = True;
|
|
e.xclient.format = 32;
|
|
int r = XSendEvent ( display, target, False, mask, &e ) ? 1 : 0;
|
|
XFlush ( display );
|
|
return r;
|
|
}
|
|
|
|
// find the dimensions of the monitor displaying point x,y
|
|
static void monitor_dimensions ( Screen *screen, int x, int y, workarea *mon )
|
|
{
|
|
memset ( mon, 0, sizeof ( workarea ) );
|
|
mon->w = WidthOfScreen ( screen );
|
|
mon->h = HeightOfScreen ( screen );
|
|
|
|
// locate the current monitor
|
|
if ( XineramaIsActive ( display ) ) {
|
|
int monitors;
|
|
XineramaScreenInfo *info = XineramaQueryScreens ( display, &monitors );
|
|
|
|
if ( info ) {
|
|
for ( int i = 0; i < monitors; i++ ) {
|
|
if ( INTERSECT ( x, y, 1, 1, info[i].x_org, info[i].y_org, info[i].width, info[i].height ) ) {
|
|
mon->x = info[i].x_org;
|
|
mon->y = info[i].y_org;
|
|
mon->w = info[i].width;
|
|
mon->h = info[i].height;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
XFree ( info );
|
|
}
|
|
}
|
|
|
|
// determine which monitor holds the active window, or failing that the mouse pointer
|
|
static void monitor_active ( workarea *mon )
|
|
{
|
|
Screen *screen = DefaultScreenOfDisplay ( display );
|
|
Window root = RootWindow ( display, XScreenNumberOfScreen ( screen ) );
|
|
int x, y;
|
|
|
|
Window id;
|
|
Atom type;
|
|
int count;
|
|
if ( window_get_prop ( root, netatoms[_NET_ACTIVE_WINDOW], &type, &count, &id, sizeof ( Window ) )
|
|
&& type == XA_WINDOW && count > 0 ) {
|
|
XWindowAttributes *attr = window_get_attributes ( id );
|
|
if ( attr != NULL ) {
|
|
Window junkwin;
|
|
if ( XTranslateCoordinates ( display, id, attr->root,
|
|
-attr->border_width,
|
|
-attr->border_width,
|
|
&x, &y, &junkwin ) == True ) {
|
|
monitor_dimensions ( screen, x, y, mon );
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
if ( pointer_get ( display, root, &x, &y ) ) {
|
|
monitor_dimensions ( screen, x, y, mon );
|
|
return;
|
|
}
|
|
|
|
monitor_dimensions ( screen, 0, 0, mon );
|
|
}
|
|
|
|
// _NET_WM_STATE_*
|
|
static int client_has_state ( client *c, Atom state )
|
|
{
|
|
int i;
|
|
|
|
for ( i = 0; i < c->states; i++ ) {
|
|
if ( c->state[i] == state ) {
|
|
return 1;
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
// collect info on any window
|
|
// doesn't have to be a window we'll end up managing
|
|
static client* window_client ( Window win )
|
|
{
|
|
if ( win == None ) {
|
|
return NULL;
|
|
}
|
|
|
|
int idx = winlist_find ( cache_client, win );
|
|
|
|
if ( idx >= 0 ) {
|
|
return cache_client->data[idx];
|
|
}
|
|
|
|
// if this fails, we're up that creek
|
|
XWindowAttributes *attr = window_get_attributes ( win );
|
|
|
|
if ( !attr ) {
|
|
return NULL;
|
|
}
|
|
|
|
client *c = g_malloc0 ( sizeof ( client ) );
|
|
c->window = win;
|
|
|
|
// copy xattr so we don't have to care when stuff is freed
|
|
memmove ( &c->xattr, attr, sizeof ( XWindowAttributes ) );
|
|
XGetTransientForHint ( display, win, &c->trans );
|
|
|
|
c->states = window_get_atom_prop ( win, netatoms[_NET_WM_STATE], c->state, CLIENTSTATE );
|
|
|
|
char *name;
|
|
|
|
if ( ( name = window_get_text_prop ( c->window, netatoms[_NET_WM_NAME] ) ) && name ) {
|
|
snprintf ( c->title, CLIENTTITLE, "%s", name );
|
|
g_free ( name );
|
|
}
|
|
else if ( XFetchName ( display, c->window, &name ) ) {
|
|
snprintf ( c->title, CLIENTTITLE, "%s", name );
|
|
XFree ( name );
|
|
}
|
|
|
|
name = window_get_text_prop ( c->window, XInternAtom ( display, "WM_WINDOW_ROLE", False ) );
|
|
|
|
if ( name != NULL ) {
|
|
snprintf ( c->role, CLIENTROLE, "%s", name );
|
|
XFree ( name );
|
|
}
|
|
|
|
XClassHint chint;
|
|
|
|
if ( XGetClassHint ( display, c->window, &chint ) ) {
|
|
snprintf ( c->class, CLIENTCLASS, "%s", chint.res_class );
|
|
snprintf ( c->name, CLIENTNAME, "%s", chint.res_name );
|
|
XFree ( chint.res_class );
|
|
XFree ( chint.res_name );
|
|
}
|
|
|
|
monitor_dimensions ( c->xattr.screen, c->xattr.x, c->xattr.y, &c->monitor );
|
|
winlist_append ( cache_client, c->window, c );
|
|
return c;
|
|
}
|
|
|
|
|
|
|
|
static void menu_hide_arrow_text ( int filtered_lines, int selected, int max_elements,
|
|
textbox *arrowbox_top, textbox *arrowbox_bottom )
|
|
{
|
|
if ( arrowbox_top == NULL || arrowbox_bottom == NULL ) {
|
|
return;
|
|
}
|
|
int page = ( filtered_lines > 0 ) ? selected / max_elements : 0;
|
|
int npages = ( filtered_lines > 0 ) ? ( ( filtered_lines + max_elements - 1 ) / max_elements ) : 1;
|
|
if ( !( page != 0 && npages > 1 ) ) {
|
|
textbox_hide ( arrowbox_top );
|
|
}
|
|
if ( !( ( npages - 1 ) != page && npages > 1 ) ) {
|
|
textbox_hide ( arrowbox_bottom );
|
|
}
|
|
}
|
|
|
|
static void menu_set_arrow_text ( int filtered_lines, int selected, int max_elements,
|
|
textbox *arrowbox_top, textbox *arrowbox_bottom )
|
|
{
|
|
if ( arrowbox_top == NULL || arrowbox_bottom == NULL ) {
|
|
return;
|
|
}
|
|
if ( filtered_lines == 0 || max_elements == 0 ) {
|
|
return;
|
|
}
|
|
int page = ( filtered_lines > 0 ) ? selected / max_elements : 0;
|
|
int npages = ( filtered_lines > 0 ) ? ( ( filtered_lines + max_elements - 1 ) / max_elements ) : 1;
|
|
int entry = selected % max_elements;
|
|
if ( page != 0 && npages > 1 ) {
|
|
textbox_show ( arrowbox_top );
|
|
textbox_font ( arrowbox_top, ( entry != 0 ) ? NORMAL : HIGHLIGHT );
|
|
textbox_draw ( arrowbox_top );
|
|
}
|
|
if ( ( npages - 1 ) != page && npages > 1 ) {
|
|
textbox_show ( arrowbox_bottom );
|
|
textbox_font ( arrowbox_bottom, ( entry != ( max_elements - 1 ) ) ? NORMAL : HIGHLIGHT );
|
|
textbox_draw ( arrowbox_bottom );
|
|
}
|
|
}
|
|
|
|
|
|
static int window_match ( char **tokens, __attribute__( ( unused ) ) const char *input,
|
|
int case_sensitive, int index, void *data )
|
|
{
|
|
int match = 1;
|
|
winlist *ids = ( winlist * ) data;
|
|
client *c = window_client ( ids->array[index] );
|
|
|
|
if ( tokens ) {
|
|
for ( int j = 0; match && tokens[j]; j++ ) {
|
|
int test = 0;
|
|
|
|
if ( !test && c->title[0] != '\0' ) {
|
|
char *key = token_collate_key ( c->title, case_sensitive );
|
|
test = ( strstr ( key, tokens[j] ) != NULL );
|
|
g_free ( key );
|
|
}
|
|
|
|
if ( !test && c->class[0] != '\0' ) {
|
|
char *key = token_collate_key ( c->class, case_sensitive );
|
|
test = ( strstr ( key, tokens[j] ) != NULL );
|
|
g_free ( key );
|
|
}
|
|
|
|
if ( !test && c->role[0] != '\0' ) {
|
|
char *key = token_collate_key ( c->role, case_sensitive );
|
|
test = ( strstr ( key, tokens[j] ) != NULL );
|
|
g_free ( key );
|
|
}
|
|
|
|
if ( !test && c->name[0] != '\0' ) {
|
|
char *key = token_collate_key ( c->name, case_sensitive );
|
|
test = ( strstr ( key, tokens[j] ) != NULL );
|
|
g_free ( key );
|
|
}
|
|
|
|
if ( test == 0 ) {
|
|
match = 0;
|
|
}
|
|
}
|
|
}
|
|
|
|
return match;
|
|
}
|
|
|
|
static int lev_sort ( const void *p1, const void *p2, void *arg )
|
|
{
|
|
const int *a = p1;
|
|
const int *b = p2;
|
|
int *distances = arg;
|
|
|
|
return distances[*a] - distances[*b];
|
|
}
|
|
|
|
static int dist ( const char *s, const char *t, int *d, int ls, int lt, int i, int j )
|
|
{
|
|
if ( d[i * ( lt + 1 ) + j] >= 0 ) {
|
|
return d[i * ( lt + 1 ) + j];
|
|
}
|
|
|
|
int x;
|
|
if ( i == ls ) {
|
|
x = lt - j;
|
|
}
|
|
else if ( j == lt ) {
|
|
x = ls - i;
|
|
}
|
|
else if ( s[i] == t[j] ) {
|
|
x = dist ( s, t, d, ls, lt, i + 1, j + 1 );
|
|
}
|
|
else {
|
|
x = dist ( s, t, d, ls, lt, i + 1, j + 1 );
|
|
|
|
int y;
|
|
if ( ( y = dist ( s, t, d, ls, lt, i, j + 1 ) ) < x ) {
|
|
x = y;
|
|
}
|
|
if ( ( y = dist ( s, t, d, ls, lt, i + 1, j ) ) < x ) {
|
|
x = y;
|
|
}
|
|
x++;
|
|
}
|
|
return d[i * ( lt + 1 ) + j] = x;
|
|
}
|
|
static int levenshtein ( const char *s, const char *t )
|
|
{
|
|
int ls = strlen ( s ), lt = strlen ( t );
|
|
size_t array_length = ( ls + 1 ) * ( lt + 1 );
|
|
|
|
// For some reason Coverity does not get that I initialize the
|
|
// array in for loop.
|
|
int *d = g_malloc_n ( array_length, sizeof ( int ) );
|
|
|
|
for ( size_t i = 0; i < array_length; i++ ) {
|
|
d[i] = -1;
|
|
}
|
|
|
|
int retv = dist ( s, t, d, ls, lt, 0, 0 );
|
|
g_free ( d );
|
|
return retv;
|
|
}
|
|
|
|
static void window_set_opacity ( Display *display, Window box, unsigned int opacity )
|
|
{
|
|
// Hack to set window opacity.
|
|
unsigned int opacity_set = ( unsigned int ) ( ( opacity / 100.0 ) * UINT32_MAX );
|
|
XChangeProperty ( display, box, netatoms[_NET_WM_WINDOW_OPACITY],
|
|
XA_CARDINAL, 32, PropModeReplace,
|
|
( unsigned char * ) &opacity_set, 1L );
|
|
}
|
|
|
|
Window create_window ( Display *display )
|
|
{
|
|
XSetWindowAttributes attr;
|
|
attr.colormap = map;
|
|
attr.border_pixel = color_get ( display, config.menu_bc );
|
|
attr.background_pixel = color_get ( display, config.menu_bg );
|
|
|
|
Window box = XCreateWindow ( display, DefaultRootWindow ( display ),
|
|
0, 0, 200, 100, config.menu_bw, vinfo.depth, InputOutput,
|
|
vinfo.visual, CWColormap | CWBorderPixel | CWBackPixel, &attr );
|
|
XSelectInput ( display, box, ExposureMask | ButtonPressMask );
|
|
|
|
gc = XCreateGC ( display, box, 0, 0 );
|
|
XSetLineAttributes ( display, gc, 2, LineOnOffDash, CapButt, JoinMiter );
|
|
XSetForeground ( display, gc, color_get ( display, config.menu_bc ) );
|
|
// make it an unmanaged window
|
|
window_set_atom_prop ( box, netatoms[_NET_WM_STATE], &netatoms[_NET_WM_STATE_ABOVE], 1 );
|
|
XSetWindowAttributes sattr;
|
|
sattr.override_redirect = True;
|
|
XChangeWindowAttributes ( display, box, CWOverrideRedirect, &sattr );
|
|
|
|
// Set the WM_NAME
|
|
XStoreName ( display, box, "rofi" );
|
|
|
|
window_set_opacity ( display, box, config.window_opacity );
|
|
return box;
|
|
}
|
|
|
|
// State of the menu.
|
|
|
|
typedef struct MenuState
|
|
{
|
|
unsigned int menu_lines;
|
|
unsigned int max_elements;
|
|
unsigned int max_rows;
|
|
unsigned int columns;
|
|
|
|
// window width,height
|
|
unsigned int w, h;
|
|
int x, y;
|
|
unsigned int element_width;
|
|
|
|
// Update/Refilter list.
|
|
int update;
|
|
int refilter;
|
|
|
|
// Entries
|
|
textbox *text;
|
|
textbox *prompt_tb;
|
|
textbox *case_indicator;
|
|
textbox *arrowbox_top;
|
|
textbox *arrowbox_bottom;
|
|
textbox **boxes;
|
|
char **filtered;
|
|
int *distance;
|
|
int *line_map;
|
|
|
|
unsigned int num_lines;
|
|
|
|
// Selected element.
|
|
unsigned int selected;
|
|
unsigned int filtered_lines;
|
|
// Last offset in paginating.
|
|
unsigned int last_offset;
|
|
|
|
KeySym prev_key;
|
|
Time last_button_press;
|
|
|
|
int quit;
|
|
int init;
|
|
// Return state
|
|
int *selected_line;
|
|
MenuReturn retv;
|
|
char **lines;
|
|
}MenuState;
|
|
|
|
/**
|
|
* @param state Internal state of the menu.
|
|
*
|
|
* Free the allocated fields in the state.
|
|
*/
|
|
static void menu_free_state ( MenuState *state )
|
|
{
|
|
textbox_free ( state->text );
|
|
textbox_free ( state->prompt_tb );
|
|
textbox_free ( state->case_indicator );
|
|
textbox_free ( state->arrowbox_bottom );
|
|
textbox_free ( state->arrowbox_top );
|
|
|
|
for ( unsigned int i = 0; i < state->max_elements; i++ ) {
|
|
textbox_free ( state->boxes[i] );
|
|
}
|
|
|
|
g_free ( state->boxes );
|
|
|
|
g_free ( state->filtered );
|
|
g_free ( state->line_map );
|
|
g_free ( state->distance );
|
|
}
|
|
|
|
/**
|
|
* @param x [out] the calculated x position.
|
|
* @param y [out] the calculated y position.
|
|
* @param mon the workarea.
|
|
* @param h the required height of the window.
|
|
* @param w the required width of the window.
|
|
*/
|
|
static void calculate_window_position ( MenuState *state, const workarea *mon )
|
|
{
|
|
// Default location is center.
|
|
state->y = mon->y + ( mon->h - state->h - config.menu_bw * 2 ) / 2;
|
|
state->x = mon->x + ( mon->w - state->w - config.menu_bw * 2 ) / 2;
|
|
// Determine window location
|
|
switch ( config.location )
|
|
{
|
|
case WL_NORTH_WEST:
|
|
state->x = mon->x;
|
|
|
|
case WL_NORTH:
|
|
state->y = mon->y;
|
|
break;
|
|
|
|
case WL_NORTH_EAST:
|
|
state->y = mon->y;
|
|
|
|
case WL_EAST:
|
|
state->x = mon->x + mon->w - state->w - config.menu_bw * 2;
|
|
break;
|
|
|
|
case WL_EAST_SOUTH:
|
|
state->x = mon->x + mon->w - state->w - config.menu_bw * 2;
|
|
|
|
case WL_SOUTH:
|
|
state->y = mon->y + mon->h - state->h - config.menu_bw * 2;
|
|
break;
|
|
|
|
case WL_SOUTH_WEST:
|
|
state->y = mon->y + mon->h - state->h - config.menu_bw * 2;
|
|
|
|
case WL_WEST:
|
|
state->x = mon->x;
|
|
break;
|
|
|
|
case WL_CENTER:
|
|
default:
|
|
break;
|
|
}
|
|
// Apply offset.
|
|
state->x += config.x_offset;
|
|
state->y += config.y_offset;
|
|
}
|
|
|
|
|
|
/**
|
|
* @param state Internal state of the menu.
|
|
* @param num_lines the number of entries passed to the menu.
|
|
*
|
|
* Calculate the number of rows, columns and elements to display based on the
|
|
* configuration and available data.
|
|
*/
|
|
static void menu_calculate_rows_columns ( MenuState *state )
|
|
{
|
|
state->columns = config.menu_columns;
|
|
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
|
|
// (num elements is better name) then dividing by columns.
|
|
state->max_rows = MIN ( state->menu_lines,
|
|
(unsigned int) (
|
|
( state->num_lines + ( state->columns - state->num_lines % state->columns ) %
|
|
state->columns ) / ( state->columns )
|
|
) );
|
|
|
|
if ( config.fixed_num_lines == TRUE ) {
|
|
state->max_elements = state->menu_lines * state->columns;
|
|
state->max_rows = state->menu_lines;
|
|
// If it would fit in one column, only use one column.
|
|
if ( state->num_lines < state->max_elements ) {
|
|
state->columns = ( state->num_lines + ( state->max_rows - state->num_lines % state->max_rows ) %
|
|
state->max_rows ) / state->max_rows;
|
|
state->max_elements = state->menu_lines * state->columns;
|
|
}
|
|
// Sanitize.
|
|
if ( state->columns == 0 ) {
|
|
state->columns = 1;
|
|
}
|
|
}
|
|
// More hacks.
|
|
if ( config.hmode == TRUE ) {
|
|
state->max_rows = 1;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* @param state Internal state of the menu.
|
|
* @param mon the dimensions of the monitor rofi is displayed on.
|
|
*
|
|
* Calculate the width of the window and the width of an element.
|
|
*/
|
|
static void menu_calculate_window_and_element_width ( MenuState *state, workarea *mon )
|
|
{
|
|
if ( config.menu_width < 0 ) {
|
|
double fw = textbox_get_estimated_char_width ( );
|
|
state->w = -( fw * config.menu_width );
|
|
state->w += 2 * config.padding + 4; // 4 = 2*SIDE_MARGIN
|
|
// Compensate for border width.
|
|
state->w -= config.menu_bw * 2;
|
|
}
|
|
else{
|
|
// Calculate as float to stop silly, big rounding down errors.
|
|
state->w = config.menu_width < 101 ? ( mon->w / 100.0f ) * ( float ) config.menu_width : config.menu_width;
|
|
// Compensate for border width.
|
|
state->w -= config.menu_bw * 2;
|
|
}
|
|
|
|
if ( state->columns > 0 ) {
|
|
state->element_width = state->w - ( 2 * ( config.padding ) );
|
|
// Divide by the # columns
|
|
state->element_width = ( state->element_width - ( state->columns - 1 ) * LINE_MARGIN ) / state->columns;
|
|
if ( config.hmode == TRUE ) {
|
|
state->element_width = ( state->w - ( 2 * ( config.padding ) ) - state->max_elements * LINE_MARGIN ) / (
|
|
state->max_elements + 1 );
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Nav helper functions, to avoid duplicate code.
|
|
*/
|
|
/**
|
|
* @param state The current MenuState
|
|
*
|
|
* Move the selection one column to the right.
|
|
* - No wrap around.
|
|
* - Do not move to top row when at start.
|
|
*/
|
|
inline static void menu_nav_right ( MenuState *state )
|
|
{
|
|
if ( ( state->selected + state->max_rows ) < state->filtered_lines ) {
|
|
state->selected += state->max_rows;
|
|
state->update = TRUE;
|
|
}
|
|
else if ( state->selected < ( state->filtered_lines - 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 = state->selected / state->max_rows;
|
|
// Check total number of columns.
|
|
int ncol = state->filtered_lines / state->max_rows;
|
|
// If there is an extra column, move.
|
|
if ( col != ncol ) {
|
|
state->selected = state->filtered_lines - 1;
|
|
state->update = TRUE;
|
|
}
|
|
}
|
|
}
|
|
/**
|
|
* @param state The current MenuState
|
|
*
|
|
* Move the selection one column to the left.
|
|
* - No wrap around.
|
|
*/
|
|
inline static void menu_nav_left ( MenuState *state )
|
|
{
|
|
if ( state->selected >= state->max_rows ) {
|
|
state->selected -= state->max_rows;
|
|
state->update = TRUE;
|
|
}
|
|
}
|
|
/**
|
|
* @param state The current MenuState
|
|
*
|
|
* Move the selection one row up.
|
|
* - Wrap around.
|
|
*/
|
|
inline static void menu_nav_up ( MenuState *state )
|
|
{
|
|
// Wrap around.
|
|
if ( state->selected == 0 ) {
|
|
state->selected = state->filtered_lines;
|
|
}
|
|
|
|
if ( state->selected > 0 ) {
|
|
state->selected--;
|
|
}
|
|
state->update = TRUE;
|
|
}
|
|
/**
|
|
* @param state The current MenuState
|
|
*
|
|
* Move the selection one row down.
|
|
* - Wrap around.
|
|
*/
|
|
inline static void menu_nav_down ( MenuState *state )
|
|
{
|
|
state->selected = state->selected < state->filtered_lines - 1 ? MIN (
|
|
state->filtered_lines - 1, state->selected + 1 ) : 0;
|
|
state->update = TRUE;
|
|
}
|
|
/**
|
|
* @param state Internal state of the menu.
|
|
* @param key the Key being pressed.
|
|
* @param modstate the modifier state.
|
|
*
|
|
* Keyboard navigation through the elements.
|
|
*/
|
|
static void menu_keyboard_navigation ( MenuState *state, KeySym key, unsigned int modstate )
|
|
{
|
|
if ( key == XK_Escape
|
|
// pressing one of the global key bindings closes the switcher. this allows fast closing of the menu if an item is not selected
|
|
|| ( ( windows_modmask == AnyModifier || modstate & windows_modmask ) && key == windows_keysym )
|
|
|| ( ( rundialog_modmask == AnyModifier || modstate & rundialog_modmask ) && key == rundialog_keysym )
|
|
|| ( ( sshdialog_modmask == AnyModifier || modstate & sshdialog_modmask ) && key == sshdialog_keysym )
|
|
) {
|
|
state->retv = MENU_CANCEL;
|
|
state->quit = TRUE;
|
|
}
|
|
// Up, Ctrl-p or Shift-Tab
|
|
else if ( key == XK_Up || ( key == XK_Tab && modstate & ShiftMask ) ||
|
|
( key == XK_p && modstate & ControlMask ) ) {
|
|
menu_nav_up ( state );
|
|
}
|
|
else if ( key == XK_Tab ) {
|
|
if ( state->filtered_lines == 1 ) {
|
|
if ( state->filtered[state->selected] ) {
|
|
state->retv = MENU_OK;
|
|
*( state->selected_line ) = state->line_map[state->selected];
|
|
state->quit = 1;
|
|
}
|
|
else{
|
|
fprintf ( stderr, "We should never hit this." );
|
|
abort ();
|
|
}
|
|
return;
|
|
}
|
|
|
|
// Double tab!
|
|
if ( state->filtered_lines == 0 && key == state->prev_key ) {
|
|
state->retv = MENU_NEXT;
|
|
*( state->selected_line ) = 0;
|
|
state->quit = TRUE;
|
|
}
|
|
else{
|
|
state->selected = state->selected < state->filtered_lines - 1 ? MIN (
|
|
state->filtered_lines - 1, state->selected + 1 ) : 0;
|
|
state->update = TRUE;
|
|
}
|
|
}
|
|
// Down, Ctrl-n
|
|
else if ( key == XK_Down ||
|
|
( key == XK_n && ( modstate & ControlMask ) ) ) {
|
|
menu_nav_down ( state );
|
|
}
|
|
else if ( key == XK_Page_Up && ( modstate & ControlMask ) ) {
|
|
menu_nav_left ( state );
|
|
}
|
|
else if ( key == XK_Page_Down && ( modstate & ControlMask ) ) {
|
|
menu_nav_right ( state );
|
|
}
|
|
else if ( key == XK_Page_Up ) {
|
|
if ( state->selected < state->max_elements ) {
|
|
state->selected = 0;
|
|
}
|
|
else{
|
|
state->selected -= ( state->max_elements );
|
|
}
|
|
state->update = TRUE;
|
|
}
|
|
else if ( key == XK_Page_Down ) {
|
|
state->selected += ( state->max_elements );
|
|
if ( state->selected >= state->filtered_lines ) {
|
|
state->selected = state->filtered_lines - 1;
|
|
}
|
|
state->update = TRUE;
|
|
}
|
|
else if ( key == XK_Home || key == XK_KP_Home ) {
|
|
state->selected = 0;
|
|
state->update = TRUE;
|
|
}
|
|
else if ( key == XK_End || key == XK_KP_End ) {
|
|
state->selected = state->filtered_lines - 1;
|
|
state->update = TRUE;
|
|
}
|
|
else if ( key == XK_space && ( modstate & ControlMask ) == ControlMask ) {
|
|
// If a valid item is selected, return that..
|
|
if ( state->selected < state->filtered_lines && state->filtered[state->selected] != NULL ) {
|
|
textbox_text ( state->text, state->lines[state->line_map[state->selected]] );
|
|
textbox_cursor_end ( state->text );
|
|
state->update = TRUE;
|
|
state->refilter = TRUE;
|
|
}
|
|
}
|
|
state->prev_key = key;
|
|
}
|
|
|
|
/**
|
|
* @param state Internal state of the menu.
|
|
* @param xbe The mouse button press event.
|
|
*
|
|
* mouse navigation through the elements.
|
|
*
|
|
* TODO: Scroll wheel.
|
|
*/
|
|
static void menu_mouse_navigation ( MenuState *state, XButtonEvent *xbe )
|
|
{
|
|
// Scroll event
|
|
if ( xbe->button > 3 ) {
|
|
if ( xbe->button == 4 ) {
|
|
menu_nav_up ( state );
|
|
}
|
|
else if ( xbe->button == 5 ) {
|
|
menu_nav_down ( state );
|
|
}
|
|
else if ( xbe->button == 6 ) {
|
|
menu_nav_left ( state );
|
|
}
|
|
else if ( xbe->button == 7 ) {
|
|
menu_nav_right ( state );
|
|
}
|
|
return;
|
|
}
|
|
if ( xbe->window == state->arrowbox_top->window ) {
|
|
// Page up.
|
|
if ( state->selected < state->max_rows ) {
|
|
state->selected = 0;
|
|
}
|
|
else{
|
|
state->selected -= state->max_elements;
|
|
}
|
|
state->update = TRUE;
|
|
}
|
|
else if ( xbe->window == state->arrowbox_bottom->window ) {
|
|
// Page down.
|
|
state->selected += state->max_elements;
|
|
if ( state->selected >= state->filtered_lines ) {
|
|
state->selected = state->filtered_lines - 1;
|
|
}
|
|
state->update = TRUE;
|
|
}
|
|
else {
|
|
for ( unsigned int i = 0; config.sidebar_mode == TRUE && i < num_switchers; i++ ) {
|
|
if ( switchers[i].tb->window == ( xbe->window ) ) {
|
|
*( state->selected_line ) = i;
|
|
state->retv = MENU_QUICK_SWITCH;
|
|
state->quit = TRUE;
|
|
return;
|
|
}
|
|
}
|
|
for ( unsigned int i = 0; i < state->max_elements; i++ ) {
|
|
if ( ( xbe->window ) == ( state->boxes[i]->window ) ) {
|
|
// Only allow items that are visible to be selected.
|
|
if ( ( state->last_offset + i ) >= state->filtered_lines ) {
|
|
break;
|
|
}
|
|
//
|
|
state->selected = state->last_offset + i;
|
|
state->update = TRUE;
|
|
if ( ( xbe->time - state->last_button_press ) < 200 ) {
|
|
state->retv = MENU_OK;
|
|
*( state->selected_line ) = state->line_map[state->selected];
|
|
// Quit
|
|
state->quit = TRUE;
|
|
}
|
|
state->last_button_press = xbe->time;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
static void menu_refilter ( MenuState *state, char **lines, menu_match_cb mmc, void *mmc_data,
|
|
int sorting, int case_sensitive )
|
|
{
|
|
if ( strlen ( state->text->text ) > 0 ) {
|
|
unsigned int j = 0;
|
|
char **tokens = tokenize ( state->text->text, case_sensitive );
|
|
|
|
// input changed
|
|
for ( unsigned int i = 0; i < state->num_lines; i++ ) {
|
|
int match = mmc ( tokens, lines[i], case_sensitive, i, mmc_data );
|
|
|
|
// If each token was matched, add it to list.
|
|
if ( match ) {
|
|
state->line_map[j] = i;
|
|
if ( sorting ) {
|
|
state->distance[i] = levenshtein ( state->text->text, lines[i] );
|
|
}
|
|
// Try to look-up the selected line and highlight that.
|
|
// This is needed 'hack' to fix the dmenu 'next row' modi.
|
|
// int to unsigned int is valid because we check negativeness of
|
|
// selected_line
|
|
if ( state->init == TRUE && ( state->selected_line ) != NULL &&
|
|
( *( state->selected_line ) ) >= 0 &&
|
|
( (unsigned int) ( *( state->selected_line ) ) ) == i ) {
|
|
state->selected = j;
|
|
state->init = FALSE;
|
|
}
|
|
j++;
|
|
}
|
|
}
|
|
if ( sorting ) {
|
|
qsort_r ( state->line_map, j, sizeof ( int ), lev_sort, state->distance );
|
|
}
|
|
// Update the filtered list.
|
|
for ( unsigned int i = 0; i < j; i++ ) {
|
|
state->filtered[i] = lines[state->line_map[i]];
|
|
}
|
|
for ( unsigned int i = j; i < state->num_lines; i++ ) {
|
|
state->filtered[i] = NULL;
|
|
}
|
|
|
|
// Cleanup + bookkeeping.
|
|
state->filtered_lines = j;
|
|
g_strfreev ( tokens );
|
|
}
|
|
else{
|
|
for ( unsigned int i = 0; i < state->num_lines; i++ ) {
|
|
state->filtered[i] = lines[i];
|
|
state->line_map[i] = i;
|
|
}
|
|
state->filtered_lines = state->num_lines;
|
|
}
|
|
state->selected = MIN ( state->selected, state->filtered_lines - 1 );
|
|
|
|
state->refilter = FALSE;
|
|
}
|
|
|
|
|
|
static void menu_draw ( MenuState *state )
|
|
{
|
|
unsigned int i, offset = 0;
|
|
|
|
// selected row is always visible.
|
|
// If selected is visible do not scroll.
|
|
if ( ( ( state->selected - ( state->last_offset ) ) < ( state->max_elements ) )
|
|
&& ( state->selected >= ( state->last_offset ) ) ) {
|
|
offset = state->last_offset;
|
|
}
|
|
else{
|
|
// Do paginating
|
|
int page = ( state->max_elements > 0 ) ? ( state->selected / state->max_elements ) : 0;
|
|
offset = page * state->max_elements;
|
|
state->last_offset = offset;
|
|
}
|
|
|
|
for ( i = 0; i < state->max_elements; i++ ) {
|
|
if ( ( i + offset ) >= state->num_lines || state->filtered[i + offset] == NULL ) {
|
|
textbox_font ( state->boxes[i], NORMAL );
|
|
textbox_text ( state->boxes[i], "" );
|
|
}
|
|
else{
|
|
TextBoxFontType type = ( ( ( i % state->max_rows ) & 1 ) == 0 ) ? NORMAL : ALT;
|
|
char *text = state->filtered[i + offset];
|
|
TextBoxFontType tbft = ( i + offset ) == state->selected ? HIGHLIGHT : type;
|
|
textbox_font ( state->boxes[i], tbft );
|
|
textbox_text ( state->boxes[i], text );
|
|
}
|
|
|
|
textbox_draw ( state->boxes[i] );
|
|
}
|
|
}
|
|
|
|
static void menu_update ( MenuState *state )
|
|
{
|
|
menu_hide_arrow_text ( state->filtered_lines, state->selected,
|
|
state->max_elements, state->arrowbox_top,
|
|
state->arrowbox_bottom );
|
|
textbox_draw ( state->case_indicator );
|
|
textbox_draw ( state->prompt_tb );
|
|
textbox_draw ( state->text );
|
|
menu_draw ( state );
|
|
menu_set_arrow_text ( state->filtered_lines, state->selected,
|
|
state->max_elements, state->arrowbox_top,
|
|
state->arrowbox_bottom );
|
|
// Why do we need the specian -1?
|
|
if ( config.hmode == FALSE ) {
|
|
int line_height = textbox_get_height ( state->text );
|
|
XDrawLine ( display, main_window, gc, ( config.padding ),
|
|
line_height + ( config.padding ) + ( LINE_MARGIN ) / 2,
|
|
state->w - ( ( config.padding ) ) - 1,
|
|
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 ) - 1,
|
|
state->w - ( ( config.padding ) ) - 1,
|
|
state->h - line_height - ( config.padding ) - 1 );
|
|
for ( int j = 0; j < num_switchers; j++ ) {
|
|
textbox_draw ( switchers[j].tb );
|
|
}
|
|
}
|
|
|
|
|
|
state->update = FALSE;
|
|
}
|
|
|
|
/**
|
|
* @param state Internal state of the menu.
|
|
* @param xse X selection event.
|
|
*
|
|
* Handle paste event.
|
|
*/
|
|
static void menu_paste ( MenuState *state, XSelectionEvent *xse )
|
|
{
|
|
if ( xse->property == netatoms[UTF8_STRING] ) {
|
|
char *pbuf = NULL;
|
|
int di;
|
|
unsigned long dl, rm;
|
|
Atom da;
|
|
|
|
/* we have been given the current selection, now insert it into input */
|
|
XGetWindowProperty (
|
|
display,
|
|
main_window,
|
|
netatoms[UTF8_STRING],
|
|
0,
|
|
256 / 4, // max length in words.
|
|
False, // Do not delete clipboard.
|
|
netatoms[UTF8_STRING], &da, &di, &dl, &rm, (unsigned char * *) &pbuf );
|
|
// If There was remaining data left.. lets ignore this.
|
|
// Only accept it when we get bytes!
|
|
if ( di == 8 ) {
|
|
char *index;
|
|
if ( ( index = strchr ( pbuf, '\n' ) ) != NULL ) {
|
|
// Calc new length;
|
|
dl = index - pbuf;
|
|
}
|
|
// Create a NULL terminated string. I am not sure how the data is returned.
|
|
// With or without trailing 0
|
|
char str[dl + 1];
|
|
memcpy ( str, pbuf, dl );
|
|
str[dl] = '\0';
|
|
|
|
// Insert string move cursor.
|
|
textbox_insert ( state->text, state->text->cursor, str );
|
|
textbox_cursor ( state->text, state->text->cursor + dl );
|
|
// Force a redraw and refiltering of the text.
|
|
state->update = TRUE;
|
|
state->refilter = TRUE;
|
|
}
|
|
XFree ( pbuf );
|
|
}
|
|
}
|
|
|
|
MenuReturn menu ( char **lines, unsigned int num_lines, char **input, char *prompt, Time *time,
|
|
int *shift, menu_match_cb mmc, void *mmc_data, int *selected_line, int sorting )
|
|
{
|
|
MenuState state = {
|
|
.selected_line = selected_line,
|
|
.retv = MENU_CANCEL,
|
|
.prev_key = 0,
|
|
.last_button_press = 0,
|
|
.last_offset = 0,
|
|
.num_lines = num_lines,
|
|
.distance = NULL,
|
|
.init = TRUE,
|
|
.quit = FALSE,
|
|
.filtered_lines = 0,
|
|
.max_elements = 0,
|
|
// We want to filter on the first run.
|
|
.refilter = TRUE,
|
|
.update = FALSE,
|
|
.lines = lines
|
|
};
|
|
unsigned int i;
|
|
workarea mon;
|
|
|
|
|
|
|
|
// main window isn't explicitly destroyed in case we switch modes. Reusing it prevents flicker
|
|
XWindowAttributes attr;
|
|
if ( main_window == None || XGetWindowAttributes ( display, main_window, &attr ) == 0 ) {
|
|
main_window = create_window ( display );
|
|
}
|
|
// Get active monitor size.
|
|
monitor_active ( &mon );
|
|
|
|
|
|
// we need this at this point so we can get height.
|
|
state.case_indicator = textbox_create ( main_window, &vinfo, map, TB_AUTOHEIGHT | TB_AUTOWIDTH,
|
|
( config.padding ), ( config.padding ),
|
|
0, 0,
|
|
NORMAL, "*" );
|
|
|
|
// Height of a row.
|
|
int line_height = textbox_get_height ( state.case_indicator );
|
|
if ( config.menu_lines == 0 ) {
|
|
// Autosize it.
|
|
int h = mon.h - config.padding * 2 - LINE_MARGIN - config.menu_bw * 2;
|
|
int r = ( h ) / ( line_height * config.element_height ) - 1 - config.sidebar_mode;
|
|
state.menu_lines = r;
|
|
}
|
|
else {
|
|
state.menu_lines = config.menu_lines;
|
|
}
|
|
menu_calculate_rows_columns ( &state );
|
|
menu_calculate_window_and_element_width ( &state, &mon );
|
|
|
|
// Prompt box.
|
|
state.prompt_tb = textbox_create ( main_window, &vinfo, map, TB_AUTOHEIGHT | TB_AUTOWIDTH,
|
|
( config.padding ),
|
|
( config.padding ),
|
|
0, 0, NORMAL, prompt );
|
|
// Entry box
|
|
int entrybox_width = (
|
|
( config.hmode == TRUE ) ? state.element_width : ( state.w - ( 2 * ( config.padding ) ) ) )
|
|
- textbox_get_width ( state.prompt_tb )
|
|
- textbox_get_width ( state.case_indicator );
|
|
|
|
state.text = textbox_create ( main_window, &vinfo, map, TB_AUTOHEIGHT | TB_EDITABLE,
|
|
( config.padding ) + textbox_get_width ( state.prompt_tb ),
|
|
( config.padding ),
|
|
entrybox_width, 0,
|
|
NORMAL,
|
|
( input != NULL ) ? *input : "" );
|
|
// Move indicator to end.
|
|
textbox_move ( state.case_indicator,
|
|
config.padding + textbox_get_width ( state.prompt_tb ) + entrybox_width,
|
|
0 );
|
|
|
|
textbox_show ( state.text );
|
|
textbox_show ( state.prompt_tb );
|
|
|
|
if ( config.case_sensitive ) {
|
|
textbox_show ( state.case_indicator );
|
|
}
|
|
|
|
|
|
|
|
|
|
int element_height = line_height * config.element_height;
|
|
// filtered list display
|
|
state.boxes = g_malloc0_n ( state.max_elements, sizeof ( textbox* ) );
|
|
|
|
int y_offset = config.padding + ( ( config.hmode == FALSE ) ? line_height : 0 );
|
|
int x_offset = config.padding + ( ( config.hmode == FALSE ) ? 0 : ( state.element_width + LINE_MARGIN ) );
|
|
|
|
for ( i = 0; i < state.max_elements; i++ ) {
|
|
int line = ( i ) % state.max_rows;
|
|
int col = ( i ) / state.max_rows;
|
|
|
|
int ex = col * ( state.element_width + LINE_MARGIN );
|
|
int ey = line * element_height + ( ( config.hmode == TRUE ) ? 0 : LINE_MARGIN );
|
|
|
|
state.boxes[i] = textbox_create ( main_window, &vinfo, map, 0,
|
|
ex + x_offset,
|
|
ey + y_offset,
|
|
state.element_width, element_height, NORMAL, "" );
|
|
textbox_show ( state.boxes[i] );
|
|
}
|
|
// Arrows
|
|
state.arrowbox_top = textbox_create ( main_window, &vinfo, map, TB_AUTOHEIGHT | TB_AUTOWIDTH,
|
|
( config.padding ),
|
|
( config.padding ),
|
|
0, 0,
|
|
NORMAL,
|
|
( config.hmode == FALSE ) ? "↑" : "←" );
|
|
state.arrowbox_bottom = textbox_create ( main_window, &vinfo, map, TB_AUTOHEIGHT | TB_AUTOWIDTH,
|
|
( config.padding ),
|
|
( config.padding ),
|
|
0, 0,
|
|
NORMAL,
|
|
( config.hmode == FALSE ) ? "↓" : "→" );
|
|
|
|
if ( config.hmode == FALSE ) {
|
|
textbox_move ( state.arrowbox_top,
|
|
state.w - config.padding - state.arrowbox_top->w,
|
|
config.padding + line_height + LINE_MARGIN );
|
|
textbox_move ( state.arrowbox_bottom,
|
|
state.w - config.padding - state.arrowbox_bottom->w,
|
|
config.padding + state.max_rows * element_height + LINE_MARGIN );
|
|
}
|
|
else {
|
|
textbox_move ( state.arrowbox_bottom,
|
|
state.w - config.padding - state.arrowbox_top->w,
|
|
config.padding );
|
|
textbox_move ( state.arrowbox_top,
|
|
state.w - config.padding - state.arrowbox_bottom->w - state.arrowbox_top->w,
|
|
config.padding );
|
|
}
|
|
|
|
// filtered list
|
|
state.filtered = (char * *) g_malloc0_n ( state.num_lines, sizeof ( char* ) );
|
|
state.line_map = g_malloc0_n ( state.num_lines, sizeof ( int ) );
|
|
if ( sorting ) {
|
|
state.distance = (int *) g_malloc0_n ( state.num_lines, sizeof ( int ) );
|
|
}
|
|
|
|
// resize window vertically to suit
|
|
// Subtract the margin of the last row.
|
|
state.h = line_height + element_height * state.max_rows + ( config.padding ) * 2 + LINE_MARGIN;
|
|
if ( config.hmode == TRUE ) {
|
|
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.
|
|
calculate_window_position ( &state, &mon );
|
|
|
|
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, &vinfo, map, TB_CENTER,
|
|
config.padding + j * width, state.h - line_height - config.padding,
|
|
width, line_height, ( j == curr_switcher ) ? HIGHLIGHT : NORMAL, switchers[j].name );
|
|
textbox_show ( switchers[j].tb );
|
|
}
|
|
}
|
|
|
|
// Display it.
|
|
XMoveResizeWindow ( display, main_window, state.x, state.y, state.w, state.h );
|
|
XMapRaised ( display, main_window );
|
|
|
|
// if grabbing keyboard failed, fall through
|
|
state.selected = 0;
|
|
// The cast to unsigned in here is valid, we checked if selected_line > 0.
|
|
// So its maximum range is 0-2³¹, well within the num_lines range.
|
|
if ( ( *( state.selected_line ) ) >= 0 && (unsigned int) ( *( state.selected_line ) ) <= state.num_lines ) {
|
|
state.selected = *( state.selected_line );
|
|
}
|
|
|
|
state.quit = FALSE;
|
|
menu_refilter ( &state, lines, mmc, mmc_data, sorting, config.case_sensitive );
|
|
|
|
int x11_fd = ConnectionNumber ( display );
|
|
int has_keyboard = take_keyboard ( display, main_window );
|
|
while ( !state.quit ) {
|
|
// Update if requested.
|
|
if ( state.update ) {
|
|
menu_update ( &state );
|
|
}
|
|
|
|
// Wait for event.
|
|
XEvent ev;
|
|
// Only use lazy mode above 5000 lines.
|
|
// Or if we still need to get window.
|
|
MainLoopEvent mle = ML_XEVENT;
|
|
// If we are in lazy mode, or trying to grab keyboard, go into timeout.
|
|
// Otherwise continue like we had an XEvent (and we will block on fetching this event).
|
|
if ( !has_keyboard || ( state.refilter && state.num_lines > config.lazy_filter_limit ) ) {
|
|
mle = wait_for_xevent_or_timeout ( display, x11_fd );
|
|
// Whatever happened, try to get keyboard.
|
|
has_keyboard = take_keyboard ( display, main_window );
|
|
}
|
|
// If not in lazy mode, refilter.
|
|
if ( state.num_lines <= config.lazy_filter_limit ) {
|
|
if ( state.refilter ) {
|
|
menu_refilter ( &state, lines, mmc, mmc_data, sorting, config.case_sensitive );
|
|
menu_update ( &state );
|
|
}
|
|
}
|
|
else if ( mle == ML_TIMEOUT ) {
|
|
// When timeout (and in lazy filter mode)
|
|
// We refilter then loop back and wait for Xevent.
|
|
if ( state.refilter ) {
|
|
menu_refilter ( &state, lines, mmc, mmc_data, sorting, config.case_sensitive );
|
|
menu_update ( &state );
|
|
}
|
|
}
|
|
if ( mle == ML_TIMEOUT ) {
|
|
continue;
|
|
}
|
|
// Get next event. (might block)
|
|
XNextEvent ( display, &ev );
|
|
|
|
// Handle event.
|
|
if ( ev.type == Expose ) {
|
|
while ( XCheckTypedEvent ( display, Expose, &ev ) ) {
|
|
;
|
|
}
|
|
state.update = TRUE;
|
|
}
|
|
// Button press event.
|
|
else if ( ev.type == ButtonPress ) {
|
|
while ( XCheckTypedEvent ( display, ButtonPress, &ev ) ) {
|
|
;
|
|
}
|
|
menu_mouse_navigation ( &state, &( ev.xbutton ) );
|
|
}
|
|
// Paste event.
|
|
else if ( ev.type == SelectionNotify ) {
|
|
while ( XCheckTypedEvent ( display, SelectionNotify, &ev ) ) {
|
|
;
|
|
}
|
|
menu_paste ( &state, &( ev.xselection ) );
|
|
}
|
|
// Key press event.
|
|
else if ( ev.type == KeyPress ) {
|
|
do {
|
|
if ( time ) {
|
|
*time = ev.xkey.time;
|
|
}
|
|
|
|
KeySym key = XkbKeycodeToKeysym ( display, ev.xkey.keycode, 0, 0 );
|
|
|
|
// Handling of paste
|
|
if ( ( ( ( ev.xkey.state & ControlMask ) == ControlMask ) && key == XK_v ) ) {
|
|
XConvertSelection ( display, ( ev.xkey.state & ShiftMask ) ?
|
|
XA_PRIMARY : netatoms[CLIPBOARD],
|
|
netatoms[UTF8_STRING], netatoms[UTF8_STRING], main_window, CurrentTime );
|
|
}
|
|
else if ( key == XK_Insert ) {
|
|
XConvertSelection ( display, ( ev.xkey.state & ShiftMask ) ?
|
|
XA_PRIMARY : netatoms[CLIPBOARD],
|
|
netatoms[UTF8_STRING], netatoms[UTF8_STRING], main_window, CurrentTime );
|
|
}
|
|
else if ( ( ( ev.xkey.state & ControlMask ) == ControlMask ) && key == XK_slash ) {
|
|
state.retv = MENU_PREVIOUS;
|
|
*( state.selected_line ) = 0;
|
|
state.quit = TRUE;
|
|
break;
|
|
}
|
|
// Menu navigation.
|
|
else if ( ( ( ev.xkey.state & ShiftMask ) == ShiftMask ) &&
|
|
key == XK_slash ) {
|
|
state.retv = MENU_NEXT;
|
|
*( state.selected_line ) = 0;
|
|
state.quit = TRUE;
|
|
break;
|
|
}
|
|
// Toggle case sensitivity.
|
|
else if ( key == XK_grave || key == XK_dead_grave
|
|
|| key == XK_acute ) {
|
|
config.case_sensitive = !config.case_sensitive;
|
|
*( state.selected_line ) = 0;
|
|
state.refilter = TRUE;
|
|
state.update = TRUE;
|
|
if ( config.case_sensitive ) {
|
|
textbox_show ( state.case_indicator );
|
|
}
|
|
else {
|
|
textbox_hide ( state.case_indicator );
|
|
}
|
|
}
|
|
// Switcher short-cut
|
|
else if ( ( ( ev.xkey.state & Mod1Mask ) == Mod1Mask ) &&
|
|
key >= XK_1 && key <= XK_9 ) {
|
|
*( state.selected_line ) = ( key - XK_1 );
|
|
state.retv = MENU_QUICK_SWITCH;
|
|
state.quit = TRUE;
|
|
break;
|
|
}
|
|
// Special delete entry command.
|
|
else if ( ( ( ev.xkey.state & ShiftMask ) == ShiftMask ) &&
|
|
key == XK_Delete ) {
|
|
if ( state.filtered[state.selected] != NULL ) {
|
|
*( state.selected_line ) = state.line_map[state.selected];
|
|
state.retv = MENU_ENTRY_DELETE;
|
|
state.quit = TRUE;
|
|
break;
|
|
}
|
|
}
|
|
else{
|
|
int rc = textbox_keypress ( state.text, &ev );
|
|
// Row is accepted.
|
|
if ( rc < 0 ) {
|
|
if ( shift != NULL ) {
|
|
( *shift ) = ( ( ev.xkey.state & ShiftMask ) == ShiftMask );
|
|
}
|
|
|
|
// If a valid item is selected, return that..
|
|
if ( state.selected < state.filtered_lines && state.filtered[state.selected] != NULL ) {
|
|
*( state.selected_line ) = state.line_map[state.selected];
|
|
if ( strlen ( state.text->text ) > 0 && rc == -2 ) {
|
|
state.retv = MENU_CUSTOM_INPUT;
|
|
}
|
|
else {
|
|
state.retv = MENU_OK;
|
|
}
|
|
}
|
|
else if ( strlen ( state.text->text ) > 0 ) {
|
|
state.retv = MENU_CUSTOM_INPUT;
|
|
}
|
|
else{
|
|
// Nothing entered and nothing selected.
|
|
state.retv = MENU_CANCEL;
|
|
}
|
|
|
|
state.quit = TRUE;
|
|
}
|
|
// Key press is handled by entry box.
|
|
else if ( rc > 0 ) {
|
|
state.refilter = TRUE;
|
|
state.update = TRUE;
|
|
}
|
|
// Other keys.
|
|
else{
|
|
// unhandled key
|
|
menu_keyboard_navigation ( &state, key, ev.xkey.state );
|
|
}
|
|
}
|
|
} while ( XCheckTypedEvent ( display, KeyPress, &ev ) );
|
|
}
|
|
}
|
|
|
|
release_keyboard ( display );
|
|
// Update input string.
|
|
g_free ( *input );
|
|
*input = g_strdup ( state.text->text );
|
|
|
|
int retv = state.retv;
|
|
menu_free_state ( &state );
|
|
|
|
// Free the switcher boxes.
|
|
// When state is free'ed we should no longer need these.
|
|
if ( config.sidebar_mode == TRUE ) {
|
|
for ( int j = 0; j < num_switchers; j++ ) {
|
|
textbox_free ( switchers[j].tb );
|
|
switchers[j].tb = NULL;
|
|
}
|
|
}
|
|
|
|
return retv;
|
|
}
|
|
|
|
void error_dialog ( char *msg )
|
|
{
|
|
MenuState state = {
|
|
.selected_line = NULL,
|
|
.retv = MENU_CANCEL,
|
|
.prev_key = 0,
|
|
.last_button_press = 0,
|
|
.last_offset = 0,
|
|
.num_lines = 0,
|
|
.distance = NULL,
|
|
.init = FALSE,
|
|
.quit = FALSE,
|
|
.filtered_lines = 0,
|
|
.columns = 0,
|
|
.update = TRUE,
|
|
};
|
|
workarea mon;
|
|
// Get active monitor size.
|
|
monitor_active ( &mon );
|
|
// main window isn't explicitly destroyed in case we switch modes. Reusing it prevents flicker
|
|
XWindowAttributes attr;
|
|
if ( main_window == None || XGetWindowAttributes ( display, main_window, &attr ) == 0 ) {
|
|
main_window = create_window ( display );
|
|
}
|
|
|
|
|
|
menu_calculate_window_and_element_width ( &state, &mon );
|
|
state.max_elements = 0;
|
|
|
|
state.text = textbox_create ( main_window, &vinfo, map, TB_AUTOHEIGHT,
|
|
( config.padding ),
|
|
( config.padding ),
|
|
( state.w - ( 2 * ( config.padding ) ) ),
|
|
1,
|
|
NORMAL,
|
|
( msg != NULL ) ? msg : "" );
|
|
textbox_show ( state.text );
|
|
int line_height = textbox_get_height ( state.text );
|
|
|
|
// resize window vertically to suit
|
|
state.h = line_height + ( config.padding ) * 2;
|
|
|
|
// Move the window to the correct x,y position.
|
|
calculate_window_position ( &state, &mon );
|
|
XMoveResizeWindow ( display, main_window, state.x, state.y, state.w, state.h );
|
|
|
|
// Display it.
|
|
XMapRaised ( display, main_window );
|
|
|
|
int x11_fd = ConnectionNumber ( display );
|
|
int has_keyboard = take_keyboard ( display, main_window );
|
|
while ( !state.quit ) {
|
|
// Update if requested.
|
|
if ( state.update ) {
|
|
textbox_draw ( state.text );
|
|
state.update = FALSE;
|
|
}
|
|
// Wait for event.
|
|
XEvent ev;
|
|
MainLoopEvent mle = ML_XEVENT;
|
|
if ( !has_keyboard ) {
|
|
mle = wait_for_xevent_or_timeout ( display, x11_fd );
|
|
has_keyboard = take_keyboard ( display, main_window );
|
|
}
|
|
if ( mle == ML_TIMEOUT ) {
|
|
// Loop.
|
|
continue;
|
|
}
|
|
XNextEvent ( display, &ev );
|
|
|
|
|
|
// Handle event.
|
|
if ( ev.type == Expose ) {
|
|
while ( XCheckTypedEvent ( display, Expose, &ev ) ) {
|
|
;
|
|
}
|
|
state.update = TRUE;
|
|
}
|
|
// Key press event.
|
|
else if ( ev.type == KeyPress ) {
|
|
while ( XCheckTypedEvent ( display, KeyPress, &ev ) ) {
|
|
;
|
|
}
|
|
state.quit = TRUE;
|
|
}
|
|
}
|
|
release_keyboard ( display );
|
|
}
|
|
|
|
SwitcherMode run_switcher_window ( char **input, G_GNUC_UNUSED void *data )
|
|
{
|
|
Screen *screen = DefaultScreenOfDisplay ( display );
|
|
Window root = RootWindow ( display, XScreenNumberOfScreen ( screen ) );
|
|
SwitcherMode retv = MODE_EXIT;
|
|
// find window list
|
|
Atom type;
|
|
int nwins;
|
|
Window wins[100];
|
|
int count = 0;
|
|
Window curr_win_id = 0;
|
|
|
|
// Get the active window so we can highlight this.
|
|
if ( !( window_get_prop ( root, netatoms[_NET_ACTIVE_WINDOW], &type,
|
|
&count, &curr_win_id, sizeof ( Window ) )
|
|
&& type == XA_WINDOW && count > 0 ) ) {
|
|
curr_win_id = 0;
|
|
}
|
|
|
|
if ( window_get_prop ( root, netatoms[_NET_CLIENT_LIST_STACKING],
|
|
&type, &nwins, wins, 100 * sizeof ( Window ) )
|
|
&& type == XA_WINDOW ) {
|
|
char pattern[50];
|
|
int i;
|
|
unsigned int classfield = 0;
|
|
unsigned long desktops = 0;
|
|
// windows we actually display. may be slightly different to _NET_CLIENT_LIST_STACKING
|
|
// if we happen to have a window destroyed while we're working...
|
|
winlist *ids = winlist_new ();
|
|
|
|
|
|
|
|
// calc widths of fields
|
|
for ( i = nwins - 1; i > -1; i-- ) {
|
|
client *c;
|
|
|
|
if ( ( c = window_client ( wins[i] ) )
|
|
&& !c->xattr.override_redirect
|
|
&& !client_has_state ( c, netatoms[_NET_WM_STATE_SKIP_PAGER] )
|
|
&& !client_has_state ( c, netatoms[_NET_WM_STATE_SKIP_TASKBAR] ) ) {
|
|
classfield = MAX ( classfield, strlen ( c->class ) );
|
|
|
|
#ifdef HAVE_I3_IPC_H
|
|
|
|
// In i3 mode, skip the i3bar completely.
|
|
if ( config_i3_mode && strstr ( c->class, "i3bar" ) != NULL ) {
|
|
continue;
|
|
}
|
|
|
|
#endif
|
|
if ( c->window == curr_win_id ) {
|
|
c->active = TRUE;
|
|
}
|
|
winlist_append ( ids, c->window, NULL );
|
|
}
|
|
}
|
|
|
|
// Create pattern for printing the line.
|
|
if ( !window_get_cardinal_prop ( root, netatoms[_NET_NUMBER_OF_DESKTOPS], &desktops, 1 ) ) {
|
|
desktops = 1;
|
|
}
|
|
#ifdef HAVE_I3_IPC_H
|
|
if ( config_i3_mode ) {
|
|
sprintf ( pattern, "%%-%ds %%s", MAX ( 5, classfield ) );
|
|
}
|
|
else{
|
|
#endif
|
|
sprintf ( pattern, "%%-%ds %%-%ds %%s", desktops < 10 ? 1 : 2, MAX ( 5, classfield ) );
|
|
#ifdef HAVE_I3_IPC_H
|
|
}
|
|
#endif
|
|
char **list = g_malloc0_n ( ( ids->len + 1 ), sizeof ( char* ) );
|
|
unsigned int lines = 0;
|
|
|
|
// build the actual list
|
|
for ( i = 0; i < ( ids->len ); i++ ) {
|
|
Window w = ids->array[i];
|
|
client *c;
|
|
|
|
if ( ( c = window_client ( w ) ) ) {
|
|
// final line format
|
|
unsigned long wmdesktop;
|
|
char desktop[5];
|
|
desktop[0] = 0;
|
|
char *line = g_malloc ( strlen ( c->title ) + strlen ( c->class ) + classfield + 50 );
|
|
#ifdef HAVE_I3_IPC_H
|
|
if ( !config_i3_mode ) {
|
|
#endif
|
|
// find client's desktop. this is zero-based, so we adjust by since most
|
|
// normal people don't think like this :-)
|
|
if ( !window_get_cardinal_prop ( c->window, netatoms[_NET_WM_DESKTOP], &wmdesktop, 1 ) ) {
|
|
wmdesktop = 0xFFFFFFFF;
|
|
}
|
|
|
|
if ( wmdesktop < 0xFFFFFFFF ) {
|
|
sprintf ( desktop, "%d", (int) wmdesktop + 1 );
|
|
}
|
|
|
|
sprintf ( line, pattern, desktop, c->class, c->title );
|
|
#ifdef HAVE_I3_IPC_H
|
|
}
|
|
else{
|
|
sprintf ( line, pattern, c->class, c->title );
|
|
}
|
|
#endif
|
|
|
|
list[lines++] = line;
|
|
}
|
|
}
|
|
Time time;
|
|
int selected_line = 0;
|
|
MenuReturn mretv = menu ( list, lines, input, "window:", &time, NULL,
|
|
window_match, ids, &selected_line, config.levenshtein_sort );
|
|
|
|
if ( mretv == MENU_NEXT ) {
|
|
retv = NEXT_DIALOG;
|
|
}
|
|
else if ( mretv == MENU_PREVIOUS ) {
|
|
retv = PREVIOUS_DIALOG;
|
|
}
|
|
else if ( mretv == MENU_QUICK_SWITCH ) {
|
|
retv = selected_line;
|
|
}
|
|
else if ( ( mretv == MENU_OK || mretv == MENU_CUSTOM_INPUT ) && list[selected_line] ) {
|
|
#ifdef HAVE_I3_IPC_H
|
|
|
|
if ( config_i3_mode ) {
|
|
// Hack for i3.
|
|
focus_window_i3 ( i3_socket_path, ids->array[selected_line] );
|
|
}
|
|
else
|
|
#endif
|
|
{
|
|
// Change to the desktop of the selected window/client.
|
|
// TODO: get rid of strtol
|
|
window_send_message ( root, root, netatoms[_NET_CURRENT_DESKTOP], strtol ( list[selected_line], NULL, 10 ) - 1,
|
|
SubstructureNotifyMask | SubstructureRedirectMask, time );
|
|
XSync ( display, False );
|
|
|
|
window_send_message ( root, ids->array[selected_line], netatoms[_NET_ACTIVE_WINDOW], 2, // 2 = pager
|
|
SubstructureNotifyMask | SubstructureRedirectMask, time );
|
|
}
|
|
}
|
|
|
|
g_strfreev ( list );
|
|
winlist_free ( ids );
|
|
}
|
|
|
|
return retv;
|
|
}
|
|
|
|
/**
|
|
* Start dmenu mode.
|
|
*/
|
|
static int run_dmenu ()
|
|
{
|
|
// Request truecolor visual.
|
|
create_visual_and_colormap ();
|
|
int ret_state;
|
|
textbox_setup ( &vinfo, map,
|
|
config.menu_bg, config.menu_bg_alt, config.menu_fg,
|
|
config.menu_hlbg,
|
|
config.menu_hlfg );
|
|
char *input = NULL;
|
|
|
|
// Dmenu modi has a return state.
|
|
ret_state = dmenu_switcher_dialog ( &input );
|
|
|
|
g_free ( input );
|
|
|
|
// Cleanup font setup.
|
|
textbox_cleanup ( );
|
|
|
|
if ( map != None ) {
|
|
XFreeColormap ( display, map );
|
|
}
|
|
return ret_state;
|
|
}
|
|
|
|
static void run_switcher ( int do_fork, SwitcherMode mode )
|
|
{
|
|
// we fork because it's technically possible to have multiple window
|
|
// lists up at once on a zaphod multihead X setup.
|
|
// this also happens to isolate the Xft font stuff in a child process
|
|
// that gets cleaned up every time. that library shows some valgrind
|
|
// strangeness...
|
|
if ( do_fork == TRUE ) {
|
|
if ( fork () ) {
|
|
return;
|
|
}
|
|
|
|
display = XOpenDisplay ( display_str );
|
|
XSync ( display, True );
|
|
}
|
|
|
|
create_visual_and_colormap ();
|
|
|
|
// Because of the above fork, we want to do this here.
|
|
// Make sure this is isolated to its own thread.
|
|
textbox_setup ( &vinfo, map,
|
|
config.menu_bg, config.menu_bg_alt, config.menu_fg,
|
|
config.menu_hlbg,
|
|
config.menu_hlfg );
|
|
char *input = NULL;
|
|
// Otherwise check if requested mode is enabled.
|
|
if ( switchers[mode].cb != NULL ) {
|
|
do {
|
|
SwitcherMode retv;
|
|
|
|
curr_switcher = mode;
|
|
retv = switchers[mode].cb ( &input, switchers[mode].cb_data );
|
|
// Find next enabled
|
|
if ( retv == NEXT_DIALOG ) {
|
|
mode = ( mode + 1 ) % num_switchers;
|
|
}
|
|
else if ( retv == PREVIOUS_DIALOG ) {
|
|
mode = ( mode - 1 ) % num_switchers;
|
|
if ( mode < 0 ) {
|
|
mode = num_switchers - 1;
|
|
}
|
|
}
|
|
else if ( retv == RELOAD_DIALOG ) {
|
|
// do nothing.
|
|
}
|
|
else if ( retv < MODE_EXIT ) {
|
|
mode = ( retv ) % num_switchers;
|
|
}
|
|
else {
|
|
mode = retv;
|
|
}
|
|
} while ( mode != MODE_EXIT );
|
|
}
|
|
g_free ( input );
|
|
|
|
// Cleanup font setup.
|
|
textbox_cleanup ( );
|
|
if ( map != None ) {
|
|
XFreeColormap ( display, map );
|
|
}
|
|
|
|
if ( do_fork == TRUE ) {
|
|
exit ( EXIT_SUCCESS );
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Function that listens for global key-presses.
|
|
* This is only used when in daemon mode.
|
|
*/
|
|
static void handle_keypress ( XEvent *ev )
|
|
{
|
|
KeySym key = XkbKeycodeToKeysym ( display, ev->xkey.keycode, 0, 0 );
|
|
|
|
if ( ( windows_modmask == AnyModifier || ev->xkey.state & windows_modmask ) &&
|
|
key == windows_keysym ) {
|
|
int index = switcher_get ( "window" );
|
|
if ( index >= 0 ) {
|
|
run_switcher ( TRUE, index );
|
|
}
|
|
}
|
|
|
|
if ( ( rundialog_modmask == AnyModifier || ev->xkey.state & rundialog_modmask ) &&
|
|
key == rundialog_keysym ) {
|
|
int index = switcher_get ( "run" );
|
|
if ( index >= 0 ) {
|
|
run_switcher ( TRUE, index );
|
|
}
|
|
}
|
|
|
|
if ( ( sshdialog_modmask == AnyModifier || ev->xkey.state & sshdialog_modmask ) &&
|
|
key == sshdialog_keysym ) {
|
|
int index = switcher_get ( "ssh" );
|
|
if ( index >= 0 ) {
|
|
run_switcher ( TRUE, index );
|
|
}
|
|
}
|
|
}
|
|
|
|
// convert a Mod+key arg to mod mask and keysym
|
|
static void parse_key ( char *combo, unsigned int *mod, KeySym *key )
|
|
{
|
|
unsigned int modmask = 0;
|
|
|
|
if ( strcasestr ( combo, "shift" ) ) {
|
|
modmask |= ShiftMask;
|
|
}
|
|
|
|
if ( strcasestr ( combo, "control" ) ) {
|
|
modmask |= ControlMask;
|
|
}
|
|
|
|
if ( strcasestr ( combo, "mod1" ) ) {
|
|
modmask |= Mod1Mask;
|
|
}
|
|
|
|
if ( strcasestr ( combo, "alt" ) ) {
|
|
modmask |= Mod1Mask;
|
|
}
|
|
|
|
if ( strcasestr ( combo, "mod2" ) ) {
|
|
modmask |= Mod2Mask;
|
|
}
|
|
|
|
if ( strcasestr ( combo, "mod3" ) ) {
|
|
modmask |= Mod3Mask;
|
|
}
|
|
|
|
if ( strcasestr ( combo, "mod4" ) ) {
|
|
modmask |= Mod4Mask;
|
|
}
|
|
|
|
if ( strcasestr ( combo, "mod5" ) ) {
|
|
modmask |= Mod5Mask;
|
|
}
|
|
|
|
// If no modifier mask is set, allow any modifier.
|
|
*mod = modmask ? modmask : AnyModifier;
|
|
|
|
// Skip modifier (if exist( and parse key.
|
|
char i = strlen ( combo );
|
|
|
|
while ( i > 0 && !strchr ( "-+", combo[i - 1] ) ) {
|
|
i--;
|
|
}
|
|
|
|
KeySym sym = XStringToKeysym ( combo + i );
|
|
|
|
if ( sym == NoSymbol || ( !modmask && ( strchr ( combo, '-' ) || strchr ( combo, '+' ) ) ) ) {
|
|
fprintf ( stderr, "sorry, cannot understand key combination: %s\n", combo );
|
|
exit ( EXIT_FAILURE );
|
|
}
|
|
|
|
*key = sym;
|
|
}
|
|
|
|
// bind a key combination on a root window, compensating for Lock* states
|
|
static void grab_key ( Display *display, unsigned int modmask, KeySym key )
|
|
{
|
|
Screen *screen = DefaultScreenOfDisplay ( display );
|
|
Window root = RootWindow ( display, XScreenNumberOfScreen ( screen ) );
|
|
KeyCode keycode = XKeysymToKeycode ( display, key );
|
|
XUngrabKey ( display, keycode, AnyModifier, root );
|
|
|
|
if ( modmask != AnyModifier ) {
|
|
// bind to combinations of mod and lock masks, so caps and numlock don't confuse people
|
|
XGrabKey ( display, keycode, modmask, root, True, GrabModeAsync, GrabModeAsync );
|
|
XGrabKey ( display, keycode, modmask | LockMask, root, True, GrabModeAsync, GrabModeAsync );
|
|
|
|
if ( NumlockMask ) {
|
|
XGrabKey ( display, keycode, modmask | NumlockMask, root, True, GrabModeAsync, GrabModeAsync );
|
|
XGrabKey ( display, keycode, modmask | NumlockMask | LockMask, root, True, GrabModeAsync, GrabModeAsync );
|
|
}
|
|
}
|
|
else{
|
|
// nice simple single key bind
|
|
XGrabKey ( display, keycode, AnyModifier, root, True, GrabModeAsync, GrabModeAsync );
|
|
}
|
|
}
|
|
|
|
|
|
#ifdef HAVE_I3_IPC_H
|
|
/**
|
|
* Get the i3 socket from the X root window.
|
|
*/
|
|
static inline void display_get_i3_path ( Display *display )
|
|
{
|
|
// Get the default screen.
|
|
Screen *screen = DefaultScreenOfDisplay ( display );
|
|
// Find the root window (each X has one.).
|
|
Window root = RootWindow ( display, XScreenNumberOfScreen ( screen ) );
|
|
// Get the i3 path property.
|
|
i3_socket_path = window_get_text_prop ( root, netatoms[I3_SOCKET_PATH] );
|
|
// If we find it, go into i3 mode.
|
|
config_i3_mode = ( i3_socket_path != NULL ) ? TRUE : FALSE;
|
|
}
|
|
#endif //HAVE_I3_IPC_H
|
|
|
|
|
|
/**
|
|
* Help function. This calls man.
|
|
*/
|
|
static void help ()
|
|
{
|
|
int code = execlp ( "man", "man", MANPAGE_PATH, NULL );
|
|
|
|
if ( code == -1 ) {
|
|
fprintf ( stderr, "Failed to execute man: %s\n", strerror ( errno ) );
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Parse commandline options.
|
|
*/
|
|
static void parse_cmd_options ( int argc, char ** argv )
|
|
{
|
|
// catch help request
|
|
if ( find_arg ( argc, argv, "-h" ) >= 0 ||
|
|
find_arg ( argc, argv, "-help" ) >= 0 ) {
|
|
help ();
|
|
exit ( EXIT_SUCCESS );
|
|
}
|
|
|
|
if ( find_arg ( argc, argv, "-v" ) >= 0 ||
|
|
find_arg ( argc, argv, "-version" ) >= 0 ) {
|
|
fprintf ( stdout, "Version: "VERSION "\n" );
|
|
exit ( EXIT_SUCCESS );
|
|
}
|
|
|
|
find_arg_str_alloc ( argc, argv, "-pid", &( pidfile ) );
|
|
|
|
find_arg_str ( argc, argv, "-switchers", &( config.switchers ) );
|
|
// Parse commandline arguments about the looks.
|
|
find_arg_uint ( argc, argv, "-opacity", &( config.window_opacity ) );
|
|
|
|
find_arg_int ( argc, argv, "-width", &( config.menu_width ) );
|
|
|
|
find_arg_uint ( argc, argv, "-lines", &( config.menu_lines ) );
|
|
find_arg_uint ( argc, argv, "-columns", &( config.menu_columns ) );
|
|
|
|
find_arg_str ( argc, argv, "-font", &( config.menu_font ) );
|
|
find_arg_str ( argc, argv, "-fg", &( config.menu_fg ) );
|
|
find_arg_str ( argc, argv, "-bg", &( config.menu_bg ) );
|
|
find_arg_str ( argc, argv, "-bgalt", &( config.menu_bg_alt ) );
|
|
find_arg_str ( argc, argv, "-hlfg", &( config.menu_hlfg ) );
|
|
find_arg_str ( argc, argv, "-hlbg", &( config.menu_hlbg ) );
|
|
find_arg_str ( argc, argv, "-bc", &( config.menu_bc ) );
|
|
find_arg_uint ( argc, argv, "-bw", &( config.menu_bw ) );
|
|
|
|
// Parse commandline arguments about size and position
|
|
find_arg_uint ( argc, argv, "-location", &( config.location ) );
|
|
find_arg_uint ( argc, argv, "-padding", &( config.padding ) );
|
|
find_arg_int ( argc, argv, "-xoffset", &( config.x_offset ) );
|
|
find_arg_int ( argc, argv, "-yoffset", &( config.y_offset ) );
|
|
if ( find_arg ( argc, argv, "-fixed-num-lines" ) >= 0 ) {
|
|
config.fixed_num_lines = 1;
|
|
}
|
|
if ( find_arg ( argc, argv, "-disable-history" ) >= 0 ) {
|
|
config.disable_history = TRUE;
|
|
}
|
|
if ( find_arg ( argc, argv, "-levenshtein-sort" ) >= 0 ) {
|
|
config.levenshtein_sort = TRUE;
|
|
}
|
|
if ( find_arg ( argc, argv, "-case-sensitive" ) >= 0 ) {
|
|
config.case_sensitive = TRUE;
|
|
}
|
|
|
|
// Parse commandline arguments about behavior
|
|
find_arg_str ( argc, argv, "-terminal", &( config.terminal_emulator ) );
|
|
|
|
if ( find_arg ( argc, argv, "-hmode" ) >= 0 ) {
|
|
config.hmode = TRUE;
|
|
}
|
|
|
|
find_arg_str ( argc, argv, "-ssh-client", &( config.ssh_client ) );
|
|
find_arg_str ( argc, argv, "-ssh-command", &( config.ssh_command ) );
|
|
find_arg_str ( argc, argv, "-run-command", &( config.run_command ) );
|
|
find_arg_str ( argc, argv, "-run-list-command", &( config.run_list_command ) );
|
|
find_arg_str ( argc, argv, "-run-shell-command", &( config.run_shell_command ) );
|
|
|
|
// Keybindings
|
|
find_arg_str ( argc, argv, "-key", &( config.window_key ) );
|
|
find_arg_str ( argc, argv, "-rkey", &( config.run_key ) );
|
|
find_arg_str ( argc, argv, "-skey", &( config.ssh_key ) );
|
|
|
|
|
|
|
|
find_arg_char ( argc, argv, "-sep", &( config.separator ) );
|
|
|
|
|
|
find_arg_int ( argc, argv, "-eh", &( config.element_height ) );
|
|
|
|
find_arg_int ( argc, argv, "-lazy-filter-limit", &( config.lazy_filter_limit ) );
|
|
|
|
if ( find_arg ( argc, argv, "-sidebar-mode" ) >= 0 ) {
|
|
config.sidebar_mode = TRUE;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Function bound by 'atexit'.
|
|
* Cleanup globally allocated memory.
|
|
*/
|
|
static void cleanup ()
|
|
{
|
|
// Cleanup
|
|
if ( display != NULL ) {
|
|
if ( main_window != None ) {
|
|
XFreeGC ( display, gc );
|
|
XDestroyWindow ( display, main_window );
|
|
XCloseDisplay ( display );
|
|
}
|
|
}
|
|
if ( cache_xattr != NULL ) {
|
|
winlist_free ( cache_xattr );
|
|
}
|
|
if ( cache_client != NULL ) {
|
|
winlist_free ( cache_client );
|
|
}
|
|
#ifdef HAVE_I3_IPC_H
|
|
|
|
if ( i3_socket_path != NULL ) {
|
|
g_free ( i3_socket_path );
|
|
}
|
|
|
|
#endif
|
|
|
|
// Cleaning up memory allocated by the Xresources file.
|
|
// TODO, not happy with this.
|
|
parse_xresource_free ();
|
|
|
|
for ( unsigned int i = 0; i < num_switchers; i++ ) {
|
|
// only used for script dialog.
|
|
if ( switchers[i].cb_data != NULL ) {
|
|
script_switcher_free_options ( switchers[i].cb_data );
|
|
}
|
|
}
|
|
g_free ( switchers );
|
|
|
|
// Cleanup pid file.
|
|
if ( pidfile ) {
|
|
unlink ( pidfile );
|
|
g_free ( pidfile );
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Do some input validation, especially the first few could break things.
|
|
* It is good to catch them beforehand.
|
|
*
|
|
* This functions exits the program with 1 when it finds an invalid configuration.
|
|
*/
|
|
static void config_sanity_check ( void )
|
|
{
|
|
if ( config.element_height < 1 ) {
|
|
fprintf ( stderr, "config.element_height is invalid. It needs to be atleast 1 line high.\n" );
|
|
exit ( 1 );
|
|
}
|
|
if ( config.menu_columns == 0 ) {
|
|
fprintf ( stderr, "config.menu_columns is invalid. You need at least one visible column.\n" );
|
|
exit ( 1 );
|
|
}
|
|
if ( config.menu_width == 0 ) {
|
|
fprintf ( stderr, "config.menu_width is invalid. You cannot have a window with no width.\n" );
|
|
exit ( 1 );
|
|
}
|
|
if ( !( config.location >= WL_CENTER && config.location <= WL_WEST ) ) {
|
|
fprintf ( stderr, "config.location is invalid. ( %d >= %d >= %d) does not hold.\n",
|
|
WL_WEST, config.location, WL_CENTER );
|
|
exit ( 1 );
|
|
}
|
|
if ( !( config.hmode == TRUE || config.hmode == FALSE ) ) {
|
|
fprintf ( stderr, "config.hmode is invalid.\n" );
|
|
exit ( 1 );
|
|
}
|
|
// If alternative row is not set, copy the normal background color.
|
|
if ( config.menu_bg_alt == NULL ) {
|
|
config.menu_bg_alt = config.menu_bg;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Parse the switcher string, into internal array of type Switcher.
|
|
*
|
|
* String is split on separator ','
|
|
* First the three build-in modi are checked: window, run, ssh
|
|
* if that fails, a script-switcher is created.
|
|
*/
|
|
static void setup_switchers ( void )
|
|
{
|
|
char *savept = NULL;
|
|
// Make a copy, as strtok will modify it.
|
|
char *switcher_str = g_strdup ( config.switchers );
|
|
// Split token on ','. This modifies switcher_str.
|
|
for ( char *token = strtok_r ( switcher_str, ",", &savept );
|
|
token != NULL;
|
|
token = strtok_r ( NULL, ",", &savept ) ) {
|
|
// Window switcher.
|
|
if ( strcasecmp ( token, "window" ) == 0 ) {
|
|
// Resize and add entry.
|
|
switchers = (Switcher *) g_realloc ( switchers, sizeof ( Switcher ) * ( num_switchers + 1 ) );
|
|
g_strlcpy ( switchers[num_switchers].name, "window", 32 );
|
|
switchers[num_switchers].cb = run_switcher_window;
|
|
switchers[num_switchers].cb_data = NULL;
|
|
num_switchers++;
|
|
}
|
|
// SSh dialog
|
|
else if ( strcasecmp ( token, "ssh" ) == 0 ) {
|
|
// Resize and add entry.
|
|
switchers = (Switcher *) g_realloc ( switchers, sizeof ( Switcher ) * ( num_switchers + 1 ) );
|
|
g_strlcpy ( switchers[num_switchers].name, "ssh", 32 );
|
|
switchers[num_switchers].cb = ssh_switcher_dialog;
|
|
switchers[num_switchers].cb_data = NULL;
|
|
num_switchers++;
|
|
}
|
|
// Run dialog
|
|
else if ( strcasecmp ( token, "run" ) == 0 ) {
|
|
// Resize and add entry.
|
|
switchers = (Switcher *) g_realloc ( switchers, sizeof ( Switcher ) * ( num_switchers + 1 ) );
|
|
g_strlcpy ( switchers[num_switchers].name, "run", 32 );
|
|
switchers[num_switchers].cb = run_switcher_dialog;
|
|
switchers[num_switchers].cb_data = NULL;
|
|
num_switchers++;
|
|
}
|
|
else {
|
|
// If not build in, use custom switchers.
|
|
ScriptOptions *sw = script_switcher_parse_setup ( token );
|
|
if ( sw != NULL ) {
|
|
// Resize and add entry.
|
|
switchers = (Switcher *) g_realloc ( switchers, sizeof ( Switcher ) * ( num_switchers + 1 ) );
|
|
g_strlcpy ( switchers[num_switchers].name, sw->name, 32 );
|
|
switchers[num_switchers].cb = script_switcher_dialog;
|
|
switchers[num_switchers].cb_data = sw;
|
|
num_switchers++;
|
|
}
|
|
else{
|
|
// Report error, don't continue.
|
|
fprintf ( stderr, "Invalid script switcher: %s\n", token );
|
|
token = NULL;
|
|
}
|
|
}
|
|
}
|
|
// Free string that was modified by strtok_r
|
|
g_free ( switcher_str );
|
|
}
|
|
|
|
/**
|
|
* Keep a copy of arc, argv around, so we can use the same parsing method
|
|
*/
|
|
static int stored_argc;
|
|
static char **stored_argv;
|
|
|
|
/**
|
|
* @param display Pointer to the X connection to use.
|
|
* Load configuration.
|
|
* Following priority: (current), X, commandline arguments
|
|
*/
|
|
static inline void load_configuration ( Display *display )
|
|
{
|
|
// Load in config from X resources.
|
|
parse_xresource_options ( display );
|
|
|
|
// Parse command line for settings.
|
|
parse_cmd_options ( stored_argc, stored_argv );
|
|
|
|
// Sanity check
|
|
config_sanity_check ();
|
|
}
|
|
|
|
/**
|
|
* Handle sighub request.
|
|
* Currently we just reload the configuration.
|
|
*/
|
|
static void hup_action_handler ( int num )
|
|
{
|
|
/**
|
|
* Open new connection to X. It seems the XResources do not get updated
|
|
* on the old connection.
|
|
*/
|
|
Display *display = XOpenDisplay ( display_str );
|
|
if ( display ) {
|
|
load_configuration ( display );
|
|
XCloseDisplay ( display );
|
|
}
|
|
}
|
|
/**
|
|
* @param pidfile The pidfile to create.
|
|
*/
|
|
static void create_pid_file ( const char *pidfile )
|
|
{
|
|
if ( pidfile == NULL ) {
|
|
return;
|
|
}
|
|
|
|
int fd = open ( pidfile, O_RDWR | O_CREAT, S_IRUSR | S_IWUSR );
|
|
if ( fd < 0 ) {
|
|
fprintf ( stderr, "Failed to create pid file." );
|
|
exit ( 1 );
|
|
}
|
|
// Set it to close the File Descriptor on exit.
|
|
int flags = fcntl ( fd, F_GETFD, NULL );
|
|
flags = flags | FD_CLOEXEC;
|
|
if ( fcntl ( fd, F_SETFD, flags, NULL ) < 0 ) {
|
|
fprintf ( stderr, "Failed to set CLOEXEC on pidfile." );
|
|
close ( fd );
|
|
exit ( 1 );
|
|
}
|
|
// Try to get exclusive write lock on FD
|
|
int retv = flock ( fd, LOCK_EX | LOCK_NB );
|
|
if ( retv != 0 ) {
|
|
fprintf ( stderr, "Failed to set lock on pidfile: Rofi already running?\n" );
|
|
fprintf ( stderr, "%d %s\n", retv, strerror ( errno ) );
|
|
exit ( 1 );
|
|
}
|
|
ftruncate ( fd, (off_t) 0 );
|
|
|
|
// Write pid.
|
|
char buffer[64];
|
|
int length = snprintf ( buffer, 64, "%i", getpid () );
|
|
write ( fd, buffer, length );
|
|
}
|
|
|
|
int main ( int argc, char *argv[] )
|
|
{
|
|
stored_argc = argc;
|
|
stored_argv = argv;
|
|
|
|
// Get the path to the cache dir.
|
|
cache_dir = g_get_user_cache_dir ();
|
|
|
|
// Create pid file path.
|
|
const char *path = g_get_user_runtime_dir ();
|
|
if ( path ) {
|
|
pidfile = g_build_filename ( path, "rofi.pid", NULL );
|
|
}
|
|
|
|
// Register cleanup function.
|
|
atexit ( cleanup );
|
|
|
|
// Get DISPLAY
|
|
display_str = getenv ( "DISPLAY" );
|
|
find_arg_str ( argc, argv, "-display", &display_str );
|
|
|
|
if ( !( display = XOpenDisplay ( display_str ) ) ) {
|
|
fprintf ( stderr, "cannot open display!\n" );
|
|
return EXIT_FAILURE;
|
|
}
|
|
|
|
|
|
load_configuration ( display );
|
|
|
|
// Dump.
|
|
if ( find_arg ( argc, argv, "-dump-xresources" ) >= 0 ) {
|
|
xresource_dump ();
|
|
exit ( EXIT_SUCCESS );
|
|
}
|
|
|
|
// Create pid file
|
|
if ( pidfile != NULL ) {
|
|
create_pid_file ( pidfile );
|
|
}
|
|
|
|
// setup_switchers
|
|
setup_switchers ();
|
|
|
|
// Set up X interaction.
|
|
signal ( SIGCHLD, catch_exit );
|
|
|
|
// Set error handle
|
|
XSync ( display, False );
|
|
xerror = XSetErrorHandler ( display_oops );
|
|
XSync ( display, False );
|
|
|
|
// determine numlock mask so we can bind on keys with and without it
|
|
XModifierKeymap *modmap = XGetModifierMapping ( display );
|
|
KeyCode kc = XKeysymToKeycode ( display, XK_Num_Lock );
|
|
for ( int i = 0; i < 8; i++ ) {
|
|
for ( int j = 0; j < ( int ) modmap->max_keypermod; j++ ) {
|
|
if ( modmap->modifiermap[i * modmap->max_keypermod + j] == kc ) {
|
|
NumlockMask = ( 1 << i );
|
|
}
|
|
}
|
|
}
|
|
XFreeModifiermap ( modmap );
|
|
|
|
cache_client = winlist_new ();
|
|
cache_xattr = winlist_new ();
|
|
|
|
// X atom values
|
|
for ( int i = 0; i < NUM_NETATOMS; i++ ) {
|
|
netatoms[i] = XInternAtom ( display, netatom_names[i], False );
|
|
}
|
|
|
|
#ifdef HAVE_I3_IPC_H
|
|
// Check for i3
|
|
display_get_i3_path ( display );
|
|
#endif
|
|
|
|
char *msg = NULL;
|
|
if ( find_arg_str ( argc, argv, "-e", &( msg ) ) ) {
|
|
// Request truecolor visual.
|
|
create_visual_and_colormap ();
|
|
textbox_setup ( &vinfo, map,
|
|
config.menu_bg, config.menu_bg_alt, config.menu_fg,
|
|
config.menu_hlbg,
|
|
config.menu_hlfg );
|
|
error_dialog ( msg );
|
|
textbox_cleanup ( );
|
|
if ( map != None ) {
|
|
XFreeColormap ( display, map );
|
|
}
|
|
exit ( EXIT_SUCCESS );
|
|
}
|
|
|
|
|
|
// flags to run immediately and exit
|
|
char *sname = NULL;
|
|
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_int ( argc, argv, "-l", &dmenu_selected_line );
|
|
int retv = run_dmenu ();
|
|
|
|
// User canceled the operation.
|
|
if ( retv == FALSE ) {
|
|
return EXIT_FAILURE;
|
|
}
|
|
}
|
|
else if ( find_arg_str ( argc, argv, "-show", &sname ) == TRUE ) {
|
|
int index = switcher_get ( sname );
|
|
if ( index >= 0 ) {
|
|
run_switcher ( FALSE, index );
|
|
}
|
|
else {
|
|
fprintf ( stderr, "The %s switcher has not been enabled\n", sname );
|
|
}
|
|
}
|
|
// Old modi.
|
|
else if ( find_arg ( argc, argv, "-now" ) >= 0 ) {
|
|
int index = switcher_get ( "window" );
|
|
if ( index >= 0 ) {
|
|
run_switcher ( FALSE, index );
|
|
}
|
|
else {
|
|
fprintf ( stderr, "The window switcher has not been enabled\n" );
|
|
}
|
|
}
|
|
else if ( find_arg ( argc, argv, "-rnow" ) >= 0 ) {
|
|
int index = switcher_get ( "run" );
|
|
if ( index >= 0 ) {
|
|
run_switcher ( FALSE, index );
|
|
}
|
|
else {
|
|
fprintf ( stderr, "The run dialog has not been enabled\n" );
|
|
}
|
|
}
|
|
else if ( find_arg ( argc, argv, "-snow" ) >= 0 ) {
|
|
int index = switcher_get ( "ssh" );
|
|
if ( index >= 0 ) {
|
|
run_switcher ( FALSE, index );
|
|
}
|
|
else {
|
|
fprintf ( stderr, "The ssh dialog has not been enabled\n" );
|
|
}
|
|
}
|
|
else{
|
|
// Daemon mode, Listen to key presses..
|
|
if ( switcher_get ( "window" ) >= 0 ) {
|
|
parse_key ( config.window_key, &windows_modmask, &windows_keysym );
|
|
grab_key ( display, windows_modmask, windows_keysym );
|
|
}
|
|
|
|
if ( switcher_get ( "run" ) >= 0 ) {
|
|
parse_key ( config.run_key, &rundialog_modmask, &rundialog_keysym );
|
|
grab_key ( display, rundialog_modmask, rundialog_keysym );
|
|
}
|
|
|
|
if ( switcher_get ( "ssh" ) >= 0 ) {
|
|
parse_key ( config.ssh_key, &sshdialog_modmask, &sshdialog_keysym );
|
|
grab_key ( display, sshdialog_modmask, sshdialog_keysym );
|
|
}
|
|
|
|
// Setup handler for sighub (reload config)
|
|
const struct sigaction hup_action = { hup_action_handler, };
|
|
sigaction ( SIGHUP, &hup_action, NULL );
|
|
|
|
|
|
// Application Main loop.
|
|
// This listens in the background for any events on the Xserver
|
|
// catching global key presses.
|
|
for (;; ) {
|
|
XEvent ev;
|
|
// caches only live for a single event
|
|
winlist_empty ( cache_xattr );
|
|
winlist_empty ( cache_client );
|
|
|
|
// block and wait for something
|
|
XNextEvent ( display, &ev );
|
|
// If we get an event that does not belong to a window:
|
|
// Ignore it.
|
|
if ( ev.xany.window == None ) {
|
|
continue;
|
|
}
|
|
// If keypress, handle it.
|
|
if ( ev.type == KeyPress ) {
|
|
handle_keypress ( &ev );
|
|
}
|
|
}
|
|
}
|
|
|
|
return EXIT_SUCCESS;
|
|
}
|