bimg/vips.h

660 lines
18 KiB
C
Raw Normal View History

2015-03-30 04:55:04 +08:00
#include <stdlib.h>
2017-01-25 02:14:34 +08:00
#include <string.h>
2015-03-30 04:55:04 +08:00
#include <vips/vips.h>
#include <vips/foreign.h>
2015-03-30 04:55:04 +08:00
#include <vips/vips7compat.h>
2021-07-06 19:59:12 +08:00
#include <vips/vector.h>
2015-03-30 04:55:04 +08:00
2015-09-09 04:54:32 +08:00
/**
* Starting libvips 7.41, VIPS_ANGLE_x has been renamed to VIPS_ANGLE_Dx
* "to help python". So we provide the macro to correctly build for versions
* before 7.41.x.
* https://github.com/jcupitt/libvips/blob/master/ChangeLog#L128
*/
#if (VIPS_MAJOR_VERSION == 7 && VIPS_MINOR_VERSION < 41)
#define VIPS_ANGLE_D0 VIPS_ANGLE_0
#define VIPS_ANGLE_D90 VIPS_ANGLE_90
#define VIPS_ANGLE_D180 VIPS_ANGLE_180
#define VIPS_ANGLE_D270 VIPS_ANGLE_270
#endif
2015-10-16 21:25:32 +08:00
#define EXIF_IFD0_ORIENTATION "exif-ifd0-Orientation"
#define INT_TO_GBOOLEAN(bool) (bool > 0 ? TRUE : FALSE)
2015-04-04 22:12:45 +08:00
enum types {
UNKNOWN = 0,
JPEG,
WEBP,
PNG,
TIFF,
GIF,
PDF,
SVG,
MAGICK,
HEIF,
AVIF
2015-04-04 22:12:45 +08:00
};
2015-04-11 08:49:11 +08:00
typedef struct {
2015-04-11 18:12:29 +08:00
const char *Text;
const char *Font;
} WatermarkTextOptions;
2015-04-11 08:49:11 +08:00
typedef struct {
2015-04-11 08:56:59 +08:00
int Width;
int DPI;
int Margin;
int NoReplicate;
float Opacity;
2015-04-11 08:49:11 +08:00
double Background[3];
} WatermarkOptions;
2015-04-11 08:49:11 +08:00
2017-01-25 02:14:34 +08:00
typedef struct {
int Left;
int Top;
float Opacity;
} WatermarkImageOptions;
static unsigned long
has_profile_embed(VipsImage *image) {
return vips_image_get_typeof(image, VIPS_META_ICC_NAME);
}
static void
remove_profile(VipsImage *image) {
vips_image_remove(image, VIPS_META_ICC_NAME);
}
static int
has_alpha_channel(VipsImage *image) {
return (
(image->Bands == 2 && image->Type == VIPS_INTERPRETATION_B_W) ||
(image->Bands == 4 && image->Type != VIPS_INTERPRETATION_CMYK) ||
(image->Bands == 5 && image->Type == VIPS_INTERPRETATION_CMYK)
) ? 1 : 0;
}
2015-09-09 04:54:32 +08:00
/**
* This method is here to handle the weird initialization of the vips lib.
* libvips use a macro VIPS_INIT() that call vips__init() in version < 7.41,
* or calls vips_init() in version >= 7.41.
2015-09-09 04:54:32 +08:00
*
* Anyway, it's not possible to build bimg on Debian Jessie with libvips 7.40.x,
* as vips_init() is a macro to VIPS_INIT(), which is also a macro, hence, cgo
* is unable to determine the return type of vips_init(), making the build impossible.
* In order to correctly build bimg, for version < 7.41, we should undef vips_init and
* creates a vips_init() method that calls VIPS_INIT().
*/
2015-09-09 04:54:32 +08:00
#if (VIPS_MAJOR_VERSION == 7 && VIPS_MINOR_VERSION < 41)
2015-09-09 04:54:32 +08:00
#undef vips_init
int
vips_init(const char *argv0)
{
return VIPS_INIT(argv0);
}
#endif
2015-04-27 02:07:47 +08:00
void
vips_enable_cache_set_trace() {
2015-04-27 02:07:47 +08:00
vips_cache_set_trace(TRUE);
}
2015-04-27 02:07:47 +08:00
2015-03-30 04:55:04 +08:00
int
vips_affine_interpolator(VipsImage *in, VipsImage **out, double a, double b, double c, double d, VipsInterpolate *interpolator, int extend) {
return vips_affine(in, out, a, b, c, d, "interpolate", interpolator, "extend", extend, NULL);
}
2015-03-30 04:55:04 +08:00
int
vips_jpegload_buffer_shrink(void *buf, size_t len, VipsImage **out, int shrink) {
2015-03-30 04:55:04 +08:00
return vips_jpegload_buffer(buf, len, out, "shrink", shrink, NULL);
}
2015-03-30 04:55:04 +08:00
int
vips_webpload_buffer_shrink(void *buf, size_t len, VipsImage **out, int shrink) {
return vips_webpload_buffer(buf, len, out, "shrink", shrink, NULL);
}
2015-03-30 08:41:16 +08:00
int
vips_flip_bridge(VipsImage *in, VipsImage **out, int direction) {
return vips_flip(in, out, direction, NULL);
}
2015-03-30 08:41:16 +08:00
2015-03-30 04:55:04 +08:00
int
vips_shrink_bridge(VipsImage *in, VipsImage **out, double xshrink, double yshrink) {
2015-03-30 04:55:04 +08:00
return vips_shrink(in, out, xshrink, yshrink, NULL);
}
2015-03-30 04:55:04 +08:00
2017-07-04 09:33:13 +08:00
int
vips_reduce_bridge(VipsImage *in, VipsImage **out, double xshrink, double yshrink) {
return vips_reduce(in, out, xshrink, yshrink, NULL);
}
2016-09-28 06:56:18 +08:00
int
vips_type_find_bridge(int t) {
if (t == GIF) {
return vips_type_find("VipsOperation", "gifload");
}
if (t == PDF) {
return vips_type_find("VipsOperation", "pdfload");
}
if (t == TIFF) {
return vips_type_find("VipsOperation", "tiffload");
}
if (t == SVG) {
return vips_type_find("VipsOperation", "svgload");
}
if (t == WEBP) {
return vips_type_find("VipsOperation", "webpload");
}
if (t == PNG) {
return vips_type_find("VipsOperation", "pngload");
}
if (t == JPEG) {
return vips_type_find("VipsOperation", "jpegload");
}
if (t == MAGICK) {
return vips_type_find("VipsOperation", "magickload");
}
#if (VIPS_MAJOR_VERSION > 8 || (VIPS_MAJOR_VERSION == 8 && VIPS_MINOR_VERSION >= 8))
if (t == HEIF) {
return vips_type_find("VipsOperation", "heifload");
}
#endif
2016-09-28 06:56:18 +08:00
return 0;
}
int
vips_type_find_save_bridge(int t) {
if (t == TIFF) {
return vips_type_find("VipsOperation", "tiffsave_buffer");
}
if (t == WEBP) {
return vips_type_find("VipsOperation", "webpsave_buffer");
}
if (t == PNG) {
return vips_type_find("VipsOperation", "pngsave_buffer");
}
if (t == JPEG) {
return vips_type_find("VipsOperation", "jpegsave_buffer");
}
#if (VIPS_MAJOR_VERSION > 8 || (VIPS_MAJOR_VERSION == 8 && VIPS_MINOR_VERSION >= 8))
if (t == HEIF) {
return vips_type_find("VipsOperation", "heifsave_buffer");
}
#endif
return 0;
}
2015-03-30 04:55:04 +08:00
int
vips_rotate_bridge(VipsImage *in, VipsImage **out, int angle) {
2015-03-30 04:55:04 +08:00
int rotate = VIPS_ANGLE_D0;
2016-08-25 21:28:15 +08:00
angle %= 360;
if (angle == 45) {
rotate = VIPS_ANGLE45_D45;
} else if (angle == 90) {
2015-03-30 08:43:24 +08:00
rotate = VIPS_ANGLE_D90;
2016-08-25 21:28:15 +08:00
} else if (angle == 135) {
rotate = VIPS_ANGLE45_D135;
2015-03-30 08:43:24 +08:00
} else if (angle == 180) {
rotate = VIPS_ANGLE_D180;
2016-08-25 21:28:15 +08:00
} else if (angle == 225) {
rotate = VIPS_ANGLE45_D225;
2015-03-30 08:43:24 +08:00
} else if (angle == 270) {
rotate = VIPS_ANGLE_D270;
2016-08-25 21:28:15 +08:00
} else if (angle == 315) {
rotate = VIPS_ANGLE45_D315;
} else {
angle = 0;
2015-03-30 08:43:24 +08:00
}
2015-03-30 04:55:04 +08:00
2016-08-25 21:28:15 +08:00
if (angle > 0 && angle % 90 != 0) {
return vips_rot45(in, out, "angle", rotate, NULL);
} else {
return vips_rot(in, out, rotate, NULL);
}
}
2015-03-30 04:55:04 +08:00
2020-08-05 01:40:07 +08:00
int
vips_autorot_bridge(VipsImage *in, VipsImage **out) {
return vips_autorot(in, out, NULL);
}
2020-07-30 22:23:26 +08:00
const char *
vips_exif_tag(VipsImage *image, const char *tag) {
const char *exif;
2015-04-06 23:06:44 +08:00
if (
2020-07-30 22:23:26 +08:00
vips_image_get_typeof(image, tag) != 0 &&
!vips_image_get_string(image, tag, &exif)
2015-04-06 23:06:44 +08:00
) {
2020-07-30 22:23:26 +08:00
return &exif[0];
}
return "";
}
int
2020-08-04 17:24:15 +08:00
vips_exif_tag_to_int(VipsImage *image, const char *tag) {
int value = 0;
const char *exif = vips_exif_tag(image, tag);
2020-07-30 22:23:26 +08:00
if (strcmp(exif, "")) {
2020-08-04 17:24:15 +08:00
value = atoi(&exif[0]);
2015-04-06 23:06:44 +08:00
}
2020-08-04 17:24:15 +08:00
return value;
}
2015-04-06 19:25:01 +08:00
2020-08-04 17:24:15 +08:00
int
vips_exif_orientation(VipsImage *image) {
return vips_exif_tag_to_int(image, EXIF_IFD0_ORIENTATION);
2020-07-30 22:23:26 +08:00
}
2015-04-06 19:25:01 +08:00
int
interpolator_window_size(char const *name) {
2015-04-06 23:06:44 +08:00
VipsInterpolate *interpolator = vips_interpolate_new(name);
int window_size = vips_interpolate_get_window_size(interpolator);
g_object_unref(interpolator);
return window_size;
}
2015-04-06 19:25:01 +08:00
const char *
vips_enum_nick_bridge(VipsImage *image) {
return vips_enum_nick(VIPS_TYPE_INTERPRETATION, image->Type);
}
2015-04-06 19:25:01 +08:00
2015-04-09 07:44:54 +08:00
int
vips_zoom_bridge(VipsImage *in, VipsImage **out, int xfac, int yfac) {
return vips_zoom(in, out, xfac, yfac, NULL);
}
2015-04-09 07:44:54 +08:00
2015-03-30 04:55:04 +08:00
int
vips_embed_bridge(VipsImage *in, VipsImage **out, int left, int top, int width, int height, int extend, double r, double g, double b) {
if (extend == VIPS_EXTEND_BACKGROUND) {
if (has_alpha_channel(in) == 1) {
double background[4] = {r, g, b, 0.0};
VipsArrayDouble *vipsBackground = vips_array_double_new(background, 4);
return vips_embed(in, out, left, top, width, height, "extend", extend, "background", vipsBackground, NULL);
} else {
double background[3] = {r, g, b};
VipsArrayDouble *vipsBackground = vips_array_double_new(background, 3);
return vips_embed(in, out, left, top, width, height, "extend", extend, "background", vipsBackground, NULL);}
}
2015-04-11 08:49:11 +08:00
return vips_embed(in, out, left, top, width, height, "extend", extend, NULL);
}
2015-03-30 04:55:04 +08:00
int
vips_extract_area_bridge(VipsImage *in, VipsImage **out, int left, int top, int width, int height) {
2015-03-30 04:55:04 +08:00
return vips_extract_area(in, out, left, top, width, height, NULL);
}
2015-03-30 04:55:04 +08:00
int
vips_colourspace_issupported_bridge(VipsImage *in) {
return vips_colourspace_issupported(in) ? 1 : 0;
}
VipsInterpretation
vips_image_guess_interpretation_bridge(VipsImage *in) {
return vips_image_guess_interpretation(in);
}
2015-06-14 03:16:28 +08:00
int
vips_colourspace_bridge(VipsImage *in, VipsImage **out, VipsInterpretation space) {
return vips_colourspace(in, out, space, NULL);
}
int
vips_icc_transform_bridge (VipsImage *in, VipsImage **out, const char *output_icc_profile) {
// `output_icc_profile` represents the absolute path to the output ICC profile file
return vips_icc_transform(in, out, output_icc_profile, "embedded", TRUE, NULL);
}
int
vips_icc_transform_with_default_bridge (VipsImage *in, VipsImage **out, const char *output_icc_profile, const char *input_icc_profile) {
// `output_icc_profile` represents the absolute path to the output ICC profile file
return vips_icc_transform(in, out, output_icc_profile, "input_profile", input_icc_profile, "embedded", FALSE, NULL);
}
2015-03-30 04:55:04 +08:00
int
vips_jpegsave_bridge(VipsImage *in, void **buf, size_t *len, int strip, int quality, int interlace) {
return vips_jpegsave_buffer(in, buf, len,
"strip", INT_TO_GBOOLEAN(strip),
"Q", quality,
"optimize_coding", TRUE,
"interlace", INT_TO_GBOOLEAN(interlace),
NULL
);
}
2015-04-06 05:39:15 +08:00
int
2020-06-08 16:43:04 +08:00
vips_pngsave_bridge(VipsImage *in, void **buf, size_t *len, int strip, int compression, int quality, int interlace, int palette) {
2020-06-09 20:50:16 +08:00
#if (VIPS_MAJOR_VERSION >= 8 && VIPS_MINOR_VERSION >= 7)
return vips_pngsave_buffer(in, buf, len,
"strip", INT_TO_GBOOLEAN(strip),
"compression", compression,
"interlace", INT_TO_GBOOLEAN(interlace),
"filter", VIPS_FOREIGN_PNG_FILTER_ALL,
"palette", INT_TO_GBOOLEAN(palette),
NULL
);
2015-04-06 05:39:15 +08:00
#else
return vips_pngsave_buffer(in, buf, len,
"strip", INT_TO_GBOOLEAN(strip),
"compression", compression,
"interlace", INT_TO_GBOOLEAN(interlace),
NULL
);
2015-04-06 05:39:15 +08:00
#endif
}
2015-04-06 05:39:15 +08:00
int
2017-11-11 11:56:12 +08:00
vips_webpsave_bridge(VipsImage *in, void **buf, size_t *len, int strip, int quality, int lossless) {
return vips_webpsave_buffer(in, buf, len,
"strip", INT_TO_GBOOLEAN(strip),
"Q", quality,
2017-11-11 11:56:12 +08:00
"lossless", INT_TO_GBOOLEAN(lossless),
NULL
);
}
2015-04-06 05:39:15 +08:00
int
vips_tiffsave_bridge(VipsImage *in, void **buf, size_t *len) {
#if (VIPS_MAJOR_VERSION >= 8 && VIPS_MINOR_VERSION >= 5)
return vips_tiffsave_buffer(in, buf, len, NULL);
#else
return 0;
#endif
}
int
vips_avifsave_bridge(VipsImage *in, void **buf, size_t *len, int strip, int quality, int lossless, int speed) {
#if (VIPS_MAJOR_VERSION > 8 || (VIPS_MAJOR_VERSION >= 8 && VIPS_MINOR_VERSION > 10) || (VIPS_MAJOR_VERSION >= 8 && VIPS_MINOR_VERSION >= 10 && VIPS_MICRO_VERSION >= 2))
return vips_heifsave_buffer(in, buf, len,
"strip", INT_TO_GBOOLEAN(strip),
"Q", quality,
"lossless", INT_TO_GBOOLEAN(lossless),
"compression", VIPS_FOREIGN_HEIF_COMPRESSION_AV1,
"speed", speed,
NULL
);
#elif (VIPS_MAJOR_VERSION > 8 || (VIPS_MAJOR_VERSION == 8 && VIPS_MINOR_VERSION >= 9))
return vips_heifsave_buffer(in, buf, len,
"strip", INT_TO_GBOOLEAN(strip),
"Q", quality,
"lossless", INT_TO_GBOOLEAN(lossless),
"compression", VIPS_FOREIGN_HEIF_COMPRESSION_AV1,
NULL
);
#else
return 0;
#endif
}
int
vips_heifsave_bridge(VipsImage *in, void **buf, size_t *len, int strip, int quality, int lossless) {
#if (VIPS_MAJOR_VERSION > 8 || (VIPS_MAJOR_VERSION == 8 && VIPS_MINOR_VERSION >= 8))
return vips_heifsave_buffer(in, buf, len,
"strip", INT_TO_GBOOLEAN(strip),
"Q", quality,
"lossless", INT_TO_GBOOLEAN(lossless),
NULL
);
#else
return 0;
#endif
}
int
vips_is_16bit (VipsInterpretation interpretation) {
return interpretation == VIPS_INTERPRETATION_RGB16 || interpretation == VIPS_INTERPRETATION_GREY16;
}
int
vips_flatten_background_brigde(VipsImage *in, VipsImage **out, double r, double g, double b) {
2016-11-11 04:15:52 +08:00
if (vips_is_16bit(in->Type)) {
r = 65535 * r / 255;
g = 65535 * g / 255;
b = 65535 * b / 255;
}
double background[3] = {r, g, b};
VipsArrayDouble *vipsBackground = vips_array_double_new(background, 3);
return vips_flatten(in, out,
2016-11-11 04:15:52 +08:00
"background", vipsBackground,
"max_alpha", vips_is_16bit(in->Type) ? 65535.0 : 255.0,
NULL
);
}
2015-04-06 05:39:15 +08:00
int
vips_init_image (void *buf, size_t len, int imageType, VipsImage **out) {
2015-04-06 23:06:44 +08:00
int code = 1;
if (imageType == JPEG) {
2015-05-06 02:28:39 +08:00
code = vips_jpegload_buffer(buf, len, out, "access", VIPS_ACCESS_RANDOM, NULL);
2015-04-06 23:06:44 +08:00
} else if (imageType == PNG) {
2015-05-06 02:28:39 +08:00
code = vips_pngload_buffer(buf, len, out, "access", VIPS_ACCESS_RANDOM, NULL);
2015-04-06 23:06:44 +08:00
} else if (imageType == WEBP) {
2015-05-06 02:28:39 +08:00
code = vips_webpload_buffer(buf, len, out, "access", VIPS_ACCESS_RANDOM, NULL);
2015-04-06 23:06:44 +08:00
} else if (imageType == TIFF) {
2015-05-06 02:28:39 +08:00
code = vips_tiffload_buffer(buf, len, out, "access", VIPS_ACCESS_RANDOM, NULL);
2015-04-08 05:02:31 +08:00
#if (VIPS_MAJOR_VERSION >= 8)
#if (VIPS_MINOR_VERSION >= 3)
} else if (imageType == GIF) {
code = vips_gifload_buffer(buf, len, out, "access", VIPS_ACCESS_RANDOM, NULL);
} else if (imageType == PDF) {
code = vips_pdfload_buffer(buf, len, out, "access", VIPS_ACCESS_RANDOM, NULL);
} else if (imageType == SVG) {
code = vips_svgload_buffer(buf, len, out, "access", VIPS_ACCESS_RANDOM, NULL);
#endif
2015-04-06 23:06:44 +08:00
} else if (imageType == MAGICK) {
2015-05-06 02:28:39 +08:00
code = vips_magickload_buffer(buf, len, out, "access", VIPS_ACCESS_RANDOM, NULL);
#endif
#if (VIPS_MAJOR_VERSION > 8 || (VIPS_MAJOR_VERSION == 8 && VIPS_MINOR_VERSION >= 8))
} else if (imageType == HEIF) {
code = vips_heifload_buffer(buf, len, out, "access", VIPS_ACCESS_RANDOM, NULL);
#endif
#if (VIPS_MAJOR_VERSION == 8 && VIPS_MINOR_VERSION >= 9)
} else if (imageType == AVIF) {
code = vips_heifload_buffer(buf, len, out, "access", VIPS_ACCESS_RANDOM, NULL);
2015-04-08 05:02:31 +08:00
#endif
2015-04-06 23:06:44 +08:00
}
2015-04-06 05:39:15 +08:00
2015-04-06 23:06:44 +08:00
return code;
}
2015-04-11 08:49:11 +08:00
2015-04-11 18:58:15 +08:00
int
vips_watermark_replicate (VipsImage *orig, VipsImage *in, VipsImage **out) {
2015-04-11 18:58:15 +08:00
VipsImage *cache = vips_image_new();
if (
vips_replicate(in, &cache,
1 + orig->Xsize / in->Xsize,
1 + orig->Ysize / in->Ysize, NULL) ||
vips_crop(cache, out, 0, 0, orig->Xsize, orig->Ysize, NULL)
) {
g_object_unref(cache);
return 1;
}
g_object_unref(cache);
2015-04-11 18:58:15 +08:00
return 0;
}
2015-04-11 18:58:15 +08:00
2015-04-11 08:49:11 +08:00
int
vips_watermark(VipsImage *in, VipsImage **out, WatermarkTextOptions *to, WatermarkOptions *o) {
2015-04-11 09:12:49 +08:00
double ones[3] = { 1, 1, 1 };
VipsImage *base = vips_image_new();
2015-04-11 18:58:15 +08:00
VipsImage **t = (VipsImage **) vips_object_local_array(VIPS_OBJECT(base), 10);
2015-04-11 09:12:49 +08:00
t[0] = in;
// Make the mask.
if (
vips_text(&t[1], to->Text,
"width", o->Width,
"dpi", o->DPI,
"font", to->Font,
NULL) ||
vips_linear1(t[1], &t[2], o->Opacity, 0.0, NULL) ||
vips_cast(t[2], &t[3], VIPS_FORMAT_UCHAR, NULL) ||
vips_embed(t[3], &t[4], 100, 100, t[3]->Xsize + o->Margin, t[3]->Ysize + o->Margin, NULL)
) {
g_object_unref(base);
2015-04-11 18:58:15 +08:00
return 1;
2015-04-11 09:12:49 +08:00
}
// Replicate if necessary
2015-04-11 18:58:15 +08:00
if (o->NoReplicate != 1) {
VipsImage *cache = vips_image_new();
if (vips_watermark_replicate(t[0], t[4], &cache)) {
g_object_unref(cache);
g_object_unref(base);
return 1;
}
g_object_unref(t[4]);
t[4] = cache;
2015-04-11 09:12:49 +08:00
}
// Make the constant image to paint the text with.
if (
2015-04-11 18:58:15 +08:00
vips_black(&t[5], 1, 1, NULL) ||
vips_linear(t[5], &t[6], ones, o->Background, 3, NULL) ||
vips_cast(t[6], &t[7], VIPS_FORMAT_UCHAR, NULL) ||
vips_copy(t[7], &t[8], "interpretation", t[0]->Type, NULL) ||
vips_embed(t[8], &t[9], 0, 0, t[0]->Xsize, t[0]->Ysize, "extend", VIPS_EXTEND_COPY, NULL)
2015-04-11 09:12:49 +08:00
) {
g_object_unref(base);
2015-04-11 18:58:15 +08:00
return 1;
2015-04-11 09:12:49 +08:00
}
// Blend the mask and text and write to output.
2015-04-11 18:58:15 +08:00
if (vips_ifthenelse(t[4], t[9], t[0], out, "blend", TRUE, NULL)) {
2015-04-11 09:12:49 +08:00
g_object_unref(base);
2015-04-11 18:58:15 +08:00
return 1;
2015-04-11 09:12:49 +08:00
}
g_object_unref(base);
2015-04-11 18:58:15 +08:00
return 0;
}
int
vips_gaussblur_bridge(VipsImage *in, VipsImage **out, double sigma, double min_ampl) {
#if (VIPS_MAJOR_VERSION == 7 && VIPS_MINOR_VERSION < 41)
return vips_gaussblur(in, out, (int) sigma, NULL);
#else
return vips_gaussblur(in, out, sigma, NULL, "min_ampl", min_ampl, NULL);
#endif
}
2015-11-17 07:46:08 +08:00
int
vips_sharpen_bridge(VipsImage *in, VipsImage **out, int radius, double x1, double y2, double y3, double m1, double m2) {
#if (VIPS_MAJOR_VERSION == 7 && VIPS_MINOR_VERSION < 41)
return vips_sharpen(in, out, radius, x1, y2, y3, m1, m2, NULL);
#else
return vips_sharpen(in, out, "radius", radius, "x1", x1, "y2", y2, "y3", y3, "m1", m1, "m2", m2, NULL);
#endif
}
2017-01-25 02:14:34 +08:00
int
vips_add_band(VipsImage *in, VipsImage **out, double c) {
#if (VIPS_MAJOR_VERSION > 8 || (VIPS_MAJOR_VERSION >= 8 && VIPS_MINOR_VERSION >= 2))
return vips_bandjoin_const1(in, out, c, NULL);
#else
VipsImage *base = vips_image_new();
if (
vips_black(&base, in->Xsize, in->Ysize, NULL) ||
vips_linear1(base, &base, 1, c, NULL)) {
g_object_unref(base);
return 1;
}
g_object_unref(base);
return vips_bandjoin2(in, base, out, c, NULL);
#endif
2017-01-25 02:14:34 +08:00
}
int
vips_watermark_image(VipsImage *in, VipsImage *sub, VipsImage **out, WatermarkImageOptions *o) {
VipsImage *base = vips_image_new();
VipsImage **t = (VipsImage **) vips_object_local_array(VIPS_OBJECT(base), 10);
// add in and sub for unreffing and later use
t[0] = in;
t[1] = sub;
if (has_alpha_channel(in) == 0) {
vips_add_band(in, &t[0], 255.0);
// in is no longer in the array and won't be unreffed, so add it at the end
t[8] = in;
}
if (has_alpha_channel(sub) == 0) {
vips_add_band(sub, &t[1], 255.0);
// sub is no longer in the array and won't be unreffed, so add it at the end
t[9] = sub;
}
// Place watermark image in the right place and size it to the size of the
// image that should be watermarked
if (
vips_embed(t[1], &t[2], o->Left, o->Top, t[0]->Xsize, t[0]->Ysize, NULL)) {
g_object_unref(base);
return 1;
}
// Create a mask image based on the alpha band from the watermark image
// and place it in the right position
if (
vips_extract_band(t[1], &t[3], t[1]->Bands - 1, "n", 1, NULL) ||
vips_linear1(t[3], &t[4], o->Opacity, 0.0, NULL) ||
vips_cast(t[4], &t[5], VIPS_FORMAT_UCHAR, NULL) ||
vips_copy(t[5], &t[6], "interpretation", t[0]->Type, NULL) ||
vips_embed(t[6], &t[7], o->Left, o->Top, t[0]->Xsize, t[0]->Ysize, NULL)) {
g_object_unref(base);
return 1;
}
// Blend the mask and watermark image and write to output.
if (vips_ifthenelse(t[7], t[2], t[0], out, "blend", TRUE, NULL)) {
g_object_unref(base);
return 1;
}
g_object_unref(base);
return 0;
2017-01-25 02:14:34 +08:00
}
int
vips_smartcrop_bridge(VipsImage *in, VipsImage **out, int width, int height) {
#if (VIPS_MAJOR_VERSION >= 8 && VIPS_MINOR_VERSION >= 5)
return vips_smartcrop(in, out, width, height, NULL);
#else
return 0;
#endif
}
int vips_find_trim_bridge(VipsImage *in, int *top, int *left, int *width, int *height, double r, double g, double b, double threshold) {
#if (VIPS_MAJOR_VERSION >= 8 && VIPS_MINOR_VERSION >= 6)
if (vips_is_16bit(in->Type)) {
r = 65535 * r / 255;
g = 65535 * g / 255;
b = 65535 * b / 255;
}
double background[3] = {r, g, b};
VipsArrayDouble *vipsBackground = vips_array_double_new(background, 3);
return vips_find_trim(in, top, left, width, height, "background", vipsBackground, "threshold", threshold, NULL);
#else
return 0;
#endif
}
2019-08-01 14:30:44 +08:00
int vips_gamma_bridge(VipsImage *in, VipsImage **out, double exponent)
{
return vips_gamma(in, out, "exponent", 1.0 / exponent, NULL);
}