#ifndef BLUR_CPP #define BLUR_CPP /* * Copyright 2007 Jani Huhtanen <jani.huhtanen@tut.fi> * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU Library General Public License version 2 as * published by the Free Software Foundation * * 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 Library General Public * License along with this program; if not, write to the * Free Software Foundation, Inc., * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. */ #include <cmath> // Exponential blur, Jani Huhtanen, 2006 // template<int aprec, int zprec> static inline void blurinner(unsigned char *bptr, int &zR, int &zG, int &zB, int &zA, int alpha); template<int aprec,int zprec> static inline void blurrow(QImage &im, int line, int alpha); template<int aprec, int zprec> static inline void blurcol(QImage &im, int col, int alpha); /* * expblur(QImage &img, int radius) * * In-place blur of image 'img' with kernel * of approximate radius 'radius'. * * Blurs with two sided exponential impulse * response. * * aprec = precision of alpha parameter * in fixed-point format 0.aprec * * zprec = precision of state parameters * zR,zG,zB and zA in fp format 8.zprec */ template<int aprec,int zprec> void expblur(QImage &img, int radius) { if (radius < 1) { return; } img = img.convertToFormat(QImage::Format_ARGB32_Premultiplied); /* Calculate the alpha such that 90% of the kernel is within the radius. (Kernel extends to infinity) */ int alpha = (int)((1 << aprec) * (1.0f - std::exp(-2.3f / (radius + 1.f)))); int height = img.height(); int width = img.width(); for (int row=0; row<height; row++) { blurrow<aprec,zprec>(img, row, alpha); } for (int col=0; col<width; col++) { blurcol<aprec,zprec>(img, col, alpha); } return; } template<int aprec, int zprec> static inline void blurinner(unsigned char *bptr, int &zR, int &zG, int &zB, int &zA, int alpha) { int R, G, B, A; R = *bptr; G = *(bptr + 1); B = *(bptr + 2); A = *(bptr + 3); zR += (alpha * ((R << zprec) - zR)) >> aprec; zG += (alpha * ((G << zprec) - zG)) >> aprec; zB += (alpha * ((B << zprec) - zB)) >> aprec; zA += (alpha * ((A << zprec) - zA)) >> aprec; *bptr = zR >> zprec; *(bptr+1) = zG >> zprec; *(bptr+2) = zB >> zprec; *(bptr+3) = zA >> zprec; } template<int aprec,int zprec> static inline void blurrow(QImage &im, int line, int alpha) { int zR, zG, zB, zA; QRgb *ptr = (QRgb *)im.scanLine(line); int width = im.width(); zR = *((unsigned char *)ptr ) << zprec; zG = *((unsigned char *)ptr + 1) << zprec; zB = *((unsigned char *)ptr + 2) << zprec; zA = *((unsigned char *)ptr + 3) << zprec; for (int index=1; index<width; index++) { blurinner<aprec,zprec>((unsigned char *)&ptr[index],zR,zG,zB,zA,alpha); } for (int index=width-2; index>=0; index--) { blurinner<aprec,zprec>((unsigned char *)&ptr[index],zR,zG,zB,zA,alpha); } } template<int aprec, int zprec> static inline void blurcol(QImage &im, int col, int alpha) { int zR, zG, zB, zA; QRgb *ptr = (QRgb *)im.bits(); ptr += col; int height = im.height(); int width = im.width(); zR = *((unsigned char *)ptr ) << zprec; zG = *((unsigned char *)ptr + 1) << zprec; zB = *((unsigned char *)ptr + 2) << zprec; zA = *((unsigned char *)ptr + 3) << zprec; for (int index=width; index<(height-1)*width; index+=width) { blurinner<aprec,zprec>((unsigned char *)&ptr[index], zR, zG, zB, zA, alpha); } for (int index=(height-2)*width; index>=0; index-=width) { blurinner<aprec,zprec>((unsigned char *)&ptr[index], zR, zG, zB, zA, alpha); } } template<class T> inline const T &qClamp(const T &x, const T &low, const T &high) { if (x < low) { return low; } else if (x > high) { return high; } else { return x; } } #endif