主頁 > 軟體設計 > cytoscape.js進階篇

cytoscape.js進階篇

2020-09-30 11:36:50 軟體設計

cytoscape.js進階篇

  • 系列文章
  • 前言
  • cytoscape依賴參考
  • cytoscape擴展依賴參考
  • 升級Cytoscape組件
    • 擴展Cytoscape組件模板容器
    • 擴展Cytoscape組件方法
    • 完善Cytoscape組件高亮樣式
    • 完善Cytoscape組件中圓形選單中的高亮鄰居指令
    • Cytoscape組件中完善內容方法詳解
      • 放大縮小
      • 合適大小
      • 重繪布局
      • 高亮鄰居
    • 完整的Cytoscape組件代碼
  • Cytoscape組件的使用
    • 創建Cytoscape測驗模板
    • Cytoscape組件參考
    • Cytoscape測驗模板方法
    • 完整的Cytoscape測驗模板代碼
  • 完整截圖
  • 注意事項

很久之前寫過一篇cytoscape.js基礎篇, 地址是https://blog.csdn.net/dahaiaixiaohai/article/details/89669526. 可以適當的參考一下其內容.

很久之前寫過一篇cytoscape.js初級篇, 地址是https://blog.csdn.net/dahaiaixiaohai/article/details/108862390. 可以適當的參考一下其內容.

很久之前寫過一篇cytoscape.js Cxtmenu圓形選單篇, 地址是https://blog.csdn.net/dahaiaixiaohai/article/details/108867486. 可以適當的參考一下其內容.

本篇章, 正式使用cytoscape.js實作一些簡單的節點操作, 并加上左上角的操作工具列. 所提供的工具列功能有: 放大, 縮小, 自由布局, 方形布局, 圓形布局, 高亮鄰居. 實作這些功能后, 開發者可以自由實作其他功能. 擴展功能將會在后期繼續更新發布.

在這里插入圖片描述
在這里插入圖片描述

本文對應的源代碼寄托于github, cytoscape.js進階篇 , 在此感謝github提供的服務.

系列文章

  • cytoscape.js基礎篇CSDN博客
  • cytoscape.js初級篇CSDN博客
  • cytoscape.js初級篇github源代碼
  • cytoscape.js Cxtmenu圓形選單篇
  • cytoscape.js Cxtmenu圓形選單篇github源代碼
  • cytoscape.js進階篇
  • cytoscape.js進階篇github源代碼
  • cytoscape.js專案篇
  • cytoscape.js專案篇github源代碼

前言

本實體通過Vue開發測驗, 其他開發方式可以參考本篇章, 適當的修改后使用.

cytoscape依賴參考

  • npm : npm install cytoscape --save
  • bower : bower install cytoscape
  • jspm : jspm install npm:cytoscape

cytoscape擴展依賴參考

  • 本篇章主要依賴cytoscape-avsdf, cytoscape-cola, cytoscape-cose-bilkent組件. 所以務必保證該些組件安裝成功.

npm 布局及擴展依賴參考

# Cytoscape.js的圓形可滑動背景關系擴展選單
npm install cytoscape-cxtmenu --save
# Cytoscape.js的布局方式
npm install cytoscape-avsdf --save
npm install cytoscape-cola --save
npm install cytoscape-cose-bilkent --save

升級Cytoscape組件

  • 本篇章是在 cytoscape.js Cxtmenu圓形選單篇 的基礎上繼續完善實作的, 沒有 cytoscape.js Cxtmenu圓形選單篇 準備作業或后面代碼看起來有些吃力的小伙伴們可以先看一下cytoscape.js Cxtmenu圓形選單篇, cytoscape.js初級篇 , 甚至可以先看一下 cytoscape.js基礎篇 .

擴展Cytoscape組件模板容器

  • 擴展Cytoscape模板容器后的樣式, 在左上角會多出一組工具列. 該工具列是由組件本身提供, 不需要參考者單獨開發工具列.
<style scoped>
  .tools {
    display: inline-block;
    height: 45px;
    width: 45px;
    vertical-align: middle;
  }

  .center-center {
    height: 100%;
    display: flex;
    align-items: center;
    align-content: center;
    justify-items: center;
    justify-content: center;
  }
</style>
<template>
  <div style="position: relative; height: 100%; width: 100%; z-index: 0;">
    <div id="cytoscape_id" style="height: 100%; width: 100%; z-index: 1;"></div>
    <div id="cytoolbar_id" style="position: absolute; left: 5pt; top: 5pt; z-index: 2; background-color: rgba(249, 249, 249, 0.9);">
      <div class="tools">
        <div class="center-center">
          <Icon style="font-size: 32px; cursor: pointer;" title="放大" type="ios-add-circle-outline" @click="magnifying()"/>
        </div>
      </div>
      <div class="tools">
        <div class="center-center">
          <Icon style="font-size: 32px; cursor: pointer;" title="縮小" type="ios-remove-circle-outline" @click="contractible()"/>
        </div>
      </div>
      <div class="tools">
        <div class="center-center">
          <Icon style="font-size: 32px; cursor: pointer;" title="合適大小" type="ios-resize" @click="resize()"/>
        </div>
      </div>
      <div class="tools">
        <div class="center-center">
          <Icon style="font-size: 32px; cursor: pointer;" title="高亮鄰居" type="ios-color-wand-outline" @click="highlight()"/>
        </div>
      </div>
      <div class="tools">
        <div class="center-center">
          <Icon style="font-size: 32px; cursor: pointer;" title="重繪布局" type="ios-sync" @click="refresh({name: 'cola'})"/>
        </div>
      </div>
      <div class="tools">
        <div class="center-center">
          <Icon style="font-size: 32px; cursor: pointer;" title="網格布局" type="ios-apps-outline" @click="refresh({name: 'grid'})"/>
        </div>
      </div>
      <div class="tools">
        <div class="center-center">
          <Icon style="font-size: 32px; cursor: pointer;" title="環形布局" type="ios-globe-outline" @click="refresh({name: 'circle'})"/>
        </div>
      </div>
    </div>
  </div>
</template>

擴展Cytoscape組件方法

