| 1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561156215631564156515661567156815691570157115721573157415751576157715781579158015811582158315841585158615871588158915901591159215931594159515961597159815991600160116021603160416051606160716081609161016111612161316141615161616171618161916201621162216231624162516261627162816291630163116321633163416351636163716381639164016411642164316441645164616471648164916501651165216531654165516561657165816591660166116621663166416651666 |
- /*!
- Copyright 2013 Lovell Fuller and others.
- SPDX-License-Identifier: Apache-2.0
- */
- const path = require('node:path');
- const is = require('./is');
- const sharp = require('./sharp');
- const formats = new Map([
- ['heic', 'heif'],
- ['heif', 'heif'],
- ['avif', 'avif'],
- ['jpeg', 'jpeg'],
- ['jpg', 'jpeg'],
- ['jpe', 'jpeg'],
- ['tile', 'tile'],
- ['dz', 'tile'],
- ['png', 'png'],
- ['raw', 'raw'],
- ['tiff', 'tiff'],
- ['tif', 'tiff'],
- ['webp', 'webp'],
- ['gif', 'gif'],
- ['jp2', 'jp2'],
- ['jpx', 'jp2'],
- ['j2k', 'jp2'],
- ['j2c', 'jp2'],
- ['jxl', 'jxl']
- ]);
- const jp2Regex = /\.(jp[2x]|j2[kc])$/i;
- const errJp2Save = () => new Error('JP2 output requires libvips with support for OpenJPEG');
- const bitdepthFromColourCount = (colours) => 1 << 31 - Math.clz32(Math.ceil(Math.log2(colours)));
- /**
- * Write output image data to a file.
- *
- * If an explicit output format is not selected, it will be inferred from the extension,
- * with JPEG, PNG, WebP, AVIF, TIFF, GIF, DZI, and libvips' V format supported.
- * Note that raw pixel data is only supported for buffer output.
- *
- * By default all metadata will be removed, which includes EXIF-based orientation.
- * See {@link #withmetadata withMetadata} for control over this.
- *
- * The caller is responsible for ensuring directory structures and permissions exist.
- *
- * A `Promise` is returned when `callback` is not provided.
- *
- * @example
- * sharp(input)
- * .toFile('output.png', (err, info) => { ... });
- *
- * @example
- * sharp(input)
- * .toFile('output.png')
- * .then(info => { ... })
- * .catch(err => { ... });
- *
- * @param {string} fileOut - the path to write the image data to.
- * @param {Function} [callback] - called on completion with two arguments `(err, info)`.
- * `info` contains the output image `format`, `size` (bytes), `width`, `height`,
- * `channels` and `premultiplied` (indicating if premultiplication was used).
- * When using a crop strategy also contains `cropOffsetLeft` and `cropOffsetTop`.
- * When using the attention crop strategy also contains `attentionX` and `attentionY`, the focal point of the cropped region.
- * Animated output will also contain `pageHeight` and `pages`.
- * May also contain `textAutofitDpi` (dpi the font was rendered at) if image was created from text.
- * @returns {Promise<Object>} - when no callback is provided
- * @throws {Error} Invalid parameters
- */
- function toFile (fileOut, callback) {
- let err;
- if (!is.string(fileOut)) {
- err = new Error('Missing output file path');
- } else if (is.string(this.options.input.file) && path.resolve(this.options.input.file) === path.resolve(fileOut)) {
- err = new Error('Cannot use same file for input and output');
- } else if (jp2Regex.test(path.extname(fileOut)) && !this.constructor.format.jp2k.output.file) {
- err = errJp2Save();
- }
- if (err) {
- if (is.fn(callback)) {
- callback(err);
- } else {
- return Promise.reject(err);
- }
- } else {
- this.options.fileOut = fileOut;
- const stack = Error();
- return this._pipeline(callback, stack);
- }
- return this;
- }
- /**
- * Write output to a Buffer.
- * JPEG, PNG, WebP, AVIF, TIFF, GIF and raw pixel data output are supported.
- *
- * Use {@link #toformat toFormat} or one of the format-specific functions such as {@link #jpeg jpeg}, {@link #png png} etc. to set the output format.
- *
- * If no explicit format is set, the output format will match the input image, except SVG input which becomes PNG output.
- *
- * By default all metadata will be removed, which includes EXIF-based orientation.
- * See {@link #withmetadata withMetadata} for control over this.
- *
- * `callback`, if present, gets three arguments `(err, data, info)` where:
- * - `err` is an error, if any.
- * - `data` is the output image data.
- * - `info` contains the output image `format`, `size` (bytes), `width`, `height`,
- * `channels` and `premultiplied` (indicating if premultiplication was used).
- * When using a crop strategy also contains `cropOffsetLeft` and `cropOffsetTop`.
- * Animated output will also contain `pageHeight` and `pages`.
- * May also contain `textAutofitDpi` (dpi the font was rendered at) if image was created from text.
- *
- * A `Promise` is returned when `callback` is not provided.
- *
- * @example
- * sharp(input)
- * .toBuffer((err, data, info) => { ... });
- *
- * @example
- * sharp(input)
- * .toBuffer()
- * .then(data => { ... })
- * .catch(err => { ... });
- *
- * @example
- * sharp(input)
- * .png()
- * .toBuffer({ resolveWithObject: true })
- * .then(({ data, info }) => { ... })
- * .catch(err => { ... });
- *
- * @example
- * const { data, info } = await sharp('my-image.jpg')
- * // output the raw pixels
- * .raw()
- * .toBuffer({ resolveWithObject: true });
- *
- * // create a more type safe way to work with the raw pixel data
- * // this will not copy the data, instead it will change `data`s underlying ArrayBuffer
- * // so `data` and `pixelArray` point to the same memory location
- * const pixelArray = new Uint8ClampedArray(data.buffer);
- *
- * // When you are done changing the pixelArray, sharp takes the `pixelArray` as an input
- * const { width, height, channels } = info;
- * await sharp(pixelArray, { raw: { width, height, channels } })
- * .toFile('my-changed-image.jpg');
- *
- * @param {Object} [options]
- * @param {boolean} [options.resolveWithObject] Resolve the Promise with an Object containing `data` and `info` properties instead of resolving only with `data`.
- * @param {Function} [callback]
- * @returns {Promise<Buffer>} - when no callback is provided
- */
- function toBuffer (options, callback) {
- if (is.object(options)) {
- this._setBooleanOption('resolveWithObject', options.resolveWithObject);
- } else if (this.options.resolveWithObject) {
- this.options.resolveWithObject = false;
- }
- this.options.fileOut = '';
- const stack = Error();
- return this._pipeline(is.fn(options) ? options : callback, stack);
- }
- /**
- * Keep all EXIF metadata from the input image in the output image.
- *
- * EXIF metadata is unsupported for TIFF output.
- *
- * @since 0.33.0
- *
- * @example
- * const outputWithExif = await sharp(inputWithExif)
- * .keepExif()
- * .toBuffer();
- *
- * @returns {Sharp}
- */
- function keepExif () {
- this.options.keepMetadata |= 0b00001;
- return this;
- }
- /**
- * Set EXIF metadata in the output image, ignoring any EXIF in the input image.
- *
- * @since 0.33.0
- *
- * @example
- * const dataWithExif = await sharp(input)
- * .withExif({
- * IFD0: {
- * Copyright: 'The National Gallery'
- * },
- * IFD3: {
- * GPSLatitudeRef: 'N',
- * GPSLatitude: '51/1 30/1 3230/100',
- * GPSLongitudeRef: 'W',
- * GPSLongitude: '0/1 7/1 4366/100'
- * }
- * })
- * .toBuffer();
- *
- * @param {Object<string, Object<string, string>>} exif Object keyed by IFD0, IFD1 etc. of key/value string pairs to write as EXIF data.
- * @returns {Sharp}
- * @throws {Error} Invalid parameters
- */
- function withExif (exif) {
- if (is.object(exif)) {
- for (const [ifd, entries] of Object.entries(exif)) {
- if (is.object(entries)) {
- for (const [k, v] of Object.entries(entries)) {
- if (is.string(v)) {
- this.options.withExif[`exif-${ifd.toLowerCase()}-${k}`] = v;
- } else {
- throw is.invalidParameterError(`${ifd}.${k}`, 'string', v);
- }
- }
- } else {
- throw is.invalidParameterError(ifd, 'object', entries);
- }
- }
- } else {
- throw is.invalidParameterError('exif', 'object', exif);
- }
- this.options.withExifMerge = false;
- return this.keepExif();
- }
- /**
- * Update EXIF metadata from the input image in the output image.
- *
- * @since 0.33.0
- *
- * @example
- * const dataWithMergedExif = await sharp(inputWithExif)
- * .withExifMerge({
- * IFD0: {
- * Copyright: 'The National Gallery'
- * }
- * })
- * .toBuffer();
- *
- * @param {Object<string, Object<string, string>>} exif Object keyed by IFD0, IFD1 etc. of key/value string pairs to write as EXIF data.
- * @returns {Sharp}
- * @throws {Error} Invalid parameters
- */
- function withExifMerge (exif) {
- this.withExif(exif);
- this.options.withExifMerge = true;
- return this;
- }
- /**
- * Keep ICC profile from the input image in the output image.
- *
- * When input and output colour spaces differ, use with {@link /api-colour/#tocolourspace toColourspace} and optionally {@link /api-colour/#pipelinecolourspace pipelineColourspace}.
- *
- * @since 0.33.0
- *
- * @example
- * const outputWithIccProfile = await sharp(inputWithIccProfile)
- * .keepIccProfile()
- * .toBuffer();
- *
- * @example
- * const cmykOutputWithIccProfile = await sharp(cmykInputWithIccProfile)
- * .pipelineColourspace('cmyk')
- * .toColourspace('cmyk')
- * .keepIccProfile()
- * .toBuffer();
- *
- * @returns {Sharp}
- */
- function keepIccProfile () {
- this.options.keepMetadata |= 0b01000;
- return this;
- }
- /**
- * Transform using an ICC profile and attach to the output image.
- *
- * This can either be an absolute filesystem path or
- * built-in profile name (`srgb`, `p3`, `cmyk`).
- *
- * @since 0.33.0
- *
- * @example
- * const outputWithP3 = await sharp(input)
- * .withIccProfile('p3')
- * .toBuffer();
- *
- * @param {string} icc - Absolute filesystem path to output ICC profile or built-in profile name (srgb, p3, cmyk).
- * @param {Object} [options]
- * @param {number} [options.attach=true] Should the ICC profile be included in the output image metadata?
- * @returns {Sharp}
- * @throws {Error} Invalid parameters
- */
- function withIccProfile (icc, options) {
- if (is.string(icc)) {
- this.options.withIccProfile = icc;
- } else {
- throw is.invalidParameterError('icc', 'string', icc);
- }
- this.keepIccProfile();
- if (is.object(options)) {
- if (is.defined(options.attach)) {
- if (is.bool(options.attach)) {
- if (!options.attach) {
- this.options.keepMetadata &= ~0b01000;
- }
- } else {
- throw is.invalidParameterError('attach', 'boolean', options.attach);
- }
- }
- }
- return this;
- }
- /**
- * Keep XMP metadata from the input image in the output image.
- *
- * @since 0.34.3
- *
- * @example
- * const outputWithXmp = await sharp(inputWithXmp)
- * .keepXmp()
- * .toBuffer();
- *
- * @returns {Sharp}
- */
- function keepXmp () {
- this.options.keepMetadata |= 0b00010;
- return this;
- }
- /**
- * Set XMP metadata in the output image.
- *
- * Supported by PNG, JPEG, WebP, and TIFF output.
- *
- * @since 0.34.3
- *
- * @example
- * const xmpString = `
- * <?xml version="1.0"?>
- * <x:xmpmeta xmlns:x="adobe:ns:meta/">
- * <rdf:RDF xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#">
- * <rdf:Description rdf:about="" xmlns:dc="http://purl.org/dc/elements/1.1/">
- * <dc:creator><rdf:Seq><rdf:li>John Doe</rdf:li></rdf:Seq></dc:creator>
- * </rdf:Description>
- * </rdf:RDF>
- * </x:xmpmeta>`;
- *
- * const data = await sharp(input)
- * .withXmp(xmpString)
- * .toBuffer();
- *
- * @param {string} xmp String containing XMP metadata to be embedded in the output image.
- * @returns {Sharp}
- * @throws {Error} Invalid parameters
- */
- function withXmp (xmp) {
- if (is.string(xmp) && xmp.length > 0) {
- this.options.withXmp = xmp;
- this.options.keepMetadata |= 0b00010;
- } else {
- throw is.invalidParameterError('xmp', 'non-empty string', xmp);
- }
- return this;
- }
- /**
- * Keep all metadata (EXIF, ICC, XMP, IPTC) from the input image in the output image.
- *
- * The default behaviour, when `keepMetadata` is not used, is to convert to the device-independent
- * sRGB colour space and strip all metadata, including the removal of any ICC profile.
- *
- * @since 0.33.0
- *
- * @example
- * const outputWithMetadata = await sharp(inputWithMetadata)
- * .keepMetadata()
- * .toBuffer();
- *
- * @returns {Sharp}
- */
- function keepMetadata () {
- this.options.keepMetadata = 0b11111;
- return this;
- }
- /**
- * Keep most metadata (EXIF, XMP, IPTC) from the input image in the output image.
- *
- * This will also convert to and add a web-friendly sRGB ICC profile if appropriate.
- *
- * Allows orientation and density to be set or updated.
- *
- * @example
- * const outputSrgbWithMetadata = await sharp(inputRgbWithMetadata)
- * .withMetadata()
- * .toBuffer();
- *
- * @example
- * // Set output metadata to 96 DPI
- * const data = await sharp(input)
- * .withMetadata({ density: 96 })
- * .toBuffer();
- *
- * @param {Object} [options]
- * @param {number} [options.orientation] Used to update the EXIF `Orientation` tag, integer between 1 and 8.
- * @param {number} [options.density] Number of pixels per inch (DPI).
- * @returns {Sharp}
- * @throws {Error} Invalid parameters
- */
- function withMetadata (options) {
- this.keepMetadata();
- this.withIccProfile('srgb');
- if (is.object(options)) {
- if (is.defined(options.orientation)) {
- if (is.integer(options.orientation) && is.inRange(options.orientation, 1, 8)) {
- this.options.withMetadataOrientation = options.orientation;
- } else {
- throw is.invalidParameterError('orientation', 'integer between 1 and 8', options.orientation);
- }
- }
- if (is.defined(options.density)) {
- if (is.number(options.density) && options.density > 0) {
- this.options.withMetadataDensity = options.density;
- } else {
- throw is.invalidParameterError('density', 'positive number', options.density);
- }
- }
- if (is.defined(options.icc)) {
- this.withIccProfile(options.icc);
- }
- if (is.defined(options.exif)) {
- this.withExifMerge(options.exif);
- }
- }
- return this;
- }
- /**
- * Force output to a given format.
- *
- * @example
- * // Convert any input to PNG output
- * const data = await sharp(input)
- * .toFormat('png')
- * .toBuffer();
- *
- * @param {(string|Object)} format - as a string or an Object with an 'id' attribute
- * @param {Object} options - output options
- * @returns {Sharp}
- * @throws {Error} unsupported format or options
- */
- function toFormat (format, options) {
- const actualFormat = formats.get((is.object(format) && is.string(format.id) ? format.id : format).toLowerCase());
- if (!actualFormat) {
- throw is.invalidParameterError('format', `one of: ${[...formats.keys()].join(', ')}`, format);
- }
- return this[actualFormat](options);
- }
- /**
- * Use these JPEG options for output image.
- *
- * @example
- * // Convert any input to very high quality JPEG output
- * const data = await sharp(input)
- * .jpeg({
- * quality: 100,
- * chromaSubsampling: '4:4:4'
- * })
- * .toBuffer();
- *
- * @example
- * // Use mozjpeg to reduce output JPEG file size (slower)
- * const data = await sharp(input)
- * .jpeg({ mozjpeg: true })
- * .toBuffer();
- *
- * @param {Object} [options] - output options
- * @param {number} [options.quality=80] - quality, integer 1-100
- * @param {boolean} [options.progressive=false] - use progressive (interlace) scan
- * @param {string} [options.chromaSubsampling='4:2:0'] - set to '4:4:4' to prevent chroma subsampling otherwise defaults to '4:2:0' chroma subsampling
- * @param {boolean} [options.optimiseCoding=true] - optimise Huffman coding tables
- * @param {boolean} [options.optimizeCoding=true] - alternative spelling of optimiseCoding
- * @param {boolean} [options.mozjpeg=false] - use mozjpeg defaults, equivalent to `{ trellisQuantisation: true, overshootDeringing: true, optimiseScans: true, quantisationTable: 3 }`
- * @param {boolean} [options.trellisQuantisation=false] - apply trellis quantisation
- * @param {boolean} [options.overshootDeringing=false] - apply overshoot deringing
- * @param {boolean} [options.optimiseScans=false] - optimise progressive scans, forces progressive
- * @param {boolean} [options.optimizeScans=false] - alternative spelling of optimiseScans
- * @param {number} [options.quantisationTable=0] - quantization table to use, integer 0-8
- * @param {number} [options.quantizationTable=0] - alternative spelling of quantisationTable
- * @param {boolean} [options.force=true] - force JPEG output, otherwise attempt to use input format
- * @returns {Sharp}
- * @throws {Error} Invalid options
- */
- function jpeg (options) {
- if (is.object(options)) {
- if (is.defined(options.quality)) {
- if (is.integer(options.quality) && is.inRange(options.quality, 1, 100)) {
- this.options.jpegQuality = options.quality;
- } else {
- throw is.invalidParameterError('quality', 'integer between 1 and 100', options.quality);
- }
- }
- if (is.defined(options.progressive)) {
- this._setBooleanOption('jpegProgressive', options.progressive);
- }
- if (is.defined(options.chromaSubsampling)) {
- if (is.string(options.chromaSubsampling) && is.inArray(options.chromaSubsampling, ['4:2:0', '4:4:4'])) {
- this.options.jpegChromaSubsampling = options.chromaSubsampling;
- } else {
- throw is.invalidParameterError('chromaSubsampling', 'one of: 4:2:0, 4:4:4', options.chromaSubsampling);
- }
- }
- const optimiseCoding = is.bool(options.optimizeCoding) ? options.optimizeCoding : options.optimiseCoding;
- if (is.defined(optimiseCoding)) {
- this._setBooleanOption('jpegOptimiseCoding', optimiseCoding);
- }
- if (is.defined(options.mozjpeg)) {
- if (is.bool(options.mozjpeg)) {
- if (options.mozjpeg) {
- this.options.jpegTrellisQuantisation = true;
- this.options.jpegOvershootDeringing = true;
- this.options.jpegOptimiseScans = true;
- this.options.jpegProgressive = true;
- this.options.jpegQuantisationTable = 3;
- }
- } else {
- throw is.invalidParameterError('mozjpeg', 'boolean', options.mozjpeg);
- }
- }
- const trellisQuantisation = is.bool(options.trellisQuantization) ? options.trellisQuantization : options.trellisQuantisation;
- if (is.defined(trellisQuantisation)) {
- this._setBooleanOption('jpegTrellisQuantisation', trellisQuantisation);
- }
- if (is.defined(options.overshootDeringing)) {
- this._setBooleanOption('jpegOvershootDeringing', options.overshootDeringing);
- }
- const optimiseScans = is.bool(options.optimizeScans) ? options.optimizeScans : options.optimiseScans;
- if (is.defined(optimiseScans)) {
- this._setBooleanOption('jpegOptimiseScans', optimiseScans);
- if (optimiseScans) {
- this.options.jpegProgressive = true;
- }
- }
- const quantisationTable = is.number(options.quantizationTable) ? options.quantizationTable : options.quantisationTable;
- if (is.defined(quantisationTable)) {
- if (is.integer(quantisationTable) && is.inRange(quantisationTable, 0, 8)) {
- this.options.jpegQuantisationTable = quantisationTable;
- } else {
- throw is.invalidParameterError('quantisationTable', 'integer between 0 and 8', quantisationTable);
- }
- }
- }
- return this._updateFormatOut('jpeg', options);
- }
- /**
- * Use these PNG options for output image.
- *
- * By default, PNG output is full colour at 8 bits per pixel.
- *
- * Indexed PNG input at 1, 2 or 4 bits per pixel is converted to 8 bits per pixel.
- * Set `palette` to `true` for slower, indexed PNG output.
- *
- * For 16 bits per pixel output, convert to `rgb16` via
- * {@link /api-colour/#tocolourspace toColourspace}.
- *
- * @example
- * // Convert any input to full colour PNG output
- * const data = await sharp(input)
- * .png()
- * .toBuffer();
- *
- * @example
- * // Convert any input to indexed PNG output (slower)
- * const data = await sharp(input)
- * .png({ palette: true })
- * .toBuffer();
- *
- * @example
- * // Output 16 bits per pixel RGB(A)
- * const data = await sharp(input)
- * .toColourspace('rgb16')
- * .png()
- * .toBuffer();
- *
- * @param {Object} [options]
- * @param {boolean} [options.progressive=false] - use progressive (interlace) scan
- * @param {number} [options.compressionLevel=6] - zlib compression level, 0 (fastest, largest) to 9 (slowest, smallest)
- * @param {boolean} [options.adaptiveFiltering=false] - use adaptive row filtering
- * @param {boolean} [options.palette=false] - quantise to a palette-based image with alpha transparency support
- * @param {number} [options.quality=100] - use the lowest number of colours needed to achieve given quality, sets `palette` to `true`
- * @param {number} [options.effort=7] - CPU effort, between 1 (fastest) and 10 (slowest), sets `palette` to `true`
- * @param {number} [options.colours=256] - maximum number of palette entries, sets `palette` to `true`
- * @param {number} [options.colors=256] - alternative spelling of `options.colours`, sets `palette` to `true`
- * @param {number} [options.dither=1.0] - level of Floyd-Steinberg error diffusion, sets `palette` to `true`
- * @param {boolean} [options.force=true] - force PNG output, otherwise attempt to use input format
- * @returns {Sharp}
- * @throws {Error} Invalid options
- */
- function png (options) {
- if (is.object(options)) {
- if (is.defined(options.progressive)) {
- this._setBooleanOption('pngProgressive', options.progressive);
- }
- if (is.defined(options.compressionLevel)) {
- if (is.integer(options.compressionLevel) && is.inRange(options.compressionLevel, 0, 9)) {
- this.options.pngCompressionLevel = options.compressionLevel;
- } else {
- throw is.invalidParameterError('compressionLevel', 'integer between 0 and 9', options.compressionLevel);
- }
- }
- if (is.defined(options.adaptiveFiltering)) {
- this._setBooleanOption('pngAdaptiveFiltering', options.adaptiveFiltering);
- }
- const colours = options.colours || options.colors;
- if (is.defined(colours)) {
- if (is.integer(colours) && is.inRange(colours, 2, 256)) {
- this.options.pngBitdepth = bitdepthFromColourCount(colours);
- } else {
- throw is.invalidParameterError('colours', 'integer between 2 and 256', colours);
- }
- }
- if (is.defined(options.palette)) {
- this._setBooleanOption('pngPalette', options.palette);
- } else if ([options.quality, options.effort, options.colours, options.colors, options.dither].some(is.defined)) {
- this._setBooleanOption('pngPalette', true);
- }
- if (this.options.pngPalette) {
- if (is.defined(options.quality)) {
- if (is.integer(options.quality) && is.inRange(options.quality, 0, 100)) {
- this.options.pngQuality = options.quality;
- } else {
- throw is.invalidParameterError('quality', 'integer between 0 and 100', options.quality);
- }
- }
- if (is.defined(options.effort)) {
- if (is.integer(options.effort) && is.inRange(options.effort, 1, 10)) {
- this.options.pngEffort = options.effort;
- } else {
- throw is.invalidParameterError('effort', 'integer between 1 and 10', options.effort);
- }
- }
- if (is.defined(options.dither)) {
- if (is.number(options.dither) && is.inRange(options.dither, 0, 1)) {
- this.options.pngDither = options.dither;
- } else {
- throw is.invalidParameterError('dither', 'number between 0.0 and 1.0', options.dither);
- }
- }
- }
- }
- return this._updateFormatOut('png', options);
- }
- /**
- * Use these WebP options for output image.
- *
- * @example
- * // Convert any input to lossless WebP output
- * const data = await sharp(input)
- * .webp({ lossless: true })
- * .toBuffer();
- *
- * @example
- * // Optimise the file size of an animated WebP
- * const outputWebp = await sharp(inputWebp, { animated: true })
- * .webp({ effort: 6 })
- * .toBuffer();
- *
- * @param {Object} [options] - output options
- * @param {number} [options.quality=80] - quality, integer 1-100
- * @param {number} [options.alphaQuality=100] - quality of alpha layer, integer 0-100
- * @param {boolean} [options.lossless=false] - use lossless compression mode
- * @param {boolean} [options.nearLossless=false] - use near_lossless compression mode
- * @param {boolean} [options.smartSubsample=false] - use high quality chroma subsampling
- * @param {boolean} [options.smartDeblock=false] - auto-adjust the deblocking filter, can improve low contrast edges (slow)
- * @param {string} [options.preset='default'] - named preset for preprocessing/filtering, one of: default, photo, picture, drawing, icon, text
- * @param {number} [options.effort=4] - CPU effort, between 0 (fastest) and 6 (slowest)
- * @param {number} [options.loop=0] - number of animation iterations, use 0 for infinite animation
- * @param {number|number[]} [options.delay] - delay(s) between animation frames (in milliseconds)
- * @param {boolean} [options.minSize=false] - prevent use of animation key frames to minimise file size (slow)
- * @param {boolean} [options.mixed=false] - allow mixture of lossy and lossless animation frames (slow)
- * @param {boolean} [options.force=true] - force WebP output, otherwise attempt to use input format
- * @returns {Sharp}
- * @throws {Error} Invalid options
- */
- function webp (options) {
- if (is.object(options)) {
- if (is.defined(options.quality)) {
- if (is.integer(options.quality) && is.inRange(options.quality, 1, 100)) {
- this.options.webpQuality = options.quality;
- } else {
- throw is.invalidParameterError('quality', 'integer between 1 and 100', options.quality);
- }
- }
- if (is.defined(options.alphaQuality)) {
- if (is.integer(options.alphaQuality) && is.inRange(options.alphaQuality, 0, 100)) {
- this.options.webpAlphaQuality = options.alphaQuality;
- } else {
- throw is.invalidParameterError('alphaQuality', 'integer between 0 and 100', options.alphaQuality);
- }
- }
- if (is.defined(options.lossless)) {
- this._setBooleanOption('webpLossless', options.lossless);
- }
- if (is.defined(options.nearLossless)) {
- this._setBooleanOption('webpNearLossless', options.nearLossless);
- }
- if (is.defined(options.smartSubsample)) {
- this._setBooleanOption('webpSmartSubsample', options.smartSubsample);
- }
- if (is.defined(options.smartDeblock)) {
- this._setBooleanOption('webpSmartDeblock', options.smartDeblock);
- }
- if (is.defined(options.preset)) {
- if (is.string(options.preset) && is.inArray(options.preset, ['default', 'photo', 'picture', 'drawing', 'icon', 'text'])) {
- this.options.webpPreset = options.preset;
- } else {
- throw is.invalidParameterError('preset', 'one of: default, photo, picture, drawing, icon, text', options.preset);
- }
- }
- if (is.defined(options.effort)) {
- if (is.integer(options.effort) && is.inRange(options.effort, 0, 6)) {
- this.options.webpEffort = options.effort;
- } else {
- throw is.invalidParameterError('effort', 'integer between 0 and 6', options.effort);
- }
- }
- if (is.defined(options.minSize)) {
- this._setBooleanOption('webpMinSize', options.minSize);
- }
- if (is.defined(options.mixed)) {
- this._setBooleanOption('webpMixed', options.mixed);
- }
- }
- trySetAnimationOptions(options, this.options);
- return this._updateFormatOut('webp', options);
- }
- /**
- * Use these GIF options for the output image.
- *
- * The first entry in the palette is reserved for transparency.
- *
- * The palette of the input image will be re-used if possible.
- *
- * @since 0.30.0
- *
- * @example
- * // Convert PNG to GIF
- * await sharp(pngBuffer)
- * .gif()
- * .toBuffer();
- *
- * @example
- * // Convert animated WebP to animated GIF
- * await sharp('animated.webp', { animated: true })
- * .toFile('animated.gif');
- *
- * @example
- * // Create a 128x128, cropped, non-dithered, animated thumbnail of an animated GIF
- * const out = await sharp('in.gif', { animated: true })
- * .resize({ width: 128, height: 128 })
- * .gif({ dither: 0 })
- * .toBuffer();
- *
- * @example
- * // Lossy file size reduction of animated GIF
- * await sharp('in.gif', { animated: true })
- * .gif({ interFrameMaxError: 8 })
- * .toFile('optim.gif');
- *
- * @param {Object} [options] - output options
- * @param {boolean} [options.reuse=true] - re-use existing palette, otherwise generate new (slow)
- * @param {boolean} [options.progressive=false] - use progressive (interlace) scan
- * @param {number} [options.colours=256] - maximum number of palette entries, including transparency, between 2 and 256
- * @param {number} [options.colors=256] - alternative spelling of `options.colours`
- * @param {number} [options.effort=7] - CPU effort, between 1 (fastest) and 10 (slowest)
- * @param {number} [options.dither=1.0] - level of Floyd-Steinberg error diffusion, between 0 (least) and 1 (most)
- * @param {number} [options.interFrameMaxError=0] - maximum inter-frame error for transparency, between 0 (lossless) and 32
- * @param {number} [options.interPaletteMaxError=3] - maximum inter-palette error for palette reuse, between 0 and 256
- * @param {boolean} [options.keepDuplicateFrames=false] - keep duplicate frames in the output instead of combining them
- * @param {number} [options.loop=0] - number of animation iterations, use 0 for infinite animation
- * @param {number|number[]} [options.delay] - delay(s) between animation frames (in milliseconds)
- * @param {boolean} [options.force=true] - force GIF output, otherwise attempt to use input format
- * @returns {Sharp}
- * @throws {Error} Invalid options
- */
- function gif (options) {
- if (is.object(options)) {
- if (is.defined(options.reuse)) {
- this._setBooleanOption('gifReuse', options.reuse);
- }
- if (is.defined(options.progressive)) {
- this._setBooleanOption('gifProgressive', options.progressive);
- }
- const colours = options.colours || options.colors;
- if (is.defined(colours)) {
- if (is.integer(colours) && is.inRange(colours, 2, 256)) {
- this.options.gifBitdepth = bitdepthFromColourCount(colours);
- } else {
- throw is.invalidParameterError('colours', 'integer between 2 and 256', colours);
- }
- }
- if (is.defined(options.effort)) {
- if (is.number(options.effort) && is.inRange(options.effort, 1, 10)) {
- this.options.gifEffort = options.effort;
- } else {
- throw is.invalidParameterError('effort', 'integer between 1 and 10', options.effort);
- }
- }
- if (is.defined(options.dither)) {
- if (is.number(options.dither) && is.inRange(options.dither, 0, 1)) {
- this.options.gifDither = options.dither;
- } else {
- throw is.invalidParameterError('dither', 'number between 0.0 and 1.0', options.dither);
- }
- }
- if (is.defined(options.interFrameMaxError)) {
- if (is.number(options.interFrameMaxError) && is.inRange(options.interFrameMaxError, 0, 32)) {
- this.options.gifInterFrameMaxError = options.interFrameMaxError;
- } else {
- throw is.invalidParameterError('interFrameMaxError', 'number between 0.0 and 32.0', options.interFrameMaxError);
- }
- }
- if (is.defined(options.interPaletteMaxError)) {
- if (is.number(options.interPaletteMaxError) && is.inRange(options.interPaletteMaxError, 0, 256)) {
- this.options.gifInterPaletteMaxError = options.interPaletteMaxError;
- } else {
- throw is.invalidParameterError('interPaletteMaxError', 'number between 0.0 and 256.0', options.interPaletteMaxError);
- }
- }
- if (is.defined(options.keepDuplicateFrames)) {
- if (is.bool(options.keepDuplicateFrames)) {
- this._setBooleanOption('gifKeepDuplicateFrames', options.keepDuplicateFrames);
- } else {
- throw is.invalidParameterError('keepDuplicateFrames', 'boolean', options.keepDuplicateFrames);
- }
- }
- }
- trySetAnimationOptions(options, this.options);
- return this._updateFormatOut('gif', options);
- }
- /**
- * Use these JP2 options for output image.
- *
- * Requires libvips compiled with support for OpenJPEG.
- * The prebuilt binaries do not include this - see
- * {@link /install#custom-libvips installing a custom libvips}.
- *
- * @example
- * // Convert any input to lossless JP2 output
- * const data = await sharp(input)
- * .jp2({ lossless: true })
- * .toBuffer();
- *
- * @example
- * // Convert any input to very high quality JP2 output
- * const data = await sharp(input)
- * .jp2({
- * quality: 100,
- * chromaSubsampling: '4:4:4'
- * })
- * .toBuffer();
- *
- * @since 0.29.1
- *
- * @param {Object} [options] - output options
- * @param {number} [options.quality=80] - quality, integer 1-100
- * @param {boolean} [options.lossless=false] - use lossless compression mode
- * @param {number} [options.tileWidth=512] - horizontal tile size
- * @param {number} [options.tileHeight=512] - vertical tile size
- * @param {string} [options.chromaSubsampling='4:4:4'] - set to '4:2:0' to use chroma subsampling
- * @returns {Sharp}
- * @throws {Error} Invalid options
- */
- function jp2 (options) {
- /* node:coverage ignore next 41 */
- if (!this.constructor.format.jp2k.output.buffer) {
- throw errJp2Save();
- }
- if (is.object(options)) {
- if (is.defined(options.quality)) {
- if (is.integer(options.quality) && is.inRange(options.quality, 1, 100)) {
- this.options.jp2Quality = options.quality;
- } else {
- throw is.invalidParameterError('quality', 'integer between 1 and 100', options.quality);
- }
- }
- if (is.defined(options.lossless)) {
- if (is.bool(options.lossless)) {
- this.options.jp2Lossless = options.lossless;
- } else {
- throw is.invalidParameterError('lossless', 'boolean', options.lossless);
- }
- }
- if (is.defined(options.tileWidth)) {
- if (is.integer(options.tileWidth) && is.inRange(options.tileWidth, 1, 32768)) {
- this.options.jp2TileWidth = options.tileWidth;
- } else {
- throw is.invalidParameterError('tileWidth', 'integer between 1 and 32768', options.tileWidth);
- }
- }
- if (is.defined(options.tileHeight)) {
- if (is.integer(options.tileHeight) && is.inRange(options.tileHeight, 1, 32768)) {
- this.options.jp2TileHeight = options.tileHeight;
- } else {
- throw is.invalidParameterError('tileHeight', 'integer between 1 and 32768', options.tileHeight);
- }
- }
- if (is.defined(options.chromaSubsampling)) {
- if (is.string(options.chromaSubsampling) && is.inArray(options.chromaSubsampling, ['4:2:0', '4:4:4'])) {
- this.options.jp2ChromaSubsampling = options.chromaSubsampling;
- } else {
- throw is.invalidParameterError('chromaSubsampling', 'one of: 4:2:0, 4:4:4', options.chromaSubsampling);
- }
- }
- }
- return this._updateFormatOut('jp2', options);
- }
- /**
- * Set animation options if available.
- * @private
- *
- * @param {Object} [source] - output options
- * @param {number} [source.loop=0] - number of animation iterations, use 0 for infinite animation
- * @param {number[]} [source.delay] - list of delays between animation frames (in milliseconds)
- * @param {Object} [target] - target object for valid options
- * @throws {Error} Invalid options
- */
- function trySetAnimationOptions (source, target) {
- if (is.object(source) && is.defined(source.loop)) {
- if (is.integer(source.loop) && is.inRange(source.loop, 0, 65535)) {
- target.loop = source.loop;
- } else {
- throw is.invalidParameterError('loop', 'integer between 0 and 65535', source.loop);
- }
- }
- if (is.object(source) && is.defined(source.delay)) {
- // We allow singular values as well
- if (is.integer(source.delay) && is.inRange(source.delay, 0, 65535)) {
- target.delay = [source.delay];
- } else if (
- Array.isArray(source.delay) &&
- source.delay.every(is.integer) &&
- source.delay.every(v => is.inRange(v, 0, 65535))) {
- target.delay = source.delay;
- } else {
- throw is.invalidParameterError('delay', 'integer or an array of integers between 0 and 65535', source.delay);
- }
- }
- }
- /**
- * Use these TIFF options for output image.
- *
- * The `density` can be set in pixels/inch via {@link #withmetadata withMetadata}
- * instead of providing `xres` and `yres` in pixels/mm.
- *
- * @example
- * // Convert SVG input to LZW-compressed, 1 bit per pixel TIFF output
- * sharp('input.svg')
- * .tiff({
- * compression: 'lzw',
- * bitdepth: 1
- * })
- * .toFile('1-bpp-output.tiff')
- * .then(info => { ... });
- *
- * @param {Object} [options] - output options
- * @param {number} [options.quality=80] - quality, integer 1-100
- * @param {boolean} [options.force=true] - force TIFF output, otherwise attempt to use input format
- * @param {string} [options.compression='jpeg'] - compression options: none, jpeg, deflate, packbits, ccittfax4, lzw, webp, zstd, jp2k
- * @param {boolean} [options.bigtiff=false] - use BigTIFF variant (has no effect when compression is none)
- * @param {string} [options.predictor='horizontal'] - compression predictor options: none, horizontal, float
- * @param {boolean} [options.pyramid=false] - write an image pyramid
- * @param {boolean} [options.tile=false] - write a tiled tiff
- * @param {number} [options.tileWidth=256] - horizontal tile size
- * @param {number} [options.tileHeight=256] - vertical tile size
- * @param {number} [options.xres=1.0] - horizontal resolution in pixels/mm
- * @param {number} [options.yres=1.0] - vertical resolution in pixels/mm
- * @param {string} [options.resolutionUnit='inch'] - resolution unit options: inch, cm
- * @param {number} [options.bitdepth=8] - reduce bitdepth to 1, 2 or 4 bit
- * @param {boolean} [options.miniswhite=false] - write 1-bit images as miniswhite
- * @returns {Sharp}
- * @throws {Error} Invalid options
- */
- function tiff (options) {
- if (is.object(options)) {
- if (is.defined(options.quality)) {
- if (is.integer(options.quality) && is.inRange(options.quality, 1, 100)) {
- this.options.tiffQuality = options.quality;
- } else {
- throw is.invalidParameterError('quality', 'integer between 1 and 100', options.quality);
- }
- }
- if (is.defined(options.bitdepth)) {
- if (is.integer(options.bitdepth) && is.inArray(options.bitdepth, [1, 2, 4, 8])) {
- this.options.tiffBitdepth = options.bitdepth;
- } else {
- throw is.invalidParameterError('bitdepth', '1, 2, 4 or 8', options.bitdepth);
- }
- }
- // tiling
- if (is.defined(options.tile)) {
- this._setBooleanOption('tiffTile', options.tile);
- }
- if (is.defined(options.tileWidth)) {
- if (is.integer(options.tileWidth) && options.tileWidth > 0) {
- this.options.tiffTileWidth = options.tileWidth;
- } else {
- throw is.invalidParameterError('tileWidth', 'integer greater than zero', options.tileWidth);
- }
- }
- if (is.defined(options.tileHeight)) {
- if (is.integer(options.tileHeight) && options.tileHeight > 0) {
- this.options.tiffTileHeight = options.tileHeight;
- } else {
- throw is.invalidParameterError('tileHeight', 'integer greater than zero', options.tileHeight);
- }
- }
- // miniswhite
- if (is.defined(options.miniswhite)) {
- this._setBooleanOption('tiffMiniswhite', options.miniswhite);
- }
- // pyramid
- if (is.defined(options.pyramid)) {
- this._setBooleanOption('tiffPyramid', options.pyramid);
- }
- // resolution
- if (is.defined(options.xres)) {
- if (is.number(options.xres) && options.xres > 0) {
- this.options.tiffXres = options.xres;
- } else {
- throw is.invalidParameterError('xres', 'number greater than zero', options.xres);
- }
- }
- if (is.defined(options.yres)) {
- if (is.number(options.yres) && options.yres > 0) {
- this.options.tiffYres = options.yres;
- } else {
- throw is.invalidParameterError('yres', 'number greater than zero', options.yres);
- }
- }
- // compression
- if (is.defined(options.compression)) {
- if (is.string(options.compression) && is.inArray(options.compression, ['none', 'jpeg', 'deflate', 'packbits', 'ccittfax4', 'lzw', 'webp', 'zstd', 'jp2k'])) {
- this.options.tiffCompression = options.compression;
- } else {
- throw is.invalidParameterError('compression', 'one of: none, jpeg, deflate, packbits, ccittfax4, lzw, webp, zstd, jp2k', options.compression);
- }
- }
- // bigtiff
- if (is.defined(options.bigtiff)) {
- this._setBooleanOption('tiffBigtiff', options.bigtiff);
- }
- // predictor
- if (is.defined(options.predictor)) {
- if (is.string(options.predictor) && is.inArray(options.predictor, ['none', 'horizontal', 'float'])) {
- this.options.tiffPredictor = options.predictor;
- } else {
- throw is.invalidParameterError('predictor', 'one of: none, horizontal, float', options.predictor);
- }
- }
- // resolutionUnit
- if (is.defined(options.resolutionUnit)) {
- if (is.string(options.resolutionUnit) && is.inArray(options.resolutionUnit, ['inch', 'cm'])) {
- this.options.tiffResolutionUnit = options.resolutionUnit;
- } else {
- throw is.invalidParameterError('resolutionUnit', 'one of: inch, cm', options.resolutionUnit);
- }
- }
- }
- return this._updateFormatOut('tiff', options);
- }
- /**
- * Use these AVIF options for output image.
- *
- * AVIF image sequences are not supported.
- * Prebuilt binaries support a bitdepth of 8 only.
- *
- * This feature is experimental on the Windows ARM64 platform
- * and requires a CPU with ARM64v8.4 or later.
- *
- * @example
- * const data = await sharp(input)
- * .avif({ effort: 2 })
- * .toBuffer();
- *
- * @example
- * const data = await sharp(input)
- * .avif({ lossless: true })
- * .toBuffer();
- *
- * @since 0.27.0
- *
- * @param {Object} [options] - output options
- * @param {number} [options.quality=50] - quality, integer 1-100
- * @param {boolean} [options.lossless=false] - use lossless compression
- * @param {number} [options.effort=4] - CPU effort, between 0 (fastest) and 9 (slowest)
- * @param {string} [options.chromaSubsampling='4:4:4'] - set to '4:2:0' to use chroma subsampling
- * @param {number} [options.bitdepth=8] - set bitdepth to 8, 10 or 12 bit
- * @returns {Sharp}
- * @throws {Error} Invalid options
- */
- function avif (options) {
- return this.heif({ ...options, compression: 'av1' });
- }
- /**
- * Use these HEIF options for output image.
- *
- * Support for patent-encumbered HEIC images using `hevc` compression requires the use of a
- * globally-installed libvips compiled with support for libheif, libde265 and x265.
- *
- * @example
- * const data = await sharp(input)
- * .heif({ compression: 'hevc' })
- * .toBuffer();
- *
- * @since 0.23.0
- *
- * @param {Object} options - output options
- * @param {string} options.compression - compression format: av1, hevc
- * @param {number} [options.quality=50] - quality, integer 1-100
- * @param {boolean} [options.lossless=false] - use lossless compression
- * @param {number} [options.effort=4] - CPU effort, between 0 (fastest) and 9 (slowest)
- * @param {string} [options.chromaSubsampling='4:4:4'] - set to '4:2:0' to use chroma subsampling
- * @param {number} [options.bitdepth=8] - set bitdepth to 8, 10 or 12 bit
- * @returns {Sharp}
- * @throws {Error} Invalid options
- */
- function heif (options) {
- if (is.object(options)) {
- if (is.string(options.compression) && is.inArray(options.compression, ['av1', 'hevc'])) {
- this.options.heifCompression = options.compression;
- } else {
- throw is.invalidParameterError('compression', 'one of: av1, hevc', options.compression);
- }
- if (is.defined(options.quality)) {
- if (is.integer(options.quality) && is.inRange(options.quality, 1, 100)) {
- this.options.heifQuality = options.quality;
- } else {
- throw is.invalidParameterError('quality', 'integer between 1 and 100', options.quality);
- }
- }
- if (is.defined(options.lossless)) {
- if (is.bool(options.lossless)) {
- this.options.heifLossless = options.lossless;
- } else {
- throw is.invalidParameterError('lossless', 'boolean', options.lossless);
- }
- }
- if (is.defined(options.effort)) {
- if (is.integer(options.effort) && is.inRange(options.effort, 0, 9)) {
- this.options.heifEffort = options.effort;
- } else {
- throw is.invalidParameterError('effort', 'integer between 0 and 9', options.effort);
- }
- }
- if (is.defined(options.chromaSubsampling)) {
- if (is.string(options.chromaSubsampling) && is.inArray(options.chromaSubsampling, ['4:2:0', '4:4:4'])) {
- this.options.heifChromaSubsampling = options.chromaSubsampling;
- } else {
- throw is.invalidParameterError('chromaSubsampling', 'one of: 4:2:0, 4:4:4', options.chromaSubsampling);
- }
- }
- if (is.defined(options.bitdepth)) {
- if (is.integer(options.bitdepth) && is.inArray(options.bitdepth, [8, 10, 12])) {
- if (options.bitdepth !== 8 && this.constructor.versions.heif) {
- throw is.invalidParameterError('bitdepth when using prebuilt binaries', 8, options.bitdepth);
- }
- this.options.heifBitdepth = options.bitdepth;
- } else {
- throw is.invalidParameterError('bitdepth', '8, 10 or 12', options.bitdepth);
- }
- }
- } else {
- throw is.invalidParameterError('options', 'Object', options);
- }
- return this._updateFormatOut('heif', options);
- }
- /**
- * Use these JPEG-XL (JXL) options for output image.
- *
- * This feature is experimental, please do not use in production systems.
- *
- * Requires libvips compiled with support for libjxl.
- * The prebuilt binaries do not include this - see
- * {@link /install/#custom-libvips installing a custom libvips}.
- *
- * @since 0.31.3
- *
- * @param {Object} [options] - output options
- * @param {number} [options.distance=1.0] - maximum encoding error, between 0 (highest quality) and 15 (lowest quality)
- * @param {number} [options.quality] - calculate `distance` based on JPEG-like quality, between 1 and 100, overrides distance if specified
- * @param {number} [options.decodingTier=0] - target decode speed tier, between 0 (highest quality) and 4 (lowest quality)
- * @param {boolean} [options.lossless=false] - use lossless compression
- * @param {number} [options.effort=7] - CPU effort, between 1 (fastest) and 9 (slowest)
- * @param {number} [options.loop=0] - number of animation iterations, use 0 for infinite animation
- * @param {number|number[]} [options.delay] - delay(s) between animation frames (in milliseconds)
- * @returns {Sharp}
- * @throws {Error} Invalid options
- */
- function jxl (options) {
- if (is.object(options)) {
- if (is.defined(options.quality)) {
- if (is.integer(options.quality) && is.inRange(options.quality, 1, 100)) {
- // https://github.com/libjxl/libjxl/blob/0aeea7f180bafd6893c1db8072dcb67d2aa5b03d/tools/cjxl_main.cc#L640-L644
- this.options.jxlDistance = options.quality >= 30
- ? 0.1 + (100 - options.quality) * 0.09
- : 53 / 3000 * options.quality * options.quality - 23 / 20 * options.quality + 25;
- } else {
- throw is.invalidParameterError('quality', 'integer between 1 and 100', options.quality);
- }
- } else if (is.defined(options.distance)) {
- if (is.number(options.distance) && is.inRange(options.distance, 0, 15)) {
- this.options.jxlDistance = options.distance;
- } else {
- throw is.invalidParameterError('distance', 'number between 0.0 and 15.0', options.distance);
- }
- }
- if (is.defined(options.decodingTier)) {
- if (is.integer(options.decodingTier) && is.inRange(options.decodingTier, 0, 4)) {
- this.options.jxlDecodingTier = options.decodingTier;
- } else {
- throw is.invalidParameterError('decodingTier', 'integer between 0 and 4', options.decodingTier);
- }
- }
- if (is.defined(options.lossless)) {
- if (is.bool(options.lossless)) {
- this.options.jxlLossless = options.lossless;
- } else {
- throw is.invalidParameterError('lossless', 'boolean', options.lossless);
- }
- }
- if (is.defined(options.effort)) {
- if (is.integer(options.effort) && is.inRange(options.effort, 1, 9)) {
- this.options.jxlEffort = options.effort;
- } else {
- throw is.invalidParameterError('effort', 'integer between 1 and 9', options.effort);
- }
- }
- }
- trySetAnimationOptions(options, this.options);
- return this._updateFormatOut('jxl', options);
- }
- /**
- * Force output to be raw, uncompressed pixel data.
- * Pixel ordering is left-to-right, top-to-bottom, without padding.
- * Channel ordering will be RGB or RGBA for non-greyscale colourspaces.
- *
- * @example
- * // Extract raw, unsigned 8-bit RGB pixel data from JPEG input
- * const { data, info } = await sharp('input.jpg')
- * .raw()
- * .toBuffer({ resolveWithObject: true });
- *
- * @example
- * // Extract alpha channel as raw, unsigned 16-bit pixel data from PNG input
- * const data = await sharp('input.png')
- * .ensureAlpha()
- * .extractChannel(3)
- * .toColourspace('b-w')
- * .raw({ depth: 'ushort' })
- * .toBuffer();
- *
- * @param {Object} [options] - output options
- * @param {string} [options.depth='uchar'] - bit depth, one of: char, uchar (default), short, ushort, int, uint, float, complex, double, dpcomplex
- * @returns {Sharp}
- * @throws {Error} Invalid options
- */
- function raw (options) {
- if (is.object(options)) {
- if (is.defined(options.depth)) {
- if (is.string(options.depth) && is.inArray(options.depth,
- ['char', 'uchar', 'short', 'ushort', 'int', 'uint', 'float', 'complex', 'double', 'dpcomplex']
- )) {
- this.options.rawDepth = options.depth;
- } else {
- throw is.invalidParameterError('depth', 'one of: char, uchar, short, ushort, int, uint, float, complex, double, dpcomplex', options.depth);
- }
- }
- }
- return this._updateFormatOut('raw');
- }
- /**
- * Use tile-based deep zoom (image pyramid) output.
- *
- * Set the format and options for tile images via the `toFormat`, `jpeg`, `png` or `webp` functions.
- * Use a `.zip` or `.szi` file extension with `toFile` to write to a compressed archive file format.
- *
- * The container will be set to `zip` when the output is a Buffer or Stream, otherwise it will default to `fs`.
- *
- * @example
- * sharp('input.tiff')
- * .png()
- * .tile({
- * size: 512
- * })
- * .toFile('output.dz', function(err, info) {
- * // output.dzi is the Deep Zoom XML definition
- * // output_files contains 512x512 tiles grouped by zoom level
- * });
- *
- * @example
- * const zipFileWithTiles = await sharp(input)
- * .tile({ basename: "tiles" })
- * .toBuffer();
- *
- * @example
- * const iiififier = sharp().tile({ layout: "iiif" });
- * readableStream
- * .pipe(iiififier)
- * .pipe(writeableStream);
- *
- * @param {Object} [options]
- * @param {number} [options.size=256] tile size in pixels, a value between 1 and 8192.
- * @param {number} [options.overlap=0] tile overlap in pixels, a value between 0 and 8192.
- * @param {number} [options.angle=0] tile angle of rotation, must be a multiple of 90.
- * @param {string|Object} [options.background={r: 255, g: 255, b: 255, alpha: 1}] - background colour, parsed by the [color](https://www.npmjs.org/package/color) module, defaults to white without transparency.
- * @param {string} [options.depth] how deep to make the pyramid, possible values are `onepixel`, `onetile` or `one`, default based on layout.
- * @param {number} [options.skipBlanks=-1] Threshold to skip tile generation. Range is 0-255 for 8-bit images, 0-65535 for 16-bit images. Default is 5 for `google` layout, -1 (no skip) otherwise.
- * @param {string} [options.container='fs'] tile container, with value `fs` (filesystem) or `zip` (compressed file).
- * @param {string} [options.layout='dz'] filesystem layout, possible values are `dz`, `iiif`, `iiif3`, `zoomify` or `google`.
- * @param {boolean} [options.centre=false] centre image in tile.
- * @param {boolean} [options.center=false] alternative spelling of centre.
- * @param {string} [options.id='https://example.com/iiif'] when `layout` is `iiif`/`iiif3`, sets the `@id`/`id` attribute of `info.json`
- * @param {string} [options.basename] the name of the directory within the zip file when container is `zip`.
- * @returns {Sharp}
- * @throws {Error} Invalid parameters
- */
- function tile (options) {
- if (is.object(options)) {
- // Size of square tiles, in pixels
- if (is.defined(options.size)) {
- if (is.integer(options.size) && is.inRange(options.size, 1, 8192)) {
- this.options.tileSize = options.size;
- } else {
- throw is.invalidParameterError('size', 'integer between 1 and 8192', options.size);
- }
- }
- // Overlap of tiles, in pixels
- if (is.defined(options.overlap)) {
- if (is.integer(options.overlap) && is.inRange(options.overlap, 0, 8192)) {
- if (options.overlap > this.options.tileSize) {
- throw is.invalidParameterError('overlap', `<= size (${this.options.tileSize})`, options.overlap);
- }
- this.options.tileOverlap = options.overlap;
- } else {
- throw is.invalidParameterError('overlap', 'integer between 0 and 8192', options.overlap);
- }
- }
- // Container
- if (is.defined(options.container)) {
- if (is.string(options.container) && is.inArray(options.container, ['fs', 'zip'])) {
- this.options.tileContainer = options.container;
- } else {
- throw is.invalidParameterError('container', 'one of: fs, zip', options.container);
- }
- }
- // Layout
- if (is.defined(options.layout)) {
- if (is.string(options.layout) && is.inArray(options.layout, ['dz', 'google', 'iiif', 'iiif3', 'zoomify'])) {
- this.options.tileLayout = options.layout;
- } else {
- throw is.invalidParameterError('layout', 'one of: dz, google, iiif, iiif3, zoomify', options.layout);
- }
- }
- // Angle of rotation,
- if (is.defined(options.angle)) {
- if (is.integer(options.angle) && !(options.angle % 90)) {
- this.options.tileAngle = options.angle;
- } else {
- throw is.invalidParameterError('angle', 'positive/negative multiple of 90', options.angle);
- }
- }
- // Background colour
- this._setBackgroundColourOption('tileBackground', options.background);
- // Depth of tiles
- if (is.defined(options.depth)) {
- if (is.string(options.depth) && is.inArray(options.depth, ['onepixel', 'onetile', 'one'])) {
- this.options.tileDepth = options.depth;
- } else {
- throw is.invalidParameterError('depth', 'one of: onepixel, onetile, one', options.depth);
- }
- }
- // Threshold to skip blank tiles
- if (is.defined(options.skipBlanks)) {
- if (is.integer(options.skipBlanks) && is.inRange(options.skipBlanks, -1, 65535)) {
- this.options.tileSkipBlanks = options.skipBlanks;
- } else {
- throw is.invalidParameterError('skipBlanks', 'integer between -1 and 255/65535', options.skipBlanks);
- }
- } else if (is.defined(options.layout) && options.layout === 'google') {
- this.options.tileSkipBlanks = 5;
- }
- // Center image in tile
- const centre = is.bool(options.center) ? options.center : options.centre;
- if (is.defined(centre)) {
- this._setBooleanOption('tileCentre', centre);
- }
- // @id attribute for IIIF layout
- if (is.defined(options.id)) {
- if (is.string(options.id)) {
- this.options.tileId = options.id;
- } else {
- throw is.invalidParameterError('id', 'string', options.id);
- }
- }
- // Basename for zip container
- if (is.defined(options.basename)) {
- if (is.string(options.basename)) {
- this.options.tileBasename = options.basename;
- } else {
- throw is.invalidParameterError('basename', 'string', options.basename);
- }
- }
- }
- // Format
- if (is.inArray(this.options.formatOut, ['jpeg', 'png', 'webp'])) {
- this.options.tileFormat = this.options.formatOut;
- } else if (this.options.formatOut !== 'input') {
- throw is.invalidParameterError('format', 'one of: jpeg, png, webp', this.options.formatOut);
- }
- return this._updateFormatOut('dz');
- }
- /**
- * Set a timeout for processing, in seconds.
- * Use a value of zero to continue processing indefinitely, the default behaviour.
- *
- * The clock starts when libvips opens an input image for processing.
- * Time spent waiting for a libuv thread to become available is not included.
- *
- * @example
- * // Ensure processing takes no longer than 3 seconds
- * try {
- * const data = await sharp(input)
- * .blur(1000)
- * .timeout({ seconds: 3 })
- * .toBuffer();
- * } catch (err) {
- * if (err.message.includes('timeout')) { ... }
- * }
- *
- * @since 0.29.2
- *
- * @param {Object} options
- * @param {number} options.seconds - Number of seconds after which processing will be stopped
- * @returns {Sharp}
- */
- function timeout (options) {
- if (!is.plainObject(options)) {
- throw is.invalidParameterError('options', 'object', options);
- }
- if (is.integer(options.seconds) && is.inRange(options.seconds, 0, 3600)) {
- this.options.timeoutSeconds = options.seconds;
- } else {
- throw is.invalidParameterError('seconds', 'integer between 0 and 3600', options.seconds);
- }
- return this;
- }
- /**
- * Update the output format unless options.force is false,
- * in which case revert to input format.
- * @private
- * @param {string} formatOut
- * @param {Object} [options]
- * @param {boolean} [options.force=true] - force output format, otherwise attempt to use input format
- * @returns {Sharp}
- */
- function _updateFormatOut (formatOut, options) {
- if (!(is.object(options) && options.force === false)) {
- this.options.formatOut = formatOut;
- }
- return this;
- }
- /**
- * Update a boolean attribute of the this.options Object.
- * @private
- * @param {string} key
- * @param {boolean} val
- * @throws {Error} Invalid key
- */
- function _setBooleanOption (key, val) {
- if (is.bool(val)) {
- this.options[key] = val;
- } else {
- throw is.invalidParameterError(key, 'boolean', val);
- }
- }
- /**
- * Called by a WriteableStream to notify us it is ready for data.
- * @private
- */
- function _read () {
- if (!this.options.streamOut) {
- this.options.streamOut = true;
- const stack = Error();
- this._pipeline(undefined, stack);
- }
- }
- /**
- * Invoke the C++ image processing pipeline
- * Supports callback, stream and promise variants
- * @private
- */
- function _pipeline (callback, stack) {
- if (typeof callback === 'function') {
- // output=file/buffer
- if (this._isStreamInput()) {
- // output=file/buffer, input=stream
- this.on('finish', () => {
- this._flattenBufferIn();
- sharp.pipeline(this.options, (err, data, info) => {
- if (err) {
- callback(is.nativeError(err, stack));
- } else {
- callback(null, data, info);
- }
- });
- });
- } else {
- // output=file/buffer, input=file/buffer
- sharp.pipeline(this.options, (err, data, info) => {
- if (err) {
- callback(is.nativeError(err, stack));
- } else {
- callback(null, data, info);
- }
- });
- }
- return this;
- } else if (this.options.streamOut) {
- // output=stream
- if (this._isStreamInput()) {
- // output=stream, input=stream
- this.once('finish', () => {
- this._flattenBufferIn();
- sharp.pipeline(this.options, (err, data, info) => {
- if (err) {
- this.emit('error', is.nativeError(err, stack));
- } else {
- this.emit('info', info);
- this.push(data);
- }
- this.push(null);
- this.on('end', () => this.emit('close'));
- });
- });
- if (this.streamInFinished) {
- this.emit('finish');
- }
- } else {
- // output=stream, input=file/buffer
- sharp.pipeline(this.options, (err, data, info) => {
- if (err) {
- this.emit('error', is.nativeError(err, stack));
- } else {
- this.emit('info', info);
- this.push(data);
- }
- this.push(null);
- this.on('end', () => this.emit('close'));
- });
- }
- return this;
- } else {
- // output=promise
- if (this._isStreamInput()) {
- // output=promise, input=stream
- return new Promise((resolve, reject) => {
- this.once('finish', () => {
- this._flattenBufferIn();
- sharp.pipeline(this.options, (err, data, info) => {
- if (err) {
- reject(is.nativeError(err, stack));
- } else {
- if (this.options.resolveWithObject) {
- resolve({ data, info });
- } else {
- resolve(data);
- }
- }
- });
- });
- });
- } else {
- // output=promise, input=file/buffer
- return new Promise((resolve, reject) => {
- sharp.pipeline(this.options, (err, data, info) => {
- if (err) {
- reject(is.nativeError(err, stack));
- } else {
- if (this.options.resolveWithObject) {
- resolve({ data, info });
- } else {
- resolve(data);
- }
- }
- });
- });
- }
- }
- }
- /**
- * Decorate the Sharp prototype with output-related functions.
- * @module Sharp
- * @private
- */
- module.exports = (Sharp) => {
- Object.assign(Sharp.prototype, {
- // Public
- toFile,
- toBuffer,
- keepExif,
- withExif,
- withExifMerge,
- keepIccProfile,
- withIccProfile,
- keepXmp,
- withXmp,
- keepMetadata,
- withMetadata,
- toFormat,
- jpeg,
- jp2,
- png,
- webp,
- tiff,
- avif,
- heif,
- jxl,
- gif,
- raw,
- tile,
- timeout,
- // Private
- _updateFormatOut,
- _setBooleanOption,
- _read,
- _pipeline
- });
- };
|