前端实现可拖拽课程表【纯HTML、CSS、JS】

news/2024/7/19 13:26:03 标签: 前端, html, css, js

前言

hello,今天实现点小动画,帮助学习理解Web api的拖拽效果,这里实现的是可拖拽的课程表!#

效果图

在这里插入图片描述
附:作者没钱去除水印,就这样看一下简单的看一下效果吧!

实现前言知识

这里我使用事件委托,统一将拖拽事件委托给父元素contaniner。

实现该元素可拖拽

这里我们使用dom的一个属性draggable,将该属性设置为true,即可拖拽

   <div  draggable="true"  class="color-1 item">语文</div>

拖拽开始事件

container.ondragstart,记录拖拽开始时触发的事件,只触发一次,返回拖拽元素本身的dom节点

//拖拽开始事件,只触发一次
container.ondragstart = e => {
    console.log('start', e.target)
}

在这里插入图片描述

拖拽结束事件

表示该元素拖拽到哪个元素之上,不断触发

container.ondragover = e => {
    e.preventDefault();
    // console.log('over', e.target)
}

拖拽移入事件

类似于mouse enter,记录该元素拖拽经过哪些元素,经过只触发一次

container.ondragenter = e => {
    // console.log('enter', e.target)
}

拖拽松开事件

拖拽放手在哪个元素,table,tr,td,这些元素是不允许元素拖拽到他的上面的,因此是不会触发这个事件的。为了触发需要阻止浏览器默认事件,在ondragover中阻止

container.ondrop =e=>{
// console.log('drop', e.target)
}
container.ondragover = e => {
    e.preventDefault();
    // console.log('over', e.target)
}

实现步骤

设定移动区域划分

首先,我这边有两块可以移入的区域,第一是课程类区域,第二是课程表格区域,因此,我们使用data-drop="move"data-drop="copy",分别代表课程类区域和课程表格区域。
这里我使用的是flex布局,将两块区域分为左右两边,也就是left和right。
在这里插入图片描述

 <div class="left" data-drop="move">
                <div data-effect="copy" draggable="true"  class="color-1 item">语文</div>
                <div data-effect="copy" draggable="true" class="color-2 item">数学</div>
                <div data-effect="copy" draggable="true" class="color-3 item">英语</div>
                <div data-effect="copy" draggable="true" class="color-4 item">音乐</div>
                <div data-effect="copy" draggable="true" class="color-5 item">政治</div>
                <div data-effect="copy" draggable="true" class="color-6 item">历史</div>
            </div>
 <div class="right">
                <table border="1">
                    <colgroup>
                        <col span="6" style="background-color:#fff">
                    </colgroup>
                    <tr>
                        <th>星期一</th>
                        <th>星期二</th>
                        <th>星期三</th>
                         <th>星期四</th>
                          <th>星期五</th>
                           <th>星期六</th>
                    </tr>
                    <tr>
                        <td data-drop="copy"></td>
                        <td data-drop="copy"></td>
                        <td data-drop="copy"></td>
                          <td></td>
                        <td data-drop="copy"></td>
                        <td data-drop="copy"></td>
                    </tr>
                         <tr>
                        <td data-drop="copy"></td>
                        <td data-drop="copy"></td>
                        <td data-drop="copy"></td>
                          <td  data-drop="copy"></td>
                        <td data-drop="copy"></td>
                        <td data-drop="copy"></td>
                    </tr>
                         <tr>
                        <td data-drop="copy"></td>
                        <td data-drop="copy"></td>
                        <td data-drop="copy"></td>
                          <td  data-drop="copy"></td>
                        <td data-drop="copy"></td>
                        <td data-drop="copy"></td>
                    </tr>
                </table>
            </div>

获取拖拽节点,设定节点本身的属性

当拖拽该元素时,该节点下面会有一个‘+’的符号,这个是浏览器默认给他加上的,我们需要自定义该属性,当他拖拽到某个区域时候,就设定该属性的值为区域的值

