G6關(guān)系圖隱藏節(jié)點(diǎn)算法探究

最近遇到一個(gè)需求:在一個(gè)關(guān)系圖上進(jìn)行特定節(jié)點(diǎn)的隱藏乍炉,并生成新的關(guān)系圖抡锈。項(xiàng)目里面用的可視化框架是g6怜珍,然后我去g6的文檔里面搜索了一下“數(shù)據(jù)過(guò)濾”渡紫,好家伙,出現(xiàn)了一個(gè)左側(cè)目錄里面根本不存在的頁(yè)面考赛,而且并沒(méi)有任何代碼講解惕澎,只是在介紹一個(gè)項(xiàng)目,不知道這個(gè)項(xiàng)目是曾經(jīng)存在過(guò)還是以后將要上線......最后很無(wú)奈颜骤,不管數(shù)據(jù)過(guò)濾這個(gè)功能是曾經(jīng)有還是未來(lái)會(huì)有唧喉,反正當(dāng)前版本是沒(méi)找著的,只有自力更生了忍抽。

關(guān)系圖基本說(shuō)明

其實(shí)不管是g6還是其他的可視化框架八孝,普通的關(guān)系圖只需要兩組數(shù)據(jù)(不考慮兩個(gè)節(jié)點(diǎn)之間多條連線),準(zhǔn)確的說(shuō)是兩個(gè)數(shù)組鸠项,第一個(gè)是點(diǎn)的集合干跛,每一項(xiàng)儲(chǔ)存每個(gè)節(jié)點(diǎn)的id,樣式以及其他的信息祟绊,至少要有一個(gè)唯一id楼入;第二個(gè)是節(jié)點(diǎn)關(guān)系的集合,每一項(xiàng)有一個(gè)上游節(jié)點(diǎn)id和一個(gè)下游節(jié)點(diǎn)id牧抽,這兩個(gè)id就能畫(huà)出兩個(gè)節(jié)點(diǎn)和一條線了嘉熊。最終無(wú)數(shù)個(gè)節(jié)點(diǎn)和節(jié)點(diǎn)關(guān)系就能組成一個(gè)復(fù)雜的關(guān)系圖。
框架會(huì)幫我們把節(jié)點(diǎn)畫(huà)到畫(huà)布上并分配每個(gè)節(jié)點(diǎn)的坐標(biāo)扬舒,我們要做的就是傳入點(diǎn)的集合和關(guān)系的集合阐肤。
而要實(shí)現(xiàn)開(kāi)頭的需求整體上需要兩步,第一步是將要隱藏的點(diǎn)從原始的點(diǎn)集合里面去掉呼巴,第二步是生成新的節(jié)點(diǎn)關(guān)系數(shù)組泽腮。第一步進(jìn)行數(shù)組過(guò)濾就能實(shí)現(xiàn),關(guān)鍵就在于第二步衣赶。

生成基本的關(guān)系圖

寫(xiě)了一個(gè)demo,整個(gè)步驟官網(wǎng)上都有就不贅述了厚满,只將數(shù)據(jù)放上來(lái)府瞄,加上了一點(diǎn)樣式便于區(qū)分,使用的是g6的dagre布局碘箍。項(xiàng)目里面一開(kāi)始沒(méi)考慮到需要隱藏的節(jié)點(diǎn)有兩兩相鄰的情況遵馆,所以當(dāng)時(shí)多花了一天才搞定。

    const data = {
      // 點(diǎn)集
      nodes: [
        {
          id: 'node1',
          label: 'n1',
          type: 'normal' // normal-留下的點(diǎn)hide-需要隱藏的點(diǎn)
        },
        {
          id: 'node2',
          label: 'n2',
          type: 'normal'
        },
        {
          id: 'node3',
          label: 'n3',
          type: 'normal'
        },
        {
          id: 'node4',
          label: 'n4',
          type: 'normal'
        },
        {
          id: 'hide1',
          label: 'h1',
          style: {
            fill: 'red'
          },
          type: 'hide'
        },
        {
          id: 'hide2',
          label: 'h2',
          style: {
            fill: 'red'
          },
          type: 'hide'
        }
      ],
      // 邊集
      edges: [
        {
          source: 'node1', // 上游節(jié)點(diǎn)
          target: 'hide1' // 下游節(jié)點(diǎn)
        },
        {
          source: 'node2',
          target: 'hide1'
        },
        {
          source: 'hide1',
          target: 'hide2'
        },
        {
          source: 'hide2',
          target: 'node3'
        },
        {
          source: 'hide2',
          target: 'node4'
        }
      ],
    };

