Build a Fully-Onchain Image Gallery
The FlowtoBooth tutorial series teaches you how to build a fun benchmark app and provides inspiration for the greater scope of possibilities building on Flow thanks to gas being so much less expensive.
It is not a production best-practice. While everything in these tutorials works, you'll run into the following problems at production scale:
- RPC Providers will likely rate-limit you for reading this much data at once
- NFT marketplaces may not display the images, likely due to the above
- 256*256 images are huge by blockchain data standards, but too small for modern devices
If you search for resources on how to store images of any significant size onchain, you'll be told it's either prohibitively expensive or even completely impossible. The reason for this is two-fold - first the size limit for data on transactions is about 40kb. Second, saving 40kb takes almost all of the 30 million gas limit on most blockchains.
The former constraint is immutable (though many chains are slowly increasing this limit), which limits the app to images about 256*256 pixels in size. The latter is heavily dependent on which chain you choose.
At current gas prices on most chains, using all 30 million gas in a block costs several dollars - or potentially thousands on ETH mainnet. At current prices on Flow, spending 30 million gas costs less than a penny, usually 1 or 2 tenths of a cent.
Much more computation is available at prices you or your users will be willing to pay for regular interactions. Including, but not limited to:
- Airdropping hundreds of NFTs with one transaction, for pennies
- Generation of large mazes
- Generating large amounts of random numbers (with free native VRF)
- Extensive string manipulation onchain
- Simple game AI logic
In this tutorial, we'll build a smart contract that can store and retrieve images onchain. We'll also build a simple frontend to interact with the contract on Flow and another chain.
Objectives
After completing this guide, you'll be able to:
- Construct a composable onchain image gallery that can be used permissionlessly by onchain apps and other contracts to store and retrieve images
- Build an onchain app that can interact with this contract to save and display images
- Compare the price of spending 30 million gas on Flow with the price on other chains
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.
Solidity
You don't need to be an expert, but you should be comfortable writing code in Solidity. You can use Hardhat, Foundry, or even Remix.
Build an Image Gallery Contract
Start a new smart contract project in the toolchain of your choice and install the OpenZeppelin contracts.
In your project, stub out a new contract for your image gallery that inherits from the Ownable contract:
_10// ImageGallery.sol_10_10// SPDX-License-Identifier: MIT_10pragma solidity ^0.8.28;_10_10import "@openzeppelin/contracts/access/Ownable.sol";_10_10contract ImageGallery is Ownable {_10 constructor(address _owner) Ownable(_owner) {}_10}
We're passing the original owner of the contract as an argument in the constructor to give greater flexibility for ownership when this contract is deployed.
Set Up Storage for Images
We'll store the images in a simple struct
that holds the image as a base64
encoded string
and also contains a string
for the description. Doing so allows the image to be directly used in html and makes it easier to test the contract directly with a block explorer, but has the downside of making the images 33% bigger. Another format would be more efficient.
These will be held in array:
_10struct Image {_10 string description;_10 string base64EncodedImage;_10}_10_10Image[] public images;
Construct Functions to Add and Delete Images
Next, add a function that accepts a _description
and _base64EncodedImage
and adds them to the array.
_10function addImage(_10 string memory _description,_10 string memory _base64EncodedImage_10) public onlyOwner {_10 images.push(Image(_description, _base64EncodedImage));_10}
Then, add one to delete the image at a given index:
_10function deleteImage(uint256 index) public onlyOwner {_10 if (index >= images.length) {_10 revert ImageIndexOutOfBounds(index, images.length);_10 }_10 for (uint256 i = index; i < images.length - 1; i++) {_10 images[i] = images[i + 1];_10 }_10 images.pop();_10}
If the array gets big enough that calling deleteImage
takes more than 30 million gas, it will brick this function. A safer and more gas-efficient method is to use a mapping
with a counter as the index, and handling for the case where an index is empty.
We're doing it this way to provide a way to delete accidentally uploaded images without making things too complex.
Retrieval Functions
Finally, add functions to get one image, get all of the images, and get the number of images in the collection.
_14function getImages() public view returns (Image[] memory) {_14 return images;_14}_14_14function getImage(uint256 index) public view returns (Image memory) {_14 if (index >= images.length) {_14 revert ImageIndexOutOfBounds(index, images.length);_14 }_14 return images[index];_14}_14_14function getImageCount() public view returns (uint256) {_14 return images.length;_14}
Final Contract
After completing the above, you'll end up with a contract similar to:
_49// SPDX-License-Identifier: UNLICENSED_49pragma solidity ^0.8.28;_49_49import "@openzeppelin/contracts/access/Ownable.sol";_49_49contract ImageGallery is Ownable {_49 struct Image {_49 string description;_49 string base64EncodedImage;_49 }_49_49 Image[] public images;_49_49 error ImageIndexOutOfBounds(uint256 index, uint256 length);_49_49 constructor(address _owner) Ownable(_owner) {}_49_49 function addImage(_49 string memory _description,_49 string memory _base64EncodedImage_49 ) public onlyOwner {_49 images.push(Image(_description, _base64EncodedImage));_49 }_49_49 function deleteImage(uint256 index) public onlyOwner {_49 if (index >= images.length) {_49 revert ImageIndexOutOfBounds(index, images.length);_49 }_49 for (uint256 i = index; i < images.length - 1; i++) {_49 images[i] = images[i + 1];_49 }_49 images.pop();_49 }_49_49 function getImages() public view returns (Image[] memory) {_49 return images;_49 }_49_49 function getImage(uint256 index) public view returns (Image memory) {_49 if (index >= images.length) {_49 revert ImageIndexOutOfBounds(index, images.length);_49 }_49 return images[index];_49 }_49_49 function getImageCount() public view returns (uint256) {_49 return images.length;_49 }_49}
Create a Factory
The image gallery contract you've just constructed is intended to be a utility for other contracts and apps to use freely. You don't want just one gallery for everyone, you need to give the ability for any app or contract to create and deploy private galleries freely.
Build a factory to deploy image galleries:
_13pragma solidity ^0.8.28;_13_13import "@openzeppelin/contracts/access/Ownable.sol";_13import "./ImageGallery.sol";_13_13contract ImageGalleryFactory {_13 event ImageGalleryCreated(address indexed owner, address gallery);_13_13 function createImageGallery(address _owner) public {_13 ImageGallery gallery = new ImageGallery(_owner);_13 emit ImageGalleryCreated(_owner, address(gallery));_13 }_13}
Tracking Factories
Some app designs may need multiple galleries for each user. For example, you might want to be able to give users the ability to collect images in separate galleries for separate topics, dates, or events, similar to how many photo apps work on smartphones.
To facilitate this feature, update your contract to keep track of which galleries have been created by which users. You'll end up with:
_23// SPDX-License-Identifier: UNLICENSED_23pragma solidity ^0.8.28;_23_23import "@openzeppelin/contracts/access/Ownable.sol";_23import "./ImageGallery.sol";_23_23contract ImageGalleryFactory {_23 event ImageGalleryCreated(address indexed owner, address gallery);_23_23 mapping(address => address[]) userToGalleries;_23_23 function createImageGallery(address _owner) public {_23 ImageGallery gallery = new ImageGallery(_owner);_23 emit ImageGalleryCreated(_owner, address(gallery));_23 userToGalleries[_owner].push(address(gallery));_23 }_23_23 function getGalleries(_23 address _owner_23 ) public view returns (address[] memory) {_23 return userToGalleries[_owner];_23 }_23}
Testing the Factory
Write appropriate unit tests, then deploy and verify the factory on Flow Testnet.
If you need help, check out:
Navigate to evm-testnet.flowscan.io, search for your contract, and navigate to the contracts
tab, then Read/Write contract
. You'll see something similar to:
Connect
your wallet. Use the Flow Wallet if you want automatically sponsored gas on both mainnet and testnet, or use the Flow Faucet to grab some testnet funds if you prefer to use another wallet.
Expand the createImageGallery
function, click the self
button, and then Write
the function.
Approve the transaction and wait for it to complete. Then, call getGalleries
for your address to find the address of the gallery you've created.
Testing the Image Gallery
Search for the address of your image gallery contract. It won't
be verified, but if you're using our exact contract, you will see a message from Flowscan that a verified contract with the same bytecode was found in the Blockscout DB. Click the provided link to complete the verification process.
The easiest way to get an ABI for the image gallery is to deploy one. You can do that now if you like.
If you're following along, but used your own contract, simply deploy and verify one copy of the contract directly, refresh the page, then complete the above.
You could test addImage
with a random string, but it's better to use a base64-encoded image. Search for and navigate to one of the many online tools that will base64 encode images.
Most sites of this nature are free tools created by helpful programmers and are funded with ads, donations, or the generosity of the creator. But you never know who made them or what they're caching.
Never upload or convert sensitive data on a free site.
Use the tool to convert an image that is ~30kb or smaller. Copy the string and paste it into the field in addImage
. You can also add a description
, but the bytes used will count towards the ~40kb limit.
Click Write
and approve the transaction. Take note of the cost! You've saved an image onchain forever for just a little bit of gas!
Once the transaction goes through, call getImage
with 0
as the index to retrieve your description and base64-encoded image.
Paste your image string as the src
for an img
tag in an html snippet to confirm it worked.
_10<div>_10 <img_10 src=""_10 />_10</div>
Building the Frontend
Now that your contracts are sorted and working, it's time to build an app to interact with it. We'll use Next.js for this, but the components we provide will be adaptable to other React frameworks.
Run:
_10npx create-next-app
We're using the default options.
Next, install rainbowkit, wagmi, and their related dependencies:
_10npm install @rainbow-me/rainbowkit wagmi viem@2.x @tanstack/react-query
Provider Setup
Add a file called providers
inside the app
folder. In it, add your config and providers for wagmi and rainbowkit. You'll need to add the Flow Wallet as a custom wallet. It's not included by default because it has special features that aren't compatible with other blockchains.
_113'use client';_113_113import { connectorsForWallets } from '@rainbow-me/rainbowkit';_113import { Wallet, getWalletConnectConnector } from '@rainbow-me/rainbowkit';_113import { QueryClient, QueryClientProvider } from '@tanstack/react-query';_113import { createConfig, WagmiProvider } from 'wagmi';_113import { RainbowKitProvider } from '@rainbow-me/rainbowkit';_113import { flowTestnet } from 'viem/chains';_113import { http } from 'wagmi';_113_113const projectId = '51407fcf066d74968d9a1a4c6da0d994'; // Replace with your actual project ID_113_113export interface MyWalletOptions {_113 projectId: string;_113}_113_113const flowWallet = ({ projectId }: MyWalletOptions): Wallet => ({_113 id: 'flow-wallet',_113 name: 'Flow Wallet',_113 iconUrl: 'https://lilico.app/logo_mobile.png',_113 iconBackground: '#41CC5D',_113 downloadUrls: {_113 android:_113 'https://play.google.com/store/apps/details?id=com.flowfoundation.wallet',_113 ios: 'https://apps.apple.com/ca/app/flow-wallet-nfts-and-crypto/id6478996750',_113 chrome:_113 'https://chromewebstore.google.com/detail/flow-wallet/hpclkefagolihohboafpheddmmgdffjm',_113 qrCode: 'https://link.lilico.app',_113 },_113 mobile: {_113 getUri: (uri: string) => uri,_113 },_113 qrCode: {_113 getUri: (uri: string) => uri,_113 instructions: {_113 learnMoreUrl: 'https://wallet.flow.com',_113 steps: [_113 {_113 description:_113 'We recommend putting Flow Wallet on your home screen for faster access to your wallet.',_113 step: 'install',_113 title: 'Open the Flow Wallet app',_113 },_113 {_113 description:_113 'You can find the scan button on home page, a connection prompt will appear for you to connect your wallet.',_113 step: 'scan',_113 title: 'Tap the scan button',_113 },_113 ],_113 },_113 },_113 extension: {_113 instructions: {_113 learnMoreUrl: 'https://wallet.flow.com',_113 steps: [_113 {_113 description:_113 'We recommend pinning Flow Wallet to your taskbar for quicker access to your wallet.',_113 step: 'install',_113 title: 'Install the Flow Wallet extension',_113 },_113 {_113 description:_113 'Be sure to back up your wallet using a secure method. Never share your secret phrase with anyone.',_113 step: 'create',_113 title: 'Create or Import a Wallet',_113 },_113 {_113 description:_113 'Once you set up your wallet, click below to refresh the browser and load up the extension.',_113 step: 'refresh',_113 title: 'Refresh your browser',_113 },_113 ],_113 },_113 },_113 createConnector: getWalletConnectConnector({ projectId }),_113});_113_113const connectors = connectorsForWallets(_113 [_113 {_113 groupName: 'Recommended',_113 wallets: [flowWallet],_113 },_113 ],_113 {_113 appName: 'Onchain Image Gallery',_113 projectId: projectId,_113 },_113);_113_113const wagmiConfig = createConfig({_113 connectors,_113 chains: [flowTestnet],_113 ssr: true,_113 transports: {_113 [flowTestnet.id]: http(),_113 },_113});_113_113export default function Providers({ children }: { children: React.ReactNode }) {_113 const queryClient = new QueryClient();_113_113 return (_113 <WagmiProvider config={wagmiConfig}>_113 <QueryClientProvider client={queryClient}>_113 <RainbowKitProvider>{children}</RainbowKitProvider>_113 </QueryClientProvider>_113 </WagmiProvider>_113 );_113}
Add the Connect Button
Open page.tsx
and clear out the default content. Replace it with a message about what your app does and add the rainbowkit Connect
button. Don't forget to import rainbowkit's css file and the ConnectButton
component:
_25import '@rainbow-me/rainbowkit/styles.css';_25_25import { ConnectButton } from '@rainbow-me/rainbowkit';_25_25export default function Home() {_25 return (_25 <div className="grid grid-rows-[20px_1fr_20px] items-center justify-items-center min-h-screen p-8 pb-20 gap-16 sm:p-20 font-[family-name:var(--font-geist-sans)]">_25 <main className="flex flex-col gap-8 row-start-2 items-center sm:items-start">_25 <h1 className="text-4xl font-bold">Image Gallery</h1>_25 <p className="text-lg text-center sm:text-left">_25 A decentralized image gallery built on Flow blockchain. All images_25 saved directly onchain._25 </p>_25 <p className="text-lg text-center sm:text-left">_25 A fun benchmark, not best practice for production!_25 </p>_25 <p className="text-lg text-center sm:text-left">_25 Free with gas sponsored by Flow with the Flow wallet. Sub-cent to save_25 an image with other wallets._25 </p>_25 <ConnectButton />_25 </main>_25 </div>_25 );_25}
Test the app and make sure you can connect your wallet.
Import Your Contracts
Next, you'll need to get your contract ABI and address into your frontend. If you're using Hardhat, you can use the artifacts produced by the Ignition deployment process. If you're using Foundry or Remix, you can adapt this process to the format of artifacts produced by those toolchains.
If you didn't deploy the Image Gallery contract, do so now to generate an artifact containing the ABI.
Add a folder in app
called contracts
. Copy the following files from your smart contract project, located in the ignition
and ignition/deployments/chain-545
folders:
deployed_addresses.json
ImageGallery#ImageGallery.json
ImageGalleryFactory#ImageGalleryFactory.json
Additionally, add a file called contracts.ts
. In it, create a hook to provide the ABI and addresses of your contracts conveniently:
_22import { useMemo } from 'react';_22import { Abi } from 'viem';_22_22import imageGalleryFactory from './ImageGalleryFactory#ImageGalleryFactory.json';_22import imageGallery from './ImageGallery#ImageGallery.json';_22import addresses from './deployed_addresses.json';_22_22export default function useContracts() {_22 return useMemo(() => {_22 return {_22 imageGalleryFactory: {_22 address: addresses[_22 'ImageGalleryFactory#ImageGalleryFactory'_22 ] as `0x${string}`,_22 abi: imageGalleryFactory.abi as Abi,_22 },_22 imageGallery: {_22 abi: imageGallery.abi as Abi,_22 },_22 };_22 }, []);_22}
Note that we're not including an address
for the imageGallery
itself. We'll need to set this dynamically as users might have more than one gallery.
Add Content
You can use a few strategies to organize the components that interact with the blockchain. One is to create a centralized component that stores all of the state related to smart contracts and uses a single instance of useWriteContract
. Doing so makes it easier to convey the transaction lifecycle to your users, at the cost of re-fetching all the data from your RPC provider after every transaction. This becomes sub-optimal if your app interacts with many contracts, or even different read functions within the same contract.
Add a folder in app
called components
, and create a file called Content.tsx
. In it, add the following:
- Imports for React, wagmi, your contracts, and Tanstack
- State variables for:
- When a reload is needed
- When you are waiting on a transaction response
- The list of gallery addresses for the connected wallet
- Hooks for:
useAccount()
useQueryClient()
useContracts()
useWriteContract()
useWaitForTransactionReceipt()
useEffects
to:- Listen for a receipt and set
reload
to true andawaitingResponse
false - Listen for needing a reload and invalidating the query for galleryAddresses
- Error handling
- Receipt of gallery addresses
- Listen for a receipt and set
- A
useReadContract
to fetch the list of gallery addresses for this user - Frontend code to display the button to create a gallery if the user is signed in
You'll end up with something similar to:
_103'use client';_103_103import { useEffect, useState } from 'react';_103import {_103 useAccount,_103 useReadContract,_103 useWaitForTransactionReceipt,_103 useWriteContract,_103} from 'wagmi';_103import useContracts from '../contracts/contracts';_103import { useQueryClient } from '@tanstack/react-query';_103_103export default function Content() {_103 const [reload, setReload] = useState(false);_103 const [awaitingResponse, setAwaitingResponse] = useState(false);_103 const [galleryAddresses, setGalleryAddresses] = useState<string[]>([]);_103_103 const account = useAccount();_103 const queryClient = useQueryClient();_103 const { imageGalleryFactory } = useContracts();_103_103 const { data, writeContract, error: writeError } = useWriteContract();_103_103 const { data: receipt, error: receiptError } = useWaitForTransactionReceipt({_103 hash: data,_103 });_103_103 useEffect(() => {_103 if (receipt) {_103 setReload(true);_103 setAwaitingResponse(false);_103 }_103 }, [receipt]);_103_103 useEffect(() => {_103 if (reload) {_103 setReload(false);_103 queryClient.invalidateQueries({ queryKey: galleryAddressesQueryKey });_103 }_103 }, [reload]);_103_103 useEffect(() => {_103 if (writeError) {_103 console.error(writeError);_103 setAwaitingResponse(false);_103 }_103 }, [writeError]);_103_103 useEffect(() => {_103 if (receiptError) {_103 console.error(receiptError);_103 setAwaitingResponse(false);_103 }_103 }, [receiptError]);_103_103 const { data: galleryAddressesData, queryKey: galleryAddressesQueryKey } =_103 useReadContract({_103 abi: imageGalleryFactory.abi,_103 address: imageGalleryFactory.address,_103 functionName: 'getGalleries',_103 args: [account.address],_103 });_103_103 useEffect(() => {_103 if (galleryAddressesData) {_103 const newAddresses = galleryAddressesData as string[];_103 newAddresses.reverse();_103 setGalleryAddresses(newAddresses);_103 }_103 }, [galleryAddressesData]);_103_103 function handleCreateGallery() {_103 setAwaitingResponse(true);_103 writeContract({_103 abi: imageGalleryFactory.abi,_103 address: imageGalleryFactory.address,_103 functionName: 'createImageGallery',_103 args: [account.address],_103 });_103 }_103_103 return (_103 <div className="card gap-1">_103 {account.isConnected && (_103 <div>_103 <div className="mb-4">_103 <button_103 onClick={handleCreateGallery}_103 disabled={awaitingResponse}_103 className={`px-4 py-2 rounded-lg text-white ${_103 !awaitingResponse_103 ? 'bg-blue-500 hover:bg-blue-600'_103 : 'bg-gray-300 cursor-not-allowed'_103 }`}_103 >_103 {awaitingResponse ? 'Processing...' : 'Create Gallery'}_103 </button>_103 </div>_103 </div>_103 )}_103 </div>_103 );_103}
Don't forget to add your <Content />
component to page.tsx
, below the <ConnectButton />
component.
Test the app and make sure you can complete the transaction to create a gallery.
Gallery List
Next, you'll need to display the list of a user's galleries and enable them to select which one they want to interact with. A dropdown list will serve this function well. Add a component called AddressList.tsx
, and in it add:
_42import React, { useEffect, useState } from 'react';_42_42type AddressDropdownProps = {_42 addresses: string[]; // Array of EVM addresses_42 handleSetActiveAddress: Function;_42};_42_42const AddressDropdown: React.FC<AddressDropdownProps> = ({_42 addresses,_42 handleSetActiveAddress,_42}) => {_42 const [selectedAddress, setSelectedAddress] = useState('');_42_42 useEffect(() => {_42 if (selectedAddress) {_42 console.log(selectedAddress);_42 handleSetActiveAddress(selectedAddress);_42 }_42 }, [selectedAddress]);_42_42 return (_42 <div className="container mx-auto px-4">_42 <h1 className="text-2xl font-bold text-center mb-6">Select a Gallery</h1>_42 <div className="flex flex-col items-center space-y-4">_42 <select_42 value={selectedAddress}_42 onChange={(e) => setSelectedAddress(e.target.value)}_42 className="w-full max-w-md border border-gray-300 rounded-lg p-2 bg-white shadow-md focus:outline-none focus:ring-2 focus:ring-blue-500"_42 >_42 <option value="">Select an address</option>_42 {addresses.map((address, index) => (_42 <option key={index} value={address}>_42 {address}_42 </option>_42 ))}_42 </select>_42 </div>_42 </div>_42 );_42};_42_42export default AddressDropdown;
This component doesn't interact directly with the blockchain. It accepts the array of addresses
and a function to handle setting the activeAddress
.
To use it in Content.tsx
, you'll need to add a new state variable for the activeAddress
:
_10const [activeAddress, setActiveAddress] = useState<string | null>(null);
You'll also need a handler for when the activeAddress
is set. You can't just use the setActiveAddress()
function because you need to tell the app to reload if the user changes which gallery is active, so that the images in that gallery are loaded.
_10function handleSetActiveAddress(address: string) {_10 setReload(true);_10 setActiveAddress(address);_10}
Finally, add the new component under the <button>
:
_10<AddressList_10 addresses={galleryAddresses}_10 handleSetActiveAddress={handleSetActiveAddress}_10/>
Test again, and confirm that the address of the gallery you created is in the dropdown and is selectable. The provided code contains a console log as well, to make it easier to copy the address in case you need to check it on Flowscan.
Display the Images
Next, you need to pull the images for the selected gallery from the contract.
Make sure you're using the same gallery you added an image too earlier. Otherwise, there won't be an image to pull and display!
Create a component called ImageGallery
. All this needs to do is accept a list of images and descriptions and display them. You can style this nicely if you'd like, or use the basic implementation here:
_55export type ImageGalleryImage = {_55 description: string;_55 base64EncodedImage: string;_55};_55_55type ImageGalleryProps = {_55 images: ImageGalleryImage[]; // Array of image objects_55};_55_55const ImageGallery: React.FC<ImageGalleryProps> = ({ images }) => {_55 if (images.length === 0) {_55 return (_55 <div className="container mx-auto px-4">_55 <p className="text-center text-xl font-bold">No images to display</p>_55 </div>_55 );_55 }_55_55 return (_55 <div className="container mx-auto px-4">_55 <div className="grid grid-cols-2 md:grid-cols-3 lg:grid-cols-4 gap-4">_55 {images.map((image, index) => {_55 const isValidBase64Image =_55 typeof image.base64EncodedImage === 'string' &&_55 image.base64EncodedImage.startsWith('data:image/') &&_55 image.base64EncodedImage.includes('base64,');_55_55 return (_55 <div_55 key={index}_55 className="border border-gray-200 rounded-lg overflow-hidden shadow-md"_55 >_55 {isValidBase64Image ? (_55 <img_55 src={image.base64EncodedImage}_55 alt={image.description || `Image ${index + 1}`}_55 className="w-full h-auto object-cover"_55 />_55 ) : (_55 <div className="p-4 text-center text-red-500">_55 Invalid image data_55 </div>_55 )}_55 <div className="p-2 bg-gray-100 text-center text-sm text-gray-700">_55 {image.description || 'No description available'}_55 </div>_55 </div>_55 );_55 })}_55 </div>_55 </div>_55 );_55};_55_55export default ImageGallery;
Implementing the gallery display will take more additions to Content.tsx
. You'll need to:
- Add a state variable for the list of images
- Implement a second
useContractRead
hook to pull the images from the currently selected gallery address - Hook the gallery into the refresh logic
First, add the state variable to store the gallery array:
_10const [images, setImages] = useState<ImageGalleryImage[]>([]);
Next, add a useReadContract
to read from the gallery. Use the activeAddress
for the address
property. Don't forget to destructure imageGallery
from useContracts
_10const [images, setImages] = useState<ImageGalleryImage[]>([]);
_10const { data: galleryData, queryKey: galleryQueryKey } = useReadContract({_10 abi: imageGallery.abi,_10 address: activeAddress as `0x${string}`,_10 functionName: 'getImages',_10});
Hook the new query key into the refresh system:
_10useEffect(() => {_10 if (reload) {_10 setReload(false);_10 queryClient.invalidateQueries({ queryKey: galleryAddressesQueryKey });_10 // Added to existing `useEffect`_10 queryClient.invalidateQueries({ queryKey: galleryQueryKey });_10 }_10}, [reload]);
Then, add a useEffect
to update the images
in state when galleryData
is received. Users expect the newest images to be shown first, so reverse
the array before setting it to state.
_10useEffect(() => {_10 if (galleryData) {_10 const newImages = galleryData as ImageGalleryImage[];_10 // reverse the array so the latest images are shown first_10 newImages.reverse();_10 setImages(newImages);_10 }_10}, [galleryData]);
Finally, implement the gallery itself in the return
:
_28return (_28 <div className="card gap-1">_28 {account.isConnected && (_28 <div>_28 <div className="mb-4">_28 <button_28 onClick={handleCreateGallery}_28 disabled={awaitingResponse}_28 className={`px-4 py-2 rounded-lg text-white ${_28 !awaitingResponse_28 ? 'bg-blue-500 hover:bg-blue-600'_28 : 'bg-gray-300 cursor-not-allowed'_28 }`}_28 >_28 {awaitingResponse ? 'Processing...' : 'Create Gallery'}_28 </button>_28 <AddressList_28 addresses={galleryAddresses}_28 handleSetActiveAddress={handleSetActiveAddress}_28 />_28 </div>_28 <div className="mb-4">_28 <ImageGallery images={images} />_28 </div>_28 </div>_28 )}_28 </div>_28);
Run the app, log in with your wallet that has the gallery you created for testing and select the gallery.
You're now displaying an image that is stored onchain forever!
Image Uploader
The last thing to do for this initial implementation is to add functionality so that users can upload their own images through the app and save them onchain without needing to do the base64 conversion on their own.
For now, we'll just generate an error if the file is too big, but later on we can do that for the user as well.
Add the ImageUploader
component. This needs to handle uploading the image and displaying any errors. We'll keep the state for the image itself in Content
so that it's accessible to other components:
_64import React, { useState } from 'react';_64_64type ImageUploaderProps = {_64 setUploadedBase64Image: (base64: string) => void; // Function to set the uploaded base64 image_64};_64_64const ImageUploader: React.FC<ImageUploaderProps> = ({_64 setUploadedBase64Image,_64}) => {_64 const [error, setError] = useState<string | null>(null);_64_64 const handleImageUpload = (event: React.ChangeEvent<HTMLInputElement>) => {_64 const file = event.target.files?.[0];_64_64 if (!file) {_64 setError('No file selected');_64 return;_64 }_64_64 if (!file.type.startsWith('image/')) {_64 setError('Only image files are allowed');_64 return;_64 }_64_64 if (file.size > 30 * 1024) {_64 setError('Image size must be 30KB or smaller');_64 return;_64 }_64_64 const reader = new FileReader();_64 reader.onload = () => {_64 const base64 = reader.result as string;_64 setUploadedBase64Image(base64);_64 setError(null);_64 };_64 reader.onerror = () => {_64 setError('Failed to read file');_64 };_64 reader.readAsDataURL(file);_64 };_64_64 return (_64 <div className="container mx-auto px-4">_64 <div className="flex flex-col items-center space-y-4">_64 <label_64 htmlFor="image-upload"_64 className="cursor-pointer bg-blue-500 text-white px-4 py-2 rounded-lg shadow-md hover:bg-blue-600"_64 >_64 Upload Image_64 </label>_64 <input_64 id="image-upload"_64 type="file"_64 accept="image/*"_64 onChange={handleImageUpload}_64 className="hidden"_64 />_64 {error && <p className="text-red-500 text-sm">{error}</p>}_64 </div>_64 </div>_64 );_64};_64_64export default ImageUploader;
As before, we'll need to make some updates to Content.tsx
to complete the implementation.
First, add a state variable for the image:
_10const [uploadedBase64Image, setUploadedBase64Image] = useState<string>('');
Then add the ImageUploader
to the return
:
_10<ImageUploader setUploadedBase64Image={setUploadedBase64Image} />
Later on, you'll probably want to make a component for displaying the uploaded image, but for now just add it below the uploader button component:
_11{_11 uploadedBase64Image && (_11 <div className="mt-6 text-center">_11 <img_11 src={uploadedBase64Image}_11 alt="Uploaded"_11 className="max-w-xs mx-auto rounded-lg shadow-md"_11 />_11 </div>_11 );_11}
Finally, you need to add a button and a handler to call the smart contract function to save the image onchain.
_10function handleSaveOnchain() {_10 // console.log(uploadedBase64Image);_10 setAwaitingResponse(true);_10 writeContract({_10 abi: imageGallery.abi,_10 address: activeAddress as `0x${string}`,_10 functionName: 'addImage',_10 args: ['', uploadedBase64Image],_10 });_10}
Add the button inside the check for an uploadedBase64Image
so that it only displays when there is an image to upload:
_22{_22 uploadedBase64Image && (_22 <div className="mt-6 text-center">_22 <img_22 src={uploadedBase64Image}_22 alt="Uploaded"_22 className="max-w-xs mx-auto rounded-lg shadow-md"_22 />_22 <button_22 onClick={handleSaveOnchain}_22 disabled={awaitingResponse}_22 className={`px-4 py-2 rounded-lg text-white ${_22 !awaitingResponse_22 ? 'bg-blue-500 hover:bg-blue-600'_22 : 'bg-gray-300 cursor-not-allowed'_22 }`}_22 >_22 {awaitingResponse ? 'Loading...' : 'Save Onchain'}_22 </button>_22 </div>_22 );_22}
Test the app to save your new image, and make sure the error displays if you try to upload an image that is too large.
Conclusion
In this tutorial, you built a fully functional onchain image gallery using Flow EVM. You created smart contracts that can store images directly on the blockchain and a modern React frontend that allows users to interact with these contracts. The implementation demonstrates how Flow's efficient gas pricing makes operations that would be prohibitively expensive on other chains not just possible, but practical.
Now that you have completed the tutorial, you should be able to:
- Construct a composable onchain image gallery that can be used permissionlessly by onchain apps and other contracts to store and retrieve images
- Build an onchain app that can interact with this contract to save and display images
- Compare the price of spending 30 million gas on Flow with the price on other chains
Now that you've completed this tutorial, you're ready to explore more complex onchain storage patterns and build applications that take advantage of Flow's unique capabilities for storing and processing larger amounts of data than traditionally possible on other chains.