kwin/tiling.cpp

505 lines
14 KiB
C++
Raw Normal View History

/********************************************************************
KWin - the KDE window manager
This file is part of the KDE project.
Copyright (C) 2009 Nikhil Marathe <nsm.nikhil@gmail.com>
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 <http://www.gnu.org/licenses/>.
*********************************************************************/
// 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 <knotification.h>
#include <klocale.h>
#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<Tile *> 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) << 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<Tile *> 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