效果圖如下


點(diǎn)擊切換節(jié)點(diǎn)實(shí)現(xiàn)紅色的點(diǎn)隱藏/顯示

現(xiàn)在要求h1和h2隱藏之后n1要連接到n3和n4丰榴,n2也要連接到n3和n4货邓。

整體思路

  1. 過(guò)濾出需要隱藏的點(diǎn)和不需要隱藏的點(diǎn)
  2. 從現(xiàn)有的節(jié)點(diǎn)關(guān)系中過(guò)濾出無(wú)需改變的節(jié)點(diǎn)關(guān)系(這個(gè)demo比較簡(jiǎn)單,所有的節(jié)點(diǎn)關(guān)系都要改變四濒,假如只有h1需要隱藏换况,那么h2-n3,h2-n4這兩個(gè)節(jié)點(diǎn)關(guān)系就過(guò)濾出來(lái)了)
  3. 從現(xiàn)有的節(jié)點(diǎn)關(guān)系中遍歷出需要隱藏的節(jié)點(diǎn)的上游節(jié)點(diǎn)和下游節(jié)點(diǎn)
  4. 通過(guò)雙重遍歷上下游節(jié)點(diǎn)集合將新的節(jié)點(diǎn)關(guān)系加入到第一步生成的無(wú)需改變的節(jié)點(diǎn)關(guān)系中
  5. 將新的節(jié)點(diǎn)關(guān)系去重职辨,隨后通過(guò)不需要隱藏的點(diǎn)和新的節(jié)點(diǎn)關(guān)系生成新的關(guān)系圖

上述第三步里面需要判斷獲取到的上游或下游節(jié)點(diǎn)是否也是隱藏節(jié)點(diǎn)(遞歸)

詳細(xì)過(guò)程

首先是準(zhǔn)備工作,過(guò)濾出需要的數(shù)組(以下代碼里面使用的'_'都代表lodash中的方法)

      // 需要隱藏的節(jié)點(diǎn)的id集合(方便后續(xù)計(jì)算)
      const hideNodes = data.nodes.filter(v=>v.type==='hide').map(v=>v.id);
      // 剩余正常顯示的節(jié)點(diǎn)
      const restNodes = data.nodes.filter(v=>v.type!=='hide');
      // 舊的節(jié)點(diǎn)關(guān)系(因?yàn)間6使用data數(shù)據(jù)生成圖之后data里面會(huì)多出來(lái)很多屬性戈二,所以過(guò)濾無(wú)關(guān)屬性方便去重)
      const oldEdges = data.edges.map(v => {
          return {
              target: v.target,
              source: v.source
          }
      });
      // 新的節(jié)點(diǎn)關(guān)系
      let newEdges = [];
      // 無(wú)需改變的節(jié)點(diǎn)關(guān)系
      const noChanges = oldEdges.filter(v=>{
        return !hideNodes.includes(v.target) && !hideNodes.includes(v.source);
      });

獲取被隱藏節(jié)點(diǎn)的下游節(jié)點(diǎn)和上游節(jié)點(diǎn)舒裤,當(dāng)上游或下游節(jié)點(diǎn)也是被隱藏的節(jié)點(diǎn)時(shí)需要遞歸繼續(xù)判斷,這里為了思考方便我寫(xiě)了兩個(gè)方法觉吭,最后項(xiàng)目里面寫(xiě)一個(gè)方法就行

  // hideItem是hideNodes遍歷時(shí)的每一項(xiàng)
    function getTarget(oldEdges, hideItem, hideNodes) {
        const result = [];
        oldEdges.forEach(v=>{
          if(v.source === hideItem) {
            if(hideNodes.includes(v.target)){
              const currentItem = v.target;
              const restHideNodes = _.without(hideNodes, v.target); // 當(dāng)前值從隱藏節(jié)點(diǎn)數(shù)組中去掉
              const res = getTarget(oldEdges, currentItem, restHideNodes);
              result.push(...res);
            }else{
              result.push(v.target);
            }
          }
        });
        return result;
      }
      function getSource(oldEdges, hideItem, hideNodes) {
        const result = [];
        oldEdges.forEach(v=>{
          if(v.target === hideItem) {
            if(hideNodes.includes(v.source)){
              const currentItem = v.source;
              const restHideNodes = _.without(hideNodes, v.source); // 當(dāng)前值從隱藏節(jié)點(diǎn)數(shù)組中去掉
              const res = getSource(oldEdges, currentItem, restHideNodes);
              result.push(...res);
            }else{
              result.push(v.source);
            }
          }
        });
        return result;
      }

