现代 CSS 高阶技巧:实现平滑内凹圆角的工程化实践

news2025/7/20 6:47:04

通过 数学计算 + CSS mask 复合遮罩 实现的真正几何内凹效果:

在这里插入图片描述

背景是一张图片,用来证明中间的凹陷是透明的。


完整代码

app.js

import FormPage from "./pages/formPage";
import "./App.css";
const App = () => {
  return (
    <div className="box">
      <div className="block"></div>
    </div>
  );
};

export default App;

app.css

.box {
  background: url(../src/pages/img/10.jpg) 100%;
  height: 500px;
}

.block {
  /* 增大圆角半径 */
  --r: 6px;
  /* 适当增加斜切长度 */
  --s: 69px;
  /* 微调角度使过渡更自然 */
  --a: 44deg;
  
  /* 更柔和的边缘过渡 */
  --_m: 0/calc(2*var(--r)) var(--r) no-repeat radial-gradient(50% 100% at bottom, #000 calc(100% - 0.8px), transparent);

  /* 自动计算的水平偏移 */
  --_d: (var(--s) + var(--r)) * cos(var(--a));


  width: 300px;
  height: 100px;
  background-color: antiquewhite;
  border-radius: 150px;
  mask:
    calc(50% + var(--_d)) var(--_m), calc(50% - var(--_d)) var(--_m),
    radial-gradient(var(--s) at 50% calc(-1*sin(var(--a))*var(--s)),
      transparent 100%, #000 calc(100% + 0.8px)) 0 calc(var(--r)*(1 - sin(var(--a)))),
    linear-gradient(90deg, #000 calc(50% - var(--_d)), transparent 0 calc(50% + var(--_d)), #000 0);
  mask-repeat: no-repeat;
}

🧠 整体思路

这段代码的核心思想是:

使用多个 mask 图层组合,通过 radial-gradient 和 linear-gradient 的叠加,形成一个“中间凹进去、边缘过渡柔和”的视觉效果。

它利用了 CSS 中的变量(--r, --s, --a)、三角函数和 mask 属性,实现了动态可配置的内凹圆角效果。


📏 变量解析(CSS Custom Properties)

--r: 6px;        /* 凹陷区域的半径(控制凹陷大小) */
--s: 69px;       /* 圆弧的半径(控制凹陷的位置) */
--a: 44deg;      /* 角度(用于三角函数计算) */

这些变量可以方便地调整最终的视觉效果。


🧮 计算变量(关键逻辑)

--_m:定义一个遮罩图层(更柔和的边缘过渡)

--_m: 0 / calc(2*var(--r)) var(--r) no-repeat
    radial-gradient(50% 100% at bottom, #000 calc(100% - 0.8px), transparent);
  • 创建一个宽度为 2 * --r,高度为 --r 的径向渐变。
  • 渐变从底部开始,颜色从不透明到透明,形成一个边缘模糊的遮罩条带
  • 这个条带会在最终 mask 中作为“边缘柔化”层使用。

--_d:水平偏移距离(基于三角函数自动计算)

--_d: (var(--s) + var(--r)) * cos(var(--a));
  • 使用余弦函数计算出一个水平方向上的偏移值。
  • 这个偏移值决定了凹陷区域在水平方向上的位置。
  • 结合下面的 mask 设置,使凹陷区域对称分布在中心两侧。

🎭 Mask 图层详解(这是整个效果的关键)

mask:
  calc(50% + var(--_d)) var(--_m),
  calc(50% - var(--_d)) var(--_m),
  radial-gradient(var(--s) at 50% calc(-1*sin(var(--a))*var(--s)),
    transparent 100%, #000 calc(100% + 0.8px)) 0 calc(var(--r)*(1 - sin(var(--a)))),
  linear-gradient(90deg, #000 calc(50% - var(--_d)), transparent 0 calc(50% + var(--_d)), #000 0);

我们来逐行拆解这四个 mask 图层:


🔹 第一层 & 第二层(边缘柔化层)

calc(50% + var(--_d)) var(--_m),
calc(50% - var(--_d)) var(--_m)
  • 这两个图层使用的是之前定义好的 --_m 遮罩条带。
  • 分别放置在中心左右各偏移 --_d 的位置。
  • 作用是柔和边缘,避免生硬的裁剪边界。

🔸 第三层(核心凹陷层)

radial-gradient(var(--s) at 50% calc(-1*sin(var(--a))*var(--s)),
  transparent 100%, #000 calc(100% + 0.8px))
  0 calc(var(--r)*(1 - sin(var(--a))));
  • 创建一个以中心为圆心、向上偏移一定距离的径向渐变。
  • 半径为 --s,位于垂直方向上偏移 sin(a) * s
  • 渐变从透明到黑色,超出部分变为不透明。
  • 最后定位在 0 calc(...),即垂直方向向下偏移一点,让凹陷区域更贴合整体形状。

✅ 这一层是形成“内凹”视觉的核心图层


🔷 第四层(背景遮罩层)

linear-gradient(90deg, #000 calc(50% - var(--_d)), transparent 0 calc(50% + var(--_d)), #000 0)
  • 水平方向的线性渐变:

    • 左侧和右侧为黑色(显示区域)
    • 中间一段为透明(隐藏区域)
  • 作用是遮住中间的凹陷区域,只保留两边的内容。


🧪 最终效果总结

图层作用
第一、二层边缘柔化处理(避免锯齿感)
第三层核心凹陷区域(模拟“内凹”形状)
第四层背景遮罩(只显示两边,中间隐藏)

结合起来就形成了一个中间凹陷、边缘柔和、对称分布的视觉效果,非常适合用在按钮、卡片等需要轻微凹陷质感的 UI 元素中。


🖼️ 示例效果预览(文字描述)

想象一个椭圆形的盒子(border-radius: 150px),原本是完整的圆形。但在它的正中央,有一个向上弯曲的凹陷区域,就像轻轻按下按钮时那种感觉,边缘还有轻微的阴影过渡。

这种效果常见于 macOS 的菜单栏按钮、iOS 控件等现代 UI 设计中。


🛠️ 如何调整?

你可以通过修改以下变量来实时调整视觉效果:

变量默认值调整建议影响
--r6px增大 → 凹陷更大凹陷区域大小
--s69px增大 → 凹陷更深凹陷位置与深度
--a44deg增大 → 凹陷更靠上凹陷角度与位置
background-colorantiquewhite更浅/深色整体对比度和质感

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

【运维自动化-标准运维】如何实现在不同步骤间传递参数

当流程有多个步骤时&#xff0c;经常需要把前面某个个步骤处理的结果传递给下一个或后面的步骤使用&#xff08;输出作为输入&#xff09;&#xff0c;这就是跨步骤传参的场景&#xff0c;标准运维通过特有的标记符号"<SOPS_VAR>key:value</SOPS_VAR> "来…

[AI]主流大模型、ChatGPTDeepseek、国内免费大模型API服务推荐(支持LangChain.js集成)

主流大模型特色对比表 模型核心优势适用场景局限性DeepSeek- 数学/代码能力卓越&#xff08;GSM8K准确率82.3%&#xff09;1- 开源生态完善&#xff08;支持医疗/金融领域&#xff09;7- 成本极低&#xff08;API价格仅为ChatGPT的2%-3%&#xff09;5科研辅助、代码开发、数据…

手机IP地址更换的影响与操作指南

在移动互联网时代&#xff0c;IP地址如同手机的“网络身份证”&#xff0c;其变更可能对上网体验、隐私安全及服务访问产生连锁反应。无论是为了绕过地域限制、保护隐私&#xff0c;还是解决网络冲突&#xff0c;了解IP更换的影响与正确操作方法都至关重要。本文将系统分析影响…

MongoDB 安全机制详解:全方位保障数据安全

在当今数据驱动的时代&#xff0c;数据库安全至关重要。MongoDB 作为一款流行的 NoSQL 数据库&#xff0c;广泛应用于 Web 应用、大数据分析和物联网等领域。然而&#xff0c;随着 MongoDB 的普及&#xff0c;其安全性也面临诸多挑战&#xff0c;如未授权访问、数据泄露和注入攻…

嵌入式学习之系统编程(六)线程

目录 一、线程 &#xff08;一&#xff09;线程概念 &#xff08;二&#xff09;特征 &#xff08;三&#xff09;优缺点 二、线程与进程的区别&#xff08;面问&#xff09; 三、多线程程序设计步骤 四、线程的创建&#xff08;相关函数&#xff09; 1、pthread_create…

大语言模型 提示词的少样本案例的 演示选择与排序新突破

提示词中 演示示例的选择与排序 这篇论文《Rapid Selection and Ordering of In-Context Demonstrations via Prompt Embedding Clustering》聚焦于提升大语言模型(LLMs)在自适应上下文学习(ICL)场景中演示示例的选择与排序效率 一、论文要解决的问题 在上下文学习(ICL)…

【算法篇】二分查找算法:基础篇

题目链接&#xff1a; 34.在排序数组中查找元素的第一个和最后一个位置 题目描述&#xff1a; 给你一个按照非递减顺序排列的整数数组 nums&#xff0c;和一个目标值 target。请你找出给定目标值在数组中的开始位置和结束位置。 如果数组中不存在目标值 target&#xff0c;返…

重磅发布 | 复旦533页《大规模语言模型:从理论到实践(第2版)》(免费下载)

在人工智能浪潮席卷全球的今天&#xff0c;大语言模型正以前所未有的速度推动着科技进步和产业变革。从 ChatGPT 到各类行业应用&#xff0c;LLM 不仅重塑了人机交互的方式&#xff0c;更成为推动学术研究与产业创新的关键技术。 面对这一飞速演进的技术体系&#xff0c;如何系…

智能体赋能效率,企业知识库沉淀价值:UMI企业智脑的双轮驱动!

智能体企业知识库&#xff1a;UMI企业智脑的核心功能与价值 在人工智能技术飞速发展的今天&#xff0c;企业智能化转型已经成为不可逆转的趋势。作为企业级AI智能体开发平台的佼佼者&#xff0c;优秘智能推出的UMI企业智脑&#xff0c;以其强大的智能体开发能力和全面的企业知…

vue项目 build时@vue-office/docx报错

我在打包vue项目时&#xff0c; 开始用的npm run build和cnpm run build&#xff0c;总是提示 vue-office/docx 错误&#xff0c;尝试过用cnpm重新安装node_modules几次都没用。类似下面的提示一直有。 Error: [commonjs--resolver] Failed to resolve entry for package "…

#RabbitMQ# 消息队列入门

目录 一 MQ技术选型 1 运行rabbitmq 2 基本介绍 3 快速入门 1 交换机负责路由消息给队列 2 数据隔离 二 Java客户端 1 快速入门 2 WorkQueue 3 FanOut交换机 4 Direct交换机 5 Topic交换机 *6 声明队列交换机 1 在配置类当中声明 2 使用注解的方式指定 7 消息转…

EPT(Efficient Prompt Tuning)方法,旨在解决提示调优(Prompt Tuning)中效率与准确性平衡和跨任务一致性的问题

EPT(Efficient Prompt Tuning)方法,旨在解决提示调优(Prompt Tuning)中效率与准确性平衡和跨任务一致性的问题 一、核心原理:分解提示与多空间投影 1. 提示分解:用低秩矩阵压缩长提示 传统问题: 长提示(如100个token)精度高但训练慢,短提示(如20个token)速度快但…

云原生安全核心:云安全责任共担模型(Shared Responsibility Model)详解

&#x1f525;「炎码工坊」技术弹药已装填&#xff01; 点击关注 → 解锁工业级干货【工具实测|项目避坑|源码燃烧指南】 1. 基础概念 什么是云安全责任共担模型&#xff1f; 云安全责任共担模型&#xff08;Shared Responsibility Model, SRM&#xff09;是云服务提供商&…

go并发与锁之sync.Mutex入门

sync.Mutex 原理&#xff1a;一个共享的变量&#xff0c;哪个线程握到了&#xff0c;哪个线程可以执行代码 功能&#xff1a;一个性能不错的悲观锁&#xff0c;使用方式和Java的ReentrantLock很像&#xff0c;就是手动Lock&#xff0c;手动UnLock。 使用例子&#xff1a; v…

[Java恶补day8] 3. 无重复字符的最长子串

给定一个字符串 s &#xff0c;请你找出其中不含有重复字符的 最长 子串 的长度。 示例 1: 输入: s “abcabcbb” 输出: 3 解释: 因为无重复字符的最长子串是 “abc”&#xff0c;所以其长度为 3。 示例 2: 输入: s “bbbbb” 输出: 1 解释: 因为无重复字符的最长子串是 “…

LabVIEW教学用开发平台

一、培训目标 基础编程&#xff1a;掌握 LabVIEW 数据类型、程序结构、子 VI 设计与调试技巧。 硬件通信&#xff1a;精通 RS-232/485、TCP/IP、Modbus、PLC 等工业通信协议及实现。 高级设计模式&#xff1a;熟练运用状态机、生产者 - 消费者模式构建复杂测控系统。 项目实…

Package Size Comparison – 6 Leads

Package Size Comparison 6 LeadsTSOP SOT SM SMT SOT23 SC-74 SC-59 SC-88 SOT363 US6 UMT6 SC-70 SOT563 ES EMT SC-75-6

python打卡day38

Dataset和DataLoader 知识点回顾&#xff1a; Dataset类的__getitem__和__len__方法&#xff08;本质是python的特殊方法&#xff09;Dataloader类minist手写数据集的了解 作业&#xff1a;了解下cifar数据集&#xff0c;尝试获取其中一张图片 在遇到大规模数据集时&#xff0c…

vLLM 核心技术 PagedAttention 原理详解

本文是 vLLM 系列文章的第二篇&#xff0c;介绍 vLLM 核心技术 PagedAttention 的设计理念与实现机制。 vLLM PagedAttention 论文精读视频可以在这里观看&#xff1a;https://www.bilibili.com/video/BV1GWjjzfE1b 往期文章&#xff1a; vLLM 快速部署指南 1 引言&#xf…

《软件工程》第 2 章 -UML 与 RUP 统一过程

在软件工程领域&#xff0c;UML&#xff08;统一建模语言&#xff09;与 RUP&#xff08;统一过程&#xff09;是进行面向对象软件开发的重要工具和方法。接下来&#xff0c;我们将深入探讨第 2 章的内容&#xff0c;通过案例和代码&#xff0c;帮助大家理解和掌握相关知识。 …