<canvas id="render"></canvas>
html, body {
  margin: 0;
  padding: 0;
  overflow: hidden;
}

canvas {
  width: 100%;
  height: 100%;
}
View Compiled
const canvas = document.getElementById("render");
const context = canvas.getContext('2d');

const toggleTime = 300; // 300ms

class Easing {
  static easeInOutQuad(x) {
    return x < 0.5 ? 2 * x * x : 1 - Math.pow(-2 * x + 2, 2) / 2;
  }

  static easeOutBounce(x) {
    const n1 = 7.5625;
    const d1 = 2.75;
    
    if (x < 1 / d1) {
        return n1 * x * x;
    } else if (x < 2 / d1) {
        return n1 * (x -= 1.5 / d1) * x + 0.75;
    } else if (x < 2.5 / d1) {
        return n1 * (x -= 2.25 / d1) * x + 0.9375;
    } else {
        return n1 * (x -= 2.625 / d1) * x + 0.984375;
    }
  }

  static easeInOutBounce(x) {
    return x < 0.5
      ? (1 - Easing.easeOutBounce(1 - 2 * x)) / 2
      : (1 + Easing.easeOutBounce(2 * x - 1)) / 2;
  }

  static easeInOutCirc(x) {
    return x < 0.5
      ? (1 - Math.sqrt(1 - Math.pow(2 * x, 2))) / 2
      : (Math.sqrt(1 - Math.pow(-2 * x + 2, 2)) + 1) / 2;
  }
}

class RGB {
  static clamp(value, min, max) {
    if (value < min)
      return min;
    if (value > max)
      return max;
    return value;
  }

  constructor(r, g, b) {
    this.r = RGB.clamp(r, 0, 255);
    this.g = RGB.clamp(g, 0, 255);
    this.b = RGB.clamp(b, 0, 255);
  }

  toString() {
    return `rgb(${this.r}, ${this.g}, ${this.b})`;
  }

  clone() {
    return new RGB(this.r, this.g, this.b);
  }
};

class RGBA extends RGB {
  constructor(r, g, b, a = 1) {
    super(r, g, b);
    this.a = RGB.clamp(a, 0, 1);
  }

  toString() {
    return `rgba(${this.r}, ${this.g}, ${this.b}, ${this.a})`;
  }

  clone() {
    return new RGBA(this.r, this.g, this.b, this.a);
  }
};

class Annotation {
  constructor(text, side, color) {
    this.text = text;
    this.side = side;
    this.color = color.clone();
  }

  render(context, position, size) {
    context.save();
    
    context.font = (size / 2) + "px sans-serif";

    if (this.side === "right" || this.side === "left") {
      context.fillStyle = this.color.toString();
      context.textBaseline = "middle";
      context.textAlign = this.side === "right" ? "left" : "right";
      context.fillText(this.text, position.x + (this.side === "right" ? size : -size), position.y);
    } else if (this.side === "top" || this.side === "bottom") {
      context.translate(position.x, position.y);
      context.rotate(-(Math.PI / 2));
      context.translate(-position.x, -position.y);
      context.fillStyle = this.color.toString();
      context.textBaseline = "middle";
      context.textAlign = this.side === "top" ? "left" : "right";
      context.fillText(this.text, position.x + (this.side === "top" ? size : -size), position.y);      
    }

    context.restore();
  }
};

class BinaryNumber {
  constructor(value, valueWidth, color, index, total, annotations) {
    this.value = [];
    this.old = [];
    this.cols = [];
    this.anno = annotations || {};

    for (let i = 0; i < valueWidth; ++i) {
      this.value.push(0);
      this.old.push(0);
      this.cols.push(color.clone());
    }

    this.valueWidth = valueWidth;
    this.setValue(value);
    
    this.bitSize = 30;
    this.bitSpacing = 10;
    this.color = color;
    this.index = index;
    this.total = total;
    this.width = (total * this.bitSize) + ((total - 1) * this.bitSpacing);
    this.height = (this.value.length * this.bitSize) + ((this.value.length - 1) * this.bitSpacing);
    this.offset = (this.index * this.bitSize) + (this.index * this.bitSpacing);
    this.future = 0;
    this.past = 0;
  }
  
  setValue(value) {
    this.old    = [...this.value];
    this.value  = [...value.toString(2)].map(v => +v);
    this.future = (new Date()).getTime() + toggleTime;

    if (this.value.length > this.valueWidth)
      throw TypeError("invalid value width provided");

    for (let i = this.value.length; i < this.valueWidth; ++i)
      this.value.unshift(0);

    if (this.value.length !== this.old.length || this.value.length != this.cols.length)
      throw TypeError("catastrophic failure"); 
  }
  
  step() {
    let now = (new Date()).getTime();
    let delta = 1;

    if (now < this.future) {
      delta = (this.future - now) / toggleTime;
    } else {
      return;
    }

    for (let index in this.value) {
      let value = this.value[index];
      let old   = this.old[index];
      let color = this.cols[index];

      if (value === old) {
        color.a = (value === 1) ? 1 : 0;
        continue;
      }

      delta = Easing.easeInOutQuad(delta);

      color.a = (value > old) ? (1 - delta) : (delta);
    }
  }
  
