Antv X6 动态连线

news2025/1/24 22:29:00

这是我实际开发项目中,利用 X6 开发的一个关系图。具备连线功能。这里我尽可能全的记录整个开发思路和部分编码,如果你也用了 X6 希望对你有帮助。

创建画布

代码有删减,以下展示的代码全都有删减
index.vue

<template>
  <div id="dag-view"></div>
</template>

<script>
import { Graph } from '@antv/x6'
import { GraphCells } from './GraphCells'
export default {
  name: 'index',
  data () {
    return {
      graph: null,
      graphCells: new GraphCells()
    }
  },
  methods: {
    init () {
      this.graphCells.clear()
      this.graph = new Graph(this.graphCells.graphOptions({}))
      this.graphCells.graph = this.graph
    }
  },
  mounted () {
    this.init()
  }
}
</script>

GraphCells.js

class GraphCells {
  constructor () {
    this.graph = null
    this.cells = new Map()
    this.edges = []
  }

  clear () {
    this.graph = null
    this.cells.clear()
    this.edges = []
  }

  graphOptions (options) {
    return {
      container: document.getElementById('dag-view'),
      autoResize: true,
      // 是否可以拖动
      panning: false,
      grid: {
        size: 10,
        type: 'dot', // 'dot' | 'fixedDot' | 'mesh'
        visible: true,
        args: {
          color: '#a0a0a0', // 网格线/点颜色
          thickness: 1 // 网格线宽度/网格点大小
        }
      },
      highlighting: {
        magnetAvailable: {
          name: 'stroke',
          args: {
            attrs: {
              stroke: '#47C769'
            }
          }
        },
        magnetAdsorbed: {
          name: 'stroke',
          args: {
            attrs: {
              fill: '#fff',
              stroke: '#31d0c6'
            }
          }
        }
      },
      mousewheel: {
        enabled: true,
        modifiers: 'ctrl',
        factor: 1.1,
        maxScale: 1.5,
        minScale: 0.5
      },
      scroller: {
        enabled: true,
        pageVisible: false,
        pageBreak: false,
        pannable: true
      },
      connecting: {
        // 当 snap 设置为 true 时连线的过程中距离节点或者连接桩 50px 时会触发自动吸附
        snap: true,
        // 是否允许连接到画布空白位置的点
        allowBlank: false,
        // 是否允许创建循环连线
        allowLoop: false,
        // 拖动边时,是否高亮显示所有可用的连接桩或节点
        highlight: true
      },
      ...options
    }
  }
  
  ...
}

export {
  GraphCells
}

创建节点

DagNode.js

import { Node, ObjectExt, Dom } from '@antv/x6'

class DagNode extends Node {
  constructor (options) {
    super(options)
    this.options = options
  }
}

