Seam carving is a method of content-aware image resizing first presented by Shai Avidan and Ariel Shamir in a landmark paper in 2007. A seam of an image I is defined as a monotonic and connected path of pixels walked from the leftmost column of I to the rightmost column (or from top to bottom). By imposing an energy function E(i,j) upon I, one effectively reformulates the image resizing problem as a brute-force search for minimal-energy seams (which is made practical through dynamic programming). The following repository details the implementation of Seam, an interactive editor and visualizer for said operation. In particular, Seam implements the improved, forward-energy formulation of seam carving published by Michael Rubinstein, Shamir, and Avidan in 2008.
Seam is a minimal viable product hastily written in C, self-contained with the exception of: i) an interface built on SDL2; and ii) image decoding/encoding performed through Lode Vandevenne's lightweight LodePNG.
Per iteration, Seam computes a discrete gradient through convolution with a Sobel kernel, which is then used to compute an accumulative cost matrix M (as in dynamic programming). The optimal solution is then back-traced through M to yield a seam of pixels which may be deleted from I to produce an image one column (or row) smaller than I.
The result is a resizing operation that may be less destructive of source features than linear or cubic interpolation:
(Left: Seam carving animation generated with the -rs flag. Right: Corresponding heatmap animation generated with the -re flag.)
(Left: Result of linear interpolation. Right: Result of seam carving (note the aberration around Mt. Fuji). Source: Hokusai, The Great Wave off Kanagawa)
(Left: Result of linear interpolation. Right: Result of seam carving (note that the head of Jesus is protected by a high energy boundary). Source: Leonardo da Vinci, The Last Supper)
Download Seam by running
git clone https://github.com/brekekekex/seam.git
Note that building Seam requires SDL2, which may be obtained through
sudo apt-get install libsdl2-dev
In addition, one should git clone LodePNG into the /src directory and rename lodepng.cpp to lodepng.c.
One may now build the binary with
make
For usage instructions, run
./seam --help
which should produce
usage: ./seam FILENAME [-rr] [-re] [-rs] [-tw TARGET_WIDTH] [-th TARGET_HEIGHT]
-rr: dump intermediate images to disk
-re: dump intermediate energy maps to disk
-rs: dump intermediate seams to disk
-tw TARGET_WIDTH: resize image width to TARGET_WIDTH
-th TARGET_HEIGHT: resize image height to TARGET_HEIGHT
omit -tw and -th flags to enter interactive mode
Seam-carve a PNG image.
./seam implements Rubinstein, Avidan, and Shamir's forward-energy-based content resizing algorithm.
Note that FILENAME must be a PNG. The [-tw TARGET_WIDTH] and [-th TARGET_HEIGHT] may be used
together or independently (in which case ./seam will maintain the source aspect ratio).
By default, FILENAME (without target dimensions) launches the editor in interactive mode.
EXIT the editor with Q or ESC.
SAVE the current VIEW with W.
Toggle SEAMS with S.
Toggle HEATMAP with H.
As of now, Seam will attempt to produce animated GIFS from any intermediate files written by the -rr, -re, and -rs flags through a system() call for ImageMagick. This behavior (which can be quite slow) may be turned off by reflagging #define ANIMATION 1 to #define ANIMATION 0.
(Source: Grant Wood, American Gothic)
(Source: Charles C. Ebbets, Lunch atop a Skyscraper)