这里我们在拖拽元素上添加一个属性data-effect="copy",通过e.target.dataset.effect获取值,并赋值给该属性

let source;

//拖拽开始事件,只触发一次
container.ondragstart = e => {
    //移动取消出现+号的效果
    e.dataTransfer.effectAllowed = e.target.dataset.effect;
    source = e.target
    console.log('start', e.target)
}

拖拽移动,区域样式变化

当我们拖拽移动的时候,对应的区域样式变化。
首先我们要先阻止浏览器的默认行为,否则不会触发ondrop事件

container.ondragover = e => {
    e.preventDefault();
    // console.log('over', e.target)
}

同时,进入某个区域的时候,样式发生变化,但是由于它移入可以会经过其他元素,所以我们要先清除之前拖拽的样式。

//清除移动的样式
function removeDropStyle() {
    document.querySelectorAll('.drop-over').forEach((node) => {
        node.classList.remove('drop-over')
    })
}

container.ondragenter = e => {
    //清除之前拖拽的样式
    removeDropStyle()
    const dropNode = getDropNode(e.target)
    if (dropNode && dropNode.dataset.drop === e.dataTransfer.effectAllowed) {
        //该节点能够接受目前拖拽节点
        e.target.classList.add('drop-over')
    }

    // console.log('enter', e.target)
}

到这里,我们就实现了拖拽该元素,并且移动到哪个区域,哪个区域的样式就会发生改变,接下来就是松开拖拽,对应区域就拥有拖拽元素。

拖拽松开,区域变化

我们要解决的问题有:
一、判断该元素是否有drop属性,如果没有,就去父元素找。
二、清除之前的元素
三、如果是move区域,就直接移除元素即可

container.ondrop = e => {
    //清除拖拽的样式
    removeDropStyle()
    console.log('drop', e.target)
    const dropNode = getDropNode(e.target)
    //该节点能够接受目前拖拽节点
    if (dropNode && dropNode.dataset.drop === e.dataTransfer.effectAllowed) {
        //两种情况,是拖拽到哪个区域,如果是copy区域
        if (dropNode.dataset.drop === 'copy') {
            //清除之前的元素
            dropNode.innerHTML = ''
            //把该元素复制一份
            const cloned = source.cloneNode(true)
            cloned.dataset.effect = 'move'
            dropNode.appendChild(cloned)
            //如果是move区域
        } else {
            console.log("1111111")
            source.remove()
        }
    }
}
//获取该元素是否有父元素
function getDropNode(node) {
    while (node) {
        if (node.dataset.drop) {
            return node
        }
        node = node.parentNode
    }
}

完整代码

index.js

//直接监控父元素,使用事件委托
const container = document.querySelector('.container');

let source;

//拖拽开始事件,只触发一次
container.ondragstart = e => {
    //移动取消出现+号的效果
    e.dataTransfer.effectAllowed = e.target.dataset.effect;
    source = e.target
    console.log('start', e.target)
}

//拖拽结束事件,表示拖拽这个元素到哪个元素之上,不断触发
container.ondragover = e => {
    e.preventDefault();
    // console.log('over', e.target)
}


//拖拽移入事件,只触发一次
container.ondragenter = e => {
    //清除之前拖拽的样式
    removeDropStyle()
    const dropNode = getDropNode(e.target)
    if (dropNode && dropNode.dataset.drop === e.dataTransfer.effectAllowed) {
        //该节点能够接受目前拖拽节点
        e.target.classList.add('drop-over')
    }

    // console.log('enter', e.target)
}

