2007-11-27 19:40:25 +00:00
|
|
|
/********************************************************************
|
2007-04-29 17:35:43 +00:00
|
|
|
KWin - the KDE window manager
|
|
|
|
This file is part of the KDE project.
|
|
|
|
|
2007-11-27 19:40:25 +00:00
|
|
|
Copyright (C) 2007 Lubos Lunak <l.lunak@kde.org>
|
2007-11-13 16:20:52 +00:00
|
|
|
Copyright (C) 2007 Christian Nitschkowski <christian.nitschkowski@kdemail.net>
|
2011-08-20 08:53:36 +00:00
|
|
|
Copyright (C) 2011 Martin Gräßlin <mgraesslin@kde.org>
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2007-11-27 19:40:25 +00:00
|
|
|
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/>.
|
|
|
|
*********************************************************************/
|
2007-04-29 17:35:43 +00:00
|
|
|
|
|
|
|
#include "magnifier.h"
|
|
|
|
|
2008-05-30 17:47:24 +00:00
|
|
|
#include <kwinconfig.h>
|
|
|
|
|
2007-04-29 17:35:43 +00:00
|
|
|
#include <kaction.h>
|
|
|
|
#include <kactioncollection.h>
|
2007-11-13 16:20:52 +00:00
|
|
|
#include <kconfiggroup.h>
|
2007-04-29 17:35:43 +00:00
|
|
|
#include <kstandardaction.h>
|
|
|
|
|
2011-01-06 13:29:13 +00:00
|
|
|
#include <kwinglutils.h>
|
2012-05-26 07:05:05 +00:00
|
|
|
#ifdef KWIN_HAVE_XRENDER_COMPOSITING
|
2012-03-09 01:20:01 +00:00
|
|
|
#include <kwinxrenderutils.h>
|
2012-05-26 07:05:05 +00:00
|
|
|
#endif
|
2007-04-29 17:35:43 +00:00
|
|
|
|
|
|
|
namespace KWin
|
|
|
|
{
|
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
KWIN_EFFECT(magnifier, MagnifierEffect)
|
2011-06-26 16:51:08 +00:00
|
|
|
KWIN_EFFECT_SUPPORTED(magnifier, MagnifierEffect::supported())
|
2007-04-29 17:35:43 +00:00
|
|
|
|
|
|
|
const int FRAME_WIDTH = 5;
|
|
|
|
|
|
|
|
MagnifierEffect::MagnifierEffect()
|
2011-01-30 14:34:42 +00:00
|
|
|
: zoom(1)
|
|
|
|
, target_zoom(1)
|
|
|
|
, polling(false)
|
2011-08-20 08:53:36 +00:00
|
|
|
, m_texture(0)
|
|
|
|
, m_fbo(0)
|
2012-03-09 01:20:01 +00:00
|
|
|
, m_pixmap(0)
|
2011-01-30 14:34:42 +00:00
|
|
|
{
|
|
|
|
KActionCollection* actionCollection = new KActionCollection(this);
|
2007-04-29 17:35:43 +00:00
|
|
|
KAction* a;
|
2011-01-30 14:34:42 +00:00
|
|
|
a = static_cast< KAction* >(actionCollection->addAction(KStandardAction::ZoomIn, this, SLOT(zoomIn())));
|
2007-04-29 17:35:43 +00:00
|
|
|
a->setGlobalShortcut(KShortcut(Qt::META + Qt::Key_Equal));
|
2011-01-30 14:34:42 +00:00
|
|
|
a = static_cast< KAction* >(actionCollection->addAction(KStandardAction::ZoomOut, this, SLOT(zoomOut())));
|
2007-04-29 17:35:43 +00:00
|
|
|
a->setGlobalShortcut(KShortcut(Qt::META + Qt::Key_Minus));
|
2011-01-30 14:34:42 +00:00
|
|
|
a = static_cast< KAction* >(actionCollection->addAction(KStandardAction::ActualSize, this, SLOT(toggle())));
|
2007-04-29 17:35:43 +00:00
|
|
|
a->setGlobalShortcut(KShortcut(Qt::META + Qt::Key_0));
|
2011-03-12 13:37:30 +00:00
|
|
|
connect(effects, SIGNAL(mouseChanged(QPoint,QPoint,Qt::MouseButtons,Qt::MouseButtons,Qt::KeyboardModifiers,Qt::KeyboardModifiers)),
|
|
|
|
this, SLOT(slotMouseChanged(QPoint,QPoint,Qt::MouseButtons,Qt::MouseButtons,Qt::KeyboardModifiers,Qt::KeyboardModifiers)));
|
2011-01-30 14:34:42 +00:00
|
|
|
reconfigure(ReconfigureAll);
|
|
|
|
}
|
2007-11-13 16:20:52 +00:00
|
|
|
|
2011-08-20 08:53:36 +00:00
|
|
|
MagnifierEffect::~MagnifierEffect()
|
|
|
|
{
|
|
|
|
delete m_fbo;
|
|
|
|
delete m_texture;
|
2012-03-09 01:20:01 +00:00
|
|
|
delete m_pixmap;
|
2012-04-13 10:05:23 +00:00
|
|
|
// Save the zoom value.
|
|
|
|
KConfigGroup conf = EffectsHandler::effectConfig("Magnifier");
|
|
|
|
conf.writeEntry("InitialZoom", target_zoom);
|
|
|
|
conf.sync();
|
2011-08-20 08:53:36 +00:00
|
|
|
}
|
|
|
|
|
2011-06-26 16:51:08 +00:00
|
|
|
bool MagnifierEffect::supported()
|
|
|
|
{
|
2012-03-09 01:20:01 +00:00
|
|
|
return effects->compositingType() == XRenderCompositing ||
|
|
|
|
(effects->compositingType() == OpenGLCompositing && GLRenderTarget::blitSupported());
|
2011-06-26 16:51:08 +00:00
|
|
|
}
|
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
void MagnifierEffect::reconfigure(ReconfigureFlags)
|
|
|
|
{
|
2008-10-02 09:27:32 +00:00
|
|
|
KConfigGroup conf = EffectsHandler::effectConfig("Magnifier");
|
2007-11-13 16:20:52 +00:00
|
|
|
int width, height;
|
|
|
|
width = conf.readEntry("Width", 200);
|
|
|
|
height = conf.readEntry("Height", 200);
|
2011-01-30 14:34:42 +00:00
|
|
|
magnifier_size = QSize(width, height);
|
2012-04-13 10:05:23 +00:00
|
|
|
// Load the saved zoom value.
|
|
|
|
target_zoom = conf.readEntry("InitialZoom", target_zoom);
|
|
|
|
if (target_zoom != zoom)
|
|
|
|
toggle();
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
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);
|
2011-09-07 07:31:31 +00:00
|
|
|
else {
|
2011-01-30 14:34:42 +00:00
|
|
|
zoom = qMax(zoom * qMin(1 - diff, 0.8), target_zoom);
|
2011-09-07 07:31:31 +00:00
|
|
|
if (zoom == 1.0) {
|
|
|
|
// zoom ended - delete FBO and texture
|
|
|
|
delete m_fbo;
|
|
|
|
delete m_texture;
|
2012-03-09 01:20:01 +00:00
|
|
|
delete m_pixmap;
|
2011-09-07 07:31:31 +00:00
|
|
|
m_fbo = NULL;
|
|
|
|
m_texture = NULL;
|
2012-03-09 01:20:01 +00:00
|
|
|
m_pixmap = NULL;
|
2011-09-07 07:31:31 +00:00
|
|
|
}
|
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
}
|
2011-01-30 14:34:42 +00:00
|
|
|
effects->prePaintScreen(data, time);
|
|
|
|
if (zoom != 1.0)
|
|
|
|
data.paint |= magnifierArea().adjusted(-FRAME_WIDTH, -FRAME_WIDTH, FRAME_WIDTH, FRAME_WIDTH);
|
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
void MagnifierEffect::paintScreen(int mask, QRegion region, ScreenPaintData& data)
|
|
|
|
{
|
|
|
|
effects->paintScreen(mask, region, data); // paint normal screen
|
|
|
|
if (zoom != 1.0) {
|
2011-08-20 08:53:36 +00:00
|
|
|
// get the right area from the current rendered screen
|
|
|
|
const QRect area = magnifierArea();
|
|
|
|
const QPoint cursor = cursorPos();
|
2012-03-09 01:20:01 +00:00
|
|
|
|
|
|
|
QRect srcArea(cursor.x() - (double)area.width() / (zoom*2),
|
|
|
|
cursor.y() - (double)area.height() / (zoom*2),
|
|
|
|
(double)area.width() / zoom, (double)area.height() / zoom);
|
|
|
|
if (effects->compositingType() == OpenGLCompositing) {
|
|
|
|
m_fbo->blitFromFramebuffer(srcArea);
|
|
|
|
// paint magnifier
|
|
|
|
m_texture->bind();
|
|
|
|
m_texture->render(infiniteRegion(), area);
|
|
|
|
m_texture->unbind();
|
|
|
|
QVector<float> verts;
|
|
|
|
GLVertexBuffer *vbo = GLVertexBuffer::streamingBuffer();
|
|
|
|
vbo->reset();
|
|
|
|
vbo->setColor(QColor(0, 0, 0));
|
|
|
|
// top frame
|
|
|
|
verts << area.right() + FRAME_WIDTH << area.top() - FRAME_WIDTH;
|
|
|
|
verts << area.left() - FRAME_WIDTH << area.top() - FRAME_WIDTH;
|
|
|
|
verts << area.left() - FRAME_WIDTH << area.top() - 1;
|
|
|
|
verts << area.left() - FRAME_WIDTH << area.top() - 1;
|
|
|
|
verts << area.right() + FRAME_WIDTH << area.top() - 1;
|
|
|
|
verts << area.right() + FRAME_WIDTH << area.top() - FRAME_WIDTH;
|
|
|
|
// left frame
|
|
|
|
verts << area.left() - 1 << area.top() - FRAME_WIDTH;
|
|
|
|
verts << area.left() - FRAME_WIDTH << area.top() - FRAME_WIDTH;
|
|
|
|
verts << area.left() - FRAME_WIDTH << area.bottom() + FRAME_WIDTH;
|
|
|
|
verts << area.left() - FRAME_WIDTH << area.bottom() + FRAME_WIDTH;
|
|
|
|
verts << area.left() - 1 << area.bottom() + FRAME_WIDTH;
|
|
|
|
verts << area.left() - 1 << area.top() - FRAME_WIDTH;
|
|
|
|
// right frame
|
|
|
|
verts << area.right() + FRAME_WIDTH << area.top() - FRAME_WIDTH;
|
|
|
|
verts << area.right() + 1 << area.top() - FRAME_WIDTH;
|
|
|
|
verts << area.right() + 1 << area.bottom() + FRAME_WIDTH;
|
|
|
|
verts << area.right() + 1 << area.bottom() + FRAME_WIDTH;
|
|
|
|
verts << area.right() + FRAME_WIDTH << area.bottom() + FRAME_WIDTH;
|
|
|
|
verts << area.right() + FRAME_WIDTH << area.top() - FRAME_WIDTH;
|
|
|
|
// bottom frame
|
|
|
|
verts << area.right() + FRAME_WIDTH << area.bottom() + 1;
|
|
|
|
verts << area.left() - FRAME_WIDTH << area.bottom() + 1;
|
|
|
|
verts << area.left() - FRAME_WIDTH << area.bottom() + FRAME_WIDTH;
|
|
|
|
verts << area.left() - FRAME_WIDTH << area.bottom() + FRAME_WIDTH;
|
|
|
|
verts << area.right() + FRAME_WIDTH << area.bottom() + FRAME_WIDTH;
|
|
|
|
verts << area.right() + FRAME_WIDTH << area.bottom() + 1;
|
|
|
|
vbo->setData(verts.size() / 2, 2, verts.constData(), NULL);
|
|
|
|
if (ShaderManager::instance()->isValid()) {
|
|
|
|
ShaderManager::instance()->pushShader(ShaderManager::ColorShader);
|
|
|
|
}
|
|
|
|
vbo->render(GL_TRIANGLES);
|
|
|
|
if (ShaderManager::instance()->isValid()) {
|
|
|
|
ShaderManager::instance()->popShader();
|
|
|
|
}
|
2011-01-06 13:29:13 +00:00
|
|
|
}
|
2012-03-09 01:20:01 +00:00
|
|
|
if (effects->compositingType() == XRenderCompositing) {
|
2012-05-26 07:05:05 +00:00
|
|
|
#ifdef KWIN_HAVE_XRENDER_COMPOSITING
|
2012-03-09 01:20:01 +00:00
|
|
|
if (!m_pixmap || m_pixmap->size() != srcArea.size()) {
|
|
|
|
delete m_pixmap;
|
|
|
|
m_pixmap = new QPixmap(srcArea.size());
|
|
|
|
}
|
|
|
|
static XTransform identity = {{
|
|
|
|
{ XDoubleToFixed(1), XDoubleToFixed(0), XDoubleToFixed(0) },
|
|
|
|
{ XDoubleToFixed(0), XDoubleToFixed(1), XDoubleToFixed(0) },
|
|
|
|
{ XDoubleToFixed(0), XDoubleToFixed(0), XDoubleToFixed(1) }
|
|
|
|
}
|
|
|
|
};
|
|
|
|
static XTransform xform = {{
|
|
|
|
{ XDoubleToFixed(1), XDoubleToFixed(0), XDoubleToFixed(0) },
|
|
|
|
{ XDoubleToFixed(0), XDoubleToFixed(1), XDoubleToFixed(0) },
|
|
|
|
{ XDoubleToFixed(0), XDoubleToFixed(0), XDoubleToFixed(1) }
|
|
|
|
}
|
|
|
|
};
|
|
|
|
XRenderComposite( display(), PictOpSrc, effects->xrenderBufferPicture(), 0, m_pixmap->x11PictureHandle(),
|
|
|
|
srcArea.x(), srcArea.y(), 0, 0, 0, 0, srcArea.width(), srcArea.height() );
|
|
|
|
XFlush(display());
|
|
|
|
xform.matrix[0][0] = XDoubleToFixed(1.0/zoom);
|
|
|
|
xform.matrix[1][1] = XDoubleToFixed(1.0/zoom);
|
|
|
|
XRenderSetPictureTransform(display(), m_pixmap->x11PictureHandle(), &xform);
|
|
|
|
XRenderSetPictureFilter(display(), m_pixmap->x11PictureHandle(), const_cast<char*>("good"), NULL, 0);
|
|
|
|
XRenderComposite( display(), PictOpSrc, m_pixmap->x11PictureHandle(), 0, effects->xrenderBufferPicture(),
|
|
|
|
0, 0, 0, 0, area.x(), area.y(), area.width(), area.height() );
|
|
|
|
XRenderSetPictureFilter(display(), m_pixmap->x11PictureHandle(), const_cast<char*>("fast"), NULL, 0);
|
|
|
|
XRenderSetPictureTransform(display(), m_pixmap->x11PictureHandle(), &identity);
|
|
|
|
const XRectangle rects[4] = { { area.x()+FRAME_WIDTH, area.y(), area.width()-FRAME_WIDTH, FRAME_WIDTH},
|
|
|
|
{ area.right()-FRAME_WIDTH, area.y()+FRAME_WIDTH, FRAME_WIDTH, area.height()-FRAME_WIDTH},
|
|
|
|
{ area.x(), area.bottom()-FRAME_WIDTH, area.width()-FRAME_WIDTH, FRAME_WIDTH},
|
|
|
|
{ area.x(), area.y(), FRAME_WIDTH, area.height()-FRAME_WIDTH} };
|
|
|
|
XRenderColor c = preMultiply(QColor(0,0,0,255));
|
|
|
|
XRenderFillRectangles(display(), PictOpSrc, effects->xrenderBufferPicture(), &c, rects, 4);
|
2012-05-26 07:05:05 +00:00
|
|
|
#endif
|
2011-01-06 13:29:13 +00:00
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
}
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
|
|
|
|
void MagnifierEffect::postPaintScreen()
|
2011-01-30 14:34:42 +00:00
|
|
|
{
|
|
|
|
if (zoom != target_zoom) {
|
|
|
|
QRect framedarea = magnifierArea().adjusted(-FRAME_WIDTH, -FRAME_WIDTH, FRAME_WIDTH, FRAME_WIDTH);
|
|
|
|
effects->addRepaint(framedarea);
|
2007-09-14 13:40:26 +00:00
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
effects->postPaintScreen();
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2011-01-30 14:34:42 +00:00
|
|
|
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());
|
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
|
|
|
|
void MagnifierEffect::zoomIn()
|
2011-01-30 14:34:42 +00:00
|
|
|
{
|
2007-04-29 17:35:43 +00:00
|
|
|
target_zoom *= 1.2;
|
2011-01-30 14:34:42 +00:00
|
|
|
if (!polling) {
|
2009-02-01 15:16:52 +00:00
|
|
|
polling = true;
|
|
|
|
effects->startMousePolling();
|
2007-04-29 17:35:43 +00:00
|
|
|
}
|
2011-08-20 08:53:36 +00:00
|
|
|
if (!m_texture) {
|
|
|
|
m_texture = new GLTexture(magnifier_size);
|
|
|
|
m_texture->setYInverted(false);
|
2011-07-17 15:57:30 +00:00
|
|
|
m_fbo = new GLRenderTarget(*m_texture);
|
2011-08-20 08:53:36 +00:00
|
|
|
}
|
2011-01-30 14:34:42 +00:00
|
|
|
effects->addRepaint(magnifierArea().adjusted(-FRAME_WIDTH, -FRAME_WIDTH, FRAME_WIDTH, FRAME_WIDTH));
|
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
|
|
|
|
void MagnifierEffect::zoomOut()
|
2011-01-30 14:34:42 +00:00
|
|
|
{
|
2007-04-29 17:35:43 +00:00
|
|
|
target_zoom /= 1.2;
|
2011-01-30 14:34:42 +00:00
|
|
|
if (target_zoom < 1) {
|
2009-04-27 20:20:05 +00:00
|
|
|
target_zoom = 1;
|
2011-01-30 14:34:42 +00:00
|
|
|
if (polling) {
|
2009-04-27 20:20:05 +00:00
|
|
|
polling = false;
|
|
|
|
effects->stopMousePolling();
|
2009-02-01 15:16:52 +00:00
|
|
|
}
|
2011-09-07 07:31:31 +00:00
|
|
|
if (zoom == target_zoom) {
|
|
|
|
delete m_fbo;
|
|
|
|
delete m_texture;
|
2012-03-09 01:20:01 +00:00
|
|
|
delete m_pixmap;
|
2011-09-07 07:31:31 +00:00
|
|
|
m_fbo = NULL;
|
|
|
|
m_texture = NULL;
|
2012-03-09 01:20:01 +00:00
|
|
|
m_pixmap = NULL;
|
2011-09-07 07:31:31 +00:00
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
}
|
2011-01-30 14:34:42 +00:00
|
|
|
effects->addRepaint(magnifierArea().adjusted(-FRAME_WIDTH, -FRAME_WIDTH, FRAME_WIDTH, FRAME_WIDTH));
|
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
|
|
|
|
void MagnifierEffect::toggle()
|
2011-01-30 14:34:42 +00:00
|
|
|
{
|
2012-04-13 10:05:23 +00:00
|
|
|
if (zoom == 1.0) {
|
|
|
|
if (target_zoom == 1.0) {
|
|
|
|
target_zoom = 2;
|
|
|
|
}
|
2011-01-30 14:34:42 +00:00
|
|
|
if (!polling) {
|
2009-04-27 20:20:05 +00:00
|
|
|
polling = true;
|
|
|
|
effects->startMousePolling();
|
|
|
|
}
|
2011-08-20 08:53:36 +00:00
|
|
|
if (!m_texture) {
|
|
|
|
m_texture = new GLTexture(magnifier_size);
|
|
|
|
m_texture->setYInverted(false);
|
2011-07-17 15:57:30 +00:00
|
|
|
m_fbo = new GLRenderTarget(*m_texture);
|
2011-08-20 08:53:36 +00:00
|
|
|
}
|
2011-01-30 14:34:42 +00:00
|
|
|
} else {
|
2007-04-29 17:35:43 +00:00
|
|
|
target_zoom = 1;
|
2011-01-30 14:34:42 +00:00
|
|
|
if (polling) {
|
2009-04-27 20:20:05 +00:00
|
|
|
polling = false;
|
|
|
|
effects->stopMousePolling();
|
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
}
|
2011-01-30 14:34:42 +00:00
|
|
|
effects->addRepaint(magnifierArea().adjusted(-FRAME_WIDTH, -FRAME_WIDTH, FRAME_WIDTH, FRAME_WIDTH));
|
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2011-03-12 13:37:30 +00:00
|
|
|
void MagnifierEffect::slotMouseChanged(const QPoint& pos, const QPoint& old,
|
2011-01-30 14:34:42 +00:00
|
|
|
Qt::MouseButtons, Qt::MouseButtons, Qt::KeyboardModifiers, Qt::KeyboardModifiers)
|
|
|
|
{
|
|
|
|
if (pos != old && zoom != 1)
|
2009-04-27 17:36:20 +00:00
|
|
|
// need full repaint as we might lose some change events on fast mouse movements
|
|
|
|
// see Bug 187658
|
|
|
|
effects->addRepaintFull();
|
2011-01-30 14:34:42 +00:00
|
|
|
}
|
2007-04-29 17:35:43 +00:00
|
|
|
|
2011-08-27 09:21:31 +00:00
|
|
|
bool MagnifierEffect::isActive() const
|
|
|
|
{
|
|
|
|
return zoom != 1.0 || zoom != target_zoom;
|
|
|
|
}
|
|
|
|
|
2007-04-29 17:35:43 +00:00
|
|
|
} // namespace
|
|
|
|
|
|
|
|
#include "magnifier.moc"
|