plan9port/src/cmd/rio/menu.c

289 lines
5.2 KiB
C
Raw Normal View History

2004-03-21 04:27:28 +00:00
/* Copyright (c) 1994-1996 David Hogan, see README for licence details */
#include <stdio.h>
#include <signal.h>
#include <unistd.h>
#include <stdlib.h>
#include <sys/wait.h>
2004-03-21 04:27:28 +00:00
#include <X11/X.h>
#include <X11/Xlib.h>
#include <X11/Xutil.h>
#include "dat.h"
#include "fns.h"
Client *hiddenc[MAXHIDDEN];
int numhidden;
char *b3items[B3FIXED+MAXHIDDEN+1] =
{
"New",
"Reshape",
"Move",
"Delete",
"Hide",
0,
};
Menu b3menu =
{
b3items,
};
Menu egg =
{
version,
};
void
button(XButtonEvent *e)
{
int n, shift;
Client *c;
Window dw;
ScreenInfo *s;
curtime = e->time;
s = getscreen(e->root);
if (s == 0)
return;
c = getclient(e->window, 0);
2004-04-19 05:54:21 +00:00
if(c){
if (debug) fprintf(stderr, "but: e x=%d y=%d c x=%d y=%d dx=%d dy=%d BORDR %d\n",
e->x, e->y, c->x, c->y, c->dx, c->dy, BORDER);
2004-04-19 05:54:21 +00:00
if(borderorient(c, e->x, e->y) != BorderUnknown){
switch (e->button) {
case Button1:
case Button2:
reshape(c, e->button, pull, e);
return;
case Button3:
move(c, Button3);
return;
default:
return;
}
2004-04-19 05:54:21 +00:00
}
2004-03-21 04:27:28 +00:00
e->x += c->x - BORDER;
e->y += c->y - BORDER;
2004-04-19 05:54:21 +00:00
} else if (e->window != e->root) {
if (debug) fprintf(stderr, "but no client: e x=%d y=%d\n",
e->x, e->y);
2004-03-21 04:27:28 +00:00
XTranslateCoordinates(dpy, e->window, s->root, e->x, e->y,
&e->x, &e->y, &dw);
}
2004-03-21 04:27:28 +00:00
switch (e->button) {
case Button1:
if (c) {
XMapRaised(dpy, c->parent);
top(c);
active(c);
}
return;
case Button2:
if ((e->state&(ShiftMask|ControlMask))==(ShiftMask|ControlMask))
menuhit(e, &egg);
return;
default:
return;
case Button3:
break;
}
if (current && current->screen == s)
cmapnofocus(s);
switch (n = menuhit(e, &b3menu)) {
case 0: /* New */
spawn(s);
break;
case 1: /* Reshape */
reshape(selectwin(1, 0, s), Button3, sweep, 0);
2004-03-21 04:27:28 +00:00
break;
case 2: /* Move */
move(selectwin(0, 0, s), Button3);
2004-03-21 04:27:28 +00:00
break;
case 3: /* Delete */
shift = 0;
c = selectwin(1, &shift, s);
delete(c, shift);
break;
case 4: /* Hide */
hide(selectwin(1, 0, s));
break;
default: /* unhide window */
unhide(n - B3FIXED, 1);
break;
case -1: /* nothing */
break;
}
if (current && current->screen == s)
cmapfocus(current);
}
void
spawn(ScreenInfo *s)
{
/*
* ugly dance to cause sweeping for terminals.
* the very next window created will require sweeping.
* hope it's created by the program we're about to
* exec!
*/
isNew = 1;
2004-03-21 04:27:28 +00:00
/*
* ugly dance to avoid leaving zombies. Could use SIGCHLD,
* but it's not very portable.
*/
if (fork() == 0) {
if (fork() == 0) {
close(ConnectionNumber(dpy));
if (s->display[0] != '\0')
putenv(s->display);
2004-09-20 15:08:57 +00:00
signal(SIGINT, SIG_DFL);
signal(SIGTERM, SIG_DFL);
signal(SIGHUP, SIG_DFL);
2004-03-21 04:27:28 +00:00
if (termprog != NULL) {
execl(shell, shell, "-c", termprog, 0);
2004-04-19 19:35:17 +00:00
fprintf(stderr, "rio: exec %s", shell);
2004-03-21 04:27:28 +00:00
perror(" failed");
}
execlp("9term", "9term", "-w", 0);
execlp("xterm", "xterm", "-ut", 0);
2004-04-19 19:35:17 +00:00
perror("rio: exec 9term/xterm failed");
2004-03-21 04:27:28 +00:00
exit(1);
}
exit(0);
}
wait((int *) 0);
}
void
reshape(Client *c, int but, int (*fn)(Client*, int, XButtonEvent *), XButtonEvent *e)
2004-03-21 04:27:28 +00:00
{
int odx, ody;
if (c == 0)
return;
odx = c->dx;
ody = c->dy;
if (fn(c, but, e) == 0)
2004-03-21 04:27:28 +00:00
return;
active(c);
top(c);
XRaiseWindow(dpy, c->parent);
XMoveResizeWindow(dpy, c->parent, c->x-BORDER, c->y-BORDER,
c->dx+2*BORDER, c->dy+2*BORDER);
if (c->dx == odx && c->dy == ody)
sendconfig(c);
else
XMoveResizeWindow(dpy, c->window, BORDER, BORDER, c->dx, c->dy);
}
void
move(Client *c, int but)
2004-03-21 04:27:28 +00:00
{
if (c == 0)
return;
if (drag(c, but) == 0)
2004-03-21 04:27:28 +00:00
return;
active(c);
top(c);
XRaiseWindow(dpy, c->parent);
XMoveWindow(dpy, c->parent, c->x-BORDER, c->y-BORDER);
sendconfig(c);
}
void
delete(Client *c, int shift)
{
if (c == 0)
return;
if ((c->proto & Pdelete) && !shift)
2004-08-06 14:14:38 +00:00
sendcmessage(c->window, wm_protocols, wm_delete, 0, 0);
2004-03-21 04:27:28 +00:00
else
XKillClient(dpy, c->window); /* let event clean up */
}
void
hide(Client *c)
{
if (c == 0 || numhidden == MAXHIDDEN)
return;
if (hidden(c)) {
2004-04-19 19:35:17 +00:00
fprintf(stderr, "rio: already hidden: %s\n", c->label);
2004-03-21 04:27:28 +00:00
return;
}
XUnmapWindow(dpy, c->parent);
XUnmapWindow(dpy, c->window);
setstate(c, IconicState);
if (c == current)
nofocus();
hiddenc[numhidden] = c;
b3items[B3FIXED+numhidden] = c->label;
numhidden++;
b3items[B3FIXED+numhidden] = 0;
}
void
unhide(int n, int map)
{
Client *c;
int i;
if (n >= numhidden) {
2004-04-19 19:35:17 +00:00
fprintf(stderr, "rio: unhide: n %d numhidden %d\n", n, numhidden);
2004-03-21 04:27:28 +00:00
return;
}
c = hiddenc[n];
if (!hidden(c)) {
2004-04-19 19:35:17 +00:00
fprintf(stderr, "rio: unhide: not hidden: %s(0x%x)\n",
c->label, (int)c->window);
2004-03-21 04:27:28 +00:00
return;
}
if (map) {
XMapWindow(dpy, c->window);
XMapRaised(dpy, c->parent);
setstate(c, NormalState);
active(c);
top(c);
}
numhidden--;
for (i = n; i < numhidden; i ++) {
hiddenc[i] = hiddenc[i+1];
b3items[B3FIXED+i] = b3items[B3FIXED+i+1];
}
b3items[B3FIXED+numhidden] = 0;
}
void
unhidec(Client *c, int map)
{
int i;
for (i = 0; i < numhidden; i++)
if (c == hiddenc[i]) {
unhide(i, map);
return;
}
2004-04-19 19:35:17 +00:00
fprintf(stderr, "rio: unhidec: not hidden: %s(0x%x)\n",
c->label, (int)c->window);
2004-03-21 04:27:28 +00:00
}
void
renamec(Client *c, char *name)
{
int i;
if (name == 0)
name = "???";
c->label = name;
if (!hidden(c))
return;
for (i = 0; i < numhidden; i++)
if (c == hiddenc[i]) {
b3items[B3FIXED+i] = name;
return;
}
}