广告

买白酒,找南将

Skip to content

线条高级用法-插槽

Vue2 版本

adv-line-slot2.vue

javascript
<template>
  <div>
    <div ref="myPage" style="height:calc(100vh);">
      <RelationGraph
        ref="graphRef"
        :options="graphOptions">
        <template #line="{line, link}">
          <MyLine :link="link" :relation="line" />
        </template>
      </RelationGraph>
    </div>
  </div>
</template>

<script>
// 如果您没有在main.js文件中使用Vue.use(RelationGraph); 就需要使用下面这一行代码来引入relation-graph
// import RelationGraph from 'relation-graph';
import MyLine from './Demo4AdvLineSlot2LineSlot';
export default {
  name: 'Demo4AdvLineSlot',
  components: { MyLine },
  data() {
    return {
      graphOptions: {
        allowSwitchLineShape: true,
        allowSwitchJunctionPoint: true,
        defaultJunctionPoint: 'border',
        defaultLineColor: '#2E74B5',
        defaultNodeColor: '#2E74B5',
        defaultNodeBorderWidth: 2,
        defaultNodeBorderColor: '#2E74B5',
        defaultNodeFontColor: '#ffffff',
        // 这里可以参考"Graph 图谱"中的参数进行设置
      }
    };
  },
  mounted() {
    this.showGraph();
  },
  methods: {
    showGraph() {
      const __graph_json_data = {
        rootId: '2',
        nodes: [
          // 注意:在节点配置信息中,你的自定义属性需要像下面这样放到data标签中,否则数据会丢失
          { id: '1', text: '节点-1', data: { myicon: 'el-icon-star-on' }},
          { id: '2', text: '节点-2', data: { myicon: 'el-icon-setting' }},
          { id: '4', text: '节点-4', data: { myicon: 'el-icon-star-on' }},
          { id: '6', text: '节点-6', data: { myicon: 'el-icon-setting' }},
          { id: '7', text: '节点-7', data: { myicon: 'el-icon-setting' }},
          { id: '8', text: '节点-8', data: { myicon: 'el-icon-star-on' }},
          { id: '9', text: '节点-9', data: { myicon: 'el-icon-headset' }}
        ],
        lines: [
          { from: '1', to: '2', text: '投资' },
          { from: '4', to: '2', text: '高管' },
          { from: '6', to: '2', text: '高管' },
          { from: '7', to: '2', text: '高管' },
          { from: '8', to: '2', text: '高管' },
          { from: '9', to: '2', text: '高管' }
        ]
      };
      this.$refs.graphRef.setJsonData(__graph_json_data, (graphInstance) => {
        // 这些写上当图谱初始化完成后需要执行的代码
      });
    }
  }
};
</script>

<style lang="scss">

</style>

<style lang="scss" scoped>
.c-node-menu-item{
  line-height: 30px;padding-left: 10px;cursor: pointer;color: #444444;font-size: 14px;border-top:#efefef solid 1px;
}
.c-node-menu-item:hover{
  background-color: rgba(66,187,66,0.2);
}
</style>

Demo4AdvLineSlot2LineSlot.vue

javascript
<template>
  <g>
    <!-- 常规方式 -->
    <path
      :d="pathData.path"
      :class="['c-rg-line', relation.styleClass, checked ? 'c-rg-line-checked' : '']"
      :stroke="
        checked
          ? options.checkedLineColor
          : relation.color
          ? relation.color
          : options.defaultLineColor
      "
      :style="{
        'opacity': relation.opacity,
        'stroke-width':
          (relation.lineWidth
            ? relation.lineWidth
            : options.defaultLineWidth) + 'px',
      }"
      :marker-start="showStartArrow"
      :marker-end="showEndArrow"
      fill="none"
      @click="onClick(relation, $event)"
    />
    <g
      v-if="
        options.defaultShowLineLabel &&
        options.canvasZoom > 40
      "
      :transform="pathData.textTransform"
    >
      <rect
        :key="'t-' + relation.seeks_id"
        rx="15" ry="15"
        :x="-30"
        :y="-15"
        :style="{
          opacity: relation.opacity,
          fill: checked
            ? options.checkedLineColor
            : relation.fontColor
            ? relation.fontColor
            : relation.color
            ? relation.color
            : undefined,
        }"
        class="c-rg-line-text-bg"
        @click="onClick(relation, $event)"
      >
      </rect>
      <text
          :x="0"
          :y="4"
          :style="{
          opacity: relation.opacity
        }"
          class="c-rg-line-text"
          @click="onClick(relation, $event)"
      >
        {{ relation.text }}
      </text>
    </g>
  </g>
