news 2026/6/9 22:07:12

从一个按钮实例入门CSS in JS之styled-components

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
从一个按钮实例入门CSS in JS之styled-components

前言

最近接手一个react项目,项目中使用的 CSS in JS方案来实现CSS模块化,它使用的包是styled-components

我以前的项目Vue居多的,Vue是很简单的通过scoped就可以实现模块化CSS,当然也在项目中用过CSS Modules方案,但CSS in JS还是第一次接触到,正好趁这个时机学习一下,在做demo时候把平时开发用的比较高频的CSS特性都试了一下,最终完成了一个小按钮demo,突然发现它是一个不错的用来入门styled-components不错的小案例,特记录

styled-components介绍

主要作用是它可以编写实际的CSS代码来设计组件样式,也不需要组件和样式之间的映射,即创建后就是一个正常的React组件,在一个组件内会将结构、样式和逻辑写在一起,虽然这违背了关注点分离的原则,但是这有利于组件间的隔离

使用styled-components不需要再使用className属性来控制样式,而是将样式写成更具语义化的组件的形式,使用style-components会随机生成一个class名称,这样保证每个元素的样式名都是唯一的,当然因为随机生成,维护会增加难度,很难通过样式名定位到元素

入门styled-components的小按钮效果

Screen Recording 2025-10-31 at 14.42.32.gif

小按钮的关键代码

import styled, { keyframes, createGlobalStyle, css } from 'styled-components'

const Wrap = styled.div`

width: 100%;

height: 100vh;

display: flex;

flex-direction: column;

justify-content: center;

align-items: center;

gap: 20px;

`

// css帧动画

const aniTag = keyframes`

0% {

opacity: 0;

transform: translateY(-100%);

}

100% {

opacity: 1;

transform: translateY(0);

}

`

const MyButton = styled.div`

width: 400px;

height: 80px;

border:1px solid gray;

border-radius: 10px;

overflow: hidden;

display: flex;

flex-direction: column;

justify-content: center;

align-items: center;

position: relative;

// 传参

background-color: ${props => props.bgNormalColor ? props.bgNormalColor : 'blue'};

transition: background-color 0.3s ease-in-out;

// 伪元素

&::after{

content: 'after';

height: 20px;

font-size: 14px;

line-height: 20px;

padding:0 5px;

border-radius: 0 0 6px 6px;

position: absolute;

top: 0;

right: 10px;

background-color: yellow;

color: ${props => props.hightightColor ? props.hightightColor : 'white'};

opacity: 0;

}

// 子孙选择器

p {

line-height: 24px;

color: ${props => props.normalColor ? props.normalColor : 'black'};

transition: color 0.3s ease-in-out;

}

// 当有全局样式的时候,使用&&提高局部样式优先级来覆盖全局样式

&& span{

color: rgba(255,0,255,0.6);

}

// 类选择器

.emphasize{

text-decoration: underline;

}

// 伪类

&:hover{

background-color: ${props => props.bgHighlightColor ? props.bgHighlightColor : 'yellow'};

p{

color: ${props => props.hightightColor ? props.hightightColor : 'white'};

}

// css animation动画

&::after{

animation: ${aniTag} 0.4s ease-in-out forwards;

}

}

`

// 全局样式

const GlobalStyle = createGlobalStyle`

*{

padding: 0;

margin: 0;

}

${MyButton}{

span{

color: rgba(255,255,255,0.6);

font-size: 12px;

line-height: 20px;

}

}

`

export default function StyleComponentsTest() {

return (

<Wrap>

<GlobalStyle />

<MyButton

bgNormalColor="gray"

bgHighlightColor="white"

normalColor='rgba(255,255,255,0.6)'

hightightColor="red"

>

<p><strong className="emphasize">styled components</strong> 入门</p>

<span>我是只小小鸟</span>

</MyButton>

</Wrap>

)

}

这个小demo有用到styled-components的关键技术点:

其本语法,见Wrap

动态样式实现,通过$

怎么写全局样式,见createGlobalStyle

怎么实现子/孙元素样式控制,见p和strong元素样式控制

怎么实现伪类,见:hover

怎么实现伪元素,见::after

怎么实现动画,见transition/keyframes

当全局样式和自带样式冲突的,怎么提高样式优先级,见span

示例代码都做了详细的注释,可以浏览代码了解详情

补充

其实通过实现上面示例基本算已经入门styled-components了,但是开发经常要考虑代码复用,那styled-components怎么实现CSS复用,我来优化一下demo

最终实现的效果如下:

Screen Recording 2025-10-31 at 15.16.54.gif

关键代码如下:

import styled, { keyframes, createGlobalStyle, css } from 'styled-components'

// 样式复用

const flexColCenter = css`

display: flex;

flex-direction: column;

justify-content: center;

align-items: center;

`

const textHighlightColor = css`

color: ${props => props.hightightColor ? props.hightightColor : 'white'};

`

const Wrap = styled.div`

width: 100%;

height: 100vh;

${flexColCenter}

gap: 20px;

`

// css帧动画

const aniTag = keyframes`

0% {

opacity: 0;

transform: translateY(-100%);

}

100% {

opacity: 1;

transform: translateY(0);

}

`

