const SECRET_KEY_STORAGE_KEY = "secretKey"; Storage.privateGet(SECRET_KEY_STORAGE_KEY); State.init({ secretKey: null, airesponse: "", progressText: "", aiquestion: `A blue sky, and green fields`, accountId: "", iframeMessage: null, }); function init_iframe() { const secretKey = Storage.privateGet(SECRET_KEY_STORAGE_KEY); State.update({ secretKey, iframeMessage: secretKey ? { command: "useaccount", secretKey: secretKey, } : { command: "createaccount", }, }); } function ask_ai() { State.update({ iframeMessage: { command: "ask_ai", aiquestion: state.aiquestion, ts: new Date().getTime(), }, }); console.log("state updated", state.iframeMessage); } function changeSecretKey(secretKey) { State.update({ secretKey }); Storage.privateSet(SECRET_KEY_STORAGE_KEY, secretKey); init_iframe(); } function handleMessage(msg) { switch (msg.command) { case "accountcreated": Storage.privateSet(SECRET_KEY_STORAGE_KEY, msg.secretKey); State.update({ accountId: msg.accountId, secretKey: msg.secretKey }); break; case "airesponse": State.update({ airesponse: msg.airesponse, progress: false, error: msg.error, }); break; case "aiprogress": State.update({ progressText: state.progressText + msg.progressmessage, progress: true, }); break; case "usingaccount": State.update({ accountId: msg.accountId }); break; case "mint": State.update({ error: "" }); Near.call( "jsinrustnft.near", "nft_mint", Object.assign({}, msg.args, { title: msg.args.token_id, description: state.aiquestion, }), undefined, 1_000_00000_00000_00000_00000n.toString() ); break; case "error": State.update({ error: msg.error }); break; } } const ProgressWrapper = styled.div` .progress-border { height: 50px; width: 100%; } .progress-text { position: absolute; right: 0px; white-space: nowrap; color: #fff; padding-top: 6px; font-size: 20px; } .progress-fill { background-color: rgba(0,130,0, 0.5); z-index: 100; height: 50px; width: 25%; animation-name: indeterminate; animation-duration: 2s; animation-iteration-count: infinite; } @keyframes indeterminate { 0% { margin-left: 0%; width: 25%;} 25% { width: 40%; } 50% { margin-left: 75%; width: 25%; } 75% { width: 40%; } 100% { margin-left: 0%; width: 25%; } } `; const progressIndicator = ( <> {state.progress ? ( <ProgressWrapper> <div id="main-progress-bar" class="progress-border"> <div class="progress-text">{state.progressText}</div> <div class="progress-fill"></div> </div> </ProgressWrapper> ) : ( <button onClick={ask_ai}>Ask ChatGPT</button> )} </> ); const iframe = ( <iframe message={state.iframeMessage} onMessage={handleMessage} onLoad={init_iframe} src="data:text/html;base64,<!DOCTYPE html>
<html>
<head>
    <meta charset="UTF-8">
    <meta name="viewport" content="width=device-width, initial-scale=1" />
    <style>
        body {
            font-family: monospace;
            font-size: 14px;
        }

        input {
            padding: 5px;
            border: #55c solid 4px;
            background-color: white;
            color: black;
        }
        input.error {
            border: red solid 6px;  
        }

        button {
            padding: 10px;
            border: none;
            border: #114 solid 4px;
            background-color: white;
            color: black;
        }

        button:hover {
            background-color: #555;
            color: white;
        }


        #previewresultview {
            width: 500px;
            max-width: 100%;
        }

        a:visited, a {
            color: white;
        }

        #contractidspan {
            background-color: white;
            color: black;
            padding: 5px;
        }
    </style>
</head>

<body>
    <p>
        Token id: <input type="text" id="token_id_input" value="22" />
        &nbsp; Font size: <input type="number" id="font_size_input" value="3" min="1" max="4" />
    </p>
    <p>
        Owner: <input type="text" id="owner_input" value="" />
    </p>
    <button id="mint_button">Mint</button>
    <button id="preview_button">Preview</button>
    </p>
    <div>
        <pre id="mintresultview"></pre>
    </div>
    <input type="text" id="color_input" style="width: 50px;position: absolute; display: none; z-index: 1000;" />
    <div id="previewresultview">

    </div>
