Compare commits
62 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
34d5252242 | ||
|
|
f31e4d2869 | ||
|
|
c695c40abc | ||
|
|
fd1ca1dbb2 | ||
|
|
f25dbd5f61 | ||
|
|
541e7104fd | ||
|
|
94945cf6ac | ||
|
|
db76e655f8 | ||
|
|
d43c7b581d | ||
|
|
383b933e26 | ||
|
|
d26ccf6294 | ||
|
|
6f9699f605 | ||
|
|
1e9093d781 | ||
|
|
9dc6492e52 | ||
|
|
d22f7cae6a | ||
|
|
473afaab45 | ||
|
|
dcd68303a4 | ||
|
|
03394556b5 | ||
|
|
1c4f6f75f3 | ||
|
|
f00928dedb | ||
|
|
a48f8fbb61 | ||
|
|
1fa388370e | ||
|
|
95ef6b3f71 | ||
|
|
de11d36d00 | ||
|
|
d77c2adabe | ||
|
|
c89c055ae0 | ||
|
|
dac8117f32 | ||
|
|
937b091bab | ||
|
|
019e6a1bfe | ||
|
|
1565e58fcf | ||
|
|
c22e2a17ef | ||
|
|
fd2a10ccea | ||
|
|
0725378257 | ||
|
|
c431909f35 | ||
|
|
db4df6f0b2 | ||
|
|
17f942c802 | ||
|
|
60438ebfe5 | ||
|
|
21fbe546b8 | ||
|
|
11900945eb | ||
|
|
ea5270221b | ||
|
|
a64844689e | ||
|
|
6007e13a22 | ||
|
|
c3274e480b | ||
|
|
3c54eeda5b | ||
|
|
6236e4b97d | ||
|
|
796738da65 | ||
|
|
37d385fafa | ||
|
|
db2af42ee7 | ||
|
|
24b42ef192 | ||
|
|
2ce166ab0a | ||
|
|
71755b69e4 | ||
|
|
1106aac2d8 | ||
|
|
93aac660a3 | ||
|
|
0ce8ad7130 | ||
|
|
deacd553bf | ||
|
|
c8ff7e11a9 | ||
|
|
4cff62258c | ||
|
|
0144358afb | ||
|
|
136097efe7 | ||
|
|
374c6959d7 | ||
|
|
7d48a5ccf4 | ||
|
|
bf3254cb16 |
89
.travis.yml
@@ -1,34 +1,83 @@
|
|||||||
language: node_js
|
|
||||||
matrix:
|
matrix:
|
||||||
include:
|
include:
|
||||||
- os: linux
|
- name: "Linux (glibc) - Node 6"
|
||||||
dist: trusty
|
os: linux
|
||||||
sudo: false
|
|
||||||
node_js: "4"
|
|
||||||
- os: linux
|
|
||||||
dist: trusty
|
dist: trusty
|
||||||
sudo: false
|
sudo: false
|
||||||
|
language: node_js
|
||||||
node_js: "6"
|
node_js: "6"
|
||||||
- os: linux
|
- name: "Linux (glibc) - Node 8"
|
||||||
|
os: linux
|
||||||
dist: trusty
|
dist: trusty
|
||||||
sudo: false
|
sudo: false
|
||||||
|
language: node_js
|
||||||
node_js: "8"
|
node_js: "8"
|
||||||
- os: linux
|
- name: "Linux (glibc) - Node 10"
|
||||||
|
os: linux
|
||||||
dist: trusty
|
dist: trusty
|
||||||
sudo: false
|
sudo: false
|
||||||
|
language: node_js
|
||||||
node_js: "10"
|
node_js: "10"
|
||||||
- os: osx
|
after_success:
|
||||||
osx_image: xcode8.3
|
- npm install coveralls
|
||||||
node_js: "4"
|
- cat ./coverage/lcov.info | ./node_modules/coveralls/bin/coveralls.js
|
||||||
- os: osx
|
- name: "Linux (glibc) - Node 11 (Experimental)"
|
||||||
osx_image: xcode8.3
|
os: linux
|
||||||
|
dist: trusty
|
||||||
|
sudo: false
|
||||||
|
language: node_js
|
||||||
|
node_js: "11"
|
||||||
|
before_install:
|
||||||
|
- unset PREBUILD_TOKEN
|
||||||
|
- name: "Linux (musl) - Node 8"
|
||||||
|
os: linux
|
||||||
|
dist: trusty
|
||||||
|
sudo: true
|
||||||
|
language: minimal
|
||||||
|
before_install:
|
||||||
|
- sudo docker run -dit --name sharp --env CI --env PREBUILD_TOKEN --volume "${PWD}:/mnt/sharp" --workdir /mnt/sharp node:8-alpine
|
||||||
|
- sudo docker exec sharp apk add build-base git python2 --update-cache
|
||||||
|
install: sudo docker exec sharp sh -c "npm install --unsafe-perm"
|
||||||
|
script: sudo docker exec sharp sh -c "npm test"
|
||||||
|
- name: "Linux (musl) - Node 10"
|
||||||
|
os: linux
|
||||||
|
dist: trusty
|
||||||
|
sudo: true
|
||||||
|
language: minimal
|
||||||
|
before_install:
|
||||||
|
- sudo docker run -dit --name sharp --env CI --env PREBUILD_TOKEN --volume "${PWD}:/mnt/sharp" --workdir /mnt/sharp node:10-alpine
|
||||||
|
- sudo docker exec sharp apk add build-base git python2 --update-cache
|
||||||
|
install: sudo docker exec sharp sh -c "npm install --unsafe-perm"
|
||||||
|
script: sudo docker exec sharp sh -c "npm test"
|
||||||
|
- name: "Linux (musl) - Node 11 (Experimental)"
|
||||||
|
os: linux
|
||||||
|
dist: trusty
|
||||||
|
sudo: true
|
||||||
|
language: minimal
|
||||||
|
before_install:
|
||||||
|
- sudo docker run -dit --name sharp --env CI --volume "${PWD}:/mnt/sharp" --workdir /mnt/sharp node:11-alpine
|
||||||
|
- sudo docker exec sharp apk add build-base git python2 --update-cache
|
||||||
|
install: sudo docker exec sharp sh -c "npm install --unsafe-perm"
|
||||||
|
script: sudo docker exec sharp sh -c "npm test"
|
||||||
|
- name: "OS X - Node 6"
|
||||||
|
os: osx
|
||||||
|
osx_image: xcode9.2
|
||||||
|
language: node_js
|
||||||
node_js: "6"
|
node_js: "6"
|
||||||
- os: osx
|
- name: "OS X - Node 8"
|
||||||
osx_image: xcode8.3
|
os: osx
|
||||||
|
osx_image: xcode9.2
|
||||||
|
language: node_js
|
||||||
node_js: "8"
|
node_js: "8"
|
||||||
- os: osx
|
- name: "OS X - Node 10"
|
||||||
osx_image: xcode8.3
|
os: osx
|
||||||
|
osx_image: xcode9.2
|
||||||
|
language: node_js
|
||||||
node_js: "10"
|
node_js: "10"
|
||||||
after_success:
|
- name: "OS X - Node 11 (Experimental)"
|
||||||
- npm install coveralls
|
os: osx
|
||||||
- cat ./coverage/lcov.info | ./node_modules/coveralls/bin/coveralls.js
|
osx_image: xcode9.2
|
||||||
|
language: node_js
|
||||||
|
node_js: "11"
|
||||||
|
before_install:
|
||||||
|
- unset PREBUILD_TOKEN
|
||||||
|
|||||||
@@ -14,7 +14,7 @@ New bugs are assigned a `triage` label whilst under investigation.
|
|||||||
|
|
||||||
If a [similar request](https://github.com/lovell/sharp/labels/enhancement) exists, it's probably fastest to add a comment to it about your requirement.
|
If a [similar request](https://github.com/lovell/sharp/labels/enhancement) exists, it's probably fastest to add a comment to it about your requirement.
|
||||||
|
|
||||||
Implementation is usually straightforward if _libvips_ [already supports](https://jcupitt.github.io/libvips/API/current/) the feature you need.
|
Implementation is usually straightforward if _libvips_ [already supports](https://libvips.github.io/libvips/API/current/) the feature you need.
|
||||||
|
|
||||||
## Submit a Pull Request to fix a bug
|
## Submit a Pull Request to fix a bug
|
||||||
|
|
||||||
|
|||||||
@@ -21,8 +21,8 @@ Lanczos resampling ensures quality is not sacrificed for speed.
|
|||||||
As well as image resizing, operations such as
|
As well as image resizing, operations such as
|
||||||
rotation, extraction, compositing and gamma correction are available.
|
rotation, extraction, compositing and gamma correction are available.
|
||||||
|
|
||||||
Most modern 64-bit OS X, Windows and Linux (glibc) systems running
|
Most modern 64-bit OS X, Windows and Linux systems running
|
||||||
Node versions 4, 6, 8 and 10
|
Node versions 6, 8 and 10
|
||||||
do not require any additional install or runtime dependencies.
|
do not require any additional install or runtime dependencies.
|
||||||
|
|
||||||
## Examples
|
## Examples
|
||||||
@@ -78,7 +78,7 @@ Visit [sharp.pixelplumbing.com](http://sharp.pixelplumbing.com/) for complete
|
|||||||
A [guide for contributors](https://github.com/lovell/sharp/blob/master/CONTRIBUTING.md)
|
A [guide for contributors](https://github.com/lovell/sharp/blob/master/CONTRIBUTING.md)
|
||||||
covers reporting bugs, requesting features and submitting code changes.
|
covers reporting bugs, requesting features and submitting code changes.
|
||||||
|
|
||||||
### Licence
|
### Licensing
|
||||||
|
|
||||||
Copyright 2013, 2014, 2015, 2016, 2017, 2018 Lovell Fuller and contributors.
|
Copyright 2013, 2014, 2015, 2016, 2017, 2018 Lovell Fuller and contributors.
|
||||||
|
|
||||||
|
|||||||
@@ -4,10 +4,11 @@ build: off
|
|||||||
platform: x64
|
platform: x64
|
||||||
environment:
|
environment:
|
||||||
matrix:
|
matrix:
|
||||||
- nodejs_version: "4"
|
|
||||||
- nodejs_version: "6"
|
- nodejs_version: "6"
|
||||||
- nodejs_version: "8"
|
- nodejs_version: "8"
|
||||||
- nodejs_version: "10"
|
- nodejs_version: "10"
|
||||||
|
- nodejs_version: "11"
|
||||||
|
PREBUILD_TOKEN: ""
|
||||||
install:
|
install:
|
||||||
- ps: Install-Product node $env:nodejs_version x64
|
- ps: Install-Product node $env:nodejs_version x64
|
||||||
- npm install -g npm@5
|
- npm install -g npm@5
|
||||||
|
|||||||
17
binding.gyp
@@ -128,9 +128,11 @@
|
|||||||
'../vendor/lib/libcairo.so',
|
'../vendor/lib/libcairo.so',
|
||||||
'../vendor/lib/libcroco-0.6.so',
|
'../vendor/lib/libcroco-0.6.so',
|
||||||
'../vendor/lib/libexif.so',
|
'../vendor/lib/libexif.so',
|
||||||
|
'../vendor/lib/libexpat.so',
|
||||||
'../vendor/lib/libffi.so',
|
'../vendor/lib/libffi.so',
|
||||||
'../vendor/lib/libfontconfig.so',
|
'../vendor/lib/libfontconfig.so',
|
||||||
'../vendor/lib/libfreetype.so',
|
'../vendor/lib/libfreetype.so',
|
||||||
|
'../vendor/lib/libfribidi.so',
|
||||||
'../vendor/lib/libgdk_pixbuf-2.0.so',
|
'../vendor/lib/libgdk_pixbuf-2.0.so',
|
||||||
'../vendor/lib/libgif.so',
|
'../vendor/lib/libgif.so',
|
||||||
'../vendor/lib/libgio-2.0.so',
|
'../vendor/lib/libgio-2.0.so',
|
||||||
@@ -138,6 +140,7 @@
|
|||||||
'../vendor/lib/libgsf-1.so',
|
'../vendor/lib/libgsf-1.so',
|
||||||
'../vendor/lib/libgthread-2.0.so',
|
'../vendor/lib/libgthread-2.0.so',
|
||||||
'../vendor/lib/libharfbuzz.so',
|
'../vendor/lib/libharfbuzz.so',
|
||||||
|
'../vendor/lib/libharfbuzz-subset.so.0',
|
||||||
'../vendor/lib/libjpeg.so',
|
'../vendor/lib/libjpeg.so',
|
||||||
'../vendor/lib/liblcms2.so',
|
'../vendor/lib/liblcms2.so',
|
||||||
'../vendor/lib/liborc-0.4.so',
|
'../vendor/lib/liborc-0.4.so',
|
||||||
@@ -149,6 +152,8 @@
|
|||||||
'../vendor/lib/librsvg-2.so',
|
'../vendor/lib/librsvg-2.so',
|
||||||
'../vendor/lib/libtiff.so',
|
'../vendor/lib/libtiff.so',
|
||||||
'../vendor/lib/libwebp.so',
|
'../vendor/lib/libwebp.so',
|
||||||
|
'../vendor/lib/libwebpdemux.so',
|
||||||
|
'../vendor/lib/libwebpmux.so',
|
||||||
'../vendor/lib/libxml2.so',
|
'../vendor/lib/libxml2.so',
|
||||||
'../vendor/lib/libz.so',
|
'../vendor/lib/libz.so',
|
||||||
# Ensure runtime linking is relative to sharp.node
|
# Ensure runtime linking is relative to sharp.node
|
||||||
@@ -178,13 +183,23 @@
|
|||||||
},
|
},
|
||||||
'configurations': {
|
'configurations': {
|
||||||
'Release': {
|
'Release': {
|
||||||
|
'cflags_cc': [
|
||||||
|
'-Wno-cast-function-type',
|
||||||
|
'-Wno-deprecated-declarations'
|
||||||
|
],
|
||||||
|
'xcode_settings': {
|
||||||
|
'OTHER_CPLUSPLUSFLAGS': [
|
||||||
|
'-Wno-deprecated-declarations'
|
||||||
|
]
|
||||||
|
},
|
||||||
'msvs_settings': {
|
'msvs_settings': {
|
||||||
'VCCLCompilerTool': {
|
'VCCLCompilerTool': {
|
||||||
'ExceptionHandling': 1
|
'ExceptionHandling': 1
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
'msvs_disabled_warnings': [
|
'msvs_disabled_warnings': [
|
||||||
4275
|
4275,
|
||||||
|
4996
|
||||||
]
|
]
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|||||||
@@ -1,23 +1,5 @@
|
|||||||
<!-- Generated by documentation.js. Update this documentation by updating the source code. -->
|
<!-- Generated by documentation.js. Update this documentation by updating the source code. -->
|
||||||
|
|
||||||
## background
|
|
||||||
|
|
||||||
Set the background for the `embed`, `flatten` and `extend` operations.
|
|
||||||
The default background is `{r: 0, g: 0, b: 0, alpha: 1}`, black without transparency.
|
|
||||||
|
|
||||||
Delegates to the _color_ module, which can throw an Error
|
|
||||||
but is liberal in what it accepts, clipping values to sensible min/max.
|
|
||||||
The alpha value is a float between `0` (transparent) and `1` (opaque).
|
|
||||||
|
|
||||||
### Parameters
|
|
||||||
|
|
||||||
- `rgba` **([String][1] \| [Object][2])** parsed by the [color][3] module to extract values for red, green, blue and alpha.
|
|
||||||
|
|
||||||
|
|
||||||
- Throws **[Error][4]** Invalid parameter
|
|
||||||
|
|
||||||
Returns **Sharp**
|
|
||||||
|
|
||||||
## tint
|
## tint
|
||||||
|
|
||||||
Tint the image using the provided chroma while preserving the image luminance.
|
Tint the image using the provided chroma while preserving the image luminance.
|
||||||
@@ -94,4 +76,4 @@ Returns **Sharp**
|
|||||||
|
|
||||||
[5]: https://developer.mozilla.org/docs/Web/JavaScript/Reference/Global_Objects/Boolean
|
[5]: https://developer.mozilla.org/docs/Web/JavaScript/Reference/Global_Objects/Boolean
|
||||||
|
|
||||||
[6]: https://github.com/jcupitt/libvips/blob/master/libvips/iofuncs/enumtypes.c#L568
|
[6]: https://github.com/libvips/libvips/blob/master/libvips/iofuncs/enumtypes.c#L568
|
||||||
|
|||||||
@@ -18,7 +18,7 @@ If the overlay image contains an alpha channel then composition with premultipli
|
|||||||
- `options.left` **[Number][4]?** the pixel offset from the left edge.
|
- `options.left` **[Number][4]?** the pixel offset from the left edge.
|
||||||
- `options.tile` **[Boolean][5]** set to true to repeat the overlay image across the entire image with the given `gravity`. (optional, default `false`)
|
- `options.tile` **[Boolean][5]** set to true to repeat the overlay image across the entire image with the given `gravity`. (optional, default `false`)
|
||||||
- `options.cutout` **[Boolean][5]** set to true to apply only the alpha channel of the overlay image to the input image, giving the appearance of one image being cut out of another. (optional, default `false`)
|
- `options.cutout` **[Boolean][5]** set to true to apply only the alpha channel of the overlay image to the input image, giving the appearance of one image being cut out of another. (optional, default `false`)
|
||||||
- `options.density` **[Number][4]** integral number representing the DPI for vector overlay image. (optional, default `72`)
|
- `options.density` **[Number][4]** number representing the DPI for vector overlay image. (optional, default `72`)
|
||||||
- `options.raw` **[Object][3]?** describes overlay when using raw pixel data.
|
- `options.raw` **[Object][3]?** describes overlay when using raw pixel data.
|
||||||
- `options.raw.width` **[Number][4]?**
|
- `options.raw.width` **[Number][4]?**
|
||||||
- `options.raw.height` **[Number][4]?**
|
- `options.raw.height` **[Number][4]?**
|
||||||
|
|||||||
@@ -12,7 +12,7 @@
|
|||||||
- `options.failOnError` **[Boolean][4]** by default apply a "best effort"
|
- `options.failOnError` **[Boolean][4]** by default apply a "best effort"
|
||||||
to decode images, even if the data is corrupt or invalid. Set this flag to true
|
to decode images, even if the data is corrupt or invalid. Set this flag to true
|
||||||
if you'd rather halt processing and raise an error when loading invalid images. (optional, default `false`)
|
if you'd rather halt processing and raise an error when loading invalid images. (optional, default `false`)
|
||||||
- `options.density` **[Number][5]** integral number representing the DPI for vector images. (optional, default `72`)
|
- `options.density` **[Number][5]** number representing the DPI for vector images. (optional, default `72`)
|
||||||
- `options.page` **[Number][5]** page number to extract for multi-page input (GIF, TIFF) (optional, default `0`)
|
- `options.page` **[Number][5]** page number to extract for multi-page input (GIF, TIFF) (optional, default `0`)
|
||||||
- `options.raw` **[Object][3]?** describes raw pixel input image data. See `raw()` for pixel ordering.
|
- `options.raw` **[Object][3]?** describes raw pixel input image data. See `raw()` for pixel ordering.
|
||||||
- `options.raw.width` **[Number][5]?**
|
- `options.raw.width` **[Number][5]?**
|
||||||
|
|||||||
@@ -25,12 +25,15 @@ Fast access to (uncached) image metadata without decoding any compressed image d
|
|||||||
A Promises/A+ promise is returned when `callback` is not provided.
|
A Promises/A+ promise is returned when `callback` is not provided.
|
||||||
|
|
||||||
- `format`: Name of decoder used to decompress image data e.g. `jpeg`, `png`, `webp`, `gif`, `svg`
|
- `format`: Name of decoder used to decompress image data e.g. `jpeg`, `png`, `webp`, `gif`, `svg`
|
||||||
|
- `size`: Total size of image in bytes, for Stream and Buffer input only
|
||||||
- `width`: Number of pixels wide (EXIF orientation is not taken into consideration)
|
- `width`: Number of pixels wide (EXIF orientation is not taken into consideration)
|
||||||
- `height`: Number of pixels high (EXIF orientation is not taken into consideration)
|
- `height`: Number of pixels high (EXIF orientation is not taken into consideration)
|
||||||
- `space`: Name of colour space interpretation e.g. `srgb`, `rgb`, `cmyk`, `lab`, `b-w` [...][1]
|
- `space`: Name of colour space interpretation e.g. `srgb`, `rgb`, `cmyk`, `lab`, `b-w` [...][1]
|
||||||
- `channels`: Number of bands e.g. `3` for sRGB, `4` for CMYK
|
- `channels`: Number of bands e.g. `3` for sRGB, `4` for CMYK
|
||||||
- `depth`: Name of pixel depth format e.g. `uchar`, `char`, `ushort`, `float` [...][2]
|
- `depth`: Name of pixel depth format e.g. `uchar`, `char`, `ushort`, `float` [...][2]
|
||||||
- `density`: Number of pixels per inch (DPI), if present
|
- `density`: Number of pixels per inch (DPI), if present
|
||||||
|
- `chromaSubsampling`: String containing JPEG chroma subsampling, `4:2:0` or `4:4:4` for RGB, `4:2:0:4` or `4:4:4:4` for CMYK
|
||||||
|
- `isProgressive`: Boolean indicating whether the image is interlaced using a progressive scan
|
||||||
- `hasProfile`: Boolean indicating the presence of an embedded ICC profile
|
- `hasProfile`: Boolean indicating the presence of an embedded ICC profile
|
||||||
- `hasAlpha`: Boolean indicating the presence of an alpha transparency channel
|
- `hasAlpha`: Boolean indicating the presence of an alpha transparency channel
|
||||||
- `orientation`: Number value of the EXIF Orientation header, if present
|
- `orientation`: Number value of the EXIF Orientation header, if present
|
||||||
@@ -126,9 +129,9 @@ The default behaviour _before_ function call is `false`, meaning the libvips acc
|
|||||||
|
|
||||||
Returns **Sharp**
|
Returns **Sharp**
|
||||||
|
|
||||||
[1]: https://github.com/jcupitt/libvips/blob/master/libvips/iofuncs/enumtypes.c#L636
|
[1]: https://github.com/libvips/libvips/blob/master/libvips/iofuncs/enumtypes.c#L636
|
||||||
|
|
||||||
[2]: https://github.com/jcupitt/libvips/blob/master/libvips/iofuncs/enumtypes.c#L672
|
[2]: https://github.com/libvips/libvips/blob/master/libvips/iofuncs/enumtypes.c#L672
|
||||||
|
|
||||||
[3]: https://www.npmjs.com/package/icc
|
[3]: https://www.npmjs.com/package/icc
|
||||||
|
|
||||||
|
|||||||
@@ -5,9 +5,12 @@
|
|||||||
Rotate the output image by either an explicit angle
|
Rotate the output image by either an explicit angle
|
||||||
or auto-orient based on the EXIF `Orientation` tag.
|
or auto-orient based on the EXIF `Orientation` tag.
|
||||||
|
|
||||||
If an angle is provided, it is converted to a valid 90/180/270deg rotation.
|
If an angle is provided, it is converted to a valid positive degree rotation.
|
||||||
For example, `-450` will produce a 270deg rotation.
|
For example, `-450` will produce a 270deg rotation.
|
||||||
|
|
||||||
|
When rotating by an angle other than a multiple of 90,
|
||||||
|
the background colour can be provided with the `background` option.
|
||||||
|
|
||||||
If no angle is provided, it is determined from the EXIF data.
|
If no angle is provided, it is determined from the EXIF data.
|
||||||
Mirroring is supported and may infer the use of a flip operation.
|
Mirroring is supported and may infer the use of a flip operation.
|
||||||
|
|
||||||
@@ -18,7 +21,9 @@ for example `rotate(x).extract(y)` will produce a different result to `extract(y
|
|||||||
|
|
||||||
### Parameters
|
### Parameters
|
||||||
|
|
||||||
- `angle` **[Number][1]** angle of rotation, must be a multiple of 90. (optional, default `auto`)
|
- `angle` **[Number][1]** angle of rotation. (optional, default `auto`)
|
||||||
|
- `options` **[Object][2]?** if present, is an Object with optional attributes.
|
||||||
|
- `options.background` **([String][3] \| [Object][2])** parsed by the [color][4] module to extract values for red, green, blue and alpha. (optional, default `"#000000"`)
|
||||||
|
|
||||||
### Examples
|
### Examples
|
||||||
|
|
||||||
@@ -34,47 +39,7 @@ const pipeline = sharp()
|
|||||||
readableStream.pipe(pipeline);
|
readableStream.pipe(pipeline);
|
||||||
```
|
```
|
||||||
|
|
||||||
- Throws **[Error][2]** Invalid parameters
|
- Throws **[Error][5]** Invalid parameters
|
||||||
|
|
||||||
Returns **Sharp**
|
|
||||||
|
|
||||||
## extract
|
|
||||||
|
|
||||||
Extract a region of the image.
|
|
||||||
|
|
||||||
- Use `extract` before `resize` for pre-resize extraction.
|
|
||||||
- Use `extract` after `resize` for post-resize extraction.
|
|
||||||
- Use `extract` before and after for both.
|
|
||||||
|
|
||||||
### Parameters
|
|
||||||
|
|
||||||
- `options` **[Object][3]**
|
|
||||||
- `options.left` **[Number][1]** zero-indexed offset from left edge
|
|
||||||
- `options.top` **[Number][1]** zero-indexed offset from top edge
|
|
||||||
- `options.width` **[Number][1]** dimension of extracted image
|
|
||||||
- `options.height` **[Number][1]** dimension of extracted image
|
|
||||||
|
|
||||||
### Examples
|
|
||||||
|
|
||||||
```javascript
|
|
||||||
sharp(input)
|
|
||||||
.extract({ left: left, top: top, width: width, height: height })
|
|
||||||
.toFile(output, function(err) {
|
|
||||||
// Extract a region of the input image, saving in the same format.
|
|
||||||
});
|
|
||||||
```
|
|
||||||
|
|
||||||
```javascript
|
|
||||||
sharp(input)
|
|
||||||
.extract({ left: leftOffsetPre, top: topOffsetPre, width: widthPre, height: heightPre })
|
|
||||||
.resize(width, height)
|
|
||||||
.extract({ left: leftOffsetPost, top: topOffsetPost, width: widthPost, height: heightPost })
|
|
||||||
.toFile(output, function(err) {
|
|
||||||
// Extract a region, resize, then extract from the resized image
|
|
||||||
});
|
|
||||||
```
|
|
||||||
|
|
||||||
- Throws **[Error][2]** Invalid parameters
|
|
||||||
|
|
||||||
Returns **Sharp**
|
Returns **Sharp**
|
||||||
|
|
||||||
@@ -85,7 +50,7 @@ The use of `flip` implies the removal of the EXIF `Orientation` tag, if any.
|
|||||||
|
|
||||||
### Parameters
|
### Parameters
|
||||||
|
|
||||||
- `flip` **[Boolean][4]** (optional, default `true`)
|
- `flip` **[Boolean][6]** (optional, default `true`)
|
||||||
|
|
||||||
Returns **Sharp**
|
Returns **Sharp**
|
||||||
|
|
||||||
@@ -96,7 +61,7 @@ The use of `flop` implies the removal of the EXIF `Orientation` tag, if any.
|
|||||||
|
|
||||||
### Parameters
|
### Parameters
|
||||||
|
|
||||||
- `flop` **[Boolean][4]** (optional, default `true`)
|
- `flop` **[Boolean][6]** (optional, default `true`)
|
||||||
|
|
||||||
Returns **Sharp**
|
Returns **Sharp**
|
||||||
|
|
||||||
@@ -114,7 +79,7 @@ Separate control over the level of sharpening in "flat" and "jagged" areas is av
|
|||||||
- `jagged` **[Number][1]** the level of sharpening to apply to "jagged" areas. (optional, default `2.0`)
|
- `jagged` **[Number][1]** the level of sharpening to apply to "jagged" areas. (optional, default `2.0`)
|
||||||
|
|
||||||
|
|
||||||
- Throws **[Error][2]** Invalid parameters
|
- Throws **[Error][5]** Invalid parameters
|
||||||
|
|
||||||
Returns **Sharp**
|
Returns **Sharp**
|
||||||
|
|
||||||
@@ -128,7 +93,7 @@ When used without parameters the default window is 3x3.
|
|||||||
- `size` **[Number][1]** square mask size: size x size (optional, default `3`)
|
- `size` **[Number][1]** square mask size: size x size (optional, default `3`)
|
||||||
|
|
||||||
|
|
||||||
- Throws **[Error][2]** Invalid parameters
|
- Throws **[Error][5]** Invalid parameters
|
||||||
|
|
||||||
Returns **Sharp**
|
Returns **Sharp**
|
||||||
|
|
||||||
@@ -143,59 +108,18 @@ When a `sigma` is provided, performs a slower, more accurate Gaussian blur.
|
|||||||
- `sigma` **[Number][1]?** a value between 0.3 and 1000 representing the sigma of the Gaussian mask, where `sigma = 1 + radius / 2`.
|
- `sigma` **[Number][1]?** a value between 0.3 and 1000 representing the sigma of the Gaussian mask, where `sigma = 1 + radius / 2`.
|
||||||
|
|
||||||
|
|
||||||
- Throws **[Error][2]** Invalid parameters
|
- Throws **[Error][5]** Invalid parameters
|
||||||
|
|
||||||
Returns **Sharp**
|
|
||||||
|
|
||||||
## extend
|
|
||||||
|
|
||||||
Extends/pads the edges of the image with the colour provided to the `background` method.
|
|
||||||
This operation will always occur after resizing and extraction, if any.
|
|
||||||
|
|
||||||
### Parameters
|
|
||||||
|
|
||||||
- `extend` **([Number][1] \| [Object][3])** single pixel count to add to all edges or an Object with per-edge counts
|
|
||||||
- `extend.top` **[Number][1]?**
|
|
||||||
- `extend.left` **[Number][1]?**
|
|
||||||
- `extend.bottom` **[Number][1]?**
|
|
||||||
- `extend.right` **[Number][1]?**
|
|
||||||
|
|
||||||
### Examples
|
|
||||||
|
|
||||||
```javascript
|
|
||||||
// Resize to 140 pixels wide, then add 10 transparent pixels
|
|
||||||
// to the top, left and right edges and 20 to the bottom edge
|
|
||||||
sharp(input)
|
|
||||||
.resize(140)
|
|
||||||
.background({r: 0, g: 0, b: 0, alpha: 0})
|
|
||||||
.extend({top: 10, bottom: 20, left: 10, right: 10})
|
|
||||||
...
|
|
||||||
```
|
|
||||||
|
|
||||||
- Throws **[Error][2]** Invalid parameters
|
|
||||||
|
|
||||||
Returns **Sharp**
|
Returns **Sharp**
|
||||||
|
|
||||||
## flatten
|
## flatten
|
||||||
|
|
||||||
Merge alpha transparency channel, if any, with `background`.
|
Merge alpha transparency channel, if any, with a background.
|
||||||
|
|
||||||
### Parameters
|
### Parameters
|
||||||
|
|
||||||
- `flatten` **[Boolean][4]** (optional, default `true`)
|
- `options` **[Object][2]?**
|
||||||
|
- `options.background` **([String][3] \| [Object][2])** background colour, parsed by the [color][4] module, defaults to black. (optional, default `{r:0,g:0,b:0}`)
|
||||||
Returns **Sharp**
|
|
||||||
|
|
||||||
## trim
|
|
||||||
|
|
||||||
Trim "boring" pixels from all edges that contain values within a percentage similarity of the top-left pixel.
|
|
||||||
|
|
||||||
### Parameters
|
|
||||||
|
|
||||||
- `tolerance` **[Number][1]** value between 1 and 99 representing the percentage similarity. (optional, default `10`)
|
|
||||||
|
|
||||||
|
|
||||||
- Throws **[Error][2]** Invalid parameters
|
|
||||||
|
|
||||||
Returns **Sharp**
|
Returns **Sharp**
|
||||||
|
|
||||||
@@ -207,12 +131,15 @@ This can improve the perceived brightness of a resized image in non-linear colou
|
|||||||
JPEG and WebP input images will not take advantage of the shrink-on-load performance optimisation
|
JPEG and WebP input images will not take advantage of the shrink-on-load performance optimisation
|
||||||
when applying a gamma correction.
|
when applying a gamma correction.
|
||||||
|
|
||||||
|
Supply a second argument to use a different output gamma value, otherwise the first value is used in both cases.
|
||||||
|
|
||||||
### Parameters
|
### Parameters
|
||||||
|
|
||||||
- `gamma` **[Number][1]** value between 1.0 and 3.0. (optional, default `2.2`)
|
- `gamma` **[Number][1]** value between 1.0 and 3.0. (optional, default `2.2`)
|
||||||
|
- `gammaOut` **[Number][1]?** value between 1.0 and 3.0. (optional, defaults to same as `gamma`)
|
||||||
|
|
||||||
|
|
||||||
- Throws **[Error][2]** Invalid parameters
|
- Throws **[Error][5]** Invalid parameters
|
||||||
|
|
||||||
Returns **Sharp**
|
Returns **Sharp**
|
||||||
|
|
||||||
@@ -222,7 +149,7 @@ Produce the "negative" of the image.
|
|||||||
|
|
||||||
### Parameters
|
### Parameters
|
||||||
|
|
||||||
- `negate` **[Boolean][4]** (optional, default `true`)
|
- `negate` **[Boolean][6]** (optional, default `true`)
|
||||||
|
|
||||||
Returns **Sharp**
|
Returns **Sharp**
|
||||||
|
|
||||||
@@ -232,7 +159,7 @@ Enhance output image contrast by stretching its luminance to cover the full dyna
|
|||||||
|
|
||||||
### Parameters
|
### Parameters
|
||||||
|
|
||||||
- `normalise` **[Boolean][4]** (optional, default `true`)
|
- `normalise` **[Boolean][6]** (optional, default `true`)
|
||||||
|
|
||||||
Returns **Sharp**
|
Returns **Sharp**
|
||||||
|
|
||||||
@@ -242,7 +169,7 @@ Alternative spelling of normalise.
|
|||||||
|
|
||||||
### Parameters
|
### Parameters
|
||||||
|
|
||||||
- `normalize` **[Boolean][4]** (optional, default `true`)
|
- `normalize` **[Boolean][6]** (optional, default `true`)
|
||||||
|
|
||||||
Returns **Sharp**
|
Returns **Sharp**
|
||||||
|
|
||||||
@@ -252,10 +179,10 @@ Convolve the image with the specified kernel.
|
|||||||
|
|
||||||
### Parameters
|
### Parameters
|
||||||
|
|
||||||
- `kernel` **[Object][3]**
|
- `kernel` **[Object][2]**
|
||||||
- `kernel.width` **[Number][1]** width of the kernel in pixels.
|
- `kernel.width` **[Number][1]** width of the kernel in pixels.
|
||||||
- `kernel.height` **[Number][1]** width of the kernel in pixels.
|
- `kernel.height` **[Number][1]** width of the kernel in pixels.
|
||||||
- `kernel.kernel` **[Array][5]<[Number][1]>** Array of length `width*height` containing the kernel values.
|
- `kernel.kernel` **[Array][7]<[Number][1]>** Array of length `width*height` containing the kernel values.
|
||||||
- `kernel.scale` **[Number][1]** the scale of the kernel in pixels. (optional, default `sum`)
|
- `kernel.scale` **[Number][1]** the scale of the kernel in pixels. (optional, default `sum`)
|
||||||
- `kernel.offset` **[Number][1]** the offset of the kernel in pixels. (optional, default `0`)
|
- `kernel.offset` **[Number][1]** the offset of the kernel in pixels. (optional, default `0`)
|
||||||
|
|
||||||
@@ -275,7 +202,7 @@ sharp(input)
|
|||||||
});
|
});
|
||||||
```
|
```
|
||||||
|
|
||||||
- Throws **[Error][2]** Invalid parameters
|
- Throws **[Error][5]** Invalid parameters
|
||||||
|
|
||||||
Returns **Sharp**
|
Returns **Sharp**
|
||||||
|
|
||||||
@@ -286,12 +213,12 @@ Any pixel value greather than or equal to the threshold value will be set to 255
|
|||||||
### Parameters
|
### Parameters
|
||||||
|
|
||||||
- `threshold` **[Number][1]** a value in the range 0-255 representing the level at which the threshold will be applied. (optional, default `128`)
|
- `threshold` **[Number][1]** a value in the range 0-255 representing the level at which the threshold will be applied. (optional, default `128`)
|
||||||
- `options` **[Object][3]?**
|
- `options` **[Object][2]?**
|
||||||
- `options.greyscale` **[Boolean][4]** convert to single channel greyscale. (optional, default `true`)
|
- `options.greyscale` **[Boolean][6]** convert to single channel greyscale. (optional, default `true`)
|
||||||
- `options.grayscale` **[Boolean][4]** alternative spelling for greyscale. (optional, default `true`)
|
- `options.grayscale` **[Boolean][6]** alternative spelling for greyscale. (optional, default `true`)
|
||||||
|
|
||||||
|
|
||||||
- Throws **[Error][2]** Invalid parameters
|
- Throws **[Error][5]** Invalid parameters
|
||||||
|
|
||||||
Returns **Sharp**
|
Returns **Sharp**
|
||||||
|
|
||||||
@@ -304,16 +231,16 @@ the selected bitwise boolean `operation` between the corresponding pixels of the
|
|||||||
|
|
||||||
### Parameters
|
### Parameters
|
||||||
|
|
||||||
- `operand` **([Buffer][6] \| [String][7])** Buffer containing image data or String containing the path to an image file.
|
- `operand` **([Buffer][8] \| [String][3])** Buffer containing image data or String containing the path to an image file.
|
||||||
- `operator` **[String][7]** one of `and`, `or` or `eor` to perform that bitwise operation, like the C logic operators `&`, `|` and `^` respectively.
|
- `operator` **[String][3]** one of `and`, `or` or `eor` to perform that bitwise operation, like the C logic operators `&`, `|` and `^` respectively.
|
||||||
- `options` **[Object][3]?**
|
- `options` **[Object][2]?**
|
||||||
- `options.raw` **[Object][3]?** describes operand when using raw pixel data.
|
- `options.raw` **[Object][2]?** describes operand when using raw pixel data.
|
||||||
- `options.raw.width` **[Number][1]?**
|
- `options.raw.width` **[Number][1]?**
|
||||||
- `options.raw.height` **[Number][1]?**
|
- `options.raw.height` **[Number][1]?**
|
||||||
- `options.raw.channels` **[Number][1]?**
|
- `options.raw.channels` **[Number][1]?**
|
||||||
|
|
||||||
|
|
||||||
- Throws **[Error][2]** Invalid parameters
|
- Throws **[Error][5]** Invalid parameters
|
||||||
|
|
||||||
Returns **Sharp**
|
Returns **Sharp**
|
||||||
|
|
||||||
@@ -327,20 +254,51 @@ Apply the linear formula a \* input + b to the image (levels adjustment)
|
|||||||
- `b` **[Number][1]** offset (optional, default `0.0`)
|
- `b` **[Number][1]** offset (optional, default `0.0`)
|
||||||
|
|
||||||
|
|
||||||
- Throws **[Error][2]** Invalid parameters
|
- Throws **[Error][5]** Invalid parameters
|
||||||
|
|
||||||
|
Returns **Sharp**
|
||||||
|
|
||||||
|
## recomb
|
||||||
|
|
||||||
|
Recomb the image with the specified matrix.
|
||||||
|
|
||||||
|
### Parameters
|
||||||
|
|
||||||
|
- `inputMatrix`
|
||||||
|
- `3x3` **[Array][7]<[Array][7]<[Number][1]>>** Recombination matrix
|
||||||
|
|
||||||
|
### Examples
|
||||||
|
|
||||||
|
```javascript
|
||||||
|
sharp(input)
|
||||||
|
.recomb([
|
||||||
|
[0.3588, 0.7044, 0.1368],
|
||||||
|
[0.2990, 0.5870, 0.1140],
|
||||||
|
[0.2392, 0.4696, 0.0912],
|
||||||
|
])
|
||||||
|
.raw()
|
||||||
|
.toBuffer(function(err, data, info) {
|
||||||
|
// data contains the raw pixel data after applying the recomb
|
||||||
|
// With this example input, a sepia filter has been applied
|
||||||
|
});
|
||||||
|
```
|
||||||
|
|
||||||
|
- Throws **[Error][5]** Invalid parameters
|
||||||
|
|
||||||
Returns **Sharp**
|
Returns **Sharp**
|
||||||
|
|
||||||
[1]: https://developer.mozilla.org/docs/Web/JavaScript/Reference/Global_Objects/Number
|
[1]: https://developer.mozilla.org/docs/Web/JavaScript/Reference/Global_Objects/Number
|
||||||
|
|
||||||
[2]: https://developer.mozilla.org/docs/Web/JavaScript/Reference/Global_Objects/Error
|
[2]: https://developer.mozilla.org/docs/Web/JavaScript/Reference/Global_Objects/Object
|
||||||
|
|
||||||
[3]: https://developer.mozilla.org/docs/Web/JavaScript/Reference/Global_Objects/Object
|
[3]: https://developer.mozilla.org/docs/Web/JavaScript/Reference/Global_Objects/String
|
||||||
|
|
||||||
[4]: https://developer.mozilla.org/docs/Web/JavaScript/Reference/Global_Objects/Boolean
|
[4]: https://www.npmjs.org/package/color
|
||||||
|
|
||||||
[5]: https://developer.mozilla.org/docs/Web/JavaScript/Reference/Global_Objects/Array
|
[5]: https://developer.mozilla.org/docs/Web/JavaScript/Reference/Global_Objects/Error
|
||||||
|
|
||||||
[6]: https://nodejs.org/api/buffer.html
|
[6]: https://developer.mozilla.org/docs/Web/JavaScript/Reference/Global_Objects/Boolean
|
||||||
|
|
||||||
[7]: https://developer.mozilla.org/docs/Web/JavaScript/Reference/Global_Objects/String
|
[7]: https://developer.mozilla.org/docs/Web/JavaScript/Reference/Global_Objects/Array
|
||||||
|
|
||||||
|
[8]: https://nodejs.org/api/buffer.html
|
||||||
|
|||||||
@@ -206,6 +206,10 @@ Use these TIFF options for output image.
|
|||||||
- `options.force` **[Boolean][6]** force TIFF output, otherwise attempt to use input format (optional, default `true`)
|
- `options.force` **[Boolean][6]** force TIFF output, otherwise attempt to use input format (optional, default `true`)
|
||||||
- `options.compression` **[Boolean][6]** compression options: lzw, deflate, jpeg, ccittfax4 (optional, default `'jpeg'`)
|
- `options.compression` **[Boolean][6]** compression options: lzw, deflate, jpeg, ccittfax4 (optional, default `'jpeg'`)
|
||||||
- `options.predictor` **[Boolean][6]** compression predictor options: none, horizontal, float (optional, default `'horizontal'`)
|
- `options.predictor` **[Boolean][6]** compression predictor options: none, horizontal, float (optional, default `'horizontal'`)
|
||||||
|
- `options.pyramid` **[Boolean][6]** write an image pyramid (optional, default `false`)
|
||||||
|
- `options.tile` **[Boolean][6]** write a tiled tiff (optional, default `false`)
|
||||||
|
- `options.tileWidth` **[Boolean][6]** horizontal tile size (optional, default `256`)
|
||||||
|
- `options.tileHeight` **[Boolean][6]** vertical tile size (optional, default `256`)
|
||||||
- `options.xres` **[Number][8]** horizontal resolution in pixels/mm (optional, default `1.0`)
|
- `options.xres` **[Number][8]** horizontal resolution in pixels/mm (optional, default `1.0`)
|
||||||
- `options.yres` **[Number][8]** vertical resolution in pixels/mm (optional, default `1.0`)
|
- `options.yres` **[Number][8]** vertical resolution in pixels/mm (optional, default `1.0`)
|
||||||
- `options.squash` **[Boolean][6]** squash 8-bit images down to 1 bit (optional, default `false`)
|
- `options.squash` **[Boolean][6]** squash 8-bit images down to 1 bit (optional, default `false`)
|
||||||
|
|||||||
@@ -2,181 +2,233 @@
|
|||||||
|
|
||||||
## resize
|
## resize
|
||||||
|
|
||||||
Resize image to `width` x `height`.
|
Resize image to `width`, `height` or `width x height`.
|
||||||
By default, the resized image is centre cropped to the exact size specified.
|
|
||||||
|
|
||||||
Possible kernels are:
|
When both a `width` and `height` are provided, the possible methods by which the image should **fit** these are:
|
||||||
|
|
||||||
- `nearest`: Use [nearest neighbour interpolation][1].
|
- `cover`: Crop to cover both provided dimensions (the default).
|
||||||
- `cubic`: Use a [Catmull-Rom spline][2].
|
- `contain`: Embed within both provided dimensions.
|
||||||
- `lanczos2`: Use a [Lanczos kernel][3] with `a=2`.
|
- `fill`: Ignore the aspect ratio of the input and stretch to both provided dimensions.
|
||||||
- `lanczos3`: Use a Lanczos kernel with `a=3` (the default).
|
- `inside`: Preserving aspect ratio, resize the image to be as large as possible while ensuring its dimensions are less than or equal to both those specified.
|
||||||
|
- `outside`: Preserving aspect ratio, resize the image to be as small as possible while ensuring its dimensions are greater than or equal to both those specified.
|
||||||
|
Some of these values are based on the [object-fit][1] CSS property.
|
||||||
|
|
||||||
### Parameters
|
When using a `fit` of `cover` or `contain`, the default **position** is `centre`. Other options are:
|
||||||
|
|
||||||
- `width` **[Number][4]?** pixels wide the resultant image should be. Use `null` or `undefined` to auto-scale the width to match the height.
|
- `sharp.position`: `top`, `right top`, `right`, `right bottom`, `bottom`, `left bottom`, `left`, `left top`.
|
||||||
- `height` **[Number][4]?** pixels high the resultant image should be. Use `null` or `undefined` to auto-scale the height to match the width.
|
- `sharp.gravity`: `north`, `northeast`, `east`, `southeast`, `south`, `southwest`, `west`, `northwest`, `center` or `centre`.
|
||||||
- `options` **[Object][5]?**
|
- `sharp.strategy`: `cover` only, dynamically crop using either the `entropy` or `attention` strategy.
|
||||||
- `options.kernel` **[String][6]** the kernel to use for image reduction. (optional, default `'lanczos3'`)
|
Some of these values are based on the [object-position][2] CSS property.
|
||||||
- `options.fastShrinkOnLoad` **[Boolean][7]** take greater advantage of the JPEG and WebP shrink-on-load feature, which can lead to a slight moiré pattern on some images. (optional, default `true`)
|
|
||||||
|
|
||||||
### Examples
|
|
||||||
|
|
||||||
```javascript
|
|
||||||
sharp(inputBuffer)
|
|
||||||
.resize(200, 300, {
|
|
||||||
kernel: sharp.kernel.nearest
|
|
||||||
})
|
|
||||||
.background('white')
|
|
||||||
.embed()
|
|
||||||
.toFile('output.tiff')
|
|
||||||
.then(function() {
|
|
||||||
// output.tiff is a 200 pixels wide and 300 pixels high image
|
|
||||||
// containing a nearest-neighbour scaled version, embedded on a white canvas,
|
|
||||||
// of the image data in inputBuffer
|
|
||||||
});
|
|
||||||
```
|
|
||||||
|
|
||||||
- Throws **[Error][8]** Invalid parameters
|
|
||||||
|
|
||||||
Returns **Sharp**
|
|
||||||
|
|
||||||
## crop
|
|
||||||
|
|
||||||
Crop the resized image to the exact size specified, the default behaviour.
|
|
||||||
|
|
||||||
Possible attributes of the optional `sharp.gravity` are `north`, `northeast`, `east`, `southeast`, `south`,
|
|
||||||
`southwest`, `west`, `northwest`, `center` and `centre`.
|
|
||||||
|
|
||||||
The experimental strategy-based approach resizes so one dimension is at its target length
|
The experimental strategy-based approach resizes so one dimension is at its target length
|
||||||
then repeatedly ranks edge regions, discarding the edge with the lowest score based on the selected strategy.
|
then repeatedly ranks edge regions, discarding the edge with the lowest score based on the selected strategy.
|
||||||
|
|
||||||
- `entropy`: focus on the region with the highest [Shannon entropy][9].
|
- `entropy`: focus on the region with the highest [Shannon entropy][3].
|
||||||
- `attention`: focus on the region with the highest luminance frequency, colour saturation and presence of skin tones.
|
- `attention`: focus on the region with the highest luminance frequency, colour saturation and presence of skin tones.
|
||||||
|
|
||||||
|
Possible interpolation kernels are:
|
||||||
|
|
||||||
|
- `nearest`: Use [nearest neighbour interpolation][4].
|
||||||
|
- `cubic`: Use a [Catmull-Rom spline][5].
|
||||||
|
- `mitchell`: Use a [Mitchell-Netravali spline][6].
|
||||||
|
- `lanczos2`: Use a [Lanczos kernel][7] with `a=2`.
|
||||||
|
- `lanczos3`: Use a Lanczos kernel with `a=3` (the default).
|
||||||
|
|
||||||
### Parameters
|
### Parameters
|
||||||
|
|
||||||
- `crop` **[String][6]** A member of `sharp.gravity` to crop to an edge/corner or `sharp.strategy` to crop dynamically. (optional, default `'centre'`)
|
- `width` **[Number][8]?** pixels wide the resultant image should be. Use `null` or `undefined` to auto-scale the width to match the height.
|
||||||
|
- `height` **[Number][8]?** pixels high the resultant image should be. Use `null` or `undefined` to auto-scale the height to match the width.
|
||||||
|
- `options` **[Object][9]?**
|
||||||
|
- `options.width` **[String][10]?** alternative means of specifying `width`. If both are present this take priority.
|
||||||
|
- `options.height` **[String][10]?** alternative means of specifying `height`. If both are present this take priority.
|
||||||
|
- `options.fit` **[String][10]** how the image should be resized to fit both provided dimensions, one of `cover`, `contain`, `fill`, `inside` or `outside`. (optional, default `'cover'`)
|
||||||
|
- `options.position` **[String][10]** position, gravity or strategy to use when `fit` is `cover` or `contain`. (optional, default `'centre'`)
|
||||||
|
- `options.background` **([String][10] \| [Object][9])** background colour when using a `fit` of `contain`, parsed by the [color][11] module, defaults to black without transparency. (optional, default `{r:0,g:0,b:0,alpha:1}`)
|
||||||
|
- `options.kernel` **[String][10]** the kernel to use for image reduction. (optional, default `'lanczos3'`)
|
||||||
|
- `options.withoutEnlargement` **[Boolean][12]** do not enlarge if the width _or_ height are already less than the specified dimensions, equivalent to GraphicsMagick's `>` geometry option. (optional, default `false`)
|
||||||
|
- `options.fastShrinkOnLoad` **[Boolean][12]** take greater advantage of the JPEG and WebP shrink-on-load feature, which can lead to a slight moiré pattern on some images. (optional, default `true`)
|
||||||
|
|
||||||
### Examples
|
### Examples
|
||||||
|
|
||||||
|
```javascript
|
||||||
|
sharp(input)
|
||||||
|
.resize({ width: 100 })
|
||||||
|
.toBuffer()
|
||||||
|
.then(data => {
|
||||||
|
// 100 pixels wide, auto-scaled height
|
||||||
|
});
|
||||||
|
```
|
||||||
|
|
||||||
|
```javascript
|
||||||
|
sharp(input)
|
||||||
|
.resize({ height: 100 })
|
||||||
|
.toBuffer()
|
||||||
|
.then(data => {
|
||||||
|
// 100 pixels high, auto-scaled width
|
||||||
|
});
|
||||||
|
```
|
||||||
|
|
||||||
|
```javascript
|
||||||
|
sharp(input)
|
||||||
|
.resize(200, 300, {
|
||||||
|
kernel: sharp.kernel.nearest,
|
||||||
|
fit: 'contain',
|
||||||
|
position: 'right top',
|
||||||
|
background: { r: 255, g: 255, b: 255, alpha: 0.5 }
|
||||||
|
})
|
||||||
|
.toFile('output.png')
|
||||||
|
.then(() => {
|
||||||
|
// output.png is a 200 pixels wide and 300 pixels high image
|
||||||
|
// containing a nearest-neighbour scaled version
|
||||||
|
// contained within the north-east corner of a semi-transparent white canvas
|
||||||
|
});
|
||||||
|
```
|
||||||
|
|
||||||
```javascript
|
```javascript
|
||||||
const transformer = sharp()
|
const transformer = sharp()
|
||||||
.resize(200, 200)
|
.resize({
|
||||||
.crop(sharp.strategy.entropy)
|
width: 200,
|
||||||
.on('error', function(err) {
|
height: 200,
|
||||||
console.log(err);
|
fit: sharp.fit.cover,
|
||||||
|
position: sharp.strategy.entropy
|
||||||
});
|
});
|
||||||
// Read image data from readableStream
|
// Read image data from readableStream
|
||||||
// Write 200px square auto-cropped image data to writableStream
|
// Write 200px square auto-cropped image data to writableStream
|
||||||
readableStream.pipe(transformer).pipe(writableStream);
|
readableStream
|
||||||
|
.pipe(transformer)
|
||||||
|
.pipe(writableStream);
|
||||||
```
|
```
|
||||||
|
|
||||||
- Throws **[Error][8]** Invalid parameters
|
|
||||||
|
|
||||||
Returns **Sharp**
|
|
||||||
|
|
||||||
## embed
|
|
||||||
|
|
||||||
Preserving aspect ratio, resize the image to the maximum `width` or `height` specified
|
|
||||||
then embed on a background of the exact `width` and `height` specified.
|
|
||||||
|
|
||||||
If the background contains an alpha value then WebP and PNG format output images will
|
|
||||||
contain an alpha channel, even when the input image does not.
|
|
||||||
|
|
||||||
### Parameters
|
|
||||||
|
|
||||||
- `embed` **[String][6]** A member of `sharp.gravity` to embed to an edge/corner. (optional, default `'centre'`)
|
|
||||||
|
|
||||||
### Examples
|
|
||||||
|
|
||||||
```javascript
|
```javascript
|
||||||
sharp('input.gif')
|
sharp(input)
|
||||||
.resize(200, 300)
|
.resize(200, 200, {
|
||||||
.background({r: 0, g: 0, b: 0, alpha: 0})
|
fit: sharp.fit.inside,
|
||||||
.embed()
|
withoutEnlargement: true
|
||||||
.toFormat(sharp.format.webp)
|
})
|
||||||
.toBuffer(function(err, outputBuffer) {
|
|
||||||
if (err) {
|
|
||||||
throw err;
|
|
||||||
}
|
|
||||||
// outputBuffer contains WebP image data of a 200 pixels wide and 300 pixels high
|
|
||||||
// containing a scaled version, embedded on a transparent canvas, of input.gif
|
|
||||||
});
|
|
||||||
```
|
|
||||||
|
|
||||||
- Throws **[Error][8]** Invalid parameters
|
|
||||||
|
|
||||||
Returns **Sharp**
|
|
||||||
|
|
||||||
## max
|
|
||||||
|
|
||||||
Preserving aspect ratio, resize the image to be as large as possible
|
|
||||||
while ensuring its dimensions are less than or equal to the `width` and `height` specified.
|
|
||||||
|
|
||||||
Both `width` and `height` must be provided via `resize` otherwise the behaviour will default to `crop`.
|
|
||||||
|
|
||||||
### Examples
|
|
||||||
|
|
||||||
```javascript
|
|
||||||
sharp(inputBuffer)
|
|
||||||
.resize(200, 200)
|
|
||||||
.max()
|
|
||||||
.toFormat('jpeg')
|
.toFormat('jpeg')
|
||||||
.toBuffer()
|
.toBuffer()
|
||||||
.then(function(outputBuffer) {
|
.then(function(outputBuffer) {
|
||||||
// outputBuffer contains JPEG image data no wider than 200 pixels and no higher
|
// outputBuffer contains JPEG image data
|
||||||
// than 200 pixels regardless of the inputBuffer image dimensions
|
// no wider and no higher than 200 pixels
|
||||||
|
// and no larger than the input image
|
||||||
});
|
});
|
||||||
```
|
```
|
||||||
|
|
||||||
Returns **Sharp**
|
- Throws **[Error][13]** Invalid parameters
|
||||||
|
|
||||||
## min
|
|
||||||
|
|
||||||
Preserving aspect ratio, resize the image to be as small as possible
|
|
||||||
while ensuring its dimensions are greater than or equal to the `width` and `height` specified.
|
|
||||||
|
|
||||||
Both `width` and `height` must be provided via `resize` otherwise the behaviour will default to `crop`.
|
|
||||||
|
|
||||||
Returns **Sharp**
|
Returns **Sharp**
|
||||||
|
|
||||||
## ignoreAspectRatio
|
## extend
|
||||||
|
|
||||||
Ignoring the aspect ratio of the input, stretch the image to
|
Extends/pads the edges of the image with the provided background colour.
|
||||||
the exact `width` and/or `height` provided via `resize`.
|
This operation will always occur after resizing and extraction, if any.
|
||||||
|
|
||||||
Returns **Sharp**
|
|
||||||
|
|
||||||
## withoutEnlargement
|
|
||||||
|
|
||||||
Do not enlarge the output image if the input image width _or_ height are already less than the required dimensions.
|
|
||||||
This is equivalent to GraphicsMagick's `>` geometry option:
|
|
||||||
"_change the dimensions of the image only if its width or height exceeds the geometry specification_".
|
|
||||||
Use with `max()` to preserve the image's aspect ratio.
|
|
||||||
|
|
||||||
The default behaviour _before_ function call is `false`, meaning the image will be enlarged.
|
|
||||||
|
|
||||||
### Parameters
|
### Parameters
|
||||||
|
|
||||||
- `withoutEnlargement` **[Boolean][7]** (optional, default `true`)
|
- `extend` **([Number][8] \| [Object][9])** single pixel count to add to all edges or an Object with per-edge counts
|
||||||
|
- `extend.top` **[Number][8]?**
|
||||||
|
- `extend.left` **[Number][8]?**
|
||||||
|
- `extend.bottom` **[Number][8]?**
|
||||||
|
- `extend.right` **[Number][8]?**
|
||||||
|
- `extend.background` **([String][10] \| [Object][9])** background colour, parsed by the [color][11] module, defaults to black without transparency. (optional, default `{r:0,g:0,b:0,alpha:1}`)
|
||||||
|
|
||||||
|
### Examples
|
||||||
|
|
||||||
|
```javascript
|
||||||
|
// Resize to 140 pixels wide, then add 10 transparent pixels
|
||||||
|
// to the top, left and right edges and 20 to the bottom edge
|
||||||
|
sharp(input)
|
||||||
|
.resize(140)
|
||||||
|
.extend({
|
||||||
|
top: 10,
|
||||||
|
bottom: 20,
|
||||||
|
left: 10,
|
||||||
|
right: 10
|
||||||
|
background: { r: 0, g: 0, b: 0, alpha: 0 }
|
||||||
|
})
|
||||||
|
...
|
||||||
|
```
|
||||||
|
|
||||||
|
- Throws **[Error][13]** Invalid parameters
|
||||||
|
|
||||||
Returns **Sharp**
|
Returns **Sharp**
|
||||||
|
|
||||||
[1]: http://en.wikipedia.org/wiki/Nearest-neighbor_interpolation
|
## extract
|
||||||
|
|
||||||
[2]: https://en.wikipedia.org/wiki/Centripetal_Catmull%E2%80%93Rom_spline
|
Extract a region of the image.
|
||||||
|
|
||||||
[3]: https://en.wikipedia.org/wiki/Lanczos_resampling#Lanczos_kernel
|
- Use `extract` before `resize` for pre-resize extraction.
|
||||||
|
- Use `extract` after `resize` for post-resize extraction.
|
||||||
|
- Use `extract` before and after for both.
|
||||||
|
|
||||||
[4]: https://developer.mozilla.org/docs/Web/JavaScript/Reference/Global_Objects/Number
|
### Parameters
|
||||||
|
|
||||||
[5]: https://developer.mozilla.org/docs/Web/JavaScript/Reference/Global_Objects/Object
|
- `options` **[Object][9]**
|
||||||
|
- `options.left` **[Number][8]** zero-indexed offset from left edge
|
||||||
|
- `options.top` **[Number][8]** zero-indexed offset from top edge
|
||||||
|
- `options.width` **[Number][8]** dimension of extracted image
|
||||||
|
- `options.height` **[Number][8]** dimension of extracted image
|
||||||
|
|
||||||
[6]: https://developer.mozilla.org/docs/Web/JavaScript/Reference/Global_Objects/String
|
### Examples
|
||||||
|
|
||||||
[7]: https://developer.mozilla.org/docs/Web/JavaScript/Reference/Global_Objects/Boolean
|
```javascript
|
||||||
|
sharp(input)
|
||||||
|
.extract({ left: left, top: top, width: width, height: height })
|
||||||
|
.toFile(output, function(err) {
|
||||||
|
// Extract a region of the input image, saving in the same format.
|
||||||
|
});
|
||||||
|
```
|
||||||
|
|
||||||
[8]: https://developer.mozilla.org/docs/Web/JavaScript/Reference/Global_Objects/Error
|
```javascript
|
||||||
|
sharp(input)
|
||||||
|
.extract({ left: leftOffsetPre, top: topOffsetPre, width: widthPre, height: heightPre })
|
||||||
|
.resize(width, height)
|
||||||
|
.extract({ left: leftOffsetPost, top: topOffsetPost, width: widthPost, height: heightPost })
|
||||||
|
.toFile(output, function(err) {
|
||||||
|
// Extract a region, resize, then extract from the resized image
|
||||||
|
});
|
||||||
|
```
|
||||||
|
|
||||||
[9]: https://en.wikipedia.org/wiki/Entropy_%28information_theory%29
|
- Throws **[Error][13]** Invalid parameters
|
||||||
|
|
||||||
|
Returns **Sharp**
|
||||||
|
|
||||||
|
## trim
|
||||||
|
|
||||||
|
Trim "boring" pixels from all edges that contain values similar to the top-left pixel.
|
||||||
|
The `info` response Object will contain `trimOffsetLeft` and `trimOffsetTop` properties.
|
||||||
|
|
||||||
|
### Parameters
|
||||||
|
|
||||||
|
- `threshold` **[Number][8]** the allowed difference from the top-left pixel, a number greater than zero. (optional, default `10`)
|
||||||
|
|
||||||
|
|
||||||
|
- Throws **[Error][13]** Invalid parameters
|
||||||
|
|
||||||
|
Returns **Sharp**
|
||||||
|
|
||||||
|
[1]: https://developer.mozilla.org/en-US/docs/Web/CSS/object-fit
|
||||||
|
|
||||||
|
[2]: https://developer.mozilla.org/en-US/docs/Web/CSS/object-position
|
||||||
|
|
||||||
|
[3]: https://en.wikipedia.org/wiki/Entropy_%28information_theory%29
|
||||||
|
|
||||||
|
[4]: http://en.wikipedia.org/wiki/Nearest-neighbor_interpolation
|
||||||
|
|
||||||
|
[5]: https://en.wikipedia.org/wiki/Centripetal_Catmull%E2%80%93Rom_spline
|
||||||
|
|
||||||
|
[6]: https://www.cs.utexas.edu/~fussell/courses/cs384g-fall2013/lectures/mitchell/Mitchell.pdf
|
||||||
|
|
||||||
|
[7]: https://en.wikipedia.org/wiki/Lanczos_resampling#Lanczos_kernel
|
||||||
|
|
||||||
|
[8]: https://developer.mozilla.org/docs/Web/JavaScript/Reference/Global_Objects/Number
|
||||||
|
|
||||||
|
[9]: https://developer.mozilla.org/docs/Web/JavaScript/Reference/Global_Objects/Object
|
||||||
|
|
||||||
|
[10]: https://developer.mozilla.org/docs/Web/JavaScript/Reference/Global_Objects/String
|
||||||
|
|
||||||
|
[11]: https://www.npmjs.org/package/color
|
||||||
|
|
||||||
|
[12]: https://developer.mozilla.org/docs/Web/JavaScript/Reference/Global_Objects/Boolean
|
||||||
|
|
||||||
|
[13]: https://developer.mozilla.org/docs/Web/JavaScript/Reference/Global_Objects/Error
|
||||||
|
|||||||
@@ -77,23 +77,20 @@ Requires libvips to have been compiled with liborc support.
|
|||||||
Improves the performance of `resize`, `blur` and `sharpen` operations
|
Improves the performance of `resize`, `blur` and `sharpen` operations
|
||||||
by taking advantage of the SIMD vector unit of the CPU, e.g. Intel SSE and ARM NEON.
|
by taking advantage of the SIMD vector unit of the CPU, e.g. Intel SSE and ARM NEON.
|
||||||
|
|
||||||
This feature is currently off by default but future versions may reverse this.
|
|
||||||
Versions of liborc prior to 0.4.25 are known to segfault under heavy load.
|
|
||||||
|
|
||||||
### Parameters
|
### Parameters
|
||||||
|
|
||||||
- `simd` **[Boolean][2]** (optional, default `false`)
|
- `simd` **[Boolean][2]** (optional, default `true`)
|
||||||
|
|
||||||
### Examples
|
### Examples
|
||||||
|
|
||||||
```javascript
|
```javascript
|
||||||
const simd = sharp.simd();
|
const simd = sharp.simd();
|
||||||
// simd is `true` if SIMD is currently enabled
|
// simd is `true` if the runtime use of liborc is currently enabled
|
||||||
```
|
```
|
||||||
|
|
||||||
```javascript
|
```javascript
|
||||||
const simd = sharp.simd(true);
|
const simd = sharp.simd(false);
|
||||||
// attempts to enable the use of SIMD, returning true if available
|
// prevent libvips from using liborc at runtime
|
||||||
```
|
```
|
||||||
|
|
||||||
Returns **[Boolean][2]**
|
Returns **[Boolean][2]**
|
||||||
|
|||||||
@@ -1,9 +1,89 @@
|
|||||||
# Changelog
|
# Changelog
|
||||||
|
|
||||||
|
### v0.21 - "*teeth*"
|
||||||
|
|
||||||
|
Requires libvips v8.7.0.
|
||||||
|
|
||||||
|
#### v0.21.1 - 7<sup>th</sup> December 2018
|
||||||
|
|
||||||
|
* Install: support `sharp_dist_base_url` npm config, like existing `SHARP_DIST_BASE_URL`.
|
||||||
|
[#1422](https://github.com/lovell/sharp/pull/1422)
|
||||||
|
[@SethWen](https://github.com/SethWen)
|
||||||
|
|
||||||
|
* Ensure `channel` metadata is correct for raw, greyscale output.
|
||||||
|
[#1425](https://github.com/lovell/sharp/issues/1425)
|
||||||
|
|
||||||
|
* Add support for the "mitchell" kernel for image reductions.
|
||||||
|
[#1438](https://github.com/lovell/sharp/pull/1438)
|
||||||
|
[@Daiz](https://github.com/Daiz)
|
||||||
|
|
||||||
|
* Allow separate parameters for gamma encoding and decoding.
|
||||||
|
[#1439](https://github.com/lovell/sharp/pull/1439)
|
||||||
|
[@Daiz](https://github.com/Daiz)
|
||||||
|
|
||||||
|
* Build prototype with `Object.assign` to allow minification.
|
||||||
|
[#1475](https://github.com/lovell/sharp/pull/1475)
|
||||||
|
[@jaubourg](https://github.com/jaubourg)
|
||||||
|
|
||||||
|
* Expose libvips' recombination matrix operation.
|
||||||
|
[#1477](https://github.com/lovell/sharp/pull/1477)
|
||||||
|
[@fromkeith](https://github.com/fromkeith)
|
||||||
|
|
||||||
|
* Expose libvips' pyramid/tile options for TIFF output.
|
||||||
|
[#1483](https://github.com/lovell/sharp/pull/1483)
|
||||||
|
[@mbklein](https://github.com/mbklein)
|
||||||
|
|
||||||
|
#### v0.21.0 - 4<sup>th</sup> October 2018
|
||||||
|
|
||||||
|
* Deprecate the following resize-related functions:
|
||||||
|
`crop`, `embed`, `ignoreAspectRatio`, `max`, `min` and `withoutEnlargement`.
|
||||||
|
Access to these is now via options passed to the `resize` function.
|
||||||
|
For example:
|
||||||
|
`embed('north')` is now `resize(width, height, { fit: 'contain', position: 'north' })`,
|
||||||
|
`crop('attention')` is now `resize(width, height, { fit: 'cover', position: 'attention' })`,
|
||||||
|
`max().withoutEnlargement()` is now `resize(width, height, { fit: 'inside', withoutEnlargement: true })`.
|
||||||
|
[#1135](https://github.com/lovell/sharp/issues/1135)
|
||||||
|
|
||||||
|
* Deprecate the `background` function.
|
||||||
|
Per-operation `background` options added to `resize`, `extend` and `flatten` operations.
|
||||||
|
[#1392](https://github.com/lovell/sharp/issues/1392)
|
||||||
|
|
||||||
|
* Add `size` to `metadata` response (Stream and Buffer input only).
|
||||||
|
[#695](https://github.com/lovell/sharp/issues/695)
|
||||||
|
|
||||||
|
* Switch from custom trim operation to `vips_find_trim`.
|
||||||
|
[#914](https://github.com/lovell/sharp/issues/914)
|
||||||
|
|
||||||
|
* Add `chromaSubsampling` and `isProgressive` properties to `metadata` response.
|
||||||
|
[#1186](https://github.com/lovell/sharp/issues/1186)
|
||||||
|
|
||||||
|
* Drop Node 4 support.
|
||||||
|
[#1212](https://github.com/lovell/sharp/issues/1212)
|
||||||
|
|
||||||
|
* Enable SIMD convolution by default.
|
||||||
|
[#1213](https://github.com/lovell/sharp/issues/1213)
|
||||||
|
|
||||||
|
* Add experimental prebuilt binaries for musl-based Linux.
|
||||||
|
[#1379](https://github.com/lovell/sharp/issues/1379)
|
||||||
|
|
||||||
|
* Add support for arbitrary rotation angle via vips_rotate.
|
||||||
|
[#1385](https://github.com/lovell/sharp/pull/1385)
|
||||||
|
[@freezy](https://github.com/freezy)
|
||||||
|
|
||||||
### v0.20 - "*prebuild*"
|
### v0.20 - "*prebuild*"
|
||||||
|
|
||||||
Requires libvips v8.6.1.
|
Requires libvips v8.6.1.
|
||||||
|
|
||||||
|
#### v0.20.8 - 5<sup>th</sup> September 2018
|
||||||
|
|
||||||
|
* Avoid race conditions when creating directories during installation.
|
||||||
|
[#1358](https://github.com/lovell/sharp/pull/1358)
|
||||||
|
[@ajhool](https://github.com/ajhool)
|
||||||
|
|
||||||
|
* Accept floating point values for input density parameter.
|
||||||
|
[#1362](https://github.com/lovell/sharp/pull/1362)
|
||||||
|
[@aeirola](https://github.com/aeirola)
|
||||||
|
|
||||||
#### v0.20.7 - 21<sup>st</sup> August 2018
|
#### v0.20.7 - 21<sup>st</sup> August 2018
|
||||||
|
|
||||||
* Use copy+unlink if rename operation fails during installation.
|
* Use copy+unlink if rename operation fails during installation.
|
||||||
|
|||||||
@@ -13,8 +13,8 @@ Lanczos resampling ensures quality is not sacrificed for speed.
|
|||||||
As well as image resizing, operations such as
|
As well as image resizing, operations such as
|
||||||
rotation, extraction, compositing and gamma correction are available.
|
rotation, extraction, compositing and gamma correction are available.
|
||||||
|
|
||||||
Most 64-bit OS X, Windows and Linux (glibc) systems running
|
Most modern 64-bit OS X, Windows and Linux systems running
|
||||||
Node versions 4, 6, 8 and 10
|
Node versions 6, 8 and 10
|
||||||
do not require any additional install or runtime dependencies.
|
do not require any additional install or runtime dependencies.
|
||||||
|
|
||||||
[](https://coveralls.io/r/lovell/sharp?branch=master)
|
[](https://coveralls.io/r/lovell/sharp?branch=master)
|
||||||
@@ -37,7 +37,7 @@ and [Leaflet](https://github.com/turban/Leaflet.Zoomify).
|
|||||||
### Fast
|
### Fast
|
||||||
|
|
||||||
This module is powered by the blazingly fast
|
This module is powered by the blazingly fast
|
||||||
[libvips](https://github.com/jcupitt/libvips) image processing library,
|
[libvips](https://github.com/libvips/libvips) image processing library,
|
||||||
originally created in 1989 at Birkbeck College
|
originally created in 1989 at Birkbeck College
|
||||||
and currently maintained by
|
and currently maintained by
|
||||||
[John Cupitt](https://github.com/jcupitt).
|
[John Cupitt](https://github.com/jcupitt).
|
||||||
@@ -116,10 +116,16 @@ the help and code contributions of the following people:
|
|||||||
* [Espen Hovlandsdal](https://github.com/rexxars)
|
* [Espen Hovlandsdal](https://github.com/rexxars)
|
||||||
* [Sylvain Dumont](https://github.com/sylvaindumont)
|
* [Sylvain Dumont](https://github.com/sylvaindumont)
|
||||||
* [Alun Davies](https://github.com/alundavies)
|
* [Alun Davies](https://github.com/alundavies)
|
||||||
|
* [Aidan Hoolachan](https://github.com/ajhool)
|
||||||
|
* [Axel Eirola](https://github.com/aeirola)
|
||||||
|
* [Freezy](https://github.com/freezy)
|
||||||
|
* [Julian Aubourg](https://github.com/jaubourg)
|
||||||
|
* [Keith Belovay](https://github.com/fromkeith)
|
||||||
|
* [Michael B. Klein](https://github.com/mbklein)
|
||||||
|
|
||||||
Thank you!
|
Thank you!
|
||||||
|
|
||||||
### Licence
|
### Licensing
|
||||||
|
|
||||||
Copyright 2013, 2014, 2015, 2016, 2017, 2018 Lovell Fuller and contributors.
|
Copyright 2013, 2014, 2015, 2016, 2017, 2018 Lovell Fuller and contributors.
|
||||||
|
|
||||||
|
|||||||
@@ -15,7 +15,7 @@ yarn add sharp
|
|||||||
### Building from source
|
### Building from source
|
||||||
|
|
||||||
Pre-compiled binaries for sharp are provided for use with
|
Pre-compiled binaries for sharp are provided for use with
|
||||||
Node versions 4, 6, 8 and 10 on
|
Node versions 6, 8 and 10 on
|
||||||
64-bit Windows, OS X and Linux platforms.
|
64-bit Windows, OS X and Linux platforms.
|
||||||
|
|
||||||
Sharp will be built from source at install time when:
|
Sharp will be built from source at install time when:
|
||||||
@@ -27,7 +27,7 @@ Sharp will be built from source at install time when:
|
|||||||
Building from source requires:
|
Building from source requires:
|
||||||
|
|
||||||
* C++11 compatible compiler such as gcc 4.8+, clang 3.0+ or MSVC 2013+
|
* C++11 compatible compiler such as gcc 4.8+, clang 3.0+ or MSVC 2013+
|
||||||
* [node-gyp](https://github.com/TooTallNate/node-gyp#installation) and its dependencies (includes Python)
|
* [node-gyp](https://github.com/nodejs/node-gyp#installation) and its dependencies (includes Python 2.7)
|
||||||
|
|
||||||
## libvips
|
## libvips
|
||||||
|
|
||||||
@@ -36,13 +36,14 @@ Building from source requires:
|
|||||||
[](https://travis-ci.org/lovell/sharp)
|
[](https://travis-ci.org/lovell/sharp)
|
||||||
|
|
||||||
libvips and its dependencies are fetched and stored within `node_modules/sharp/vendor` during `npm install`.
|
libvips and its dependencies are fetched and stored within `node_modules/sharp/vendor` during `npm install`.
|
||||||
This involves an automated HTTPS download of approximately 7MB.
|
This involves an automated HTTPS download of approximately 8MB.
|
||||||
|
|
||||||
Most recent Linux-based operating systems with glibc running on x64 and ARMv6+ CPUs should "just work", e.g.:
|
Most Linux-based (glibc, musl) operating systems running on x64 and ARMv6+ CPUs should "just work", e.g.:
|
||||||
|
|
||||||
* Debian 7+
|
* Debian 7+
|
||||||
* Ubuntu 14.04+
|
* Ubuntu 14.04+
|
||||||
* Centos 7+
|
* Centos 7+
|
||||||
|
* Alpine 3.8+ (Node 8 and 10)
|
||||||
* Fedora
|
* Fedora
|
||||||
* openSUSE 13.2+
|
* openSUSE 13.2+
|
||||||
* Archlinux
|
* Archlinux
|
||||||
@@ -61,9 +62,9 @@ and `LD_LIBRARY_PATH` at runtime.
|
|||||||
This allows the use of newer versions of libvips with older versions of sharp.
|
This allows the use of newer versions of libvips with older versions of sharp.
|
||||||
|
|
||||||
For 32-bit Intel CPUs and older Linux-based operating systems such as Centos 6,
|
For 32-bit Intel CPUs and older Linux-based operating systems such as Centos 6,
|
||||||
it is recommended to install a system-wide installation of libvips from source:
|
compiling libvips from source is recommended.
|
||||||
|
|
||||||
https://jcupitt.github.io/libvips/install.html#building-libvips-from-a-source-tarball
|
[https://libvips.github.io/libvips/install.html#building-libvips-from-a-source-tarball](https://libvips.github.io/libvips/install.html#building-libvips-from-a-source-tarball)
|
||||||
|
|
||||||
#### Alpine Linux
|
#### Alpine Linux
|
||||||
|
|
||||||
@@ -71,7 +72,9 @@ libvips is available in the
|
|||||||
[testing repository](https://pkgs.alpinelinux.org/packages?name=vips-dev):
|
[testing repository](https://pkgs.alpinelinux.org/packages?name=vips-dev):
|
||||||
|
|
||||||
```sh
|
```sh
|
||||||
apk add vips-dev fftw-dev --update-cache --repository https://dl-3.alpinelinux.org/alpine/edge/testing/
|
apk add vips-dev fftw-dev build-base --update-cache \
|
||||||
|
--repository https://alpine.global.ssl.fastly.net/alpine/edge/testing/ \
|
||||||
|
--repository https://alpine.global.ssl.fastly.net/alpine/edge/main
|
||||||
```
|
```
|
||||||
|
|
||||||
The smaller stack size of musl libc means
|
The smaller stack size of musl libc means
|
||||||
@@ -94,7 +97,8 @@ that it can be located using `pkg-config --modversion vips-cpp`.
|
|||||||
[](https://ci.appveyor.com/project/lovell/sharp)
|
[](https://ci.appveyor.com/project/lovell/sharp)
|
||||||
|
|
||||||
libvips and its dependencies are fetched and stored within `node_modules\sharp\vendor` during `npm install`.
|
libvips and its dependencies are fetched and stored within `node_modules\sharp\vendor` during `npm install`.
|
||||||
This involves an automated HTTPS download of approximately 12MB.
|
This involves an automated HTTPS download of approximately 13MB. If you are having issues during
|
||||||
|
installation consider removing the directory ```C:\Users\[user]\AppData\Roaming\npm-cache\_libvips```.
|
||||||
|
|
||||||
Only 64-bit (x64) `node.exe` is supported.
|
Only 64-bit (x64) `node.exe` is supported.
|
||||||
|
|
||||||
@@ -117,9 +121,6 @@ https://bugs.freebsd.org/bugzilla/show_bug.cgi?id=193528
|
|||||||
|
|
||||||
### Heroku
|
### Heroku
|
||||||
|
|
||||||
libvips and its dependencies are fetched and stored within `node_modules\sharp\vendor` during `npm install`.
|
|
||||||
This involves an automated HTTPS download of approximately 7MB.
|
|
||||||
|
|
||||||
Set [NODE_MODULES_CACHE](https://devcenter.heroku.com/articles/nodejs-support#cache-behavior)
|
Set [NODE_MODULES_CACHE](https://devcenter.heroku.com/articles/nodejs-support#cache-behavior)
|
||||||
to `false` when using the `yarn` package manager.
|
to `false` when using the `yarn` package manager.
|
||||||
|
|
||||||
@@ -148,18 +149,28 @@ docker pull tailor/docker-libvips
|
|||||||
|
|
||||||
### AWS Lambda
|
### AWS Lambda
|
||||||
|
|
||||||
A [deployment package](http://docs.aws.amazon.com/lambda/latest/dg/nodejs-create-deployment-pkg.html) for the
|
Set the Lambda runtime to Node.js 8.10.
|
||||||
[Lambda Execution Environment](http://docs.aws.amazon.com/lambda/latest/dg/current-supported-versions.html)
|
|
||||||
can be built using Docker.
|
The binaries in the `node_modules` directory of the
|
||||||
|
[deployment package](https://docs.aws.amazon.com/lambda/latest/dg/nodejs-create-deployment-pkg.html)
|
||||||
|
must be for the Linux x64 platform/architecture.
|
||||||
|
|
||||||
|
On non-Linux machines such as OS X and Windows run the following:
|
||||||
|
|
||||||
```sh
|
```sh
|
||||||
rm -rf node_modules/sharp
|
rm -rf node_modules/sharp
|
||||||
docker run -v "$PWD":/var/task lambci/lambda:build-nodejs6.10 npm install
|
npm install --arch=x64 --platform=linux --target=8.10.0 sharp
|
||||||
```
|
```
|
||||||
|
|
||||||
Set the Lambda runtime to Node.js 6.10.
|
Alternatively a Docker container closely matching the Lambda runtime can be used:
|
||||||
|
|
||||||
To get the best performance select the largest memory available. A 1536 MB function provides ~12x more CPU time than a 128 MB function.
|
```sh
|
||||||
|
rm -rf node_modules/sharp
|
||||||
|
docker run -v "$PWD":/var/task lambci/lambda:build-nodejs8.10 npm install sharp
|
||||||
|
```
|
||||||
|
|
||||||
|
To get the best performance select the largest memory available.
|
||||||
|
A 1536 MB function provides ~12x more CPU time than a 128 MB function.
|
||||||
|
|
||||||
### NW.js
|
### NW.js
|
||||||
|
|
||||||
@@ -171,7 +182,7 @@ nw-gyp rebuild --arch=x64 --target=[your nw version]
|
|||||||
node node_modules/sharp/install/dll-copy
|
node node_modules/sharp/install/dll-copy
|
||||||
```
|
```
|
||||||
|
|
||||||
See also http://docs.nwjs.io/en/latest/For%20Users/Advanced/Use%20Native%20Node%20Modules/
|
[http://docs.nwjs.io/en/latest/For%20Users/Advanced/Use%20Native%20Node%20Modules/](http://docs.nwjs.io/en/latest/For%20Users/Advanced/Use%20Native%20Node%20Modules/)
|
||||||
|
|
||||||
### Build tools
|
### Build tools
|
||||||
|
|
||||||
@@ -199,28 +210,6 @@ and [Valgrind](http://valgrind.org/) have been used to test
|
|||||||
the most popular web-based formats, as well as libvips itself,
|
the most popular web-based formats, as well as libvips itself,
|
||||||
you are advised to perform your own testing and sandboxing.
|
you are advised to perform your own testing and sandboxing.
|
||||||
|
|
||||||
ImageMagick in particular has a relatively large attack surface,
|
|
||||||
which can be partially mitigated with a
|
|
||||||
[policy.xml](http://www.imagemagick.org/script/resources.php)
|
|
||||||
configuration file to prevent the use of coders known to be vulnerable.
|
|
||||||
|
|
||||||
```xml
|
|
||||||
<policymap>
|
|
||||||
<policy domain="coder" rights="none" pattern="EPHEMERAL" />
|
|
||||||
<policy domain="coder" rights="none" pattern="URL" />
|
|
||||||
<policy domain="coder" rights="none" pattern="HTTPS" />
|
|
||||||
<policy domain="coder" rights="none" pattern="MVG" />
|
|
||||||
<policy domain="coder" rights="none" pattern="MSL" />
|
|
||||||
<policy domain="coder" rights="none" pattern="TEXT" />
|
|
||||||
<policy domain="coder" rights="none" pattern="SHOW" />
|
|
||||||
<policy domain="coder" rights="none" pattern="WIN" />
|
|
||||||
<policy domain="coder" rights="none" pattern="PLT" />
|
|
||||||
</policymap>
|
|
||||||
```
|
|
||||||
|
|
||||||
Set the `MAGICK_CONFIGURE_PATH` environment variable
|
|
||||||
to the directory containing the `policy.xml` file.
|
|
||||||
|
|
||||||
### Pre-compiled libvips binaries
|
### Pre-compiled libvips binaries
|
||||||
|
|
||||||
This module will attempt to download a pre-compiled bundle of libvips
|
This module will attempt to download a pre-compiled bundle of libvips
|
||||||
@@ -236,10 +225,18 @@ SHARP_IGNORE_GLOBAL_LIBVIPS=1 npm install sharp
|
|||||||
```
|
```
|
||||||
|
|
||||||
Should you need to manually download and inspect these files,
|
Should you need to manually download and inspect these files,
|
||||||
you can do so via https://github.com/lovell/sharp-libvips/releases
|
you can do so via
|
||||||
|
[https://github.com/lovell/sharp-libvips/releases](https://github.com/lovell/sharp-libvips/releases)
|
||||||
|
|
||||||
Should you wish to install these from your own location,
|
Should you wish to install these from your own location,
|
||||||
set the `SHARP_DIST_BASE_URL` environment variable, e.g.
|
set the `sharp_dist_base_url` npm config option, e.g.
|
||||||
|
|
||||||
|
```sh
|
||||||
|
npm config set sharp_dist_base_url "https://hostname/path/"
|
||||||
|
npm install sharp
|
||||||
|
```
|
||||||
|
|
||||||
|
or set the `SHARP_DIST_BASE_URL` environment variable, e.g.
|
||||||
|
|
||||||
```sh
|
```sh
|
||||||
SHARP_DIST_BASE_URL="https://hostname/path/" npm install sharp
|
SHARP_DIST_BASE_URL="https://hostname/path/" npm install sharp
|
||||||
@@ -265,6 +262,8 @@ Use of libraries under the terms of the LGPLv3 is via the
|
|||||||
| expat | MIT Licence |
|
| expat | MIT Licence |
|
||||||
| fontconfig | [fontconfig Licence](https://cgit.freedesktop.org/fontconfig/tree/COPYING) (BSD-like) |
|
| fontconfig | [fontconfig Licence](https://cgit.freedesktop.org/fontconfig/tree/COPYING) (BSD-like) |
|
||||||
| freetype | [freetype Licence](http://git.savannah.gnu.org/cgit/freetype/freetype2.git/tree/docs/FTL.TXT) (BSD-like) |
|
| freetype | [freetype Licence](http://git.savannah.gnu.org/cgit/freetype/freetype2.git/tree/docs/FTL.TXT) (BSD-like) |
|
||||||
|
| fribidi | LGPLv3 |
|
||||||
|
| gettext | LGPLv3 |
|
||||||
| giflib | MIT Licence |
|
| giflib | MIT Licence |
|
||||||
| glib | LGPLv3 |
|
| glib | LGPLv3 |
|
||||||
| harfbuzz | MIT Licence |
|
| harfbuzz | MIT Licence |
|
||||||
|
|||||||
@@ -3,19 +3,17 @@
|
|||||||
### Test environment
|
### Test environment
|
||||||
|
|
||||||
* AWS EC2 eu-west-1 [c5.large](https://aws.amazon.com/ec2/instance-types/c5/) (2x Xeon Platinum 8124M CPU @ 3.00GHz)
|
* AWS EC2 eu-west-1 [c5.large](https://aws.amazon.com/ec2/instance-types/c5/) (2x Xeon Platinum 8124M CPU @ 3.00GHz)
|
||||||
* Ubuntu 17.10 (hvm:ebs-ssd, 20180102, ami-0741d47e)
|
* Ubuntu 18.04 (hvm-ssd/ubuntu-bionic-18.04-amd64-server-20180912 ami-00035f41c82244dab)
|
||||||
* Node.js v8.9.4
|
* Node.js v10.11.0
|
||||||
|
|
||||||
### The contenders
|
### The contenders
|
||||||
|
|
||||||
* [jimp](https://www.npmjs.com/package/jimp) v0.2.28 - Image processing in pure JavaScript. Bilinear interpolation only.
|
* [jimp](https://www.npmjs.com/package/jimp) v0.5.3 - Image processing in pure JavaScript. Provides bicubic interpolation.
|
||||||
* [pajk-lwip](https://www.npmjs.com/package/pajk-lwip) v0.2.0 (fork) - Wrapper around CImg that compiles dependencies from source.
|
* [mapnik](https://www.npmjs.org/package/mapnik) v4.0.1 - Whilst primarily a map renderer, Mapnik contains bitmap image utilities.
|
||||||
* [mapnik](https://www.npmjs.org/package/mapnik) v3.6.2 - Whilst primarily a map renderer, Mapnik contains bitmap image utilities.
|
|
||||||
* [imagemagick-native](https://www.npmjs.com/package/imagemagick-native) v1.9.3 - Wrapper around libmagick++, supports Buffers only.
|
* [imagemagick-native](https://www.npmjs.com/package/imagemagick-native) v1.9.3 - Wrapper around libmagick++, supports Buffers only.
|
||||||
* [imagemagick](https://www.npmjs.com/package/imagemagick) v0.1.3 - Supports filesystem only and "*has been unmaintained for a long time*".
|
* [imagemagick](https://www.npmjs.com/package/imagemagick) v0.1.3 - Supports filesystem only and "*has been unmaintained for a long time*".
|
||||||
* [gm](https://www.npmjs.com/package/gm) v1.23.1 - Fully featured wrapper around GraphicsMagick's `gm` command line utility.
|
* [gm](https://www.npmjs.com/package/gm) v1.23.1 - Fully featured wrapper around GraphicsMagick's `gm` command line utility.
|
||||||
* [images](https://www.npmjs.com/package/images) v3.0.1 - Compiles dependencies from source. Provides bicubic interpolation.
|
* sharp v0.21.0 / libvips v8.7.0 - Caching within libvips disabled to ensure a fair comparison.
|
||||||
* sharp v0.19.0 / libvips v8.6.1 - Caching within libvips disabled to ensure a fair comparison.
|
|
||||||
|
|
||||||
### The task
|
### The task
|
||||||
|
|
||||||
@@ -27,19 +25,14 @@ then compress to JPEG at a "quality" setting of 80.
|
|||||||
|
|
||||||
| Module | Input | Output | Ops/sec | Speed-up |
|
| Module | Input | Output | Ops/sec | Speed-up |
|
||||||
| :----------------- | :----- | :----- | ------: | -------: |
|
| :----------------- | :----- | :----- | ------: | -------: |
|
||||||
| jimp (bilinear) | buffer | buffer | 1.14 | 1.0 |
|
| jimp | buffer | buffer | 0.71 | 1.0 |
|
||||||
| lwip | buffer | buffer | 1.86 | 1.6 |
|
| mapnik | buffer | buffer | 3.32 | 4.7 |
|
||||||
| mapnik | buffer | buffer | 3.34 | 2.9 |
|
| gm | buffer | buffer | 3.97 | 5.6 |
|
||||||
| imagemagick-native | buffer | buffer | 4.13 | 3.6 |
|
| imagemagick-native | buffer | buffer | 4.06 | 5.7 |
|
||||||
| gm | buffer | buffer | 4.21 | 3.7 |
|
| imagemagick | file | file | 4.24 | 6.0 |
|
||||||
| gm | file | file | 4.27 | 3.7 |
|
| sharp | stream | stream | 25.30 | 35.6 |
|
||||||
| imagemagick | file | file | 4.67 | 4.1 |
|
| sharp | file | file | 26.17 | 36.9 |
|
||||||
| images (bicubic) | file | file | 6.22 | 5.5 |
|
| sharp | buffer | buffer | 26.45 | 37.3 |
|
||||||
| sharp | stream | stream | 24.43 | 21.4 |
|
|
||||||
| sharp | file | file | 25.97 | 22.7 |
|
|
||||||
| sharp | file | buffer | 26.00 | 22.8 |
|
|
||||||
| sharp | buffer | file | 26.33 | 23.0 |
|
|
||||||
| sharp | buffer | buffer | 26.43 | 23.1 |
|
|
||||||
|
|
||||||
Greater libvips performance can be expected with caching enabled (default)
|
Greater libvips performance can be expected with caching enabled (default)
|
||||||
and using 8+ core machines, especially those with larger L1/L2 CPU caches.
|
and using 8+ core machines, especially those with larger L1/L2 CPU caches.
|
||||||
@@ -57,7 +50,7 @@ brew install mapnik
|
|||||||
```
|
```
|
||||||
|
|
||||||
```sh
|
```sh
|
||||||
sudo apt-get install imagemagick libmagick++-dev graphicsmagick mapnik-dev
|
sudo apt-get install imagemagick libmagick++-dev graphicsmagick libmapnik-dev
|
||||||
```
|
```
|
||||||
|
|
||||||
```sh
|
```sh
|
||||||
|
|||||||
@@ -4,6 +4,7 @@ const fs = require('fs');
|
|||||||
const path = require('path');
|
const path = require('path');
|
||||||
|
|
||||||
const copyFileSync = require('fs-copy-file-sync');
|
const copyFileSync = require('fs-copy-file-sync');
|
||||||
|
const libvips = require('../lib/libvips');
|
||||||
const npmLog = require('npmlog');
|
const npmLog = require('npmlog');
|
||||||
|
|
||||||
if (process.platform === 'win32') {
|
if (process.platform === 'win32') {
|
||||||
@@ -11,8 +12,8 @@ if (process.platform === 'win32') {
|
|||||||
const buildReleaseDir = path.join(buildDir, 'Release');
|
const buildReleaseDir = path.join(buildDir, 'Release');
|
||||||
npmLog.info('sharp', `Creating ${buildReleaseDir}`);
|
npmLog.info('sharp', `Creating ${buildReleaseDir}`);
|
||||||
try {
|
try {
|
||||||
fs.mkdirSync(buildDir);
|
libvips.mkdirSync(buildDir);
|
||||||
fs.mkdirSync(buildReleaseDir);
|
libvips.mkdirSync(buildReleaseDir);
|
||||||
} catch (err) {}
|
} catch (err) {}
|
||||||
const vendorLibDir = path.join(__dirname, '..', 'vendor', 'lib');
|
const vendorLibDir = path.join(__dirname, '..', 'vendor', 'lib');
|
||||||
npmLog.info('sharp', `Copying DLLs from ${vendorLibDir} to ${buildReleaseDir}`);
|
npmLog.info('sharp', `Copying DLLs from ${vendorLibDir} to ${buildReleaseDir}`);
|
||||||
|
|||||||
@@ -16,22 +16,24 @@ const libvips = require('../lib/libvips');
|
|||||||
const platform = require('../lib/platform');
|
const platform = require('../lib/platform');
|
||||||
|
|
||||||
const minimumLibvipsVersion = libvips.minimumLibvipsVersion;
|
const minimumLibvipsVersion = libvips.minimumLibvipsVersion;
|
||||||
const distBaseUrl = process.env.SHARP_DIST_BASE_URL || `https://github.com/lovell/sharp-libvips/releases/download/v${minimumLibvipsVersion}/`;
|
const distBaseUrl = process.env.npm_config_sharp_dist_base_url || process.env.SHARP_DIST_BASE_URL || `https://github.com/lovell/sharp-libvips/releases/download/v${minimumLibvipsVersion}/`;
|
||||||
|
|
||||||
|
const fail = function (err) {
|
||||||
|
npmLog.error('sharp', err.message);
|
||||||
|
npmLog.error('sharp', 'Please see http://sharp.pixelplumbing.com/page/install');
|
||||||
|
process.exit(1);
|
||||||
|
};
|
||||||
|
|
||||||
const extractTarball = function (tarPath) {
|
const extractTarball = function (tarPath) {
|
||||||
const vendorPath = path.join(__dirname, '..', 'vendor');
|
const vendorPath = path.join(__dirname, '..', 'vendor');
|
||||||
if (!fs.existsSync(vendorPath)) {
|
libvips.mkdirSync(vendorPath);
|
||||||
fs.mkdirSync(vendorPath);
|
|
||||||
}
|
|
||||||
tar
|
tar
|
||||||
.extract({
|
.extract({
|
||||||
file: tarPath,
|
file: tarPath,
|
||||||
cwd: vendorPath,
|
cwd: vendorPath,
|
||||||
strict: true
|
strict: true
|
||||||
})
|
})
|
||||||
.catch(function (err) {
|
.catch(fail);
|
||||||
throw err;
|
|
||||||
});
|
|
||||||
};
|
};
|
||||||
|
|
||||||
try {
|
try {
|
||||||
@@ -53,11 +55,8 @@ try {
|
|||||||
if (arch === 'ia32') {
|
if (arch === 'ia32') {
|
||||||
throw new Error(`Intel Architecture 32-bit systems require manual installation of libvips >= ${minimumLibvipsVersion}`);
|
throw new Error(`Intel Architecture 32-bit systems require manual installation of libvips >= ${minimumLibvipsVersion}`);
|
||||||
}
|
}
|
||||||
if (platformAndArch === 'freebsd-x64') {
|
if (platformAndArch === 'freebsd-x64' || platformAndArch === 'openbsd-x64' || platformAndArch === 'sunos-x64') {
|
||||||
throw new Error(`FreeBSD systems require manual installation of libvips >= ${minimumLibvipsVersion}`);
|
throw new Error(`BSD/SunOS systems require manual installation of libvips >= ${minimumLibvipsVersion}`);
|
||||||
}
|
|
||||||
if (detectLibc.isNonGlibcLinux) {
|
|
||||||
throw new Error(`Use with ${detectLibc.family} libc requires manual installation of libvips >= ${minimumLibvipsVersion}`);
|
|
||||||
}
|
}
|
||||||
if (detectLibc.family === detectLibc.GLIBC && detectLibc.version && semver.lt(`${detectLibc.version}.0`, '2.13.0')) {
|
if (detectLibc.family === detectLibc.GLIBC && detectLibc.version && semver.lt(`${detectLibc.version}.0`, '2.13.0')) {
|
||||||
throw new Error(`Use with glibc version ${detectLibc.version} requires manual installation of libvips >= ${minimumLibvipsVersion}`);
|
throw new Error(`Use with glibc version ${detectLibc.version} requires manual installation of libvips >= ${minimumLibvipsVersion}`);
|
||||||
@@ -82,21 +81,21 @@ try {
|
|||||||
}
|
}
|
||||||
response.pipe(tmpFile);
|
response.pipe(tmpFile);
|
||||||
});
|
});
|
||||||
tmpFile.on('close', function () {
|
tmpFile
|
||||||
try {
|
.on('error', fail)
|
||||||
// Attempt to rename
|
.on('close', function () {
|
||||||
fs.renameSync(tarPathTemp, tarPathCache);
|
try {
|
||||||
} catch (err) {
|
// Attempt to rename
|
||||||
// Fall back to copy and unlink
|
fs.renameSync(tarPathTemp, tarPathCache);
|
||||||
copyFileSync(tarPathTemp, tarPathCache);
|
} catch (err) {
|
||||||
fs.unlinkSync(tarPathTemp);
|
// Fall back to copy and unlink
|
||||||
}
|
copyFileSync(tarPathTemp, tarPathCache);
|
||||||
extractTarball(tarPathCache);
|
fs.unlinkSync(tarPathTemp);
|
||||||
});
|
}
|
||||||
|
extractTarball(tarPathCache);
|
||||||
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
npmLog.error('sharp', err.message);
|
fail(err);
|
||||||
npmLog.error('sharp', 'Please see http://sharp.pixelplumbing.com/page/install');
|
|
||||||
process.exit(1);
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -117,14 +117,12 @@ function bandbool (boolOp) {
|
|||||||
* @private
|
* @private
|
||||||
*/
|
*/
|
||||||
module.exports = function (Sharp) {
|
module.exports = function (Sharp) {
|
||||||
// Public instance functions
|
Object.assign(Sharp.prototype, {
|
||||||
[
|
// Public instance functions
|
||||||
removeAlpha,
|
removeAlpha,
|
||||||
extractChannel,
|
extractChannel,
|
||||||
joinChannel,
|
joinChannel,
|
||||||
bandbool
|
bandbool
|
||||||
].forEach(function (f) {
|
|
||||||
Sharp.prototype[f.name] = f;
|
|
||||||
});
|
});
|
||||||
// Class attributes
|
// Class attributes
|
||||||
Sharp.bool = bool;
|
Sharp.bool = bool;
|
||||||
|
|||||||
@@ -1,5 +1,7 @@
|
|||||||
'use strict';
|
'use strict';
|
||||||
|
|
||||||
|
const deprecate = require('util').deprecate;
|
||||||
|
|
||||||
const color = require('color');
|
const color = require('color');
|
||||||
const is = require('./is');
|
const is = require('./is');
|
||||||
|
|
||||||
@@ -16,25 +18,20 @@ const colourspace = {
|
|||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Set the background for the `embed`, `flatten` and `extend` operations.
|
* @deprecated
|
||||||
* The default background is `{r: 0, g: 0, b: 0, alpha: 1}`, black without transparency.
|
* @private
|
||||||
*
|
|
||||||
* Delegates to the _color_ module, which can throw an Error
|
|
||||||
* but is liberal in what it accepts, clipping values to sensible min/max.
|
|
||||||
* The alpha value is a float between `0` (transparent) and `1` (opaque).
|
|
||||||
*
|
|
||||||
* @param {String|Object} rgba - parsed by the [color](https://www.npmjs.org/package/color) module to extract values for red, green, blue and alpha.
|
|
||||||
* @returns {Sharp}
|
|
||||||
* @throws {Error} Invalid parameter
|
|
||||||
*/
|
*/
|
||||||
function background (rgba) {
|
function background (rgba) {
|
||||||
const colour = color(rgba);
|
const colour = color(rgba);
|
||||||
this.options.background = [
|
const background = [
|
||||||
colour.red(),
|
colour.red(),
|
||||||
colour.green(),
|
colour.green(),
|
||||||
colour.blue(),
|
colour.blue(),
|
||||||
Math.round(colour.alpha() * 255)
|
Math.round(colour.alpha() * 255)
|
||||||
];
|
];
|
||||||
|
this.options.resizeBackground = background;
|
||||||
|
this.options.extendBackground = background;
|
||||||
|
this.options.flattenBackground = background.slice(0, 3);
|
||||||
return this;
|
return this;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -80,7 +77,7 @@ function grayscale (grayscale) {
|
|||||||
/**
|
/**
|
||||||
* Set the output colourspace.
|
* Set the output colourspace.
|
||||||
* By default output image will be web-friendly sRGB, with additional channels interpreted as alpha channels.
|
* By default output image will be web-friendly sRGB, with additional channels interpreted as alpha channels.
|
||||||
* @param {String} [colourspace] - output colourspace e.g. `srgb`, `rgb`, `cmyk`, `lab`, `b-w` [...](https://github.com/jcupitt/libvips/blob/master/libvips/iofuncs/enumtypes.c#L568)
|
* @param {String} [colourspace] - output colourspace e.g. `srgb`, `rgb`, `cmyk`, `lab`, `b-w` [...](https://github.com/libvips/libvips/blob/master/libvips/iofuncs/enumtypes.c#L568)
|
||||||
* @returns {Sharp}
|
* @returns {Sharp}
|
||||||
* @throws {Error} Invalid parameters
|
* @throws {Error} Invalid parameters
|
||||||
*/
|
*/
|
||||||
@@ -102,23 +99,43 @@ function toColorspace (colorspace) {
|
|||||||
return this.toColourspace(colorspace);
|
return this.toColourspace(colorspace);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Update a colour attribute of the this.options Object.
|
||||||
|
* @private
|
||||||
|
* @param {String} key
|
||||||
|
* @param {String|Object} val
|
||||||
|
* @throws {Error} Invalid key
|
||||||
|
*/
|
||||||
|
function _setColourOption (key, val) {
|
||||||
|
if (is.object(val) || is.string(val)) {
|
||||||
|
const colour = color(val);
|
||||||
|
this.options[key] = [
|
||||||
|
colour.red(),
|
||||||
|
colour.green(),
|
||||||
|
colour.blue(),
|
||||||
|
Math.round(colour.alpha() * 255)
|
||||||
|
];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Decorate the Sharp prototype with colour-related functions.
|
* Decorate the Sharp prototype with colour-related functions.
|
||||||
* @private
|
* @private
|
||||||
*/
|
*/
|
||||||
module.exports = function (Sharp) {
|
module.exports = function (Sharp) {
|
||||||
// Public instance functions
|
Object.assign(Sharp.prototype, {
|
||||||
[
|
// Public
|
||||||
background,
|
|
||||||
tint,
|
tint,
|
||||||
greyscale,
|
greyscale,
|
||||||
grayscale,
|
grayscale,
|
||||||
toColourspace,
|
toColourspace,
|
||||||
toColorspace
|
toColorspace,
|
||||||
].forEach(function (f) {
|
// Private
|
||||||
Sharp.prototype[f.name] = f;
|
_setColourOption
|
||||||
});
|
});
|
||||||
// Class attributes
|
// Class attributes
|
||||||
Sharp.colourspace = colourspace;
|
Sharp.colourspace = colourspace;
|
||||||
Sharp.colorspace = colourspace;
|
Sharp.colorspace = colourspace;
|
||||||
|
// Deprecated
|
||||||
|
Sharp.prototype.background = deprecate(background, 'background(background) is deprecated, use resize({ background }), extend({ background }) or flatten({ background }) instead');
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -34,7 +34,7 @@ const is = require('./is');
|
|||||||
* @param {Number} [options.left] - the pixel offset from the left edge.
|
* @param {Number} [options.left] - the pixel offset from the left edge.
|
||||||
* @param {Boolean} [options.tile=false] - set to true to repeat the overlay image across the entire image with the given `gravity`.
|
* @param {Boolean} [options.tile=false] - set to true to repeat the overlay image across the entire image with the given `gravity`.
|
||||||
* @param {Boolean} [options.cutout=false] - set to true to apply only the alpha channel of the overlay image to the input image, giving the appearance of one image being cut out of another.
|
* @param {Boolean} [options.cutout=false] - set to true to apply only the alpha channel of the overlay image to the input image, giving the appearance of one image being cut out of another.
|
||||||
* @param {Number} [options.density=72] - integral number representing the DPI for vector overlay image.
|
* @param {Number} [options.density=72] - number representing the DPI for vector overlay image.
|
||||||
* @param {Object} [options.raw] - describes overlay when using raw pixel data.
|
* @param {Object} [options.raw] - describes overlay when using raw pixel data.
|
||||||
* @param {Number} [options.raw.width]
|
* @param {Number} [options.raw.width]
|
||||||
* @param {Number} [options.raw.height]
|
* @param {Number} [options.raw.height]
|
||||||
|
|||||||
@@ -4,42 +4,10 @@ const path = require('path');
|
|||||||
const util = require('util');
|
const util = require('util');
|
||||||
const stream = require('stream');
|
const stream = require('stream');
|
||||||
const events = require('events');
|
const events = require('events');
|
||||||
const semver = require('semver');
|
|
||||||
const is = require('./is');
|
const is = require('./is');
|
||||||
const platform = require('./platform');
|
|
||||||
const sharp = require('../build/Release/sharp.node');
|
|
||||||
|
|
||||||
// Vendor platform
|
require('./libvips').hasVendoredLibvips();
|
||||||
(function () {
|
const sharp = require('bindings')('sharp.node');
|
||||||
let vendorPlatformId;
|
|
||||||
try {
|
|
||||||
vendorPlatformId = require('../vendor/platform.json');
|
|
||||||
} catch (err) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
const currentPlatformId = platform();
|
|
||||||
/* istanbul ignore if */
|
|
||||||
if (currentPlatformId !== vendorPlatformId) {
|
|
||||||
throw new Error(`'${vendorPlatformId}' binaries cannot be used on the '${currentPlatformId}' platform. Please remove the 'node_modules/sharp/vendor' directory and run 'npm rebuild'.`);
|
|
||||||
}
|
|
||||||
})();
|
|
||||||
|
|
||||||
// Versioning
|
|
||||||
let versions = {
|
|
||||||
vips: sharp.libvipsVersion()
|
|
||||||
};
|
|
||||||
(function () {
|
|
||||||
// Does libvips meet minimum requirement?
|
|
||||||
const libvipsVersionMin = require('../package.json').config.libvips;
|
|
||||||
/* istanbul ignore if */
|
|
||||||
if (semver.lt(versions.vips, libvipsVersionMin)) {
|
|
||||||
throw new Error('Found libvips ' + versions.vips + ' but require at least ' + libvipsVersionMin);
|
|
||||||
}
|
|
||||||
// Include versions of dependencies, if present
|
|
||||||
try {
|
|
||||||
versions = require('../vendor/versions.json');
|
|
||||||
} catch (err) {}
|
|
||||||
})();
|
|
||||||
|
|
||||||
// Use NODE_DEBUG=sharp to enable libvips warnings
|
// Use NODE_DEBUG=sharp to enable libvips warnings
|
||||||
const debuglog = util.debuglog('sharp');
|
const debuglog = util.debuglog('sharp');
|
||||||
@@ -96,7 +64,7 @@ const debuglog = util.debuglog('sharp');
|
|||||||
* @param {Boolean} [options.failOnError=false] - by default apply a "best effort"
|
* @param {Boolean} [options.failOnError=false] - by default apply a "best effort"
|
||||||
* to decode images, even if the data is corrupt or invalid. Set this flag to true
|
* to decode images, even if the data is corrupt or invalid. Set this flag to true
|
||||||
* if you'd rather halt processing and raise an error when loading invalid images.
|
* if you'd rather halt processing and raise an error when loading invalid images.
|
||||||
* @param {Number} [options.density=72] - integral number representing the DPI for vector images.
|
* @param {Number} [options.density=72] - number representing the DPI for vector images.
|
||||||
* @param {Number} [options.page=0] - page number to extract for multi-page input (GIF, TIFF)
|
* @param {Number} [options.page=0] - page number to extract for multi-page input (GIF, TIFF)
|
||||||
* @param {Object} [options.raw] - describes raw pixel input image data. See `raw()` for pixel ordering.
|
* @param {Object} [options.raw] - describes raw pixel input image data. See `raw()` for pixel ordering.
|
||||||
* @param {Number} [options.raw.width]
|
* @param {Number} [options.raw.width]
|
||||||
@@ -136,10 +104,12 @@ const Sharp = function (input, options) {
|
|||||||
width: -1,
|
width: -1,
|
||||||
height: -1,
|
height: -1,
|
||||||
canvas: 'crop',
|
canvas: 'crop',
|
||||||
crop: 0,
|
position: 0,
|
||||||
embed: 0,
|
resizeBackground: [0, 0, 0, 255],
|
||||||
useExifOrientation: false,
|
useExifOrientation: false,
|
||||||
angle: 0,
|
angle: 0,
|
||||||
|
rotationAngle: 0,
|
||||||
|
rotationBackground: [0, 0, 0, 255],
|
||||||
rotateBeforePreExtract: false,
|
rotateBeforePreExtract: false,
|
||||||
flip: false,
|
flip: false,
|
||||||
flop: false,
|
flop: false,
|
||||||
@@ -147,14 +117,15 @@ const Sharp = function (input, options) {
|
|||||||
extendBottom: 0,
|
extendBottom: 0,
|
||||||
extendLeft: 0,
|
extendLeft: 0,
|
||||||
extendRight: 0,
|
extendRight: 0,
|
||||||
|
extendBackground: [0, 0, 0, 255],
|
||||||
withoutEnlargement: false,
|
withoutEnlargement: false,
|
||||||
kernel: 'lanczos3',
|
kernel: 'lanczos3',
|
||||||
fastShrinkOnLoad: true,
|
fastShrinkOnLoad: true,
|
||||||
// operations
|
// operations
|
||||||
background: [0, 0, 0, 255],
|
|
||||||
tintA: 128,
|
tintA: 128,
|
||||||
tintB: 128,
|
tintB: 128,
|
||||||
flatten: false,
|
flatten: false,
|
||||||
|
flattenBackground: [0, 0, 0],
|
||||||
negate: false,
|
negate: false,
|
||||||
medianSize: 0,
|
medianSize: 0,
|
||||||
blurSigma: 0,
|
blurSigma: 0,
|
||||||
@@ -163,8 +134,9 @@ const Sharp = function (input, options) {
|
|||||||
sharpenJagged: 2,
|
sharpenJagged: 2,
|
||||||
threshold: 0,
|
threshold: 0,
|
||||||
thresholdGrayscale: true,
|
thresholdGrayscale: true,
|
||||||
trimTolerance: 0,
|
trimThreshold: 0,
|
||||||
gamma: 0,
|
gamma: 0,
|
||||||
|
gammaOut: 0,
|
||||||
greyscale: false,
|
greyscale: false,
|
||||||
normalise: 0,
|
normalise: 0,
|
||||||
booleanBufferIn: null,
|
booleanBufferIn: null,
|
||||||
@@ -205,7 +177,11 @@ const Sharp = function (input, options) {
|
|||||||
tiffQuality: 80,
|
tiffQuality: 80,
|
||||||
tiffCompression: 'jpeg',
|
tiffCompression: 'jpeg',
|
||||||
tiffPredictor: 'horizontal',
|
tiffPredictor: 'horizontal',
|
||||||
|
tiffPyramid: false,
|
||||||
tiffSquash: false,
|
tiffSquash: false,
|
||||||
|
tiffTile: false,
|
||||||
|
tiffTileHeight: 256,
|
||||||
|
tiffTileWidth: 256,
|
||||||
tiffXres: 1.0,
|
tiffXres: 1.0,
|
||||||
tiffYres: 1.0,
|
tiffYres: 1.0,
|
||||||
tileSize: 256,
|
tileSize: 256,
|
||||||
@@ -251,7 +227,12 @@ Sharp.format = sharp.format();
|
|||||||
* @example
|
* @example
|
||||||
* console.log(sharp.versions);
|
* console.log(sharp.versions);
|
||||||
*/
|
*/
|
||||||
Sharp.versions = versions;
|
Sharp.versions = {
|
||||||
|
vips: sharp.libvipsVersion()
|
||||||
|
};
|
||||||
|
try {
|
||||||
|
Sharp.versions = require('../vendor/versions.json');
|
||||||
|
} catch (err) {}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Export constructor.
|
* Export constructor.
|
||||||
|
|||||||
13
lib/input.js
@@ -36,7 +36,7 @@ function _createInputDescriptor (input, inputOptions, containerOptions) {
|
|||||||
}
|
}
|
||||||
// Density
|
// Density
|
||||||
if (is.defined(inputOptions.density)) {
|
if (is.defined(inputOptions.density)) {
|
||||||
if (is.integer(inputOptions.density) && is.inRange(inputOptions.density, 1, 2400)) {
|
if (is.inRange(inputOptions.density, 1, 2400)) {
|
||||||
inputDescriptor.density = inputOptions.density;
|
inputDescriptor.density = inputOptions.density;
|
||||||
} else {
|
} else {
|
||||||
throw new Error('Invalid density (1 to 2400) ' + inputOptions.density);
|
throw new Error('Invalid density (1 to 2400) ' + inputOptions.density);
|
||||||
@@ -177,12 +177,15 @@ function clone () {
|
|||||||
* A Promises/A+ promise is returned when `callback` is not provided.
|
* A Promises/A+ promise is returned when `callback` is not provided.
|
||||||
*
|
*
|
||||||
* - `format`: Name of decoder used to decompress image data e.g. `jpeg`, `png`, `webp`, `gif`, `svg`
|
* - `format`: Name of decoder used to decompress image data e.g. `jpeg`, `png`, `webp`, `gif`, `svg`
|
||||||
|
* - `size`: Total size of image in bytes, for Stream and Buffer input only
|
||||||
* - `width`: Number of pixels wide (EXIF orientation is not taken into consideration)
|
* - `width`: Number of pixels wide (EXIF orientation is not taken into consideration)
|
||||||
* - `height`: Number of pixels high (EXIF orientation is not taken into consideration)
|
* - `height`: Number of pixels high (EXIF orientation is not taken into consideration)
|
||||||
* - `space`: Name of colour space interpretation e.g. `srgb`, `rgb`, `cmyk`, `lab`, `b-w` [...](https://github.com/jcupitt/libvips/blob/master/libvips/iofuncs/enumtypes.c#L636)
|
* - `space`: Name of colour space interpretation e.g. `srgb`, `rgb`, `cmyk`, `lab`, `b-w` [...](https://github.com/libvips/libvips/blob/master/libvips/iofuncs/enumtypes.c#L636)
|
||||||
* - `channels`: Number of bands e.g. `3` for sRGB, `4` for CMYK
|
* - `channels`: Number of bands e.g. `3` for sRGB, `4` for CMYK
|
||||||
* - `depth`: Name of pixel depth format e.g. `uchar`, `char`, `ushort`, `float` [...](https://github.com/jcupitt/libvips/blob/master/libvips/iofuncs/enumtypes.c#L672)
|
* - `depth`: Name of pixel depth format e.g. `uchar`, `char`, `ushort`, `float` [...](https://github.com/libvips/libvips/blob/master/libvips/iofuncs/enumtypes.c#L672)
|
||||||
* - `density`: Number of pixels per inch (DPI), if present
|
* - `density`: Number of pixels per inch (DPI), if present
|
||||||
|
* - `chromaSubsampling`: String containing JPEG chroma subsampling, `4:2:0` or `4:4:4` for RGB, `4:2:0:4` or `4:4:4:4` for CMYK
|
||||||
|
* - `isProgressive`: Boolean indicating whether the image is interlaced using a progressive scan
|
||||||
* - `hasProfile`: Boolean indicating the presence of an embedded ICC profile
|
* - `hasProfile`: Boolean indicating the presence of an embedded ICC profile
|
||||||
* - `hasAlpha`: Boolean indicating the presence of an alpha transparency channel
|
* - `hasAlpha`: Boolean indicating the presence of an alpha transparency channel
|
||||||
* - `orientation`: Number value of the EXIF Orientation header, if present
|
* - `orientation`: Number value of the EXIF Orientation header, if present
|
||||||
@@ -359,7 +362,7 @@ function sequentialRead (sequentialRead) {
|
|||||||
* @private
|
* @private
|
||||||
*/
|
*/
|
||||||
module.exports = function (Sharp) {
|
module.exports = function (Sharp) {
|
||||||
[
|
Object.assign(Sharp.prototype, {
|
||||||
// Private
|
// Private
|
||||||
_createInputDescriptor,
|
_createInputDescriptor,
|
||||||
_write,
|
_write,
|
||||||
@@ -371,7 +374,5 @@ module.exports = function (Sharp) {
|
|||||||
stats,
|
stats,
|
||||||
limitInputPixels,
|
limitInputPixels,
|
||||||
sequentialRead
|
sequentialRead
|
||||||
].forEach(function (f) {
|
|
||||||
Sharp.prototype[f.name] = f;
|
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -15,15 +15,21 @@ const spawnSyncOptions = {
|
|||||||
shell: true
|
shell: true
|
||||||
};
|
};
|
||||||
|
|
||||||
|
const mkdirSync = function (dirPath) {
|
||||||
|
try {
|
||||||
|
fs.mkdirSync(dirPath);
|
||||||
|
} catch (err) {
|
||||||
|
if (err.code !== 'EEXIST') {
|
||||||
|
throw err;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
const cachePath = function () {
|
const cachePath = function () {
|
||||||
const npmCachePath = env.npm_config_cache || (env.APPDATA ? path.join(env.APPDATA, 'npm-cache') : path.join(os.homedir(), '.npm'));
|
const npmCachePath = env.npm_config_cache || (env.APPDATA ? path.join(env.APPDATA, 'npm-cache') : path.join(os.homedir(), '.npm'));
|
||||||
if (!fs.existsSync(npmCachePath)) {
|
mkdirSync(npmCachePath);
|
||||||
fs.mkdirSync(npmCachePath);
|
|
||||||
}
|
|
||||||
const libvipsCachePath = path.join(npmCachePath, '_libvips');
|
const libvipsCachePath = path.join(npmCachePath, '_libvips');
|
||||||
if (!fs.existsSync(libvipsCachePath)) {
|
mkdirSync(libvipsCachePath);
|
||||||
fs.mkdirSync(libvipsCachePath);
|
|
||||||
}
|
|
||||||
return libvipsCachePath;
|
return libvipsCachePath;
|
||||||
};
|
};
|
||||||
|
|
||||||
@@ -38,10 +44,16 @@ const globalLibvipsVersion = function () {
|
|||||||
|
|
||||||
const hasVendoredLibvips = function () {
|
const hasVendoredLibvips = function () {
|
||||||
const currentPlatformId = platform();
|
const currentPlatformId = platform();
|
||||||
|
const vendorPath = path.join(__dirname, '..', 'vendor');
|
||||||
|
let vendorVersionId;
|
||||||
let vendorPlatformId;
|
let vendorPlatformId;
|
||||||
try {
|
try {
|
||||||
vendorPlatformId = require(path.join(__dirname, '..', 'vendor', 'platform.json'));
|
vendorVersionId = require(path.join(vendorPath, 'versions.json')).vips;
|
||||||
|
vendorPlatformId = require(path.join(vendorPath, 'platform.json'));
|
||||||
} catch (err) {}
|
} catch (err) {}
|
||||||
|
if (vendorVersionId && vendorVersionId !== minimumLibvipsVersion) {
|
||||||
|
throw new Error(`Found vendored libvips v${vendorVersionId} but require v${minimumLibvipsVersion}. Please remove the 'node_modules/sharp/vendor' directory and run 'npm install'.`);
|
||||||
|
}
|
||||||
if (vendorPlatformId) {
|
if (vendorPlatformId) {
|
||||||
if (currentPlatformId === vendorPlatformId) {
|
if (currentPlatformId === vendorPlatformId) {
|
||||||
return true;
|
return true;
|
||||||
@@ -78,5 +90,6 @@ module.exports = {
|
|||||||
globalLibvipsVersion: globalLibvipsVersion,
|
globalLibvipsVersion: globalLibvipsVersion,
|
||||||
hasVendoredLibvips: hasVendoredLibvips,
|
hasVendoredLibvips: hasVendoredLibvips,
|
||||||
pkgConfigPath: pkgConfigPath,
|
pkgConfigPath: pkgConfigPath,
|
||||||
useGlobalLibvips: useGlobalLibvips
|
useGlobalLibvips: useGlobalLibvips,
|
||||||
|
mkdirSync: mkdirSync
|
||||||
};
|
};
|
||||||
|
|||||||
204
lib/operation.js
@@ -1,14 +1,18 @@
|
|||||||
'use strict';
|
'use strict';
|
||||||
|
|
||||||
|
const color = require('color');
|
||||||
const is = require('./is');
|
const is = require('./is');
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Rotate the output image by either an explicit angle
|
* Rotate the output image by either an explicit angle
|
||||||
* or auto-orient based on the EXIF `Orientation` tag.
|
* or auto-orient based on the EXIF `Orientation` tag.
|
||||||
*
|
*
|
||||||
* If an angle is provided, it is converted to a valid 90/180/270deg rotation.
|
* If an angle is provided, it is converted to a valid positive degree rotation.
|
||||||
* For example, `-450` will produce a 270deg rotation.
|
* For example, `-450` will produce a 270deg rotation.
|
||||||
*
|
*
|
||||||
|
* When rotating by an angle other than a multiple of 90,
|
||||||
|
* the background colour can be provided with the `background` option.
|
||||||
|
*
|
||||||
* If no angle is provided, it is determined from the EXIF data.
|
* If no angle is provided, it is determined from the EXIF data.
|
||||||
* Mirroring is supported and may infer the use of a flip operation.
|
* Mirroring is supported and may infer the use of a flip operation.
|
||||||
*
|
*
|
||||||
@@ -28,64 +32,30 @@ const is = require('./is');
|
|||||||
* });
|
* });
|
||||||
* readableStream.pipe(pipeline);
|
* readableStream.pipe(pipeline);
|
||||||
*
|
*
|
||||||
* @param {Number} [angle=auto] angle of rotation, must be a multiple of 90.
|
* @param {Number} [angle=auto] angle of rotation.
|
||||||
|
* @param {Object} [options] - if present, is an Object with optional attributes.
|
||||||
|
* @param {String|Object} [options.background="#000000"] parsed by the [color](https://www.npmjs.org/package/color) module to extract values for red, green, blue and alpha.
|
||||||
* @returns {Sharp}
|
* @returns {Sharp}
|
||||||
* @throws {Error} Invalid parameters
|
* @throws {Error} Invalid parameters
|
||||||
*/
|
*/
|
||||||
function rotate (angle) {
|
function rotate (angle, options) {
|
||||||
if (!is.defined(angle)) {
|
if (!is.defined(angle)) {
|
||||||
this.options.useExifOrientation = true;
|
this.options.useExifOrientation = true;
|
||||||
} else if (is.integer(angle) && !(angle % 90)) {
|
} else if (is.integer(angle) && !(angle % 90)) {
|
||||||
this.options.angle = angle;
|
this.options.angle = angle;
|
||||||
} else {
|
} else if (is.number(angle)) {
|
||||||
throw new Error('Unsupported angle: angle must be a positive/negative multiple of 90 ' + angle);
|
this.options.rotationAngle = angle;
|
||||||
}
|
if (is.object(options) && options.background) {
|
||||||
return this;
|
const backgroundColour = color(options.background);
|
||||||
}
|
this.options.rotationBackground = [
|
||||||
|
backgroundColour.red(),
|
||||||
/**
|
backgroundColour.green(),
|
||||||
* Extract a region of the image.
|
backgroundColour.blue(),
|
||||||
*
|
Math.round(backgroundColour.alpha() * 255)
|
||||||
* - Use `extract` before `resize` for pre-resize extraction.
|
];
|
||||||
* - Use `extract` after `resize` for post-resize extraction.
|
|
||||||
* - Use `extract` before and after for both.
|
|
||||||
*
|
|
||||||
* @example
|
|
||||||
* sharp(input)
|
|
||||||
* .extract({ left: left, top: top, width: width, height: height })
|
|
||||||
* .toFile(output, function(err) {
|
|
||||||
* // Extract a region of the input image, saving in the same format.
|
|
||||||
* });
|
|
||||||
* @example
|
|
||||||
* sharp(input)
|
|
||||||
* .extract({ left: leftOffsetPre, top: topOffsetPre, width: widthPre, height: heightPre })
|
|
||||||
* .resize(width, height)
|
|
||||||
* .extract({ left: leftOffsetPost, top: topOffsetPost, width: widthPost, height: heightPost })
|
|
||||||
* .toFile(output, function(err) {
|
|
||||||
* // Extract a region, resize, then extract from the resized image
|
|
||||||
* });
|
|
||||||
*
|
|
||||||
* @param {Object} options
|
|
||||||
* @param {Number} options.left - zero-indexed offset from left edge
|
|
||||||
* @param {Number} options.top - zero-indexed offset from top edge
|
|
||||||
* @param {Number} options.width - dimension of extracted image
|
|
||||||
* @param {Number} options.height - dimension of extracted image
|
|
||||||
* @returns {Sharp}
|
|
||||||
* @throws {Error} Invalid parameters
|
|
||||||
*/
|
|
||||||
function extract (options) {
|
|
||||||
const suffix = this.options.width === -1 && this.options.height === -1 ? 'Pre' : 'Post';
|
|
||||||
['left', 'top', 'width', 'height'].forEach(function (name) {
|
|
||||||
const value = options[name];
|
|
||||||
if (is.integer(value) && value >= 0) {
|
|
||||||
this.options[name + (name === 'left' || name === 'top' ? 'Offset' : '') + suffix] = value;
|
|
||||||
} else {
|
|
||||||
throw new Error('Non-integer value for ' + name + ' of ' + value);
|
|
||||||
}
|
}
|
||||||
}, this);
|
} else {
|
||||||
// Ensure existing rotation occurs before pre-resize extraction
|
throw new Error('Unsupported angle: must be a number.');
|
||||||
if (suffix === 'Pre' && ((this.options.angle % 360) !== 0 || this.options.useExifOrientation === true)) {
|
|
||||||
this.options.rotateBeforePreExtract = true;
|
|
||||||
}
|
}
|
||||||
return this;
|
return this;
|
||||||
}
|
}
|
||||||
@@ -201,72 +171,15 @@ function blur (sigma) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Extends/pads the edges of the image with the colour provided to the `background` method.
|
* Merge alpha transparency channel, if any, with a background.
|
||||||
* This operation will always occur after resizing and extraction, if any.
|
* @param {Object} [options]
|
||||||
*
|
* @param {String|Object} [options.background={r: 0, g: 0, b: 0}] - background colour, parsed by the [color](https://www.npmjs.org/package/color) module, defaults to black.
|
||||||
* @example
|
|
||||||
* // Resize to 140 pixels wide, then add 10 transparent pixels
|
|
||||||
* // to the top, left and right edges and 20 to the bottom edge
|
|
||||||
* sharp(input)
|
|
||||||
* .resize(140)
|
|
||||||
* .background({r: 0, g: 0, b: 0, alpha: 0})
|
|
||||||
* .extend({top: 10, bottom: 20, left: 10, right: 10})
|
|
||||||
* ...
|
|
||||||
*
|
|
||||||
* @param {(Number|Object)} extend - single pixel count to add to all edges or an Object with per-edge counts
|
|
||||||
* @param {Number} [extend.top]
|
|
||||||
* @param {Number} [extend.left]
|
|
||||||
* @param {Number} [extend.bottom]
|
|
||||||
* @param {Number} [extend.right]
|
|
||||||
* @returns {Sharp}
|
|
||||||
* @throws {Error} Invalid parameters
|
|
||||||
*/
|
|
||||||
function extend (extend) {
|
|
||||||
if (is.integer(extend) && extend > 0) {
|
|
||||||
this.options.extendTop = extend;
|
|
||||||
this.options.extendBottom = extend;
|
|
||||||
this.options.extendLeft = extend;
|
|
||||||
this.options.extendRight = extend;
|
|
||||||
} else if (
|
|
||||||
is.object(extend) &&
|
|
||||||
is.integer(extend.top) && extend.top >= 0 &&
|
|
||||||
is.integer(extend.bottom) && extend.bottom >= 0 &&
|
|
||||||
is.integer(extend.left) && extend.left >= 0 &&
|
|
||||||
is.integer(extend.right) && extend.right >= 0
|
|
||||||
) {
|
|
||||||
this.options.extendTop = extend.top;
|
|
||||||
this.options.extendBottom = extend.bottom;
|
|
||||||
this.options.extendLeft = extend.left;
|
|
||||||
this.options.extendRight = extend.right;
|
|
||||||
} else {
|
|
||||||
throw new Error('Invalid edge extension ' + extend);
|
|
||||||
}
|
|
||||||
return this;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Merge alpha transparency channel, if any, with `background`.
|
|
||||||
* @param {Boolean} [flatten=true]
|
|
||||||
* @returns {Sharp}
|
* @returns {Sharp}
|
||||||
*/
|
*/
|
||||||
function flatten (flatten) {
|
function flatten (options) {
|
||||||
this.options.flatten = is.bool(flatten) ? flatten : true;
|
this.options.flatten = is.bool(options) ? options : true;
|
||||||
return this;
|
if (is.object(options)) {
|
||||||
}
|
this._setColourOption('flattenBackground', options.background);
|
||||||
|
|
||||||
/**
|
|
||||||
* Trim "boring" pixels from all edges that contain values within a percentage similarity of the top-left pixel.
|
|
||||||
* @param {Number} [tolerance=10] value between 1 and 99 representing the percentage similarity.
|
|
||||||
* @returns {Sharp}
|
|
||||||
* @throws {Error} Invalid parameters
|
|
||||||
*/
|
|
||||||
function trim (tolerance) {
|
|
||||||
if (!is.defined(tolerance)) {
|
|
||||||
this.options.trimTolerance = 10;
|
|
||||||
} else if (is.integer(tolerance) && is.inRange(tolerance, 1, 99)) {
|
|
||||||
this.options.trimTolerance = tolerance;
|
|
||||||
} else {
|
|
||||||
throw new Error('Invalid trim tolerance (1 to 99) ' + tolerance);
|
|
||||||
}
|
}
|
||||||
return this;
|
return this;
|
||||||
}
|
}
|
||||||
@@ -277,11 +190,15 @@ function trim (tolerance) {
|
|||||||
* This can improve the perceived brightness of a resized image in non-linear colour spaces.
|
* This can improve the perceived brightness of a resized image in non-linear colour spaces.
|
||||||
* JPEG and WebP input images will not take advantage of the shrink-on-load performance optimisation
|
* JPEG and WebP input images will not take advantage of the shrink-on-load performance optimisation
|
||||||
* when applying a gamma correction.
|
* when applying a gamma correction.
|
||||||
|
*
|
||||||
|
* Supply a second argument to use a different output gamma value, otherwise the first value is used in both cases.
|
||||||
|
*
|
||||||
* @param {Number} [gamma=2.2] value between 1.0 and 3.0.
|
* @param {Number} [gamma=2.2] value between 1.0 and 3.0.
|
||||||
|
* @param {Number} [gammaOut] value between 1.0 and 3.0. (optional, defaults to same as `gamma`)
|
||||||
* @returns {Sharp}
|
* @returns {Sharp}
|
||||||
* @throws {Error} Invalid parameters
|
* @throws {Error} Invalid parameters
|
||||||
*/
|
*/
|
||||||
function gamma (gamma) {
|
function gamma (gamma, gammaOut) {
|
||||||
if (!is.defined(gamma)) {
|
if (!is.defined(gamma)) {
|
||||||
// Default gamma correction of 2.2 (sRGB)
|
// Default gamma correction of 2.2 (sRGB)
|
||||||
this.options.gamma = 2.2;
|
this.options.gamma = 2.2;
|
||||||
@@ -290,6 +207,14 @@ function gamma (gamma) {
|
|||||||
} else {
|
} else {
|
||||||
throw new Error('Invalid gamma correction (1.0 to 3.0) ' + gamma);
|
throw new Error('Invalid gamma correction (1.0 to 3.0) ' + gamma);
|
||||||
}
|
}
|
||||||
|
if (!is.defined(gammaOut)) {
|
||||||
|
// Default gamma correction for output is same as input
|
||||||
|
this.options.gammaOut = this.options.gamma;
|
||||||
|
} else if (is.number(gammaOut) && is.inRange(gammaOut, 1, 3)) {
|
||||||
|
this.options.gammaOut = gammaOut;
|
||||||
|
} else {
|
||||||
|
throw new Error('Invalid output gamma correction (1.0 to 3.0) ' + gammaOut);
|
||||||
|
}
|
||||||
return this;
|
return this;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -453,22 +378,56 @@ function linear (a, b) {
|
|||||||
return this;
|
return this;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Recomb the image with the specified matrix.
|
||||||
|
*
|
||||||
|
* @example
|
||||||
|
* sharp(input)
|
||||||
|
* .recomb([
|
||||||
|
* [0.3588, 0.7044, 0.1368],
|
||||||
|
* [0.2990, 0.5870, 0.1140],
|
||||||
|
* [0.2392, 0.4696, 0.0912],
|
||||||
|
* ])
|
||||||
|
* .raw()
|
||||||
|
* .toBuffer(function(err, data, info) {
|
||||||
|
* // data contains the raw pixel data after applying the recomb
|
||||||
|
* // With this example input, a sepia filter has been applied
|
||||||
|
* });
|
||||||
|
*
|
||||||
|
* @param {Array<Array<Number>>} 3x3 Recombination matrix
|
||||||
|
* @returns {Sharp}
|
||||||
|
* @throws {Error} Invalid parameters
|
||||||
|
*/
|
||||||
|
function recomb (inputMatrix) {
|
||||||
|
if (!Array.isArray(inputMatrix) || inputMatrix.length !== 3 ||
|
||||||
|
inputMatrix[0].length !== 3 ||
|
||||||
|
inputMatrix[1].length !== 3 ||
|
||||||
|
inputMatrix[2].length !== 3
|
||||||
|
) {
|
||||||
|
// must pass in a kernel
|
||||||
|
throw new Error('Invalid Recomb Matrix');
|
||||||
|
}
|
||||||
|
this.options.recombMatrix = [
|
||||||
|
inputMatrix[0][0], inputMatrix[0][1], inputMatrix[0][2],
|
||||||
|
inputMatrix[1][0], inputMatrix[1][1], inputMatrix[1][2],
|
||||||
|
inputMatrix[2][0], inputMatrix[2][1], inputMatrix[2][2]
|
||||||
|
].map(Number);
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Decorate the Sharp prototype with operation-related functions.
|
* Decorate the Sharp prototype with operation-related functions.
|
||||||
* @private
|
* @private
|
||||||
*/
|
*/
|
||||||
module.exports = function (Sharp) {
|
module.exports = function (Sharp) {
|
||||||
[
|
Object.assign(Sharp.prototype, {
|
||||||
rotate,
|
rotate,
|
||||||
extract,
|
|
||||||
flip,
|
flip,
|
||||||
flop,
|
flop,
|
||||||
sharpen,
|
sharpen,
|
||||||
median,
|
median,
|
||||||
blur,
|
blur,
|
||||||
extend,
|
|
||||||
flatten,
|
flatten,
|
||||||
trim,
|
|
||||||
gamma,
|
gamma,
|
||||||
negate,
|
negate,
|
||||||
normalise,
|
normalise,
|
||||||
@@ -476,8 +435,7 @@ module.exports = function (Sharp) {
|
|||||||
convolve,
|
convolve,
|
||||||
threshold,
|
threshold,
|
||||||
boolean,
|
boolean,
|
||||||
linear
|
linear,
|
||||||
].forEach(function (f) {
|
recomb
|
||||||
Sharp.prototype[f.name] = f;
|
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|||||||
120
lib/output.js
@@ -304,6 +304,10 @@ function webp (options) {
|
|||||||
* @param {Boolean} [options.force=true] - force TIFF output, otherwise attempt to use input format
|
* @param {Boolean} [options.force=true] - force TIFF output, otherwise attempt to use input format
|
||||||
* @param {Boolean} [options.compression='jpeg'] - compression options: lzw, deflate, jpeg, ccittfax4
|
* @param {Boolean} [options.compression='jpeg'] - compression options: lzw, deflate, jpeg, ccittfax4
|
||||||
* @param {Boolean} [options.predictor='horizontal'] - compression predictor options: none, horizontal, float
|
* @param {Boolean} [options.predictor='horizontal'] - compression predictor options: none, horizontal, float
|
||||||
|
* @param {Boolean} [options.pyramid=false] - write an image pyramid
|
||||||
|
* @param {Boolean} [options.tile=false] - write a tiled tiff
|
||||||
|
* @param {Boolean} [options.tileWidth=256] - horizontal tile size
|
||||||
|
* @param {Boolean} [options.tileHeight=256] - vertical tile size
|
||||||
* @param {Number} [options.xres=1.0] - horizontal resolution in pixels/mm
|
* @param {Number} [options.xres=1.0] - horizontal resolution in pixels/mm
|
||||||
* @param {Number} [options.yres=1.0] - vertical resolution in pixels/mm
|
* @param {Number} [options.yres=1.0] - vertical resolution in pixels/mm
|
||||||
* @param {Boolean} [options.squash=false] - squash 8-bit images down to 1 bit
|
* @param {Boolean} [options.squash=false] - squash 8-bit images down to 1 bit
|
||||||
@@ -311,51 +315,83 @@ function webp (options) {
|
|||||||
* @throws {Error} Invalid options
|
* @throws {Error} Invalid options
|
||||||
*/
|
*/
|
||||||
function tiff (options) {
|
function tiff (options) {
|
||||||
if (is.object(options) && is.defined(options.quality)) {
|
if (is.object(options)) {
|
||||||
if (is.integer(options.quality) && is.inRange(options.quality, 1, 100)) {
|
if (is.defined(options.quality)) {
|
||||||
this.options.tiffQuality = options.quality;
|
if (is.integer(options.quality) && is.inRange(options.quality, 1, 100)) {
|
||||||
} else {
|
this.options.tiffQuality = options.quality;
|
||||||
throw new Error('Invalid quality (integer, 1-100) ' + options.quality);
|
} else {
|
||||||
|
throw new Error('Invalid quality (integer, 1-100) ' + options.quality);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
if (is.defined(options.squash)) {
|
||||||
if (is.object(options) && is.defined(options.squash)) {
|
if (is.bool(options.squash)) {
|
||||||
if (is.bool(options.squash)) {
|
this.options.tiffSquash = options.squash;
|
||||||
this.options.tiffSquash = options.squash;
|
} else {
|
||||||
} else {
|
throw new Error('Invalid Value for squash ' + options.squash + ' Only Boolean Values allowed for options.squash.');
|
||||||
throw new Error('Invalid Value for squash ' + options.squash + ' Only Boolean Values allowed for options.squash.');
|
}
|
||||||
}
|
}
|
||||||
}
|
// tiling
|
||||||
// resolution
|
if (is.defined(options.tile)) {
|
||||||
if (is.object(options) && is.defined(options.xres)) {
|
if (is.bool(options.tile)) {
|
||||||
if (is.number(options.xres)) {
|
this.options.tiffTile = options.tile;
|
||||||
this.options.tiffXres = options.xres;
|
} else {
|
||||||
} else {
|
throw new Error('Invalid Value for tile ' + options.tile + ' Only Boolean values allowed for options.tile');
|
||||||
throw new Error('Invalid Value for xres ' + options.xres + ' Only numeric values allowed for options.xres');
|
}
|
||||||
}
|
}
|
||||||
}
|
if (is.defined(options.tileWidth)) {
|
||||||
if (is.object(options) && is.defined(options.yres)) {
|
if (is.number(options.tileWidth) && options.tileWidth > 0) {
|
||||||
if (is.number(options.yres)) {
|
this.options.tiffTileWidth = options.tileWidth;
|
||||||
this.options.tiffYres = options.yres;
|
} else {
|
||||||
} else {
|
throw new Error('Invalid Value for tileWidth ' + options.tileWidth + ' Only positive numeric values allowed for options.tileWidth');
|
||||||
throw new Error('Invalid Value for yres ' + options.yres + ' Only numeric values allowed for options.yres');
|
}
|
||||||
}
|
}
|
||||||
}
|
if (is.defined(options.tileHeight)) {
|
||||||
// compression
|
if (is.number(options.tileHeight) && options.tileHeight > 0) {
|
||||||
if (is.defined(options) && is.defined(options.compression)) {
|
this.options.tiffTileHeight = options.tileHeight;
|
||||||
if (is.string(options.compression) && is.inArray(options.compression, ['lzw', 'deflate', 'jpeg', 'ccittfax4', 'none'])) {
|
} else {
|
||||||
this.options.tiffCompression = options.compression;
|
throw new Error('Invalid Value for tileHeight ' + options.tileHeight + ' Only positive numeric values allowed for options.tileHeight');
|
||||||
} else {
|
}
|
||||||
const message = `Invalid compression option "${options.compression}". Should be one of: lzw, deflate, jpeg, ccittfax4, none`;
|
|
||||||
throw new Error(message);
|
|
||||||
}
|
}
|
||||||
}
|
// pyramid
|
||||||
// predictor
|
if (is.defined(options.pyramid)) {
|
||||||
if (is.defined(options) && is.defined(options.predictor)) {
|
if (is.bool(options.pyramid)) {
|
||||||
if (is.string(options.predictor) && is.inArray(options.predictor, ['none', 'horizontal', 'float'])) {
|
this.options.tiffPyramid = options.pyramid;
|
||||||
this.options.tiffPredictor = options.predictor;
|
} else {
|
||||||
} else {
|
throw new Error('Invalid Value for pyramid ' + options.pyramid + ' Only Boolean values allowed for options.pyramid');
|
||||||
const message = `Invalid predictor option "${options.predictor}". Should be one of: none, horizontal, float`;
|
}
|
||||||
throw new Error(message);
|
}
|
||||||
|
// resolution
|
||||||
|
if (is.defined(options.xres)) {
|
||||||
|
if (is.number(options.xres)) {
|
||||||
|
this.options.tiffXres = options.xres;
|
||||||
|
} else {
|
||||||
|
throw new Error('Invalid Value for xres ' + options.xres + ' Only numeric values allowed for options.xres');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (is.defined(options.yres)) {
|
||||||
|
if (is.number(options.yres)) {
|
||||||
|
this.options.tiffYres = options.yres;
|
||||||
|
} else {
|
||||||
|
throw new Error('Invalid Value for yres ' + options.yres + ' Only numeric values allowed for options.yres');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// compression
|
||||||
|
if (is.defined(options.compression)) {
|
||||||
|
if (is.string(options.compression) && is.inArray(options.compression, ['lzw', 'deflate', 'jpeg', 'ccittfax4', 'none'])) {
|
||||||
|
this.options.tiffCompression = options.compression;
|
||||||
|
} else {
|
||||||
|
const message = `Invalid compression option "${options.compression}". Should be one of: lzw, deflate, jpeg, ccittfax4, none`;
|
||||||
|
throw new Error(message);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// predictor
|
||||||
|
if (is.defined(options.predictor)) {
|
||||||
|
if (is.string(options.predictor) && is.inArray(options.predictor, ['none', 'horizontal', 'float'])) {
|
||||||
|
this.options.tiffPredictor = options.predictor;
|
||||||
|
} else {
|
||||||
|
const message = `Invalid predictor option "${options.predictor}". Should be one of: none, horizontal, float`;
|
||||||
|
throw new Error(message);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return this._updateFormatOut('tiff', options);
|
return this._updateFormatOut('tiff', options);
|
||||||
@@ -641,7 +677,7 @@ function _pipeline (callback) {
|
|||||||
* @private
|
* @private
|
||||||
*/
|
*/
|
||||||
module.exports = function (Sharp) {
|
module.exports = function (Sharp) {
|
||||||
[
|
Object.assign(Sharp.prototype, {
|
||||||
// Public
|
// Public
|
||||||
toFile,
|
toFile,
|
||||||
toBuffer,
|
toBuffer,
|
||||||
@@ -658,7 +694,5 @@ module.exports = function (Sharp) {
|
|||||||
_setBooleanOption,
|
_setBooleanOption,
|
||||||
_read,
|
_read,
|
||||||
_pipeline
|
_pipeline
|
||||||
].forEach(function (f) {
|
|
||||||
Sharp.prototype[f.name] = f;
|
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -1,10 +1,13 @@
|
|||||||
'use strict';
|
'use strict';
|
||||||
|
|
||||||
|
const detectLibc = require('detect-libc');
|
||||||
|
|
||||||
module.exports = function () {
|
module.exports = function () {
|
||||||
const arch = process.env.npm_config_arch || process.arch;
|
const arch = process.env.npm_config_arch || process.arch;
|
||||||
const platform = process.env.npm_config_platform || process.platform;
|
const platform = process.env.npm_config_platform || process.platform;
|
||||||
|
const libc = (platform === 'linux' && detectLibc.isNonGlibcLinux) ? detectLibc.family : '';
|
||||||
|
|
||||||
const platformId = [platform];
|
const platformId = [`${platform}${libc}`];
|
||||||
if (arch === 'arm' || arch === 'armhf' || arch === 'arm64') {
|
if (arch === 'arm' || arch === 'armhf' || arch === 'arm64') {
|
||||||
const armVersion = (arch === 'arm64') ? '8' : process.env.npm_config_armv || process.config.variables.arm_version || '6';
|
const armVersion = (arch === 'arm64') ? '8' : process.env.npm_config_armv || process.config.variables.arm_version || '6';
|
||||||
platformId.push(`armv${armVersion}`);
|
platformId.push(`armv${armVersion}`);
|
||||||
|
|||||||
423
lib/resize.js
@@ -1,9 +1,10 @@
|
|||||||
'use strict';
|
'use strict';
|
||||||
|
|
||||||
|
const deprecate = require('util').deprecate;
|
||||||
const is = require('./is');
|
const is = require('./is');
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Weighting to apply to image crop.
|
* Weighting to apply when using contain/cover fit.
|
||||||
* @member
|
* @member
|
||||||
* @private
|
* @private
|
||||||
*/
|
*/
|
||||||
@@ -21,7 +22,23 @@ const gravity = {
|
|||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Strategies for automagic crop behaviour.
|
* Position to apply when using contain/cover fit.
|
||||||
|
* @member
|
||||||
|
* @private
|
||||||
|
*/
|
||||||
|
const position = {
|
||||||
|
top: 1,
|
||||||
|
right: 2,
|
||||||
|
bottom: 3,
|
||||||
|
left: 4,
|
||||||
|
'right top': 5,
|
||||||
|
'right bottom': 6,
|
||||||
|
'left bottom': 7,
|
||||||
|
'left top': 8
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Strategies for automagic cover behaviour.
|
||||||
* @member
|
* @member
|
||||||
* @private
|
* @private
|
||||||
*/
|
*/
|
||||||
@@ -38,45 +55,144 @@ const strategy = {
|
|||||||
const kernel = {
|
const kernel = {
|
||||||
nearest: 'nearest',
|
nearest: 'nearest',
|
||||||
cubic: 'cubic',
|
cubic: 'cubic',
|
||||||
|
mitchell: 'mitchell',
|
||||||
lanczos2: 'lanczos2',
|
lanczos2: 'lanczos2',
|
||||||
lanczos3: 'lanczos3'
|
lanczos3: 'lanczos3'
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Resize image to `width` x `height`.
|
* Methods by which an image can be resized to fit the provided dimensions.
|
||||||
* By default, the resized image is centre cropped to the exact size specified.
|
* @member
|
||||||
|
* @private
|
||||||
|
*/
|
||||||
|
const fit = {
|
||||||
|
contain: 'contain',
|
||||||
|
cover: 'cover',
|
||||||
|
fill: 'fill',
|
||||||
|
inside: 'inside',
|
||||||
|
outside: 'outside'
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Map external fit property to internal canvas property.
|
||||||
|
* @member
|
||||||
|
* @private
|
||||||
|
*/
|
||||||
|
const mapFitToCanvas = {
|
||||||
|
contain: 'embed',
|
||||||
|
cover: 'crop',
|
||||||
|
fill: 'ignore_aspect',
|
||||||
|
inside: 'max',
|
||||||
|
outside: 'min'
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Resize image to `width`, `height` or `width x height`.
|
||||||
*
|
*
|
||||||
* Possible kernels are:
|
* When both a `width` and `height` are provided, the possible methods by which the image should **fit** these are:
|
||||||
|
* - `cover`: Crop to cover both provided dimensions (the default).
|
||||||
|
* - `contain`: Embed within both provided dimensions.
|
||||||
|
* - `fill`: Ignore the aspect ratio of the input and stretch to both provided dimensions.
|
||||||
|
* - `inside`: Preserving aspect ratio, resize the image to be as large as possible while ensuring its dimensions are less than or equal to both those specified.
|
||||||
|
* - `outside`: Preserving aspect ratio, resize the image to be as small as possible while ensuring its dimensions are greater than or equal to both those specified.
|
||||||
|
* Some of these values are based on the [object-fit](https://developer.mozilla.org/en-US/docs/Web/CSS/object-fit) CSS property.
|
||||||
|
*
|
||||||
|
* When using a `fit` of `cover` or `contain`, the default **position** is `centre`. Other options are:
|
||||||
|
* - `sharp.position`: `top`, `right top`, `right`, `right bottom`, `bottom`, `left bottom`, `left`, `left top`.
|
||||||
|
* - `sharp.gravity`: `north`, `northeast`, `east`, `southeast`, `south`, `southwest`, `west`, `northwest`, `center` or `centre`.
|
||||||
|
* - `sharp.strategy`: `cover` only, dynamically crop using either the `entropy` or `attention` strategy.
|
||||||
|
* Some of these values are based on the [object-position](https://developer.mozilla.org/en-US/docs/Web/CSS/object-position) CSS property.
|
||||||
|
*
|
||||||
|
* The experimental strategy-based approach resizes so one dimension is at its target length
|
||||||
|
* then repeatedly ranks edge regions, discarding the edge with the lowest score based on the selected strategy.
|
||||||
|
* - `entropy`: focus on the region with the highest [Shannon entropy](https://en.wikipedia.org/wiki/Entropy_%28information_theory%29).
|
||||||
|
* - `attention`: focus on the region with the highest luminance frequency, colour saturation and presence of skin tones.
|
||||||
|
*
|
||||||
|
* Possible interpolation kernels are:
|
||||||
* - `nearest`: Use [nearest neighbour interpolation](http://en.wikipedia.org/wiki/Nearest-neighbor_interpolation).
|
* - `nearest`: Use [nearest neighbour interpolation](http://en.wikipedia.org/wiki/Nearest-neighbor_interpolation).
|
||||||
* - `cubic`: Use a [Catmull-Rom spline](https://en.wikipedia.org/wiki/Centripetal_Catmull%E2%80%93Rom_spline).
|
* - `cubic`: Use a [Catmull-Rom spline](https://en.wikipedia.org/wiki/Centripetal_Catmull%E2%80%93Rom_spline).
|
||||||
|
* - `mitchell`: Use a [Mitchell-Netravali spline](https://www.cs.utexas.edu/~fussell/courses/cs384g-fall2013/lectures/mitchell/Mitchell.pdf).
|
||||||
* - `lanczos2`: Use a [Lanczos kernel](https://en.wikipedia.org/wiki/Lanczos_resampling#Lanczos_kernel) with `a=2`.
|
* - `lanczos2`: Use a [Lanczos kernel](https://en.wikipedia.org/wiki/Lanczos_resampling#Lanczos_kernel) with `a=2`.
|
||||||
* - `lanczos3`: Use a Lanczos kernel with `a=3` (the default).
|
* - `lanczos3`: Use a Lanczos kernel with `a=3` (the default).
|
||||||
*
|
*
|
||||||
* @example
|
* @example
|
||||||
* sharp(inputBuffer)
|
* sharp(input)
|
||||||
|
* .resize({ width: 100 })
|
||||||
|
* .toBuffer()
|
||||||
|
* .then(data => {
|
||||||
|
* // 100 pixels wide, auto-scaled height
|
||||||
|
* });
|
||||||
|
*
|
||||||
|
* @example
|
||||||
|
* sharp(input)
|
||||||
|
* .resize({ height: 100 })
|
||||||
|
* .toBuffer()
|
||||||
|
* .then(data => {
|
||||||
|
* // 100 pixels high, auto-scaled width
|
||||||
|
* });
|
||||||
|
*
|
||||||
|
* @example
|
||||||
|
* sharp(input)
|
||||||
* .resize(200, 300, {
|
* .resize(200, 300, {
|
||||||
* kernel: sharp.kernel.nearest
|
* kernel: sharp.kernel.nearest,
|
||||||
|
* fit: 'contain',
|
||||||
|
* position: 'right top',
|
||||||
|
* background: { r: 255, g: 255, b: 255, alpha: 0.5 }
|
||||||
* })
|
* })
|
||||||
* .background('white')
|
* .toFile('output.png')
|
||||||
* .embed()
|
* .then(() => {
|
||||||
* .toFile('output.tiff')
|
* // output.png is a 200 pixels wide and 300 pixels high image
|
||||||
* .then(function() {
|
* // containing a nearest-neighbour scaled version
|
||||||
* // output.tiff is a 200 pixels wide and 300 pixels high image
|
* // contained within the north-east corner of a semi-transparent white canvas
|
||||||
* // containing a nearest-neighbour scaled version, embedded on a white canvas,
|
* });
|
||||||
* // of the image data in inputBuffer
|
*
|
||||||
|
* @example
|
||||||
|
* const transformer = sharp()
|
||||||
|
* .resize({
|
||||||
|
* width: 200,
|
||||||
|
* height: 200,
|
||||||
|
* fit: sharp.fit.cover,
|
||||||
|
* position: sharp.strategy.entropy
|
||||||
|
* });
|
||||||
|
* // Read image data from readableStream
|
||||||
|
* // Write 200px square auto-cropped image data to writableStream
|
||||||
|
* readableStream
|
||||||
|
* .pipe(transformer)
|
||||||
|
* .pipe(writableStream);
|
||||||
|
*
|
||||||
|
* @example
|
||||||
|
* sharp(input)
|
||||||
|
* .resize(200, 200, {
|
||||||
|
* fit: sharp.fit.inside,
|
||||||
|
* withoutEnlargement: true
|
||||||
|
* })
|
||||||
|
* .toFormat('jpeg')
|
||||||
|
* .toBuffer()
|
||||||
|
* .then(function(outputBuffer) {
|
||||||
|
* // outputBuffer contains JPEG image data
|
||||||
|
* // no wider and no higher than 200 pixels
|
||||||
|
* // and no larger than the input image
|
||||||
* });
|
* });
|
||||||
*
|
*
|
||||||
* @param {Number} [width] - pixels wide the resultant image should be. Use `null` or `undefined` to auto-scale the width to match the height.
|
* @param {Number} [width] - pixels wide the resultant image should be. Use `null` or `undefined` to auto-scale the width to match the height.
|
||||||
* @param {Number} [height] - pixels high the resultant image should be. Use `null` or `undefined` to auto-scale the height to match the width.
|
* @param {Number} [height] - pixels high the resultant image should be. Use `null` or `undefined` to auto-scale the height to match the width.
|
||||||
* @param {Object} [options]
|
* @param {Object} [options]
|
||||||
|
* @param {String} [options.width] - alternative means of specifying `width`. If both are present this take priority.
|
||||||
|
* @param {String} [options.height] - alternative means of specifying `height`. If both are present this take priority.
|
||||||
|
* @param {String} [options.fit='cover'] - how the image should be resized to fit both provided dimensions, one of `cover`, `contain`, `fill`, `inside` or `outside`.
|
||||||
|
* @param {String} [options.position='centre'] - position, gravity or strategy to use when `fit` is `cover` or `contain`.
|
||||||
|
* @param {String|Object} [options.background={r: 0, g: 0, b: 0, alpha: 1}] - background colour when using a `fit` of `contain`, parsed by the [color](https://www.npmjs.org/package/color) module, defaults to black without transparency.
|
||||||
* @param {String} [options.kernel='lanczos3'] - the kernel to use for image reduction.
|
* @param {String} [options.kernel='lanczos3'] - the kernel to use for image reduction.
|
||||||
|
* @param {Boolean} [options.withoutEnlargement=false] - do not enlarge if the width *or* height are already less than the specified dimensions, equivalent to GraphicsMagick's `>` geometry option.
|
||||||
* @param {Boolean} [options.fastShrinkOnLoad=true] - take greater advantage of the JPEG and WebP shrink-on-load feature, which can lead to a slight moiré pattern on some images.
|
* @param {Boolean} [options.fastShrinkOnLoad=true] - take greater advantage of the JPEG and WebP shrink-on-load feature, which can lead to a slight moiré pattern on some images.
|
||||||
* @returns {Sharp}
|
* @returns {Sharp}
|
||||||
* @throws {Error} Invalid parameters
|
* @throws {Error} Invalid parameters
|
||||||
*/
|
*/
|
||||||
function resize (width, height, options) {
|
function resize (width, height, options) {
|
||||||
if (is.defined(width)) {
|
if (is.defined(width)) {
|
||||||
if (is.integer(width) && width > 0) {
|
if (is.object(width) && !is.defined(options)) {
|
||||||
|
options = width;
|
||||||
|
} else if (is.integer(width) && width > 0) {
|
||||||
this.options.width = width;
|
this.options.width = width;
|
||||||
} else {
|
} else {
|
||||||
throw is.invalidParameterError('width', 'positive integer', width);
|
throw is.invalidParameterError('width', 'positive integer', width);
|
||||||
@@ -94,6 +210,38 @@ function resize (width, height, options) {
|
|||||||
this.options.height = -1;
|
this.options.height = -1;
|
||||||
}
|
}
|
||||||
if (is.object(options)) {
|
if (is.object(options)) {
|
||||||
|
// Width
|
||||||
|
if (is.integer(options.width) && options.width > 0) {
|
||||||
|
this.options.width = options.width;
|
||||||
|
}
|
||||||
|
// Height
|
||||||
|
if (is.integer(options.height) && options.height > 0) {
|
||||||
|
this.options.height = options.height;
|
||||||
|
}
|
||||||
|
// Fit
|
||||||
|
if (is.defined(options.fit)) {
|
||||||
|
const canvas = mapFitToCanvas[options.fit];
|
||||||
|
if (is.string(canvas)) {
|
||||||
|
this.options.canvas = canvas;
|
||||||
|
} else {
|
||||||
|
throw is.invalidParameterError('fit', 'valid fit', options.fit);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// Position
|
||||||
|
if (is.defined(options.position)) {
|
||||||
|
const pos = is.integer(options.position)
|
||||||
|
? options.position
|
||||||
|
: strategy[options.position] || position[options.position] || gravity[options.position];
|
||||||
|
if (is.integer(pos) && (is.inRange(pos, 0, 8) || is.inRange(pos, 16, 17))) {
|
||||||
|
this.options.position = pos;
|
||||||
|
} else {
|
||||||
|
throw is.invalidParameterError('position', 'valid position/gravity/strategy', options.position);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// Background
|
||||||
|
if (is.defined(options.background)) {
|
||||||
|
this._setColourOption('resizeBackground', options.background);
|
||||||
|
}
|
||||||
// Kernel
|
// Kernel
|
||||||
if (is.defined(options.kernel)) {
|
if (is.defined(options.kernel)) {
|
||||||
if (is.string(kernel[options.kernel])) {
|
if (is.string(kernel[options.kernel])) {
|
||||||
@@ -102,6 +250,10 @@ function resize (width, height, options) {
|
|||||||
throw is.invalidParameterError('kernel', 'valid kernel name', options.kernel);
|
throw is.invalidParameterError('kernel', 'valid kernel name', options.kernel);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
// Without enlargement
|
||||||
|
if (is.defined(options.withoutEnlargement)) {
|
||||||
|
this._setBooleanOption('withoutEnlargement', options.withoutEnlargement);
|
||||||
|
}
|
||||||
// Shrink on load
|
// Shrink on load
|
||||||
if (is.defined(options.fastShrinkOnLoad)) {
|
if (is.defined(options.fastShrinkOnLoad)) {
|
||||||
this._setBooleanOption('fastShrinkOnLoad', options.fastShrinkOnLoad);
|
this._setBooleanOption('fastShrinkOnLoad', options.fastShrinkOnLoad);
|
||||||
@@ -111,48 +263,144 @@ function resize (width, height, options) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Crop the resized image to the exact size specified, the default behaviour.
|
* Extends/pads the edges of the image with the provided background colour.
|
||||||
*
|
* This operation will always occur after resizing and extraction, if any.
|
||||||
* Possible attributes of the optional `sharp.gravity` are `north`, `northeast`, `east`, `southeast`, `south`,
|
|
||||||
* `southwest`, `west`, `northwest`, `center` and `centre`.
|
|
||||||
*
|
|
||||||
* The experimental strategy-based approach resizes so one dimension is at its target length
|
|
||||||
* then repeatedly ranks edge regions, discarding the edge with the lowest score based on the selected strategy.
|
|
||||||
* - `entropy`: focus on the region with the highest [Shannon entropy](https://en.wikipedia.org/wiki/Entropy_%28information_theory%29).
|
|
||||||
* - `attention`: focus on the region with the highest luminance frequency, colour saturation and presence of skin tones.
|
|
||||||
*
|
*
|
||||||
* @example
|
* @example
|
||||||
* const transformer = sharp()
|
* // Resize to 140 pixels wide, then add 10 transparent pixels
|
||||||
* .resize(200, 200)
|
* // to the top, left and right edges and 20 to the bottom edge
|
||||||
* .crop(sharp.strategy.entropy)
|
* sharp(input)
|
||||||
* .on('error', function(err) {
|
* .resize(140)
|
||||||
* console.log(err);
|
* .extend({
|
||||||
* });
|
* top: 10,
|
||||||
* // Read image data from readableStream
|
* bottom: 20,
|
||||||
* // Write 200px square auto-cropped image data to writableStream
|
* left: 10,
|
||||||
* readableStream.pipe(transformer).pipe(writableStream);
|
* right: 10
|
||||||
|
* background: { r: 0, g: 0, b: 0, alpha: 0 }
|
||||||
|
* })
|
||||||
|
* ...
|
||||||
*
|
*
|
||||||
* @param {String} [crop='centre'] - A member of `sharp.gravity` to crop to an edge/corner or `sharp.strategy` to crop dynamically.
|
* @param {(Number|Object)} extend - single pixel count to add to all edges or an Object with per-edge counts
|
||||||
|
* @param {Number} [extend.top]
|
||||||
|
* @param {Number} [extend.left]
|
||||||
|
* @param {Number} [extend.bottom]
|
||||||
|
* @param {Number} [extend.right]
|
||||||
|
* @param {String|Object} [extend.background={r: 0, g: 0, b: 0, alpha: 1}] - background colour, parsed by the [color](https://www.npmjs.org/package/color) module, defaults to black without transparency.
|
||||||
* @returns {Sharp}
|
* @returns {Sharp}
|
||||||
* @throws {Error} Invalid parameters
|
* @throws {Error} Invalid parameters
|
||||||
|
*/
|
||||||
|
function extend (extend) {
|
||||||
|
if (is.integer(extend) && extend > 0) {
|
||||||
|
this.options.extendTop = extend;
|
||||||
|
this.options.extendBottom = extend;
|
||||||
|
this.options.extendLeft = extend;
|
||||||
|
this.options.extendRight = extend;
|
||||||
|
} else if (
|
||||||
|
is.object(extend) &&
|
||||||
|
is.integer(extend.top) && extend.top >= 0 &&
|
||||||
|
is.integer(extend.bottom) && extend.bottom >= 0 &&
|
||||||
|
is.integer(extend.left) && extend.left >= 0 &&
|
||||||
|
is.integer(extend.right) && extend.right >= 0
|
||||||
|
) {
|
||||||
|
this.options.extendTop = extend.top;
|
||||||
|
this.options.extendBottom = extend.bottom;
|
||||||
|
this.options.extendLeft = extend.left;
|
||||||
|
this.options.extendRight = extend.right;
|
||||||
|
this._setColourOption('extendBackground', extend.background);
|
||||||
|
} else {
|
||||||
|
throw new Error('Invalid edge extension ' + extend);
|
||||||
|
}
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Extract a region of the image.
|
||||||
|
*
|
||||||
|
* - Use `extract` before `resize` for pre-resize extraction.
|
||||||
|
* - Use `extract` after `resize` for post-resize extraction.
|
||||||
|
* - Use `extract` before and after for both.
|
||||||
|
*
|
||||||
|
* @example
|
||||||
|
* sharp(input)
|
||||||
|
* .extract({ left: left, top: top, width: width, height: height })
|
||||||
|
* .toFile(output, function(err) {
|
||||||
|
* // Extract a region of the input image, saving in the same format.
|
||||||
|
* });
|
||||||
|
* @example
|
||||||
|
* sharp(input)
|
||||||
|
* .extract({ left: leftOffsetPre, top: topOffsetPre, width: widthPre, height: heightPre })
|
||||||
|
* .resize(width, height)
|
||||||
|
* .extract({ left: leftOffsetPost, top: topOffsetPost, width: widthPost, height: heightPost })
|
||||||
|
* .toFile(output, function(err) {
|
||||||
|
* // Extract a region, resize, then extract from the resized image
|
||||||
|
* });
|
||||||
|
*
|
||||||
|
* @param {Object} options
|
||||||
|
* @param {Number} options.left - zero-indexed offset from left edge
|
||||||
|
* @param {Number} options.top - zero-indexed offset from top edge
|
||||||
|
* @param {Number} options.width - dimension of extracted image
|
||||||
|
* @param {Number} options.height - dimension of extracted image
|
||||||
|
* @returns {Sharp}
|
||||||
|
* @throws {Error} Invalid parameters
|
||||||
|
*/
|
||||||
|
function extract (options) {
|
||||||
|
const suffix = this.options.width === -1 && this.options.height === -1 ? 'Pre' : 'Post';
|
||||||
|
['left', 'top', 'width', 'height'].forEach(function (name) {
|
||||||
|
const value = options[name];
|
||||||
|
if (is.integer(value) && value >= 0) {
|
||||||
|
this.options[name + (name === 'left' || name === 'top' ? 'Offset' : '') + suffix] = value;
|
||||||
|
} else {
|
||||||
|
throw new Error('Non-integer value for ' + name + ' of ' + value);
|
||||||
|
}
|
||||||
|
}, this);
|
||||||
|
// Ensure existing rotation occurs before pre-resize extraction
|
||||||
|
if (suffix === 'Pre' && ((this.options.angle % 360) !== 0 || this.options.useExifOrientation === true)) {
|
||||||
|
this.options.rotateBeforePreExtract = true;
|
||||||
|
}
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Trim "boring" pixels from all edges that contain values similar to the top-left pixel.
|
||||||
|
* The `info` response Object will contain `trimOffsetLeft` and `trimOffsetTop` properties.
|
||||||
|
* @param {Number} [threshold=10] the allowed difference from the top-left pixel, a number greater than zero.
|
||||||
|
* @returns {Sharp}
|
||||||
|
* @throws {Error} Invalid parameters
|
||||||
|
*/
|
||||||
|
function trim (threshold) {
|
||||||
|
if (!is.defined(threshold)) {
|
||||||
|
this.options.trimThreshold = 10;
|
||||||
|
} else if (is.number(threshold) && threshold > 0) {
|
||||||
|
this.options.trimThreshold = threshold;
|
||||||
|
} else {
|
||||||
|
throw is.invalidParameterError('threshold', 'number greater than zero', threshold);
|
||||||
|
}
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Deprecated functions
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @deprecated
|
||||||
|
* @private
|
||||||
*/
|
*/
|
||||||
function crop (crop) {
|
function crop (crop) {
|
||||||
this.options.canvas = 'crop';
|
this.options.canvas = 'crop';
|
||||||
if (!is.defined(crop)) {
|
if (!is.defined(crop)) {
|
||||||
// Default
|
// Default
|
||||||
this.options.crop = gravity.center;
|
this.options.position = gravity.center;
|
||||||
} else if (is.integer(crop) && is.inRange(crop, 0, 8)) {
|
} else if (is.integer(crop) && is.inRange(crop, 0, 8)) {
|
||||||
// Gravity (numeric)
|
// Gravity (numeric)
|
||||||
this.options.crop = crop;
|
this.options.position = crop;
|
||||||
} else if (is.string(crop) && is.integer(gravity[crop])) {
|
} else if (is.string(crop) && is.integer(gravity[crop])) {
|
||||||
// Gravity (string)
|
// Gravity (string)
|
||||||
this.options.crop = gravity[crop];
|
this.options.position = gravity[crop];
|
||||||
} else if (is.integer(crop) && crop >= strategy.entropy) {
|
} else if (is.integer(crop) && crop >= strategy.entropy) {
|
||||||
// Strategy
|
// Strategy
|
||||||
this.options.crop = crop;
|
this.options.position = crop;
|
||||||
} else if (is.string(crop) && is.integer(strategy[crop])) {
|
} else if (is.string(crop) && is.integer(strategy[crop])) {
|
||||||
// Strategy (string)
|
// Strategy (string)
|
||||||
this.options.crop = strategy[crop];
|
this.options.position = strategy[crop];
|
||||||
} else {
|
} else {
|
||||||
throw is.invalidParameterError('crop', 'valid crop id/name/strategy', crop);
|
throw is.invalidParameterError('crop', 'valid crop id/name/strategy', crop);
|
||||||
}
|
}
|
||||||
@@ -160,66 +408,29 @@ function crop (crop) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Preserving aspect ratio, resize the image to the maximum `width` or `height` specified
|
* @deprecated
|
||||||
* then embed on a background of the exact `width` and `height` specified.
|
* @private
|
||||||
*
|
|
||||||
* If the background contains an alpha value then WebP and PNG format output images will
|
|
||||||
* contain an alpha channel, even when the input image does not.
|
|
||||||
*
|
|
||||||
* @example
|
|
||||||
* sharp('input.gif')
|
|
||||||
* .resize(200, 300)
|
|
||||||
* .background({r: 0, g: 0, b: 0, alpha: 0})
|
|
||||||
* .embed()
|
|
||||||
* .toFormat(sharp.format.webp)
|
|
||||||
* .toBuffer(function(err, outputBuffer) {
|
|
||||||
* if (err) {
|
|
||||||
* throw err;
|
|
||||||
* }
|
|
||||||
* // outputBuffer contains WebP image data of a 200 pixels wide and 300 pixels high
|
|
||||||
* // containing a scaled version, embedded on a transparent canvas, of input.gif
|
|
||||||
* });
|
|
||||||
* @param {String} [embed='centre'] - A member of `sharp.gravity` to embed to an edge/corner.
|
|
||||||
* @returns {Sharp}
|
|
||||||
* @throws {Error} Invalid parameters
|
|
||||||
*/
|
*/
|
||||||
function embed (embed) {
|
function embed (embed) {
|
||||||
this.options.canvas = 'embed';
|
this.options.canvas = 'embed';
|
||||||
|
|
||||||
if (!is.defined(embed)) {
|
if (!is.defined(embed)) {
|
||||||
// Default
|
// Default
|
||||||
this.options.embed = gravity.center;
|
this.options.position = gravity.center;
|
||||||
} else if (is.integer(embed) && is.inRange(embed, 0, 8)) {
|
} else if (is.integer(embed) && is.inRange(embed, 0, 8)) {
|
||||||
// Gravity (numeric)
|
// Gravity (numeric)
|
||||||
this.options.embed = embed;
|
this.options.position = embed;
|
||||||
} else if (is.string(embed) && is.integer(gravity[embed])) {
|
} else if (is.string(embed) && is.integer(gravity[embed])) {
|
||||||
// Gravity (string)
|
// Gravity (string)
|
||||||
this.options.embed = gravity[embed];
|
this.options.position = gravity[embed];
|
||||||
} else {
|
} else {
|
||||||
throw is.invalidParameterError('embed', 'valid embed id/name', embed);
|
throw is.invalidParameterError('embed', 'valid embed id/name', embed);
|
||||||
}
|
}
|
||||||
|
|
||||||
return this;
|
return this;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Preserving aspect ratio, resize the image to be as large as possible
|
* @deprecated
|
||||||
* while ensuring its dimensions are less than or equal to the `width` and `height` specified.
|
* @private
|
||||||
*
|
|
||||||
* Both `width` and `height` must be provided via `resize` otherwise the behaviour will default to `crop`.
|
|
||||||
*
|
|
||||||
* @example
|
|
||||||
* sharp(inputBuffer)
|
|
||||||
* .resize(200, 200)
|
|
||||||
* .max()
|
|
||||||
* .toFormat('jpeg')
|
|
||||||
* .toBuffer()
|
|
||||||
* .then(function(outputBuffer) {
|
|
||||||
* // outputBuffer contains JPEG image data no wider than 200 pixels and no higher
|
|
||||||
* // than 200 pixels regardless of the inputBuffer image dimensions
|
|
||||||
* });
|
|
||||||
*
|
|
||||||
* @returns {Sharp}
|
|
||||||
*/
|
*/
|
||||||
function max () {
|
function max () {
|
||||||
this.options.canvas = 'max';
|
this.options.canvas = 'max';
|
||||||
@@ -227,12 +438,8 @@ function max () {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Preserving aspect ratio, resize the image to be as small as possible
|
* @deprecated
|
||||||
* while ensuring its dimensions are greater than or equal to the `width` and `height` specified.
|
* @private
|
||||||
*
|
|
||||||
* Both `width` and `height` must be provided via `resize` otherwise the behaviour will default to `crop`.
|
|
||||||
*
|
|
||||||
* @returns {Sharp}
|
|
||||||
*/
|
*/
|
||||||
function min () {
|
function min () {
|
||||||
this.options.canvas = 'min';
|
this.options.canvas = 'min';
|
||||||
@@ -240,9 +447,8 @@ function min () {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Ignoring the aspect ratio of the input, stretch the image to
|
* @deprecated
|
||||||
* the exact `width` and/or `height` provided via `resize`.
|
* @private
|
||||||
* @returns {Sharp}
|
|
||||||
*/
|
*/
|
||||||
function ignoreAspectRatio () {
|
function ignoreAspectRatio () {
|
||||||
this.options.canvas = 'ignore_aspect';
|
this.options.canvas = 'ignore_aspect';
|
||||||
@@ -250,15 +456,8 @@ function ignoreAspectRatio () {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Do not enlarge the output image if the input image width *or* height are already less than the required dimensions.
|
* @deprecated
|
||||||
* This is equivalent to GraphicsMagick's `>` geometry option:
|
* @private
|
||||||
* "*change the dimensions of the image only if its width or height exceeds the geometry specification*".
|
|
||||||
* Use with `max()` to preserve the image's aspect ratio.
|
|
||||||
*
|
|
||||||
* The default behaviour *before* function call is `false`, meaning the image will be enlarged.
|
|
||||||
*
|
|
||||||
* @param {Boolean} [withoutEnlargement=true]
|
|
||||||
* @returns {Sharp}
|
|
||||||
*/
|
*/
|
||||||
function withoutEnlargement (withoutEnlargement) {
|
function withoutEnlargement (withoutEnlargement) {
|
||||||
this.options.withoutEnlargement = is.bool(withoutEnlargement) ? withoutEnlargement : true;
|
this.options.withoutEnlargement = is.bool(withoutEnlargement) ? withoutEnlargement : true;
|
||||||
@@ -270,19 +469,23 @@ function withoutEnlargement (withoutEnlargement) {
|
|||||||
* @private
|
* @private
|
||||||
*/
|
*/
|
||||||
module.exports = function (Sharp) {
|
module.exports = function (Sharp) {
|
||||||
[
|
Object.assign(Sharp.prototype, {
|
||||||
resize,
|
resize,
|
||||||
crop,
|
extend,
|
||||||
embed,
|
extract,
|
||||||
max,
|
trim
|
||||||
min,
|
|
||||||
ignoreAspectRatio,
|
|
||||||
withoutEnlargement
|
|
||||||
].forEach(function (f) {
|
|
||||||
Sharp.prototype[f.name] = f;
|
|
||||||
});
|
});
|
||||||
// Class attributes
|
// Class attributes
|
||||||
Sharp.gravity = gravity;
|
Sharp.gravity = gravity;
|
||||||
Sharp.strategy = strategy;
|
Sharp.strategy = strategy;
|
||||||
Sharp.kernel = kernel;
|
Sharp.kernel = kernel;
|
||||||
|
Sharp.fit = fit;
|
||||||
|
Sharp.position = position;
|
||||||
|
// Deprecated functions, to be removed in v0.22.0
|
||||||
|
Sharp.prototype.crop = deprecate(crop, 'crop(position) is deprecated, use resize({ fit: "cover", position }) instead');
|
||||||
|
Sharp.prototype.embed = deprecate(embed, 'embed(position) is deprecated, use resize({ fit: "contain", position }) instead');
|
||||||
|
Sharp.prototype.max = deprecate(max, 'max() is deprecated, use resize({ fit: "inside" }) instead');
|
||||||
|
Sharp.prototype.min = deprecate(min, 'min() is deprecated, use resize({ fit: "outside" }) instead');
|
||||||
|
Sharp.prototype.ignoreAspectRatio = deprecate(ignoreAspectRatio, 'ignoreAspectRatio() is deprecated, use resize({ fit: "fill" }) instead');
|
||||||
|
Sharp.prototype.withoutEnlargement = deprecate(withoutEnlargement, 'withoutEnlargement() is deprecated, use resize({ withoutEnlargement: true }) instead');
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -82,23 +82,20 @@ function counters () {
|
|||||||
* Improves the performance of `resize`, `blur` and `sharpen` operations
|
* Improves the performance of `resize`, `blur` and `sharpen` operations
|
||||||
* by taking advantage of the SIMD vector unit of the CPU, e.g. Intel SSE and ARM NEON.
|
* by taking advantage of the SIMD vector unit of the CPU, e.g. Intel SSE and ARM NEON.
|
||||||
*
|
*
|
||||||
* This feature is currently off by default but future versions may reverse this.
|
|
||||||
* Versions of liborc prior to 0.4.25 are known to segfault under heavy load.
|
|
||||||
*
|
|
||||||
* @example
|
* @example
|
||||||
* const simd = sharp.simd();
|
* const simd = sharp.simd();
|
||||||
* // simd is `true` if SIMD is currently enabled
|
* // simd is `true` if the runtime use of liborc is currently enabled
|
||||||
* @example
|
* @example
|
||||||
* const simd = sharp.simd(true);
|
* const simd = sharp.simd(false);
|
||||||
* // attempts to enable the use of SIMD, returning true if available
|
* // prevent libvips from using liborc at runtime
|
||||||
*
|
*
|
||||||
* @param {Boolean} [simd=false]
|
* @param {Boolean} [simd=true]
|
||||||
* @returns {Boolean}
|
* @returns {Boolean}
|
||||||
*/
|
*/
|
||||||
function simd (simd) {
|
function simd (simd) {
|
||||||
return sharp.simd(is.bool(simd) ? simd : null);
|
return sharp.simd(is.bool(simd) ? simd : null);
|
||||||
}
|
}
|
||||||
simd(false);
|
simd(true);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Decorate the Sharp class with utility-related functions.
|
* Decorate the Sharp class with utility-related functions.
|
||||||
|
|||||||
48
package.json
@@ -1,7 +1,7 @@
|
|||||||
{
|
{
|
||||||
"name": "sharp",
|
"name": "sharp",
|
||||||
"description": "High performance Node.js image processing, the fastest module to resize JPEG, PNG, WebP and TIFF images",
|
"description": "High performance Node.js image processing, the fastest module to resize JPEG, PNG, WebP and TIFF images",
|
||||||
"version": "0.20.7",
|
"version": "0.21.1",
|
||||||
"author": "Lovell Fuller <npm@lovell.info>",
|
"author": "Lovell Fuller <npm@lovell.info>",
|
||||||
"homepage": "https://github.com/lovell/sharp",
|
"homepage": "https://github.com/lovell/sharp",
|
||||||
"contributors": [
|
"contributors": [
|
||||||
@@ -52,13 +52,22 @@
|
|||||||
"Tom Lokhorst <tom@lokhorst.eu>",
|
"Tom Lokhorst <tom@lokhorst.eu>",
|
||||||
"Espen Hovlandsdal <espen@hovlandsdal.com>",
|
"Espen Hovlandsdal <espen@hovlandsdal.com>",
|
||||||
"Sylvain Dumont <sylvain.dumont35@gmail.com>",
|
"Sylvain Dumont <sylvain.dumont35@gmail.com>",
|
||||||
"Alun Davies <alun.owain.davies@googlemail.com>"
|
"Alun Davies <alun.owain.davies@googlemail.com>",
|
||||||
|
"Aidan Hoolachan <ajhoolachan21@gmail.com>",
|
||||||
|
"Axel Eirola <axel.eirola@iki.fi>",
|
||||||
|
"Freezy <freezy@xbmc.org>",
|
||||||
|
"Daiz <taneli.vatanen@gmail.com>",
|
||||||
|
"Julian Aubourg <j@ubourg.net>",
|
||||||
|
"Keith Belovay <keith@picthrive.com>",
|
||||||
|
"Michael B. Klein <mbklein@gmail.com>"
|
||||||
],
|
],
|
||||||
"scripts": {
|
"scripts": {
|
||||||
"install": "(node install/libvips && node install/dll-copy && prebuild-install) || (node-gyp rebuild && node install/dll-copy)",
|
"install": "(node install/libvips && node install/dll-copy && prebuild-install) || (node-gyp rebuild && node install/dll-copy)",
|
||||||
"clean": "rm -rf node_modules/ build/ vendor/ coverage/ test/fixtures/output.*",
|
"clean": "rm -rf node_modules/ build/ vendor/ .nyc_output/ coverage/ test/fixtures/output.*",
|
||||||
"test": "semistandard && cc && nyc --reporter=lcov --branches=99 mocha --slow=5000 --timeout=60000 ./test/unit/*.js && prebuild-ci",
|
"test": "semistandard && cc && npm run test-unit && npm run test-licensing && prebuild-ci",
|
||||||
"coverage": "./test/coverage/report.sh",
|
"test-unit": "nyc --reporter=lcov --branches=99 mocha --slow=5000 --timeout=60000 ./test/unit/*.js",
|
||||||
|
"test-licensing": "license-checker --production --summary --onlyAllow=\"Apache-2.0;BSD;ISC;MIT\"",
|
||||||
|
"test-coverage": "./test/coverage/report.sh",
|
||||||
"test-leak": "./test/leak/leak.sh",
|
"test-leak": "./test/leak/leak.sh",
|
||||||
"docs": "for m in constructor input resize composite operation colour channel output utility; do documentation build --shallow --format=md --markdown-toc=false lib/$m.js >docs/api-$m.md; done"
|
"docs": "for m in constructor input resize composite operation colour channel output utility; do documentation build --shallow --format=md --markdown-toc=false lib/$m.js >docs/api-$m.md; done"
|
||||||
},
|
},
|
||||||
@@ -84,37 +93,40 @@
|
|||||||
"vips"
|
"vips"
|
||||||
],
|
],
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"color": "^3.0.0",
|
"bindings": "^1.3.1",
|
||||||
|
"color": "^3.1.0",
|
||||||
"detect-libc": "^1.0.3",
|
"detect-libc": "^1.0.3",
|
||||||
"nan": "^2.10.0",
|
|
||||||
"fs-copy-file-sync": "^1.1.1",
|
"fs-copy-file-sync": "^1.1.1",
|
||||||
|
"nan": "^2.11.1",
|
||||||
"npmlog": "^4.1.2",
|
"npmlog": "^4.1.2",
|
||||||
"prebuild-install": "^4.0.0",
|
"prebuild-install": "^5.2.2",
|
||||||
"semver": "^5.5.1",
|
"semver": "^5.6.0",
|
||||||
"simple-get": "^2.8.1",
|
"simple-get": "^3.0.3",
|
||||||
"tar": "^4.4.6",
|
"tar": "^4.4.8",
|
||||||
"tunnel-agent": "^0.6.0"
|
"tunnel-agent": "^0.6.0"
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"async": "^2.6.1",
|
"async": "^2.6.1",
|
||||||
"cc": "^1.0.2",
|
"cc": "^1.0.2",
|
||||||
"decompress-zip": "^0.3.1",
|
"decompress-zip": "^0.3.1",
|
||||||
"documentation": "^8.1.1",
|
"documentation": "^8.1.2",
|
||||||
"exif-reader": "^1.0.2",
|
"exif-reader": "^1.0.2",
|
||||||
"icc": "^1.0.0",
|
"icc": "^1.0.0",
|
||||||
|
"license-checker": "^24.0.1",
|
||||||
"mocha": "^5.2.0",
|
"mocha": "^5.2.0",
|
||||||
"nyc": "^12.0.2",
|
"mock-fs": "^4.7.0",
|
||||||
"prebuild": "^7.6.2",
|
"nyc": "^13.1.0",
|
||||||
"prebuild-ci": "^2.2.3",
|
"prebuild": "^8.1.2",
|
||||||
|
"prebuild-ci": "^2.3.0",
|
||||||
"rimraf": "^2.6.2",
|
"rimraf": "^2.6.2",
|
||||||
"semistandard": "^12.0.1"
|
"semistandard": "^13.0.1"
|
||||||
},
|
},
|
||||||
"license": "Apache-2.0",
|
"license": "Apache-2.0",
|
||||||
"config": {
|
"config": {
|
||||||
"libvips": "8.6.1"
|
"libvips": "8.7.0"
|
||||||
},
|
},
|
||||||
"engines": {
|
"engines": {
|
||||||
"node": ">=4.5.0"
|
"node": ">=6"
|
||||||
},
|
},
|
||||||
"semistandard": {
|
"semistandard": {
|
||||||
"env": [
|
"env": [
|
||||||
|
|||||||
@@ -37,6 +37,14 @@ namespace sharp {
|
|||||||
std::string AttrAsStr(v8::Handle<v8::Object> obj, std::string attr) {
|
std::string AttrAsStr(v8::Handle<v8::Object> obj, std::string attr) {
|
||||||
return *Nan::Utf8String(Nan::Get(obj, Nan::New(attr).ToLocalChecked()).ToLocalChecked());
|
return *Nan::Utf8String(Nan::Get(obj, Nan::New(attr).ToLocalChecked()).ToLocalChecked());
|
||||||
}
|
}
|
||||||
|
std::vector<double> AttrAsRgba(v8::Handle<v8::Object> obj, std::string attr) {
|
||||||
|
v8::Local<v8::Object> background = AttrAs<v8::Object>(obj, attr);
|
||||||
|
std::vector<double> rgba(4);
|
||||||
|
for (unsigned int i = 0; i < 4; i++) {
|
||||||
|
rgba[i] = AttrTo<double>(background, i);
|
||||||
|
}
|
||||||
|
return rgba;
|
||||||
|
}
|
||||||
|
|
||||||
// Create an InputDescriptor instance from a v8::Object describing an input image
|
// Create an InputDescriptor instance from a v8::Object describing an input image
|
||||||
InputDescriptor* CreateInputDescriptor(
|
InputDescriptor* CreateInputDescriptor(
|
||||||
@@ -55,7 +63,7 @@ namespace sharp {
|
|||||||
descriptor->failOnError = AttrTo<bool>(input, "failOnError");
|
descriptor->failOnError = AttrTo<bool>(input, "failOnError");
|
||||||
// Density for vector-based input
|
// Density for vector-based input
|
||||||
if (HasAttr(input, "density")) {
|
if (HasAttr(input, "density")) {
|
||||||
descriptor->density = AttrTo<uint32_t>(input, "density");
|
descriptor->density = AttrTo<double>(input, "density");
|
||||||
}
|
}
|
||||||
// Raw pixel input
|
// Raw pixel input
|
||||||
if (HasAttr(input, "rawChannels")) {
|
if (HasAttr(input, "rawChannels")) {
|
||||||
@@ -72,10 +80,7 @@ namespace sharp {
|
|||||||
descriptor->createChannels = AttrTo<uint32_t>(input, "createChannels");
|
descriptor->createChannels = AttrTo<uint32_t>(input, "createChannels");
|
||||||
descriptor->createWidth = AttrTo<uint32_t>(input, "createWidth");
|
descriptor->createWidth = AttrTo<uint32_t>(input, "createWidth");
|
||||||
descriptor->createHeight = AttrTo<uint32_t>(input, "createHeight");
|
descriptor->createHeight = AttrTo<uint32_t>(input, "createHeight");
|
||||||
v8::Local<v8::Object> createBackground = AttrAs<v8::Object>(input, "createBackground");
|
descriptor->createBackground = AttrAsRgba(input, "createBackground");
|
||||||
for (unsigned int i = 0; i < 4; i++) {
|
|
||||||
descriptor->createBackground[i] = AttrTo<double>(createBackground, i);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
return descriptor;
|
return descriptor;
|
||||||
}
|
}
|
||||||
@@ -228,7 +233,7 @@ namespace sharp {
|
|||||||
->set("access", accessMethod)
|
->set("access", accessMethod)
|
||||||
->set("fail", descriptor->failOnError);
|
->set("fail", descriptor->failOnError);
|
||||||
if (imageType == ImageType::SVG || imageType == ImageType::PDF) {
|
if (imageType == ImageType::SVG || imageType == ImageType::PDF) {
|
||||||
option->set("dpi", static_cast<double>(descriptor->density));
|
option->set("dpi", descriptor->density);
|
||||||
}
|
}
|
||||||
if (imageType == ImageType::MAGICK) {
|
if (imageType == ImageType::MAGICK) {
|
||||||
option->set("density", std::to_string(descriptor->density).data());
|
option->set("density", std::to_string(descriptor->density).data());
|
||||||
@@ -270,7 +275,7 @@ namespace sharp {
|
|||||||
->set("access", accessMethod)
|
->set("access", accessMethod)
|
||||||
->set("fail", descriptor->failOnError);
|
->set("fail", descriptor->failOnError);
|
||||||
if (imageType == ImageType::SVG || imageType == ImageType::PDF) {
|
if (imageType == ImageType::SVG || imageType == ImageType::PDF) {
|
||||||
option->set("dpi", static_cast<double>(descriptor->density));
|
option->set("dpi", descriptor->density);
|
||||||
}
|
}
|
||||||
if (imageType == ImageType::MAGICK) {
|
if (imageType == ImageType::MAGICK) {
|
||||||
option->set("density", std::to_string(descriptor->density).data());
|
option->set("density", std::to_string(descriptor->density).data());
|
||||||
@@ -355,8 +360,8 @@ namespace sharp {
|
|||||||
/*
|
/*
|
||||||
Set pixels/mm resolution based on a pixels/inch density.
|
Set pixels/mm resolution based on a pixels/inch density.
|
||||||
*/
|
*/
|
||||||
void SetDensity(VImage image, const int density) {
|
void SetDensity(VImage image, const double density) {
|
||||||
const double pixelsPerMm = static_cast<double>(density) / 25.4;
|
const double pixelsPerMm = density / 25.4;
|
||||||
image.set("Xres", pixelsPerMm);
|
image.set("Xres", pixelsPerMm);
|
||||||
image.set("Yres", pixelsPerMm);
|
image.set("Yres", pixelsPerMm);
|
||||||
image.set(VIPS_META_RESOLUTION_UNIT, "in");
|
image.set(VIPS_META_RESOLUTION_UNIT, "in");
|
||||||
@@ -370,10 +375,6 @@ namespace sharp {
|
|||||||
if (image.width() > 65535 || image.height() > 65535) {
|
if (image.width() > 65535 || image.height() > 65535) {
|
||||||
throw vips::VError("Processed image is too large for the JPEG format");
|
throw vips::VError("Processed image is too large for the JPEG format");
|
||||||
}
|
}
|
||||||
} else if (imageType == ImageType::PNG) {
|
|
||||||
if (image.width() > 2147483647 || image.height() > 2147483647) {
|
|
||||||
throw vips::VError("Processed image is too large for the PNG format");
|
|
||||||
}
|
|
||||||
} else if (imageType == ImageType::WEBP) {
|
} else if (imageType == ImageType::WEBP) {
|
||||||
if (image.width() > 16383 || image.height() > 16383) {
|
if (image.width() > 16383 || image.height() > 16383) {
|
||||||
throw vips::VError("Processed image is too large for the WebP format");
|
throw vips::VError("Processed image is too large for the WebP format");
|
||||||
@@ -606,4 +607,40 @@ namespace sharp {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
Apply the alpha channel to a given colour
|
||||||
|
*/
|
||||||
|
std::tuple<VImage, std::vector<double>> ApplyAlpha(VImage image, std::vector<double> colour) {
|
||||||
|
// Scale up 8-bit values to match 16-bit input image
|
||||||
|
double const multiplier = sharp::Is16Bit(image.interpretation()) ? 256.0 : 1.0;
|
||||||
|
// Create alphaColour colour
|
||||||
|
std::vector<double> alphaColour;
|
||||||
|
if (image.bands() > 2) {
|
||||||
|
alphaColour = {
|
||||||
|
multiplier * colour[0],
|
||||||
|
multiplier * colour[1],
|
||||||
|
multiplier * colour[2]
|
||||||
|
};
|
||||||
|
} else {
|
||||||
|
// Convert sRGB to greyscale
|
||||||
|
alphaColour = { multiplier * (
|
||||||
|
0.2126 * colour[0] +
|
||||||
|
0.7152 * colour[1] +
|
||||||
|
0.0722 * colour[2])
|
||||||
|
};
|
||||||
|
}
|
||||||
|
// Add alpha channel to alphaColour colour
|
||||||
|
if (colour[3] < 255.0 || HasAlpha(image)) {
|
||||||
|
alphaColour.push_back(colour[3] * multiplier);
|
||||||
|
}
|
||||||
|
// Ensure alphaColour colour uses correct colourspace
|
||||||
|
alphaColour = sharp::GetRgbaAsColourspace(alphaColour, image.interpretation());
|
||||||
|
// Add non-transparent alpha channel, if required
|
||||||
|
if (colour[3] < 255.0 && !HasAlpha(image)) {
|
||||||
|
image = image.bandjoin(
|
||||||
|
VImage::new_matrix(image.width(), image.height()).new_from_image(255 * multiplier));
|
||||||
|
}
|
||||||
|
return std::make_tuple(image, alphaColour);
|
||||||
|
}
|
||||||
|
|
||||||
} // namespace sharp
|
} // namespace sharp
|
||||||
|
|||||||
22
src/common.h
@@ -49,7 +49,7 @@ namespace sharp {
|
|||||||
char *buffer;
|
char *buffer;
|
||||||
bool failOnError;
|
bool failOnError;
|
||||||
size_t bufferLength;
|
size_t bufferLength;
|
||||||
int density;
|
double density;
|
||||||
int rawChannels;
|
int rawChannels;
|
||||||
int rawWidth;
|
int rawWidth;
|
||||||
int rawHeight;
|
int rawHeight;
|
||||||
@@ -57,30 +57,27 @@ namespace sharp {
|
|||||||
int createChannels;
|
int createChannels;
|
||||||
int createWidth;
|
int createWidth;
|
||||||
int createHeight;
|
int createHeight;
|
||||||
double createBackground[4];
|
std::vector<double> createBackground;
|
||||||
|
|
||||||
InputDescriptor():
|
InputDescriptor():
|
||||||
buffer(nullptr),
|
buffer(nullptr),
|
||||||
failOnError(FALSE),
|
failOnError(FALSE),
|
||||||
bufferLength(0),
|
bufferLength(0),
|
||||||
density(72),
|
density(72.0),
|
||||||
rawChannels(0),
|
rawChannels(0),
|
||||||
rawWidth(0),
|
rawWidth(0),
|
||||||
rawHeight(0),
|
rawHeight(0),
|
||||||
page(0),
|
page(0),
|
||||||
createChannels(0),
|
createChannels(0),
|
||||||
createWidth(0),
|
createWidth(0),
|
||||||
createHeight(0) {
|
createHeight(0),
|
||||||
createBackground[0] = 0.0;
|
createBackground{ 0.0, 0.0, 0.0, 255.0 } {}
|
||||||
createBackground[1] = 0.0;
|
|
||||||
createBackground[2] = 0.0;
|
|
||||||
createBackground[3] = 255.0;
|
|
||||||
}
|
|
||||||
};
|
};
|
||||||
|
|
||||||
// Convenience methods to access the attributes of a v8::Object
|
// Convenience methods to access the attributes of a v8::Object
|
||||||
bool HasAttr(v8::Handle<v8::Object> obj, std::string attr);
|
bool HasAttr(v8::Handle<v8::Object> obj, std::string attr);
|
||||||
std::string AttrAsStr(v8::Handle<v8::Object> obj, std::string attr);
|
std::string AttrAsStr(v8::Handle<v8::Object> obj, std::string attr);
|
||||||
|
std::vector<double> AttrAsRgba(v8::Handle<v8::Object> obj, std::string attr);
|
||||||
template<typename T> v8::Local<T> AttrAs(v8::Handle<v8::Object> obj, std::string attr) {
|
template<typename T> v8::Local<T> AttrAs(v8::Handle<v8::Object> obj, std::string attr) {
|
||||||
return Nan::Get(obj, Nan::New(attr).ToLocalChecked()).ToLocalChecked().As<T>();
|
return Nan::Get(obj, Nan::New(attr).ToLocalChecked()).ToLocalChecked().As<T>();
|
||||||
}
|
}
|
||||||
@@ -186,7 +183,7 @@ namespace sharp {
|
|||||||
/*
|
/*
|
||||||
Set pixels/mm resolution based on a pixels/inch density.
|
Set pixels/mm resolution based on a pixels/inch density.
|
||||||
*/
|
*/
|
||||||
void SetDensity(VImage image, const int density);
|
void SetDensity(VImage image, const double density);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Check the proposed format supports the current dimensions.
|
Check the proposed format supports the current dimensions.
|
||||||
@@ -255,6 +252,11 @@ namespace sharp {
|
|||||||
*/
|
*/
|
||||||
std::vector<double> GetRgbaAsColourspace(std::vector<double> const rgba, VipsInterpretation const interpretation);
|
std::vector<double> GetRgbaAsColourspace(std::vector<double> const rgba, VipsInterpretation const interpretation);
|
||||||
|
|
||||||
|
/*
|
||||||
|
Apply the alpha channel to a given colour
|
||||||
|
*/
|
||||||
|
std::tuple<VImage, std::vector<double>> ApplyAlpha(VImage image, std::vector<double> colour);
|
||||||
|
|
||||||
} // namespace sharp
|
} // namespace sharp
|
||||||
|
|
||||||
#endif // SRC_COMMON_H_
|
#endif // SRC_COMMON_H_
|
||||||
|
|||||||
@@ -613,7 +613,7 @@ VImage::new_matrixv( int width, int height, ... )
|
|||||||
}
|
}
|
||||||
|
|
||||||
VImage
|
VImage
|
||||||
VImage::write( VImage out )
|
VImage::write( VImage out ) const
|
||||||
{
|
{
|
||||||
if( vips_image_write( this->get_image(), out.get_image() ) )
|
if( vips_image_write( this->get_image(), out.get_image() ) )
|
||||||
throw VError();
|
throw VError();
|
||||||
@@ -622,7 +622,7 @@ VImage::write( VImage out )
|
|||||||
}
|
}
|
||||||
|
|
||||||
void
|
void
|
||||||
VImage::write_to_file( const char *name, VOption *options )
|
VImage::write_to_file( const char *name, VOption *options ) const
|
||||||
{
|
{
|
||||||
char filename[VIPS_PATH_MAX];
|
char filename[VIPS_PATH_MAX];
|
||||||
char option_string[VIPS_PATH_MAX];
|
char option_string[VIPS_PATH_MAX];
|
||||||
@@ -642,7 +642,7 @@ VImage::write_to_file( const char *name, VOption *options )
|
|||||||
|
|
||||||
void
|
void
|
||||||
VImage::write_to_buffer( const char *suffix, void **buf, size_t *size,
|
VImage::write_to_buffer( const char *suffix, void **buf, size_t *size,
|
||||||
VOption *options )
|
VOption *options ) const
|
||||||
{
|
{
|
||||||
char filename[VIPS_PATH_MAX];
|
char filename[VIPS_PATH_MAX];
|
||||||
char option_string[VIPS_PATH_MAX];
|
char option_string[VIPS_PATH_MAX];
|
||||||
@@ -675,7 +675,7 @@ VImage::write_to_buffer( const char *suffix, void **buf, size_t *size,
|
|||||||
#include "vips-operators.cpp"
|
#include "vips-operators.cpp"
|
||||||
|
|
||||||
std::vector<VImage>
|
std::vector<VImage>
|
||||||
VImage::bandsplit( VOption *options )
|
VImage::bandsplit( VOption *options ) const
|
||||||
{
|
{
|
||||||
std::vector<VImage> b;
|
std::vector<VImage> b;
|
||||||
|
|
||||||
@@ -686,7 +686,7 @@ VImage::bandsplit( VOption *options )
|
|||||||
}
|
}
|
||||||
|
|
||||||
VImage
|
VImage
|
||||||
VImage::bandjoin( VImage other, VOption *options )
|
VImage::bandjoin( VImage other, VOption *options ) const
|
||||||
{
|
{
|
||||||
VImage v[2] = { *this, other };
|
VImage v[2] = { *this, other };
|
||||||
std::vector<VImage> vec( v, v + VIPS_NUMBER( v ) );
|
std::vector<VImage> vec( v, v + VIPS_NUMBER( v ) );
|
||||||
@@ -695,7 +695,7 @@ VImage::bandjoin( VImage other, VOption *options )
|
|||||||
}
|
}
|
||||||
|
|
||||||
VImage
|
VImage
|
||||||
VImage::composite( VImage other, VipsBlendMode mode, VOption *options )
|
VImage::composite( VImage other, VipsBlendMode mode, VOption *options ) const
|
||||||
{
|
{
|
||||||
VImage v[2] = { *this, other };
|
VImage v[2] = { *this, other };
|
||||||
std::vector<VImage> ivec( v, v + VIPS_NUMBER( v ) );
|
std::vector<VImage> ivec( v, v + VIPS_NUMBER( v ) );
|
||||||
@@ -706,7 +706,7 @@ VImage::composite( VImage other, VipsBlendMode mode, VOption *options )
|
|||||||
}
|
}
|
||||||
|
|
||||||
std::complex<double>
|
std::complex<double>
|
||||||
VImage::minpos( VOption *options )
|
VImage::minpos( VOption *options ) const
|
||||||
{
|
{
|
||||||
double x, y;
|
double x, y;
|
||||||
|
|
||||||
@@ -719,7 +719,7 @@ VImage::minpos( VOption *options )
|
|||||||
}
|
}
|
||||||
|
|
||||||
std::complex<double>
|
std::complex<double>
|
||||||
VImage::maxpos( VOption *options )
|
VImage::maxpos( VOption *options ) const
|
||||||
{
|
{
|
||||||
double x, y;
|
double x, y;
|
||||||
|
|
||||||
@@ -734,43 +734,43 @@ VImage::maxpos( VOption *options )
|
|||||||
// Operator overloads
|
// Operator overloads
|
||||||
|
|
||||||
VImage
|
VImage
|
||||||
VImage::operator[]( int index )
|
VImage::operator[]( int index ) const
|
||||||
{
|
{
|
||||||
return( this->extract_band( index ) );
|
return( this->extract_band( index ) );
|
||||||
}
|
}
|
||||||
|
|
||||||
std::vector<double>
|
std::vector<double>
|
||||||
VImage::operator()( int x, int y )
|
VImage::operator()( int x, int y ) const
|
||||||
{
|
{
|
||||||
return( this->getpoint( x, y ) );
|
return( this->getpoint( x, y ) );
|
||||||
}
|
}
|
||||||
|
|
||||||
VImage
|
VImage
|
||||||
operator+( VImage a, VImage b )
|
operator+( const VImage a, const VImage b )
|
||||||
{
|
{
|
||||||
return( a.add( b ) );
|
return( a.add( b ) );
|
||||||
}
|
}
|
||||||
|
|
||||||
VImage
|
VImage
|
||||||
operator+( double a, VImage b )
|
operator+( double a, const VImage b )
|
||||||
{
|
{
|
||||||
return( b.linear( 1.0, a ) );
|
return( b.linear( 1.0, a ) );
|
||||||
}
|
}
|
||||||
|
|
||||||
VImage
|
VImage
|
||||||
operator+( VImage a, double b )
|
operator+( const VImage a, double b )
|
||||||
{
|
{
|
||||||
return( a.linear( 1.0, b ) );
|
return( a.linear( 1.0, b ) );
|
||||||
}
|
}
|
||||||
|
|
||||||
VImage
|
VImage
|
||||||
operator+( std::vector<double> a, VImage b )
|
operator+( const std::vector<double> a, const VImage b )
|
||||||
{
|
{
|
||||||
return( b.linear( 1.0, a ) );
|
return( b.linear( 1.0, a ) );
|
||||||
}
|
}
|
||||||
|
|
||||||
VImage
|
VImage
|
||||||
operator+( VImage a, std::vector<double> b )
|
operator+( const VImage a, const std::vector<double> b )
|
||||||
{
|
{
|
||||||
return( a.linear( 1.0, b ) );
|
return( a.linear( 1.0, b ) );
|
||||||
}
|
}
|
||||||
@@ -788,37 +788,37 @@ operator+=( VImage &a, const double b )
|
|||||||
}
|
}
|
||||||
|
|
||||||
VImage &
|
VImage &
|
||||||
operator+=( VImage &a, std::vector<double> b )
|
operator+=( VImage &a, const std::vector<double> b )
|
||||||
{
|
{
|
||||||
return( a = a + b );
|
return( a = a + b );
|
||||||
}
|
}
|
||||||
|
|
||||||
VImage
|
VImage
|
||||||
operator-( VImage a, VImage b )
|
operator-( const VImage a, const VImage b )
|
||||||
{
|
{
|
||||||
return( a.subtract( b ) );
|
return( a.subtract( b ) );
|
||||||
}
|
}
|
||||||
|
|
||||||
VImage
|
VImage
|
||||||
operator-( double a, VImage b )
|
operator-( double a, const VImage b )
|
||||||
{
|
{
|
||||||
return( b.linear( -1.0, a ) );
|
return( b.linear( -1.0, a ) );
|
||||||
}
|
}
|
||||||
|
|
||||||
VImage
|
VImage
|
||||||
operator-( VImage a, double b )
|
operator-( const VImage a, double b )
|
||||||
{
|
{
|
||||||
return( a.linear( 1.0, -b ) );
|
return( a.linear( 1.0, -b ) );
|
||||||
}
|
}
|
||||||
|
|
||||||
VImage
|
VImage
|
||||||
operator-( std::vector<double> a, VImage b )
|
operator-( const std::vector<double> a, const VImage b )
|
||||||
{
|
{
|
||||||
return( b.linear( -1.0, a ) );
|
return( b.linear( -1.0, a ) );
|
||||||
}
|
}
|
||||||
|
|
||||||
VImage
|
VImage
|
||||||
operator-( VImage a, std::vector<double> b )
|
operator-( const VImage a, const std::vector<double> b )
|
||||||
{
|
{
|
||||||
return( a.linear( 1.0, vips::negate( b ) ) );
|
return( a.linear( 1.0, vips::negate( b ) ) );
|
||||||
}
|
}
|
||||||
@@ -836,43 +836,43 @@ operator-=( VImage &a, const double b )
|
|||||||
}
|
}
|
||||||
|
|
||||||
VImage &
|
VImage &
|
||||||
operator-=( VImage &a, std::vector<double> b )
|
operator-=( VImage &a, const std::vector<double> b )
|
||||||
{
|
{
|
||||||
return( a = a - b );
|
return( a = a - b );
|
||||||
}
|
}
|
||||||
|
|
||||||
VImage
|
VImage
|
||||||
operator-( VImage a )
|
operator-( const VImage a )
|
||||||
{
|
{
|
||||||
return( a * -1 );
|
return( a * -1 );
|
||||||
}
|
}
|
||||||
|
|
||||||
VImage
|
VImage
|
||||||
operator*( VImage a, VImage b )
|
operator*( const VImage a, const VImage b )
|
||||||
{
|
{
|
||||||
return( a.multiply( b ) );
|
return( a.multiply( b ) );
|
||||||
}
|
}
|
||||||
|
|
||||||
VImage
|
VImage
|
||||||
operator*( double a, VImage b )
|
operator*( double a, const VImage b )
|
||||||
{
|
{
|
||||||
return( b.linear( a, 0.0 ) );
|
return( b.linear( a, 0.0 ) );
|
||||||
}
|
}
|
||||||
|
|
||||||
VImage
|
VImage
|
||||||
operator*( VImage a, double b )
|
operator*( const VImage a, double b )
|
||||||
{
|
{
|
||||||
return( a.linear( b, 0.0 ) );
|
return( a.linear( b, 0.0 ) );
|
||||||
}
|
}
|
||||||
|
|
||||||
VImage
|
VImage
|
||||||
operator*( std::vector<double> a, VImage b )
|
operator*( const std::vector<double> a, const VImage b )
|
||||||
{
|
{
|
||||||
return( b.linear( a, 0.0 ) );
|
return( b.linear( a, 0.0 ) );
|
||||||
}
|
}
|
||||||
|
|
||||||
VImage
|
VImage
|
||||||
operator*( VImage a, std::vector<double> b )
|
operator*( const VImage a, const std::vector<double> b )
|
||||||
{
|
{
|
||||||
return( a.linear( b, 0.0 ) );
|
return( a.linear( b, 0.0 ) );
|
||||||
}
|
}
|
||||||
@@ -890,37 +890,37 @@ operator*=( VImage &a, const double b )
|
|||||||
}
|
}
|
||||||
|
|
||||||
VImage &
|
VImage &
|
||||||
operator*=( VImage &a, std::vector<double> b )
|
operator*=( VImage &a, const std::vector<double> b )
|
||||||
{
|
{
|
||||||
return( a = a * b );
|
return( a = a * b );
|
||||||
}
|
}
|
||||||
|
|
||||||
VImage
|
VImage
|
||||||
operator/( VImage a, VImage b )
|
operator/( const VImage a, const VImage b )
|
||||||
{
|
{
|
||||||
return( a.divide( b ) );
|
return( a.divide( b ) );
|
||||||
}
|
}
|
||||||
|
|
||||||
VImage
|
VImage
|
||||||
operator/( double a, VImage b )
|
operator/( double a, const VImage b )
|
||||||
{
|
{
|
||||||
return( b.pow( -1.0 ).linear( a, 0.0 ) );
|
return( b.pow( -1.0 ).linear( a, 0.0 ) );
|
||||||
}
|
}
|
||||||
|
|
||||||
VImage
|
VImage
|
||||||
operator/( VImage a, double b )
|
operator/( const VImage a, double b )
|
||||||
{
|
{
|
||||||
return( a.linear( 1.0 / b, 0.0 ) );
|
return( a.linear( 1.0 / b, 0.0 ) );
|
||||||
}
|
}
|
||||||
|
|
||||||
VImage
|
VImage
|
||||||
operator/( std::vector<double> a, VImage b )
|
operator/( const std::vector<double> a, const VImage b )
|
||||||
{
|
{
|
||||||
return( b.pow( -1.0 ).linear( a, 0.0 ) );
|
return( b.pow( -1.0 ).linear( a, 0.0 ) );
|
||||||
}
|
}
|
||||||
|
|
||||||
VImage
|
VImage
|
||||||
operator/( VImage a, std::vector<double> b )
|
operator/( const VImage a, const std::vector<double> b )
|
||||||
{
|
{
|
||||||
return( a.linear( vips::invert( b ), 0.0 ) );
|
return( a.linear( vips::invert( b ), 0.0 ) );
|
||||||
}
|
}
|
||||||
@@ -938,25 +938,25 @@ operator/=( VImage &a, const double b )
|
|||||||
}
|
}
|
||||||
|
|
||||||
VImage &
|
VImage &
|
||||||
operator/=( VImage &a, std::vector<double> b )
|
operator/=( VImage &a, const std::vector<double> b )
|
||||||
{
|
{
|
||||||
return( a = a / b );
|
return( a = a / b );
|
||||||
}
|
}
|
||||||
|
|
||||||
VImage
|
VImage
|
||||||
operator%( VImage a, VImage b )
|
operator%( const VImage a, const VImage b )
|
||||||
{
|
{
|
||||||
return( a.remainder( b ) );
|
return( a.remainder( b ) );
|
||||||
}
|
}
|
||||||
|
|
||||||
VImage
|
VImage
|
||||||
operator%( VImage a, double b )
|
operator%( const VImage a, const double b )
|
||||||
{
|
{
|
||||||
return( a.remainder_const( to_vector( b ) ) );
|
return( a.remainder_const( to_vector( b ) ) );
|
||||||
}
|
}
|
||||||
|
|
||||||
VImage
|
VImage
|
||||||
operator%( VImage a, std::vector<double> b )
|
operator%( const VImage a, const std::vector<double> b )
|
||||||
{
|
{
|
||||||
return( a.remainder_const( b ) );
|
return( a.remainder_const( b ) );
|
||||||
}
|
}
|
||||||
@@ -974,243 +974,243 @@ operator%=( VImage &a, const double b )
|
|||||||
}
|
}
|
||||||
|
|
||||||
VImage &
|
VImage &
|
||||||
operator%=( VImage &a, std::vector<double> b )
|
operator%=( VImage &a, const std::vector<double> b )
|
||||||
{
|
{
|
||||||
return( a = a % b );
|
return( a = a % b );
|
||||||
}
|
}
|
||||||
|
|
||||||
VImage
|
VImage
|
||||||
operator<( VImage a, VImage b )
|
operator<( const VImage a, const VImage b )
|
||||||
{
|
{
|
||||||
return( a.relational( b, VIPS_OPERATION_RELATIONAL_LESS ) );
|
return( a.relational( b, VIPS_OPERATION_RELATIONAL_LESS ) );
|
||||||
}
|
}
|
||||||
|
|
||||||
VImage
|
VImage
|
||||||
operator<( double a, VImage b )
|
operator<( const double a, const VImage b )
|
||||||
{
|
{
|
||||||
return( b.relational_const( VIPS_OPERATION_RELATIONAL_MORE,
|
return( b.relational_const( VIPS_OPERATION_RELATIONAL_MORE,
|
||||||
to_vector( a ) ) );
|
to_vector( a ) ) );
|
||||||
}
|
}
|
||||||
|
|
||||||
VImage
|
VImage
|
||||||
operator<( VImage a, double b )
|
operator<( const VImage a, const double b )
|
||||||
{
|
{
|
||||||
return( a.relational_const( VIPS_OPERATION_RELATIONAL_LESS,
|
return( a.relational_const( VIPS_OPERATION_RELATIONAL_LESS,
|
||||||
to_vector( b ) ) );
|
to_vector( b ) ) );
|
||||||
}
|
}
|
||||||
|
|
||||||
VImage
|
VImage
|
||||||
operator<( std::vector<double> a, VImage b )
|
operator<( const std::vector<double> a, const VImage b )
|
||||||
{
|
{
|
||||||
return( b.relational_const( VIPS_OPERATION_RELATIONAL_MORE,
|
return( b.relational_const( VIPS_OPERATION_RELATIONAL_MORE,
|
||||||
a ) );
|
a ) );
|
||||||
}
|
}
|
||||||
|
|
||||||
VImage
|
VImage
|
||||||
operator<( VImage a, std::vector<double> b )
|
operator<( const VImage a, const std::vector<double> b )
|
||||||
{
|
{
|
||||||
return( a.relational_const( VIPS_OPERATION_RELATIONAL_LESS,
|
return( a.relational_const( VIPS_OPERATION_RELATIONAL_LESS,
|
||||||
b ) );
|
b ) );
|
||||||
}
|
}
|
||||||
|
|
||||||
VImage
|
VImage
|
||||||
operator<=( VImage a, VImage b )
|
operator<=( const VImage a, const VImage b )
|
||||||
{
|
{
|
||||||
return( a.relational( b, VIPS_OPERATION_RELATIONAL_LESSEQ ) );
|
return( a.relational( b, VIPS_OPERATION_RELATIONAL_LESSEQ ) );
|
||||||
}
|
}
|
||||||
|
|
||||||
VImage
|
VImage
|
||||||
operator<=( double a, VImage b )
|
operator<=( const double a, const VImage b )
|
||||||
{
|
{
|
||||||
return( b.relational_const( VIPS_OPERATION_RELATIONAL_MOREEQ,
|
return( b.relational_const( VIPS_OPERATION_RELATIONAL_MOREEQ,
|
||||||
to_vector( a ) ) );
|
to_vector( a ) ) );
|
||||||
}
|
}
|
||||||
|
|
||||||
VImage
|
VImage
|
||||||
operator<=( VImage a, double b )
|
operator<=( const VImage a, const double b )
|
||||||
{
|
{
|
||||||
return( a.relational_const( VIPS_OPERATION_RELATIONAL_LESSEQ,
|
return( a.relational_const( VIPS_OPERATION_RELATIONAL_LESSEQ,
|
||||||
to_vector( b ) ) );
|
to_vector( b ) ) );
|
||||||
}
|
}
|
||||||
|
|
||||||
VImage
|
VImage
|
||||||
operator<=( std::vector<double> a, VImage b )
|
operator<=( const std::vector<double> a, const VImage b )
|
||||||
{
|
{
|
||||||
return( b.relational_const( VIPS_OPERATION_RELATIONAL_MOREEQ,
|
return( b.relational_const( VIPS_OPERATION_RELATIONAL_MOREEQ,
|
||||||
a ) );
|
a ) );
|
||||||
}
|
}
|
||||||
|
|
||||||
VImage
|
VImage
|
||||||
operator<=( VImage a, std::vector<double> b )
|
operator<=( const VImage a, const std::vector<double> b )
|
||||||
{
|
{
|
||||||
return( a.relational_const( VIPS_OPERATION_RELATIONAL_LESSEQ,
|
return( a.relational_const( VIPS_OPERATION_RELATIONAL_LESSEQ,
|
||||||
b ) );
|
b ) );
|
||||||
}
|
}
|
||||||
|
|
||||||
VImage
|
VImage
|
||||||
operator>( VImage a, VImage b )
|
operator>( const VImage a, const VImage b )
|
||||||
{
|
{
|
||||||
return( a.relational( b, VIPS_OPERATION_RELATIONAL_MORE ) );
|
return( a.relational( b, VIPS_OPERATION_RELATIONAL_MORE ) );
|
||||||
}
|
}
|
||||||
|
|
||||||
VImage
|
VImage
|
||||||
operator>( double a, VImage b )
|
operator>( const double a, const VImage b )
|
||||||
{
|
{
|
||||||
return( b.relational_const( VIPS_OPERATION_RELATIONAL_LESS,
|
return( b.relational_const( VIPS_OPERATION_RELATIONAL_LESS,
|
||||||
to_vector( a ) ) );
|
to_vector( a ) ) );
|
||||||
}
|
}
|
||||||
|
|
||||||
VImage
|
VImage
|
||||||
operator>( VImage a, double b )
|
operator>( const VImage a, const double b )
|
||||||
{
|
{
|
||||||
return( a.relational_const( VIPS_OPERATION_RELATIONAL_MORE,
|
return( a.relational_const( VIPS_OPERATION_RELATIONAL_MORE,
|
||||||
to_vector( b ) ) );
|
to_vector( b ) ) );
|
||||||
}
|
}
|
||||||
|
|
||||||
VImage
|
VImage
|
||||||
operator>( std::vector<double> a, VImage b )
|
operator>( const std::vector<double> a, const VImage b )
|
||||||
{
|
{
|
||||||
return( b.relational_const( VIPS_OPERATION_RELATIONAL_LESS,
|
return( b.relational_const( VIPS_OPERATION_RELATIONAL_LESS,
|
||||||
a ) );
|
a ) );
|
||||||
}
|
}
|
||||||
|
|
||||||
VImage
|
VImage
|
||||||
operator>( VImage a, std::vector<double> b )
|
operator>( const VImage a, const std::vector<double> b )
|
||||||
{
|
{
|
||||||
return( a.relational_const( VIPS_OPERATION_RELATIONAL_MORE,
|
return( a.relational_const( VIPS_OPERATION_RELATIONAL_MORE,
|
||||||
b ) );
|
b ) );
|
||||||
}
|
}
|
||||||
|
|
||||||
VImage
|
VImage
|
||||||
operator>=( VImage a, VImage b )
|
operator>=( const VImage a, const VImage b )
|
||||||
{
|
{
|
||||||
return( a.relational( b, VIPS_OPERATION_RELATIONAL_MOREEQ ) );
|
return( a.relational( b, VIPS_OPERATION_RELATIONAL_MOREEQ ) );
|
||||||
}
|
}
|
||||||
|
|
||||||
VImage
|
VImage
|
||||||
operator>=( double a, VImage b )
|
operator>=( const double a, const VImage b )
|
||||||
{
|
{
|
||||||
return( b.relational_const( VIPS_OPERATION_RELATIONAL_LESSEQ,
|
return( b.relational_const( VIPS_OPERATION_RELATIONAL_LESSEQ,
|
||||||
to_vector( a ) ) );
|
to_vector( a ) ) );
|
||||||
}
|
}
|
||||||
|
|
||||||
VImage
|
VImage
|
||||||
operator>=( VImage a, double b )
|
operator>=( const VImage a, const double b )
|
||||||
{
|
{
|
||||||
return( a.relational_const( VIPS_OPERATION_RELATIONAL_MOREEQ,
|
return( a.relational_const( VIPS_OPERATION_RELATIONAL_MOREEQ,
|
||||||
to_vector( b ) ) );
|
to_vector( b ) ) );
|
||||||
}
|
}
|
||||||
|
|
||||||
VImage
|
VImage
|
||||||
operator>=( std::vector<double> a, VImage b )
|
operator>=( const std::vector<double> a, const VImage b )
|
||||||
{
|
{
|
||||||
return( b.relational_const( VIPS_OPERATION_RELATIONAL_LESSEQ,
|
return( b.relational_const( VIPS_OPERATION_RELATIONAL_LESSEQ,
|
||||||
a ) );
|
a ) );
|
||||||
}
|
}
|
||||||
|
|
||||||
VImage
|
VImage
|
||||||
operator>=( VImage a, std::vector<double> b )
|
operator>=( const VImage a, const std::vector<double> b )
|
||||||
{
|
{
|
||||||
return( a.relational_const( VIPS_OPERATION_RELATIONAL_MOREEQ,
|
return( a.relational_const( VIPS_OPERATION_RELATIONAL_MOREEQ,
|
||||||
b ) );
|
b ) );
|
||||||
}
|
}
|
||||||
|
|
||||||
VImage
|
VImage
|
||||||
operator==( VImage a, VImage b )
|
operator==( const VImage a, const VImage b )
|
||||||
{
|
{
|
||||||
return( a.relational( b, VIPS_OPERATION_RELATIONAL_EQUAL ) );
|
return( a.relational( b, VIPS_OPERATION_RELATIONAL_EQUAL ) );
|
||||||
}
|
}
|
||||||
|
|
||||||
VImage
|
VImage
|
||||||
operator==( double a, VImage b )
|
operator==( const double a, const VImage b )
|
||||||
{
|
{
|
||||||
return( b.relational_const( VIPS_OPERATION_RELATIONAL_EQUAL,
|
return( b.relational_const( VIPS_OPERATION_RELATIONAL_EQUAL,
|
||||||
to_vector( a ) ) );
|
to_vector( a ) ) );
|
||||||
}
|
}
|
||||||
|
|
||||||
VImage
|
VImage
|
||||||
operator==( VImage a, double b )
|
operator==( const VImage a, const double b )
|
||||||
{
|
{
|
||||||
return( a.relational_const( VIPS_OPERATION_RELATIONAL_EQUAL,
|
return( a.relational_const( VIPS_OPERATION_RELATIONAL_EQUAL,
|
||||||
to_vector( b ) ) );
|
to_vector( b ) ) );
|
||||||
}
|
}
|
||||||
|
|
||||||
VImage
|
VImage
|
||||||
operator==( std::vector<double> a, VImage b )
|
operator==( const std::vector<double> a, const VImage b )
|
||||||
{
|
{
|
||||||
return( b.relational_const( VIPS_OPERATION_RELATIONAL_EQUAL,
|
return( b.relational_const( VIPS_OPERATION_RELATIONAL_EQUAL,
|
||||||
a ) );
|
a ) );
|
||||||
}
|
}
|
||||||
|
|
||||||
VImage
|
VImage
|
||||||
operator==( VImage a, std::vector<double> b )
|
operator==( const VImage a, const std::vector<double> b )
|
||||||
{
|
{
|
||||||
return( a.relational_const( VIPS_OPERATION_RELATIONAL_EQUAL,
|
return( a.relational_const( VIPS_OPERATION_RELATIONAL_EQUAL,
|
||||||
b ) );
|
b ) );
|
||||||
}
|
}
|
||||||
|
|
||||||
VImage
|
VImage
|
||||||
operator!=( VImage a, VImage b )
|
operator!=( const VImage a, const VImage b )
|
||||||
{
|
{
|
||||||
return( a.relational( b, VIPS_OPERATION_RELATIONAL_NOTEQ ) );
|
return( a.relational( b, VIPS_OPERATION_RELATIONAL_NOTEQ ) );
|
||||||
}
|
}
|
||||||
|
|
||||||
VImage
|
VImage
|
||||||
operator!=( double a, VImage b )
|
operator!=( const double a, const VImage b )
|
||||||
{
|
{
|
||||||
return( b.relational_const( VIPS_OPERATION_RELATIONAL_NOTEQ,
|
return( b.relational_const( VIPS_OPERATION_RELATIONAL_NOTEQ,
|
||||||
to_vector( a ) ) );
|
to_vector( a ) ) );
|
||||||
}
|
}
|
||||||
|
|
||||||
VImage
|
VImage
|
||||||
operator!=( VImage a, double b )
|
operator!=( const VImage a, const double b )
|
||||||
{
|
{
|
||||||
return( a.relational_const( VIPS_OPERATION_RELATIONAL_NOTEQ,
|
return( a.relational_const( VIPS_OPERATION_RELATIONAL_NOTEQ,
|
||||||
to_vector( b ) ) );
|
to_vector( b ) ) );
|
||||||
}
|
}
|
||||||
|
|
||||||
VImage
|
VImage
|
||||||
operator!=( std::vector<double> a, VImage b )
|
operator!=( const std::vector<double> a, const VImage b )
|
||||||
{
|
{
|
||||||
return( b.relational_const( VIPS_OPERATION_RELATIONAL_NOTEQ,
|
return( b.relational_const( VIPS_OPERATION_RELATIONAL_NOTEQ,
|
||||||
a ) );
|
a ) );
|
||||||
}
|
}
|
||||||
|
|
||||||
VImage
|
VImage
|
||||||
operator!=( VImage a, std::vector<double> b )
|
operator!=( const VImage a, const std::vector<double> b )
|
||||||
{
|
{
|
||||||
return( a.relational_const( VIPS_OPERATION_RELATIONAL_NOTEQ,
|
return( a.relational_const( VIPS_OPERATION_RELATIONAL_NOTEQ,
|
||||||
b ) );
|
b ) );
|
||||||
}
|
}
|
||||||
|
|
||||||
VImage
|
VImage
|
||||||
operator&( VImage a, VImage b )
|
operator&( const VImage a, const VImage b )
|
||||||
{
|
{
|
||||||
return( a.boolean( b, VIPS_OPERATION_BOOLEAN_AND ) );
|
return( a.boolean( b, VIPS_OPERATION_BOOLEAN_AND ) );
|
||||||
}
|
}
|
||||||
|
|
||||||
VImage
|
VImage
|
||||||
operator&( double a, VImage b )
|
operator&( const double a, const VImage b )
|
||||||
{
|
{
|
||||||
return( b.boolean_const( VIPS_OPERATION_BOOLEAN_AND,
|
return( b.boolean_const( VIPS_OPERATION_BOOLEAN_AND,
|
||||||
to_vector( a ) ) );
|
to_vector( a ) ) );
|
||||||
}
|
}
|
||||||
|
|
||||||
VImage
|
VImage
|
||||||
operator&( VImage a, double b )
|
operator&( const VImage a, const double b )
|
||||||
{
|
{
|
||||||
return( a.boolean_const( VIPS_OPERATION_BOOLEAN_AND,
|
return( a.boolean_const( VIPS_OPERATION_BOOLEAN_AND,
|
||||||
to_vector( b ) ) );
|
to_vector( b ) ) );
|
||||||
}
|
}
|
||||||
|
|
||||||
VImage
|
VImage
|
||||||
operator&( std::vector<double> a, VImage b )
|
operator&( const std::vector<double> a, const VImage b )
|
||||||
{
|
{
|
||||||
return( b.boolean_const( VIPS_OPERATION_BOOLEAN_AND, a ) );
|
return( b.boolean_const( VIPS_OPERATION_BOOLEAN_AND, a ) );
|
||||||
}
|
}
|
||||||
|
|
||||||
VImage
|
VImage
|
||||||
operator&( VImage a, std::vector<double> b )
|
operator&( const VImage a, const std::vector<double> b )
|
||||||
{
|
{
|
||||||
return( a.boolean_const( VIPS_OPERATION_BOOLEAN_AND, b ) );
|
return( a.boolean_const( VIPS_OPERATION_BOOLEAN_AND, b ) );
|
||||||
}
|
}
|
||||||
@@ -1228,40 +1228,40 @@ operator&=( VImage &a, const double b )
|
|||||||
}
|
}
|
||||||
|
|
||||||
VImage &
|
VImage &
|
||||||
operator&=( VImage &a, std::vector<double> b )
|
operator&=( VImage &a, const std::vector<double> b )
|
||||||
{
|
{
|
||||||
return( a = a & b );
|
return( a = a & b );
|
||||||
}
|
}
|
||||||
|
|
||||||
VImage
|
VImage
|
||||||
operator|( VImage a, VImage b )
|
operator|( const VImage a, const VImage b )
|
||||||
{
|
{
|
||||||
return( a.boolean( b, VIPS_OPERATION_BOOLEAN_OR ) );
|
return( a.boolean( b, VIPS_OPERATION_BOOLEAN_OR ) );
|
||||||
}
|
}
|
||||||
|
|
||||||
VImage
|
VImage
|
||||||
operator|( double a, VImage b )
|
operator|( const double a, const VImage b )
|
||||||
{
|
{
|
||||||
return( b.boolean_const( VIPS_OPERATION_BOOLEAN_OR,
|
return( b.boolean_const( VIPS_OPERATION_BOOLEAN_OR,
|
||||||
to_vector( a ) ) );
|
to_vector( a ) ) );
|
||||||
}
|
}
|
||||||
|
|
||||||
VImage
|
VImage
|
||||||
operator|( VImage a, double b )
|
operator|( const VImage a, const double b )
|
||||||
{
|
{
|
||||||
return( a.boolean_const( VIPS_OPERATION_BOOLEAN_OR,
|
return( a.boolean_const( VIPS_OPERATION_BOOLEAN_OR,
|
||||||
to_vector( b ) ) );
|
to_vector( b ) ) );
|
||||||
}
|
}
|
||||||
|
|
||||||
VImage
|
VImage
|
||||||
operator|( std::vector<double> a, VImage b )
|
operator|( const std::vector<double> a, const VImage b )
|
||||||
{
|
{
|
||||||
return( b.boolean_const( VIPS_OPERATION_BOOLEAN_OR,
|
return( b.boolean_const( VIPS_OPERATION_BOOLEAN_OR,
|
||||||
a ) );
|
a ) );
|
||||||
}
|
}
|
||||||
|
|
||||||
VImage
|
VImage
|
||||||
operator|( VImage a, std::vector<double> b )
|
operator|( const VImage a, const std::vector<double> b )
|
||||||
{
|
{
|
||||||
return( a.boolean_const( VIPS_OPERATION_BOOLEAN_OR,
|
return( a.boolean_const( VIPS_OPERATION_BOOLEAN_OR,
|
||||||
b ) );
|
b ) );
|
||||||
@@ -1280,40 +1280,40 @@ operator|=( VImage &a, const double b )
|
|||||||
}
|
}
|
||||||
|
|
||||||
VImage &
|
VImage &
|
||||||
operator|=( VImage &a, std::vector<double> b )
|
operator|=( VImage &a, const std::vector<double> b )
|
||||||
{
|
{
|
||||||
return( a = a | b );
|
return( a = a | b );
|
||||||
}
|
}
|
||||||
|
|
||||||
VImage
|
VImage
|
||||||
operator^( VImage a, VImage b )
|
operator^( const VImage a, const VImage b )
|
||||||
{
|
{
|
||||||
return( a.boolean( b, VIPS_OPERATION_BOOLEAN_EOR ) );
|
return( a.boolean( b, VIPS_OPERATION_BOOLEAN_EOR ) );
|
||||||
}
|
}
|
||||||
|
|
||||||
VImage
|
VImage
|
||||||
operator^( double a, VImage b )
|
operator^( const double a, const VImage b )
|
||||||
{
|
{
|
||||||
return( b.boolean_const( VIPS_OPERATION_BOOLEAN_EOR,
|
return( b.boolean_const( VIPS_OPERATION_BOOLEAN_EOR,
|
||||||
to_vector( a ) ) );
|
to_vector( a ) ) );
|
||||||
}
|
}
|
||||||
|
|
||||||
VImage
|
VImage
|
||||||
operator^( VImage a, double b )
|
operator^( const VImage a, const double b )
|
||||||
{
|
{
|
||||||
return( a.boolean_const( VIPS_OPERATION_BOOLEAN_EOR,
|
return( a.boolean_const( VIPS_OPERATION_BOOLEAN_EOR,
|
||||||
to_vector( b ) ) );
|
to_vector( b ) ) );
|
||||||
}
|
}
|
||||||
|
|
||||||
VImage
|
VImage
|
||||||
operator^( std::vector<double> a, VImage b )
|
operator^( const std::vector<double> a, const VImage b )
|
||||||
{
|
{
|
||||||
return( b.boolean_const( VIPS_OPERATION_BOOLEAN_EOR,
|
return( b.boolean_const( VIPS_OPERATION_BOOLEAN_EOR,
|
||||||
a ) );
|
a ) );
|
||||||
}
|
}
|
||||||
|
|
||||||
VImage
|
VImage
|
||||||
operator^( VImage a, std::vector<double> b )
|
operator^( const VImage a, const std::vector<double> b )
|
||||||
{
|
{
|
||||||
return( a.boolean_const( VIPS_OPERATION_BOOLEAN_EOR,
|
return( a.boolean_const( VIPS_OPERATION_BOOLEAN_EOR,
|
||||||
b ) );
|
b ) );
|
||||||
@@ -1332,26 +1332,26 @@ operator^=( VImage &a, const double b )
|
|||||||
}
|
}
|
||||||
|
|
||||||
VImage &
|
VImage &
|
||||||
operator^=( VImage &a, std::vector<double> b )
|
operator^=( VImage &a, const std::vector<double> b )
|
||||||
{
|
{
|
||||||
return( a = a ^ b );
|
return( a = a ^ b );
|
||||||
}
|
}
|
||||||
|
|
||||||
VImage
|
VImage
|
||||||
operator<<( VImage a, VImage b )
|
operator<<( const VImage a, const VImage b )
|
||||||
{
|
{
|
||||||
return( a.boolean( b, VIPS_OPERATION_BOOLEAN_LSHIFT ) );
|
return( a.boolean( b, VIPS_OPERATION_BOOLEAN_LSHIFT ) );
|
||||||
}
|
}
|
||||||
|
|
||||||
VImage
|
VImage
|
||||||
operator<<( VImage a, double b )
|
operator<<( const VImage a, const double b )
|
||||||
{
|
{
|
||||||
return( a.boolean_const( VIPS_OPERATION_BOOLEAN_LSHIFT,
|
return( a.boolean_const( VIPS_OPERATION_BOOLEAN_LSHIFT,
|
||||||
to_vector( b ) ) );
|
to_vector( b ) ) );
|
||||||
}
|
}
|
||||||
|
|
||||||
VImage
|
VImage
|
||||||
operator<<( VImage a, std::vector<double> b )
|
operator<<( const VImage a, const std::vector<double> b )
|
||||||
{
|
{
|
||||||
return( a.boolean_const( VIPS_OPERATION_BOOLEAN_LSHIFT,
|
return( a.boolean_const( VIPS_OPERATION_BOOLEAN_LSHIFT,
|
||||||
b ) );
|
b ) );
|
||||||
@@ -1370,26 +1370,26 @@ operator<<=( VImage &a, const double b )
|
|||||||
}
|
}
|
||||||
|
|
||||||
VImage &
|
VImage &
|
||||||
operator<<=( VImage &a, std::vector<double> b )
|
operator<<=( VImage &a, const std::vector<double> b )
|
||||||
{
|
{
|
||||||
return( a = a << b );
|
return( a = a << b );
|
||||||
}
|
}
|
||||||
|
|
||||||
VImage
|
VImage
|
||||||
operator>>( VImage a, VImage b )
|
operator>>( const VImage a, const VImage b )
|
||||||
{
|
{
|
||||||
return( a.boolean( b, VIPS_OPERATION_BOOLEAN_RSHIFT ) );
|
return( a.boolean( b, VIPS_OPERATION_BOOLEAN_RSHIFT ) );
|
||||||
}
|
}
|
||||||
|
|
||||||
VImage
|
VImage
|
||||||
operator>>( VImage a, double b )
|
operator>>( const VImage a, const double b )
|
||||||
{
|
{
|
||||||
return( a.boolean_const( VIPS_OPERATION_BOOLEAN_RSHIFT,
|
return( a.boolean_const( VIPS_OPERATION_BOOLEAN_RSHIFT,
|
||||||
to_vector( b ) ) );
|
to_vector( b ) ) );
|
||||||
}
|
}
|
||||||
|
|
||||||
VImage
|
VImage
|
||||||
operator>>( VImage a, std::vector<double> b )
|
operator>>( const VImage a, const std::vector<double> b )
|
||||||
{
|
{
|
||||||
return( a.boolean_const( VIPS_OPERATION_BOOLEAN_RSHIFT,
|
return( a.boolean_const( VIPS_OPERATION_BOOLEAN_RSHIFT,
|
||||||
b ) );
|
b ) );
|
||||||
@@ -1408,7 +1408,7 @@ operator>>=( VImage &a, const double b )
|
|||||||
}
|
}
|
||||||
|
|
||||||
VImage &
|
VImage &
|
||||||
operator>>=( VImage &a, std::vector<double> b )
|
operator>>=( VImage &a, const std::vector<double> b )
|
||||||
{
|
{
|
||||||
return( a = a << b );
|
return( a = a << b );
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -62,6 +62,12 @@ class MetadataWorker : public Nan::AsyncWorker {
|
|||||||
if (sharp::HasDensity(image)) {
|
if (sharp::HasDensity(image)) {
|
||||||
baton->density = sharp::GetDensity(image);
|
baton->density = sharp::GetDensity(image);
|
||||||
}
|
}
|
||||||
|
if (image.get_typeof("jpeg-chroma-subsample") == VIPS_TYPE_REF_STRING) {
|
||||||
|
baton->chromaSubsampling = image.get_string("jpeg-chroma-subsample");
|
||||||
|
}
|
||||||
|
if (image.get_typeof("interlaced") == G_TYPE_INT) {
|
||||||
|
baton->isProgressive = image.get_int("interlaced") == 1;
|
||||||
|
}
|
||||||
baton->hasProfile = sharp::HasProfile(image);
|
baton->hasProfile = sharp::HasProfile(image);
|
||||||
// Derived attributes
|
// Derived attributes
|
||||||
baton->hasAlpha = sharp::HasAlpha(image);
|
baton->hasAlpha = sharp::HasAlpha(image);
|
||||||
@@ -117,6 +123,9 @@ class MetadataWorker : public Nan::AsyncWorker {
|
|||||||
// Metadata Object
|
// Metadata Object
|
||||||
v8::Local<v8::Object> info = New<v8::Object>();
|
v8::Local<v8::Object> info = New<v8::Object>();
|
||||||
Set(info, New("format").ToLocalChecked(), New<v8::String>(baton->format).ToLocalChecked());
|
Set(info, New("format").ToLocalChecked(), New<v8::String>(baton->format).ToLocalChecked());
|
||||||
|
if (baton->input->bufferLength > 0) {
|
||||||
|
Set(info, New("size").ToLocalChecked(), New<v8::Uint32>(static_cast<uint32_t>(baton->input->bufferLength)));
|
||||||
|
}
|
||||||
Set(info, New("width").ToLocalChecked(), New<v8::Uint32>(baton->width));
|
Set(info, New("width").ToLocalChecked(), New<v8::Uint32>(baton->width));
|
||||||
Set(info, New("height").ToLocalChecked(), New<v8::Uint32>(baton->height));
|
Set(info, New("height").ToLocalChecked(), New<v8::Uint32>(baton->height));
|
||||||
Set(info, New("space").ToLocalChecked(), New<v8::String>(baton->space).ToLocalChecked());
|
Set(info, New("space").ToLocalChecked(), New<v8::String>(baton->space).ToLocalChecked());
|
||||||
@@ -125,6 +134,12 @@ class MetadataWorker : public Nan::AsyncWorker {
|
|||||||
if (baton->density > 0) {
|
if (baton->density > 0) {
|
||||||
Set(info, New("density").ToLocalChecked(), New<v8::Uint32>(baton->density));
|
Set(info, New("density").ToLocalChecked(), New<v8::Uint32>(baton->density));
|
||||||
}
|
}
|
||||||
|
if (!baton->chromaSubsampling.empty()) {
|
||||||
|
Set(info,
|
||||||
|
New("chromaSubsampling").ToLocalChecked(),
|
||||||
|
New<v8::String>(baton->chromaSubsampling).ToLocalChecked());
|
||||||
|
}
|
||||||
|
Set(info, New("isProgressive").ToLocalChecked(), New<v8::Boolean>(baton->isProgressive));
|
||||||
Set(info, New("hasProfile").ToLocalChecked(), New<v8::Boolean>(baton->hasProfile));
|
Set(info, New("hasProfile").ToLocalChecked(), New<v8::Boolean>(baton->hasProfile));
|
||||||
Set(info, New("hasAlpha").ToLocalChecked(), New<v8::Boolean>(baton->hasAlpha));
|
Set(info, New("hasAlpha").ToLocalChecked(), New<v8::Boolean>(baton->hasAlpha));
|
||||||
if (baton->orientation > 0) {
|
if (baton->orientation > 0) {
|
||||||
|
|||||||
@@ -31,6 +31,8 @@ struct MetadataBaton {
|
|||||||
int channels;
|
int channels;
|
||||||
std::string depth;
|
std::string depth;
|
||||||
int density;
|
int density;
|
||||||
|
std::string chromaSubsampling;
|
||||||
|
bool isProgressive;
|
||||||
bool hasProfile;
|
bool hasProfile;
|
||||||
bool hasAlpha;
|
bool hasAlpha;
|
||||||
int orientation;
|
int orientation;
|
||||||
@@ -50,6 +52,7 @@ struct MetadataBaton {
|
|||||||
height(0),
|
height(0),
|
||||||
channels(0),
|
channels(0),
|
||||||
density(0),
|
density(0),
|
||||||
|
isProgressive(false),
|
||||||
hasProfile(false),
|
hasProfile(false),
|
||||||
hasAlpha(false),
|
hasAlpha(false),
|
||||||
orientation(0),
|
orientation(0),
|
||||||
|
|||||||
@@ -78,7 +78,7 @@ namespace sharp {
|
|||||||
//
|
//
|
||||||
// References:
|
// References:
|
||||||
// - http://en.wikipedia.org/wiki/Alpha_compositing#Alpha_blending
|
// - http://en.wikipedia.org/wiki/Alpha_compositing#Alpha_blending
|
||||||
// - https://github.com/jcupitt/ruby-vips/issues/28#issuecomment-9014826
|
// - https://github.com/libvips/ruby-vips/issues/28#issuecomment-9014826
|
||||||
//
|
//
|
||||||
// out_a = src_a + dst_a * (1 - src_a)
|
// out_a = src_a + dst_a * (1 - src_a)
|
||||||
// ^^^^^^^^^^^
|
// ^^^^^^^^^^^
|
||||||
@@ -278,6 +278,25 @@ namespace sharp {
|
|||||||
return image.conv(kernel);
|
return image.conv(kernel);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Recomb with a Matrix of the given bands/channel size.
|
||||||
|
* Eg. RGB will be a 3x3 matrix.
|
||||||
|
*/
|
||||||
|
VImage Recomb(VImage image, std::unique_ptr<double[]> const &matrix) {
|
||||||
|
double *m = matrix.get();
|
||||||
|
return image
|
||||||
|
.colourspace(VIPS_INTERPRETATION_sRGB)
|
||||||
|
.recomb(image.bands() == 3
|
||||||
|
? VImage::new_from_memory(
|
||||||
|
m, 9 * sizeof(double), 3, 3, 1, VIPS_FORMAT_DOUBLE
|
||||||
|
)
|
||||||
|
: VImage::new_matrixv(4, 4,
|
||||||
|
m[0], m[1], m[2], 0.0,
|
||||||
|
m[3], m[4], m[5], 0.0,
|
||||||
|
m[6], m[7], m[8], 0.0,
|
||||||
|
0.0, 0.0, 0.0, 1.0));
|
||||||
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Sharpen flat and jagged areas. Use sigma of -1.0 for fast sharpen.
|
* Sharpen flat and jagged areas. Use sigma of -1.0 for fast sharpen.
|
||||||
*/
|
*/
|
||||||
@@ -324,55 +343,22 @@ namespace sharp {
|
|||||||
return image.boolean(imageR, boolean);
|
return image.boolean(imageR, boolean);
|
||||||
}
|
}
|
||||||
|
|
||||||
VImage Trim(VImage image, int const tolerance) {
|
/*
|
||||||
using sharp::MaximumImageAlpha;
|
Trim an image
|
||||||
// An equivalent of ImageMagick's -trim in C++ ... automatically remove
|
*/
|
||||||
// "boring" image edges.
|
VImage Trim(VImage image, double const threshold) {
|
||||||
|
// Top-left pixel provides the background colour
|
||||||
// We use .project to sum the rows and columns of a 0/255 mask image, the first
|
VImage background = image.extract_area(0, 0, 1, 1);
|
||||||
// non-zero row or column is the object edge. We make the mask image with an
|
if (HasAlpha(background)) {
|
||||||
// amount-different-from-background image plus a threshold.
|
background = background.flatten();
|
||||||
|
}
|
||||||
// find the value of the pixel at (0, 0) ... we will search for all pixels
|
int top, width, height;
|
||||||
// significantly different from this
|
int const left = image.find_trim(&top, &width, &height, VImage::option()
|
||||||
std::vector<double> background = image(0, 0);
|
->set("background", background(0, 0))
|
||||||
|
->set("threshold", threshold));
|
||||||
double const max = MaximumImageAlpha(image.interpretation());
|
if (width == 0 || height == 0) {
|
||||||
|
|
||||||
// we need to smooth the image, subtract the background from every pixel, take
|
|
||||||
// the absolute value of the difference, then threshold
|
|
||||||
VImage mask = (image.median(3) - background).abs() > (max * tolerance / 100);
|
|
||||||
|
|
||||||
// sum mask rows and columns, then search for the first non-zero sum in each
|
|
||||||
// direction
|
|
||||||
VImage rows;
|
|
||||||
VImage columns = mask.project(&rows);
|
|
||||||
|
|
||||||
VImage profileLeftV;
|
|
||||||
VImage profileLeftH = columns.profile(&profileLeftV);
|
|
||||||
|
|
||||||
VImage profileRightV;
|
|
||||||
VImage profileRightH = columns.fliphor().profile(&profileRightV);
|
|
||||||
|
|
||||||
VImage profileTopV;
|
|
||||||
VImage profileTopH = rows.profile(&profileTopV);
|
|
||||||
|
|
||||||
VImage profileBottomV;
|
|
||||||
VImage profileBottomH = rows.flipver().profile(&profileBottomV);
|
|
||||||
|
|
||||||
int left = static_cast<int>(floor(profileLeftV.min()));
|
|
||||||
int right = columns.width() - static_cast<int>(floor(profileRightV.min()));
|
|
||||||
int top = static_cast<int>(floor(profileTopH.min()));
|
|
||||||
int bottom = rows.height() - static_cast<int>(floor(profileBottomH.min()));
|
|
||||||
|
|
||||||
int width = right - left;
|
|
||||||
int height = bottom - top;
|
|
||||||
|
|
||||||
if (width <= 0 || height <= 0) {
|
|
||||||
throw VError("Unexpected error while trimming. Try to lower the tolerance");
|
throw VError("Unexpected error while trimming. Try to lower the tolerance");
|
||||||
}
|
}
|
||||||
|
|
||||||
// and now crop the original image
|
|
||||||
return image.extract_area(left, top, width, height);
|
return image.extract_area(left, top, width, height);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -100,13 +100,19 @@ namespace sharp {
|
|||||||
/*
|
/*
|
||||||
Trim an image
|
Trim an image
|
||||||
*/
|
*/
|
||||||
VImage Trim(VImage image, int const tolerance);
|
VImage Trim(VImage image, double const threshold);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Linear adjustment (a * in + b)
|
* Linear adjustment (a * in + b)
|
||||||
*/
|
*/
|
||||||
VImage Linear(VImage image, double const a, double const b);
|
VImage Linear(VImage image, double const a, double const b);
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Recomb with a Matrix of the given bands/channel size.
|
||||||
|
* Eg. RGB will be a 3x3 matrix.
|
||||||
|
*/
|
||||||
|
VImage Recomb(VImage image, std::unique_ptr<double[]> const &matrix);
|
||||||
|
|
||||||
} // namespace sharp
|
} // namespace sharp
|
||||||
|
|
||||||
#endif // SRC_OPERATIONS_H_
|
#endif // SRC_OPERATIONS_H_
|
||||||
|
|||||||
151
src/pipeline.cc
@@ -100,8 +100,10 @@ class PipelineWorker : public Nan::AsyncWorker {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Trim
|
// Trim
|
||||||
if (baton->trimTolerance != 0) {
|
if (baton->trimThreshold > 0.0) {
|
||||||
image = sharp::Trim(image, baton->trimTolerance);
|
image = sharp::Trim(image, baton->trimThreshold);
|
||||||
|
baton->trimOffsetLeft = image.xoffset();
|
||||||
|
baton->trimOffsetTop = image.yoffset();
|
||||||
}
|
}
|
||||||
|
|
||||||
// Pre extraction
|
// Pre extraction
|
||||||
@@ -233,7 +235,7 @@ class PipelineWorker : public Nan::AsyncWorker {
|
|||||||
if (
|
if (
|
||||||
xshrink == yshrink && xshrink >= 2 * shrink_on_load_factor &&
|
xshrink == yshrink && xshrink >= 2 * shrink_on_load_factor &&
|
||||||
(inputImageType == ImageType::JPEG || inputImageType == ImageType::WEBP) &&
|
(inputImageType == ImageType::JPEG || inputImageType == ImageType::WEBP) &&
|
||||||
baton->gamma == 0 && baton->topOffsetPre == -1 && baton->trimTolerance == 0
|
baton->gamma == 0 && baton->topOffsetPre == -1 && baton->trimThreshold == 0.0
|
||||||
) {
|
) {
|
||||||
if (xshrink >= 8 * shrink_on_load_factor) {
|
if (xshrink >= 8 * shrink_on_load_factor) {
|
||||||
xfactor = xfactor / 8;
|
xfactor = xfactor / 8;
|
||||||
@@ -318,9 +320,9 @@ class PipelineWorker : public Nan::AsyncWorker {
|
|||||||
double const multiplier = sharp::Is16Bit(image.interpretation()) ? 256.0 : 1.0;
|
double const multiplier = sharp::Is16Bit(image.interpretation()) ? 256.0 : 1.0;
|
||||||
// Background colour
|
// Background colour
|
||||||
std::vector<double> background {
|
std::vector<double> background {
|
||||||
baton->background[0] * multiplier,
|
baton->flattenBackground[0] * multiplier,
|
||||||
baton->background[1] * multiplier,
|
baton->flattenBackground[1] * multiplier,
|
||||||
baton->background[2] * multiplier
|
baton->flattenBackground[2] * multiplier
|
||||||
};
|
};
|
||||||
image = image.flatten(VImage::option()
|
image = image.flatten(VImage::option()
|
||||||
->set("background", background));
|
->set("background", background));
|
||||||
@@ -379,7 +381,7 @@ class PipelineWorker : public Nan::AsyncWorker {
|
|||||||
vips_enum_from_nick(nullptr, VIPS_TYPE_KERNEL, baton->kernel.data()));
|
vips_enum_from_nick(nullptr, VIPS_TYPE_KERNEL, baton->kernel.data()));
|
||||||
if (
|
if (
|
||||||
kernel != VIPS_KERNEL_NEAREST && kernel != VIPS_KERNEL_CUBIC && kernel != VIPS_KERNEL_LANCZOS2 &&
|
kernel != VIPS_KERNEL_NEAREST && kernel != VIPS_KERNEL_CUBIC && kernel != VIPS_KERNEL_LANCZOS2 &&
|
||||||
kernel != VIPS_KERNEL_LANCZOS3
|
kernel != VIPS_KERNEL_LANCZOS3 && kernel != VIPS_KERNEL_MITCHELL
|
||||||
) {
|
) {
|
||||||
throw vips::VError("Unknown kernel");
|
throw vips::VError("Unknown kernel");
|
||||||
}
|
}
|
||||||
@@ -421,35 +423,8 @@ class PipelineWorker : public Nan::AsyncWorker {
|
|||||||
// Crop/embed
|
// Crop/embed
|
||||||
if (image.width() != baton->width || image.height() != baton->height) {
|
if (image.width() != baton->width || image.height() != baton->height) {
|
||||||
if (baton->canvas == Canvas::EMBED) {
|
if (baton->canvas == Canvas::EMBED) {
|
||||||
// Scale up 8-bit values to match 16-bit input image
|
|
||||||
double const multiplier = sharp::Is16Bit(image.interpretation()) ? 256.0 : 1.0;
|
|
||||||
// Create background colour
|
|
||||||
std::vector<double> background;
|
std::vector<double> background;
|
||||||
if (image.bands() > 2) {
|
std::tie(image, background) = sharp::ApplyAlpha(image, baton->resizeBackground);
|
||||||
background = {
|
|
||||||
multiplier * baton->background[0],
|
|
||||||
multiplier * baton->background[1],
|
|
||||||
multiplier * baton->background[2]
|
|
||||||
};
|
|
||||||
} else {
|
|
||||||
// Convert sRGB to greyscale
|
|
||||||
background = { multiplier * (
|
|
||||||
0.2126 * baton->background[0] +
|
|
||||||
0.7152 * baton->background[1] +
|
|
||||||
0.0722 * baton->background[2])
|
|
||||||
};
|
|
||||||
}
|
|
||||||
// Add alpha channel to background colour
|
|
||||||
if (baton->background[3] < 255.0 || HasAlpha(image)) {
|
|
||||||
background.push_back(baton->background[3] * multiplier);
|
|
||||||
}
|
|
||||||
// Ensure background colour uses correct colourspace
|
|
||||||
background = sharp::GetRgbaAsColourspace(background, image.interpretation());
|
|
||||||
// Add non-transparent alpha channel, if required
|
|
||||||
if (baton->background[3] < 255.0 && !HasAlpha(image)) {
|
|
||||||
image = image.bandjoin(
|
|
||||||
VImage::new_matrix(image.width(), image.height()).new_from_image(255 * multiplier));
|
|
||||||
}
|
|
||||||
|
|
||||||
// Embed
|
// Embed
|
||||||
|
|
||||||
@@ -463,7 +438,7 @@ class PipelineWorker : public Nan::AsyncWorker {
|
|||||||
int width = std::max(image.width(), baton->width);
|
int width = std::max(image.width(), baton->width);
|
||||||
int height = std::max(image.height(), baton->height);
|
int height = std::max(image.height(), baton->height);
|
||||||
std::tie(left, top) = sharp::CalculateEmbedPosition(
|
std::tie(left, top) = sharp::CalculateEmbedPosition(
|
||||||
image.width(), image.height(), baton->width, baton->height, baton->embed);
|
image.width(), image.height(), baton->width, baton->height, baton->position);
|
||||||
|
|
||||||
image = image.embed(left, top, width, height, VImage::option()
|
image = image.embed(left, top, width, height, VImage::option()
|
||||||
->set("extend", VIPS_EXTEND_BACKGROUND)
|
->set("extend", VIPS_EXTEND_BACKGROUND)
|
||||||
@@ -474,12 +449,12 @@ class PipelineWorker : public Nan::AsyncWorker {
|
|||||||
(image.width() > baton->width || image.height() > baton->height)
|
(image.width() > baton->width || image.height() > baton->height)
|
||||||
) {
|
) {
|
||||||
// Crop/max/min
|
// Crop/max/min
|
||||||
if (baton->crop < 9) {
|
if (baton->position < 9) {
|
||||||
// Gravity-based crop
|
// Gravity-based crop
|
||||||
int left;
|
int left;
|
||||||
int top;
|
int top;
|
||||||
std::tie(left, top) = sharp::CalculateCrop(
|
std::tie(left, top) = sharp::CalculateCrop(
|
||||||
image.width(), image.height(), baton->width, baton->height, baton->crop);
|
image.width(), image.height(), baton->width, baton->height, baton->position);
|
||||||
int width = std::min(image.width(), baton->width);
|
int width = std::min(image.width(), baton->width);
|
||||||
int height = std::min(image.height(), baton->height);
|
int height = std::min(image.height(), baton->height);
|
||||||
image = image.extract_area(left, top, width, height);
|
image = image.extract_area(left, top, width, height);
|
||||||
@@ -495,7 +470,7 @@ class PipelineWorker : public Nan::AsyncWorker {
|
|||||||
->set("access", baton->accessMethod)
|
->set("access", baton->accessMethod)
|
||||||
->set("threaded", TRUE));
|
->set("threaded", TRUE));
|
||||||
image = image.smartcrop(baton->width, baton->height, VImage::option()
|
image = image.smartcrop(baton->width, baton->height, VImage::option()
|
||||||
->set("interesting", baton->crop == 16 ? VIPS_INTERESTING_ENTROPY : VIPS_INTERESTING_ATTENTION));
|
->set("interesting", baton->position == 16 ? VIPS_INTERESTING_ENTROPY : VIPS_INTERESTING_ATTENTION));
|
||||||
baton->hasCropOffset = true;
|
baton->hasCropOffset = true;
|
||||||
baton->cropOffsetLeft = static_cast<int>(image.xoffset());
|
baton->cropOffsetLeft = static_cast<int>(image.xoffset());
|
||||||
baton->cropOffsetTop = static_cast<int>(image.yoffset());
|
baton->cropOffsetTop = static_cast<int>(image.yoffset());
|
||||||
@@ -503,6 +478,13 @@ class PipelineWorker : public Nan::AsyncWorker {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Rotate by degree
|
||||||
|
if (baton->rotationAngle != 0.0) {
|
||||||
|
std::vector<double> background;
|
||||||
|
std::tie(image, background) = sharp::ApplyAlpha(image, baton->rotationBackground);
|
||||||
|
image = image.rotate(baton->rotationAngle, VImage::option()->set("background", background));
|
||||||
|
}
|
||||||
|
|
||||||
// Post extraction
|
// Post extraction
|
||||||
if (baton->topOffsetPost != -1) {
|
if (baton->topOffsetPost != -1) {
|
||||||
image = image.extract_area(
|
image = image.extract_area(
|
||||||
@@ -511,35 +493,9 @@ class PipelineWorker : public Nan::AsyncWorker {
|
|||||||
|
|
||||||
// Extend edges
|
// Extend edges
|
||||||
if (baton->extendTop > 0 || baton->extendBottom > 0 || baton->extendLeft > 0 || baton->extendRight > 0) {
|
if (baton->extendTop > 0 || baton->extendBottom > 0 || baton->extendLeft > 0 || baton->extendRight > 0) {
|
||||||
// Scale up 8-bit values to match 16-bit input image
|
|
||||||
double const multiplier = sharp::Is16Bit(image.interpretation()) ? 256.0 : 1.0;
|
|
||||||
// Create background colour
|
|
||||||
std::vector<double> background;
|
std::vector<double> background;
|
||||||
if (image.bands() > 2) {
|
std::tie(image, background) = sharp::ApplyAlpha(image, baton->extendBackground);
|
||||||
background = {
|
|
||||||
multiplier * baton->background[0],
|
|
||||||
multiplier * baton->background[1],
|
|
||||||
multiplier * baton->background[2]
|
|
||||||
};
|
|
||||||
} else {
|
|
||||||
// Convert sRGB to greyscale
|
|
||||||
background = { multiplier * (
|
|
||||||
0.2126 * baton->background[0] +
|
|
||||||
0.7152 * baton->background[1] +
|
|
||||||
0.0722 * baton->background[2])
|
|
||||||
};
|
|
||||||
}
|
|
||||||
// Add alpha channel to background colour
|
|
||||||
if (baton->background[3] < 255.0 || HasAlpha(image)) {
|
|
||||||
background.push_back(baton->background[3] * multiplier);
|
|
||||||
}
|
|
||||||
// Ensure background colour uses correct colourspace
|
|
||||||
background = sharp::GetRgbaAsColourspace(background, image.interpretation());
|
|
||||||
// Add non-transparent alpha channel, if required
|
|
||||||
if (baton->background[3] < 255.0 && !HasAlpha(image)) {
|
|
||||||
image = image.bandjoin(
|
|
||||||
VImage::new_matrix(image.width(), image.height()).new_from_image(255 * multiplier));
|
|
||||||
}
|
|
||||||
// Embed
|
// Embed
|
||||||
baton->width = image.width() + baton->extendLeft + baton->extendRight;
|
baton->width = image.width() + baton->extendLeft + baton->extendRight;
|
||||||
baton->height = image.height() + baton->extendTop + baton->extendBottom;
|
baton->height = image.height() + baton->extendTop + baton->extendBottom;
|
||||||
@@ -569,6 +525,11 @@ class PipelineWorker : public Nan::AsyncWorker {
|
|||||||
baton->convKernel);
|
baton->convKernel);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Recomb
|
||||||
|
if (baton->recombMatrix != NULL) {
|
||||||
|
image = sharp::Recomb(image, baton->recombMatrix);
|
||||||
|
}
|
||||||
|
|
||||||
// Sharpen
|
// Sharpen
|
||||||
if (shouldSharpen) {
|
if (shouldSharpen) {
|
||||||
image = sharp::Sharpen(image, baton->sharpenSigma, baton->sharpenFlat, baton->sharpenJagged);
|
image = sharp::Sharpen(image, baton->sharpenSigma, baton->sharpenFlat, baton->sharpenJagged);
|
||||||
@@ -656,8 +617,8 @@ class PipelineWorker : public Nan::AsyncWorker {
|
|||||||
baton->premultiplied = shouldPremultiplyAlpha;
|
baton->premultiplied = shouldPremultiplyAlpha;
|
||||||
|
|
||||||
// Gamma decoding (brighten)
|
// Gamma decoding (brighten)
|
||||||
if (baton->gamma >= 1 && baton->gamma <= 3) {
|
if (baton->gammaOut >= 1 && baton->gammaOut <= 3) {
|
||||||
image = sharp::Gamma(image, baton->gamma);
|
image = sharp::Gamma(image, baton->gammaOut);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Linear adjustment (a * in + b)
|
// Linear adjustment (a * in + b)
|
||||||
@@ -802,6 +763,10 @@ class PipelineWorker : public Nan::AsyncWorker {
|
|||||||
->set("squash", baton->tiffSquash)
|
->set("squash", baton->tiffSquash)
|
||||||
->set("compression", baton->tiffCompression)
|
->set("compression", baton->tiffCompression)
|
||||||
->set("predictor", baton->tiffPredictor)
|
->set("predictor", baton->tiffPredictor)
|
||||||
|
->set("pyramid", baton->tiffPyramid)
|
||||||
|
->set("tile", baton->tiffTile)
|
||||||
|
->set("tile_height", baton->tiffTileHeight)
|
||||||
|
->set("tile_width", baton->tiffTileWidth)
|
||||||
->set("xres", baton->tiffXres)
|
->set("xres", baton->tiffXres)
|
||||||
->set("yres", baton->tiffYres)));
|
->set("yres", baton->tiffYres)));
|
||||||
baton->bufferOut = static_cast<char*>(area->data);
|
baton->bufferOut = static_cast<char*>(area->data);
|
||||||
@@ -815,6 +780,7 @@ class PipelineWorker : public Nan::AsyncWorker {
|
|||||||
if (baton->greyscale || image.interpretation() == VIPS_INTERPRETATION_B_W) {
|
if (baton->greyscale || image.interpretation() == VIPS_INTERPRETATION_B_W) {
|
||||||
// Extract first band for greyscale image
|
// Extract first band for greyscale image
|
||||||
image = image[0];
|
image = image[0];
|
||||||
|
baton->channels = 1;
|
||||||
}
|
}
|
||||||
if (image.format() != VIPS_FORMAT_UCHAR) {
|
if (image.format() != VIPS_FORMAT_UCHAR) {
|
||||||
// Cast pixels to uint8 (unsigned char)
|
// Cast pixels to uint8 (unsigned char)
|
||||||
@@ -894,16 +860,16 @@ class PipelineWorker : public Nan::AsyncWorker {
|
|||||||
if (baton->tiffCompression == VIPS_FOREIGN_TIFF_COMPRESSION_JPEG) {
|
if (baton->tiffCompression == VIPS_FOREIGN_TIFF_COMPRESSION_JPEG) {
|
||||||
sharp::AssertImageTypeDimensions(image, ImageType::JPEG);
|
sharp::AssertImageTypeDimensions(image, ImageType::JPEG);
|
||||||
}
|
}
|
||||||
// Cast pixel values to float, if required
|
|
||||||
if (baton->tiffPredictor == VIPS_FOREIGN_TIFF_PREDICTOR_FLOAT) {
|
|
||||||
image = image.cast(VIPS_FORMAT_FLOAT);
|
|
||||||
}
|
|
||||||
image.tiffsave(const_cast<char*>(baton->fileOut.data()), VImage::option()
|
image.tiffsave(const_cast<char*>(baton->fileOut.data()), VImage::option()
|
||||||
->set("strip", !baton->withMetadata)
|
->set("strip", !baton->withMetadata)
|
||||||
->set("Q", baton->tiffQuality)
|
->set("Q", baton->tiffQuality)
|
||||||
->set("squash", baton->tiffSquash)
|
->set("squash", baton->tiffSquash)
|
||||||
->set("compression", baton->tiffCompression)
|
->set("compression", baton->tiffCompression)
|
||||||
->set("predictor", baton->tiffPredictor)
|
->set("predictor", baton->tiffPredictor)
|
||||||
|
->set("pyramid", baton->tiffPyramid)
|
||||||
|
->set("tile", baton->tiffTile)
|
||||||
|
->set("tile_height", baton->tiffTileHeight)
|
||||||
|
->set("tile_width", baton->tiffTileWidth)
|
||||||
->set("xres", baton->tiffXres)
|
->set("xres", baton->tiffXres)
|
||||||
->set("yres", baton->tiffYres));
|
->set("yres", baton->tiffYres));
|
||||||
baton->formatOut = "tiff";
|
baton->formatOut = "tiff";
|
||||||
@@ -1016,6 +982,12 @@ class PipelineWorker : public Nan::AsyncWorker {
|
|||||||
Set(info, New("cropOffsetTop").ToLocalChecked(),
|
Set(info, New("cropOffsetTop").ToLocalChecked(),
|
||||||
New<v8::Int32>(static_cast<int32_t>(baton->cropOffsetTop)));
|
New<v8::Int32>(static_cast<int32_t>(baton->cropOffsetTop)));
|
||||||
}
|
}
|
||||||
|
if (baton->trimThreshold > 0.0) {
|
||||||
|
Set(info, New("trimOffsetLeft").ToLocalChecked(),
|
||||||
|
New<v8::Int32>(static_cast<int32_t>(baton->trimOffsetLeft)));
|
||||||
|
Set(info, New("trimOffsetTop").ToLocalChecked(),
|
||||||
|
New<v8::Int32>(static_cast<int32_t>(baton->trimOffsetTop)));
|
||||||
|
}
|
||||||
|
|
||||||
if (baton->bufferOutLength > 0) {
|
if (baton->bufferOutLength > 0) {
|
||||||
// Pass ownership of output data to Buffer instance
|
// Pass ownership of output data to Buffer instance
|
||||||
@@ -1147,6 +1119,7 @@ NAN_METHOD(pipeline) {
|
|||||||
using sharp::AttrTo;
|
using sharp::AttrTo;
|
||||||
using sharp::AttrAs;
|
using sharp::AttrAs;
|
||||||
using sharp::AttrAsStr;
|
using sharp::AttrAsStr;
|
||||||
|
using sharp::AttrAsRgba;
|
||||||
using sharp::CreateInputDescriptor;
|
using sharp::CreateInputDescriptor;
|
||||||
|
|
||||||
// Input Buffers must not undergo GC compaction during processing
|
// Input Buffers must not undergo GC compaction during processing
|
||||||
@@ -1190,11 +1163,6 @@ NAN_METHOD(pipeline) {
|
|||||||
} else if (canvas == "ignore_aspect") {
|
} else if (canvas == "ignore_aspect") {
|
||||||
baton->canvas = Canvas::IGNORE_ASPECT;
|
baton->canvas = Canvas::IGNORE_ASPECT;
|
||||||
}
|
}
|
||||||
// Background colour
|
|
||||||
v8::Local<v8::Object> background = AttrAs<v8::Object>(options, "background");
|
|
||||||
for (unsigned int i = 0; i < 4; i++) {
|
|
||||||
baton->background[i] = AttrTo<double>(background, i);
|
|
||||||
}
|
|
||||||
// Tint chroma
|
// Tint chroma
|
||||||
baton->tintA = AttrTo<double>(options, "tintA");
|
baton->tintA = AttrTo<double>(options, "tintA");
|
||||||
baton->tintB = AttrTo<double>(options, "tintB");
|
baton->tintB = AttrTo<double>(options, "tintB");
|
||||||
@@ -1209,8 +1177,8 @@ NAN_METHOD(pipeline) {
|
|||||||
}
|
}
|
||||||
// Resize options
|
// Resize options
|
||||||
baton->withoutEnlargement = AttrTo<bool>(options, "withoutEnlargement");
|
baton->withoutEnlargement = AttrTo<bool>(options, "withoutEnlargement");
|
||||||
baton->crop = AttrTo<int32_t>(options, "crop");
|
baton->position = AttrTo<int32_t>(options, "position");
|
||||||
baton->embed = AttrTo<int32_t>(options, "embed");
|
baton->resizeBackground = AttrAsRgba(options, "resizeBackground");
|
||||||
baton->kernel = AttrAsStr(options, "kernel");
|
baton->kernel = AttrAsStr(options, "kernel");
|
||||||
baton->fastShrinkOnLoad = AttrTo<bool>(options, "fastShrinkOnLoad");
|
baton->fastShrinkOnLoad = AttrTo<bool>(options, "fastShrinkOnLoad");
|
||||||
// Join Channel Options
|
// Join Channel Options
|
||||||
@@ -1228,6 +1196,7 @@ NAN_METHOD(pipeline) {
|
|||||||
}
|
}
|
||||||
// Operators
|
// Operators
|
||||||
baton->flatten = AttrTo<bool>(options, "flatten");
|
baton->flatten = AttrTo<bool>(options, "flatten");
|
||||||
|
baton->flattenBackground = AttrAsRgba(options, "flattenBackground");
|
||||||
baton->negate = AttrTo<bool>(options, "negate");
|
baton->negate = AttrTo<bool>(options, "negate");
|
||||||
baton->blurSigma = AttrTo<double>(options, "blurSigma");
|
baton->blurSigma = AttrTo<double>(options, "blurSigma");
|
||||||
baton->medianSize = AttrTo<uint32_t>(options, "medianSize");
|
baton->medianSize = AttrTo<uint32_t>(options, "medianSize");
|
||||||
@@ -1236,14 +1205,17 @@ NAN_METHOD(pipeline) {
|
|||||||
baton->sharpenJagged = AttrTo<double>(options, "sharpenJagged");
|
baton->sharpenJagged = AttrTo<double>(options, "sharpenJagged");
|
||||||
baton->threshold = AttrTo<int32_t>(options, "threshold");
|
baton->threshold = AttrTo<int32_t>(options, "threshold");
|
||||||
baton->thresholdGrayscale = AttrTo<bool>(options, "thresholdGrayscale");
|
baton->thresholdGrayscale = AttrTo<bool>(options, "thresholdGrayscale");
|
||||||
baton->trimTolerance = AttrTo<int32_t>(options, "trimTolerance");
|
baton->trimThreshold = AttrTo<double>(options, "trimThreshold");
|
||||||
baton->gamma = AttrTo<double>(options, "gamma");
|
baton->gamma = AttrTo<double>(options, "gamma");
|
||||||
|
baton->gammaOut = AttrTo<double>(options, "gammaOut");
|
||||||
baton->linearA = AttrTo<double>(options, "linearA");
|
baton->linearA = AttrTo<double>(options, "linearA");
|
||||||
baton->linearB = AttrTo<double>(options, "linearB");
|
baton->linearB = AttrTo<double>(options, "linearB");
|
||||||
baton->greyscale = AttrTo<bool>(options, "greyscale");
|
baton->greyscale = AttrTo<bool>(options, "greyscale");
|
||||||
baton->normalise = AttrTo<bool>(options, "normalise");
|
baton->normalise = AttrTo<bool>(options, "normalise");
|
||||||
baton->useExifOrientation = AttrTo<bool>(options, "useExifOrientation");
|
baton->useExifOrientation = AttrTo<bool>(options, "useExifOrientation");
|
||||||
baton->angle = AttrTo<int32_t>(options, "angle");
|
baton->angle = AttrTo<int32_t>(options, "angle");
|
||||||
|
baton->rotationAngle = AttrTo<double>(options, "rotationAngle");
|
||||||
|
baton->rotationBackground = AttrAsRgba(options, "rotationBackground");
|
||||||
baton->rotateBeforePreExtract = AttrTo<bool>(options, "rotateBeforePreExtract");
|
baton->rotateBeforePreExtract = AttrTo<bool>(options, "rotateBeforePreExtract");
|
||||||
baton->flip = AttrTo<bool>(options, "flip");
|
baton->flip = AttrTo<bool>(options, "flip");
|
||||||
baton->flop = AttrTo<bool>(options, "flop");
|
baton->flop = AttrTo<bool>(options, "flop");
|
||||||
@@ -1251,7 +1223,9 @@ NAN_METHOD(pipeline) {
|
|||||||
baton->extendBottom = AttrTo<int32_t>(options, "extendBottom");
|
baton->extendBottom = AttrTo<int32_t>(options, "extendBottom");
|
||||||
baton->extendLeft = AttrTo<int32_t>(options, "extendLeft");
|
baton->extendLeft = AttrTo<int32_t>(options, "extendLeft");
|
||||||
baton->extendRight = AttrTo<int32_t>(options, "extendRight");
|
baton->extendRight = AttrTo<int32_t>(options, "extendRight");
|
||||||
|
baton->extendBackground = AttrAsRgba(options, "extendBackground");
|
||||||
baton->extractChannel = AttrTo<int32_t>(options, "extractChannel");
|
baton->extractChannel = AttrTo<int32_t>(options, "extractChannel");
|
||||||
|
|
||||||
baton->removeAlpha = AttrTo<bool>(options, "removeAlpha");
|
baton->removeAlpha = AttrTo<bool>(options, "removeAlpha");
|
||||||
if (HasAttr(options, "boolean")) {
|
if (HasAttr(options, "boolean")) {
|
||||||
baton->boolean = CreateInputDescriptor(AttrAs<v8::Object>(options, "boolean"), buffersToPersist);
|
baton->boolean = CreateInputDescriptor(AttrAs<v8::Object>(options, "boolean"), buffersToPersist);
|
||||||
@@ -1273,6 +1247,13 @@ NAN_METHOD(pipeline) {
|
|||||||
baton->convKernel[i] = AttrTo<double>(kdata, i);
|
baton->convKernel[i] = AttrTo<double>(kdata, i);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
if (HasAttr(options, "recombMatrix")) {
|
||||||
|
baton->recombMatrix = std::unique_ptr<double[]>(new double[9]);
|
||||||
|
v8::Local<v8::Array> recombMatrix = AttrAs<v8::Array>(options, "recombMatrix");
|
||||||
|
for (unsigned int i = 0; i < 9; i++) {
|
||||||
|
baton->recombMatrix[i] = AttrTo<double>(recombMatrix, i);
|
||||||
|
}
|
||||||
|
}
|
||||||
baton->colourspace = sharp::GetInterpretation(AttrAsStr(options, "colourspace"));
|
baton->colourspace = sharp::GetInterpretation(AttrAsStr(options, "colourspace"));
|
||||||
if (baton->colourspace == VIPS_INTERPRETATION_ERROR) {
|
if (baton->colourspace == VIPS_INTERPRETATION_ERROR) {
|
||||||
baton->colourspace = VIPS_INTERPRETATION_sRGB;
|
baton->colourspace = VIPS_INTERPRETATION_sRGB;
|
||||||
@@ -1299,7 +1280,11 @@ NAN_METHOD(pipeline) {
|
|||||||
baton->webpLossless = AttrTo<bool>(options, "webpLossless");
|
baton->webpLossless = AttrTo<bool>(options, "webpLossless");
|
||||||
baton->webpNearLossless = AttrTo<bool>(options, "webpNearLossless");
|
baton->webpNearLossless = AttrTo<bool>(options, "webpNearLossless");
|
||||||
baton->tiffQuality = AttrTo<uint32_t>(options, "tiffQuality");
|
baton->tiffQuality = AttrTo<uint32_t>(options, "tiffQuality");
|
||||||
|
baton->tiffPyramid = AttrTo<bool>(options, "tiffPyramid");
|
||||||
baton->tiffSquash = AttrTo<bool>(options, "tiffSquash");
|
baton->tiffSquash = AttrTo<bool>(options, "tiffSquash");
|
||||||
|
baton->tiffTile = AttrTo<bool>(options, "tiffTile");
|
||||||
|
baton->tiffTileWidth = AttrTo<uint32_t>(options, "tiffTileWidth");
|
||||||
|
baton->tiffTileHeight = AttrTo<uint32_t>(options, "tiffTileHeight");
|
||||||
baton->tiffXres = AttrTo<double>(options, "tiffXres");
|
baton->tiffXres = AttrTo<double>(options, "tiffXres");
|
||||||
baton->tiffYres = AttrTo<double>(options, "tiffYres");
|
baton->tiffYres = AttrTo<double>(options, "tiffYres");
|
||||||
// tiff compression options
|
// tiff compression options
|
||||||
@@ -1342,8 +1327,8 @@ NAN_METHOD(pipeline) {
|
|||||||
}
|
}
|
||||||
// Force random access for certain operations
|
// Force random access for certain operations
|
||||||
if (baton->accessMethod == VIPS_ACCESS_SEQUENTIAL && (
|
if (baton->accessMethod == VIPS_ACCESS_SEQUENTIAL && (
|
||||||
baton->trimTolerance != 0 || baton->normalise ||
|
baton->trimThreshold > 0.0 || baton->normalise ||
|
||||||
baton->crop == 16 || baton->crop == 17)) {
|
baton->position == 16 || baton->position == 17)) {
|
||||||
baton->accessMethod = VIPS_ACCESS_RANDOM;
|
baton->accessMethod = VIPS_ACCESS_RANDOM;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -61,18 +61,18 @@ struct PipelineBaton {
|
|||||||
int height;
|
int height;
|
||||||
int channels;
|
int channels;
|
||||||
Canvas canvas;
|
Canvas canvas;
|
||||||
int crop;
|
int position;
|
||||||
int embed;
|
std::vector<double> resizeBackground;
|
||||||
bool hasCropOffset;
|
bool hasCropOffset;
|
||||||
int cropOffsetLeft;
|
int cropOffsetLeft;
|
||||||
int cropOffsetTop;
|
int cropOffsetTop;
|
||||||
bool premultiplied;
|
bool premultiplied;
|
||||||
std::string kernel;
|
std::string kernel;
|
||||||
bool fastShrinkOnLoad;
|
bool fastShrinkOnLoad;
|
||||||
double background[4];
|
|
||||||
double tintA;
|
double tintA;
|
||||||
double tintB;
|
double tintB;
|
||||||
bool flatten;
|
bool flatten;
|
||||||
|
std::vector<double> flattenBackground;
|
||||||
bool negate;
|
bool negate;
|
||||||
double blurSigma;
|
double blurSigma;
|
||||||
int medianSize;
|
int medianSize;
|
||||||
@@ -81,14 +81,19 @@ struct PipelineBaton {
|
|||||||
double sharpenJagged;
|
double sharpenJagged;
|
||||||
int threshold;
|
int threshold;
|
||||||
bool thresholdGrayscale;
|
bool thresholdGrayscale;
|
||||||
int trimTolerance;
|
double trimThreshold;
|
||||||
|
int trimOffsetLeft;
|
||||||
|
int trimOffsetTop;
|
||||||
double linearA;
|
double linearA;
|
||||||
double linearB;
|
double linearB;
|
||||||
double gamma;
|
double gamma;
|
||||||
|
double gammaOut;
|
||||||
bool greyscale;
|
bool greyscale;
|
||||||
bool normalise;
|
bool normalise;
|
||||||
bool useExifOrientation;
|
bool useExifOrientation;
|
||||||
int angle;
|
int angle;
|
||||||
|
double rotationAngle;
|
||||||
|
std::vector<double> rotationBackground;
|
||||||
bool rotateBeforePreExtract;
|
bool rotateBeforePreExtract;
|
||||||
bool flip;
|
bool flip;
|
||||||
bool flop;
|
bool flop;
|
||||||
@@ -96,6 +101,7 @@ struct PipelineBaton {
|
|||||||
int extendBottom;
|
int extendBottom;
|
||||||
int extendLeft;
|
int extendLeft;
|
||||||
int extendRight;
|
int extendRight;
|
||||||
|
std::vector<double> extendBackground;
|
||||||
bool withoutEnlargement;
|
bool withoutEnlargement;
|
||||||
VipsAccess accessMethod;
|
VipsAccess accessMethod;
|
||||||
int jpegQuality;
|
int jpegQuality;
|
||||||
@@ -116,7 +122,11 @@ struct PipelineBaton {
|
|||||||
int tiffQuality;
|
int tiffQuality;
|
||||||
VipsForeignTiffCompression tiffCompression;
|
VipsForeignTiffCompression tiffCompression;
|
||||||
VipsForeignTiffPredictor tiffPredictor;
|
VipsForeignTiffPredictor tiffPredictor;
|
||||||
|
bool tiffPyramid;
|
||||||
bool tiffSquash;
|
bool tiffSquash;
|
||||||
|
bool tiffTile;
|
||||||
|
int tiffTileHeight;
|
||||||
|
int tiffTileWidth;
|
||||||
double tiffXres;
|
double tiffXres;
|
||||||
double tiffYres;
|
double tiffYres;
|
||||||
std::string err;
|
std::string err;
|
||||||
@@ -140,6 +150,7 @@ struct PipelineBaton {
|
|||||||
std::string tileFormat;
|
std::string tileFormat;
|
||||||
int tileAngle;
|
int tileAngle;
|
||||||
VipsForeignDzDepth tileDepth;
|
VipsForeignDzDepth tileDepth;
|
||||||
|
std::unique_ptr<double[]> recombMatrix;
|
||||||
|
|
||||||
PipelineBaton():
|
PipelineBaton():
|
||||||
input(nullptr),
|
input(nullptr),
|
||||||
@@ -155,8 +166,8 @@ struct PipelineBaton {
|
|||||||
topOffsetPost(-1),
|
topOffsetPost(-1),
|
||||||
channels(0),
|
channels(0),
|
||||||
canvas(Canvas::CROP),
|
canvas(Canvas::CROP),
|
||||||
crop(0),
|
position(0),
|
||||||
embed(0),
|
resizeBackground{ 0.0, 0.0, 0.0, 255.0 },
|
||||||
hasCropOffset(false),
|
hasCropOffset(false),
|
||||||
cropOffsetLeft(0),
|
cropOffsetLeft(0),
|
||||||
cropOffsetTop(0),
|
cropOffsetTop(0),
|
||||||
@@ -164,6 +175,7 @@ struct PipelineBaton {
|
|||||||
tintA(128.0),
|
tintA(128.0),
|
||||||
tintB(128.0),
|
tintB(128.0),
|
||||||
flatten(false),
|
flatten(false),
|
||||||
|
flattenBackground{ 0.0, 0.0, 0.0 },
|
||||||
negate(false),
|
negate(false),
|
||||||
blurSigma(0.0),
|
blurSigma(0.0),
|
||||||
medianSize(0),
|
medianSize(0),
|
||||||
@@ -172,7 +184,9 @@ struct PipelineBaton {
|
|||||||
sharpenJagged(2.0),
|
sharpenJagged(2.0),
|
||||||
threshold(0),
|
threshold(0),
|
||||||
thresholdGrayscale(true),
|
thresholdGrayscale(true),
|
||||||
trimTolerance(0),
|
trimThreshold(0.0),
|
||||||
|
trimOffsetLeft(0),
|
||||||
|
trimOffsetTop(0),
|
||||||
linearA(1.0),
|
linearA(1.0),
|
||||||
linearB(0.0),
|
linearB(0.0),
|
||||||
gamma(0.0),
|
gamma(0.0),
|
||||||
@@ -180,12 +194,15 @@ struct PipelineBaton {
|
|||||||
normalise(false),
|
normalise(false),
|
||||||
useExifOrientation(false),
|
useExifOrientation(false),
|
||||||
angle(0),
|
angle(0),
|
||||||
|
rotationAngle(0.0),
|
||||||
|
rotationBackground{ 0.0, 0.0, 0.0, 255.0 },
|
||||||
flip(false),
|
flip(false),
|
||||||
flop(false),
|
flop(false),
|
||||||
extendTop(0),
|
extendTop(0),
|
||||||
extendBottom(0),
|
extendBottom(0),
|
||||||
extendLeft(0),
|
extendLeft(0),
|
||||||
extendRight(0),
|
extendRight(0),
|
||||||
|
extendBackground{ 0.0, 0.0, 0.0, 255.0 },
|
||||||
withoutEnlargement(false),
|
withoutEnlargement(false),
|
||||||
jpegQuality(80),
|
jpegQuality(80),
|
||||||
jpegProgressive(false),
|
jpegProgressive(false),
|
||||||
@@ -202,7 +219,11 @@ struct PipelineBaton {
|
|||||||
tiffQuality(80),
|
tiffQuality(80),
|
||||||
tiffCompression(VIPS_FOREIGN_TIFF_COMPRESSION_JPEG),
|
tiffCompression(VIPS_FOREIGN_TIFF_COMPRESSION_JPEG),
|
||||||
tiffPredictor(VIPS_FOREIGN_TIFF_PREDICTOR_HORIZONTAL),
|
tiffPredictor(VIPS_FOREIGN_TIFF_PREDICTOR_HORIZONTAL),
|
||||||
|
tiffPyramid(false),
|
||||||
tiffSquash(false),
|
tiffSquash(false),
|
||||||
|
tiffTile(false),
|
||||||
|
tiffTileHeight(256),
|
||||||
|
tiffTileWidth(256),
|
||||||
tiffXres(1.0),
|
tiffXres(1.0),
|
||||||
tiffYres(1.0),
|
tiffYres(1.0),
|
||||||
withMetadata(false),
|
withMetadata(false),
|
||||||
@@ -222,12 +243,7 @@ struct PipelineBaton {
|
|||||||
tileContainer(VIPS_FOREIGN_DZ_CONTAINER_FS),
|
tileContainer(VIPS_FOREIGN_DZ_CONTAINER_FS),
|
||||||
tileLayout(VIPS_FOREIGN_DZ_LAYOUT_DZ),
|
tileLayout(VIPS_FOREIGN_DZ_LAYOUT_DZ),
|
||||||
tileAngle(0),
|
tileAngle(0),
|
||||||
tileDepth(VIPS_FOREIGN_DZ_DEPTH_LAST){
|
tileDepth(VIPS_FOREIGN_DZ_DEPTH_LAST) {}
|
||||||
background[0] = 0.0;
|
|
||||||
background[1] = 0.0;
|
|
||||||
background[2] = 0.0;
|
|
||||||
background[3] = 255.0;
|
|
||||||
}
|
|
||||||
};
|
};
|
||||||
|
|
||||||
#endif // SRC_PIPELINE_H_
|
#endif // SRC_PIPELINE_H_
|
||||||
|
|||||||
@@ -259,7 +259,7 @@ NAN_METHOD(_maxColourDistance) {
|
|||||||
}
|
}
|
||||||
// Calculate colour distance
|
// Calculate colour distance
|
||||||
maxColourDistance = image1.dE00(image2).max();
|
maxColourDistance = image1.dE00(image2).max();
|
||||||
} catch (VError err) {
|
} catch (VError const &err) {
|
||||||
return ThrowError(err.what());
|
return ThrowError(err.what());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -8,19 +8,17 @@
|
|||||||
"test": "node perf && node random && node parallel"
|
"test": "node perf && node random && node parallel"
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"async": "^2.6.0",
|
"async": "^2.6.1",
|
||||||
"benchmark": "^2.1.4",
|
"benchmark": "^2.1.4",
|
||||||
"gm": "^1.23.1",
|
"gm": "^1.23.1",
|
||||||
"imagemagick": "^0.1.3",
|
"imagemagick": "^0.1.3",
|
||||||
"imagemagick-native": "^1.9.3",
|
"imagemagick-native": "^1.9.3",
|
||||||
"images": "^3.0.1",
|
"jimp": "^0.5.3",
|
||||||
"jimp": "^0.2.28",
|
"mapnik": "^4.0.1",
|
||||||
"mapnik": "^3.6.2",
|
"semver": "^5.5.1"
|
||||||
"pajk-lwip": "^0.2.0",
|
|
||||||
"semver": "^5.4.1"
|
|
||||||
},
|
},
|
||||||
"license": "Apache-2.0",
|
"license": "Apache-2.0",
|
||||||
"engines": {
|
"engines": {
|
||||||
"node": ">=4"
|
"node": ">=6"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -12,24 +12,12 @@ const gm = require('gm');
|
|||||||
const imagemagick = require('imagemagick');
|
const imagemagick = require('imagemagick');
|
||||||
const mapnik = require('mapnik');
|
const mapnik = require('mapnik');
|
||||||
const jimp = require('jimp');
|
const jimp = require('jimp');
|
||||||
let images;
|
|
||||||
try {
|
|
||||||
images = require('images');
|
|
||||||
} catch (err) {
|
|
||||||
console.log('Excluding node-images');
|
|
||||||
}
|
|
||||||
let imagemagickNative;
|
let imagemagickNative;
|
||||||
try {
|
try {
|
||||||
imagemagickNative = require('imagemagick-native');
|
imagemagickNative = require('imagemagick-native');
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
console.log('Excluding imagemagick-native');
|
console.log('Excluding imagemagick-native');
|
||||||
}
|
}
|
||||||
let lwip;
|
|
||||||
try {
|
|
||||||
lwip = require('pajk-lwip');
|
|
||||||
} catch (err) {
|
|
||||||
console.log('Excluding lwip');
|
|
||||||
}
|
|
||||||
|
|
||||||
const fixtures = require('../fixtures');
|
const fixtures = require('../fixtures');
|
||||||
|
|
||||||
@@ -38,8 +26,6 @@ const height = 588;
|
|||||||
|
|
||||||
// Disable libvips cache to ensure tests are as fair as they can be
|
// Disable libvips cache to ensure tests are as fair as they can be
|
||||||
sharp.cache(false);
|
sharp.cache(false);
|
||||||
// Enable use of SIMD
|
|
||||||
sharp.simd(true);
|
|
||||||
|
|
||||||
async.series({
|
async.series({
|
||||||
'jpeg': function (callback) {
|
'jpeg': function (callback) {
|
||||||
@@ -54,7 +40,7 @@ async.series({
|
|||||||
throw err;
|
throw err;
|
||||||
} else {
|
} else {
|
||||||
image
|
image
|
||||||
.resize(width, height)
|
.resize(width, height, jimp.RESIZE_BICUBIC)
|
||||||
.quality(80)
|
.quality(80)
|
||||||
.getBuffer(jimp.MIME_JPEG, function (err) {
|
.getBuffer(jimp.MIME_JPEG, function (err) {
|
||||||
if (err) {
|
if (err) {
|
||||||
@@ -74,7 +60,7 @@ async.series({
|
|||||||
throw err;
|
throw err;
|
||||||
} else {
|
} else {
|
||||||
image
|
image
|
||||||
.resize(width, height)
|
.resize(width, height, jimp.RESIZE_BICUBIC)
|
||||||
.quality(80)
|
.quality(80)
|
||||||
.write(fixtures.outputJpg, function (err) {
|
.write(fixtures.outputJpg, function (err) {
|
||||||
if (err) {
|
if (err) {
|
||||||
@@ -87,51 +73,6 @@ async.series({
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
// lwip
|
|
||||||
if (typeof lwip !== 'undefined') {
|
|
||||||
jpegSuite.add('lwip-file-file', {
|
|
||||||
defer: true,
|
|
||||||
fn: function (deferred) {
|
|
||||||
lwip.open(fixtures.inputJpg, function (err, image) {
|
|
||||||
if (err) {
|
|
||||||
throw err;
|
|
||||||
}
|
|
||||||
image.resize(width, height, 'lanczos', function (err, image) {
|
|
||||||
if (err) {
|
|
||||||
throw err;
|
|
||||||
}
|
|
||||||
image.writeFile(fixtures.outputJpg, {quality: 80}, function (err) {
|
|
||||||
if (err) {
|
|
||||||
throw err;
|
|
||||||
}
|
|
||||||
deferred.resolve();
|
|
||||||
});
|
|
||||||
});
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}).add('lwip-buffer-buffer', {
|
|
||||||
defer: true,
|
|
||||||
fn: function (deferred) {
|
|
||||||
lwip.open(inputJpgBuffer, 'jpg', function (err, image) {
|
|
||||||
if (err) {
|
|
||||||
throw err;
|
|
||||||
}
|
|
||||||
image.resize(width, height, 'lanczos', function (err, image) {
|
|
||||||
if (err) {
|
|
||||||
throw err;
|
|
||||||
}
|
|
||||||
image.toBuffer('jpg', {quality: 80}, function (err, buffer) {
|
|
||||||
if (err) {
|
|
||||||
throw err;
|
|
||||||
}
|
|
||||||
assert.notStrictEqual(null, buffer);
|
|
||||||
deferred.resolve();
|
|
||||||
});
|
|
||||||
});
|
|
||||||
});
|
|
||||||
}
|
|
||||||
});
|
|
||||||
}
|
|
||||||
// mapnik
|
// mapnik
|
||||||
jpegSuite.add('mapnik-file-file', {
|
jpegSuite.add('mapnik-file-file', {
|
||||||
defer: true,
|
defer: true,
|
||||||
@@ -272,14 +213,6 @@ async.series({
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
// images
|
|
||||||
if (typeof images !== 'undefined') {
|
|
||||||
jpegSuite.add('images-file-file', function () {
|
|
||||||
images(fixtures.inputJpg)
|
|
||||||
.resize(width, height)
|
|
||||||
.save(fixtures.outputJpg, { quality: 80 });
|
|
||||||
});
|
|
||||||
}
|
|
||||||
// sharp
|
// sharp
|
||||||
jpegSuite.add('sharp-buffer-file', {
|
jpegSuite.add('sharp-buffer-file', {
|
||||||
defer: true,
|
defer: true,
|
||||||
@@ -569,8 +502,10 @@ async.series({
|
|||||||
defer: true,
|
defer: true,
|
||||||
fn: function (deferred) {
|
fn: function (deferred) {
|
||||||
sharp(inputJpgBuffer)
|
sharp(inputJpgBuffer)
|
||||||
.resize(width, height)
|
.resize(width, height, {
|
||||||
.crop(sharp.strategy.entropy)
|
fit: 'cover',
|
||||||
|
position: sharp.strategy.entropy
|
||||||
|
})
|
||||||
.toBuffer(function (err, buffer) {
|
.toBuffer(function (err, buffer) {
|
||||||
if (err) {
|
if (err) {
|
||||||
throw err;
|
throw err;
|
||||||
@@ -584,8 +519,10 @@ async.series({
|
|||||||
defer: true,
|
defer: true,
|
||||||
fn: function (deferred) {
|
fn: function (deferred) {
|
||||||
sharp(inputJpgBuffer)
|
sharp(inputJpgBuffer)
|
||||||
.resize(width, height)
|
.resize(width, height, {
|
||||||
.crop(sharp.strategy.attention)
|
fit: 'cover',
|
||||||
|
position: sharp.strategy.attention
|
||||||
|
})
|
||||||
.toBuffer(function (err, buffer) {
|
.toBuffer(function (err, buffer) {
|
||||||
if (err) {
|
if (err) {
|
||||||
throw err;
|
throw err;
|
||||||
@@ -696,31 +633,6 @@ async.series({
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
// lwip
|
|
||||||
if (typeof lwip !== 'undefined') {
|
|
||||||
pngSuite.add('lwip-buffer-buffer', {
|
|
||||||
defer: true,
|
|
||||||
fn: function (deferred) {
|
|
||||||
lwip.open(inputPngBuffer, 'png', function (err, image) {
|
|
||||||
if (err) {
|
|
||||||
throw err;
|
|
||||||
}
|
|
||||||
image.resize(width, height, 'lanczos', function (err, image) {
|
|
||||||
if (err) {
|
|
||||||
throw err;
|
|
||||||
}
|
|
||||||
image.toBuffer('png', function (err, buffer) {
|
|
||||||
if (err) {
|
|
||||||
throw err;
|
|
||||||
}
|
|
||||||
assert.notStrictEqual(null, buffer);
|
|
||||||
deferred.resolve();
|
|
||||||
});
|
|
||||||
});
|
|
||||||
});
|
|
||||||
}
|
|
||||||
});
|
|
||||||
}
|
|
||||||
// mapnik
|
// mapnik
|
||||||
pngSuite.add('mapnik-file-file', {
|
pngSuite.add('mapnik-file-file', {
|
||||||
defer: true,
|
defer: true,
|
||||||
@@ -833,14 +745,6 @@ async.series({
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
// images
|
|
||||||
if (typeof images !== 'undefined') {
|
|
||||||
pngSuite.add('images-file-file', function () {
|
|
||||||
images(fixtures.inputPng)
|
|
||||||
.resize(width, height)
|
|
||||||
.save(fixtures.outputPng);
|
|
||||||
});
|
|
||||||
}
|
|
||||||
// sharp
|
// sharp
|
||||||
pngSuite.add('sharp-buffer-file', {
|
pngSuite.add('sharp-buffer-file', {
|
||||||
defer: true,
|
defer: true,
|
||||||
|
|||||||
BIN
test/fixtures/expected/Landscape_1-recomb-saturation.jpg
vendored
Normal file
|
After Width: | Height: | Size: 82 KiB |
BIN
test/fixtures/expected/Landscape_1-recomb-sepia.jpg
vendored
Normal file
|
After Width: | Height: | Size: 77 KiB |
BIN
test/fixtures/expected/Landscape_1-recomb-sepia2.jpg
vendored
Normal file
|
After Width: | Height: | Size: 85 KiB |
BIN
test/fixtures/expected/alpha-recomb-sepia.png
vendored
Normal file
|
After Width: | Height: | Size: 209 KiB |
BIN
test/fixtures/expected/extract-lch.jpg
vendored
|
Before Width: | Height: | Size: 13 KiB After Width: | Height: | Size: 13 KiB |
BIN
test/fixtures/expected/fast-shrink-on-load-false.png
vendored
|
Before Width: | Height: | Size: 258 B After Width: | Height: | Size: 270 B |
BIN
test/fixtures/expected/fast-shrink-on-load-true.png
vendored
|
Before Width: | Height: | Size: 263 B After Width: | Height: | Size: 265 B |
BIN
test/fixtures/expected/gamma-in-2.2-out-3.0.jpg
vendored
Normal file
|
After Width: | Height: | Size: 1.8 KiB |
BIN
test/fixtures/expected/rotate-solid-bg.jpg
vendored
Normal file
|
After Width: | Height: | Size: 19 KiB |
BIN
test/fixtures/expected/rotate-transparent-bg.png
vendored
Normal file
|
After Width: | Height: | Size: 238 KiB |
BIN
test/fixtures/expected/svg14.4.png
vendored
Normal file
|
After Width: | Height: | Size: 340 B |
BIN
test/fixtures/expected/tint-blue.jpg
vendored
|
Before Width: | Height: | Size: 15 KiB After Width: | Height: | Size: 15 KiB |
BIN
test/fixtures/expected/tint-cmyk.jpg
vendored
|
Before Width: | Height: | Size: 26 KiB After Width: | Height: | Size: 26 KiB |
BIN
test/fixtures/expected/tint-green.jpg
vendored
|
Before Width: | Height: | Size: 12 KiB After Width: | Height: | Size: 13 KiB |
BIN
test/fixtures/expected/tint-red.jpg
vendored
|
Before Width: | Height: | Size: 12 KiB After Width: | Height: | Size: 12 KiB |
BIN
test/fixtures/expected/tint-sepia.jpg
vendored
|
Before Width: | Height: | Size: 14 KiB After Width: | Height: | Size: 14 KiB |
3
test/fixtures/index.js
vendored
@@ -15,8 +15,7 @@ const fingerprint = function (image, callback) {
|
|||||||
sharp(image)
|
sharp(image)
|
||||||
.greyscale()
|
.greyscale()
|
||||||
.normalise()
|
.normalise()
|
||||||
.resize(9, 8)
|
.resize(9, 8, { fit: sharp.fit.fill })
|
||||||
.ignoreAspectRatio()
|
|
||||||
.raw()
|
.raw()
|
||||||
.toBuffer(function (err, data) {
|
.toBuffer(function (err, data) {
|
||||||
if (err) {
|
if (err) {
|
||||||
|
|||||||
@@ -5,7 +5,7 @@ if ! type valgrind >/dev/null; then
|
|||||||
exit 1
|
exit 1
|
||||||
fi
|
fi
|
||||||
|
|
||||||
curl -o ./test/leak/libvips.supp https://raw.githubusercontent.com/jcupitt/libvips/master/libvips.supp
|
curl -s -o ./test/leak/libvips.supp https://raw.githubusercontent.com/libvips/libvips/master/libvips.supp
|
||||||
|
|
||||||
for test in ./test/unit/*.js; do
|
for test in ./test/unit/*.js; do
|
||||||
G_SLICE=always-malloc G_DEBUG=gc-friendly valgrind \
|
G_SLICE=always-malloc G_DEBUG=gc-friendly valgrind \
|
||||||
@@ -16,5 +16,5 @@ for test in ./test/unit/*.js; do
|
|||||||
--show-leak-kinds=definite,indirect,possible \
|
--show-leak-kinds=definite,indirect,possible \
|
||||||
--num-callers=20 \
|
--num-callers=20 \
|
||||||
--trace-children=yes \
|
--trace-children=yes \
|
||||||
mocha --slow=60000 --timeout=120000 "$test";
|
node node_modules/.bin/mocha --slow=60000 --timeout=120000 --file test/unit/beforeEach.js "$test";
|
||||||
done
|
done
|
||||||
|
|||||||
@@ -233,7 +233,13 @@
|
|||||||
...
|
...
|
||||||
fun:uv__work_done
|
fun:uv__work_done
|
||||||
}
|
}
|
||||||
|
{
|
||||||
|
leak_libuv_FlushForegroundTasks
|
||||||
|
Memcheck:Leak
|
||||||
|
match-leak-kinds: possible
|
||||||
|
...
|
||||||
|
fun:_ZN4node12NodePlatform28FlushForegroundTasksInternalEv
|
||||||
|
}
|
||||||
# nodejs warnings
|
# nodejs warnings
|
||||||
{
|
{
|
||||||
param_nodejs_write_buffer
|
param_nodejs_write_buffer
|
||||||
@@ -360,6 +366,17 @@
|
|||||||
...
|
...
|
||||||
fun:_ZN2v84base6Thread5StartEv
|
fun:_ZN2v84base6Thread5StartEv
|
||||||
}
|
}
|
||||||
|
{
|
||||||
|
leak_nodejs_thread_TracingController
|
||||||
|
Memcheck:Leak
|
||||||
|
match-leak-kinds: possible
|
||||||
|
fun:calloc
|
||||||
|
fun:allocate_dtv
|
||||||
|
fun:_dl_allocate_tls
|
||||||
|
fun:allocate_stack
|
||||||
|
...
|
||||||
|
fun:_ZN4node12NodePlatformC1EiPN2v817TracingControllerE
|
||||||
|
}
|
||||||
{
|
{
|
||||||
leak_nan_FunctionCallbackInfo
|
leak_nan_FunctionCallbackInfo
|
||||||
Memcheck:Leak
|
Memcheck:Leak
|
||||||
|
|||||||
@@ -7,7 +7,6 @@ const async = require('async');
|
|||||||
const sharp = require('../../');
|
const sharp = require('../../');
|
||||||
|
|
||||||
const crops = {
|
const crops = {
|
||||||
centre: sharp.gravity.centre,
|
|
||||||
entropy: sharp.strategy.entropy,
|
entropy: sharp.strategy.entropy,
|
||||||
attention: sharp.strategy.attention
|
attention: sharp.strategy.attention
|
||||||
};
|
};
|
||||||
@@ -34,23 +33,35 @@ async.eachLimit(files, concurrency, function (file, done) {
|
|||||||
const salientHeight = userData[file].bottom - userData[file].top;
|
const salientHeight = userData[file].bottom - userData[file].top;
|
||||||
sharp(filename).metadata(function (err, metadata) {
|
sharp(filename).metadata(function (err, metadata) {
|
||||||
if (err) console.log(err);
|
if (err) console.log(err);
|
||||||
|
const marginWidth = metadata.width - salientWidth;
|
||||||
|
const marginHeight = metadata.height - salientHeight;
|
||||||
async.each(Object.keys(crops), function (crop, done) {
|
async.each(Object.keys(crops), function (crop, done) {
|
||||||
async.parallel([
|
async.parallel([
|
||||||
// Left edge accuracy
|
// Left edge accuracy
|
||||||
function (done) {
|
function (done) {
|
||||||
sharp(filename).resize(salientWidth, metadata.height).crop(crops[crop]).toBuffer(function (err, data, info) {
|
if (marginWidth) {
|
||||||
const accuracy = Math.round(Math.abs(userData[file].left - info.cropCalcLeft) / (metadata.width - salientWidth) * 100);
|
sharp(filename).resize(salientWidth, metadata.height).crop(crops[crop]).toBuffer(function (err, data, info) {
|
||||||
incrementScore(accuracy, crop);
|
const delta = Math.abs(userData[file].left + info.cropOffsetLeft);
|
||||||
done(err);
|
const accuracy = Math.round(marginWidth / (marginWidth + delta) * 100);
|
||||||
});
|
incrementScore(accuracy, crop);
|
||||||
|
done(err);
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
done();
|
||||||
|
}
|
||||||
},
|
},
|
||||||
// Top edge accuracy
|
// Top edge accuracy
|
||||||
function (done) {
|
function (done) {
|
||||||
sharp(filename).resize(metadata.width, salientHeight).crop(crops[crop]).toBuffer(function (err, data, info) {
|
if (marginHeight) {
|
||||||
const accuracy = Math.round(Math.abs(userData[file].top - info.cropCalcTop) / (metadata.height - salientHeight) * 100);
|
sharp(filename).resize(metadata.width, salientHeight).crop(crops[crop]).toBuffer(function (err, data, info) {
|
||||||
incrementScore(accuracy, crop);
|
const delta = Math.abs(userData[file].top + info.cropOffsetTop);
|
||||||
done(err);
|
const accuracy = Math.round(marginHeight / (marginHeight + delta) * 100);
|
||||||
});
|
incrementScore(accuracy, crop);
|
||||||
|
done(err);
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
done();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
], done);
|
], done);
|
||||||
}, done);
|
}, done);
|
||||||
@@ -60,7 +71,7 @@ async.eachLimit(files, concurrency, function (file, done) {
|
|||||||
Object.keys(scores).forEach(function (accuracy) {
|
Object.keys(scores).forEach(function (accuracy) {
|
||||||
report.push(
|
report.push(
|
||||||
Object.assign({
|
Object.assign({
|
||||||
accuracy: parseInt(accuracy, 10)
|
accuracy: Number(accuracy)
|
||||||
}, scores[accuracy])
|
}, scores[accuracy])
|
||||||
);
|
);
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -22,7 +22,7 @@ const median = function (values) {
|
|||||||
// List of files
|
// List of files
|
||||||
fs.readdirSync(userDataDir).forEach(function (file) {
|
fs.readdirSync(userDataDir).forEach(function (file) {
|
||||||
// Contents of file
|
// Contents of file
|
||||||
const lines = fs.readFileSync(path.join(userDataDir, file), {encoding: 'utf-8'}).split(/\r\n/);
|
const lines = fs.readFileSync(path.join(userDataDir, file), { encoding: 'utf-8' }).split(/\r\n/);
|
||||||
// First line = number of entries
|
// First line = number of entries
|
||||||
const entries = parseInt(lines[0], 10);
|
const entries = parseInt(lines[0], 10);
|
||||||
// Verify number of entries
|
// Verify number of entries
|
||||||
|
|||||||
@@ -19,9 +19,10 @@ describe('Alpha transparency', function () {
|
|||||||
|
|
||||||
it('Flatten to RGB orange', function (done) {
|
it('Flatten to RGB orange', function (done) {
|
||||||
sharp(fixtures.inputPngWithTransparency)
|
sharp(fixtures.inputPngWithTransparency)
|
||||||
.flatten()
|
|
||||||
.background({r: 255, g: 102, b: 0})
|
|
||||||
.resize(400, 300)
|
.resize(400, 300)
|
||||||
|
.flatten({
|
||||||
|
background: { r: 255, g: 102, b: 0 }
|
||||||
|
})
|
||||||
.toBuffer(function (err, data, info) {
|
.toBuffer(function (err, data, info) {
|
||||||
if (err) throw err;
|
if (err) throw err;
|
||||||
assert.strictEqual(400, info.width);
|
assert.strictEqual(400, info.width);
|
||||||
@@ -32,9 +33,8 @@ describe('Alpha transparency', function () {
|
|||||||
|
|
||||||
it('Flatten to CSS/hex orange', function (done) {
|
it('Flatten to CSS/hex orange', function (done) {
|
||||||
sharp(fixtures.inputPngWithTransparency)
|
sharp(fixtures.inputPngWithTransparency)
|
||||||
.flatten()
|
|
||||||
.background('#ff6600')
|
|
||||||
.resize(400, 300)
|
.resize(400, 300)
|
||||||
|
.flatten({ background: '#ff6600' })
|
||||||
.toBuffer(function (err, data, info) {
|
.toBuffer(function (err, data, info) {
|
||||||
if (err) throw err;
|
if (err) throw err;
|
||||||
assert.strictEqual(400, info.width);
|
assert.strictEqual(400, info.width);
|
||||||
@@ -46,8 +46,9 @@ describe('Alpha transparency', function () {
|
|||||||
it('Flatten 16-bit PNG with transparency to orange', function (done) {
|
it('Flatten 16-bit PNG with transparency to orange', function (done) {
|
||||||
const output = fixtures.path('output.flatten-rgb16-orange.jpg');
|
const output = fixtures.path('output.flatten-rgb16-orange.jpg');
|
||||||
sharp(fixtures.inputPngWithTransparency16bit)
|
sharp(fixtures.inputPngWithTransparency16bit)
|
||||||
.flatten()
|
.flatten({
|
||||||
.background({r: 255, g: 102, b: 0})
|
background: { r: 255, g: 102, b: 0 }
|
||||||
|
})
|
||||||
.toFile(output, function (err, info) {
|
.toFile(output, function (err, info) {
|
||||||
if (err) throw err;
|
if (err) throw err;
|
||||||
assert.strictEqual(true, info.size > 0);
|
assert.strictEqual(true, info.size > 0);
|
||||||
@@ -71,8 +72,7 @@ describe('Alpha transparency', function () {
|
|||||||
|
|
||||||
it('Ignored for JPEG', function (done) {
|
it('Ignored for JPEG', function (done) {
|
||||||
sharp(fixtures.inputJpg)
|
sharp(fixtures.inputJpg)
|
||||||
.background('#ff0000')
|
.flatten({ background: '#ff0000' })
|
||||||
.flatten()
|
|
||||||
.toBuffer(function (err, data, info) {
|
.toBuffer(function (err, data, info) {
|
||||||
if (err) throw err;
|
if (err) throw err;
|
||||||
assert.strictEqual('jpeg', info.format);
|
assert.strictEqual('jpeg', info.format);
|
||||||
|
|||||||
12
test/unit/beforeEach.js
Normal file
@@ -0,0 +1,12 @@
|
|||||||
|
'use strict';
|
||||||
|
|
||||||
|
const detectLibc = require('detect-libc');
|
||||||
|
const sharp = require('../../');
|
||||||
|
|
||||||
|
const usingCache = detectLibc.family !== detectLibc.MUSL;
|
||||||
|
const usingSimd = !process.env.G_DEBUG;
|
||||||
|
|
||||||
|
beforeEach(function () {
|
||||||
|
sharp.cache(usingCache);
|
||||||
|
sharp.simd(usingSimd);
|
||||||
|
});
|
||||||
@@ -1,9 +0,0 @@
|
|||||||
'use strict';
|
|
||||||
|
|
||||||
const sharp = require('../../');
|
|
||||||
|
|
||||||
// Define SHARP_TEST_WITHOUT_CACHE environment variable to prevent use of libvips' cache
|
|
||||||
|
|
||||||
beforeEach(function () {
|
|
||||||
sharp.cache(!process.env.SHARP_TEST_WITHOUT_CACHE);
|
|
||||||
});
|
|
||||||
@@ -69,9 +69,10 @@ describe('Colour space conversion', function () {
|
|||||||
|
|
||||||
it('From CMYK to sRGB with white background, not yellow', function (done) {
|
it('From CMYK to sRGB with white background, not yellow', function (done) {
|
||||||
sharp(fixtures.inputJpgWithCmykProfile)
|
sharp(fixtures.inputJpgWithCmykProfile)
|
||||||
.resize(320, 240)
|
.resize(320, 240, {
|
||||||
.background('white')
|
fit: sharp.fit.contain,
|
||||||
.embed()
|
background: 'white'
|
||||||
|
})
|
||||||
.toBuffer(function (err, data, info) {
|
.toBuffer(function (err, data, info) {
|
||||||
if (err) throw err;
|
if (err) throw err;
|
||||||
assert.strictEqual('jpeg', info.format);
|
assert.strictEqual('jpeg', info.format);
|
||||||
|
|||||||
73
test/unit/deprecated-background.js
Normal file
@@ -0,0 +1,73 @@
|
|||||||
|
'use strict';
|
||||||
|
|
||||||
|
const assert = require('assert');
|
||||||
|
const fixtures = require('../fixtures');
|
||||||
|
const sharp = require('../../');
|
||||||
|
|
||||||
|
describe('Deprecated background', function () {
|
||||||
|
it('Flatten to RGB orange', function (done) {
|
||||||
|
sharp(fixtures.inputPngWithTransparency)
|
||||||
|
.flatten()
|
||||||
|
.background({ r: 255, g: 102, b: 0 })
|
||||||
|
.resize(400, 300)
|
||||||
|
.toBuffer(function (err, data, info) {
|
||||||
|
if (err) throw err;
|
||||||
|
assert.strictEqual(400, info.width);
|
||||||
|
assert.strictEqual(300, info.height);
|
||||||
|
fixtures.assertSimilar(fixtures.expected('flatten-orange.jpg'), data, done);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('Flatten to CSS/hex orange', function (done) {
|
||||||
|
sharp(fixtures.inputPngWithTransparency)
|
||||||
|
.flatten()
|
||||||
|
.background('#ff6600')
|
||||||
|
.resize(400, 300)
|
||||||
|
.toBuffer(function (err, data, info) {
|
||||||
|
if (err) throw err;
|
||||||
|
assert.strictEqual(400, info.width);
|
||||||
|
assert.strictEqual(300, info.height);
|
||||||
|
fixtures.assertSimilar(fixtures.expected('flatten-orange.jpg'), data, done);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('Flatten 16-bit PNG with transparency to orange', function (done) {
|
||||||
|
const output = fixtures.path('output.flatten-rgb16-orange.jpg');
|
||||||
|
sharp(fixtures.inputPngWithTransparency16bit)
|
||||||
|
.flatten()
|
||||||
|
.background({ r: 255, g: 102, b: 0 })
|
||||||
|
.toFile(output, function (err, info) {
|
||||||
|
if (err) throw err;
|
||||||
|
assert.strictEqual(true, info.size > 0);
|
||||||
|
assert.strictEqual(32, info.width);
|
||||||
|
assert.strictEqual(32, info.height);
|
||||||
|
fixtures.assertMaxColourDistance(output, fixtures.expected('flatten-rgb16-orange.jpg'), 25);
|
||||||
|
done();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('Ignored for JPEG', function (done) {
|
||||||
|
sharp(fixtures.inputJpg)
|
||||||
|
.background('#ff0000')
|
||||||
|
.flatten()
|
||||||
|
.toBuffer(function (err, data, info) {
|
||||||
|
if (err) throw err;
|
||||||
|
assert.strictEqual('jpeg', info.format);
|
||||||
|
assert.strictEqual(3, info.channels);
|
||||||
|
done();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('extend all sides equally with RGB', function (done) {
|
||||||
|
sharp(fixtures.inputJpg)
|
||||||
|
.resize(120)
|
||||||
|
.background({ r: 255, g: 0, b: 0 })
|
||||||
|
.extend(10)
|
||||||
|
.toBuffer(function (err, data, info) {
|
||||||
|
if (err) throw err;
|
||||||
|
assert.strictEqual(140, info.width);
|
||||||
|
assert.strictEqual(118, info.height);
|
||||||
|
fixtures.assertSimilar(fixtures.expected('extend-equal.jpg'), data, done);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
@@ -5,7 +5,7 @@ const assert = require('assert');
|
|||||||
const sharp = require('../../');
|
const sharp = require('../../');
|
||||||
const fixtures = require('../fixtures');
|
const fixtures = require('../fixtures');
|
||||||
|
|
||||||
describe('Crop', function () {
|
describe('Deprecated crop', function () {
|
||||||
[
|
[
|
||||||
{
|
{
|
||||||
name: 'North',
|
name: 'North',
|
||||||
@@ -5,7 +5,7 @@ const assert = require('assert');
|
|||||||
const sharp = require('../../');
|
const sharp = require('../../');
|
||||||
const fixtures = require('../fixtures');
|
const fixtures = require('../fixtures');
|
||||||
|
|
||||||
describe('Embed', function () {
|
describe('Deprecated embed', function () {
|
||||||
it('Allows specifying the gravity as a string', function (done) {
|
it('Allows specifying the gravity as a string', function (done) {
|
||||||
sharp(fixtures.inputJpg)
|
sharp(fixtures.inputJpg)
|
||||||
.resize(320, 240)
|
.resize(320, 240)
|
||||||
@@ -38,7 +38,7 @@ describe('Embed', function () {
|
|||||||
it('JPEG within WebP, to include alpha channel', function (done) {
|
it('JPEG within WebP, to include alpha channel', function (done) {
|
||||||
sharp(fixtures.inputJpg)
|
sharp(fixtures.inputJpg)
|
||||||
.resize(320, 240)
|
.resize(320, 240)
|
||||||
.background({r: 0, g: 0, b: 0, alpha: 0})
|
.background({ r: 0, g: 0, b: 0, alpha: 0 })
|
||||||
.embed()
|
.embed()
|
||||||
.webp()
|
.webp()
|
||||||
.toBuffer(function (err, data, info) {
|
.toBuffer(function (err, data, info) {
|
||||||
@@ -86,7 +86,7 @@ describe('Embed', function () {
|
|||||||
sharp(fixtures.inputPngWithTransparency16bit)
|
sharp(fixtures.inputPngWithTransparency16bit)
|
||||||
.resize(32, 16)
|
.resize(32, 16)
|
||||||
.embed()
|
.embed()
|
||||||
.background({r: 0, g: 0, b: 0, alpha: 0})
|
.background({ r: 0, g: 0, b: 0, alpha: 0 })
|
||||||
.toBuffer(function (err, data, info) {
|
.toBuffer(function (err, data, info) {
|
||||||
if (err) throw err;
|
if (err) throw err;
|
||||||
assert.strictEqual(true, data.length > 0);
|
assert.strictEqual(true, data.length > 0);
|
||||||
@@ -102,7 +102,7 @@ describe('Embed', function () {
|
|||||||
sharp(fixtures.inputPngWithGreyAlpha)
|
sharp(fixtures.inputPngWithGreyAlpha)
|
||||||
.resize(32, 16)
|
.resize(32, 16)
|
||||||
.embed()
|
.embed()
|
||||||
.background({r: 0, g: 0, b: 0, alpha: 0})
|
.background({ r: 0, g: 0, b: 0, alpha: 0 })
|
||||||
.toBuffer(function (err, data, info) {
|
.toBuffer(function (err, data, info) {
|
||||||
if (err) throw err;
|
if (err) throw err;
|
||||||
assert.strictEqual(true, data.length > 0);
|
assert.strictEqual(true, data.length > 0);
|
||||||
@@ -114,23 +114,6 @@ describe('Embed', function () {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
it.skip('embed TIFF in LAB colourspace onto RGBA background', function (done) {
|
|
||||||
sharp(fixtures.inputTiffCielab)
|
|
||||||
.resize(64, 128)
|
|
||||||
.embed()
|
|
||||||
.background({r: 255, g: 102, b: 0, alpha: 0.5})
|
|
||||||
.png()
|
|
||||||
.toBuffer(function (err, data, info) {
|
|
||||||
if (err) throw err;
|
|
||||||
assert.strictEqual(true, data.length > 0);
|
|
||||||
assert.strictEqual('png', info.format);
|
|
||||||
assert.strictEqual(64, info.width);
|
|
||||||
assert.strictEqual(128, info.height);
|
|
||||||
assert.strictEqual(4, info.channels);
|
|
||||||
fixtures.assertSimilar(fixtures.expected('embed-lab-into-rgba.png'), data, done);
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
it('Enlarge and embed', function (done) {
|
it('Enlarge and embed', function (done) {
|
||||||
sharp(fixtures.inputPngWithOneColor)
|
sharp(fixtures.inputPngWithOneColor)
|
||||||
.embed()
|
.embed()
|
||||||
@@ -170,7 +153,7 @@ describe('Embed', function () {
|
|||||||
it('Embed gravity horizontal northwest', function (done) {
|
it('Embed gravity horizontal northwest', function (done) {
|
||||||
sharp(fixtures.inputPngEmbed)
|
sharp(fixtures.inputPngEmbed)
|
||||||
.resize(200, 100)
|
.resize(200, 100)
|
||||||
.background({r: 0, g: 0, b: 0, alpha: 0})
|
.background({ r: 0, g: 0, b: 0, alpha: 0 })
|
||||||
.embed(sharp.gravity.northwest)
|
.embed(sharp.gravity.northwest)
|
||||||
.toBuffer(function (err, data, info) {
|
.toBuffer(function (err, data, info) {
|
||||||
if (err) throw err;
|
if (err) throw err;
|
||||||
@@ -186,7 +169,7 @@ describe('Embed', function () {
|
|||||||
it('Embed gravity horizontal north', function (done) {
|
it('Embed gravity horizontal north', function (done) {
|
||||||
sharp(fixtures.inputPngEmbed)
|
sharp(fixtures.inputPngEmbed)
|
||||||
.resize(200, 100)
|
.resize(200, 100)
|
||||||
.background({r: 0, g: 0, b: 0, alpha: 0})
|
.background({ r: 0, g: 0, b: 0, alpha: 0 })
|
||||||
.embed(sharp.gravity.north)
|
.embed(sharp.gravity.north)
|
||||||
.toBuffer(function (err, data, info) {
|
.toBuffer(function (err, data, info) {
|
||||||
if (err) throw err;
|
if (err) throw err;
|
||||||
@@ -202,7 +185,7 @@ describe('Embed', function () {
|
|||||||
it('Embed gravity horizontal northeast', function (done) {
|
it('Embed gravity horizontal northeast', function (done) {
|
||||||
sharp(fixtures.inputPngEmbed)
|
sharp(fixtures.inputPngEmbed)
|
||||||
.resize(200, 100)
|
.resize(200, 100)
|
||||||
.background({r: 0, g: 0, b: 0, alpha: 0})
|
.background({ r: 0, g: 0, b: 0, alpha: 0 })
|
||||||
.embed(sharp.gravity.northeast)
|
.embed(sharp.gravity.northeast)
|
||||||
.toBuffer(function (err, data, info) {
|
.toBuffer(function (err, data, info) {
|
||||||
if (err) throw err;
|
if (err) throw err;
|
||||||
@@ -218,7 +201,7 @@ describe('Embed', function () {
|
|||||||
it('Embed gravity horizontal east', function (done) {
|
it('Embed gravity horizontal east', function (done) {
|
||||||
sharp(fixtures.inputPngEmbed)
|
sharp(fixtures.inputPngEmbed)
|
||||||
.resize(200, 100)
|
.resize(200, 100)
|
||||||
.background({r: 0, g: 0, b: 0, alpha: 0})
|
.background({ r: 0, g: 0, b: 0, alpha: 0 })
|
||||||
.embed(sharp.gravity.east)
|
.embed(sharp.gravity.east)
|
||||||
.toBuffer(function (err, data, info) {
|
.toBuffer(function (err, data, info) {
|
||||||
if (err) throw err;
|
if (err) throw err;
|
||||||
@@ -234,7 +217,7 @@ describe('Embed', function () {
|
|||||||
it('Embed gravity horizontal southeast', function (done) {
|
it('Embed gravity horizontal southeast', function (done) {
|
||||||
sharp(fixtures.inputPngEmbed)
|
sharp(fixtures.inputPngEmbed)
|
||||||
.resize(200, 100)
|
.resize(200, 100)
|
||||||
.background({r: 0, g: 0, b: 0, alpha: 0})
|
.background({ r: 0, g: 0, b: 0, alpha: 0 })
|
||||||
.embed(sharp.gravity.southeast)
|
.embed(sharp.gravity.southeast)
|
||||||
.toBuffer(function (err, data, info) {
|
.toBuffer(function (err, data, info) {
|
||||||
if (err) throw err;
|
if (err) throw err;
|
||||||
@@ -250,7 +233,7 @@ describe('Embed', function () {
|
|||||||
it('Embed gravity horizontal south', function (done) {
|
it('Embed gravity horizontal south', function (done) {
|
||||||
sharp(fixtures.inputPngEmbed)
|
sharp(fixtures.inputPngEmbed)
|
||||||
.resize(200, 100)
|
.resize(200, 100)
|
||||||
.background({r: 0, g: 0, b: 0, alpha: 0})
|
.background({ r: 0, g: 0, b: 0, alpha: 0 })
|
||||||
.embed(sharp.gravity.south)
|
.embed(sharp.gravity.south)
|
||||||
.toBuffer(function (err, data, info) {
|
.toBuffer(function (err, data, info) {
|
||||||
if (err) throw err;
|
if (err) throw err;
|
||||||
@@ -266,7 +249,7 @@ describe('Embed', function () {
|
|||||||
it('Embed gravity horizontal southwest', function (done) {
|
it('Embed gravity horizontal southwest', function (done) {
|
||||||
sharp(fixtures.inputPngEmbed)
|
sharp(fixtures.inputPngEmbed)
|
||||||
.resize(200, 100)
|
.resize(200, 100)
|
||||||
.background({r: 0, g: 0, b: 0, alpha: 0})
|
.background({ r: 0, g: 0, b: 0, alpha: 0 })
|
||||||
.embed(sharp.gravity.southwest)
|
.embed(sharp.gravity.southwest)
|
||||||
.toBuffer(function (err, data, info) {
|
.toBuffer(function (err, data, info) {
|
||||||
if (err) throw err;
|
if (err) throw err;
|
||||||
@@ -282,7 +265,7 @@ describe('Embed', function () {
|
|||||||
it('Embed gravity horizontal west', function (done) {
|
it('Embed gravity horizontal west', function (done) {
|
||||||
sharp(fixtures.inputPngEmbed)
|
sharp(fixtures.inputPngEmbed)
|
||||||
.resize(200, 100)
|
.resize(200, 100)
|
||||||
.background({r: 0, g: 0, b: 0, alpha: 0})
|
.background({ r: 0, g: 0, b: 0, alpha: 0 })
|
||||||
.embed(sharp.gravity.west)
|
.embed(sharp.gravity.west)
|
||||||
.toBuffer(function (err, data, info) {
|
.toBuffer(function (err, data, info) {
|
||||||
if (err) throw err;
|
if (err) throw err;
|
||||||
@@ -298,7 +281,7 @@ describe('Embed', function () {
|
|||||||
it('Embed gravity horizontal center', function (done) {
|
it('Embed gravity horizontal center', function (done) {
|
||||||
sharp(fixtures.inputPngEmbed)
|
sharp(fixtures.inputPngEmbed)
|
||||||
.resize(200, 100)
|
.resize(200, 100)
|
||||||
.background({r: 0, g: 0, b: 0, alpha: 0})
|
.background({ r: 0, g: 0, b: 0, alpha: 0 })
|
||||||
.embed(sharp.gravity.center)
|
.embed(sharp.gravity.center)
|
||||||
.toBuffer(function (err, data, info) {
|
.toBuffer(function (err, data, info) {
|
||||||
if (err) throw err;
|
if (err) throw err;
|
||||||
@@ -314,7 +297,7 @@ describe('Embed', function () {
|
|||||||
it('Embed gravity vertical northwest', function (done) {
|
it('Embed gravity vertical northwest', function (done) {
|
||||||
sharp(fixtures.inputPngEmbed)
|
sharp(fixtures.inputPngEmbed)
|
||||||
.resize(200, 200)
|
.resize(200, 200)
|
||||||
.background({r: 0, g: 0, b: 0, alpha: 0})
|
.background({ r: 0, g: 0, b: 0, alpha: 0 })
|
||||||
.embed(sharp.gravity.northwest)
|
.embed(sharp.gravity.northwest)
|
||||||
.toBuffer(function (err, data, info) {
|
.toBuffer(function (err, data, info) {
|
||||||
if (err) throw err;
|
if (err) throw err;
|
||||||
@@ -330,7 +313,7 @@ describe('Embed', function () {
|
|||||||
it('Embed gravity vertical north', function (done) {
|
it('Embed gravity vertical north', function (done) {
|
||||||
sharp(fixtures.inputPngEmbed)
|
sharp(fixtures.inputPngEmbed)
|
||||||
.resize(200, 200)
|
.resize(200, 200)
|
||||||
.background({r: 0, g: 0, b: 0, alpha: 0})
|
.background({ r: 0, g: 0, b: 0, alpha: 0 })
|
||||||
.embed(sharp.gravity.north)
|
.embed(sharp.gravity.north)
|
||||||
.toBuffer(function (err, data, info) {
|
.toBuffer(function (err, data, info) {
|
||||||
if (err) throw err;
|
if (err) throw err;
|
||||||
@@ -346,7 +329,7 @@ describe('Embed', function () {
|
|||||||
it('Embed gravity vertical northeast', function (done) {
|
it('Embed gravity vertical northeast', function (done) {
|
||||||
sharp(fixtures.inputPngEmbed)
|
sharp(fixtures.inputPngEmbed)
|
||||||
.resize(200, 200)
|
.resize(200, 200)
|
||||||
.background({r: 0, g: 0, b: 0, alpha: 0})
|
.background({ r: 0, g: 0, b: 0, alpha: 0 })
|
||||||
.embed(sharp.gravity.northeast)
|
.embed(sharp.gravity.northeast)
|
||||||
.toBuffer(function (err, data, info) {
|
.toBuffer(function (err, data, info) {
|
||||||
if (err) throw err;
|
if (err) throw err;
|
||||||
@@ -362,7 +345,7 @@ describe('Embed', function () {
|
|||||||
it('Embed gravity vertical east', function (done) {
|
it('Embed gravity vertical east', function (done) {
|
||||||
sharp(fixtures.inputPngEmbed)
|
sharp(fixtures.inputPngEmbed)
|
||||||
.resize(200, 200)
|
.resize(200, 200)
|
||||||
.background({r: 0, g: 0, b: 0, alpha: 0})
|
.background({ r: 0, g: 0, b: 0, alpha: 0 })
|
||||||
.embed(sharp.gravity.east)
|
.embed(sharp.gravity.east)
|
||||||
.toBuffer(function (err, data, info) {
|
.toBuffer(function (err, data, info) {
|
||||||
if (err) throw err;
|
if (err) throw err;
|
||||||
@@ -378,7 +361,7 @@ describe('Embed', function () {
|
|||||||
it('Embed gravity vertical southeast', function (done) {
|
it('Embed gravity vertical southeast', function (done) {
|
||||||
sharp(fixtures.inputPngEmbed)
|
sharp(fixtures.inputPngEmbed)
|
||||||
.resize(200, 200)
|
.resize(200, 200)
|
||||||
.background({r: 0, g: 0, b: 0, alpha: 0})
|
.background({ r: 0, g: 0, b: 0, alpha: 0 })
|
||||||
.embed(sharp.gravity.southeast)
|
.embed(sharp.gravity.southeast)
|
||||||
.toBuffer(function (err, data, info) {
|
.toBuffer(function (err, data, info) {
|
||||||
if (err) throw err;
|
if (err) throw err;
|
||||||
@@ -394,7 +377,7 @@ describe('Embed', function () {
|
|||||||
it('Embed gravity vertical south', function (done) {
|
it('Embed gravity vertical south', function (done) {
|
||||||
sharp(fixtures.inputPngEmbed)
|
sharp(fixtures.inputPngEmbed)
|
||||||
.resize(200, 200)
|
.resize(200, 200)
|
||||||
.background({r: 0, g: 0, b: 0, alpha: 0})
|
.background({ r: 0, g: 0, b: 0, alpha: 0 })
|
||||||
.embed(sharp.gravity.south)
|
.embed(sharp.gravity.south)
|
||||||
.toBuffer(function (err, data, info) {
|
.toBuffer(function (err, data, info) {
|
||||||
if (err) throw err;
|
if (err) throw err;
|
||||||
@@ -410,7 +393,7 @@ describe('Embed', function () {
|
|||||||
it('Embed gravity vertical southwest', function (done) {
|
it('Embed gravity vertical southwest', function (done) {
|
||||||
sharp(fixtures.inputPngEmbed)
|
sharp(fixtures.inputPngEmbed)
|
||||||
.resize(200, 200)
|
.resize(200, 200)
|
||||||
.background({r: 0, g: 0, b: 0, alpha: 0})
|
.background({ r: 0, g: 0, b: 0, alpha: 0 })
|
||||||
.embed(sharp.gravity.southwest)
|
.embed(sharp.gravity.southwest)
|
||||||
.toBuffer(function (err, data, info) {
|
.toBuffer(function (err, data, info) {
|
||||||
if (err) throw err;
|
if (err) throw err;
|
||||||
@@ -426,7 +409,7 @@ describe('Embed', function () {
|
|||||||
it('Embed gravity vertical west', function (done) {
|
it('Embed gravity vertical west', function (done) {
|
||||||
sharp(fixtures.inputPngEmbed)
|
sharp(fixtures.inputPngEmbed)
|
||||||
.resize(200, 200)
|
.resize(200, 200)
|
||||||
.background({r: 0, g: 0, b: 0, alpha: 0})
|
.background({ r: 0, g: 0, b: 0, alpha: 0 })
|
||||||
.embed(sharp.gravity.west)
|
.embed(sharp.gravity.west)
|
||||||
.toBuffer(function (err, data, info) {
|
.toBuffer(function (err, data, info) {
|
||||||
if (err) throw err;
|
if (err) throw err;
|
||||||
@@ -442,7 +425,7 @@ describe('Embed', function () {
|
|||||||
it('Embed gravity vertical center', function (done) {
|
it('Embed gravity vertical center', function (done) {
|
||||||
sharp(fixtures.inputPngEmbed)
|
sharp(fixtures.inputPngEmbed)
|
||||||
.resize(200, 200)
|
.resize(200, 200)
|
||||||
.background({r: 0, g: 0, b: 0, alpha: 0})
|
.background({ r: 0, g: 0, b: 0, alpha: 0 })
|
||||||
.embed(sharp.gravity.center)
|
.embed(sharp.gravity.center)
|
||||||
.toBuffer(function (err, data, info) {
|
.toBuffer(function (err, data, info) {
|
||||||
if (err) throw err;
|
if (err) throw err;
|
||||||
261
test/unit/deprecated-resize.js
Normal file
@@ -0,0 +1,261 @@
|
|||||||
|
'use strict';
|
||||||
|
|
||||||
|
const assert = require('assert');
|
||||||
|
|
||||||
|
const sharp = require('../../');
|
||||||
|
const fixtures = require('../fixtures');
|
||||||
|
|
||||||
|
describe('Deprecated resize-related functions', function () {
|
||||||
|
it('Max width or height considering ratio (portrait)', function (done) {
|
||||||
|
sharp(fixtures.inputTiff)
|
||||||
|
.resize(320, 320)
|
||||||
|
.max()
|
||||||
|
.jpeg()
|
||||||
|
.toBuffer(function (err, data, info) {
|
||||||
|
if (err) throw err;
|
||||||
|
assert.strictEqual(true, data.length > 0);
|
||||||
|
assert.strictEqual('jpeg', info.format);
|
||||||
|
assert.strictEqual(243, info.width);
|
||||||
|
assert.strictEqual(320, info.height);
|
||||||
|
done();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('Min width or height considering ratio (portrait)', function (done) {
|
||||||
|
sharp(fixtures.inputTiff)
|
||||||
|
.resize(320, 320)
|
||||||
|
.min()
|
||||||
|
.jpeg()
|
||||||
|
.toBuffer(function (err, data, info) {
|
||||||
|
if (err) throw err;
|
||||||
|
assert.strictEqual(true, data.length > 0);
|
||||||
|
assert.strictEqual('jpeg', info.format);
|
||||||
|
assert.strictEqual(320, info.width);
|
||||||
|
assert.strictEqual(422, info.height);
|
||||||
|
done();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('Max width or height considering ratio (landscape)', function (done) {
|
||||||
|
sharp(fixtures.inputJpg)
|
||||||
|
.resize(320, 320)
|
||||||
|
.max()
|
||||||
|
.toBuffer(function (err, data, info) {
|
||||||
|
if (err) throw err;
|
||||||
|
assert.strictEqual(true, data.length > 0);
|
||||||
|
assert.strictEqual('jpeg', info.format);
|
||||||
|
assert.strictEqual(320, info.width);
|
||||||
|
assert.strictEqual(261, info.height);
|
||||||
|
done();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('Provide only one dimension with max, should default to crop', function (done) {
|
||||||
|
sharp(fixtures.inputJpg)
|
||||||
|
.resize(320)
|
||||||
|
.max()
|
||||||
|
.toBuffer(function (err, data, info) {
|
||||||
|
if (err) throw err;
|
||||||
|
assert.strictEqual(true, data.length > 0);
|
||||||
|
assert.strictEqual('jpeg', info.format);
|
||||||
|
assert.strictEqual(320, info.width);
|
||||||
|
assert.strictEqual(261, info.height);
|
||||||
|
done();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('Min width or height considering ratio (landscape)', function (done) {
|
||||||
|
sharp(fixtures.inputJpg)
|
||||||
|
.resize(320, 320)
|
||||||
|
.min()
|
||||||
|
.toBuffer(function (err, data, info) {
|
||||||
|
if (err) throw err;
|
||||||
|
assert.strictEqual(true, data.length > 0);
|
||||||
|
assert.strictEqual('jpeg', info.format);
|
||||||
|
assert.strictEqual(392, info.width);
|
||||||
|
assert.strictEqual(320, info.height);
|
||||||
|
done();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('Provide only one dimension with min, should default to crop', function (done) {
|
||||||
|
sharp(fixtures.inputJpg)
|
||||||
|
.resize(320)
|
||||||
|
.min()
|
||||||
|
.toBuffer(function (err, data, info) {
|
||||||
|
if (err) throw err;
|
||||||
|
assert.strictEqual(true, data.length > 0);
|
||||||
|
assert.strictEqual('jpeg', info.format);
|
||||||
|
assert.strictEqual(320, info.width);
|
||||||
|
assert.strictEqual(261, info.height);
|
||||||
|
done();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('Do not enlarge when input width is already less than output width', function (done) {
|
||||||
|
sharp(fixtures.inputJpg)
|
||||||
|
.resize(2800)
|
||||||
|
.withoutEnlargement()
|
||||||
|
.toBuffer(function (err, data, info) {
|
||||||
|
if (err) throw err;
|
||||||
|
assert.strictEqual(true, data.length > 0);
|
||||||
|
assert.strictEqual('jpeg', info.format);
|
||||||
|
assert.strictEqual(2725, info.width);
|
||||||
|
assert.strictEqual(2225, info.height);
|
||||||
|
done();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('Do not enlarge when input height is already less than output height', function (done) {
|
||||||
|
sharp(fixtures.inputJpg)
|
||||||
|
.resize(null, 2300)
|
||||||
|
.withoutEnlargement()
|
||||||
|
.toBuffer(function (err, data, info) {
|
||||||
|
if (err) throw err;
|
||||||
|
assert.strictEqual(true, data.length > 0);
|
||||||
|
assert.strictEqual('jpeg', info.format);
|
||||||
|
assert.strictEqual(2725, info.width);
|
||||||
|
assert.strictEqual(2225, info.height);
|
||||||
|
done();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('Do enlarge when input width is less than output width', function (done) {
|
||||||
|
sharp(fixtures.inputJpg)
|
||||||
|
.resize(2800)
|
||||||
|
.withoutEnlargement(false)
|
||||||
|
.toBuffer(function (err, data, info) {
|
||||||
|
if (err) throw err;
|
||||||
|
assert.strictEqual(true, data.length > 0);
|
||||||
|
assert.strictEqual('jpeg', info.format);
|
||||||
|
assert.strictEqual(2800, info.width);
|
||||||
|
assert.strictEqual(2286, info.height);
|
||||||
|
done();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('Downscale width and height, ignoring aspect ratio', function (done) {
|
||||||
|
sharp(fixtures.inputJpg)
|
||||||
|
.resize(320, 320)
|
||||||
|
.ignoreAspectRatio()
|
||||||
|
.toBuffer(function (err, data, info) {
|
||||||
|
if (err) throw err;
|
||||||
|
assert.strictEqual(true, data.length > 0);
|
||||||
|
assert.strictEqual('jpeg', info.format);
|
||||||
|
assert.strictEqual(320, info.width);
|
||||||
|
assert.strictEqual(320, info.height);
|
||||||
|
done();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('Downscale width, ignoring aspect ratio', function (done) {
|
||||||
|
sharp(fixtures.inputJpg)
|
||||||
|
.resize(320)
|
||||||
|
.ignoreAspectRatio()
|
||||||
|
.toBuffer(function (err, data, info) {
|
||||||
|
if (err) throw err;
|
||||||
|
assert.strictEqual(true, data.length > 0);
|
||||||
|
assert.strictEqual('jpeg', info.format);
|
||||||
|
assert.strictEqual(320, info.width);
|
||||||
|
assert.strictEqual(2225, info.height);
|
||||||
|
done();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('Downscale height, ignoring aspect ratio', function (done) {
|
||||||
|
sharp(fixtures.inputJpg)
|
||||||
|
.resize(null, 320)
|
||||||
|
.ignoreAspectRatio()
|
||||||
|
.toBuffer(function (err, data, info) {
|
||||||
|
if (err) throw err;
|
||||||
|
assert.strictEqual(true, data.length > 0);
|
||||||
|
assert.strictEqual('jpeg', info.format);
|
||||||
|
assert.strictEqual(2725, info.width);
|
||||||
|
assert.strictEqual(320, info.height);
|
||||||
|
done();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('Upscale width and height, ignoring aspect ratio', function (done) {
|
||||||
|
sharp(fixtures.inputJpg)
|
||||||
|
.resize(3000, 3000)
|
||||||
|
.ignoreAspectRatio()
|
||||||
|
.toBuffer(function (err, data, info) {
|
||||||
|
if (err) throw err;
|
||||||
|
assert.strictEqual(true, data.length > 0);
|
||||||
|
assert.strictEqual('jpeg', info.format);
|
||||||
|
assert.strictEqual(3000, info.width);
|
||||||
|
assert.strictEqual(3000, info.height);
|
||||||
|
done();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('Upscale width, ignoring aspect ratio', function (done) {
|
||||||
|
sharp(fixtures.inputJpg)
|
||||||
|
.resize(3000)
|
||||||
|
.ignoreAspectRatio()
|
||||||
|
.toBuffer(function (err, data, info) {
|
||||||
|
if (err) throw err;
|
||||||
|
assert.strictEqual(true, data.length > 0);
|
||||||
|
assert.strictEqual('jpeg', info.format);
|
||||||
|
assert.strictEqual(3000, info.width);
|
||||||
|
assert.strictEqual(2225, info.height);
|
||||||
|
done();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('Upscale height, ignoring aspect ratio', function (done) {
|
||||||
|
sharp(fixtures.inputJpg)
|
||||||
|
.resize(null, 3000)
|
||||||
|
.ignoreAspectRatio()
|
||||||
|
.toBuffer(function (err, data, info) {
|
||||||
|
if (err) throw err;
|
||||||
|
assert.strictEqual(true, data.length > 0);
|
||||||
|
assert.strictEqual('jpeg', info.format);
|
||||||
|
assert.strictEqual(2725, info.width);
|
||||||
|
assert.strictEqual(3000, info.height);
|
||||||
|
done();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('Downscale width, upscale height, ignoring aspect ratio', function (done) {
|
||||||
|
sharp(fixtures.inputJpg)
|
||||||
|
.resize(320, 3000)
|
||||||
|
.ignoreAspectRatio()
|
||||||
|
.toBuffer(function (err, data, info) {
|
||||||
|
if (err) throw err;
|
||||||
|
assert.strictEqual(true, data.length > 0);
|
||||||
|
assert.strictEqual('jpeg', info.format);
|
||||||
|
assert.strictEqual(320, info.width);
|
||||||
|
assert.strictEqual(3000, info.height);
|
||||||
|
done();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('Upscale width, downscale height, ignoring aspect ratio', function (done) {
|
||||||
|
sharp(fixtures.inputJpg)
|
||||||
|
.resize(3000, 320)
|
||||||
|
.ignoreAspectRatio()
|
||||||
|
.toBuffer(function (err, data, info) {
|
||||||
|
if (err) throw err;
|
||||||
|
assert.strictEqual(true, data.length > 0);
|
||||||
|
assert.strictEqual('jpeg', info.format);
|
||||||
|
assert.strictEqual(3000, info.width);
|
||||||
|
assert.strictEqual(320, info.height);
|
||||||
|
done();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('Identity transform, ignoring aspect ratio', function (done) {
|
||||||
|
sharp(fixtures.inputJpg)
|
||||||
|
.ignoreAspectRatio()
|
||||||
|
.toBuffer(function (err, data, info) {
|
||||||
|
if (err) throw err;
|
||||||
|
assert.strictEqual(true, data.length > 0);
|
||||||
|
assert.strictEqual('jpeg', info.format);
|
||||||
|
assert.strictEqual(2725, info.width);
|
||||||
|
assert.strictEqual(2225, info.height);
|
||||||
|
done();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
@@ -9,8 +9,13 @@ describe('Extend', function () {
|
|||||||
it('extend all sides equally with RGB', function (done) {
|
it('extend all sides equally with RGB', function (done) {
|
||||||
sharp(fixtures.inputJpg)
|
sharp(fixtures.inputJpg)
|
||||||
.resize(120)
|
.resize(120)
|
||||||
.background({r: 255, g: 0, b: 0})
|
.extend({
|
||||||
.extend(10)
|
top: 10,
|
||||||
|
bottom: 10,
|
||||||
|
left: 10,
|
||||||
|
right: 10,
|
||||||
|
background: { r: 255, g: 0, b: 0 }
|
||||||
|
})
|
||||||
.toBuffer(function (err, data, info) {
|
.toBuffer(function (err, data, info) {
|
||||||
if (err) throw err;
|
if (err) throw err;
|
||||||
assert.strictEqual(140, info.width);
|
assert.strictEqual(140, info.width);
|
||||||
@@ -22,8 +27,13 @@ describe('Extend', function () {
|
|||||||
it('extend sides unequally with RGBA', function (done) {
|
it('extend sides unequally with RGBA', function (done) {
|
||||||
sharp(fixtures.inputPngWithTransparency16bit)
|
sharp(fixtures.inputPngWithTransparency16bit)
|
||||||
.resize(120)
|
.resize(120)
|
||||||
.background({r: 0, g: 0, b: 0, alpha: 0})
|
.extend({
|
||||||
.extend({top: 50, bottom: 0, left: 10, right: 35})
|
top: 50,
|
||||||
|
bottom: 0,
|
||||||
|
left: 10,
|
||||||
|
right: 35,
|
||||||
|
background: { r: 0, g: 0, b: 0, alpha: 0 }
|
||||||
|
})
|
||||||
.toBuffer(function (err, data, info) {
|
.toBuffer(function (err, data, info) {
|
||||||
if (err) throw err;
|
if (err) throw err;
|
||||||
assert.strictEqual(165, info.width);
|
assert.strictEqual(165, info.width);
|
||||||
@@ -44,15 +54,20 @@ describe('Extend', function () {
|
|||||||
});
|
});
|
||||||
it('partial object fails', function () {
|
it('partial object fails', function () {
|
||||||
assert.throws(function () {
|
assert.throws(function () {
|
||||||
sharp().extend({top: 1});
|
sharp().extend({ top: 1 });
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should add alpha channel before extending with a transparent Background', function (done) {
|
it('should add alpha channel before extending with a transparent Background', function (done) {
|
||||||
sharp(fixtures.inputJpgWithLandscapeExif1)
|
sharp(fixtures.inputJpgWithLandscapeExif1)
|
||||||
.background({r: 0, g: 0, b: 0, alpha: 0})
|
.extend({
|
||||||
|
top: 0,
|
||||||
|
bottom: 10,
|
||||||
|
left: 0,
|
||||||
|
right: 10,
|
||||||
|
background: { r: 0, g: 0, b: 0, alpha: 0 }
|
||||||
|
})
|
||||||
.toFormat(sharp.format.png)
|
.toFormat(sharp.format.png)
|
||||||
.extend({top: 0, bottom: 10, left: 0, right: 10})
|
|
||||||
.toBuffer(function (err, data, info) {
|
.toBuffer(function (err, data, info) {
|
||||||
if (err) throw err;
|
if (err) throw err;
|
||||||
assert.strictEqual(610, info.width);
|
assert.strictEqual(610, info.width);
|
||||||
@@ -63,8 +78,13 @@ describe('Extend', function () {
|
|||||||
|
|
||||||
it('PNG with 2 channels', function (done) {
|
it('PNG with 2 channels', function (done) {
|
||||||
sharp(fixtures.inputPngWithGreyAlpha)
|
sharp(fixtures.inputPngWithGreyAlpha)
|
||||||
.background('transparent')
|
.extend({
|
||||||
.extend({top: 0, bottom: 20, left: 0, right: 20})
|
top: 0,
|
||||||
|
bottom: 20,
|
||||||
|
left: 0,
|
||||||
|
right: 20,
|
||||||
|
background: 'transparent'
|
||||||
|
})
|
||||||
.toBuffer(function (err, data, info) {
|
.toBuffer(function (err, data, info) {
|
||||||
if (err) throw err;
|
if (err) throw err;
|
||||||
assert.strictEqual(true, data.length > 0);
|
assert.strictEqual(true, data.length > 0);
|
||||||
|
|||||||
@@ -69,8 +69,9 @@ describe('Partial image extraction', function () {
|
|||||||
|
|
||||||
it('After resize and crop', function (done) {
|
it('After resize and crop', function (done) {
|
||||||
sharp(fixtures.inputJpg)
|
sharp(fixtures.inputJpg)
|
||||||
.resize(500, 500)
|
.resize(500, 500, {
|
||||||
.crop(sharp.gravity.north)
|
position: sharp.gravity.north
|
||||||
|
})
|
||||||
.extract({ left: 10, top: 10, width: 100, height: 100 })
|
.extract({ left: 10, top: 10, width: 100, height: 100 })
|
||||||
.toBuffer(function (err, data, info) {
|
.toBuffer(function (err, data, info) {
|
||||||
if (err) throw err;
|
if (err) throw err;
|
||||||
@@ -83,8 +84,9 @@ describe('Partial image extraction', function () {
|
|||||||
it('Before and after resize and crop', function (done) {
|
it('Before and after resize and crop', function (done) {
|
||||||
sharp(fixtures.inputJpg)
|
sharp(fixtures.inputJpg)
|
||||||
.extract({ left: 0, top: 0, width: 700, height: 700 })
|
.extract({ left: 0, top: 0, width: 700, height: 700 })
|
||||||
.resize(500, 500)
|
.resize(500, 500, {
|
||||||
.crop(sharp.gravity.north)
|
position: sharp.gravity.north
|
||||||
|
})
|
||||||
.extract({ left: 10, top: 10, width: 100, height: 100 })
|
.extract({ left: 10, top: 10, width: 100, height: 100 })
|
||||||
.toBuffer(function (err, data, info) {
|
.toBuffer(function (err, data, info) {
|
||||||
if (err) throw err;
|
if (err) throw err;
|
||||||
@@ -115,7 +117,7 @@ describe('Partial image extraction', function () {
|
|||||||
if (err) throw err;
|
if (err) throw err;
|
||||||
assert.strictEqual(280, info.width);
|
assert.strictEqual(280, info.width);
|
||||||
assert.strictEqual(380, info.height);
|
assert.strictEqual(380, info.height);
|
||||||
fixtures.assertSimilar(fixtures.expected('rotate-extract.jpg'), data, { threshold: 6 }, done);
|
fixtures.assertSimilar(fixtures.expected('rotate-extract.jpg'), data, { threshold: 7 }, done);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|||||||
@@ -59,12 +59,12 @@ describe('Image channel extraction', function () {
|
|||||||
sharp(fixtures.inputJpg)
|
sharp(fixtures.inputJpg)
|
||||||
.toColourspace('lch')
|
.toColourspace('lch')
|
||||||
.extractChannel(1)
|
.extractChannel(1)
|
||||||
.resize(320, 240)
|
.resize(320, 240, { fastShrinkOnLoad: false })
|
||||||
.toFile(output, function (err, info) {
|
.toFile(output, function (err, info) {
|
||||||
if (err) throw err;
|
if (err) throw err;
|
||||||
assert.strictEqual(320, info.width);
|
assert.strictEqual(320, info.width);
|
||||||
assert.strictEqual(240, info.height);
|
assert.strictEqual(240, info.height);
|
||||||
fixtures.assertMaxColourDistance(output, fixtures.expected('extract-lch.jpg'));
|
fixtures.assertMaxColourDistance(output, fixtures.expected('extract-lch.jpg'), 9);
|
||||||
done();
|
done();
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -49,8 +49,8 @@ describe('failOnError', function () {
|
|||||||
it('returns errors to callback for truncated JPEG when failOnError is set', function (done) {
|
it('returns errors to callback for truncated JPEG when failOnError is set', function (done) {
|
||||||
sharp(fixtures.inputJpgTruncated, { failOnError: true }).toBuffer(function (err, data, info) {
|
sharp(fixtures.inputJpgTruncated, { failOnError: true }).toBuffer(function (err, data, info) {
|
||||||
assert.ok(err.message.includes('VipsJpeg: Premature end of JPEG file'), err);
|
assert.ok(err.message.includes('VipsJpeg: Premature end of JPEG file'), err);
|
||||||
assert.equal(data, null);
|
assert.strictEqual(data, null);
|
||||||
assert.equal(info, null);
|
assert.strictEqual(info, null);
|
||||||
done();
|
done();
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
@@ -58,8 +58,8 @@ describe('failOnError', function () {
|
|||||||
it('returns errors to callback for truncated PNG when failOnError is set', function (done) {
|
it('returns errors to callback for truncated PNG when failOnError is set', function (done) {
|
||||||
sharp(fixtures.inputPngTruncated, { failOnError: true }).toBuffer(function (err, data, info) {
|
sharp(fixtures.inputPngTruncated, { failOnError: true }).toBuffer(function (err, data, info) {
|
||||||
assert.ok(err.message.includes('vipspng: libpng read error'), err);
|
assert.ok(err.message.includes('vipspng: libpng read error'), err);
|
||||||
assert.equal(data, null);
|
assert.strictEqual(data, null);
|
||||||
assert.equal(info, null);
|
assert.strictEqual(info, null);
|
||||||
done();
|
done();
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -44,6 +44,19 @@ describe('Gamma correction', function () {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
it('input value of 2.2, output value of 3.0', function (done) {
|
||||||
|
sharp(fixtures.inputJpgWithGammaHoliness)
|
||||||
|
.resize(129, 111)
|
||||||
|
.gamma(2.2, 3.0)
|
||||||
|
.toBuffer(function (err, data, info) {
|
||||||
|
if (err) throw err;
|
||||||
|
assert.strictEqual('jpeg', info.format);
|
||||||
|
assert.strictEqual(129, info.width);
|
||||||
|
assert.strictEqual(111, info.height);
|
||||||
|
fixtures.assertSimilar(fixtures.expected('gamma-in-2.2-out-3.0.jpg'), data, { threshold: 6 }, done);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
it('alpha transparency', function (done) {
|
it('alpha transparency', function (done) {
|
||||||
sharp(fixtures.inputPngOverlayLayer1)
|
sharp(fixtures.inputPngOverlayLayer1)
|
||||||
.resize(320)
|
.resize(320)
|
||||||
@@ -57,9 +70,15 @@ describe('Gamma correction', function () {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
it('invalid value', function () {
|
it('invalid first parameter value', function () {
|
||||||
assert.throws(function () {
|
assert.throws(function () {
|
||||||
sharp(fixtures.inputJpgWithGammaHoliness).gamma(4);
|
sharp(fixtures.inputJpgWithGammaHoliness).gamma(4);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
it('invalid second parameter value', function () {
|
||||||
|
assert.throws(function () {
|
||||||
|
sharp(fixtures.inputJpgWithGammaHoliness).gamma(2.2, 4);
|
||||||
|
});
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
|||||||
205
test/unit/io.js
@@ -2,6 +2,7 @@
|
|||||||
|
|
||||||
const fs = require('fs');
|
const fs = require('fs');
|
||||||
const assert = require('assert');
|
const assert = require('assert');
|
||||||
|
const rimraf = require('rimraf');
|
||||||
|
|
||||||
const sharp = require('../../');
|
const sharp = require('../../');
|
||||||
const fixtures = require('../fixtures');
|
const fixtures = require('../fixtures');
|
||||||
@@ -16,7 +17,7 @@ describe('Input/output', function () {
|
|||||||
|
|
||||||
it('Read from File and write to Stream', function (done) {
|
it('Read from File and write to Stream', function (done) {
|
||||||
const writable = fs.createWriteStream(fixtures.outputJpg);
|
const writable = fs.createWriteStream(fixtures.outputJpg);
|
||||||
writable.on('finish', function () {
|
writable.on('close', function () {
|
||||||
sharp(fixtures.outputJpg).toBuffer(function (err, data, info) {
|
sharp(fixtures.outputJpg).toBuffer(function (err, data, info) {
|
||||||
if (err) throw err;
|
if (err) throw err;
|
||||||
assert.strictEqual(true, data.length > 0);
|
assert.strictEqual(true, data.length > 0);
|
||||||
@@ -24,8 +25,7 @@ describe('Input/output', function () {
|
|||||||
assert.strictEqual('jpeg', info.format);
|
assert.strictEqual('jpeg', info.format);
|
||||||
assert.strictEqual(320, info.width);
|
assert.strictEqual(320, info.width);
|
||||||
assert.strictEqual(240, info.height);
|
assert.strictEqual(240, info.height);
|
||||||
fs.unlinkSync(fixtures.outputJpg);
|
rimraf(fixtures.outputJpg, done);
|
||||||
done();
|
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
sharp(fixtures.inputJpg).resize(320, 240).pipe(writable);
|
sharp(fixtures.inputJpg).resize(320, 240).pipe(writable);
|
||||||
@@ -34,7 +34,7 @@ describe('Input/output', function () {
|
|||||||
it('Read from Buffer and write to Stream', function (done) {
|
it('Read from Buffer and write to Stream', function (done) {
|
||||||
const inputJpgBuffer = fs.readFileSync(fixtures.inputJpg);
|
const inputJpgBuffer = fs.readFileSync(fixtures.inputJpg);
|
||||||
const writable = fs.createWriteStream(fixtures.outputJpg);
|
const writable = fs.createWriteStream(fixtures.outputJpg);
|
||||||
writable.on('finish', function () {
|
writable.on('close', function () {
|
||||||
sharp(fixtures.outputJpg).toBuffer(function (err, data, info) {
|
sharp(fixtures.outputJpg).toBuffer(function (err, data, info) {
|
||||||
if (err) throw err;
|
if (err) throw err;
|
||||||
assert.strictEqual(true, data.length > 0);
|
assert.strictEqual(true, data.length > 0);
|
||||||
@@ -42,8 +42,7 @@ describe('Input/output', function () {
|
|||||||
assert.strictEqual('jpeg', info.format);
|
assert.strictEqual('jpeg', info.format);
|
||||||
assert.strictEqual(320, info.width);
|
assert.strictEqual(320, info.width);
|
||||||
assert.strictEqual(240, info.height);
|
assert.strictEqual(240, info.height);
|
||||||
fs.unlinkSync(fixtures.outputJpg);
|
rimraf(fixtures.outputJpg, done);
|
||||||
done();
|
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
sharp(inputJpgBuffer).resize(320, 240).pipe(writable);
|
sharp(inputJpgBuffer).resize(320, 240).pipe(writable);
|
||||||
@@ -57,8 +56,7 @@ describe('Input/output', function () {
|
|||||||
assert.strictEqual('jpeg', info.format);
|
assert.strictEqual('jpeg', info.format);
|
||||||
assert.strictEqual(320, info.width);
|
assert.strictEqual(320, info.width);
|
||||||
assert.strictEqual(240, info.height);
|
assert.strictEqual(240, info.height);
|
||||||
fs.unlinkSync(fixtures.outputJpg);
|
rimraf(fixtures.outputJpg, done);
|
||||||
done();
|
|
||||||
});
|
});
|
||||||
readable.pipe(pipeline);
|
readable.pipe(pipeline);
|
||||||
});
|
});
|
||||||
@@ -81,7 +79,7 @@ describe('Input/output', function () {
|
|||||||
const pipeline = sharp().resize(1, 1);
|
const pipeline = sharp().resize(1, 1);
|
||||||
fs.createReadStream(fixtures.inputJpg).pipe(pipeline);
|
fs.createReadStream(fixtures.inputJpg).pipe(pipeline);
|
||||||
return pipeline
|
return pipeline
|
||||||
.toBuffer({resolveWithObject: false})
|
.toBuffer({ resolveWithObject: false })
|
||||||
.then(function (data) {
|
.then(function (data) {
|
||||||
assert.strictEqual(true, data instanceof Buffer);
|
assert.strictEqual(true, data instanceof Buffer);
|
||||||
assert.strictEqual(true, data.length > 0);
|
assert.strictEqual(true, data.length > 0);
|
||||||
@@ -92,7 +90,7 @@ describe('Input/output', function () {
|
|||||||
const pipeline = sharp().resize(1, 1);
|
const pipeline = sharp().resize(1, 1);
|
||||||
fs.createReadStream(fixtures.inputJpg).pipe(pipeline);
|
fs.createReadStream(fixtures.inputJpg).pipe(pipeline);
|
||||||
return pipeline
|
return pipeline
|
||||||
.toBuffer({resolveWithObject: true})
|
.toBuffer({ resolveWithObject: true })
|
||||||
.then(function (object) {
|
.then(function (object) {
|
||||||
assert.strictEqual('object', typeof object);
|
assert.strictEqual('object', typeof object);
|
||||||
assert.strictEqual('object', typeof object.info);
|
assert.strictEqual('object', typeof object.info);
|
||||||
@@ -108,7 +106,7 @@ describe('Input/output', function () {
|
|||||||
it('Read from File and write to Buffer via Promise resolved with Buffer', function () {
|
it('Read from File and write to Buffer via Promise resolved with Buffer', function () {
|
||||||
return sharp(fixtures.inputJpg)
|
return sharp(fixtures.inputJpg)
|
||||||
.resize(1, 1)
|
.resize(1, 1)
|
||||||
.toBuffer({resolveWithObject: false})
|
.toBuffer({ resolveWithObject: false })
|
||||||
.then(function (data) {
|
.then(function (data) {
|
||||||
assert.strictEqual(true, data instanceof Buffer);
|
assert.strictEqual(true, data instanceof Buffer);
|
||||||
assert.strictEqual(true, data.length > 0);
|
assert.strictEqual(true, data.length > 0);
|
||||||
@@ -118,7 +116,7 @@ describe('Input/output', function () {
|
|||||||
it('Read from File and write to Buffer via Promise resolved with Object', function () {
|
it('Read from File and write to Buffer via Promise resolved with Object', function () {
|
||||||
return sharp(fixtures.inputJpg)
|
return sharp(fixtures.inputJpg)
|
||||||
.resize(1, 1)
|
.resize(1, 1)
|
||||||
.toBuffer({resolveWithObject: true})
|
.toBuffer({ resolveWithObject: true })
|
||||||
.then(function (object) {
|
.then(function (object) {
|
||||||
assert.strictEqual('object', typeof object);
|
assert.strictEqual('object', typeof object);
|
||||||
assert.strictEqual('object', typeof object.info);
|
assert.strictEqual('object', typeof object.info);
|
||||||
@@ -134,7 +132,7 @@ describe('Input/output', function () {
|
|||||||
it('Read from Stream and write to Stream', function (done) {
|
it('Read from Stream and write to Stream', function (done) {
|
||||||
const readable = fs.createReadStream(fixtures.inputJpg);
|
const readable = fs.createReadStream(fixtures.inputJpg);
|
||||||
const writable = fs.createWriteStream(fixtures.outputJpg);
|
const writable = fs.createWriteStream(fixtures.outputJpg);
|
||||||
writable.on('finish', function () {
|
writable.on('close', function () {
|
||||||
sharp(fixtures.outputJpg).toBuffer(function (err, data, info) {
|
sharp(fixtures.outputJpg).toBuffer(function (err, data, info) {
|
||||||
if (err) throw err;
|
if (err) throw err;
|
||||||
assert.strictEqual(true, data.length > 0);
|
assert.strictEqual(true, data.length > 0);
|
||||||
@@ -142,8 +140,7 @@ describe('Input/output', function () {
|
|||||||
assert.strictEqual('jpeg', info.format);
|
assert.strictEqual('jpeg', info.format);
|
||||||
assert.strictEqual(320, info.width);
|
assert.strictEqual(320, info.width);
|
||||||
assert.strictEqual(240, info.height);
|
assert.strictEqual(240, info.height);
|
||||||
fs.unlinkSync(fixtures.outputJpg);
|
rimraf(fixtures.outputJpg, done);
|
||||||
done();
|
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
const pipeline = sharp().resize(320, 240);
|
const pipeline = sharp().resize(320, 240);
|
||||||
@@ -162,10 +159,9 @@ describe('Input/output', function () {
|
|||||||
assert.strictEqual(3, info.channels);
|
assert.strictEqual(3, info.channels);
|
||||||
infoEventEmitted = true;
|
infoEventEmitted = true;
|
||||||
});
|
});
|
||||||
writable.on('finish', function () {
|
writable.on('close', function () {
|
||||||
assert.strictEqual(true, infoEventEmitted);
|
assert.strictEqual(true, infoEventEmitted);
|
||||||
fs.unlinkSync(fixtures.outputJpg);
|
rimraf(fixtures.outputJpg, done);
|
||||||
done();
|
|
||||||
});
|
});
|
||||||
readable.pipe(pipeline).pipe(writable);
|
readable.pipe(pipeline).pipe(writable);
|
||||||
});
|
});
|
||||||
@@ -177,8 +173,7 @@ describe('Input/output', function () {
|
|||||||
anErrorWasEmitted = !!err;
|
anErrorWasEmitted = !!err;
|
||||||
}).on('end', function () {
|
}).on('end', function () {
|
||||||
assert(anErrorWasEmitted);
|
assert(anErrorWasEmitted);
|
||||||
fs.unlinkSync(fixtures.outputJpg);
|
rimraf(fixtures.outputJpg, done);
|
||||||
done();
|
|
||||||
});
|
});
|
||||||
const readableButNotAnImage = fs.createReadStream(__filename);
|
const readableButNotAnImage = fs.createReadStream(__filename);
|
||||||
const writable = fs.createWriteStream(fixtures.outputJpg);
|
const writable = fs.createWriteStream(fixtures.outputJpg);
|
||||||
@@ -192,8 +187,7 @@ describe('Input/output', function () {
|
|||||||
anErrorWasEmitted = !!err;
|
anErrorWasEmitted = !!err;
|
||||||
}).on('end', function () {
|
}).on('end', function () {
|
||||||
assert(anErrorWasEmitted);
|
assert(anErrorWasEmitted);
|
||||||
fs.unlinkSync(fixtures.outputJpg);
|
rimraf(fixtures.outputJpg, done);
|
||||||
done();
|
|
||||||
});
|
});
|
||||||
const writable = fs.createWriteStream(fixtures.outputJpg);
|
const writable = fs.createWriteStream(fixtures.outputJpg);
|
||||||
readableButNotAnImage.pipe(writable);
|
readableButNotAnImage.pipe(writable);
|
||||||
@@ -202,7 +196,7 @@ describe('Input/output', function () {
|
|||||||
it('Readable side of Stream can start flowing after Writable side has finished', function (done) {
|
it('Readable side of Stream can start flowing after Writable side has finished', function (done) {
|
||||||
const readable = fs.createReadStream(fixtures.inputJpg);
|
const readable = fs.createReadStream(fixtures.inputJpg);
|
||||||
const writable = fs.createWriteStream(fixtures.outputJpg);
|
const writable = fs.createWriteStream(fixtures.outputJpg);
|
||||||
writable.on('finish', function () {
|
writable.on('close', function () {
|
||||||
sharp(fixtures.outputJpg).toBuffer(function (err, data, info) {
|
sharp(fixtures.outputJpg).toBuffer(function (err, data, info) {
|
||||||
if (err) throw err;
|
if (err) throw err;
|
||||||
assert.strictEqual(true, data.length > 0);
|
assert.strictEqual(true, data.length > 0);
|
||||||
@@ -210,8 +204,7 @@ describe('Input/output', function () {
|
|||||||
assert.strictEqual('jpeg', info.format);
|
assert.strictEqual('jpeg', info.format);
|
||||||
assert.strictEqual(320, info.width);
|
assert.strictEqual(320, info.width);
|
||||||
assert.strictEqual(240, info.height);
|
assert.strictEqual(240, info.height);
|
||||||
fs.unlinkSync(fixtures.outputJpg);
|
rimraf(fixtures.outputJpg, done);
|
||||||
done();
|
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
const pipeline = sharp().resize(320, 240);
|
const pipeline = sharp().resize(320, 240);
|
||||||
@@ -469,7 +462,7 @@ describe('Input/output', function () {
|
|||||||
|
|
||||||
it('should work for webp alpha quality', function (done) {
|
it('should work for webp alpha quality', function (done) {
|
||||||
sharp(fixtures.inputPngAlphaPremultiplicationSmall)
|
sharp(fixtures.inputPngAlphaPremultiplicationSmall)
|
||||||
.webp({alphaQuality: 80})
|
.webp({ alphaQuality: 80 })
|
||||||
.toBuffer(function (err, data, info) {
|
.toBuffer(function (err, data, info) {
|
||||||
if (err) throw err;
|
if (err) throw err;
|
||||||
assert.strictEqual(true, data.length > 0);
|
assert.strictEqual(true, data.length > 0);
|
||||||
@@ -480,7 +473,7 @@ describe('Input/output', function () {
|
|||||||
|
|
||||||
it('should work for webp lossless', function (done) {
|
it('should work for webp lossless', function (done) {
|
||||||
sharp(fixtures.inputPngAlphaPremultiplicationSmall)
|
sharp(fixtures.inputPngAlphaPremultiplicationSmall)
|
||||||
.webp({lossless: true})
|
.webp({ lossless: true })
|
||||||
.toBuffer(function (err, data, info) {
|
.toBuffer(function (err, data, info) {
|
||||||
if (err) throw err;
|
if (err) throw err;
|
||||||
assert.strictEqual(true, data.length > 0);
|
assert.strictEqual(true, data.length > 0);
|
||||||
@@ -491,7 +484,7 @@ describe('Input/output', function () {
|
|||||||
|
|
||||||
it('should work for webp near-lossless', function (done) {
|
it('should work for webp near-lossless', function (done) {
|
||||||
sharp(fixtures.inputPngAlphaPremultiplicationSmall)
|
sharp(fixtures.inputPngAlphaPremultiplicationSmall)
|
||||||
.webp({nearLossless: true, quality: 50})
|
.webp({ nearLossless: true, quality: 50 })
|
||||||
.toBuffer(function (err50, data50, info50) {
|
.toBuffer(function (err50, data50, info50) {
|
||||||
if (err50) throw err50;
|
if (err50) throw err50;
|
||||||
assert.strictEqual(true, data50.length > 0);
|
assert.strictEqual(true, data50.length > 0);
|
||||||
@@ -502,7 +495,7 @@ describe('Input/output', function () {
|
|||||||
|
|
||||||
it('should use near-lossless when both lossless and nearLossless are specified', function (done) {
|
it('should use near-lossless when both lossless and nearLossless are specified', function (done) {
|
||||||
sharp(fixtures.inputPngAlphaPremultiplicationSmall)
|
sharp(fixtures.inputPngAlphaPremultiplicationSmall)
|
||||||
.webp({nearLossless: true, quality: 50, lossless: true})
|
.webp({ nearLossless: true, quality: 50, lossless: true })
|
||||||
.toBuffer(function (err50, data50, info50) {
|
.toBuffer(function (err50, data50, info50) {
|
||||||
if (err50) throw err50;
|
if (err50) throw err50;
|
||||||
assert.strictEqual(true, data50.length > 0);
|
assert.strictEqual(true, data50.length > 0);
|
||||||
@@ -564,8 +557,7 @@ describe('Input/output', function () {
|
|||||||
assert.strictEqual('jpeg', info.format);
|
assert.strictEqual('jpeg', info.format);
|
||||||
assert.strictEqual(320, info.width);
|
assert.strictEqual(320, info.width);
|
||||||
assert.strictEqual(80, info.height);
|
assert.strictEqual(80, info.height);
|
||||||
fs.unlinkSync(fixtures.outputZoinks);
|
rimraf(fixtures.outputZoinks, done);
|
||||||
done();
|
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -578,8 +570,7 @@ describe('Input/output', function () {
|
|||||||
assert.strictEqual('png', info.format);
|
assert.strictEqual('png', info.format);
|
||||||
assert.strictEqual(320, info.width);
|
assert.strictEqual(320, info.width);
|
||||||
assert.strictEqual(80, info.height);
|
assert.strictEqual(80, info.height);
|
||||||
fs.unlinkSync(fixtures.outputZoinks);
|
rimraf(fixtures.outputZoinks, done);
|
||||||
done();
|
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -592,8 +583,7 @@ describe('Input/output', function () {
|
|||||||
assert.strictEqual('webp', info.format);
|
assert.strictEqual('webp', info.format);
|
||||||
assert.strictEqual(320, info.width);
|
assert.strictEqual(320, info.width);
|
||||||
assert.strictEqual(80, info.height);
|
assert.strictEqual(80, info.height);
|
||||||
fs.unlinkSync(fixtures.outputZoinks);
|
rimraf(fixtures.outputZoinks, done);
|
||||||
done();
|
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -606,8 +596,7 @@ describe('Input/output', function () {
|
|||||||
assert.strictEqual('tiff', info.format);
|
assert.strictEqual('tiff', info.format);
|
||||||
assert.strictEqual(320, info.width);
|
assert.strictEqual(320, info.width);
|
||||||
assert.strictEqual(80, info.height);
|
assert.strictEqual(80, info.height);
|
||||||
fs.unlinkSync(fixtures.outputZoinks);
|
rimraf(fixtures.outputZoinks, done);
|
||||||
done();
|
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -620,8 +609,7 @@ describe('Input/output', function () {
|
|||||||
assert.strictEqual('png', info.format);
|
assert.strictEqual('png', info.format);
|
||||||
assert.strictEqual(320, info.width);
|
assert.strictEqual(320, info.width);
|
||||||
assert.strictEqual(80, info.height);
|
assert.strictEqual(80, info.height);
|
||||||
fs.unlinkSync(fixtures.outputZoinks);
|
rimraf(fixtures.outputZoinks, done);
|
||||||
done();
|
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -635,8 +623,7 @@ describe('Input/output', function () {
|
|||||||
assert.strictEqual('jpeg', info.format);
|
assert.strictEqual('jpeg', info.format);
|
||||||
assert.strictEqual(320, info.width);
|
assert.strictEqual(320, info.width);
|
||||||
assert.strictEqual(80, info.height);
|
assert.strictEqual(80, info.height);
|
||||||
fs.unlinkSync(fixtures.outputZoinks);
|
rimraf(fixtures.outputZoinks, done);
|
||||||
done();
|
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
@@ -830,7 +817,7 @@ describe('Input/output', function () {
|
|||||||
assert.strictEqual(320, withInfo.width);
|
assert.strictEqual(320, withInfo.width);
|
||||||
assert.strictEqual(240, withInfo.height);
|
assert.strictEqual(240, withInfo.height);
|
||||||
// Verify image is of a different size (progressive output even without mozjpeg)
|
// Verify image is of a different size (progressive output even without mozjpeg)
|
||||||
assert.notEqual(withData.length, withoutData.length);
|
assert.notStrictEqual(withData.length, withoutData.length);
|
||||||
done();
|
done();
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
@@ -900,7 +887,7 @@ describe('Input/output', function () {
|
|||||||
it('Convert SVG to PNG at default 72DPI', function (done) {
|
it('Convert SVG to PNG at default 72DPI', function (done) {
|
||||||
sharp(fixtures.inputSvg)
|
sharp(fixtures.inputSvg)
|
||||||
.resize(1024)
|
.resize(1024)
|
||||||
.extract({left: 290, top: 760, width: 40, height: 40})
|
.extract({ left: 290, top: 760, width: 40, height: 40 })
|
||||||
.toFormat('png')
|
.toFormat('png')
|
||||||
.toBuffer(function (err, data, info) {
|
.toBuffer(function (err, data, info) {
|
||||||
if (err) throw err;
|
if (err) throw err;
|
||||||
@@ -921,7 +908,7 @@ describe('Input/output', function () {
|
|||||||
it('Convert SVG to PNG at 1200DPI', function (done) {
|
it('Convert SVG to PNG at 1200DPI', function (done) {
|
||||||
sharp(fixtures.inputSvg, { density: 1200 })
|
sharp(fixtures.inputSvg, { density: 1200 })
|
||||||
.resize(1024)
|
.resize(1024)
|
||||||
.extract({left: 290, top: 760, width: 40, height: 40})
|
.extract({ left: 290, top: 760, width: 40, height: 40 })
|
||||||
.toFormat('png')
|
.toFormat('png')
|
||||||
.toBuffer(function (err, data, info) {
|
.toBuffer(function (err, data, info) {
|
||||||
if (err) throw err;
|
if (err) throw err;
|
||||||
@@ -939,6 +926,21 @@ describe('Input/output', function () {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
it('Convert SVG to PNG at 14.4DPI', function (done) {
|
||||||
|
sharp(fixtures.inputSvg, { density: 14.4 })
|
||||||
|
.toFormat('png')
|
||||||
|
.toBuffer(function (err, data, info) {
|
||||||
|
if (err) throw err;
|
||||||
|
assert.strictEqual('png', info.format);
|
||||||
|
assert.strictEqual(20, info.width);
|
||||||
|
assert.strictEqual(20, info.height);
|
||||||
|
fixtures.assertSimilar(fixtures.expected('svg14.4.png'), data, function (err) {
|
||||||
|
if (err) throw err;
|
||||||
|
done();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
it('Convert SVG with embedded images to PNG, respecting dimensions, autoconvert to PNG', function (done) {
|
it('Convert SVG with embedded images to PNG, respecting dimensions, autoconvert to PNG', function (done) {
|
||||||
sharp(fixtures.inputSvgWithEmbeddedImages)
|
sharp(fixtures.inputSvgWithEmbeddedImages)
|
||||||
.toBuffer(function (err, data, info) {
|
.toBuffer(function (err, data, info) {
|
||||||
@@ -1065,7 +1067,7 @@ describe('Input/output', function () {
|
|||||||
if (err) throw err;
|
if (err) throw err;
|
||||||
assert.strictEqual('tiff', info.format);
|
assert.strictEqual('tiff', info.format);
|
||||||
assert(info.size === startSize);
|
assert(info.size === startSize);
|
||||||
fs.unlink(fixtures.outputTiff, done);
|
rimraf(fixtures.outputTiff, done);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -1082,7 +1084,7 @@ describe('Input/output', function () {
|
|||||||
if (err) throw err;
|
if (err) throw err;
|
||||||
assert.strictEqual('tiff', info.format);
|
assert.strictEqual('tiff', info.format);
|
||||||
assert(info.size < (startSize / 2));
|
assert(info.size < (startSize / 2));
|
||||||
fs.unlink(fixtures.outputTiff, done);
|
rimraf(fixtures.outputTiff, done);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -1105,7 +1107,7 @@ describe('Input/output', function () {
|
|||||||
sharp(fixtures.outputTiff).metadata(function (err, metadata) {
|
sharp(fixtures.outputTiff).metadata(function (err, metadata) {
|
||||||
if (err) throw err;
|
if (err) throw err;
|
||||||
assert.strictEqual(metadata.density, res * 2.54); // convert to dpi
|
assert.strictEqual(metadata.density, res * 2.54); // convert to dpi
|
||||||
fs.unlink(fixtures.outputTiff, done);
|
rimraf(fixtures.outputTiff, done);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
@@ -1150,7 +1152,7 @@ describe('Input/output', function () {
|
|||||||
if (err) throw err;
|
if (err) throw err;
|
||||||
assert.strictEqual('tiff', info.format);
|
assert.strictEqual('tiff', info.format);
|
||||||
assert(info.size < startSize);
|
assert(info.size < startSize);
|
||||||
fs.unlink(fixtures.outputTiff, done);
|
rimraf(fixtures.outputTiff, done);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -1166,7 +1168,7 @@ describe('Input/output', function () {
|
|||||||
if (err) throw err;
|
if (err) throw err;
|
||||||
assert.strictEqual('tiff', info.format);
|
assert.strictEqual('tiff', info.format);
|
||||||
assert(info.size < startSize);
|
assert(info.size < startSize);
|
||||||
fs.unlink(fixtures.outputTiff, done);
|
rimraf(fixtures.outputTiff, done);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -1181,7 +1183,7 @@ describe('Input/output', function () {
|
|||||||
if (err) throw err;
|
if (err) throw err;
|
||||||
assert.strictEqual('tiff', info.format);
|
assert.strictEqual('tiff', info.format);
|
||||||
assert(info.size < startSize);
|
assert(info.size < startSize);
|
||||||
fs.unlink(fixtures.outputTiff, done);
|
rimraf(fixtures.outputTiff, done);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -1196,7 +1198,7 @@ describe('Input/output', function () {
|
|||||||
if (err) throw err;
|
if (err) throw err;
|
||||||
assert.strictEqual('tiff', info.format);
|
assert.strictEqual('tiff', info.format);
|
||||||
assert(info.size < startSize);
|
assert(info.size < startSize);
|
||||||
fs.unlink(fixtures.outputTiff, done);
|
rimraf(fixtures.outputTiff, done);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -1211,7 +1213,7 @@ describe('Input/output', function () {
|
|||||||
if (err) throw err;
|
if (err) throw err;
|
||||||
assert.strictEqual('tiff', info.format);
|
assert.strictEqual('tiff', info.format);
|
||||||
assert(info.size < startSize);
|
assert(info.size < startSize);
|
||||||
fs.unlink(fixtures.outputTiff, done);
|
rimraf(fixtures.outputTiff, done);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -1225,7 +1227,7 @@ describe('Input/output', function () {
|
|||||||
if (err) throw err;
|
if (err) throw err;
|
||||||
assert.strictEqual('tiff', info.format);
|
assert.strictEqual('tiff', info.format);
|
||||||
assert(info.size < startSize);
|
assert(info.size < startSize);
|
||||||
fs.unlink(fixtures.outputTiff, done);
|
rimraf(fixtures.outputTiff, done);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -1283,6 +1285,84 @@ describe('Input/output', function () {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
it('TIFF tiled pyramid image without compression enlarges test file', function (done) {
|
||||||
|
const startSize = fs.statSync(fixtures.inputTiffUncompressed).size;
|
||||||
|
sharp(fixtures.inputTiffUncompressed)
|
||||||
|
.tiff({
|
||||||
|
compression: 'none',
|
||||||
|
pyramid: true,
|
||||||
|
tile: true,
|
||||||
|
tileHeight: 256,
|
||||||
|
tileWidth: 256
|
||||||
|
})
|
||||||
|
.toFile(fixtures.outputTiff, (err, info) => {
|
||||||
|
if (err) throw err;
|
||||||
|
assert.strictEqual('tiff', info.format);
|
||||||
|
assert(info.size > startSize);
|
||||||
|
rimraf(fixtures.outputTiff, done);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('TIFF pyramid true value does not throw error', function () {
|
||||||
|
assert.doesNotThrow(function () {
|
||||||
|
sharp().tiff({ pyramid: true });
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('Invalid TIFF pyramid value throws error', function () {
|
||||||
|
assert.throws(function () {
|
||||||
|
sharp().tiff({ pyramid: 'true' });
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('Invalid TIFF tile value throws error', function () {
|
||||||
|
assert.throws(function () {
|
||||||
|
sharp().tiff({ tile: 'true' });
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('TIFF tile true value does not throw error', function () {
|
||||||
|
assert.doesNotThrow(function () {
|
||||||
|
sharp().tiff({ tile: true });
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('Valid TIFF tileHeight value does not throw error', function () {
|
||||||
|
assert.doesNotThrow(function () {
|
||||||
|
sharp().tiff({ tileHeight: 512 });
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('Valid TIFF tileWidth value does not throw error', function () {
|
||||||
|
assert.doesNotThrow(function () {
|
||||||
|
sharp().tiff({ tileWidth: 512 });
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('Invalid TIFF tileHeight value throws error', function () {
|
||||||
|
assert.throws(function () {
|
||||||
|
sharp().tiff({ tileHeight: '256' });
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('Invalid TIFF tileWidth value throws error', function () {
|
||||||
|
assert.throws(function () {
|
||||||
|
sharp().tiff({ tileWidth: '256' });
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('Invalid TIFF tileHeight value throws error', function () {
|
||||||
|
assert.throws(function () {
|
||||||
|
sharp().tiff({ tileHeight: 0 });
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('Invalid TIFF tileWidth value throws error', function () {
|
||||||
|
assert.throws(function () {
|
||||||
|
sharp().tiff({ tileWidth: 0 });
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
it('Input and output formats match when not forcing', function (done) {
|
it('Input and output formats match when not forcing', function (done) {
|
||||||
sharp(fixtures.inputJpg)
|
sharp(fixtures.inputJpg)
|
||||||
.resize(320, 240)
|
.resize(320, 240)
|
||||||
@@ -1362,15 +1442,14 @@ describe('Input/output', function () {
|
|||||||
|
|
||||||
it('Save Vips V file', function (done) {
|
it('Save Vips V file', function (done) {
|
||||||
sharp(fixtures.inputJpg)
|
sharp(fixtures.inputJpg)
|
||||||
.extract({left: 910, top: 1105, width: 70, height: 60})
|
.extract({ left: 910, top: 1105, width: 70, height: 60 })
|
||||||
.toFile(fixtures.outputV, function (err, info) {
|
.toFile(fixtures.outputV, function (err, info) {
|
||||||
if (err) throw err;
|
if (err) throw err;
|
||||||
assert.strictEqual(true, info.size > 0);
|
assert.strictEqual(true, info.size > 0);
|
||||||
assert.strictEqual('v', info.format);
|
assert.strictEqual('v', info.format);
|
||||||
assert.strictEqual(70, info.width);
|
assert.strictEqual(70, info.width);
|
||||||
assert.strictEqual(60, info.height);
|
assert.strictEqual(60, info.height);
|
||||||
fs.unlinkSync(fixtures.outputV);
|
rimraf(fixtures.outputV, done);
|
||||||
done();
|
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -1387,6 +1466,7 @@ describe('Input/output', function () {
|
|||||||
assert.strictEqual('raw', info.format);
|
assert.strictEqual('raw', info.format);
|
||||||
assert.strictEqual(32, info.width);
|
assert.strictEqual(32, info.width);
|
||||||
assert.strictEqual(24, info.height);
|
assert.strictEqual(24, info.height);
|
||||||
|
assert.strictEqual(1, info.channels);
|
||||||
done();
|
done();
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
@@ -1506,11 +1586,6 @@ describe('Input/output', function () {
|
|||||||
sharp(null, { density: 'zoinks' });
|
sharp(null, { density: 'zoinks' });
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
it('Invalid density: float', function () {
|
|
||||||
assert.throws(function () {
|
|
||||||
sharp(null, { density: 0.5 });
|
|
||||||
});
|
|
||||||
});
|
|
||||||
it('Ignore unknown attribute', function () {
|
it('Ignore unknown attribute', function () {
|
||||||
sharp(null, { unknown: true });
|
sharp(null, { unknown: true });
|
||||||
});
|
});
|
||||||
@@ -1558,7 +1633,7 @@ describe('Input/output', function () {
|
|||||||
width: info.width,
|
width: info.width,
|
||||||
height: info.height,
|
height: info.height,
|
||||||
channels: info.channels
|
channels: info.channels
|
||||||
}})
|
} })
|
||||||
.jpeg()
|
.jpeg()
|
||||||
.toBuffer(function (err, data, info) {
|
.toBuffer(function (err, data, info) {
|
||||||
if (err) throw err;
|
if (err) throw err;
|
||||||
@@ -1585,7 +1660,7 @@ describe('Input/output', function () {
|
|||||||
width: info.width,
|
width: info.width,
|
||||||
height: info.height,
|
height: info.height,
|
||||||
channels: info.channels
|
channels: info.channels
|
||||||
}})
|
} })
|
||||||
.png()
|
.png()
|
||||||
.toBuffer(function (err, data, info) {
|
.toBuffer(function (err, data, info) {
|
||||||
if (err) throw err;
|
if (err) throw err;
|
||||||
@@ -1687,7 +1762,7 @@ describe('Input/output', function () {
|
|||||||
assert.strictEqual(472, info.height);
|
assert.strictEqual(472, info.height);
|
||||||
assert.strictEqual(3, info.channels);
|
assert.strictEqual(3, info.channels);
|
||||||
});
|
});
|
||||||
const badPipeline = sharp(null, {raw: {width: 840, height: 500, channels: 3}})
|
const badPipeline = sharp(null, { raw: { width: 840, height: 500, channels: 3 } })
|
||||||
.toFormat('jpeg')
|
.toFormat('jpeg')
|
||||||
.toBuffer(function (err, data, info) {
|
.toBuffer(function (err, data, info) {
|
||||||
assert.strictEqual(err.message.indexOf('memory area too small') > 0, true);
|
assert.strictEqual(err.message.indexOf('memory area too small') > 0, true);
|
||||||
@@ -1695,7 +1770,7 @@ describe('Input/output', function () {
|
|||||||
const inPipeline = sharp()
|
const inPipeline = sharp()
|
||||||
.resize(840, 472)
|
.resize(840, 472)
|
||||||
.raw();
|
.raw();
|
||||||
const goodPipeline = sharp(null, {raw: {width: 840, height: 472, channels: 3}})
|
const goodPipeline = sharp(null, { raw: { width: 840, height: 472, channels: 3 } })
|
||||||
.toFormat('jpeg')
|
.toFormat('jpeg')
|
||||||
.toBuffer(function (err, data, info) {
|
.toBuffer(function (err, data, info) {
|
||||||
if (err) throw err;
|
if (err) throw err;
|
||||||
|
|||||||
@@ -138,7 +138,7 @@ describe('Image channel insertion', function () {
|
|||||||
|
|
||||||
it('Invalid raw buffer description', function () {
|
it('Invalid raw buffer description', function () {
|
||||||
assert.throws(function () {
|
assert.throws(function () {
|
||||||
sharp().joinChannel(fs.readFileSync(fixtures.inputPng), {raw: {}});
|
sharp().joinChannel(fs.readFileSync(fixtures.inputPng), { raw: {} });
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|||||||
@@ -4,6 +4,7 @@ const assert = require('assert');
|
|||||||
const fs = require('fs');
|
const fs = require('fs');
|
||||||
const semver = require('semver');
|
const semver = require('semver');
|
||||||
const libvips = require('../../lib/libvips');
|
const libvips = require('../../lib/libvips');
|
||||||
|
const mockFS = require('mock-fs');
|
||||||
|
|
||||||
const originalPlatform = process.platform;
|
const originalPlatform = process.platform;
|
||||||
|
|
||||||
@@ -74,4 +75,34 @@ describe('libvips binaries', function () {
|
|||||||
assert.strictEqual(true, fs.existsSync(cachePath));
|
assert.strictEqual(true, fs.existsSync(cachePath));
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
describe('safe directory creation', function () {
|
||||||
|
before(function () {
|
||||||
|
mockFS({
|
||||||
|
exampleDirA: {
|
||||||
|
exampleDirB: {
|
||||||
|
exampleFile: 'Example test file'
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
after(function () { mockFS.restore(); });
|
||||||
|
|
||||||
|
it('mkdirSync creates a directory', function () {
|
||||||
|
const dirPath = 'createdDir';
|
||||||
|
|
||||||
|
libvips.mkdirSync(dirPath);
|
||||||
|
assert.strictEqual(true, fs.existsSync(dirPath));
|
||||||
|
});
|
||||||
|
it('mkdirSync does not throw error or overwrite an existing dir', function () {
|
||||||
|
const dirPath = 'exampleDirA';
|
||||||
|
const nestedDirPath = 'exampleDirA/exampleDirB';
|
||||||
|
assert.strictEqual(true, fs.existsSync(dirPath));
|
||||||
|
|
||||||
|
libvips.mkdirSync(dirPath);
|
||||||
|
|
||||||
|
assert.strictEqual(true, fs.existsSync(dirPath));
|
||||||
|
assert.strictEqual(true, fs.existsSync(nestedDirPath));
|
||||||
|
});
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -13,12 +13,15 @@ describe('Image metadata', function () {
|
|||||||
sharp(fixtures.inputJpg).metadata(function (err, metadata) {
|
sharp(fixtures.inputJpg).metadata(function (err, metadata) {
|
||||||
if (err) throw err;
|
if (err) throw err;
|
||||||
assert.strictEqual('jpeg', metadata.format);
|
assert.strictEqual('jpeg', metadata.format);
|
||||||
|
assert.strictEqual('undefined', typeof metadata.size);
|
||||||
assert.strictEqual(2725, metadata.width);
|
assert.strictEqual(2725, metadata.width);
|
||||||
assert.strictEqual(2225, metadata.height);
|
assert.strictEqual(2225, metadata.height);
|
||||||
assert.strictEqual('srgb', metadata.space);
|
assert.strictEqual('srgb', metadata.space);
|
||||||
assert.strictEqual(3, metadata.channels);
|
assert.strictEqual(3, metadata.channels);
|
||||||
assert.strictEqual('uchar', metadata.depth);
|
assert.strictEqual('uchar', metadata.depth);
|
||||||
assert.strictEqual('undefined', typeof metadata.density);
|
assert.strictEqual('undefined', typeof metadata.density);
|
||||||
|
assert.strictEqual('4:2:0', metadata.chromaSubsampling);
|
||||||
|
assert.strictEqual(false, metadata.isProgressive);
|
||||||
assert.strictEqual(false, metadata.hasProfile);
|
assert.strictEqual(false, metadata.hasProfile);
|
||||||
assert.strictEqual(false, metadata.hasAlpha);
|
assert.strictEqual(false, metadata.hasAlpha);
|
||||||
assert.strictEqual('undefined', typeof metadata.orientation);
|
assert.strictEqual('undefined', typeof metadata.orientation);
|
||||||
@@ -32,12 +35,15 @@ describe('Image metadata', function () {
|
|||||||
sharp(fixtures.inputJpgWithExif).metadata(function (err, metadata) {
|
sharp(fixtures.inputJpgWithExif).metadata(function (err, metadata) {
|
||||||
if (err) throw err;
|
if (err) throw err;
|
||||||
assert.strictEqual('jpeg', metadata.format);
|
assert.strictEqual('jpeg', metadata.format);
|
||||||
|
assert.strictEqual('undefined', typeof metadata.size);
|
||||||
assert.strictEqual(450, metadata.width);
|
assert.strictEqual(450, metadata.width);
|
||||||
assert.strictEqual(600, metadata.height);
|
assert.strictEqual(600, metadata.height);
|
||||||
assert.strictEqual('srgb', metadata.space);
|
assert.strictEqual('srgb', metadata.space);
|
||||||
assert.strictEqual(3, metadata.channels);
|
assert.strictEqual(3, metadata.channels);
|
||||||
assert.strictEqual('uchar', metadata.depth);
|
assert.strictEqual('uchar', metadata.depth);
|
||||||
assert.strictEqual(72, metadata.density);
|
assert.strictEqual(72, metadata.density);
|
||||||
|
assert.strictEqual('4:2:0', metadata.chromaSubsampling);
|
||||||
|
assert.strictEqual(false, metadata.isProgressive);
|
||||||
assert.strictEqual(true, metadata.hasProfile);
|
assert.strictEqual(true, metadata.hasProfile);
|
||||||
assert.strictEqual(false, metadata.hasAlpha);
|
assert.strictEqual(false, metadata.hasAlpha);
|
||||||
assert.strictEqual(8, metadata.orientation);
|
assert.strictEqual(8, metadata.orientation);
|
||||||
@@ -79,12 +85,15 @@ describe('Image metadata', function () {
|
|||||||
sharp(fixtures.inputTiff).metadata(function (err, metadata) {
|
sharp(fixtures.inputTiff).metadata(function (err, metadata) {
|
||||||
if (err) throw err;
|
if (err) throw err;
|
||||||
assert.strictEqual('tiff', metadata.format);
|
assert.strictEqual('tiff', metadata.format);
|
||||||
|
assert.strictEqual('undefined', typeof metadata.size);
|
||||||
assert.strictEqual(2464, metadata.width);
|
assert.strictEqual(2464, metadata.width);
|
||||||
assert.strictEqual(3248, metadata.height);
|
assert.strictEqual(3248, metadata.height);
|
||||||
assert.strictEqual('b-w', metadata.space);
|
assert.strictEqual('b-w', metadata.space);
|
||||||
assert.strictEqual(1, metadata.channels);
|
assert.strictEqual(1, metadata.channels);
|
||||||
assert.strictEqual('uchar', metadata.depth);
|
assert.strictEqual('uchar', metadata.depth);
|
||||||
assert.strictEqual(300, metadata.density);
|
assert.strictEqual(300, metadata.density);
|
||||||
|
assert.strictEqual('undefined', typeof metadata.chromaSubsampling);
|
||||||
|
assert.strictEqual(false, metadata.isProgressive);
|
||||||
assert.strictEqual(false, metadata.hasProfile);
|
assert.strictEqual(false, metadata.hasProfile);
|
||||||
assert.strictEqual(false, metadata.hasAlpha);
|
assert.strictEqual(false, metadata.hasAlpha);
|
||||||
assert.strictEqual(1, metadata.orientation);
|
assert.strictEqual(1, metadata.orientation);
|
||||||
@@ -98,12 +107,15 @@ describe('Image metadata', function () {
|
|||||||
sharp(fixtures.inputPng).metadata(function (err, metadata) {
|
sharp(fixtures.inputPng).metadata(function (err, metadata) {
|
||||||
if (err) throw err;
|
if (err) throw err;
|
||||||
assert.strictEqual('png', metadata.format);
|
assert.strictEqual('png', metadata.format);
|
||||||
|
assert.strictEqual('undefined', typeof metadata.size);
|
||||||
assert.strictEqual(2809, metadata.width);
|
assert.strictEqual(2809, metadata.width);
|
||||||
assert.strictEqual(2074, metadata.height);
|
assert.strictEqual(2074, metadata.height);
|
||||||
assert.strictEqual('b-w', metadata.space);
|
assert.strictEqual('b-w', metadata.space);
|
||||||
assert.strictEqual(1, metadata.channels);
|
assert.strictEqual(1, metadata.channels);
|
||||||
assert.strictEqual('uchar', metadata.depth);
|
assert.strictEqual('uchar', metadata.depth);
|
||||||
assert.strictEqual(300, metadata.density);
|
assert.strictEqual(300, metadata.density);
|
||||||
|
assert.strictEqual('undefined', typeof metadata.chromaSubsampling);
|
||||||
|
assert.strictEqual(false, metadata.isProgressive);
|
||||||
assert.strictEqual(false, metadata.hasProfile);
|
assert.strictEqual(false, metadata.hasProfile);
|
||||||
assert.strictEqual(false, metadata.hasAlpha);
|
assert.strictEqual(false, metadata.hasAlpha);
|
||||||
assert.strictEqual('undefined', typeof metadata.orientation);
|
assert.strictEqual('undefined', typeof metadata.orientation);
|
||||||
@@ -117,12 +129,15 @@ describe('Image metadata', function () {
|
|||||||
sharp(fixtures.inputPngWithTransparency).metadata(function (err, metadata) {
|
sharp(fixtures.inputPngWithTransparency).metadata(function (err, metadata) {
|
||||||
if (err) throw err;
|
if (err) throw err;
|
||||||
assert.strictEqual('png', metadata.format);
|
assert.strictEqual('png', metadata.format);
|
||||||
|
assert.strictEqual('undefined', typeof metadata.size);
|
||||||
assert.strictEqual(2048, metadata.width);
|
assert.strictEqual(2048, metadata.width);
|
||||||
assert.strictEqual(1536, metadata.height);
|
assert.strictEqual(1536, metadata.height);
|
||||||
assert.strictEqual('srgb', metadata.space);
|
assert.strictEqual('srgb', metadata.space);
|
||||||
assert.strictEqual(4, metadata.channels);
|
assert.strictEqual(4, metadata.channels);
|
||||||
assert.strictEqual('uchar', metadata.depth);
|
assert.strictEqual('uchar', metadata.depth);
|
||||||
assert.strictEqual(72, metadata.density);
|
assert.strictEqual(72, metadata.density);
|
||||||
|
assert.strictEqual('undefined', typeof metadata.chromaSubsampling);
|
||||||
|
assert.strictEqual(false, metadata.isProgressive);
|
||||||
assert.strictEqual(false, metadata.hasProfile);
|
assert.strictEqual(false, metadata.hasProfile);
|
||||||
assert.strictEqual(true, metadata.hasAlpha);
|
assert.strictEqual(true, metadata.hasAlpha);
|
||||||
assert.strictEqual('undefined', typeof metadata.orientation);
|
assert.strictEqual('undefined', typeof metadata.orientation);
|
||||||
@@ -136,12 +151,15 @@ describe('Image metadata', function () {
|
|||||||
sharp(fixtures.inputWebP).metadata(function (err, metadata) {
|
sharp(fixtures.inputWebP).metadata(function (err, metadata) {
|
||||||
if (err) throw err;
|
if (err) throw err;
|
||||||
assert.strictEqual('webp', metadata.format);
|
assert.strictEqual('webp', metadata.format);
|
||||||
|
assert.strictEqual('undefined', typeof metadata.size);
|
||||||
assert.strictEqual(1024, metadata.width);
|
assert.strictEqual(1024, metadata.width);
|
||||||
assert.strictEqual(772, metadata.height);
|
assert.strictEqual(772, metadata.height);
|
||||||
assert.strictEqual('srgb', metadata.space);
|
assert.strictEqual('srgb', metadata.space);
|
||||||
assert.strictEqual(3, metadata.channels);
|
assert.strictEqual(3, metadata.channels);
|
||||||
assert.strictEqual('uchar', metadata.depth);
|
assert.strictEqual('uchar', metadata.depth);
|
||||||
assert.strictEqual('undefined', typeof metadata.density);
|
assert.strictEqual('undefined', typeof metadata.density);
|
||||||
|
assert.strictEqual('undefined', typeof metadata.chromaSubsampling);
|
||||||
|
assert.strictEqual(false, metadata.isProgressive);
|
||||||
assert.strictEqual(false, metadata.hasProfile);
|
assert.strictEqual(false, metadata.hasProfile);
|
||||||
assert.strictEqual(false, metadata.hasAlpha);
|
assert.strictEqual(false, metadata.hasAlpha);
|
||||||
assert.strictEqual('undefined', typeof metadata.orientation);
|
assert.strictEqual('undefined', typeof metadata.orientation);
|
||||||
@@ -155,11 +173,14 @@ describe('Image metadata', function () {
|
|||||||
sharp(fixtures.inputGif).metadata(function (err, metadata) {
|
sharp(fixtures.inputGif).metadata(function (err, metadata) {
|
||||||
if (err) throw err;
|
if (err) throw err;
|
||||||
assert.strictEqual('gif', metadata.format);
|
assert.strictEqual('gif', metadata.format);
|
||||||
|
assert.strictEqual('undefined', typeof metadata.size);
|
||||||
assert.strictEqual(800, metadata.width);
|
assert.strictEqual(800, metadata.width);
|
||||||
assert.strictEqual(533, metadata.height);
|
assert.strictEqual(533, metadata.height);
|
||||||
assert.strictEqual(3, metadata.channels);
|
assert.strictEqual(3, metadata.channels);
|
||||||
assert.strictEqual('uchar', metadata.depth);
|
assert.strictEqual('uchar', metadata.depth);
|
||||||
assert.strictEqual('undefined', typeof metadata.density);
|
assert.strictEqual('undefined', typeof metadata.density);
|
||||||
|
assert.strictEqual('undefined', typeof metadata.chromaSubsampling);
|
||||||
|
assert.strictEqual(false, metadata.isProgressive);
|
||||||
assert.strictEqual(false, metadata.hasProfile);
|
assert.strictEqual(false, metadata.hasProfile);
|
||||||
assert.strictEqual(false, metadata.hasAlpha);
|
assert.strictEqual(false, metadata.hasAlpha);
|
||||||
assert.strictEqual('undefined', typeof metadata.orientation);
|
assert.strictEqual('undefined', typeof metadata.orientation);
|
||||||
@@ -172,11 +193,14 @@ describe('Image metadata', function () {
|
|||||||
sharp(fixtures.inputGifGreyPlusAlpha).metadata(function (err, metadata) {
|
sharp(fixtures.inputGifGreyPlusAlpha).metadata(function (err, metadata) {
|
||||||
if (err) throw err;
|
if (err) throw err;
|
||||||
assert.strictEqual('gif', metadata.format);
|
assert.strictEqual('gif', metadata.format);
|
||||||
|
assert.strictEqual('undefined', typeof metadata.size);
|
||||||
assert.strictEqual(2, metadata.width);
|
assert.strictEqual(2, metadata.width);
|
||||||
assert.strictEqual(1, metadata.height);
|
assert.strictEqual(1, metadata.height);
|
||||||
assert.strictEqual(2, metadata.channels);
|
assert.strictEqual(2, metadata.channels);
|
||||||
assert.strictEqual('uchar', metadata.depth);
|
assert.strictEqual('uchar', metadata.depth);
|
||||||
assert.strictEqual('undefined', typeof metadata.density);
|
assert.strictEqual('undefined', typeof metadata.density);
|
||||||
|
assert.strictEqual('undefined', typeof metadata.chromaSubsampling);
|
||||||
|
assert.strictEqual(false, metadata.isProgressive);
|
||||||
assert.strictEqual(false, metadata.hasProfile);
|
assert.strictEqual(false, metadata.hasProfile);
|
||||||
assert.strictEqual(true, metadata.hasAlpha);
|
assert.strictEqual(true, metadata.hasAlpha);
|
||||||
assert.strictEqual('undefined', typeof metadata.orientation);
|
assert.strictEqual('undefined', typeof metadata.orientation);
|
||||||
@@ -189,12 +213,15 @@ describe('Image metadata', function () {
|
|||||||
it('File in, Promise out', function (done) {
|
it('File in, Promise out', function (done) {
|
||||||
sharp(fixtures.inputJpg).metadata().then(function (metadata) {
|
sharp(fixtures.inputJpg).metadata().then(function (metadata) {
|
||||||
assert.strictEqual('jpeg', metadata.format);
|
assert.strictEqual('jpeg', metadata.format);
|
||||||
|
assert.strictEqual('undefined', typeof metadata.size);
|
||||||
assert.strictEqual(2725, metadata.width);
|
assert.strictEqual(2725, metadata.width);
|
||||||
assert.strictEqual(2225, metadata.height);
|
assert.strictEqual(2225, metadata.height);
|
||||||
assert.strictEqual('srgb', metadata.space);
|
assert.strictEqual('srgb', metadata.space);
|
||||||
assert.strictEqual(3, metadata.channels);
|
assert.strictEqual(3, metadata.channels);
|
||||||
assert.strictEqual('uchar', metadata.depth);
|
assert.strictEqual('uchar', metadata.depth);
|
||||||
assert.strictEqual('undefined', typeof metadata.density);
|
assert.strictEqual('undefined', typeof metadata.density);
|
||||||
|
assert.strictEqual('4:2:0', metadata.chromaSubsampling);
|
||||||
|
assert.strictEqual(false, metadata.isProgressive);
|
||||||
assert.strictEqual(false, metadata.hasProfile);
|
assert.strictEqual(false, metadata.hasProfile);
|
||||||
assert.strictEqual(false, metadata.hasAlpha);
|
assert.strictEqual(false, metadata.hasAlpha);
|
||||||
assert.strictEqual('undefined', typeof metadata.orientation);
|
assert.strictEqual('undefined', typeof metadata.orientation);
|
||||||
@@ -218,21 +245,22 @@ describe('Image metadata', function () {
|
|||||||
const pipeline = sharp();
|
const pipeline = sharp();
|
||||||
pipeline.metadata().then(function (metadata) {
|
pipeline.metadata().then(function (metadata) {
|
||||||
assert.strictEqual('jpeg', metadata.format);
|
assert.strictEqual('jpeg', metadata.format);
|
||||||
|
assert.strictEqual(829183, metadata.size);
|
||||||
assert.strictEqual(2725, metadata.width);
|
assert.strictEqual(2725, metadata.width);
|
||||||
assert.strictEqual(2225, metadata.height);
|
assert.strictEqual(2225, metadata.height);
|
||||||
assert.strictEqual('srgb', metadata.space);
|
assert.strictEqual('srgb', metadata.space);
|
||||||
assert.strictEqual(3, metadata.channels);
|
assert.strictEqual(3, metadata.channels);
|
||||||
assert.strictEqual('uchar', metadata.depth);
|
assert.strictEqual('uchar', metadata.depth);
|
||||||
assert.strictEqual('undefined', typeof metadata.density);
|
assert.strictEqual('undefined', typeof metadata.density);
|
||||||
|
assert.strictEqual('4:2:0', metadata.chromaSubsampling);
|
||||||
|
assert.strictEqual(false, metadata.isProgressive);
|
||||||
assert.strictEqual(false, metadata.hasProfile);
|
assert.strictEqual(false, metadata.hasProfile);
|
||||||
assert.strictEqual(false, metadata.hasAlpha);
|
assert.strictEqual(false, metadata.hasAlpha);
|
||||||
assert.strictEqual('undefined', typeof metadata.orientation);
|
assert.strictEqual('undefined', typeof metadata.orientation);
|
||||||
assert.strictEqual('undefined', typeof metadata.exif);
|
assert.strictEqual('undefined', typeof metadata.exif);
|
||||||
assert.strictEqual('undefined', typeof metadata.icc);
|
assert.strictEqual('undefined', typeof metadata.icc);
|
||||||
done();
|
done();
|
||||||
}).catch(function (err) {
|
}).catch(done);
|
||||||
throw err;
|
|
||||||
});
|
|
||||||
readable.pipe(pipeline);
|
readable.pipe(pipeline);
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -241,12 +269,15 @@ describe('Image metadata', function () {
|
|||||||
const pipeline = sharp().metadata(function (err, metadata) {
|
const pipeline = sharp().metadata(function (err, metadata) {
|
||||||
if (err) throw err;
|
if (err) throw err;
|
||||||
assert.strictEqual('jpeg', metadata.format);
|
assert.strictEqual('jpeg', metadata.format);
|
||||||
|
assert.strictEqual(829183, metadata.size);
|
||||||
assert.strictEqual(2725, metadata.width);
|
assert.strictEqual(2725, metadata.width);
|
||||||
assert.strictEqual(2225, metadata.height);
|
assert.strictEqual(2225, metadata.height);
|
||||||
assert.strictEqual('srgb', metadata.space);
|
assert.strictEqual('srgb', metadata.space);
|
||||||
assert.strictEqual(3, metadata.channels);
|
assert.strictEqual(3, metadata.channels);
|
||||||
assert.strictEqual('uchar', metadata.depth);
|
assert.strictEqual('uchar', metadata.depth);
|
||||||
assert.strictEqual('undefined', typeof metadata.density);
|
assert.strictEqual('undefined', typeof metadata.density);
|
||||||
|
assert.strictEqual('4:2:0', metadata.chromaSubsampling);
|
||||||
|
assert.strictEqual(false, metadata.isProgressive);
|
||||||
assert.strictEqual(false, metadata.hasProfile);
|
assert.strictEqual(false, metadata.hasProfile);
|
||||||
assert.strictEqual(false, metadata.hasAlpha);
|
assert.strictEqual(false, metadata.hasAlpha);
|
||||||
assert.strictEqual('undefined', typeof metadata.orientation);
|
assert.strictEqual('undefined', typeof metadata.orientation);
|
||||||
@@ -262,12 +293,15 @@ describe('Image metadata', function () {
|
|||||||
image.metadata(function (err, metadata) {
|
image.metadata(function (err, metadata) {
|
||||||
if (err) throw err;
|
if (err) throw err;
|
||||||
assert.strictEqual('jpeg', metadata.format);
|
assert.strictEqual('jpeg', metadata.format);
|
||||||
|
assert.strictEqual('undefined', typeof metadata.size);
|
||||||
assert.strictEqual(2725, metadata.width);
|
assert.strictEqual(2725, metadata.width);
|
||||||
assert.strictEqual(2225, metadata.height);
|
assert.strictEqual(2225, metadata.height);
|
||||||
assert.strictEqual('srgb', metadata.space);
|
assert.strictEqual('srgb', metadata.space);
|
||||||
assert.strictEqual(3, metadata.channels);
|
assert.strictEqual(3, metadata.channels);
|
||||||
assert.strictEqual('uchar', metadata.depth);
|
assert.strictEqual('uchar', metadata.depth);
|
||||||
assert.strictEqual('undefined', typeof metadata.density);
|
assert.strictEqual('undefined', typeof metadata.density);
|
||||||
|
assert.strictEqual('4:2:0', metadata.chromaSubsampling);
|
||||||
|
assert.strictEqual(false, metadata.isProgressive);
|
||||||
assert.strictEqual(false, metadata.hasProfile);
|
assert.strictEqual(false, metadata.hasProfile);
|
||||||
assert.strictEqual(false, metadata.hasAlpha);
|
assert.strictEqual(false, metadata.hasAlpha);
|
||||||
assert.strictEqual('undefined', typeof metadata.orientation);
|
assert.strictEqual('undefined', typeof metadata.orientation);
|
||||||
@@ -346,6 +380,56 @@ describe('Image metadata', function () {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
it('chromaSubsampling 4:4:4:4 CMYK JPEG', function () {
|
||||||
|
return sharp(fixtures.inputJpgWithCmykProfile)
|
||||||
|
.metadata()
|
||||||
|
.then(function (metadata) {
|
||||||
|
assert.strictEqual('4:4:4:4', metadata.chromaSubsampling);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('chromaSubsampling 4:4:4 RGB JPEG', function () {
|
||||||
|
return sharp(fixtures.inputJpg)
|
||||||
|
.resize(10, 10)
|
||||||
|
.jpeg({ chromaSubsampling: '4:4:4' })
|
||||||
|
.toBuffer()
|
||||||
|
.then(function (data) {
|
||||||
|
return sharp(data)
|
||||||
|
.metadata()
|
||||||
|
.then(function (metadata) {
|
||||||
|
assert.strictEqual('4:4:4', metadata.chromaSubsampling);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('isProgressive JPEG', function () {
|
||||||
|
return sharp(fixtures.inputJpg)
|
||||||
|
.resize(10, 10)
|
||||||
|
.jpeg({ progressive: true })
|
||||||
|
.toBuffer()
|
||||||
|
.then(function (data) {
|
||||||
|
return sharp(data)
|
||||||
|
.metadata()
|
||||||
|
.then(function (metadata) {
|
||||||
|
assert.strictEqual(true, metadata.isProgressive);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('isProgressive PNG', function () {
|
||||||
|
return sharp(fixtures.inputJpg)
|
||||||
|
.resize(10, 10)
|
||||||
|
.png({ progressive: true })
|
||||||
|
.toBuffer()
|
||||||
|
.then(function (data) {
|
||||||
|
return sharp(data)
|
||||||
|
.metadata()
|
||||||
|
.then(function (metadata) {
|
||||||
|
assert.strictEqual(true, metadata.isProgressive);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
it('File input with corrupt header fails gracefully', function (done) {
|
it('File input with corrupt header fails gracefully', function (done) {
|
||||||
sharp(fixtures.inputJpgWithCorruptHeader)
|
sharp(fixtures.inputJpgWithCorruptHeader)
|
||||||
.metadata(function (err) {
|
.metadata(function (err) {
|
||||||
@@ -365,12 +449,12 @@ describe('Image metadata', function () {
|
|||||||
describe('Invalid withMetadata parameters', function () {
|
describe('Invalid withMetadata parameters', function () {
|
||||||
it('String orientation', function () {
|
it('String orientation', function () {
|
||||||
assert.throws(function () {
|
assert.throws(function () {
|
||||||
sharp().withMetadata({orientation: 'zoinks'});
|
sharp().withMetadata({ orientation: 'zoinks' });
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
it('Negative orientation', function () {
|
it('Negative orientation', function () {
|
||||||
assert.throws(function () {
|
assert.throws(function () {
|
||||||
sharp().withMetadata({orientation: -1});
|
sharp().withMetadata({ orientation: -1 });
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
it('Zero orientation', function () {
|
it('Zero orientation', function () {
|
||||||
@@ -380,7 +464,7 @@ describe('Image metadata', function () {
|
|||||||
});
|
});
|
||||||
it('Too large orientation', function () {
|
it('Too large orientation', function () {
|
||||||
assert.throws(function () {
|
assert.throws(function () {
|
||||||
sharp().withMetadata({orientation: 9});
|
sharp().withMetadata({ orientation: 9 });
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|||||||
135
test/unit/recomb.js
Normal file
@@ -0,0 +1,135 @@
|
|||||||
|
'use strict';
|
||||||
|
|
||||||
|
const assert = require('assert');
|
||||||
|
|
||||||
|
const sharp = require('../../');
|
||||||
|
const fixtures = require('../fixtures');
|
||||||
|
|
||||||
|
describe('Recomb', function () {
|
||||||
|
it('applies a sepia filter using recomb', function (done) {
|
||||||
|
const output = fixtures.path('output.recomb-sepia.jpg');
|
||||||
|
sharp(fixtures.inputJpgWithLandscapeExif1)
|
||||||
|
.recomb([
|
||||||
|
[0.3588, 0.7044, 0.1368],
|
||||||
|
[0.299, 0.587, 0.114],
|
||||||
|
[0.2392, 0.4696, 0.0912]
|
||||||
|
])
|
||||||
|
.toFile(output, function (err, info) {
|
||||||
|
if (err) throw err;
|
||||||
|
assert.strictEqual('jpeg', info.format);
|
||||||
|
assert.strictEqual(600, info.width);
|
||||||
|
assert.strictEqual(450, info.height);
|
||||||
|
fixtures.assertMaxColourDistance(
|
||||||
|
output,
|
||||||
|
fixtures.expected('Landscape_1-recomb-sepia.jpg'),
|
||||||
|
17
|
||||||
|
);
|
||||||
|
done();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('applies a sepia filter using recomb to an PNG with Alpha', function (done) {
|
||||||
|
const output = fixtures.path('output.recomb-sepia.png');
|
||||||
|
sharp(fixtures.inputPngAlphaPremultiplicationSmall)
|
||||||
|
.recomb([
|
||||||
|
[0.3588, 0.7044, 0.1368],
|
||||||
|
[0.299, 0.587, 0.114],
|
||||||
|
[0.2392, 0.4696, 0.0912]
|
||||||
|
])
|
||||||
|
.toFile(output, function (err, info) {
|
||||||
|
if (err) throw err;
|
||||||
|
assert.strictEqual('png', info.format);
|
||||||
|
assert.strictEqual(1024, info.width);
|
||||||
|
assert.strictEqual(768, info.height);
|
||||||
|
fixtures.assertMaxColourDistance(
|
||||||
|
output,
|
||||||
|
fixtures.expected('alpha-recomb-sepia.png'),
|
||||||
|
17
|
||||||
|
);
|
||||||
|
done();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('applies a different sepia filter using recomb', function (done) {
|
||||||
|
const output = fixtures.path('output.recomb-sepia2.jpg');
|
||||||
|
sharp(fixtures.inputJpgWithLandscapeExif1)
|
||||||
|
.recomb([
|
||||||
|
[0.393, 0.769, 0.189],
|
||||||
|
[0.349, 0.686, 0.168],
|
||||||
|
[0.272, 0.534, 0.131]
|
||||||
|
])
|
||||||
|
.toFile(output, function (err, info) {
|
||||||
|
if (err) throw err;
|
||||||
|
assert.strictEqual('jpeg', info.format);
|
||||||
|
assert.strictEqual(600, info.width);
|
||||||
|
assert.strictEqual(450, info.height);
|
||||||
|
fixtures.assertMaxColourDistance(
|
||||||
|
output,
|
||||||
|
fixtures.expected('Landscape_1-recomb-sepia2.jpg'),
|
||||||
|
17
|
||||||
|
);
|
||||||
|
done();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
it('increases the saturation of the image', function (done) {
|
||||||
|
const saturationLevel = 1;
|
||||||
|
const output = fixtures.path('output.recomb-saturation.jpg');
|
||||||
|
sharp(fixtures.inputJpgWithLandscapeExif1)
|
||||||
|
.recomb([
|
||||||
|
[
|
||||||
|
saturationLevel + 1 - 0.2989,
|
||||||
|
-0.587 * saturationLevel,
|
||||||
|
-0.114 * saturationLevel
|
||||||
|
],
|
||||||
|
[
|
||||||
|
-0.2989 * saturationLevel,
|
||||||
|
saturationLevel + 1 - 0.587,
|
||||||
|
-0.114 * saturationLevel
|
||||||
|
],
|
||||||
|
[
|
||||||
|
-0.2989 * saturationLevel,
|
||||||
|
-0.587 * saturationLevel,
|
||||||
|
saturationLevel + 1 - 0.114
|
||||||
|
]
|
||||||
|
])
|
||||||
|
.toFile(output, function (err, info) {
|
||||||
|
if (err) throw err;
|
||||||
|
assert.strictEqual('jpeg', info.format);
|
||||||
|
assert.strictEqual(600, info.width);
|
||||||
|
assert.strictEqual(450, info.height);
|
||||||
|
fixtures.assertMaxColourDistance(
|
||||||
|
output,
|
||||||
|
fixtures.expected('Landscape_1-recomb-saturation.jpg'),
|
||||||
|
37
|
||||||
|
);
|
||||||
|
done();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('invalid matrix specification', function () {
|
||||||
|
it('missing', function () {
|
||||||
|
assert.throws(function () {
|
||||||
|
sharp(fixtures.inputJpg).recomb();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
it('incorrect flat data', function () {
|
||||||
|
assert.throws(function () {
|
||||||
|
sharp(fixtures.inputJpg).recomb([1, 2, 3, 4, 5, 6, 7, 8, 9]);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
it('incorrect sub size', function () {
|
||||||
|
assert.throws(function () {
|
||||||
|
sharp(fixtures.inputJpg).recomb([
|
||||||
|
[1, 2, 3, 4],
|
||||||
|
[5, 6, 7, 8],
|
||||||
|
[1, 2, 9, 6]
|
||||||
|
]);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
it('incorrect top size', function () {
|
||||||
|
assert.throws(function () {
|
||||||
|
sharp(fixtures.inputJpg).recomb([[1, 2, 3, 4], [5, 6, 7, 8]]);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
770
test/unit/resize-contain.js
Normal file
@@ -0,0 +1,770 @@
|
|||||||
|
'use strict';
|
||||||
|
|
||||||
|
const assert = require('assert');
|
||||||
|
|
||||||
|
const sharp = require('../../');
|
||||||
|
const fixtures = require('../fixtures');
|
||||||
|
|
||||||
|
describe('Resize fit=contain', function () {
|
||||||
|
it('Allows specifying the position as a string', function (done) {
|
||||||
|
sharp(fixtures.inputJpg)
|
||||||
|
.resize(320, 240, {
|
||||||
|
fit: 'contain',
|
||||||
|
position: 'center'
|
||||||
|
})
|
||||||
|
.png()
|
||||||
|
.toBuffer(function (err, data, info) {
|
||||||
|
if (err) throw err;
|
||||||
|
assert.strictEqual(320, info.width);
|
||||||
|
assert.strictEqual(240, info.height);
|
||||||
|
fixtures.assertSimilar(fixtures.expected('embed-3-into-3.png'), data, done);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('JPEG within PNG, no alpha channel', function (done) {
|
||||||
|
sharp(fixtures.inputJpg)
|
||||||
|
.resize(320, 240, { fit: 'contain' })
|
||||||
|
.png()
|
||||||
|
.toBuffer(function (err, data, info) {
|
||||||
|
if (err) throw err;
|
||||||
|
assert.strictEqual(true, data.length > 0);
|
||||||
|
assert.strictEqual('png', info.format);
|
||||||
|
assert.strictEqual(320, info.width);
|
||||||
|
assert.strictEqual(240, info.height);
|
||||||
|
assert.strictEqual(3, info.channels);
|
||||||
|
fixtures.assertSimilar(fixtures.expected('embed-3-into-3.png'), data, done);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('JPEG within WebP, to include alpha channel', function (done) {
|
||||||
|
sharp(fixtures.inputJpg)
|
||||||
|
.resize(320, 240, {
|
||||||
|
fit: 'contain',
|
||||||
|
background: { r: 0, g: 0, b: 0, alpha: 0 }
|
||||||
|
})
|
||||||
|
.webp()
|
||||||
|
.toBuffer(function (err, data, info) {
|
||||||
|
if (err) throw err;
|
||||||
|
assert.strictEqual(true, data.length > 0);
|
||||||
|
assert.strictEqual('webp', info.format);
|
||||||
|
assert.strictEqual(320, info.width);
|
||||||
|
assert.strictEqual(240, info.height);
|
||||||
|
assert.strictEqual(4, info.channels);
|
||||||
|
fixtures.assertSimilar(fixtures.expected('embed-3-into-4.webp'), data, done);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('PNG with alpha channel', function (done) {
|
||||||
|
sharp(fixtures.inputPngWithTransparency)
|
||||||
|
.resize(50, 50, { fit: 'contain' })
|
||||||
|
.toBuffer(function (err, data, info) {
|
||||||
|
if (err) throw err;
|
||||||
|
assert.strictEqual(true, data.length > 0);
|
||||||
|
assert.strictEqual('png', info.format);
|
||||||
|
assert.strictEqual(50, info.width);
|
||||||
|
assert.strictEqual(50, info.height);
|
||||||
|
assert.strictEqual(4, info.channels);
|
||||||
|
fixtures.assertSimilar(fixtures.expected('embed-4-into-4.png'), data, done);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('16-bit PNG with alpha channel', function (done) {
|
||||||
|
sharp(fixtures.inputPngWithTransparency16bit)
|
||||||
|
.resize(32, 16, { fit: 'contain' })
|
||||||
|
.toBuffer(function (err, data, info) {
|
||||||
|
if (err) throw err;
|
||||||
|
assert.strictEqual(true, data.length > 0);
|
||||||
|
assert.strictEqual('png', info.format);
|
||||||
|
assert.strictEqual(32, info.width);
|
||||||
|
assert.strictEqual(16, info.height);
|
||||||
|
assert.strictEqual(4, info.channels);
|
||||||
|
fixtures.assertSimilar(fixtures.expected('embed-16bit.png'), data, done);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('16-bit PNG with alpha channel onto RGBA', function (done) {
|
||||||
|
sharp(fixtures.inputPngWithTransparency16bit)
|
||||||
|
.resize(32, 16, {
|
||||||
|
fit: 'contain',
|
||||||
|
background: { r: 0, g: 0, b: 0, alpha: 0 }
|
||||||
|
})
|
||||||
|
.toBuffer(function (err, data, info) {
|
||||||
|
if (err) throw err;
|
||||||
|
assert.strictEqual(true, data.length > 0);
|
||||||
|
assert.strictEqual('png', info.format);
|
||||||
|
assert.strictEqual(32, info.width);
|
||||||
|
assert.strictEqual(16, info.height);
|
||||||
|
assert.strictEqual(4, info.channels);
|
||||||
|
fixtures.assertSimilar(fixtures.expected('embed-16bit-rgba.png'), data, done);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('PNG with 2 channels', function (done) {
|
||||||
|
sharp(fixtures.inputPngWithGreyAlpha)
|
||||||
|
.resize(32, 16, {
|
||||||
|
fit: 'contain',
|
||||||
|
background: { r: 0, g: 0, b: 0, alpha: 0 }
|
||||||
|
})
|
||||||
|
.toBuffer(function (err, data, info) {
|
||||||
|
if (err) throw err;
|
||||||
|
assert.strictEqual(true, data.length > 0);
|
||||||
|
assert.strictEqual('png', info.format);
|
||||||
|
assert.strictEqual(32, info.width);
|
||||||
|
assert.strictEqual(16, info.height);
|
||||||
|
assert.strictEqual(4, info.channels);
|
||||||
|
fixtures.assertSimilar(fixtures.expected('embed-2channel.png'), data, done);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it.skip('TIFF in LAB colourspace onto RGBA background', function (done) {
|
||||||
|
sharp(fixtures.inputTiffCielab)
|
||||||
|
.resize(64, 128, {
|
||||||
|
fit: 'contain',
|
||||||
|
background: { r: 255, g: 102, b: 0, alpha: 0.5 }
|
||||||
|
})
|
||||||
|
.png()
|
||||||
|
.toBuffer(function (err, data, info) {
|
||||||
|
if (err) throw err;
|
||||||
|
assert.strictEqual(true, data.length > 0);
|
||||||
|
assert.strictEqual('png', info.format);
|
||||||
|
assert.strictEqual(64, info.width);
|
||||||
|
assert.strictEqual(128, info.height);
|
||||||
|
assert.strictEqual(4, info.channels);
|
||||||
|
fixtures.assertSimilar(fixtures.expected('embed-lab-into-rgba.png'), data, done);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('Enlarge', function (done) {
|
||||||
|
sharp(fixtures.inputPngWithOneColor)
|
||||||
|
.resize(320, 240, { fit: 'contain' })
|
||||||
|
.toBuffer(function (err, data, info) {
|
||||||
|
if (err) throw err;
|
||||||
|
assert.strictEqual(true, data.length > 0);
|
||||||
|
assert.strictEqual('png', info.format);
|
||||||
|
assert.strictEqual(320, info.width);
|
||||||
|
assert.strictEqual(240, info.height);
|
||||||
|
assert.strictEqual(3, info.channels);
|
||||||
|
fixtures.assertSimilar(fixtures.expected('embed-enlarge.png'), data, done);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('Invalid position values should fail', function () {
|
||||||
|
[-1, 8.1, 9, 1000000, false, 'vallejo'].forEach(function (position) {
|
||||||
|
assert.throws(function () {
|
||||||
|
sharp().resize(null, null, { fit: 'contain', position });
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('Position horizontal top', function (done) {
|
||||||
|
sharp(fixtures.inputPngEmbed)
|
||||||
|
.resize(200, 100, {
|
||||||
|
fit: sharp.fit.contain,
|
||||||
|
background: { r: 0, g: 0, b: 0, alpha: 0 },
|
||||||
|
position: 'top'
|
||||||
|
})
|
||||||
|
.toBuffer(function (err, data, info) {
|
||||||
|
if (err) throw err;
|
||||||
|
assert.strictEqual(true, data.length > 0);
|
||||||
|
assert.strictEqual('png', info.format);
|
||||||
|
assert.strictEqual(200, info.width);
|
||||||
|
assert.strictEqual(100, info.height);
|
||||||
|
assert.strictEqual(4, info.channels);
|
||||||
|
fixtures.assertSimilar(fixtures.expected('./embedgravitybird/a2-n.png'), data, done);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('Position horizontal right top', function (done) {
|
||||||
|
sharp(fixtures.inputPngEmbed)
|
||||||
|
.resize(200, 100, {
|
||||||
|
fit: sharp.fit.contain,
|
||||||
|
background: { r: 0, g: 0, b: 0, alpha: 0 },
|
||||||
|
position: 'right top'
|
||||||
|
})
|
||||||
|
.toBuffer(function (err, data, info) {
|
||||||
|
if (err) throw err;
|
||||||
|
assert.strictEqual(true, data.length > 0);
|
||||||
|
assert.strictEqual('png', info.format);
|
||||||
|
assert.strictEqual(200, info.width);
|
||||||
|
assert.strictEqual(100, info.height);
|
||||||
|
assert.strictEqual(4, info.channels);
|
||||||
|
fixtures.assertSimilar(fixtures.expected('./embedgravitybird/a3-ne.png'), data, done);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('Position horizontal right', function (done) {
|
||||||
|
sharp(fixtures.inputPngEmbed)
|
||||||
|
.resize(200, 100, {
|
||||||
|
fit: sharp.fit.contain,
|
||||||
|
background: { r: 0, g: 0, b: 0, alpha: 0 },
|
||||||
|
position: 'right'
|
||||||
|
})
|
||||||
|
.toBuffer(function (err, data, info) {
|
||||||
|
if (err) throw err;
|
||||||
|
assert.strictEqual(true, data.length > 0);
|
||||||
|
assert.strictEqual('png', info.format);
|
||||||
|
assert.strictEqual(200, info.width);
|
||||||
|
assert.strictEqual(100, info.height);
|
||||||
|
assert.strictEqual(4, info.channels);
|
||||||
|
fixtures.assertSimilar(fixtures.expected('./embedgravitybird/a4-e.png'), data, done);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('Position horizontal right bottom', function (done) {
|
||||||
|
sharp(fixtures.inputPngEmbed)
|
||||||
|
.resize(200, 100, {
|
||||||
|
fit: sharp.fit.contain,
|
||||||
|
background: { r: 0, g: 0, b: 0, alpha: 0 },
|
||||||
|
position: 'right bottom'
|
||||||
|
})
|
||||||
|
.toBuffer(function (err, data, info) {
|
||||||
|
if (err) throw err;
|
||||||
|
assert.strictEqual(true, data.length > 0);
|
||||||
|
assert.strictEqual('png', info.format);
|
||||||
|
assert.strictEqual(200, info.width);
|
||||||
|
assert.strictEqual(100, info.height);
|
||||||
|
assert.strictEqual(4, info.channels);
|
||||||
|
fixtures.assertSimilar(fixtures.expected('./embedgravitybird/a5-se.png'), data, done);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('Position horizontal bottom', function (done) {
|
||||||
|
sharp(fixtures.inputPngEmbed)
|
||||||
|
.resize(200, 100, {
|
||||||
|
fit: sharp.fit.contain,
|
||||||
|
background: { r: 0, g: 0, b: 0, alpha: 0 },
|
||||||
|
position: 'bottom'
|
||||||
|
})
|
||||||
|
.toBuffer(function (err, data, info) {
|
||||||
|
if (err) throw err;
|
||||||
|
assert.strictEqual(true, data.length > 0);
|
||||||
|
assert.strictEqual('png', info.format);
|
||||||
|
assert.strictEqual(200, info.width);
|
||||||
|
assert.strictEqual(100, info.height);
|
||||||
|
assert.strictEqual(4, info.channels);
|
||||||
|
fixtures.assertSimilar(fixtures.expected('./embedgravitybird/a6-s.png'), data, done);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('Position horizontal left bottom', function (done) {
|
||||||
|
sharp(fixtures.inputPngEmbed)
|
||||||
|
.resize(200, 100, {
|
||||||
|
fit: sharp.fit.contain,
|
||||||
|
background: { r: 0, g: 0, b: 0, alpha: 0 },
|
||||||
|
position: 'left bottom'
|
||||||
|
})
|
||||||
|
.toBuffer(function (err, data, info) {
|
||||||
|
if (err) throw err;
|
||||||
|
assert.strictEqual(true, data.length > 0);
|
||||||
|
assert.strictEqual('png', info.format);
|
||||||
|
assert.strictEqual(200, info.width);
|
||||||
|
assert.strictEqual(100, info.height);
|
||||||
|
assert.strictEqual(4, info.channels);
|
||||||
|
fixtures.assertSimilar(fixtures.expected('./embedgravitybird/a7-sw.png'), data, done);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('Position horizontal left', function (done) {
|
||||||
|
sharp(fixtures.inputPngEmbed)
|
||||||
|
.resize(200, 100, {
|
||||||
|
fit: sharp.fit.contain,
|
||||||
|
background: { r: 0, g: 0, b: 0, alpha: 0 },
|
||||||
|
position: 'left'
|
||||||
|
})
|
||||||
|
.toBuffer(function (err, data, info) {
|
||||||
|
if (err) throw err;
|
||||||
|
assert.strictEqual(true, data.length > 0);
|
||||||
|
assert.strictEqual('png', info.format);
|
||||||
|
assert.strictEqual(200, info.width);
|
||||||
|
assert.strictEqual(100, info.height);
|
||||||
|
assert.strictEqual(4, info.channels);
|
||||||
|
fixtures.assertSimilar(fixtures.expected('./embedgravitybird/a8-w.png'), data, done);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('Position horizontal left top', function (done) {
|
||||||
|
sharp(fixtures.inputPngEmbed)
|
||||||
|
.resize(200, 100, {
|
||||||
|
fit: sharp.fit.contain,
|
||||||
|
background: { r: 0, g: 0, b: 0, alpha: 0 },
|
||||||
|
position: 'left top'
|
||||||
|
})
|
||||||
|
.toBuffer(function (err, data, info) {
|
||||||
|
if (err) throw err;
|
||||||
|
assert.strictEqual(true, data.length > 0);
|
||||||
|
assert.strictEqual('png', info.format);
|
||||||
|
assert.strictEqual(200, info.width);
|
||||||
|
assert.strictEqual(100, info.height);
|
||||||
|
assert.strictEqual(4, info.channels);
|
||||||
|
fixtures.assertSimilar(fixtures.expected('./embedgravitybird/a1-nw.png'), data, done);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('Position horizontal north', function (done) {
|
||||||
|
sharp(fixtures.inputPngEmbed)
|
||||||
|
.resize(200, 100, {
|
||||||
|
fit: sharp.fit.contain,
|
||||||
|
background: { r: 0, g: 0, b: 0, alpha: 0 },
|
||||||
|
position: sharp.gravity.north
|
||||||
|
})
|
||||||
|
.toBuffer(function (err, data, info) {
|
||||||
|
if (err) throw err;
|
||||||
|
assert.strictEqual(true, data.length > 0);
|
||||||
|
assert.strictEqual('png', info.format);
|
||||||
|
assert.strictEqual(200, info.width);
|
||||||
|
assert.strictEqual(100, info.height);
|
||||||
|
assert.strictEqual(4, info.channels);
|
||||||
|
fixtures.assertSimilar(fixtures.expected('./embedgravitybird/a2-n.png'), data, done);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('Position horizontal northeast', function (done) {
|
||||||
|
sharp(fixtures.inputPngEmbed)
|
||||||
|
.resize(200, 100, {
|
||||||
|
fit: sharp.fit.contain,
|
||||||
|
background: { r: 0, g: 0, b: 0, alpha: 0 },
|
||||||
|
position: sharp.gravity.northeast
|
||||||
|
})
|
||||||
|
.toBuffer(function (err, data, info) {
|
||||||
|
if (err) throw err;
|
||||||
|
assert.strictEqual(true, data.length > 0);
|
||||||
|
assert.strictEqual('png', info.format);
|
||||||
|
assert.strictEqual(200, info.width);
|
||||||
|
assert.strictEqual(100, info.height);
|
||||||
|
assert.strictEqual(4, info.channels);
|
||||||
|
fixtures.assertSimilar(fixtures.expected('./embedgravitybird/a3-ne.png'), data, done);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('Position horizontal east', function (done) {
|
||||||
|
sharp(fixtures.inputPngEmbed)
|
||||||
|
.resize(200, 100, {
|
||||||
|
fit: sharp.fit.contain,
|
||||||
|
background: { r: 0, g: 0, b: 0, alpha: 0 },
|
||||||
|
position: sharp.gravity.east
|
||||||
|
})
|
||||||
|
.toBuffer(function (err, data, info) {
|
||||||
|
if (err) throw err;
|
||||||
|
assert.strictEqual(true, data.length > 0);
|
||||||
|
assert.strictEqual('png', info.format);
|
||||||
|
assert.strictEqual(200, info.width);
|
||||||
|
assert.strictEqual(100, info.height);
|
||||||
|
assert.strictEqual(4, info.channels);
|
||||||
|
fixtures.assertSimilar(fixtures.expected('./embedgravitybird/a4-e.png'), data, done);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('Position horizontal southeast', function (done) {
|
||||||
|
sharp(fixtures.inputPngEmbed)
|
||||||
|
.resize(200, 100, {
|
||||||
|
fit: sharp.fit.contain,
|
||||||
|
background: { r: 0, g: 0, b: 0, alpha: 0 },
|
||||||
|
position: sharp.gravity.southeast
|
||||||
|
})
|
||||||
|
.toBuffer(function (err, data, info) {
|
||||||
|
if (err) throw err;
|
||||||
|
assert.strictEqual(true, data.length > 0);
|
||||||
|
assert.strictEqual('png', info.format);
|
||||||
|
assert.strictEqual(200, info.width);
|
||||||
|
assert.strictEqual(100, info.height);
|
||||||
|
assert.strictEqual(4, info.channels);
|
||||||
|
fixtures.assertSimilar(fixtures.expected('./embedgravitybird/a5-se.png'), data, done);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('Position horizontal south', function (done) {
|
||||||
|
sharp(fixtures.inputPngEmbed)
|
||||||
|
.resize(200, 100, {
|
||||||
|
fit: sharp.fit.contain,
|
||||||
|
background: { r: 0, g: 0, b: 0, alpha: 0 },
|
||||||
|
position: sharp.gravity.south
|
||||||
|
})
|
||||||
|
.toBuffer(function (err, data, info) {
|
||||||
|
if (err) throw err;
|
||||||
|
assert.strictEqual(true, data.length > 0);
|
||||||
|
assert.strictEqual('png', info.format);
|
||||||
|
assert.strictEqual(200, info.width);
|
||||||
|
assert.strictEqual(100, info.height);
|
||||||
|
assert.strictEqual(4, info.channels);
|
||||||
|
fixtures.assertSimilar(fixtures.expected('./embedgravitybird/a6-s.png'), data, done);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('Position horizontal southwest', function (done) {
|
||||||
|
sharp(fixtures.inputPngEmbed)
|
||||||
|
.resize(200, 100, {
|
||||||
|
fit: sharp.fit.contain,
|
||||||
|
background: { r: 0, g: 0, b: 0, alpha: 0 },
|
||||||
|
position: sharp.gravity.southwest
|
||||||
|
})
|
||||||
|
.toBuffer(function (err, data, info) {
|
||||||
|
if (err) throw err;
|
||||||
|
assert.strictEqual(true, data.length > 0);
|
||||||
|
assert.strictEqual('png', info.format);
|
||||||
|
assert.strictEqual(200, info.width);
|
||||||
|
assert.strictEqual(100, info.height);
|
||||||
|
assert.strictEqual(4, info.channels);
|
||||||
|
fixtures.assertSimilar(fixtures.expected('./embedgravitybird/a7-sw.png'), data, done);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('Position horizontal west', function (done) {
|
||||||
|
sharp(fixtures.inputPngEmbed)
|
||||||
|
.resize(200, 100, {
|
||||||
|
fit: sharp.fit.contain,
|
||||||
|
background: { r: 0, g: 0, b: 0, alpha: 0 },
|
||||||
|
position: sharp.gravity.west
|
||||||
|
})
|
||||||
|
.toBuffer(function (err, data, info) {
|
||||||
|
if (err) throw err;
|
||||||
|
assert.strictEqual(true, data.length > 0);
|
||||||
|
assert.strictEqual('png', info.format);
|
||||||
|
assert.strictEqual(200, info.width);
|
||||||
|
assert.strictEqual(100, info.height);
|
||||||
|
assert.strictEqual(4, info.channels);
|
||||||
|
fixtures.assertSimilar(fixtures.expected('./embedgravitybird/a8-w.png'), data, done);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('Position horizontal northwest', function (done) {
|
||||||
|
sharp(fixtures.inputPngEmbed)
|
||||||
|
.resize(200, 100, {
|
||||||
|
fit: sharp.fit.contain,
|
||||||
|
background: { r: 0, g: 0, b: 0, alpha: 0 },
|
||||||
|
position: sharp.gravity.northwest
|
||||||
|
})
|
||||||
|
.toBuffer(function (err, data, info) {
|
||||||
|
if (err) throw err;
|
||||||
|
assert.strictEqual(true, data.length > 0);
|
||||||
|
assert.strictEqual('png', info.format);
|
||||||
|
assert.strictEqual(200, info.width);
|
||||||
|
assert.strictEqual(100, info.height);
|
||||||
|
assert.strictEqual(4, info.channels);
|
||||||
|
fixtures.assertSimilar(fixtures.expected('./embedgravitybird/a1-nw.png'), data, done);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('Position horizontal center', function (done) {
|
||||||
|
sharp(fixtures.inputPngEmbed)
|
||||||
|
.resize(200, 100, {
|
||||||
|
fit: sharp.fit.contain,
|
||||||
|
background: { r: 0, g: 0, b: 0, alpha: 0 },
|
||||||
|
position: sharp.gravity.center
|
||||||
|
})
|
||||||
|
.toBuffer(function (err, data, info) {
|
||||||
|
if (err) throw err;
|
||||||
|
assert.strictEqual(true, data.length > 0);
|
||||||
|
assert.strictEqual('png', info.format);
|
||||||
|
assert.strictEqual(200, info.width);
|
||||||
|
assert.strictEqual(100, info.height);
|
||||||
|
assert.strictEqual(4, info.channels);
|
||||||
|
fixtures.assertSimilar(fixtures.expected('./embedgravitybird/a9-c.png'), data, done);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('Position vertical top', function (done) {
|
||||||
|
sharp(fixtures.inputPngEmbed)
|
||||||
|
.resize(200, 200, {
|
||||||
|
fit: sharp.fit.contain,
|
||||||
|
background: { r: 0, g: 0, b: 0, alpha: 0 },
|
||||||
|
position: 'top'
|
||||||
|
})
|
||||||
|
.toBuffer(function (err, data, info) {
|
||||||
|
if (err) throw err;
|
||||||
|
assert.strictEqual(true, data.length > 0);
|
||||||
|
assert.strictEqual('png', info.format);
|
||||||
|
assert.strictEqual(200, info.width);
|
||||||
|
assert.strictEqual(200, info.height);
|
||||||
|
assert.strictEqual(4, info.channels);
|
||||||
|
fixtures.assertSimilar(fixtures.expected('./embedgravitybird/2-n.png'), data, done);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('Position vertical right top', function (done) {
|
||||||
|
sharp(fixtures.inputPngEmbed)
|
||||||
|
.resize(200, 200, {
|
||||||
|
fit: sharp.fit.contain,
|
||||||
|
background: { r: 0, g: 0, b: 0, alpha: 0 },
|
||||||
|
position: 'right top'
|
||||||
|
})
|
||||||
|
.toBuffer(function (err, data, info) {
|
||||||
|
if (err) throw err;
|
||||||
|
assert.strictEqual(true, data.length > 0);
|
||||||
|
assert.strictEqual('png', info.format);
|
||||||
|
assert.strictEqual(200, info.width);
|
||||||
|
assert.strictEqual(200, info.height);
|
||||||
|
assert.strictEqual(4, info.channels);
|
||||||
|
fixtures.assertSimilar(fixtures.expected('./embedgravitybird/3-ne.png'), data, done);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('Position vertical right', function (done) {
|
||||||
|
sharp(fixtures.inputPngEmbed)
|
||||||
|
.resize(200, 200, {
|
||||||
|
fit: sharp.fit.contain,
|
||||||
|
background: { r: 0, g: 0, b: 0, alpha: 0 },
|
||||||
|
position: 'right'
|
||||||
|
})
|
||||||
|
.toBuffer(function (err, data, info) {
|
||||||
|
if (err) throw err;
|
||||||
|
assert.strictEqual(true, data.length > 0);
|
||||||
|
assert.strictEqual('png', info.format);
|
||||||
|
assert.strictEqual(200, info.width);
|
||||||
|
assert.strictEqual(200, info.height);
|
||||||
|
assert.strictEqual(4, info.channels);
|
||||||
|
fixtures.assertSimilar(fixtures.expected('./embedgravitybird/4-e.png'), data, done);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('Position vertical right bottom', function (done) {
|
||||||
|
sharp(fixtures.inputPngEmbed)
|
||||||
|
.resize(200, 200, {
|
||||||
|
fit: sharp.fit.contain,
|
||||||
|
background: { r: 0, g: 0, b: 0, alpha: 0 },
|
||||||
|
position: 'right bottom'
|
||||||
|
})
|
||||||
|
.toBuffer(function (err, data, info) {
|
||||||
|
if (err) throw err;
|
||||||
|
assert.strictEqual(true, data.length > 0);
|
||||||
|
assert.strictEqual('png', info.format);
|
||||||
|
assert.strictEqual(200, info.width);
|
||||||
|
assert.strictEqual(200, info.height);
|
||||||
|
assert.strictEqual(4, info.channels);
|
||||||
|
fixtures.assertSimilar(fixtures.expected('./embedgravitybird/5-se.png'), data, done);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('Position vertical bottom', function (done) {
|
||||||
|
sharp(fixtures.inputPngEmbed)
|
||||||
|
.resize(200, 200, {
|
||||||
|
fit: sharp.fit.contain,
|
||||||
|
background: { r: 0, g: 0, b: 0, alpha: 0 },
|
||||||
|
position: 'bottom'
|
||||||
|
})
|
||||||
|
.toBuffer(function (err, data, info) {
|
||||||
|
if (err) throw err;
|
||||||
|
assert.strictEqual(true, data.length > 0);
|
||||||
|
assert.strictEqual('png', info.format);
|
||||||
|
assert.strictEqual(200, info.width);
|
||||||
|
assert.strictEqual(200, info.height);
|
||||||
|
assert.strictEqual(4, info.channels);
|
||||||
|
fixtures.assertSimilar(fixtures.expected('./embedgravitybird/6-s.png'), data, done);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('Position vertical left bottom', function (done) {
|
||||||
|
sharp(fixtures.inputPngEmbed)
|
||||||
|
.resize(200, 200, {
|
||||||
|
fit: sharp.fit.contain,
|
||||||
|
background: { r: 0, g: 0, b: 0, alpha: 0 },
|
||||||
|
position: 'left bottom'
|
||||||
|
})
|
||||||
|
.toBuffer(function (err, data, info) {
|
||||||
|
if (err) throw err;
|
||||||
|
assert.strictEqual(true, data.length > 0);
|
||||||
|
assert.strictEqual('png', info.format);
|
||||||
|
assert.strictEqual(200, info.width);
|
||||||
|
assert.strictEqual(200, info.height);
|
||||||
|
assert.strictEqual(4, info.channels);
|
||||||
|
fixtures.assertSimilar(fixtures.expected('./embedgravitybird/7-sw.png'), data, done);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('Position vertical left', function (done) {
|
||||||
|
sharp(fixtures.inputPngEmbed)
|
||||||
|
.resize(200, 200, {
|
||||||
|
fit: sharp.fit.contain,
|
||||||
|
background: { r: 0, g: 0, b: 0, alpha: 0 },
|
||||||
|
position: 'left'
|
||||||
|
})
|
||||||
|
.toBuffer(function (err, data, info) {
|
||||||
|
if (err) throw err;
|
||||||
|
assert.strictEqual(true, data.length > 0);
|
||||||
|
assert.strictEqual('png', info.format);
|
||||||
|
assert.strictEqual(200, info.width);
|
||||||
|
assert.strictEqual(200, info.height);
|
||||||
|
assert.strictEqual(4, info.channels);
|
||||||
|
fixtures.assertSimilar(fixtures.expected('./embedgravitybird/8-w.png'), data, done);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('Position vertical left top', function (done) {
|
||||||
|
sharp(fixtures.inputPngEmbed)
|
||||||
|
.resize(200, 200, {
|
||||||
|
fit: sharp.fit.contain,
|
||||||
|
background: { r: 0, g: 0, b: 0, alpha: 0 },
|
||||||
|
position: 'left top'
|
||||||
|
})
|
||||||
|
.toBuffer(function (err, data, info) {
|
||||||
|
if (err) throw err;
|
||||||
|
assert.strictEqual(true, data.length > 0);
|
||||||
|
assert.strictEqual('png', info.format);
|
||||||
|
assert.strictEqual(200, info.width);
|
||||||
|
assert.strictEqual(200, info.height);
|
||||||
|
assert.strictEqual(4, info.channels);
|
||||||
|
fixtures.assertSimilar(fixtures.expected('./embedgravitybird/1-nw.png'), data, done);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('Position vertical north', function (done) {
|
||||||
|
sharp(fixtures.inputPngEmbed)
|
||||||
|
.resize(200, 200, {
|
||||||
|
fit: sharp.fit.contain,
|
||||||
|
background: { r: 0, g: 0, b: 0, alpha: 0 },
|
||||||
|
position: sharp.gravity.north
|
||||||
|
})
|
||||||
|
.toBuffer(function (err, data, info) {
|
||||||
|
if (err) throw err;
|
||||||
|
assert.strictEqual(true, data.length > 0);
|
||||||
|
assert.strictEqual('png', info.format);
|
||||||
|
assert.strictEqual(200, info.width);
|
||||||
|
assert.strictEqual(200, info.height);
|
||||||
|
assert.strictEqual(4, info.channels);
|
||||||
|
fixtures.assertSimilar(fixtures.expected('./embedgravitybird/2-n.png'), data, done);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('Position vertical northeast', function (done) {
|
||||||
|
sharp(fixtures.inputPngEmbed)
|
||||||
|
.resize(200, 200, {
|
||||||
|
fit: sharp.fit.contain,
|
||||||
|
background: { r: 0, g: 0, b: 0, alpha: 0 },
|
||||||
|
position: sharp.gravity.northeast
|
||||||
|
})
|
||||||
|
.toBuffer(function (err, data, info) {
|
||||||
|
if (err) throw err;
|
||||||
|
assert.strictEqual(true, data.length > 0);
|
||||||
|
assert.strictEqual('png', info.format);
|
||||||
|
assert.strictEqual(200, info.width);
|
||||||
|
assert.strictEqual(200, info.height);
|
||||||
|
assert.strictEqual(4, info.channels);
|
||||||
|
fixtures.assertSimilar(fixtures.expected('./embedgravitybird/3-ne.png'), data, done);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('Position vertical east', function (done) {
|
||||||
|
sharp(fixtures.inputPngEmbed)
|
||||||
|
.resize(200, 200, {
|
||||||
|
fit: sharp.fit.contain,
|
||||||
|
background: { r: 0, g: 0, b: 0, alpha: 0 },
|
||||||
|
position: sharp.gravity.east
|
||||||
|
})
|
||||||
|
.toBuffer(function (err, data, info) {
|
||||||
|
if (err) throw err;
|
||||||
|
assert.strictEqual(true, data.length > 0);
|
||||||
|
assert.strictEqual('png', info.format);
|
||||||
|
assert.strictEqual(200, info.width);
|
||||||
|
assert.strictEqual(200, info.height);
|
||||||
|
assert.strictEqual(4, info.channels);
|
||||||
|
fixtures.assertSimilar(fixtures.expected('./embedgravitybird/4-e.png'), data, done);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('Position vertical southeast', function (done) {
|
||||||
|
sharp(fixtures.inputPngEmbed)
|
||||||
|
.resize(200, 200, {
|
||||||
|
fit: sharp.fit.contain,
|
||||||
|
background: { r: 0, g: 0, b: 0, alpha: 0 },
|
||||||
|
position: sharp.gravity.southeast
|
||||||
|
})
|
||||||
|
.toBuffer(function (err, data, info) {
|
||||||
|
if (err) throw err;
|
||||||
|
assert.strictEqual(true, data.length > 0);
|
||||||
|
assert.strictEqual('png', info.format);
|
||||||
|
assert.strictEqual(200, info.width);
|
||||||
|
assert.strictEqual(200, info.height);
|
||||||
|
assert.strictEqual(4, info.channels);
|
||||||
|
fixtures.assertSimilar(fixtures.expected('./embedgravitybird/5-se.png'), data, done);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('Position vertical south', function (done) {
|
||||||
|
sharp(fixtures.inputPngEmbed)
|
||||||
|
.resize(200, 200, {
|
||||||
|
fit: sharp.fit.contain,
|
||||||
|
background: { r: 0, g: 0, b: 0, alpha: 0 },
|
||||||
|
position: sharp.gravity.south
|
||||||
|
})
|
||||||
|
.toBuffer(function (err, data, info) {
|
||||||
|
if (err) throw err;
|
||||||
|
assert.strictEqual(true, data.length > 0);
|
||||||
|
assert.strictEqual('png', info.format);
|
||||||
|
assert.strictEqual(200, info.width);
|
||||||
|
assert.strictEqual(200, info.height);
|
||||||
|
assert.strictEqual(4, info.channels);
|
||||||
|
fixtures.assertSimilar(fixtures.expected('./embedgravitybird/6-s.png'), data, done);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('Position vertical southwest', function (done) {
|
||||||
|
sharp(fixtures.inputPngEmbed)
|
||||||
|
.resize(200, 200, {
|
||||||
|
fit: sharp.fit.contain,
|
||||||
|
background: { r: 0, g: 0, b: 0, alpha: 0 },
|
||||||
|
position: sharp.gravity.southwest
|
||||||
|
})
|
||||||
|
.toBuffer(function (err, data, info) {
|
||||||
|
if (err) throw err;
|
||||||
|
assert.strictEqual(true, data.length > 0);
|
||||||
|
assert.strictEqual('png', info.format);
|
||||||
|
assert.strictEqual(200, info.width);
|
||||||
|
assert.strictEqual(200, info.height);
|
||||||
|
assert.strictEqual(4, info.channels);
|
||||||
|
fixtures.assertSimilar(fixtures.expected('./embedgravitybird/7-sw.png'), data, done);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('Position vertical west', function (done) {
|
||||||
|
sharp(fixtures.inputPngEmbed)
|
||||||
|
.resize(200, 200, {
|
||||||
|
fit: sharp.fit.contain,
|
||||||
|
background: { r: 0, g: 0, b: 0, alpha: 0 },
|
||||||
|
position: sharp.gravity.west
|
||||||
|
})
|
||||||
|
.toBuffer(function (err, data, info) {
|
||||||
|
if (err) throw err;
|
||||||
|
assert.strictEqual(true, data.length > 0);
|
||||||
|
assert.strictEqual('png', info.format);
|
||||||
|
assert.strictEqual(200, info.width);
|
||||||
|
assert.strictEqual(200, info.height);
|
||||||
|
assert.strictEqual(4, info.channels);
|
||||||
|
fixtures.assertSimilar(fixtures.expected('./embedgravitybird/8-w.png'), data, done);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('Position vertical northwest', function (done) {
|
||||||
|
sharp(fixtures.inputPngEmbed)
|
||||||
|
.resize(200, 200, {
|
||||||
|
fit: sharp.fit.contain,
|
||||||
|
background: { r: 0, g: 0, b: 0, alpha: 0 },
|
||||||
|
position: sharp.gravity.northwest
|
||||||
|
})
|
||||||
|
.toBuffer(function (err, data, info) {
|
||||||
|
if (err) throw err;
|
||||||
|
assert.strictEqual(true, data.length > 0);
|
||||||
|
assert.strictEqual('png', info.format);
|
||||||
|
assert.strictEqual(200, info.width);
|
||||||
|
assert.strictEqual(200, info.height);
|
||||||
|
assert.strictEqual(4, info.channels);
|
||||||
|
fixtures.assertSimilar(fixtures.expected('./embedgravitybird/1-nw.png'), data, done);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('Position vertical center', function (done) {
|
||||||
|
sharp(fixtures.inputPngEmbed)
|
||||||
|
.resize(200, 200, {
|
||||||
|
fit: sharp.fit.contain,
|
||||||
|
background: { r: 0, g: 0, b: 0, alpha: 0 },
|
||||||
|
position: sharp.gravity.center
|
||||||
|
})
|
||||||
|
.toBuffer(function (err, data, info) {
|
||||||
|
if (err) throw err;
|
||||||
|
assert.strictEqual(true, data.length > 0);
|
||||||
|
assert.strictEqual('png', info.format);
|
||||||
|
assert.strictEqual(200, info.width);
|
||||||
|
assert.strictEqual(200, info.height);
|
||||||
|
assert.strictEqual(4, info.channels);
|
||||||
|
fixtures.assertSimilar(fixtures.expected('./embedgravitybird/9-c.png'), data, done);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
383
test/unit/resize-cover.js
Normal file
@@ -0,0 +1,383 @@
|
|||||||
|
'use strict';
|
||||||
|
|
||||||
|
const assert = require('assert');
|
||||||
|
|
||||||
|
const sharp = require('../../');
|
||||||
|
const fixtures = require('../fixtures');
|
||||||
|
|
||||||
|
describe('Resize fit=cover', function () {
|
||||||
|
[
|
||||||
|
// Position
|
||||||
|
{
|
||||||
|
name: 'Position: top',
|
||||||
|
width: 320,
|
||||||
|
height: 80,
|
||||||
|
gravity: sharp.position.top,
|
||||||
|
fixture: 'gravity-north.jpg'
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'Position: right',
|
||||||
|
width: 80,
|
||||||
|
height: 320,
|
||||||
|
gravity: sharp.position.right,
|
||||||
|
fixture: 'gravity-east.jpg'
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'Position: bottom',
|
||||||
|
width: 320,
|
||||||
|
height: 80,
|
||||||
|
gravity: sharp.position.bottom,
|
||||||
|
fixture: 'gravity-south.jpg'
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'Position: left',
|
||||||
|
width: 80,
|
||||||
|
height: 320,
|
||||||
|
gravity: sharp.position.left,
|
||||||
|
fixture: 'gravity-west.jpg'
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'Position: right top (top)',
|
||||||
|
width: 320,
|
||||||
|
height: 80,
|
||||||
|
gravity: sharp.position['right top'],
|
||||||
|
fixture: 'gravity-north.jpg'
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'Position: right top (right)',
|
||||||
|
width: 80,
|
||||||
|
height: 320,
|
||||||
|
gravity: sharp.position['right top'],
|
||||||
|
fixture: 'gravity-east.jpg'
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'Position: right bottom (bottom)',
|
||||||
|
width: 320,
|
||||||
|
height: 80,
|
||||||
|
gravity: sharp.position['right bottom'],
|
||||||
|
fixture: 'gravity-south.jpg'
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'Position: right bottom (right)',
|
||||||
|
width: 80,
|
||||||
|
height: 320,
|
||||||
|
gravity: sharp.position['right bottom'],
|
||||||
|
fixture: 'gravity-east.jpg'
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'Position: left bottom (bottom)',
|
||||||
|
width: 320,
|
||||||
|
height: 80,
|
||||||
|
gravity: sharp.position['left bottom'],
|
||||||
|
fixture: 'gravity-south.jpg'
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'Position: left bottom (left)',
|
||||||
|
width: 80,
|
||||||
|
height: 320,
|
||||||
|
gravity: sharp.position['left bottom'],
|
||||||
|
fixture: 'gravity-west.jpg'
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'Position: left top (top)',
|
||||||
|
width: 320,
|
||||||
|
height: 80,
|
||||||
|
gravity: sharp.position['left top'],
|
||||||
|
fixture: 'gravity-north.jpg'
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'Position: left top (left)',
|
||||||
|
width: 80,
|
||||||
|
height: 320,
|
||||||
|
gravity: sharp.position['left top'],
|
||||||
|
fixture: 'gravity-west.jpg'
|
||||||
|
},
|
||||||
|
// Gravity
|
||||||
|
{
|
||||||
|
name: 'Gravity: north',
|
||||||
|
width: 320,
|
||||||
|
height: 80,
|
||||||
|
gravity: sharp.gravity.north,
|
||||||
|
fixture: 'gravity-north.jpg'
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'Gravity: east',
|
||||||
|
width: 80,
|
||||||
|
height: 320,
|
||||||
|
gravity: sharp.gravity.east,
|
||||||
|
fixture: 'gravity-east.jpg'
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'Gravity: south',
|
||||||
|
width: 320,
|
||||||
|
height: 80,
|
||||||
|
gravity: sharp.gravity.south,
|
||||||
|
fixture: 'gravity-south.jpg'
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'Gravity: west',
|
||||||
|
width: 80,
|
||||||
|
height: 320,
|
||||||
|
gravity: sharp.gravity.west,
|
||||||
|
fixture: 'gravity-west.jpg'
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'Gravity: center',
|
||||||
|
width: 320,
|
||||||
|
height: 80,
|
||||||
|
gravity: sharp.gravity.center,
|
||||||
|
fixture: 'gravity-center.jpg'
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'Gravity: centre',
|
||||||
|
width: 80,
|
||||||
|
height: 320,
|
||||||
|
gravity: sharp.gravity.centre,
|
||||||
|
fixture: 'gravity-centre.jpg'
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'Default (centre)',
|
||||||
|
width: 80,
|
||||||
|
height: 320,
|
||||||
|
gravity: undefined,
|
||||||
|
fixture: 'gravity-centre.jpg'
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'Gravity: northeast (north)',
|
||||||
|
width: 320,
|
||||||
|
height: 80,
|
||||||
|
gravity: sharp.gravity.northeast,
|
||||||
|
fixture: 'gravity-north.jpg'
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'Gravity: northeast (east)',
|
||||||
|
width: 80,
|
||||||
|
height: 320,
|
||||||
|
gravity: sharp.gravity.northeast,
|
||||||
|
fixture: 'gravity-east.jpg'
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'Gravity: southeast (south)',
|
||||||
|
width: 320,
|
||||||
|
height: 80,
|
||||||
|
gravity: sharp.gravity.southeast,
|
||||||
|
fixture: 'gravity-south.jpg'
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'Gravity: southeast (east)',
|
||||||
|
width: 80,
|
||||||
|
height: 320,
|
||||||
|
gravity: sharp.gravity.southeast,
|
||||||
|
fixture: 'gravity-east.jpg'
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'Gravity: southwest (south)',
|
||||||
|
width: 320,
|
||||||
|
height: 80,
|
||||||
|
gravity: sharp.gravity.southwest,
|
||||||
|
fixture: 'gravity-south.jpg'
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'Gravity: southwest (west)',
|
||||||
|
width: 80,
|
||||||
|
height: 320,
|
||||||
|
gravity: sharp.gravity.southwest,
|
||||||
|
fixture: 'gravity-west.jpg'
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'Gravity: northwest (north)',
|
||||||
|
width: 320,
|
||||||
|
height: 80,
|
||||||
|
gravity: sharp.gravity.northwest,
|
||||||
|
fixture: 'gravity-north.jpg'
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'Gravity: northwest (west)',
|
||||||
|
width: 80,
|
||||||
|
height: 320,
|
||||||
|
gravity: sharp.gravity.northwest,
|
||||||
|
fixture: 'gravity-west.jpg'
|
||||||
|
}
|
||||||
|
].forEach(function (settings) {
|
||||||
|
it(settings.name, function (done) {
|
||||||
|
sharp(fixtures.inputJpg)
|
||||||
|
.resize(settings.width, settings.height, {
|
||||||
|
fit: sharp.fit.cover,
|
||||||
|
position: settings.gravity
|
||||||
|
})
|
||||||
|
.toBuffer(function (err, data, info) {
|
||||||
|
if (err) throw err;
|
||||||
|
assert.strictEqual(settings.width, info.width);
|
||||||
|
assert.strictEqual(settings.height, info.height);
|
||||||
|
fixtures.assertSimilar(fixtures.expected(settings.fixture), data, done);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('Allows specifying the gravity as a string', function (done) {
|
||||||
|
sharp(fixtures.inputJpg)
|
||||||
|
.resize(80, 320, {
|
||||||
|
fit: sharp.fit.cover,
|
||||||
|
position: 'east'
|
||||||
|
})
|
||||||
|
.toBuffer(function (err, data, info) {
|
||||||
|
if (err) throw err;
|
||||||
|
assert.strictEqual(80, info.width);
|
||||||
|
assert.strictEqual(320, info.height);
|
||||||
|
fixtures.assertSimilar(fixtures.expected('gravity-east.jpg'), data, done);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('Invalid position values fail', function () {
|
||||||
|
assert.throws(function () {
|
||||||
|
sharp().resize(null, null, { fit: 'cover', position: 9 });
|
||||||
|
}, /Expected valid position\/gravity\/strategy for position but received 9 of type number/);
|
||||||
|
assert.throws(function () {
|
||||||
|
sharp().resize(null, null, { fit: 'cover', position: 1.1 });
|
||||||
|
}, /Expected valid position\/gravity\/strategy for position but received 1.1 of type number/);
|
||||||
|
assert.throws(function () {
|
||||||
|
sharp().resize(null, null, { fit: 'cover', position: -1 });
|
||||||
|
}, /Expected valid position\/gravity\/strategy for position but received -1 of type number/);
|
||||||
|
assert.throws(function () {
|
||||||
|
sharp().resize(null, null, { fit: 'cover', position: 'zoinks' }).crop();
|
||||||
|
}, /Expected valid position\/gravity\/strategy for position but received zoinks of type string/);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('Uses default value when none specified', function () {
|
||||||
|
assert.doesNotThrow(function () {
|
||||||
|
sharp().resize(null, null, { fit: 'cover' });
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('Skip crop when post-resize dimensions are at target', function () {
|
||||||
|
return sharp(fixtures.inputJpg)
|
||||||
|
.resize(1600, 1200)
|
||||||
|
.toBuffer()
|
||||||
|
.then(function (input) {
|
||||||
|
return sharp(input)
|
||||||
|
.resize(1110, null, {
|
||||||
|
fit: sharp.fit.cover,
|
||||||
|
position: sharp.strategy.attention
|
||||||
|
})
|
||||||
|
.toBuffer({ resolveWithObject: true })
|
||||||
|
.then(function (result) {
|
||||||
|
assert.strictEqual(1110, result.info.width);
|
||||||
|
assert.strictEqual(832, result.info.height);
|
||||||
|
assert.strictEqual(undefined, result.info.cropOffsetLeft);
|
||||||
|
assert.strictEqual(undefined, result.info.cropOffsetTop);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('Entropy-based strategy', function () {
|
||||||
|
it('JPEG', function (done) {
|
||||||
|
sharp(fixtures.inputJpg)
|
||||||
|
.resize(80, 320, {
|
||||||
|
fit: 'cover',
|
||||||
|
position: sharp.strategy.entropy
|
||||||
|
})
|
||||||
|
.toBuffer(function (err, data, info) {
|
||||||
|
if (err) throw err;
|
||||||
|
assert.strictEqual('jpeg', info.format);
|
||||||
|
assert.strictEqual(3, info.channels);
|
||||||
|
assert.strictEqual(80, info.width);
|
||||||
|
assert.strictEqual(320, info.height);
|
||||||
|
assert.strictEqual(-117, info.cropOffsetLeft);
|
||||||
|
assert.strictEqual(0, info.cropOffsetTop);
|
||||||
|
fixtures.assertSimilar(fixtures.expected('crop-strategy-entropy.jpg'), data, done);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('PNG', function (done) {
|
||||||
|
sharp(fixtures.inputPngWithTransparency)
|
||||||
|
.resize(320, 80, {
|
||||||
|
fit: 'cover',
|
||||||
|
position: sharp.strategy.entropy
|
||||||
|
})
|
||||||
|
.toBuffer(function (err, data, info) {
|
||||||
|
if (err) throw err;
|
||||||
|
assert.strictEqual('png', info.format);
|
||||||
|
assert.strictEqual(4, info.channels);
|
||||||
|
assert.strictEqual(320, info.width);
|
||||||
|
assert.strictEqual(80, info.height);
|
||||||
|
assert.strictEqual(0, info.cropOffsetLeft);
|
||||||
|
assert.strictEqual(-80, info.cropOffsetTop);
|
||||||
|
fixtures.assertSimilar(fixtures.expected('crop-strategy.png'), data, done);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('supports the strategy passed as a string', function (done) {
|
||||||
|
sharp(fixtures.inputPngWithTransparency)
|
||||||
|
.resize(320, 80, {
|
||||||
|
fit: 'cover',
|
||||||
|
position: 'entropy'
|
||||||
|
})
|
||||||
|
.toBuffer(function (err, data, info) {
|
||||||
|
if (err) throw err;
|
||||||
|
assert.strictEqual('png', info.format);
|
||||||
|
assert.strictEqual(4, info.channels);
|
||||||
|
assert.strictEqual(320, info.width);
|
||||||
|
assert.strictEqual(80, info.height);
|
||||||
|
assert.strictEqual(0, info.cropOffsetLeft);
|
||||||
|
assert.strictEqual(-80, info.cropOffsetTop);
|
||||||
|
fixtures.assertSimilar(fixtures.expected('crop-strategy.png'), data, done);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('Attention strategy', function () {
|
||||||
|
it('JPEG', function (done) {
|
||||||
|
sharp(fixtures.inputJpg)
|
||||||
|
.resize(80, 320, {
|
||||||
|
fit: 'cover',
|
||||||
|
position: sharp.strategy.attention
|
||||||
|
})
|
||||||
|
.toBuffer(function (err, data, info) {
|
||||||
|
if (err) throw err;
|
||||||
|
assert.strictEqual('jpeg', info.format);
|
||||||
|
assert.strictEqual(3, info.channels);
|
||||||
|
assert.strictEqual(80, info.width);
|
||||||
|
assert.strictEqual(320, info.height);
|
||||||
|
assert.strictEqual(-143, info.cropOffsetLeft);
|
||||||
|
assert.strictEqual(0, info.cropOffsetTop);
|
||||||
|
fixtures.assertSimilar(fixtures.expected('crop-strategy-attention.jpg'), data, done);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('PNG', function (done) {
|
||||||
|
sharp(fixtures.inputPngWithTransparency)
|
||||||
|
.resize(320, 80, {
|
||||||
|
fit: 'cover',
|
||||||
|
position: sharp.strategy.attention
|
||||||
|
})
|
||||||
|
.toBuffer(function (err, data, info) {
|
||||||
|
if (err) throw err;
|
||||||
|
assert.strictEqual('png', info.format);
|
||||||
|
assert.strictEqual(4, info.channels);
|
||||||
|
assert.strictEqual(320, info.width);
|
||||||
|
assert.strictEqual(80, info.height);
|
||||||
|
assert.strictEqual(0, info.cropOffsetLeft);
|
||||||
|
assert.strictEqual(0, info.cropOffsetTop);
|
||||||
|
fixtures.assertSimilar(fixtures.expected('crop-strategy.png'), data, done);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('supports the strategy passed as a string', function (done) {
|
||||||
|
sharp(fixtures.inputPngWithTransparency)
|
||||||
|
.resize(320, 80, {
|
||||||
|
fit: 'cover',
|
||||||
|
position: 'attention'
|
||||||
|
})
|
||||||
|
.toBuffer(function (err, data, info) {
|
||||||
|
if (err) throw err;
|
||||||
|
assert.strictEqual('png', info.format);
|
||||||
|
assert.strictEqual(4, info.channels);
|
||||||
|
assert.strictEqual(320, info.width);
|
||||||
|
assert.strictEqual(80, info.height);
|
||||||
|
assert.strictEqual(0, info.cropOffsetLeft);
|
||||||
|
assert.strictEqual(0, info.cropOffsetTop);
|
||||||
|
fixtures.assertSimilar(fixtures.expected('crop-strategy.png'), data, done);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
@@ -151,8 +151,7 @@ describe('Resize dimensions', function () {
|
|||||||
|
|
||||||
it('TIFF embed known to cause rounding errors', function (done) {
|
it('TIFF embed known to cause rounding errors', function (done) {
|
||||||
sharp(fixtures.inputTiff)
|
sharp(fixtures.inputTiff)
|
||||||
.resize(240, 320)
|
.resize(240, 320, { fit: sharp.fit.contain })
|
||||||
.embed()
|
|
||||||
.jpeg()
|
.jpeg()
|
||||||
.toBuffer(function (err, data, info) {
|
.toBuffer(function (err, data, info) {
|
||||||
if (err) throw err;
|
if (err) throw err;
|
||||||
@@ -178,10 +177,9 @@ describe('Resize dimensions', function () {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
it('Max width or height considering ratio (portrait)', function (done) {
|
it('fit=inside, portrait', function (done) {
|
||||||
sharp(fixtures.inputTiff)
|
sharp(fixtures.inputTiff)
|
||||||
.resize(320, 320)
|
.resize(320, 320, { fit: sharp.fit.inside })
|
||||||
.max()
|
|
||||||
.jpeg()
|
.jpeg()
|
||||||
.toBuffer(function (err, data, info) {
|
.toBuffer(function (err, data, info) {
|
||||||
if (err) throw err;
|
if (err) throw err;
|
||||||
@@ -193,10 +191,9 @@ describe('Resize dimensions', function () {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
it('Min width or height considering ratio (portrait)', function (done) {
|
it('fit=outside, portrait', function (done) {
|
||||||
sharp(fixtures.inputTiff)
|
sharp(fixtures.inputTiff)
|
||||||
.resize(320, 320)
|
.resize(320, 320, { fit: sharp.fit.outside })
|
||||||
.min()
|
|
||||||
.jpeg()
|
.jpeg()
|
||||||
.toBuffer(function (err, data, info) {
|
.toBuffer(function (err, data, info) {
|
||||||
if (err) throw err;
|
if (err) throw err;
|
||||||
@@ -208,10 +205,9 @@ describe('Resize dimensions', function () {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
it('Max width or height considering ratio (landscape)', function (done) {
|
it('fit=inside, landscape', function (done) {
|
||||||
sharp(fixtures.inputJpg)
|
sharp(fixtures.inputJpg)
|
||||||
.resize(320, 320)
|
.resize(320, 320, { fit: sharp.fit.inside })
|
||||||
.max()
|
|
||||||
.toBuffer(function (err, data, info) {
|
.toBuffer(function (err, data, info) {
|
||||||
if (err) throw err;
|
if (err) throw err;
|
||||||
assert.strictEqual(true, data.length > 0);
|
assert.strictEqual(true, data.length > 0);
|
||||||
@@ -222,24 +218,9 @@ describe('Resize dimensions', function () {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
it('Provide only one dimension with max, should default to crop', function (done) {
|
it('fit=outside, landscape', function (done) {
|
||||||
sharp(fixtures.inputJpg)
|
sharp(fixtures.inputJpg)
|
||||||
.resize(320)
|
.resize(320, 320, { fit: sharp.fit.outside })
|
||||||
.max()
|
|
||||||
.toBuffer(function (err, data, info) {
|
|
||||||
if (err) throw err;
|
|
||||||
assert.strictEqual(true, data.length > 0);
|
|
||||||
assert.strictEqual('jpeg', info.format);
|
|
||||||
assert.strictEqual(320, info.width);
|
|
||||||
assert.strictEqual(261, info.height);
|
|
||||||
done();
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
it('Min width or height considering ratio (landscape)', function (done) {
|
|
||||||
sharp(fixtures.inputJpg)
|
|
||||||
.resize(320, 320)
|
|
||||||
.min()
|
|
||||||
.toBuffer(function (err, data, info) {
|
.toBuffer(function (err, data, info) {
|
||||||
if (err) throw err;
|
if (err) throw err;
|
||||||
assert.strictEqual(true, data.length > 0);
|
assert.strictEqual(true, data.length > 0);
|
||||||
@@ -250,10 +231,28 @@ describe('Resize dimensions', function () {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
it('Provide only one dimension with min, should default to crop', function (done) {
|
it('fit=inside, provide only one dimension', function (done) {
|
||||||
sharp(fixtures.inputJpg)
|
sharp(fixtures.inputJpg)
|
||||||
.resize(320)
|
.resize({
|
||||||
.min()
|
width: 320,
|
||||||
|
fit: sharp.fit.inside
|
||||||
|
})
|
||||||
|
.toBuffer(function (err, data, info) {
|
||||||
|
if (err) throw err;
|
||||||
|
assert.strictEqual(true, data.length > 0);
|
||||||
|
assert.strictEqual('jpeg', info.format);
|
||||||
|
assert.strictEqual(320, info.width);
|
||||||
|
assert.strictEqual(261, info.height);
|
||||||
|
done();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('fit=outside, provide only one dimension', function (done) {
|
||||||
|
sharp(fixtures.inputJpg)
|
||||||
|
.resize({
|
||||||
|
width: 320,
|
||||||
|
fit: sharp.fit.outside
|
||||||
|
})
|
||||||
.toBuffer(function (err, data, info) {
|
.toBuffer(function (err, data, info) {
|
||||||
if (err) throw err;
|
if (err) throw err;
|
||||||
assert.strictEqual(true, data.length > 0);
|
assert.strictEqual(true, data.length > 0);
|
||||||
@@ -266,8 +265,10 @@ describe('Resize dimensions', function () {
|
|||||||
|
|
||||||
it('Do not enlarge when input width is already less than output width', function (done) {
|
it('Do not enlarge when input width is already less than output width', function (done) {
|
||||||
sharp(fixtures.inputJpg)
|
sharp(fixtures.inputJpg)
|
||||||
.resize(2800)
|
.resize({
|
||||||
.withoutEnlargement()
|
width: 2800,
|
||||||
|
withoutEnlargement: true
|
||||||
|
})
|
||||||
.toBuffer(function (err, data, info) {
|
.toBuffer(function (err, data, info) {
|
||||||
if (err) throw err;
|
if (err) throw err;
|
||||||
assert.strictEqual(true, data.length > 0);
|
assert.strictEqual(true, data.length > 0);
|
||||||
@@ -280,8 +281,10 @@ describe('Resize dimensions', function () {
|
|||||||
|
|
||||||
it('Do not enlarge when input height is already less than output height', function (done) {
|
it('Do not enlarge when input height is already less than output height', function (done) {
|
||||||
sharp(fixtures.inputJpg)
|
sharp(fixtures.inputJpg)
|
||||||
.resize(null, 2300)
|
.resize({
|
||||||
.withoutEnlargement()
|
height: 2300,
|
||||||
|
withoutEnlargement: true
|
||||||
|
})
|
||||||
.toBuffer(function (err, data, info) {
|
.toBuffer(function (err, data, info) {
|
||||||
if (err) throw err;
|
if (err) throw err;
|
||||||
assert.strictEqual(true, data.length > 0);
|
assert.strictEqual(true, data.length > 0);
|
||||||
@@ -294,8 +297,10 @@ describe('Resize dimensions', function () {
|
|||||||
|
|
||||||
it('Do enlarge when input width is less than output width', function (done) {
|
it('Do enlarge when input width is less than output width', function (done) {
|
||||||
sharp(fixtures.inputJpg)
|
sharp(fixtures.inputJpg)
|
||||||
.resize(2800)
|
.resize({
|
||||||
.withoutEnlargement(false)
|
width: 2800,
|
||||||
|
withoutEnlargement: false
|
||||||
|
})
|
||||||
.toBuffer(function (err, data, info) {
|
.toBuffer(function (err, data, info) {
|
||||||
if (err) throw err;
|
if (err) throw err;
|
||||||
assert.strictEqual(true, data.length > 0);
|
assert.strictEqual(true, data.length > 0);
|
||||||
@@ -306,103 +311,127 @@ describe('Resize dimensions', function () {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
it('Downscale width and height, ignoring aspect ratio', function (done) {
|
it('fit=fill, downscale width and height', function (done) {
|
||||||
sharp(fixtures.inputJpg).resize(320, 320).ignoreAspectRatio().toBuffer(function (err, data, info) {
|
sharp(fixtures.inputJpg)
|
||||||
if (err) throw err;
|
.resize(320, 320, { fit: 'fill' })
|
||||||
assert.strictEqual(true, data.length > 0);
|
.toBuffer(function (err, data, info) {
|
||||||
assert.strictEqual('jpeg', info.format);
|
if (err) throw err;
|
||||||
assert.strictEqual(320, info.width);
|
assert.strictEqual(true, data.length > 0);
|
||||||
assert.strictEqual(320, info.height);
|
assert.strictEqual('jpeg', info.format);
|
||||||
done();
|
assert.strictEqual(320, info.width);
|
||||||
});
|
assert.strictEqual(320, info.height);
|
||||||
|
done();
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
it('Downscale width, ignoring aspect ratio', function (done) {
|
it('fit=fill, downscale width', function (done) {
|
||||||
sharp(fixtures.inputJpg).resize(320).ignoreAspectRatio().toBuffer(function (err, data, info) {
|
sharp(fixtures.inputJpg)
|
||||||
if (err) throw err;
|
.resize({
|
||||||
assert.strictEqual(true, data.length > 0);
|
width: 320,
|
||||||
assert.strictEqual('jpeg', info.format);
|
fit: 'fill'
|
||||||
assert.strictEqual(320, info.width);
|
})
|
||||||
assert.strictEqual(2225, info.height);
|
.toBuffer(function (err, data, info) {
|
||||||
done();
|
if (err) throw err;
|
||||||
});
|
assert.strictEqual(true, data.length > 0);
|
||||||
|
assert.strictEqual('jpeg', info.format);
|
||||||
|
assert.strictEqual(320, info.width);
|
||||||
|
assert.strictEqual(2225, info.height);
|
||||||
|
done();
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
it('Downscale height, ignoring aspect ratio', function (done) {
|
it('fit=fill, downscale height', function (done) {
|
||||||
sharp(fixtures.inputJpg).resize(null, 320).ignoreAspectRatio().toBuffer(function (err, data, info) {
|
sharp(fixtures.inputJpg)
|
||||||
if (err) throw err;
|
.resize({
|
||||||
assert.strictEqual(true, data.length > 0);
|
height: 320,
|
||||||
assert.strictEqual('jpeg', info.format);
|
fit: 'fill'
|
||||||
assert.strictEqual(2725, info.width);
|
})
|
||||||
assert.strictEqual(320, info.height);
|
.toBuffer(function (err, data, info) {
|
||||||
done();
|
if (err) throw err;
|
||||||
});
|
assert.strictEqual(true, data.length > 0);
|
||||||
|
assert.strictEqual('jpeg', info.format);
|
||||||
|
assert.strictEqual(2725, info.width);
|
||||||
|
assert.strictEqual(320, info.height);
|
||||||
|
done();
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
it('Upscale width and height, ignoring aspect ratio', function (done) {
|
it('fit=fill, upscale width and height', function (done) {
|
||||||
sharp(fixtures.inputJpg).resize(3000, 3000).ignoreAspectRatio().toBuffer(function (err, data, info) {
|
sharp(fixtures.inputJpg)
|
||||||
if (err) throw err;
|
.resize(3000, 3000, { fit: 'fill' })
|
||||||
assert.strictEqual(true, data.length > 0);
|
.toBuffer(function (err, data, info) {
|
||||||
assert.strictEqual('jpeg', info.format);
|
if (err) throw err;
|
||||||
assert.strictEqual(3000, info.width);
|
assert.strictEqual(true, data.length > 0);
|
||||||
assert.strictEqual(3000, info.height);
|
assert.strictEqual('jpeg', info.format);
|
||||||
done();
|
assert.strictEqual(3000, info.width);
|
||||||
});
|
assert.strictEqual(3000, info.height);
|
||||||
|
done();
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
it('Upscale width, ignoring aspect ratio', function (done) {
|
it('fit=fill, upscale width', function (done) {
|
||||||
sharp(fixtures.inputJpg).resize(3000).ignoreAspectRatio().toBuffer(function (err, data, info) {
|
sharp(fixtures.inputJpg)
|
||||||
if (err) throw err;
|
.resize(3000, null, { fit: 'fill' })
|
||||||
assert.strictEqual(true, data.length > 0);
|
.toBuffer(function (err, data, info) {
|
||||||
assert.strictEqual('jpeg', info.format);
|
if (err) throw err;
|
||||||
assert.strictEqual(3000, info.width);
|
assert.strictEqual(true, data.length > 0);
|
||||||
assert.strictEqual(2225, info.height);
|
assert.strictEqual('jpeg', info.format);
|
||||||
done();
|
assert.strictEqual(3000, info.width);
|
||||||
});
|
assert.strictEqual(2225, info.height);
|
||||||
|
done();
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
it('Upscale height, ignoring aspect ratio', function (done) {
|
it('fit=fill, upscale height', function (done) {
|
||||||
sharp(fixtures.inputJpg).resize(null, 3000).ignoreAspectRatio().toBuffer(function (err, data, info) {
|
sharp(fixtures.inputJpg)
|
||||||
if (err) throw err;
|
.resize(null, 3000, { fit: 'fill' })
|
||||||
assert.strictEqual(true, data.length > 0);
|
.toBuffer(function (err, data, info) {
|
||||||
assert.strictEqual('jpeg', info.format);
|
if (err) throw err;
|
||||||
assert.strictEqual(2725, info.width);
|
assert.strictEqual(true, data.length > 0);
|
||||||
assert.strictEqual(3000, info.height);
|
assert.strictEqual('jpeg', info.format);
|
||||||
done();
|
assert.strictEqual(2725, info.width);
|
||||||
});
|
assert.strictEqual(3000, info.height);
|
||||||
|
done();
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
it('Downscale width, upscale height, ignoring aspect ratio', function (done) {
|
it('fit=fill, downscale width, upscale height', function (done) {
|
||||||
sharp(fixtures.inputJpg).resize(320, 3000).ignoreAspectRatio().toBuffer(function (err, data, info) {
|
sharp(fixtures.inputJpg)
|
||||||
if (err) throw err;
|
.resize(320, 3000, { fit: 'fill' })
|
||||||
assert.strictEqual(true, data.length > 0);
|
.toBuffer(function (err, data, info) {
|
||||||
assert.strictEqual('jpeg', info.format);
|
if (err) throw err;
|
||||||
assert.strictEqual(320, info.width);
|
assert.strictEqual(true, data.length > 0);
|
||||||
assert.strictEqual(3000, info.height);
|
assert.strictEqual('jpeg', info.format);
|
||||||
done();
|
assert.strictEqual(320, info.width);
|
||||||
});
|
assert.strictEqual(3000, info.height);
|
||||||
|
done();
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
it('Upscale width, downscale height, ignoring aspect ratio', function (done) {
|
it('fit=fill, upscale width, downscale height', function (done) {
|
||||||
sharp(fixtures.inputJpg).resize(3000, 320).ignoreAspectRatio().toBuffer(function (err, data, info) {
|
sharp(fixtures.inputJpg)
|
||||||
if (err) throw err;
|
.resize(3000, 320, { fit: 'fill' })
|
||||||
assert.strictEqual(true, data.length > 0);
|
.toBuffer(function (err, data, info) {
|
||||||
assert.strictEqual('jpeg', info.format);
|
if (err) throw err;
|
||||||
assert.strictEqual(3000, info.width);
|
assert.strictEqual(true, data.length > 0);
|
||||||
assert.strictEqual(320, info.height);
|
assert.strictEqual('jpeg', info.format);
|
||||||
done();
|
assert.strictEqual(3000, info.width);
|
||||||
});
|
assert.strictEqual(320, info.height);
|
||||||
|
done();
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
it('Identity transform, ignoring aspect ratio', function (done) {
|
it('fit=fill, identity transform', function (done) {
|
||||||
sharp(fixtures.inputJpg).ignoreAspectRatio().toBuffer(function (err, data, info) {
|
sharp(fixtures.inputJpg)
|
||||||
if (err) throw err;
|
.resize(null, null, { fit: 'fill' })
|
||||||
assert.strictEqual(true, data.length > 0);
|
.toBuffer(function (err, data, info) {
|
||||||
assert.strictEqual('jpeg', info.format);
|
if (err) throw err;
|
||||||
assert.strictEqual(2725, info.width);
|
assert.strictEqual(true, data.length > 0);
|
||||||
assert.strictEqual(2225, info.height);
|
assert.strictEqual('jpeg', info.format);
|
||||||
done();
|
assert.strictEqual(2725, info.width);
|
||||||
});
|
assert.strictEqual(2225, info.height);
|
||||||
|
done();
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
it('Dimensions that result in differing even shrinks on each axis', function (done) {
|
it('Dimensions that result in differing even shrinks on each axis', function (done) {
|
||||||
@@ -468,6 +497,7 @@ describe('Resize dimensions', function () {
|
|||||||
[
|
[
|
||||||
sharp.kernel.nearest,
|
sharp.kernel.nearest,
|
||||||
sharp.kernel.cubic,
|
sharp.kernel.cubic,
|
||||||
|
sharp.kernel.mitchell,
|
||||||
sharp.kernel.lanczos2,
|
sharp.kernel.lanczos2,
|
||||||
sharp.kernel.lanczos3
|
sharp.kernel.lanczos3
|
||||||
].forEach(function (kernel) {
|
].forEach(function (kernel) {
|
||||||
@@ -500,4 +530,16 @@ describe('Resize dimensions', function () {
|
|||||||
sharp().resize(null, null, { kernel: 'unknown' });
|
sharp().resize(null, null, { kernel: 'unknown' });
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
it('unknown fit throws', function () {
|
||||||
|
assert.throws(function () {
|
||||||
|
sharp().resize(null, null, { fit: 'unknown' });
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('unknown position throws', function () {
|
||||||
|
assert.throws(function () {
|
||||||
|
sharp().resize(null, null, { position: 'unknown' });
|
||||||
|
});
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -23,6 +23,33 @@ describe('Rotation', function () {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
it('Rotate by 30 degrees with semi-transparent background', function (done) {
|
||||||
|
sharp(fixtures.inputJpg)
|
||||||
|
.rotate(30, { background: { r: 255, g: 0, b: 0, alpha: 0.5 } })
|
||||||
|
.resize(320)
|
||||||
|
.png()
|
||||||
|
.toBuffer(function (err, data, info) {
|
||||||
|
if (err) throw err;
|
||||||
|
assert.strictEqual('png', info.format);
|
||||||
|
assert.strictEqual(408, info.width);
|
||||||
|
assert.strictEqual(386, info.height);
|
||||||
|
fixtures.assertSimilar(fixtures.expected('rotate-transparent-bg.png'), data, done);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('Rotate by 30 degrees with solid background', function (done) {
|
||||||
|
sharp(fixtures.inputJpg)
|
||||||
|
.rotate(30, { background: { r: 255, g: 0, b: 0, alpha: 0.5 } })
|
||||||
|
.resize(320)
|
||||||
|
.toBuffer(function (err, data, info) {
|
||||||
|
if (err) throw err;
|
||||||
|
assert.strictEqual('jpeg', info.format);
|
||||||
|
assert.strictEqual(408, info.width);
|
||||||
|
assert.strictEqual(386, info.height);
|
||||||
|
fixtures.assertSimilar(fixtures.expected('rotate-solid-bg.jpg'), data, done);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
it('Rotate by 90 degrees, respecting output input size', function (done) {
|
it('Rotate by 90 degrees, respecting output input size', function (done) {
|
||||||
sharp(fixtures.inputJpg).rotate(90).resize(320, 240).toBuffer(function (err, data, info) {
|
sharp(fixtures.inputJpg).rotate(90).resize(320, 240).toBuffer(function (err, data, info) {
|
||||||
if (err) throw err;
|
if (err) throw err;
|
||||||
@@ -34,6 +61,17 @@ describe('Rotation', function () {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
it('Rotate by 30 degrees, respecting output input size', function (done) {
|
||||||
|
sharp(fixtures.inputJpg).rotate(30).resize(320, 240).toBuffer(function (err, data, info) {
|
||||||
|
if (err) throw err;
|
||||||
|
assert.strictEqual(true, data.length > 0);
|
||||||
|
assert.strictEqual('jpeg', info.format);
|
||||||
|
assert.strictEqual(397, info.width);
|
||||||
|
assert.strictEqual(368, info.height);
|
||||||
|
done();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
[-3690, -450, -90, 90, 450, 3690].forEach(function (angle) {
|
[-3690, -450, -90, 90, 450, 3690].forEach(function (angle) {
|
||||||
it('Rotate by any 90-multiple angle (' + angle + 'deg)', function (done) {
|
it('Rotate by any 90-multiple angle (' + angle + 'deg)', function (done) {
|
||||||
sharp(fixtures.inputJpg320x240).rotate(angle).toBuffer(function (err, data, info) {
|
sharp(fixtures.inputJpg320x240).rotate(angle).toBuffer(function (err, data, info) {
|
||||||
@@ -45,6 +83,17 @@ describe('Rotation', function () {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
[-3750, -510, -150, 30, 390, 3630].forEach(function (angle) {
|
||||||
|
it('Rotate by any 30-multiple angle (' + angle + 'deg)', function (done) {
|
||||||
|
sharp(fixtures.inputJpg320x240).rotate(angle).toBuffer(function (err, data, info) {
|
||||||
|
if (err) throw err;
|
||||||
|
assert.strictEqual(397, info.width);
|
||||||
|
assert.strictEqual(368, info.height);
|
||||||
|
done();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
[-3780, -540, 0, 180, 540, 3780].forEach(function (angle) {
|
[-3780, -540, 0, 180, 540, 3780].forEach(function (angle) {
|
||||||
it('Rotate by any 180-multiple angle (' + angle + 'deg)', function (done) {
|
it('Rotate by any 180-multiple angle (' + angle + 'deg)', function (done) {
|
||||||
sharp(fixtures.inputJpg320x240).rotate(angle).toBuffer(function (err, data, info) {
|
sharp(fixtures.inputJpg320x240).rotate(angle).toBuffer(function (err, data, info) {
|
||||||
@@ -58,8 +107,7 @@ describe('Rotation', function () {
|
|||||||
|
|
||||||
it('Rotate by 270 degrees, square output ignoring aspect ratio', function (done) {
|
it('Rotate by 270 degrees, square output ignoring aspect ratio', function (done) {
|
||||||
sharp(fixtures.inputJpg)
|
sharp(fixtures.inputJpg)
|
||||||
.resize(240, 240)
|
.resize(240, 240, { fit: sharp.fit.fill })
|
||||||
.ignoreAspectRatio()
|
|
||||||
.rotate(270)
|
.rotate(270)
|
||||||
.toBuffer(function (err, data, info) {
|
.toBuffer(function (err, data, info) {
|
||||||
if (err) throw err;
|
if (err) throw err;
|
||||||
@@ -74,10 +122,26 @@ describe('Rotation', function () {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
it('Rotate by 315 degrees, square output ignoring aspect ratio', function (done) {
|
||||||
|
sharp(fixtures.inputJpg)
|
||||||
|
.resize(240, 240, { fit: sharp.fit.fill })
|
||||||
|
.rotate(315)
|
||||||
|
.toBuffer(function (err, data, info) {
|
||||||
|
if (err) throw err;
|
||||||
|
assert.strictEqual(339, info.width);
|
||||||
|
assert.strictEqual(339, info.height);
|
||||||
|
sharp(data).metadata(function (err, metadata) {
|
||||||
|
if (err) throw err;
|
||||||
|
assert.strictEqual(339, metadata.width);
|
||||||
|
assert.strictEqual(339, metadata.height);
|
||||||
|
done();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
it('Rotate by 270 degrees, rectangular output ignoring aspect ratio', function (done) {
|
it('Rotate by 270 degrees, rectangular output ignoring aspect ratio', function (done) {
|
||||||
sharp(fixtures.inputJpg)
|
sharp(fixtures.inputJpg)
|
||||||
.resize(320, 240)
|
.resize(320, 240, { fit: sharp.fit.fill })
|
||||||
.ignoreAspectRatio()
|
|
||||||
.rotate(270)
|
.rotate(270)
|
||||||
.toBuffer(function (err, data, info) {
|
.toBuffer(function (err, data, info) {
|
||||||
if (err) throw err;
|
if (err) throw err;
|
||||||
@@ -92,6 +156,23 @@ describe('Rotation', function () {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
it('Rotate by 30 degrees, rectangular output ignoring aspect ratio', function (done) {
|
||||||
|
sharp(fixtures.inputJpg)
|
||||||
|
.resize(320, 240, { fit: sharp.fit.fill })
|
||||||
|
.rotate(30)
|
||||||
|
.toBuffer(function (err, data, info) {
|
||||||
|
if (err) throw err;
|
||||||
|
assert.strictEqual(397, info.width);
|
||||||
|
assert.strictEqual(368, info.height);
|
||||||
|
sharp(data).metadata(function (err, metadata) {
|
||||||
|
if (err) throw err;
|
||||||
|
assert.strictEqual(397, metadata.width);
|
||||||
|
assert.strictEqual(368, metadata.height);
|
||||||
|
done();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
it('Input image has Orientation EXIF tag but do not rotate output', function (done) {
|
it('Input image has Orientation EXIF tag but do not rotate output', function (done) {
|
||||||
sharp(fixtures.inputJpgWithExif)
|
sharp(fixtures.inputJpgWithExif)
|
||||||
.resize(320)
|
.resize(320)
|
||||||
@@ -127,7 +208,7 @@ describe('Rotation', function () {
|
|||||||
sharp(fixtures.inputJpgWithExif)
|
sharp(fixtures.inputJpgWithExif)
|
||||||
.rotate()
|
.rotate()
|
||||||
.resize(320)
|
.resize(320)
|
||||||
.withMetadata({orientation: 3})
|
.withMetadata({ orientation: 3 })
|
||||||
.toBuffer(function (err, data, info) {
|
.toBuffer(function (err, data, info) {
|
||||||
if (err) throw err;
|
if (err) throw err;
|
||||||
assert.strictEqual('jpeg', info.format);
|
assert.strictEqual('jpeg', info.format);
|
||||||
@@ -185,9 +266,9 @@ describe('Rotation', function () {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
it('Rotate to an invalid angle, should fail', function () {
|
it('Rotate with a string argument, should fail', function () {
|
||||||
assert.throws(function () {
|
assert.throws(function () {
|
||||||
sharp(fixtures.inputJpg).rotate(1);
|
sharp(fixtures.inputJpg).rotate('not-a-number');
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|||||||
@@ -132,7 +132,7 @@ describe('Threshold', function () {
|
|||||||
it('color threshold', function (done) {
|
it('color threshold', function (done) {
|
||||||
sharp(fixtures.inputJpg)
|
sharp(fixtures.inputJpg)
|
||||||
.resize(320, 240)
|
.resize(320, 240)
|
||||||
.threshold(128, {'grayscale': false})
|
.threshold(128, { grayscale: false })
|
||||||
.toBuffer(function (err, data, info) {
|
.toBuffer(function (err, data, info) {
|
||||||
if (err) throw err;
|
if (err) throw err;
|
||||||
assert.strictEqual('jpeg', info.format);
|
assert.strictEqual('jpeg', info.format);
|
||||||
|
|||||||
@@ -9,12 +9,12 @@ describe('Tint', function () {
|
|||||||
it('tints rgb image red', function (done) {
|
it('tints rgb image red', function (done) {
|
||||||
const output = fixtures.path('output.tint-red.jpg');
|
const output = fixtures.path('output.tint-red.jpg');
|
||||||
sharp(fixtures.inputJpg)
|
sharp(fixtures.inputJpg)
|
||||||
.resize(320, 240)
|
.resize(320, 240, { fastShrinkOnLoad: false })
|
||||||
.tint('#FF0000')
|
.tint('#FF0000')
|
||||||
.toFile(output, function (err, info) {
|
.toFile(output, function (err, info) {
|
||||||
if (err) throw err;
|
if (err) throw err;
|
||||||
assert.strictEqual(true, info.size > 0);
|
assert.strictEqual(true, info.size > 0);
|
||||||
fixtures.assertMaxColourDistance(output, fixtures.expected('tint-red.jpg'), 10);
|
fixtures.assertMaxColourDistance(output, fixtures.expected('tint-red.jpg'), 18);
|
||||||
done();
|
done();
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
@@ -22,12 +22,12 @@ describe('Tint', function () {
|
|||||||
it('tints rgb image green', function (done) {
|
it('tints rgb image green', function (done) {
|
||||||
const output = fixtures.path('output.tint-green.jpg');
|
const output = fixtures.path('output.tint-green.jpg');
|
||||||
sharp(fixtures.inputJpg)
|
sharp(fixtures.inputJpg)
|
||||||
.resize(320, 240)
|
.resize(320, 240, { fastShrinkOnLoad: false })
|
||||||
.tint('#00FF00')
|
.tint('#00FF00')
|
||||||
.toFile(output, function (err, info) {
|
.toFile(output, function (err, info) {
|
||||||
if (err) throw err;
|
if (err) throw err;
|
||||||
assert.strictEqual(true, info.size > 0);
|
assert.strictEqual(true, info.size > 0);
|
||||||
fixtures.assertMaxColourDistance(output, fixtures.expected('tint-green.jpg'), 10);
|
fixtures.assertMaxColourDistance(output, fixtures.expected('tint-green.jpg'), 27);
|
||||||
done();
|
done();
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
@@ -35,12 +35,12 @@ describe('Tint', function () {
|
|||||||
it('tints rgb image blue', function (done) {
|
it('tints rgb image blue', function (done) {
|
||||||
const output = fixtures.path('output.tint-blue.jpg');
|
const output = fixtures.path('output.tint-blue.jpg');
|
||||||
sharp(fixtures.inputJpg)
|
sharp(fixtures.inputJpg)
|
||||||
.resize(320, 240)
|
.resize(320, 240, { fastShrinkOnLoad: false })
|
||||||
.tint('#0000FF')
|
.tint('#0000FF')
|
||||||
.toFile(output, function (err, info) {
|
.toFile(output, function (err, info) {
|
||||||
if (err) throw err;
|
if (err) throw err;
|
||||||
assert.strictEqual(true, info.size > 0);
|
assert.strictEqual(true, info.size > 0);
|
||||||
fixtures.assertMaxColourDistance(output, fixtures.expected('tint-blue.jpg'), 10);
|
fixtures.assertMaxColourDistance(output, fixtures.expected('tint-blue.jpg'), 14);
|
||||||
done();
|
done();
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
@@ -48,7 +48,7 @@ describe('Tint', function () {
|
|||||||
it('tints rgb image with sepia tone', function (done) {
|
it('tints rgb image with sepia tone', function (done) {
|
||||||
const output = fixtures.path('output.tint-sepia.jpg');
|
const output = fixtures.path('output.tint-sepia.jpg');
|
||||||
sharp(fixtures.inputJpg)
|
sharp(fixtures.inputJpg)
|
||||||
.resize(320, 240)
|
.resize(320, 240, { fastShrinkOnLoad: false })
|
||||||
.tint('#704214')
|
.tint('#704214')
|
||||||
.toFile(output, function (err, info) {
|
.toFile(output, function (err, info) {
|
||||||
if (err) throw err;
|
if (err) throw err;
|
||||||
@@ -62,7 +62,7 @@ describe('Tint', function () {
|
|||||||
it('tints rgb image with sepia tone with rgb colour', function (done) {
|
it('tints rgb image with sepia tone with rgb colour', function (done) {
|
||||||
const output = fixtures.path('output.tint-sepia.jpg');
|
const output = fixtures.path('output.tint-sepia.jpg');
|
||||||
sharp(fixtures.inputJpg)
|
sharp(fixtures.inputJpg)
|
||||||
.resize(320, 240)
|
.resize(320, 240, { fastShrinkOnLoad: false })
|
||||||
.tint([112, 66, 20])
|
.tint([112, 66, 20])
|
||||||
.toFile(output, function (err, info) {
|
.toFile(output, function (err, info) {
|
||||||
if (err) throw err;
|
if (err) throw err;
|
||||||
@@ -76,7 +76,7 @@ describe('Tint', function () {
|
|||||||
it('tints rgb image with alpha channel', function (done) {
|
it('tints rgb image with alpha channel', function (done) {
|
||||||
const output = fixtures.path('output.tint-alpha.png');
|
const output = fixtures.path('output.tint-alpha.png');
|
||||||
sharp(fixtures.inputPngRGBWithAlpha)
|
sharp(fixtures.inputPngRGBWithAlpha)
|
||||||
.resize(320, 240)
|
.resize(320, 240, { fastShrinkOnLoad: false })
|
||||||
.tint('#704214')
|
.tint('#704214')
|
||||||
.toFile(output, function (err, info) {
|
.toFile(output, function (err, info) {
|
||||||
if (err) throw err;
|
if (err) throw err;
|
||||||
@@ -90,12 +90,12 @@ describe('Tint', function () {
|
|||||||
it('tints cmyk image red', function (done) {
|
it('tints cmyk image red', function (done) {
|
||||||
const output = fixtures.path('output.tint-cmyk.jpg');
|
const output = fixtures.path('output.tint-cmyk.jpg');
|
||||||
sharp(fixtures.inputJpgWithCmykProfile)
|
sharp(fixtures.inputJpgWithCmykProfile)
|
||||||
.resize(320, 240)
|
.resize(320, 240, { fastShrinkOnLoad: false })
|
||||||
.tint('#FF0000')
|
.tint('#FF0000')
|
||||||
.toFile(output, function (err, info) {
|
.toFile(output, function (err, info) {
|
||||||
if (err) throw err;
|
if (err) throw err;
|
||||||
assert.strictEqual(true, info.size > 0);
|
assert.strictEqual(true, info.size > 0);
|
||||||
fixtures.assertMaxColourDistance(output, fixtures.expected('tint-cmyk.jpg'), 10);
|
fixtures.assertMaxColourDistance(output, fixtures.expected('tint-cmyk.jpg'), 15);
|
||||||
done();
|
done();
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -3,6 +3,7 @@
|
|||||||
const assert = require('assert');
|
const assert = require('assert');
|
||||||
|
|
||||||
const sharp = require('../../');
|
const sharp = require('../../');
|
||||||
|
const inRange = require('../../lib/is').inRange;
|
||||||
const fixtures = require('../fixtures');
|
const fixtures = require('../fixtures');
|
||||||
|
|
||||||
describe('Trim borders', function () {
|
describe('Trim borders', function () {
|
||||||
@@ -16,6 +17,8 @@ describe('Trim borders', function () {
|
|||||||
assert.strictEqual('png', info.format);
|
assert.strictEqual('png', info.format);
|
||||||
assert.strictEqual(450, info.width);
|
assert.strictEqual(450, info.width);
|
||||||
assert.strictEqual(322, info.height);
|
assert.strictEqual(322, info.height);
|
||||||
|
assert.strictEqual(-204, info.trimOffsetLeft);
|
||||||
|
assert.strictEqual(0, info.trimOffsetTop);
|
||||||
fixtures.assertSimilar(expected, data, done);
|
fixtures.assertSimilar(expected, data, done);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
@@ -24,12 +27,16 @@ describe('Trim borders', function () {
|
|||||||
const expected = fixtures.expected('alpha-layer-2-trim-resize.jpg');
|
const expected = fixtures.expected('alpha-layer-2-trim-resize.jpg');
|
||||||
sharp(fixtures.inputJpgOverlayLayer2)
|
sharp(fixtures.inputJpgOverlayLayer2)
|
||||||
.trim()
|
.trim()
|
||||||
.resize(300)
|
.resize({
|
||||||
|
width: 300,
|
||||||
|
fastShrinkOnLoad: false
|
||||||
|
})
|
||||||
.toBuffer(function (err, data, info) {
|
.toBuffer(function (err, data, info) {
|
||||||
if (err) throw err;
|
if (err) throw err;
|
||||||
assert.strictEqual('jpeg', info.format);
|
assert.strictEqual('jpeg', info.format);
|
||||||
assert.strictEqual(300, info.width);
|
assert.strictEqual(300, info.width);
|
||||||
assert.strictEqual(300, info.height);
|
assert.strictEqual(true, inRange(info.trimOffsetLeft, -873, -870));
|
||||||
|
assert.strictEqual(-554, info.trimOffsetTop);
|
||||||
fixtures.assertSimilar(expected, data, done);
|
fixtures.assertSimilar(expected, data, done);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
@@ -45,12 +52,14 @@ describe('Trim borders', function () {
|
|||||||
assert.strictEqual(32, info.width);
|
assert.strictEqual(32, info.width);
|
||||||
assert.strictEqual(32, info.height);
|
assert.strictEqual(32, info.height);
|
||||||
assert.strictEqual(4, info.channels);
|
assert.strictEqual(4, info.channels);
|
||||||
|
assert.strictEqual(-2, info.trimOffsetLeft);
|
||||||
|
assert.strictEqual(-2, info.trimOffsetTop);
|
||||||
fixtures.assertSimilar(fixtures.expected('trim-16bit-rgba.png'), data, done);
|
fixtures.assertSimilar(fixtures.expected('trim-16bit-rgba.png'), data, done);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
describe('Invalid thresholds', function () {
|
describe('Invalid thresholds', function () {
|
||||||
[-1, 100, 'fail', {}].forEach(function (threshold) {
|
[-1, 'fail', {}].forEach(function (threshold) {
|
||||||
it(JSON.stringify(threshold), function () {
|
it(JSON.stringify(threshold), function () {
|
||||||
assert.throws(function () {
|
assert.throws(function () {
|
||||||
sharp().trim(threshold);
|
sharp().trim(threshold);
|
||||||
|
|||||||