广告

买白酒,找南将

Skip to content

节点-内容及样式 5

Vue2 版本

models-relationship.vue

javascript
<template>
  <div>
    <div style="height:calc(100vh);background-color: #222222;">
      <RelationGraph ref="graphRef" :options="graphOptions" :on-node-click="onNodeClick">
        <template #canvas-plug>
          <!--- You can put any content here, and it will be displayed in behind of the node. --->
        </template>
        <!--- You can put any content here, and it will be displayed in above of the node. --->
        <!-- node slot -->
        <template #node="{ node }">
          <SingleAIModel :node="node" :dataTypeColorMap="dataTypeColorMap" />
        </template>
      </RelationGraph>
    </div>
  </div>
</template>

<script>
// 如果您没有在main.js文件中使用Vue.use(RelationGraph); 就需要使用下面这一行代码来引入relation-graph
// import RelationGraph from 'relation-graph';
import SingleAIModel from './VipDemo4AIModel.vue'

export default {
  name: 'AIModelRelationship',
  components: { SingleAIModel },
  data() {
    return {
      isShowCodePanel: false,
      dataTypeColorMap: {
        1: '#2cbb03',
        2: '#1da9f5',
        3: '#ff8080',
        4: '#f39930',
        5: '#f39930',
        6: '#f12de0',
        7: '#5b05f1',
        8: '#f39930',
        9: '#ffffff',
      },
      graphOptions: {
        debug: false,
        defaultJunctionPoint: 'border',
        backgroundColor: 'transparent',
        defaultNodeColor: 'rgba(100, 100, 100, 0.5)',
        placeOtherNodes: false,
        placeSingleNode: false,
        defaultLineMarker: {
          markerWidth: 20,
          markerHeight: 20,
          refX: 3,
          refY: 3,
          data: 'M 0 0, V 6, L 4 3, Z',
        },
        layout: {
          layoutName: 'fixed',
        },
      },
    }
  },
  mounted() {
    this.showGraph()
    // You can adjust the position of the nodes on the page, and then use the following code to print or save the positions of all current nodes, and use these positions the next time the page is opened.
    // setInterval(() => {
    //   const graphInstance = this.$refs.graphRef.getInstance();
    //   const nodePositons = graphInstance.getNodes().map(n => ({id:n.id,x:n.x,y:n.y}));
    //   console.log(JSON.stringify(nodePositons));
    // }, 5000);;
  },
  methods: {
    showGraph() {
      const __graph_json_data = {
        rootId: 'a',
        nodes: [
          {
            id: 'm-a',
            nodeShape: 1,
            text: 'Load Video',
            data: {
              options: [
                { name: 'video_path', type: 'string', value: '/data/My-World-1080P.mp4' },
                { name: 'force_rate', type: 'number', value: '10' },
                { name: 'force_size', type: 'boolean', value: '0' },
                { name: 'frame_load_cap', type: 'number', value: '2048' },
                { name: 'skip_first_frames', type: 'number', value: '0' },
              ],
              input: [],
              output: [
                { name: 'current_image', type: 1 },
                { name: 'frame_count', type: 1 },
              ],
            },
          },
          {
            id: 'm-b',
            nodeShape: 1,
            text: 'Preview Image',
            data: {
              options: [],
              input: [{ name: 'image', type: 1 }],
              output: [],
            },
          },
          {
            id: 'm-c',
            nodeShape: 1,
            text: 'Empty Latent Image',
            data: {
              options: [],
              input: [
                { name: 'batch_size', type: 2 },
                { name: 'width', type: 2 },
                { name: 'height', type: 2 },
              ],
              output: [{ name: 'LATENT', type: 5 }],
            },
          },
          {
            id: 'm-d',
            nodeShape: 1,
            text: 'Primitive',
            data: {
              options: [
                { name: 'value', type: 'number', value: '512' },
                { name: 'control_affer_generate', type: 'string', value: 'fixed' },
              ],
              input: [],
              output: [{ name: 'INT', type: 2 }],
            },
          },
          {
            id: 'm-dd',
            nodeShape: 1,
            text: 'Primitive',
            data: {
              options: [
                { name: 'value', type: 'number', value: '24' },
                { name: 'control_affer_generate', type: 'string', value: 'fixed' },
              ],
              input: [],
              output: [{ name: 'INT', type: 2 }],
            },
          },
          {
            id: 'm-e',
            nodeShape: 1,
            text: 'Primitive',
            data: {
              options: [
                { name: 'value', type: 'number', value: '688' },
                { name: 'control_affer_generate', type: 'string', value: 'fixed' },
              ],
              input: [],
              output: [{ name: 'INT', type: 2 }],
            },
          },
          {
            id: 'm-f',
            nodeShape: 1,
            text: 'AnimateDiff Loader',
            color: 'rgba(139,108,46,0.58)',
            data: {
              options: [
                { name: 'model_name', type: 'string', value: 'diff-v1.safetensors' },
                { name: 'beta_schedule', type: 'string', value: 'sqrt_linearr(AnimateDiff)' },
              ],
              input: [
                { name: 'model', type: 0 },
                { name: 'context_options', type: 8 },
                { name: 'montion_lora', type: 9 },
              ],
              output: [{ name: 'MODEL', type: 0 }],
            },
          },
          {
            id: 'm-g',
            nodeShape: 1,
            text: 'KSampler',
            data: {
              options: [
                { name: 'steps', type: 'number', value: '30' },
                { name: 'cfg', type: 'number', value: '8.0' },
                { name: 'sampler_name', type: 'string', value: 'dpmpp_sde' },
                { name: 'scheduler', type: 'string', value: 'karras' },
                { name: 'denoise', type: 'number', value: '1.00' },
              ],
              input: [
                { name: 'model', type: 0 },
                { name: 'positive', type: 0 },
                { name: 'negative', type: 0 },
                { name: 'latent_image', type: 1 },
                { name: 'seed', type: 2 },
              ],
              output: [{ name: 'LATENT', type: 5 }],
            },
          },
          {
            id: 'm-h',
            nodeShape: 1,
            text: 'CLIP Set Last Layer',
            data: {
              options: [{ name: 'stop_at_clip_layer', type: 'number', value: '-2' }],
              input: [{ name: 'clip', type: 3 }],
              output: [{ name: 'CLIP', type: 5 }],
            },
          },
          {
            id: 'm-i',
            nodeShape: 1,
            text: 'CLIP Text Encode(Prompt)',
            color: 'rgba(125,55,53,0.6)',
            data: {
              options: [{ name: 'stop_at_clip_layer', type: 'number', value: '-2' }],
              input: [{ name: 'clip', type: 3 }],
              output: [{ name: 'CONDITIONING', type: 5 }],
            },
          },
          {
            id: 'm-j',
            nodeShape: 1,
            text: 'Load Checkpoint',
            data: {
              options: [{ name: 'ckp', type: 'string', value: 'vision51_v51VAE_safetensors' }],
              input: [],
              output: [
                { name: 'MODEL', type: 5 },
                { name: 'CLIP', type: 3 },
                { name: 'VAE', type: 6 },
              ],
            },
          },
          {
            id: 'm-k',
            nodeShape: 1,
            text: 'BatchPrompSchedule',
            data: {
              options: [
                {
                  name: '_textarea',
                  type: 'string',
                  value: `"0" : "yummy food, seafood pizza",
                  "83" : "student's desk, books, pen, sketchbook, earaser, ruler",
                  "167" : "jewelery, diamonds, crystals, marbles",
                  `,
                },
                { name: 'pre_text', type: 'string', value: 'photoreealistic photography, to' },
                { name: 'app_text', type: 'string', value: 'black white interior,top angl' },
                { name: 'pw_a', type: 'number', value: '0.0' },
                { name: 'pw_b', type: 'number', value: '0.0' },
                { name: 'pw_c', type: 'number', value: '0.0' },
                { name: 'pw_d', type: 'number', value: '0.0' },
              ],
              input: [
                { name: 'clip', type: 5 },
                { name: 'max_frames', type: 5 },
              ],
              output: [{ name: 'CONDITIONING', type: 5 }],
            },
          },
          {
            id: 'm-l',
            nodeShape: 1,
            text: 'Uniform Context Options',
            color: 'rgba(139,108,46,0.58)',
            data: {
              options: [
                { name: 'context_length', type: 'number', value: '0.0' },
                { name: 'context_stride', type: 'number', value: '0.0' },
                { name: 'context_overlap', type: 'number', value: '0.0' },
                { name: 'context_schedule', type: 'string', value: 'uniform' },
                { name: 'closed_loop', type: 'boolean', value: 'false' },
              ],
              input: [],
              output: [{ name: 'CONTEXT_OPTIONS', type: 5 }],
            },
          },
          {
            id: 'm-m',
            nodeShape: 1,
            text: 'Apply ControlNet(Advanced)',
            color: 'rgba(141,32,141,0.5)',
            data: {
              options: [
                { name: 'strength', type: 'number', value: '1.0' },
                { name: 'start_percent', type: 'number', value: '0.000' },
                { name: 'end_percent', type: 'number', value: '1.000' },
              ],
              input: [
                { name: 'positive', type: 5 },
                { name: 'negative', type: 5 },
                { name: 'control_net', type: 4 },
                { name: 'image', type: 4 },
              ],
              output: [
                { name: 'positive', type: 5 },
                { name: 'negative', type: 5 },
              ],
            },
          },
          {
            id: 'm-n',
            nodeShape: 1,
            text: 'Apply ControlNet(Advanced)',
            color: 'rgba(141,32,141,0.5)',
            data: {
              options: [
                { name: 'strength', type: 'number', value: '0.50' },
                { name: 'start_percent', type: 'number', value: '0.002' },
                { name: 'end_percent', type: 'number', value: '0.663' },
              ],
              input: [
                { name: 'positive', type: 5 },
                { name: 'negative', type: 5 },
                { name: 'control_net', type: 4 },
                { name: 'image', type: 4 },
              ],
              output: [
                { name: 'positive', type: 5 },
                { name: 'negative', type: 5 },
              ],
            },
          },
          {
            id: 'm-o',
            nodeShape: 1,
            text: 'Apply ControlNet(Advanced)',
            color: 'rgba(141,32,141,0.5)',
            data: {
              options: [
                { name: 'strength', type: 'number', value: '0.45' },
                { name: 'start_percent', type: 'number', value: '0.002' },
                { name: 'end_percent', type: 'number', value: '0.550' },
              ],
              input: [
                { name: 'positive', type: 5 },
                { name: 'negative', type: 5 },
                { name: 'control_net', type: 4 },
                { name: 'image', type: 4 },
              ],
              output: [
                { name: 'positive', type: 5 },
                { name: 'negative', type: 5 },
              ],
            },
          },
          {
            id: 'm-mc',
            nodeShape: 1,
            text: 'Load ControlNet Model(Advanced)',
            color: 'rgba(141,32,141,0.5)',
            data: {
              options: [
                {
                  name: 'control_net_name',
                  type: 'string',
                  value: 'control_v11p_sd15_openpose.pth',
                },
              ],
              input: [],
              output: [{ name: 'CONTROL_NET', type: 1 }],
            },
          },
          {
            id: 'm-nc',
            nodeShape: 1,
            text: 'Load ControlNet Model(Advanced)',
            color: 'rgba(141,32,141,0.5)',
            data: {
              options: [
                { name: 'control_net_name', type: 'string', value: 'control_v11p_sd15_depth.pth' },
              ],
              input: [],
              output: [{ name: 'CONTROL_NET', type: 1 }],
            },
          },
          {
            id: 'm-oc',
            nodeShape: 1,
            text: 'Load ControlNet Model(Advanced)',
            color: 'rgba(141,32,141,0.5)',
            data: {
              options: [
                {
                  name: 'control_net_name',
                  type: 'string',
                  value: 'control_v11p_sd15_lineart.pth',
                },
              ],
              input: [],
              output: [{ name: 'CONTROL_NET', type: 1 }],
            },
          },
          {
            id: 'm-p',
            nodeShape: 1,
            text: 'Load VAE',
            data: {
              options: [
                { name: 'vae_name', type: 'string', value: 'vae-ft-mse-840000.safetensors' },
              ],
              input: [],
              output: [{ name: 'VAE', type: 5 }],
            },
          },
          {
            id: 'm-q',
            nodeShape: 1,
            text: 'VAE Decode',
            data: {
              options: [],
              input: [
                { name: 'samples', type: 5 },
                { name: 'vae', type: 5 },
              ],
              output: [{ name: 'IMAGE', type: 5 }],
            },
          },
          {
            id: 'm-r',
            nodeShape: 1,
            text: 'Video Combine',
            color: 'rgba(50,58,132,0.65)',
            data: {
              options: [
                { name: 'frome_rate', type: 'number', value: '10' },
                { name: 'loop_count', type: 'number', value: '0' },
              ],
              input: [{ name: 'images', type: 5 }],
              output: [{ name: 'GIF', type: 5 }],
            },
          },
          {
            id: 'm-final',
            nodeShape: 1,
            text: 'Video Combine',
            color: 'rgba(50,58,132,0.65)',
            data: {
              options: [
                { name: '_image', type: 'string', value: '/demo-images/open-all-close-all.gif' },
              ],
              input: [{ name: 'GIF', type: 5 }],
              output: [],
            },
          },
          {
            id: 'm-s',
            nodeShape: 1,
            text: 'Normal Lineart',
            data: {
              options: [
                { name: 'coarse', type: 'string', value: 'disable' },
                { name: 'resolution', type: 'number', value: '512' },
              ],
              input: [{ name: 'image', type: 5 }],
              output: [{ name: 'IMAGE', type: 6 }],
            },
          },
          {
            id: 'm-t',
            nodeShape: 1,
            text: 'Zoe - Depth Map',
            data: {
              options: [{ name: 'resolution', type: 'number', value: '512' }],
              input: [{ name: 'image', type: 5 }],
              output: [{ name: 'IMAGE', type: 6 }],
            },
          },
          {
            id: 'm-u',
            nodeShape: 1,
            text: 'MediaPipe-FaceMeshPreprocessor',
            data: {
              options: [
                { name: 'max_faces', type: 'number', value: '2' },
                { name: 'min_confidence', type: 'number', value: '0.50' },
                { name: 'resolution', type: 'number', value: '512' },
              ],
              input: [{ name: 'image', type: 5 }],
              output: [{ name: 'IMAGE', type: 6 }],
            },
          },
          {
            id: 'm-v',
            nodeShape: 1,
            text: 'Upscale Image',
            data: {
              options: [
                { name: 'upscale_method', type: 'string', value: 'nearest-exact' },
                { name: 'ccrop', type: 'string', value: 'center' },
              ],
              input: [
                { name: 'image', type: 5 },
                { name: 'width', type: 5 },
                { name: 'height', type: 5 },
              ],
              output: [{ name: 'IMAGE', type: 6 }],
            },
          },
        ],
        lines: [],
        elementLines: [],
      }
      const modelRelations = [
        {
          output_model: 'm-d',
          output_param_name: 'INT',
          input_model: 'm-c',
          input_param_name: 'width',
        },
        {
          output_model: 'm-e',
          output_param_name: 'INT',
          input_model: 'm-c',
          input_param_name: 'height',
        },
        {
          output_model: 'm-dd',
          output_param_name: 'INT',
          input_model: 'm-g',
          input_param_name: 'seed',
        },
        {
          output_model: 'm-a',
          output_param_name: 'frame_count',
          input_model: 'm-c',
          input_param_name: 'batch_size',
        },
        {
          output_model: 'm-a',
          output_param_name: 'current_image',
          input_model: 'm-b',
          input_param_name: 'image',
        },
        {
          output_model: 'm-c',
          output_param_name: 'LATENT',
          input_model: 'm-g',
          input_param_name: 'latent_image',
        },
        {
          output_model: 'm-f',
          output_param_name: 'MODEL',
          input_model: 'm-g',
          input_param_name: 'model',
        },
        {
          output_model: 'm-j',
          output_param_name: 'CLIP',
          input_model: 'm-h',
          input_param_name: 'clip',
        },
        {
          output_model: 'm-j',
          output_param_name: 'CLIP',
          input_model: 'm-i',
          input_param_name: 'clip',
        },
        {
          output_model: 'm-j',
          output_param_name: 'MODEL',
          input_model: 'm-f',
          input_param_name: 'model',
        },
        {
          output_model: 'm-a',
          output_param_name: 'frame_count',
          input_model: 'm-k',
          input_param_name: 'max_frames',
        },
        {
          output_model: 'm-h',
          output_param_name: 'CLIP',
          input_model: 'm-k',
          input_param_name: 'clip',
        },
        {
          output_model: 'm-l',
          output_param_name: 'CONTEXT_OPTIONS',
          input_model: 'm-f',
          input_param_name: 'context_options',
        },
        {
          output_model: 'm-i',
          output_param_name: 'CONDITIONING',
          input_model: 'm-m',
          input_param_name: 'negative',
        },
        {
          output_model: 'm-k',
          output_param_name: 'CONDITIONING',
          input_model: 'm-m',
          input_param_name: 'positive',
        },
        {
          output_model: 'm-m',
          output_param_name: 'negative',
          input_model: 'm-n',
          input_param_name: 'negative',
        },
        {
          output_model: 'm-m',
          output_param_name: 'positive',
          input_model: 'm-n',
          input_param_name: 'positive',
        },
        {
          output_model: 'm-n',
          output_param_name: 'negative',
          input_model: 'm-o',
          input_param_name: 'negative',
        },
        {
          output_model: 'm-n',
          output_param_name: 'positive',
          input_model: 'm-o',
          input_param_name: 'positive',
        },
        {
          output_model: 'm-mc',
          output_param_name: 'CONTROL_NET',
          input_model: 'm-m',
          input_param_name: 'control_net',
        },
        {
          output_model: 'm-nc',
          output_param_name: 'CONTROL_NET',
          input_model: 'm-n',
          input_param_name: 'control_net',
        },
        {
          output_model: 'm-oc',
          output_param_name: 'CONTROL_NET',
          input_model: 'm-o',
          input_param_name: 'control_net',
        },
        {
          output_model: 'm-p',
          output_param_name: 'VAE',
          input_model: 'm-q',
          input_param_name: 'vae',
        },
        {
          output_model: 'm-g',
          output_param_name: 'LATENT',
          input_model: 'm-q',
          input_param_name: 'samples',
        },
        {
          output_model: 'm-q',
          output_param_name: 'IMAGE',
          input_model: 'm-r',
          input_param_name: 'images',
        },
        {
          output_model: 'm-s',
          output_param_name: 'IMAGE',
          input_model: 'm-o',
          input_param_name: 'image',
        },
        {
          output_model: 'm-t',
          output_param_name: 'IMAGE',
          input_model: 'm-n',
          input_param_name: 'image',
        },
        {
          output_model: 'm-u',
          output_param_name: 'IMAGE',
          input_model: 'm-m',
          input_param_name: 'image',
        },
        {
          output_model: 'm-u',
          output_param_name: 'IMAGE',
          input_model: 'm-t',
          input_param_name: 'image',
        },
        {
          output_model: 'm-v',
          output_param_name: 'IMAGE',
          input_model: 'm-u',
          input_param_name: 'image',
        },
        {
          output_model: 'm-v',
          output_param_name: 'IMAGE',
          input_model: 'm-s',
          input_param_name: 'image',
        },
        {
          output_model: 'm-a',
          output_param_name: 'current_image',
          input_model: 'm-v',
          input_param_name: 'image',
        },
        {
          output_model: 'm-d',
          output_param_name: 'INT',
          input_model: 'm-v',
          input_param_name: 'width',
        },
        {
          output_model: 'm-e',
          output_param_name: 'INT',
          input_model: 'm-v',
          input_param_name: 'height',
        },
        {
          output_model: 'm-r',
          output_param_name: 'GIF',
          input_model: 'm-final',
          input_param_name: 'GIF',
        },
      ]
      const getModel = (modelId) => {
        return __graph_json_data.nodes.find((model) => model.id === modelId)
      }
      const getModelOutputParam = (model, outputParamName) => {
        return model.data.output.find((param) => param.name === outputParamName)
      }
      const getModelInputParam = (model, inputParamName) => {
        return model.data.input.find((param) => param.name === inputParamName)
      }
      __graph_json_data.elementLines = modelRelations.map((relation) => {
        const outputModel = getModel(relation.output_model)
        const outputParam = getModelOutputParam(outputModel, relation.output_param_name)
        const inputModel = getModel(relation.input_model)
        const inputParam = getModelInputParam(inputModel, relation.input_param_name)
        inputParam.type = outputParam.type // fix data bug
        return {
          from: `O_${relation.output_model}-${relation.output_param_name}`,
          fromJunctionPoint: 'right',
          to: `I_${relation.input_model}-${relation.input_param_name}`,
          toJunctionPoint: 'left',
          text: '',
          lineShape: 6,
          lineWidth: 3,
          color: this.dataTypeColorMap[outputParam.type],
        }
      })
      // Using the position set for the node, it can be adjusted, saved to the database, and then reloaded to be displayed here.
      const nodesPosition = [
        { id: 'm-a', x: -930.1706372723322, y: -357.62950890917 },
        { id: 'm-b', x: -456.66745683694836, y: -468.67267196080763 },
        { id: 'm-c', x: 18.272727272727273, y: -297.27272727272725 },
        { id: 'm-d', x: -459.07692307692304, y: -383.7692307692307 },
        { id: 'm-dd', x: -458.12172573189525, y: -81.8335901386749 },
        { id: 'm-e', x: -459.9230769230769, y: -240.23076923076923 },
        { id: 'm-f', x: 502.27272727272725, y: 150 },
        { id: 'm-g', x: 982.421052631579, y: -146.69856459330143 },
        { id: 'm-h', x: 498.14285714285717, y: -301.29870129870125 },
        { id: 'm-i', x: 503, y: -158.63636363636365 },
        { id: 'm-j', x: 15.420875420875419, y: -173.28956228956224 },
        { id: 'm-k', x: -22.447552447552397, y: -3.681688681688712 },
        { id: 'm-l', x: -456.8423948932424, y: 72.69742719892274 },
        { id: 'm-m', x: 16.882352941176407, y: -545.5347593582887 },
        { id: 'm-n', x: 495.5050505050505, y: -550.5555555555555 },
        { id: 'm-o', x: 976.8311688311688, y: -554.1428571428571 },
        { id: 'm-mc', x: 18.012987012987004, y: -684.6883116883117 },
        { id: 'm-nc', x: 492.012987012987, y: -688.3766233766235 },
        { id: 'm-oc', x: 971.7272727272727, y: -692.1038961038962 },
        { id: 'm-p', x: 502.5717703349282, y: -12.715311004784702 },
        { id: 'm-q', x: 982.2282980177716, y: -302.84347231715657 },
        { id: 'm-r', x: 1457.1701982228299, y: -235.41695146958307 },
        { id: 'm-final', x: 1977.9903165303608, y: -245.81082613290445 },
        { id: 'm-s', x: 967.2727272727273, y: -845.6818181818181 },
        { id: 'm-t', x: 494.0909090909091, y: -817.7272727272727 },
        { id: 'm-u', x: 17.762845849802375, y: -868.7193675889328 },
        { id: 'm-v', x: -456.2667477450086, y: -678.9085334954902 },
      ]
      __graph_json_data.nodes.forEach((model, index) => {
        const position = nodesPosition.find((n) => n.id === model.id)
        if (position) {
          model.x = position.x
          model.y = position.y
        } else {
          model.x = index * 50
          model.y = index * 50
        }
      })
      this.$refs.graphRef.setJsonData(__graph_json_data, async (graphInstance) => {
        // 这些写上当图谱初始化完成后需要执行的代码
        await graphInstance.moveToCenter()
        await graphInstance.zoomToFit()
      })
    },
    onNodeClick(node) {
      console.log('onNodeClick:', node.text)
      const graphInstance = this.$refs.graphRef.getInstance()
      const elLines = graphInstance.getElementLines()
      elLines.forEach((line) => {
        const keywords = `_${node.id}-`
        const refed = line.from.includes(keywords) || line.to.includes(keywords)
        line.animation = refed && 2
      })
    },
  },
}
</script>
<style lang="scss" scoped>
::v-deep .rel-node-shape-1 {
  overflow: hidden;
  border-radius: 10px;
}
::v-deep .rel-toolbar {
  color: #ffffff;
  .c-current-zoom {
    color: #ffffff;
  }
}
</style>

