Movement is powerful way of establishing meaning and relationships between visual objects. In the example above, agency and a number of complex interactions are established through movement of simple shapes.
Motion perception is a complex theme, but some aspects are explained in simple terms in Gestalt psychology under “The Law of Common Fate”. This law stipulates that objects that move together, are perceived as single whole or group, something that is often utilised in GUI design.
Animation Easing
The code below uses a bell curve to generate the animation. The advantage of using this function is that the size of the points return to there original state once the transformation is over. There is however a boundless number of functions that can be used for animation, many of which have been documented by Robert Penner for several programming languages http://easings.net/.
When used for animation, such functions are often called "easing". Easing can be used to create more natural movements and for creating physics like effects such as bouncing.
let arrayWidth = 20; let arrayHeight = 20; let dist_x = 20; let dist_y = 20; let shapeWidth = 10; let shapeHeight = 10; let duration = 120; let scaleTimer = 0; let amplitude = 2; let wavelength = 30; let xOffset; let yOffset; var reactorPosition; function setup() { createCanvas(800, 700); xOffset = floor(width/2-(dist_x*arrayWidth/2)); yOffset = floor(height/2-(dist_y*arrayHeight/2)); reactorPosition = createVector(0, 0); noStroke(); fill(0); } function draw() { background(255); scaleTimer+=3; push(); translate(xOffset, yOffset); for (let i = 0; i<arrayWidth; i++ ) { for (let j = 0; j<arrayHeight; j++ ) { rectMode(CENTER); var myPos = createVector(i*dist_x, j*dist_y); var reactorDistance = dist(reactorPosition.x, reactorPosition.y, myPos.x, myPos.y); var myStartTime = int(scaleTimer-reactorDistance); var scaler = bellCurve(myStartTime, shapeWidth, amplitude, wavelength); ellipse(myPos.x, myPos.y, scaler, scaler); } } pop(); } function mouse_X() { return (mouseX - xOffset); //correct positions matrix translations } function mouse_Y() { return (mouseY - xOffset); //correct positions matrix translations } function mouseClicked() { scaleTimer = -100; reactorPosition.x = mouse_X(); reactorPosition.y = mouse_Y(); } function bellCurve(t, a, b, c) { // see https://en.wikipedia.org/wiki/Gaussian_function // t = time // a = start value // b = amplitude // c = wavelength var scaler = 1+(c/sqrt((c*c)+(t*t)))*b; //bell curve return scaler*a; } // alternative equation for easing function easeInOutQuart (t, b, c, d) { // seee http://www.robertpenner.com/easing/ // t = time // b = start value // c = change in value // d = duration if ((t/=d/2) < 1) return c/2*t*t + b; return -c/2 * ((--t)*(t-2) - 1) + b; }