CSS 之 图片九宫格变幻效果

一、简介

​ 本篇博客用于讲解如何实现图片九宫格变幻的样式效果,将图片分为九块填充在3×3的的九宫格子元素中,并结合gridhovertransition等CSS属性,实现元素hover时,九宫格子元素合并为一张完整图片的动画效果。

​ 为了简化代码,demo中通过JS设置CSS变量的方法,优化了元素背景的设置过程,减少了代码的繁杂度。

​ 最后还结合js的点击事件实现了一个简易的点击拼图demo。

效果示意图:

在这里插入图片描述

二、具体实现

1、九宫格子元素显示部分图片,然后从外向内组合变幻成完整图片
页面效果:

在这里插入图片描述

实现思路:

​ ① 首先在一个父元素内,创建九个子元素,并通过display: grid;实现3×3的九宫格布局,使其聚集在父元素中间,此时子元素之间无间隙。

​ ② 给这九个子元素通过background的相关属性,分别设置各个子元素的背景图片为总图片的一部分(1/9),其聚集在一起时,正好能够组成一张完成的图片。

​ ③ 在设置子元素背景时,可以通过依次设置九个子元素的背景样式来实现,但是代码过于繁琐。为了优化代码,可以通过js获取相应元素,然后遍历获取到的元素设置CSS变量,最后再统一通过子级选择器+CSS变量的形式,统一设置元素背景。

​ ④ 设置好背景后,利用父元素grid布局的gap属性,统一设置子元素之间的间距,使九个子元素按照固定间距,形成九宫格布局。

​ ⑤ 父元素hover时,修改gap属性的值,使子元素再次聚集在一起,并通过transition实现缓动动画。

核心思路为:变化主体为父元素,子元素不改变,改变父元素控制间隙的属性。

具体代码:
<style>.d1 {/* 设置grid布局 */display: grid;grid-template-columns: 100px 100px 100px;grid-template-rows: 100px 100px 100px;/* 设置子元素间隔 */gap: 20px;width: 340px;height: 340px;/* 设置子元素居中在父元素中心 */justify-content: center;align-content: center;cursor: pointer;/* 设置缓动动画 */transition: all 0.3s linear;border: 2px solid red;}/* hover时消除子元素间隔 */.d1:hover {gap: 0;}/* 统一设置子元素的背景 */.d1>div {background: url(../image/test.jpg) no-repeat;/* 设置背景图片大小 这个大小等于后面拼接成的完成图片的大小 */background-size: 300px 300px;/* 利用JS设置的CSS变量控制子元素的背景 */background-position: var(--bgx, 0) var(--bgy, 0);}
</style><div class="d1"><div class="d1-1"></div><div class="d1-2"></div><div class="d1-3"></div><div class="d1-4"></div><div class="d1-5"></div><div class="d1-6"></div><div class="d1-7"></div><div class="d1-8"></div><div class="d1-9"></div>
</div><script>// 获取所有子元素const d1List = document.querySelectorAll('.d1 > div');// 遍历所有子元素d1List.forEach((item, index) => {// 计算当前子元素所在行 0~2const row = Math.floor(index / 3);// 计算当前子元素所在列 0~2const col = index % 3;console.log(row, col);// 根据子元素所在行和列 设置CSS变量 决定显示的背景图片的偏移量 从而决定背景图片的内容item.style.setProperty('--bgx', `${-col * 100}px`);item.style.setProperty('--bgy', `${-row * 100}px`);})
</script>
2、九宫格子元素显示部分图片,然后从内向外组合变幻成完整图片
页面效果:

在这里插入图片描述

实现思路:

​ ① 首先在一个父元素内,创建九个子元素,并通过display: grid;实现3×3的九宫格布局,初始设置子元素的宽高小于grid布局设置的单元格宽高,然后通过相关属性设置子元素在对应单元格内水平垂直居中。

​ ② 给这九个子元素通过background的相关属性,分别设置背景图片为总图片的一部分。

​ ③ 在设置子元素背景时,可以通过依次设置九个子元素的背景样式来实现,但是代码过于繁琐。为了优化代码,可以通过js获取相应元素,然后遍历获取到的元素设置CSS变量,最后再统一通过子级选择器+CSS变量的形式,统一设置元素背景,使九个子元素各自占据背景图片的一部分。但由于此时子元素的宽高小于grid布局设置的单元格宽高,所以此时子元素之间存在间隙,而且所有子元素组合拼接的背景图片并不完整。