DagNode.config({
  zIndex: 2,
  width: 100,
  height: 28,
  markup: [
    {
      tagName: 'rect',
      selector: 'body'
    },
    {
      // 使用 foreignObject 渲染 HTML 片段
      tagName: 'foreignObject',
      attrs: {
      },
      children: [
        {
          // 当 tagName 指定的标签是 HTML 元素时,需要使用 HTML 元素的命名空间
          ns: Dom.ns.xhtml,
          tagName: 'body',
          attrs: {
            xmlns: Dom.ns.xhtml
          },
          style: {
            display: 'table-cell',
            // 设置上左和下左边框radius。在svg元素中很难做到一侧radius,所以这里选择html元素
            borderTopLeftRadius: '4px',
            borderBottomLeftRadius: '4px',
            // 背景颜色(下面介绍怎么动态设置)
            backgroundColor: '',
            textAlign: 'center',
            verticalAlign: 'middle',
            height: 28,
            padding: '0 5px'
          },
          children: [
            {
              tagName: 'i',
              attrs: {
                // 设置图标字体class
                class: 'iconfont iconzirenwu'
              },
              style: {
                fontSize: '16px',
                // 图标颜色(下面介绍怎么动态设置)
                color: ''
              }
            }
          ]
        }
      ]
    },
    {
      tagName: 'text',
      selector: 'label'
    }
  ],
  attrs: {
    body: {
      refWidth: '100%',
      refHeight: '100%',
      strokeWidth: 0,
      fill: '#ffffff',
      // stroke: '#5F95FF',
      rx: 4,
      ry: 4,
      filter: {
        name: 'highlight',
        args: {
          color: '#BFBFBF',
          width: 2,
          blur: 2,
          opacity: 0.5
        }
      }
    },
    label: {
      textWrap: {
        // text: 'lorem ipsum dolor lorem ipsum dolor lorem ipsum dolor',
        ellipsis: true,
        breakWord: true,
        width: -35
      },
      textAnchor: 'middle',
      textVerticalAnchor: 'middle',
      refX: 60,
      refY: '50%',
      refWidth: '80%',
      fontSize: 12,
      fill: '#333'
    }
  },
  // 定义连接桩
  ports: {
    groups: {
      right: {
        position: { name: 'right' },
        zIndex: 2,
        attrs: {
          portBody: {
            magnet: true,
            r: 3
          }
        }
      },
      in: {
        position: { name: 'left' },
        zIndex: 2,
        attrs: {
          portBody: {
            magnet: false,
            r: 0
          }
        }
      },
      out: {
        position: { name: 'right' },
        zIndex: 2,
        attrs: {
          portBody: {
            magnet: false,
            r: 0
          }
        }
      }
    }
  },
  portMarkup: {
    tagName: 'circle',
    selector: 'portBody',
    attrs: {
      fill: '#fff',
      stroke: '#F08BB4',
      strokeWidth: 1
    }
  },
  propHooks (metadata) {
    const { label, ...others } = metadata
    if (label) {
      ObjectExt.setByPath(others, 'attrs/label/textWrap/text', label)
    }
    return others
  }
})

Node.registry.register('dag-node', DagNode, true)

export default DagNode

在 index.vue 中引,并添加两个节点到画布中

<template>
  <div id="dag-view"></div>
</template>

<script>
import { Graph } from '@antv/x6'
import { GraphCells } from './GraphCells'
// 注册自定义节点
import './DagNode'
export default {
  name: 'index',
  data () {
    return {
      graph: null,
      graphCells: new GraphCells()
    }
  },
  methods: {
    init () {
      this.graphCells.clear()
      this.graph = new Graph(this.graphCells.graphOptions({}))
      this.graphCells.graph = this.graph
      // 添加节点
      this.graph.addNode({
        shape: 'dag-node',
        x: 100,
        y: 100,
        label: 'foofoo' // 显示文本。这样写归功于自定义节点中的propHooks
      })

      this.graph.addNode({
        shape: 'dag-node',
        x: 300,
        y: 100,
        label: 'barbar'
      })
    }
  },
  mounted () {
    this.init()
  }
}
</script>

显示效果:
在这里插入图片描述

创建连线

DagEdge.js

import { Shape, Edge } from '@antv/x6'

/* 连线1 */
class CommonEdge extends Shape.Edge {
  // ...
}
CommonEdge.config({
  zIndex: 1,
  router: {
    name: 'er',
    args: {
      offset: 24,
      direction: 'H'
    }
  },
  connector: 'rounded',
  connectionPoint: 'boundary',
  attrs: {
    line: {
      stroke: '#BFBFBF',
      strokeWidth: 1,
      targetMarker: null
    }
  }
})
Edge.registry.register('common-edge', CommonEdge, true)

export {
  CommonEdge
}

在 index.vue 中引入 DagEdge,并添加连线

<template>
  <div id="dag-view"></div>
</template>

