Skip to content

2D Line Chart

js
// SPDX-License-Identifier: Apache-2.0
// Copyright : J.P. Morgan Chase & Co.

import * as anu from '@jpmorganchase/anu';
import * as d3 from 'd3';
import { Scene, HemisphericLight, ArcRotateCamera, Mesh, Vector3, Color3, StandardMaterial } from '@babylonjs/core';
import stocks from './data/stocks.csv';  //Our data

//Create and export a function that takes a Babylon engine and returns a Babylon Scene
export function linechart2D(engine){

  //Create an empty Scene
  const scene = new Scene(engine);

  //Add some lighting
  new HemisphericLight('light1', new Vector3(0, 10, 0), scene);

  //Add a camera that rotates around the origin and adjust its properties
  const camera = new ArcRotateCamera("Camera", 0, 0, 10, new Vector3(0, 0, 0), scene);
  camera.wheelPrecision = 20; // Adjust the sensitivity of the mouse wheel's zooming
  camera.minZ = 0;            // Adjust the distance of the camera's near plane
  camera.attachControl(true); // Allow the camera to respond to user controls
  camera.position = new Vector3(0, 0, -3.5);

  //Filter to only show a single stock
  let data = stocks.filter(d => d.symbol === "GOOG");
  
  //Create D3 functions to parse the time and date
  let parseTime = d3.timeParse("%b %d %Y");
  let dateFormat = d3.timeFormat("%Y");

  //Get all of the dates in our dataset into an array
  let dates = data.map((d) => parseTime(d.date));

  //Create the D3 functions that we will use to scale our data dimensions to desired output ranges for our visualization
  //In this case, we create scale functions that correspond to the x and y positions
  let scaleX = d3.scaleTime().domain(d3.extent(dates)).range([-1, 1]);  //Time
  let scaleY = d3.scaleLinear().domain([0, Math.max(...data.map(d => d.price))]).range([-1, 1]).nice();

  //Create an array of Vector3 that correspond to the positions along the stock price timeseries using the scales we created
  let path = data.map((row) => new Vector3(scaleX(parseTime(row.date)), scaleY(row.price), 0));

  //Since we also want to color in the area below the line using a ribbon, we create another array of Vector at y=0 that forms a 2D mesh
  let zeroPath = path.map((value) => new Vector3(value.x, scaleY(0), 0));

  //Create a Center of Transform TransformNode using create() that serves the parent node for all our meshes that make up our chart
  let CoT = anu.create("cot", "cot");
  //We need to make an Anu Selection separately, as create() does not return a Section but the created Babylon TransformNode
  let chart = anu.selectName("cot", scene);

  //Create the lineSystem that will render the paths we had created
  let lines = chart.bind("lineSystem", { lines: [path] }) //lines expects an array of arrays of Vector3, where each sub-array is its own line
                   .attr("color", Color3.Blue())

  //Create the ribbon that will render the area below the line
  let ribbon = chart.bind("ribbon",
    { pathArray: [path, zeroPath],      //pathArray expects an array of arrays of Vector3, where each subarray is the "edge" of a ribbon segment
      sideOrientation: Mesh.DOUBLESIDE  //Double side so that the ribbon can be seen behind the chart as well
    })
    .material((d,n,i) => {              //Set a material to change its color
      let mat = new StandardMaterial("ribbonMat");
      mat.diffuseColor = Color3.FromHexString("#4287f5");
      mat.alpha = 1;
      return mat;
  })
    .positionZ(-0.001);   //Move ribbon forward to prevent z-fighting

  //Use the createAxes() Anu helper function to create the axes for us based on our D3 scale functions
  //Also adjust its visual properties to properly format the axes labels
  anu.createAxes("axes", scene, { parent: chart,
                                  scale: { x: scaleX, y: scaleY },
                                  domainMaterialOptions: { "color": Color3.Black(), width: 0.05 },
                                  labelTicks: { x: scaleX.ticks(d3.timeYear) },
                                  labelFormat: { x: dateFormat, y: (text) => '$' + text }
  });

  return scene;
}