kwin/effects/taskbarthumbnail/taskbarthumbnail.cpp
Fredrik Höglund cbb4248a2a Use a high quality lanczos filter to scale the window thumbnails when
GLSL shaders are supported.

Currently disabled when the NVIDIA driver is used because of a driver
issue that will need to be worked around.

svn path=/trunk/KDE/kdebase/workspace/; revision=1125672
2010-05-11 22:05:00 +00:00

380 lines
14 KiB
C++

/********************************************************************
KWin - the KDE window manager
This file is part of the KDE project.
Copyright (C) 2007 Rivo Laks <rivolaks@hot.ee>
Copyright (C) 2007 Lubos Lunak <l.lunak@kde.org>
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/>.
*********************************************************************/
#include "taskbarthumbnail.h"
#include <kdebug.h>
#ifdef KWIN_HAVE_OPENGL_COMPOSITING
#include <kwinglutils.h>
#endif
#include <qmath.h>
#include <cmath>
// This effect shows a preview inside a window that has a special property set
// on it that says which window and where to render. It is used by the taskbar
// to show window previews in tooltips.
namespace KWin
{
KWIN_EFFECT( taskbarthumbnail, TaskbarThumbnailEffect )
TaskbarThumbnailEffect::TaskbarThumbnailEffect()
{
atom = XInternAtom( display(), "_KDE_WINDOW_PREVIEW", False );
effects->registerPropertyType( atom, true );
// TODO hackish way to announce support, make better after 4.0
unsigned char dummy = 0;
XChangeProperty( display(), rootWindow(), atom, atom, 8, PropModeReplace, &dummy, 1 );
// With the NVIDIA driver the alpha values are set to 0 when drawing a window texture
// without an alpha channel on an FBO that has an alpha channel.
// This creates problems when the FBO is drawn on the backbuffer with blending enabled,
// so for now we only do high quality scaling with windows with alpha channels with
// the NVIDIA driver.
const QByteArray vendor = (const char *) glGetString( GL_VENDOR );
alphaWindowsOnly = vendor.startsWith("NVIDIA");
if ( GLShader::fragmentShaderSupported() &&
GLShader::vertexShaderSupported() &&
GLRenderTarget::supported() )
{
shader = new GLShader(":/taskbarthumbnail/vertex.glsl", ":/taskbarthumbnail/fragment.glsl");
shader->bind();
uTexUnit = shader->uniformLocation("texUnit");
uKernelSize = shader->uniformLocation("kernelSize");
uKernel = shader->uniformLocation("kernel");
uOffsets = shader->uniformLocation("offsets");
shader->unbind();
}
else
{
shader = 0;
}
offscreenTex = 0;
offscreenTarget = 0;
}
TaskbarThumbnailEffect::~TaskbarThumbnailEffect()
{
XDeleteProperty( display(), rootWindow(), atom );
effects->registerPropertyType( atom, false );
delete offscreenTarget;
delete offscreenTex;
delete shader;
}
void TaskbarThumbnailEffect::prePaintScreen( ScreenPrePaintData& data, int time )
{
// if( thumbnails.count() > 0 )
// // TODO this should not be needed (it causes whole screen repaint)
// data.mask |= PAINT_SCREEN_WITH_TRANSFORMED_WINDOWS;
effects->prePaintScreen(data, time);
}
void TaskbarThumbnailEffect::prePaintWindow( EffectWindow* w, WindowPrePaintData& data, int time )
{
// TODO what if of the windows is translucent and one not? change data.mask?
effects->prePaintWindow( w, data, time );
}
void TaskbarThumbnailEffect::updateOffscreenSurfaces()
{
int w = displayWidth();
int h = displayHeight();
if ( !GLTexture::NPOTTextureSupported() )
{
w = nearestPowerOfTwo( w );
h = nearestPowerOfTwo( h );
}
if ( !offscreenTex || offscreenTex->width() != w || offscreenTex->height() != h )
{
if ( offscreenTex )
{
delete offscreenTex;
delete offscreenTarget;
}
offscreenTex = new GLTexture( w, h );
offscreenTex->setFilter( GL_LINEAR );
offscreenTex->setWrapMode( GL_CLAMP_TO_EDGE );
offscreenTarget = new GLRenderTarget( offscreenTex );
}
}
static float sinc( float x )
{
return std::sin( x * M_PI ) / ( x * M_PI );
}
static float lanczos( float x, float a )
{
if ( qFuzzyCompare( x + 1.0, 1.0 ) )
return 1.0;
if ( qAbs( x ) >= a )
return 0.0;
return sinc( x ) * sinc( x / a );
}
QVector<QVector4D> TaskbarThumbnailEffect::createKernel( float delta )
{
const float a = 2.0;
// The two outermost samples always fall at points where the lanczos
// function returns 0, so we'll skip them.
const int sampleCount = qBound( 3, qCeil(delta * a) * 2 + 1 - 2, 49 );
const int center = sampleCount / 2;
const int kernelSize = center + 1;
const float factor = 1.0 / delta;
QVector<float> values( kernelSize );
QVector<QVector4D> kernel( kernelSize );
float sum = 0;
for ( int i = 0; i < kernelSize; i++ ) {
const float val = lanczos( i * factor, a );
sum += i > 0 ? val * 2 : val;
values[i] = val;
}
// Normalize the kernel
for ( int i = 0; i < kernelSize; i++ ) {
const float val = values[i] / sum;
kernel[i] = QVector4D( val, val, val, val );
}
return kernel;
}
QVector<QVector2D> TaskbarThumbnailEffect::createOffsets( int count, float width, Qt::Orientation direction )
{
QVector<QVector2D> offsets( count );
for ( int i = 0; i < count; i++ ) {
offsets[i] = ( direction == Qt::Horizontal ) ?
QVector2D( i / width, 0 ) : QVector2D( 0, i / width );
}
return offsets;
}
void TaskbarThumbnailEffect::paintWindow( EffectWindow* w, int mask, QRegion region, WindowPaintData& data )
{
effects->paintWindow( w, mask, region, data ); // paint window first
if( thumbnails.contains( w ))
{ // paint thumbnails on it
int mask = PAINT_WINDOW_TRANSFORMED;
if( data.opacity == 1.0 )
mask |= PAINT_WINDOW_OPAQUE;
else
mask |= PAINT_WINDOW_TRANSLUCENT;
foreach( const Data &thumb, thumbnails.values( w ))
{
EffectWindow* thumbw = effects->findWindow( thumb.window );
if( thumbw == NULL )
continue;
WindowPaintData thumbData( thumbw );
thumbData.opacity = data.opacity;
QRect r;
#ifdef KWIN_HAVE_OPENGL_COMPOSITING
if( effects->compositingType() == KWin::OpenGLCompositing )
{
if ( shader && (!alphaWindowsOnly || thumbw->hasAlpha()) )
{
int tx = thumb.rect.x() + w->x();
int ty = thumb.rect.y() + w->y();
int tw = thumb.rect.width();
int th = thumb.rect.height();
int sw = thumbw->width();
int sh = thumbw->height();
setPositionTransformations( thumbData, r,
thumbw, QRect(0, 0, sw, sh), Qt::KeepAspectRatio );
// Bind the offscreen FBO and draw the window on it unscaled
updateOffscreenSurfaces();
effects->pushRenderTarget( offscreenTarget );
glClear( GL_COLOR_BUFFER_BIT );
effects->drawWindow( thumbw, mask, r, thumbData );
// Create a scratch texture and copy the rendered window into it
GLTexture tex( sw, sh );
tex.setFilter( GL_LINEAR );
tex.setWrapMode( GL_CLAMP_TO_EDGE );
tex.bind();
glCopyTexSubImage2D( GL_TEXTURE_2D, 0, 0, 0, 0, offscreenTex->height() - sh, sw, sh );
// Set up the shader for horizontal scaling
float dx = sw / float(tw);
QVector<QVector4D> kernel = createKernel( dx );
QVector<QVector2D> offsets = createOffsets( kernel.size(), sw, Qt::Horizontal );
shader->bind();
glUniform1i( uTexUnit, 0 );
glUniform1i( uKernelSize, kernel.size() );
glUniform2fv( uOffsets, offsets.size(), (const GLfloat*)offsets.constData() );
glUniform4fv( uKernel, kernel.size(), (const GLfloat*)kernel.constData() );
// Draw the window back into the FBO, this time scaled horizontally
glClear( GL_COLOR_BUFFER_BIT );
glBegin( GL_QUADS );
glTexCoord2f( 0, 0 ); glVertex2i( 0, 0 ); // Top left
glTexCoord2f( 1, 0 ); glVertex2i( tw, 0 ); // Top right
glTexCoord2f( 1, 1 ); glVertex2i( tw, sh ); // Bottom right
glTexCoord2f( 0, 1 ); glVertex2i( 0, sh ); // Bottom left
glEnd();
// At this point we don't need the scratch texture anymore
tex.unbind();
tex.discard();
// Unbind the FBO
effects->popRenderTarget();
// Set up the shader for vertical scaling
float dy = sh / float(th);
kernel = createKernel( dy );
offsets = createOffsets( kernel.size(), offscreenTex->height(), Qt::Vertical );
glUniform1i( uKernelSize, kernel.size() );
glUniform2fv( uOffsets, offsets.size(), (const GLfloat*)offsets.constData() );
glUniform4fv( uKernel, kernel.size(), (const GLfloat*)kernel.constData() );
float sx2 = tw / float(offscreenTex->width());
float sy2 = 1 - (sh / float(offscreenTex->height()));
// Now draw the horizontally scaled window in the FBO at the right
// coordinates on the screen, while scaling it vertically and blending it.
offscreenTex->bind();
glPushAttrib( GL_COLOR_BUFFER_BIT );
glEnable( GL_BLEND );
glBlendFunc( GL_ONE, GL_ONE_MINUS_SRC_ALPHA );
glBegin( GL_QUADS );
glTexCoord2f( 0, sy2 ); glVertex2i( tx, ty ); // Top left
glTexCoord2f( sx2, sy2 ); glVertex2i( tx + tw, ty ); // Top right
glTexCoord2f( sx2, 1 ); glVertex2i( tx + tw, ty + th ); // Bottom right
glTexCoord2f( 0, 1 ); glVertex2i( tx, ty + th ); // Bottom left
glEnd();
glPopAttrib();
offscreenTex->unbind();
shader->unbind();
// Delete the offscreen surface after 5 seconds
timer.start( 5000, this );
return;
}
if ( data.shader )
{
// there is a shader - update texture width and height
int texw = thumbw->width();
int texh = thumbw->height();
if( !GLTexture::NPOTTextureSupported() )
{
kWarning( 1212 ) << "NPOT textures not supported, wasting some memory" ;
texw = nearestPowerOfTwo( texw );
texh = nearestPowerOfTwo( texh );
}
thumbData.shader = data.shader;
thumbData.shader->setTextureWidth( (float)texw );
thumbData.shader->setTextureHeight( (float)texh );
}
} // if ( effects->compositingType() == KWin::OpenGLCompositing )
#endif
setPositionTransformations( thumbData, r,
thumbw, thumb.rect.translated( w->pos()), Qt::KeepAspectRatio );
effects->drawWindow( thumbw, mask, r, thumbData );
}
}
} // End of function
void TaskbarThumbnailEffect::windowDamaged( EffectWindow* w, const QRect& damage )
{
Q_UNUSED( damage );
// Update the thumbnail if the window was damaged
foreach( EffectWindow* window, thumbnails.uniqueKeys() )
foreach( const Data &thumb, thumbnails.values( window ))
if( w == effects->findWindow( thumb.window ))
effects->addRepaint( thumb.rect.translated( window->pos() ));
}
void TaskbarThumbnailEffect::windowAdded( EffectWindow* w )
{
propertyNotify( w, atom ); // read initial value
}
void TaskbarThumbnailEffect::windowDeleted( EffectWindow* w )
{
thumbnails.remove( w );
}
void TaskbarThumbnailEffect::propertyNotify( EffectWindow* w, long a )
{
if( !w || a != atom )
return;
thumbnails.remove( w );
QByteArray data = w->readProperty( atom, atom, 32 );
if( data.length() < 1 )
return;
long* d = reinterpret_cast< long* >( data.data());
int len = data.length() / sizeof( d[ 0 ] );
int pos = 0;
int cnt = d[ 0 ];
++pos;
for( int i = 0;
i < cnt;
++i )
{
int size = d[ pos ];
if( len - pos < size )
return; // format error
++pos;
Data data;
data.window = d[ pos ];
data.rect = QRect( d[ pos + 1 ], d[ pos + 2 ], d[ pos + 3 ], d[ pos + 4 ] );
thumbnails.insert( w, data );
pos += size;
}
}
void TaskbarThumbnailEffect::timerEvent( QTimerEvent *event )
{
if (event->timerId() == timer.timerId())
{
timer.stop();
delete offscreenTarget;
delete offscreenTex;
offscreenTarget = 0;
offscreenTex = 0;
}
}
} // namespace