Skip to content

An adventure game template on SKALE blockchain with NFT, SFT, AI

License

Notifications You must be signed in to change notification settings

andreykobal/skale-nft-game

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 

History

49 Commits
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 

Repository files navigation

SKALE OpenVerse: Adventure Game Template With NFT, SFT, AI

logo-skale

Welcome to the SKALE OpenVerse: Adventure Game Template GitHub repository! This repository contains the Unity game project along with the smart contracts used in the game. Adventure Game Template is an exciting 3D game where players use NFT avatars, complete quests, collect diamonds, and explore magical worlds.

Project Setup

To set up the Adventure Game Template project, please follow these steps:

  1. Clone the repository to your local machine using the following command:

    git clone https://github.com/andreykobal/skale-nft-game.git
    
  2. Open the Unity game project that is located in /unity-project using Unity Hub or your preferred Unity editor.

  3. Install the necessary dependencies and packages required for the project.

  4. Configure the Unity project settings as needed and ensure that the project is properly configured for your target platform.

  5. Set up the Metamask extension in your preferred browser and create a wallet.

  6. Configure the Metamask wallet by connecting it to the SKALE network and obtaining the necessary testnet tokens SKALE Quick Start.

  7. Build and run the game to start playing Adventure Game Template!

Smart Contract Deployment

To deploy the smart contracts using Hardhat, please follow these steps:

  • Clone this and run npm i in terminal.
  • Add .env file with:
MNEMONIC=privatekey. not the seedphrase
  • Edit the deploy script to pass in your name and ticker/uri depending on standard
  • Edit the contractUri method in the contract and add your collection metadata URI
  • Edit the mint script and add your token uri, contract address and account address of the account you want to mint to.
  • Deploy with npx hardhat run --network scaleTestnet scripts/deploy721.js for ERC-721
  • Deploy with npx hardhat run --network scaleTestnet scripts/deploy1155.js for ERC-1155
  • Mint 721 with npx hardhat run --network scaleTestnet scripts/mint.js
  • Mint 1155 with npx hardhat run --network scaleTestnet scripts/mint1155.js
  • Check balance of ERC-1155 with npx hardhat run --network scaleTestnet scripts/balances.js
  • Get metadata of ERC-721 with npx hardhat run --network scaleTestnet scripts/getTokensOfOwner.js

Contract code inspired from Opensea: https://github.com/ProjectOpenSea/meta-transactions/blob/main/contracts/ERC721MetaTransactionMaticSample.sol This is for gas-less transactions when transferring assets. Users dont have to pay that extra gas, and get a better experience.

Smart Contract Code Highlights

GameItem.sol

The GameItem.sol contract is responsible for managing the 721 non-fungible token (NFT) avatars in Adventure Game Template. Players can purchase avatars and own them using this contract.

Important code snippets from GameItem.sol:

  • mintItem function: This function mints a new avatar NFT with the provided token URI and assigns it to the caller's address.
    function mintItem(string memory tokenURI) public payable {
        // Mint a new avatar NFT and assign it to the caller's address
        // ...
    }

GameItems.sol

The GameItems.sol contract handles the 1155 semi-fungible token (SFT) diamonds in Adventure Game Template. Players can collect these diamonds and use them to claim rewards.

Important code snippets from GameItems.sol:

  • batchMint function: This function allows the batch minting of diamond tokens with the provided token URIs and assigns them to the caller's address.

    function batchMint(string[] memory tokenURIs) public {
        // Batch mint diamond tokens and assign them to the caller's address
        // ...
    }
  • getTotalBalance function: This function returns the total balance of diamonds owned by a specific address.

    function getTotalBalance(address account) public view returns (uint256) {
        // Get the total balance of diamonds owned by the specified address
        // ...
    }

Game Code Highlights

Here are some C# code highlights from the Adventure Game Template project:

MintNFT.cs

async public void mintItem(int avatarIndex)
{
    string chainId = "1351057110";
    var tokenURI = "https://bafkreiczapdwomdlotjqt4yaojyizlgarn4kq57smi3ptkwn5lug5yz7yu.ipfs.nftstorage.link/";

    string contractAbi = "";
    string contractAddress = contractAddresses[avatarIndex];
    string method = "mintItem";
    
    var provider = new JsonRpcProvider("https://staging-v3.skalenodes.com/v1/staging-fast-active-bellatrix");
    var contract = new Contract(contractAbi, contractAddress, provider);
    Debug.Log("Contract: " + contract);
    
    var calldata = contract.Calldata(method, new object[]
    {
        tokenURI
    });
    
    string response = await Web3Wallet.SendTransaction(chainId, contractAddress, "500000000000000000", calldata, "", "");

    Debug.LogError(response);
    
    StartCoroutine(WaitAndCheckBalance(avatarIndex));
}

CheckNFTOwnership.cs

async public void CheckNFTBalance(string contractAddress, int avatarIndex)
{
    string contractAbi = "";
    string method = "getBalance";

    var provider = new JsonRpcProvider("https://staging-v3.skalenodes.com/v1/staging-fast-active-bellatrix");
    var contract = new Contract(contractAbi, contractAddress, provider);
    Debug.Log("Contract: " + contract);


    // Call the getBalance method
    var walletAddress = PlayerPrefs.GetString("Account");
    Debug.Log("Wallet address is: " + walletAddress);
    var calldata = await contract.Call(method, new object[]
    {
        // Pass the wallet address for which you want to retrieve the balance
        walletAddress
    });
    avatarBalances[avatarIndex] = int.Parse(calldata[0].ToString());
    print("ERC-721 Token Balance for avatar " + (avatarIndex + 1) + ": " + avatarBalances[avatarIndex]);

    string buttonText = avatarBalances[avatarIndex] > 0 ? "Select" : "Buy 0.5 $SKL";
    ui.UpdateButtonLabel(avatarIndex, buttonText);
}