  render(context) {
    let position = {
      x: Math.floor(((context.canvas.width / 2) - (this.width / 2)) + this.offset),
      y: Math.floor(((context.canvas.height / 2) - (this.height / 2)) + this.height)
    };

    context.save();

    for (let index = this.value.length - 1; index >= 0; --index) {
        let value = this.value[index];
        let color = this.cols[index];
        let offColor = color.clone();
        offColor.a = .1;

        context.beginPath();
        context.fillStyle = offColor.toString();
        context.arc(position.x, position.y, this.bitSize / 2, 0, 2*Math.PI,true);
        context.fill();
        context.closePath();

        if (value === 1) {
          let shadowColor = color.clone();
          context.save();

          context.shadowBlur = 15;
          context.shadowColor = shadowColor.toString();

          let gradient = context.createRadialGradient(
            position.x, position.y, 2, position.x, position.y, this.bitSize / 2);
          gradient.addColorStop(0, color.clone().toString());
          gradient.addColorStop(1, new RGBA(color.r - 50, color.g - 50, color.b - 50, color.a).toString());

          context.beginPath();
          context.fillStyle = gradient;
          context.arc(position.x, position.y, this.bitSize / 2, 0, 2*Math.PI,true);
          context.fill();
          context.closePath();

          context.restore();
        }
        
        if (index in this.anno)
          this.anno[index].forEach(anno => anno.render(context, position, this.bitSize));

        position.y -= (this.bitSize + this.bitSpacing);
    }
    
    context.restore();
  }
};

function updateBinaryValues(numberArray) {
  let today = new Date();
  numberArray[0].setValue(+today.getYear().toString().substr(-2));
  numberArray[1].setValue(today.getMonth() + 1);
  numberArray[2].setValue(today.getDate());
  numberArray[3].setValue(today.getHours());
  numberArray[4].setValue(today.getMinutes());
  numberArray[5].setValue(today.getSeconds());
}

const annotationColor = new RGBA(50, 50, 50, 1);

// // https://www.schemecolor.com/rainbow-pastels-color-scheme.php
// const yearColor = new RGBA(255, 154, 162);
// const monthColor = new RGBA(255, 183, 178);
// const dayColor = new RGBA(255, 218, 193);
// const hourColor = new RGBA(226, 240, 203);
// const minuteColor = new RGBA(181, 234, 215);
// const secondColor = new RGBA(199, 206, 234);

// https://www.schemecolor.com/pastel-rainbow.php
const yearColor = new RGBA(204, 153, 201);
const monthColor = new RGBA(158, 193, 207);
const dayColor = new RGBA(158, 224, 158);
const hourColor = new RGBA(253, 253, 151);
const minuteColor = new RGBA(254, 177, 68);
const secondColor = new RGBA(255, 102, 99);


let numbers = [
  new BinaryNumber(0, 6, yearColor, 0, 6, {
    5: [
      new Annotation("1", "left", annotationColor),
      new Annotation("Year", "bottom", annotationColor)
    ],
    4: [new Annotation("2", "left", annotationColor)],
    3: [new Annotation("4", "left", annotationColor)],
    2: [new Annotation("8", "left", annotationColor)],
    1: [new Annotation("16", "left", annotationColor)],
    0: [
      new Annotation("32", "left", annotationColor),
      new Annotation("Year", "top", annotationColor)
    ],
  }),
  new BinaryNumber(0, 6, monthColor, 1, 6, {
    5: [new Annotation("Month", "bottom", annotationColor)],
    0: [new Annotation("Month", "top", annotationColor)]
  }),
  new BinaryNumber(0, 6, dayColor, 2, 6, {
    5: [new Annotation("Day", "bottom", annotationColor)],
    0: [new Annotation("Day", "top", annotationColor)]
  }),
  new BinaryNumber(0, 6, hourColor, 3, 6, {
    5: [new Annotation("Hour", "bottom", annotationColor)],
    0: [new Annotation("Hour", "top", annotationColor)]
  }),
  new BinaryNumber(0, 6, minuteColor, 4, 6, {
    5: [new Annotation("Minute", "bottom", annotationColor)],
    0: [new Annotation("Minute", "top", annotationColor)]
  }),
  new BinaryNumber(0, 6, secondColor, 5, 6, {
    5: [
      new Annotation("1", "right", annotationColor),
      new Annotation("Second", "bottom", annotationColor)
    ],
    4: [new Annotation("2", "right", annotationColor)],
    3: [new Annotation("4", "right", annotationColor)],
    2: [new Annotation("8", "right", annotationColor)],
    1: [new Annotation("16", "right", annotationColor)],
    0: [
      new Annotation("32", "right", annotationColor),
      new Annotation("Second", "top", annotationColor)
    ],
  })
];

function renderLoop() {
  canvas.width = window.innerWidth;
  canvas.height = window.innerHeight;
  
  context.fillStyle = 'black';
  context.beginPath();
  context.rect(0, 0, canvas.width, canvas.height);
  context.fill();
  context.closePath();

  for (let index in numbers) {
      let number = numbers[index]; 
      number.step();
      number.render(context);
  }
  
  requestAnimationFrame(renderLoop);
}

function main() {

  setInterval(() => {
    updateBinaryValues(numbers);
  }, 1000);

  requestAnimationFrame(renderLoop);
}

window.addEventListener('load', main);

External CSS

This Pen doesn't use any external CSS resources.

External JavaScript

This Pen doesn't use any external JavaScript resources.