I spent the last week learning Clojure and generating jigsaw puzzles as part of my one-week programming retreat at the Recurse Center.
Why jigsaw puzzles? I was motivated by two things: first, I wanted a good-sized language-learning project. Secondly, I was heavily inspired by the amazing, beautiful, intricate jigsaw puzzles produced by Nervous Systems and wanted to experiment with similar-ish generative methods. (I’m a sucker for generative things and hadn’t played with generative algorithms too much before.) If you want a crazy cool puzzle, seriously, go buy one from Nervous.
I think they turned out well!
data:image/s3,"s3://crabby-images/cb524/cb524a89bda341c3b68dca714626bc7104606671" alt="A hand holding the cat-shaped puzzle piece."
Because I have access to NYC Resistor’s laser cutter, the obvious thing to do was generate SVGs which I could then laser cut. If you haven’t worked with SVG before, it’s an XML-based format for describing vector graphics. It’s pretty easy to generate “by hand”.
Here is what an SVG looks like, if you open it up with a text editor:
<svg width="100" height="100">
<circle
cx="50"
cy="50"
r="40"
stroke="green"
stroke-width="4"
fill="yellow"
/>
</svg>
This produces a yellow circle, centered at (50, 50), with a radius of 40, and a four-pixel green outline. You can view an SVG file in any web browser, or edit it in an editor like Inkscape.
data:image/s3,"s3://crabby-images/e7359/e7359dc150ae70c10e7a698e798dc4fb7f7885ee" alt="A yellow circle"
See? Easy-peasy.
I started by generating a “classic” jigsaw puzzle shape, and figuring out how to tile it.
data:image/s3,"s3://crabby-images/15eb2/15eb261fcb2af51adba96bfd292953738724c953" alt="A grid of mostly-identical jigsaw puzzles"
But I wanted something more interesting than just a grid of similar puzzle pieces! My next step was to use a Voronoi diagram to draw more irregularly-sized polygons around “seed” points. At first this created some amusing failures:
data:image/s3,"s3://crabby-images/19f7a/19f7ad25e7cc2611a01da41aae02f260f6b6bd9e" alt="A grid of orange dots and some black lines between them. Something looks weird - the lines skew off in random directions!"
Oops. This is what happens when you draw points at (x, x) instead of (x, y). Let’s fix those coordinates.
data:image/s3,"s3://crabby-images/e8ae0/e8ae032389fec308fae431fc561846d2e60d760e" alt="A grid of orange dots, surrounded by black lines representing voronoi edges. Now they actually are enclosed cells, like they're supposed to be."
If we replace those straight lines with puzzle-piece edges, we get something that starts to look like a more interesting puzzle. There are still obviously flaws to be ironed out here (e.g. edge overlap).
data:image/s3,"s3://crabby-images/33c63/33c6306c27be66dfa639dcb9cb858749056f7f44" alt="Similar to the previous image, polygons constructed by voronoi tiling then have their edges deformed using puzzle-piece-like squiggles. There is some overlap between lines so this would not make a good puzzle."
I wanted to make more novel puzzle piece shapes, though, so I turned to the SVG path
type. You can draw Bézier curves in SVG pretty easily:
<path d="M 0 0 C 0 -100 50 -100 50 0 S 100 100 100 0"
stroke="blue"
fill="transparent"
transform="translate(0 400)"/>
data:image/s3,"s3://crabby-images/bcb0c/bcb0c1e6b495a95dc4017a8f261ab4c87c1f2b1a" alt="A blue curve that dips up then down."
OK, here’s what it looks like when we replace our puzzle piece shape with some random-ish curves:
data:image/s3,"s3://crabby-images/efe35/efe352e0658c75591eef16fcc5d2a85fbfde8826" alt="More puzzle pieces, now with squiggles."
Add more curves and it gets even better!
data:image/s3,"s3://crabby-images/5b60d/5b60d35f1f1c6cbd346f970e5837a72fb1a5d985" alt="Very squiggly pieces."
I also experimented with variations on how to place the seed points for my puzzle generation. Here’s one that’s based on a circular point distribution.
data:image/s3,"s3://crabby-images/b00e2/b00e20d072f1077b1ebbeb85216d2a0df3044d47" alt="A puzzle arranged by concentric circles of squiggles."
Now I had some monstrously-irregular puzzle pieces to play with. Cool! I wanted to take it one step further by implementing whimsy pieces. In jigsaw jargon, a whimsy piece is a themed, recognizably-shaped puzzle piece. They might be butterflies or people or letters or…you name it!
I modified my puzzle-generator to clear space for a whimsy piece, first testing it with circular whimsy pieces.
data:image/s3,"s3://crabby-images/3639e/3639e0c126483a313f23e8ac4b77dec4f72fb662" alt="A puzzle with two circular pieces placed inside it."
Then, using a kd-tree, I identified the whimsy piece’s nearest-neighbors and connected it back to the rest of the puzzle. Here’s a cat!
data:image/s3,"s3://crabby-images/83cd6/83cd6e1926885b065bc3fa0f5bc37f95773f7c24" alt="Same image as earlier - a puzzle with a cat-shaped piece in it."
And, finally, I took these files over to NYC Resistor and lasered them.
data:image/s3,"s3://crabby-images/0f644/0f6449a6cefcfb9829dd343c36eb1847c7ce8d4f" alt="Laser cutting cutting puzzle pieces into white acrylic"
data:image/s3,"s3://crabby-images/8eb77/8eb77208f9ba3e992b2289cdf929bb609b2bfda4" alt="The completed puzzle, with the cat whimsy removed."
It took a group of us about ninety minutes to solve the cat puzzle. Not bad for four days’ work!
data:image/s3,"s3://crabby-images/c947e/c947ed958f27904a530d23f807170607432da54c" alt="Several people gathered around the puzzle, working on solving it."
All of the code is available on Github at bonniee/svg-puzzle-gen. (It’s my first Clojure program, so I’m sure there are plenty of non-idiomatic things happening there.)
Dependencies / thank-yous:
Testimonials from playtesters:
- “This is awesome!”
- “This is horrible!”
- “This is amazing! And by amazing I mean terrible!”
- “Why are all the puzzle pieces the same ???”
- “Is this supposed to be evil?”
- “How can I get one?”