diff --git a/app/analytics/page.tsx b/app/analytics/page.tsx index 26b92589..d0971ec8 100644 --- a/app/analytics/page.tsx +++ b/app/analytics/page.tsx @@ -17,12 +17,12 @@ export default function Page() { const { address } = useAccount(); const [loading, setLoading] = useState(true); - const [quests, setQuests] = useState ({} as QuestList); + const [quests, setQuests] = useState({} as QuestList); const fetchQuests = useCallback(async () => { try { setLoading(true); - const res = await getQuests() || {}; + const res = (await getQuests()) || {}; setQuests(res); setLoading(false); } catch (error) { @@ -39,34 +39,42 @@ export default function Page() {
router.back()} />
- Quest Analytics + + Quest Analytics +
{loading ? ( ) : ( - (Object.keys(quests) as (keyof typeof quests)[]).map((categoryName: keyof typeof quests) => { - const categoryValue = quests[categoryName]; - if (Array.isArray(categoryValue)) { - return categoryValue.map((quest: QuestDocument) => { - return ( - router.push(`/analytics/${quest.id}`)} - imgSrc={quest.img_card} - issuer={{ - name: quest.issuer, - logoFavicon: quest.logo, - }} - reward={quest.rewards_title} - id={quest.id} - expired={false} - /> - ); - }); + (Object.keys(quests) as (keyof typeof quests)[]).map( + (categoryName: keyof typeof quests) => { + const categoryValue = quests[categoryName]; + if (Array.isArray(categoryValue)) { + return categoryValue.map((quest: QuestDocument) => { + return ( + router.push(`/analytics/${quest.id}`)} + imgSrc={quest.img_card} + issuer={{ + name: quest.issuer, + logoFavicon: quest.logo, + }} + reward={quest.rewards_title} + id={quest.id} + expired={false} + /> + ); + }); + } + return null; } - return null; - }) + ) )}
diff --git a/app/provider.tsx b/app/provider.tsx index 12aa217a..e601d4ef 100644 --- a/app/provider.tsx +++ b/app/provider.tsx @@ -1,9 +1,6 @@ "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 { Connector, @@ -11,10 +8,9 @@ import { jsonRpcProvider, } from "@starknet-react/core"; import { StarknetIdJsProvider } from "@context/StarknetIdJsProvider"; -import { ThemeProvider, createTheme } from "@mui/material"; +import { ThemeProvider } 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"; diff --git a/context/QuestsProvider.tsx b/context/QuestsProvider.tsx index 6dcfb124..e9799cad 100644 --- a/context/QuestsProvider.tsx +++ b/context/QuestsProvider.tsx @@ -138,7 +138,6 @@ export const QuestsContextProvider = ({ !quest.expired ) : undefined; - setFeaturedQuest( lastBoostedQuest || notExpired[Math.floor(Math.random() * notExpired.length)] diff --git a/jest.config.js b/jest.config.js index 4b91002b..385c7f18 100644 --- a/jest.config.js +++ b/jest.config.js @@ -2,7 +2,7 @@ module.exports = { testEnvironment: "node", transform: { - "\\.[jt]sx?$": "ts-jest", + "^.+\\.(js|jsx|ts|tsx)$": ["babel-jest", { presets: ["next/babel"] }], }, setupFilesAfterEnv: ["./jest.setup.js"], moduleNameMapper: { @@ -16,9 +16,10 @@ module.exports = { coverageThreshold: { "./utils/": { branches: 75, - functions: 90, + functions: 80, lines: 90, - statements: -10, + statements: -23, }, }, -}; \ No newline at end of file + transformIgnorePatterns: ["node_modules/(?!(@starknet-react|@starknetkit)/)"], +}; diff --git a/package.json b/package.json index 1c695cb4..e0773eed 100644 --- a/package.json +++ b/package.json @@ -30,7 +30,8 @@ "chart.js": "^4.3.0", "dayjs": "^1.11.11", "eslint-config-next": "^14.0.1", - "get-starknet-core": "^3.2.0", + "get-starknet": "^3.3.3", + "get-starknet-core": "^3.3.3", "jotai": "^2.5.0", "jwt-decode": "^4.0.0", "lottie-react": "^2.4.0", @@ -51,7 +52,7 @@ "recharts": "^2.12.2", "starknet": "^6.11.0", "starknetid.js": "^4.0.1", - "starknetkit": "^2.3.3", + "starknetkit": "^2.6.1", "three": "^0.155.0", "twitter-api-sdk": "^1.2.1" }, diff --git a/public/visuals/braavosMobile.webp b/public/visuals/braavosMobile.webp new file mode 100644 index 00000000..c9acf4d4 Binary files /dev/null and b/public/visuals/braavosMobile.webp differ diff --git a/tests/utils/braavosMobile.test.js b/tests/utils/braavosMobile.test.js new file mode 100644 index 00000000..f7511826 --- /dev/null +++ b/tests/utils/braavosMobile.test.js @@ -0,0 +1,86 @@ +import { mainnet, sepolia } from "@starknet-react/chains"; +import { + ConnectorNotConnectedError, + UserNotConnectedError, +} from "@starknet-react/core"; +import { BraavosMobileConnector, getBraavosMobile } from "@utils/braavosMobile"; + +describe("BraavosMobileConnector class", () => { + let connector; + + beforeEach(() => { + connector = new BraavosMobileConnector(); + }); + + describe("id getter", () => { + it("should return 'braavosMobile'", () => { + expect(connector.id).toBe("braavosMobile"); + }); + }); + + describe("icon getter", () => { + it("should return the same icon for light and dark modes", () => { + expect(connector.icon.light).toBeDefined(); + expect(connector.icon.dark).toBeDefined(); + expect(connector.icon.light).toBe(connector.icon.dark); + }); + }); + + describe("name getter", () => { + it("should return 'Braavos (mobile)'", () => { + expect(connector.name).toBe("Braavos (mobile)"); + }); + }); + + describe("available method", () => { + it("should return true", () => { + expect(connector.available()).toBe(true); + }); + }); + + describe("wallet getter", () => { + it("should throw a ConnectorNotConnectedError", () => { + expect(() => connector.wallet).toThrow(ConnectorNotConnectedError); + }); + }); + + describe("disconnect method", () => { + it("should throw a UserNotConnectedError", () => { + expect(() => connector.disconnect()).toThrow(UserNotConnectedError); + }); + }); + + describe("account method", () => { + it("should throw a ConnectorNotConnectedError", () => { + expect(() => connector.account()).toThrow(ConnectorNotConnectedError); + }); + }); + + describe("chainId method", () => { + it("should return sepolia.id when NEXT_PUBLIC_IS_TESTNET is 'true'", async () => { + process.env.NEXT_PUBLIC_IS_TESTNET = "true"; + const result = await connector.chainId(); + expect(result).toBe(sepolia.id); + }); + + it("should return mainnet.id when NEXT_PUBLIC_IS_TESTNET is not 'true'", async () => { + process.env.NEXT_PUBLIC_IS_TESTNET = "false"; + const result = await connector.chainId(); + expect(result).toBe(mainnet.id); + }); + }); + + describe("ready method", () => { + it("should return true as a Promise", async () => { + const result = await connector.ready(); + expect(result).toBe(true); + }); + }); +}); + +describe("getBraavosMobile function", () => { + it("should return an instance of BraavosMobileConnector", () => { + const instance = getBraavosMobile(); + expect(instance).toBeInstanceOf(BraavosMobileConnector); + }); +}); diff --git a/tests/utils/domainService.test.js b/tests/utils/domainService.test.js index e17b5b35..ccdb6f4e 100644 --- a/tests/utils/domainService.test.js +++ b/tests/utils/domainService.test.js @@ -1,23 +1,26 @@ -import { getDomainFromAddress } from '@utils/domainService'; +import { getDomainFromAddress } from "@utils/domainService"; const API_URL = process.env.NEXT_PUBLIC_STARKNET_ID_API_LINK; describe("getDomainFromAddress function", () => { -it("should return a valid domain for a valid address", async () => { - const validAddress = "0x061b6c0a78f9edf13cea17b50719f3344533fadd470b8cb29c2b4318014f52d3"; + it("should return a valid domain for a valid address", async () => { + const validAddress = + "0x061b6c0a78f9edf13cea17b50719f3344533fadd470b8cb29c2b4318014f52d3"; const expectedDomain = "fricoben.stark"; // Mocking the fetch function to return a predefined response const mockResponse = { domain: expectedDomain }; const fetchMock = jest.fn().mockResolvedValue({ - json: jest.fn().mockResolvedValue(mockResponse), + json: jest.fn().mockResolvedValue(mockResponse), }); - (fetch as jest.Mock) = fetchMock; + fetch = fetchMock; const domain = await getDomainFromAddress(validAddress); expect(domain).toBe(expectedDomain); - expect(fetchMock).toHaveBeenCalledWith(`${API_URL}/addr_to_domain?addr=${validAddress}`); + expect(fetchMock).toHaveBeenCalledWith( + `${API_URL}/addr_to_domain?addr=${validAddress}` + ); }); it("should return an empty string for an invalid address", async () => { @@ -27,13 +30,15 @@ it("should return a valid domain for a valid address", async () => { // Mocking the fetch function to simulate an error const mockResponse = { domain: expectedDomain }; const fetchMock = jest.fn().mockResolvedValue({ - json: jest.fn().mockResolvedValue(mockResponse), - }); - (fetch as jest.Mock) = fetchMock; + json: jest.fn().mockResolvedValue(mockResponse), + }); + fetch = fetchMock; const domain = await getDomainFromAddress(invalidAddress); expect(domain).toBe(""); - expect(fetchMock).toHaveBeenCalledWith(`${API_URL}/addr_to_domain?addr=${invalidAddress}`); + expect(fetchMock).toHaveBeenCalledWith( + `${API_URL}/addr_to_domain?addr=${invalidAddress}` + ); }); -}); \ No newline at end of file +}); diff --git a/tests/utils/walletConfig.test.js b/tests/utils/walletConfig.test.js index befc29ca..871109c9 100644 --- a/tests/utils/walletConfig.test.js +++ b/tests/utils/walletConfig.test.js @@ -3,7 +3,7 @@ import { getConnectorIcon, getConnectorName, getConnectorDiscovery, -} from "../../utils/walletConfig"; +} from "@utils/walletConfig"; class MockConnector { constructor(id, name, isAvailable) { this.id = id; @@ -65,10 +65,10 @@ describe("sortConnectors function", () => { describe("getConnectorIcon function", () => { it("should return the icon for a wallet when the ID matches", () => { expect(getConnectorIcon("argentX")).toBe( - "", + "" ); expect(getConnectorIcon("braavos")).toBe( - "", + "" ); }); @@ -123,26 +123,26 @@ describe("getConnectorDiscovery function", () => { it("should return the wallet's website URL when no downloads are available", () => { expect(getConnectorDiscovery("argentWebWallet")).toBe( - "https://www.argent.xyz/argent-x/", + "https://www.argent.xyz/argent-x/" ); }); it("should return the correct download link for the user's browser if available", () => { expect(getConnectorDiscovery("braavos")).toBe( - "https://chrome.google.com/webstore/detail/braavos-wallet/jnlgamecbpmbajjfhmmmlhejkemejdma", + "https://chrome.google.com/webstore/detail/braavos-wallet/jnlgamecbpmbajjfhmmmlhejkemejdma" ); }); it("should return the ecosystem website if no wallet data is found", () => { expect(getConnectorDiscovery("wallet")).toBe( - "https://www.starknet-ecosystem.com", + "https://www.starknet-ecosystem.com" ); }); it("should return the wallet's website when navigator is undefined", () => { Object.defineProperty(global, "navigator", { value: undefined }); expect(getConnectorDiscovery("argentX")).toBe( - "https://www.argent.xyz/argent-x/", + "https://www.argent.xyz/argent-x/" ); }); }); diff --git a/utils/braavosMobile.ts b/utils/braavosMobile.ts new file mode 100644 index 00000000..f9031c6d --- /dev/null +++ b/utils/braavosMobile.ts @@ -0,0 +1,64 @@ +import { mainnet, sepolia } from "@starknet-react/chains"; +import { + Connector, + ConnectorNotConnectedError, + UserNotConnectedError, +} from "@starknet-react/core"; +import { ConnectorData } from "starknetkit"; + +export const braavosMobileIcon = + ""; + +// @ts-ignore +export class BraavosMobileConnector extends Connector { + get id(): string { + return "braavosMobile"; + } + + get icon(): { dark: string; light: string } { + return { + dark: braavosMobileIcon, + light: braavosMobileIcon, + }; + } + + get name(): string { + return "Braavos (mobile)"; + } + + available(): boolean { + return true; + } + + connect(): Promise { + const link = `https://link.braavos.app/dapp/${window.location.host}`; + window.open(link, "_blank"); + return null as unknown as Promise; + } + + get wallet(): never { + throw new ConnectorNotConnectedError(); + } + + disconnect(): never { + throw new UserNotConnectedError(); + } + + account(): never { + throw new ConnectorNotConnectedError(); + } + + chainId(): Promise { + return process.env.NEXT_PUBLIC_IS_TESTNET === "true" + ? Promise.resolve(sepolia.id) + : Promise.resolve(mainnet.id); + } + + ready(): Promise { + return Promise.resolve(true); + } +} + +export function getBraavosMobile(): BraavosMobileConnector { + return new BraavosMobileConnector(); +} diff --git a/utils/walletConfig.ts b/utils/walletConfig.ts index 857dd4a0..220cb7d2 100644 --- a/utils/walletConfig.ts +++ b/utils/walletConfig.ts @@ -1,10 +1,11 @@ import { constants } from "starknet"; import { InjectedConnector } from "starknetkit/injected"; import { WebWalletConnector } from "starknetkit/webwallet"; -import { Connector, StarknetWindowObject } from "starknetkit"; +import { Connector } from "starknetkit"; import { ArgentMobileConnector } from "starknetkit/argentMobile"; import { getCurrentNetwork } from "./network"; import { getBrowser } from "./browserService"; +import { braavosMobileIcon, getBraavosMobile } from "./braavosMobile"; export type WalletStore = "chrome" | "firefox" | "edge" | "safari"; type WalletDownload = Partial>; @@ -89,6 +90,18 @@ export const wallets: Wallet[] = [ }, website: "https://www.keplr.app/", }, + { + id: "braavosMobile", + name: "Braavos Mobile", + icon: braavosMobileIcon, + downloads: { + chrome: `https://link.braavos.app/dapp/starknet.quest`, + firefox: `https://link.braavos.app/dapp/starknet.quest`, + edge: `https://link.braavos.app/dapp/starknet.quest`, + safari: `https://link.braavos.app/dapp/starknet.quest`, + }, + website: `https://link.braavos.app/dapp/starknet.quest`, + }, ]; // Check if the Bitget wallet is available on the window object @@ -109,7 +122,6 @@ export const getConnectors = () => { }, }), new InjectedConnector({ options: { id: "okxwallet", name: "Okx Wallet" } }), - new WebWalletConnector({ url: getCurrentNetwork() === "TESTNET" @@ -126,7 +138,11 @@ export const getConnectors = () => { }), new InjectedConnector({ options: { id: "keplr", name: "Keplr" } }), - ]; + + typeof window === "undefined" || window.innerWidth < 768 + ? getBraavosMobile() + : [], + ].flat(); return connectors; }; @@ -140,6 +156,7 @@ export const sortConnectors = (connectors: Connector[]): Connector[] => { const notAvailable: Connector[] = []; connectors.forEach((connector) => { + console.log(connector); connector.available() ? available.push(connector) : notAvailable.push(connector);