const MyButton = styled.div`

width: 400px;

height: 80px;

border:1px solid gray;

border-radius: 10px;

overflow: hidden;

${flexColCenter}

position: relative;

// 传参

background-color: ${props => props.bgNormalColor ? props.bgNormalColor : 'blue'};

transition: background-color 0.3s ease-in-out;

// 伪元素

&::after{

content: 'after';

height: 20px;

font-size: 14px;

line-height: 20px;

padding:0 5px;

border-radius: 0 0 6px 6px;

position: absolute;

top: 0;

right: 10px;

background-color: yellow;

${textHighlightColor}

opacity: 0;

}

// 子孙选择器

p {

line-height: 24px;

color: ${props => props.normalColor ? props.normalColor : 'black'};

transition: color 0.3s ease-in-out;

}

// 当有全局样式的时候,使用&&提高局部样式优先级来覆盖全局样式

&& span{

color: rgba(255,0,255,0.6);

}

// 类选择器

.emphasize{

text-decoration: underline;

}

// 伪类

&:hover{

background-color: ${props => props.bgHighlightColor ? props.bgHighlightColor : 'yellow'};

p{

${textHighlightColor}

}

// css animation动画

&::after{

animation: ${aniTag} 0.4s ease-in-out forwards;

}

}

`

const MyButton1 = styled(MyButton)`

border:4px solid gray;

border-radius: 20px;

`

// 全局样式

const GlobalStyle = createGlobalStyle`

*{

padding: 0;

margin: 0;

}

${MyButton}{

span{

color: rgba(255,255,255,0.6);

font-size: 12px;

line-height: 20px;

}

}

`

export default function StyleComponentsTest() {

return (

<Wrap>

<GlobalStyle />

<MyButton

bgNormalColor="gray"

bgHighlightColor="white"

normalColor='rgba(255,255,255,0.6)'

hightightColor="red"

>

<p><strong className="emphasize">styled components</strong> 入门</p>

<span>我是只小小鸟</span>

</MyButton>

<MyButton1

bgNormalColor="gray"

bgHighlightColor="white"

normalColor='rgba(255,255,255,0.6)'

hightightColor="red"

>

<p><strong className="emphasize">styled components</strong> 入门1</p>

<span>我是只小小鸟1</span>

</MyButton1>

</Wrap>

)

}

关键点:

样式复用,见flexColCenter/textHighlightColor

样式继承重写,见MyButton1

本来是想放一个在线测试案例的,找了很久都找到能用的React的playground,如果你想自己跑起示例来也是很简单的,初始化一个最简单的React示例项目并安装styled-components依赖,新建一个组件,把上面代码复制进去即可,推荐直接使用vite初始化项目

// 1命令行执行如下命令,再选择react即可

npm create vite@latest

或者直接clone我的测试仓库,你本地测试即可:demo仓库

小结

个人的知识和能力是有限的,本文只是个人练习styled-components的小demo,应该还有很多styled-components高

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/5/23 14:20:24

2025-2030年下一代音频编解码技术展望与产业深度研究报告2025-2030年下一代音频编解码技术展望与产业深度研究报告

2025-2030年下一代音频编解码技术展望与产业深度研究报告 1. 宏观背景与执行摘要 1.1 全球音频技术转折点&#xff1a;从压缩到重构 2025年标志着全球音频技术产业的一个关键转折点。在过去的三十年里&#xff0c;音频编解码技术的发展主线一直是“比特率战争”&#xff08;…

作者头像 李华
网站建设 2026/6/7 14:27:33

DLL注入器 Xenos 64注入器

链接&#xff1a;https://pan.quark.cn/s/b9aa8d6b1c02[玫瑰]还在为找不到安全好用的注入器发愁吗[玫瑰]还在怕自己的DLL注入被VAC吗[玫瑰]CSGO、GTA5等游戏通用[玫瑰]好用的DLL插件注入器!【软件名称】&#xff1a;Xenos64注入器

作者头像 李华
网站建设 2026/6/9 20:03:38

DeepSeek-V3.2-Exp动态注意力机制如何突破传统Transformer性能瓶颈

DeepSeek-V3.2-Exp动态注意力机制如何突破传统Transformer性能瓶颈 【免费下载链接】DeepSeek-V3.2-Exp-Base 项目地址: https://ai.gitcode.com/hf_mirrors/deepseek-ai/DeepSeek-V3.2-Exp-Base 面对超长文本处理中的注意力稀释难题&#xff0c;传统Transformer架构在…

作者头像 李华
网站建设 2026/6/9 2:23:41

光刻胶用增感剂:乙氧基/丙氧基改性吡唑啉有机物

1. 基本信息乙氧基/丙氧基改性的吡唑啉有机物是一种专门设计用于光刻胶的增感剂。其核心结构是在吡唑啉环上引入了乙氧基&#xff08;-CH₂-CH₂-O-&#xff09;和/或丙氧基&#xff08;-CH(CH₃)-CH₂-O-&#xff09;链段。这种独特的分子设计使其吸收波段通常在360-400nm之间…

作者头像 李华
网站建设 2026/6/9 19:42:00

TCN-GRU回归+特征贡献SHAP分析+新数据预测+多输出,MATLAB代码

MATLAB代码实现了一个TCN-GRU混合神经网络模型&#xff0c;用于多输出回归预测任务&#xff0c;并集成了SHAP特征重要性分析和新数据预测功能。以下是详细分析&#xff1a; 一、主要功能 TCN-GRU混合模型构建与训练&#xff1a; 结合时序卷积网络&#xff08;TCN&#xff09;和…

作者头像 李华