vue实现多栏布局拖拽

 更新时间:2021年09月08日 14:44:27   作者:你的小余童鞋  
这篇文章主要为大家详细介绍了vue实现多栏布局拖拽,改变盒子的宽度,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

本文实例为大家分享了vue实现多栏布局拖拽的具体代码,供大家参考,具体内容如下

一、目标

vue 实现多个盒子(用户根据实际场景决定盒子数量)自由拖拽,改变宽度。

二、应用场景

可自由拖动宽度的多栏布局。

最典型的案例:编辑器(eg:vscode,idea等)

三、组件设计

由于该组件盒子数量不确定,所以我们设计组件时参考了Vuetify中的Form和FormItem的设计。即:外层大盒子处理分发的拖拽事件,里层的盒子负责展示各个Item的内容。

组件设计实现目标:

<drag-box style="width: 100%; height: 100%;">
   <drag-item>item1</drag-item>
   <drag-item>item2</drag-item>
   <drag-item>item3</drag-item>
   <drag-item>item4</drag-item>
</drag-box>

四、实现

4.1 dragBox 静态页面

(通过插槽实现子元素的嵌套)

<template>
    <div ref='dragBox' style='display: flex; width: 100%; height: 100%;'>
        <slot></slot>
    </div>
</template>

4.2 dragItem 页面

(通过插槽实现drag-item内部元素的嵌套)

<template>
    <div ref="container" class="d-flex" style="min-width: 200px; position: relative;">
        <div style="width: 100%; height: 100%;">
            <slot>默认信息</slot>
        </div>
  <!-- 拖拽条 -->
        <div v-if="resizeShow" class="resize" />
    </div>
</template>
<script>
export default {
  props: {
  // 控制拖拽条的是否显示,默认显示
    resizeShow: {
      type: Boolean,
      default: true
    }
  }
}
</script>
<style>
.resize {
    position: absolute;
    top: 0;
    right: 0;
    width: 4px;
    height: 100%;
    cursor: col-resize;
    background-color: #d6d6d6;
}
</style>

4.3 拖拽逻辑

拖拽的逻辑应当交给dragBox处理,而非dragItem。

4.3.1 在实现拖拽之前,应当给子元素(即:dragItem)进行合理布局。

当用户未给 dragItem 分配初始宽度时,则默认flex:1(平均分配剩余空间)。具体逻辑如下:

 // 如果dragItem 没有定义宽度,则flex=1
    setDragItemFlex () {
      const dragBox = this.$refs.dragBox
      const childsLen = dragBox.children.length

      for (let i = 0; i < childsLen; i++) {
        const node = dragBox.children[i]
        if (!node.style.width) {
          // 如果没有定义宽度,则flex=1
          node.style.flex = 1
        }
      }
    },

4.3.2 拖拽实现逻辑

应当为每个dragItem的拖动条添加拖动事件。完整的拖动事件包括:鼠标按下,鼠标移动,鼠标抬起(拖动结束)。

循环为每个拖动条添加事件:

dragControllerDiv () {
  const resize = document.getElementsByClassName('resize') // 拖拽条
  // 循环为每个拖拽条添加事件
  for (let i = 0; i < resize.length; i++) {
    // 鼠标按下事件
    resize[i].addEventListener('mousedown', this.onMouseDown)
  }
},

鼠标按下逻辑:获取鼠标按下的初始位置,改变拖拽条颜色,添加move和up的监听事件。

onMouseDown (e) {
  this.resizeBox = e.target
  this.currentBox = this.resizeBox.parentNode// 当前盒子
  this.rightBox = this.getNextElement(this.currentBox)// 当前盒子的下个兄弟节点
  if (!this.rightBox) return
  this.curLen = this.currentBox.clientWidth
  this.otherBoxWidth = this.$refs.dragBox.clientWidth - this.currentBox.clientWidth - this.rightBox.clientWidth // 其他盒子的宽度
  // 颜色改变提醒
  this.resizeBox.style.background = '#818181'
  this.startX = e.clientX
  document.addEventListener('mousemove', this.onMousemove)
  document.addEventListener('mouseup', this.onMouseup)
},

// 获取下一个兄弟元素的兼容函数
getNextElement (element) {
  if (element.nextElementSibling) {
    return element.nextElementSibling
  } else {
    var next = element.nextSibling// 下一个兄弟节点
    while (next && next.nodeType !== 1) { // 有 并且 不是我想要的
      next = next.nextSibling
    }
    return next
  }
}

鼠标移动事件:计算并设置当前盒子和右侧盒子的宽度。

