@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.
Files changed (129) hide show
  1. package/mcp/skills/aipk_init_project/template/AGENTS_TEMPLATE.md +1 -1
  2. package/mcp/src/server.js +1 -1
  3. package/package.json +1 -1
  4. package/skills/aipk_design/GURU_AI.md +119 -7
  5. package/skills/aipk_design/SKILL.md +8 -7
  6. package/skills/aipk_design/{auto_panel_splitter → update-requirements-from-design}/SKILL.md +18 -16
  7. package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/.package-lock.json +113 -0
  8. package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/@img/colour/LICENSE.md +82 -0
  9. package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/@img/colour/README.md +15 -0
  10. package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/@img/colour/color.cjs +1594 -0
  11. package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/@img/colour/index.cjs +1 -0
  12. package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/@img/colour/package.json +45 -0
  13. package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/@img/sharp-darwin-arm64/LICENSE +191 -0
  14. package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/@img/sharp-darwin-arm64/README.md +18 -0
  15. package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/@img/sharp-darwin-arm64/lib/sharp-darwin-arm64.node +0 -0
  16. package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/@img/sharp-darwin-arm64/package.json +40 -0
  17. package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/@img/sharp-libvips-darwin-arm64/README.md +46 -0
  18. 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
  19. package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/@img/sharp-libvips-darwin-arm64/lib/index.js +1 -0
  20. 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
  21. package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/@img/sharp-libvips-darwin-arm64/package.json +36 -0
  22. package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/@img/sharp-libvips-darwin-arm64/versions.json +30 -0
  23. package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/detect-libc/LICENSE +201 -0
  24. package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/detect-libc/README.md +163 -0
  25. package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/detect-libc/index.d.ts +14 -0
  26. package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/detect-libc/lib/detect-libc.js +313 -0
  27. package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/detect-libc/lib/elf.js +39 -0
  28. package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/detect-libc/lib/filesystem.js +51 -0
  29. package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/detect-libc/lib/process.js +24 -0
  30. package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/detect-libc/package.json +44 -0
  31. package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/semver/LICENSE +15 -0
  32. package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/semver/README.md +664 -0
  33. package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/semver/bin/semver.js +191 -0
  34. package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/semver/classes/comparator.js +143 -0
  35. package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/semver/classes/index.js +7 -0
  36. package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/semver/classes/range.js +557 -0
  37. package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/semver/classes/semver.js +333 -0
  38. package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/semver/functions/clean.js +8 -0
  39. package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/semver/functions/cmp.js +54 -0
  40. package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/semver/functions/coerce.js +62 -0
  41. package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/semver/functions/compare-build.js +9 -0
  42. package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/semver/functions/compare-loose.js +5 -0
  43. package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/semver/functions/compare.js +7 -0
  44. package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/semver/functions/diff.js +60 -0
  45. package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/semver/functions/eq.js +5 -0
  46. package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/semver/functions/gt.js +5 -0
  47. package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/semver/functions/gte.js +5 -0
  48. package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/semver/functions/inc.js +21 -0
  49. package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/semver/functions/lt.js +5 -0
  50. package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/semver/functions/lte.js +5 -0
  51. package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/semver/functions/major.js +5 -0
  52. package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/semver/functions/minor.js +5 -0
  53. package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/semver/functions/neq.js +5 -0
  54. package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/semver/functions/parse.js +18 -0
  55. package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/semver/functions/patch.js +5 -0
  56. package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/semver/functions/prerelease.js +8 -0
  57. package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/semver/functions/rcompare.js +5 -0
  58. package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/semver/functions/rsort.js +5 -0
  59. package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/semver/functions/satisfies.js +12 -0
  60. package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/semver/functions/sort.js +5 -0
  61. package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/semver/functions/valid.js +8 -0
  62. package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/semver/index.js +91 -0
  63. package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/semver/internal/constants.js +37 -0
  64. package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/semver/internal/debug.js +11 -0
  65. package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/semver/internal/identifiers.js +29 -0
  66. package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/semver/internal/lrucache.js +42 -0
  67. package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/semver/internal/parse-options.js +17 -0
  68. package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/semver/internal/re.js +223 -0
  69. package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/semver/package.json +78 -0
  70. package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/semver/preload.js +4 -0
  71. package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/semver/range.bnf +16 -0
  72. package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/semver/ranges/gtr.js +6 -0
  73. package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/semver/ranges/intersects.js +9 -0
  74. package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/semver/ranges/ltr.js +6 -0
  75. package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/semver/ranges/max-satisfying.js +27 -0
  76. package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/semver/ranges/min-satisfying.js +26 -0
  77. package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/semver/ranges/min-version.js +63 -0
  78. package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/semver/ranges/outside.js +82 -0
  79. package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/semver/ranges/simplify.js +49 -0
  80. package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/semver/ranges/subset.js +249 -0
  81. package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/semver/ranges/to-comparators.js +10 -0
  82. package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/semver/ranges/valid.js +13 -0
  83. package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/sharp/LICENSE +191 -0
  84. package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/sharp/README.md +118 -0
  85. package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/sharp/install/build.js +38 -0
  86. package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/sharp/install/check.js +14 -0
  87. package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/sharp/lib/channel.js +177 -0
  88. package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/sharp/lib/colour.js +195 -0
  89. package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/sharp/lib/composite.js +212 -0
  90. package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/sharp/lib/constructor.js +499 -0
  91. package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/sharp/lib/index.d.ts +1971 -0
  92. package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/sharp/lib/index.js +16 -0
  93. package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/sharp/lib/input.js +809 -0
  94. package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/sharp/lib/is.js +143 -0
  95. package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/sharp/lib/libvips.js +207 -0
  96. package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/sharp/lib/operation.js +1016 -0
  97. package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/sharp/lib/output.js +1666 -0
  98. package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/sharp/lib/resize.js +595 -0
  99. package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/sharp/lib/sharp.js +121 -0
  100. package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/sharp/lib/utility.js +291 -0
  101. package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/sharp/package.json +202 -0
  102. package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/sharp/src/binding.gyp +298 -0
  103. package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/sharp/src/common.cc +1130 -0
  104. package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/sharp/src/common.h +402 -0
  105. package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/sharp/src/metadata.cc +346 -0
  106. package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/sharp/src/metadata.h +90 -0
  107. package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/sharp/src/operations.cc +499 -0
  108. package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/sharp/src/operations.h +137 -0
  109. package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/sharp/src/pipeline.cc +1814 -0
  110. package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/sharp/src/pipeline.h +408 -0
  111. package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/sharp/src/sharp.cc +43 -0
  112. package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/sharp/src/stats.cc +186 -0
  113. package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/sharp/src/stats.h +62 -0
  114. package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/sharp/src/utilities.cc +288 -0
  115. package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/node_modules/sharp/src/utilities.h +22 -0
  116. package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/package-lock.json +529 -0
  117. package/skills/aipk_design/update-requirements-from-design/scripts/sharp-runtime/package.json +5 -0
  118. 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
  119. 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
  120. package/skills/aipk_development/GURU_AI.md +1 -1
  121. package/skills/aipk_init_project/GURU_AI.md +2 -2
  122. package/skills/aipk_init_project/template/AGENTS_TEMPLATE.md +1 -1
  123. package/skills/aipk_operations/GURU_AI.md +1 -1
  124. package/skills/aipk_requirements/GURU_AI.md +2 -2
  125. package/skills/aipk_requirements/documentation/SKILL.md +23 -1
  126. package/skills/aipk_skill_generate/GURU_AI.md +1 -1
  127. package/skills/aipk_tool_prompts/GURU_AI.md +1 -1
  128. /package/skills/aipk_design/{auto_panel_splitter → update-requirements-from-design}/scripts/auto_panel_splitter.js +0 -0
  129. /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