<script>
  export default {
    methods: {
      // ......
      /***************************工具列************************/
      /**
       * 縮放大小.
       * @param zoom 增減幅度.
       */
      zoom(zoom) {
        /** 獲取已選擇內容 */
        let selectedEles = this.$cy.elements('node:selected');
        /** 獲取已選擇內容中得第一個, 沒有選擇為null */
        let selectedEle = selectedEles && selectedEles.length ? selectedEles[0] : null;
        /** 獲取畫布偏移位置 */
        let pan = this.$cy.pan();
        /** 計算原點坐標 */
        let [x, y] = selectedEle ? [selectedEle.position('x'), selectedEle.position('y')] : [pan.x, pan.y];
        let level = this.$cy.zoom() + zoom;
        (level > this.$cy.maxZoom) && (level = this.$cy.maxZoom);
        (level < this.$cy.minZoom) && (level = this.$cy.minZoom);
        this.$cy.zoom({level: level, renderedPosition: {x: x, y: y}});
      },
      /** 放大 */
      magnifying() {
        this.zoom(0.3);
      },
      /** 縮小 */
      contractible() {
        this.zoom(-0.3);
      },
      /** 合適大小 */
      resize() {
        this.$cy.fit();
      },
      /**
       * 高亮.
       * @param ele 某元素ID
       */
      lightOn(ele) {
        this.$cy.startBatch();
        this.$cy.batch(() => {
          this.$cy.elements().addClass("light-off"); //*添加樣式*/
          let elements = ((Array.isArray ? Array.isArray(ele) : null != ele && ele instanceof Array) ? ele : [ele]);
          elements.forEach(__ => {
            this.$cy.getElementById(__).removeClass("light-off");
            this.$cy.getElementById(__).neighborhood().removeClass("light-off");
          });
        });
        this.$cy.once('click', () => this.lightOff());
        this.$cy.endBatch();
      },
      /**
       * 取消高亮.
       */
      lightOff() {
        this.$cy.startBatch();
        this.$cy.batch(() => this.$cy.elements().removeClass("light-off") /*移除樣式*/);
        this.$cy.endBatch();
      },
      /** 高亮鄰居 */
      highlight() {
        /** 獲取已選擇內容 */
        let selectedEles = this.$cy.elements('node:selected');
        /** 獲取已選擇內容中得第一個, 沒有選擇為null */
        let selectedEle = selectedEles && selectedEles.length ? selectedEles[0] : null;
        (selectedEle) && (this.lightOn(selectedEle.id()));
      },
      /**
       * 重繪布局.
       * name取值范圍:
       * ['grid', 'circle', 'cola', 'avsdf', 'cose-bilkent', ]
       * @param {name = 'cola......', randomize = true | false, animate = true | false}
       */
      refresh({name = 'cola', randomize = false, animate = true} = {}) {
        this.$cy.layout({name: name, randomize: randomize, animate: animate,}).run();
      },
      /***************************工具列************************/
    },
  }
</script>

完善Cytoscape組件高亮樣式

  • 高亮樣式核心樣式配置this.$cy.style().selector('.light-off').style({'opacity': '0.1',}).
// 通用的樣式
this.$cy.style()
  /*未選擇節點樣式*/
  .selector('node')
  .style({'label': 'data(name)', 'font-size': '10pt', 'width': '8pt', 'height': '8pt'})
  /*已選擇節點樣式*/
  .selector('node:selected')
  .style({'border-color': '#c84e40', 'border-width': "1px",})
  /*未選擇節點樣式*/
  .selector('edge')
  .style({......})
  /*已選擇節點樣式*/
  .selector('edge:selected')
  .style({......})
  /*高亮樣式*/
  .selector('.light-off')
  .style({'opacity': '0.1',})
  ;

完善Cytoscape組件中圓形選單中的高亮鄰居指令

{
  // fillColor: 'rgba(200, 200, 200, 0.75)', // optional: custom background color for item
  content: '高亮鄰居', // html/text content to be displayed in the menu
  // contentStyle: {}, // css key:value pairs to set the command's css in js if you want
  select: (ele) => this.lightOn([ele.id()]),  // a function to execute when the command is selected
  enabled: true, // whether the command is selectable
},

Cytoscape組件中完善內容方法詳解

放大縮小

      /**
       * 縮放大小.
       * @param zoom 增減幅度.
       */
      zoom(zoom) {
        /** 獲取已選擇內容 */
        let selectedEles = this.$cy.elements('node:selected');
        /** 獲取已選擇內容中得第一個, 沒有選擇為null */
        let selectedEle = selectedEles && selectedEles.length ? selectedEles[0] : null;
        /** 獲取畫布偏移位置 */
        let pan = this.$cy.pan();
        /** 計算原點坐標 */
        let [x, y] = selectedEle ? [selectedEle.position('x'), selectedEle.position('y')] : [pan.x, pan.y];
        let level = this.$cy.zoom() + zoom;
        (level > this.$cy.maxZoom) && (level = this.$cy.maxZoom);
        (level < this.$cy.minZoom) && (level = this.$cy.minZoom);
        this.$cy.zoom({level: level, renderedPosition: {x: x, y: y}});
      },
      /** 放大 */
      magnifying() {
        this.zoom(0.3);
      },
      /** 縮小 */
      contractible() {
        this.zoom(-0.3);
      },

合適大小

      /** 合適大小 */
      resize() {
        this.$cy.fit();
      },

重繪布局

  • 依賴上述依賴組件: cytoscape-avsdf, cytoscape-cola, cytoscape-cose-bilkent組件.
      /**
       * 重繪布局.
       * name取值范圍:
       * ['grid', 'circle', 'cola', 'avsdf', 'cose-bilkent', ]
       * @param {name = 'cola......', randomize = true | false, animate = true | false}
       */
      refresh({name = 'cola', randomize = false, animate = true} = {}) {
        this.$cy.layout({name: name, randomize: randomize, animate: animate,}).run();
      },

