Blog About
Table of Contents
  • เครื่องมือที่ใช้

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

Apisit N.
02 Oct 2021

สวัสดีครับ ในตอนนี้จะมาหัดเขียน frontend โดยใช้ React + CSS Modules เพื่อให้หน้าตาออกมาเหมือนกับ uniswap โดยในตอนนี้จะเขียนแต่ ui ส่วนของการแลกเปลี่ยนเหรียญ

uniswap ผมมองว่า ui ออกแบบมาสวย เรียบง่ายดีและก็มีเว็บอื่น ๆ ที่ผมเข้าบ่อย ๆ มีหน้าตาก็คล้ายกันมากก็เลยอยากรู้ว่าเขียนยังไงจึงเป็นที่มาของบล็อกนี้ครับ

เครื่องมือที่ใช้

  1. nodejs
  2. React

มาเริ่มกันเลย

Terminal window
npx create-react-app frontend

เมื่อสร้างโปรเจคเสร็จแล้ว run ด้วยคำสั่ง

Terminal window
cd frontend
yarn start

สร้างไฟล์ใหม่ src/styles.module.css ไฟล์นี้เราจะเขียน css modules กันจากนั้นก็ import เพื่อเรียกให้งานใน App.js

App.js
1
import React, { useEffect, useState } from "react";
2
import styles from "./App2.module.css";
3
...

ข้อดีของการใช้งาน css modules ในมุมมองของผมคือการตั้งชื่อ classname ซ้ำได้แต่ก็ไม่ควรตั้งซ้ำกันอยู่ดีจะไม่ได้สับสน ปัญหาการเขียน css แบบเดิมคือเมื่อโปรเจคใหญ่ขึ้นจะมีไฟล์เยอะขึ้นตาม ต้องสร้างไฟล์ css เยอะถ้าโปรเจคนั้นไม่ได้เขียนแค่คนเดียวอีกจะรู้ได้ยังไงว่าคนอื่นจะตั้งชื่อ classname ซ้ำกับเราหรือเปล่า จากประสบการณ์ที่ผมได้เขียนมาก็เป็นแบบนี้จริงๆ เพราะตอนแรกผมเขียนแบบเดิมทำให้จะแก้ ui ยากมากไม่รู้ว่ามันใช้ classname จากไฟล์ไหนกันแน่ จะปรับให้เป็นแบบ css modules ก็สายไปแล้วเพราะใน css modules มันใช้ชื่อด้วยตัว ( - ) ขีดแดท(Dashes)แบบเดิมไม่ได้ ต้องเปลี่ยนชื่อ classname ทั้งหมด

หลังจากที่มีประสบการณ์แบบนั้นแล้ว ผมจึงลองหัดเขียน css แบบใหม่ดูบ้างเช่น css modules โดยจะใช้ 1 ไฟล์.js ต่อ 1 ไฟล์.module.css ไปเลยง่ายดี แยกแต่ละไฟล์แต่ละหน้าออกจากกันเลย

หน้าจอแลกเปลี่ยนเหรียญ

หน้าจอเลือกเหรียญ

