Skip to content

Commit

Permalink
Show voted candidates and allow to change vote
Browse files Browse the repository at this point in the history
  • Loading branch information
damianmarti committed Sep 24, 2024
1 parent 1434aae commit d9113a4
Show file tree
Hide file tree
Showing 2 changed files with 75 additions and 27 deletions.
84 changes: 60 additions & 24 deletions packages/nextjs/components/PollDetail.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -26,6 +26,8 @@ export default function PollDetail({ id }: { id: bigint }) {
const isAnyInvalid = Object.values(isVotesInvalid).some(v => v);
const [result, setResult] = useState<{ candidate: string; votes: number }[] | null>(null);
const [status, setStatus] = useState<PollStatus>();
const [voted, setVoted] = useState<boolean>(false);
const [voting, setVoting] = useState<boolean>(false);

useEffect(() => {
if (!poll || !poll.metadata) {
Expand Down Expand Up @@ -130,6 +132,8 @@ export default function PollDetail({ id }: { id: bigint }) {
return;
}

setVoting(true);

const votesToMessage = votes.map((v, i) =>
getMessageAndEncKeyPair(
stateIndex,
Expand Down Expand Up @@ -167,9 +171,12 @@ export default function PollDetail({ id }: { id: bigint }) {
}

notification.success("Vote casted successfully");
setVoted(true);
} catch (err) {
console.log("err", err);
notification.error("Casting vote failed, please try again ");
} finally {
setVoting(false);
}
};

Expand Down Expand Up @@ -224,32 +231,61 @@ export default function PollDetail({ id }: { id: bigint }) {
<div className="container mx-auto pt-10">
<div className="flex h-full flex-col md:w-2/3 lg:w-1/2 mx-auto">
<div className="flex flex-row items-center my-5">
<div className="text-2xl font-bold ">Vote for {poll?.name}</div>
</div>
{poll?.options.map((candidate, index) => (
<div className="pb-5 flex" key={index}>
<VoteCard
pollOpen={status === PollStatus.OPEN}
index={index}
candidate={candidate}
clicked={false}
pollType={pollType}
onChange={(checked, votes) => voteUpdated(index, checked, votes)}
isInvalid={Boolean(isVotesInvalid[index])}
setIsInvalid={status => setIsVotesInvalid({ ...isVotesInvalid, [index]: status })}
/>
<div className="text-2xl font-bold ">
Vote for {poll?.name}
{status === PollStatus.CLOSED && " (Closed)"}
</div>
))}
{status === PollStatus.OPEN && (
<div className={`mt-2 shadow-2xl`}>
<button
onClick={castVote}
disabled={!true}
className="hover:border-black border-2 border-accent w-full text-lg text-center bg-accent py-3 rounded-xl font-bold"
>
{true ? "Vote Now" : "Voting Closed"}{" "}
</button>
</div>
{voted ? (
<div>
<p className="font-bold">Voted:</p>
<ul>
{votes.map(vote => (
<li key={vote.index} className="bg-primary flex w-full px-2 py-2 rounded-lg mb-2">
{poll?.options[vote.index]}: {vote.votes} votes
</li>
))}
</ul>
{status === PollStatus.OPEN && (
<div className={`mt-2 shadow-2xl`}>
<button
onClick={() => setVoted(false)}
className="hover:border-black border-2 border-accent w-full text-lg text-center bg-accent py-3 rounded-xl font-bold mt-4"
>
Change Vote
</button>
</div>
)}
</div>
) : (
<>
{poll?.options.map((candidate, index) => (
<div className="pb-5 flex" key={index}>
<VoteCard
pollOpen={status === PollStatus.OPEN}
index={index}
candidate={candidate}
clicked={false}
currentVotes={votes.find(v => v.index === index)?.votes}
pollType={pollType}
onChange={(checked, votes) => voteUpdated(index, checked, votes)}
isInvalid={Boolean(isVotesInvalid[index])}
setIsInvalid={status => setIsVotesInvalid({ ...isVotesInvalid, [index]: status })}
/>
</div>
))}
{status === PollStatus.OPEN && (
<div className={`mt-2 shadow-2xl`}>
<button
onClick={castVote}
disabled={voting}
className="hover:border-black border-2 border-accent w-full text-lg text-center bg-accent py-3 rounded-xl font-bold disabled:cursor-not-allowed disabled:border-none"
>
Vote Now
</button>
</div>
)}
</>
)}

{result && (
Expand Down
18 changes: 15 additions & 3 deletions packages/nextjs/components/card/VoteCard.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -10,11 +10,21 @@ type VoteCardProps = {
setIsInvalid: (value: boolean) => void;
isInvalid: boolean;
pollOpen: boolean;
currentVotes?: number;
};

const VoteCard = ({ index, candidate, onChange, pollType, isInvalid, setIsInvalid, pollOpen }: VoteCardProps) => {
const [selected, setSelected] = useState(false);
const [votes, setVotes] = useState(0);
const VoteCard = ({
index,
candidate,
onChange,
pollType,
isInvalid,
setIsInvalid,
pollOpen,
currentVotes,
}: VoteCardProps) => {
const [selected, setSelected] = useState(currentVotes && currentVotes > 0 ? true : false);
const [votes, setVotes] = useState(currentVotes || 0);
const votesFieldRef = useRef<HTMLInputElement>(null);

return (
Expand All @@ -25,6 +35,7 @@ const VoteCard = ({ index, candidate, onChange, pollType, isInvalid, setIsInvali
type={pollType === PollType.SINGLE_VOTE ? "radio" : "checkbox"}
className="mr-2"
value={index}
checked={selected}
onChange={e => {
console.log(e.target.checked);
setSelected(e.target.checked);
Expand Down Expand Up @@ -72,6 +83,7 @@ const VoteCard = ({ index, candidate, onChange, pollType, isInvalid, setIsInvali
placeholder="Votes"
min={0}
step={1}
defaultValue={currentVotes || ""}
onChange={function (e) {
if (
Number(e.currentTarget.value) < 0 ||
Expand Down

0 comments on commit d9113a4

Please sign in to comment.