Creating a Frontend
Now that we have successfully created a contract, it's time to build a frontend to provide a user-friendly interface for interacting with it. Up until now, we have been using the CLI to send transactions and view the contract's state. However, frontends offer a more intuitive way for end users to interact with the contract. They can display all the relevant information in one place, allow users to make calls with a simple button click, and only require a wallet as a prerequisite.
Frontend structureβ
Navigate to the auction frontend.
cd frotends/01-frontend
Here we have a simple Next.js frontend that we'll walk through to understand the basics of creating a frontend for a NEAR smart contract.
For starters, let's take a look at how the code in the frontend is structured by doing a quick overview of the important files.
File | Description |
---|---|
_app.js | Responsible for rending the page, initiates the wallet object and adds it to global context |
index.js | The main page where the project's components are loaded into and contains most of the logic for the application like viewing the state of the contract and logic for placing a bid |
near.js | Contains the wallet class that has methods to interact with the wallet and blockchain |
context.js | Holds the global context - the wallet object and the signed-in account ID - that can be accessed anywhere |
config.js | Specifies the account ID of the auction contract |
Navigation.jsx | A component that contains a button to sign users in and out of wallets |
Bid.jsx | A component allowing a user to make a bid |
LastBid.jsx | A component that displays the highest bid and when the highest bid will next refresh |
Timer.jsx | A component that shows how long till the auction is over, or, if over, displays a button to claim the auction and then states the auction is over |
Specifying the contractβ
We have a config file that specifies the contract name of the auction that the frontend will interact with. There has been an example auction contract deployed and specified already but feel free to change the contract to your own auction contract you deployed.
- config.js
Loading...
Setting up walletsβ
To be able to fully interact with the contract - send bids and claim the auction - you'll need a wallet
to sign transactions. Wallets securely store your private keys and allow you to sign transactions without exposing your private key to the frontend. The wallet selector allows users to choose between a selection of wallets.
We abstract the wallet selector in our near.js
file by exposing methods to complete various tasks. Feel free to explore the file to understand fully how the wallet selector is implemented.
The wallet object is initiated in the app.js
file and its added to the global context along with the account that is signed in to make it easier to access anywhere in the application.
- _app.js
- context.js
Loading...
Loading...
On NEAR, in additional to normal full access keys, we have function-call access keys
that are given to applications to allow them to sign non-payable
transactions on behalf of the user. This is so the wallet doesn't have to pop up for each non-critical transaction, improving the user experience. When creating the wallet object you can decide to create an access key for the application to use. However, in this example we opt out since the main function - bid
- we'll be calling is payable
.
You can read further about NEAR's key model here.
We add a sign-in and sign-out button in the navigation
component to call the respective methods in the near.js
file.
- Navigation.jsx
- near.js
Loading...
Loading...
Displaying the highest bidβ
To get the highest bid from the auction and who made it we call get_highest_bid
. Since this function returns the highest bid in yoctoNEAR
we divide by 10^24
to get the amount in NEAR.
- index.js
- near.js
Loading...
Loading...
In the wallet file, you'll see that we make a query to the RPC provider, since we are not signing a transaction the wallet isn't required here. Here we are using https://rpc.testnet.near.org but note there are many different providers available. We are querying the RPC with optimistic finality, which queries the latest block recorded on the node. Alternatively, one could use final finality where the block has been validated by at least 66% of the validators on the network but this will provide slightly delayed information (only by a couple of seconds).
We then pass the information about the highest bidder into the LastBid
component to display the bid amount and the bidder's account Id.
- index.js
- LastBid.jsx
Loading...
Loading...
Updating the highest bidβ
We want to know the highest bid at all times, someone else could have placed a higher bid since the page was loaded. To solve this we fetch the contract information every 20 seconds using setInterval
and update the highest bid if it has changed. In reality you would want to refresh the bid amount more requently but for the sake of saving on RPC calls we are doing it every 20 seconds.
- index.js
Loading...
Auction end timeβ
The contract stores the end time of the auction in the number of nanoseconds since the Unix epoch (1 January 1970 00:00:00 UTC). In our frontend we will display the time left in days, hours, minutes, and seconds.
- Timer.jsx
Loading...
Making a bidβ
To make a bid we make a call to the contract using the bid
function. We specify the deposit amount in yoctoNEAR
which will be the bid amount. The input box will take the bid amount in NEAR so we multiply by 10^24
to get the correct amount to send. We also specify the amount of gas to attach to the transaction, here we are attaching 30Tgas which is more than enough for the transaction to go through, we are refunded any unused gas anyway.
Here, since the user is changing the state of the contract, not just viewing it, the user needs to sign the transaction. Thus the wallet will pop up displaying the transaction details.
- index.js
- near.js
Loading...
Loading...
Claiming the auctionβ
Once the auction is over (the current time is greater than the end time) the auction can be claimed. At this point, the timer will be hidden and a button to claim the auction will be displayed. Once clicked the claim
function will be called on the auction contract to send the highest bidder the NFT and the auctioneer the FTs.
- index.js
Loading...
Conclusionβ
In this part of the tutorial, we have implemented a simple frontend for a NEAR contract. Along the way, you have learned how to use the wallet selector to sign the user in and out, how to view the contractβs state, how to sign and send transactions.
While we can see the highest bid, we may want to see the auction's bidding history. Since the contract only stores the most recent bid (to reduce storage costs), we need to use an indexer to pull historical data. In the next part of the tutorial, we'll look at querying historical data using an API endpoint.