likes
comments
collection
share

低代码实现之-拖拽与辅助线

作者站长头像
站长
· 阅读数 50

先上效果图炸波🐟

低代码实现之-拖拽与辅助线

拖拽实现

这里我们使用react-draggable三方库,由于它默认的Draggable是去动态修改元素的transform,和我们设计的初衷背离,于是使用其DraggableCore模块去自定义拖拽方式,具体实现参考官方文档,这里不再多赘述

<DraggableCore
        onDrag={drag}
        onStop={stop}
        onStart={start}
      >
        {
          React.cloneElement(
            this.props.children
          )
        }
</DraggableCore>

辅助线

如何得到正确的辅助线 需要几步?

  • 一:得到当前拖拽元素的left top值
  • 二:获得当前拖拽元素 和 参考元素(画布上非当前拖拽元素的其他元素)的相关信息,包括:width,height,left,top等
  • 三:遍历对比元素,通过如下x、y轴的分析生成辅助线数组
  • 四:根据当前拖拽元素的left top筛选出应该显示的辅助线

步骤一 获取元素的left top

drag = (ev: DraggableEvent, b: DraggableData) => {
  // 动态计算当前的left top
  const dragX = b.lastX - this.lastX;
  const dragY = b.lastY - this.lastY;
};

步骤二 获取当前元素 和 参考元素

简单概述就是一次map和一次filter 此处省略...

步骤三 生成辅助线

y轴辅助线情况分解

图中B为拖拽中的当前元素,A为静态参考元素,triggerTop为横辅助线触发条件,lineTop为横辅助线的y坐标

顶对底应理解为当前元素的顶部对齐参考元素的底部,其他方向也应如此理解,前者为当前元素,后者为参考元素 低代码实现之-拖拽与辅助线

  • 顶对顶 : triggerTop = A.top, lineTop = A.top
  • 顶对底 : triggerTop = A.top - B.height, lineTop = A.top
  • 中对中 : triggerTop = A.top + A.height/2 - B.height/2, lineTop = A.top + A/height/2
  • 底对顶 : triggerTop = A.top + A.height, lineTop = A.top + A.height
  • 底对底 : triggerTop = A.tio + A.height - B.height, lineTop = A.top + A.height

如图所示,产生y轴方向的辅助线有以上5种情况,根据这5种情况,可以用 当前元素(target)参考元素(compare) 的信息创建出一组我们需要的数据

x轴辅助线情况分解

有了y轴的分析经验,相信x轴也会不攻自破,triggerLeft为竖辅助线触发条件,lineLeft为竖辅助线的x坐标

低代码实现之-拖拽与辅助线

  • 左对左 : triggerLeft = A.left, lineLeft = A.left
  • 左对右 : triggerLeft = A.left + A.width, lineLeft = A.left + A.width
  • 中对中 : triggerLeft = A.left+ A.width/2 - B.width/2, lineLeft = A.left + A.width/ 2
  • 右对右 : triggerLeft: A.left + A.width - B.width, lineLeft = A.left + A.width
  • 右对左 : triggerLeft = A.left - B.width, lineLeft = A.left

转换成代码:

const directionsMap = {
  y: [
    {
      name: '顶对顶',
      calc: (compare) => ({
        lineTop: compare.t,
        triggerTop: compare.t
      })
    },
    {
      name: '底对顶',
      calc: (compare, target) => ({
        lineTop: compare.t,
        triggerTop: compare.t - target.h
      })
    },
    {
      name: '中对中',
      calc: (compare, target) => ({
        lineTop: compare.t + compare.h / 2,
        triggerTop: compare.t + compare.h / 2 - target.h / 2
      })
    },
    {
      name: '顶对底',
      calc: (compare) => ({
        lineTop: compare.t + compare.h,
        triggerTop: compare.t + compare.h
      })
    },
    {
      name: '底对底',
      calc: (compare, target) => ({
        lineTop: compare.t + compare.h,
        triggerTop: compare.t + compare.h - target.h
      })
    }],
  x: [
    {
      name: '左对左',
      calc: (compare) => ({
        lineLeft: compare.l,
        triggerLeft: compare.l
      })
    },
    {
      name: '左对右',
      calc: (compare) => ({
        lineLeft: compare.l + compare.w,
        triggerLeft: compare.l + compare.w
      })
    },
    {
      name: '中对中',
      calc: (compare, target) => ({
        lineLeft: compare.l + compare.w / 2,
        triggerLeft: compare.l + compare.w / 2 - target.w / 2
      })
    },
    {
      name: '右对右',
      calc: (compare, target) => ({
        lineLeft: compare.l + compare.w,
        triggerLeft: compare.l + compare.w - target.w
      })
    },
    {
      name: '右对左',
      calc: (compare, target) => ({
        lineLeft: compare.l,
        triggerLeft: compare.l - target.w
      })
    },
  ],
};


  calcLines(values, target, compares) {
    let {x, y} = values;
    // threshold为临界阀值 默认5
    const { threshold } = this.props;
    const lines: { y: YLine[], x: XLine[] } = {
      y: [],
      x: []
    };

    for (let i = 0; i < compares.length; i++) { // 和其他所有参考物遍历
      const compare = compares[i];
      directionsMap.y.forEach((item) => { // 构造横辅助线数据
        lines.y.push(item.calc(compare, target));
      });
      directionsMap.x.forEach((item) => { // 构造竖辅助线数据
        lines.x.push(item.calc(compare, target));
      })
    }
  }

步骤四 筛选辅助线


    let yLine = null, xLine = null;

    for (let i = 0; i < lines.y.length; i++) {
      const { triggerTop, lineTop } = lines.y[i];
      if (Math.abs(y - triggerTop) < threshold) {
        // 满足竖辅助线出现条件 为xLine赋值
        yLine = lineTop;
        // 实现吸附效果
        y = triggerTop;
        break;
      }
    }
    for (let j = 0; j <lines.x.length; j++) {
      const { triggerLeft, lineLeft } = lines.x[j];
      if (Math.abs(x - triggerLeft) < threshold) {
        // 满足竖辅助线出现条件 为xLine赋值
        xLine = lineLeft;
        // 实现吸附效果
        x = triggerLeft;
        break;
      }
    }

    if (yLine !== this.state.yLine || xLine !== this.state.xLine) {
      // 需要展示的横、竖辅助线
      this.setState({
        yLine,
        xLine,
      });
    }

    return { x, y }; // 拖拽元素的最终位置
转载自:https://juejin.cn/post/7047810320923885575
评论
请登录