​ ④ 设置父元素hover时,改变子元素的宽高,使其充满单元格宽高,承载的背景图片比例也相应变大,此时子元素之间紧密连接,并组合拼接为一张完整的图片。

核心思路为:变化主体为子元素,子元素改变宽高属性,父元素不变。

具体代码:
<style>.d2 {/* 设置grid布局 */display: grid;grid-template-columns: 100px 100px 100px;grid-template-rows: 100px 100px 100px;width: 300px;height: 300px;cursor: pointer;border: 2px solid red;}.d2>div {/* 设置子元素的初始宽高 */width: 60px;height: 60px;/* 设置子元素位于grid布局网格的中心位置 */justify-self: center;align-self: center;/* 设置子元素缓动动画 */transition: all 0.3s linear;/* 设置子元素背景图片 */background: url(../image/test.jpg) no-repeat;/* 设置背景图片的大小 这个大小等于后面拼接成的完成图片的大小 */background-size: 300px 300px;/* 利用JS设置的CSS变量控制子元素的背景偏移 */background-position: var(--bgx, 0) var(--bgy, 0);}/* 父元素hover之后 修改子元素的宽高 使子元素能连接在一起 */.d2:hover div {width: 100px;height: 100px;}
</style><div class="d2"><div class="d2-1"></div><div class="d2-2"></div><div class="d2-3"></div><div class="d2-4"></div><div class="d2-5"></div><div class="d2-6"></div><div class="d2-7"></div><div class="d2-8"></div><div class="d2-9"></div>
</div><script>// 获取所有子元素const d2List = document.querySelectorAll('.d2 > div');// 遍历所有子元素d2List.forEach((item, index) => {// 计算当前子元素所在行 0~2const row = Math.floor(index / 3);// 计算当前子元素所在列 0~2const col = index % 3;console.log(row, col);// 根据子元素所在行和列 设置CSS变量 决定显示的背景图片偏移量item.style.setProperty('--bgx', `${-col * 100}px`);item.style.setProperty('--bgy', `${-row * 100}px`);})
</script>
3、九宫格子元素显示完整小图,然后从外向内合并成一张完整大图
页面效果:

在这里插入图片描述

实现思路:

​ ① 首先在一个父元素内,创建九个子元素,并通过display: grid;实现3×3的九宫格布局,并且设置子元素之间存在固定间隙。

​ ② 给每一个子元素都设置背景图片,并且设置背景图片的大小等于子元素的大小,使背景图片在子元素中完整展示。

​ ③ 在父元素hover时,修改子元素之间的间隙为0,并且修改子元素的背景图片大小和背景图片的偏移量,使子元素只显示背景图片的一部分,并且所有子元素拼接起来时,能够构成一张完整的背景图片。

核心思路:同时改变父元素和子元素,改变父元素控制间隙的属性,改变子元素的背景属性。

具体代码:
<style>.d4 {/* 设置grid布局 */display: grid;grid-template-columns: 100px 100px 100px;grid-template-rows: 100px 100px 100px;/* 设置子元素之间的间隙 */gap: 20px;width: 340px;height: 340px;/* 设置子元素整体内容区域相对于父元素水平垂直居中 */justify-content: center;align-content: center;cursor: pointer;/* 设置父元素缓动动画 */transition: all 0.4s linear;border: 2px solid red;}.d4>div {/* 设置子元素缓动动画 */transition: all 0.4s linear;/* 设置子元素的背景 背景图片大小适配元素大小 */background: url(../image/test.jpg) no-repeat;background-size: 100% 100%;}/* 父元素hover时 将子元素之间的间隙变为0  */.d4:hover {gap: 0;}/* 父元素hover时 修改子元素背景图片的大小 以及背景图片偏移量 */.d4:hover>div {/* 设置的背景图片大小 就是子元素拼接成一张图片时 图片的大小 */background-size: 300px 300px;/* 根据JS设置的CSS变量 设置背景图片的偏移量 */background-position: var(--bgx, 0) var(--bgy, 0);}
</style><div class="d4"><div class="d4-1"></div><div class="d4-2"></div><div class="d4-3"></div><div class="d4-4"></div><div class="d4-5"></div><div class="d4-6"></div><div class="d4-7"></div><div class="d4-8"></div><div class="d4-9"></div>
</div><script>// 获取所有子元素const d4List = document.querySelectorAll('.d4 > div');// 遍历所有子元素d4List.forEach((item, index) => {// 计算当前子元素所在行 0~2const row = Math.floor(index / 3);// 计算当前子元素所在列 0~2const col = index % 3;console.log(row, col);// 根据子元素所在行和列 设置CSS变量 决定显示的背景图片偏移量item.style.setProperty('--bgx', `${-col * 100}px`);item.style.setProperty('--bgy', `${-row * 100}px`);})
</script>