最后雙重for循環(huán)獲取新的節(jié)點(diǎn)關(guān)系

      hideNodes.forEach(hideItem=>{
        // 被隱藏節(jié)點(diǎn)的下游節(jié)點(diǎn)
        const targetArr = getTarget(oldEdges, hideItem, hideNodes);
        // 被隱藏節(jié)點(diǎn)的上游節(jié)點(diǎn)
        const sourceArr = getSource(oldEdges, hideItem, hideNodes);
        for (const target of targetArr) {
          for (const source of sourceArr) {
            noChanges.push({ source, target });
          }
        }
        newEdges.push(...noChanges);
        // 去重
        newEdges = _.uniqWith(newEdges, _.isEqual);
      });

至此這個(gè)功能就算是完成了腾供,項(xiàng)目里面用起來(lái)暫時(shí)還沒(méi)什么問(wèn)題,不過(guò)像這種稍微復(fù)雜的過(guò)程可能還有進(jìn)一步優(yōu)化的空間鲜滩,以我的實(shí)力也就只能做到這種地步了伴鳖,這個(gè)計(jì)算過(guò)程還是我與小伙伴討論前后完善了3天才寫(xiě)出來(lái)的,這就是吃了不懂算法的虧啊徙硅。

最后附上完整代碼

<!DOCTYPE html>
<html lang="zh-CN">
<head>
    <meta charset="UTF-8">
    <title>Document</title>
</head>
<body>
  <button id="hide">切換節(jié)點(diǎn)</button>
  <div id="mountNode"></div>
  <script src="https://gw.alipayobjects.com/os/lib/antv/g6/4.0.3/dist/g6.min.js"></script>
  <script src="https://cdn.bootcdn.net/ajax/libs/lodash.js/4.17.20/lodash.min.js"></script>
    <script>
    let isShow = true;
    const data = {...}; // 見(jiàn)上文

    const graph = new G6.Graph({
      container: 'mountNode',
      width: 800,
      height: 500,
      layout: {
          type: 'dagre',
          rankdir: 'LR'
      }
    });
    graph.data(data);
    graph.render();
    const hideBtn = document.getElementById('hide');
    hideBtn.onclick = ()=>{
      isShow = !isShow;
      if(isShow) {
        graph.data(data);
        graph.render();
        return;
      }
      // 需要隱藏的節(jié)點(diǎn)的id集合
      const hideNodes = data.nodes.filter(v=>v.type==='hide').map(v=>v.id);
      // 剩余正常顯示的節(jié)點(diǎn)
      const restNodes = data.nodes.filter(v=>v.type!=='hide');
      // 舊的節(jié)點(diǎn)關(guān)系(過(guò)濾無(wú)關(guān)屬性方便去重)
      const oldEdges = data.edges.map(v => {
          return {
              target: v.target,
              source: v.source
          }
      });
      // 新的節(jié)點(diǎn)關(guān)系
      let newEdges = [];
      // 無(wú)需改變的節(jié)點(diǎn)關(guān)系
      const noChanges = oldEdges.filter(v=>{
        return !hideNodes.includes(v.target) && !hideNodes.includes(v.source);
      });
      hideNodes.forEach(hideItem=>{
        // 被隱藏節(jié)點(diǎn)的下游節(jié)點(diǎn)
        const targetArr = getTarget(oldEdges, hideItem, hideNodes);
        // 被隱藏節(jié)點(diǎn)的上游節(jié)點(diǎn)
        const sourceArr = getSource(oldEdges, hideItem, hideNodes);
        for (const target of targetArr) {
          for (const source of sourceArr) {
            noChanges.push({ source, target });
          }
        }
        newEdges.push(...noChanges);
        // 去重
        newEdges = _.uniqWith(newEdges, _.isEqual);
      });
      const newData = {
        nodes: restNodes,
        edges: newEdges
      };
      
      graph.data(newData);
      graph.render();
    }
    function getTarget(oldEdges, hideItem, hideNodes) {
        const result = [];
        oldEdges.forEach(v=>{
          if(v.source === hideItem) {
            if(hideNodes.includes(v.target)){
              const currentItem = v.target;
              const restHideNodes = _.without(hideNodes, v.target); // 當(dāng)前值從隱藏節(jié)點(diǎn)數(shù)組中去掉
              const res = getTarget(oldEdges, currentItem, restHideNodes);
              result.push(...res);
            }else{
              result.push(v.target);
            }
          }
        });
        return result;
      }
      function getSource(oldEdges, hideItem, hideNodes) {
        const result = [];
        oldEdges.forEach(v=>{
          if(v.target === hideItem) {
            if(hideNodes.includes(v.source)){
              const currentItem = v.source;
              const restHideNodes = _.without(hideNodes, v.source); // 當(dāng)前值從隱藏節(jié)點(diǎn)數(shù)組中去掉
              const res = getSource(oldEdges, currentItem, restHideNodes);
              result.push(...res);
            }else{
              result.push(v.source);
            }
          }
        });
        return result;
      }
    </script>
