<div id="container">
  <div id="message">
    <a id="animate" href="#">Transmit</a>
  </div>
</div>
@import compass

$color: #B95

body
  overflow: hidden
  padding: 0
  margin: 0

p, a
  text-shadow: 0 0 8px rgba(lighten($color, 50%), 0.3)
  font-family: Courier, monospace
  font-smoothing: antialiased
  margin: 0.8em 0
  font-size: 18px
  color: $color

.ghost,
.glitch
  opacity: 0.25

#message
  margin-left: -300px
  margin-top: -130px
  position: absolute
  height: 260px
  width: 600px
  left: 50%
  top: 50%

#animate
  $color: darken(desaturate($color, 10%), 30%) 
  text-shadow: 0 0 8px rgba(lighten($color, 50%), 0.1)
  position: absolute
  color: $color
  bottom: 0
  
#container
  $color1: #20202D
  $color2: #101018
  $color3: #060612
  background: -webkit-radial-gradient(50% 50%, ellipse, $color1 0%, $color2 75%, $color3 100%)
  background: -moz-radial-gradient(50% 50%, ellipse, $color1 0%, $color2 75%, $color3 100%)
  position: absolute
  height: 100%
  width: 100%
  left: 0
  top: 0

  &:before
    pointer-events: none
    position: absolute
    height: 100%
    width: 100%
    content: ''
    left: 0
    top: 0
    background-image: url()
View Compiled
MESSAGES = []
MESSAGES.push delay:0,    text:"Incoming transmission..."
MESSAGES.push delay:1200, text:"You don't talk to anybody."
MESSAGES.push delay:2200, text:"You don't interact with anybody."
MESSAGES.push delay:3600, text:"Your whole sense of reality is, pretty warped..."
MESSAGES.push delay:5200, text:"Does it bother you that we're not real?"

$container = $("#container")
$message = $("#message")
$animate = $("#animate")
$paragraph = null

scramble = (element, text, options) ->

  # Default properties.
  defaults =
    probability: 0.2
    glitches: '-|/\\'
    blank: ''
    duration: text.length * 40
    ease: 'easeInOutQuad'
    delay: 0.0

  # Convert the element to a jQuery object and build the settings object.
  $element = $(element)
  settings = $.extend defaults, options

  # Convenience methods.
  shuffle = () -> if Math.random() < 0.5 then 1 else -1
  wrap = (text, classes) -> """<span class="#{classes}">#{text}</span>"""

  # Glitch values.
  glitchText = settings.glitches
  glitchCharacters = glitchText.split ''
  glitchLength = glitchCharacters.length
  glitchProbability = settings.probability
  glitches = ((wrap letter, 'glitch') for letter in glitchCharacters)

  # Ghost values.
  ghostText = $element.text()
  ghostCharacters = ghostText.split ''
  ghostLength = ghostCharacters.length
  ghosts = ((wrap letter, 'ghost') for letter in ghostCharacters)

  # Text values.
  textCharacters = text.split ''
  textLength = textCharacters.length

  # Order and output arrays.
  order = [0...textLength].sort @shuffle
  output = []

  # Build the output array.
  for i in [0...textLength]
    glitchIndex = Math.floor Math.random() * (glitchLength - 1)
    glitchCharacter = glitches[glitchIndex]
    ghostCharacter = ghosts[i] or settings.blank
    addGlitch = Math.random() < glitchProbability
    character = if addGlitch then glitchCharacter else ghostCharacter
    output.push character

  # Animate the text.
  object = value:0
  target = value:1
  parameters =
    duration:settings.duration
    ease:settings.ease
    step: ->
      progress = Math.floor object.value * (textLength - 1)
      for i in [0..progress]
        index = order[i]
        output[index] = textCharacters[index]
      $element.html output.join ''
    complete: ->
      $element.html text

  # Animate the text.
  $(object).delay(settings.delay).animate target, parameters



animate = () ->
  for data, index in MESSAGES
    element = $paragraph.get index
    element.innerText = ''
    options = delay: data.delay
    scramble element, data.text, options
  return

initialise = () ->
  $animate.click animate
  $message.append "<p>" for text, index in MESSAGES
  $paragraph = $container.find "p"
  animate()
  return

initialise()
View Compiled
Run Pen

External CSS

This Pen doesn't use any external CSS resources.

External JavaScript

  1. //cdnjs.cloudflare.com/ajax/libs/jquery/2.1.3/jquery.min.js