Skip to content

Commit 74e3f73

Browse files
antonmarsdenlovell
authored andcommitted
Expand linear operation to allow use of per-channel arrays #3303
1 parent b9261c2 commit 74e3f73

10 files changed

+104
-34
lines changed

docs/api-operation.md

+22-4
Original file line numberDiff line numberDiff line change
@@ -466,14 +466,32 @@ Returns **Sharp** 
466466

467467
## linear
468468

469-
Apply the linear formula a \* input + b to the image (levels adjustment)
469+
Apply the linear formula `a` \* input + `b` to the image to adjust image levels.
470+
471+
When a single number is provided, it will be used for all image channels.
472+
When an array of numbers is provided, the array length must match the number of channels.
470473

471474
### Parameters
472475

473-
* `a` **[number][1]** multiplier (optional, default `1.0`)
474-
* `b` **[number][1]** offset (optional, default `0.0`)
476+
* `a` **([number][1] | [Array][7]<[number][1]>)** multiplier (optional, default `[]`)
477+
* `b` **([number][1] | [Array][7]<[number][1]>)** offset (optional, default `[]`)
475478

476-
<!---->
479+
### Examples
480+
481+
```javascript
482+
await sharp(input)
483+
.linear(0.5, 2)
484+
.toBuffer();
485+
```
486+
487+
```javascript
488+
await sharp(rgbInput)
489+
.linear(
490+
[0.25, 0.5, 0.75],
491+
[150, 100, 50]
492+
)
493+
.toBuffer();
494+
```
477495

478496
* Throws **[Error][5]** Invalid parameters
479497

docs/search-index.json

+1-1
Large diffs are not rendered by default.

lib/constructor.js

