React.js Workshop
淺入淺出 React.js
Denny Ku
data:image/s3,"s3://crabby-images/ec9f4/ec9f4c23dd46cd071210d329ef52b44fe05c4f75" alt=""
我是誰
Denny Ku
- Sudo:Front-end developer
- 和沛科技:Front-end engineer
- github: abalone0204
data:image/s3,"s3://crabby-images/ec9f4/ec9f4c23dd46cd071210d329ef52b44fe05c4f75" alt=""
今天要講什麼
- 為什麼 React 重要
- 基本的 React 教學
- 一般人懂的 React 教學
data:image/s3,"s3://crabby-images/ec9f4/ec9f4c23dd46cd071210d329ef52b44fe05c4f75" alt=""
Outline
- 概念
- 簡介 MVC、MVVM
- What is React.js
- Pros & Cons
- 環境建置
- React basic
- Data flow
- State, Props, Life-cycle method
- React intermediate
- Router
- Redux
- Redux Saga
- CSS Modules
data:image/s3,"s3://crabby-images/6b26d/6b26d3b3cd5fdf10d7ded1b1eff35730fce06e12" alt=""
基本上
就是我的工作流程
data:image/s3,"s3://crabby-images/6b26d/6b26d3b3cd5fdf10d7ded1b1eff35730fce06e12" alt=""
概念
前言
- 每個人的定義的 MV* 可能都不一樣
- 去爭吵什麼是 Model 、什麼是 Controller 沒有意義
- 真正要在意的是這些想法在軟體架構上的意義
data:image/s3,"s3://crabby-images/ec9f4/ec9f4c23dd46cd071210d329ef52b44fe05c4f75" alt=""
前言
- 可變與不可預測的狀態是萬惡之源
- 但需求永遠都是可變、不可預測的
所以我們應該消滅製造需求的人-
所以我們應該要在讓可變的狀態變得好預測
- Easy to reason about
- 好讀
- 符合常理
data:image/s3,"s3://crabby-images/ec9f4/ec9f4c23dd46cd071210d329ef52b44fe05c4f75" alt=""
MVC 是什麼
data:image/s3,"s3://crabby-images/ecc00/ecc00023531c3743df951ed941ba5d92bbe946c9" alt=""
data:image/s3,"s3://crabby-images/2d09b/2d09b5a0a83737e3d71df42dad4a27f93b4a6d1b" alt=""
MVVM 是什麼
多了一個 View Model
data:image/s3,"s3://crabby-images/ecc00/ecc00023531c3743df951ed941ba5d92bbe946c9" alt=""
data:image/s3,"s3://crabby-images/d0b0f/d0b0fdd5aac13acf47cca6278cf05a097a95fff8" alt=""
MVVM 是什麼
- ViewModel 不需要瞭解 View 的實作細節
- View 不需要知道 ViewModel 的實作細節,但是會聲明自己需要哪些數據、而且知道如何渲染畫面
data:image/s3,"s3://crabby-images/ecc00/ecc00023531c3743df951ed941ba5d92bbe946c9" alt=""
data:image/s3,"s3://crabby-images/d0b0f/d0b0fdd5aac13acf47cca6278cf05a097a95fff8" alt=""
所以
- 我們需要一個 ViewModel 來做 action binding
- 需要一個 state less 的 View
data:image/s3,"s3://crabby-images/ecc00/ecc00023531c3743df951ed941ba5d92bbe946c9" alt=""
建置環境
data:image/s3,"s3://crabby-images/ecc00/ecc00023531c3743df951ed941ba5d92bbe946c9" alt=""
建一個 React App
data:image/s3,"s3://crabby-images/c3417/c3417a52c2c01670841f9df626124fd16e3e0ab5" alt=""
React Create App
$> npm install -g react-create-app
$> create-react-app hello-world
$> cd hello-world && npm install && npm start
React basic
data:image/s3,"s3://crabby-images/ecc00/ecc00023531c3743df951ed941ba5d92bbe946c9" alt=""
f(Data, Template) = View
- 一個 Stateless 的 View 是什麼?
- Function 的特性:
- 好預測
Recap
data:image/s3,"s3://crabby-images/ecc00/ecc00023531c3743df951ed941ba5d92bbe946c9" alt=""
- ViewModel 不需要瞭解 View 的實作細節
- View 不需要知道 ViewModel 的實作細節,但是會聲明自己需要哪些數據、而且知道如何渲染畫面
data:image/s3,"s3://crabby-images/d0b0f/d0b0fdd5aac13acf47cca6278cf05a097a95fff8" alt=""
Thinking in React
data:image/s3,"s3://crabby-images/ecc00/ecc00023531c3743df951ed941ba5d92bbe946c9" alt=""
- 如果你是個前端工程師,你會怎麼拆解這個畫面?
data:image/s3,"s3://crabby-images/7518b/7518b98b3581428a5bcb792cfe2d6702991627aa" alt=""
data:image/s3,"s3://crabby-images/b418c/b418c6272a321c362fb836919576cff8a813d8f7" alt=""
關鍵的概念:hierarchy
恰巧 HTML 最大的優點就是
它是個 markup languate
它能夠很好的描述各種 hierarchy
data:image/s3,"s3://crabby-images/ecc00/ecc00023531c3743df951ed941ba5d92bbe946c9" alt=""
Component base
data:image/s3,"s3://crabby-images/ecc00/ecc00023531c3743df951ed941ba5d92bbe946c9" alt=""
- 一次只要關注在一個小點上
- 決定這個 View 需要什麼樣的資料
data:image/s3,"s3://crabby-images/b418c/b418c6272a321c362fb836919576cff8a813d8f7" alt=""
Web Component
data:image/s3,"s3://crabby-images/ecc00/ecc00023531c3743df951ed941ba5d92bbe946c9" alt=""
- HTML + JavaScript + CSS
data:image/s3,"s3://crabby-images/b418c/b418c6272a321c362fb836919576cff8a813d8f7" alt=""
<FilterableProductTable>
<SearchBar />
<ProductTable>
<ProductCategoryRow/>
<ProductRow/>
</ProductTable>
</FilterableProductTable>
JSX
data:image/s3,"s3://crabby-images/ecc00/ecc00023531c3743df951ed941ba5d92bbe946c9" alt=""
- HTML + JavaScript + CSS
data:image/s3,"s3://crabby-images/b418c/b418c6272a321c362fb836919576cff8a813d8f7" alt=""
<FilterableProductTable>
<SearchBar />
<ProductTable>
<ProductCategoryRow/>
<ProductRow/>
</ProductTable>
</FilterableProductTable>
Hello world
data:image/s3,"s3://crabby-images/ecc00/ecc00023531c3743df951ed941ba5d92bbe946c9" alt=""
// /static/js/bundle.js
import ReactDOM from 'react-dom';
ReactDOM.render(
<h1>Hello, world!</h1>,
document.getElementById('example')
);
<!doctype html>
<html lang="en">
<body>
<div id="example"></div>
</body>
<script type="text/javascript" src="/static/js/bundle.js"></script>
</html>
data:image/s3,"s3://crabby-images/5fd18/5fd183dc84620e15684463e575a143423495fc61" alt=""
Hello world
data:image/s3,"s3://crabby-images/ecc00/ecc00023531c3743df951ed941ba5d92bbe946c9" alt=""
// /static/js/bundle.js
import ReactDOM from 'react-dom';
import Hello from 'components/Hello';
ReactDOM.render(
<Hello />,
document.getElementById('example')
);
<!doctype html>
<html lang="en">
<body>
<div id="example"></div>
</body>
<script type="text/javascript" src="/static/js/bundle.js"></script>
</html>
data:image/s3,"s3://crabby-images/5fd18/5fd183dc84620e15684463e575a143423495fc61" alt=""
const Hello = () => (
<h1>Hello, world!</h1>
);
export default Hello;
What is View
data:image/s3,"s3://crabby-images/ecc00/ecc00023531c3743df951ed941ba5d92bbe946c9" alt=""
f (Data, Template) -> View
const Hello = () => (
<h1>Hello, world!</h1>
);
export default Hello;
const Hello = ({text}) => (
<h1>{text}</h1>
);
export default Hello;
f () -> View
React 中的 Props 跟 State
data:image/s3,"s3://crabby-images/ecc00/ecc00023531c3743df951ed941ba5d92bbe946c9" alt=""
// CSS 待會再說
<Hello text={"yo"}/>
const Hello = ({text}) => (
<h1>{text}</h1>
);
export default Hello;
這就是 Props
那 State 呢
data:image/s3,"s3://crabby-images/ecc00/ecc00023531c3743df951ed941ba5d92bbe946c9" alt=""
<div className="App">
<h1>Hello world</h1>
<input type="text" ref='input'/>
<button onClick={this.clickHandler}>
send
</button>
<TodoList todos={todos} />
</div>
clickHandler = (e) => {
const {
value,
} = this.refs.input
const {
todos,
} = this.state;
this.setState({
todos: todos.concat(value)
}, ()=> {
this.refs.input.value = '';
})
}
Live demo
State 的壞處
- 如果每一個小 component 都有自己的 state
data:image/s3,"s3://crabby-images/6b26d/6b26d3b3cd5fdf10d7ded1b1eff35730fce06e12" alt=""
State
State
State
State
gan gan gan
data:image/s3,"s3://crabby-images/cfe22/cfe2275c8491130e728e45e235a6d7e649ae50fe" alt=""
Make State great again
data:image/s3,"s3://crabby-images/ecc00/ecc00023531c3743df951ed941ba5d92bbe946c9" alt=""
State
User action
還可以更好
這樣做的結果是會得到一個過度複雜的 Component
data:image/s3,"s3://crabby-images/ecc00/ecc00023531c3743df951ed941ba5d92bbe946c9" alt=""
data:image/s3,"s3://crabby-images/8d154/8d154ef817eee078ab761501ad4df28c291a806b" alt=""
React intermediate
data:image/s3,"s3://crabby-images/ecc00/ecc00023531c3743df951ed941ba5d92bbe946c9" alt=""
- 這裡會介紹這些東西在做什麼以及該怎樣學
Redux
data:image/s3,"s3://crabby-images/6b26d/6b26d3b3cd5fdf10d7ded1b1eff35730fce06e12" alt=""
data:image/s3,"s3://crabby-images/b8386/b838636445057b1c0e50affc8e91b196212eb6fc" alt=""
Redux
data:image/s3,"s3://crabby-images/6b26d/6b26d3b3cd5fdf10d7ded1b1eff35730fce06e12" alt=""
- 優點:
- 沒有 local state 了
- functional way
- 好讀、好寫、好測試
- 會需要寫一些 boiler plate 的 code
- 不足的地方:
- 對於 Async 的 action 處理還不夠好
Redux - Async?
data:image/s3,"s3://crabby-images/6b26d/6b26d3b3cd5fdf10d7ded1b1eff35730fce06e12" alt=""
送出後不會馬上做完的事情
data:image/s3,"s3://crabby-images/f0ac7/f0ac7e804e1bb83134e22677c8956772fb77e42e" alt=""
Redux - 學習資源
data:image/s3,"s3://crabby-images/6b26d/6b26d3b3cd5fdf10d7ded1b1eff35730fce06e12" alt=""
redux-saga
data:image/s3,"s3://crabby-images/6b26d/6b26d3b3cd5fdf10d7ded1b1eff35730fce06e12" alt=""
處理 Side effect、Async action 的 Middle ware
redux-saga - saga 是什麼
data:image/s3,"s3://crabby-images/6b26d/6b26d3b3cd5fdf10d7ded1b1eff35730fce06e12" alt=""
data:image/s3,"s3://crabby-images/9e536/9e536b964c73f5066983f5e5a732afc29906a6d1" alt=""
redux-saga - saga 是什麼
data:image/s3,"s3://crabby-images/6b26d/6b26d3b3cd5fdf10d7ded1b1eff35730fce06e12" alt=""
data:image/s3,"s3://crabby-images/776f8/776f89b9a94608ee5b050ac203a6af135deb8cae" alt=""
redux-saga - 學習資源
data:image/s3,"s3://crabby-images/6b26d/6b26d3b3cd5fdf10d7ded1b1eff35730fce06e12" alt=""
react-router
data:image/s3,"s3://crabby-images/6b26d/6b26d3b3cd5fdf10d7ded1b1eff35730fce06e12" alt=""
Single Page Application 的 Router 問題
關鍵概念:
If route is nested , then component should be nested.
CSS Modules
data:image/s3,"s3://crabby-images/6b26d/6b26d3b3cd5fdf10d7ded1b1eff35730fce06e12" alt=""
- local scope
- global scope
- composes
- 官方教學
CSS Modules
data:image/s3,"s3://crabby-images/6b26d/6b26d3b3cd5fdf10d7ded1b1eff35730fce06e12" alt=""
data:image/s3,"s3://crabby-images/4f793/4f793f7bc8851574a47205949d46b86fd6afc83c" alt=""
import cssModules from 'react-css-modules';
import styles from './styles.css';
/* eslint-disable global-require*/
const Loading = () => (
<div styleName="container">
<img
styleName="loading-icon"
src={require('../../../assets/components/loading-icon.png')} alt=""
/>
</div>
);
/* eslint-enable */
export default cssModules(Loading, styles);
@keyframes spin { 100% {transform: rotate(360deg); }}
.container {
position: relative;
padding: 30px;
text-align: center;
}
.loading-icon {
animation: spin 0.7s linear infinite;
}
CSS Modules
data:image/s3,"s3://crabby-images/6b26d/6b26d3b3cd5fdf10d7ded1b1eff35730fce06e12" alt=""
- 設置簡單
- 沒有 global 污染問題
- no more !important
- 符合組件的開發思想
CSS Modules - 結論
data:image/s3,"s3://crabby-images/6b26d/6b26d3b3cd5fdf10d7ded1b1eff35730fce06e12" alt=""
- 時間不夠了,總之讚讚讚
結論
data:image/s3,"s3://crabby-images/6b26d/6b26d3b3cd5fdf10d7ded1b1eff35730fce06e12" alt=""
- React 可能會收掉
- 但組件化的思想不會
- 可以了解底下怎麼做到的
- 但,越少需要寫 config 的東西最好
- 你可以 follow vuejs, cyclejs 等等 library
- 更好一點也可以 follow Reactive programming 的想法
- 也可以 follow 或我的網誌
結論
data:image/s3,"s3://crabby-images/6b26d/6b26d3b3cd5fdf10d7ded1b1eff35730fce06e12" alt=""
-
你使用的語言被人嘲笑
職業被藐視
解決問題的努力被視為噱頭
自詡不弄髒手的人輕視你面對的問題
執行引擎到處是實作者從未重視過的 Bug
所有跨平台的承諾都在增加你要解決的問題
UX認為要的效果很簡單
QA給你的回報難對上問題
最後,報酬通常低於平均水準你說,你想成為前端工程師
-
參考資料
data:image/s3,"s3://crabby-images/ecc00/ecc00023531c3743df951ed941ba5d92bbe946c9" alt=""
淺入淺出我流 React.js
By Denny Ku
淺入淺出我流 React.js
- 2,581