使用CSS3与JavaScript实现炫酷的3D旋转魔方及九宫格交换动效

文章目录

  • 前言
  • 一、项目需求背景
  • 二、CSS3 3D基础知识介绍
    • 2.1 什么是CSS3 3D?
    • 2.2 主要使用的CSS属性
  • 三、使用HTML和CSS搭建魔方结构
  • 四、让魔方动起来:CSS3动画
  • 五、九宫格数字交换的JavaScript实现
    • 5.1 九宫格布局
    • 5.2 随机交换数字
  • 六、随机交换与相邻格子的逻辑
  • 七、项目总结与完整代码
  • 总结


前言

在前端开发中,视觉效果的提升一直是提升用户体验的重要手段。通过CSS3和JavaScript的组合,我们可以实现很多令人惊叹的动画效果。在本文中,我们将通过一个实际的案例,展示如何实现一个旋转魔方的3D效果,并且在魔方的九宫格上随机交换数字。这个动效看似复杂,但通过合理的CSS与JavaScript编排,你会发现它不仅有趣且非常具有实用性。


一、项目需求背景

在日常的网页设计中,3D效果和动效已经逐渐成为了增强用户体验的重要元素。这次实现的效果,我希望通过一个3D魔方展示如何将CSS3的3D变换与JavaScript的动态交互相结合。魔方不仅会进行旋转,魔方面上的数字也会以随机相邻交换的方式进行变化。

二、CSS3 3D基础知识介绍

2.1 什么是CSS3 3D?

CSS3引入了transform属性,允许开发者对元素进行三维操作。我们可以通过rotateX、rotateY等方法在三维空间中旋转元素。本文中的魔方正是基于这些技术实现的,它不仅在X、Y、Z轴上进行旋转,同时使用perspective属性给予了它空间感。

2.2 主要使用的CSS属性

•	transform: 进行三维旋转、平移的核心属性。
•	transform-style: 确保元素的子元素在3D空间中呈现。
•	@keyframes: 定义动画效果的关键帧,确保魔方能流畅旋转。

三、使用HTML和CSS搭建魔方结构

在实现复杂动效前,我们首先需要搭建魔方的基础结构。魔方由6个面组成,每个面都是一个九宫格布局。具体的HTML结构如下:

<div class="magic-cube">
    <div class="face front"></div>
    <div class="face back"></div>
    <div class="face left"></div>
    <div class="face right"></div>
    <div class="face top"></div>
    <div class="face bottom"></div>
</div>

每个面对应一个CSS类,通过position和transform属性定义它们在3D空间中的位置。

四、让魔方动起来:CSS3动画

使用CSS的@keyframes,我们可以让魔方进行自动旋转。通过多个关键帧的设定,魔方可以围绕X、Y、Z三个轴进行不同方向的旋转。

@keyframes complexRotate {
    0%   { transform: rotateX(-30deg) rotateY(-45deg) rotateZ(0deg); }
    100% { transform: rotateX(120deg) rotateY(180deg) rotateZ(-80deg); }
}

五、九宫格数字交换的JavaScript实现

5.1 九宫格布局

我们通过HTML动态生成九宫格的每个格子,并为每个格子指定初始的数字和位置。

positions.forEach((pos, i) => {
    const cell = document.createElement('div');
    cell.textContent = i + 1;
    cell.style.top = pos.top;
    cell.style.left = pos.left;
    face.appendChild(cell);
});

5.2 随机交换数字

为了让数字的变化更具动感,我们通过JavaScript随机交换相邻的格子位置。交换的动画由CSS的transition属性负责,实现平滑过渡。

六、随机交换与相邻格子的逻辑

要确保数字交换的规则是合理的,我们引入了一个函数来判断两个格子是否相邻。通过计算九宫格的行列位置,我们限制了交换只能发生在相邻的格子之间。

function areAdjacent(index1, index2) {
    const row1 = Math.floor(index1 / 3), col1 = index1 % 3;
    const row2 = Math.floor(index2 / 3), col2 = index2 % 3;
    return (row1 === row2 && Math.abs(col1 - col2) === 1) || (col1 === col2 && Math.abs(row1 - row2) === 1);
}

七、项目总结与完整代码

