<!DOCTYPE html>
<html>
  <meta charSet="utf-8" />
  <title>mezza.biz</title>

  <meta name="description" content="Invest today" />

  <meta id="viewport" name="viewport" content="width=device-width, initial-scale=1.0" />

  <style>
    * {
      box-sizing: border-box;
    }

    html {
      --accent: #ed1c24;
      --foreground: #000;
      --background: #fff;
      --background-alt: #f5f5f5;

      font-family: 'Open Sans', Arial, sans-serif;
      font-weight: 800;
      background: var(--background);
      color: var(--foreground);
      -ms-text-size-adjust: 100%;
      -webkit-text-size-adjust: 100%;
      -moz-osx-font-smoothing: grayscale;
      -webkit-font-smoothing: antialiased;
      box-sizing: border-box;
      width: 100vw;
      height: 100vh;
      margin: 0;
      font-size: 1vw;
    }

    body {
      margin: 0;
    }

    .video {
      height: 100vh;
      width: 100vw;
      object-fit: cover;
      position: fixed;
      z-index: -1;
      top: 0%;
      left: 0;
    }

    @keyframes bling {
      0% {
        text-shadow: 0px 0px 0.0rem gold;
      }

      30% {
        text-shadow: 0px 0px 0.0rem gold;
      }

      50% {
        text-shadow: 0px 0px 0.5rem gold;
      }

      70% {
        text-shadow: 0px 0px 0.0rem gold;
      }

      100% {
        text-shadow: 0px 0px 0.0rem gold;
      }
    }

    @keyframes scale {
      0% {
        transform: scale(1);
      }

      50% {
        transform: scale(2);
      }

      100% {
        transform: scale(1);
      }
    }

    #ad {
      position: fixed;
      z-index: 1;
      bottom: 0;
      right: 0;
      left: 0;
      padding: 0 16px;
      color: gold;
      font-size: 5vh;
      text-shadow: 0px 0px 0.5rem gold;
      animation-name: bling;
      animation-duration: 1s;
      animation-timing-function: linear;
      animation-iteration-count: infinite;
      white-space: nowrap;
    }

    @keyframes rotate {
      from {
        transform: rotateY(0deg);
      }

      to {
        transform: rotateY(360deg);
      }
    }

    @keyframes depth {
      0% { text-shadow: 0 0 #5b5015; }
      25% { text-shadow: 4px 0 #5b5015, 8px 0 #5b5015, 12px 0 #5b5015, 16px 0 #5b5015, 20px 0 #5b5015; }
      50% { text-shadow: 0 0 #5b5015; }
      75% { text-shadow: -4px 0 #5b5015, -8px 0 #5b5015, -12px 0 #5b5015, -16px 0 #5b5015, -20px 0 #5b5015; }
      100% { text-shadow: 0 0 #5b5015; }
    }

    .money {
      position: fixed;
      top: 5vh;
      left: 5vw;
      z-index: 1;
      color: gold;
      font-size: 20vw;
      animation-name: rotate, depth;
      animation-timing-function: linear;
      animation-iteration-count: infinite;
      animation-duration: 5s;
    }
  </style>

  <script
    defer
    type="module"
  >
    import * as d3 from "https://cdn.jsdelivr.net/npm/d3@7/+esm";

    // Copyright 2021 Observable, Inc.
    // Released under the ISC license.
    // https://observablehq.com/@d3/candlestick-chart
    function CandlestickChart(data, {
      date = d => d.date, // given d in data, returns the (temporal) x-value
      open = d => d.open, // given d in data, returns a (quantitative) y-value
      close = d => d.close, // given d in data, returns a (quantitative) y-value
      high = d => d.high, // given d in data, returns a (quantitative) y-value
      low = d => d.low, // given d in data, returns a (quantitative) y-value
      title, // given d in data, returns the title text
      marginTop = 20, // top margin, in pixels
      marginRight = 30, // right margin, in pixels
      marginBottom = 30, // bottom margin, in pixels
      marginLeft = 40, // left margin, in pixels
      width = 640, // outer width, in pixels
      height = 400, // outer height, in pixels
      xDomain, // array of x-values (defaults to every weekday)
      xRange = [marginLeft, width - marginRight], // [left, right]
      xPadding = 0.2,
      xTicks, // array of x-values to label (defaults to every other Monday)
      yType = d3.scaleLinear, // type of y-scale
      yDomain, // [ymin, ymax]
      yRange = [height - marginBottom, marginTop], // [bottom, top]
      xFormat = "%b %-d", // a format specifier for the date on the x-axis
      yFormat = "~f", // a format specifier for the value on the y-axis
      yLabel, // a label for the y-axis
      stroke = "gold", // stroke color for the daily rule
      strokeLinecap = "round", // stroke line cap for the rules
      colors = ["gold", "white", "black"] // [up, no change, down]
    } = {}) {
      // Compute values.
      const X = d3.map(data, date);
      const Yo = d3.map(data, open);
      const Yc = d3.map(data, close);
      const Yh = d3.map(data, high);
      const Yl = d3.map(data, low);
      const I = d3.range(X.length);

      const weeks = (start, stop, stride) => d3.utcMonday.every(stride).range(start, +stop + 1);
      const weekdays = (start, stop) => d3.utcDays(start, +stop + 1).filter(d => d.getUTCDay() !== 0 && d.getUTCDay() !== 6);

      // Compute default domains and ticks.
      if (xDomain === undefined) xDomain = weekdays(d3.min(X), d3.max(X));
      if (yDomain === undefined) yDomain = [d3.min(Yl), d3.max(Yh)];
      if (xTicks === undefined) xTicks = weeks(d3.min(xDomain), d3.max(xDomain), 2);

      // Construct scales and axes.
      // If you were to plot a stock using d3.scaleUtc, you’d see distracting gaps
      // every weekend. This chart therefore uses a d3.scaleBand whose domain is every
      // weekday in the dataset. A few gaps remain for holiday weekdays, such as
      // Christmas, but these are infrequent and allow the labeling of Mondays. As a
      // band scale, we specify explicit tick values.
      const xScale = d3.scaleBand(xDomain, xRange).padding(xPadding);
      const yScale = yType(yDomain, yRange);

      // Compute titles.
      if (title === undefined) {
        const formatDate = d3.utcFormat("%B %-d, %Y");
        const formatValue = d3.format(".2f");
        const formatChange = (f => (y0, y1) => f((y1 - y0) / y0))(d3.format("+.2%"));
        title = i => `${formatDate(X[i])}
    Open: ${formatValue(Yo[i])}
    Close: ${formatValue(Yc[i])} (${formatChange(Yo[i], Yc[i])})
    Low: ${formatValue(Yl[i])}
    High: ${formatValue(Yh[i])}`;
      } else if (title !== null) {
        const T = d3.map(data, title);
        title = i => T[i];
      }

      const svg = d3.create("svg")
          .attr("width", width)
          .attr("height", height)
          .attr("viewBox", [0, 0, width, height])
          .attr("style", "max-width: 100%; height: auto; height: intrinsic;");

      const g = svg.append("g")
          .attr("stroke", stroke)
          .attr("stroke-linecap", strokeLinecap)
        .selectAll("g")
        .data(I)
        .join("g")
          .attr("transform", i => {
            return `translate(${xScale.step() * i},0)`;
          });

      g.append("line")
          .attr("y1", i => yScale(Yl[i]))
          .attr("y2", i => yScale(Yh[i]));

      g.append("line")
          .attr("y1", i => yScale(Yo[i]))
          .attr("y2", i => yScale(Yc[i]))
          .attr("stroke-width", xScale.bandwidth())
          .attr("stroke", i => colors[1 + Math.sign(Yo[i] - Yc[i])]);

      if (title) g.append("title")
          .text(title);

      return svg.node();
    }

    const getNewDataPoint = (oldPoint) => ({
      date: new Date(+oldPoint.date + (24 * 60 * 60 * 1000)), 
      open: oldPoint.close,
      high: oldPoint.close + Math.random() * 5000 - 1000,
      low: oldPoint.close - Math.random() * 2000 + 1000,
      close: oldPoint.close + Math.random() * 2000 - Math.random() * 1000,
      volume: Math.random() * 21899500 / 2 + 4229132,
    });

    const open = Math.random() * 200;
    const initialPoint = {
      date: new Date(),
      open,
      high: open + Math.random() * 400,
      low: open - Math.random() * 200,
      close: open + Math.random() * 2000 - Math.random() * 1000,
      volume: 21899500,
    };

    const baseArray = new Array(200)
      .fill(null)
      .reduce((list, _, i) => [
        ...list,
        getNewDataPoint(list[i]),
      ], [initialPoint]);

    const createChart = () => {
      const svgs = document.body.getElementsByTagName('svg');
      for (let svg of svgs) {
        document.body.removeChild(svg);
      }

      const chart = CandlestickChart(baseArray, {
        date: d => d.date,
        high: d => d.high,
        low: d => d.low,
        open: d => d.open,
        close: d => d.close,
        yLabel: "↑ Price ($)",
        width: window.innerWidth,
        height: window.innerHeight,
      });

      document.body.appendChild(chart);
    }

    createChart();

    setInterval(() => {
      baseArray.shift();
      baseArray.push(getNewDataPoint(baseArray[baseArray.length - 1]));
      createChart();
    }, 500);
  </script>
  <body>
    <video
      class="video"
      src="/assets/money-rains.mp4"
      autoplay="true"
      loop="true"
      muted="true"
    ></video>
    <div id="charts"></div>
    <marquee
      id="ad"
      scrolldelay="10"
      truespeed="true"
      behavior="scroll"
    >
      mezza
      nineninenine //
      here for YOUR business //
      we make that line go UP //
      NEVER settle for less //
      mail NOW to talk to your personal finance savior //
      biz@mezza.biz
    </marquee>
    <div class="money">$</div>
  </body>
</html>