At first, I wanted to do something that was static (e.g. putting everything in setup) since that was an option presented by the professor. However, the more I thought about incorporating randomness into my project, I thought that having some animation or movement would be the best way to do that. I still wanted to keep the artwork simple and focused on making use of a clear, structured pattern while allowing for unexpected variations.
I decided to create a looping animation that generates geometric designs at five key positions on the canvas: center, top-left, top-right, bottom-left, and bottom-right.
let positions = [ [width / 2, height / 2], // cent [0, 0], // tl [width, 0], // tr [0, height], // bl [width, height] // br ];
Each of these positions contains a series of concentric shapes that shift over time, creating a dynamic, evolving effect. The rotation is driven by angleOffset * j, which ensures that each circle is slightly different in position based on frameCount, adding to the sense of movement. The shapes themselves are defined using beginShape() and endShape(CLOSE), which connect the calculated points to form a jagged circular pattern.
let angleOffset = frameCount * 0.1; for (let i = 0; i < positions.length; i++) { push(); translate(positions[i][0], positions[i][1]); stroke(colors[i]); for (let j = 0; j < numCircles; j++) { let radius = map(j, 0, numCircles, 20, maxRadius); beginShape(); for (let angle = 0; angle < TWO_PI; angle += PI / 6) { let x = radius * cos(angle + angleOffset * j); let y = radius * sin(angle + angleOffset * j); vertex(x, y); } endShape(CLOSE); } pop(); }
I wanted to do a bit more and add an interactive element on top of the randomness. So, I added a function that shuffles the colors whenever you click on the canvas. This makes sure that while keeping the overall structure of the artwork the same, each interaction with the piece introduces a fresh, randomized arrangement of colors, keeping it visually engaging.
function mousePressed() { shuffle(colors, true); }
This project helped me appreciate the balance between structure and randomness in p5.js/computational art. By keeping certain elements predictable (such as placement and form) while allowing colors and movement to shift dynamically, I was able to create an interactive piece that feels both intentional and surprising. Well…at least I hope! Check it out: