Skip to content

Instantly share code, notes, and snippets.

@rrag
Last active July 31, 2021 16:26
Show Gist options
  • Save rrag/23dd469fe32d7fc1977c to your computer and use it in GitHub Desktop.
Save rrag/23dd469fe32d7fc1977c to your computer and use it in GitHub Desktop.
Chart with a dark theme
license: MIT
height: 770
"use strict";
var { ChartCanvas, Chart, series, scale, coordinates, tooltip, axes, annotation, indicator, helper } = ReStock;
var { CandlestickSeries, BarSeries, LineSeries, AreaSeries, StochasticSeries, BollingerSeries } = series;
var { discontinuousTimeScaleProvider } = scale;
var { CrossHairCursor, MouseCoordinateX, MouseCoordinateY, CurrentCoordinate } = coordinates;
var { EdgeIndicator } = coordinates;
var { OHLCTooltip, MovingAverageTooltip, BollingerBandTooltip, StochasticTooltip } = tooltip;
var { XAxis, YAxis } = axes;
var { stochasticOscillator, ema, bollingerBand } = indicator;
var { fitWidth, TypeChooser } = helper;
class CandleStickChartWithDarkTheme extends React.Component {
render() {
var height = 750;
var { data, type, width, ratio } = this.props;
var margin = {left: 70, right: 70, top:20, bottom: 30};
var gridHeight = height - margin.top - margin.bottom;
var gridWidth = width - margin.left - margin.right;
var showGrid = true;
var yGrid = showGrid ? { innerTickSize: -1 * gridWidth, tickStrokeOpacity: 0.2 } : {};
var xGrid = showGrid ? { innerTickSize: -1 * gridHeight, tickStrokeOpacity: 0.2 } : {};
var ema20 = ema()
.id(0)
.windowSize(20)
.merge((d, c) => {d.ema20 = c})
.accessor(d => d.ema20);
var ema50 = ema()
.id(2)
.windowSize(50)
.merge((d, c) => {d.ema50 = c})
.accessor(d => d.ema50);
var slowSTO = stochasticOscillator()
.windowSize(14)
.kWindowSize(1)
.stroke({ D: "#ea2bff", K: "#74d400", top: "#37a600", middle: "#b8ab00", bottom: "#37a600" })
.merge((d, c) => {d.slowSTO = c})
.accessor(d => d.slowSTO);
var fastSTO = stochasticOscillator()
.windowSize(14)
.kWindowSize(3)
.stroke({ D: "#ea2bff", K: "#74d400", top: "#37a600", middle: "#b8ab00", bottom: "#37a600" })
.merge((d, c) => {d.fastSTO = c})
.accessor(d => d.fastSTO);
var fullSTO = stochasticOscillator()
.windowSize(14)
.kWindowSize(3)
.dWindowSize(4)
.stroke({ D: "#ea2bff", K: "#74d400" })
.merge((d, c) => {d.fullSTO = c})
.accessor(d => d.fullSTO);
var bb = bollingerBand()
.stroke({ top: "#964B00", middle: "#8c9900", bottom: "#964B00" })
.fill("#adffaf");
return (
<ChartCanvas ratio={ratio} width={width} height={750}
margin={margin} type={type}
seriesName="MSFT"
data={data} calculator={[ema20, ema50, slowSTO, fastSTO, fullSTO, bb]}
xAccessor={d => d.date} xScaleProvider={discontinuousTimeScaleProvider}
xExtents={[new Date(2012, 0, 1), new Date(2012, 6, 2)]}>
<Chart id={1} height={325}
yExtents={[d => [d.high, d.low], bb.accessor(), ema20.accessor(), ema50.accessor()]}
padding={{ top: 10, bottom: 20 }}>
<YAxis axisAt="right" orient="right" ticks={5} {...yGrid} inverted={true}
tickStroke="#FFFFFF" />
<XAxis axisAt="bottom" orient="bottom" showTicks={false} outerTickSize={0}
stroke="#FFFFFF" opacity={0.5}/>
<MouseCoordinateY
at="right"
orient="right"
displayFormat={d3.format(".2f")} />
<CandlestickSeries
wickStroke={d => d.close > d.open ? "#6BA583" : "#DB0000"}
fill={d => d.close > d.open ? "#6BA583" : "#DB0000"} />
<LineSeries yAccessor={ema20.accessor()} stroke={ema20.stroke()}/>
<LineSeries yAccessor={ema50.accessor()} stroke={ema50.stroke()}/>
<BollingerSeries calculator={bb} />
<CurrentCoordinate yAccessor={ema20.accessor()} fill={ema20.stroke()} />
<CurrentCoordinate yAccessor={ema50.accessor()} fill={ema50.stroke()} />
<EdgeIndicator itemType="last" orient="right" edgeAt="right"
yAccessor={d => d.close} fill={d => d.close > d.open ? "#6BA583" : "#DB0000"}/>
<OHLCTooltip origin={[-40, -10]}/>
<MovingAverageTooltip onClick={(e) => console.log(e)} origin={[-38, 10]}
calculators={[ema20, ema50]}/>
<BollingerBandTooltip origin={[-38, 60]} calculator={bb} />
</Chart>
<Chart id={2}
yExtents={d => d.volume}
height={100} origin={(w, h) => [0, h - 475]} >
<YAxis axisAt="left" orient="left" ticks={5} tickFormat={d3.format(".0s")}
tickStroke="#FFFFFF"/>
<BarSeries
yAccessor={d => d.volume}
fill={d => d.close > d.open ? "#6BA583" : "#DB0000"} />
</Chart>
<Chart id={3}
yExtents={slowSTO.domain()}
height={125} origin={(w, h) => [0, h - 375]} padding={{ top: 10, bottom: 10 }} >
<XAxis axisAt="bottom" orient="bottom" showTicks={false} outerTickSize={0}
stroke="#FFFFFF" opacity={0.5} />
<YAxis axisAt="right" orient="right" ticks={2} tickValues={slowSTO.tickValues()}
tickStroke="#FFFFFF"/>
<MouseCoordinateY
at="right"
orient="right"
displayFormat={d3.format(".2f")} />
<StochasticSeries calculator={slowSTO}
stroke={{ top: "#37a600", middle: "#b8ab00", bottom: "#37a600" }}/>
<StochasticTooltip calculator={slowSTO} origin={[-38, 15]}>Fast STO</StochasticTooltip>
</Chart>
<Chart id={4}
yExtents={fastSTO.domain()}
height={125} origin={(w, h) => [0, h - 250]} padding={{ top: 10, bottom: 10 }} >
<XAxis axisAt="bottom" orient="bottom" showTicks={false} outerTickSize={0}
stroke="#FFFFFF" opacity={0.5} />
<YAxis axisAt="right" orient="right" ticks={2} tickValues={fastSTO.tickValues()}
tickStroke="#FFFFFF"/>
<MouseCoordinateY
at="right"
orient="right"
displayFormat={d3.format(".2f")} />
<StochasticSeries calculator={fastSTO}
stroke={{ top: "#37a600", middle: "#b8ab00", bottom: "#37a600" }}/>
<StochasticTooltip calculator={fastSTO} origin={[-38, 15]}>Slow STO</StochasticTooltip>
</Chart>
<Chart id={5}
yExtents={fullSTO.domain()}
height={125} origin={(w, h) => [0, h - 125]} padding={{ top: 10, bottom: 10 }} >
<XAxis axisAt="bottom" orient="bottom" {...xGrid}
tickStroke="#FFFFFF" stroke="#FFFFFF" />
<YAxis axisAt="right" orient="right" ticks={2} tickValues={fullSTO.tickValues()}
tickStroke="#FFFFFF"/>
<MouseCoordinateX
at="bottom"
orient="bottom"
displayFormat={d3.timeFormat("%Y-%m-%d")} />
<MouseCoordinateY
at="right"
orient="right"
displayFormat={d3.format(".2f")} />
<StochasticSeries calculator={fullSTO}
stroke={{ top: "#37a600", middle: "#b8ab00", bottom: "#37a600" }}/>
<StochasticTooltip calculator={fullSTO} origin={[-38, 15]}>Full STO</StochasticTooltip>
</Chart>
<CrossHairCursor stroke="#FFFFFF" />
</ChartCanvas>
);
}
}
CandleStickChartWithDarkTheme.propTypes = {
data: React.PropTypes.array.isRequired,
width: React.PropTypes.number.isRequired,
ratio: React.PropTypes.number.isRequired,
type: React.PropTypes.oneOf(["svg", "hybrid"]).isRequired,
};
CandleStickChartWithDarkTheme.defaultProps = {
type: "svg",
};
CandleStickChartWithDarkTheme = fitWidth(CandleStickChartWithDarkTheme);
var parseDate = d3.timeParse("%Y-%m-%d %H:%M:%S");
d3["tsv"]("//rrag.github.io/react-stockcharts/data/MSFT.tsv", (err, data) => {
/* change MSFT.tsv to MSFT_full.tsv above to see how this works with lots of data points */
data.forEach((d, i) => {
d.date = new Date(d3.timeParse("%Y-%m-%d")(d.date).getTime());
d.open = +d.open;
d.high = +d.high;
d.low = +d.low;
d.close = +d.close;
d.volume = +d.volume;
// console.log(d);
});
/* change the type from hybrid to svg to compare the performance between svg and canvas */
ReactDOM.render(<TypeChooser type="hybrid">{type => <CandleStickChartWithDarkTheme data={data} type={type} />}</TypeChooser>, document.getElementById("chart"));
});
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="utf-8">
<title>React Stockcharts - CandleStickChartWithDarkTheme Example</title>
<link href="https://maxcdn.bootstrapcdn.com/bootstrap/3.3.6/css/bootstrap.min.css" rel="stylesheet">
<script type="text/javascript" src="https://cdnjs.cloudflare.com/ajax/libs/d3/4.2.7/d3.min.js"></script>
<script type="text/javascript" src="https://cdnjs.cloudflare.com/ajax/libs/react/15.3.2/react.min.js"></script>
<script type="text/javascript" src="https://cdnjs.cloudflare.com/ajax/libs/react/15.3.2/react-dom.min.js"></script>
<script type="text/javascript" src="https://cdnjs.cloudflare.com/ajax/libs/babel-standalone/6.17.0/babel.min.js"></script>
<!-- <script type="text/javascript" src="//unpkg.com/react-stockcharts@latest/dist/react-stockcharts.min.js"></script> -->
<script type="text/javascript" src="//rrag.github.io/react-stockcharts/dist/react-stockcharts.min.js"></script>
<style>
body {
background: #303030;
}
.react-stockcharts-tooltip {
fill: white;
}
.react-stockcharts-tooltip-label {
fill: yellow;
}
</style>
</head>
<body>
<span id="iconPreload" class="glyphicon glyphicon-arrow-down"></span>
<div id="chart">
</div>
<script>
// Use babel transform so the examples work on the browser
d3.request("./CandleStickChartWithDarkTheme.jsx")
.get(function(err, data) {
var outputEl = document.getElementById('chart');
try {
var output = Babel.transform(data.responseText, { presets: ["es2015", "react", "stage-3"] }).code;
eval(output);
} catch (ex) {
outputEl.innerHTML = 'ERROR: ' + ex.message;
}
})
</script>
</body>
</html>
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment