1
0
Fork 0
mirror of https://github.com/davatorium/rofi.git synced 2024-11-03 04:23:42 -05:00
rofi/source/rofi.c

2034 lines
64 KiB
C
Raw Normal View History

/**
2014-03-01 11:27:52 -05:00
* rofi
2014-01-10 04:35:38 -05:00
*
* MIT/X11 License
* Copyright (c) 2012 Sean Pringle <sean.pringle@gmail.com>
* Modified 2013-2014 Qball Cow <qball@gmpclient.org>
2014-01-10 04:35:38 -05:00
*
* 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:
2014-01-10 04:35:38 -05:00
*
* The above copyright notice and this permission notice shall be
* included in all copies or substantial portions of the Software.
2014-01-10 04:35:38 -05:00
*
* 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.
2014-01-10 04:35:38 -05:00
*
*/
2012-06-29 13:10:13 -04:00
#define _GNU_SOURCE
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <stdarg.h>
#include <unistd.h>
#include <ctype.h>
#include <signal.h>
#include <sys/wait.h>
#include <fcntl.h>
#include <err.h>
2014-01-21 08:56:25 -05:00
#include <errno.h>
2014-03-16 08:47:44 -04:00
#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>
2012-06-29 13:10:13 -04:00
#include <X11/extensions/Xinerama.h>
2014-03-18 05:55:25 -04:00
#include "rofi.h"
2014-03-18 05:38:30 -04:00
#ifdef HAVE_I3_IPC_H
#include <errno.h>
#include <linux/un.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <i3/ipc.h>
#endif
#include <basedir.h>
#include "run-dialog.h"
2014-01-21 04:01:55 -05:00
#include "ssh-dialog.h"
2014-01-29 18:47:23 -05:00
#include "dmenu-dialog.h"
2012-06-29 13:10:13 -04:00
#include "xrmoptions.h"
2014-03-22 16:04:19 -04:00
#define LINE_MARGIN 4
2014-05-20 05:22:03 -04:00
#ifdef HAVE_I3_IPC_H
2014-05-20 05:41:39 -04:00
#define I3_SOCKET_PATH_PROP "I3_SOCKET_PATH"
2014-05-20 05:22:03 -04:00
// This setting is no longer user configurable, but partial to this file:
int config_i3_mode = 0;
2014-05-20 05:47:05 -04:00
// Path to HAVE_I3_IPC_H socket.
char *i3_socket_path = NULL;
2014-05-20 05:22:03 -04:00
#endif
2012-08-23 21:28:39 -04:00
2014-06-04 15:29:23 -04:00
xdgHandle xdg_handle;
const char *cache_dir = NULL;
char *active_font = NULL;
unsigned int NumlockMask = 0;
Display *display = NULL;
2014-02-03 16:49:07 -05:00
/**
* Shared 'token_match' function.
* Matches tokenized.
*/
int token_match ( char **tokens, const char *input,
2014-03-22 16:04:19 -04:00
__attribute__( ( unused ) ) int index,
__attribute__( ( unused ) ) void *data )
2014-02-03 16:49:07 -05:00
{
int match = 1;
2014-02-03 16:49:07 -05:00
// Do a tokenized match.
2014-06-04 15:29:23 -04:00
if ( tokens ) {
for ( int j = 1; match && tokens[j]; j++ ) {
2014-03-22 16:04:19 -04:00
match = ( strcasestr ( input, tokens[j] ) != NULL );
2014-02-03 16:49:07 -05:00
}
2014-03-22 16:04:19 -04:00
}
2014-02-03 16:49:07 -05:00
return match;
}
2014-03-22 16:04:19 -04:00
static char **tokenize ( const char *input )
2013-02-04 03:32:10 -05:00
{
2014-06-04 15:29:23 -04:00
if ( input == NULL ) {
2014-03-22 16:04:19 -04:00
return NULL;
}
2014-01-10 04:35:38 -05:00
2013-02-04 03:32:10 -05:00
char *saveptr = NULL, *token;
2014-03-22 16:04:19 -04:00
char **retv = NULL;
2013-02-04 03:32:10 -05:00
// First entry is always full (modified) stringtext.
2014-03-22 16:04:19 -04:00
int num_tokens = 1;
2013-02-04 03:32:10 -05:00
//First entry is string that is modified.
retv = malloc ( 2 * sizeof ( char* ) );
2014-03-22 16:04:19 -04:00
retv[0] = strdup ( input );
2013-02-04 03:32:10 -05:00
retv[1] = NULL;
// Iterate over tokens.
2014-02-25 03:13:34 -05:00
// strtok should still be valid for utf8.
2014-06-04 15:29:23 -04:00
for ( token = strtok_r ( retv[0], " ", &saveptr );
token != NULL;
token = strtok_r ( NULL, " ", &saveptr ) ) {
char **tr = realloc ( retv, sizeof ( char* ) * ( num_tokens + 2 ) );
2014-06-04 15:29:23 -04:00
if ( tr != NULL ) {
2014-05-26 04:32:26 -04:00
retv = tr;
retv[num_tokens + 1] = NULL;
retv[num_tokens] = token;
num_tokens++;
}
2013-02-04 03:32:10 -05:00
}
return retv;
}
2014-03-22 16:04:19 -04:00
static inline void tokenize_free ( char **ip )
2013-02-04 03:32:10 -05:00
{
2014-06-04 15:29:23 -04:00
if ( ip == NULL ) {
2014-03-22 16:04:19 -04:00
return;
}
2014-01-10 04:35:38 -05:00
2014-05-22 03:33:32 -04:00
free ( ip[0] );
2014-03-22 16:04:19 -04:00
free ( ip );
2013-02-04 03:32:10 -05:00
}
2014-03-18 05:38:30 -04:00
#ifdef HAVE_I3_IPC_H
// Focus window on HAVE_I3_IPC_H window manager.
2014-03-22 16:04:19 -04:00
static void focus_window_i3 ( const char *socket_path, int id )
{
2014-03-22 16:04:19 -04:00
i3_ipc_header_t head;
char command[128];
int s, t, len;
struct sockaddr_un remote;
2014-06-04 15:29:23 -04:00
if ( strlen ( socket_path ) > UNIX_PATH_MAX ) {
2014-03-22 16:04:19 -04:00
fprintf ( stderr, "Socket path is to long. %zd > %d\n", strlen ( socket_path ), UNIX_PATH_MAX );
return;
}
2014-06-04 15:29:23 -04:00
if ( ( s = socket ( AF_UNIX, SOCK_STREAM, 0 ) ) == -1 ) {
2014-03-22 16:04:19 -04:00
fprintf ( stderr, "Failed to open connection to I3: %s\n", strerror ( errno ) );
return;
}
remote.sun_family = AF_UNIX;
2014-03-22 16:04:19 -04:00
strcpy ( remote.sun_path, socket_path );
len = strlen ( remote.sun_path ) + sizeof ( remote.sun_family );
2014-06-04 15:29:23 -04:00
if ( connect ( s, ( struct sockaddr * ) &remote, len ) == -1 ) {
2014-03-22 16:04:19 -04:00
fprintf ( stderr, "Failed to connect to I3 (%s): %s\n", socket_path, strerror ( errno ) );
close ( s );
return;
}
2014-02-03 17:07:04 -05:00
// Formulate command
2014-03-22 16:04:19 -04:00
snprintf ( command, 128, "[id=\"%d\"] focus", id );
2014-01-26 06:59:10 -05:00
// Prepare header.
2014-03-22 16:04:19 -04:00
memcpy ( head.magic, I3_IPC_MAGIC, 6 );
head.size = strlen ( command );
2014-01-26 06:59:10 -05:00
head.type = I3_IPC_MESSAGE_TYPE_COMMAND;
// Send header.
2014-03-22 16:04:19 -04:00
send ( s, &head, sizeof ( head ), 0 );
2014-01-26 06:59:10 -05:00
// Send message
2014-03-22 16:04:19 -04:00
send ( s, command, strlen ( command ), 0 );
2014-01-26 06:59:10 -05:00
// Receive header.
2014-03-22 16:04:19 -04:00
t = recv ( s, &head, sizeof ( head ), 0 );
2014-01-26 06:59:10 -05:00
2014-06-04 15:29:23 -04:00
if ( t == sizeof ( head ) ) {
2014-03-22 16:04:19 -04:00
recv ( s, command, head.size, 0 );
2014-01-26 06:59:10 -05:00
}
2014-03-22 16:04:19 -04:00
close ( s );
}
#endif
2014-03-22 16:04:19 -04:00
void catch_exit ( __attribute__( ( unused ) ) int sig )
2012-06-29 13:10:13 -04:00
{
2014-06-04 15:29:23 -04:00
while ( 0 < waitpid ( -1, NULL, WNOHANG ) ) {
2014-03-22 16:04:19 -04:00
;
}
2012-06-29 13:10:13 -04:00
}
2014-01-16 12:47:19 -05:00
2012-06-29 13:10:13 -04:00
// cli arg handling
2014-03-22 16:04:19 -04:00
static int find_arg ( const int argc, char * const argv[], const char * const key )
2012-06-29 13:10:13 -04:00
{
2014-01-10 04:35:38 -05:00
int i;
2014-06-04 15:29:23 -04:00
for ( i = 0; i < argc && strcasecmp ( argv[i], key ); i++ ) {
2014-03-22 16:04:19 -04:00
;
}
2014-01-10 04:35:38 -05:00
2014-03-22 16:04:19 -04:00
return i < argc ? i : -1;
2012-06-29 13:10:13 -04:00
}
2014-03-22 16:04:19 -04:00
static void find_arg_str ( int argc, char *argv[], char *key, char** val )
2012-06-29 13:10:13 -04:00
{
2014-03-22 16:04:19 -04:00
int i = find_arg ( argc, argv, key );
2014-01-23 05:39:12 -05:00
2014-06-04 15:29:23 -04:00
if ( val != NULL && i > 0 && i < argc - 1 ) {
2014-03-22 16:04:19 -04:00
*val = argv[i + 1];
2014-01-23 05:39:12 -05:00
}
2012-06-29 13:10:13 -04:00
}
2014-01-23 05:39:12 -05:00
static void find_arg_int ( int argc, char *argv[], char *key, unsigned int *val )
2012-06-29 13:10:13 -04:00
{
2014-03-22 16:04:19 -04:00
int i = find_arg ( argc, argv, key );
2014-01-23 05:39:12 -05:00
2014-06-04 15:29:23 -04:00
if ( val != NULL && i > 0 && i < ( argc - 1 ) ) {
2014-03-22 16:04:19 -04:00
*val = strtol ( argv[i + 1], NULL, 10 );
2014-01-23 05:39:12 -05:00
}
2012-06-29 13:10:13 -04:00
}
2014-03-22 16:04:19 -04:00
2014-05-26 04:32:26 -04:00
static int ( *xerror )( Display *, XErrorEvent * );
2014-03-22 16:04:19 -04:00
#define ATOM_ENUM( x ) x
#define ATOM_CHAR( x ) # x
2014-06-04 15:29:23 -04:00
#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 ), \
2014-06-05 02:11:01 -04:00
X ( _NET_WM_DESKTOP ), \
X ( _NET_WM_WINDOW_OPACITY )
2014-03-22 16:04:19 -04:00
enum { EWMH_ATOMS ( ATOM_ENUM ), NETATOMS };
const char *netatom_names[] = { EWMH_ATOMS ( ATOM_CHAR ) };
Atom netatoms[NETATOMS];
2012-06-29 13:10:13 -04:00
// X error handler
2014-05-20 05:41:39 -04:00
static int display_oops ( __attribute__( ( unused ) ) Display *d, XErrorEvent *ee )
2012-06-29 13:10:13 -04:00
{
2014-01-10 04:35:38 -05:00
if ( ee->error_code == BadWindow
|| ( ee->request_code == X_GrabButton && ee->error_code == BadAccess )
|| ( ee->request_code == X_GrabKey && ee->error_code == BadAccess )
2014-06-04 15:29:23 -04:00
) {
2014-03-22 16:04:19 -04:00
return 0;
}
2014-01-10 04:35:38 -05:00
2014-03-22 16:04:19 -04:00
fprintf ( stderr, "error: request code=%d, error code=%d\n", ee->request_code, ee->error_code );
return xerror ( display, ee );
2012-06-29 13:10:13 -04:00
}
// usable space on a monitor
2014-03-22 16:04:19 -04:00
typedef struct
{
2014-01-10 04:35:38 -05:00
int x, y, w, h;
int l, r, t, b;
2012-06-29 13:10:13 -04:00
} workarea;
2014-01-20 18:20:09 -05:00
// window lists
2014-03-22 16:04:19 -04:00
typedef struct
{
2014-01-20 18:20:09 -05:00
Window *array;
2014-03-22 16:04:19 -04:00
void **data;
int len;
2014-01-20 18:20:09 -05:00
} winlist;
2014-05-19 15:02:05 -04:00
winlist *cache_client = NULL;
winlist *cache_xattr = NULL;
2012-06-29 13:10:13 -04:00
2014-03-22 16:04:19 -04:00
#define winlist_ascend( l, i, w ) for ( ( i ) = 0; ( i ) < ( l )->len && ( ( ( w ) = ( l )->array[i] ) || 1 ); ( i )++ )
#define winlist_descend( l, i, w ) for ( ( i ) = ( l )->len - 1; ( i ) >= 0 && ( ( ( w ) = ( l )->array[i] ) || 1 ); ( i )-- )
2012-06-29 13:10:13 -04:00
2014-03-22 16:04:19 -04:00
#define WINLIST 32
2012-06-29 13:10:13 -04:00
2014-03-22 16:04:19 -04:00
winlist* winlist_new ()
2012-06-29 13:10:13 -04:00
{
winlist *l = malloc ( sizeof ( winlist ) );
2014-03-22 16:04:19 -04:00
l->len = 0;
l->array = malloc ( sizeof ( Window ) * ( WINLIST + 1 ) );
l->data = malloc ( sizeof ( void* ) * ( WINLIST + 1 ) );
2014-01-10 04:35:38 -05:00
return l;
2012-06-29 13:10:13 -04:00
}
2014-03-22 16:04:19 -04:00
int winlist_append ( winlist *l, Window w, void *d )
2012-06-29 13:10:13 -04:00
{
2014-06-04 15:29:23 -04:00
if ( l->len > 0 && !( l->len % WINLIST ) ) {
l->array = realloc ( l->array, sizeof ( Window ) * ( l->len + WINLIST + 1 ) );
l->data = realloc ( l->data, sizeof ( void* ) * ( l->len + WINLIST + 1 ) );
2014-01-10 04:35:38 -05:00
}
2014-03-11 15:16:44 -04:00
// Make clang-check happy.
// TODO: make clang-check clear this should never be 0.
2014-06-04 15:29:23 -04:00
if ( l->data == NULL || l->array == NULL ) {
2014-03-22 16:04:19 -04:00
return 0;
}
2014-01-10 04:35:38 -05:00
2014-03-22 16:04:19 -04:00
l->data[l->len] = d;
2014-01-10 04:35:38 -05:00
l->array[l->len++] = w;
2014-03-22 16:04:19 -04:00
return l->len - 1;
2012-06-29 13:10:13 -04:00
}
2014-03-22 16:04:19 -04:00
void winlist_empty ( winlist *l )
2012-06-29 13:10:13 -04:00
{
2014-06-04 15:29:23 -04:00
while ( l->len > 0 ) {
2014-03-22 16:04:19 -04:00
free ( l->data[--( l->len )] );
}
2012-06-29 13:10:13 -04:00
}
2014-03-22 16:04:19 -04:00
void winlist_free ( winlist *l )
2012-06-29 13:10:13 -04:00
{
2014-03-22 16:04:19 -04:00
winlist_empty ( l );
free ( l->array );
free ( l->data );
free ( l );
2012-06-29 13:10:13 -04:00
}
2014-03-22 16:04:19 -04:00
int winlist_find ( winlist *l, Window w )
2012-06-29 13:10:13 -04:00
{
// iterate backwards. theory is: windows most often accessed will be
// nearer the end. testing with kcachegrind seems to support this...
2014-03-22 16:04:19 -04:00
int i;
2014-02-26 16:48:49 -05:00
Window o = 0;
2014-01-10 04:35:38 -05:00
2014-06-04 15:29:23 -04:00
winlist_descend ( l, i, o ) if ( w == o ) {
2014-03-22 16:04:19 -04:00
return i;
}
2014-01-10 04:35:38 -05:00
return -1;
2012-06-29 13:10:13 -04:00
}
2014-03-22 16:04:19 -04:00
#define CLIENTTITLE 100
#define CLIENTCLASS 50
#define CLIENTNAME 50
#define CLIENTSTATE 10
#define CLIENTROLE 50
2012-06-29 13:10:13 -04:00
// a managable window
2014-03-22 16:04:19 -04:00
typedef struct
{
Window window, trans;
2014-01-10 04:35:38 -05:00
XWindowAttributes xattr;
2014-03-22 16:04:19 -04:00
char title[CLIENTTITLE];
char class[CLIENTCLASS];
char name[CLIENTNAME];
char role[CLIENTROLE];
int states;
Atom state[CLIENTSTATE];
workarea monitor;
int active;
2012-06-29 13:10:13 -04:00
} client;
2014-01-23 05:39:12 -05:00
2012-07-18 08:14:13 -04:00
// malloc a pixel value for an X named color
2014-03-22 16:04:19 -04:00
static unsigned int color_get ( const char *const name )
2012-06-29 13:10:13 -04:00
{
2014-05-26 04:32:26 -04:00
int screen_id = DefaultScreen ( display );
2014-03-22 16:04:19 -04:00
XColor color;
Colormap map = DefaultColormap ( display, screen_id );
return XAllocNamedColor ( display, map, name, &color, &color ) ? color.pixel : None;
2012-06-29 13:10:13 -04:00
}
// find mouse pointer location
2014-03-22 16:04:19 -04:00
int pointer_get ( Window root, int *x, int *y )
2012-06-29 13:10:13 -04:00
{
2014-01-10 04:35:38 -05:00
*x = 0;
*y = 0;
2014-03-22 16:04:19 -04:00
Window rr, cr;
int rxr, ryr, wxr, wyr;
2014-01-10 04:35:38 -05:00
unsigned int mr;
2014-06-04 15:29:23 -04:00
if ( XQueryPointer ( display, root, &rr, &cr, &rxr, &ryr, &wxr, &wyr, &mr ) ) {
2014-01-10 04:35:38 -05:00
*x = rxr;
*y = ryr;
return 1;
}
return 0;
2012-06-29 13:10:13 -04:00
}
static int take_keyboard ( Window w )
2012-07-18 08:14:13 -04:00
{
2014-01-10 04:35:38 -05:00
int i;
2014-06-04 15:29:23 -04:00
for ( i = 0; i < 1000; i++ ) {
if ( XGrabKeyboard ( display, w, True, GrabModeAsync, GrabModeAsync, CurrentTime ) == GrabSuccess ) {
2014-01-10 04:35:38 -05:00
return 1;
2014-03-22 16:04:19 -04:00
}
2014-01-10 04:35:38 -05:00
2014-05-26 09:37:33 -04:00
struct timespec rsl = { 0, 100000L };
2014-03-22 16:04:19 -04:00
nanosleep ( &rsl, NULL );
2014-01-10 04:35:38 -05:00
}
return 0;
2012-07-18 08:14:13 -04:00
}
2014-03-22 16:04:19 -04:00
void release_keyboard ()
2012-07-18 08:14:13 -04:00
{
2014-03-22 16:04:19 -04:00
XUngrabKeyboard ( display, CurrentTime );
2012-07-18 08:14:13 -04:00
}
2012-06-29 13:10:13 -04:00
// XGetWindowAttributes with caching
2014-03-22 16:04:19 -04:00
XWindowAttributes* window_get_attributes ( Window w )
2012-06-29 13:10:13 -04:00
{
2014-03-22 16:04:19 -04:00
int idx = winlist_find ( cache_xattr, w );
2014-01-10 04:35:38 -05:00
2014-06-04 15:29:23 -04:00
if ( idx < 0 ) {
XWindowAttributes *cattr = malloc ( sizeof ( XWindowAttributes ) );
2014-01-10 04:35:38 -05:00
2014-06-04 15:29:23 -04:00
if ( XGetWindowAttributes ( display, w, cattr ) ) {
2014-03-22 16:04:19 -04:00
winlist_append ( cache_xattr, w, cattr );
2014-01-10 04:35:38 -05:00
return cattr;
}
2014-03-22 16:04:19 -04:00
free ( cattr );
2014-01-10 04:35:38 -05:00
return NULL;
}
return cache_xattr->data[idx];
2012-06-29 13:10:13 -04:00
}
// retrieve a property of any type from a window
2014-03-22 16:04:19 -04:00
int window_get_prop ( Window w, Atom prop, Atom *type, int *items, void *buffer, unsigned int bytes )
2012-06-29 13:10:13 -04:00
{
2014-01-10 04:35:38 -05:00
Atom _type;
2014-06-04 15:29:23 -04:00
if ( !type ) {
2014-03-22 16:04:19 -04:00
type = &_type;
}
2014-01-10 04:35:38 -05:00
int _items;
2014-06-04 15:29:23 -04:00
if ( !items ) {
2014-03-22 16:04:19 -04:00
items = &_items;
}
2014-01-10 04:35:38 -05:00
2014-03-22 16:04:19 -04:00
int format;
2014-01-10 04:35:38 -05:00
unsigned long nitems, nbytes;
unsigned char *ret = NULL;
2014-03-22 16:04:19 -04:00
memset ( buffer, 0, bytes );
if ( XGetWindowProperty ( display, w, prop, 0, bytes / 4, False, AnyPropertyType, type,
2014-06-04 15:29:23 -04:00
&format, &nitems, &nbytes, &ret ) == Success && ret && *type != None && format ) {
if ( format == 8 ) {
2014-03-22 16:04:19 -04:00
memmove ( buffer, ret, MIN ( bytes, nitems ) );
}
2014-01-10 04:35:38 -05:00
2014-06-04 15:29:23 -04:00
if ( format == 16 ) {
2014-03-22 16:04:19 -04:00
memmove ( buffer, ret, MIN ( bytes, nitems * sizeof ( short ) ) );
}
2014-01-10 04:35:38 -05:00
2014-06-04 15:29:23 -04:00
if ( format == 32 ) {
2014-03-22 16:04:19 -04:00
memmove ( buffer, ret, MIN ( bytes, nitems * sizeof ( long ) ) );
}
2014-01-10 04:35:38 -05:00
2014-03-22 16:04:19 -04:00
*items = ( int ) nitems;
XFree ( ret );
2014-01-10 04:35:38 -05:00
return 1;
}
return 0;
2012-06-29 13:10:13 -04:00
}
// retrieve a text property from a window
// technically we could use window_get_prop(), but this is better for character set support
2014-03-22 16:04:19 -04:00
char* window_get_text_prop ( Window w, Atom atom )
2012-06-29 13:10:13 -04:00
{
2014-01-10 04:35:38 -05:00
XTextProperty prop;
2014-03-22 16:04:19 -04:00
char *res = NULL;
char **list = NULL;
int count;
2014-06-04 15:29:23 -04:00
if ( XGetTextProperty ( display, w, &prop, atom ) && prop.value && prop.nitems ) {
if ( prop.encoding == XA_STRING ) {
res = malloc ( strlen ( ( char * ) prop.value ) + 1 );
2014-03-11 15:16:44 -04:00
// make clang-check happy.
2014-06-04 15:29:23 -04:00
if ( res ) {
2014-03-22 16:04:19 -04:00
strcpy ( res, ( char * ) prop.value );
2014-03-11 15:16:44 -04:00
}
2014-03-22 16:04:19 -04:00
}
2014-06-04 15:29:23 -04:00
else if ( Xutf8TextPropertyToTextList ( display, &prop, &list, &count ) >= Success && count > 0 && *list ) {
res = malloc ( strlen ( *list ) + 1 );
2014-03-11 15:16:44 -04:00
// make clang-check happy.
2014-06-04 15:29:23 -04:00
if ( res ) {
2014-03-22 16:04:19 -04:00
strcpy ( res, *list );
2014-03-11 15:16:44 -04:00
}
2014-03-22 16:04:19 -04:00
XFreeStringList ( list );
2014-01-10 04:35:38 -05:00
}
}
2014-06-04 15:29:23 -04:00
if ( prop.value ) {
2014-03-22 16:04:19 -04:00
XFree ( prop.value );
}
2014-01-10 04:35:38 -05:00
return res;
2012-06-29 13:10:13 -04:00
}
2014-03-22 16:04:19 -04:00
int window_get_atom_prop ( Window w, Atom atom, Atom *list, int count )
2012-06-29 13:10:13 -04:00
{
2014-01-10 04:35:38 -05:00
Atom type;
2014-03-22 16:04:19 -04:00
int items;
return window_get_prop ( w, atom, &type, &items, list, count * sizeof ( Atom ) ) && type == XA_ATOM ? items : 0;
2012-06-29 13:10:13 -04:00
}
2014-03-22 16:04:19 -04:00
void window_set_atom_prop ( Window w, Atom prop, Atom *atoms, int count )
2012-06-29 13:10:13 -04:00
{
2014-03-22 16:04:19 -04:00
XChangeProperty ( display, w, prop, XA_ATOM, 32, PropModeReplace, ( unsigned char * ) atoms, count );
2012-06-29 13:10:13 -04:00
}
2014-03-22 16:04:19 -04:00
int window_get_cardinal_prop ( Window w, Atom atom, unsigned long *list, int count )
{
Atom type; int items;
2014-03-22 16:04:19 -04:00
return window_get_prop ( w, atom, &type, &items, list, count * sizeof ( unsigned long ) ) && type == XA_CARDINAL ? items : 0;
}
2012-06-29 13:10:13 -04:00
// a ClientMessage
2014-03-22 16:04:19 -04:00
int window_send_message ( Window target, Window subject, Atom atom, unsigned long protocol, unsigned long mask, Time time )
2012-06-29 13:10:13 -04:00
{
2014-01-10 04:35:38 -05:00
XEvent e;
2014-03-22 16:04:19 -04:00
memset ( &e, 0, sizeof ( XEvent ) );
e.xclient.type = ClientMessage;
2014-01-10 04:35:38 -05:00
e.xclient.message_type = atom;
2014-03-22 16:04:19 -04:00
e.xclient.window = subject;
2014-01-10 04:35:38 -05:00
e.xclient.data.l[0] = protocol;
2014-03-22 16:04:19 -04:00
e.xclient.data.l[1] = time;
2014-01-10 04:35:38 -05:00
e.xclient.send_event = True;
2014-03-22 16:04:19 -04:00
e.xclient.format = 32;
int r = XSendEvent ( display, target, False, mask, &e ) ? 1 : 0;
XFlush ( display );
2014-01-10 04:35:38 -05:00
return r;
2012-06-29 13:10:13 -04:00
}
// find the dimensions of the monitor displaying point x,y
2014-03-22 16:04:19 -04:00
void monitor_dimensions ( Screen *screen, int x, int y, workarea *mon )
2012-06-29 13:10:13 -04:00
{
2014-03-22 16:04:19 -04:00
memset ( mon, 0, sizeof ( workarea ) );
mon->w = WidthOfScreen ( screen );
mon->h = HeightOfScreen ( screen );
2014-01-10 04:35:38 -05:00
// locate the current monitor
2014-06-04 15:29:23 -04:00
if ( XineramaIsActive ( display ) ) {
2014-03-22 16:04:19 -04:00
int monitors;
XineramaScreenInfo *info = XineramaQueryScreens ( display, &monitors );
2014-06-04 15:29:23 -04:00
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 ) ) {
2014-01-10 04:35:38 -05:00
mon->x = info[i].x_org;
mon->y = info[i].y_org;
mon->w = info[i].width;
mon->h = info[i].height;
break;
}
}
2014-02-27 14:26:35 -05:00
}
2014-01-10 04:35:38 -05:00
2014-03-22 16:04:19 -04:00
XFree ( info );
2014-01-10 04:35:38 -05:00
}
2012-06-29 13:10:13 -04:00
}
// determine which monitor holds the active window, or failing that the mouse pointer
2014-03-22 16:04:19 -04:00
void monitor_active ( workarea *mon )
2012-06-29 13:10:13 -04:00
{
2014-05-26 04:32:26 -04:00
Screen *screen = DefaultScreenOfDisplay ( display );
Window root = RootWindow ( display, XScreenNumberOfScreen ( screen ) );
2014-06-04 15:29:23 -04:00
int x, y;
2014-01-10 04:35:38 -05:00
2014-06-04 15:29:23 -04:00
if ( pointer_get ( root, &x, &y ) ) {
2014-03-22 16:04:19 -04:00
monitor_dimensions ( screen, x, y, mon );
2014-01-10 04:35:38 -05:00
return;
}
2014-03-22 16:04:19 -04:00
monitor_dimensions ( screen, 0, 0, mon );
2012-06-29 13:10:13 -04:00
}
// _NET_WM_STATE_*
2014-03-22 16:04:19 -04:00
int client_has_state ( client *c, Atom state )
2012-06-29 13:10:13 -04:00
{
2014-01-10 04:35:38 -05:00
int i;
2014-06-04 15:29:23 -04:00
for ( i = 0; i < c->states; i++ ) {
if ( c->state[i] == state ) {
2014-03-22 16:04:19 -04:00
return 1;
}
}
2014-01-10 04:35:38 -05:00
return 0;
2012-06-29 13:10:13 -04:00
}
// collect info on any window
// doesn't have to be a window we'll end up managing
2014-03-22 16:04:19 -04:00
client* window_client ( Window win )
2012-06-29 13:10:13 -04:00
{
2014-06-04 15:29:23 -04:00
if ( win == None ) {
2014-03-22 16:04:19 -04:00
return NULL;
}
2014-01-10 04:35:38 -05:00
2014-03-22 16:04:19 -04:00
int idx = winlist_find ( cache_client, win );
2014-06-04 15:29:23 -04:00
if ( idx >= 0 ) {
2014-01-18 17:25:09 -05:00
return cache_client->data[idx];
}
2014-01-10 04:35:38 -05:00
2014-05-24 05:02:13 -04:00
// if this fails, we're up that creek
2014-03-22 16:04:19 -04:00
XWindowAttributes *attr = window_get_attributes ( win );
2014-01-10 04:35:38 -05:00
2014-06-04 15:29:23 -04:00
if ( !attr ) {
2014-03-22 16:04:19 -04:00
return NULL;
}
2014-01-10 04:35:38 -05:00
client *c = calloc ( 1, sizeof ( client ) );
2014-01-10 04:35:38 -05:00
c->window = win;
2014-05-24 05:02:13 -04:00
// copy xattr so we don't have to care when stuff is freed
2014-03-22 16:04:19 -04:00
memmove ( &c->xattr, attr, sizeof ( XWindowAttributes ) );
XGetTransientForHint ( display, win, &c->trans );
2014-01-10 04:35:38 -05:00
2014-03-22 16:04:19 -04:00
c->states = window_get_atom_prop ( win, netatoms[_NET_WM_STATE], c->state, CLIENTSTATE );
2014-01-10 04:35:38 -05:00
char *name;
2014-06-04 15:29:23 -04:00
if ( ( name = window_get_text_prop ( c->window, netatoms[_NET_WM_NAME] ) ) && name ) {
2014-03-22 16:04:19 -04:00
snprintf ( c->title, CLIENTTITLE, "%s", name );
free ( name );
}
2014-06-04 15:29:23 -04:00
else if ( XFetchName ( display, c->window, &name ) ) {
2014-03-22 16:04:19 -04:00
snprintf ( c->title, CLIENTTITLE, "%s", name );
XFree ( name );
2014-01-10 04:35:38 -05:00
}
2014-03-22 16:04:19 -04:00
name = window_get_text_prop ( c->window, XInternAtom ( display, "WM_WINDOW_ROLE", False ) );
2014-01-18 17:12:45 -05:00
2014-06-04 15:29:23 -04:00
if ( name != NULL ) {
2014-03-22 16:04:19 -04:00
snprintf ( c->role, CLIENTROLE, "%s", name );
XFree ( name );
2014-01-18 17:12:45 -05:00
}
2014-01-10 04:35:38 -05:00
XClassHint chint;
2014-06-04 15:29:23 -04:00
if ( XGetClassHint ( display, c->window, &chint ) ) {
2014-03-22 16:04:19 -04:00
snprintf ( c->class, CLIENTCLASS, "%s", chint.res_class );
snprintf ( c->name, CLIENTNAME, "%s", chint.res_name );
XFree ( chint.res_class );
XFree ( chint.res_name );
2014-01-10 04:35:38 -05:00
}
2014-03-22 16:04:19 -04:00
monitor_dimensions ( c->xattr.screen, c->xattr.x, c->xattr.y, &c->monitor );
winlist_append ( cache_client, c->window, c );
2014-01-10 04:35:38 -05:00
return c;
2012-06-29 13:10:13 -04:00
}
2014-01-10 04:35:38 -05:00
unsigned int windows_modmask;
2014-03-22 16:04:19 -04:00
KeySym windows_keysym;
2014-01-14 00:36:12 -05:00
unsigned int rundialog_modmask;
2014-03-22 16:04:19 -04:00
KeySym rundialog_keysym;
2014-01-21 04:13:42 -05:00
unsigned int sshdialog_modmask;
2014-03-22 16:04:19 -04:00
KeySym sshdialog_keysym;
2014-01-26 09:39:50 -05:00
2014-03-22 16:04:19 -04:00
Window main_window = None;
GC gc = NULL;
#include "textbox.h"
2014-05-14 14:56:38 -04:00
2014-05-19 15:58:13 -04:00
void menu_hide_arrow_text ( int filtered_lines, int selected, int max_elements,
textbox *arrowbox_top, textbox *arrowbox_bottom )
2014-05-14 14:56:38 -04:00
{
2014-06-04 15:29:23 -04:00
if ( arrowbox_top == NULL || arrowbox_bottom == NULL ) {
2014-05-19 15:58:13 -04:00
return;
2014-05-14 14:56:38 -04:00
}
2014-05-19 15:58:13 -04:00
int page = ( filtered_lines > 0 ) ? selected / max_elements : 0;
int npages = ( filtered_lines > 0 ) ? ( ( filtered_lines + max_elements - 1 ) / max_elements ) : 1;
2014-06-04 15:29:23 -04:00
if ( !( page != 0 && npages > 1 ) ) {
2014-05-19 15:58:13 -04:00
textbox_hide ( arrowbox_top );
}
2014-06-04 15:29:23 -04:00
if ( !( ( npages - 1 ) != page && npages > 1 ) ) {
2014-05-19 15:58:13 -04:00
textbox_hide ( arrowbox_bottom );
2014-05-14 14:56:38 -04:00
}
}
2014-05-19 15:58:13 -04:00
void menu_set_arrow_text ( int filtered_lines, int selected, int max_elements,
textbox *arrowbox_top, textbox *arrowbox_bottom )
{
2014-06-04 15:29:23 -04:00
if ( arrowbox_top == NULL || arrowbox_bottom == NULL ) {
2014-05-19 15:58:13 -04:00
return;
}
2014-06-04 15:29:23 -04:00
if ( filtered_lines == 0 || max_elements == 0 ) {
2014-06-02 15:51:22 -04:00
return;
}
2014-05-19 15:58:13 -04:00
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;
2014-06-04 15:29:23 -04:00
if ( page != 0 && npages > 1 ) {
2014-05-19 15:58:13 -04:00
textbox_show ( arrowbox_top );
2014-05-25 17:58:05 -04:00
textbox_font ( arrowbox_top, ( entry != 0 ) ? NORMAL : HIGHLIGHT );
2014-05-19 15:58:13 -04:00
textbox_draw ( arrowbox_top );
2014-05-14 14:56:38 -04:00
}
2014-06-04 15:29:23 -04:00
if ( ( npages - 1 ) != page && npages > 1 ) {
2014-05-19 15:58:13 -04:00
textbox_show ( arrowbox_bottom );
2014-05-25 17:58:05 -04:00
textbox_font ( arrowbox_bottom, ( entry != ( max_elements - 1 ) ) ? NORMAL : HIGHLIGHT );
2014-05-19 15:58:13 -04:00
textbox_draw ( arrowbox_bottom );
2014-05-14 14:56:38 -04:00
}
}
2012-07-18 08:14:13 -04:00
void menu_draw ( textbox **boxes,
2014-05-19 03:50:09 -04:00
int max_elements,
2014-04-15 03:14:00 -04:00
int num_lines,
int *last_offset,
int selected,
char **filtered )
2012-06-29 13:10:13 -04:00
{
2014-03-22 16:04:19 -04:00
int i, offset = 0;
2014-01-10 04:35:38 -05:00
2014-01-31 07:41:08 -05:00
// selected row is always visible.
// If selected is visible do not scroll.
2014-06-04 15:29:23 -04:00
if ( ( selected - ( *last_offset ) ) < ( max_elements ) && ( selected - ( *last_offset ) ) >= 0 ) {
2014-03-22 16:04:19 -04:00
offset = *last_offset;
}
2014-06-04 15:29:23 -04:00
else{
2014-04-28 15:18:13 -04:00
// Do paginating
2014-05-19 15:58:13 -04:00
int page = ( max_elements > 0 ) ? ( selected / max_elements ) : 0;
2014-05-19 03:50:09 -04:00
offset = page * max_elements;
2014-01-31 07:41:08 -05:00
*last_offset = offset;
}
2014-06-04 15:29:23 -04:00
for ( i = 0; i < max_elements; i++ ) {
if ( ( i + offset ) >= num_lines || filtered[i + offset] == NULL ) {
textbox_font ( boxes[i], NORMAL );
2014-03-22 16:04:19 -04:00
textbox_text ( boxes[i], "" );
}
2014-06-04 15:29:23 -04:00
else{
char *text = filtered[i + offset];
TextBoxFontType tbft = ( i + offset ) == selected ? HIGHLIGHT : NORMAL;
// Check for active
2014-06-04 15:29:23 -04:00
if ( text[0] == '*' ) {
// Skip the '*'
text++;
// Use the active version of font.
tbft = ( tbft == HIGHLIGHT ) ? ACTIVE_HIGHLIGHT : ACTIVE;
}
textbox_font ( boxes[i], tbft );
textbox_text ( boxes[i], text );
2014-01-13 07:36:41 -05:00
}
2014-03-22 16:04:19 -04:00
textbox_draw ( boxes[i] );
2014-01-10 04:35:38 -05:00
}
2012-06-29 13:10:13 -04:00
}
/* Very bad implementation of tab completion.
2014-03-22 16:04:19 -04:00
* It will complete to the common prefix
*/
2014-05-19 03:50:09 -04:00
static int calculate_common_prefix ( char **filtered, int max_elements )
{
2014-02-27 14:26:35 -05:00
int length_prefix = 0;
2014-01-10 04:35:38 -05:00
2014-06-04 15:29:23 -04:00
if ( filtered && filtered[0] != NULL ) {
2014-03-22 16:04:19 -04:00
int found = 1;
char *p = filtered[0];
2014-01-10 04:35:38 -05:00
2014-06-04 15:29:23 -04:00
do {
found = 1;
2014-01-10 04:35:38 -05:00
2014-06-04 15:29:23 -04:00
for ( int j = 0; j < max_elements && filtered[j] != NULL; j++ ) {
if ( filtered[j][length_prefix] == '\0' || filtered[j][length_prefix] != *p ) {
if ( found ) {
2014-03-22 16:04:19 -04:00
found = 0;
}
2014-01-10 04:35:38 -05:00
break;
}
}
2014-01-10 04:35:38 -05:00
2014-06-04 15:29:23 -04:00
if ( found ) {
length_prefix++;
2014-03-22 16:04:19 -04:00
}
2014-01-10 04:35:38 -05:00
p++;
2014-01-10 04:35:38 -05:00
} while ( found );
2014-02-25 03:13:34 -05:00
// cut off to be valid utf8.
2014-06-04 15:29:23 -04:00
for ( int j = 0; j < length_prefix; ) {
if ( ( filtered[0][j] & 0x80 ) == 0 ) {
2014-03-22 16:04:19 -04:00
j++;
}
2014-06-04 15:29:23 -04:00
else if ( ( filtered[0][j] & 0xf0 ) == 0xc0 ) {
2014-02-25 03:13:34 -05:00
// 2 bytes
2014-06-04 15:29:23 -04:00
if ( ( j + 2 ) >= length_prefix ) {
2014-03-22 16:04:19 -04:00
length_prefix = j;
}
j += 2;
}
2014-06-04 15:29:23 -04:00
else if ( ( filtered[0][j] & 0xf0 ) == 0xe0 ) {
2014-02-25 03:13:34 -05:00
// 3 bytes
2014-06-04 15:29:23 -04:00
if ( ( j + 3 ) >= length_prefix ) {
2014-03-22 16:04:19 -04:00
length_prefix = j;
}
j += 3;
}
2014-06-04 15:29:23 -04:00
else if ( ( filtered[0][j] & 0xf0 ) == 0xf0 ) {
2014-02-25 03:13:34 -05:00
// 4 bytes
2014-06-04 15:29:23 -04:00
if ( ( j + 4 ) >= length_prefix ) {
2014-03-22 16:04:19 -04:00
length_prefix = j;
}
j += 4;
}
2014-06-04 15:29:23 -04:00
else{
2014-03-22 16:04:19 -04:00
j++;
};
2014-02-25 03:13:34 -05:00
}
}
return length_prefix;
}
2014-01-20 18:20:09 -05:00
2014-03-22 16:04:19 -04:00
int window_match ( char **tokens, __attribute__( ( unused ) ) const char *input, int index, void *data )
2014-01-20 18:20:09 -05:00
{
2014-03-22 16:04:19 -04:00
int match = 1;
winlist *ids = ( winlist * ) data;
client *c = window_client ( ids->array[index] );
2014-06-04 15:29:23 -04:00
if ( tokens ) {
for ( int j = 1; match && tokens[j]; j++ ) {
2014-01-21 04:13:42 -05:00
int test = 0;
2014-01-20 18:20:09 -05:00
2014-06-04 15:29:23 -04:00
if ( !test && c->title[0] != '\0' ) {
2014-03-22 16:04:19 -04:00
test = ( strcasestr ( c->title, tokens[j] ) != NULL );
}
2014-01-20 18:20:09 -05:00
2014-06-04 15:29:23 -04:00
if ( !test && c->class[0] != '\0' ) {
2014-03-22 16:04:19 -04:00
test = ( strcasestr ( c->class, tokens[j] ) != NULL );
}
2014-01-20 18:20:09 -05:00
2014-06-04 15:29:23 -04:00
if ( !test && c->role[0] != '\0' ) {
2014-03-22 16:04:19 -04:00
test = ( strcasestr ( c->role, tokens[j] ) != NULL );
}
2014-01-20 18:20:09 -05:00
2014-06-04 15:29:23 -04:00
if ( !test && c->name[0] != '\0' ) {
2014-03-22 16:04:19 -04:00
test = ( strcasestr ( c->name, tokens[j] ) != NULL );
}
2014-01-20 18:20:09 -05:00
2014-06-04 15:29:23 -04:00
if ( test == 0 ) {
2014-03-22 16:04:19 -04:00
match = 0;
}
2014-01-21 04:13:42 -05:00
}
2014-03-22 16:04:19 -04:00
}
2014-01-20 18:20:09 -05:00
return match;
2014-01-21 04:13:42 -05:00
}
2014-01-20 18:20:09 -05:00
2014-03-22 16:04:19 -04:00
MenuReturn menu ( char **lines, char **input, char *prompt, Time *time, int *shift,
menu_match_cb mmc, void *mmc_data, int *selected_line )
2012-06-29 13:10:13 -04:00
{
2014-03-22 16:04:19 -04:00
int retv = MENU_CANCEL;
unsigned int i, j;
unsigned int columns = config.menu_columns;
2014-03-22 16:04:19 -04:00
workarea mon;
2014-05-19 03:50:09 -04:00
2014-03-22 16:04:19 -04:00
unsigned int num_lines = 0;
int last_offset = 0;
2014-06-04 15:29:23 -04:00
for (; lines != NULL && lines[num_lines]; num_lines++ ) {
2014-03-22 16:04:19 -04:00
;
}
2014-01-10 04:35:38 -05:00
2014-05-17 16:17:07 -04:00
unsigned int max_elements = MIN ( config.menu_lines * columns, num_lines );
2014-05-19 03:50:09 -04:00
// TODO, clean this up.
// 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.
2014-05-19 15:58:13 -04:00
unsigned int max_rows = MIN ( config.menu_lines,
(unsigned int) (
2014-05-24 10:23:01 -04:00
( num_lines + ( columns - num_lines % columns ) % columns ) /
( columns )
2014-05-19 15:58:13 -04:00
) );
2014-01-10 04:35:38 -05:00
2014-06-04 15:29:23 -04:00
if ( config.fixed_num_lines == TRUE ) {
max_elements = config.menu_lines * columns;
2014-05-19 15:58:13 -04:00
max_rows = config.menu_lines;
// If it would fit in one column, only use one column.
2014-06-04 15:29:23 -04:00
if ( num_lines < max_elements ) {
2014-05-24 10:23:01 -04:00
columns = ( num_lines + ( max_rows - num_lines % max_rows ) % max_rows ) / max_rows;
max_elements = config.menu_lines * columns;
}
// Sanitize.
2014-06-04 15:29:23 -04:00
if ( columns == 0 ) {
columns = 1;
}
2014-05-19 03:50:09 -04:00
}
// More hacks.
2014-06-04 15:29:23 -04:00
if ( config.hmode == TRUE ) {
2014-05-19 03:50:09 -04:00
max_rows = 1;
2014-05-17 16:17:07 -04:00
}
2014-05-19 03:50:09 -04:00
// Get active monitor size.
monitor_active ( &mon );
2014-01-26 06:59:10 -05:00
// Calculate as float to stop silly, big rounding down errors.
2014-03-22 16:04:19 -04:00
int w = config.menu_width < 101 ? ( mon.w / 100.0f ) * ( float ) config.menu_width : config.menu_width;
int x = mon.x + ( mon.w - w ) / 2;
int element_width = w - ( 2 * ( config.padding ) );
2014-05-19 03:50:09 -04:00
// Divide by the # columns
element_width /= columns;
2014-06-04 15:29:23 -04:00
if ( config.hmode == TRUE ) {
2014-05-19 03:50:09 -04:00
element_width = ( w - ( 2 * ( config.padding ) ) - max_elements * LINE_MARGIN ) / ( max_elements + 1 );
2014-01-26 06:59:10 -05:00
}
2014-01-10 04:35:38 -05:00
2014-03-22 16:04:19 -04:00
Window box;
2014-01-10 04:35:38 -05:00
XWindowAttributes attr;
// main window isn't explicitly destroyed in case we switch modes. Reusing it prevents flicker
2014-06-04 15:29:23 -04:00
if ( main_window != None && XGetWindowAttributes ( display, main_window, &attr ) ) {
2014-01-10 04:35:38 -05:00
box = main_window;
2014-03-22 16:04:19 -04:00
}
2014-06-04 15:29:23 -04:00
else{
2014-05-26 04:32:26 -04:00
Screen *screen = DefaultScreenOfDisplay ( display );
Window root = RootWindow ( display, XScreenNumberOfScreen ( screen ) );
2014-03-22 16:04:19 -04:00
box = XCreateSimpleWindow ( display, root, x, 0, w, 300,
config.menu_bw, color_get ( config.menu_bc ),
color_get ( config.menu_bg ) );
XSelectInput ( display, box, ExposureMask );
2014-01-22 07:07:52 -05:00
2014-03-22 16:04:19 -04:00
gc = XCreateGC ( display, box, 0, 0 );
XSetLineAttributes ( display, gc, 2, LineOnOffDash, CapButt, JoinMiter );
XSetForeground ( display, gc, color_get ( config.menu_bc ) );
2014-01-10 04:35:38 -05:00
// make it an unmanaged window
2014-03-22 16:04:19 -04:00
window_set_atom_prop ( box, netatoms[_NET_WM_STATE], &netatoms[_NET_WM_STATE_ABOVE], 1 );
2014-01-10 04:35:38 -05:00
XSetWindowAttributes sattr;
sattr.override_redirect = True;
2014-03-22 16:04:19 -04:00
XChangeWindowAttributes ( display, box, CWOverrideRedirect, &sattr );
2014-01-10 04:35:38 -05:00
main_window = box;
2013-01-20 07:15:44 -05:00
// Set the WM_NAME
2014-03-22 16:04:19 -04:00
XStoreName ( display, box, "rofi" );
// Hack to set window opacity.
2014-06-05 02:11:01 -04:00
unsigned int opacity_set = ( unsigned int ) ( ( config.window_opacity / 100.0 ) * UINT32_MAX);
XChangeProperty ( display, box, netatoms[_NET_WM_WINDOW_OPACITY],
2014-03-22 16:04:19 -04:00
XA_CARDINAL, 32, PropModeReplace,
( unsigned char * ) &opacity_set, 1L );
}
2012-08-23 21:28:39 -04:00
2014-01-10 04:35:38 -05:00
// search text input
2014-05-14 14:56:38 -04:00
textbox *prompt_tb = textbox_create ( box, TB_AUTOHEIGHT | TB_AUTOWIDTH,
( config.padding ),
( config.padding ),
0, 0,
NORMAL,
prompt );
2014-03-22 16:04:19 -04:00
textbox *text = textbox_create ( box, TB_AUTOHEIGHT | TB_EDITABLE,
( config.padding ) + prompt_tb->w,
2014-03-22 16:04:19 -04:00
( config.padding ),
( ( config.hmode == TRUE ) ?
element_width : ( w - ( 2 * ( config.padding ) ) ) ) - prompt_tb->w, 1,
NORMAL,
( input != NULL ) ? *input : "" );
2012-08-23 21:28:39 -04:00
2014-01-10 04:35:38 -05:00
int line_height = text->font->ascent + text->font->descent;
2012-08-23 21:28:39 -04:00
2014-05-14 14:56:38 -04:00
textbox_show ( text );
textbox_show ( prompt_tb );
2014-05-14 14:56:38 -04:00
2014-01-10 04:35:38 -05:00
// filtered list display
textbox **boxes = calloc ( 1, sizeof ( textbox* ) * max_elements );
2014-01-26 06:59:10 -05:00
2014-06-04 15:29:23 -04:00
for ( i = 0; i < max_elements; i++ ) {
int line = ( i ) % max_rows + ( ( config.hmode == FALSE ) ? 1 : 0 );
int col = ( i ) / max_rows + ( ( config.hmode == FALSE ) ? 0 : 1 );
2014-03-22 16:04:19 -04:00
boxes[i] = textbox_create ( box,
0,
( config.padding ) + col * ( element_width + LINE_MARGIN ), // X
line * line_height + config.padding + ( ( config.hmode == TRUE ) ? 0 : LINE_MARGIN ), // y
element_width, // w
line_height, // h
NORMAL, "" );
2014-03-22 16:04:19 -04:00
textbox_show ( boxes[i] );
2014-01-10 04:35:38 -05:00
}
2014-05-14 14:56:38 -04:00
// Arrows
2014-05-14 16:21:16 -04:00
textbox *arrowbox_top = NULL, *arrowbox_bottom = NULL;
2014-06-04 15:29:23 -04:00
if ( config.hmode == FALSE ) {
2014-05-14 16:21:16 -04:00
arrowbox_top = textbox_create ( box, TB_AUTOHEIGHT | TB_AUTOWIDTH,
2014-05-19 15:58:13 -04:00
( config.padding ),
( config.padding ),
0, 0,
NORMAL,
"" );
2014-05-14 16:21:16 -04:00
arrowbox_bottom = textbox_create ( box, TB_AUTOHEIGHT | TB_AUTOWIDTH,
2014-05-19 15:58:13 -04:00
( config.padding ),
( config.padding ),
0, 0,
NORMAL,
"" );
2014-05-14 16:21:16 -04:00
textbox_move ( arrowbox_top,
2014-05-19 15:58:13 -04:00
w - config.padding - arrowbox_top->w,
config.padding + line_height + LINE_MARGIN );
2014-05-14 16:21:16 -04:00
textbox_move ( arrowbox_bottom,
2014-05-19 15:58:13 -04:00
w - config.padding - arrowbox_bottom->w,
config.padding + max_rows * line_height + LINE_MARGIN );
2014-05-14 16:21:16 -04:00
}
2012-08-23 21:28:39 -04:00
2014-01-10 04:35:38 -05:00
// filtered list
2014-05-19 15:58:13 -04:00
char **filtered = calloc ( num_lines, sizeof ( char* ) );
int *line_map = calloc ( num_lines, sizeof ( int ) );
2014-01-23 05:39:12 -05:00
unsigned int filtered_lines = 0;
2012-08-23 21:28:39 -04:00
2014-06-04 15:29:23 -04:00
if ( input && *input ) {
2014-03-22 16:04:19 -04:00
char **tokens = tokenize ( *input );
// input changed
2014-06-04 15:29:23 -04:00
for ( i = 0, j = 0; i < num_lines; i++ ) {
2014-03-22 16:04:19 -04:00
int match = mmc ( tokens, lines[i], i, mmc_data );
// If each token was matched, add it to list.
2014-06-04 15:29:23 -04:00
if ( match ) {
2014-03-22 16:04:19 -04:00
line_map[j] = i;
filtered[j++] = lines[i];
filtered_lines++;
}
}
2014-01-16 12:47:19 -05:00
2014-03-22 16:04:19 -04:00
tokenize_free ( tokens );
}
2014-06-04 15:29:23 -04:00
else{
2014-02-27 14:26:35 -05:00
int jin = 0;
2014-06-04 15:29:23 -04:00
for ( i = 0; i < num_lines; i++ ) {
filtered[jin] = lines[i];
line_map[jin] = i;
jin++;
filtered_lines++;
}
2014-01-10 04:35:38 -05:00
}
// resize window vertically to suit
// Subtract the margin of the last row.
2014-05-19 03:50:09 -04:00
int h = line_height * ( max_rows + 1 ) + ( config.padding ) * 2 + LINE_MARGIN;
2014-01-26 06:59:10 -05:00
2014-05-17 16:17:07 -04:00
2014-06-04 15:29:23 -04:00
if ( config.hmode == TRUE ) {
2014-03-22 16:04:19 -04:00
h = line_height + ( config.padding ) * 2;
2014-01-26 06:59:10 -05:00
}
// Default location is center.
2014-03-22 16:04:19 -04:00
int y = mon.y + ( mon.h - h ) / 2;
2014-01-26 06:59:10 -05:00
// Determine window location
2014-03-22 16:04:19 -04:00
switch ( config.location )
{
2014-05-19 15:58:13 -04:00
case WL_NORTH_WEST:
x = mon.x;
2014-01-26 06:59:10 -05:00
2014-05-19 15:58:13 -04:00
case WL_NORTH:
y = mon.y;
break;
2014-01-26 06:59:10 -05:00
2014-05-19 15:58:13 -04:00
case WL_NORTH_EAST:
y = mon.y;
2014-01-26 06:59:10 -05:00
2014-05-19 15:58:13 -04:00
case WL_EAST:
x = mon.x + mon.w - w - config.menu_bw * 2;
break;
2014-01-26 06:59:10 -05:00
2014-05-19 15:58:13 -04:00
case WL_EAST_SOUTH:
x = mon.x + mon.w - w - config.menu_bw * 2;
2014-01-26 06:59:10 -05:00
2014-05-19 15:58:13 -04:00
case WL_SOUTH:
y = mon.y + mon.h - h - config.menu_bw * 2;
break;
2014-01-26 06:59:10 -05:00
2014-05-19 15:58:13 -04:00
case WL_SOUTH_WEST:
y = mon.y + mon.h - h - config.menu_bw * 2;
2014-01-26 06:59:10 -05:00
2014-05-19 15:58:13 -04:00
case WL_WEST:
x = mon.x;
break;
2014-01-26 06:59:10 -05:00
2014-05-19 15:58:13 -04:00
case WL_CENTER:
default:
break;
}
2014-05-15 10:54:35 -04:00
// Apply offset.
2014-05-19 15:58:13 -04:00
y += config.y_offset;
x += config.x_offset;
2014-03-22 16:04:19 -04:00
XMoveResizeWindow ( display, box, x, y, w, h );
XMapRaised ( display, box );
2014-01-10 04:35:38 -05:00
// if grabbing keyboard failed, fall through
2014-06-04 15:29:23 -04:00
if ( take_keyboard ( box ) ) {
2014-05-19 15:58:13 -04:00
KeySym prev_key = 0;
2014-05-14 14:56:38 -04:00
unsigned int selected = 0;
2014-06-04 15:29:23 -04:00
for (;; ) {
2014-05-27 12:55:48 -04:00
int refilter = FALSE;
int update = FALSE;
XEvent ev;
XNextEvent ( display, &ev );
2014-06-04 15:29:23 -04:00
if ( ev.type == Expose ) {
while ( XCheckTypedEvent ( display, Expose, &ev ) ) {
;
}
2014-01-10 04:35:38 -05:00
textbox_draw ( text );
textbox_draw ( prompt_tb );
menu_draw ( boxes, max_elements, num_lines, &last_offset, selected, filtered );
2014-05-19 15:58:13 -04:00
menu_set_arrow_text ( filtered_lines, selected,
max_elements, arrowbox_top,
arrowbox_bottom );
// Why do we need the specian -1?
2014-06-04 15:29:23 -04:00
if ( config.hmode == FALSE && max_elements > 0 ) {
XDrawLine ( display, main_window, gc, ( config.padding ),
2014-05-19 15:58:13 -04:00
line_height + ( config.padding ) + ( LINE_MARGIN - 2 ) / 2,
w - ( ( config.padding ) ) - 1,
line_height + ( config.padding ) + ( LINE_MARGIN - 2 ) / 2 );
}
}
2014-06-04 15:29:23 -04:00
else if ( ev.type == SelectionNotify ) {
2014-05-27 06:55:47 -04:00
// TODO move this.
Atom utf8 = XInternAtom ( display, "UTF8_STRING", False );
2014-06-04 15:29:23 -04:00
if ( ev.xselection.property == utf8 ) {
2014-05-27 06:55:47 -04:00
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,
utf8,
0,
256 / 4, // max length in words.
False, // Do not delete clipboard.
utf8, &da, &di, &dl, &rm, (unsigned char * *) &pbuf );
// If There was remaining data left.. lets ignore this.
// Only accept it when we get bytes!
2014-06-04 15:29:23 -04:00
if ( di == 8 ) {
2014-05-27 06:55:47 -04:00
char *index;
2014-06-04 15:29:23 -04:00
if ( ( index = strchr ( pbuf, '\n' ) ) != NULL ) {
2014-05-27 06:55:47 -04:00
// 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 ( text, text->cursor, str );
textbox_cursor ( text, text->cursor + dl );
2014-05-27 12:55:48 -04:00
// Force a redraw and refiltering of the text.
update = TRUE;
refilter = TRUE;
2014-05-27 06:55:47 -04:00
}
XFree ( pbuf );
}
}
2014-06-04 15:29:23 -04:00
else if ( ev.type == KeyPress ) {
while ( XCheckTypedEvent ( display, KeyPress, &ev ) ) {
;
2014-03-22 16:04:19 -04:00
}
2014-01-15 03:27:03 -05:00
2014-06-04 15:29:23 -04:00
if ( time ) {
*time = ev.xkey.time;
}
KeySym key = XkbKeycodeToKeysym ( display, ev.xkey.keycode, 0, 0 );
2014-01-10 04:35:38 -05:00
if ( ( ( ( ev.xkey.state & ControlMask ) == ControlMask ) && key == XK_v ) ||
2014-06-04 15:29:23 -04:00
key == XK_Insert ) {
2014-05-27 06:55:47 -04:00
// TODO move these.
Atom clip = XInternAtom ( display, "CLIPBOARD", False );
Atom utf8 = XInternAtom ( display, "UTF8_STRING", False );
XConvertSelection ( display, ( ev.xkey.state & ShiftMask ) ? clip : XA_PRIMARY,
2014-05-27 12:41:32 -04:00
utf8, utf8, main_window, CurrentTime );
2014-05-27 06:55:47 -04:00
}
else if ( ( ( ev.xkey.state & ShiftMask ) == ShiftMask ) &&
2014-06-04 15:29:23 -04:00
key == XK_slash ) {
retv = MENU_NEXT;
*selected_line = 0;
break;
}
else if ( ( ( ev.xkey.state & ShiftMask ) == ShiftMask ) &&
2014-06-04 15:29:23 -04:00
key == XK_Delete ) {
if ( filtered[selected] != NULL ) {
*selected_line = line_map[selected];
retv = MENU_ENTRY_DELETE;
break;
}
}
2014-01-10 04:35:38 -05:00
int rc = textbox_keypress ( text, &ev );
2014-01-10 04:35:38 -05:00
2014-06-04 15:29:23 -04:00
if ( rc < 0 ) {
if ( shift != NULL ) {
( *shift ) = ( ( ev.xkey.state & ShiftMask ) == ShiftMask );
}
2014-01-10 04:35:38 -05:00
2014-06-04 15:29:23 -04:00
if ( filtered && filtered[selected] ) {
2014-03-22 16:04:19 -04:00
retv = MENU_OK;
*selected_line = line_map[selected];
2014-03-22 16:04:19 -04:00
}
2014-06-04 15:29:23 -04:00
else{
retv = MENU_CUSTOM_INPUT;
}
break;
}
2014-06-04 15:29:23 -04:00
else if ( rc ) {
2014-05-27 12:55:48 -04:00
refilter = TRUE;
update = TRUE;
}
2014-06-04 15:29:23 -04:00
else{
// unhandled key
KeySym key = XkbKeycodeToKeysym ( display, ev.xkey.keycode, 0, 0 );
if ( key == XK_Escape
2014-05-19 15:58:13 -04:00
// 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 || ev.xkey.state & windows_modmask ) && key == windows_keysym )
|| ( ( rundialog_modmask == AnyModifier || ev.xkey.state & rundialog_modmask ) && key == rundialog_keysym )
|| ( ( sshdialog_modmask == AnyModifier || ev.xkey.state & sshdialog_modmask ) && key == sshdialog_keysym )
2014-06-04 15:29:23 -04:00
) {
retv = MENU_CANCEL;
break;
2014-03-22 16:04:19 -04:00
}
2014-06-04 15:29:23 -04:00
else{
// Up or Shift-Tab
if ( key == XK_Up || ( key == XK_Tab && ev.xkey.state & ShiftMask ) ||
2014-06-04 15:29:23 -04:00
( key == XK_j && ev.xkey.state & ControlMask ) ) {
if ( selected == 0 ) {
selected = filtered_lines;
}
2014-06-04 15:29:23 -04:00
if ( selected > 0 ) {
selected--;
}
2014-05-27 12:55:48 -04:00
update = TRUE;
}
else if ( key == XK_Down ||
2014-06-04 15:29:23 -04:00
( key == XK_k && ev.xkey.state & ControlMask ) ) {
selected = selected < filtered_lines - 1 ? MIN ( filtered_lines - 1, selected + 1 ) : 0;
2014-05-27 12:55:48 -04:00
update = TRUE;
}
2014-06-04 15:29:23 -04:00
else if ( key == XK_Page_Up ) {
if ( selected < max_elements ) {
selected = 0;
2014-03-22 16:04:19 -04:00
}
2014-06-04 15:29:23 -04:00
else{
2014-05-19 03:50:09 -04:00
selected -= ( max_elements - 1 );
}
2014-05-27 12:55:48 -04:00
update = TRUE;
2014-01-10 04:35:38 -05:00
}
2014-06-04 15:29:23 -04:00
else if ( key == XK_Page_Down ) {
2014-05-19 03:50:09 -04:00
selected += ( max_elements - 1 );
2014-01-23 05:39:12 -05:00
2014-06-04 15:29:23 -04:00
if ( selected >= num_lines ) {
selected = num_lines - 1;
}
2014-05-27 12:55:48 -04:00
update = TRUE;
}
2014-06-04 15:29:23 -04:00
else if ( key == XK_h && ev.xkey.state & ControlMask ) {
if ( selected < max_rows ) {
2014-05-19 03:50:09 -04:00
selected = 0;
}
2014-06-04 15:29:23 -04:00
else{
2014-05-19 15:58:13 -04:00
selected -= max_rows;
2014-05-19 03:50:09 -04:00
}
2014-05-27 12:55:48 -04:00
update = TRUE;
2014-05-19 03:50:09 -04:00
}
2014-06-04 15:29:23 -04:00
else if ( key == XK_l && ev.xkey.state & ControlMask ) {
2014-05-19 03:50:09 -04:00
selected += max_rows;
2014-06-04 15:29:23 -04:00
if ( selected >= num_lines ) {
2014-05-19 15:58:13 -04:00
selected = num_lines - 1;
2014-05-19 03:50:09 -04:00
}
2014-05-27 12:55:48 -04:00
update = TRUE;
2014-05-19 03:50:09 -04:00
}
2014-06-04 15:29:23 -04:00
else if ( key == XK_Home || key == XK_KP_Home ) {
selected = 0;
2014-05-27 12:55:48 -04:00
update = TRUE;
2014-03-22 16:04:19 -04:00
}
2014-06-04 15:29:23 -04:00
else if ( key == XK_End || key == XK_KP_End ) {
selected = num_lines - 1;
2014-05-27 12:55:48 -04:00
update = TRUE;
}
2014-06-04 15:29:23 -04:00
else if ( key == XK_Tab ) {
if ( filtered_lines == 1 ) {
if ( filtered[selected] ) {
retv = MENU_OK;
*selected_line = line_map[selected];
}
2014-06-04 15:29:23 -04:00
else{
fprintf ( stderr, "We should never hit this." );
abort ();
}
break;
}
int length_prefix = calculate_common_prefix ( filtered, num_lines );
// TODO: memcmp to utf8 aware cmp.
2014-06-04 15:29:23 -04:00
if ( length_prefix && memcmp ( filtered[0], text->text, length_prefix ) ) {
// Do not want to modify original string, so make copy.
// not eff..
char * str = malloc ( sizeof ( char ) * ( length_prefix + 1 ) );
memcpy ( str, filtered[0], length_prefix );
str[length_prefix] = '\0';
textbox_text ( text, str );
textbox_cursor_end ( text );
free ( str );
2014-05-27 12:55:48 -04:00
update = TRUE;
}
// Double tab!
2014-06-04 15:29:23 -04:00
else if ( filtered_lines == 0 && key == prev_key ) {
retv = MENU_NEXT;
*selected_line = 0;
break;
}
2014-06-04 15:29:23 -04:00
else{
selected = selected < filtered_lines - 1 ? MIN ( filtered_lines - 1, selected + 1 ) : 0;
2014-05-27 12:55:48 -04:00
update = TRUE;
}
}
}
}
2014-05-27 12:55:48 -04:00
prev_key = key;
}
// If something changed, refilter the list. (paste or text entered)
2014-06-04 15:29:23 -04:00
if ( refilter ) {
2014-05-27 12:55:48 -04:00
char **tokens = tokenize ( text->text );
// input changed
2014-06-04 15:29:23 -04:00
for ( i = 0, j = 0; i < num_lines; i++ ) {
2014-05-27 12:55:48 -04:00
int match = mmc ( tokens, lines[i], i, mmc_data );
// If each token was matched, add it to list.
2014-06-04 15:29:23 -04:00
if ( match ) {
2014-05-27 12:55:48 -04:00
line_map[j] = i;
filtered[j++] = lines[i];
}
}
// Cleanup + bookkeeping.
filtered_lines = j;
selected = MIN ( selected, j - 1 );
2014-06-04 15:29:23 -04:00
for (; j < num_lines; j++ ) {
2014-05-27 12:55:48 -04:00
filtered[j] = NULL;
}
2014-06-04 15:29:23 -04:00
if ( config.zeltak_mode && filtered_lines == 1 ) {
if ( filtered[selected] ) {
2014-05-27 12:55:48 -04:00
retv = MENU_OK;
*selected_line = line_map[selected];
}
2014-06-04 15:29:23 -04:00
else{
2014-05-27 12:55:48 -04:00
fprintf ( stderr, "We should never hit this." );
abort ();
}
break;
}
tokenize_free ( tokens );
}
// Update if requested.
2014-06-04 15:29:23 -04:00
if ( update ) {
2014-05-19 15:58:13 -04:00
menu_hide_arrow_text ( filtered_lines, selected,
max_elements, arrowbox_top,
arrowbox_bottom );
textbox_draw ( text );
textbox_draw ( prompt_tb );
menu_draw ( boxes, max_elements, num_lines, &last_offset, selected, filtered );
2014-05-19 15:58:13 -04:00
menu_set_arrow_text ( filtered_lines, selected,
max_elements, arrowbox_top,
arrowbox_bottom );
}
2014-01-10 04:35:38 -05:00
}
release_keyboard ();
}
2012-08-23 21:28:39 -04:00
2014-05-22 03:33:32 -04:00
free ( *input );
2014-03-22 16:04:19 -04:00
*input = strdup ( text->text );
2014-01-16 12:47:19 -05:00
2014-03-22 16:04:19 -04:00
textbox_free ( text );
textbox_free ( prompt_tb );
textbox_free ( arrowbox_bottom );
textbox_free ( arrowbox_top );
2014-06-04 15:29:23 -04:00
for ( i = 0; i < max_elements; i++ ) {
2014-03-22 16:04:19 -04:00
textbox_free ( boxes[i] );
}
2012-08-23 21:28:39 -04:00
2014-03-22 16:04:19 -04:00
free ( boxes );
2014-01-10 04:35:38 -05:00
2014-03-22 16:04:19 -04:00
free ( filtered );
free ( line_map );
2014-01-10 04:35:38 -05:00
return retv;
2012-06-29 13:10:13 -04:00
}
2014-01-15 03:27:03 -05:00
2014-01-20 16:58:10 -05:00
SwitcherMode run_switcher_window ( char **input )
{
2014-05-26 04:32:26 -04:00
Screen *screen = DefaultScreenOfDisplay ( display );
Window root = RootWindow ( display, XScreenNumberOfScreen ( screen ) );
SwitcherMode retv = MODE_EXIT;
2014-01-20 16:58:10 -05:00
// 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 ) )
2014-06-04 15:29:23 -04:00
&& type == XA_WINDOW && count > 0 ) ) {
curr_win_id = 0;
}
2014-01-20 16:58:10 -05:00
2014-03-22 16:04:19 -04:00
if ( window_get_prop ( root, netatoms[_NET_CLIENT_LIST_STACKING],
&type, &nwins, wins, 100 * sizeof ( Window ) )
2014-06-04 15:29:23 -04:00
&& type == XA_WINDOW ) {
2014-03-22 16:04:19 -04:00
char pattern[50];
int i;
unsigned int classfield = 0;
unsigned long desktops = 0;
2014-01-20 16:58:10 -05:00
// 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...
2014-03-22 16:04:19 -04:00
winlist *ids = winlist_new ();
2014-01-20 16:58:10 -05:00
2014-01-20 16:58:10 -05:00
// calc widths of fields
2014-06-04 15:29:23 -04:00
for ( i = nwins - 1; i > -1; i-- ) {
2014-01-14 00:36:12 -05:00
client *c;
2014-03-22 16:04:19 -04:00
if ( ( c = window_client ( wins[i] ) )
2014-01-20 16:58:10 -05:00
&& !c->xattr.override_redirect
2014-03-22 16:04:19 -04:00
&& !client_has_state ( c, netatoms[_NET_WM_STATE_SKIP_PAGER] )
2014-06-04 15:29:23 -04:00
&& !client_has_state ( c, netatoms[_NET_WM_STATE_SKIP_TASKBAR] ) ) {
2014-03-22 16:04:19 -04:00
classfield = MAX ( classfield, strlen ( c->class ) );
2014-01-20 16:58:10 -05:00
2014-03-18 05:38:30 -04:00
#ifdef HAVE_I3_IPC_H
2014-01-10 04:35:38 -05:00
2014-01-20 16:58:10 -05:00
// In i3 mode, skip the i3bar completely.
2014-06-04 15:29:23 -04:00
if ( config_i3_mode && strstr ( c->class, "i3bar" ) != NULL ) {
2014-03-22 16:04:19 -04:00
continue;
}
2012-06-29 13:10:13 -04:00
#endif
2014-06-04 15:29:23 -04:00
if ( c->window == curr_win_id ) {
c->active = TRUE;
}
2014-03-22 16:04:19 -04:00
winlist_append ( ids, c->window, NULL );
2014-01-20 16:58:10 -05:00
}
}
2014-01-20 16:58:10 -05:00
// Create pattern for printing the line.
2014-06-04 15:29:23 -04:00
if ( !window_get_cardinal_prop ( root, netatoms[_NET_NUMBER_OF_DESKTOPS], &desktops, 1 ) ) {
desktops = 1;
2014-03-22 16:04:19 -04:00
}
2014-03-18 05:38:30 -04:00
#ifdef HAVE_I3_IPC_H
2014-06-04 15:29:23 -04:00
if ( config_i3_mode ) {
sprintf ( pattern, "%%s%%-%ds %%s", MAX ( 5, classfield ) );
2014-03-22 16:04:19 -04:00
}
2014-06-04 15:29:23 -04:00
else{
2014-03-02 06:53:06 -05:00
#endif
sprintf ( pattern, "%%s%%-%ds %%-%ds %%s", desktops < 10 ? 1 : 2, MAX ( 5, classfield ) );
2014-03-18 05:38:30 -04:00
#ifdef HAVE_I3_IPC_H
2014-03-22 16:04:19 -04:00
}
2014-03-02 06:53:06 -05:00
#endif
char **list = calloc ( ( ids->len + 1 ), sizeof ( char* ) );
2014-03-22 16:04:19 -04:00
int lines = 0;
2013-02-04 03:32:10 -05:00
2014-01-20 16:58:10 -05:00
// build the actual list
2014-02-26 16:48:49 -05:00
Window w = 0;
2014-03-22 16:04:19 -04:00
winlist_ascend ( ids, i, w )
{
2014-01-20 16:58:10 -05:00
client *c;
2012-06-29 13:10:13 -04:00
2014-06-04 15:29:23 -04:00
if ( ( c = window_client ( w ) ) ) {
2014-01-20 16:58:10 -05:00
// final line format
unsigned long wmdesktop;
2014-03-22 16:04:19 -04:00
char desktop[5];
desktop[0] = 0;
char *line = malloc ( strlen ( c->title ) + strlen ( c->class ) + classfield + 50 );
2014-03-18 05:38:30 -04:00
#ifdef HAVE_I3_IPC_H
2014-06-04 15:29:23 -04:00
if ( !config_i3_mode ) {
2014-03-02 06:53:06 -05:00
#endif
2014-03-22 16:04:19 -04:00
// find client's desktop. this is zero-based, so we adjust by since most
// normal people don't think like this :-)
2014-06-04 15:29:23 -04:00
if ( !window_get_cardinal_prop ( c->window, netatoms[_NET_WM_DESKTOP], &wmdesktop, 1 ) ) {
2014-03-22 16:04:19 -04:00
wmdesktop = 0xFFFFFFFF;
}
2014-01-10 04:35:38 -05:00
2014-06-04 15:29:23 -04:00
if ( wmdesktop < 0xFFFFFFFF ) {
2014-03-22 16:04:19 -04:00
sprintf ( desktop, "%d", (int) wmdesktop + 1 );
}
sprintf ( line, pattern, ( c->active ) ? "*" : "", desktop, c->class, c->title );
2014-03-18 05:38:30 -04:00
#ifdef HAVE_I3_IPC_H
2014-03-22 16:04:19 -04:00
}
2014-06-04 15:29:23 -04:00
else{
sprintf ( line, pattern, ( c->active ) ? "*" : "", c->class, c->title );
2014-03-22 16:04:19 -04:00
}
2014-03-02 06:53:06 -05:00
#endif
2014-01-20 16:58:10 -05:00
list[lines++] = line;
}
}
Time time;
int selected_line = 0;
MenuReturn mretv = menu ( list, input, "window:", &time, NULL, window_match, ids, &selected_line );
2014-01-14 00:36:12 -05:00
2014-06-04 15:29:23 -04:00
if ( mretv == MENU_NEXT ) {
2014-01-22 03:24:31 -05:00
retv = NEXT_DIALOG;
2014-03-22 16:04:19 -04:00
}
2014-06-04 15:29:23 -04:00
else if ( mretv == MENU_OK && list[selected_line] ) {
2014-03-18 05:38:30 -04:00
#ifdef HAVE_I3_IPC_H
2014-06-04 15:29:23 -04:00
if ( config_i3_mode ) {
2014-01-20 16:58:10 -05:00
// Hack for i3.
2014-03-22 16:04:19 -04:00
focus_window_i3 ( i3_socket_path, ids->array[selected_line] );
}
else
#endif
2014-01-20 16:58:10 -05:00
{
// Change to the desktop of the selected window/client.
// TODO: get rid of strtol
2014-03-22 16:04:19 -04:00
window_send_message ( root, root, netatoms[_NET_CURRENT_DESKTOP], strtol ( list[selected_line], NULL, 10 ) - 1,
SubstructureNotifyMask | SubstructureRedirectMask, time );
XSync ( display, False );
2014-03-22 16:04:19 -04:00
window_send_message ( root, ids->array[selected_line], netatoms[_NET_ACTIVE_WINDOW], 2, // 2 = pager
SubstructureNotifyMask | SubstructureRedirectMask, time );
2014-01-14 00:36:12 -05:00
}
2014-01-20 16:58:10 -05:00
}
2014-01-10 04:35:38 -05:00
2014-06-04 15:29:23 -04:00
for ( i = 0; i < lines; i++ ) {
2014-03-22 16:04:19 -04:00
free ( list[i] );
}
2014-01-10 04:35:38 -05:00
2014-03-22 16:04:19 -04:00
free ( list );
winlist_free ( ids );
2014-01-20 16:58:10 -05:00
}
2014-01-14 00:36:12 -05:00
2014-01-20 16:58:10 -05:00
return retv;
}
2014-01-14 00:36:12 -05:00
2014-05-24 05:02:13 -04:00
static void run_switcher ( int do_fork, SwitcherMode mode )
2014-01-20 16:58:10 -05:00
{
// 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...
2014-06-04 15:29:23 -04:00
if ( do_fork == TRUE ) {
if ( fork () ) {
2014-03-22 16:04:19 -04:00
return;
}
2014-01-14 00:36:12 -05:00
2014-03-22 16:04:19 -04:00
display = XOpenDisplay ( 0 );
XSync ( display, True );
2014-01-20 16:58:10 -05:00
}
2014-05-26 03:00:14 -04:00
// Because of the above fork, we want to do this here.
// Make sure this is isolated to its own thread.
textbox_setup ( config.menu_font, active_font,
config.menu_bg, config.menu_fg,
config.menu_hlbg,
config.menu_hlfg );
2014-01-20 16:58:10 -05:00
char *input = NULL;
2014-01-14 00:36:12 -05:00
2014-06-04 15:29:23 -04:00
do {
2014-01-22 03:24:31 -05:00
SwitcherMode retv = MODE_EXIT;
2014-01-22 07:07:52 -05:00
2014-06-04 15:29:23 -04:00
if ( mode == WINDOW_SWITCHER ) {
2014-03-22 16:04:19 -04:00
retv = run_switcher_window ( &input );
}
2014-06-04 15:29:23 -04:00
else if ( mode == RUN_DIALOG ) {
2014-03-22 16:04:19 -04:00
retv = run_switcher_dialog ( &input );
}
2014-06-04 15:29:23 -04:00
else if ( mode == SSH_DIALOG ) {
2014-03-22 16:04:19 -04:00
retv = ssh_switcher_dialog ( &input );
2014-01-22 03:24:31 -05:00
}
2014-06-04 15:29:23 -04:00
else if ( mode == DMENU_DIALOG ) {
2014-01-29 18:47:23 -05:00
retv = dmenu_switcher_dialog ( &input );
}
2014-01-26 10:47:26 -05:00
2014-06-04 15:29:23 -04:00
if ( retv == NEXT_DIALOG ) {
2014-03-22 16:04:19 -04:00
mode = ( mode + 1 ) % NUM_DIALOGS;
}
2014-06-04 15:29:23 -04:00
else{
2014-01-22 03:24:31 -05:00
mode = retv;
2014-01-14 00:36:12 -05:00
}
} while ( mode != MODE_EXIT );
2014-01-16 12:47:19 -05:00
2014-05-22 03:33:32 -04:00
free ( input );
2014-01-16 12:47:19 -05:00
2014-05-26 03:00:14 -04:00
// Cleanup font setup.
textbox_cleanup ();
2014-06-04 15:29:23 -04:00
if ( do_fork == TRUE ) {
2014-03-22 16:04:19 -04:00
exit ( EXIT_SUCCESS );
2014-01-20 16:58:10 -05:00
}
2012-06-29 13:10:13 -04:00
}
// KeyPress event
static void handle_keypress ( XEvent *ev )
2012-06-29 13:10:13 -04:00
{
2014-03-22 16:04:19 -04:00
KeySym key = XkbKeycodeToKeysym ( display, ev->xkey.keycode, 0, 0 );
2012-06-29 13:10:13 -04:00
2014-01-14 00:36:12 -05:00
if ( ( windows_modmask == AnyModifier || ev->xkey.state & windows_modmask ) &&
2014-06-04 15:29:23 -04:00
key == windows_keysym ) {
2014-05-24 05:02:13 -04:00
run_switcher ( TRUE, WINDOW_SWITCHER );
2014-01-14 00:36:12 -05:00
}
if ( ( rundialog_modmask == AnyModifier || ev->xkey.state & rundialog_modmask ) &&
2014-06-04 15:29:23 -04:00
key == rundialog_keysym ) {
2014-05-24 05:02:13 -04:00
run_switcher ( TRUE, RUN_DIALOG );
2014-01-14 00:36:12 -05:00
}
2014-01-21 04:13:42 -05:00
if ( ( sshdialog_modmask == AnyModifier || ev->xkey.state & sshdialog_modmask ) &&
2014-06-04 15:29:23 -04:00
key == sshdialog_keysym ) {
2014-05-24 05:02:13 -04:00
run_switcher ( TRUE, SSH_DIALOG );
2014-01-21 04:13:42 -05:00
}
2012-06-29 13:10:13 -04:00
}
// convert a Mod+key arg to mod mask and keysym
2014-05-27 02:31:59 -04:00
static void parse_key ( char *combo, unsigned int *mod, KeySym *key )
2012-06-29 13:10:13 -04:00
{
2014-01-10 04:35:38 -05:00
unsigned int modmask = 0;
2014-06-04 15:29:23 -04:00
if ( strcasestr ( combo, "shift" ) ) {
2014-03-22 16:04:19 -04:00
modmask |= ShiftMask;
}
2014-01-10 04:35:38 -05:00
2014-06-04 15:29:23 -04:00
if ( strcasestr ( combo, "control" ) ) {
2014-03-22 16:04:19 -04:00
modmask |= ControlMask;
}
2014-01-10 04:35:38 -05:00
2014-06-04 15:29:23 -04:00
if ( strcasestr ( combo, "mod1" ) ) {
2014-03-22 16:04:19 -04:00
modmask |= Mod1Mask;
}
2014-01-10 04:35:38 -05:00
2014-06-04 15:29:23 -04:00
if ( strcasestr ( combo, "alt" ) ) {
2014-03-22 16:04:19 -04:00
modmask |= Mod1Mask;
}
2014-01-14 00:36:12 -05:00
2014-06-04 15:29:23 -04:00
if ( strcasestr ( combo, "mod2" ) ) {
2014-03-22 16:04:19 -04:00
modmask |= Mod2Mask;
}
2014-01-10 04:35:38 -05:00
2014-06-04 15:29:23 -04:00
if ( strcasestr ( combo, "mod3" ) ) {
2014-03-22 16:04:19 -04:00
modmask |= Mod3Mask;
}
2014-01-10 04:35:38 -05:00
2014-06-04 15:29:23 -04:00
if ( strcasestr ( combo, "mod4" ) ) {
2014-03-22 16:04:19 -04:00
modmask |= Mod4Mask;
}
2014-01-10 04:35:38 -05:00
2014-06-04 15:29:23 -04:00
if ( strcasestr ( combo, "mod5" ) ) {
2014-03-22 16:04:19 -04:00
modmask |= Mod5Mask;
}
2014-01-10 04:35:38 -05:00
2014-03-22 16:04:19 -04:00
*mod = modmask ? modmask : AnyModifier;
2014-01-10 04:35:38 -05:00
2014-03-22 16:04:19 -04:00
char i = strlen ( combo );
2014-01-10 04:35:38 -05:00
2014-06-04 15:29:23 -04:00
while ( i > 0 && !strchr ( "-+", combo[i - 1] ) ) {
2014-03-22 16:04:19 -04:00
i--;
}
2014-01-10 04:35:38 -05:00
2014-03-22 16:04:19 -04:00
KeySym sym = XStringToKeysym ( combo + i );
2014-01-10 04:35:38 -05:00
2014-06-04 15:29:23 -04:00
if ( sym == NoSymbol || ( !modmask && ( strchr ( combo, '-' ) || strchr ( combo, '+' ) ) ) ) {
2014-03-22 16:04:19 -04:00
fprintf ( stderr, "sorry, cannot understand key combination: %s\n", combo );
exit ( EXIT_FAILURE );
2014-01-10 04:35:38 -05:00
}
*key = sym;
2012-06-29 13:10:13 -04:00
}
// bind a key combination on a root window, compensating for Lock* states
2014-05-27 02:31:59 -04:00
static void grab_key ( unsigned int modmask, KeySym key )
2012-06-29 13:10:13 -04:00
{
2014-05-26 04:32:26 -04:00
Screen *screen = DefaultScreenOfDisplay ( display );
Window root = RootWindow ( display, XScreenNumberOfScreen ( screen ) );
2014-03-22 16:04:19 -04:00
KeyCode keycode = XKeysymToKeycode ( display, key );
XUngrabKey ( display, keycode, AnyModifier, root );
2014-01-10 04:35:38 -05:00
2014-06-04 15:29:23 -04:00
if ( modmask != AnyModifier ) {
2014-01-10 04:35:38 -05:00
// bind to combinations of mod and lock masks, so caps and numlock don't confuse people
2014-03-22 16:04:19 -04:00
XGrabKey ( display, keycode, modmask, root, True, GrabModeAsync, GrabModeAsync );
XGrabKey ( display, keycode, modmask | LockMask, root, True, GrabModeAsync, GrabModeAsync );
2014-01-10 04:35:38 -05:00
2014-06-04 15:29:23 -04:00
if ( NumlockMask ) {
2014-03-22 16:04:19 -04:00
XGrabKey ( display, keycode, modmask | NumlockMask, root, True, GrabModeAsync, GrabModeAsync );
XGrabKey ( display, keycode, modmask | NumlockMask | LockMask, root, True, GrabModeAsync, GrabModeAsync );
2014-01-10 04:35:38 -05:00
}
2014-03-22 16:04:19 -04:00
}
2014-06-04 15:29:23 -04:00
else{
2014-01-10 04:35:38 -05:00
// nice simple single key bind
2014-03-22 16:04:19 -04:00
XGrabKey ( display, keycode, AnyModifier, root, True, GrabModeAsync, GrabModeAsync );
2014-01-10 04:35:38 -05:00
}
2012-06-29 13:10:13 -04:00
}
2014-03-18 05:38:30 -04:00
#ifdef HAVE_I3_IPC_H
2014-03-22 16:04:19 -04:00
static inline void display_get_i3_path ( Display *display )
{
2014-05-26 04:32:26 -04:00
Screen *screen = DefaultScreenOfDisplay ( display );
Window root = RootWindow ( display, XScreenNumberOfScreen ( screen ) );
Atom atom = XInternAtom ( display, I3_SOCKET_PATH_PROP, True );
2014-05-20 05:22:03 -04:00
config_i3_mode = 0;
2014-06-04 15:29:23 -04:00
if ( atom != None ) {
2014-03-22 16:04:19 -04:00
i3_socket_path = window_get_text_prop ( root, atom );
2014-06-04 15:29:23 -04:00
if ( i3_socket_path != NULL ) {
2014-05-20 05:22:03 -04:00
config_i3_mode = 1;
}
}
}
2014-03-18 05:38:30 -04:00
#endif //HAVE_I3_IPC_H
/**
* Help function. This calls man.
*/
2014-05-27 02:31:59 -04:00
static void help ()
{
2014-03-22 16:04:19 -04:00
int code = execlp ( "man", "man", MANPAGE_PATH, NULL );
2014-06-04 15:29:23 -04:00
if ( code == -1 ) {
2014-03-22 16:04:19 -04:00
fprintf ( stderr, "Failed to execute man: %s\n", strerror ( errno ) );
}
}
2014-05-19 15:58:13 -04:00
static void parse_cmd_options ( int argc, char ** argv )
{
2014-05-20 05:41:39 -04:00
// catch help request
2014-05-24 05:02:13 -04:00
if ( find_arg ( argc, argv, "-h" ) >= 0 ||
2014-06-04 15:29:23 -04:00
find_arg ( argc, argv, "-help" ) >= 0 ) {
2014-05-20 05:41:39 -04:00
help ();
exit ( EXIT_SUCCESS );
2014-05-20 05:41:39 -04:00
}
if ( find_arg ( argc, argv, "-v" ) >= 0 ||
2014-06-04 15:29:23 -04:00
find_arg ( argc, argv, "-version" ) >= 0 ) {
2014-05-20 05:41:39 -04:00
fprintf ( stdout, "Version: "VERSION "\n" );
exit ( EXIT_SUCCESS );
2014-05-20 05:41:39 -04:00
}
// Parse commandline arguments about the looks.
find_arg_int ( argc, argv, "-opacity", &( config.window_opacity ) );
find_arg_int ( argc, argv, "-width", &( config.menu_width ) );
find_arg_int ( argc, argv, "-lines", &( config.menu_lines ) );
find_arg_int ( 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, "-hlfg", &( config.menu_hlfg ) );
find_arg_str ( argc, argv, "-hlbg", &( config.menu_hlbg ) );
find_arg_str ( argc, argv, "-bc", &( config.menu_bc ) );
find_arg_int ( argc, argv, "-bw", &( config.menu_bw ) );
// Parse commandline arguments about size and position
find_arg_int ( argc, argv, "-loc", &( config.location ) );
find_arg_int ( argc, argv, "-padding", &( config.padding ) );
find_arg_int ( argc, argv, "-xoffset", &( config.x_offset ) );
find_arg_int ( argc, argv, "-yoffset", &( config.y_offset ) );
2014-06-04 15:29:23 -04:00
if ( find_arg ( argc, argv, "-fixed-num-lines" ) >= 0 ) {
config.fixed_num_lines = 1;
}
// Parse commandline arguments about behavior
find_arg_str ( argc, argv, "-terminal", &( config.terminal_emulator ) );
2014-06-04 15:29:23 -04:00
if ( find_arg ( argc, argv, "-zeltak" ) >= 0 ) {
config.zeltak_mode = 1;
}
2014-06-04 15:29:23 -04:00
if ( find_arg ( argc, argv, "-hmode" ) >= 0 ) {
config.hmode = TRUE;
}
2014-05-20 03:43:45 -04:00
2014-06-04 15:29:23 -04:00
if ( find_arg ( argc, argv, "-ssh-set-title" ) >= 0 ) {
2014-05-22 04:03:36 -04:00
char *value;
find_arg_str ( argc, argv, "-ssh-set-title", &value );
2014-06-04 15:29:23 -04:00
if ( strcasecmp ( value, "true" ) == 0 ) {
2014-05-22 04:03:36 -04:00
config.ssh_set_title = TRUE;
}
2014-06-04 15:29:23 -04:00
else{
2014-05-22 04:03:36 -04:00
config.ssh_set_title = FALSE;
}
}
2014-05-20 03:43:45 -04:00
// 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 ) );
2014-05-20 05:41:39 -04:00
// Dump.
2014-06-04 15:29:23 -04:00
if ( find_arg ( argc, argv, "-dump-xresources" ) >= 0 ) {
2014-05-22 15:56:57 -04:00
xresource_dump ();
exit ( EXIT_SUCCESS );
}
}
2014-05-19 15:58:13 -04:00
static void cleanup ()
2014-05-19 15:02:05 -04:00
{
// Cleanup
2014-06-04 15:29:23 -04:00
if ( display != NULL ) {
if ( main_window != None ) {
2014-05-19 15:02:05 -04:00
XFreeGC ( display, gc );
XDestroyWindow ( display, main_window );
XCloseDisplay ( display );
}
}
2014-06-04 15:29:23 -04:00
if ( cache_xattr != NULL ) {
2014-05-19 15:02:05 -04:00
winlist_free ( cache_xattr );
}
2014-06-04 15:29:23 -04:00
if ( cache_client != NULL ) {
2014-05-19 15:02:05 -04:00
winlist_free ( cache_client );
}
#ifdef HAVE_I3_IPC_H
2014-06-04 15:29:23 -04:00
if ( i3_socket_path != NULL ) {
2014-05-19 15:02:05 -04:00
free ( i3_socket_path );
}
#endif
2014-05-19 15:58:13 -04:00
// Cleaning up memory allocated by the Xresources file.
// TODO, not happy with this.
parse_xresource_free ();
// Whipe the handle.. (not working)
xdgWipeHandle ( &xdg_handle );
free ( active_font );
2014-05-19 15:02:05 -04:00
}
2014-05-20 05:41:39 -04:00
2014-05-22 03:33:32 -04:00
/**
* 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.
*/
2014-06-05 02:11:01 -04:00
static void config_sanity_check ( void )
2012-06-29 13:10:13 -04:00
{
2014-06-04 15:29:23 -04:00
if ( config.menu_lines == 0 ) {
2014-05-22 03:33:32 -04:00
fprintf ( stderr, "config.menu_lines is invalid. You need at least one visible line.\n" );
exit ( 1 );
}
2014-06-04 15:29:23 -04:00
if ( config.menu_columns == 0 ) {
2014-05-22 03:33:32 -04:00
fprintf ( stderr, "config.menu_columns is invalid. You need at least one visible column.\n" );
exit ( 1 );
}
2014-06-04 15:29:23 -04:00
if ( config.menu_width == 0 ) {
2014-05-22 03:33:32 -04:00
fprintf ( stderr, "config.menu_width is invalid. You cannot have a window with no width.\n" );
exit ( 1 );
}
2014-06-04 15:29:23 -04:00
if ( !( config.location >= WL_CENTER && config.location <= WL_WEST ) ) {
2014-05-22 03:33:32 -04:00
fprintf ( stderr, "config.location is invalid. ( %d >= %d >= %d) does not hold.\n",
WL_WEST, config.location, WL_CENTER );
exit ( 1 );
}
2014-06-04 15:29:23 -04:00
if ( !( config.hmode == TRUE || config.hmode == FALSE ) ) {
2014-05-22 03:33:32 -04:00
fprintf ( stderr, "config.hmode is invalid.\n" );
exit ( 1 );
}
}
int main ( int argc, char *argv[] )
{
2014-05-20 05:41:39 -04:00
// Initialize xdg, so we can grab the xdgCacheHome
2014-06-04 15:29:23 -04:00
if ( xdgInitHandle ( &xdg_handle ) == NULL ) {
2014-05-20 05:41:39 -04:00
fprintf ( stderr, "Failed to initialize XDG\n" );
return EXIT_FAILURE;
2014-01-10 04:35:38 -05:00
}
2014-05-20 05:41:39 -04:00
// Get the path to the cache dir.
cache_dir = xdgCacheHome ( &xdg_handle );
2014-05-20 05:41:39 -04:00
// Register cleanup function.
atexit ( cleanup );
2014-05-19 15:02:05 -04:00
// Get DISPLAY
2014-03-22 16:04:19 -04:00
char *display_str = getenv ( "DISPLAY" );
find_arg_str ( argc, argv, "-display", &display_str );
2014-01-15 03:27:03 -05:00
2014-06-04 15:29:23 -04:00
if ( !( display = XOpenDisplay ( display_str ) ) ) {
2014-03-22 16:04:19 -04:00
fprintf ( stderr, "cannot open display!\n" );
2014-01-10 04:35:38 -05:00
return EXIT_FAILURE;
}
2014-05-20 05:41:39 -04:00
// Load in config from X resources.
parse_xresource_options ( display );
2014-01-25 17:37:37 -05:00
2014-05-20 05:41:39 -04:00
// Parse command line for settings.
parse_cmd_options ( argc, argv );
2014-05-19 15:02:05 -04:00
2014-05-22 03:33:32 -04:00
// Sanity check
config_sanity_check ();
2014-01-25 17:37:37 -05:00
// Generate the font string for the line that indicates a selected item.
2014-06-04 15:29:23 -04:00
if ( asprintf ( &active_font, "%s:slant=italic", config.menu_font ) < 0 ) {
fprintf ( stderr, "Failed to construct active string: %s\n", strerror ( errno ) );
return EXIT_FAILURE;
}
2014-05-22 03:33:32 -04:00
// Set up X interaction.
2014-03-22 16:04:19 -04:00
signal ( SIGCHLD, catch_exit );
2014-05-26 04:32:26 -04:00
2014-05-20 05:41:39 -04:00
// Set error handle
2014-03-22 16:04:19 -04:00
XSync ( display, False );
2014-05-20 05:41:39 -04:00
xerror = XSetErrorHandler ( display_oops );
2014-03-22 16:04:19 -04:00
XSync ( display, False );
2014-01-10 04:35:38 -05:00
// determine numlock mask so we can bind on keys with and without it
2014-03-22 16:04:19 -04:00
XModifierKeymap *modmap = XGetModifierMapping ( display );
2014-01-10 04:35:38 -05:00
2014-06-04 15:29:23 -04:00
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] == XKeysymToKeycode ( display, XK_Num_Lock ) ) {
2014-03-22 16:04:19 -04:00
NumlockMask = ( 1 << i );
}
}
}
2014-01-10 04:35:38 -05:00
2014-03-22 16:04:19 -04:00
XFreeModifiermap ( modmap );
2014-01-10 04:35:38 -05:00
2014-03-22 16:04:19 -04:00
cache_client = winlist_new ();
cache_xattr = winlist_new ();
2014-01-10 04:35:38 -05:00
// X atom values
2014-06-04 15:29:23 -04:00
for ( int i = 0; i < NETATOMS; i++ ) {
2014-03-22 16:04:19 -04:00
netatoms[i] = XInternAtom ( display, netatom_names[i], False );
}
2014-01-23 05:39:12 -05:00
2014-04-24 02:23:16 -04:00
#ifdef HAVE_I3_IPC_H
// Check for i3
display_get_i3_path ( display );
#endif
2014-05-20 03:43:45 -04:00
2014-05-19 15:02:05 -04:00
2014-01-10 04:35:38 -05:00
// flags to run immediately and exit
2014-06-04 15:29:23 -04:00
if ( find_arg ( argc, argv, "-now" ) >= 0 ) {
2014-05-24 05:02:13 -04:00
run_switcher ( FALSE, WINDOW_SWITCHER );
2014-03-22 16:04:19 -04:00
}
2014-06-04 15:29:23 -04:00
else if ( find_arg ( argc, argv, "-rnow" ) >= 0 ) {
2014-05-24 05:02:13 -04:00
run_switcher ( FALSE, RUN_DIALOG );
2014-03-22 16:04:19 -04:00
}
2014-06-04 15:29:23 -04:00
else if ( find_arg ( argc, argv, "-snow" ) >= 0 ) {
2014-05-24 05:02:13 -04:00
run_switcher ( FALSE, SSH_DIALOG );
2014-03-22 16:04:19 -04:00
}
2014-06-04 15:29:23 -04:00
else if ( find_arg ( argc, argv, "-dmenu" ) >= 0 ) {
2014-03-22 16:04:19 -04:00
find_arg_str ( argc, argv, "-p", &dmenu_prompt );
2014-05-24 05:02:13 -04:00
run_switcher ( FALSE, DMENU_DIALOG );
2014-03-22 16:04:19 -04:00
}
2014-06-04 15:29:23 -04:00
else{
2014-01-23 14:33:13 -05:00
// Daemon mode, Listen to key presses..
2014-03-22 16:04:19 -04:00
parse_key ( config.window_key, &windows_modmask, &windows_keysym );
grab_key ( windows_modmask, windows_keysym );
2014-01-26 09:39:50 -05:00
2014-03-22 16:04:19 -04:00
parse_key ( config.run_key, &rundialog_modmask, &rundialog_keysym );
grab_key ( rundialog_modmask, rundialog_keysym );
2014-01-26 09:39:50 -05:00
2014-03-22 16:04:19 -04:00
parse_key ( config.ssh_key, &sshdialog_modmask, &sshdialog_keysym );
grab_key ( sshdialog_modmask, sshdialog_keysym );
2014-01-10 04:35:38 -05:00
2014-04-24 02:23:16 -04:00
// Main loop
2014-06-04 15:29:23 -04:00
for (;; ) {
2014-04-24 02:23:16 -04:00
XEvent ev;
// caches only live for a single event
2014-03-22 16:04:19 -04:00
winlist_empty ( cache_xattr );
winlist_empty ( cache_client );
2014-01-10 04:35:38 -05:00
// block and wait for something
2014-03-22 16:04:19 -04:00
XNextEvent ( display, &ev );
2014-01-10 04:35:38 -05:00
2014-06-04 15:29:23 -04:00
if ( ev.xany.window == None ) {
2014-03-22 16:04:19 -04:00
continue;
}
2014-01-10 04:35:38 -05:00
2014-06-04 15:29:23 -04:00
if ( ev.type == KeyPress ) {
2014-03-22 16:04:19 -04:00
handle_keypress ( &ev );
}
}
}
2014-01-23 14:33:13 -05:00
return EXIT_SUCCESS;
}