React 練習 - 便條紙

更新於 發佈於 閱讀時間約 9 分鐘

最近正在學 React,看見課程中要做出一個類似便條紙的頁面,老師說靈感來自 Google Keep,紀錄自己的想法後丟上去。

google keep頁面

google keep頁面

課程裡做出來的:

post-it 頁面

post-it 頁面


主要是練習 React 中的 props 跟怎麼使用狀態,以下是寫出來的 component 之間的關係。

<App>
<Header />
<CreateNote />
<Note />
<Footer />
<App />

使用 useState 將表單元素轉成可控制元件

首先是新增便條紙 <CreateNote /> 的部分,要將不可控的元素轉成可控制元素(controlled elements),因為表單元素是由 DOM 管理輸入的狀態,​轉成受控元素就能允許 React 保持對輸入的控制。

步驟:

  • 使用 useState 來初始化狀態
const [note, setNote] = useState({ title: '', content: '' })
  • 建立處理事件的函數
function handleChange(e) {
const { name, value } = e.target
setNote((note) => {
return { ...note, [name]: value }
})
}
  • 將狀態綁定與函數在元素
<input
// ...
onChange={handleChange}
value={note.title}/>
<textarea
// ...
onChange={handleChange}
value={note.content}/>

這時候元素的輸入值就可以預測(因為是我們在控制的),也能進行動態更新。

提升狀態到父元件 Lift states up

使用者在新增便條紙後,接收值的元件是 <Note /> ,它不是 <CreateNote /> 的子元件。

由於 React 中資料只能由上往下流動,當父元件或其他兄弟元件需要使用同個狀態時,必須將狀態提升到上一層的公共父元件,在這裡就是 <App />

步驟:

  • 提升狀態到 App 元件
const [notes, setNotes] = useState([])

function handleAddNote(newNote) {
setNotes((notes) => [...prevNotes, newNote])
}

function handleDeleteNote(id) {
setNotes((notes) => notes.filter((note, i) => i !== id))
}
  • 用 props 傳遞給 CreateNoteNote 元件
<CreateNote onAddNote={handleAddNote} />
注意 notes 是陣列 要用 map 迭代資料給 Note
{notes.map((note, i) => {
return (
<Note
key={i}
id={i}
title={note.title}
content={note.content}
onDeleteNote={handleDeleteNote}/>
)
})}
  • 使用 props
function CreateNote({ onAddNote }) {
// ...
function handleClick(e) {
e.preventDefault()
onAddNote(note)
}
return (
// ...
<button onClick={handleClick}>Add</button>
)
}
function Note({ id, title, content, onDeleteNote }) {
return (
<div className="note">
<h2>{title}</h2>
<p>{content}</p>
<button onClick={() => onDeleteNote(id)}>DELETE</button>
</div>
)
}

現在新增便條紙後,發現就算沒有寫任何內容還是可以新增,所以回到 CreateNote 加上一個預防動作:

function handleClick(e) {
e.preventDefault()
// 新增以下這行
if (note.title === '' || note.content === '') return
// 當沒有內容時 就不再執行​
onAddNote(note)
}

使用 useEffect 將資料存在 localStorage

新增完便條紙後,如果重新整理網頁,狀態就會回到初始狀態,也就是空的陣列:

const [notes, setNotes] = useState([]) // 初始狀態為空陣列

於是就想到可以把資料存在 localStorage,讓使用者下次打開頁面時,可以再將儲存的資料渲染出來。

步驟:

  • 將上面那段改寫
const [notes, setNotes] = useState(function () {
const storageValue = localStorage.getItem('notes')
return storageValue ? JSON.parse(storageValue) : []
// 如果 localStorage 沒有資料 返回空陣列
})
  • 新增 useEffect
useEffect(
function () {
localStorage.setItem('notes', JSON.stringify(notes))
},
[notes]
// 當 notes 更新時 執行上面的函數
)

完成!

為了提高重用度 改寫成 Custom Hook

很多時候,將資料儲存到 localStorage 是常有的事,如果可以把上面那一大段程式碼寫成自定義的 Hook,未來其他專案要使用時,也能輕鬆移植過去。

最後就是要來寫 Custom Hook:

  • 程式碼寫在另一個檔案 使用 use 命名 Custom Hook
