IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> JavaScript知识库 -> javascript css实现3d轮播图 -> 正文阅读

[JavaScript知识库]javascript css实现3d轮播图

?Vue 代码

<template>
  <div class="mobile-unit">
    <div class="mobile-unit-container">
      <div class="items">
        <div
          class="mobile-unit-item five-in-one"
          :class="[{ 'is-active': currentComponent === 'FiveInOne' }]"
          @click="toggleComponent('FiveInOne')"
          v-show="equipmentData.data.indexOf('五合一') !== -1"
          ref="FiveInOne"
        >
          <i></i>
          <span>五合一</span>
        </div>
        <div
          class="mobile-unit-item ozone-radar"
          :class="[{ 'is-active': currentComponent === 'OzoneRadar' }]"
          @click="toggleComponent('OzoneRadar')"
          v-show="equipmentData.data.indexOf('雷达') !== -1"
          ref="OzoneRadar"
        >
          <i></i>
          <span>雷达</span>
        </div>
        <div
          class="mobile-unit-item mass-spectrometer"
          :class="[{ 'is-active': currentComponent === 'MassSpectrometer' }]"
          @click="toggleComponent('MassSpectrometer')"
          v-show="equipmentData.data.indexOf('质谱仪') !== -1"
          ref="MassSpectrometer"
        >
          <i></i>
          <span>质谱仪</span>
        </div>
        <div
          class="mobile-unit-item ship"
          :class="[{ 'is-active': currentComponent === 'UnmannedShip' }]"
          @click="toggleComponent('UnmannedShip')"
          v-show="equipmentData.data.indexOf('无人船') !== -1"
          ref="UnmannedShip"
        >
          <i></i>
          <span>无人船</span>
        </div>
        <div
          class="mobile-unit-item portable-equipment"
          :class="[{ 'is-active': currentComponent === 'PortableEquipment' }]"
          @click="toggleComponent('PortableEquipment')"
          v-show="equipmentData.data.indexOf('便携设备') !== -1"
          ref="PortableEquipment"
        >
          <i></i>
          <span>便携设备</span>
        </div>
        <div
          class="mobile-unit-item uav"
          :class="[{ 'is-active': currentComponent === 'Uav' }]"
          @click="toggleComponent('Uav')"
          v-show="equipmentData.data.indexOf('无人机') !== -1"
          ref="Uav"
        >
          <i></i>
          <span>无人机</span>
        </div>
      </div>
    </div>
  </div>
</template>

<script>
import {
  ref,
  reactive,
  watch,
  getCurrentInstance,
  onMounted,
  onUnmounted,
  onDeactivated,
} from "vue";

