Next.js + Move 石头剪刀布

  • Nigdle
  • 更新于 2024-11-08 22:38
  • 阅读 914

Next.js + Move 石头剪刀布

rock-paper-scissors

写在前面

本地

源码:https://github.com/zcy1024/SuiStudy/tree/main/rock-paper-scissors

# 或其它等价的命令来安装依赖并将项目跑起来
pnpm install
pnpm run dev
# http://localhost:3000/

在线(如果没过期的话)

https://rock-paper-scissors.walrus.site/

前端(样式布局)

初始化

Sui dApp 项目生成器创建一个使用 Sui Testnet 的 Next.js 框架,一切按照提示进行,生成器指南中有详细操作流程,这里就不再赘述。

config/index.ts是根据你所选择的网络环境进行初始化配置的文件,我们这里不需要动它。<br>contracts文件夹用来存储智能合约,lib/contracts文件夹用来存储前端对链上合约的调用,public文件夹用来存储静态文件,utils文件夹用来存储通用的函数或工具,清空这些文件夹里生成的样板代码(直接删除ts文件)。<br>app文件夹是网页的主体,我们也来对其做一些样板代码清理工作。首先,删除favicon.ico,这是网页标签页的图标,然后,进入page.tsx,将里面的代码除了基础结构外全部清除,就像这样:

'use client'

import Image from 'next/image'

export default function Home() {
    return (
        &lt;div className="">
            &lt;Image src="/logo/logo.jpg" alt="Sui Logo" width={80} height={40} />
        &lt;/div>
    );
}

准备好HOH社区的logo,将其替换掉/logo/logo.jpg,接下来更改app/layout.tsx文件中的metadata,该结构中的参数会影响网页标签页的展示内容:

export const metadata: Metadata = {
    title: "Rock Paper Scissors",
    description: "Classic Game: Rock Paper Scissors",
    icons: "/logo/logo.jpeg"
};

在 Next.js 中,使用public文件夹中的静态文件的时候,直接用/来表示public/,上面的&lt;Image src="/logo/logo.jpg"... />也是同理。

最后,简单了解一下app下的其它内容:fonts/fonts.tsglobals.css是创建 Next.js 框架时自带的字体样式(处理)和全局css样式(已配置好tailwindcss),providers.tsx用来初始化 Sui 网络环境、钱包等配置。

至此,我们运行项目,应该能得到如下界面:

init.png

页面搭建

整体布局

  • 页面上方一条导航栏,左侧放logo,右侧放连接钱包的按钮。
  • 剩余部分都用来作为石头剪刀布的游戏区域。

得益于 Sui dApp 项目生成器的配置,连接钱包的按钮就只需要调用@mysten/dapp-kit中提供的组件ConnectButton

为了让布局更直观,我们暂时为上下两块区域加上背景颜色:

'use client'

import Image from 'next/image'
import {ConnectButton} from "@mysten/dapp-kit";

export default function Home() {
    return (
        &lt;div className="flex flex-col h-screen mx-64">
            &lt;div className="bg-red-600 flex justify-between items-center">
                &lt;Image src="/logo/logo.jpeg" alt="HOH Logo" width={80} height={80} priority={true} />
                &lt;ConnectButton />
            &lt;/div>
            &lt;div className="flex-1 bg-yellow-600">&lt;/div>
        &lt;/div>
    );
}

global_layout.png

游戏区域布局

寻找石头、剪刀、布的图片,存储至public/game/目录下,分别以rock.pngscissors.pngpaper.png命名,以其中任意一张作为样本,将其放到游戏区域的中央,这将作为游戏开始的点击按键,同时为其绑定触发函数:

const playGame = () => {
    console.log('play game');
}

&lt;div className="relative flex-1 bg-yellow-600">
    &lt;div className="absolute top-1/2 left-1/2 -translate-x-1/2 -translate-y-1/2 cursor-pointer" onClick={playGame}>
        &lt;Image src="/game/rock.png" alt="start button" width={100} height={100} priority={true} className="w-auto h-auto" />
    &lt;/div>