<script>
import { Graph } from '@antv/x6'
import { GraphCells } from './GraphCells'
// 注册自定义节点
import './DagNode'
// 引入自定义连线  
import './DagEdge'  
export default {
  name: 'index',
  data () {
    return {
      graph: null,
      graphCells: new GraphCells()
    }
  },
  methods: {
    init () {
      this.graphCells.clear()
      this.graph = new Graph(this.graphCells.graphOptions({}))
      this.graphCells.graph = this.graph
      // 添加节点
      this.graph.addNode({
        id: 'node1',
        shape: 'dag-node',
        x: 100,
        y: 100,
        label: 'foofoo',
        // 创建连接桩
        ports: {
          items: [
            {
              id: 'node1_out',
              group: 'out'
            }
          ]
        }
      })

      this.graph.addNode({
        id: 'node2',
        shape: 'dag-node',
        x: 300,
        y: 100,
        label: 'barbar',
        ports: {
          items: [
            {
              id: 'node2_in',
              group: 'in'
            }
          ]
        }
      })

      this.graph.addEdge({
        shape: 'common-edge',
        source: { cell: 'node1', port: 'node1_out' },
        target: { cell: 'node2', port: 'node2_in' }
      })
    }
  },
  mounted () {
    this.init()
  }
}
</script>

此时显示效果:
在这里插入图片描述

自定义连接

如果再添加一个节点 bazbaz,此时我想手动在 barbar 和 bazbaz 之间连接一条线,这条线和已有的线还不一样,该怎么做呢?
在 DagEdge 中添加另一条自定义连线:

import { Shape, Edge, Graph } from '@antv/x6'

/* 连线1 */
class CommonEdge extends Shape.Edge {
  // ...
}
CommonEdge.config({
  zIndex: 1,
  router: {
    name: 'er',
    args: {
      offset: 24,
      direction: 'H'
    }
  },
  connector: 'rounded',
  connectionPoint: 'boundary',
  attrs: {
    line: {
      stroke: '#BFBFBF',
      strokeWidth: 1,
      targetMarker: null
    }
  }
})
Edge.registry.register('common-edge', CommonEdge, true)

/* 连线2 */
class RelationEdge extends Shape.Edge {
  // hover加粗线
  hoverLine () {
    this.attr('line', {
      strokeWidth: 8
    })
  }

  // 清除hover
  clearHover () {
    this.attr('line', {
      strokeWidth: 1
    })
  }

  // 清除箭头和虚线
  clearMarker () {
    this.attr('line', {
      strokeDasharray: '',
      targetMarker: ''
    })
  }
}
RelationEdge.config({
  zIndex: 1,
  connector: 'rounded',
  connectionPoint: 'boundary',
  router: {
    name: 'oneSide',
    args: { side: 'right' }
  },
  attrs: {
    line: {
      stroke: '#F08BB4',
      strokeWidth: 1,
      strokeDasharray: 5, // 控制虚线间隔
      targetMarker: {
        name: 'classic',
        size: 5
      }
    }
  }
})
Edge.registry.register('relation-edge', RelationEdge, true)

export {
  CommonEdge,
  RelationEdge
}

在 GraphCells.js 中添加

import { RelationEdge } from './DagEdge'
graphOptions (options) {
    return {
      ...
      connecting: {
        ...
        // 连接的过程中创建新的边
        createEdge () {
          return new RelationEdge()
        },
        // 在移动边的时候判断连接是否有效,如果返回 false,当鼠标放开的时候,不会连接到当前元素,否则会连接到当前元素
        validateConnection ({ sourceView, targetView, targetMagnet }) {
          if (!targetMagnet) {
            return false
          }
          if (targetMagnet.getAttribute('port-group') !== 'right') {
            return false
          }
          return true
        }
      }
    }
  }

在 index.vue 中添加 bazbaz 节点

<template>
  <div id="dag-view"></div>
</template>

<script>
import { Graph } from '@antv/x6'
import { GraphCells } from './GraphCells'
// 注册自定义节点
import './DagNode'
// 引入自定义连线  
import './DagEdge'  
export default {
  name: 'index',
  data () {
    return {
      graph: null,
      graphCells: new GraphCells()
    }
  },
  methods: {
    init () {
      this.graphCells.clear()
      this.graph = new Graph(this.graphCells.graphOptions({}))
      this.graphCells.graph = this.graph
      // 添加节点
      this.graph.addNode({
        id: 'node1',
        shape: 'dag-node',
        x: 100,
        y: 100,
        label: 'foofoo',
        ports: {
          items: [
            {
              id: 'node1_out',
              group: 'out'
            }
          ]
        }
      })

      this.graph.addNode({
        id: 'node2',
        shape: 'dag-node',
        x: 300,
        y: 100,
        label: 'barbar',
        ports: {
          items: [
            {
              id: 'node2_in',
              group: 'in'
            },
            {
              id: 'node3_right',
              group: 'right'
            }
          ]
        }
      })

      this.graph.addNode({
        id: 'node3',
        shape: 'dag-node',
        x: 300,
        y: 200,
        label: 'bazbaz',
        ports: {
          items: [
            {
              id: 'node3_right',
              group: 'right'
            }
          ]
        }
      })

      this.graph.addEdge({
        shape: 'common-edge',
        source: { cell: 'node1', port: 'node1_out' },
        target: { cell: 'node2', port: 'node2_in' }
      })
    }
  },
  mounted () {
    this.init()
  }
}
</script>

