Week 4 : Generative text

For this week’s project, I wanted to create something with a cosmic theme that felt both interactive and magical. I focused on shades of blue and purple to match the theme and added a twinkling star effect as the background. The core of the project is the interactive bubbles—users can click on the screen to generate bubbles, and each bubble will display a random star sign. Clicking on an existing bubble reveals a short message related to that star sign. The letters of the message then fall down in an effect imitating a meteor shower.

CLICK TO CREATE AND POP BUBBLES

One part of the code that I’m particularly proud of is the way the bubbles and the falling messages behave. When the user clicks on a bubble, the bubble displays a message tied to the star sign. After a few seconds, the letters from the message “fall” like meteors. This falling effect was challenging to create because I had to ensure the letters moved smoothly and looked dynamic without overlapping or bunching together. Balancing the timing and position of each letter took some effort, but I’m happy with how it turned out. It adds a playful touch to the overall design.

I’m also proud of how the bubbles behave when they’re generated. Perfecting their collision and bounce behavior was tricky—it was difficult to make sure they didn’t overlap or get stuck together without affecting their smooth movement across the screen. It took a lot of experimenting to perfect the constraints that controlled their movement, Despite the challenges, I found it rewarding to see how small tweaks could make such a big improvement in the final product. These interactions made the overall experience feel more dynamic and immersive, which is exactly what I was aiming for with the cosmic theme.

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
class Fortune {
constructor(x, y, message) {
this.x = x;
this.y = y;
this.message = message;
this.alpha = 0; // Start invisible for fade-in effect
this.timer = 90; // Now lasts for 3 seconds
this.fadeInSpeed = 5; // Controls how fast it fades in
this.released = false;
this.floatOffset = 0; // For a slight floating effect
}
update() {
if (this.timer > 0) {
this.timer--;
if (this.alpha < 255) this.alpha += this.fadeInSpeed; // Gradually appear
this.floatOffset = sin(frameCount * 0.1) * 2; // Subtle floating effect
} else if (!this.released) {
this.releaseLetters();
this.released = true;
}
}
display() {
if (this.timer > 0) {
push();
translate(this.x, this.y + this.floatOffset);
// Glowing text effect
fill(255, 255, 255, this.alpha);
textSize(14);
textAlign(CENTER, CENTER);
drawingContext.shadowBlur = 10;
drawingContext.shadowColor = color(255, 200, 255, this.alpha);
text(this.message, 0, 0);
pop();
}
}
releaseLetters() {
for (let i = 0; i < this.message.length; i++) {
rainingLetters.push(new Letter(this.x, this.y, this.message.charAt(i)));
}
}
}
class Letter {
constructor(x, y, char) {
this.x = x + random(-10, 10);
this.y = y;
this.char = char;
this.speed = random(1, 3);
this.alpha = 255;
}
update() {
this.y += this.speed;
this.alpha -= 3;
}
display() {
push();
translate(this.x, this.y);
fill(255, this.alpha);
textSize(20);
textAlign(CENTER, CENTER);
text(this.char, 0, 0);
pop();
}
}
class Fortune { constructor(x, y, message) { this.x = x; this.y = y; this.message = message; this.alpha = 0; // Start invisible for fade-in effect this.timer = 90; // Now lasts for 3 seconds this.fadeInSpeed = 5; // Controls how fast it fades in this.released = false; this.floatOffset = 0; // For a slight floating effect } update() { if (this.timer > 0) { this.timer--; if (this.alpha < 255) this.alpha += this.fadeInSpeed; // Gradually appear this.floatOffset = sin(frameCount * 0.1) * 2; // Subtle floating effect } else if (!this.released) { this.releaseLetters(); this.released = true; } } display() { if (this.timer > 0) { push(); translate(this.x, this.y + this.floatOffset); // Glowing text effect fill(255, 255, 255, this.alpha); textSize(14); textAlign(CENTER, CENTER); drawingContext.shadowBlur = 10; drawingContext.shadowColor = color(255, 200, 255, this.alpha); text(this.message, 0, 0); pop(); } } releaseLetters() { for (let i = 0; i < this.message.length; i++) { rainingLetters.push(new Letter(this.x, this.y, this.message.charAt(i))); } } } class Letter { constructor(x, y, char) { this.x = x + random(-10, 10); this.y = y; this.char = char; this.speed = random(1, 3); this.alpha = 255; } update() { this.y += this.speed; this.alpha -= 3; } display() { push(); translate(this.x, this.y); fill(255, this.alpha); textSize(20); textAlign(CENTER, CENTER); text(this.char, 0, 0); pop(); } }
class Fortune {
  constructor(x, y, message) {
    this.x = x;
    this.y = y;
    this.message = message;
    this.alpha = 0; // Start invisible for fade-in effect
    this.timer = 90; // Now lasts for 3 seconds
    this.fadeInSpeed = 5; // Controls how fast it fades in
    this.released = false;
    this.floatOffset = 0; // For a slight floating effect
  }

  update() {
    if (this.timer > 0) {
      this.timer--;
      if (this.alpha < 255) this.alpha += this.fadeInSpeed; // Gradually appear
      this.floatOffset = sin(frameCount * 0.1) * 2; // Subtle floating effect
    } else if (!this.released) {
      this.releaseLetters();
      this.released = true;
    }
  }

  display() {
    if (this.timer > 0) {
      push();
      translate(this.x, this.y + this.floatOffset);

      // Glowing text effect
      fill(255, 255, 255, this.alpha);
      textSize(14);
      textAlign(CENTER, CENTER);
      drawingContext.shadowBlur = 10;
      drawingContext.shadowColor = color(255, 200, 255, this.alpha);

      text(this.message, 0, 0);

      pop();
    }
  }

  releaseLetters() {
    for (let i = 0; i < this.message.length; i++) {
      rainingLetters.push(new Letter(this.x, this.y, this.message.charAt(i)));
    }
  }
}


class Letter {
  constructor(x, y, char) {
    this.x = x + random(-10, 10);
    this.y = y;
    this.char = char;
    this.speed = random(1, 3);
    this.alpha = 255;
  }

  update() {
    this.y += this.speed;
    this.alpha -= 3;
  }

  display() {
    push();
    translate(this.x, this.y);
    fill(255, this.alpha);
    textSize(20);
    textAlign(CENTER, CENTER);
    text(this.char, 0, 0);
    pop();
  }
}

Reflections and Future Ideas

Overall, I really enjoyed working on this project, even though it was a bit frustrating at times. Looking back, I feel like I could improve the overall aesthetic of the bubbles by adding a shine to make them look like they’re reflecting light. I’d also like to enhance the appearance of the text since the current font is quite simple. Another idea I have is to experiment with different styles for the falling letters—and maybe space them out more and slow the fall for a better visual experience, to better match the cosmic theme.

One thought on “Week 4 : Generative text”

  1. Nice! Have the bubbles start in different directions would make the sketch feel more dynamic. If each sign had it’s own colour it would also make for a more interesting “composition” as the bubbles are added.

Leave a Reply