&lt;/div>

click_play.png

在这个开始按键的上方,是敌方(链上随机)选择区域;下方则是我方(鼠标点击)选择区域。类似的,用flex规划好区域后往里面填充内容:

&lt;div className="relative flex-1 bg-yellow-600">
    &lt;div className="absolute top-1/2 left-1/2 -translate-x-1/2 -translate-y-1/2 cursor-pointer" onClick={playGame}>
        &lt;Image src="/game/rock.png" alt="start button" width={100} height={100} priority={true} className="w-auto h-auto" />
    &lt;/div>
    &lt;div className="absolute top-0 left-0 w-full h-1/2 flex justify-evenly items-center">
        &lt;Image src="/game/rock.png" alt="enemy" width={100} height={100} priority={true} className="w-auto h-auto" />
    &lt;/div>
    &lt;div className="absolute bottom-0 left-0 w-full h-1/2 flex justify-evenly items-center">
        &lt;Image src="/game/rock.png" alt="rock" width={100} height={100} priority={true} className="w-auto h-auto" />
        &lt;Image src="/game/scissors.png" alt="scissors" width={100} height={100} priority={true} className="w-auto h-auto" />
        &lt;Image src="/game/paper.png" alt="paper" width={100} height={100} priority={true} className="w-auto h-auto" />
    &lt;/div>
&lt;/div>

init_choose.png

这时我们发现,中间的开始按钮失去了作用,仔细观察不难发现,这是因为后续的敌我双方的布局覆盖在了上方,最简单的办法是将开始按钮的代码向下移或者为其自定义z-index属性。

之后,我们来思考一个逻辑,开始按钮和后续的出拳选择是否真的需要同时出现?

我们完全可以先将出拳选择区域隐藏,在点击开始后再让其显现出来,相对应的,开始按钮则需要在点击后隐藏。不难发现,它们的显隐状态归根结底都由一个数据进行控制 —— 是否开始游戏。

const [isPlaying, setIsPlaying] = useState&lt;boolean>(false);

用一个布尔值isPlaying来判断,点击后通过setIsPlaying将其设为真。<br>对于需要根据该值隐藏的内容,通过className={"..." + (isPlaying ? "..." : "...")}来设置。<br>为了消失和显现不那么突然,可以增加transition-opacity来实现渐隐渐显效果。

export default function Home() {
    const [isPlaying, setIsPlaying] = useState&lt;boolean>(false);

    const playGame = () => {
        setIsPlaying(true);
    }

    return (
        &lt;div className="flex flex-col h-screen mx-64">
            &lt;div className="bg-red-600 flex justify-between items-center">
                &lt;Image src="/logo/logo.jpeg" alt="HOH Logo" width={80} height={80} priority={true}/>
                &lt;ConnectButton/>
            &lt;/div>
            &lt;div className="relative flex-1 bg-yellow-600">
                &lt;div
                    className={"absolute top-1/2 left-1/2 -translate-x-1/2 -translate-y-1/2 z-10 transition-opacity " + (isPlaying ? "opacity-0" : "cursor-pointer opacity-100")}
                    onClick={!isPlaying ? playGame : () => {
                    }}>
                    &lt;Image src="/game/rock.png" alt="start button" width={100} height={100} priority={true}
                           className="w-auto h-auto"/>
                &lt;/div>
                &lt;div className={"transition-opacity " + (isPlaying ? "opacity-100" : "opacity-0")}>
                    &lt;div className="absolute top-0 left-0 w-full h-1/2 flex justify-evenly items-center">
                        &lt;Image src="/game/rock.png" alt="enemy" width={100} height={100} priority={true}
                               className="w-auto h-auto"/>
                    &lt;/div>
                    &lt;div className="absolute bottom-0 left-0 w-full h-1/2 flex justify-evenly items-center">
                        &lt;Image src="/game/rock.png" alt="rock" width={100} height={100} priority={true}
                               className="w-auto h-auto"/>
                        &lt;Image src="/game/scissors.png" alt="scissors" width={100} height={100} priority={true}
                               className="w-auto h-auto"/>
                        &lt;Image src="/game/paper.png" alt="paper" width={100} height={100} priority={true}
                               className="w-auto h-auto"/>
                    &lt;/div>
                &lt;/div>
            &lt;/div>
        &lt;/div>
    );
}