高亮鄰居

  • 高亮的程序: 判斷是否有選擇節點, 多選擇節點, 只取第0個節點.
  • 給所有節點添加高亮樣式, 獲取選擇節點去除高亮樣式.
  • 添加一次性操作onse(‘click’, () => {}), 取消高亮樣式.
  • 異步執行.
      /**
       * 高亮.
       * @param ele 某元素ID
       */
      lightOn(ele) {
        this.$cy.startBatch();
        this.$cy.batch(() => {
          this.$cy.elements().addClass("light-off"); //*添加樣式*/
          let elements = ((Array.isArray ? Array.isArray(ele) : null != ele && ele instanceof Array) ? ele : [ele]);
          elements.forEach(__ => {
            this.$cy.getElementById(__).removeClass("light-off");
            this.$cy.getElementById(__).neighborhood().removeClass("light-off");
          });
        });
        this.$cy.once('click', () => this.lightOff());
        this.$cy.endBatch();
      },
      /**
       * 取消高亮.
       */
      lightOff() {
        this.$cy.startBatch();
        this.$cy.batch(() => this.$cy.elements().removeClass("light-off") /*移除樣式*/);
        this.$cy.endBatch();
      },
      /** 高亮鄰居 */
      highlight() {
        /** 獲取已選擇內容 */
        let selectedEles = this.$cy.elements('node:selected');
        /** 獲取已選擇內容中得第一個, 沒有選擇為null */
        let selectedEle = selectedEles && selectedEles.length ? selectedEles[0] : null;
        (selectedEle) && (this.lightOn(selectedEle.id()));
      },

完整的Cytoscape組件代碼

<style scoped>
  .tools {
    display: inline-block;
    height: 45px;
    width: 45px;
    vertical-align: middle;
  }

  .center-center {
    height: 100%;
    display: flex;
    align-items: center;
    align-content: center;
    justify-items: center;
    justify-content: center;
  }
</style>

<template>
  <div style="position: relative; height: 100%; width: 100%; z-index: 0;">
    <div id="cytoscape_id" style="height: 100%; width: 100%; z-index: 1;"></div>
    <div id="cytoolbar_id" style="position: absolute; left: 5pt; top: 5pt; z-index: 2; background-color: rgba(249, 249, 249, 0.9);">
      <div class="tools">
        <div class="center-center">
          <Icon style="font-size: 32px; cursor: pointer;" title="放大" type="ios-add-circle-outline" @click="magnifying()"/>
        </div>
      </div>
      <div class="tools">
        <div class="center-center">
          <Icon style="font-size: 32px; cursor: pointer;" title="縮小" type="ios-remove-circle-outline" @click="contractible()"/>
        </div>
      </div>
      <div class="tools">
        <div class="center-center">
          <Icon style="font-size: 32px; cursor: pointer;" title="合適大小" type="ios-resize" @click="resize()"/>
        </div>
      </div>
      <div class="tools">
        <div class="center-center">
          <Icon style="font-size: 32px; cursor: pointer;" title="高亮鄰居" type="ios-color-wand-outline" @click="highlight()"/>
        </div>
      </div>
      <div class="tools">
        <div class="center-center">
          <Icon style="font-size: 32px; cursor: pointer;" title="重繪布局" type="ios-sync" @click="refresh({name: 'cola'})"/>
        </div>
      </div>
      <div class="tools">
        <div class="center-center">
          <Icon style="font-size: 32px; cursor: pointer;" title="網格布局" type="ios-apps-outline" @click="refresh({name: 'grid'})"/>
        </div>
      </div>
      <div class="tools">
        <div class="center-center">
          <Icon style="font-size: 32px; cursor: pointer;" title="環形布局" type="ios-globe-outline" @click="refresh({name: 'circle'})"/>
        </div>
      </div>
    </div>
  </div>
</template>