</template>

<script>
export default {
  name: 'Demo4AdvLineSlot2LineSlot',
  props: {
    link: {
      mustUseProp: true,
      default: () => {
        return {};
      },
      type: Object,
    },
    relation: {
      mustUseProp: true,
      default: () => {
        return {};
      },
      type: Object,
    },
    relationIndex: {
      mustUseProp: true,
      default: () => {
        return 0;
      },
      type: Number,
    },
  },
  data() {
    return {
      is_flashing: false,
    };
  },
  inject: ['graph', 'graphInstance'],
  computed: {
    checked() {
      return this.relation.id === this.options.checkedLineId;
    },
    showStartArrow() {
      return this.relation.showStartArrow && this.relationGraph.getArrow(this.relation, this.link, true);
    },
    showEndArrow() {
      return this.relation.showEndArrow && this.relationGraph.getArrow(this.relation, this.link, false);
    },
    pathData() {
      try {
        const { path, textPosition } = this.relationGraph.createLinePath(
          this.link,
          this.relation,
          this.relationIndex
        );
        let textTransform = {}
        try {
          textTransform = this.relationGraph.getTextTransform(
              this.relation,
              textPosition.x,
              textPosition.y,
              textPosition.rotate
          )
        } catch (e) {
        }
        return {
          path,
          textTransform
        };
      } catch (e) {
      }
      return {path:null, textTransform: null};
    },
    options() {
      return this.graph.options;
    },
    relationGraph() {
      return this.graphInstance();
    }
  },
  show() {
    this.isShow = true;
  },
  watch: {},
  methods: {
    onClick(line, e) {
      // RGStore.commit('setCurrentLineId', this.lineProps.id)
      this.relationGraph.onLineClick(line, this.link, e);
    }
  },
};
</script>

<style scoped>
/*.RGLine-enter-active {*/
/*transition: all .3s ease;*/
/*}*/
/*.RGLine-leave-active {*/
/*transition: all .8s cubic-bezier(1.0, 0.5, 0.8, 1.0);*/
/*}*/
.c-rg-line-text-bg {
  fill: #2E74B5;
  width: 80px;
  height:30px;
  font-size: 12px;
  border-radius: 10px;
}
.c-rg-line-text {
  fill: #ffffff;
  font-size: 12px;
}
.c-rg-line {
  z-index: 1000;
  fill-rule: nonzero;
  /*marker-end: url('#arrow');*/
  /* firefox bug fix - won't rotate at 90deg angles */
  /*-moz-transform: rotate(-89deg) translateX(-190px);*/
  /*animation-timing-function:linear;*/
  /*animation: ACTRGLineInit 1s;*/
}
.c-rg-line-tool {
  stroke-dasharray: 5, 5, 5;
}
.c-rg-line-flash {
  /* firefox bug fix - won't rotate at 90deg angles */
  -moz-transform: rotate(-89deg) translateX(-190px);
  animation-timing-function: linear;
  animation: ACTRGLineChecked 10s;
}
@keyframes ACTRGLineInit {
  from {
    stroke-dashoffset: 10px;
    stroke-dasharray: 20, 20, 20;
  }

  to {
    stroke-dashoffset: 0;
    stroke-dasharray: 0, 0, 0;
  }
}
@keyframes ACTRGLineChecked {
  from {
    stroke-dashoffset: 352px;
  }

  to {
    stroke-dashoffset: 0;
  }
}
.c-rg-line-checked {
  /*stroke: var(--stroke);*/
  /*marker-end: var(--markerEndChecked) !important;*/
  stroke-width: 2px;
  stroke-dasharray: 5, 5, 5;
  stroke-dashoffset: 3px;
  stroke-linecap: butt;
  /*stroke: #FD8B37;*/
  stroke-linejoin: bevel;
  /* firefox bug fix - won't rotate at 90deg angles */
  /*-moz-transform: rotate(-89deg) translateX(-190px);*/
  animation-timing-function: linear;
  animation: ACTRGLineChecked 10s;
}
</style>