click_animation.gif

类似的,我们为我方选择区域的三个图添加点击事件,由于它们都是&lt;Image ... />,可以通过同一个类型的点击事件进行获取,最后通过alt属性来区分究竟选择的是石头、剪刀还是布。

const clickChoose = (e: MouseEvent&lt;HTMLImageElement>) => {
    console.log(e.currentTarget.alt);
}

&lt;div className="absolute bottom-0 left-0 w-full h-1/2 flex justify-evenly items-center">
    &lt;Image src="/game/rock.png" alt="rock" width={100} height={100} priority={true}
           className={"w-auto h-auto " + (isPlaying ? "cursor-pointer" : "")}
           onClick={isPlaying ? clickChoose : () => {
           }}/>
    &lt;Image src="/game/scissors.png" alt="scissors" width={100} height={100} priority={true}
           className={"w-auto h-auto " + (isPlaying ? "cursor-pointer" : "")}
           onClick={isPlaying ? clickChoose : () => {
           }}/>
    &lt;Image src="/game/paper.png" alt="paper" width={100} height={100} priority={true}
           className={"w-auto h-auto " + (isPlaying ? "cursor-pointer" : "")}
           onClick={isPlaying ? clickChoose : () => {
           }}/>
&lt;/div>

image_event.gif

游戏区域的布局是完成了,我们可以把之前用来辨别区域的背景去掉,纯白色太刺眼,就再加一点点灰色缓冲,但是,为什么开始按钮是石头?不如让它动起来,循环切换石头、剪刀、布,包括敌人(链上随机)在返回结果时也不应该固定显示。

大致思路:将三张图片的文件名放到一个数组中,通过不断加一再对数组长度取余使得下标达成循环,根据当前下标所对应的文件名进行显示渲染。

utils文件夹下创建三个文件sleep.tsnext.tsindex.ts

sleep.ts:顾名思义,让程序睡眠,等待多少时间后再继续向下运行。

export default function sleep(ms: number) {
    return new Promise(resolve => setTimeout(resolve, ms));
}

next.ts:在一个数组中循环不断地取下一个。

// 这里显示标注返回值中依次的类型,方便解构赋值后按照次序获得确切的类型
export default function next&lt;T>(index: number, array: T[]): [number, T] {
    const len = array.length;
    index = (index + 1) % len;
    return [index, array[index]];
}

index.ts:将utils目录下所有导出的东西归档再一同导出,方便其它地方导入。由于这里只有两个函数,便捷性提升得不明显。

import sleep from './sleep';
import next from './next';

export {
    sleep,
    next
}

回到page.tsx,借助上面的两个小函数来实现每隔一小段时间切换图片的功能:

  1. 图片名更新要实时作用到页面中,所以需要useState来创建一个字符串以及改变该字符串的函数:<br>const [loopName, setLoopName] = useState&lt;string>("rock");
  2. 定义一个下标,表示当前循环到了数组中的哪一项,很自然地想到用let index = 0;不过,用let定义的变量,除非放到全局,否则每次渲染都会重置。<br>类似于useState,有一个钩子函数useRef可以解决这个问题:const index = useRef&lt;number>(0);<br>当需要取值时,用index.current,需要更改值时,也只需要将新值赋值给index.current
  3. 定义一个包含三张图片名的数组:const array = ["rock", "scissors", "paper"];
  4. 实现一个异步函数,在里面依次调用上面两个小工具,获得数据进行更新,而这个函数则放到useEffect当中,这个useEffect的依赖项设置为loopName,即每次loopName改变后重新执行。
