@guru-ai-product/ai-product-kit 0.2.251113194913 → 0.2.251113205658
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/mcp/skills/aipk_init_project/template/AGENTS_TEMPLATE.md +1 -1
- package/mcp/src/server.js +1 -1
- package/package.json +1 -1
- package/skills/aipk_design/GURU_AI.md +119 -7
- package/skills/aipk_design/SKILL.md +8 -7
- package/skills/aipk_design/{auto_panel_splitter → update-requirements-from-design}/SKILL.md +18 -16
- package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/.package-lock.json +113 -0
- package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/@img/colour/LICENSE.md +82 -0
- package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/@img/colour/README.md +15 -0
- package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/@img/colour/color.cjs +1594 -0
- package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/@img/colour/index.cjs +1 -0
- package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/@img/colour/package.json +45 -0
- package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/@img/sharp-darwin-arm64/LICENSE +191 -0
- package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/@img/sharp-darwin-arm64/README.md +18 -0
- package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/@img/sharp-darwin-arm64/lib/sharp-darwin-arm64.node +0 -0
- package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/@img/sharp-darwin-arm64/package.json +40 -0
- package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/@img/sharp-libvips-darwin-arm64/README.md +46 -0
- package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/@img/sharp-libvips-darwin-arm64/lib/glib-2.0/include/glibconfig.h +220 -0
- package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/@img/sharp-libvips-darwin-arm64/lib/index.js +1 -0
- package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/@img/sharp-libvips-darwin-arm64/lib/libvips-cpp.8.17.3.dylib +0 -0
- package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/@img/sharp-libvips-darwin-arm64/package.json +36 -0
- package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/@img/sharp-libvips-darwin-arm64/versions.json +30 -0
- package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/detect-libc/LICENSE +201 -0
- package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/detect-libc/README.md +163 -0
- package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/detect-libc/index.d.ts +14 -0
- package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/detect-libc/lib/detect-libc.js +313 -0
- package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/detect-libc/lib/elf.js +39 -0
- package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/detect-libc/lib/filesystem.js +51 -0
- package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/detect-libc/lib/process.js +24 -0
- package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/detect-libc/package.json +44 -0
- package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/semver/LICENSE +15 -0
- package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/semver/README.md +664 -0
- package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/semver/bin/semver.js +191 -0
- package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/semver/classes/comparator.js +143 -0
- package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/semver/classes/index.js +7 -0
- package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/semver/classes/range.js +557 -0
- package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/semver/classes/semver.js +333 -0
- package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/semver/functions/clean.js +8 -0
- package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/semver/functions/cmp.js +54 -0
- package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/semver/functions/coerce.js +62 -0
- package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/semver/functions/compare-build.js +9 -0
- package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/semver/functions/compare-loose.js +5 -0
- package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/semver/functions/compare.js +7 -0
- package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/semver/functions/diff.js +60 -0
- package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/semver/functions/eq.js +5 -0
- package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/semver/functions/gt.js +5 -0
- package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/semver/functions/gte.js +5 -0
- package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/semver/functions/inc.js +21 -0
- package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/semver/functions/lt.js +5 -0
- package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/semver/functions/lte.js +5 -0
- package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/semver/functions/major.js +5 -0
- package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/semver/functions/minor.js +5 -0
- package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/semver/functions/neq.js +5 -0
- package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/semver/functions/parse.js +18 -0
- package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/semver/functions/patch.js +5 -0
- package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/semver/functions/prerelease.js +8 -0
- package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/semver/functions/rcompare.js +5 -0
- package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/semver/functions/rsort.js +5 -0
- package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/semver/functions/satisfies.js +12 -0
- package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/semver/functions/sort.js +5 -0
- package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/semver/functions/valid.js +8 -0
- package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/semver/index.js +91 -0
- package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/semver/internal/constants.js +37 -0
- package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/semver/internal/debug.js +11 -0
- package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/semver/internal/identifiers.js +29 -0
- package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/semver/internal/lrucache.js +42 -0
- package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/semver/internal/parse-options.js +17 -0
- package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/semver/internal/re.js +223 -0
- package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/semver/package.json +78 -0
- package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/semver/preload.js +4 -0
- package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/semver/range.bnf +16 -0
- package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/semver/ranges/gtr.js +6 -0
- package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/semver/ranges/intersects.js +9 -0
- package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/semver/ranges/ltr.js +6 -0
- package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/semver/ranges/max-satisfying.js +27 -0
- package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/semver/ranges/min-satisfying.js +26 -0
- package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/semver/ranges/min-version.js +63 -0
- package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/semver/ranges/outside.js +82 -0
- package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/semver/ranges/simplify.js +49 -0
- package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/semver/ranges/subset.js +249 -0
- package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/semver/ranges/to-comparators.js +10 -0
- package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/semver/ranges/valid.js +13 -0
- package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/sharp/LICENSE +191 -0
- package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/sharp/README.md +118 -0
- package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/sharp/install/build.js +38 -0
- package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/sharp/install/check.js +14 -0
- package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/sharp/lib/channel.js +177 -0
- package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/sharp/lib/colour.js +195 -0
- package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/sharp/lib/composite.js +212 -0
- package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/sharp/lib/constructor.js +499 -0
- package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/sharp/lib/index.d.ts +1971 -0
- package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/sharp/lib/index.js +16 -0
- package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/sharp/lib/input.js +809 -0
- package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/sharp/lib/is.js +143 -0
- package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/sharp/lib/libvips.js +207 -0
- package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/sharp/lib/operation.js +1016 -0
- package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/sharp/lib/output.js +1666 -0
- package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/sharp/lib/resize.js +595 -0
- package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/sharp/lib/sharp.js +121 -0
- package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/sharp/lib/utility.js +291 -0
- package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/sharp/package.json +202 -0
- package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/sharp/src/binding.gyp +298 -0
- package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/sharp/src/common.cc +1130 -0
- package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/sharp/src/common.h +402 -0
- package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/sharp/src/metadata.cc +346 -0
- package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/sharp/src/metadata.h +90 -0
- package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/sharp/src/operations.cc +499 -0
- package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/sharp/src/operations.h +137 -0
- package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/sharp/src/pipeline.cc +1814 -0
- package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/sharp/src/pipeline.h +408 -0
- package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/sharp/src/sharp.cc +43 -0
- package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/sharp/src/stats.cc +186 -0
- package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/sharp/src/stats.h +62 -0
- package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/sharp/src/utilities.cc +288 -0
- package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/sharp/src/utilities.h +22 -0
- package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/package-lock.json +529 -0
- package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/package.json +5 -0
- package/skills/aipk_design/{auto_panel_splitter/scripts/auto_panel_splitter_bundle.js → update-requirements-from-design/scripts/split-design-boards-bundle.js} +10 -1
- package/skills/aipk_design/{auto_panel_splitter/scripts/panel_asset_mapper_bundle.js → update-requirements-from-design/scripts/sync-design-to-requirements-bundle.js} +10 -1
- package/skills/aipk_development/GURU_AI.md +1 -1
- package/skills/aipk_init_project/GURU_AI.md +2 -2
- package/skills/aipk_init_project/template/AGENTS_TEMPLATE.md +1 -1
- package/skills/aipk_operations/GURU_AI.md +1 -1
- package/skills/aipk_requirements/GURU_AI.md +2 -2
- package/skills/aipk_requirements/documentation/SKILL.md +23 -1
- package/skills/aipk_skill_generate/GURU_AI.md +1 -1
- package/skills/aipk_tool_prompts/GURU_AI.md +1 -1
- /package/skills/aipk_design/{auto_panel_splitter → update-requirements-from-design}/scripts/auto_panel_splitter.js +0 -0
- /package/skills/aipk_design/{auto_panel_splitter → update-requirements-from-design}/scripts/panel_asset_mapper.js +0 -0
|
@@ -0,0 +1,1130 @@
|
|
|
1
|
+
/*!
|
|
2
|
+
Copyright 2013 Lovell Fuller and others.
|
|
3
|
+
SPDX-License-Identifier: Apache-2.0
|
|
4
|
+
*/
|
|
5
|
+
|
|
6
|
+
#include <algorithm>
|
|
7
|
+
#include <cstdlib>
|
|
8
|
+
#include <map>
|
|
9
|
+
#include <mutex>
|
|
10
|
+
#include <queue>
|
|
11
|
+
#include <string>
|
|
12
|
+
#include <tuple>
|
|
13
|
+
#include <utility>
|
|
14
|
+
#include <vector>
|
|
15
|
+
|
|
16
|
+
#include <napi.h>
|
|
17
|
+
#include <vips/vips8>
|
|
18
|
+
|
|
19
|
+
#include "./common.h"
|
|
20
|
+
|
|
21
|
+
using vips::VImage;
|
|
22
|
+
|
|
23
|
+
namespace sharp {
|
|
24
|
+
|
|
25
|
+
// Convenience methods to access the attributes of a Napi::Object
|
|
26
|
+
bool HasAttr(Napi::Object obj, std::string attr) {
|
|
27
|
+
return obj.Has(attr);
|
|
28
|
+
}
|
|
29
|
+
std::string AttrAsStr(Napi::Object obj, std::string attr) {
|
|
30
|
+
return obj.Get(attr).As<Napi::String>();
|
|
31
|
+
}
|
|
32
|
+
std::string AttrAsStr(Napi::Object obj, unsigned int const attr) {
|
|
33
|
+
return obj.Get(attr).As<Napi::String>();
|
|
34
|
+
}
|
|
35
|
+
uint32_t AttrAsUint32(Napi::Object obj, std::string attr) {
|
|
36
|
+
return obj.Get(attr).As<Napi::Number>().Uint32Value();
|
|
37
|
+
}
|
|
38
|
+
int32_t AttrAsInt32(Napi::Object obj, std::string attr) {
|
|
39
|
+
return obj.Get(attr).As<Napi::Number>().Int32Value();
|
|
40
|
+
}
|
|
41
|
+
int32_t AttrAsInt32(Napi::Object obj, unsigned int const attr) {
|
|
42
|
+
return obj.Get(attr).As<Napi::Number>().Int32Value();
|
|
43
|
+
}
|
|
44
|
+
int64_t AttrAsInt64(Napi::Object obj, std::string attr) {
|
|
45
|
+
return obj.Get(attr).As<Napi::Number>().Int64Value();
|
|
46
|
+
}
|
|
47
|
+
double AttrAsDouble(Napi::Object obj, std::string attr) {
|
|
48
|
+
return obj.Get(attr).As<Napi::Number>().DoubleValue();
|
|
49
|
+
}
|
|
50
|
+
double AttrAsDouble(Napi::Object obj, unsigned int const attr) {
|
|
51
|
+
return obj.Get(attr).As<Napi::Number>().DoubleValue();
|
|
52
|
+
}
|
|
53
|
+
bool AttrAsBool(Napi::Object obj, std::string attr) {
|
|
54
|
+
return obj.Get(attr).As<Napi::Boolean>().Value();
|
|
55
|
+
}
|
|
56
|
+
std::vector<double> AttrAsVectorOfDouble(Napi::Object obj, std::string attr) {
|
|
57
|
+
Napi::Array napiArray = obj.Get(attr).As<Napi::Array>();
|
|
58
|
+
std::vector<double> vectorOfDouble(napiArray.Length());
|
|
59
|
+
for (unsigned int i = 0; i < napiArray.Length(); i++) {
|
|
60
|
+
vectorOfDouble[i] = AttrAsDouble(napiArray, i);
|
|
61
|
+
}
|
|
62
|
+
return vectorOfDouble;
|
|
63
|
+
}
|
|
64
|
+
std::vector<int32_t> AttrAsInt32Vector(Napi::Object obj, std::string attr) {
|
|
65
|
+
Napi::Array array = obj.Get(attr).As<Napi::Array>();
|
|
66
|
+
std::vector<int32_t> vector(array.Length());
|
|
67
|
+
for (unsigned int i = 0; i < array.Length(); i++) {
|
|
68
|
+
vector[i] = AttrAsInt32(array, i);
|
|
69
|
+
}
|
|
70
|
+
return vector;
|
|
71
|
+
}
|
|
72
|
+
|
|
73
|
+
// Create an InputDescriptor instance from a Napi::Object describing an input image
|
|
74
|
+
InputDescriptor* CreateInputDescriptor(Napi::Object input) {
|
|
75
|
+
InputDescriptor *descriptor = new InputDescriptor;
|
|
76
|
+
if (HasAttr(input, "file")) {
|
|
77
|
+
descriptor->file = AttrAsStr(input, "file");
|
|
78
|
+
} else if (HasAttr(input, "buffer")) {
|
|
79
|
+
Napi::Buffer<char> buffer = input.Get("buffer").As<Napi::Buffer<char>>();
|
|
80
|
+
descriptor->bufferLength = buffer.Length();
|
|
81
|
+
descriptor->buffer = buffer.Data();
|
|
82
|
+
descriptor->isBuffer = true;
|
|
83
|
+
}
|
|
84
|
+
descriptor->failOn = AttrAsEnum<VipsFailOn>(input, "failOn", VIPS_TYPE_FAIL_ON);
|
|
85
|
+
// Density for vector-based input
|
|
86
|
+
if (HasAttr(input, "density")) {
|
|
87
|
+
descriptor->density = AttrAsDouble(input, "density");
|
|
88
|
+
}
|
|
89
|
+
// Should we ignore any embedded ICC profile
|
|
90
|
+
if (HasAttr(input, "ignoreIcc")) {
|
|
91
|
+
descriptor->ignoreIcc = AttrAsBool(input, "ignoreIcc");
|
|
92
|
+
}
|
|
93
|
+
// Raw pixel input
|
|
94
|
+
if (HasAttr(input, "rawChannels")) {
|
|
95
|
+
descriptor->rawDepth = AttrAsEnum<VipsBandFormat>(input, "rawDepth", VIPS_TYPE_BAND_FORMAT);
|
|
96
|
+
descriptor->rawChannels = AttrAsUint32(input, "rawChannels");
|
|
97
|
+
descriptor->rawWidth = AttrAsUint32(input, "rawWidth");
|
|
98
|
+
descriptor->rawHeight = AttrAsUint32(input, "rawHeight");
|
|
99
|
+
descriptor->rawPremultiplied = AttrAsBool(input, "rawPremultiplied");
|
|
100
|
+
descriptor->rawPageHeight = AttrAsUint32(input, "rawPageHeight");
|
|
101
|
+
}
|
|
102
|
+
// Multi-page input (GIF, TIFF, PDF)
|
|
103
|
+
if (HasAttr(input, "pages")) {
|
|
104
|
+
descriptor->pages = AttrAsInt32(input, "pages");
|
|
105
|
+
}
|
|
106
|
+
if (HasAttr(input, "page")) {
|
|
107
|
+
descriptor->page = AttrAsUint32(input, "page");
|
|
108
|
+
}
|
|
109
|
+
// SVG
|
|
110
|
+
if (HasAttr(input, "svgStylesheet")) {
|
|
111
|
+
descriptor->svgStylesheet = AttrAsStr(input, "svgStylesheet");
|
|
112
|
+
}
|
|
113
|
+
if (HasAttr(input, "svgHighBitdepth")) {
|
|
114
|
+
descriptor->svgHighBitdepth = AttrAsBool(input, "svgHighBitdepth");
|
|
115
|
+
}
|
|
116
|
+
// Multi-level input (OpenSlide)
|
|
117
|
+
if (HasAttr(input, "openSlideLevel")) {
|
|
118
|
+
descriptor->openSlideLevel = AttrAsUint32(input, "openSlideLevel");
|
|
119
|
+
}
|
|
120
|
+
// subIFD (OME-TIFF)
|
|
121
|
+
if (HasAttr(input, "subifd")) {
|
|
122
|
+
descriptor->tiffSubifd = AttrAsInt32(input, "tiffSubifd");
|
|
123
|
+
}
|
|
124
|
+
// // PDF background color
|
|
125
|
+
if (HasAttr(input, "pdfBackground")) {
|
|
126
|
+
descriptor->pdfBackground = AttrAsVectorOfDouble(input, "pdfBackground");
|
|
127
|
+
}
|
|
128
|
+
// Use JPEG 2000 oneshot mode?
|
|
129
|
+
if (HasAttr(input, "jp2Oneshot")) {
|
|
130
|
+
descriptor->jp2Oneshot = AttrAsBool(input, "jp2Oneshot");
|
|
131
|
+
}
|
|
132
|
+
// Create new image
|
|
133
|
+
if (HasAttr(input, "createChannels")) {
|
|
134
|
+
descriptor->createChannels = AttrAsUint32(input, "createChannels");
|
|
135
|
+
descriptor->createWidth = AttrAsUint32(input, "createWidth");
|
|
136
|
+
descriptor->createHeight = AttrAsUint32(input, "createHeight");
|
|
137
|
+
descriptor->createPageHeight = AttrAsUint32(input, "createPageHeight");
|
|
138
|
+
if (HasAttr(input, "createNoiseType")) {
|
|
139
|
+
descriptor->createNoiseType = AttrAsStr(input, "createNoiseType");
|
|
140
|
+
descriptor->createNoiseMean = AttrAsDouble(input, "createNoiseMean");
|
|
141
|
+
descriptor->createNoiseSigma = AttrAsDouble(input, "createNoiseSigma");
|
|
142
|
+
} else {
|
|
143
|
+
descriptor->createBackground = AttrAsVectorOfDouble(input, "createBackground");
|
|
144
|
+
}
|
|
145
|
+
}
|
|
146
|
+
// Create new image with text
|
|
147
|
+
if (HasAttr(input, "textValue")) {
|
|
148
|
+
descriptor->textValue = AttrAsStr(input, "textValue");
|
|
149
|
+
if (HasAttr(input, "textFont")) {
|
|
150
|
+
descriptor->textFont = AttrAsStr(input, "textFont");
|
|
151
|
+
}
|
|
152
|
+
if (HasAttr(input, "textFontfile")) {
|
|
153
|
+
descriptor->textFontfile = AttrAsStr(input, "textFontfile");
|
|
154
|
+
}
|
|
155
|
+
if (HasAttr(input, "textWidth")) {
|
|
156
|
+
descriptor->textWidth = AttrAsUint32(input, "textWidth");
|
|
157
|
+
}
|
|
158
|
+
if (HasAttr(input, "textHeight")) {
|
|
159
|
+
descriptor->textHeight = AttrAsUint32(input, "textHeight");
|
|
160
|
+
}
|
|
161
|
+
if (HasAttr(input, "textAlign")) {
|
|
162
|
+
descriptor->textAlign = AttrAsEnum<VipsAlign>(input, "textAlign", VIPS_TYPE_ALIGN);
|
|
163
|
+
}
|
|
164
|
+
if (HasAttr(input, "textJustify")) {
|
|
165
|
+
descriptor->textJustify = AttrAsBool(input, "textJustify");
|
|
166
|
+
}
|
|
167
|
+
if (HasAttr(input, "textDpi")) {
|
|
168
|
+
descriptor->textDpi = AttrAsUint32(input, "textDpi");
|
|
169
|
+
}
|
|
170
|
+
if (HasAttr(input, "textRgba")) {
|
|
171
|
+
descriptor->textRgba = AttrAsBool(input, "textRgba");
|
|
172
|
+
}
|
|
173
|
+
if (HasAttr(input, "textSpacing")) {
|
|
174
|
+
descriptor->textSpacing = AttrAsUint32(input, "textSpacing");
|
|
175
|
+
}
|
|
176
|
+
if (HasAttr(input, "textWrap")) {
|
|
177
|
+
descriptor->textWrap = AttrAsEnum<VipsTextWrap>(input, "textWrap", VIPS_TYPE_TEXT_WRAP);
|
|
178
|
+
}
|
|
179
|
+
}
|
|
180
|
+
// Join images together
|
|
181
|
+
if (HasAttr(input, "joinAnimated")) {
|
|
182
|
+
descriptor->joinAnimated = AttrAsBool(input, "joinAnimated");
|
|
183
|
+
}
|
|
184
|
+
if (HasAttr(input, "joinAcross")) {
|
|
185
|
+
descriptor->joinAcross = AttrAsUint32(input, "joinAcross");
|
|
186
|
+
}
|
|
187
|
+
if (HasAttr(input, "joinShim")) {
|
|
188
|
+
descriptor->joinShim = AttrAsUint32(input, "joinShim");
|
|
189
|
+
}
|
|
190
|
+
if (HasAttr(input, "joinBackground")) {
|
|
191
|
+
descriptor->joinBackground = AttrAsVectorOfDouble(input, "joinBackground");
|
|
192
|
+
}
|
|
193
|
+
if (HasAttr(input, "joinHalign")) {
|
|
194
|
+
descriptor->joinHalign = AttrAsEnum<VipsAlign>(input, "joinHalign", VIPS_TYPE_ALIGN);
|
|
195
|
+
}
|
|
196
|
+
if (HasAttr(input, "joinValign")) {
|
|
197
|
+
descriptor->joinValign = AttrAsEnum<VipsAlign>(input, "joinValign", VIPS_TYPE_ALIGN);
|
|
198
|
+
}
|
|
199
|
+
// Limit input images to a given number of pixels, where pixels = width * height
|
|
200
|
+
descriptor->limitInputPixels = static_cast<uint64_t>(AttrAsInt64(input, "limitInputPixels"));
|
|
201
|
+
if (HasAttr(input, "access")) {
|
|
202
|
+
descriptor->access = AttrAsBool(input, "sequentialRead") ? VIPS_ACCESS_SEQUENTIAL : VIPS_ACCESS_RANDOM;
|
|
203
|
+
}
|
|
204
|
+
// Remove safety features and allow unlimited input
|
|
205
|
+
descriptor->unlimited = AttrAsBool(input, "unlimited");
|
|
206
|
+
// Use the EXIF orientation to auto orient the image
|
|
207
|
+
descriptor->autoOrient = AttrAsBool(input, "autoOrient");
|
|
208
|
+
return descriptor;
|
|
209
|
+
}
|
|
210
|
+
|
|
211
|
+
// How many tasks are in the queue?
|
|
212
|
+
std::atomic<int> counterQueue{0};
|
|
213
|
+
|
|
214
|
+
// How many tasks are being processed?
|
|
215
|
+
std::atomic<int> counterProcess{0};
|
|
216
|
+
|
|
217
|
+
// Filename extension checkers
|
|
218
|
+
static bool EndsWith(std::string const &str, std::string const &end) {
|
|
219
|
+
return str.length() >= end.length() && 0 == str.compare(str.length() - end.length(), end.length(), end);
|
|
220
|
+
}
|
|
221
|
+
bool IsJpeg(std::string const &str) {
|
|
222
|
+
return EndsWith(str, ".jpg") || EndsWith(str, ".jpeg") || EndsWith(str, ".JPG") || EndsWith(str, ".JPEG");
|
|
223
|
+
}
|
|
224
|
+
bool IsPng(std::string const &str) {
|
|
225
|
+
return EndsWith(str, ".png") || EndsWith(str, ".PNG");
|
|
226
|
+
}
|
|
227
|
+
bool IsWebp(std::string const &str) {
|
|
228
|
+
return EndsWith(str, ".webp") || EndsWith(str, ".WEBP");
|
|
229
|
+
}
|
|
230
|
+
bool IsGif(std::string const &str) {
|
|
231
|
+
return EndsWith(str, ".gif") || EndsWith(str, ".GIF");
|
|
232
|
+
}
|
|
233
|
+
bool IsJp2(std::string const &str) {
|
|
234
|
+
return EndsWith(str, ".jp2") || EndsWith(str, ".jpx") || EndsWith(str, ".j2k") || EndsWith(str, ".j2c")
|
|
235
|
+
|| EndsWith(str, ".JP2") || EndsWith(str, ".JPX") || EndsWith(str, ".J2K") || EndsWith(str, ".J2C");
|
|
236
|
+
}
|
|
237
|
+
bool IsTiff(std::string const &str) {
|
|
238
|
+
return EndsWith(str, ".tif") || EndsWith(str, ".tiff") || EndsWith(str, ".TIF") || EndsWith(str, ".TIFF");
|
|
239
|
+
}
|
|
240
|
+
bool IsHeic(std::string const &str) {
|
|
241
|
+
return EndsWith(str, ".heic") || EndsWith(str, ".HEIC");
|
|
242
|
+
}
|
|
243
|
+
bool IsHeif(std::string const &str) {
|
|
244
|
+
return EndsWith(str, ".heif") || EndsWith(str, ".HEIF") || IsHeic(str) || IsAvif(str);
|
|
245
|
+
}
|
|
246
|
+
bool IsAvif(std::string const &str) {
|
|
247
|
+
return EndsWith(str, ".avif") || EndsWith(str, ".AVIF");
|
|
248
|
+
}
|
|
249
|
+
bool IsJxl(std::string const &str) {
|
|
250
|
+
return EndsWith(str, ".jxl") || EndsWith(str, ".JXL");
|
|
251
|
+
}
|
|
252
|
+
bool IsDz(std::string const &str) {
|
|
253
|
+
return EndsWith(str, ".dzi") || EndsWith(str, ".DZI");
|
|
254
|
+
}
|
|
255
|
+
bool IsDzZip(std::string const &str) {
|
|
256
|
+
return EndsWith(str, ".zip") || EndsWith(str, ".ZIP") || EndsWith(str, ".szi") || EndsWith(str, ".SZI");
|
|
257
|
+
}
|
|
258
|
+
bool IsV(std::string const &str) {
|
|
259
|
+
return EndsWith(str, ".v") || EndsWith(str, ".V") || EndsWith(str, ".vips") || EndsWith(str, ".VIPS");
|
|
260
|
+
}
|
|
261
|
+
|
|
262
|
+
/*
|
|
263
|
+
Trim space from end of string.
|
|
264
|
+
*/
|
|
265
|
+
std::string TrimEnd(std::string const &str) {
|
|
266
|
+
return str.substr(0, str.find_last_not_of(" \n\r\f") + 1);
|
|
267
|
+
}
|
|
268
|
+
|
|
269
|
+
/*
|
|
270
|
+
Provide a string identifier for the given image type.
|
|
271
|
+
*/
|
|
272
|
+
std::string ImageTypeId(ImageType const imageType) {
|
|
273
|
+
std::string id;
|
|
274
|
+
switch (imageType) {
|
|
275
|
+
case ImageType::JPEG: id = "jpeg"; break;
|
|
276
|
+
case ImageType::PNG: id = "png"; break;
|
|
277
|
+
case ImageType::WEBP: id = "webp"; break;
|
|
278
|
+
case ImageType::TIFF: id = "tiff"; break;
|
|
279
|
+
case ImageType::GIF: id = "gif"; break;
|
|
280
|
+
case ImageType::JP2: id = "jp2"; break;
|
|
281
|
+
case ImageType::SVG: id = "svg"; break;
|
|
282
|
+
case ImageType::HEIF: id = "heif"; break;
|
|
283
|
+
case ImageType::PDF: id = "pdf"; break;
|
|
284
|
+
case ImageType::MAGICK: id = "magick"; break;
|
|
285
|
+
case ImageType::OPENSLIDE: id = "openslide"; break;
|
|
286
|
+
case ImageType::PPM: id = "ppm"; break;
|
|
287
|
+
case ImageType::FITS: id = "fits"; break;
|
|
288
|
+
case ImageType::EXR: id = "exr"; break;
|
|
289
|
+
case ImageType::JXL: id = "jxl"; break;
|
|
290
|
+
case ImageType::RAD: id = "rad"; break;
|
|
291
|
+
case ImageType::DCRAW: id = "dcraw"; break;
|
|
292
|
+
case ImageType::VIPS: id = "vips"; break;
|
|
293
|
+
case ImageType::RAW: id = "raw"; break;
|
|
294
|
+
case ImageType::UNKNOWN: id = "unknown"; break;
|
|
295
|
+
case ImageType::MISSING: id = "missing"; break;
|
|
296
|
+
}
|
|
297
|
+
return id;
|
|
298
|
+
}
|
|
299
|
+
|
|
300
|
+
/**
|
|
301
|
+
* Regenerate this table with something like:
|
|
302
|
+
*
|
|
303
|
+
* $ vips -l foreign | grep -i load | awk '{ print $2, $1; }'
|
|
304
|
+
*
|
|
305
|
+
* Plus a bit of editing.
|
|
306
|
+
*/
|
|
307
|
+
std::map<std::string, ImageType> loaderToType = {
|
|
308
|
+
{ "VipsForeignLoadJpegFile", ImageType::JPEG },
|
|
309
|
+
{ "VipsForeignLoadJpegBuffer", ImageType::JPEG },
|
|
310
|
+
{ "VipsForeignLoadPngFile", ImageType::PNG },
|
|
311
|
+
{ "VipsForeignLoadPngBuffer", ImageType::PNG },
|
|
312
|
+
{ "VipsForeignLoadWebpFile", ImageType::WEBP },
|
|
313
|
+
{ "VipsForeignLoadWebpBuffer", ImageType::WEBP },
|
|
314
|
+
{ "VipsForeignLoadTiffFile", ImageType::TIFF },
|
|
315
|
+
{ "VipsForeignLoadTiffBuffer", ImageType::TIFF },
|
|
316
|
+
{ "VipsForeignLoadGifFile", ImageType::GIF },
|
|
317
|
+
{ "VipsForeignLoadGifBuffer", ImageType::GIF },
|
|
318
|
+
{ "VipsForeignLoadNsgifFile", ImageType::GIF },
|
|
319
|
+
{ "VipsForeignLoadNsgifBuffer", ImageType::GIF },
|
|
320
|
+
{ "VipsForeignLoadJp2kBuffer", ImageType::JP2 },
|
|
321
|
+
{ "VipsForeignLoadJp2kFile", ImageType::JP2 },
|
|
322
|
+
{ "VipsForeignLoadSvgFile", ImageType::SVG },
|
|
323
|
+
{ "VipsForeignLoadSvgBuffer", ImageType::SVG },
|
|
324
|
+
{ "VipsForeignLoadHeifFile", ImageType::HEIF },
|
|
325
|
+
{ "VipsForeignLoadHeifBuffer", ImageType::HEIF },
|
|
326
|
+
{ "VipsForeignLoadPdfFile", ImageType::PDF },
|
|
327
|
+
{ "VipsForeignLoadPdfBuffer", ImageType::PDF },
|
|
328
|
+
{ "VipsForeignLoadMagickFile", ImageType::MAGICK },
|
|
329
|
+
{ "VipsForeignLoadMagickBuffer", ImageType::MAGICK },
|
|
330
|
+
{ "VipsForeignLoadMagick7File", ImageType::MAGICK },
|
|
331
|
+
{ "VipsForeignLoadMagick7Buffer", ImageType::MAGICK },
|
|
332
|
+
{ "VipsForeignLoadOpenslideFile", ImageType::OPENSLIDE },
|
|
333
|
+
{ "VipsForeignLoadPpmFile", ImageType::PPM },
|
|
334
|
+
{ "VipsForeignLoadFitsFile", ImageType::FITS },
|
|
335
|
+
{ "VipsForeignLoadOpenexr", ImageType::EXR },
|
|
336
|
+
{ "VipsForeignLoadJxlFile", ImageType::JXL },
|
|
337
|
+
{ "VipsForeignLoadJxlBuffer", ImageType::JXL },
|
|
338
|
+
{ "VipsForeignLoadRadFile", ImageType::RAD },
|
|
339
|
+
{ "VipsForeignLoadRadBuffer", ImageType::RAD },
|
|
340
|
+
{ "VipsForeignLoadDcRawFile", ImageType::DCRAW },
|
|
341
|
+
{ "VipsForeignLoadDcRawBuffer", ImageType::DCRAW },
|
|
342
|
+
{ "VipsForeignLoadVips", ImageType::VIPS },
|
|
343
|
+
{ "VipsForeignLoadVipsFile", ImageType::VIPS },
|
|
344
|
+
{ "VipsForeignLoadRaw", ImageType::RAW }
|
|
345
|
+
};
|
|
346
|
+
|
|
347
|
+
/*
|
|
348
|
+
Determine image format of a buffer.
|
|
349
|
+
*/
|
|
350
|
+
ImageType DetermineImageType(void *buffer, size_t const length) {
|
|
351
|
+
ImageType imageType = ImageType::UNKNOWN;
|
|
352
|
+
char const *load = vips_foreign_find_load_buffer(buffer, length);
|
|
353
|
+
if (load != nullptr) {
|
|
354
|
+
auto it = loaderToType.find(load);
|
|
355
|
+
if (it != loaderToType.end()) {
|
|
356
|
+
imageType = it->second;
|
|
357
|
+
}
|
|
358
|
+
}
|
|
359
|
+
return imageType;
|
|
360
|
+
}
|
|
361
|
+
|
|
362
|
+
/*
|
|
363
|
+
Determine image format, reads the first few bytes of the file
|
|
364
|
+
*/
|
|
365
|
+
ImageType DetermineImageType(char const *file) {
|
|
366
|
+
ImageType imageType = ImageType::UNKNOWN;
|
|
367
|
+
char const *load = vips_foreign_find_load(file);
|
|
368
|
+
if (load != nullptr) {
|
|
369
|
+
auto it = loaderToType.find(load);
|
|
370
|
+
if (it != loaderToType.end()) {
|
|
371
|
+
imageType = it->second;
|
|
372
|
+
}
|
|
373
|
+
} else {
|
|
374
|
+
if (EndsWith(vips::VError().what(), " does not exist\n")) {
|
|
375
|
+
imageType = ImageType::MISSING;
|
|
376
|
+
}
|
|
377
|
+
}
|
|
378
|
+
return imageType;
|
|
379
|
+
}
|
|
380
|
+
|
|
381
|
+
/*
|
|
382
|
+
Does this image type support multiple pages?
|
|
383
|
+
*/
|
|
384
|
+
bool ImageTypeSupportsPage(ImageType imageType) {
|
|
385
|
+
return
|
|
386
|
+
imageType == ImageType::WEBP ||
|
|
387
|
+
imageType == ImageType::MAGICK ||
|
|
388
|
+
imageType == ImageType::GIF ||
|
|
389
|
+
imageType == ImageType::JP2 ||
|
|
390
|
+
imageType == ImageType::TIFF ||
|
|
391
|
+
imageType == ImageType::HEIF ||
|
|
392
|
+
imageType == ImageType::PDF;
|
|
393
|
+
}
|
|
394
|
+
|
|
395
|
+
/*
|
|
396
|
+
Does this image type support removal of safety limits?
|
|
397
|
+
*/
|
|
398
|
+
bool ImageTypeSupportsUnlimited(ImageType imageType) {
|
|
399
|
+
return
|
|
400
|
+
imageType == ImageType::JPEG ||
|
|
401
|
+
imageType == ImageType::PNG ||
|
|
402
|
+
imageType == ImageType::SVG ||
|
|
403
|
+
imageType == ImageType::TIFF ||
|
|
404
|
+
imageType == ImageType::HEIF;
|
|
405
|
+
}
|
|
406
|
+
|
|
407
|
+
/*
|
|
408
|
+
Format-specific options builder
|
|
409
|
+
*/
|
|
410
|
+
vips::VOption* GetOptionsForImageType(ImageType imageType, InputDescriptor *descriptor) {
|
|
411
|
+
vips::VOption *option = VImage::option()
|
|
412
|
+
->set("access", descriptor->access)
|
|
413
|
+
->set("fail_on", descriptor->failOn);
|
|
414
|
+
if (descriptor->unlimited && ImageTypeSupportsUnlimited(imageType)) {
|
|
415
|
+
option->set("unlimited", true);
|
|
416
|
+
}
|
|
417
|
+
if (ImageTypeSupportsPage(imageType)) {
|
|
418
|
+
option->set("n", descriptor->pages);
|
|
419
|
+
option->set("page", descriptor->page);
|
|
420
|
+
}
|
|
421
|
+
switch (imageType) {
|
|
422
|
+
case ImageType::SVG:
|
|
423
|
+
option->set("dpi", descriptor->density)
|
|
424
|
+
->set("stylesheet", descriptor->svgStylesheet.data())
|
|
425
|
+
->set("high_bitdepth", descriptor->svgHighBitdepth);
|
|
426
|
+
break;
|
|
427
|
+
case ImageType::TIFF:
|
|
428
|
+
option->set("subifd", descriptor->tiffSubifd);
|
|
429
|
+
break;
|
|
430
|
+
case ImageType::PDF:
|
|
431
|
+
option->set("dpi", descriptor->density)
|
|
432
|
+
->set("background", descriptor->pdfBackground);
|
|
433
|
+
break;
|
|
434
|
+
case ImageType::OPENSLIDE:
|
|
435
|
+
option->set("level", descriptor->openSlideLevel);
|
|
436
|
+
break;
|
|
437
|
+
case ImageType::JP2:
|
|
438
|
+
option->set("oneshot", descriptor->jp2Oneshot);
|
|
439
|
+
break;
|
|
440
|
+
case ImageType::MAGICK:
|
|
441
|
+
option->set("density", std::to_string(descriptor->density).data());
|
|
442
|
+
break;
|
|
443
|
+
default:
|
|
444
|
+
break;
|
|
445
|
+
}
|
|
446
|
+
return option;
|
|
447
|
+
}
|
|
448
|
+
|
|
449
|
+
/*
|
|
450
|
+
Open an image from the given InputDescriptor (filesystem, compressed buffer, raw pixel data)
|
|
451
|
+
*/
|
|
452
|
+
std::tuple<VImage, ImageType> OpenInput(InputDescriptor *descriptor) {
|
|
453
|
+
VImage image;
|
|
454
|
+
ImageType imageType;
|
|
455
|
+
if (descriptor->isBuffer) {
|
|
456
|
+
if (descriptor->rawChannels > 0) {
|
|
457
|
+
// Raw, uncompressed pixel data
|
|
458
|
+
bool const is8bit = vips_band_format_is8bit(descriptor->rawDepth);
|
|
459
|
+
image = VImage::new_from_memory(descriptor->buffer, descriptor->bufferLength,
|
|
460
|
+
descriptor->rawWidth, descriptor->rawHeight, descriptor->rawChannels, descriptor->rawDepth);
|
|
461
|
+
if (descriptor->rawChannels < 3) {
|
|
462
|
+
image.get_image()->Type = is8bit ? VIPS_INTERPRETATION_B_W : VIPS_INTERPRETATION_GREY16;
|
|
463
|
+
} else {
|
|
464
|
+
image.get_image()->Type = is8bit ? VIPS_INTERPRETATION_sRGB : VIPS_INTERPRETATION_RGB16;
|
|
465
|
+
}
|
|
466
|
+
if (descriptor->rawPageHeight > 0) {
|
|
467
|
+
image.set(VIPS_META_PAGE_HEIGHT, descriptor->rawPageHeight);
|
|
468
|
+
image.set(VIPS_META_N_PAGES, static_cast<int>(descriptor->rawHeight / descriptor->rawPageHeight));
|
|
469
|
+
}
|
|
470
|
+
if (descriptor->rawPremultiplied) {
|
|
471
|
+
image = image.unpremultiply();
|
|
472
|
+
}
|
|
473
|
+
imageType = ImageType::RAW;
|
|
474
|
+
} else {
|
|
475
|
+
// Compressed data
|
|
476
|
+
imageType = DetermineImageType(descriptor->buffer, descriptor->bufferLength);
|
|
477
|
+
if (imageType != ImageType::UNKNOWN) {
|
|
478
|
+
try {
|
|
479
|
+
vips::VOption *option = GetOptionsForImageType(imageType, descriptor);
|
|
480
|
+
image = VImage::new_from_buffer(descriptor->buffer, descriptor->bufferLength, nullptr, option);
|
|
481
|
+
if (imageType == ImageType::SVG || imageType == ImageType::PDF || imageType == ImageType::MAGICK) {
|
|
482
|
+
image = SetDensity(image, descriptor->density);
|
|
483
|
+
}
|
|
484
|
+
} catch (vips::VError const &err) {
|
|
485
|
+
throw vips::VError(std::string("Input buffer has corrupt header: ") + err.what());
|
|
486
|
+
}
|
|
487
|
+
} else {
|
|
488
|
+
throw vips::VError("Input buffer contains unsupported image format");
|
|
489
|
+
}
|
|
490
|
+
}
|
|
491
|
+
} else {
|
|
492
|
+
int const channels = descriptor->createChannels;
|
|
493
|
+
if (channels > 0) {
|
|
494
|
+
// Create new image
|
|
495
|
+
if (descriptor->createNoiseType == "gaussian") {
|
|
496
|
+
std::vector<VImage> bands = {};
|
|
497
|
+
bands.reserve(channels);
|
|
498
|
+
for (int _band = 0; _band < channels; _band++) {
|
|
499
|
+
bands.push_back(VImage::gaussnoise(descriptor->createWidth, descriptor->createHeight, VImage::option()
|
|
500
|
+
->set("mean", descriptor->createNoiseMean)
|
|
501
|
+
->set("sigma", descriptor->createNoiseSigma)));
|
|
502
|
+
}
|
|
503
|
+
image = VImage::bandjoin(bands).copy(VImage::option()->set("interpretation",
|
|
504
|
+
channels < 3 ? VIPS_INTERPRETATION_B_W: VIPS_INTERPRETATION_sRGB));
|
|
505
|
+
} else {
|
|
506
|
+
std::vector<double> background = {
|
|
507
|
+
descriptor->createBackground[0],
|
|
508
|
+
descriptor->createBackground[1],
|
|
509
|
+
descriptor->createBackground[2]
|
|
510
|
+
};
|
|
511
|
+
if (channels == 4) {
|
|
512
|
+
background.push_back(descriptor->createBackground[3]);
|
|
513
|
+
}
|
|
514
|
+
image = VImage::new_matrix(descriptor->createWidth, descriptor->createHeight)
|
|
515
|
+
.copy(VImage::option()->set("interpretation",
|
|
516
|
+
channels < 3 ? VIPS_INTERPRETATION_B_W : VIPS_INTERPRETATION_sRGB))
|
|
517
|
+
.new_from_image(background);
|
|
518
|
+
}
|
|
519
|
+
if (descriptor->createPageHeight > 0) {
|
|
520
|
+
image.set(VIPS_META_PAGE_HEIGHT, descriptor->createPageHeight);
|
|
521
|
+
image.set(VIPS_META_N_PAGES, static_cast<int>(descriptor->createHeight / descriptor->createPageHeight));
|
|
522
|
+
}
|
|
523
|
+
image = image.cast(VIPS_FORMAT_UCHAR);
|
|
524
|
+
imageType = ImageType::RAW;
|
|
525
|
+
} else if (descriptor->textValue.length() > 0) {
|
|
526
|
+
// Create a new image with text
|
|
527
|
+
vips::VOption *textOptions = VImage::option()
|
|
528
|
+
->set("align", descriptor->textAlign)
|
|
529
|
+
->set("justify", descriptor->textJustify)
|
|
530
|
+
->set("rgba", descriptor->textRgba)
|
|
531
|
+
->set("spacing", descriptor->textSpacing)
|
|
532
|
+
->set("wrap", descriptor->textWrap)
|
|
533
|
+
->set("autofit_dpi", &descriptor->textAutofitDpi);
|
|
534
|
+
if (descriptor->textWidth > 0) {
|
|
535
|
+
textOptions->set("width", descriptor->textWidth);
|
|
536
|
+
}
|
|
537
|
+
// Ignore dpi if height is set
|
|
538
|
+
if (descriptor->textWidth > 0 && descriptor->textHeight > 0) {
|
|
539
|
+
textOptions->set("height", descriptor->textHeight);
|
|
540
|
+
} else if (descriptor->textDpi > 0) {
|
|
541
|
+
textOptions->set("dpi", descriptor->textDpi);
|
|
542
|
+
}
|
|
543
|
+
if (descriptor->textFont.length() > 0) {
|
|
544
|
+
textOptions->set("font", const_cast<char*>(descriptor->textFont.data()));
|
|
545
|
+
}
|
|
546
|
+
if (descriptor->textFontfile.length() > 0) {
|
|
547
|
+
textOptions->set("fontfile", const_cast<char*>(descriptor->textFontfile.data()));
|
|
548
|
+
}
|
|
549
|
+
image = VImage::text(const_cast<char *>(descriptor->textValue.data()), textOptions);
|
|
550
|
+
if (!descriptor->textRgba) {
|
|
551
|
+
image = image.copy(VImage::option()->set("interpretation", VIPS_INTERPRETATION_B_W));
|
|
552
|
+
}
|
|
553
|
+
imageType = ImageType::RAW;
|
|
554
|
+
} else {
|
|
555
|
+
// From filesystem
|
|
556
|
+
imageType = DetermineImageType(descriptor->file.data());
|
|
557
|
+
if (imageType == ImageType::MISSING) {
|
|
558
|
+
if (descriptor->file.find("<svg") != std::string::npos) {
|
|
559
|
+
throw vips::VError("Input file is missing, did you mean "
|
|
560
|
+
"sharp(Buffer.from('" + descriptor->file.substr(0, 8) + "...')?");
|
|
561
|
+
}
|
|
562
|
+
throw vips::VError("Input file is missing: " + descriptor->file);
|
|
563
|
+
}
|
|
564
|
+
if (imageType != ImageType::UNKNOWN) {
|
|
565
|
+
try {
|
|
566
|
+
vips::VOption *option = GetOptionsForImageType(imageType, descriptor);
|
|
567
|
+
image = VImage::new_from_file(descriptor->file.data(), option);
|
|
568
|
+
if (imageType == ImageType::SVG || imageType == ImageType::PDF || imageType == ImageType::MAGICK) {
|
|
569
|
+
image = SetDensity(image, descriptor->density);
|
|
570
|
+
}
|
|
571
|
+
} catch (vips::VError const &err) {
|
|
572
|
+
throw vips::VError(std::string("Input file has corrupt header: ") + err.what());
|
|
573
|
+
}
|
|
574
|
+
} else {
|
|
575
|
+
throw vips::VError("Input file contains unsupported image format");
|
|
576
|
+
}
|
|
577
|
+
}
|
|
578
|
+
}
|
|
579
|
+
|
|
580
|
+
// Limit input images to a given number of pixels, where pixels = width * height
|
|
581
|
+
if (descriptor->limitInputPixels > 0 &&
|
|
582
|
+
static_cast<uint64_t>(image.width()) * image.height() > descriptor->limitInputPixels) {
|
|
583
|
+
throw vips::VError("Input image exceeds pixel limit");
|
|
584
|
+
}
|
|
585
|
+
return std::make_tuple(image, imageType);
|
|
586
|
+
}
|
|
587
|
+
|
|
588
|
+
/*
|
|
589
|
+
Does this image have an embedded profile?
|
|
590
|
+
*/
|
|
591
|
+
bool HasProfile(VImage image) {
|
|
592
|
+
return image.get_typeof(VIPS_META_ICC_NAME) == VIPS_TYPE_BLOB;
|
|
593
|
+
}
|
|
594
|
+
|
|
595
|
+
/*
|
|
596
|
+
Get copy of embedded profile.
|
|
597
|
+
*/
|
|
598
|
+
std::pair<char*, size_t> GetProfile(VImage image) {
|
|
599
|
+
std::pair<char*, size_t> icc(nullptr, 0);
|
|
600
|
+
if (HasProfile(image)) {
|
|
601
|
+
size_t length;
|
|
602
|
+
const void *data = image.get_blob(VIPS_META_ICC_NAME, &length);
|
|
603
|
+
icc.first = static_cast<char*>(g_malloc(length));
|
|
604
|
+
icc.second = length;
|
|
605
|
+
memcpy(icc.first, data, length);
|
|
606
|
+
}
|
|
607
|
+
return icc;
|
|
608
|
+
}
|
|
609
|
+
|
|
610
|
+
/*
|
|
611
|
+
Set embedded profile.
|
|
612
|
+
*/
|
|
613
|
+
VImage SetProfile(VImage image, std::pair<char*, size_t> icc) {
|
|
614
|
+
if (icc.first != nullptr) {
|
|
615
|
+
image = image.copy();
|
|
616
|
+
image.set(VIPS_META_ICC_NAME, reinterpret_cast<VipsCallbackFn>(vips_area_free_cb), icc.first, icc.second);
|
|
617
|
+
}
|
|
618
|
+
return image;
|
|
619
|
+
}
|
|
620
|
+
|
|
621
|
+
static void* RemoveExifCallback(VipsImage *image, char const *field, GValue *value, void *data) {
|
|
622
|
+
std::vector<std::string> *fieldNames = static_cast<std::vector<std::string> *>(data);
|
|
623
|
+
std::string fieldName(field);
|
|
624
|
+
if (fieldName.substr(0, 8) == ("exif-ifd")) {
|
|
625
|
+
fieldNames->push_back(fieldName);
|
|
626
|
+
}
|
|
627
|
+
return nullptr;
|
|
628
|
+
}
|
|
629
|
+
|
|
630
|
+
/*
|
|
631
|
+
Remove all EXIF-related image fields.
|
|
632
|
+
*/
|
|
633
|
+
VImage RemoveExif(VImage image) {
|
|
634
|
+
std::vector<std::string> fieldNames;
|
|
635
|
+
vips_image_map(image.get_image(), static_cast<VipsImageMapFn>(RemoveExifCallback), &fieldNames);
|
|
636
|
+
for (const auto& f : fieldNames) {
|
|
637
|
+
image.remove(f.data());
|
|
638
|
+
}
|
|
639
|
+
return image;
|
|
640
|
+
}
|
|
641
|
+
|
|
642
|
+
/*
|
|
643
|
+
Get EXIF Orientation of image, if any.
|
|
644
|
+
*/
|
|
645
|
+
int ExifOrientation(VImage image) {
|
|
646
|
+
int orientation = 0;
|
|
647
|
+
if (image.get_typeof(VIPS_META_ORIENTATION) != 0) {
|
|
648
|
+
orientation = image.get_int(VIPS_META_ORIENTATION);
|
|
649
|
+
}
|
|
650
|
+
return orientation;
|
|
651
|
+
}
|
|
652
|
+
|
|
653
|
+
/*
|
|
654
|
+
Set EXIF Orientation of image.
|
|
655
|
+
*/
|
|
656
|
+
VImage SetExifOrientation(VImage image, int const orientation) {
|
|
657
|
+
VImage copy = image.copy();
|
|
658
|
+
copy.set(VIPS_META_ORIENTATION, orientation);
|
|
659
|
+
return copy;
|
|
660
|
+
}
|
|
661
|
+
|
|
662
|
+
/*
|
|
663
|
+
Remove EXIF Orientation from image.
|
|
664
|
+
*/
|
|
665
|
+
VImage RemoveExifOrientation(VImage image) {
|
|
666
|
+
VImage copy = image.copy();
|
|
667
|
+
copy.remove(VIPS_META_ORIENTATION);
|
|
668
|
+
copy.remove("exif-ifd0-Orientation");
|
|
669
|
+
return copy;
|
|
670
|
+
}
|
|
671
|
+
|
|
672
|
+
/*
|
|
673
|
+
Set animation properties if necessary.
|
|
674
|
+
*/
|
|
675
|
+
VImage SetAnimationProperties(VImage image, int nPages, int pageHeight, std::vector<int> delay, int loop) {
|
|
676
|
+
bool hasDelay = !delay.empty();
|
|
677
|
+
VImage copy = image.copy();
|
|
678
|
+
|
|
679
|
+
// Only set page-height if we have more than one page, or this could
|
|
680
|
+
// accidentally turn into an animated image later.
|
|
681
|
+
if (nPages > 1) copy.set(VIPS_META_PAGE_HEIGHT, pageHeight);
|
|
682
|
+
if (hasDelay) {
|
|
683
|
+
if (delay.size() == 1) {
|
|
684
|
+
// We have just one delay, repeat that value for all frames.
|
|
685
|
+
delay.insert(delay.end(), nPages - 1, delay[0]);
|
|
686
|
+
}
|
|
687
|
+
copy.set("delay", delay);
|
|
688
|
+
}
|
|
689
|
+
if (nPages == 1 && !hasDelay && loop == -1) {
|
|
690
|
+
loop = 1;
|
|
691
|
+
}
|
|
692
|
+
if (loop != -1) copy.set("loop", loop);
|
|
693
|
+
|
|
694
|
+
return copy;
|
|
695
|
+
}
|
|
696
|
+
|
|
697
|
+
/*
|
|
698
|
+
Remove animation properties from image.
|
|
699
|
+
*/
|
|
700
|
+
VImage RemoveAnimationProperties(VImage image) {
|
|
701
|
+
VImage copy = image.copy();
|
|
702
|
+
copy.remove(VIPS_META_PAGE_HEIGHT);
|
|
703
|
+
copy.remove("delay");
|
|
704
|
+
copy.remove("loop");
|
|
705
|
+
return copy;
|
|
706
|
+
}
|
|
707
|
+
|
|
708
|
+
/*
|
|
709
|
+
Remove GIF palette from image.
|
|
710
|
+
*/
|
|
711
|
+
VImage RemoveGifPalette(VImage image) {
|
|
712
|
+
VImage copy = image.copy();
|
|
713
|
+
copy.remove("gif-palette");
|
|
714
|
+
return copy;
|
|
715
|
+
}
|
|
716
|
+
|
|
717
|
+
/*
|
|
718
|
+
Does this image have a non-default density?
|
|
719
|
+
*/
|
|
720
|
+
bool HasDensity(VImage image) {
|
|
721
|
+
return image.xres() > 1.0;
|
|
722
|
+
}
|
|
723
|
+
|
|
724
|
+
/*
|
|
725
|
+
Get pixels/mm resolution as pixels/inch density.
|
|
726
|
+
*/
|
|
727
|
+
int GetDensity(VImage image) {
|
|
728
|
+
return static_cast<int>(round(image.xres() * 25.4));
|
|
729
|
+
}
|
|
730
|
+
|
|
731
|
+
/*
|
|
732
|
+
Set pixels/mm resolution based on a pixels/inch density.
|
|
733
|
+
*/
|
|
734
|
+
VImage SetDensity(VImage image, const double density) {
|
|
735
|
+
const double pixelsPerMm = density / 25.4;
|
|
736
|
+
VImage copy = image.copy();
|
|
737
|
+
copy.get_image()->Xres = pixelsPerMm;
|
|
738
|
+
copy.get_image()->Yres = pixelsPerMm;
|
|
739
|
+
return copy;
|
|
740
|
+
}
|
|
741
|
+
|
|
742
|
+
/*
|
|
743
|
+
Multi-page images can have a page height. Fetch it, and sanity check it.
|
|
744
|
+
If page-height is not set, it defaults to the image height
|
|
745
|
+
*/
|
|
746
|
+
int GetPageHeight(VImage image) {
|
|
747
|
+
return vips_image_get_page_height(image.get_image());
|
|
748
|
+
}
|
|
749
|
+
|
|
750
|
+
/*
|
|
751
|
+
Check the proposed format supports the current dimensions.
|
|
752
|
+
*/
|
|
753
|
+
void AssertImageTypeDimensions(VImage image, ImageType const imageType) {
|
|
754
|
+
const int height = image.get_typeof(VIPS_META_PAGE_HEIGHT) == G_TYPE_INT
|
|
755
|
+
? image.get_int(VIPS_META_PAGE_HEIGHT)
|
|
756
|
+
: image.height();
|
|
757
|
+
if (imageType == ImageType::JPEG) {
|
|
758
|
+
if (image.width() > 65535 || height > 65535) {
|
|
759
|
+
throw vips::VError("Processed image is too large for the JPEG format");
|
|
760
|
+
}
|
|
761
|
+
} else if (imageType == ImageType::WEBP) {
|
|
762
|
+
if (image.width() > 16383 || height > 16383) {
|
|
763
|
+
throw vips::VError("Processed image is too large for the WebP format");
|
|
764
|
+
}
|
|
765
|
+
} else if (imageType == ImageType::GIF) {
|
|
766
|
+
if (image.width() > 65535 || height > 65535) {
|
|
767
|
+
throw vips::VError("Processed image is too large for the GIF format");
|
|
768
|
+
}
|
|
769
|
+
} else if (imageType == ImageType::HEIF) {
|
|
770
|
+
if (image.width() > 16384 || height > 16384) {
|
|
771
|
+
throw vips::VError("Processed image is too large for the HEIF format");
|
|
772
|
+
}
|
|
773
|
+
}
|
|
774
|
+
}
|
|
775
|
+
|
|
776
|
+
/*
|
|
777
|
+
Called when a Buffer undergoes GC, required to support mixed runtime libraries in Windows
|
|
778
|
+
*/
|
|
779
|
+
std::function<void(void*, char*)> FreeCallback = [](void*, char* data) {
|
|
780
|
+
g_free(data);
|
|
781
|
+
};
|
|
782
|
+
|
|
783
|
+
/*
|
|
784
|
+
Temporary buffer of warnings
|
|
785
|
+
*/
|
|
786
|
+
std::queue<std::string> vipsWarnings;
|
|
787
|
+
std::mutex vipsWarningsMutex;
|
|
788
|
+
|
|
789
|
+
/*
|
|
790
|
+
Called with warnings from the glib-registered "VIPS" domain
|
|
791
|
+
*/
|
|
792
|
+
void VipsWarningCallback(char const* log_domain, GLogLevelFlags log_level, char const* message, void* ignore) {
|
|
793
|
+
std::lock_guard<std::mutex> lock(vipsWarningsMutex);
|
|
794
|
+
vipsWarnings.emplace(message);
|
|
795
|
+
}
|
|
796
|
+
|
|
797
|
+
/*
|
|
798
|
+
Pop the oldest warning message from the queue
|
|
799
|
+
*/
|
|
800
|
+
std::string VipsWarningPop() {
|
|
801
|
+
std::string warning;
|
|
802
|
+
std::lock_guard<std::mutex> lock(vipsWarningsMutex);
|
|
803
|
+
if (!vipsWarnings.empty()) {
|
|
804
|
+
warning = vipsWarnings.front();
|
|
805
|
+
vipsWarnings.pop();
|
|
806
|
+
}
|
|
807
|
+
return warning;
|
|
808
|
+
}
|
|
809
|
+
|
|
810
|
+
/*
|
|
811
|
+
Attach an event listener for progress updates, used to detect timeout
|
|
812
|
+
*/
|
|
813
|
+
void SetTimeout(VImage image, int const seconds) {
|
|
814
|
+
if (seconds > 0) {
|
|
815
|
+
VipsImage *im = image.get_image();
|
|
816
|
+
if (im->progress_signal == NULL) {
|
|
817
|
+
int *timeout = VIPS_NEW(im, int);
|
|
818
|
+
*timeout = seconds;
|
|
819
|
+
g_signal_connect(im, "eval", G_CALLBACK(VipsProgressCallBack), timeout);
|
|
820
|
+
vips_image_set_progress(im, true);
|
|
821
|
+
}
|
|
822
|
+
}
|
|
823
|
+
}
|
|
824
|
+
|
|
825
|
+
/*
|
|
826
|
+
Event listener for progress updates, used to detect timeout
|
|
827
|
+
*/
|
|
828
|
+
void VipsProgressCallBack(VipsImage *im, VipsProgress *progress, int *timeout) {
|
|
829
|
+
if (*timeout > 0 && progress->run >= *timeout) {
|
|
830
|
+
vips_image_set_kill(im, true);
|
|
831
|
+
vips_error("timeout", "%d%% complete", progress->percent);
|
|
832
|
+
*timeout = 0;
|
|
833
|
+
}
|
|
834
|
+
}
|
|
835
|
+
|
|
836
|
+
/*
|
|
837
|
+
Calculate the (left, top) coordinates of the output image
|
|
838
|
+
within the input image, applying the given gravity during an embed.
|
|
839
|
+
|
|
840
|
+
@Azurebyte: We are basically swapping the inWidth and outWidth, inHeight and outHeight from the CalculateCrop function.
|
|
841
|
+
*/
|
|
842
|
+
std::tuple<int, int> CalculateEmbedPosition(int const inWidth, int const inHeight,
|
|
843
|
+
int const outWidth, int const outHeight, int const gravity) {
|
|
844
|
+
|
|
845
|
+
int left = 0;
|
|
846
|
+
int top = 0;
|
|
847
|
+
switch (gravity) {
|
|
848
|
+
case 1:
|
|
849
|
+
// North
|
|
850
|
+
left = (outWidth - inWidth) / 2;
|
|
851
|
+
break;
|
|
852
|
+
case 2:
|
|
853
|
+
// East
|
|
854
|
+
left = outWidth - inWidth;
|
|
855
|
+
top = (outHeight - inHeight) / 2;
|
|
856
|
+
break;
|
|
857
|
+
case 3:
|
|
858
|
+
// South
|
|
859
|
+
left = (outWidth - inWidth) / 2;
|
|
860
|
+
top = outHeight - inHeight;
|
|
861
|
+
break;
|
|
862
|
+
case 4:
|
|
863
|
+
// West
|
|
864
|
+
top = (outHeight - inHeight) / 2;
|
|
865
|
+
break;
|
|
866
|
+
case 5:
|
|
867
|
+
// Northeast
|
|
868
|
+
left = outWidth - inWidth;
|
|
869
|
+
break;
|
|
870
|
+
case 6:
|
|
871
|
+
// Southeast
|
|
872
|
+
left = outWidth - inWidth;
|
|
873
|
+
top = outHeight - inHeight;
|
|
874
|
+
break;
|
|
875
|
+
case 7:
|
|
876
|
+
// Southwest
|
|
877
|
+
top = outHeight - inHeight;
|
|
878
|
+
break;
|
|
879
|
+
case 8:
|
|
880
|
+
// Northwest
|
|
881
|
+
// Which is the default is 0,0 so we do not assign anything here.
|
|
882
|
+
break;
|
|
883
|
+
default:
|
|
884
|
+
// Centre
|
|
885
|
+
left = (outWidth - inWidth) / 2;
|
|
886
|
+
top = (outHeight - inHeight) / 2;
|
|
887
|
+
}
|
|
888
|
+
return std::make_tuple(left, top);
|
|
889
|
+
}
|
|
890
|
+
|
|
891
|
+
/*
|
|
892
|
+
Calculate the (left, top) coordinates of the output image
|
|
893
|
+
within the input image, applying the given gravity during a crop.
|
|
894
|
+
*/
|
|
895
|
+
std::tuple<int, int> CalculateCrop(int const inWidth, int const inHeight,
|
|
896
|
+
int const outWidth, int const outHeight, int const gravity) {
|
|
897
|
+
|
|
898
|
+
int left = 0;
|
|
899
|
+
int top = 0;
|
|
900
|
+
switch (gravity) {
|
|
901
|
+
case 1:
|
|
902
|
+
// North
|
|
903
|
+
left = (inWidth - outWidth + 1) / 2;
|
|
904
|
+
break;
|
|
905
|
+
case 2:
|
|
906
|
+
// East
|
|
907
|
+
left = inWidth - outWidth;
|
|
908
|
+
top = (inHeight - outHeight + 1) / 2;
|
|
909
|
+
break;
|
|
910
|
+
case 3:
|
|
911
|
+
// South
|
|
912
|
+
left = (inWidth - outWidth + 1) / 2;
|
|
913
|
+
top = inHeight - outHeight;
|
|
914
|
+
break;
|
|
915
|
+
case 4:
|
|
916
|
+
// West
|
|
917
|
+
top = (inHeight - outHeight + 1) / 2;
|
|
918
|
+
break;
|
|
919
|
+
case 5:
|
|
920
|
+
// Northeast
|
|
921
|
+
left = inWidth - outWidth;
|
|
922
|
+
break;
|
|
923
|
+
case 6:
|
|
924
|
+
// Southeast
|
|
925
|
+
left = inWidth - outWidth;
|
|
926
|
+
top = inHeight - outHeight;
|
|
927
|
+
break;
|
|
928
|
+
case 7:
|
|
929
|
+
// Southwest
|
|
930
|
+
top = inHeight - outHeight;
|
|
931
|
+
break;
|
|
932
|
+
case 8:
|
|
933
|
+
// Northwest
|
|
934
|
+
break;
|
|
935
|
+
default:
|
|
936
|
+
// Centre
|
|
937
|
+
left = (inWidth - outWidth + 1) / 2;
|
|
938
|
+
top = (inHeight - outHeight + 1) / 2;
|
|
939
|
+
}
|
|
940
|
+
return std::make_tuple(left, top);
|
|
941
|
+
}
|
|
942
|
+
|
|
943
|
+
/*
|
|
944
|
+
Calculate the (left, top) coordinates of the output image
|
|
945
|
+
within the input image, applying the given x and y offsets.
|
|
946
|
+
*/
|
|
947
|
+
std::tuple<int, int> CalculateCrop(int const inWidth, int const inHeight,
|
|
948
|
+
int const outWidth, int const outHeight, int const x, int const y) {
|
|
949
|
+
|
|
950
|
+
// default values
|
|
951
|
+
int left = 0;
|
|
952
|
+
int top = 0;
|
|
953
|
+
|
|
954
|
+
// assign only if valid
|
|
955
|
+
if (x < (inWidth - outWidth)) {
|
|
956
|
+
left = x;
|
|
957
|
+
} else if (x >= (inWidth - outWidth)) {
|
|
958
|
+
left = inWidth - outWidth;
|
|
959
|
+
}
|
|
960
|
+
|
|
961
|
+
if (y < (inHeight - outHeight)) {
|
|
962
|
+
top = y;
|
|
963
|
+
} else if (y >= (inHeight - outHeight)) {
|
|
964
|
+
top = inHeight - outHeight;
|
|
965
|
+
}
|
|
966
|
+
|
|
967
|
+
return std::make_tuple(left, top);
|
|
968
|
+
}
|
|
969
|
+
|
|
970
|
+
/*
|
|
971
|
+
Are pixel values in this image 16-bit integer?
|
|
972
|
+
*/
|
|
973
|
+
bool Is16Bit(VipsInterpretation const interpretation) {
|
|
974
|
+
return interpretation == VIPS_INTERPRETATION_RGB16 || interpretation == VIPS_INTERPRETATION_GREY16;
|
|
975
|
+
}
|
|
976
|
+
|
|
977
|
+
/*
|
|
978
|
+
Convert RGBA value to another colourspace
|
|
979
|
+
*/
|
|
980
|
+
std::vector<double> GetRgbaAsColourspace(std::vector<double> const rgba,
|
|
981
|
+
VipsInterpretation const interpretation, bool premultiply) {
|
|
982
|
+
int const bands = static_cast<int>(rgba.size());
|
|
983
|
+
if (bands < 3) {
|
|
984
|
+
return rgba;
|
|
985
|
+
}
|
|
986
|
+
VImage pixel = VImage::new_matrix(1, 1);
|
|
987
|
+
pixel.set("bands", bands);
|
|
988
|
+
pixel = pixel
|
|
989
|
+
.new_from_image(rgba)
|
|
990
|
+
.colourspace(interpretation, VImage::option()->set("source_space", VIPS_INTERPRETATION_sRGB));
|
|
991
|
+
if (premultiply) {
|
|
992
|
+
pixel = pixel.premultiply();
|
|
993
|
+
}
|
|
994
|
+
return pixel(0, 0);
|
|
995
|
+
}
|
|
996
|
+
|
|
997
|
+
/*
|
|
998
|
+
Apply the alpha channel to a given colour
|
|
999
|
+
*/
|
|
1000
|
+
std::tuple<VImage, std::vector<double>> ApplyAlpha(VImage image, std::vector<double> colour, bool premultiply) {
|
|
1001
|
+
// Scale up 8-bit values to match 16-bit input image
|
|
1002
|
+
double const multiplier = sharp::Is16Bit(image.interpretation()) ? 256.0 : 1.0;
|
|
1003
|
+
// Create alphaColour colour
|
|
1004
|
+
std::vector<double> alphaColour;
|
|
1005
|
+
if (image.bands() > 2) {
|
|
1006
|
+
alphaColour = {
|
|
1007
|
+
multiplier * colour[0],
|
|
1008
|
+
multiplier * colour[1],
|
|
1009
|
+
multiplier * colour[2]
|
|
1010
|
+
};
|
|
1011
|
+
} else {
|
|
1012
|
+
// Convert sRGB to greyscale
|
|
1013
|
+
alphaColour = { multiplier * (
|
|
1014
|
+
0.2126 * colour[0] +
|
|
1015
|
+
0.7152 * colour[1] +
|
|
1016
|
+
0.0722 * colour[2])
|
|
1017
|
+
};
|
|
1018
|
+
}
|
|
1019
|
+
// Add alpha channel(s) to alphaColour colour
|
|
1020
|
+
if (colour[3] < 255.0 || image.has_alpha()) {
|
|
1021
|
+
int extraBands = image.bands() > 4 ? image.bands() - 3 : 1;
|
|
1022
|
+
alphaColour.insert(alphaColour.end(), extraBands, colour[3] * multiplier);
|
|
1023
|
+
}
|
|
1024
|
+
// Ensure alphaColour colour uses correct colourspace
|
|
1025
|
+
alphaColour = sharp::GetRgbaAsColourspace(alphaColour, image.interpretation(), premultiply);
|
|
1026
|
+
// Add non-transparent alpha channel, if required
|
|
1027
|
+
if (colour[3] < 255.0 && !image.has_alpha()) {
|
|
1028
|
+
image = image.bandjoin_const({ 255 * multiplier });
|
|
1029
|
+
}
|
|
1030
|
+
return std::make_tuple(image, alphaColour);
|
|
1031
|
+
}
|
|
1032
|
+
|
|
1033
|
+
/*
|
|
1034
|
+
Removes alpha channels, if any.
|
|
1035
|
+
*/
|
|
1036
|
+
VImage RemoveAlpha(VImage image) {
|
|
1037
|
+
while (image.bands() > 1 && image.has_alpha()) {
|
|
1038
|
+
image = image.extract_band(0, VImage::option()->set("n", image.bands() - 1));
|
|
1039
|
+
}
|
|
1040
|
+
return image;
|
|
1041
|
+
}
|
|
1042
|
+
|
|
1043
|
+
/*
|
|
1044
|
+
Ensures alpha channel, if missing.
|
|
1045
|
+
*/
|
|
1046
|
+
VImage EnsureAlpha(VImage image, double const value) {
|
|
1047
|
+
if (!image.has_alpha()) {
|
|
1048
|
+
image = image.bandjoin_const({ value * vips_interpretation_max_alpha(image.interpretation()) });
|
|
1049
|
+
}
|
|
1050
|
+
return image;
|
|
1051
|
+
}
|
|
1052
|
+
|
|
1053
|
+
std::pair<double, double> ResolveShrink(int width, int height, int targetWidth, int targetHeight,
|
|
1054
|
+
Canvas canvas, bool withoutEnlargement, bool withoutReduction) {
|
|
1055
|
+
double hshrink = 1.0;
|
|
1056
|
+
double vshrink = 1.0;
|
|
1057
|
+
|
|
1058
|
+
if (targetWidth > 0 && targetHeight > 0) {
|
|
1059
|
+
// Fixed width and height
|
|
1060
|
+
hshrink = static_cast<double>(width) / targetWidth;
|
|
1061
|
+
vshrink = static_cast<double>(height) / targetHeight;
|
|
1062
|
+
|
|
1063
|
+
switch (canvas) {
|
|
1064
|
+
case Canvas::CROP:
|
|
1065
|
+
case Canvas::MIN:
|
|
1066
|
+
if (hshrink < vshrink) {
|
|
1067
|
+
vshrink = hshrink;
|
|
1068
|
+
} else {
|
|
1069
|
+
hshrink = vshrink;
|
|
1070
|
+
}
|
|
1071
|
+
break;
|
|
1072
|
+
case Canvas::EMBED:
|
|
1073
|
+
case Canvas::MAX:
|
|
1074
|
+
if (hshrink > vshrink) {
|
|
1075
|
+
vshrink = hshrink;
|
|
1076
|
+
} else {
|
|
1077
|
+
hshrink = vshrink;
|
|
1078
|
+
}
|
|
1079
|
+
break;
|
|
1080
|
+
case Canvas::IGNORE_ASPECT:
|
|
1081
|
+
break;
|
|
1082
|
+
}
|
|
1083
|
+
} else if (targetWidth > 0) {
|
|
1084
|
+
// Fixed width
|
|
1085
|
+
hshrink = static_cast<double>(width) / targetWidth;
|
|
1086
|
+
|
|
1087
|
+
if (canvas != Canvas::IGNORE_ASPECT) {
|
|
1088
|
+
// Auto height
|
|
1089
|
+
vshrink = hshrink;
|
|
1090
|
+
}
|
|
1091
|
+
} else if (targetHeight > 0) {
|
|
1092
|
+
// Fixed height
|
|
1093
|
+
vshrink = static_cast<double>(height) / targetHeight;
|
|
1094
|
+
|
|
1095
|
+
if (canvas != Canvas::IGNORE_ASPECT) {
|
|
1096
|
+
// Auto width
|
|
1097
|
+
hshrink = vshrink;
|
|
1098
|
+
}
|
|
1099
|
+
}
|
|
1100
|
+
|
|
1101
|
+
// We should not reduce or enlarge the output image, if
|
|
1102
|
+
// withoutReduction or withoutEnlargement is specified.
|
|
1103
|
+
if (withoutReduction) {
|
|
1104
|
+
// Equivalent of VIPS_SIZE_UP
|
|
1105
|
+
hshrink = std::min(1.0, hshrink);
|
|
1106
|
+
vshrink = std::min(1.0, vshrink);
|
|
1107
|
+
} else if (withoutEnlargement) {
|
|
1108
|
+
// Equivalent of VIPS_SIZE_DOWN
|
|
1109
|
+
hshrink = std::max(1.0, hshrink);
|
|
1110
|
+
vshrink = std::max(1.0, vshrink);
|
|
1111
|
+
}
|
|
1112
|
+
|
|
1113
|
+
// We don't want to shrink so much that we send an axis to 0
|
|
1114
|
+
hshrink = std::min(hshrink, static_cast<double>(width));
|
|
1115
|
+
vshrink = std::min(vshrink, static_cast<double>(height));
|
|
1116
|
+
|
|
1117
|
+
return std::make_pair(hshrink, vshrink);
|
|
1118
|
+
}
|
|
1119
|
+
|
|
1120
|
+
/*
|
|
1121
|
+
Ensure decoding remains sequential.
|
|
1122
|
+
*/
|
|
1123
|
+
VImage StaySequential(VImage image, bool condition) {
|
|
1124
|
+
if (vips_image_is_sequential(image.get_image()) && condition) {
|
|
1125
|
+
image = image.copy_memory().copy();
|
|
1126
|
+
image.remove(VIPS_META_SEQUENTIAL);
|
|
1127
|
+
}
|
|
1128
|
+
return image;
|
|
1129
|
+
}
|
|
1130
|
+
} // namespace sharp
|