export function useLocalStorageState(initialValue, key) {}
  • 將剛剛寫的 useState、useEffect 剪下貼過來
export function useLocalStorageState(initialValue, key) {
const [value, setValue] = useState(function () {
const storageValue = localStorage.getItem(key)
return storageValue ? JSON.parse(storageValue) : initialValue
})

useEffect(
function () {
localStorage.setItem(key, JSON.stringify(value))
},
[value, key]
)
}
  • 返回陣列或物件
export function useLocalStorageState(initialValue, key) {
const [value, setValue] = useState(function () {
const storageValue = localStorage.getItem(key)
return storageValue ? JSON.parse(storageValue) : initialValue
})

useEffect(
function () {
localStorage.setItem(key, JSON.stringify(value))
},
[value, key]
)

return [value, setValue] // 返回陣列
}
  • 在元件中導入 Custom Hook
const [notes, setNotes] = useLocalStorageState([], 'notes')

這樣就大功告成啦!

重新整理也不會遺失寫好的便條紙

重新整理也不會遺失寫好的便條紙



練習 repo 頁面 ▶ Post-it - build with React

avatar-img
5會員
14內容數
本來是理科生,在被物理放棄之後成為了文科生,有時理性思考,偶爾卻會脫口出感性的字句;喜歡打字的聲音,以生活為靈感寫下過去、現在與未來。
留言0
查看全部
avatar-img
發表第一個留言支持創作者!
碎碎念 的其他內容
C2 開始終於使用到 visual studio code, 這個階段學到 JavaScript 核心概念、DOM 操作、API 串接、MVC 架構, 每天都在 coding,每週都在追進度, 用壓縮到極致的時間寫完作業, 但跟著課程內容寫程式碼,真的有學到嗎? 有,吧。 畢竟在寫最後的電影清單
這次的 Frontend Mentor challenge 是新手村最後一個挑戰 ▶ Intro component with sign up form 網頁,使用 jQuery validation plugin。
開始接觸網頁開發後,發現自己缺乏實作經驗, 於是老師推了 Frontend Mentor challenge , 免費的通常是獨立頁面,花錢則是可以挑戰多頁式網站, 初學還沒學到 javascript 或 API 也沒關係, 新手也有提供 HTML/CSS 的練習。 很適合用來增加自己寫網頁的經驗,
理解問題,然後拆解,最後解決它。 通常網頁開發的課程流是 HTML - CSS - JavaScript , 但在 AC 一開始就先淺談 JavaScript , 就是要先建立運算思維:
孔子說三十而立,我卻三十而迷失了自己, 這個世界給得價值觀太過沈重, 原本不在意的,在褪去青春之後,都成為了後悔的來源; 後悔沒有選OO系、沒有攻讀研究所、沒有超前部署自己的人生, 最後遺失了勇氣,想要往前邁進得步伐依舊跨不出舒適圈, 於是季節繼續更迭、身邊的夥伴來來去去, 好像只有我被自己困住了。
C2 開始終於使用到 visual studio code, 這個階段學到 JavaScript 核心概念、DOM 操作、API 串接、MVC 架構, 每天都在 coding,每週都在追進度, 用壓縮到極致的時間寫完作業, 但跟著課程內容寫程式碼,真的有學到嗎? 有,吧。 畢竟在寫最後的電影清單
這次的 Frontend Mentor challenge 是新手村最後一個挑戰 ▶ Intro component with sign up form 網頁,使用 jQuery validation plugin。
開始接觸網頁開發後,發現自己缺乏實作經驗, 於是老師推了 Frontend Mentor challenge , 免費的通常是獨立頁面,花錢則是可以挑戰多頁式網站, 初學還沒學到 javascript 或 API 也沒關係, 新手也有提供 HTML/CSS 的練習。 很適合用來增加自己寫網頁的經驗,
理解問題,然後拆解,最後解決它。 通常網頁開發的課程流是 HTML - CSS - JavaScript , 但在 AC 一開始就先淺談 JavaScript , 就是要先建立運算思維:
孔子說三十而立,我卻三十而迷失了自己, 這個世界給得價值觀太過沈重, 原本不在意的,在褪去青春之後,都成為了後悔的來源; 後悔沒有選OO系、沒有攻讀研究所、沒有超前部署自己的人生, 最後遺失了勇氣,想要往前邁進得步伐依舊跨不出舒適圈, 於是季節繼續更迭、身邊的夥伴來來去去, 好像只有我被自己困住了。
你可能也想看
Google News 追蹤
Thumbnail
隨著理財資訊的普及,越來越多台灣人不再將資產侷限於台股,而是將視野拓展到國際市場。特別是美國市場,其豐富的理財選擇,讓不少人開始思考將資金配置於海外市場的可能性。 然而,要參與美國市場並不只是盲目跟隨標的這麼簡單,而是需要策略和方式,尤其對新手而言,除了選股以外還會遇到語言、開戶流程、Ap
Thumbnail
嘿,大家新年快樂~ 新年大家都在做什麼呢? 跨年夜的我趕工製作某個外包設計案,在工作告一段落時趕上倒數。 然後和兩個小孩過了一個忙亂的元旦。在深夜時刻,看到朋友傳來的解籤網站,興致勃勃熬夜體驗了一下,覺得非常好玩,或許有人玩過了,但還是想寫上來分享紀錄一下~
Thumbnail
第一次使用這平台 處於摸索狀態 透過文字希望可以好好紀錄自己的生活
Thumbnail
在程式任何地方都能修改各種react組件狀態的做法分享
Thumbnail
#誠實書寫練習 #Day112 Q.本日書寫主題: 回想本週看見的人事物,找尋有印象或有感覺的事情並寫下來。 連結並記錄下3個相關過往的記憶抽屜中的小事。 ✍️這一週行前準備,以往都是跟團,要不就是親友幫忙處理。 而這一回有幾件事情需要我一一處理,必須要保持頭腦清晰,要很仔細填寫資料,所以感到有
IG重新整理了一下,試著做圖文,這篇談分享。
Thumbnail
【Paged Notes】是一款簡單、清晰、直觀的記事軟體,提供自訂記事分類、清單記事和備忘記事等功能。同時支援標示記事重要性和自訂記事底色,並可自動連結網址、電話和E-mail。未來將持續新增相關模組與功能。
Thumbnail
今天紀錄的是因應工作需要而學習的小技能。 我想達到的目標:當欄位裡面出現「寄出」這個關鍵字的時候整欄變色,以辨別該項目的狀態。
Thumbnail
隨著理財資訊的普及,越來越多台灣人不再將資產侷限於台股,而是將視野拓展到國際市場。特別是美國市場,其豐富的理財選擇,讓不少人開始思考將資金配置於海外市場的可能性。 然而,要參與美國市場並不只是盲目跟隨標的這麼簡單,而是需要策略和方式,尤其對新手而言,除了選股以外還會遇到語言、開戶流程、Ap
Thumbnail
嘿,大家新年快樂~ 新年大家都在做什麼呢? 跨年夜的我趕工製作某個外包設計案,在工作告一段落時趕上倒數。 然後和兩個小孩過了一個忙亂的元旦。在深夜時刻,看到朋友傳來的解籤網站,興致勃勃熬夜體驗了一下,覺得非常好玩,或許有人玩過了,但還是想寫上來分享紀錄一下~
Thumbnail
第一次使用這平台 處於摸索狀態 透過文字希望可以好好紀錄自己的生活
Thumbnail
在程式任何地方都能修改各種react組件狀態的做法分享
Thumbnail
#誠實書寫練習 #Day112 Q.本日書寫主題: 回想本週看見的人事物,找尋有印象或有感覺的事情並寫下來。 連結並記錄下3個相關過往的記憶抽屜中的小事。 ✍️這一週行前準備,以往都是跟團,要不就是親友幫忙處理。 而這一回有幾件事情需要我一一處理,必須要保持頭腦清晰,要很仔細填寫資料,所以感到有
IG重新整理了一下,試著做圖文,這篇談分享。
Thumbnail
【Paged Notes】是一款簡單、清晰、直觀的記事軟體,提供自訂記事分類、清單記事和備忘記事等功能。同時支援標示記事重要性和自訂記事底色,並可自動連結網址、電話和E-mail。未來將持續新增相關模組與功能。
Thumbnail
今天紀錄的是因應工作需要而學習的小技能。 我想達到的目標:當欄位裡面出現「寄出」這個關鍵字的時候整欄變色,以辨別該項目的狀態。