const waitToDispatch = async () => {
    await sleep(222);
    const [ne_idx, name] = next(index.current, array);
    index.current = ne_idx;
    setLoopName(name);
}
useEffect(() => {
    waitToDispatch().then();
}, [loopName]);

// 最后,将写死的&lt;Image src="/game/rock.png" ... />改为&lt;Image src={`/game/${loopName}.png`} ... />
// 每次loopName变化,src也会跟着变化

至此,功能已经实现且能够正常运行,不过,如果尝试build会发现其中还有一些警告,接下去来尝试解决一下:

useEffect`中用到了`waitToDispatch`,提示我们最好将其添加为依赖项,于是:`useEffect(..., [loopName, waitToDispatch])

再次build获得一个新的警告,由于waitToDispatchuseEffect的依赖项,所以它当前定义实现的位置,可能会因为重新渲染等因素出现潜在的问题。提示了两个解决方案,一个是转移实现waitToDispatch的位置,另一个是用useCallback包裹它。<br>用useCallback实现的函数,它不会因为页面重新渲染而改变,除非它检测到它的依赖项发生变化才会更新其中的逻辑,起到缓存、提升性能的作用。<br>于是,我们用其包裹:

const waitToDispatch = useCallback(async () => {
    await sleep(222);
    const [ne_idx, name] = next(index.current, array);
    index.current = ne_idx;
    setLoopName(name);
}, [index, array]);

我们知道,array内部的值其实是不会改变的,所以只需要依赖index变化来变化就可以,实际上项目也可以运行,不过又会在build时警告,所以我们将其加上。不过,加上之后,又报了个新的warning,说是由于arrayuseCallback的依赖项,当前位置可能会出现潜在的问题,需要我们转移array定义的位置,或者用useMemo将其包裹。<br>useMemouseCallback类似,但是,useMemo得到的是经过逻辑运算后的值,并将这个值缓存下来,以避免重复进行(大量的)逻辑运算,除非它的依赖项的值发生了变化才会重新进行计算。<br>于是,我们用其包裹:const array = useMemo(() => ["rock", "scissors", "paper"], []);

终于,我们解决了所有警告!附上当下page.tsx的完整代码以及演示动图:

'use client'

import Image from 'next/image'
import {ConnectButton} from "@mysten/dapp-kit";
import {MouseEvent, useCallback, useEffect, useMemo, useRef, useState} from "react";
import {sleep, next} from "@/utils"

export default function Home() {
    const [isPlaying, setIsPlaying] = useState&lt;boolean>(false);

    const playGame = () => {
        setIsPlaying(true);
    }

    const clickChoose = (e: MouseEvent&lt;HTMLImageElement>) => {
        console.log(e.currentTarget.alt);
    }

    const [loopName, setLoopName] = useState&lt;string>("rock");
    const index = useRef&lt;number>(0);
    const array = useMemo(() => ["rock", "scissors", "paper"], []);
    const waitToDispatch = useCallback(async () => {
        await sleep(222);
        const [ne_idx, name] = next(index.current, array);
        index.current = ne_idx;
        setLoopName(name);
    }, [index, array]);
    useEffect(() => {
        waitToDispatch().then();
    }, [loopName, waitToDispatch]);

    return (
        &lt;div className="flex flex-col h-screen mx-64 bg-gray-50 shadow-md">
            &lt;div className="flex justify-between items-center">
                &lt;Image src="/logo/logo.jpeg" alt="HOH Logo" width={80} height={80} priority={true}/>
                &lt;ConnectButton/>
            &lt;/div>
            &lt;div className="relative flex-1">
                &lt;div
                    className={"absolute top-1/2 left-1/2 -translate-x-1/2 -translate-y-1/2 z-10 transition-opacity " + (isPlaying ? "opacity-0" : "cursor-pointer opacity-100")}
                    onClick={!isPlaying ? playGame : () => {
                    }}>
                    &lt;Image src={`/game/${loopName}.png`} alt="start button" width={100} height={100} priority={true}
                           className="w-auto h-auto"/>
                &lt;/div>
                &lt;div className={"transition-opacity " + (isPlaying ? "opacity-100" : "opacity-0")}>
                    &lt;div className="absolute top-0 left-0 w-full h-1/2 flex justify-evenly items-center">
                        &lt;Image src={`/game/${loopName}.png`} alt="enemy" width={100} height={100} priority={true}
                               className="w-auto h-auto"/>
                    &lt;/div>
                    &lt;div className="absolute bottom-0 left-0 w-full h-1/2 flex justify-evenly items-center">
                        &lt;Image src="/game/rock.png" alt="rock" width={100} height={100} priority={true}
                               className={"w-auto h-auto " + (isPlaying ? "cursor-pointer" : "")}
                               onClick={isPlaying ? clickChoose : () => {
                               }}/>
                        &lt;Image src="/game/scissors.png" alt="scissors" width={100} height={100} priority={true}
                               className={"w-auto h-auto " + (isPlaying ? "cursor-pointer" : "")}
                               onClick={isPlaying ? clickChoose : () => {
                               }}/>
                        &lt;Image src="/game/paper.png" alt="paper" width={100} height={100} priority={true}
                               className={"w-auto h-auto " + (isPlaying ? "cursor-pointer" : "")}
                               onClick={isPlaying ? clickChoose : () => {
                               }}/>
                    &lt;/div>
                &lt;/div>
            &lt;/div>
        &lt;/div>
    );
}

