• 欢迎访问搞代码网站,推荐使用最新版火狐浏览器和Chrome浏览器访问本网站!
  • 如果您觉得本站非常有看点,那么赶紧使用Ctrl+D 收藏搞代码吧

vue实现多栏布局拖拽

vue 搞代码 4年前 (2022-01-08) 34次浏览 已收录 0个评论
文章目录[隐藏]

这篇文章主要为大家详细介绍了vue实现多栏布局拖拽,改变盒子的宽度,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

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

本文来源gaodai.ma#com搞##代!^码@网3

内容如下

一、目标

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

二、应用场景

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

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

三、组件设计

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

组件设计实现目标:

  item1item2item3item4

四、实现

4.1 dragBox 静态页面

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

  <div style='width: 100%;height: 100%'> </div>

4.2 dragItem 页面

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

  <div class="d-flex" style="min-width: 200px"> <div style="width: 100%;height: 100%"> 默认信息</div><!-- 拖拽条 --><div class="resize" /></div> .resize { position: absolute; top: 0; right: 0; width: 4px; height: 100%; cursor: col-resize; background-color: #d6d6d6; } 

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() },

引入并注册使用该组件:

  <div id="app" style="width: 100%;height: 100vh;border:1px solid #ccc">  item1item2item3</div>

五、运行结果

具体样式可后期修改。

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持gaodaima搞代码网

以上就是vue实现多栏布局拖拽的详细内容,更多请关注gaodaima搞代码网其它相关文章!


搞代码网(gaodaima.com)提供的所有资源部分来自互联网,如果有侵犯您的版权或其他权益,请说明详细缘由并提供版权或权益证明然后发送到邮箱[email protected],我们会在看到邮件的第一时间内为您处理,或直接联系QQ:872152909。本网站采用BY-NC-SA协议进行授权
转载请注明原文链接:vue实现多栏布局拖拽

喜欢 (0)
[搞代码]
分享 (0)
发表我的评论
取消评论

表情 贴图 加粗 删除线 居中 斜体 签到

Hi,您需要填写昵称和邮箱!

  • 昵称 (必填)
  • 邮箱 (必填)
  • 网址