From c147bbf2f5f5ede726a217a9f1712772b43a2659 Mon Sep 17 00:00:00 2001 From: "dependabot[bot]" <49699333+dependabot[bot]@users.noreply.github.com> Date: Thu, 12 Dec 2024 21:31:53 +0000 Subject: [PATCH 1/5] build(deps): bump nanoid from 3.3.6 to 3.3.8 Bumps [nanoid](https://github.com/ai/nanoid) from 3.3.6 to 3.3.8. - [Release notes](https://github.com/ai/nanoid/releases) - [Changelog](https://github.com/ai/nanoid/blob/main/CHANGELOG.md) - [Commits](https://github.com/ai/nanoid/compare/3.3.6...3.3.8) --- updated-dependencies: - dependency-name: nanoid dependency-type: indirect ... Signed-off-by: dependabot[bot] --- yarn.lock | 13 ++++--------- 1 file changed, 4 insertions(+), 9 deletions(-) diff --git a/yarn.lock b/yarn.lock index b21200bdc..3310818ea 100644 --- a/yarn.lock +++ b/yarn.lock @@ -13579,15 +13579,10 @@ nanoassert@^2.0.0: resolved "https://registry.yarnpkg.com/nanoassert/-/nanoassert-2.0.0.tgz#a05f86de6c7a51618038a620f88878ed1e490c09" integrity sha512-7vO7n28+aYO4J+8w96AzhmU8G+Y/xpPDJz/se19ICsqj/momRbb9mh9ZUtkoJ5X3nTnPdhEJyc0qnM6yAsHBaA== -nanoid@^3.3.6: - version "3.3.6" - resolved "https://registry.yarnpkg.com/nanoid/-/nanoid-3.3.6.tgz#443380c856d6e9f9824267d960b4236ad583ea4c" - integrity sha512-BGcqMMJuToF7i1rt+2PWSNVnWIkGCU78jBG3RxO/bZlnZPK2Cmi2QaffxGO/2RvWi9sL+FAiRiXMgsyxQ1DIDA== - -nanoid@^3.3.7: - version "3.3.7" - resolved "https://registry.yarnpkg.com/nanoid/-/nanoid-3.3.7.tgz#d0c301a691bc8d54efa0a2226ccf3fe2fd656bd8" - integrity sha512-eSRppjcPIatRIMC1U6UngP8XFcz8MQWGQdt1MTBQ7NaAmvXDfvNxbvWV3x2y6CdEUciCSsDHDQZbhYaB8QEo2g== +nanoid@^3.3.6, nanoid@^3.3.7: + version "3.3.8" + resolved "https://registry.yarnpkg.com/nanoid/-/nanoid-3.3.8.tgz#b1be3030bee36aaff18bacb375e5cce521684baf" + integrity sha512-WNLf5Sd8oZxOm+TzppcYk8gVOgP+l58xNy58D0nbUnOxOWRWvlcCV4kUF7ltmI6PsrLl/BgKEyS4mqsGChFN0w== natural-compare-lite@^1.4.0: version "1.4.0" From eb2b78fed8339da4bfbfc7cbb4e41e722cc9e803 Mon Sep 17 00:00:00 2001 From: Chen Yu Date: Mon, 23 Dec 2024 11:33:01 +0800 Subject: [PATCH 2/5] Enable fiber nodes (#1829) --- .eslintrc.js | 1 + public/images/tokens/ckb_token.svg | 9 + .../GraphChannelList/index.module.scss | 139 +++++++ src/components/GraphChannelList/index.tsx | 156 ++++++++ .../Search/AggregateSearchResults.tsx | 16 + src/components/Search/index.tsx | 1 + src/components/Search/utils.ts | 6 + src/constants/fiberChainHash.ts | 4 + src/locales/en.json | 52 ++- src/locales/zh.json | 3 +- src/pages/Fiber/Channel/fiber.module.scss | 11 + src/pages/Fiber/Channel/fiber.tsx | 25 ++ src/pages/Fiber/Channel/index.module.scss | 114 ++++++ src/pages/Fiber/Channel/index.tsx | 150 ++++++++ .../Fiber/GraphChannelList/index.module.scss | 46 +++ src/pages/Fiber/GraphChannelList/index.tsx | 53 +++ src/pages/Fiber/GraphNode/index.module.scss | 231 ++++++++++++ src/pages/Fiber/GraphNode/index.tsx | 340 ++++++++++++++++++ .../Fiber/GraphNodeList/index.module.scss | 167 +++++++++ src/pages/Fiber/GraphNodeList/index.tsx | 221 ++++++++++++ src/pages/Fiber/Pagination/index.module.scss | 95 +++++ src/pages/Fiber/Pagination/index.tsx | 68 ++++ src/pages/Fiber/Peer/index.module.scss | 151 ++++++++ src/pages/Fiber/Peer/index.tsx | 194 ++++++++++ .../Fiber/PeerList/AddPeerForm.module.scss | 74 ++++ src/pages/Fiber/PeerList/AddPeerForm.tsx | 105 ++++++ src/pages/Fiber/PeerList/index.module.scss | 144 ++++++++ src/pages/Fiber/PeerList/index.tsx | 188 ++++++++++ src/pages/Fiber/utils/index.tsx | 42 +++ src/pages/Home/Banner/index.tsx | 15 +- src/routes/index.tsx | 32 ++ src/services/ExplorerService/fetcher.ts | 222 ++++++++++++ src/styles/card.module.scss | 6 + src/styles/table.module.scss | 43 +++ src/styles/text.module.scss | 21 ++ 35 files changed, 3129 insertions(+), 16 deletions(-) create mode 100644 public/images/tokens/ckb_token.svg create mode 100644 src/components/GraphChannelList/index.module.scss create mode 100644 src/components/GraphChannelList/index.tsx create mode 100644 src/constants/fiberChainHash.ts create mode 100644 src/pages/Fiber/Channel/fiber.module.scss create mode 100644 src/pages/Fiber/Channel/fiber.tsx create mode 100644 src/pages/Fiber/Channel/index.module.scss create mode 100644 src/pages/Fiber/Channel/index.tsx create mode 100644 src/pages/Fiber/GraphChannelList/index.module.scss create mode 100644 src/pages/Fiber/GraphChannelList/index.tsx create mode 100644 src/pages/Fiber/GraphNode/index.module.scss create mode 100644 src/pages/Fiber/GraphNode/index.tsx create mode 100644 src/pages/Fiber/GraphNodeList/index.module.scss create mode 100644 src/pages/Fiber/GraphNodeList/index.tsx create mode 100644 src/pages/Fiber/Pagination/index.module.scss create mode 100644 src/pages/Fiber/Pagination/index.tsx create mode 100644 src/pages/Fiber/Peer/index.module.scss create mode 100644 src/pages/Fiber/Peer/index.tsx create mode 100644 src/pages/Fiber/PeerList/AddPeerForm.module.scss create mode 100644 src/pages/Fiber/PeerList/AddPeerForm.tsx create mode 100644 src/pages/Fiber/PeerList/index.module.scss create mode 100644 src/pages/Fiber/PeerList/index.tsx create mode 100644 src/pages/Fiber/utils/index.tsx create mode 100644 src/styles/card.module.scss create mode 100644 src/styles/table.module.scss create mode 100644 src/styles/text.module.scss diff --git a/.eslintrc.js b/.eslintrc.js index 47549e586..86d7a97b2 100644 --- a/.eslintrc.js +++ b/.eslintrc.js @@ -115,6 +115,7 @@ module.exports = { 'jsx-a11y/label-has-associated-control': 'off', 'jsx-a11y/no-static-element-interactions': 'off', 'jsx-a11y/no-noninteractive-element-interactions': 'off', + 'jsx-a11y/click-events-have-key-events': 'off', }, env: { jest: true, diff --git a/public/images/tokens/ckb_token.svg b/public/images/tokens/ckb_token.svg new file mode 100644 index 000000000..27302bd6f --- /dev/null +++ b/public/images/tokens/ckb_token.svg @@ -0,0 +1,9 @@ + + + + + + + + diff --git a/src/components/GraphChannelList/index.module.scss b/src/components/GraphChannelList/index.module.scss new file mode 100644 index 000000000..399a0e332 --- /dev/null +++ b/src/components/GraphChannelList/index.module.scss @@ -0,0 +1,139 @@ +@import '../../styles/variables.module'; +@import '../../styles/text.module'; + +.container { + font-size: 0.875rem; + + a { + color: var(--primary-color); + } + + svg { + pointer-events: none; + } + + dl { + display: flex; + gap: 4px; + + @media screen and (width <= $mobileBreakPoint) { + display: block; + } + } + + dl, + dd, + dt { + margin: 0; + + /* white-space: pre; */ + + /* flex-wrap: wrap; */ + } + + dt { + &::after { + content: ':'; + } + } + + dd { + display: flex; + align-items: center; + gap: 4px; + } + + .channel { + margin-bottom: 4px; + background: #fff; + padding: 8px 40px; + + @media screen and (width <= $largeBreakPoint) { + padding: 8px; + } + + h1 { + font-size: 1.2rem; + } + } + + .funding { + display: flex; + gap: 4px; + flex-wrap: nowrap; + overflow: hidden; + + dd { + overflow: hidden; + } + + a.address { + @extend %hash; + + min-width: 180px; + } + } + + .outPoint { + dd { + overflow: hidden; + } + + a { + @extend %hash; + } + } + + .nodesContainer { + border-radius: 6px; + border: 1px solid #ccc; + padding: 8px; + margin-top: 8px; + background: rgb(0 0 0 / 3%); + + dt, + dd { + display: flex; + flex-wrap: nowrap; + } + } + + .nodes { + display: flex; + + &[data-is-full-width='false'] { + flex-direction: column; + } + + h3 { + display: flex; + align-items: center; + gap: 4px; + font-size: 1rem; + + span { + display: flex; + align-items: center; + } + } + + gap: 20px; + + .node { + flex: 1; + overflow: hidden; + + dd { + overflow: hidden; + } + + a { + @extend %hash; + } + } + + @media screen and (width <= $mobileBreakPoint) { + flex-direction: column; + } + } +} diff --git a/src/components/GraphChannelList/index.tsx b/src/components/GraphChannelList/index.tsx new file mode 100644 index 000000000..a32a02b11 --- /dev/null +++ b/src/components/GraphChannelList/index.tsx @@ -0,0 +1,156 @@ +import { CopyIcon, HomeIcon, GlobeIcon } from '@radix-ui/react-icons' +import { Tooltip } from 'antd' +import dayjs from 'dayjs' +import type { FC } from 'react' +import { Link } from 'react-router-dom' +import type { Fiber } from '../../services/ExplorerService/fetcher' +import { parseNumericAbbr } from '../../utils/chart' +import { localeNumberString } from '../../utils/number' +import { shannonToCkb } from '../../utils/util' +import styles from './index.module.scss' + +const TIME_TEMPLATE = 'YYYY/MM/DD hh:mm:ss' + +const GraphChannelList: FC<{ list: Fiber.Graph.Channel[]; node?: string }> = ({ list, node }) => { + if (!list.length) { + return
No Channels
+ } + + return ( +
+ {list.map((channel, i) => { + const outPoint = { + txHash: channel.channelOutpoint.slice(0, -8), + index: parseInt(channel.channelOutpoint.slice(-8), 16), + } + + const ckb = shannonToCkb(channel.capacity) + const amount = parseNumericAbbr(ckb) + + const fundingCkb = shannonToCkb(channel.openTransactionInfo.capacity) + const fundingCkbAmount = parseNumericAbbr(fundingCkb) + + const fundingUdtAmount = channel.openTransactionInfo.udtAmount + ? parseNumericAbbr(channel.openTransactionInfo.udtAmount) + : null + + const outpoint = `${outPoint.txHash}#${outPoint.index}` + + return ( +
+

Channel #{i + 1}

+
+
+
Out Point
+
+ + +
{outpoint.slice(0, -15)}
+
{outpoint.slice(-15)}
+ +
+ +
+
+ +
+
Capacity
+
+ + {`${amount} CKB`} + +
+
+
+
Source
+
+ {fundingUdtAmount || ( + + {`${fundingCkbAmount} CKB`} + + )} + from + + +
{channel.openTransactionInfo.address.slice(0, -15)}
+
{channel.openTransactionInfo.address.slice(-15)}
+ +
+
+
+
+
Position
+
+ On + + + {localeNumberString(channel.fundingTxBlockNumber)} + + +
+
+
+ +
+

Nodes

+
+
+

+ First Node + {node ? {node === channel.node1 ? : } : null} +

+
+
ID
+
+ + +
{`0x${channel.node1.slice(0, -8)}`}
+
{channel.node1.slice(-8)}
+ +
+ +
+
+
+
Fee Rate
+
{`${localeNumberString(channel.node1ToNode2FeeRate)} shannon/kB`}
+
+
+
+

+ Second Node + {node ? {node === channel.node2 ? : } : null} +

+
+
ID
+
+ + +
{`0x${channel.node2.slice(0, -8)}`}
+
{channel.node2.slice(-8)}
+ +
+ +
+
+
+
Fee Rate
+
{`${localeNumberString(channel.node2ToNode1FeeRate)} shannon/kB`}
+
+
+
+
+
+ ) + })} +
+ ) +} + +export default GraphChannelList diff --git a/src/components/Search/AggregateSearchResults.tsx b/src/components/Search/AggregateSearchResults.tsx index de54979bc..6f0d451ff 100644 --- a/src/components/Search/AggregateSearchResults.tsx +++ b/src/components/Search/AggregateSearchResults.tsx @@ -259,6 +259,22 @@ const SearchResultItem: FC<{ keyword?: string; item: AggregateSearchResult }> = ) } + if (item.type === SearchResultType.FiberGraphNode) { + return ( + +
+ + +
+ + {t('search.fiber_graph_node')} # {localeNumberString(item.attributes.alias)} + +
+
+ + ) + } + return (
diff --git a/src/components/Search/index.tsx b/src/components/Search/index.tsx index 8123799b5..8d628193a 100644 --- a/src/components/Search/index.tsx +++ b/src/components/Search/index.tsx @@ -45,6 +45,7 @@ const ALLOW_SEARCH_TYPES = [ SearchResultType.UDT, SearchResultType.DID, SearchResultType.BtcAddress, + SearchResultType.FiberGraphNode, ] async function fetchAggregateSearchResult(searchValue: string): Promise { diff --git a/src/components/Search/utils.ts b/src/components/Search/utils.ts index 0425169ef..0bb9527b8 100644 --- a/src/components/Search/utils.ts +++ b/src/components/Search/utils.ts @@ -54,6 +54,9 @@ export const getURLByAggregateSearchResult = (result: AggregateSearchResult) => case SearchResultType.BtcAddress: return `/address/${attributes.addressHash}` + case SearchResultType.FiberGraphNode: + return `/fiber/graph/node/${attributes.nodeId}` + default: break } @@ -97,4 +100,7 @@ export const getDisplayNameByAggregateSearchResult = (result: AggregateSearchRes if (type === SearchResultType.BtcAddress) { return attributes.addressHash } + if (type === SearchResultType.FiberGraphNode) { + return attributes.peerId + } } diff --git a/src/constants/fiberChainHash.ts b/src/constants/fiberChainHash.ts new file mode 100644 index 000000000..eaad4f749 --- /dev/null +++ b/src/constants/fiberChainHash.ts @@ -0,0 +1,4 @@ +export const ChainHash = new Map([ + ['0x0000000000000000000000000000000000000000000000000000000000000000', 'CKB Testnet'], + ['0x10639e0895502b5688a6be8cf69460d76541bfa4821629d86d62ba0aae3f9606', 'CKB Testnet'], +]) diff --git a/src/locales/en.json b/src/locales/en.json index a2b804a44..fe766623a 100644 --- a/src/locales/en.json +++ b/src/locales/en.json @@ -177,7 +177,7 @@ "sUDT": "sUDT", "inscriptions": "Inscriptions", "docs": "Docs", - "search_placeholder": "Block/Transaction/Address/Script Hash/Args/BTC Address/BTC Txid/DID", + "search_placeholder": "Block/Transaction/Address/Script Hash/Args/BTC Address/BTC Txid/DID/Fiber Peer Id/Fiber Node Id", "search_by_name_placeholder": "Token Name", "more": "More", "mainnet": "LINA", @@ -377,7 +377,7 @@ "loading": "Loading...", "no_search_result": "Oops! Your search did not match any record.", "empty_result": "Oops! Your search did not match any record. \n\nPlease make sure input contains only one of the following items:\n", - "empty_result_items": "Block Number/ Block Hash/ Transaction Hash/ Address/ Script Hash/ Args/ BTC Address/ BTC Txid/DID", + "empty_result_items": "Block Number/ Block Hash/ Transaction Hash/ Address/ Script Hash/ Args/ BTC Address/ BTC Txid/DID/Fiber Peer Id/Fiber Node Id", "address_type_testnet_error": "Testnet address detected,please goto", "address_type_mainnet_error": "Mainnet address detected,please goto", "address_type_testnet_url": "testnet explorer", @@ -401,7 +401,8 @@ "bitcoin_address": "BTC Address", "token_collection": "Token Collection", "token_item": "Token Item", - "did": "DID" + "did": "DID", + "fiber_graph_node": "Fiber Graph Node" }, "cell": { "live_cell": "Live Cell", @@ -1085,6 +1086,51 @@ "view_address": "View Address", "view_btc_utxo": "View BTC UTXO", "view_cell_info": "View Cell Info" + }, + "fiber": { + "peer": { + "peer_id": "ID", + "name": "Name", + "channels_count": "Channels", + "channels": "Channels", + "open_time": "First Channel", + "update_time": "Last Update", + "total_local_balance": "Local Balance", + "rpc_addr": "RPC Address", + "connect_id": "Connect" + }, + "channel": { + "channel_id": "Channel ID", + "fiber_peers": "Fiber Peers", + "state": "State", + "open_time": "Open Time", + "update_time": "Last Update Time", + "shutdown_time": "Shutdown Time", + "balance": "Balance", + "local": "Local", + "remote": "Remote", + "tlc_balance": "TLC Balance", + "offered": "Offered", + "received": "Received" + }, + "graph": { + "node": { + "id": "Node ID", + "name": "Name", + "alias": "Alias", + "auto_accept_funding_amount": "Auto-accepting Threshold", + "first_seen": "First Seen", + "node_id": "Node ID", + "chain": "Chain", + "chain_hash": "Chain Hash", + "addresses": "Addresses", + "total_capacity": "Capacity", + "open_channels": "Open Channels" + }, + "channel": { + "connected_node": "Connected Node" + } + } } } } diff --git a/src/locales/zh.json b/src/locales/zh.json index fae856f2b..605431e21 100644 --- a/src/locales/zh.json +++ b/src/locales/zh.json @@ -416,7 +416,8 @@ "bitcoin_address": "BTC 地址", "token_collection": "藏品集", "token_item": "藏品", - "did": "分布式数字身份(DID)" + "did": "分布式数字身份(DID)", + "fiber_graph_node": "Fiber Graph Node" }, "address": { "address": "地址", diff --git a/src/pages/Fiber/Channel/fiber.module.scss b/src/pages/Fiber/Channel/fiber.module.scss new file mode 100644 index 000000000..d2e8e270a --- /dev/null +++ b/src/pages/Fiber/Channel/fiber.module.scss @@ -0,0 +1,11 @@ +.container { + display: flex; + flex-direction: column; + + .id { + display: flex; + align-items: center; + gap: 4px; + font-size: 14px; + } +} diff --git a/src/pages/Fiber/Channel/fiber.tsx b/src/pages/Fiber/Channel/fiber.tsx new file mode 100644 index 000000000..b0c9462b5 --- /dev/null +++ b/src/pages/Fiber/Channel/fiber.tsx @@ -0,0 +1,25 @@ +import { CopyIcon } from '@radix-ui/react-icons' +import { Tooltip } from 'antd' +import { Link } from 'react-router-dom' +import type { Fiber } from '../../../services/ExplorerService/fetcher' +import styles from './fiber.module.scss' + +const FiberPeerInfo = ({ peer }: { peer: Fiber.Channel.Peer }) => { + return ( +
+ + {peer.name || 'Untitled Node'} + +
+ + {`${peer.peerId.slice(0, 8)}...${peer.peerId.slice(-8)}`} + + +
+
+ ) +} + +export default FiberPeerInfo diff --git a/src/pages/Fiber/Channel/index.module.scss b/src/pages/Fiber/Channel/index.module.scss new file mode 100644 index 000000000..9d063904b --- /dev/null +++ b/src/pages/Fiber/Channel/index.module.scss @@ -0,0 +1,114 @@ +@import '../../../styles/variables.module'; +@import '../../../styles/card.module'; + +.container { + text-wrap: nowrap; + display: flex; + flex-direction: column; + align-items: stretch; + margin: 24px 120px; + font-size: 1rem; + + a { + color: var(--primary-color); + } + + svg { + pointer-events: none; + } + + dl { + display: flex; + + dt, + dd { + display: flex; + align-items: center; + gap: 4px; + margin: 0; + padding: 0; + } + + dt::after { + content: ':'; + margin-right: 4px; + } + } + + button { + display: flex; + align-items: center; + appearance: none; + padding: 0; + border: none; + background: none; + cursor: pointer; + + &:hover { + color: var(--primary-color); + } + } + + .overview { + @extend %base-card; + } + + .id { + overflow: hidden; + + & > span:first-child { + overflow: hidden; + text-overflow: ellipsis; + flex-shrink: 1; + } + } + + .transactions { + @extend %base-card; + + margin-top: 8px; + padding-top: 16px; + + h3 { + margin: 0; + padding: 0; + } + } + + .peers { + margin-top: 8px; + display: flex; + flex-wrap: wrap; + + .local, + .remote { + @extend %base-card; + + flex: 1 0 40%; + border: 1px solid #ccc; + padding: 32px; + + dl:first-child { + dt { + align-items: flex-start; + } + + a { + font-weight: 400; + } + } + + dl:last-child { + margin: 0; + } + } + } + + @media screen and (width < $extraLargeBreakPoint) { + margin: 24px 20px; + } + + @media screen and (width < 1030px) { + font-size: 14px; + } +} diff --git a/src/pages/Fiber/Channel/index.tsx b/src/pages/Fiber/Channel/index.tsx new file mode 100644 index 000000000..d5787feda --- /dev/null +++ b/src/pages/Fiber/Channel/index.tsx @@ -0,0 +1,150 @@ +import { useTranslation } from 'react-i18next' +import { useParams } from 'react-router-dom' +import { useQuery } from '@tanstack/react-query' +import { CopyIcon } from '@radix-ui/react-icons' +import BigNumber from 'bignumber.js' +import dayjs from 'dayjs' +import Content from '../../../components/Content' +import { useSetToast } from '../../../components/Toast' +import Loading from '../../../components/Loading' +import { explorerService } from '../../../services/ExplorerService' +import styles from './index.module.scss' +import { shannonToCkb } from '../../../utils/util' +import { localeNumberString } from '../../../utils/number' +import FiberPeerInfo from './fiber' + +const TIME_TEMPLATE = 'YYYY/MM/DD hh:mm:ss' + +const Channel = () => { + const [t] = useTranslation() + const { id } = useParams<{ id: string }>() + const setToast = useSetToast() + + const { data, isLoading } = useQuery({ + queryKey: ['fiber', 'channels', id], + queryFn: () => { + return explorerService.api.getFiberChannel(id) + }, + enabled: !!id, + }) + + if (isLoading) { + return + } + + if (!data) { + return
Fiber Peer Not Found
+ } + const channel = data.data + + const handleCopy = (e: React.SyntheticEvent) => { + const elm = e.target + if (!(elm instanceof HTMLElement)) return + const { copyText } = elm.dataset + if (!copyText) return + e.stopPropagation() + e.preventDefault() + navigator?.clipboard.writeText(copyText).then(() => setToast({ message: t('common.copied') })) + } + + const totalBalance = BigNumber(channel.localBalance).plus(BigNumber(channel.remoteBalance)) + const totalTLCBalance = BigNumber(channel.offeredTlcBalance).plus(BigNumber(channel.receivedTlcBalance)) + + return ( + +
+
+
+
+
{t('fiber.channel.channel_id')}
+
+ {channel.channelId} + +
+
+
+
{t('fiber.channel.state')}
+
{channel.stateName}
+
+ +
+
{t('fiber.channel.balance')}
+
{`${localeNumberString( + shannonToCkb(totalBalance.toFormat({ groupSeparator: '' })), + )} CKB(Total) | ${localeNumberString( + shannonToCkb(totalTLCBalance.toFormat({ groupSeparator: '' })), + )} CKB(TLC)`}
+
+
+
{t('fiber.channel.open_time')}
+
+ +
+
+
+
{t('fiber.channel.update_time')}
+
+ +
+
+ {channel.shutdownAt ? ( +
+
{t('fiber.channel.shutdown_time')}
+
+ +
+
+ ) : null} +
+
+
+
+
Fiber Peer
+
+ +
+
+ +
+
{t('fiber.channel.balance')}
+
{`${localeNumberString(shannonToCkb(channel.localBalance))} CKB`}
+
+ +
+
{t('fiber.channel.tlc_balance')}
+
{`${localeNumberString(shannonToCkb(channel.offeredTlcBalance))} CKB`}
+
+
+ +
+
+
Fiber Peer
+
+ +
+
+ +
+
{t('fiber.channel.balance')}
+
{`${localeNumberString(shannonToCkb(channel.remoteBalance))} CKB`}
+
+ +
+
{t('fiber.channel.tlc_balance')}
+
{`${localeNumberString(shannonToCkb(channel.receivedTlcBalance))} CKB`}
+
+
+
+
+
+

Open | Close Transactions

+
Coming soon
+
+
+
+ ) +} + +export default Channel diff --git a/src/pages/Fiber/GraphChannelList/index.module.scss b/src/pages/Fiber/GraphChannelList/index.module.scss new file mode 100644 index 000000000..48060a63e --- /dev/null +++ b/src/pages/Fiber/GraphChannelList/index.module.scss @@ -0,0 +1,46 @@ +@import '../../../styles/variables.module'; +@import '../../../styles/table.module'; + +.container { + margin: 24px 120px; + font-size: 1rem; + + .channels { + border-radius: 6px; + box-shadow: rgb(0 0 0 / 12%) 0 2px 6px 0; + overflow: hidden; + } + + button { + display: flex; + align-items: center; + appearance: none; + padding: 0; + border: none; + background: none; + cursor: pointer; + + &:hover { + color: var(--primary-color); + } + } + + .header { + font-size: 1.5rem; + margin-bottom: 20px; + } + + .pagination { + background: #fff; + padding: 8px 40px; + margin-top: 4px; + } + + @media screen and (width < $extraLargeBreakPoint) { + margin: 24px 20px; + } + + @media screen and (width < 1330px) { + font-size: 14px; + } +} diff --git a/src/pages/Fiber/GraphChannelList/index.tsx b/src/pages/Fiber/GraphChannelList/index.tsx new file mode 100644 index 000000000..9d928ab8b --- /dev/null +++ b/src/pages/Fiber/GraphChannelList/index.tsx @@ -0,0 +1,53 @@ +import { useQuery } from '@tanstack/react-query' +import { useTranslation } from 'react-i18next' +import Content from '../../../components/Content' +import { useSetToast } from '../../../components/Toast' +import { explorerService } from '../../../services/ExplorerService' +import styles from './index.module.scss' +import Pagination from '../Pagination' +import { PAGE_SIZE } from '../../../constants/common' +import GraphChannelListComp from '../../../components/GraphChannelList' +import { useSearchParams } from '../../../hooks' + +const GraphChannelList = () => { + const [t] = useTranslation() + const setToast = useSetToast() + const { page = 1, page_size: pageSize = PAGE_SIZE } = useSearchParams('page', 'page_size') + + const { data } = useQuery({ + queryKey: ['fiber', 'graph', 'channels', +page, +pageSize], + queryFn: () => explorerService.api.getGraphChannels(+page, +pageSize), + }) + + const list = data?.data.fiberGraphChannels ?? [] + const pageInfo = data?.meta ?? { total: 1, pageSize: PAGE_SIZE } + const totalPages = Math.ceil(pageInfo.total / pageInfo.pageSize) + + const handleCopy = (e: React.SyntheticEvent) => { + const elm = e.target + if (!(elm instanceof HTMLElement)) return + const { copyText } = elm.dataset + if (!copyText) return + e.stopPropagation() + e.preventDefault() + navigator?.clipboard.writeText(copyText).then(() => setToast({ message: t('common.copied') })) + } + + return ( + +
+

+ CKB Fiber Graph Channels +

+
+ +
+ +
+
+
+
+ ) +} + +export default GraphChannelList diff --git a/src/pages/Fiber/GraphNode/index.module.scss b/src/pages/Fiber/GraphNode/index.module.scss new file mode 100644 index 000000000..d93a43d4f --- /dev/null +++ b/src/pages/Fiber/GraphNode/index.module.scss @@ -0,0 +1,231 @@ +@import '../../../styles/variables.module'; +@import '../../../styles/card.module'; +@import '../../../styles/text.module'; + +.container { + text-wrap: nowrap; + display: flex; + flex-direction: column; + align-items: stretch; + margin: 24px 120px; + font-size: 1rem; + + a { + color: var(--primary-color); + } + + dl { + display: flex; + + dt, + dd { + display: flex; + align-items: center; + gap: 4px; + margin: 0; + padding: 0; + } + + dt::after { + content: ':'; + margin-right: 4px; + } + } + + table { + width: 100%; + text-align: left; + cursor: default; + + td, + th { + padding: 8px; + padding-right: 16px; + + &:last-child { + text-align: right; + } + } + + tbody { + tr:hover { + background: #ccc; + } + } + } + + svg { + pointer-events: none; + } + + button { + display: flex; + align-items: center; + appearance: none; + padding: 0; + border: none; + background: none; + cursor: pointer; + + &:hover { + color: var(--primary-color); + } + } + + .overview { + @extend %base-card; + + display: flex; + justify-content: space-between; + flex-wrap: wrap; + + .fields { + overflow: hidden; + } + } + + .id, + .connectId { + overflow: hidden; + + & > span:first-child { + overflow: hidden; + text-overflow: ellipsis; + flex-shrink: 1; + } + } + + .thresholds { + dt { + align-items: start; + } + + dd { + margin-left: 8px; + display: flex; + flex-direction: column; + align-items: flex-start; + + img { + flex: 0 0; + } + + .token { + display: flex; + gap: 8px; + align-items: center; + } + } + } + + .activities { + display: flex; + gap: 16px; + margin-top: 16px; + + .channels, + .transactions { + overflow: hidden; + flex-basis: 50%; + background: #fff; + border-radius: 6px; + padding: 16px; + box-shadow: 0 2px 6px 0 #4d4d4d33; + font-size: 0.8em; + + * { + font-size: inherit; + } + + h3 { + margin: 0; + padding: 0; + } + } + + @media screen and (width < $mobileBreakPoint) { + flex-direction: column; + } + + @media screen and (width < 500px) { + thead { + display: none; + } + + tbody { + tr { + display: flex; + flex-direction: column; + padding: 16px 0; + + &:not(:last-child) { + border-bottom: 1px solid #ccc; + } + + td { + text-align: left; + padding: 0; + } + } + } + } + } + + @media screen and (width < $extraLargeBreakPoint) { + margin: 24px 20px; + } + + @media screen and (width < 1030px) { + font-size: 14px; + } +} + +.addresses { + select { + max-width: 80%; + overflow: hidden; + text-overflow: ellipsis; + } +} + +.tx { + padding: 8px 40px; + display: flex; + flex-direction: column; + + @media screen and (width < $extraLargeBreakPoint) { + padding: 8px; + } + + time { + margin-right: auto; + } + + & > div { + display: flex; + align-items: center; + gap: 4px; + } + + .addr { + @extend %hash; + } + + a { + @extend %monospace; + + display: flex; + align-items: center; + overflow: hidden; + user-select: none; + + div { + font-family: inherit; + } + + div:first-child { + overflow: hidden; + text-overflow: ellipsis; + } + } +} diff --git a/src/pages/Fiber/GraphNode/index.tsx b/src/pages/Fiber/GraphNode/index.tsx new file mode 100644 index 000000000..cc3ff4390 --- /dev/null +++ b/src/pages/Fiber/GraphNode/index.tsx @@ -0,0 +1,340 @@ +import { useEffect, useMemo, useRef, useState } from 'react' +import { useTranslation } from 'react-i18next' +import { useParams } from 'react-router-dom' +import { useQuery } from '@tanstack/react-query' +import { CopyIcon, Link1Icon, LinkBreak1Icon, OpenInNewWindowIcon } from '@radix-ui/react-icons' +import { Tooltip } from 'antd' +import QRCode from 'qrcode' +import dayjs from 'dayjs' +import Content from '../../../components/Content' +import { explorerService } from '../../../services/ExplorerService' +import { useSetToast } from '../../../components/Toast' +import styles from './index.module.scss' +import Loading from '../../../components/Loading' +import GraphChannelList from '../../../components/GraphChannelList' +import { getFundingThreshold } from '../utils' +import { shannonToCkb } from '../../../utils/util' +import { parseNumericAbbr } from '../../../utils/chart' +import { ChainHash } from '../../../constants/fiberChainHash' +import { Link } from '../../../components/Link' +import { localeNumberString } from '../../../utils/number' + +const TIME_TEMPLATE = 'YYYY/MM/DD hh:mm:ss' + +const GraphNode = () => { + const [t] = useTranslation() + const [addr, setAddr] = useState('') + const { id } = useParams<{ id: string }>() + const qrRef = useRef(null) + + const setToast = useSetToast() + + const { data, isLoading } = useQuery({ + queryKey: ['fiber', 'graph', 'node', id], + queryFn: () => { + return explorerService.api.getGraphNodeDetail(id) + }, + enabled: !!id, + }) + + const node = data?.data + + const connectId = addr + + const handleAddrSelect = (e: React.ChangeEvent) => { + e.stopPropagation() + e.preventDefault() + const r = e.currentTarget.value + if (r) { + setAddr(r) + } + } + + useEffect(() => { + const firstAddr = node?.addresses[0] + if (firstAddr) { + setAddr(firstAddr) + } + }, [node, setAddr]) + + useEffect(() => { + const cvs = qrRef.current + if (!cvs || !connectId) return + QRCode.toCanvas( + cvs, + connectId, + { + margin: 5, + errorCorrectionLevel: 'H', + width: 144, + }, + err => { + if (err) { + console.error(err) + } + }, + ) + }, [qrRef, connectId]) + + const openAndClosedTxs = useMemo(() => { + const list: { + hash: string + index?: string + block: { + number: number + timestamp: number + } + isUdt: boolean + isOpen: boolean + accounts: Record<'amount' | 'address', string>[] + }[] = [] + + if (!node?.fiberGraphChannels) return list + + node.fiberGraphChannels.forEach(c => { + const isUdt = !!c.openTransactionInfo.udtAmount + const open = { + isOpen: true, + isUdt, + hash: c.openTransactionInfo.txHash, + index: c.fundingTxIndex, + block: { + number: c.openTransactionInfo.blockNumber, + timestamp: c.openTransactionInfo.blockTimestamp, + }, + accounts: [ + { + address: c.openTransactionInfo.address, + amount: + c.openTransactionInfo.udtAmount ?? + `${localeNumberString(shannonToCkb(c.openTransactionInfo.capacity))} CKB`, + }, + ], + } + + list.push(open) + + const close = c.closedTransactionInfo?.txHash + ? { + isOpen: false, + hash: c.closedTransactionInfo.txHash, + block: { + number: c.closedTransactionInfo.blockNumber, + timestamp: c.closedTransactionInfo.blockTimestamp, + }, + isUdt, + accounts: c.closedTransactionInfo.closeAccounts.map(acc => { + return { + amount: acc.udtAmount ?? `${localeNumberString(shannonToCkb(acc.capacity))} CKB`, + address: acc.address, + } + }), + } + : null + if (close) { + list.push(close) + } + }) + return list.sort((a, b) => a.block.timestamp - b.block.timestamp) + }, [node]) + + if (isLoading) { + return + } + + if (!node) { + return
Fiber Peer Not Found
+ } + const channels = node.fiberGraphChannels.filter(c => !c.closedTransactionInfo) + + const thresholds = getFundingThreshold(node) + + const totalCkb = parseNumericAbbr(shannonToCkb(node.totalCapacity)) + + const handleCopy = (e: React.SyntheticEvent) => { + const elm = e.target + if (!(elm instanceof HTMLElement)) return + const { copyText } = elm.dataset + if (!copyText) return + e.stopPropagation() + e.preventDefault() + navigator?.clipboard.writeText(copyText).then(() => setToast({ message: t('common.copied') })) + } + + const chain = ChainHash.get(node.chainHash) ?? '-' + + return ( + +
+
+
+ {node.alias ? ( +
+
{t('fiber.graph.alias')}
+
+ {node.alias} + +
+
+ ) : null} +
+
{t('fiber.graph.node.id')}
+
+ {`0x${node.nodeId}`} + +
+
+
+
+ +
+
+ + + + + +
+
+
+
{t('fiber.graph.node.first_seen')}
+
{dayjs(+node.timestamp).format(TIME_TEMPLATE)}
+
+
+
{t('fiber.graph.node.chain')}
+
+ {chain} +
+
+
+
{t('fiber.graph.node.total_capacity')}
+
{totalCkb}
+
+
+
{t('fiber.graph.node.auto_accept_funding_amount')}
+
+ {thresholds.map(threshold => { + return ( + + + icon + {threshold.display} + + + ) + })} +
+
+
+ {connectId ? ( +
+ +
+ ) : null} +
+
+
+

{`${t('fiber.peer.channels')}(${channels.length})`}

+ +
+
+

Open & Closed Transactions

+
+ {openAndClosedTxs.map(tx => { + const key = tx.isOpen ? `${tx.hash}#${tx.index}` : tx.hash + if (tx.isOpen) { + const account = tx.accounts[0]! + return ( +
+
+ + at + + + + + + +
+
+ By + + + +
{account.address.slice(0, -8)}
+
{account.address.slice(-8)}
+ +
+
+ ({account.amount}) +
+
+ ) + } + const [acc1, acc2] = tx.accounts + return ( +
+
+ + at + + + + + + +
+
+ To + + + +
{acc1.address.slice(0, -8)}
+
{acc1.address.slice(-8)}
+ +
+
+ ({acc1.amount}) +
+
+ And + + + +
{acc2.address.slice(0, -8)}
+
{acc2.address.slice(-8)}
+ +
+
+ ({acc2.amount}) +
+
+ ) + })} +
+
+
+
+
+ ) +} + +export default GraphNode diff --git a/src/pages/Fiber/GraphNodeList/index.module.scss b/src/pages/Fiber/GraphNodeList/index.module.scss new file mode 100644 index 000000000..4356a5086 --- /dev/null +++ b/src/pages/Fiber/GraphNodeList/index.module.scss @@ -0,0 +1,167 @@ +@import '../../../styles/variables.module'; +@import '../../../styles/table.module'; + +.container { + text-wrap: nowrap; + display: flex; + flex-direction: column; + margin: 24px 120px; + font-size: 1rem; + + a { + color: var(--primary-color); + } + + table { + @extend %base-table; + + tr[data-role='pagination']:hover { + background: #fff; + } + } + + svg { + pointer-events: none; + } + + button { + display: flex; + align-items: center; + appearance: none; + padding: 0; + border: none; + background: none; + cursor: pointer; + + &:hover { + color: var(--primary-color); + } + } + + .name { + max-width: 100px; + overflow: hidden; + text-overflow: ellipsis; + } + + .funding { + display: flex; + flex-direction: column; + + .token { + display: flex; + gap: 8px; + align-items: center; + } + } + + .nodeId, + .chainHash { + display: flex; + gap: 4px; + } + + .address { + display: flex; + align-items: center; + justify-content: flex-end; + flex-wrap: nowrap; + gap: 4px; + + & > span:first-child { + display: block; + max-width: 240px; + overflow: hidden; + text-overflow: ellipsis; + } + + button, + a, + .more { + display: flex; + align-items: center; + } + } + + .header { + display: flex; + justify-content: space-between; + align-items: center; + font-size: 1.5rem; + margin-bottom: 20px; + + button { + font-size: 0.875rem; + color: var(--primary-color); + padding-left: 8px; + } + } + + .amount { + display: flex; + flex-direction: column; + } + + @media screen and (width < $extraLargeBreakPoint) { + margin: 24px 20px; + } + + @media screen and (width < 1330px) { + font-size: 14px; + + table { + th, + td { + &:nth-child(6) { + display: none; + } + } + } + } + + @media screen and (width < 810px) { + table { + tr:not([data-role='pagination']) { + th, + td { + &:last-child { + display: none; + } + } + } + } + } + + @media screen and (width < 900px) { + table { + th, + td { + &:nth-child(5) { + display: none; + } + } + } + } + + @media screen and (width < 700px) { + table { + th, + td { + &:nth-child(3) { + display: none; + } + } + } + } + + @media screen and (width < 520px) { + table { + th, + td { + &:first-child { + display: none; + } + } + } + } +} diff --git a/src/pages/Fiber/GraphNodeList/index.tsx b/src/pages/Fiber/GraphNodeList/index.tsx new file mode 100644 index 000000000..0dcc283d4 --- /dev/null +++ b/src/pages/Fiber/GraphNodeList/index.tsx @@ -0,0 +1,221 @@ +import { useQuery } from '@tanstack/react-query' +import { useTranslation } from 'react-i18next' +import { Link } from 'react-router-dom' +import { Tooltip } from 'antd' +import { CopyIcon, InfoCircledIcon } from '@radix-ui/react-icons' +import dayjs from 'dayjs' +import Content from '../../../components/Content' +import { useSetToast } from '../../../components/Toast' +import { explorerService } from '../../../services/ExplorerService' +import type { Fiber } from '../../../services/ExplorerService/fetcher' +import Pagination from '../Pagination' +import { PAGE_SIZE } from '../../../constants/common' +import { useSearchParams } from '../../../hooks' +import { getFundingThreshold } from '../utils' +import styles from './index.module.scss' +import { shannonToCkb } from '../../../utils/util' +import { parseNumericAbbr } from '../../../utils/chart' +import { localeNumberString } from '../../../utils/number' +import { ChainHash } from '../../../constants/fiberChainHash' + +const TIME_TEMPLATE = 'YYYY/MM/DD hh:mm:ss' + +const fields = [ + { + key: 'alias', + label: 'alias', + transformer: (v: unknown, i: Fiber.Graph.Node) => { + if (typeof v !== 'string') return v + return ( + +
+ {v || Untitled} +
+
+ ) + }, + }, + { + key: 'autoAcceptMinCkbFundingAmount', + label: 'auto_accept_funding_amount', + transformer: (_: unknown, n: Fiber.Graph.Node) => { + const thresholds = getFundingThreshold(n) + + return ( +
+ {thresholds.map(threshold => { + return ( + + + icon + {threshold.display} + + + ) + })} +
+ ) + }, + }, + { + key: 'totalCapacity', + label: 'total_capacity', + transformer: (v: unknown) => { + if (typeof v !== 'string') return v + + const ckb = shannonToCkb(v) + const amount = parseNumericAbbr(ckb) + return ( + + {`${amount} CKB`} + + ) + }, + }, + { + key: 'openChannelsCount', + label: 'open_channels', + transformer: (v: unknown) => { + if (typeof v !== 'string') return v + + return localeNumberString(v) + }, + }, + { + key: 'timestamp', + label: 'first_seen', + transformer: (v: unknown) => { + if (typeof v !== 'string') return v + return dayjs(+v).format(TIME_TEMPLATE) + }, + }, + { + key: 'nodeId', + label: 'node_id', + transformer: (v: unknown) => { + if (typeof v !== 'string') return v + return ( + + + + {v.length > 16 ? `0x${v.slice(0, 8)}...${v.slice(-8)}` : `0x${v}`} + + + + + ) + }, + }, + { + key: 'chainHash', + label: 'chain', + transformer: (v: unknown) => { + if (typeof v !== 'string') return v + const chain = ChainHash.get(v) ?? '-' + return ( + + {chain} + + + ) + }, + }, + { + key: 'addresses', + label: 'addresses', + transformer: (v: unknown) => { + if (!Array.isArray(v)) return v + const addr = v[0] + if (!addr || typeof addr !== 'string') return v + return ( + + + {addr} + + + {/* */} + {/* */} + {/* */} + {v.length > 1 ? ( + + + + + + ) : null} + + ) + }, + }, +] + +const GraphNodeList = () => { + const [t] = useTranslation() + const setToast = useSetToast() + const { page = 1, page_size: pageSize = PAGE_SIZE } = useSearchParams('page', 'page_size') + + const { data } = useQuery({ + queryKey: ['fiber', 'graph', 'nodes', +page, +pageSize], + queryFn: () => explorerService.api.getGraphNodes(+page, +pageSize), + }) + + const list = data?.data.fiberGraphNodes ?? [] + const pageInfo = data?.meta ?? { total: 1, pageSize: PAGE_SIZE } + const totalPages = Math.ceil(pageInfo.total / pageInfo.pageSize) + + const handleCopy = (e: React.SyntheticEvent) => { + const elm = e.target + if (!(elm instanceof HTMLElement)) return + const { copyText } = elm.dataset + if (!copyText) return + e.stopPropagation() + e.preventDefault() + navigator?.clipboard.writeText(copyText).then(() => setToast({ message: t('common.copied') })) + } + + return ( + +
+

+ CKB Fiber Graph Nodes +

+ + + + {fields.map(f => { + return + })} + + +
+
+ {list.map(i => { + return ( + + {fields.map(f => { + const v = i[f.key as keyof typeof i] + return + })} + + ) + })} +
+
+ + + +
{t(`fiber.graph.node.${f.label}`)}
{f.transformer?.(v, i) ?? v}
+ +
+
+
+ ) +} + +export default GraphNodeList diff --git a/src/pages/Fiber/Pagination/index.module.scss b/src/pages/Fiber/Pagination/index.module.scss new file mode 100644 index 000000000..2a189997a --- /dev/null +++ b/src/pages/Fiber/Pagination/index.module.scss @@ -0,0 +1,95 @@ +@import '../../../styles/variables.module'; + +.container { + display: flex; + background: #fff; + justify-content: space-between; + height: 34px; + + form { + display: flex; + gap: 20px; + + label { + display: flex; + align-items: center; + } + + button { + display: flex; + justify-content: center; + width: 50px; + background: #f5f5f5; + border-radius: 6px; + } + + input { + width: 100; + padding: 0 10px; + color: #969696; + border-radius: 6px; + background: #f5f5f5; + border: none; + } + } + + .pager { + display: flex; + gap: 20px; + + .pageNo { + display: flex; + align-items: center; + } + } + + a { + display: flex; + justify-content: center; + align-items: center; + color: #969696; + + &:hover { + color: var(--primary-color); + } + + &[aria-disabled='true'] { + pointer-events: none; + opacity: 0.5; + } + + &[data-role='first-page'], + &[data-role='last-page'] { + width: 50px; + background: #f5f5f5; + border-radius: 6px; + } + + &[data-role='prev-page'], + &[data-role='next-page'] { + width: 30px; + height: 30px; + background: #f5f5f5; + border-radius: 6px; + } + } + + @media screen and (width < $mobileBreakPoint) { + font-size: 12px; + + .pager { + gap: 4px; + } + + a { + &[data-role='first-page'], + &[data-role='last-page'] { + display: none; + } + } + + .pageNo { + order: 3; + } + } +} diff --git a/src/pages/Fiber/Pagination/index.tsx b/src/pages/Fiber/Pagination/index.tsx new file mode 100644 index 000000000..5e60d5cba --- /dev/null +++ b/src/pages/Fiber/Pagination/index.tsx @@ -0,0 +1,68 @@ +import React from 'react' +import { Link, useHistory } from 'react-router-dom' +import { ChevronLeftIcon, ChevronRightIcon } from '@radix-ui/react-icons' +import { useSearchParams } from '../../../hooks' +import styles from './index.module.scss' + +interface PaginationProps { + totalPages: number +} + +const getPageUrl = (page: number, search: URLSearchParams) => { + search.set('page', page.toString()) + return `${window.location.pathname}?${search.toString()}` +} + +const Pagination: React.FC = ({ totalPages }) => { + const history = useHistory() + const { page: p } = useSearchParams('page') + + // Get the current page from the URL query parameter, defaulting to 1 if not set + const currentPage = Number(p) || 1 + const search = new URLSearchParams(window.location.search) + + const handleGo = (e: React.SyntheticEvent) => { + e.stopPropagation() + e.preventDefault() + + const { page } = e.currentTarget + if (!(page instanceof HTMLInputElement)) return + const go = page.value + if (+go < 1) { + history.push(getPageUrl(1, search)) + return + } + if (+go > totalPages) { + history.push(getPageUrl(totalPages, search)) + return + } + history.push(getPageUrl(+go, search)) + } + + return ( +
+
+ + First + + + + + {`Page ${currentPage} of ${totalPages}`} + + + + + Last + +
+
+ + + +
+
+ ) +} + +export default Pagination diff --git a/src/pages/Fiber/Peer/index.module.scss b/src/pages/Fiber/Peer/index.module.scss new file mode 100644 index 000000000..ace74b7cd --- /dev/null +++ b/src/pages/Fiber/Peer/index.module.scss @@ -0,0 +1,151 @@ +@import '../../../styles/variables.module'; +@import '../../../styles/card.module'; + +.container { + text-wrap: nowrap; + display: flex; + flex-direction: column; + align-items: stretch; + margin: 24px 120px; + font-size: 1rem; + + a { + color: var(--primary-color); + } + + dl { + display: flex; + + dt, + dd { + display: flex; + align-items: center; + gap: 4px; + margin: 0; + padding: 0; + } + + dt::after { + content: ':'; + margin-right: 4px; + } + } + + table { + width: 100%; + text-align: left; + cursor: default; + + td, + th { + padding: 8px; + padding-right: 16px; + + &:last-child { + text-align: right; + } + } + + tbody { + tr:hover { + background: #ccc; + } + } + } + + svg { + pointer-events: none; + } + + button { + display: flex; + align-items: center; + appearance: none; + padding: 0; + border: none; + background: none; + cursor: pointer; + + &:hover { + color: var(--primary-color); + } + } + + .overview { + @extend %base-card; + + display: flex; + justify-content: space-between; + flex-wrap: wrap; + + .fields { + overflow: hidden; + } + } + + .id, + .connectId { + overflow: hidden; + + & > span:first-child { + overflow: hidden; + text-overflow: ellipsis; + flex-shrink: 1; + } + } + + .activities { + display: flex; + gap: 16px; + margin-top: 16px; + + .channels, + .transactions { + flex: 1; + background: #fff; + border-radius: 6px; + padding: 16px; + box-shadow: 0 2px 6px 0 #4d4d4d33; + + h3 { + margin: 0; + padding: 0; + } + } + + @media screen and (width < 960px) { + flex-direction: column; + } + + @media screen and (width < 500px) { + thead { + display: none; + } + + tbody { + tr { + display: flex; + flex-direction: column; + padding: 16px 0; + + &:not(:last-child) { + border-bottom: 1px solid #ccc; + } + + td { + text-align: left; + padding: 0; + } + } + } + } + } + + @media screen and (width < $extraLargeBreakPoint) { + margin: 24px 20px; + } + + @media screen and (width < 1030px) { + font-size: 14px; + } +} diff --git a/src/pages/Fiber/Peer/index.tsx b/src/pages/Fiber/Peer/index.tsx new file mode 100644 index 000000000..e096dee4d --- /dev/null +++ b/src/pages/Fiber/Peer/index.tsx @@ -0,0 +1,194 @@ +import { useEffect, useRef, useState } from 'react' +import { useTranslation } from 'react-i18next' +import { Link, useParams } from 'react-router-dom' +import { useQuery } from '@tanstack/react-query' +import { CopyIcon, OpenInNewWindowIcon } from '@radix-ui/react-icons' +import QRCode from 'qrcode' +import { Tooltip } from 'antd' +import Content from '../../../components/Content' +import { explorerService } from '../../../services/ExplorerService' +import { useSetToast } from '../../../components/Toast' +import styles from './index.module.scss' +import Loading from '../../../components/Loading' + +const Peer = () => { + const [t] = useTranslation() + const [rpcAddr, setRpcAddr] = useState('') + const { id } = useParams<{ id: string }>() + const qrRef = useRef(null) + + const setToast = useSetToast() + + const { data, isLoading } = useQuery({ + queryKey: ['fiber', 'peers', id], + queryFn: () => { + return explorerService.api.getFiberPeerDetail(id) + }, + enabled: !!id, + }) + + const peer = data?.data + + const connectId = peer && rpcAddr ? `${peer.peerId}@${rpcAddr}` : null + + const handleRpcAddrSelect = (e: React.ChangeEvent) => { + e.stopPropagation() + e.preventDefault() + const r = e.currentTarget.value + if (r) { + setRpcAddr(r) + } + } + + useEffect(() => { + const firstRpcAddr = peer?.rpcListeningAddr[0] + if (firstRpcAddr) { + setRpcAddr(firstRpcAddr) + } + }, [peer, setRpcAddr]) + + useEffect(() => { + const cvs = qrRef.current + if (!cvs || !connectId) return + QRCode.toCanvas( + cvs, + connectId, + { + margin: 5, + errorCorrectionLevel: 'H', + width: 144, + }, + err => { + if (err) { + console.error(err) + } + }, + ) + }, [qrRef, connectId]) + + if (isLoading) { + return + } + + if (!peer) { + return
Fiber Peer Not Found
+ } + const channels = peer.fiberChannels + + const handleCopy = (e: React.SyntheticEvent) => { + const elm = e.target + if (!(elm instanceof HTMLElement)) return + const { copyText } = elm.dataset + if (!copyText) return + e.stopPropagation() + e.preventDefault() + navigator?.clipboard.writeText(copyText).then(() => setToast({ message: t('common.copied') })) + } + + return ( + +
+
+
+
+
{t('fiber.peer.peer_id')}
+
+ {peer.peerId} + +
+
+
+
+ +
+
+ + + + + +
+
+ {connectId ? ( +
+
{t('fiber.peer.connect_id')}
+
+ + {connectId} + + +
+
+ ) : null} +
+
{t('fiber.peer.open_time')}
+
+ Coming soon +
+
+
+
{t('fiber.peer.update_time')}
+
+ Coming soon +
+
+
+ {connectId ? ( +
+ +
+ ) : null} +
+
+
+

{`${t('fiber.peer.channels')}(${channels.length})`}

+ + + + + + + + + {channels.map(c => { + return ( + + + + + ) + })} + +
{t('fiber.channel.channel_id')}{t('fiber.channel.state')}
+ + + {`${c.channelId.slice(0, 10)}...${c.channelId.slice(-10)}`} + + + {c.stateName}
+
+
+

Open | Close Transactions

+ Coming soon +
+
+
+
+ ) +} + +export default Peer diff --git a/src/pages/Fiber/PeerList/AddPeerForm.module.scss b/src/pages/Fiber/PeerList/AddPeerForm.module.scss new file mode 100644 index 000000000..2fa4288b5 --- /dev/null +++ b/src/pages/Fiber/PeerList/AddPeerForm.module.scss @@ -0,0 +1,74 @@ +.container { + font-size: 1rem; + + &::backdrop { + background: rgb(0 0 0 / 40%); + } + + form { + background: #fff; + position: fixed; + transform: translateX(-50%) translateY(-50%); + font-size: 0.875rem; + width: min-content; + margin: 16px auto; + top: 50%; + left: 50%; + padding: 23px 40px; + border-radius: 4px; + } + + h3 { + display: flex; + justify-content: space-between; + align-items: center; + padding-bottom: 22px; + border-bottom: 1px solid #e5e5e5; + margin-bottom: 16px; + + button { + svg { + color: #000; + } + } + } + + fieldset { + display: flex; + flex-direction: column; + margin-bottom: 16px; + } + + label { + display: flex; + align-items: center; + gap: 4px; + color: #666; + margin-bottom: 12px; + + &[data-required]::after { + content: '*'; + color: var(--accent-color); + } + } + + input { + width: 320px; + padding: 9px 12px; + border: 1px solid #e5e5e5; + border-radius: 4px; + font-size: inherit; + } + + button[type='submit'] { + padding: 14px 40px; + color: #fff; + background: var(--primary-color); + border-radius: 4px; + font-size: 1rem; + } + + @media screen and (width < 1030px) { + font-size: 14px; + } +} diff --git a/src/pages/Fiber/PeerList/AddPeerForm.tsx b/src/pages/Fiber/PeerList/AddPeerForm.tsx new file mode 100644 index 000000000..a0d2ae5d2 --- /dev/null +++ b/src/pages/Fiber/PeerList/AddPeerForm.tsx @@ -0,0 +1,105 @@ +import { Cross2Icon } from '@radix-ui/react-icons' +import { type FC, useRef, useEffect } from 'react' +import { useTranslation } from 'react-i18next' +import { useSetToast } from '../../../components/Toast' +import { explorerService } from '../../../services/ExplorerService' +import styles from './AddPeerForm.module.scss' + +interface AddPeerFormProps { + onSuccess: () => void +} +const AddPeerForm: FC = ({ onSuccess }) => { + const dialogRef = useRef(null) + const [t] = useTranslation() + const setToast = useSetToast() + + useEffect(() => { + const handleKeyDown = (e: KeyboardEvent) => { + if (e.key === 'Escape') { + dialogRef.current?.close() + } + } + + window.addEventListener('keydown', handleKeyDown) + + return () => { + window.removeEventListener('keydown', handleKeyDown) + } + }, []) + + const handleClose = () => { + dialogRef.current?.close() + } + + const handleSubmit = async (e: React.SyntheticEvent) => { + e.preventDefault() + e.stopPropagation() + const form = e.currentTarget + + const { peer_id, peer_name, rpc } = form + const params: Parameters[0] = { + rpc: rpc instanceof HTMLInputElement ? rpc.value : '', + id: peer_id instanceof HTMLInputElement ? peer_id.value : '', + name: peer_name instanceof HTMLInputElement ? peer_name.value : undefined, + } + + if (params.rpc && params.id) { + try { + await explorerService.api.addFiberPeer(params) + setToast({ message: 'submitted' }) + onSuccess() + } catch (e) { + const message = e instanceof Error ? e.message : JSON.stringify(e) + setToast({ message }) + } + } + } + + const handleClickOutside = (e: React.MouseEvent) => { + if (e.target === dialogRef.current) { + dialogRef.current?.close() + } + } + + const handleOpen = () => { + dialogRef.current?.showModal() + } + + return ( + <> + + + +
+

+ Add Fiber Peer + +

+
+ + +
+
+ + +
+
+ + +
+ +
+
+ + ) +} + +export default AddPeerForm diff --git a/src/pages/Fiber/PeerList/index.module.scss b/src/pages/Fiber/PeerList/index.module.scss new file mode 100644 index 000000000..25e11803e --- /dev/null +++ b/src/pages/Fiber/PeerList/index.module.scss @@ -0,0 +1,144 @@ +@import '../../../styles/variables.module'; +@import '../../../styles/table.module'; + +.container { + text-wrap: nowrap; + display: flex; + flex-direction: column; + margin: 24px 120px; + font-size: 1rem; + + a { + color: var(--primary-color); + } + + table { + @extend %base-table; + + tr[data-role='pagination']:hover { + background: #fff; + } + } + + svg { + pointer-events: none; + } + + button { + display: flex; + align-items: center; + appearance: none; + padding: 0; + border: none; + background: none; + cursor: pointer; + + &:hover { + color: var(--primary-color); + } + } + + .name { + max-width: 100px; + overflow: hidden; + text-overflow: ellipsis; + } + + .peerId { + display: flex; + gap: 4px; + } + + .rpc { + display: flex; + align-items: center; + justify-content: flex-end; + flex-wrap: nowrap; + gap: 4px; + + & > span:first-child { + display: block; + max-width: 500px; + overflow: hidden; + text-overflow: ellipsis; + } + + button, + a, + .more { + display: flex; + align-items: center; + } + } + + .header { + display: flex; + justify-content: space-between; + align-items: center; + font-size: 1.5rem; + margin-bottom: 20px; + + button { + font-size: 0.875rem; + color: var(--primary-color); + padding-left: 8px; + } + } + + .balance { + display: flex; + flex-direction: column; + } + + @media screen and (width < $extraLargeBreakPoint) { + margin: 24px 20px; + } + + @media screen and (width < 1030px) { + font-size: 14px; + + table { + th, + td { + &:nth-child(4) { + display: none; + } + } + } + } + + @media screen and (width < 810px) { + table { + tr:not([data-role='pagination']) { + th, + td { + &:last-child { + display: none; + } + } + } + } + } + + @media screen and (width < 600px) { + table { + th, + td { + &:nth-child(6) { + display: none; + } + } + } + } + + @media screen and (width < 420px) { + table { + th, + td { + &:nth-child(5) { + display: none; + } + } + } + } +} diff --git a/src/pages/Fiber/PeerList/index.tsx b/src/pages/Fiber/PeerList/index.tsx new file mode 100644 index 000000000..35aa9476a --- /dev/null +++ b/src/pages/Fiber/PeerList/index.tsx @@ -0,0 +1,188 @@ +import { useQuery } from '@tanstack/react-query' +import { useTranslation } from 'react-i18next' +import { Link } from 'react-router-dom' +import { Tooltip } from 'antd' +import { CopyIcon, InfoCircledIcon, OpenInNewWindowIcon } from '@radix-ui/react-icons' +import Content from '../../../components/Content' +import { useSetToast } from '../../../components/Toast' +import { explorerService } from '../../../services/ExplorerService' +import type { Fiber } from '../../../services/ExplorerService/fetcher' +import { shannonToCkb } from '../../../utils/util' +import { localeNumberString } from '../../../utils/number' +import { parseNumericAbbr } from '../../../utils/chart' +import styles from './index.module.scss' +import AddPeerForm from './AddPeerForm' +import Pagination from '../Pagination' +import { PAGE_SIZE } from '../../../constants/common' +import { useSearchParams } from '../../../hooks' + +const fields = [ + { + key: 'name', + label: 'name', + transformer: (v: unknown, i: Fiber.Peer.ItemInList) => { + if (typeof v !== 'string') return v + return ( + +
+ {v} +
+
+ ) + }, + }, + { + key: 'channelsCount', + label: 'channels_count', + transformer: (v: unknown) => { + if (typeof v !== 'number') return v + return localeNumberString(v) + }, + }, + { + key: 'totalLocalBalance', + label: 'total_local_balance', + transformer: (v: unknown) => { + if (typeof v !== 'string' || Number.isNaN(+v)) return v + const ckb = shannonToCkb(v) + const amount = parseNumericAbbr(ckb) + return ( +
+ + {`${amount} CKB`} + + Share: coming soon +
+ ) + }, + }, + { + key: 'firstChannelOpenedAt', + label: 'open_time', + transformer: () => { + return Coming soon + }, + }, + { + key: 'lastChannelUpdatedAt', + label: 'update_time', + transformer: () => { + return Coming soon + }, + }, + { + key: 'peerId', + label: 'peer_id', + transformer: (v: unknown) => { + if (typeof v !== 'string') return v + return ( + + + + {v.length > 16 ? `${v.slice(0, 8)}...${v.slice(-8)}` : v} + + + + + ) + }, + }, + { + key: 'rpcListeningAddr', + label: 'rpc_addr', + transformer: (v: unknown) => { + if (!Array.isArray(v)) return v + const rpcAddr = v[0] + if (!rpcAddr || typeof rpcAddr !== 'string') return v + return ( + + + {rpcAddr} + + + + + + {v.length > 1 ? ( + + + + + + ) : null} + + ) + }, + }, +] + +const PeerList = () => { + const [t] = useTranslation() + const setToast = useSetToast() + const { page = 1, page_size: pageSize = PAGE_SIZE } = useSearchParams('page', 'page_size') + + const { data, refetch: refetchList } = useQuery({ + queryKey: ['fiber', 'peers', +page, +pageSize], + queryFn: () => explorerService.api.getFiberPeerList(+page, +pageSize), + }) + + const list = data?.data.fiberPeers ?? [] + const pageInfo = data?.meta ?? { total: 1, pageSize: PAGE_SIZE } + const totalPages = Math.ceil(pageInfo.total / pageInfo.pageSize) + + const handleCopy = (e: React.SyntheticEvent) => { + const elm = e.target + if (!(elm instanceof HTMLElement)) return + const { copyText } = elm.dataset + if (!copyText) return + e.stopPropagation() + e.preventDefault() + navigator?.clipboard.writeText(copyText).then(() => setToast({ message: t('common.copied') })) + } + + return ( + +
+

+ CKB Fiber Peers + + refetchList()} /> +

+ + + + {fields.map(f => { + return + })} + + +
+
+ {list.map(i => { + return ( + + {fields.map(f => { + const v = i[f.key as keyof typeof i] + return + })} + + ) + })} +
+
+ + + +
{t(`fiber.peer.${f.label}`)}
{f.transformer?.(v, i) ?? v}
+ +
+
+
+ ) +} + +export default PeerList diff --git a/src/pages/Fiber/utils/index.tsx b/src/pages/Fiber/utils/index.tsx new file mode 100644 index 000000000..2513b3bae --- /dev/null +++ b/src/pages/Fiber/utils/index.tsx @@ -0,0 +1,42 @@ +import { scriptToHash } from '@nervosnetwork/ckb-sdk-utils' +import type { Fiber } from '../../../services/ExplorerService/fetcher' +import { parseNumericAbbr } from '../../../utils/chart' +import { localeNumberString, parseUDTAmount } from '../../../utils/number' +import { shannonToCkb } from '../../../utils/util' + +export const getFundingThreshold = (n: Fiber.Graph.Node) => { + const ckb = shannonToCkb(n.autoAcceptMinCkbFundingAmount) + const amount = parseNumericAbbr(ckb) + + const tokens: { title: string; display: string; id: string; icon?: string }[] = [ + { + title: `${localeNumberString(ckb)} CKB`, + display: `${amount} CKB`, + id: 'ckb', + icon: '/images/tokens/ckb_token.svg', + }, + ] + + n.udtCfgInfos.forEach(udt => { + if (udt && udt.autoAcceptAmount && typeof udt.decimal === 'number' && udt.symbol) { + try { + const udtAmount = parseUDTAmount(udt.autoAcceptAmount, udt.decimal) + const id = scriptToHash({ + codeHash: udt.codeHash, + hashType: udt.hashType, + args: udt.args, + }) + tokens.push({ + title: `${localeNumberString(udtAmount)} ${udt.symbol}`, + display: `${parseNumericAbbr(udtAmount)} ${udt.symbol}`, + icon: udt.iconFile, + id, + }) + } catch (e) { + console.error(e) + } + } + }) + + return tokens +} diff --git a/src/pages/Home/Banner/index.tsx b/src/pages/Home/Banner/index.tsx index 686576b48..49f3ca9a0 100644 --- a/src/pages/Home/Banner/index.tsx +++ b/src/pages/Home/Banner/index.tsx @@ -1,7 +1,6 @@ import { useQuery } from '@tanstack/react-query' import { BarChartIcon } from '@radix-ui/react-icons' import { useTranslation } from 'react-i18next' -import { Tooltip } from 'antd' import { Link } from '../../../components/Link' import config from '../../../config' import styles from './index.module.scss' @@ -56,17 +55,9 @@ export default () => { {t(`banner.learn_more`)} - - ) => { - e.preventDefault() - }} - > - {t('banner.find_nodes')} - - + + {t('banner.find_nodes')} +
) diff --git a/src/routes/index.tsx b/src/routes/index.tsx index 2fbe2276c..a88663c4a 100644 --- a/src/routes/index.tsx +++ b/src/routes/index.tsx @@ -81,6 +81,14 @@ const Hasher = lazy(() => import('../pages/Tools/Hasher')) const BroadcastTx = lazy(() => import('../pages/Tools/BroadcastTx')) const CamelCase = lazy(() => import('../pages/Tools/CamelCase')) const MoleculeParser = lazy(() => import('../pages/Tools/MoleculeParser')) +// ====== +const FiberPeerList = lazy(() => import('../pages/Fiber/PeerList')) +const FiberPeer = lazy(() => import('../pages/Fiber/Peer')) +const FiberChannel = lazy(() => import('../pages/Fiber/Channel')) +const FiberGraphNodeList = lazy(() => import('../pages/Fiber/GraphNodeList')) +const FiberGraphNode = lazy(() => import('../pages/Fiber/GraphNode')) +const FiberGraphChannelList = lazy(() => import('../pages/Fiber/GraphChannelList')) +// ====== const routes: RouteProps[] = [ { @@ -346,6 +354,30 @@ const routes: RouteProps[] = [ path: '/tools/molecule-parser', component: MoleculeParser, }, + { + path: '/fiber/peers', + component: FiberPeerList, + }, + { + path: '/fiber/peers/:id', + component: FiberPeer, + }, + { + path: '/fiber/channels/:id', + component: FiberChannel, + }, + { + path: '/fiber/graph/nodes', + component: FiberGraphNodeList, + }, + { + path: '/fiber/graph/node/:id', + component: FiberGraphNode, + }, + { + path: '/fiber/graph/channels', + component: FiberGraphChannelList, + }, ] type PageErrorBoundaryState = { diff --git a/src/services/ExplorerService/fetcher.ts b/src/services/ExplorerService/fetcher.ts index bd68e347e..9f0ed87e8 100644 --- a/src/services/ExplorerService/fetcher.ts +++ b/src/services/ExplorerService/fetcher.ts @@ -70,6 +70,7 @@ export enum SearchResultType { TokenItem = 'token_item', DID = 'did', BtcAddress = 'bitcoin_address', + FiberGraphNode = 'fiber_graph_node', } enum SearchQueryType { @@ -113,6 +114,14 @@ export type AggregateSearchResult = }, SearchResultType.BtcAddress > + | Response.Wrapper< + { + alias: string + nodeId: string + peerId: string + }, + SearchResultType.FiberGraphNode + > export const getBtcTxList = (idList: string[]): Promise> => { if (idList.length === 0) return Promise.resolve({}) @@ -1195,6 +1204,108 @@ export const apiFetcher = { }, }), getBtcTxList, + + // ================== + // Fiber + // ================== + getFiberPeerList: (page = 1, pageSize = 10) => { + return requesterV2 + .get( + `/fiber/peers?${new URLSearchParams({ + page: page.toString(), + page_size: pageSize.toString(), + })}`, + ) + .then(res => + toCamelcase< + Response.Response<{ + fiberPeers: Fiber.Peer.ItemInList[] + meta: { + total: number + pageSize: number + } + }> + >(res.data), + ) + }, + + getFiberPeerDetail: (id: string) => { + return requesterV2 + .get(`/fiber/peers/${id}`) + .then(res => toCamelcase>(res.data)) + }, + + getFiberChannel: (id: string) => { + return requesterV2 + .get(`/fiber/channels/${id}`) + .then(res => toCamelcase>(res.data)) + }, + + addFiberPeer: (params: { rpc: string; id: string; name?: string }) => { + return requesterV2 + .post(`/fiber/peers`, { + name: params.name, + rpc_listening_addr: params.rpc, + peer_id: params.id, + }) + .catch(e => { + if (Array.isArray(e.response?.data)) { + const res = e.response.data[0] + if (res) { + throw new Error(res.title) + } + } + throw e + }) + }, + + getGraphNodes: (page = 1, pageSize = 10) => { + return requesterV2 + .get( + `/fiber/graph_nodes?${new URLSearchParams({ + page: page.toString(), + page_size: pageSize.toString(), + })}`, + ) + .then(res => + toCamelcase< + Response.Response<{ + fiberGraphNodes: Fiber.Graph.Node[] + meta: { + total: number + pageSize: number + } + }> + >(res.data), + ) + }, + + getGraphNodeDetail: (id: string) => { + return requesterV2 + .get(`/fiber/graph_nodes/${id}`) + .then(res => toCamelcase>(res.data)) + }, + getGraphChannels: (page = 1, pageSize = 10) => { + return requesterV2 + .get( + `/fiber/graph_channels?${new URLSearchParams({ + page: page.toString(), + page_size: pageSize.toString(), + status: 'open', + })}`, + ) + .then(res => + toCamelcase< + Response.Response<{ + fiberGraphChannels: Fiber.Graph.Channel[] + meta: { + total: number + pageSize: number + } + }> + >(res.data), + ) + }, } // ==================== @@ -1386,3 +1497,114 @@ export interface RGBTransaction { rgbCellChanges: number rgbTxid: string } + +export namespace Fiber { + export namespace Peer { + interface Base { + peerId: string + rpcListeningAddr: string[] + firstChannelOpenedAt: null // TODO + lastChannelUpdatedAt: null // TODO + } + export interface ItemInList extends Base { + name: string + channelsCount: number + totalLocalBalance: string // shannon amount + } + + export interface Detail extends Base { + fiberChannels: { + peerId: string + channelId: string + stateName: string // TODO: should be enum + stateFlags: [] // TODO + }[] + } + } + export namespace Channel { + export interface Peer { + name?: string + peerId: string + rpcListeningAddr: string[] + } + export interface Detail { + channelId: string + stateName: string // TODO should be name + stateFlags: [] // TODO + shutdownAt: null // TODO + createdAt: string // utc time + updatedAt: string // utc time + localBalance: string // shannon + offeredTlcBalance: string // shannon + receivedTlcBalance: string // shannon + remoteBalance: string // shannon + localPeer: Peer + remotePeer: Peer + } + } + + export namespace Graph { + interface UdtConfigInfo { + args: string + codeHash: string + hashType: HashType + decimal?: number + fullName?: string + iconFile?: string + symbol?: string + autoAcceptAmount: string + } + + interface OpenTransactionInfo { + address: string + blockNumber: number + blockTimestamp: number + capacity: string + txHash: string + udtAmount?: string + } + + interface ClosedTransactionInfo { + blockNumber: number + blockTimestamp: number + txHash: string + closeAccounts: { + address: string + capacity: string + udtAmount: string | null + }[] + } + + export interface Node { + alias: string + nodeId: string + addresses: string[] + timestamp: string + chainHash: string + autoAcceptMinCkbFundingAmount: string + udtCfgInfos: UdtConfigInfo[] + totalCapacity: string + connectedNodeIds: string[] + openChannelsCount: number + } + + export interface Channel { + channelOutpoint: string + node1: string + node2: string + chainHash: string + fundingTxBlockNumber: string + fundingTxIndex: string // number + lastUpdatedTimestamp: string + node1ToNode2FeeRate: string + node2ToNode1FeeRate: string + capacity: string + openTransactionInfo: OpenTransactionInfo + closedTransactionInfo: ClosedTransactionInfo + } + + export interface NodeDetail extends Node { + fiberGraphChannels: Channel[] + } + } +} diff --git a/src/styles/card.module.scss b/src/styles/card.module.scss new file mode 100644 index 000000000..6d5d707bd --- /dev/null +++ b/src/styles/card.module.scss @@ -0,0 +1,6 @@ +%base-card { + background: #fff; + border-radius: 6px; + padding: 16px; + box-shadow: 0 2px 6px 0 #4d4d4d33; +} diff --git a/src/styles/table.module.scss b/src/styles/table.module.scss new file mode 100644 index 000000000..c5052bc78 --- /dev/null +++ b/src/styles/table.module.scss @@ -0,0 +1,43 @@ +%base-table { + width: 100%; + text-align: left; + cursor: default; + overflow: hidden; + border-radius: 6px; + box-shadow: rgb(0 0 0 / 12%) 0 2px 6px 0; + font-size: 0.875rem; + + tr { + background: #fff; + } + + td, + th { + padding: 8px; + padding-right: 1rem; + + &:first-child { + padding-left: 40px; + } + + &:last-child { + text-align: right; + } + } + + thead { + th { + height: 3.5rem; + } + } + + tbody { + tr:hover { + background: #ccc; + } + } + + .tableSeparator { + height: 4px; + } +} diff --git a/src/styles/text.module.scss b/src/styles/text.module.scss new file mode 100644 index 000000000..cbb2415db --- /dev/null +++ b/src/styles/text.module.scss @@ -0,0 +1,21 @@ +%monospace { + font-family: source-code-pro, Menlo, Monaco, Consolas, 'Courier New', monospace; +} + +%hash { + @extend %monospace; + + display: flex; + align-items: center; + overflow: hidden; + user-select: none; + + div { + font-family: inherit; + } + + div:first-child { + overflow: hidden; + text-overflow: ellipsis; + } +} From f467cfd10da09b52bd5e046fc895e156c355c00c Mon Sep 17 00:00:00 2001 From: Keith Date: Fri, 3 Jan 2025 12:47:14 +0900 Subject: [PATCH 3/5] fix: ractify breakpoint of layouts --- src/pages/NftCollections/index.tsx | 10 +++++----- src/pages/NftCollections/styles.module.scss | 12 ++++++++++-- 2 files changed, 15 insertions(+), 7 deletions(-) diff --git a/src/pages/NftCollections/index.tsx b/src/pages/NftCollections/index.tsx index 15ddd60cb..d79292671 100644 --- a/src/pages/NftCollections/index.tsx +++ b/src/pages/NftCollections/index.tsx @@ -7,7 +7,7 @@ import Pagination from '../../components/Pagination' import { getPrimaryColor } from '../../constants/common' import { explorerService } from '../../services/ExplorerService' import { udtSubmitEmail } from '../../utils/util' -import { useIsMobile, useSearchParams } from '../../hooks' +import { useSearchParams } from '../../hooks' import styles from './styles.module.scss' import { useNFTCollectionsSortParam } from './util' @@ -24,7 +24,6 @@ const NftCollections = () => { const { search } = useLocation() const { page = '1', type, tags } = useSearchParams('page', 'type', 'tags') const { sort } = useNFTCollectionsSortParam() - const isMobile = useIsMobile() const isValidFilter = isTxFilterType(type) && type !== 'all' @@ -61,11 +60,12 @@ const NftCollections = () => {
- {isMobile ? ( +
- ) : ( +
+
- )} +
Date: Fri, 3 Jan 2025 12:47:38 +0900 Subject: [PATCH 4/5] feat: disable link to fiber graph nodes --- src/pages/Home/Banner/index.tsx | 15 ++++++++++++--- 1 file changed, 12 insertions(+), 3 deletions(-) diff --git a/src/pages/Home/Banner/index.tsx b/src/pages/Home/Banner/index.tsx index 49f3ca9a0..a111eac82 100644 --- a/src/pages/Home/Banner/index.tsx +++ b/src/pages/Home/Banner/index.tsx @@ -1,6 +1,7 @@ import { useQuery } from '@tanstack/react-query' import { BarChartIcon } from '@radix-ui/react-icons' import { useTranslation } from 'react-i18next' +import { Tooltip } from 'antd' import { Link } from '../../../components/Link' import config from '../../../config' import styles from './index.module.scss' @@ -55,9 +56,17 @@ export default () => { {t(`banner.learn_more`)} - - {t('banner.find_nodes')} - + + ) => { + e.preventDefault() + }} + > + {t('banner.find_nodes')} + + ) From 3d3d8c7bf3748a622f968572e73a4ab1bd8d74e7 Mon Sep 17 00:00:00 2001 From: Keith Date: Sat, 4 Jan 2025 17:44:08 +0900 Subject: [PATCH 5/5] feat: add active addresses chart --- src/locales/en.json | 31 +++- src/locales/zh.json | 4 + .../activities/ActiveAddressesChart.tsx | 151 ++++++++++++++++++ .../activities/CkbHodlWave.tsx | 87 +++++----- src/pages/StatisticsChart/index.tsx | 7 + src/routes/index.tsx | 5 + src/services/ExplorerService/fetcher.ts | 12 ++ src/services/ExplorerService/types.ts | 5 + src/utils/chart.ts | 45 ++++++ 9 files changed, 301 insertions(+), 46 deletions(-) create mode 100644 src/pages/StatisticsChart/activities/ActiveAddressesChart.tsx diff --git a/src/locales/en.json b/src/locales/en.json index fe766623a..9aadc6ca1 100644 --- a/src/locales/en.json +++ b/src/locales/en.json @@ -292,6 +292,7 @@ "circulating_supply": "Circulating Supply", "burnt": "Burnt", "locked": "Unvested", + "week": "Time (week)", "year": "Time (year)", "nominal_apc": "Nominal DAO Compensation Rate", "inflation_rate": "Inflation Rate", @@ -326,6 +327,9 @@ "ckb_amount": "CKB Amount", "contract_resource_distributed": "Contract Resource Distribution", "contract_resource_distributed_description": "The x axis represents contract's unique address count, the y axis represents the contract's CKB amount, the symbol size represents the contract's transaction count.", + "active_addresses": "Active Addresses", + "active_addresses_description": "The number of unique addresses that have participated in the network as a sender or receiver.", + "active_address_count": "Active Addresses Count", "country": "Country/Region", "node_country_distribution": "Nodes distribution by Country/Region", "top_50_holders": "Top 50 Holders", @@ -340,7 +344,32 @@ "over_three_years": "> 3y", "ckb_hodl_wave": "CKB HODL Wave", "h24_transaction_count": "24hr Transaction Count", - "holder_count": "Holder Count" + "holder_count": "Holder Count", + "address_label": { + "anyoneCanPayLock": "Anyone Can Pay", + "btcTimeLock": "BTC Time", + "cheque": "Cheque", + "flashSigner": "Flash Signer", + "godwokenCustodianLock": "Godwoken Custodian", + "godwokenDepositLock": "Godwoken Deposit", + "godwokenStakeLock": "Godwoken Stake", + "godwokenWithdrawalLock": "Godwoken Withdrawal", + "iCkbLogic": "iCKB Logic", + "joyId": "Joy ID", + "nostr": "Nostr", + "omniLockV1": "Omni Lock V1", + "omniLockV2": "Omni Lock V2", + "pwLock": "Portal Wallet", + "rgb++": "RGB++", + "secp256K1/blake160": "Secp256k1/Blake160", + "secp256K1/multisig": "Secp256k1/Multisig", + "singleUseLock": "Single Use", + "udtLimitOrderr": "UDT Limit Order", + "unipassV2": "Unipass V2", + "unipassV3": "Unipass V3", + "wrOwnedOwner": "WR Owned Owner", + "others": "Others" + } }, "home": { "height": "Height", diff --git a/src/locales/zh.json b/src/locales/zh.json index 605431e21..cca1b5de6 100644 --- a/src/locales/zh.json +++ b/src/locales/zh.json @@ -306,6 +306,7 @@ "circulating_supply": "总流通量", "burnt": "销毁量", "locked": "未解锁数量", + "week": "时间 (周)", "year": "时间 (年)", "nominal_apc": "基准补偿率", "inflation_rate": "通胀率", @@ -340,6 +341,9 @@ "ckb_amount": "CKB 数量", "contract_resource_distributed": "合约资源分布图", "contract_resource_distributed_description": "横轴表示合约的唯一地址数量, 纵轴是合约的 CKB 数量, 图标大小代表合约的交易数量", + "active_addresses": "活跃地址", + "active_addresses_description": "活跃地址是指在特定时间段内有交易记录的地址", + "active_address_count": "活跃地址数量", "country": "国家(地区)", "node_country_distribution": "节点国家(地区)分布图", "top_50_holders": "前 50 持有地址", diff --git a/src/pages/StatisticsChart/activities/ActiveAddressesChart.tsx b/src/pages/StatisticsChart/activities/ActiveAddressesChart.tsx new file mode 100644 index 000000000..1cd52ede6 --- /dev/null +++ b/src/pages/StatisticsChart/activities/ActiveAddressesChart.tsx @@ -0,0 +1,151 @@ +import { useTranslation } from 'react-i18next' +import type { ChartColorConfig } from '../../../constants/common' +import { SmartChartPage } from '../common' +import { DATA_ZOOM_CONFIG, handleAxis, variantColors } from '../../../utils/chart' +import { explorerService } from '../../../services/ExplorerService' + +// Helper function to get ISO week number and year +function getWeekNumber(timestamp: string) { + const date = new Date(+timestamp * 1000) + const firstDayOfYear = new Date(date.getFullYear(), 0, 1) + const days = Math.floor((date.getTime() - firstDayOfYear.getTime()) / (24 * 60 * 60 * 1000)) + const weekNumber = Math.ceil((days + firstDayOfYear.getDay() + 1) / 7) + return `${date.getFullYear()}-W${weekNumber}` +} + +const useOption = ( + activeAddresses: { + createdAtUnixtimestamp: string + distribution: Record + }[], + _: ChartColorConfig, + isMobile: boolean, + isThumbnail = false, +): echarts.EChartOption => { + const { t } = useTranslation() + const gridThumbnail = { + left: '4%', + right: '10%', + top: '8%', + bottom: '6%', + containLabel: true, + } + const grid = { + left: '4%', + right: '8%', + top: '12%', + bottom: '5%', + containLabel: true, + } + + const aggregatedByWeek = activeAddresses.reduce((acc, item) => { + const week = getWeekNumber(item.createdAtUnixtimestamp) + + if (!acc[week]) { + acc[week] = { + createdAtWeek: week, + distribution: {}, + } + } + + Object.entries(item.distribution).forEach(([key, value]) => { + acc[week].distribution[key] = (acc[week].distribution[key] || 0) + value + }) + + return acc + }, {} as Record }>) + + const aggregatedDdata = Object.values(aggregatedByWeek) + const dataset = aggregatedDdata.slice(0, aggregatedDdata.length - 1) // Remove the last week data because it's not complete + const xAxisData = dataset.map(item => item.createdAtWeek) + const allKeys = Array.from(new Set(dataset.flatMap(item => Object.keys(item.distribution)))).sort((a, b) => { + if (a === 'others') return 1 + if (b === 'others') return -1 + return a.localeCompare(b) + }) + const series = allKeys.map(key => ({ + name: t(`statistic.address_label.${key}`), + type: 'line', + stack: 'total', + areaStyle: {}, + lineStyle: { + width: 0, + }, + symbol: 'none', + emphasis: { + focus: 'series', + }, + data: dataset.map(item => item.distribution[key] || 0), + })) + const colors = variantColors(allKeys.length) + + return { + color: colors, + tooltip: !isThumbnail + ? { + trigger: 'axis', + axisPointer: { type: 'cross' }, + formatter: params => { + // Filter out fields with value 0 + if (!Array.isArray(params)) return '' + const filteredParams = params.filter(item => item.value !== 0) + + // Construct the tooltip content + if (filteredParams.length === 0) return '' // No fields to display + + const header = `${filteredParams[0].axisValue}
` // Show week + const sum = ` +${t('statistic.active_address_count')}: ${filteredParams.reduce( + (acc, item) => acc + Number(item.value), + 0, + )}

` + const body = filteredParams + .map( + item => + ` + ${item.seriesName}: ${item.value}`, + ) + .join('
') + + return header + sum + body + }, + } + : undefined, + grid: isThumbnail ? gridThumbnail : grid, + dataZoom: isThumbnail ? [] : DATA_ZOOM_CONFIG, + legend: { data: isThumbnail ? [] : allKeys.map(key => t(`statistic.address_label.${key}`) as string) }, + xAxis: { + type: 'category', + boundaryGap: false, + data: xAxisData, + axisLabel: { + formatter: (value: string) => value, // Display week labels + }, + name: isMobile || isThumbnail ? '' : t('statistic.week'), + }, + yAxis: { + type: 'value', + name: isMobile || isThumbnail ? '' : `${t('statistic.active_address_count')}`, + axisLabel: { + formatter: (value: string) => handleAxis(+value), + }, + }, + series, + } +} + +export const ActiveAddressesChart = ({ isThumbnail = false }: { isThumbnail?: boolean }) => { + const [t] = useTranslation() + return ( + + ) +} + +export default ActiveAddressesChart diff --git a/src/pages/StatisticsChart/activities/CkbHodlWave.tsx b/src/pages/StatisticsChart/activities/CkbHodlWave.tsx index f168ed069..7bdac08a7 100644 --- a/src/pages/StatisticsChart/activities/CkbHodlWave.tsx +++ b/src/pages/StatisticsChart/activities/CkbHodlWave.tsx @@ -1,5 +1,6 @@ import { useTranslation } from 'react-i18next' import dayjs from 'dayjs' +import type { ChartColorConfig } from '../../../constants/common' import { SupportedLng, useCurrentLanguage } from '../../../utils/i18n' import { DATA_ZOOM_CONFIG, @@ -7,10 +8,10 @@ import { assertSerialsItem, assertSerialsDataIsStringArrayOf10, handleAxis, + variantColors, } from '../../../utils/chart' import { tooltipColor, tooltipWidth, SeriesItem, SmartChartPage } from '../common' import { ChartItem, explorerService } from '../../../services/ExplorerService' -import { ChartColorConfig } from '../../../constants/common' const widthSpan = (value: string, currentLanguage: SupportedLng) => tooltipWidth(value, currentLanguage === 'en' ? 125 : 80) @@ -82,7 +83,7 @@ const useTooltip = () => { const useOption = ( statisticCkbHodlWaves: ChartItem.CkbHodlWaveHolderCount[], - chartColor: ChartColorConfig, + _: ChartColorConfig, isMobile: boolean, isThumbnail = false, ): echarts.EChartOption => { @@ -103,7 +104,36 @@ const useOption = ( containLabel: true, } const parseTooltip = useTooltip() - const colors = [...chartColor.moreColors].slice(0, 9) + const legends = [ + { + name: t('statistic.24h'), + }, + { + name: t('statistic.day_to_one_week'), + }, + { + name: t('statistic.one_week_to_one_month'), + }, + { + name: t('statistic.one_month_to_three_months'), + }, + { + name: t('statistic.three_months_to_six_months'), + }, + { + name: t('statistic.six_months_to_one_year'), + }, + { + name: t('statistic.one_year_to_three_years'), + }, + { + name: t('statistic.over_three_years'), + }, + { + name: t('statistic.holder_count'), + }, + ] + const colors = variantColors(legends.length) return { color: colors, tooltip: !isThumbnail @@ -123,48 +153,7 @@ const useOption = ( } : undefined, legend: { - data: isThumbnail - ? [] - : [ - { - name: t('statistic.24h'), - }, - { - name: t('statistic.day_to_one_week'), - }, - { - name: t('statistic.one_week_to_one_month'), - }, - { - name: t('statistic.one_month_to_three_months'), - }, - { - name: t('statistic.three_months_to_six_months'), - }, - { - name: t('statistic.six_months_to_one_year'), - }, - { - name: t('statistic.one_year_to_three_years'), - }, - { - name: t('statistic.over_three_years'), - }, - { - name: t('statistic.holder_count'), - }, - ], - selected: { - [t('statistic.24h')]: true, - [t('statistic.day_to_one_week')]: true, - [t('statistic.one_week_to_one_month')]: true, - [t('statistic.one_month_to_three_months')]: true, - [t('statistic.three_months_to_six_months')]: true, - [t('statistic.six_months_to_one_year')]: true, - [t('statistic.one_year_to_three_years')]: true, - [t('statistic.over_three_years')]: true, - [t('statistic.holder_count')]: true, - }, + data: isThumbnail ? [] : legends, }, grid: isThumbnail ? gridThumbnail : grid, dataZoom: isThumbnail ? [] : DATA_ZOOM_CONFIG, @@ -214,6 +203,7 @@ const useOption = ( areaStyle: { color: colors[0], }, + lineStyle: { width: 0 }, }, { name: t('statistic.day_to_one_week'), @@ -225,6 +215,7 @@ const useOption = ( areaStyle: { color: colors[1], }, + lineStyle: { width: 0 }, }, { name: t('statistic.one_week_to_one_month'), @@ -236,6 +227,7 @@ const useOption = ( areaStyle: { color: colors[2], }, + lineStyle: { width: 0 }, }, { name: t('statistic.one_month_to_three_months'), @@ -247,6 +239,7 @@ const useOption = ( areaStyle: { color: colors[3], }, + lineStyle: { width: 0 }, }, { name: t('statistic.three_months_to_six_months'), @@ -258,6 +251,7 @@ const useOption = ( areaStyle: { color: colors[4], }, + lineStyle: { width: 0 }, }, { name: t('statistic.six_months_to_one_year'), @@ -269,6 +263,7 @@ const useOption = ( areaStyle: { color: colors[5], }, + lineStyle: { width: 0 }, }, { name: t('statistic.one_year_to_three_years'), @@ -280,6 +275,7 @@ const useOption = ( areaStyle: { color: colors[6], }, + lineStyle: { width: 0 }, }, { name: t('statistic.over_three_years'), @@ -291,6 +287,7 @@ const useOption = ( areaStyle: { color: colors[7], }, + lineStyle: { width: 0 }, }, { name: t('statistic.holder_count'), diff --git a/src/pages/StatisticsChart/index.tsx b/src/pages/StatisticsChart/index.tsx index c85c9e457..f54088bef 100644 --- a/src/pages/StatisticsChart/index.tsx +++ b/src/pages/StatisticsChart/index.tsx @@ -17,6 +17,7 @@ import { HashRateChart } from './mining/HashRate' import { UncleRateChart } from './mining/UncleRate' import { BalanceDistributionChart } from './activities/BalanceDistribution' import { ContractResourceDistributedChart } from './activities/ContractResourceDistributed' +import { ActiveAddressesChart } from './activities/ActiveAddressesChart' import { TxFeeHistoryChart } from './activities/TxFeeHistory' import { BlockTimeDistributionChart } from './block/BlockTimeDistribution' import { EpochTimeDistributionChart } from './block/EpochTimeDistribution' @@ -121,6 +122,12 @@ const useChartsData = () => { path: '/charts/contract-resource-distributed', description: t('statistic.contract_resource_distributed_description'), }, + { + title: `${t('statistic.active_addresses')}`, + chart: , + path: '/charts/active-addresses', + description: t('statistic.active_addresses_description'), + }, { title: `${t('statistic.knowledge_size')}`, chart: , diff --git a/src/routes/index.tsx b/src/routes/index.tsx index a88663c4a..f146eebc1 100644 --- a/src/routes/index.tsx +++ b/src/routes/index.tsx @@ -55,6 +55,7 @@ const CellCountChart = lazy(() => import('../pages/StatisticsChart/activities/Ce const ContractResourceDistributedChart = lazy( () => import('../pages/StatisticsChart/activities/ContractResourceDistributed'), ) +const ActiveAddressesChart = lazy(() => import('../pages/StatisticsChart/activities/ActiveAddressesChart')) const KnowledgeSizeChart = lazy(() => import('../pages/StatisticsChart/activities/KnowledgeSize')) const CkbHodlWaveChart = lazy(() => import('../pages/StatisticsChart/activities/CkbHodlWave')) const AddressBalanceRankChart = lazy(() => import('../pages/StatisticsChart/activities/AddressBalanceRank')) @@ -265,6 +266,10 @@ const routes: RouteProps[] = [ path: '/charts/contract-resource-distributed', component: ContractResourceDistributedChart, }, + { + path: '/charts/active-addresses', + component: ActiveAddressesChart, + }, { path: '/charts/knowledge-size', component: KnowledgeSizeChart, diff --git a/src/services/ExplorerService/fetcher.ts b/src/services/ExplorerService/fetcher.ts index 9f0ed87e8..34f98ee04 100644 --- a/src/services/ExplorerService/fetcher.ts +++ b/src/services/ExplorerService/fetcher.ts @@ -818,6 +818,18 @@ export const apiFetcher = { })) }), + fetchStatisticActiveAddresses: () => + v1GetUnwrappedList(`/daily_statistics/activity_address_contract_distribution`).then( + items => + items.map<{ + createdAtUnixtimestamp: string + distribution: Record + }>(({ createdAtUnixtimestamp, ...list }) => ({ + createdAtUnixtimestamp, + distribution: Object.assign({}, ...list.activityAddressContractDistribution), + })), + ), + fetchFlushChartCache: () => v1GetUnwrapped<{ flushCacheInfo: string[] }>(`statistics/flush_cache_info`), fetchSimpleUDT: (typeHash: string) => v1GetUnwrapped(`/udts/${typeHash}`), diff --git a/src/services/ExplorerService/types.ts b/src/services/ExplorerService/types.ts index 07f8399b3..24cb18357 100644 --- a/src/services/ExplorerService/types.ts +++ b/src/services/ExplorerService/types.ts @@ -201,6 +201,11 @@ export namespace ChartItem { createdAtUnixtimestamp: string knowledgeSize: number } + + export interface ActiveAddresses { + createdAtUnixtimestamp: string + activityAddressContractDistribution: Record[] + } } export interface NervosDaoDepositor { diff --git a/src/utils/chart.ts b/src/utils/chart.ts index 436664c57..df4ea3afd 100644 --- a/src/utils/chart.ts +++ b/src/utils/chart.ts @@ -1,5 +1,6 @@ import BigNumber from 'bignumber.js' import { EChartOption } from 'echarts' +import { ChartColor } from '../constants/common' import { SeriesItem } from '../pages/StatisticsChart/common' import type { FeeRateTracker } from '../services/ExplorerService/fetcher' @@ -210,3 +211,47 @@ export const assertSerialsDataIsStringArrayOf10: (value: EChartOption.Tooltip.Fo throw new Error('invalid SeriesItem length of 10') } } + +const BASE_COLORS = [ + ...ChartColor.colors.slice(0, 2), + '#FF5733', + '#FFC300', + '#DAF7A6', + '#33FF57', + '#33C1FF', + '#8A33FF', + '#FF33A8', + '#FF33F6', + '#FF8C33', + '#FFE733', +] + +export const variantColors = (count: number, baseColors: string[] = BASE_COLORS) => { + // Helper function to adjust brightness + function adjustColor(color: string, factor: number) { + const hex = color.replace('#', '') + const r = Math.min(255, Math.max(0, parseInt(hex.substring(0, 2), 16) + factor)) + const g = Math.min(255, Math.max(0, parseInt(hex.substring(2, 4), 16) + factor)) + const b = Math.min(255, Math.max(0, parseInt(hex.substring(4, 6), 16) + factor)) + return `#${r.toString(16).padStart(2, '0')}${g.toString(16).padStart(2, '0')}${b.toString(16).padStart(2, '0')}` + } + + const colors = [] + let variantIndex = 0 + + for (let i = 0; i < count; i++) { + const baseColor = baseColors[i % baseColors.length] + let adjustmentFactor = 0 + if (variantIndex % 3 === 1) { + adjustmentFactor = 30 + } else if (variantIndex % 3 === 2) { + variantIndex = -30 + } + colors.push(adjustColor(baseColor, adjustmentFactor)) + if ((i + 1) % baseColors.length === 0) { + variantIndex++ + } + } + + return colors +}