openbox/plugins/placement/placement.c

178 lines
5.4 KiB
C
Raw Normal View History

#include "kernel/dispatch.h"
#include "kernel/client.h"
2003-07-30 17:06:09 +00:00
#include "kernel/group.h"
#include "kernel/frame.h"
#include "kernel/screen.h"
#include "kernel/openbox.h"
#include "parser/parse.h"
2003-03-21 07:53:29 +00:00
#include "history.h"
2003-03-21 07:33:07 +00:00
#include <glib.h>
static gboolean history;
2003-05-24 15:35:26 +00:00
static void parse_xml(xmlDocPtr doc, xmlNodePtr node, void *d)
{
2003-05-24 15:35:26 +00:00
xmlNodePtr n;
node = node->xmlChildrenNode;
2003-05-24 15:35:26 +00:00
if ((n = parse_find_node("remember", node)))
history = parse_bool(doc, n);
}
2003-03-21 07:53:29 +00:00
2003-03-23 19:42:47 +00:00
void plugin_setup_config()
{
history = TRUE;
2003-05-24 15:35:26 +00:00
parse_register("placement", parse_xml, NULL);
2003-03-23 19:42:47 +00:00
}
static Rect* pick_head(ObClient *c)
{
/* try direct parent first */
if (c->transient_for && c->transient_for != TRAN_GROUP) {
return screen_area_monitor(c->desktop,
client_monitor(c->transient_for));
}
/* more than one guy in his group (more than just him) */
if (c->group && c->group->members->next) {
GSList *it;
/* try on the client's desktop */
for (it = c->group->members; it; it = g_slist_next(it)) {
ObClient *itc = it->data;
if (itc != c &&
(itc->desktop == c->desktop ||
itc->desktop == DESKTOP_ALL || c->desktop == DESKTOP_ALL))
return screen_area_monitor(c->desktop,
client_monitor(it->data));
}
/* try on all desktops */
for (it = c->group->members; it; it = g_slist_next(it)) {
ObClient *itc = it->data;
if (itc != c)
return screen_area_monitor(c->desktop,
client_monitor(it->data));
}
}
return NULL;
}
2003-07-10 06:38:42 +00:00
static void place_random(ObClient *c)
2003-03-21 07:33:07 +00:00
{
int l, r, t, b;
int x, y;
Rect *area;
if (ob_state() == OB_STATE_STARTING) return;
area = pick_head(c);
if (!area)
area = screen_area_monitor(c->desktop,
g_random_int_range(0, screen_num_monitors));
2003-03-21 07:33:07 +00:00
l = area->x;
t = area->y;
r = area->x + area->width - c->frame->area.width;
b = area->y + area->height - c->frame->area.height;
2003-03-21 07:53:29 +00:00
if (r > l) x = g_random_int_range(l, r + 1);
else x = 0;
if (b > t) y = g_random_int_range(t, b + 1);
else y = 0;
2003-03-21 07:33:07 +00:00
frame_frame_gravity(c->frame, &x, &y); /* get where the client should be */
client_configure(c, OB_CORNER_TOPLEFT, x, y, c->area.width, c->area.height,
2003-03-21 07:33:07 +00:00
TRUE, TRUE);
}
2003-03-21 09:31:36 +00:00
static void event(ObEvent *e, void *foo)
2003-03-21 07:33:07 +00:00
{
g_assert(e->type == Event_Client_New);
/* requested a position */
if (e->data.c.client->positioned) return;
2003-07-30 17:06:09 +00:00
if (e->data.c.client->transient_for) {
if (e->data.c.client->transient_for != OB_TRAN_GROUP) {
ObClient *c = e->data.c.client;
ObClient *p = e->data.c.client->transient_for;
2003-08-02 06:27:05 +00:00
int x = (p->frame->area.width - c->frame->area.width) / 2 +
2003-07-30 17:06:09 +00:00
p->frame->area.x;
2003-08-02 06:27:05 +00:00
int y = (p->frame->area.height - c->frame->area.height) / 2 +
2003-07-30 17:06:09 +00:00
p->frame->area.y;
client_configure(c, OB_CORNER_TOPLEFT, x, y,
c->area.width, c->area.height,
TRUE, TRUE);
return;
} else {
GSList *it;
ObClient *c = e->data.c.client;
gboolean first = TRUE;
int l, r, t, b;
for (it = c->group->members; it; it = it->next) {
ObClient *m = it->data;
if (!(m == c || m->transient_for)) {
if (first) {
l = m->frame->area.x;
t = m->frame->area.y;
r = m->frame->area.x + m->frame->area.width - 1;
b = m->frame->area.y + m->frame->area.height - 1;
first = FALSE;
} else {
l = MIN(l, m->frame->area.x);
t = MIN(t, m->frame->area.y);
r = MAX(r, m->frame->area.x +m->frame->area.width - 1);
b = MAX(b, m->frame->area.y +m->frame->area.height -1);
}
}
}
if (!first) {
int x = ((r + 1 - l) - c->frame->area.width) / 2 + l;
int y = ((b + 1 - t) - c->frame->area.height) / 2 + t;
client_configure(c, OB_CORNER_TOPLEFT, x, y,
c->area.width, c->area.height,
TRUE, TRUE);
return;
}
}
}
/* center parentless dialogs on the screen */
if (e->data.c.client->type == OB_CLIENT_TYPE_DIALOG) {
Rect *area;
ObClient *c = e->data.c.client;
int x, y;
area = pick_head(c);
if (!area)
area = screen_area_monitor(c->desktop, 0);
x = (area->width - c->frame->area.width) / 2 + area->x;
y = (area->height - c->frame->area.height) / 2 + area->y;
client_configure(c, OB_CORNER_TOPLEFT, x, y,
c->area.width, c->area.height,
TRUE, TRUE);
return;
}
if (!history || !place_history(e->data.c.client))
2003-03-21 07:53:29 +00:00
place_random(e->data.c.client);
2003-03-21 07:33:07 +00:00
}
void plugin_startup()
{
2003-03-21 09:31:36 +00:00
dispatch_register(Event_Client_New, (EventHandler)event, NULL);
2003-03-21 07:53:29 +00:00
history_startup();
2003-03-21 07:33:07 +00:00
}
void plugin_shutdown()
{
2003-03-21 09:31:36 +00:00
dispatch_register(0, (EventHandler)event, NULL);
2003-03-21 07:53:29 +00:00
history_shutdown();
2003-03-21 07:33:07 +00:00
}