export default {
  setup() {
    let currentComponent = ref(null);
    let FiveInOne = ref(null);
    let OzoneRadar = ref(null);
    let MassSpectrometer = ref(null);
    let UnmannedShip = ref(null);
    let PortableEquipment = ref(null);
    let Uav = ref(null);
    let timers = [];
    const toggleComponent = (value) => {
      currentComponent.value = currentComponent.value !== value ? value : null;
      if (equipmentData.data.length <= 1) return;
      if (currentComponent.value) {
        timers.forEach((item) => {
          clearInterval(item);
        });
      } else {
        starRotate(true);
      }
    };

    let equipmentData = reactive({
      data: ["便携设备", "无人机", "无人船", "质谱仪", "雷达", "五合一"],
    });

    onDeactivated(() => {
      timers.forEach((item) => {
        clearInterval(item);
      });
    });
    onUnmounted(() => {
      timers.forEach((item) => {
        clearInterval(item);
      });
    });
    onMounted(() => {
      console.log("onmounted了");

      watch(
        () => equipmentData.data,
        (value) => {
          starRotate(false);
        },
        { deep: true, immediate: true }
      );
    });

    function starRotate(isPause) {
      if (equipmentData.data.length <= 1) return;
      let part = 360 / equipmentData.data.length;
      let PortableEquipmentStyle = PortableEquipment.value.style.transform;
      let UavStyle = Uav.value.style.transform;
      let UnmannedShipStyle = UnmannedShip.value.style.transform;
      let MassSpectrometerStyle = MassSpectrometer.value.style.transform;
      let OzoneRadarStyle = OzoneRadar.value.style.transform;
      let FiveInOneStyle = FiveInOne.value.style.transform;

      equipmentData.data.forEach((item, index) => {
        switch (item) {
          case "便携设备":
            //获取角度
            let PortableEquipmentAngle = isPause
              ? PortableEquipmentStyle.split("deg")[0].split("(")[1]
              : part * index;
            //判断是否是暂停
            PortableEquipmentStyle = isPause
              ? PortableEquipmentStyle
              : `rotateY(${PortableEquipmentAngle}deg) translateZ(210px)`;
            timers.push(
              rotate(PortableEquipment, PortableEquipmentAngle, part)
            );
            break;

          case "无人机":
            //获取角度
            let UAVAngle = isPause
              ? UavStyle.split("deg")[0].split("(")[1]
              : part * index;
            //判断是否是暂停
            UavStyle = isPause
              ? UavStyle
              : `rotateY(${UAVAngle}deg) translateZ(210px)`;
            timers.push(rotate(Uav, UAVAngle, part));
            break;
          case "无人船":
            //获取角度
            let UnmannedShipAngle = isPause
              ? UnmannedShipStyle.split("deg")[0].split("(")[1]
              : part * index;
            //判断是否是暂停
            UnmannedShipStyle = isPause
              ? UnmannedShipStyle
              : `rotateY(${UnmannedShipAngle}deg) translateZ(210px)`;
            timers.push(rotate(UnmannedShip, UnmannedShipAngle, part));
            break;
          case "质谱仪":
            //获取角度
            let MassSpectrometerAngle = isPause
              ? MassSpectrometerStyle.split("deg")[0].split("(")[1]
              : part * index;
            //判断是否是暂停
            MassSpectrometerStyle = isPause
              ? MassSpectrometerStyle
              : `rotateY(${MassSpectrometerAngle}deg) translateZ(210px)`;
            timers.push(rotate(MassSpectrometer, MassSpectrometerAngle, part));
            break;
          case "雷达":
            //获取角度
            let OzoneRadarAngle = isPause
              ? OzoneRadarStyle.split("deg")[0].split("(")[1]
              : part * index;
            //判断是否是暂停
            OzoneRadarStyle = isPause
              ? OzoneRadarStyle
              : `rotateY(${OzoneRadarAngle}deg) translateZ(210px)`;
            timers.push(rotate(OzoneRadar, OzoneRadarAngle, part));
            break;
          case "五合一":
            //获取角度
            let FiveInOneAngle = isPause
              ? FiveInOneStyle.split("deg")[0].split("(")[1]
              : part * index;
            //判断是否是暂停
            FiveInOneStyle = isPause
              ? FiveInOneStyle
              : `rotateY(${FiveInOneAngle}deg) translateZ(210px)`;
            timers.push(rotate(FiveInOne, FiveInOneAngle, part));
            break;
          default:
            break;
        }
      });
    }

    function rotate(dom, angle, part) {
      if (!dom.value) return;
      return setInterval(() => {
        //减去每个步长即每块div所占的角度
        angle -= part;
        dom.value.style.transform = `rotateY(${angle}deg) translateZ(210px)`;
      }, 2000);
    }

    return {
      currentComponent,
      toggleComponent,
      equipmentData,
      FiveInOne,
      OzoneRadar,
      MassSpectrometer,
      UnmannedShip,
      PortableEquipment,
      Uav,
    };
  },
};
</script>