显示效果:
在这里插入图片描述

添加事件

当鼠标移到连线上时,线加粗,并且显示删除按钮。
index.vue

<template>
  <div id="dag-view"></div>
</template>

<script>
import { Graph } from '@antv/x6'
import { GraphCells } from './GraphCells'
// 注册自定义节点
import './DagNode'
// 引入自定义连线  
import {RelationEdge} from './DagEdge'  
export default {
  name: 'index',
  data () {
    return {
      graph: null,
      graphCells: new GraphCells()
    }
  },
  methods: {
    init () {
      this.graphCells.clear()
      this.graph = new Graph(this.graphCells.graphOptions({}))
      this.graphCells.graph = this.graph
      // 添加节点
      this.graph.addNode({
        id: 'node1',
        shape: 'dag-node',
        x: 100,
        y: 100,
        label: 'foofoo',
        ports: {
          items: [
            {
              id: 'node1_out',
              group: 'out'
            }
          ]
        }
      })

      this.graph.addNode({
        id: 'node2',
        shape: 'dag-node',
        x: 300,
        y: 100,
        label: 'barbar',
        ports: {
          items: [
            {
              id: 'node2_in',
              group: 'in'
            },
            {
              id: 'node3_right',
              group: 'right'
            }
          ]
        }
      })

      this.graph.addNode({
        id: 'node3',
        shape: 'dag-node',
        x: 300,
        y: 200,
        label: 'bazbaz',
        ports: {
          items: [
            {
              id: 'node3_right',
              group: 'right'
            }
          ]
        }
      })

      this.graph.addEdge({
        shape: 'common-edge',
        source: { cell: 'node1', port: 'node1_out' },
        target: { cell: 'node2', port: 'node2_in' }
      })
     
      /* 添加事件 */
      this.graph.on('edge:mouseenter', ({ edge }) => {
        if (edge instanceof RelationEdge) {
          // 在 RelationEdge 中已定义
          edge.hoverLine()
          // X6 提供的小工具 https://x6.antv.vision/zh/docs/api/registry/edge-tool
          edge.addTools([
            {
              name: 'button-remove',
              args: {
                distance: '50%',
                offset: 0,
                // 删除回调
                // onClick ({ cell }) {
                // }
              }
            }
          ])
        }
      })
      
      this.graph.on('edge:mouseleave', ({ edge }) => {
        if (edge instanceof RelationEdge) {
          edge.clearHover()
          edge.removeTools()
        }
      })
      
      this.graph.on('edge:connected', ({ isNew, edge }) => {
        if (isNew) {
          edge.clearMarker()
        }
      })
    }
  },
  mounted () {
    this.init()
  }
}
</script>

此时效果:
在这里插入图片描述

使用布局

使用布局使节点按照一定形式排列
index.vue

<template>
  <div id="dag-view"></div>
</template>

