}
+static inline unsigned char clamp(int x)
+{
+ if (x <= 0) {
+ return 0;
+ }
+ if (x > 255) {
+ return 255;
+ }
+ return (unsigned char) x;
+}
+
+static inline unsigned char clamp(float x)
+{
+ if (x <= 0.0f) {
+ return 0;
+ }
+ if (x > 255.0f) {
+ return 255;
+ }
+ return (unsigned char) (x + 0.5f);
+}
+
+
QImage
-ApiSurface::qimageFromRawImage(const image::Image *image)
+ApiSurface::qimageFromRawImage(const image::Image *image,
+ float lowerValue,
+ float upperValue,
+ bool opaque,
+ bool alpha)
{
QImage img;
int width = image->width;
img = QImage(width, height, QImage::Format_ARGB32);
+ int offset = - lowerValue * 255;
+ int scale = 256 / (upperValue - lowerValue);
+
+ float offset_f = - lowerValue;
+ float scale_f = 255.0f / (upperValue - lowerValue);
+
+ int aMask = (opaque || alpha) ? 0xff : 0;
+
const unsigned char *srcRow = image->start();
for (int y = 0; y < height; ++y) {
QRgb *dst = (QRgb *)img.scanLine(y);
for (int x = 0; x < width; ++x) {
unsigned char rgba[4] = {0, 0, 0, 0xff};
for (int c = 0; c < image->channels; ++c) {
- rgba[c] = *src++;
+ rgba[c] = clamp(((*src++ + offset) * scale) >> 8);
}
if (image->channels == 1) {
// Use gray-scale instead of red
rgba[1] = rgba[0];
rgba[2] = rgba[0];
}
+ if (alpha) {
+ rgba[2] = rgba[1] = rgba[0] = rgba[3];
+ }
+ rgba[3] |= aMask;
dst[x] = qRgba(rgba[0], rgba[1], rgba[2], rgba[3]);
}
} else {
for (int x = 0; x < width; ++x) {
unsigned char rgba[4] = {0, 0, 0, 0xff};
for (int c = 0; c < image->channels; ++c) {
- float f = *src++;
- unsigned char u;
- if (f >= 1.0f) {
- u = 255;
- } else if (f <= 0.0f) {
- u = 0;
- } else {
- u = f * 255 + 0.5;
- }
- rgba[c] = u;
+ rgba[c] = clamp((*src++ + offset_f)*scale_f);
}
if (image->channels == 1) {
// Use gray-scale instead of red
rgba[1] = rgba[0];
rgba[2] = rgba[0];
}
+ if (alpha) {
+ rgba[2] = rgba[1] = rgba[0] = rgba[3];
+ }
+ rgba[3] |= aMask;
dst[x] = qRgba(rgba[0], rgba[1], rgba[2], rgba[3]);
}
}
QImage thumb() const;
static image::Image *imageFromBase64(const QByteArray &data);
- static QImage qimageFromRawImage(const image::Image *img);
+ static QImage qimageFromRawImage(const image::Image *img,
+ float lowerValue = 0.0f,
+ float upperValue = 1.0f,
+ bool opaque = false,
+ bool alpha = false);
private:
QSize m_size;
updateGeometry();
}
-static inline int clamp(int x)
-{
- if (x <= 0) {
- return 0;
- }
- if (x > 255) {
- return 255;
- }
- return x;
-}
-
void ImageViewer::slotUpdate()
{
m_convertedImage =
bool opaque = opaqueCheckBox->isChecked();
bool alpha = alphaCheckBox->isChecked();
- if (lowerValue != 0.0 || upperValue != 1.0 || opaque || alpha) {
- /*
- * Rescale the image.
- *
- * XXX: This would be much more useful if done with the full precision
- * of the original image
- */
-
- int offset = - lowerValue * 255;
- int scale = 256 / (upperValue - lowerValue);
-
- m_convertedImage =
- m_convertedImage.convertToFormat(QImage::Format_ARGB32);
-
- if (0) {
- qDebug()
- << "offset = " << offset << "\n"
- << "scale = " << scale << "\n";
- }
-
- int width = m_convertedImage.width();
- int height = m_convertedImage.height();
-
- int aMask = opaque ? 0xff : 0;
-
- for (int y = 0; y < height; ++y) {
- QRgb *scanline = (QRgb *)m_convertedImage.scanLine(y);
- for (int x = 0; x < width; ++x) {
- QRgb pixel = scanline[x];
- int r = qRed(pixel);
- int g = qGreen(pixel);
- int b = qBlue(pixel);
- int a = qAlpha(pixel);
- if (alpha) {
- a = clamp(((a + offset) * scale) >> 8);
- scanline[x] = qRgba(a, a, a, 0xff);
- } else {
- r = clamp(((r + offset) * scale) >> 8);
- g = clamp(((g + offset) * scale) >> 8);
- b = clamp(((b + offset) * scale) >> 8);
- a |= aMask;
- scanline[x] = qRgba(r, g, b, a);
- }
- }
- }
+ m_convertedImage = ApiSurface::qimageFromRawImage(m_image,
+ lowerValue, upperValue,
+ opaque, alpha);
+
+ if (flipCheckBox->isChecked()) {
+ m_convertedImage = m_convertedImage.mirrored(false, true);
}
m_pixelWidget->setSurface(m_convertedImage);