广告

买白酒,找南将

Skip to content

禁用拖动、缩放画布;禁用拖动、选中节点

Vue2 版本

disable-effect.vue

javascript
<template>
  <div>
    <div style="height: calc(100vh)">
      <RelationGraph
        ref="graphRef"
        :options="graphOptions"
      >
        <template #graph-plug>
          <div class="c-my-panel">
            <div>
              <el-switch
                  size="mini"
                  style="margin-left:30px;"
                  v-model="graphOptions.disableZoom"
                  @change="upddateGraphOptions"
                  active-text="禁用缩放">
              </el-switch>
            </div>
            <div>
              <el-switch
                  size="mini"
                  style="margin-left:30px;"
                  v-model="graphOptions.disableDragCanvas"
                  @change="upddateGraphOptions"
                  active-text="禁用画布拖动">
              </el-switch>
            </div>
            <div>
              <el-switch
                  size="mini"
                  style="margin-left:30px;"
                  v-model="graphOptions.disableDragNode"
                  @change="upddateGraphOptions"
                  active-text="禁用节点拖动">
              </el-switch>
            </div>
            <div>
              <el-switch
                  size="mini"
                  style="margin-left:30px;"
                  v-model="graphOptions.disableNodeClickEffect"
                  @change="upddateGraphOptions"
                  active-text="禁用节点选中效果">
              </el-switch>
            </div>
          </div>
        </template>
      </RelationGraph>
    </div>
  </div>
</template>

<script>
// 如果您没有在main.js文件中使用Vue.use(RelationGraph); 就需要使用下面这一行代码来引入relation-graph
// import RelationGraph from "relation-graph";
const graphOptions = {
  useAnimationWhenRefresh: false,
  defaultFocusRootNode: false,
  disableLineClickEffect: true,
  allowSwitchLineShape: true,
  defaultNodeBorderWidth: 0,
  defaultLineColor: "#a9bdc6",
  defaultLineWidth:  2,
  defaultNodeShape: 0,
  disableZoom: true,
  disableDragCanvas: true,
  disableDragNode: true,
  disableNodeClickEffect: true,
  defaultPolyLineRadius: 10,
  defaultLineShape: 4,
  layout: {
    layoutName: "tree",
    from: 'top',
    levelDistance: '300,300,300,300',
    min_per_width: 100,
    max_per_width: 200
  },
  disableDefaultClickEffect: true
};
export default {
  name: 'Demo',
  components: { },
  data() {
    return {
      isShowCodePanel: false,
      graphOptions
    };
  },
  mounted() {
    setTimeout(() => {
      this.showGraph();
    }, 100)
  },
  methods: {
    showGraph() {
      const __graph_json_data = {"rootId":"a","nodes":[{"id":"a","text":"a"},{"id":"b","text":"b"},{"id":"b1","text":"b1"},{"id":"b2","text":"b2"},{"id":"b3","text":"b3"},{"id":"b4","text":"b4"},{"id":"b5","text":"b5"},{"id":"b6","text":"b6"},{"id":"c","text":"c"},{"id":"c1","text":"c1"},{"id":"c2","text":"c2"},{"id":"c3","text":"c3"},{"id":"d","text":"d"},{"id":"d1","text":"d1"},{"id":"d2","text":"d2"},{"id":"d3","text":"d3"},{"id":"d4","text":"d4"},{"id":"e","text":"e"},{"id":"e1","text":"e1"},{"id":"e2","text":"e2"}],"lines":[{"from":"a","to":"b"},{"from":"b","to":"b1"},{"from":"b","to":"b2"},{"from":"b","to":"b3"},{"from":"b","to":"b4"},{"from":"b","to":"b5"},{"from":"b","to":"b6"},{"from":"a","to":"c"},{"from":"c","to":"c1"},{"from":"c","to":"c2"},{"from":"c","to":"c3"},{"from":"a","to":"d"},{"from":"d","to":"d1"},{"from":"d","to":"d2"},{"from":"d","to":"d3"},{"from":"d","to":"d4"},{"from":"a","to":"e"},{"from":"e","to":"e1"},{"from":"e","to":"e2"}]};
      const graphRef = this.$refs.graphRef
      graphRef.setJsonData(__graph_json_data, (graphInstance) => {
          // 这些写上当图谱初始化完成后需要执行的代码.
        }
      );
    },
    async upddateGraphOptions() {
      const graphInstance = this.$refs.graphRef.getInstance();
      await graphInstance.setOptions(this.graphOptions, true);
      await graphInstance.refresh();
    }
  },
};
</script>