</body>
</html>
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
禁止轉(zhuǎn)載黎侈,如需轉(zhuǎn)載請(qǐng)通過(guò)簡(jiǎn)信或評(píng)論聯(lián)系作者。
  • 序言:七十年代末闷游,一起剝皮案震驚了整個(gè)濱河市峻汉,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌脐往,老刑警劉巖休吠,帶你破解...
    沈念sama閱讀 221,198評(píng)論 6 514
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異业簿,居然都是意外死亡瘤礁,警方通過(guò)查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 94,334評(píng)論 3 398
  • 文/潘曉璐 我一進(jìn)店門(mén)梅尤,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)柜思,“玉大人,你說(shuō)我怎么就攤上這事巷燥∩呐蹋” “怎么了?”我有些...
    開(kāi)封第一講書(shū)人閱讀 167,643評(píng)論 0 360
  • 文/不壞的土叔 我叫張陵缰揪,是天一觀的道長(zhǎng)陨享。 經(jīng)常有香客問(wèn)我,道長(zhǎng)钝腺,這世上最難降的妖魔是什么抛姑? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 59,495評(píng)論 1 296
  • 正文 為了忘掉前任,我火速辦了婚禮艳狐,結(jié)果婚禮上定硝,老公的妹妹穿的比我還像新娘。我一直安慰自己毫目,他們只是感情好蔬啡,可當(dāng)我...
    茶點(diǎn)故事閱讀 68,502評(píng)論 6 397
  • 文/花漫 我一把揭開(kāi)白布诲侮。 她就那樣靜靜地躺著,像睡著了一般星爪。 火紅的嫁衣襯著肌膚如雪浆西。 梳的紋絲不亂的頭發(fā)上,一...
    開(kāi)封第一講書(shū)人閱讀 52,156評(píng)論 1 308
  • 那天顽腾,我揣著相機(jī)與錄音近零,去河邊找鬼。 笑死抄肖,一個(gè)胖子當(dāng)著我的面吹牛久信,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播漓摩,決...
    沈念sama閱讀 40,743評(píng)論 3 421
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼裙士,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來(lái)了管毙?” 一聲冷哼從身側(cè)響起腿椎,我...
    開(kāi)封第一講書(shū)人閱讀 39,659評(píng)論 0 276
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎夭咬,沒(méi)想到半個(gè)月后啃炸,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 46,200評(píng)論 1 319
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡卓舵,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 38,282評(píng)論 3 340
  • 正文 我和宋清朗相戀三年南用,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片掏湾。...
    茶點(diǎn)故事閱讀 40,424評(píng)論 1 352
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡裹虫,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出融击,到底是詐尸還是另有隱情筑公,我是刑警寧澤,帶...
    沈念sama閱讀 36,107評(píng)論 5 349
  • 正文 年R本政府宣布砚嘴,位于F島的核電站十酣,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏际长。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,789評(píng)論 3 333
  • 文/蒙蒙 一兴泥、第九天 我趴在偏房一處隱蔽的房頂上張望工育。 院中可真熱鬧,春花似錦搓彻、人聲如沸如绸。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 32,264評(píng)論 0 23
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)怔接。三九已至搪泳,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間扼脐,已是汗流浹背岸军。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 33,390評(píng)論 1 271
  • 我被黑心中介騙來(lái)泰國(guó)打工, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留瓦侮,地道東北人艰赞。 一個(gè)月前我還...
    沈念sama閱讀 48,798評(píng)論 3 376
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像肚吏,于是被迫代替她去往敵國(guó)和親方妖。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,435評(píng)論 2 359

推薦閱讀更多精彩內(nèi)容