Vue3 版本

adv-line-slot2.vue

javascript
<template>
  <div>
    <div ref="myPage" style="height:calc(100vh);">
      <RelationGraph ref="graphRef" :options="graphOptions">
        <template #line="{link, line}">
          <MyLine :link="link" :line="line" />
        </template>
      </RelationGraph>
    </div>
  </div>
</template>

<script lang="ts" setup>
import { defineComponent, onMounted, ref } from 'vue';
import RelationGraph from 'relation-graph-vue3';
import { RelationGraphComponent, RGJsonData, RGOptions, RGNode, RGLine, RGLink, RGUserEvent } from 'relation-graph-vue3';
import MyLine from './Demo4AdvLineSlot2LineSlot';

const graphRef = ref<RelationGraphComponent>();
const graphOptions: RGOptions = {
    allowSwitchLineShape: true,
    allowSwitchJunctionPoint: true,
    defaultJunctionPoint: 'border',
    defaultLineColor: '#2E74B5',
    defaultNodeColor: '#2E74B5',
    defaultNodeBorderWidth: 2,
    defaultNodeBorderColor: '#2E74B5',
    defaultNodeFontColor: '#ffffff',
    // 这里可以参考"Graph 图谱"中的参数进行设置

};

onMounted(() => {
    showGraph();
});

const showGraph = async() => {
    const __graph_json_data: RGJsonData = {
        rootId: '2',
        nodes: [
            // 注意:在节点配置信息中,你的自定义属性需要像下面这样放到data标签中,否则数据会丢失

            { id: '1', text: '节点-1', data: { myicon: 'el-icon-star-on' }},
            { id: '2', text: '节点-2', data: { myicon: 'el-icon-setting' }},
            { id: '4', text: '节点-4', data: { myicon: 'el-icon-star-on' }},
            { id: '6', text: '节点-6', data: { myicon: 'el-icon-setting' }},
            { id: '7', text: '节点-7', data: { myicon: 'el-icon-setting' }},
            { id: '8', text: '节点-8', data: { myicon: 'el-icon-star-on' }},
            { id: '9', text: '节点-9', data: { myicon: 'el-icon-headset' }}
        ],
        lines: [
            { from: '1', to: '2', text: '投资' },
            { from: '4', to: '2', text: '高管' },
            { from: '6', to: '2', text: '高管' },
            { from: '7', to: '2', text: '高管' },
            { from: '8', to: '2', text: '高管' },
            { from: '9', to: '2', text: '高管' }
        ]
    };
    const graphInstance = graphRef.value!.getInstance();
    await graphInstance.setJsonData(__graph_json_data);
    await graphInstance.moveToCenter();
    await graphInstance.zoomToFit();
};
</script>

<style lang="scss">

</style>

<style lang="scss" scoped>
.c-node-menu-item{
  line-height: 30px;padding-left: 10px;cursor: pointer;color: #444444;font-size: 14px;border-top:#efefef solid 1px;
}
.c-node-menu-item:hover{
  background-color: rgba(66,187,66,0.2);
}
</style>

Demo4AdvLineSlot2LineSlot.vue

javascript
<template>
  <g>
    <!-- Regular way -->
    <path

      :d="pathData.path"
      :class="['c-rg-line', line.styleClass, checked ? 'c-rg-line-checked' : '']"
      :stroke="checked ? options.checkedLineColor : line.color ? line.color : options.defaultLineColor"
      :style="{
        'opacity': line.opacity,
        'stroke-width': (line.lineWidth ? line.lineWidth : options.defaultLineWidth) + 'px',
      }"
      :marker-start="showStartArrow"
      :marker-end="showEndArrow"
      fill="none"
      @click="onClick(line, $event)"
    />
    <g v-if="options.defaultShowLineLabel && options.canvasZoom > 40" :transform="pathData.textTransform">
      <rect

        :key="'t-' + line.seeks_id"
        rx="15"
        ry="15"
        :x="-30"
        :y="-15"
        :style="{
          opacity: line.opacity,
          fill: checked ? options.checkedLineColor : line.fontColor ? line.fontColor : line.color ? line.color : undefined,
        }"
        class="c-rg-line-text-bg"
        @click="onClick(line, $event)"
      />
      <text

        :x="0"
        :y="4"
        :style="{
          opacity: line.opacity

        }"
        class="c-rg-line-text"
        @click="onClick(line, $event)"
      >
        {{ line.text }}
      </text>
    </g>
  </g>