<style lang="scss" scoped>
.c-my-panel{
  width: 260px;
  position: absolute;
  left: 10px;
  top: 10px;
  border-radius: 10px;
  z-index: 800;
  background-color: #efefef;
  border: #eeeeee solid 1px;
  padding: 10px;
  .c-option-name{
    color: #666666;
    font-size: 14px;
    line-height: 40px;
    padding-left:10px;
    padding-right:10px;
  }
  & > div {
    padding: 5px;
  }
}
</style>

Vue3 版本

disable-effect.vue

javascript
<template>
  <div>
    <div style="height: calc(100vh)">
      <RelationGraph ref="graphRef" :options="graphOptions">
        <template #graph-plug>
          <div class="c-my-panel">
            <div>
              <el-switch

                size="small"
                style="margin-left:30px;"
                v-model="graphOptions.disableZoom"
                @change="updateGraphOptions"
                active-text="Disable Zoom"
              ></el-switch>
            </div>
            <div>
              <el-switch

                size="small"
                style="margin-left:30px;"
                v-model="graphOptions.disableDragCanvas"
                @change="updateGraphOptions"
                active-text="Disable Canvas Drag"
              ></el-switch>
            </div>
            <div>
              <el-switch

                size="small"
                style="margin-left:30px;"
                v-model="graphOptions.disableDragNode"
                @change="updateGraphOptions"
                active-text="Disable Node Drag"
              ></el-switch>
            </div>
            <div>
              <el-switch

                size="small"
                style="margin-left:30px;"
                v-model="graphOptions.disableNodeClickEffect"
                @change="updateGraphOptions"
                active-text="Disable Node Click Effect"
              ></el-switch>
            </div>
          </div>
        </template>
      </RelationGraph>
    </div>
  </div>
</template>

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

const graphOptions: RGOptions = {
    useAnimationWhenRefresh: false,
    defaultFocusRootNode: false,
    disableLineClickEffect: true,
    allowSwitchLineShape: true,
    defaultNodeBorderWidth: 0,
    defaultLineColor: '#a9bdc6',
    defaultLineWidth: 2,
    defaultNodeShape: 0,
    disableZoom: true,
    disableDragCanvas: true,
    disableDragNode: true,
    disableNodeClickEffect: true,
    defaultPolyLineRadius: 10,
    defaultLineShape: 4,
    layout: {
        layoutName: 'tree',
        from: 'top',
        levelDistance: '300,300,300,300',
        min_per_width: 100,
        max_per_width: 200

    }
};

const graphRef = ref<RelationGraphComponent>();

onMounted(() => {
    setTimeout(() => {
        showGraph();
    }, 100);
});

const showGraph = () => {
    const __graph_json_data: RGJsonData = {
        rootId: 'a',
        nodes: [
            { id: 'a', text: 'a' },
            { id: 'b', text: 'b' },
            { id: 'b1', text: 'b1' },
            { id: 'b2', text: 'b2' },
            { id: 'b3', text: 'b3' },
            { id: 'b4', text: 'b4' },
            { id: 'b5', text: 'b5' },
            { id: 'b6', text: 'b6' },
            { id: 'c', text: 'c' },
            { id: 'c1', text: 'c1' },
            { id: 'c2', text: 'c2' },
            { id: 'c3', text: 'c3' },
            { id: 'd', text: 'd' },
            { id: 'd1', text: 'd1' },
            { id: 'd2', text: 'd2' },
            { id: 'd3', text: 'd3' },
            { id: 'd4', text: 'd4' },
            { id: 'e', text: 'e' },
            { id: 'e1', text: 'e1' },
            { id: 'e2', text: 'e2' }
        ],
        lines: [
            { from: 'a', to: 'b' },
            { from: 'b', to: 'b1' },
            { from: 'b', to: 'b2' },
            { from: 'b', to: 'b3' },
            { from: 'b', to: 'b4' },
            { from: 'b', to: 'b5' },
            { from: 'b', to: 'b6' },
            { from: 'a', to: 'c' },
            { from: 'c', to: 'c1' },
            { from: 'c', to: 'c2' },
            { from: 'c', to: 'c3' },
            { from: 'a', to: 'd' },
            { from: 'd', to: 'd1' },
            { from: 'd', to: 'd2' },
            { from: 'd', to: 'd3' },
            { from: 'd', to: 'd4' },
            { from: 'a', to: 'e' },
            { from: 'e', to: 'e1' },
            { from: 'e', to: 'e2' }
        ]
    };

    graphRef.value!.setJsonData(__graph_json_data, (graphInstance) => {
        // 这些写上当图谱初始化完成后需要执行的代码.
    });
};