VipDemo4AIModel.vue

javascript
<template>
  <div class="c-model">
    <div class="c-model-title">{{node.text}}</div>
    <div class="c-model-io">
      <div class="c-col-left" >
        <template v-for="input of node.data.input">
          <div :key="input.name" :id="`I_${node.id}-${input.name}`">
            <i class="el-icon-info c-dot" :style="{color:dataTypeColorMap[input.type]}" />
            {{input.name}}
          </div>
        </template>
      </div>
      <div class="c-col-right">
        <template v-for="output of node.data.output">
          <div :key="output.name" :id="`O_${node.id}-${output.name}`">
            {{output.name}}
            <i class="el-icon-info c-dot" :style="{color:dataTypeColorMap[output.type]}" />
          </div>
        </template>
      </div>
    </div>
    <div class="c-model-options">
      <template v-for="opt of node.data.options">
        <div :key="opt.name">
          <div v-if="opt.name === '_textarea'" class="c-model-opt" style="padding: 0px;overflow: hidden;width:400px;">
            <textarea style="height:100px;width:100%;background-color: transparent;border-radius: 10px;color: #ffffff;">{{opt.value}}</textarea>
          </div>
          <div v-else-if="opt.name === '_image'" class="c-model-opt" style="padding: 0px;overflow: hidden;">
            <img :src="opt.value" style="width:100%;user-select: none;-webkit-user-drag: none;">
          </div>
          <div v-else class="c-model-opt">
            <div class="c-col-left">
              <div> {{opt.name}}</div>
            </div>
            <div class="c-col-right">
              <div>{{opt.value}}</div>
            </div>
          </div>
        </div>
      </template>
    </div>
  </div>
