矢量Chart圖表嵌入HTML5網絡拓撲圖的應用

使用 HT for Web (如下簡稱 HT)開發HTML5網絡拓撲圖的開發者有 Chart 需求的項目的時候,感受很痛苦,HT 集成的 Chart 組件中,並不包含有座標,在展示方面不是很直觀,可是也不是沒有解決方案,接下來咱們就來聊聊具體的解決方案。 javascript

首先,第一種解決方案是,在定義 Chart 矢量的時候在 comps 中除了定義 Chart 外,再添加幾個你自定義的繪製區域來繪製你想要的座標軸,效果及 example 以下:java

 

Chart 的定義代碼見附錄1(代碼的定義太長),代碼雖然長,可是代碼的邏輯並不亂,各個模塊間的矢量描述仍是比較清晰的,具體能夠參考 HT 的矢量手冊,看到如此長的代碼,我本身都沒信心去維護它,維護這樣的代碼純粹是體力活,並且複用性也不高,每個不一樣的 Chart 都要相似如此地繪製,繪製一兩個這樣的圖表感受還好,繪製多了,真心會感受很噁心,再這上面很浪費時間。網絡

其次,第二種解決方案是,經過數據綁定來自定義繪製座標軸。實現以上相同效果,其代碼見附錄2。能夠明顯看出其代碼量會比第一種解決方案好不少,並且代碼能夠複用。在其餘的圖表中,能夠將橫軸和縱軸的文本內容設置到 data 的 attr 屬性上,並在定義 chart 時使用上以下代碼就能夠實現座標文本的效果:學習

ht.Default.setImage('chartName', {
    width: Number,
    height: Number,
    comps: [
        {
            // define chart
        },
        {
            type: 'xAxis',
            rect: Array
        },
        {
            type: 'yAxis',
            rect: Array
        }
    ]
});

在這裏我已經經過 ht.Default.setCompType('typeName', function(g, rect, comp, data, view){}) 的方法定義了名字爲 xAxis 和 yAxis 的 CompType,這兩個 CompType 分別繪製了橫軸和縱軸的座標文本,代替了第一種方案制定多個 CompType 爲 text 的寫法,稍微優化了下代碼,提升代碼的可維護性。優化

可是,這樣但使用方法總剛覺有些彆扭,明明座標軸是 Chart 的一部分,在定義 Chart 上卻要硬生生地將圖表和座標部分分開,那若是用戶還要在定義標題、座標刻度、座標說明等需求,那這個方案仍是沒法爽快的解決大部分通用的需求,須要定義許多 CompType 來渲染不一樣的需求,並且在使用上也不是那麼爽快。接下來要說明的方案三,就是來解決使用上及維護上的問題。spa

最後,第三種解決方案是,和第二種解決方案差很少,都是經過 ht.Default.setCompType('typeName', function(g, rect, comp, data, view){}) 的方法來定義名字爲 axisChart 的 CompType,不一樣的是,數據並非設置到 data 中,而是在 ht.Default.setImage() 的 comps 中直接定義其相關屬性。具體的配置屬性說明及其具體的代碼實現能夠查看附件,使用方式很簡單,在引入 ht.js 核心文件的前提下,引入附件1的 axisChart.js 文件便可。設計

接下來來看下 axisChart 的具體使用及幾個簡單的例子:3d

例1:設計同一時刻不一樣小區之間的電流電壓狀況的柱狀圖柱狀圖:rest

 

代碼以下:orm

