nsxiv/main.c

340 lines
6.2 KiB
C
Raw Normal View History

2011-01-17 13:57:59 +00:00
/* sxiv: main.c
* Copyright (c) 2011 Bert Muennich <muennich at informatik.hu-berlin.de>
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 2 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
*/
2011-01-17 15:41:50 +00:00
#include <stdlib.h>
#include <stdio.h>
2011-01-22 22:27:29 +00:00
#include <sys/select.h>
2011-01-17 15:41:50 +00:00
#include <X11/Xlib.h>
2011-01-21 09:30:10 +00:00
#include <X11/Xutil.h>
#include <X11/keysym.h>
2011-01-17 13:57:59 +00:00
#include "sxiv.h"
#include "image.h"
#include "options.h"
#include "window.h"
void on_keypress(XEvent*);
void on_buttonpress(XEvent*);
2011-01-26 17:38:54 +00:00
void on_configurenotify(XEvent*);
void update_title();
static void (*handler[LASTEvent])(XEvent*) = {
2011-01-20 20:44:34 +00:00
[KeyPress] = on_keypress,
2011-01-26 17:38:54 +00:00
[ButtonPress] = on_buttonpress,
2011-01-20 20:44:34 +00:00
[ConfigureNotify] = on_configurenotify
};
img_t img;
win_t win;
const char **filenames;
int filecnt, fileidx;
2011-01-22 22:27:29 +00:00
unsigned char timeout;
#define TITLE_LEN 256
char win_title[TITLE_LEN];
void run() {
2011-01-22 22:27:29 +00:00
int xfd;
fd_set fds;
struct timeval t;
XEvent ev;
2011-01-17 13:57:59 +00:00
2011-01-22 22:27:29 +00:00
timeout = 0;
while (1) {
if (timeout) {
t.tv_sec = 0;
t.tv_usec = 250;
xfd = ConnectionNumber(win.env.dpy);
FD_ZERO(&fds);
FD_SET(xfd, &fds);
if (!XPending(win.env.dpy) && !select(xfd + 1, &fds, 0, 0, &t)) {
img_render(&img, &win);
timeout = 0;
}
}
if (!XNextEvent(win.env.dpy, &ev) && handler[ev.type])
handler[ev.type](&ev);
}
}
2011-01-17 13:57:59 +00:00
int main(int argc, char **argv) {
int i;
2011-01-19 17:16:44 +00:00
parse_options(argc, argv);
if (!options->filecnt) {
print_usage();
exit(1);
}
if (!(filenames = (const char**) malloc(options->filecnt * sizeof(char*))))
DIE("could not allocate memory");
fileidx = 0;
filecnt = 0;
for (i = 0; i < options->filecnt; ++i) {
if (!(img_load(&img, options->filenames[i]) < 0))
filenames[filecnt++] = options->filenames[i];
}
if (!filecnt) {
fprintf(stderr, "sxiv: no valid image filename given, aborting\n");
exit(1);
}
win_open(&win);
2011-01-21 11:57:35 +00:00
img_init(&img, &win);
2011-01-17 15:41:50 +00:00
img_load(&img, filenames[fileidx]);
2011-01-21 11:13:52 +00:00
img_render(&img, &win);
update_title();
run();
cleanup();
2011-01-17 15:41:50 +00:00
2011-01-17 13:57:59 +00:00
return 0;
}
2011-01-18 14:33:25 +00:00
void cleanup() {
static int in = 0;
if (!in++) {
2011-01-21 11:57:35 +00:00
img_free(&img);
win_close(&win);
}
}
void on_keypress(XEvent *ev) {
2011-01-21 09:30:10 +00:00
char key;
KeySym keysym;
2011-01-21 12:48:02 +00:00
int changed;
if (!ev)
return;
2011-01-21 12:48:02 +00:00
XLookupString(&ev->xkey, &key, 1, &keysym, NULL);
changed = 0;
switch (keysym) {
case XK_Escape:
cleanup();
2011-01-21 09:30:10 +00:00
exit(2);
case XK_space:
key = 'n';
break;
case XK_BackSpace:
key = 'p';
break;
2011-01-28 12:42:39 +00:00
case XK_Left:
key = 'h';
break;
case XK_Down:
key = 'j';
break;
case XK_Up:
key = 'k';
break;
case XK_Right:
key = 'l';
break;
2011-01-21 09:30:10 +00:00
}
switch (key) {
case 'q':
cleanup();
exit(0);
2011-01-21 12:48:02 +00:00
2011-01-23 17:20:08 +00:00
/* navigate image list */
2011-01-21 09:30:10 +00:00
case 'n':
if (fileidx + 1 < filecnt) {
img_load(&img, filenames[++fileidx]);
2011-01-21 12:48:02 +00:00
changed = 1;
2011-01-20 15:04:34 +00:00
}
break;
2011-01-21 09:30:10 +00:00
case 'p':
2011-01-20 15:04:34 +00:00
if (fileidx > 0) {
img_load(&img, filenames[--fileidx]);
changed = 1;
}
break;
case '[':
if (fileidx != 0) {
fileidx = MAX(0, fileidx - 10);
img_load(&img, filenames[fileidx]);
changed = 1;
}
break;
case ']':
if (fileidx != filecnt - 1) {
fileidx = MIN(fileidx + 10, filecnt - 1);
img_load(&img, filenames[fileidx]);
changed = 1;
}
break;
case 'g':
if (fileidx != 0) {
fileidx = 0;
img_load(&img, filenames[fileidx]);
changed = 1;
}
break;
case 'G':
if (fileidx != filecnt - 1) {
fileidx = filecnt - 1;
img_load(&img, filenames[fileidx]);
2011-01-21 12:48:02 +00:00
changed = 1;
2011-01-20 15:04:34 +00:00
}
break;
2011-01-21 12:48:02 +00:00
/* zooming */
2011-01-21 09:30:10 +00:00
case '+':
case '=':
2011-01-21 12:48:02 +00:00
changed = img_zoom_in(&img);
2011-01-20 22:22:00 +00:00
break;
2011-01-21 09:30:10 +00:00
case '-':
2011-01-21 12:48:02 +00:00
changed = img_zoom_out(&img);
2011-01-20 22:22:00 +00:00
break;
2011-01-21 12:48:02 +00:00
/* panning */
case 'h':
changed = img_pan(&img, &win, PAN_LEFT);
break;
case 'j':
changed = img_pan(&img, &win, PAN_DOWN);
break;
case 'k':
changed = img_pan(&img, &win, PAN_UP);
break;
case 'l':
changed = img_pan(&img, &win, PAN_RIGHT);
break;
2011-01-23 15:14:41 +00:00
2011-01-26 13:42:10 +00:00
/* rotation */
case '<':
changed = img_rotate_left(&img, &win);
break;
case '>':
changed = img_rotate_right(&img, &win);
break;
/* control window */
2011-01-23 15:14:41 +00:00
case 'f':
win_toggle_fullscreen(&win);
break;
/* miscellaneous */
case 'a':
changed = img_toggle_antialias(&img);
break;
2011-01-21 12:48:02 +00:00
}
if (changed) {
img_render(&img, &win);
update_title();
2011-01-22 22:27:29 +00:00
timeout = 0;
}
}
2011-01-26 17:38:54 +00:00
void on_buttonpress(XEvent *ev) {
int changed;
unsigned int mask;
if (!ev)
return;
mask = CLEANMASK(ev->xbutton.state);
changed = 0;
switch (ev->xbutton.button) {
case Button1:
if (fileidx + 1 < filecnt) {
img_load(&img, filenames[++fileidx]);
changed = 1;
}
break;
case Button3:
if (fileidx > 0) {
img_load(&img, filenames[--fileidx]);
changed = 1;
}
break;
case Button4:
if (mask == ControlMask)
changed = img_zoom_in(&img);
else if (mask == ShiftMask)
changed = img_pan(&img, &win, PAN_LEFT);
else
changed = img_pan(&img, &win, PAN_UP);
break;
case Button5:
if (mask == ControlMask)
changed = img_zoom_out(&img);
else if (mask == ShiftMask)
changed = img_pan(&img, &win, PAN_RIGHT);
else
changed = img_pan(&img, &win, PAN_DOWN);
break;
case 6:
changed = img_pan(&img, &win, PAN_LEFT);
break;
case 7:
changed = img_pan(&img, &win, PAN_RIGHT);
break;
}
if (changed) {
img_render(&img, &win);
update_title();
timeout = 0;
}
}
void on_configurenotify(XEvent *ev) {
if (!ev)
return;
2011-01-22 22:27:29 +00:00
if (win_configure(&win, &ev->xconfigure)) {
img.checkpan = 1;
timeout = 1;
}
}
void update_title() {
int n;
n = snprintf(win_title, TITLE_LEN, "sxiv: [%d/%d] <%d%%> %s", fileidx + 1,
filecnt, (int) (img.zoom * 100.0), filenames[fileidx]);
if (n >= TITLE_LEN) {
win_title[TITLE_LEN - 2] = '.';
win_title[TITLE_LEN - 3] = '.';
win_title[TITLE_LEN - 4] = '.';
}
win_set_title(&win, win_title);
}