data:image/s3,"s3://crabby-images/25b84/25b8473bf7d97498cdd01f864de29f20e7575b59" alt="pixel"
data:image/s3,"s3://crabby-images/0f868/0f86812acbe1896364aaa9cb8e55205fb553493c" alt="Twitter Follow"
This project is part of the
@thi.ng/umbrella monorepo.
About
Typedarray integer & float pixel buffers w/ customizable formats, blitting, drawing, convolution.
data:image/s3,"s3://crabby-images/79df4/79df4bccab1ad31acb8503f2d8277910cf121ddc" alt="screenshot"
- Buffer creation from HTML image elements w/ opt resize & format
conversion (browser only)
- Buffer-to-buffer blitting w/ automatic format conversion
- Buffer-to-canvas blitting
- Buffer-to-buffer blending w/ Porter-Duff
operators
- Pre/post-multiply alpha
- Region / sub-image extraction
- Single-channel manipulation / extraction / replacement / conversion
- k-means based dominant color extraction (float buffers only)
- Accessors for normalized channel value
- Image sampling, resizing, pooling
- Filters: nearest neighbor, bilinear, bicubic
- Wrap behaviors: clamp, wrap, repeat
- Pooling: mean/min/max
- Invert image
- Convolution w/ arbitrary shaped/sized kernels, pooling, striding
- Convolution kernel & pooling kernels presets
- Higher order kernel generators (Gaussian, Lanczos)
- Image pyramid generation (w/ customizable kernels)
- Customizable normal map generation (i.e. X/Y gradients plus static Z component)
- XY full pixel & channel-only accessors
- 12 packed integer and 6 floating point preset formats (see table below)
- Declarative custom format & optimized code generation
- HTML canvas creation &
ImageData
utilities
Integer pixel formats
All integer formats use the canvas native ABGR 32bit format as common
intermediate for conversions. During conversion to ABGR, channels with sizes
smaller than 8 bits will be scaled appropriately to ensure an as full-range and
as linear as possible mapping. E.g. a 4 bit channel will be scaled by 255 / 15 =
17.
Format specs can freely control channel layout within current limits:
- Channel sizes: 1 - 32 bits.
- Storage: 8, 16 or 32 bits per pixel
New formats can be defined via defIntFormat()
.
Format ID | Bits per pixel | Description |
---|
ALPHA8 | 8 | 8 bit channel (alpha only) |
GRAY8 | 8 | 8 bit single channel (grayscale conv) |
GRAY_ALPHA8 | 16 | 8 bit single channel (grayscale conv), 8 bit alpha |
GRAY16 | 16 | 16 bit single channel (grayscale conv) |
GRAY_ALPHA16 | 32 | 16 bit single channel (grayscale conv), 16 bit alpha |
ARGB4444 | 16 | 4 channels @ 4 bits each |
ARGB1555 | 16 | 5 bits each for RGB, 1 bit alpha |
RGB565 | 16 | 5 bits red, 6 bits green, 5 bits blue |
RGB888 | 32 (24 effective) | 3 channels @ 8 bits each |
ARGB8888 | 32 | 4 channels @ 8 bits each |
BGR888 | 32 (24 effective) | 3 channels @ 8 bits each |
ABGR8888 | 32 | 4 channels @ 8 bits each |
ALPHA8
is mapped from/to ABGR alpha channelGRAY8/16
, GRAY_ALPHA8/16
compute grayscale/luminance when
converting from ABGR and in return produce grayscale ABGR- In all built-in formats supporting it, the alpha channel always
occupies the most-significant bits (up to format size)
Floating point pixel formats
Strided floating point format presets for use with floatBuffer()
. New
formats can be defined via defFloatFormat()
.
Format ID | Channel count | Description |
---|
FLOAT_GRAY | 1 | Single channel / grayscale |
FLOAT_GRAY_ALPHA | 2 | Grayscale and alpha channel |
FLOAT_NORMAL | 3 | Normal map (signed values) |
FLOAT_RGB | 3 | Red, Green, Blue |
FLOAT_RGBA | 4 | Red, Green, Blue, Alpha |
- All color channels are unclamped (but can be clamped via
buf.clamp()
). For
conversion to packed int formats assumed to contain normalized data (i.e.
[0..1] interval, with exception of FLOAT_NORMAL
which uses [-1..1] range) - Conversion between float formats is currently unsupported
Filtered image sampling and resizing
Available (and optimized) for both integer & floating point formats, image
samplers can be created with the following filters & wrap modes:
Filters
"nearest"
- nearest neighbor"linear"
- bilinear interpolation"cubic"
- bicubic interpolation
Wrap mode
"clamp"
- outside values return 0"wrap"
- infinite tiling"repeat"
- edge pixels are repeated
const src = intBuffer(4, 4, ABGR8888);
src.forEach((_,i) => 0xff000000 | Math.random() * 0xffffff);
const sampler = defSampler(src, "linear", "repeat");
sampler(-1.1, 0.5).toString(16)
const img = src.resize(1024, 256, "cubic");
Filter | |
---|
"nearest" | data:image/s3,"s3://crabby-images/32faf/32fafe5d937986a14c23c9666ace45772c34249c" alt="resized image w/ nearest neighbor sampling" |
"linear" | data:image/s3,"s3://crabby-images/33e57/33e57fe1773b75cc8f5f2f5d3c2b3c9f27152039" alt="resized image w/ bilinear sampling" |
"cubic" | data:image/s3,"s3://crabby-images/005cb/005cb20eb61ee3f2a8432b7c51d5928b6bbe6c30" alt="resized image w/ bicubic sampling" |
Strided convolution & pooling
Floating point buffers can be processed using arbitrary convolution kernels. The
following convolution kernel presets are provided for convenience:
Kernel | Size |
---|
BOX_BLUR3 | 3x3 |
BOX_BLUR5 | 5x5 |
GAUSSIAN_BLUR3 | 3x3 |
GAUSSIAN_BLUR5 | 5x5 |
GAUSSIAN(n) | 2n+1 x 2n+1 |
HIGHPASS3 | 3x3 |
LANCZOS(a,s) | as+1 x as+1 |
SHARPEN3 | 3x3 |
SOBEL_X | 3x3 |
SOBEL_Y | 3x3 |
UNSHARP_MASK5 | 5x5 |
Custom kernels can be defined (and code generated) using an array of
coefficients and a given kernel size. See above presets and
defKernel()
for
reference.
Furthermore, convolution supports striding (i.e. only processing & keeping every
nth pixel column/row, aka downscaling) and pixel pooling (e.g. for ML
applications). Available pooling kernel presets (kernel sizes must be configured
independently):
Kernel | Description |
---|
POOL_MEAN | Moving average |
POOL_MAX | Local maximum |
POOL_MIN | Local minimum |
POOL_NEAREST | Nearest neighbor |
POOL_THRESHOLD(bias) | Adaptive threshold |
Convolution can be applied to single, multiple or all channels of a
FloatBuffer
. See
convolveChannel()
and
convolveImage()
.
See
ConvolveOpts
for config options.
src = floatBuffer(read("test.ppm"), FLOAT_RGB);
convolveImage(src, { kernel: SOBEL_X, stride: 2, scale: 4 });
Normal map generation
Normal maps can be created via normalMap()
. This function uses an adjustable
convolution kernel size to control gradient smoothness & details. Result X/Y
gradients can also be scaled (uniform or anisotropic) and the Z component can be
customized to (default: 1.0). The resulting image is in FLOAT_NORMAL
format,
using signed channel values. This channel format is auto-translating these into
unsigned values when the image is converted into an integer format.
Step | Scale = 1 | Scale = 2 | Scale = 4 | Scale = 8 |
---|
0 | data:image/s3,"s3://crabby-images/df912/df91213a65d088f6776ec1d1e3505059c3dd3577" alt="" | data:image/s3,"s3://crabby-images/67707/67707aad3bcbdb32e0ce2af1f63e5d04821f5072" alt="" | data:image/s3,"s3://crabby-images/52599/525997372651d39e525f984039a270b448aa21ce" alt="" | data:image/s3,"s3://crabby-images/200cc/200cc081d87067a773caaf8c0b518eb8205eb3b5" alt="" |
1 | data:image/s3,"s3://crabby-images/5de2f/5de2fc49ed92df0405a618ec986ea55f0bc222a9" alt="" | data:image/s3,"s3://crabby-images/38ed4/38ed4b3a2ebf5e272bb23e3639a5ccb58219a339" alt="" | data:image/s3,"s3://crabby-images/cf4b9/cf4b932b8aea2a7da861b6eaee9ad0243f4272ba" alt="" | data:image/s3,"s3://crabby-images/fb6f5/fb6f53fd31531257a3b66e97c50aa2776988a79a" alt="" |
2 | data:image/s3,"s3://crabby-images/6fd52/6fd528946e52fabf8499b740fd6aa8519562af11" alt="" | data:image/s3,"s3://crabby-images/d651e/d651ec768633812d79b3fb54b22b4aef75eb3ad3" alt="" | data:image/s3,"s3://crabby-images/db368/db36826325b7da0b1b7f92dc4d771358bb4cd7fc" alt="" | data:image/s3,"s3://crabby-images/f63f9/f63f926edf69002b535bba391111f72f940d1072" alt="" |
3 | data:image/s3,"s3://crabby-images/ab70a/ab70ae0f973f219294f88773dbe15f957a8cb1ca" alt="" | data:image/s3,"s3://crabby-images/e9ec3/e9ec396eb608022b0576b52feb8b482411a9a11c" alt="" | data:image/s3,"s3://crabby-images/8890c/8890c6e928dbcc2ddcde1699f002d4b198dd15aa" alt="" | data:image/s3,"s3://crabby-images/185d9/185d99def41e50997463ecd1f2077086ecd3144d" alt="" |
import { floatBuffer, normalMap, FLOAT_GRAY, RGB888 } from "@thi.ng/pixel";
import { asPPM, read } from "@thi.ng/pixel-io-netpbm";
const src = floatBuffer(read(readFileSync("noise.pgm")), FLOAT_GRAY);
const nmap = normalMap(src, { step: 0, scale: 1 });
nmap.getAt(10, 10);
writeFileSync("noise-normal.ppm", asPPM(nmap.as(RGB888)));
The dominantColors()
function applies k-means
clustering to
extract the dominant colors from the given image. The clustering can be
configured. The function returns an array of { color, area }
objects (sorted
by descending area), where color
is a cluster's dominant color (in the format
of the source image) and area
the normalized cluster size (number of selected
pixels over total number of pixels in the image).
data:image/s3,"s3://crabby-images/e5a4f/e5a4ff35351b6f666f5252f713ac5846fe2603ba" alt="Example image & extracted dominant colors"
Picture credit: /u/kristophershinn
const img = floatBuffer(read(readFileSync(`test.ppm`)), FLOAT_RGB);
const clusters = dominantColors(img, 5);
console.log(clusters);
Status
STABLE - used in production
Search or submit any issues for this package
Support packages
Related packages
- @thi.ng/color - Array-based color types, CSS parsing, conversions, transformations, declarative theme generation, gradients, presets
- @thi.ng/porter-duff - Porter-Duff operators for packed ints & float-array alpha compositing
- @thi.ng/rasterize - 2D shape drawing & rasterization
- @thi.ng/shader-ast - DSL to define shader code in TypeScript and cross-compile to GLSL, JS and other targets
- @thi.ng/webgl - WebGL & GLSL abstraction layer
Installation
yarn add @thi.ng/pixel
ES module import:
<script type="module" src="https://cdn.skypack.dev/@thi.ng/pixel"></script>
Skypack documentation
For Node.js REPL:
# with flag only for < v16
node --experimental-repl-await
> const pixel = await import("@thi.ng/pixel");
Package sizes (gzipped, pre-treeshake): ESM: 9.15 KB
Dependencies
Usage examples
Several demos in this repo's
/examples
directory are using this package.
A selection:
Screenshot | Description | Live demo | Source |
---|
data:image/s3,"s3://crabby-images/9013f/9013fc6ce023b0426613ef73408064fcae5f0c66" alt="" | Interactive image processing (adaptive threshold) | Demo | Source |
data:image/s3,"s3://crabby-images/77bdd/77bdd98581e66c503f11f4c64e4637874ee4ec5f" alt="" | Color palette generation via dominant color extraction from uploaded images | Demo | Source |
data:image/s3,"s3://crabby-images/79df4/79df4bccab1ad31acb8503f2d8277910cf121ddc" alt="" | Pixel buffer manipulations | Demo | Source |
data:image/s3,"s3://crabby-images/a4e6f/a4e6f07e702e11a1a5d57e51bbff8bebc5cf4747" alt="" | Showcase of various dithering algorithms | Demo | Source |
data:image/s3,"s3://crabby-images/6ae20/6ae2085ef9594c554a82cfa71fb5c80a251b4905" alt="" | Image dithering and remapping using indexed palettes | Demo | Source |
data:image/s3,"s3://crabby-images/87587/87587e77ea6f21bb6cb1fcfdc16b1bd1a0b54329" alt="" | Interactive pixel sorting tool using thi.ng/color & thi.ng/pixel | Demo | Source |
data:image/s3,"s3://crabby-images/bac9d/bac9d0240f42c84fe8ec3b783f8ffd57182fe50a" alt="" | Port-Duff image compositing / alpha blending | Demo | Source |
data:image/s3,"s3://crabby-images/572a1/572a186a5c8ca33537cceb267d1ae6129e857cc5" alt="" | 2D scenegraph & image map based geometry manipulation | Demo | Source |
data:image/s3,"s3://crabby-images/3d420/3d42087459e8e93395acd51bb856f7197c3aeae8" alt="" | WebGL & Canvas2D textured tunnel shader | Demo | Source |
data:image/s3,"s3://crabby-images/91d4b/91d4b6c3d9312ed162cdecd230fbe8860a9d5f27" alt="" | Fork-join worker-based raymarch renderer (JS/CPU only) | Demo | Source |
data:image/s3,"s3://crabby-images/255ab/255ab6f4a2eab511faf685214f1274e58cc2928d" alt="" | Textmode image warping w/ 16bit color output | Demo | Source |
| Minimal multi-pass / GPGPU example | Demo | Source |
API
Generated API docs
import * as pix from "@thi.ng/pixel";
import { SRC_OVER_I } from "@thi.ng/porter-duff";
import IMG from "../assets/haystack.jpg";
import LOGO from "../assets/logo-64.png";
Promise
.all([IMG, LOGO].map(pix.imagePromise))
.then(([img, logo]) => {
const buf = intBufferFromImage(img, pix.RGB565, 256, 256);
pix.intBufferFromImage(logo, pix.GRAY_ALPHA88)
.premultiply()
.blend(SRC_OVER_I, buf, {
dx: 10,
dy: 10
});
const region = buf.getRegion(32, 96, 128, 64);
region.blit(buf, { dx: 96, dy: 32 });
const ctx = pix.canvas2d(buf.width, buf.height * 3, document.body);
buf.blitCanvas(ctx.canvas);
const id = 0;
const ch = buf.getChannel(id).invert();
for (let y = 0; y < ch.height; y += 2) {
for (let x = (y >> 1) & 1; x < ch.width; x += 2) {
ch.setAt(x, y, 0xff);
}
}
buf.setChannel(id, ch);
buf.blitCanvas(ctx.canvas, 0, buf.height);
buf.as(GRAY8).blitCanvas(ctx.canvas, 0, buf.height * 2);
});
TODO see examples & source comments for now
Authors
Karsten Schmidt
If this project contributes to an academic publication, please cite it as:
@misc{thing-pixel,
title = "@thi.ng/pixel",
author = "Karsten Schmidt",
note = "https://thi.ng/pixel",
year = 2019
}
License
© 2019 - 2021 Karsten Schmidt // Apache Software License 2.0