Skip to main content

Part 2: Create an Ionic mobile application

A web3 mobile application is not different from a web2 one in terms of its basic functionality and user interface. Both types of applications can run on smartphones, tablets, and other mobile devices, and both can access the internet and provide various services to users. However, a web3 mobile application differs from a web2 one in terms of its underlying architecture and design principles. A web3 mobile application is built on decentralized technologies, such as blockchain, smart contracts, and peer-to-peer networks, that enable more transparency, security, and autonomy for users and developers.

Create the Mobile app

Ionic React is a good hybrid solution for creating mobile applications and compatible with the Typescript version of the BeaconSDK. The behavior is equivalent to a classical web development, so for a web developer the ramp up is easy.

Beacon: the protocol of communication between the dapp and the wallet.

Note: As of today, it is not recommended to develop a native dApp in Flutter, React Native or native tools as it requires additional UI works (ex: missing wallet popup mechanism to confirm transactions).

  1. Install Ionic:

    npm install -g @ionic/cli
    ionic start app blank --type react
  2. Generate smart contract types from the taqueria plugin:

    This command generates Typescript classes from the smart contract interface definition that is used on the frontend.

    taq install @taqueria/plugin-contract-types
    taq generate types ./app/src
  3. Uninstall the conflicting old jest libraries/react-scripts and install the required Tezos web3 dependencies and Vite framework:

    cd app
    npm uninstall -S @testing-library/jest-dom @testing-library/react @testing-library/user-event @types/jest
    rm -rf src/components src/pages/Home.tsx src/pages/Home.css
    rm src/setupTests.ts src/App.test.tsx
    echo '/// <reference types="vite/client" />' > src/vite-env.d.ts

    npm install -S @taquito/taquito @taquito/beacon-wallet @airgap/beacon-sdk @tzkt/sdk-api
    npm install -S -D @airgap/beacon-types vite @vitejs/plugin-react-swc @types/react @types/node @types/react@18.2.42
  4. Polyfill issues fix:

    ⚠️ Polyfill issues fix: Add the following dependencies in order to avoid polyfill issues. The reason is that some dependencies are Node APIs and are not included in browsers.

    1. Install the missing libraries:

      npm i -D process buffer crypto-browserify stream-browserify assert stream-http https-browserify os-browserify url path-browserify
    2. Create a new file nodeSpecific.ts in the src folder of your project:

      touch src/nodeSpecific.ts
    3. Edit it to look like this:

      import { Buffer } from 'buffer';
      globalThis.Buffer = Buffer;
    4. Edit the vite.config.ts file:

      import react from '@vitejs/plugin-react-swc';
      import path from 'path';
      import { defineConfig } from 'vite';
      // https://vitejs.dev/config/
      export default ({ command }) => {
      const isBuild = command === 'build';

      return defineConfig({
      define: { 'process.env': process.env, global: {} },
      plugins: [react()],
      build: {
      commonjsOptions: {
      transformMixedEsModules: true,
      },
      },
      resolve: {
      alias: {
      // dedupe @airgap/beacon-sdk
      // I almost have no idea why it needs `cjs` on dev and `esm` on build, but this is how it works 🤷‍♂️
      '@airgap/beacon-sdk': path.resolve(
      path.resolve(),
      `./node_modules/@airgap/beacon-sdk/dist/${
      isBuild ? 'esm' : 'cjs'
      }/index.js`
      ),
      stream: 'stream-browserify',
      os: 'os-browserify/browser',
      util: 'util',
      process: 'process/browser',
      buffer: 'buffer',
      crypto: 'crypto-browserify',
      assert: 'assert',
      http: 'stream-http',
      https: 'https-browserify',
      url: 'url',
      path: 'path-browserify',
      },
      },
      });
      };
  5. Adapt Ionic for Vite:

    1. Edit index.html to fix the Node buffer issue with nodeSpecific.ts file and point to the CSS file:

      <!DOCTYPE html>
      <html lang="en">
      <head>
      <meta charset="utf-8" />
      <title>Ionic App</title>

      <base href="/" />

      <meta name="color-scheme" content="light dark" />
      <meta
      name="viewport"
      content="viewport-fit=cover, width=device-width, initial-scale=1.0, minimum-scale=1.0, maximum-scale=1.0, user-scalable=no"
      />
      <meta name="format-detection" content="telephone=no" />
      <meta name="msapplication-tap-highlight" content="no" />

      <link rel="manifest" href="/manifest.json" />
      <link href="assets/styles.css" rel="stylesheet" />

      <link rel="shortcut icon" type="image/png" href="/favicon.png" />

      <!-- add to homescreen for ios -->
      <meta name="apple-mobile-web-app-capable" content="yes" />
      <meta name="apple-mobile-web-app-title" content="Ionic App" />
      <meta name="apple-mobile-web-app-status-bar-style" content="black" />
      </head>
      <body>
      <div id="root"></div>
      <script type="module" src="/src/nodeSpecific.ts"></script>
      <script type="module" src="/src/main.tsx"></script>
      </body>
      </html>
    2. Edit src/main.tsx to force dark mode and remove React strict mode:

      import { createRoot } from 'react-dom/client';
      import App from './App';

      const container = document.getElementById('root');
      const root = createRoot(container!);

      // Add or remove the "dark" class based on if the media query matches
      document.body.classList.add('dark');

      root.render(<App />);
    3. Modify the default package.json default scripts to use Vite instead of the default React scripts:

        "scripts": {
      "dev": "jq -r '\"VITE_CONTRACT_ADDRESS=\" + last(.tasks[]).output[0].address' ../.taq/testing-state.json > .env && vite --host",
      "ionic:build": "tsc -v && tsc && vite build",
      "build": " tsc -v && tsc && vite build",
      "lint": "eslint src --ext ts,tsx --report-unused-disable-directives --max-warnings 0",
      "preview": "vite preview",
      "ionic:serve": "vite dev --host",
      "sync": "npm run build && ionic cap sync --no-build"
      },
  6. Edit the default application file src/App.tsx to configure page routing and add the style:

    import {
    IonApp,
    IonRouterOutlet,
    RefresherEventDetail,
    setupIonicReact,
    } from '@ionic/react';
    import { IonReactRouter } from '@ionic/react-router';
    import { Redirect, Route } from 'react-router-dom';

    /* Core CSS required for Ionic components to work properly */
    import '@ionic/react/css/core.css';

    /* Basic CSS for apps built with Ionic */
    import '@ionic/react/css/normalize.css';
    import '@ionic/react/css/structure.css';
    import '@ionic/react/css/typography.css';

    /* Optional CSS utils that can be commented out */
    import '@ionic/react/css/display.css';
    import '@ionic/react/css/flex-utils.css';
    import '@ionic/react/css/float-elements.css';
    import '@ionic/react/css/padding.css';
    import '@ionic/react/css/text-alignment.css';
    import '@ionic/react/css/text-transformation.css';

    /* Theme variables */
    import './theme/variables.css';

    import { NetworkType } from '@airgap/beacon-types';
    import { BeaconWallet } from '@taquito/beacon-wallet';
    import { InternalOperationResult } from '@taquito/rpc';
    import {
    PollingSubscribeProvider,
    Subscription,
    TezosToolkit,
    } from '@taquito/taquito';
    import React, { Dispatch, SetStateAction, useEffect, useState } from 'react';
    import { MainWalletType, Storage } from './main.types';
    import { HomeScreen } from './pages/HomeScreen';
    import { RulesScreen } from './pages/Rules';
    import { SessionScreen } from './pages/SessionScreen';
    import { TopPlayersScreen } from './pages/TopPlayersScreen';
    import {
    MMap,
    address,
    bytes,
    mutez,
    nat,
    timestamp,
    unit,
    } from './type-aliases';

    setupIonicReact();

    export class Action implements ActionCisor, ActionPaper, ActionStone {
    cisor?: unit;
    paper?: unit;
    stone?: unit;
    constructor(cisor?: unit, paper?: unit, stone?: unit) {
    this.cisor = cisor;
    this.paper = paper;
    this.stone = stone;
    }
    }
    export type ActionCisor = { cisor?: unit };
    export type ActionPaper = { paper?: unit };
    export type ActionStone = { stone?: unit };

    export type Session = {
    asleep: timestamp;
    board: MMap<nat, { Some: address } | null>;
    current_round: nat;
    decoded_rounds: MMap<
    nat,
    Array<{
    action: { cisor: unit } | { paper: unit } | { stone: unit };
    player: address;
    }>
    >;
    players: Array<address>;
    pool: mutez;
    result: { draw: unit } | { inplay: unit } | { winner: address };
    rounds: MMap<
    nat,
    Array<{
    action: bytes;
    player: address;
    }>
    >;
    total_rounds: nat;
    };

    export type UserContextType = {
    storage: Storage | null;
    setStorage: Dispatch<SetStateAction<Storage | null>>;
    userAddress: string;
    setUserAddress: Dispatch<SetStateAction<string>>;
    userBalance: number;
    setUserBalance: Dispatch<SetStateAction<number>>;
    Tezos: TezosToolkit;
    wallet: BeaconWallet;
    mainWalletType: MainWalletType | null;
    loading: boolean;
    setLoading: Dispatch<SetStateAction<boolean>>;
    refreshStorage: (
    event?: CustomEvent<RefresherEventDetail>
    ) => Promise<void>;
    subReveal: Subscription<InternalOperationResult> | undefined;
    subNewRound: Subscription<InternalOperationResult> | undefined;
    };
    export const UserContext = React.createContext<UserContextType | null>(null);

    const App: React.FC = () => {
    const Tezos = new TezosToolkit('https://ghostnet.tezos.marigold.dev');

    const wallet = new BeaconWallet({
    name: 'Training',
    preferredNetwork: NetworkType.GHOSTNET,
    });

    Tezos.setWalletProvider(wallet);
    Tezos.setStreamProvider(
    Tezos.getFactory(PollingSubscribeProvider)({
    shouldObservableSubscriptionRetry: true,
    pollingIntervalMilliseconds: 1500,
    })
    );

    const [userAddress, setUserAddress] = useState<string>('');
    const [userBalance, setUserBalance] = useState<number>(0);
    const [storage, setStorage] = useState<Storage | null>(null);
    const [mainWalletType, setMainWalletType] =
    useState<MainWalletType | null>(null);
    const [loading, setLoading] = useState<boolean>(false);

    const [subscriptionsDone, setSubscriptionsDone] = useState<boolean>(false);
    const [subReveal, setSubReveal] =
    useState<Subscription<InternalOperationResult>>();
    const [subNewRound, setSubNewRound] =
    useState<Subscription<InternalOperationResult>>();

    const refreshStorage = async (
    event?: CustomEvent<RefresherEventDetail>
    ): Promise<void> => {
    try {
    if (!userAddress) {
    const activeAccount = await wallet.client.getActiveAccount();
    let userAddress: string;
    if (activeAccount) {
    userAddress = activeAccount.address;
    setUserAddress(userAddress);
    const balance = await Tezos.tz.getBalance(userAddress);
    setUserBalance(balance.toNumber());
    }
    }

    console.log(
    'VITE_CONTRACT_ADDRESS:',
    import.meta.env.VITE_CONTRACT_ADDRESS
    );
    const mainWalletType: MainWalletType =
    await Tezos.wallet.at<MainWalletType>(
    import.meta.env.VITE_CONTRACT_ADDRESS
    );
    const storage: Storage = await mainWalletType.storage();
    setMainWalletType(mainWalletType);
    setStorage(storage);
    console.log('Storage refreshed');

    event?.detail.complete();
    } catch (error) {
    console.log('error refreshing storage', error);
    }
    };

    useEffect(() => {
    try {
    if (!subscriptionsDone) {
    const sub = Tezos.stream.subscribeEvent({
    tag: 'gameStatus',
    address: import.meta.env.VITE_CONTRACT_ADDRESS!,
    });

    sub.on('data', (e) => {
    console.log('on gameStatus event :', e);
    refreshStorage();
    });

    setSubReveal(
    Tezos.stream.subscribeEvent({
    tag: 'reveal',
    address: import.meta.env.VITE_CONTRACT_ADDRESS,
    })
    );

    setSubNewRound(
    Tezos.stream.subscribeEvent({
    tag: 'newRound',
    address: import.meta.env.VITE_CONTRACT_ADDRESS,
    })
    );
    } else {
    console.warn(
    'Tezos.stream.subscribeEvent already done ... ignoring'
    );
    }
    } catch (e) {
    console.log('Error with Smart contract event pooling', e);
    }

    console.log('Tezos.stream.subscribeEvent DONE');
    setSubscriptionsDone(true);
    }, []);

    useEffect(() => {
    if (userAddress) {
    console.warn('userAddress changed', wallet);
    (async () => await refreshStorage())();
    }
    }, [userAddress]);

    return (
    <IonApp>
    <UserContext.Provider
    value={{
    userAddress,
    userBalance,
    Tezos,
    wallet,
    storage,
    mainWalletType,
    setUserAddress,
    setUserBalance,
    setStorage,
    loading,
    setLoading,
    refreshStorage,
    subReveal,
    subNewRound,
    }}
    >
    <IonReactRouter>
    <IonRouterOutlet>
    <Route path={PAGES.HOME} component={HomeScreen} />
    <Route path={`${PAGES.SESSION}/:id`} component={SessionScreen} />
    <Route path={PAGES.TOPPLAYERS} component={TopPlayersScreen} />
    <Route path={PAGES.RULES} component={RulesScreen} />
    <Redirect exact from="/" to={PAGES.HOME} />
    </IonRouterOutlet>
    </IonReactRouter>
    </UserContext.Provider>
    </IonApp>
    );
    };

    export enum PAGES {
    HOME = '/home',
    SESSION = '/session',
    TOPPLAYERS = '/topplayers',
    RULES = '/rules',
    }

    export default App;

    Explanations:

    • import "@ionic...": Default standard Ionic imports.
    • import ... from "@airgap/beacon-types" ... from "@taquito/beacon-wallet" ... from "@taquito/taquito": Require libraries to interact with the Tezos node and the wallet.
    • export class Action implements ActionCisor, ActionPaper, ActionStone {...}: Representation of the Ligo variant Action in Typescript, which is needed when passing arguments on Play function.
    • export type Session = {...}: Taqueria exports the global storage type but not this sub-type from the storage type; it is needed for later, so extract a copy.
    • export const UserContext = React.createContext<UserContextType | null>(null): Global React context that is passed along pages. More info on React context here.
    • const refreshStorage = async (event?: CustomEvent<RefresherEventDetail>): Promise<void> => {...: A useful function to force the smart contract storage to refresh on React state changes (user balance, state of the game).
    • useEffect(() => { ... Tezos.setStreamProvider(...) ... Tezos.stream.subscribeEvent({...: During application initialization, it configures the wallet, the websocket listening to smart contract events.
    • <IonApp><UserContext.Provider ... ><IonReactRouter><IonRouterOutlet><Route path={PAGES.HOME} component={HomeScreen} /> ... : Injects the React context to all pages and declares the global routing of the application.
    • export enum PAGES { HOME = "/home", ...: Declaration of the global routes.
  7. Add the default theming (CSS, pictures, etc.) via copying the content of the git repository folder named assets folder to your local project (considering you cloned the repo and assets folder is on root folder).

    cp -r ../../assets/* .
  8. Create two React Button components to connect and disconnect the wallet and add code to fetch the user public hash key and balanceL

    1. Create the 2 missing component files in the app folder:

      touch src/ConnectWallet.tsx
      touch src/DisconnectWallet.tsx
    2. In the ConnectWallet.tsx file, create a button that creates an instance of the wallet, gets user permissions via a popup, and retrieves account information.

      import { NetworkType } from '@airgap/beacon-types';
      import { IonButton } from '@ionic/react';
      import { BeaconWallet } from '@taquito/beacon-wallet';
      import { TezosToolkit } from '@taquito/taquito';
      import { Dispatch, SetStateAction } from 'react';

      type ButtonProps = {
      Tezos: TezosToolkit;
      setUserAddress: Dispatch<SetStateAction<string>>;
      setUserBalance: Dispatch<SetStateAction<number>>;
      wallet: BeaconWallet;
      };

      const ConnectButton = ({
      Tezos,
      setUserAddress,
      setUserBalance,
      wallet,
      }: ButtonProps): JSX.Element => {
      const connectWallet = async (): Promise<void> => {
      try {
      console.log('before requestPermissions');

      await wallet.requestPermissions({
      network: {
      type: NetworkType.GHOSTNET,
      rpcUrl: 'https://ghostnet.tezos.marigold.dev',
      },
      });
      console.log('after requestPermissions');

      // gets user's address
      const userAddress = await wallet.getPKH();
      const balance = await Tezos.tz.getBalance(userAddress);
      setUserBalance(balance.toNumber());
      setUserAddress(userAddress);
      } catch (error) {
      console.log('error connectWallet', error);
      }
      };

      return (
      <IonButton expand="full" onClick={connectWallet}>
      Connect Wallet
      </IonButton>
      );
      };

      export default ConnectButton;
    3. In the DisconnectWallet.tsx file, create a button that cleans the wallet instance and all linked objects:

      import { IonFab, IonFabButton, IonIcon } from '@ionic/react';
      import { BeaconWallet } from '@taquito/beacon-wallet';
      import { power } from 'ionicons/icons';
      import { Dispatch, SetStateAction } from 'react';

      interface ButtonProps {
      wallet: BeaconWallet;
      setUserAddress: Dispatch<SetStateAction<string>>;
      setUserBalance: Dispatch<SetStateAction<number>>;
      }

      const DisconnectButton = ({
      wallet,
      setUserAddress,
      setUserBalance,
      }: ButtonProps): JSX.Element => {
      const disconnectWallet = async (): Promise<void> => {
      setUserAddress('');
      setUserBalance(0);
      console.log('disconnecting wallet');
      await wallet.clearActiveAccount();
      };

      return (
      <IonFab slot="fixed" vertical="top" horizontal="end">
      <IonFabButton>
      <IonIcon icon={power} onClick={disconnectWallet} />
      </IonFabButton>
      </IonFab>
      );
      };

      export default DisconnectButton;
    4. Save both files.

  9. Create the missing pages and the error utility class:

    touch src/pages/HomeScreen.tsx
    touch src/pages/SessionScreen.tsx
    touch src/pages/Rules.tsx
    touch src/pages/TopPlayersScreen.tsx
    touch src/TransactionInvalidBeaconError.ts

    The TransactionInvalidBeaconError.ts utility class is used to display human readable message from Beacon errors.

  10. Make these updates to the files:

    • HomeScreen.tsx: the home page where you can access all other pages.

      import {
      IonButton,
      IonButtons,
      IonContent,
      IonFooter,
      IonHeader,
      IonIcon,
      IonImg,
      IonInput,
      IonItem,
      IonLabel,
      IonList,
      IonModal,
      IonPage,
      IonRefresher,
      IonRefresherContent,
      IonSpinner,
      IonTitle,
      IonToolbar,
      useIonAlert,
      } from '@ionic/react';
      import { BigNumber } from 'bignumber.js';
      import { person } from 'ionicons/icons';
      import React, { useEffect, useRef, useState } from 'react';
      import { useHistory } from 'react-router-dom';
      import { PAGES, Session, UserContext, UserContextType } from '../App';
      import ConnectButton from '../ConnectWallet';
      import DisconnectButton from '../DisconnectWallet';
      import { TransactionInvalidBeaconError } from '../TransactionInvalidBeaconError';
      import Paper from '../assets/paper-logo.webp';
      import Scissor from '../assets/scissor-logo.webp';
      import Stone from '../assets/stone-logo.webp';
      import XTZLogo from '../assets/xtz.webp';
      import { SelectMembers } from '../components/TzCommunitySelectMembers';
      import { address, nat } from '../type-aliases';

      export const HomeScreen: React.FC = () => {
      const [presentAlert] = useIonAlert();
      const { push } = useHistory();

      const createGameModal = useRef<HTMLIonModalElement>(null);
      const selectGameModal = useRef<HTMLIonModalElement>(null);
      function dismissCreateGameModal() {
      console.log('dismissCreateGameModal');
      createGameModal.current?.dismiss();
      }
      function dismissSelectGameModal() {
      selectGameModal.current?.dismiss();
      const element = document.getElementById('home');
      setTimeout(() => {
      return element && element.remove();
      }, 1000); // Give a little time to properly unmount your previous page before removing the old one
      }

      const {
      Tezos,
      wallet,
      userAddress,
      userBalance,
      storage,
      mainWalletType,
      setStorage,
      setUserAddress,
      setUserBalance,
      setLoading,
      loading,
      refreshStorage,
      } = React.useContext(UserContext) as UserContextType;

      const [newPlayer, setNewPlayer] = useState<address>('' as address);
      const [total_rounds, setTotal_rounds] = useState<nat>(
      new BigNumber(1) as nat
      );
      const [myGames, setMyGames] = useState<Map<nat, Session>>();

      useEffect(() => {
      (async () => {
      if (storage) {
      const myGames = new Map(); //filtering our games
      Array.from(storage.sessions.keys()).forEach((key) => {
      const session = storage.sessions.get(key);

      if (
      session.players.indexOf(userAddress as address) >= 0 &&
      'inplay' in session.result
      ) {
      myGames.set(key, session);
      }
      });
      setMyGames(myGames);
      } else {
      console.log('storage is not ready yet');
      }
      })();
      }, [storage]);

      const createSession = async (
      e: React.MouseEvent<HTMLIonButtonElement, MouseEvent>
      ) => {
      console.log('createSession');
      e.preventDefault();

      try {
      setLoading(true);
      const op = await mainWalletType?.methods
      .createSession(total_rounds, [userAddress as address, newPlayer])
      .send();
      await op?.confirmation();
      const newStorage = await mainWalletType?.storage();
      setStorage(newStorage!);
      setLoading(false);
      dismissCreateGameModal();
      setTimeout(
      () => push(PAGES.SESSION + '/' + storage?.next_session.toString()),
      500
      );
      //it was the id created
      console.log('newStorage', newStorage);
      } catch (error) {
      console.table(`Error: ${JSON.stringify(error, null, 2)}`);
      const tibe: TransactionInvalidBeaconError =
      new TransactionInvalidBeaconError(error);
      presentAlert({
      header: 'Error',
      message: tibe.data_message,
      buttons: ['Close'],
      });
      setLoading(false);
      }
      setLoading(false);
      };

      return (
      <IonPage className="container">
      <IonHeader>
      <IonToolbar>
      <IonTitle>Shifumi</IonTitle>
      </IonToolbar>
      </IonHeader>
      <IonContent fullscreen>
      <IonRefresher slot="fixed" onIonRefresh={refreshStorage}>
      <IonRefresherContent></IonRefresherContent>
      </IonRefresher>

      {loading ? (
      <div className="loading">
      <IonItem>
      <IonLabel>Refreshing ...</IonLabel>
      <IonSpinner className="spinner"></IonSpinner>
      </IonItem>
      </div>
      ) : (
      <IonList inset={true}>
      {!userAddress ? (
      <>
      <div
      style={{
      display: 'flex',
      flexDirection: 'row',
      padding: '4em',
      justifyContent: 'space-around',
      }}
      >
      <IonImg src={Stone} className="logo" />
      <IonImg src={Paper} className="logo" />
      <IonImg src={Scissor} className="logo" />
      </div>
      <IonList inset={true}>
      <ConnectButton
      Tezos={Tezos}
      setUserAddress={setUserAddress}
      setUserBalance={setUserBalance}
      wallet={wallet}
      />
      </IonList>
      </>
      ) : (
      <IonList>
      <IonItem style={{ padding: 0, margin: 0 }}>
      <IonIcon icon={person} />
      <IonLabel
      style={{ fontSize: '0.8em', direction: 'rtl' }}
      >
      {userAddress}
      </IonLabel>
      </IonItem>
      <IonItem style={{ padding: 0, margin: 0 }}>
      <IonImg
      style={{ height: 24, width: 24 }}
      src={XTZLogo}
      />
      <IonLabel style={{ direction: 'rtl' }}>
      {userBalance / 1000000}
      </IonLabel>
      </IonItem>

      <div
      style={{
      display: 'flex',
      flexDirection: 'row',
      paddingTop: '10vh',
      paddingBottom: '10vh',
      justifyContent: 'space-around',
      width: '100%',
      }}
      >
      <IonImg src={Stone} className="logo" />
      <IonImg src={Paper} className="logo" />
      <IonImg src={Scissor} className="logo" />
      </div>

      <IonButton id="createGameModalVisible" expand="full">
      New game
      </IonButton>
      <IonModal
      ref={createGameModal}
      trigger="createGameModalVisible"
      >
      <IonHeader>
      <IonToolbar>
      <IonButtons slot="start">
      <IonButton
      onClick={() => dismissCreateGameModal()}
      >
      Cancel
      </IonButton>
      </IonButtons>
      <IonTitle>New Game</IonTitle>
      <IonButtons slot="end">
      <IonButton
      strong={true}
      onClick={(e) => createSession(e)}
      id="createGameModal"
      >
      Create
      </IonButton>
      </IonButtons>
      </IonToolbar>
      </IonHeader>
      <IonContent className="ion-padding">
      <h2>How many total rounds ?</h2>

      <IonItem key="total_rounds">
      <IonLabel
      position="stacked"
      className="text"
      ></IonLabel>
      <IonInput
      onIonChange={(str: any) => {
      if (str.detail.value === undefined) return;
      setTotal_rounds(
      new BigNumber(str.target.value) as nat
      );
      }}
      value={total_rounds.toString()}
      placeholder="total_rounds"
      type="number"
      label="Total Rounds"
      />
      </IonItem>

      <h2>Choose your opponent player</h2>

      <SelectMembers
      Tezos={Tezos}
      member={newPlayer}
      setMember={setNewPlayer}
      />

      <IonItem key="newPlayer">
      <IonInput
      onIonChange={(str: any) => {
      if (str.detail.value === undefined) return;
      setNewPlayer(str.detail.value as address);
      }}
      labelPlacement="floating"
      class="address"
      value={newPlayer}
      placeholder="...tz1"
      type="text"
      label="Tezos Address "
      />
      </IonItem>
      </IonContent>
      </IonModal>

      <IonButton id="selectGameModalVisible" expand="full">
      Join game
      </IonButton>
      <IonModal
      ref={selectGameModal}
      trigger="selectGameModalVisible"
      >
      <IonHeader>
      <IonToolbar>
      <IonButtons slot="start">
      <IonButton
      onClick={() => dismissSelectGameModal()}
      >
      Cancel
      </IonButton>
      </IonButtons>
      <IonTitle>Select Game</IonTitle>
      </IonToolbar>
      </IonHeader>
      <IonContent>
      <IonList inset={true}>
      {myGames
      ? Array.from(myGames.entries()).map(([key, _]) => (
      <IonButton
      key={'Game-' + key.toString()}
      expand="full"
      routerLink={
      PAGES.SESSION + '/' + key.toString()
      }
      onClick={dismissSelectGameModal}
      >
      {'Game n°' + key.toString()}
      </IonButton>
      ))
      : []}
      </IonList>
      </IonContent>
      </IonModal>

      <IonButton routerLink={PAGES.TOPPLAYERS} expand="full">
      Top Players
      </IonButton>
      </IonList>
      )}
      </IonList>
      )}
      </IonContent>
      <IonFooter>
      <IonToolbar>
      <IonTitle>
      <IonButton
      color="primary"
      routerLink={PAGES.RULES}
      expand="full"
      >
      Rules
      </IonButton>
      </IonTitle>
      </IonToolbar>
      </IonFooter>

      {userAddress ? (
      <DisconnectButton
      wallet={wallet}
      setUserAddress={setUserAddress}
      setUserBalance={setUserBalance}
      />
      ) : (
      <></>
      )}
      </IonPage>
      );
      };

      Explanation:

      • const createGameModal: The popup to create a new game.
      • const selectGameModal: The popup to select a game to join.
      • const [newPlayer, setNewPlayer] = useState<address>("" as address): Used on the New Game popup form to add an opponent.
      • const [total_rounds, setTotal_rounds] = useState<nat>(new BigNumber(1) as nat): Used on the New Game popup form to set number of round for one game.
      • const [myGames, setMyGames] = useState<Map<nat, Session>>(): Used on the Join Game popup window to display the games created or with invitation.
      • Array.from(storage.sessions.keys()).forEach((key) => { ... if (session.players.indexOf(userAddress as address) >= 0 && "inplay" in session.result ...: On storage change event, fetch and filter only games which the user can join and play (that is, with inplay status and where user appears on the player list).
      • const createSession = async (...) => { ... const op = await mainWalletType!.methods.createSession([userAddress as address, newPlayer], total_rounds).send(); ... : This function calls the smart contract entrypoint passing these arguments: current user address, opponent address, and total rounds. Then it redirects to the newly created game page.
      • {...<IonButton ... routerLink={PAGES.SESSION + "/" + key.toString()}: If you click on a game button from the list it redirects you to the game to play.
    • SessionScreen.tsx: This page lets you play on limited rounds and shows the result of the game.

      import { IonPage } from '@ionic/react';
      import React from 'react';

      export const SessionScreen: React.FC = () => {
      return <IonPage className="container"></IonPage>;
      };

      You will add more to this file later.

    • TopPlayersScreen.tsx: The player ranking page.

      import { IonPage } from '@ionic/react';
      import React from 'react';

      export const TopPlayersScreen: React.FC = () => {
      return <IonPage className="container"></IonPage>;
      };

      You will add more to this file later.

    • Rules.tsx: Just some information about game rules.

      import {
      IonButton,
      IonButtons,
      IonContent,
      IonHeader,
      IonImg,
      IonItem,
      IonList,
      IonPage,
      IonTitle,
      IonToolbar,
      } from '@ionic/react';
      import React from 'react';
      import { useHistory } from 'react-router-dom';
      import Clock from '../assets/clock.webp';
      import Legend from '../assets/legend.webp';
      import Paper from '../assets/paper-logo.webp';
      import Scissor from '../assets/scissor-logo.webp';
      import Stone from '../assets/stone-logo.webp';

      export const RulesScreen: React.FC = () => {
      const { goBack } = useHistory();

      /* 2. Get the param */
      return (
      <IonPage className="container">
      <IonHeader>
      <IonToolbar>
      <IonButtons slot="start">
      <IonButton onClick={goBack}>Back</IonButton>
      </IonButtons>
      <IonTitle>Rules</IonTitle>
      </IonToolbar>
      </IonHeader>
      <IonContent fullscreen>
      <div style={{ textAlign: 'left' }}>
      <IonList>
      <IonItem className="nopm">
      <IonImg src={Stone} className="logo" />
      Stone (Clenched Fist). Rock beats the scissors by hitting it
      </IonItem>
      <IonItem className="nopm">
      <IonImg src={Paper} className="logo" />
      Paper (open and extended hand) . Paper wins over stone by enveloping
      it
      </IonItem>
      <IonItem className="nopm">
      <IonImg src={Scissor} className="logo" />
      Scissors (closed hand with the two fingers) . Scissors wins paper
      cutting it
      </IonItem>

      <IonItem className="nopm">
      <IonImg src={Clock} className="logo" />
      If you are inactive for more than 10 minutes your opponent can
      claim the victory
      </IonItem>

      <IonItem className="nopm">
      <IonImg src={Legend} className="logo" />
      <ul style={{ listStyle: 'none' }}>
      <li className="win">Won round</li>
      <li className="lose">Lost round</li>
      <li className="draw">Draw</li>
      <li className="current">Current Round</li>
      <li className="missing">Missing Rounds</li>
      </ul>
      </IonItem>
      </IonList>
      </div>
      </IonContent>
      </IonPage>
      );
      };
    • TransactionInvalidBeaconError.ts: The utility class that formats Beacon errors.

      export class TransactionInvalidBeaconError {
      name: string;
      title: string;
      message: string;
      description: string;
      data_contract_handle: string;
      data_expected_form: string;
      data_message: string;

      /**
      *
      * @param transactionInvalidBeaconError {
      "name": "UnknownBeaconError",
      "title": "Aborted",
      "message": "[ABORTED_ERROR]:The action was aborted by the user.",
      "description": "The action was aborted by the user."
      }
      */

      constructor(transactionInvalidBeaconError: any) {
      this.name = transactionInvalidBeaconError.name;
      this.title = transactionInvalidBeaconError.title;
      this.message = transactionInvalidBeaconError.message;
      this.description = transactionInvalidBeaconError.description;
      this.data_contract_handle = '';
      this.data_expected_form = '';
      this.data_message = this.message;
      if (transactionInvalidBeaconError.data !== undefined) {
      let dataArray = Array.from<any>(
      new Map(
      Object.entries<any>(transactionInvalidBeaconError.data)
      ).values()
      );
      let contract_handle = dataArray.find(
      (obj) => obj.contract_handle !== undefined
      );
      this.data_contract_handle =
      contract_handle !== undefined
      ? contract_handle.contract_handle
      : '';
      let expected_form = dataArray.find(
      (obj) => obj.expected_form !== undefined
      );
      this.data_expected_form =
      expected_form !== undefined
      ? expected_form.expected_form +
      ':' +
      expected_form.wrong_expression.string
      : '';
      this.data_message =
      (this.data_contract_handle
      ? 'Error on contract: ' + this.data_contract_handle + ' '
      : '') +
      (this.data_expected_form
      ? 'error: ' + this.data_expected_form + ' '
      : '');
      }
      }
      }
  11. Test the application:

    To test in web mode, run this command:

    npm run dev

    Make sure that your wallet is has some tez on Ghostnet and click on the Connect button.

    Note: If you don't have tokens, to get some free XTZ on Ghostnet, follow this link to the faucet.

    On the popup, select your wallet, then your account and connect.

    You are logged.

    Optional: Click the Disconnect button to test the logout.

Summary

You have a mobile application where you can connect and disconnect a wallet, some default UI components and styles but not yet an interaction with your smart contract. The next step is to be able to create a game, join a game and play a session.

When you are ready, continue to Part 3: Create the game pages.