</template>

<script>
// 如果您没有在main.js文件中使用Vue.use(RelationGraph); 就需要使用下面这一行代码来引入relation-graph
// import RelationGraph from 'relation-graph';
export default {
  name: 'SingleAIModel',
  components: { },
  props: {
    node: {
      mustUseProp: true,
      default: {},
      type: Object
    },
    dataTypeColorMap: {
      mustUseProp: true,
      default: {},
      type: Object
    }
  },
  data() {
    return {
    };
  },
  mounted() {
  },
  methods: {

  }
};
</script>
<style lang="scss" scoped>
.c-model{
  min-width: 350px;
  border-radius: 10px;
  overflow: hidden;
  .c-dot{
    color: #2cbb03;
    font-size: 10px;
  }
  .c-model-title {
    background-color: rgba(0, 0, 0, 0.2);
    color: #cccccc;
    line-height: 28px;
  }
  .c-col-left{
    width:50%;text-align: left;
  }
  .c-col-right{
    width:50%;text-align: right;
  }
  .c-model-io {
    display: flex;
    padding-left:10px;
    padding-right:10px;
    line-height: 20px;
    .c-col-left{

    }
  }
  .c-model-options {
    padding-left:20px;
    padding-right:20px;
    padding-bottom:5px;
    .c-model-opt {
      display: flex;
      background-color: rgba(0, 0, 0, 0.4);
      border-radius: 10px;
      line-height: 20px;
      margin-top:10px;
      color: #ffffff;
      padding-left: 20px;
      padding-right: 20px;
    }
  }
}
</style>

Vue3 版本

models-relationship.vue

javascript
<template>
  <div>
    <div style="height:calc(100vh);background-color: #222222;">
      <RelationGraph ref="graphRef" :options="graphOptions" :on-node-click="onNodeClick">
        <template #canvas-plug>
          <!--- You can put any content here, and it will be displayed in behind of the node. --->
        </template>
        <!--- You can put any content here, and it will be displayed in above of the node. --->
        <!-- node slot -->
        <template #node="{ node }">
          <AIModel :node="node" :data-type-color-map="dataTypeColorMap" />
        </template>
      </RelationGraph>
    </div>
  </div>
</template>

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

const isShowCodePanel = ref(false)
const dataTypeColorMap = {
  1: '#2cbb03',
  2: '#1da9f5',
  3: '#ff8080',
  4: '#f39930',
  5: '#f39930',
  6: '#f12de0',
  7: '#5b05f1',
  8: '#f39930',
  9: '#ffffff',
}
const graphOptions: RGOptions = {
  debug: false,
  defaultJunctionPoint: 'border',
  backgroundColor: 'transparent',
  defaultNodeColor: 'rgba(100, 100, 100, 0.5)',
  placeOtherNodes: false,
  placeSingleNode: false,
  defaultLineMarker: {
    markerWidth: 20,
    markerHeight: 20,
    refX: 3,
    refY: 3,
    data: 'M 0 0, V 6, L 4 3, Z',
  },
  layout: {
    layoutName: 'fixed',
  },
}