<script>
import { Graph } from '@antv/x6'
import { GraphCells } from './GraphCells'
import { DagreLayout } from '@antv/layout'  
// 注册自定义节点
import './DagNode'
// 引入自定义连线  
import {RelationEdge} from './DagEdge'  
export default {
  name: 'index',
  data () {
    return {
      graph: null,
      dagreLayout: null,
      graphCells: new GraphCells()
    }
  },
  methods: {
    init () {
      this.graphCells.clear()
      this.graph = new Graph(this.graphCells.graphOptions({}))
      this.graphCells.graph = this.graph
     
      /* 添加事件 */
      this.graph.on('edge:mouseenter', ({ edge }) => {
        if (edge instanceof RelationEdge) {
          // 在 RelationEdge 中已定义
          edge.hoverLine()
          // X6 提供的小工具 https://x6.antv.vision/zh/docs/api/registry/edge-tool
          edge.addTools([
            {
              name: 'button-remove',
              args: {
                distance: '50%',
                offset: 0,
                // 删除回调
                // onClick ({ cell }) {
                // }
              }
            }
          ])
        }
      })
      
      this.graph.on('edge:mouseleave', ({ edge }) => {
        if (edge instanceof RelationEdge) {
          edge.clearHover()
          edge.removeTools()
        }
      })
      
      this.graph.on('edge:connected', ({ isNew, edge }) => {
        if (isNew) {
          edge.clearMarker()
        }
      })
      
      this.dagreLayout = new DagreLayout({
        type: 'dagre',
        rankdir: 'LR',
        align: undefined,
        ranksep: 50,
        nodesep: 5,
        controlPoints: true,
        begin: [50, 80]
      })
    },
      
    // 布局渲染
    graphLayout () {
      const newModel = this.dagreLayout.layout(this.graphCells.getModel())
      this.graph.fromJSON(newModel)
      this.graph.centerContent()
    },
      
    getData () {
      // 接口获取节点和连线信息。添加到 graphCells 中
      // this.graphCells.setCell(cell)
      //  this.graphCells.setEdge(edge)
      
      this.$nextTick(() => {
        this.graph && this.graphLayout()
      })
    }
  },
  created () {
    this.getData()
  },  
  mounted () {
    this.init()
  }
}
</script>

GraphCell.js

import { Dom, Shape } from '@antv/x6'
import { RelationEdge } from './DagEdge'

class GraphCells {
  constructor () {
    this.graph = null
    this.cells = new Map()
    this.edges = []
  }

  clear () {
    this.graph = null
    this.cells.clear()
    this.edges = []
  }

  graphOptions (options) {
    return {
      container: document.getElementById('dag-view'),
      autoResize: true,
      // 是否可以拖动
      panning: false,
      grid: {
        size: 10,
        type: 'dot', // 'dot' | 'fixedDot' | 'mesh'
        visible: true,
        args: {
          color: '#a0a0a0', // 网格线/点颜色
          thickness: 1 // 网格线宽度/网格点大小
        }
      },
      highlighting: {
        magnetAvailable: {
          name: 'stroke',
          args: {
            attrs: {
              stroke: '#47C769'
            }
          }
        },
        magnetAdsorbed: {
          name: 'stroke',
          args: {
            attrs: {
              fill: '#fff',
              stroke: '#31d0c6'
            }
          }
        }
      },
      mousewheel: {
        enabled: true,
        modifiers: 'ctrl',
        factor: 1.1,
        maxScale: 1.5,
        minScale: 0.5
      },
      scroller: {
        enabled: true,
        pageVisible: false,
        pageBreak: false,
        pannable: true
      },
      connecting: {
        // 当 snap 设置为 true 时连线的过程中距离节点或者连接桩 50px 时会触发自动吸附
        snap: true,
        // 是否允许连接到画布空白位置的点
        allowBlank: false,
        // 是否允许创建循环连线
        allowLoop: false,
        // 拖动边时,是否高亮显示所有可用的连接桩或节点
        highlight: true,
        createEdge () {
          return new RelationEdge()
        },
        validateConnection ({ sourceView, targetView, targetMagnet }) {
          if (!targetMagnet) {
            return false
          }
          if (targetMagnet.getAttribute('port-group') !== 'right') {
            return false
          }
          return true
        }
      },
      ...options
    }
  }

  // 根据节点类型生成连接桩
  getPortItems (id) {
    const items = []
    const inPort = {
      id: `port_${id}_in`,
      group: 'in'
    }
    const outPort = {
      id: `port_${id}_out`,
      group: 'out'
    }
    const rightPort = {
      id: `port_${id}_right`,
      group: 'right'
    }
    return items
  }