<script>
  import cytoscape from 'cytoscape';
  import cxtmenu from 'cytoscape-cxtmenu';
  import cola from 'cytoscape-cola';
  import avsdf from 'cytoscape-avsdf';
  import coseBilkent from 'cytoscape-cose-bilkent';


  export default {
    name: "CJS",
    beforeCreate() {
      this.$cy && this.$cy.destroyed() && this.$cy.destroy();
      delete this.$cy;
    },
    beforeDestroy() {
      this.$cy && this.$cy.destroyed() && this.$cy.destroy();
      delete this.$cy;
    },
    watch: {},
    props: {},
    mounted() {
      // Cxtmenu圓形選單主要依賴組件
      if (!cytoscape().cxtmenu) {
        cytoscape.use(cxtmenu);
        cytoscape.use(cola);
        cytoscape.use(avsdf);
        cytoscape.use(coseBilkent);
      }

      this.$cy = cytoscape({
        // initial viewport state:
        zoom: 1, // 圖表的初始縮放級別.可以設定options.minZoom和options.maxZoom設定縮放級別的限制.
        pan: {x: 0, y: 0}, // 圖表的初始平移位置.
        // interaction options:
        minZoom: 1e-50, // 圖表縮放級別的最小界限.視口的縮放比例不能小于此縮放級別.
        maxZoom: 1e50, // 圖表縮放級別的最大界限.視口的縮放比例不能大于此縮放級別.
        zoomingEnabled: true, // 是否通過用戶事件和編程方式啟用縮放圖形.
        userZoomingEnabled: true, // 是否允許用戶事件(例如滑鼠滾輪,捏合縮放)縮放圖形.對此縮放的編程更改不受此選項的影響.
        panningEnabled: true, // 是否通過用戶事件和編程方式啟用平移圖形.
        userPanningEnabled: true, // 是否允許用戶事件(例如拖動圖形背景)平移圖形.平移的程式化更改不受此選項的影響.
        boxSelectionEnabled: true, // 是否啟用了框選擇(即拖動框疊加,并將其釋放為選擇).如果啟用,則用戶必須點擊以平移圖表.
        selectionType: 'single', // 一個字串,指示用戶輸入的選擇行為.對于'additive',用戶進行的新選擇將添加到當前所選元素的集合中.對于'single',用戶做出的新選擇成為當前所選元素的整個集合.
        touchTapThreshold: 8, // 非負整數,分別表示用戶在輕擊手勢期間可以在觸摸設備和桌面設備上移動的最大允許距離.這使得用戶更容易點擊.
                              // 這些值具有合理的默認值,因此建議不要更改這些選項,除非您有充分的理由這樣做.大值幾乎肯定會產生不良后果.
        desktopTapThreshold: 4, // 非負整數,分別表示用戶在輕擊手勢期間可以在觸摸設備和桌面設備上移動的最大允許距離.這使得用戶更容易點擊.
                                // 這些值具有合理的默認值,因此建議不要更改這些選項,除非您有充分的理由這樣做.大值幾乎肯定會產生不良后果.
        autolock: false, // 默認情況下是否應鎖定節點(根本不可拖動,如果true覆寫單個節點狀態).
        autoungrabify: false, // 默認情況下節點是否不允許被拾取(用戶不可抓取,如果true覆寫單個節點狀態).
        autounselectify: false, // 默認情況下節點是否允許被選擇(不可變選擇狀態,如果true覆寫單個元素狀態).
        // rendering options:
        headless: false, // true:空運行,不顯示不需要容器容納.false:顯示需要容器容納.
        styleEnabled: true, // 一個布林值,指示是否應用樣式.
        hideEdgesOnViewport: true, // 渲染提示,設定為true在渲染視窗時,不渲染邊.例如,移動某個頂點時或縮放時,邊資訊會被臨時隱藏,移動結束后,邊資訊會被執行一次渲染.由于性能增強,此選項現在基本上沒有實際意義.
        hideLabelsOnViewport: true, // 渲染提示,當設定為true使渲染器在平移和縮放期間使用紋理而不是繪制元素時,使大圖更具回應性.由于性能增強,此選項現在基本上沒有實際意義.
        textureOnViewport: true, // 渲染提示,當設定為true使渲染器在平移和縮放期間使用紋理而不是繪制元素時,使大圖更具回應性.由于性能增強,此選項現在基本上沒有實際意義.
        motionBlur: true, // 渲染提示,設定為true使渲染器使用運動模糊效果使幀之間的過渡看起來更平滑.這可以增加大圖的感知性能.由于性能增強,此選項現在基本上沒有實際意義.
        motionBlurOpacity: 0.2, // 當motionBlur:true,此值控制運動模糊幀的不透明度.值越高,運動模糊效果越明顯.由于性能增強,此選項現在基本上沒有實際意義.
        wheelSensitivity: 0.3, // 縮放時更改滾輪靈敏度.這是一個乘法修飾符.因此,0到1之間的值會降低靈敏度(變焦較慢),而大于1的值會增加靈敏度(變焦更快).
        pixelRatio: 'auto', // 使用手動設定值覆寫螢屏像素比率(1.0建議,如果已設定).這可以通過減少需要渲染的有效區域來提高高密度顯示幕的性能,
                            // 盡管在最近的瀏覽器版本中這是不太必要的.如果要使用硬體的實際像素比,可以設定pixelRatio: 'auto'(默認).
        // DOM容器,決定內容展示的位置,方式一(原生):document.getElementById('xx'),方式二(jQuery):$('#xx')
        container: document.getElementById('cytoscape_id'),
        // 一個指定布局選項的普通物件.
        layout: {name: 'random'},
      });
      // Cxtmenu圓形選單--開始
      this.$cy.cxtmenu({
        menuRadius: 80, // the radius of the circular menu in pixels
        selector: 'node', // elements matching this Cytoscape.js selector will trigger cxtmenus
        commands: (element) => {
          return [
            {
              fillColor: 'rgba(200, 200, 200, 0.75)', // optional: custom background color for item
              content: '<span class="fa fa-flash fa-2x">操作1</span>', // html/text content to be displayed in the menu
              contentStyle: {}, // css key:value pairs to set the command's css in js if you want
              select: function (ele) { // a function to execute when the command is selected
                alert(ele.id()); // `ele` holds the reference to the active element
              },
              enabled: true, // whether the command is selectable
            },
            {
              fillColor: 'rgba(200, 200, 200, 0.75)', // optional: custom background color for item
              content: '<span class="fa fa-flash fa-2x">操作2</span>', // html/text content to be displayed in the menu
              contentStyle: {}, // css key:value pairs to set the command's css in js if you want
              select: function (ele) { // a function to execute when the command is selected
                alert(ele.id()); // `ele` holds the reference to the active element
              },
              enabled: true, // whether the command is selectable
            },
            {
              // fillColor: 'rgba(200, 200, 200, 0.75)', // optional: custom background color for item
              content: '高亮鄰居', // html/text content to be displayed in the menu
              // contentStyle: {}, // css key:value pairs to set the command's css in js if you want
              select: (ele) => this.lightOn([ele.id()]),  // a function to execute when the command is selected
              enabled: true, // whether the command is selectable
            },
            {
              // fillColor: 'rgba(200, 200, 200, 0.75)', // optional: custom background color for item
              content: '禁用', // html/text content to be displayed in the menu
              // contentStyle: {}, // css key:value pairs to set the command's css in js if you want
              select: (ele) => alert(ele.id()),  // a function to execute when the command is selected
              enabled: false, // whether the command is selectable
            }
          ]
        },
        fillColor: 'rgba(0, 0, 0, 0.75)', // 指令默認顏色(the background colour of the menu)
        activeFillColor: 'rgba(1, 105, 217, 0.75)', // 所選指令的顏色(the colour used to indicate the selected command)
        activePadding: 10, // additional size in pixels for the active command
        indicatorSize: 14, // the size in pixels of the pointer to the active command
        separatorWidth: 4, //連續命令之間的空白間隔(以像素為單位)
        spotlightPadding: 10, //元素和聚光燈之間的額外間距(以像素為單位)
        minSpotlightRadius: 10, // the minimum radius in pixels of the spotlight
        maxSpotlightRadius: 14, // the maximum radius in pixels of the spotlight
        openMenuEvents: 'cxttapstart taphold', // space-separated cytoscape events that will open the menu; only `cxttapstart` and/or `taphold` work here
        itemColor: 'white', // 各指令元素內字體顏色
        itemTextShadowColor: 'red', // 各指令元素內字體陰影顏色
        zIndex: 9999, // the z-index of the ui div
        atMouse: true, // draw menu at mouse position
      });
      //Cxtmenu圓形選單--結束
      // 不同節點的樣式
      this.$cy
        .style()
        .selector('.classes-A')
        .style({'background-color': '#FF0000', 'border-color': '#FF0000', 'border-width': "1px",})
        .selector('.classes-B')
        .style({'background-color': '#00FF00', 'border-color': '#00FF00', 'border-width': "1px",})
        .selector('.classes-C')
        .style({'background-color': '#0000FF', 'border-color': '#0000FF', 'border-width': "1px",})
        .selector('.classes-D')
        .style({'background-color': '#E0E0E0', 'border-color': '#E0E0E0', 'border-width': "1px",})
      ;
      // 通用的樣式
      this.$cy.style()
        /*未選擇節點樣式*/
        .selector('node')
        .style({'label': 'data(name)', 'font-size': '10pt', 'width': '8pt', 'height': '8pt'})
        /*已選擇節點樣式*/
        .selector('node:selected')
        .style({'border-color': '#c84e40', 'border-width': "1px",})
        /*未選擇節點樣式*/
        .selector('edge')
        .style({
          'label': 'data(name)',
          'target-arrow-shape': 'triangle-backcurve', /*箭頭樣式*/
          'target-arrow-size': '1px', /*箭頭大小*/
          'target-arrow-color': '#999999', /*箭頭顏色*/
          'curve-style': 'bezier', /*線條樣式曲線*/
          'line-color': '#999999', /*線條顏色*/
          'width': '1px', /*線條寬度*/
          'font-size': '10px', /*標簽字體大小*/
          'color': '#000000', /*標簽字體大小*/
          'text-outline-color': 'white', /*文本輪廓顏色*/
          'text-outline-width': '1px', /*文本輪廓寬度*/
          'text-rotation': 'autorotate', /*標簽方向*/
        })
        /*已選擇節點樣式*/
        .selector('edge:selected')
        .style({
          'color': '#3165fc', /*標簽字體大小*/
          'target-arrow-color': '#61bffc', /*箭頭顏色*/
          'line-color': '#61bffc', /*線條顏色*/
        })
        /*高亮樣式*/
        .selector('.light-off')
        .style({'opacity': '0.1',})
      ;
    },
    data() {
      return {}
    },
    methods: {
      /**
       * eles : Array or Map.
       * node_eg: {group: 'nodes', data: {id: 'nid1', name: 'name1', label: 'l1 l2', others: 'others'}, classes: 'like label', position: {x: 100, y: 100}};
       * edge_eg: {group: 'edges', data: {id: 'eid1', name: 'name1', source: 'A', target: 'B', label: 'l1 l2', others: 'others'}, classes: 'like label', position: {x: 100, y: 100}};
       * node_eg: [
       *   {group: 'nodes', data: {id: 'nid1', name: 'name1', label: 'l1 l2', others: 'others'}, classes: 'like label', position: {x: 100, y: 100}};
       *   {group: 'nodes', data: {id: 'nid2', name: 'name2', label: 'l1 l2', others: 'others'}, classes: 'like label', position: {x: 100, y: 100}};
       * ];
       * edge_eg: [
       *   {group: 'edges', data: {id: 'eid1', name: 'name1', source: 'nid1', target: 'nid2', label: 'l1 l2', others: 'others'}, classes: 'like label', position: {x: 100, y: 100}};
       *   {group: 'edges', data: {id: 'eid2', name: 'name1', source: 'nid2', target: 'nid3', label: 'l1 l2', others: 'others'}, classes: 'like label', position: {x: 100, y: 100}};
       * ];
       * @param eles 元素集合.
       */
      addEles(eles) {
        if (eles) {
          this.$cy.startBatch();
          this.$cy.batch(() => {
            let elements = ((Array.isArray ? Array.isArray(eles) : null != eles && eles instanceof Array) ? eles : [eles]);
            let filterElements = elements.filter(__ => !this.$cy.getElementById(__.data.id).length)
            this.$cy.add(filterElements);
          });
          this.$cy.endBatch();
        }
      },
      /**
       * 洗掉選擇的內容(可能是頂點, 也可能是關系)
       */
      delEles() {
        this.$cy.startBatch();
        this.$cy.batch(() => {
          let selectedEles = this.$cy.elements(':selected');
          // 未選擇不進行操作
          if (!selectedEles || 1 > selectedEles.length) {
            return false;
          }
          selectedEles.remove();
        });
        this.$cy.endBatch();
      },
      /***************************工具列************************/
      /**
       * 縮放大小.
       * @param zoom 增減幅度.
       */
      zoom(zoom) {
        /** 獲取已選擇內容 */
        let selectedEles = this.$cy.elements('node:selected');
        /** 獲取已選擇內容中得第一個, 沒有選擇為null */
        let selectedEle = selectedEles && selectedEles.length ? selectedEles[0] : null;
        /** 獲取畫布偏移位置 */
        let pan = this.$cy.pan();
        /** 計算原點坐標 */
        let [x, y] = selectedEle ? [selectedEle.position('x'), selectedEle.position('y')] : [pan.x, pan.y];
        let level = this.$cy.zoom() + zoom;
        (level > this.$cy.maxZoom) && (level = this.$cy.maxZoom);
        (level < this.$cy.minZoom) && (level = this.$cy.minZoom);
        this.$cy.zoom({level: level, renderedPosition: {x: x, y: y}});
      },
      /** 放大 */
      magnifying() {
        this.zoom(0.3);
      },
      /** 縮小 */
      contractible() {
        this.zoom(-0.3);
      },
      /** 合適大小 */
      resize() {
        this.$cy.fit();
      },
      /**
       * 高亮.
       * @param ele 某元素ID
       */
      lightOn(ele) {
        this.$cy.startBatch();
        this.$cy.batch(() => {
          this.$cy.elements().addClass("light-off"); //*添加樣式*/
          let elements = ((Array.isArray ? Array.isArray(ele) : null != ele && ele instanceof Array) ? ele : [ele]);
          elements.forEach(__ => {
            this.$cy.getElementById(__).removeClass("light-off");
            this.$cy.getElementById(__).neighborhood().removeClass("light-off");
          });
        });
        this.$cy.once('click', () => this.lightOff());
        this.$cy.endBatch();
      },
      /**
       * 取消高亮.
       */
      lightOff() {
        this.$cy.startBatch();
        this.$cy.batch(() => this.$cy.elements().removeClass("light-off") /*移除樣式*/);
        this.$cy.endBatch();
      },
      /** 高亮鄰居 */
      highlight() {
        /** 獲取已選擇內容 */
        let selectedEles = this.$cy.elements('node:selected');
        /** 獲取已選擇內容中得第一個, 沒有選擇為null */
        let selectedEle = selectedEles && selectedEles.length ? selectedEles[0] : null;
        (selectedEle) && (this.lightOn(selectedEle.id()));
      },
      /**
       * 重繪布局.
       * name取值范圍:
       * ['grid', 'circle', 'cola', 'avsdf', 'cose-bilkent', ]
       * @param {name = 'cola......', randomize = true | false, animate = true | false}
       */
      refresh({name = 'cola', randomize = false, animate = true} = {}) {
        this.$cy.layout({name: name, randomize: randomize, animate: animate,}).run();
      },
      /***************************工具列************************/
    },
  }
