zoukankan      html  css  js  c++  java
  • 用Canvas画一棵二叉树

    笔墨伺候

    var canvas = document.getElementById('canvas');
    var ctx = canvas.getContext('2d');
    // 然后便可以挥毫泼墨了

    树的样子

    clipboard.png

    const root = {
          value: 'A',
          label: '100',
          left: {
            value: 'B',
            label: '70',
            left: {
              value: 'D',
              label: '40',
              left: {
                value: 'H',
                label: '20',
                left: null,
                right: null
              },
              right: {
                value: 'I',
                label: '20',
                left: null,
                right: null
              }
            },
            right: {
              value: 'E',
              label: '30',
              left: null,
              right: null
            }
          },
          right: {
            value: 'C',
            label: '30',
            left: {
              value: 'F',
              label: '15',
              left: null,
              right: null
            },
            right: {
              value: 'G',
              label: '15',
              left: null,
              right: null
            }
          }
        }

    构思构思

    这样一幅大作,无非就是由黑色的正方形+线段构成
    这正方形怎么画

    function drawRect(text, x, y, unit) {
      ctx.fillRect(x, y, unit, unit)
      // fillRect(x, y, width, height) 
      // x与y指定了在canvas画布上所绘制的矩形的左上角(相对于原点)的坐标
      // width和height设置矩形的尺寸。
      ctx.font = "14px serif"
      ctx.fillText(text, x + unit, y + unit) // 再给每个正方形加个名字
    }

    这直线怎么画

    function drawLine(x1, y1, x2, y2) {
      ctx.moveTo(x1, y1)
      ctx.lineTo(x2, y2)
      ctx.stroke()
    }

    这关系怎么画

    // 前序遍历二叉树
    function preOrderTraverse(root, x, y){
      drawRect(root.value, x, y)
      if(root.left){
        drawLine(x, y, ...)
        preOrderTraverse(root.left, ...)
      }
      if(root.right){
        drawLine(x, y, ...)
        preOrderTraverse(root.right, ...)
      }
    }

    现在遇到个小问题,如何确定节点的子节的位置?

    clipboard.png

    父节点与子结点在y轴上的距离固定,为正方形长度unit的两倍;父节点与子结点在x轴上的距离满足n2=(n1+2)*2-2,其中设父节点与子结点在x轴上最短的距离n0=1,即unit,而父节点与子结点在x轴上最长的距离取决于该树的层数。
    如何得到树的深度?

    function getDeepOfTree(root) {
      if (!root) {
        return 0
      }
      let left = getDeepOfTree(root.left)
      let right = getDeepOfTree(root.right)
      return (left > right) ? left + 1 : right + 1
    }

    这样父节点与子结点在x轴上最长的距离

    let distance = 1
    const deep = getDeepOfTree(root)
    for (let i = 2; i < deep; i++) {
      distance = (distance + 2) * 2 - 2
    }
    // distance*unit 即为父节点与子结点在x轴上最长的距离

    unit为正方形的长度,如何确定,假设canvas的宽度为1000,由深度deep可知,树的最大宽度为Math.pow(2, deep - 1),最底层的正方形占据4个unit

    clipboard.png

    所以unit是如此计算,const unit = 1000 / (Math.pow(2, deep - 1) * 4 + 8)+8是个备用空间。

    代码

    <html>
    
    <body>
      <canvas id="canvas" width="1000"></canvas>
      <script>
        const root = {
          value: 'A',
          label: '100',
          left: {
            value: 'B',
            label: '70',
            left: {
              value: 'D',
              label: '40',
              left: {
                value: 'H',
                label: '20',
                left: null,
                right: null
              },
              right: {
                value: 'I',
                label: '20',
                left: null,
                right: null
              }
            },
            right: {
              value: 'E',
              label: '30',
              left: null,
              right: null
            }
          },
          right: {
            value: 'C',
            label: '30',
            left: {
              value: 'F',
              label: '15',
              left: null,
              right: null
            },
            right: {
              value: 'G',
              label: '15',
              left: null,
              right: null
            }
          }
        }
        const canvas = document.getElementById('canvas')
        const ctx = canvas.getContext('2d')
    
        const deep = getDeepOfTree(root)
        let distance = 1
        for (let i = 2; i < deep; i++) {
          distance = (distance + 2) * 2 - 2
        }
        const unit = 1000 / (Math.pow(2, deep - 1) * 4 + 8)
        canvas.setAttribute('height', deep * unit * 4)
    
        const rootX = (1000 - unit) / 2
        const rootY = unit
        preOrderTraverse(root, rootX, rootY, distance)
        
        // 得到该树的高度
        function getDeepOfTree(root) {
          if (!root) {
            return 0
          }
          let left = getDeepOfTree(root.left)
          let right = getDeepOfTree(root.right)
          return (left > right) ? left + 1 : right + 1
        }
    
        function preOrderTraverse(root, x, y, distance) {
          drawRect(root.value, x, y) // 绘制节点
          if (root.left) {
            drawLeftLine(x, y + unit, distance)
            preOrderTraverse(root.left, x - (distance + 1) * unit, y + 3 * unit, distance / 2 - 1)
          }
          if (root.right) {
            drawRightLine(x + unit, y + unit, distance)
            preOrderTraverse(root.right, x + (distance + 1) * unit, y + 3 * unit, distance / 2 - 1)
          }
        }
    
        function drawRect(text, x, y) {
          ctx.fillRect(x, y, unit, unit)
          ctx.font = "14px serif"
          ctx.fillText(text, x + unit, y + unit)
        }
    
        function drawLeftLine (x, y, distance) {
          ctx.moveTo(x, y)
          ctx.lineTo(x - distance * unit, y + 2 * unit)
          ctx.stroke()
        }
    
        function drawRightLine (x, y, distance) {
          ctx.moveTo(x, y)
          ctx.lineTo(x + distance * unit, y + 2 * unit)
          ctx.stroke()
        }
      </script>
    </body>
    
    </html>

    来点互动

    实现移动至节点出现tooltip

    首先要有tooltip

    <div id="tooltip" style="position:absolute;"></div>
    ...
    const tooltip = document.getElementById('tooltip')

    由于canvas是一个整体元素,所以只能给canvas绑定事件,根据鼠标的坐标,判断是否落在某个正方形区域内
    这里有个关健个函数

    ctx.rect(0, 0, 100, 100)
    ctx.isPointInPath(x, y)
    // 判断x,y是否落在刚刚由path绘制出的区域内

    所以在绘制正方形时还要将其path记下来

    let pathArr = []
    function preOrderTraverse(root, x, y, distance) {
      pathArr.push({
        x,
        y,
        value: root.value,
        label: root.label
      })
      // 记录正方形左上角的位置,就可以重绘路径
      drawRect(root.value, x, y) // 绘制节点
      if (root.left) {
        drawLeftLine(x, y + unit, distance)
        preOrderTraverse(root.left, x - (distance + 1) * unit, y + 3 * unit, distance / 2 - 1)
      }
      if (root.right) {
        drawRightLine(x + unit, y + unit, distance)
        preOrderTraverse(root.right, x + (distance + 1) * unit, y + 3 * unit, distance / 2 - 1)
      }
    }

    绑定事件

    // 模拟鼠标hover效果
    canvas.addEventListener('mousemove', (e) => {
      let i = 0
      while (i < pathArr.length) {
        ctx.beginPath()
        ctx.rect(pathArr[i].x, pathArr[i].y, unit, unit)
        if (ctx.isPointInPath(e.offsetX, e.offsetY)) {
          canvas.style.cursor = 'pointer'
          tooltip.innerHTML = `<span style="font-size:14px;">${pathArr[i].label}</span>`
          tooltip.style.top = `${pathArr[i].y + unit + 4}px`
          tooltip.style.left = `${pathArr[i].x + unit}px`
          break
        } else {
          i++
        }
      }
      if (i === pathArr.length) {
        canvas.style.cursor = 'default'
        tooltip.innerHTML = ``
      }
    })

    线上demo

    JSBin地址

  • 相关阅读:
    vue项目搭建过程2 -- 使用 vue cli 4.0 搭建 vue 项目
    vue项目搭建过程1 -- 环境搭建
    升级node.js版本
    git的初步了解
    期末总结
    四则运算的封装
    用户故事
    0~10的随机整数运算
    创业近一年在博客园总结一下,希望给来者一点借鉴
    PV与并发之间换算的算法换算公式
  • 原文地址:https://www.cnblogs.com/10manongit/p/12839717.html
Copyright © 2011-2022 走看看