</body>
<script type="module">import"https://cdn.jsdelivr.net/npm/near-api-js@2.1.3/dist/near-api-js.min.js";import"https://cdn.jsdelivr.net/npm/js-sha256@0.9.0/src/sha256.min.js";const e="mainnet",t=new nearApi.keyStores.InMemoryKeyStore;let n;const a={keyStore:t,networkId:e,nodeUrl:`https://rpc.${e}.near.org`,walletUrl:`https://wallet.${e}.near.org`,helperUrl:`https://helper.${e}.near.org`,explorerUrl:`https://explorer.${e}.near.org`},r=await nearApi.connect(a);async function o(a){const o=nearApi.utils.KeyPair.fromString(a),s=Buffer.from(o.publicKey.data).toString("hex");return await t.setKey(e,s,o),n=await r.account(s),s}async function s(){const e=new nearApi.Contract(n,d,{viewMethods:["call_js_func"]}),t=await e.call_js_func({function_name:"svg_preview",token_id:document.getElementById("token_id_input").value,font_size:document.getElementById("font_size_input").value,colors:c});document.getElementById("previewresultview").innerHTML=t.svg}async function i(e){window.parent.postMessage({command:"aiprogress",progressmessage:"creating request"},globalThis.parentOrigin);const t=await async function(e){const t=n.accountId,a=JSON.stringify(e),r=sha256(a),o=await n.connection.signer.getPublicKey(n.accountId,n.connection.networkId),s=await n.findAccessKey();if(!s)throw new Error(`Account has no funds. From your wallet, send a small amount to ${n.accountId}`);const i=s.accessKey,c=++i.nonce,l=nearApi.utils.serialize.base_decode(i.block_hash),d=nearApi.transactions.createTransaction(n.accountId,o,"jsinrust.near",c,[nearApi.transactions.functionCall("ask_ai",{message_hash:r},"30000000000000",5000000000000000000000n)],l),[u,p]=await nearApi.transactions.signTransaction(d,n.connection.signer,n.accountId,n.connection.networkId);return JSON.stringify({signed_transaction:Buffer.from(p.encode()).toString("base64"),transaction_hash:nearApi.utils.serialize.base_encode(u),sender_account_id:t,messages:e})}(e);window.parent.postMessage({command:"aiprogress",progressmessage:"sending request"},globalThis.parentOrigin);const a=await fetch("https://near-openai.vercel.app/api/openaistream",{method:"POST",body:t});if(!a.ok)throw new Error(`${a.status} ${a.statusText}\n${await a.text()}\n`);const r=a.body.getReader(),o=[];for(;;){const{done:e,value:t}=await r.read();if(e)break;const n=(new TextDecoder).decode(t);o.push(n),window.parent.postMessage({command:"aiprogress",progressmessage:n},globalThis.parentOrigin);let a=o.join("");const s=a.indexOf("[");if(s>-1){a=a.substring(s);const e=a.lastIndexOf('",');if(e>-1){a=a.substring(0,e+1)+"]]";const t=Array.from(document.querySelectorAll("#previewresultview svg rect"));JSON.parse(a).flat().forEach(((e,n)=>t[n].attributes.fill.value=e))}}}return o.join("")}let c;document.getElementById("mint_button").addEventListener("click",(async()=>{const e=document.getElementById("owner_input"),t=document.getElementById("owner_input").value.trim();if(!t)return e.classList.add("error"),void window.parent.postMessage({command:"error",error:"You must provide an owner account for the NFT"},globalThis.parentOrigin);e.classList.remove("error");try{await r.connection.provider.query({request_type:"view_account",finality:"final",account_id:t})}catch(n){return window.parent.postMessage({command:"error",error:`Unknown account: ${t}`},globalThis.parentOrigin),void e.classList.add("error")}e.classList.remove("error");const a=document.getElementById("token_id_input"),o=a.value.trim();if(!o)return a.classList.add("error"),void window.parent.postMessage({command:"error",error:"Token ID cannot be empty"},globalThis.parentOrigin);a.classList.remove("error");const s=new nearApi.Contract(n,d,{viewMethods:["nft_token"]}),i=await s.nft_token({token_id:o});if(console.log(i),i)return a.classList.add("error"),void window.parent.postMessage({command:"error",error:`Token ID "${o}" is already taken, must be unique`},globalThis.parentOrigin);a.classList.remove("error"),window.parent.postMessage({command:"mint",args:{token_id:o,token_owner_id:t,font_size:document.getElementById("font_size_input").value,colors:c}},globalThis.parentOrigin)}));const l=document.getElementById("preview_button"),d="jsinrustnft.near";l.addEventListener("click",(async()=>{document.getElementById("previewresultview").innerHTML="Please wait while generating preview";try{await s();const e=document.getElementById("color_input");c=[],Array.from(document.querySelectorAll("#previewresultview svg rect")).forEach(((t,n)=>{c.push(t.attributes.fill.value),t.addEventListener("click",(a=>{e.style.top=`${a.clientY}px`,e.style.left=`${a.clientX}px`,e.value=t.attributes.fill.value,e.style.display="block",console.log(e),e.onblur=()=>{t.attributes.fill.value=e.value,c[n]=e.value,e.style.display="none"}}))}))}catch(e){document.getElementById("mintresultview").innerHTML=e.toString()}})),window.onmessage=async r=>{switch(globalThis.parentOrigin=r.origin,console.log("iframe got message",r.data),r.data.command){case"createaccount":const{secretKey:d,accountId:u}=await async function(){const r=nearApi.utils.KeyPairEd25519.fromRandom(),o=Buffer.from(r.publicKey.data).toString("hex");await t.setKey(e,o,r);const s=await nearApi.connect(a);return n=await s.account(o),{secretKey:r.secretKey,accountId:o}}();window.parent.postMessage({command:"accountcreated",secretKey:d,accountId:u},globalThis.parentOrigin);break;case"useaccount":window.parent.postMessage({command:"usingaccount",accountId:await o(r.data.secretKey)},globalThis.parentOrigin);break;case"ask_ai":let p,g;try{c=new Array(81).fill("black"),await s(),g=await i([{role:"user",content:"In the next message there will be a description that you should use to create 9x9 pixel art, and as inspiration for a word to be used as a token id. If the description is weak, then be creative."},{role:"user",content:r.data.aiquestion},{role:"user",content:"\n                    Give me only a json result that I can parse directly, and no other surrounding context. The json should contain a property called image which is a 9x9 array with string of CSS color codes representing the pixel art. The other property should be named token_id and contain the word for the token id."}]);const e=JSON.parse(g);c=e.image.flat(),document.getElementById("token_id_input").value=e.token_id,l.click()}catch(e){p=`Error:\n${e.message??""}\n\n${g??""}\n                `}window.parent.postMessage({command:"airesponse",airesponse:g,error:p},globalThis.parentOrigin)}},console.log("iframe loaded");
</script>
</html>" style={{ width: "100%", height: "700px", border: "none" }} ></iframe> ); const secretKeyToggle = state.showSecretKey ? ( <> <button onClick={() => State.update({ showSecretKey: false })}>Hide</button> <input type="text" value={state.secretKey} onChange={(e) => changeSecretKey(e.target.value)} ></input> </> ) : ( <button onClick={() => State.update({ showSecretKey: true })}>Show</button> ); const responseArea = state.error ? ( <div style={{ color: "red", backgroundColor: "#f8f8f8" }}> <Markdown text={state.error} /> </div> ) : ( "" ); const accountArea = ( <> <p> Spending account ID: <pre>{state.accountId}</pre> </p> <p>Spending account secret key: {secretKeyToggle}</p> </> ); const aiquestionarea = ( <textarea style={{ width: "100%" }} onChange={(e) => State.update({ aiquestion: e.target.value })} value={state.aiquestion} ></textarea> ); return ( <> <p> Create some image and text and mint your own NFT that you can list and trade on{" "} <a href="https://www.mintbase.xyz/contract/jsinrustnft.near/nfts/all/0" target="_blank" > Mintbase </a> </p> <p> <b>NOTE:</b> Each request to ChatGPT costs about 0.005 NEAR. Make sure the spending account below is funded, and you can also get full access to that account by using the secret key. Only you have the key to this account, so don't loose it. </p> {aiquestionarea} {progressIndicator} {responseArea} {iframe} <p></p> {accountArea} </> );