<style lang="scss" scoped>
.mobile-unit {
  margin-top: 20px;
  width: 100%;
  height: 100%;
  background: rgba(12, 12, 57, 0.2);

  .mobile-unit-container {
    position: relative;
    width: 420px;
    height: 350px;
    margin: 3px auto;
    perspective: 10000px;
    background: url(~@/assets/img/dark/round.png) no-repeat center center;
    // transform-style: preserve-3d;
    .items {
      width: 112px;
      height: 124px;
      transition: transform 1s;
      transform-style: preserve-3d;
      position: absolute;
      left: 37%;
      top: 30%;
      transform: rotateX(340deg);
    }

    .navigation-vehicle {
      position: relative;

      margin: 0 auto;
      width: 276px;
      height: 248px;

      background: url(~@/assets/img/dark/round.png) no-repeat;
      background-size: contain;
      background-position-y: 70px;

      cursor: pointer;

      > canvas {
        position: absolute;
        top: 0;
        left: calc(50% - 150px);

        width: 300px;
        height: 200px;

        outline: none;
      }
    }
    .mobile-unit-item {
      position: absolute;
      bottom: 0;
      display: flex;
      flex-direction: column;
      align-items: center;
      justify-content: space-between;
      width: 112px;
      height: 124px;
      padding: 10px 0px 25px 0;
      background: url(~@/assets/img/dark/equipment.png) no-repeat;
      transition: all 1s;
      i {
        width: 45px;
        height: 45px;
        background-color: rgba(0, 244, 255, 0.1);
        background-position: center;
        background-repeat: no-repeat;
        // border: 1px dashed #00f4ff;
        border-radius: 50%;
        cursor: pointer;
      }
      span {
        font-size: 14px;

        color: #fff;
        cursor: pointer;
      }
    }
    .mobile-unit-item.is-active {
      i {
        border-color: #ff8104;
      }

      span {
        color: #ff8104;
      }
    }
    .is-active {
      background-image: url(~@/assets/img/dark/equipment-active.png);
    }
    .five-in-one {
      //transform: rotateY(0) translateZ(210px);
      i {
        background-image: url(~@/assets/img/home/mobileUnit/icon-five-in-one.png);
      }
    }
    .five-in-one.is-active {
      i {
        background-image: url(~@/assets/img/home/mobileUnit/icon-five-in-one-active.png);
      }
    }

    .ozone-radar {
      //transform: rotateY(60deg) translateZ(210px);
      i {
        background-image: url(~@/assets/img/home/mobileUnit/icon-ozone-radar.png);
      }
    }
    .ozone-radar.is-active {
      i {
        background-image: url(~@/assets/img/home/mobileUnit/icon-ozone-radar-active.png);
      }
    }

    .mass-spectrometer {
      //transform: rotateY(120deg) translateZ(210px);
      i {
        background-image: url(~@/assets/img/home/mobileUnit/icon-mass-spectrometer.png);
      }
    }
    .mass-spectrometer.is-active {
      i {
        background-image: url(~@/assets/img/home/mobileUnit/icon-mass-spectrometer-active.png);
      }
    }

    .ship {
      //transform: rotateY(180deg) translateZ(210px);
      i {
        background-image: url(~@/assets/img/home/mobileUnit/icon-ship.png);
      }
    }
    .ship.is-active {
      i {
        background-image: url(~@/assets/img/home/mobileUnit/icon-ship-active.png);
      }
    }

    .portable-equipment {
      //transform: rotateY(240deg) translateZ(210px);
      i {
        background-image: url(~@/assets/img/home/mobileUnit/icon-portable-equipment.png);
      }
    }
    .portable-equipment.is-active {
      i {
        background-image: url(~@/assets/img/home/mobileUnit/icon-portable-equipment-active.png);
      }
    }

    .uav {
      //transform: rotateY(300deg) translateZ(210px);
      i {
        background-image: url(~@/assets/img/home/mobileUnit/icon-uav.png);
      }
    }
    .uav.is-active {
      i {
        background-image: url(~@/assets/img/home/mobileUnit/icon-uav-active.png);
      }
    }
  }
}
@keyframes rotate {
}
</style>

  JavaScript知识库 最新文章
ES6的相关知识点
react 函数式组件 & react其他一些总结
Vue基础超详细
前端JS也可以连点成线(Vue中运用 AntVG6)
Vue事件处理的基本使用
Vue后台项目的记录 (一)
前后端分离vue跨域,devServer配置proxy代理
TypeScript
初识vuex
vue项目安装包指令收集
上一篇文章      下一篇文章      查看所有文章
加:2022-04-01 23:16:14  更:2022-04-01 23:18:33 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2025年1日历 -2025/1/10 20:56:30-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码