#include "apisurface.h"
+#include "thumbnail.h"
+
+#include <sstream>
#include <QDebug>
#include <QSysInfo>
+#include "image/image.hpp"
+
+
ApiSurface::ApiSurface()
{
}
m_size = size;
}
-int ApiSurface::numChannels() const
+struct ByteArrayBuf : public std::streambuf
{
- return m_numChannels;
-}
+ ByteArrayBuf(QByteArray & a)
+ {
+ setg(a.data(), a.data(), a.data() + a.size());
+ }
+};
-void ApiSurface::setNumChannels(int numChannels)
+void ApiSurface::contentsFromBase64(const QByteArray &base64)
{
- m_numChannels = numChannels;
+ m_base64Data = base64;
+
+ /*
+ * We need to do the conversion to create the thumbnail
+ */
+ image::Image *image = imageFromBase64(base64);
+ Q_ASSERT(image);
+ QImage img = qimageFromRawImage(image);
+ m_thumb = thumbnail(img);
+ delete image;
}
-static inline int
-rgba8_to_argb(quint8 r, quint8 g, quint8 b, quint8 a)
+QByteArray ApiSurface::base64Data() const
{
- return (a << 24 | r << 16 | g << 8 | b);
+ return m_base64Data;
}
-static inline int
-rgbaf2argb(float r, float g, float b, float a)
+QImage ApiSurface::thumb() const
{
- quint8 rb = r * 255;
- quint8 gb = g * 255;
- quint8 bb = b * 255;
- quint8 ab = a * 255;
-
- return (ab << 24 | rb << 16 | gb << 8 | bb);
+ return m_thumb;
}
-void ApiSurface::contentsFromBase64(const QByteArray &base64)
+int ApiSurface::depth() const
{
- QByteArray dataArray = QByteArray::fromBase64(base64);
- const quint8 *data = (const quint8*)dataArray.data();
- int width = m_size.width();
- int height = m_size.height();
-
- if (width <= 0 || height <= 0)
- return;
-
- int *pixelData = (int*)malloc(sizeof(int) * width * height);
-
- //XXX not sure if this will work when
- // QSysInfo::ByteOrder == QSysInfo::BigEndian
-
- if (m_numChannels == 4) {
- for (int y = 0; y < height; ++y) {
- for (int x = 0; x < width; ++x) {
- int pixel = rgba8_to_argb(data[(y * width + x) * 4 + 0],
- data[(y * width + x) * 4 + 1],
- data[(y * width + x) * 4 + 2],
- data[(y * width + x) * 4 + 3]);
- pixelData[y * width + x] = pixel;
- }
- }
- } else if (m_numChannels == 1) {
- for (int y = 0; y < height; ++y) {
- for (int x = 0; x < width; ++x) {
- int pixel = rgba8_to_argb(data[y * width + x],
- data[y * width + x],
- data[y * width + x],
- 255);
- pixelData[y * width + x] = pixel;
- }
- }
- } else {
- Q_ASSERT(0);
- }
-
- m_image = QImage((uchar*)pixelData,
- width, height,
- QImage::Format_ARGB32).mirrored();
- m_thumb = m_image.scaled(64, 64, Qt::KeepAspectRatio);
- //m_image.save("testoutput.png");
+ return m_depth;
+}
- free(pixelData);
+void ApiSurface::setDepth(int depth)
+{
+ m_depth = depth;
}
-QImage ApiSurface::image() const
+QString ApiSurface::formatName() const
{
- return m_image;
+ return m_formatName;
}
-QImage ApiSurface::thumb() const
+void ApiSurface::setFormatName(const QString &str)
{
- return m_thumb;
+ m_formatName = str;
}
+
ApiTexture::ApiTexture()
- : ApiSurface(),
- m_unit(0),
- m_level(0)
+ : ApiSurface()
{
}
-int ApiTexture::unit() const
+QString ApiTexture::label() const
{
- return m_unit;
+ return m_label;
}
-void ApiTexture::setUnit(int un)
+void ApiTexture::setLabel(const QString &str)
{
- m_unit = un;
+ m_label = str;
}
-QString ApiTexture::target() const
+ApiFramebuffer::ApiFramebuffer()
+ : ApiSurface()
{
- return m_target;
}
-void ApiTexture::setTarget(const QString &str)
+QString ApiFramebuffer::type() const
{
- m_target = str;
+ return m_type;
}
-int ApiTexture::level() const
+void ApiFramebuffer::setType(const QString &str)
{
- return m_level;
+ m_type = str;
}
-void ApiTexture::setLevel(int l)
+image::Image *
+ApiSurface::imageFromBase64(const QByteArray &base64)
{
- m_level = l;
+ QByteArray dataArray = QByteArray::fromBase64(base64);
+ image::Image *image;
+
+ /*
+ * Detect the PNG vs PFM images.
+ */
+ const char pngSignature[] = {(char)0x89, 0x50, 0x4E, 0x47, 0x0D, 0x0A, 0x1A, 0x0A, 0};
+ if (dataArray.startsWith(pngSignature)) {
+ ByteArrayBuf buf(dataArray);
+ std::istream istr(&buf);
+ image = image::readPNG(istr);
+ } else {
+ image = image::readPNM(dataArray.data(), dataArray.size());
+ }
+
+ return image;
}
-ApiFramebuffer::ApiFramebuffer()
- : ApiSurface()
+
+static inline unsigned char clamp(int x)
{
+ if (x <= 0) {
+ return 0;
+ }
+ if (x > 255) {
+ return 255;
+ }
+ return (unsigned char) x;
}
-QString ApiFramebuffer::type() const
+static inline unsigned char clamp(float x)
{
- return m_type;
+ if (x <= 0.0f) {
+ return 0;
+ }
+ if (x > 255.0f) {
+ return 255;
+ }
+ return (unsigned char) (x + 0.5f);
}
-void ApiFramebuffer::setType(const QString &str)
+
+QImage
+ApiSurface::qimageFromRawImage(const image::Image *image,
+ float lowerValue,
+ float upperValue,
+ bool opaque,
+ bool alpha)
{
- m_type = str;
+ QImage img;
+ int width = image->width;
+ int height = image->height;
+
+ 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);
+
+ if (image->channelType == image::TYPE_UNORM8) {
+ const unsigned char *src = srcRow;
+ 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] = 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 {
+ const float *src = (const float *)srcRow;
+ 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] = 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]);
+ }
+ }
+
+ srcRow += image->stride();
+ }
+
+ return img;
}