ht.Default.setImage('c1', {
    width: 600,
    height: 400,
    comps: [
        {
            type: 'axisChart',
            rect: [0, 0, 600, 400],
            yAxis: [
                {
                    name: '單位:V',
                    max: 270,
                    min: 150,
                    splitNumber: 10,
                    axisTitle: {
                        text: '電壓',
                        rotate: -90
                    },
                    axisLine: {
                        arrow: true
                    }
                },
                {
                    position: 'right',
                    name: '單位:I',
                    max: 20,
                    splitNumber: 20,
                    axisTitle: {
                        text: '電流',
                        rotate: 90
                    },
                    axisLabel: {
                        interval: 1
                    },
                    axisLine: {
                        arrow: true
                    }
                }
            ],
            xAxis: [
                {
                    type: 'category',
                    data: ['撫梅源', '藕花深處', '紫東花園', '紫金苑', '華府山水', '水雲間', '瑞景新城'],
                    axisTitle: {
                        text: '小區名稱'
                    }
                }
            ],
            series: [
                {
                    label: function(value){
                        return value + ' V';
                    },
                    data: {
                        values: [220, 210, 200, 209, 230, 215, 218],
                        color: '#f90'
                    }
                },
                {
                    yAxisPosition: 'right',
                    label: true,
                    data: {
                        values: [10, 4, 15, 9, 12, 18, 7],
                        color: '#af0'
                    }
                }
            ]
        }
    ]
});

例2: 不一樣時刻,不一樣小區的電壓狀況的折線圖:

 

代碼以下:

ht.Default.setImage('c2', {
    width: 600,
    height: 400,
    comps: [
        {
            type: 'axisChart',
            rect: [0, 0, 600, 400],
            yAxis: [
                {
                    name: '單位:V',
                    max: 240,
                    min: 190,
                    splitNumber: 10,
                    axisTitle: {
                        text: '電壓',
                        rotate: -90
                    },
                    axisLine: {
                        arrow: true
                    }
                }
            ],
            xAxis: [
                {
                    type: 'time',
                    min: new Date(2015, 0, 1),
                    max: new Date(2015, 0, 2),
                    splitNumber: 25,
                    axisTitle: {
                        text: '時間'
                    },
                    axisLabel: {
                        interval: 2,
                        formatter: function(value, index, min, max){
                            return value.format('dd-hh');
                        }
                    },
                    axisLine: {
                        arrow: true
                    }
                }
            ],
            series: [
                {
                    type: 'line',
                    data: {
                        values: [220, 210, 200, 209, 230, 215, 218, 220, 210, 200, 209, 230, 215, 218, 209, 230, 215, 218, 220, 210, 200, 209, 230, 215, 218],
                        color: '#f90'
                    }
                },
                {
                    type: 'line',
                    data: {
                        values: [225, 209, 208, 206, 205, 221, 213, 224, 218, 224, 205, 208, 216, 220, 208, 210, 219, 219, 210, 209, 219, 207, 222, 222, 215],
                        color: '#7d0'
                    }
                },
                {
                    type: 'line',
                    linePoint: true,
                    line3d: true,
                    data: {
                        values: [211, 216, 215, 205, 206, 206, 223, 217, 217, 215, 212, 221, 219, 222, 205, 208, 205, 218, 223, 222, 207, 215, 215, 222, 223],
                        color: '#ab0'
                    }
                }
            ]
        }
    ]
});

最後,還有一點要說明,axisChart 的代碼並非那麼的無懈可擊,我我的以爲代碼設計上仍是有些欠缺,全部的代碼總共有差很少 1000 行,我以爲太臃腫了,在設計上本身也感受不是那麼的友好,等想修改的時候發現已經投入太多時間了,還有好多事情等待着我我去學習、去探討,因此也就線這樣吧,等有空了再重構一番,可是我相信在功能上仍是可以知足大部分的需求,在設計上,或者是實現的方法上,仍是在使用過程當中發現的 bug,還望你們可以不吝賜教。

附錄1