const updateGraphOptions = async () => {
    const graphInstance = graphRef.value?.getInstance();
    if (graphInstance) {
        await graphInstance.setOptions(graphOptions, true);
        await graphInstance.refresh();
    }
};
</script>

<style scoped lang="scss">
.c-my-panel {
  width: 260px;
  position: absolute;
  left: 10px;
  top: 10px;
  border-radius: 10px;
  z-index: 800;
  background-color: #efefef;
  border: #eeeeee solid 1px;
  padding: 10px;
  .c-option-name {
    color: #666666;
    font-size: 14px;
    line-height: 40px;
    padding-left: 10px;
    padding-right: 10px;
  }
  & > div {
    padding: 5px;
  }
}
</style>

React 版本

disable-effect.tsx

javascript
import React, {useEffect, useRef, useState} from 'react';
import RelationGraphReact, { RGOptions, RGJsonData, RGNode, RGLine, RGLink, RGUserEvent, RelationGraphComponent } from 'relation-graph-react';
import {MySwitch} from "./RGDemoComponents/MyUIComponents";

const MyComponent = () => {
  const graphRef = useRef<RelationGraphComponent|null>(null);
  const [graphOptions, setGraphOptions] = useState<RGOptions>({
    useAnimationWhenRefresh: false,
    defaultFocusRootNode: false,
    defaultNodeBorderWidth: 0,
    defaultLineColor: '#a9bdc6',
    defaultLineWidth: 2,
    defaultNodeShape: 0,
    disableZoom: true,
    disableDragCanvas: true,
    disableDragNode: true,
    defaultPolyLineRadius: 10,
    defaultLineShape: 4,
    layout: {
      layoutName: 'tree',
      from: 'top',
      levelDistance: '300,300,300,300',
      min_per_width: 100,
      max_per_width: 200
    },
    disableLineClickEffect: true,
    disableNodeClickEffect: true
  });
  useEffect(() => {
    const showGraph = () => {
      const __graph_json_data: RGJsonData = {
        rootId: 'a',
        nodes: [
          { id: 'a', text: 'a' },
          { id: 'b', text: 'b' },
          { id: 'b1', text: 'b1' },
          { id: 'b2', text: 'b2' },
          { id: 'b3', text: 'b3' },
          { id: 'b4', text: 'b4' },
          { id: 'b5', text: 'b5' },
          { id: 'b6', text: 'b6' },
          { id: 'c', text: 'c' },
          { id: 'c1', text: 'c1' },
          { id: 'c2', text: 'c2' },
          { id: 'c3', text: 'c3' },
          { id: 'd', text: 'd' },
          { id: 'd1', text: 'd1' },
          { id: 'd2', text: 'd2' },
          { id: 'd3', text: 'd3' },
          { id: 'd4', text: 'd4' },
          { id: 'e', text: 'e' },
          { id: 'e1', text: 'e1' },
          { id: 'e2', text: 'e2' }
        ],
        lines: [
          { from: 'a', to: 'b' },
          { from: 'b', to: 'b1' },
          { from: 'b', to: 'b2' },
          { from: 'b', to: 'b3' },
          { from: 'b', to: 'b4' },
          { from: 'b', to: 'b5' },
          { from: 'b', to: 'b6' },
          { from: 'a', to: 'c' },
          { from: 'c', to: 'c1' },
          { from: 'c', to: 'c2' },
          { from: 'c', to: 'c3' },
          { from: 'a', to: 'd' },
          { from: 'd', to: 'd1' },
          { from: 'd', to: 'd2' },
          { from: 'd', to: 'd3' },
          { from: 'd', to: 'd4' },
          { from: 'a', to: 'e' },
          { from: 'e', to: 'e1' },
          { from: 'e', to: 'e2' }
        ]
      };

      graphRef.current?.setJsonData(__graph_json_data, (graphInstance) => {
        // 这些写上当图谱初始化完成后需要执行的代码.
      });
    };

    setTimeout(() => {
      showGraph();
    }, 100);
  }, []);

  const updateGraphOptions = async () => {
    const graphInstance = graphRef.current!.getInstance();
    if (graphInstance) {
      await graphInstance.setOptions(graphOptions, true);
      await graphInstance.refresh();
    }
  };
  useEffect(() => {
    updateGraphOptions();
  }, [graphOptions]);
  return (
    <div>
      <div style={{ height: '100vh' }}>
        <div className="w-96 rounded-lg absolute left-20 top-20 z-20 p-4 bg-white border-solid border-2 border-black shadow-lg">
          <div className="flex gap-2">
            <MySwitch
              currentValue={graphOptions.disableZoom!}
              onChange={(newValue) => {
                graphOptions.disableZoom = newValue;
                setGraphOptions(JSON.parse(JSON.stringify(graphOptions)));
              }}
            />
            <div className="w-72">Disable Zoom</div>
          </div>
          <div className="flex gap-2">
            <MySwitch
              currentValue={graphOptions.disableDragCanvas!}
              onChange={(newValue) => {
                graphOptions.disableDragCanvas = newValue;
                setGraphOptions(JSON.parse(JSON.stringify(graphOptions)));
              }}
            />
            <div className="w-72">Disable Canvas Drag</div>
          </div>
          <div className="flex gap-2">
            <MySwitch
              currentValue={graphOptions.disableDragNode!}
              onChange={(newValue) => {
                graphOptions.disableDragNode = newValue;
                setGraphOptions(JSON.parse(JSON.stringify(graphOptions)));
              }}
            />
            <div className="w-72">Disable Node Drag</div>
          </div>
          <div className="flex gap-2">
            <MySwitch
              currentValue={graphOptions.disableNodeClickEffect!}
              onChange={(newValue) => {
                graphOptions.disableNodeClickEffect = newValue;
                setGraphOptions(JSON.parse(JSON.stringify(graphOptions)));
              }}
            />
            <div className="w-72">Disable Node Click Effect</div>
          </div>
        </div>
        <RelationGraphReact ref={graphRef} options={graphOptions}>
        </RelationGraphReact>
      </div>
    </div>
  );
};