styles.module.css
1
* {
2
margin: 0;
3
padding: 0;
4
box-sizing: border-box;
5
}
6
.modal {
7
position: relative;
8
max-width: 480px;
9
width: 100%;
10
background: rgb(255, 255, 255);
11
box-shadow: rgb(0 0 0 / 1%) 0px 0px 1px, rgb(0 0 0 / 4%) 0px 4px 8px,
12
rgb(0 0 0 / 4%) 0px 16px 24px, rgb(0 0 0 / 1%) 0px 24px 32px;
13
border-radius: 24px;
14
margin-top: 1rem;
15
16
margin-left: auto;
17
margin-right: auto;
18
}
19
.modal__header {
20
padding: 1rem 1.25rem 0.5rem;
21
width: 100%;
22
color: rgb(86, 90, 105);
23
}
24
.modal__header__flex {
25
display: flex;
26
justify-content: space-between;
27
align-items: center;
28
}
29
.modal__header__flex__item {
30
width: fit-content;
31
}
32
.modal__body {
33
/* background: rgb(254, 2, 9); */
34
/* height: 200px; */
35
36
position: relative;
37
padding: 8px;
38
}
39
.modal__body__flex {
40
display: grid;
41
grid-auto-rows: auto;
42
/* row-gap: 8px; */
43
}
44
.modal__body__flex__item {
45
border-radius: 20px;
46
border: 1px solid rgb(237, 238, 242);
47
background: rgb(247, 248, 250);
48
width: initial;
49
}
50
.modal__body__flex__item__topflex {
51
display: flex;
52
justify-content: space-between;
53
align-items: center;
54
55
line-height: 1rem;
56
padding: 1rem 1rem 0.75rem;
57
}
58
.modal__body__flex__item__topflex__item {
59
}
60
.modal__body__flex__item__topflex input {
61
color: rgb(0, 0, 0);
62
width: 0px;
63
position: relative;
64
font-weight: 500;
65
outline: none;
66
border: none;
67
flex: 1 1 auto;
68
background: rgb(247, 248, 250);
69
font-size: 24px;
70
white-space: nowrap;
71
overflow: hidden;
72
text-overflow: ellipsis;
73
padding: 0px;
74
appearance: textfield;
75
text-align: right;
76
}
77
.modal__body__flex__item__bottomflex {
78
display: flex;
79
justify-content: space-between;
80
align-items: center;
81
82
line-height: 1rem;
83
padding: 0px 1rem 1rem;
84
}
85
.modal__body__flex__item__bottomflex__item {
86
}
87
.modal__footer {
88
/* background: rgb(254, 127, 206); */
89
display: grid;
90
grid-auto-rows: auto;
91
row-gap: 8px;
92
93
padding: 8px;
94
border-bottom-left-radius: 20px;
95
border-bottom-right-radius: 20px;
96
}
97
98
.swap__button {
99
padding: 16px;
100
width: 100%;
101
font-weight: 500;
102
text-align: center;
103
border-radius: 20px;
104
outline: none;
105
border: 1px solid transparent;
106
/* color: white; */
107
text-decoration: none;
108
display: flex;
109
-webkit-box-pack: center;
110
justify-content: center;
111
flex-wrap: nowrap;
112
-webkit-box-align: center;
113
align-items: center;
114
/* cursor: pointer; */
115
position: relative;
116
z-index: 1;
117
will-change: transform;
118
transition: transform 450ms ease 0s;
119
transform: perspective(1px) translateZ(0px);
120
121
background: rgb(237, 238, 242);
122
color: rgb(86, 90, 105);
123
box-shadow: none;
124
}
125
.swap__button__price_impact_too_high {
126
border: 1px solid rgb(218, 45, 43);
127
opacity: 0.5;
128
background: rgb(218, 45, 43);
129
color: white;
130
131
cursor: auto;
132
}
133
.swap__button__connect_wallet {
134
opacity: 0.5;
135
background: rgb(253, 234, 241);
136
color: rgb(213, 0, 102);
137
138
cursor: auto;
139
}
140
.swap__button div {
141
box-sizing: border-box;
142
margin: 0px;
143
min-width: 0px;
144
font-size: 20px;
145
font-weight: 500;
146
}
147
148
.btn {
149
background: rgb(255, 255, 255);
150
display: flex;
151
align-items: center;
152
153
border: none;
154
border-radius: 20px;
155
height: 2.4rem;
156
width: initial;
157
padding: 0px 8px;
158
margin-right: 12px;
159
160
font-size: 24px;
161
font-weight: 500;
162
box-shadow: rgb(0 0 0 / 8%) 0px 6px 10px;
163
164
cursor: pointer;
165
}
166
.btn:hover {
167
background: rgb(237, 238, 242);
168
}
169
.btn.btn_select_a_token:hover {
170
background-color: rgb(207, 0, 99);
171
}
172
.btn_select_a_token {
173
background: rgb(232, 0, 111);
174
color: rgb(255, 255, 255);
175
}
176
.btn_select_a_token svg path {
177
stroke: rgb(255, 255, 255);
178
stroke-width: 1.5px;
179
}
180
.btn__flex {
181
display: flex;
182
align-items: center;
183
}
184
.btn__flex__symbol {
185
display: flex;
186
align-items: center;
187
}
188
.btn__flex__symbol__pic {
189
width: 24px;
190
height: 24px;
191
box-shadow: rgb(0 0 0 / 8%) 0px 6px 10px;
192
border-radius: 24px;
193
}
194
.btn__flex__symbol__name {
195
margin: 0px 0.25rem;
196
font-size: 18px;
197
}
198
.switch {
199
padding: 4px;
200
border-radius: 12px;
201
height: 32px;
202
width: 32px;
203
position: relative;
204
margin-top: -14px;
205
margin-bottom: -14px;
206
left: calc(50% - 16px);
207
background: rgb(247, 248, 250);
208
border: 4px solid rgb(255, 255, 255);
209
z-index: 2;
210
211
cursor: pointer;
212
}
213
214
/* -------------------------------------------------------------------------------------------- */
215
/* -------------------------------------------------------------------------------------------- */
216
/* -------------------------------------------------------------------------------------------- */
217
/* -------------------------------------------------------------------------------------------- */
218
/* -------------------------------------------------------------------------------------------- */
219
/* -------------------------------------------------------------------------------------------- */
220
/* -------------------------------------------------------------------------------------------- */
221
/* -------------------------------------------------------------------------------------------- */
222
/* -------------------------------------------------------------------------------------------- */
223
/* -------------------------------------------------------------------------------------------- */
224
225
.swap {
226
border-radius: 24px;
227
position: relative;
228
max-width: 480px;
229
width: 100%;
230
background: rgb(255, 255, 255);
231
box-shadow: rgb(0 0 0 / 1%) 0px 0px 1px, rgb(0 0 0 / 4%) 0px 4px 8px,
232
rgb(0 0 0 / 4%) 0px 16px 24px, rgb(0 0 0 / 1%) 0px 24px 32px;
233
border-radius: 24px;
234
margin-top: 1rem;
235
236
margin-left: auto;
237
margin-right: auto;
238
239
padding-bottom: 20px;
240
}
241
.swap__header {
242
}
243
.swap__header__wrapper {
244
display: grid;
245
grid-template-columns: 1fr;
246
247
padding: 8px;
248
}
249
.swap__header__wrapper__previos {
250
position: absolute;
251
text-align: center;
252
padding: 4px 10px;
253
width: fit-content;
254
border-radius: 8px;
255
256
background: rgb(255, 255, 255);
257
cursor: pointer;
258
}
259
.swap__header__wrapper__previos:hover {
260
background: rgb(247, 248, 250);
261
}
262
.swap__header__wrapper__title {
263
display: block;
264
position: inherit;
265
266
width: 100%;
267
text-align: center;
268
269
font-size: 20px;
270
font-weight: 500;
271
text-align: center;
272
}
273
274
.swap__body {
275
}
276
.swap__body__warpper {
277
}
278
.swap__body__warpper__input {
279
padding: 8px;
280
}
281
.swap__body__warpper__input input {
282
width: 100%;
283
border: none;
284
border-radius: 20px;
285
line-height: 18px;
286
padding: 16px 40px 16px 16px;
287
288
background: rgb(247, 248, 250);
289
}
290
291
.swap__body__warpper__suggest {
292
display: flex;
293
flex-flow: row wrap;
294
295
padding: 8px;
296
}
297
.swap__body__warpper__suggest__flex {
298
display: flex;
299
flex-flow: row nowrap;
300
301
border-radius: 8px;
302
303
margin: 4px;
304
padding: 8px 11px;
305
background: rgb(255, 255, 255);
306
box-shadow: rgb(0 0 0 / 1%) 0px 0px 1px, rgb(0 0 0 / 4%) 0px 4px 8px,
307
rgb(0 0 0 / 4%) 0px 16px 24px, rgb(0 0 0 / 1%) 0px 24px 32px;
308
309
cursor: pointer;
310
}
311
.swap__body__warpper__suggest__flex:hover {
312
background: rgb(237, 238, 242);
313
}
314
.swap__body__warpper__suggest__flex img {
315
width: 24px;
316
height: 24px;
317
box-shadow: rgb(0 0 0 / 8%) 0px 6px 10px;
318
border-radius: 24px;
319
}
320
321
.swap__body__warpper__list {
322
position: relative;
323
height: 149px;
324
width: 100%;
325
overflow: auto;
326
will-change: transform;
327
direction: ltr;
328
}
329
.swap__body__warpper__list::-webkit-scrollbar {
330
width: 4px;
331
}
332
.swap__body__warpper__list::-webkit-scrollbar-track {
333
-webkit-box-shadow: inset 0 0 6px rgba(0, 0, 0, 0.3);
334
border-radius: 10px;
335
}
336
.swap__body__warpper__list::-webkit-scrollbar-thumb {
337
border-radius: 10px;
338
-webkit-box-shadow: inset 0 0 6px rgba(0, 0, 0, 0.5);
339
}
340
.list {
341
display: grid;
342
grid-template-columns: auto 1fr auto;
343
344
padding: 8px;
345
cursor: pointer;
346
}
347
.list:hover {
348
background: rgb(237, 238, 242);
349
}
350
.list__pic img {
351
width: 40px;
352
height: 40px;
353
margin-right: 16px;
354
355
box-shadow: rgb(0 0 0 / 8%) 0px 6px 10px;
356
border-radius: 24px;
357
}
358
.list__flex {
359
display: flex;
360
justify-content: space-between;
361
}
362
.list__flex__left {
363
}
364
.list__flex__left__name {
365
display: block;
366
}
367
.list__flex__left__balance {
368
display: block;
369
}
370
.list__flex__right {
371
display: block;
372
}
373
.list__pin {
374
}
375
.swap__footer {
376
}

