diff --git a/.github/ISSUE_TEMPLATE/bug_report.md b/.github/ISSUE_TEMPLATE/bug_report.md deleted file mode 100644 index 59896d45..00000000 --- a/.github/ISSUE_TEMPLATE/bug_report.md +++ /dev/null @@ -1,40 +0,0 @@ ---- -name: Bug report -about: Create a report to help us improve -title: "" -labels: "Type: bug" -assignees: "" ---- - -**Describe the bug** -A clear and concise description of what the bug is. - -**To Reproduce** -Steps to reproduce the behavior: - -1. Go to '...' -2. Click on '....' -3. Scroll down to '....' -4. See error - -**Expected behavior** -A clear and concise description of what you expected to happen. - -**Screenshots** -If applicable, add screenshots to help explain your problem. - -**Desktop (please complete the following information):** - -- OS: [e.g. iOS] -- Browser [e.g. chrome, safari] -- Version [e.g. 22] - -**Smartphone (please complete the following information):** - -- Device: [e.g. iPhone6] -- OS: [e.g. iOS8.1] -- Browser [e.g. stock browser, safari] -- Version [e.g. 22] - -**Additional context** -Add any other context about the problem here. diff --git a/.github/ISSUE_TEMPLATE/contributor-issue.md b/.github/ISSUE_TEMPLATE/contributor-issue.md new file mode 100644 index 00000000..0465d4f6 --- /dev/null +++ b/.github/ISSUE_TEMPLATE/contributor-issue.md @@ -0,0 +1,47 @@ +--- +name: Contributor Issue +about: Create an issue for it to be taken by a contributor +title: Name your issue +labels: Good first issue, OD Hack, open for contribution +assignees: '' + +--- + +## Description 📹 + +**[Insert clear description here]** + +## Proposed Actions 🛠️ + +Here’s a checklist of actions to follow for resolving this issue: + +1. **Fork and Create Branch**: + Fork the repository and create a new branch using the issue number: + ```bash + git checkout -b fix-[issue-number] + ``` + +2. **Implement Changes**: + [Insert Code snippet if needed with a mardown todo list] + +3. **Run Tests and Commit Changes**: + Make sure your changes don't break existing functionality and commit with a clear message: + ```bash + git commit -m "Fix: [Short description of the fix]" + ``` + +## Required 📋 + +To keep our workflow smooth, please make sure you follow these guidelines: + +- **Assignment**: Don't create a pull request if you weren’t assigned to this issue. +- **Timeframe**: Complete the task within **3 business days**. +- **Closing the Issue**: In your PR description, close the issue by writing `Close #[issue_id]`. +- **Review Process**: + - Once you've submitted your PR, change the label to **"ready for review"**. + - If changes are requested, address them and then update the label back to **"ready for review"** once done. +- **Testing**: Test your PR locally before pushing, and verify that tests and build are working after pushing. + +Thank you for your contribution 🙏 + +⚠️ WARNING: Failure to follow the requirements above may result in being added to the OnlyDust blacklist, affecting your ability to receive future rewards. diff --git a/.github/ISSUE_TEMPLATE/feature_request.md b/.github/ISSUE_TEMPLATE/feature_request.md deleted file mode 100644 index 96e97b42..00000000 --- a/.github/ISSUE_TEMPLATE/feature_request.md +++ /dev/null @@ -1,19 +0,0 @@ ---- -name: Feature request -about: Describe the feature you want and propose some steps to achieve it -title: "" -labels: "" -assignees: "" ---- - -## Description -A clear and concise description of what the problem is. Ex. I'm always frustrated when [...] - -## PROPOSED TODO - -- [ ] Your task number 1 -- [ ] Your task number 2 -- [ ] Your task number 3 -... - - diff --git a/README.md b/README.md index 31d253be..56879184 100644 --- a/README.md +++ b/README.md @@ -10,7 +10,7 @@ Leading quests platform on Starknet to onboard your next million users. We provide a platform for users to discover new apps and for companies to pitch your products and new features to users and get user feedback. -[![Pull Requests welcome](https://img.shields.io/badge/PRs-welcome-ff69b4.svg?style=flat-square)](https://github.com/starknet-id/starknet.quest/issues?q=is:issue+is:open+label:%22open+for+contribution%22) +[![Pull Requests welcome](https://img.shields.io/badge/PRs-welcome-ff69b4.svg?style=flat-square)](https://github.com/lfglabs-dev/starknet.quest/pulls) ## About @@ -49,6 +49,17 @@ npm i npm run dev ``` +If you encounter installation issues, try these steps: + +* Clear npm cache: `npm cache clean --force` +* Delete node_modules: `rm -rf node_modules` +* Delete package-lock.json: `rm package-lock.json` +* Retry installation: `npm i` + +If issues persist, you can try `npm i --legacy-peer-deps` or `npm i --force` as a last resort. + +Note that using these flags may lead to dependency conflicts. + You should see something like this: ```sh @@ -100,4 +111,4 @@ Thanks go to these wonderful people: This project follows the [all-contributors](https://github.com/all-contributors/all-contributors) specification. -Contributions of any kind welcome! \ No newline at end of file +Contributions of any kind welcome! diff --git a/app/[addressOrDomain]/page.tsx b/app/[addressOrDomain]/page.tsx index 534a8205..308b54b9 100644 --- a/app/[addressOrDomain]/page.tsx +++ b/app/[addressOrDomain]/page.tsx @@ -12,13 +12,19 @@ import { useAccount } from "@starknet-react/core"; import Blur from "@components/shapes/blur"; import { utils } from "starknetid.js"; import { StarknetIdJsContext } from "@context/StarknetIdJsProvider"; -import { hexToDecimal } from "@utils/feltService"; +import { hexToDecimal, tokenToDecimal } from "@utils/feltService"; import { isHexString, minifyAddress } from "@utils/stringService"; import ProfileCardSkeleton from "@components/skeletons/profileCardSkeleton"; import { getDataFromId } from "@services/starknetIdService"; import { usePathname, useRouter } from "next/navigation"; import ErrorScreen from "@components/UI/screens/errorScreen"; -import { CompletedQuests } from "../../types/backTypes"; +import { + ArgentDappMap, + ArgentTokenMap, + ArgentUserDapp, + ArgentUserToken, + CompletedQuests, +} from "../../types/backTypes"; import QuestSkeleton from "@components/skeletons/questsSkeleton"; import QuestCardCustomised from "@components/dashboard/CustomisedQuestCard"; import QuestStyles from "@styles/Home.module.css"; @@ -31,6 +37,17 @@ import { TEXT_TYPE } from "@constants/typography"; import { a11yProps } from "@components/UI/tabs/a11y"; import { CustomTabPanel } from "@components/UI/tabs/customTab"; import SuggestedQuests from "@components/dashboard/SuggestedQuests"; +import PortfolioSummary from "@components/dashboard/PortfolioSummary"; +import { useNotification } from "@context/NotificationProvider"; +import { + calculateTokenPrice, + fetchDapps, + fetchTokens, + fetchUserDapps, + fetchUserTokens, +} from "@services/argentPortfolioService"; +import PortfolioSummarySkeleton from "@components/skeletons/portfolioSummarySkeleton"; +import { useDisplayName } from "@hooks/displayName.tsx"; type AddressOrDomainProps = { params: { @@ -38,10 +55,23 @@ type AddressOrDomainProps = { }; }; +type ChartItemMap = { + [dappId: string]: ChartItem; +}; + +type DebtStatus = { + hasDebt: boolean; + tokens: { + dappId: string; + tokenAddress: string; + tokenBalance: number; + }[]; +}; + export default function Page({ params }: AddressOrDomainProps) { const router = useRouter(); const addressOrDomain = params.addressOrDomain; - const { address } = useAccount(); + const { showNotification } = useNotification(); const { starknetIdNavigator } = useContext(StarknetIdJsContext); const [initProfile, setInitProfile] = useState(false); const { getBoostClaimStatus } = useBoost(); @@ -62,6 +92,22 @@ export default function Page({ params }: AddressOrDomainProps) { const [questsLoading, setQuestsLoading] = useState(true); const [tabIndex, setTabIndex] = React.useState(0); const [claimableQuests, setClaimableQuests] = useState([]); + const [portfolioAssets, setPortfolioAssets] = useState([]); + const [portfolioProtocols, setPortfolioProtocols] = useState([]); + const [loadingProtocols, setLoadingProtocols] = useState(true); + const { address } = useAccount(); + + useEffect(() => { + if (!address) return; + + // Redirect to the new address profile + const updateUrl = (address: string) => { + const newUrl = `/${address}`; + router.replace(newUrl); + }; + + updateUrl(address); + }, [address, router]); const handleChangeTab = useCallback( (event: React.SyntheticEvent, newValue: number) => { @@ -168,10 +214,404 @@ export default function Page({ params }: AddressOrDomainProps) { setQuestsLoading(false); }, []); + const calculateAssetPercentages = async ( + userTokens: ArgentUserToken[], + tokens: ArgentTokenMap, + dapps: ArgentDappMap, + userDapps: ArgentUserDapp[] + ) => { + let totalValue = 0; + const assetValues: { [symbol: string]: number } = {}; + + // Process user tokens in parallel + const userTokenPromises = userTokens.map(async (token) => { + const tokenInfo = tokens[token.tokenAddress]; + if (!tokenInfo || token.tokenBalance === "0") return null; + + // Skip protocol tokens (like LPT pair tokens, staking, etc.) + if (tokenInfo.dappId) { + return null; + } + + try { + const value = await calculateTokenPrice( + token.tokenAddress, + tokenToDecimal(token.tokenBalance, tokenInfo.decimals), + "USD" + ); + return { + value, + symbol: tokenInfo.symbol || "Unknown", + isProtocolToken: !!tokenInfo.dappId, + }; + } catch (err) { + console.log( + `Error calculating price for token ${token.tokenAddress}:`, + err + ); + return null; + } + }); + + // Flatten userDapps into an array of token balances + const dappBalances = userDapps.flatMap( + (dapp) => + dapp.products[0]?.positions.flatMap((position) => + Object.entries(position.totalBalances).map( + ([tokenAddress, balance]) => ({ + tokenAddress, + balance, + dappId: dapp.dappId, + }) + ) + ) ?? [] + ); + + // Process all balances in parallel + const balancePromises = dappBalances.map( + async ({ tokenAddress, balance, dappId }) => { + const tokenInfo = tokens[tokenAddress]; + if (!tokenInfo || balance === "0") return null; + + try { + const value = await calculateTokenPrice( + tokenAddress, + tokenToDecimal(balance, tokenInfo.decimals), + "USD" + ); + + return { + value, + symbol: tokenInfo.symbol || "Unknown", + isProtocolToken: !!tokenInfo.dappId, + }; + } catch (err) { + console.log( + `Error calculating price for token ${tokenAddress}:`, + err + ); + return null; + } + } + ); + + // Process results + const results = ( + await Promise.all([...balancePromises, ...userTokenPromises]) + ).filter(Boolean); + + results.forEach((result) => { + if (!result) return; + const { value, symbol, isProtocolToken } = result; + + if (value < 0) return; // Skip negative balances + + totalValue += value; + + if (!isProtocolToken) { + assetValues[symbol] = (assetValues[symbol] || 0) + value; + } + }); + // Convert to percentages and format + const sortedAssets = Object.entries(assetValues) + .sort(([, a], [, b]) => b - a) + .map(([symbol, value]) => ({ + itemLabel: symbol, + itemValue: ((value / totalValue) * 100).toFixed(2), + itemValueSymbol: "%", + color: "", // Colors will be assigned later + })); + + // Handle "Others" category if needed + if (sortedAssets.length > 4) { + const others = sortedAssets + .slice(4) + .reduce((sum, asset) => sum + parseFloat(asset.itemValue), 0); + sortedAssets.splice(4); + sortedAssets.push({ + itemLabel: "Others", + itemValue: others.toFixed(2), + itemValueSymbol: "%", + color: "", + }); + } + + // Assign colors + const colors = ["#1E2097", "#637DEB", "#2775CA", "#5CE3FE", "#F4FAFF"]; + sortedAssets.forEach((asset, index) => { + asset.color = colors[index % colors.length]; // Use modulo to recycle colors if needed + }); + return sortedAssets; + }; + + const fetchPortfolioAssets = useCallback( + async (data: { + dapps: ArgentDappMap; + tokens: ArgentTokenMap; + userTokens: ArgentUserToken[]; + userDapps: ArgentUserDapp[]; + }) => { + const { dapps, tokens, userTokens, userDapps } = data; + try { + if (!tokens || !userTokens || !dapps || !userDapps) { + console.warn("Missing required data for portfolio calculation"); + return; + } + const assets = await calculateAssetPercentages( + userTokens, + tokens, + dapps, + userDapps + ); + setPortfolioAssets(assets); + } catch (error) { + showNotification("Error while fetching portfolio assets", "error"); + console.log("Error while fetching portfolio assets", error); + } + }, + [] + ); + + const userHasDebt = (userDapps: ArgentUserDapp[]) => { + let debt: DebtStatus = { hasDebt: false, tokens: [] }; + + for (const dapp of userDapps) { + if (!dapp.products[0]) { + continue; + } + for (const position of dapp.products[0].positions) { + for (const tokenAddress of Object.keys(position.totalBalances)) { + const tokenBalance = Number(position.totalBalances[tokenAddress]); + if (tokenBalance < 0) { + debt.hasDebt = true; + debt.tokens.push({ + dappId: dapp.dappId, + tokenAddress, + tokenBalance, + }); + } + } + } + } + return debt; + }; + + const handleDebt = async ( + protocolsMap: ChartItemMap, + userDapps: ArgentUserDapp[], + tokens: ArgentTokenMap + ) => { + const debtStatus = userHasDebt(userDapps); + if (!debtStatus || !debtStatus.hasDebt) { + return; + } + + for await (const debt of debtStatus.tokens) { + let value = Number(protocolsMap[debt.dappId].itemValue); + value += await calculateTokenPrice( + debt.tokenAddress, + tokenToDecimal( + debt.tokenBalance.toString(), + tokens[debt.tokenAddress].decimals + ), + "USD" + ); + + protocolsMap[debt.dappId].itemValue = value.toFixed(2); + } + }; + + const getProtocolsFromTokens = async ( + protocolsMap: ChartItemMap, + userTokens: ArgentUserToken[], + tokens: ArgentTokenMap, + dapps: ArgentDappMap + ) => { + for await (const token of userTokens) { + const tokenInfo = tokens[token.tokenAddress]; + if (tokenInfo.dappId && token.tokenBalance != "0") { + let itemValue = 0; + const currentTokenBalance = await calculateTokenPrice( + token.tokenAddress, + tokenToDecimal(token.tokenBalance, tokenInfo.decimals), + "USD" + ); + + if (protocolsMap[tokenInfo.dappId]?.itemValue) { + itemValue = + Number(protocolsMap[tokenInfo.dappId].itemValue) + + currentTokenBalance; + } else { + itemValue = currentTokenBalance; + } + + protocolsMap[tokenInfo.dappId] = { + color: "", + itemLabel: dapps[tokenInfo.dappId].name, + itemValueSymbol: "$", + itemValue: itemValue.toFixed(2), + }; + } + } + }; + + const getProtocolsFromDapps = async ( + protocolsMap: ChartItemMap, + userDapps: ArgentUserDapp[], + tokens: ArgentTokenMap, + dapps: ArgentDappMap + ) => { + for await (const userDapp of userDapps) { + if (protocolsMap[userDapp.dappId]) { + continue; + } // Ignore entry if already present in the map + + let protocolBalance = 0; + if (!userDapp.products[0]) { + return; + } + for await (const position of userDapp.products[0].positions) { + for await (const tokenAddress of Object.keys(position.totalBalances)) { + protocolBalance += await calculateTokenPrice( + tokenAddress, + tokenToDecimal( + position.totalBalances[tokenAddress], + tokens[tokenAddress].decimals + ), + "USD" + ); + } + } + + protocolsMap[userDapp.dappId] = { + color: "", + itemLabel: dapps[userDapp.dappId].name, + itemValueSymbol: "$", + itemValue: protocolBalance.toFixed(2), + }; + } + }; + + const sortProtocols = (protocolsMap: ChartItemMap) => { + return Object.values(protocolsMap).sort( + (a, b) => parseFloat(b.itemValue) - parseFloat(a.itemValue) + ); + }; + + const handleExtraProtocols = (sortedProtocols: ChartItem[]) => { + let otherProtocols = + sortedProtocols.length > 5 ? sortedProtocols.splice(4) : []; + if (otherProtocols.length === 0) { + return; + } + sortedProtocols.push({ + itemLabel: "Others", + itemValue: otherProtocols + .reduce( + (valueSum, protocol) => valueSum + Number(protocol.itemValue), + 0 + ) + .toFixed(2), + itemValueSymbol: "$", + color: "", + }); + }; + + const assignProtocolColors = (sortedProtocols: ChartItem[]) => { + const portfolioProtocolColors = [ + "#278015", + "#23F51F", + "#DEFE5C", + "#9EFABB", + "#F4FAFF", + ]; + sortedProtocols.forEach((protocol, index) => { + protocol.color = portfolioProtocolColors[index]; + }); + }; + + const fetchPortfolioProtocols = useCallback( + async (data: { + dapps: ArgentDappMap; + tokens: ArgentTokenMap; + userTokens: ArgentUserToken[]; + userDapps: ArgentUserDapp[]; + }) => { + const { dapps, tokens, userTokens, userDapps } = data; + + if (!dapps || !tokens || (!userTokens && !userDapps)) return; + let protocolsMap: ChartItemMap = {}; + + try { + await getProtocolsFromTokens(protocolsMap, userTokens, tokens, dapps); + await handleDebt(protocolsMap, userDapps, tokens); // Tokens show debt as balance 0, so need to handle it manually + await getProtocolsFromDapps(protocolsMap, userDapps, tokens, dapps); + + let sortedProtocols = sortProtocols(protocolsMap); + handleExtraProtocols(sortedProtocols); + assignProtocolColors(sortedProtocols); + + setPortfolioProtocols(sortedProtocols); + } catch (error) { + showNotification( + "Error while calculating address portfolio stats", + "error" + ); + console.log("Error while calculating address portfolio stats", error); + } + }, + [address] + ); + + const fetchPortfolioData = useCallback( + async (addr: string, abortController: AbortController) => { + setLoadingProtocols(true); + try { + // Argent API requires lowercase address + const normalizedAddr = addr.toLowerCase(); + const [dappsData, tokensData, userTokensData, userDappsData] = + await Promise.all([ + fetchDapps({ signal: abortController.signal }), + fetchTokens({ signal: abortController.signal }), + fetchUserTokens(normalizedAddr, { signal: abortController.signal }), + fetchUserDapps(normalizedAddr, { signal: abortController.signal }), + ]); + + const data = { + dapps: dappsData, + tokens: tokensData, + userTokens: userTokensData, + userDapps: userDappsData, + }; + + await Promise.all([ + fetchPortfolioProtocols(data), + fetchPortfolioAssets(data), + ]); + } catch (error) { + console.log("Error while fetching address portfolio", error); + if (error instanceof Error && error.name === "AbortError") { + // Do not show notification for AbortError + return; + } + + showNotification("Error while fetching address portfolio", "error"); + } finally { + setLoadingProtocols(false); + } + }, + [fetchPortfolioProtocols, fetchPortfolioAssets] + ); + useEffect(() => { + const abortController = new AbortController(); + if (!identity) return; fetchQuestData(identity.owner); fetchPageData(identity.owner); + fetchPortfolioData(identity.owner, abortController); + + return () => abortController.abort(); }, [identity]); useEffect(() => setNotFound(false), [dynamicRoute]); @@ -296,8 +736,8 @@ export default function Page({ params }: AddressOrDomainProps) { if (notFound) { return ( router.push("/")} /> ); @@ -315,7 +755,6 @@ export default function Page({ params }: AddressOrDomainProps) { {initProfile && identity ? ( + {/* Portfolio charts */} +
+ {loadingProtocols ? ( // Change for corresponding state + + ) : ( + sum + Number(item.itemValue), + 0 + )} + isProtocol={false} + /> + )} + {loadingProtocols ? ( + + ) : ( + sum + Number(item.itemValue), + 0 + )} + isProtocol={true} + /> + )} +
+ {/* Completed Quests */}
@@ -332,11 +801,11 @@ export default function Page({ params }: AddressOrDomainProps) { style={{ borderBottom: "0.5px solid rgba(224, 224, 224, 0.3)", }} - className="pb-6" + className='pb-6' value={tabIndex} onChange={handleChangeTab} - aria-label="quests and collectons tabs" - indicatorColor="secondary" + aria-label='quests and collectons tabs' + indicatorColor='secondary' >
- +
{questsLoading ? ( ) : completedQuests?.length === 0 ? ( - isOwner - ? - : User has not completed any quests at the moment + isOwner ? ( + + ) : ( + + User has not completed any quests at the moment + + ) ) : (
{completedQuests?.length > 0 && completedQuests?.map((quest) => ( - + ))}
@@ -391,11 +873,14 @@ export default function Page({ params }: AddressOrDomainProps) {
- + {questsLoading ? ( ) : ( -
+
{claimableQuests && claimableQuests.map((quest) => ( ({ + best_users: [], + total_users: 0, + }); + const isTop50RankedView = useMemo( () => !currentSearchedAddress && @@ -87,18 +94,21 @@ export default function Page() { // set user address on wallet connect and disconnect useEffect(() => { - setTimeout(() => { - setApiCallDelay(true); - }, 1000); - if (address === "") return; + const timeoutId = setTimeout(() => setApiCallDelay(true), 1000); if (address) setUserAddress(address); if (status === "disconnected") setUserAddress(""); + return () => clearTimeout(timeoutId); // Cleanup }, [address, status]); useEffect(() => { if (!apiCallDelay) return; - fetchPageData(); - }, [apiCallDelay]); + const fetchTimeout = setTimeout(() => { + fetchPageData(); + }, 500); + + return () => clearTimeout(fetchTimeout); +}, [apiCallDelay]); + const fetchRankingResults = useCallback( async (requestBody: LeaderboardRankingParams) => { @@ -113,10 +123,9 @@ export default function Page() { const addRankingResults = useCallback( async (requestBody: LeaderboardRankingParams) => { const response = await fetchLeaderboardRankings(requestBody); - if (response) - setRanking((prev) => { - return { ...prev, ranking: [...prev.ranking, ...response.ranking] }; - }); + if (response) { + setRanking((prev) => ({ ...prev, ranking: [...prev.ranking, ...response.ranking] })); + } }, [] ); @@ -131,10 +140,7 @@ export default function Page() { const fetchPageData = useCallback(async () => { const requestBody = { - addr: - status === "connected" - ? hexToDecimal(address && address?.length > 0 ? address : userAddress) - : "", + addr: status === "connected" ? (address || userAddress) : "", page_size: 10, shift: 0, duration: timeFrameMap(duration), @@ -154,11 +160,7 @@ export default function Page() { status, ]); - const [leaderboardToppers, setLeaderboardToppers] = - useState({ - best_users: [], - total_users: 0, - }); + const contract = useMemo(() => { return new Contract( @@ -206,10 +208,7 @@ export default function Page() { useEffect(() => { const checkIfValidAddress = async (address: string) => { try { - let domain = address; - if (isStarkDomain(address)) { - domain = getDomainWithoutStark(address); - } + const domain = isStarkDomain(address) ? getDomainWithoutStark(address) : address; const res: { message: boolean } = await verifyDomain(domain); if (res.message) { setSearchResults([domain.concat(".stark")]); @@ -260,12 +259,7 @@ export default function Page() { } if (!checkIfLastPage && viewMore) { const requestBody = { - addr: - currentSearchedAddress.length > 0 - ? currentSearchedAddress - : userAddress - ? hexToDecimal(userAddress) - : "", + addr: currentSearchedAddress || (userAddress ? (userAddress) : ""), page_size: rowsPerPage, shift: currentPage, duration: timeFrameMap(duration), @@ -294,14 +288,7 @@ export default function Page() { */ useEffect(() => { const requestBody = { - addr: - currentSearchedAddress.length > 0 - ? currentSearchedAddress - : userAddress - ? hexToDecimal(userAddress) - : address - ? address - : "", + addr: currentSearchedAddress || (userAddress ? (userAddress) : address || ""), page_size: rowsPerPage, shift: 0, duration: timeFrameMap(duration), @@ -326,14 +313,7 @@ export default function Page() { useEffect(() => { if (inititalFetchTop50 && address && duration !== TOP_50_TAB_STRING) { const requestBody = { - addr: - currentSearchedAddress.length > 0 - ? currentSearchedAddress - : userAddress - ? hexToDecimal(userAddress) - : address - ? address - : "", + addr: currentSearchedAddress || (userAddress ? (userAddress) : address || ""), page_size: rowsPerPage, shift: 0, duration: timeFrameMap(duration), @@ -453,7 +433,7 @@ export default function Page() { 0 - ? currentSearchedAddress + ? decimalToHex(currentSearchedAddress) : userAddress } /> @@ -544,7 +524,7 @@ export default function Page() { selectedAddress={ currentSearchedAddress.length > 0 ? currentSearchedAddress - : hexToDecimal(userAddress) + : (userAddress) } searchedAddress={currentSearchedAddress} leaderboardToppers={leaderboardToppers} diff --git a/app/not-connected/page.tsx b/app/not-connected/page.tsx index 952eea78..7886c4ce 100644 --- a/app/not-connected/page.tsx +++ b/app/not-connected/page.tsx @@ -1,6 +1,6 @@ "use client"; -import { useAccount, useConnect } from "@starknet-react/core"; +import { useAccount, useConnect, Connector } from "@starknet-react/core"; import React, { useEffect } from "react"; import { useRouter } from "next/navigation"; import ErrorScreen from "@components/UI/screens/errorScreen"; @@ -12,7 +12,7 @@ export default function Page() { const { connectAsync } = useConnect(); const { push } = useRouter(); const { starknetkitConnectModal } = useStarknetkitConnectModal({ - connectors: availableConnectors, + connectors: availableConnectors as any, modalTheme: "dark", }); @@ -25,14 +25,14 @@ export default function Page() { if (!connector) { return; } - await connectAsync({ connector }); + await connectAsync({ connector: connector as Connector }); // Type casted localStorage.setItem("SQ-connectedWallet", connector.id); }; return ( <> diff --git a/app/provider.tsx b/app/provider.tsx index a85944a5..0ec6b2fb 100644 --- a/app/provider.tsx +++ b/app/provider.tsx @@ -1,30 +1,30 @@ -"use client"; +'use client'; -import React from "react"; -import { InjectedConnector } from "starknetkit/injected"; -import { WebWalletConnector } from "starknetkit/webwallet"; -import { ArgentMobileConnector } from "starknetkit/argentMobile"; -import { Chain, mainnet, sepolia } from "@starknet-react/chains"; +import React from 'react'; +import { InjectedConnector } from 'starknetkit/injected'; +import { WebWalletConnector } from 'starknetkit/webwallet'; +import { ArgentMobileConnector } from 'starknetkit/argentMobile'; +import { Chain, mainnet, sepolia } from '@starknet-react/chains'; import { Connector, StarknetConfig, jsonRpcProvider, -} from "@starknet-react/core"; -import { StarknetIdJsProvider } from "@context/StarknetIdJsProvider"; -import { ThemeProvider, createTheme } from "@mui/material"; -import { QuestsContextProvider } from "@context/QuestsProvider"; -import { getCurrentNetwork } from "@utils/network"; -import { constants } from "starknet"; -import { PostHogProvider } from "posthog-js/react"; -import posthog from "posthog-js"; -import { NotificationProvider } from "@context/NotificationProvider"; -import { LocalizationProvider } from "@mui/x-date-pickers"; -import { AdapterDayjs } from "@mui/x-date-pickers/AdapterDayjs"; +} from '@starknet-react/core'; +import { StarknetIdJsProvider } from '@context/StarknetIdJsProvider'; +import { ThemeProvider, createTheme } from '@mui/material'; +import { QuestsContextProvider } from '@context/QuestsProvider'; +import { getCurrentNetwork } from '@utils/network'; +import { constants } from 'starknet'; +import { PostHogProvider } from 'posthog-js/react'; +import posthog from 'posthog-js'; +import { NotificationProvider } from '@context/NotificationProvider'; +import { LocalizationProvider } from '@mui/x-date-pickers'; +import { AdapterDayjs } from '@mui/x-date-pickers/AdapterDayjs'; // Traffic measures -if (typeof window !== "undefined") { +if (typeof window !== 'undefined') { posthog.init(process.env.NEXT_PUBLIC_POSTHOG_KEY as string, { - api_host: "https://app.posthog.com", + api_host: 'https://app.posthog.com', session_recording: { recordCrossOriginIframes: true, }, @@ -34,26 +34,46 @@ if (typeof window !== "undefined") { (window as any).posthog = posthog; } +function isBitgetWalletInstalled() { + // Check if the Bitget wallet is available on the window object + return typeof window !== "undefined" && window.bitget !== undefined; +} + export const availableConnectors = [ new InjectedConnector({ options: { id: "braavos", name: "Braavos" } }), new InjectedConnector({ options: { id: "argentX", name: "Argent X" } }), + new InjectedConnector({ + options: { + id: "bitkeep", + name: isBitgetWalletInstalled() + ? "Bitget Wallet" + : "Install Bitget Wallet", + icon: "", + }, + }), + new InjectedConnector({ options: { id: "okxwallet", name: "Okx Wallet" } }), + new WebWalletConnector({ url: - getCurrentNetwork() === "TESTNET" - ? "https://web.hydrogen.argent47.net" - : "https://web.argent.xyz/", + getCurrentNetwork() === 'TESTNET' + ? 'https://web.hydrogen.argent47.net' + : 'https://web.argent.xyz/', }), - new ArgentMobileConnector({ - dappName: "Starknet Quest", - url: process.env.NEXT_PUBLIC_APP_LINK as string, - chainId: constants.NetworkName.SN_MAIN, - icons: ["https://starknet.quest/visuals/starknetquestLogo.svg"], + ArgentMobileConnector.init({ + options: { + dappName: "Starknet Quest", + url: process.env.NEXT_PUBLIC_APP_LINK as string, + chainId: constants.NetworkName.SN_MAIN, + icons: ["https://starknet.quest/visuals/starknetquestLogo.svg"], + }, }), + + new InjectedConnector({ options: { id: 'keplr', name: 'Keplr' } }), ]; export function Providers({ children }: { children: React.ReactNode }) { const network = getCurrentNetwork(); - const chains = [network === "TESTNET" ? sepolia : mainnet]; + const chains = [network === 'TESTNET' ? sepolia : mainnet]; const provider = jsonRpcProvider({ // eslint-disable-next-line @typescript-eslint/no-unused-vars rpc: (_chain: Chain) => ({ @@ -64,33 +84,33 @@ export function Providers({ children }: { children: React.ReactNode }) { const theme = createTheme({ palette: { primary: { - main: "#6affaf", - light: "#5ce3fe", + main: '#6affaf', + light: '#5ce3fe', }, secondary: { - main: "#f4faff", - dark: "#eae0d5", + main: '#f4faff', + dark: '#eae0d5', }, background: { - default: "#29282b", + default: '#29282b', }, }, components: { MuiTabs: { styleOverrides: { root: { - "& .MuiTabs-flexContainer": { - display: "flex", - flexDirection: "column", // For mobile versions - alignItems: "center", - ["@media (min-width:768px)"]: { - flexDirection: "row", // For desktop versions + '& .MuiTabs-flexContainer': { + display: 'flex', + flexDirection: 'column', // For mobile versions + alignItems: 'center', + ['@media (min-width:768px)']: { + flexDirection: 'row', // For desktop versions }, }, }, // Overrides the styles for the selected tab indicator indicator: { - backgroundColor: "transparent", + backgroundColor: 'transparent', }, }, }, @@ -98,14 +118,14 @@ export function Providers({ children }: { children: React.ReactNode }) { styleOverrides: { // Overrides the styles for unselected tabs root: { - color: "#E1DCEA", // Text color for unselected tabs - width: "100%", - ["@media (min-width:768px)"]: { - width: "fit-content", + color: '#E1DCEA', // Text color for unselected tabs + width: '100%', + ['@media (min-width:768px)']: { + width: 'fit-content', }, - "&.Mui-selected": { - color: "#000", // Text color for the selected tab - backgroundColor: "#fff", // Background of the selected tab + '&.Mui-selected': { + color: '#000', // Text color for the selected tab + backgroundColor: '#fff', // Background of the selected tab }, }, }, @@ -125,7 +145,7 @@ export function Providers({ children }: { children: React.ReactNode }) { - {children} + {children} @@ -133,4 +153,4 @@ export function Providers({ children }: { children: React.ReactNode }) { ); -} +} \ No newline at end of file diff --git a/app/quest/[questPage]/quest.tsx b/app/quest/[questPage]/quest.tsx index fce58c6f..82226f77 100644 --- a/app/quest/[questPage]/quest.tsx +++ b/app/quest/[questPage]/quest.tsx @@ -54,8 +54,7 @@ const Quest: FunctionComponent = ({ } }, []); - // this fetches quest data - useEffect(() => { + const fetchQuestData = useCallback(async () => { getQuestById(questId) .then((data) => { if (!data) { @@ -79,6 +78,11 @@ const Quest: FunctionComponent = ({ }); }, [questId]); + useEffect(() => { + if (!questId) return; + fetchQuestData(); + }, [questId]); + useEffect(() => { // dont log if questId is not present if (!questId) return; @@ -147,6 +151,7 @@ const Quest: FunctionComponent = ({ setShowDomainPopup={setShowDomainPopup} hasRootDomain={hasRootDomain} hasNftReward={hasNftReward} + fetchQuestData={fetchQuestData} />
diff --git a/components/UI/avatar.tsx b/components/UI/avatar.tsx index 61ae9b90..2f6e0f16 100644 --- a/components/UI/avatar.tsx +++ b/components/UI/avatar.tsx @@ -1,7 +1,13 @@ -import React, { FunctionComponent } from "react"; +import React, { + FunctionComponent, + useContext, + useEffect, + useState, +} from "react"; import ProfilIcon from "@components/UI/iconsComponents/icons/profilIcon"; import theme from "@styles/theme"; -import { useStarkProfile } from "@starknet-react/core"; +import { StarknetIdJsContext } from "@context/StarknetIdJsProvider"; +import { StarkProfile } from "starknetid.js"; type AvatarProps = { address: string; @@ -9,17 +15,24 @@ type AvatarProps = { }; const Avatar: FunctionComponent = ({ address, width = "32" }) => { - const { data: profileData } = useStarkProfile({ address }); - + const { starknetIdNavigator } = useContext(StarknetIdJsContext); + const [profile, setProfile] = useState(null); + useEffect(() => { + if (!starknetIdNavigator) return; + starknetIdNavigator.getProfileData(address).then((profile) => { + setProfile(profile); + }); + }, [starknetIdNavigator, address]); return ( <> - {profileData?.profilePicture ? ( + {profile?.profilePicture ? ( {`${profile?.name?.length ) : ( diff --git a/components/UI/changeWallet.tsx b/components/UI/changeWallet.tsx index b35c14a1..966cd973 100644 --- a/components/UI/changeWallet.tsx +++ b/components/UI/changeWallet.tsx @@ -10,6 +10,10 @@ import useGetDiscoveryWallets from "@hooks/useGetDiscoveryWallets"; import Typography from "./typography/typography"; import { TEXT_TYPE } from "@constants/typography"; +// Define an array of wallet IDs with explicit types +const INSTALLABLE_WALLETS = ["braavos", "argentX", "bitkeep"] as const; +type WalletId = typeof INSTALLABLE_WALLETS[number]; + type ChangeWalletProps = { closeWallet: () => void; hasWallet: boolean; @@ -53,7 +57,13 @@ const ChangeWallet: FunctionComponent = ({ > - Change wallet + + Change wallet + {connectors.map((connector) => { if (connector.available()) { return ( @@ -61,8 +71,7 @@ const ChangeWallet: FunctionComponent = ({
); }; + export default ChangeWallet; diff --git a/components/UI/iconsComponents/icons/eyeIcon.tsx b/components/UI/iconsComponents/icons/eyeIcon.tsx new file mode 100644 index 00000000..dfcbd537 --- /dev/null +++ b/components/UI/iconsComponents/icons/eyeIcon.tsx @@ -0,0 +1,54 @@ +import React, { FunctionComponent } from "react"; + +interface IconProps { + color?: string; + width?: number; +} + +const EyeIcon: FunctionComponent = ({ + color = "#E1DCEA", + width = 17, +}) => { + return ( + + + + + ); +}; + +const EyeIconSlashed: FunctionComponent = ({ + color = "#E1DCEA", + width = 17, +}) => { + return ( + + + + ); +}; + +export { EyeIcon, EyeIconSlashed }; diff --git a/components/UI/iconsComponents/icons/walletIcons.tsx b/components/UI/iconsComponents/icons/walletIcons.tsx index 89a4552e..efc72565 100644 --- a/components/UI/iconsComponents/icons/walletIcons.tsx +++ b/components/UI/iconsComponents/icons/walletIcons.tsx @@ -33,6 +33,59 @@ const WalletIcons: FunctionComponent = ({ id }) => { ); + if (id === "bitkeep") + return ( + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + ); + + return ( { const currentNetwork = getCurrentNetwork(); @@ -53,10 +59,11 @@ const Navbar: FunctionComponent = () => { linkText: "", }, ]); + const sortedConnectors = sortConnectors(availableConnectors); + const { starknetkitConnectModal } = useStarknetkitConnectModal({ - connectors: availableConnectors, + connectors: sortedConnectors as any, }); - const fetchAndUpdateNotifications = async () => { if (!address) return; const res = await getPendingBoostClaims(hexToDecimal(address)); @@ -94,7 +101,7 @@ const Navbar: FunctionComponent = () => { const connector = availableConnectors.find( (item) => item.id === connectordId ); - await connectAsync({ connector }); + await connectAsync({ connector: connector as Connector }); } }; connectToStarknet(); @@ -121,7 +128,7 @@ const Navbar: FunctionComponent = () => { if (!connector) { return; } - await connectAsync({ connector }); + await connectAsync({ connector: connector as Connector }); localStorage.setItem("SQ-connectedWallet", connector.id); }; @@ -174,18 +181,24 @@ const Navbar: FunctionComponent = () => { return ( <> -