หัดเขียน frontend เลียนแบบเว็บ uniswap ด้วย React


สวัสดีครับ ในตอนนี้จะมาหัดเขียน frontend โดยใช้ React + CSS Modules เพื่อให้หน้าตาออกมาเหมือนกับ uniswap โดยในตอนนี้จะเขียนแต่ ui ส่วนของการแลกเปลี่ยนเหรียญ
uniswap ผมมองว่า ui ออกแบบมาสวย เรียบง่ายดีและก็มีเว็บอื่น ๆ ที่ผมเข้าบ่อย ๆ มีหน้าตาก็คล้ายกันมากก็เลยอยากรู้ว่าเขียนยังไงจึงเป็นที่มาของบล็อกนี้ครับ
เครื่องมือที่ใช้
- nodejs
- React
มาเริ่มกันเลย
npx create-react-app frontend
เมื่อสร้างโปรเจคเสร็จแล้ว run ด้วยคำสั่ง
cd frontendyarn start
สร้างไฟล์ใหม่ src/styles.module.css ไฟล์นี้เราจะเขียน css modules กันจากนั้นก็ import เพื่อเรียกให้งานใน App.js
import React, { useEffect, useState } from "react";import styles from "./App2.module.css";...
ข้อดีของการใช้งาน css modules ในมุมมองของผมคือการตั้งชื่อ classname ซ้ำได้แต่ก็ไม่ควรตั้งซ้ำกันอยู่ดีจะไม่ได้สับสน ปัญหาการเขียน css แบบเดิมคือเมื่อโปรเจคใหญ่ขึ้นจะมีไฟล์เยอะขึ้นตาม ต้องสร้างไฟล์ css เยอะถ้าโปรเจคนั้นไม่ได้เขียนแค่คนเดียวอีกจะรู้ได้ยังไงว่าคนอื่นจะตั้งชื่อ classname ซ้ำกับเราหรือเปล่า จากประสบการณ์ที่ผมได้เขียนมาก็เป็นแบบนี้จริงๆ เพราะตอนแรกผมเขียนแบบเดิมทำให้จะแก้ ui ยากมากไม่รู้ว่ามันใช้ classname จากไฟล์ไหนกันแน่ จะปรับให้เป็นแบบ css modules ก็สายไปแล้วเพราะใน css modules มันใช้ชื่อด้วยตัว ( - ) ขีดแดท(Dashes)แบบเดิมไม่ได้ ต้องเปลี่ยนชื่อ classname ทั้งหมด
หลังจากที่มีประสบการณ์แบบนั้นแล้ว ผมจึงลองหัดเขียน css แบบใหม่ดูบ้างเช่น css modules โดยจะใช้ 1 ไฟล์.js ต่อ 1 ไฟล์.module.css ไปเลยง่ายดี แยกแต่ละไฟล์แต่ละหน้าออกจากกันเลย
* { margin: 0; padding: 0; box-sizing: border-box;}.modal { position: relative; max-width: 480px; width: 100%; background: rgb(255, 255, 255); box-shadow: rgb(0 0 0 / 1%) 0px 0px 1px, rgb(0 0 0 / 4%) 0px 4px 8px, rgb(0 0 0 / 4%) 0px 16px 24px, rgb(0 0 0 / 1%) 0px 24px 32px; border-radius: 24px; margin-top: 1rem;
margin-left: auto; margin-right: auto;}.modal__header { padding: 1rem 1.25rem 0.5rem; width: 100%; color: rgb(86, 90, 105);}.modal__header__flex { display: flex; justify-content: space-between; align-items: center;}.modal__header__flex__item { width: fit-content;}.modal__body { /* background: rgb(254, 2, 9); */ /* height: 200px; */
position: relative; padding: 8px;}.modal__body__flex { display: grid; grid-auto-rows: auto; /* row-gap: 8px; */}.modal__body__flex__item { border-radius: 20px; border: 1px solid rgb(237, 238, 242); background: rgb(247, 248, 250); width: initial;}.modal__body__flex__item__topflex { display: flex; justify-content: space-between; align-items: center;
line-height: 1rem; padding: 1rem 1rem 0.75rem;}.modal__body__flex__item__topflex__item {}.modal__body__flex__item__topflex input { color: rgb(0, 0, 0); width: 0px; position: relative; font-weight: 500; outline: none; border: none; flex: 1 1 auto; background: rgb(247, 248, 250); font-size: 24px; white-space: nowrap; overflow: hidden; text-overflow: ellipsis; padding: 0px; appearance: textfield; text-align: right;}.modal__body__flex__item__bottomflex { display: flex; justify-content: space-between; align-items: center;
line-height: 1rem; padding: 0px 1rem 1rem;}.modal__body__flex__item__bottomflex__item {}.modal__footer { /* background: rgb(254, 127, 206); */ display: grid; grid-auto-rows: auto; row-gap: 8px;
padding: 8px; border-bottom-left-radius: 20px; border-bottom-right-radius: 20px;}
.swap__button { padding: 16px; width: 100%; font-weight: 500; text-align: center; border-radius: 20px; outline: none; border: 1px solid transparent; /* color: white; */ text-decoration: none; display: flex; -webkit-box-pack: center; justify-content: center; flex-wrap: nowrap; -webkit-box-align: center; align-items: center; /* cursor: pointer; */ position: relative; z-index: 1; will-change: transform; transition: transform 450ms ease 0s; transform: perspective(1px) translateZ(0px);
background: rgb(237, 238, 242); color: rgb(86, 90, 105); box-shadow: none;}.swap__button__price_impact_too_high { border: 1px solid rgb(218, 45, 43); opacity: 0.5; background: rgb(218, 45, 43); color: white;
cursor: auto;}.swap__button__connect_wallet { opacity: 0.5; background: rgb(253, 234, 241); color: rgb(213, 0, 102);
cursor: auto;}.swap__button div { box-sizing: border-box; margin: 0px; min-width: 0px; font-size: 20px; font-weight: 500;}
.btn { background: rgb(255, 255, 255); display: flex; align-items: center;
border: none; border-radius: 20px; height: 2.4rem; width: initial; padding: 0px 8px; margin-right: 12px;
font-size: 24px; font-weight: 500; box-shadow: rgb(0 0 0 / 8%) 0px 6px 10px;
cursor: pointer;}.btn:hover { background: rgb(237, 238, 242);}.btn.btn_select_a_token:hover { background-color: rgb(207, 0, 99);}.btn_select_a_token { background: rgb(232, 0, 111); color: rgb(255, 255, 255);}.btn_select_a_token svg path { stroke: rgb(255, 255, 255); stroke-width: 1.5px;}.btn__flex { display: flex; align-items: center;}.btn__flex__symbol { display: flex; align-items: center;}.btn__flex__symbol__pic { width: 24px; height: 24px; box-shadow: rgb(0 0 0 / 8%) 0px 6px 10px; border-radius: 24px;}.btn__flex__symbol__name { margin: 0px 0.25rem; font-size: 18px;}.switch { padding: 4px; border-radius: 12px; height: 32px; width: 32px; position: relative; margin-top: -14px; margin-bottom: -14px; left: calc(50% - 16px); background: rgb(247, 248, 250); border: 4px solid rgb(255, 255, 255); z-index: 2;
cursor: pointer;}
/* -------------------------------------------------------------------------------------------- *//* -------------------------------------------------------------------------------------------- *//* -------------------------------------------------------------------------------------------- *//* -------------------------------------------------------------------------------------------- *//* -------------------------------------------------------------------------------------------- *//* -------------------------------------------------------------------------------------------- *//* -------------------------------------------------------------------------------------------- *//* -------------------------------------------------------------------------------------------- *//* -------------------------------------------------------------------------------------------- *//* -------------------------------------------------------------------------------------------- */
.swap { border-radius: 24px; position: relative; max-width: 480px; width: 100%; background: rgb(255, 255, 255); box-shadow: rgb(0 0 0 / 1%) 0px 0px 1px, rgb(0 0 0 / 4%) 0px 4px 8px, rgb(0 0 0 / 4%) 0px 16px 24px, rgb(0 0 0 / 1%) 0px 24px 32px; border-radius: 24px; margin-top: 1rem;
margin-left: auto; margin-right: auto;
padding-bottom: 20px;}.swap__header {}.swap__header__wrapper { display: grid; grid-template-columns: 1fr;
padding: 8px;}.swap__header__wrapper__previos { position: absolute; text-align: center; padding: 4px 10px; width: fit-content; border-radius: 8px;
background: rgb(255, 255, 255); cursor: pointer;}.swap__header__wrapper__previos:hover { background: rgb(247, 248, 250);}.swap__header__wrapper__title { display: block; position: inherit;
width: 100%; text-align: center;
font-size: 20px; font-weight: 500; text-align: center;}
.swap__body {}.swap__body__warpper {}.swap__body__warpper__input { padding: 8px;}.swap__body__warpper__input input { width: 100%; border: none; border-radius: 20px; line-height: 18px; padding: 16px 40px 16px 16px;
background: rgb(247, 248, 250);}
.swap__body__warpper__suggest { display: flex; flex-flow: row wrap;
padding: 8px;}.swap__body__warpper__suggest__flex { display: flex; flex-flow: row nowrap;
border-radius: 8px;
margin: 4px; padding: 8px 11px; background: rgb(255, 255, 255); box-shadow: rgb(0 0 0 / 1%) 0px 0px 1px, rgb(0 0 0 / 4%) 0px 4px 8px, rgb(0 0 0 / 4%) 0px 16px 24px, rgb(0 0 0 / 1%) 0px 24px 32px;
cursor: pointer;}.swap__body__warpper__suggest__flex:hover { background: rgb(237, 238, 242);}.swap__body__warpper__suggest__flex img { width: 24px; height: 24px; box-shadow: rgb(0 0 0 / 8%) 0px 6px 10px; border-radius: 24px;}
.swap__body__warpper__list { position: relative; height: 149px; width: 100%; overflow: auto; will-change: transform; direction: ltr;}.swap__body__warpper__list::-webkit-scrollbar { width: 4px;}.swap__body__warpper__list::-webkit-scrollbar-track { -webkit-box-shadow: inset 0 0 6px rgba(0, 0, 0, 0.3); border-radius: 10px;}.swap__body__warpper__list::-webkit-scrollbar-thumb { border-radius: 10px; -webkit-box-shadow: inset 0 0 6px rgba(0, 0, 0, 0.5);}.list { display: grid; grid-template-columns: auto 1fr auto;
padding: 8px; cursor: pointer;}.list:hover { background: rgb(237, 238, 242);}.list__pic img { width: 40px; height: 40px; margin-right: 16px;
box-shadow: rgb(0 0 0 / 8%) 0px 6px 10px; border-radius: 24px;}.list__flex { display: flex; justify-content: space-between;}.list__flex__left {}.list__flex__left__name { display: block;}.list__flex__left__balance { display: block;}.list__flex__right { display: block;}.list__pin {}.swap__footer {}
css ส่วนใหญ่ก็แกะโค้ดมากจากทาง uniswap เลยครับ
import React, { useEffect, useState } from "react";import styles from "./App2.module.css";
function App2() { const [toggle, setToggle] = useState(false); const [selectSourceOrDestination, setSelectSourceOrDestination] = useState(); const [input, setInput] = useState();
function handleChange(e) { const { target } = e; const { name } = target; const value = name === 'term' ? target.checked : target.value; setInput({ ...input,
[name]: { symbol: input[name].symbol, value: value, } }); } // --------- DEBUG --------- // useEffect(() => { // console.log(input); // }, [input]) // --------- DEBUG --------- function handleToggle(select) { setToggle(!toggle);
if (select === "source" || select === "destination") { setSelectSourceOrDestination(select); } } function setSelectToken(token) { if (selectSourceOrDestination === "source") { let data = { source: { symbol: token }, destination: { symbol: input?.destination?.symbol ? input.destination.symbol : null } } setInput(data); setToggle(!toggle); } else if (selectSourceOrDestination === "destination") { let data = { source: { symbol: input?.source?.symbol ? input.source.symbol : null }, destination: { symbol: token } } setInput(data); setToggle(!toggle); } } function Switching() { setInput({ source: input.destination, destination: input.source }); } return ( <div> <div> { toggle === false ? <div className={styles.modal}> <div className={styles.modal__header}> <div className={styles.modal__header__flex}> <div className={styles.modal__header__flex__item}> <div>Swap</div> </div> <div className={styles.modal__header__flex__item}> <div>SETTING</div> </div> </div> </div> <div className={styles.modal__body}> <div className={styles.modal__body__flex}> <div className={styles.modal__body__flex__item}> <div className={styles.modal__body__flex__item__topflex}> <div className={styles.modal__body__flex__item__topflex__item}> <button className={styles.btn} onClick={() => handleToggle("source")}> <span className={styles.btn__flex}> <div className={styles.btn__flex__symbol}> <img style={{marginRight: "0.5rem"}} className={styles.btn__flex__symbol__pic} src="" alt="ethereum logo" /> <span className={styles.btn__flex__symbol__name}> ETH </span> </div> <svg width="12" height="7" viewBox="0 0 12 7" fill="none" xmlns="http://www.w3.org/2000/svg" className="sc-33m4yg-8 khlnVY"><path d="M0.97168 1L6.20532 6L11.439 1" stroke="#AEAEAE"></path></svg> </span> </button> </div> <input onChange={(e) => handleChange(e)} name="source" style={{ flex: 2 }} className="" inputMode="decimal" autoComplete="off" autoCorrect="off" type="text" pattern="^[0-9]*[.,]?[0-9]*$" placeholder="0.0" minLength="1" maxLength="79" spellCheck="false" /> </div> <div className={styles.modal__body__flex__item__bottomflex}> <div className={styles.modal__body__flex__item__bottomflex__item}>Balance: 0.123456 ETH</div> <div className={styles.modal__body__flex__item__bottomflex__item}></div> </div> </div>
<div className={styles.switch} onClick={() => Switching()}> <svg xmlns="http://www.w3.org/2000/svg" width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="#000000" strokeWidth="2" strokeLinecap="round" strokeLinejoin="round"><line x1="12" y1="5" x2="12" y2="19"></line><polyline points="19 12 12 19 5 12"></polyline></svg> </div>
<div className={styles.modal__body__flex__item}> <div className={styles.modal__body__flex__item__topflex}> <div className={styles.modal__body__flex__item__topflex__item}> { input?.destination?.symbol ? ( <button className={styles.btn} onClick={() => handleToggle("destination")}> <span className={styles.btn__flex}> <div className={styles.btn__flex__symbol}> <img style={{marginRight: "0.5rem"}} className={styles.btn__flex__symbol__pic} src="https://ethereum-optimism.github.io/logos/USDT.png" alt="ethereum logo" /> <span className={styles.btn__flex__symbol__name}> USDT </span> </div> <svg width="12" height="7" viewBox="0 0 12 7" fill="none" xmlns="http://www.w3.org/2000/svg" className="sc-33m4yg-8 khlnVY"><path d="M0.97168 1L6.20532 6L11.439 1" stroke="#AEAEAE"></path></svg> </span> </button> ) : ( <button className={`${styles.btn} ${styles.btn_select_a_token}`} onClick={() => handleToggle("destination")}> <span className={styles.btn__flex}> <div className={styles.btn__flex__symbol}> <span className={styles.btn__flex__symbol__name}> Select a token </span> </div> <svg width="12" height="7" viewBox="0 0 12 7" fill="none" xmlns="http://www.w3.org/2000/svg" className="sc-33m4yg-8 khlnVY"><path d="M0.97168 1L6.20532 6L11.439 1" stroke="#AEAEAE"></path></svg> </span> </button> ) } </div> <input onChange={(e) => handleChange(e)} name="destination" style={{ flex: 2 }} className="" inputMode="decimal" autoComplete="off" autoCorrect="off" type="text" pattern="^[0-9]*[.,]?[0-9]*$" placeholder="0.0" minLength="1" maxLength="79" spellCheck="false" /> </div> <div className={styles.modal__body__flex__item__bottomflex}> <div className={styles.modal__body__flex__item__bottomflex__item}>Balance: 950.123456 USDT</div> <div className={styles.modal__body__flex__item__bottomflex__item}></div> </div> </div> </div> </div> <div className={styles.modal__footer}> <button className={styles.swap__button} disabled=""> <div className="">Swap</div> </button> <button className={styles.swap__button} disabled=""> <div className="">Enter an amount</div> </button> <button className={`${styles.swap__button} ${styles.swap__button__price_impact_too_high}`} disabled=""> <div className="">Price Impact Too High</div> </button> <button className={`${styles.swap__button} ${styles.swap__button__connect_wallet}`} disabled=""> <div className="">Connect Wallet</div> </button> </div> </div> : <div className={styles.swap}> <div className={styles.swap__header}> <div className={styles.swap__header__wrapper}> <div className={styles.swap__header__wrapper__previos} onClick={() => handleToggle()}> {'<'} </div> <div className={styles.swap__header__wrapper__title}> Select a token </div> </div> </div> <div style={{ width: "100%", height: "1px", backgroundColor: "rgb(237, 238, 242)" }}></div> <div className={styles.swap__body}> <div className={styles.swap__body__warpper}> <div className={styles.swap__body__warpper__input}> <input placeholder="Search by name or paste address" /> </div> <div className={styles.swap__body__warpper__suggest}> <div className={styles.swap__body__warpper__suggest__flex} onClick={() => setSelectToken("ETH")}> <img style={{marginRight: "0.5rem"}} src="" alt="ethereum logo" /> <span>ETH</span> </div> <div className={styles.swap__body__warpper__suggest__flex} onClick={() => setSelectToken("SNX")}> <img style={{marginRight: "0.5rem"}} src="" alt="ethereum logo" /> <span>SNX</span> </div> <div className={styles.swap__body__warpper__suggest__flex} onClick={() => setSelectToken("DAI")}> <img style={{marginRight: "0.5rem"}} src="" alt="ethereum logo" /> <span>DAI</span> </div> <div className={styles.swap__body__warpper__suggest__flex} onClick={() => setSelectToken("USDT")}> <img style={{marginRight: "0.5rem"}} src="" alt="ethereum logo" /> <span>USDT</span> </div> <div className={styles.swap__body__warpper__suggest__flex} onClick={() => setSelectToken("WBTC")}> <img style={{marginRight: "0.5rem"}} src="" alt="ethereum logo" /> <span>WBTC</span> </div> <div className={styles.swap__body__warpper__suggest__flex} onClick={() => setSelectToken("LINK")}> <img style={{marginRight: "0.5rem"}} src="" alt="ethereum logo" /> <span>LINK</span> </div> </div> <div style={{ width: "100%", height: "1px", backgroundColor: "rgb(237, 238, 242)" }}></div> <div className={styles.swap__body__warpper__list}> { [1,2,3,2,3,2,3,2,3,2,3,2,3,2,3,2,3,2,3,2,3,2,3,2,3,2,3,2,3,2,3,2,3,2,3,2,3].map((item, index) => { return ( <React.Fragment key={index}> <div className={styles.list} onClick={() => setSelectToken("sUSD")}> <div className={styles.list__pic}> <img style={{marginRight: "0.5rem"}} src="" alt="ethereum logo" /> </div> <div className={styles.list__flex}> <div className={styles.list__flex__left}> <div className={styles.list__flex__left__name}> Synth sUSD </div> <div className={styles.list__flex__left__balance}> 80.0310 sUSD </div> </div> <div className={styles.list__flex__right}>$80.03</div> </div> {/* <div className={styles.list__pin}>Pin</div> */} </div> </React.Fragment> ) }) } </div> </div> </div> <div className={styles.swap__footer}>
</div> </div>
} </div> </div> );}
export default App2;
โครงสร้าง html ส่วนใหญ่ก็แกะโค้ดมากจากทาง uniswap ครับ หน้าเลือกเหรียญจะทำแบบ 1inch
โค้ด v1.0.0 https://github.com/apisit110/uiswap.git
สุดท้าย ก็หวังว่าบทความนี้จะเป็นประโยชน์กับใครหลาย ๆ คนนะครับ