From 81398ff02108f482f2406a1e9f5d892e1ad80bec Mon Sep 17 00:00:00 2001 From: Anton Mushnin Date: Mon, 6 May 2024 11:45:39 +0300 Subject: [PATCH 01/24] new icon --- web/src/components/icons/QuestionMarkIcon.tsx | 29 +++++++++++++++++++ 1 file changed, 29 insertions(+) create mode 100644 web/src/components/icons/QuestionMarkIcon.tsx diff --git a/web/src/components/icons/QuestionMarkIcon.tsx b/web/src/components/icons/QuestionMarkIcon.tsx new file mode 100644 index 00000000..db955472 --- /dev/null +++ b/web/src/components/icons/QuestionMarkIcon.tsx @@ -0,0 +1,29 @@ +import React from "react"; + +const QuestionMarkIcon: React.FC> = (props) => ( + + + + +); + +export default QuestionMarkIcon; From 9c1a1b37613bf8b9fcd401d07e042ecab2d33c13 Mon Sep 17 00:00:00 2001 From: Anton Mushnin Date: Mon, 6 May 2024 12:10:20 +0300 Subject: [PATCH 02/24] hide scores and footer for progress === 2 --- web/src/components/atbat/AtBatView.tsx | 14 +++++++++----- 1 file changed, 9 insertions(+), 5 deletions(-) diff --git a/web/src/components/atbat/AtBatView.tsx b/web/src/components/atbat/AtBatView.tsx index d24e8bab..d054bfb3 100644 --- a/web/src/components/atbat/AtBatView.tsx +++ b/web/src/components/atbat/AtBatView.tsx @@ -238,7 +238,7 @@ const AtBatView: React.FC = () => { )} - {atBatState.data && !isBigView && ( + {atBatState.data && !isBigView && atBatState.data.atBat.pitches[0].progress !== 2 && ( { {atBatState.data && isBigView && atBatState.data?.atBat.outcome === 0 && - !showPitchOutcome && ( + !showPitchOutcome && + atBatState.data.atBat.pitches[0].progress !== 2 && ( { )} )} - {atBatState.data && !showPitchOutcome && !isBigView && ( - - )} + {atBatState.data && + !showPitchOutcome && + !isBigView && + atBatState.data.atBat.pitches[0].progress !== 2 && ( + + )} ); }; From c28833821ae5834dd84ead6fc1c354af99d81c6e Mon Sep 17 00:00:00 2001 From: Anton Mushnin Date: Mon, 6 May 2024 12:12:46 +0300 Subject: [PATCH 03/24] invite prompt --- web/src/components/atbat/AtBatView.module.css | 16 ++++++++++++++++ web/src/components/atbat/AtBatView.tsx | 7 +++++++ 2 files changed, 23 insertions(+) diff --git a/web/src/components/atbat/AtBatView.module.css b/web/src/components/atbat/AtBatView.module.css index 5adae758..8fa08ba5 100644 --- a/web/src/components/atbat/AtBatView.module.css +++ b/web/src/components/atbat/AtBatView.module.css @@ -80,6 +80,22 @@ height: 98px; } +.invitePrompt { + color: #262019; + position: absolute; + right: 50%; + transform: translateX(50%); + z-index: 3; + text-align: center; + font-family: Bangers, cursive; + font-style: normal; + font-weight: 400; + line-height: normal; + top: 88.5px; + font-size: 24px; + width: 320px; +} + .negativeOutcome2 { stroke: #CD7676; color: #CD7676; diff --git a/web/src/components/atbat/AtBatView.tsx b/web/src/components/atbat/AtBatView.tsx index d054bfb3..30204161 100644 --- a/web/src/components/atbat/AtBatView.tsx +++ b/web/src/components/atbat/AtBatView.tsx @@ -295,6 +295,13 @@ const AtBatView: React.FC = () => { : "you lose!"} )} + {atBatState.data && atBatState.data.atBat.pitches[0].progress === 2 && ( +
+ Waiting for Opponent. +
+ Invite Friend? +
+ )} {atBatState.data?.atBat.outcome === 0 && !showPitchOutcome && atBatState.data.atBat.pitches[atBatState.data.atBat.numberOfSessions - 1].progress !== 2 && From eb971245d66e17e1a54ba2aaa6ddef513a5c9eb6 Mon Sep 17 00:00:00 2001 From: Anton Mushnin Date: Mon, 6 May 2024 12:51:07 +0300 Subject: [PATCH 04/24] new icons --- web/src/components/icons/BallIconWhite.tsx | 42 ++++++++++++++++++++++ web/src/components/icons/BatIconWhite.tsx | 42 ++++++++++++++++++++++ 2 files changed, 84 insertions(+) create mode 100644 web/src/components/icons/BallIconWhite.tsx create mode 100644 web/src/components/icons/BatIconWhite.tsx diff --git a/web/src/components/icons/BallIconWhite.tsx b/web/src/components/icons/BallIconWhite.tsx new file mode 100644 index 00000000..138faadc --- /dev/null +++ b/web/src/components/icons/BallIconWhite.tsx @@ -0,0 +1,42 @@ +import React from "react"; + +const BallIconWhite: React.FC> = (props) => ( + + + + + + + + +); + +export default BallIconWhite; diff --git a/web/src/components/icons/BatIconWhite.tsx b/web/src/components/icons/BatIconWhite.tsx new file mode 100644 index 00000000..dae213b1 --- /dev/null +++ b/web/src/components/icons/BatIconWhite.tsx @@ -0,0 +1,42 @@ +import React from "react"; + +const BatIconWhite: React.FC> = (props) => ( + + + + + + + + +); + +export default BatIconWhite; From 720b577bc90d1b627a8f051befd4a3cfb32b30af Mon Sep 17 00:00:00 2001 From: Anton Mushnin Date: Mon, 6 May 2024 12:51:18 +0300 Subject: [PATCH 05/24] link icon --- web/src/components/icons/LinkIcon.tsx | 36 +++++++++++++++++++++++++++ 1 file changed, 36 insertions(+) create mode 100644 web/src/components/icons/LinkIcon.tsx diff --git a/web/src/components/icons/LinkIcon.tsx b/web/src/components/icons/LinkIcon.tsx new file mode 100644 index 00000000..38b18cee --- /dev/null +++ b/web/src/components/icons/LinkIcon.tsx @@ -0,0 +1,36 @@ +import React from "react"; + +const LinkIcon: React.FC> = (props) => ( + + + + + + + + + + + +); + +export default LinkIcon; From 64a0698485ec7dc729edb6232f701f10fd9ce878 Mon Sep 17 00:00:00 2001 From: Anton Mushnin Date: Mon, 6 May 2024 12:52:04 +0300 Subject: [PATCH 06/24] tokens in invite view --- .../atbat/InviteLinkView.module.css | 97 +++++++++++++++++++ web/src/components/atbat/InviteLinkView.tsx | 40 ++++++++ 2 files changed, 137 insertions(+) create mode 100644 web/src/components/atbat/InviteLinkView.module.css create mode 100644 web/src/components/atbat/InviteLinkView.tsx diff --git a/web/src/components/atbat/InviteLinkView.module.css b/web/src/components/atbat/InviteLinkView.module.css new file mode 100644 index 00000000..fb26ef9f --- /dev/null +++ b/web/src/components/atbat/InviteLinkView.module.css @@ -0,0 +1,97 @@ +.container { + display: flex; + flex-direction: column; + align-items: center; + gap: 15px; + height: 100%; + width: 100vw; + justify-content: center; +} +.tokens { + display: flex; + width: 100%; + padding: 10px; + justify-content: center; + align-items: center; + gap: 15px; + + border-top: 1px solid #262019; + border-bottom: 1px solid #262019; + background: #FCECD9; +} + +.player, .opponent { + width: 110px; + height: 110px; + position: relative; +} + +.player { + border: 4px solid #328449; + position: relative; + +} + +.opponent { + display: flex; + border: 1px solid #262019; + background: linear-gradient(0deg, #EBE4DC 0%, #EBE4DC 100%), #F9F0E3; + align-items: center; + justify-content: center; +} + +.vs { + color: #262019; + text-align: center; + font-family: Bangers, cursive; + font-size: 24px; + font-style: normal; + font-weight: 400; + line-height: 120%; /* 28.8px */ +} + +.linkContainer { + display: flex; + width: 270px; + align-items: center; + gap: -1px; +} + +.link { + display: flex; + padding: 10px; + align-items: center; + gap: 10px; + flex: 1 0 0; + background: #262019; +} + +.copyButton { + display: flex; + padding: 10px; + align-items: center; + gap: 10px; + border: 1px solid #262019; + background: #328449; + color: #FFF; + font-family: Pangolin, cursive; + font-size: 14px; + font-style: normal; + font-weight: 400; + line-height: 100%; /* 14px */ + letter-spacing: 0.7px; +} + +.ball { + position: absolute; + left: 0; + top: 50%; + transform: translateY(-50%) translateX(-50%); +} + +.bat { + position: absolute; + right: 0; + top: 50%; + transform: translateY(-50%) translateX(50%); +} \ No newline at end of file diff --git a/web/src/components/atbat/InviteLinkView.tsx b/web/src/components/atbat/InviteLinkView.tsx new file mode 100644 index 00000000..9fcb754f --- /dev/null +++ b/web/src/components/atbat/InviteLinkView.tsx @@ -0,0 +1,40 @@ +import styles from "./InviteLinkView.module.css"; +import { AtBatStatus } from "../../types"; +import { Image } from "@chakra-ui/react"; +import QuestionMarkIcon from "../icons/QuestionMarkIcon"; +import BallIconWhite from "../icons/BallIconWhite"; +import BatIconWhite from "../icons/BatIconWhite"; + +const InviteLinkView = ({ atBat }: { atBat: AtBatStatus }) => { + return ( +
+
+ {atBat.pitcher ? ( +
+ {atBat.pitcher.name} + +
+ ) : ( +
+ + +
+ )} +
VS
+ {atBat.batter ? ( +
+ {atBat.batter.name} + +
+ ) : ( +
+ + +
+ )} +
+
+ ); +}; + +export default InviteLinkView; From 2d6a8681d90c91f38bbf0bf011d8bc946361e2b7 Mon Sep 17 00:00:00 2001 From: Anton Mushnin Date: Mon, 6 May 2024 13:27:29 +0300 Subject: [PATCH 07/24] InviteLinkView --- .../atbat/InviteLinkView.module.css | 20 +++++++++++++--- web/src/components/atbat/InviteLinkView.tsx | 23 ++++++++++++++++++- 2 files changed, 39 insertions(+), 4 deletions(-) diff --git a/web/src/components/atbat/InviteLinkView.module.css b/web/src/components/atbat/InviteLinkView.module.css index fb26ef9f..b4146fe2 100644 --- a/web/src/components/atbat/InviteLinkView.module.css +++ b/web/src/components/atbat/InviteLinkView.module.css @@ -52,7 +52,7 @@ .linkContainer { display: flex; - width: 270px; + max-width: 270px; align-items: center; gap: -1px; } @@ -64,11 +64,20 @@ gap: 10px; flex: 1 0 0; background: #262019; + white-space: nowrap; + overflow: clip; + color: #FFF; + font-family: Pangolin, cursive; + font-size: 14px; + font-style: normal; + font-weight: 400; + line-height: 100%; /* 14px */ + letter-spacing: 0.7px; } -.copyButton { +.copyButton, .copyButtonSuccess { display: flex; - padding: 10px; + padding: 9px; align-items: center; gap: 10px; border: 1px solid #262019; @@ -82,6 +91,11 @@ letter-spacing: 0.7px; } +.copyButtonSuccess { + background: #01641d; + +} + .ball { position: absolute; left: 0; diff --git a/web/src/components/atbat/InviteLinkView.tsx b/web/src/components/atbat/InviteLinkView.tsx index 9fcb754f..ae041054 100644 --- a/web/src/components/atbat/InviteLinkView.tsx +++ b/web/src/components/atbat/InviteLinkView.tsx @@ -1,11 +1,19 @@ import styles from "./InviteLinkView.module.css"; import { AtBatStatus } from "../../types"; -import { Image } from "@chakra-ui/react"; +import { Image, useClipboard } from "@chakra-ui/react"; import QuestionMarkIcon from "../icons/QuestionMarkIcon"; import BallIconWhite from "../icons/BallIconWhite"; import BatIconWhite from "../icons/BatIconWhite"; +import LinkIcon from "../icons/LinkIcon"; const InviteLinkView = ({ atBat }: { atBat: AtBatStatus }) => { + const link = `${window.location.protocol}//${ + window.location.host + }/?invite_from=${encodeURIComponent( + atBat.pitcher ? atBat.pitcher.name : atBat.batter ? atBat.batter.name : "", + )}&id=${atBat.id}`; + const { onCopy, hasCopied } = useClipboard(link); + return (
@@ -33,6 +41,19 @@ const InviteLinkView = ({ atBat }: { atBat: AtBatStatus }) => {
)}
+
+
+ {link} +
+ +
); }; From 0f19713bf48ebc9ecc9b943f9dd79fc9e0d0ab4f Mon Sep 17 00:00:00 2001 From: Anton Mushnin Date: Mon, 6 May 2024 22:27:55 +0300 Subject: [PATCH 08/24] Choose Token View --- .../components/tokens/ChooseToken.module.css | 10 +- web/src/components/tokens/ChooseToken.tsx | 112 ++++++++++++++++-- 2 files changed, 112 insertions(+), 10 deletions(-) diff --git a/web/src/components/tokens/ChooseToken.module.css b/web/src/components/tokens/ChooseToken.module.css index 464ae622..746cad8e 100644 --- a/web/src/components/tokens/ChooseToken.module.css +++ b/web/src/components/tokens/ChooseToken.module.css @@ -1,13 +1,19 @@ .container { + position: absolute; + top: 50vh; + left: 50%; + border: 1px solid black; + transform: translateX(-50%) translateY(-50%); flex-direction: column; display: flex; - height: 100vh; - width: 100vw; + height: 75vh; + width: 320px; padding: 20px; justify-content: center; align-items: center; gap: 20px; background: #FCECD9; + z-index: 3; } .header { diff --git a/web/src/components/tokens/ChooseToken.tsx b/web/src/components/tokens/ChooseToken.tsx index 1d5c8a3f..19fa4645 100644 --- a/web/src/components/tokens/ChooseToken.tsx +++ b/web/src/components/tokens/ChooseToken.tsx @@ -1,23 +1,37 @@ import styles from "./ChooseToken.module.css"; import parentStyles from "./CreateNewCharacter.module.css"; -import { OwnedToken } from "../../types"; +import { AtBat, OwnedToken, Token } from "../../types"; import TokenCard from "./TokenCard"; -import { useGameContext } from "../../contexts/GameContext"; import { useEffect, useRef, useState } from "react"; import NewCharacterButton from "./NewCharacterButton"; +import { useMutation, useQueryClient } from "react-query"; +import { joinSessionFullcountPlayer } from "../../tokenInterfaces/FullcountPlayerAPI"; +import router from "next/router"; +import { sendReport } from "../../utils/humbug"; +import useMoonToast from "../../hooks/useMoonToast"; +import useUser from "../../contexts/UserContext"; +import { useSound } from "../../hooks/useSound"; +import { Spinner } from "@chakra-ui/react"; const ChooseToken = ({ tokens, - onChoose, onClose, + sessionID, + inviteCode, + inviteFrom, }: { tokens: OwnedToken[]; - onChoose: (token: OwnedToken) => void; onClose: () => void; + sessionID: number; + inviteCode: string; + inviteFrom: string; }) => { const [selectedTokenIdx, setSelectedTokenIdx] = useState(0); const elementRef = useRef(null); const [drawBottomLine, setDrawBottomLine] = useState(false); + const toast = useMoonToast(); + const { user } = useUser(); + const playSound = useSound(); useEffect(() => { const element = elementRef.current; @@ -26,6 +40,88 @@ const ChooseToken = ({ } }, [tokens]); + const handleClick = () => { + playSound("batButton"); + joinSession.mutate({ sessionID, token: tokens[selectedTokenIdx], inviteCode }); + }; + + const queryClient = useQueryClient(); + const joinSession = useMutation( + async ({ + sessionID, + token, + inviteCode, + }: { + sessionID: number; + token: OwnedToken; + inviteCode: string; + }): Promise => { + return joinSessionFullcountPlayer({ token, sessionID, inviteCode }); + }, + { + onSuccess: async (data, variables) => { + let atBatId: number | undefined = undefined; + queryClient.setQueryData( + ["atBats"], + (oldData: { atBats: AtBat[]; tokens: Token[] } | undefined) => { + console.log(oldData); + if (!oldData) { + return { atBats: [], tokens: [] }; + } + const newAtBats = oldData.atBats.map((atBat) => { + if (atBat.lastSessionId !== variables.sessionID) { + return atBat; + } + atBatId = atBat.id; + if (!atBat.pitcher) { + return { ...atBat, progress: 3, pitcher: { ...variables.token } }; + } + if (!atBat.batter) { + return { ...atBat, progress: 3, batter: { ...variables.token } }; + } + return atBat; + }); + + return { atBats: newAtBats, tokens: oldData.tokens }; + }, + ); + queryClient.setQueryData(["owned_tokens", user], (oldData: OwnedToken[] | undefined) => { + console.log(oldData); + if (!oldData) { + return []; + } + return oldData.map((t) => { + if (t.address === variables.token.address && t.id === variables.token.id) { + return { + ...t, + isStaked: true, + stakedSessionID: variables.sessionID, + tokenProgress: 3, + }; + } + return t; + }); + }); + queryClient.invalidateQueries("owned_tokens"); + if (atBatId) { + router.push(`atbats/?id=${atBatId}`); + } + }, + retryDelay: (attemptIndex) => (attemptIndex < 1 ? 5000 : 10000), + retry: (failureCount, error) => { + console.log(error); + if (failureCount < 3) { + console.log("Will retry in 5, maybe 10 seconds"); + } + return failureCount < 3; + }, + onError: (e: Error) => { + toast("Join failed" + e?.message, "error"); + sendReport("Error toast", { error: e }, ["type:error_toast"]); + }, + }, + ); + return (
Choose character
@@ -33,7 +129,7 @@ const ChooseToken = ({ className={styles.content} style={{ borderBottom: drawBottomLine ? "1px solid #7e8e7f" : "none" }} > -
Play
+ {/*
Play
*/}
Choose a character to play with.
{tokens.map((t, idx) => ( @@ -51,9 +147,9 @@ const ChooseToken = ({
Cancel
-
onChoose(tokens[selectedTokenIdx])}> - Play -
+
); From 5ca6061282539e39fae29d4b96d12668acccf390 Mon Sep 17 00:00:00 2001 From: Anton Mushnin Date: Mon, 6 May 2024 22:28:35 +0300 Subject: [PATCH 09/24] New character button update --- web/src/components/tokens/NewCharacterButton.tsx | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/web/src/components/tokens/NewCharacterButton.tsx b/web/src/components/tokens/NewCharacterButton.tsx index 31a13766..cf571ba4 100644 --- a/web/src/components/tokens/NewCharacterButton.tsx +++ b/web/src/components/tokens/NewCharacterButton.tsx @@ -15,7 +15,7 @@ const NewCharacterButton = ({ small }: { small: boolean }) => { updateContext({ isCreateCharacter: true }); }} > - {small ? : "+ Mint new Beer League Baller"} + {small ? : "+ Create new Beer League Baller"} ); }; From eb868e7530badc99f8f19c9c46550cc2567af42a Mon Sep 17 00:00:00 2001 From: Anton Mushnin Date: Mon, 6 May 2024 22:29:09 +0300 Subject: [PATCH 10/24] showing invite link in new atbat --- web/src/components/atbat/AtBatView.tsx | 14 +++++++++----- 1 file changed, 9 insertions(+), 5 deletions(-) diff --git a/web/src/components/atbat/AtBatView.tsx b/web/src/components/atbat/AtBatView.tsx index 30204161..32b782c2 100644 --- a/web/src/components/atbat/AtBatView.tsx +++ b/web/src/components/atbat/AtBatView.tsx @@ -21,6 +21,7 @@ import ExitDialog from "./ExitDialog"; import useUser from "../../contexts/UserContext"; import { fetchFullcountPlayerTokens } from "../../tokenInterfaces/FullcountPlayerAPI"; import { useSound } from "../../hooks/useSound"; +import InviteLinkView from "./InviteLinkView"; export const outcomes = [ "In Progress", @@ -296,11 +297,14 @@ const AtBatView: React.FC = () => { )} {atBatState.data && atBatState.data.atBat.pitches[0].progress === 2 && ( -
- Waiting for Opponent. -
- Invite Friend? -
+ <> +
+ Waiting for Opponent. +
+ Invite Friend? +
+ + )} {atBatState.data?.atBat.outcome === 0 && !showPitchOutcome && From 185dc0fe3fc38189a89129a5f521c042a234aced Mon Sep 17 00:00:00 2001 From: Anton Mushnin Date: Mon, 6 May 2024 22:32:03 +0300 Subject: [PATCH 11/24] handling invites --- web/src/components/Playing.tsx | 38 ++++++++++++++++++++++++---------- 1 file changed, 27 insertions(+), 11 deletions(-) diff --git a/web/src/components/Playing.tsx b/web/src/components/Playing.tsx index 1d956596..3f4bb9c3 100644 --- a/web/src/components/Playing.tsx +++ b/web/src/components/Playing.tsx @@ -22,6 +22,7 @@ import { getMulticallResults } from "../utils/multicall"; import { AbiItem } from "web3-utils"; import FullcountABIImported from "../web3/abi/FullcountABI.json"; import { useSound } from "../hooks/useSound"; +import router from "next/router"; const FullcountABI = FullcountABIImported as unknown as AbiItem[]; const Playing = () => { @@ -38,6 +39,21 @@ const Playing = () => { } = useGameContext(); const { user } = useUser(); const playSound = useSound(); + const [inviteFrom, setInviteFrom] = useState(""); + const [inviteSession, setInviteSession] = useState(""); + const [inviteCode, setInviteCode] = useState(""); + + useEffect(() => { + if (router.query.invite_from && typeof router.query.invite_from === "string") { + setInviteFrom(router.query.invite_from); + } + if (router.query.id && typeof router.query.id === "string") { + setInviteSession(router.query.id); + } + if (router.query.invite_code && typeof router.query.invite_code === "string") { + setInviteCode(router.query.invite_code); + } + }, [router.query.invite_from, router.query.id, router.query.invite_code]); const ownedTokens = useQuery( ["owned_tokens", user], @@ -192,26 +208,26 @@ const Playing = () => { ownedTokens.data && ownedTokens.data.length >= 1 && !invitedTo && + !inviteFrom && !isCreateCharacter && ( )} - - {invitedTo && ownedTokens.data && !isCreateCharacter && ( + {inviteFrom && inviteSession && ownedTokens.data && ( { - updateContext({ selectedToken: token, invitedTo: undefined }); + tokens={ownedTokens.data.filter((t) => !t.isStaked)} + sessionID={Number(inviteSession)} + inviteCode={inviteCode} + inviteFrom={inviteFrom} + onClose={() => { + router.push("/"); + setInviteCode(""); + setInviteFrom(""); + setInviteSession(""); }} - onClose={() => updateContext({ invitedTo: undefined })} /> )} - - {selectedSession && watchingToken && } - {selectedSession && !watchingToken && selectedToken && ( - - )} ); }; From 410b4ca9c2ff2ef9bcd04a70f371e0c3d3100d41 Mon Sep 17 00:00:00 2001 From: Anton Mushnin Date: Tue, 7 May 2024 13:49:35 +0300 Subject: [PATCH 12/24] making PVPIcon respect fill prop --- web/src/components/icons/PvPIcon.tsx | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/web/src/components/icons/PvPIcon.tsx b/web/src/components/icons/PvPIcon.tsx index c0e237ce..0ebba71e 100644 --- a/web/src/components/icons/PvPIcon.tsx +++ b/web/src/components/icons/PvPIcon.tsx @@ -6,19 +6,19 @@ const PvPIcon: React.FC> = (props) => ( width="50" height="51" viewBox="0 0 50 51" - fill="none" {...props} + fill="none" //fill here should be none > ); From bf9cf0fcfe7a0d981fb7b4591923f8b55833643d Mon Sep 17 00:00:00 2001 From: Anton Mushnin Date: Tue, 7 May 2024 13:49:55 +0300 Subject: [PATCH 13/24] invite to session instead of atbat --- web/src/components/atbat/InviteLinkView.tsx | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/web/src/components/atbat/InviteLinkView.tsx b/web/src/components/atbat/InviteLinkView.tsx index ae041054..0c4fd0bc 100644 --- a/web/src/components/atbat/InviteLinkView.tsx +++ b/web/src/components/atbat/InviteLinkView.tsx @@ -11,7 +11,7 @@ const InviteLinkView = ({ atBat }: { atBat: AtBatStatus }) => { window.location.host }/?invite_from=${encodeURIComponent( atBat.pitcher ? atBat.pitcher.name : atBat.batter ? atBat.batter.name : "", - )}&id=${atBat.id}`; + )}&id=${atBat.pitches[0].sessionID}`; const { onCopy, hasCopied } = useClipboard(link); return ( From 57db991f59724279b93927bceeddc471e337a681 Mon Sep 17 00:00:00 2001 From: Anton Mushnin Date: Tue, 7 May 2024 13:51:05 +0300 Subject: [PATCH 14/24] opened invite layout --- .../components/tokens/ChooseToken.module.css | 61 ++++++++++++++++--- 1 file changed, 51 insertions(+), 10 deletions(-) diff --git a/web/src/components/tokens/ChooseToken.module.css b/web/src/components/tokens/ChooseToken.module.css index 746cad8e..059f1e67 100644 --- a/web/src/components/tokens/ChooseToken.module.css +++ b/web/src/components/tokens/ChooseToken.module.css @@ -1,13 +1,25 @@ +.background { + display: flex; + flex-direction: column; + justify-content: center; + align-items: center; + min-height: 100vh; + width: 100%; + background-size: cover; + background-position: center; +} + + + .container { position: absolute; top: 50vh; left: 50%; - border: 1px solid black; transform: translateX(-50%) translateY(-50%); flex-direction: column; display: flex; - height: 75vh; - width: 320px; + height: 603px; + min-width: 322px; padding: 20px; justify-content: center; align-items: center; @@ -16,15 +28,35 @@ z-index: 3; } + + +.headerContainer { + display: flex; + flex-direction: column; + justify-content: center; + align-items: center; + gap: 10px; + align-self: stretch; +} + .header { - color: #262019; + color: #328449; text-align: center; font-family: Bangers, cursive; font-size: 30px; font-style: normal; font-weight: 400; - line-height: 100%; /* 30px */ - letter-spacing: 1.5px; + line-height: normal; +} + +.prompt { + color: #262019; + text-align: center; + font-family: Pangolin, cursive; + font-size: 14px; + font-style: normal; + font-weight: 400; + line-height: normal; } .content { @@ -33,7 +65,7 @@ align-items: center; gap: 15px; flex: 1 0 0; - width: 286px; + max-height: 310px; } .title { @@ -57,11 +89,20 @@ .cards { - width: 279px; display: flex; - justify-content: flex-start; + justify-content: center; gap: 15px; flex-wrap: wrap; overflow-y: auto; - max-height: calc(100vh - 221px); /*console.log(20 + 30 + 20 + 25.5 + 15 + 17.5 + 15 + 20 + 37.5 + 20);*/ + max-height: calc(100vh - 221px); + padding-bottom: 15px; +} + +@media (min-width: 362px) and (min-height: 643px) { + .background { + background-image: url('https://static.fullcount.xyz/web/landing/Bl-banner-background-tinted.jpeg'); + } + .container { + border: 1px solid black; + } } \ No newline at end of file From 427af6f5fd93a556cb73d0b496e87ee13f99d1c4 Mon Sep 17 00:00:00 2001 From: Anton Mushnin Date: Tue, 7 May 2024 13:52:07 +0300 Subject: [PATCH 15/24] choosing token form update --- web/src/components/tokens/ChooseToken.tsx | 80 ++++++++++++++--------- 1 file changed, 48 insertions(+), 32 deletions(-) diff --git a/web/src/components/tokens/ChooseToken.tsx b/web/src/components/tokens/ChooseToken.tsx index 19fa4645..d9fc9b96 100644 --- a/web/src/components/tokens/ChooseToken.tsx +++ b/web/src/components/tokens/ChooseToken.tsx @@ -2,7 +2,7 @@ import styles from "./ChooseToken.module.css"; import parentStyles from "./CreateNewCharacter.module.css"; import { AtBat, OwnedToken, Token } from "../../types"; import TokenCard from "./TokenCard"; -import { useEffect, useRef, useState } from "react"; +import React, { useEffect, useRef, useState } from "react"; import NewCharacterButton from "./NewCharacterButton"; import { useMutation, useQueryClient } from "react-query"; import { joinSessionFullcountPlayer } from "../../tokenInterfaces/FullcountPlayerAPI"; @@ -12,6 +12,8 @@ import useMoonToast from "../../hooks/useMoonToast"; import useUser from "../../contexts/UserContext"; import { useSound } from "../../hooks/useSound"; import { Spinner } from "@chakra-ui/react"; +import PvPIcon from "../icons/PvPIcon"; +import { useGameContext } from "../../contexts/GameContext"; const ChooseToken = ({ tokens, @@ -26,12 +28,14 @@ const ChooseToken = ({ inviteCode: string; inviteFrom: string; }) => { - const [selectedTokenIdx, setSelectedTokenIdx] = useState(0); + const { updateContext, selectedTokenIdx } = useGameContext(); + const elementRef = useRef(null); const [drawBottomLine, setDrawBottomLine] = useState(false); const toast = useMoonToast(); const { user } = useUser(); const playSound = useSound(); + const { isCreateCharacter } = useGameContext(); useEffect(() => { const element = elementRef.current; @@ -64,7 +68,6 @@ const ChooseToken = ({ queryClient.setQueryData( ["atBats"], (oldData: { atBats: AtBat[]; tokens: Token[] } | undefined) => { - console.log(oldData); if (!oldData) { return { atBats: [], tokens: [] }; } @@ -86,7 +89,6 @@ const ChooseToken = ({ }, ); queryClient.setQueryData(["owned_tokens", user], (oldData: OwnedToken[] | undefined) => { - console.log(oldData); if (!oldData) { return []; } @@ -123,35 +125,49 @@ const ChooseToken = ({ ); return ( -
-
Choose character
-
- {/*
Play
*/} -
Choose a character to play with.
-
- {tokens.map((t, idx) => ( - setSelectedTokenIdx(idx)} - /> - ))} - -
-
-
-
- Cancel + <> + {!isCreateCharacter && ( +
+
+
+ +
BATTER UP
+
+ +
+ {`${inviteFrom} is inviting you to play Fullcount.xyz.`}
+ {`Choose a character to play with.`} +
+
+
+ {tokens.map((t, idx) => ( + + updateContext({ selectedToken: { ...tokens[idx] }, selectedTokenIdx: idx }) + } + /> + ))} + +
+
+
+ + +
+
- -
-
+ )} + ); }; From 6e6a82a91505520eee9b994a2c98e162a042d635 Mon Sep 17 00:00:00 2001 From: Anton Mushnin Date: Tue, 7 May 2024 13:52:30 +0300 Subject: [PATCH 16/24] cursor for clickable element --- web/src/components/tokens/TokenCard.module.css | 1 + 1 file changed, 1 insertion(+) diff --git a/web/src/components/tokens/TokenCard.module.css b/web/src/components/tokens/TokenCard.module.css index 0812ae7f..94cba682 100644 --- a/web/src/components/tokens/TokenCard.module.css +++ b/web/src/components/tokens/TokenCard.module.css @@ -5,6 +5,7 @@ align-items: center; border: 1px solid #7E8E7F; background: #262019; + cursor: pointer; } From 5e77ff3b9d3535a856a6ff48c53eeca6026081ef Mon Sep 17 00:00:00 2001 From: Anton Mushnin Date: Wed, 8 May 2024 14:21:44 +0300 Subject: [PATCH 17/24] remove duplicate class --- web/src/components/tokens/ChooseToken.module.css | 11 ----------- 1 file changed, 11 deletions(-) diff --git a/web/src/components/tokens/ChooseToken.module.css b/web/src/components/tokens/ChooseToken.module.css index 059f1e67..5f8b4640 100644 --- a/web/src/components/tokens/ChooseToken.module.css +++ b/web/src/components/tokens/ChooseToken.module.css @@ -77,17 +77,6 @@ line-height: normal; } -.prompt { - color: #262019; - text-align: center; - font-family: Pangolin, cursive; - font-size: 14px; - font-style: normal; - font-weight: 400; - line-height: normal; -} - - .cards { display: flex; justify-content: center; From 4f5a36b9fd25cd1d4590d6e8dc5575de8614936a Mon Sep 17 00:00:00 2001 From: Anton Mushnin Date: Wed, 8 May 2024 14:23:01 +0300 Subject: [PATCH 18/24] show invite info on login and signup --- web/src/components/account/Account.module.css | 11 +++++++++++ web/src/components/account/LoginForm.tsx | 16 ++++++++++++++-- web/src/components/account/SignUpForm.tsx | 16 ++++++++++++++-- 3 files changed, 39 insertions(+), 4 deletions(-) diff --git a/web/src/components/account/Account.module.css b/web/src/components/account/Account.module.css index a775d05b..e8d0601b 100644 --- a/web/src/components/account/Account.module.css +++ b/web/src/components/account/Account.module.css @@ -58,3 +58,14 @@ opacity: 0; transition: opacity 1s ease-out; } + +.invitePrompt { + color: #262019; + text-align: center; + font-family: Pangolin, cursive; + font-size: 14px; + font-style: normal; + font-weight: 700; + line-height: normal; + margin-bottom: 20px; +} diff --git a/web/src/components/account/LoginForm.tsx b/web/src/components/account/LoginForm.tsx index f27f4dcc..f957ce0f 100644 --- a/web/src/components/account/LoginForm.tsx +++ b/web/src/components/account/LoginForm.tsx @@ -5,7 +5,13 @@ import { Spinner } from "@chakra-ui/react"; import styles from "./Account.module.css"; import useLogin from "../../hooks/useLogin"; -const LoginForm = ({ setIsSuccess }: { setIsSuccess: (isSuccess: boolean) => void }) => { +const LoginForm = ({ + setIsSuccess, + inviteFrom, +}: { + setIsSuccess: (isSuccess: boolean) => void; + inviteFrom?: string; +}) => { const [username, setUsername] = useState(""); const [password, setPassword] = useState(""); const { login, isLoading, isSuccess } = useLogin(); @@ -30,7 +36,13 @@ const LoginForm = ({ setIsSuccess }: { setIsSuccess: (isSuccess: boolean) => voi return (
{/*
*/} -
Welcome back!
+ {inviteFrom ? ( +
+ {`${inviteFrom} is inviting you to play Fullcount.xyz.`} +
+ ) : ( +
Welcome back!
+ )} void }) => { +const SignUpForm = ({ + setIsSuccess, + inviteFrom, +}: { + setIsSuccess: (isSuccess: boolean) => void; + inviteFrom?: string; +}) => { const [username, setUsername] = useState(""); const [password, setPassword] = useState(""); const [email, setEmail] = useState(""); @@ -32,7 +38,13 @@ const SignUpForm = ({ setIsSuccess }: { setIsSuccess: (isSuccess: boolean) => vo return ( {/*
*/} -
Welcome!
+ {inviteFrom ? ( +
+ {`${inviteFrom} is inviting you to play Fullcount.xyz.`} +
+ ) : ( +
Welcome!
+ )} Date: Wed, 8 May 2024 14:23:41 +0300 Subject: [PATCH 19/24] choosing token for invite if there is a token --- web/src/components/Playing.tsx | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/web/src/components/Playing.tsx b/web/src/components/Playing.tsx index 3f4bb9c3..5625809a 100644 --- a/web/src/components/Playing.tsx +++ b/web/src/components/Playing.tsx @@ -214,7 +214,7 @@ const Playing = () => { )} - {inviteFrom && inviteSession && ownedTokens.data && ( + {inviteFrom && inviteSession && ownedTokens.data && ownedTokens.data.length > 1 && ( !t.isStaked)} sessionID={Number(inviteSession)} From ac84f38033a43a1318e5146850a0ad77e64efac1 Mon Sep 17 00:00:00 2001 From: Anton Mushnin Date: Wed, 8 May 2024 14:57:58 +0300 Subject: [PATCH 20/24] updating token after creation --- web/src/components/Playing.tsx | 9 +++++++++ 1 file changed, 9 insertions(+) diff --git a/web/src/components/Playing.tsx b/web/src/components/Playing.tsx index 5625809a..03780709 100644 --- a/web/src/components/Playing.tsx +++ b/web/src/components/Playing.tsx @@ -63,6 +63,15 @@ const Playing = () => { if (ownedTokens.length > 0 && !selectedToken && ownedTokens[selectedTokenIdx]) { updateContext({ selectedToken: { ...ownedTokens[selectedTokenIdx] } }); } + //first response from the FCPLayer API after token creation has empty name and p0 image + if ( + selectedToken && + !selectedToken.name && + ownedTokens[selectedTokenIdx] && + selectedToken.id === ownedTokens[selectedTokenIdx].id + ) { + updateContext({ selectedToken: { ...ownedTokens[selectedTokenIdx] } }); + } return ownedTokens; }, { From a03bec7f0152c22dfd2e2995214542fb4f466ce5 Mon Sep 17 00:00:00 2001 From: Anton Mushnin Date: Wed, 8 May 2024 15:00:36 +0300 Subject: [PATCH 21/24] invite flow for not logged in user --- web/src/components/TitleScreen.tsx | 29 +++++++++++++++++++++++++---- 1 file changed, 25 insertions(+), 4 deletions(-) diff --git a/web/src/components/TitleScreen.tsx b/web/src/components/TitleScreen.tsx index dade513f..b34db863 100644 --- a/web/src/components/TitleScreen.tsx +++ b/web/src/components/TitleScreen.tsx @@ -7,19 +7,34 @@ import Playing from "./Playing"; import SignUpForm from "./account/SignUpForm"; import LoginForm from "./account/LoginForm"; -import { FULLCOUNT_ASSETS, FULLCOUNT_ASSETS_PATH } from "../constants"; +import { FULLCOUNT_ASSETS } from "../constants"; import LoadingView from "./HomePage/LoadingView"; import LaunchForm from "./LaunchForm"; import MoonstreamLogo2 from "./icons/MoonstreamLogo2"; import { useGameContext } from "../contexts/GameContext"; +import { useRouter } from "next/router"; const TitleScreen = () => { const { user, isLoading } = useUser(); + const router = useRouter(); const [isLogging, setIsLogging] = useState(false); // login or signUp const [isSuccess, setIsSuccess] = useState(false); const [isFirstSeconds, setIsFirstSeconds] = useState(true); const { isLaunching, updateContext } = useGameContext(); + const [inviteFrom, setInviteFrom] = useState(""); + + useEffect(() => { + if ( + router.isReady && + router.query.invite_from && + typeof router.query.invite_from === "string" + ) { + setInviteFrom(router.query.invite_from); + setIsLogging(true); + } + }, [router.isReady, router.query]); + useEffect(() => { setTimeout(() => setIsFirstSeconds(false), 2000); }, []); @@ -40,14 +55,20 @@ const TitleScreen = () => { src={`${FULLCOUNT_ASSETS}/banners/Bl-banner-updated-logo.jpeg`} alt={""} /> - {isLaunching ? ( + {isLaunching && !inviteFrom ? ( updateContext({ isLaunching: false })} /> ) : ( <> {!isLogging ? ( - setIsSuccess(value)} /> + setIsSuccess(value)} + /> ) : ( - setIsSuccess(value)} /> + setIsSuccess(value)} + /> )} {!isSuccess && (
From c49b958e67abd43b50e843ddbeec2f0ad07e3c30 Mon Sep 17 00:00:00 2001 From: Anton Mushnin Date: Wed, 8 May 2024 15:17:18 +0300 Subject: [PATCH 22/24] updating ownedTokens after mint --- .../components/tokens/CreateCharacterForm.tsx | 25 ++++++++++++++++--- 1 file changed, 21 insertions(+), 4 deletions(-) diff --git a/web/src/components/tokens/CreateCharacterForm.tsx b/web/src/components/tokens/CreateCharacterForm.tsx index d86a76b6..5f87aa8f 100644 --- a/web/src/components/tokens/CreateCharacterForm.tsx +++ b/web/src/components/tokens/CreateCharacterForm.tsx @@ -1,15 +1,16 @@ -import { Flex, Spinner } from "@chakra-ui/react"; +import { Spinner } from "@chakra-ui/react"; import Image from "next/image"; import styles from "./CreateNewCharacter.module.css"; import React, { useEffect, useState } from "react"; -import { TokenSource } from "../../types"; +import { OwnedToken, TokenSource } from "../../types"; import { useMutation, useQueryClient } from "react-query"; import { mintFullcountPlayerToken } from "../../tokenInterfaces/FullcountPlayerAPI"; import useMoonToast from "../../hooks/useMoonToast"; import { blbImage } from "../../constants"; import { sendReport } from "../../utils/humbug"; import { useSound } from "../../hooks/useSound"; +import useUser from "../../contexts/UserContext"; const NUMBER_OF_IMAGES = 8; const images: number[] = []; @@ -22,6 +23,7 @@ const CreateCharacterForm = ({ onClose }: { onClose?: () => void }) => { const [imageIndex, setImageIndex] = useState(0); const source: TokenSource = "FullcountPlayerAPI"; const queryClient = useQueryClient(); + const { user } = useUser(); const toast = useMoonToast(); const playSound = useSound(); @@ -35,11 +37,26 @@ const CreateCharacterForm = ({ onClose }: { onClose?: () => void }) => { } }, { - onSuccess: () => { + onSuccess: (data, variables) => { if (onClose) { onClose(); } - queryClient.invalidateQueries("owned_tokens"); //TODO data update + queryClient.setQueryData(["owned_tokens", user], (oldData: OwnedToken[] | undefined) => { + if (!oldData) { + return []; + } + const newToken: OwnedToken = { + address: data.erc721_address, + id: data.token_id, + image: blbImage(variables.imageIndex), + isStaked: false, + name: variables.name, + source: "FullcountPlayerAPI", + stakedSessionID: 0, + tokenProgress: 0, + }; + return [...oldData, newToken]; + }); }, onError: (e: Error) => { console.log(e); From 11c151fe3f01166e00e044f9db53821445e3010f Mon Sep 17 00:00:00 2001 From: Anton Mushnin Date: Wed, 8 May 2024 16:38:14 +0300 Subject: [PATCH 23/24] checking if invite session is valid --- .../components/tokens/ChooseToken.module.css | 9 ++ web/src/components/tokens/ChooseToken.tsx | 107 ++++++++++++------ 2 files changed, 83 insertions(+), 33 deletions(-) diff --git a/web/src/components/tokens/ChooseToken.module.css b/web/src/components/tokens/ChooseToken.module.css index 5f8b4640..3aaf01dd 100644 --- a/web/src/components/tokens/ChooseToken.module.css +++ b/web/src/components/tokens/ChooseToken.module.css @@ -59,6 +59,15 @@ line-height: normal; } +.error { + color: #CD7676; + text-align: center; + font-size: 14px; + font-style: normal; + font-weight: 400; + line-height: normal; +} + .content { display: flex; flex-direction: column; diff --git a/web/src/components/tokens/ChooseToken.tsx b/web/src/components/tokens/ChooseToken.tsx index d9fc9b96..2d49ab05 100644 --- a/web/src/components/tokens/ChooseToken.tsx +++ b/web/src/components/tokens/ChooseToken.tsx @@ -4,9 +4,9 @@ import { AtBat, OwnedToken, Token } from "../../types"; import TokenCard from "./TokenCard"; import React, { useEffect, useRef, useState } from "react"; import NewCharacterButton from "./NewCharacterButton"; -import { useMutation, useQueryClient } from "react-query"; +import { useMutation, useQuery, useQueryClient } from "react-query"; import { joinSessionFullcountPlayer } from "../../tokenInterfaces/FullcountPlayerAPI"; -import router from "next/router"; +import { useRouter } from "next/router"; import { sendReport } from "../../utils/humbug"; import useMoonToast from "../../hooks/useMoonToast"; import useUser from "../../contexts/UserContext"; @@ -14,6 +14,18 @@ import { useSound } from "../../hooks/useSound"; import { Spinner } from "@chakra-ui/react"; import PvPIcon from "../icons/PvPIcon"; import { useGameContext } from "../../contexts/GameContext"; +import { getContracts } from "../../utils/getWeb3Contracts"; + +const getErrorMessage = (sessionProgress: number) => { + switch (sessionProgress) { + case 0: + return "At-bat not found"; + case 2: + return undefined; + default: + return "Invitation is no longer valid"; + } +}; const ChooseToken = ({ tokens, @@ -36,6 +48,7 @@ const ChooseToken = ({ const { user } = useUser(); const playSound = useSound(); const { isCreateCharacter } = useGameContext(); + const router = useRouter(); useEffect(() => { const element = elementRef.current; @@ -49,6 +62,16 @@ const ChooseToken = ({ joinSession.mutate({ sessionID, token: tokens[selectedTokenIdx], inviteCode }); }; + const handleExitClick = () => { + router.push("/"); + sendReport("Invite is invalid", { sessionID }, ["type:click", "click:close_invite"]); + }; + + const sessionProgress = useQuery(["session_status", sessionID], () => { + const { gameContract } = getContracts(); + return gameContract.methods.sessionProgress(sessionID).call(); + }); + const queryClient = useQueryClient(); const joinSession = useMutation( async ({ @@ -133,37 +156,55 @@ const ChooseToken = ({
BATTER UP
- -
- {`${inviteFrom} is inviting you to play Fullcount.xyz.`}
- {`Choose a character to play with.`} -
-
-
- {tokens.map((t, idx) => ( - - updateContext({ selectedToken: { ...tokens[idx] }, selectedTokenIdx: idx }) - } - /> - ))} - -
-
-
- - -
+ {sessionProgress.data && getErrorMessage(Number(sessionProgress.data)) ? ( + <> +
+ {`${inviteFrom} is inviting you to play Fullcount.xyz.`}
+
+
{getErrorMessage(Number(sessionProgress.data))}
+
+ +
+ + ) : ( + <> +
+ {`${inviteFrom} is inviting you to play Fullcount.xyz.`}
+ {`Choose a character to play with.`} +
+
+
+ {tokens.map((t, idx) => ( + + updateContext({ + selectedToken: { ...tokens[idx] }, + selectedTokenIdx: idx, + }) + } + /> + ))} + +
+
+
+ + +
+ + )}
)} From 2eceedf6dfa64a9021509456a3dcfdd220fb67ba Mon Sep 17 00:00:00 2001 From: Anton Mushnin Date: Wed, 8 May 2024 16:39:15 +0300 Subject: [PATCH 24/24] clear invite state after route change --- web/src/components/Playing.tsx | 11 +++++++++-- 1 file changed, 9 insertions(+), 2 deletions(-) diff --git a/web/src/components/Playing.tsx b/web/src/components/Playing.tsx index 03780709..df541c37 100644 --- a/web/src/components/Playing.tsx +++ b/web/src/components/Playing.tsx @@ -22,7 +22,7 @@ import { getMulticallResults } from "../utils/multicall"; import { AbiItem } from "web3-utils"; import FullcountABIImported from "../web3/abi/FullcountABI.json"; import { useSound } from "../hooks/useSound"; -import router from "next/router"; +import { useRouter } from "next/router"; const FullcountABI = FullcountABIImported as unknown as AbiItem[]; const Playing = () => { @@ -42,16 +42,23 @@ const Playing = () => { const [inviteFrom, setInviteFrom] = useState(""); const [inviteSession, setInviteSession] = useState(""); const [inviteCode, setInviteCode] = useState(""); + const router = useRouter(); useEffect(() => { if (router.query.invite_from && typeof router.query.invite_from === "string") { setInviteFrom(router.query.invite_from); + } else { + setInviteFrom(""); } if (router.query.id && typeof router.query.id === "string") { setInviteSession(router.query.id); + } else { + setInviteSession(""); } if (router.query.invite_code && typeof router.query.invite_code === "string") { setInviteCode(router.query.invite_code); + } else { + setInviteCode(""); } }, [router.query.invite_from, router.query.id, router.query.invite_code]); @@ -223,7 +230,7 @@ const Playing = () => { )} - {inviteFrom && inviteSession && ownedTokens.data && ownedTokens.data.length > 1 && ( + {inviteFrom && inviteSession && ownedTokens.data && ownedTokens.data.length > 0 && ( !t.isStaked)} sessionID={Number(inviteSession)}