</script>

Cytoscape組件的使用

  • Cytoscape測驗模板與 cytoscape.js初級篇 的Cytoscape測驗模板幾乎完全一致.
  • 唯一差距在: 將添加, 洗掉按鍵下移, 給工具列騰出位置.

創建Cytoscape測驗模板

<template>
  <div style="height: 100%; width: 100%; border: 1px solid #00F;">
    <div style="position: fixed; left: 20pt; top: 50pt; z-index: 99999;">
      <Button size="small" @click="addEles">添加</Button>
      <Button size="small" @click="delEles">洗掉</Button>
    </div>
    <CJS ref="ref_CJS"></CJS>
  </div>
</template>
  • 樣式如下
    測驗組件樣式

Cytoscape組件參考

  • 測驗模板參考Cytoscape組件, 并命名為CJS.
<script>
  import CJS from './components/cjs';
  
  export default {
    name: "Test",
    components: {CJS,},
    // ......
  }
</script>

Cytoscape測驗模板方法

methods: {
  addEles() {
	this.$refs['ref_CJS'].addEles([
	  {group: 'nodes', data: {'id': '魯A123456', 'name': '魯A123456',}, classes: 'classes-A', position: {x: 200, y: 50}},
	  {group: 'nodes', data: {'id': '魯B123456', 'name': '魯B123456',}, classes: 'classes-A', position: {x: 500, y: 50}},
	  {group: 'nodes', data: {'id': '魯C123456', 'name': '魯C123456',}, classes: 'classes-A', display: 'hide', position: {x: 200, y: 150}},
	  {group: 'nodes', data: {'id': '魯D123456', 'name': '魯D123456',}, classes: 'classes-A', position: {x: 500, y: 150}},
	  {group: 'nodes', data: {'id': '小王', 'name': '小王',}, classes: 'classes-B', position: {x: 100, y: 100}},
	  {group: 'nodes', data: {'id': '小趙', 'name': '小趙',}, classes: 'classes-B', position: {x: 400, y: 100}},
	  {group: 'nodes', data: {'id': '川川某公司', 'name': '川川某公司',}, classes: 'classes-C', display: 'hide', position: {x: 300, y: 100}},
	  {group: 'nodes', data: {'id': '京京某單位', 'name': '京京某單位',}, classes: 'classes-D', position: {x: 300, y: 200}},
	  {group: 'edges', data: {id: 'e0', name: '擁有', source: '小王', target: '魯A123456'}},
	  {group: 'edges', data: {id: 'e1', name: '擁有', source: '小趙', target: '魯B123456'}},
	  {group: 'edges', data: {id: 'e2', name: '擁有', source: '小王', target: '魯C123456'}},
	  {group: 'edges', data: {id: 'e3', name: '擁有', source: '小趙', target: '魯D123456'}},
	  {group: 'edges', data: {id: 'e4', name: '就職', source: '小王', target: '川川某公司'}},
	  {group: 'edges', data: {id: 'e5', name: '就職', source: '小趙', target: '川川某公司'}},
	  {group: 'edges', data: {id: 'e6', name: '租用', source: '川川某公司', target: '魯A123456'}},
	  {group: 'edges', data: {id: 'e7', name: '租用', source: '川川某公司', target: '魯B123456'}}
	]);
  },
  delEles() {
	this.$refs['ref_CJS'].delEles();
  },
},

