/*************************************************************************** CImage.cpp (c) 2000-2017 BenoƮt Minisini This program is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; either version 2, or (at your option) any later version. 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 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. ***************************************************************************/ #define __CIMAGE_CPP #include #include "gambas.h" #include "kimageeffect.h" #include "effect.h" #include "main.h" #include "CImage.h" static void return_copy(GB_IMAGE img) { GB_IMG *info = (GB_IMG *)img; GB_IMG *copy; copy = IMAGE.Create(info->width, info->height, info->format, info->data); GB.ReturnObject(copy); } BEGIN_METHOD(CIMAGE_gradient, GB_INTEGER w; GB_INTEGER h; GB_INTEGER cs; GB_INTEGER cd; GB_INTEGER orient; GB_FLOAT x; GB_FLOAT y) QSize size(VARG(w), VARG(h)); QColor cs(VARG(cs)); QColor cd(VARG(cd)); if (MISSING(x) || MISSING(y)) { QImage img = KImageEffect::gradient(size, cs, cd, (KImageEffect::GradientType)VARG(orient)); GB.ReturnObject(img.object()); } else { QImage img = KImageEffect::unbalancedGradient(size, cs, cd, (KImageEffect::GradientType)VARG(orient), (int)(VARG(x) * 200), (int)(VARG(y) * 200)); GB.ReturnObject(img.object()); } END_METHOD BEGIN_METHOD(CIMAGE_intensity, GB_FLOAT val; GB_INTEGER channel) KImageEffect::RGBComponent channel = KImageEffect::All; QImage img(THIS); if (!MISSING(channel)) channel = (KImageEffect::RGBComponent)VARG(channel); if (channel == KImageEffect::All) KImageEffect::intensity(img, VARG(val)); else KImageEffect::channelIntensity(img, VARG(val), channel); END_METHOD BEGIN_METHOD(CIMAGE_flatten, GB_INTEGER c1; GB_INTEGER c2) QImage img(THIS); KImageEffect::flatten(img, VARG(c1), VARG(c2)); END_METHOD BEGIN_METHOD(CIMAGE_fade, GB_INTEGER col; GB_FLOAT val) QImage img(THIS); KImageEffect::fade(img, VARG(val), VARG(col)); END_METHOD /*BEGIN_METHOD_VOID(CIMAGE_gray) QImage img(THIS); KImageEffect::toGray(img, false); END_METHOD*/ /*BEGIN_METHOD(CIMAGE_desaturate, GB_FLOAT val) QImage img(THIS); KImageEffect::desaturate(img, VARGOPT(val, 0.3)); END_METHOD*/ BEGIN_METHOD(CIMAGE_threshold, GB_FLOAT val) QImage img(THIS); KImageEffect::threshold(img, (uint)(VARG(val) * 255)); END_METHOD BEGIN_METHOD(CIMAGE_solarize, GB_FLOAT val) QImage img(THIS); KImageEffect::solarize(img, VARG(val) * 100); END_METHOD BEGIN_METHOD_VOID(CIMAGE_normalize) QImage img(THIS); KImageEffect::normalize(img); END_METHOD BEGIN_METHOD_VOID(CIMAGE_equalize) QImage img(THIS); KImageEffect::equalize(img); END_METHOD BEGIN_METHOD(CIMAGE_balance, GB_FLOAT brightness; GB_FLOAT contrast; GB_FLOAT gamma; GB_INTEGER channel) Effect::balance(THIS, VARGOPT(channel, Effect::All), (int)(VARG(brightness) * 50), (int)(VARG(contrast) * 50), (int)(VARG(gamma) * 50)); END_METHOD BEGIN_METHOD(CIMAGE_invert, GB_INTEGER channel) int channel = VARGOPT(channel, Effect::All); if (channel == -1) // Compatibilty with Image.Invert() in 'gb.image'. IMAGE.Invert((GB_IMG *)THIS, true); else Effect::invert(THIS, channel); END_METHOD BEGIN_METHOD_VOID(CIMAGE_emboss) QImage src(THIS); QImage dest = KImageEffect::emboss(src, 0, 1); GB.ReturnObject(dest.object()); END_METHOD BEGIN_METHOD(CIMAGE_edge, GB_FLOAT radius) double radius = VARGOPT(radius, -1); if (radius != 0) { if (radius < 0) radius = 0; QImage src(THIS); QImage dest = KImageEffect::edge(src, radius); GB.ReturnObject(dest.object()); } else return_copy(THIS); END_METHOD BEGIN_METHOD_VOID(CIMAGE_despeckle) QImage src(THIS); QImage dest = KImageEffect::despeckle(src); GB.ReturnObject(dest.object()); END_METHOD BEGIN_METHOD(CIMAGE_blur, GB_FLOAT val) double val = VARGOPT(val, 0.2); double sigma = 0.5 + val * (4.0 - 0.5); double radius = 8; QImage src(THIS); QImage dest = KImageEffect::blur(src, radius, sigma); GB.ReturnObject(dest.object()); END_METHOD BEGIN_METHOD(CIMAGE_sharpen, GB_FLOAT val) double val = VARGOPT(val, 0.2); double radius = 0.1 + val * (2.5 - 0.1); double sigma = radius < 1 ? radius : sqrt(radius); QImage src(THIS); QImage dest = KImageEffect::sharpen(src, radius, sigma); GB.ReturnObject(dest.object()); END_METHOD BEGIN_METHOD(CIMAGE_charcoal, GB_FLOAT radius) double radius = VARGOPT(radius, -1); if (radius != 0) { if (radius < 0) radius = 0; QImage src(THIS); QImage dest = KImageEffect::charcoal(src, radius, 0.5); GB.ReturnObject(dest.object()); } else return_copy(THIS); END_METHOD BEGIN_METHOD(CIMAGE_oil_paint, GB_FLOAT radius) double radius = VARGOPT(radius, -1); if (radius != 0) { if (radius < 0) radius = 0; QImage src(THIS); QImage dest = KImageEffect::oilPaintConvolve(src, radius); GB.ReturnObject(dest.object()); } else return_copy(THIS); END_METHOD BEGIN_METHOD(CIMAGE_spread, GB_INTEGER amount) int amount = VARGOPT(amount, 0); if (amount > 0) { QImage src(THIS); QImage dest = KImageEffect::spread(src, VARGOPT(amount, 3)); GB.ReturnObject(dest.object()); } else return_copy(THIS); END_METHOD BEGIN_METHOD(CIMAGE_shade, GB_FLOAT azimuth; GB_FLOAT elevation) QImage src(THIS); QImage dest = KImageEffect::shade(src, true, VARGOPT(azimuth, M_PI/6), VARGOPT(elevation, M_PI/6)); GB.ReturnObject(dest.object()); END_METHOD BEGIN_METHOD(CIMAGE_swirl, GB_FLOAT angle; GB_INTEGER background) QImage src(THIS); QImage dest = KImageEffect::swirl(src, VARGOPT(angle, M_PI/3), VARGOPT(background, 0xFFFFFF) ^ 0xFF000000); GB.ReturnObject(dest.object()); END_METHOD BEGIN_METHOD(CIMAGE_wave, GB_FLOAT amp; GB_FLOAT freq; GB_INTEGER background) QImage src(THIS); QImage dest = KImageEffect::wave(src, VARGOPT(amp, 25.0), VARGOPT(freq, 150.0), VARGOPT(background, 0xFFFFFF) ^ 0xFF000000); GB.ReturnObject(dest.object()); END_METHOD BEGIN_METHOD(CIMAGE_noise, GB_INTEGER type) QImage src(THIS); QImage dest = KImageEffect::addNoise(src, (KImageEffect::NoiseType)VARG(type)); GB.ReturnObject(dest.object()); END_METHOD BEGIN_METHOD(CIMAGE_implode, GB_FLOAT factor; GB_INTEGER background) QImage src(THIS); QImage dest = KImageEffect::implode(src, VARGOPT(factor, 1.0) * 100.0, VARGOPT(background, 0xFFFFFF) ^ 0xFF000000); GB.ReturnObject(dest.object()); END_METHOD BEGIN_METHOD_VOID(Image_Histogram) CIMAGEHISTOGRAM *hist; int *histogram; QImage image(THIS); unsigned int *p, *pm; hist = (CIMAGEHISTOGRAM *)GB.New(GB.FindClass("ImageHistogram"), NULL, NULL); GB.Alloc(POINTER(&histogram), sizeof(int) * 256 * 4); memset(histogram, 0, 256 * 4 * sizeof(int)); p = (unsigned int *)image.bits(); pm = &p[image.width() * image.height()]; if (image.inverted()) { while (p < pm) { histogram[qRed(*p)]++; histogram[qGreen(*p) + 256]++; histogram[qBlue(*p) + 256 * 2]++; histogram[qAlpha(*p) + 256 * 3]++; p++; } } else { while (p < pm) { histogram[qBlue(*p)]++; histogram[qGreen(*p) + 256]++; histogram[qRed(*p) + 256 * 2]++; histogram[qAlpha(*p) + 256 * 3]++; p++; } } hist->histogram = histogram; GB.ReturnObject(hist); END_METHOD BEGIN_METHOD_VOID(ImageHistogram_free) GB.Free(POINTER(&THIS_HISTOGRAM->histogram)); END_METHOD BEGIN_METHOD(ImageHistogram_get, GB_INTEGER channel; GB_INTEGER value) int channel; int value; switch(VARG(channel)) { case KImageEffect::Blue: channel = 0; break; case KImageEffect::Green: channel = 1; break; case KImageEffect::Red: channel = 2; break; case KImageEffect::Alpha: channel = 3; break; default: GB.Error("Bad channel"); return; } value = VARG(value); if (value < 0 || value > 255) { GB.Error("Out of bounds"); return; } GB.ReturnInteger(THIS_HISTOGRAM->histogram[channel * 256 + value]); END_METHOD GB_DESC ImageHistogramDesc[] = { GB_DECLARE("ImageHistogram", sizeof(CIMAGEHISTOGRAM)), GB_NOT_CREATABLE(), GB_METHOD("_free", NULL, ImageHistogram_free, NULL), GB_METHOD("_get", "i", ImageHistogram_get, "(Channel)i(Value)i"), GB_END_DECLARE }; GB_DESC CImageDesc[] = { GB_DECLARE("Image", 0), GB_CONSTANT("Vertical", "i", KImageEffect::VerticalGradient), GB_CONSTANT("Horizontal", "i", KImageEffect::HorizontalGradient), GB_CONSTANT("Diagonal", "i", KImageEffect::DiagonalGradient), GB_CONSTANT("Cross", "i", KImageEffect::CrossDiagonalGradient), GB_CONSTANT("Pyramid", "i", KImageEffect::PyramidGradient), GB_CONSTANT("Rectangle", "i", KImageEffect::RectangleGradient), GB_CONSTANT("PipeCross", "i", KImageEffect::PipeCrossGradient), GB_CONSTANT("Elliptic", "i", KImageEffect::EllipticGradient), GB_CONSTANT("All", "i", KImageEffect::All), GB_CONSTANT("Red", "i", KImageEffect::Red), GB_CONSTANT("Green", "i", KImageEffect::Green), GB_CONSTANT("Blue", "i", KImageEffect::Blue), GB_CONSTANT("Alpha", "i", KImageEffect::Alpha), GB_CONSTANT("Uniform", "i", KImageEffect::UniformNoise), GB_CONSTANT("Gaussian", "i", KImageEffect::GaussianNoise), GB_CONSTANT("Multiplicative", "i", KImageEffect::MultiplicativeGaussianNoise), GB_CONSTANT("Impulse", "i", KImageEffect::ImpulseNoise), GB_CONSTANT("Laplacian", "i", KImageEffect::LaplacianNoise), GB_CONSTANT("Poisson", "i", KImageEffect::PoissonNoise), GB_STATIC_METHOD("Gradient", "Image", CIMAGE_gradient, "(Width)i(Height)i(SrcColor)i(DstColor)i(Orientation)i[(XDecay)f(YDecay)f]"), GB_METHOD("Intensity", NULL, CIMAGE_intensity, "(Value)f[(Channel)i]"), GB_METHOD("Flatten", NULL, CIMAGE_flatten, "(DarkColor)i(BrightColor)i"), GB_METHOD("Fade", NULL, CIMAGE_fade, "(Color)i(Value)f"), //GB_METHOD("Gray", NULL, CIMAGE_gray, NULL), //GB_METHOD("Desaturate", NULL, CIMAGE_desaturate, "[(Value)f]"), GB_METHOD("Threshold", NULL, CIMAGE_threshold, "(Value)f"), GB_METHOD("Solarize", NULL, CIMAGE_solarize, "(Value)f"), GB_METHOD("Normalize", NULL, CIMAGE_normalize, NULL), GB_METHOD("Equalize", NULL, CIMAGE_equalize, NULL), GB_METHOD("Balance", NULL, CIMAGE_balance, "(Brightness)f(Contrast)f(Gamma)f[(Channel)i]"), GB_METHOD("Invert", NULL, CIMAGE_invert, "[(Channel)i]"), GB_METHOD("Emboss", "Image", CIMAGE_emboss, NULL), GB_METHOD("Edge", "Image", CIMAGE_edge, "[(Radius)f]"), GB_METHOD("Despeckle", "Image", CIMAGE_despeckle, NULL), GB_METHOD("Charcoal", "Image", CIMAGE_charcoal, "[(Radius)f]"), GB_METHOD("OilPaint", "Image", CIMAGE_oil_paint, "[(Radius)f]"), GB_METHOD("Blur", "Image", CIMAGE_blur, "[(Value)f]"), GB_METHOD("Sharpen", "Image", CIMAGE_sharpen, "[(Value)f]"), GB_METHOD("Spread", "Image", CIMAGE_spread, "[(Amount)i]"), GB_METHOD("Shade", "Image", CIMAGE_shade, "[(Azimuth)f(Elevation)f]"), GB_METHOD("Swirl", "Image", CIMAGE_swirl, "[(Angle)f(Background)i]"), GB_METHOD("Wave", "Image", CIMAGE_wave, "[(Amplitude)f(WaveLength)f(Background)i]"), GB_METHOD("Noise", "Image", CIMAGE_noise, "(Noise)i"), GB_METHOD("Implode", "Image", CIMAGE_implode, "[(Factor)f(Background)i]"), GB_METHOD("Histogram", "ImageHistogram", Image_Histogram, NULL), GB_END_DECLARE };