export default MyComponent;

📂 RGDemoComponents

MyUIComponents.tsx

javascript
import React from "react";

export interface MySelectorProps {
  small?: boolean
  currentValue: string|number
  data:{value: string|number, text:string}[]
  onChange: (newValue:string|number, label:string) => void
}
export const MySelector:React.FC<MySelectorProps> = ({small, data, onChange, currentValue}) => {
  return (
    <div className="flex flex-wrap justify-center rounded-lg border border-gray-900 overflow-hidden">
      {
        data.map(item =>
          <div key={item.value}
               className={`border-r w-auto text-xs cursor-pointer whitespace-nowrap ${currentValue === item.value && 'bg-blue-500 text-white'} ${small?' px-2 h-6 leading-6':'h-8 px-3 leading-8'}`}
               onClick={() => {onChange(item.value, item.text);}}
          >
          {item.text}
        </div>)
      }
    </div>
  );
};
export interface MySwitchProps {
  currentValue: boolean
  onChange: (newValue:boolean) => void
}
export const MySwitch:React.FC<MySwitchProps> = ({onChange, currentValue}) => {
  return (
    <div className={`w-14 flex rounded-full border p-0.5 ${currentValue ? 'justify-end border-blue-500' : 'justify-start border-gray-500'}`}>
      <div
        className={`w-8 h-5 leading-8 rounded-full w-auto px-3 text-xs cursor-pointer whitespace-nowrap ${currentValue ? 'bg-blue-500' : 'bg-gray-500'}`} onClick={() => {onChange(!currentValue);}}>
      </div>
    </div>
  );
};
export interface MySliderProps {
  min: number
  max: number
  step: number
  currentValue: number
  onChange: (newValue:number) => void
}
export const MySlider:React.FC<MySliderProps> = ({min, max, step, currentValue, onChange}) => {
  return (
    <div>
      <input
        type="range"
        className="w-72"
        min={min}
        max={max}
        step={step}
        value={currentValue}
        onChange={(e) => { onChange(parseFloat(e.target.value))}}
      />
    </div>
  );
};