三、延伸扩展

1、简易的九宫格拼图游戏

​ 上面的三个案例Demo中,都是在父元素hover时,通过控制CSS样式子元素进行整体变化,无法做到针对单个元素进行变化。如果我们结合JS的点击事件,以及子元素的定位属性,就可以实现针对单个元素的变化。比如实现简单的拼图游戏效果等等。

页面效果:

在这里插入图片描述

实现思路:

​ ① 首先在一个父元素内,创建九个子元素,并通过display: grid;实现3×3的九宫格布局,并且设置子元素之间无间隙,紧密结合在一起,并且在父元素中水平垂直居中。

​ ② 通过JS获取所有子元素,然后遍历获取到的子元素,根据子元素所在行和所在列,设置表示背景偏移的CSS变量,最后再统一通过子级选择器+CSS变量的形式,统一设置元素背景,使九个子元素各自占据背景图片的一部分。

​ ③ 在上一步遍历所有子元素时,根据子元素所在行和所在列,同步设置表示子元素偏移的CSS变量,然后再给子元素设置 position: relative;相对定位,最后再统一通过子级选择器+CSS变量的形式,利用CSS变量设置子元素的偏移量,使其形成有监理的九宫格布局。

​ ④ 在遍历所有子元素的同时,给所有子元素绑定点击事件,根据事件触发元素上的表示元素偏移量的CSS变量的值,决定元素该向内还是向外偏移,然后根据元素所在的行和列,决定具体的元素偏移量。

具体代码:
<style>.d3 {/* 设置grid布局 */display: grid;grid-template-columns: 100px 100px 100px;grid-template-rows: 100px 100px 100px;width: 340px;height: 340px;/* 设置子元素整体内容区域相对于父元素水平垂直居中 */justify-content: center;align-content: center;cursor: pointer;border: 2px solid red;}.d3>div {/* 设置子元素的偏移 使子元素之间出现间隙 形成九宫格布局 */position: relative;/* 利用JS的定义的CSS变量设置元素偏移量 */left: var(--pleft, 0);top: var(--ptop, 0);z-index: 1;width: 100px;height: 100px;/* 设置子元素的背景图片 */background: url(../image/test.jpg) no-repeat;/* 设置背景图片尺寸 该尺寸等同于子元素拼接后图片的尺寸 */background-size: 300px 300px;/* 利用JS的定义的CSS变量设置背景图片的偏移量 */background-position: var(--bgx, 0) var(--bgy, 0);/* 设置子元素的缓动动画 */transition: all 0.3s linear;}
</style><div class="d3"><div class="d3-1"></div><div class="d3-2"></div><div class="d3-3"></div><div class="d3-4"></div><div class="d3-5"></div><div class="d3-6"></div><div class="d3-7"></div><div class="d3-8"></div><div class="d3-9"></div>
</div><script>// 获取所有子元素const d3List = document.querySelectorAll('.d3 > div');// 遍历所有子元素d3List.forEach((item, index) => {// 计算当前子元素所在行 0~2const row = Math.floor(index / 3);// 计算当前子元素所在列 0~2const col = index % 3;// 根据子元素所在行和列 设置CSS变量 决定显示的背景图片偏移量item.style.setProperty('--bgx', `${-col * 100}px`);item.style.setProperty('--bgy', `${-row * 100}px`);// 根据子元素所在行和列 设置CSS变量 决定元素的偏移量item.style.setProperty('--pleft', `${(col - 1) * 20}px`);item.style.setProperty('--ptop', `${(row - 1) * 20}px`);// 为当前子元素添加点击事件item.addEventListener('click', (e) => {// 获取当前子元素表示元素偏移量的CSS变量值const left = e.target.style.getPropertyValue('--pleft');const top = e.target.style.getPropertyValue('--ptop');// 判断当前子元素是否在原始位置if (left === '0px' && top === '0px') {// 如果在原始位置 则根据子元素所在行和列 设置CSS变量 决定使元素偏移e.target.style.setProperty('--pleft', `${(col - 1) * 20}px`);e.target.style.setProperty('--ptop', `${(row - 1) * 20}px`);} else {// 如果不在原始位置 则将CSS变量值设置为0px 将其移回原始位置e.target.style.setProperty('--pleft', `0px`);e.target.style.setProperty('--ptop', `0px`);}});})
</script>

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

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

