/src/gui/image/qpixmapfilter.cpp

https://bitbucket.org/ultra_iter/qt-vtl · C++ · 1380 lines · 766 code · 158 blank · 456 comment · 128 complexity · 46b24501c5dc488d7d713edfb9a41519 MD5 · raw file

  1. /****************************************************************************
  2. **
  3. ** Copyright (C) 2012 Nokia Corporation and/or its subsidiary(-ies).
  4. ** All rights reserved.
  5. ** Contact: Nokia Corporation (qt-info@nokia.com)
  6. **
  7. ** This file is part of the QtGui module of the Qt Toolkit.
  8. **
  9. ** $QT_BEGIN_LICENSE:LGPL$
  10. ** GNU Lesser General Public License Usage
  11. ** This file may be used under the terms of the GNU Lesser General Public
  12. ** License version 2.1 as published by the Free Software Foundation and
  13. ** appearing in the file LICENSE.LGPL included in the packaging of this
  14. ** file. Please review the following information to ensure the GNU Lesser
  15. ** General Public License version 2.1 requirements will be met:
  16. ** http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html.
  17. **
  18. ** In addition, as a special exception, Nokia gives you certain additional
  19. ** rights. These rights are described in the Nokia Qt LGPL Exception
  20. ** version 1.1, included in the file LGPL_EXCEPTION.txt in this package.
  21. **
  22. ** GNU General Public License Usage
  23. ** Alternatively, this file may be used under the terms of the GNU General
  24. ** Public License version 3.0 as published by the Free Software Foundation
  25. ** and appearing in the file LICENSE.GPL included in the packaging of this
  26. ** file. Please review the following information to ensure the GNU General
  27. ** Public License version 3.0 requirements will be met:
  28. ** http://www.gnu.org/copyleft/gpl.html.
  29. **
  30. ** Other Usage
  31. ** Alternatively, this file may be used in accordance with the terms and
  32. ** conditions contained in a signed written agreement between you and Nokia.
  33. **
  34. **
  35. **
  36. **
  37. **
  38. ** $QT_END_LICENSE$
  39. **
  40. ****************************************************************************/
  41. #include <qglobal.h>
  42. #include <QDebug>
  43. #include "qpainter.h"
  44. #include "qpixmap.h"
  45. #include "qpixmapfilter_p.h"
  46. #include "qvarlengtharray.h"
  47. #include "private/qapplication_p.h"
  48. #include "private/qgraphicssystem_p.h"
  49. #include "private/qpaintengineex_p.h"
  50. #include "private/qpaintengine_raster_p.h"
  51. #include "qmath.h"
  52. #include "private/qmath_p.h"
  53. #include "private/qmemrotate_p.h"
  54. #include "private/qdrawhelper_p.h"
  55. #ifndef QT_NO_GRAPHICSEFFECT
  56. QT_BEGIN_NAMESPACE
  57. class QPixmapFilterPrivate : public QObjectPrivate
  58. {
  59. Q_DECLARE_PUBLIC(QPixmapFilter)
  60. public:
  61. QPixmapFilter::FilterType type;
  62. };
  63. /*!
  64. \class QPixmapFilter
  65. \since 4.5
  66. \ingroup painting
  67. \brief The QPixmapFilter class provides the basic functionality for
  68. pixmap filter classes. Pixmap filter can be for example colorize or blur.
  69. QPixmapFilter is the base class for every pixmap filter. QPixmapFilter is
  70. an abstract class and cannot itself be instantiated. It provides a standard
  71. interface for filter processing.
  72. \internal
  73. */
  74. /*!
  75. \enum QPixmapFilter::FilterType
  76. \internal
  77. This enum describes the types of filter that can be applied to pixmaps.
  78. \value ConvolutionFilter A filter that is used to calculate the convolution
  79. of the image with a kernel. See
  80. QPixmapConvolutionFilter for more information.
  81. \value ColorizeFilter A filter that is used to change the overall color
  82. of an image. See QPixmapColorizeFilter for more
  83. information.
  84. \value DropShadowFilter A filter that is used to add a drop shadow to an
  85. image. See QPixmapDropShadowFilter for more
  86. information.
  87. \value BlurFilter A filter that is used to blur an image using
  88. a simple blur radius. See QPixmapBlurFilter
  89. for more information.
  90. \value UserFilter The first filter type that can be used for
  91. application-specific purposes.
  92. */
  93. /*!
  94. Constructs a default QPixmapFilter with the given \a type.
  95. This constructor should be used when subclassing QPixmapFilter to
  96. create custom user filters.
  97. \internal
  98. */
  99. QPixmapFilter::QPixmapFilter(FilterType type, QObject *parent)
  100. : QObject(*new QPixmapFilterPrivate, parent)
  101. {
  102. d_func()->type = type;
  103. }
  104. /*!
  105. \internal
  106. */
  107. QPixmapFilter::QPixmapFilter(QPixmapFilterPrivate&d, QPixmapFilter::FilterType type, QObject *parent)
  108. : QObject(d, parent)
  109. {
  110. d_func()->type = type;
  111. }
  112. /*!
  113. Destroys the pixmap filter.
  114. \internal
  115. */
  116. QPixmapFilter::~QPixmapFilter()
  117. {
  118. }
  119. /*!
  120. Returns the type of the filter. All standard pixmap filter classes
  121. are associated with a unique value.
  122. \internal
  123. */
  124. QPixmapFilter::FilterType QPixmapFilter::type() const
  125. {
  126. Q_D(const QPixmapFilter);
  127. return d->type;
  128. }
  129. /*!
  130. Returns the bounding rectangle that is affected by the pixmap
  131. filter if the filter is applied to the specified \a rect.
  132. \internal
  133. */
  134. QRectF QPixmapFilter::boundingRectFor(const QRectF &rect) const
  135. {
  136. return rect;
  137. }
  138. /*!
  139. \fn void QPixmapFilter::draw(QPainter *painter, const QPointF &p, const QPixmap &src, const QRectF& srcRect) const
  140. Uses \a painter to draw filtered result of \a src at the point
  141. specified by \a p. If \a srcRect is specified the it will
  142. be used as a source rectangle to only draw a part of the source.
  143. draw() will affect the area which boundingRectFor() returns.
  144. \internal
  145. */
  146. /*!
  147. \class QPixmapConvolutionFilter
  148. \since 4.5
  149. \ingroup painting
  150. \brief The QPixmapConvolutionFilter class provides convolution
  151. filtering for pixmaps.
  152. QPixmapConvolutionFilter implements a convolution pixmap filter,
  153. which is applied when \l{QPixmapFilter::}{draw()} is called. A
  154. convolution filter lets you distort an image by setting the values
  155. of a matrix of qreal values called its
  156. \l{setConvolutionKernel()}{kernel}. The matrix's values are
  157. usually between -1.0 and 1.0.
  158. \omit
  159. In convolution filtering, the pixel value is calculated from the
  160. neighboring pixels based on the weighting convolution kernel.
  161. This needs explaining to be useful.
  162. \endomit
  163. Example:
  164. \snippet doc/src/snippets/code/src_gui_image_qpixmapfilter.cpp 1
  165. \sa {Pixmap Filters Example}, QPixmapColorizeFilter, QPixmapDropShadowFilter
  166. \internal
  167. */
  168. class QPixmapConvolutionFilterPrivate : public QPixmapFilterPrivate
  169. {
  170. public:
  171. QPixmapConvolutionFilterPrivate(): convolutionKernel(0), kernelWidth(0), kernelHeight(0), convoluteAlpha(false) {}
  172. ~QPixmapConvolutionFilterPrivate() {
  173. delete[] convolutionKernel;
  174. }
  175. qreal *convolutionKernel;
  176. int kernelWidth;
  177. int kernelHeight;
  178. bool convoluteAlpha;
  179. };
  180. /*!
  181. Constructs a pixmap convolution filter.
  182. By default there is no convolution kernel.
  183. \internal
  184. */
  185. QPixmapConvolutionFilter::QPixmapConvolutionFilter(QObject *parent)
  186. : QPixmapFilter(*new QPixmapConvolutionFilterPrivate, ConvolutionFilter, parent)
  187. {
  188. Q_D(QPixmapConvolutionFilter);
  189. d->convoluteAlpha = true;
  190. }
  191. /*!
  192. Destructor of pixmap convolution filter.
  193. \internal
  194. */
  195. QPixmapConvolutionFilter::~QPixmapConvolutionFilter()
  196. {
  197. }
  198. /*!
  199. Sets convolution kernel with the given number of \a rows and \a columns.
  200. Values from \a kernel are copied to internal data structure.
  201. To preserve the intensity of the pixmap, the sum of all the
  202. values in the convolution kernel should add up to 1.0. A sum
  203. greater than 1.0 produces a lighter result and a sum less than 1.0
  204. produces a darker and transparent result.
  205. \internal
  206. */
  207. void QPixmapConvolutionFilter::setConvolutionKernel(const qreal *kernel, int rows, int columns)
  208. {
  209. Q_D(QPixmapConvolutionFilter);
  210. delete [] d->convolutionKernel;
  211. d->convolutionKernel = new qreal[rows * columns];
  212. memcpy(d->convolutionKernel, kernel, sizeof(qreal) * rows * columns);
  213. d->kernelWidth = columns;
  214. d->kernelHeight = rows;
  215. }
  216. /*!
  217. Gets the convolution kernel data.
  218. \internal
  219. */
  220. const qreal *QPixmapConvolutionFilter::convolutionKernel() const
  221. {
  222. Q_D(const QPixmapConvolutionFilter);
  223. return d->convolutionKernel;
  224. }
  225. /*!
  226. Gets the number of rows in the convolution kernel.
  227. \internal
  228. */
  229. int QPixmapConvolutionFilter::rows() const
  230. {
  231. Q_D(const QPixmapConvolutionFilter);
  232. return d->kernelHeight;
  233. }
  234. /*!
  235. Gets the number of columns in the convolution kernel.
  236. \internal
  237. */
  238. int QPixmapConvolutionFilter::columns() const
  239. {
  240. Q_D(const QPixmapConvolutionFilter);
  241. return d->kernelWidth;
  242. }
  243. /*!
  244. \internal
  245. */
  246. QRectF QPixmapConvolutionFilter::boundingRectFor(const QRectF &rect) const
  247. {
  248. Q_D(const QPixmapConvolutionFilter);
  249. return rect.adjusted(-d->kernelWidth / 2, -d->kernelHeight / 2, (d->kernelWidth - 1) / 2, (d->kernelHeight - 1) / 2);
  250. }
  251. // Convolutes the image
  252. static void convolute(
  253. QImage *destImage,
  254. const QPointF &pos,
  255. const QImage &srcImage,
  256. const QRectF &srcRect,
  257. QPainter::CompositionMode mode,
  258. qreal *kernel,
  259. int kernelWidth,
  260. int kernelHeight )
  261. {
  262. const QImage processImage = (srcImage.format() != QImage::Format_ARGB32_Premultiplied ) ? srcImage.convertToFormat(QImage::Format_ARGB32_Premultiplied) : srcImage;
  263. // TODO: support also other formats directly without copying
  264. int *fixedKernel = new int[kernelWidth*kernelHeight];
  265. for(int i = 0; i < kernelWidth*kernelHeight; i++)
  266. {
  267. fixedKernel[i] = (int)(65536 * kernel[i]);
  268. }
  269. QRectF trect = srcRect.isNull() ? processImage.rect() : srcRect;
  270. trect.moveTo(pos);
  271. QRectF bounded = trect.adjusted(-kernelWidth / 2, -kernelHeight / 2, (kernelWidth - 1) / 2, (kernelHeight - 1) / 2);
  272. QRect rect = bounded.toAlignedRect();
  273. QRect targetRect = rect.intersected(destImage->rect());
  274. QRectF srect = srcRect.isNull() ? processImage.rect() : srcRect;
  275. QRectF sbounded = srect.adjusted(-kernelWidth / 2, -kernelHeight / 2, (kernelWidth - 1) / 2, (kernelHeight - 1) / 2);
  276. QPoint srcStartPoint = sbounded.toAlignedRect().topLeft()+(targetRect.topLeft()-rect.topLeft());
  277. const uint *sourceStart = (uint*)processImage.scanLine(0);
  278. uint *outputStart = (uint*)destImage->scanLine(0);
  279. int yk = srcStartPoint.y();
  280. for (int y = targetRect.top(); y <= targetRect.bottom(); y++) {
  281. uint* output = outputStart + (destImage->bytesPerLine()/sizeof(uint))*y+targetRect.left();
  282. int xk = srcStartPoint.x();
  283. for(int x = targetRect.left(); x <= targetRect.right(); x++) {
  284. int r = 0;
  285. int g = 0;
  286. int b = 0;
  287. int a = 0;
  288. // some out of bounds pre-checking to avoid inner-loop ifs
  289. int kernely = -kernelHeight/2;
  290. int starty = 0;
  291. int endy = kernelHeight;
  292. if(yk+kernely+endy >= srcImage.height())
  293. endy = kernelHeight-((yk+kernely+endy)-srcImage.height())-1;
  294. if(yk+kernely < 0)
  295. starty = -(yk+kernely);
  296. int kernelx = -kernelWidth/2;
  297. int startx = 0;
  298. int endx = kernelWidth;
  299. if(xk+kernelx+endx >= srcImage.width())
  300. endx = kernelWidth-((xk+kernelx+endx)-srcImage.width())-1;
  301. if(xk+kernelx < 0)
  302. startx = -(xk+kernelx);
  303. for (int ys = starty; ys < endy; ys ++) {
  304. const uint *pix = sourceStart + (processImage.bytesPerLine()/sizeof(uint))*(yk+kernely+ys) + ((xk+kernelx+startx));
  305. const uint *endPix = pix+endx-startx;
  306. int kernelPos = ys*kernelWidth+startx;
  307. while (pix < endPix) {
  308. int factor = fixedKernel[kernelPos++];
  309. a += (((*pix) & 0xff000000)>>24) * factor;
  310. r += (((*pix) & 0x00ff0000)>>16) * factor;
  311. g += (((*pix) & 0x0000ff00)>>8 ) * factor;
  312. b += (((*pix) & 0x000000ff) ) * factor;
  313. pix++;
  314. }
  315. }
  316. r = qBound((int)0, r >> 16, (int)255);
  317. g = qBound((int)0, g >> 16, (int)255);
  318. b = qBound((int)0, b >> 16, (int)255);
  319. a = qBound((int)0, a >> 16, (int)255);
  320. // composition mode checking could be moved outside of loop
  321. if(mode == QPainter::CompositionMode_Source) {
  322. uint color = (a<<24)+(r<<16)+(g<<8)+b;
  323. *output++ = color;
  324. } else {
  325. uint current = *output;
  326. uchar ca = (current&0xff000000)>>24;
  327. uchar cr = (current&0x00ff0000)>>16;
  328. uchar cg = (current&0x0000ff00)>>8;
  329. uchar cb = (current&0x000000ff);
  330. uint color =
  331. (((ca*(255-a) >> 8)+a) << 24)+
  332. (((cr*(255-a) >> 8)+r) << 16)+
  333. (((cg*(255-a) >> 8)+g) << 8)+
  334. (((cb*(255-a) >> 8)+b));
  335. *output++ = color;;
  336. }
  337. xk++;
  338. }
  339. yk++;
  340. }
  341. delete[] fixedKernel;
  342. }
  343. /*!
  344. \internal
  345. */
  346. void QPixmapConvolutionFilter::draw(QPainter *painter, const QPointF &p, const QPixmap &src, const QRectF& srcRect) const
  347. {
  348. Q_D(const QPixmapConvolutionFilter);
  349. if (!painter->isActive())
  350. return;
  351. if(d->kernelWidth<=0 || d->kernelHeight <= 0)
  352. return;
  353. if (src.isNull())
  354. return;
  355. QPixmapFilter *filter = painter->paintEngine() && painter->paintEngine()->isExtended() ?
  356. static_cast<QPaintEngineEx *>(painter->paintEngine())->pixmapFilter(type(), this) : 0;
  357. QPixmapConvolutionFilter *convolutionFilter = static_cast<QPixmapConvolutionFilter*>(filter);
  358. if (convolutionFilter) {
  359. convolutionFilter->setConvolutionKernel(d->convolutionKernel, d->kernelWidth, d->kernelHeight);
  360. convolutionFilter->d_func()->convoluteAlpha = d->convoluteAlpha;
  361. convolutionFilter->draw(painter, p, src, srcRect);
  362. return;
  363. }
  364. // falling back to raster implementation
  365. QImage *target = 0;
  366. if (painter->paintEngine()->paintDevice()->devType() == QInternal::Image) {
  367. target = static_cast<QImage *>(painter->paintEngine()->paintDevice());
  368. QTransform mat = painter->combinedTransform();
  369. if (mat.type() > QTransform::TxTranslate) {
  370. // Disabled because of transformation...
  371. target = 0;
  372. } else {
  373. QRasterPaintEngine *pe = static_cast<QRasterPaintEngine *>(painter->paintEngine());
  374. if (pe->clipType() == QRasterPaintEngine::ComplexClip)
  375. // disabled because of complex clipping...
  376. target = 0;
  377. else {
  378. QRectF clip = pe->clipBoundingRect();
  379. QRectF rect = boundingRectFor(srcRect.isEmpty() ? src.rect() : srcRect);
  380. QTransform x = painter->deviceTransform();
  381. if (!clip.contains(rect.translated(x.dx() + p.x(), x.dy() + p.y()))) {
  382. target = 0;
  383. }
  384. }
  385. }
  386. }
  387. if (target) {
  388. QTransform x = painter->deviceTransform();
  389. QPointF offset(x.dx(), x.dy());
  390. convolute(target, p+offset, src.toImage(), srcRect, QPainter::CompositionMode_SourceOver, d->convolutionKernel, d->kernelWidth, d->kernelHeight);
  391. } else {
  392. QRect srect = srcRect.isNull() ? src.rect() : srcRect.toRect();
  393. QRect rect = boundingRectFor(srect).toRect();
  394. QImage result = QImage(rect.size(), QImage::Format_ARGB32_Premultiplied);
  395. QPoint offset = srect.topLeft() - rect.topLeft();
  396. convolute(&result,
  397. offset,
  398. src.toImage(),
  399. srect,
  400. QPainter::CompositionMode_Source,
  401. d->convolutionKernel,
  402. d->kernelWidth,
  403. d->kernelHeight);
  404. painter->drawImage(p - offset, result);
  405. }
  406. }
  407. /*!
  408. \class QPixmapBlurFilter
  409. \since 4.6
  410. \ingroup multimedia
  411. \brief The QPixmapBlurFilter class provides blur filtering
  412. for pixmaps.
  413. QPixmapBlurFilter implements a blur pixmap filter,
  414. which is applied when \l{QPixmapFilter::}{draw()} is called.
  415. The filter lets you specialize the radius of the blur as well
  416. as hints as to whether to prefer performance or quality.
  417. By default, the blur effect is produced by applying an exponential
  418. filter generated from the specified blurRadius(). Paint engines
  419. may override this with a custom blur that is faster on the
  420. underlying hardware.
  421. \sa {Pixmap Filters Example}, QPixmapConvolutionFilter, QPixmapDropShadowFilter
  422. \internal
  423. */
  424. class QPixmapBlurFilterPrivate : public QPixmapFilterPrivate
  425. {
  426. public:
  427. QPixmapBlurFilterPrivate() : radius(5), hints(QGraphicsBlurEffect::PerformanceHint) {}
  428. qreal radius;
  429. QGraphicsBlurEffect::BlurHints hints;
  430. };
  431. /*!
  432. Constructs a pixmap blur filter.
  433. \internal
  434. */
  435. QPixmapBlurFilter::QPixmapBlurFilter(QObject *parent)
  436. : QPixmapFilter(*new QPixmapBlurFilterPrivate, BlurFilter, parent)
  437. {
  438. }
  439. /*!
  440. Destructor of pixmap blur filter.
  441. \internal
  442. */
  443. QPixmapBlurFilter::~QPixmapBlurFilter()
  444. {
  445. }
  446. /*!
  447. Sets the radius of the blur filter. Higher radius produces increased blurriness.
  448. \internal
  449. */
  450. void QPixmapBlurFilter::setRadius(qreal radius)
  451. {
  452. Q_D(QPixmapBlurFilter);
  453. d->radius = radius;
  454. }
  455. /*!
  456. Gets the radius of the blur filter.
  457. \internal
  458. */
  459. qreal QPixmapBlurFilter::radius() const
  460. {
  461. Q_D(const QPixmapBlurFilter);
  462. return d->radius;
  463. }
  464. /*!
  465. Setting the blur hints to PerformanceHint causes the implementation
  466. to trade off visual quality to blur the image faster. Setting the
  467. blur hints to QualityHint causes the implementation to improve
  468. visual quality at the expense of speed.
  469. AnimationHint causes the implementation to optimize for animating
  470. the blur radius, possibly by caching blurred versions of the source
  471. pixmap.
  472. The implementation is free to ignore this value if it only has a single
  473. blur algorithm.
  474. \internal
  475. */
  476. void QPixmapBlurFilter::setBlurHints(QGraphicsBlurEffect::BlurHints hints)
  477. {
  478. Q_D(QPixmapBlurFilter);
  479. d->hints = hints;
  480. }
  481. /*!
  482. Gets the blur hints of the blur filter.
  483. \internal
  484. */
  485. QGraphicsBlurEffect::BlurHints QPixmapBlurFilter::blurHints() const
  486. {
  487. Q_D(const QPixmapBlurFilter);
  488. return d->hints;
  489. }
  490. const qreal radiusScale = qreal(2.5);
  491. /*!
  492. \internal
  493. */
  494. QRectF QPixmapBlurFilter::boundingRectFor(const QRectF &rect) const
  495. {
  496. Q_D(const QPixmapBlurFilter);
  497. const qreal delta = radiusScale * d->radius + 1;
  498. return rect.adjusted(-delta, -delta, delta, delta);
  499. }
  500. template <int shift>
  501. inline int qt_static_shift(int value)
  502. {
  503. if (shift == 0)
  504. return value;
  505. else if (shift > 0)
  506. return value << (uint(shift) & 0x1f);
  507. else
  508. return value >> (uint(-shift) & 0x1f);
  509. }
  510. template<int aprec, int zprec>
  511. inline void qt_blurinner(uchar *bptr, int &zR, int &zG, int &zB, int &zA, int alpha)
  512. {
  513. QRgb *pixel = (QRgb *)bptr;
  514. #define Z_MASK (0xff << zprec)
  515. const int A_zprec = qt_static_shift<zprec - 24>(*pixel) & Z_MASK;
  516. const int R_zprec = qt_static_shift<zprec - 16>(*pixel) & Z_MASK;
  517. const int G_zprec = qt_static_shift<zprec - 8>(*pixel) & Z_MASK;
  518. const int B_zprec = qt_static_shift<zprec>(*pixel) & Z_MASK;
  519. #undef Z_MASK
  520. const int zR_zprec = zR >> aprec;
  521. const int zG_zprec = zG >> aprec;
  522. const int zB_zprec = zB >> aprec;
  523. const int zA_zprec = zA >> aprec;
  524. zR += alpha * (R_zprec - zR_zprec);
  525. zG += alpha * (G_zprec - zG_zprec);
  526. zB += alpha * (B_zprec - zB_zprec);
  527. zA += alpha * (A_zprec - zA_zprec);
  528. #define ZA_MASK (0xff << (zprec + aprec))
  529. *pixel =
  530. qt_static_shift<24 - zprec - aprec>(zA & ZA_MASK)
  531. | qt_static_shift<16 - zprec - aprec>(zR & ZA_MASK)
  532. | qt_static_shift<8 - zprec - aprec>(zG & ZA_MASK)
  533. | qt_static_shift<-zprec - aprec>(zB & ZA_MASK);
  534. #undef ZA_MASK
  535. }
  536. const int alphaIndex = (QSysInfo::ByteOrder == QSysInfo::BigEndian ? 0 : 3);
  537. template<int aprec, int zprec>
  538. inline void qt_blurinner_alphaOnly(uchar *bptr, int &z, int alpha)
  539. {
  540. const int A_zprec = int(*(bptr)) << zprec;
  541. const int z_zprec = z >> aprec;
  542. z += alpha * (A_zprec - z_zprec);
  543. *(bptr) = z >> (zprec + aprec);
  544. }
  545. template<int aprec, int zprec, bool alphaOnly>
  546. inline void qt_blurrow(QImage & im, int line, int alpha)
  547. {
  548. uchar *bptr = im.scanLine(line);
  549. int zR = 0, zG = 0, zB = 0, zA = 0;
  550. if (alphaOnly && im.format() != QImage::Format_Indexed8)
  551. bptr += alphaIndex;
  552. const int stride = im.depth() >> 3;
  553. const int im_width = im.width();
  554. for (int index = 0; index < im_width; ++index) {
  555. if (alphaOnly)
  556. qt_blurinner_alphaOnly<aprec, zprec>(bptr, zA, alpha);
  557. else
  558. qt_blurinner<aprec, zprec>(bptr, zR, zG, zB, zA, alpha);
  559. bptr += stride;
  560. }
  561. bptr -= stride;
  562. for (int index = im_width - 2; index >= 0; --index) {
  563. bptr -= stride;
  564. if (alphaOnly)
  565. qt_blurinner_alphaOnly<aprec, zprec>(bptr, zA, alpha);
  566. else
  567. qt_blurinner<aprec, zprec>(bptr, zR, zG, zB, zA, alpha);
  568. }
  569. }
  570. /*
  571. * expblur(QImage &img, int radius)
  572. *
  573. * Based on exponential blur algorithm by Jani Huhtanen
  574. *
  575. * In-place blur of image 'img' with kernel
  576. * of approximate radius 'radius'.
  577. *
  578. * Blurs with two sided exponential impulse
  579. * response.
  580. *
  581. * aprec = precision of alpha parameter
  582. * in fixed-point format 0.aprec
  583. *
  584. * zprec = precision of state parameters
  585. * zR,zG,zB and zA in fp format 8.zprec
  586. */
  587. template <int aprec, int zprec, bool alphaOnly>
  588. void expblur(QImage &img, qreal radius, bool improvedQuality = false, int transposed = 0)
  589. {
  590. // halve the radius if we're using two passes
  591. if (improvedQuality)
  592. radius *= qreal(0.5);
  593. Q_ASSERT(img.format() == QImage::Format_ARGB32_Premultiplied
  594. || img.format() == QImage::Format_RGB32
  595. || img.format() == QImage::Format_Indexed8);
  596. // choose the alpha such that pixels at radius distance from a fully
  597. // saturated pixel will have an alpha component of no greater than
  598. // the cutOffIntensity
  599. const qreal cutOffIntensity = 2;
  600. int alpha = radius <= qreal(1e-5)
  601. ? ((1 << aprec)-1)
  602. : qRound((1<<aprec)*(1 - qPow(cutOffIntensity * (1 / qreal(255)), 1 / radius)));
  603. int img_height = img.height();
  604. for (int row = 0; row < img_height; ++row) {
  605. for (int i = 0; i <= int(improvedQuality); ++i)
  606. qt_blurrow<aprec, zprec, alphaOnly>(img, row, alpha);
  607. }
  608. QImage temp(img.height(), img.width(), img.format());
  609. if (transposed >= 0) {
  610. if (img.depth() == 8) {
  611. qt_memrotate270(reinterpret_cast<const quint8*>(img.bits()),
  612. img.width(), img.height(), img.bytesPerLine(),
  613. reinterpret_cast<quint8*>(temp.bits()),
  614. temp.bytesPerLine());
  615. } else {
  616. qt_memrotate270(reinterpret_cast<const quint32*>(img.bits()),
  617. img.width(), img.height(), img.bytesPerLine(),
  618. reinterpret_cast<quint32*>(temp.bits()),
  619. temp.bytesPerLine());
  620. }
  621. } else {
  622. if (img.depth() == 8) {
  623. qt_memrotate90(reinterpret_cast<const quint8*>(img.bits()),
  624. img.width(), img.height(), img.bytesPerLine(),
  625. reinterpret_cast<quint8*>(temp.bits()),
  626. temp.bytesPerLine());
  627. } else {
  628. qt_memrotate90(reinterpret_cast<const quint32*>(img.bits()),
  629. img.width(), img.height(), img.bytesPerLine(),
  630. reinterpret_cast<quint32*>(temp.bits()),
  631. temp.bytesPerLine());
  632. }
  633. }
  634. img_height = temp.height();
  635. for (int row = 0; row < img_height; ++row) {
  636. for (int i = 0; i <= int(improvedQuality); ++i)
  637. qt_blurrow<aprec, zprec, alphaOnly>(temp, row, alpha);
  638. }
  639. if (transposed == 0) {
  640. if (img.depth() == 8) {
  641. qt_memrotate90(reinterpret_cast<const quint8*>(temp.bits()),
  642. temp.width(), temp.height(), temp.bytesPerLine(),
  643. reinterpret_cast<quint8*>(img.bits()),
  644. img.bytesPerLine());
  645. } else {
  646. qt_memrotate90(reinterpret_cast<const quint32*>(temp.bits()),
  647. temp.width(), temp.height(), temp.bytesPerLine(),
  648. reinterpret_cast<quint32*>(img.bits()),
  649. img.bytesPerLine());
  650. }
  651. } else {
  652. img = temp;
  653. }
  654. }
  655. #define AVG(a,b) ( ((((a)^(b)) & 0xfefefefeUL) >> 1) + ((a)&(b)) )
  656. #define AVG16(a,b) ( ((((a)^(b)) & 0xf7deUL) >> 1) + ((a)&(b)) )
  657. Q_GUI_EXPORT QImage qt_halfScaled(const QImage &source)
  658. {
  659. if (source.width() < 2 || source.height() < 2)
  660. return QImage();
  661. QImage srcImage = source;
  662. if (source.format() == QImage::Format_Indexed8) {
  663. // assumes grayscale
  664. QImage dest(source.width() / 2, source.height() / 2, srcImage.format());
  665. const uchar *src = reinterpret_cast<const uchar*>(const_cast<const QImage &>(srcImage).bits());
  666. int sx = srcImage.bytesPerLine();
  667. int sx2 = sx << 1;
  668. uchar *dst = reinterpret_cast<uchar*>(dest.bits());
  669. int dx = dest.bytesPerLine();
  670. int ww = dest.width();
  671. int hh = dest.height();
  672. for (int y = hh; y; --y, dst += dx, src += sx2) {
  673. const uchar *p1 = src;
  674. const uchar *p2 = src + sx;
  675. uchar *q = dst;
  676. for (int x = ww; x; --x, ++q, p1 += 2, p2 += 2)
  677. *q = ((int(p1[0]) + int(p1[1]) + int(p2[0]) + int(p2[1])) + 2) >> 2;
  678. }
  679. return dest;
  680. } else if (source.format() == QImage::Format_ARGB8565_Premultiplied) {
  681. QImage dest(source.width() / 2, source.height() / 2, srcImage.format());
  682. const uchar *src = reinterpret_cast<const uchar*>(const_cast<const QImage &>(srcImage).bits());
  683. int sx = srcImage.bytesPerLine();
  684. int sx2 = sx << 1;
  685. uchar *dst = reinterpret_cast<uchar*>(dest.bits());
  686. int dx = dest.bytesPerLine();
  687. int ww = dest.width();
  688. int hh = dest.height();
  689. for (int y = hh; y; --y, dst += dx, src += sx2) {
  690. const uchar *p1 = src;
  691. const uchar *p2 = src + sx;
  692. uchar *q = dst;
  693. for (int x = ww; x; --x, q += 3, p1 += 6, p2 += 6) {
  694. // alpha
  695. q[0] = AVG(AVG(p1[0], p1[3]), AVG(p2[0], p2[3]));
  696. // rgb
  697. const quint16 p16_1 = (p1[2] << 8) | p1[1];
  698. const quint16 p16_2 = (p1[5] << 8) | p1[4];
  699. const quint16 p16_3 = (p2[2] << 8) | p2[1];
  700. const quint16 p16_4 = (p2[5] << 8) | p2[4];
  701. const quint16 result = AVG16(AVG16(p16_1, p16_2), AVG16(p16_3, p16_4));
  702. q[1] = result & 0xff;
  703. q[2] = result >> 8;
  704. }
  705. }
  706. return dest;
  707. } else if (source.format() != QImage::Format_ARGB32_Premultiplied
  708. && source.format() != QImage::Format_RGB32)
  709. {
  710. srcImage = source.convertToFormat(QImage::Format_ARGB32_Premultiplied);
  711. }
  712. QImage dest(source.width() / 2, source.height() / 2, srcImage.format());
  713. const quint32 *src = reinterpret_cast<const quint32*>(const_cast<const QImage &>(srcImage).bits());
  714. int sx = srcImage.bytesPerLine() >> 2;
  715. int sx2 = sx << 1;
  716. quint32 *dst = reinterpret_cast<quint32*>(dest.bits());
  717. int dx = dest.bytesPerLine() >> 2;
  718. int ww = dest.width();
  719. int hh = dest.height();
  720. for (int y = hh; y; --y, dst += dx, src += sx2) {
  721. const quint32 *p1 = src;
  722. const quint32 *p2 = src + sx;
  723. quint32 *q = dst;
  724. for (int x = ww; x; --x, q++, p1 += 2, p2 += 2)
  725. *q = AVG(AVG(p1[0], p1[1]), AVG(p2[0], p2[1]));
  726. }
  727. return dest;
  728. }
  729. Q_GUI_EXPORT void qt_blurImage(QPainter *p, QImage &blurImage, qreal radius, bool quality, bool alphaOnly, int transposed = 0)
  730. {
  731. if (blurImage.format() != QImage::Format_ARGB32_Premultiplied
  732. && blurImage.format() != QImage::Format_RGB32)
  733. {
  734. blurImage = blurImage.convertToFormat(QImage::Format_ARGB32_Premultiplied);
  735. }
  736. qreal scale = 1;
  737. if (radius >= 4 && blurImage.width() >= 2 && blurImage.height() >= 2) {
  738. blurImage = qt_halfScaled(blurImage);
  739. scale = 2;
  740. radius *= qreal(0.5);
  741. }
  742. if (alphaOnly)
  743. expblur<12, 10, true>(blurImage, radius, quality, transposed);
  744. else
  745. expblur<12, 10, false>(blurImage, radius, quality, transposed);
  746. if (p) {
  747. p->scale(scale, scale);
  748. p->setRenderHint(QPainter::SmoothPixmapTransform);
  749. p->drawImage(QRect(0, 0, blurImage.width(), blurImage.height()), blurImage);
  750. }
  751. }
  752. Q_GUI_EXPORT void qt_blurImage(QImage &blurImage, qreal radius, bool quality, int transposed = 0)
  753. {
  754. if (blurImage.format() == QImage::Format_Indexed8)
  755. expblur<12, 10, true>(blurImage, radius, quality, transposed);
  756. else
  757. expblur<12, 10, false>(blurImage, radius, quality, transposed);
  758. }
  759. Q_GUI_EXPORT bool qt_scaleForTransform(const QTransform &transform, qreal *scale);
  760. /*!
  761. \internal
  762. */
  763. void QPixmapBlurFilter::draw(QPainter *painter, const QPointF &p, const QPixmap &src, const QRectF &rect) const
  764. {
  765. Q_D(const QPixmapBlurFilter);
  766. if (!painter->isActive())
  767. return;
  768. if (src.isNull())
  769. return;
  770. QRectF srcRect = rect;
  771. if (srcRect.isNull())
  772. srcRect = src.rect();
  773. if (d->radius <= 1) {
  774. painter->drawPixmap(srcRect.translated(p), src, srcRect);
  775. return;
  776. }
  777. qreal scaledRadius = radiusScale * d->radius;
  778. qreal scale;
  779. if (qt_scaleForTransform(painter->transform(), &scale))
  780. scaledRadius /= scale;
  781. QPixmapFilter *filter = painter->paintEngine() && painter->paintEngine()->isExtended() ?
  782. static_cast<QPaintEngineEx *>(painter->paintEngine())->pixmapFilter(type(), this) : 0;
  783. QPixmapBlurFilter *blurFilter = static_cast<QPixmapBlurFilter*>(filter);
  784. if (blurFilter) {
  785. blurFilter->setRadius(scaledRadius);
  786. blurFilter->setBlurHints(d->hints);
  787. blurFilter->draw(painter, p, src, srcRect);
  788. return;
  789. }
  790. QImage srcImage;
  791. QImage destImage;
  792. if (srcRect == src.rect()) {
  793. srcImage = src.toImage();
  794. } else {
  795. QRect rect = srcRect.toAlignedRect().intersected(src.rect());
  796. srcImage = src.copy(rect).toImage();
  797. }
  798. QTransform transform = painter->worldTransform();
  799. painter->translate(p);
  800. qt_blurImage(painter, srcImage, scaledRadius, (d->hints & QGraphicsBlurEffect::QualityHint), false);
  801. painter->setWorldTransform(transform);
  802. }
  803. // grayscales the image to dest (could be same). If rect isn't defined
  804. // destination image size is used to determine the dimension of grayscaling
  805. // process.
  806. static void grayscale(const QImage &image, QImage &dest, const QRect& rect = QRect())
  807. {
  808. QRect destRect = rect;
  809. QRect srcRect = rect;
  810. if (rect.isNull()) {
  811. srcRect = dest.rect();
  812. destRect = dest.rect();
  813. }
  814. if (&image != &dest) {
  815. destRect.moveTo(QPoint(0, 0));
  816. }
  817. unsigned int *data = (unsigned int *)image.bits();
  818. unsigned int *outData = (unsigned int *)dest.bits();
  819. if (dest.size() == image.size() && image.rect() == srcRect) {
  820. // a bit faster loop for grayscaling everything
  821. int pixels = dest.width() * dest.height();
  822. for (int i = 0; i < pixels; ++i) {
  823. int val = qGray(data[i]);
  824. outData[i] = qRgba(val, val, val, qAlpha(data[i]));
  825. }
  826. } else {
  827. int yd = destRect.top();
  828. for (int y = srcRect.top(); y <= srcRect.bottom() && y < image.height(); y++) {
  829. data = (unsigned int*)image.scanLine(y);
  830. outData = (unsigned int*)dest.scanLine(yd++);
  831. int xd = destRect.left();
  832. for (int x = srcRect.left(); x <= srcRect.right() && x < image.width(); x++) {
  833. int val = qGray(data[x]);
  834. outData[xd++] = qRgba(val, val, val, qAlpha(data[x]));
  835. }
  836. }
  837. }
  838. }
  839. /*!
  840. \class QPixmapColorizeFilter
  841. \since 4.5
  842. \ingroup painting
  843. \brief The QPixmapColorizeFilter class provides colorizing
  844. filtering for pixmaps.
  845. A colorize filter gives the pixmap a tint of its color(). The
  846. filter first grayscales the pixmap and then converts those to
  847. colorized values using QPainter::CompositionMode_Screen with the
  848. chosen color. The alpha-channel is not changed.
  849. Example:
  850. \snippet doc/src/snippets/code/src_gui_image_qpixmapfilter.cpp 0
  851. \sa QPainter::CompositionMode
  852. \internal
  853. */
  854. class QPixmapColorizeFilterPrivate : public QPixmapFilterPrivate
  855. {
  856. Q_DECLARE_PUBLIC(QPixmapColorizeFilter)
  857. public:
  858. QColor color;
  859. qreal strength;
  860. quint32 opaque : 1;
  861. quint32 alphaBlend : 1;
  862. quint32 padding : 30;
  863. };
  864. /*!
  865. Constructs an pixmap colorize filter.
  866. Default color value for colorizing is QColor(0, 0, 192).
  867. \internal
  868. */
  869. QPixmapColorizeFilter::QPixmapColorizeFilter(QObject *parent)
  870. : QPixmapFilter(*new QPixmapColorizeFilterPrivate, ColorizeFilter, parent)
  871. {
  872. Q_D(QPixmapColorizeFilter);
  873. d->color = QColor(0, 0, 192);
  874. d->strength = qreal(1);
  875. d->opaque = true;
  876. d->alphaBlend = false;
  877. }
  878. /*!
  879. Gets the color of the colorize filter.
  880. \internal
  881. */
  882. QColor QPixmapColorizeFilter::color() const
  883. {
  884. Q_D(const QPixmapColorizeFilter);
  885. return d->color;
  886. }
  887. /*!
  888. Sets the color of the colorize filter to the \a color specified.
  889. \internal
  890. */
  891. void QPixmapColorizeFilter::setColor(const QColor &color)
  892. {
  893. Q_D(QPixmapColorizeFilter);
  894. d->color = color;
  895. }
  896. /*!
  897. Gets the strength of the colorize filter, 1.0 means full colorized while
  898. 0.0 equals to no filtering at all.
  899. \internal
  900. */
  901. qreal QPixmapColorizeFilter::strength() const
  902. {
  903. Q_D(const QPixmapColorizeFilter);
  904. return d->strength;
  905. }
  906. /*!
  907. Sets the strength of the colorize filter to \a strength.
  908. \internal
  909. */
  910. void QPixmapColorizeFilter::setStrength(qreal strength)
  911. {
  912. Q_D(QPixmapColorizeFilter);
  913. d->strength = qBound(qreal(0), strength, qreal(1));
  914. d->opaque = !qFuzzyIsNull(d->strength);
  915. d->alphaBlend = !qFuzzyIsNull(d->strength - 1);
  916. }
  917. /*!
  918. \internal
  919. */
  920. void QPixmapColorizeFilter::draw(QPainter *painter, const QPointF &dest, const QPixmap &src, const QRectF &srcRect) const
  921. {
  922. Q_D(const QPixmapColorizeFilter);
  923. if (src.isNull())
  924. return;
  925. QPixmapFilter *filter = painter->paintEngine() && painter->paintEngine()->isExtended() ?
  926. static_cast<QPaintEngineEx *>(painter->paintEngine())->pixmapFilter(type(), this) : 0;
  927. QPixmapColorizeFilter *colorizeFilter = static_cast<QPixmapColorizeFilter*>(filter);
  928. if (colorizeFilter) {
  929. colorizeFilter->setColor(d->color);
  930. colorizeFilter->setStrength(d->strength);
  931. colorizeFilter->draw(painter, dest, src, srcRect);
  932. return;
  933. }
  934. // falling back to raster implementation
  935. if (!d->opaque) {
  936. painter->drawPixmap(dest, src, srcRect);
  937. return;
  938. }
  939. QImage srcImage;
  940. QImage destImage;
  941. if (srcRect.isNull()) {
  942. srcImage = src.toImage();
  943. srcImage = srcImage.convertToFormat(srcImage.hasAlphaChannel() ? QImage::Format_ARGB32_Premultiplied : QImage::Format_RGB32);
  944. destImage = QImage(srcImage.size(), srcImage.format());
  945. } else {
  946. QRect rect = srcRect.toAlignedRect().intersected(src.rect());
  947. srcImage = src.copy(rect).toImage();
  948. srcImage = srcImage.convertToFormat(srcImage.hasAlphaChannel() ? QImage::Format_ARGB32_Premultiplied : QImage::Format_RGB32);
  949. destImage = QImage(rect.size(), srcImage.format());
  950. }
  951. // do colorizing
  952. QPainter destPainter(&destImage);
  953. grayscale(srcImage, destImage, srcImage.rect());
  954. destPainter.setCompositionMode(QPainter::CompositionMode_Screen);
  955. destPainter.fillRect(srcImage.rect(), d->color);
  956. destPainter.end();
  957. if (d->alphaBlend) {
  958. // alpha blending srcImage and destImage
  959. QImage buffer = srcImage;
  960. QPainter bufPainter(&buffer);
  961. bufPainter.setOpacity(d->strength);
  962. bufPainter.drawImage(0, 0, destImage);
  963. bufPainter.end();
  964. destImage = buffer;
  965. }
  966. if (srcImage.hasAlphaChannel())
  967. destImage.setAlphaChannel(srcImage.alphaChannel());
  968. painter->drawImage(dest, destImage);
  969. }
  970. class QPixmapDropShadowFilterPrivate : public QPixmapFilterPrivate
  971. {
  972. public:
  973. QPixmapDropShadowFilterPrivate()
  974. : offset(8, 8), color(63, 63, 63, 180), radius(1) {}
  975. QPointF offset;
  976. QColor color;
  977. qreal radius;
  978. };
  979. /*!
  980. \class QPixmapDropShadowFilter
  981. \since 4.5
  982. \ingroup painting
  983. \brief The QPixmapDropShadowFilter class is a convenience class
  984. for drawing pixmaps with drop shadows.
  985. The drop shadow is produced by taking a copy of the source pixmap
  986. and applying a color to the copy using a
  987. QPainter::CompositionMode_DestinationIn operation. This produces a
  988. homogeneously-colored pixmap which is then drawn using a
  989. QPixmapConvolutionFilter at an offset. The original pixmap is
  990. drawn on top.
  991. The QPixmapDropShadowFilter class provides some customization
  992. options to specify how the drop shadow should appear. The color of
  993. the drop shadow can be modified using the setColor() function, the
  994. drop shadow offset can be modified using the setOffset() function,
  995. and the blur radius of the drop shadow can be changed through the
  996. setBlurRadius() function.
  997. By default, the drop shadow is a dark gray shadow, blurred with a
  998. radius of 1 at an offset of 8 pixels towards the lower right.
  999. Example:
  1000. \snippet doc/src/snippets/code/src_gui_image_qpixmapfilter.cpp 2
  1001. \sa QPixmapColorizeFilter, QPixmapConvolutionFilter
  1002. \internal
  1003. */
  1004. /*!
  1005. Constructs drop shadow filter.
  1006. \internal
  1007. */
  1008. QPixmapDropShadowFilter::QPixmapDropShadowFilter(QObject *parent)
  1009. : QPixmapFilter(*new QPixmapDropShadowFilterPrivate, DropShadowFilter, parent)
  1010. {
  1011. }
  1012. /*!
  1013. Destroys drop shadow filter.
  1014. \internal
  1015. */
  1016. QPixmapDropShadowFilter::~QPixmapDropShadowFilter()
  1017. {
  1018. }
  1019. /*!
  1020. Returns the radius in pixels of the blur on the drop shadow.
  1021. A smaller radius results in a sharper shadow.
  1022. \sa color(), offset()
  1023. \internal
  1024. */
  1025. qreal QPixmapDropShadowFilter::blurRadius() const
  1026. {
  1027. Q_D(const QPixmapDropShadowFilter);
  1028. return d->radius;
  1029. }
  1030. /*!
  1031. Sets the radius in pixels of the blur on the drop shadow to the \a radius specified.
  1032. Using a smaller radius results in a sharper shadow.
  1033. \sa setColor(), setOffset()
  1034. \internal
  1035. */
  1036. void QPixmapDropShadowFilter::setBlurRadius(qreal radius)
  1037. {
  1038. Q_D(QPixmapDropShadowFilter);
  1039. d->radius = radius;
  1040. }
  1041. /*!
  1042. Returns the color of the drop shadow.
  1043. \sa blurRadius(), offset()
  1044. \internal
  1045. */
  1046. QColor QPixmapDropShadowFilter::color() const
  1047. {
  1048. Q_D(const QPixmapDropShadowFilter);
  1049. return d->color;
  1050. }
  1051. /*!
  1052. Sets the color of the drop shadow to the \a color specified.
  1053. \sa setBlurRadius(), setOffset()
  1054. \internal
  1055. */
  1056. void QPixmapDropShadowFilter::setColor(const QColor &color)
  1057. {
  1058. Q_D(QPixmapDropShadowFilter);
  1059. d->color = color;
  1060. }
  1061. /*!
  1062. Returns the shadow offset in pixels.
  1063. \sa blurRadius(), color()
  1064. \internal
  1065. */
  1066. QPointF QPixmapDropShadowFilter::offset() const
  1067. {
  1068. Q_D(const QPixmapDropShadowFilter);
  1069. return d->offset;
  1070. }
  1071. /*!
  1072. Sets the shadow offset in pixels to the \a offset specified.
  1073. \sa setBlurRadius(), setColor()
  1074. \internal
  1075. */
  1076. void QPixmapDropShadowFilter::setOffset(const QPointF &offset)
  1077. {
  1078. Q_D(QPixmapDropShadowFilter);
  1079. d->offset = offset;
  1080. }
  1081. /*!
  1082. \fn void QPixmapDropShadowFilter::setOffset(qreal dx, qreal dy)
  1083. \overload
  1084. Sets the shadow offset in pixels to be the displacement specified by the
  1085. horizontal \a dx and vertical \a dy coordinates.
  1086. \sa setBlurRadius(), setColor()
  1087. \internal
  1088. */
  1089. /*!
  1090. \internal
  1091. */
  1092. QRectF QPixmapDropShadowFilter::boundingRectFor(const QRectF &rect) const
  1093. {
  1094. Q_D(const QPixmapDropShadowFilter);
  1095. return rect.united(rect.translated(d->offset).adjusted(-d->radius, -d->radius, d->radius, d->radius));
  1096. }
  1097. /*!
  1098. \internal
  1099. */
  1100. void QPixmapDropShadowFilter::draw(QPainter *p,
  1101. const QPointF &pos,
  1102. const QPixmap &px,
  1103. const QRectF &src) const
  1104. {
  1105. Q_D(const QPixmapDropShadowFilter);
  1106. if (px.isNull())
  1107. return;
  1108. QPixmapFilter *filter = p->paintEngine() && p->paintEngine()->isExtended() ?
  1109. static_cast<QPaintEngineEx *>(p->paintEngine())->pixmapFilter(type(), this) : 0;
  1110. QPixmapDropShadowFilter *dropShadowFilter = static_cast<QPixmapDropShadowFilter*>(filter);
  1111. if (dropShadowFilter) {
  1112. dropShadowFilter->setColor(d->color);
  1113. dropShadowFilter->setBlurRadius(d->radius);
  1114. dropShadowFilter->setOffset(d->offset);
  1115. dropShadowFilter->draw(p, pos, px, src);
  1116. return;
  1117. }
  1118. QImage tmp(px.size(), QImage::Format_ARGB32_Premultiplied);
  1119. tmp.fill(0);
  1120. QPainter tmpPainter(&tmp);
  1121. tmpPainter.setCompositionMode(QPainter::CompositionMode_Source);
  1122. tmpPainter.drawPixmap(d->offset, px);
  1123. tmpPainter.end();
  1124. // blur the alpha channel
  1125. QImage blurred(tmp.size(), QImage::Format_ARGB32_Premultiplied);
  1126. blurred.fill(0);
  1127. QPainter blurPainter(&blurred);
  1128. qt_blurImage(&blurPainter, tmp, d->radius, false, true);
  1129. blurPainter.end();
  1130. // blacken the image...
  1131. QPainter blackenPainter(&blurred);
  1132. blackenPainter.setCompositionMode(QPainter::CompositionMode_SourceIn);
  1133. blackenPainter.fillRect(blurred.rect(), d->color);
  1134. blackenPainter.end();
  1135. // draw the blurred drop shadow...
  1136. p->drawImage(pos, blurred);
  1137. // Draw the actual pixmap...
  1138. p->drawPixmap(pos, px, src);
  1139. }
  1140. QT_END_NAMESPACE
  1141. #endif //QT_NO_GRAPHICSEFFECT