  /**
   * @function 添加节点
   * */
  setCell (options) {
    const id = options.id
    const cell = {
      shape: 'dag-node',
      // 注意添加节点数据时传入的makup会覆盖DagNode.config中的markup。
      // 所以DagNode中的markup可以删除,在这动态定义,就可以根据节点类型不同改变色值和图标
      markup: [
        {
          tagName: 'rect',
          selector: 'body'
        },
        {
          tagName: 'foreignObject',
          attrs: {
          },
          children: [
            {
              ns: Dom.ns.xhtml,
              tagName: 'body',
              attrs: {
                xmlns: Dom.ns.xhtml
              },
              style: {
                display: 'table-cell',
                borderTopLeftRadius: '4px',
                borderBottomLeftRadius: '4px',
                // 动态传入颜色
                backgroundColor: '',
                textAlign: 'center',
                verticalAlign: 'middle',
                height: 28,
                padding: '0 5px'
              },
              children: [
                {
                  tagName: 'i',
                  attrs: {
                    // 动态传入图标class
                    class: ''
                  },
                  style: {
                    fontSize: '16px',
                    // 动态传入颜色
                    color: ''
                  }
                }
              ]
            }
          ]
        },
        {
          tagName: 'text',
          selector: 'label'
        }
      ],
      ports: {
        items: this.getPortItems(id)
      },
      ...options
    }
    this.cells.set(id, cell)
  }

  /**
   * @function 添加连线
   * @param edge {object} 连线配置
   * */
  setEdge (edge) {
    const { source, target } = edge
    if (this.cells.has(source.cell) && this.cells.has(target.cell)) {
      this.edges.push({
        // 默认普通连线
        shape: 'common-edge',
        ...edge
      })
    }
  }

  /* graph model */
  getModel () {
    return {
      nodes: [...this.cells.values()],
      edges: this.edges
    }
  }
}

export {
  GraphCells
}

Tooltip

当节点 label 文本过长会显示 … ,鼠标移入显示 tooltip 显示全名。在 X6 中没有找到合适的小工具,我自己写了一个 Tooltip 组件。这里只是配合我的代码结构使用,并非封装完美的 Tooltip 组件。仅供大家参考。
Tooltip.vue

<template>
  <div id="lz_tooltip_container" class="lz_tooltip_container">
    <div class="tooltip">
      <span class="arrow"></span>
      <div class="cont"></div>
    </div>
  </div>
</template>

<script>
export default {
  name: 'Tooltip'
}
</script>

<style lang="scss">
  .lz_tooltip_container {
    position: fixed;
    left: -1000px;
    top: -1000px;
    z-index: 999;
  }

  .lz_tooltip_container > .tooltip {
    position: absolute;
    background: black;
    color: white;
    top: -40px;
    font-size: 14px;
    padding: 8px 16px;
    border-radius: 5px;
  }

  .lz_tooltip_container > .tooltip > .cont {
    white-space: nowrap
  }

  .lz_tooltip_container > .tooltip > .arrow {
    position: absolute;
    width: 0;
    height: 0;
    bottom: -6px;
    left: 24px;
    border-left: 6px solid transparent;
    border-right: 6px solid transparent;
    border-top: 6px solid #000000;
    filter: drop-shadow(0 2px 12px rgba(0,0,0, .03));
  }

  .lz_tooltip_container > .tooltip_hidden {
    display: none;
  }
</style>

我使用的是相对于屏幕的固定定位 fixed。因为在画布会上下滚动,左右平移,所以要根据节点相对画布的位置,转换成相对屏幕的位置然后动态设置 Tooltip 的 left top 属性。
在 index.vue 中 添加事件

<template>
  <div id="dag-view"></div>
</template>

<script>
import Tooltip from './Tooltip'  
export default {
  name: 'index',
   components: {
    Tooltip
  },
  methods: {
    init () {
      ...
      this.graph.on('node:mouseenter', ({ e, node, view }) => {
        // 节点相对画布位置
        const pos = node.position({ relative: true })
        // 转为相对屏幕位置
        const { x, y } = this.graph.localToClient(pos.x, pos.y)
        const tooltip = document.getElementById('lz_tooltip_container')
        if (!tooltip) return
        const cont = tooltip.querySelector('.cont')
        if (!cont) return
        cont.innerText = node.options.label
        tooltip.style.left = `${x + 20}px`
        tooltip.style.top = `${y - 5}px`
      })

      this.graph.on('node:mouseleave', ({ node }) => {
        const tooltip = document.getElementById('lz_tooltip_container')
        if (!tooltip) return
        tooltip.style.left = '-1000px'
        tooltip.style.top = '-1000px'
      })
    }
  }
}
</script>

