首頁 web前端 H5教程 如何建立 HTML5 Canvas 電信網路拓撲圖

如何建立 HTML5 Canvas 電信網路拓撲圖

Jun 14, 2018 am 11:37 AM
html5

這篇文章主要介紹了快速創建 HTML5 Canvas 電信網路拓撲圖的範例程式碼的相關資料,內容挺不錯的,現在分享給大家,也給大家做個參考。

前言

屬性清單想必大家都不會陌生,正常用HTML5 來做的屬性清單大概就是用下拉式選單之類的,而且很多情況下,下拉清單還不夠好看,怎麼辦?我試著用 HT for Web 來實現屬性欄點擊按鈕彈出多功能選框,對傳入的數據進行選擇的功能,感覺整體實踐起來還是比較簡單方便的,所以在這邊跟大家分享一下。

效果圖

#程式碼實作

拓撲圖

從上面的效果圖中我們可以看到,整個頁面分成3 個部分,左邊的graphView 拓樸圖部分,右下角的 tableView 表格部分,以及右上角的propertyView 屬性部分。我們先把整個場景分割出來,然後再向各個部分來加入具體的內容:

gv = new ht.graph.GraphView();
var tablePane = new ht.widget.TablePane(gv.dm());//表格面板组件
propertyView = new ht.widget.PropertyView(gv.dm());//formPane是在propertyView里的,所以要先定义

var rightView = new ht.widget.SplitView(propertyView, tablePane, 'v', 0.4);//分割组件,v分为上下层,比例为0.4:0.6
rightView.getView().style.borderLeft = '1px solid #000';
var borderPane = new ht.widget.BorderPane();//边框面板组件
borderPane.setRightView(rightView, 400);//设置 borderPane 右边组件为 rightView,宽度为400
borderPane.setCenterView(gv);//设置 borderPane 中间组件为 gv 
borderPane.addToDOM();//将 borderPane 组件添加进 body 中
登入後複製

上面程式碼中new 出來的部分都是HT 封裝好的組件,相當於“類別”,這裡解釋一下SplitView 分割組件,分割組件用於左右或上下分割兩個子組件, 子組件可為HT 框架提供的組件,也可為HTML 的原生組件,子組件以position為absolute 方式進行絕對定位,這個組件中的參數分別為(左組件或上組件, 右組件或下組件,h 表示左右分割v 表示上下分割,分割位置預設值為0.5,如果設定值為0~1則以百分比分割,大於1 代表左組件或上組件的絕對寬或高,小於1 代表右組件或下組件的絕對寬或高);還有BorderPane 面板組件為佈局容器,可在上、下、左、右、中的五個區域位置擺放子元件, 子元件可為HT 框架提供的元件,也可為HTML 原生元件,子元件以position 為absolute 方式進行絕對定位。這裡我結合了 SplitView 和 BorderPane 將場景分成了三個部分。最後記得一定要將最終的佈局容器加入到body 中或任一個HTML 標籤中,這樣才能在介面上顯示,其中addToDOM 的定義內容如下:

addToDOM = function(){   
    var self = this,
        view = self.getView(), //获取这个组件的底层 p 
        style = view.style;//获取底层 p 的样式 style 属性
    document.body.appendChild(view); //将底层 p 添加进 body 中
    style.left = '0';//HT 默认将组件都定义为决定定位 absolute,所以需要设置位置
    style.right = '0';
    style.top = '0';
    style.bottom = '0';      
    window.addEventListener('resize', function () { self.iv(); }, false);            
}
登入後複製

#HT 的元件通常會嵌入BorderPane、SplitView 和TabView 等容器中使用,而最外層的HT 元件則需要使用者手動將getView() 傳回的底層p 元素加入到頁面的DOM 元素中,這裡需要注意的是,當父容器大小變化時,如果父容器是BorderPane 和SplitView 等這些HT 預先定義的容器元件,則HT 的容器會自動遞歸呼叫孩子元件invalidate 函數通知更新。但如果父容器是原生的 html 元素, 則 HT 元件無法獲知需要更新,因此最外層的 HT 元件一般需要監聽 window 的視窗大小變化事件,呼叫最外層元件 invalidate 函式進行更新。

場景創建完畢。為了顯示不同節點對應的屬性的不同,我們在拓樸圖上新增了七個節點:

