diff --git a/docs/build/basics/flow-token.md b/docs/build/basics/flow-token.md
index 38eb4755c5..efabb75de9 100644
--- a/docs/build/basics/flow-token.md
+++ b/docs/build/basics/flow-token.md
@@ -107,4 +107,4 @@ The Flow SDKs also allow polling for events using the Flow Access API,
## How to Build with FLOW
-To get started building on Flow, please see the [Flow App Quickstart](../guides/flow-app-quickstart.md)
+To get started building on Flow, please see the [Flow App Quickstart](../getting-started/fcl-quickstart.md)
diff --git a/docs/build/explore-more.md b/docs/build/explore-more.md
index 2b2a959d81..2beafbda87 100644
--- a/docs/build/explore-more.md
+++ b/docs/build/explore-more.md
@@ -16,7 +16,7 @@ Below are some additional tutorials to help you get started with Flow:
{
type: 'link',
label: 'Flow App Quickstart',
- href: '/build/guides/flow-app-quickstart',
+ href: '/build/getting-started/fcl-quickstart',
description: 'Simple walkthrough building a web3 app using the Flow Client Library (FCL)',
customProps: {
icon: ,
@@ -58,7 +58,7 @@ Below are some additional tutorials to help you get started with Flow:
{
type: 'link',
label: 'Walkthrough Guides',
- href: '/build/guides/flow-app-quickstart',
+ href: '/build/getting-started/fcl-quickstart',
description: 'Longer form guides to help you get started with Flow',
customProps: {
icon: ,
diff --git a/docs/build/flow.md b/docs/build/flow.md
index f8a5c12b8e..ea35f585cb 100644
--- a/docs/build/flow.md
+++ b/docs/build/flow.md
@@ -19,7 +19,7 @@ The following chapters summarize the content in this section. Read on more for d
## App Development
-The [Flow App Quickstart](./guides/flow-app-quickstart.md) covers the Flow core concepts, including:
+The [Flow App Quickstart](./getting-started/fcl-quickstart.md) covers the Flow core concepts, including:
- **App Client:** The app client is the interface through which users interact with your app. Web and mobile applications are typical examples of app clients.
- **Smart Contract:** A smart contract is a collection of code deployed to a permanent location on the blockchain that defines the core logic for a dApp.
diff --git a/docs/build/getting-started/contract-interaction.md b/docs/build/getting-started/contract-interaction.md
new file mode 100644
index 0000000000..883c10c2b2
--- /dev/null
+++ b/docs/build/getting-started/contract-interaction.md
@@ -0,0 +1,133 @@
+---
+sidebar_position: 1
+sidebar_label: Contract Interaction
+---
+
+import VerticalSplit from "./vertical-split.svg"
+
+# Contract Interaction
+
+In this quickstart guide, you'll interact with your first smart contract on the Flow Testnet. `Testnet` is a public instance of the Flow blockchain designed for experimentation, where you can deploy and invoke smart contracts without incurring any real-world costs.
+
+Smart contracts on Flow are permanent pieces of code that live on the blockchain. They allow you to encode business logic, define digital assets, and much more. By leveraging smart contracts, you can create decentralized applications (dApps) that are transparent, secure, and open to anyone.
+
+Flow supports modern smart contracts written in [Cadence], a resource-oriented programming language designed specifically for smart contracts. Cadence focuses on safety and security, making it easier to write robust contracts. Flow also supports traditional [EVM]-compatible smart contracts written in Solidity, allowing developers to port their existing Ethereum contracts to Flow. In this guide, we'll focus on interacting with Cadence smart contracts.
+
+## Objectives
+
+After completing this guide, you'll be able to:
+
+* Read data from a [Cadence] smart contract deployed on Flow.
+* Understand how to interact with contracts on Flow's `testnet`.
+* Retrieve and display data from a deployed smart contract via scripts.
+
+In later steps, you'll learn how to:
+
+* Create a Flow project using the [Flow CLI](../../tools/flow-cli/index.md).
+* Add an already-deployed contract to your project with the [Dependency Manager](../../tools/flow-cli/dependency-manager.md).
+* Deploy a smart contract locally to the [Flow Emulator](../../tools/emulator/index.md).
+* Write and execute transactions to interact with a deployed smart contract.
+* Display data from a Cadence smart contract on a React frontend using the [Flow Client Library](../../tools/clients/fcl-js/index.md).
+
+## Calling a Contract With a Script
+
+The `Counter` contract exposes a public function named `getCount()` that returns the current value of the counter. We can retrieve its value using a simple script written in the [Cadence] programming language. Scripts in Cadence are read-only operations that allow you to query data from the blockchain without changing any state.
+
+Here's the script:
+
+```cadence
+import Counter from 0x8a4dce54554b225d
+
+access(all)
+fun main(): Int {
+ return Counter.getCount()
+}
+```
+
+Let's break down what this script does:
+
+- **Import Statement**: `import Counter from 0x8a4dce54554b225d` tells the script to use the `Counter` contract deployed at the address `0x8a4dce54554b225d` on the `testnet`.
+- **Main Function**: `access(all) fun main(): Int` defines the entry point of the script, which returns an `Int`.
+- **Return Statement**: `return Counter.getCount()` calls the `getCount()` function from the `Counter` contract and returns its value.
+
+### Steps to Execute the Script
+
+- **Run the Script**: Click the Run button to execute the script.
+- **View the Output**: Observe the output returned by the script. You should see the current value of the `count` variable, which is `0` unless it has been modified.
+
+
+
+## Understanding the `Counter` Contract
+
+To fully grasp how the script works, it's important to understand the structure of the `Counter` contract. Below is the source code for the contract:
+
+```cadence
+access(all) contract Counter {
+
+ access(all) var count: Int
+
+ // Event to be emitted when the counter is incremented
+ access(all) event CounterIncremented(newCount: Int)
+
+ // Event to be emitted when the counter is decremented
+ access(all) event CounterDecremented(newCount: Int)
+
+ init() {
+ self.count = 0
+ }
+
+ // Public function to increment the counter
+ access(all) fun increment() {
+ self.count = self.count + 1
+ emit CounterIncremented(newCount: self.count)
+ }
+
+ // Public function to decrement the counter
+ access(all) fun decrement() {
+ self.count = self.count - 1
+ emit CounterDecremented(newCount: self.count)
+ }
+
+ // Public function to get the current count
+ view access(all) fun getCount(): Int {
+ return self.count
+ }
+}
+```
+
+### Breakdown of the Contract
+
+- **Contract Declaration**: `access(all) contract Counter` declares a new contract named `Counter` that is accessible to everyone.
+- **State Variable**: `access(all) var count: Int` declares a public variable `count` of type `Int`. The `access(all)` modifier means that this variable can be read by anyone.
+- **Events**: Two events are declared:
+ - `CounterIncremented(newCount: Int)`: Emitted when the counter is incremented.
+ - `CounterDecremented(newCount: Int)`: Emitted when the counter is decremented.
+- **Initializer**: The `init()` function initializes the `count` variable to `0` when the contract is deployed.
+- **Public Functions**:
+ - `increment()`: Increases the `count` by `1` and emits the `CounterIncremented` event.
+ - `decrement()`: Decreases the `count` by `1` and emits the `CounterDecremented` event.
+ - `getCount()`: Returns the current value of `count`. The `view` modifier indicates that this function does not modify the contract's state.
+
+### Key Points
+
+- **Public Access**: The `count` variable and the functions `increment()`, `decrement()`, and `getCount()` are all public, allowing anyone to interact with them.
+- **State Modification**: The `increment()` and `decrement()` functions modify the state of the contract by changing the value of `count` and emitting events.
+- **Read Costs**: Reading data from the blockchain is free on Flow. Executing scripts like the one you ran does not incur any costs. However, transactions that modify state, such as calling `increment()` or `decrement()`, will incur costs and require proper authorization.
+
+### What's Next?
+
+In the upcoming tutorials, you'll learn how to:
+
+- **Modify the Counter**: Invoke the `increment()` and `decrement()` functions to update the `count` value.
+- **Deploy Contracts**: Use the Flow CLI to deploy your own smart contracts.
+- **Interact with Contracts Locally**: Use the Flow Emulator to test contracts in a local development environment.
+- **Build Frontend Applications**: Display data from smart contracts in a React application using the Flow Client Library.
+
+By understanding the `Counter` contract and how to interact with it, you're building a solid foundation for developing more complex applications on the Flow blockchain.
+
+Proceed to the next tutorial to learn how to create your own contracts and deploy them live using the Flow CLI.
+
+
+
+[Cadence]: https://cadence-lang.org/
+[EVM]: https://flow.com/upgrade/crescendo/evm
\ No newline at end of file
diff --git a/docs/build/getting-started/fcl-quickstart.md b/docs/build/getting-started/fcl-quickstart.md
index bd532d649a..eb492da01c 100644
--- a/docs/build/getting-started/fcl-quickstart.md
+++ b/docs/build/getting-started/fcl-quickstart.md
@@ -5,164 +5,515 @@ sidebar_label: Simple Frontend
# Simple Frontend
-[Flow Client Library] (FCL), is a JavaScript library developed to facilitate interactions with the Flow blockchain. It provides developers with tools to build, integrate, and interact with Flow directly from web applications. This web app quickstart will get you interacting with a contract already deployed to Flow.
-
-For this tutorial, we're going to be making a [React] app with [Create React App]. We'll try and keep the code as simple as possible in case you're coming from another framework.
+Building upon the `Counter` contract you interacted with in [Step 1: Contract Interaction](contract-interaction.md) and deployed locally in [Step 2: Local Development](./flow-cli.md), this tutorial will guide you through creating a simple frontend application using [Next.js] to interact with the `Counter` smart contract on the local Flow emulator. Using the [Flow Client Library] (FCL), you'll learn how to read and modify the contract's state from a React web application, set up wallet authentication using FCL's Discovery UI connected to the local emulator, and query the chain to read data from smart contracts.
## Objectives
After completing this guide, you'll be able to:
-* Display data from a [Cadence] smart contract on a React frontend using the [Flow Client Library]
+- Display data from a [Cadence] smart contract (`Counter`) on a Next.js frontend using the [Flow Client Library].
+- Query the chain to read data from smart contracts on the local emulator.
+- Mutate the state of a smart contract by sending transactions using FCL and a wallet connected to the local emulator.
+- Set up the Discovery UI to use a wallet for authentication with the local emulator.
+
+## Prerequisites
+
+- Completion of [Step 1: Contract Interaction](contract-interaction.md) and [Step 2: Local Development](./flow-cli.md).
+- Flow CLI installed.
+- Node.js and npm installed.
+
+## Setting Up the Next.js App
+
+Assuming you're in your project directory from Steps 1 and 2, we'll create a Next.js frontend application to interact with your smart contract deployed on the local Flow emulator.
+
+### Step 1: Create a New Next.js App
-## Creating the App
+First, we'll create a new Next.js application using `npx create-next-app`. We'll create it inside your existing project directory and then move it up to the root directory.
-First, let's create our app and then navigate to it with the following terminal commands. From the root of where you keep your source code:
+**Assumption**: You are already in your project directory.
-```zsh
-npx create-react-app fcl-app-quickstart
-cd fcl-app-quickstart
+Run the following command:
+
+```bash
+npx create-next-app@latest fcl-app-quickstart
```
-Open the new project in a new window in your editor.
+During the setup process, you'll be prompted with several options. Choose the following:
-It comes with a default layout, but let's remove it in `src/App.js` with something simple. Copy and paste this:
+- **TypeScript**: **No**
+- **Use src directory**: **Yes**
+- **Use App Router**: **Yes**
-```tsx
-// src/App.js
+This command will create a new Next.js project named `fcl-app-quickstart` inside your current directory.
-import './App.css';
+### Step 2: Move the Next.js App Up a Directory
-function App() {
- return (
-
-
FCL App Quickstart
-
- );
-}
+Now, we'll move the contents of the `fcl-app-quickstart` directory up to your project root directory.
+
+**Note**: Moving the Next.js app into your existing project may overwrite existing files such as `package.json`, `package-lock.json`, `.gitignore`, etc. **Make sure to back up any important files before proceeding.** You may need to merge configurations manually.
+
+#### Remove the README File
+
+Before moving the files, let's remove the `README.md` file from the `fcl-app-quickstart` directory to avoid conflicts:
-export default App;
+```bash
+rm fcl-app-quickstart/README.md
```
-Now let's run our app with the following `npm` command:
+#### Merge `.gitignore` Files and Move Contents
-```zsh
-npm start
+To merge the `.gitignore` files, you can use the `cat` command to concatenate them and then remove duplicates:
+
+```bash
+cat .gitignore fcl-app-quickstart/.gitignore | sort | uniq > temp_gitignore
+mv temp_gitignore .gitignore
```
-You'll see a blank page with `FCL App Quickstart` at the top.
+Now, move the contents of the `fcl-app-quickstart` directory to your project root:
+
+On macOS/Linux:
-## Setting Up FCL
+```bash
+mv fcl-app-quickstart/* .
+mv fcl-app-quickstart/.* . # This moves hidden files like .env.local if any
+rm -r fcl-app-quickstart
+```
-In order to use FCL, we need to install it. Shut the server down then run the following to download the library and set it as a dependency in our project:
+On Windows (PowerShell):
+```powershell
+Move-Item -Path .\fcl-app-quickstart\* -Destination . -Force
+Move-Item -Path .\fcl-app-quickstart\.* -Destination . -Force
+Remove-Item -Recurse -Force .\fcl-app-quickstart
```
-npm install @onflow/fcl --save
+
+**Note**: When moving hidden files (those starting with a dot, like `.gitignore`), ensure you don't overwrite important files in your root directory.
+
+### Step 3: Install FCL
+
+Now, install the Flow Client Library (FCL) in your project. FCL is a JavaScript library that simplifies interaction with the Flow blockchain:
+
+```bash
+npm install @onflow/fcl
```
-Next we'll want to add to our FCL configuration. There's a lot you can do here, but for this simple example, let's configure `accessNode.api` to talk to the Testnet Flow Access Node. An Access Node serves as the primary point of interaction for clients, such as wallets, dapps, and other services, to communicate with the Flow network. It provides a gateway for these clients to submit transactions, query data, and retrieve information without having to connect to the entire network or maintain a full copy of the blockchain.
+## Setting Up the Local Flow Emulator and Dev Wallet
-For our example, we are going to point at a free Access Node provided by Flow. Add the following config code to your `src/App.js`:
+Before proceeding, ensure that both the Flow emulator and the Dev Wallet are running.
-```tsx
-// src/App.js
+### Step 1: Start the Flow Emulator
-import * as fcl from '@onflow/fcl';
+In a new terminal window, navigate to your project directory and run:
-fcl.config({
- 'accessNode.api': 'https://rest-testnet.onflow.org'
-});
+```bash
+flow emulator start
```
+This starts the Flow emulator on `http://localhost:8888`.
+
+### Step 2: Start the FCL Dev Wallet
+
+In another terminal window, run:
+
+```bash
+flow dev-wallet
+```
+
+This starts the Dev Wallet, which listens on `http://localhost:8701`. The Dev Wallet is a local wallet that allows you to authenticate with the Flow blockchain and sign transactions on the local emulator. This is the wallet we'll select in Discovery UI when authenticating.
+
## Querying the Chain
-On Flow, you can interact with a contract by reading from the chain with a script or changing its state with a transaction. Reading is free and is done with FCL by passing a [Cadence] script to `fcl.query`.
+Now, let's read data from the `Counter` smart contract deployed on the local Flow emulator.
-For our example we are going to read from a `HelloWorld` contract deployed to the account `0xa1296b1e2e90ca5b` on `testnet` (you can [view the contract here] to see what it looks like).
+Since you've already deployed the `Counter` contract in [Step 2: Local Development](./flow-cli.md), we can proceed to query it.
-In the same `src/App.js` file, let's create app state to store our greeting and query the chain when the component renders in order to fetch the greeting state from the `HelloWorld` contract.
+### Step 1: Update the Home Page
-```tsx
-const [greeting, setGreeting] = useState("");
+Open `src/app/page.js` in your editor.
-useEffect(() => {
- const queryChain = async () => {
- const res = await fcl.query({
- cadence: `
- import HelloWorld from 0xa1296b1e2e90ca5b
+#### Adding the FCL Configuration Before the Rest
- access(all) fun main(): String {
- return HelloWorld.greeting
- }
- `
- });
+At the top of your `page.js` file, before the rest of the code, we'll add the FCL configuration. This ensures that FCL is properly configured before we use it.
- console.log(res);
- setGreeting(res);
- }
+Add the following code:
- queryChain();
-}, []);
+```jsx
+import * as fcl from "@onflow/fcl";
+
+// FCL Configuration
+fcl.config({
+ "flow.network": "local",
+ "accessNode.api": "http://localhost:8888", // Flow Emulator
+ "discovery.wallet": "http://localhost:8701/fcl/authn", // Local Wallet Discovery
+});
```
-At this point our entire `src/App.js` file should look like this:
+This configuration code sets up FCL to work with the local Flow emulator and Dev Wallet. The `flow.network` and `accessNode.api` properties point to the local emulator, while `discovery.wallet` points to the local Dev Wallet for authentication.
-```tsx
-import { useEffect, useState } from 'react';
-import './App.css';
-import * as fcl from '@onflow/fcl';
+For more information on Discovery configurations, refer to the [Wallet Discovery Guide](../../tools/clients/fcl-js/discovery.md).
+#### Implementing the Component
+
+Now, we'll implement the component to query the count from the `Counter` contract.
+
+Update your `page.js` file to the following:
+
+```jsx
+// src/app/page.js
+
+"use client"; // This directive is necessary when using useState and useEffect in Next.js App Router
+
+import { useState, useEffect } from "react";
+import * as fcl from "@onflow/fcl";
+
+// FCL Configuration
fcl.config({
- 'accessNode.api': 'https://rest-testnet.onflow.org'
+ "flow.network": "local",
+ "accessNode.api": "http://localhost:8888",
+ "discovery.wallet": "http://localhost:8701/fcl/authn", // Local Dev Wallet
});
-function App() {
- const [greeting, setGreeting] = useState("");
+export default function Home() {
+ const [count, setCount] = useState(0);
+
+ const queryCount = async () => {
+ try {
+ const res = await fcl.query({
+ cadence: `
+ import Counter from 0xf8d6e0586b0a20c7
+ import NumberFormatter from 0xf8d6e0586b0a20c7
+
+ access(all)
+ fun main(): String {
+ // Retrieve the count from the Counter contract
+ let count: Int = Counter.getCount()
+
+ // Format the count using NumberFormatter
+ let formattedCount = NumberFormatter.formatWithCommas(number: count)
+
+ // Return the formatted count
+ return formattedCount
+ }
+ `,
+ });
+ setCount(res);
+ } catch (error) {
+ console.error("Error querying count:", error);
+ }
+ };
+
+ useEffect(() => {
+ queryCount();
+ }, []);
+
+ return (
+
+
FCL App Quickstart
+
Count: {count}
+
+ );
+}
+```
+
+In the above code:
- useEffect(() => {
- const queryChain = async () => {
- const res = await fcl.query({
- cadence: `
- import HelloWorld from 0xa1296b1e2e90ca5b
+- We import the necessary React hooks (`useState` and `useEffect`) and the FCL library.
+- We define the `Home` component, which is the main page of our app.
+- We set up a state variable `count` using the `useState` hook to store the count value.
+- We define an `async` function `queryCount` to query the count from the `Counter` contract.
+- We use the `useEffect` hook to call `queryCount` when the component mounts.
+- We return a simple JSX structure that displays the count value on the page.
+- If an error occurs during the query, we log it to the console.
+- We use the script from Step 2 to query the count from the `Counter` contract and format it using the `NumberFormatter` contract.
- access(all) fun main(): String {
- return HelloWorld.greeting
- }
- `
- });
+### Step 2: Run the App
- console.log(res);
- setGreeting(res);
- };
+Start your development server:
+
+```bash
+npm run dev
+```
- queryChain();
- }, []);
+Visit `http://localhost:3000` in your browser. You should see the current count displayed on the page, formatted according to the `NumberFormatter` contract.
- return (
-
-
FCL App Quickstart
-
{greeting}
+## Mutating the Chain State
+
+Now that we've successfully read data from the Flow blockchain emulator, let's modify the state by incrementing the `count` in the `Counter` contract. We'll set up wallet authentication and send a transaction to the blockchain emulator.
+
+### Adding Authentication and Transaction Functionality
+
+#### Step 1: Manage Authentication State
+
+In `src/app/page.js`, add new state variables to manage the user's authentication state:
+
+```jsx
+const [user, setUser] = useState({ loggedIn: false });
+```
+
+#### Step 2: Subscribe to Authentication Changes
+
+Update the `useEffect` hook to subscribe to the current user's authentication state:
+
+```jsx
+useEffect(() => {
+ fcl.currentUser.subscribe(setUser);
+ queryCount();
+}, []);
+```
+
+The `currentUser.subscribe` method listens for changes to the current user's authentication state and updates the `user` state accordingly.
+
+#### Step 3: Define Log In and Log Out Functions
+
+Define the `logIn` and `logOut` functions:
+
+```jsx
+const logIn = () => {
+ fcl.authenticate();
+};
+
+const logOut = () => {
+ fcl.unauthenticate();
+};
+```
+
+The `authenticate` method opens the Discovery UI for the user to log in, while `unauthenticate` logs the user out.
+
+#### Step 4: Define the `incrementCount` Function
+
+Add the `incrementCount` function:
+
+```jsx
+const incrementCount = async () => {
+ try {
+ const transactionId = await fcl.mutate({
+ cadence: `
+ import Counter from 0xf8d6e0586b0a20c7
+
+ transaction {
+
+ prepare(acct: &Account) {
+ // Authorizes the transaction
+ }
+
+ execute {
+ // Increment the counter
+ Counter.increment()
+
+ // Retrieve the new count and log it
+ let newCount = Counter.getCount()
+ log("New count after incrementing: ".concat(newCount.toString()))
+ }
+ }
+ `,
+ proposer: fcl.currentUser,
+ payer: fcl.currentUser,
+ authorizations: [fcl.currentUser.authorization],
+ limit: 50,
+ });
+
+ console.log("Transaction Id", transactionId);
+
+ await fcl.tx(transactionId).onceSealed();
+ console.log("Transaction Sealed");
+
+ queryCount();
+ } catch (error) {
+ console.error("Transaction Failed", error);
+ }
+};
+```
+
+In the above code:
+
+- We define an `async` function `incrementCount` to send a transaction to increment the count in the `Counter` contract.
+- We use the `mutate` method to send a transaction to the blockchain emulator.
+- The transaction increments the count in the `Counter` contract and logs the new count.
+- We use the `proposer`, `payer`, and `authorizations` properties to set the transaction's proposer, payer, and authorizations to the current user.
+- The `limit` property sets the gas limit for the transaction.
+- We log the transaction ID and wait for the transaction to be sealed before querying the updated count.
+- If an error occurs during the transaction, we log it to the console.
+- After the transaction is sealed, we call `queryCount` to fetch and display the updated count.
+- We use the transaction from Step 2 to increment the count in the `Counter` contract.
+
+#### Step 5: Update the Return Statement
+
+Update the `return` statement to include authentication buttons and display the user's address when they're logged in:
+
+```jsx
+return (
+
+ );
+}
```
-You just built an app on Flow!
+Visit `http://localhost:3000` in your browser.
+
+- **Log In**:
+ - Click the "Log In" button.
+ - The Discovery UI will appear, showing the available wallets. Select the "Dev Wallet" option.
+ - Select the account to log in with.
+ - If prompted, create a new account or use an existing one.
+
+- **Increment Count**:
+ - After logging in, you'll see your account address displayed.
+ - Click the "Increment Count" button.
+ - Your wallet will prompt you to approve the transaction.
+ - Approve the transaction to send it to the Flow emulator.
+
+- **View Updated Count**:
+ - Once the transaction is sealed, the app will automatically fetch and display the updated count.
+ - You should see the count incremented on the page, formatted using the `NumberFormatter` contract.
-Run `npm start` again. After a moment, the greeting from `HelloWorld` will appear!
+## Conclusion
-## Mutating Chain State and More
+By following these steps, you've successfully created a simple frontend application using Next.js that interacts with the `Counter` smart contract on the Flow blockchain emulator. You've learned how to:
-For a deeper dive into writing an FCL app, such as how to change the chain state with FCL, check out [the app quickstart guide] or the [FCL documentation].
+- Add the FCL configuration before the rest of your code within the `page.js` file.
+- Configure FCL to work with the local Flow emulator and Dev Wallet.
+- Start the Dev Wallet using `flow dev-wallet` to enable local authentication.
+- Read data from the local blockchain emulator, utilizing multiple contracts (`Counter` and `NumberFormatter`).
+- Authenticate users using the local Dev Wallet.
+- Send transactions to mutate the state of a smart contract on the local emulator.
-
+## Additional Resources
-[Flow Client Library]: ../../tools/clients/fcl-js/index.md
-[Cadence]: https://cadence-lang.org
-[React]: https://react.dev/learn
-[Create React App]: https://create-react-app.dev
-[view the contract here]: https://f.dnz.dev/0xa1296b1e2e90ca5b/HelloWorld
-[the app quickstart guide]: ../guides/flow-app-quickstart.md
-[FCL documentation]: ../../tools/clients/fcl-js/index.md
\ No newline at end of file
+[Flow Client Library]: https://github.com/onflow/fcl-js
+[Cadence]: https://developers.flow.com/cadence
+[Next.js]: https://nextjs.org/docs/getting-started
+[Flow Emulator]: https://developers.flow.com/tools/emulator
+[Flow Dev Wallet]: https://github.com/onflow/fcl-dev-wallet
\ No newline at end of file
diff --git a/docs/build/getting-started/flow-cli.md b/docs/build/getting-started/flow-cli.md
index 7b667444fa..9d2f98ab67 100644
--- a/docs/build/getting-started/flow-cli.md
+++ b/docs/build/getting-started/flow-cli.md
@@ -12,6 +12,7 @@ The [Flow Command Line Interface] (CLI) is a set of tools that developers can us
After completing this guide, you'll be able to:
* Create a Flow project using the [Flow Command Line Interface]
+* Run tests for a smart contract
* Add an already-deployed contract to your project with the [Dependency Manager]
* Deploy a smart contract locally to the Flow Emulator
* Write and execute scripts to interact with a deployed smart contract
@@ -26,24 +27,28 @@ brew install flow-cli
For other ways of installing, please refer to the [installation guide].
-## Configuration
+## Creating a New Project
-Lets first create a project directory and navigate to it:
+To create a new project, navigate to the directory where you want to create your project and run:
```zsh
-mkdir cli-quickstart
-cd cli-quickstart
+flow init
```
-Next, we'll initialize a new Flow project with the CLI:
+Upon running this command, you'll be prompted to enter a project name. Enter a name and press `Enter`.
-```zsh
-flow init --config-only
-```
+You'll also be asked if you'd like to install any core contracts (such as `FungibleToken`, `NonFungibleToken`, etc.) using the [Dependency Manager](../../tools/flow-cli/dependency-manager.md). For this tutorial, you can select `No`.
+
+The `init` command will create a new directory with the project name and the following files:
+
+- `flow.json`: This file contains the configuration for your project.
+- `emulator-account.pkey`: This file contains the private key for the default emulator account.
+- `flow.json`: This file contains the configuration for your project.
+- `cadence/`: This directory contains your Cadence code. Inside there are subdirectories for contracts, scripts, transactions, and tests.
-This will create a `flow.json` file in your project directory. This file is used to configure your project and describe the setup of your contracts, networks, and accounts.
+Inside the `cadence/contracts` directory, you'll find a `Counter.cdc` file. This is the same as the `Counter` contract in the previous step.
-It will also have a default `emulator-account` created for you. We'll use this account to interact with the emulator later on.
+Next, `cd` into your new project directory.
:::info
@@ -51,40 +56,19 @@ For additional details on how `flow.json` is configured, review the [configurati
:::
-## Grabbing the `HelloWorld` Contract
+### Running the Tests
-For this demo, we are going to be interacting with a simple `HelloWorld` contract, written in [Cadence], that is already deployed on Flow's `testnet` network on account [0xa1296b1e2e90ca5b]. In order to grab this project dependency, we'll use Flow's [Dependency Manager] to install it into our project using a source string that defines the network, address, and contract name of the contract we want to import.
+To run the example test for the `Counter` contract located in `cadence/tests`, you can run:
```zsh
-flow dependencies add testnet://0xa1296b1e2e90ca5b.HelloWorld
+flow test
```
-This will add the `HelloWorld` contract and any of its dependencies to an `imports` directory in your project. We recommend adding this directory to your `.gitignore` file, which is done by the script by default. It will also add any dependencies to your `flow.json` file.
-
-During the install you'll be prompted to specify which account to deploy the contracts to. For this tutorial, you can select the default `emulator-account`. Leave the alias address for HelloWorld on mainnet blank.
-
-Review the `π Dependency Manager Actions Summary` for a list of actions completed by the script.
-
-Open `imports/a1296b1e2e90ca5b/HelloWorld.cdc` in your editor. You will see the following:
-
-```cadence
-access(all) contract HelloWorld {
-
- access(all)
- var greeting: String
-
- access(all)
- fun changeGreeting(newGreeting: String) {
- self.greeting = newGreeting
- }
+:::tip
- init() {
- self.greeting = "Hello, World!"
- }
-}
-```
+For a more detailed guide on running Cadence tests, check out the [tests documentation](../../tools/flow-cli/tests.md).
-This contract has a `greeting` variable that can be read and changed. It also has a `changeGreeting` function that allows you to change the greeting.
+:::
## Deploying the Contract to Emulator
@@ -96,50 +80,63 @@ Before we deploy, let's open a new terminal window and run the emulator. From t
flow emulator start
```
-:::warning
+Your emulator should now be running.
-If you see a message that configuration is missing, you are in the wrong directory. Do **not** run `flow init`!.
+### Deploying a Contract
+#### Creating an Account
-> π Configuration is missing, initialize it with: 'flow init' and then rerun this command.
+When you created a project you'll see that a `Counter` contract was added to your `flow.json` configuration file, but it's not set up for deployment yet. We could deploy it to the `emulator-account`, but for this example lets also create a new account on the emulator to deploy it to.
-:::
+With your emulator running, run the following command:
+
+```zsh
+flow accounts create
+```
+
+When prompted, give your account the name `test-account` and select `Emulator` as the network. You'll now see this account in your `flow.json`.
-To deploy the `HelloWorld` contract to the emulator, return to your first terminal and run the following command:
+> Note: We won't use this much in this example, but it's good to know how to create an account.
+
+#### Configuring the Deployment
+
+To deploy the `Counter` contract to the emulator, you'll need to add it to your project configuration. You can do this by running:
```zsh
-flow project deploy
+flow config add deployment
```
-You should see:
+You'll be prompted to select the contract you want to deploy. Select `Counter` and then select the account you want to deploy it to. For this example, select `emulator-account`.
+
+#### Deploying the Contract
+
+To deploy the `Counter` contract to the emulator, run:
```zsh
-π All contracts deployed successfully
+flow project deploy
```
-The contract will now have been deployed to the default `emulator-account`. You can now interact with it using a script.
+That's it! You've just deployed your first contract to the Flow Emulator.
## Running Scripts
Scripts are used to read data from the Flow blockchain. There is no state modification. In our case, we are going to read a greeting from the `HelloWorld` contract.
-Let's create a script file. We can generate a boilerplate script file with the following command:
+If we wanted to generate a new script, we could run:
```zsh
-flow generate script ReadGreeting
+flow generate script ScriptName
```
-This will create a file called `ReadGreeting.cdc` in the `cadence/scripts` directory. Let's update the script to read the greeting from the `HelloWorld` contract. Replace the existing coded with:
+But the default project already has a `GetCounter` script for reading the count of the `Counter` contract. Open `cadence/scripts/GetCounter.cdc` in your editor to see the script.
-```cadence
-import "HelloWorld"
+To run the script, you can run:
-access(all) fun main(): String {
- return HelloWorld.greeting
-}
+```zsh
+flow scripts execute cadence/scripts/GetCounter.cdc
```
-The import syntax will automatically resolve the address of the contract on the network you are running the script on. This is determined by your `flow.json` configuration.
+You should see zero as the result since the `Counter` contract initializes the count to zero and we haven't run any transactions to increment it.
:::tip
@@ -147,42 +144,17 @@ If you'll like to learn more about writing scripts, please check out the docs fo
:::
-To run the script, we'll run this from the CLI:
-
-```zsh
-flow scripts execute cadence/scripts/ReadGreeting.cdc
-```
-
-You should see the result of the greeting. `Result: "Hello, world!"`
-
-## Creating an Account and Running a Transaction
+## Executing Transactions
-To change state on the Flow Blockchain, you need to run a transaction. Let's create a simple transaction file. We can use to modify the `greeting` on the `HelloWorld` contract.
+Transactions are used to modify the state of the blockchain. In our case, we want to increment the count of the `Counter` contract. Luckily, we already have a transaction for that in the project that was generated for us. Open `cadence/transactions/IncrementCounter.cdc` in your editor to see the transaction.
-First, create a file called `cadence/transactions/ChangeGreeting.cdc` with the following command:
+To run the transaction, you can run:
```zsh
-flow generate transaction ChangeGreeting
-```
-
-Open the new file - `cadence/transactions/ChangeGreeting.cdc`. Update the boilerplate transaction to look like this:
-
-```cadence
-import "HelloWorld"
-
-transaction(greeting: String) {
-
- prepare(acct: &Account) {
- log(acct.address)
- }
-
- execute {
- HelloWorld.changeGreeting(newGreeting: greeting)
- }
-}
+flow transactions send cadence/transactions/IncrementCounter.cdc
```
-This will log the account signing the transaction, call the `changeGreeting` method of the `HelloWorld` contract, and pass in the new greeting.
+By default, this uses the `emulator-account` to sign the transaction and the emulator network. If you want to use your `test-account` account, you can specify the `--signer` flag with the account name.
:::tip
@@ -190,51 +162,63 @@ If you want to learn more about writing transactions, please read the docs for [
:::
-In order to run a transaction, the signing account needs to pay for it. You could run the transaction on emulator using the default `emulator-account` account, but a better test is to run it with a new test account.
+## Installing & Interacting With External Dependencies
+
+In addition to creating your own contracts, you can also install contracts that have already been deployed to the network by using the [Dependency Manager]. This is useful for interacting with contracts that are part of the Flow ecosystem or that have been deployed by other developers.
-Let's learn the command for creating accounts.
+For example, let's say we want to format the result of our `GetCounter` script so that we display the number with commas if it's greater than 999. To do that we can install a contract called [`NumberFormatter`](https://contractbrowser.com/A.8a4dce54554b225d.NumberFormatter) from `testnet` that has a function to format numbers.
-The easiest way to create an account using CLI is with:
+To grab it, run:
```zsh
-flow accounts create
+flow dependencies add testnet://8a4dce54554b225d.NumberFormatter
```
-Remember, your emulator should still be running at this point in another terminal.
+When prompted for the account to deploy the contract to, select any account and ignore the prompt for an alias. This is if you wanted to configure a `mainnet` address for the contract.
+
+This will add the `NumberFormatter` contract and any of its dependencies to an `imports` directory in your project. It will also add any dependencies to your `flow.json` file. In addition, the prompt will configure the deployment of the contract to the account you selected. Make sure to select the `emulator-account` account to deploy the contract to the emulator.
-Give your account the name `emulator-tester`, then select `Emulator` as the network. You'll now see this account in your `flow.json`.
+You should then see the `NumberFormatter` in your deployments for emulator in your `flow.json`. If you messed this up, you can always run `flow config add deployment` to add the contract to your deployments.
-To run a transaction with this new account, you can run the following:
+Now we can deploy the `NumberFormatter` contract to the emulator by running:
```zsh
-flow transactions send cadence/transactions/ChangeGreeting.cdc "Hello, me" --signer emulator-tester --network emulator
+flow project deploy
```
-You've just modified the state of the Flow Blockchain! At least on the emulator. You'll know it worked if you see the receipt. Yours will be similar to:
+Now that we have the `NumberFormatter` contract deployed, we can update our `GetCounter` script to format the result. Open `cadence/scripts/GetCounter.cdc` and update it to use the following code:
-```zsh
-Transaction ID: 2ff6cbb8125103595fca0abaead94cd00510d29902ceae9f5dc480e927ab7334
-
-Block ID 36bbf6fc573129fa9a3c78a43e257d3b627a3af78fd9e64eeb133d981819cc69
-Block Height 3
-Status β SEALED
-ID 2ff6cbb8125103595fca0abaead94cd00510d29902ceae9f5dc480e927ab7334
-Payer 179b6b1cb6755e31
-Authorizers [179b6b1cb6755e31]
-```
+```cadence
+import "Counter"
+import "NumberFormatter"
-You can also re-run the `ReadGreeting` script with:
+access(all)
+fun main(): String {
+ // Retrieve the count from the Counter contract
+ let count: Int = Counter.getCount()
-```zsh
-flow scripts execute cadence/scripts/ReadGreeting.cdc
+ // Format the count using NumberFormatter
+ let formattedCount = NumberFormatter.formatWithCommas(number: count)
+
+ // Return the formatted count
+ return formattedCount
+}
```
-You'll now see:
+The things to note here are:
+
+- We import the `NumberFormatter` contract.
+- We call the `formatWithCommas` function from the `NumberFormatter` contract to format the count.
+- We return the formatted count as a `String`.
+
+Now, to run the updated script, you can run:
```zsh
-Result: "Hello, me"
+flow scripts execute cadence/scripts/GetCounter.cdc
```
+You should now see the result. You won't see the commas unless the number is greater than 999.
+
## More
If you want to continue on generating your own contracts, you can also use the the `generate` subcommand to create a new contract file. See more in the [`generate` documentation].
diff --git a/docs/build/getting-started/hello-world.md b/docs/build/getting-started/hello-world.md
deleted file mode 100644
index 0bbb9c62b3..0000000000
--- a/docs/build/getting-started/hello-world.md
+++ /dev/null
@@ -1,107 +0,0 @@
----
-sidebar_position: 1
-sidebar_label: Hello World
----
-import VerticalSplit from "./vertical-split.svg"
-
-# Hello World
-
-In this quickstart guide, you'll interact with your first smart contract on the Flow Testnet. `Testnet` is a public instance of the Flow blockchain designed for experimentation, where you can deploy and invoke smart contracts without incurring any real-world costs.
-
-Smart contracts on Flow are permanent pieces of code that live on the blockchain. They allow you to encode business logic, define digital assets, and much more. By leveraging smart contracts, you can create decentralized applications (dApps) that are transparent, secure, and open to anyone.
-
-Flow supports modern smart contracts written in [Cadence], a resource-oriented programming language designed specifically for smart contracts. Cadence focuses on safety and security, making it easier to write robust contracts. Flow also supports traditional [EVM]-compatible smart contracts written in Solidity, allowing developers to port their existing Ethereum contracts to Flow. In this guide, we'll focus on interacting with Cadence smart contracts.
-
-## Objectives
-
-After completing this guide, you'll be able to:
-
-* Read a public variable on a [Cadence] smart contract deployed on Flow.
-* Understand how to interact with contracts on Flow's `testnet`.
-* Retrieve and display data from a deployed smart contract via scripts.
-
-In later steps, you'll learn how to:
-
-* Create a Flow project using the [Flow CLI](../../tools/flow-cli/index.md).
-* Add an already-deployed contract to your project with the [Dependency Manager](../../tools/flow-cli/dependency-manager.md).
-* Deploy a smart contract locally to the [Flow Emulator](../../tools/emulator/index.md).
-* Write and execute transactions to interact with a deployed smart contract.
-* Display data from a Cadence smart contract on a React frontend using the [Flow Client Library](../../tools/clients/fcl-js/index.md).
-
-
-## Calling a Contract With a Script
-
-The `HelloWorld` contract exposes a public variable named `greeting` that is accessible to everything outside the contract. We can retrieve its value using a simple script written in the [Cadence] programming language. Scripts in Cadence are read-only operations that allow you to query data from the blockchain without changing any state.
-
-Here's the script:
-
-```cadence
-import HelloWorld from 0xa1296b1e2e90ca5b
-
-access(all) fun main(): String {
- return HelloWorld.greeting
-}
-```
-
-Let's break down what this script does:
-
-- **Import Statement**: `import HelloWorld from 0xa1296b1e2e90ca5b` tells the script to use the `HelloWorld` contract deployed at the address `0xa1296b1e2e90ca5b` on the `testnet`.
-- **Main Function**: `access(all) fun main(): String` defines the entry point of the script, which returns a `String`.
-- **Return Statement**: `return HelloWorld.greeting` accesses the greeting variable from the `HelloWorld` contract and returns its value.
-
-### Steps to Execute the Script
-
-- **Run the Script**: Click the Run button to execute the script.
-- **View the Output**: Observe the output returned by the script. You should see the current value of the `greeting` variable, which is `"Hello, World!"`.
-
-
-
-## Understanding the `HelloWorld` Contract
-
-To fully grasp how the script works, it's important to understand the structure of the `HelloWorld` contract. Below is the source code for the contract:
-
-```cadence
-access(all) contract HelloWorld {
-
- access(all) var greeting: String
-
- access(account) fun changeGreeting(newGreeting: String) {
- self.greeting = newGreeting
- }
-
- init() {
- self.greeting = "Hello, World!"
- }
-}
-```
-
-### Breakdown of the Contract
-
-- **Contract Declaration**: `access(all) contract HelloWorld` declares a new contract named `HelloWorld` that is accessible to everyone.
-- **State Variable**: `access(all) var greeting: String` declares a public variable `greeting` of type `String`. The `access(all)` modifier means that this variable can be read by anyone.
-- **Function to Change Greeting**: `access(account) fun changeGreeting(newGreeting: String)` defines a function that allows changing the value of `greeting`. The `access(account)` modifier restricts this function so that only the account that deployed the contract (the owner) can call it.
-- **Initializer**: The `init()` function is called when the contract is deployed. It sets the initial value of `greeting` to `"Hello, World!"`.
-
-### Key Points
-
-- **Public Access**: The `greeting` variable is public, allowing anyone to read its value without any restrictions.
-- **Restricted Modification**: Only the contract owner can modify the `greeting` variable using the `changeGreeting` function. This ensures that unauthorized accounts cannot change the contract's state.
-- **No Read Costs**: Reading data from the blockchain is free on Flow. Executing scripts like the one you ran does not incur any costs.
-
-### What's Next?
-
-In the upcoming tutorials, you'll learn how to:
-
-- **Modify the Greeting**: Invoke the `changeGreeting` function to update the `greeting` value.
-- **Deploy Contracts**: Use the Flow CLI to deploy your own smart contracts.
-- **Interact with Contracts Locally**: Use the Flow Emulator to test contracts in a local development environment.
-- **Build Frontend Applications**: Display data from smart contracts in a React application using the Flow Client Library.
-
-By understanding the `HelloWorld` contract and how to interact with it, you're building a solid foundation for developing more complex applications on the Flow blockchain.
-
-Proceed to the next tutorial to learn how to create your own contracts and deploy them live using the Flow CLI.
-
-
-
-[Cadence]: https://cadence-lang.org/
-[EVM]: https://flow.com/upgrade/crescendo/evm
\ No newline at end of file
diff --git a/docs/build/guides/account-linking-with-dapper.md b/docs/build/guides/account-linking-with-dapper.md
new file mode 100644
index 0000000000..89b8f79ac9
--- /dev/null
+++ b/docs/build/guides/account-linking-with-dapper.md
@@ -0,0 +1,752 @@
+---
+title: Account Linking With NBA Top Shot
+description: Use Account Linking between the Dapper Wallet and Flow Wallet to effortlessly use NBA Top Shot Moments in your app.
+sidebar_position: 5
+sidebar_custom_props:
+ icon: βοΈ
+---
+
+# Account Linking With NBA Top Shot
+
+[Account Linking] is a powerful Flow feature that allows users to connect their wallets, enabling linked wallets to view and manage assets in one wallet with another. This feature helps reduce or even eliminate the challenges posed by other account abstraction solutions, which often lead to multiple isolated wallets and fragmented assets.
+
+In this tutorial, you'll build a [simple onchain app] that allows users to sign into your app with their Flow wallet and view [NBA Top Shot] Moments that reside in their [Dapper Wallet] - without those users needing to sign in with Dapper.
+
+## Objectives
+
+After completing this guide, you'll be able to:
+
+* Pull your users' NBA Top Shot Moments into your Flow app without needing to transfer them out of their Dapper wallet
+* Retrieve and list all NFT collections in any child wallet linked to a given Flow address
+* Write a [Cadence] script to iterate through the storage of a Flow wallet to find NFT collections
+* Run Cadence Scripts from the frontend
+
+## Prerequisites
+
+### Next.js and Modern Frontend Development
+
+This tutorial uses [Next.js]. You don't need to be an expert, but it's helpful to be comfortable with development using a current React framework. You'll be on your own to select and use a package manager, manage Node versions, and other frontend environment tasks. If you don't have your own preference, you can just follow along with us and use [Yarn].
+
+### Flow Wallet
+
+You'll need a [Flow Wallet], but you don't need to deposit any funds.
+
+## Moments NFTs
+
+You'll need a [Dapper Wallet] containing some Moments NFTs, such as [NBA Top Shot] Moments.
+
+## Getting Started
+
+This tutorial will use a [Next.js] project as the foundation of the frontend. Create a new project with:
+
+```zsh
+npx create-next-app@latest
+```
+
+We will be using TypeScript and the App Router, in this tutorial.
+
+Open your new project in the editor of your choice, install dependencies, and run the project.
+
+```zsh
+yarn install
+yarn run dev
+```
+
+If everything is working properly, you'll be able to navigate to `localhost:3000` and see the default [Next.js] page.
+
+## Flow Cadence Setup
+
+You'll need a few more dependencies to efficiently work with Cadence inside of your app.
+
+### Flow CLI and Types
+
+The [Flow CLI] contains a number of command-line tools for interacting with the Flow ecosystem. If you don't already have it installed, you can add it with Brew (or using [other installation methods]):
+
+```zsh
+brew install flow-cli
+```
+
+Once it's installed, you'll need to initialize Flow in your Next.js project. From the root, run:
+
+```zsh
+flow init --config-only
+```
+
+The `--config-only` flag [initializes a project] with the just the config file. This allows the Flow CLI to interact with your project without adding any unnecessary files.
+
+Next, you'll need to do a little bit of config work so that your project knows how to read Cadence files. Install the Flow Cadence Plugin:
+
+```zsh
+yarn add flow-cadence-plugin --dev
+```
+
+Finally, open `next.config.ts` and update it to use the plugin with Raw Loader:
+
+```tsx
+// next.config.ts
+import type { NextConfig } from "next";
+import FlowCadencePlugin from "flow-cadence-plugin";
+
+const nextConfig: NextConfig = {
+ webpack: (config) => {
+ config.plugins.push(new FlowCadencePlugin())
+
+ return config;
+ },
+};
+
+export default nextConfig;
+```
+
+## Frontend Setup
+
+We'll use the Flow Client Library [FCL] to manage blockchain interaction from the frontend. It's similar to viem, ethers, or web3.js, but works with the Flow blockchain and transactions and scripts written in Cadence.
+
+```zsh
+yarn add @onflow/fcl
+```
+
+Go ahead and install `dotenv` as well:
+
+```
+yarn add dotenv
+```
+
+### Provider Setup
+
+A fair amount of boilerplate code is needed to set up your provider. We'll provide it, but since it's not the purpose of this tutorial, we'll be brief on explanations. For more details, check out the [App Quickstart Guide].
+
+Add `app/providers/AuthProvider.tsx`:
+
+```tsx
+'use client';
+/* eslint-disable @typescript-eslint/no-explicit-any */
+
+import { createContext, useContext, ReactNode } from 'react';
+import useCurrentUser from '../hooks/use-current-user.hook';
+
+interface State {
+ user: any;
+ loggedIn: any;
+ logIn: any;
+ logOut: any;
+}
+
+const AuthContext = createContext(undefined);
+
+interface AuthProviderProps {
+ children: ReactNode;
+}
+
+const AuthProvider: React.FC = ({ children }) => {
+ const [user, loggedIn, logIn, logOut] = useCurrentUser();
+
+ return (
+
+ {children}
+
+ );
+};
+
+export default AuthProvider;
+
+export const useAuth = (): State => {
+ const context = useContext(AuthContext);
+
+ if (context === undefined) {
+ throw new Error('useAuth must be used within a AuthProvider');
+ }
+
+ return context;
+};
+```
+
+Then, add `app/hooks/use-current-user-hook.tsx`:
+
+```tsx
+import { useEffect, useState } from 'react';
+import * as fcl from '@onflow/fcl';
+
+export default function useCurrentUser() {
+ const [user, setUser] = useState({ addr: null });
+
+ const logIn = () => {
+ fcl.authenticate();
+ };
+
+ const logOut = () => {
+ fcl.unauthenticate();
+ };
+
+ useEffect(() => {
+ fcl.currentUser().subscribe(setUser);
+ }, []);
+
+ return {user, loggedIn: user?.addr != null, logIn, logOut};
+}
+```
+
+## .env
+
+Add a `.env` to the root and fill it with:
+
+```text
+NEXT_PUBLIC_ACCESS_NODE_API="https://rest-mainnet.onflow.org"
+NEXT_PUBLIC_FLOW_NETWORK="mainnet"
+NEXT_PUBLIC_WALLETCONNECT_ID=
+```
+
+:::warning
+
+Don't forget to replace `` with your own [Wallet Connect] app id!
+
+:::
+
+### Implement the Provider and Flow Config
+
+Finally, open `layout.tsx`. Start by importing Flow dependencies and the AuthProvider:
+
+```tsx
+import flowJSON from '../flow.json'
+import * as fcl from "@onflow/fcl";
+
+import AuthProvider from "./providers/AuthProvider";
+```
+
+Then add your Flow config:
+
+```tsx
+fcl.config({
+ "discovery.wallet": "https://fcl-discovery.onflow.org/authn",
+ 'accessNode.api': process.env.NEXT_PUBLIC_ACCESS_NODE_API,
+ 'flow.network': process.env.NEXT_PUBLIC_FLOW_NETWORK,
+ 'walletconnect.projectId': process.env.NEXT_PUBLIC_WALLETCONNECT_ID
+}).load({ flowJSON });
+```
+
+:::warning
+
+We're going to force some things client side to get this proof-of-concept working quickly. Use Next.js best practices for a production app.
+
+:::
+
+Add a `'use client';` directive to the top of the file and **delete** the import for Metadata and fonts, as well as the code related to them.
+
+Finally, update the `` to remove the font references and suppress hydration warnings:
+
+```tsx
+
+```
+
+Your code should be:
+
+```tsx
+// layout.tsx
+'use client';
+import "./globals.css";
+import flowJSON from '../flow.json'
+import * as fcl from "@onflow/fcl";
+
+import AuthProvider from "./providers/AuthProvider";
+
+fcl.config({
+ "discovery.wallet": "https://fcl-discovery.onflow.org/authn",
+ 'accessNode.api': process.env.NEXT_PUBLIC_ACCESS_NODE_API,
+ 'flow.network': process.env.NEXT_PUBLIC_FLOW_NETWORK,
+ 'walletconnect.projectId': process.env.NEXT_PUBLIC_WALLETCONNECT_ID
+}).load({ flowJSON });
+
+export default function RootLayout({
+ children,
+}: {
+ children: React.ReactNode;
+}) {
+ return (
+
+
+
+ {children}
+
+
+
+ );
+}
+```
+
+### Add the Connect Button
+
+Open `page.tsx` and clean up the demo code leaving only the `` block:
+
+```tsx
+import Image from "next/image";
+
+export default function Home() {
+ return (
+
+
+
TODO
+
+
+ );
+}
+```
+
+Add a `'use client';` directive, import the `useAuth` hook and instantiate it in the `Home` function:
+
+```tsx
+'use client';
+import { useAuth } from "./providers/AuthProvider";
+```
+
+```tsx
+const { user, loggedIn, logIn, logOut } = useAuth();
+```
+
+Then add a button in the `` to handle logging in or out:
+
+```tsx
+
+
Welcome
+
+
+```
+
+## Testing Pass
+
+Run the app:
+
+```zsh
+yarn dev
+```
+
+You'll see your `Log In` button in the middle of the window.
+
+![Welcome](welcome.png)
+
+Click the button and log in with your Flow wallet.
+
+![Flow Wallet](flow-wallet.png)
+
+## Account Linking
+
+Now that your app is set up, you can make use of [Account Linking] to to pull your NFTs from your Dapper Wallet, through your Flow Wallet, and into the app.
+
+### Setting Up Account Linking
+
+If you haven't yet, you'll need to [link your Dapper Wallet] to your Flow Wallet.
+
+:::warning
+
+The Dapper Wallet requires that you complete KYC before you can use Account Linking. While this may frustrate some members of the community, it makes it much easier for app developers to design onboarding rewards and bonuses that are less farmable.
+
+:::
+
+### Discovering the NFTs with a Script
+
+With your accounts linked, your Flow Wallet now has a set of capabilities related to your Dapper Wallet and it's permitted to use those to view and even manipulate those NFTs and assets.
+
+Before you can add a script that can handle this, you'll need to import the `HybridCustody` contract using the [Flow Dependency Manager]:
+
+```zsh
+flow dependencies add mainnet://d8a7e05a7ac670c0.HybridCustody
+```
+
+Choose `none` to skip deploying on the `emulator` and skip adding testnet aliases. There's no point, these NFTs are on mainnet!
+
+You'll get a complete summary from the Dependency Manager:
+
+```zsh
+π Dependency Manager Actions Summary
+
+ποΈ File System Actions:
+β οΈ Contract HybridCustody from d8a7e05a7ac670c0 on mainnet installed
+β οΈ Contract MetadataViews from 1d7e57aa55817448 on mainnet installed
+β οΈ Contract FungibleToken from f233dcee88fe0abe on mainnet installed
+β οΈ Contract ViewResolver from 1d7e57aa55817448 on mainnet installed
+β οΈ Contract Burner from f233dcee88fe0abe on mainnet installed
+β οΈ Contract NonFungibleToken from 1d7e57aa55817448 on mainnet installed
+β οΈ Contract CapabilityFactory from d8a7e05a7ac670c0 on mainnet installed
+β οΈ Contract CapabilityDelegator from d8a7e05a7ac670c0 on mainnet installed
+β οΈ Contract CapabilityFilter from d8a7e05a7ac670c0 on mainnet installed
+
+πΎ State Updates:
+β HybridCustody added to emulator deployments
+β Alias added for HybridCustody on mainnet
+β HybridCustody added to flow.json
+β MetadataViews added to flow.json
+β FungibleToken added to flow.json
+β ViewResolver added to flow.json
+β Burner added to flow.json
+β NonFungibleToken added to flow.json
+β CapabilityFactory added to emulator deployments
+β Alias added for CapabilityFactory on mainnet
+β CapabilityFactory added to flow.json
+β CapabilityDelegator added to emulator deployments
+β Alias added for CapabilityDelegator on mainnet
+β CapabilityDelegator added to flow.json
+β CapabilityFilter added to emulator deployments
+β Alias added for CapabilityFilter on mainnet
+β CapabilityFilter added to flow.json
+```
+
+Add `app/cadence/scripts/FetchNFTsFromLinkedAccts.cdc`. In it, add this script. Review the inline comments to see what each step is doing:
+
+```cadence
+import "HybridCustody"
+import "NonFungibleToken"
+import "MetadataViews"
+
+// This script iterates through a parent's child accounts,
+// identifies private paths with an accessible NonFungibleToken.Provider, and returns the corresponding typeIds
+
+access(all) fun main(addr: Address): AnyStruct {
+ let manager = getAuthAccount(addr).storage.borrow(from: HybridCustody.ManagerStoragePath)
+ ?? panic ("manager does not exist")
+
+ var typeIdsWithProvider: {Address: [String]} = {}
+ var nftViews: {Address: {UInt64: MetadataViews.Display}} = {}
+
+ let providerType = Type()
+ let collectionType: Type = Type<@{NonFungibleToken.CollectionPublic}>()
+
+ for address in manager.getChildAddresses() {
+ let acct = getAuthAccount(address)
+ let foundTypes: [String] = []
+ let views: {UInt64: MetadataViews.Display} = {}
+ let childAcct = manager.borrowAccount(addr: address) ?? panic("child account not found")
+
+ // Iterate through storage paths to find NFTs that are controlled by the parent account
+ // To just find NFTs, check if thing stored is nft collection and borrow it as NFT collection and get IDs
+ for s in acct.storage.storagePaths {
+ // Iterate through capabilities
+ for c in acct.capabilities.storage.getControllers(forPath: s) {
+ if !c.borrowType.isSubtype(of: providerType){
+ // If this doen't have providerType, it's not an NFT collection
+ continue
+ }
+
+ // We're dealing with a Collection but we need to check if accessible from the parent account
+ if let cap: Capability = childAcct.getCapability(controllerID: c.capabilityID, type: providerType) { // Part 1
+ let providerCap = cap as! Capability<&{NonFungibleToken.Provider}>
+
+ if !providerCap.check(){
+ // If I don't have access to control the account, skip it.
+ // Disable this check to do something else.
+ //
+ continue
+ }
+
+ foundTypes.append(cap.borrow<&AnyResource>()!.getType().identifier)
+ typeIdsWithProvider[address] = foundTypes
+ // Don't need to keep looking at capabilities, we can control NFT from parent account
+ break
+ }
+ }
+ }
+
+ // Iterate storage, check if typeIdsWithProvider contains the typeId, if so, add to views
+ acct.storage.forEachStored(fun (path: StoragePath, type: Type): Bool {
+
+ if typeIdsWithProvider[address] == nil {
+ return true
+ }
+
+ for key in typeIdsWithProvider.keys {
+ for idx, value in typeIdsWithProvider[key]! {
+ let value = typeIdsWithProvider[key]!
+
+ if value[idx] != type.identifier {
+ continue
+ } else {
+ if type.isInstance(collectionType) {
+ continue
+ }
+ if let collection = acct.storage.borrow<&{NonFungibleToken.CollectionPublic}>(from: path) {
+ // Iterate over IDs & resolve the Display view
+ for id in collection.getIDs() {
+ let nft = collection.borrowNFT(id)!
+ if let display = nft.resolveView(Type())! as? MetadataViews.Display {
+ views.insert(key: id, display)
+ }
+ }
+ }
+ continue
+ }
+ }
+ }
+ return true
+ })
+ nftViews[address] = views
+ }
+ return nftViews
+}
+```
+
+:::warning
+
+The above script is a relatively naive implementation. For production, you'll want to filter for only the collections you care about, and you will eventually need to add handling for very large collections in a wallet.
+
+:::
+
+### Running the Script and Displaying the NFTs
+
+Add a component in `app/components` called `DisplayLinkedNFTs.cdc`.
+
+In it, import dependencies from React and FCL, as well as the script you just added:
+
+```tsx
+import React, { useState, useEffect } from 'react';
+import * as fcl from "@onflow/fcl";
+import * as t from '@onflow/types';
+
+import FetchNFTs from '../cadence/scripts/FetchNFTsFromLinkedAccts.cdc';
+```
+
+As we're using TypeScript, you should add some types as well to manage the data from the NFTs nicely. For now, just add them to this file:
+
+```typescript
+type Thumbnail = {
+ url: string;
+};
+
+type Moment = {
+ name: string;
+ description: string;
+ thumbnail: Thumbnail;
+};
+
+type MomentsData = {
+ [momentId: string]: Moment;
+};
+
+type ApiResponse = {
+ [address: string]: MomentsData;
+};
+
+interface AddressDisplayProps {
+ address: string;
+}
+```
+
+Then, add the function for the component:
+
+```tsx
+const DisplayLinkedNFTs: React.FC = ({ address }) => {
+ // TODO...
+
+ return (
+
Nothing here yet
+ )
+}
+
+export default DisplayLinkedNFTs;
+```
+
+In the function, add a state variable to store the data retrieved by the script:
+
+```typescript
+const [responseData, setResponseData] = useState(null);
+```
+
+Then, use `useEffect` to fetch the NFTs with the script and `fcl.query`:
+
+```tsx
+useEffect(() => {
+ const fetchLinkedAddresses = async () => {
+ if (!address) return;
+
+ try {
+ const cadenceScript = FetchNFTs;
+
+ // Fetch the linked addresses
+ const response: ApiResponse = await fcl.query({
+ cadence: cadenceScript,
+ args: () => [fcl.arg(address, t.Address)],
+ });
+
+ console.log(JSON.stringify(response, null, 2));
+
+ setResponseData(response);
+ } catch (error) {
+ console.error("Error fetching linked addresses:", error);
+ }
+ };
+
+ fetchLinkedAddresses();
+}, [address]);
+```
+
+Return to `page.tsx`, import your new component, and add an instance of `` that passes in the user's address and is only displayed while `loggedIn`.
+
+```tsx
+{loggedIn && }
+```
+
+### Testing
+
+Run the app again. If you have linked your account and have NFTs in that account, you'll see them in the console!
+
+### Displaying the Moments
+
+Now that they're here, all to do is display them nicely! Return to `DisplayLinkedNFTs.tsx`. Add a helper function to confirm each returned NFT matches the Moments format. You can update this to handle other NFTs you'd like to show as well.
+
+:::warning
+
+Remember, you'll also need to update the script in a production app to filter for only the collections you want, and handle large collections.
+
+:::
+
+```tsx
+// Type-checking function to validate moment structure
+// eslint-disable-next-line @typescript-eslint/no-explicit-any
+const isValidMoment = (moment: any): moment is Moment => {
+ const isValid =
+ typeof moment.name === 'string' &&
+ typeof moment.description === 'string' &&
+ moment.thumbnail &&
+ typeof moment.thumbnail.url === 'string';
+
+ if (!isValid) {
+ console.warn('Invalid moment data:', moment);
+ }
+
+ return isValid;
+};
+```
+
+Next, add a rendering function with some basic styling:
+
+```tsx
+// Function to render moments with validation
+const renderMoments = (data: ApiResponse) => {
+ return Object.entries(data).map(([addr, moments]) => (
+
+);
+```
+
+### Further Polish
+
+Finally, you can polish up your `page.tsx` to look a little nicer, and guide your users to the Account Linking process in the Dapper Wallet:
+
+```tsx
+'use client';
+import DisplayLinkedNFTs from "./components/DisplayLinkedNFTs";
+import { useAuth } from "./providers/AuthProvider";
+
+export default function Home() {
+ const { user, loggedIn, logIn, logOut } = useAuth();
+
+ return (
+
+
+ {/* Message visible for all users */}
+
+ Please link your Dapper wallet to view your NFTs. For more information, check the{" "}
+
+ Account Linking and FAQ
+ .
+
+
+
+ {/* Display user address or linked NFTs if logged in */}
+ {loggedIn ? (
+
+ Address: {user.addr}
+
+ ) : (
+
+ Please log in to view your linked NFTs.
+
+ )}
+
+ {/* Login/Logout Button */}
+
+
+
+ {/* Display NFTs if logged in */}
+ {loggedIn && }
+
+
+ );
+}
+```
+
+Your app will now look like the [simple onchain app] demo!
+
+## Conclusion
+
+In this tutorial, you took your first steps towards building powerful new experiences that meet you customers where they are. They can keep their assets in the wallet associate with one app, but also give your app the ability to use them - seamlessly, safely, and beautifully!
+
+[Account Linking]: ./account-linking/index.md
+[NBA Top Shot]: https://nbatopshot.com
+[simple onchain app]: https://nextjs-topshot-account-linking.vercel.app
+[Dapper Wallet]: https://meetdapper.com
+[Cadence]: https://cadence-lang.org/docs
+[Next.js]: https://nextjs.org/docs/app/getting-started/installation
+[Yarn]: https://yarnpkg.com
+[Flow CLI]: ../../tools/flow-cli/index.md
+[other installation methods]: ../../tools/flow-cli/install.md
+[initializes a project]: ../../tools/flow-cli/super-commands.md#init
+[Flow Dependency Manager]: ../../tools/flow-cli/dependency-manager.md
+[FCL]: ../../tools/clients/fcl-js/index.md
+[App Quickstart Guide]: ../getting-started/fcl-quickstart.md
+[Wallet Connect]: https://cloud.walletconnect.com/sign-in
+[Flow Wallet]: https://wallet.flow.com
+[link your Dapper Wallet]: https://support.meetdapper.com/hc/en-us/articles/20744347884819-Account-Linking-and-FAQ
\ No newline at end of file
diff --git a/docs/build/guides/flow-app-quickstart.md b/docs/build/guides/flow-app-quickstart.md
deleted file mode 100644
index 80cb9fa5d5..0000000000
--- a/docs/build/guides/flow-app-quickstart.md
+++ /dev/null
@@ -1,407 +0,0 @@
----
-description: Building an application on Flow
-sidebar_position: 4
-sidebar_custom_props:
- icon: ποΈ
----
-
-# Create a Web App on Flow
-
----
-
-This guide is a simple walkthrough to get started building a web3 app using the Flow Client Library (FCL).
-
-:::info
-
-If you are looking for a scaffolds, check out [scaffolds available in the Flow CLI](../../tools/flow-cli/super-commands.md#using-scaffolds).
-
-:::
-
-This simple guide uses the "Hello World" scaffold, it is meant to provide all the code needed to get a web application up and running.
-
-## Introduction
-
-![Flow app anatomy](flow-dapp-anatomy.png)
-
-:::info
-
-FCL (aka Flow Client Library) wraps the logic needed to communicate with the Flow blockchain. It's a npm package, [More Information](../../tools/clients/fcl-js)
-
-:::
-
-This guide assumes a good understanding of React. The concepts are easy to understand and apply to other libraries and framework. A strong understanding of Cadence (Flow's smart contract language) is not required. More information on Cadence, [learning the Cadence language](https://cadence-lang.org/docs/language/).
-
-### FCL concepts covered:
-
-- [Installation](#installation)
-- [Configuration](#configuration)
-- [Authenticate a user](#authenticate-a-user)
-- [Deploy contracts](#deploy-contracts)
-- [Query the Blockchain](#query-the-blockchain)
-- [Mutate the Blockchain](#mutate-the-blockchain)
-
-For more help, [Discord](https://discord.com/invite/J6fFnh2xx6). See links at the end of this article for diving deeper into building on Flow.
-
-## Installation
-
-Make sure you have Flow CLI installed. [installation instructions](../../tools/flow-cli/install.md).
-
-## Configuration
-
-### Setting up Flow
-
-Using the Flow CLI scaffold, create a basic hello world web project with `Web Scaffolds` -> `[5] FCL Web Dapp`. This will create a new `flow.json` file in the `hello-world` folder. This file will contain the Flow configuration for your project.
-
-```sh
-flow init hello-world --scaffold
-# select scaffold 5
-cd hello-world
-npm install
-# run the app in a later step
-```
-
-We don't recommend keeping private keys in your `flow.json`, notice that Flow CLI already moved the emulator private key to a `emulator.key` file and point to it using the [key/location pattern](../../tools/flow-cli/flow.json/security.md#private-account-configuration-file). This file should be added to your `.gitignore` file, so it won't be committed to your repository.
-
-We won't be using emulator and running contracts locally in this quickstart, but FCL will complain if it finds private keys in your `flow.json` file.
-
-Your `flow.json` file should look like this:
-
-```
-{
- "networks": {
- "emulator": "127.0.0.1:3569",
- "mainnet": "access.mainnet.nodes.onflow.org:9000",
- "testnet": "access.devnet.nodes.onflow.org:9000"
- },
- "accounts": {
- "emulator-account": {
- "address": "f8d6e0586b0a20c7",
- "key": {
- "type": "file",
- "location": "./emulator.key"
- }
- }
- }
-}
-```
-
-:::info
-
-The `flow.json` file is used to keep track of deployed contracts and accounts. [More Information](../../tools/clients/fcl-js/api#using-flowjson)
-
-:::
-
-### Configuring FCL
-
-Next, notice `@onflow/fcl` has been added to the `package.json` and the web application is ready to be run.
-
-:::info
-
-There are a lot of benefits to getting familiar with existing Flow CLI scaffolds. For example the `hello-world` scaffold already has fcl configuration settings to run on local emulator.
-
-:::
-
-
-The `hello-world` web application comes with convenience npm commands to facilitate a quick start. The following command will preform:
- 1. Start emulator
- 2. Start dev wallet
- 3. Start web app
-
-:::info
-
-Emulator is a local blockchain [More Information](../../tools/emulator/index.md). Emulator has all the features as testnet and mainnet blockchains
-
-:::
-
-```sh
-npm run dev:local
-```
-
-Now that your app is running. FCL loads the configuration in `config/fcl.ts` This file contains configuration information for FCL, such as what Access Node and wallet discovery endpoint and which network to use (e.g. testnet or a local emulator).
-
-- `accessNode.api` key specifies the address of a Flow access node. There are publically available access nodes, but have to rate limit. Alternatively, applications might want to run an Observer node [Run a Node](../../networks/node-ops/access-onchain-data/light-nodes/observer-node.md).
-- `discovery.wallet` is an address that points to a service that lists FCL compatible wallets. Flow's FCL Discovery service is a service that FCL wallet providers can be added to, and be made 'discoverable' to any application that uses the `discovery.wallet` endpoint.
-
-Also, notice that `package.json` uses `NEXT_PUBLIC_FLOW_NETWORK=local` for `dev` command, this is used to set the network in `config/fcl.ts`.
-
-:::info
-
-Learn more about [configuring Discovery](../../tools/clients/fcl-js/discovery.md) or [setting configuration values](../../tools/clients/fcl-js/api.md#setting-configuration-values).
-
-:::
-
-The main Next.js app component is located in `pages/_app.tsx`. It should import the config file `config/fcl.ts` already and should look like this:
-
-```jsx
-import '../styles/globals.css'
-import DefaultLayout from '../layouts/DefaultLayout'
-
-// Import FCL config
-import '../config/fcl'
-
-function MyApp({ Component, pageProps }) {
- return (
-
-
-
- )
-}
-
-export default MyApp
-
-```
-
-The main page for the Next.js app is located in `pages/index.tsx`. It should contain some basic UI and should look like this:
-
-```jsx
-import Head from 'next/head'
-import styles from '../styles/Home.module.css'
-import Links from '../components/Links'
-import Container from '../components/Container'
-import useCurrentUser from '../hooks/useCurrentUser'
-
-export default function Home() {
- const { loggedIn } = useCurrentUser()
-
- return (
-
- )
-}
-
-```
-
-Now we're ready to start talking to Flow!
-
-:::warning
-
-The web app will run, but there are no contracts deployed to local emulator. This is a step in [Query the Blockchain](#query-the-blockchain) section.
-
-:::
-
-## Authenticate a User
-
-Navigate to [localhost:3000](http://localhost:3000) in your browser.
-
-![User Login](./user-login-hello-world.gif)
-
-Note: in the code above `useCurrentUser` is used to determine if there is a logged in user.
-
-There are two methods to allow the user to login. `fcl.logIn()` or `fcl.authenticate()` [More Information on, authenticate](../../tools/clients/fcl-js/api#authenticate)
-
-In `components/Navbar.tsx` a button wires up the authentication method `fcl.authenticate()`. It is used to bring up the list of supported wallets. See below
-
-Once authenticated, FCL uses a hook `const user = useCurrentUser()` to get the user data, when user is signed in `user.loggedIn` flag is `true`. For more information on the `currentUser`, read more [here](../../tools/clients/fcl-js/api.md#current-user).
-
-```jsx
-import Head from 'next/head'
-import * as fcl from '@onflow/fcl'
-import useCurrentUser from '../hooks/useCurrentUser'
-import navbarStyles from '../styles/Navbar.module.css'
-import elementStyles from '../styles/Elements.module.css'
-
-export default function Navbar() {
- const user = useCurrentUser()
-
- return (
-
- )
-}
-
-```
-
-You should now be able to log in or sign up users and unauthenticate them. Upon logging in or signing up your users will see a popup where they can choose between wallet providers. Choose the `dev wallet` to use the same account that deployed the `HelloWorld` contract, this is needed for mutation. Upon completing authentication, you'll see the component change and the user's wallet address appear on the screen if you've completed this properly.
-
-:::info
-
-More on wallets, [Flow Core wallet](https://core.flow.com/) is a Reference Flow wallet.
-
-:::
-
-## Deploy contracts
-
-Hello World scaffold does come with a Cadence contract. You will notice `HelloWorld.cdc` has been deployed when running `npm run dev:local`. Look at hello-world `package.json` to see the commands that get run, `flow dev` deploys contracts to the emulator.
-In the `flow.json` make sure the emulator endpoint is correct. Look at the terminal the emulator is running,
-
-![Emulator Output](./emulator-output.png)
-
-1. Make sure the emulator is using the same port as `gRPC` and
-2. The `deployment` section of `flow.json` should look something like this:
-```json
- ...
- "networks": {
- "emulator": "127.0.0.1:3569",
- ...
- },
- "deployments": {
- "emulator": {
- "default": [
- "HelloWorld"
- ]
- }
-}
-
-```
-
-Verify that `flow.json` updates with HelloWorld contract information, `contracts` has the `HelloWorld` contract and `deployments` shows that `HelloWorld` has been deployed.
-
-
-```json
-{
- "contracts": {
- "HelloWorld": "cadence/contracts/HelloWorld.cdc"
- },
- "networks": {
- "emulator": "127.0.0.1:3569",
- "mainnet": "access.mainnet.nodes.onflow.org:9000",
- "testnet": "access.devnet.nodes.onflow.org:9000"
- },
- "accounts": {
- "default": {
- "address": "01cf0e2f2f715450",
- "key": "..."
- },
- "emulator-account": {
- "address": "f8d6e0586b0a20c7",
- "key": {
- "type": "file",
- "location": "./emulator.key"
- }
- }
- },
- "deployments": {
- "emulator": {
- "default": [
- "HelloWorld"
- ]
- }
- }
-}
-```
-
-For more information on deployments check [Flow CLI](../../tools/flow-cli/flow.json/manage-configuration.md) documentation
-
-## Query the Blockchain
-
-Now that all the pieces are in place, you can query the `HelloWorld` contract.
-
-Navigate again to [localhost:3000](http://localhost:3000) in your browser.
-
-![Query HelloWorld Contract](./query-helloWorld-contract.gif)
-
-
-The script that queries the Hello World contract is located `hello-world/cadence/scripts/ReadHelloWorld.cdc`
-
-```cadence
-import "HelloWorld"
-
-access(all) fun main(): String {
- return HelloWorld.greeting
-}
-```
-
-In `components/Container.tsx` file, `fcl.query` is used to set the Cadence script and query the contract.
-
-```jsx
-
- const queryChain = async () => {
- const res = await fcl.query({
- cadence: ReadHelloWorld
- })
-
- setChainGreeting(res)
- }
-```
-
-It is that simple!
-
-## Mutate the Blockchain
-
-Update the `HelloWorld` contract greeting. Notice that Greeting gets changed when the transaction gets processed.
-
-![Update contract](./hello-world-update-contract.gif)
-
-
-In `components/Container.tsx` file, the `mutateGreeting` method `fcl.mutate` sends `UpdateHelloWorld` cadence which triggers a transaction that the user signs.
-
-```javascript
- const mutateGreeting = async (event) => {
- event.preventDefault()
-
- if (!userGreetingInput.length) {
- throw new Error('Please add a new greeting string.')
- }
-
- const transactionId = await fcl.mutate({
- cadence: UpdateHelloWorld,
- args: (arg, t) => [arg(userGreetingInput, t.String)],
- })
-
- setLastTransactionId(transactionId)
- }
-
-```
-
-## More information
-
-That's it! You now have Flow app that uses auth, query and mutate the chain. This is just the beginning. There is so much more to know. Next steps:
-
-**Cadence**
-- [Cadence Playground Tutorials](https://cadence-lang.org/docs/tutorial/first-steps)
-- [Cadence Hello World Video](https://www.youtube.com/watch?v=pRz7EzrWchs)
-- [Why Cadence?](https://www.onflow.org/post/flow-blockchain-cadence-programming-language-resources-assets)
-
-**FCL Scaffolds**
-- [FCL Next TypeScript Scaffold](https://github.com/chasefleming/fcl-next-scaffold)
-- [FCL React Native Scaffold](https://github.com/jribbink/fcl-react-native-scaffold)
-
-**Full Stack NFT Marketplace Example**
-- [Beginner Example: CryptoDappy](https://github.com/bebner/crypto-dappy)
-
-**More FCL**
-- [FCL API Quick Reference](../../tools/clients/fcl-js/api)
-- [More on Scripts](../../tools/clients/fcl-js/scripts.md)
-- [More on Transactions](../../tools/clients/fcl-js/transactions.md)
-- [User Signatures](../../tools/clients/fcl-js/user-signatures.md)
-- [Proving Account Ownership](../../tools/clients/fcl-js/proving-authentication.mdx)
diff --git a/docs/build/guides/flow-wallet.png b/docs/build/guides/flow-wallet.png
new file mode 100644
index 0000000000..e9e9695c1f
Binary files /dev/null and b/docs/build/guides/flow-wallet.png differ
diff --git a/docs/build/guides/welcome.png b/docs/build/guides/welcome.png
new file mode 100644
index 0000000000..f65093e060
Binary files /dev/null and b/docs/build/guides/welcome.png differ
diff --git a/docs/tools/clients/fcl-js/api.md b/docs/tools/clients/fcl-js/api.md
index 102b526f97..d7716fd7c0 100644
--- a/docs/tools/clients/fcl-js/api.md
+++ b/docs/tools/clients/fcl-js/api.md
@@ -434,7 +434,7 @@ export const signMessage = async () => {
Discovery abstracts away code so that developers don't have to deal with the discovery of Flow compatible wallets, integration, or authentication. Using `discovery` from FCL allows dapps to list and authenticate with wallets while having full control over the UI. Common use cases for this are login or registration pages.
-(Alternatively, if you don't need control over your UI you can continue to use the `discovery.wallet` config value documented in the [Quickstart](../../../build/guides/flow-app-quickstart.md) for the simplest configuration.)
+(Alternatively, if you don't need control over your UI you can continue to use the `discovery.wallet` config value documented in the [Quickstart](../../../build/getting-started/fcl-quickstart.md) for the simplest configuration.)
> β οΈ**The following methods can only be used in web browsers.**
@@ -503,7 +503,7 @@ config({
| `Dapper Wallet` | 0x82ec283f88a62e65 | 0xead892083b3e2c6c |
| `Ledger` | 0x9d2e44203cb13051 | 0xe5cd26afebe62781 |
-For more details on wallets, view the [service list here](https://github.com/onflow/fcl-discovery/blob/master/data/services.json).
+For more details on wallets, view the [service list here](https://github.com/onflow/fcl-discovery/blob/87e172db85d185882d9fde007c95f08bc2a1cccb/data/services.json).
---
@@ -2092,4 +2092,4 @@ Signature objects are used to represent a signature for a particular message as
| ----------- | ------------------------------------------------------------- | ---------------------------------------------------------------------------------------------------------------------------------------------------------------- |
| `addr` | [Address](#address) | the address of the account which this signature has been generated for |
| `keyId` | number | The index of the key to use during authorization. (Multiple keys on an account is possible).|
-| `signature` | string | a hexidecimal-encoded string representation of the generated signature |
\ No newline at end of file
+| `signature` | string | a hexidecimal-encoded string representation of the generated signature |
diff --git a/docs/tools/clients/fcl-js/authentication.md b/docs/tools/clients/fcl-js/authentication.md
index 28cb268974..a1f029aa65 100644
--- a/docs/tools/clients/fcl-js/authentication.md
+++ b/docs/tools/clients/fcl-js/authentication.md
@@ -42,6 +42,6 @@ The TL;DR is to call `fcl.authenticate()` and `fcl.unauthenticate()` respectivel
On Flow mainnet, you wont even need to configure anything for this to work, the users of your dapp will go through the authentication process and be able to use any FCL compatible wallet providers.
During development you will probably want to configure your dapp to use [`@onflow/dev-wallet`](https://github.com/onflow/fcl-dev-wallet).
-The [Quick Start](../../../build/guides/flow-app-quickstart.md) guide will walk you through using it.
+The [Quick Start](../../../build/getting-started/fcl-quickstart.md) guide will walk you through using it.
We know this can all be fairly overwhelming, we are committed to help though. If you run into any problems, reach out to us on [Discord](https://discord.gg/flow), we are more than happy to help out.
diff --git a/docs/tools/clients/fcl-js/index.md b/docs/tools/clients/fcl-js/index.md
index 2864b53af2..8cad59b29b 100644
--- a/docs/tools/clients/fcl-js/index.md
+++ b/docs/tools/clients/fcl-js/index.md
@@ -114,7 +114,7 @@ const txId = await fcl.mutate({
## Next Steps
-See the [Flow App Quick Start](../../../build/guides/flow-app-quickstart.md).
+See the [Flow App Quick Start](../../../build/getting-started/fcl-quickstart.md).
See the full [API Reference](./api.md) for all FCL functionality.
diff --git a/docs/tools/clients/fcl-js/sdk-guidelines.md b/docs/tools/clients/fcl-js/sdk-guidelines.md
index 51c05a0736..70284ae3b0 100644
--- a/docs/tools/clients/fcl-js/sdk-guidelines.md
+++ b/docs/tools/clients/fcl-js/sdk-guidelines.md
@@ -52,24 +52,6 @@ config({
})
```
-π **gRPC Access API URLs** can be found [here](../../../networks/node-ops/access-onchain-data/access-nodes/accessing-data/access-api.md#flow-access-node-endpoints). `sdk.transport` must be specified if you wish to use the gRPC API. The public Flow gRPC access nodes are accessible at:
-- Testnet `https://access-testnet.onflow.org`
-- Mainnet `https://access-mainnet.onflow.org`
-- Local Emulator `127.0.0.1:3569`
-
-For local development, use the [flow emulator](https://github.com/onflow/flow-emulator) which once started provides an HTTP access endpoint at `127.0.0.1:8888` and a gRPC access endpoint at `127.0.0.1:3569`.
-
-If using the gRPC Access API, the `sdk.transport` configuration key must be populated as this value defaults to the HTTP API transport. The SDK can be configured to use the gRPC API transport as follows:
-```javascript
-import { config } from "@onflow/fcl"
-import { send as transportGRPC } from "@onflow/transport-grpc"
-
-config({
- "accessNode.api": "https://access-testnet.onflow.org",
- "sdk.transport": transportGRPC
-})
-```
-
## Querying the Flow Network
After you have established a connection with an access node, you can query the Flow network to retrieve data about blocks, accounts, events and transactions. We will explore how to retrieve each of these entities in the sections below.
diff --git a/docs/tools/clients/index.md b/docs/tools/clients/index.md
index b0a28dc45b..29998aa1c8 100644
--- a/docs/tools/clients/index.md
+++ b/docs/tools/clients/index.md
@@ -16,10 +16,6 @@
[Flow JVM SDK](https://github.com/onflow/flow-jvm-sdk) is a library for JVM-compatible languages, implemented in Kotlin, that provides utilities to interact with the Flow blockchain.
-## Unity
-
-[Flow SDK for Unity](./unity-sdk/index.md) allows Unity developers to integrate their games and applications with the Flow blockchain.
-
## JavaScript (FCL)
[Flow Client Library (FCL)](./fcl-js/index.md) is a package used to interact with user wallets and the Flow blockchain.
diff --git a/docs/tools/clients/unity-sdk/api-reference.md b/docs/tools/clients/unity-sdk/api-reference.md
deleted file mode 100644
index b234b8b08c..0000000000
--- a/docs/tools/clients/unity-sdk/api-reference.md
+++ /dev/null
@@ -1,9 +0,0 @@
----
-title: API Reference
-sidebar_position: 1
-sidebar_label: API Reference
----
-
-Click on the following link to view the API Reference for the Flow SDK for Unity.
-
-[](https://unity-flow-sdk-api-docs.vercel.app/)
\ No newline at end of file
diff --git a/docs/tools/clients/unity-sdk/changelog.md b/docs/tools/clients/unity-sdk/changelog.md
deleted file mode 100644
index 80f81f3c0a..0000000000
--- a/docs/tools/clients/unity-sdk/changelog.md
+++ /dev/null
@@ -1,49 +0,0 @@
----
-title: Changelog
-sidebar_position: 4
-sidebar_label: Changelog
----
-## Version 2.0.0
-
-- Added new wallet provider Wallet Connect
-- Added iOS as a supported platform
-- Added Cadence Convert feature for simple de/serialization
-- Changed how wallet providers are instantiated (see Breaking Changes)
-
-### Breaking Changes in 2.0.0
-
-Previously, wallet providers (ie DevWallet) were instantiated as follows:
-
-```csharp
-FlowSDK.RegisterWalletProvider(ScriptableObject.CreateInstance());
-```
-
-From 2.0.0, this should now be:
-
-```csharp
-FlowSDK.RegisterWalletProvider(new DevWalletProvider());
-```
-
-## Version 1.0.3
-
-- Add Example NFT sample
-
-## Version 1.0.2
-
-- Fixed an issue where the latest Flow emulator would cause the editor to freeze
-
-## Version 1.0.1
-
-- Added wallet authentication to contract and account creation tools
-- Fixed all Unity warnings in all samples
-- Removed unused and commented code
-- Added README files to all samples
-- Emulator listens on all local IP addresses
-- Improved error handling from all API requests
-- Improved error handling and feedback from Flow Control tools
-- Limited Events.GetForBlockIds request to 50 block ids
-- Fixed login panel on Flow Words Tutorial sample
-
-## Version 1.0.0
-
-- Initial release of the Flow SDK for Unity
\ No newline at end of file
diff --git a/docs/tools/clients/unity-sdk/guides/_category_.yml b/docs/tools/clients/unity-sdk/guides/_category_.yml
deleted file mode 100644
index c65b5e747e..0000000000
--- a/docs/tools/clients/unity-sdk/guides/_category_.yml
+++ /dev/null
@@ -1,2 +0,0 @@
-label: Guides
-position: 2
\ No newline at end of file
diff --git a/docs/tools/clients/unity-sdk/guides/convert.md b/docs/tools/clients/unity-sdk/guides/convert.md
deleted file mode 100644
index 517c319d4a..0000000000
--- a/docs/tools/clients/unity-sdk/guides/convert.md
+++ /dev/null
@@ -1,231 +0,0 @@
-# Flow SDK Convert
-
-The FlowSDK provides a Convert class to convert between Cadence and C# datatypes. It provides two functions, ToCadence and FromCadence.
-
-The conversion is strict in regards to the types of the data. C# types *must* match the Cadence type being converted from/to or an exception will be thrown. The following are the required types:
-
-Cadence|C#
----|---
-Int|BigInteger
-UInt|BigInteger
-Int8|SByte
-Int16|Int16
-Int32|Int32
-Int64|Int64
-Int128|BigInteger
-Int256|BigInteger
-UInt8|Byte
-UInt16|UInt16
-UInt32|UInt32
-UInt64|UInt64
-UInt128|BigInteger
-UInt256|BigInteger
-Word8|Byte
-Word16|UInt16
-Word32|UInt32
-Word64|UInt64
-Fix64|Decimal
-UFix64|Decimal
-Address|String
-String|String
-Bool|Boolean
-Path|CadencePathValue
-Capability|CadenceCapabilityValue
-`Array ([T])`|List<T>
-`Dictionary ({T:R})`|Dictionary<T,R>
-Struct|Class or Struct
-
-
-## Usage
-
-### ToCadence
-
-The Convert.ToCadence function has the following signature:
-```csharp
-public static CadenceBase ToCadence(object source, string destinationType)
-```
-
-ToCadence will return a CadenceBase that can be passed into Script and Transaction functions. Example:
-
-```csharp
-CadenceBase cb = Convert.ToCadence((Int64)44, "Int64");
-```
-
-This will result in a CadenceNumber with a value of "44" and a type of "Int64". However:
-
-```csharp
-CadenceBase cb = Convert.ToCadence(44, "Int64");
-```
-
-will result in an exception "Exception: Can not convert System.Int32 into Int64. Requires source to be System.Int64."
-
-Similarly, the Cadence Int type is an arbitrary precision integer type and thus must be created using a System.BigInteger.
-
-```csharp
-CadenceBase cb = Convert.ToCadence(new BigInteger(44), "Int");
-```
-
-### FromCadence
-
-The Convert.FromCadence has the following signature:
-
-```csharp
-public static T FromCadence(CadenceBase cadence)
-```
-where T is the C# you would like the Cadence converted into.
-
-FromCadence will return a value of the requested type given a CadenceBase value. The Cadence and C# types must match.
-
-```csharp
-CadenceBase cb = new CadenceNumber(CadenceNumberType.Int16, "44");
-Int16 i = Convert.FromCadence(cb);
-```
-
-If the requested and source types do not match, an exception will be thrown.
-
-```csharp
-CadenceBase cb = new CadenceNumber(CadenceNumberType.Int16, "44");
-Int64 i = Convert.FromCadence(cb);
-```
-
-The above results in an exception: "Exception: Attempt to convert cadence to invalid type. Cadence type Int16 expects System.Int16, got System.Int64".
-
-### Composite Types
-
-Using the Convert class on primitive types isn't much easier than constructing the Cadence types yourself. Using it on Composite types is much more useful.
-
-Convert can convert between Cadence composite types (Struct, Enum, Resource, Event, Contract) and C# structs/classes. In order to annotate your C# classes/structs a CadenceAttribute is provided.
-The CadenceAttribute has two properties. The Name and CadenceType properties allow you to provide hints to the converter as to what each C# types should be when converted to Cadence.
-
-Given the following C# class:
-
-```csharp
-public class TestStruct
-{
- public Int32 i32;
- public String s;
-}
-```
-
-and the following Cadence struct:
-
-```cadence
-access(all) struct TestStruct {
- access(all) let i32: Int32
- access(all) let s: String
-}
-```
-
-you can convert from Cadence to C# using:
-
-```csharp
-TestStruct ts = Convert.FromCadence(cadence);
-```
-
-It will match based on the field names, and the types are compatible. Converting from C# to Cadence, on the other hand, requires annotations.
-
-```csharp
-[Cadence(CadenceType="A.XXX.CCC.TestStruct")]
-public class TestStruct
-{
- [Cadence(CadenceType="Int32", Name="i32")]
- public Int32 i;
- [Cadence(CadenceType="String")]
- public String s;
-}
-```
-
-This is because a C# string could be a Cadence Address or Cadence String. For consistency, Convert requires all C#->Cadence conversions to be annotated with a CadenceType. If a field is not
-annotated, it will be skipped when converting from C# to Cadence.
-
-You can also use the Name parameter to account for differences in field naming. In the above example we mapped the Cadence "i32" field to the C# "i" field. The Name property is optional and
-it will use the field name if no Name property is given.
-
-***Note: The CadenceType annotation on C# classes/structs (`[Cadence(CadenceType="A.XXX.CCC.TestStruct")]` in the above example) is ignored when converting from Cadence to C#.
-Convert.FromCadence will populate all fields that have matching names/types regardless of the type of Cadence struct that is being converted. This allows you to convert a Cadence struct defined
-int a Cadence script into C# even if you do not know what the Cadence type is.***
-
-The class annotation (A.XXX.CCC.TestStruct) is required when converting from C# to Cadence. XXX should be the address of the account that contains the contract where the struct is
-defined, without a leading "0x". The CCC is the name of the contract.
-
-Here's an example using the NBA TopShot contract (https://github.com/dapperlabs/nba-smart-contracts/blob/master/contracts/TopShot.cdc) on TestNet, defining a class to hold the Play struct:
-
-```csharp
-[Cadence(CadenceType="A.877931736ee77cff.TopShot.Play")]
-public class Play
-{
- [Cadence(CadenceType="UInt32")]
- public UInt32 playID;
- [Cadence(CadenceType="{String:String}")]
- public Dictionary metadata;
-}
-```
-
-## Structs inside Structs
-
-If a Cadence struct contains another struct, the field should be annotated as a "Struct". Given the following Cadence:
-
-```cadence
-access(all) struct Other {
- access(all) let i: Int16
-
- access(all) init(i:Int16) {
- self.i=i
- }
-}
-
-access(all) struct Test {
- access(all) let o : Other
-
- access(all) init(i: Int16) {
- self.o = Other(i:i)
- }
-}
-```
-
-you could use this C#:
-
-```csharp
-[Cadence(CadenceType="A.xxx.ccc.Nested")]
-public class Other
-{
- [Cadence(CadenceType="Int16")]
- public Int16 i;
-}
-
-[Cadence(CadenceType="A.xxx.ccc.Test")]
-public class Test
-{
- [Cadence(CadenceType="Struct")]
- public Other o;
-}
-```
-
-If you have a Cadence Test struct you can convert into C# using:
-```csharp
-TestStruct ts = Convert.FromCadence(cadence);
-```
-
-If you have a C# Test object, you can convert to Cadence using:
-```csharp
-CadenceBase cb = Convert.ToCadence(ts, "Struct");
-```
-
-## Optionals
-
-Cadence optionals are indicated by appending a ?. For instance the Cadence type Int16? can contain either an Int16 value or nil. If the C# type is a reference type, no additional work is required.
-For instance a Cadence String? will have the C# equivalent type of String. This is because the C# String is a reference type, which can natively be set to null. On the other hand, the Cadence Int16? requires the C# type Int16? which wraps type the value type Int16 in a `Nullable<>`.
-
-```csharp
-//c is a Cadence Int16? type
-Int16? i = Convert.FromCadence(c);
-```
-
-Trying to convert a Cadence optional into a non-nullable type results in an exception:
-
-```csharp
-//c is a Cadence Int16? type
-Int16 i = Convert.FromCadence(c);
-```
-
-"Exception: CadenceOptional requires a nullable type"
\ No newline at end of file
diff --git a/docs/tools/clients/unity-sdk/guides/dev-wallet.md b/docs/tools/clients/unity-sdk/guides/dev-wallet.md
deleted file mode 100644
index 612026040b..0000000000
--- a/docs/tools/clients/unity-sdk/guides/dev-wallet.md
+++ /dev/null
@@ -1,114 +0,0 @@
-# Dev Wallet
-
-## Introduction
-
-Dev Wallet is an implementation of a wallet provider for development purposes. It provides the following:
-
-- An example of how to implement the `IWallet` interface, should you wish to implement your own wallet provider.
-- An easy way to sign transactions during development, without the burden of connecting to real wallets.
-
-***IMPORTANT:*** Never use Dev Wallet in a production build, as it is not secure.
-
-The initial v1.0.0 release of the Flow SDK for Unity does not provide a production ready wallet provider. We are working hard to release one in the very near future. In the meantime, you can attempt to create your own wallet provider implementing the `IWallet` interface, or stick with Dev Wallet while you develop your game.
-
-## What are Wallets?
-
-A wallet is a piece of software or hardware that stores the private key associated with a Flow account. The term *custodian* is used to refer to the party that stores the private key. Hardware Wallets (eg Ledger), typically USB devices, allow users to be their own custodian, whereas hosted software wallets (eg Dapper Wallet) act as the custodian on behalf of the user.
-
-## What is a Wallet Provider?
-
-In terms of the Flow SDK for Unity, a Wallet Provider is a class which implements the `IWallet` interface and allows users to interact with specific hardware or software wallets. This includes authenticating with a wallet, retrieving the user's Flow account address from the wallet, and requesting the wallet to sign transactions on behalf of the user.
-
-As of v1.0.0, the Flow SDK for Unity only contains a single wallet provider - Dev Wallet. This is not a real wallet provider, but a mock wallet provider for development purposes. It simulates the same functionality as a real wallet provider, but doesn't require users to create and integrate with real wallets. This is a great feature for developers, because it allows them to focus on making the game and designing their on-chain architecture in a purely sandbox environment. A production ready wallet provider will be released in a future version of the SDK as a priority feature.
-
-## How to use Dev Wallet
-
-Dev Wallet uses the accounts listed in the Accounts tab of the Flow Control Window. You can "authenticate" as one of these accounts, then sign transactions with that account. For more information on setting up these accounts, see [flow-control#flowcontrol-accounts](../../unity-sdk/guides/flow-control.md#flowcontrol-accounts).
-
-### Registering
-
-To use Dev Wallet, you must register it as follows:
-
-```csharp
-FlowSDK.RegisterWalletProvider(ScriptableObject.CreateInstance());
-```
-
-This should be done at the same time as initializing the Flow SDK. Here is a complete example of initializing the Flow SDK to use the emulator and registering Dev Wallet:
-
-```csharp
-using DapperLabs.Flow.Sdk;
-using DapperLabs.Flow.Sdk.DevWallet;
-
-FlowConfig flowConfig = new FlowConfig();
-flowConfig.NetworkUrl = FlowControl.Data.EmulatorSettings.emulatorEndpoint // local emulator
-flowConfig.Protocol = FlowConfig.NetworkProtocol.HTTP;
-
-FlowSDK.Init(flowConfig);
-FlowSDK.RegisterWalletProvider(ScriptableObject.CreateInstance());
-```
-
-### Authenticating
-
-The `IWallet.Authenticate` method is as follows:
-
-```csharp
-public void Authenticate(string username, System.Action OnAuthSuccess, System.Action OnAuthFailed);
-```
-
-`username` is a string that can be used to identify a user's wallet. For Dev Wallet, this corresponds to the `Name` field of an account in the Accounts tab of the Flow Control Window. If you know which account you want to authenticate with, supply the account name as this argument. If you pass in a blank string, the user will get a dialog where they can choose which account to authenticate as. As this is a development tool, there are no passwords - it is simply simulating authentication.
-`OnAuthSuccess` is a function that will be called when you have successfully authenticated. The callback function must take a `string` argument, which will contain the authenticated account's Flow address.
-`OnAuthFailed` is a function that will be called if authentication failed. In Dev Wallet, this would only fail if you passed in a `username` that doesn't exist.
-
-Here is an example of authenticating as the `user1` account from game code:
-
-```csharp
-FlowSDK.GetWalletProvider().Authenticate("user1", (string flowAddress) =>
-{
- Debug.Log($"Authenticated - Flow account address is {flowAddress}");
-}, () =>
-{
- Debug.Log("Authentication failed.");
-});
-```
-
-Here is an example of calling `Authenticate` to display a list of Dev Wallet accounts to choose from:
-
-```csharp
-FlowSDK.GetWalletProvider().Authenticate("", (string flowAddress) =>
-{
- Debug.Log($"Authenticated - Flow account address is {flowAddress}");
-}, () =>
-{
- Debug.Log("Authentication failed.");
-});
-```
-
-### Signing Transactions
-
-If you are using the Flow SDK to sign transactions then you do not need to worry about this, as it is handled automatically. When you submit a transaction, the SDK will request Dev Wallet to sign the transaction as the authenticated user. A dialog will be displayed requesting the user to approve the transaction.
-
-For full disclosure, here are the methods on the `IWallet` interface to sign a transaction:
-
-```csharp
-public Task SignTransactionPayload(FlowTransaction txn);
-
-public Task SignTransactionEnvelope(FlowTransaction txn);
-```
-
-In Flow, there are two parts of a transaction that can be signed - the Payload and the Authorization Envelope. The envelope must always be signed, and is the last thing to be signed by the Payer of the transaction fees. The Payload is only signed by the Proposer and\or the Authorizers IF they are not also the Payer (i.e. nobody signs the transaction twice). For more information see [transaction signing](../../../../build/basics/transactions.md#signing-a-transaction).
-
-The following is an example of how to call `SignTransactionPayload`, but as mentioned, this is automatically done by the SDK's `Transactions.Submit` function. It's an asynchronous so is therefore `await`ed, and returns the signature as a byte array.
-
-```csharp
-byte[] signature = await FlowSDK.GetWalletProvider().SignTransactionPayload(txRequest);
-```
-
-### Unauthenticating
-
-You can unauthenticate from Dev Wallet by calling the following:
-
-```csharp
-FlowSDK.GetWalletProvider().Unauthenticate();
-```
-
-This will clear Dev Wallet's internal cache of who is authenticated. Note that trying to submit transactions without anyone authenticated will result in an error.
diff --git a/docs/tools/clients/unity-sdk/guides/flow-control.md b/docs/tools/clients/unity-sdk/guides/flow-control.md
deleted file mode 100644
index 2389a7abdc..0000000000
--- a/docs/tools/clients/unity-sdk/guides/flow-control.md
+++ /dev/null
@@ -1,115 +0,0 @@
-ο»Ώ# FlowControl
-
-FlowControl is a set of Unity tools and functions designed to streamline development and interactions with the Flow blockchain.
-
-After installing the Unity Flow SDK, open the FlowControl Manager window by clicking on Windows->Flow->Flow Control.
-
-This will open the FlowControl Manager window. It will also add a Flow Control object to your scene and create a FlowControlData resource to store your data if they are not already present. You can have multiple FlowControlData resources to act as presets and switch between them as needed.
-
-## FlowControl Emulator Settings
-
-The first panel that displays will be your emulator settings.
-
-![FlowControl Emulator Settings](../media/FlowControlEmulator.png)
-
-**Installation Detection:** FlowControl will attempt to find the flow binary in your path. If it can't detect it, it will display a button with the label "Install" linking you to the install instructions.
-
-**Emulator Data Directory:** This field is required if you wish to control the emulator from FlowControl. Set this to the **directory** where you would like to store your persisted
-emulator state (flowdb), and your initial emulator configuration (flow.json). This should typically be outside your project folder. The folder must exist.
-
-**Emulator endpoint:** If you want to connect to an emulator running on a different computer, such as when testing on mobile devices or several clients all connecting
-to the same emulator instance, you can specify an emulator endpoint address here.
-
-**Run emulator in play mode?** This determines if the emulator should automatically start when you enter play mode.
-
-**Start Emulator** button: This will start the emulator in the background to allow you to interact with it in edit mode.
-
-**Stop Emulator** button: This will stop a running emulator process.
-
-**Clear Persistent Data** button: The Flow emulator maintains the state of the emulated blockchain in a database. This allows you to persist data between runs. If you would like to reset the emulated blockchain to its initial state, use this button. This will delete the flowdb directory located in the Emulator Data Directory. It will be recreated the next time you start the emulator.
-
-**Show Emulator Log** button: This opens the Emulator Output window. You can use this to verify that scripts you run against the emulator execute
-properly and will display debug messages if something goes wrong. You can also open the log using Window->Flow->Emulator Output
-
-## Advanced Options
-
-Some options are not exposed in the Flow Control Manager, but can be changed directly in the FlowControlData asset.
-![FlowControlData asset](../media/FlowControlDataAsset.png)
-
-**Flow Executable Path Override:** This allows you to select the flow binary you would like to use. This will override the auto detected flow binary.
-
-**Persist Data:** This allows you disable persisting the emulated blockchain to disk. If this is disabled, all changes you make to the emulated blockchain will be lost when the emulator stops or restarts.
-
-**Kill Other Flow:** When starting or restarting the emulator, any running flow processes will be terminated. If you wish to run your own emulator instance outside the control of Flow Control, uncheck this, as well as "Run Emulator in Play Mode" to ensure it doesn't get terminated.
-
-**Verbose:** This controls the verbosity of emulator output. Verbose output is usually desired, but if you would like non-verbose output for any reason, this allows you to control that setting.
-
-## FlowControl Accounts
-
-The Accounts panel allows you to store account information for easy access at runtime.
-***Note: Adding accounts to this panel does not create accounts on the blockchain***
-
-![FlowControl Accounts](../media/FlowControlAccounts.png)
-
-The "+" button at the top will add a blank account entry for you to fill in. The fields may differ depending on which Gateway is selected, but Gateway and Name are always required.
-
-The emulator_service_account account will be created/updated each time you start the emulator so that it will always be valid.
-
-If it is a Testnet account, a link to the Testnet Flowdiver page for that account will be provided. Clicking on it will open the page in your default browser.
-
-## FlowControl Text Replacement
-
-![FlowControl Text Replacement](../media/FlowControlTextReplacement.png)
-
-The Text Replacement panel allows you to manage string replacement inside your Cadence scripts. If you're switching between emulator and Testnet, this makes it easy to replace account addresses and other strings without having to edit the scripts each time.
-
-All instances of Original Text will be replaced with Replacement Text. There is no required format, but it is recommended that you enclose the placeholder text in a separator character to ensure that there are no conflicting replacements.
-
-The Description field is optional and is just used to make it easy to remember what each replacement is for.
-
-The Active checkbox indicates if this text replacement should be performed.
-
-"Apply to accounts" allows you to select specific accounts this replacement should apply to. Any transactions or script executions performed via these accounts will have this replacement applied.
-
-"Apply to gateways" allows you to select specific gateways this replacement should apply to. Any transactions or script executions performed via an account utilizing these gateways will have this replacement performed.
-
-Accounts must meet both the account and gateway requirements for a text replacement to take place. If an account is listed in the accounts list, but uses a gateway that is not in the gateways list, the replacement will not be processed and vice-versa.
-
-## FlowControl Tools
-
-The FlowControl Tools panel can be used to deploy contracts and create accounts.
-
-![FlowControl Tools](../media/FlowControlTools.png)
-
-### Manage Contracts
-
-The Contract Name is the name of the contract as it will appear on the blockchain. It must match the contract name in the contract file. Text replacements are done on the contract file before deploying.
-
-Contract is the cadence contract file that you want to deploy.
-
-Account is the account that should deploy the contract. The contract will be deployed to the same network that the account gateway uses.
-
-The Deploy Contract button will attempt to deploy the contract. Check the emulator logs or flowdiver.io to monitor the progress of deployment and check for errors.
-
-The Update Contract button will attempt to update an existing contract. There are rules as to what can change in a contract for an update to be accepted. See [cadence/language/contract-updatability](https://cadence-lang.org/docs/language/contract-updatability) for more information.
-
-The Remove Contract button will attempt to delete the contract with the given name from the specified account. Once a contract is deleted, a new contract with the same name can not be created on the same account. When using the emulator, clearing the database will allow you to recreate the contract. When using testnet or mainnet, that contract name will become unavailable for the account. This is done to prevent bypassing the contract update rules by deleting and recreating a contract with incompatible changes.
-
-
-### Transactions
-
-This section will allow you to execute a transaction against the gateway associated with the Signer account. Currently it does not allow you to pass arguments to the transaction, but you can use a Text Replacement to work around this.
-
-
-### Create New Account
-
-The Paying account is the account that will initially fund the new account. The new account name is not stored on the blockchain, and is only used to identify the new account in the Accounts panel. After the account is created, the account information will be added to the Accounts panel. Make sure to save a copy of this information somewhere. If the private key is lost, that account will not be accessible. You can change the account Name in the Accounts panel. Note that changing the Address or Private Key fields will not change these on the blockchain.
-
-
-## FlowControl inspector
-![FlowControl Inspector](../media/FlowControlInspector.png)
-
-Clicking the FlowControl object in your scene will give you access to the inspector. This will allow you to select which FlowControlData resource you would like to use
-as well as export to/import from JSON. JSON is useful if you would like to modify FlowControl variables from a built project, since you will not be able to save the
-FlowControlData resource from a built project. The Log JSON button will serialize the FlowControlData resource into a string and display it. The Load JSON button will
-deserialize the displayed JSON into the selected FlowControlData resource.
diff --git a/docs/tools/clients/unity-sdk/guides/ios-support.md b/docs/tools/clients/unity-sdk/guides/ios-support.md
deleted file mode 100644
index c3972f741b..0000000000
--- a/docs/tools/clients/unity-sdk/guides/ios-support.md
+++ /dev/null
@@ -1,28 +0,0 @@
-# iOS Support
-
-To use the Flow SDK in iOS projects, there are a couple more settings you must configure in your Unity project.
-
-## Provisioning
-
-This is required for all iOS projects, so anyone with experience setting up iOS projects in Unity should be familiar with it. If you are new to developing for iOS on Unity, please follow [Unity's documentation on how to set up a project for iOS](https://docs.unity3d.com/Manual/iphone-GettingStarted.html).
-
-With iOS selected as the active platform, open the Player Settings (Edit - Project Settings - Player). Scroll down to **Identification** and enter your provisioning details. The fastest way to get up and running is to enter your **Signing Team ID** and check **Automatically Sign**. For a description of all these fields, please refer to [Unity's documentation](https://docs.unity3d.com/Manual/class-PlayerSettingsiOS.html#Identification).
-
-## IL2CPP Code Generation setting
-
-If your version of Unity is older than 2022.1 (keeping in mind the minimum supported version is 2021.3) you will need to change the following setting:
-
-1. Open Build Settings (under File).
-2. If iOS is not already the active platform, select iOS and click **Switch Platform**.
-3. Change **IL2CPP Code Generation** to **Faster (smaller) builds**.
-
-The reason this must be changed is because the Flow SDK utilises generic sharing of value types. For a detailed description on this problem and how it has been fixed in 2022.1, please read this [Unity blog post](https://blog.unity.com/engine-platform/il2cpp-full-generic-sharing-in-unity-2022-1-beta).
-
-## Managed Stripping Level
-
-Similar to the previous setting, sometimes automatic code stripping on iOS can strip out functions that the optimizer thinks aren't needed, but they actually are. We highly recommend you change this setting to avoid any of these issues.
-
-1. Open Project Settings (under Edit).
-2. Go to the Player tab.
-3. Expand Other Settings and scroll down to Optimization.
-4. Change **Managed Stripping Level** to **Minimal**.
\ No newline at end of file
diff --git a/docs/tools/clients/unity-sdk/guides/wallet-connect.md b/docs/tools/clients/unity-sdk/guides/wallet-connect.md
deleted file mode 100644
index c017608a21..0000000000
--- a/docs/tools/clients/unity-sdk/guides/wallet-connect.md
+++ /dev/null
@@ -1,128 +0,0 @@
-# Wallet Connect
-
-## Introduction
-
-Wallet Connect is a production ready wallet provider which implements the `IWallet` interface. It's not an actual wallet itself - it's a bridge that connects your Unity game with Flow Wallets that support Wallet Connect.
-
-## What are Wallets?
-
-A wallet is a piece of software or hardware that stores the private key associated with a Flow account. The term *custodian* is used to refer to the party that stores the private key. Hardware Wallets (eg Ledger), typically USB devices, allow users to be their own custodian, whereas hosted software wallets (eg Dapper Wallet) act as the custodian on behalf of the user.
-
-## What is a Wallet Provider?
-
-In terms of the Flow SDK for Unity, a Wallet Provider is a class which implements the `IWallet` interface and allows users to interact with specific hardware or software wallets. This includes authenticating with a wallet, retrieving the user's Flow account address from the wallet, and requesting the wallet to sign transactions on behalf of the user.
-
-As of v2.0.0, the Flow SDK for Unity contains two wallet providers - [Dev Wallet](./dev-wallet.md) and Wallet Connect. Dev Wallet is a mock wallet provider to make development easier, while Wallet Connect connects to real wallets and is therefore used for production. You could also implement your own wallet provider by implementing the `IWallet` interface.
-
-## How to implement Wallet Connect
-
-To implement Wallet Connect, you must first register your project in the Wallet Connect dashboard to obtain a Project ID, then register the provider with the Flow SDK.
-
-### Obtain Project ID
-
-1. Go to [https://cloud.walletconnect.com/sign-in](https://cloud.walletconnect.com/sign-in) and sign in, or create an account if you don't have one.
-2. Click on New Project and provide a name.
-3. Copy the Project ID.
-
-### Registering
-
-Create an instance of WalletConnectProvider and initialize it with the required config, then register it with the Flow SDK. Here is an example:
-
-```csharp
-using DapperLabs.Flow.Sdk;
-using DapperLabs.Flow.Sdk.WalletConnect;
-using DapperLabs.Flow.Sdk.Crypto;
-
-IWallet walletProvider = ScriptableObject.CreateInstance();
-walletProvider.Init(new WalletConnectConfig
-{
- ProjectId = "xxxxxxxxxxxxxxxxxxxxx", // the Project ID from the previous step
- ProjectDescription = "An example project to showcase Wallet Connect", // a description for your project
- ProjectIconUrl = "https://walletconnect.com/meta/favicon.ico", // URL for an icon for your project
- ProjectName = "Dapper Unity Example", // the name of your project
- ProjectUrl = "https://dapperlabs.com" // URL for your project
-});
-FlowSDK.RegisterWalletProvider(walletProvider);
-```
-
-The description, icon, name and URL that you provide will appear in your user's wallet apps, when they connect the game to their wallet.
-
-### Authenticating
-
-The `IWallet.Authenticate` method is as follows:
-
-```csharp
-public void Authenticate(string username, System.Action OnAuthSuccess, System.Action OnAuthFailed);
-```
-
-`username` is ignored in the Wallet Connect provider.
-`OnAuthSuccess` is a function that will be called when you have successfully authenticated with a wallet. The callback function must take a `string` argument, which will contain the authenticated account's Flow address.
-`OnAuthFailed` is a function that will be called if authentication failed, for example if the user rejected the request in their wallet app.
-
-Here is an example of authenticating from game code:
-
-```csharp
-FlowSDK.GetWalletProvider().Authenticate("", (string flowAddress) =>
-{
- Debug.Log($"Authenticated - Flow account address is {flowAddress}");
-}, () =>
-{
- Debug.Log("Authentication failed.");
-});
-```
-
-### Signing Transactions
-
-If you are using the Flow SDK to sign transactions then you do not need to worry about this, as it is handled automatically. When you submit a transaction, the SDK will request Wallet Connect to sign the transaction as the authenticated user. The user will receive a notification in their wallet app to approve the transaction.
-
-For full disclosure, here are the methods on the `IWallet` interface to sign a transaction:
-
-```csharp
-public Task SignTransactionPayload(FlowTransaction txn);
-
-public Task SignTransactionEnvelope(FlowTransaction txn);
-```
-
-In Flow, there are two parts of a transaction that can be signed - the Payload and the Authorization Envelope. The envelope must always be signed, and is the last thing to be signed by the Payer of the transaction fees. The Payload is only signed by the Proposer and\or the Authorizers IF they are not also the Payer (i.e. nobody signs the transaction twice). For more information on transaction signing, see [transaction signing](../../../../build/basics/transactions.md#signing-a-transaction).
-
-The following is an example of how to call `SignTransactionEnvelope`, but as mentioned, this is automatically done by the SDK's `Transactions.Submit` function. It is asynchronous so is therefore `await`ed, and returns the signature as a byte array.
-
-```csharp
-byte[] signature = await FlowSDK.GetWalletProvider().SignTransactionEnvelope(txRequest);
-```
-
-## What your users will see
-
-When your game calls `Authenticate`, a dialog box will appear. The contents of the dialog box will depend on what platform the user is on.
-
-### Desktop builds (Windows, MacOS)
-
-The authenticate dialog box will contain a QR code. The user must scan the QR code with their wallet app, after which their app will ask them to approve the connection with your game. Once they have approved the connection, they will receive any transaction requests submitted by the game in their wallet app. They can approve or reject these transaction requests.
-
-### Mobile builds (iOS, Android)
-
-The authenticate dialog box will contain a list of wallet apps that support Flow and Wallet Connect. If an app is installed on the device, it will say `Installed` next to the name. Selecting an app will either open the app if it's installed, or direct the user to download the app from the App Store (iOS) or Play Store (Android). If the app was installed and is opened, the user can approve the connection with your game. Once they have approved the connection, they will receive any transaction requests submitted by the game in their wallet app. They can approve or reject these transaction requests.
-
-## Customising the Authentication Dialogs
-
-You can customise the UI of both the QR Code (desktop builds) and Wallet Select (mobile builds) dialogs, allowing you to keep the same UI theme as the rest of your game. To do this, supply your custom prefabs to the Wallet Connect Config object during initialization, such as in the following example:
-
-```csharp
-// Register WalletConnect
-IWallet walletProvider = new WalletConnectProvider();
-walletProvider.Init(new WalletConnectConfig
-{
- ProjectId = "xxxxxxxxxxxxxxxxxxxxx",
- ProjectDescription = "An example project to showcase Wallet Connect",
- ProjectIconUrl = "https://walletconnect.com/meta/favicon.ico",
- ProjectName = "Dapper Unity Example",
- ProjectUrl = "https://dapperlabs.com"
- QrCodeDialogPrefab = qrCodeCustomPrefab, // custom prefab for QR Code dialog (desktop builds)
- WalletSelectDialogPrefab = walletSelectCustomPrefab // custom prefab for Wallet Select dialog (mobile builds)
-});
-FlowSDK.RegisterWalletProvider(walletProvider);
-```
-
-> **Note**: \
-Your custom QR Code prefab must have the `QRCodeDialog` script added as a component. \
-Your custom Wallet Select prefab must have the `WalletSelectDialog` script added as a component.
diff --git a/docs/tools/clients/unity-sdk/index.md b/docs/tools/clients/unity-sdk/index.md
deleted file mode 100644
index 014447fbd5..0000000000
--- a/docs/tools/clients/unity-sdk/index.md
+++ /dev/null
@@ -1,84 +0,0 @@
-# Flow SDK for Unity
-
-The Flow SDK for Unity allows Unity developers to integrate their games and applications with the Flow blockchain.
-
-## Introduction
-
-As games developers ourselves, we understand that most games developers donβt have any background in blockchain technology or even understand why it is used. The Flow SDK for Unity has been developed by games developers, for games developers. Our aim is to make blockchain integration as easy as possible, so that you can focus on making a great game.
-
-First, it helps to understand why you would want to make a game on blockchain, and what you can do.
-
-## Why Blockchain?
-
-Here at Dapper Labs weβre big believers in Web3. The two main advantages of blockchain that we believe in are:
-
-**Trustlessness** β there is no trust required in any particular platform or service. Once something is on the blockchain, itβs there to stay. Users have full confidence that the data you put on-chain will always be available β there are no servers or databases that can be taken down.
-
-**Composability** β once your Smart Contracts have been deployed, other developers can utilize them. They can deploy additional Smart Contracts that leverage your own, or develop other client applications which use them. Creating composable pieces of software like this gives new experiences to existing audiences β it's a win-win for everyone.
-
-## So blockchains are awesome... why Flow?
-
-Flow is a fast, decentralized and developer-friendly layer-1 blockchain. It was created to solve problems that Dapper Labs encountered while scaling CryptoKitties on Ethereum. Flow differs from other blockchains because it was explicitly designed to support games and consumer applications on day one, with the throughput necessary to scale to millions of active users. Some points which are unique to Flow include:
-
-- Multi-node architecture which facilitates scalability
-- Very low transaction fees
-- Eco-friendly due to its proof-of-stake consensus and multi-node architecture
-- Cadence β Flow's Smart Contract language, a resource-oriented programming language which has in-built security features
-- Upgradable Smart Contracts
-
-## Overview of the Flow SDK
-
-The Flow SDK allows Unity games to read and write from\\to a Flow blockchain. There are three chains\\environments\\networks developers should be aware of:
-
-- Emulator β this is an executable you can run locally, or on your own network, which emulates a Flow blockchain.
-- Testnet β the publicly available Flow blockchain used for testing.
-- Mainnet β the publicly available Flow blockchain used for production.
-
-We highly recommend you do most of your development against a local emulator, and only deploy to Testnet when your Smart Contracts are finalised. While Flow does have βupgradable Smart Contractsβ, there are many caveats to this, and you should get into the mindset that Smart Contracts are completely immutable once deployed. On an emulator, you can simply erase the entire chain and start again.
-
-The types of things you can do with the Flow blockchain include:
-
-- Mint, burn and trade NFTs and Fungible Tokens
-- Store game data
-- Read any publicly available information
-- Run game logic
-
-It is completely up to you, the developer, to decide how much data\\logic you want to store\\run on-chain, and how much (if any) on off-chain architecture such as servers and databases. We encourage you to experiment on an emulator, and when youβre ready, test on Testnet to get an indication of performance.
-
-## Adding the Flow SDK
-
-To add the Flow SDK for Unity, go to it in the Asset Store and add it to My Assets: [https://assetstore.unity.com/packages/decentralization/flow-sdk-237172](https://assetstore.unity.com/packages/decentralization/flow-sdk-237172)
-
-Once you've added it to your account, follow these instructions to import it into your project: [https://docs.unity3d.com/Manual/upm-ui-import.html](https://docs.unity3d.com/Manual/upm-ui-import.html)
-
-Alternatively, the Flow SDK for Unity is hosted on github here: [https://github.com/onflow/UnityFlowSDK](https://github.com/onflow/UnityFlowSDK)
-
-To add it to your Unity project directly from the github repo:
-
-1. In the Unity Editor, go to Window, Package Manager.
-2. Click the + dropdown in the top left corner.
-4. Select "Add package from git URL" and enter `https://github.com/onflow/UnityFlowSDK.git`.
-
-## Updating the Flow SDK
-
-Check the Package Manager regularly for updates to the Flow SDK.
-
-1. In the Unity Editor, go to Window, Package Manager.
-2. Under Packages - Dapper Labs, select Flow SDK.
-3. Click Update. If there is no Update button then you are already up to date.
-
-## Samples
-
-To add any of the SDKs Samples, click the related import button, from the Flow SDK package panel, in the Unity Package Manager window.
-
-## Requirements
-
-The Flow SDK is compatible with Unity version 2021.3 or higher.
-
-Supported Platforms: Windows, OSX, Android
-
-## Support
-
-Email: unity-sdk-support@dapperlabs.com
-
-Discord: https://discord.gg/flow
diff --git a/docs/tools/clients/unity-sdk/license.md b/docs/tools/clients/unity-sdk/license.md
deleted file mode 100644
index 99ca02f7a7..0000000000
--- a/docs/tools/clients/unity-sdk/license.md
+++ /dev/null
@@ -1,14 +0,0 @@
----
-title: License
-sidebar_position: 4
----
-
-MIT License
-
-Copyright 2023 Dapper Labs
-
-Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:
-
-The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software.
-
-THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
\ No newline at end of file
diff --git a/docs/tools/clients/unity-sdk/media/079fee6ed2cb0777c5ee99322d3d6039.png b/docs/tools/clients/unity-sdk/media/079fee6ed2cb0777c5ee99322d3d6039.png
deleted file mode 100644
index 91f241a615..0000000000
Binary files a/docs/tools/clients/unity-sdk/media/079fee6ed2cb0777c5ee99322d3d6039.png and /dev/null differ
diff --git a/docs/tools/clients/unity-sdk/media/079fee6ed2cb0777c5ee99322d3d6040.png b/docs/tools/clients/unity-sdk/media/079fee6ed2cb0777c5ee99322d3d6040.png
deleted file mode 100644
index 429035c13d..0000000000
Binary files a/docs/tools/clients/unity-sdk/media/079fee6ed2cb0777c5ee99322d3d6040.png and /dev/null differ
diff --git a/docs/tools/clients/unity-sdk/media/079fee6ed2cb0777c5ee99322d3d6041.png b/docs/tools/clients/unity-sdk/media/079fee6ed2cb0777c5ee99322d3d6041.png
deleted file mode 100644
index 318c7dc0f9..0000000000
Binary files a/docs/tools/clients/unity-sdk/media/079fee6ed2cb0777c5ee99322d3d6041.png and /dev/null differ
diff --git a/docs/tools/clients/unity-sdk/media/0933fe23c63eedcd9ba2c4f3aac2878c.png b/docs/tools/clients/unity-sdk/media/0933fe23c63eedcd9ba2c4f3aac2878c.png
deleted file mode 100644
index 0a46df5faa..0000000000
Binary files a/docs/tools/clients/unity-sdk/media/0933fe23c63eedcd9ba2c4f3aac2878c.png and /dev/null differ
diff --git a/docs/tools/clients/unity-sdk/media/0bc22650b47abb1cf606b2c7381a6377.png b/docs/tools/clients/unity-sdk/media/0bc22650b47abb1cf606b2c7381a6377.png
deleted file mode 100644
index dfad1b07ec..0000000000
Binary files a/docs/tools/clients/unity-sdk/media/0bc22650b47abb1cf606b2c7381a6377.png and /dev/null differ
diff --git a/docs/tools/clients/unity-sdk/media/15840cee138659b172d6023791b73155.png b/docs/tools/clients/unity-sdk/media/15840cee138659b172d6023791b73155.png
deleted file mode 100644
index b78e2d718f..0000000000
Binary files a/docs/tools/clients/unity-sdk/media/15840cee138659b172d6023791b73155.png and /dev/null differ
diff --git a/docs/tools/clients/unity-sdk/media/184e8a2ee8440b6b700072a9212e92bb.png b/docs/tools/clients/unity-sdk/media/184e8a2ee8440b6b700072a9212e92bb.png
deleted file mode 100644
index e60c288f2c..0000000000
Binary files a/docs/tools/clients/unity-sdk/media/184e8a2ee8440b6b700072a9212e92bb.png and /dev/null differ
diff --git a/docs/tools/clients/unity-sdk/media/2466ac711f979da15f6ad5a6cdb6421a.png b/docs/tools/clients/unity-sdk/media/2466ac711f979da15f6ad5a6cdb6421a.png
deleted file mode 100644
index 99079bcc7e..0000000000
Binary files a/docs/tools/clients/unity-sdk/media/2466ac711f979da15f6ad5a6cdb6421a.png and /dev/null differ
diff --git a/docs/tools/clients/unity-sdk/media/252129f67efc7d03357d9e5070d67ece.png b/docs/tools/clients/unity-sdk/media/252129f67efc7d03357d9e5070d67ece.png
deleted file mode 100644
index d5fd50f104..0000000000
Binary files a/docs/tools/clients/unity-sdk/media/252129f67efc7d03357d9e5070d67ece.png and /dev/null differ
diff --git a/docs/tools/clients/unity-sdk/media/270ed3d6ac5df815d35cc82f588a4a18.png b/docs/tools/clients/unity-sdk/media/270ed3d6ac5df815d35cc82f588a4a18.png
deleted file mode 100644
index f516dfbfa0..0000000000
Binary files a/docs/tools/clients/unity-sdk/media/270ed3d6ac5df815d35cc82f588a4a18.png and /dev/null differ
diff --git a/docs/tools/clients/unity-sdk/media/2ce641e868905901df937c4dd09de105.png b/docs/tools/clients/unity-sdk/media/2ce641e868905901df937c4dd09de105.png
deleted file mode 100644
index 85ad600a72..0000000000
Binary files a/docs/tools/clients/unity-sdk/media/2ce641e868905901df937c4dd09de105.png and /dev/null differ
diff --git a/docs/tools/clients/unity-sdk/media/2e08cbac6f2ff6a2cde89e7cd6c6be62.png b/docs/tools/clients/unity-sdk/media/2e08cbac6f2ff6a2cde89e7cd6c6be62.png
deleted file mode 100644
index 6fb49cc012..0000000000
Binary files a/docs/tools/clients/unity-sdk/media/2e08cbac6f2ff6a2cde89e7cd6c6be62.png and /dev/null differ
diff --git a/docs/tools/clients/unity-sdk/media/3421ca4434716e8f2efc89bbf5f3c7ad.png b/docs/tools/clients/unity-sdk/media/3421ca4434716e8f2efc89bbf5f3c7ad.png
deleted file mode 100644
index 7c85a41037..0000000000
Binary files a/docs/tools/clients/unity-sdk/media/3421ca4434716e8f2efc89bbf5f3c7ad.png and /dev/null differ
diff --git a/docs/tools/clients/unity-sdk/media/35f927dd8c61a096c387264c358eba6a.png b/docs/tools/clients/unity-sdk/media/35f927dd8c61a096c387264c358eba6a.png
deleted file mode 100644
index 0d8c777b06..0000000000
Binary files a/docs/tools/clients/unity-sdk/media/35f927dd8c61a096c387264c358eba6a.png and /dev/null differ
diff --git a/docs/tools/clients/unity-sdk/media/4444607a3effafd2c5bb8430b8fade3f.png b/docs/tools/clients/unity-sdk/media/4444607a3effafd2c5bb8430b8fade3f.png
deleted file mode 100644
index e87e7ed378..0000000000
Binary files a/docs/tools/clients/unity-sdk/media/4444607a3effafd2c5bb8430b8fade3f.png and /dev/null differ
diff --git a/docs/tools/clients/unity-sdk/media/4eaf391ee18bc8b70452e4495b5b317a.png b/docs/tools/clients/unity-sdk/media/4eaf391ee18bc8b70452e4495b5b317a.png
deleted file mode 100644
index e10342b6b5..0000000000
Binary files a/docs/tools/clients/unity-sdk/media/4eaf391ee18bc8b70452e4495b5b317a.png and /dev/null differ
diff --git a/docs/tools/clients/unity-sdk/media/51d4f1e5f01dc36b0fdd0853eb08b0bd.png b/docs/tools/clients/unity-sdk/media/51d4f1e5f01dc36b0fdd0853eb08b0bd.png
deleted file mode 100644
index 978a72cc50..0000000000
Binary files a/docs/tools/clients/unity-sdk/media/51d4f1e5f01dc36b0fdd0853eb08b0bd.png and /dev/null differ
diff --git a/docs/tools/clients/unity-sdk/media/54f8a5235b61d70f09904c38ed3e5e5c.png b/docs/tools/clients/unity-sdk/media/54f8a5235b61d70f09904c38ed3e5e5c.png
deleted file mode 100644
index 8a2579c91e..0000000000
Binary files a/docs/tools/clients/unity-sdk/media/54f8a5235b61d70f09904c38ed3e5e5c.png and /dev/null differ
diff --git a/docs/tools/clients/unity-sdk/media/5569416d261ff8e9d05b53c082c8a1b1.png b/docs/tools/clients/unity-sdk/media/5569416d261ff8e9d05b53c082c8a1b1.png
deleted file mode 100644
index da39d30c54..0000000000
Binary files a/docs/tools/clients/unity-sdk/media/5569416d261ff8e9d05b53c082c8a1b1.png and /dev/null differ
diff --git a/docs/tools/clients/unity-sdk/media/55bad7978ba3915550c8367b319a9944.png b/docs/tools/clients/unity-sdk/media/55bad7978ba3915550c8367b319a9944.png
deleted file mode 100644
index a64d81cede..0000000000
Binary files a/docs/tools/clients/unity-sdk/media/55bad7978ba3915550c8367b319a9944.png and /dev/null differ
diff --git a/docs/tools/clients/unity-sdk/media/58d03fbcda193c672c0287da2f3f04ba.png b/docs/tools/clients/unity-sdk/media/58d03fbcda193c672c0287da2f3f04ba.png
deleted file mode 100644
index c7970487b6..0000000000
Binary files a/docs/tools/clients/unity-sdk/media/58d03fbcda193c672c0287da2f3f04ba.png and /dev/null differ
diff --git a/docs/tools/clients/unity-sdk/media/5ba73beddcefba8eade5c9ce4322a86d.png b/docs/tools/clients/unity-sdk/media/5ba73beddcefba8eade5c9ce4322a86d.png
deleted file mode 100644
index a05b117a7d..0000000000
Binary files a/docs/tools/clients/unity-sdk/media/5ba73beddcefba8eade5c9ce4322a86d.png and /dev/null differ
diff --git a/docs/tools/clients/unity-sdk/media/61267a392954311426b4144345f25dfd.png b/docs/tools/clients/unity-sdk/media/61267a392954311426b4144345f25dfd.png
deleted file mode 100644
index b97cdf2a2f..0000000000
Binary files a/docs/tools/clients/unity-sdk/media/61267a392954311426b4144345f25dfd.png and /dev/null differ
diff --git a/docs/tools/clients/unity-sdk/media/6279d33bd3d9eea312ef2f9b05000dbf.png b/docs/tools/clients/unity-sdk/media/6279d33bd3d9eea312ef2f9b05000dbf.png
deleted file mode 100644
index 41a57f0920..0000000000
Binary files a/docs/tools/clients/unity-sdk/media/6279d33bd3d9eea312ef2f9b05000dbf.png and /dev/null differ
diff --git a/docs/tools/clients/unity-sdk/media/62ec9ee52b31041e7420ac7ad7132684.png b/docs/tools/clients/unity-sdk/media/62ec9ee52b31041e7420ac7ad7132684.png
deleted file mode 100644
index d8cbfacee6..0000000000
Binary files a/docs/tools/clients/unity-sdk/media/62ec9ee52b31041e7420ac7ad7132684.png and /dev/null differ
diff --git a/docs/tools/clients/unity-sdk/media/6d1ca6e13da7d1c65496fd803a3c5f78.png b/docs/tools/clients/unity-sdk/media/6d1ca6e13da7d1c65496fd803a3c5f78.png
deleted file mode 100644
index c17adc6cac..0000000000
Binary files a/docs/tools/clients/unity-sdk/media/6d1ca6e13da7d1c65496fd803a3c5f78.png and /dev/null differ
diff --git a/docs/tools/clients/unity-sdk/media/6d33d20949b4923d1e805e8e21884935.png b/docs/tools/clients/unity-sdk/media/6d33d20949b4923d1e805e8e21884935.png
deleted file mode 100644
index 8a43e38f78..0000000000
Binary files a/docs/tools/clients/unity-sdk/media/6d33d20949b4923d1e805e8e21884935.png and /dev/null differ
diff --git a/docs/tools/clients/unity-sdk/media/741a3e12d817f800c2bced3c927912ba.png b/docs/tools/clients/unity-sdk/media/741a3e12d817f800c2bced3c927912ba.png
deleted file mode 100644
index 594db4701c..0000000000
Binary files a/docs/tools/clients/unity-sdk/media/741a3e12d817f800c2bced3c927912ba.png and /dev/null differ
diff --git a/docs/tools/clients/unity-sdk/media/7e85df9713fbb484d0955d698ccd1668.png b/docs/tools/clients/unity-sdk/media/7e85df9713fbb484d0955d698ccd1668.png
deleted file mode 100644
index 7bfab1aec2..0000000000
Binary files a/docs/tools/clients/unity-sdk/media/7e85df9713fbb484d0955d698ccd1668.png and /dev/null differ
diff --git a/docs/tools/clients/unity-sdk/media/814d81d58a609d94bc2d7bc9a6e90afe.png b/docs/tools/clients/unity-sdk/media/814d81d58a609d94bc2d7bc9a6e90afe.png
deleted file mode 100644
index 076b06be17..0000000000
Binary files a/docs/tools/clients/unity-sdk/media/814d81d58a609d94bc2d7bc9a6e90afe.png and /dev/null differ
diff --git a/docs/tools/clients/unity-sdk/media/82ea27a997c12eab231ad8bf2e49d335.png b/docs/tools/clients/unity-sdk/media/82ea27a997c12eab231ad8bf2e49d335.png
deleted file mode 100644
index 35e517c6c4..0000000000
Binary files a/docs/tools/clients/unity-sdk/media/82ea27a997c12eab231ad8bf2e49d335.png and /dev/null differ
diff --git a/docs/tools/clients/unity-sdk/media/84ffa7104bc4de009c328960c1c9c6e7.png b/docs/tools/clients/unity-sdk/media/84ffa7104bc4de009c328960c1c9c6e7.png
deleted file mode 100644
index dba674dc41..0000000000
Binary files a/docs/tools/clients/unity-sdk/media/84ffa7104bc4de009c328960c1c9c6e7.png and /dev/null differ
diff --git a/docs/tools/clients/unity-sdk/media/881d1efa0c8e4ff9bffe3382274a255c.png b/docs/tools/clients/unity-sdk/media/881d1efa0c8e4ff9bffe3382274a255c.png
deleted file mode 100644
index 7b86d6d77a..0000000000
Binary files a/docs/tools/clients/unity-sdk/media/881d1efa0c8e4ff9bffe3382274a255c.png and /dev/null differ
diff --git a/docs/tools/clients/unity-sdk/media/894e40fa8ac7a3ebbc60e44c60be135a.png b/docs/tools/clients/unity-sdk/media/894e40fa8ac7a3ebbc60e44c60be135a.png
deleted file mode 100644
index 7d6eb37db9..0000000000
Binary files a/docs/tools/clients/unity-sdk/media/894e40fa8ac7a3ebbc60e44c60be135a.png and /dev/null differ
diff --git a/docs/tools/clients/unity-sdk/media/8cd5f4e2d4ac66693e0c2a6f796d1b38.png b/docs/tools/clients/unity-sdk/media/8cd5f4e2d4ac66693e0c2a6f796d1b38.png
deleted file mode 100644
index bc3f78265f..0000000000
Binary files a/docs/tools/clients/unity-sdk/media/8cd5f4e2d4ac66693e0c2a6f796d1b38.png and /dev/null differ
diff --git a/docs/tools/clients/unity-sdk/media/999cca57ee98b89bfe5f012ee445766c.png b/docs/tools/clients/unity-sdk/media/999cca57ee98b89bfe5f012ee445766c.png
deleted file mode 100644
index 542df16076..0000000000
Binary files a/docs/tools/clients/unity-sdk/media/999cca57ee98b89bfe5f012ee445766c.png and /dev/null differ
diff --git a/docs/tools/clients/unity-sdk/media/999cca57ee98b89bfe5f012ee445766d.png b/docs/tools/clients/unity-sdk/media/999cca57ee98b89bfe5f012ee445766d.png
deleted file mode 100644
index 26268c1944..0000000000
Binary files a/docs/tools/clients/unity-sdk/media/999cca57ee98b89bfe5f012ee445766d.png and /dev/null differ
diff --git a/docs/tools/clients/unity-sdk/media/AppendixDeployContract.png b/docs/tools/clients/unity-sdk/media/AppendixDeployContract.png
deleted file mode 100644
index 0095d61632..0000000000
Binary files a/docs/tools/clients/unity-sdk/media/AppendixDeployContract.png and /dev/null differ
diff --git a/docs/tools/clients/unity-sdk/media/AppendixNewAccount.png b/docs/tools/clients/unity-sdk/media/AppendixNewAccount.png
deleted file mode 100644
index 179e1fc0da..0000000000
Binary files a/docs/tools/clients/unity-sdk/media/AppendixNewAccount.png and /dev/null differ
diff --git a/docs/tools/clients/unity-sdk/media/AppendixTextReplacement.png b/docs/tools/clients/unity-sdk/media/AppendixTextReplacement.png
deleted file mode 100644
index 7929157caa..0000000000
Binary files a/docs/tools/clients/unity-sdk/media/AppendixTextReplacement.png and /dev/null differ
diff --git a/docs/tools/clients/unity-sdk/media/FlowControlAccounts.png b/docs/tools/clients/unity-sdk/media/FlowControlAccounts.png
deleted file mode 100644
index eb393676ac..0000000000
Binary files a/docs/tools/clients/unity-sdk/media/FlowControlAccounts.png and /dev/null differ
diff --git a/docs/tools/clients/unity-sdk/media/FlowControlDataAsset.png b/docs/tools/clients/unity-sdk/media/FlowControlDataAsset.png
deleted file mode 100644
index 881a0dfcec..0000000000
Binary files a/docs/tools/clients/unity-sdk/media/FlowControlDataAsset.png and /dev/null differ
diff --git a/docs/tools/clients/unity-sdk/media/FlowControlEmulator.png b/docs/tools/clients/unity-sdk/media/FlowControlEmulator.png
deleted file mode 100644
index ea76b99074..0000000000
Binary files a/docs/tools/clients/unity-sdk/media/FlowControlEmulator.png and /dev/null differ
diff --git a/docs/tools/clients/unity-sdk/media/FlowControlInspector.png b/docs/tools/clients/unity-sdk/media/FlowControlInspector.png
deleted file mode 100644
index 5ce9a5d081..0000000000
Binary files a/docs/tools/clients/unity-sdk/media/FlowControlInspector.png and /dev/null differ
diff --git a/docs/tools/clients/unity-sdk/media/FlowControlTextReplacement.png b/docs/tools/clients/unity-sdk/media/FlowControlTextReplacement.png
deleted file mode 100644
index 218dc7d40c..0000000000
Binary files a/docs/tools/clients/unity-sdk/media/FlowControlTextReplacement.png and /dev/null differ
diff --git a/docs/tools/clients/unity-sdk/media/FlowControlTools.png b/docs/tools/clients/unity-sdk/media/FlowControlTools.png
deleted file mode 100644
index 073e87cdfa..0000000000
Binary files a/docs/tools/clients/unity-sdk/media/FlowControlTools.png and /dev/null differ
diff --git a/docs/tools/clients/unity-sdk/media/Lilico_Settings.jpg b/docs/tools/clients/unity-sdk/media/Lilico_Settings.jpg
deleted file mode 100644
index 9cdad3e6d4..0000000000
Binary files a/docs/tools/clients/unity-sdk/media/Lilico_Settings.jpg and /dev/null differ
diff --git a/docs/tools/clients/unity-sdk/media/Lilico_TestnetEnable.jpg b/docs/tools/clients/unity-sdk/media/Lilico_TestnetEnable.jpg
deleted file mode 100644
index cc34602410..0000000000
Binary files a/docs/tools/clients/unity-sdk/media/Lilico_TestnetEnable.jpg and /dev/null differ
diff --git a/docs/tools/clients/unity-sdk/media/aeda525491f7b1b329e57ac7d07fa997.png b/docs/tools/clients/unity-sdk/media/aeda525491f7b1b329e57ac7d07fa997.png
deleted file mode 100644
index 79b00a7689..0000000000
Binary files a/docs/tools/clients/unity-sdk/media/aeda525491f7b1b329e57ac7d07fa997.png and /dev/null differ
diff --git a/docs/tools/clients/unity-sdk/media/b33bd58a6b1085948b1fbc9b40b2f2b3.png b/docs/tools/clients/unity-sdk/media/b33bd58a6b1085948b1fbc9b40b2f2b3.png
deleted file mode 100644
index 0727939486..0000000000
Binary files a/docs/tools/clients/unity-sdk/media/b33bd58a6b1085948b1fbc9b40b2f2b3.png and /dev/null differ
diff --git a/docs/tools/clients/unity-sdk/media/bac9d81383b8aeacab6e2f757f6c5f1c.png b/docs/tools/clients/unity-sdk/media/bac9d81383b8aeacab6e2f757f6c5f1c.png
deleted file mode 100644
index 001ff814c9..0000000000
Binary files a/docs/tools/clients/unity-sdk/media/bac9d81383b8aeacab6e2f757f6c5f1c.png and /dev/null differ
diff --git a/docs/tools/clients/unity-sdk/media/bd9262a67638aa9c519921629e8c9075.png b/docs/tools/clients/unity-sdk/media/bd9262a67638aa9c519921629e8c9075.png
deleted file mode 100644
index afa7f72039..0000000000
Binary files a/docs/tools/clients/unity-sdk/media/bd9262a67638aa9c519921629e8c9075.png and /dev/null differ
diff --git a/docs/tools/clients/unity-sdk/media/bd9b6a0a7261d3c605aad9760a304220.png b/docs/tools/clients/unity-sdk/media/bd9b6a0a7261d3c605aad9760a304220.png
deleted file mode 100644
index 5fbf304deb..0000000000
Binary files a/docs/tools/clients/unity-sdk/media/bd9b6a0a7261d3c605aad9760a304220.png and /dev/null differ
diff --git a/docs/tools/clients/unity-sdk/media/be2bc6569ca5a475b3c147cf34f07c1a.png b/docs/tools/clients/unity-sdk/media/be2bc6569ca5a475b3c147cf34f07c1a.png
deleted file mode 100644
index 9c0b7e2702..0000000000
Binary files a/docs/tools/clients/unity-sdk/media/be2bc6569ca5a475b3c147cf34f07c1a.png and /dev/null differ
diff --git a/docs/tools/clients/unity-sdk/media/c680c8f205102d2dd4254e0a5a4664b0.png b/docs/tools/clients/unity-sdk/media/c680c8f205102d2dd4254e0a5a4664b0.png
deleted file mode 100644
index 77a365a70c..0000000000
Binary files a/docs/tools/clients/unity-sdk/media/c680c8f205102d2dd4254e0a5a4664b0.png and /dev/null differ
diff --git a/docs/tools/clients/unity-sdk/media/cc91513990fc19f6348a32ba3770c543.png b/docs/tools/clients/unity-sdk/media/cc91513990fc19f6348a32ba3770c543.png
deleted file mode 100644
index f96e553926..0000000000
Binary files a/docs/tools/clients/unity-sdk/media/cc91513990fc19f6348a32ba3770c543.png and /dev/null differ
diff --git a/docs/tools/clients/unity-sdk/media/d4ef2f464230b54400ff46b92f2eb01e.png b/docs/tools/clients/unity-sdk/media/d4ef2f464230b54400ff46b92f2eb01e.png
deleted file mode 100644
index bc390b442c..0000000000
Binary files a/docs/tools/clients/unity-sdk/media/d4ef2f464230b54400ff46b92f2eb01e.png and /dev/null differ
diff --git a/docs/tools/clients/unity-sdk/media/db2741c083bc677b767e5f2973f63e1b.png b/docs/tools/clients/unity-sdk/media/db2741c083bc677b767e5f2973f63e1b.png
deleted file mode 100644
index a2e9a41edf..0000000000
Binary files a/docs/tools/clients/unity-sdk/media/db2741c083bc677b767e5f2973f63e1b.png and /dev/null differ
diff --git a/docs/tools/clients/unity-sdk/media/e6a3c62a95440af5c38b47c9efa27a72.png b/docs/tools/clients/unity-sdk/media/e6a3c62a95440af5c38b47c9efa27a72.png
deleted file mode 100644
index 9bbfae2b07..0000000000
Binary files a/docs/tools/clients/unity-sdk/media/e6a3c62a95440af5c38b47c9efa27a72.png and /dev/null differ
diff --git a/docs/tools/clients/unity-sdk/media/e81ec10f19a1f9d51f1ffaa102115244.png b/docs/tools/clients/unity-sdk/media/e81ec10f19a1f9d51f1ffaa102115244.png
deleted file mode 100644
index 5ac70245e6..0000000000
Binary files a/docs/tools/clients/unity-sdk/media/e81ec10f19a1f9d51f1ffaa102115244.png and /dev/null differ
diff --git a/docs/tools/clients/unity-sdk/media/e96f948858a786b5e26fa08cbcc2c260.png b/docs/tools/clients/unity-sdk/media/e96f948858a786b5e26fa08cbcc2c260.png
deleted file mode 100644
index 3c51e693f0..0000000000
Binary files a/docs/tools/clients/unity-sdk/media/e96f948858a786b5e26fa08cbcc2c260.png and /dev/null differ
diff --git a/docs/tools/clients/unity-sdk/media/eb0fba4dbd9a040424afb3182e5ce345.png b/docs/tools/clients/unity-sdk/media/eb0fba4dbd9a040424afb3182e5ce345.png
deleted file mode 100644
index 20cfd357ef..0000000000
Binary files a/docs/tools/clients/unity-sdk/media/eb0fba4dbd9a040424afb3182e5ce345.png and /dev/null differ
diff --git a/docs/tools/clients/unity-sdk/media/ecac86c8af7414ac22e95f64340acc44.png b/docs/tools/clients/unity-sdk/media/ecac86c8af7414ac22e95f64340acc44.png
deleted file mode 100644
index 29d7d4e2bc..0000000000
Binary files a/docs/tools/clients/unity-sdk/media/ecac86c8af7414ac22e95f64340acc44.png and /dev/null differ
diff --git a/docs/tools/clients/unity-sdk/media/ed8abd462a98bd71a9abe6a66c04d958.png b/docs/tools/clients/unity-sdk/media/ed8abd462a98bd71a9abe6a66c04d958.png
deleted file mode 100644
index 0fe1eadf07..0000000000
Binary files a/docs/tools/clients/unity-sdk/media/ed8abd462a98bd71a9abe6a66c04d958.png and /dev/null differ
diff --git a/docs/tools/clients/unity-sdk/media/f0dfb75d21cf7462d7f0aaf199d97b27.png b/docs/tools/clients/unity-sdk/media/f0dfb75d21cf7462d7f0aaf199d97b27.png
deleted file mode 100644
index e8d0f4d010..0000000000
Binary files a/docs/tools/clients/unity-sdk/media/f0dfb75d21cf7462d7f0aaf199d97b27.png and /dev/null differ
diff --git a/docs/tools/clients/unity-sdk/media/f30282a8889db37965bcd49b209ac614.png b/docs/tools/clients/unity-sdk/media/f30282a8889db37965bcd49b209ac614.png
deleted file mode 100644
index 10bc679e4d..0000000000
Binary files a/docs/tools/clients/unity-sdk/media/f30282a8889db37965bcd49b209ac614.png and /dev/null differ
diff --git a/docs/tools/clients/unity-sdk/media/f6a8324547c8ec499a2eb68248091ba3.png b/docs/tools/clients/unity-sdk/media/f6a8324547c8ec499a2eb68248091ba3.png
deleted file mode 100644
index 4c09790bd9..0000000000
Binary files a/docs/tools/clients/unity-sdk/media/f6a8324547c8ec499a2eb68248091ba3.png and /dev/null differ
diff --git a/docs/tools/clients/unity-sdk/media/f8bf765df950289fd3d1b7dc8e73d545.png b/docs/tools/clients/unity-sdk/media/f8bf765df950289fd3d1b7dc8e73d545.png
deleted file mode 100644
index 4a73f6228e..0000000000
Binary files a/docs/tools/clients/unity-sdk/media/f8bf765df950289fd3d1b7dc8e73d545.png and /dev/null differ
diff --git a/docs/tools/clients/unity-sdk/media/fd52a768f2367b763980b22d85479305.png b/docs/tools/clients/unity-sdk/media/fd52a768f2367b763980b22d85479305.png
deleted file mode 100644
index 1cef1b26fb..0000000000
Binary files a/docs/tools/clients/unity-sdk/media/fd52a768f2367b763980b22d85479305.png and /dev/null differ
diff --git a/docs/tools/clients/unity-sdk/media/fe0ccbbb8fb583dd7c461a1436d9206c.png b/docs/tools/clients/unity-sdk/media/fe0ccbbb8fb583dd7c461a1436d9206c.png
deleted file mode 100644
index 0609be63a5..0000000000
Binary files a/docs/tools/clients/unity-sdk/media/fe0ccbbb8fb583dd7c461a1436d9206c.png and /dev/null differ
diff --git a/docs/tools/clients/unity-sdk/samples/_category_.yml b/docs/tools/clients/unity-sdk/samples/_category_.yml
deleted file mode 100644
index 1d2ab7fc93..0000000000
--- a/docs/tools/clients/unity-sdk/samples/_category_.yml
+++ /dev/null
@@ -1,2 +0,0 @@
-label: Samples
-position: 3
\ No newline at end of file
diff --git a/docs/tools/clients/unity-sdk/samples/flow-words-sample.md b/docs/tools/clients/unity-sdk/samples/flow-words-sample.md
deleted file mode 100644
index e4dbfd01b4..0000000000
--- a/docs/tools/clients/unity-sdk/samples/flow-words-sample.md
+++ /dev/null
@@ -1,30 +0,0 @@
-# Flow Unity SDK - FlowWords Sample
-
-This sample is a working pre-deployed version of our FlowWords tutorial sample, running on FLOW Testnet.\
-To use this sample you will need a WalletConnect compatible Testnet wallet. e.g. Lilico.
-
-If you do not already have a compatible wallet, you can sign up for one here: [https://lilico.app](https://lilico.app)\
-You will need to install the mobile phone app. (iOS or Android)
-
-Once installed, you will need to enable dev mode in the Lilico app, do so as follows;
-1. Go to the settings menu, by clicking the profile icon on the bottom right corner of the screen.
-2. Select Developer Mode to enter the Developer mode submenu
-
- ![Lilico Settings](../media/Lilico_Settings.jpg)
-
-3. Enable Developer Mode with the toggle at the top of the screen, and then set the network to TestNet.
-
- ![Lilico Enable Devmode](../media/Lilico_TestnetEnable.jpg)
-
-4. Back out to the wallet dashboard by clicking the back button at the top left of the screen, then clicking the Dashboard button (small $) in the bottom left corner of the screen.
-5. At this point your wallet should say TESTNET at the top of the screen. If your FLOW balance is zero, you can fund your Testnet account via the FLOW Testnet Faucet here: [https://testnet-faucet.onflow.org/fund-account](https://testnet-faucet.onflow.org/fund-account)
- 1. Enter your Testnet account address, as shown in the Lilico app
- 2. Complete the captcha, and click 'Fund Your Account'
- 3. Refresh your Lilico app on your device to see your new FLOW balance! (swipe down on the dashboard screen)
-
-We assume you have already added the SDK package to your Unity project, and Imported the FlowWords Sample using the Unity Package Manager.
-If you have not already done so, do this now. You can find instructions on how to do so [here.](../index.md#adding-the-flow-sdk)
-
-1. Open Game.scene from the FlowWords sample folder. (Samples\Flow SDK\\<version>\Flow Words\Scenes)
-2. Press Play in Unity to run the game sample.
-3. Whenever the game presents you with a QR code, scan the code with your mobile Wallet App, and approve the transactions on your mobile device!
\ No newline at end of file
diff --git a/docs/tools/clients/unity-sdk/samples/flow-words-tutorial.md b/docs/tools/clients/unity-sdk/samples/flow-words-tutorial.md
deleted file mode 100644
index 4381369967..0000000000
--- a/docs/tools/clients/unity-sdk/samples/flow-words-tutorial.md
+++ /dev/null
@@ -1,1192 +0,0 @@
-# Tutorial - How to Build FlowWords
-
-## Outline
-
-The FlowWords tutorial example project is the complete FlowWords game, with the FlowSDK and any SDK related code removed.
-
-In this tutorial, you will learn how to;
-
-1. Configure the FlowSDK and local emulator for use with Unity
-2. Deploy contracts, create Accounts and run Transactions on the Flow emulator from within Unity.
-3. Incorporate Flow Accounts, Transactions and Scripts into Unity projects using Code
-
-## Requirements
-
-FlowWords Tutorial has been tested on Unity version 2021.3.6f1
-
-This tutorial assumes you have created a blank project, and installed the FlowSDK package from the Unity Asset Store, and FlowWords Tutorial Sample.
-
-## Step 1 β Configure the FlowSDK and local emulator
-
-1. Open the scene at Assets\\Samples\\Flow SDK\\<version>\\Flow Words Tutorial Assets\\Scenes\\Game.unity
-
- ![Open Game scene example](../media/bac9d81383b8aeacab6e2f757f6c5f1c.png)
-
-2. Open the Flow Control Manager window, at Window-\>Flow-\>Flow Control.
- This will create a Flow Control GameObject in the Game scene, and a FlowControlData asset in the Resources folder if they do not already exist.
-
- ![Open Flow Control Manager example](../media/f8bf765df950289fd3d1b7dc8e73d545.png)
-
-3. Go to the Emulator Settings tab. If the manager states the Flow Executable cannot be found, use the Install button provided and follow the instructions to install the Flow CLI.
- Note: If Unity still does not detect the emulator after installing the Flow CLI, you may need to restart your computer for the install to take effect.
-4. Set a directory for our Emulator data to reside in. This is where the emulator stores the state of the emulated Flow blockchain.
- For the purposes of this tutorial, we will create a new folder called FlowEmulator inside our project folder.
-
- ![Set Emulator data directory example](../media/4eaf391ee18bc8b70452e4495b5b317a.png)
-
-5. Click Start Emulator to start the emulator and create the emulator_service_account, and ensure βRun emulator in play modeβ is checked.
-
- ![Start Emulator example](../media/4444607a3effafd2c5bb8430b8fade3f.png)
-
- NOTE: If this is the first time you have run the emulator on Windows, you may be presented with a dialog to allow access through your firewall. This is safe to allow.
-
-6. Go to the Accounts tab and verify that the service account has been created.
- This is the account we will use to create more accounts, and deploy our game contract on to the emulator.
-
- ![Flow Control Manager Accounts final state](../media/6279d33bd3d9eea312ef2f9b05000dbf.png)
-
-7. Back on the Emulator Settings tab, you can click Show Emulator Log, to view the output.
-
- ![Flow Emulator Output example](../media/d4ef2f464230b54400ff46b92f2eb01e.png)
-
-## Step 2 β Deploy Contracts, create Accounts and run Transactions
-
-We have provided you with the FlowWords game contract, but before we can interact with the contract, we have to deploy it to the blockchain.
-
-We also have to set up some text replacements as, once deployed, our scripts will require hardcoded references to the contract name and deployed address.
-
-### Set up Text Replacements
-
-Text replacements allow us to set and update references across all of our local Cadence scripts, without having to update each file individually.
-
-1. Open the Flow Control Manager window, and navigate to the Text Replacements tab.
-2. Set up the text replacements as follows. You can add text replacements by clicking the β+β button at the top of the panel.
-
- ![Flow Control Manager Text Replacement final state](../media/bd9262a67638aa9c519921629e8c9075.png)
-
- The anatomy of a text replacement is as follows;
- **Description**: A friendly description, has no bearing on functionality.
- **Original Text**: The original text, in your script files, which you want to replace.
- **Replacement Text**: The text that you wish to replace the original text with. This is what gets submitted to the chain.
- **Active**: This checkbox enables or disables the text replacement.
- **Apply to Accounts**: Select which, if any, accounts this text replacement should apply to.
- **Apply to Gateways**: Select which, if any, gateways (such as Emulator, or TestNet) this replacement should apply to.
-
-### Create User Accounts
-
-While it would be perfectly possible to play our game with the emulator service account, we will often want to test our contracts with multiple different user accounts.
-
-To create a new user account;
-
-1. Open the Flow Control Manager window, and navigate to the Tools tab.
-2. In the Create New Account section;
-
- ![Flow Control Manager new account creation example](../media/aeda525491f7b1b329e57ac7d07fa997.png)
-
- 1. Select the paying account. This will usually be the emulator_service_account
- 2. Enter a friendly name for the new account. This name is just for your own reference and does not get written to the chain.
- 3. Click Create
-3. If successful, the new account will appear under the Flow Control Manager Accounts tab.
-
- ![Flow Control Manager new account example final state](../media/84ffa7104bc4de009c328960c1c9c6e7.png)
-
-### Deploy the Contract
-
-Before anyone can interact with a contract, it must be deployed to the blockchain.
-
-We are going to deploy our game contract to the emulator, for local testing. But for deploying to Testnet, or Mainnet follow the same process.
-
-1. Go to the Emulator Settings tab, in the Flow Control Manager, and Start the Emulator if it is not already running.
-2. Go to the Tools tab. In the Manage Contracts section, enter the contract name of βFlowWordsβ β this should match our CONTRACT_NAME text replacement precisely.
-3. Populate the Contract field with game-contract.cdc, which can be found in Resources/contracts
-
- ![Resources folder example](../media/741a3e12d817f800c2bced3c927912ba.png)
-
-4. Finally, ensure that the Account field contains emulator_service_account, and click the Deploy Contract button.
-
- ![Flow Control Manager contract deployment example](../media/ed8abd462a98bd71a9abe6a66c04d958.png)
-
-5. To check if the contract has deployed successfully, open the Flow Emulator Output window. Successful deployment will look as follows;
-
- ![Flow Emulator Output example](../media/0933fe23c63eedcd9ba2c4f3aac2878c.png)
-
-### Submit Transactions
-
-For administration purposes, it is sometimes useful to be able to directly submit transactions to the chain.
-
-We can use a transaction to check out text replacements are set up correctly, and our game contract has successfully deployed.
-
-1. In Flow Control Manager, navigate to tools.
-2. In the Transactions section;
-
- ![Flow Control Manager contract deployment example](../media/2ce641e868905901df937c4dd09de105.png)
-
- 1. Populate the Transaction field with check-contract-deployed.cdc, located in Resources/transactions
-
- ![Resources folder example](../media/0bc22650b47abb1cf606b2c7381a6377.png)
-
- 2. Set the Signer field to the new Account we just created earlier. βPlayer1β
- 3. Click Execute Transaction
-3. If you have successfully configured the SDK and deployed the game contract, you will see the following message;
-
- ![Flow Emulator Output example](../media/bd9b6a0a7261d3c605aad9760a304220.png)
-
-## Step 3 β Incorporate Flow Accounts, Transactions and Scripts into Code
-
-We have our SDK and emulator configured, and our game contract deployed.
-Now we are going to create the code which will allow our game to send and receive data from the blockchain.
-
-Our FlowWords tutorial project contains a script called FlowInterface.cs, which can be found in the Scripts folder. This script contains all of our blockchain interfacing functions, which are called from GameManager.cs and UIManager.cs.
-
-Our game FlowInterface has 5 main functions:
-
-- Login
-- Logout
-- GetGameDataFromChain
-- SubmitGuess
-- LoadHighScoresFromChain
-
-Our functions are going to need to access a number of cadence scripts and transactions.
-These have been provided for you, and can be found in the Resources\\scripts folder and Resources\\transactions folder.
-
-![Resources folder example](../media/e6a3c62a95440af5c38b47c9efa27a72.png)
-
-FlowInterface.cs has a number of Serialized TextAsset fields, which can be populated via the Unity inspector. Select the GameFlowInterface gameobject in the Game.unity scene, and populate the fields as follows, using the scripts and transactions in the aforementioned folders; (you may find these have already been populated for you)
-
-![GameFlowInterface script assignment final state example](../media/5569416d261ff8e9d05b53c082c8a1b1.png)
-
-### Login
-Open FlowInterface.cs and find the Login function stub.
-
-The Login functionβs role is to take the credentials entered by the user, create a FlowControl.Account object with which we can submit transactions to the chain, and run the login.cdc transaction.
-
-At the top of the file, add the following using statements to grant us easy access to the Flow SDK structures.
-
-```csharp
-using DapperLabs.Flow.Sdk.Cadence;
-using DapperLabs.Flow.Sdk.DataObjects;
-using DapperLabs.Flow.Sdk.DevWallet;
-using Convert = DapperLabs.Flow.Sdk.Cadence.Convert;
-```
-
-Your file should now look like this:
-
-```csharp
-using System;
-using System.Collections;
-using System.Collections.Generic;
-using System.Linq;
-using System.Numerics;
-using System.Threading.Tasks;
-using UnityEngine;
-using DapperLabs.Flow.Sdk;
-using DapperLabs.Flow.Sdk.Unity;
-using DapperLabs.Flow.Sdk.Cadence;
-using DapperLabs.Flow.Sdk.DataObjects;
-using DapperLabs.Flow.Sdk.DevWallet;
-using Convert = DapperLabs.Flow.Sdk.Cadence.Convert;
-```
-
-Uncomment the following at line 56:
-```csharp
-//private FlowControl.Account FLOW_ACCOUNT = null;
-```
-
-We now have to register a wallet provider with the Flow SDK. We are going to use DevWallet, which comes with the Flow SDK, and is only intended for development purposes on emulator and testnet. Add the following to the `Start` function:
-
-```cs
-// Set up SDK to access Emulator
-FlowConfig flowConfig = new FlowConfig()
-{
- NetworkUrl = "http://127.0.0.1:8888/v1", // emulator
- Protocol = FlowConfig.NetworkProtocol.HTTP
-};
-FlowSDK.Init(flowConfig);
-
-// Register DevWallet wallet provider with SDK
-FlowSDK.RegisterWalletProvider(new DevWalletProvider());
-```
-Your Start function should now look like this:
-
-```csharp
-private void Start()
-{
- if (Instance != this)
- {
- Destroy(this);
- }
-
- // Set up SDK to access Emulator
- FlowConfig flowConfig = new FlowConfig()
- {
- NetworkUrl = "http://127.0.0.1:8888/v1", // emulator
- Protocol = FlowConfig.NetworkProtocol.HTTP
- };
- FlowSDK.Init(flowConfig);
-
- // Register DevWallet wallet provider with SDK
- FlowSDK.RegisterWalletProvider(new DevWalletProvider());
-}
-```
-
-> **WARNING:** Do not use DevWallet in production builds. It is only intended for development purposes and does NOT securely store keys.\
-Having the Address and Private Keys to a blockchain account gives your application full access to all of that accountβs funds and storage. They should be treated with extreme care.
-
-Next, we will fill out the body of the Login function.
-
-```csharp
-///
-/// Attempts to log in by executing a transaction using the provided credentials
-///
-/// An arbitrary username the player would like to be known by on the leaderboards
-/// Function that should be called when login is successful
-/// Function that should be called when login fails
-public void Login(string username, System.Action onSuccessCallback, System.Action onFailureCallback)
-{
- // Authenticate an account with DevWallet
-
-}
-```
-
-First, we have to invoke the wallet provider to authenticate the user and get their flow address. Add the following code to the Login function;
-
-```cs
-// Authenticate an account with DevWallet
-FlowSDK.GetWalletProvider().Authenticate(
- "", // blank string will show list of accounts from Accounts tab of Flow Control Window
- (string address) => onSuccessCallback(address, username),
- onFailureCallback);
-```
-
-The Authenticate function takes parameters as follows;
-
-- The first parameter is a username which corresponds to the name of an account in the Accounts tab. If you leave this string blank (as above), a dialog will be shown to the user to select an account from the Accounts tab.
-- The second parameter is a success callback for `Authenticate()`. We pass in a lambda function which starts a coroutine to run our async function `OnAuthSuccess`. This takes the flow address that we got from `Authenticate()`, as well as a few other parameters.
-- The third parameter is a callback for if `Authenticate()` fails. We pass through the fail callback that was passed to Login.
-
-Your completed Login function should look as follows;
-
-```csharp
-///
-/// Attempts to log in by executing a transaction using the provided credentials
-///
-/// An arbitrary username the player would like to be known by on the leaderboards
-/// Function that should be called when login is successful
-/// Function that should be called when login fails
-public void Login(string username, System.Action onSuccessCallback, System.Action onFailureCallback)
-{
- // Authenticate an account with DevWallet
- if (FlowSDK.GetWalletProvider().IsAuthenticated() == false)
- {
- FlowSDK.GetWalletProvider().Authenticate(
- "", // blank string will show list of accounts from Accounts tab of Flow Control Window
- (string address) => onSuccessCallback(address, username),
- onFailureCallback);
- }
-}
-
-```
-Now we need to implement the `GetGameDataFromChain` function for when we successfully authenticate our user and get their flow address.\
-At this point we have successfully authenticated our user, and all subsequent Scripts and Transactions will be submitted via the authenticated account.
-
-
-### GetGameDataFromChain
-
-This function executes the get-current-gamestate.cdc transaction on the chain, and then processes the emitted events to get the CurrentGameState for the logged in account, and current GameStartTime for the game of the day, which we use to show time remaining.
-
-```csharp
-///
-/// Attempts to get the current game state for the user from chain.
-///
-/// An arbitrary username the player would like to be known by on the leaderboards
-/// Callback on success
-/// Callback on failure
-public IEnumerator GetGameDataFromChain(string username, System.Action, Dictionary> onSuccessCallback, System.Action onFailureCallback)
-{
- // get FLOW_ACCOUNT object for text replacements
-
- // execute getCurrentGameState transaction on chain
-
- // check for error. if so, break.
-
- // transaction success, get data from emitted events
-
- // process currentGameState event
-
- // process gameStartTime event
-
- // call GameManager to set game state
-
- yield return null;
-}
-```
-
-As you will recall, we previously configured a number of text replacements.\
-These are useful, and allow us to easily change some hardcoded data from a contract or transaction, such as the deploy address of the contract, from one central location without having to edit every cadence script in our project.\
-Our transactions and scripts will need to access the text replacement function, which lives in the FlowControl.Account class, so we need to first create a FlowControl.Account object as follows;
-
-```cs
-// get FLOW_ACCOUNT object for text replacements
-FLOW_ACCOUNT = new FlowControl.Account
-{
- GatewayName = "Emulator", // the network to match
- AccountConfig = new Dictionary { { "Address", FlowSDK.GetWalletProvider().GetAuthenticatedAccount().Address } } // the account address to match
-};
-```
-
-We are about to pull down the user's saved game data. To make processing event data easier, we have declared three classes to hold the results of events emitted by transactions:
-
-```csharp
-public class StatePayload
-{
- public List currentState;
-}
-
-public class TimePayload
-{
- public Decimal startTime;
-}
-
-public class GuessResultPayload
-{
- public string result;
-}
-```
-
-Let's compare these to the payloads for each event in the contract:
-
-```cadence
-access(all) event CurrentState(currentState: [UserGuess])
-access(all) event LastGameStart(startTime: UFix64)
-access(all) event GuessResult(result: String)
-
-access(all) struct UserGuess
-{
- access(all) let Guess: String
- access(all) let Result: String
- init(guess: String, result: String)
- {
- self.Guess = guess
- self.Result = result
- }
-}
-```
-
-When using Cadence.Convert, cadence arrays are converted into C# Lists, as shown in the StatePayload class and CurrentState event.
-
-In GameManager.cs, the GuessResult class is declared as:
-
-```csharp
-public class GuessResult
-{
- ///
- /// the guess that was submitted
- ///
- [Cadence(CadenceType = "String", Name = "Guess")]
- public string word;
- ///
- /// A 5 letter code indicating the result and resulting color a cell should be.
- ///
- ///
- /// "p" = the letter at this position was in the word and in the correct (p)osition, color the cell green.
- /// "w" = the letter at this position was in the (w)ord, but in the incorrect position, color the cell yellow.
- /// "n" = the letter at this position was (n)ot in the word.
- ///
- [Cadence(CadenceType = "String", Name = "Result")]
- public string colorMap;
-}
-```
-
-We want the Cadence fields ```Guess``` and ```Result``` to map to the C# fields ```word``` and ```colorMap```. To do this, we add a Cadence attribute to each field with a ```Name```
-parameter that tells it which Cadence fields maps to that class field.
-
-We did not have to do that with the three payload classes we defined earlier because the C# field names exactly match the Cadence field names in the contract.
-
-Now that we have an Account object for text replacement, we can use it with the Transactions class to Submit our login.cdc transaction.
-
-Add the following code to the GetGameDataFromChain function;
-
-```cs
-// execute getCurrentGameState transaction on chain
-Task getStateTask = Transactions.SubmitAndWaitUntilExecuted(FLOW_ACCOUNT.DoTextReplacements(loginTxn.text), new CadenceString(username));
-
-while (!getStateTask.IsCompleted)
-{
- int dots = ((int)(Time.time * 2.0f) % 4);
-
- UIManager.Instance.SetStatus($"Retrieving data from chain" + new string('.', dots));
-
- yield return null;
-}
-```
-Because transactions can take quite some time on chain, we create an asynchronous Task by calling SubmitAndWaitUntilExecuted from the Transactions class, to prevent blocking the main game thread.
-Into SubmitAndWaitUntilExecuted, we pass the script that we want to execute, and any parameters.
-
-For our script, we refer to the serialized TextAsset field, loginTxn, to which we will assign login.cdc in the inspector.
-We pass our script into SubmitAndWaitUntilExecuted via the DoTextReplacements function on our FLOW_ACCOUNT object, which will parse the cadence script and replace any of our defined text replacements.
-
-For parameters, the login.cdc script is expecting a single String parameter with the playerβs display name in it. We pass in a new CadenceString object, which we create inline from the encapsulating functionβs username string parameter.
-
-Next, we simply wait until our asynchronous task.IsCompleted.
-While we wait, we update the UI with a simple animated βConnectingβ¦β text status, and yield to the Unity engine to prevent blocking the thread.
-
-Once our transaction has completed, we want to check if it was successful on chain. Add the following code beneath the transaction submission code;
-
-```cs
-// check for error. if so, break.
-if (getStateTask.Result.Error != null || getStateTask.Result.ErrorMessage != string.Empty || getStateTask.Result.Status == FlowTransactionStatus.EXPIRED)
-{
- onFailureCallback();
- yield break;
-}
-```
-
-Here we must check the transaction Result for three conditions:
-
-- Error: Was there an error submitting the transaction to the blockchain?
-- ErrorMessage: Was there an error during processing on the blockchain?
-- Status: A status of EXPIRED means that the transaction did not execute on time and was discarded.
-
-Any error here, and we are simply going to fail the login, and call our onFailureCallback.
-
-Next, we process the result of our transaction.
-This transaction is designed to return the game state for the user, and the time remaining on the word of the day, via emitted events.
-We can access these emitted events via the .Result.Events property on our task.
-To do so, add the following code below our submission logic;
-
-```cs
-// transaction success, get data from emitted events
-List events = getStateTask.Result.Events;
-FlowEvent currentStateEvent = events.Find(x => x.Type.EndsWith(".CurrentState"));
-FlowEvent startTimeEvent = events.Find(x => x.Type.EndsWith(".LastGameStart"));
-
-if (currentStateEvent == null || startTimeEvent == null)
-{
- onFailureCallback();
- yield break;
-}
-```
-
-This chunk accesses the returned Events list, and attempts to find events ending with β.CurrentStateβ and β.LastGameStartβ. We use .EndsWith, as the transaction returns fully qualified event names, and the name of the deployed contract may change during development.
-Finally, we check that we do indeed have both of our required events, and if not, call the onFailure callback and break.
-
-Next, we will parse the contents of each event. Add the following code to the function;
-
-```cs
-// process current game state event
-Decimal gameStartTime = 0;
-Dictionary letterStatuses = new Dictionary();
-List results = Convert.FromCadence(currentStateEvent.Payload).currentState;
-foreach (GuessResult newResult in results)
-{
- newResult.word = newResult.word.ToUpper();
- for (int i = 0; i < 5; i++)
- {
- bool letterAlreadyExists = letterStatuses.ContainsKey(newResult.word[i].ToString());
- string currentStatus = letterAlreadyExists ? letterStatuses[newResult.word[i].ToString()] : "";
- switch (currentStatus)
- {
- case "":
- letterStatuses[newResult.word[i].ToString()] = newResult.colorMap[i].ToString();
- break;
- case "p":
- break;
- case "w":
- if (newResult.colorMap[i] == 'p')
- {
- letterStatuses[newResult.word[i].ToString()] = newResult.colorMap[i].ToString();
- }
- break;
- case "n":
- if (newResult.colorMap[i] == 'p' || newResult.colorMap[i] == 'w')
- {
- letterStatuses[newResult.word[i].ToString()] = newResult.colorMap[i].ToString();
- }
- break;
- }
- }
-}
-
-// get game start time event
-gameStartTime = Convert.FromCadence(startTimeEvent.Payload).startTime;
-```
-From the contract, we know that the CurrentState event returns a list of ```UserGuess``` structs. We want to convert these to a C# ```List```.
-
-```csharp
-List results = Convert.FromCadence(currentStateEvent.Payload).currentState;
-```
-
-This converts the Payload of the currentStateEvent event into a ```StatePayload``` object, then sets results to the ```currentState``` field of that object.
-
-Then, we iterate over the results list and update the letterStatuses that we display.
-
-
-
-The GameStartTime event is processed similarly:
-
-```csharp
-gameStartTime = Convert.FromCadence(startTimeEvent.Payload).startTime;
-```
-
-The startTimeEvent payload is converted into a TimePayload object and the startTime field is extracted from that. Because the Cadence type is UFix64, we get back a C# Decimal struct.
-
-Finally, we call our onSuccess callback to return our results to our caller.
-Add the following lines to the bottom of the function;
-
-```cs
-// call GameManager to set game state
-onSuccessCallback(gameStartTime, results, letterStatuses);
-```
-
-You can now remove the ```yield return null``` at the base of the function if you wish.
-
-Your completed function should now look like this;
-
-```csharp
-///
-/// Attempts to get the current game state for the user from chain.
-///
-/// An arbitrary username the player would like to be known by on the leaderboards
-/// Callback on success
-/// Callback on failure
-public IEnumerator GetGameDataFromChain(string username, System.Action, Dictionary> onSuccessCallback, System.Action onFailureCallback)
-{
- // get FLOW_ACCOUNT object for text replacements
- FLOW_ACCOUNT = new FlowControl.Account
- {
- GatewayName = "Emulator", // the network to match
- AccountConfig = new Dictionary { { "Address", FlowSDK.GetWalletProvider().GetAuthenticatedAccount().Address } } // the account address to match
- };
-
- // execute getCurrentGameState transaction on chain
- Task getStateTask = Transactions.SubmitAndWaitUntilExecuted(FLOW_ACCOUNT.DoTextReplacements(loginTxn.text), new CadenceString(username));
-
- while (!getStateTask.IsCompleted)
- {
- int dots = ((int)(Time.time * 2.0f) % 4);
-
- UIManager.Instance.SetStatus($"Retrieving data from chain" + new string('.', dots));
-
- yield return null;
- }
-
- // check for error. if so, break.
- if (getStateTask.Result.Error != null || getStateTask.Result.ErrorMessage != string.Empty || getStateTask.Result.Status == FlowTransactionStatus.EXPIRED)
- {
- onFailureCallback();
- yield break;
- }
-
- // transaction success, get data from emitted events
- List events = getStateTask.Result.Events;
- FlowEvent currentStateEvent = events.Find(x => x.Type.EndsWith(".CurrentState"));
- FlowEvent startTimeEvent = events.Find(x => x.Type.EndsWith(".LastGameStart"));
-
- if (currentStateEvent == null || startTimeEvent == null)
- {
- onFailureCallback();
- yield break;
- }
-
- // process current game state event
- Decimal gameStartTime = 0;
- Dictionary letterStatuses = new Dictionary();
- List results = Convert.FromCadence(currentStateEvent.Payload).currentState;
- foreach (GuessResult newResult in results)
- {
- newResult.word = newResult.word.ToUpper();
- for (int i = 0; i < 5; i++)
- {
- bool letterAlreadyExists = letterStatuses.ContainsKey(newResult.word[i].ToString());
- string currentStatus = letterAlreadyExists ? letterStatuses[newResult.word[i].ToString()] : "";
- switch (currentStatus)
- {
- case "":
- letterStatuses[newResult.word[i].ToString()] = newResult.colorMap[i].ToString();
- break;
- case "p":
- break;
- case "w":
- if (newResult.colorMap[i] == 'p')
- {
- letterStatuses[newResult.word[i].ToString()] = newResult.colorMap[i].ToString();
- }
- break;
- case "n":
- if (newResult.colorMap[i] == 'p' || newResult.colorMap[i] == 'w')
- {
- letterStatuses[newResult.word[i].ToString()] = newResult.colorMap[i].ToString();
- }
- break;
- }
- }
- }
-
- // get game start time event
- gameStartTime = Convert.FromCadence(startTimeEvent.Payload).startTime;
-
- // call GameManager to set game state
- onSuccessCallback(gameStartTime, results, letterStatuses);
-}
-```
-
-
-### Logout
-
-The Logout functionβs role is to disconnect the authenticated wallet from your app, and to clear the FlowControl.Account object, to prevent any more transactions from being executed with those account credentials.
-
-The Logout function is very simple. Simply add the following line;
-
-```cs
-FLOW_ACCOUNT = null;
-FlowSDK.GetWalletProvider().Unauthenticate();
-```
-
-This clears the FlowAccount object, preventing any more transactions from being submitted with it, and unauthenticates the user from the wallet provider.
-
-Your completed Logout function should now look like this;
-
-```csharp
-///
-/// Clear the FLOW account object
-///
-public void Logout()
-{
- FLOW_ACCOUNT = null;
- FlowSDK.GetWalletProvider().Unauthenticate();
-}
-```
-
-
-### SubmitGuess
-
-This function has two phases. First, it checks that the entered word is valid by submitting the check-word.cdc script to chain, and processing the returned value.
-
-If the word is deemed valid, it then submits the word guess to the game contract using the currently logged in userβs credentials, by executing the submit-guess.cdc transaction script on chain, and then processing the emitted events.
-
-For phase one, enter the following code at the top of the SubmitGuess function;
-
-```cs
-// submit word via checkWord script to FLOW chain to check if word is valid
-Task checkWordTask = Scripts.ExecuteAtLatestBlock(FLOW_ACCOUNT.DoTextReplacements(checkWordScript.text), new CadenceString(word.ToLower()));
-
-while (!checkWordTask.IsCompleted)
-{
- int dots = ((int)(Time.time * 2.0f) % 4);
- UIManager.Instance.SetStatus("Waiting for server" + new string('.', dots));
- yield return null;
-}
-
-if (checkWordTask.Result.Error != null)
-{
- onFailureCallback();
- UIManager.Instance.SetStatus("Error checking word validity.");
- yield break;
-}
-
-bool wordValid = ((checkWordTask.Result.Value as CadenceString).Value == "OK");
-if (wordValid == false)
-{
- onFailureCallback();
- yield break;
-}
-```
-
-This code starts by calling ExecuteAtLatestBlock, passing in the checkWordScript and our guess word as a CadenceString object, to create an async Task using our Flow Account object.
-
-Scripts on Cadence can be thought of as read-only transactions, which are performed very quickly.
-Since scripts are read only, they do not require signing, and are best to use when you need to quickly get publicly available data from chain.
-
-As with our previous transactions, we then wait until our task.IsCompleted, and then check for any errors in the result. With scripts we only have to check the Result.Error, as this catches all possible failure modes.
-
-We then process the return value of the script, which can be found in the Result.Value property on our completed task object. Scripts do not emit events like transactions, but have return values like a regular function.
-The return value is of the generic base type CadenceBase, which we cast to CadenceString, as we are expecting a string type return value.
-
-If the word guess is deemed to be invalid we call the onFailure callback and break, otherwise we proceed onto the guess submission phase.
-
-For the second phase of the function, add the following code below phase one;
-
-```cs
-// if word is valid, submit guess via transaction to FLOW chain
-Task submitGuessTask = Transactions.SubmitAndWaitUntilExecuted(FLOW_ACCOUNT.DoTextReplacements(submitGuessTxn.text), new CadenceString(word.ToLower()));
-while (!submitGuessTask.IsCompleted)
-{
- int dots = ((int)(Time.time * 2.0f) % 4);
- UIManager.Instance.SetStatus("Waiting for server" + new string('.', dots));
- yield return null;
-}
-
-if (submitGuessTask.Result.Error != null || submitGuessTask.Result.ErrorMessage != string.Empty || submitGuessTask.Result.Status == FlowTransactionStatus.EXPIRED)
-{
- onFailureCallback();
- yield break;
-}
-
-// get wordscore
-string wordScore = "";
-FlowEvent ourEvent = submitGuessTask.Result.Events.Find(x => x.Type.EndsWith(".GuessResult"));
-if (ourEvent != null)
-{
- wordScore = Convert.FromCadence(ourEvent.Payload).result;
-
- // check if we are out of guesses
- if (wordScore == "OutOfGuesses")
- {
- onFailureCallback();
- UIManager.Instance.SetStatus("Out Of Guesses. Try again tomorrow.");
- yield break;
- }
-
- // process result
- onSuccessCallback(word, wordScore);
-}
-else
-{
- onFailureCallback();
-}
-```
-
-This phase begins by submitting the submit-guess.cdc transaction, passing in our guess word as a new CadenceString parameter. We then wait for the task to complete as usual, and check for any errors.
-As this is a transaction, we once again check the three possible failure modes, and call the onFailure callback if the transaction failed.
-
-Next we parse our transactionβs emitted events.
-
-```cadence
-access(all) event GuessResult(result: String)
-```
-
-We are expecting an event called GuessResult, with a single string parameter called result. We created a C# version of that event: ```GuessResultPayload```.
-
-```csharp
-// get wordscore
-string wordScore = "";
-FlowEvent ourEvent = submitGuessTask.Result.Events.Find(x => x.Type.EndsWith(".GuessResult"));
-if (ourEvent != null)
-{
- wordScore = Convert.FromCadence(ourEvent.Payload).result;
-
- // check if we are out of guesses
- if (wordScore == "OutOfGuesses")
- {
- onFailureCallback();
- UIManager.Instance.SetStatus("Out Of Guesses. Try again tomorrow.");
- yield break;
- }
-
- // process result
- onSuccessCallback(word, wordScore);
-}
-else
-{
- onFailureCallback();
-}
-```
-
-We first find our event in the Result.Events list on our task object.
-
-If our event is found, we then convert the payload to a ```GuessResultPayload``` and store the ```result``` field as ```wordScore```. We then pass the guess word, and the result back to our caller via the onSuccess callback.
-
-If the GuessResult event cannot be found in the Result.Events list, we call the onFailure callback.
-
-Once complete, your SubmitGuess function should look like this:
-
-```csharp
-public IEnumerator SubmitGuess(string word, System.Action onSuccessCallback, System.Action onFailureCallback)
-{
- // submit word via checkWord script to FLOW chain to check if word is valid
- Task checkWordTask = Scripts.ExecuteAtLatestBlock(FLOW_ACCOUNT.DoTextReplacements(checkWordScript.text), new CadenceString(word.ToLower()));
- while (!checkWordTask.IsCompleted)
- {
- int dots = ((int)(Time.time * 2.0f) % 4);
- UIManager.Instance.SetStatus("Waiting for server" + new string('.', dots));
- yield return null;
- }
-
- if (checkWordTask.Result.Error != null)
- {
- onFailureCallback();
- UIManager.Instance.SetStatus("Error checking word validity.");
- yield break;
- }
-
- bool wordValid = ((checkWordTask.Result.Value as CadenceString).Value == "OK");
- if (wordValid == false)
- {
- onFailureCallback();
- yield break;
- }
-
- // if word is valid, submit guess via transaction to FLOW chain
- Task submitGuessTask = Transactions.SubmitAndWaitUntilExecuted(FLOW_ACCOUNT.DoTextReplacements(submitGuessTxn.text), new CadenceString(word.ToLower()));
- while (!submitGuessTask.IsCompleted)
- {
- int dots = ((int)(Time.time * 2.0f) % 4);
- UIManager.Instance.SetStatus("Waiting for server" + new string('.', dots));
- yield return null;
- }
-
- if (submitGuessTask.Result.Error != null || submitGuessTask.Result.ErrorMessage != string.Empty || submitGuessTask.Result.Status == FlowTransactionStatus.EXPIRED)
- {
- onFailureCallback();
- yield break;
- }
-
- // get wordscore
- string wordScore = "";
- FlowEvent ourEvent = submitGuessTask.Result.Events.Find(x => x.Type.EndsWith(".GuessResult"));
- if (ourEvent != null)
- {
- wordScore = Convert.FromCadence(ourEvent.Payload).result;
-
- // check if we are out of guesses
- if (wordScore == "OutOfGuesses")
- {
- onFailureCallback();
- UIManager.Instance.SetStatus("Out Of Guesses. Try again tomorrow.");
- yield break;
- }
-
- // process result
- onSuccessCallback(word, wordScore);
- }
- else
- {
- onFailureCallback();
- }
-}
-```
-
-### LoadHighScoresFromChain
-
-This function fires off a number of small scripts simultaneously, which pull publicly available high score data from the game contract on chain using;
-
-- get-highscores.cdc
-- get-player-cumulativescore.cdc
-- get-player-guess-distribution.cdc
-- get-player-maxstreak.cdc
-- get-player-scores.cdc
-- get-player-streak.cdc.
-
-It then processes their returned values, and passes them out to the onSuccess call for the high scores UI to render.
-
-For this function, we are going to first fire off a number of simultaneous scripts on the blockchain. This is something you want to avoid with transactions, as transaction order of execution cannot be guaranteed due to the distributed nature of blockchain, however as scripts are read-only, and do not mutate the chain, order of execution is far less likely to matter.
-
-To execute the scripts, add the following code to the top of the function;
-
-```cs
-// get player's wallet public address
-string playerWalletAddress = FlowSDK.GetWalletProvider().GetAuthenticatedAccount().Address;
-
-// execute scripts to get highscore data
-Dictionary> tasks = new Dictionary>();
-tasks.Add("GetHighScores", Scripts.ExecuteAtLatestBlock(FLOW_ACCOUNT.DoTextReplacements(GetHighScores.text)));
-tasks.Add("GetPlayerCumulativeScore", Scripts.ExecuteAtLatestBlock(FLOW_ACCOUNT.DoTextReplacements(GetPlayerCumulativeScore.text), new CadenceAddress(playerWalletAddress)));
-tasks.Add("GetPlayerWinningStreak", Scripts.ExecuteAtLatestBlock(FLOW_ACCOUNT.DoTextReplacements(GetPlayerWinningStreak.text), new CadenceAddress(playerWalletAddress)));
-tasks.Add("GetPlayerMaxWinningStreak", Scripts.ExecuteAtLatestBlock(FLOW_ACCOUNT.DoTextReplacements(GetPlayerMaxWinningStreak.text), new CadenceAddress(playerWalletAddress)));
-tasks.Add("GetGuessDistribution", Scripts.ExecuteAtLatestBlock(FLOW_ACCOUNT.DoTextReplacements(GetGuessDistribution.text), new CadenceAddress(playerWalletAddress)));
-
-// wait for completion
-bool complete = false;
-while (!complete)
-{
- complete = true;
- foreach (KeyValuePair> task in tasks)
- {
- complete = complete && task.Value.IsCompleted;
- }
- yield return null;
-}
-
-// check for errors
-foreach (KeyValuePair> task in tasks)
-{
- if (task.Value.Result.Error != null)
- {
- onFailureCallback();
- yield break;
- }
-}
-```
-
-This block of code first obtains the public address of the current authenticated player and creates a ```Dictionary``` to store our concurrent script tasks.
-We then sequentially create async Tasks for each script that we want to execute, using ExecuteAtLatestBlock, and add them to the Task dictionary.
-
-In our use case, we want all of the tasks to complete before we display any results, so our wait for completion code block iterates over every Task in the dictionary, and only moves on once every task.IsComplete.
-
-Checking for errors is similarly done using a foreach loop, where every task is checked to ensure the Error field is null. If even one task has an Error, we call the onFailure callback and break.
-
-Next we need to process the returned values. Add the following code beneath the previous;
-
-```cs
-// load global highscores
-List GlobalScores = Convert.FromCadence>(tasks["GetHighScores"].Result.Value);
-GlobalScores = GlobalScores.OrderByDescending(score => score.Score).Take(10).ToList();
-
-// load player scores
-BigInteger PlayerCumulativeScore = Convert.FromCadence(tasks["GetPlayerCumulativeScore"].Result.Value);
-BigInteger PlayerWinningStreak = Convert.FromCadence(tasks["GetPlayerWinningStreak"].Result.Value);
-BigInteger PlayerMaximumWinningStreak = Convert.FromCadence(tasks["GetPlayerMaxWinningStreak"].Result.Value);
-List PlayerGuessDistribution = Convert.FromCadence>(tasks["GetGuessDistribution"].Result.Value);
-
-// callback
-onSuccessCallback(GlobalScores, PlayerCumulativeScore, PlayerWinningStreak, PlayerMaximumWinningStreak, PlayerGuessDistribution);
-```
-
-Our global highscores are an array of Scores objects in the contract.
-
-```cadence
-access(contract) let TopScores : [Scores]
-access(all) struct Scores
-{
- access(all) let AccId : Address
- access(all) let Name : String
- access(all) let Score : UInt
-}
-```
-
-We have a ScoreStruct defined HighScoresPanel.cs as:
-
-```csharp
-public struct ScoreStruct
-{
- public string Name;
- public BigInteger Score;
-}
-```
-
-```csharp
-List GlobalScores = Convert.FromCadence>(tasks["GetHighScores"].Result.Value);
-GlobalScores = GlobalScores.OrderByDescending(score => score.Score).Take(10).ToList();
-```
-
-Here we get the result of the GetHighScores task and convert it into a ```List```. Then we reorder the list and keep only the highest ten values.
-
-Next, we parse the detailed statistics for the current player, using ```Convert.FromCadence``` to convert from the Cadence values into the C# types we want.
-
-Finally, we call the onSuccess callback, passing in all of our parsed results.
-
-Once complete, your function should look as follows;
-
-```csharp
-public IEnumerator LoadHighScoresFromChain(System.Action, BigInteger, BigInteger, BigInteger, List> onSuccessCallback, System.Action onFailureCallback)
-{
- // get player's wallet public address
- string playerWalletAddress = FlowSDK.GetWalletProvider().GetAuthenticatedAccount().Address;
-
- // execute scripts to get highscore data
- Dictionary> tasks = new Dictionary>();
- tasks.Add("GetHighScores", Scripts.ExecuteAtLatestBlock(FLOW_ACCOUNT.DoTextReplacements(GetHighScores.text)));
- tasks.Add("GetPlayerCumulativeScore", Scripts.ExecuteAtLatestBlock(FLOW_ACCOUNT.DoTextReplacements(GetPlayerCumulativeScore.text), new CadenceAddress(playerWalletAddress)));
- tasks.Add("GetPlayerWinningStreak", Scripts.ExecuteAtLatestBlock(FLOW_ACCOUNT.DoTextReplacements(GetPlayerWinningStreak.text), new CadenceAddress(playerWalletAddress)));
- tasks.Add("GetPlayerMaxWinningStreak", Scripts.ExecuteAtLatestBlock(FLOW_ACCOUNT.DoTextReplacements(GetPlayerMaxWinningStreak.text), new CadenceAddress(playerWalletAddress)));
- tasks.Add("GetGuessDistribution", Scripts.ExecuteAtLatestBlock(FLOW_ACCOUNT.DoTextReplacements(GetGuessDistribution.text), new CadenceAddress(playerWalletAddress)));
-
- // wait for completion
- bool complete = false;
- while (!complete)
- {
- complete = true;
- foreach (KeyValuePair> task in tasks)
- {
- complete = complete && task.Value.IsCompleted;
- }
- yield return null;
- }
-
- // check for errors
- foreach (KeyValuePair> task in tasks)
- {
- if (task.Value.Result.Error != null)
- {
- onFailureCallback();
- yield break;
- }
- }
-
- // load global highscores
- List GlobalScores = Convert.FromCadence>(tasks["GetHighScores"].Result.Value);
- GlobalScores = GlobalScores.OrderByDescending(score => score.Score).Take(10).ToList();
-
- // load player scores
- BigInteger PlayerCumulativeScore = Convert.FromCadence(tasks["GetPlayerCumulativeScore"].Result.Value);
- BigInteger PlayerWinningStreak = Convert.FromCadence(tasks["GetPlayerWinningStreak"].Result.Value);
- BigInteger PlayerMaximumWinningStreak = Convert.FromCadence(tasks["GetPlayerMaxWinningStreak"].Result.Value);
- List PlayerGuessDistribution = Convert.FromCadence>(tasks["GetGuessDistribution"].Result.Value);
-
- // callback
- onSuccessCallback(GlobalScores, PlayerCumulativeScore, PlayerWinningStreak, PlayerMaximumWinningStreak, PlayerGuessDistribution);
-
- yield return null;
-}
-```
-
-## Step 5 β Play FlowWords!
-
-If you have correctly followed all of the steps above, you will now have a working project.
-
-1. Return to the Unity editor, and press the Play button.
-2. Enter a friendly name - this will appear on the leaderboard.
-3. Click Log In.
-4. Select an account from the dialog that appears to authenticate with.
-
-You should see the login screen, say Connectingβ¦, and then Loadingβ¦
-
-![Login Panel User Interface](../media/fe0ccbbb8fb583dd7c461a1436d9206c.png)
-
-Followed shortly thereafter by the game screen;
-
-![Game User Interface](../media/8cd5f4e2d4ac66693e0c2a6f796d1b38.png)
-
-And the High Scores screen (if you click the button);
-
-![HighScores User Interface](../media/eb0fba4dbd9a040424afb3182e5ce345.png)
-
-## Step 6 β Further Experimentation
-
-For an extra challenge, try some of the following;
-
-- Add more accounts and play with some friends, hot seat style
-- Modify the game-contract.cdc to make a new game every 5 minutes instead of every 24 hours.
-- Try to remove and redeploy the contract
- (hint: on testnet and mainnet, once removed, a contractβs name can never be used again on the same account)
- (extra hint: delete-game-resources.cdc)
-- Poke about in the game contracts, scripts and transactions to see what they do!
-
-If you ever get the emulator into a messy state, you can always hit the Clear Persistent Data button, which will wipe the emulator back to its blank state. This will of course lose all deployed contracts and high score and game history.
-
-## Appendix β How to convert FlowWords to run on TestNet
-
-To modify the tutorial project to run on TestNet, only minor modifications are required.
-
-1. Change the network configuration to point to TestNet
-2. Replace the DevWallet provider with a more secure solution. e.g. WalletConnect
-3. Configure Account and update Text Replacements
-4. Deploy the contract to TestNet
-
-### Change the network configuration to TestNet
-
-To change the network configuration, simply modify the start function as follows;
-
-```cs
-private void Start()
-{
- if (Instance != this)
- {
- Destroy(this);
- }
-
- // Set up SDK to access TestNet
- FlowConfig flowConfig = new FlowConfig()
- {
- NetworkUrl = "https://rest-testnet.onflow.org/v1", // testnet
- Protocol = FlowConfig.NetworkProtocol.HTTP
- };
- FlowSDK.Init(flowConfig);
-
- // Register DevWallet wallet provider with SDK
- FlowSDK.RegisterWalletProvider(new DevWalletProvider());
-}
-```
-
-We have now replaced the emulator address with the address for the TestNet access point, and all subsequent transactions and scripts will be directed to TestNet.
-
-### Replace DevWallet with WalletConnect
-
-To now change the wallet provider, simply modify the start function as follows;
-
-```cs
-private void Start()
-{
- if (Instance != this)
- {
- Destroy(this);
- }
-
- // Set up SDK to access TestNet
- FlowConfig flowConfig = new FlowConfig()
- {
- NetworkUrl = "https://rest-testnet.onflow.org/v1", // testnet
- Protocol = FlowConfig.NetworkProtocol.HTTP
- };
- FlowSDK.Init(flowConfig);
-
- // Create WalletConnect wallet provider
- IWallet walletProvider = new WalletConnectProvider();
- walletProvider.Init(new WalletConnectConfig
- {
- ProjectId = "", // insert Project ID from Wallet Connect dashboard
- ProjectDescription = "A simple word guessing game built on FLOW!",
- ProjectIconUrl = "https://walletconnect.com/meta/favicon.ico",
- ProjectName = "FlowWords",
- ProjectUrl = "https://dapperlabs.com"
- });
-
- // Register WalletConnect wallet provider with SDK
- FlowSDK.RegisterWalletProvider(walletProvider);
-}
-```
-
-You will also need to add the following using declarations to the top of the file;
-
-```cs
-using DapperLabs.Flow.Sdk.WalletConnect;
-using DapperLabs.Flow.Sdk.Crypto;
-```
-
-For this modification we have created a new WalletConnectProvider, and initialized it, and then registered our new WalletConnectProvider.
-
-The only thing missing is a Project Id.\
-Each WalletConnect application requires its own project id. You can get one by going to https://cloud.walletconnect.com and signing up for an account.\
-You can then create a new project on the website, give it a name (we suggest FlowWords), and enter the project id provided into your code.
-
-Finally, we need one more change to make sure our text replacement still functions correctly.
-
-At the beginning of the OnAuthSuccess function, change the FLOW_ACCOUNT GatewayName from "Emulator" to "Flow Testnet".
-
-```cs
-// get FLOW account - we are only going to use this for text replacements
-FLOW_ACCOUNT = new FlowControl.Account
-{
- GatewayName = "Flow Testnet",
- AccountConfig = new Dictionary { { "Address", FlowSDK.GetWalletProvider().GetAuthenticatedAccount().Address } }
-};
-```
-
-### Configure Account and update Text Replacements
-
-To deploy the contract to TestNet, the only current way to do so via the FlowSDK, is to use a TestNet account to which you know the private key.
-
-First, in the Accounts tab of the Flow Control window, create a new account by clicking the plus '+' icon at the top of the window.\
-Create the new account as follows, replacing the Address and Private Key with your own valid TestNet address / private key pair.
-
-![New Account Example](../media/AppendixNewAccount.png)
-
-Switch to the Text Replacements tab of the Flow Control window now, and create a new text replacement definition as follows, replacing the given replacement text with the address to your TestNet account.
-
-![New Text Replacement Example](../media/AppendixTextReplacement.png)
-
-Now modify the existing CONTRACT_NAME text replacement, by changing the Apply to Gateways field from Emulator, to All. This will make sure this replacement also applies to TestNet transactions.
-
-### Deploy the Contract to TestNet
-
-Now that all of the configuration is done, you can deploy the contract.
-
-On the Tools tab of the Flow Control window, set up the Manage Contracts section as follows;
-
-![New Account Example](../media/AppendixDeployContract.png)
-
-Where;
-- Contract Name is the name of the contract on the blockchain. This should match the text replacement configured for CONTRACT_NAME
-- Contract is the game-contract.cdc asset included with the project sample in the Assets\\Samples\\Flow SDK\\<version>\\Flow Words Tutorial Assets\\Resources directory.
-- Account is the new Deploy Account that you created in the previous step.
-
-Once you have the Manage Contracts section filled in as per the example, click Deploy Contract.
-
-If all goes well, you will see the text 'Executing Transaction' appear below the Deploy Contract button, followed by a clickable link to the transaction result on [Flowdiver](https://testnet.flowdiver.io/).
-
-Congratulations! You can now run FlowWords, and play the game with a WalletConnect compatible wallet!
-
-> **NOTE:** TestNet requires that all contracts deployed to an account have a unique name, does not allow contract removal without authorisation, and only allows contract updates that do not break interfaces or data structures or introduce undefined data to existing resources.\
-\
-Due to these limitations, iterative development should always be done on Emulator before attempting to push anything to a live network.
-
diff --git a/docs/tools/clients/unity-sdk/samples/nft-browser.md b/docs/tools/clients/unity-sdk/samples/nft-browser.md
deleted file mode 100644
index a4ecdf2dc0..0000000000
--- a/docs/tools/clients/unity-sdk/samples/nft-browser.md
+++ /dev/null
@@ -1,291 +0,0 @@
-# Simple NFT Viewer
-
-This example project will show you how to build a simple viewer that will allow you to view NFTs that conform to the [NFT](https://github.com/onflow/flow-nft) and [MetadataViews](https://github.com/onflow/flow-nft#nft-metadata) standards.
-
-This tutorial will mostly ignore the C# code that actually displays the NFTs and focus on a high level summary of the steps used.
-
-## Overview
-
-When querying the blockchain we utilize four scripts:
-```
-* [GetCollections.cdc](Cadence/GetCollections.cdc) - Gets a list of Collections that conform to NFT.Collection for a given address
-* [GetNftIdsForCollection.cdc](Cadence/GetNftIdsForCollection.cdc) - Gets a list of all NFT IDs that are contained in a given collection
-* [GetDisplayDataForIDs.cdc](Cadence/GetDisplayDataForIDs.cdc) - Gets just the display data for a given NFT
-* [GetFullDataForID.cdc](Cadence/GetFullDataForID.cdc) - Gets a more comprehensive set of data for a single NFT.
-```
-While we could use a single script to query for all the data, larger collections will cause the script to time out. Instead we query for just the data we need to reduce the chances of a timeout occurring.
-
-## Finding Collections
-
-First we need to get a list of all collections on an account that are a subtype of NFT.Collection.
-
-```cadence
-import NonFungibleToken from 0x1d7e57aa55817448
-
-access(all) fun main(addr: Address) : [StoragePath] {
- // Get the authorized Account object for the given address.
- // The authorized account is needed because we're going to be looking into the Storage of the user
- var acct = getAuthAccount(addr)
-
- //Array that we will fill with all valid storage paths
- var paths : [StoragePath] = []
-
- //Uses the storage iteration API to iterate through all storage paths on the account
- acct.forEachStored(fun (path: StoragePath, type:Type): Bool {
- //Check to see if the resource at this location is a subtype of NonFungibleToken.Collection.
- if type.isSubtype(of: Type<@NonFungibleToken.Collection>()) {
- //Add this path to the array
- paths.append(path)
- }
-
- //returning true tells the iterator to continue to the next entry
- return true
- });
-
- //Return the array that we built
- return paths
-}
-```
-
-We use the [Storage Iteration API](https://cadence-lang.org/docs/language/accounts#storage-iteration) to look at everything the account has in it's storage and see if it is an NFT Collection. We return a list of all found NFT Collections.
-
-## Getting NFT IDs Contained in a Collection
-
-We use this to create a list of collection paths a user can pick from. When the user selects a path to view, we fetch a list of IDs contained in that collection:
-
-```cadence
-import NonFungibleToken from 0x1d7e57aa55817448
-
-access(all) fun main(addr: Address, path: StoragePath) : [UInt64] {
- // Get the authorized Account object for the given address.
- // The authorized account is needed because we're going to be looking into the Storage of the user
- var acct = getAuthAccount(addr)
-
- //Get a reference to an interface of type NonFungibleToken.Collection public backed by the resource located at path
- var ref = acct.borrow<&{NonFungibleToken.CollectionPublic}>(from: path)!
-
- //Return the list of NFT IDs contained in this collection
- return ref!.getIDs()
-}
-```
-
-## Getting Display Data for an NFT
-
-After we get a list of the available NFT IDs, we need to get some basic data about the NFT to display the thumbnail icon.
-
-```cadence
-import NonFungibleToken from 0x1d7e57aa55817448
-import MetadataViews from 0x1d7e57aa55817448
-
-access(all) fun main(addr: Address, path: StoragePath, ids: [UInt64]) : {UInt64:AnyStruct?} {
- //Array to hold the NFT display data that we will return
- //We use AnyStruct? because that is the type that is returned by resolveView.
- var returnData: {UInt64:AnyStruct?} = {}
-
- // Get the authorized Account object for the given address.
- // The authorized account is needed because we're going to be looking into the Storage of the user
- var acct = getAuthAccount(addr)
-
- //Get a reference to a capability to the storage path as a NonFungibleToken.CollectionPublic
- var ref = acct.borrow<&{NonFungibleToken.CollectionPublic}>(from: path)!
-
- //Loop through the requested IDs
- for id in ids {
- //Get a reference to the NFT we're interested in
- var nftRef = ref.borrowNFT(id: id)
-
- //If for some reason we couldn't borrow a reference, continue onto the next NFT
- if nftRef == nil {
- continue
- }
-
- //Fetch the information we're interested in and store it in our NFT structure
- returnData[id] = nftRef.resolveView(Type())
- }
-
- return returnData
-}
-```
-This gives us a dictionary that maps NFT IDs to Display structs (```{UInt64:MetadataViews.Display}```). Because accessing this information can be tedious in C#, we can define some C# classes to make our lives easier:
-
-```csharp
-public class File
-{
- public string url;
- public string cid;
- public string path;
-}
-
-public class Display
-{
- public String name;
- public String description;
- public File thumbnail;
-}
-```
-
-This will allow us to use Cadence.Convert to convert from the CadenceBase that the script returns into a Display class.
-
-This line in NFTViewer.cs is an example of converting using Cadence.Convert:
-
-```csharp
-Dictionary displayData = Convert.FromCadence>(scriptResponseTask.Result.Value);
-```
-
-You might ask whey we don't combine GetNftIdsForCollection.cdc and GetDisplayDataForIDs.cdc to get the Display data at the same time we get the list of IDs. This approach would work in many cases, but when an account contains large numbers of NFTs, this could cause a script timeout. Getting the list of IDs is a cheap call because the NFT contains this list in an array already.
-By getting just the NFT IDs, we could implement paging and use multiple script calls to each fetch a portion of the display data.
-This example doesn't currently do this type of paging, but could do so without modifying the cadence scripts.
-
-## Getting Complete NFT Data
-
-When a user selects a particular NFT to view in more detail, we need to fetch that detail.
-
-```cadence
-import NonFungibleToken from 0x1d7e57aa55817448
-import MetadataViews from 0x1d7e57aa55817448
-
-//Structure that will hold all the data we want for an NFT
-access(all) struct NFTData {
- access(all) var NFTView: AnyStruct?
- access(all) var Display : AnyStruct?
- access(all) var HTTPFile: AnyStruct?
- access(all) var IPFSFile: AnyStruct?
- access(all) var Edition: AnyStruct?
- access(all) var Editions: AnyStruct?
- access(all) var Serial: AnyStruct?
- access(all) var Royalty: AnyStruct?
- access(all) var Royalties: AnyStruct?
- access(all) var Media: AnyStruct?
- access(all) var Medias: AnyStruct?
- access(all) var License: AnyStruct?
- access(all) var ExternalURL: AnyStruct?
- access(all) var NFTCollectionDisplay: AnyStruct?
- access(all) var Rarity: AnyStruct?
- access(all) var Trait: AnyStruct?
- access(all) var Traits: AnyStruct?
-
- init() {
- self.NFTView = nil
- self.Display = nil
- self.HTTPFile = nil
- self.IPFSFile = nil
- self.Edition = nil
- self.Editions = nil
- self.Serial = nil
- self.Royalty = nil
- self.Royalties = nil
- self.Media = nil
- self.Medias = nil
- self.License = nil
- self.ExternalURL = nil
- self.NFTCollectionDisplay = nil
- self.Rarity = nil
- self.Trait = nil
- self.Traits = nil
- }
-}
-
-access(all) fun main(addr: Address, path: StoragePath, id: UInt64) : NFTData? {
- // Get the authorized Account object for the given address.
- // The authorized account is needed because we're going to be looking into the Storage of the user
- var acct = getAuthAccount(addr)
-
- //Get a reference to a capability to the storage path as a NonFungibleToken.CollectionPublic
- var ref = acct.borrow<&{NonFungibleToken.CollectionPublic}>(from: path)!
-
- //Get a reference to the NFT we're interested in
- var nftRef = ref.borrowNFT(id: id)
-
- //If for some reason we couldn't borrow a reference, continue onto the next NFT
- if nftRef == nil {
- return nil
- }
-
- var nftData : NFTData = NFTData()
-
- //Fetch the information we're interested in and store it in our NFT structure
- nftData.Display = nftRef.resolveView(Type())
- nftData.NFTView = nftRef.resolveView(Type())
- nftData.HTTPFile = nftRef.resolveView(Type())
- nftData.IPFSFile = nftRef.resolveView(Type())
- nftData.Edition = nftRef.resolveView(Type())
- nftData.Editions = nftRef.resolveView(Type())
- nftData.Serial = nftRef.resolveView(Type())
- nftData.Media = nftRef.resolveView(Type())
- nftData.Rarity = nftRef.resolveView(Type())
- nftData.Trait = nftRef.resolveView(Type())
- nftData.Traits = nftRef.resolveView(Type())
- nftData.Medias = nftRef.resolveView(Type())
- nftData.ExternalURL = nftRef.resolveView(Type())
- nftData.Royalty = nftRef.resolveView(Type())
- nftData.Royalties = nftRef.resolveView(Type())
- nftData.License = nftRef.resolveView(Type())
- nftData.NFTCollectionDisplay = nftRef.resolveView(Type())
-
- return nftData
-}
-```
-Here we define a struct NFTData that will contain all the different information we want and fill the struct via multiple resolveView calls.
-
-## C# Classes for Easy Converting
-
-The end of NFTViewer.cs contains classes that we use to more easily convert from Cadence into C#. One thing to note is that the Cadence structs contain Optionals, like:
-
-```var IPFSFile: AnyStruct?```
-
-while the C# versions do not, such as
-
-```public IPFSFile IPFSFile;```
-
-This is because we are declaring them as Classes, not Structs. Classes in C# are reference types, which can automatically be null. We could have used Structs, in which case we'd have to use:
-
-```public IPFSFile? IPFSFile```
-
-This would wrap the IPFSFile struct in a Nullable, which would allow it to be null if the Cadence value was nil.
-
-Another thing to note is the declaration of the C# File class:
-
-```csharp
-public class File
-{
- public string url;
- public string cid;
- public string path;
-
- public string GetURL()
- {
- if (string.IsNullOrEmpty(url) && !string.IsNullOrEmpty(cid))
- {
- return $"https://ipfs.io/ipfs/{cid}";
- }
-
- return url;
- }
-}
-```
-
-Compare this to the File interface in the MetadataViews contract:
-
-```cadence
- access(all) struct interface File {
- access(all) fun uri(): String
- }
-```
-
-The MetadataViews.File interface doesn't actually contain any fields, only a single method. Because only two things in MetadataViews implement the
-File interface (HTTPFile and IPFSFile), we chose to combine the possible fields into our File class.
-
-```cadence
-access(all) struct HTTPFile: File {
- access(all) let url: String
-}
-
-access(all) struct IPFSFile: File {
- access(all) let cid: String
- access(all) let path: String?
-}
-```
-
-This allows Cadence.Convert to convert either an HTTPFile or an IPFSFile into a File object. We can then check which fields are populated to determine which it was initially.
-
-This works fine for this simple viewer, but a more robust approach might be to create a ResolvedFile struct in the cadence script which has a single uri field and populates it by calling the uri() function on whatever File type was retrieved.
diff --git a/docs/tools/clients/unity-sdk/samples/nft-example.md b/docs/tools/clients/unity-sdk/samples/nft-example.md
deleted file mode 100644
index 6deb513a6b..0000000000
--- a/docs/tools/clients/unity-sdk/samples/nft-example.md
+++ /dev/null
@@ -1,419 +0,0 @@
-# Simple NFT demo
-
-This tutorial will show you how to create, mint and list a simple NFT. It follows the
-Non Fungible Token standard (https://github.com/onflow/flow-nft/blob/master/contracts/NonFungibleToken.cdc),
-but does not implement the MetadataViews interface. If you would like to make your NFT compatible with
-marketplaces, look at implementing MetadataViews (https://github.com/onflow/flow-nft/blob/master/contracts/MetadataViews.cdc)
-
-The following are the main points of this tutorial:
-1. Creating a contract that implements INFT
-2. Deploying the contract
-3. Listing, minting and storing NFTs defined by the contract via a transaction
-
-
-### Getting started
-Load the Samples/Flow SDK/x.x.x/Example NFT/Scenes/NFTExampleScene scene.
-Press play and approve the transactions that come up (only on first time run)
-Click Authenticate and choose the emulator_service_account.
-Click Mint
-Fill in the Text and URL fields and click Mint
-Approve the transaction
-Click List to refresh the NFT display panel and show your newly minted NFT
-Repeat Mint and List as desired to make your list grow
-
-Now we'll show you how this works.
-
-### Creating an NFT contract
-
-When creating an NFT it is recommended (but not required) to implement the NonFungibleToken.INFT
-interface. We will be doing so in this case.
-
-At its simplest, an NFT on Flow is a resource with a unique id. A Collection is a resource
-that will allow you to store, list, deposit, and withdraw NFTs of a specific type.
-
-We recommend reading through the [NFT tutorial](https://cadence-lang.org/docs/tutorial/non-fungible-tokens-1)
-to understand what is happening, as well as reviewing the contents of Cadence/Contracts/SDKExampleNFT.cdc
-
-The SDKExampleNFT minter allows for anyone to mint an SDKExampleNFT. Typically you would restrict
-minting to an authorized account.
-
-This tutorial will not delve deeply into the NFT contract or Cadence, instead focusing on interacting
-with them using the functionality the Unity SDK provides.
-
-### Deploying the contracts
-
-Open up Example.cs to follow along.
-
-Our Start function looks like this:
-
-```csharp
-public void Start()
-{
- //Initialize the FlowSDK, connecting to an emulator using HTTP
- FlowSDK.Init(new FlowConfig
- {
- NetworkUrl = FlowControl.Data.EmulatorSettings.emulatorEndpoint,
- Protocol = FlowConfig.NetworkProtocol.HTTP
- });
-
- //Register the DevWallet provider that we will be using
- FlowSDK.RegisterWalletProvider(new DevWalletProvider());
-
- //Deploy the NonFungibleToken and SDKExampleNFT contracts if they are not already deployed
- StartCoroutine(DeployContracts());
-}
-```
-
-This initializes the FlowSDK to connect to the emulator, creates and registers a DevWalletProvioder, then
-starts a coroutine to deploy our contract if needed.
-
-Contracts can be deployed via the FlowControl Tools window, but we will deploy them via code for ease
-of use.
-
-The DeployContracts coroutine:
-
-```csharp
-public IEnumerator DeployContracts()
-{
- statusText.text = "Verifying contracts";
- //Wait 1 second to ensure emulator has started up and service account information has been captured.
- yield return new WaitForSeconds(1.0f);
-
- //Get the address of the emulator_service_account, then get an account object for that account.
- Task accountTask = Accounts.GetByAddress(FlowControl.Data.Accounts.Find(acct => acct.Name == "emulator_service_account").AccountConfig["Address"]);
- //Wait until the account fetch is complete
- yield return new WaitUntil(() => accountTask.IsCompleted);
-
- //Check for errors.
- if (accountTask.Result.Error != null)
- {
- Debug.LogError(accountTask.Result.Error.Message);
- Debug.LogError(accountTask.Result.Error.StackTrace);
- }
-
- //We now have an Account object, which contains the contracts deployed to that account. Check if the NonFungileToken and SDKExampleNFT contracts are deployed
- if (!accountTask.Result.Contracts.Exists(x => x.Name == "SDKExampleNFT") || !accountTask.Result.Contracts.Exists(x => x.Name == "NonFungibleToken"))
- {
- statusText.text = "Deploying contracts,\napprove transactions";
-
- //First authenticate as the emulator_service_account using DevWallet
- FlowSDK.GetWalletProvider().Authenticate("emulator_service_account", null, null);
-
- //Ensure that we authenticated properly
- if (FlowSDK.GetWalletProvider().GetAuthenticatedAccount() == null)
- {
- Debug.LogError("No authenticated account.");
- yield break;
- }
-
- //Deploy the NonFungibleToken contract
- Task txResponse = CommonTransactions.DeployContract("NonFungibleToken", NonFungibleTokenContract.text);
- yield return new WaitUntil(() => txResponse.IsCompleted);
- if (txResponse.Result.Error != null)
- {
- Debug.LogError(txResponse.Result.Error.Message);
- Debug.LogError(txResponse.Result.Error.StackTrace);
- yield break;
- }
-
- //Wait until the transaction finishes executing
- Task txResult = Transactions.GetResult(txResponse.Result.Id);
- yield return new WaitUntil(() => txResult.IsCompleted);
-
- //Deploy the SDKExampleNFT contract
- txResponse = CommonTransactions.DeployContract("SDKExampleNFT", SDKExampleNFTContract.text);
- yield return new WaitUntil(() => txResponse.IsCompleted);
- if (txResponse.Result.Error != null)
- {
- Debug.LogError(txResponse.Result.Error.Message);
- Debug.LogError(txResponse.Result.Error.StackTrace);
- yield break;
- }
-
- //Wait until the transaction finishes executing
- txResult = Transactions.GetResult(txResponse.Result.Id);
- yield return new WaitUntil(() => txResult.IsCompleted);
-
- //Unauthenticate as the emulator_service_account
- FlowSDK.GetWalletProvider().Unauthenticate();
- }
-
- //Enable the Authenticate button.
- authenticateButton.interactable = true;
- statusText.text = "";
-}
-```
-
-We start by waiting one second. This ensures that the emulator has finished initializing and
-the required service account has been populated.
-
-Next we fetch the emulator_service_account Account. This Account object will contain the contracts
-that are deployed to the account. We check if both the required contracts are deployed,
-and if they are not, we deploy them.
-
-Upon first running the scene, you will be presented with two popups by DevWallet. This authorizes
-the transactions that will deploy the contracts. You will not see these popups during subsequent
-runs because the contracts will already be present on the account. If you purge the emulator
-data, you will see the popups again the next time you play the scene.
-
-When using Testnet or Mainnet, the NonFungibleToken contract will already be deployed at a known location.
-Launching the emulator with the --contracts flag will also deploy this contract. I this case we are running
-without --contracts, so we will deploy the NonFungibleToken contract ourselves.
-
-### Listing, minting, and storing NFTs
-
-Now that the contracts are in place, the Authenticate button will be clickable. This uses the
-registered wallet provider (DevWalletProvider) to authenticate. Unless you create another account
-using the FlowControl Tools panel, only emulator_service_account will be available.
-
-After clicking Authenticate, it will prompt you to select an account to authenticate as. Choose
-emulator_service_account. This is done with the following functions:
-
-```csharp
- public void Authenticate()
- {
- FlowSDK.GetWalletProvider().Authenticate("", OnAuthSuccess, OnAuthFailed);
- }
-
- private void OnAuthFailed()
- {
- Debug.LogError("Authentication failed!");
- accountText.text = $"Account: {FlowSDK.GetWalletProvider().GetAuthenticatedAccount()?.Address??"None"}";
- if (FlowSDK.GetWalletProvider().GetAuthenticatedAccount() == null)
- {
- mintPanelButton.interactable = false;
- listButton.interactable = false;
- }
- }
-
- private void OnAuthSuccess(string obj)
- {
- accountText.text = $"Account: {FlowSDK.GetWalletProvider().GetAuthenticatedAccount().Address}";
- mintPanelButton.interactable = true;
- listButton.interactable = true;
- }
-```
-
-If authentication succeeds, a coroutine is started that will make the Mint button available.
-
-Clicking on the Mint button displays the Minting panel that will allow you to customize the NFT that will
-be minted:
-
-```csharp
-public void ShowMintPanel()
- {
- textInputField.text = "";
- URLInputField.text = "";
- mintPanel.SetActive(true);
- }
-```
-
-### Minting
-Clicking Mint in the Mint panel will trigger the creation of the NFT with the supplied text.
-
-
-```csharp
-public void MintNFT()
- {
- if(FlowSDK.GetWalletProvider() != null && FlowSDK.GetWalletProvider().IsAuthenticated())
- {
- StartCoroutine(MintNFTCoroutine());
- }
-
- mintPanel.SetActive(false);
- }
-```
-
-```csharp
- public IEnumerator MintNFTCoroutine()
- {
- statusText.text = "Minting...";
- List args = new List
- {
- Convert.ToCadence(new Dictionary
- {
- ["Text"] = textInputField.text,
- ["URL"] = URLInputField.text
- }, "{String:String}")
- };
-
- Task txResponse = Transactions.Submit(mintTransaction.text, args);
-
- while(!txResponse.IsCompleted)
- {
- yield return null;
- }
-
- if (txResponse.Result.Error != null)
- {
- statusText.text = "Error, see log";
- Debug.LogError(txResponse.Result.Error.Message);
- yield break;
- }
-
- Task txResult = Transactions.GetResult(txResponse.Result.Id);
-
- while (!txResult.IsCompleted)
- {
- yield return null;
- }
-
- if (txResult.Result.Error != null)
- {
- statusText.text = "Error, see log";
- Debug.LogError(txResult.Result.Error.Message);
- yield break;
- }
-
- statusText.text = "";
- }
-```
-
-Because transactions can take a while, they are done in coroutines to prevent the interface from locking
-up.
-
-First we construct a list of arguments we are going to pass to the transaction in MintAndSave.cdc. This
-list consists of a single Dictionary containing the "Text" and "URL" keys and String values from the Mint
-panel. We use Cadence.Convert to convert from a `Dictionary` into a Cadence `{String:String}`
-for the argument.
-
-The MintAndSave.cdc file contains the transaction that will be executed.
-
-```cadence
-import SDKExampleNFT from 0xf8d6e0586b0a20c7
-import NonFungibleToken from 0xf8d6e0586b0a20c7
-
-transaction(md: {String:String}) {
- let acct : auth(Storage, Capabilities) &Account
-
- prepare(signer: auth(Storage, Capabilities) &Account) {
- self.acct = signer
- }
-
- execute {
- // Create collection if it doesn't exist
- if self.acct.storage.borrow<&SDKExampleNFT.Collection>(from: SDKExampleNFT.CollectionStoragePath) == nil
- {
- // Create a new empty collection
- let collection <- SDKExampleNFT.createEmptyCollection()
- // save it to the account
- self.acct.save(<-collection, to: SDKExampleNFT.CollectionStoragePath)
- // link a public capability for the collection
- let newCap = self.acct.capabilities.storage.issue<&{SDKExampleNFT.CollectionPublic, NonFungibleToken.CollectionPublic}>(
- SDKExampleNFT.CollectionStoragePath
- )
- self.acct.capabilities.publish(newCap, to: SDKExampleNFT.CollectionPublicPath)
- }
-
- //Get a reference to the minter
- let minter = getAccount(0xf8d6e0586b0a20c7)
- .capabilities.get(SDKExampleNFT.MinterPublicPath)
- .borrow<&{SDKExampleNFT.PublicMinter}>()
-
-
- //Get a CollectionPublic reference to the collection
- let collection = self.acct.capabilities.get(SDKExampleNFT.CollectionPublicPath)
- .borrow<&{NonFungibleToken.CollectionPublic}>()
-
- //Mint a new NFT and deposit into the authorizers account
- minter?.mintNFT(recipient: collection!, metadata: md)
- }
-}
-```
-
-This transaction checks to see if an SDKExampleNFT collection exists on the account, creating/saving/linking it if it does
-not. Then it calls the contract to mint a new NFT with the desired metadata and saves it to the collection.
-
-### Listing NFTs
-
-The List button calls the UpdateNFTPanelCoroutine function that is responsible for populating the panel with information
-about the SDKExampleNFT resources in the account you are authenticated as.
-
-```csharp
-public IEnumerator UpdateNFTPanelCoroutine()
-{
- //Create the script request. We use the text in the GetNFTsOnAccount.cdc file and pass the address of the
- //authenticated account as the address of the account we want to query.
- FlowScriptRequest scriptRequest = new FlowScriptRequest
- {
- Script = listScript.text,
- Arguments = new List
- {
- new CadenceAddress(FlowSDK.GetWalletProvider().GetAuthenticatedAccount().Address)
- }
- };
-
- //Execute the script and wait until it is completed.
- Task scriptResponse = Scripts.ExecuteAtLatestBlock(scriptRequest);
- yield return new WaitUntil(() => scriptResponse.IsCompleted);
-
- //Destroy existing NFT display prefabs
- foreach (TMP_Text child in NFTContentPanel.GetComponentsInChildren())
- {
- Destroy(child.transform.parent.gameObject);
- }
-
- //Iterate over the returned dictionary
- Dictionary> results = Convert.FromCadence>>(scriptResponse.Result.Value);
- //Iterate over the returned dictionary
- foreach (KeyValuePair> nft in results)
- {
- //Create a prefab for the NFT
- GameObject prefab = Instantiate(NFTPrefab, NFTContentPanel.transform);
-
- //Set the text
- string text = $"ID: {nft.Key}\n";
- foreach (KeyValuePair pair in nft.Value)
- {
- text += $" {pair.Key}: {pair.Value}\n";
- }
-
- prefab.GetComponentInChildren().text = text;
- }
-}
-```
-
-When running a script, you can query any account. In this case we will only query the account
-that is authenticated with the wallet provider.
-
-It executes the script defined in GetNFTsOnAccount.cdc:
-
-```cadence
-import SDKExampleNFT from 0xf8d6e0586b0a20c7
-
-access(all) fun main(addr:Address): {UInt64:{String:String}} {
-
- //Get a capability to the SDKExampleNFT collection if it exists. Return an empty dictionary if it does not
- let collectionCap = getAccount(addr).capabilities.get<&{SDKExampleNFT.CollectionPublic}>(SDKExampleNFT.CollectionPublicPath)
- if(!collectionCap.check())
- {
- return {}
- }
-
- //Borrow a reference to the capability, returning an empty dictionary if it can not borrow
- let collection = collectionCap.borrow()
- if(collection == nil)
- {
- return {}
- }
-
- //Create a variable to store the information we extract from the NFTs
- var output : {UInt64:{String:String}} = {}
-
- //Iterate through the NFTs, extracting id and metadata from each.
- for id in collection?.getIDs()! {
- log(collection!.borrowSDKExampleNFT(id:id))
- log(collection!.borrowSDKExampleNFT(id:id)!.metadata)
- output[id] = collection!.borrowSDKExampleNFT(id:id)!.metadata;
- }
-
- //Return the constructed data
- return output
-}
-```
-
-This ensures that an SDKExampleNFT.Collection resource exists at the proper path, then creates and returns
-a ```{UInt64:{String:String}}``` containing the information of all SDKExampleNFTs in the collection. We use
-Cadence.Convert to convert this into a C# ```Dictionary>```
-
-After that we Instantiate prefabs to display the data of each of the returned NFTs.
\ No newline at end of file
diff --git a/docs/tools/clients/unity-sdk/samples/quickstart.md b/docs/tools/clients/unity-sdk/samples/quickstart.md
deleted file mode 100644
index 08bba4f5ff..0000000000
--- a/docs/tools/clients/unity-sdk/samples/quickstart.md
+++ /dev/null
@@ -1,378 +0,0 @@
-ο»Ώ# FlowControl scripting example
-This example shows how to use FlowControl to perform common tasks purely from C#, without using
-the FlowControl editor interface. This will ignore any of the GUI related code and focus on
-using the FlowControl and Flow SDK.
-
-## Prerequisites
-Ensure you have flow-cli installed. This will allow us to use an emulated flow environment.
-You can install it by following the instructions at [Flow CLI](../../../../tools/flow-cli/install.md)
-
-## Sample walk through
-You can follow along in FlowControlExample.cs
-
-The first thing to notice is that we declare Start() to be an IEnumerator. This makes Start a coroutine.
-You will always want to run FlowSDK functions inside a coroutine because they can take a while to complete
-and you don't want to lock up your game while they are processed.
-
-```csharp
-private IEnumerator Start()
-{
- // Your start code
-}
-```
-
-## Checking emulator state
-The next thing we do is ensure the emulator is running. We give it a few seconds to start:
-
-```csharp
-//Wait up to 2.5 seconds for the emulator to start.
-int waited = 0;
-
-while (!FlowControl.IsEmulatorRunning && waited < 5)
-{
- waited++;
- yield return new WaitForSeconds(.5f);
-}
-
-if (!FlowControl.IsEmulatorRunning)
-{
- //Stop execution if the emulator is not running by now.
- yield break;
-}
-```
-
-## Creating a FlowControl Account
-Next we'll create a FlowControl account to use ***ONLY*** for running scripts. The Flow network
-doesn't require an account to run scripts, but FlowControl uses Accounts to determine which network
-to connect to.
-
-```csharp
-FlowControl.Account scriptOnlyAccount = new FlowControl.Account
-{
- GatewayName = "Emulator"
-};
-```
-
-Because this account doesn't have any keys associated with it, it can't be used to run transactions.
-It does define which Gateway to use, in this case the "Emulator" gateway, so it can be used
-to run scripts.
-
-## Running scripts
-
-Next, we'll use this account to run a script on the emulator. Scripts on Flow are written in Cadence.
-More information is available at [Developer Portal](https://cadence-lang.org/docs/language/)
-
-First we'll define the script that we want to run:
-
-```csharp
-const string code = @"access(all) fun main(message: String): Int{
- log(message)
- return 42
-}";
-```
-
-This script requires a Cadence String as input, returns a Cadence Int, and will log the input
-string to the emulator log.
-
-Now we execute this script:
-
-```csharp
-Task task = scriptOnlyAccount.ExecuteScript(code, Convert.ToCadence("Test", "String"));
-```
-
-FlowControl uses an Account oriented approach. Everything is done using an Account object. In this
-case we'll use the scriptOnlyAccount account that we created earlier to call ExecuteScript.
-
-A script is code that can not permanently mutate the state of the blockchain. It is read-only.
-It ***CAN*** call functions that would change the state of the blockchain, but any changes that are
-made will be discarded once the script finishes running.
-
-We pass in the Cadence code we want to run and any arguments that are required by the script.
-We need to use Cadence specific data types, so we construct a new CadenceString using the string
-"Test".
-
-This returns a `Task`. This is an asynchronous Task that will result
-in a FlowScriptResponse when it is complete.
-
-Next, we need to wait for the Task to complete. Inside a Unity coroutine we can use the `WaitUntil`
-function as follows:
-
-```csharp
-yield return new WaitUntil(() => task.IsCompleted);
-```
-
-`WaitUntil` takes a function that returns a bool (`Func`), so we construct an anonymous one that
-returns the IsCompleted field of the task. This cause Unity to pause execution of the current coroutine
-until the task is completed.
-
-We then check to see if an error occured, and if so, log it to the console.
-
-```csharp
-if (task.Result.Error != null)
-{
- Debug.LogError($"Error: {task.Result.Error.Message}");
- yield break;
-}
-```
-
-If there is no error, the script should have returned a Cadence Int value. We can access it as follows:
-
-```csharp
-Debug.Log($"Script result: {Convert.FromCadence(task.Result.Value)}");
-```
-
-This might be a bit confusing. The Task will have a Result. The result could contain an error,
-but we checked for that earlier. If it doesn't contain an error, then it will contain a Value.
-
-That Value will be of type CadenceBase, which is the base type for all Cadence data types. We
-know that the script returns a number, so we can convert it to an appropriate data type using Convert.FromCadence.
-
-## Creating an SdkAccount
-
-Next, let's create an account that can be used to execute transactions that mutate the state of
-the blockchain. This will also demonstrate how you can use both FlowControl and the base SDK
-together.
-
-```csharp
-SdkAccount emulatorSdkAccount = FlowControl.GetSdkAccountByName("emulator_service_account");
-if (emulatorSdkAccount == null)
-{
- Debug.LogError("Error getting SdkAccount for emulator_service_account");
- yield break;
-}
-```
-
-When the emulator is started, FlowControl automatically creates an emulator_service_account FlowControl.Account
-for you to use to access the built in emulator service account. We'll use that account to create a new account.
-
-Because the `CreateAccount` function is an SDK function, and not a FlowControl function, we'll need to create a
-temporary `SdkAccount` from the FlowControl Account. The `GetSdkAccountByName` function will construct an
-SdkAccount object from a FlowControl.Account object.
-
-If the name you pass to `FlowControl.GetSdkAccountByName` does not exist, it will return null, so we check
-for that and stop execution if it fails.
-
-## Creating an account on the blockchain
-
-Now we'll use this new SdkAccount object to create a new Flow account on the emulated blockchain.
-
-```csharp
-FlowSDK.RegisterWalletProvider(ScriptableObject.CreateInstance());
-
-string authAddress = "";
-FlowSDK.GetWalletProvider().Authenticate("", (string address) =>
-{
- authAddress = address;
-}, null);
-
-yield return new WaitUntil(() => { return authAddress != ""; });
-
-//Convert FlowAccount to SdkAccount
-SdkAccount emulatorSdkAccount = FlowControl.GetSdkAccountByAddress(authAddress);
-if (emulatorSdkAccount == null)
-{
- Debug.LogError("Error getting SdkAccount for emulator_service_account");
- yield break;
-}
-
-//Create a new account with the name "User"
-Task newAccountTask = CommonTransactions.CreateAccount("User");
-yield return new WaitUntil(() => newAccountTask.IsCompleted);
-
-if (newAccountTask.Result.Error != null)
-{
- Debug.LogError($"Error creating new account: {newAccountTask.Result.Error.Message}");
- yield break;
-}
-
-outputText.text += "DONE\n\n";
-
-//Here we have an SdkAccount
-SdkAccount userSdkAccount = newAccountTask.Result;
-
-```
-First we create and register a new `DevWalletProvider`. Any time a transaction is run, it calls the provided wallet provider. The `DevWalletProvider`
-is an implementation of IWallet that shows a simulated wallet interface. It will allow you to view and authorize the submitted transaction.
-
-After creating and registering the wallet provider, we call `Authenticate` to display a popup that will allow you to select any of the accounts in the FlowControl
-Accounts tab. You should choose emulator_service_account when prompted when running the demo.
-
-We then wait until the user has selected an account.
-
-`CommonTransactions` contains some utility functions to make performing frequent operations a little easier.
-One of these is `CreateAccount`. It expects a `Name`, which is not placed on the blockchain, and the SdkAccount
-that should pay for the creation of the new account. That returns a Task that is handled similarly to
-before.
-
-If there is no error, the Result field of the task will contain the newly create account info.
-
-Now, in order to use this new account with FlowControl, we'll need to create a FlowControl.Account from
-the SdkAccount we have.
-
-```csharp
-FlowControl.Account userAccount = new FlowControl.Account
-{
- Name = userSdkAccount.Name,
- GatewayName = "Emulator",
- AccountConfig = new Dictionary
- {
- ["Address"] = userSdkAccount.Address,
- ["Private Key"] = userSdkAccount.PrivateKey
- }
-};
-```
-
-Then we store this account in the FlowControlData object so that we can look it up by name later.
-
-```csharp
-FlowControl.Data.Accounts.Add(userAccount);
-```
-
-## Deploying a contract
-
-The next section shows how to deploy a contract to the Flow network. Because this is another utility
-function from `CommonTransactions`, it needs an SdkAccount. We'll use userSdkAccount we created earlier.
-
-First we need to define the contract we wish to deploy.
-
-```csharp
-const string contractCode = @"
- access(all) contract HelloWorld {
- access(all) let greeting: String
-
- access(all) event TestEvent(field: String)
-
- init() {
- self.greeting = ""Hello, World!""
- }
-
- access(all) fun hello(data: String): String {
- emit TestEvent(field:data)
- return self.greeting
- }
- }";
-```
-
-We won't discuss how to write Flow contracts in depth here, but simply put this contract defines a single
-function that will emit an event and return the string "Hello World!" when run.
-
-Then we use the same pattern we've used before to deploy this contract using the `CommonTransaction.DeployContract`
-function. Note that we should register a new wallet provider since we are changing the account we want to run the transaction
-as.
-
-```csharp
-FlowSDK.GetWalletProvider().Authenticate(userAccount.Name, null, null);
-Task deployContractTask =
- CommonTransactions.DeployContract("HelloWorld", contractCode);
-
-yield return new WaitUntil(() => deployContractTask.IsCompleted);
-
-if (deployContractTask.Result.Error != null)
-{
- Debug.LogError($"Error deploying contract: {deployContractTask.Result.Error.Message}");
- yield break;
-}
-```
-We'll reauthenticate with the wallet provider to tell it to use the new newly created account. Because we pass in a name this time, it
-won't display the select account pop-up.
-
-The first argument to `DeployContract` is the contract name. This must match the name in the contract
-data itself. The second argument is the Cadence code that defines the contract, and the third argument
-is the SdkAccount that the contract should be deployed to.
-
-## Replacement text
-
-Next we'll see how to add a `ReplacementText` entry to FlowControl. This is typically done via the
-FlowControl editor interface, but can be done programatically as shown.
-
-```csharp
-FlowControl.TextReplacement newTextReplacement = new FlowControl.TextReplacement
-{
- description = "User Address",
- originalText = "%USERADDRESS%",
- replacementText = userSdkAccount.Address,
- active = true,
- ApplyToAccounts = new List { "User" },
- ApplyToGateways = new List { "Emulator" }
-};
-
-FlowControl.Data.TextReplacements.Add(newTextReplacement);
-```
-
-Note that we are setting `ApplyToAccounts` and `ApplyToGateways` so that this `TextReplacement` will be
-performed any time the FlowControl.Account account with the name "User" executes a function against the emulator.
-
-This new `TextReplacement` will be used when we execute a transaction using the contract we just deployed.
-
-## Transactions
-
-First we'll write the transaction we want to execute.
-
-```csharp
-string transaction = @"
- import HelloWorld from %USERADDRESS%
- transaction {
- prepare(acct: &Account) {
- log(""Transaction Test"")
- HelloWorld.hello(data:""Test Event"")
- }
- }";
-```
-
-Based on the `TextReplacement` we created earlier, `%USERADDRESS%` will be replaced with the Flow address
-of the user account we created. This will then call the `hello` function on the `HelloWorld` contract
-we deployed to the user account.
-
-Next we follow a similar pattern to before:
-
-```csharp
-Task transactionTask = userAccount.SubmitAndWaitUntilSealed(transaction);
-yield return new WaitUntil(() => transactionTask.IsCompleted);
-
-if (transactionTask.Result.Error != null || !string.IsNullOrEmpty(transactionTask.Result.ErrorMessage))
-{
- Debug.LogError($"Error executing transaction: {transactionTask.Result.Error?.Message??transactionTask.Result.ErrorMessage}");
- yield break;
-}
-```
-
-Here, we're using the `SubmitAndWaitUntilSealed` FlowControl function. This combines two SDK functions
-together. It first submits the transaction to the network. Then it polls the network until the network
-indicates that the transaction has been sealed and then returns the results.
-
-Because this is combining two operations together, there are two potential failure points. The first
-is a network error or syntax error that causes the submission to be rejected. This will be indicated
-in the `Result.Error` field. The second is something that goes wrong during the processing of the
-transaction after submission was successful. This will be indicated in the Result.ErrorMessage field.
-When using SubmitAndWaitUntilSealed or SubmitAndWaitUntilExecuted, you will want to check both of the
-error fields to ensure it has completed successfully.
-
-Finally, we check the events emitted by the transaction. Because submitting transactions returns before
-the transaction is actually processed, you can't return data directly from a transaction like you can
-with a script. Instead, you emit events that can be retrieved. We'll check the events of the completed
-transaction as follows:
-
-## Transaction Events
-
-```csharp
-FlowEvent txEvent = transactionTask.Result.Events.Find(x => x.Type.Contains("TestEvent"));
-
-//Show that the transaction finished and display the value of the event that was emitted during execution.
-//The Payload of the returned FlowEvent will be a CadenceComposite. We want the value associated with the
-//"field" field as a string
-Debug.Log($"Executed transaction. Event type: {txEvent.Type}. Event payload: {Convert.FromCadence(txEvent.Payload).field}");
-```
-
-We end up a with a list of Events that were emitted by a transaction in the `Result.Events` object. We
-use LINQ to find the event we're interested in. It will contain "TestEvent" in it.
-
-We need something to convert the Cadence TestEvent into, so we declared a C# class earlier:
-
-```csharp
-public class TestEvent
-{
- public String field;
-}
-```
-
-Then we have to get the payload from the event to display. We'll convert it into our TestEvent class and access the ```field``` field.
diff --git a/docs/tools/clients/unity-sdk/samples/ui-usage.md b/docs/tools/clients/unity-sdk/samples/ui-usage.md
deleted file mode 100644
index 0b9eb8f8d7..0000000000
--- a/docs/tools/clients/unity-sdk/samples/ui-usage.md
+++ /dev/null
@@ -1,364 +0,0 @@
-
-# Sample - FlowSDK Usage Demo
-
-This sample demonstrates how to call each of the APIs within the SDK. It's a simple UI containing input fields and buttons for testing out each API call, and displays information that is returned by the APIs. You will need to set up an emulator with applicable test data to use this sample.
-
-## Importing the Sample
-
-To import the sample, first add the Flow SDK to your project, then follow these steps.
-
-1. Open Package Manager.
-2. Select the Flow SDK package.
-3. On the right, expand Samples and click Import next to FlowSDK Usage Demo.
-
-![](../media/fd52a768f2367b763980b22d85479305.png)
-
-This will import the sample into a Samples folder under your Assets:
-
-![](../media/58d03fbcda193c672c0287da2f3f04ba.png)
-
-## Installing the Flow Emulator
-
-To install the Flow emulator, please follow the instructions for your platform here:
-
-[flow-cli/install](../../../../tools/flow-cli/index.md)
-
-Note that the Flow CLI and Flow emulator are bundled into the same command-line executable.
-
-To test that the Flow emulator is installed correctly, open a command line interface, type `flow` and press enter. You should see something like this:
-
-![](../media/be2bc6569ca5a475b3c147cf34f07c1a.png)
-
-## Configuring the Emulator
-
-The directory that you run the emulator from must contain a flow.json file. A default one can be created by running the command `flow init`, but this sample contains a flow.json file which is specific to the sample:
-
-![](../media/54f8a5235b61d70f09904c38ed3e5e5c.png)
-
-Firstly, open the scene that comes with the sample:
-
-![](../media/881d1efa0c8e4ff9bffe3382274a255c.png)
-
-Select the UI Canvas in the hierarchy:
-
-![](../media/82ea27a997c12eab231ad8bf2e49d335.png)
-
-In the inspector, scroll down to the bottom and click "Open Flow Control Window":
-
-![](../media/61267a392954311426b4144345f25dfd.png)
-
-The Flow Control Window contains all the editor tools that come with the Flow SDK. It should look like this:
-
-![](../media/6d33d20949b4923d1e805e8e21884935.png)
-
-If you see a message asking you to install the Flow emulator, then Unity has not detected your Flow installation. Click the Install button and follow the directions to install the Flow CLI \\ emulator.
-
-To run the emulator, we need to specify which directory to run it from, remembering that it requires a flow.json file. Click the browse button and select the folder in the sample that contains the flow.json file (it should be in **Assets/Samples/Flow SDK/\/FlowSDK Usage Demo/Scripts/Cadence**):
-
-![](../media/62ec9ee52b31041e7420ac7ad7132684.png)
-
-Click on Start Emulator. The message at the bottom of the window should say **Emulator Status: Running**. Click on Emulator Log. This will open another window showing the output of the emulator:
-
-![](../media/f6a8324547c8ec499a2eb68248091ba3.png)
-
-To test the emulator, open a command line interface and enter the command `flow blocks get latest`. You should see something like this:
-
-![](../media/270ed3d6ac5df815d35cc82f588a4a18.png)
-
-The command retrieves information about the latest block on the blockchain. The latest block is currently block height 0, meaning that there is only one block on the blockchain, the root block (hence why Parent ID is null). Now we're going to add some more blocks.
-
-In your file explorer, navigate to the folder that the emulator is running from. Execute the **emulator_test_data** script for your platform (**.bat** for windows, **.sh** for mac/linux). When the script has finished, go back to your command line interface and enter the command `flow blocks get latest` again. Now you should see something like this:
-
-![](../media/2466ac711f979da15f6ad5a6cdb6421a.png)
-
-There are now 11 blocks on your blockchain emulator - the latest block is at block height 10 (remember the first one was block height 0). Every block has its own ID and the ID of its parent, the previous block in the chain.
-
-You are now ready to run the sample app.
-
-## Running the FlowSDK Usage Demo
-
-Click the Play button in the editor. You will see the following:
-
-![](../media/3421ca4434716e8f2efc89bbf5f3c7ad.png)
-
-The buttons along the bottom display different tabs which correspond to different API calls within the SDK. The tabs demonstrate the following API calls:
-
-**Blocks**
-
-*GetById*
-*GetByHeight*
-*GetLatest*
-
-**Collections**
-
-*GetById*
-
-**Events**
-
-*GetForBlockHeightRange*
-*GetForBlockIds*
-
-**Scripts**
-
-*ExecuteAtLatestBlock*
-
-**Transactions**
-
-*Submit*
-*GetById*
-*GetResult*
-
-**Accounts**
-
-*GetByAddress*
-
-We will now discuss each of the tests throughout the sample. As you read through each of these, it's recommended to look at the code calling these APIs. All of the code is contained here:
-
-![](../media/15840cee138659b172d6023791b73155.png)
-
-### Blocks
-
-Blocks are the foundation of blockchains, and Flow is no different. A Block primarily consists of two things - Collections and Block Seals. Collections contain Transactions that are yet to be executed, while Block Seals contain the results of Transactions that have been executed and verified, and are ready to be sealed.
-
-**Get Block by ID**
-
-*MainUIController.BlocksGetById()*
-
-Enter a valid Block ID and click the button to retrieve information about the block.
-
-**Get Block by Height**
-
-*MainUIController.BlocksGetByHeight()*
-
-Enter a valid Block Height to retrieve information about the block at that height.
-
-**Get Latest Block**
-
-*MainUIController.BlocksGetLatest()*
-
-Simply click the button to retrieve information about the latest block.
-
-### Collections
-
-A Collection contains a list of Transactions to be executed.
-
-**Get Collection by ID**
-
-*MainUIController.CollectionsGetById()*
-
-Enter a valid Collection ID to retrieve information about the collection.
-
-### Events
-
-Events are defined and emitted in your Smart Contracts. You would emit an event when you want to later query that something has happened.
-
-**Get Events for Height Range**
-
-*MainUIController.EventsGetForHeightRange()*
-
-This will retrieve any events of the given type emitted in a given block height range. Enter the Type, Start Block Height and End Block Height and click the button.
-
-**Get Events for Block Ids**
-
-*MainUIController.EventsGetForBlockIds()*
-
-Similar to the previous API, but instead of providing a block height range, you provide a comma separated list of Block Ids. Any events matching the Type that were emitted in those Block Ids will be returned.
-
-For both of the API calls, the dropdown in the bottom left will populate with a list of blocks. Selecting a block in the list will populate the right-hand side with the event results.
-
-#### Event Types
-
-For both of these API calls, the Event Type is a fully qualified type in the following format:
-
-```
-A.{contract address}.{contract name}.{event name}
-```
-
-Contract address - the account that the Smart Contract which defines the event is deployed to.
-Contract name - the name of the Smart Contract which defines the event.
-Event name - the name of the event as defined in the Smart Contract.
-
-Here is an example of an Event Type:
-
-```
-A.7e60df042a9c0868.FlowToken.TokensInitialized
-```
-
-#### Core Events
-
-There are a few core events that are emitted directly by Flow and don't follow the standard naming convention. These event types are:
-
-```
-flow.AccountCreated
-flow.AccountKeyAdded
-flow.AccountKeyRemoved
-flow.AccountContractAdded
-flow.AccountContractUpdated
-flow.AccountContractRemoved
-```
-
-### Scripts
-
-Scripts are cadence code that you write and are executed on the blockchain. They can contain arguments and return values, and can interact with Smart Contracts. Scripts are read-only - they cannot mutate anything on the blockchain. Anything held in local memory is discarded when the script finishes execution. For more information on programming see [Cadence](https://cadence-lang.org/docs).
-
-**Execute Simple Script At Latest Block**
-
-*MainUIController.ExecuteSimpleScript()*
-
-This executes a very simple script which returns a string message. The message is printed to the UI.
-
-**Execute Script With Args At Latest Block**
-
-*MainUIController.ExecuteScriptWithArgs()*
-
-This demonstrates sending arguments with a simple script. Enter a number in each of the input fields and click the button. The script will add the numbers together and return the result, printing it to the UI.
-
-**Execute Get Tokens At Latest Block**
-
-*MainUIController.ExecuteGetTokens()*
-
-The **emulator_test_data** script that you ran earlier deployed a Smart Contract for fungible tokens called FlowSDKSampleToken. It also created a few accounts and minted some of these tokens into a couple of accounts.
-
-This test demonstrates passing an account address as an argument to a script, which returns a single value - the number of FlowSDKSampleTokens that account has.
-
-In the Flow Control Window, click the Accounts tab. This shows a few accounts which you can get the address of to test the script.
-
-![](../media/51d4f1e5f01dc36b0fdd0853eb08b0bd.png)
-
-If you would like to see the cadence script itself, it is located here:
-
-![](../media/f30282a8889db37965bcd49b209ac614.png)
-
-**Execute Print NFTs At Latest Block**
-
-*MainUIController.ExecutePrintNfts()*
-
-The **emulator_test_data** script also deployed a contract called FlowSDKSampleNFT, an example implementation of an NFT. It also minted a few of these NFTs and deposited them into a couple of accounts.
-
-This test is similar to the previous one - enter an account address into the input field and click the button to retrieve a list of NFTs that account owns. Instead of returning a single value, the script returns an array of values (NFT Ids).
-
-The cadence script for this test is in the same location as the previous test, called **print-nfts.cdc**.
-
-**Execute Get NFTs For Sale At Latest Block**
-
-*MainUIController.ExecuteGetNftsForSale()*
-
-This script does not have any arguments, but demonstrates returning a complex data structure - an array of structs. The struct, which has multiple properties, is defined in the script itself.
-
-The data being returned is a list of NFTs that are for sale, and contains the Id, price and owner's address for each NFT. Right now it will not return anything, because there are no NFTs for sale. You will need to run the **Submit Transaction List NFT** test in the Transactions tab for this script to display anything.
-
-The cadence script for this test is in the same location as the others, called **get-nfts-for-sale.cdc**.
-
-### Transactions
-
-Like Scripts, Transactions are cadence code that you write and are executed on the blockchain, but that is where the similarities end. The purpose of Transactions is to mutate data on the blockchain. To do this, the Transaction must be signed by the account/s that are going to be affected. The Transaction code can contain arguments, but can't return a value. This is because there is a delay for the Transaction to execute, because it has to go through the collection/consensus/execution/verification cycle.
-
-For more information about Transactions, see [cadence/language/transactions](https://cadence-lang.org/docs/language/transactions).
-
-For more information about Transaction signing, see [concepts/transaction-signing](../../../../build/basics/transactions.md#signing-a-transaction).
-
-For more information about Cadence programming, see [Cadence](https://cadence-lang.org/docs).
-
-**Signing In**
-
-To sign transactions, you must sign-in to a wallet provider. This sample uses DevWallet, an example wallet provider specifically made for development. For more information see [DevWallet](../guides/dev-wallet.md).
-
-Click the Sign In button. This will open a dialog asking to select an account to sign in with. The list of accounts is populated from the Accounts tab of the Flow Control Window. Select an account and click Ok. The address of the account you select will show underneath "Signed in as". You can now run the three transaction tests.
-
-**Submit Transaction**
-
-*MainUIController.SubmitTxSinglePayerProposerAuthorizer()*
-
-This is a very simple Transaction that logs out a message on the blockchain. Simply click the first Submit Transaction button on the left. Successfully submitting a Transaction will return a Transaction Id.
-
-Now go to the Flow Emulator Output and scroll to the bottom. You should see several things - the transaction was submitted, executed, the log message in the cadence code was printed, and a new block was committed.
-
-![](../media/db2741c083bc677b767e5f2973f63e1b.png)
-
-**Submit Transaction With Args**
-
-*MainUIController.SubmitTxWithArgs()*
-
-This is a very simple Transaction like the previous one which doesn't actually mutate anything, but simply demonstrates passing arguments to the Transaction. Enter a string argument and a number argument, then click the next Submit Transaction button. Like the previous test, the arguments are logged out on the blockchain.
-
-![](../media/f0dfb75d21cf7462d7f0aaf199d97b27.png)
-
-**Submit Transaction List NFT**
-
-*MainUIController.SubmitTxListNft()*
-
-This Transaction demonstrates an actual use case of listing an account's NFT for sale. Enter the NFT Id and price, then click the last Submit Transaction button. To get a list of NFT Ids for a given account, go back to the Scripts tab and run the "Execute Print NFTs At Latest Block" script.
-
-Note: The NFT Price must be a decimal number, eg 23.5 or 12.0
-
-If you'd like to see the cadence code for this Transaction, it's located here:
-
-![](../media/5ba73beddcefba8eade5c9ce4322a86d.png)
-
-**Get Transaction By Id**
-
-*MainUIController.GetTransactionById()*
-
-This API call will return all of the information that was submitted with the Transaction. For ease of use, the input field for this is automatically populated with the Transaction Id of the previous Transaction that was submitted. You can of course enter a different valid Transaction Id. Clicking the button will populate all the fields under the Transaction heading.
-
-**Get Transaction Result**
-
-*MainUIController.GetTransactionResult()*
-
-This demonstrates how to retrieve the result of a Transaction. You must supply the Transaction Id using the same input as the previous test. As previously mentioned, the input is automatically populated with the Transaction Id of the last submitted Transaction.
-
-The fields under the Result heading are populated when running this API call. The following information is returned.
-
-Status: Current status of the Transaction. Can be Unknown, Pending, Finalized, Executed, Sealed or Expired.
-Error Message: If there was an error executing the Transaction (eg a cadence runtime error), this will be populated with the error message from the blockchain. When running against an emulator, you will also see the error in the Flow Emulator Output.
-Events: A list of events that were emitted during execution of the Transaction.
-
-### Accounts
-
-**Signing In**
-
-Like the Transactions tab, you must be signed in to a wallet provider to run most of these tests. That's because they are actually transactions, and therefore require signing. If you already signed in on the Transactions tab, you will still be signed in as the same user here. Otherwise, Click Sign In and select an account, then click Ok.
-
-**Get Account**
-
-*MainUIController.GetAccount()*
-
-Retrieves details about the given account address, including Flow token balance, public keys and any deployed Smart Contracts. This is not a transaction, therefore you do not need to be signed in to run this test.
-
-**Create Account**
-
-*MainUIController.CreateAccount()*
-
-To create an account on Flow, it needs to be sponsored by another account. Every account on Flow must have a minimum of 0.001 Flow tokens in their balance, so the sponsor must pay this to the new account. To test this, enter a name for the new account, then click Create Account. The account you are signed in as will sign the transaction and pay the 0.001 Flow tokens to the new account. If it fails, ensure the account has enough Flow tokens to execute this transaction and still retain the minimum 0.001 Flow tokens.
-
-A new public\\private keypair will be generated and assigned to the new account.
-
-Note: the Account Name is not actually stored on the blockchain. It is only used by the SDK.
-
-**Deploy Contract**
-
-*MainUIController.DeployContract()*
-
-This demonstrates deploying a Smart Contract to an account. Enter the name of the contract and the path of its source file, then click Deploy Contract.
-
-Note that the Contract Name must match the name of the contract in the source file.
-
-The Contract Source File location is relative to the project's Assets directory.
-
-**Remove Contract**
-
-*MainUIController.RemoveContract()*
-
-Removes a Smart Contract from an account. Enter the name of the contract then click Remove Contract.
-
-**Update Contract**
-
-*MainUIController.UpdateContract()*
-
-Updates a Smart Contract which is already deployed to the signed-in account. Unlike Deploy Contract, the `init()` function of the contract is not executed. Enter the Contract Name and the path of the updated source file, then click Update Contract.
-
-Note that the Contract Name must match the name of the contract in the source file, and must also match the name of the existing contract deployed to the account.
-
-The Contract Source File location is relative to the project's Assets directory.
-
-Only certain things in a Smart Contract can be modified. See this link about [Contract Updatability](https://cadence-lang.org/docs/language/contract-updatability).
diff --git a/docs/tools/flow-cli/tests.md b/docs/tools/flow-cli/tests.md
new file mode 100644
index 0000000000..c5d83a4c08
--- /dev/null
+++ b/docs/tools/flow-cli/tests.md
@@ -0,0 +1,227 @@
+---
+title: Running Cadence Tests
+sidebar_label: Running Cadence Tests
+description: How to run Cadence tests from the CLI
+sidebar_position: 11
+---
+
+The Flow CLI provides a straightforward command to execute Cadence tests, enabling developers to validate their scripts and smart contracts effectively.
+
+To run all tests in your project, simply use:
+
+```shell
+flow test
+```
+
+The `flow test` command automatically discovers and runs all test scripts in your project that end with `_test.cdc`.
+
+> **Note:** The `test` command requires a properly initialized configuration. If you havenβt set up your Flow project yet, refer to the [flow init](flow.json/initialize-configuration.md) guide for assistance.
+
+## Prerequisites
+
+Before running your tests, ensure that your contracts are properly configured in your `flow.json` file, including any necessary testing aliases.
+
+### Setting Up Testing Aliases in Contracts
+
+If your tests involve deploying or interacting with contracts, you need to add your contracts to the `contracts` section in the `flow.json` configuration file. Specifically, include the contract name, source location, and an address alias for the `testing` environment.
+
+Example `flow.json` configuration:
+
+```json
+{
+ "contracts": {
+ "Counter": {
+ "source": "cadence/contracts/Counter.cdc",
+ "aliases": {
+ "testing": "0x0000000000000007"
+ }
+ }
+ },
+ "networks": {
+ // ... your network configurations
+ },
+ "accounts": {
+ // ... your account configurations
+ },
+ "deployments": {
+ // ... your deployment configurations
+ }
+}
+```
+
+For the `testing` alias, you can use one of the following addresses:
+
+- `0x0000000000000005`
+- `0x0000000000000006`
+- `0x0000000000000007`
+- `0x0000000000000008`
+- `0x0000000000000009`
+- `0x000000000000000A`
+- `0x000000000000000B`
+- `0x000000000000000C`
+- `0x000000000000000D`
+- `0x000000000000000E`
+
+> **Note**: For more information on setting up contracts and aliases, refer to the [Flow CLI Configuration](flow.json/initialize-configuration.md) documentation.
+
+## Example Usage
+
+Assuming you have a test script named `test_script_test.cdc` in your project directory, which verifies the functionality of a Cadence script executed in the testing environment:
+
+```cadence
+// test_script_test.cdc
+import Test
+
+access(all) let blockchain = Test.newEmulatorBlockchain()
+
+access(all) fun testSumOfTwo() {
+ let scriptResult = blockchain.executeScript(
+ "access(all) fun main(a: Int, b: Int): Int { return a + b }",
+ [2, 3]
+ )
+
+ Test.expect(scriptResult, Test.beSucceeded())
+
+ let sum = scriptResult.returnValue! as! Int
+ Test.assertEqual(5, sum)
+}
+```
+
+This script defines a single test case, `testSumOfTwo`, which checks if a Cadence script that adds two integers `(a + b)` works as expected. The test passes if the result matches the expected value of `5`.
+
+You can run all tests in your project using the CLI:
+
+```shell
+$ flow test
+```
+
+The Flow CLI will discover all test scripts ending with `_test.cdc` and execute them. The results will be displayed in the terminal:
+
+```shell
+Test results:
+- PASS: test_script_test.cdc > testSumOfTwo
+```
+
+To learn more about writing tests in Cadence, visit the [Cadence Testing Framework](../../build/smart-contracts/testing.md) documentation.
+
+---
+
+### Running Specific Tests
+
+If you wish to run a specific test script rather than all tests, you can provide the path to the test file:
+
+```shell
+flow test path/to/your/test_script_test.cdc
+```
+
+This will execute only the tests contained in the specified file.
+
+---
+
+## Flags
+
+The `flow test` command supports several flags that provide additional functionality for managing test execution and coverage reporting.
+
+### **Coverage Report**
+
+- **Flag:** `--cover`
+- **Default:** `false`
+
+The `--cover` flag calculates the coverage of the code being tested, helping you identify untested parts of your scripts and contracts.
+
+```shell
+$ flow test --cover
+```
+
+Sample output:
+
+```shell
+Test results:
+- PASS: test_script_test.cdc > testSumOfTwo
+Coverage: 96.5% of statements
+```
+
+---
+
+### Coverage Report Output File
+
+- **Flag:** `--coverprofile`
+- **Valid Inputs:** A valid filename with extension `.json` or `.lcov`
+- **Default:** `"coverage.json"`
+
+Use the `--coverprofile` flag to specify the output file for the coverage report.
+
+Example:
+
+```shell
+$ flow test --cover --coverprofile="coverage.lcov"
+```
+
+The generated coverage file can then be inspected:
+
+```shell
+$ cat coverage.lcov
+```
+
+### Coverage Code Type
+
+- **Flag:** `--covercode`
+- **Valid Inputs:** `"all"` (default) or `"contracts"`
+- **Default:** `"all"`
+
+The `--covercode` flag lets you limit the coverage report to specific types of code. Setting the value to `"contracts"` excludes scripts and transactions from the coverage analysis.
+
+```shell
+$ flow test --cover --covercode="contracts"
+```
+
+Sample output when no contracts are present:
+
+```shell
+Test results:
+- PASS: test_script_test.cdc > testSumOfTwo
+There are no statements to cover
+```
+
+> **Note:** In this example, the coverage report is empty because the `--covercode` flag is set to `"contracts"`, and the test script only contains scripts, not contracts.
+
+### Random Execution of Test Cases
+
+- **Flag:** `--random`
+- **Default:** `false`
+
+Use the `--random` flag to execute test cases in a random order. This can help identify issues that may arise due to test dependencies or the order in which tests are run.
+
+```shell
+flow test --random
+```
+
+### Seed for Random Execution
+
+- **Flag:** `--seed`
+- **Default:** `0`
+
+Use the `--seed` flag to specify a seed value for the random execution order of test cases. This allows you to reproduce a specific random order by using the same seed value, which is helpful for debugging flaky tests.
+
+```shell
+flow test --seed=12345
+```
+
+> **Note:** If both `--random` and `--seed` are provided, the `--random` flag will be ignored, and the seed value from `--seed` will be used for randomization.
+
+---
+
+### Run Specific Test by Name
+
+- **Flag:** `--name`
+- **Default:** `""` (empty string)
+
+Use the `--name` flag to run only tests that match the given name. This is useful when you want to execute a specific test function within your test scripts.
+
+```shell
+flow test --name=testSumOfTwo
+```
+
+This command will run only the test function named `testSumOfTwo` across all test scripts that contain it.
+
+To dive deeper into testing the functionality of your Cadence scripts and contracts, explore the [Cadence Testing Framework](https://cadence-lang.org/docs/testing-framework) documentation.
\ No newline at end of file
diff --git a/docs/tools/flow-cli/tests/_category_.json b/docs/tools/flow-cli/tests/_category_.json
deleted file mode 100644
index b6f8c563b2..0000000000
--- a/docs/tools/flow-cli/tests/_category_.json
+++ /dev/null
@@ -1,4 +0,0 @@
-{
- "label": "Tests",
- "position": 13
-}
\ No newline at end of file
diff --git a/docs/tools/flow-cli/tests/run-tests.md b/docs/tools/flow-cli/tests/run-tests.md
deleted file mode 100644
index 3e3b6b8840..0000000000
--- a/docs/tools/flow-cli/tests/run-tests.md
+++ /dev/null
@@ -1,96 +0,0 @@
----
-title: Run Cadence tests
-description: How to run Cadence tests from the command line
-sidebar_position: 11
----
-
-The Flow CLI provides a command to run Cadence tests.
-
-```shell
-flow test /path/to/test_script.cdc
-```
-
-β οΈ The `test` command expects configuration to be initialized. See [flow init](../flow.json/initialize-configuration.md) command.
-
-## Example Usage
-
-A simple Cadence script `test_script.cdc`, which has a test case for running a cadence script on-chain:
-```cadence
-import Test
-
-access(all) let blockchain = Test.newEmulatorBlockchain()
-
-access(all) fun testSumOfTwo() {
- let scriptResult = blockchain.executeScript(
- "access(all) fun main(a: Int, b: Int): Int { return a + b }",
- [2, 3]
- )
-
- Test.expect(scriptResult, Test.beSucceeded())
-
- let sum = scriptResult.returnValue! as! Int
- Test.assertEqual(5, sum)
-}
-```
-The above test-script can be run with the CLI as follows, and the test results will be printed on the console.
-```shell
-$ flow test test_script.cdc
-
-Test results: "test_script.cdc"
-- PASS: testSumOfTwo
-
-```
-
-To learn more about writing tests in Cadence, take a look at the [Cadence testing framework](../../../build/smart-contracts/testing.md).
-
-## Flags
-
-### Coverage
-
-- Flag: `--cover`
-- Default: `false`
-
-Use the `cover` flag to calculate coverage report for the code being tested.
-```shell
-$ flow test --cover test_script.cdc
-
-Test results: "test_script.cdc"
-- PASS: testSumOfTwo
-Coverage: 96.5% of statements
-
-```
-
-### Coverage Report File
-
-- Flag: `--coverprofile`
-- Valid inputs: valid filename and extension
-- Default: `"coverage.json"`
-
-Use the `coverprofile` to specify the filename where the calculated coverage report is to be written. Supported filename extensions are `.json` and `.lcov`.
-```shell
-$ flow test --cover test_script.cdc
-
-$ cat coverage.json
-
-$ flow test --cover --coverprofile="coverage.lcov" test_script.cdc
-
-$ cat coverage.lcov
-```
-
-### Coverage Code Type
-
-- Flag: `--covercode`
-- Valid inputs: `"all"`, `"contracts"`
-- Default: `"all"`
-
-Use the `covercode` flag to calculate coverage report only for certain types of code. A value of `"contracts"` will exclude scripts and transactions from the coverage report.
-```shell
-$ flow test --cover --covercode="contracts" test_script.cdc
-
-Test results: "tests/test_script.cdc"
-- PASS: testSumOfTwo
-There are no statements to cover
-```
-
-Since we did not use any contracts in our sample test script, there is no coverage percentage to be reported.
-
diff --git a/docs/tools/vscode-extension/index.md b/docs/tools/vscode-extension/index.md
index daec9d922e..dc12aa9da5 100644
--- a/docs/tools/vscode-extension/index.md
+++ b/docs/tools/vscode-extension/index.md
@@ -14,30 +14,8 @@ Note that most editing features (type checking, code completion, etc.) are imple
## Installation
-To install the extension, ensure you [have VS Code installed](https://code.visualstudio.com/docs/setup/mac)
-and have configured the [`code` command line interface](https://code.visualstudio.com/docs/setup/mac#_launching-from-the-command-line).
-
-### Using the Flow CLI
-
-The recommended way to install the latest released version is to use the Flow CLI.
-
-```shell script
-brew install flow-cli
-```
-
-Check that it's been installed correctly.
-
-```shell script
-flow version
-```
-
-Next, use the CLI to install the VS Code extension.
-
-```shell script
-flow cadence install-vscode-extension
-```
-
-Restart VS Code and the extension should be installed!
+To install the extension, ensure you have the [VS Code IDE installed](https://code.visualstudio.com/docs/setup/mac).
+Then, you can install the Cadence extension from the [VS Code Marketplace](https://marketplace.visualstudio.com/items?itemName=onflow.cadence).
## Developing the Extension
diff --git a/docs/tooling/wallet-provider-spec/assets/fcl-ars-auth-v1.excalidraw b/docs/tools/wallet-provider-spec/assets/fcl-ars-auth-v1.excalidraw
similarity index 100%
rename from docs/tooling/wallet-provider-spec/assets/fcl-ars-auth-v1.excalidraw
rename to docs/tools/wallet-provider-spec/assets/fcl-ars-auth-v1.excalidraw
diff --git a/docs/tooling/wallet-provider-spec/assets/fcl-ars-auth-v1.png b/docs/tools/wallet-provider-spec/assets/fcl-ars-auth-v1.png
similarity index 100%
rename from docs/tooling/wallet-provider-spec/assets/fcl-ars-auth-v1.png
rename to docs/tools/wallet-provider-spec/assets/fcl-ars-auth-v1.png
diff --git a/docs/tooling/wallet-provider-spec/assets/fcl-ars-auth-v2.excalidraw b/docs/tools/wallet-provider-spec/assets/fcl-ars-auth-v2.excalidraw
similarity index 100%
rename from docs/tooling/wallet-provider-spec/assets/fcl-ars-auth-v2.excalidraw
rename to docs/tools/wallet-provider-spec/assets/fcl-ars-auth-v2.excalidraw
diff --git a/docs/tooling/wallet-provider-spec/assets/fcl-ars-auth-v2.png b/docs/tools/wallet-provider-spec/assets/fcl-ars-auth-v2.png
similarity index 100%
rename from docs/tooling/wallet-provider-spec/assets/fcl-ars-auth-v2.png
rename to docs/tools/wallet-provider-spec/assets/fcl-ars-auth-v2.png
diff --git a/docs/tooling/wallet-provider-spec/assets/fcl-ars-auth-v3.1.excalidraw b/docs/tools/wallet-provider-spec/assets/fcl-ars-auth-v3.1.excalidraw
similarity index 100%
rename from docs/tooling/wallet-provider-spec/assets/fcl-ars-auth-v3.1.excalidraw
rename to docs/tools/wallet-provider-spec/assets/fcl-ars-auth-v3.1.excalidraw
diff --git a/docs/tooling/wallet-provider-spec/assets/fcl-ars-auth-v3.1.png b/docs/tools/wallet-provider-spec/assets/fcl-ars-auth-v3.1.png
similarity index 100%
rename from docs/tooling/wallet-provider-spec/assets/fcl-ars-auth-v3.1.png
rename to docs/tools/wallet-provider-spec/assets/fcl-ars-auth-v3.1.png
diff --git a/docs/tooling/wallet-provider-spec/assets/fcl-ars-auth-v3.2.excalidraw b/docs/tools/wallet-provider-spec/assets/fcl-ars-auth-v3.2.excalidraw
similarity index 100%
rename from docs/tooling/wallet-provider-spec/assets/fcl-ars-auth-v3.2.excalidraw
rename to docs/tools/wallet-provider-spec/assets/fcl-ars-auth-v3.2.excalidraw
diff --git a/docs/tooling/wallet-provider-spec/assets/fcl-ars-auth-v3.2.png b/docs/tools/wallet-provider-spec/assets/fcl-ars-auth-v3.2.png
similarity index 100%
rename from docs/tooling/wallet-provider-spec/assets/fcl-ars-auth-v3.2.png
rename to docs/tools/wallet-provider-spec/assets/fcl-ars-auth-v3.2.png
diff --git a/docs/tooling/wallet-provider-spec/assets/fcl-ars-auth-v3.excalidraw b/docs/tools/wallet-provider-spec/assets/fcl-ars-auth-v3.excalidraw
similarity index 100%
rename from docs/tooling/wallet-provider-spec/assets/fcl-ars-auth-v3.excalidraw
rename to docs/tools/wallet-provider-spec/assets/fcl-ars-auth-v3.excalidraw
diff --git a/docs/tooling/wallet-provider-spec/assets/fcl-ars-auth-v3.png b/docs/tools/wallet-provider-spec/assets/fcl-ars-auth-v3.png
similarity index 100%
rename from docs/tooling/wallet-provider-spec/assets/fcl-ars-auth-v3.png
rename to docs/tools/wallet-provider-spec/assets/fcl-ars-auth-v3.png
diff --git a/docs/tooling/wallet-provider-spec/authorization-function.md b/docs/tools/wallet-provider-spec/authorization-function.md
similarity index 100%
rename from docs/tooling/wallet-provider-spec/authorization-function.md
rename to docs/tools/wallet-provider-spec/authorization-function.md
diff --git a/docs/tooling/wallet-provider-spec/custodial.md b/docs/tools/wallet-provider-spec/custodial.md
similarity index 100%
rename from docs/tooling/wallet-provider-spec/custodial.md
rename to docs/tools/wallet-provider-spec/custodial.md
diff --git a/docs/tooling/wallet-provider-spec/index.md b/docs/tools/wallet-provider-spec/index.md
similarity index 100%
rename from docs/tooling/wallet-provider-spec/index.md
rename to docs/tools/wallet-provider-spec/index.md
diff --git a/docs/tooling/wallet-provider-spec/provable-authn.md b/docs/tools/wallet-provider-spec/provable-authn.md
similarity index 100%
rename from docs/tooling/wallet-provider-spec/provable-authn.md
rename to docs/tools/wallet-provider-spec/provable-authn.md
diff --git a/docs/tooling/wallet-provider-spec/user-signature.md b/docs/tools/wallet-provider-spec/user-signature.md
similarity index 100%
rename from docs/tooling/wallet-provider-spec/user-signature.md
rename to docs/tools/wallet-provider-spec/user-signature.md
diff --git a/docusaurus.config.js b/docusaurus.config.js
index f138c94f64..7c2e1cc563 100644
--- a/docusaurus.config.js
+++ b/docusaurus.config.js
@@ -304,7 +304,7 @@ const config = {
items: [
{
label: 'Getting Started',
- to: '/build/getting-started/hello-world',
+ to: '/build/getting-started/contract-interaction',
},
{
label: "SDK's & Tools",
diff --git a/src/data/tools/index.ts b/src/data/tools/index.ts
index 0e1acd2581..f840eba38d 100644
--- a/src/data/tools/index.ts
+++ b/src/data/tools/index.ts
@@ -629,5 +629,4 @@ export {
niftoryTool,
flowviewTool,
dapperSelfCustodyWallet,
- unitySDK,
};
diff --git a/src/ui/design-system/src/lib/Components/HomepageStartListCadence/index.tsx b/src/ui/design-system/src/lib/Components/HomepageStartListCadence/index.tsx
index 9837786578..a617064a9a 100644
--- a/src/ui/design-system/src/lib/Components/HomepageStartListCadence/index.tsx
+++ b/src/ui/design-system/src/lib/Components/HomepageStartListCadence/index.tsx
@@ -14,7 +14,7 @@ const homepageData: Record = {
icon: 'cadence-course',
},
'beginner-dapp': {
- link: '/build/getting-started/hello-world',
+ link: '/build/getting-started/contract-interaction',
icon: 'start-here',
},
'flow-quest': {
diff --git a/src/ui/design-system/src/lib/Components/LandingHeaderHome/index.tsx b/src/ui/design-system/src/lib/Components/LandingHeaderHome/index.tsx
index 3d84638a0f..67ebb33ebd 100644
--- a/src/ui/design-system/src/lib/Components/LandingHeaderHome/index.tsx
+++ b/src/ui/design-system/src/lib/Components/LandingHeaderHome/index.tsx
@@ -32,7 +32,7 @@ export function LandingHeaderHome({
Get Building
diff --git a/src/ui/design-system/src/lib/Components/LinkGrid/index.tsx b/src/ui/design-system/src/lib/Components/LinkGrid/index.tsx
index 6706c684f2..f3bd407ec6 100644
--- a/src/ui/design-system/src/lib/Components/LinkGrid/index.tsx
+++ b/src/ui/design-system/src/lib/Components/LinkGrid/index.tsx
@@ -25,7 +25,7 @@ const sections: LinkGridSection[] = [
links: [
{
title: 'Hello World Tutorial',
- href: '/build/getting-started/hello-world',
+ href: '/build/getting-started/contract-interaction',
},
{
title: 'App Architecture',
diff --git a/vercel.json b/vercel.json
index 8baa83c87f..9352f4601a 100644
--- a/vercel.json
+++ b/vercel.json
@@ -252,11 +252,6 @@
"destination": "/tools/clients/flow-go-sdk",
"permanent": true
},
- {
- "source": "/tools/unity-sdk/:path*",
- "destination": "/tools/clients/unity-sdk/:path*",
- "permanent": true
- },
{
"source": "/account-linking/:path*",
"destination": "/build/advanced-concepts/account-linking/:path*",
@@ -329,7 +324,7 @@
},
{
"source": "/tutorials/flow-dapp-anatomy",
- "destination": "/build/getting-started/app-architecture",
+ "destination": "/build/app-architecture",
"permanent": true
},
{
@@ -692,11 +687,6 @@
"destination": "/tools/clients/flow-go-sdk/:path*",
"permanent": true
},
- {
- "source": "/tooling/unity-sdk/:path*",
- "destination": "/tools/clients/unity-sdk/:path*",
- "permanent": true
- },
{
"source": "/tooling/:path*",
"destination": "/tools/:path*",
@@ -1248,6 +1238,11 @@
"destination": "/build/getting-started/hello-world",
"permanent": true
},
+ {
+ "source": "/build/getting-started/hello-world",
+ "destination": "/build/getting-started/contract-interaction",
+ "permanent": true
+ },
{
"source": "/build/getting-started/quickstarts/flow-cli",
"destination": "/build/getting-started/flow-cli",
@@ -1263,10 +1258,30 @@
"destination": "https://cadence-lang.org/docs/",
"permanent": true
},
+ {
+ "source": "/build/guides/flow-app-quickstart",
+ "destination": "/build/getting-started/fcl-quickstart",
+ "permanent": true
+ },
{
"source": "/build/guides/flow-cli",
"destination": "/tools/flow-cli/install",
"permanent": true
+ },
+ {
+ "source": "/flow-cli/:path*",
+ "destination": "/tools/flow-cli/:path*",
+ "permanent": true
+ },
+ {
+ "source": "/evm/vm-bridge",
+ "destination": "/evm/cadence/vm-bridge",
+ "permanent": true
+ },
+ {
+ "source": "/tools/flow-cli/tests/run-tests",
+ "destination": "/tools/flow-cli/tests",
+ "permanent": true
}
]
}
\ No newline at end of file