[Interest] QImage + pixel lambdas
Jason H
jhihn at gmx.com
Fri Jun 1 22:36:45 CEST 2018
channel ARGB32 6000000 pixels 87/88/93 msec (lambda, scanline)
channel ARGB32 6000000 pixels 148/146/153 msec QImage::pixel
> Sent: Friday, June 01, 2018 at 4:13 PM
> From: "Jason H" <jhihn at gmx.com>
> To: "Jason H" <jhihn at gmx.com>
> Cc: "interestqt-project.org" <interest at qt-project.org>
> Subject: Re: [Interest] QImage + pixel lambdas
>
> Incase anyone is wondering:
> accumulator ARGB32 6000000 pixels 24/25/26 msec (lamda, scanline)
> accumulator ARGB32 6000000 pixels 40/42/44 msec (lambda or no lambda, QImage::pixel)
>
>
> > Sent: Friday, June 01, 2018 at 9:38 AM
> > From: "Jason H" <jhihn at gmx.com>
> > To: "interestqt-project.org" <interest at qt-project.org>
> > Subject: [Interest] QImage + pixel lambdas
> >
> > So I'm doing some qimage analysis/manipulation. I think it would be really good if QImage came with a per-pixel function. (Examples below)
> > I'm tired of writing the x/y for loops. The really neat thing (not in the examples) is using QImage::scanLine(int y) for faster access, and automatically choosing the value of p in the lambda based on bit depth.
> >
> > Anyone else think this is a good idea?
> >
> > //Really sorry about this SFINAE
> > template <typename T, typename std::enable_if<std::is_same<void, decltype(std::declval<T>()(1))>::value, int>::type = 0>
> > void eachPixel(const QImage &image, T pixelFunc, const QRect &bounds=QRect()) {
> > QRect imageRect(QRect(0,0, image.width(), image.height()));
> > QRect region = bounds.isNull() ? imageRect : bounds.intersected(imageRect);
> >
> > for (int y=region.y(); y<=region.bottom(); y++) {
> > for (int x=region.x(); x<=region.right(); x++) {
> > pixelFunc(image.pixel(x,y));
> > }
> > }
> > }
> >
> > //Really sorry about this SFINAE too.
> > template <typename T, typename std::enable_if<std::is_same<uint, decltype(std::declval<T>()(1))>::value, int>::type = 0>
> > QImage eachPixel(const QImage &image, T pixelFunc, const QRect &bounds=QRect()) {
> > QRect imageRect(QRect(0,0, image.width(), image.height()));
> > QRect region = bounds.isNull() ? imageRect : bounds.intersected(imageRect);
> >
> > QImage out(image.size(), image.format());
> > for (int y=region.y(); y<=region.bottom(); y++) {
> > for (int x=region.x(); x<=region.right(); x++) {
> > out.setPixel(x,y, pixelFunc(image.pixel(x,y)));
> > }
> > }
> > return out;
> > }
> >
> > void test_pixelFunc() {
> > QImage image(300,200, QImage::Format_ARGB32);
> > image.fill(Qt::blue);
> >
> > QImage out = eachPixel(image, [] (uint p) {
> > return qRgb(qRed(p), qBlue(p), qGreen(p)); // swap green and blue channels
> > }, QRect (5,5, 300, 200)); // for a portion of the image
> > out.save("test_pixelfunc.png");
> >
> >
> > int accumulator=0;
> > int min = 255;
> > int max = 0
> > eachPixel(image, [&accumulator, &min, &max](uint p) { // This one can be used for statistics,
> > accumulator +=p; // if you're not making a new image
> > if (p < min) min = p;
> > if (p > max) max = p;
> > }, QRect (5,5, 300, 200));
> > qDebug() << "accumulator" << accumulator << "min" << min << "max" << max;
> >
> > }
> > _______________________________________________
> > Interest mailing list
> > Interest at qt-project.org
> > http://lists.qt-project.org/mailman/listinfo/interest
> >
>
More information about the Interest
mailing list