HTML + CSS 创建图片倒影的 5 种方法
目标:掌握多种生成“图片倒影 / Reflection”效果的实现思路,理解兼容性、性能差异与最佳实践,方便在真实业务(商品展示、相册、登陆页面视觉强化)中安全使用。
总览对比
方法 | 核心技术 | 代码量 | 兼容性 | 动态内容适配 | 可控性 | 适用场景 |
---|---|---|---|---|---|---|
1. -webkit-box-reflect | 私有 CSS 属性 | 最少 | 仅 WebKit (Chrome / Safari / Edge) | 自动 | 中 | 快速 Demo / 营销页 |
2. 伪元素 + transform: scaleY(-1) | 标准 CSS | 少 | 所有现代浏览器 | 良好 | 高(可自定义遮罩) | 通用首选 |
3. 伪元素 + mask-image / -webkit-mask | CSS Mask | 中 | Safari / Chromium (Firefox 部分实验) | 良好 | 很高(渐隐更自然) | 高端展示 |
4. SVG 复制 + 渐变遮罩 | 内联 SVG | 中 | 全面 (IE 除外) | 良好 | 很高(滤镜/形变) | 复杂视觉 / 批量渲染 |
5. Canvas 二次绘制 | <canvas> | 高 | 全面 | 需手动重绘 | 最高(像素级) | 动态生成 / 后处理 |
选择建议:
- 追求最少代码 & 不顾部分浏览器:用 1。
- 需要兼容性 + 易维护:用 2。
- 想要柔和渐隐过渡、无多余 DOM:用 3。
- 大型可视化 / 复杂滤镜链:用 4。
- 需要最终导出合成图 / 动态内容(如生成分享海报):用 5。
方法一:-webkit-box-reflect
(最简单 / 兼容性受限)
<style>.reflect-webkit {width: 240px;-webkit-box-reflect: below 6px linear-gradient(to bottom, rgba(0, 0, 0, 0.25), transparent70%);}
</style>
<img class="reflect-webkit" src="demo.jpg" alt="Product" />
说明:
- 语法:
-webkit-box-reflect: <direction> <offset> <mask>
linear-gradient
充当倒影的渐隐遮罩。
优点:单行 + 自动跟随宽高。
缺点:仅 WebKit 内核(Firefox 不支持)。
适合:临时视觉增强、非核心信息。
方法二:伪元素复制 + 反转(推荐通用方案)
思路:利用容器包裹图片,伪元素 ::after
再绘制同一张图像,垂直翻转并添加渐变。
<div class="reflection-box"><img src="demo.jpg" alt="Phone" />
</div><style>.reflection-box {position: relative;width: 240px;}.reflection-box img {display: block;width: 100%;}.reflection-box::after {content: '';position: absolute;left: 0;right: 0;top: 100%;height: 100%;background: url('demo.jpg') center/cover no-repeat;transform: scaleY(-1);transform-origin: top;opacity: 0.6;/* 渐隐叠加 */mask-image: linear-gradient(to bottom,rgba(0, 0, 0, 0.8),rgba(0, 0, 0, 0));-webkit-mask-image: linear-gradient(to bottom,rgba(0, 0, 0, 0.8),rgba(0, 0, 0, 0));}
</style>
处理动态 src:
- 如果图片 URL 需动态绑定,可用行内
style="--src:url('xxx')"
+background: var(--src)
或用 JS 设置伪元素。 - 亦可直接复制
<img>
节点再scaleY(-1)
,如下:
<div class="reflect-wrap"><img src="demo.jpg" alt="Phone" class="origin" /><img src="demo.jpg" alt="Phone reflection" class="mirror" />
</div><style>.reflect-wrap {width: 240px;position: relative;}.reflect-wrap .origin {display: block;width: 100%;}.reflect-wrap .mirror {display: block;width: 100%;transform: scaleY(-1);transform-origin: top;margin-top: 6px;mask-image: linear-gradient(to bottom, rgba(0, 0, 0, 0.6), transparent);-webkit-mask-image: linear-gradient(to bottom,rgba(0, 0, 0, 0.6),transparent);opacity: 0.8;}
</style>
优点:标准、兼容好、可扩展。
注意:双份图片会触发两次解码,可用 <img decoding="async" loading="lazy">
或复用绘制(JS 画到 canvas 再生成 DataURL)。
方法三:伪元素 + Mask(强化渐隐 & 灵活形状)
核心区别:不直接复制图片,而是使用 CSS 变量引用同一来源,搭配 mask-image
形成更可控的透明衰减(可变换曲线)。
<figure class="mask-reflect" style="--src:url('demo.jpg')"><img src="demo.jpg" alt="Laptop" />
</figure><style>.mask-reflect {position: relative;width: 260px;}.mask-reflect img {width: 100%;display: block;}.mask-reflect::after {content: '';position: absolute;inset: 0;top: 100%;height: 100%;background: var(--src) center/cover no-repeat;transform: scaleY(-1);transform-origin: top;/* 自定义遮罩——使用非线性渐变模拟更柔的衰减 */mask-image: linear-gradient(to bottom,rgba(0, 0, 0, 0.9) 0%,rgba(0, 0, 0, 0.4) 35%,rgba(0, 0, 0, 0.15) 60%,rgba(0, 0, 0, 0) 100%);-webkit-mask-image: linear-gradient(to bottom,rgba(0, 0, 0, 0.9) 0%,rgba(0, 0, 0, 0.4) 35%,rgba(0, 0, 0, 0.15) 60%,rgba(0, 0, 0, 0) 100%);opacity: 0.85;filter: blur(0.4px) saturate(0.95);}
</style>
拓展:
- 横向拉伸 / 倾斜效果:附加
transform: scaleY(-1) skewX(3deg);
- 波纹倒影:叠加
filter: url(#turbulence)
(需 SVG filter)。
方法四:SVG 复制 + 渐变遮罩
适合批量渲染(一个 SVG 中包含多个倒影)或需要滤镜(模糊、色偏、波浪)。
<svg width="260" viewBox="0 0 260 360" class="svg-reflect"><defs><linearGradient id="fade" x1="0" y1="0" x2="0" y2="1"><stop offset="0%" stop-color="black" stop-opacity="0.8" /><stop offset="70%" stop-color="black" stop-opacity="0" /></linearGradient><mask id="mask-fade" maskUnits="userSpaceOnUse"><rect x="0" y="180" width="260" height="180" fill="url(#fade)" /></mask></defs><!-- 原图 --><image href="demo.jpg" x="0" y="0" width="260" height="180" /><!-- 倒影 --><g transform="translate(0,180) scale(1,-1)" mask="url(#mask-fade)"><image href="demo.jpg" x="0" y="0" width="260" height="180" opacity="0.9" /></g>
</svg>
优点:
- 所有可视操作可在 SVG 内完成(模糊、波纹、颜色矩阵)。
- 单文件可复制多份资源。
缺点:语法冗长;与普通 DOM 混排需处理层级。
扩展滤镜(波纹):
<filter id="ripple" x="-20%" y="-20%" width="140%" height="140%"><feTurbulence baseFrequency="0.01 0.15" numOctaves="2" result="turb"/><feDisplacementMap in2="turb" in="SourceGraphic" scale="6" xChannelSelector="R" yChannelSelector="G"/>
</filter>
然后在倒影 <g>
里加 filter="url(#ripple)"
。
方法五:Canvas 动态绘制(可导出 / 像素级)
适合:
- 需要合成单张最终图(下载 / 分享)。
- 倒影需要与原图进行额外像素操作(模糊、曲面映射、噪声)。
<canvas id="c" width="260" height="360"></canvas>
<script>const img = new Image();img.src = 'demo.jpg';img.onload = () => {const h = 180;const w = 260;const canvas = document.getElementById('c');const ctx = canvas.getContext('2d');// 原图ctx.drawImage(img, 0, 0, w, h);// 倒影:缩放坐标系ctx.save();ctx.translate(0, h * 2); // 移到下面ctx.scale(1, -1);ctx.drawImage(img, 0, 0, w, h);ctx.restore();// 渐隐:创建渐变遮罩const gradient = ctx.createLinearGradient(0, h, 0, h * 2);gradient.addColorStop(0, 'rgba(0,0,0,0.6)');gradient.addColorStop(0.7, 'rgba(0,0,0,0)');ctx.globalCompositeOperation = 'destination-in';ctx.fillStyle = gradient;ctx.fillRect(0, h, w, h);ctx.globalCompositeOperation = 'source-over';};
</script>
延伸:
- 添加 Blur:使用离屏 canvas 再
ctx.filter = 'blur(2px)'
。 - 曲面:对每一行像素裁剪后重新绘制(实现较复杂)。
- 封装为函数供多个图片批量处理。
性能与内存考量
关注点 | 说明 | 建议 |
---|---|---|
重复解码 | 伪元素背景与 <img> 均触发 | 使用缓存(同 src 浏览器已缓存),或使用单 <img> +CSS 复制(方法二第二种 DOM 方式) |
Reflow | 倒影高度改变影响布局 | 给容器固定高度 / 使用绝对定位避免抖动 |
重绘成本 | CSS 滤镜 / SVG 滤镜会增加渲染成本 | 降低滤镜强度;按需加载(IntersectionObserver) |
Canvas 占用 | 多张大图 + 离屏操作 | 控制尺寸,复用 canvas |
动态数据 | src 频繁变化 | Debounce 更新;避免在视口外构建倒影 |
常见坑 & 解决方案
问题 | 场景 | 解决 |
---|---|---|
图片加载时闪烁 | 背景方式倒影先出现空白 | 用 onload 后再添加反射 class |
高 DPI 模糊 | Canvas 导出在 Retina 模糊 | canvas.width = cssWidth * devicePixelRatio 再缩放绘制 |
Mask 失效 | Firefox 对 mask-image 支持差异 | 降级:在 Firefox 检测后改用透明渐变 PNG 覆盖 |
滤镜太重卡顿 | 多个 blur(8px) | 降级为较小 blur + opacity |
SEO 影响 | 额外 <img> 计入图片索引 | 第二种 DOM 复制时给倒影图 aria-hidden="true" + alt="" |
可复用 CSS 片段(方法二封装)
.reflect {position: relative;display: inline-block;
}
.reflect > img {display: block;
}
.reflect::after {content: '';position: absolute;left: 0;right: 0;top: 100%;height: var(--reflect-height, 100%);background: var(--reflect-src) center/contain no-repeat;transform: scaleY(-1);transform-origin: top;opacity: var(--reflect-opacity, 0.7);mask-image: linear-gradient(to bottom, rgba(0, 0, 0, 0.9), transparent);-webkit-mask-image: linear-gradient(to bottom,rgba(0, 0, 0, 0.9),transparent);
}
使用:
<divclass="reflect"style="--reflect-src:url('demo.jpg'); --reflect-height:100%; --reflect-opacity:.6"
><img src="demo.jpg" alt="Phone" />
</div>
JS 动态设置:
const wrap = document.querySelector('.reflect');
const img = wrap.querySelector('img');
img.addEventListener('load', () => {wrap.style.setProperty('--reflect-src',`url('${img.currentSrc || img.src}')`);
});
进阶创意玩法
创意 | 说明 | 关键点 |
---|---|---|
波浪倒影 | 倒影加 feTurbulence 位移 | SVG 过滤或 Canvas 位移 |
动态闪光 | 倒影上加渐变动画 | keyframes 改变 mask 的角度 |
颜色偏移 | 倒影轻微冷色 | filter: hue-rotate(5deg) saturate(.9) |
模糊加速 | 滚动远离时加大模糊 | IntersectionObserver + class 切换 |
多层折射 | 多个伪元素多次缩放 | 性能注意:限制层数 |
选择决策树(简化)
- 是否需要导出最终合成图?→ 是:Canvas。
- 是否需要复杂滤镜 / 波纹?→ 是:SVG / Canvas。
- 是否关键页面全平台必须统一?→ 是:伪元素方案。
- 是否只是临时视觉点缀且不在意 Firefox?→
-webkit-box-reflect
。 - 想要更丝滑衰减边缘?→ 增强 Mask(方法三)。
快速小抄(Cheat Sheet)
目标 | 推荐代码片段 |
---|---|
最快 Demo | -webkit-box-reflect: below 4px linear-gradient(...) |
通用实现 | 伪元素 + scaleY(-1) + mask-image |
柔和渐隐 | 多段 stop 的 linear-gradient 遮罩 |
波纹特效 | SVG feTurbulence + feDisplacementMap |
导出图片 | Canvas 绘制 + toDataURL() |
性能优化 | 只为首屏可见元素添加倒影 / 复用 src |
总结
图片倒影本质是:复制 → 翻转 → 衰减透明度 +(可选)附加滤镜。选型关键在 兼容性、性能、是否需后处理 / 导出、视觉精细度 之间的平衡。实际项目中建议封装成组件(接收:图片源、倒影高度、衰减曲线、滤镜强度),统一管理,避免随处散落临时写法。
思路优先,技巧其次。掌握“复制 + 翻转 + 遮罩”内核,再延伸任意创意。
(完)