const showGraph = async () => {
  const __graph_json_data: RGJsonData = {
    rootId: 'a',
    nodes: [
      {
        id: 'm-a',
        nodeShape: 1,
        text: 'Load Video',
        data: {
          options: [
            { name: 'video_path', type: 'string', value: '/data/My-World-1080P.mp4' },
            { name: 'force_rate', type: 'number', value: '10' },
            { name: 'force_size', type: 'boolean', value: '0' },
            { name: 'frame_load_cap', type: 'number', value: '2048' },
            { name: 'skip_first_frames', type: 'number', value: '0' },
          ],
          input: [],
          output: [
            { name: 'current_image', type: 1 },
            { name: 'frame_count', type: 1 },
          ],
        },
      },
      {
        id: 'm-b',
        nodeShape: 1,
        text: 'Preview Image',
        data: {
          options: [],
          input: [{ name: 'image', type: 1 }],
          output: [],
        },
      },
      {
        id: 'm-c',
        nodeShape: 1,
        text: 'Empty Latent Image',
        data: {
          options: [],
          input: [
            { name: 'batch_size', type: 2 },
            { name: 'width', type: 2 },
            { name: 'height', type: 2 },
          ],
          output: [{ name: 'LATENT', type: 5 }],
        },
      },
      {
        id: 'm-d',
        nodeShape: 1,
        text: 'Primitive',
        data: {
          options: [
            { name: 'value', type: 'number', value: '512' },
            { name: 'control_affer_generate', type: 'string', value: 'fixed' },
          ],
          input: [],
          output: [{ name: 'INT', type: 2 }],
        },
      },
      {
        id: 'm-dd',
        nodeShape: 1,
        text: 'Primitive',
        data: {
          options: [
            { name: 'value', type: 'number', value: '24' },
            { name: 'control_affer_generate', type: 'string', value: 'fixed' },
          ],
          input: [],
          output: [{ name: 'INT', type: 2 }],
        },
      },
      {
        id: 'm-e',
        nodeShape: 1,
        text: 'Primitive',
        data: {
          options: [
            { name: 'value', type: 'number', value: '688' },
            { name: 'control_affer_generate', type: 'string', value: 'fixed' },
          ],
          input: [],
          output: [{ name: 'INT', type: 2 }],
        },
      },
      {
        id: 'm-f',
        nodeShape: 1,
        text: 'AnimateDiff Loader',
        color: 'rgba(139,108,46,0.58)',
        data: {
          options: [
            { name: 'model_name', type: 'string', value: 'diff-v1.safetensors' },
            { name: 'beta_schedule', type: 'string', value: 'sqrt_linearr(AnimateDiff)' },
          ],
          input: [
            { name: 'model', type: 0 },
            { name: 'context_options', type: 8 },
            { name: 'montion_lora', type: 9 },
          ],
          output: [{ name: 'MODEL', type: 0 }],
        },
      },
      {
        id: 'm-g',
        nodeShape: 1,
        text: 'KSampler',
        data: {
          options: [
            { name: 'steps', type: 'number', value: '30' },
            { name: 'cfg', type: 'number', value: '8.0' },
            { name: 'sampler_name', type: 'string', value: 'dpmpp_sde' },
            { name: 'scheduler', type: 'string', value: 'karras' },
            { name: 'denoise', type: 'number', value: '1.00' },
          ],
          input: [
            { name: 'model', type: 0 },
            { name: 'positive', type: 0 },
            { name: 'negative', type: 0 },
            { name: 'latent_image', type: 1 },
            { name: 'seed', type: 2 },
          ],
          output: [{ name: 'LATENT', type: 5 }],
        },
      },
      {
        id: 'm-h',
        nodeShape: 1,
        text: 'CLIP Set Last Layer',
        data: {
          options: [{ name: 'stop_at_clip_layer', type: 'number', value: '-2' }],
          input: [{ name: 'clip', type: 3 }],
          output: [{ name: 'CLIP', type: 5 }],
        },
      },
      {
        id: 'm-i',
        nodeShape: 1,
        text: 'CLIP Text Encode(Prompt)',
        color: 'rgba(125,55,53,0.6)',
        data: {
          options: [{ name: 'stop_at_clip_layer', type: 'number', value: '-2' }],
          input: [{ name: 'clip', type: 3 }],
          output: [{ name: 'CONDITIONING', type: 5 }],
        },
      },
      {
        id: 'm-j',
        nodeShape: 1,
        text: 'Load Checkpoint',
        data: {
          options: [{ name: 'ckp', type: 'string', value: 'vision51_v51VAE_safetensors' }],
          input: [],
          output: [
            { name: 'MODEL', type: 5 },
            { name: 'CLIP', type: 3 },
            { name: 'VAE', type: 6 },
          ],
        },
      },
      {
        id: 'm-k',
        nodeShape: 1,
        text: 'BatchPrompSchedule',
        data: {
          options: [
            {
              name: '_textarea',
              type: 'string',
              value: `"0" : "yummy food, seafood pizza",
"83" : "student's desk, books, pen, sketchbook, earaser, ruler",
"167" : "jewelery, diamonds, crystals, marbles",
                `,
            },
            { name: 'pre_text', type: 'string', value: 'photoreealistic photography, to' },
            { name: 'app_text', type: 'string', value: 'black white interior,top angl' },
            { name: 'pw_a', type: 'number', value: '0.0' },
            { name: 'pw_b', type: 'number', value: '0.0' },
            { name: 'pw_c', type: 'number', value: '0.0' },
            { name: 'pw_d', type: 'number', value: '0.0' },
          ],
          input: [
            { name: 'clip', type: 5 },
            { name: 'max_frames', type: 5 },
          ],
          output: [{ name: 'CONDITIONING', type: 5 }],
        },
      },
      {
        id: 'm-l',
        nodeShape: 1,
        text: 'Uniform Context Options',
        color: 'rgba(139,108,46,0.58)',
        data: {
          options: [
            { name: 'context_length', type: 'number', value: '0.0' },
            { name: 'context_stride', type: 'number', value: '0.0' },
            { name: 'context_overlap', type: 'number', value: '0.0' },
            { name: 'context_schedule', type: 'string', value: 'uniform' },
            { name: 'closed_loop', type: 'boolean', value: 'false' },
          ],
          input: [],
          output: [{ name: 'CONTEXT_OPTIONS', type: 5 }],
        },
      },
      {
        id: 'm-m',
        nodeShape: 1,
        text: 'Apply ControlNet(Advanced)',
        color: 'rgba(141,32,141,0.5)',
        data: {
          options: [
            { name: 'strength', type: 'number', value: '1.0' },
            { name: 'start_percent', type: 'number', value: '0.000' },
            { name: 'end_percent', type: 'number', value: '1.000' },
          ],
          input: [
            { name: 'positive', type: 5 },
            { name: 'negative', type: 5 },
            { name: 'control_net', type: 4 },
            { name: 'image', type: 4 },
          ],
          output: [
            { name: 'positive', type: 5 },
            { name: 'negative', type: 5 },
          ],
        },
      },
      {
        id: 'm-n',
        nodeShape: 1,
        text: 'Apply ControlNet(Advanced)',
        color: 'rgba(141,32,141,0.5)',
        data: {
          options: [
            { name: 'strength', type: 'number', value: '0.50' },
            { name: 'start_percent', type: 'number', value: '0.002' },
            { name: 'end_percent', type: 'number', value: '0.663' },
          ],
          input: [
            { name: 'positive', type: 5 },
            { name: 'negative', type: 5 },
            { name: 'control_net', type: 4 },
            { name: 'image', type: 4 },
          ],
          output: [
            { name: 'positive', type: 5 },
            { name: 'negative', type: 5 },
          ],
        },
      },
      {
        id: 'm-o',
        nodeShape: 1,
        text: 'Apply ControlNet(Advanced)',
        color: 'rgba(141,32,141,0.5)',
        data: {
          options: [
            { name: 'strength', type: 'number', value: '0.45' },
            { name: 'start_percent', type: 'number', value: '0.002' },
            { name: 'end_percent', type: 'number', value: '0.550' },
          ],
          input: [
            { name: 'positive', type: 5 },
            { name: 'negative', type: 5 },
            { name: 'control_net', type: 4 },
            { name: 'image', type: 4 },
          ],
          output: [
            { name: 'positive', type: 5 },
            { name: 'negative', type: 5 },
          ],
        },
      },
      {
        id: 'm-mc',
        nodeShape: 1,
        text: 'Load ControlNet Model(Advanced)',
        color: 'rgba(141,32,141,0.5)',
        data: {
          options: [
            { name: 'control_net_name', type: 'string', value: 'control_v11p_sd15_openpose.pth' },
          ],
          input: [],
          output: [{ name: 'CONTROL_NET', type: 1 }],
        },
      },
      {
        id: 'm-nc',
        nodeShape: 1,
        text: 'Load ControlNet Model(Advanced)',
        color: 'rgba(141,32,141,0.5)',
        data: {
          options: [
            { name: 'control_net_name', type: 'string', value: 'control_v11p_sd15_depth.pth' },
          ],
          input: [],
          output: [{ name: 'CONTROL_NET', type: 1 }],
        },
      },
      {
        id: 'm-oc',
        nodeShape: 1,
        text: 'Load ControlNet Model(Advanced)',
        color: 'rgba(141,32,141,0.5)',
        data: {
          options: [
            { name: 'control_net_name', type: 'string', value: 'control_v11p_sd15_lineart.pth' },
          ],
          input: [],
          output: [{ name: 'CONTROL_NET', type: 1 }],
        },
      },
      {
        id: 'm-p',
        nodeShape: 1,
        text: 'Load VAE',
        data: {
          options: [{ name: 'vae_name', type: 'string', value: 'vae-ft-mse-840000.safetensors' }],
          input: [],
          output: [{ name: 'VAE', type: 5 }],
        },
      },
      {
        id: 'm-q',
        nodeShape: 1,
        text: 'VAE Decode',
        data: {
          options: [],
          input: [
            { name: 'samples', type: 5 },
            { name: 'vae', type: 5 },
          ],
          output: [{ name: 'IMAGE', type: 5 }],
        },
      },
      {
        id: 'm-r',
        nodeShape: 1,
        text: 'Video Combine',
        color: 'rgba(50,58,132,0.65)',
        data: {
          options: [
            { name: 'frome_rate', type: 'number', value: '10' },
            { name: 'loop_count', type: 'number', value: '0' },
          ],
          input: [{ name: 'images', type: 5 }],
          output: [{ name: 'GIF', type: 5 }],
        },
      },
      {
        id: 'm-final',
        nodeShape: 1,
        text: 'Video Combine',
        color: 'rgba(50,58,132,0.65)',
        data: {
          options: [
            { name: '_image', type: 'string', value: '/demo-images/open-all-close-all.gif' },
          ],
          input: [{ name: 'GIF', type: 5 }],
          output: [],
        },
      },
      {
        id: 'm-s',
        nodeShape: 1,
        text: 'Normal Lineart',
        data: {
          options: [
            { name: 'coarse', type: 'string', value: 'disable' },
            { name: 'resolution', type: 'number', value: '512' },
          ],
          input: [{ name: 'image', type: 5 }],
          output: [{ name: 'IMAGE', type: 6 }],
        },
      },
      {
        id: 'm-t',
        nodeShape: 1,
        text: 'Zoe - Depth Map',
        data: {
          options: [{ name: 'resolution', type: 'number', value: '512' }],
          input: [{ name: 'image', type: 5 }],
          output: [{ name: 'IMAGE', type: 6 }],
        },
      },
      {
        id: 'm-u',
        nodeShape: 1,
        text: 'MediaPipe-FaceMeshPreprocessor',
        data: {
          options: [
            { name: 'max_faces', type: 'number', value: '2' },
            { name: 'min_confidence', type: 'number', value: '0.50' },
            { name: 'resolution', type: 'number', value: '512' },
          ],
          input: [{ name: 'image', type: 5 }],
          output: [{ name: 'IMAGE', type: 6 }],
        },
      },
      {
        id: 'm-v',
        nodeShape: 1,
        text: 'Upscale Image',
        data: {
          options: [
            { name: 'upscale_method', type: 'string', value: 'nearest-exact' },
            { name: 'ccrop', type: 'string', value: 'center' },
          ],
          input: [
            { name: 'image', type: 5 },
            { name: 'width', type: 5 },
            { name: 'height', type: 5 },
          ],
          output: [{ name: 'IMAGE', type: 6 }],
        },
      },
    ],
    lines: [],
    elementLines: [],
  }
  const modelRelations = [
    {
      output_model: 'm-d',
      output_param_name: 'INT',
      input_model: 'm-c',
      input_param_name: 'width',
    },
    {
      output_model: 'm-e',
      output_param_name: 'INT',
      input_model: 'm-c',
      input_param_name: 'height',
    },
    {
      output_model: 'm-dd',
      output_param_name: 'INT',
      input_model: 'm-g',
      input_param_name: 'seed',
    },
    {
      output_model: 'm-a',
      output_param_name: 'frame_count',
      input_model: 'm-c',
      input_param_name: 'batch_size',
    },
    {
      output_model: 'm-a',
      output_param_name: 'current_image',
      input_model: 'm-b',
      input_param_name: 'image',
    },
    {
      output_model: 'm-c',
      output_param_name: 'LATENT',
      input_model: 'm-g',
      input_param_name: 'latent_image',
    },
    {
      output_model: 'm-f',
      output_param_name: 'MODEL',
      input_model: 'm-g',
      input_param_name: 'model',
    },
    {
      output_model: 'm-j',
      output_param_name: 'CLIP',
      input_model: 'm-h',
      input_param_name: 'clip',
    },
    {
      output_model: 'm-j',
      output_param_name: 'CLIP',
      input_model: 'm-i',
      input_param_name: 'clip',
    },
    {
      output_model: 'm-j',
      output_param_name: 'MODEL',
      input_model: 'm-f',
      input_param_name: 'model',
    },
    {
      output_model: 'm-a',
      output_param_name: 'frame_count',
      input_model: 'm-k',
      input_param_name: 'max_frames',
    },
    {
      output_model: 'm-h',
      output_param_name: 'CLIP',
      input_model: 'm-k',
      input_param_name: 'clip',
    },
    {
      output_model: 'm-l',
      output_param_name: 'CONTEXT_OPTIONS',
      input_model: 'm-f',
      input_param_name: 'context_options',
    },
    {
      output_model: 'm-i',
      output_param_name: 'CONDITIONING',
      input_model: 'm-m',
      input_param_name: 'negative',
    },
    {
      output_model: 'm-k',
      output_param_name: 'CONDITIONING',
      input_model: 'm-m',
      input_param_name: 'positive',
    },
    {
      output_model: 'm-m',
      output_param_name: 'negative',
      input_model: 'm-n',
      input_param_name: 'negative',
    },
    {
      output_model: 'm-m',
      output_param_name: 'positive',
      input_model: 'm-n',
      input_param_name: 'positive',
    },
    {
      output_model: 'm-n',
      output_param_name: 'negative',
      input_model: 'm-o',
      input_param_name: 'negative',
    },
    {
      output_model: 'm-n',
      output_param_name: 'positive',
      input_model: 'm-o',
      input_param_name: 'positive',
    },
    {
      output_model: 'm-mc',
      output_param_name: 'CONTROL_NET',
      input_model: 'm-m',
      input_param_name: 'control_net',
    },
    {
      output_model: 'm-nc',
      output_param_name: 'CONTROL_NET',
      input_model: 'm-n',
      input_param_name: 'control_net',
    },
    {
      output_model: 'm-oc',
      output_param_name: 'CONTROL_NET',
      input_model: 'm-o',
      input_param_name: 'control_net',
    },
    { output_model: 'm-p', output_param_name: 'VAE', input_model: 'm-q', input_param_name: 'vae' },
    {
      output_model: 'm-g',
      output_param_name: 'LATENT',
      input_model: 'm-q',
      input_param_name: 'samples',
    },
    {
      output_model: 'm-q',
      output_param_name: 'IMAGE',
      input_model: 'm-r',
      input_param_name: 'images',
    },
    {
      output_model: 'm-s',
      output_param_name: 'IMAGE',
      input_model: 'm-o',
      input_param_name: 'image',
    },
    {
      output_model: 'm-t',
      output_param_name: 'IMAGE',
      input_model: 'm-n',
      input_param_name: 'image',
    },
    {
      output_model: 'm-u',
      output_param_name: 'IMAGE',
      input_model: 'm-m',
      input_param_name: 'image',
    },
    {
      output_model: 'm-u',
      output_param_name: 'IMAGE',
      input_model: 'm-t',
      input_param_name: 'image',
    },
    {
      output_model: 'm-v',
      output_param_name: 'IMAGE',
      input_model: 'm-u',
      input_param_name: 'image',
    },
    {
      output_model: 'm-v',
      output_param_name: 'IMAGE',
      input_model: 'm-s',
      input_param_name: 'image',
    },
    {
      output_model: 'm-a',
      output_param_name: 'current_image',
      input_model: 'm-v',
      input_param_name: 'image',
    },
    {
      output_model: 'm-d',
      output_param_name: 'INT',
      input_model: 'm-v',
      input_param_name: 'width',
    },
    {
      output_model: 'm-e',
      output_param_name: 'INT',
      input_model: 'm-v',
      input_param_name: 'height',
    },
    {
      output_model: 'm-r',
      output_param_name: 'GIF',
      input_model: 'm-final',
      input_param_name: 'GIF',
    },
  ]
  const getModel = (modelId: string) => {
    return __graph_json_data.nodes.find((model) => model.id === modelId)
  }
  const getModelOutputParam = (model: RGNode, outputParamName: string) => {
    return model.data.output.find((param) => param.name === outputParamName)
  }
  const getModelInputParam = (model: RGNode, inputParamName: string) => {
    return model.data.input.find((param) => param.name === inputParamName)
  }
  __graph_json_data.elementLines = modelRelations.map((relation) => {
    const outputModel = getModel(relation.output_model)
    const outputParam = getModelOutputParam(outputModel, relation.output_param_name)
    const inputModel = getModel(relation.input_model)
    const inputParam = getModelInputParam(inputModel, relation.input_param_name)
    inputParam.type = outputParam.type // fix data bug

    return {
      from: `O_${relation.output_model}-${relation.output_param_name}`,
      fromJunctionPoint: 'right',
      to: `I_${relation.input_model}-${relation.input_param_name}`,
      toJunctionPoint: 'left',
      text: '',
      lineShape: 6,
      lineWidth: 3,
      color: dataTypeColorMap[outputParam.type],
    }
  })
  // Using the position set for the node, it can be adjusted, saved to the database, and then reloaded to be displayed here.
  const nodesPosition = [
    { id: 'm-a', x: -930.1706372723322, y: -357.62950890917 },
    { id: 'm-b', x: -456.66745683694836, y: -468.67267196080763 },
    { id: 'm-c', x: 18.272727272727273, y: -297.27272727272725 },
    { id: 'm-d', x: -459.07692307692304, y: -383.7692307692307 },
    { id: 'm-dd', x: -458.12172573189525, y: -81.8335901386749 },
    { id: 'm-e', x: -459.9230769230769, y: -240.23076923076923 },
    { id: 'm-f', x: 502.27272727272725, y: 150 },
    { id: 'm-g', x: 982.421052631579, y: -146.69856459330143 },
    { id: 'm-h', x: 498.14285714285717, y: -301.29870129870125 },
    { id: 'm-i', x: 503, y: -158.63636363636365 },
    { id: 'm-j', x: 15.420875420875419, y: -173.28956228956224 },
    { id: 'm-k', x: -22.447552447552397, y: -3.681688681688712 },
    { id: 'm-l', x: -456.8423948932424, y: 72.69742719892274 },
    { id: 'm-m', x: 16.882352941176407, y: -545.5347593582887 },
    { id: 'm-n', x: 495.5050505050505, y: -550.5555555555555 },
    { id: 'm-o', x: 976.8311688311688, y: -554.1428571428571 },
    { id: 'm-mc', x: 18.012987012987004, y: -684.6883116883117 },
    { id: 'm-nc', x: 492.012987012987, y: -688.3766233766235 },
    { id: 'm-oc', x: 971.7272727272727, y: -692.1038961038962 },
    { id: 'm-p', x: 502.5717703349282, y: -12.715311004784702 },
    { id: 'm-q', x: 982.2282980177716, y: -302.84347231715657 },
    { id: 'm-r', x: 1457.1701982228299, y: -235.41695146958307 },
    { id: 'm-final', x: 1977.9903165303608, y: -245.81082613290445 },
    { id: 'm-s', x: 967.2727272727273, y: -845.6818181818181 },
    { id: 'm-t', x: 494.0909090909091, y: -817.7272727272727 },
    { id: 'm-u', x: 17.762845849802375, y: -868.7193675889328 },
    { id: 'm-v', x: -456.2667477450086, y: -678.9085334954902 },
  ]
  __graph_json_data.nodes.forEach((model, index) => {
    const position = nodesPosition.find((n) => n.id === model.id)
    if (position) {
      model.x = position.x
      model.y = position.y
    } else {
      model.x = index * 50
      model.y = index * 50
    }
  })
  const graphInstance = graphRef.value!.getInstance()
  await graphInstance.setJsonData(__graph_json_data)
  await graphInstance.moveToCenter()
  await graphInstance.zoomToFit()
}