+2-2
Original file line numberDiff line numberDiff line change
@@ -319,8 +319,8 @@ const Sharp = function (input, options) {
319319
tileId: 'https://example.com/iiif',
320320
tileBasename: '',
321321
timeoutSeconds: 0,
322-
linearA: 1,
323-
linearB: 0,
322+
linearA: [],
323+
linearB: [],
324324
// Function to notify of libvips warnings
325325
debuglog: warning => {
326326
this.emit('warning', warning);

lib/operation.js

+36-7
Original file line numberDiff line numberDiff line change
@@ -644,26 +644,55 @@ function boolean (operand, operator, options) {
644644
}
645645

646646
/**
647-
* Apply the linear formula a * input + b to the image (levels adjustment)
648-
* @param {number} [a=1.0] multiplier
649-
* @param {number} [b=0.0] offset
647+
* Apply the linear formula `a` * input + `b` to the image to adjust image levels.
648+
*
649+
* When a single number is provided, it will be used for all image channels.
650+
* When an array of numbers is provided, the array length must match the number of channels.
651+
*
652+
* @example
653+
* await sharp(input)
654+
* .linear(0.5, 2)
655+
* .toBuffer();
656+
*
657+
* @example
658+
* await sharp(rgbInput)
659+
* .linear(
660+
* [0.25, 0.5, 0.75],
661+
* [150, 100, 50]
662+
* )
663+
* .toBuffer();
664+
*
665+
* @param {(number|number[])} [a=[]] multiplier
666+
* @param {(number|number[])} [b=[]] offset
650667
* @returns {Sharp}
651668
* @throws {Error} Invalid parameters
652669
*/
653670
function linear (a, b) {
671+
if (!is.defined(a) && is.number(b)) {
672+
a = 1.0;
673+
} else if (is.number(a) && !is.defined(b)) {
674+
b = 0.0;
675+
}
654676
if (!is.defined(a)) {
655-
this.options.linearA = 1.0;
677+
this.options.linearA = [];
656678
} else if (is.number(a)) {
679+
this.options.linearA = [a];
680+
} else if (Array.isArray(a) && a.length && a.every(is.number)) {
657681
this.options.linearA = a;
658682
} else {
659-
throw is.invalidParameterError('a', 'numeric', a);
683+
throw is.invalidParameterError('a', 'number or array of numbers', a);
660684
}
661685
if (!is.defined(b)) {
662-
this.options.linearB = 0.0;
686+
this.options.linearB = [];
663687
} else if (is.number(b)) {
688+
this.options.linearB = [b];
689+
} else if (Array.isArray(b) && b.length && b.every(is.number)) {
664690
this.options.linearB = b;
665691
} else {
666-
throw is.invalidParameterError('b', 'numeric', b);
692+
throw is.invalidParameterError('b', 'number or array of numbers', b);
693+
}
694+
if (this.options.linearA.length !== this.options.linearB.length) {
695+
throw new Error('Expected a and b to be arrays of the same length');
667696
}
668697
return this;
669698
}

src/operations.cc

+7-3
Original file line numberDiff line numberDiff line change
@@ -306,10 +306,14 @@ namespace sharp {
306306
/*
307307
* Calculate (a * in + b)
308308
*/
309-
VImage Linear(VImage image, double const a, double const b) {
310-
if (HasAlpha(image)) {
309+
VImage Linear(VImage image, std::vector<double> const a, std::vector<double> const b) {
310+
size_t const bands = static_cast<size_t>(image.bands());
311+
if (a.size() > bands) {
312+
throw VError("Band expansion using linear is unsupported");
313+
}
314+
if (HasAlpha(image) && a.size() != bands && (a.size() == 1 || a.size() == bands - 1 || bands - 1 == 1)) {
311315
// Separate alpha channel
312-
VImage alpha = image[image.bands() - 1];
316+
VImage alpha = image[bands - 1];
313317
return RemoveAlpha(image).linear(a, b).bandjoin(alpha);
314318
} else {
315319
return image.linear(a, b);

src/operations.h

+1-1
Original file line numberDiff line numberDiff line change
@@ -90,7 +90,7 @@ namespace sharp {
9090
/*
9191
* Linear adjustment (a * in + b)
9292
*/
93-
VImage Linear(VImage image, double const a, double const b);
93+
VImage Linear(VImage image, std::vector<double> const a, std::vector<double> const b);
9494

9595
/*
9696
* Recomb with a Matrix of the given bands/channel size.

src/pipeline.cc

+3-3
Original file line numberDiff line numberDiff line change
@@ -688,7 +688,7 @@ class PipelineWorker : public Napi::AsyncWorker {
688688
}
689689

690690
// Linear adjustment (a * in + b)
691-
if (baton->linearA != 1.0 || baton->linearB != 0.0) {
691+
if (!baton->linearA.empty()) {
692692
image = sharp::Linear(image, baton->linearA, baton->linearB);
693693
}
694694

@@ -1454,8 +1454,8 @@ Napi::Value pipeline(const Napi::CallbackInfo& info) {
14541454
baton->trimThreshold = sharp::AttrAsDouble(options, "trimThreshold");
14551455
baton->gamma = sharp::AttrAsDouble(options, "gamma");
14561456
baton->gammaOut = sharp::AttrAsDouble(options, "gammaOut");
1457-
baton->linearA = sharp::AttrAsDouble(options, "linearA");
1458-
baton->linearB = sharp::AttrAsDouble(options, "linearB");
1457+
baton->linearA = sharp::AttrAsVectorOfDouble(options, "linearA");
1458+
baton->linearB = sharp::AttrAsVectorOfDouble(options, "linearB");
14591459
baton->greyscale = sharp::AttrAsBool(options, "greyscale");
14601460
baton->normalise = sharp::AttrAsBool(options, "normalise");
14611461
baton->claheWidth = sharp::AttrAsUint32(options, "claheWidth");

src/pipeline.h

+4-4
Original file line numberDiff line numberDiff line change
@@ -100,8 +100,8 @@ struct PipelineBaton {
100100
double trimThreshold;
101101
int trimOffsetLeft;
102102
int trimOffsetTop;
103-
double linearA;
104-
double linearB;
103+
std::vector<double> linearA;
104+
std::vector<double> linearB;
105105
double gamma;
106106
double gammaOut;
107107
bool greyscale;
@@ -251,8 +251,8 @@ struct PipelineBaton {
251251
trimThreshold(0.0),
252252
trimOffsetLeft(0),
253253
trimOffsetTop(0),
254-
linearA(1.0),
255-
linearB(0.0),
254+
linearA{},
255+
linearB{},
256256
gamma(0.0),
257257
greyscale(false),
258258
normalise(false),
161 KB
Loading

test/unit/linear.js

+28-9
Original file line numberDiff line numberDiff line change
@@ -65,15 +65,34 @@ describe('Linear adjustment', function () {
6565
});
6666
});
6767

68-
it('Invalid linear arguments', function () {
69-
assert.throws(function () {
70-
sharp(fixtures.inputPngOverlayLayer1)
71-
.linear('foo');
72-
});
68+
it('per channel level adjustment', function (done) {
69+
sharp(fixtures.inputWebP)
70+
.linear([0.25, 0.5, 0.75], [150, 100, 50]).toBuffer(function (err, data, info) {
71+
if (err) throw err;
72+
fixtures.assertSimilar(fixtures.expected('linear-per-channel.jpg'), data, done);
73+
});
74+
});
7375

74-
assert.throws(function () {
75-
sharp(fixtures.inputPngOverlayLayer1)
76-
.linear(undefined, { bar: 'baz' });
77-
});
76+
it('Invalid linear arguments', function () {
77+
assert.throws(
78+
() => sharp().linear('foo'),
79+
/Expected number or array of numbers for a but received foo of type string/
80+
);
81+
assert.throws(
82+
() => sharp().linear(undefined, { bar: 'baz' }),
83+
/Expected number or array of numbers for b but received \[object Object\] of type object/
84+
);
85+
assert.throws(
86+
() => sharp().linear([], [1]),
87+
/Expected number or array of numbers for a but received {2}of type object/
88+
);
89+
assert.throws(
90+
() => sharp().linear([1, 2], [1]),
91+
/Expected a and b to be arrays of the same length/
92+
);
93+
assert.throws(
94+
() => sharp().linear([1]),
95+
/Expected a and b to be arrays of the same length/
96+
);
7897
});
7998
});

0 commit comments

Comments
 (0)