Check1155Balance.cs

async public void Check1155TotalBalance()
{
    string contractAbi = "";
    string contractAddress = "0x6721De8B1865A6cD98C64165305611B1f28B95e4";
    string method = "getTotalBalance";
    
    var walletAddress = PlayerPrefs.GetString("Account");
    var provider = new JsonRpcProvider("https://staging-v3.skalenodes.com/v1/staging-fast-active-bellatrix");
    var contract = new Contract(contractAbi, contractAddress, provider);
    var calldata = await contract.Call(method, new object[]
    {
        // Pass the account address as the parameter
        walletAddress
    });
    var totalDiamonds = calldata[0];
    
    print("Contract 1155 Balance (Diamonds) Total: " + totalDiamonds);
    
    diamondsTotalLabel = uiDocument.rootVisualElement.Q<Label>("DiamondsTotal");
    diamondsTotalLabel.text = totalDiamonds.ToString();
}

MintDiamonds.cs

private async void Start()
{
    // Find the "Claim" button in the UI document
    Button claimButton = GetComponent<UIDocument>().rootVisualElement.Q<Button>("ClaimButton");
    claimElement = GetComponent<UIDocument>().rootVisualElement.Q("Claim");

    // Add a click event handler to the "Claim" button
    claimButton.clicked += Mint1155Diamonds;
}

async public void Mint1155Diamonds()
{
    string chainId = "1351057110";
    string contractAddress = "0x6721De8B1865A6cD98C64165305611B1f28B95e4";
    string value = "0";
    string abi = "";
    string method = "batchMint";
    string gasLimit = "";
    string gasPrice = "";
    string[] uri = { "https://bafkreiczapdwomdlotjqt4yaojyizlgarn4kq57smi3ptkwn5lug5yz7yu.ipfs.nftstorage.link/" };
    var contract = new Contract(abi, contractAddress);
    var calldata = contract.Calldata(method, new object[]
    {
        uri
    });

    string response = await Web3Wallet.SendTransaction(chainId, contractAddress, value, calldata, gasLimit, gasPrice);
    print(response);

    // Check if there is a response
    if (!string.IsNullOrEmpty(response))
    {
        claimElement.visible = false;
    }
}

GetTokensMetadata.cs

public class GetTokensMetadata : MonoBehaviour
{
    async public void CheckTokensOwned()
    {
        // ABI for GameItem contract
        string contractAbi = "YOUR_CONTRACT_ABI";
        // address of contract
        string contractAddress = "0x643D5cf6fdd9Faa3825e194e925D07E290E993D2"; // Replace with your contract's address.
        string method = "getTokensOfOwner";
        // you can use this to create a provider for hardcoding and parse this instead of rpc get instance
        var provider = new JsonRpcProvider("https://staging-v3.skalenodes.com/v1/staging-fast-active-bellatrix");
        var contract = new Contract(contractAbi, contractAddress, provider);
        Debug.Log("Contract: " + contract);

        // address of the owner whose tokens we want to check
        string ownerAddress = PlayerPrefs.GetString("Account"); // Replace with the address of the owner whose tokens you want to check.

    var calldata = await contract.Call(method, new object[]
    {
        ownerAddress // parameter for the getTokensOfOwner function
    });

    try
    {
        List<string> tokenURIs = calldata[0] as List<string>;

        if(tokenURIs != null)
        {
            foreach (string tokenURI in tokenURIs)
            {
                print("Token URI: " + tokenURI);
            }
        }
        else
        {
            print("Could not cast returned data to List<string>.");
        }
    }
    catch(Exception e)
    {
        print("Exception when casting: " + e.Message);
    }
    }
}

ClaimController.cs

Distributing sFUEL to users is an important part of building on SKALE. Docs: sFUEL Distribution

public class ClaimController : MonoBehaviour
{
    private const string BaseUrl = "https://skale-sfuel-77334b37033d.herokuapp.com";

    private void Start() {
        string currentAccount = PlayerPrefs.GetString("Account");
        Claim(currentAccount);
    }

    public void Claim(string ethereumAddress)
    {
        string url = $"{BaseUrl}/claim/{ethereumAddress}";
        StartCoroutine(SendRequest(url));
    }

    private IEnumerator SendRequest(string url)
    {
        using (UnityWebRequest webRequest = UnityWebRequest.Get(url))
        {
            yield return webRequest.SendWebRequest();

            if (webRequest.result == UnityWebRequest.Result.Success)
            {
                Debug.Log("sFuel Claim Request successful");
                Debug.Log(webRequest.downloadHandler.text);
            }
            else
            {
                Debug.LogError("sFuel Claim Request failed");
                Debug.LogError(webRequest.error);
            }
        }
    }
}

Contributing

Thank you for your interest in contributing

to Adventure Game Template! If you would like to contribute, please follow these guidelines:

  • Fork the repository and create a new branch for your contributions.
  • Make your changes and ensure that the code adheres to the project's coding style and conventions.
  • Test your changes thoroughly.
  • Submit a pull request describing your changes and the rationale behind them.

We appreciate your contributions to make Adventure Game Template an even more exciting game!

License

This project is licensed under the MIT License. Feel free to use, modify, and distribute the code for personal or commercial purposes.

Acknowledgements

We would like to express our gratitude to the open-source community for their valuable contributions and the resources that helped in the development of Adventure Game Template.

If you have any questions or encounter any issues, please don't hesitate to reach out. Enjoy playing Adventure Game Template!

About

An adventure game template on SKALE blockchain with NFT, SFT, AI

Resources

License

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published