const onNodeClick = (node: RGNode, $event: RGUserEvent) => {
  console.log('onNodeClick:', node.text)
  const graphInstance = graphRef.value!.getInstance()
  const elLines = graphInstance.getElementLines()
  elLines.forEach((line: RGLine) => {
    const keywords = `_${node.id}-`
    const refed = line.from.includes(keywords) || line.to.includes(keywords)
    line.animation = refed ? 2 : undefined
  })
}

const graphRef = ref<RelationGraphComponent>()

onMounted(() => {
  console.log('graphRef.value:', graphRef.value)
  showGraph()
  // You can adjust the position of the nodes on the page, and then use the following code to print or save the positions of all current nodes, and use these positions the next time the page is opened.
  // setInterval(() => {
  //   const graphInstance = graphRef.current!.getInstance();
  //   const nodePositons = graphInstance.getNodes().map(n => ({id:n.id,x:n.x,y:n.y}));
  //   console.log(JSON.stringify(nodePositons));
  // }, 5000);;
})
</script>

<style lang="scss" scoped>
::v-deep(.relation-graph) {
  .rel-node-shape-1 {
    overflow: hidden;
    border-radius: 10px;
  }
  .rel-toolbar {
    color: #ffffff;
    .c-current-zoom {
      color: #ffffff;
    }
  }
}
</style>

