The code that powers a jigsaw puzzle solving robot. This reliably solves a few 1000-piece all-white puzzles. I'm making this somewhat easier on myself by assuming the puzzle pieces come to form a grid, where each piece has four sides.
I made a Youtube video with Mark Rober here.
Fourty-four pages of detail can be found here.
-
Manually lay pieces out in the staging area
- Face up
- Must have a large enough gap between pieces for there to be 1+ dark pixels in the photos taken
- Need not form a grid of any sort
- On a black backdrop
-
Provide configuration about our setup:
- Edit config.py
-
Robot takes pictures of the staging area along a cornrow path
-
Process each photo
- Crop the photo by a configurable amount. This allows for faster processing as it is preferable to have photos with pieces closest to the center
- Segment the photo into a binary bitmap: 0s are the black background and 1s are pieces (or dust)
- Extract each piece from the photo by finding large connected islands of 1s
- Clean the extracted piece by removing fine details like dust and hairs
- Save off the piece as a binary bitmap
-
Process each piece
- Find the edge of the piece in the binary image
- Walk along the edge, creating a dense vector path
- Detect the four corners of the piece with an algorithm that finds the best four candidates based on a handful of heuristics
- "Enhance" the corners by finding where the two sides would intersect, to account for slightly dinged or rounded-off corners
- Extract the four sides by yanking all vertices between two consecutive corners
- Note which sides are edges by calculating how close to perfectly straight each side is
- Compute the best point inside the piece for the robot to grip the piece from, by computing an approximate incenter - the point inside a polygon furthest from the nearest side
- Save off the piece's data and metadata about its sides, position in the input photo, etc.
-
Deduplicate pieces that were seen in multiple images
- We use the vector data because our test puzzles had printed patterns on them useful for debugging
- Compare each piece to every other piece
- Eliminate pieces that were taken far enough away in gripper-space
- If two pieces were geographically proximal, we compare how similar their four sides are
- We only keep one of each duplicate found, and we select the one closest to the center of its photo, to minimize parallax
-
Find all other pieces each piece can feasibly connect with
- For each side in each piece, compare the geometric fit to all other pieces' sides
- ASCII art showing the solver comparing two sides (green and yellow) and their proximity (white=intersection):
- Save off a list of the most likely fits for each side, sorted by their geometric similarity
- Note: small improvements to this part of the algorithm have an outsized impace on solve times. This is because the current solver's runtime complexity blows up quickly as the connectivity of the graph (i.e. how many sides could match) gets denser
- For each side in each piece, compare the geometric fit to all other pieces' sides
-
Solve the puzzle
- Grab one of the four corner pieces (i.e. a piece whose two adjacent sides are edges)
- Walk around the edge of the puzzle, exploring each feasible piece that could be connected
- TODO add a graphic showing the spiral solving pattern
- TODO add a graphic showing the corner being evaluated with a piece that fits snuggly but isn't an edge, vs one that is
- We keep track of not just which position each piece goes in, but its orientation
- We do an exhaustive depth-first search until we've placed all 100 pieces
-
Determine how to move each piece from the staging area to the solution area
-
Assemble the puzzle in spiral-order
- e.g. spiral from edge, spiral from center, evens then odds, ...
-
Robot executes these movements
To start, I solved a kid's puzzle. To help with debugging, I randomly labeled each piece with Sharpie, but the algorithm doesn't use this data at all.
The solution of a 100 piece puzzle, showing each piece's position and orientation:
3^ 2^ 1^ 10< 40> 37> 36v 84< 32< 34v
6^ 5> 4^ 11^ 38> 31^ 24^ 23^ 42< 19<
9^ 8^ 7^ 12< 48v 41^ 39< 22^ 52^ 20<
51< 50^ 49< 13< 14< 47^ 65^ 66^ 54< 21<
62< 81> 63< 30^ 64^ 15< 78^ 93^ 57^ 33<
70^ 71< 46> 79< 72^ 16< 82^ 100> 55v 53v
69< 68< 85v 94^ 96^ 98^ 95v 67< 25< 56>
99^ 17< 88< 29< 59> 28< 83^ 77> 26^ 75^
58^ 80v 86> 18< 87< 97< 92^ 76> 27< 43<
45^ 61v 89^ 91< 73< 90< 60^ 74< 44^ 35v