Skip to main content

Build a Fully-Onchain Image Gallery

info

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.

stage-1

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.

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
_10
pragma solidity ^0.8.28;
_10
_10
import "@openzeppelin/contracts/access/Ownable.sol";
_10
_10
contract 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 stringand 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:


_10
struct Image {
_10
string description;
_10
string base64EncodedImage;
_10
}
_10
_10
Image[] 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.


_10
function 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:


_10
function 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
}

warning

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.


_14
function getImages() public view returns (Image[] memory) {
_14
return images;
_14
}
_14
_14
function 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
_14
function 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
_49
pragma solidity ^0.8.28;
_49
_49
import "@openzeppelin/contracts/access/Ownable.sol";
_49
_49
contract 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:


_13
pragma solidity ^0.8.28;
_13
_13
import "@openzeppelin/contracts/access/Ownable.sol";
_13
import "./ImageGallery.sol";
_13
_13
contract 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
_23
pragma solidity ^0.8.28;
_23
_23
import "@openzeppelin/contracts/access/Ownable.sol";
_23
import "./ImageGallery.sol";
_23
_23
contract 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:

Factory on Flowscan

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.

createImageGallery

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.

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.

info

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.

danger

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.

addImage

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:


_10
npx create-next-app

We're using the default options.

Next, install rainbowkit, wagmi, and their related dependencies:


_10
npm 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
_113
import { connectorsForWallets } from '@rainbow-me/rainbowkit';
_113
import { Wallet, getWalletConnectConnector } from '@rainbow-me/rainbowkit';
_113
import { QueryClient, QueryClientProvider } from '@tanstack/react-query';
_113
import { createConfig, WagmiProvider } from 'wagmi';
_113
import { RainbowKitProvider } from '@rainbow-me/rainbowkit';
_113
import { flowTestnet } from 'viem/chains';
_113
import { http } from 'wagmi';
_113
_113
const projectId = '51407fcf066d74968d9a1a4c6da0d994'; // Replace with your actual project ID
_113
_113
export interface MyWalletOptions {
_113
projectId: string;
_113
}
_113
_113
const 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
_113
const connectors = connectorsForWallets(
_113
[
_113
{
_113
groupName: 'Recommended',
_113
wallets: [flowWallet],
_113
},
_113
],
_113
{
_113
appName: 'Onchain Image Gallery',
_113
projectId: projectId,
_113
},
_113
);
_113
_113
const wagmiConfig = createConfig({
_113
connectors,
_113
chains: [flowTestnet],
_113
ssr: true,
_113
transports: {
_113
[flowTestnet.id]: http(),
_113
},
_113
});
_113
_113
export 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:


_25
import '@rainbow-me/rainbowkit/styles.css';
_25
_25
import { ConnectButton } from '@rainbow-me/rainbowkit';
_25
_25
export 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.

tip

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:


_22
import { useMemo } from 'react';
_22
import { Abi } from 'viem';
_22
_22
import imageGalleryFactory from './ImageGalleryFactory#ImageGalleryFactory.json';
_22
import imageGallery from './ImageGallery#ImageGallery.json';
_22
import addresses from './deployed_addresses.json';
_22
_22
export 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
}

info

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 and awaitingResponse false
    • Listen for needing a reload and invalidating the query for galleryAddresses
    • Error handling
    • Receipt of gallery addresses
  • 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
_103
import { useEffect, useState } from 'react';
_103
import {
_103
useAccount,
_103
useReadContract,
_103
useWaitForTransactionReceipt,
_103
useWriteContract,
_103
} from 'wagmi';
_103
import useContracts from '../contracts/contracts';
_103
import { useQueryClient } from '@tanstack/react-query';
_103
_103
export 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.

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:


_42
import React, { useEffect, useState } from 'react';
_42
_42
type AddressDropdownProps = {
_42
addresses: string[]; // Array of EVM addresses
_42
handleSetActiveAddress: Function;
_42
};
_42
_42
const 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
_42
export 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:


_10
const [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.


_10
function 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.

warning

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:


_55
export type ImageGalleryImage = {
_55
description: string;
_55
base64EncodedImage: string;
_55
};
_55
_55
type ImageGalleryProps = {
_55
images: ImageGalleryImage[]; // Array of image objects
_55
};
_55
_55
const 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
_55
export 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:


_10
const [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


_10
const [images, setImages] = useState<ImageGalleryImage[]>([]);


_10
const { 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:


_10
useEffect(() => {
_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.


_10
useEffect(() => {
_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:


_28
return (
_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:


_64
import React, { useState } from 'react';
_64
_64
type ImageUploaderProps = {
_64
setUploadedBase64Image: (base64: string) => void; // Function to set the uploaded base64 image
_64
};
_64
_64
const 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
_64
export 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:


_10
const [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.


_10
function 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.