VipDemo4AIModel.vue

javascript
<template>
  <div class="c-model">
    <div class="c-model-title">{{ node.text }}</div>
    <div class="c-model-io">
      <div class="c-col-left">
        <template v-for="input of node.data.input" :key="input.name">
          <div :id="`I_${node.id}-${input.name}`">
            <i class="el-icon-info c-dot" :style="{color:dataTypeColorMap[input.type]}" />
            {{ input.name }}
          </div>
        </template>
      </div>
      <div class="c-col-right">
        <template v-for="output of node.data.output" :key="output.name">
          <div :id="`O_${node.id}-${output.name}`">
            {{ output.name }}
            <i class="el-icon-info c-dot" :style="{color:dataTypeColorMap[output.type]}" />
          </div>
        </template>
      </div>
    </div>
    <div class="c-model-options">
      <template v-for="opt of node.data.options" :key="opt.name">
        <div>
          <div v-if="opt.name === '_textarea'" class="c-model-opt" style="padding: 0px;overflow: hidden;width:400px;">
            <textarea style="height:100px;width:100%;background-color: transparent;border-radius: 10px;color: #ffffff;">{{ opt.value }}</textarea>
          </div>
          <div v-else-if="opt.name === '_image'" class="c-model-opt" style="padding: 0px;overflow: hidden;">
            <img :src="opt.value" style="width:100%;user-select: none;-webkit-user-drag: none;">
          </div>
          <div v-else class="c-model-opt">
            <div class="c-col-left">
              <div> {{ opt.name }}</div>
            </div>
            <div class="c-col-right">
              <div>{{ opt.value }}</div>
            </div>
          </div>
        </div>
      </template>
    </div>
  </div>
</template>

<script lang="ts" setup>
// 如果您没有在main.js文件中使用Vue.use(RelationGraph); 就需要使用下面这一行代码来引入relation-graph
// import RelationGraph from 'relation-graph';
import {RGNode} from "relation-graph-vue3";

const {node, dataTypeColorMap} = defineProps<{
    node: RGNode,
    dataTypeColorMap: any
}>();
</script>
<style lang="scss" scoped>
.c-model{
  min-width: 350px;
  border-radius: 10px;
  overflow: hidden;
  .c-dot{
    color: #2cbb03;
    font-size: 10px;
  }
  .c-model-title {
    background-color: rgba(0, 0, 0, 0.2);
    color: #cccccc;
    line-height: 28px;
  }
  .c-col-left{
    width:50%;text-align: left;
  }
  .c-col-right{
    width:50%;text-align: right;
  }
  .c-model-io {
    display: flex;
    padding-left:10px;
    padding-right:10px;
    line-height: 20px;
    .c-col-left{

    }
  }
  .c-model-options {
    padding-left:20px;
    padding-right:20px;
    padding-bottom:5px;
    .c-model-opt {
      display: flex;
      background-color: rgba(0, 0, 0, 0.4);
      border-radius: 10px;
      line-height: 20px;
      margin-top:10px;
      color: #ffffff;
      padding-left: 20px;
      padding-right: 20px;
    }
  }
}
</style>

React 版本

models-relationship.tsx

javascript
import React, { useEffect, useRef } from 'react';
import RelationGraph, {JsonNode} from 'relation-graph-react';
import { RGJsonData, RGOptions, RGNode, RGLine, RGUserEvent, RelationGraphComponent } from 'relation-graph-react';
import {MyAIModel, dataTypeColorMap} from './VipDemo4AIModel.tsx';
import './models-relationship.scss';