</template>

<script lang="ts" setup>
import {defineComponent, inject, ref, onMounted, computed} from 'vue';
import type { RGJsonData, RGLine, RGLink, RGNode, RGOptions, RGUserEvent, RelationGraphComponent } from 'relation-graph-vue3';
import {RelationGraphInstance, graphKey} from "relation-graph-vue3";

const {link,line} = defineProps<{
    link: RGLink,
    line: RGLine,
}>();
const graph = inject<any>(graphKey);
const isFlashing = ref(false);
console.log(graph);
const options = computed(() => {
    return graph.options;
});
const checked = computed(() => {
    return line.id === graph.options.checkedLineId;
});

const showStartArrow = computed(() => {
    return line.showStartArrow && graph.instance.getArrow(line, link, true);
});

const showEndArrow = computed(() => {
    return line.showEndArrow && graph.instance.getArrow(line, link, false);
});

const pathData = computed(() => {
    try {
        const { path, textPosition } = graph.instance.createLinePath(link, line, 0);
        let textTransform = {};
        try {
            textTransform = graph.instance.getTextTransform(
                line,
                textPosition.x,
                textPosition.y,
                textPosition.rotate

            );
        } catch (e) {}
        return {
            path,
            textTransform,
        };
    } catch (e) {}
    return { path: null, textTransform: null };
});


const onClick = (lineObject: RGLine, linkObject: RGLink, $event: RGUserEvent) => {
    graph.instance.onLineClick(lineObject, linkObject, $event);
};

onMounted(() => {
    isFlashing.value = true;
});
</script>

<style scoped>
/*.RGLine-enter-active {*/
/*transition: all .3s ease;*/
/*}*/
/*.RGLine-leave-active {*/
/*transition: all .8s cubic-bezier(1.0, 0.5, 0.8, 1.0);*/
/*}*/
.c-rg-line-text-bg {
  fill: #2E74B5;
  width: 80px;
  height: 30px;
  font-size: 12px;
  border-radius: 10px;
}
.c-rg-line-text {
  fill: #ffffff;
  font-size: 12px;
}
.c-rg-line {
  z-index: 1000;
  fill-rule: nonzero;
  /*marker-end: url('#arrow');*/
  /* firefox bug fix - won't rotate at 90deg angles */
  /*-moz-transform: rotate(-89deg) translateX(-190px);*/
  /*animation-timing-function:linear;*/
  /*animation: ACTRGLineInit 1s;*/
}
.c-rg-line-tool {
  stroke-dasharray: 5, 5, 5;
}
.c-rg-line-flash {
  /* firefox bug fix - won't rotate at 90deg angles */
  -moz-transform: rotate(-89deg) translateX(-190px);
  animation-timing-function: linear;
  animation: ACTRGLineChecked 10s;
}
@keyframes ACTRGLineInit {
  from {
    stroke-dashoffset: 10px;
    stroke-dasharray: 20, 20, 20;
  }

  to {
    stroke-dashoffset: 0;
    stroke-dasharray: 0, 0, 0;
  }
}
@keyframes ACTRGLineChecked {
  from {
    stroke-dashoffset: 352px;
  }

  to {
    stroke-dashoffset: 0;
  }
}
.c-rg-line-checked {
  /*stroke: var(--stroke);*/
  /*marker-end: var(--markerEndChecked) !important;*/
  stroke-width: 2px;
  stroke-dasharray: 5, 5, 5;
  stroke-dashoffset: 3px;
  stroke-linecap: butt;
  /*stroke: #FD8B37;*/
  stroke-linejoin: bevel;
  /* firefox bug fix - won't rotate at 90deg angles */
  /*-moz-transform: rotate(-89deg) translateX(-190px);*/
  animation-timing-function: linear;
  animation: ACTRGLineChecked 10s;
}
</style>

React 版本

adv-line-slot2.tsx

