class + typescript 用法
現在大多的React新專案都改成用hooks了,但還是有一些老舊專案會遇到,這邊會來記錄一下用法。
生命週期
- 當這個組件安裝到畫面上的時候
1 2 3
| componentDidMount() { console.log('mount') }
|
- 當這個組件要被更新的時候
1 2 3
| componentDidUpdate() { console.log('update', this.state) }
|
- 當這個組件要解除安裝的時候
1 2 3
| componentWillUnmount() { console.log('btn unmount') }
|
App.tsx
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73
| import React from 'react'
type BtnProps = { clickHandler: () => void } type BtnState = {}
class Btn extends React.Component<BtnProps, BtnState> { constructor(props: BtnProps) { super(props) }
componentWillUnmount() { console.log('btn unmount') }
render(): React.ReactNode { return <button onClick={this.props.clickHandler}>+1</button> } }
type AppProps = {} type AppState = { count: number }
class App extends React.Component<AppProps, AppState> { constructor(props: AppProps) { super(props) this.state = { count: 0 }
this.countClickHandler = this.countClickHandler.bind(this) }
countClickHandler() { console.log('this', this) this.setState((prevState) => { return { count: prevState.count + 1 } }) }
componentDidMount() { console.log('mount') }
componentDidUpdate() { console.log('update', this.state) }
render(): React.ReactNode { return <> <h1>Count: {this.state.count}</h1> {this.state.count === 5 ? null : <Btn clickHandler={this.countClickHandler} />} </> } }
export default App
|
React Hook 筆記
useState
冒泡事件範例
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19
| const Parent: React.FC = () => { let [count, setCount] = useState(0); return ( <div onClick={() => setCount(prev => prev + 1)}> Parent clicked {count} times <Child /> </div> ); }
const Child: React.FC = () => { let [count, setCount] = useState(0); return ( <button onClick={() => setCount(count + 1)}> Child clicked {count} times </button> ); }
|
callback Function 沿用前一次狀態
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26
| const Counter: React.FC = () => { const [counter, setCounter] = useState(0);
const handleClick = () => { function callbackFunc (prev: number) { return prev + 1 } setCounter(callbackFunc); setCounter(callbackFunc); }
return ( <div className='App'> <h1>Counter 組件</h1> <div> counter: {counter} </div> <br/> <button onClick={handleClick}>Click me</button> </div> ); }
export default Counter
|
useEffect
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31
| useEffect(() => { console.log('hello world') }, [])
let counter = 1; useEffect(() => { console.log('hello world') }, [counter])
useEffect(() => { console.log('hello world') }, [counter, num])
let interval : NodeJS.Timeout | null = null;
useEffect(() => { interval = setInterval(() => { console.log('hello world'); }, 1000)
return ()=>{ if (interval !== null) { clearInterval(interval); } } }, [counter])
|
遠端api抓資料 / 自定義Hook
useRef
- 記憶當下的狀態current,但不會觸發render,useState則會
1 2
| const sumRef = useRef(0); console.log(sumRef.current);
|
1 2 3 4 5 6 7 8 9 10 11
| const App: React.FC = () => { const h2Ref = useRef<HTMLHeadingElement>(null);
useEffect(() => { console.log(h2Ref.current); }, [])
return (<h2 ref={h2Ref}>title</h2>) }
export default App
|
useContext
定義 context
ModalContext.tsx
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22
| import React, { createContext, useState, useContext } from 'react';
const defaultValue = { isShow: false }
const ModalContext = createContext(defaultValue)
export const ModalProvider: React.FC = ({ children }) => { const [isShow, setIsShow] = useState(true) return <> <ModalContext.Provider value={{isShow}}> {children} </ModalContext.Provider> </> }
export const useModalContext = () => { return useContext(ModalContext) }
|
App.tsx
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18
| import React, { useState } from "react" import { ModalProvider, useModalContext } from './context/ModalContext'
const Box: React.FC = () => { const data = useModalContext() console.log(data.isShow); return <div>Box</div> }
const App: React.FC = () => { return (<ModalProvider> <p>hello world</p> <Box /> </ModalProvider>) }
export default App
|
useMemo & useCallback & memo
如果使用useEffect的dependency是物件類型,每次render物件的記憶體位置不同,所以會被視為不同值。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32
| import React, { useEffect, useState, useMemo, useCallback } from "react"
const Box: React.FC = memo({num}) => { return <div>Box {num}</div> }
const App: React.FC = () => { const [num, setNum] = useState(0); const [value, setValue] = useState(false) const memoObj = useMemo(() => { const obj = {title: 'abc', data: {}}; return obj; }, [])
const memoFunc = useCallback(function() { console.log('memoFunc') }, [])
useEffect(() => { console.log('useEffect callback') }, [memoObj, memoFunc1])
return <> <Box num={num} /> //若沒有用memo記憶這個元件,每次setValue會重新render到子元件 <button onClick={() => {setValue(!value)}}> render </button> </> }
export default App
|