css ส่วนใหญ่ก็แกะโค้ดมากจากทาง uniswap เลยครับ

App.js
1
import React, { useEffect, useState } from "react";
2
import styles from "./App2.module.css";
3
4
function App2() {
5
const [toggle, setToggle] = useState(false);
6
const [selectSourceOrDestination, setSelectSourceOrDestination] = useState();
7
const [input, setInput] = useState();
8
9
function handleChange(e) {
10
const { target } = e;
11
const { name } = target;
12
const value = name === 'term' ? target.checked : target.value;
13
setInput({
14
...input,
15
16
[name]: {
17
symbol: input[name].symbol,
18
value: value,
19
}
20
});
21
}
22
// --------- DEBUG ---------
23
// useEffect(() => {
24
// console.log(input);
25
// }, [input])
26
// --------- DEBUG ---------
27
function handleToggle(select) {
28
setToggle(!toggle);
29
30
if (select === "source" || select === "destination") {
31
setSelectSourceOrDestination(select);
32
}
33
}
34
function setSelectToken(token) {
35
if (selectSourceOrDestination === "source") {
36
let data = {
37
source: {
38
symbol: token
39
},
40
destination: {
41
symbol: input?.destination?.symbol ? input.destination.symbol : null
42
}
43
}
44
setInput(data);
45
setToggle(!toggle);
46
} else if (selectSourceOrDestination === "destination") {
47
let data = {
48
source: {
49
symbol: input?.source?.symbol ? input.source.symbol : null
50
},
51
destination: {
52
symbol: token
53
}
54
}
55
setInput(data);
56
setToggle(!toggle);
57
}
58
}
59
function Switching() {
60
setInput({
61
source: input.destination,
62
destination: input.source
63
});
64
}
65
return (
66
<div>
67
<div>
68
{
69
toggle === false ?
70
<div className={styles.modal}>
71
<div className={styles.modal__header}>
72
<div className={styles.modal__header__flex}>
73
<div className={styles.modal__header__flex__item}>
74
<div>Swap</div>
75
</div>
76
<div className={styles.modal__header__flex__item}>
77
<div>SETTING</div>
78
</div>
79
</div>
80
</div>
81
<div className={styles.modal__body}>
82
<div className={styles.modal__body__flex}>
83
<div className={styles.modal__body__flex__item}>
84
<div className={styles.modal__body__flex__item__topflex}>
85
<div className={styles.modal__body__flex__item__topflex__item}>
86
<button className={styles.btn} onClick={() => handleToggle("source")}>
87
<span className={styles.btn__flex}>
88
<div className={styles.btn__flex__symbol}>
89
<img style={{marginRight: "0.5rem"}} className={styles.btn__flex__symbol__pic} src="" alt="ethereum logo" />
90
<span className={styles.btn__flex__symbol__name}>
91
ETH
92
</span>
93
</div>
94
<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>
95
</span>
96
</button>
97
</div>
98
<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" />
99
</div>
100
<div className={styles.modal__body__flex__item__bottomflex}>
101
<div className={styles.modal__body__flex__item__bottomflex__item}>Balance: 0.123456 ETH</div>
102
<div className={styles.modal__body__flex__item__bottomflex__item}></div>
103
</div>
104
</div>
105
106
<div className={styles.switch} onClick={() => Switching()}>
107
<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>
108
</div>
109
110
<div className={styles.modal__body__flex__item}>
111
<div className={styles.modal__body__flex__item__topflex}>
112
<div className={styles.modal__body__flex__item__topflex__item}>
113
{
114
input?.destination?.symbol ? (
115
<button className={styles.btn} onClick={() => handleToggle("destination")}>
116
<span className={styles.btn__flex}>
117
<div className={styles.btn__flex__symbol}>
118
<img style={{marginRight: "0.5rem"}} className={styles.btn__flex__symbol__pic} src="https://ethereum-optimism.github.io/logos/USDT.png" alt="ethereum logo" />
119
<span className={styles.btn__flex__symbol__name}>
120
USDT
121
</span>
122
</div>
123
<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>
124
</span>
125
</button>
126
)
127
:
128
(
129
<button className={`${styles.btn} ${styles.btn_select_a_token}`} onClick={() => handleToggle("destination")}>
130
<span className={styles.btn__flex}>
131
<div className={styles.btn__flex__symbol}>
132
<span className={styles.btn__flex__symbol__name}>
133
Select a token
134
</span>
135
</div>
136
<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>
137
</span>
138
</button>
139
)
140
}
141
</div>
142
<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" />
143
</div>
144
<div className={styles.modal__body__flex__item__bottomflex}>
145
<div className={styles.modal__body__flex__item__bottomflex__item}>Balance: 950.123456 USDT</div>
146
<div className={styles.modal__body__flex__item__bottomflex__item}></div>
147
</div>
148
</div>
149
</div>
150
</div>
151
<div className={styles.modal__footer}>
152
<button className={styles.swap__button} disabled="">
153
<div className="">Swap</div>
154
</button>
155
<button className={styles.swap__button} disabled="">
156
<div className="">Enter an amount</div>
157
</button>
158
<button className={`${styles.swap__button} ${styles.swap__button__price_impact_too_high}`} disabled="">
159
<div className="">Price Impact Too High</div>
160
</button>
161
<button className={`${styles.swap__button} ${styles.swap__button__connect_wallet}`} disabled="">
162
<div className="">Connect Wallet</div>
163
</button>
164
</div>
165
</div>
166
:
167
<div className={styles.swap}>
168
<div className={styles.swap__header}>
169
<div className={styles.swap__header__wrapper}>
170
<div className={styles.swap__header__wrapper__previos} onClick={() => handleToggle()}>
171
{'<'}
172
</div>
173
<div className={styles.swap__header__wrapper__title}>
174
Select a token
175
</div>
176
</div>
177
</div>
178
<div style={{ width: "100%", height: "1px", backgroundColor: "rgb(237, 238, 242)" }}></div>
179
<div className={styles.swap__body}>
180
<div className={styles.swap__body__warpper}>
181
<div className={styles.swap__body__warpper__input}>
182
<input placeholder="Search by name or paste address" />
183
</div>
184
<div className={styles.swap__body__warpper__suggest}>
185
<div className={styles.swap__body__warpper__suggest__flex} onClick={() => setSelectToken("ETH")}>
186
<img style={{marginRight: "0.5rem"}} src="" alt="ethereum logo" />
187
<span>ETH</span>
188
</div>
189
<div className={styles.swap__body__warpper__suggest__flex} onClick={() => setSelectToken("SNX")}>
190
<img style={{marginRight: "0.5rem"}} src="" alt="ethereum logo" />
191
<span>SNX</span>
192
</div>
193
<div className={styles.swap__body__warpper__suggest__flex} onClick={() => setSelectToken("DAI")}>
194
<img style={{marginRight: "0.5rem"}} src="" alt="ethereum logo" />
195
<span>DAI</span>
196
</div>
197
<div className={styles.swap__body__warpper__suggest__flex} onClick={() => setSelectToken("USDT")}>
198
<img style={{marginRight: "0.5rem"}} src="" alt="ethereum logo" />
199
<span>USDT</span>
200
</div>
201
<div className={styles.swap__body__warpper__suggest__flex} onClick={() => setSelectToken("WBTC")}>
202
<img style={{marginRight: "0.5rem"}} src="" alt="ethereum logo" />
203
<span>WBTC</span>
204
</div>
205
<div className={styles.swap__body__warpper__suggest__flex} onClick={() => setSelectToken("LINK")}>
206
<img style={{marginRight: "0.5rem"}} src="" alt="ethereum logo" />
207
<span>LINK</span>
208
</div>
209
</div>
210
<div style={{ width: "100%", height: "1px", backgroundColor: "rgb(237, 238, 242)" }}></div>
211
<div className={styles.swap__body__warpper__list}>
212
{
213
[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) => {
214
return (
215
<React.Fragment key={index}>
216
<div className={styles.list} onClick={() => setSelectToken("sUSD")}>
217
<div className={styles.list__pic}>
218
<img style={{marginRight: "0.5rem"}} src="" alt="ethereum logo" />
219
</div>
220
<div className={styles.list__flex}>
221
<div className={styles.list__flex__left}>
222
<div className={styles.list__flex__left__name}>
223
Synth sUSD
224
</div>
225
<div className={styles.list__flex__left__balance}>
226
80.0310 sUSD
227
</div>
228
</div>
229
<div className={styles.list__flex__right}>$80.03</div>
230
</div>
231
{/* <div className={styles.list__pin}>Pin</div> */}
232
</div>
233
</React.Fragment>
234
)
235
})
236
}
237
</div>
238
</div>
239
</div>
240
<div className={styles.swap__footer}>
241
242
</div>
243
</div>
244
245
}
246
</div>
247
</div>
248
);
249
}
250
251
export default App2;

โครงสร้าง html ส่วนใหญ่ก็แกะโค้ดมากจากทาง uniswap ครับ หน้าเลือกเหรียญจะทำแบบ 1inch

โค้ด v1.0.0 https://github.com/apisit110/uiswap.git

สุดท้าย ก็หวังว่าบทความนี้จะเป็นประโยชน์กับใครหลาย ๆ คนนะครับ

© 2025 Apisit N.