Skip to main content

User Interface

This app consists of multiple components coming together to create this single screen that we will later powerup with our contract

Demo image

Components

For the design of this application we're keeping it simple by creating components using Tailwind for styling and the DaisyUI tailwind plugin to provide us with components.

An overview of the above screen can be found in this snippet:

<template>
<main class="min-h-[100vh] flex flex-col items-center gap-4 justify-center">
<div class="flex gap-2 justify-center">
<ChainSelect />
<ConnectWallet />
</div>

<div class="relative w-80 md:w-1/2 max-w-lg">
<ChatHeader />

<div class="bg-black px-4 pb-4 max-h-[400px] min-h-[200px] overflow-y-auto chat-container rounded-b-lg">
<div class="chat-block">
<div class="chat chat-end">
<div class="chat-header mb-1">Last msg sent via contract</div>
<ChatBlock :key="lastMessage" username="Anon" :msg="lastMessage" />
</div>
</div>

<ChatFooter />
</div>
</div>
</main>
</template>

Individual Components

Taking the above image and snippet into consideration we can go further into explaining how the components work:

Chain Select

  • This is a simple select input that allows us choose the network we want to work with.
<template>
<select class="select select-primary" v-model="selected">
<option
v-for="chain in supportedChains"
:key="chain.name"
:value="chain.name"
>
{{ chain.name }}
</option>
</select>
</template>

Connect Wallet

  • This is the button that allows us connect our wallet to our app and use it to interact with our contract.
  • It also has the ability to disconnect our wallet from our app.
<template>
<div v-if="!address">
<button class="btn btn-accent" @click="handleConnect">
Connect Wallet
</button>
<p class="text-[0.6rem] text-center mt-1">Freighter only</p>
</div>

<button v-else class="bg-primary p-4 rounded-2xl text-black" @click="handleDisconnect">
Account: <span class="font-bold">{{ getShortAddress(address) }}</span>
</button>
</template>

Chat Header

  • This exists to indicate the name we want to call our app and also point us to an external link to see all interactions the Greeting contract has had since its creation.
<template>
<div class="bg-primary absolute top-0 left-0 p-5 w-full text-center text-black rounded-t-lg">
<p>
Greeter -
<NuxtLink external :to="explorerLink" target="_blank" class="underline underline-offset-4">
Explore Contract
</NuxtLink>
</p>
</div>

<div class="h-[60px]" />
</template>

Chat Block

  • This is used displaty the last message that was sent by calling the contracts read_title function.
<template>
<ChatAvatar :username="username" />
<div class="chat-bubble">{{ msg }}</div>
</template>
  • This contains the input box we type our intended message and a send button to pass on that message by calling the contract's set_title method.
<template>
<div class="bg-black absolute bottom-0 left-0 px-4 w-full rounded-b-lg">
<div class="flex items-center gap-1 my-4">
<div class="h-[40px]">
<ChatAvatar :username="getShortAddress(address)" />
</div>

<input type="text" placeholder="Type message here..." class="input w-full rounded-3xl" v-model="message" />

<button class="btn btn-primary rounded-3xl" :disabled="!address || isSending" @click="handleSend">
{{ !isSending ? "Send" : "Sending..." }}
</button>
</div>
</div>
</template>

ChatAvatar

  • A simple component to have an avatar image, purely for aesthetics.
<template>
<div class="chat-image avatar">
<div class="w-10 rounded-full">
<img alt="Tailwind CSS chat bubble component" width="200px" height="200px"
:src="`https://robohash.org/${Boolean(props.username) ? props.username : defaultName}`" class="bg-neutral" />
</div>
</div>
</template>

This is how our UI is built.

Next Step

The next step is to this app to interact with the blockchain using the contract.

  • We will be using this package; @stellar/stellar-sdk.

  • And some Custom implementations based off packages from soroban-react to tie our contract with the above interface.