front_end.gif

智能合约

前端页面布局暂告一段落,从这里开始将用Move编写一个简单的石头剪刀布的智能合约。

来到contracts目录,通过sui move new game命令新建合约代码。删除tests目录,里面用来编写测试代码,我们暂时用不上。打开sources/game.move准备编写合约。

我们想要达成的效果很简单,就是当玩家选择好自己是石头、剪刀还是布之后,通过链上随机的方式得到对方出什么。

于是,我们就需要编写以下内容:

  • 通过触发事件的方式来得到随机结果。
  • 随机函数。

最终,move代码如下:

module game::game {
    use sui::event;
    use sui::random::Random;

    public struct RandomEvent has copy, drop {
        chosen: u8
    }

    entry fun play(random: &Random, ctx: &mut TxContext) {
        let mut generator = random.new_generator(ctx);
        event::emit(RandomEvent {
            chosen: generator.generate_u8_in_range(1, 3)
        });
    }
}

随机得到1~3中的数,由前端处理其对应到石头、剪刀和布,sui move build没问题,sui client publish发布,成功后得到一串信息。

通过命令行调用初步观察结果:

export PACKAGE=0x5780ec9a0ab44c86b957855eab35fa3e0dacb71d683109e40c50f94fca2f411b
sui client call --package $PACKAGE --module game --function play --args 0x8
# output:
╭─────────────────────────────────────────────────────────────────────────────────────────────────────╮
│ Transaction Block Events                                                                            │
├─────────────────────────────────────────────────────────────────────────────────────────────────────┤
│  ┌──                                                                                                │
│  │ EventID: o82egWHVDnqSABWre6tustng5zb6vDBfyQvPBtDGnQs:0                                           │
│  │ PackageID: 0x5780ec9a0ab44c86b957855eab35fa3e0dacb71d683109e40c50f94fca2f411b                    │
│  │ Transaction Module: game                                                                         │
│  │ Sender: 0x9e4092b6a894e6b168aa1c6c009f5c1c1fcb83fb95e5aa39144e1d2be4ee0d67                       │
│  │ EventType: 0x5780ec9a0ab44c86b957855eab35fa3e0dacb71d683109e40c50f94fca2f411b::game::RandomEvent │
│  │ ParsedJSON:                                                                                      │
│  │   ┌────────┬───┐                                                                                 │
│  │   │ chosen │ 3 │                                                                                 │
│  │   └────────┴───┘                                                                                 │
│  └──                                                                                                │
╰─────────────────────────────────────────────────────────────────────────────────────────────────────╯

