效果演示图
可拖拽侧边栏的使用情况非常多啊,博客园后台管理左侧边栏就可以拖拽哟!废话不多说,本随笔实现的可拖拽侧边栏效果演示图如下:
HTML 代码
<div class="container">
<div class="left">
<div class="resize-bar"></div>
</div>
<div class="right">
<div class="resize-bar"></div>
</div>
</div>
CSS 代码
html,
body {
display: flex;
justify-content: center;
align-items: center;
align-content: center;
padding: 0 !important;
margin: 0 !important;
}
.container {
width: 80vw;
height: 100vh;
display: flex;
justify-content: space-between;
align-items: center;
align-content: center;
}
.left {
position: relative;
width: 100px;
height: 100%;
background-color: rgb(160, 212, 233);
}
.left .resize-bar {
position: absolute;
top: 0;
left: 100px;
width: 3px;
height: 100%;
opacity: 0;
}
.left .resize-bar:hover {
cursor: col-resize;
opacity: 1;
background-color: rgb(210, 85, 50);
}
.right {
position: relative;
width: 100px;
height: 100%;
background-color: rgb(36, 107, 214);
}
.right .resize-bar {
position: absolute;
top: 0;
right: 100px;
width: 3px;
height: 100%;
opacity: 0;
}
.right .resize-bar:hover {
cursor: col-resize;
opacity: 1;
background-color: rgb(211, 36, 164);
}
JS 代码
拖拽右侧边栏
如上图,红色方框是 container 的区域,我们的侧边栏可移动范围也是在 container 移动范围内。
鼠标按住不放拖拽右侧边栏,右侧边栏的宽度此时是逐渐增大的趋势。鼠标移动多少 px 根据用户移动鼠标的速率决定。假如,鼠标移动到 1200px 的位置,右侧边栏左边缘往左移动 startWidth 个像素点的距离。那么,startWidth 该怎么计算呢?
初始阶段,右侧边栏边缘(startWidth)应该是整个 container 的宽度减去右侧边栏的宽度。假如,container 的宽度是 1355 px,右侧边栏的宽度是 100 px;那么,startWidth = 1355px - 100px = 1255px。此时的鼠标移动到 1200px,那么右侧边栏应该从边缘部分起移动 1255px - 1200px = 55px 的距离。也就是说,原来的右侧边栏宽度 100px 应该变成 100px + 55px = 155px。
const container = document.querySelector(".container");
const right = document.querySelector(".right");
const rightResizeBar = document.querySelector(".right .resize-bar");
function moveRightBar(event) {
setTimeout(() => {
let startWidth = container.clientWidth - right.clientWidth;
let shiftWidth = startWidth - event.pageX + container.offsetLeft + right.clientWidth;
right.style.width = shiftWidth + "px";
rightResizeBar.style.right = shiftWidth + "px";
}, 200);
}
这里要特别声明,pageX、clientX 都是一样的数值,用哪个都可以。而且,代码中还加了一个 container.offsetLeft。因为我们的 container 可能是被 flex 布局设置了居中,那么此时 container 与浏览器窗口有一个 left 偏移量,所以需要加上这个 left 以保证移动距离的准确性。如效果演示图中,container 并没有紧挨着浏览器窗口的左边边缘处。
添加监听器
鼠标点击到 rightResizeBar 元素时,就应该开启整个 container 的 mousemove 事件:
// 1. 鼠标按下 rightResizeBar 元素,开启 container 的事件监听
rightResizeBar.addEventListener("mousedown", () => {
container.addEventListener("mousemove", moveRightBar);
});
// 2. 当鼠标从 rightResizeBar 放下时,取消监听
rightResizeBar.addEventListener("mouseup", () => {
container.removeEventListener("mousemove", moveRightBar);
});
// 3. 当鼠标从 container 范围内放下时,取消监听
container.addEventListener("mouseup", () => {
container.removeEventListener("mousemove", moveRightBar);
});
鼠标放下时,清除事件监听。上面的代码中 rightResizeBar 元素和 container 都清除了 container 的 mousemove 事件,这是为了确保清除干净才这样做的。
拖拽左侧边栏
左侧边栏就非常简单了,因为鼠标的移动与浏览器的 x、y 有关系,所以,左侧边栏的右边缘处移动正符合上面右侧边栏那样的 startWidth,而且不需要计算。但是,必须要加上 container 的 left 偏移量,同上。
function moveLeftBar(event) {
setTimeout(() => {
let shiftWidth = event.pageX - container.offsetLeft;
left.style.width = shiftWidth + "px";
leftResizeBar.style.left = shiftWidth + "px";
}, 200);
}
添加事件监听
leftResizeBar.addEventListener("mousedown", () => {
container.addEventListener("mousemove", moveLeftBar);
});
leftResizeBar.addEventListener("mouseup", () => {
container.removeEventListener("mousemove", moveLeftBar);
});
container.addEventListener("mouseup", () => {
container.removeEventListener("mousemove", moveLeftBar);
});