openbox/actions/moveresizeto.c (raw)
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 |
#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_moveresizeto_startup() { actions_register("MoveResizeTo", setup_func, free_func, run_func, NULL, NULL); } static void parse_coord(xmlDocPtr doc, xmlNodePtr n, gint *pos, gboolean *opposite, gboolean *center) { gchar *s = parse_string(doc, n); if (!g_ascii_strcasecmp(s, "center")) *center = TRUE; else { if (s[0] == '-') *opposite = TRUE; if (s[0] == '-' || s[0] == '+') *pos = atoi(s+1); else *pos = atoi(s); } g_free(s); } 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))) parse_coord(doc, n, &o->x, &o->xopposite, &o->xcenter); if ((n = parse_find_node("y", node))) parse_coord(doc, n, &o->y, &o->yopposite, &o->ycenter); 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; } |