Vue3 实现课程表

  1. 效果
  2. 代码
    <template>
      <div class="class-compopnents">
        <!-- 容器用于放置所有课程分类 -->
        <div class="class-container">
          <!-- 循环生成可拖拽的课程分类 -->
          <div
            v-for="(item, index) in classTypeList"
            class="class-type"
            :key="index"
            :style="{ 'background-color': item.color }"
            :draggable="item.draggable"
            @dragstart="onDragStart(index)"
            @dragover.prevent="() => onDragOver(index)"
            @dragend="onDragEnd"
            :class="{ dragging: draggingIndex === index }"
          >
            {{ item.name }}
          </div>
        </div>
        <div class="course-schedule-container">
          <table>
            <tbody>
              <tr class="header-tr">
                <td colspan="2" :canInsert="false"></td>
                <td
                  :canInsert="false"
                  v-for="(item, index) in numberOfDays"
                  :key="index"
                  class="notSelect"
                >
                  {{ item }}
                </td>
              </tr>
              <tr v-for="(item, index) in classScheduleCardList" :key="index">
                <td
                  rowspan="4"
                  v-if="item.type === 'morning' || item.type === 'afternoon'"
                  :canInsert="false"
                  class="notSelect"
                >
                  {{ item.title }}
                </td>
                <td :canInsert="false">{{ item.section }}</td>
                <td
                  v-for="(v, i) in item.courseList"
                  :key="i"
                  :parentIndex="index"
                  @dragstart="(e) => tdDragStart(index, i, e)"
                  @dragover.prevent="(e) => tdDragOver(index, i, e)"
                  @dragleave="(e) => tdDragleave(index, i, e)"
                  @dragend="(e) => tdDragend(index, i, e)"
                  :draggable="v && v.draggable"
                  :style="{
                    'background-color': v ? v.color : '#fff',
                    cursor: 'grab',
                  }"
                >
                  <div class="course-name">
                    <span class="title notSelect">
                      {{ v ? v.name : "" }}
                    </span>
                    <div
                      class="remove"
                      v-if="v"
                      @click.stop="() => remove(index, i)"
                    >
                      +
                    </div>
                  </div>
                </td>
              </tr>
            </tbody>
          </table>
        </div>
      </div>
    </template>
    
    <script lang="ts" setup>
    import { ref } from "vue";
    
    // 定义课程分类数据
    const classTypeList = ref([
      { name: "语文", color: "#8dedfb", draggable: true },
      { name: "数学", color: "#f1957b", draggable: true },
      { name: "英语", color: "#f6f1a9", draggable: true },
      { name: "物理", color: "#8cdb27", draggable: true },
      { name: "化学", color: "#d3dc28", draggable: true },
      { name: "生物", color: "#e5b46f", draggable: true },
      { name: "地理", color: "#656ef9", draggable: true },
      { name: "历史", color: "#f6efa1", draggable: true },
    ]);
    
    const numberOfDays = ref([
      "星期一",
      "星期二",
      "星期三",
      "星期四",
      "星期五",
      "星期六",
      "星期天",
    ]);
    const classScheduleCardList = ref<any>([
      {
        type: "morning",
        title: "上午",
        rowspan: 4,
        section: "第一节",
        courseList: [null, null, null, null, null, null, null],
      },
      {
        section: "第二节",
        courseList: [null, null, null, null, null, null, null],
      },
      {
        section: "第三节",
        courseList: [null, null, null, null, null, null, null],
      },
      {
        section: "第四节",
        courseList: [null, null, null, null, null, null, null],
      },
      {
        type: "afternoon",
        title: "下午",
        rowspan: 4,
        section: "第五节",
        courseList: [null, null, null, null, null, null, null],
      },
      {
        section: "第六节",
        courseList: [null, null, null, null, null, null, null],
      },
      {
        section: "第七节",
        courseList: [null, null, null, null, null, null, null],
      },
      {
        section: "第八节",
        courseList: [null, null, null, null, null, null, null],
      },
    ]);
    
    // 当前正在拖拽的元素索引
    const draggingIndex = ref<number | null>(null);
    
    // 拖拽开始:记录当前拖拽的索引
    function onDragStart(index: number) {
      draggingIndex.value = index;
    }
    
    // 拖拽经过:动态调整拖拽顺序
    function onDragOver(targetIndex: number) {
      if (draggingIndex.value !== null && draggingIndex.value !== targetIndex) {
        // 拖拽元素和目标位置互换
        const draggedItem = classTypeList.value[draggingIndex.value];
        classTypeList.value.splice(draggingIndex.value, 1); // 移除拖拽的元素
        classTypeList.value.splice(targetIndex, 0, draggedItem); // 插入到目标位置
        draggingIndex.value = targetIndex; // 更新拖拽元素的当前索引
      }
    }
    
    // 拖拽结束:清空拖拽状态
    function onDragEnd() {
      draggingIndex.value = null;
      currentTDinfo.value = null;
      processCourseData();
    }
    
    // 处理课程数据
    function processCourseData() {
      // 拖拽结束后, 将课程表进行处理,添加上 自定义删除的标识,避免其他元素移出后,将其置空
      const list = JSON.parse(JSON.stringify(classScheduleCardList.value));
      list.forEach((item: any) => {
        item.courseList.forEach((v: any) => {
          if (v) v.customDelete = true;
        });
      });
      classScheduleCardList.value = list;
    }
    
    // 拖拽中的 td 元素
    const dragingTdInfo = ref<any>(null);
    // 移动到的 td 信息
    const currentTDinfo = ref<any>(null);
    
    // td 中开始拖拽拖拽
    function tdDragStart(parentIndex: number, index: number, e: any) {
      if (!classScheduleCardList.value[parentIndex].courseList[index]) return;
      const item = JSON.parse(
        JSON.stringify(classScheduleCardList.value[parentIndex].courseList[index])
      );
      item.customDelete = false;
      dragingTdInfo.value = item;
      classScheduleCardList.value[parentIndex].courseList[index] = null;
    }
    
    // 移动到 td 上触发
    function tdDragOver(parentIndex: number, index: number, e: any) {
      currentTDinfo.value = { parentIndex, index };
      const currentTd = classScheduleCardList.value[parentIndex].courseList[index];
      if (draggingIndex.value !== null && !currentTd) {
        const draggedItem = classTypeList.value[draggingIndex.value];
        // 将对应的课程数据 暂时 放入到td中
        classScheduleCardList.value[parentIndex].courseList[index] = draggedItem;
      }
    
      // td 中进行拖拽
      if (dragingTdInfo.value && !currentTd) {
        classScheduleCardList.value[parentIndex].courseList[index] = JSON.parse(
          JSON.stringify(dragingTdInfo.value)
        );
      }
    }
    
    // 课程在 td 上离开
    function tdDragleave(parentIndex: number, index: number, event: any) {
      // 检查 relatedTarget 是否在目标元素内
      const relatedTarget = event.relatedTarget as HTMLElement;
      const isInside = relatedTarget?.closest("td") === event.currentTarget;
      if (!isInside) {
        // 如果存在课程 return ,只有用户删除之后才可以添加新的课程
        // 存在 customDelete 删除属性,只能用户主动删除才可以清空,其余情况 拖拽元素移出就 清空
        const currentTd =
          classScheduleCardList.value[parentIndex].courseList[index];
        // 从课程类型中拖拽到 td 中
        if (draggingIndex.value !== null && currentTd && !currentTd.customDelete) {
          classScheduleCardList.value[parentIndex].courseList[index] = null;
        }
    
        // td 中进行拖拽
        if (
          dragingTdInfo.value &&
          !dragingTdInfo.value.customDelete &&
          currentTd &&
          !currentTd.customDelete
        ) {
          classScheduleCardList.value[parentIndex].courseList[index] = null;
        }
      }
    }
    
    //  td中拖拽课程结束
    function tdDragend(parentIndex: number, index: number, e: any) {
      const { parentIndex: currentParentIndex, index: currentIndex } =
        currentTDinfo.value;
    
      const currentTd =
        classScheduleCardList.value[currentParentIndex].courseList[currentIndex];
      if (!currentTd) return;
    
      const isSameTd = currentParentIndex === parentIndex && currentIndex === index;
      const tdHasCourses =
        classScheduleCardList.value[currentParentIndex].courseList[currentIndex]
          .customDelete;
      if (isSameTd || tdHasCourses) {
        classScheduleCardList.value[parentIndex].courseList[index] = JSON.parse(
          JSON.stringify(dragingTdInfo.value)
        );
      }
    
      dragingTdInfo.value = null;
      currentTDinfo.value = null;
      processCourseData();
    }
    
    // 删除
    function remove(parentIndex: number, index: number) {
      classScheduleCardList.value[parentIndex].courseList[index] = null;
    }
    </script>
    
    <style lang="scss" scoped>
    .class-compopnents {
      display: flex;
      justify-content: flex-start;
    }
    .class-container {
      display: flex;
      flex-direction: column;
      width: 120px;
      margin-right: 20px;
    }
    
    .class-type {
      width: 100px;
      padding: 20px 10px;
      border-radius: 10px;
      margin-bottom: 10px;
      text-align: center;
      cursor: grab;
      user-select: none; /* 防止拖拽时选中文字 */
      transition: transform 0.2s, background-color 0.2s;
    
      &:hover {
        transform: scale(1.05); /* 鼠标悬浮时的缩放效果 */
      }
    }
    
    /* 正在拖拽的元素样式 */
    .dragging {
      opacity: 0.5;
      transform: scale(1.1); /* 拖拽中的缩放效果 */
      //   background-color: #d3d3d3;
    }
    
    .class {
      width: 100%;
      height: 100%;
    }
    
    .course-schedule-container {
      width: 100%;
      height: 100%;
    }
    
    table {
      width: 100%;
      border-radius: 10px;
    }
    
    .header-tr {
      background-color: #f8f8f8;
      color: #aaa;
    }
    
    .notSelect {
      user-select: none;
    }
    .course-name {
      width: 100%;
      height: 100%;
      position: relative;
      .title {
      }
      .remove {
        width: 20px;
        height: 20px;
        text-align: center;
        line-height: 20px;
        position: absolute;
        top: -20px;
        right: -20px;
        font-size: 20px;
        transform: rotate(45deg);
        border-radius: 50%;
        border: 1px solid #000;
        cursor: pointer;
      }
    }
    table,
    th,
    td {
      border: 1px solid #ccc;
      border-collapse: collapse; /* 消除双边框效果 */
      transition: all 0.3s;
    }
    
    th,
    td {
      padding: 20px;
    }
    </style>
    
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

清云随笔

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值