Compare commits
162 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
531a0402f7 | ||
|
|
cb10f9a9c8 | ||
|
|
c808139b02 | ||
|
|
e0d58266be | ||
|
|
1b7c5816fc | ||
|
|
b224874332 | ||
|
|
ef61da3051 | ||
|
|
f214269aa1 | ||
|
|
6bc2ea8dc7 | ||
|
|
71fb839e2b | ||
|
|
8c9c070caf | ||
|
|
b2d7d4c4a9 | ||
|
|
0ac7fbfc07 | ||
|
|
ebfc897bcf | ||
|
|
c66495b66c | ||
|
|
24fb0c33c2 | ||
|
|
25b63a2fb4 | ||
|
|
e576165cf1 | ||
|
|
fe2eccef39 | ||
|
|
0e0e746a0d | ||
|
|
5b4f4b0672 | ||
|
|
185fcfe635 | ||
|
|
2034efcf55 | ||
|
|
38ddb3b866 | ||
|
|
f950294f70 | ||
|
|
86815bc9c4 | ||
|
|
bb37dc1ea6 | ||
|
|
d92ea31858 | ||
|
|
55f204c6f9 | ||
|
|
e97909f776 | ||
|
|
c210ac73cc | ||
|
|
962c91daf0 | ||
|
|
df33c3024a | ||
|
|
62e04f7784 | ||
|
|
32fcb771ca | ||
|
|
a21760b374 | ||
|
|
cd05c7814a | ||
|
|
7b12f091e8 | ||
|
|
e149e60c7a | ||
|
|
bdac84059d | ||
|
|
2d05804fc3 | ||
|
|
2a56de69cc | ||
|
|
6ca2a4a9cd | ||
|
|
a9eb65c462 | ||
|
|
afb30b3695 | ||
|
|
09b019ed13 | ||
|
|
d46ac3a478 | ||
|
|
677b2b9089 | ||
|
|
5c1067c63f | ||
|
|
736c04a7a4 | ||
|
|
0e29c55d13 | ||
|
|
ca49e6079c | ||
|
|
320a7464c7 | ||
|
|
da74cd078f | ||
|
|
322aa60891 | ||
|
|
e380576da2 | ||
|
|
cf7664a854 | ||
|
|
56508e8d79 | ||
|
|
2656c69d99 | ||
|
|
57c1e3ae26 | ||
|
|
2675b2265b | ||
|
|
41e50770d1 | ||
|
|
b3d6e94984 | ||
|
|
5c9c17f1f6 | ||
|
|
11329d5e09 | ||
|
|
8843211e12 | ||
|
|
20e75dc50b | ||
|
|
d2e5441d6e | ||
|
|
0ffa1e72d0 | ||
|
|
a0e034a9e9 | ||
|
|
3c7cbf8685 | ||
|
|
7541dfcab2 | ||
|
|
dc2b79ac9a | ||
|
|
6d62051877 | ||
|
|
61b86744d7 | ||
|
|
fd5b4a131f | ||
|
|
32c4b9eff1 | ||
|
|
95cf35efc5 | ||
|
|
58e6368525 | ||
|
|
16e0d54b15 | ||
|
|
be381e4440 | ||
|
|
9982182926 | ||
|
|
607d157b76 | ||
|
|
e21277ceba | ||
|
|
8012733a52 | ||
|
|
01a1377972 | ||
|
|
37e4b9b5ba | ||
|
|
8a3098604c | ||
|
|
5febce7a59 | ||
|
|
3dbedf1fb6 | ||
|
|
0ae619dfc5 | ||
|
|
05dd191e17 | ||
|
|
c9ecc7a517 | ||
|
|
434a433a09 | ||
|
|
3de54d897c | ||
|
|
530c2a9fcf | ||
|
|
60b8b92630 | ||
|
|
5842da22d8 | ||
|
|
9850e3dae0 | ||
|
|
3af62446fc | ||
|
|
1f71dade67 | ||
|
|
8be664b66f | ||
|
|
c0be4f1307 | ||
|
|
d9c754f5c1 | ||
|
|
33a175eafb | ||
|
|
7c990b3ab3 | ||
|
|
5dfeaa9fd1 | ||
|
|
84fd1caa46 | ||
|
|
2678d761ba | ||
|
|
ede2ee9ce3 | ||
|
|
20f468991f | ||
|
|
58d9e0fef7 | ||
|
|
d7278f022b | ||
|
|
7383596f8c | ||
|
|
f6831ab46b | ||
|
|
7cf0f95ed1 | ||
|
|
bf6b894480 | ||
|
|
ee8fcb6109 | ||
|
|
05cec013fe | ||
|
|
f4cbbd7b79 | ||
|
|
2129adfcc3 | ||
|
|
9f59a2aebf | ||
|
|
26fb75bf3f | ||
|
|
25e5f27785 | ||
|
|
ef62daccf9 | ||
|
|
9067c0a000 | ||
|
|
79470d2e07 | ||
|
|
3cefa6f2bf | ||
|
|
75d954a6bc | ||
|
|
1b7e3746cc | ||
|
|
29252d9dbb | ||
|
|
23e14861be | ||
|
|
97960b5f91 | ||
|
|
18c4ad9adf | ||
|
|
b240c53633 | ||
|
|
660f3d58be | ||
|
|
b6d75cda8e | ||
|
|
e07356c11c | ||
|
|
82e215a42e | ||
|
|
cc1c36d891 | ||
|
|
a1a2d7de5c | ||
|
|
6dce2deb82 | ||
|
|
cdad84edc6 | ||
|
|
de842a67d8 | ||
|
|
918bbe88c6 | ||
|
|
7d3891989c | ||
|
|
168fe7c8d9 | ||
|
|
29ab8408fb | ||
|
|
692e2d4df4 | ||
|
|
85d86dbede | ||
|
|
ce2d7b8efc | ||
|
|
be00d72d82 | ||
|
|
5b376364f5 | ||
|
|
409d15c624 | ||
|
|
ce22388c3b | ||
|
|
30143cf509 | ||
|
|
78f31d2b0d | ||
|
|
4e67a5025a | ||
|
|
b7e0a6f277 | ||
|
|
045680fba5 | ||
|
|
692347cc6b | ||
|
|
faa515d969 |
12
.editorconfig
Normal file
@@ -0,0 +1,12 @@
|
|||||||
|
# http://editorconfig.org
|
||||||
|
root = true
|
||||||
|
|
||||||
|
[*]
|
||||||
|
indent_style = space
|
||||||
|
indent_size = 2
|
||||||
|
charset = utf-8
|
||||||
|
trim_trailing_whitespace = true
|
||||||
|
insert_final_newline = true
|
||||||
|
|
||||||
|
[*.md]
|
||||||
|
trim_trailing_whitespace = false
|
||||||
6
.gitignore
vendored
@@ -4,6 +4,8 @@ coverage
|
|||||||
test/bench/node_modules
|
test/bench/node_modules
|
||||||
test/fixtures/output*
|
test/fixtures/output*
|
||||||
test/leak/libvips.supp
|
test/leak/libvips.supp
|
||||||
|
lib
|
||||||
# Mac OS X
|
include
|
||||||
|
packaging/libvips*
|
||||||
|
packaging/*.log
|
||||||
.DS_Store
|
.DS_Store
|
||||||
|
|||||||
@@ -4,7 +4,8 @@
|
|||||||
"maxparams": 4,
|
"maxparams": 4,
|
||||||
"maxcomplexity": 13,
|
"maxcomplexity": 13,
|
||||||
"globals": {
|
"globals": {
|
||||||
"before": true,
|
"beforeEach": true,
|
||||||
|
"afterEach": true,
|
||||||
"describe": true,
|
"describe": true,
|
||||||
"it": true
|
"it": true
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,10 +1,16 @@
|
|||||||
build
|
build
|
||||||
node_modules
|
node_modules
|
||||||
coverage
|
coverage
|
||||||
|
.editorconfig
|
||||||
.jshintignore
|
.jshintignore
|
||||||
.jshintrc
|
.jshintrc
|
||||||
.gitignore
|
.gitignore
|
||||||
test
|
test
|
||||||
.travis.yml
|
.travis.yml
|
||||||
appveyor.yml
|
appveyor.yml
|
||||||
|
circle.yml
|
||||||
mkdocs.yml
|
mkdocs.yml
|
||||||
|
lib
|
||||||
|
include
|
||||||
|
packaging
|
||||||
|
preinstall.sh
|
||||||
|
|||||||
18
.travis.yml
@@ -2,9 +2,21 @@ language: node_js
|
|||||||
node_js:
|
node_js:
|
||||||
- "0.10"
|
- "0.10"
|
||||||
- "0.12"
|
- "0.12"
|
||||||
- "iojs-v1"
|
- "4"
|
||||||
- "iojs-v2"
|
- "5"
|
||||||
|
os:
|
||||||
|
- linux
|
||||||
|
- osx
|
||||||
|
sudo: false
|
||||||
|
addons:
|
||||||
|
apt:
|
||||||
|
sources:
|
||||||
|
- ubuntu-toolchain-r-test
|
||||||
|
packages:
|
||||||
|
- g++-4.8
|
||||||
|
osx_image: xcode7.3
|
||||||
before_install:
|
before_install:
|
||||||
- curl -s https://raw.githubusercontent.com/lovell/sharp/master/preinstall.sh | sudo bash -
|
- if [[ "$TRAVIS_OS_NAME" == "linux" ]]; then export CXX=g++-4.8; fi
|
||||||
|
- if [[ "$TRAVIS_OS_NAME" == "osx" ]]; then brew install homebrew/science/vips; fi
|
||||||
after_success:
|
after_success:
|
||||||
- cat ./coverage/lcov.info | ./node_modules/coveralls/bin/coveralls.js
|
- cat ./coverage/lcov.info | ./node_modules/coveralls/bin/coveralls.js
|
||||||
|
|||||||
@@ -6,16 +6,10 @@ Hello, thank you for your interest in helping!
|
|||||||
|
|
||||||
Please create a [new issue](https://github.com/lovell/sharp/issues/new) containing the steps to reproduce the problem.
|
Please create a [new issue](https://github.com/lovell/sharp/issues/new) containing the steps to reproduce the problem.
|
||||||
|
|
||||||
|
If you're having installation problems, please include the output of running `npm install --verbose sharp`.
|
||||||
|
|
||||||
New bugs are assigned a `triage` label whilst under investigation.
|
New bugs are assigned a `triage` label whilst under investigation.
|
||||||
|
|
||||||
If you're having problems with `npm install sharp`, please include the output of the following commands, perhaps as a [gist](https://gist.github.com/):
|
|
||||||
|
|
||||||
```sh
|
|
||||||
vips -v
|
|
||||||
pkg-config --print-provides vips
|
|
||||||
npm install --verbose sharp
|
|
||||||
```
|
|
||||||
|
|
||||||
## Submit a new feature request
|
## Submit a new feature request
|
||||||
|
|
||||||
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.
|
||||||
@@ -47,9 +41,9 @@ Any change that modifies the existing public API should be added to the relevant
|
|||||||
|
|
||||||
| Release | WIP branch |
|
| Release | WIP branch |
|
||||||
| ------: | :--------- |
|
| ------: | :--------- |
|
||||||
| v0.11.0 | knife |
|
| v0.15.0 | outfit |
|
||||||
| v0.12.0 | look |
|
| v0.16.0 | pencil |
|
||||||
| v0.13.0 | mind |
|
| v0.17.0 | quill |
|
||||||
|
|
||||||
Please squash your changes into a single commit using a command like `git rebase -i upstream/<wip-branch>`.
|
Please squash your changes into a single commit using a command like `git rebase -i upstream/<wip-branch>`.
|
||||||
|
|
||||||
@@ -85,6 +79,15 @@ Requires [Valgrind](http://valgrind.org/).
|
|||||||
npm run test-leak
|
npm run test-leak
|
||||||
```
|
```
|
||||||
|
|
||||||
|
### Packaging tests
|
||||||
|
|
||||||
|
Tests the installation on a number of Linux-based operating systems.
|
||||||
|
Requires docker.
|
||||||
|
|
||||||
|
```sh
|
||||||
|
npm run test-packaging
|
||||||
|
```
|
||||||
|
|
||||||
## Finally
|
## Finally
|
||||||
|
|
||||||
Please feel free to ask any questions via a [new issue](https://github.com/lovell/sharp/issues/new) or contact me by [e-mail](https://github.com/lovell/sharp/blob/master/package.json#L4).
|
Please feel free to ask any questions via a [new issue](https://github.com/lovell/sharp/issues/new) or contact me by [e-mail](https://github.com/lovell/sharp/blob/master/package.json#L4).
|
||||||
|
|||||||
25
README.md
@@ -4,13 +4,30 @@ The typical use case for this high speed Node.js module
|
|||||||
is to convert large images of many formats to
|
is to convert large images of many formats to
|
||||||
smaller, web-friendly JPEG, PNG and WebP images of varying dimensions.
|
smaller, web-friendly JPEG, PNG and WebP images of varying dimensions.
|
||||||
|
|
||||||
|
Resizing an image is typically 4x faster than using the
|
||||||
|
quickest ImageMagick and GraphicsMagick settings.
|
||||||
|
|
||||||
|
Colour spaces, embedded ICC profiles and alpha transparency channels are all handled correctly.
|
||||||
|
Bicubic interpolation with Lanczos anti-alias filtering ensures quality is not sacrificed for speed.
|
||||||
|
|
||||||
|
As well as image resizing, operations such as
|
||||||
|
rotation, extraction, compositing and gamma correction are available.
|
||||||
|
|
||||||
|
Most Windows (x64), Linux and ARMv6+ systems do not require
|
||||||
|
the installation of any external runtime dependencies.
|
||||||
|
|
||||||
|
Use with OS X is as simple as running `brew install homebrew/science/vips`
|
||||||
|
to install the libvips dependency.
|
||||||
|
|
||||||
[](https://coveralls.io/r/lovell/sharp?branch=master)
|
[](https://coveralls.io/r/lovell/sharp?branch=master)
|
||||||
|
|
||||||
### Documentation
|
### Documentation
|
||||||
|
|
||||||
Visit [sharp.dimens.io](http://sharp.dimens.io/) for
|
Visit [sharp.dimens.io](http://sharp.dimens.io/) for complete
|
||||||
complete installation instructions, API documentation,
|
[installation instructions](http://sharp.dimens.io/page/install),
|
||||||
benchmark tests and a changelog.
|
[API documentation](http://sharp.dimens.io/page/api),
|
||||||
|
[benchmark tests](http://sharp.dimens.io/page/performance) and
|
||||||
|
[changelog](http://sharp.dimens.io/page/changelog).
|
||||||
|
|
||||||
### Contributing
|
### Contributing
|
||||||
|
|
||||||
@@ -19,7 +36,7 @@ covers reporting bugs, requesting features and submitting code changes.
|
|||||||
|
|
||||||
### Licence
|
### Licence
|
||||||
|
|
||||||
Copyright 2013, 2014, 2015 Lovell Fuller and contributors.
|
Copyright 2013, 2014, 2015, 2016 Lovell Fuller and contributors.
|
||||||
|
|
||||||
Licensed under the Apache License, Version 2.0 (the "License");
|
Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
you may not use this file except in compliance with the License.
|
you may not use this file except in compliance with the License.
|
||||||
|
|||||||
21
appveyor.yml
@@ -3,24 +3,13 @@ version: "{build}"
|
|||||||
build: off
|
build: off
|
||||||
platform: x64
|
platform: x64
|
||||||
environment:
|
environment:
|
||||||
VIPS_VERSION_MAJOR_MINOR: 8.0
|
|
||||||
VIPS_VERSION_PATCH: 2
|
|
||||||
VIPS_WARNING: 0
|
VIPS_WARNING: 0
|
||||||
matrix:
|
matrix:
|
||||||
- nodejs_version: "0.10"
|
|
||||||
nodejs_exec: "node"
|
|
||||||
- nodejs_version: "0.12"
|
- nodejs_version: "0.12"
|
||||||
nodejs_exec: "node"
|
- nodejs_version: "4"
|
||||||
- nodejs_version: "2"
|
- nodejs_version: "5"
|
||||||
nodejs_exec: "iojs"
|
|
||||||
install:
|
install:
|
||||||
- ps: $env:VIPS_VERSION = "$env:VIPS_VERSION_MAJOR_MINOR.$env:VIPS_VERSION_PATCH"
|
- ps: Install-Product node $env:nodejs_version x64
|
||||||
- ps: Write-Output "Fetching http://www.vips.ecs.soton.ac.uk/supported/$env:VIPS_VERSION_MAJOR_MINOR/win32/vips-dev-$env:VIPS_VERSION.zip"
|
- npm install
|
||||||
- ps: Start-FileDownload http://www.vips.ecs.soton.ac.uk/supported/$env:VIPS_VERSION_MAJOR_MINOR/win32/vips-dev-$env:VIPS_VERSION.zip -FileName c:\vips-dev-$env:VIPS_VERSION.zip
|
|
||||||
- ps: Invoke-Expression "& 7z -y x c:\vips-dev-$env:VIPS_VERSION.zip -oc:\ | FIND /V `"ing `""
|
|
||||||
- ps: $env:VIPS_HOME = "c:\vips-dev-$env:VIPS_VERSION"
|
|
||||||
- ps: $env:PATH = "$env:VIPS_HOME\bin;$env:PATH"
|
|
||||||
- ps: Install-Product node $env:nodejs_version x86
|
|
||||||
- npm install --arch=ia32 --msvs_version=2013
|
|
||||||
test_script:
|
test_script:
|
||||||
- npm run-script test-win32-%nodejs_exec%
|
- npm run-script test-win
|
||||||
|
|||||||
252
binding.gyp
Executable file → Normal file
@@ -1,6 +1,87 @@
|
|||||||
{
|
{
|
||||||
'targets': [{
|
'targets': [{
|
||||||
|
'target_name': 'libvips-cpp',
|
||||||
|
'conditions': [
|
||||||
|
['OS == "win"', {
|
||||||
|
# Build libvips C++ binding for Windows due to MSVC std library ABI changes
|
||||||
|
'type': 'shared_library',
|
||||||
|
'variables': {
|
||||||
|
'download_vips': '<!(node -e "require(\'./binding\').download_vips()")'
|
||||||
|
},
|
||||||
|
'defines': [
|
||||||
|
'VIPS_CPLUSPLUS_EXPORTS'
|
||||||
|
],
|
||||||
|
'sources': [
|
||||||
|
'src/libvips/cplusplus/VError.cpp',
|
||||||
|
'src/libvips/cplusplus/VInterpolate.cpp',
|
||||||
|
'src/libvips/cplusplus/VImage.cpp'
|
||||||
|
],
|
||||||
|
'include_dirs': [
|
||||||
|
'<(module_root_dir)/include',
|
||||||
|
'<(module_root_dir)/include/glib-2.0',
|
||||||
|
'<(module_root_dir)/lib/glib-2.0/include'
|
||||||
|
],
|
||||||
|
'libraries': [
|
||||||
|
'<(module_root_dir)/lib/libvips.lib',
|
||||||
|
'<(module_root_dir)/lib/libglib-2.0.lib',
|
||||||
|
'<(module_root_dir)/lib/libgobject-2.0.lib'
|
||||||
|
],
|
||||||
|
'configurations': {
|
||||||
|
'Release': {
|
||||||
|
'msvs_settings': {
|
||||||
|
'VCCLCompilerTool': {
|
||||||
|
'ExceptionHandling': 1
|
||||||
|
}
|
||||||
|
},
|
||||||
|
'msvs_disabled_warnings': [
|
||||||
|
4275
|
||||||
|
]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}, {
|
||||||
|
# Ignore this target for non-Windows
|
||||||
|
'type': 'none'
|
||||||
|
}]
|
||||||
|
]
|
||||||
|
}, {
|
||||||
'target_name': 'sharp',
|
'target_name': 'sharp',
|
||||||
|
'dependencies': [
|
||||||
|
'libvips-cpp'
|
||||||
|
],
|
||||||
|
# Nested variables "pattern" borrowed from http://src.chromium.org/viewvc/chrome/trunk/src/build/common.gypi
|
||||||
|
'variables': {
|
||||||
|
'variables': {
|
||||||
|
'variables': {
|
||||||
|
'conditions': [
|
||||||
|
['OS != "win"', {
|
||||||
|
# Build the PKG_CONFIG_PATH environment variable with all possible combinations
|
||||||
|
'pkg_config_path': '<!(which brew >/dev/null 2>&1 && eval $(brew --env) && echo $PKG_CONFIG_LIBDIR || true):$PKG_CONFIG_PATH:/usr/local/lib/pkgconfig:/usr/lib/pkgconfig'
|
||||||
|
}, {
|
||||||
|
'pkg_config_path': ''
|
||||||
|
}]
|
||||||
|
],
|
||||||
|
},
|
||||||
|
'conditions': [
|
||||||
|
['OS != "win"', {
|
||||||
|
# Which version, if any, of libvips is available globally via pkg-config?
|
||||||
|
'global_vips_version': '<!(PKG_CONFIG_PATH="<(pkg_config_path)" pkg-config --modversion vips-cpp 2>/dev/null || true)'
|
||||||
|
}, {
|
||||||
|
'global_vips_version': ''
|
||||||
|
}]
|
||||||
|
],
|
||||||
|
'pkg_config_path%': '<(pkg_config_path)'
|
||||||
|
},
|
||||||
|
'pkg_config_path%': '<(pkg_config_path)',
|
||||||
|
'runtime_link%': 'shared',
|
||||||
|
'conditions': [
|
||||||
|
['OS != "win"', {
|
||||||
|
# Does the globally available version of libvips, if any, meet the minimum version requirement?
|
||||||
|
'use_global_vips': '<!(GLOBAL_VIPS_VERSION="<(global_vips_version)" node -e "require(\'./binding\').use_global_vips()")'
|
||||||
|
}, {
|
||||||
|
'use_global_vips': ''
|
||||||
|
}]
|
||||||
|
]
|
||||||
|
},
|
||||||
'sources': [
|
'sources': [
|
||||||
'src/common.cc',
|
'src/common.cc',
|
||||||
'src/metadata.cc',
|
'src/metadata.cc',
|
||||||
@@ -9,50 +90,71 @@
|
|||||||
'src/sharp.cc',
|
'src/sharp.cc',
|
||||||
'src/utilities.cc'
|
'src/utilities.cc'
|
||||||
],
|
],
|
||||||
|
'defines': [
|
||||||
|
'_GLIBCXX_USE_CXX11_ABI=0'
|
||||||
|
],
|
||||||
|
'include_dirs': [
|
||||||
|
'<!(node -e "require(\'nan\')")'
|
||||||
|
],
|
||||||
'conditions': [
|
'conditions': [
|
||||||
['OS=="win"', {
|
['use_global_vips == "true"', {
|
||||||
'library_dirs': [
|
# Use pkg-config for include and lib
|
||||||
'$(VIPS_HOME)/lib'
|
'include_dirs': ['<!@(PKG_CONFIG_PATH="<(pkg_config_path)" pkg-config --cflags-only-I vips-cpp vips glib-2.0 | sed s\/-I//g)'],
|
||||||
],
|
'conditions': [
|
||||||
|
['runtime_link == "static"', {
|
||||||
|
'libraries': ['<!@(PKG_CONFIG_PATH="<(pkg_config_path)" pkg-config --libs --static vips-cpp)']
|
||||||
|
}, {
|
||||||
|
'libraries': ['<!@(PKG_CONFIG_PATH="<(pkg_config_path)" pkg-config --libs vips-cpp)']
|
||||||
|
}]
|
||||||
|
]
|
||||||
|
}, {
|
||||||
|
# Attempt to download pre-built libvips and install locally within node_modules
|
||||||
|
'include_dirs': [
|
||||||
|
'<(module_root_dir)/include',
|
||||||
|
'<(module_root_dir)/include/glib-2.0',
|
||||||
|
'<(module_root_dir)/lib/glib-2.0/include'
|
||||||
|
],
|
||||||
|
'conditions': [
|
||||||
|
['OS == "win"', {
|
||||||
'libraries': [
|
'libraries': [
|
||||||
'libvips.dll.a',
|
'<(module_root_dir)/lib/libvips.lib',
|
||||||
'glib-2.0.lib',
|
'<(module_root_dir)/lib/libglib-2.0.lib',
|
||||||
'gobject-2.0.lib',
|
'<(module_root_dir)/lib/libgobject-2.0.lib'
|
||||||
'gthread-2.0.lib',
|
|
||||||
'gmodule-2.0.lib',
|
|
||||||
'liblcms2.dll.a',
|
|
||||||
'libxml2.lib',
|
|
||||||
'intl.lib',
|
|
||||||
'libjpeg.dll.a',
|
|
||||||
'libexif.dll.a',
|
|
||||||
'libpng.lib',
|
|
||||||
'libtiff.dll.a',
|
|
||||||
'libMagickWand-6.Q16.dll.a',
|
|
||||||
'libMagickCore-6.Q16.dll.a',
|
|
||||||
'pango-1.0.lib',
|
|
||||||
'pangoft2-1.0.lib',
|
|
||||||
'libgsf-1.dll.a',
|
|
||||||
'libopenslide.dll.a',
|
|
||||||
'libfftw3.dll.a'
|
|
||||||
],
|
|
||||||
'include_dirs': [
|
|
||||||
'$(VIPS_HOME)/include',
|
|
||||||
'$(VIPS_HOME)/include/glib-2.0',
|
|
||||||
'$(VIPS_HOME)/lib/glib-2.0/include',
|
|
||||||
'<!(node -e "require(\'nan\')")'
|
|
||||||
]
|
]
|
||||||
}, {
|
}],
|
||||||
|
['OS == "linux"', {
|
||||||
'variables': {
|
'variables': {
|
||||||
'PKG_CONFIG_PATH': '<!(which brew >/dev/null 2>&1 && eval $(brew --env) && echo $PKG_CONFIG_LIBDIR || true):$PKG_CONFIG_PATH:/usr/local/lib/pkgconfig:/usr/lib/pkgconfig'
|
'download_vips': '<!(LDD_VERSION="<!(ldd --version 2>&1 || true)" node -e "require(\'./binding\').download_vips()")'
|
||||||
},
|
},
|
||||||
'libraries': [
|
'libraries': [
|
||||||
'<!(PKG_CONFIG_PATH="<(PKG_CONFIG_PATH)" pkg-config --libs vips)'
|
'<(module_root_dir)/lib/libvips-cpp.so',
|
||||||
],
|
'<(module_root_dir)/lib/libvips.so',
|
||||||
'include_dirs': [
|
'<(module_root_dir)/lib/libglib-2.0.so',
|
||||||
'<!(PKG_CONFIG_PATH="<(PKG_CONFIG_PATH)" pkg-config --cflags vips glib-2.0)',
|
'<(module_root_dir)/lib/libgobject-2.0.so',
|
||||||
'<!(node -e "require(\'nan\')")'
|
# Dependencies of dependencies, included for openSUSE support
|
||||||
|
'<(module_root_dir)/lib/libGraphicsMagick.so',
|
||||||
|
'<(module_root_dir)/lib/libGraphicsMagickWand.so',
|
||||||
|
'<(module_root_dir)/lib/libexif.so',
|
||||||
|
'<(module_root_dir)/lib/libgio-2.0.so',
|
||||||
|
'<(module_root_dir)/lib/libgmodule-2.0.so',
|
||||||
|
'<(module_root_dir)/lib/libgsf-1.so',
|
||||||
|
'<(module_root_dir)/lib/libjpeg.so',
|
||||||
|
'<(module_root_dir)/lib/libpng.so',
|
||||||
|
'<(module_root_dir)/lib/libtiff.so',
|
||||||
|
'<(module_root_dir)/lib/libwebp.so',
|
||||||
|
'<(module_root_dir)/lib/libz.so',
|
||||||
|
'<(module_root_dir)/lib/libffi.so',
|
||||||
|
'<(module_root_dir)/lib/libgthread-2.0.so',
|
||||||
|
'<(module_root_dir)/lib/liblcms2.so',
|
||||||
|
'<(module_root_dir)/lib/libpng16.so',
|
||||||
|
'<(module_root_dir)/lib/libxml2.so',
|
||||||
|
'<(module_root_dir)/lib/liborc-0.4.so',
|
||||||
|
# Ensure runtime linking is relative to sharp.node
|
||||||
|
'-Wl,-rpath=\'$${ORIGIN}/../../lib\''
|
||||||
]
|
]
|
||||||
}]
|
}]
|
||||||
|
]
|
||||||
|
}]
|
||||||
],
|
],
|
||||||
'cflags_cc': [
|
'cflags_cc': [
|
||||||
'-std=c++0x',
|
'-std=c++0x',
|
||||||
@@ -61,28 +163,82 @@
|
|||||||
'-O3'
|
'-O3'
|
||||||
],
|
],
|
||||||
'xcode_settings': {
|
'xcode_settings': {
|
||||||
|
'CLANG_CXX_LANGUAGE_STANDARD': 'c++11',
|
||||||
|
'CLANG_CXX_LIBRARY': 'libc++',
|
||||||
|
'MACOSX_DEPLOYMENT_TARGET': '10.7',
|
||||||
|
'GCC_ENABLE_CPP_EXCEPTIONS': 'YES',
|
||||||
|
'GCC_ENABLE_CPP_RTTI': 'YES',
|
||||||
'OTHER_CPLUSPLUSFLAGS': [
|
'OTHER_CPLUSPLUSFLAGS': [
|
||||||
'-std=c++11',
|
|
||||||
'-stdlib=libc++',
|
|
||||||
'-fexceptions',
|
'-fexceptions',
|
||||||
'-Wall',
|
'-Wall',
|
||||||
'-O3'
|
'-O3'
|
||||||
],
|
]
|
||||||
'MACOSX_DEPLOYMENT_TARGET': '10.7'
|
|
||||||
},
|
|
||||||
'msvs_settings': {
|
|
||||||
'VCCLCompilerTool': {
|
|
||||||
'ExceptionHandling': 1 # /EHsc
|
|
||||||
}
|
|
||||||
},
|
},
|
||||||
'configurations': {
|
'configurations': {
|
||||||
'Release': {
|
'Release': {
|
||||||
'msvs_settings': {
|
'msvs_settings': {
|
||||||
'VCCLCompilerTool': {
|
'VCCLCompilerTool': {
|
||||||
'ExceptionHandling': 1,
|
'ExceptionHandling': 1
|
||||||
}
|
}
|
||||||
}
|
},
|
||||||
|
'msvs_disabled_warnings': [
|
||||||
|
4275
|
||||||
|
]
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
}, {
|
||||||
|
'target_name': 'win_copy_dlls',
|
||||||
|
'type': 'none',
|
||||||
|
'dependencies': [
|
||||||
|
'sharp'
|
||||||
|
],
|
||||||
|
'conditions': [
|
||||||
|
['OS == "win"', {
|
||||||
|
# Windows lacks support for rpath
|
||||||
|
'copies': [{
|
||||||
|
'destination': '<(module_root_dir)/build/Release',
|
||||||
|
'files': [
|
||||||
|
'<(module_root_dir)/lib/GNU.Gettext.dll',
|
||||||
|
'<(module_root_dir)/lib/libMagickCore-6.Q16-2.dll',
|
||||||
|
'<(module_root_dir)/lib/libMagickWand-6.Q16-2.dll',
|
||||||
|
'<(module_root_dir)/lib/libasprintf-0.dll',
|
||||||
|
'<(module_root_dir)/lib/libcairo-2.dll',
|
||||||
|
'<(module_root_dir)/lib/libcairo-gobject-2.dll',
|
||||||
|
'<(module_root_dir)/lib/libcairo-script-interpreter-2.dll',
|
||||||
|
'<(module_root_dir)/lib/libexif-12.dll',
|
||||||
|
'<(module_root_dir)/lib/libexpat-1.dll',
|
||||||
|
'<(module_root_dir)/lib/libffi-6.dll',
|
||||||
|
'<(module_root_dir)/lib/libfftw3-3.dll',
|
||||||
|
'<(module_root_dir)/lib/libfontconfig-1.dll',
|
||||||
|
'<(module_root_dir)/lib/libfreetype-6.dll',
|
||||||
|
'<(module_root_dir)/lib/libgcc_s_seh-1.dll',
|
||||||
|
'<(module_root_dir)/lib/libgdk_pixbuf-2.0-0.dll',
|
||||||
|
'<(module_root_dir)/lib/libgio-2.0-0.dll',
|
||||||
|
'<(module_root_dir)/lib/libglib-2.0-0.dll',
|
||||||
|
'<(module_root_dir)/lib/libgmodule-2.0-0.dll',
|
||||||
|
'<(module_root_dir)/lib/libgobject-2.0-0.dll',
|
||||||
|
'<(module_root_dir)/lib/libgsf-1-114.dll',
|
||||||
|
'<(module_root_dir)/lib/libgthread-2.0-0.dll',
|
||||||
|
'<(module_root_dir)/lib/libintl-8.dll',
|
||||||
|
'<(module_root_dir)/lib/libjpeg-62.dll',
|
||||||
|
'<(module_root_dir)/lib/liblcms2-2.dll',
|
||||||
|
'<(module_root_dir)/lib/libopenjp2.dll',
|
||||||
|
'<(module_root_dir)/lib/libopenslide-0.dll',
|
||||||
|
'<(module_root_dir)/lib/libpango-1.0-0.dll',
|
||||||
|
'<(module_root_dir)/lib/libpangocairo-1.0-0.dll',
|
||||||
|
'<(module_root_dir)/lib/libpangowin32-1.0-0.dll',
|
||||||
|
'<(module_root_dir)/lib/libpixman-1-0.dll',
|
||||||
|
'<(module_root_dir)/lib/libpng16-16.dll',
|
||||||
|
'<(module_root_dir)/lib/libquadmath-0.dll',
|
||||||
|
'<(module_root_dir)/lib/libsqlite3-0.dll',
|
||||||
|
'<(module_root_dir)/lib/libssp-0.dll',
|
||||||
|
'<(module_root_dir)/lib/libtiff-5.dll',
|
||||||
|
'<(module_root_dir)/lib/libvips-42.dll',
|
||||||
|
'<(module_root_dir)/lib/libxml2-2.dll',
|
||||||
|
'<(module_root_dir)/lib/zlib1.dll'
|
||||||
|
]
|
||||||
|
}]
|
||||||
|
}]
|
||||||
|
]
|
||||||
}]
|
}]
|
||||||
}
|
}
|
||||||
|
|||||||
134
binding.js
Normal file
@@ -0,0 +1,134 @@
|
|||||||
|
'use strict';
|
||||||
|
|
||||||
|
var fs = require('fs');
|
||||||
|
var path = require('path');
|
||||||
|
var zlib = require('zlib');
|
||||||
|
|
||||||
|
var semver = require('semver');
|
||||||
|
var request = require('request');
|
||||||
|
var tar = require('tar');
|
||||||
|
|
||||||
|
var tmp = require('os').tmpdir();
|
||||||
|
|
||||||
|
var distBaseUrl = 'https://dl.bintray.com/lovell/sharp/';
|
||||||
|
|
||||||
|
// Use NPM-provided environment variable where available, falling back to require-based method for Electron
|
||||||
|
var minimumLibvipsVersion = process.env.npm_package_config_libvips || require('./package.json').config.libvips;
|
||||||
|
|
||||||
|
var vipsHeaderPath = path.join(__dirname, 'include', 'vips', 'vips.h');
|
||||||
|
|
||||||
|
// -- Helpers
|
||||||
|
|
||||||
|
// Does this file exist?
|
||||||
|
var isFile = function(file) {
|
||||||
|
var exists = false;
|
||||||
|
try {
|
||||||
|
exists = fs.statSync(file).isFile();
|
||||||
|
} catch (err) {}
|
||||||
|
return exists;
|
||||||
|
};
|
||||||
|
|
||||||
|
var unpack = function(tarPath, done) {
|
||||||
|
var extractor = tar.Extract({
|
||||||
|
path: __dirname
|
||||||
|
});
|
||||||
|
extractor.on('error', error);
|
||||||
|
extractor.on('end', function() {
|
||||||
|
if (!isFile(vipsHeaderPath)) {
|
||||||
|
error('Could not unpack ' + tarPath);
|
||||||
|
}
|
||||||
|
if (typeof done === 'function') {
|
||||||
|
done();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
fs.createReadStream(tarPath).on('error', error)
|
||||||
|
.pipe(zlib.Unzip())
|
||||||
|
.pipe(extractor);
|
||||||
|
};
|
||||||
|
|
||||||
|
// Error
|
||||||
|
var error = function(msg) {
|
||||||
|
if (msg instanceof Error) {
|
||||||
|
msg = msg.message;
|
||||||
|
}
|
||||||
|
process.stderr.write('ERROR: ' + msg + '\n');
|
||||||
|
process.exit(1);
|
||||||
|
};
|
||||||
|
|
||||||
|
// -- Binary downloaders
|
||||||
|
|
||||||
|
module.exports.download_vips = function() {
|
||||||
|
// Has vips been installed locally?
|
||||||
|
if (!isFile(vipsHeaderPath)) {
|
||||||
|
// Ensure Intel 64-bit or ARM
|
||||||
|
if (process.arch === 'ia32') {
|
||||||
|
error('Intel Architecture 32-bit systems require manual installation - please see http://sharp.dimens.io/en/stable/install/');
|
||||||
|
}
|
||||||
|
// Ensure glibc >= 2.15
|
||||||
|
var lddVersion = process.env.LDD_VERSION;
|
||||||
|
if (lddVersion) {
|
||||||
|
if (/(glibc|gnu libc)/i.test(lddVersion)) {
|
||||||
|
var glibcVersion = lddVersion ? lddVersion.split(/\n/)[0].split(' ').slice(-1)[0].trim() : '';
|
||||||
|
if (glibcVersion && semver.lt(glibcVersion + '.0', '2.13.0')) {
|
||||||
|
error('glibc version ' + glibcVersion + ' requires manual installation - please see http://sharp.dimens.io/en/stable/install/');
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
error(lddVersion.split(/\n/)[0] + ' requires manual installation - please see http://sharp.dimens.io/en/stable/install/');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// Arch/platform-specific .tar.gz
|
||||||
|
var platform = (process.arch === 'arm') ? 'arm' : process.platform.substr(0, 3);
|
||||||
|
var tarFilename = ['libvips', minimumLibvipsVersion, platform].join('-') + '.tar.gz';
|
||||||
|
var tarPath = path.join(__dirname, 'packaging', tarFilename);
|
||||||
|
if (isFile(tarPath)) {
|
||||||
|
unpack(tarPath);
|
||||||
|
} else {
|
||||||
|
// Download to per-process temporary file
|
||||||
|
tarPath = path.join(tmp, process.pid + '-' + tarFilename);
|
||||||
|
var tmpFile = fs.createWriteStream(tarPath).on('finish', function() {
|
||||||
|
unpack(tarPath, function() {
|
||||||
|
// Attempt to remove temporary file
|
||||||
|
try {
|
||||||
|
fs.unlinkSync(tarPath);
|
||||||
|
} catch (err) {}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
var options = {
|
||||||
|
url: distBaseUrl + tarFilename
|
||||||
|
};
|
||||||
|
if (process.env.npm_config_https_proxy) {
|
||||||
|
// Use the NPM-configured HTTPS proxy
|
||||||
|
options.proxy = process.env.npm_config_https_proxy;
|
||||||
|
}
|
||||||
|
request(options).on('response', function(response) {
|
||||||
|
if (response.statusCode !== 200) {
|
||||||
|
error(distBaseUrl + tarFilename + ' status code ' + response.statusCode);
|
||||||
|
}
|
||||||
|
}).on('error', function(err) {
|
||||||
|
error('Download from ' + distBaseUrl + tarFilename + ' failed: ' + err.message);
|
||||||
|
}).pipe(tmpFile);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
module.exports.use_global_vips = function() {
|
||||||
|
var useGlobalVips = false;
|
||||||
|
var globalVipsVersion = process.env.GLOBAL_VIPS_VERSION;
|
||||||
|
if (globalVipsVersion) {
|
||||||
|
useGlobalVips = semver.gte(
|
||||||
|
globalVipsVersion,
|
||||||
|
minimumLibvipsVersion
|
||||||
|
);
|
||||||
|
}
|
||||||
|
if (process.platform === 'darwin' && !useGlobalVips) {
|
||||||
|
if (globalVipsVersion) {
|
||||||
|
error(
|
||||||
|
'Found libvips ' + globalVipsVersion + ' but require ' + minimumLibvipsVersion +
|
||||||
|
'\nPlease upgrade libvips by running: brew update && brew upgrade'
|
||||||
|
);
|
||||||
|
} else {
|
||||||
|
error('Please install libvips by running: brew install homebrew/science/vips --with-webp --with-graphicsmagick');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
process.stdout.write(useGlobalVips ? 'true' : 'false');
|
||||||
|
};
|
||||||
6
circle.yml
Normal file
@@ -0,0 +1,6 @@
|
|||||||
|
machine:
|
||||||
|
services:
|
||||||
|
- docker
|
||||||
|
test:
|
||||||
|
override:
|
||||||
|
- ./packaging/test.sh
|
||||||
266
docs/api.md
@@ -6,22 +6,28 @@ var sharp = require('sharp');
|
|||||||
|
|
||||||
### Input
|
### Input
|
||||||
|
|
||||||
#### sharp([input])
|
#### sharp([input], [options])
|
||||||
|
|
||||||
Constructor to which further methods are chained. `input`, if present, can be one of:
|
Constructor to which further methods are chained.
|
||||||
|
|
||||||
* Buffer containing JPEG, PNG, WebP, GIF* or TIFF image data, or
|
`input`, if present, can be one of:
|
||||||
* String containing the filename of an image, with most major formats supported.
|
|
||||||
|
|
||||||
The object returned implements the
|
* Buffer containing JPEG, PNG, WebP, GIF, SVG, TIFF or raw pixel image data, or
|
||||||
|
* String containing the path to an image file, with most major formats supported.
|
||||||
|
|
||||||
|
JPEG, PNG, WebP, GIF, SVG, TIFF or raw pixel image data
|
||||||
|
can be streamed into the object when `input` is `null` or `undefined`.
|
||||||
|
|
||||||
|
`options`, if present, is an Object with the following optional attributes:
|
||||||
|
|
||||||
|
* `density` an integral number representing the DPI for vector images, defaulting to 72.
|
||||||
|
* `raw` an Object containing `width`, `height` and `channels` when providing uncompressed data. See `raw()` for pixel ordering.
|
||||||
|
|
||||||
|
The object returned by the constructor implements the
|
||||||
[stream.Duplex](http://nodejs.org/api/stream.html#stream_class_stream_duplex) class.
|
[stream.Duplex](http://nodejs.org/api/stream.html#stream_class_stream_duplex) class.
|
||||||
|
|
||||||
JPEG, PNG, WebP, GIF* or TIFF format image data
|
|
||||||
can be streamed into the object when `input` is not provided.
|
|
||||||
|
|
||||||
JPEG, PNG or WebP format image data can be streamed out from this object.
|
JPEG, PNG or WebP format image data can be streamed out from this object.
|
||||||
|
When using Stream based output, derived attributes are available from the `info` event.
|
||||||
\* libvips 8.0.0+ is required for Buffer/Stream input of GIF and other `magick` formats.
|
|
||||||
|
|
||||||
```javascript
|
```javascript
|
||||||
sharp('input.jpg')
|
sharp('input.jpg')
|
||||||
@@ -32,17 +38,31 @@ sharp('input.jpg')
|
|||||||
});
|
});
|
||||||
```
|
```
|
||||||
|
|
||||||
|
```javascript
|
||||||
|
// Read image data from readableStream,
|
||||||
|
// resize to 300 pixels wide,
|
||||||
|
// emit an 'info' event with calculated dimensions
|
||||||
|
// and finally write image data to writableStream
|
||||||
|
var transformer = sharp()
|
||||||
|
.resize(300)
|
||||||
|
.on('info', function(info) {
|
||||||
|
console.log('Image height is ' + info.height);
|
||||||
|
});
|
||||||
|
readableStream.pipe(transformer).pipe(writableStream);
|
||||||
|
```
|
||||||
|
|
||||||
#### metadata([callback])
|
#### metadata([callback])
|
||||||
|
|
||||||
Fast access to image metadata without decoding any compressed image data.
|
Fast access to image metadata without decoding any compressed image data.
|
||||||
|
|
||||||
`callback`, if present, gets the arguments `(err, metadata)` where `metadata` has the attributes:
|
`callback`, if present, gets the arguments `(err, metadata)` where `metadata` has the attributes:
|
||||||
|
|
||||||
* `format`: Name of decoder to be used to decompress image data e.g. `jpeg`, `png`, `webp` (for file-based input additionally `tiff`, `magick` and `openslide`)
|
* `format`: Name of decoder to be used to decompress image data e.g. `jpeg`, `png`, `webp` (for file-based input additionally `tiff`, `magick`, `openslide`, `ppm`, `fits`)
|
||||||
* `width`: Number of pixels wide
|
* `width`: Number of pixels wide
|
||||||
* `height`: Number of pixels high
|
* `height`: Number of pixels high
|
||||||
* `space`: Name of colour space interpretation e.g. `srgb`, `rgb`, `scrgb`, `cmyk`, `lab`, `xyz`, `b-w` [...](https://github.com/jcupitt/libvips/blob/master/libvips/iofuncs/enumtypes.c#L522)
|
* `space`: Name of colour space interpretation e.g. `srgb`, `rgb`, `scrgb`, `cmyk`, `lab`, `xyz`, `b-w` [...](https://github.com/jcupitt/libvips/blob/master/libvips/iofuncs/enumtypes.c#L522)
|
||||||
* `channels`: Number of bands e.g. `3` for sRGB, `4` for CMYK
|
* `channels`: Number of bands e.g. `3` for sRGB, `4` for CMYK
|
||||||
|
* `density`: Number of pixels per inch (DPI), if present
|
||||||
* `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
|
||||||
@@ -78,7 +98,7 @@ to share a single input Stream.
|
|||||||
```javascript
|
```javascript
|
||||||
var pipeline = sharp().rotate();
|
var pipeline = sharp().rotate();
|
||||||
pipeline.clone().resize(800, 600).pipe(firstWritableStream);
|
pipeline.clone().resize(800, 600).pipe(firstWritableStream);
|
||||||
pipeline.clone().extract(20, 20, 100, 100).pipe(secondWritableStream);
|
pipeline.clone().extract({ left: 20, top: 20, width: 100, height: 100 }).pipe(secondWritableStream);
|
||||||
readableStream.pipe(pipeline);
|
readableStream.pipe(pipeline);
|
||||||
// firstWritableStream receives auto-rotated, resized readableStream
|
// firstWritableStream receives auto-rotated, resized readableStream
|
||||||
// secondWritableStream receives auto-rotated, extracted region of readableStream
|
// secondWritableStream receives auto-rotated, extracted region of readableStream
|
||||||
@@ -93,7 +113,8 @@ This will reduce memory usage and can improve performance on some systems.
|
|||||||
|
|
||||||
Do not process input images where the number of pixels (width * height) exceeds this limit.
|
Do not process input images where the number of pixels (width * height) exceeds this limit.
|
||||||
|
|
||||||
`pixels` is the integral Number of pixels, with a value between 1 and the default 268402689 (0x3FFF * 0x3FFF).
|
`pixels` is either an integral Number of pixels, with a value between 1 and the default 268402689 (0x3FFF * 0x3FFF) or
|
||||||
|
a boolean. `false` will disable checking while `true` will revert to the default limit.
|
||||||
|
|
||||||
### Resizing
|
### Resizing
|
||||||
|
|
||||||
@@ -105,22 +126,37 @@ Scale output to `width` x `height`. By default, the resized image is cropped to
|
|||||||
|
|
||||||
`height` is the integral Number of pixels high the resultant image should be, between 1 and 16383. Use `null` or `undefined` to auto-scale the height to match the width.
|
`height` is the integral Number of pixels high the resultant image should be, between 1 and 16383. Use `null` or `undefined` to auto-scale the height to match the width.
|
||||||
|
|
||||||
#### crop([gravity])
|
#### crop([option])
|
||||||
|
|
||||||
Crop the resized image to the exact size specified, the default behaviour.
|
Crop the resized image to the exact size specified, the default behaviour.
|
||||||
|
|
||||||
`gravity`, if present, is an attribute of the `sharp.gravity` Object e.g. `sharp.gravity.north`.
|
`option`, if present, is an attribute of:
|
||||||
|
|
||||||
Possible values are `north`, `east`, `south`, `west`, `center` and `centre`. The default gravity is `center`/`centre`.
|
* `sharp.gravity` e.g. `sharp.gravity.north`, to crop to an edge or corner, or
|
||||||
|
* `sharp.strategy` e.g. `sharp.strategy.entropy`, to crop dynamically.
|
||||||
|
|
||||||
|
Possible attributes of `sharp.gravity` are
|
||||||
|
`north`, `northeast`, `east`, `southeast`, `south`,
|
||||||
|
`southwest`, `west`, `northwest`, `center` and `centre`.
|
||||||
|
|
||||||
|
Possible attributes of the experimental `sharp.strategy` are:
|
||||||
|
|
||||||
|
* `entropy`: resize so one dimension is at its target size
|
||||||
|
then repeatedly remove pixels from the edge with the lowest
|
||||||
|
[Shannon entropy](https://en.wikipedia.org/wiki/Entropy_%28information_theory%29)
|
||||||
|
until it too reaches the target size.
|
||||||
|
|
||||||
|
The default crop option is a `center`/`centre` gravity.
|
||||||
|
|
||||||
```javascript
|
```javascript
|
||||||
var transformer = sharp()
|
var transformer = sharp()
|
||||||
.resize(300, 200)
|
.resize(200, 200)
|
||||||
.crop(sharp.gravity.north)
|
.crop(sharp.strategy.entropy)
|
||||||
.on('error', function(err) {
|
.on('error', function(err) {
|
||||||
console.log(err);
|
console.log(err);
|
||||||
});
|
});
|
||||||
// Read image data from readableStream, resize and write image data to writableStream
|
// Read image data from readableStream
|
||||||
|
// Write 200px square auto-cropped image data to writableStream
|
||||||
readableStream.pipe(transformer).pipe(writableStream);
|
readableStream.pipe(transformer).pipe(writableStream);
|
||||||
```
|
```
|
||||||
|
|
||||||
@@ -200,14 +236,18 @@ Ignoring the aspect ratio of the input, stretch the image to the exact `width` a
|
|||||||
|
|
||||||
Use the given interpolator for image resizing, where `interpolator` is an attribute of the `sharp.interpolator` Object e.g. `sharp.interpolator.bicubic`.
|
Use the given interpolator for image resizing, where `interpolator` is an attribute of the `sharp.interpolator` Object e.g. `sharp.interpolator.bicubic`.
|
||||||
|
|
||||||
|
The default interpolator is `bicubic`, providing a general-purpose interpolator that is both fast and of good quality.
|
||||||
|
|
||||||
Possible interpolators, in order of performance, are:
|
Possible interpolators, in order of performance, are:
|
||||||
|
|
||||||
* `nearest`: Use [nearest neighbour interpolation](http://en.wikipedia.org/wiki/Nearest-neighbor_interpolation), suitable for image enlargement only.
|
* `nearest`: Use [nearest neighbour interpolation](http://en.wikipedia.org/wiki/Nearest-neighbor_interpolation), suitable for image enlargement only.
|
||||||
* `bilinear`: Use [bilinear interpolation](http://en.wikipedia.org/wiki/Bilinear_interpolation), the default and fastest image reduction interpolation.
|
* `bilinear`: Use [bilinear interpolation](http://en.wikipedia.org/wiki/Bilinear_interpolation), faster than bicubic but with less smooth results.
|
||||||
* `bicubic`: Use [bicubic interpolation](http://en.wikipedia.org/wiki/Bicubic_interpolation), which typically reduces performance by 5%.
|
* `vertexSplitQuadraticBasisSpline`: Use the smoother [VSQBS interpolation](https://github.com/jcupitt/libvips/blob/master/libvips/resample/vsqbs.cpp#L48) to prevent "staircasing" when enlarging.
|
||||||
* `vertexSplitQuadraticBasisSpline`: Use [VSQBS interpolation](https://github.com/jcupitt/libvips/blob/master/libvips/resample/vsqbs.cpp#L48), which prevents "staircasing" and typically reduces performance by 5%.
|
* `bicubic`: Use [bicubic interpolation](http://en.wikipedia.org/wiki/Bicubic_interpolation) (the default).
|
||||||
* `locallyBoundedBicubic`: Use [LBB interpolation](https://github.com/jcupitt/libvips/blob/master/libvips/resample/lbb.cpp#L100), which prevents some "[acutance](http://en.wikipedia.org/wiki/Acutance)" and typically reduces performance by a factor of 2.
|
* `locallyBoundedBicubic`: Use [LBB interpolation](https://github.com/jcupitt/libvips/blob/master/libvips/resample/lbb.cpp#L100), which prevents some "[acutance](http://en.wikipedia.org/wiki/Acutance)" but typically reduces performance by a factor of 2.
|
||||||
* `nohalo`: Use [Nohalo interpolation](http://eprints.soton.ac.uk/268086/), which prevents acutance and typically reduces performance by a factor of 3.
|
* `nohalo`: Use [Nohalo interpolation](http://eprints.soton.ac.uk/268086/), which prevents acutance but typically reduces performance by a factor of 3.
|
||||||
|
|
||||||
|
[Compare the output of these interpolators](https://github.com/lovell/sharp/tree/master/test/interpolators)
|
||||||
|
|
||||||
```javascript
|
```javascript
|
||||||
sharp(inputBuffer)
|
sharp(inputBuffer)
|
||||||
@@ -225,11 +265,11 @@ sharp(inputBuffer)
|
|||||||
|
|
||||||
### Operations
|
### Operations
|
||||||
|
|
||||||
#### extract(top, left, width, height)
|
#### extract({ left: left, top: top, width: width, height: height })
|
||||||
|
|
||||||
Extract a region of the image. Can be used with or without a `resize` operation.
|
Extract a region of the image. Can be used with or without a `resize` operation.
|
||||||
|
|
||||||
`top` and `left` are the offset, in pixels, from the top-left corner.
|
`left` and `top` are the offset, in pixels, from the top-left corner.
|
||||||
|
|
||||||
`width` and `height` are the dimensions of the extracted image.
|
`width` and `height` are the dimensions of the extracted image.
|
||||||
|
|
||||||
@@ -237,7 +277,7 @@ Use `extract` before `resize` for pre-resize extraction. Use `extract` after `re
|
|||||||
|
|
||||||
```javascript
|
```javascript
|
||||||
sharp(input)
|
sharp(input)
|
||||||
.extract(top, left, width, height)
|
.extract({ left: left, top: top, width: width, height: height })
|
||||||
.toFile(output, function(err) {
|
.toFile(output, function(err) {
|
||||||
// Extract a region of the input image, saving in the same format.
|
// Extract a region of the input image, saving in the same format.
|
||||||
});
|
});
|
||||||
@@ -245,9 +285,9 @@ sharp(input)
|
|||||||
|
|
||||||
```javascript
|
```javascript
|
||||||
sharp(input)
|
sharp(input)
|
||||||
.extract(topOffsetPre, leftOffsetPre, widthPre, heightPre)
|
.extract({ left: leftOffsetPre, top: topOffsetPre, width: widthPre, height: heightPre })
|
||||||
.resize(width, height)
|
.resize(width, height)
|
||||||
.extract(topOffsetPost, leftOffsetPost, widthPost, heightPost)
|
.extract({ left: leftOffsetPost, top: topOffsetPost, width: widthPost, height: heightPost })
|
||||||
.toFile(output, function(err) {
|
.toFile(output, function(err) {
|
||||||
// Extract a region, resize, then extract from the resized image
|
// Extract a region, resize, then extract from the resized image
|
||||||
});
|
});
|
||||||
@@ -255,7 +295,7 @@ sharp(input)
|
|||||||
|
|
||||||
#### background(rgba)
|
#### background(rgba)
|
||||||
|
|
||||||
Set the background for the `embed` and `flatten` operations.
|
Set the background for the `embed`, `flatten` and `extend` operations.
|
||||||
|
|
||||||
`rgba` is parsed by the [color](https://www.npmjs.org/package/color) module to extract values for red, green, blue and alpha.
|
`rgba` is parsed by the [color](https://www.npmjs.org/package/color) module to extract values for red, green, blue and alpha.
|
||||||
|
|
||||||
@@ -267,6 +307,29 @@ The default background is `{r: 0, g: 0, b: 0, a: 1}`, black without transparency
|
|||||||
|
|
||||||
Merge alpha transparency channel, if any, with `background`.
|
Merge alpha transparency channel, if any, with `background`.
|
||||||
|
|
||||||
|
#### extend(extension)
|
||||||
|
|
||||||
|
Extends/pads the edges of the image with `background`, where `extension` is one of:
|
||||||
|
|
||||||
|
* a Number representing the pixel count to add to each edge, or
|
||||||
|
* an Object containing `top`, `left`, `bottom` and `right` attributes, each a Number of pixels to add to that edge.
|
||||||
|
|
||||||
|
This operation will always occur after resizing and extraction, if any.
|
||||||
|
|
||||||
|
```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, a: 0})
|
||||||
|
.extend({top: 10, bottom: 20, left: 10, right: 10})
|
||||||
|
...
|
||||||
|
```
|
||||||
|
|
||||||
|
#### negate()
|
||||||
|
|
||||||
|
Produces the "negative" of the image. White => Black, Black => White, Blue => Yellow, etc.
|
||||||
|
|
||||||
#### rotate([angle])
|
#### rotate([angle])
|
||||||
|
|
||||||
Rotate the output image by either an explicit angle or auto-orient based on the EXIF `Orientation` tag.
|
Rotate the output image by either an explicit angle or auto-orient based on the EXIF `Orientation` tag.
|
||||||
@@ -323,11 +386,17 @@ When a `radius` is provided, performs a slower, more accurate sharpen of the L c
|
|||||||
* `flat`, if present, is a Number representing the level of sharpening to apply to "flat" areas, defaulting to a value of 1.0.
|
* `flat`, if present, is a Number representing the level of sharpening to apply to "flat" areas, defaulting to a value of 1.0.
|
||||||
* `jagged`, if present, is a Number representing the level of sharpening to apply to "jagged" areas, defaulting to a value of 2.0.
|
* `jagged`, if present, is a Number representing the level of sharpening to apply to "jagged" areas, defaulting to a value of 2.0.
|
||||||
|
|
||||||
|
#### threshold([threshold])
|
||||||
|
|
||||||
|
Converts all pixels in the image to greyscale white or black. Any pixel greather-than-or-equal-to the threshold (0..255) will be white. All others will be black.
|
||||||
|
|
||||||
|
* `threshold`, if present, is a Number, representing the level above which pixels will be forced to white.
|
||||||
|
|
||||||
#### gamma([gamma])
|
#### gamma([gamma])
|
||||||
|
|
||||||
Apply a gamma correction by reducing the encoding (darken) pre-resize at a factor of `1/gamma` then increasing the encoding (brighten) post-resize at a factor of `gamma`.
|
Apply a gamma correction by reducing the encoding (darken) pre-resize at a factor of `1/gamma` then increasing the encoding (brighten) post-resize at a factor of `gamma`.
|
||||||
|
|
||||||
`gamma`, if present, is a Number betweem 1 and 3. The default value is `2.2`, a suitable approximation for sRGB images.
|
`gamma`, if present, is a Number between 1 and 3. The default value is `2.2`, a suitable approximation for sRGB images.
|
||||||
|
|
||||||
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.
|
||||||
|
|
||||||
@@ -345,13 +414,18 @@ The output image will still be web-friendly sRGB and contain three (identical) c
|
|||||||
|
|
||||||
Enhance output image contrast by stretching its luminance to cover the full dynamic range. This typically reduces performance by 30%.
|
Enhance output image contrast by stretching its luminance to cover the full dynamic range. This typically reduces performance by 30%.
|
||||||
|
|
||||||
#### overlayWith(filename)
|
#### overlayWith(image, [options])
|
||||||
|
|
||||||
_Experimental_
|
Overlay (composite) a image containing an alpha channel over the processed (resized, extracted etc.) image.
|
||||||
|
|
||||||
Alpha composite `filename` over the processed (resized, extracted) image. The dimensions of the two images must match.
|
`image` is one of the following, and must be the same size or smaller than the processed image:
|
||||||
|
|
||||||
* `filename` is a String containing the filename of an image with an alpha channel.
|
* Buffer containing PNG, WebP, GIF or SVG image data, or
|
||||||
|
* String containing the path to an image file, with most major transparency formats supported.
|
||||||
|
|
||||||
|
`options`, if present, is an Object with the following optional attributes:
|
||||||
|
|
||||||
|
* `gravity` is a String or an attribute of the `sharp.gravity` Object e.g. `sharp.gravity.north` at which to place the overlay, defaulting to `center`/`centre`.
|
||||||
|
|
||||||
```javascript
|
```javascript
|
||||||
sharp('input.png')
|
sharp('input.png')
|
||||||
@@ -359,7 +433,7 @@ sharp('input.png')
|
|||||||
.resize(300)
|
.resize(300)
|
||||||
.flatten()
|
.flatten()
|
||||||
.background('#ff6600')
|
.background('#ff6600')
|
||||||
.overlayWith('overlay.png')
|
.overlayWith('overlay.png', { gravity: sharp.gravity.southeast } )
|
||||||
.sharpen()
|
.sharpen()
|
||||||
.withMetadata()
|
.withMetadata()
|
||||||
.quality(90)
|
.quality(90)
|
||||||
@@ -367,21 +441,23 @@ sharp('input.png')
|
|||||||
.toBuffer()
|
.toBuffer()
|
||||||
.then(function(outputBuffer) {
|
.then(function(outputBuffer) {
|
||||||
// outputBuffer contains upside down, 300px wide, alpha channel flattened
|
// outputBuffer contains upside down, 300px wide, alpha channel flattened
|
||||||
// onto orange background, composited with overlay.png, sharpened,
|
// onto orange background, composited with overlay.png with SE gravity,
|
||||||
// with metadata, 90% quality WebP image data. Phew!
|
// sharpened, with metadata, 90% quality WebP image data. Phew!
|
||||||
});
|
});
|
||||||
```
|
```
|
||||||
|
|
||||||
### Output
|
### Output
|
||||||
|
|
||||||
#### toFile(filename, [callback])
|
#### toFile(path, [callback])
|
||||||
|
|
||||||
`filename` is a String containing the filename to write the image data to. The format is inferred from the extension, with JPEG, PNG, WebP, TIFF and DZI supported.
|
`path` is a String containing the path to write the image data to.
|
||||||
|
|
||||||
|
If an explicit output format is not selected, it will be inferred from the extension, with JPEG, PNG, WebP, TIFF and DZI supported.
|
||||||
|
|
||||||
`callback`, if present, is called with two arguments `(err, info)` where:
|
`callback`, if present, is called with two arguments `(err, info)` where:
|
||||||
|
|
||||||
* `err` contains an error message, if any.
|
* `err` contains an error message, if any.
|
||||||
* `info` contains the output image `format`, `size` (bytes), `width` and `height`.
|
* `info` contains the output image `format`, `size` (bytes), `width`, `height` and `channels`.
|
||||||
|
|
||||||
A Promises/A+ promise is returned when `callback` is not provided.
|
A Promises/A+ promise is returned when `callback` is not provided.
|
||||||
|
|
||||||
@@ -393,7 +469,7 @@ Write image data to a Buffer, the format of which will match the input image by
|
|||||||
|
|
||||||
* `err` is an error message, if any.
|
* `err` is an error message, if any.
|
||||||
* `buffer` is the output image data.
|
* `buffer` is the output image data.
|
||||||
* `info` contains the output image `format`, `size` (bytes), `width` and `height`.
|
* `info` contains the output image `format`, `size` (bytes), `width`, `height` and `channels`.
|
||||||
|
|
||||||
A Promises/A+ promise is returned when `callback` is not provided.
|
A Promises/A+ promise is returned when `callback` is not provided.
|
||||||
|
|
||||||
@@ -411,8 +487,6 @@ Use WebP format for the output image.
|
|||||||
|
|
||||||
#### raw()
|
#### raw()
|
||||||
|
|
||||||
_Requires libvips 7.42.0+_
|
|
||||||
|
|
||||||
Provide raw, uncompressed uint8 (unsigned char) image data for Buffer and Stream based output.
|
Provide raw, uncompressed uint8 (unsigned char) image data for Buffer and Stream based output.
|
||||||
|
|
||||||
The number of channels depends on the input image and selected options.
|
The number of channels depends on the input image and selected options.
|
||||||
@@ -446,23 +520,33 @@ This will also convert to and add the latest web-friendly v2 sRGB ICC profile.
|
|||||||
The optional `metadata` parameter, if present, is an Object with the attributes to update.
|
The optional `metadata` parameter, if present, is an Object with the attributes to update.
|
||||||
New attributes cannot be inserted, only existing attributes updated.
|
New attributes cannot be inserted, only existing attributes updated.
|
||||||
|
|
||||||
* `orientation` is an integral Number between 0 and 7, used to update the value of the EXIF `Orientation` tag.
|
* `orientation` is an integral Number between 1 and 8, used to update the value of the EXIF `Orientation` tag.
|
||||||
This has no effect if the input image does not have an EXIF `Orientation` tag.
|
This has no effect if the input image does not have an EXIF `Orientation` tag.
|
||||||
|
|
||||||
The default behaviour, when `withMetadata` is not used, is to strip all metadata and convert to the device-independent sRGB colour space.
|
The default behaviour, when `withMetadata` is not used, is to strip all metadata and convert to the device-independent sRGB colour space.
|
||||||
|
|
||||||
#### tile([size], [overlap])
|
#### tile(options)
|
||||||
|
|
||||||
The size and overlap, in pixels, of square Deep Zoom image pyramid tiles.
|
The size, overlap, container and directory layout to use when generating square Deep Zoom image pyramid tiles.
|
||||||
|
|
||||||
|
`options` is an Object with one or more of the following attributes:
|
||||||
|
|
||||||
* `size` is an integral Number between 1 and 8192. The default value is 256 pixels.
|
* `size` is an integral Number between 1 and 8192. The default value is 256 pixels.
|
||||||
* `overlap` is an integral Number between 0 and 8192. The default value is 0 pixels.
|
* `overlap` is an integral Number between 0 and 8192. The default value is 0 pixels.
|
||||||
|
* `container` is a String, with value `fs` or `zip`. The default value is `fs`.
|
||||||
|
* `layout` is a String, with value `dz`, `zoomify` or `google`. The default value is `dz`.
|
||||||
|
|
||||||
|
You can also use the file extension .zip or .szi to write to a ZIP container instead of the filesystem.
|
||||||
|
|
||||||
```javascript
|
```javascript
|
||||||
sharp('input.tiff').tile(256).toFile('output.dzi', function(err, info) {
|
sharp('input.tiff')
|
||||||
// The output.dzi file is the XML format Deep Zoom definition
|
.tile({
|
||||||
// The output_files directory contains 256x256 pixel tiles grouped by zoom level
|
size: 512
|
||||||
});
|
})
|
||||||
|
.toFile('output.dzi', function(err, info) {
|
||||||
|
// output.dzi is the Deep Zoom XML definition
|
||||||
|
// output_files contains 512x512 tiles grouped by zoom level
|
||||||
|
});
|
||||||
```
|
```
|
||||||
|
|
||||||
#### withoutChromaSubsampling()
|
#### withoutChromaSubsampling()
|
||||||
@@ -482,13 +566,11 @@ An advanced setting for the _zlib_ compression level of the lossless PNG output
|
|||||||
|
|
||||||
#### withoutAdaptiveFiltering()
|
#### withoutAdaptiveFiltering()
|
||||||
|
|
||||||
_Requires libvips 7.42.0+_
|
|
||||||
|
|
||||||
An advanced setting to disable adaptive row filtering for the lossless PNG output format.
|
An advanced setting to disable adaptive row filtering for the lossless PNG output format.
|
||||||
|
|
||||||
#### trellisQuantisation() / trellisQuantization()
|
#### trellisQuantisation() / trellisQuantization()
|
||||||
|
|
||||||
_Requires libvips 8.0.0+ compiled against mozjpeg 3.0+_
|
_Requires libvips to have been compiled with mozjpeg support_
|
||||||
|
|
||||||
An advanced setting to apply the use of
|
An advanced setting to apply the use of
|
||||||
[trellis quantisation](http://en.wikipedia.org/wiki/Trellis_quantization) with JPEG output.
|
[trellis quantisation](http://en.wikipedia.org/wiki/Trellis_quantization) with JPEG output.
|
||||||
@@ -496,7 +578,7 @@ Reduces file size and slightly increases relative quality at the cost of increas
|
|||||||
|
|
||||||
#### overshootDeringing()
|
#### overshootDeringing()
|
||||||
|
|
||||||
_Requires libvips 8.0.0+ compiled against mozjpeg 3.0+_
|
_Requires libvips to have been compiled with mozjpeg support_
|
||||||
|
|
||||||
An advanced setting to reduce the effects of
|
An advanced setting to reduce the effects of
|
||||||
[ringing](http://en.wikipedia.org/wiki/Ringing_%28signal%29) in JPEG output,
|
[ringing](http://en.wikipedia.org/wiki/Ringing_%28signal%29) in JPEG output,
|
||||||
@@ -504,7 +586,7 @@ in particular where black text appears on a white background (or vice versa).
|
|||||||
|
|
||||||
#### optimiseScans() / optimizeScans()
|
#### optimiseScans() / optimizeScans()
|
||||||
|
|
||||||
_Requires libvips 8.0.0+ compiled against mozjpeg 3.0+_
|
_Requires libvips to have been compiled with mozjpeg support_
|
||||||
|
|
||||||
An advanced setting for progressive (interlace) JPEG output.
|
An advanced setting for progressive (interlace) JPEG output.
|
||||||
Calculates which spectrum of DCT coefficients uses the fewest bits.
|
Calculates which spectrum of DCT coefficients uses the fewest bits.
|
||||||
@@ -554,21 +636,51 @@ sharp.queue.on('change', function(queueLength) {
|
|||||||
});
|
});
|
||||||
```
|
```
|
||||||
|
|
||||||
|
#### versions
|
||||||
|
|
||||||
|
An Object containing the version numbers of libvips and, on Linux, its dependencies.
|
||||||
|
|
||||||
|
```javascript
|
||||||
|
> console.log(sharp.versions);
|
||||||
|
|
||||||
|
{ zlib: '1.2.8',
|
||||||
|
ffi: '3.2.1',
|
||||||
|
glib: '2.46.2',
|
||||||
|
xml: '2.9.2',
|
||||||
|
gsf: '1.14.34',
|
||||||
|
exif: '0.6.21',
|
||||||
|
jpeg: '1.4.2',
|
||||||
|
png: '1.6.19',
|
||||||
|
lcms: '2.7',
|
||||||
|
webp: '0.4.4',
|
||||||
|
tiff: '4.0.6',
|
||||||
|
magick: '6.9.2-6',
|
||||||
|
orc: '0.4.24',
|
||||||
|
vips: '8.1.1' }
|
||||||
|
```
|
||||||
|
|
||||||
### Utilities
|
### Utilities
|
||||||
|
|
||||||
#### sharp.cache([memory], [items])
|
#### sharp.cache([options])
|
||||||
|
|
||||||
If `memory` or `items` are provided, set the limits of _libvips'_ operation cache.
|
If `options` is provided, sets the limits of _libvips'_ operation cache.
|
||||||
|
|
||||||
* `memory` is the maximum memory in MB to use for this cache, with a default value of 100
|
* `options.memory` is the maximum memory in MB to use for this cache, with a default value of 50
|
||||||
* `items` is the maximum number of operations to cache, with a default value of 500
|
* `options.files` is the maximum number of files to hold open, with a default value of 20
|
||||||
|
* `options.items` is the maximum number of operations to cache, with a default value of 100
|
||||||
|
|
||||||
|
`options` can also be a boolean, where `true` enables the default cache settings and `false` disables all caching.
|
||||||
|
|
||||||
This method always returns cache statistics, useful for determining how much working memory is required for a particular task.
|
This method always returns cache statistics, useful for determining how much working memory is required for a particular task.
|
||||||
|
|
||||||
```javascript
|
```javascript
|
||||||
var stats = sharp.cache(); // { current: 75, high: 99, memory: 100, items: 500 }
|
var stats = sharp.cache();
|
||||||
sharp.cache(200); // { current: 75, high: 99, memory: 200, items: 500 }
|
```
|
||||||
sharp.cache(50, 200); // { current: 49, high: 99, memory: 50, items: 200}
|
|
||||||
|
```javascript
|
||||||
|
sharp.cache( { items: 200 } );
|
||||||
|
sharp.cache( { files: 0 } );
|
||||||
|
sharp.cache(false);
|
||||||
```
|
```
|
||||||
|
|
||||||
#### sharp.concurrency([threads])
|
#### sharp.concurrency([threads])
|
||||||
@@ -595,3 +707,31 @@ Provides access to internal task counters.
|
|||||||
```javascript
|
```javascript
|
||||||
var counters = sharp.counters(); // { queue: 2, process: 4 }
|
var counters = sharp.counters(); // { queue: 2, process: 4 }
|
||||||
```
|
```
|
||||||
|
|
||||||
|
#### sharp.simd([enable])
|
||||||
|
|
||||||
|
_Requires libvips to have been compiled with liborc support_
|
||||||
|
|
||||||
|
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.
|
||||||
|
|
||||||
|
* `enable`, if present, is a boolean where `true` enables and `false` disables the use of SIMD.
|
||||||
|
|
||||||
|
This method always returns the current state.
|
||||||
|
|
||||||
|
This feature is currently disabled by default
|
||||||
|
but future versions may enable it by default.
|
||||||
|
|
||||||
|
When enabled, versions of liborc prior to 0.4.24
|
||||||
|
and versions of libvips prior to 8.2.0
|
||||||
|
have been known to crash under heavy load.
|
||||||
|
|
||||||
|
```javascript
|
||||||
|
var simd = sharp.simd();
|
||||||
|
// simd is `true` if SIMD is currently enabled
|
||||||
|
```
|
||||||
|
|
||||||
|
```javascript
|
||||||
|
var simd = sharp.simd(true);
|
||||||
|
// attempts to enable the use of SIMD, returning true if available
|
||||||
|
```
|
||||||
|
|||||||
@@ -1,7 +1,223 @@
|
|||||||
# Changelog
|
# Changelog
|
||||||
|
|
||||||
|
### v0.14 - "*needle*"
|
||||||
|
|
||||||
|
Requires libvips v8.2.3
|
||||||
|
|
||||||
|
#### v0.14.1 - 16<sup>th</sup> April 2016
|
||||||
|
|
||||||
|
* Allow removal of limitation on input pixel count via limitInputPixels. Use with care.
|
||||||
|
[#250](https://github.com/lovell/sharp/issues/250)
|
||||||
|
[#316](https://github.com/lovell/sharp/pull/316)
|
||||||
|
[@anandthakker](https://github.com/anandthakker)
|
||||||
|
[@kentongray](https://github.com/kentongray)
|
||||||
|
|
||||||
|
* Use final output image for metadata passed to callback.
|
||||||
|
[#399](https://github.com/lovell/sharp/pull/399)
|
||||||
|
[@salzhrani](https://github.com/salzhrani)
|
||||||
|
|
||||||
|
* Add support for writing tiled images to a zip container.
|
||||||
|
[#402](https://github.com/lovell/sharp/pull/402)
|
||||||
|
[@felixbuenemann](https://github.com/felixbuenemann)
|
||||||
|
|
||||||
|
* Allow use of embed with 1 and 2 channel images.
|
||||||
|
[#411](https://github.com/lovell/sharp/issues/411)
|
||||||
|
[@janaz](https://github.com/janaz)
|
||||||
|
|
||||||
|
* Improve Electron compatibility by allowing node-gyp rebuilds without npm.
|
||||||
|
[#412](https://github.com/lovell/sharp/issues/412)
|
||||||
|
[@nouh](https://github.com/nouh)
|
||||||
|
|
||||||
|
#### v0.14.0 - 2<sup>nd</sup> April 2016
|
||||||
|
|
||||||
|
* Add ability to extend (pad) the edges of an image.
|
||||||
|
[#128](https://github.com/lovell/sharp/issues/128)
|
||||||
|
[@blowsie](https://github.com/blowsie)
|
||||||
|
|
||||||
|
* Add support for Zoomify and Google tile layouts. Breaks existing tile API.
|
||||||
|
[#223](https://github.com/lovell/sharp/issues/223)
|
||||||
|
[@bdunnette](https://github.com/bdunnette)
|
||||||
|
|
||||||
|
* Improvements to overlayWith: differing sizes/formats, gravity, buffer input.
|
||||||
|
[#239](https://github.com/lovell/sharp/issues/239)
|
||||||
|
[@chrisriley](https://github.com/chrisriley)
|
||||||
|
|
||||||
|
* Add entropy-based crop strategy to remove least interesting edges.
|
||||||
|
[#295](https://github.com/lovell/sharp/issues/295)
|
||||||
|
[@rightaway](https://github.com/rightaway)
|
||||||
|
|
||||||
|
* Expose density metadata; set density of images from vector input.
|
||||||
|
[#338](https://github.com/lovell/sharp/issues/338)
|
||||||
|
[@lookfirst](https://github.com/lookfirst)
|
||||||
|
|
||||||
|
* Emit post-processing 'info' event for Stream output.
|
||||||
|
[#367](https://github.com/lovell/sharp/issues/367)
|
||||||
|
[@salzhrani](https://github.com/salzhrani)
|
||||||
|
|
||||||
|
* Ensure output image EXIF Orientation values are within 1-8 range.
|
||||||
|
[#385](https://github.com/lovell/sharp/pull/385)
|
||||||
|
[@jtobinisaniceguy](https://github.com/jtobinisaniceguy)
|
||||||
|
|
||||||
|
* Ensure ratios are not swapped when rotating 90/270 and ignoring aspect.
|
||||||
|
[#387](https://github.com/lovell/sharp/issues/387)
|
||||||
|
[@kleisauke](https://github.com/kleisauke)
|
||||||
|
|
||||||
|
### v0.13 - "*mind*"
|
||||||
|
|
||||||
|
Requires libvips v8.2.2
|
||||||
|
|
||||||
|
#### v0.13.1 - 27<sup>th</sup> February 2016
|
||||||
|
|
||||||
|
* Fix embedding onto transparent backgrounds; regression introduced in v0.13.0.
|
||||||
|
[#366](https://github.com/lovell/sharp/issues/366)
|
||||||
|
[@diegocsandrim](https://github.com/diegocsandrim)
|
||||||
|
|
||||||
|
#### v0.13.0 - 15<sup>th</sup> February 2016
|
||||||
|
|
||||||
|
* Improve vector image support by allowing control of density/DPI.
|
||||||
|
Switch pre-built libs from Imagemagick to Graphicsmagick.
|
||||||
|
[#110](https://github.com/lovell/sharp/issues/110)
|
||||||
|
[@bradisbell](https://github.com/bradisbell)
|
||||||
|
|
||||||
|
* Add support for raw, uncompressed pixel Buffer/Stream input.
|
||||||
|
[#220](https://github.com/lovell/sharp/issues/220)
|
||||||
|
[@mikemorris](https://github.com/mikemorris)
|
||||||
|
|
||||||
|
* Switch from libvips' C to C++ bindings, requires upgrade to v8.2.2.
|
||||||
|
[#299](https://github.com/lovell/sharp/issues/299)
|
||||||
|
|
||||||
|
* Control number of open files in libvips' cache; breaks existing `cache` behaviour.
|
||||||
|
[#315](https://github.com/lovell/sharp/issues/315)
|
||||||
|
[@impomezia](https://github.com/impomezia)
|
||||||
|
|
||||||
|
* Ensure 16-bit input images can be normalised and embedded onto transparent backgrounds.
|
||||||
|
[#339](https://github.com/lovell/sharp/issues/339)
|
||||||
|
[#340](https://github.com/lovell/sharp/issues/340)
|
||||||
|
[@janaz](https://github.com/janaz)
|
||||||
|
|
||||||
|
* Ensure selected format takes precedence over any unknown output filename extension.
|
||||||
|
[#344](https://github.com/lovell/sharp/issues/344)
|
||||||
|
[@ubaltaci](https://github.com/ubaltaci)
|
||||||
|
|
||||||
|
* Add support for libvips' PBM, PGM, PPM and FITS image format loaders.
|
||||||
|
[#347](https://github.com/lovell/sharp/issues/347)
|
||||||
|
[@oaleynik](https://github.com/oaleynik)
|
||||||
|
|
||||||
|
* Ensure default crop gravity is center/centre.
|
||||||
|
[#351](https://github.com/lovell/sharp/pull/351)
|
||||||
|
[@joelmukuthu](https://github.com/joelmukuthu)
|
||||||
|
|
||||||
|
* Improve support for musl libc systems e.g. Alpine Linux.
|
||||||
|
[#354](https://github.com/lovell/sharp/issues/354)
|
||||||
|
[#359](https://github.com/lovell/sharp/pull/359)
|
||||||
|
[@download13](https://github.com/download13)
|
||||||
|
[@wjordan](https://github.com/wjordan)
|
||||||
|
|
||||||
|
* Small optimisation when reducing by an integral factor to favour shrink over affine.
|
||||||
|
|
||||||
|
* Add support for gamma correction of images with an alpha channel.
|
||||||
|
|
||||||
|
### v0.12 - "*look*"
|
||||||
|
|
||||||
|
Requires libvips v8.2.0
|
||||||
|
|
||||||
|
#### v0.12.2 - 16<sup>th</sup> January 2016
|
||||||
|
|
||||||
|
* Upgrade libvips to v8.2.0 for improved vips_shrink.
|
||||||
|
|
||||||
|
* Add pre-compiled libvips for ARMv6+ CPUs.
|
||||||
|
|
||||||
|
* Ensure 16-bit input images work with embed option.
|
||||||
|
[#325](https://github.com/lovell/sharp/issues/325)
|
||||||
|
[@janaz](https://github.com/janaz)
|
||||||
|
|
||||||
|
* Allow compilation with gmake to provide FreeBSD support.
|
||||||
|
[#326](https://github.com/lovell/sharp/issues/326)
|
||||||
|
[@c0decafe](https://github.com/c0decafe)
|
||||||
|
|
||||||
|
* Attempt to remove temporary file after installation.
|
||||||
|
[#331](https://github.com/lovell/sharp/issues/331)
|
||||||
|
[@dtoubelis](https://github.com/dtoubelis)
|
||||||
|
|
||||||
|
#### v0.12.1 - 12<sup>th</sup> December 2015
|
||||||
|
|
||||||
|
* Allow use of SIMD vector instructions (via liborc) to be toggled on/off.
|
||||||
|
[#172](https://github.com/lovell/sharp/issues/172)
|
||||||
|
[@bkw](https://github.com/bkw)
|
||||||
|
[@puzrin](https://github.com/puzrin)
|
||||||
|
|
||||||
|
* Ensure embedded ICC profiles output with perceptual intent.
|
||||||
|
[#321](https://github.com/lovell/sharp/issues/321)
|
||||||
|
[@vlapo](https://github.com/vlapo)
|
||||||
|
|
||||||
|
* Use the NPM-configured HTTPS proxy, if any, for binary downloads.
|
||||||
|
|
||||||
|
#### v0.12.0 - 23<sup>rd</sup> November 2015
|
||||||
|
|
||||||
|
* Bundle pre-compiled libvips and its dependencies for 64-bit Linux and Windows.
|
||||||
|
[#42](https://github.com/lovell/sharp/issues/42)
|
||||||
|
|
||||||
|
* Take advantage of libvips v8.1.0+ features.
|
||||||
|
[#152](https://github.com/lovell/sharp/issues/152)
|
||||||
|
|
||||||
|
* Add support for 64-bit Windows. Drop support for 32-bit Windows.
|
||||||
|
[#224](https://github.com/lovell/sharp/issues/224)
|
||||||
|
[@sabrehagen](https://github.com/sabrehagen)
|
||||||
|
|
||||||
|
* Switch default interpolator to bicubic.
|
||||||
|
[#289](https://github.com/lovell/sharp/issues/289)
|
||||||
|
[@mahnunchik](https://github.com/mahnunchik)
|
||||||
|
|
||||||
|
* Pre-extract rotatation should not swap width/height.
|
||||||
|
[#296](https://github.com/lovell/sharp/issues/296)
|
||||||
|
[@asilvas](https://github.com/asilvas)
|
||||||
|
|
||||||
|
* Ensure 16-bit+alpha input images are (un)premultiplied correctly.
|
||||||
|
[#301](https://github.com/lovell/sharp/issues/301)
|
||||||
|
[@izaakschroeder](https://github.com/izaakschroeder)
|
||||||
|
|
||||||
|
* Add `threshold` operation.
|
||||||
|
[#303](https://github.com/lovell/sharp/pull/303)
|
||||||
|
[@dacarley](https://github.com/dacarley)
|
||||||
|
|
||||||
|
* Add `negate` operation.
|
||||||
|
[#306](https://github.com/lovell/sharp/pull/306)
|
||||||
|
[@dacarley](https://github.com/dacarley)
|
||||||
|
|
||||||
|
* Support `options` Object with existing `extract` operation.
|
||||||
|
[#309](https://github.com/lovell/sharp/pull/309)
|
||||||
|
[@papandreou](https://github.com/papandreou)
|
||||||
|
|
||||||
### v0.11 - "*knife*"
|
### v0.11 - "*knife*"
|
||||||
|
|
||||||
|
#### v0.11.4 - 5<sup>th</sup> November 2015
|
||||||
|
|
||||||
|
* Add corners, e.g. `northeast`, to existing `gravity` option.
|
||||||
|
[#291](https://github.com/lovell/sharp/pull/291)
|
||||||
|
[@brandonaaron](https://github.com/brandonaaron)
|
||||||
|
|
||||||
|
* Ensure correct auto-rotation for EXIF Orientation values 2 and 4.
|
||||||
|
[#288](https://github.com/lovell/sharp/pull/288)
|
||||||
|
[@brandonaaron](https://github.com/brandonaaron)
|
||||||
|
|
||||||
|
* Make static linking possible via `--runtime_link` install option.
|
||||||
|
[#287](https://github.com/lovell/sharp/pull/287)
|
||||||
|
[@vlapo](https://github.com/vlapo)
|
||||||
|
|
||||||
|
#### v0.11.3 - 8<sup>th</sup> September 2015
|
||||||
|
|
||||||
|
* Intrepret blurSigma, sharpenFlat, and sharpenJagged as double precision.
|
||||||
|
[#263](https://github.com/lovell/sharp/pull/263)
|
||||||
|
[@chrisriley](https://github.com/chrisriley)
|
||||||
|
|
||||||
|
#### v0.11.2 - 28<sup>th</sup> August 2015
|
||||||
|
|
||||||
|
* Allow crop gravity to be provided as a String.
|
||||||
|
[#255](https://github.com/lovell/sharp/pull/255)
|
||||||
|
[@papandreou](https://github.com/papandreou)
|
||||||
|
* Add support for io.js v3 and Node v4.
|
||||||
|
[#246](https://github.com/lovell/sharp/issues/246)
|
||||||
|
|
||||||
#### v0.11.1 - 12<sup>th</sup> August 2015
|
#### v0.11.1 - 12<sup>th</sup> August 2015
|
||||||
|
|
||||||
* Silence MSVC warning: "C4530: C++ exception handler used, but unwind semantics are not enabled".
|
* Silence MSVC warning: "C4530: C++ exception handler used, but unwind semantics are not enabled".
|
||||||
|
|||||||
@@ -4,15 +4,27 @@ The typical use case for this high speed Node.js module
|
|||||||
is to convert large images of many formats to
|
is to convert large images of many formats to
|
||||||
smaller, web-friendly JPEG, PNG and WebP images of varying dimensions.
|
smaller, web-friendly JPEG, PNG and WebP images of varying dimensions.
|
||||||
|
|
||||||
Resizing an image is typically 4x faster than using
|
Resizing an image is typically 4x faster than using the
|
||||||
the quickest ImageMagick and GraphicsMagick settings.
|
quickest ImageMagick and GraphicsMagick settings.
|
||||||
|
|
||||||
|
Colour spaces, embedded ICC profiles and alpha transparency channels are all handled correctly.
|
||||||
|
Bicubic interpolation with Lanczos anti-alias filtering ensures quality is not sacrificed for speed.
|
||||||
|
|
||||||
|
As well as image resizing, operations such as
|
||||||
|
rotation, extraction, compositing and gamma correction are available.
|
||||||
|
|
||||||
|
Most Windows (x64), Linux and ARMv6+ systems do not require
|
||||||
|
the installation of any external runtime dependencies.
|
||||||
|
|
||||||
|
Use with OS X is as simple as running `brew install homebrew/science/vips`
|
||||||
|
to install the libvips dependency.
|
||||||
|
|
||||||
[](https://coveralls.io/r/lovell/sharp?branch=master)
|
[](https://coveralls.io/r/lovell/sharp?branch=master)
|
||||||
|
|
||||||
### Formats
|
### Formats
|
||||||
|
|
||||||
This module supports reading JPEG, PNG, WebP, TIFF, OpenSlide,
|
This module supports reading JPEG, PNG, WebP, TIFF, OpenSlide,
|
||||||
GIF and other libmagick-supported formats.
|
GIF and most other libmagick-supported formats.
|
||||||
|
|
||||||
Output images can be in JPEG, PNG and WebP formats as well as uncompressed raw pixel data.
|
Output images can be in JPEG, PNG and WebP formats as well as uncompressed raw pixel data.
|
||||||
|
|
||||||
@@ -25,14 +37,6 @@ suitable for use with "slippy map" tile viewers like
|
|||||||
[OpenSeadragon](https://github.com/openseadragon/openseadragon)
|
[OpenSeadragon](https://github.com/openseadragon/openseadragon)
|
||||||
and [Leaflet](https://github.com/turban/Leaflet.Zoomify).
|
and [Leaflet](https://github.com/turban/Leaflet.Zoomify).
|
||||||
|
|
||||||
### Features
|
|
||||||
|
|
||||||
As well as image resizing, operations such as
|
|
||||||
rotation, extraction, compositing and gamma correction are available.
|
|
||||||
|
|
||||||
Colour spaces, embedded ICC profiles and alpha transparency channels
|
|
||||||
are all handled correctly.
|
|
||||||
|
|
||||||
### Fast
|
### Fast
|
||||||
|
|
||||||
This module is powered by the blazingly fast
|
This module is powered by the blazingly fast
|
||||||
@@ -84,6 +88,7 @@ the help and code contributions of the following people:
|
|||||||
* [Victor Mateevitsi](https://github.com/mvictoras)
|
* [Victor Mateevitsi](https://github.com/mvictoras)
|
||||||
* [Alaric Holloway](https://github.com/skedastik)
|
* [Alaric Holloway](https://github.com/skedastik)
|
||||||
* [Bernhard K. Weisshuhn](https://github.com/bkw)
|
* [Bernhard K. Weisshuhn](https://github.com/bkw)
|
||||||
|
* [David A. Carley](https://github.com/dacarley)
|
||||||
|
|
||||||
Thank you!
|
Thank you!
|
||||||
|
|
||||||
|
|||||||
157
docs/install.md
@@ -6,74 +6,73 @@ npm install sharp
|
|||||||
|
|
||||||
### Prerequisites
|
### Prerequisites
|
||||||
|
|
||||||
* Node.js v0.10+ or io.js
|
* C++11 compatible compiler such as gcc 4.6+ (Node v4+ requires gcc 4.8+), clang 3.0+ or MSVC 2013
|
||||||
* [libvips](https://github.com/jcupitt/libvips) v7.40.0+ (7.42.0+ recommended)
|
* [node-gyp](https://github.com/TooTallNate/node-gyp#installation)
|
||||||
* C++11 compatible compiler such as gcc 4.6+, clang 3.0+ or MSVC 2013
|
|
||||||
|
|
||||||
### Linux
|
### Linux
|
||||||
|
|
||||||
[](https://travis-ci.org/lovell/sharp)
|
[](https://travis-ci.org/lovell/sharp)
|
||||||
[](https://snap-ci.com/lovell/sharp/branch/master)
|
[](https://circleci.com/gh/lovell/sharp)
|
||||||
|
|
||||||
For a system-wide installation of the most suitable version of
|
libvips and its dependencies are fetched and stored within `node_modules/sharp/lib` during `npm install`.
|
||||||
libvips and its dependencies on the following Operating Systems:
|
This involves an automated HTTPS download of approximately 6MB.
|
||||||
|
|
||||||
|
Most recent Linux-based operating systems with glibc running on x64 and ARMv6+ CPUs should "just work", e.g.:
|
||||||
|
|
||||||
* Debian 7, 8
|
* Debian 7, 8
|
||||||
* Ubuntu 12.04, 14.04, 14.10, 15.04
|
* Ubuntu 12.04, 14.04, 14.10, 15.04, 15.10
|
||||||
* Mint 13, 17
|
* Centos 7
|
||||||
* RHEL/Centos/Scientific 6, 7
|
* Fedora 21, 22, 23
|
||||||
* Fedora 21, 22
|
* openSUSE 13.2
|
||||||
* Amazon Linux 2014.09, 2015.03
|
* Archlinux 2015.06.01
|
||||||
* OpenSuse 13
|
* Raspbian Jessie
|
||||||
|
* Amazon Linux 2015.03, 2015.09
|
||||||
|
|
||||||
run the following as a user with `sudo` access:
|
To use your own version of libvips instead of the provided binaries, make sure it is
|
||||||
|
at least the version listed under `config.libvips` in the `package.json` file that it
|
||||||
|
can be located using `pkg-config`. If you are using non-stadard paths (anything other
|
||||||
|
than `/usr` or `/usr/local`), you might need to set `PKG_CONFIG_PATH` during `npm install`
|
||||||
|
and `LD_LIBRARY_PATH` at runtime.
|
||||||
|
|
||||||
|
You can print the detected vips version using: `pkg-config --modversion vips-cpp`
|
||||||
|
|
||||||
|
This allows the use of newer versions of libvips with older versions of sharp.
|
||||||
|
|
||||||
|
For older Linux-based operating systems and 32-bit Intel CPUs,
|
||||||
|
a system-wide installation of the most suitable version of
|
||||||
|
libvips and its dependencies can be achieved by running
|
||||||
|
the following command as a user with `sudo` access
|
||||||
|
(requires `curl` and `pkg-config`):
|
||||||
|
|
||||||
```sh
|
```sh
|
||||||
curl -s https://raw.githubusercontent.com/lovell/sharp/master/preinstall.sh | sudo bash -
|
curl -s https://raw.githubusercontent.com/lovell/sharp/master/preinstall.sh | sudo bash -
|
||||||
```
|
```
|
||||||
|
|
||||||
or run the following as `root`:
|
For Linux-based operating systems such as Alpine that use musl libc,
|
||||||
|
the smaller stack size means libvips' cache should be disabled
|
||||||
|
via `sharp.cache(false)` to avoid a stack overflow.
|
||||||
|
|
||||||
```sh
|
Beware of Linux OS upgrades that introduce v5.1+ of the `g++` compiler due to
|
||||||
curl -s https://raw.githubusercontent.com/lovell/sharp/master/preinstall.sh | bash -
|
[changes](https://gcc.gnu.org/onlinedocs/libstdc++/manual/using_dual_abi.html)
|
||||||
```
|
in the C++11 ABI.
|
||||||
|
This module assumes the previous behaviour, which can be enforced by setting the
|
||||||
The [preinstall.sh](https://github.com/lovell/sharp/blob/master/preinstall.sh) script requires `curl` and `pkg-config`.
|
`_GLIBCXX_USE_CXX11_ABI=0` environment variable at libvips' compile time.
|
||||||
|
|
||||||
Add `--with-openslide` to enable OpenSlide support:
|
|
||||||
|
|
||||||
```sh
|
|
||||||
curl -s https://raw.githubusercontent.com/lovell/sharp/master/preinstall.sh | sudo bash -s -- --with-openslide
|
|
||||||
```
|
|
||||||
|
|
||||||
#### Ubuntu LTS
|
|
||||||
|
|
||||||
libvips v7.40.6 is available via a PPA.
|
|
||||||
|
|
||||||
##### 12.04
|
|
||||||
|
|
||||||
```sh
|
|
||||||
sudo add-apt-repository -y ppa:lovell/precise-backport-vips
|
|
||||||
sudo apt-get update
|
|
||||||
sudo apt-get install -y libvips-dev libgsf-1-dev
|
|
||||||
```
|
|
||||||
|
|
||||||
##### 14.04
|
|
||||||
|
|
||||||
```sh
|
|
||||||
sudo add-apt-repository -y ppa:lovell/trusty-backport-vips
|
|
||||||
sudo apt-get update
|
|
||||||
sudo apt-get install -y libvips-dev libgsf-1-dev
|
|
||||||
```
|
|
||||||
|
|
||||||
### Mac OS
|
### Mac OS
|
||||||
|
|
||||||
[](https://travis-ci.org/lovell/sharp-osx-ci)
|
[](https://travis-ci.org/lovell/sharp)
|
||||||
|
|
||||||
Install libvips via homebrew:
|
libvips must be installed before `npm install` is run.
|
||||||
|
This can be achieved via homebrew:
|
||||||
|
|
||||||
```sh
|
```sh
|
||||||
brew install homebrew/science/vips --with-webp --with-graphicsmagick
|
brew install homebrew/science/vips
|
||||||
|
```
|
||||||
|
|
||||||
|
For GIF input and WebP output suppport use:
|
||||||
|
|
||||||
|
```sh
|
||||||
|
brew install homebrew/science/vips --with-imagemagick --with-webp
|
||||||
```
|
```
|
||||||
|
|
||||||
A missing or incorrectly configured _Xcode Command Line Tools_ installation
|
A missing or incorrectly configured _Xcode Command Line Tools_ installation
|
||||||
@@ -88,18 +87,22 @@ If so, please try `brew link gettext --force`.
|
|||||||
|
|
||||||
### Windows
|
### Windows
|
||||||
|
|
||||||
[](https://ci.appveyor.com/project/lovell/sharp)
|
[](https://ci.appveyor.com/project/lovell/sharp)
|
||||||
|
|
||||||
Requires x86 32-bit Node.js or io.js (use `iojs.exe` rather than `node.exe`).
|
libvips and its dependencies are fetched and stored within `node_modules\sharp` during `npm install`.
|
||||||
|
This involves an automated HTTPS download of approximately 9MB.
|
||||||
|
|
||||||
The WebP format is currently unsupported.
|
Only 64-bit (x64) `node.exe` is supported.
|
||||||
|
The WebP format is currently unavailable on Windows.
|
||||||
|
|
||||||
1. Ensure the [node-gyp prerequisites](https://github.com/TooTallNate/node-gyp#installation) are met.
|
### FreeBSD
|
||||||
2. [Download](http://www.vips.ecs.soton.ac.uk/supported/current/win32/) and unzip `vips-dev.x.y.z.zip`.
|
|
||||||
3. Set the `VIPS_HOME` environment variable to the full path of the `vips-dev-x.y.z` directory.
|
|
||||||
4. Add `vips-dev-x.y.z\bin` to `PATH`.
|
|
||||||
|
|
||||||
Versions of MSVC more recent than 2013 may require the use of `npm install --arch=ia32 --msvs_version=2013`.
|
libvips must be installed before `npm install` is run.
|
||||||
|
This can be achieved via [FreshPorts](https://www.freshports.org/graphics/vips/):
|
||||||
|
|
||||||
|
```sh
|
||||||
|
cd /usr/ports/graphics/vips/ && make install clean
|
||||||
|
```
|
||||||
|
|
||||||
### Heroku
|
### Heroku
|
||||||
|
|
||||||
@@ -109,15 +112,49 @@ and its dependencies.
|
|||||||
|
|
||||||
### Docker
|
### Docker
|
||||||
|
|
||||||
[Marc Bachmann](https://github.com/marcbachmann) maintains a
|
[Marc Bachmann](https://github.com/marcbachmann) maintains an
|
||||||
[Dockerfile for libvips](https://github.com/marcbachmann/dockerfile-libvips).
|
[Ubuntu-based Dockerfile for libvips](https://github.com/marcbachmann/dockerfile-libvips).
|
||||||
|
|
||||||
```sh
|
```sh
|
||||||
docker pull marcbachmann/libvips
|
docker pull marcbachmann/libvips
|
||||||
```
|
```
|
||||||
|
|
||||||
|
[Will Jordan](https://github.com/wjordan) maintains an
|
||||||
|
[Alpine-based Dockerfile for libvips](https://github.com/wjordan/dockerfile-libvips).
|
||||||
|
|
||||||
|
```sh
|
||||||
|
docker pull wjordan/libvips
|
||||||
|
```
|
||||||
|
|
||||||
|
### AWS Lambda
|
||||||
|
|
||||||
|
In order to use sharp on AWS Lambda, you need to [create a deployment package](http://docs.aws.amazon.com/lambda/latest/dg/nodejs-create-deployment-pkg.html). Because sharp
|
||||||
|
downloads and links libraries for the current platform during `npm install` you have to
|
||||||
|
do this on a system similar to the [Lambda Execution Environment](http://docs.aws.amazon.com/lambda/latest/dg/current-supported-versions.html). The easiest ways to do this, is to setup a
|
||||||
|
small t2.micro instance using the AMI ID listed in the previous link, ssh into it as ec2-user
|
||||||
|
and follow the instructions below.
|
||||||
|
|
||||||
|
Install depencies:
|
||||||
|
|
||||||
|
```sh
|
||||||
|
sudo yum-config-manager --enable epel
|
||||||
|
sudo yum install -y nodejs gcc-c++
|
||||||
|
curl -s https://www.npmjs.com/install.sh | sudo sh
|
||||||
|
```
|
||||||
|
|
||||||
|
Copy your code and package.json to the instance using `scp` and create a deployment package:
|
||||||
|
|
||||||
|
```sh
|
||||||
|
cd sharp-lambda-example
|
||||||
|
npm install
|
||||||
|
zip -ur9 ../sharp-lambda-example.zip index.js node_modules
|
||||||
|
```
|
||||||
|
|
||||||
|
You can now download your deployment ZIP using `scp` and upload it to Lambda.
|
||||||
|
|
||||||
|
**Performance Tip:** To get the best performance on Lambda choose the largest memory available because this also gives you the most cpu time (a 1536 MB function is 12x faster than a 128 MB function).
|
||||||
|
|
||||||
### Build tools
|
### Build tools
|
||||||
|
|
||||||
* [gulp-responsive](https://www.npmjs.com/package/gulp-responsive)
|
* [gulp-responsive](https://www.npmjs.com/package/gulp-responsive)
|
||||||
* [gulp-sharp](https://www.npmjs.com/package/gulp-sharp)
|
|
||||||
* [grunt-sharp](https://www.npmjs.com/package/grunt-sharp)
|
* [grunt-sharp](https://www.npmjs.com/package/grunt-sharp)
|
||||||
|
|||||||
@@ -2,39 +2,40 @@
|
|||||||
|
|
||||||
### Test environment
|
### Test environment
|
||||||
|
|
||||||
* AWS EC2 [c4.xlarge](http://aws.amazon.com/ec2/instance-types/#c4)
|
* AWS EC2 [c4.xlarge](http://aws.amazon.com/ec2/instance-types/#c4) (4x E5-2666 v3 @2.90GHz)
|
||||||
* Ubuntu 15.04
|
* Amazon Linux 2015.09.1
|
||||||
* Node.js 0.12.7
|
* Node.js v5.5.0
|
||||||
* libvips 8.0.2
|
|
||||||
* liborc 0.4.22
|
|
||||||
|
|
||||||
### The contenders
|
### The contenders
|
||||||
|
|
||||||
* [lwip](https://www.npmjs.com/package/lwip) 0.0.7 - Wrapper around CImg, compiles dependencies from source
|
* [jimp](https://www.npmjs.com/package/jimp) v0.2.20 - Image processing in pure JavaScript. Bilinear interpolation only.
|
||||||
* [imagemagick-native](https://www.npmjs.com/package/imagemagick-native) git@45d4e2e - Wrapper around libmagick++, supports Buffers only.
|
* [lwip](https://www.npmjs.com/package/lwip) v0.0.8 - Wrapper around CImg, compiles dependencies from source.
|
||||||
* [imagemagick](https://www.npmjs.com/package/imagemagick) 0.1.3 - Supports filesystem only and "*has been unmaintained for a long time*".
|
* [imagemagick-native](https://www.npmjs.com/package/imagemagick-native) @47c7329 - Wrapper around libmagick++, supports Buffers only.
|
||||||
* [gm](https://www.npmjs.com/package/gm) 1.18.1 - Fully featured wrapper around GraphicsMagick's `convert` command line utility.
|
* [imagemagick](https://www.npmjs.com/package/imagemagick) v0.1.3 - Supports filesystem only and "*has been unmaintained for a long time*".
|
||||||
* sharp 0.11.0 - Caching within libvips disabled to ensure a fair comparison.
|
* [gm](https://www.npmjs.com/package/gm) v1.21.0 - Fully featured wrapper around GraphicsMagick's `gm` command line utility.
|
||||||
|
* sharp v0.13.0 / libvips v8.2.2 - Caching within libvips disabled to ensure a fair comparison.
|
||||||
|
|
||||||
### The task
|
### The task
|
||||||
|
|
||||||
Decompress a 2725x2225 JPEG image, resize to 720x480 using bilinear interpolation, then compress to JPEG.
|
Decompress a 2725x2225 JPEG image, resize to 720x480 using bicubic interpolation (where available), then compress to JPEG.
|
||||||
|
|
||||||
### Results
|
### Results
|
||||||
|
|
||||||
| Module | Input | Output | Ops/sec | Speed-up |
|
| Module | Input | Output | Ops/sec | Speed-up |
|
||||||
| :----------------- | :----- | :----- | ------: | -------: |
|
| :----------------- | :----- | :----- | ------: | -------: |
|
||||||
| lwip | file | file | 1.75 | 1 |
|
| jimp (bilinear) | file | file | 1.04 | 1.0 |
|
||||||
| lwip | buffer | buffer | 2.21 | 1.3 |
|
| jimp (bilinear) | buffer | buffer | 1.07 | 1.0 |
|
||||||
| imagemagick-native | buffer | buffer | 7.13 | 4.1 |
|
| lwip | file | file | 1.13 | 1.1 |
|
||||||
| gm | buffer | buffer | 7.27 | 4.2 |
|
| lwip | buffer | buffer | 1.13 | 1.1 |
|
||||||
| gm | file | file | 7.33 | 4.2 |
|
| imagemagick-native | buffer | buffer | 1.65 | 1.6 |
|
||||||
| imagemagick | file | file | 10.04 | 5.7 |
|
| imagemagick | file | file | 5.02 | 4.8 |
|
||||||
| sharp | stream | stream | 23.12 | 13.2 |
|
| gm | buffer | buffer | 5.36 | 5.2 |
|
||||||
| sharp | file | file | 24.43 | 14.0 |
|
| gm | file | file | 5.39 | 5.2 |
|
||||||
| sharp | file | buffer | 24.55 | 14.0 |
|
| sharp | stream | stream | 22.00 | 21.2 |
|
||||||
| sharp | buffer | file | 24.86 | 14.2 |
|
| sharp | file | file | 22.87 | 22.0 |
|
||||||
| sharp | buffer | buffer | 24.92 | 14.2 |
|
| sharp | file | buffer | 23.03 | 22.1 |
|
||||||
|
| sharp | buffer | file | 23.10 | 22.2 |
|
||||||
|
| sharp | buffer | buffer | 23.21 | 22.3 |
|
||||||
|
|
||||||
Greater performance can be expected with caching enabled (default) and using 8+ core machines.
|
Greater performance can be expected with caching enabled (default) and using 8+ core machines.
|
||||||
|
|
||||||
@@ -50,14 +51,20 @@ brew install graphicsmagick
|
|||||||
```
|
```
|
||||||
|
|
||||||
```sh
|
```sh
|
||||||
sudo apt-get install imagemagick graphicsmagick libmagick++-dev
|
sudo apt-get install imagemagick libmagick++-dev graphicsmagick
|
||||||
|
```
|
||||||
|
|
||||||
|
```sh
|
||||||
|
sudo yum install ImageMagick-devel ImageMagick-c++-devel GraphicsMagick
|
||||||
```
|
```
|
||||||
|
|
||||||
### Running the benchmark test
|
### Running the benchmark test
|
||||||
|
|
||||||
```sh
|
```sh
|
||||||
git clone https://github.com/lovell/sharp.git
|
git clone https://github.com/lovell/sharp.git
|
||||||
cd sharp/test/bench
|
cd sharp
|
||||||
|
npm install
|
||||||
|
cd test/bench
|
||||||
npm install
|
npm install
|
||||||
npm test
|
npm test
|
||||||
```
|
```
|
||||||
|
|||||||
432
index.js
Executable file → Normal file
@@ -10,17 +10,34 @@ var color = require('color');
|
|||||||
var BluebirdPromise = require('bluebird');
|
var BluebirdPromise = require('bluebird');
|
||||||
|
|
||||||
var sharp = require('./build/Release/sharp');
|
var sharp = require('./build/Release/sharp');
|
||||||
var libvipsVersion = sharp.libvipsVersion();
|
|
||||||
|
|
||||||
|
// Versioning
|
||||||
|
var versions = {
|
||||||
|
vips: sharp.libvipsVersion()
|
||||||
|
};
|
||||||
|
(function() {
|
||||||
|
// Does libvips meet minimum requirement?
|
||||||
|
var libvipsVersionMin = require('./package.json').config.libvips;
|
||||||
|
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('./lib/versions.json');
|
||||||
|
} catch (err) {}
|
||||||
|
})();
|
||||||
|
|
||||||
|
// Limits
|
||||||
var maximum = {
|
var maximum = {
|
||||||
width: 0x3FFF,
|
width: 0x3FFF,
|
||||||
height: 0x3FFF,
|
height: 0x3FFF,
|
||||||
pixels: Math.pow(0x3FFF, 2)
|
pixels: Math.pow(0x3FFF, 2)
|
||||||
};
|
};
|
||||||
|
|
||||||
var Sharp = function(input) {
|
// Constructor-factory
|
||||||
|
var Sharp = function(input, options) {
|
||||||
if (!(this instanceof Sharp)) {
|
if (!(this instanceof Sharp)) {
|
||||||
return new Sharp(input);
|
return new Sharp(input, options);
|
||||||
}
|
}
|
||||||
stream.Duplex.call(this);
|
stream.Duplex.call(this);
|
||||||
this.options = {
|
this.options = {
|
||||||
@@ -29,6 +46,10 @@ var Sharp = function(input) {
|
|||||||
streamIn: false,
|
streamIn: false,
|
||||||
sequentialRead: false,
|
sequentialRead: false,
|
||||||
limitInputPixels: maximum.pixels,
|
limitInputPixels: maximum.pixels,
|
||||||
|
density: 72,
|
||||||
|
rawWidth: 0,
|
||||||
|
rawHeight: 0,
|
||||||
|
rawChannels: 0,
|
||||||
// ICC profiles
|
// ICC profiles
|
||||||
iccProfilePath: path.join(__dirname, 'icc') + path.sep,
|
iccProfilePath: path.join(__dirname, 'icc') + path.sep,
|
||||||
// resize options
|
// resize options
|
||||||
@@ -43,27 +64,36 @@ var Sharp = function(input) {
|
|||||||
width: -1,
|
width: -1,
|
||||||
height: -1,
|
height: -1,
|
||||||
canvas: 'crop',
|
canvas: 'crop',
|
||||||
gravity: 0,
|
crop: 0,
|
||||||
angle: 0,
|
angle: 0,
|
||||||
rotateBeforePreExtract: false,
|
rotateBeforePreExtract: false,
|
||||||
flip: false,
|
flip: false,
|
||||||
flop: false,
|
flop: false,
|
||||||
|
extendTop: 0,
|
||||||
|
extendBottom: 0,
|
||||||
|
extendLeft: 0,
|
||||||
|
extendRight: 0,
|
||||||
withoutEnlargement: false,
|
withoutEnlargement: false,
|
||||||
interpolator: 'bilinear',
|
interpolator: 'bicubic',
|
||||||
// operations
|
// operations
|
||||||
background: [0, 0, 0, 255],
|
background: [0, 0, 0, 255],
|
||||||
flatten: false,
|
flatten: false,
|
||||||
|
negate: false,
|
||||||
blurSigma: 0,
|
blurSigma: 0,
|
||||||
sharpenRadius: 0,
|
sharpenRadius: 0,
|
||||||
sharpenFlat: 1,
|
sharpenFlat: 1,
|
||||||
sharpenJagged: 2,
|
sharpenJagged: 2,
|
||||||
|
threshold: 0,
|
||||||
gamma: 0,
|
gamma: 0,
|
||||||
greyscale: false,
|
greyscale: false,
|
||||||
normalize: 0,
|
normalize: 0,
|
||||||
// overlay
|
// overlay
|
||||||
overlayPath: '',
|
overlayFileIn: '',
|
||||||
|
overlayBufferIn: null,
|
||||||
|
overlayGravity: 0,
|
||||||
// output options
|
// output options
|
||||||
output: '__input',
|
formatOut: 'input',
|
||||||
|
fileOut: '',
|
||||||
progressive: false,
|
progressive: false,
|
||||||
quality: 80,
|
quality: 80,
|
||||||
compressionLevel: 6,
|
compressionLevel: 6,
|
||||||
@@ -82,18 +112,19 @@ var Sharp = function(input) {
|
|||||||
module.exports.queue.emit('change', queueLength);
|
module.exports.queue.emit('change', queueLength);
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
if (typeof input === 'string') {
|
if (isString(input)) {
|
||||||
// input=file
|
// input=file
|
||||||
this.options.fileIn = input;
|
this.options.fileIn = input;
|
||||||
} else if (typeof input === 'object' && input instanceof Buffer) {
|
} else if (isBuffer(input)) {
|
||||||
// input=buffer
|
// input=buffer
|
||||||
this.options.bufferIn = input;
|
this.options.bufferIn = input;
|
||||||
} else if (typeof input === 'undefined') {
|
} else if (!isDefined(input)) {
|
||||||
// input=stream
|
// input=stream
|
||||||
this.options.streamIn = true;
|
this.options.streamIn = true;
|
||||||
} else {
|
} else {
|
||||||
throw new Error('Unsupported input ' + typeof input);
|
throw new Error('Unsupported input ' + typeof input);
|
||||||
}
|
}
|
||||||
|
this._inputOptions(options);
|
||||||
return this;
|
return this;
|
||||||
};
|
};
|
||||||
module.exports = Sharp;
|
module.exports = Sharp;
|
||||||
@@ -109,6 +140,70 @@ module.exports.queue = new events.EventEmitter();
|
|||||||
*/
|
*/
|
||||||
module.exports.format = sharp.format();
|
module.exports.format = sharp.format();
|
||||||
|
|
||||||
|
/*
|
||||||
|
Version numbers of libvips and its dependencies
|
||||||
|
*/
|
||||||
|
module.exports.versions = versions;
|
||||||
|
|
||||||
|
/*
|
||||||
|
Validation helpers
|
||||||
|
*/
|
||||||
|
var isDefined = function(val) {
|
||||||
|
return typeof val !== 'undefined' && val !== null;
|
||||||
|
};
|
||||||
|
var isObject = function(val) {
|
||||||
|
return typeof val === 'object';
|
||||||
|
};
|
||||||
|
var isBuffer = function(val) {
|
||||||
|
return typeof val === 'object' && val instanceof Buffer;
|
||||||
|
};
|
||||||
|
var isString = function(val) {
|
||||||
|
return typeof val === 'string' && val.length > 0;
|
||||||
|
};
|
||||||
|
var isInteger = function(val) {
|
||||||
|
return typeof val === 'number' && !Number.isNaN(val) && val % 1 === 0;
|
||||||
|
};
|
||||||
|
var inRange = function(val, min, max) {
|
||||||
|
return val >= min && val <= max;
|
||||||
|
};
|
||||||
|
var contains = function(val, list) {
|
||||||
|
return list.indexOf(val) !== -1;
|
||||||
|
};
|
||||||
|
|
||||||
|
/*
|
||||||
|
Set input-related options
|
||||||
|
density: DPI at which to load vector images via libmagick
|
||||||
|
*/
|
||||||
|
Sharp.prototype._inputOptions = function(options) {
|
||||||
|
if (isObject(options)) {
|
||||||
|
// Density
|
||||||
|
if (isDefined(options.density)) {
|
||||||
|
if (isInteger(options.density) && inRange(options.density, 1, 2400)) {
|
||||||
|
this.options.density = options.density;
|
||||||
|
} else {
|
||||||
|
throw new Error('Invalid density (1 to 2400) ' + options.density);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// Raw pixel input
|
||||||
|
if (isDefined(options.raw)) {
|
||||||
|
if (
|
||||||
|
isObject(options.raw) &&
|
||||||
|
isInteger(options.raw.width) && inRange(options.raw.width, 1, maximum.width) &&
|
||||||
|
isInteger(options.raw.height) && inRange(options.raw.height, 1, maximum.height) &&
|
||||||
|
isInteger(options.raw.channels) && inRange(options.raw.channels, 1, 4)
|
||||||
|
) {
|
||||||
|
this.options.rawWidth = options.raw.width;
|
||||||
|
this.options.rawHeight = options.raw.height;
|
||||||
|
this.options.rawChannels = options.raw.channels;
|
||||||
|
} else {
|
||||||
|
throw new Error('Expected width, height and channels for raw pixel input');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else if (isDefined(options)) {
|
||||||
|
throw new Error('Invalid input options ' + options);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Handle incoming chunk on Writable Stream
|
Handle incoming chunk on Writable Stream
|
||||||
*/
|
*/
|
||||||
@@ -136,30 +231,58 @@ Sharp.prototype._write = function(chunk, encoding, callback) {
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
// Crop this part of the resized image (Center/Centre, North, East, South, West)
|
// Weighting to apply to image crop
|
||||||
module.exports.gravity = {'center': 0, 'centre': 0, 'north': 1, 'east': 2, 'south': 3, 'west': 4};
|
module.exports.gravity = {
|
||||||
|
center: 0,
|
||||||
|
centre: 0,
|
||||||
|
north: 1,
|
||||||
|
east: 2,
|
||||||
|
south: 3,
|
||||||
|
west: 4,
|
||||||
|
northeast: 5,
|
||||||
|
southeast: 6,
|
||||||
|
southwest: 7,
|
||||||
|
northwest: 8
|
||||||
|
};
|
||||||
|
|
||||||
Sharp.prototype.crop = function(gravity) {
|
// Strategies for automagic behaviour
|
||||||
|
module.exports.strategy = {
|
||||||
|
entropy: 16
|
||||||
|
};
|
||||||
|
|
||||||
|
/*
|
||||||
|
What part of the image should be retained when cropping?
|
||||||
|
*/
|
||||||
|
Sharp.prototype.crop = function(crop) {
|
||||||
this.options.canvas = 'crop';
|
this.options.canvas = 'crop';
|
||||||
if (typeof gravity === 'number' && !Number.isNaN(gravity) && gravity >= 0 && gravity <= 4) {
|
if (!isDefined(crop)) {
|
||||||
this.options.gravity = gravity;
|
// Default
|
||||||
|
this.options.crop = module.exports.gravity.center;
|
||||||
|
} else if (isInteger(crop) && inRange(crop, 0, 8)) {
|
||||||
|
// Gravity (numeric)
|
||||||
|
this.options.crop = crop;
|
||||||
|
} else if (isString(crop) && isInteger(module.exports.gravity[crop])) {
|
||||||
|
// Gravity (string)
|
||||||
|
this.options.crop = module.exports.gravity[crop];
|
||||||
|
} else if (isInteger(crop) && crop === module.exports.strategy.entropy) {
|
||||||
|
// Strategy
|
||||||
|
this.options.crop = crop;
|
||||||
} else {
|
} else {
|
||||||
throw new Error('Unsupported crop gravity ' + gravity);
|
throw new Error('Unsupported crop ' + crop);
|
||||||
}
|
}
|
||||||
return this;
|
return this;
|
||||||
};
|
};
|
||||||
|
|
||||||
Sharp.prototype.extract = function(topOffset, leftOffset, width, height) {
|
Sharp.prototype.extract = function(options) {
|
||||||
/*jslint unused: false */
|
|
||||||
var suffix = this.options.width === -1 && this.options.height === -1 ? 'Pre' : 'Post';
|
var suffix = this.options.width === -1 && this.options.height === -1 ? 'Pre' : 'Post';
|
||||||
var values = arguments;
|
['left', 'top', 'width', 'height'].forEach(function (name) {
|
||||||
['topOffset', 'leftOffset', 'width', 'height'].forEach(function(name, index) {
|
var value = options[name];
|
||||||
if (typeof values[index] === 'number' && !Number.isNaN(values[index]) && (values[index] % 1 === 0) && values[index] >= 0) {
|
if (isInteger(value) && value >= 0) {
|
||||||
this.options[name + suffix] = values[index];
|
this.options[name + (name === 'left' || name === 'top' ? 'Offset' : '') + suffix] = value;
|
||||||
} else {
|
} else {
|
||||||
throw new Error('Non-integer value for ' + name + ' of ' + values[index]);
|
throw new Error('Non-integer value for ' + name + ' of ' + value);
|
||||||
}
|
}
|
||||||
}.bind(this));
|
}, this);
|
||||||
// Ensure existing rotation occurs before pre-resize extraction
|
// Ensure existing rotation occurs before pre-resize extraction
|
||||||
if (suffix === 'Pre' && this.options.angle !== 0) {
|
if (suffix === 'Pre' && this.options.angle !== 0) {
|
||||||
this.options.rotateBeforePreExtract = true;
|
this.options.rotateBeforePreExtract = true;
|
||||||
@@ -208,14 +331,31 @@ Sharp.prototype.flatten = function(flatten) {
|
|||||||
return this;
|
return this;
|
||||||
};
|
};
|
||||||
|
|
||||||
Sharp.prototype.overlayWith = function(overlayPath) {
|
Sharp.prototype.negate = function(negate) {
|
||||||
if (typeof overlayPath !== 'string') {
|
this.options.negate = (typeof negate === 'boolean') ? negate : true;
|
||||||
throw new Error('The overlay path must be a string');
|
return this;
|
||||||
|
};
|
||||||
|
|
||||||
|
/*
|
||||||
|
Overlay with another image, using an optional gravity
|
||||||
|
*/
|
||||||
|
Sharp.prototype.overlayWith = function(overlay, options) {
|
||||||
|
if (isString(overlay)) {
|
||||||
|
this.options.overlayFileIn = overlay;
|
||||||
|
} else if (isBuffer(overlay)) {
|
||||||
|
this.options.overlayBufferIn = overlay;
|
||||||
|
} else {
|
||||||
|
throw new Error('Unsupported overlay ' + typeof overlay);
|
||||||
}
|
}
|
||||||
if (overlayPath === '') {
|
if (isObject(options)) {
|
||||||
throw new Error('The overlay path cannot be empty');
|
if (isInteger(options.gravity) && inRange(options.gravity, 0, 8)) {
|
||||||
|
this.options.overlayGravity = options.gravity;
|
||||||
|
} else if (isString(options.gravity) && isInteger(module.exports.gravity[options.gravity])) {
|
||||||
|
this.options.overlayGravity = module.exports.gravity[options.gravity];
|
||||||
|
} else if (isDefined(options.gravity)) {
|
||||||
|
throw new Error('Unsupported overlay gravity ' + options.gravity);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
this.options.overlayPath = overlayPath;
|
|
||||||
return this;
|
return this;
|
||||||
};
|
};
|
||||||
|
|
||||||
@@ -321,6 +461,19 @@ Sharp.prototype.sharpen = function(radius, flat, jagged) {
|
|||||||
return this;
|
return this;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
Sharp.prototype.threshold = function(threshold) {
|
||||||
|
if (typeof threshold === 'undefined') {
|
||||||
|
this.options.threshold = 128;
|
||||||
|
} else if (typeof threshold === 'boolean') {
|
||||||
|
this.options.threshold = threshold ? 128 : 0;
|
||||||
|
} else if (typeof threshold === 'number' && !Number.isNaN(threshold) && (threshold % 1 === 0) && threshold >= 0 && threshold <= 255) {
|
||||||
|
this.options.threshold = threshold;
|
||||||
|
} else {
|
||||||
|
throw new Error('Invalid threshold (0 to 255) ' + threshold);
|
||||||
|
}
|
||||||
|
return this;
|
||||||
|
};
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Set the interpolator to use for the affine transformation
|
Set the interpolator to use for the affine transformation
|
||||||
*/
|
*/
|
||||||
@@ -368,11 +521,7 @@ Sharp.prototype.gamma = function(gamma) {
|
|||||||
Enhance output image contrast by stretching its luminance to cover the full dynamic range
|
Enhance output image contrast by stretching its luminance to cover the full dynamic range
|
||||||
*/
|
*/
|
||||||
Sharp.prototype.normalize = function(normalize) {
|
Sharp.prototype.normalize = function(normalize) {
|
||||||
if (process.platform !== 'win32') {
|
this.options.normalize = (typeof normalize === 'boolean') ? normalize : true;
|
||||||
this.options.normalize = (typeof normalize === 'boolean') ? normalize : true;
|
|
||||||
} else {
|
|
||||||
console.error('normalize unavailable on win32 platform');
|
|
||||||
}
|
|
||||||
return this;
|
return this;
|
||||||
};
|
};
|
||||||
Sharp.prototype.normalise = Sharp.prototype.normalize;
|
Sharp.prototype.normalise = Sharp.prototype.normalize;
|
||||||
@@ -418,14 +567,10 @@ Sharp.prototype.compressionLevel = function(compressionLevel) {
|
|||||||
};
|
};
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Disable the use of adaptive row filtering for PNG output - requires libvips 7.42.0+
|
Disable the use of adaptive row filtering for PNG output
|
||||||
*/
|
*/
|
||||||
Sharp.prototype.withoutAdaptiveFiltering = function(withoutAdaptiveFiltering) {
|
Sharp.prototype.withoutAdaptiveFiltering = function(withoutAdaptiveFiltering) {
|
||||||
if (semver.gte(libvipsVersion, '7.42.0')) {
|
this.options.withoutAdaptiveFiltering = (typeof withoutAdaptiveFiltering === 'boolean') ? withoutAdaptiveFiltering : true;
|
||||||
this.options.withoutAdaptiveFiltering = (typeof withoutAdaptiveFiltering === 'boolean') ? withoutAdaptiveFiltering : true;
|
|
||||||
} else {
|
|
||||||
console.error('withoutAdaptiveFiltering requires libvips 7.41.0+');
|
|
||||||
}
|
|
||||||
return this;
|
return this;
|
||||||
};
|
};
|
||||||
|
|
||||||
@@ -438,41 +583,29 @@ Sharp.prototype.withoutChromaSubsampling = function(withoutChromaSubsampling) {
|
|||||||
};
|
};
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Apply trellis quantisation to JPEG output - requires libvips 8.0.0+ compiled against mozjpeg 3.0+
|
Apply trellis quantisation to JPEG output - requires mozjpeg 3.0+
|
||||||
*/
|
*/
|
||||||
Sharp.prototype.trellisQuantisation = function(trellisQuantisation) {
|
Sharp.prototype.trellisQuantisation = function(trellisQuantisation) {
|
||||||
if (semver.gte(libvipsVersion, '8.0.0')) {
|
this.options.trellisQuantisation = (typeof trellisQuantisation === 'boolean') ? trellisQuantisation : true;
|
||||||
this.options.trellisQuantisation = (typeof trellisQuantisation === 'boolean') ? trellisQuantisation : true;
|
|
||||||
} else {
|
|
||||||
console.error('trellisQuantisation requires libvips 8.0.0+');
|
|
||||||
}
|
|
||||||
return this;
|
return this;
|
||||||
};
|
};
|
||||||
Sharp.prototype.trellisQuantization = Sharp.prototype.trellisQuantisation;
|
Sharp.prototype.trellisQuantization = Sharp.prototype.trellisQuantisation;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Apply overshoot deringing to JPEG output - requires libvips 8.0.0+ compiled against mozjpeg 3.0+
|
Apply overshoot deringing to JPEG output - requires mozjpeg 3.0+
|
||||||
*/
|
*/
|
||||||
Sharp.prototype.overshootDeringing = function(overshootDeringing) {
|
Sharp.prototype.overshootDeringing = function(overshootDeringing) {
|
||||||
if (semver.gte(libvipsVersion, '8.0.0')) {
|
this.options.overshootDeringing = (typeof overshootDeringing === 'boolean') ? overshootDeringing : true;
|
||||||
this.options.overshootDeringing = (typeof overshootDeringing === 'boolean') ? overshootDeringing : true;
|
|
||||||
} else {
|
|
||||||
console.error('overshootDeringing requires libvips 8.0.0+');
|
|
||||||
}
|
|
||||||
return this;
|
return this;
|
||||||
};
|
};
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Optimise scans in progressive JPEG output - requires libvips 8.0.0+ compiled against mozjpeg 3.0+
|
Optimise scans in progressive JPEG output - requires mozjpeg 3.0+
|
||||||
*/
|
*/
|
||||||
Sharp.prototype.optimiseScans = function(optimiseScans) {
|
Sharp.prototype.optimiseScans = function(optimiseScans) {
|
||||||
if (semver.gte(libvipsVersion, '8.0.0')) {
|
this.options.optimiseScans = (typeof optimiseScans === 'boolean') ? optimiseScans : true;
|
||||||
this.options.optimiseScans = (typeof optimiseScans === 'boolean') ? optimiseScans : true;
|
if (this.options.optimiseScans) {
|
||||||
if (this.options.optimiseScans) {
|
this.progressive();
|
||||||
this.progressive();
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
console.error('optimiseScans requires libvips 8.0.0+');
|
|
||||||
}
|
}
|
||||||
return this;
|
return this;
|
||||||
};
|
};
|
||||||
@@ -491,12 +624,12 @@ Sharp.prototype.withMetadata = function(withMetadata) {
|
|||||||
typeof withMetadata.orientation === 'number' &&
|
typeof withMetadata.orientation === 'number' &&
|
||||||
!Number.isNaN(withMetadata.orientation) &&
|
!Number.isNaN(withMetadata.orientation) &&
|
||||||
withMetadata.orientation % 1 === 0 &&
|
withMetadata.orientation % 1 === 0 &&
|
||||||
withMetadata.orientation >=0 &&
|
withMetadata.orientation >= 1 &&
|
||||||
withMetadata.orientation <= 7
|
withMetadata.orientation <= 8
|
||||||
) {
|
) {
|
||||||
this.options.withMetadataOrientation = withMetadata.orientation;
|
this.options.withMetadataOrientation = withMetadata.orientation;
|
||||||
} else {
|
} else {
|
||||||
throw new Error('Invalid orientation (0 to 7) ' + withMetadata.orientation);
|
throw new Error('Invalid orientation (1 to 8) ' + withMetadata.orientation);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -504,27 +637,71 @@ Sharp.prototype.withMetadata = function(withMetadata) {
|
|||||||
};
|
};
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Tile size and overlap for Deep Zoom output
|
Tile-based deep zoom output options: size, overlap, layout
|
||||||
*/
|
*/
|
||||||
Sharp.prototype.tile = function(size, overlap) {
|
Sharp.prototype.tile = function(tile) {
|
||||||
// Size of square tiles, in pixels
|
if (isObject(tile)) {
|
||||||
if (typeof size !== 'undefined' && size !== null) {
|
// Size of square tiles, in pixels
|
||||||
if (!Number.isNaN(size) && size % 1 === 0 && size >= 1 && size <= 8192) {
|
if (isDefined(tile.size)) {
|
||||||
this.options.tileSize = size;
|
if (isInteger(tile.size) && inRange(tile.size, 1, 8192)) {
|
||||||
} else {
|
this.options.tileSize = tile.size;
|
||||||
throw new Error('Invalid tile size (1 to 8192) ' + size);
|
} else {
|
||||||
|
throw new Error('Invalid tile size (1 to 8192) ' + tile.size);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// Overlap of tiles, in pixels
|
||||||
|
if (isDefined(tile.overlap)) {
|
||||||
|
if (isInteger(tile.overlap) && inRange(tile.overlap, 0, 8192)) {
|
||||||
|
if (tile.overlap > this.options.tileSize) {
|
||||||
|
throw new Error('Tile overlap ' + tile.overlap + ' cannot be larger than tile size ' + this.options.tileSize);
|
||||||
|
}
|
||||||
|
this.options.tileOverlap = tile.overlap;
|
||||||
|
} else {
|
||||||
|
throw new Error('Invalid tile overlap (0 to 8192) ' + tile.overlap);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// Container
|
||||||
|
if (isDefined(tile.container)) {
|
||||||
|
if (isString(tile.container) && contains(tile.container, ['fs', 'zip'])) {
|
||||||
|
this.options.tileContainer = tile.container;
|
||||||
|
} else {
|
||||||
|
throw new Error('Invalid tile container ' + tile.container);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// Layout
|
||||||
|
if (isDefined(tile.layout)) {
|
||||||
|
if (isString(tile.layout) && contains(tile.layout, ['dz', 'google', 'zoomify'])) {
|
||||||
|
this.options.tileLayout = tile.layout;
|
||||||
|
} else {
|
||||||
|
throw new Error('Invalid tile layout ' + tile.layout);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
// Overlap of tiles, in pixels
|
return this;
|
||||||
if (typeof overlap !== 'undefined' && overlap !== null) {
|
};
|
||||||
if (!Number.isNaN(overlap) && overlap % 1 === 0 && overlap >=0 && overlap <= 8192) {
|
|
||||||
if (overlap > this.options.tileSize) {
|
/*
|
||||||
throw new Error('Tile overlap ' + overlap + ' cannot be larger than tile size ' + this.options.tileSize);
|
Extend edges
|
||||||
}
|
*/
|
||||||
this.options.tileOverlap = overlap;
|
Sharp.prototype.extend = function(extend) {
|
||||||
} else {
|
if (isInteger(extend) && extend > 0) {
|
||||||
throw new Error('Invalid tile overlap (0 to 8192) ' + overlap);
|
this.options.extendTop = extend;
|
||||||
}
|
this.options.extendBottom = extend;
|
||||||
|
this.options.extendLeft = extend;
|
||||||
|
this.options.extendRight = extend;
|
||||||
|
} else if (
|
||||||
|
isObject(extend) &&
|
||||||
|
isInteger(extend.top) && extend.top >= 0 &&
|
||||||
|
isInteger(extend.bottom) && extend.bottom >= 0 &&
|
||||||
|
isInteger(extend.left) && extend.left >= 0 &&
|
||||||
|
isInteger(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;
|
return this;
|
||||||
};
|
};
|
||||||
@@ -553,10 +730,17 @@ Sharp.prototype.resize = function(width, height) {
|
|||||||
|
|
||||||
/*
|
/*
|
||||||
Limit the total number of pixels for input images
|
Limit the total number of pixels for input images
|
||||||
Assumes the image dimensions contained in the file header can be trusted
|
Assumes the image dimensions contained in the file header can be trusted.
|
||||||
|
Alternatively can use boolean to disable or reset to default (maximum pixels)
|
||||||
*/
|
*/
|
||||||
Sharp.prototype.limitInputPixels = function(limit) {
|
Sharp.prototype.limitInputPixels = function(limit) {
|
||||||
if (typeof limit === 'number' && !Number.isNaN(limit) && limit % 1 === 0 && limit > 0) {
|
//if we pass in false we represent the integer as 0 to disable
|
||||||
|
if(limit === false) {
|
||||||
|
limit = 0;
|
||||||
|
} else if(limit === true) {
|
||||||
|
limit = maximum.pixels;
|
||||||
|
}
|
||||||
|
if (typeof limit === 'number' && !Number.isNaN(limit) && limit % 1 === 0 && limit >= 0) {
|
||||||
this.options.limitInputPixels = limit;
|
this.options.limitInputPixels = limit;
|
||||||
} else {
|
} else {
|
||||||
throw new Error('Invalid pixel limit (1 to ' + maximum.pixels + ') ' + limit);
|
throw new Error('Invalid pixel limit (1 to ' + maximum.pixels + ') ' + limit);
|
||||||
@@ -567,8 +751,8 @@ Sharp.prototype.limitInputPixels = function(limit) {
|
|||||||
/*
|
/*
|
||||||
Write output image data to a file
|
Write output image data to a file
|
||||||
*/
|
*/
|
||||||
Sharp.prototype.toFile = function(output, callback) {
|
Sharp.prototype.toFile = function(fileOut, callback) {
|
||||||
if (!output || output.length === 0) {
|
if (!fileOut || fileOut.length === 0) {
|
||||||
var errOutputInvalid = new Error('Invalid output');
|
var errOutputInvalid = new Error('Invalid output');
|
||||||
if (typeof callback === 'function') {
|
if (typeof callback === 'function') {
|
||||||
callback(errOutputInvalid);
|
callback(errOutputInvalid);
|
||||||
@@ -576,7 +760,7 @@ Sharp.prototype.toFile = function(output, callback) {
|
|||||||
return BluebirdPromise.reject(errOutputInvalid);
|
return BluebirdPromise.reject(errOutputInvalid);
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
if (this.options.fileIn === output) {
|
if (this.options.fileIn === fileOut) {
|
||||||
var errOutputIsInput = new Error('Cannot use same file for input and output');
|
var errOutputIsInput = new Error('Cannot use same file for input and output');
|
||||||
if (typeof callback === 'function') {
|
if (typeof callback === 'function') {
|
||||||
callback(errOutputIsInput);
|
callback(errOutputIsInput);
|
||||||
@@ -584,7 +768,7 @@ Sharp.prototype.toFile = function(output, callback) {
|
|||||||
return BluebirdPromise.reject(errOutputIsInput);
|
return BluebirdPromise.reject(errOutputIsInput);
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
this.options.output = output;
|
this.options.fileOut = fileOut;
|
||||||
return this._pipeline(callback);
|
return this._pipeline(callback);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -602,7 +786,7 @@ Sharp.prototype.toBuffer = function(callback) {
|
|||||||
Force JPEG output
|
Force JPEG output
|
||||||
*/
|
*/
|
||||||
Sharp.prototype.jpeg = function() {
|
Sharp.prototype.jpeg = function() {
|
||||||
this.options.output = '__jpeg';
|
this.options.formatOut = 'jpeg';
|
||||||
return this;
|
return this;
|
||||||
};
|
};
|
||||||
|
|
||||||
@@ -610,7 +794,7 @@ Sharp.prototype.jpeg = function() {
|
|||||||
Force PNG output
|
Force PNG output
|
||||||
*/
|
*/
|
||||||
Sharp.prototype.png = function() {
|
Sharp.prototype.png = function() {
|
||||||
this.options.output = '__png';
|
this.options.formatOut = 'png';
|
||||||
return this;
|
return this;
|
||||||
};
|
};
|
||||||
|
|
||||||
@@ -618,7 +802,7 @@ Sharp.prototype.png = function() {
|
|||||||
Force WebP output
|
Force WebP output
|
||||||
*/
|
*/
|
||||||
Sharp.prototype.webp = function() {
|
Sharp.prototype.webp = function() {
|
||||||
this.options.output = '__webp';
|
this.options.formatOut = 'webp';
|
||||||
return this;
|
return this;
|
||||||
};
|
};
|
||||||
|
|
||||||
@@ -626,12 +810,7 @@ Sharp.prototype.webp = function() {
|
|||||||
Force raw, uint8 output
|
Force raw, uint8 output
|
||||||
*/
|
*/
|
||||||
Sharp.prototype.raw = function() {
|
Sharp.prototype.raw = function() {
|
||||||
var supportsRawOutput = module.exports.format.raw.output;
|
this.options.formatOut = 'raw';
|
||||||
if (supportsRawOutput.file || supportsRawOutput.buffer || supportsRawOutput.stream) {
|
|
||||||
this.options.output = '__raw';
|
|
||||||
} else {
|
|
||||||
console.error('Raw output requires libvips 7.42.0+');
|
|
||||||
}
|
|
||||||
return this;
|
return this;
|
||||||
};
|
};
|
||||||
|
|
||||||
@@ -639,15 +818,17 @@ Sharp.prototype.raw = function() {
|
|||||||
Force output to a given format
|
Force output to a given format
|
||||||
@param format is either the id as a String or an Object with an 'id' attribute
|
@param format is either the id as a String or an Object with an 'id' attribute
|
||||||
*/
|
*/
|
||||||
Sharp.prototype.toFormat = function(format) {
|
Sharp.prototype.toFormat = function(formatOut) {
|
||||||
var id = format;
|
if (isObject(formatOut) && isDefined(formatOut.id)) {
|
||||||
if (typeof format === 'object') {
|
formatOut = formatOut.id;
|
||||||
id = format.id;
|
|
||||||
}
|
}
|
||||||
if (typeof id === 'string' && typeof module.exports.format[id] === 'object' && typeof this[id] === 'function') {
|
if (
|
||||||
this[id]();
|
isDefined(formatOut) &&
|
||||||
|
['jpeg', 'png', 'webp', 'raw', 'tiff', 'dz', 'input'].indexOf(formatOut) !== -1
|
||||||
|
) {
|
||||||
|
this.options.formatOut = formatOut;
|
||||||
} else {
|
} else {
|
||||||
throw new Error('Unsupported format ' + format);
|
throw new Error('Unsupported output format ' + formatOut);
|
||||||
}
|
}
|
||||||
return this;
|
return this;
|
||||||
};
|
};
|
||||||
@@ -685,10 +866,11 @@ Sharp.prototype._pipeline = function(callback) {
|
|||||||
if (this.options.streamIn) {
|
if (this.options.streamIn) {
|
||||||
// output=stream, input=stream
|
// output=stream, input=stream
|
||||||
this.on('finish', function() {
|
this.on('finish', function() {
|
||||||
sharp.pipeline(that.options, function(err, data) {
|
sharp.pipeline(that.options, function(err, data, info) {
|
||||||
if (err) {
|
if (err) {
|
||||||
that.emit('error', err);
|
that.emit('error', err);
|
||||||
} else {
|
} else {
|
||||||
|
that.emit('info', info);
|
||||||
that.push(data);
|
that.push(data);
|
||||||
}
|
}
|
||||||
that.push(null);
|
that.push(null);
|
||||||
@@ -696,10 +878,11 @@ Sharp.prototype._pipeline = function(callback) {
|
|||||||
});
|
});
|
||||||
} else {
|
} else {
|
||||||
// output=stream, input=file/buffer
|
// output=stream, input=file/buffer
|
||||||
sharp.pipeline(this.options, function(err, data) {
|
sharp.pipeline(this.options, function(err, data, info) {
|
||||||
if (err) {
|
if (err) {
|
||||||
that.emit('error', err);
|
that.emit('error', err);
|
||||||
} else {
|
} else {
|
||||||
|
that.emit('info', info);
|
||||||
that.push(data);
|
that.push(data);
|
||||||
}
|
}
|
||||||
that.push(null);
|
that.push(null);
|
||||||
@@ -786,7 +969,6 @@ Sharp.prototype.clone = function() {
|
|||||||
// Clone existing options
|
// Clone existing options
|
||||||
var clone = new Sharp();
|
var clone = new Sharp();
|
||||||
util._extend(clone.options, this.options);
|
util._extend(clone.options, this.options);
|
||||||
clone.streamIn = false;
|
|
||||||
// Pass 'finish' event to clone for Stream-based input
|
// Pass 'finish' event to clone for Stream-based input
|
||||||
this.on('finish', function() {
|
this.on('finish', function() {
|
||||||
// Clone inherits input data
|
// Clone inherits input data
|
||||||
@@ -796,18 +978,25 @@ Sharp.prototype.clone = function() {
|
|||||||
return clone;
|
return clone;
|
||||||
};
|
};
|
||||||
|
|
||||||
/*
|
/**
|
||||||
Get and set cache memory and item limits
|
Get and set cache memory, file and item limits
|
||||||
*/
|
*/
|
||||||
module.exports.cache = function(memory, items) {
|
module.exports.cache = function(options) {
|
||||||
if (typeof memory !== 'number' || Number.isNaN(memory)) {
|
if (typeof options === 'boolean') {
|
||||||
memory = null;
|
if (options) {
|
||||||
|
// Default cache settings of 50MB, 20 files, 100 items
|
||||||
|
return sharp.cache(50, 20, 100);
|
||||||
|
} else {
|
||||||
|
return sharp.cache(0, 0, 0);
|
||||||
|
}
|
||||||
|
} else if (typeof options === 'object') {
|
||||||
|
return sharp.cache(options.memory, options.files, options.items);
|
||||||
|
} else {
|
||||||
|
return sharp.cache();
|
||||||
}
|
}
|
||||||
if (typeof items !== 'number' || Number.isNaN(items)) {
|
|
||||||
items = null;
|
|
||||||
}
|
|
||||||
return sharp.cache(memory, items);
|
|
||||||
};
|
};
|
||||||
|
// Ensure default cache settings are set
|
||||||
|
module.exports.cache(true);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Get and set size of thread pool
|
Get and set size of thread pool
|
||||||
@@ -827,8 +1016,13 @@ module.exports.counters = function() {
|
|||||||
};
|
};
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Get the version of the libvips library
|
Get and set use of SIMD vector unit instructions
|
||||||
*/
|
*/
|
||||||
module.exports.libvipsVersion = function() {
|
module.exports.simd = function(simd) {
|
||||||
return libvipsVersion;
|
if (typeof simd !== 'boolean') {
|
||||||
|
simd = null;
|
||||||
|
}
|
||||||
|
return sharp.simd(simd);
|
||||||
};
|
};
|
||||||
|
// Switch off default
|
||||||
|
module.exports.simd(false);
|
||||||
|
|||||||
51
package.json
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "sharp",
|
"name": "sharp",
|
||||||
"version": "0.11.1",
|
"version": "0.14.1",
|
||||||
"author": "Lovell Fuller <npm@lovell.info>",
|
"author": "Lovell Fuller <npm@lovell.info>",
|
||||||
"contributors": [
|
"contributors": [
|
||||||
"Pierre Inglebert <pierre.inglebert@gmail.com>",
|
"Pierre Inglebert <pierre.inglebert@gmail.com>",
|
||||||
@@ -16,16 +16,22 @@
|
|||||||
"Linus Unnebäck <linus@folkdatorn.se>",
|
"Linus Unnebäck <linus@folkdatorn.se>",
|
||||||
"Victor Mateevitsi <mvictoras@gmail.com>",
|
"Victor Mateevitsi <mvictoras@gmail.com>",
|
||||||
"Alaric Holloway <alaric.holloway@gmail.com>",
|
"Alaric Holloway <alaric.holloway@gmail.com>",
|
||||||
"Bernhard K. Weisshuhn <bkw@codingforce.com>"
|
"Bernhard K. Weisshuhn <bkw@codingforce.com>",
|
||||||
|
"Chris Riley <criley@primedia.com>",
|
||||||
|
"David Carley <dacarley@gmail.com>",
|
||||||
|
"John Tobin <john@limelightmobileinc.com>",
|
||||||
|
"Kenton Gray <kentongray@gmail.com>",
|
||||||
|
"Felix Bünemann <Felix.Buenemann@gmail.com>",
|
||||||
|
"Samy Al Zahrani <samyalzahrany@gmail.com>"
|
||||||
],
|
],
|
||||||
"description": "High performance Node.js module to resize JPEG, PNG, WebP and TIFF images using the libvips library",
|
"description": "High performance Node.js module to resize JPEG, PNG, WebP and TIFF images using the libvips library",
|
||||||
"scripts": {
|
"scripts": {
|
||||||
"clean": "rm -rf test/fixtures/output.*",
|
"clean": "rm -rf node_modules/ build/ include/ lib/ coverage/ test/fixtures/output.*",
|
||||||
"test": "VIPS_WARNING=0 node ./node_modules/istanbul/lib/cli.js cover ./node_modules/mocha/bin/_mocha -- --slow=5000 --timeout=20000 ./test/unit/*.js",
|
"test": "VIPS_WARNING=0 node ./node_modules/istanbul/lib/cli.js cover ./node_modules/mocha/bin/_mocha -- --slow=5000 --timeout=30000 ./test/unit/*.js",
|
||||||
"test-clean": "npm run clean && npm install && npm test",
|
"test-win": "node ./node_modules/mocha/bin/mocha --slow=5000 --timeout=60000 ./test/unit/*.js",
|
||||||
"test-leak": "cd test/leak; ./leak.sh; cd - > /dev/null",
|
"test-leak": "./test/leak/leak.sh",
|
||||||
"test-win32-node": "node ./node_modules/mocha/bin/mocha --slow=5000 --timeout=30000 ./test/unit/*.js",
|
"test-packaging": "./packaging/test.sh",
|
||||||
"test-win32-iojs": "iojs ./node_modules/mocha/bin/mocha --slow=5000 --timeout=30000 ./test/unit/*.js"
|
"test-clean": "rm -rf coverage/ test/fixtures/output.* && npm install && npm test"
|
||||||
},
|
},
|
||||||
"main": "index.js",
|
"main": "index.js",
|
||||||
"repository": {
|
"repository": {
|
||||||
@@ -45,23 +51,30 @@
|
|||||||
"vips"
|
"vips"
|
||||||
],
|
],
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"bluebird": "^2.9.33",
|
"bluebird": "^3.3.5",
|
||||||
"color": "^0.10.1",
|
"color": "^0.11.1",
|
||||||
"nan": "^1.8.4",
|
"nan": "^2.2.1",
|
||||||
"semver": "^5.0.1"
|
"semver": "^5.1.0",
|
||||||
|
"request": "^2.71.0",
|
||||||
|
"tar": "^2.2.1"
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"async": "^1.4.2",
|
"async": "^1.5.2",
|
||||||
"coveralls": "^2.11.2",
|
"bufferutil": "^1.2.1",
|
||||||
"exif-reader": "1.0.0",
|
"coveralls": "^2.11.9",
|
||||||
|
"exif-reader": "^1.0.0",
|
||||||
"icc": "^0.0.2",
|
"icc": "^0.0.2",
|
||||||
"istanbul": "^0.3.17",
|
"istanbul": "^0.4.3",
|
||||||
"mocha": "^2.2.5",
|
"mocha": "^2.4.5",
|
||||||
"mocha-jshint": "^2.2.3",
|
"mocha-jshint": "^2.3.1",
|
||||||
"node-cpplint": "^0.4.0",
|
"node-cpplint": "^0.4.0",
|
||||||
"rimraf": "^2.4.1"
|
"rimraf": "^2.5.2",
|
||||||
|
"unzip": "^0.1.11"
|
||||||
},
|
},
|
||||||
"license": "Apache-2.0",
|
"license": "Apache-2.0",
|
||||||
|
"config": {
|
||||||
|
"libvips": "8.2.3"
|
||||||
|
},
|
||||||
"engines": {
|
"engines": {
|
||||||
"node": ">=0.10"
|
"node": ">=0.10"
|
||||||
}
|
}
|
||||||
|
|||||||
34
packaging/arm-build.sh
Executable file
@@ -0,0 +1,34 @@
|
|||||||
|
#!/bin/sh
|
||||||
|
|
||||||
|
if [ $# -lt 1 ]; then
|
||||||
|
echo "Usage: $0 IP"
|
||||||
|
echo "Build libvips for ARM using Docker, where IP is"
|
||||||
|
echo "the address of a Raspberry Pi running HypriotOS"
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
IP="$1"
|
||||||
|
|
||||||
|
echo "Verifying connectivity to $IP"
|
||||||
|
if ! ping -c 1 $IP; then
|
||||||
|
echo "Could not connect to $IP"
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
if ! type sshpass >/dev/null; then
|
||||||
|
echo "Please install sshpass"
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
export SSHPASS=hypriot
|
||||||
|
|
||||||
|
echo "Copying arm/Dockerfile and arm/build.sh to device"
|
||||||
|
sshpass -e scp -o PreferredAuthentications=password -r arm root@${IP}:/root
|
||||||
|
|
||||||
|
echo "Building Raspbian-based container"
|
||||||
|
sshpass -e ssh -o PreferredAuthentications=password -t root@${IP} "docker build -t vips-dev-arm arm"
|
||||||
|
|
||||||
|
echo "Running arm/build.sh within container"
|
||||||
|
sshpass -e ssh -o PreferredAuthentications=password -t root@${IP} "docker run -i -t --rm -v \${PWD}/arm:/arm vips-dev-arm sh -c 'cd /arm && ./build.sh' | tee arm/build.log"
|
||||||
|
|
||||||
|
echo "Copying resultant tar.gz file from device"
|
||||||
|
sshpass -e scp -o PreferredAuthentications=password root@${IP}:/root/arm/*.tar.gz .
|
||||||
28
packaging/arm-test.sh
Executable file
@@ -0,0 +1,28 @@
|
|||||||
|
#!/bin/sh
|
||||||
|
|
||||||
|
if [ $# -lt 1 ]; then
|
||||||
|
echo "Usage: $0 IP"
|
||||||
|
echo "Test sharp on ARM using Docker, where IP is"
|
||||||
|
echo "the address of a Raspberry Pi running HypriotOS"
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
IP="$1"
|
||||||
|
|
||||||
|
echo "Verifying connectivity to $IP"
|
||||||
|
if ! ping -c 1 $IP; then
|
||||||
|
echo "Could not connect to $IP"
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
if ! type sshpass >/dev/null; then
|
||||||
|
echo "Please install sshpass"
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
export SSHPASS=hypriot
|
||||||
|
|
||||||
|
echo "Copying sharp source to device"
|
||||||
|
sshpass -e scp -o PreferredAuthentications=password -r ../../sharp root@${IP}:/root/sharp
|
||||||
|
|
||||||
|
echo "Compile and test within container"
|
||||||
|
sshpass -e ssh -o PreferredAuthentications=password -t root@${IP} "docker run -i -t --rm -v \${PWD}/sharp:/s hypriot/rpi-node:5 sh -c 'cd /s && npm install --unsafe-perm && npm test'"
|
||||||
5
packaging/arm/Dockerfile
Normal file
@@ -0,0 +1,5 @@
|
|||||||
|
FROM resin/rpi-raspbian:jessie
|
||||||
|
MAINTAINER Lovell Fuller <npm@lovell.info>
|
||||||
|
|
||||||
|
# Build dependencies
|
||||||
|
RUN apt-get update && apt-get install -y build-essential autoconf libtool nasm gtk-doc-tools texinfo curl
|
||||||
137
packaging/arm/build.sh
Executable file
@@ -0,0 +1,137 @@
|
|||||||
|
#!/bin/sh
|
||||||
|
|
||||||
|
# To be run inside a Raspbian container
|
||||||
|
|
||||||
|
# Working directories
|
||||||
|
DEPS=/deps
|
||||||
|
TARGET=/target
|
||||||
|
mkdir ${DEPS}
|
||||||
|
mkdir ${TARGET}
|
||||||
|
|
||||||
|
# Common build paths and flags
|
||||||
|
export PKG_CONFIG_PATH="${PKG_CONFIG_PATH}:${TARGET}/lib/pkgconfig"
|
||||||
|
export PATH="${PATH}:${TARGET}/bin"
|
||||||
|
export CPPFLAGS="-I${TARGET}/include"
|
||||||
|
export LDFLAGS="-L${TARGET}/lib"
|
||||||
|
export CFLAGS="-O3"
|
||||||
|
export CXXFLAGS="-O3"
|
||||||
|
|
||||||
|
# Dependency version numbers
|
||||||
|
VERSION_ZLIB=1.2.8
|
||||||
|
VERSION_FFI=3.2.1
|
||||||
|
VERSION_GLIB=2.47.6
|
||||||
|
VERSION_XML2=2.9.3
|
||||||
|
VERSION_GSF=1.14.34
|
||||||
|
VERSION_EXIF=0.6.21
|
||||||
|
VERSION_LCMS2=2.7
|
||||||
|
VERSION_GM=1.3.23
|
||||||
|
VERSION_JPEG=1.4.2
|
||||||
|
VERSION_PNG16=1.6.21
|
||||||
|
VERSION_WEBP=0.5.0
|
||||||
|
VERSION_TIFF=4.0.6
|
||||||
|
VERSION_ORC=0.4.25
|
||||||
|
VERSION_VIPS=8.2.3
|
||||||
|
|
||||||
|
mkdir ${DEPS}/zlib
|
||||||
|
curl -Ls http://zlib.net/zlib-${VERSION_ZLIB}.tar.xz | tar xJC ${DEPS}/zlib --strip-components=1
|
||||||
|
cd ${DEPS}/zlib
|
||||||
|
./configure --prefix=${TARGET} && make install
|
||||||
|
rm ${TARGET}/lib/libz.a
|
||||||
|
|
||||||
|
mkdir ${DEPS}/ffi
|
||||||
|
curl -Ls ftp://sourceware.org/pub/libffi/libffi-${VERSION_FFI}.tar.gz | tar xzC ${DEPS}/ffi --strip-components=1
|
||||||
|
cd ${DEPS}/ffi
|
||||||
|
./configure --prefix=${TARGET} --enable-shared --disable-static --disable-dependency-tracking --disable-builddir && make install-strip
|
||||||
|
|
||||||
|
mkdir ${DEPS}/glib
|
||||||
|
curl -Ls https://download.gnome.org/sources/glib/2.47/glib-${VERSION_GLIB}.tar.xz | tar xJC ${DEPS}/glib --strip-components=1
|
||||||
|
cd ${DEPS}/glib
|
||||||
|
./configure --prefix=${TARGET} --enable-shared --disable-static --disable-dependency-tracking --with-pcre=internal && make install-strip
|
||||||
|
|
||||||
|
mkdir ${DEPS}/xml2
|
||||||
|
curl -Ls http://xmlsoft.org/sources/libxml2-${VERSION_XML2}.tar.gz | tar xzC ${DEPS}/xml2 --strip-components=1
|
||||||
|
cd ${DEPS}/xml2
|
||||||
|
./configure --prefix=${TARGET} --enable-shared --disable-static --disable-dependency-tracking --without-python --with-zlib=${TARGET} && make install-strip
|
||||||
|
|
||||||
|
mkdir ${DEPS}/gsf
|
||||||
|
curl -Ls https://download.gnome.org/sources/libgsf/1.14/libgsf-${VERSION_GSF}.tar.xz | tar xJC ${DEPS}/gsf --strip-components=1
|
||||||
|
cd ${DEPS}/gsf
|
||||||
|
./configure --prefix=${TARGET} --enable-shared --disable-static --disable-dependency-tracking && make install-strip
|
||||||
|
|
||||||
|
mkdir ${DEPS}/exif
|
||||||
|
curl -Ls http://heanet.dl.sourceforge.net/project/libexif/libexif/${VERSION_EXIF}/libexif-${VERSION_EXIF}.tar.bz2 | tar xjC ${DEPS}/exif --strip-components=1
|
||||||
|
cd ${DEPS}/exif
|
||||||
|
./configure --prefix=${TARGET} --enable-shared --disable-static --disable-dependency-tracking && make install-strip
|
||||||
|
|
||||||
|
mkdir ${DEPS}/lcms2
|
||||||
|
curl -Ls http://heanet.dl.sourceforge.net/project/lcms/lcms/${VERSION_LCMS2}/lcms2-${VERSION_LCMS2}.tar.gz | tar xzC ${DEPS}/lcms2 --strip-components=1
|
||||||
|
cd ${DEPS}/lcms2
|
||||||
|
./configure --prefix=${TARGET} --enable-shared --disable-static --disable-dependency-tracking && make install-strip
|
||||||
|
|
||||||
|
mkdir ${DEPS}/gm
|
||||||
|
curl -Ls http://heanet.dl.sourceforge.net/project/graphicsmagick/graphicsmagick/${VERSION_GM}/GraphicsMagick-${VERSION_GM}.tar.xz | tar xJC ${DEPS}/gm --strip-components=1
|
||||||
|
cd ${DEPS}/gm
|
||||||
|
./configure --prefix=${TARGET} --enable-shared --disable-static --disable-dependency-tracking --without-magick-plus-plus && make install-strip
|
||||||
|
|
||||||
|
mkdir ${DEPS}/jpeg
|
||||||
|
curl -Ls http://heanet.dl.sourceforge.net/project/libjpeg-turbo/${VERSION_JPEG}/libjpeg-turbo-${VERSION_JPEG}.tar.gz | tar xzC ${DEPS}/jpeg --strip-components=1
|
||||||
|
cd ${DEPS}/jpeg
|
||||||
|
./configure --prefix=${TARGET} --enable-shared --disable-static --disable-dependency-tracking --with-jpeg8 --without-turbojpeg && make install-strip
|
||||||
|
|
||||||
|
mkdir ${DEPS}/png16
|
||||||
|
curl -Ls http://heanet.dl.sourceforge.net/project/libpng/libpng16/${VERSION_PNG16}/libpng-${VERSION_PNG16}.tar.xz | tar xJC ${DEPS}/png16 --strip-components=1
|
||||||
|
cd ${DEPS}/png16
|
||||||
|
./configure --prefix=${TARGET} --enable-shared --disable-static --disable-dependency-tracking && make install-strip
|
||||||
|
|
||||||
|
mkdir ${DEPS}/webp
|
||||||
|
curl -Ls http://downloads.webmproject.org/releases/webp/libwebp-${VERSION_WEBP}.tar.gz | tar xzC ${DEPS}/webp --strip-components=1
|
||||||
|
cd ${DEPS}/webp
|
||||||
|
./configure --prefix=${TARGET} --enable-shared --disable-static --disable-dependency-tracking && make install-strip
|
||||||
|
|
||||||
|
mkdir ${DEPS}/tiff
|
||||||
|
curl -Ls http://download.osgeo.org/libtiff/tiff-${VERSION_TIFF}.tar.gz | tar xzC ${DEPS}/tiff --strip-components=1
|
||||||
|
cd ${DEPS}/tiff
|
||||||
|
./configure --prefix=${TARGET} --enable-shared --disable-static --disable-dependency-tracking && make install-strip
|
||||||
|
rm ${TARGET}/lib/libtiffxx*
|
||||||
|
|
||||||
|
mkdir ${DEPS}/orc
|
||||||
|
curl -Ls http://gstreamer.freedesktop.org/data/src/orc/orc-${VERSION_ORC}.tar.xz | tar xJC ${DEPS}/orc --strip-components=1
|
||||||
|
cd ${DEPS}/orc
|
||||||
|
./configure --prefix=${TARGET} --enable-shared --disable-static --disable-dependency-tracking && make install-strip
|
||||||
|
|
||||||
|
mkdir ${DEPS}/vips
|
||||||
|
curl -Ls http://www.vips.ecs.soton.ac.uk/supported/8.2/vips-${VERSION_VIPS}.tar.gz | tar xzC ${DEPS}/vips --strip-components=1
|
||||||
|
cd ${DEPS}/vips
|
||||||
|
./configure --prefix=${TARGET} --enable-shared --disable-static --disable-dependency-tracking \
|
||||||
|
--disable-debug --disable-introspection --without-python --without-fftw --with-magickpackage=GraphicsMagick \
|
||||||
|
--with-zip-includes=${TARGET}/include --with-zip-libraries=${TARGET}/lib \
|
||||||
|
--with-jpeg-includes=${TARGET}/include --with-jpeg-libraries=${TARGET}/lib \
|
||||||
|
&& make install-strip
|
||||||
|
|
||||||
|
# Remove the old C++ bindings
|
||||||
|
cd ${TARGET}/include
|
||||||
|
rm -rf vips/vipsc++.h vips/vipscpp.h
|
||||||
|
cd ${TARGET}/lib
|
||||||
|
rm -rf pkgconfig .libs *.la libvipsCC*
|
||||||
|
|
||||||
|
# Create JSON file of version numbers
|
||||||
|
cd ${TARGET}
|
||||||
|
echo "{\n\
|
||||||
|
\"exif\": \"${VERSION_EXIF}\",\n\
|
||||||
|
\"ffi\": \"${VERSION_FFI}\",\n\
|
||||||
|
\"glib\": \"${VERSION_GLIB}\",\n\
|
||||||
|
\"gsf\": \"${VERSION_GSF}\",\n\
|
||||||
|
\"jpeg\": \"${VERSION_JPEG}\",\n\
|
||||||
|
\"lcms\": \"${VERSION_LCMS2}\",\n\
|
||||||
|
\"gm\": \"${VERSION_GM}\",\n\
|
||||||
|
\"orc\": \"${VERSION_ORC}\",\n\
|
||||||
|
\"png\": \"${VERSION_PNG16}\",\n\
|
||||||
|
\"tiff\": \"${VERSION_TIFF}\",\n\
|
||||||
|
\"vips\": \"${VERSION_VIPS}\"\n\
|
||||||
|
\"webp\": \"${VERSION_WEBP}\",\n\
|
||||||
|
\"xml\": \"${VERSION_XML2}\",\n\
|
||||||
|
\"zlib\": \"${VERSION_ZLIB}\",\n\
|
||||||
|
}" >lib/versions.json
|
||||||
|
|
||||||
|
# Create .tar.gz
|
||||||
|
GZIP=-9 tar czf /arm/libvips-${VERSION_VIPS}-arm.tar.gz include lib
|
||||||
30
packaging/build.sh
Executable file
@@ -0,0 +1,30 @@
|
|||||||
|
#!/bin/sh
|
||||||
|
|
||||||
|
VERSION_VIPS=8.2.3
|
||||||
|
|
||||||
|
# Is docker available?
|
||||||
|
|
||||||
|
if ! type docker >/dev/null; then
|
||||||
|
echo "Please install docker"
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
# TODO: docker v1.9.0 will allow build-time args - https://github.com/docker/docker/pull/15182
|
||||||
|
|
||||||
|
# Windows
|
||||||
|
|
||||||
|
docker build -t vips-dev-win win
|
||||||
|
WIN_CONTAINER_ID=$(docker run -d vips-dev-win)
|
||||||
|
docker cp "${WIN_CONTAINER_ID}:/libvips-${VERSION_VIPS}-win.tar.gz" .
|
||||||
|
docker rm "${WIN_CONTAINER_ID}"
|
||||||
|
|
||||||
|
# Linux
|
||||||
|
|
||||||
|
docker build -t vips-dev-lin lin
|
||||||
|
LIN_CONTAINER_ID=$(docker run -d vips-dev-lin)
|
||||||
|
docker cp "${LIN_CONTAINER_ID}:/libvips-${VERSION_VIPS}-lin.tar.gz" .
|
||||||
|
docker rm "${LIN_CONTAINER_ID}"
|
||||||
|
|
||||||
|
# Checksums
|
||||||
|
|
||||||
|
sha256sum *.tar.gz
|
||||||
139
packaging/lin/Dockerfile
Normal file
@@ -0,0 +1,139 @@
|
|||||||
|
FROM debian:wheezy
|
||||||
|
MAINTAINER Lovell Fuller <npm@lovell.info>
|
||||||
|
|
||||||
|
# Build dependencies
|
||||||
|
RUN apt-get update && apt-get install -y build-essential autoconf libtool nasm gtk-doc-tools texinfo
|
||||||
|
|
||||||
|
# Create working directories
|
||||||
|
ENV DEPS=/deps \
|
||||||
|
TARGET=/target
|
||||||
|
RUN mkdir ${DEPS} && mkdir ${TARGET}
|
||||||
|
|
||||||
|
# Common build paths and flags
|
||||||
|
ENV PKG_CONFIG_PATH=${PKG_CONFIG_PATH}:${TARGET}/lib/pkgconfig \
|
||||||
|
PATH=${PATH}:${TARGET}/bin \
|
||||||
|
CPPFLAGS=-I${TARGET}/include \
|
||||||
|
LDFLAGS=-L${TARGET}/lib \
|
||||||
|
CFLAGS="-O3" \
|
||||||
|
CXXFLAGS="-O3"
|
||||||
|
|
||||||
|
# Dependency version numbers
|
||||||
|
ENV VERSION_ZLIB=1.2.8 \
|
||||||
|
VERSION_FFI=3.2.1 \
|
||||||
|
VERSION_GLIB=2.47.6 \
|
||||||
|
VERSION_XML2=2.9.3 \
|
||||||
|
VERSION_GSF=1.14.34 \
|
||||||
|
VERSION_EXIF=0.6.21 \
|
||||||
|
VERSION_LCMS2=2.7 \
|
||||||
|
VERSION_GM=1.3.23 \
|
||||||
|
VERSION_JPEG=1.4.2 \
|
||||||
|
VERSION_PNG16=1.6.21 \
|
||||||
|
VERSION_WEBP=0.5.0 \
|
||||||
|
VERSION_TIFF=4.0.6 \
|
||||||
|
VERSION_ORC=0.4.25 \
|
||||||
|
VERSION_VIPS=8.2.3
|
||||||
|
|
||||||
|
RUN mkdir ${DEPS}/zlib
|
||||||
|
RUN curl -Ls http://zlib.net/zlib-${VERSION_ZLIB}.tar.xz | tar xJC ${DEPS}/zlib --strip-components=1
|
||||||
|
WORKDIR ${DEPS}/zlib
|
||||||
|
RUN ./configure --prefix=${TARGET} && make install
|
||||||
|
RUN rm ${TARGET}/lib/libz.a
|
||||||
|
|
||||||
|
RUN mkdir ${DEPS}/ffi
|
||||||
|
RUN curl -Ls ftp://sourceware.org/pub/libffi/libffi-${VERSION_FFI}.tar.gz | tar xzC ${DEPS}/ffi --strip-components=1
|
||||||
|
WORKDIR ${DEPS}/ffi
|
||||||
|
RUN ./configure --prefix=${TARGET} --enable-shared --disable-static --disable-dependency-tracking --disable-builddir && make install-strip
|
||||||
|
|
||||||
|
RUN mkdir ${DEPS}/glib
|
||||||
|
RUN curl -Ls https://download.gnome.org/sources/glib/2.47/glib-${VERSION_GLIB}.tar.xz | tar xJC ${DEPS}/glib --strip-components=1
|
||||||
|
WORKDIR ${DEPS}/glib
|
||||||
|
RUN ./configure --prefix=${TARGET} --enable-shared --disable-static --disable-dependency-tracking --with-pcre=internal && make install-strip
|
||||||
|
|
||||||
|
RUN mkdir ${DEPS}/xml2
|
||||||
|
RUN curl -Ls http://xmlsoft.org/sources/libxml2-${VERSION_XML2}.tar.gz | tar xzC ${DEPS}/xml2 --strip-components=1
|
||||||
|
WORKDIR ${DEPS}/xml2
|
||||||
|
RUN ./configure --prefix=${TARGET} --enable-shared --disable-static --disable-dependency-tracking --without-python --with-zlib=${TARGET} && make install-strip
|
||||||
|
|
||||||
|
RUN mkdir ${DEPS}/gsf
|
||||||
|
RUN curl -Ls https://download.gnome.org/sources/libgsf/1.14/libgsf-${VERSION_GSF}.tar.xz | tar xJC ${DEPS}/gsf --strip-components=1
|
||||||
|
WORKDIR ${DEPS}/gsf
|
||||||
|
RUN ./configure --prefix=${TARGET} --enable-shared --disable-static --disable-dependency-tracking && make install-strip
|
||||||
|
|
||||||
|
RUN mkdir ${DEPS}/exif
|
||||||
|
RUN curl -Ls http://heanet.dl.sourceforge.net/project/libexif/libexif/${VERSION_EXIF}/libexif-${VERSION_EXIF}.tar.bz2 | tar xjC ${DEPS}/exif --strip-components=1
|
||||||
|
WORKDIR ${DEPS}/exif
|
||||||
|
RUN ./configure --prefix=${TARGET} --enable-shared --disable-static --disable-dependency-tracking && make install-strip
|
||||||
|
|
||||||
|
RUN mkdir ${DEPS}/lcms2
|
||||||
|
RUN curl -Ls http://heanet.dl.sourceforge.net/project/lcms/lcms/${VERSION_LCMS2}/lcms2-${VERSION_LCMS2}.tar.gz | tar xzC ${DEPS}/lcms2 --strip-components=1
|
||||||
|
WORKDIR ${DEPS}/lcms2
|
||||||
|
RUN ./configure --prefix=${TARGET} --enable-shared --disable-static --disable-dependency-tracking && make install-strip
|
||||||
|
|
||||||
|
RUN mkdir ${DEPS}/gm
|
||||||
|
RUN curl -Ls http://heanet.dl.sourceforge.net/project/graphicsmagick/graphicsmagick/${VERSION_GM}/GraphicsMagick-${VERSION_GM}.tar.xz | tar xJC ${DEPS}/gm --strip-components=1
|
||||||
|
WORKDIR ${DEPS}/gm
|
||||||
|
RUN ./configure --prefix=${TARGET} --enable-shared --disable-static --disable-dependency-tracking --without-magick-plus-plus && make install-strip
|
||||||
|
|
||||||
|
RUN mkdir ${DEPS}/jpeg
|
||||||
|
RUN curl -Ls http://heanet.dl.sourceforge.net/project/libjpeg-turbo/${VERSION_JPEG}/libjpeg-turbo-${VERSION_JPEG}.tar.gz | tar xzC ${DEPS}/jpeg --strip-components=1
|
||||||
|
WORKDIR ${DEPS}/jpeg
|
||||||
|
RUN ./configure --prefix=${TARGET} --enable-shared --disable-static --disable-dependency-tracking --with-jpeg8 --without-turbojpeg && make install-strip
|
||||||
|
|
||||||
|
RUN mkdir ${DEPS}/png16
|
||||||
|
RUN curl -Ls http://heanet.dl.sourceforge.net/project/libpng/libpng16/${VERSION_PNG16}/libpng-${VERSION_PNG16}.tar.xz | tar xJC ${DEPS}/png16 --strip-components=1
|
||||||
|
WORKDIR ${DEPS}/png16
|
||||||
|
RUN ./configure --prefix=${TARGET} --enable-shared --disable-static --disable-dependency-tracking && make install-strip
|
||||||
|
|
||||||
|
RUN mkdir ${DEPS}/webp
|
||||||
|
RUN curl -Ls http://downloads.webmproject.org/releases/webp/libwebp-${VERSION_WEBP}.tar.gz | tar xzC ${DEPS}/webp --strip-components=1
|
||||||
|
WORKDIR ${DEPS}/webp
|
||||||
|
RUN ./configure --prefix=${TARGET} --enable-shared --disable-static --disable-dependency-tracking && make install-strip
|
||||||
|
|
||||||
|
RUN mkdir ${DEPS}/tiff
|
||||||
|
RUN curl -Ls http://download.osgeo.org/libtiff/tiff-${VERSION_TIFF}.tar.gz | tar xzC ${DEPS}/tiff --strip-components=1
|
||||||
|
WORKDIR ${DEPS}/tiff
|
||||||
|
RUN ./configure --prefix=${TARGET} --enable-shared --disable-static --disable-dependency-tracking && make install-strip
|
||||||
|
RUN rm ${TARGET}/lib/libtiffxx*
|
||||||
|
|
||||||
|
RUN mkdir ${DEPS}/orc
|
||||||
|
RUN curl -Ls http://gstreamer.freedesktop.org/data/src/orc/orc-${VERSION_ORC}.tar.xz | tar xJC ${DEPS}/orc --strip-components=1
|
||||||
|
WORKDIR ${DEPS}/orc
|
||||||
|
RUN ./configure --prefix=${TARGET} --enable-shared --disable-static --disable-dependency-tracking && make install-strip
|
||||||
|
|
||||||
|
RUN mkdir ${DEPS}/vips
|
||||||
|
RUN curl -Ls http://www.vips.ecs.soton.ac.uk/supported/8.2/vips-${VERSION_VIPS}.tar.gz | tar xzC ${DEPS}/vips --strip-components=1
|
||||||
|
WORKDIR ${DEPS}/vips
|
||||||
|
RUN ./configure --prefix=${TARGET} --enable-shared --disable-static --disable-dependency-tracking \
|
||||||
|
--disable-debug --disable-introspection --without-python --without-fftw --with-magickpackage=GraphicsMagick \
|
||||||
|
--with-zip-includes=${TARGET}/include --with-zip-libraries=${TARGET}/lib \
|
||||||
|
--with-jpeg-includes=${TARGET}/include --with-jpeg-libraries=${TARGET}/lib \
|
||||||
|
&& make install-strip
|
||||||
|
|
||||||
|
# Remove the old C++ bindings
|
||||||
|
WORKDIR ${TARGET}/include
|
||||||
|
RUN rm -rf vips/vipsc++.h vips/vipscpp.h
|
||||||
|
WORKDIR ${TARGET}/lib
|
||||||
|
RUN rm -rf pkgconfig .libs *.la libvipsCC*
|
||||||
|
|
||||||
|
# Create JSON file of version numbers
|
||||||
|
WORKDIR ${TARGET}
|
||||||
|
RUN echo "{\n\
|
||||||
|
\"exif\": \"${VERSION_EXIF}\",\n\
|
||||||
|
\"ffi\": \"${VERSION_FFI}\",\n\
|
||||||
|
\"glib\": \"${VERSION_GLIB}\",\n\
|
||||||
|
\"gsf\": \"${VERSION_GSF}\",\n\
|
||||||
|
\"jpeg\": \"${VERSION_JPEG}\",\n\
|
||||||
|
\"lcms\": \"${VERSION_LCMS2}\",\n\
|
||||||
|
\"gm\": \"${VERSION_GM}\",\n\
|
||||||
|
\"orc\": \"${VERSION_ORC}\",\n\
|
||||||
|
\"png\": \"${VERSION_PNG16}\",\n\
|
||||||
|
\"tiff\": \"${VERSION_TIFF}\",\n\
|
||||||
|
\"vips\": \"${VERSION_VIPS}\"\n\
|
||||||
|
\"webp\": \"${VERSION_WEBP}\",\n\
|
||||||
|
\"xml\": \"${VERSION_XML2}\",\n\
|
||||||
|
\"zlib\": \"${VERSION_ZLIB}\",\n\
|
||||||
|
}" >lib/versions.json
|
||||||
|
|
||||||
|
# Create .tar.gz
|
||||||
|
WORKDIR ${TARGET}
|
||||||
|
RUN GZIP=-9 tar czf /libvips-${VERSION_VIPS}-lin.tar.gz include lib
|
||||||
57
packaging/test.sh
Executable file
@@ -0,0 +1,57 @@
|
|||||||
|
#!/bin/sh
|
||||||
|
|
||||||
|
# Verify docker is available
|
||||||
|
if ! type docker >/dev/null; then
|
||||||
|
echo "Please install docker"
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
test="npm run clean; npm install --unsafe-perm; npm test"
|
||||||
|
|
||||||
|
# Debian 7, 8
|
||||||
|
# Ubuntu 12.04, 14.04
|
||||||
|
for dist in wheezy jessie precise trusty; do
|
||||||
|
echo "Testing $dist..."
|
||||||
|
if docker run -i -t --rm -v $PWD:/v -e "NODE_ENV=development" nodesource/$dist:0.12 >packaging/$dist.log 2>&1 sh -c "cd /v; ./packaging/test/debian.sh; $test";
|
||||||
|
then echo "$dist OK"
|
||||||
|
else echo "$dist fail" && cat packaging/$dist.log
|
||||||
|
fi
|
||||||
|
done
|
||||||
|
|
||||||
|
# Centos 6
|
||||||
|
echo "Testing centos6..."
|
||||||
|
if docker run -i -t --rm -v $PWD:/v -e "NODE_ENV=development" nodesource/centos6:0.12 >packaging/centos6.log 2>&1 sh -c "cd /v; source ./packaging/test/centos6.sh; ./preinstall.sh; $test";
|
||||||
|
then echo "centos6 OK"
|
||||||
|
else echo "centos6 fail" && cat packaging/centos6.log
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Centos 7
|
||||||
|
# Fedora 20, 21
|
||||||
|
for dist in centos7 fedora20 fedora21; do
|
||||||
|
echo "Testing $dist..."
|
||||||
|
if docker run -i -t --rm -v $PWD:/v -e "NODE_ENV=development" nodesource/$dist:0.12 >packaging/$dist.log 2>&1 sh -c "cd /v; $test";
|
||||||
|
then echo "$dist OK"
|
||||||
|
else echo "$dist fail" && cat packaging/$dist.log
|
||||||
|
fi
|
||||||
|
done
|
||||||
|
|
||||||
|
# openSUSE 13.2
|
||||||
|
echo "Testing opensuse..."
|
||||||
|
if docker run -i -t --rm -v $PWD:/v opensuse:13.2 >packaging/opensuse.log 2>&1 /bin/sh -c "cd /v; ./packaging/test/opensuse.sh; $test";
|
||||||
|
then echo "opensuse OK"
|
||||||
|
else echo "opensuse fail" && cat packaging/opensuse.log
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Archlinux 2015.06.01
|
||||||
|
echo "Testing archlinux..."
|
||||||
|
if docker run -i -t --rm -v $PWD:/v base/archlinux:2015.06.01 >packaging/archlinux.log 2>&1 sh -c "cd /v; ./packaging/test/archlinux.sh; $test";
|
||||||
|
then echo "archlinux OK"
|
||||||
|
else echo "archlinux fail" && cat packaging/archlinux.log
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Alpine
|
||||||
|
echo "Testing alpine..."
|
||||||
|
if docker run -i -t --rm -v $PWD:/v -e "SHARP_TEST_WITHOUT_CACHE=0" alpine:edge >packaging/alpine.log 2>&1 sh -c "cd /v; ./packaging/test/alpine.sh; $test";
|
||||||
|
then echo "alpine OK"
|
||||||
|
else echo "alpine fail" && cat packaging/alpine.log
|
||||||
|
fi
|
||||||
7
packaging/test/alpine.sh
Executable file
@@ -0,0 +1,7 @@
|
|||||||
|
#!/bin/sh
|
||||||
|
|
||||||
|
# Install build dependencies
|
||||||
|
apk add --update make gcc g++ python nodejs
|
||||||
|
|
||||||
|
# Install libvips with build headers and dependencies
|
||||||
|
apk add libvips-dev --update --repository https://s3.amazonaws.com/wjordan-apk --allow-untrusted
|
||||||
5
packaging/test/archlinux.sh
Executable file
@@ -0,0 +1,5 @@
|
|||||||
|
#!/bin/sh
|
||||||
|
|
||||||
|
# Install Node.js on Archlinux
|
||||||
|
pacman -Sy --noconfirm gcc make python2 nodejs npm | cat
|
||||||
|
ln -s /usr/bin/python2 /usr/bin/python
|
||||||
8
packaging/test/centos6.sh
Executable file
@@ -0,0 +1,8 @@
|
|||||||
|
#!/bin/sh
|
||||||
|
|
||||||
|
# Install C++11 compatible version of g++ on Centos 6
|
||||||
|
curl -o /etc/yum.repos.d/devtools-1.1.repo http://people.centos.org/tru/devtools-1.1/devtools-1.1.repo
|
||||||
|
yum install -y devtoolset-1.1
|
||||||
|
export CC=/opt/centos/devtoolset-1.1/root/usr/bin/gcc
|
||||||
|
export CPP=/opt/centos/devtoolset-1.1/root/usr/bin/cpp
|
||||||
|
export CXX=/opt/centos/devtoolset-1.1/root/usr/bin/c++
|
||||||
5
packaging/test/debian.sh
Executable file
@@ -0,0 +1,5 @@
|
|||||||
|
#!/bin/sh
|
||||||
|
|
||||||
|
# Install pkg-config on Debian/Ubuntu
|
||||||
|
apt-get update
|
||||||
|
apt-get install -y pkg-config
|
||||||
7
packaging/test/opensuse.sh
Executable file
@@ -0,0 +1,7 @@
|
|||||||
|
#!/bin/sh
|
||||||
|
|
||||||
|
# Install Node.js on openSUSE 13.2
|
||||||
|
zypper addrepo http://download.opensuse.org/repositories/devel:languages:nodejs/openSUSE_13.2/devel:languages:nodejs.repo
|
||||||
|
zypper --gpg-auto-import-keys refresh
|
||||||
|
zypper --non-interactive install gcc-c++ make nodejs-devel npm
|
||||||
|
npm install -g npm
|
||||||
20
packaging/win/Dockerfile
Normal file
@@ -0,0 +1,20 @@
|
|||||||
|
FROM ubuntu:precise
|
||||||
|
MAINTAINER Lovell Fuller <npm@lovell.info>
|
||||||
|
|
||||||
|
RUN apt-get update && apt-get install -y curl zip
|
||||||
|
|
||||||
|
ENV VERSION_VIPS=8.2.3
|
||||||
|
|
||||||
|
# Fetch and unzip
|
||||||
|
RUN mkdir /vips
|
||||||
|
WORKDIR /vips
|
||||||
|
RUN curl -O http://www.vips.ecs.soton.ac.uk/supported/8.2/win32/vips-dev-w64-8.2.zip
|
||||||
|
RUN unzip vips-dev-w64-8.2.zip
|
||||||
|
|
||||||
|
# Clean and zip
|
||||||
|
WORKDIR /vips/vips-dev-8.2
|
||||||
|
RUN rm bin/libvipsCC-42.dll bin/libvips-cpp-42.dll bin/libgsf-win32-1-114.dll bin/libstdc++-6.dll
|
||||||
|
RUN cp bin/*.dll lib/
|
||||||
|
RUN cp -r lib64/* lib/
|
||||||
|
|
||||||
|
RUN GZIP=-9 tar czf /libvips-${VERSION_VIPS}-win.tar.gz include lib/glib-2.0 lib/libvips.lib lib/libglib-2.0.lib lib/libgobject-2.0.lib lib/*.dll
|
||||||
@@ -1,19 +1,28 @@
|
|||||||
#!/bin/sh
|
#!/bin/sh
|
||||||
|
|
||||||
# Ensures libvips is installed and attempts to install it if not
|
# This script is no longer required on most
|
||||||
# Currently supports:
|
# 64-bit Linux systems when using sharp v0.12.0+
|
||||||
|
|
||||||
|
# See http://sharp.dimens.io/page/install#linux
|
||||||
|
|
||||||
|
# If you really need this script, it will attempt to
|
||||||
|
# globally install libvips if not already available.
|
||||||
|
|
||||||
|
# Supports:
|
||||||
# * Debian Linux
|
# * Debian Linux
|
||||||
# * Debian 7, 8
|
# * Debian 7, 8
|
||||||
# * Ubuntu 12.04, 14.04, 14.10, 15.04
|
# * Ubuntu 12.04, 14.04, 14.10, 15.04, 15.10
|
||||||
# * Mint 13, 17
|
# * Mint 13, 17
|
||||||
|
# * Elementary 0.3
|
||||||
# * Red Hat Linux
|
# * Red Hat Linux
|
||||||
# * RHEL/Centos/Scientific 6, 7
|
# * RHEL/Centos/Scientific 6, 7
|
||||||
# * Fedora 21, 22
|
# * Fedora 21, 22, 23
|
||||||
# * Amazon Linux 2014.09, 2015.03
|
# * Amazon Linux 2015.03, 2015.09
|
||||||
|
# * OpenSuse 13
|
||||||
|
|
||||||
vips_version_minimum=7.40.0
|
vips_version_minimum=8.2.3
|
||||||
vips_version_latest_major_minor=8.0
|
vips_version_latest_major_minor=8.2
|
||||||
vips_version_latest_patch=2
|
vips_version_latest_patch=3
|
||||||
|
|
||||||
openslide_version_minimum=3.4.0
|
openslide_version_minimum=3.4.0
|
||||||
openslide_version_latest_major_minor=3.4
|
openslide_version_latest_major_minor=3.4
|
||||||
@@ -24,14 +33,14 @@ install_libvips_from_source() {
|
|||||||
curl -O http://www.vips.ecs.soton.ac.uk/supported/$vips_version_latest_major_minor/vips-$vips_version_latest_major_minor.$vips_version_latest_patch.tar.gz
|
curl -O http://www.vips.ecs.soton.ac.uk/supported/$vips_version_latest_major_minor/vips-$vips_version_latest_major_minor.$vips_version_latest_patch.tar.gz
|
||||||
tar zvxf vips-$vips_version_latest_major_minor.$vips_version_latest_patch.tar.gz
|
tar zvxf vips-$vips_version_latest_major_minor.$vips_version_latest_patch.tar.gz
|
||||||
cd vips-$vips_version_latest_major_minor.$vips_version_latest_patch
|
cd vips-$vips_version_latest_major_minor.$vips_version_latest_patch
|
||||||
./configure --disable-debug --disable-docs --disable-static --disable-introspection --enable-cxx=yes --without-python --without-orc --without-fftw $1
|
./configure --disable-debug --disable-docs --disable-static --disable-introspection --disable-dependency-tracking --enable-cxx=yes --without-python --without-orc --without-fftw $1
|
||||||
make
|
make
|
||||||
make install
|
make install
|
||||||
cd ..
|
cd ..
|
||||||
rm -rf vips-$vips_version_latest_major_minor.$vips_version_latest_patch
|
rm -rf vips-$vips_version_latest_major_minor.$vips_version_latest_patch
|
||||||
rm vips-$vips_version_latest_major_minor.$vips_version_latest_patch.tar.gz
|
rm vips-$vips_version_latest_major_minor.$vips_version_latest_patch.tar.gz
|
||||||
ldconfig
|
ldconfig
|
||||||
echo "Installed libvips $vips_version_latest_major_minor.$vips_version_latest_patch"
|
echo "Installed libvips $(PKG_CONFIG_PATH=$PKG_CONFIG_PATH:/usr/local/lib/pkgconfig:/usr/lib/pkgconfig pkg-config --modversion vips)"
|
||||||
}
|
}
|
||||||
|
|
||||||
install_libopenslide_from_source() {
|
install_libopenslide_from_source() {
|
||||||
@@ -117,6 +126,11 @@ if [ "$(id -u)" -ne "0" ]; then
|
|||||||
exit 1
|
exit 1
|
||||||
fi
|
fi
|
||||||
|
|
||||||
|
# Deprecation warning
|
||||||
|
if [ "$(arch)" == "x86_64" ]; then
|
||||||
|
echo "This script is no longer required on most 64-bit Linux systems when using sharp v0.12.0+"
|
||||||
|
fi
|
||||||
|
|
||||||
# OS-specific installations of libopenslide follows
|
# OS-specific installations of libopenslide follows
|
||||||
# Either openslide does not exist, or vips is installed without openslide support
|
# Either openslide does not exist, or vips is installed without openslide support
|
||||||
if [ $enable_openslide -eq 1 ] && [ -z $vips_with_openslide ] && [ $openslide_exists -eq 0 ]; then
|
if [ $enable_openslide -eq 1 ] && [ -z $vips_with_openslide ] && [ $openslide_exists -eq 0 ]; then
|
||||||
@@ -125,12 +139,12 @@ if [ $enable_openslide -eq 1 ] && [ -z $vips_with_openslide ] && [ $openslide_ex
|
|||||||
DISTRO=$(lsb_release -c -s)
|
DISTRO=$(lsb_release -c -s)
|
||||||
echo "Detected Debian Linux '$DISTRO'"
|
echo "Detected Debian Linux '$DISTRO'"
|
||||||
case "$DISTRO" in
|
case "$DISTRO" in
|
||||||
jessie|vivid)
|
jessie|vivid|wily)
|
||||||
# Debian 8, Ubuntu 15
|
# Debian 8, Ubuntu 15
|
||||||
echo "Installing libopenslide via apt-get"
|
echo "Installing libopenslide via apt-get"
|
||||||
apt-get install -y libopenslide-dev
|
apt-get install -y libopenslide-dev
|
||||||
;;
|
;;
|
||||||
trusty|utopic|qiana|rebecca|rafaela)
|
trusty|utopic|qiana|rebecca|rafaela|freya)
|
||||||
# Ubuntu 14, Mint 17
|
# Ubuntu 14, Mint 17
|
||||||
echo "Installing libopenslide dependencies via apt-get"
|
echo "Installing libopenslide dependencies via apt-get"
|
||||||
apt-get install -y automake build-essential curl zlib1g-dev libopenjpeg-dev libpng12-dev libjpeg-dev libtiff5-dev libgdk-pixbuf2.0-dev libxml2-dev libsqlite3-dev libcairo2-dev libglib2.0-dev sqlite3 libsqlite3-dev
|
apt-get install -y automake build-essential curl zlib1g-dev libopenjpeg-dev libpng12-dev libjpeg-dev libtiff5-dev libgdk-pixbuf2.0-dev libxml2-dev libsqlite3-dev libcairo2-dev libglib2.0-dev sqlite3 libsqlite3-dev
|
||||||
@@ -209,18 +223,8 @@ if [ -f /etc/debian_version ]; then
|
|||||||
DISTRO=$(lsb_release -c -s)
|
DISTRO=$(lsb_release -c -s)
|
||||||
echo "Detected Debian Linux '$DISTRO'"
|
echo "Detected Debian Linux '$DISTRO'"
|
||||||
case "$DISTRO" in
|
case "$DISTRO" in
|
||||||
jessie|vivid)
|
jessie|trusty|utopic|vivid|wily|xenial|qiana|rebecca|rafaela|freya)
|
||||||
# Debian 8, Ubuntu 15
|
# Debian 8, Ubuntu 14.04+, Mint 17
|
||||||
if [ $enable_openslide -eq 1 ]; then
|
|
||||||
echo "Recompiling vips with openslide support"
|
|
||||||
install_libvips_from_source
|
|
||||||
else
|
|
||||||
echo "Installing libvips via apt-get"
|
|
||||||
apt-get install -y libvips-dev libgsf-1-dev
|
|
||||||
fi
|
|
||||||
;;
|
|
||||||
trusty|utopic|qiana|rebecca|rafaela)
|
|
||||||
# Ubuntu 14, Mint 17
|
|
||||||
echo "Installing libvips dependencies via apt-get"
|
echo "Installing libvips dependencies via apt-get"
|
||||||
apt-get install -y automake build-essential gobject-introspection gtk-doc-tools libglib2.0-dev libjpeg-dev libpng12-dev libwebp-dev libtiff5-dev libexif-dev libgsf-1-dev liblcms2-dev libxml2-dev swig libmagickcore-dev curl
|
apt-get install -y automake build-essential gobject-introspection gtk-doc-tools libglib2.0-dev libjpeg-dev libpng12-dev libwebp-dev libtiff5-dev libexif-dev libgsf-1-dev liblcms2-dev libxml2-dev swig libmagickcore-dev curl
|
||||||
install_libvips_from_source
|
install_libvips_from_source
|
||||||
@@ -247,32 +251,26 @@ elif [ -f /etc/redhat-release ]; then
|
|||||||
# RHEL/CentOS 7
|
# RHEL/CentOS 7
|
||||||
echo "Installing libvips dependencies via yum"
|
echo "Installing libvips dependencies via yum"
|
||||||
yum groupinstall -y "Development Tools"
|
yum groupinstall -y "Development Tools"
|
||||||
yum install -y gtk-doc libxml2-devel libjpeg-turbo-devel libpng-devel libtiff-devel libexif-devel libgsf-devel lcms-devel ImageMagick-devel gobject-introspection-devel libwebp-devel curl
|
yum install -y tar curl gtk-doc libxml2-devel libjpeg-turbo-devel libpng-devel libtiff-devel libexif-devel libgsf-devel lcms2-devel ImageMagick-devel gobject-introspection-devel libwebp-devel
|
||||||
install_libvips_from_source "--prefix=/usr"
|
install_libvips_from_source "--prefix=/usr"
|
||||||
;;
|
;;
|
||||||
"Red Hat Enterprise Linux release 6."*|"CentOS release 6."*|"Scientific Linux release 6."*)
|
"Red Hat Enterprise Linux release 6."*|"CentOS release 6."*|"Scientific Linux release 6."*)
|
||||||
# RHEL/CentOS 6
|
# RHEL/CentOS 6
|
||||||
echo "Installing libvips dependencies via yum"
|
echo "Installing libvips dependencies via yum"
|
||||||
yum groupinstall -y "Development Tools"
|
yum groupinstall -y "Development Tools"
|
||||||
yum install -y gtk-doc libxml2-devel libjpeg-turbo-devel libpng-devel libtiff-devel libexif-devel libgsf-devel lcms-devel ImageMagick-devel curl
|
yum install -y tar curl gtk-doc libxml2-devel libjpeg-turbo-devel libpng-devel libtiff-devel libexif-devel libgsf-devel lcms-devel ImageMagick-devel
|
||||||
yum install -y http://li.nux.ro/download/nux/dextop/el6/x86_64/nux-dextop-release-0-2.el6.nux.noarch.rpm
|
yum install -y http://li.nux.ro/download/nux/dextop/el6/x86_64/nux-dextop-release-0-2.el6.nux.noarch.rpm
|
||||||
yum install -y --enablerepo=nux-dextop gobject-introspection-devel
|
yum install -y --enablerepo=nux-dextop gobject-introspection-devel
|
||||||
yum install -y http://rpms.famillecollet.com/enterprise/remi-release-6.rpm
|
yum install -y http://rpms.famillecollet.com/enterprise/remi-release-6.rpm
|
||||||
yum install -y --enablerepo=remi libwebp-devel
|
yum install -y --enablerepo=remi libwebp-devel
|
||||||
install_libvips_from_source "--prefix=/usr"
|
install_libvips_from_source "--prefix=/usr"
|
||||||
;;
|
;;
|
||||||
"Fedora release 21 "*|"Fedora release 22 "*)
|
"Fedora"*)
|
||||||
# Fedora 21, 22
|
# Fedora 21, 22, 23
|
||||||
if [ $enable_openslide -eq 1 ]; then
|
echo "Installing libvips dependencies via yum"
|
||||||
echo "Installing libvips dependencies via yum"
|
yum groupinstall -y "Development Tools"
|
||||||
yum groupinstall -y "Development Tools"
|
yum install -y gcc-c++ gtk-doc libxml2-devel libjpeg-turbo-devel libpng-devel libtiff-devel libexif-devel lcms-devel ImageMagick-devel gobject-introspection-devel libwebp-devel curl
|
||||||
yum install -y gcc-c++ gtk-doc libxml2-devel libjpeg-turbo-devel libpng-devel libtiff-devel libexif-devel lcms-devel ImageMagick-devel gobject-introspection-devel libwebp-devel curl
|
install_libvips_from_source "--prefix=/usr"
|
||||||
echo "Compiling vips with openslide support"
|
|
||||||
install_libvips_from_source "--prefix=/usr"
|
|
||||||
else
|
|
||||||
echo "Installing libvips via yum"
|
|
||||||
yum install -y vips-devel
|
|
||||||
fi
|
|
||||||
;;
|
;;
|
||||||
*)
|
*)
|
||||||
# Unsupported RHEL-based OS
|
# Unsupported RHEL-based OS
|
||||||
@@ -283,12 +281,12 @@ elif [ -f /etc/system-release ]; then
|
|||||||
# Probably Amazon Linux
|
# Probably Amazon Linux
|
||||||
RELEASE=$(cat /etc/system-release)
|
RELEASE=$(cat /etc/system-release)
|
||||||
case $RELEASE in
|
case $RELEASE in
|
||||||
"Amazon Linux AMI release 2014.09"|"Amazon Linux AMI release 2015.03")
|
"Amazon Linux AMI release 2015.03"|"Amazon Linux AMI release 2015.09")
|
||||||
# Amazon Linux
|
# Amazon Linux
|
||||||
echo "Detected '$RELEASE'"
|
echo "Detected '$RELEASE'"
|
||||||
echo "Installing libvips dependencies via yum"
|
echo "Installing libvips dependencies via yum"
|
||||||
yum groupinstall -y "Development Tools"
|
yum groupinstall -y "Development Tools"
|
||||||
yum install -y gtk-doc libxml2-devel libjpeg-turbo-devel libpng-devel libtiff-devel libexif-devel libgsf-devel lcms-devel ImageMagick-devel gobject-introspection-devel libwebp-devel curl
|
yum install -y gtk-doc libxml2-devel libjpeg-turbo-devel libpng-devel libtiff-devel libexif-devel libgsf-devel lcms2-devel ImageMagick-devel gobject-introspection-devel libwebp-devel curl
|
||||||
install_libvips_from_source "--prefix=/usr"
|
install_libvips_from_source "--prefix=/usr"
|
||||||
;;
|
;;
|
||||||
*)
|
*)
|
||||||
|
|||||||
189
src/common.cc
Executable file → Normal file
@@ -1,32 +1,30 @@
|
|||||||
#include <cstdlib>
|
#include <cstdlib>
|
||||||
#include <string>
|
#include <string>
|
||||||
#include <string.h>
|
#include <string.h>
|
||||||
#include <vips/vips.h>
|
#include <vips/vips8>
|
||||||
|
|
||||||
#include "common.h"
|
#include "common.h"
|
||||||
|
|
||||||
// Verify platform and compiler compatibility
|
// Verify platform and compiler compatibility
|
||||||
|
|
||||||
#if (VIPS_MAJOR_VERSION < 7 || (VIPS_MAJOR_VERSION == 7 && VIPS_MINOR_VERSION < 40))
|
#if (VIPS_MAJOR_VERSION < 8 || (VIPS_MAJOR_VERSION == 8 && VIPS_MINOR_VERSION < 2))
|
||||||
#error libvips version 7.40.0+ required - see http://sharp.dimens.io/page/install
|
#error libvips version 8.2.0+ required - see sharp.dimens.io/page/install
|
||||||
#endif
|
|
||||||
|
|
||||||
#ifdef _WIN64
|
|
||||||
#error Windows 64-bit is currently unsupported - see http://sharp.dimens.io/page/install#windows
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#if ((!defined(__clang__)) && defined(__GNUC__) && (__GNUC__ < 4 || (__GNUC__ == 4 && __GNUC_MINOR__ < 6)))
|
#if ((!defined(__clang__)) && defined(__GNUC__) && (__GNUC__ < 4 || (__GNUC__ == 4 && __GNUC_MINOR__ < 6)))
|
||||||
#error GCC version 4.6+ is required for C++11 features - see http://sharp.dimens.io/page/install#prerequisites
|
#error GCC version 4.6+ is required for C++11 features - see sharp.dimens.io/page/install#prerequisites
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#if (defined(__clang__) && defined(__has_feature))
|
#if (defined(__clang__) && defined(__has_feature))
|
||||||
#if (!__has_feature(cxx_range_for))
|
#if (!__has_feature(cxx_range_for))
|
||||||
#error clang version 3.0+ is required for C++11 features - see http://sharp.dimens.io/page/install#prerequisites
|
#error clang version 3.0+ is required for C++11 features - see sharp.dimens.io/page/install#prerequisites
|
||||||
#endif
|
#endif
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#define EXIF_IFD0_ORIENTATION "exif-ifd0-Orientation"
|
#define EXIF_IFD0_ORIENTATION "exif-ifd0-Orientation"
|
||||||
|
|
||||||
|
using vips::VImage;
|
||||||
|
|
||||||
namespace sharp {
|
namespace sharp {
|
||||||
|
|
||||||
// How many tasks are in the queue?
|
// How many tasks are in the queue?
|
||||||
@@ -54,6 +52,29 @@ namespace sharp {
|
|||||||
bool IsDz(std::string const &str) {
|
bool IsDz(std::string const &str) {
|
||||||
return EndsWith(str, ".dzi") || EndsWith(str, ".DZI");
|
return EndsWith(str, ".dzi") || EndsWith(str, ".DZI");
|
||||||
}
|
}
|
||||||
|
bool IsDzZip(std::string const &str) {
|
||||||
|
return EndsWith(str, ".zip") || EndsWith(str, ".ZIP") || EndsWith(str, ".szi") || EndsWith(str, ".SZI");
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
Provide a string identifier for the given image type.
|
||||||
|
*/
|
||||||
|
std::string ImageTypeId(ImageType const imageType) {
|
||||||
|
std::string id;
|
||||||
|
switch (imageType) {
|
||||||
|
case ImageType::JPEG: id = "jpeg"; break;
|
||||||
|
case ImageType::PNG: id = "png"; break;
|
||||||
|
case ImageType::WEBP: id = "webp"; break;
|
||||||
|
case ImageType::TIFF: id = "tiff"; break;
|
||||||
|
case ImageType::MAGICK: id = "magick"; break;
|
||||||
|
case ImageType::OPENSLIDE: id = "openslide"; break;
|
||||||
|
case ImageType::PPM: id = "ppm"; break;
|
||||||
|
case ImageType::FITS: id = "fits"; break;
|
||||||
|
case ImageType::RAW: id = "raw"; break;
|
||||||
|
case ImageType::UNKNOWN: id = "unknown"; break;
|
||||||
|
}
|
||||||
|
return id;
|
||||||
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Determine image format of a buffer.
|
Determine image format of a buffer.
|
||||||
@@ -62,7 +83,7 @@ namespace sharp {
|
|||||||
ImageType imageType = ImageType::UNKNOWN;
|
ImageType imageType = ImageType::UNKNOWN;
|
||||||
char const *load = vips_foreign_find_load_buffer(buffer, length);
|
char const *load = vips_foreign_find_load_buffer(buffer, length);
|
||||||
if (load != NULL) {
|
if (load != NULL) {
|
||||||
std::string loader = load;
|
std::string const loader = load;
|
||||||
if (EndsWith(loader, "JpegBuffer")) {
|
if (EndsWith(loader, "JpegBuffer")) {
|
||||||
imageType = ImageType::JPEG;
|
imageType = ImageType::JPEG;
|
||||||
} else if (EndsWith(loader, "PngBuffer")) {
|
} else if (EndsWith(loader, "PngBuffer")) {
|
||||||
@@ -78,21 +99,14 @@ namespace sharp {
|
|||||||
return imageType;
|
return imageType;
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
|
||||||
Initialise and return a VipsImage from a buffer. Supports JPEG, PNG, WebP and TIFF.
|
|
||||||
*/
|
|
||||||
VipsImage* InitImage(void *buffer, size_t const length, VipsAccess const access) {
|
|
||||||
return vips_image_new_from_buffer(buffer, length, NULL, "access", access, NULL);
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Determine image format, reads the first few bytes of the file
|
Determine image format, reads the first few bytes of the file
|
||||||
*/
|
*/
|
||||||
ImageType DetermineImageType(char const *file) {
|
ImageType DetermineImageType(char const *file) {
|
||||||
ImageType imageType = ImageType::UNKNOWN;
|
ImageType imageType = ImageType::UNKNOWN;
|
||||||
char const *load = vips_foreign_find_load(file);
|
char const *load = vips_foreign_find_load(file);
|
||||||
if (load != NULL) {
|
if (load != nullptr) {
|
||||||
std::string loader = load;
|
std::string const loader = load;
|
||||||
if (EndsWith(loader, "JpegFile")) {
|
if (EndsWith(loader, "JpegFile")) {
|
||||||
imageType = ImageType::JPEG;
|
imageType = ImageType::JPEG;
|
||||||
} else if (EndsWith(loader, "Png")) {
|
} else if (EndsWith(loader, "Png")) {
|
||||||
@@ -103,6 +117,10 @@ namespace sharp {
|
|||||||
imageType = ImageType::OPENSLIDE;
|
imageType = ImageType::OPENSLIDE;
|
||||||
} else if (EndsWith(loader, "TiffFile")) {
|
} else if (EndsWith(loader, "TiffFile")) {
|
||||||
imageType = ImageType::TIFF;
|
imageType = ImageType::TIFF;
|
||||||
|
} else if (EndsWith(loader, "Ppm")) {
|
||||||
|
imageType = ImageType::PPM;
|
||||||
|
} else if (EndsWith(loader, "Fits")) {
|
||||||
|
imageType = ImageType::FITS;
|
||||||
} else if (EndsWith(loader, "Magick") || EndsWith(loader, "MagickFile")) {
|
} else if (EndsWith(loader, "Magick") || EndsWith(loader, "MagickFile")) {
|
||||||
imageType = ImageType::MAGICK;
|
imageType = ImageType::MAGICK;
|
||||||
}
|
}
|
||||||
@@ -110,43 +128,37 @@ namespace sharp {
|
|||||||
return imageType;
|
return imageType;
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
|
||||||
Initialise and return a VipsImage from a file.
|
|
||||||
*/
|
|
||||||
VipsImage* InitImage(char const *file, VipsAccess const access) {
|
|
||||||
return vips_image_new_from_file(file, "access", access, NULL);
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Does this image have an embedded profile?
|
Does this image have an embedded profile?
|
||||||
*/
|
*/
|
||||||
bool HasProfile(VipsImage *image) {
|
bool HasProfile(VImage image) {
|
||||||
return (vips_image_get_typeof(image, VIPS_META_ICC_NAME) > 0) ? TRUE : FALSE;
|
return (image.get_typeof(VIPS_META_ICC_NAME) != 0) ? TRUE : FALSE;
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Does this image have an alpha channel?
|
Does this image have an alpha channel?
|
||||||
Uses colour space interpretation with number of channels to guess this.
|
Uses colour space interpretation with number of channels to guess this.
|
||||||
*/
|
*/
|
||||||
bool HasAlpha(VipsImage *image) {
|
bool HasAlpha(VImage image) {
|
||||||
|
int const bands = image.bands();
|
||||||
|
VipsInterpretation const interpretation = image.interpretation();
|
||||||
return (
|
return (
|
||||||
(image->Bands == 2 && image->Type == VIPS_INTERPRETATION_B_W) ||
|
(bands == 2 && interpretation == VIPS_INTERPRETATION_B_W) ||
|
||||||
(image->Bands == 4 && image->Type != VIPS_INTERPRETATION_CMYK) ||
|
(bands == 4 && interpretation != VIPS_INTERPRETATION_CMYK) ||
|
||||||
(image->Bands == 5 && image->Type == VIPS_INTERPRETATION_CMYK)
|
(bands == 5 && interpretation == VIPS_INTERPRETATION_CMYK)
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Get EXIF Orientation of image, if any.
|
Get EXIF Orientation of image, if any.
|
||||||
*/
|
*/
|
||||||
int ExifOrientation(VipsImage const *image) {
|
int ExifOrientation(VImage image) {
|
||||||
int orientation = 0;
|
int orientation = 0;
|
||||||
const char *exif;
|
if (image.get_typeof(EXIF_IFD0_ORIENTATION) != 0) {
|
||||||
if (
|
char const *exif = image.get_string(EXIF_IFD0_ORIENTATION);
|
||||||
vips_image_get_typeof(image, EXIF_IFD0_ORIENTATION) != 0 &&
|
if (exif != nullptr) {
|
||||||
!vips_image_get_string(image, EXIF_IFD0_ORIENTATION, &exif)
|
orientation = atoi(&exif[0]);
|
||||||
) {
|
}
|
||||||
orientation = atoi(&exif[0]);
|
|
||||||
}
|
}
|
||||||
return orientation;
|
return orientation;
|
||||||
}
|
}
|
||||||
@@ -154,31 +166,100 @@ namespace sharp {
|
|||||||
/*
|
/*
|
||||||
Set EXIF Orientation of image.
|
Set EXIF Orientation of image.
|
||||||
*/
|
*/
|
||||||
void SetExifOrientation(VipsImage *image, int const orientation) {
|
void SetExifOrientation(VImage image, int const orientation) {
|
||||||
char exif[3];
|
char exif[3];
|
||||||
g_snprintf(exif, sizeof(exif), "%d", orientation);
|
g_snprintf(exif, sizeof(exif), "%d", orientation);
|
||||||
vips_image_set_string(image, EXIF_IFD0_ORIENTATION, exif);
|
image.set(EXIF_IFD0_ORIENTATION, exif);
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Remove EXIF Orientation from image.
|
Remove EXIF Orientation from image.
|
||||||
*/
|
*/
|
||||||
void RemoveExifOrientation(VipsImage *image) {
|
void RemoveExifOrientation(VImage image) {
|
||||||
vips_image_remove(image, EXIF_IFD0_ORIENTATION);
|
SetExifOrientation(image, 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Returns the window size for the named interpolator. For example,
|
Does this image have a non-default density?
|
||||||
a window size of 3 means a 3x3 pixel grid is used for the calculation.
|
|
||||||
*/
|
*/
|
||||||
int InterpolatorWindowSize(char const *name) {
|
bool HasDensity(VImage image) {
|
||||||
VipsInterpolate *interpolator = vips_interpolate_new(name);
|
return image.xres() > 1.0;
|
||||||
if (interpolator == NULL) {
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
int window_size = vips_interpolate_get_window_size(interpolator);
|
|
||||||
g_object_unref(interpolator);
|
|
||||||
return window_size;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
} // namespace sharp
|
/*
|
||||||
|
Get pixels/mm resolution as pixels/inch density.
|
||||||
|
*/
|
||||||
|
int GetDensity(VImage image) {
|
||||||
|
return static_cast<int>(round(image.xres() * 25.4));
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
Set pixels/mm resolution based on a pixels/inch density.
|
||||||
|
*/
|
||||||
|
void SetDensity(VImage image, const int density) {
|
||||||
|
const double pixelsPerMm = static_cast<double>(density) / 25.4;
|
||||||
|
image.set("Xres", pixelsPerMm);
|
||||||
|
image.set("Yres", pixelsPerMm);
|
||||||
|
image.set(VIPS_META_RESOLUTION_UNIT, "in");
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
Called when a Buffer undergoes GC, required to support mixed runtime libraries in Windows
|
||||||
|
*/
|
||||||
|
void FreeCallback(char* data, void* hint) {
|
||||||
|
if (data != nullptr) {
|
||||||
|
g_free(data);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
Calculate the (left, top) coordinates of the output image
|
||||||
|
within the input image, applying the given gravity.
|
||||||
|
*/
|
||||||
|
std::tuple<int, int> CalculateCrop(int const inWidth, int const inHeight,
|
||||||
|
int const outWidth, int const outHeight, int const gravity) {
|
||||||
|
|
||||||
|
int left = 0;
|
||||||
|
int top = 0;
|
||||||
|
switch (gravity) {
|
||||||
|
case 1:
|
||||||
|
// North
|
||||||
|
left = (inWidth - outWidth + 1) / 2;
|
||||||
|
break;
|
||||||
|
case 2:
|
||||||
|
// East
|
||||||
|
left = inWidth - outWidth;
|
||||||
|
top = (inHeight - outHeight + 1) / 2;
|
||||||
|
break;
|
||||||
|
case 3:
|
||||||
|
// South
|
||||||
|
left = (inWidth - outWidth + 1) / 2;
|
||||||
|
top = inHeight - outHeight;
|
||||||
|
break;
|
||||||
|
case 4:
|
||||||
|
// West
|
||||||
|
top = (inHeight - outHeight + 1) / 2;
|
||||||
|
break;
|
||||||
|
case 5:
|
||||||
|
// Northeast
|
||||||
|
left = inWidth - outWidth;
|
||||||
|
break;
|
||||||
|
case 6:
|
||||||
|
// Southeast
|
||||||
|
left = inWidth - outWidth;
|
||||||
|
top = inHeight - outHeight;
|
||||||
|
case 7:
|
||||||
|
// Southwest
|
||||||
|
top = inHeight - outHeight;
|
||||||
|
case 8:
|
||||||
|
// Northwest
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
// Centre
|
||||||
|
left = (inWidth - outWidth + 1) / 2;
|
||||||
|
top = (inHeight - outHeight + 1) / 2;
|
||||||
|
}
|
||||||
|
return std::make_tuple(left, top);
|
||||||
|
}
|
||||||
|
|
||||||
|
} // namespace sharp
|
||||||
|
|||||||
63
src/common.h
Executable file → Normal file
@@ -2,6 +2,11 @@
|
|||||||
#define SRC_COMMON_H_
|
#define SRC_COMMON_H_
|
||||||
|
|
||||||
#include <string>
|
#include <string>
|
||||||
|
#include <tuple>
|
||||||
|
|
||||||
|
#include <vips/vips8>
|
||||||
|
|
||||||
|
using vips::VImage;
|
||||||
|
|
||||||
namespace sharp {
|
namespace sharp {
|
||||||
|
|
||||||
@@ -12,7 +17,10 @@ namespace sharp {
|
|||||||
WEBP,
|
WEBP,
|
||||||
TIFF,
|
TIFF,
|
||||||
MAGICK,
|
MAGICK,
|
||||||
OPENSLIDE
|
OPENSLIDE,
|
||||||
|
PPM,
|
||||||
|
FITS,
|
||||||
|
RAW
|
||||||
};
|
};
|
||||||
|
|
||||||
// How many tasks are in the queue?
|
// How many tasks are in the queue?
|
||||||
@@ -27,6 +35,12 @@ namespace sharp {
|
|||||||
bool IsWebp(std::string const &str);
|
bool IsWebp(std::string const &str);
|
||||||
bool IsTiff(std::string const &str);
|
bool IsTiff(std::string const &str);
|
||||||
bool IsDz(std::string const &str);
|
bool IsDz(std::string const &str);
|
||||||
|
bool IsDzZip(std::string const &str);
|
||||||
|
|
||||||
|
/*
|
||||||
|
Provide a string identifier for the given image type.
|
||||||
|
*/
|
||||||
|
std::string ImageTypeId(ImageType const imageType);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Determine image format of a buffer.
|
Determine image format of a buffer.
|
||||||
@@ -38,47 +52,58 @@ namespace sharp {
|
|||||||
*/
|
*/
|
||||||
ImageType DetermineImageType(char const *file);
|
ImageType DetermineImageType(char const *file);
|
||||||
|
|
||||||
/*
|
|
||||||
Initialise and return a VipsImage from a buffer. Supports JPEG, PNG, WebP and TIFF.
|
|
||||||
*/
|
|
||||||
VipsImage* InitImage(void *buffer, size_t const length, VipsAccess const access);
|
|
||||||
|
|
||||||
/*
|
|
||||||
Initialise and return a VipsImage from a file.
|
|
||||||
*/
|
|
||||||
VipsImage* InitImage(char const *file, VipsAccess const access);
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Does this image have an embedded profile?
|
Does this image have an embedded profile?
|
||||||
*/
|
*/
|
||||||
bool HasProfile(VipsImage *image);
|
bool HasProfile(VImage image);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Does this image have an alpha channel?
|
Does this image have an alpha channel?
|
||||||
Uses colour space interpretation with number of channels to guess this.
|
Uses colour space interpretation with number of channels to guess this.
|
||||||
*/
|
*/
|
||||||
bool HasAlpha(VipsImage *image);
|
bool HasAlpha(VImage image);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Get EXIF Orientation of image, if any.
|
Get EXIF Orientation of image, if any.
|
||||||
*/
|
*/
|
||||||
int ExifOrientation(VipsImage const *image);
|
int ExifOrientation(VImage image);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Set EXIF Orientation of image.
|
Set EXIF Orientation of image.
|
||||||
*/
|
*/
|
||||||
void SetExifOrientation(VipsImage *image, int const orientation);
|
void SetExifOrientation(VImage image, int const orientation);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Remove EXIF Orientation from image.
|
Remove EXIF Orientation from image.
|
||||||
*/
|
*/
|
||||||
void RemoveExifOrientation(VipsImage *image);
|
void RemoveExifOrientation(VImage image);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Returns the window size for the named interpolator. For example,
|
Does this image have a non-default density?
|
||||||
a window size of 3 means a 3x3 pixel grid is used for the calculation.
|
|
||||||
*/
|
*/
|
||||||
int InterpolatorWindowSize(char const *name);
|
bool HasDensity(VImage image);
|
||||||
|
|
||||||
|
/*
|
||||||
|
Get pixels/mm resolution as pixels/inch density.
|
||||||
|
*/
|
||||||
|
int GetDensity(VImage image);
|
||||||
|
|
||||||
|
/*
|
||||||
|
Set pixels/mm resolution based on a pixels/inch density.
|
||||||
|
*/
|
||||||
|
void SetDensity(VImage image, const int density);
|
||||||
|
|
||||||
|
/*
|
||||||
|
Called when a Buffer undergoes GC, required to support mixed runtime libraries in Windows
|
||||||
|
*/
|
||||||
|
void FreeCallback(char* data, void* hint);
|
||||||
|
|
||||||
|
/*
|
||||||
|
Calculate the (left, top) coordinates of the output image
|
||||||
|
within the input image, applying the given gravity.
|
||||||
|
*/
|
||||||
|
std::tuple<int, int> CalculateCrop(int const inWidth, int const inHeight,
|
||||||
|
int const outWidth, int const outHeight, int const gravity);
|
||||||
|
|
||||||
} // namespace sharp
|
} // namespace sharp
|
||||||
|
|
||||||
|
|||||||
52
src/libvips/cplusplus/VError.cpp
Normal file
@@ -0,0 +1,52 @@
|
|||||||
|
// Code for error type
|
||||||
|
|
||||||
|
/*
|
||||||
|
|
||||||
|
Copyright (C) 1991-2001 The National Gallery
|
||||||
|
|
||||||
|
This program is free software; you can redistribute it and/or modify
|
||||||
|
it under the terms of the GNU Lesser General Public License as published by
|
||||||
|
the Free Software Foundation; either version 2 of the License, or
|
||||||
|
(at your option) any later version.
|
||||||
|
|
||||||
|
This program is distributed in the hope that it will be useful,
|
||||||
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
GNU Lesser General Public License for more details.
|
||||||
|
|
||||||
|
You should have received a copy of the GNU Lesser General Public License
|
||||||
|
along with this program; if not, write to the Free Software
|
||||||
|
Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
|
||||||
|
02110-1301 USA
|
||||||
|
|
||||||
|
*/
|
||||||
|
|
||||||
|
/*
|
||||||
|
|
||||||
|
These files are distributed with VIPS - http://www.vips.ecs.soton.ac.uk
|
||||||
|
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifdef HAVE_CONFIG_H
|
||||||
|
#include <config.h>
|
||||||
|
#endif /*HAVE_CONFIG_H*/
|
||||||
|
#include <vips/intl.h>
|
||||||
|
|
||||||
|
#include <iostream>
|
||||||
|
|
||||||
|
#include <vips/vips8>
|
||||||
|
|
||||||
|
VIPS_NAMESPACE_START
|
||||||
|
|
||||||
|
std::ostream &operator<<( std::ostream &file, const VError &err )
|
||||||
|
{
|
||||||
|
err.ostream_print( file );
|
||||||
|
return( file );
|
||||||
|
}
|
||||||
|
|
||||||
|
void VError::ostream_print( std::ostream &file ) const
|
||||||
|
{
|
||||||
|
file << _what;
|
||||||
|
}
|
||||||
|
|
||||||
|
VIPS_NAMESPACE_END
|
||||||
714
src/libvips/cplusplus/VImage.cpp
Normal file
@@ -0,0 +1,714 @@
|
|||||||
|
/* Object part of VImage class
|
||||||
|
*
|
||||||
|
* 30/12/14
|
||||||
|
* - allow set enum value from string
|
||||||
|
*/
|
||||||
|
|
||||||
|
/*
|
||||||
|
|
||||||
|
Copyright (C) 1991-2001 The National Gallery
|
||||||
|
|
||||||
|
This program is free software; you can redistribute it and/or modify
|
||||||
|
it under the terms of the GNU Lesser General Public License as published by
|
||||||
|
the Free Software Foundation; either version 2 of the License, or
|
||||||
|
(at your option) any later version.
|
||||||
|
|
||||||
|
This program is distributed in the hope that it will be useful,
|
||||||
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
GNU Lesser General Public License for more details.
|
||||||
|
|
||||||
|
You should have received a copy of the GNU Lesser General Public License
|
||||||
|
along with this program; if not, write to the Free Software
|
||||||
|
Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
|
||||||
|
02110-1301 USA
|
||||||
|
|
||||||
|
*/
|
||||||
|
|
||||||
|
/*
|
||||||
|
|
||||||
|
These files are distributed with VIPS - http://www.vips.ecs.soton.ac.uk
|
||||||
|
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifdef HAVE_CONFIG_H
|
||||||
|
#include <config.h>
|
||||||
|
#endif /*HAVE_CONFIG_H*/
|
||||||
|
#include <vips/intl.h>
|
||||||
|
|
||||||
|
#include <vips/vips8>
|
||||||
|
|
||||||
|
#include <vips/debug.h>
|
||||||
|
|
||||||
|
/*
|
||||||
|
#define VIPS_DEBUG
|
||||||
|
#define VIPS_DEBUG_VERBOSE
|
||||||
|
*/
|
||||||
|
|
||||||
|
VIPS_NAMESPACE_START
|
||||||
|
|
||||||
|
std::vector<double>
|
||||||
|
to_vectorv( int n, ... )
|
||||||
|
{
|
||||||
|
std::vector<double> vector( n );
|
||||||
|
va_list ap;
|
||||||
|
|
||||||
|
va_start( ap, n );
|
||||||
|
for( int i = 0; i < n; i++ )
|
||||||
|
vector[i] = va_arg( ap, double );
|
||||||
|
va_end( ap );
|
||||||
|
|
||||||
|
return( vector );
|
||||||
|
}
|
||||||
|
|
||||||
|
std::vector<double>
|
||||||
|
to_vector( double value )
|
||||||
|
{
|
||||||
|
return( to_vectorv( 1, value ) );
|
||||||
|
}
|
||||||
|
|
||||||
|
std::vector<double>
|
||||||
|
to_vector( int n, double array[] )
|
||||||
|
{
|
||||||
|
std::vector<double> vector( n );
|
||||||
|
|
||||||
|
for( int i = 0; i < n; i++ )
|
||||||
|
vector[i] = array[i];
|
||||||
|
|
||||||
|
return( vector );
|
||||||
|
}
|
||||||
|
|
||||||
|
std::vector<double>
|
||||||
|
negate( std::vector<double> vector )
|
||||||
|
{
|
||||||
|
std::vector<double> new_vector( vector.size() );
|
||||||
|
|
||||||
|
for( unsigned int i = 0; i < vector.size(); i++ )
|
||||||
|
new_vector[i] = vector[i] * -1;
|
||||||
|
|
||||||
|
return( new_vector );
|
||||||
|
}
|
||||||
|
|
||||||
|
std::vector<double>
|
||||||
|
invert( std::vector<double> vector )
|
||||||
|
{
|
||||||
|
std::vector<double> new_vector( vector.size() );
|
||||||
|
|
||||||
|
for( unsigned int i = 0; i < vector.size(); i++ )
|
||||||
|
new_vector[i] = 1.0 / vector[i];
|
||||||
|
|
||||||
|
return( new_vector );
|
||||||
|
}
|
||||||
|
|
||||||
|
VOption::~VOption()
|
||||||
|
{
|
||||||
|
std::list<Pair *>::iterator i;
|
||||||
|
|
||||||
|
for( i = options.begin(); i != options.end(); ++i )
|
||||||
|
delete *i;
|
||||||
|
}
|
||||||
|
|
||||||
|
// input bool
|
||||||
|
VOption *
|
||||||
|
VOption::set( const char *name, bool value )
|
||||||
|
{
|
||||||
|
Pair *pair = new Pair( name );
|
||||||
|
|
||||||
|
pair->input = true;
|
||||||
|
g_value_init( &pair->value, G_TYPE_BOOLEAN );
|
||||||
|
g_value_set_boolean( &pair->value, value );
|
||||||
|
options.push_back( pair );
|
||||||
|
|
||||||
|
return( this );
|
||||||
|
}
|
||||||
|
|
||||||
|
// input int ... this path is used for enums as well
|
||||||
|
VOption *
|
||||||
|
VOption::set( const char *name, int value )
|
||||||
|
{
|
||||||
|
Pair *pair = new Pair( name );
|
||||||
|
|
||||||
|
pair->input = true;
|
||||||
|
g_value_init( &pair->value, G_TYPE_INT );
|
||||||
|
g_value_set_int( &pair->value, value );
|
||||||
|
options.push_back( pair );
|
||||||
|
|
||||||
|
return( this );
|
||||||
|
}
|
||||||
|
|
||||||
|
// input double
|
||||||
|
VOption *
|
||||||
|
VOption::set( const char *name, double value )
|
||||||
|
{
|
||||||
|
Pair *pair = new Pair( name );
|
||||||
|
|
||||||
|
pair->input = true;
|
||||||
|
g_value_init( &pair->value, G_TYPE_DOUBLE );
|
||||||
|
g_value_set_double( &pair->value, value );
|
||||||
|
options.push_back( pair );
|
||||||
|
|
||||||
|
return( this );
|
||||||
|
}
|
||||||
|
|
||||||
|
VOption *
|
||||||
|
VOption::set( const char *name, const char *value )
|
||||||
|
{
|
||||||
|
Pair *pair = new Pair( name );
|
||||||
|
|
||||||
|
pair->input = true;
|
||||||
|
g_value_init( &pair->value, G_TYPE_STRING );
|
||||||
|
g_value_set_string( &pair->value, value );
|
||||||
|
options.push_back( pair );
|
||||||
|
|
||||||
|
return( this );
|
||||||
|
}
|
||||||
|
|
||||||
|
// input image
|
||||||
|
VOption *
|
||||||
|
VOption::set( const char *name, VImage value )
|
||||||
|
{
|
||||||
|
Pair *pair = new Pair( name );
|
||||||
|
|
||||||
|
pair->input = true;
|
||||||
|
g_value_init( &pair->value, VIPS_TYPE_IMAGE );
|
||||||
|
g_value_set_object( &pair->value, value.get_image() );
|
||||||
|
options.push_back( pair );
|
||||||
|
|
||||||
|
return( this );
|
||||||
|
}
|
||||||
|
|
||||||
|
// input double array
|
||||||
|
VOption *
|
||||||
|
VOption::set( const char *name, std::vector<double> value )
|
||||||
|
{
|
||||||
|
Pair *pair = new Pair( name );
|
||||||
|
|
||||||
|
double *array;
|
||||||
|
unsigned int i;
|
||||||
|
|
||||||
|
pair->input = true;
|
||||||
|
|
||||||
|
g_value_init( &pair->value, VIPS_TYPE_ARRAY_DOUBLE );
|
||||||
|
vips_value_set_array_double( &pair->value, NULL,
|
||||||
|
static_cast< int >( value.size() ) );
|
||||||
|
array = vips_value_get_array_double( &pair->value, NULL );
|
||||||
|
|
||||||
|
for( i = 0; i < value.size(); i++ )
|
||||||
|
array[i] = value[i];
|
||||||
|
|
||||||
|
options.push_back( pair );
|
||||||
|
|
||||||
|
return( this );
|
||||||
|
}
|
||||||
|
|
||||||
|
// input image array
|
||||||
|
VOption *
|
||||||
|
VOption::set( const char *name, std::vector<VImage> value )
|
||||||
|
{
|
||||||
|
Pair *pair = new Pair( name );
|
||||||
|
|
||||||
|
VipsImage **array;
|
||||||
|
unsigned int i;
|
||||||
|
|
||||||
|
pair->input = true;
|
||||||
|
|
||||||
|
g_value_init( &pair->value, VIPS_TYPE_ARRAY_IMAGE );
|
||||||
|
vips_value_set_array_image( &pair->value,
|
||||||
|
static_cast< int >( value.size() ) );
|
||||||
|
array = vips_value_get_array_image( &pair->value, NULL );
|
||||||
|
|
||||||
|
for( i = 0; i < value.size(); i++ ) {
|
||||||
|
VipsImage *vips_image = value[i].get_image();
|
||||||
|
|
||||||
|
array[i] = vips_image;
|
||||||
|
g_object_ref( vips_image );
|
||||||
|
}
|
||||||
|
|
||||||
|
options.push_back( pair );
|
||||||
|
|
||||||
|
return( this );
|
||||||
|
}
|
||||||
|
|
||||||
|
// input blob
|
||||||
|
VOption *
|
||||||
|
VOption::set( const char *name, VipsBlob *value )
|
||||||
|
{
|
||||||
|
Pair *pair = new Pair( name );
|
||||||
|
|
||||||
|
pair->input = true;
|
||||||
|
g_value_init( &pair->value, VIPS_TYPE_BLOB );
|
||||||
|
g_value_set_boxed( &pair->value, value );
|
||||||
|
options.push_back( pair );
|
||||||
|
|
||||||
|
return( this );
|
||||||
|
}
|
||||||
|
|
||||||
|
// output bool
|
||||||
|
VOption *
|
||||||
|
VOption::set( const char *name, bool *value )
|
||||||
|
{
|
||||||
|
Pair *pair = new Pair( name );
|
||||||
|
|
||||||
|
pair->input = false;
|
||||||
|
pair->vbool = value;
|
||||||
|
g_value_init( &pair->value, G_TYPE_BOOLEAN );
|
||||||
|
|
||||||
|
options.push_back( pair );
|
||||||
|
|
||||||
|
return( this );
|
||||||
|
}
|
||||||
|
|
||||||
|
// output int
|
||||||
|
VOption *
|
||||||
|
VOption::set( const char *name, int *value )
|
||||||
|
{
|
||||||
|
Pair *pair = new Pair( name );
|
||||||
|
|
||||||
|
pair->input = false;
|
||||||
|
pair->vint = value;
|
||||||
|
g_value_init( &pair->value, G_TYPE_INT );
|
||||||
|
|
||||||
|
options.push_back( pair );
|
||||||
|
|
||||||
|
return( this );
|
||||||
|
}
|
||||||
|
|
||||||
|
// output double
|
||||||
|
VOption *
|
||||||
|
VOption::set( const char *name, double *value )
|
||||||
|
{
|
||||||
|
Pair *pair = new Pair( name );
|
||||||
|
|
||||||
|
pair->input = false;
|
||||||
|
pair->vdouble = value;
|
||||||
|
g_value_init( &pair->value, G_TYPE_DOUBLE );
|
||||||
|
|
||||||
|
options.push_back( pair );
|
||||||
|
|
||||||
|
return( this );
|
||||||
|
}
|
||||||
|
|
||||||
|
// output image
|
||||||
|
VOption *
|
||||||
|
VOption::set( const char *name, VImage *value )
|
||||||
|
{
|
||||||
|
Pair *pair = new Pair( name );
|
||||||
|
|
||||||
|
pair->input = false;
|
||||||
|
pair->vimage = value;
|
||||||
|
g_value_init( &pair->value, VIPS_TYPE_IMAGE );
|
||||||
|
|
||||||
|
options.push_back( pair );
|
||||||
|
|
||||||
|
return( this );
|
||||||
|
}
|
||||||
|
|
||||||
|
// output doublearray
|
||||||
|
VOption *
|
||||||
|
VOption::set( const char *name, std::vector<double> *value )
|
||||||
|
{
|
||||||
|
Pair *pair = new Pair( name );
|
||||||
|
|
||||||
|
pair->input = false;
|
||||||
|
pair->vvector = value;
|
||||||
|
g_value_init( &pair->value, VIPS_TYPE_ARRAY_DOUBLE );
|
||||||
|
|
||||||
|
options.push_back( pair );
|
||||||
|
|
||||||
|
return( this );
|
||||||
|
}
|
||||||
|
|
||||||
|
// output blob
|
||||||
|
VOption *
|
||||||
|
VOption::set( const char *name, VipsBlob **value )
|
||||||
|
{
|
||||||
|
Pair *pair = new Pair( name );
|
||||||
|
|
||||||
|
pair->input = false;
|
||||||
|
pair->vblob = value;
|
||||||
|
g_value_init( &pair->value, VIPS_TYPE_BLOB );
|
||||||
|
|
||||||
|
options.push_back( pair );
|
||||||
|
|
||||||
|
return( this );
|
||||||
|
}
|
||||||
|
|
||||||
|
// just g_object_set_property(), except we allow set enum from string
|
||||||
|
static void
|
||||||
|
set_property( VipsObject *object, const char *name, const GValue *value )
|
||||||
|
{
|
||||||
|
VipsObjectClass *object_class = VIPS_OBJECT_GET_CLASS( object );
|
||||||
|
GType type = G_VALUE_TYPE( value );
|
||||||
|
|
||||||
|
GParamSpec *pspec;
|
||||||
|
VipsArgumentClass *argument_class;
|
||||||
|
VipsArgumentInstance *argument_instance;
|
||||||
|
|
||||||
|
if( vips_object_get_argument( object, name,
|
||||||
|
&pspec, &argument_class, &argument_instance ) ) {
|
||||||
|
vips_warn( NULL, "%s", vips_error_buffer() );
|
||||||
|
vips_error_clear();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if( G_IS_PARAM_SPEC_ENUM( pspec ) &&
|
||||||
|
type == G_TYPE_STRING ) {
|
||||||
|
GType pspec_type = G_PARAM_SPEC_VALUE_TYPE( pspec );
|
||||||
|
|
||||||
|
int enum_value;
|
||||||
|
GValue value2 = { 0 };
|
||||||
|
|
||||||
|
if( (enum_value = vips_enum_from_nick( object_class->nickname,
|
||||||
|
pspec_type, g_value_get_string( value ) )) < 0 ) {
|
||||||
|
vips_warn( NULL, "%s", vips_error_buffer() );
|
||||||
|
vips_error_clear();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
g_value_init( &value2, pspec_type );
|
||||||
|
g_value_set_enum( &value2, enum_value );
|
||||||
|
g_object_set_property( G_OBJECT( object ), name, &value2 );
|
||||||
|
g_value_unset( &value2 );
|
||||||
|
}
|
||||||
|
else
|
||||||
|
g_object_set_property( G_OBJECT( object ), name, value );
|
||||||
|
}
|
||||||
|
|
||||||
|
// walk the options and set props on the operation
|
||||||
|
void
|
||||||
|
VOption::set_operation( VipsOperation *operation )
|
||||||
|
{
|
||||||
|
std::list<Pair *>::iterator i;
|
||||||
|
|
||||||
|
for( i = options.begin(); i != options.end(); ++i )
|
||||||
|
if( (*i)->input ) {
|
||||||
|
#ifdef VIPS_DEBUG_VERBOSE
|
||||||
|
printf( "set_operation: " );
|
||||||
|
vips_object_print_name( VIPS_OBJECT( operation ) );
|
||||||
|
char *str_value = g_strdup_value_contents( &(*i)->value );
|
||||||
|
printf( ".%s = %s\n", (*i)->name, str_value );
|
||||||
|
g_free( str_value );
|
||||||
|
#endif /*VIPS_DEBUG_VERBOSE*/
|
||||||
|
|
||||||
|
set_property( VIPS_OBJECT( operation ),
|
||||||
|
(*i)->name, &(*i)->value );
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// walk the options and fetch any requested outputs
|
||||||
|
void
|
||||||
|
VOption::get_operation( VipsOperation *operation )
|
||||||
|
{
|
||||||
|
std::list<Pair *>::iterator i;
|
||||||
|
|
||||||
|
for( i = options.begin(); i != options.end(); ++i )
|
||||||
|
if( ! (*i)->input ) {
|
||||||
|
const char *name = (*i)->name;
|
||||||
|
|
||||||
|
g_object_get_property( G_OBJECT( operation ),
|
||||||
|
name, &(*i)->value );
|
||||||
|
|
||||||
|
#ifdef VIPS_DEBUG_VERBOSE
|
||||||
|
printf( "get_operation: " );
|
||||||
|
vips_object_print_name( VIPS_OBJECT( operation ) );
|
||||||
|
char *str_value = g_strdup_value_contents(
|
||||||
|
&(*i)->value );
|
||||||
|
printf( ".%s = %s\n", name, str_value );
|
||||||
|
g_free( str_value );
|
||||||
|
#endif /*VIPS_DEBUG_VERBOSE*/
|
||||||
|
|
||||||
|
GValue *value = &(*i)->value;
|
||||||
|
GType type = G_VALUE_TYPE( value );
|
||||||
|
|
||||||
|
if( type == VIPS_TYPE_IMAGE ) {
|
||||||
|
// rebox object
|
||||||
|
VipsImage *image = VIPS_IMAGE(
|
||||||
|
g_value_get_object( value ) );
|
||||||
|
*((*i)->vimage) = VImage( image );
|
||||||
|
}
|
||||||
|
else if( type == G_TYPE_INT )
|
||||||
|
*((*i)->vint) = g_value_get_int( value );
|
||||||
|
else if( type == G_TYPE_BOOLEAN )
|
||||||
|
*((*i)->vbool) = g_value_get_boolean( value );
|
||||||
|
else if( type == G_TYPE_DOUBLE )
|
||||||
|
*((*i)->vdouble) = g_value_get_double( value );
|
||||||
|
else if( type == VIPS_TYPE_ARRAY_DOUBLE ) {
|
||||||
|
int length;
|
||||||
|
double *array =
|
||||||
|
vips_value_get_array_double( value,
|
||||||
|
&length );
|
||||||
|
int j;
|
||||||
|
|
||||||
|
((*i)->vvector)->resize( length );
|
||||||
|
for( j = 0; j < length; j++ )
|
||||||
|
(*((*i)->vvector))[j] = array[j];
|
||||||
|
}
|
||||||
|
else if( type == VIPS_TYPE_BLOB ) {
|
||||||
|
// our caller gets a reference
|
||||||
|
*((*i)->vblob) =
|
||||||
|
(VipsBlob *) g_value_dup_boxed( value );
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
VImage::call_option_string( const char *operation_name,
|
||||||
|
const char *option_string, VOption *options )
|
||||||
|
{
|
||||||
|
VipsOperation *operation;
|
||||||
|
|
||||||
|
VIPS_DEBUG_MSG( "vips_call_by_name: starting for %s ...\n",
|
||||||
|
operation_name );
|
||||||
|
|
||||||
|
if( !(operation = vips_operation_new( operation_name )) ) {
|
||||||
|
if( options )
|
||||||
|
delete options;
|
||||||
|
throw( VError() );
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Set str options before vargs options, so the user can't
|
||||||
|
* override things we set deliberately.
|
||||||
|
*/
|
||||||
|
if( option_string &&
|
||||||
|
vips_object_set_from_string( VIPS_OBJECT( operation ),
|
||||||
|
option_string ) ) {
|
||||||
|
vips_object_unref_outputs( VIPS_OBJECT( operation ) );
|
||||||
|
g_object_unref( operation );
|
||||||
|
delete options;
|
||||||
|
throw( VError() );
|
||||||
|
}
|
||||||
|
|
||||||
|
if( options )
|
||||||
|
options->set_operation( operation );
|
||||||
|
|
||||||
|
/* Build from cache.
|
||||||
|
*/
|
||||||
|
if( vips_cache_operation_buildp( &operation ) ) {
|
||||||
|
vips_object_unref_outputs( VIPS_OBJECT( operation ) );
|
||||||
|
delete options;
|
||||||
|
throw( VError() );
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Walk args again, writing output.
|
||||||
|
*/
|
||||||
|
if( options )
|
||||||
|
options->get_operation( operation );
|
||||||
|
|
||||||
|
/* We're done with options!
|
||||||
|
*/
|
||||||
|
delete options;
|
||||||
|
|
||||||
|
/* The operation we have built should now have been reffed by
|
||||||
|
* one of its arguments or have finished its work. Either
|
||||||
|
* way, we can unref.
|
||||||
|
*/
|
||||||
|
g_object_unref( operation );
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
VImage::call( const char *operation_name, VOption *options )
|
||||||
|
{
|
||||||
|
call_option_string( operation_name, NULL, options );
|
||||||
|
}
|
||||||
|
|
||||||
|
VImage
|
||||||
|
VImage::new_from_file( const char *name, VOption *options )
|
||||||
|
{
|
||||||
|
char filename[VIPS_PATH_MAX];
|
||||||
|
char option_string[VIPS_PATH_MAX];
|
||||||
|
const char *operation_name;
|
||||||
|
|
||||||
|
VImage out;
|
||||||
|
|
||||||
|
vips__filename_split8( name, filename, option_string );
|
||||||
|
if( !(operation_name = vips_foreign_find_load( filename )) ) {
|
||||||
|
delete options;
|
||||||
|
throw VError();
|
||||||
|
}
|
||||||
|
|
||||||
|
call_option_string( operation_name, option_string,
|
||||||
|
(options ? options : VImage::option())->
|
||||||
|
set( "filename", filename )->
|
||||||
|
set( "out", &out ) );
|
||||||
|
|
||||||
|
return( out );
|
||||||
|
}
|
||||||
|
|
||||||
|
VImage
|
||||||
|
VImage::new_from_buffer( void *buf, size_t len, const char *option_string,
|
||||||
|
VOption *options )
|
||||||
|
{
|
||||||
|
const char *operation_name;
|
||||||
|
VipsBlob *blob;
|
||||||
|
VImage out;
|
||||||
|
|
||||||
|
if( !(operation_name = vips_foreign_find_load_buffer( buf, len )) ) {
|
||||||
|
delete options;
|
||||||
|
throw( VError() );
|
||||||
|
}
|
||||||
|
|
||||||
|
/* We don't take a copy of the data or free it.
|
||||||
|
*/
|
||||||
|
blob = vips_blob_new( NULL, buf, len );
|
||||||
|
options = (options ? options : VImage::option())->
|
||||||
|
set( "buffer", blob )->
|
||||||
|
set( "out", &out );
|
||||||
|
vips_area_unref( VIPS_AREA( blob ) );
|
||||||
|
|
||||||
|
call_option_string( operation_name, option_string, options );
|
||||||
|
|
||||||
|
return( out );
|
||||||
|
}
|
||||||
|
|
||||||
|
VImage
|
||||||
|
VImage::new_from_image( std::vector<double> pixel )
|
||||||
|
{
|
||||||
|
VImage onepx = VImage::black( 1, 1,
|
||||||
|
VImage::option()->set( "bands", bands() ) );
|
||||||
|
|
||||||
|
onepx = onepx.linear( to_vectorv( 1, 1.0 ), pixel ).cast( format() );
|
||||||
|
|
||||||
|
VImage big = onepx.embed( 0, 0, width(), height(),
|
||||||
|
VImage::option()->set( "extend", VIPS_EXTEND_COPY ) );
|
||||||
|
|
||||||
|
big = big.copy(
|
||||||
|
VImage::option()->
|
||||||
|
set( "interpretation", interpretation() )->
|
||||||
|
set( "xres", xres() )->
|
||||||
|
set( "yres", yres() )->
|
||||||
|
set( "xoffset", xres() )->
|
||||||
|
set( "yoffset", yres() ) );
|
||||||
|
|
||||||
|
return( big );
|
||||||
|
}
|
||||||
|
|
||||||
|
VImage
|
||||||
|
VImage::new_from_image( double pixel )
|
||||||
|
{
|
||||||
|
return( new_from_image( to_vectorv( 1, pixel ) ) );
|
||||||
|
}
|
||||||
|
|
||||||
|
VImage
|
||||||
|
VImage::new_matrix( int width, int height )
|
||||||
|
{
|
||||||
|
return( VImage( vips_image_new_matrix( width, height ) ) );
|
||||||
|
}
|
||||||
|
|
||||||
|
VImage
|
||||||
|
VImage::new_matrixv( int width, int height, ... )
|
||||||
|
{
|
||||||
|
VImage matrix = new_matrix( width, height );
|
||||||
|
VipsImage *vips_matrix = matrix.get_image();
|
||||||
|
|
||||||
|
va_list ap;
|
||||||
|
|
||||||
|
va_start( ap, height );
|
||||||
|
for( int y = 0; y < height; y++ )
|
||||||
|
for( int x = 0; x < width; x++ )
|
||||||
|
*VIPS_MATRIX( vips_matrix, x, y ) =
|
||||||
|
va_arg( ap, double );
|
||||||
|
va_end( ap );
|
||||||
|
|
||||||
|
return( matrix );
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
VImage::write_to_file( const char *name, VOption *options )
|
||||||
|
{
|
||||||
|
char filename[VIPS_PATH_MAX];
|
||||||
|
char option_string[VIPS_PATH_MAX];
|
||||||
|
const char *operation_name;
|
||||||
|
|
||||||
|
vips__filename_split8( name, filename, option_string );
|
||||||
|
if( !(operation_name = vips_foreign_find_save( filename )) ) {
|
||||||
|
delete options;
|
||||||
|
throw VError();
|
||||||
|
}
|
||||||
|
|
||||||
|
call_option_string( operation_name, option_string,
|
||||||
|
(options ? options : VImage::option())->
|
||||||
|
set( "in", *this )->
|
||||||
|
set( "filename", filename ) );
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
VImage::write_to_buffer( const char *suffix, void **buf, size_t *size,
|
||||||
|
VOption *options )
|
||||||
|
{
|
||||||
|
char filename[VIPS_PATH_MAX];
|
||||||
|
char option_string[VIPS_PATH_MAX];
|
||||||
|
const char *operation_name;
|
||||||
|
VipsBlob *blob;
|
||||||
|
|
||||||
|
vips__filename_split8( suffix, filename, option_string );
|
||||||
|
if( !(operation_name = vips_foreign_find_save_buffer( filename )) ) {
|
||||||
|
delete options;
|
||||||
|
throw VError();
|
||||||
|
}
|
||||||
|
|
||||||
|
call_option_string( operation_name, option_string,
|
||||||
|
(options ? options : VImage::option())->
|
||||||
|
set( "in", *this )->
|
||||||
|
set( "buffer", &blob ) );
|
||||||
|
|
||||||
|
if( blob ) {
|
||||||
|
if( buf ) {
|
||||||
|
*buf = VIPS_AREA( blob )->data;
|
||||||
|
VIPS_AREA( blob )->free_fn = NULL;
|
||||||
|
}
|
||||||
|
if( size )
|
||||||
|
*size = VIPS_AREA( blob )->length;
|
||||||
|
|
||||||
|
vips_area_unref( VIPS_AREA( blob ) );
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#include "vips-operators.cpp"
|
||||||
|
|
||||||
|
std::vector<VImage>
|
||||||
|
VImage::bandsplit( VOption *options )
|
||||||
|
{
|
||||||
|
std::vector<VImage> b;
|
||||||
|
|
||||||
|
for( int i = 0; i < bands(); i++ )
|
||||||
|
b.push_back( extract_band( i ) );
|
||||||
|
|
||||||
|
return( b );
|
||||||
|
}
|
||||||
|
|
||||||
|
VImage
|
||||||
|
VImage::bandjoin( VImage other, VOption *options )
|
||||||
|
{
|
||||||
|
VImage v[2] = { *this, other };
|
||||||
|
std::vector<VImage> vec( v, v + VIPS_NUMBER( v ) );
|
||||||
|
|
||||||
|
return( bandjoin( vec, options ) );
|
||||||
|
}
|
||||||
|
|
||||||
|
std::complex<double>
|
||||||
|
VImage::minpos( VOption *options )
|
||||||
|
{
|
||||||
|
double x, y;
|
||||||
|
|
||||||
|
(void) min(
|
||||||
|
(options ? options : VImage::option()) ->
|
||||||
|
set( "x", &x ) ->
|
||||||
|
set( "y", &y ) );
|
||||||
|
|
||||||
|
return( std::complex<double>( x, y ) );
|
||||||
|
}
|
||||||
|
|
||||||
|
std::complex<double>
|
||||||
|
VImage::maxpos( VOption *options )
|
||||||
|
{
|
||||||
|
double x, y;
|
||||||
|
|
||||||
|
(void) max(
|
||||||
|
(options ? options : VImage::option()) ->
|
||||||
|
set( "x", &x ) ->
|
||||||
|
set( "y", &y ) );
|
||||||
|
|
||||||
|
return( std::complex<double>( x, y ) );
|
||||||
|
}
|
||||||
|
|
||||||
|
VIPS_NAMESPACE_END
|
||||||
76
src/libvips/cplusplus/VInterpolate.cpp
Normal file
@@ -0,0 +1,76 @@
|
|||||||
|
/* Object part of VInterpolate class
|
||||||
|
*/
|
||||||
|
|
||||||
|
/*
|
||||||
|
|
||||||
|
Copyright (C) 1991-2001 The National Gallery
|
||||||
|
|
||||||
|
This program is free software; you can redistribute it and/or modify
|
||||||
|
it under the terms of the GNU Lesser General Public License as published by
|
||||||
|
the Free Software Foundation; either version 2 of the License, or
|
||||||
|
(at your option) any later version.
|
||||||
|
|
||||||
|
This program is distributed in the hope that it will be useful,
|
||||||
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
GNU Lesser General Public License for more details.
|
||||||
|
|
||||||
|
You should have received a copy of the GNU Lesser General Public License
|
||||||
|
along with this program; if not, write to the Free Software
|
||||||
|
Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
|
||||||
|
02110-1301 USA
|
||||||
|
|
||||||
|
*/
|
||||||
|
|
||||||
|
/*
|
||||||
|
|
||||||
|
These files are distributed with VIPS - http://www.vips.ecs.soton.ac.uk
|
||||||
|
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifdef HAVE_CONFIG_H
|
||||||
|
#include <config.h>
|
||||||
|
#endif /*HAVE_CONFIG_H*/
|
||||||
|
#include <vips/intl.h>
|
||||||
|
|
||||||
|
#include <vips/vips8>
|
||||||
|
|
||||||
|
#include <vips/debug.h>
|
||||||
|
|
||||||
|
/*
|
||||||
|
#define VIPS_DEBUG
|
||||||
|
#define VIPS_DEBUG_VERBOSE
|
||||||
|
*/
|
||||||
|
|
||||||
|
VIPS_NAMESPACE_START
|
||||||
|
|
||||||
|
VInterpolate
|
||||||
|
VInterpolate::new_from_name( const char *name, VOption *options )
|
||||||
|
{
|
||||||
|
VipsInterpolate *interp;
|
||||||
|
|
||||||
|
if( !(interp = vips_interpolate_new( name )) ) {
|
||||||
|
delete options;
|
||||||
|
throw VError();
|
||||||
|
}
|
||||||
|
delete options;
|
||||||
|
|
||||||
|
VInterpolate out( interp );
|
||||||
|
|
||||||
|
return( out );
|
||||||
|
}
|
||||||
|
|
||||||
|
VOption *
|
||||||
|
VOption::set( const char *name, VInterpolate value )
|
||||||
|
{
|
||||||
|
Pair *pair = new Pair( name );
|
||||||
|
|
||||||
|
pair->input = true;
|
||||||
|
g_value_init( &pair->value, VIPS_TYPE_INTERPOLATE );
|
||||||
|
g_value_set_object( &pair->value, value.get_interpolate() );
|
||||||
|
options.push_back( pair );
|
||||||
|
|
||||||
|
return( this );
|
||||||
|
}
|
||||||
|
|
||||||
|
VIPS_NAMESPACE_END
|
||||||
2738
src/libvips/cplusplus/vips-operators.cpp
Normal file
180
src/metadata.cc
Executable file → Normal file
@@ -1,5 +1,5 @@
|
|||||||
#include <node.h>
|
#include <node.h>
|
||||||
#include <vips/vips.h>
|
#include <vips/vips8>
|
||||||
|
|
||||||
#include "nan.h"
|
#include "nan.h"
|
||||||
|
|
||||||
@@ -16,12 +16,31 @@ using v8::Boolean;
|
|||||||
using v8::Function;
|
using v8::Function;
|
||||||
using v8::Exception;
|
using v8::Exception;
|
||||||
|
|
||||||
|
using Nan::AsyncQueueWorker;
|
||||||
|
using Nan::AsyncWorker;
|
||||||
|
using Nan::Callback;
|
||||||
|
using Nan::HandleScope;
|
||||||
|
using Nan::Utf8String;
|
||||||
|
using Nan::Has;
|
||||||
|
using Nan::Get;
|
||||||
|
using Nan::Set;
|
||||||
|
using Nan::New;
|
||||||
|
using Nan::NewBuffer;
|
||||||
|
using Nan::Null;
|
||||||
|
using Nan::Error;
|
||||||
|
|
||||||
|
using vips::VImage;
|
||||||
|
using vips::VError;
|
||||||
|
|
||||||
using sharp::ImageType;
|
using sharp::ImageType;
|
||||||
|
using sharp::ImageTypeId;
|
||||||
using sharp::DetermineImageType;
|
using sharp::DetermineImageType;
|
||||||
using sharp::InitImage;
|
|
||||||
using sharp::HasProfile;
|
using sharp::HasProfile;
|
||||||
using sharp::HasAlpha;
|
using sharp::HasAlpha;
|
||||||
using sharp::ExifOrientation;
|
using sharp::ExifOrientation;
|
||||||
|
using sharp::HasDensity;
|
||||||
|
using sharp::GetDensity;
|
||||||
|
using sharp::FreeCallback;
|
||||||
using sharp::counterQueue;
|
using sharp::counterQueue;
|
||||||
|
|
||||||
struct MetadataBaton {
|
struct MetadataBaton {
|
||||||
@@ -35,6 +54,7 @@ struct MetadataBaton {
|
|||||||
int height;
|
int height;
|
||||||
std::string space;
|
std::string space;
|
||||||
int channels;
|
int channels;
|
||||||
|
int density;
|
||||||
bool hasProfile;
|
bool hasProfile;
|
||||||
bool hasAlpha;
|
bool hasAlpha;
|
||||||
int orientation;
|
int orientation;
|
||||||
@@ -46,25 +66,20 @@ struct MetadataBaton {
|
|||||||
|
|
||||||
MetadataBaton():
|
MetadataBaton():
|
||||||
bufferInLength(0),
|
bufferInLength(0),
|
||||||
|
density(0),
|
||||||
orientation(0),
|
orientation(0),
|
||||||
exifLength(0),
|
exifLength(0),
|
||||||
iccLength(0) {}
|
iccLength(0) {}
|
||||||
};
|
};
|
||||||
|
|
||||||
/*
|
class MetadataWorker : public AsyncWorker {
|
||||||
Delete input char[] buffer and notify V8 of memory deallocation
|
|
||||||
Used as the callback function for the "postclose" signal
|
|
||||||
*/
|
|
||||||
static void DeleteBuffer(VipsObject *object, char *buffer) {
|
|
||||||
if (buffer != NULL) {
|
|
||||||
delete[] buffer;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
class MetadataWorker : public NanAsyncWorker {
|
|
||||||
|
|
||||||
public:
|
public:
|
||||||
MetadataWorker(NanCallback *callback, MetadataBaton *baton) : NanAsyncWorker(callback), baton(baton) {}
|
MetadataWorker(Callback *callback, MetadataBaton *baton, const Local<Object> &bufferIn) :
|
||||||
|
AsyncWorker(callback), baton(baton) {
|
||||||
|
if (baton->bufferInLength > 0) {
|
||||||
|
SaveToPersistent("bufferIn", bufferIn);
|
||||||
|
}
|
||||||
|
}
|
||||||
~MetadataWorker() {}
|
~MetadataWorker() {}
|
||||||
|
|
||||||
void Execute() {
|
void Execute() {
|
||||||
@@ -72,79 +87,65 @@ class MetadataWorker : public NanAsyncWorker {
|
|||||||
g_atomic_int_dec_and_test(&counterQueue);
|
g_atomic_int_dec_and_test(&counterQueue);
|
||||||
|
|
||||||
ImageType imageType = ImageType::UNKNOWN;
|
ImageType imageType = ImageType::UNKNOWN;
|
||||||
VipsImage *image = NULL;
|
VImage image;
|
||||||
if (baton->bufferInLength > 0) {
|
if (baton->bufferInLength > 0) {
|
||||||
// From buffer
|
// From buffer
|
||||||
imageType = DetermineImageType(baton->bufferIn, baton->bufferInLength);
|
imageType = DetermineImageType(baton->bufferIn, baton->bufferInLength);
|
||||||
if (imageType != ImageType::UNKNOWN) {
|
if (imageType != ImageType::UNKNOWN) {
|
||||||
image = InitImage(baton->bufferIn, baton->bufferInLength, VIPS_ACCESS_RANDOM);
|
try {
|
||||||
if (image != NULL) {
|
image = VImage::new_from_buffer(baton->bufferIn, baton->bufferInLength, nullptr);
|
||||||
// Listen for "postclose" signal to delete input buffer
|
} catch (...) {
|
||||||
g_signal_connect(image, "postclose", G_CALLBACK(DeleteBuffer), baton->bufferIn);
|
|
||||||
} else {
|
|
||||||
(baton->err).append("Input buffer has corrupt header");
|
(baton->err).append("Input buffer has corrupt header");
|
||||||
imageType = ImageType::UNKNOWN;
|
imageType = ImageType::UNKNOWN;
|
||||||
DeleteBuffer(NULL, baton->bufferIn);
|
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
(baton->err).append("Input buffer contains unsupported image format");
|
(baton->err).append("Input buffer contains unsupported image format");
|
||||||
DeleteBuffer(NULL, baton->bufferIn);
|
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
// From file
|
// From file
|
||||||
imageType = DetermineImageType(baton->fileIn.c_str());
|
imageType = DetermineImageType(baton->fileIn.data());
|
||||||
if (imageType != ImageType::UNKNOWN) {
|
if (imageType != ImageType::UNKNOWN) {
|
||||||
image = InitImage(baton->fileIn.c_str(), VIPS_ACCESS_RANDOM);
|
try {
|
||||||
if (image == NULL) {
|
image = VImage::new_from_file(baton->fileIn.data());
|
||||||
|
} catch (...) {
|
||||||
(baton->err).append("Input file has corrupt header");
|
(baton->err).append("Input file has corrupt header");
|
||||||
imageType = ImageType::UNKNOWN;
|
imageType = ImageType::UNKNOWN;
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
(baton->err).append("Input file is of an unsupported image format");
|
(baton->err).append("Input file is missing or of an unsupported image format");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (image != NULL && imageType != ImageType::UNKNOWN) {
|
if (imageType != ImageType::UNKNOWN) {
|
||||||
// Image type
|
// Image type
|
||||||
switch (imageType) {
|
baton->format = ImageTypeId(imageType);
|
||||||
case ImageType::JPEG: baton->format = "jpeg"; break;
|
|
||||||
case ImageType::PNG: baton->format = "png"; break;
|
|
||||||
case ImageType::WEBP: baton->format = "webp"; break;
|
|
||||||
case ImageType::TIFF: baton->format = "tiff"; break;
|
|
||||||
case ImageType::MAGICK: baton->format = "magick"; break;
|
|
||||||
case ImageType::OPENSLIDE: baton->format = "openslide"; break;
|
|
||||||
case ImageType::UNKNOWN: break;
|
|
||||||
}
|
|
||||||
// VipsImage attributes
|
// VipsImage attributes
|
||||||
baton->width = image->Xsize;
|
baton->width = image.width();
|
||||||
baton->height = image->Ysize;
|
baton->height = image.height();
|
||||||
baton->space = vips_enum_nick(VIPS_TYPE_INTERPRETATION, image->Type);
|
baton->space = vips_enum_nick(VIPS_TYPE_INTERPRETATION, image.interpretation());
|
||||||
baton->channels = image->Bands;
|
baton->channels = image.bands();
|
||||||
|
if (HasDensity(image)) {
|
||||||
|
baton->density = GetDensity(image);
|
||||||
|
}
|
||||||
baton->hasProfile = HasProfile(image);
|
baton->hasProfile = HasProfile(image);
|
||||||
// Derived attributes
|
// Derived attributes
|
||||||
baton->hasAlpha = HasAlpha(image);
|
baton->hasAlpha = HasAlpha(image);
|
||||||
baton->orientation = ExifOrientation(image);
|
baton->orientation = ExifOrientation(image);
|
||||||
// EXIF
|
// EXIF
|
||||||
if (vips_image_get_typeof(image, VIPS_META_EXIF_NAME) == VIPS_TYPE_BLOB) {
|
if (image.get_typeof(VIPS_META_EXIF_NAME) == VIPS_TYPE_BLOB) {
|
||||||
void* exif;
|
|
||||||
size_t exifLength;
|
size_t exifLength;
|
||||||
if (!vips_image_get_blob(image, VIPS_META_EXIF_NAME, &exif, &exifLength)) {
|
void const *exif = image.get_blob(VIPS_META_EXIF_NAME, &exifLength);
|
||||||
baton->exifLength = exifLength;
|
baton->exif = static_cast<char*>(g_malloc(exifLength));
|
||||||
baton->exif = new char[exifLength];
|
memcpy(baton->exif, exif, exifLength);
|
||||||
memcpy(baton->exif, exif, exifLength);
|
baton->exifLength = exifLength;
|
||||||
}
|
|
||||||
}
|
}
|
||||||
// ICC profile
|
// ICC profile
|
||||||
if (vips_image_get_typeof(image, VIPS_META_ICC_NAME) == VIPS_TYPE_BLOB) {
|
if (image.get_typeof(VIPS_META_ICC_NAME) == VIPS_TYPE_BLOB) {
|
||||||
void* icc;
|
|
||||||
size_t iccLength;
|
size_t iccLength;
|
||||||
if (!vips_image_get_blob(image, VIPS_META_ICC_NAME, &icc, &iccLength)) {
|
void const *icc = image.get_blob(VIPS_META_ICC_NAME, &iccLength);
|
||||||
baton->iccLength = iccLength;
|
baton->icc = static_cast<char*>(g_malloc(iccLength));
|
||||||
baton->icc = new char[iccLength];
|
memcpy(baton->icc, icc, iccLength);
|
||||||
memcpy(baton->icc, icc, iccLength);
|
baton->iccLength = iccLength;
|
||||||
}
|
|
||||||
}
|
}
|
||||||
// Drop image reference
|
|
||||||
g_object_unref(image);
|
|
||||||
}
|
}
|
||||||
// Clean up
|
// Clean up
|
||||||
vips_error_clear();
|
vips_error_clear();
|
||||||
@@ -152,33 +153,47 @@ class MetadataWorker : public NanAsyncWorker {
|
|||||||
}
|
}
|
||||||
|
|
||||||
void HandleOKCallback () {
|
void HandleOKCallback () {
|
||||||
NanScope();
|
HandleScope();
|
||||||
|
|
||||||
Handle<Value> argv[2] = { NanNull(), NanNull() };
|
Local<Value> argv[2] = { Null(), Null() };
|
||||||
if (!baton->err.empty()) {
|
if (!baton->err.empty()) {
|
||||||
// Error
|
// Error
|
||||||
argv[0] = Exception::Error(NanNew<String>(baton->err.data(), baton->err.size()));
|
argv[0] = Error(baton->err.data());
|
||||||
} else {
|
} else {
|
||||||
// Metadata Object
|
// Metadata Object
|
||||||
Local<Object> info = NanNew<Object>();
|
Local<Object> info = New<Object>();
|
||||||
info->Set(NanNew<String>("format"), NanNew<String>(baton->format));
|
Set(info, New("format").ToLocalChecked(), New<String>(baton->format).ToLocalChecked());
|
||||||
info->Set(NanNew<String>("width"), NanNew<Number>(baton->width));
|
Set(info, New("width").ToLocalChecked(), New<Number>(baton->width));
|
||||||
info->Set(NanNew<String>("height"), NanNew<Number>(baton->height));
|
Set(info, New("height").ToLocalChecked(), New<Number>(baton->height));
|
||||||
info->Set(NanNew<String>("space"), NanNew<String>(baton->space));
|
Set(info, New("space").ToLocalChecked(), New<String>(baton->space).ToLocalChecked());
|
||||||
info->Set(NanNew<String>("channels"), NanNew<Number>(baton->channels));
|
Set(info, New("channels").ToLocalChecked(), New<Number>(baton->channels));
|
||||||
info->Set(NanNew<String>("hasProfile"), NanNew<Boolean>(baton->hasProfile));
|
if (baton->density > 0) {
|
||||||
info->Set(NanNew<String>("hasAlpha"), NanNew<Boolean>(baton->hasAlpha));
|
Set(info, New("density").ToLocalChecked(), New<Number>(baton->density));
|
||||||
|
}
|
||||||
|
Set(info, New("hasProfile").ToLocalChecked(), New<Boolean>(baton->hasProfile));
|
||||||
|
Set(info, New("hasAlpha").ToLocalChecked(), New<Boolean>(baton->hasAlpha));
|
||||||
if (baton->orientation > 0) {
|
if (baton->orientation > 0) {
|
||||||
info->Set(NanNew<String>("orientation"), NanNew<Number>(baton->orientation));
|
Set(info, New("orientation").ToLocalChecked(), New<Number>(baton->orientation));
|
||||||
}
|
}
|
||||||
if (baton->exifLength > 0) {
|
if (baton->exifLength > 0) {
|
||||||
info->Set(NanNew<String>("exif"), NanBufferUse(baton->exif, baton->exifLength));
|
Set(info,
|
||||||
|
New("exif").ToLocalChecked(),
|
||||||
|
NewBuffer(baton->exif, baton->exifLength, FreeCallback, nullptr).ToLocalChecked()
|
||||||
|
);
|
||||||
}
|
}
|
||||||
if (baton->iccLength > 0) {
|
if (baton->iccLength > 0) {
|
||||||
info->Set(NanNew<String>("icc"), NanBufferUse(baton->icc, baton->iccLength));
|
Set(info,
|
||||||
|
New("icc").ToLocalChecked(),
|
||||||
|
NewBuffer(baton->icc, baton->iccLength, FreeCallback, nullptr).ToLocalChecked()
|
||||||
|
);
|
||||||
}
|
}
|
||||||
argv[1] = info;
|
argv[1] = info;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Dispose of Persistent wrapper around input Buffer so it can be garbage collected
|
||||||
|
if (baton->bufferInLength > 0) {
|
||||||
|
GetFromPersistent("bufferIn");
|
||||||
|
}
|
||||||
delete baton;
|
delete baton;
|
||||||
|
|
||||||
// Return to JavaScript
|
// Return to JavaScript
|
||||||
@@ -193,29 +208,26 @@ class MetadataWorker : public NanAsyncWorker {
|
|||||||
metadata(options, callback)
|
metadata(options, callback)
|
||||||
*/
|
*/
|
||||||
NAN_METHOD(metadata) {
|
NAN_METHOD(metadata) {
|
||||||
NanScope();
|
HandleScope();
|
||||||
|
|
||||||
// V8 objects are converted to non-V8 types held in the baton struct
|
// V8 objects are converted to non-V8 types held in the baton struct
|
||||||
MetadataBaton *baton = new MetadataBaton;
|
MetadataBaton *baton = new MetadataBaton;
|
||||||
Local<Object> options = args[0]->ToObject();
|
Local<Object> options = info[0].As<Object>();
|
||||||
|
|
||||||
// Input filename
|
// Input filename
|
||||||
baton->fileIn = *String::Utf8Value(options->Get(NanNew<String>("fileIn"))->ToString());
|
baton->fileIn = *Utf8String(Get(options, New("fileIn").ToLocalChecked()).ToLocalChecked());
|
||||||
// Input Buffer object
|
// Input Buffer object
|
||||||
if (options->Get(NanNew<String>("bufferIn"))->IsObject()) {
|
Local<Object> bufferIn;
|
||||||
Local<Object> buffer = options->Get(NanNew<String>("bufferIn"))->ToObject();
|
if (node::Buffer::HasInstance(Get(options, New("bufferIn").ToLocalChecked()).ToLocalChecked())) {
|
||||||
// Take a copy of the input Buffer to avoid problems with V8 heap compaction
|
bufferIn = Get(options, New("bufferIn").ToLocalChecked()).ToLocalChecked().As<Object>();
|
||||||
baton->bufferInLength = node::Buffer::Length(buffer);
|
baton->bufferInLength = node::Buffer::Length(bufferIn);
|
||||||
baton->bufferIn = new char[baton->bufferInLength];
|
baton->bufferIn = node::Buffer::Data(bufferIn);
|
||||||
memcpy(baton->bufferIn, node::Buffer::Data(buffer), baton->bufferInLength);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Join queue for worker thread
|
// Join queue for worker thread
|
||||||
NanCallback *callback = new NanCallback(args[1].As<v8::Function>());
|
Callback *callback = new Callback(info[1].As<Function>());
|
||||||
NanAsyncQueueWorker(new MetadataWorker(callback, baton));
|
AsyncQueueWorker(new MetadataWorker(callback, baton, bufferIn));
|
||||||
|
|
||||||
// Increment queued task counter
|
// Increment queued task counter
|
||||||
g_atomic_int_inc(&counterQueue);
|
g_atomic_int_inc(&counterQueue);
|
||||||
|
|
||||||
NanReturnUndefined();
|
|
||||||
}
|
}
|
||||||
|
|||||||
0
src/metadata.h
Executable file → Normal file
406
src/operations.cc
Executable file → Normal file
@@ -1,66 +1,54 @@
|
|||||||
#include <vips/vips.h>
|
#include <algorithm>
|
||||||
|
#include <tuple>
|
||||||
|
#include <vips/vips8>
|
||||||
|
|
||||||
#include "common.h"
|
#include "common.h"
|
||||||
#include "operations.h"
|
#include "operations.h"
|
||||||
|
|
||||||
|
using vips::VImage;
|
||||||
|
using vips::VError;
|
||||||
|
|
||||||
namespace sharp {
|
namespace sharp {
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Alpha composite src over dst
|
Alpha composite src over dst with given gravity.
|
||||||
Assumes alpha channels are already premultiplied and will be unpremultiplied after
|
Assumes alpha channels are already premultiplied and will be unpremultiplied after.
|
||||||
*/
|
*/
|
||||||
int Composite(VipsObject *context, VipsImage *src, VipsImage *dst, VipsImage **out) {
|
VImage Composite(VImage src, VImage dst, const int gravity) {
|
||||||
|
using sharp::CalculateCrop;
|
||||||
using sharp::HasAlpha;
|
using sharp::HasAlpha;
|
||||||
|
|
||||||
// Split src into non-alpha and alpha
|
if (!HasAlpha(src)) {
|
||||||
VipsImage *srcWithoutAlpha;
|
throw VError("Overlay image must have an alpha channel");
|
||||||
if (vips_extract_band(src, &srcWithoutAlpha, 0, "n", src->Bands - 1, NULL))
|
}
|
||||||
return -1;
|
if (!HasAlpha(dst)) {
|
||||||
vips_object_local(context, srcWithoutAlpha);
|
throw VError("Image to be overlaid must have an alpha channel");
|
||||||
VipsImage *srcAlpha;
|
}
|
||||||
if (vips_extract_band(src, &srcAlpha, src->Bands - 1, "n", 1, NULL))
|
if (src.width() > dst.width() || src.height() > dst.height()) {
|
||||||
return -1;
|
throw VError("Overlay image must have same dimensions or smaller");
|
||||||
vips_object_local(context, srcAlpha);
|
|
||||||
|
|
||||||
// Split dst into non-alpha and alpha channels
|
|
||||||
VipsImage *dstWithoutAlpha;
|
|
||||||
VipsImage *dstAlpha;
|
|
||||||
if (HasAlpha(dst)) {
|
|
||||||
// Non-alpha: extract all-but-last channel
|
|
||||||
if (vips_extract_band(dst, &dstWithoutAlpha, 0, "n", dst->Bands - 1, NULL)) {
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
vips_object_local(context, dstWithoutAlpha);
|
|
||||||
// Alpha: Extract last channel
|
|
||||||
if (vips_extract_band(dst, &dstAlpha, dst->Bands - 1, "n", 1, NULL)) {
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
vips_object_local(context, dstAlpha);
|
|
||||||
} else {
|
|
||||||
// Non-alpha: Copy reference
|
|
||||||
dstWithoutAlpha = dst;
|
|
||||||
// Alpha: Use blank, opaque (0xFF) image
|
|
||||||
VipsImage *black;
|
|
||||||
if (vips_black(&black, dst->Xsize, dst->Ysize, NULL)) {
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
vips_object_local(context, black);
|
|
||||||
if (vips_invert(black, &dstAlpha, NULL)) {
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
vips_object_local(context, dstAlpha);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Compute normalized input alpha channels:
|
// Enlarge overlay src, if required
|
||||||
VipsImage *srcAlphaNormalized;
|
if (src.width() < dst.width() || src.height() < dst.height()) {
|
||||||
if (vips_linear1(srcAlpha, &srcAlphaNormalized, 1.0 / 255.0, 0.0, NULL))
|
// Calculate the (left, top) coordinates of the output image within the input image, applying the given gravity.
|
||||||
return -1;
|
int left;
|
||||||
vips_object_local(context, srcAlphaNormalized);
|
int top;
|
||||||
|
std::tie(left, top) = CalculateCrop(dst.width(), dst.height(), src.width(), src.height(), gravity);
|
||||||
|
// Embed onto transparent background
|
||||||
|
std::vector<double> background { 0.0, 0.0, 0.0, 0.0 };
|
||||||
|
src = src.embed(left, top, dst.width(), dst.height(), VImage::option()
|
||||||
|
->set("extend", VIPS_EXTEND_BACKGROUND)
|
||||||
|
->set("background", background)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
VipsImage *dstAlphaNormalized;
|
// Split src into non-alpha and alpha channels
|
||||||
if (vips_linear1(dstAlpha, &dstAlphaNormalized, 1.0 / 255.0, 0.0, NULL))
|
VImage srcWithoutAlpha = src.extract_band(0, VImage::option()->set("n", src.bands() - 1));
|
||||||
return -1;
|
VImage srcAlpha = src[src.bands() - 1] * (1.0 / 255.0);
|
||||||
vips_object_local(context, dstAlphaNormalized);
|
|
||||||
|
// Split dst into non-alpha and alpha channels
|
||||||
|
VImage dstWithoutAlpha = dst.extract_band(0, VImage::option()->set("n", dst.bands() - 1));
|
||||||
|
VImage dstAlpha = dst[dst.bands() - 1] * (1.0 / 255.0);
|
||||||
|
|
||||||
//
|
//
|
||||||
// Compute normalized output alpha channel:
|
// Compute normalized output alpha channel:
|
||||||
@@ -72,22 +60,8 @@ namespace sharp {
|
|||||||
// out_a = src_a + dst_a * (1 - src_a)
|
// out_a = src_a + dst_a * (1 - src_a)
|
||||||
// ^^^^^^^^^^^
|
// ^^^^^^^^^^^
|
||||||
// t0
|
// t0
|
||||||
// ^^^^^^^^^^^^^^^^^^^
|
VImage t0 = srcAlpha.linear(-1.0, 1.0);
|
||||||
// t1
|
VImage outAlphaNormalized = srcAlpha + dstAlpha * t0;
|
||||||
VipsImage *t0;
|
|
||||||
if (vips_linear1(srcAlphaNormalized, &t0, -1.0, 1.0, NULL))
|
|
||||||
return -1;
|
|
||||||
vips_object_local(context, t0);
|
|
||||||
|
|
||||||
VipsImage *t1;
|
|
||||||
if (vips_multiply(dstAlphaNormalized, t0, &t1, NULL))
|
|
||||||
return -1;
|
|
||||||
vips_object_local(context, t1);
|
|
||||||
|
|
||||||
VipsImage *outAlphaNormalized;
|
|
||||||
if (vips_add(srcAlphaNormalized, t1, &outAlphaNormalized, NULL))
|
|
||||||
return -1;
|
|
||||||
vips_object_local(context, outAlphaNormalized);
|
|
||||||
|
|
||||||
//
|
//
|
||||||
// Compute output RGB channels:
|
// Compute output RGB channels:
|
||||||
@@ -101,233 +75,179 @@ namespace sharp {
|
|||||||
// premultiplied RGBA image as reversal of premultiplication is handled
|
// premultiplied RGBA image as reversal of premultiplication is handled
|
||||||
// externally.
|
// externally.
|
||||||
//
|
//
|
||||||
VipsImage *t2;
|
VImage outRGBPremultiplied = srcWithoutAlpha + dstWithoutAlpha * t0;
|
||||||
if (vips_multiply(dstWithoutAlpha, t0, &t2, NULL))
|
|
||||||
return -1;
|
|
||||||
vips_object_local(context, t2);
|
|
||||||
|
|
||||||
VipsImage *outRGBPremultiplied;
|
|
||||||
if (vips_add(srcWithoutAlpha, t2, &outRGBPremultiplied, NULL))
|
|
||||||
return -1;
|
|
||||||
vips_object_local(context, outRGBPremultiplied);
|
|
||||||
|
|
||||||
// Denormalize output alpha channel:
|
|
||||||
VipsImage *outAlpha;
|
|
||||||
if (vips_linear1(outAlphaNormalized, &outAlpha, 255.0, 0.0, NULL))
|
|
||||||
return -1;
|
|
||||||
vips_object_local(context, outAlpha);
|
|
||||||
|
|
||||||
// Combine RGB and alpha channel into output image:
|
// Combine RGB and alpha channel into output image:
|
||||||
return vips_bandjoin2(outRGBPremultiplied, outAlpha, out, NULL);
|
return outRGBPremultiplied.bandjoin(outAlphaNormalized * 255.0);
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
* Premultiply alpha channel of `image`.
|
|
||||||
*/
|
|
||||||
int Premultiply(VipsObject *context, VipsImage *image, VipsImage **out) {
|
|
||||||
#if (VIPS_MAJOR_VERSION >= 9 || (VIPS_MAJOR_VERSION >= 8 && VIPS_MINOR_VERSION >= 1))
|
|
||||||
return vips_premultiply(image, out, NULL);
|
|
||||||
#else
|
|
||||||
VipsImage *imageRGB;
|
|
||||||
if (vips_extract_band(image, &imageRGB, 0, "n", image->Bands - 1, NULL))
|
|
||||||
return -1;
|
|
||||||
vips_object_local(context, imageRGB);
|
|
||||||
|
|
||||||
VipsImage *imageAlpha;
|
|
||||||
if (vips_extract_band(image, &imageAlpha, image->Bands - 1, "n", 1, NULL))
|
|
||||||
return -1;
|
|
||||||
vips_object_local(context, imageAlpha);
|
|
||||||
|
|
||||||
VipsImage *imageAlphaNormalized;
|
|
||||||
if (vips_linear1(imageAlpha, &imageAlphaNormalized, 1.0 / 255.0, 0.0, NULL))
|
|
||||||
return -1;
|
|
||||||
vips_object_local(context, imageAlphaNormalized);
|
|
||||||
|
|
||||||
VipsImage *imageRGBPremultiplied;
|
|
||||||
if (vips_multiply(imageRGB, imageAlphaNormalized, &imageRGBPremultiplied, NULL))
|
|
||||||
return -1;
|
|
||||||
vips_object_local(context, imageRGBPremultiplied);
|
|
||||||
|
|
||||||
return vips_bandjoin2(imageRGBPremultiplied, imageAlpha, out, NULL);
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
* Unpremultiply alpha channel of `image`.
|
|
||||||
*/
|
|
||||||
int Unpremultiply(VipsObject *context, VipsImage *image, VipsImage **out) {
|
|
||||||
#if (VIPS_MAJOR_VERSION >= 9 || (VIPS_MAJOR_VERSION >= 8 && VIPS_MINOR_VERSION >= 1))
|
|
||||||
return vips_unpremultiply(image, out, NULL);
|
|
||||||
#else
|
|
||||||
VipsImage *imageRGBPremultipliedTransformed;
|
|
||||||
if (vips_extract_band(image, &imageRGBPremultipliedTransformed, 0, "n", image->Bands - 1, NULL))
|
|
||||||
return -1;
|
|
||||||
vips_object_local(context, imageRGBPremultipliedTransformed);
|
|
||||||
|
|
||||||
VipsImage *imageAlphaTransformed;
|
|
||||||
if (vips_extract_band(image, &imageAlphaTransformed, image->Bands - 1, "n", 1, NULL))
|
|
||||||
return -1;
|
|
||||||
vips_object_local(context, imageAlphaTransformed);
|
|
||||||
|
|
||||||
VipsImage *imageAlphaNormalizedTransformed;
|
|
||||||
if (vips_linear1(imageAlphaTransformed, &imageAlphaNormalizedTransformed, 1.0 / 255.0, 0.0, NULL))
|
|
||||||
return -1;
|
|
||||||
vips_object_local(context, imageAlphaNormalizedTransformed);
|
|
||||||
|
|
||||||
VipsImage *imageRGBUnpremultipliedTransformed;
|
|
||||||
if (vips_divide(imageRGBPremultipliedTransformed, imageAlphaNormalizedTransformed, &imageRGBUnpremultipliedTransformed, NULL))
|
|
||||||
return -1;
|
|
||||||
vips_object_local(context, imageRGBUnpremultipliedTransformed);
|
|
||||||
|
|
||||||
return vips_bandjoin2(imageRGBUnpremultipliedTransformed, imageAlphaTransformed, out, NULL);
|
|
||||||
#endif
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Stretch luminance to cover full dynamic range.
|
* Stretch luminance to cover full dynamic range.
|
||||||
*/
|
*/
|
||||||
int Normalize(VipsObject *context, VipsImage *image, VipsImage **out) {
|
VImage Normalize(VImage image) {
|
||||||
#ifndef _WIN32
|
|
||||||
// Get original colourspace
|
// Get original colourspace
|
||||||
VipsInterpretation typeBeforeNormalize = image->Type;
|
VipsInterpretation typeBeforeNormalize = image.interpretation();
|
||||||
if (typeBeforeNormalize == VIPS_INTERPRETATION_RGB) {
|
if (typeBeforeNormalize == VIPS_INTERPRETATION_RGB) {
|
||||||
typeBeforeNormalize = VIPS_INTERPRETATION_sRGB;
|
typeBeforeNormalize = VIPS_INTERPRETATION_sRGB;
|
||||||
}
|
}
|
||||||
// Convert to LAB colourspace
|
// Convert to LAB colourspace
|
||||||
VipsImage *lab;
|
VImage lab = image.colourspace(VIPS_INTERPRETATION_LAB);
|
||||||
if (vips_colourspace(image, &lab, VIPS_INTERPRETATION_LAB, NULL)) {
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
vips_object_local(context, lab);
|
|
||||||
// Extract luminance
|
// Extract luminance
|
||||||
VipsImage *luminance;
|
VImage luminance = lab[0];
|
||||||
if (vips_extract_band(lab, &luminance, 0, "n", 1, NULL)) {
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
vips_object_local(context, luminance);
|
|
||||||
// Extract chroma
|
|
||||||
VipsImage *chroma;
|
|
||||||
if (vips_extract_band(lab, &chroma, 1, "n", 2, NULL)) {
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
vips_object_local(context, chroma);
|
|
||||||
// Find luminance range
|
// Find luminance range
|
||||||
VipsImage *stats;
|
VImage stats = luminance.stats();
|
||||||
if (vips_stats(luminance, &stats, NULL)) {
|
double min = stats(0, 0)[0];
|
||||||
return -1;
|
double max = stats(1, 0)[0];
|
||||||
}
|
|
||||||
vips_object_local(context, stats);
|
|
||||||
double min = *VIPS_MATRIX(stats, 0, 0);
|
|
||||||
double max = *VIPS_MATRIX(stats, 1, 0);
|
|
||||||
if (min != max) {
|
if (min != max) {
|
||||||
|
// Extract chroma
|
||||||
|
VImage chroma = lab.extract_band(1, VImage::option()->set("n", 2));
|
||||||
|
// Calculate multiplication factor and addition
|
||||||
double f = 100.0 / (max - min);
|
double f = 100.0 / (max - min);
|
||||||
double a = -(min * f);
|
double a = -(min * f);
|
||||||
// Scale luminance
|
// Scale luminance, join to chroma, convert back to original colourspace
|
||||||
VipsImage *luminance100;
|
VImage normalized = luminance.linear(f, a).bandjoin(chroma).colourspace(typeBeforeNormalize);
|
||||||
if (vips_linear1(luminance, &luminance100, f, a, NULL)) {
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
vips_object_local(context, luminance100);
|
|
||||||
// Join scaled luminance to chroma
|
|
||||||
VipsImage *normalizedLab;
|
|
||||||
if (vips_bandjoin2(luminance100, chroma, &normalizedLab, NULL)) {
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
vips_object_local(context, normalizedLab);
|
|
||||||
// Convert to original colourspace
|
|
||||||
VipsImage *normalized;
|
|
||||||
if (vips_colourspace(normalizedLab, &normalized, typeBeforeNormalize, NULL)) {
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
vips_object_local(context, normalized);
|
|
||||||
// Attach original alpha channel, if any
|
// Attach original alpha channel, if any
|
||||||
if (HasAlpha(image)) {
|
if (HasAlpha(image)) {
|
||||||
// Extract original alpha channel
|
// Extract original alpha channel
|
||||||
VipsImage *alpha;
|
VImage alpha = image[image.bands() - 1];
|
||||||
if (vips_extract_band(image, &alpha, image->Bands - 1, "n", 1, NULL)) {
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
vips_object_local(context, alpha);
|
|
||||||
// Join alpha channel to normalised image
|
// Join alpha channel to normalised image
|
||||||
VipsImage *normalizedAlpha;
|
return normalized.bandjoin(alpha);
|
||||||
if (vips_bandjoin2(normalized, alpha, &normalizedAlpha, NULL)) {
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
vips_object_local(context, normalizedAlpha);
|
|
||||||
*out = normalizedAlpha;
|
|
||||||
} else {
|
} else {
|
||||||
*out = normalized;
|
return normalized;
|
||||||
}
|
}
|
||||||
} else {
|
|
||||||
// Cannot normalise zero-range image
|
|
||||||
*out = image;
|
|
||||||
}
|
}
|
||||||
#else
|
return image;
|
||||||
// The normalize operation is currently unsupported on Windows
|
}
|
||||||
// See https://github.com/lovell/sharp/issues/152
|
|
||||||
*out = image;
|
/*
|
||||||
#endif
|
* Gamma encoding/decoding
|
||||||
return 0;
|
*/
|
||||||
|
VImage Gamma(VImage image, double const exponent) {
|
||||||
|
if (HasAlpha(image)) {
|
||||||
|
// Separate alpha channel
|
||||||
|
VImage imageWithoutAlpha = image.extract_band(0,
|
||||||
|
VImage::option()->set("n", image.bands() - 1));
|
||||||
|
VImage alpha = image[image.bands() - 1];
|
||||||
|
return imageWithoutAlpha.gamma(VImage::option()->set("exponent", exponent)).bandjoin(alpha);
|
||||||
|
} else {
|
||||||
|
return image.gamma(VImage::option()->set("exponent", exponent));
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Gaussian blur (use sigma <0 for fast blur)
|
* Gaussian blur (use sigma <0 for fast blur)
|
||||||
*/
|
*/
|
||||||
int Blur(VipsObject *context, VipsImage *image, VipsImage **out, double sigma) {
|
VImage Blur(VImage image, double const sigma) {
|
||||||
VipsImage *blurred;
|
|
||||||
if (sigma < 0.0) {
|
if (sigma < 0.0) {
|
||||||
// Fast, mild blur - averages neighbouring pixels
|
// Fast, mild blur - averages neighbouring pixels
|
||||||
VipsImage *blur = vips_image_new_matrixv(3, 3,
|
VImage blur = VImage::new_matrixv(3, 3,
|
||||||
1.0, 1.0, 1.0,
|
1.0, 1.0, 1.0,
|
||||||
1.0, 1.0, 1.0,
|
1.0, 1.0, 1.0,
|
||||||
1.0, 1.0, 1.0);
|
1.0, 1.0, 1.0);
|
||||||
vips_image_set_double(blur, "scale", 9);
|
blur.set("scale", 9.0);
|
||||||
vips_object_local(context, blur);
|
return image.conv(blur);
|
||||||
if (vips_conv(image, &blurred, blur, NULL)) {
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
} else {
|
} else {
|
||||||
// Slower, accurate Gaussian blur
|
// Slower, accurate Gaussian blur
|
||||||
// Create Gaussian function for standard deviation
|
return image.gaussblur(sigma);
|
||||||
VipsImage *gaussian;
|
|
||||||
if (vips_gaussmat(&gaussian, sigma, 0.2, "separable", TRUE, "integer", TRUE, NULL)) {
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
vips_object_local(context, gaussian);
|
|
||||||
// Apply Gaussian function
|
|
||||||
if (vips_convsep(image, &blurred, gaussian, "precision", VIPS_PRECISION_INTEGER, NULL)) {
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
vips_object_local(context, blurred);
|
|
||||||
*out = blurred;
|
|
||||||
return 0;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Sharpen flat and jagged areas. Use radius of -1 for fast sharpen.
|
* Sharpen flat and jagged areas. Use radius of -1 for fast sharpen.
|
||||||
*/
|
*/
|
||||||
int Sharpen(VipsObject *context, VipsImage *image, VipsImage **out, int radius, double flat, double jagged) {
|
VImage Sharpen(VImage image, int const radius, double const flat, double const jagged) {
|
||||||
VipsImage *sharpened;
|
|
||||||
if (radius == -1) {
|
if (radius == -1) {
|
||||||
// Fast, mild sharpen
|
// Fast, mild sharpen
|
||||||
VipsImage *sharpen = vips_image_new_matrixv(3, 3,
|
VImage sharpen = VImage::new_matrixv(3, 3,
|
||||||
-1.0, -1.0, -1.0,
|
-1.0, -1.0, -1.0,
|
||||||
-1.0, 32.0, -1.0,
|
-1.0, 32.0, -1.0,
|
||||||
-1.0, -1.0, -1.0);
|
-1.0, -1.0, -1.0);
|
||||||
vips_image_set_double(sharpen, "scale", 24);
|
sharpen.set("scale", 24.0);
|
||||||
vips_object_local(context, sharpen);
|
return image.conv(sharpen);
|
||||||
if (vips_conv(image, &sharpened, sharpen, NULL)) {
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
} else {
|
} else {
|
||||||
// Slow, accurate sharpen in LAB colour space, with control over flat vs jagged areas
|
// Slow, accurate sharpen in LAB colour space, with control over flat vs jagged areas
|
||||||
if (vips_sharpen(image, &sharpened, "radius", radius, "m1", flat, "m2", jagged, NULL)) {
|
return image.sharpen(
|
||||||
return -1;
|
VImage::option()->set("radius", radius)->set("m1", flat)->set("m2", jagged)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
Calculate crop area based on image entropy
|
||||||
|
*/
|
||||||
|
std::tuple<int, int> EntropyCrop(VImage image, int const outWidth, int const outHeight) {
|
||||||
|
int left = 0;
|
||||||
|
int top = 0;
|
||||||
|
int const inWidth = image.width();
|
||||||
|
int const inHeight = image.height();
|
||||||
|
if (inWidth > outWidth) {
|
||||||
|
// Reduce width by repeated removing slices from edge with lowest entropy
|
||||||
|
int width = inWidth;
|
||||||
|
double leftEntropy = 0.0;
|
||||||
|
double rightEntropy = 0.0;
|
||||||
|
// Max width of each slice
|
||||||
|
int const maxSliceWidth = static_cast<int>(ceil((inWidth - outWidth) / 8.0));
|
||||||
|
while (width > outWidth) {
|
||||||
|
// Width of current slice
|
||||||
|
int const slice = std::min(width - outWidth, maxSliceWidth);
|
||||||
|
if (leftEntropy == 0.0) {
|
||||||
|
// Update entropy of left slice
|
||||||
|
leftEntropy = Entropy(image.extract_area(left, 0, slice, inHeight));
|
||||||
|
}
|
||||||
|
if (rightEntropy == 0.0) {
|
||||||
|
// Update entropy of right slice
|
||||||
|
rightEntropy = Entropy(image.extract_area(width - slice - 1, 0, slice, inHeight));
|
||||||
|
}
|
||||||
|
// Keep slice with highest entropy
|
||||||
|
if (leftEntropy >= rightEntropy) {
|
||||||
|
// Discard right slice
|
||||||
|
rightEntropy = 0.0;
|
||||||
|
} else {
|
||||||
|
// Discard left slice
|
||||||
|
leftEntropy = 0.0;
|
||||||
|
left = left + slice;
|
||||||
|
}
|
||||||
|
width = width - slice;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
vips_object_local(context, sharpened);
|
if (inHeight > outHeight) {
|
||||||
*out = sharpened;
|
// Reduce height by repeated removing slices from edge with lowest entropy
|
||||||
return 0;
|
int height = inHeight;
|
||||||
|
double topEntropy = 0.0;
|
||||||
|
double bottomEntropy = 0.0;
|
||||||
|
// Max height of each slice
|
||||||
|
int const maxSliceHeight = static_cast<int>(ceil((inHeight - outHeight) / 8.0));
|
||||||
|
while (height > outHeight) {
|
||||||
|
// Height of current slice
|
||||||
|
int const slice = std::min(height - outHeight, maxSliceHeight);
|
||||||
|
if (topEntropy == 0.0) {
|
||||||
|
// Update entropy of top slice
|
||||||
|
topEntropy = Entropy(image.extract_area(0, top, inWidth, slice));
|
||||||
|
}
|
||||||
|
if (bottomEntropy == 0.0) {
|
||||||
|
// Update entropy of bottom slice
|
||||||
|
bottomEntropy = Entropy(image.extract_area(0, height - slice - 1, inWidth, slice));
|
||||||
|
}
|
||||||
|
// Keep slice with highest entropy
|
||||||
|
if (topEntropy >= bottomEntropy) {
|
||||||
|
// Discard bottom slice
|
||||||
|
bottomEntropy = 0.0;
|
||||||
|
} else {
|
||||||
|
// Discard top slice
|
||||||
|
topEntropy = 0.0;
|
||||||
|
top = top + slice;
|
||||||
|
}
|
||||||
|
height = height - slice;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return std::make_tuple(left, top);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
Calculate the Shannon entropy for an image
|
||||||
|
*/
|
||||||
|
double Entropy(VImage image) {
|
||||||
|
return image.hist_find().hist_entropy();
|
||||||
|
}
|
||||||
|
|
||||||
} // namespace sharp
|
} // namespace sharp
|
||||||
|
|||||||
42
src/operations.h
Executable file → Normal file
@@ -1,38 +1,48 @@
|
|||||||
#ifndef SRC_OPERATIONS_H_
|
#ifndef SRC_OPERATIONS_H_
|
||||||
#define SRC_OPERATIONS_H_
|
#define SRC_OPERATIONS_H_
|
||||||
|
|
||||||
|
#include <tuple>
|
||||||
|
#include <vips/vips8>
|
||||||
|
|
||||||
|
using vips::VImage;
|
||||||
|
|
||||||
namespace sharp {
|
namespace sharp {
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Composite images `src` and `dst` with premultiplied alpha channel and output
|
Alpha composite src over dst with given gravity.
|
||||||
image with premultiplied alpha.
|
Assumes alpha channels are already premultiplied and will be unpremultiplied after.
|
||||||
*/
|
*/
|
||||||
int Composite(VipsObject *context, VipsImage *src, VipsImage *dst, VipsImage **out);
|
VImage Composite(VImage src, VImage dst, const int gravity);
|
||||||
|
|
||||||
/*
|
|
||||||
* Premultiply alpha channel of `image`.
|
|
||||||
*/
|
|
||||||
int Premultiply(VipsObject *context, VipsImage *image, VipsImage **out);
|
|
||||||
|
|
||||||
/*
|
|
||||||
* Unpremultiply alpha channel of `image`.
|
|
||||||
*/
|
|
||||||
int Unpremultiply(VipsObject *context, VipsImage *image, VipsImage **out);
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Stretch luminance to cover full dynamic range.
|
* Stretch luminance to cover full dynamic range.
|
||||||
*/
|
*/
|
||||||
int Normalize(VipsObject *context, VipsImage *image, VipsImage **out);
|
VImage Normalize(VImage image);
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Gamma encoding/decoding
|
||||||
|
*/
|
||||||
|
VImage Gamma(VImage image, double const exponent);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Gaussian blur. Use sigma of -1 for fast blur.
|
* Gaussian blur. Use sigma of -1 for fast blur.
|
||||||
*/
|
*/
|
||||||
int Blur(VipsObject *context, VipsImage *image, VipsImage **out, double sigma);
|
VImage Blur(VImage image, double const sigma);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Sharpen flat and jagged areas. Use radius of -1 for fast sharpen.
|
* Sharpen flat and jagged areas. Use radius of -1 for fast sharpen.
|
||||||
*/
|
*/
|
||||||
int Sharpen(VipsObject *context, VipsImage *image, VipsImage **out, int radius, double flat, double jagged);
|
VImage Sharpen(VImage image, int const radius, double const flat, double const jagged);
|
||||||
|
|
||||||
|
/*
|
||||||
|
Calculate crop area based on image entropy
|
||||||
|
*/
|
||||||
|
std::tuple<int, int> EntropyCrop(VImage image, int const outWidth, int const outHeight);
|
||||||
|
|
||||||
|
/*
|
||||||
|
Calculate the Shannon entropy for an image
|
||||||
|
*/
|
||||||
|
double Entropy(VImage image);
|
||||||
|
|
||||||
} // namespace sharp
|
} // namespace sharp
|
||||||
|
|
||||||
|
|||||||
1846
src/pipeline.cc
Executable file → Normal file
135
src/pipeline.h
Executable file → Normal file
@@ -1,8 +1,143 @@
|
|||||||
#ifndef SRC_PIPELINE_H_
|
#ifndef SRC_PIPELINE_H_
|
||||||
#define SRC_PIPELINE_H_
|
#define SRC_PIPELINE_H_
|
||||||
|
|
||||||
|
#include <vips/vips8>
|
||||||
|
|
||||||
#include "nan.h"
|
#include "nan.h"
|
||||||
|
|
||||||
NAN_METHOD(pipeline);
|
NAN_METHOD(pipeline);
|
||||||
|
|
||||||
|
enum class Canvas {
|
||||||
|
CROP,
|
||||||
|
EMBED,
|
||||||
|
MAX,
|
||||||
|
MIN,
|
||||||
|
IGNORE_ASPECT
|
||||||
|
};
|
||||||
|
|
||||||
|
struct PipelineBaton {
|
||||||
|
std::string fileIn;
|
||||||
|
char *bufferIn;
|
||||||
|
size_t bufferInLength;
|
||||||
|
std::string iccProfilePath;
|
||||||
|
int limitInputPixels;
|
||||||
|
int density;
|
||||||
|
int rawWidth;
|
||||||
|
int rawHeight;
|
||||||
|
int rawChannels;
|
||||||
|
std::string formatOut;
|
||||||
|
std::string fileOut;
|
||||||
|
void *bufferOut;
|
||||||
|
size_t bufferOutLength;
|
||||||
|
std::string overlayFileIn;
|
||||||
|
char *overlayBufferIn;
|
||||||
|
size_t overlayBufferInLength;
|
||||||
|
int overlayGravity;
|
||||||
|
int topOffsetPre;
|
||||||
|
int leftOffsetPre;
|
||||||
|
int widthPre;
|
||||||
|
int heightPre;
|
||||||
|
int topOffsetPost;
|
||||||
|
int leftOffsetPost;
|
||||||
|
int widthPost;
|
||||||
|
int heightPost;
|
||||||
|
int width;
|
||||||
|
int height;
|
||||||
|
int channels;
|
||||||
|
Canvas canvas;
|
||||||
|
int crop;
|
||||||
|
std::string interpolator;
|
||||||
|
double background[4];
|
||||||
|
bool flatten;
|
||||||
|
bool negate;
|
||||||
|
double blurSigma;
|
||||||
|
int sharpenRadius;
|
||||||
|
double sharpenFlat;
|
||||||
|
double sharpenJagged;
|
||||||
|
int threshold;
|
||||||
|
double gamma;
|
||||||
|
bool greyscale;
|
||||||
|
bool normalize;
|
||||||
|
int angle;
|
||||||
|
bool rotateBeforePreExtract;
|
||||||
|
bool flip;
|
||||||
|
bool flop;
|
||||||
|
int extendTop;
|
||||||
|
int extendBottom;
|
||||||
|
int extendLeft;
|
||||||
|
int extendRight;
|
||||||
|
bool progressive;
|
||||||
|
bool withoutEnlargement;
|
||||||
|
VipsAccess accessMethod;
|
||||||
|
int quality;
|
||||||
|
int compressionLevel;
|
||||||
|
bool withoutAdaptiveFiltering;
|
||||||
|
bool withoutChromaSubsampling;
|
||||||
|
bool trellisQuantisation;
|
||||||
|
bool overshootDeringing;
|
||||||
|
bool optimiseScans;
|
||||||
|
std::string err;
|
||||||
|
bool withMetadata;
|
||||||
|
int withMetadataOrientation;
|
||||||
|
int tileSize;
|
||||||
|
int tileOverlap;
|
||||||
|
VipsForeignDzContainer tileContainer;
|
||||||
|
VipsForeignDzLayout tileLayout;
|
||||||
|
|
||||||
|
PipelineBaton():
|
||||||
|
bufferInLength(0),
|
||||||
|
limitInputPixels(0),
|
||||||
|
density(72),
|
||||||
|
rawWidth(0),
|
||||||
|
rawHeight(0),
|
||||||
|
rawChannels(0),
|
||||||
|
formatOut(""),
|
||||||
|
fileOut(""),
|
||||||
|
bufferOutLength(0),
|
||||||
|
overlayBufferInLength(0),
|
||||||
|
overlayGravity(0),
|
||||||
|
topOffsetPre(-1),
|
||||||
|
topOffsetPost(-1),
|
||||||
|
channels(0),
|
||||||
|
canvas(Canvas::CROP),
|
||||||
|
crop(0),
|
||||||
|
flatten(false),
|
||||||
|
negate(false),
|
||||||
|
blurSigma(0.0),
|
||||||
|
sharpenRadius(0),
|
||||||
|
sharpenFlat(1.0),
|
||||||
|
sharpenJagged(2.0),
|
||||||
|
threshold(0),
|
||||||
|
gamma(0.0),
|
||||||
|
greyscale(false),
|
||||||
|
normalize(false),
|
||||||
|
angle(0),
|
||||||
|
flip(false),
|
||||||
|
flop(false),
|
||||||
|
extendTop(0),
|
||||||
|
extendBottom(0),
|
||||||
|
extendLeft(0),
|
||||||
|
extendRight(0),
|
||||||
|
progressive(false),
|
||||||
|
withoutEnlargement(false),
|
||||||
|
quality(80),
|
||||||
|
compressionLevel(6),
|
||||||
|
withoutAdaptiveFiltering(false),
|
||||||
|
withoutChromaSubsampling(false),
|
||||||
|
trellisQuantisation(false),
|
||||||
|
overshootDeringing(false),
|
||||||
|
optimiseScans(false),
|
||||||
|
withMetadata(false),
|
||||||
|
withMetadataOrientation(-1),
|
||||||
|
tileSize(256),
|
||||||
|
tileOverlap(0),
|
||||||
|
tileContainer(VIPS_FOREIGN_DZ_CONTAINER_FS),
|
||||||
|
tileLayout(VIPS_FOREIGN_DZ_LAYOUT_DZ) {
|
||||||
|
background[0] = 0.0;
|
||||||
|
background[1] = 0.0;
|
||||||
|
background[2] = 0.0;
|
||||||
|
background[3] = 255.0;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
#endif // SRC_PIPELINE_H_
|
#endif // SRC_PIPELINE_H_
|
||||||
|
|||||||
35
src/sharp.cc
Executable file → Normal file
@@ -1,5 +1,5 @@
|
|||||||
#include <node.h>
|
#include <node.h>
|
||||||
#include <vips/vips.h>
|
#include <vips/vips8>
|
||||||
|
|
||||||
#include "nan.h"
|
#include "nan.h"
|
||||||
|
|
||||||
@@ -8,23 +8,28 @@
|
|||||||
#include "pipeline.h"
|
#include "pipeline.h"
|
||||||
#include "utilities.h"
|
#include "utilities.h"
|
||||||
|
|
||||||
extern "C" void init(v8::Handle<v8::Object> target) {
|
NAN_MODULE_INIT(init) {
|
||||||
NanScope();
|
|
||||||
vips_init("sharp");
|
vips_init("sharp");
|
||||||
|
|
||||||
// Set libvips operation cache limits
|
|
||||||
vips_cache_set_max_mem(100 * 1024 * 1024); // 100 MB
|
|
||||||
vips_cache_set_max(500); // 500 operations
|
|
||||||
|
|
||||||
// Methods available to JavaScript
|
// Methods available to JavaScript
|
||||||
NODE_SET_METHOD(target, "metadata", metadata);
|
Nan::Set(target, Nan::New("metadata").ToLocalChecked(),
|
||||||
NODE_SET_METHOD(target, "pipeline", pipeline);
|
Nan::GetFunction(Nan::New<v8::FunctionTemplate>(metadata)).ToLocalChecked());
|
||||||
NODE_SET_METHOD(target, "cache", cache);
|
Nan::Set(target, Nan::New("pipeline").ToLocalChecked(),
|
||||||
NODE_SET_METHOD(target, "concurrency", concurrency);
|
Nan::GetFunction(Nan::New<v8::FunctionTemplate>(pipeline)).ToLocalChecked());
|
||||||
NODE_SET_METHOD(target, "counters", counters);
|
Nan::Set(target, Nan::New("cache").ToLocalChecked(),
|
||||||
NODE_SET_METHOD(target, "libvipsVersion", libvipsVersion);
|
Nan::GetFunction(Nan::New<v8::FunctionTemplate>(cache)).ToLocalChecked());
|
||||||
NODE_SET_METHOD(target, "format", format);
|
Nan::Set(target, Nan::New("concurrency").ToLocalChecked(),
|
||||||
NODE_SET_METHOD(target, "_maxColourDistance", _maxColourDistance);
|
Nan::GetFunction(Nan::New<v8::FunctionTemplate>(concurrency)).ToLocalChecked());
|
||||||
|
Nan::Set(target, Nan::New("counters").ToLocalChecked(),
|
||||||
|
Nan::GetFunction(Nan::New<v8::FunctionTemplate>(counters)).ToLocalChecked());
|
||||||
|
Nan::Set(target, Nan::New("simd").ToLocalChecked(),
|
||||||
|
Nan::GetFunction(Nan::New<v8::FunctionTemplate>(simd)).ToLocalChecked());
|
||||||
|
Nan::Set(target, Nan::New("libvipsVersion").ToLocalChecked(),
|
||||||
|
Nan::GetFunction(Nan::New<v8::FunctionTemplate>(libvipsVersion)).ToLocalChecked());
|
||||||
|
Nan::Set(target, Nan::New("format").ToLocalChecked(),
|
||||||
|
Nan::GetFunction(Nan::New<v8::FunctionTemplate>(format)).ToLocalChecked());
|
||||||
|
Nan::Set(target, Nan::New("_maxColourDistance").ToLocalChecked(),
|
||||||
|
Nan::GetFunction(Nan::New<v8::FunctionTemplate>(_maxColourDistance)).ToLocalChecked());
|
||||||
}
|
}
|
||||||
|
|
||||||
NODE_MODULE(sharp, init)
|
NODE_MODULE(sharp, init)
|
||||||
|
|||||||
322
src/utilities.cc
Executable file → Normal file
@@ -1,5 +1,7 @@
|
|||||||
|
#include <cmath>
|
||||||
#include <node.h>
|
#include <node.h>
|
||||||
#include <vips/vips.h>
|
#include <vips/vips8>
|
||||||
|
#include <vips/vector.h>
|
||||||
|
|
||||||
#include "nan.h"
|
#include "nan.h"
|
||||||
|
|
||||||
@@ -7,54 +9,79 @@
|
|||||||
#include "operations.h"
|
#include "operations.h"
|
||||||
#include "utilities.h"
|
#include "utilities.h"
|
||||||
|
|
||||||
using v8::Local;
|
|
||||||
using v8::Object;
|
|
||||||
using v8::Number;
|
|
||||||
using v8::String;
|
|
||||||
using v8::Boolean;
|
using v8::Boolean;
|
||||||
|
using v8::Integer;
|
||||||
|
using v8::Local;
|
||||||
|
using v8::Number;
|
||||||
|
using v8::Object;
|
||||||
|
using v8::String;
|
||||||
|
|
||||||
|
using Nan::HandleScope;
|
||||||
|
using Nan::New;
|
||||||
|
using Nan::Set;
|
||||||
|
using Nan::ThrowError;
|
||||||
|
using Nan::To;
|
||||||
|
using Nan::Utf8String;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Get and set cache memory and item limits
|
Get and set cache limits
|
||||||
*/
|
*/
|
||||||
NAN_METHOD(cache) {
|
NAN_METHOD(cache) {
|
||||||
NanScope();
|
HandleScope();
|
||||||
|
|
||||||
// Set cache memory limit
|
// Set memory limit
|
||||||
if (args[0]->IsInt32()) {
|
if (info[0]->IsInt32()) {
|
||||||
int newMax = args[0]->Int32Value() * 1048576;
|
vips_cache_set_max_mem(To<int32_t>(info[0]).FromJust() * 1048576);
|
||||||
int oldMax = vips_cache_get_max_mem();
|
}
|
||||||
vips_cache_set_max_mem(newMax);
|
// Set file limit
|
||||||
|
if (info[1]->IsInt32()) {
|
||||||
// Notify the V8 garbage collector of delta in max cache size
|
vips_cache_set_max_files(To<int32_t>(info[1]).FromJust());
|
||||||
NanAdjustExternalMemory(newMax - oldMax);
|
}
|
||||||
|
// Set items limit
|
||||||
|
if (info[2]->IsInt32()) {
|
||||||
|
vips_cache_set_max(To<int32_t>(info[2]).FromJust());
|
||||||
}
|
}
|
||||||
|
|
||||||
// Set cache items limit
|
// Get memory stats
|
||||||
if (args[1]->IsInt32()) {
|
Local<Object> memory = New<Object>();
|
||||||
vips_cache_set_max(args[1]->Int32Value());
|
Set(memory, New("current").ToLocalChecked(),
|
||||||
}
|
New<Integer>(static_cast<int>(round(vips_tracked_get_mem() / 1048576)))
|
||||||
|
);
|
||||||
|
Set(memory, New("high").ToLocalChecked(),
|
||||||
|
New<Integer>(static_cast<int>(round(vips_tracked_get_mem_highwater() / 1048576)))
|
||||||
|
);
|
||||||
|
Set(memory, New("max").ToLocalChecked(),
|
||||||
|
New<Integer>(static_cast<int>(round(vips_cache_get_max_mem() / 1048576)))
|
||||||
|
);
|
||||||
|
// Get file stats
|
||||||
|
Local<Object> files = New<Object>();
|
||||||
|
Set(files, New("current").ToLocalChecked(), New<Integer>(vips_tracked_get_files()));
|
||||||
|
Set(files, New("max").ToLocalChecked(), New<Integer>(vips_cache_get_max_files()));
|
||||||
|
|
||||||
// Get cache statistics
|
// Get item stats
|
||||||
Local<Object> cache = NanNew<Object>();
|
Local<Object> items = New<Object>();
|
||||||
cache->Set(NanNew<String>("current"), NanNew<Number>(vips_tracked_get_mem() / 1048576));
|
Set(items, New("current").ToLocalChecked(), New<Integer>(vips_cache_get_size()));
|
||||||
cache->Set(NanNew<String>("high"), NanNew<Number>(vips_tracked_get_mem_highwater() / 1048576));
|
Set(items, New("max").ToLocalChecked(), New<Integer>(vips_cache_get_max()));
|
||||||
cache->Set(NanNew<String>("memory"), NanNew<Number>(vips_cache_get_max_mem() / 1048576));
|
|
||||||
cache->Set(NanNew<String>("items"), NanNew<Number>(vips_cache_get_max()));
|
Local<Object> cache = New<Object>();
|
||||||
NanReturnValue(cache);
|
Set(cache, New("memory").ToLocalChecked(), memory);
|
||||||
|
Set(cache, New("files").ToLocalChecked(), files);
|
||||||
|
Set(cache, New("items").ToLocalChecked(), items);
|
||||||
|
info.GetReturnValue().Set(cache);
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Get and set size of thread pool
|
Get and set size of thread pool
|
||||||
*/
|
*/
|
||||||
NAN_METHOD(concurrency) {
|
NAN_METHOD(concurrency) {
|
||||||
NanScope();
|
HandleScope();
|
||||||
|
|
||||||
// Set concurrency
|
// Set concurrency
|
||||||
if (args[0]->IsInt32()) {
|
if (info[0]->IsInt32()) {
|
||||||
vips_concurrency_set(args[0]->Int32Value());
|
vips_concurrency_set(To<int32_t>(info[0]).FromJust());
|
||||||
}
|
}
|
||||||
// Get concurrency
|
// Get concurrency
|
||||||
NanReturnValue(NanNew<Number>(vips_concurrency_get()));
|
info.GetReturnValue().Set(New<Integer>(vips_concurrency_get()));
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
@@ -64,84 +91,101 @@ NAN_METHOD(counters) {
|
|||||||
using sharp::counterProcess;
|
using sharp::counterProcess;
|
||||||
using sharp::counterQueue;
|
using sharp::counterQueue;
|
||||||
|
|
||||||
NanScope();
|
HandleScope();
|
||||||
Local<Object> counters = NanNew<Object>();
|
Local<Object> counters = New<Object>();
|
||||||
counters->Set(NanNew<String>("queue"), NanNew<Number>(counterQueue));
|
Set(counters, New("queue").ToLocalChecked(), New<Integer>(counterQueue));
|
||||||
counters->Set(NanNew<String>("process"), NanNew<Number>(counterProcess));
|
Set(counters, New("process").ToLocalChecked(), New<Integer>(counterProcess));
|
||||||
NanReturnValue(counters);
|
info.GetReturnValue().Set(counters);
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
Get and set use of SIMD vector unit instructions
|
||||||
|
*/
|
||||||
|
NAN_METHOD(simd) {
|
||||||
|
HandleScope();
|
||||||
|
|
||||||
|
// Set state
|
||||||
|
if (info[0]->IsBoolean()) {
|
||||||
|
vips_vector_set_enabled(To<bool>(info[0]).FromJust());
|
||||||
|
}
|
||||||
|
// Get state
|
||||||
|
info.GetReturnValue().Set(New<Boolean>(vips_vector_isenabled()));
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Get libvips version
|
Get libvips version
|
||||||
*/
|
*/
|
||||||
NAN_METHOD(libvipsVersion) {
|
NAN_METHOD(libvipsVersion) {
|
||||||
NanScope();
|
HandleScope();
|
||||||
char version[9];
|
char version[9];
|
||||||
g_snprintf(version, sizeof(version), "%d.%d.%d", vips_version(0), vips_version(1), vips_version(2));
|
g_snprintf(version, sizeof(version), "%d.%d.%d", vips_version(0), vips_version(1), vips_version(2));
|
||||||
NanReturnValue(NanNew<String>(version));
|
info.GetReturnValue().Set(New(version).ToLocalChecked());
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Get available input/output file/buffer/stream formats
|
Get available input/output file/buffer/stream formats
|
||||||
*/
|
*/
|
||||||
NAN_METHOD(format) {
|
NAN_METHOD(format) {
|
||||||
NanScope();
|
HandleScope();
|
||||||
|
|
||||||
// Attribute names
|
// Attribute names
|
||||||
Local<String> attrId = NanNew<String>("id");
|
Local<String> attrId = New("id").ToLocalChecked();
|
||||||
Local<String> attrInput = NanNew<String>("input");
|
Local<String> attrInput = New("input").ToLocalChecked();
|
||||||
Local<String> attrOutput = NanNew<String>("output");
|
Local<String> attrOutput = New("output").ToLocalChecked();
|
||||||
Local<String> attrFile = NanNew<String>("file");
|
Local<String> attrFile = New("file").ToLocalChecked();
|
||||||
Local<String> attrBuffer = NanNew<String>("buffer");
|
Local<String> attrBuffer = New("buffer").ToLocalChecked();
|
||||||
Local<String> attrStream = NanNew<String>("stream");
|
Local<String> attrStream = New("stream").ToLocalChecked();
|
||||||
|
|
||||||
// Which load/save operations are available for each compressed format?
|
// Which load/save operations are available for each compressed format?
|
||||||
Local<Object> format = NanNew<Object>();
|
Local<Object> format = New<Object>();
|
||||||
for (std::string f : {"jpeg", "png", "webp", "tiff", "magick", "openslide", "dz"}) {
|
for (std::string f : {"jpeg", "png", "webp", "tiff", "magick", "openslide", "dz"}) {
|
||||||
// Input
|
// Input
|
||||||
Local<Object> input = NanNew<Object>();
|
Local<Boolean> hasInputFile =
|
||||||
input->Set(attrFile, NanNew<Boolean>(
|
New<Boolean>(vips_type_find("VipsOperation", (f + "load").c_str()));
|
||||||
vips_type_find("VipsOperation", (f + "load").c_str())));
|
Local<Boolean> hasInputBuffer =
|
||||||
input->Set(attrBuffer, NanNew<Boolean>(
|
New<Boolean>(vips_type_find("VipsOperation", (f + "load_buffer").c_str()));
|
||||||
vips_type_find("VipsOperation", (f + "load_buffer").c_str())));
|
Local<Object> input = New<Object>();
|
||||||
input->Set(attrStream, input->Get(attrBuffer));
|
Set(input, attrFile, hasInputFile);
|
||||||
|
Set(input, attrBuffer, hasInputBuffer);
|
||||||
|
Set(input, attrStream, hasInputBuffer);
|
||||||
// Output
|
// Output
|
||||||
Local<Object> output = NanNew<Object>();
|
Local<Boolean> hasOutputFile =
|
||||||
output->Set(attrFile, NanNew<Boolean>(
|
New<Boolean>(vips_type_find("VipsOperation", (f + "save").c_str()));
|
||||||
vips_type_find("VipsOperation", (f + "save").c_str())));
|
Local<Boolean> hasOutputBuffer =
|
||||||
output->Set(attrBuffer, NanNew<Boolean>(
|
New<Boolean>(vips_type_find("VipsOperation", (f + "save_buffer").c_str()));
|
||||||
vips_type_find("VipsOperation", (f + "save_buffer").c_str())));
|
Local<Object> output = New<Object>();
|
||||||
output->Set(attrStream, output->Get(attrBuffer));
|
Set(output, attrFile, hasOutputFile);
|
||||||
|
Set(output, attrBuffer, hasOutputBuffer);
|
||||||
|
Set(output, attrStream, hasOutputBuffer);
|
||||||
// Other attributes
|
// Other attributes
|
||||||
Local<Object> container = NanNew<Object>();
|
Local<Object> container = New<Object>();
|
||||||
Local<String> formatId = NanNew<String>(f);
|
Local<String> formatId = New(f).ToLocalChecked();
|
||||||
container->Set(attrId, formatId);
|
Set(container, attrId, formatId);
|
||||||
container->Set(attrInput, input);
|
Set(container, attrInput, input);
|
||||||
container->Set(attrOutput, output);
|
Set(container, attrOutput, output);
|
||||||
// Add to set of formats
|
// Add to set of formats
|
||||||
format->Set(formatId, container);
|
Set(format, formatId, container);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Raw, uncompressed data
|
// Raw, uncompressed data
|
||||||
Local<Object> raw = NanNew<Object>();
|
Local<Object> raw = New<Object>();
|
||||||
raw->Set(attrId, NanNew<String>("raw"));
|
Local<String> rawId = New("raw").ToLocalChecked();
|
||||||
format->Set(NanNew<String>("raw"), raw);
|
Set(raw, attrId, rawId);
|
||||||
// No support for raw input yet, so always false
|
Set(format, rawId, raw);
|
||||||
Local<Boolean> unsupported = NanNew<Boolean>(false);
|
Local<Boolean> supported = New<Boolean>(true);
|
||||||
Local<Object> rawInput = NanNew<Object>();
|
Local<Boolean> unsupported = New<Boolean>(false);
|
||||||
rawInput->Set(attrFile, unsupported);
|
Local<Object> rawInput = New<Object>();
|
||||||
rawInput->Set(attrBuffer, unsupported);
|
Set(rawInput, attrFile, unsupported);
|
||||||
rawInput->Set(attrStream, unsupported);
|
Set(rawInput, attrBuffer, supported);
|
||||||
raw->Set(attrInput, rawInput);
|
Set(rawInput, attrStream, supported);
|
||||||
// Raw output via Buffer/Stream is available in libvips >= 7.42.0
|
Set(raw, attrInput, rawInput);
|
||||||
Local<Boolean> supportsRawOutput = NanNew<Boolean>(vips_version(0) >= 8 || (vips_version(0) == 7 && vips_version(1) >= 42));
|
Local<Object> rawOutput = New<Object>();
|
||||||
Local<Object> rawOutput = NanNew<Object>();
|
Set(rawOutput, attrFile, unsupported);
|
||||||
rawOutput->Set(attrFile, unsupported);
|
Set(rawOutput, attrBuffer, supported);
|
||||||
rawOutput->Set(attrBuffer, supportsRawOutput);
|
Set(rawOutput, attrStream, supported);
|
||||||
rawOutput->Set(attrStream, supportsRawOutput);
|
Set(raw, attrOutput, rawOutput);
|
||||||
raw->Set(attrOutput, rawOutput);
|
|
||||||
|
|
||||||
NanReturnValue(format);
|
info.GetReturnValue().Set(format);
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
@@ -150,102 +194,64 @@ NAN_METHOD(format) {
|
|||||||
between two images of the same dimensions and number of channels.
|
between two images of the same dimensions and number of channels.
|
||||||
*/
|
*/
|
||||||
NAN_METHOD(_maxColourDistance) {
|
NAN_METHOD(_maxColourDistance) {
|
||||||
using sharp::Premultiply;
|
using vips::VImage;
|
||||||
|
using vips::VError;
|
||||||
using sharp::DetermineImageType;
|
using sharp::DetermineImageType;
|
||||||
using sharp::ImageType;
|
using sharp::ImageType;
|
||||||
using sharp::InitImage;
|
|
||||||
using sharp::HasAlpha;
|
using sharp::HasAlpha;
|
||||||
|
|
||||||
NanScope();
|
HandleScope();
|
||||||
|
|
||||||
// Create "hook" VipsObject to hang image references from
|
|
||||||
VipsObject *hook = reinterpret_cast<VipsObject*>(vips_image_new());
|
|
||||||
|
|
||||||
// Open input files
|
// Open input files
|
||||||
VipsImage *image1 = NULL;
|
VImage image1;
|
||||||
ImageType imageType1 = DetermineImageType(*String::Utf8Value(args[0]));
|
ImageType imageType1 = DetermineImageType(*Utf8String(info[0]));
|
||||||
if (imageType1 != ImageType::UNKNOWN) {
|
if (imageType1 != ImageType::UNKNOWN) {
|
||||||
image1 = InitImage(*String::Utf8Value(args[0]), VIPS_ACCESS_SEQUENTIAL);
|
try {
|
||||||
if (image1 == NULL) {
|
image1 = VImage::new_from_file(*Utf8String(info[0]));
|
||||||
g_object_unref(hook);
|
} catch (...) {
|
||||||
return NanThrowError("Input file 1 has corrupt header");
|
return ThrowError("Input file 1 has corrupt header");
|
||||||
} else {
|
|
||||||
vips_object_local(hook, image1);
|
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
g_object_unref(hook);
|
return ThrowError("Input file 1 is of an unsupported image format");
|
||||||
return NanThrowError("Input file 1 is of an unsupported image format");
|
|
||||||
}
|
}
|
||||||
VipsImage *image2 = NULL;
|
VImage image2;
|
||||||
ImageType imageType2 = DetermineImageType(*String::Utf8Value(args[1]));
|
ImageType imageType2 = DetermineImageType(*Utf8String(info[1]));
|
||||||
if (imageType2 != ImageType::UNKNOWN) {
|
if (imageType2 != ImageType::UNKNOWN) {
|
||||||
image2 = InitImage(*String::Utf8Value(args[1]), VIPS_ACCESS_SEQUENTIAL);
|
try {
|
||||||
if (image2 == NULL) {
|
image2 = VImage::new_from_file(*Utf8String(info[1]));
|
||||||
g_object_unref(hook);
|
} catch (...) {
|
||||||
return NanThrowError("Input file 2 has corrupt header");
|
return ThrowError("Input file 2 has corrupt header");
|
||||||
} else {
|
|
||||||
vips_object_local(hook, image2);
|
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
g_object_unref(hook);
|
return ThrowError("Input file 2 is of an unsupported image format");
|
||||||
return NanThrowError("Input file 2 is of an unsupported image format");
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Ensure same number of channels
|
// Ensure same number of channels
|
||||||
if (image1->Bands != image2->Bands) {
|
if (image1.bands() != image2.bands()) {
|
||||||
g_object_unref(hook);
|
return ThrowError("mismatchedBands");
|
||||||
return NanThrowError("mismatchedBands");
|
|
||||||
}
|
}
|
||||||
// Ensure same dimensions
|
// Ensure same dimensions
|
||||||
if (image1->Xsize != image2->Xsize || image1->Ysize != image2->Ysize) {
|
if (image1.width() != image2.width() || image1.height() != image2.height()) {
|
||||||
g_object_unref(hook);
|
return ThrowError("mismatchedDimensions");
|
||||||
return NanThrowError("mismatchedDimensions");
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Premultiply and remove alpha
|
|
||||||
if (HasAlpha(image1)) {
|
|
||||||
VipsImage *imagePremultiplied1;
|
|
||||||
if (Premultiply(hook, image1, &imagePremultiplied1)) {
|
|
||||||
g_object_unref(hook);
|
|
||||||
return NanThrowError(vips_error_buffer());
|
|
||||||
}
|
|
||||||
vips_object_local(hook, imagePremultiplied1);
|
|
||||||
VipsImage *imagePremultipliedNoAlpha1;
|
|
||||||
if (vips_extract_band(image1, &imagePremultipliedNoAlpha1, 1, "n", image1->Bands - 1, NULL)) {
|
|
||||||
g_object_unref(hook);
|
|
||||||
return NanThrowError(vips_error_buffer());
|
|
||||||
}
|
|
||||||
vips_object_local(hook, imagePremultipliedNoAlpha1);
|
|
||||||
image1 = imagePremultipliedNoAlpha1;
|
|
||||||
}
|
|
||||||
if (HasAlpha(image2)) {
|
|
||||||
VipsImage *imagePremultiplied2;
|
|
||||||
if (Premultiply(hook, image2, &imagePremultiplied2)) {
|
|
||||||
g_object_unref(hook);
|
|
||||||
return NanThrowError(vips_error_buffer());
|
|
||||||
}
|
|
||||||
vips_object_local(hook, imagePremultiplied2);
|
|
||||||
VipsImage *imagePremultipliedNoAlpha2;
|
|
||||||
if (vips_extract_band(image2, &imagePremultipliedNoAlpha2, 1, "n", image2->Bands - 1, NULL)) {
|
|
||||||
g_object_unref(hook);
|
|
||||||
return NanThrowError(vips_error_buffer());
|
|
||||||
}
|
|
||||||
vips_object_local(hook, imagePremultipliedNoAlpha2);
|
|
||||||
image2 = imagePremultipliedNoAlpha2;
|
|
||||||
}
|
|
||||||
// Calculate colour distance
|
|
||||||
VipsImage *difference;
|
|
||||||
if (vips_dE00(image1, image2, &difference, NULL)) {
|
|
||||||
g_object_unref(hook);
|
|
||||||
return NanThrowError(vips_error_buffer());
|
|
||||||
}
|
|
||||||
vips_object_local(hook, difference);
|
|
||||||
// Extract maximum distance
|
|
||||||
double maxColourDistance;
|
double maxColourDistance;
|
||||||
if (vips_max(difference, &maxColourDistance, NULL)) {
|
try {
|
||||||
g_object_unref(hook);
|
// Premultiply and remove alpha
|
||||||
return NanThrowError(vips_error_buffer());
|
if (HasAlpha(image1)) {
|
||||||
|
image1 = image1.premultiply().extract_band(1, VImage::option()->set("n", image1.bands() - 1));
|
||||||
|
}
|
||||||
|
if (HasAlpha(image2)) {
|
||||||
|
image2 = image2.premultiply().extract_band(1, VImage::option()->set("n", image2.bands() - 1));
|
||||||
|
}
|
||||||
|
// Calculate colour distance
|
||||||
|
maxColourDistance = image1.dE00(image2).max();
|
||||||
|
} catch (VError err) {
|
||||||
|
return ThrowError(err.what());
|
||||||
}
|
}
|
||||||
g_object_unref(hook);
|
|
||||||
NanReturnValue(maxColourDistance);
|
// Clean up libvips' per-request data and threads
|
||||||
|
vips_error_clear();
|
||||||
|
vips_thread_shutdown();
|
||||||
|
|
||||||
|
info.GetReturnValue().Set(New<Number>(maxColourDistance));
|
||||||
}
|
}
|
||||||
|
|||||||
1
src/utilities.h
Executable file → Normal file
@@ -6,6 +6,7 @@
|
|||||||
NAN_METHOD(cache);
|
NAN_METHOD(cache);
|
||||||
NAN_METHOD(concurrency);
|
NAN_METHOD(concurrency);
|
||||||
NAN_METHOD(counters);
|
NAN_METHOD(counters);
|
||||||
|
NAN_METHOD(simd);
|
||||||
NAN_METHOD(libvipsVersion);
|
NAN_METHOD(libvipsVersion);
|
||||||
NAN_METHOD(format);
|
NAN_METHOD(format);
|
||||||
NAN_METHOD(_maxColourDistance);
|
NAN_METHOD(_maxColourDistance);
|
||||||
|
|||||||
15
test/bench/package.json
Executable file → Normal file
@@ -5,16 +5,17 @@
|
|||||||
"author": "Lovell Fuller <npm@lovell.info>",
|
"author": "Lovell Fuller <npm@lovell.info>",
|
||||||
"description": "Benchmark and performance tests for sharp",
|
"description": "Benchmark and performance tests for sharp",
|
||||||
"scripts": {
|
"scripts": {
|
||||||
"test": "node perf && node random && node parallel"
|
"test": "VIPS_WARNING=0 node perf && node random && node parallel"
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
|
"async": "^1.5.2",
|
||||||
|
"benchmark": "^2.1.0",
|
||||||
|
"gm": "^1.21.0",
|
||||||
"imagemagick": "^0.1.3",
|
"imagemagick": "^0.1.3",
|
||||||
"imagemagick-native": "^1.8.0",
|
"imagemagick-native": "^1.9.2",
|
||||||
"gm": "^1.18.1",
|
"jimp": "^0.2.20",
|
||||||
"lwip": "^0.0.7",
|
"lwip": "^0.0.8",
|
||||||
"async": "^1.4.2",
|
"semver": "^5.1.0"
|
||||||
"semver": "^5.0.1",
|
|
||||||
"benchmark": "^1.0.0"
|
|
||||||
},
|
},
|
||||||
"license": "Apache-2.0",
|
"license": "Apache-2.0",
|
||||||
"engines": {
|
"engines": {
|
||||||
|
|||||||
5
test/bench/parallel.js
Executable file → Normal file
@@ -1,5 +1,7 @@
|
|||||||
'use strict';
|
'use strict';
|
||||||
|
|
||||||
|
process.env.UV_THREADPOOL_SIZE = 64;
|
||||||
|
|
||||||
var assert = require('assert');
|
var assert = require('assert');
|
||||||
var async = require('async');
|
var async = require('async');
|
||||||
|
|
||||||
@@ -10,12 +12,13 @@ var width = 720;
|
|||||||
var height = 480;
|
var height = 480;
|
||||||
|
|
||||||
sharp.concurrency(1);
|
sharp.concurrency(1);
|
||||||
|
sharp.simd(true);
|
||||||
|
|
||||||
var timer = setInterval(function() {
|
var timer = setInterval(function() {
|
||||||
console.dir(sharp.counters());
|
console.dir(sharp.counters());
|
||||||
}, 100);
|
}, 100);
|
||||||
|
|
||||||
async.mapSeries([1, 1, 2, 4, 8, 16, 32, 64, 128], function(parallelism, next) {
|
async.mapSeries([1, 1, 2, 4, 8, 16, 32, 64], function(parallelism, next) {
|
||||||
var start = new Date().getTime();
|
var start = new Date().getTime();
|
||||||
async.times(parallelism,
|
async.times(parallelism,
|
||||||
function(id, callback) {
|
function(id, callback) {
|
||||||
|
|||||||
1270
test/bench/perf.js
Executable file → Normal file
4
test/bench/random.js
Executable file → Normal file
@@ -8,6 +8,8 @@ var Benchmark = require('benchmark');
|
|||||||
var sharp = require('../../index');
|
var sharp = require('../../index');
|
||||||
var fixtures = require('../fixtures');
|
var fixtures = require('../fixtures');
|
||||||
|
|
||||||
|
sharp.simd(true);
|
||||||
|
|
||||||
var min = 320;
|
var min = 320;
|
||||||
var max = 960;
|
var max = 960;
|
||||||
|
|
||||||
@@ -68,7 +70,7 @@ new Benchmark.Suite('random').add('imagemagick', {
|
|||||||
}).on('cycle', function(event) {
|
}).on('cycle', function(event) {
|
||||||
console.log(String(event.target));
|
console.log(String(event.target));
|
||||||
}).on('complete', function() {
|
}).on('complete', function() {
|
||||||
var winner = this.filter('fastest').pluck('name');
|
var winner = this.filter('fastest').map('name');
|
||||||
assert.strictEqual('sharp', String(winner), 'sharp was slower than ' + winner);
|
assert.strictEqual('sharp', String(winner), 'sharp was slower than ' + winner);
|
||||||
console.dir(sharp.cache());
|
console.dir(sharp.cache());
|
||||||
}).run();
|
}).run();
|
||||||
|
|||||||
BIN
test/fixtures/Landscape_1.jpg
vendored
Normal file
|
After Width: | Height: | Size: 136 KiB |
BIN
test/fixtures/Landscape_2.jpg
vendored
Normal file
|
After Width: | Height: | Size: 134 KiB |
BIN
test/fixtures/Landscape_3.jpg
vendored
Normal file
|
After Width: | Height: | Size: 138 KiB |
BIN
test/fixtures/Landscape_4.jpg
vendored
Normal file
|
After Width: | Height: | Size: 137 KiB |
BIN
test/fixtures/Landscape_6.jpg
vendored
Normal file
|
After Width: | Height: | Size: 134 KiB |
BIN
test/fixtures/Landscape_7.jpg
vendored
Normal file
|
After Width: | Height: | Size: 137 KiB |
BIN
test/fixtures/Portrait_1.jpg
vendored
Normal file
|
After Width: | Height: | Size: 126 KiB |
BIN
test/fixtures/Portrait_2.jpg
vendored
Normal file
|
After Width: | Height: | Size: 133 KiB |
BIN
test/fixtures/Portrait_3.jpg
vendored
Normal file
|
After Width: | Height: | Size: 133 KiB |
BIN
test/fixtures/Portrait_4.jpg
vendored
Normal file
|
After Width: | Height: | Size: 128 KiB |
BIN
test/fixtures/Portrait_5.jpg
vendored
Normal file
|
After Width: | Height: | Size: 131 KiB |
BIN
test/fixtures/Portrait_6.jpg
vendored
Normal file
|
After Width: | Height: | Size: 133 KiB |
BIN
test/fixtures/Portrait_7.jpg
vendored
Normal file
|
After Width: | Height: | Size: 132 KiB |
BIN
test/fixtures/Portrait_8.jpg
vendored
Normal file
|
After Width: | Height: | Size: 129 KiB |
17
test/fixtures/Wikimedia-logo.svg
vendored
@@ -1,17 +0,0 @@
|
|||||||
<?xml version="1.0" standalone="yes"?>
|
|
||||||
<!DOCTYPE svg PUBLIC "-//W3C//DTD SVG 1.1//EN" "http://www.w3.org/Graphics/SVG/1.1/DTD/svg11.dtd">
|
|
||||||
|
|
||||||
<svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" version="1.1"
|
|
||||||
id="Wikimedia logo"
|
|
||||||
viewBox="-599 -599 1198 1198" width="1024" height="1024">
|
|
||||||
<defs>
|
|
||||||
<clipPath id="mask">
|
|
||||||
<path d="M 47.5,-87.5 v 425 h -95 v -425 l -552,-552 v 1250 h 1199 v -1250 z" />
|
|
||||||
</clipPath>
|
|
||||||
</defs>
|
|
||||||
<g clip-path="url(#mask)">
|
|
||||||
<circle id="green parts" fill="#396" r="336.5"/>
|
|
||||||
<circle id="blue arc" fill="none" stroke="#069" r="480.25" stroke-width="135.5" />
|
|
||||||
</g>
|
|
||||||
<circle fill="#900" cy="-379.5" r="184.5" id="red circle"/>
|
|
||||||
</svg>
|
|
||||||
|
Before Width: | Height: | Size: 692 B |
3
test/fixtures/check.svg
vendored
Normal file
@@ -0,0 +1,3 @@
|
|||||||
|
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 100 100">
|
||||||
|
<path d="M30,76q6-14,13-26q6-12,14-23q8-12,13-17q3-4,6-6q1-1,5-2q8-1,12-1q1,0,1,1q0,1-1,2q-13,11-27,33q-14,21-24,44q-4,9-5,11q-1,2-9,2q-5,0-6-1q-1-1-5-6q-5-8-12-15q-3-4-3-6q0-2,4-5q3-2,6-2q3,0,8,3q5,4,10,14z" fill="green" />
|
||||||
|
</svg>
|
||||||
|
After Width: | Height: | Size: 301 B |
BIN
test/fixtures/expected/Landscape_1-out.jpg
vendored
Normal file
|
After Width: | Height: | Size: 109 KiB |
BIN
test/fixtures/expected/Landscape_2-out.jpg
vendored
Normal file
|
After Width: | Height: | Size: 103 KiB |
BIN
test/fixtures/expected/Landscape_3-out.jpg
vendored
Normal file
|
After Width: | Height: | Size: 103 KiB |
BIN
test/fixtures/expected/Landscape_4-out.jpg
vendored
Normal file
|
After Width: | Height: | Size: 103 KiB |
BIN
test/fixtures/expected/Landscape_5-out.jpg
vendored
Normal file
|
After Width: | Height: | Size: 103 KiB |
BIN
test/fixtures/expected/Landscape_6-out.jpg
vendored
Normal file
|
After Width: | Height: | Size: 103 KiB |
BIN
test/fixtures/expected/Landscape_7-out.jpg
vendored
Normal file
|
After Width: | Height: | Size: 103 KiB |
BIN
test/fixtures/expected/Landscape_8-out.jpg
vendored
Normal file
|
After Width: | Height: | Size: 103 KiB |
BIN
test/fixtures/expected/Portrait_1-out.jpg
vendored
Normal file
|
After Width: | Height: | Size: 169 KiB |
BIN
test/fixtures/expected/Portrait_2-out.jpg
vendored
Normal file
|
After Width: | Height: | Size: 166 KiB |
BIN
test/fixtures/expected/Portrait_3-out.jpg
vendored
Normal file
|
After Width: | Height: | Size: 165 KiB |
BIN
test/fixtures/expected/Portrait_4-out.jpg
vendored
Normal file
|
After Width: | Height: | Size: 166 KiB |
BIN
test/fixtures/expected/Portrait_5-out.jpg
vendored
Normal file
|
After Width: | Height: | Size: 172 KiB |
BIN
test/fixtures/expected/Portrait_6-out.jpg
vendored
Normal file
|
After Width: | Height: | Size: 167 KiB |
BIN
test/fixtures/expected/Portrait_7-out.jpg
vendored
Normal file
|
After Width: | Height: | Size: 164 KiB |
BIN
test/fixtures/expected/Portrait_8-out.jpg
vendored
Normal file
|
After Width: | Height: | Size: 165 KiB |
BIN
test/fixtures/expected/blur-0.3.jpg
vendored
Normal file
|
After Width: | Height: | Size: 13 KiB |
BIN
test/fixtures/expected/blur-1.jpg
vendored
Normal file
|
After Width: | Height: | Size: 10 KiB |
BIN
test/fixtures/expected/blur-10.jpg
vendored
Normal file
|
After Width: | Height: | Size: 4.8 KiB |
BIN
test/fixtures/expected/blur-mild.jpg
vendored
Normal file
|
After Width: | Height: | Size: 10 KiB |
BIN
test/fixtures/expected/colourspace.cmyk-without-profile.jpg
vendored
Normal file
|
After Width: | Height: | Size: 24 KiB |
BIN
test/fixtures/expected/colourspace.cmyk.jpg
vendored
Normal file
|
After Width: | Height: | Size: 25 KiB |
BIN
test/fixtures/expected/crop-entropy.jpg
vendored
Normal file
|
After Width: | Height: | Size: 8.3 KiB |
BIN
test/fixtures/expected/crop-entropy.png
vendored
Normal file
|
After Width: | Height: | Size: 6.0 KiB |
BIN
test/fixtures/expected/embed-16bit-rgba.png
vendored
Normal file
|
After Width: | Height: | Size: 755 B |
BIN
test/fixtures/expected/embed-16bit.png
vendored
Normal file
|
After Width: | Height: | Size: 827 B |
BIN
test/fixtures/expected/embed-2channel.png
vendored
Normal file
|
After Width: | Height: | Size: 703 B |
BIN
test/fixtures/expected/embed-3-into-3.png
vendored
Normal file
|
After Width: | Height: | Size: 116 KiB |
BIN
test/fixtures/expected/embed-3-into-4.webp
vendored
Normal file
|
After Width: | Height: | Size: 10 KiB |
BIN
test/fixtures/expected/embed-4-into-4.png
vendored
Normal file
|
After Width: | Height: | Size: 670 B |
BIN
test/fixtures/expected/extend-equal.jpg
vendored
Normal file
|
After Width: | Height: | Size: 4.1 KiB |
BIN
test/fixtures/expected/extend-unequal.png
vendored
Normal file
|
After Width: | Height: | Size: 18 KiB |
BIN
test/fixtures/expected/extract-resize.jpg
vendored
|
Before Width: | Height: | Size: 506 B After Width: | Height: | Size: 519 B |