diff --git a/apis/quiz.ts b/apis/quiz.ts new file mode 100644 index 00000000..a884950c --- /dev/null +++ b/apis/quiz.ts @@ -0,0 +1,19 @@ +export const getCompletedQuizzes = async (wallet: string) => { + try { + const result = await fetch( + `https://api.mngo.cloud/data/v4/user-data/all-completed-quizzes?wallet-pk=${wallet}`, + ) + const solved = await result.json() + return solved?.length + ? (solved as { + wallet_pk: string + mango_account: string + quiz_id: number + points: number + }[]) + : [] + } catch (e) { + console.log(e) + return [] + } +} diff --git a/components/modals/UserSetupModal.tsx b/components/modals/UserSetupModal.tsx index 3ee02b9e..c74c91d0 100644 --- a/components/modals/UserSetupModal.tsx +++ b/components/modals/UserSetupModal.tsx @@ -276,7 +276,7 @@ const UserSetupModal = ({ src="/logos/logo-mark.svg" alt="next" /> -
+
{ + const router = useRouter() + const queryClient = useQueryClient() + const { connected, publicKey, signMessage } = useWallet() + const { mangoAccountAddress } = useMangoAccount() + const { data: solved } = useCompletedQuizzes(publicKey?.toBase58()) + const [currentQuestion, setCurrentQuestion] = useState(0) + const [answerIndex, setAnswerIndex] = useState(null) + const [isCorrectAnswer, setIsCorrectAnswer] = useState(null) + const [result, setResult] = useState(DEFAULT_RESULT) + const [showIntro, setShowIntro] = useState(true) + const [showResult, setShowResult] = useState(false) + const { questions, intro } = quiz + const { question, choices, description, correctAnswer } = + questions[currentQuestion] + + const handleAnswer = (answer: string, index: number) => { + setAnswerIndex(index) + if (answer === correctAnswer) { + setIsCorrectAnswer(true) + } else { + setIsCorrectAnswer(false) + } + } + + const handleNext = () => { + setAnswerIndex(null) + setResult((prev) => + isCorrectAnswer + ? { + ...prev, + correctAnswers: prev.correctAnswers + 1, + } + : { + ...prev, + wrongAnswers: [...prev.wrongAnswers, questions[currentQuestion]], + }, + ) + + if (currentQuestion !== questions.length - 1) { + setCurrentQuestion((prev) => prev + 1) + } else { + setCurrentQuestion(0) + setShowResult(true) + } + } + + const handleTryAgain = () => { + setResult(DEFAULT_RESULT) + setShowResult(false) + } + + const getResultsHeadingText = (score: number) => { + if (!score) { + return 'Whoops 😲' + } else if (score < 50) { + return 'Try Again' + } else if (score < 100) { + return 'Almost There...' + } else return 'Congratulations 🎉' + } + + const completeQuiz = async () => { + const message = new TextEncoder().encode(mangoAccountAddress) + const signature = await signMessage!(message) + const rawResponse = await fetch( + 'https://api.mngo.cloud/data/v4/user-data/complete-quiz', + { + method: 'POST', + headers: { + Accept: 'application/json', + 'Content-Type': 'application/json', + }, + body: JSON.stringify({ + wallet_pk: publicKey?.toBase58(), + quiz_id: quiz.id, + mango_account: mangoAccountAddress, + signature: bs58.encode(signature), + }), + }, + ) + await rawResponse.json() + queryClient.invalidateQueries(['completed-quizzes', publicKey?.toBase58()]) + notify({ + type: 'success', + title: 'Well done!', + description: 'Well done! 50 Rewards Points are on the way', + }) + router.push('/learn', undefined, { shallow: true }) + } + + const canClaimPoints = connected && mangoAccountAddress + + return ( + <> +
+
+ router.push('/learn', undefined, { shallow: true })} + > + + +

{quiz.name} Quiz

+
+ {showIntro || showResult ? null : ( +
+ {currentQuestion + 1} + /{quiz.questions.length} +
+ )} +
+
+
+
+
+
+ {showIntro ? ( + <> + {quiz.imagePath ? ( + Quiz Image + ) : null} +

{intro.title}

+

{intro.description}

+ {intro?.docs ? ( + + {intro.docs.linkText} + + ) : null} + +
+

+ {!connected + ? 'Connect wallet to earn rewards points' + : solved?.find((x) => x.quiz_id === quiz.id) + ? 'Rewards Points Claimed' + : mangoAccountAddress + ? `Score ${quiz.questions.length}/${quiz.questions.length} to earn rewards points` + : 'Create a Mango Account to earn rewards points'} +

+
+ + ) : !showResult ? ( + <> +

{question}

+ {description ?

: null} +
+ {choices.map((choice, index) => ( + + ))} +
+
+ +
+ + ) : ( + <> +

+ {getResultsHeadingText( + (result.correctAnswers / questions.length) * 100, + )} +

+

You scored

+ + {((result.correctAnswers / questions.length) * 100).toFixed()}% + + {result.correctAnswers !== questions.length ? ( +
+

+ Try again to earn rewards points. +

+
+ ) : null} +
+
+
+ +

Correct Answers

+
+

+ {result.correctAnswers} +

+
+ {result.wrongAnswers?.length ? ( + + {({ open }) => ( + <> + +
+
+ +

Wrong Answers

+
+
+

+ {result.wrongAnswers.length} +

+ +
+
+
+ + {result.wrongAnswers.map((answer) => ( +
+ + {({ open }) => ( + <> +
+
+ +

+ {answer.question} +

+
+ + + {open + ? 'Hide Answer' + : 'Reveal Answer'} + + +
+ + {answer.explanation ? ( +
+
+ +

+ {answer.explanation} +

+
+
+ ) : null} + {answer.choices.map((choice, index) => ( +
+
+ + {String.fromCharCode( + 97 + index, + ).toUpperCase()} + +
+ {choice} +
+
+ ))} + + + )} + +
+ ))} +
+ + )} +
+ ) : ( +
+
+ +

Wrong Answers

+
+

0

+
+ )} +
+
+ {solved?.find((x) => x.quiz_id === quiz.id) || + !canClaimPoints ? ( + + ) : result.correctAnswers === questions.length ? ( + + ) : ( + <> + + + + )} +
+ + )} +
+
+ + ) +} + +export default Quiz diff --git a/hooks/useQuiz.ts b/hooks/useQuiz.ts new file mode 100644 index 00000000..8fcf5b2e --- /dev/null +++ b/hooks/useQuiz.ts @@ -0,0 +1,16 @@ +import { useQuery } from '@tanstack/react-query' +import { getCompletedQuizzes } from 'apis/quiz' + +export const useCompletedQuizzes = (wallet?: string) => { + return useQuery( + ['completed-quizzes', wallet], + () => getCompletedQuizzes(wallet!), + { + cacheTime: 1000 * 60 * 10, + staleTime: 1000 * 60, + retry: 3, + refetchOnWindowFocus: false, + enabled: !!wallet, + }, + ) +} diff --git a/pages/learn.tsx b/pages/learn.tsx new file mode 100644 index 00000000..0a3c74dd --- /dev/null +++ b/pages/learn.tsx @@ -0,0 +1,140 @@ +import { serverSideTranslations } from 'next-i18next/serverSideTranslations' +import { NextPage } from 'next' +import { Quiz as QuizType, quizzes } from 'utils/quiz' +import { useMemo } from 'react' +import { CheckCircleIcon, ChevronRightIcon } from '@heroicons/react/20/solid' +import { useRouter } from 'next/router' +import Image from 'next/image' +import { useCompletedQuizzes } from 'hooks/useQuiz' +import Quiz from '@components/quiz/Quiz' +import { useWallet } from '@solana/wallet-adapter-react' + +export async function getStaticProps({ locale }: { locale: string }) { + return { + props: { + ...(await serverSideTranslations(locale, [ + 'account', + 'close-account', + 'common', + 'notifications', + 'onboarding', + 'profile', + 'search', + 'settings', + ])), + // Will be passed to the page component as props + }, + } +} + +const shuffleQuiz = (quiz: QuizType) => { + for (let i = quiz.questions.length - 1; i > 0; i--) { + const j = Math.floor(Math.random() * (i + 1)) + ;[quiz.questions[i], quiz.questions[j]] = [ + quiz.questions[j], + quiz.questions[i], + ] + } + return quiz +} + +const Learn: NextPage = () => { + // const { t } = useTranslation('common') + const router = useRouter() + const { quiz } = router.query + + const quizToShow = useMemo(() => { + if (!quiz) return + const index = quizzes.findIndex((q) => q.slug === quiz) + const result = quizzes.find((q) => q.slug === quiz) + return { + quiz: result, + index, + } + }, [quiz]) + + return !quizToShow?.quiz ? ( +
+

Learn 2 Earn

+

+ Earn rewards points for becoming a quiz master. +

+
+ {[...quizzes].reverse().map((quiz, index) => ( + + ))} +
+
+ ) : ( +
+ +
+ ) +} + +const QuizCard = ({ quiz }: { quiz: QuizType }) => { + const { publicKey } = useWallet() + const { data: solved } = useCompletedQuizzes(publicKey?.toBase58()) + const router = useRouter() + const goToQuiz = (quiz: string) => { + const query = { ...router.query, ['quiz']: quiz } + router.push({ pathname: router.pathname, query }, undefined, { + shallow: true, + }) + } + + return ( + + ) +} + +export default Learn diff --git a/utils/quiz.ts b/utils/quiz.ts new file mode 100644 index 00000000..20b92360 --- /dev/null +++ b/utils/quiz.ts @@ -0,0 +1,428 @@ +export type Quiz = { + id: number + name: string + description: string + imagePath?: string + intro: { + title: string + description: string + docs?: { + url: string + linkText: string + } + } + questions: QuizQuestion[] + slug: string +} + +export type QuizQuestion = { + question: string + description?: string + explanation?: string // explanation of the correct answer + choices: string[] + correctAnswer: string +} + +const healthQuiz = [ + { + id: 0, + question: + 'Which of the following is NOT factored in the account health calculation', + choices: ['Collateral', 'Borrows', 'Perp Positions', 'Account Value'], + explanation: + "All variants of account health are calculated as a weighted sum of an accounts assets minus liabilities. Account value is the notional value of an account's equity", + correctAnswer: 'Account Value', + }, + { + id: 1, + question: 'How many types of account health are there?', + choices: ['1', '2', '3', '4'], + explanation: + 'There are three types of health: Initial Health is used to check if new positions can be openend, Maintenance Health controls when liquidation starts and Liquidation End Health determines when liquidation ends.', + correctAnswer: '3', + }, + { + id: 2, + question: 'Which of the following is NOT a type of account health?', + choices: [ + 'Initial Health', + 'Liquidation End Health', + 'Maintenance Health', + 'Collateral Health', + ], + explanation: + 'There are three types of health: Initial Health is used to check if new positions can be openend, Maintenance Health controls when liquidation starts and Liquidation End Health determines when liquidation ends.', + correctAnswer: 'Collateral Health', + }, + { + id: 3, + question: 'What is Initial Health used for?', + choices: [ + 'Liquidations', + 'Free Collateral', + 'Account Value', + 'Funding Rates', + ], + explanation: + "Initial Health essentially represents your free collateral. If this value reaches 0 you won't be able to enter new positions or withdraw collateral", + correctAnswer: 'Free Collateral', + }, + { + id: 4, + question: + 'True or false? Assets and liabilities are weighted equally across health types.', + choices: ['True', 'False'], + explanation: + 'Initial Health is weighted more conservatively than Maintenance Health and the weights are dynamicly determined by the risk engine.', + correctAnswer: 'False', + }, + { + id: 5, + question: 'What happens when your account health ratio reaches 0%?', + choices: [ + "New positions can't be opened", + 'Account will be liquidated', + 'Account withdrawals will be locked', + 'Account will get an airdrop', + ], + explanation: + 'Your account health ratio percentage is a representation of Maintenance Health. If it reaches 0% your account will be liquidated.', + correctAnswer: 'Account will be liquidated', + }, + { + id: 6, + question: 'What is the Stable Price?', + choices: [ + 'A special price just for stablecoins', + 'The average between the spot best bid and ask prices minus the oracle price', + "A safety mechanism that smooths the oracle price when it's changing rapidly", + 'The oracle price from 1 hour ago', + ], + explanation: + 'Stable price is a safety mechanism that limits your ability to enter risky positions when the oracle price is changing rapidly.', + correctAnswer: + "A safety mechanism that smooths the oracle price when it's changing rapidly", + }, + { + id: 7, + question: 'How are assets valued for Maintenance Health?', + choices: [ + 'Maint asset weight multiplied by stable price', + 'Maint asset weight multiplied by oracle price', + 'Init asset weight multiplied by oracle price', + 'Maint asset weight multiplied by the lesser of oracle price and stable price', + ], + explanation: + 'Maintenance Health determines if an account can be liquidated. The value of assets in this case is the maintenance asset weight multiplied by the oracle price.', + correctAnswer: 'Maint asset weight multiplied by oracle price', + }, + { + id: 8, + question: 'How are liabilities valued for Initial Health?', + choices: [ + 'Maint liability weight multiplied by stable price', + 'Maint liability weight multiplied by oracle price', + 'Init liability weight multiplied by oracle price', + 'Init liability weight multiplied by the greater of oracle price and stable price', + ], + explanation: + 'Initial Health determines if new positions can be opened. The value of liabilities in this case is the init liability weight multiplied by the greater of oracle price and stable price. Including stable price prevents new overly risky positions.', + correctAnswer: + 'Init liability weight multiplied by the greater of oracle price and stable price', + }, +] + +// const nosanaQuiz = [ +// { +// id: 0, +// question: 'What is the primary purpose of the Nosana Network?', +// choices: [ +// 'To offer a decentralized cryptocurrency exchange', +// 'To provide a distributed GPU grid for rent', +// 'To serve as a gaming platform', +// 'To act as a digital wallet for cryptocurrencies', +// ], +// correctAnswer: 'To provide a distributed GPU grid for rent', +// }, +// { +// id: 1, +// question: +// 'What unique advantage does Nosana offer to AI users and GPU owners?', +// choices: [ +// 'Exclusive access to online gaming', +// 'High-speed internet services', +// 'Affordable GPUs for AI users and income for GPU owners', +// 'Free cryptocurrency mining software', +// ], +// correctAnswer: 'Affordable GPUs for AI users and income for GPU owners', +// }, +// { +// id: 2, +// question: "What is Nosana's ultimate vision for the future of computing?", +// choices: [ +// 'To centralize all computing power under one corporation', +// 'To decentralize essential computations using blockchain technology', +// 'To monopolize the GPU market', +// 'To replace all traditional computing hardware', +// ], +// correctAnswer: +// 'To decentralize essential computations using blockchain technology', +// }, +// { +// id: 3, +// question: 'Which blockchain powers the Nosana programs?', +// choices: ['Ethereum', 'Bitcoin', 'Solana', 'Cardano'], +// correctAnswer: 'Solana', +// }, +// { +// id: 4, +// question: 'What can users do with the Nosana Token (NOS)?', +// choices: [ +// 'Only trade it for other cryptocurrencies', +// 'Use it to run AI workloads on the network', +// 'Use it as a voting right in governance decisions', +// 'Convert it directly into fiat currency', +// ], +// correctAnswer: 'Use it as a voting right in governance decisions', +// }, +// { +// id: 5, +// question: 'What is the total supply of Nosana Tokens (NOS)?', +// choices: ['1,000,000', '10,000,000', '50,000,000', '100,000,000'], +// correctAnswer: '100,000,000', +// }, +// { +// id: 6, +// question: +// 'How are the mining tokens distributed to nodes within the Nosana Network?', +// choices: [ +// 'Equally to all participants', +// 'Based on the computational power provided', +// 'In a linear fashion over 24 months', +// 'As a lump sum at the start of their participation', +// ], +// correctAnswer: 'In a linear fashion over 24 months', +// }, +// { +// id: 7, +// question: +// 'What percentage of the total NOS tokens are allocated to team members?', +// choices: ['5%', '10%', '20%', '25%'], +// correctAnswer: '20%', +// }, +// { +// id: 8, +// question: +// 'Which of the following is NOT a use case for company tokens within the Nosana Network?', +// choices: [ +// 'Marketing', +// 'Engineering', +// 'Business Development', +// 'Personal expenses of team members', +// ], +// correctAnswer: 'Personal expenses of team members', +// }, +// { +// id: 9, +// question: 'Over what period are backers tokens released?', +// choices: ['9 months', '12 months', '24 months', '36 months'], +// correctAnswer: '9 months', +// }, +// ] + +const stepFinanceQuiz = [ + { + question: + 'What key functionality does Step Finance provide to its users on the Solana blockchain?', + choices: [ + 'Crypto mining software', + 'Portfolio tracking and management', + 'Virtual real estate services', + 'Social media platform for traders', + ], + correctAnswer: 'Portfolio tracking and management', + }, + { + question: 'What feature does Step Finance offer for NFT owners?', + choices: [ + 'Minting NFTs', + 'Viewing NFTs in a personal gallery', + 'Bidding on NFT auctions', + 'Creating NFTs', + ], + correctAnswer: 'Viewing NFTs in a personal gallery', + }, + { + question: + 'Which exchange is integrated with Step Finance for swapping tokens?', + choices: ['Uniswap', 'PancakeSwap', 'Jupiter Exchange', 'SushiSwap'], + correctAnswer: 'Jupiter Exchange', + }, + { + question: + 'What can STEP token holders do to earn a share of Step protocol revenue?', + choices: [ + 'Participate in governance voting', + 'Stake STEP for xSTEP', + 'Trade STEP tokens frequently', + 'Hold STEP tokens in a wallet', + ], + correctAnswer: 'Stake STEP for xSTEP', + }, + { + question: 'How does Step Finance help users find DeFi opportunities?', + choices: [ + 'By providing a list of top cryptocurrencies', + 'Through a dedicated Opportunities Page', + 'By offering a crypto news feed', + 'Through email alerts', + ], + correctAnswer: 'Through a dedicated Opportunities Page', + }, + { + question: + 'What unique tool does Step Finance provide for tracking transactions?', + choices: [ + 'Blockchain explorer', + 'Transaction history tool', + 'A private ledger', + 'An external API', + ], + correctAnswer: 'Transaction history tool', + }, + { + question: 'How can users support charities through Step Finance?', + choices: [ + 'By mining cryptocurrencies', + 'Through direct token swaps', + 'By donating USDC directly from the app', + 'By staking NFTs', + ], + correctAnswer: 'By donating USDC directly from the app', + }, + { + question: + 'What is the primary utility of the STEP token within the ecosystem?', + choices: [ + 'To pay transaction fees', + 'To be used throughout the ecosystem for various utilities', + 'To vote on future Solana validators', + 'Solely for speculative trading', + ], + correctAnswer: 'To be used throughout the ecosystem for various utilities', + }, + { + question: + 'Which service is integrated into Step Finance for private messaging?', + choices: ['Signal', 'Telegram', 'Dialect', 'WhatsApp'], + correctAnswer: 'Dialect', + }, + { + question: + 'What does Step Finance offer to help monitor and manage yield farms and staking rewards?', + choices: [ + 'DeFi News aggregator', + 'Real-time market data dashboard', + 'Protocol integrated management dashboard', + 'Automated trading bots', + ], + correctAnswer: 'Protocol integrated management dashboard', + }, + { + question: + 'What allows users to settle DEX balances and close token accounts on Step Finance?', + choices: [ + 'The Step Wallet', + 'A specific tool on the Step Dashboard', + 'A third-party service', + 'Manual transactions only', + ], + correctAnswer: 'A specific tool on the Step Dashboard', + }, + { + question: 'How does Step Finance enhance its token swapping feature?', + choices: [ + 'By offering the highest interest rates for swaps', + 'Through reward incentives', + 'By limiting swaps to high-volume tokens only', + 'Using AI to predict the best swap times', + ], + correctAnswer: 'Through reward incentives', + }, + { + question: + 'What feature does Step Finance provide for a more engaging DeFi experience?', + choices: [ + 'Virtual reality trading rooms', + 'An interactive game for earning tokens', + 'Detailed token data on SPL tokens', + 'Live streams of DeFi events', + ], + correctAnswer: 'Detailed token data on SPL tokens', + }, + { + question: 'How does Step Finance claim to benefit builders on Solana?', + choices: [ + 'By offering free advertising space', + 'Through funding and project support', + 'Providing a platform for token launches', + 'Free hosting services for DeFi projects', + ], + correctAnswer: 'Through funding and project support', + }, + { + question: + 'Which tool within Step Finance helps users explore and sort different yield farming APRs, lending pools data, and margin trading platforms?', + choices: [ + 'The DeFi Analysis Tool', + 'The APR Calculator', + 'The DeFi Strategy Simulator', + 'The Opportunities Page', + ], + correctAnswer: 'The Opportunities Page', + }, +] + +export const quizzes = [ + // { + // name: 'Know Nosana', + // id: 1, + // description: 'Think you know Nosana?', + // imagePath: '/icons/nos.svg', + // intro: { + // title: 'Know Nosana', + // description: + // 'Test your knowledge on the basics of the Nosana Network and the NOS token.', + // }, + // questions: nosanaQuiz, + // slug: 'nosana', + // }, + { + name: 'Account Health', + id: 2, + description: 'Learn how account health works on Mango.', + intro: { + title: 'Health is Wealth', + description: + 'Understanding account health is very important. Take a few minutes to check out the Docs before taking the quiz.', + docs: { + url: 'https://docs.mango.markets/mango-markets/health-overview', + linkText: 'Read the Health Docs', + }, + }, + questions: healthQuiz, + slug: 'health', + }, + { + name: 'Step Finance', + id: 3, + description: 'Step up your knowledge on Step Finance', + imagePath: '/icons/step.svg', + intro: { + title: 'Step Finance Quiz', + description: 'How well do you know Step Finance?', + }, + questions: stepFinanceQuiz, + slug: 'step-finance', + }, +]