70acd11ecb
svn path=/trunk/KDE/kdebase/workspace/; revision=1118827
504 lines
14 KiB
C++
504 lines
14 KiB
C++
/********************************************************************
|
|
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) << static_cast<Toplevel *>(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
|
|
|