Quickstart
Follow these steps to install the Tezos SDK for Unity in an existing Unity project and start using it.
These instructions cover:
- Installing the SDK into an existing Unity project
- Testing that the SDK works in your project
- Connecting to a user's Tezos wallet
- Creating and managing tokens
- Prompting the user to sign messages
Installing the SDK
-
Make sure that you have Unity Editor version 2021.3.23f1 or later.
-
Install a Tezos-compatible wallet on a mobile device that has a camera and can scan QR codes. For instructions, see Installing and funding a wallet.
-
Switch the wallet to Ghostnet and use the Ghostnet faucet to get some tez that you can use to send transactions to your Unity projects.
-
Install the SDK from GitHub:
- From the SDK's GitHub page at https://github.com/trilitech/tezos-unity-sdk/releases, download the latest release of the SDK, which is a file named
TezosUnitySdk.unitypackage
. - In the Unity Editor, click Assets > Import Package > Custom Package and import the SDK file.
- Verify that the SDK is installed by opening the Project panel and expanding the Assets > TezosSDK folder.
If you see the TezosSDK folder with sub-folders including Editor, Examples, Resources, and Runtime, the SDK is installed correctly, as in this picture:
- From the SDK's GitHub page at https://github.com/trilitech/tezos-unity-sdk/releases, download the latest release of the SDK, which is a file named
-
Verify that the SDK works in your project by running the WalletConnection example scene, which demonstrates how to connect to a Tezos wallet in a Unity project:
-
In the Project panel, expand Assets > TezosSDK > Examples > WalletConnection and open the
_WalletConnection
scene. -
Run the scene, which shows a QR code.
-
Use your wallet app to scan the barcode and confirm the connection to the Unity project. The project UI shows the address of the connected account, as in this picture:
-
Connecting to wallets
Connecting to a user's wallet is a prerequisite to working with Tezos in any application. Accessing the wallet allows your project to see the tokens in it and to prompt the user to submit transactions, but it does not give your project direct control over the wallet. Users must still confirm all transactions in their wallet application.
Using a wallet application in this way saves you from having to implement payment processing and security in your application. Game developers can also use the wallet and its account as a unique account identifier and as the user's inventory.
-
Copy the
MainThreadExecutor
andTezosManager
prefabs to your scene. These prefabs provide prerequisites to use Tezos in a scene. TheTezosManager
fields control what users see in their wallet applications before connecting to the project, as shown in this picture of the Inspector panel: -
Add features to your project that connect to users' wallets. You can copy objects from the
_WalletConnection
scene, including the QRCode, LogoutButton, and AccountAddress objects.The
TezosSDK/Examples/WalletConnection/Scripts/QRImageGenerator.cs
file receives a handshake from theTezosManager
object and uses the handshake information to generate a URI and encode that URI to a QR code image:private void SetQrCode(HandshakeData handshake_data)
{
if (encoded)
{
return;
}
var uri = "tezos://?type=tzip10&data=" + handshake_data.PairingData;
EncodeTextToQrCode(uri);
}You can adjust this code and the bound Unity object to control when and where the QR code appears.
-
Add features to your project to use the connected account. For example, the
TezosSDK/Examples/Common/Scripts/AccountInfoUI.cs
file responds to theAccountConnected
event, which runs when the user scans the QR code and approves the connection in their wallet app. When the event runs, it uses theTezosManager.Instance.Wallet
object to get information about the connected account, such as its address:private void Start()
{
addressText.text = notConnectedText;
TezosManager.Instance.MessageReceiver.AccountConnected += OnAccountConnected;
TezosManager.Instance.MessageReceiver.AccountDisconnected += OnAccountDisconnected;
}
private void OnAccountDisconnected(AccountInfo account_info)
{
addressText.text = notConnectedText;
}
private void OnAccountConnected(AccountInfo account_info)
{
addressText.text = TezosManager.Instance.Wallet.GetActiveAddress();
// OR
addressText.text = account_info.Address;
}You can use this address as a user's account ID because Tezos account addresses are unique.
-
To respond to other events, add listeners for the events that the SDK provides. You can see these events and their return values in the MessageReceiver object.
Note that if you stop the project while your wallet is connected and restart the project later, it remembers that your wallet is connected. The SDK uses the Beacon SDK to connect to wallets and remember connections.
Managing tokens
To create and work with tokens in your project, you need a smart contract to manage them. Smart contracts are programs on the blockchain that are permanent and hosted automatically by Tezos nodes, so you don't have to worry about where to host contracts and whether they will run when users call them.
Smart contracts can do many tasks, but for gaming they have two main purposes:
- They handle tokens, which are digital assets stored on the blockchain
- They provide backend logic that users can trust because it cannot change
You can create your own smart contracts or use the built-in contract that the SDK provides for managing tokens in Unity projects. The built-in contract is compatible with the FA2 token standard, which means that you can use a single smart contract to manage any number of types of tokens. Each token type can be:
- Fungible tokens, which are collections of interchangeable tokens with a quantity that you define. Fungible tokens can be quantifiable commodities like in-game currency, fuel, ammunition, or energy, or they can be identical items with a limited quantity.
- Non-fungible tokens (NFTs), which are unique assets with only one unit. Games use NFTs for items that are unique and must not be duplicated.
For more information about smart contracts, see Smart contracts.
-
To deploy the built-in contract, call the
TezosManager.Instance.Tezos.TokenContract.Deploy
method and pass a callback function:public void DeployContract()
{
TezosManager
.Instance
.Tezos
.TokenContract
.Deploy(OnContractDeployed);
}
private void OnContractDeployed(string contractAddress)
{
Debug.Log(contractAddress);
}The project sends the deployment transaction to the connected wallet, which must approve the transaction and pay the related fees.
The callback function receives the address of the deployed contract, which the project uses to send requests to the contract. It can take a few minutes for the contract to deploy and be confirmed in multiple blocks on the blockchain. The SDK stores the address of the contract as
TezosManager.Instance.Tezos.TokenContract.Address
.The address that deployed the contract becomes the administrator of the contract and is the only account that can create tokens.
-
To create tokens, call the contract's
mint
entrypoint.The built-in contract has a
mint
method that accepts these parameters and calls the deployed contract'smint
entrypoint:- A callback function to run when the token is created
- The metadata for the token, which includes a name and description, URIs to preview media or thumbnails, and how many decimal places the token can be broken into
- The destination account that owns the new tokens, which can be a user account, this smart contract, or any other smart contract
- The number of tokens to create
For example, this code creates a token type with a quantity of 100:
var initialOwner = TezosManager
.Instance
.Wallet
.GetActiveAddress();
const string imageAddress = "ipfs://QmX4t8ikQgjvLdqTtL51v6iVun9tNE7y7Txiw4piGQVNgK";
var tokenMetadata = new TokenMetadata
{
Name = "My token",
Description = "Description for my token",
Symbol = "MYTOKEN",
Decimals = "0",
DisplayUri = imageAddress,
ArtifactUri = imageAddress,
ThumbnailUri = imageAddress
};
TezosManager
.Instance
.Tezos
.TokenContract
.Mint(
completedCallback: OnTokenMinted,
tokenMetadata: tokenMetadata,
destination: initialOwner,
amount: 100);
private void OnTokenMinted(TokenBalance tokenBalance)
{
Debug.Log($"Successfully minted token with Token ID {tokenBalance.TokenId}");
}For a complete example of creating tokens, see the file
TezosSDK/Examples/Contract/Scripts/MintToken.cs
and the Contract example scene.Note that developers usually don't store large files such as images or large metadata files directly on the blockchain. Instead, they set up distributed storage for the file with the InterPlanetary File System (IPFS). For example, Pinata provides an API to store files in IPFS. The tutorial Create a contract and web app that mints NFTs covers setting up an IPFS account and storing files on IPFS. The code in the previous example assumes that the image that represents the token is already stored on IPFS.
-
To transfer tokens, call the contract's
Transfer
entrypoint.Like the
mint
entrypoint, the built-in contract has atransfer
method that accepts these parameters and calls the deployed contract'stransfer
entrypoint:- A callback function to run when the transfer is complete
- The account to transfer the tokens to
- The ID of the token
- The amount of tokens to transfer
This example transfers 12 tokens with the ID 5 to the account in the variable
destinationAccountAddress
.public void HandleTransfer()
{
TezosManager
.Instance
.Tezos
.TokenContract
.Transfer(
completedCallback: TransferCompleted,
destination: destinationAccountAddress,
tokenId: 5,
amount: 12);
}
private void TransferCompleted(string txHash)
{
Logger.LogDebug($"Transfer complete with transaction hash {txHash}");
}For a complete example, see the Transfer example scene.
-
To get the tokens that the connected account owns, call the
TezosManager.Instance.Tezos.API.GetTokensForOwner
method in a coroutine. This example prints information about the tokens that the account owns to the log:using Beacon.Sdk.Beacon;
using TezosSDK.Beacon;
using TezosSDK.Tezos;
using TezosSDK.Helpers;
using TezosSDK.Tezos.API.Models.Tokens;
using TezosSDK.Tezos.API.Models.Filters;
using System.Linq;
// ...
void Start()
{
TezosManager.Instance.MessageReceiver.AccountConnected += OnAccountConnected;
}
private void OnAccountConnected(AccountInfo account_info)
{
var address = TezosManager.Instance.Wallet.GetActiveAddress();
var routine = TezosManager.Instance.Tezos.API.GetTokensForOwner(
callback: onTokenBalances,
owner: address,
withMetadata: true,
maxItems: 10_000,
orderBy: new TokensForOwnerOrder.ByLastTimeAsc(0)
);
StartCoroutine(routine);
}
private void onTokenBalances(IEnumerable<TokenBalance> tokenBalances)
{
var address = TezosManager.Instance.Wallet.GetActiveAddress();
List<TokenBalance> tokens = new List<TokenBalance>(tokenBalances);
// Filter to the tokens in the active contract
List<TokenBalance> filteredTokens = tokens.Where(tb => tb.TokenContract.Address == TezosManager.Instance.Tezos.TokenContract.Address).ToList();
if (filteredTokens.Count > 0)
{
foreach (var tb in filteredTokens)
{
Debug.Log(
$"{address} has {tb.Balance} tokens on contract {tb.TokenContract.Address}");
Debug.Log(tb.TokenMetadata);
}
}
else
{
Debug.Log($"{address} has no tokens in the active contract");
}
} -
To see all of the tokens that the contract contains, copy the contract address, which starts with
KT1
, and search for that address on a block explorer such as Better Call Dev. For example, this image shows the Tokens tab of Better Call Dev and the three types of tokens in the contract:
Signing messages
You can also use the connection to the user's wallet to prompt them to sign messages. Signing a message proves that it came from a specific user's wallet because the wallet encrypts the message with the user's account's key. You can prompt users to sign messages for several reasons, including:
- Ensuring that a message came from them
- Ensuring that a user has the private key for a specific account
- Proving that they own a specific token, such as if they want to offer the token for sale
- Requiring operations to run only when multiple users sign messages
- Pre-signing transactions that another application can run later and pay the fees for on the signer's behalf
For example, this code prompts the user to sign the message "This message came from my account."
using Beacon.Sdk.Beacon.Sign;
// ...
string payload = "This message came from my account.";
void Start()
{
TezosManager.Instance.MessageReceiver.PayloadSigned += OnPayloadSigned;
TezosManager.Instance.Wallet.RequestSignPayload(SignPayloadType.micheline, payload);
}
private void OnPayloadSigned(SignResult obj)
{
// result is true if the message is signed correctly
// and that it came from the currently-connected account
var result = TezosManager.Instance.Wallet.VerifySignedPayload(SignPayloadType.micheline, payload);
Debug.Log($"Payload verification response: {result}");
}
The RequestSignPayload
method sends the request to the user's wallet.
If the user approves the request, the wallet returns the signed payload, which includes encrypted information about the account that it came from.
The callback uses the VerifySignedPayload
method to verify the signed payload.
This method compares the specified message to the most recent signature it received and returns true if the messages are the same and the signature matches the currently connected account.
For more examples of how to work with the SDK, see the scenes in the TezosSDK/Examples
folder.