小白水平理解面试经典题目LeetCode 594 Longest Harmonious Subsequence(最大和谐字符串)

594 最大和谐字符串 这道题属于字符串类型题目&#xff0c;解决的办法还是有很多的&#xff0c;暴力算法&#xff0c;二分法&#xff0c;双指针等等。 题目描述 和谐数组是指一个数组里元素的最大值和最小值之间的差别 正好是 1 。 现在&#xff0c;给你一个整数数组 nums …

BAT学习笔记:详解环境变量及其所有创建方法

文章目录 一、初识环境变量二、什么是环境变量三、为什么需要环境变量四、环境变量的分类五、环境变量的设置 一、初识环境变量 1.windows 的搜索框中输入 查看高级系统设置。点击打开系统属性窗口。 2. 在系统属性窗口中&#xff0c;点击右下方的“环境变量”打开环境变量设…

取消Vscode在输入符号时自动补全

取消Vscode在输入符号时自动补全 取消Vscode在输入符号时自动补全问题演示解决方法 取消Vscode在输入符号时自动补全 问题演示 在此状态下输入/会直接自动补全, 如下图 笔者想要达到的效果为可以正常输入/而不进行补全, 如下图 解决方法 在设置->文本编辑器->建议, 取消…

Making Large Language Models Perform Better in Knowledge Graph Completion论文阅读

文章目录 摘要1.问题的提出引出当前研究的不足与问题KGC方法LLM幻觉现象解决方案 2.数据集和模型构建数据集模型方法基线方法任务模型方法基于LLM的KGC的知识前缀适配器知识前缀适配器 与其他结构信息引入方法对比 3.实验结果与分析结果分析&#xff1a;可移植性实验&#xff1…

C++进阶(六)map和set

&#x1f4d8;北尘_&#xff1a;个人主页 &#x1f30e;个人专栏:《Linux操作系统》《经典算法试题 》《C》 《数据结构与算法》 ☀️走在路上&#xff0c;不忘来时的初心 文章目录 一、序列式容器与关联式容器二、键值对三、树形结构的关联式容器1、set1、set的介绍2、set的使…

在windows环境下安装hadoop

Hadoop是一个分布式系统基础架构。用户可以在不了解分布式底层细节的情况下&#xff0c;开发分布式程序。但这个架构是基于java语言开发的&#xff0c;所以要先进行jdk的安装&#xff0c;如果电脑已经配置过jdk或者是曾经运行成功过java文件&#xff0c;那就可以跳过第一步。 …

vue3+naiveUI二次封装的v-model 联动输入框

根据官网说明使用 源码 <template><div class"clw-input pt-3"><n-inputref"input":value"modelValue":type"type":title"title"clearable:disabled"disabled":size"size"placeholder&…

深度学习(6)--Keras项目详解

目录 一.项目介绍 二.项目流程详解 2.1.导入所需要的工具包 2.2.输入参数 2.3.获取图像路径并遍历读取数据 2.4.数据集的切分和标签转换 2.5.网络模型构建 2.6.绘制结果曲线并将结果保存到本地 三.完整代码 四.首次运行结果 五.学习率对结果的影响 六.Dropout操作…

RBD —— 不同材质破碎

目录 Working with concrete Chipping Details Proxy geometry Constraints Working with glass Chipping Proxy geometry Constraints Resolving issues with glass fracturing Working with wood Clustering Using custom cutters Working with concrete Concr…

echarts的tooltip样式内容调整

项目场景 浦东大道 交通态势 预期效果 初始效果 没有数据显示undefined 代码实现 tooltip: {trigger: axis,axisPointer: {type: none},formatter: function (val) {let ret val.forEach((item) > {// console.log(图表里面的eitem, item)if (item.seriesType ! pictor…

QT 范例阅读:Vector Deformation

效果图&#xff1a; 主要代码&#xff1a; 实现放大镜效果QPainter painter;//两种方式if (1) {m_lens_image QImage(bounds.size(), QImage::Format_ARGB32_Premultiplied);m_lens_image.fill(0);painter.begin(&m_lens_image);} else {m_lens_pixmap QPixmap(bounds.si…

jar手动上传到Nexus私服

jar手动上传到Nexus私服 一简介二上传方法1.登录nexus,打开nexus控制台。upload---maven-release2.上传jar3.验证jar包是否上传成功4、注意事项 一简介 在maven项目中我们可以通过相关配置可以将项目打包成jar&#xff0c;然后deploy上传到私服中。要是别人提供的第三方jar&am…