Vue整合d3.v5.js製做--折線圖(line)

先上效果圖(x軸固定爲時間軸):vue

圖中出現的懸浮框是鼠標懸停效果node

 

一、環境說明:git

vue版本:"vue": "^2.5.2"
d3版本:"d3": "^5.9.1"

二、Line.vue源碼github

  1 <template>
  2   <div class="d3line" :id="id">
  3   </div>
  4 </template>
  5 
  6 <script>
  7   import * as d3 from 'd3'
  8   export default {
  9     name: 'd3line',
 10     props: {
 11       id: String,
 12       width: Number,
 13       height: Number,
 14       dataset: Array
 15     },
 16     mounted() {
 17       this.init();
 18     },
 19     methods: {
 20       init() {
 21         d3.select("#svg" + this.id).remove();
 22         let width = this.width ? this.width : 600;
 23         let height = this.height ? this.height : 600;
 24         let padding = {
 25           left: 80,
 26           right: 50,
 27           top: 50,
 28           bottom: 50
 29         }
 30         let colorZ = d3.scaleOrdinal(d3.schemeDark2)
 31         let parseTime = d3.timeParse("%Y-%m-%d")
 32         let xScale = d3.scaleTime().range([0, width - padding.left - padding.right])
 33         let dates = this.dataset.flatMap((d) => d.value.map(v => parseTime(v.key)))
 34         xScale.domain([d3.min(dates), d3.max(dates)])
 35         let yScale = d3.scaleLinear().range([height - padding.top - padding.bottom, 0])
 36         yScale.domain([0, d3.max(this.dataset.flatMap((d) =>  d.value.map( v => v.value) )) + 2])
 37         let xAxis = d3.axisBottom(xScale).tickFormat(d => d.getFullYear() + '-' + (d.getMonth() + 1) + '-' + d.getDate())
 38         let yAxis = d3.axisLeft(yScale)
 39         let svg = d3.select("#" + this.id).append("svg").attr("width", width).attr("height", height).attr("id", "svg" + this.id);
 40         svg.append('g')
 41           .attr('transform', 'translate(' + padding.left + ',' + (height - padding.bottom) + ')')
 42           .call(xAxis)
 43           .selectAll('text')
 44           .attr('dx', -20)
 45           .attr('dy', 10)
 46           .attr('transform', 'rotate(-20)')
 47           .style('font-weight', 'bold')
 48         svg.append('g')
 49           .attr('transform', 'translate(' + padding.left + ',' + padding.top + ')')
 50           .call(yAxis)
 51           .selectAll('text')
 52           .style('font-weight', 'bold')
 53         let line = d3.line().x(d => xScale(parseTime(d.key))).y(d => yScale(d.value))
 54         this.dataset.forEach((v, vi) => {
 55           let tp_x = 0,
 56             tp_y =0;
 57           svg.append("path")
 58             .attr('d' , line(v.value))
 59             .attr('transform', 'translate(' + padding.left + ',' + padding.top + ')')
 60             .attr('fill', 'none')
 61             .attr('stroke', (d, i) => colorZ(vi))
 62             .attr("stroke-width", 2)
 63             .style('stroke-dasharray', function(d, i) {
 64               return d3.select(this).node().getTotalLength();
 65             })
 66             .style('stroke-dashoffset', function(d, i) {
 67               return d3.select(this).node().getTotalLength();
 68             })
 69             .transition()
 70             .duration(2000)
 71             .ease(d3.easePolyOut)
 72             .delay((d, i) => i * 200)
 73             .style('stroke-dashoffset', 0)
 74           svg.selectAll('circle1')
 75             .data(v.value)
 76             .enter()
 77             .append('circle')
 78             .attr('cx', (d, i) => {
 79               let x = xScale(parseTime(d.key))
 80               if (i === v.value.length - 1) tp_x = x - 40
 81               return x
 82             })
 83             .attr('cy', (d, i) => {
 84               let y = yScale(d.value)
 85               if (i === v.value.length - 1) tp_y = y - 10
 86               return y
 87             })
 88             .attr('r', 2)
 89             .attr('transform', 'translate(' + padding.left + ',' + padding.top + ')')
 90             .style("fill", (d, i) => colorZ(vi))
 91             .on("mouseover", (d, i) => {
 92               let g = svg.append('g')
 93                 .attr('id', `hoverg${vi}${d.key}${d.value}`)
 94                 .attr("transform", "translate(" + (xScale(parseTime(d.key)) - 20) + "," + (yScale(d.value) + 30)  + ")" )
 95               g.append("rect")
 96                 .attr("x", function(d){ return this.parentNode.getBBox().x - 3;})
 97                 .attr("y", function(d, i){ return  this.parentNode.getBBox().y - 20})
 98                 .attr("width", 110)
 99                 .attr("height", 25)
100                 .style("fill", "#fffbf0")
101               g.append('text')
102                 .text(`${d.key}:${d.value}`)
103                 .style("fill", colorZ(vi))
104             })
105             .on("mouseout", (d) => d3.select(`#hoverg${vi}${d.key}${d.value}`).remove())
106             .transition()
107             .duration(1500)
108             .ease(d3.easePolyIn)
109             .delay((d, i) => i * 200)
110             .attr('r', 5)
111           // svg.selectAll('text1')
112           //   .data([v.name])
113           //   .enter()
114           //   .append('text')
115           //   .attr('dx', (d, i) => tp_x)
116           //   .attr('dy', (d, i) => tp_y)
117           //   .attr('transform', 'translate(' + padding.left + ',' + padding.top + ')')
118           //   .text((d) => d)
119           //   .style("fill", (d, i) => colorZ(vi))
120           //   .style('font-weight', 'bold')
121           svg.append('text')
122             .attr('dx', tp_x)
123             .attr('dy', tp_y)
124             .attr('transform', 'translate(' + padding.left + ',' + padding.top + ')')
125             .text(v.name)
126             .style("fill", colorZ(vi))
127             .style('font-weight', 'bold')
128         })
129       }
130     },
131     watch: {
132       dataset() {
133         this.init();
134       }
135     }
136   }
137 </script>
138 
139 <style>
140 
141 </style>

