Reactで例えばグラフなんかを生成したい場合、d3やChartistを使いたくなる。
いくつかグラフ用のコンポーネントは散見されるが、簡単なものであればスッと利用できそうだが、ちょっと複雑なカスタマイズなどするにはなかなか厳しそうに見えた。
なのでそのまま使うにはどうするかみたいなのを考えた。
(基本的に元のライブラリにパフォーマンスが引きづられがちになったりと若干の邪道感があるやり方になっているのでそこらへんは要注意)
Example
今回はChartistを使った。
下記のような感じでRendererを用意する。
class ChartistRenderer{
render(targetSelector, props){
let {data} = props
// 必要があればここでデータ加工したり。
var chart = new Chartist.Bar(targetSelector, {
labels: data[0],
series: [ data[1] ]
})
}
}
関数名とかはReact.Componentに気持ちだけ合わせておく。
この程度ならComponent側にメソッドとして持たせても良かったのだけど、実際に扱おうとするとこの辺りの処理が肥大化するし、Reactからそこらへんの感心事は切り出したい感じしたのでやっておく。
で、あとはコンポーネント。
class Chart extends React.Component{
constructor(){
super()
this.chartClassName = "chart"
this.renderer = new ChartistRenderer()
}
componentDidMount(){
this.renderChart()
}
componentDidUpdate(){
this.renderChart()
}
renderChart(){
var targetSelector = "." + this.chartClassName
this.renderer.render(targetSelector, this.props)
}
render(){
return <div className={this.chartClassName} ></div>
}
}
renderではChartistのコンテナとなるdivだけ作る。
そしてcomponentDidUpdate
とcomponentDidMount
でこのコンテナに対してrenderingをする。
renderの際にレンダリング対象のクラス名とデータ(propsを利用)を渡す。
propsでデータを渡すことで、propsがが変更されればcomponentDidUpdate
がフックされてくれるという仕組み。
あとは起動部分。適当に。
var data = [
["foo", "baz", "bar"],
[1, 10, 20]
]
var body = document.querySelector('.container')
React.render(<Chart data={data} />, body);
おしまい!
サンプル
jsbinのbabelモードうまく使えnなかったのでcodepenで
http://codepen.io/suisho/pen/GJEwaP/
参考
あとで調べたらd3で似たようなこと考えている人いた。
http://nicolashery.com/integrating-d3js-visualizations-in-a-react-app/