//拖拽放手在哪个元素,table,tr,td,这些元素是不允许元素拖拽到他的上面的,因此是不会触发这个事件的,触发阻止浏览器默认事件,在ondragover中阻止
container.ondrop = e => {
    //清除拖拽的样式
    removeDropStyle()
    console.log('drop', e.target)
    const dropNode = getDropNode(e.target)
    //该节点能够接受目前拖拽节点
    if (dropNode && dropNode.dataset.drop === e.dataTransfer.effectAllowed) {
        //两种情况,是拖拽到哪个区域,如果是copy区域
        if (dropNode.dataset.drop === 'copy') {
            //清除之前的元素
            dropNode.innerHTML = ''
            //把该元素复制一份
            const cloned = source.cloneNode(true)
            cloned.dataset.effect = 'move'
            dropNode.appendChild(cloned)
            //如果是move区域
        } else {
            console.log("1111111")
            source.remove()
        }
    }
}

//获取该元素是否有父元素
function getDropNode(node) {
    while (node) {
        if (node.dataset.drop) {
            return node
        }
        node = node.parentNode
    }
}

//清除移动的样式
function removeDropStyle() {
    document.querySelectorAll('.drop-over').forEach((node) => {
        node.classList.remove('drop-over')
    })
}

index.css

body {
    margin: 0;
    padding: 0;
}

h1 {
    width: 100%;
    text-align: center;
}

.container {
    width: 100%;
    height: 800px;
    display: flex;
    flex-direction: row;

}

.left {
    width: 5%;
    background: #f3f4f5;
    display: flex;
    flex-direction: column;
    align-items: center;
}

.item {
    width: 100%;
    display: flex;
    justify-content: center;
    align-items: center;
    height: 70px;
    font-size: 20px;
    font-weight: 500;
    margin-bottom: 20px;
    color: #fff;
}

.right {
    margin-left: 40px;
    width: 95%;
    background: #f3f4f5;
}

.color-1 {
    background: red;
}

.color-2 {
    background: rgb(18, 49, 189);
}

.color-3 {
    background: rgb(22, 153, 33);
}

.color-4 {
    background: rgb(150, 136, 12);
}

.color-5 {
    background: rgb(110, 9, 114);
}

.color-6 {
    background: rgb(192, 118, 22);
}

td {
    width: 90px;
    height: 70px;
    ;
}

.drop-over {
    background: rgba(212, 13, 56, 0.067);
}

index.html

<!DOCTYPE html>
<html lang="en">
    <head>
        <meta charset="UTF-8">
        <meta http-equiv="X-UA-Compatible" content="IE=edge">
        <meta name="viewport" content="width=device-width, initial-scale=1.0">
        <title>可拖拽课程表</title>
        <link href="./index.css" rel="stylesheet"></link>
    </head>
    <body>
        <h1>课程表</h1>
        <div class="container">
            <div class="left" data-drop="move">
                <div data-effect="copy" draggable="true"  class="color-1 item">语文</div>
                <div data-effect="copy" draggable="true" class="color-2 item">数学</div>
                <div data-effect="copy" draggable="true" class="color-3 item">英语</div>
                <div data-effect="copy" draggable="true" class="color-4 item">音乐</div>
                <div data-effect="copy" draggable="true" class="color-5 item">政治</div>
                <div data-effect="copy" draggable="true" class="color-6 item">历史</div>
            </div>
            <div class="right">
                <table border="1">
                    <colgroup>
                        <col span="6" style="background-color:#fff">
                    </colgroup>
                    <tr>
                        <th>星期一</th>
                        <th>星期二</th>
                        <th>星期三</th>
                         <th>星期四</th>
                          <th>星期五</th>
                           <th>星期六</th>
                    </tr>
                    <tr>
                        <td data-drop="copy"></td>
                        <td data-drop="copy"></td>
                        <td data-drop="copy"></td>
                          <td></td>
                        <td data-drop="copy"></td>
                        <td data-drop="copy"></td>
                    </tr>
                         <tr>
                        <td data-drop="copy"></td>
                        <td data-drop="copy"></td>
                        <td data-drop="copy"></td>
                          <td  data-drop="copy"></td>
                        <td data-drop="copy"></td>
                        <td data-drop="copy"></td>
                    </tr>
                         <tr>
                        <td data-drop="copy"></td>
                        <td data-drop="copy"></td>
                        <td data-drop="copy"></td>
                          <td  data-drop="copy"></td>
                        <td data-drop="copy"></td>
                        <td data-drop="copy"></td>
                    </tr>
                </table>
            </div>
        </div>
        <script src="./index.js"></script>
    </body>