三、使用示例app

 1 <template>
 2   <div id="test">
 3     <!-- 必定要傳進去一個id,隨便傳一個 -->
 4     <d3line id="line" :dataset="data1"></d3line>
 5   </div>
 6 </template>
 7 
 8 <script>
 9   import d3line from '@/components/d3/Line'
10   export default {
11     name: 'test',
12     data() {
13       return {
14         data1: [
15           {
16             'name': '哈爾濱',
17             'value': [{key: '2015-1-1', value: 10}, {key: '2015-1-2', value: 12}, {key: '2015-1-3', value: 13}, {key: '2015-1-17', value: 17}]
18           },
19           {
20             'name': '海南',
21             'value': [{key: '2015-1-1', value: 9}, {key: '2015-1-2', value: 48}, {key: '2015-1-3', value: 5}, {key: '2015-1-17', value: 49}]
22           },
23           {
24             'name': '天津',
25             'value': [{key: '2015-1-2', value: 30}, {key: '2015-1-3', value: 1}, {key: '2015-1-4', value: 32}, {key: '2015-1-5', value: 10}]
26           }
27         ]
28       }
29     },
30     components: {
31       d3line
32     },
33     methods: {
34 
35     }
36   }
37 </script>
38 
39 <style scoped>
40 </style>

四、參考資料dom

https://github.com/d3/d3-scale-chromatic/blob/master/README.md#schemeCategory10svg

https://github.com/d3/d3-time-format#formatthis

https://jakearchibald.com/2013/animated-line-drawing-svg/spa

相關文章
相關標籤/搜索