重复调用几次,发现触发的事件中的值确实是随机的,接着,我们对一些信息进行存储,打开config目录新建key.ts,用来存储发布后的ID,为了方便后续前端调用,我们可以把Random的地址、调用的完整函数名以及触发的EventType也存上并导出。

// UPGRADE_CAP 本文不会用到,但是如果后续有升级合约的需求的话需要提供
export const PACKAGE = "0x5780ec9a0ab44c86b957855eab35fa3e0dacb71d683109e40c50f94fca2f411b"
export const UPGRADE_CAP = "0xb6222d0ab94ca5388b0722de9a4aab7ad10ff74bbe91a00d7c8fd1698d185c95"
export const RANDOM = "0x8"
export const FUNCTION = `${PACKAGE}::game::play`
export const EVENT = `${PACKAGE}::game::RandomEvent`

前端与合约交互

根据 Sui dApp 教学文档,我们在page.tsx中添加以下代码:

const {mutateAsync: signAndExecuteTransaction} = useSignAndExecuteTransaction({
    execute: async ({bytes, signature}) =>
        await suiClient.executeTransactionBlock({
            transactionBlock: bytes,
            signature,
            options: {
                showRawEffects: true,
                showEvents: true
            },
        })
});

通过这段代码我们将得到一个async的函数signAndExecuteTransaction,这将是我们唤起钱包签署交易的入口;<br>根据useSignAndExecuteTransaction内部的定义,在链上交易成功后会返回带有events的信息,我们所需要的随机数就在里面。<br>如果对返回值不关心,甚至可以直接const { mutate: signAndExecuteTransaction } = useSignAndExecuteTransaction();来获取入口。

假设我们已经实现了一个名为play的函数,它接受一个参数,就是上面的这个交易入口,我们将在我方游戏区域点击事件中触发:

const clickChoose = async (e: MouseEvent&lt;HTMLImageElement>) => {
    console.log(e.currentTarget.alt);
    const chosen = await play(signAndExecuteTransaction);
    console.log(chosen);
}

play函数我们放在lib/contracts目录下,搭好最基本的函数框架:

export default async function play(signAndExecuteTransaction) {
}

首先需要为这个参数指定类型,回到page.tsx将鼠标悬停在这个定义好的交易入口上,发现它的类型是:<br>UseMutateAsyncFunction&lt;SuiTransactionBlockResponse, UseSignAndExecuteTransactionError, UseSignAndExecuteTransactionArgs, unknown><br>于是:

export default async function play(signAndExecuteTransaction: UseMutateAsyncFunction&lt;SuiTransactionBlockResponse, UseSignAndExecuteTransactionError, UseSignAndExecuteTransactionArgs, unknown>) {
}

显然,全是报错,因为很多该导入的没有导入。当然可以在当前文件导入,但如果不止有这一个交易,就需要在各个文件重新做一遍类似的操作,为了更便于管理,我们在lib/contracts下新建一个type.ts,专门用来放交易过程中可能用到的(通用)东西。<br>(可能会报错包不存在,通过pnpm add -D &lt;name>或者等价的命令将其添加)

import {UseMutateFunction, UseMutateAsyncFunction} from "@tanstack/react-query";
import {SuiTransactionBlockResponse} from "@mysten/sui/client";
import type { SuiSignAndExecuteTransactionInput } from '@mysten/wallet-standard';
import { PartialBy } from "@mysten/dapp-kit/dist/cjs/types/utilityTypes";
import { WalletFeatureNotSupportedError, WalletNoAccountSelectedError, WalletNotConnectedError } from "@mysten/dapp-kit/dist/cjs/errors/walletErrors";
import {Transaction} from "@mysten/sui/transactions";

type UseSignAndExecuteTransactionError = WalletFeatureNotSupportedError | WalletNoAccountSelectedError | WalletNotConnectedError | Error;
type UseSignAndExecuteTransactionArgs = PartialBy&lt;Omit&lt;SuiSignAndExecuteTransactionInput, 'transaction'>, 'account' | 'chain'> & {
    transaction: Transaction | string;
};

