主頁 >  其他 > cytoscape.js進階篇

cytoscape.js進階篇

2020-09-30 21:33:23 其他

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/qita/143297.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)

熱門瀏覽
  • 網閘典型架構簡述

    網閘架構一般分為兩種:三主機的三系統架構網閘和雙主機的2+1架構網閘。 三主機架構分別為內端機、外端機和仲裁機。三機無論從軟體和硬體上均各自獨立。首先從硬體上來看,三機都用各自獨立的主板、記憶體及存盤設備。從軟體上來看,三機有各自獨立的作業系統。這樣能達到完全的三機獨立。對于“2+1”系統,“2”分為 ......

    uj5u.com 2020-09-10 02:00:44 more
  • 如何從xshell上傳檔案到centos linux虛擬機里

    如何從xshell上傳檔案到centos linux虛擬機里及:虛擬機CentOs下執行 yum -y install lrzsz命令,出現錯誤:鏡像無法找到軟體包 前言 一、安裝lrzsz步驟 二、上傳檔案 三、遇到的問題及解決方案 總結 前言 提示:其實很簡單,往虛擬機上安裝一個上傳檔案的工具 ......

    uj5u.com 2020-09-10 02:00:47 more
  • 一、SQLMAP入門

    一、SQLMAP入門 1、判斷是否存在注入 sqlmap.py -u 網址/id=1 id=1不可缺少。當注入點后面的引數大于兩個時。需要加雙引號, sqlmap.py -u "網址/id=1&uid=1" 2、判斷文本中的請求是否存在注入 從文本中加載http請求,SQLMAP可以從一個文本檔案中 ......

    uj5u.com 2020-09-10 02:00:50 more
  • Metasploit 簡單使用教程

    metasploit 簡單使用教程 浩先生, 2020-08-28 16:18:25 分類專欄: kail 網路安全 linux 文章標簽: linux資訊安全 編輯 著作權 metasploit 使用教程 前言 一、Metasploit是什么? 二、準備作業 三、具體步驟 前言 Msfconsole ......

    uj5u.com 2020-09-10 02:00:53 more
  • 游戲逆向之驅動層與用戶層通訊

    驅動層代碼: #pragma once #include <ntifs.h> #define add_code CTL_CODE(FILE_DEVICE_UNKNOWN,0x800,METHOD_BUFFERED,FILE_ANY_ACCESS) /* 更多游戲逆向視頻www.yxfzedu.com ......

    uj5u.com 2020-09-10 02:00:56 more
  • 北斗電力時鐘(北斗授時服務器)讓網路資料更精準

    北斗電力時鐘(北斗授時服務器)讓網路資料更精準 北斗電力時鐘(北斗授時服務器)讓網路資料更精準 京準電子科技官微——ahjzsz 近幾年,資訊技術的得了快速發展,互聯網在逐漸普及,其在人們生活和生產中都得到了廣泛應用,并且取得了不錯的應用效果。計算機網路資訊在電力系統中的應用,一方面使電力系統的運行 ......

    uj5u.com 2020-09-10 02:01:03 more
  • 【CTF】CTFHub 技能樹 彩蛋 writeup

    ?碎碎念 CTFHub:https://www.ctfhub.com/ 筆者入門CTF時時剛開始刷的是bugku的舊平臺,后來才有了CTFHub。 感覺不論是網頁UI設計,還是題目質量,賽事跟蹤,工具軟體都做得很不錯。 而且因為獨到的金幣制度的確讓人有一種想去刷題賺金幣的感覺。 個人還是非常喜歡這個 ......

    uj5u.com 2020-09-10 02:04:05 more
  • 02windows基礎操作

    我學到了一下幾點 Windows系統目錄結構與滲透的作用 常見Windows的服務詳解 Windows埠詳解 常用的Windows注冊表詳解 hacker DOS命令詳解(net user / type /md /rd/ dir /cd /net use copy、批處理 等) 利用dos命令制作 ......

    uj5u.com 2020-09-10 02:04:18 more
  • 03.Linux基礎操作

    我學到了以下幾點 01Linux系統介紹02系統安裝,密碼啊破解03Linux常用命令04LAMP 01LINUX windows: win03 8 12 16 19 配置不繁瑣 Linux:redhat,centos(紅帽社區版),Ubuntu server,suse unix:金融機構,證券,銀 ......

    uj5u.com 2020-09-10 02:04:30 more
  • 05HTML

    01HTML介紹 02頭部標簽講解03基礎標簽講解04表單標簽講解 HTML前段語言 js1.了解代碼2.根據代碼 懂得挖掘漏洞 (POST注入/XSS漏洞上傳)3.黑帽seo 白帽seo 客戶網站被黑帽植入劫持代碼如何處理4.熟悉html表單 <html><head><title>TDK標題,描述 ......

    uj5u.com 2020-09-10 02:04:36 more
