随着 React 项目日益复杂化、繁重化,React 中 css 面临很多问题,比如样式类名全局污染、命名混乱、样式覆盖等。这时, css 模块化就显得格外重要。
css 模块化的几个重要作用:
!important
或者 行内样式 来解决React 使用 css 模块化的思路:
css in js
用 js 对象方式写 css ,然后作为 style
方式赋予给 React 组件的 DOM 元素,这种写法将不需要 .css
.less
.scss
等文件,取而代之的是每一个组件都有一个写对应样式的 js
文件css Modules ,使得项目中可以像加载 js 模块一样加载 css ,本质上通过一定自定义的命名规则生成唯一性的 css 类名,从根本上解决 css 全局污染,样式覆盖的问题。对于 css modules 的配置,推荐使用 css-loader,它对 CSS Modules 的支持最好,而且很容易使用。
css-loader 配置:
{
test: /\.css$/,
use: [
'css-loader?modules'
]
}
css 文件:
.title {
color: red;
}
js 文件:
import style from './index.css'
const App = () => {
return (
<div className={style.title}>Hello World</div>
)
}
{
test: /\.css$/,
use: [
{
loader: 'css-loader',
options: {
modules: {
localIdentName: '[path][name]__[local]--[hash:base64:5]'
}
}
}
]
}
[path][name]__[local]
-> 开发环境,便于调试,可以直接找到此类名对应的文件[hash:base64:5]
-> 生产环境,便于生产环境压缩类名一旦经过 css modules 处理的 css 文件类名 ,再引用的时候就已经无效了。因为声明的类名,比如如上的 .text
已经被处理成了哈希形式。
那么怎么样快速引用声明好的全局类名呢?CSS Modules 允许使用 :global(.className)
的语法,声明一个全局类名。凡是这样声明的 class
,都不会被编译成哈希字符串。
.title {
color: red;
}
:global(.title_bg) {
background-color: pink;
}
import style from './index.css'
const App = () => {
return (
<div className={`${style.title} title_bg`}>Hello World</div>
)
}
CSS Modules 还提供一种显式的局部作用域语法: local(.title)
等同于 .title
:
.title {
color: red;
}
/* 等同于 */
:local(.title) {
color: red;
}
CSS Modules 提供了一种 composes
组合方式,实现对样式的复用:
.base {
color: blue;
}
.title {
composes: base; /* 继承 base 样式 */
font-size: 20px;
}
composes
可以将多个 class
类名添加到元素中。composes
还有一个更灵活的方法,支持动态引入别的模块下的类名。
style1.css
:
.base {
color: red;
}
style2.css
:
.title {
composes: base from './style1.css';
font-size: 20px;
}
less webpack 配置
{
test: /\.less$/,
use: [
{
loader: 'css-loader',
options: {
modules: {
localIdentName: '[path][name---[local]---[hash:base64:5]'
}
}
},
'less-loader'
]
}
index.less
:.text {
color: blue;
background-color: pink;
}
index.js
:import style from './index.less'
const App = () => {
return (
<div className={style.text}>Hello World</div>
)
}
正常情况下,React 项目可能在使用 css 处理样式之外,还会使用 scss 或者 less 预处理。
.css
文件 ,不需要做 CSS Modules 处理,这样就不需要写 :global
等繁琐语法这样就会让 React 项目更加灵活的处理 CSS 模块化。
import React from 'react';
import style from './index.less';
export default () => {
return <div>
<button className='search-btn'>Search</button>
<div className={style.text}>Less + CSS Modules</div>
</div>
}
CSS Modules 可以配合 classNames 库 实现更灵活的动态添加类名。
index.less
.base {}
.dark {
background-color: #000;
color: #fff;
}
.light {
background-color: #fff;
color: #000;
}
index.jsx
import cx from 'classnames';
import style from './index.less';
export default function Index() {
const [theme, setTheme] = useState('dark');
return <div>
<button
className={cx(style.base, style[theme])}
onClick={() => setTheme(theme === 'dark' ? 'light' : 'dark')}
>
Change Theme
</button>
</div>;
}
css .a{ .b{} }
或者重叠 css .a .b {}
CSS IN JS 相比 CSS Modules 更加简单, CSS IN JS 放弃 css ,用 js 对象形式直接写 style 。
index.jsx
import React from 'react';
import Style from './style';
export default function Index() {
return <div style={Style.boxStyle}>
<span style={Style.textStyle}>CSS in JS</span>
</div>;
}
style.js
const boxStyle = {
backgroundColor: '#000',
};
const textStyle = {
color: '#fff',
};
export default {
boxStyle,
textStyle,
};
由于 CSS IN JS 本质上就是运用 js 中对象形式保存样式, 所以 js 对象的操作方法都可以灵活的用在 CSS IN JS 上。
拓展运算符实现样式继承
const baseStyle = {
color: '#fff',
fontSize: '20px',
};
const containerStyle = {
...baseStyle,
backgroundColor: '#000',
};
动态添加样式变得更加灵活
const dark = {
backgroundColor: '#000',
};
const light = {
backgroundColor: '#fff',
};
import React from 'react';
import Style from './style';
const Index = () => {
return <div style={theme === 'light' ? Style.dark : Style.light }>
CSS in JS
</div>;
}
CSS IN JS 也可以由一些第三方库支持,如 styled-components, 可以把写好的 css 样式注入到组件中,项目中应用的已经是含有样式的组件。
基础用法
import React from 'react';
import styled from 'styled-components';
/* 给 button 标签添加样式,形成 Button React 组件 */
const Button = styled.button`
background: #6a8bad;
color: #fff;
border: none;
padding: 10px 20px;
border-radius: 5px;
font-size: 16px;
cursor: pointer;
`;
export default function Index() {
return <Button>Styled Components</Button>;
}
基于 props 动态添加样式
import React from 'react';
import styled from 'styled-components';
const Button = styled.button`
background: #6a8bad;
color: #fff;
border: none;
padding: 10px 20px;
border-radius: 5px;
font-size: 16px;
cursor: pointer;
/* 通过 props 来动态添加样式 */
${props => props.primary && `
background: #fff;
color: #6a8bad;
`}
`;
export default function Index() {
return <div>
<Button>Styled Components</Button>
<Button primary>Styled Components</Button>
</div>;
}
继承样式
import React from 'react';
import styled from 'styled-components';
const Button = styled.button`
background: #6a8bad;
color: #fff;
border: none;
padding: 10px 20px;
border-radius: 5px;
font-size: 16px;
cursor: pointer;
`;
const PrimaryButton = styled(Button)`
background: #fff;
color: #6a8bad;
`;
export default function Index() {
return <div>
<Button>Styled Components</Button>
<PrimaryButton>Styled Components</PrimaryButton>
</div>;
}
style.js
样式文件,以及快捷引入文件中的样式常量扫码关注腾讯云开发者
领取腾讯云代金券
Copyright © 2013 - 2025 Tencent Cloud. All Rights Reserved. 腾讯云 版权所有
深圳市腾讯计算机系统有限公司 ICP备案/许可证号:粤B2-20090059 深公网安备号 44030502008569
腾讯云计算(北京)有限责任公司 京ICP证150476号 | 京ICP备11018762号 | 京公网安备号11010802020287
Copyright © 2013 - 2025 Tencent Cloud.
All Rights Reserved. 腾讯云 版权所有