ht.Default.setImage('chart', {
    width: 650,
    height: 380,
    comps: [
        // column chart
        {
            type: 'columnChart',
            rect: [10, 60, 630, 260],
            label: true,
            labelColor: '#20284C',
            labelFont: '8px Arial',
            series: [
                {
                    color: '#20284C',
                    values: [471, 482, 567, 525, 590, 637, 780, 679, 848]
                },
                {
                    color: '#303F74',
                    values: [275, 290, 361, 328, 346, 341, 440, 423, 505]
                },
                {
                    color: '#7E93CD',
                    values: [82, 104, 115, 118, 135, 154, 198, 197, 247]
                },
                {
                    color: '#A9B6DB',
                    values: [65, 78, 87, 87, 113, 130, 167, 159, 213]
                }
            ]
        },
        // 豎線
        {
            type: 'rect',
            rect: [15, 60, 1, 260],
            background: '#566CB0'
        },
        {
            type: 'shape',
            rect: [5.5, 30, 20, 30],
            borderWidth: 1,
            borderColor: '#566CB0',
            points: [0, 20 / 3 * 2, 20 / 2, 0, 20, 20 / 3 * 2, 20 / 2, 0, 20 / 2, 30],
            segments: [1, 2, 2, 1, 2]
        },
        // 座標文本
        {
            type: 'text',
            rect: [0, 320 - 26 * 10 - 8, 15, 16],
            align: 'right',
            text: Math.round(84.8 * 10)
        },
        {
            type: 'text',
            rect: [0, 320 - 26 * 9 - 8, 15, 16],
            align: 'right',
            text: Math.round(84.8 * 9)
        },
        {
            type: 'text',
            rect: [0, 320 - 26 * 8 - 8, 15, 16],
            align: 'right',
            text: Math.round(84.8 * 8)
        },
        {
            type: 'text',
            rect: [0, 320 - 26 * 7 - 8, 15, 16],
            align: 'right',
            text: Math.round(84.8 * 7)
        },
        {
            type: 'text',
            rect: [0, 320 - 26 * 6 - 8, 15, 16],
            align: 'right',
            text: Math.round(84.8 * 6)
        },
        {
            type: 'text',
            rect: [0, 320 - 26 * 5 - 8, 15, 16],
            align: 'right',
            text: Math.round(84.8 * 5)
        },
        {
            type: 'text',
            rect: [0, 320 - 26 * 4 - 8, 15, 16],
            align: 'right',
            text: Math.round(84.8 * 4)
        },
        {
            type: 'text',
            rect: [0, 320 - 26 * 3 - 8, 15, 16],
            align: 'right',
            text: Math.round(84.8 * 3)
        },
        {
            type: 'text',
            rect: [0, 320 - 26 * 2 - 8, 15, 16],
            align: 'right',
            text: Math.round(84.8 * 2)
        },
        {
            type: 'text',
            rect: [0, 320 - 26 * 1 - 8, 15, 16],
            align: 'right',
            text: Math.round(84.8 * 1)
        },
        {
            type: 'text',
            rect: [0, 320 - 8, 15, 16],
            align: 'right',
            text: 0
        },
        // Q
        {
            type: 'text',
            rect: [55, 322, 0, 16],
            align: 'center',
            text: 'Q2\'11'
        },
        {
            type: 'text',
            rect: [124, 322, 0, 16],
            align: 'center',
            text: 'Q3\'11'
        },
        {
            type: 'text',
            rect: [191, 322, 0, 16],
            align: 'center',
            text: 'Q4\'11'
        },
        {
            type: 'text',
            rect: [259, 322, 0, 16],
            align: 'center',
            text: 'Q1\'12'
        },
        {
            type: 'text',
            rect: [327, 322, 0, 16],
            align: 'center',
            text: 'Q2\'12'
        },
        {
            type: 'text',
            rect: [394, 322, 0, 16],
            align: 'center',
            text: 'Q3\'12'
        },
        {
            type: 'text',
            rect: [462, 322, 0, 16],
            align: 'center',
            text: 'Q4\'12'
        },
        {
            type: 'text',
            rect: [529, 322, 0, 16],
            align: 'center',
            text: 'Q1\'13'
        },
        {
            type: 'text',
            rect: [596, 322, 0, 16],
            align: 'center',
            text: 'Q2\'13'
        },
        // line
        {
            type: 'rect',
            rect: [15, 320, 620, 1],
            background: '#566CB0'
        },
        {
            type: 'shape',
            rect: [635, 310.5, 30, 20],
            borderWidth: 1,
            borderColor: '#566CB0',
            points: [20 / 3 * 2, 0, 30, 20 / 2, 20 / 3 * 2, 20, 30, 20 / 2, 0, 20 / 2],
            segments: [1, 2, 2, 1, 2]
        }
    ]
});