完整的Cytoscape測驗模板代碼

<template>
  <div style="height: 100%; width: 100%; border: 1px solid #00F;">
    <div style="position: fixed; left: 20pt; top: 20pt; z-index: 99999;">
      <Button size="small" @click="addEles">添加</Button>
      <Button size="small" @click="delEles">洗掉</Button>
    </div>
    <CJS ref="ref_CJS"></CJS>
  </div>
</template>

<script>
  import CJS from './components/cjs';

  export default {
    name: "Test",
    components: {CJS,},
    watch: {},
    mounted() {
      this.addEles();
    },
    methods: {
      addEles() {
        this.$refs['ref_CJS'].addEles([
          {group: 'nodes', data: {'id': '魯A123456', 'name': '魯A123456',}, classes: 'classes-A', position: {x: 200, y: 50}},
          {group: 'nodes', data: {'id': '魯B123456', 'name': '魯B123456',}, classes: 'classes-A', position: {x: 500, y: 50}},
          {group: 'nodes', data: {'id': '魯C123456', 'name': '魯C123456',}, classes: 'classes-A', display: 'hide', position: {x: 200, y: 150}},
          {group: 'nodes', data: {'id': '魯D123456', 'name': '魯D123456',}, classes: 'classes-A', position: {x: 500, y: 150}},
          {group: 'nodes', data: {'id': '小王', 'name': '小王',}, classes: 'classes-B', position: {x: 100, y: 100}},
          {group: 'nodes', data: {'id': '小趙', 'name': '小趙',}, classes: 'classes-B', position: {x: 400, y: 100}},
          {group: 'nodes', data: {'id': '川川某公司', 'name': '川川某公司',}, classes: 'classes-C', display: 'hide', position: {x: 300, y: 100}},
          {group: 'nodes', data: {'id': '京京某單位', 'name': '京京某單位',}, classes: 'classes-D', position: {x: 300, y: 200}},
          {group: 'edges', data: {id: 'e0', name: '擁有', source: '小王', target: '魯A123456'}},
          {group: 'edges', data: {id: 'e1', name: '擁有', source: '小趙', target: '魯B123456'}},
          {group: 'edges', data: {id: 'e2', name: '擁有', source: '小王', target: '魯C123456'}},
          {group: 'edges', data: {id: 'e3', name: '擁有', source: '小趙', target: '魯D123456'}},
          {group: 'edges', data: {id: 'e4', name: '就職', source: '小王', target: '川川某公司'}},
          {group: 'edges', data: {id: 'e5', name: '就職', source: '小趙', target: '川川某公司'}},
          {group: 'edges', data: {id: 'e6', name: '租用', source: '川川某公司', target: '魯A123456'}},
          {group: 'edges', data: {id: 'e7', name: '租用', source: '川川某公司', target: '魯B123456'}}
        ]);
      },
      delEles() {
        this.$refs['ref_CJS'].delEles();
      },
    },
  }
