/******************************************************************** KWin - the KDE window manager This file is part of the KDE project. Copyright (C) 2007 Lubos Lunak Copyright (C) 2007 Christian Nitschkowski 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 . *********************************************************************/ #include "magnifier.h" #include #include #include #include #include #ifdef KWIN_HAVE_OPENGL_COMPOSITING #include #endif namespace KWin { KWIN_EFFECT( magnifier, MagnifierEffect ) const int FRAME_WIDTH = 5; MagnifierEffect::MagnifierEffect() : zoom( 1 ) , target_zoom( 1 ) , polling( false ) { KActionCollection* actionCollection = new KActionCollection( this ); KAction* a; a = static_cast< KAction* >( actionCollection->addAction( KStandardAction::ZoomIn, this, SLOT( zoomIn()))); a->setGlobalShortcut(KShortcut(Qt::META + Qt::Key_Equal)); a = static_cast< KAction* >( actionCollection->addAction( KStandardAction::ZoomOut, this, SLOT( zoomOut()))); a->setGlobalShortcut(KShortcut(Qt::META + Qt::Key_Minus)); a = static_cast< KAction* >( actionCollection->addAction( KStandardAction::ActualSize, this, SLOT( toggle()))); a->setGlobalShortcut(KShortcut(Qt::META + Qt::Key_0)); reconfigure( ReconfigureAll ); } void MagnifierEffect::reconfigure( ReconfigureFlags ) { KConfigGroup conf = EffectsHandler::effectConfig("Magnifier"); int width, height; width = conf.readEntry("Width", 200); height = conf.readEntry("Height", 200); magnifier_size = QSize( width, height ); } void MagnifierEffect::prePaintScreen( ScreenPrePaintData& data, int time ) { if( zoom != target_zoom ) { double diff = time / animationTime( 500.0 ); if( target_zoom > zoom ) zoom = qMin( zoom * qMax( 1 + diff, 1.2 ), target_zoom ); else zoom = qMax( zoom * qMin( 1 - diff, 0.8 ), target_zoom ); } effects->prePaintScreen( data, time ); if( zoom != 1.0 ) data.paint |= magnifierArea().adjusted( -FRAME_WIDTH, -FRAME_WIDTH, FRAME_WIDTH, FRAME_WIDTH ); } void MagnifierEffect::paintScreen( int mask, QRegion region, ScreenPaintData& data ) { ScreenPaintData data2 = data; effects->paintScreen( mask, region, data ); // paint normal screen if( zoom != 1.0 ) { // paint magnifier QRect area = magnifierArea(); PaintClipper::push( area ); // don't allow any painting outside of the area mask |= PAINT_SCREEN_TRANSFORMED; data2.xScale *= zoom; data2.yScale *= zoom; QPoint cursor = cursorPos(); // set the position so that the cursor is in the same position in the scaled view data2.xTranslate = - int( cursor.x() * ( zoom - 1 )); data2.yTranslate = - int( cursor.y() * ( zoom - 1 )); effects->paintScreen( mask, region, data2 ); PaintClipper::pop( area ); // ## TODO this should be inside KWIN_HAVE_OPENGL_COMPOSITING glPushAttrib( GL_CURRENT_BIT ); glColor4f( 0, 0, 0, 1 ); // black for( PaintClipper::Iterator iterator; !iterator.isDone(); iterator.next()) { glBegin( GL_QUADS ); glVertex2i( area.left() - FRAME_WIDTH, area.top() - FRAME_WIDTH ); // top frame glVertex2i( area.right() + FRAME_WIDTH, area.top() - FRAME_WIDTH ); glVertex2i( area.right() + FRAME_WIDTH, area.top() - 1 ); glVertex2i( area.left() - FRAME_WIDTH, area.top() - 1 ); glVertex2i( area.left() - FRAME_WIDTH, area.top() - FRAME_WIDTH ); // left frame glVertex2i( area.left() - 1, area.top() - FRAME_WIDTH ); glVertex2i( area.left() - 1, area.bottom() + FRAME_WIDTH ); glVertex2i( area.left() - FRAME_WIDTH, area.bottom() + FRAME_WIDTH ); glVertex2i( area.right() + 1, area.top() - FRAME_WIDTH ); // right frame glVertex2i( area.right() + FRAME_WIDTH, area.top() - FRAME_WIDTH ); glVertex2i( area.right() + FRAME_WIDTH, area.bottom() + FRAME_WIDTH ); glVertex2i( area.right() + 1, area.bottom() + FRAME_WIDTH ); glVertex2i( area.left() - FRAME_WIDTH, area.bottom() + 1 ); // bottom frame glVertex2i( area.right() + FRAME_WIDTH, area.bottom() + 1 ); glVertex2i( area.right() + FRAME_WIDTH, area.bottom() + FRAME_WIDTH ); glVertex2i( area.left() - FRAME_WIDTH, area.bottom() + FRAME_WIDTH ); glEnd(); } glPopAttrib(); } } void MagnifierEffect::postPaintScreen() { if( zoom != target_zoom ) { QRect framedarea = magnifierArea().adjusted( -FRAME_WIDTH, -FRAME_WIDTH, FRAME_WIDTH, FRAME_WIDTH ); effects->addRepaint( framedarea ); } effects->postPaintScreen(); } QRect MagnifierEffect::magnifierArea( QPoint pos ) const { return QRect( pos.x() - magnifier_size.width() / 2, pos.y() - magnifier_size.height() / 2, magnifier_size.width(), magnifier_size.height()); } void MagnifierEffect::zoomIn() { target_zoom *= 1.2; if( !polling ) { polling = true; effects->startMousePolling(); } effects->addRepaint( magnifierArea().adjusted( -FRAME_WIDTH, -FRAME_WIDTH, FRAME_WIDTH, FRAME_WIDTH )); } void MagnifierEffect::zoomOut() { target_zoom /= 1.2; if( target_zoom < 1 ) target_zoom = 1; if( polling ) { polling = false; effects->stopMousePolling(); } effects->addRepaint( magnifierArea().adjusted( -FRAME_WIDTH, -FRAME_WIDTH, FRAME_WIDTH, FRAME_WIDTH )); } void MagnifierEffect::toggle() { if( target_zoom == 1.0 ) target_zoom = 2; else target_zoom = 1; effects->addRepaint( magnifierArea().adjusted( -FRAME_WIDTH, -FRAME_WIDTH, FRAME_WIDTH, FRAME_WIDTH )); } void MagnifierEffect::mouseChanged( const QPoint& pos, const QPoint& old, Qt::MouseButtons, Qt::MouseButtons, Qt::KeyboardModifiers, Qt::KeyboardModifiers ) { if( pos != old && zoom != 1 ) effects->addRepaint( magnifierArea( old ).adjusted( -FRAME_WIDTH, -FRAME_WIDTH, FRAME_WIDTH, FRAME_WIDTH )); } } // namespace #include "magnifier.moc"