最新发布
  • 2023年最新微信小程式抓包教程

    01 開門見山 隔一個月發一篇文章,不過分。 首先回顧一下《微信系結手機號資料庫被脫庫事件》,我也是第一時間得知了這個訊息,然后跟蹤了整件事情的經過。下面是這起事件的相關截圖以及近日流出的一萬條資料樣本: 個人認為這件事也沒什么,還不如關注一下之前45億快遞資料查詢渠道疑似在近日復活的訊息。 訊息是 ......

    uj5u.com 2023-04-20 08:48:24 more
  • web3 產品介紹:metamask 錢包 使用最多的瀏覽器插件錢包

    Metamask錢包是一種基于區塊鏈技術的數字貨幣錢包,它允許用戶在安全、便捷的環境下管理自己的加密資產。Metamask錢包是以太坊生態系統中最流行的錢包之一,它具有易于使用、安全性高和功能強大等優點。 本文將詳細介紹Metamask錢包的功能和使用方法。 一、 Metamask錢包的功能 數字資 ......

    uj5u.com 2023-04-20 08:47:46 more
  • vulnhub_Earth

    前言 靶機地址->>>vulnhub_Earth 攻擊機ip:192.168.20.121 靶機ip:192.168.20.122 參考文章 https://www.cnblogs.com/Jing-X/archive/2022/04/03/16097695.html https://www.cnb ......

    uj5u.com 2023-04-20 07:46:20 more
  • 從4k到42k,軟體測驗工程師的漲薪史,給我看哭了

    清明節一過,盲猜大家已經無心上班,在數著日子準備過五一,但一想到銀行卡里的余額……瞬間心情就不美麗了。最近,2023年高校畢業生就業調查顯示,本科畢業月平均起薪為5825元。調查一出,便有很多同學表示自己又被平均了。看著這一資料,不免讓人想到前不久中國青年報的一項調查:近六成大學生認為畢業10年內會 ......

    uj5u.com 2023-04-20 07:44:00 more
  • 最新版本 Stable Diffusion 開源 AI 繪畫工具之中文自動提詞篇

    🎈 標簽生成器 由于輸入正向提示詞 prompt 和反向提示詞 negative prompt 都是使用英文,所以對學習母語的我們非常不友好 使用網址:https://tinygeeker.github.io/p/ai-prompt-generator 這個網址是為了讓大家在使用 AI 繪畫的時候 ......

    uj5u.com 2023-04-20 07:43:36 more
  • 漫談前端自動化測驗演進之路及測驗工具分析

    隨著前端技術的不斷發展和應用程式的日益復雜,前端自動化測驗也在不斷演進。隨著 Web 應用程式變得越來越復雜,自動化測驗的需求也越來越高。如今,自動化測驗已經成為 Web 應用程式開發程序中不可或缺的一部分,它們可以幫助開發人員更快地發現和修復錯誤,提高應用程式的性能和可靠性。 ......

    uj5u.com 2023-04-20 07:43:16 more
  • CANN開發實踐:4個DVPP記憶體問題的典型案例解讀

    摘要:由于DVPP媒體資料處理功能對存放輸入、輸出資料的記憶體有更高的要求(例如,記憶體首地址128位元組對齊),因此需呼叫專用的記憶體申請介面,那么本期就分享幾個關于DVPP記憶體問題的典型案例,并給出原因分析及解決方法。 本文分享自華為云社區《FAQ_DVPP記憶體問題案例》,作者:昇騰CANN。 DVPP ......

    uj5u.com 2023-04-20 07:43:03 more
  • msf學習

    msf學習 以kali自帶的msf為例 一、msf核心模塊與功能 msf模塊都放在/usr/share/metasploit-framework/modules目錄下 1、auxiliary 輔助模塊,輔助滲透(埠掃描、登錄密碼爆破、漏洞驗證等) 2、encoders 編碼器模塊,主要包含各種編碼 ......

    uj5u.com 2023-04-20 07:42:59 more
  • Halcon軟體安裝與界面簡介

    1. 下載Halcon17版本到到本地 2. 雙擊安裝包后 3. 步驟如下 1.2 Halcon軟體安裝 界面分為四大塊 1. Halcon的五個助手 1) 影像采集助手:與相機連接,設定相機引數,采集影像 2) 標定助手:九點標定或是其它的標定,生成標定檔案及內參外參,可以將像素單位轉換為長度單位 ......

    uj5u.com 2023-04-20 07:42:17 more
  • 在MacOS下使用Unity3D開發游戲

    第一次發博客,先發一下我的游戲開發環境吧。 去年2月份買了一臺MacBookPro2021 M1pro(以下簡稱mbp),這一年來一直在用mbp開發游戲。我大致分享一下我的開發工具以及使用體驗。 1、Unity 官網鏈接: https://unity.cn/releases 我一般使用的Apple ......

    uj5u.com 2023-04-20 07:40:19 more