附錄2

ht.Default.setCompType('yAxis', function(g, rect, comp, data, view) {
    var labels = data.a('yLabels'),
            len = labels.length,
            x = rect.x,
            y = rect.y,
            w = rect.width,
            h = rect.height,
            dh = h / (len - 1);
    g.save();
    g.font = '12px arial, sans-serif';
    g.fillStyle = 'black';
    g.textAlign = 'right';
    for(var i = 0; i < len; i++){
        g.fillText(labels[i], x, y);
        y += dh;
    }
    g.restore();
});

ht.Default.setCompType('xAxis', function(g, rect, comp, data, view) {
    var labels = data.a('xLabels'),
            len = labels.length,
            x = rect.x,
            y = rect.y,
            w = rect.width,
            h = rect.height,
            dw = w / (len * 3 + 1),
            dw3 = 3 * dw;
    x += dw * 2;
    g.save();
    g.font = '12px arial, sans-serif';
    g.fillStyle = 'black';
    g.textAlign = 'center';
    for(var i = 0; i < len; i++){
        g.fillText(labels[i], x, y);
        x += dw3;
    }
    g.restore();
});

ht.Default.setImage('chart1', {
    width: 650,
    height: 380,
    comps: [
        // column chart
        {
            type: 'columnChart',
            rect: [10, 60, 630, 260],
            label: true,
            labelColor: '#20284C',
            labelFont: '8px Arial',
            series: [
                {
                    color: '#20284C',
                    values: [471, 482, 567, 525, 590, 637, 780, 679, 848]
                },
                {
                    color: '#303F74',
                    values: [275, 290, 361, 328, 346, 341, 440, 423, 505]
                },
                {
                    color: '#7E93CD',
                    values: [82, 104, 115, 118, 135, 154, 198, 197, 247]
                },
                {
                    color: '#A9B6DB',
                    values: [65, 78, 87, 87, 113, 130, 167, 159, 213]
                }
            ]
        },
        // 豎線
        {
            type: 'rect',
            rect: [15, 60, 1, 260],
            background: '#566CB0'
        },
        // 向上的箭頭
        {
            type: 'shape',
            rect: [5.5, 30, 20, 30],
            borderWidth: 1,
            borderColor: '#566CB0',
            points: [0, 20 / 3 * 2, 20 / 2, 0, 20, 20 / 3 * 2, 20 / 2, 0, 20 / 2, 30],
            segments: [1, 2, 2, 1, 2]
        },
        // 座標文本
        {
            type: 'yAxis',
            rect: [12, 60, 15, 260]
        },
        // Q
        {
            type: 'xAxis',
            rect: [10, 330, 630, 16]
        },
        // line
        {
            type: 'rect',
            rect: [15, 320, 620, 1],
            background: '#566CB0'
        },
        // 向右的箭頭
        {
            type: 'shape',
            rect: [635, 310.5, 30, 20],
            borderWidth: 1,
            borderColor: '#566CB0',
            points: [20 / 3 * 2, 0, 30, 20 / 2, 20 / 3 * 2, 20, 30, 20 / 2, 0, 20 / 2],
            segments: [1, 2, 2, 1, 2]
        }
    ]
});
相關文章
相關標籤/搜索