效果展示:
在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/407846.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

React组件化的额外知识补充

文章目录React的额外补充Portals的使用Fragment的使用严格模式StrictModeReact的额外补充 Portals的使用 某些情况下&#xff0c;我们希望渲染的内容独立于父组件&#xff0c;甚至是独立于当前挂载到的DOM元素中(默认都是挂载到id为root的DOM 元素上的)。 Portal 提供了一种…

【vue】 vue-router安装和配置方法

vue-router 是 vue.js 官方的路由插件&#xff0c;里面组件和 URL 的映射关系由 vue-route 帮我们管理。 在 vue-router 的单页面应用中&#xff0c;页面的路径的改变就是组件的切换。 第一步&#xff1a; 1.正常初始化项目的时候&#xff0c;会有个 vue-router 供我们选择。…

在VSCode中配置代码自动 eslint 格式化(修改eslint规则、eslint忽略文件)

一、Eslint Eslint 是用来检测和规范代码格式的工具&#xff0c;应用在工程化项目中&#xff0c;可以保证项目代码格式的一致性和规范性&#xff0c;大大提升了代码的可读性。 二、配置过程 本博客是讲述对一个已经引用 eslint 依赖Nuxt项目&#xff08;vue项目应该相同&…

vue-cli脚手架的下载安装(靠谱)

找了半天才找到一个靠谱的安装教程&#xff0c;分享给你们。 1. 先下载node.js&#xff0c;下载地址&#xff1a;Download | Node.jsNode.js is a JavaScript runtime built on Chromes V8 JavaScript engine.https://nodejs.org/en/download/直接进入下载电脑对应的版本&…

vue-cli创建vue项目详细步骤

一、安装node环境&#xff08;建议使用LTS&#xff09; Download | Node.js 二、下载vue和vue-cli脚手架 命令&#xff1a;npm i -g vue ; npm i -g vue/cli 三、在想要创建的位置路径下打开cmd&#xff08;直接点击路径输入cmd即可打开当前位置的终端&#xff09; 四、创建v…

Vue使用Element-UI的table组件和后端接口进行数据交互(包含前后端代码)

前言 本次用element-ui的table组件&#xff0c;简单案例演示下前后端数据交互。 前提声明&#xff1a;如果不知道如何在vue中引入element-ui&#xff0c;可以先看下这篇文章:Vue引入并使用Element-UI组件库的两种方式 静态页面 首先先写一个静态页面吧&#xff0c;数据都是…

Vue3的vue-router路由详解

这篇文章是接着【三分钟快速搭建Vue3webpack项目】的内容做的开发&#xff0c;有基础的可以跳过 【三分钟快速搭建Vue3webpack项目】&#xff0c;直接看以下的内容。 Vue3的vue-router路由详解&#xff1a; 首先安装路由依赖模块&#xff1a; npm install vue-router4 所需…

618快到了送上自制前端小项目(html css js)

目录 &#x1f6a9;.自定义播放器 &#x1f3e0;.图片自动消失 ✨.小轮播图 &#x1f383;.旋转音乐盒 前言&#xff1a;这些小项目全都是自创的。 如果需要应用&#xff0c;或则转发的话请与 博主联系&#xff0c;感谢你们的理解&#xff0c; 1.自定义播放器 在页面中放置…

云化Web IDE,在线开发新模式

目录 前言 一、初识云IDE 二、CSDN 云IDE 1、如何使用云IDE 2、使用云IDE 三、云IDE的使用感受 四、总结 前言 工欲善其事必先利其器&#xff0c;作为程序员&#xff0c;我们在编写代码的时候&#xff0c;一定会选用一款得心应手的工具。就像行走江湖的侠客&#xff0c;手…

