js類庫styled-components快速入門教程

styled-components 是什麼?

styled-components 是一個常用的 css in js 類庫。和所有同類型的類庫一樣,通過 js 賦能解決瞭原生 css 所不具備的能力,比如變量、循環、函數等。

相對於其他預處理有什麼優點?

  • 諸如 sass&less 等預處理可以解決部分 css 的局限性,但還是要學習新的語法,而且需要對其編譯,其復雜的 webpack 配置也總是讓開發者抵觸。
  • 如果有過sass&less的經驗,也能很快的切換到styled-components,因為大部分語法都類似,比如嵌套,& 繼承等, styled-componens 很好的解決瞭學習成本與開發環境問題,很適合 React 技術棧 && React Native 的項目開發。

解決瞭什麼問題?

  • className 的寫法會讓原本寫css的寫法十分難以接受
  • 如果通過導入css的方式 會導致變量泄露成為全局 需要配置webpack讓其模塊化
  • 以及上面提到的解決瞭原生 css 所不具備的能力,能夠加速項目的快速開發

官方文檔

https://styled-components.com/docs

安裝

npm install --save styled-components

編輯器智能提示

2018-06-11更新

  • webstorm需要安裝 styled-component 插件
  • vscode已支持智能提示

最基礎的使用

import styled from 'styled-components'
const Title = styled.h1`
    font-size: 1.5em;
    text-align: center;
    color: palevioletred;
`;
// 相當於  const Title = styled.h1(xx)
const Wrapper = styled.section`
    padding: 4em;
    background: papayawhip;
`;
    render () {
        return (
            <Wrapper>
                <Title>Hello styled-components</Title>
            </Wrapper>
        )
    }

此時我們可以看到控制臺中輸出瞭一個隨機的className,這是styled-components幫我們完成的. 註意: 組件名要以大些開頭 不然會被解析成普通標簽

傳遞props

const Button = styled.button`
    background: ${props => props.primary ? 'palevioletred' : 'white'};
    color: ${props => props.primary ? 'white' : 'palevioletred'};
    font-size: 1em;
    margin: 1em;
    padding: 0.25em 1em;
    border: 2px solid palevioletred;
    border-radius: 3px;
`
render(
    <div>
        <Button>Normal</Button>
        <Button primary>Primary</Button>
    </div>
);

在組件傳遞的props都可以在定義組件時獲取到,這樣就很容易實現定制某些風格組件

props高級用法

設置默認值,在未設定必須傳值的情況下我們會給一個默認值(defaultProps)

export default class ALbum extends React.Component {
    constructor (props) {
        super(props)
        this.state = {
            // 接收傳遞的值
            imgSrc: props.imgSrc
        }
    }
    render () {
        const {imgSrc} = this.state
        return (
            <Container imgSrc={imgSrc}>
            </Container>
        )
    }
}
// 在這裡是可以拿到props的 
const Container = styled.div`
    background-size: cover;
    background-image: url(${props =>  props.imgSrc});
    width: 100%;    
    height: 300px;
`
// 當然沒傳值也沒關系  我們設置默認值
Container.defaultProps = {
    imgSrc: Cover
}

塑造組件

這個非常有用 你可能會遇到一些原本就已經是組件瞭 但是你要為他添加一些樣式,這時候該怎麼辦呢 ?

// 傳遞className 在react-native 中要使用 style
const Link = ({className , children}) => (
    <a className={className}>
        {children}
    </a>
)
const StyledLink = styled(Link)`
    color: palevioletred;
`
render(
    <div>
        <Link>普通組件</Link>
        <StyledLink>有顏色嗎?</StyledLink>
    </div>
);

組件樣式繼承

const Button = styled.button`
    color: palevioletred;
    font-size: 1em;
    margin: 1em;
    padding: 0.25em 1em;
    border: 2px solid palevioletred;
    border-radius: 3px;
`;
const TomatoButton = Button.extend`
    color: tomato;
    border-color: tomato;
`;
// TomatoButton 部分樣式繼承自 Button 這種情況下不會生成兩個class

改變組件標簽

在閑的蛋疼的情況下 我們想要改變組件的標簽 比如把 button 變成 a 標簽

// 利用上面定義的 Button 組件 調用 withComponent 方法
const Link = Button.withComponent('a')

維護其他屬性

在某種情況下,我們可能需要用到第三方庫樣式,我們可以使用這個方法輕松達到

const Input = styled.input.attrs({
    // 定義靜態 props
    type: 'password',
    // 沒傳默認使用 1em
    margin: props => props.size || '1em',
    padding: props => props.size || '1em'
})`
    color: palevioletred;
    font-size: 1em;
    border: 2px solid palevioletred;
    border-radius: 3px;
    // 動態計算props
    margin: ${props => props.margin};
    padding: ${props => props.padding}
`
render ( <Input size='1em'></Input>  <Input size='2em'></Input> )

動畫

動畫會生成一個隨機類名 而不會污染到全局

import { keyframes } from 'styled-components'
// CSS 動畫
const rotate360 = keyframes`
    from {
        transform: rotate(0);
    }
    to {
        transform: rotate(360deg);
    }
`
const Rotate = Button.extend`
    animation: ${rotate360} 2s linear infinite;
`
render ( <Rotate>  💅  </Rotate> )

結語

styled-components雖然解決瞭大部分問題,增加瞭可維護性,但是破壞瞭原生體驗,時常我們需要寫更多的代碼來達到業務要求,希望未來有更好的方案,更多關於js類庫styled-components的資料請關註WalkonNet其它相關文章!

推薦閱讀: