User Interface
This app consists of multiple components coming together to create this single screen that we will later powerup with our contract
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>
Chat Footer
- 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.