javascript
import React, { useEffect, useRef } from 'react'
import RelationGraph, {
  RGJsonData,
  RGOptions,
  RGNode,
  RGLine,
  RGLink,
  RGUserEvent,
  RGNodeSlotProps,
  RGLineSlotProps,
  RelationGraphComponent,
} from 'relation-graph-react'
import MyLine from './Demo4AdvLineSlot2LineSlot'
import './adv-line-slot2.scss'

const AdvLineSlot2 = () => {
  const graphRef = (useRef < RelationGraphComponent) | (null > null)

  const graphOptions: RGOptions = {
    allowSwitchLineShape: true,
    allowSwitchJunctionPoint: true,
    defaultJunctionPoint: 'border',
    defaultLineColor: '#2E74B5',
    defaultNodeColor: '#ffffff',
    defaultNodeBorderWidth: 2,
    defaultNodeBorderColor: '#2E74B5',
    defaultNodeFontColor: '#2E74B5',
    // 这里可以参考"Graph 图谱"中的参数进行设置
  }

  useEffect(() => {
    showGraph()
  }, [])

  const showGraph = async () => {
    const __graph_json_data: RGJsonData = {
      rootId: '2',
      nodes: [
        // 注意:在节点配置信息中,你的自定义属性需要像下面这样放到data标签中,否则数据会丢失

        { id: '1', text: '节点-1', data: { myicon: 'el-icon-star-on' } },
        { id: '2', text: '节点-2', data: { myicon: 'el-icon-setting' } },
        { id: '4', text: '节点-4', data: { myicon: 'el-icon-star-on' } },
        { id: '6', text: '节点-6', data: { myicon: 'el-icon-setting' } },
        { id: '7', text: '节点-7', data: { myicon: 'el-icon-setting' } },
        { id: '8', text: '节点-8', data: { myicon: 'el-icon-star-on' } },
        { id: '9', text: '节点-9', data: { myicon: 'el-icon-headset' } },
      ],
      lines: [
        { from: '1', to: '2', text: '投资' },
        { from: '4', to: '2', text: '高管' },
        { from: '6', to: '2', text: '高管' },
        { from: '7', to: '2', text: '高管' },
        { from: '8', to: '2', text: '高管' },
        { from: '9', to: '2', text: '高管' },
      ],
    }
    const graphInstance = graphRef.current?.getInstance()
    await graphInstance?.setJsonData(__graph_json_data)
    await graphInstance?.moveToCenter()
    await graphInstance?.zoomToFit()
  }

  const LineSlot: React.FC<RGLineSlotProps> = ({ link, line }) => {
    return <MyLine link={link} line={line} />
  }

  return (
    <div>
      <div style={{ height: '100vh' }}>
        <RelationGraph ref={graphRef} options={graphOptions} lineSlot={LineSlot} />
      </div>
    </div>
  )
}

export default AdvLineSlot2

adv-line-slot2.scss

scss
.c-node-menu-item {
  line-height: 30px;
  padding-left: 10px;
  cursor: pointer;
  color: #444444;
  font-size: 14px;
  border-top: #efefef solid 1px;
}
.c-node-menu-item:hover {
  background-color: rgba(66, 187, 66, 0.2);
}

Demo4AdvLineSlot2LineSlot.tsx

javascript
import React, {useContext, useEffect, useRef} from 'react';
import {
  RGJsonData,
  RGLine,
  RGLink,
  RGNode,
  RGOptions,
  RGUserEvent,
  RelationGraphComponent,
  RGLineSlotProps, RelationGraphStoreContext
} from 'relation-graph-react';
import './Demo4AdvLineSlot2LineSlot.scss';

interface Props {
  link: RGLink;
  line: RGLine;
}