</html>

http://www.niftyadmin.cn/n/325684.html

相关文章

Linux——对权限的理解

文章目录 总述Linux权限的概念Linux 权限管理Linux对文件访问者的分类文件类型和访问权限a.文件类型b.基本权限文件权限修改的相关方法 其他问题在首次创建时文件的权限属性是固定的吗&#xff1f;目录的权限粘滞位 总述 本篇博客将主要讲解linux系统中权限的概念&#xff0c;权…

分段存储管理方式

目录 一、分段存储管理方式的引入的需求: 1.方便编程 2.信息共享 3.信息保护 4.动态增长 5.动态链接 二、分段系统的基本原理 1.分段 2.段表 3.地址变换机构 4.分页与分段的主要区别 三、信息共享 四、段页式存储管理方式 1.基本原理 2.地址变换过程 分段与分页…

哈工大软件架构与中间件作业3

《软件架构与中间件》作业3报告 ——作业3&#xff1a;面向全球10亿用户的KWIC Web应用 姓名&#xff1a; 石卓凡 学号&#xff1a; 120L021011 目录 分布式架构设计方案 - 面向全球10亿用户的KWIC Web应用... 1 摘要... 1 架构总览... 2 1. 负…

5年测试,已失业3个月.....

我做测试5年&#xff0c;一线城市薪水拿到15K&#xff0c;中间还修了一个专升本&#xff0c;这个年限不说资深肯定也是配得上经验丰富的。今年行情不好人尽皆知&#xff0c;但我还是对我的薪水不是很满意&#xff0c;于是打算出去面试&#xff0c;希望可以搏一个高薪。 但真到面…

人脸识别2:Python实现人脸识别Face Recognition(含源码)

人脸识别2&#xff1a;Python实现人脸识别Face Recognition(含源码) 目录 人脸识别2&#xff1a;Python实现人脸识别Face Recognition(含源码) 1. 前言 2. 项目安装 3. 人脸识别系统 &#xff08;1&#xff09;人脸检测和关键点检测 &#xff08;2&#xff09;人脸校准 …

【MATLAB第32期】基于MATLAB的降维/全局敏感性分析/特征排序/数据处理分类问题MATLAB代码实现(持续更新)

【MATLAB第32期】基于MATLAB的降维/全局敏感性分析/特征排序/数据处理分类问题MATLAB代码实现(持续更新) 本文敏感性分析主要分析分类问题&#xff08;fisher、rf、arf、nca等&#xff09;。 一、降维方法&#xff08;分类&#xff09; 常见的降维方法&#xff1a; 常见的敏…

第七章结构性模式—适配器模式

文章目录 适配器模式解决的问题概念结构 类适配器模式对象适配器模式接口适配器模式应用场景JDK 源码 - Reader 与 InputStream 结构型模式描述如何将类或对象按某种布局组成更大的结构&#xff0c;有以下两种&#xff1a; 类结构型模式&#xff1a;采用继承机制来组织接口和类…

计算机网络:TCP 拥塞控制

目录 慢启动拥塞避免算法拥塞发生快速恢复 流量控制是避免「发送方」的数据填满「接收方」的缓存&#xff0c;但是并不知道网络的中发生了什么。 在网络出现拥堵时&#xff0c;如果继续发送大量数据包&#xff0c;可能会导致数据包时延、丢失等&#xff0c;这时 TCP 就会重传数…