const __graph_json_data: RGJsonData = {
  rootId: 'a',
  nodes: [
    { id: 'm-a', nodeShape: 1, text: 'Load Video', data: {
        options: [
          {name: 'video_path', type: 'string', value: '/data/My-World-1080P.mp4'},
          {name: 'force_rate', type: 'number', value: '10'},
          {name: 'force_size', type: 'boolean', value: '0'},
          {name: 'frame_load_cap', type: 'number', value: '2048'},
          {name: 'skip_first_frames', type: 'number', value: '0'},
        ],
        input: [],
        output: [
          {name: 'current_image', type: 1},
          {name: 'frame_count', type: 1},
        ]
      }
    },
    { id: 'm-b', nodeShape: 1, text: 'Preview Image', data: {
        options: [],
        input: [
          {name: 'image', type: 1}
        ],
        output: []
      }
    },
    { id: 'm-c', nodeShape: 1, text: 'Empty Latent Image', data: {
        options: [],
        input: [
          {name: 'batch_size', type: 2},
          {name: 'width', type: 2},
          {name: 'height', type: 2},
        ],
        output: [
          {name: 'LATENT', type: 5},
        ]
      }
    },
    { id: 'm-d', nodeShape: 1, text: 'Primitive', data: {
        options: [
          {name: 'value', type: 'number', value: '512'},
          {name: 'control_affer_generate', type: 'string', value: 'fixed'},
        ],
        input: [],
        output: [
          {name: 'INT', type: 2},
        ]
      }
    },
    { id: 'm-dd', nodeShape: 1, text: 'Primitive', data: {
        options: [
          {name: 'value', type: 'number', value: '24'},
          {name: 'control_affer_generate', type: 'string', value: 'fixed'},
        ],
        input: [],
        output: [
          {name: 'INT', type: 2},
        ]
      }
    },
    { id: 'm-e', nodeShape: 1, text: 'Primitive', data: {
        options: [
          {name: 'value', type: 'number', value: '688'},
          {name: 'control_affer_generate', type: 'string', value: 'fixed'},
        ],
        input: [],
        output: [
          {name: 'INT', type: 2},
        ]
      }
    },
    { id: 'm-f', nodeShape: 1, text: 'AnimateDiff Loader', color: 'rgba(139,108,46,0.58)', data: {
        options: [
          {name: 'model_name', type: 'string', value: 'diff-v1.safetensors'},
          {name: 'beta_schedule', type: 'string', value: 'sqrt_linearr(AnimateDiff)'},
        ],
        input: [
          {name: 'model', type: 0},
          {name: 'context_options', type: 8},
          {name: 'montion_lora', type: 9},
        ],
        output: [
          {name: 'MODEL', type: 0},
        ]
      }
    },
    { id: 'm-g', nodeShape: 1, text: 'KSampler', data: {
        options: [
          {name: 'steps', type: 'number', value: '30'},
          {name: 'cfg', type: 'number', value: '8.0'},
          {name: 'sampler_name', type: 'string', value: 'dpmpp_sde'},
          {name: 'scheduler', type: 'string', value: 'karras'},
          {name: 'denoise', type: 'number', value: '1.00'},
        ],
        input: [
          {name: 'model', type: 0},
          {name: 'positive', type: 0},
          {name: 'negative', type: 0},
          {name: 'latent_image', type: 1},
          {name: 'seed', type: 2},
        ],
        output: [
          {name: 'LATENT', type: 5},
        ]
      }
    },
    { id: 'm-h', nodeShape: 1, text: 'CLIP Set Last Layer', data: {
        options: [
          {name: 'stop_at_clip_layer', type: 'number', value: '-2'},
        ],
        input: [
          {name: 'clip', type: 3},
        ],
        output: [
          {name: 'CLIP', type: 5},
        ]
      }
    },
    { id: 'm-i', nodeShape: 1, text: 'CLIP Text Encode(Prompt)', color: 'rgba(125,55,53,0.6)', data: {
        options: [
          {name: 'stop_at_clip_layer', type: 'number', value: '-2'},
        ],
        input: [
          {name: 'clip', type: 3},
        ],
        output: [
          {name: 'CONDITIONING', type: 5},
        ]
      }
    },
    { id: 'm-j', nodeShape: 1, text: 'Load Checkpoint', data: {
        options: [
          {name: 'ckp', type: 'string', value: 'vision51_v51VAE_safetensors'},
        ],
        input: [
        ],
        output: [
          {name: 'MODEL', type: 5},
          {name: 'CLIP', type: 3},
          {name: 'VAE', type: 6},
        ]
      }
    },
    { id: 'm-k', nodeShape: 1, text: 'BatchPrompSchedule', data: {
        options: [
          {name: '_textarea', type: 'string', value: `"0" : "yummy food, seafood pizza",
"83" : "student's desk, books, pen, sketchbook, earaser, ruler",
"167" : "jewelery, diamonds, crystals, marbles",
            `},
          {name: 'pre_text', type: 'string', value: 'photoreealistic photography, to'},
          {name: 'app_text', type: 'string', value: 'black white interior,top angl'},
          {name: 'pw_a', type: 'number', value: '0.0'},
          {name: 'pw_b', type: 'number', value: '0.0'},
          {name: 'pw_c', type: 'number', value: '0.0'},
          {name: 'pw_d', type: 'number', value: '0.0'},
        ],
        input: [
          {name: 'clip', type: 5},
          {name: 'max_frames', type: 5},
        ],
        output: [
          {name: 'CONDITIONING', type: 5},
        ]
      }
    },
    { id: 'm-l', nodeShape: 1, text: 'Uniform Context Options', color: 'rgba(139,108,46,0.58)', data: {
        options: [
          {name: 'context_length', type: 'number', value: '0.0'},
          {name: 'context_stride', type: 'number', value: '0.0'},
          {name: 'context_overlap', type: 'number', value: '0.0'},
          {name: 'context_schedule', type: 'string', value: 'uniform'},
          {name: 'closed_loop', type: 'boolean', value: 'false'},
        ],
        input: [
        ],
        output: [
          {name: 'CONTEXT_OPTIONS', type: 5},
        ]
      }
    },
    { id: 'm-m', nodeShape: 1, text: 'Apply ControlNet(Advanced)', color: 'rgba(141,32,141,0.5)', data: {
        options: [
          {name: 'strength', type: 'number', value: '1.0'},
          {name: 'start_percent', type: 'number', value: '0.000'},
          {name: 'end_percent', type: 'number', value: '1.000'},
        ],
        input: [
          {name: 'positive', type: 5},
          {name: 'negative', type: 5},
          {name: 'control_net', type: 4},
          {name: 'image', type: 4},
        ],
        output: [
          {name: 'positive', type: 5},
          {name: 'negative', type: 5},
        ]
      }
    },
    { id: 'm-n', nodeShape: 1, text: 'Apply ControlNet(Advanced)', color: 'rgba(141,32,141,0.5)', data: {
        options: [
          {name: 'strength', type: 'number', value: '0.50'},
          {name: 'start_percent', type: 'number', value: '0.002'},
          {name: 'end_percent', type: 'number', value: '0.663'},
        ],
        input: [
          {name: 'positive', type: 5},
          {name: 'negative', type: 5},
          {name: 'control_net', type: 4},
          {name: 'image', type: 4},
        ],
        output: [
          {name: 'positive', type: 5},
          {name: 'negative', type: 5},
        ]
      }
    },
    { id: 'm-o', nodeShape: 1, text: 'Apply ControlNet(Advanced)', color: 'rgba(141,32,141,0.5)', data: {
        options: [
          {name: 'strength', type: 'number', value: '0.45'},
          {name: 'start_percent', type: 'number', value: '0.002'},
          {name: 'end_percent', type: 'number', value: '0.550'},
        ],
        input: [
          {name: 'positive', type: 5},
          {name: 'negative', type: 5},
          {name: 'control_net', type: 4},
          {name: 'image', type: 4},
        ],
        output: [
          {name: 'positive', type: 5},
          {name: 'negative', type: 5},
        ]
      }
    },
    { id: 'm-mc', nodeShape: 1, text: 'Load ControlNet Model(Advanced)', color: 'rgba(141,32,141,0.5)', data: {
        options: [
          {name: 'control_net_name', type: 'string', value: 'control_v11p_sd15_openpose.pth'},
        ],
        input: [
        ],
        output: [
          {name: 'CONTROL_NET', type: 1},
        ]
      }
    },
    { id: 'm-nc', nodeShape: 1, text: 'Load ControlNet Model(Advanced)', color: 'rgba(141,32,141,0.5)', data: {
        options: [
          {name: 'control_net_name', type: 'string', value: 'control_v11p_sd15_depth.pth'},
        ],
        input: [
        ],
        output: [
          {name: 'CONTROL_NET', type: 1},
        ]
      }
    },
    { id: 'm-oc', nodeShape: 1, text: 'Load ControlNet Model(Advanced)', color: 'rgba(141,32,141,0.5)', data: {
        options: [
          {name: 'control_net_name', type: 'string', value: 'control_v11p_sd15_lineart.pth'},
        ],
        input: [
        ],
        output: [
          {name: 'CONTROL_NET', type: 1},
        ]
      }
    },
    { id: 'm-p', nodeShape: 1, text: 'Load VAE', data: {
        options: [
          {name: 'vae_name', type: 'string', value: 'vae-ft-mse-840000.safetensors'},
        ],
        input: [
        ],
        output: [
          {name: 'VAE', type: 5},
        ]
      }
    },
    { id: 'm-q', nodeShape: 1, text: 'VAE Decode', data: {
        options: [
        ],
        input: [
          {name: 'samples', type: 5},
          {name: 'vae', type: 5},
        ],
        output: [
          {name: 'IMAGE', type: 5},
        ]
      }
    },
    { id: 'm-r', nodeShape: 1, text: 'Video Combine', color: 'rgba(50,58,132,0.65)', data: {
        options: [
          {name: 'frome_rate', type: 'number', value: '10'},
          {name: 'loop_count', type: 'number', value: '0'},
        ],
        input: [
          {name: 'images', type: 5},
        ],
        output: [
          {name: 'GIF', type: 5},
        ]
      }
    },
    { id: 'm-final', nodeShape: 1, text: 'Video Combine', color: 'rgba(50,58,132,0.65)', data: {
        options: [
          {name: '_image', type: 'string', value: 'https://relation-graph.com/demo-images/open-all-close-all.gif'},
        ],
        input: [
          {name: 'GIF', type: 5},
        ],
        output: [
        ]
      }
    },
    { id: 'm-s', nodeShape: 1, text: 'Normal Lineart', data: {
        options: [
          {name: 'coarse', type: 'string', value: 'disable'},
          {name: 'resolution', type: 'number', value: '512'},
        ],
        input: [
          {name: 'image', type: 5},
        ],
        output: [
          {name: 'IMAGE', type: 6},
        ]
      }
    },
    { id: 'm-t', nodeShape: 1, text: 'Zoe - Depth Map', data: {
        options: [
          {name: 'resolution', type: 'number', value: '512'},
        ],
        input: [
          {name: 'image', type: 5},
        ],
        output: [
          {name: 'IMAGE', type: 6},
        ]
      }
    },
    { id: 'm-u', nodeShape: 1, text: 'MediaPipe-FaceMeshPreprocessor', data: {
        options: [
          {name: 'max_faces', type: 'number', value: '2'},
          {name: 'min_confidence', type: 'number', value: '0.50'},
          {name: 'resolution', type: 'number', value: '512'},
        ],
        input: [
          {name: 'image', type: 5},
        ],
        output: [
          {name: 'IMAGE', type: 6},
        ]
      }
    },
    { id: 'm-v', nodeShape: 1, text: 'Upscale Image', data: {
        options: [
          {name: 'upscale_method', type: 'string', value: 'nearest-exact'},
          {name: 'ccrop', type: 'string', value: 'center'},
        ],
        input: [
          {name: 'image', type: 5},
          {name: 'width', type: 5},
          {name: 'height', type: 5},
        ],
        output: [
          {name: 'IMAGE', type: 6},
        ]
      }
    },
  ],
  lines: [],
  elementLines: []
};
const modelRelations = [
  { output_model: 'm-d', output_param_name: 'INT', input_model: 'm-c', input_param_name: 'width' },
  { output_model: 'm-e', output_param_name: 'INT', input_model: 'm-c', input_param_name: 'height' },
  { output_model: 'm-dd', output_param_name: 'INT', input_model: 'm-g', input_param_name: 'seed' },
  { output_model: 'm-a', output_param_name: 'frame_count', input_model: 'm-c', input_param_name: 'batch_size' },
  { output_model: 'm-a', output_param_name: 'current_image', input_model: 'm-b', input_param_name: 'image' },
  { output_model: 'm-c', output_param_name: 'LATENT', input_model: 'm-g', input_param_name: 'latent_image' },
  { output_model: 'm-f', output_param_name: 'MODEL', input_model: 'm-g', input_param_name: 'model' },
  { output_model: 'm-j', output_param_name: 'CLIP', input_model: 'm-h', input_param_name: 'clip' },
  { output_model: 'm-j', output_param_name: 'CLIP', input_model: 'm-i', input_param_name: 'clip' },
  { output_model: 'm-j', output_param_name: 'MODEL', input_model: 'm-f', input_param_name: 'model' },
  { output_model: 'm-a', output_param_name: 'frame_count', input_model: 'm-k', input_param_name: 'max_frames' },
  { output_model: 'm-h', output_param_name: 'CLIP', input_model: 'm-k', input_param_name: 'clip' },
  { output_model: 'm-l', output_param_name: 'CONTEXT_OPTIONS', input_model: 'm-f', input_param_name: 'context_options' },
  { output_model: 'm-i', output_param_name: 'CONDITIONING', input_model: 'm-m', input_param_name: 'negative' },
  { output_model: 'm-k', output_param_name: 'CONDITIONING', input_model: 'm-m', input_param_name: 'positive' },
  { output_model: 'm-m', output_param_name: 'negative', input_model: 'm-n', input_param_name: 'negative' },
  { output_model: 'm-m', output_param_name: 'positive', input_model: 'm-n', input_param_name: 'positive' },
  { output_model: 'm-n', output_param_name: 'negative', input_model: 'm-o', input_param_name: 'negative' },
  { output_model: 'm-n', output_param_name: 'positive', input_model: 'm-o', input_param_name: 'positive' },
  { output_model: 'm-mc', output_param_name: 'CONTROL_NET', input_model: 'm-m', input_param_name: 'control_net' },
  { output_model: 'm-nc', output_param_name: 'CONTROL_NET', input_model: 'm-n', input_param_name: 'control_net' },
  { output_model: 'm-oc', output_param_name: 'CONTROL_NET', input_model: 'm-o', input_param_name: 'control_net' },
  { output_model: 'm-p', output_param_name: 'VAE', input_model: 'm-q', input_param_name: 'vae' },
  { output_model: 'm-g', output_param_name: 'LATENT', input_model: 'm-q', input_param_name: 'samples' },
  { output_model: 'm-q', output_param_name: 'IMAGE', input_model: 'm-r', input_param_name: 'images' },
  { output_model: 'm-s', output_param_name: 'IMAGE', input_model: 'm-o', input_param_name: 'image' },
  { output_model: 'm-t', output_param_name: 'IMAGE', input_model: 'm-n', input_param_name: 'image' },
  { output_model: 'm-u', output_param_name: 'IMAGE', input_model: 'm-m', input_param_name: 'image' },
  { output_model: 'm-u', output_param_name: 'IMAGE', input_model: 'm-t', input_param_name: 'image' },
  { output_model: 'm-v', output_param_name: 'IMAGE', input_model: 'm-u', input_param_name: 'image' },
  { output_model: 'm-v', output_param_name: 'IMAGE', input_model: 'm-s', input_param_name: 'image' },
  { output_model: 'm-a', output_param_name: 'current_image', input_model: 'm-v', input_param_name: 'image' },
  { output_model: 'm-d', output_param_name: 'INT', input_model: 'm-v', input_param_name: 'width' },
  { output_model: 'm-e', output_param_name: 'INT', input_model: 'm-v', input_param_name: 'height' },
  { output_model: 'm-r', output_param_name: 'GIF', input_model: 'm-final', input_param_name: 'GIF' }
];