const Demo4AdvLineSlot2LineSlot: React.FC<RGLineSlotProps> = ({ link, line }) => {
  const relationGraph = useContext(RelationGraphStoreContext);
  const options = relationGraph.options;
  const checked = line.id === options?.checkedLineId;

  const showStartArrow = line.showStartArrow ? relationGraph.getArrow(line, link, true) : undefined;
  const showEndArrow = line.showEndArrow ? relationGraph.getArrow(line, link, false) : undefined;

  const pathData = relationGraph.createLinePath(link, line, 0);

  const onClick = (lineObject: RGLine, linkObject: RGLink, $event: React.MouseEvent<SVGGElement>) => {
    relationGraph.onLineClick(lineObject, linkObject, $event as RGUserEvent);
  };

  let textTransform = ''
  try {
    textTransform = relationGraph.getTextTransform(
      line,
      pathData.textPosition.x,
      pathData.textPosition.y,
      pathData.textPosition.rotate
    )
  } catch (e) {
  }
  return (
    <g>
      {/* Regular way */}
      <path

        d={pathData?.path}
        className={`c-rg-line ${line.styleClass} ${checked ? 'c-rg-line-checked' : ''}`}
        stroke={checked ? options?.checkedLineColor : line.color ? line.color : options?.defaultLineColor}
        style={{
          opacity: line.opacity,
          strokeWidth: `${line.lineWidth ? line.lineWidth : options?.defaultLineWidth}px`,
        }}
        markerStart={showStartArrow}
        markerEnd={showEndArrow}
        fill="none"
        onClick={(event) => onClick(line, link, event)}
      />
      {options?.defaultShowLineLabel && options?.canvasZoom > 40 && (
        <g transform={textTransform}>
          <rect
            key={`t-${line.seeks_id}`}
            rx="15"
            ry="15"
            x="-30"
            y="-15"
            style={{
              opacity: line.opacity,
              fill: checked ? options?.checkedLineColor : line.fontColor ? line.fontColor : line.color,
            }}
            className="c-rg-line-text-bg"
            onClick={(event) => onClick(line, link, event)}
          />
          <text
            x="0"
            y="4"
            style={{
              opacity: line.opacity,
            }}
            className="c-rg-line-text"
            onClick={(event) => onClick(line, link, event)}
          >
            {line.text}
          </text>
        </g>
      )}
    </g>
  );
};

export default Demo4AdvLineSlot2LineSlot;

Demo4AdvLineSlot2LineSlot.scss

scss
.relation-graph {
  /*.RGLine-enter-active {*/
  /*transition: all .3s ease;*/
  /*}*/
  /*.RGLine-leave-active {*/
  /*transition: all .8s cubic-bezier(1.0, 0.5, 0.8, 1.0);*/
  /*}*/
  .c-rg-line-text-bg {
    fill: #2e74b5;
    width: 80px;
    height: 30px;
    font-size: 12px;
    border-radius: 10px;
  }
  .c-rg-line-text {
    fill: #ffffff;
    font-size: 12px;
  }
  .c-rg-line {
    z-index: 1000;
    fill-rule: nonzero;
    /*marker-end: url('#arrow');*/
    /* firefox bug fix - won't rotate at 90deg angles */
    /*-moz-transform: rotate(-89deg) translateX(-190px);*/
    /*animation-timing-function:linear;*/
    /*animation: ACTRGLineInit 1s;*/
  }
  .c-rg-line-tool {
    stroke-dasharray: 5, 5, 5;
  }
  .c-rg-line-flash {
    /* firefox bug fix - won't rotate at 90deg angles */
    -moz-transform: rotate(-89deg) translateX(-190px);
    animation-timing-function: linear;
    animation: ACTRGLineChecked 10s;
  }
  @keyframes ACTRGLineInit {
    from {
      stroke-dashoffset: 10px;
      stroke-dasharray: 20, 20, 20;
    }

    to {
      stroke-dashoffset: 0;
      stroke-dasharray: 0, 0, 0;
    }
  }
  @keyframes ACTRGLineChecked {
    from {
      stroke-dashoffset: 352px;
    }

    to {
      stroke-dashoffset: 0;
    }
  }
  .c-rg-line-checked {
    /*stroke: var(--stroke);*/
    /*marker-end: var(--markerEndChecked) !important;*/
    stroke-width: 2px;
    stroke-dasharray: 5, 5, 5;
    stroke-dashoffset: 3px;
    stroke-linecap: butt;
    /*stroke: #FD8B37;*/
    stroke-linejoin: bevel;
    /* firefox bug fix - won't rotate at 90deg angles */
    /*-moz-transform: rotate(-89deg) translateX(-190px);*/
    animation-timing-function: linear;
    animation: ACTRGLineChecked 10s;
  }
}

本站搭建特别鸣谢【茂神大佬】