(本文中 dataModel = dm = 數據容器, gv = graphView = g2d = 2D 視圖)javascript
使用 vue-cli 生成項目。生成注意如下幾個問題html
1. 建議手動配置 Manually select featuresvue
2. 勾選 Router java
3. 配置設置 Indedicated config filesnode
4. 項目初始化完成後增長 vue.config.js 並配置vue-cli
module.exports = { devServer: { port: 12580 } }
vue 啓動項目默認在 8080 端口,容易衝突因此咱們把這個端口設置走npm
5. .eslintrc.js 增長配置瀏覽器
globals: { ht: true }
至此初始化項目完成。app
1. 引入 核心庫ide
將 libs 文件夾拷貝進 public 目錄下
2. 引入資源庫(若是是基礎項目可忽略這一步)
將 storage 文件夾拷貝進 public 目錄下
3. index.html 插入以下代碼
<script> window.htconfig = { Default: { convertURL: function(url){ // console.log(url); return 'storage/' + url; } } } </script> <script src='libs/core/ht.js'></script> <script src='libs/plugin/ht-modeling.js'></script> <script src='libs/plugin/ht-obj.js'></script> <script src='libs/plugin/ht-edgetype.js'></script> <script src='libs/plugin/ht-form.js'></script> <script src="libs/plugin/ht-vector.js"></script>
至此項目搭建完成,可在命令行 npm run serve 運行起來項目,在瀏覽器訪問 localhost:12580 (前面手動配置的端口) 訪問項目
在 views/Home.vue 下增長 created 生命週期,新建 ht.Node 節點並 console.log 出來,看控制檯是否有正確輸出。
<template>
<div class="home">
<img alt="Vue logo" src="../assets/logo.png">
<HelloWorld msg="Welcome to Your Vue.js App"/>
</div>
</template>
<script>
// @ is an alias to /src
import HelloWorld from '@/components/HelloWorld.vue' export default { name: 'home', components: { HelloWorld }, created: function() { console.log(ht); let n = new ht.Node(); console.log(n); } } </script>
瀏覽器打開控制檯若是正確輸出 Node 節點表示 ht 引入成功
如上輸出 Node 可進行下一步。不然檢查前面流程,好比 eslint 配置,或者 script 導入是否成功
完成至上一步後,接下來就須要把一個 ht 的 graphView 插入到 vue 組件裏,咱們在不一樣生命週期作不一樣工做,在 created 的時候建立 gv 和 dm,在 mounted 的時候把 gv 掛載到組件裏, 並新建一個圖元
1. 修改 template 以下
<template> <div class='home ht-view'> </div> </template>
2. 增長 style 參考以下
<style scoped> .ht-view { text-align: left; position: relative; width: 800px; height: 500px; margin: auto; border: 1px solid #ccc; } </style>
3. 按上述計劃修改 script,參考以下
export default { name: 'home', components: { // HelloWorld }, created: function() { this.gv = new ht.graph.GraphView(); this.dm = this.gv.dm(); }, mounted: function() { this.gv.addToDOM(this.$el); let node = new ht.Node(); this.dm.add(node); } } </script>
保存後,瀏覽器會熱更新,會獲得以下圖界面
如圖左上方出現 1/4 個小電腦,此時一個 ht-vue 最簡單的項目就搭建完成了。咱們能夠在視圖中間的區域隨意操做並修改 Home.vue 代碼來刷新界面看效果。
關於資源:在 public 下創建 storage 文件夾,而後放入一張圖片,好比我放入一張 ht.png
而後稍修改下代碼:
mounted: function() { this.gv.addToDOM(this.$el); let node = new ht.Node(); node.p(400, 250); node.setImage('ht.png'); this.dm.add(node); }
刷新視圖,實現第一個加載圖片並展現的 HT demo。
上述實現了第一個小拓撲,可是直接修改 Home 組件達到目的,若是咱們頁面裏面有不少拓撲,咱們會考慮複用這個組件,下面開始可複用組件的一點介紹。
HT 的每個做視圖展現的組件都會有一個 dataModel 來管理數據,這個 dm 是能夠複用的因此咱們但願把他做爲獨立的載體而不是組件獨享,這個咱們在教程下一步會介紹到爲何。
基於此考慮 vue 組件的 dataModel 考慮由父節點管理,子節點 dataModel 借用 props 傳遞使用,可是帶來一個問題是父節點若是沒有傳入 dm,子節點應該有一個默認的 dataModel 做使用,那麼咱們應該給 props 賦個默認值,這個值應該是子節點 graphView 默認的 dataModel。因而咱們會想到給 vue 子組件 data 包裹一個 graphView,可是不幸的是 vue 組件 props 構建先於 data,因此致使沒法給 props/dm 賦默認值,因此咱們選擇在 vue 生命週期 beforeCreate 來手動創建 graphView 節點。
於上考慮咱們在 components 下面建立一個 HT2D.vue 的組件,代碼參考以下
<template>
<div>
<div ref="htview"></div>
</div>
</template>
<script> export default { name: 'HT2D', props: { dm: { default: function() { return this.g2d.dm(); } } }, data: function() { return { } }, watch: { dm: function() { this.g2d.dm(this.dm); this.g2d.fitContent(); } }, beforeCreate: function() { this.g2d = new ht.graph.GraphView(); }, beforeMount: function() { this.g2d.dm(this.dm); }, mounted: function() { this.g2d.addToDOM(this.$refs.htview); this.g2d.fitContent(); } } </script>
而後在 views 下建一個 Embeded2D.vue 的組件使用這個 HT2D.vue,並傳入一個 dm 來構建視圖,能夠在組件生命週期 created 加入一段測試代碼,參考以下
<template>
<div>
<HT2D class='ht-view' :dm='dm' ref='g2d'></HT2D>
</div>
</template>
<script> import HT2D from '@/components/HT2D' export default { name: 'Embeded2D', components: { HT2D, }, data: function() { return { dm: new ht.DataModel(), } }, created: function() { let node = new ht.Node(); node.setImage('group_image'); this.dm.add(node); }, mounted: function() { } } </script> <style> .ht-view { text-align: left; position: relative; width: 800px; height: 500px; margin: auto; border: 1px solid #ccc; } </style>
接下來修改下路由讓這個組件展現出來,先修改下 router/index.js 的 routes 參考以下
const routes = [ { path: '/', name: 'home', component: Home }, { path: '/2d', name: '2d', component: () => import( '../views/Embeded2D.vue') }, ]
而後在修改下 APP.vue 的 route-link template 參考以下
<template> <div id="app"> <div id="nav"> <router-link to="/">Home</router-link> | <router-link to="/2d">2D</router-link> </div> <keep-alive> <router-view /> </keep-alive> </div> </template>
刷新瀏覽器並點擊路由導航 2D,會出現以下視圖,表明第一個可複用 2D 組件建立成功。
這個案例中是 Embeded2D.vue 嵌套 HT2D 組件,而後以 props 的方式傳入 dm 參數,那麼能夠在父組件中直接修改 dm 來改變視圖展現,好比在 Embeded.vue 生命週期 mounted 作個小測試,參考以下
mounted: function() { setTimeout(() => { let dm = new ht.DataModel(); let node = new ht.Node(); node.p(100, 100); dm.add(node); node = new ht.Node(); node.p(200, 200); dm.add(node); this.dm = dm; }, 3000); }
在 mounted 裏面過 3S 從新聲明一個 dm 並複製給 dm 對象。視圖會同步刷新,這裏注意下箭頭函數和 vue function this 指向的問題,如上是沒有問題的。
注意下 APP.vue 有給 router-view 加上 keep-alive 屬性,這樣在路由切換的時候不會直接銷燬老的組件,可把 keep-alive 屬性去掉,而後在 Home 和 2D 之間切換並操做下 2D 視圖,好比縮放平移,而後再切換回來就能發現問題所在。到此一個基礎 HT 案例和可複用 HT 組件已經介紹完。
上述咱們一直強調 gv 的複用,並獨立出來 dm,由於 HT 數據呈現的視圖能夠共用一個 dataModel,以一個 ListView 爲例介紹
ListView 介紹 (http://www.hightopo.com/guide/guide/core/listview/ht-listview-guide.html)
首先像 HT2D 同樣咱們在 components 下面建一個 HTList.vue 的文件,參考以下,注意去掉 fitContent 等方法,這些是 2D 視圖專用的
<template> <div> <div ref="htview"></div> </div> </template> <script> export default { name: 'HTList', props: { dm: { default: function() { return this.list.dm(); } } }, data: function() { return { } }, watch: { dm: function() { this.list.dm(this.dm); } }, beforeCreate: function() { this.list = new ht.widget.ListView(); }, beforeMount: function() { this.list.dm(this.dm); }, mounted: function() { this.list.addToDOM(this.$refs.htview); }, } </script>
而後在 views 下面建立一個 MultiHTView.vue 的組件使用 HT2D 和 HTList,並傳入相同的 dm,參考以下
<template> <div class = 'ht-view'> <HT2D class='ht-2d' :dm='dm'></HT2D> <HTList class='ht-list' :dm='dm'></HTList> </div> </template> <script> import HT2D from '@/components/HT2D' import HTList from '@/components/HTList' export default { name: 'Embeded2D', components: { HT2D, HTList, }, data: function() { return { dm: new ht.DataModel(), } }, created: function() { let node = new ht.Node(); node.setImage('group_image'); node.setName("I'm group"); this.dm.add(node); }, } </script> <style scoped> .ht-view, .ht-2d, .ht-list { position: relative; box-sizing: border-box; } .ht-view { text-align: left; width: 810px; margin: auto; } .ht-2d, .ht-list{ width: 400px; height: 300px; float: left; margin-right: 10px; border: 1px solid #ccc; } .ht-list{ margin-right: 0; } </style>
最後修改下路由,router/index.js 參考增長以下路由
{ path: '/multi', name: 'multi', component: () => import( '../views/MultiHTView.vue') }
app 增長以下 router-link
<router-link to="/multi">Multi</router-link>
而後刷新頁面進入 Multi 出現以下則成功
這裏 HT2D 和 HTList 都是 MultiHTView 組件傳入的同一個 dm,可參考前面方法對 dm 作一段測試,MultiHTView 增長一段生命週期 mounted 測試
mounted: function() { setTimeout(() => { var dm = new ht.DataModel(); let node = new ht.Node(); node.p(100, 100); node.setName("I'm n1 node"); dm.add(node); node = new ht.Node(); node.p(200, 200); node.setName("I'm n2 node"); dm.add(node); this.dm = dm; }, 3000); }
至此一個基本的 HT 2D 視圖 + HT 2D 視圖組件構建 + HT 多組件嵌套 vue 案例介紹基本完成。
序列化場景
原文出處:https://www.cnblogs.com/htdaydayup/p/11971226.html