通过本文,我们实现了一个旋转魔方的3D效果,并在每个面上实现了随机相邻格子的数字交换动效。CSS3的3D技术为魔方带来了空间感,JavaScript则为九宫格的动态变化提供了强大的支持。本文的核心在于CSS3动画与JavaScript交互的结合,通过这样的小项目,你可以掌握如何实现更加复杂且富有视觉冲击力的网页效果。

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <meta name="viewport" content="width=device-width, initial-scale=1.0">
    <title>3D Magic Cube with More Tile Movements</title>
    <style>
        body {
            display: flex;
            justify-content: center;
            align-items: center;
            height: 100vh;
            background-color: #2e3440;
            margin: 0;
        }

        .magic-cube {
            position: relative;
            width: 180px;
            height: 180px;
            transform-style: preserve-3d;
            transform: rotateX(-30deg) rotateY(-45deg);
            animation: complexRotate 8s infinite linear;
        }

        .face {
            position: absolute;
            width: 180px;
            height: 180px;
            display: flex;
            flex-wrap: wrap;
            perspective: 1000px;
        }

        .face div {
            width: calc(100% / 3);
            height: calc(100% / 3);
            display: flex;
            justify-content: center;
            align-items: center;
            font-size: 20px;
            background: rgba(255, 255, 255, 0.9);
            border: 1px solid #ccc;
            box-sizing: border-box;
            position: absolute;
            transition: all 0.5s ease;
        }

        .front  { transform: rotateY(0deg) translateZ(90px); background-color: #ff4d4d; }
        .back   { transform: rotateY(180deg) translateZ(90px); background-color: #4d94ff; }
        .left   { transform: rotateY(-90deg) translateZ(90px); background-color: #4dff4d; }
        .right  { transform: rotateY(90deg) translateZ(90px); background-color: #ffff4d; }
        .top    { transform: rotateX(90deg) translateZ(90px); background-color: #ff944d; }
        .bottom { transform: rotateX(-90deg) translateZ(90px); background-color: #944dff; }

        @keyframes complexRotate {
            0%   { transform: rotateX(-30deg) rotateY(-45deg) rotateZ(0deg); }
            25%  { transform: rotateX(30deg) rotateY(45deg) rotateZ(20deg); }
            50%  { transform: rotateX(60deg) rotateY(90deg) rotateZ(-40deg); }
            75%  { transform: rotateX(90deg) rotateY(135deg) rotateZ(60deg); }
            100% { transform: rotateX(120deg) rotateY(180deg) rotateZ(-80deg); }
        }
    </style>
</head>
<body>
    <div class="magic-cube">
        <div class="face front"></div>
        <div class="face back"></div>
        <div class="face left"></div>
        <div class="face right"></div>
        <div class="face top"></div>
        <div class="face bottom"></div>
    </div>

    <script>
        const positions = [
            {top: '0%', left: '0%'}, {top: '0%', left: '33.33%'}, {top: '0%', left: '66.66%'},
            {top: '33.33%', left: '0%'}, {top: '33.33%', left: '33.33%'}, {top: '33.33%', left: '66.66%'},
            {top: '66.66%', left: '0%'}, {top: '66.66%', left: '33.33%'}, {top: '66.66%', left: '66.66%'}
        ];

        // 判断两个索引是否相邻
        function areAdjacent(index1, index2) {
            const row1 = Math.floor(index1 / 3), col1 = index1 % 3;
            const row2 = Math.floor(index2 / 3), col2 = index2 % 3;

            // 只允许水平或者垂直方向相邻,不能跨行或者跨列
            return (row1 === row2 && Math.abs(col1 - col2) === 1) || (col1 === col2 && Math.abs(row1 - row2) === 1);
        }

        document.addEventListener("DOMContentLoaded", () => {
            const faces = document.querySelectorAll('.face');

            // 初始化每个面九宫格
            faces.forEach(face => {
                positions.forEach((pos, i) => {
                    const cell = document.createElement('div');
                    cell.textContent = i + 1;
                    cell.style.top = pos.top;
                    cell.style.left = pos.left;
                    face.appendChild(cell);
                });
            });

            // 随机交换多个相邻格子
            setInterval(() => {
                faces.forEach(face => {
                    const cells = face.querySelectorAll('div');

                    // 让多个相邻格子随机交换,增加动效复杂性
                    for (let i = 0; i < 3; i++) { // 每次交换三组格子
                        let index1 = Math.floor(Math.random() * 9);
                        let index2 = Math.floor(Math.random() * 9);

                        // 确保两个格子是水平或垂直相邻的
                        while (!areAdjacent(index1, index2)) {
                            index2 = Math.floor(Math.random() * 9);
                        }

                        // 获取两个相邻的格子
                        const cell1 = cells[index1];
                        const cell2 = cells[index2];

                        // 交换两个格子的 top 和 left 值
                        const tempTop = cell1.style.top;
                        const tempLeft = cell1.style.left;

                        cell1.style.top = cell2.style.top;
                        cell1.style.left = cell2.style.left;

                        cell2.style.top = tempTop;
                        cell2.style.left = tempLeft;
                    }
                });
            }, 1200); // 每隔1.2秒随机交换相邻格子
        });
    </script>
</body>
</html>

总结

这篇博文展示了CSS3和JavaScript结合使用的强大之处,通过这个项目,我们不仅掌握了CSS3 3D动画的基本操作,还学会了如何利用JavaScript实现页面中动态内容的变化。希望这篇文章能为你今后的项目带来一些启发,无论是复杂的动画效果还是简单的交互,CSS和JavaScript的结合总能带来惊艳的视觉效果。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mfbz.cn/a/888378.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

Unity各个操作功能+基本游戏物体创建与编辑+Unity场景概念及文件导入导出

各个操作功能 部分功能 几种操作游戏物体的方式&#xff1a; Center:有游戏物体父子关系的时候&#xff0c;中心点位置 Global/Local:世界坐标系方向/自身坐标系方向 &#xff1a;调试/暂停/下一帧 快捷键 1.Alt鼠标左键&#xff1a;可以实现巡游角度查看场景 2.鼠标滚轮…

mysql join的使用

MySQL 支持以下 JOIN 语法用于 SELECT 语句和多表 DELETE 和 UPDATE 语句中的 table_references 部分&#xff1a; table_references: 查询中涉及的一个或多个表的引用&#xff0c;可以是简单表名或 JOIN 表达式的组合。 escaped_table_reference [, escaped_table_referenc…

10.7学习

1.安全认证 ●Session 认证中最常用的一种方式&#xff0c;也是最简单的。存在多节点session丢失的情况&#xff0c;可通过nginx粘性Cookie和Redis集中式Session存储解决 ●HTTP Basic Authentication 服务端针对请求头中base64加密的Authorization 和用户名和密码进行校验。…

《贪吃蛇小游戏 1.0》源码

好久不见&#xff01; 终于搞好了简易版贪吃蛇小游戏&#xff08;C语言版&#xff09;&#xff0c;邀请你来玩一下~ 目录 Snake.h Snake.c test.c Snake.h #include<stdio.h> #include<windows.h> #include<stdbool.h> #include<stdlib.h> #inclu…

Ascend C 自定义算子开发:高效的算子实现

Ascend C 自定义算子开发&#xff1a;高效的算子实现 在 Ascend C 平台上&#xff0c;开发自定义算子能够充分发挥硬件的性能优势&#xff0c;帮助开发者针对不同的应用场景进行优化。本文将以 AddCustom 算子为例&#xff0c;介绍 Ascend C 中自定义算子的开发流程及关键技术…

FireRedTTS - 小红书最新开源AI语音克隆合成系统 免训练一键音频克隆 本地一键整合包下载

小红书技术团队FireRed最近推出了一款名为FireRedTTS的先进语音合成系统&#xff0c;该系统能够基于少量参考音频快速模仿任意音色和说话风格&#xff0c;实现独特的音频内容创造。 FireRedTTS 只需要给定文本和几秒钟参考音频&#xff0c;无需训练&#xff0c;就可模仿任意音色…

[记录]-安装pycharm

官网下载安装包&#xff1a;https://www.jetbrains.com/pycharm/ 然后按照引导安装 全部勾选

【数据管理】DAMA-元数据专题

导读&#xff1a;元数据是关于数据的组织、数据域及其关系的信息&#xff0c;是描述数据的数据。在数据治理中&#xff0c;元数据扮演着至关重要的角色&#xff0c;是数据治理的基础和支撑。以下是对数据治理中元数据专题方案的详细介绍&#xff1a; 目录 一、元数据的重要性 …

VRRP协议个人理解+报文示例+典型配置-RFC2338/RFC3768/RFC5798/RFC9568

个人认为&#xff0c;理解报文就理解了协议。通过报文中的字段可以理解协议在交互过程中相关传递的信息&#xff0c;更加便于理解协议。 因此本文将在VRRP协议报文的基础上进行介绍。 VRRP协议发展 关于VRRPv2基本原理&#xff0c;可重点参考2004年发布的RFC3768-Virtual Ro…

【Python|接口自动化测试】使用requests发送http请求时添加headers

文章目录 1.前言2.HTTP请求头的作用3.在不添加headers时4.反爬虫是什么&#xff1f;5.在请求时添加headers 1.前言 本篇文章主要讲解如何使用requests请求时添加headers&#xff0c;为什么要加headers呢&#xff1f;是因为有些接口不添加headers时&#xff0c;请求会失败。 2…

【C++ Primer Plus】4

2 字符串 字符串是存储在内存的连续字节中的一系列字符&#xff1b;C处理字符串的方式有两种&#xff0c; c-风格字符串&#xff08;C-Style string&#xff09;string 类 2.1 c-风格字符串&#xff08;C-Style string&#xff09; 2.1.1 char数组存储字符串&#xff08;c-…

Python编码规范与常见问题纠正

Python编码规范与常见问题纠正 Python 是一种以简洁和易读性著称的编程语言&#xff0c;因此&#xff0c;遵循良好的编码规范不仅能使代码易于维护&#xff0c;还能提升代码的可读性和可扩展性。编写规范的 Python 代码也是开发者职业素养的一部分&#xff0c;本文将从 Python…

Linux聊天集群开发之环境准备

一.windows下远程操作Linux 第一步&#xff1a;在Linux终端下配置openssh&#xff0c;输入netstate -tanp,查看ssh服务是否启动&#xff0c;默认端口22.。 注&#xff1a;如果openssh服务&#xff0c;则需下载。输入命令ps -e|grep ssh, 查看如否配有&#xff0c; ssh-agent …

tensorflow快速入门--如何定义张量、定义网络结构、超参数设置、模型训练???

前言 由于最近学习的东西涉及到tensorflow的使用&#xff0c;故先简单的学习了一下tensorflow中如何定义张量、定义网络结构、超参数设置、模型训练的API调用过程&#xff1b;欢迎大家&#xff0c;收藏关注&#xff0c;本人将持续更新。 文章目录 1、基本操作1、张量基础操作创…

[Python] 编程入门:理解变量类型

文章目录 [toc] 整数常见操作 浮点数字符串字符串中混用引号问题字符串长度计算字符串拼接 布尔类型动态类型特性类型转换结语 收录专栏&#xff1a;[Python] 在编程中&#xff0c;变量是用于存储数据的容器&#xff0c;而不同的变量类型则用来存储不同种类的数据。Python 与 C…

爬虫——爬虫理论+request模块

一、爬虫理论 爬虫——请求网站并提取数据的自动化程序 网络爬虫&#xff08;又被称为网页蜘蛛&#xff0c;网络机器人&#xff09;就是模拟客户端发送网络请求&#xff0c;接收请求响应&#xff0c;一种按照一定的规则&#xff0c;自动的抓取互联网信息的程序。 原则上&…

SQL第12课——联结表

三点&#xff1a;什么是联结&#xff1f;为什么使用联结&#xff1f;如何编写使用联结的select语句 12.1 联结 SQL最强大的功能之一就是能在数据查询的执行中联结&#xff08;join)表。联结是利用SQL的select能执行的最重要的操作。 在使用联结前&#xff0c;需要了解关系表…

【数据结构 | PTA】栈

文章目录 7-1 汉诺塔的非递归实现7-2 出栈序列的合法性**7-3 简单计算器**7-4 盲盒包装流水线 7-1 汉诺塔的非递归实现 借助堆栈以非递归&#xff08;循环&#xff09;方式求解汉诺塔的问题&#xff08;n, a, b, c&#xff09;&#xff0c;即将N个盘子从起始柱&#xff08;标记…

Golang | Leetcode Golang题解之第447题回旋镖的数量

题目&#xff1a; 题解&#xff1a; func numberOfBoomerangs(points [][]int) (ans int) {for _, p : range points {cnt : map[int]int{}for _, q : range points {dis : (p[0]-q[0])*(p[0]-q[0]) (p[1]-q[1])*(p[1]-q[1])cnt[dis]}for _, m : range cnt {ans m * (m - 1)…

多功能快捷回复软件

各位亲爱的客服宝宝们&#xff0c;每天面对大量的客户咨询&#xff0c;您是否还在手动一个一个地打字回复呢&#xff1f;别担心&#xff0c;我们为您带来了一款多功能快捷回复软件——客服宝。有了它&#xff0c;您的工作将变得无比轻松&#xff01; 客服宝是一款集成了内容存储…