Over at Math3ma, Tai-Danae Bradley shared the following puzzle, which she also featured in a fantastic (spoiler-free) YouTube video. If you’re seeing this for the first time, watch the video first.
Consider a square in the xy-plane, and let A (an “assassin”) and T (a “target”) be two arbitrary-but-fixed points within the square. **Suppose that the square behaves like a billiard table, so that any ray (a.k.a “shot”) from the assassin will bounce off the sides of the square, with the angle of incidence equaling the angle of reflection.
Puzzle: Is it possible to block any possible shot from A to T by placing a finite number of points in the square?
This puzzle found its way to me through Tai-Danae’s video, via category theorist Emily Riehl
See Tai-Danae’s post for a proof, which left such an impression on me I had to dig deeper. In this post I’ll discuss a visualization I made—now posted at the end of Tai-Danae’s article—as well as here and below (to avoid spoilers). In the visualization, mouse movement chooses the firing direction for the assassin, and the target is in green. Dragging the target with the mouse updates the position of the guards. The source code is on Github.
The visualization uses d3 library
The meat of the visualization is in two geometric functions.
1. Decompose a ray into a series of line segments—its path as it bounces off the walls—stopping if it intersects any of the points in the plane.
1. Compute the optimal position of the guards, given the boundary square and the positions of the assassin and target.
Both of these functions, along with all the geometry that supports them, is in geometry.js. The rest of the demo is defined in main.js, in which I oafishly trample over d3 best practices to arrive miraculously at a working product. Critiques welcome 🙂
As with most programming and software problems, the key to implementing these functions while maintaining your sanity is breaking it down into manageable pieces. Incrementalism is your friend.
We start at the bottom with a Vector class with helpful methods for adding, scaling, and computing norms and inner products.
This allows one to compute the distance between two points, e.g., with vector.subtract(otherVector).norm()
.
Next we define a class for a ray, which is represented by its center (a vector) and a direction (a vector).
The ray must be finite for us to draw it, but the length we’ve chosen is so large that, as you can see in the visualization, it’s effectively infinite. Feel free to scale it up even longer.
The interesting bit is the intersection function. We want to compute whether a ray intersects a point. To do this, we use the inner product as a decision rule
In our demo points are not infinitesimal, but rather have a small radius described by intersectionRadius
. For the sake of being able to see anything we set this to 3 pixels. If it’s too small the demo will look bad. The ray won’t stop when it should appear to stop, and it can appear to hit the target when it doesn’t.
Next up we have a class for a Rectangle, which is where the magic happens. The boilerplate and helper methods:
The function rayToPoints
that splits a ray into line segments from bouncing depends on three helper functions:
1. rayIntersection
: Compute the intersection point of a ray with the rectangle.
1. isOnVerticalWall
: Determine if a point is on a vertical or horizontal wall of the rectangle, raising an error if neither.
1. splitRay
: Split a ray into a line segment and a shorter ray that’s “bounced” off the wall of the rectangle.
(2) is trivial, computing some x- and y-coordinate distances up to some error tolerance. (1) involves parameterizing the ray and checking one of four inequalities. If the bottom left of the rectangle is and the top right is and the ray is written as , then—with some elbow grease—the following four equations provide all possibilities, with some special cases for vertical or horizontal rays:
In code:
Next, splitRay
splits a ray into a single line segment and the “remaining” ray, by computing the ray’s intersection with the rectangle, and having the “remaining” ray mirror the direction of approach with a new center that lies on the wall of the rectangle. The new ray length is appropriately shorter. If we run out of ray length, we simply return a segment with a null ray.
As you have probably guessed, rayToPoints
simply calls splitRay
over and over again until the ray hits an input “stopping point”—a guard, the target, or the assassin—or else our finite ray length has been exhausted. The output is a list of points, starting from the original ray’s center, for which adjacent pairs are interpreted as line segments to draw.
That’s sufficient to draw the shot emanating from the assassin. This method is called every time the mouse moves.
The function to compute the optimal position of the guards takes as input the containing rectangle, the assassin, and the target, and produces as output a list of 16 points.
If you read Tai-Danae’s proof, you’ll know that this construction is to
1. Compute mirrors of the target across the top, the right, and the top+right of the rectangle. Call this resulting thing the 4-mirrored-targets.
1. Replicate the 4-mirrored-targets four times, by translating three of the copies left by the entire width of the 4-mirrored-targets shape, down by the entire height, and both left-and-down.
1. Now you have 16 copies of the target, and one assassin. This gives 16 line segments from assassin-to-target-copy. Place a guard at the midpoint of each of these line segments.
1. Finally, apply the reverse translation and reverse mirroring to return the guards to the original square.
Due to WordPress being a crappy blogging platform I need to migrate off of, the code snippets below have been magically disappearing. I’ve included links to github lines as well.
Step 1 (after adding simple helper functions on Rectangle
to do the mirroring):
Step 2:
Step 3, computing the midpoints:
Step 4, returning the guards back to the original square, is harder than it seems, because the midpoint of an assassin-to-target-copy segment might not be in the same copy of the square as the target-copy being fired at. This means you have to detect which square copy the midpoint lands in, and use that to determine which operations are required to invert. This results in the final block of this massive function.
And that’s all there is to it!
There are a few improvements I’d like to make to this puzzle, but haven’t made the time (I’m writing a book, after all!).
1. Be able to drag the guards around.
1. Create new guards from an empty set of guards, with a button to “reveal” the solution.
1. Include a toggle that, when pressed, darkens the entire region of the square that can be hit by the assassin. For example, this would allow you to see if the target is in the only possible safe spot, or if there are multiple safe spots for a given configuration.
1. Perhaps darken the vulnerable spots by the number of possible paths that hit it, up to some limit.
1. The most complicated one: generalize to an arbitrary polygon (convex or not!), for which there may be no optional solution. The visualization would allow you to look for a solution using 2-4.
Pull requests are welcome if you attempt any of these improvements.
Until next time!
Like Loading…