diff --git a/public/locales/en/common.json b/public/locales/en/common.json
index e95bd49..677ced4 100644
--- a/public/locales/en/common.json
+++ b/public/locales/en/common.json
@@ -10,10 +10,12 @@
"chain": "Chain",
"chainId": "Chain ID",
"nativeToken": "Native token",
- "tvl": "TVL - L1",
+ "tvl": "TVL (L1)",
"type": "Type",
- "search": "Search",
- "notFound": "Chain not found"
+ "search": "Search (Chain)",
+ "notFound": "Chain not found",
+ "noRPC": "No RPC",
+ "noMetadata": "No Metadata"
}
},
"CHAIN": {
diff --git a/public/locales/es/common.json b/public/locales/es/common.json
index 2b7daa8..6537f38 100644
--- a/public/locales/es/common.json
+++ b/public/locales/es/common.json
@@ -10,10 +10,12 @@
"chain": "Cadena",
"chainId": "ID de cadena",
"nativeToken": "Token nativo",
- "tvl": "TVL - L1",
+ "tvl": "TVL (L1)",
"type": "Tipo",
- "search": "Buscar",
- "notFound": "Cadena no encontrada"
+ "search": "Buscar (Cadena)",
+ "notFound": "Cadena no encontrada",
+ "noRPC": "Sin RPC",
+ "noMetadata": "Sin Metadata"
}
},
"CHAIN": {
diff --git a/src/assets/icons/github.svg b/src/assets/icons/github.svg
new file mode 100644
index 0000000..2bccb51
--- /dev/null
+++ b/src/assets/icons/github.svg
@@ -0,0 +1,3 @@
+
diff --git a/src/assets/icons/infoDark.svg b/src/assets/icons/infoDark.svg
new file mode 100644
index 0000000..93d0755
--- /dev/null
+++ b/src/assets/icons/infoDark.svg
@@ -0,0 +1,5 @@
+
diff --git a/src/assets/icons/infoLight.svg b/src/assets/icons/infoLight.svg
new file mode 100644
index 0000000..aa0e9fe
--- /dev/null
+++ b/src/assets/icons/infoLight.svg
@@ -0,0 +1,5 @@
+
diff --git a/src/components/InfoTag.tsx b/src/components/InfoTag.tsx
new file mode 100644
index 0000000..1d81fd3
--- /dev/null
+++ b/src/components/InfoTag.tsx
@@ -0,0 +1,44 @@
+import { styled, Box, Typography } from '@mui/material';
+import Image from 'next/image';
+
+import { useCustomTheme } from '~/hooks';
+import informationIconDark from '~/assets/icons/infoDark.svg';
+import informationIconLight from '~/assets/icons/infoLight.svg';
+
+interface InfoTagProps {
+ information: string;
+}
+
+export const InfoTag = ({ information }: InfoTagProps) => {
+ const { theme } = useCustomTheme();
+ return (
+
+
+ {information}
+
+ );
+};
+
+const InfoTagContainer = styled(Box)(() => {
+ const { currentTheme } = useCustomTheme();
+
+ return {
+ display: 'flex',
+ alignItems: 'center',
+ fontSize: '0.7rem',
+ gap: currentTheme.gap,
+ backgroundColor: currentTheme.warningBackground,
+ borderRadius: currentTheme.borderRadius,
+ padding: '0.1rem 0.5rem 0.1rem 0.1rem',
+ border: currentTheme.warningBorder,
+ };
+});
+
+const InfoText = styled(Typography)(() => {
+ const { currentTheme } = useCustomTheme();
+
+ return {
+ fontSize: '0.7rem',
+ color: currentTheme.warningText,
+ };
+});
diff --git a/src/components/Table.tsx b/src/components/Table.tsx
index 11560e4..9eb7371 100644
--- a/src/components/Table.tsx
+++ b/src/components/Table.tsx
@@ -1,14 +1,27 @@
import { useTranslation } from 'next-i18next';
import { useRouter } from 'next/router';
+import {
+ styled,
+ TableContainer,
+ Table,
+ TableHead,
+ TableRow,
+ TableCell,
+ TableBody,
+ Typography,
+ Avatar,
+} from '@mui/material';
import { EcosystemChainData } from '~/types';
import { formatDataNumber } from '~/utils';
+import { useCustomTheme } from '~/hooks';
+import { InfoTag } from './InfoTag';
interface TableProps {
chains: EcosystemChainData[];
}
-export const Table = ({ chains }: TableProps) => {
+export const DataTable = ({ chains }: TableProps) => {
const { t } = useTranslation();
const router = useRouter();
@@ -17,26 +30,131 @@ export const Table = ({ chains }: TableProps) => {
};
return (
-
{/*
*/}
diff --git a/src/containers/Dashboard/index.tsx b/src/containers/Dashboard/index.tsx
index fb425bc..a84cddb 100644
--- a/src/containers/Dashboard/index.tsx
+++ b/src/containers/Dashboard/index.tsx
@@ -1,6 +1,6 @@
import { useTranslation } from 'next-i18next';
-import { NotFound, Table, Title } from '~/components';
+import { NotFound, DataTable, Title } from '~/components';
import { useData, useStateContext } from '~/hooks';
export const Dashboard = () => {
@@ -8,12 +8,12 @@ export const Dashboard = () => {
const { ecosystemData } = useData();
const { searchTerm } = useStateContext();
- const filteredChains = ecosystemData?.chains.filter((chain) => {
- const chainIdStr = String(chain.id);
+ const filteredChains = ecosystemData?.zkChains.filter((chain) => {
+ const chainIdStr = String(chain.chainId);
const formattedSearchTerm = String(searchTerm).toLowerCase();
// Check if either chain name or chain ID matches the search term
- const matchesName = chain.name.toLowerCase().includes(formattedSearchTerm);
+ const matchesName = chain.chainName.toLowerCase().includes(formattedSearchTerm);
const matchesId = chainIdStr.includes(formattedSearchTerm);
return matchesName || matchesId;
@@ -27,7 +27,7 @@ export const Dashboard = () => {
- {availableChains &&
}
+ {availableChains &&
}
{!availableChains &&
}
);
diff --git a/src/containers/LockedAssets/index.tsx b/src/containers/LockedAssets/index.tsx
index 3d0212d..f86c22d 100644
--- a/src/containers/LockedAssets/index.tsx
+++ b/src/containers/LockedAssets/index.tsx
@@ -6,14 +6,14 @@ import { formatDataNumber } from '~/utils';
export const LockedAssets = () => {
const { t } = useTranslation();
- const { ecosystemData } = useData();
+ const { ecosystemData, totalL1TVL } = useData();
return (
{ecosystemData && (
<>
-
-
+
+
>
)}
diff --git a/src/data/ecosystemMockData.json b/src/data/ecosystemMockData.json
index acb7d40..fe39716 100644
--- a/src/data/ecosystemMockData.json
+++ b/src/data/ecosystemMockData.json
@@ -1,92 +1,95 @@
{
- "chains": [
+ "l1Tvl": [
{
- "name": "zkSync Era",
- "id": 324,
- "nativeToken": "ETH",
- "tvl": 1000000,
- "type": "ZKRollup"
- },
- {
- "name": "Teva Chain",
- "id": 100,
- "nativeToken": "ETH",
- "tvl": 1000000,
- "type": "ZKRollup"
- },
- {
- "name": "Cronos zkEVM",
- "id": 101,
- "nativeToken": "ETH",
- "tvl": 1000000,
- "type": "Validium"
- },
- {
- "name": "GRVT",
- "id": 102,
- "nativeToken": "ETH",
- "tvl": 1000000,
- "type": "Validium"
- },
- {
- "name": "Lens",
- "id": 103,
- "nativeToken": "ETH",
- "tvl": 1000000,
- "type": "Validium"
- },
- {
- "name": "ZKChain 104",
- "id": 104,
- "nativeToken": "ETH",
- "tvl": 1000000,
- "type": "ZKRollup"
+ "token": "ETH",
+ "total": 557596566000,
+ "imageUrl": "https://coin-images.coingecko.com/coins/images/279/large/ethereum.png?1696501628"
},
{
- "name": "ZKChain 105",
- "id": 105,
- "nativeToken": "ETH",
- "tvl": 1000000,
- "type": "Validium"
- }
- ],
- "total": 700000000,
- "tvl": [
- {
- "token": "ETH",
- "value": 557596566
+ "token": "USDT",
+ "total": 114493849618,
+ "imageUrl": "https://coin-images.coingecko.com/coins/images/325/large/tether.png"
},
{
"token": "USDC",
- "value": 90091851
+ "total": 34115209093,
+ "imageUrl": "https://coin-images.coingecko.com/coins/images/6319/large/usd-coin.png"
},
{
"token": "KOI",
- "value": 32757850
+ "total": 24115209093,
+ "imageUrl": "https://assets.coingecko.com/coins/images/35766/standard/Koi_logo.png?1709782399"
},
{
- "token": "USDT",
- "value": 18021853
+ "token": "WBTC",
+ "total": 12620248,
+ "imageUrl": "https://coin-images.coingecko.com/coins/images/5757/large/wrapped-bitcoin.png"
},
{
- "token": "WBTC",
- "value": 12620248
+ "token": "wsETH",
+ "total": 3552439,
+ "imageUrl": "https://coin-images.coingecko.com/coins/images/11320/large/staked-ether.png"
+ },
+ {
+ "token": "cbETH",
+ "total": 2552439,
+ "imageUrl": "https://assets.coingecko.com/coins/images/27008/standard/cbeth.png?1709186989"
},
{
- "token": "wstETH",
- "value": 3552439
+ "token": "BAL",
+ "total": 1552439,
+ "imageUrl": "https://coin-images.coingecko.com/coins/images/671/large/balancer.png"
+ }
+ ],
+ "ethGasInfo": {
+ "gasPrice": 50,
+ "ethTransfer": 21000,
+ "erc20Transfer": 65000
+ },
+ "zkChains": [
+ {
+ "chainId": 0,
+ "chainName": "ZKsyncERA",
+ "iconUrl": "https://s2.coinmarketcap.com/static/img/coins/64x64/24091.png",
+ "chainType": "Rollup",
+ "nativeToken": "ETH",
+ "tokenImgUrl": "https://coin-images.coingecko.com/coins/images/279/large/ethereum.png?1696501628",
+ "tvl": 1000000,
+ "metadata": true,
+ "rpc": true
},
{
- "token": "MUTE",
- "value": 2071481
+ "chainId": 1,
+ "chainName": "ZKsyncERA",
+ "iconUrl": "https://s2.coinmarketcap.com/static/img/coins/64x64/24091.png",
+ "chainType": "Validium",
+ "nativeToken": "ETH",
+ "tokenImgUrl": "https://coin-images.coingecko.com/coins/images/279/large/ethereum.png?1696501628",
+ "tvl": 500000,
+ "metadata": true,
+ "rpc": false
},
{
- "token": "rETH",
- "value": 1404096
+ "chainId": 2,
+ "chainName": "ZKsyncERA",
+ "iconUrl": "https://s2.coinmarketcap.com/static/img/coins/64x64/24091.png",
+ "chainType": "Rollup",
+ "nativeToken": "ETH",
+ "tokenImgUrl": "https://coin-images.coingecko.com/coins/images/279/large/ethereum.png?1696501628",
+ "tvl": 300000,
+ "metadata": false,
+ "rpc": true
},
{
- "token": "DAI",
- "value": 1080375
+ "chainId": 3,
+ "chainName": "ZKsyncERA",
+ "iconUrl": "https://s2.coinmarketcap.com/static/img/coins/64x64/24091.png",
+ "chainType": "Rollup",
+ "nativeToken": "ETH",
+ "tokenImgUrl": "https://coin-images.coingecko.com/coins/images/279/large/ethereum.png?1696501628",
+ "tvl": 10000,
+ "metadata": false,
+ "rpc": false
}
]
}
diff --git a/src/pages/[chain]/index.tsx b/src/pages/[chain]/index.tsx
index c5b142c..1dcbc6e 100644
--- a/src/pages/[chain]/index.tsx
+++ b/src/pages/[chain]/index.tsx
@@ -19,13 +19,13 @@ const Chain = ({ chain }: InferGetStaticPropsType
) => {
const { setSelectedChainId, refetchChainData } = useData();
useEffect(() => {
- setSelectedChainId(chain?.id);
+ setSelectedChainId(chain?.chainId);
refetchChainData({ throwOnError: true, cancelRefetch: false });
- }, [chain?.id, setSelectedChainId, refetchChainData]);
+ }, [chain?.chainId, setSelectedChainId, refetchChainData]);
return (
<>
-
+
>
);
@@ -33,11 +33,11 @@ const Chain = ({ chain }: InferGetStaticPropsType) => {
export const getStaticPaths: GetStaticPaths = async () => {
const ecosystemData = await fetchEcosystemData();
- const chains = ecosystemData.chains;
+ const chains = ecosystemData.zkChains;
const paths = SUPPORTED_LANGUAGES.flatMap((locale) =>
chains.map((chain: EcosystemChainData) => ({
- params: { chain: chain.id.toString() },
+ params: { chain: chain.chainId.toString() },
locale,
})),
);
@@ -47,9 +47,9 @@ export const getStaticPaths: GetStaticPaths = async () => {
export const getStaticProps: GetStaticProps = async ({ params, locale }: GetStaticPropsContext) => {
const ecosystemData = await fetchEcosystemData();
- const chains = ecosystemData.chains;
+ const chains = ecosystemData.zkChains;
const chainId = parseInt(params?.chain as string);
- const chain = chains.find((chain: EcosystemChainData) => chain.id === chainId);
+ const chain = chains.find((chain: EcosystemChainData) => chain.chainId === chainId);
if (!chain) {
return { notFound: true };
diff --git a/src/providers/DataProvider.tsx b/src/providers/DataProvider.tsx
index 5dc964e..6fd831e 100644
--- a/src/providers/DataProvider.tsx
+++ b/src/providers/DataProvider.tsx
@@ -2,7 +2,7 @@ import { createContext, useState, useEffect, ReactNode } from 'react';
import { useQuery, UseQueryResult } from '@tanstack/react-query';
import { useRouter } from 'next/router';
-import { ChainData, EcosystemData } from '~/types';
+import { ChainData, EcosystemData, TvlData } from '~/types';
import { fetchEcosystemData, fetchChainData } from '~/utils';
type ContextType = {
@@ -15,6 +15,8 @@ type ContextType = {
ecosystemData: EcosystemData;
chainData: ChainData;
+
+ totalL1TVL: number;
};
interface DataProps {
@@ -53,6 +55,10 @@ export const DataProvider = ({ children }: DataProps) => {
}
}, [isEcosystemError, isChainError, router]);
+ const totalL1TVL = (ecosystemData?.l1Tvl || []).reduce((accumulator: number, token: TvlData) => {
+ return accumulator + (token.total || 0);
+ }, 0);
+
return (
{
ecosystemData,
chainData,
refetchChainData,
+ totalL1TVL,
}}
>
{children}
diff --git a/src/types/Data.ts b/src/types/Data.ts
index 21121c0..30bb671 100644
--- a/src/types/Data.ts
+++ b/src/types/Data.ts
@@ -38,17 +38,29 @@ export interface ChainData {
}
export interface EcosystemChainData {
- name: string;
- id: number;
+ chainName: string;
+ chainId: number;
+ iconUrl: string;
nativeToken: string;
+ tokenImgUrl: string;
tvl: number;
- type: string;
+ chainType: string;
+ metadata: boolean;
+ rpc: boolean;
}
export interface EcosystemData {
- chains: EcosystemChainData[];
+ l1Tvl: TvlData[];
+ ethGasInfo: {
+ gasPrice: number;
+ ethTransfer: number;
+ erc20Transfer: number;
+ };
+ zkChains: EcosystemChainData[];
+}
+
+export interface TvlData {
+ token: string;
total: number;
- tvl: {
- [token: string]: number;
- }[];
+ imageUrl: string;
}
diff --git a/src/types/Theme.ts b/src/types/Theme.ts
index 050dee9..dcc5dbe 100644
--- a/src/types/Theme.ts
+++ b/src/types/Theme.ts
@@ -12,9 +12,19 @@ export interface Theme {
textFontFamily: string;
borderRadius: string;
secondaryBorderRadius: string;
+ transition: string;
border: string;
gap: string;
+ warningText: string;
+ warningBackground: string;
+ warningBorder: string;
+ emptyBackground: string;
padding: string;
+ neutral: { [key: number]: string };
+ primary: { [key: number]: string };
+ warning: { [key: number]: string };
+ error: { [key: number]: string };
+ success: { [key: number]: string };
}
export interface PropTheme {