export interface MyRangeSliderProps {
  min: number
  max: number
  step: number
  currentValue: [number, number]
  onChange: (newValue:[number, number]) => void
}
export const MyRangeSlider:React.FC<MyRangeSliderProps> = ({min, max, step, currentValue, onChange}) => {
  return (
    <div className="w-72">
      <div>Min:</div>
      <input
        type="range"
        className="w-full"
        min={min}
        max={max}
        step={step}
        value={currentValue[0]}
        onChange={(e) => { if (parseFloat(e.target.value) < currentValue[1]) onChange([parseFloat(e.target.value), currentValue[1]])}}
      />
      <div>Max:</div>
      <input
        type="range"
        className="w-full"
        min={min}
        max={max}
        step={step}
        value={currentValue[1]}
        onChange={(e) => { if (parseFloat(e.target.value) > currentValue[0]) onChange([currentValue[0], parseFloat(e.target.value)])}}
      />
    </div>
  );
};
export interface MyButtonProps {
  onClick: () => void
  disabled?: boolean
}
export const MyButton:React.FC<MyButtonProps> = ({children, onClick, disabled}) => {
  return (
    <button className={`mr-2 px-2 py-1 rounded ${disabled===true ? 'bg-gray-300 text-black cursor-not-allowed':'bg-blue-500 hover:bg-blue-700 text-white'}`}
            onClick={()=>{onClick();}}>{children}</button>
  );
};
export interface MyLinkButtonProps {
  onClick: () => void
}
export const MyLinkButton:React.FC<MyLinkButtonProps> = ({children, onClick}) => {
  return (
    <div className="text-blue-600 cursor-pointer underline decoration-1" onClick={()=>{onClick();}}>
      {children}
    </div>
  );
};
export interface MyCheckBoxProps {
  currentValue: string|number
  data:{value: string|number, text:string}[]
  onChange: (newValue:string|number, label:string) => void
}
export const MyCheckBox:React.FC<MyCheckBoxProps> = ({data, onChange, currentValue}) => {
  // console.log(data);
  return (
    <div className="flex gap-2 flex-wrap">
      {
        data.map(thisItem =>
          <div
            key={thisItem.value}
            className={`px-1 py-0.5 flex justify-center place-items-center rounded-sm text-sm cursor-pointer  hover:bg-gray-300 ${currentValue === thisItem.value ? 'text-blue-600':'text-gray-500'}`}
            onClick={()=>{onChange(thisItem.value, thisItem.text);}}
          >
            <div className={`w-4 h-4 mr-1 rounded-full ${currentValue === thisItem.value ? 'border border-blue-500 bg-blue-500 text-blue-600':'border border-gray-500 text-gray-500'}`}></div>
            {thisItem.text}
          </div>
        )
      }
    </div>
  );
};
export interface CheckboxOption {
  value: string | number;
  text: string;
}

export interface MyMultiCheckBoxProps {
  currentValue: (string | number)[];
  checkboxOptions: CheckboxOption[];
  onChange: (newValue: (string | number)[]) => void;
}
export const MyMultiCheckBox:React.FC<MyMultiCheckBoxProps> = ({data, onChange, currentValue}) => {
  // console.log(data);
  const onClickItem = (item: string | number) => {
    const newValue = currentValue.includes(item)
      ? currentValue.filter((value) => value !== item)
      : [...currentValue, item];
    onChange(newValue);
  };
  return (
    <div className="flex gap-2 flex-wrap">
      {
        data.map(thisItem =>
          <div
            key={thisItem.value}
            className={`px-1 py-0.5 flex justify-center place-items-center rounded-sm text-sm cursor-pointer  hover:bg-gray-300 ${currentValue === thisItem.value ? 'text-blue-600':'text-gray-500'}`}
            onClick={()=>{onClickItem(thisItem.value);}}
          >
            <div className={`w-4 h-4 mr-1 rounded-full ${currentValue.includes(thisItem.value) ? 'border border-blue-500 bg-blue-500 text-blue-600':'border border-gray-500 text-gray-500'}`}></div>
            {thisItem.text}
          </div>
        )
      }
    </div>
  );
};
export const ElMessage = (messageObject) => {
  console.warn(messageObject);
}
export const ElNotification = (messageObject) => {
  console.warn(messageObject);
}

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