Swap page UI development
Author of this section: @ Xiaofu
in this lecture, we will implement the UI of the Swap part of Wtfswap.
The design draft is as follows:
the Swap page is the core page of Wtfswap. First, split the Swap page. The Swap page is first introduced the head of the Layout. , and then implemented Swap UI in the middle of the page; Building Swap pages requires multiple components, mainly using Card the components complete the construction of the entire framework. The complete code for this lecture can be found in here. Get.
Modification pages/wtfswap/index.tsx
introduced first. Card
component construction foundation framework:
+ import { Card } from 'antd';
import WtfLayout from "@/components/WtfLayout";
export default function Wtfswap() {
- return <WtfLayout>Wtfswap</WtfLayout>;
+ return (
+ <WtfLayout>
+ <Card title="Swap">
+ <Card></Card>
+ <Card></Card>
+ </Card>
+ </WtfLayout>
+ );
}
then create a new pages/wtfswap/swap.module.css
file, initialization style:
.swapCard {
width: 600px;
margin: 0 auto;
position: absolute;
top: 50%;
left: 50%;
transform: translate(-50%, -50%);
}
in index.tsx
use styles in:
import { Card } from 'antd';
import WtfLayout from "@/components/WtfLayout";
+ import styles from './swap.module.css';
export default function Wtfswap() {
return (
<WtfLayout>
- <Card title="Swap">
+ <Card title="Swap" className={styles.swapCard}>
<Card></Card>
<Card></Card>
</Card>
</WtfLayout>
);
}
in this way, the basic Swap page layout is built and then refined on this layout.
Next, use other components to refine the page. The selector uses Antd Design Web3 TokenSelect components, and then match Button
, Space
, Input
, Typography
and other components.
- import { Card } from 'antd';
+ import { Card, Input, Button, Space, Typography } from 'antd';
+ import { TokenSelect } from '@ant-design/web3';
+ import { SwapOutlined } from '@ant-design/icons';
import WtfLayout from "@/components/WtfLayout";
import styles from './swap.module.css';
+ const { Text } = Typography;
export default function Wtfswap() {
return (
<WtfLayout>
<Card title="Swap" className={styles.swapCard}>
<Card>
+ <Input
+ variant="borderless"
+ type="number"
+ addonAfter={
+ <TokenSelect />
+ }
+ />
+ <Space>
+ <Text type="secondary">
+ $ 0.0
+ </Text>
+ <Space>
+ <Text type="secondary">
+ Balance: 0
+ </Text>
+ <Button size="small" type="link">
+ Max
+ </Button>
+ </Space>
+ </Space>
</Card>
+ <Space>
+ <Button
+ shape="circle"
+ icon={<SwapOutlined />}
+ />
+ </Space>
<Card>
+ <Input
+ variant="borderless"
+ type="number"
+ addonAfter={
+ <TokenSelect />
+ }
+ />
+ <Space>
+ <Text type="secondary">
+ $ 0.0
+ </Text>
+ <Text type="secondary">
+ Balance: 0
+ </Text>
+ </Space>
</Card>
+ <Button type="primary" size="large" block>
+ Swap
+ </Button>
</Card>
</WtfLayout>
);
}
Now the Swap page has begun to take shape. Next, define the list of tokens that can be selected and Swap's TokenA
and TokenB
, can be derived from @ant-design/web3-assets/tokens
introduce the Token to be used:
+ import React, { useState } from 'react';
import { Card, Input, Button, Space, Typography } from 'antd';
- import { TokenSelect } from '@ant-design/web3';
+ import { TokenSelect, type Token } from '@ant-design/web3';
+ import { ETH, USDT, USDC } from '@ant-design/web3-assets/tokens';
import { SwapOutlined } from '@ant-design/icons';
// ...
export default function Wtfswap() {
+ const [tokenA, setTokenA] = useState<Token>(ETH);
+ const [tokenB, setTokenB] = useState<Token>(USDT);
+ const [amountA, setAmountA] = useState(0);
+ const [amountB, setAmountB] = useState(0);
+ const [optionsA, setOptionsA] = useState<Token[]>([ETH, USDT, USDC]);
+ const [optionsB, setOptionsB] = useState<Token[]>([USDT, ETH, USDC]);
// ...
<Input
variant="borderless"
+ value={amountA}
type="number"
addonAfter={
- <TokenSelect />
+ <TokenSelect value={tokenA} onChange={setTokenA} options={optionsA} />
}
/>
// ...
<Input
variant="borderless"
+ value={amountB}
type="number"
addonAfter={
- <TokenSelect />
+ <TokenSelect value={tokenB} onChange={setTokenB} options={optionsB} />
}
/>
// ...
now, Input
components and Button
component Add Function:
- handleAmountAChange : When the Input value of the Input component changes, it is updated synchronously.
AmountA
. Other functions are implemented in subsequent chapters. - handleSwitch : When the toggle button is clicked, swap
tokenA:TokenB
token pairs, and the corresponding valuesamountA:AmountB
.
// ...
+ const handleAmountAChange = (e: any) => {
+ setAmountA(parseFloat(e.target.value));
+ // 后续章节实现
+ };
+
+ const handleSwitch = () => {
+ setTokenA(tokenB);
+ setTokenB(tokenA);
+ setAmountA(amountB);
+ setAmountB(amountA);
+ };
// ...
<Input
variant="borderless"
value={amountA}
type="number"
+ onChange={(e) => handleAmountAChange(e)}
addonAfter={
<TokenSelect value={tokenA} onChange={setTokenA} options={optionsA} />
}
/>
// ...
<Space className={styles.switchBtn}>
<Button
shape="circle"
icon={<SwapOutlined />}
+ onClick={handleSwitch}
/>
</Space>
// ...
Then Max
button Add Function handleMax
, UI phase does not implement functions first.
// ...
+ const handleMax = () => {
+ };
// ...
- <Button size="small" type="link">
+ <Button size="small" onClick={handleMax} type="link">
Max
</Button>
// ...
The above basically realizes the Swap page, and then optimizes the style to match the content of the design draft and edit it. pages/wtfswap/swap.module.css
file:
.swapCard {
width: 600px;
margin: 0 auto;
position: absolute;
top: 50%;
left: 50%;
transform: translate(-50%, -50%);
}
+ .switchBtn {
+ font-size: '20px';
+ display: flex;
+ justify-content: center;
+ margin: 16px 0;
+ }
+
+ .swapSpace {
+ width: 99%;
+ justify-content: space-between;
+ }
+
+ .swapBtn {
+ margin-top: 10px;
+ }
in index.tsx
to introduce a newly added style:
import React, { useState } from 'react';
import { Card, Input, Button, Space, Typography } from 'antd';
import { TokenSelect, type Token } from '@ant-design/web3';
import { ETH, USDT, USDC } from '@ant-design/web3-assets/tokens';
import { SwapOutlined } from '@ant-design/icons';
import WtfLayout from "@/components/WtfLayout";
import styles from "./swap.module.css";
const { Text } = Typography;
export default function Wtfswap() {
const [tokenA, setTokenA] = useState<Token>(ETH);
const [tokenB, setTokenB] = useState<Token>(USDT);
const [amountA, setAmountA] = useState(0);
const [amountB, setAmountB] = useState(0);
const [optionsA, setOptionsA] = useState<Token[]>([ETH, USDT, USDC]);;
const [optionsB, setOptionsB] = useState<Token[]>([USDT, ETH, USDC]);;
const handleAmountAChange = (e: any) => {
setAmountA(parseFloat(e.target.value));
};
const handleSwitch = () => {
setTokenA(tokenB);
setTokenB(tokenA);
setAmountA(amountB);
setAmountB(amountA);
};
const handleMax = () => {
};
return (
<WtfLayout>
<Card title="Swap" className={styles.swapCard}>
<Card>
<Input
variant="borderless"
value={amountA}
type="number"
onChange={(e) => handleAmountAChange(e)}
addonAfter={
<TokenSelect value={tokenA} onChange={setTokenA} options={optionsA} />
}
/>
- <Space>
+ <Space className={styles.swapSpace}>
<Text type="secondary">
$ 0.0
</Text>
<Space>
<Text type="secondary">
Balance: 0
</Text>
<Button size="small" onClick={handleMax} type="link">
Max
</Button>
</Space>
</Space>
</Card>
- <Space>
+ <Space className={styles.switchBtn}>
<Button
shape="circle"
icon={<SwapOutlined />}
onClick={handleSwitch}
/>
</Space>
<Card>
<Input
variant="borderless"
value={amountB}
type="number"
addonAfter={
<TokenSelect value={tokenB} onChange={setTokenB} options={optionsB} />
}
/>
- <Space>
+ <Space className={styles.swapSpace}>
<Text type="secondary">
$ 0.0
</Text>
<Text type="secondary">
Balance: 0
</Text>
</Space>
</Card>
- <Button type="primary" size="large" block >
+ <Button type="primary" size="large" block className={styles.swapBtn}>
Swap
</Button>
</Card>
</WtfLayout>
);
}
in this way, we have realized the UI development of Swap. In order to simplify the tutorial, some details are not fully implemented according to the design draft. The page style uses CSSModules implementation, you can also use your familiar scheme. The main point of this section is to use the Antd Design Web3 TokenSelect
components for us to integrate quickly Token
into the selector.