/* Metacity Workspaces */ /* * Copyright (C) 2001 Havoc Pennington * * This program is free software; you can redistribute it and/or * modify it under the terms of the GNU General Public License as * published by the Free Software Foundation; either version 2 of the * License, or (at your option) any later version. * * This program is distributed in the hope that it will be useful, but * WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA * 02111-1307, USA. */ #include "workspace.h" #include "errors.h" #include void meta_workspace_queue_calc_showing (MetaWorkspace *workspace); static int set_number_of_spaces_hint (MetaScreen *screen); static int set_active_space_hint (MetaScreen *screen); static int set_work_area_hint (MetaScreen *screen); MetaWorkspace* meta_workspace_new (MetaScreen *screen) { MetaWorkspace *workspace; workspace = g_new (MetaWorkspace, 1); workspace->screen = screen; workspace->screen->display->workspaces = g_list_append (workspace->screen->display->workspaces, workspace); workspace->windows = NULL; workspace->work_area.x = 0; workspace->work_area.y = 0; workspace->work_area.width = screen->width; workspace->work_area.height = screen->height; workspace->work_area_invalid = TRUE; /* Update hint for current number of workspaces */ set_number_of_spaces_hint (screen); return workspace; } void meta_workspace_free (MetaWorkspace *workspace) { GList *tmp; MetaScreen *screen; g_return_if_fail (workspace != workspace->screen->active_workspace); /* Here we assume all the windows are already on another workspace * as well, so they won't be "orphaned" */ tmp = workspace->windows; while (tmp != NULL) { GList *next; MetaWindow *window = tmp->data; next = tmp->next; /* pop front of list we're iterating over */ meta_workspace_remove_window (workspace, window); g_assert (window->workspaces != NULL); tmp = next; } g_assert (workspace->windows == NULL); screen = workspace->screen; workspace->screen->display->workspaces = g_list_remove (workspace->screen->display->workspaces, workspace); g_free (workspace); /* Update hint for current number of workspaces */ set_number_of_spaces_hint (screen); } void meta_workspace_add_window (MetaWorkspace *workspace, MetaWindow *window) { g_return_if_fail (!meta_workspace_contains_window (workspace, window)); workspace->windows = g_list_prepend (workspace->windows, window); window->workspaces = g_list_prepend (window->workspaces, workspace); meta_window_set_current_workspace_hint (window); meta_window_queue_calc_showing (window); if (window->has_struts) { meta_topic (META_DEBUG_WORKAREA, "Invalidating work area of workspace %d since we're adding window %s to it\n", meta_workspace_index (workspace), window->desc); meta_workspace_invalidate_work_area (workspace); } /* queue a move_resize since changing workspaces may change * the relevant struts */ meta_window_queue_move_resize (window); } void meta_workspace_remove_window (MetaWorkspace *workspace, MetaWindow *window) { g_return_if_fail (meta_workspace_contains_window (workspace, window)); workspace->windows = g_list_remove (workspace->windows, window); window->workspaces = g_list_remove (window->workspaces, workspace); meta_window_set_current_workspace_hint (window); meta_window_queue_calc_showing (window); if (window->has_struts) { meta_topic (META_DEBUG_WORKAREA, "Invalidating work area of workspace %d since we're removing window %s from it\n", meta_workspace_index (workspace), window->desc); meta_workspace_invalidate_work_area (workspace); } /* queue a move_resize since changing workspaces may change * the relevant struts */ meta_window_queue_move_resize (window); } void meta_workspace_relocate_windows (MetaWorkspace *workspace, MetaWorkspace *new_home) { GList *tmp; GList *copy; g_return_if_fail (workspace != new_home); /* can't modify list we're iterating over */ copy = g_list_copy (workspace->windows); tmp = copy; while (tmp != NULL) { MetaWindow *window = tmp->data; meta_workspace_add_window (new_home, window); meta_workspace_remove_window (workspace, window); tmp = tmp->next; } g_list_free (copy); g_assert (workspace->windows == NULL); } gboolean meta_workspace_contains_window (MetaWorkspace *workspace, MetaWindow *window) { return g_list_find (workspace->windows, window) != NULL; } void meta_workspace_queue_calc_showing (MetaWorkspace *workspace) { GList *tmp; tmp = workspace->windows; while (tmp != NULL) { meta_window_queue_calc_showing (tmp->data); tmp = tmp->next; } } void meta_workspace_activate (MetaWorkspace *workspace) { MetaWorkspace *old; meta_verbose ("Activating workspace %d\n", meta_workspace_index (workspace)); if (workspace->screen->active_workspace == workspace) return; old = workspace->screen->active_workspace; workspace->screen->active_workspace = workspace; set_active_space_hint (workspace->screen); meta_workspace_queue_calc_showing (old); meta_workspace_queue_calc_showing (workspace); /* in mouse focus modes, this will probably get undone by an EnterNotify, * but that's OK */ meta_topic (META_DEBUG_FOCUS, "Focusing top window on new workspace\n"); meta_screen_focus_top_window (workspace->screen, NULL); } int meta_workspace_index (MetaWorkspace *workspace) { GList *tmp; int i; i = 0; tmp = workspace->screen->display->workspaces; while (tmp != NULL) { if (tmp->data == workspace) return i; ++i; tmp = tmp->next; } meta_bug ("Workspace does not exist to index!\n"); } int meta_workspace_screen_index (MetaWorkspace *workspace) { GList *tmp; int i; i = 0; tmp = workspace->screen->display->workspaces; while (tmp != NULL) { MetaWorkspace *w = tmp->data; if (tmp->data == workspace) return i; if (w->screen == workspace->screen) ++i; tmp = tmp->next; } meta_bug ("Workspace does not exist to index!\n"); } /* get windows contained on workspace, including workspace->windows * and also sticky windows. */ GList* meta_workspace_list_windows (MetaWorkspace *workspace) { GSList *display_windows; GSList *tmp; GList *workspace_windows; display_windows = meta_display_list_windows (workspace->screen->display); workspace_windows = NULL; tmp = display_windows; while (tmp != NULL) { MetaWindow *window = tmp->data; if (window->on_all_workspaces || meta_workspace_contains_window (workspace, window)) workspace_windows = g_list_prepend (workspace_windows, window); tmp = tmp->next; } g_slist_free (display_windows); return workspace_windows; } static int set_number_of_spaces_hint (MetaScreen *screen) { unsigned long data[1]; data[0] = meta_screen_get_n_workspaces (screen); meta_verbose ("Setting _NET_NUMBER_OF_DESKTOPS to %ld\n", data[0]); meta_error_trap_push (screen->display); XChangeProperty (screen->display->xdisplay, screen->xroot, screen->display->atom_net_number_of_desktops, XA_CARDINAL, 32, PropModeReplace, (guchar*) data, 1); return meta_error_trap_pop (screen->display); } static int set_active_space_hint (MetaScreen *screen) { unsigned long data[1]; data[0] = meta_workspace_screen_index (screen->active_workspace); meta_verbose ("Setting _NET_CURRENT_DESKTOP to %ld\n", data[0]); meta_error_trap_push (screen->display); XChangeProperty (screen->display->xdisplay, screen->xroot, screen->display->atom_net_current_desktop, XA_CARDINAL, 32, PropModeReplace, (guchar*) data, 1); return meta_error_trap_pop (screen->display); } static int set_work_area_hint (MetaScreen *screen) { int num_workspaces; GList *tmp_list; unsigned long *data, *tmp; MetaRectangle area; num_workspaces = meta_screen_get_n_workspaces (screen); data = g_new (unsigned long, num_workspaces * 4); tmp_list = screen->display->workspaces; tmp = data; while (tmp_list != NULL) { MetaWorkspace *workspace = tmp_list->data; if (workspace->screen == screen) { meta_workspace_get_work_area (workspace, &area); tmp[0] = area.x; tmp[1] = area.y; tmp[2] = area.width; tmp[3] = area.height; tmp += 4; } tmp_list = tmp_list->next; } meta_error_trap_push (screen->display); XChangeProperty (screen->display->xdisplay, screen->xroot, screen->display->atom_net_wm_workarea, XA_CARDINAL, 32, PropModeReplace, (guchar*) data, num_workspaces*4); g_free (data); return meta_error_trap_pop (screen->display); } static gboolean set_work_area_idle_func (void *data) { MetaScreen *screen; meta_topic (META_DEBUG_WORKAREA, "Running work area idle function\n"); screen = data; screen->work_area_idle = 0; set_work_area_hint (screen); return FALSE; } void meta_workspace_invalidate_work_area (MetaWorkspace *workspace) { GList *tmp; GList *windows; if (workspace->work_area_invalid) { meta_topic (META_DEBUG_WORKAREA, "Work area for workspace %d is already invalid\n", meta_workspace_index (workspace)); return; } meta_topic (META_DEBUG_WORKAREA, "Invalidating work area for workspace %d\n", meta_workspace_index (workspace)); workspace->work_area_invalid = TRUE; /* redo the size/position constraints on all windows */ windows = meta_workspace_list_windows (workspace); tmp = windows; while (tmp != NULL) { MetaWindow *w = tmp->data; meta_window_queue_move_resize (w); tmp = tmp->next; } g_list_free (windows); /* Recompute work area in an idle */ if (workspace->screen->work_area_idle == 0) { meta_topic (META_DEBUG_WORKAREA, "Adding work area hint idle function\n"); workspace->screen->work_area_idle = g_idle_add_full (META_PRIORITY_WORK_AREA_HINT, set_work_area_idle_func, workspace->screen, NULL); } } void meta_workspace_get_work_area (MetaWorkspace *workspace, MetaRectangle *area) { if (workspace->work_area_invalid) { int left_strut = 0; int right_strut = 0; int top_strut = 0; int bottom_strut = 0; GList *tmp; GList *windows; windows = meta_workspace_list_windows (workspace); tmp = windows; while (tmp != NULL) { MetaWindow *w = tmp->data; if (w->has_struts) { meta_topic (META_DEBUG_WORKAREA, "Merging win %s with %d %d %d %d with %d %d %d %d\n", w->desc, w->left_strut, w->right_strut, w->top_strut, w->bottom_strut, left_strut, right_strut, top_strut, bottom_strut); left_strut = MAX (left_strut, w->left_strut); right_strut = MAX (right_strut, w->right_strut); top_strut = MAX (top_strut, w->top_strut); bottom_strut = MAX (bottom_strut, w->bottom_strut); } tmp = tmp->next; } g_list_free (windows); /* Some paranoid robustness */ #define MIN_SANE_AREA 100 if ((left_strut + right_strut) > (workspace->screen->width - MIN_SANE_AREA)) { meta_topic (META_DEBUG_WORKAREA, "Making left/right struts %d %d sane\n", left_strut, right_strut); left_strut = (workspace->screen->width - MIN_SANE_AREA) / 2; right_strut = left_strut; } if ((top_strut + bottom_strut) > (workspace->screen->height - MIN_SANE_AREA)) { meta_topic (META_DEBUG_WORKAREA, "Making top/bottom struts %d %d sane\n", top_strut, bottom_strut); top_strut = (workspace->screen->height - MIN_SANE_AREA) / 2; bottom_strut = top_strut; } workspace->work_area.x = left_strut; workspace->work_area.y = top_strut; workspace->work_area.width = workspace->screen->width - left_strut - right_strut; workspace->work_area.height = workspace->screen->height - top_strut - bottom_strut; workspace->work_area_invalid = FALSE; meta_topic (META_DEBUG_WORKAREA, "Computed work area for workspace %d: %d,%d %d x %d\n", meta_workspace_index (workspace), workspace->work_area.x, workspace->work_area.y, workspace->work_area.width, workspace->work_area.height); } *area = workspace->work_area; } MetaWorkspace* meta_workspace_get_neighbor (MetaWorkspace *workspace, MetaMotionDirection direction) { int i, num_workspaces, grid_area; int rows, cols; i = meta_workspace_index (workspace); num_workspaces = meta_screen_get_n_workspaces (workspace->screen); /* * 3 rows, 4 columns, horizontal layout: * +--+--+--+--+ * | 1| 2| 3| 4| * +--+--+--+--+ * | 5| 6| 7| 8| * +--+--+--+--+ * | 9|10|11|12| * +--+--+--+--+ * * vertical layout: * +--+--+--+--+ * | 1| 4| 7|10| * +--+--+--+--+ * | 2| 5| 8|11| * +--+--+--+--+ * | 3| 6| 9|12| * +--+--+--+--+ * */ rows = workspace->screen->rows_of_workspaces; cols = workspace->screen->columns_of_workspaces; if (rows <= 0 && cols <= 0) cols = num_workspaces; if (rows <= 0) rows = num_workspaces / cols + ((num_workspaces % cols) > 0 ? 1 : 0); if (cols <= 0) cols = num_workspaces / rows + ((num_workspaces % rows) > 0 ? 1 : 0); /* paranoia */ if (rows < 1) rows = 1; if (cols < 1) cols = 1; grid_area = rows * cols; meta_verbose ("Getting neighbor rows = %d cols = %d vert = %d " "current = %d num_spaces = %d neighbor = %d\n", rows, cols, workspace->screen->vertical_workspaces, i, num_workspaces, direction); if (workspace->screen->vertical_workspaces) { switch (direction) { case META_MOTION_LEFT: if (i < rows) { i = grid_area - (i % rows) - 1; while (i >= num_workspaces) i -= rows; } else i -= rows; break; case META_MOTION_RIGHT: if ((i + rows) >= num_workspaces) i = i + rows - num_workspaces; else i += rows; break; case META_MOTION_UP: --i; if (i == -1) i = num_workspaces - 1; break; case META_MOTION_DOWN: ++i; if (i == num_workspaces) i = 0; break; } } else { switch (direction) { case META_MOTION_LEFT: --i; if (i == -1) i = num_workspaces - 1; break; case META_MOTION_RIGHT: ++i; if (i == num_workspaces) i = 0; break; case META_MOTION_UP: if (i < cols) { i = grid_area - (i % cols) - 1; while (i >= num_workspaces) i -= cols; } else i -= cols; break; case META_MOTION_DOWN: if ((i + cols) >= num_workspaces) i = i + cols - num_workspaces; else i += cols; break; } } meta_verbose ("Neighbor space is %d\n", i); return meta_display_get_workspace_by_index (workspace->screen->display, i); }