export type {
    UseMutateFunction,
    UseMutateAsyncFunction,
    SuiTransactionBlockResponse,
    UseSignAndExecuteTransactionError,
    UseSignAndExecuteTransactionArgs,

}

将上面导出的东西,导入play.ts,剩下要做的就是实现这个函数。同样的,根据Sui dApp教学,依葫芦画瓢:

const tx = new Transaction();
tx.moveCall({
    target: FUNCTION,
    arguments: [
        tx.object(RANDOM)
    ],
});
const response = await signAndExecuteTransaction({transaction: tx});

我们新建了一个交易,内容是调用FUNCTION(我们事先在config/key.ts中定义好了),调用的这个链上函数有一个参数,是一个Random对象,通过tx.object(&lt;Object Address>)来将随机数的对象地址0x8转化为对象。<br>在唤起钱包签署交易的入口里传入这一笔交易,返回的内容存储在response中。<br>接下来,只需要在其中找到(对应的)EVENT,再将其中存储的chosen返回即可:

let chosen = 0;
response.events?.forEach(event => {
    if (event.type === EVENT) {
        chosen = (event.parsedJson as ParsedJson).chosen;
    }
});
return chosen;

将项目跑起来,测试是否如我所想的那样执行:

call.gif

输赢结算

首先,在lib目录下新建一个games目录,里面建一个checkIsWinner.ts用来编写判断输赢的函数。

合约随机出的1~3分别表示石头、剪刀、布,我们将前端的选择,也就是e.currentTarget.alt按照同样的规则转化成数字,从中不难发现一个规律:<br>石头1 > 剪刀2 > 布3,当我们的选择和链上的随机数的差的绝对值小于等于1的时候,数字小的那一方获胜,否则,将两个数都对3取余数后再执行同样的判断,也就是布3 % 3 = 0 > 石头1 % 3 = 1 > 剪刀2 % 3 = 2。<br>可以证明,这个取余数的过程最多进行一次就必定会判成胜负,于是,编码如下:

function strToNumber(str: string) {
    if (str === "rock")
        return 1;
    if (str === "scissors")
        return 2;
    return 3;
}

function check(my: number, move: number) {
    if (Math.abs(my - move) > 1)
        return check(my % 3, move % 3);
    return my &lt; move;
}

export default function checkIsWinner(my_choice: string, move_choice: number) {
    return check(strToNumber(my_choice), move_choice);
}

合约交易成功后,在page.tsx中调用该函数。为了让胜负结算标签受该值控制,我们需要用useState来新建一个变量,同时为该标签绑定一个重开功能的点击事件,重开功能很容易实现,只需要将控制状态的值设为初始值即可:

const [isWinner, setIsWinner] = useState&lt;boolean | null>(null);
const clickChoose = async (e: MouseEvent&lt;HTMLImageElement>) => {
    const my_chosen = e.currentTarget.alt;
    const chosen = await play(signAndExecuteTransaction);
    setIsWinner(checkIsWinner(my_chosen, chosen));
}

const gameAgain = () => {
    setIsWinner(null);
    setIsPlaying(false);
}

{
    isWinner !== null
    &&
    &lt;div
        className="absolute w-full top-1/2 -translate-y-1/2 cursor-pointer animate-bounce text-center"
        onClick={gameAgain}>
        {isWinner ? "Congratulations, you’ve got it all!" : "No! Everyone believes you will win next time!"}
    &lt;/div>
}

game.gif

至此,大体功能已全部实现,当然,这是在没有任何误操作(比如未连接钱包开始游戏等)的前提下,同时,最后那只不断闪动的手也应该有一个最终归宿。不过这剩下的大多都是优化或者美化的环节,这里就不再详细阐述。你也可以根据自己的喜好将这一段留白增添一份天马行空的创意。

加入组织,共同进步!

点赞 0
收藏 0
分享
本文参与登链社区写作激励计划 ,好文好收益,欢迎正在阅读的你也加入。

0 条评论

请先 登录 后评论
Nigdle
Nigdle
0xa745...Fe19
江湖只有他的大名,没有他的介绍。