function initModel(){
    var name = "设备";
    var count = 0;
    var root = createNode(name + count++, name + (++count));//参数1为name,参数2为tag
    root.setImage('./symbols/机房/服务器.json');
    root.setName('服务器');
    root.s('label.position', 3);
    gv.sm().ss(root);//默认选中root节点

    for (var i = 0; i < 2; i++) {
        var iNode = createNode(name + count++, name + (++count));//参数1为name,参数2为tag
        createEdge(root, iNode);
        for (var j = 0; j < 2; j++) {
            var jNode = createNode(name + count++, name + (++count));
            createEdge(iNode, jNode);
        }
    }
}
登入後複製

createNode 函數的宣告如下:

function createNode(name, tag){//创建Node节点
    flag++;
    var node = new ht.Node();
    node.setName(name);
    node.setTag(tag);
    node.setImage(&#39;./symbols/机房/XX分系统.json&#39;);
    node.a(&#39;hidden&#39;, false);//自定义属性,可以控制node.a(&#39;hidden&#39;)来控制节点显隐

    node.a(&#39;接口类型&#39;, &#39;SATA&#39;);
    node.a(&#39;显卡&#39;, &#39;Nvidia&#39;);
    if(flag % 2 === 0){
        node.a(&#39;接口类型&#39;, &#39;IDE&#39;);
        node.a(&#39;显卡&#39;, &#39;ATI&#39;);
    }
    node.s(&#39;label.position&#39;, 11);
    gv.dm().add(node);//将节点添加进数据容器DataModel中

    node.tablePane1 = createTableView(serviceType, dataModel1);//创建表格面板
    node.tablePane2 = createTableView(serviceSize, dataModel2);
    node.tablePane3 = createTableView(version, dataModel3);

    node.formPane1 = createFormPane(node.tablePane1);//创建表单面板
    node.formPane1.title = &#39;类型&#39;;//为了给后面dialog对话框的标题做准备
    node.formPane2 = createFormPane(node.tablePane2);
    node.formPane2.title = &#39;内存&#39;;
    node.formPane3 = createFormPane(node.tablePane3);
    node.formPane3.title = &#39;型号&#39;;

    if(flag % 3 === 0){
        node.formPane3.v(&#39;tag&#39;, &#39;联想(Lenovo)服务器X3650M5 8871&#39;);
    }else{
        node.formPane3.v(&#39;tag&#39;, &#39;Lenovo IBM X3250 5458I21&#39;);
    }
    node.a(&#39;型号&#39;, node.formPane3.v(&#39;tag&#39;));
    return node;
}
登入後複製

我們透過控制這個節點的hidden 屬性,利用graphView 中的視覺過濾器setVisibleFunc 函數來控制節點的顯隱:

#
gv.setVisibleFunc(function(data){
    if(data.a(&#39;hidden&#39;)){
        return false;
    }
    return true;
});
登入後複製

屬性面板

有了節點自然就需要顯示屬性了,加上下面的tablePane 表格面板中的值,一共添加了七種屬性:

function createProperty(){//创建属性
    propertyView.addProperties([
        {
            name: &#39;name&#39;,//获取 name 属性,结合 accessType 属性最终实现对节点属性的存取,accessType 默认值为 null,如name为age,采用getAge()和setAge(98)的get/set或is/set方式存取(这边name为name,所以通过 getName() 获取)
            displayName: &#39;名称&#39;//设置属性名的显示文本值
        },
        {
            name: &#39;hidden&#39;,//获取 hidden 属性
            displayName: &#39;隐藏这个节点&#39;,
            accessType: &#39;attr&#39;,//如name为hidden,采用getAttr(&#39;hidden&#39;)和setAttr(&#39;hidden&#39;, false)的方式存取
            icon: &#39;images/alert.gif&#39;,//设置属性名左侧显示的图标
            valueType: &#39;boolean&#39;,//用于提示组件提供合适的renderer渲染 布尔类型,显示为勾选框
            editable: true//设置该属性是否可编辑
        },
        {
            name: &#39;grade&#39;,
            displayName: &#39;类型&#39;,
            accessType: &#39;attr&#39;,
            drawPropertyValue: function(g, property, value, rowIndex, x, y, w, h, data, view){//自定义属性值渲染函数
                var cb = function(v) {
                    data.a(&#39;grade&#39;, v);
                }
                return fillFormPane(data.formPane1, w, h, data.tablePane1, serviceType, cb);
            }
        },
        {
            name: &#39;number&#39;,
            displayName: &#39;内存&#39;,
            accessType: &#39;attr&#39;,
            drawPropertyValue: function(g, property, value, rowIndex, x, y, w, h, data, view){
                var cb = function(v) {
                    data.a(&#39;number&#39;, v);
                }
                return fillFormPane(data.formPane2, w, h, data.tablePane2, serviceSize, cb);
            }
        },
        {
            name: &#39;接口类型&#39;,
            accessType: &#39;attr&#39;,
            displayName: &#39;接口类型&#39;
        },
        {
            name: &#39;显卡&#39;,
            accessType: &#39;attr&#39;,
            displayName: &#39;显卡&#39;
        },
        {
            name: &#39;型号&#39;,
            accessType: &#39;attr&#39;,
            displayName: &#39;型号&#39;,
        }
    ]);
}
登入後複製

第三第四個屬性中drawPropertyValue 屬性的回傳值為fillFormPane 函數,這個函數的參數分別為(表單元件formP,表單元件寬w,表單元件高h,表單元件中按鈕點擊產生彈出框中的表格元件tableP,表格元件中的陣列內容arr,cb 函數將雙擊表格元件中的資料傳回的值賦值給form 表單中的ht. widget.TextField 文字方塊)。

第一個參數formP 表單元件的創建,表單元件的創建就是建立一個表單元件,在表單元件中加入一個文字方塊以及一個按鈕,這個步驟在HT 中也是相當的簡單:

function createFormPane(tPane) {//创建表单面板
    var formPane = new ht.widget.FormPane();
    formPane.setPadding(0);//设置表单四周与组件内容的间距
    var tField = new ht.widget.TextField();//创建一个文本框
    tField.setText(&#39;&#39;);//文本框的内容为空
    tField.setDisabled(true);//文本框不可操作
    formPane.addRow([//向表单中添加行
        {
            id: &#39;tag&#39;,//唯一标示属性,可通过formPane.getItemById(id)获取添加到对应的item对象
            element: tField//属性值可为 HTML原生元素、FormPane内部自绘制的文本信息以及HT自带组件如Button、CheckBox和ComboBox等
        },
        {
            button:{//设置了该属性后HT将根据属性值自动构建ht.widget.Button对象,并保存在element属性上
                label:&#39;...&#39;,//按钮上的文本内容
                onClicked: function(){//按钮点击事件
                    for(var i = 0; i < tPane.dm().size(); i++){//设置tablePane默认选中formPane对应的值
                        var data = tPane.dm().getDatas().get(i);
                        if(data.a(&#39;value&#39;) === formPane.v(&#39;tag&#39;)){
                            tPane.sm().ss(data);
                        }
                    }
                    return createDialog(tPane, formPane);//返回的是创建一个对话框,对话框的内容为表格面板
                }
            }
        }
    ], [0.5, 0.1]);//设置表格组件中第一个元素和第二个元素的显示比例。这个表格组件一共只有两个元素,一个文本框一个按钮,占比分别为 0.5 和 0.1
    return formPane;
}
登入後複製

createDialog 函數建立的過程也是簡潔明了,透過setConfig(config) 方法配置對話框的標題,尺寸,內容等,我給createDialog 傳了一個參數tPane 表格元件,用作對話方塊顯示的內容:

function createDialog(tPane){//创建弹出框
    dialog.setConfig({
        title: gv.sm().ld().getName()+"的"+formPane.title,//对话框的标题
        content: tPane, //直接将弹出框的内容设置为表格面板
        width: 400,//指定对话框的宽度
        height: 200,
        draggable: true,//指定对话框是否可拖拽调整位置
        closable: true,//表示是否显示关闭按钮
        maximizable: true,//表示对话框是否可被最大化
        resizeMode: "wh",//鼠标移动到对话框右下角可改变对话框的大小,wh表示宽高都可调整
        buttons: [//添加两个按钮
            {
                label: &#39;取消&#39;,
                action: function(){
                    dialog.hide()
                }
            },
            {
                label: &#39;确定&#39;,
            }
        ]
    });
    dialog.show();//显示对话框
}
登入後複製

第四個參數tableP 表格元件,也沒有什麼特別的,就是建立一個表單元件,然後在表單元件中新增列,步驟簡單,程式碼也相當簡單:

function createTableView(arr, dm){//创建表格组件
    var tableView = new ht.widget.TableView(dm);
    tableView.addColumns([//用json的数组参数方式批量添加列信息
        {
            displayName: &#39;ID&#39;,//获取表头的列名内容
            drawCell: function(g, data, selected, column, x, y, w, h, tableView){//自定义单元格渲染方式
                var id = tableView.getRowIndex(data);//返回data对象所在的行索引
                ht.Default.drawText(g, &#39;row&#39; + (id + 1), null, null, x, y, w, h, &#39;center&#39;);//绘制文字参数(g画笔对象,value文字内容,font文字字体,color文字颜色,x绘制开始的x坐标,y绘制开始的y坐标,w绘制的宽度,h绘制的高度,align文字水平对齐方式,vAlign文字垂直对齐方式)
            }
        },
        {
            displayName: &#39;Name&#39;,
            drawCell: function(g, data, selected, column, x, y, w, h, tableView){
                var id = tableView.getRowIndex(data);
                var info = arr[id];
                ht.Default.drawText(g, info, null, null, x, y, w, h, &#39;center&#39;);
            }
        }
    ]);
    return tableView;
}
登入後複製

解释完 fillFormPane 中的参数,来看看这个函数是如何定义的,基本上只差最后一步,点击 tablePane 表格组件中的元素,将这个元素返回给 formPane 表单组件中的 textField 文本框:

function fillFormPane(formP, w, h, tableP, arr, cb){//右边的formpane
    if(formP === undefined){
        return;
    }
    formP.setWidth(w);
    formP.setHeight(h);
    formP.setHGap(0);

    if(formP.v(&#39;tag&#39;) === &#39;undefined&#39; || formP.v(&#39;tag&#39;) === &#39;&#39;) {
        formP.v(&#39;tag&#39;, arr[0]);
    }

    tableP.onDataDoubleClicked = function(data){//表格组件中的data所在行被双击时回调
        var v = arr[data.a(&#39;index&#39;)];
        formP.v(&#39;tag&#39;, v);//根据id设置对应item元素值 为 setValue 的简写,这个id为tag的元素就是文本框
        dialog.hide();
        if (cb){cb(v);} //如果传入了cb参数,则设置data.a(&#39;number&#39;)/data.a(&#39;helloName&#39;)的值为表格中被双击的那一行的值,即赋值给第三第四属性
    }

    tableP.onDataClicked = function(data){//表格组件中的data所在行被单击时回调
        dialog.getConfig().buttons[1].action = function(){//点击“确定”才进行下面的操作
            var v = arr[data.a(&#39;index&#39;)];
            formP.v(&#39;tag&#39;, v);
            dialog.hide();
            if (cb){cb(v);} 
        }
    };
    return formP.getView();
}
登入後複製
登入後複製

function fillFormPane(formP, w, h, tableP, arr, cb){//右边的formpane
    if(formP === undefined){
        return;
    }
    formP.setWidth(w);
    formP.setHeight(h);
    formP.setHGap(0);

    if(formP.v(&#39;tag&#39;) === &#39;undefined&#39; || formP.v(&#39;tag&#39;) === &#39;&#39;) {
        formP.v(&#39;tag&#39;, arr[0]);
    }

    tableP.onDataDoubleClicked = function(data){//表格组件中的data所在行被双击时回调
        var v = arr[data.a(&#39;index&#39;)];
        formP.v(&#39;tag&#39;, v);//根据id设置对应item元素值 为 setValue 的简写,这个id为tag的元素就是文本框
        dialog.hide();
        if (cb){cb(v);} //如果传入了cb参数,则设置data.a(&#39;number&#39;)/data.a(&#39;helloName&#39;)的值为表格中被双击的那一行的值,即赋值给第三第四属性
    }

    tableP.onDataClicked = function(data){//表格组件中的data所在行被单击时回调
        dialog.getConfig().buttons[1].action = function(){//点击“确定”才进行下面的操作
            var v = arr[data.a(&#39;index&#39;)];
            formP.v(&#39;tag&#39;, v);
            dialog.hide();
            if (cb){cb(v);} 
        }
    };
    return formP.getView();
}
登入後複製
登入後複製

右上方属性栏的显示到这里就全部结束了,右下方的表格面板的创建方式也是雷同的,大家可以自行看代码理解。

自动布局

最后说一下整个界面节点的排布,HT 中的 autolayout 自动布局组件,即根据节点和连线关系,提供多种类型算法进行自动排布节点位置。 自动布局常用于图元较多,或连接关系教复杂时,不易于人工拖拽摆放的场景。我把各个布局方式通过按钮的方式呈现出来了,点击对应的按钮,布局方式就会根据按下的按钮设置的排布方式来自动布局:

首先,创建一个新的实例,传入需要自动布局的对象,可以为 DataModel、graphView 以及 graph3dView 三种,然后设置默认的排布方式:

autoLayout = new ht.layout.AutoLayout(gv);
setTimeout(function(){
    layout(&#39;towardsouth&#39;, true);//因为图片还没加载出来的时候,自动布局就按照节点的默认大小来布局的
}, 200);
登入後複製

接着创建 formPane 表单面板,添加进 body 中,放在 body 左上角,我不将所有代码粘出来了,就显示第一个布局的按钮就好:

function createDirectionForm(){
    var form = new ht.widget.FormPane();
    form.setWidth(200);//设置表单宽度
    form.setHeight(80);
    document.body.appendChild(form.getView());
    form.getView().style.background = &#39;#fff&#39;;
    form.getView().style.boxShadow = &#39;4px 16px 16px rgba(0, 0, 0, 0.1)&#39;;//设置阴影样式
    form.addRow([//这一行单独拿出来,作为标题
        {
            element: &#39;自动布局:&#39;,//显示的文字
        }
    ], [0.1]);//数组内只有一个对象,就只设置一个对象的宽度就好
    form.addRow([
        {
            button: {
                icon: &#39;布局/南布局.json&#39;,
                onClicked: function(){
                    layout(&#39;towardsouth&#39;, true);
                },
                background: null,
                labelColor: &#39;#fff&#39;,
                groupId: &#39;btn&#39;,
                toolTip: &#39;朝南布局&#39;,
                borderColor: null
            }
        },
        //....接下来添加剩下的6个按钮
    ], [0.1, 0.1, 0.1, 0.1, 0.1, 0.1, 0.1]);//数组中有七个对象,就要设置七个对象的宽度
    return form;
}
登入後複製

以上就是本文的全部内容,希望对大家的学习有所帮助,更多相关内容请关注PHP中文网!

相关推荐:

如何解决canvas绘图时遇到的跨域问题

以上是如何建立 HTML5 Canvas 電信網路拓撲圖的詳細內容。更多資訊請關注PHP中文網其他相關文章!

本網站聲明
本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn

熱AI工具

Undresser.AI Undress

Undresser.AI Undress

人工智慧驅動的應用程序,用於創建逼真的裸體照片

AI Clothes Remover

AI Clothes Remover

用於從照片中去除衣服的線上人工智慧工具。

Undress AI Tool

Undress AI Tool

免費脫衣圖片

Clothoff.io

Clothoff.io

AI脫衣器

Video Face Swap

Video Face Swap

使用我們完全免費的人工智慧換臉工具,輕鬆在任何影片中換臉!

熱門文章

熱工具

記事本++7.3.1

記事本++7.3.1

好用且免費的程式碼編輯器

SublimeText3漢化版

SublimeText3漢化版

中文版,非常好用

禪工作室 13.0.1

禪工作室 13.0.1

強大的PHP整合開發環境

Dreamweaver CS6

Dreamweaver CS6

視覺化網頁開發工具

SublimeText3 Mac版

SublimeText3 Mac版

神級程式碼編輯軟體(SublimeText3)

熱門話題

Java教學
1677
14
CakePHP 教程
1431
52
Laravel 教程
1334
25
PHP教程
1280
29
C# 教程
1257
24
HTML 中的表格邊框 HTML 中的表格邊框 Sep 04, 2024 pm 04:49 PM

HTML 表格邊框指南。在這裡,我們以 HTML 中的表格邊框為例,討論定義表格邊框的多種方法。

HTML 中的巢狀表 HTML 中的巢狀表 Sep 04, 2024 pm 04:49 PM

這是 HTML 中巢狀表的指南。這裡我們討論如何在表中建立表格以及對應的範例。

HTML 左邊距 HTML 左邊距 Sep 04, 2024 pm 04:48 PM

HTML 左邊距指南。在這裡,我們討論 HTML margin-left 的簡要概述及其範例及其程式碼實作。

HTML 表格佈局 HTML 表格佈局 Sep 04, 2024 pm 04:54 PM

HTML 表格佈局指南。在這裡,我們詳細討論 HTML 表格佈局的值以及範例和輸出。

HTML 輸入佔位符 HTML 輸入佔位符 Sep 04, 2024 pm 04:54 PM

HTML 輸入佔位符指南。在這裡,我們討論 HTML 輸入佔位符的範例以及程式碼和輸出。

HTML 有序列表 HTML 有序列表 Sep 04, 2024 pm 04:43 PM

HTML 有序列表指南。在這裡我們也分別討論了 HTML 有序列表和類型的介紹以及它們的範例

HTML onclick 按鈕 HTML onclick 按鈕 Sep 04, 2024 pm 04:49 PM

HTML onclick 按鈕指南。這裡我們分別討論它們的介紹、工作原理、範例以及各個事件中的onclick事件。

在 HTML 中移動文字 在 HTML 中移動文字 Sep 04, 2024 pm 04:45 PM

HTML 中的文字移動指南。在這裡我們討論一下marquee標籤如何使用語法和實作範例。

See all articles