/******************************************************************** KWin - the KDE window manager This file is part of the KDE project. Copyright (C) 2009 Nikhil Marathe 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, see . *********************************************************************/ // all the tiling related code that is extensions to existing KWin classes // Includes Workspace for now #include "client.h" #include "workspace.h" #include "tile.h" #include "tilinglayout.h" #include "tilinglayoutfactory.h" #include #include #include "lib/kdecoration.h" namespace KWin { bool Workspace::tilingMode() const { return tilingMode_; } void Workspace::setTilingMode( bool tiling ) { if( tilingMode() == tiling ) return; tilingMode_ = tiling; if( tilingMode_ ) { tilingLayouts.resize( numberOfDesktops() + 1 ); foreach( Client *c, stackingOrder() ) { createTile( c ); } } else { foreach( TilingLayout *t, tilingLayouts ) { if( t ) delete t; } tilingLayouts.clear(); } } void Workspace::slotToggleTiling() { if ( tilingMode() ) { setTilingMode( false ); QString message = i18n( "Tiling Disabled" ); KNotification::event( "tilingdisabled", message, QPixmap(), NULL, KNotification::CloseOnTimeout, KComponentData( "kwin" ) ); } else { setTilingMode( true ); QString message = i18n( "Tiling Enabled" ); KNotification::event( "tilingenabled", message, QPixmap(), NULL, KNotification::CloseOnTimeout, KComponentData( "kwin" ) ); } } void Workspace::createTile( Client *c ) { if( c == NULL ) return; if( c->desktop() < 0 || c->desktop() >= tilingLayouts.size() ) return; kDebug(1212) << "Now tiling " << c->caption(); if( !tilingMode() || !tileable(c) ) return; Tile *t = new Tile( c, clientArea( PlacementArea, c ) ); if( !tileable( c ) ) { kDebug(1212) << c->caption() << "is not tileable"; t->floatTile(); } if( !tilingLayouts.value(c->desktop()) ) { tilingLayouts[c->desktop()] = TilingLayoutFactory::createLayout( TilingLayoutFactory::DefaultL, this ); } tilingLayouts[c->desktop()]->addTile( t ); tilingLayouts[c->desktop()]->commit(); } void Workspace::removeTile( Client *c ) { if( tilingLayouts[ c->desktop() ] ) tilingLayouts[ c->desktop() ]->removeTile( c ); } bool Workspace::tileable( Client *c ) { kDebug(1212) << c->caption(); // TODO: if application specific settings // to ignore, put them here // as suggested by Lucas Murray // see Client::manage ( ~ line 270 of manage.cpp at time of writing ) // This is useful to ignore plasma widget placing. // According to the ICCCM if an application // or user requests a certain geometry // respect it long msize; XSizeHints xSizeHint; XGetWMNormalHints(display(), c->window(), &xSizeHint, &msize); if( xSizeHint.flags & PPosition || xSizeHint.flags & USPosition ) { kDebug(1212) << "Not tileable due to USPosition requirement"; return false; } kDebug(1212) << "c->isNormalWindow()" << c->isNormalWindow() ; kDebug(1212) << "c->isUtility() " << c->isUtility() ; kDebug(1212) << "c->isDialog() " << c->isDialog() ; kDebug(1212) << "c->isSplash() " << c->isSplash() ; kDebug(1212) << "c->isToolbar() " << c->isToolbar() ; kDebug(1212) << "c->isPopupMenu() " << c->isPopupMenu() ; kDebug(1212) << "c->isTransient() " << c->isTransient() ; if( !c->isNormalWindow() || c->isUtility() || c->isDialog() || c->isSplash() || c->isToolbar() || c->isPopupMenu() || c->isTransient() ) { return false; } return true; } void Workspace::belowCursor() { // TODO } Tile* Workspace::getNiceTile() const { if( !tilingMode() ) return NULL; if( !tilingLayouts.value( activeClient()->desktop() ) ) return NULL; return tilingLayouts[ activeClient()->desktop() ]->findTile( activeClient() ); // TODO } void Workspace::updateAllTiles() { foreach( TilingLayout *t, tilingLayouts ) { if( !t ) continue; t->commit(); } } /* * Resize the neighbouring clients to close any gaps */ void Workspace::notifyWindowResize( Client *c, const QRect &moveResizeGeom, const QRect &orig ) { if( tilingLayouts.value( c->desktop() ) == NULL ) return; tilingLayouts[ c->desktop() ]->clientResized( c, moveResizeGeom, orig ); } void Workspace::notifyWindowMove( Client *c, const QRect &moveResizeGeom, const QRect &orig ) { if( tilingLayouts.value( c->desktop() ) == NULL ) { c->setGeometry( moveResizeGeom ); return; } tilingLayouts[ c->desktop() ]->clientMoved( c, moveResizeGeom, orig ); updateAllTiles(); } void Workspace::notifyWindowResizeDone( Client *c, const QRect &moveResizeGeom, const QRect &orig, bool canceled ) { if( canceled ) notifyWindowResize( c, orig, moveResizeGeom ); else notifyWindowResize( c, moveResizeGeom, orig ); } void Workspace::notifyWindowMoveDone( Client *c, const QRect &moveResizeGeom, const QRect &orig, bool canceled ) { if( canceled ) notifyWindowMove( c, orig, moveResizeGeom ); else notifyWindowMove( c, moveResizeGeom, orig ); } void Workspace::notifyWindowDesktopChanged( Client *c, int old_desktop ) { if( c->desktop() < 1 || c->desktop() > numberOfDesktops() ) return; if( tilingLayouts.value( old_desktop ) ) { Tile *t = tilingLayouts[ old_desktop ]->findTile( c ); // TODO: copied from createTile(), move this into separate method? if( !tilingLayouts.value( c->desktop() ) ) { tilingLayouts[c->desktop()] = TilingLayoutFactory::createLayout( TilingLayoutFactory::DefaultL, this ); } if( t ) tilingLayouts[ c->desktop() ]->addTile( t ); tilingLayouts[ old_desktop ]->removeTile( c ); tilingLayouts[ old_desktop ]->commit(); } } // TODO: make this configurable /* * If a floating window was activated, raise all floating windows. * If a tiled window was activated, lower all floating windows. */ void Workspace::notifyWindowActivated( Client *c ) { if( c == NULL ) return; if( options->tilingRaisePolicy == 1 ) // individual raise/lowers return; if( tilingLayouts.value( c->desktop() ) ) { QList tiles = tilingLayouts[ c->desktop() ]->tiles(); StackingUpdatesBlocker blocker( this ); Tile *tile_to_raise = tilingLayouts[ c->desktop() ]->findTile( c ); if( !tile_to_raise ) { return; } kDebug(1212) << "FOUND TILE"; bool raise_floating = false; if( options->tilingRaisePolicy == 2 ) // floating always on top raise_floating = true; else raise_floating = tile_to_raise->floating(); foreach( Tile *t, tiles ) { kDebug(1212) << static_cast(t->client()) << t->floating(); if( t->floating() == raise_floating && t != tile_to_raise ) raiseClient( t->client() ); } // raise the current tile last so that it ends up on top // but only if it supposed to be raised, required to support tilingRaisePolicy kDebug(1212) << "Raise floating? " << raise_floating << "to raise is floating?" << tile_to_raise->floating(); if( tile_to_raise->floating() == raise_floating ) raiseClient( tile_to_raise->client() ); } } void Workspace::notifyWindowMinimizeToggled( Client *c ) { if( tilingLayouts.value( c->desktop() ) ) { tilingLayouts[ c->desktop() ]->clientMinimizeToggled( c ); } } void Workspace::notifyWindowMaximized( Client *c, Options::WindowOperation op ) { if( tilingLayouts.value( c->desktop() ) ) { Tile *t = tilingLayouts[ c->desktop() ]->findTile( c ); if( !t ) { createTile( c ); t = tilingLayouts[ c->desktop() ]->findTile( c ); // if still no tile, it couldn't be tiled // so ignore it if( !t ) return; } // if window IS tiled and a maximize // is attempted, make the window float. // That is all we do since that can // mess up the layout. // In all other cases, don't do // anything, let the user manage toggling // using Meta+F if ( !t->floating() && ( op == Options::MaximizeOp || op == Options::HMaximizeOp || op == Options::VMaximizeOp ) ) { tilingLayouts[ c->desktop() ]->toggleFloatTile( c ); } } } Tile* Workspace::findAdjacentTile( Tile *ref, int d ) { QRect reference = ref->geometry(); QPoint origin = reference.center(); Tile *closest = NULL; int minDist = -1; QList tiles = tilingLayouts[ ref->client()->desktop() ]->tiles(); foreach( Tile *t, tiles ) { if( t->client() == ref->client() || t->ignoreGeometry() ) continue; bool consider = false; QRect other = t->geometry(); QPoint otherCenter = other.center(); switch( d ) { case Tile::Top: consider = otherCenter.y() < origin.y() && other.bottom() < reference.top(); break; case Tile::Right: consider = otherCenter.x() > origin.x() && other.left() > reference.right(); break; case Tile::Bottom: consider = otherCenter.y() > origin.y() && other.top() > reference.bottom(); break; case Tile::Left: consider = otherCenter.x() < origin.x() && other.right() < reference.left(); break; default: abort(); } if( consider ) { int dist = ( otherCenter - origin ).manhattanLength(); if( minDist > dist || minDist < 0 ) { minDist = dist; closest = t; } } } return closest; } void Workspace::focusTile( int d ) { Tile *t = getNiceTile(); if( t ) { Tile *adj = findAdjacentTile(t, d); if( adj ) activateClient( adj->client() ); } } void Workspace::moveTile( int d ) { Tile *t = getNiceTile(); if( t ) { Tile* adj = findAdjacentTile( t, d ); tilingLayouts[ t->client()->desktop() ]->swapTiles( t, adj ); } } void Workspace::slotLeft() { focusTile( Tile::Left ); } void Workspace::slotRight() { focusTile( Tile::Right ); } void Workspace::slotTop() { focusTile( Tile::Top ); } void Workspace::slotBottom() { focusTile( Tile::Bottom ); } void Workspace::slotMoveLeft() { moveTile( Tile::Left ); } void Workspace::slotMoveRight() { moveTile( Tile::Right ); } void Workspace::slotMoveTop() { moveTile( Tile::Top ); } void Workspace::slotMoveBottom() { moveTile( Tile::Bottom ); } void Workspace::slotToggleFloating() { Client *c = activeClient(); if( tilingLayouts.value( c->desktop() ) ) { tilingLayouts[ c->desktop() ]->toggleFloatTile( c ); } } void Workspace::slotNextTileLayout() { if( tilingLayouts.value( currentDesktop() ) ) { tilingLayouts.replace( currentDesktop(), TilingLayoutFactory::nextLayout( tilingLayouts[currentDesktop()] ) ); tilingLayouts[currentDesktop()]->commit(); } } void Workspace::slotPreviousTileLayout() { if( tilingLayouts.value( currentDesktop() ) ) { tilingLayouts.replace( currentDesktop(), TilingLayoutFactory::previousLayout( tilingLayouts[currentDesktop()] ) ); tilingLayouts[currentDesktop()]->commit(); } } KDecorationDefines::Position Workspace::supportedTilingResizeMode( Client *c, KDecorationDefines::Position currentMode ) { if( tilingLayouts.value( c->desktop() ) ) { return tilingLayouts[c->desktop()]->resizeMode( c, currentMode ); } return currentMode; } void Workspace::dumpTiles() const { foreach( TilingLayout *t, tilingLayouts ) { if( !t ) { kDebug(1212) << "EMPTY DESKTOP"; continue; } kDebug(1212) << "Desktop" << tilingLayouts.indexOf( t ); foreach( Tile *tile, t->tiles() ) { tile->dumpTile( "--" ); } } } } // namespace