diff options
Diffstat (limited to 'tdecore/kiconeffect.cpp')
-rw-r--r-- | tdecore/kiconeffect.cpp | 920 |
1 files changed, 920 insertions, 0 deletions
diff --git a/tdecore/kiconeffect.cpp b/tdecore/kiconeffect.cpp new file mode 100644 index 000000000..9c76c7e4d --- /dev/null +++ b/tdecore/kiconeffect.cpp @@ -0,0 +1,920 @@ +/* vi: ts=8 sts=4 sw=4 + * $Id$ + * + * This file is part of the KDE project, module tdecore. + * Copyright (C) 2000 Geert Jansen <[email protected]> + * with minor additions and based on ideas from + * Torsten Rahn <[email protected]> + * + * This is free software; it comes under the GNU Library General + * Public License, version 2. See the file "COPYING.LIB" for the + * exact licensing terms. + */ + +#include <config.h> +#include <unistd.h> +#include <math.h> + +#include <tqstring.h> +#include <tqstringlist.h> +#include <tqbitmap.h> +#include <tqpixmap.h> +#include <tqimage.h> +#include <tqcolor.h> +#include <tqwidget.h> +#include <tqpainter.h> +#include <tqpen.h> +#include <tqapplication.h> +#include <tqpoint.h> +#include <tqrect.h> + +#include <kdebug.h> +#include <kglobal.h> +#include <kconfig.h> +#include <kglobalsettings.h> +#include <kicontheme.h> +#include "kiconeffect.h" + +#if defined(Q_WS_WIN) || defined(Q_WS_MACX) +static bool qt_use_xrender=true; +static bool qt_has_xft=true; +#else +extern bool qt_use_xrender; +extern bool qt_has_xft; +#endif +class KIconEffectPrivate +{ +public: + TQString mKey[6][3]; + TQColor mColor2[6][3]; +}; + +KIconEffect::KIconEffect() +{ + d = new KIconEffectPrivate; + init(); +} + +KIconEffect::~KIconEffect() +{ + delete d; + d = 0L; +} + +void KIconEffect::init() +{ + KConfig *config = KGlobal::config(); + + int i, j, effect=-1; + TQStringList groups; + groups += "Desktop"; + groups += "Toolbar"; + groups += "MainToolbar"; + groups += "Small"; + groups += "Panel"; + + TQStringList states; + states += "Default"; + states += "Active"; + states += "Disabled"; + + TQStringList::ConstIterator it, it2; + TQString _togray("togray"); + TQString _colorize("colorize"); + TQString _desaturate("desaturate"); + TQString _togamma("togamma"); + TQString _none("none"); + TQString _tomonochrome("tomonochrome"); + + KConfigGroupSaver cs(config, "default"); + + for (it=groups.begin(), i=0; it!=groups.end(); it++, i++) + { + // Default effects + mEffect[i][0] = NoEffect; + mEffect[i][1] = ((i==0)||(i==4)) ? ToGamma : NoEffect; + mEffect[i][2] = ToGray; + + mTrans[i][0] = false; + mTrans[i][1] = false; + mTrans[i][2] = true; + mValue[i][0] = 1.0; + mValue[i][1] = ((i==0)||(i==4)) ? 0.7 : 1.0; + mValue[i][2] = 1.0; + mColor[i][0] = TQColor(144,128,248); + mColor[i][1] = TQColor(169,156,255); + mColor[i][2] = TQColor(34,202,0); + d->mColor2[i][0] = TQColor(0,0,0); + d->mColor2[i][1] = TQColor(0,0,0); + d->mColor2[i][2] = TQColor(0,0,0); + + config->setGroup(*it + "Icons"); + for (it2=states.begin(), j=0; it2!=states.end(); it2++, j++) + { + TQString tmp = config->readEntry(*it2 + "Effect"); + if (tmp == _togray) + effect = ToGray; + else if (tmp == _colorize) + effect = Colorize; + else if (tmp == _desaturate) + effect = DeSaturate; + else if (tmp == _togamma) + effect = ToGamma; + else if (tmp == _tomonochrome) + effect = ToMonochrome; + else if (tmp == _none) + effect = NoEffect; + else + continue; + if(effect != -1) + mEffect[i][j] = effect; + mValue[i][j] = config->readDoubleNumEntry(*it2 + "Value"); + mColor[i][j] = config->readColorEntry(*it2 + "Color"); + d->mColor2[i][j] = config->readColorEntry(*it2 + "Color2"); + mTrans[i][j] = config->readBoolEntry(*it2 + "SemiTransparent"); + + } + } +} + +bool KIconEffect::hasEffect(int group, int state) const +{ + return mEffect[group][state] != NoEffect; +} + +TQString KIconEffect::fingerprint(int group, int state) const +{ + if ( group >= KIcon::LastGroup ) return ""; + TQString cached = d->mKey[group][state]; + if (cached.isEmpty()) + { + TQString tmp; + cached = tmp.setNum(mEffect[group][state]); + cached += ':'; + cached += tmp.setNum(mValue[group][state]); + cached += ':'; + cached += mTrans[group][state] ? TQString::tqfromLatin1("trans") + : TQString::tqfromLatin1("notrans"); + if (mEffect[group][state] == Colorize || mEffect[group][state] == ToMonochrome) + { + cached += ':'; + cached += mColor[group][state].name(); + } + if (mEffect[group][state] == ToMonochrome) + { + cached += ':'; + cached += d->mColor2[group][state].name(); + } + + d->mKey[group][state] = cached; + } + + return cached; +} + +TQImage KIconEffect::apply(TQImage image, int group, int state) const +{ + if (state >= KIcon::LastState) + { + kdDebug(265) << "Illegal icon state: " << state << "\n"; + return image; + } + if (group >= KIcon::LastGroup) + { + kdDebug(265) << "Illegal icon group: " << group << "\n"; + return image; + } + return apply(image, mEffect[group][state], mValue[group][state], + mColor[group][state], d->mColor2[group][state], mTrans[group][state]); +} + +TQImage KIconEffect::apply(TQImage image, int effect, float value, const TQColor col, bool trans) const +{ + return apply (image, effect, value, col, KGlobalSettings::baseColor(), trans); +} + +TQImage KIconEffect::apply(TQImage image, int effect, float value, const TQColor col, const TQColor col2, bool trans) const +{ + if (effect >= LastEffect ) + { + kdDebug(265) << "Illegal icon effect: " << effect << "\n"; + return image; + } + if (value > 1.0) + value = 1.0; + else if (value < 0.0) + value = 0.0; + switch (effect) + { + case ToGray: + toGray(image, value); + break; + case DeSaturate: + deSaturate(image, value); + break; + case Colorize: + colorize(image, col, value); + break; + case ToGamma: + toGamma(image, value); + break; + case ToMonochrome: + toMonochrome(image, col, col2, value); + break; + } + if (trans == true) + { + semiTransparent(image); + } + return image; +} + +TQPixmap KIconEffect::apply(TQPixmap pixmap, int group, int state) const +{ + if (state >= KIcon::LastState) + { + kdDebug(265) << "Illegal icon state: " << state << "\n"; + return pixmap; + } + if (group >= KIcon::LastGroup) + { + kdDebug(265) << "Illegal icon group: " << group << "\n"; + return pixmap; + } + return apply(pixmap, mEffect[group][state], mValue[group][state], + mColor[group][state], d->mColor2[group][state], mTrans[group][state]); +} + +TQPixmap KIconEffect::apply(TQPixmap pixmap, int effect, float value, + const TQColor col, bool trans) const +{ + return apply (pixmap, effect, value, col, KGlobalSettings::baseColor(), trans); +} + +TQPixmap KIconEffect::apply(TQPixmap pixmap, int effect, float value, + const TQColor col, const TQColor col2, bool trans) const +{ + TQPixmap result; + + if (effect >= LastEffect ) + { + kdDebug(265) << "Illegal icon effect: " << effect << "\n"; + return result; + } + + if ((trans == true) && (effect == NoEffect)) + { + result = pixmap; + semiTransparent(result); + } + else if ( effect != NoEffect ) + { + TQImage tmpImg = pixmap.convertToImage(); + tmpImg = apply(tmpImg, effect, value, col, col2, trans); + result.convertFromImage(tmpImg); + } + else + result = pixmap; + + return result; +} + +// Taken from KImageEffect. We don't want to link tdecore to tdeui! As long +// as this code is not too big, it doesn't seem much of a problem to me. + +void KIconEffect::toGray(TQImage &img, float value) +{ + int pixels = (img.depth() > 8) ? img.width()*img.height() + : img.numColors(); + unsigned int *data = img.depth() > 8 ? (unsigned int *) img.bits() + : (unsigned int *) img.tqcolorTable(); + int rval, gval, bval, val, alpha, i; + for (i=0; i<pixels; i++) + { + val = tqGray(data[i]); + alpha = tqAlpha(data[i]); + if (value < 1.0) + { + rval = static_cast<int>(value*val+(1.0-value)*tqRed(data[i])); + gval = static_cast<int>(value*val+(1.0-value)*tqGreen(data[i])); + bval = static_cast<int>(value*val+(1.0-value)*tqBlue(data[i])); + data[i] = tqRgba(rval, gval, bval, alpha); + } else + data[i] = tqRgba(val, val, val, alpha); + } +} + +void KIconEffect::colorize(TQImage &img, const TQColor &col, float value) +{ + int pixels = (img.depth() > 8) ? img.width()*img.height() + : img.numColors(); + unsigned int *data = img.depth() > 8 ? (unsigned int *) img.bits() + : (unsigned int *) img.tqcolorTable(); + int rval, gval, bval, val, alpha, i; + float rcol = col.red(), gcol = col.green(), bcol = col.blue(); + for (i=0; i<pixels; i++) + { + val = tqGray(data[i]); + if (val < 128) + { + rval = static_cast<int>(rcol/128*val); + gval = static_cast<int>(gcol/128*val); + bval = static_cast<int>(bcol/128*val); + } + else if (val > 128) + { + rval = static_cast<int>((val-128)*(2-rcol/128)+rcol-1); + gval = static_cast<int>((val-128)*(2-gcol/128)+gcol-1); + bval = static_cast<int>((val-128)*(2-bcol/128)+bcol-1); + } + else // val == 128 + { + rval = static_cast<int>(rcol); + gval = static_cast<int>(gcol); + bval = static_cast<int>(bcol); + } + if (value < 1.0) + { + rval = static_cast<int>(value*rval+(1.0 - value)*tqRed(data[i])); + gval = static_cast<int>(value*gval+(1.0 - value)*tqGreen(data[i])); + bval = static_cast<int>(value*bval+(1.0 - value)*tqBlue(data[i])); + } + + alpha = tqAlpha(data[i]); + data[i] = tqRgba(rval, gval, bval, alpha); + } +} + +void KIconEffect::toMonochrome(TQImage &img, const TQColor &black, const TQColor &white, float value) { + int pixels = (img.depth() > 8) ? img.width()*img.height() : img.numColors(); + unsigned int *data = img.depth() > 8 ? (unsigned int *) img.bits() + : (unsigned int *) img.tqcolorTable(); + int rval, gval, bval, alpha, i; + int rw = white.red(), gw = white.green(), bw = white.blue(); + int rb = black.red(), gb = black.green(), bb = black.blue(); + + double values = 0, sum = 0; + bool grayscale = true; + // Step 1: determine the average brightness + for (i=0; i<pixels; i++) { + sum += tqGray(data[i])*tqAlpha(data[i]) + 255*(255-tqAlpha(data[i])); + values += 255; + if ((tqRed(data[i]) != tqGreen(data[i]) ) || (tqGreen(data[i]) != tqBlue(data[i]) )) + grayscale = false; + } + double medium = sum/values; + + // Step 2: Modify the image + if (grayscale) { + for (i=0; i<pixels; i++) { + int v = tqRed(data[i]); + rval = static_cast<int>( ((255-v)*rb + v*rw)*value/255 + (1.0-value)*tqRed(data[i])); + gval = static_cast<int>( ((255-v)*gb + v*gw)*value/255 + (1.0-value)*tqGreen(data[i])); + bval = static_cast<int>( ((255-v)*bb + v*bw)*value/255 + (1.0-value)*tqBlue(data[i])); + + alpha = tqAlpha(data[i]); + data[i] = tqRgba(rval, gval, bval, alpha); + } + } + else { + for (i=0; i<pixels; i++) { + if (tqGray(data[i]) <= medium) { + rval = static_cast<int>(value*rb+(1.0-value)*tqRed(data[i])); + gval = static_cast<int>(value*gb+(1.0-value)*tqGreen(data[i])); + bval = static_cast<int>(value*bb+(1.0-value)*tqBlue(data[i])); + } + else { + rval = static_cast<int>(value*rw+(1.0-value)*tqRed(data[i])); + gval = static_cast<int>(value*gw+(1.0-value)*tqGreen(data[i])); + bval = static_cast<int>(value*bw+(1.0-value)*tqBlue(data[i])); + } + + alpha = tqAlpha(data[i]); + data[i] = tqRgba(rval, gval, bval, alpha); + } + } +} + +void KIconEffect::deSaturate(TQImage &img, float value) +{ + int pixels = (img.depth() > 8) ? img.width()*img.height() + : img.numColors(); + unsigned int *data = (img.depth() > 8) ? (unsigned int *) img.bits() + : (unsigned int *) img.tqcolorTable(); + TQColor color; + int h, s, v, i; + for (i=0; i<pixels; i++) + { + color.setRgb(data[i]); + color.hsv(&h, &s, &v); + color.setHsv(h, (int) (s * (1.0 - value) + 0.5), v); + data[i] = tqRgba(color.red(), color.green(), color.blue(), + tqAlpha(data[i])); + } +} + +void KIconEffect::toGamma(TQImage &img, float value) +{ + int pixels = (img.depth() > 8) ? img.width()*img.height() + : img.numColors(); + unsigned int *data = (img.depth() > 8) ? (unsigned int *) img.bits() + : (unsigned int *) img.tqcolorTable(); + TQColor color; + int i, rval, gval, bval; + float gamma; + gamma = 1/(2*value+0.5); + + for (i=0; i<pixels; i++) + { + color.setRgb(data[i]); + color.rgb(&rval, &gval, &bval); + rval = static_cast<int>(pow(static_cast<float>(rval)/255 , gamma)*255); + gval = static_cast<int>(pow(static_cast<float>(gval)/255 , gamma)*255); + bval = static_cast<int>(pow(static_cast<float>(bval)/255 , gamma)*255); + data[i] = tqRgba(rval, gval, bval, tqAlpha(data[i])); + } +} + +void KIconEffect::semiTransparent(TQImage &img) +{ + img.setAlphaBuffer(true); + + int x, y; + if (img.depth() == 32) + { + int width = img.width(); + int height = img.height(); + + if (qt_use_xrender && qt_has_xft ) + for (y=0; y<height; y++) + { +#ifdef WORDS_BIGENDIAN + uchar *line = (uchar*) img.scanLine(y); +#else + uchar *line = (uchar*) img.scanLine(y) + 3; +#endif + for (x=0; x<width; x++) + { + *line >>= 1; + line += 4; + } + } + else + for (y=0; y<height; y++) + { + QRgb *line = (QRgb *) img.scanLine(y); + for (x=(y%2); x<width; x+=2) + line[x] &= 0x00ffffff; + } + + } else + { + // Insert transparent pixel into the clut. + int transColor = -1; + + // search for a color that is already transparent + for (x=0; x<img.numColors(); x++) + { + // try to find already transparent pixel + if (tqAlpha(img.color(x)) < 127) + { + transColor = x; + break; + } + } + + + // FIXME: image must have transparency + if(transColor < 0 || transColor >= img.numColors()) + return; + + img.setColor(transColor, 0); + if(img.depth() == 8) + { + for (y=0; y<img.height(); y++) + { + unsigned char *line = img.scanLine(y); + for (x=(y%2); x<img.width(); x+=2) + line[x] = transColor; + } + } + else + { + // SLOOW, but simple, as we would have to + // deal with endianess etc on our own here + for (y=0; y<img.height(); y++) + for (x=(y%2); x<img.width(); x+=2) + img.setPixel(x, y, transColor); + } + } +} + +void KIconEffect::semiTransparent(TQPixmap &pix) +{ + if ( qt_use_xrender && qt_has_xft ) + { + TQImage img=pix.convertToImage(); + semiTransparent(img); + pix.convertFromImage(img); + return; + } + + TQImage img; + if (pix.mask() != 0L) + img = pix.mask()->convertToImage(); + else + { + img.create(pix.size(), 1, 2, TQImage::BigEndian); + img.fill(1); + } + + for (int y=0; y<img.height(); y++) + { + QRgb *line = (QRgb *) img.scanLine(y); + QRgb pattern = (y % 2) ? 0x55555555 : 0xaaaaaaaa; + for (int x=0; x<(img.width()+31)/32; x++) + line[x] &= pattern; + } + TQBitmap mask; + mask.convertFromImage(img); + pix.setMask(mask); +} + +TQImage KIconEffect::doublePixels(TQImage src) const +{ + TQImage dst; + if (src.depth() == 1) + { + kdDebug(265) << "image depth 1 not supported\n"; + return dst; + } + + int w = src.width(); + int h = src.height(); + dst.create(w*2, h*2, src.depth()); + dst.setAlphaBuffer(src.hasAlphaBuffer()); + + int x, y; + if (src.depth() == 32) + { + QRgb *l1, *l2; + for (y=0; y<h; y++) + { + l1 = (QRgb *) src.scanLine(y); + l2 = (QRgb *) dst.scanLine(y*2); + for (x=0; x<w; x++) + { + l2[x*2] = l2[x*2+1] = l1[x]; + } + memcpy(dst.scanLine(y*2+1), l2, dst.bytesPerLine()); + } + } else + { + for (x=0; x<src.numColors(); x++) + dst.setColor(x, src.color(x)); + + unsigned char *l1, *l2; + for (y=0; y<h; y++) + { + l1 = src.scanLine(y); + l2 = dst.scanLine(y*2); + for (x=0; x<w; x++) + { + l2[x*2] = l1[x]; + l2[x*2+1] = l1[x]; + } + memcpy(dst.scanLine(y*2+1), l2, dst.bytesPerLine()); + } + } + return dst; +} + +void KIconEffect::overlay(TQImage &src, TQImage &overlay) +{ + if (src.depth() != overlay.depth()) + { + kdDebug(265) << "Image depth src != overlay!\n"; + return; + } + if (src.size() != overlay.size()) + { + kdDebug(265) << "Image size src != overlay\n"; + return; + } + if (!overlay.hasAlphaBuffer()) + { + kdDebug(265) << "Overlay doesn't have alpha buffer!\n"; + return; + } + + int i, j; + + // We don't do 1 bpp + + if (src.depth() == 1) + { + kdDebug(265) << "1bpp not supported!\n"; + return; + } + + // Overlay at 8 bpp doesn't use alpha blending + + if (src.depth() == 8) + { + if (src.numColors() + overlay.numColors() > 255) + { + kdDebug(265) << "Too many colors in src + overlay!\n"; + return; + } + + // Find transparent pixel in overlay + int trans; + for (trans=0; trans<overlay.numColors(); trans++) + { + if (tqAlpha(overlay.color(trans)) == 0) + { + kdDebug(265) << "transparent pixel found at " << trans << "\n"; + break; + } + } + if (trans == overlay.numColors()) + { + kdDebug(265) << "transparent pixel not found!\n"; + return; + } + + // Merge color tables + int nc = src.numColors(); + src.setNumColors(nc + overlay.numColors()); + for (i=0; i<overlay.numColors(); i++) + { + src.setColor(nc+i, overlay.color(i)); + } + + // Overwrite nontransparent pixels. + unsigned char *oline, *sline; + for (i=0; i<src.height(); i++) + { + oline = overlay.scanLine(i); + sline = src.scanLine(i); + for (j=0; j<src.width(); j++) + { + if (oline[j] != trans) + sline[j] = oline[j]+nc; + } + } + } + + // Overlay at 32 bpp does use alpha blending + + if (src.depth() == 32) + { + QRgb *oline, *sline; + int r1, g1, b1, a1; + int r2, g2, b2, a2; + + for (i=0; i<src.height(); i++) + { + oline = (QRgb *) overlay.scanLine(i); + sline = (QRgb *) src.scanLine(i); + + for (j=0; j<src.width(); j++) + { + r1 = tqRed(oline[j]); + g1 = tqGreen(oline[j]); + b1 = tqBlue(oline[j]); + a1 = tqAlpha(oline[j]); + + r2 = tqRed(sline[j]); + g2 = tqGreen(sline[j]); + b2 = tqBlue(sline[j]); + a2 = tqAlpha(sline[j]); + + r2 = (a1 * r1 + (0xff - a1) * r2) >> 8; + g2 = (a1 * g1 + (0xff - a1) * g2) >> 8; + b2 = (a1 * b1 + (0xff - a1) * b2) >> 8; + a2 = QMAX(a1, a2); + + sline[j] = tqRgba(r2, g2, b2, a2); + } + } + } + + return; +} + + void +KIconEffect::visualActivate(TQWidget * widget, TQRect rect) +{ + if (!KGlobalSettings::visualActivate()) + return; + + uint actSpeed = KGlobalSettings::visualActivateSpeed(); + + uint actCount = QMIN(rect.width(), rect.height()) / 2; + + // Clip actCount to range 1..10. + + if (actCount < 1) + actCount = 1; + + else if (actCount > 10) + actCount = 10; + + // Clip actSpeed to range 1..100. + + if (actSpeed < 1) + actSpeed = 1; + + else if (actSpeed > 100) + actSpeed = 100; + + // actSpeed needs to be converted to actDelay. + // actDelay is inversely proportional to actSpeed and needs to be + // divided up into actCount portions. + // We also convert the us value to ms. + + unsigned int actDelay = (1000 * (100 - actSpeed)) / actCount; + + //kdDebug() << "actCount=" << actCount << " actDelay=" << actDelay << endl; + + TQPoint c = rect.center(); + + TQPainter p(widget); + + // Use NotROP to avoid having to tqrepaint the pixmap each time. + p.setPen(TQPen(Qt::black, 2, Qt::DotLine)); + p.setRasterOp(TQt::NotROP); + + // The spacing between the rects we draw. + // Use the minimum of width and height to avoid painting outside the + // pixmap area. + //unsigned int delta(QMIN(rect.width() / actCount, rect.height() / actCount)); + + // Support for rectangles by David + unsigned int deltaX = rect.width() / actCount; + unsigned int deltaY = rect.height() / actCount; + + for (unsigned int i = 1; i < actCount; i++) { + + int w = i * deltaX; + int h = i * deltaY; + + rect.setRect(c.x() - w / 2, c.y() - h / 2, w, h); + + p.drawRect(rect); + p.flush(); + + usleep(actDelay); + + p.drawRect(rect); + } +} + +void +KIconEffect::visualActivate(TQWidget * widget, TQRect rect, TQPixmap *pixmap) +{ + if (!KGlobalSettings::visualActivate()) + return; + + // Image too big to display smoothly + if ((rect.width() > 160) || (rect.height() > 160)) { + visualActivate(widget, rect); // call old effect + return; + } + + uint actSpeed = KGlobalSettings::visualActivateSpeed(); + uint actCount = TQMIN(rect.width(), rect.height()) / 4; + + + // Clip actCount to range 1..10. + if (actCount < 1) + actCount = 1; + + else if (actCount > 10) + actCount = 10; + + // Clip actSpeed to range 1..100. + if (actSpeed < 1) + actSpeed = 1; + + else if (actSpeed > 100) + actSpeed = 100; + + // actSpeed needs to be converted to actDelay. + // actDelay is inversely proportional to actSpeed and needs to be + // divided up into actCount portions. + // We also convert the us value to ms. + + unsigned int actDelay = (1000 * (100 - actSpeed)) / actCount; + + unsigned int deltaX = rect.width() / actCount * 1.5; + unsigned int deltaY = rect.height() / actCount * 1.5; + + TQPoint c = rect.center(); + TQRect maxRect(c.x() - (actCount * 2) * deltaX /2, + c.y() - (actCount * 2) * deltaY /2, + actCount * 2 * deltaX, + actCount * 2 * deltaY); + + // convert rect to global coordinates if needed + if ((widget->rect().width() <= maxRect.width()) + || (widget->rect().height() <= maxRect.height())) + { + TQPoint topLeft(rect.x(), rect.y()); + rect.moveLeft(widget->mapToGlobal(topLeft).x()); + rect.moveTop(widget->mapToGlobal(topLeft).y()); + c = rect.center(); + maxRect.setRect(c.x() - (actCount * 2) * deltaX /2, + c.y() - (actCount * 2) * deltaY /2, + actCount * 2 * deltaX, + actCount * 2 * deltaY); + } + + TQPainter *p; + TQImage img = pixmap->convertToImage(); + TQPixmap pix; + TQPixmap composite(maxRect.width(), maxRect.height(), -1, TQPixmap::BestOptim); + TQPainter cPainter(&composite); + TQPoint cComposite = composite.rect().center(); + + // enable alpha blending + img.setAlphaBuffer(true); + + // Ugly hack... Get "Screenshot" to blt into and even do that on the + // root window if the display area of <widget> is too small + if ((widget->rect().width() <= maxRect.width()) + || (widget->rect().height() <= maxRect.height())) + { +// p = new TQPainter(TQApplication::desktop()->screen( -1 ), TRUE); // WARNING: This was done in Qt3. It only worked in this placement due to a glitch in Qt3; it has therefore been moved below grabWidget, where it should have been in the first place. + pix = TQPixmap::grabWindow((TQApplication::desktop()->screen( -1 ))->winId(), + maxRect.x(), + maxRect.y(), + maxRect.width(), + maxRect.height()); + p = new TQPainter(TQApplication::desktop()->screen( -1 ), TRUE); + } else + { + // not as ugly as drawing directly to the screen +// p = new TQPainter(widget); // WARNING: This was done in Qt3. See above. + pix = TQPixmap::grabWidget(widget, + maxRect.x(), + maxRect.y(), + maxRect.width(), + maxRect.height()); + p = new TQPainter(widget); + } + uchar deltaAlpha = 255 / (actCount * 1.2); + + // Activate effect like MacOS X + for (unsigned int i = actCount; i < actCount * 2; i++) { + + int w = i * deltaX; + int h = i * deltaY; + + rect.setRect(cComposite.x() - w / 2, cComposite.y() - h / 2, w, h); + + // draw offscreen + cPainter.drawPixmap(0, 0, pix, 0, 0, pix.width(), pix.height()); + cPainter.drawImage(rect, img); + cPainter.flush(); + + // put onscreen + p->drawPixmap(maxRect, composite); + p->flush(); + + // Fade out Icon a bit more + int x, y; + if ((img.depth() == 32) && qt_use_xrender && qt_has_xft) + { + int width = img.width(); + int height = img.height(); + + for (y=0; y<height; y++) + { +#ifdef WORDS_BIGENDIAN + uchar *line = (uchar*) img.scanLine(y); +#else + uchar *line = (uchar*) img.scanLine(y) + 3; +#endif + for (x=0; x<width; x++) + { + *line = (*line < deltaAlpha) ? 0 : *line - deltaAlpha; + line += 4; + } + } + } + usleep(actDelay*3); + } + + // remove traces of the effect + if ((widget->rect().width() <= maxRect.width()) + || (widget->rect().height() <= maxRect.height())) + p->drawPixmap(maxRect, pix); + else { + p->drawPixmap(maxRect, pix); + widget->update(rect); + } + + delete p; +} |