Learn Javascript logic, maths and expressions by solving puzzles in a simple visual grid.
By @JakeGMaths and inspired with permission by tixy.land. You can also try my Javascript tutorial or my infinity whiteboard.
Change the code below so your pattern on the left matches the example on the right. This first puzzle uses the parameter x, the column index.
Great! Notice how although its the 2nd column, x is equal to 1, not 2. That's because rows and columns are zero-indexed - ie the first is 0 not 1.
This time you need y instead of x. In computer graphics, the origin (0, 0) is top-left rather than bottom-left, so y increases as you go down.
The current grids are 8x8 in size. Later on, the grids get bigger. Because of zero-indexing, rows and columns go from 0 to 7.
Dots are indexed with i, starting from zero top-left and increasing to the bottom-right. Some challenges have hints - press '?' if stuck.
? Try i==20 to start with then tweak away
With an 8x8 grid, i goes from 0 to 63 (1 less than 8x8=64). You can use parameters x, y and i in all puzzles, but won't need the last one t until later.
Multiple rows, columns or consecutive dots can be turned on by creating an inequality using < or >.
? Start with x>1 then tweak away
You can also use >= and <= for inequalities.
Not every challenge will have text before it, or after it when completed. You can always scroll back up to remind yourself how things work.
? Like last challenge, but with y
? An inequality involving i
If unsure, play around and experiment and/or try thinking logically, for example about the coordinates of each white dot.
? Think about x and y for each point
Puzzles may require more than one equation or inequality. Link them together with && for 'and' or || for 'or'.
? Try 2 inequalities with && between
? 2 inequalities with || between
Puzzles can be solved in multiple ways. I suggest using [...].includes(i) for this one, but other ways will also work and get a tick.
? Find numbers for [...].includes(i)
[5,10,30].includes(i) creates an array or list of numbers, then checks if i is in that list. Usually there's a more efficient way to solve the puzzles.
You've seen half of this cross before. You need to add an equation for the other half.
? Think about the x and y for each point
Notice how every nth dot is blank? i%3 gives the remainder when i is divided by 3, and dots are blank when that remainder is 0 (because 0 is 'falsy').
When things repeat or wrap around, we're dealing with modulus or clock arithmetic - like how a clock wraps around after 60 minutes.
The logical 'NOT' operator ! can be used to toggle true/false to give the reverse of which dots are on. You'll also need brackets for this puzzle.
? !(i%6) will turn on every 6th dot
Dots are on when your code evaluates to a truthy value, and blank when falsy. true, 1 and eg 2.5 are all truthy; false and 0 are falsy.
Red?! Instead of thinking of true/false, actually each dot can be -1 for red, 0 for off and 1 (or true) for white.
? For red, you need -1
Behind the scenes, your code is run in a function (t, i, x, y) { return `${yourCode}` } for each dot. The evaluation of your code within this function turns dots off/white/red.
If you negate/subtract 'true', you get -1, but if you ! a truthy value, you get false.
? 2 equations with ||, one negated
? Use remainders (%) and some negation
? Add inequalities to last one
? Every nth where n=...
? Various lines and inequalities
? Just negate one thing from last one
I lied. You can return any number between -1 and 1. Fractional values make the dots smaller.
? Just a number between 0 and 1
Values greater than 1 are treated as 1, and lower than -1 are treated as -1.
Graduated patterns can be created using simple division.
? Remember: zero-indexed
We divide by 7 not 8 because x is 7 for the last column, and 7/7=1.
? Both x and y are needed
? -1 to 0 for red of various sizes
All white dots here are 1 or above, and red dots -1 or below, because values are clamped between -1 and 1.
? A subtraction involving both x and y
Remember when you see a white dot, it could be because its 'true', 1, or any number more than 1.
The last puzzle had values outside -1 to 1 which still appeared as red or white. For this puzzle, you have to bring them all into that range yourself.
? Like last puzzle, with some division
? The index of the last dot is not 64
You need the first dot to be exactly -1 here, and the last one exactly 1.
? You probably need to double something
That was our last 8x8 grid. Our grid size is about to increase to 16x16 dots. Bear that in mind when making your calculations.
The maths function sin() returns values waving between -1 and 1 for any numeric input, perfect for our needs!
? Put a parameter inside sin()
sin() waves repeats every 360° of input. But Javascript uses radians not degrees. 360° is 2π radians, so sin(0) is roughly equal to sin(6.3).
We have 16 rows or columns, and 2π is a bit more than 6, so in the last puzzle the wave repeated ~2.5 times. Stretch a wave using division on its input.
? Divide the input to sin() by an amount
You can use all members of the Javascript Math global without the prefix - eg just sin() or PI instead of Math.sin() or Math.PI.
? Try a wave on a different parameter
We get some really pretty patterns here! You'll notice the wave as you go from left to right along each row, but also where maxima and minima line up.
Play around with different multiples and fractions of the input to sin(), and note its wavelength (distance between identical points).
Other functions such as cos() and tan() are also available for you to play around with (but aren't needed for the puzzles).
? Add some equations and inequalities
Introducing our fourth and final parameter, t, the time in seconds. Time restarts when you change code or click the grids.
? Divide t by something to slow it down
Animations will run faster if you multiply t by something, and slower if you divide it.
If the animation seems static, click it to reset time to zero. Think about what numbers you need for red, white and blank dots and the time taken.
? How can you make time start at -1?
Notice how time keeps increasing but the dots stay white? Your code is returning values > 1 which are ignored and clamped to a maximum of 1.
We can create a basic clock using t, i and a simple mathematical operation.
? Values over 1 are treated as 1
This clock is faster, and the colours are inverted.
Patterns look much cooler when they're animated with t and a sin() wave!
? Anything animated involves t
? Anything animated involves t
Sometimes we don't want fractional values. You can round() values to the nearest integers.
Sometimes we don't want fractional values. You can round() values to the nearest integers.
Sometimes we don't want fractional values. You can round() values to the nearest integers.
That's the end of the tutorial. From here on, there's a selection of cool animations on a 32x32 grid. Hit '?' to see the code for each one.
? sin(t/50*i)
How can you change where the centre of the ripple is? Or speed it up?
? sin(t-sqrt((x-15.5)**2+(y-15.5)**2))
? 1/32*tan(t/64*x*tan(i-x))
? i%(t/10)-0.5
? x%(t/2)-0.5+x/32
? cos(50*sin((y-15.5)/(x-15.5))+5*t)
Experiment and create cool stuff! Let me know any feedback @JakeGMaths and be sure to share with others!
Try out more patterns at the original tixy.land.
By @JakeGMaths. You can also try my Javascript tutorial or my infinity whiteboard.