</script>

<style scoped>
</style>

完整截圖

在這里插入圖片描述

注意事項

由于該專案是一系列文章, 逐步完善下來的, 所以在中間章節中, 并沒有詳細講解之前的內容, 如果需要了解之前的內容, 可以通過系列文章預覽閱讀早期內容或更新的內容.

本文到此結束, 更完善的內容還在后面, 敬請期待.

轉載請註明出處,本文鏈接:https://www.uj5u.com/ruanti/141906.html

標籤:其他

上一篇:JavaScript基礎速通

下一篇:Vue條件和回圈陳述句

標籤雲
其他(157675) Python(38076) JavaScript(25376) Java(17977) C(15215) 區塊鏈(8255) C#(7972) AI(7469) 爪哇(7425) MySQL(7132) html(6777) 基礎類(6313) sql(6102) 熊猫(6058) PHP(5869) 数组(5741) R(5409) Linux(5327) 反应(5209) 腳本語言(PerlPython)(5129) 非技術區(4971) Android(4554) 数据框(4311) css(4259) 节点.js(4032) C語言(3288) json(3245) 列表(3129) 扑(3119) C++語言(3117) 安卓(2998) 打字稿(2995) VBA(2789) Java相關(2746) 疑難問題(2699) 细绳(2522) 單片機工控(2479) iOS(2429) ASP.NET(2402) MongoDB(2323) 麻木的(2285) 正则表达式(2254) 字典(2211) 循环(2198) 迅速(2185) 擅长(2169) 镖(2155) 功能(1967) .NET技术(1958) Web開發(1951) python-3.x(1918) HtmlCss(1915) 弹簧靴(1913) C++(1909) xml(1889) PostgreSQL(1872) .NETCore(1853) 谷歌表格(1846) Unity3D(1843) for循环(1842)

熱門瀏覽
  • 面試突擊第一季,第二季,第三季

    第一季必考 https://www.bilibili.com/video/BV1FE411y79Y?from=search&seid=15921726601957489746 第二季分布式 https://www.bilibili.com/video/BV13f4y127ee/?spm_id_fro ......

    uj5u.com 2020-09-10 05:35:24 more
  • 第三單元作業總結

    1.前言 這應該是本學期最后一次寫作業總結了吧。總體來說,對作業的節奏也差不多掌握了,作業做起來的效率也更高了。雖然和之前的作業一樣,作業中都要用到新的知識,但是相比之前,更加懂得了如何利用工具以及資料。雖然之間卡過殼,但總體而言,這幾次作業還算完成的比較好。 2.作業程序總結 相比前兩個單元,此單 ......

    uj5u.com 2020-09-10 05:35:41 more
  • 北航OO(2020)第四單元博客作業暨課程總結博客

    北航OO(2020)第四單元博客作業暨課程總結博客 本單元作業的架構設計 在本單元中,由于UML圖具有比較清晰的樹形結構,因此我對其中需要進行查詢操作的元素進行了包裝,在樹的父節點中存盤所有孩子的參考。考慮到性能問題,我采用了快取機制,一次查詢后盡可能快取已經遍歷過的資訊,以減少遍歷次數。 本單元我 ......

    uj5u.com 2020-09-10 05:35:48 more
  • BUAA_OO_第四單元

    一、UML決議器設計 ? 先看下題目:第四單元實作一個基于JDK 8帶有效性檢查的UML(Unified Modeling Language)類圖,順序圖,狀態圖分析器 MyUmlInteraction,實際上我們要建立一個有向圖模型,UML中的物件(元素)可能與同級元素連接,也可與低級元素相連形成 ......

    uj5u.com 2020-09-10 05:35:54 more
  • 6.1邏輯運算子

    邏輯運算子 1. && 短路與 運算式1 && 運算式2 01.運算式1為true并且運算式2也為true 整體回傳為true 02.運算式1為false,將不會執行運算式2 整體回傳為false 03.只要有一個運算式為false 整體回傳為false 2. || 短路或 運算式1 || 運算式2 ......

    uj5u.com 2020-09-10 05:35:56 more
  • BUAAOO 第四單元 & 課程總結

    1. 第四單元:StarUml檔案決議 本單元采用了圖模型決議UML。 UML檔案可以抽象為圖、子圖、邊的邏輯結構。 在實作中,圖的節點包括類、介面、屬性,子圖包括狀態圖、順序圖等。 采用了三次遍歷UML元素的方法建圖,第一遍遍歷建點,第二、三次遍歷設定屬性、連邊,實作圖物件的初始化。這里借鑒了一些 ......

    uj5u.com 2020-09-10 05:36:06 more
  • 談談我對C# 多型的理解

    面向物件三要素:封裝、繼承、多型。 封裝和繼承,這兩個比較好理解,但要理解多型的話,可就稍微有點難度了。今天,我們就來講講多型的理解。 我們應該經常會看到面試題目:請談談對多型的理解。 其實呢,多型非常簡單,就一句話:呼叫同一種方法產生了不同的結果。 具體實作方式有三種。 一、多載 多載很簡單。 p ......

    uj5u.com 2020-09-10 05:36:09 more
  • Python 資料驅動工具:DDT

    背景 python 的unittest 沒有自帶資料驅動功能。 所以如果使用unittest,同時又想使用資料驅動,那么就可以使用DDT來完成。 DDT是 “Data-Driven Tests”的縮寫。 資料:http://ddt.readthedocs.io/en/latest/ 使用方法 dd. ......

    uj5u.com 2020-09-10 05:36:13 more
  • Python里面的xlrd模塊詳解

    那我就一下面積個問題對xlrd模塊進行學習一下: 1.什么是xlrd模塊? 2.為什么使用xlrd模塊? 3.怎樣使用xlrd模塊? 1.什么是xlrd模塊? ?python操作excel主要用到xlrd和xlwt這兩個庫,即xlrd是讀excel,xlwt是寫excel的庫。 今天就先來說一下xl ......

    uj5u.com 2020-09-10 05:36:28 more
  • 當我們創建HashMap時,底層到底做了什么?

    jdk1.7中的底層實作程序(底層基于陣列+鏈表) 在我們new HashMap()時,底層創建了默認長度為16的一維陣列Entry[ ] table。當我們呼叫map.put(key1,value1)方法向HashMap里添加資料的時候: 首先,呼叫key1所在類的hashCode()計算key1 ......

    uj5u.com 2020-09-10 05:36:38 more