vue项目打包优化的方法

1.按需加载第三方库 例如 ElementUI、lodash 等 a, 装包 npm install babel-plugin-component -D b, babel.config.js module.exports {"presets": ["vue/cli-plugin-babel/preset"],"plugins": [["component",{"libraryNa…

uniapp中的分享功能实现(APP,小程序,公众号)

uniapp中的分享功能实现(APP,小程序&#xff0c;公众号) 1.APP端的分享 app端的分享可以直接使用uniapp封装的方法uni.share&#xff0c;uni-app的App引擎已经封装了微信、QQ、微博的分享SDK&#xff0c;开发者可以直接调用相关功能。可以分享到微信、QQ、微博&#xff0c;每个…

【Web前端】一文带你吃透CSS(上篇)

前端学习路线小总结: 基础入门:HTML CSS JavaScript三大主流框架:VUE REACT Angular深入学习:小程序 Node jQuery TypeScript 前端工程化一起学习CSS吧! 一.CSS简介1.什么是CSS?二.CSS语法1.语法规则2.注释三.CSS选择器

Vue打包后加载太慢,访问时间太久,记录项目的整个优化过程

问题背景 最近在做一个Vue项目时&#xff0c;在打包上线的时候发现项目部署完第一次访问时间特别慢&#xff0c;整个登录页面加载用了8-10秒&#xff0c;很明显这个速度达不到项目上线的要求&#xff0c;于是开始了对项目打包之后增快加载速度的研究。 未优化前的项目加载时间…

Vue3 响应式原理

响应式原理 Vue2 使用的是 Object.defineProperty Vue3 使用的是 Proxy 2.0的不足 对象只能劫持 设置好的数据&#xff0c;新增的数据需要Vue.Set(xxx) 数组只能操作七种方法&#xff0c;修改某一项值无法劫持。 reactive和effect的实现 export const reactive <T e…

Vue 解决报错 You are using the runtime-only build of Vue where the template compiler is not available.

报错信息 [Vue warn]: You are using the runtime-only build of Vue where the template compiler is not available. Either pre-compile the templates into render functions, or use the compiler-included build. 您正在使用Vue的仅运行时版本,并而模板编译器不可用。 可…

手把手教你如何对接支付宝支付接口(奶爸级别)

支付宝支付接口步骤&#xff1a;第一步&#xff1a;第二步&#xff1a;如何让支付宝将付款成功的通知&#xff0c;回调到你本机&#xff0c;我使用的是内网穿透&#xff1a;Natapp步骤&#xff1a; 第一步&#xff1a;准备支付宝相关的参数&#xff0c;如appId、公私钥 第二步…

vite中静态资源(css、img、svg等)的加载机制及其相关配置

什么是静态资源&#xff1f; 简单来说&#xff0c;我们开发完一个项目后&#xff0c;需要把它打包&#xff08;一般是dist文件夹&#xff09;&#xff0c;并部署在服务器上。那么&#xff0c;这个打包后的dist文件夹都是静态资源&#xff1b;在我们写项目时&#xff0c;图片、…

JS类型转换

JS中的类型转换方式一般分为三类&#xff1a;转为String型&#xff0c;转为数字型&#xff0c;转为布尔型。 一.转为String型 1.toString() 通过变量.toString()可以将其他类型的变量转换为字符串型&#xff08;null和undefined没有toString()方法&#xff09; let age 18;…

如何运行vue项目

一、 1、下载node.js 安装完成后分别在cmd中执行node -v查看是否安装成功&#xff0c;出现版本号就安装成功了 2、安装 webpack npm install webpack -g 安装完成后分别在cmd中执行npm -v查看是否安装成功&#xff0c;出现版本号就安装成功了 3、安装vue-cli脚手架 cnpm i…

【vue3】使用canvas

canvas是什么&#xff1f; 一个html5支持的新标签&#xff0c;见名知意&#xff0c;canvas就是画板的意思&#xff0c;可以在canvas上画画。css画三角形很简单&#xff0c;但是要画五角星呢&#xff0c;不妨试试canvas。 在html中使用canvas 1、canvas是html5中的一个标签。…