X-Git-Url: https://git.sur5r.net/?a=blobdiff_plain;f=src%2Ffloating.c;h=2ac4afda0d39bb5e32a14aa38ca3715a50ec8905;hb=4be3178d4d360c2996217d811e61161c84d25898;hp=9d4e1cf8e438ff379bf9e9f2fbbd899dc8821936;hpb=0639a7d95ba29229db7d545ad758ca6946ad293d;p=i3%2Fi3 diff --git a/src/floating.c b/src/floating.c index 9d4e1cf8..2ac4afda 100644 --- a/src/floating.c +++ b/src/floating.c @@ -147,15 +147,30 @@ void floating_enable(Con *con, bool automatic) { } } + TAILQ_INSERT_TAIL(&(nc->nodes_head), con, nodes); + TAILQ_INSERT_TAIL(&(nc->focus_head), con, focused); + /* render the cons to get initial window_rect correct */ render_con(nc, false); render_con(con, false); - TAILQ_INSERT_TAIL(&(nc->nodes_head), con, nodes); - TAILQ_INSERT_TAIL(&(nc->focus_head), con, focused); // TODO: don’t influence focus handling when Con was not focused before. if (set_focus) con_focus(con); + + /* Check if we need to re-assign it to a different workspace because of its + * coordinates and exit if that was done successfully. */ + if (floating_maybe_reassign_ws(nc)) + return; + + /* Sanitize coordinates: Check if they are on any output */ + if (get_output_containing(nc->rect.x, nc->rect.y) != NULL) + return; + + ELOG("No output found at destination coordinates, centering floating window on current ws\n"); + Con *ws = nc->parent; + nc->rect.x = ws->rect.x + (ws->rect.width / 2) - (nc->rect.width / 2); + nc->rect.y = ws->rect.y + (ws->rect.height / 2) - (nc->rect.height / 2); } void floating_disable(Con *con, bool automatic) { @@ -225,16 +240,51 @@ void floating_raise_con(Con *con) { TAILQ_INSERT_TAIL(&(con->parent->floating_head), con, floating_windows); } +/* + * Checks if con’s coordinates are within its workspace and re-assigns it to + * the actual workspace if not. + * + */ +bool floating_maybe_reassign_ws(Con *con) { + Output *output = get_output_containing( + con->rect.x + (con->rect.width / 2), + con->rect.y + (con->rect.height / 2)); + + if (!output) { + ELOG("No output found at destination coordinates?\n"); + return false; + } + + if (con_get_output(con) == output->con) { + DLOG("still the same ws\n"); + return false; + } + + DLOG("Need to re-assign!\n"); + + Con *content = output_get_content(output->con); + Con *ws = TAILQ_FIRST(&(content->focus_head)); + DLOG("Moving con %p / %s to workspace %p / %s\n", con, con->name, ws, ws->name); + con_move_to_workspace(con, ws); + con_focus(con_descend_focused(con)); + return true; +} + DRAGGING_CB(drag_window_callback) { struct xcb_button_press_event_t *event = extra; /* Reposition the client correctly while moving */ con->rect.x = old_rect->x + (new_x - event->root_x); con->rect.y = old_rect->y + (new_y - event->root_y); - /* TODO: don’t re-render the whole tree just because we change - * coordinates of a floating window */ + + render_con(con, false); + x_push_node(con); + xcb_flush(conn); + + /* Check if we cross workspace boundaries while moving */ + if (!floating_maybe_reassign_ws(con)) + return; tree_render(); - x_push_changes(croot); } /* @@ -331,7 +381,7 @@ void floating_resize_window(Con *con, bool proportional, if (event->event_y <= (con->rect.height / 2)) corner |= BORDER_TOP; - else corner |= BORDER_RIGHT; + else corner |= BORDER_BOTTOM; struct resize_window_callback_params params = { corner, proportional, event };