onMousemove (e) {
  const endX = e.clientX
  const moveLen = endX - this.startX // (endx-startx)= 移动的距离
  const CurBoxLen = this.curLen + moveLen // resize[i].left+移动的距离=左边区域最后的宽度
  const rightBoxLen = this.$refs.dragBox.clientWidth - CurBoxLen - this.otherBoxWidth // 右侧宽度=总宽度-左侧宽度-其它盒子宽度
  // 当最小宽度时,无法继续拖动
  if (CurBoxLen <= 200 || rightBoxLen <= 200) return
  this.currentBox.style.width = CurBoxLen + 'px'// 当前盒子的宽度
  this.resizeBox.style.left = CurBoxLen // 设置左侧区域的宽度
  this.rightBox.style.width = rightBoxLen + 'px'
},

鼠标抬起事件:销毁mousedown和mousemove事件;恢复拖拽条的颜色。

onMouseup () {
 // 颜色恢复
 this.resizeBox.style.background = '#d6d6d6'
 document.removeEventListener('mousedown', this.onMouseDown)
 document.removeEventListener('mousemove', this.onMousemove)
},

在mounted钩子函数里添加对应的事件。

mounted () {
  this.setDragItemFlex()
  this.dragControllerDiv()
},

引入并注册使用该组件:

<template>
  <div id="app" style="width: 100%; height: 100vh; border:1px solid #ccc;">
    <drag-box style="width: 100%; height: 100%;">
      <drag-item style="width: 20%;">item1</drag-item>
      <drag-item>item2</drag-item>
      <drag-item style="width: 20%;" :resizeShow='false'>item3</drag-item>
    </drag-box>
  </div>
</template>

<script>
import {DragBox, DragItem} from './components/dragLayouter'

export default {
  name: 'App',
  components: {
    DragBox,
    DragItem
  }
}
</script>

五、运行结果

具体样式可后期修改。

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持脚本之家。

相关文章

  • 使用Vue绑定class和style样式的几种写法总结

    使用Vue绑定class和style样式的几种写法总结

    这篇文章主要介绍了使用Vue绑定class和style样式的几种写法,文章通过代码示例介绍的非常详细,具有一定的参考价值,需要的朋友可以参考下
    2023-07-07
  • 详解Vue-Router的安装与使用

    详解Vue-Router的安装与使用

    Vue Router 是 Vue.js 官方的路由管理器。它和 Vue.js 的核心深度集成,让构建单页面应用变得易如反掌。本文介绍下Vue Router的安装与使用
    2021-06-06
  • Element-ui tree组件自定义节点使用方法代码详解

    Element-ui tree组件自定义节点使用方法代码详解

    本文通过实例代码给大家介绍了Element-ui tree组件自定义节点使用方法 ,非常不错,具有一定的参考借鉴价值,需要的朋友可以参考下
    2018-09-09
  • vue.js修改el-popover组件显示位置

    vue.js修改el-popover组件显示位置

    el-popover是一个基于element-ui框架设计的,用于浮动展示提示或菜单的UI组件,下面这篇文章主要给大家介绍了关于vue.js修改el-popover组件显示位置的相关资料,需要的朋友可以参考下
    2023-06-06
  • Vue 2中ref属性的使用方法及注意事项

    Vue 2中ref属性的使用方法及注意事项

    这篇文章主要给大家介绍了关于Vue 2中ref属性的使用方法及注意事项的相关资料,文中通过示例代码介绍的非常详细,对大家具有一定的参考学习价值,需要的朋友们下面跟着小编一起来学习学习吧。
    2017-06-06
  • vant 自定义 van-dropdown-item的用法

    vant 自定义 van-dropdown-item的用法

    这篇文章主要介绍了vant 自定义 van-dropdown-item的用法,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2020-08-08
  • django+vue项目搭建实现前后端通信

    django+vue项目搭建实现前后端通信

    本文主要介绍了django+vue项目搭建实现前后端通信,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2023-02-02
  • vue2中使用sass并配置全局的sass样式变量的方法

    vue2中使用sass并配置全局的sass样式变量的方法

    这篇文章主要介绍了vue2中使用sass并配置全局的sass样式变量的方法,本文给大家介绍的非常详细,具有一定的参考借鉴价值,需要的朋友可以参考下
    2018-09-09
  • vue.js Router嵌套路由

    vue.js Router嵌套路由

    这篇文章主要介绍vue.js Router嵌套路由,平时访问首页,里面有新闻类的/home/news,还有信息类的/home/message这时候就需要使用到嵌套路由,下面我们就来具体学习嵌套路由的原理,需要的朋友可以参考一下
    2021-10-10
  • 详解vue + vuex + directives实现权限按钮的思路

    详解vue + vuex + directives实现权限按钮的思路

    这篇文章主要介绍了详解vue + vuex + directives实现权限按钮的思路,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2017-10-10

最新评论