The @image module contains a single class @image which represents a 2D image, and related functions.

Methods

An image is represented internally as an array of pixels, which can be manipulated. Each pixel is 32 bits: 8 bits for each of the color components (red, green, blue), and 8 bits for alpha (transparency). This allows the color of each pixel to be represented as a 32-bit integer.

The order of the pixels is left-to-right, top-to-bottom (just like text). The order of the components can be changed with :set_color_format().

Most methods return the image itself so that they they can be chained to save typing.

Most methods can take color as either number or string. When passed as a string, the color is converted to a number using :color().

The following image is used in the examples below:

image(web.dl('https://axm.dev/pic.png'))
Axiom logo :new() #
:new(path)
:new(width,height)

Create a new image.

Parameters

Return

Notes

If no parameters are specified, an empty 0 by 0 image is created.

If path is specified, the image is loaded from a file.

If width and height are specified, a blank (all black) width by height image is created.

:load(path) #

Load data from file into image.

Parameters

Return

:save(path) #

Save image to file.

Parameters

Return

:clear() #

Clear all data and make the image 0x0.

Return

Notes

:allocate(width,height) #

Allocate data for this image.

Parameters

Return

Notes

After calling, this image will be a width by height and blank (all black).

:width() #

Get the width in pixels.

Return

:height() #

Get the height in pixels.

Return

:size() #

Get the size in pixels, which is equivalent to :width() multiplied by :height().

Return

:get(x,y) #

Get the color of a pixel.

Parameters

Return

Notes

The method :rgba() can be used to get individual components from the color.

Remember that 0,0 is at the top left.

If the coordinates are out of bounds, the behavior is to return the pixel that would be mirrored from the out-of-bounds edge. This is for compatibility with image algorithms that use kernels.

:set(x,y,color) #

Set the color of a pixel

Parameters

Return

:rgba(color) #

Get components of a color.

Parameters

Return

Notes

All components are unsigned bytes, so they will be in the range [0,255].

:hsl(color) #

Get the components of a color in HSL (Hue, Saturation, Lightness) color space.

Parameters

Return

Notes

Hue will be in the range [0,360):

Saturation will be in the range [0,1], where 0 = grayscale and 1 = pure color.

Lightness will be in the range [0,1], where 0 = black, 1 = white.

:luma(color) #

Convert a color to grayscale.

Parameters

Return

:color(str) #

Create a color from string.

Parameters

Return

Notes

Colors can be specified in one of the following formats:

For the tuples, RGB must be [0,255], alpha must be [0,1], hue must be [0,360), saturation [0,1], and lightness [0,1]. If alpha is not specified, it is assumed to be 0xff (255).

If str is actually a number, no processing is done.

:get_color_format() #

Get the pixel color format.

Return

:set_color_format(format,convert) #

Set the pixel color format

Parameters

Return

:copy(s) #

Set this image to a copy of another image.

Parameters

Return

:swap(s) #

Swap data with another image.

Parameters

Return

:crop(x,y,w,h,s) #

Set this image to the cropped area of another image.

Parameters

Return

:fill(color) #

Fill the image with a single color.

Parameters

Return

:interpolate(s,p) #

Interpolate this image with another image.

Parameters

Return

Notes

Each pixel in both images are blended using the following formula: a*p + b*(1-p), where a is every pixel in this image and b is every pixel in the other image.

If p is greater than 1, then a is "extrapolated" from b.

:blend(s,p,q) #

Blend this image with another image.

Parameters

Return

Notes

Each pixel in both images are blended using the following formula: a*p + b*q, where a is this image and b is the other image.

:blur(radius) #

Blur the image.

Parameters

Return

Example

image(web.dl('https://axm.dev/pic.png')).blur().save('blur.jpg')
Axiom logo blurred :desaturate() #

Desaturate the image (convert to grayscale).

Return

Example

image(web.dl('https://axm.dev/pic.png')).desaturate().save('desaturate.jpg')
Axiom logo desaturated :lazy_sharpen(amount) #

Sharpen the image by extrapolating from the average color.

Parameters

Return

Notes

Faster than :sharpen().

:sharpen(amount,blur_radius) #

Sharpen the image by extrapolating from a blurred image.

Parameters

Return

Notes

Slower than :lazy_sharpen() but should give a nicer result.

:saturate(amount) #

Saturate the image by extrapolating from a desaturated image.

Parameters

Return

:negate() #

Negate the image (invert the colors).

Return

Example

image(web.dl('https://axm.dev/pic.png')).negate().save('negate.jpg')
Axiom logo negated :brighten(amount) #

Brighten the image by extrapolating from black.

Parameters

Return

:lazy_resize(scale) #
:lazy_resize(w,h)

Resize this image using nearest-neighbor for in-between pixels.

Parameters

Return

:resize(scale) #
:resize(w,h)

Resize this image with upsampling/downsampling for in-between pixels.

Parameters

Return

Notes

Slower than :lazy_resize() but gives a better result.

:threshold(luma) #

Perform a threshold operation on the image, which changes all pixels white if > threshold, and black if <= threshold.

Parameters

Return

:average() #

Get the average color of the image.

Return

:text(x,y,str,color,size,face) #

Draw text on this image.

Parameters

Return

Notes

The text is bound by image width and will wrap.

The "Tahoma" font face is embedded within Axiom. If any other face is specified, the current directory and the operating system font directories will be searched for a TrueType (.ttf) file with the same name.

:text_width(str,size,face) #

Get the width of the text given the font face and size.

Parameters

Return

Notes

Doesn't actually draw anything on the image; just used for layout calculations.

:line_height(size,face) #

Get the line height given the font size and face.

Parameters

Return

Notes

Doesn't actually draw anything on the image; just used for layout calculations.

:blit(s,x,y,use_alpha) #

