mutter/src/workspace.c

272 lines
6.6 KiB
C
Raw Normal View History

2001-06-06 00:47:37 -04:00
/* 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"
2001-06-09 01:14:43 -04:00
#include "errors.h"
#include <X11/Xatom.h>
2001-06-06 00:47:37 -04:00
void meta_workspace_queue_calc_showing (MetaWorkspace *workspace);
2001-06-09 01:14:43 -04:00
static int set_number_of_spaces_hint (MetaScreen *screen);
2001-06-09 23:17:15 -04:00
static int set_active_space_hint (MetaScreen *screen);
2001-06-09 01:14:43 -04:00
2001-06-06 00:47:37 -04:00
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;
2001-06-07 01:18:10 -04:00
/* This may have something to do with the strut hints
* eventually
*/
workspace->workarea.x = 0;
workspace->workarea.y = 0;
2001-07-04 00:33:31 -04:00
workspace->workarea.width = screen->width;
workspace->workarea.height = screen->height;
2001-06-09 01:14:43 -04:00
/* Update hint for current number of workspaces */
set_number_of_spaces_hint (screen);
2001-06-06 00:47:37 -04:00
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"
*/
2001-06-06 00:47:37 -04:00
tmp = workspace->windows;
while (tmp != NULL)
{
GList *next;
MetaWindow *window = tmp->data;
2001-06-06 00:47:37 -04:00
next = tmp->next;
/* pop front of list we're iterating over */
meta_workspace_remove_window (workspace, window);
g_assert (window->workspaces != NULL);
2001-06-06 00:47:37 -04:00
tmp = next;
}
g_assert (workspace->windows == NULL);
screen = workspace->screen;
2001-06-06 00:47:37 -04:00
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);
2001-06-06 00:47:37 -04:00
}
void
meta_workspace_add_window (MetaWorkspace *workspace,
MetaWindow *window)
{
2001-06-09 23:17:15 -04:00
g_return_if_fail (!meta_workspace_contains_window (workspace, window));
2001-06-06 00:47:37 -04:00
workspace->windows = g_list_prepend (workspace->windows, window);
window->workspaces = g_list_prepend (window->workspaces, workspace);
2001-06-09 23:17:15 -04:00
meta_window_set_current_workspace_hint (window);
2001-06-09 01:14:43 -04:00
2001-06-06 00:47:37 -04:00
meta_window_queue_calc_showing (window);
}
void
meta_workspace_remove_window (MetaWorkspace *workspace,
MetaWindow *window)
{
2001-06-09 23:17:15 -04:00
g_return_if_fail (meta_workspace_contains_window (workspace, window));
2001-06-06 00:47:37 -04:00
workspace->windows = g_list_remove (workspace->windows, window);
window->workspaces = g_list_remove (window->workspaces, workspace);
2001-06-09 23:17:15 -04:00
meta_window_set_current_workspace_hint (window);
2001-06-09 01:14:43 -04:00
2001-06-06 00:47:37 -04:00
meta_window_queue_calc_showing (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);
}
2001-06-09 23:17:15 -04:00
gboolean
meta_workspace_contains_window (MetaWorkspace *workspace,
MetaWindow *window)
{
return g_list_find (workspace->windows, window) != NULL;
}
2001-06-06 00:47:37 -04:00
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;
2001-06-09 23:17:15 -04:00
set_active_space_hint (workspace->screen);
2001-06-06 00:47:37 -04:00
meta_workspace_queue_calc_showing (old);
meta_workspace_queue_calc_showing (workspace);
}
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");
}
2001-06-09 01:14:43 -04:00
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");
}
static int
2001-06-09 23:17:15 -04:00
set_number_of_spaces_hint (MetaScreen *screen)
2001-06-09 01:14:43 -04:00
{
unsigned long data[1];
2001-06-09 23:17:15 -04:00
data[0] = meta_screen_get_n_workspaces (screen);
2001-06-09 01:14:43 -04:00
2001-06-09 23:17:15 -04:00
meta_verbose ("Setting _NET_NUMBER_OF_DESKTOPS to %ld\n", data[0]);
2001-06-09 01:14:43 -04:00
2001-06-09 23:17:15 -04:00
meta_error_trap_push (screen->display);
XChangeProperty (screen->display->xdisplay, screen->xroot,
screen->display->atom_net_number_of_desktops,
2001-06-09 01:14:43 -04:00
XA_CARDINAL,
32, PropModeReplace, (guchar*) data, 1);
2001-06-09 23:17:15 -04:00
return meta_error_trap_pop (screen->display);
2001-06-09 01:14:43 -04:00
}
static int
2001-06-09 23:17:15 -04:00
set_active_space_hint (MetaScreen *screen)
2001-06-09 01:14:43 -04:00
{
unsigned long data[1];
2001-06-09 23:17:15 -04:00
data[0] = meta_workspace_screen_index (screen->active_workspace);
2001-06-09 01:14:43 -04:00
2001-06-09 23:17:15 -04:00
meta_verbose ("Setting _NET_CURRENT_DESKTOP to %ld\n", data[0]);
2001-06-09 01:14:43 -04:00
meta_error_trap_push (screen->display);
XChangeProperty (screen->display->xdisplay, screen->xroot,
2001-06-09 23:17:15 -04:00
screen->display->atom_net_current_desktop,
2001-06-09 01:14:43 -04:00
XA_CARDINAL,
32, PropModeReplace, (guchar*) data, 1);
return meta_error_trap_pop (screen->display);
}