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 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:
_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.
_115'use client';_115_115import { connectorsForWallets } from '@rainbow-me/rainbowkit';_115import { Wallet, getWalletConnectConnector } from '@rainbow-me/rainbowkit';_115import { QueryClient, QueryClientProvider } from '@tanstack/react-query';_115import { createConfig, WagmiProvider } from 'wagmi';_115import { RainbowKitProvider } from '@rainbow-me/rainbowkit';_115import { flowTestnet } from 'viem/chains';_115import { http } from 'wagmi';_115_115const projectId = '51407fcf066d74968d9a1a4c6da0d994'; // Replace with your actual project ID_115_115export interface MyWalletOptions {_115  projectId: string;_115}_115_115const flowWallet = ({ projectId }: MyWalletOptions): Wallet => ({_115  id: 'flow-wallet',_115  name: 'Flow Wallet',_115  rdns: 'com.flowfoundation.wallet',_115  iconUrl: 'https://lilico.app/logo_mobile.png',_115  iconBackground: '#41CC5D',_115  downloadUrls: {_115    android:_115      'https://play.google.com/store/apps/details?id=com.flowfoundation.wallet',_115    ios: 'https://apps.apple.com/ca/app/flow-wallet-nfts-and-crypto/id6478996750',_115    chrome:_115      'https://chromewebstore.google.com/detail/flow-wallet/hpclkefagolihohboafpheddmmgdffjm',_115    qrCode: 'https://link.lilico.app',_115  },_115  mobile: {_115    getUri: (uri: string) =>_115      `https://fcw-link.lilico.app/wc?uri=${encodeURIComponent(uri)}`,_115  },_115  qrCode: {_115    getUri: (uri: string) => uri,_115    instructions: {_115      learnMoreUrl: 'https://wallet.flow.com',_115      steps: [_115        {_115          description:_115            'We recommend putting Flow Wallet on your home screen for faster access to your wallet.',_115          step: 'install',_115          title: 'Open the Flow Wallet app',_115        },_115        {_115          description:_115            'You can find the scan button on home page, a connection prompt will appear for you to connect your wallet.',_115          step: 'scan',_115          title: 'Tap the scan button',_115        },_115      ],_115    },_115  },_115  extension: {_115    instructions: {_115      learnMoreUrl: 'https://wallet.flow.com',_115      steps: [_115        {_115          description:_115            'We recommend pinning Flow Wallet to your taskbar for quicker access to your wallet.',_115          step: 'install',_115          title: 'Install the Flow Wallet extension',_115        },_115        {_115          description:_115            'Be sure to back up your wallet using a secure method. Never share your secret phrase with anyone.',_115          step: 'create',_115          title: 'Create or Import a Wallet',_115        },_115        {_115          description:_115            'Once you set up your wallet, click below to refresh the browser and load up the extension.',_115          step: 'refresh',_115          title: 'Refresh your browser',_115        },_115      ],_115    },_115  },_115  createConnector: getWalletConnectConnector({ projectId }),_115});_115_115const connectors = connectorsForWallets(_115  [_115    {_115      groupName: 'Recommended',_115      wallets: [flowWallet],_115    },_115  ],_115  {_115    appName: 'Onchain Image Gallery',_115    projectId: projectId,_115  },_115);_115_115const wagmiConfig = createConfig({_115  connectors,_115  chains: [flowTestnet],_115  ssr: true,_115  transports: {_115    [flowTestnet.id]: http(),_115  },_115});_115_115export default function Providers({ children }: { children: React.ReactNode }) {_115  const queryClient = new QueryClient();_115_115  return (_115    <WagmiProvider config={wagmiConfig}>_115      <QueryClientProvider client={queryClient}>_115        <RainbowKitProvider>{children}</RainbowKitProvider>_115      </QueryClientProvider>_115    </WagmiProvider>_115  );_115}
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()
 
- useEffectsto:- Listen for a receipt and set reloadto true andawaitingResponsefalse
- Listen for needing a reload and invalidating the query for galleryAddresses
- Error handling
- Receipt of gallery addresses
 
- Listen for a receipt and set 
- A useReadContractto 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 useContractReadhook 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.