diff options
| author | Dana Jansens <danakj@orodu.net> | 2007-07-11 20:01:47 +0000 |
|---|---|---|
| committer | Dana Jansens <danakj@orodu.net> | 2007-07-11 20:01:47 +0000 |
| commit | 2a4cd4f6bd9f6b5a43ea6a825e964fb1f4dccd80 (patch) | |
| tree | 0f779773354ae9a7a903fbc8f3c1118f75559cc9 /openbox/actions/moveresizeto.c | |
| parent | 2827b2ce899fb28ab05b369c90126f053d021b6f (diff) | |
renaming movetofromedge->movetoedge and moveto->moveresizeto
Diffstat (limited to 'openbox/actions/moveresizeto.c')
| -rw-r--r-- | openbox/actions/moveresizeto.c | 156 |
1 files changed, 156 insertions, 0 deletions
diff --git a/openbox/actions/moveresizeto.c b/openbox/actions/moveresizeto.c new file mode 100644 index 00000000..4d23be45 --- /dev/null +++ b/openbox/actions/moveresizeto.c @@ -0,0 +1,156 @@ +#include "openbox/actions.h" +#include "openbox/client.h" +#include "openbox/screen.h" +#include "openbox/frame.h" +#include <stdlib.h> /* for atoi */ + +typedef struct { + gboolean xcenter; + gboolean ycenter; + gboolean xopposite; + gboolean yopposite; + gint x; + gint y; + gint w; + gint h; + gint monitor; +} Options; + +static gpointer setup_func(ObParseInst *i, xmlDocPtr doc, xmlNodePtr node); +static void free_func(gpointer options); +static gboolean run_func(ObActionsData *data, gpointer options); + +void action_moveto_startup() +{ + actions_register("MoveTo", + setup_func, + free_func, + run_func, + NULL, NULL); +} + +static gpointer setup_func(ObParseInst *i, xmlDocPtr doc, xmlNodePtr node) +{ + xmlNodePtr n; + Options *o; + + o = g_new0(Options, 1); + o->x = G_MININT; + o->y = G_MININT; + o->w = G_MININT; + o->h = G_MININT; + o->monitor = -1; + + if ((n = parse_find_node("x", node))) { + gchar *s = parse_string(doc, n); + if (!g_ascii_strcasecmp(s, "center")) + o->xcenter = TRUE; + else { + if (s[0] == '-') + o->xopposite = TRUE; + if (s[0] == '-' || s[0] == '+') + o->x = atoi(s+1); + else + o->x = atoi(s); + } + g_free(s); + } + + if ((n = parse_find_node("y", node))) { + gchar *s = parse_string(doc, n); + if (!g_ascii_strcasecmp(s, "center")) + o->ycenter = TRUE; + else { + if (s[0] == '-') + o->yopposite = TRUE; + if (s[0] == '-' || s[0] == '+') + o->y = atoi(s+1); + else + o->y = atoi(s); + } + g_free(s); + } + + if ((n = parse_find_node("width", node))) + o->w = parse_int(doc, n) - 1; + if ((n = parse_find_node("height", node))) + o->h = parse_int(doc, n) - 1; + + if ((n = parse_find_node("monitor", node))) + o->monitor = parse_int(doc, n) - 1; + + return o; +} + +static void free_func(gpointer options) +{ + Options *o = options; + + g_free(o); +} + +/* Always return FALSE because its not interactive */ +static gboolean run_func(ObActionsData *data, gpointer options) +{ + Options *o = options; + + if (data->client) { + Rect *area, *carea; + ObClient *c; + gint mon, cmon; + gint x, y, lw, lh, w, h; + + c = data->client; + mon = o->monitor; + cmon = client_monitor(c); + if (mon < 0) mon = cmon; + area = screen_area(c->desktop, mon, NULL); + carea = screen_area(c->desktop, cmon, NULL); + + w = o->w; + if (w == G_MININT) w = c->area.width; + + h = o->h; + if (h == G_MININT) h = c->area.height; + + /* it might not be able to resize how they requested, so find out what + it will actually be resized to */ + x = c->area.x; + y = c->area.y; + client_try_configure(c, &x, &y, &w, &h, &lw, &lh, TRUE); + + /* get the frame's size */ + w += c->frame->size.left + c->frame->size.right; + h += c->frame->size.top + c->frame->size.bottom; + + x = o->x; + if (o->xcenter) x = (area->width - w) / 2; + else if (x == G_MININT) x = c->frame->area.x - carea->x; + else if (o->xopposite) x = area->width - w; + x += area->x; + + y = o->y; + if (o->ycenter) y = (area->height - h) / 2; + else if (y == G_MININT) y = c->frame->area.y - carea->y; + else if (o->yopposite) y = area->height - h; + y += area->y; + + /* get the client's size back */ + w -= c->frame->size.left + c->frame->size.right; + h -= c->frame->size.top + c->frame->size.bottom; + + frame_frame_gravity(c->frame, &x, &y); /* get the client coords */ + client_try_configure(c, &x, &y, &w, &h, &lw, &lh, TRUE); + /* force it on screen if its moving to another monitor */ + client_find_onscreen(c, &x, &y, w, h, mon != cmon); + + actions_client_move(data, TRUE); + client_configure(c, x, y, w, h, TRUE, TRUE, FALSE); + actions_client_move(data, FALSE); + + g_free(area); + g_free(carea); + } + + return FALSE; +} |
