From bf9d165916c56411a72d9229cf106db4852d619c Mon Sep 17 00:00:00 2001 From: Aahna Ashina <95955389+aahna-ashina@users.noreply.github.com> Date: Sun, 21 Apr 2024 21:42:25 +0800 Subject: [PATCH] =?UTF-8?q?=E2=9A=99=EF=B8=8F=20feat:=20ops=20skill=20leve?= =?UTF-8?q?l=20(#65)?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit close #65 --- abis/OperatorSkillLevels.json | 219 ++++++++++++++++++ .../ProfileDetailsOpsSkillRating.tsx | 55 +++++ src/pages/[passportId].tsx | 7 +- 3 files changed, 280 insertions(+), 1 deletion(-) create mode 100644 abis/OperatorSkillLevels.json create mode 100644 src/components/ProfileDetailsOpsSkillRating.tsx diff --git a/abis/OperatorSkillLevels.json b/abis/OperatorSkillLevels.json new file mode 100644 index 0000000..469be2d --- /dev/null +++ b/abis/OperatorSkillLevels.json @@ -0,0 +1,219 @@ +{ + "abi": [ + { + "inputs": [ + { + "internalType": "address", + "name": "passportUtils_", + "type": "address" + } + ], + "stateMutability": "nonpayable", + "type": "constructor" + }, + { + "inputs": [ + { + "internalType": "address", + "name": "illegalAlien", + "type": "address" + } + ], + "name": "NotPassportOwner", + "type": "error" + }, + { + "inputs": [ + { + "internalType": "address", + "name": "citizen", + "type": "address" + } + ], + "name": "PassportExpired", + "type": "error" + }, + { + "inputs": [ + { + "internalType": "uint8", + "name": "rating", + "type": "uint8" + } + ], + "name": "RatingValueError", + "type": "error" + }, + { + "anonymous": false, + "inputs": [ + { + "indexed": false, + "internalType": "address", + "name": "operator", + "type": "address" + }, + { + "indexed": false, + "internalType": "uint8", + "name": "rating", + "type": "uint8" + }, + { + "indexed": false, + "internalType": "address", + "name": "citizen", + "type": "address" + } + ], + "name": "Rated", + "type": "event" + }, + { + "inputs": [], + "name": "VERSION", + "outputs": [ + { + "internalType": "string", + "name": "", + "type": "string" + } + ], + "stateMutability": "view", + "type": "function" + }, + { + "inputs": [], + "name": "owner", + "outputs": [ + { + "internalType": "address", + "name": "", + "type": "address" + } + ], + "stateMutability": "view", + "type": "function" + }, + { + "inputs": [], + "name": "passportUtils", + "outputs": [ + { + "internalType": "contract IPassportUtils", + "name": "", + "type": "address" + } + ], + "stateMutability": "view", + "type": "function" + }, + { + "inputs": [ + { + "internalType": "address", + "name": "operator", + "type": "address" + }, + { + "internalType": "uint8", + "name": "rating", + "type": "uint8" + } + ], + "name": "rate", + "outputs": [], + "stateMutability": "nonpayable", + "type": "function" + }, + { + "inputs": [ + { + "internalType": "address", + "name": "owner_", + "type": "address" + } + ], + "name": "setOwner", + "outputs": [], + "stateMutability": "nonpayable", + "type": "function" + }, + { + "inputs": [ + { + "internalType": "address", + "name": "passportUtils_", + "type": "address" + } + ], + "name": "setPassportUtils", + "outputs": [], + "stateMutability": "nonpayable", + "type": "function" + }, + { + "inputs": [ + { + "internalType": "address", + "name": "", + "type": "address" + } + ], + "name": "skillLevelAverages", + "outputs": [ + { + "internalType": "uint256", + "name": "", + "type": "uint256" + } + ], + "stateMutability": "view", + "type": "function" + }, + { + "inputs": [ + { + "internalType": "address", + "name": "", + "type": "address" + }, + { + "internalType": "address", + "name": "", + "type": "address" + } + ], + "name": "skillLevelRatings", + "outputs": [ + { + "internalType": "uint8", + "name": "", + "type": "uint8" + } + ], + "stateMutability": "view", + "type": "function" + }, + { + "inputs": [ + { + "internalType": "address", + "name": "", + "type": "address" + } + ], + "name": "skillLevelRatingsCount", + "outputs": [ + { + "internalType": "uint8", + "name": "", + "type": "uint8" + } + ], + "stateMutability": "view", + "type": "function" + } + ] + } + \ No newline at end of file diff --git a/src/components/ProfileDetailsOpsSkillRating.tsx b/src/components/ProfileDetailsOpsSkillRating.tsx new file mode 100644 index 0000000..1466ca1 --- /dev/null +++ b/src/components/ProfileDetailsOpsSkillRating.tsx @@ -0,0 +1,55 @@ +import { useContractRead } from "wagmi" +import LoadingIndicator from "./LoadingIndicator" +import OperatorSkillLevels from "../../abis/OperatorSkillLevels.json" +import { useIsMounted } from "@/hooks/useIsMounted" +import Link from "next/link" +import { formatEther } from "viem" + +export default function ProfileDetailsOpsSkillRating({ citizen }: any) { + console.info('ProfileDetailsOpsSkillRating') + + const { data, isError, isLoading } = useContractRead({ + address: '0x8100e77899C24b0F7B516153F84868f850C034BF', + abi: OperatorSkillLevels.abi, + functionName: 'skillLevelAverages', + args: [citizen.ownerAddress] + }) + console.info('data:', data) + console.info('isError:', isError) + console.info('isLoading:', isLoading) + + if (!useIsMounted() || isLoading) { + return + } else { + let skillLevelAverage: any = null + if (data) { + skillLevelAverage = data + } + console.info('skillLevelAverage:', skillLevelAverage) + if (!skillLevelAverage) { + return ( +
+ + Rate 🔗 + + Not rated +
+ ) + } else { + const skillLevelAverageEther: number = Number(formatEther(skillLevelAverage)) + console.info('skillLevelAverageEther:', skillLevelAverageEther) + const skillLevelAverageRounded: number = Math.round(skillLevelAverageEther) + console.info('skillLevelAverageRounded:', skillLevelAverageRounded) + const ratingValues = ['★☆☆☆☆ 1/5', '★★☆☆☆ 2/5', '★★★☆☆ 3/5', '★★★★☆ 4/5', '★★★★★ 5/5'] + const ratingTextColors = ['text-orange-400', 'text-amber-400', 'text-lime-400', 'text-emerald-400', 'text-cyan-400' ] + return ( + <> + + Rate 🔗 + + {ratingValues[skillLevelAverageRounded - 1]} + + ) + } + } +} diff --git a/src/pages/[passportId].tsx b/src/pages/[passportId].tsx index d1c0c15..78f8fbd 100644 --- a/src/pages/[passportId].tsx +++ b/src/pages/[passportId].tsx @@ -19,6 +19,7 @@ import NFTImage from '@/components/NFTImage' import Link from 'next/link' import Head from 'next/head' import ProfileDetailsDevSkillRating from '@/components/ProfileDetailsDevSkillRating' +import ProfileDetailsOpsSkillRating from '@/components/ProfileDetailsOpsSkillRating' import { InformationCircleIcon } from '@heroicons/react/24/outline' const Chart = dynamic(() => import('react-apexcharts'), { ssr: false }) @@ -106,9 +107,13 @@ export default function ProfilePage({ citizen, nationCred, veNation, dework, coo Not linked
  • - Developer skill level
    + 🧙 Developer skill level
  • +
  • + ⚙️ Operator skill level
    + +
  • )}