最新发布
  • 【中介者設計模式詳解】C/Java/JS/Go/Python/TS不同語言實作

    * 中介者模式是一種行為型設計模式,它可以用來減少類之間的直接依賴關系,
    * 將物件之間的通信封裝到一個中介者物件中,從而使得各個物件之間的關系更加松散。
    * 在中介者模式中,物件之間不再直接相互互動,而是通過中介者來中轉訊息。 ......

    uj5u.com 2023-04-20 08:20:47 more
  • 露天煤礦現場調研和交流案例分享

    他們集團的資訊化公司及研究院在一個礦區正在做智能礦山的統一平臺的 試點,專案投資大概1億,包括了礦山的各方面的內容,顯示得我們這次交流有點多余。他們2年前開始做智能礦山的規劃,有很多煤礦行業專家的加持,他們的描述是非常完美,但是去年底應該上線的平臺,現在還沒有看到影子。他們確實有很多場景需求,但是被... ......

    uj5u.com 2023-04-20 08:20:25 more
  • 《社區人員管理》實戰案例設計&個人案例分享

    設計是一個讓人夢想成真程序,開始編碼、測驗、除錯之前進行需求分析和架構設計,才能保證關鍵方面都做正確 ......

    uj5u.com 2023-04-20 08:20:17 more
  • 軟體架構生態化-多角色交付的探索實踐

    作為一個技術架構師,不僅僅要緊跟行業技術趨勢,還要結合研發團隊現狀及痛點,探索新的交付方案。在日常中,你是否遇到如下問題 “ 業務需求排期長研發是瓶頸;非研發角色感受不到研發技改提效的變化;引入ISV 團隊又擔心質量和安全,培訓周期長“等等,基于此我們探索了一種新的技術體系及交付方案來解決如上問題。 ......

    uj5u.com 2023-04-20 08:20:10 more
  • 【中介者設計模式詳解】C/Java/JS/Go/Python/TS不同語言實作

    * 中介者模式是一種行為型設計模式,它可以用來減少類之間的直接依賴關系,
    * 將物件之間的通信封裝到一個中介者物件中,從而使得各個物件之間的關系更加松散。
    * 在中介者模式中,物件之間不再直接相互互動,而是通過中介者來中轉訊息。 ......

    uj5u.com 2023-04-20 08:19:44 more
  • 露天煤礦現場調研和交流案例分享

    他們集團的資訊化公司及研究院在一個礦區正在做智能礦山的統一平臺的 試點,專案投資大概1億,包括了礦山的各方面的內容,顯示得我們這次交流有點多余。他們2年前開始做智能礦山的規劃,有很多煤礦行業專家的加持,他們的描述是非常完美,但是去年底應該上線的平臺,現在還沒有看到影子。他們確實有很多場景需求,但是被... ......

    uj5u.com 2023-04-20 08:19:07 more
  • 《社區人員管理》實戰案例設計&個人案例分享

    設計是一個讓人夢想成真程序,開始編碼、測驗、除錯之前進行需求分析和架構設計,才能保證關鍵方面都做正確 ......

    uj5u.com 2023-04-20 08:18:57 more
  • 軟體架構生態化-多角色交付的探索實踐

    作為一個技術架構師,不僅僅要緊跟行業技術趨勢,還要結合研發團隊現狀及痛點,探索新的交付方案。在日常中,你是否遇到如下問題 “ 業務需求排期長研發是瓶頸;非研發角色感受不到研發技改提效的變化;引入ISV 團隊又擔心質量和安全,培訓周期長“等等,基于此我們探索了一種新的技術體系及交付方案來解決如上問題。 ......

    uj5u.com 2023-04-20 08:18:49 more
  • 05單件模式

    #經典的單件模式 public class Singleton { private static Singleton uniqueInstance; //一個靜態變數持有Singleton類的唯一實體。 // 其他有用的實體變數寫在這里 //構造器宣告為私有,只有Singleton可以實體化這個類! ......

    uj5u.com 2023-04-19 08:42:51 more
  • 【架構與設計】常見微服務分層架構的區別和落地實踐

    軟體工程的方方面面都遵循一個最基本的道理:沒有銀彈,架構分層模型更是如此,每一種都有各自優缺點,所以請根據不同的業務場景,并遵循簡單、可演進這兩個重要的架構原則選擇合適的架構分層模型即可。 ......

    uj5u.com 2023-04-19 08:42:41 more