Skip to main content

ERC-721 Onchain NFTs

https://github.com/PatrickAlphaC/all-on-chain-generated-nft

https://www.oniichain.com/

// SPDX-License-Identifier: MIT
pragma solidity ^0.8.0;

import "@openzeppelin/contracts/access/Ownable.sol";
import "@openzeppelin/contracts/token/ERC721/extensions/ERC721URIStorage.sol";
import "base64-sol/base64.sol";
import "@chainlink/contracts/src/v0.8/VRFConsumerBase.sol";

contract RandomSVG is ERC721URIStorage, VRFConsumerBase, Ownable {
uint256 public tokenCounter;

event CreatedRandomSVG(uint256 indexed tokenId, string tokenURI);
event CreatedUnfinishedRandomSVG(uint256 indexed tokenId, uint256 randomNumber);
event requestedRandomSVG(bytes32 indexed requestId, uint256 indexed tokenId);
mapping(bytes32 => address) public requestIdToSender;
mapping(uint256 => uint256) public tokenIdToRandomNumber;
mapping(bytes32 => uint256) public requestIdToTokenId;
bytes32 internal keyHash;
uint256 internal fee;
uint256 public maxNumberOfPaths;
uint256 public maxNumberOfPathCommands;
uint256 public size;
string[] public pathCommands;
string[] public colors;



constructor(address _VRFCoordinator, address _LinkToken, bytes32 _keyhash, uint256 _fee)
VRFConsumerBase(_VRFCoordinator, _LinkToken)
ERC721("RandomSVG", "rsNFT")
{
tokenCounter = 0;
keyHash = _keyhash;
fee = _fee;
maxNumberOfPaths = 10;
maxNumberOfPathCommands = 5;
size = 500;
pathCommands = ["M", "L"];
colors = ["red", "blue", "green", "yellow", "black", "white"];
}

function withdraw() public payable onlyOwner {
payable(owner()).transfer(address(this).balance);
}

function create() public returns (bytes32 requestId) {
requestId = requestRandomness(keyHash, fee);
requestIdToSender[requestId] = msg.sender;
uint256 tokenId = tokenCounter;
requestIdToTokenId[requestId] = tokenId;
tokenCounter = tokenCounter + 1;
emit requestedRandomSVG(requestId, tokenId);
}

function finishMint(uint256 tokenId) public {
require(bytes(tokenURI(tokenId)).length <= 0, "tokenURI is already set!");
require(tokenCounter > tokenId, "TokenId has not been minted yet!");
require(tokenIdToRandomNumber[tokenId] > 0, "Need to wait for the Chainlink node to respond!");
uint256 randomNumber = tokenIdToRandomNumber[tokenId];
string memory svg = generateSVG(randomNumber);
string memory imageURI = svgToImageURI(svg);
_setTokenURI(tokenId, formatTokenURI(imageURI));
emit CreatedRandomSVG(tokenId, svg);
}

function fulfillRandomness(bytes32 requestId, uint256 randomNumber) internal override {
address nftOwner = requestIdToSender[requestId];
uint256 tokenId = requestIdToTokenId[requestId];
_safeMint(nftOwner, tokenId);
tokenIdToRandomNumber[tokenId] = randomNumber;
emit CreatedUnfinishedRandomSVG(tokenId, randomNumber);
}

function generateSVG(uint256 _randomness) public view returns (string memory finalSvg) {
// We will only use the path element, with stroke and d elements
uint256 numberOfPaths = (_randomness % maxNumberOfPaths) + 1;
finalSvg = string(abi.encodePacked("<svg xmlns='http://www.w3.org/2000/svg' height='", uint2str(size), "' width='", uint2str(size), "'>"));
for(uint i = 0; i < numberOfPaths; i++) {
// we get a new random number for each path
string memory pathSvg = generatePath(uint256(keccak256(abi.encode(_randomness, i))));
finalSvg = string(abi.encodePacked(finalSvg, pathSvg));
}
finalSvg = string(abi.encodePacked(finalSvg, "</svg>"));
}

function generatePath(uint256 _randomness) public view returns(string memory pathSvg) {
uint256 numberOfPathCommands = (_randomness % maxNumberOfPathCommands) + 1;
pathSvg = "<path d='";
for(uint i = 0; i < numberOfPathCommands; i++) {
string memory pathCommand = generatePathCommand(uint256(keccak256(abi.encode(_randomness, size + i))));
pathSvg = string(abi.encodePacked(pathSvg, pathCommand));
}
string memory color = colors[_randomness % colors.length];
pathSvg = string(abi.encodePacked(pathSvg, "' fill='transparent' stroke='", color,"'/>"));
}

function generatePathCommand(uint256 _randomness) public view returns(string memory pathCommand) {
pathCommand = pathCommands[_randomness % pathCommands.length];
uint256 parameterOne = uint256(keccak256(abi.encode(_randomness, size * 2))) % size;
uint256 parameterTwo = uint256(keccak256(abi.encode(_randomness, size * 2 + 1))) % size;
pathCommand = string(abi.encodePacked(pathCommand, " ", uint2str(parameterOne), " ", uint2str(parameterTwo)));
}

// From: https://stackoverflow.com/a/65707309/11969592
function uint2str(uint _i) internal pure returns (string memory _uintAsString) {
if (_i == 0) {
return "0";
}
uint j = _i;
uint len;
while (j != 0) {
len++;
j /= 10;
}
bytes memory bstr = new bytes(len);
uint k = len;
while (_i != 0) {
k = k-1;
uint8 temp = (48 + uint8(_i - _i / 10 * 10));
bytes1 b1 = bytes1(temp);
bstr[k] = b1;
_i /= 10;
}
return string(bstr);
}




// You could also just upload the raw SVG and have solildity convert it!
function svgToImageURI(string memory svg) public pure returns (string memory) {
// example:
// <svg width='500' height='500' viewBox='0 0 285 350' fill='none' xmlns='http://www.w3.org/2000/svg'><path fill='black' d='M150,0,L75,200,L225,200,Z'></path></svg>
// 
string memory baseURL = "data:image/svg+xml;base64,";
string memory svgBase64Encoded = Base64.encode(bytes(string(abi.encodePacked(svg))));
return string(abi.encodePacked(baseURL,svgBase64Encoded));
}

function formatTokenURI(string memory imageURI) public pure returns (string memory) {
return string(
abi.encodePacked(
"data:application/json;base64,",
Base64.encode(
bytes(
abi.encodePacked(
'{"name":"',
"SVG NFT", // You can add whatever name here
'", "description":"An NFT based on SVG!", "attributes":"", "image":"',imageURI,'"}'
)
)
)
)
);
}

// remove later:
function bytes32ToString(bytes32 _bytes32) public pure returns (string memory) {
uint8 i = 0;
while(i < 32 && _bytes32[i] != 0) {
i++;
}
bytes memory bytesArray = new bytes(i);
for (i = 0; i < 32 && _bytes32[i] != 0; i++) {
bytesArray[i] = _bytes32[i];
}
return string(bytesArray);
}
}