Introduce a helper function for clamping a value to a range

Currently we have a number of spots in the code-base where we need to clamp a value to a [min, max] range. This is either implemented using `Math.min`/`Math.max` or with a local helper function, which leads to some unnecessary duplication.

Hence this patch adds and re-uses a single helper function for this, which we'll hopefully be able to remove in the future once https://github.com/tc39/proposal-math-clamp/ becomes generally available.
This commit is contained in:
Jonas Jenwald 2025-03-06 12:57:14 +01:00
parent dea35aed4a
commit 07bbbf75a5
12 changed files with 57 additions and 68 deletions

View File

@ -18,6 +18,7 @@ import {
FeatureTest, FeatureTest,
FormatError, FormatError,
info, info,
MathClamp,
shadow, shadow,
unreachable, unreachable,
warn, warn,
@ -946,7 +947,7 @@ class CalRGBCS extends ColorSpace {
#sRGBTransferFunction(color) { #sRGBTransferFunction(color) {
// See http://en.wikipedia.org/wiki/SRGB. // See http://en.wikipedia.org/wiki/SRGB.
if (color <= 0.0031308) { if (color <= 0.0031308) {
return this.#adjustToRange(0, 1, 12.92 * color); return MathClamp(12.92 * color, 0, 1);
} }
// Optimization: // Optimization:
// If color is close enough to 1, skip calling the following transform // If color is close enough to 1, skip calling the following transform
@ -957,11 +958,7 @@ class CalRGBCS extends ColorSpace {
if (color >= 0.99554525) { if (color >= 0.99554525) {
return 1; return 1;
} }
return this.#adjustToRange(0, 1, (1 + 0.055) * color ** (1 / 2.4) - 0.055); return MathClamp((1 + 0.055) * color ** (1 / 2.4) - 0.055, 0, 1);
}
#adjustToRange(min, max, value) {
return Math.max(min, Math.min(max, value));
} }
#decodeL(L) { #decodeL(L) {
@ -1057,9 +1054,9 @@ class CalRGBCS extends ColorSpace {
#toRgb(src, srcOffset, dest, destOffset, scale) { #toRgb(src, srcOffset, dest, destOffset, scale) {
// A, B and C represent a red, green and blue components of a calibrated // A, B and C represent a red, green and blue components of a calibrated
// rgb space. // rgb space.
const A = this.#adjustToRange(0, 1, src[srcOffset] * scale); const A = MathClamp(src[srcOffset] * scale, 0, 1);
const B = this.#adjustToRange(0, 1, src[srcOffset + 1] * scale); const B = MathClamp(src[srcOffset + 1] * scale, 0, 1);
const C = this.#adjustToRange(0, 1, src[srcOffset + 2] * scale); const C = MathClamp(src[srcOffset + 2] * scale, 0, 1);
// A <---> AGR in the spec // A <---> AGR in the spec
// B <---> BGG in the spec // B <---> BGG in the spec

View File

@ -26,7 +26,7 @@ import {
PatternCS, PatternCS,
} from "./colorspace.js"; } from "./colorspace.js";
import { Dict, Name, Ref } from "./primitives.js"; import { Dict, Name, Ref } from "./primitives.js";
import { shadow, unreachable, warn } from "../shared/util.js"; import { MathClamp, shadow, unreachable, warn } from "../shared/util.js";
import { IccColorSpace } from "./icc_colorspace.js"; import { IccColorSpace } from "./icc_colorspace.js";
import { MissingDataException } from "./core_utils.js"; import { MissingDataException } from "./core_utils.js";
@ -245,7 +245,7 @@ class ColorSpaceUtils {
case "I": case "I":
case "Indexed": case "Indexed":
baseCS = this.#subParse(cs[1], options); baseCS = this.#subParse(cs[1], options);
const hiVal = Math.max(0, Math.min(xref.fetchIfRef(cs[2]), 255)); const hiVal = MathClamp(xref.fetchIfRef(cs[2]), 0, 255);
const lookup = xref.fetchIfRef(cs[3]); const lookup = xref.fetchIfRef(cs[3]);
return new IndexedCS(baseCS, hiVal, lookup); return new IndexedCS(baseCS, hiVal, lookup);
case "Separation": case "Separation":

View File

@ -18,6 +18,7 @@ import {
FeatureTest, FeatureTest,
FormatError, FormatError,
info, info,
MathClamp,
shadow, shadow,
unreachable, unreachable,
} from "../shared/util.js"; } from "../shared/util.js";
@ -263,10 +264,7 @@ class PDFFunction {
// x_i' = min(max(x_i, Domain_2i), Domain_2i+1) // x_i' = min(max(x_i, Domain_2i), Domain_2i+1)
const domain_2i = domain[i][0]; const domain_2i = domain[i][0];
const domain_2i_1 = domain[i][1]; const domain_2i_1 = domain[i][1];
const xi = Math.min( const xi = MathClamp(src[srcOffset + i], domain_2i, domain_2i_1);
Math.max(src[srcOffset + i], domain_2i),
domain_2i_1
);
// e_i = Interpolate(x_i', Domain_2i, Domain_2i+1, // e_i = Interpolate(x_i', Domain_2i, Domain_2i+1,
// Encode_2i, Encode_2i+1) // Encode_2i, Encode_2i+1)
@ -280,7 +278,7 @@ class PDFFunction {
// e_i' = min(max(e_i, 0), Size_i - 1) // e_i' = min(max(e_i, 0), Size_i - 1)
const size_i = size[i]; const size_i = size[i];
e = Math.min(Math.max(e, 0), size_i - 1); e = MathClamp(e, 0, size_i - 1);
// Adjusting the cube: N and vertex sample index // Adjusting the cube: N and vertex sample index
const e0 = e < size_i - 1 ? Math.floor(e) : e - 1; // e1 = e0 + 1; const e0 = e < size_i - 1 ? Math.floor(e) : e - 1; // e1 = e0 + 1;
@ -314,7 +312,7 @@ class PDFFunction {
rj = interpolate(rj, 0, 1, decode[j][0], decode[j][1]); rj = interpolate(rj, 0, 1, decode[j][0], decode[j][1]);
// y_j = min(max(r_j, range_2j), range_2j+1) // y_j = min(max(r_j, range_2j), range_2j+1)
dest[destOffset + j] = Math.min(Math.max(rj, range[j][0]), range[j][1]); dest[destOffset + j] = MathClamp(rj, range[j][0], range[j][1]);
} }
}; };
} }
@ -361,17 +359,8 @@ class PDFFunction {
const tmpBuf = new Float32Array(1); const tmpBuf = new Float32Array(1);
return function constructStichedFn(src, srcOffset, dest, destOffset) { return function constructStichedFn(src, srcOffset, dest, destOffset) {
const clip = function constructStichedFromIRClip(v, min, max) { // Clamp to domain.
if (v > max) { const v = MathClamp(src[srcOffset], domain[0], domain[1]);
v = max;
} else if (v < min) {
v = min;
}
return v;
};
// clip to domain
const v = clip(src[srcOffset], domain[0], domain[1]);
// calculate which bound the value is in // calculate which bound the value is in
const length = bounds.length; const length = bounds.length;
let i; let i;

View File

@ -18,6 +18,7 @@ import {
FeatureTest, FeatureTest,
FormatError, FormatError,
ImageKind, ImageKind,
MathClamp,
warn, warn,
} from "../shared/util.js"; } from "../shared/util.js";
import { import {
@ -33,21 +34,6 @@ import { JpegStream } from "./jpeg_stream.js";
import { JpxImage } from "./jpx.js"; import { JpxImage } from "./jpx.js";
import { Name } from "./primitives.js"; import { Name } from "./primitives.js";
/**
* Decode and clamp a value. The formula is different from the spec because we
* don't decode to float range [0,1], we decode it in the [0,max] range.
*/
function decodeAndClamp(value, addend, coefficient, max) {
value = addend + value * coefficient;
// Clamp the value to the range
if (value < 0) {
value = 0;
} else if (value > max) {
value = max;
}
return value;
}
/** /**
* Resizes an image mask with 1 component. * Resizes an image mask with 1 component.
* @param {TypedArray} src - The source buffer. * @param {TypedArray} src - The source buffer.
@ -487,10 +473,11 @@ class PDFImage {
let index = 0; let index = 0;
for (i = 0, ii = this.width * this.height; i < ii; i++) { for (i = 0, ii = this.width * this.height; i < ii; i++) {
for (let j = 0; j < numComps; j++) { for (let j = 0; j < numComps; j++) {
buffer[index] = decodeAndClamp( // Decode and clamp. The formula is different from the spec because we
buffer[index], // don't decode to float range [0,1], we decode it in the [0,max] range.
decodeAddends[j], buffer[index] = MathClamp(
decodeCoefficients[j], decodeAddends[j] + buffer[index] * decodeCoefficients[j],
0,
max max
); );
index++; index++;

View File

@ -13,7 +13,7 @@
* limitations under the License. * limitations under the License.
*/ */
import { shadow } from "../../shared/util.js"; import { MathClamp, shadow } from "../../shared/util.js";
const dimConverters = { const dimConverters = {
pt: x => x, pt: x => x,
@ -143,7 +143,7 @@ function getColor(data, def = [0, 0, 0]) {
const color = data const color = data
.trim() .trim()
.split(/\s*,\s*/) .split(/\s*,\s*/)
.map(c => Math.min(Math.max(0, parseInt(c.trim(), 10)), 255)) .map(c => MathClamp(parseInt(c.trim(), 10), 0, 255))
.map(c => (isNaN(c) ? 0 : c)); .map(c => (isNaN(c) ? 0 : c));
if (color.length < 3) { if (color.length < 3) {

View File

@ -13,8 +13,8 @@
* limitations under the License. * limitations under the License.
*/ */
import { MathClamp, Util } from "../../../shared/util.js";
import { Outline } from "./outline.js"; import { Outline } from "./outline.js";
import { Util } from "../../../shared/util.js";
class InkDrawOutliner { class InkDrawOutliner {
// The last 3 points of the line. // The last 3 points of the line.
@ -616,10 +616,10 @@ class InkDrawOutline extends Outline {
} }
const [marginX, marginY] = this.#getMarginComponents(); const [marginX, marginY] = this.#getMarginComponents();
bbox[0] = Math.min(1, Math.max(0, bbox[0] - marginX)); bbox[0] = MathClamp(bbox[0] - marginX, 0, 1);
bbox[1] = Math.min(1, Math.max(0, bbox[1] - marginY)); bbox[1] = MathClamp(bbox[1] - marginY, 0, 1);
bbox[2] = Math.min(1, Math.max(0, bbox[2] + marginX)); bbox[2] = MathClamp(bbox[2] + marginX, 0, 1);
bbox[3] = Math.min(1, Math.max(0, bbox[3] + marginY)); bbox[3] = MathClamp(bbox[3] + marginY, 0, 1);
bbox[2] -= bbox[0]; bbox[2] -= bbox[0];
bbox[3] -= bbox[1]; bbox[3] -= bbox[1];

View File

@ -22,7 +22,12 @@ import {
ColorManager, ColorManager,
KeyboardManager, KeyboardManager,
} from "./tools.js"; } from "./tools.js";
import { FeatureTest, shadow, unreachable } from "../../shared/util.js"; import {
FeatureTest,
MathClamp,
shadow,
unreachable,
} from "../../shared/util.js";
import { noContextMenu, stopEvent } from "../display_utils.js"; import { noContextMenu, stopEvent } from "../display_utils.js";
import { AltText } from "./alt_text.js"; import { AltText } from "./alt_text.js";
import { EditorToolbar } from "./toolbar.js"; import { EditorToolbar } from "./toolbar.js";
@ -613,20 +618,20 @@ class AnnotationEditor {
if (this._mustFixPosition) { if (this._mustFixPosition) {
switch (rotation) { switch (rotation) {
case 0: case 0:
x = Math.max(0, Math.min(pageWidth - width, x)); x = MathClamp(x, 0, pageWidth - width);
y = Math.max(0, Math.min(pageHeight - height, y)); y = MathClamp(y, 0, pageHeight - height);
break; break;
case 90: case 90:
x = Math.max(0, Math.min(pageWidth - height, x)); x = MathClamp(x, 0, pageWidth - height);
y = Math.min(pageHeight, Math.max(width, y)); y = MathClamp(y, width, pageHeight);
break; break;
case 180: case 180:
x = Math.min(pageWidth, Math.max(width, x)); x = MathClamp(x, width, pageWidth);
y = Math.min(pageHeight, Math.max(height, y)); y = MathClamp(y, height, pageHeight);
break; break;
case 270: case 270:
x = Math.min(pageWidth, Math.max(height, x)); x = MathClamp(x, height, pageWidth);
y = Math.max(0, Math.min(pageHeight - width, y)); y = MathClamp(y, 0, pageHeight - width);
break; break;
} }
} }

View File

@ -33,6 +33,7 @@ import {
getUuid, getUuid,
ImageKind, ImageKind,
InvalidPDFException, InvalidPDFException,
MathClamp,
normalizeUnicode, normalizeUnicode,
OPS, OPS,
PasswordResponses, PasswordResponses,
@ -119,6 +120,7 @@ export {
isDataScheme, isDataScheme,
isPdfFile, isPdfFile,
isValidExplicitDest, isValidExplicitDest,
MathClamp,
noContextMenu, noContextMenu,
normalizeUnicode, normalizeUnicode,
OPS, OPS,

View File

@ -1128,6 +1128,12 @@ function _isValidExplicitDest(validRef, validName, dest) {
return true; return true;
} }
// TOOD: Replace all occurrences of this function with `Math.clamp` once
// https://github.com/tc39/proposal-math-clamp/ is generally available.
function MathClamp(v, min, max) {
return Math.min(Math.max(v, min), max);
}
// TODO: Remove this once `Uint8Array.prototype.toHex` is generally available. // TODO: Remove this once `Uint8Array.prototype.toHex` is generally available.
function toHexUtil(arr) { function toHexUtil(arr) {
if (Uint8Array.prototype.toHex) { if (Uint8Array.prototype.toHex) {
@ -1202,6 +1208,7 @@ export {
isNodeJS, isNodeJS,
LINE_DESCENT_FACTOR, LINE_DESCENT_FACTOR,
LINE_FACTOR, LINE_FACTOR,
MathClamp,
normalizeUnicode, normalizeUnicode,
objectFromMap, objectFromMap,
objectSize, objectSize,

View File

@ -24,6 +24,7 @@ import {
getUuid, getUuid,
ImageKind, ImageKind,
InvalidPDFException, InvalidPDFException,
MathClamp,
normalizeUnicode, normalizeUnicode,
OPS, OPS,
PasswordResponses, PasswordResponses,
@ -96,6 +97,7 @@ const expectedAPI = Object.freeze({
isDataScheme, isDataScheme,
isPdfFile, isPdfFile,
isValidExplicitDest, isValidExplicitDest,
MathClamp,
noContextMenu, noContextMenu,
normalizeUnicode, normalizeUnicode,
OPS, OPS,

View File

@ -40,6 +40,7 @@ const {
isDataScheme, isDataScheme,
isPdfFile, isPdfFile,
isValidExplicitDest, isValidExplicitDest,
MathClamp,
noContextMenu, noContextMenu,
normalizeUnicode, normalizeUnicode,
OPS, OPS,
@ -92,6 +93,7 @@ export {
isDataScheme, isDataScheme,
isPdfFile, isPdfFile,
isValidExplicitDest, isValidExplicitDest,
MathClamp,
noContextMenu, noContextMenu,
normalizeUnicode, normalizeUnicode,
OPS, OPS,

View File

@ -13,6 +13,8 @@
* limitations under the License. * limitations under the License.
*/ */
import { MathClamp } from "pdfjs-lib";
const DEFAULT_SCALE_VALUE = "auto"; const DEFAULT_SCALE_VALUE = "auto";
const DEFAULT_SCALE = 1.0; const DEFAULT_SCALE = 1.0;
const DEFAULT_SCALE_DELTA = 1.1; const DEFAULT_SCALE_DELTA = 1.1;
@ -676,10 +678,6 @@ const docStyle =
? null ? null
: document.documentElement.style; : document.documentElement.style;
function clamp(v, min, max) {
return Math.min(Math.max(v, min), max);
}
class ProgressBar { class ProgressBar {
#classList = null; #classList = null;
@ -701,7 +699,7 @@ class ProgressBar {
} }
set percent(val) { set percent(val) {
this.#percent = clamp(val, 0, 100); this.#percent = MathClamp(val, 0, 100);
if (isNaN(val)) { if (isNaN(val)) {
this.#classList.add("indeterminate"); this.#classList.add("indeterminate");