const ModelsRelationship = () => {
  const graphRef = useRef<RelationGraphComponent>(null);

  const graphOptions: RGOptions = {
    debug: false,
    defaultJunctionPoint: 'border',
    backgroundColor: '#222222',
    defaultNodeColor: 'rgba(100, 100, 100, 0.5)',
    placeOtherNodes: false,
    placeSingleNode: false,
    'defaultLineMarker': {
      'markerWidth': 20,
      'markerHeight': 20,
      'refX': 3,
      'refY': 3,
      'data': 'M 0 0, V 6, L 4 3, Z'
    },
    layout: {
      layoutName: 'fixed'
    }
  };

  const showGraph = async () => {
    const getModel = (modelId: string) => {
      return __graph_json_data.nodes.find(model => model.id === modelId);
    };
    const getModelOutputParam = (model: JsonNode, outputParamName: string) => {
      return model.data?.output.find(param => param.name === outputParamName);
    };
    const getModelInputParam = (model: JsonNode, inputParamName: string) => {
      return model.data?.input.find(param => param.name === inputParamName);
    };
    __graph_json_data.elementLines = modelRelations.map(relation => {
      const outputModel = getModel(relation.output_model);
      const outputParam = getModelOutputParam(outputModel, relation.output_param_name);
      const inputModel = getModel(relation.input_model);
      const inputParam = getModelInputParam(inputModel, relation.input_param_name);
      inputParam.type = outputParam.type; // fix data bug

      return {
        from: `O_${relation.output_model}-${relation.output_param_name}`,
        fromJunctionPoint: 'right',
        to: `I_${relation.input_model}-${relation.input_param_name}`,
        toJunctionPoint: 'left',
        text: '',
        lineShape: 6,
        lineWidth: 3,
        color: dataTypeColorMap[outputParam.type]
      };
    });
    // Using the position set for the node, it can be adjusted, saved to the database, and then reloaded to be displayed here.
    const nodesPosition = [
      {'id':'m-a','x':-930.1706372723322,'y':-357.62950890917},
      {'id':'m-b','x':-456.66745683694836,'y':-468.67267196080763},
      {'id':'m-c','x':18.272727272727273,'y':-297.27272727272725},
      {'id':'m-d','x':-459.07692307692304,'y':-383.7692307692307},
      {'id':'m-dd','x':-458.12172573189525,'y':-81.8335901386749},
      {'id':'m-e','x':-459.9230769230769,'y':-240.23076923076923},
      {'id':'m-f','x':502.27272727272725,'y':150},
      {'id':'m-g','x':982.421052631579,'y':-146.69856459330143},
      {'id':'m-h','x':498.14285714285717,'y':-301.29870129870125},
      {'id':'m-i','x':503,'y':-158.63636363636365},
      {'id':'m-j','x':15.420875420875419,'y':-173.28956228956224},
      {'id':'m-k','x':-22.447552447552397,'y':-3.681688681688712},
      {'id':'m-l','x':-456.8423948932424,'y':72.69742719892274},
      {'id':'m-m','x':16.882352941176407,'y':-545.5347593582887},
      {'id':'m-n','x':495.5050505050505,'y':-550.5555555555555},
      {'id':'m-o','x':976.8311688311688,'y':-554.1428571428571},
      {'id':'m-mc','x':18.012987012987004,'y':-684.6883116883117},
      {'id':'m-nc','x':492.012987012987,'y':-688.3766233766235},
      {'id':'m-oc','x':971.7272727272727,'y':-692.1038961038962},
      {'id':'m-p','x':502.5717703349282,'y':-12.715311004784702},
      {'id':'m-q','x':982.2282980177716,'y':-302.84347231715657},
      {'id':'m-r','x':1457.1701982228299,'y':-235.41695146958307},
      {'id':'m-final','x':1977.9903165303608,'y':-245.81082613290445},
      {'id':'m-s','x':967.2727272727273,'y':-845.6818181818181},
      {'id':'m-t','x':494.0909090909091,'y':-817.7272727272727},
      {'id':'m-u','x':17.762845849802375,'y':-868.7193675889328},
      {'id':'m-v','x':-456.2667477450086,'y':-678.9085334954902}
    ];
    __graph_json_data.nodes.forEach((model, index) => {
      const position = nodesPosition.find(n => n.id === model.id);
      if (position) {
        model.x = position.x;
        model.y = position.y;
      } else {
        model.x = index * 50;
        model.y = index * 50;
      }
    });
    const graphInstance = graphRef.current!.getInstance();
    graphInstance.setJsonData(__graph_json_data).then(() => {
      graphInstance.moveToCenter();
      graphInstance.zoomToFit();
    });
  };

  const onNodeClick = (node: RGNode, $event: RGUserEvent) => {
    console.log('onNodeClick:', node.text);
    const graphInstance = graphRef.current!.getInstance();
    const elLines = graphInstance.getElementLines();
    elLines.forEach((line: RGLine) => {
      const keywords = `_${node.id}-`;
      const refed = line.from.includes(keywords) || line.to.includes(keywords);
      line.animation = refed ? 2 : undefined;
    });
  };

  useEffect(() => {
    showGraph();
  }, []);
  return (
    <div>
      <div style={{ height: '100vh' }}>
        <RelationGraph
          ref={graphRef}
          options={graphOptions}
          onNodeClick={onNodeClick}
          nodeSlot={MyAIModel}
        >
            {/* You can put any content here, and it will be displayed in behind of the node. */}
        </RelationGraph>
      </div>
    </div>
  );
};

export default ModelsRelationship;

VipDemo4AIModel.tsx

javascript
import React from 'react'
import { RGNode, RGNodeSlotProps } from 'relation-graph-react'

import './VipDemo4AIModel.scss'

export const dataTypeColorMap = {
  1: '#2cbb03',
  2: '#1da9f5',
  3: '#ff8080',
  4: '#f39930',
  5: '#f39930',
  6: '#f12de0',
  7: '#5b05f1',
  8: '#f39930',
  9: '#ffffff',
}

export const MyAIModel: React.FC<RGNodeSlotProps> = ({ node }) => {
  return (
    <div className="c-model">
      <div className="c-model-title">{node.text}</div>
      <div className="c-model-io">
        <div className="c-col-left">
          {node.data?.input.map((input: any) => (
            <div key={input.name} id={`I_${node.id}-${input.name}`}>
              <span
                className="el-icon-info c-dot"
                style={{ backgroundColor: dataTypeColorMap[input.type] }}
              />
              {input.name}
            </div>
          ))}
        </div>
        <div className="c-col-right">
          {node.data?.output.map((output: any) => (
            <div key={output.name} id={`O_${node.id}-${output.name}`}>
              {output.name}
              <i className="el-icon-info c-dot" style={{ color: dataTypeColorMap[output.type] }} />
            </div>
          ))}
        </div>
      </div>
      <div className="c-model-options">
        {node.data?.options.map((opt: any) => (
          <div key={opt.name}>
            {opt.name === '_textarea' ? (
              <div
                className="c-model-opt"
                style={{ padding: '0px', overflow: 'hidden', width: '400px' }}
              >
                <textarea
                  value={opt.value}
                  style={{
                    height: '100px',
                    width: '100%',
                    backgroundColor: 'transparent',
                    borderRadius: '10px',
                    color: '#ffffff',
                  }}
                  readOnly={true}
                ></textarea>
              </div>
            ) : opt.name === '_image' ? (
              <div className="c-model-opt" style={{ padding: '0px', overflow: 'hidden' }}>
                <img
                  src={opt.value}
                  style={{ width: '100%', userSelect: 'none', WebkitUserDrag: 'none' }}
                />
              </div>
            ) : (
              <div className="c-model-opt">
                <div className="c-col-left">
                  <div>{opt.name}</div>
                </div>
                <div className="c-col-right">
                  <div>{opt.value}</div>
                </div>
              </div>
            )}
          </div>
        ))}
      </div>
    </div>
  )
}

models-relationship.scss

scss
.relation-graph {
  .rel-node-shape-1 {
    overflow: hidden;
    border-radius: 10px;
  }
  .rel-toolbar {
    color: #ffffff;
    .c-current-zoom {
      color: #ffffff;
    }
  }
}

VipDemo4AIModel.scss

scss
.c-model {
  min-width: 350px;
  border-radius: 10px;
  overflow: hidden;
  .c-dot {
    background-color: #2cbb03;
    font-size: 10px;
    width: 8px;
    height: 8px;
    display: inline-block;
    border-radius: 50%;
    margin-left: 5px;
  }
  .c-model-title {
    background-color: rgba(0, 0, 0, 0.2);
    color: #cccccc;
    line-height: 28px;
  }
  .c-col-left {
    width: 50%;
    text-align: left;
  }
  .c-col-right {
    width: 50%;
    text-align: right;
  }
  .c-model-io {
    display: flex;
    padding-left: 10px;
    padding-right: 10px;
    line-height: 20px;
    .c-col-left {
    }
  }
  .c-model-options {
    padding-left: 20px;
    padding-right: 20px;
    padding-bottom: 5px;
    .c-model-opt {
      display: flex;
      background-color: rgba(0, 0, 0, 0.4);
      border-radius: 10px;
      line-height: 20px;
      margin-top: 10px;
      color: #ffffff;
      padding-left: 20px;
      padding-right: 20px;
    }
  }
}

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