fluxbox/src/Gnome.cc

460 lines
16 KiB
C++
Raw Normal View History

2002-09-07 20:32:44 +00:00
// Gnome.cc for fluxbox
2003-04-14 12:13:36 +00:00
// Copyright (c) 2002 - 2003 Henrik Kinnunen (fluxgen at users.sourceforge.net)
2002-09-07 20:32:44 +00:00
//
// Permission is hereby granted, free of charge, to any person obtaining a
// copy of this software and associated documentation files (the "Software"),
// to deal in the Software without restriction, including without limitation
// the rights to use, copy, modify, merge, publish, distribute, sublicense,
// and/or sell copies of the Software, and to permit persons to whom the
// Software is furnished to do so, subject to the following conditions:
//
// The above copyright notice and this permission notice shall be included in
// all copies or substantial portions of the Software.
//
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
2002-09-07 20:32:44 +00:00
// THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
// FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
// DEALINGS IN THE SOFTWARE.
2004-06-28 13:33:05 +00:00
// $Id: Gnome.cc,v 1.36 2004/06/28 13:33:05 fluxgen Exp $
2002-09-07 20:32:44 +00:00
#include "Gnome.hh"
2002-09-10 12:23:03 +00:00
#include "fluxbox.hh"
2002-09-07 20:32:44 +00:00
#include "Window.hh"
#include "Screen.hh"
2003-04-15 00:17:59 +00:00
#include "WinClient.hh"
2003-04-25 11:15:01 +00:00
#include "Workspace.hh"
2004-06-07 11:46:05 +00:00
#include "FbTk/I18n.hh"
2002-09-07 20:32:44 +00:00
#include <iostream>
2002-11-26 19:49:40 +00:00
#include <new>
2002-09-07 20:32:44 +00:00
using namespace std;
Gnome::Gnome() {
2002-12-01 13:42:15 +00:00
createAtoms();
enableUpdate();
2002-09-07 20:32:44 +00:00
}
Gnome::~Gnome() {
2002-12-01 13:42:15 +00:00
// destroy gnome windows
while (!m_gnomewindows.empty()) {
XDestroyWindow(FbTk::App::instance()->display(), m_gnomewindows.back());
2002-12-01 13:42:15 +00:00
m_gnomewindows.pop_back();
}
2002-09-07 20:32:44 +00:00
}
2003-03-03 21:51:13 +00:00
void Gnome::initForScreen(BScreen &screen) {
Display *disp = FbTk::App::instance()->display();
2002-12-01 13:42:15 +00:00
// create the GNOME window
Window gnome_win = XCreateSimpleWindow(disp,
2003-05-10 22:59:32 +00:00
screen.rootWindow().window(), 0, 0, 5, 5, 0, 0, 0);
2002-12-01 13:42:15 +00:00
// supported WM check
2003-05-19 22:43:48 +00:00
screen.rootWindow().changeProperty(m_gnome_wm_supporting_wm_check,
2004-01-19 18:36:27 +00:00
XA_WINDOW, 32,
2003-05-19 22:43:48 +00:00
PropModeReplace, (unsigned char *) &gnome_win, 1);
2002-12-01 13:42:15 +00:00
XChangeProperty(disp, gnome_win,
m_gnome_wm_supporting_wm_check,
2004-01-19 18:36:27 +00:00
XA_WINDOW, 32, PropModeReplace, (unsigned char *) &gnome_win, 1);
2002-12-01 13:42:15 +00:00
2003-06-12 14:28:00 +00:00
// supported gnome atoms
2002-12-01 13:42:15 +00:00
Atom gnomeatomlist[] = {
m_gnome_wm_supporting_wm_check,
m_gnome_wm_win_workspace_names,
m_gnome_wm_win_client_list,
m_gnome_wm_win_state,
m_gnome_wm_win_hints,
m_gnome_wm_win_layer
2002-12-01 13:42:15 +00:00
};
//list atoms that we support
2003-05-19 22:43:48 +00:00
screen.rootWindow().changeProperty(m_gnome_wm_prot,
XA_ATOM, 32, PropModeReplace,
(unsigned char *)gnomeatomlist,
(sizeof gnomeatomlist)/sizeof gnomeatomlist[0]);
2002-09-07 20:32:44 +00:00
2002-12-01 13:42:15 +00:00
m_gnomewindows.push_back(gnome_win);
2002-09-07 20:32:44 +00:00
2002-12-01 13:42:15 +00:00
updateClientList(screen);
updateWorkspaceNames(screen);
updateWorkspaceCount(screen);
updateCurrentWorkspace(screen);
2002-09-07 20:32:44 +00:00
}
void Gnome::setupFrame(FluxboxWindow &win) {
// load gnome state (take queues from the main window of the frame)
Display *disp = FbTk::App::instance()->display();
2002-12-01 13:42:15 +00:00
Atom ret_type;
int fmt;
unsigned long nitems, bytes_after;
long flags, *data = 0;
2003-05-19 22:43:48 +00:00
if (win.winClient().property(m_gnome_wm_win_state, 0, 1, False, XA_CARDINAL,
&ret_type, &fmt, &nitems, &bytes_after,
(unsigned char **) &data) && data) {
2002-12-01 13:42:15 +00:00
flags = *data;
setState(&win, flags);
XFree (data);
} else {
updateState(win);
2002-12-01 13:42:15 +00:00
}
// load gnome layer atom
2003-05-15 11:17:29 +00:00
if (XGetWindowProperty(disp, win.clientWindow(),
m_gnome_wm_win_layer, 0, 1, False, XA_CARDINAL,
&ret_type, &fmt, &nitems, &bytes_after,
(unsigned char **) &data) == Success && data) {
flags = *data;
setLayer(&win, flags);
XFree (data);
} else {
updateLayer(win);
}
2003-02-16 17:57:54 +00:00
// load gnome workspace atom
2003-05-15 11:17:29 +00:00
if (XGetWindowProperty(disp, win.clientWindow(),
m_gnome_wm_win_workspace, 0, 1, False, XA_CARDINAL,
&ret_type, &fmt, &nitems, &bytes_after,
(unsigned char **) &data) == Success && data) {
2003-02-16 17:57:54 +00:00
unsigned int workspace_num = *data;
2003-05-15 11:17:29 +00:00
if (win.workspaceNumber() != workspace_num)
2003-05-11 13:36:12 +00:00
win.screen().reassociateWindow(&win, workspace_num, false);
XFree (data);
} else {
updateWorkspace(win);
}
2002-09-10 12:23:03 +00:00
2002-09-08 10:58:30 +00:00
}
2004-06-28 13:33:05 +00:00
bool Gnome::propertyNotify(WinClient &winclient, Atom the_property) {
if (the_property == m_gnome_wm_win_state) {
#ifdef DEBUG
cerr<<__FILE__<<"("<<__FUNCTION__<<"): _WIN_STATE"<<endl;
#endif // DEBUG
return true;
}
return false;
}
2003-03-03 21:51:13 +00:00
void Gnome::updateClientList(BScreen &screen) {
2002-12-01 13:42:15 +00:00
size_t num=0;
2003-04-15 12:22:52 +00:00
// count window clients in each workspace
2003-04-15 00:17:59 +00:00
BScreen::Workspaces::const_iterator workspace_it =
screen.getWorkspacesList().begin();
BScreen::Workspaces::const_iterator workspace_it_end =
screen.getWorkspacesList().end();
2002-12-01 13:42:15 +00:00
for (; workspace_it != workspace_it_end; ++workspace_it) {
2003-04-15 00:17:59 +00:00
Workspace::Windows::iterator win_it =
2003-05-11 17:14:41 +00:00
(*workspace_it)->windowList().begin();
2003-04-15 00:17:59 +00:00
Workspace::Windows::iterator win_it_end =
2003-05-11 17:14:41 +00:00
(*workspace_it)->windowList().end();
2003-04-15 00:17:59 +00:00
for (; win_it != win_it_end; ++win_it)
num += (*win_it)->numClients();
2002-12-01 13:42:15 +00:00
}
2002-09-07 20:32:44 +00:00
Window *wl = new Window[num];
2002-12-01 13:42:15 +00:00
if (wl == 0) {
2004-06-07 11:46:05 +00:00
_FB_USES_NLS;
cerr<<_FBTEXT(Gnome, OutOfMemoryClientList, "Fatal: Out of memory, can't allocate for GNOME client list", "")<<endl;
2002-12-01 13:42:15 +00:00
return;
}
2003-04-15 12:22:52 +00:00
//add client windows to buffer
2002-12-01 13:42:15 +00:00
workspace_it = screen.getWorkspacesList().begin();
int win=0;
for (; workspace_it != workspace_it_end; ++workspace_it) {
2002-09-07 20:32:44 +00:00
2002-12-01 13:42:15 +00:00
// Fill in array of window ID's
2003-04-15 00:17:59 +00:00
Workspace::Windows::const_iterator it =
2003-05-11 17:14:41 +00:00
(*workspace_it)->windowList().begin();
2003-04-15 00:17:59 +00:00
Workspace::Windows::const_iterator it_end =
2003-05-11 17:14:41 +00:00
(*workspace_it)->windowList().end();
2002-12-01 13:42:15 +00:00
for (; it != it_end; ++it) {
// TODO!
//check if the window don't want to be visible in the list
//if (! ( (*it)->getGnomeHints() & WIN_STATE_HIDDEN) ) {
2003-04-15 00:17:59 +00:00
std::list<WinClient *>::iterator client_it =
(*it)->clientList().begin();
std::list<WinClient *>::iterator client_it_end =
(*it)->clientList().end();
for (; client_it != client_it_end; ++client_it)
wl[win++] = (*client_it)->window();
2002-12-01 13:42:15 +00:00
}
}
//number of windows to show in client list
num = win;
2003-05-19 22:43:48 +00:00
screen.rootWindow().changeProperty(m_gnome_wm_win_client_list,
2004-01-19 18:36:27 +00:00
XA_WINDOW, 32,
2003-05-19 22:43:48 +00:00
PropModeReplace, (unsigned char *)wl, num);
2002-09-07 20:32:44 +00:00
delete[] wl;
2002-09-07 20:32:44 +00:00
}
2003-03-03 21:51:13 +00:00
void Gnome::updateWorkspaceNames(BScreen &screen) {
2003-06-12 14:28:00 +00:00
2002-12-01 13:42:15 +00:00
int number_of_desks = screen.getWorkspaceNames().size();
2003-06-12 14:28:00 +00:00
const BScreen::WorkspaceNames &workspace_names = screen.getWorkspaceNames();
// convert our desktop names to a char * so we can send it
2002-12-01 13:42:15 +00:00
char *names[number_of_desks];
2003-06-12 14:28:00 +00:00
2002-12-01 13:42:15 +00:00
for (int i = 0; i < number_of_desks; i++) {
2003-06-12 14:28:00 +00:00
names[i] = new char[workspace_names[i].size() + 1];
strcpy(names[i], workspace_names[i].c_str());
2002-12-01 13:42:15 +00:00
}
2003-06-12 14:28:00 +00:00
XTextProperty text;
2002-12-01 13:42:15 +00:00
if (XStringListToTextProperty(names, number_of_desks, &text)) {
2003-05-10 22:59:32 +00:00
XSetTextProperty(FbTk::App::instance()->display(), screen.rootWindow().window(),
2002-09-07 20:32:44 +00:00
&text, m_gnome_wm_win_workspace_names);
2002-12-01 13:42:15 +00:00
XFree(text.value);
}
2003-06-12 14:28:00 +00:00
// destroy name buffers
2002-12-01 13:42:15 +00:00
for (int i = 0; i < number_of_desks; i++)
delete [] names[i];
2002-09-07 20:32:44 +00:00
}
2003-03-03 21:51:13 +00:00
void Gnome::updateCurrentWorkspace(BScreen &screen) {
2003-05-15 12:00:46 +00:00
int workspace = screen.currentWorkspaceID();
2003-05-19 22:43:48 +00:00
screen.rootWindow().changeProperty(m_gnome_wm_win_workspace, XA_CARDINAL, 32, PropModeReplace,
(unsigned char *)&workspace, 1);
2002-09-07 20:32:44 +00:00
2002-12-01 13:42:15 +00:00
updateClientList(screen); // make sure the client list is updated too
2002-09-07 20:32:44 +00:00
}
2003-03-03 21:51:13 +00:00
void Gnome::updateWorkspaceCount(BScreen &screen) {
2002-12-01 13:42:15 +00:00
int numworkspaces = screen.getCount();
2003-05-19 22:43:48 +00:00
screen.rootWindow().changeProperty(m_gnome_wm_win_workspace_count, XA_CARDINAL, 32, PropModeReplace,
(unsigned char *)&numworkspaces, 1);
2002-09-07 20:32:44 +00:00
}
2002-09-08 10:58:30 +00:00
void Gnome::updateWorkspace(FluxboxWindow &win) {
int val = win.workspaceNumber();
if (win.isStuck()) {
val = -1;
}
2002-09-10 12:23:03 +00:00
#ifdef DEBUG
2002-12-01 13:42:15 +00:00
cerr<<__FILE__<<"("<<__LINE__<<"): setting workspace("<<val<<
") for window("<<&win<<")"<<endl;
2002-09-10 12:23:03 +00:00
#endif // DEBUG
FluxboxWindow::ClientList::iterator client_it = win.clientList().begin();
FluxboxWindow::ClientList::iterator client_it_end = win.clientList().end();
for (; client_it != client_it_end; ++client_it)
(*client_it)->changeProperty(m_gnome_wm_win_workspace,
XA_CARDINAL, 32, PropModeReplace,
(unsigned char *)&val, 1);
2002-09-08 10:58:30 +00:00
}
void Gnome::updateState(FluxboxWindow &win) {
2002-12-01 13:42:15 +00:00
//translate to gnome win state
int state=0;
if (win.isStuck())
state |= WIN_STATE_STICKY;
if (win.isIconic())
state |= WIN_STATE_MINIMIZED;
if (win.isShaded())
state |= WIN_STATE_SHADED;
2002-09-07 20:32:44 +00:00
FluxboxWindow::ClientList::iterator client_it = win.clientList().begin();
FluxboxWindow::ClientList::iterator client_it_end = win.clientList().end();
2004-06-28 13:33:05 +00:00
for (; client_it != client_it_end; ++client_it) {
(*client_it)->changeProperty(m_gnome_wm_win_state,
XA_CARDINAL, 32,
PropModeReplace, (unsigned char *)&state, 1);
2004-06-28 13:33:05 +00:00
}
2002-09-07 20:32:44 +00:00
}
void Gnome::updateLayer(FluxboxWindow &win) {
//TODO - map from flux layers to gnome ones
2003-05-08 02:50:38 +00:00
// our layers are in the opposite direction to GNOME
2003-05-15 11:17:29 +00:00
int layernum = Fluxbox::instance()->getDesktopLayer() - win.layerNum();
FluxboxWindow::ClientList::iterator client_it = win.clientList().begin();
FluxboxWindow::ClientList::iterator client_it_end = win.clientList().end();
for (; client_it != client_it_end; ++client_it)
(*client_it)->changeProperty(m_gnome_wm_win_layer,
XA_CARDINAL, 32, PropModeReplace,
(unsigned char *)&layernum, 1);
}
2002-09-08 10:58:30 +00:00
void Gnome::updateHints(FluxboxWindow &win) {
2002-12-01 13:42:15 +00:00
//TODO
2002-09-07 20:32:44 +00:00
}
bool Gnome::checkClientMessage(const XClientMessageEvent &ce, BScreen * screen, WinClient * const winclient) {
2002-12-01 13:42:15 +00:00
if (ce.message_type == m_gnome_wm_win_workspace) {
2002-09-07 20:32:44 +00:00
#ifdef DEBUG
2002-12-01 13:42:15 +00:00
cerr<<__FILE__<<"("<<__LINE__<<"): Got workspace atom="<<ce.data.l[0]<<endl;
2002-09-07 20:32:44 +00:00
#endif//!DEBUG
if ( winclient !=0 && // the message sent to client window?
2003-04-14 12:13:36 +00:00
ce.data.l[0] >= 0 &&
ce.data.l[0] < (signed)winclient->screen().getCount()) {
winclient->screen().changeWorkspaceID(ce.data.l[0]);
2002-09-07 20:32:44 +00:00
2002-12-01 13:42:15 +00:00
} else if (screen!=0 && //the message sent to root window?
ce.data.l[0] >= 0 &&
ce.data.l[0] < (signed)screen->getCount())
screen->changeWorkspaceID(ce.data.l[0]);
return true;
} else if (winclient == 0)
2002-12-01 13:42:15 +00:00
return false;
2002-09-07 20:32:44 +00:00
2002-12-01 13:42:15 +00:00
if (ce.message_type == m_gnome_wm_win_state) {
2002-09-07 20:32:44 +00:00
#ifdef DEBUG
2002-12-01 13:42:15 +00:00
cerr<<__FILE__<<"("<<__LINE__<<"): _WIN_STATE"<<endl;
2002-09-07 20:32:44 +00:00
#endif // DEBUG
#ifdef DEBUG
2002-12-01 13:42:15 +00:00
cerr<<__FILE__<<"("<<__LINE__<<"): Mask of members to change:"<<
hex<<ce.data.l[0]<<dec<<endl; // mask_of_members_to_change
cerr<<"New members:"<<ce.data.l[1]<<endl;
2002-09-07 20:32:44 +00:00
#endif // DEBUG
if (winclient && winclient->fbwindow()) {
//get new states
int flag = ce.data.l[0] & ce.data.l[1];
//don't update this when when we set new state
disableUpdate();
// convert to Fluxbox state
setState(winclient->fbwindow(), flag);
// enable update of atom states
enableUpdate();
}
2002-12-01 13:42:15 +00:00
} else if (ce.message_type == m_gnome_wm_win_hints) {
2002-09-07 20:32:44 +00:00
#ifdef DEBUG
2002-12-01 13:42:15 +00:00
cerr<<__FILE__<<"("<<__LINE__<<"): _WIN_HINTS"<<endl;
2002-09-07 20:32:44 +00:00
#endif // DEBUG
} else if (ce.message_type == m_gnome_wm_win_layer) {
#ifdef DEBUG
cerr<<__FILE__<<"("<<__LINE__<<"): _WIN_LAYER"<<endl;
#endif // DEBUG
if (winclient && winclient->fbwindow())
setLayer(winclient->fbwindow(), ce.data.l[0]);
} else
2002-12-01 13:42:15 +00:00
return false; //the gnome atom wasn't found or not supported
2002-09-07 20:32:44 +00:00
2002-12-01 13:42:15 +00:00
return true; // we handled the atom
2002-09-07 20:32:44 +00:00
}
void Gnome::setState(FluxboxWindow *win, int state) {
#ifdef DEBUG
2002-12-01 13:42:15 +00:00
cerr<<"Gnome: state=0x"<<hex<<state<<dec<<endl;
2002-09-07 20:32:44 +00:00
#endif // DEBUG
2002-12-01 13:42:15 +00:00
if (state & WIN_STATE_STICKY) {
2002-09-07 20:32:44 +00:00
#ifdef DEBUG
2002-12-01 13:42:15 +00:00
cerr<<"Gnome state: Sticky"<<endl;
2002-09-07 20:32:44 +00:00
#endif // DEBUG
2002-12-01 13:42:15 +00:00
if (!win->isStuck())
win->stick();
} else if (win->isStuck())
win->stick();
2002-09-07 20:32:44 +00:00
2002-12-01 13:42:15 +00:00
if (state & WIN_STATE_MINIMIZED) {
2002-09-07 20:32:44 +00:00
#ifdef DEBUG
2002-12-01 13:42:15 +00:00
cerr<<"Gnome state: Minimized"<<endl;
2002-09-07 20:32:44 +00:00
#endif // DEBUG
2002-12-01 13:42:15 +00:00
if (win->isIconic())
win->iconify();
} else if (win->isIconic())
win->deiconify(true, true);
2002-09-07 20:32:44 +00:00
2002-12-01 13:42:15 +00:00
if (state & WIN_STATE_SHADED) {
2002-09-07 20:32:44 +00:00
#ifdef DEBUG
2002-12-01 13:42:15 +00:00
cerr<<"Gnome state: Shade"<<endl;
2002-09-07 20:32:44 +00:00
#endif // DEBUG
2002-12-01 13:42:15 +00:00
if (!win->isShaded())
win->shade();
} else if (win->isShaded())
win->shade();
2004-01-19 18:36:27 +00:00
if (state & WIN_STATE_HIDDEN)
{
win->setFocusHidden(! win->isFocusHidden());
win->setIconHidden(! win->isIconHidden());
}
2004-01-19 18:36:27 +00:00
/*
if (state & WIN_STATE_MAXIMIZED_VERT)
cerr<<"Maximize Vert"<<endl;
if (state & WIN_STATE_MAXIMIZED_HORIZ)
cerr<<"Maximize Horiz"<<endl;
if (state & WIN_STATE_HID_WORKSPACE)
cerr<<"HID Workspace"<<endl;
if (state & WIN_STATE_HID_TRANSIENT)
cerr<<"HID Transient"<<endl;
if (state & WIN_STATE_FIXED_POSITION)
cerr<<"Fixed Position"<<endl;
if (state & WIN_STATE_ARRANGE_IGNORE)
cerr<<"Arrange Ignore"<<endl;
2002-12-01 13:42:15 +00:00
*/
2002-09-07 20:32:44 +00:00
}
void Gnome::setLayer(FluxboxWindow *win, int layer) {
if (!win) return;
2002-12-01 13:42:15 +00:00
switch (layer) {
case WIN_LAYER_DESKTOP:
layer = Fluxbox::instance()->getDesktopLayer();
2002-12-01 13:42:15 +00:00
break;
case WIN_LAYER_BELOW:
layer = Fluxbox::instance()->getBottomLayer();
2002-12-01 13:42:15 +00:00
break;
case WIN_LAYER_NORMAL:
layer = Fluxbox::instance()->getNormalLayer();
2002-12-01 13:42:15 +00:00
break;
case WIN_LAYER_ONTOP:
layer = Fluxbox::instance()->getTopLayer();
break;
2002-12-01 13:42:15 +00:00
case WIN_LAYER_DOCK:
layer = Fluxbox::instance()->getDockLayer();
break;
case WIN_LAYER_ABOVE_DOCK:
layer = Fluxbox::instance()->getAboveDockLayer();
break;
2002-12-01 13:42:15 +00:00
case WIN_LAYER_MENU:
layer = Fluxbox::instance()->getMenuLayer();
2002-12-01 13:42:15 +00:00
break;
default:
// our windows are in the opposite direction to gnome
layer = Fluxbox::instance()->getDesktopLayer() - layer;
2002-12-01 13:42:15 +00:00
break;
}
2003-02-09 14:11:14 +00:00
win->moveToLayer(layer);
2002-09-07 20:32:44 +00:00
}
void Gnome::createAtoms() {
Display *disp = FbTk::App::instance()->display();
2002-12-01 13:42:15 +00:00
m_gnome_wm_win_layer = XInternAtom(disp, "_WIN_LAYER", False);
m_gnome_wm_win_state = XInternAtom(disp, "_WIN_STATE", False);
m_gnome_wm_win_hints = XInternAtom(disp, "_WIN_HINTS", False);
m_gnome_wm_win_app_state = XInternAtom(disp, "_WIN_APP_STATE", False);
m_gnome_wm_win_expanded_size = XInternAtom(disp, "_WIN_EXPANDED_SIZE", False);
m_gnome_wm_win_icons = XInternAtom(disp, "_WIN_ICONS", False);
m_gnome_wm_win_workspace = XInternAtom(disp, "_WIN_WORKSPACE", False);
m_gnome_wm_win_workspace_count = XInternAtom(disp, "_WIN_WORKSPACE_COUNT", False);
m_gnome_wm_win_workspace_names = XInternAtom(disp, "_WIN_WORKSPACE_NAMES", False);
m_gnome_wm_win_client_list = XInternAtom(disp, "_WIN_CLIENT_LIST", False);
m_gnome_wm_prot = XInternAtom(disp, "_WIN_PROTOCOLS", False);
m_gnome_wm_supporting_wm_check = XInternAtom(disp, "_WIN_SUPPORTING_WM_CHECK", False);
2002-09-07 20:32:44 +00:00
}