1/5
_Follow along the course with this video._ --- ### SVG NFT Encoding Before we even begin, I want to say you _can_ pass the SVG itself to a constructor and encode it on-chain (and I'll show you how this works a bit later), but if we encode the SVG with base64 _first_ and pass this to our constructor, it'll save us a step. Here's the encoded SVGs again for those without base64 installed. ```bash HappySVG:  SadSVG:  ``` > ❗ **NOTE** > Those who have decided to use their own custom SVG images, remember you can acquire the encoding with the command `base64 -i <filename>` while in the `img` directory! Now, if we're going to be passing _already encoded_ imageURIs to our constructor, it's probably a good idea to adjust the naming of our storage variables for clarity. Let's do this before moving on. ```js contract MoodNft is ERC721 { uint256 private s_tokenCounter; string private s_sadSvgImageUri; string private s_happySvgImageUri; constructor(string memory sadSvgImageUri, string memory happySvgImageUri) ERC721("Mood NFT", "MN"){ s_sadSvgImageUri = sadSvgImageUri; s_happySvgImageUri = happySvgImageUri } } ``` > ❗ **IMPORTANT** >**tokenURI != imageURI** > > It's important to remember that imageURI is one property of a token's tokenURI. A tokenURI is usually a JSON object! At this point you may be asking, if the tokenURI is a JSON object, how do we store this on-chain? The answer: We can encode it in much the same way! OpenZeppelin actually offers a [**Utilities**](https://docs.openzeppelin.com/contracts/4.x/utilities) package which includes a Base64 function which we can leverage to encode our tokenURI on-chain. We've already got OpenZeppelin contracts installed, so we can just import Base64 into our NFT contract. ```js import { Base64 } from "@openzeppelin/contracts/utils/Base64.sol"; ``` Let's start off our tokenURI function by defining a variable, `string memory tokenMetadata`. We can set this equal to our JSON object in string format like so: ```js string memory tokenMetadata = abi.encodePacked( '{"name: "', name(), '", description: "An NFT that reflects your mood!", "attributes": [{"trait_type": "Mood", "value": 100}], "image": ', imageURI, '"}' ) ``` In the above, we're using `abi.encodePacked` to concatenate our disparate strings into one object. This allows us to easily parameterize things a little bit. In order to determine our imageURI we'll need to derive this from the mood which has been set to our NFT token. As such, we're going to need a way to track the mood of each token. This sounds like a mapping to me. We can even spice it up a little bit and map our choice to an `enum`, which would allow someone to set more moods, if they wanted to expand on things in the future. ```js contract MoodNft is ERC721 { uint256 private s_tokenCounter; string private s_sadSvgImageUri; string private s_happySvgImageUri; enum Mood { HAPPY, SAD } mapping(uint256 => Mood) private s_tokenIdToMood; } ``` When an NFT is minted, they'll need a default mood, let's default them to happy. ```js function mintNft() public { _safeMint(msg.sender, s_tokenCounter); s_tokenIdToMood[s_tokenCounter] = Mood.HAPPY; s_tokenCounter++; } ``` Now, back in our tokenURI function, we can define a conditional statement which will derive what our imageURI should be. ```js function tokenURI(uint256 tokenId) public view override returns (string memory){ string memory imageURI; if (s_tokenIfToMood[tokenId] == HAPPY) { imageURI = s_happySvgImageUri; } else { imageURI = s_sadSvgImageUri; } string memory tokenMetadata = abi.encodePacked( '{"name: "', name(), '", description: "An NFT that reflects your mood!", "attributes": [{"trait_type": "Mood", "value": 100}], "image": ', imageURI, '"}' ) } ``` Alright, this looks good, but we're not done yet. We'll add a way to flip our NFTs mood soon. For now, we just have our metadata as a string in our contract, we need to convert this to the hashed syntax that our browser understands. This is where things might get a little wild. Currently we have a string, in order to acquire the Base64 hash of this data, we need to first convert this string to bytes, we can do this with some typecasting. ```js bytes( abi.encodePacked( '{"name: "', name(), '", description: "An NFT that reflects your mood!", "attributes": [{"trait_type": "Mood", "value": 100}], "image": ', imageURI, '"}' ) ); ``` Now we can apply our Base64 encoding to acquire our hash. ```js Base64.encode( bytes( abi.encodePacked( '{"name: "', name(), '", description: "An NFT that reflects your mood!", "attributes": [{"trait_type": "Mood", "value": 100}], "image": ', imageURI, '"}' ) ) ); ``` At this point, our tokenURI data is formatting like our imageUris were. If you recall, we needed to prepend our data type prefix(`data:image/svg+xml;base64,`) to our Base64 hash in order for a browser to understand. A similar methodology applies to our tokenURI JSON but with a different prefix. Let's define a method to return this string for us. Fortunately the ERC721 standard has a \_baseURI function that we can override. ```js function _baseURI() internal pure override returns(string memory){ return "data:application/json;base64," } ``` Now, in our tokenURI function again, we can concatenate the result of this \_baseURI function with the Base64 encoding of our JSON object... and finally we can type cast all of this as a string to be returned by our tokenURI function. ```js return string( abi.encodePacked( _baseURI(), Base64.encode( bytes( abi.encodePacked( '{"name: "', name(), '", description: "An NFT that reflects your mood!", "attributes": [{"trait_type": "Mood", "value": 100}], "image": ', imageURI, '"}' ) ) ) ) ); ``` Admittedly, this is a lot at once. Before we add any more functionality, let's consider writing a test to make sure things are working as intended. To summarize what's happening: 1. We created a string out of our JSON object, concatenated with abi.encodePacked. 2. typecast this string as a bytes object 3. encoded the bytes object with Base64 4. concatenated the encoding with our \_baseURI prefix 5. typecast the final value as a string to be returned as our tokenURI ### Testing tokenURI Given the complexity of our tokenURI function, let's take a moment to write a quick test and assure it's returning what we'd expect it to. Create the file `test/MoodNftTest.t.sol` and set up our usual boilerplate. ```js // SPDX-License-Identifier: MIT pragma solidity ^0.8.18; import {Test} from "forge-std/Test.sol"; import {MoodNft} from "../src/MoodNft.sol"; contract MoodNftTest is Test { MoodNft moodNft; function setUp() public { } } ``` We'll need to declare our Happy and Sad SVG URIs as constants in our test, we can use these in the deployment of our MoodNft contract within the setUp function of our Test. ```js // SPDX-License-Identifier: MIT pragma solidity ^0.8.18; import {Test} from "forge-std/Test.sol"; import {MoodNft} from "../src/MoodNft.sol"; contract MoodNftTest is Test { MoodNft moodNft; string public constant HAPPY_SVG_URI = ""; string public constant SAD_SVG_URI = ""; function setUp() public { moodNft = new MoodNft(SAD_SVG_URI, HAPPY_SVG_URI); } } ``` Finally we can write a test function. All that's required is to mint one of our MoodNft tokens, and then we can console out the tokenURI of that tokenId(0)! We'll need to create a user to do this. > ❗ **PROTIP** > Don't forget to import `console`! > > ``` > import {Test, console} from "forge-std/Test.sol";` > ``` ```js // SPDX-License-Identifier: MIT pragma solidity ^0.8.18; import {Test, console} from "forge-std/Test.sol"; import {MoodNft} from "../src/MoodNft.sol"; contract MoodNftTest is Test { ... address USER = makeAddr("USER"); function setUp() public { moodNft = new MoodNft(SAD_SVG_URI, HAPPY_SVG_URI); } function testViewTokenURI() public { vm.prank(USER); moodNft.mintNft(); console.log(moodNft.tokenURI(0)); } } ``` Now let's run it (make sure you're back in your root directory)! ```bash forge test --mt testViewTokenURI -vvvv ``` ```bash Logs: data:application/json;base64,eyJuYW1lIjogIkJpUG9sYXIiLCAiZGVzY3JpcHRpb24iOiAiQW4gTkZUIHRoYXQgcmVmbGVjdHMgeW91ciBtb29kISIsICJhdHRyaWJ1dGVzIjogW3sidHJhaXRfdHlwZSI6ICJNb29kIiwgInZhbHVlIjogMTAwfV0sICJpbWFnZSI6ICJkYXRhOmltYWdlL3N2Zyt4bWw7YmFzZTY0LFBITjJaeUIyYVdWM1FtOTRQU0l3SURBZ01qQXdJREl3TUNJZ2QybGtkR2c5SWpRd01DSWdJR2hsYVdkb2REMGlOREF3SWlCNGJXeHVjejBpYUhSMGNEb3ZMM2QzZHk1M015NXZjbWN2TWpBd01DOXpkbWNpUGdvZ0lEeGphWEpqYkdVZ1kzZzlJakV3TUNJZ1kzazlJakV3TUNJZ1ptbHNiRDBpZVdWc2JHOTNJaUJ5UFNJM09DSWdjM1J5YjJ0bFBTSmliR0ZqYXlJZ2MzUnliMnRsTFhkcFpIUm9QU0l6SWk4K0NpQWdQR2NnWTJ4aGMzTTlJbVY1WlhNaVBnb2dJQ0FnUEdOcGNtTnNaU0JqZUQwaU5qRWlJR041UFNJNE1pSWdjajBpTVRJaUx6NEtJQ0FnSUR4amFYSmpiR1VnWTNnOUlqRXlOeUlnWTNrOUlqZ3lJaUJ5UFNJeE1pSXZQZ29nSUR3dlp6NEtJQ0E4Y0dGMGFDQmtQU0p0TVRNMkxqZ3hJREV4Tmk0MU0yTXVOamtnTWpZdU1UY3ROalF1TVRFZ05ESXRPREV1TlRJdExqY3pJaUJ6ZEhsc1pUMGlabWxzYkRwdWIyNWxPeUJ6ZEhKdmEyVTZJR0pzWVdOck95QnpkSEp2YTJVdGQybGtkR2c2SURNN0lpOCtDand2YzNablBnPT0ifQ== ``` This looks pretty good! If we paste this into our browser we should see... ::image{src='/foundry-nfts/13-svg-nft-encoding/svg-nft-encoding2.png' style='width: 100%; height: auto;'} ... That looks like a JSON to me! Now, let's copy that imageURI into our browser... ::image{src='/foundry-nfts/13-svg-nft-encoding/svg-nft-encoding3.png' style='width: 100%; height: auto;'} Close enough! ### Wrap Up Amazing! We've written most of our MoodNFT contract and we've gotta ahead of the game with our tests, verifying that our tokenURI function is infact returning a correctly formatting tokenURI which has been derived from our NFT's current mood setting. In the next lesson we'll set up the functionality necessary to flip our NFT's mood!
Teaches encoding SVGs in Base64 format for on-chain storage in NFTs. It covers the process of encoding and testing SVG NFTs, ensuring their proper functioning and appearance
Previous lesson
Previous
Next lesson
Next
Give us feedback
Course Overview
About the course
Advanced smart contract development
How to develop a stablecoin
How to develop a DeFi protocol
How to develop a DAO
Advanced smart contracts testing
Fuzz testing
Manual verification
Web3 Developer Relations
$85,000 - $125,000 (avg. salary)
Web3 developer
$60,000 - $150,000 (avg. salary)
Smart Contract Engineer
$100,000 - $150,000 (avg. salary)
Smart Contract Auditor
$100,000 - $200,000 (avg. salary)
Security researcher
$49,999 - $120,000 (avg. salary)
Guest lecturers:
Juliette Chevalier
Lead Developer relations at Aragon
Nader Dabit
Director of developer relations at Avara
Ally Haire
Developer relations at Protocol Labs
Harrison
Founder at GasliteGG
Last updated on November 29, 2024
Solidity Developer
Advanced FoundryDuration: 36min
Duration: 3h 06min
Duration: 5h 02min
Duration: 2h 47min
Duration: 1h 23min
Duration: 4h 28min
Duration: 1h 19min
Duration: 58min
Course Overview
About the course
Advanced smart contract development
How to develop a stablecoin
How to develop a DeFi protocol
How to develop a DAO
Advanced smart contracts testing
Fuzz testing
Manual verification
Web3 Developer Relations
$85,000 - $125,000 (avg. salary)
Web3 developer
$60,000 - $150,000 (avg. salary)
Smart Contract Engineer
$100,000 - $150,000 (avg. salary)
Smart Contract Auditor
$100,000 - $200,000 (avg. salary)
Security researcher
$49,999 - $120,000 (avg. salary)
Guest lecturers:
Juliette Chevalier
Lead Developer relations at Aragon
Nader Dabit
Director of developer relations at Avara
Ally Haire
Developer relations at Protocol Labs
Harrison
Founder at GasliteGG
Last updated on November 29, 2024
Testimonials
Read what our students have to say about this course.
Chainlink
Chainlink
Gustavo Gonzalez
Solutions Engineer at OpenZeppelin
Francesco Andreoli
Lead Devrel at Metamask
Albert Hu
DeForm Founding Engineer
Radek
Senior Developer Advocate at Ceramic
Boidushya
WalletConnect
Idris
Developer Relations Engineer at Axelar