Blit (copy image from) another image onto this image.

Parameters

Return

:rect(x,y,w,h,color,thickness,filled) #

Draw a rectangle.

Parameters

Return

:line(x1,y1,x2,y2,color,thickness) #

Draw a line.

Parameters

Return

:circle(x,y,radius,color,thickness,filled) #

Draw a circle.

Parameters

Return

:rotate_left() #

Rotate the image left (counter-clockwise).

Return

:rotate_right() #

Rotate the image right (clockwise).

Return

:flip(x_axis,y_axis) #

Flip the image across the x-axis, y-axis, or both.

Parameters

Return

:pivot() #

Flip the image across the diagonal that runs from the top-left to the bottom-right.

Return

:cinema() #

Enhance the image in a cinema-like manner; increase sharpness and contrast while reducing saturation.

Return

:find_exact(other) #

Find another image within this image.

Parameters

Return

Notes

All pixels must match exactly. This is much slower than :find_fuzzy().

:find_fuzzy(other,threshold) #

Find another image within this image using fuzzy matching.

Parameters

Return

Notes

The distance is an number in the range [0,1] which indicates how closely this image and other match. It is calculated at every possible location for other by sampling 64 uniformly-spaced pixels or every pixel in other (whichever is less) and then measuring the difference in :luma(). This is much faster than :find_exact() and can deal with cases where other does not match exactly but should still be found.

The values returned in the object are to aid with debugging. If other was not found, but was expected to be found, try increasing threshold above min_distance.

:ascii(x,y,str,color,scale) #

Draw text on this image using the ASCII 5x7 font which is typically seen on a Code Page 437 console. (The font face is pixelated and fixed width).

Parameters

Return

Notes

Each glyph is actually 5 by 8 pixels (the 8th pixel is the descender). This function leaves a 1 pixel margin around the top and left of the glyph, so the total size is actually 6 by 9 pixels per letter. (Assuming scale is 1).

The text is bound by image width, and will wrap.

:ascii_table(scale) #

Draw all ASCII 5x7 glyphs on this image.

Parameters

Return

Notes

This is so you can view the entire font (since unlike TrueType fonts which typically have a viewer built into the operating system, there is no other way to view the embedded 5x7 font). You can also view the image here.

The replaces whatever was in the image with an entirely new image.

:dither(bits) #

Reduce the bit depth of each color component, and compensate for the loss in color accuracy by dithering.

Parameters

Return

Notes

This truncates each color components of each pixel so that instead of being 8 bits, the most significant [8 - bits] bits are zeroed.

This is used within :hide() for preprocessing.

:hide(s,bits) #

Hide an image within the most significant bits of this image, also known as steganography

Parameters

Return

Notes

The image within s is dithered and has its bit depth reduced to bits. It is then hidden in the most significant bits bits of the image within this image, which makes it hard to detect.

Example

#create message
i = image(480,480).fill('#ccc')
y = i.text(30,20,'Hello','red',40) + 20
y += i.ascii(30,y,'World','green',4) + 20
i.circle(240,y+100,100,'purple',1,true)
i.rect(240,y+100,200,200,'orange',1,true)
i.save('message.png')
generated image
#hide message. must save as png or bmp (not jpg).
image(web.dl('https://axm.dev/pic.png')).resize(480,480).hide(image('message.png')).save('hide.png')
Axiom logo with hidden message

If you look closely, you can see the message image faintly within the output image. (A more detailed input image would hide the message better.)

:unhide(bits) #

Reveal the image hidden within the most significant bits of this image.

Parameters

Return

Notes

Replaces this image with the hidden image, if any. This is the opposite of :hide().

Example

image('hide.png').unhide().save('unhide.jpg')
secret message

Functions

:exif(path) #

Get EXIF metadata from an image file, typically a JPEG.

Parameters

Return

Notes

A lot of these values assume the use of a professional camera with a detachable lens (DSLR), and don't really apply to modern cell phone cameras (although cell phone cameras are getting fancier and some have multiple lenses now).

:plot(data,cols,type) #

Generate a plot from tabular data.

Parameters

Return

Example

For these examples we will use IMF GDP data from 2024: imf_gdps_2024.csv

data = io.csv.load(web.dl('https://axm.dev/imf_gdps_2024.csv'))
head = data[0]
body = data.slice(1)
body = body.sort_by_index(head.find(2023),true).slice(0,10)
data = [head]+body
image.plot(data,['Country',2023],'bar').save('bar.png')
#GDP by Country (Billions USD)
GDP by Country
data = data.transpose()
data[0][0] = 'Year'
image.plot(data,['Year','United States','China','Germany','Japan','India']).save('line.png')
#GPD by Year (Billions USD)
GDP by Year
image.plot(data,['United States','Canada'],'scatter').save('scatter.png')
print image.plot(data,['United States','Canada'],'linreg').save('linreg.png')
##
{
  "class" : 
  {
    "scope" : "@image"
  },
  "alpha" : 83.0919,
  "beta" : 0.0801964,
  "rmse" : 161.466
}
##
# GDP of United States vs Canada
GDB of United States vs Canada GDB of United States vs Canada
data = io.csv.load(web.dl('https://axm.dev/imf_gdps_2024.csv'))
head = data[0]
body = data.slice(1)
body = body.sort_by_index(head.find(2023),true).slice(0,100)
data = [head]+body
image.plot(data,['Country',2023],'pie').save('pie.png')
#Top 100 Countries by GDP (Billions USD)
Top 100 Countries by GDP :qr(s) #

Generate QR code for string.

Parameters

Return

Notes

Each element in the generated image is 5x5 pixels. The image also includes a 2x (10 pixel) "quiet zone" around the elements.

Example

image.qr('https://axm.dev/').save('qr.png')
QR code for Axiom website