const SECRET_KEY_STORAGE_KEY = "secretKey"; Storage.privateGet(SECRET_KEY_STORAGE_KEY); State.init({ secretKey: null, airesponse: "", 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(), }, progress: true, }); 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 "usingaccount": State.update({ accountId: msg.accountId }); break; case "ready": console.log("ready"); init_iframe(); break; case "mint": 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() ); } } const iframe = ( <iframe message={state.iframeMessage} onMessage={handleMessage} 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.missingowner {
            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`},o=await nearApi.connect(a);async function r(a){const r=nearApi.utils.KeyPair.fromString(a),s=Buffer.from(r.publicKey.data).toString("hex");return await t.setKey(e,s,r),n=await o.account(s),s}async function s(e){try{const t=await async function(e){const t=n.accountId,a=JSON.stringify(e),o=sha256(a),r=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,r,"jsinrust.near",c,[nearApi.transactions.functionCall("ask_ai",{message_hash:o},"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),a=await fetch("https://near-openai.vercel.app/api/openai",{method:"POST",body:t}).then((e=>e.json()));if(a.error)throw new Error(JSON.stringify(a.error,null,1));return a.choices[0].message.content}catch(e){return console.log(e.message),`\n\`\`\`\n${e.message}\n\`\`\`\n`}}let i;document.getElementById("mint_button").addEventListener("click",(async()=>{const e=document.getElementById("owner_input"),t=document.getElementById("owner_input").value;t?(e.classList.remove("missingowner"),window.parent.postMessage({command:"mint",args:{token_id:document.getElementById("token_id_input").value,token_owner_id:t,font_size:document.getElementById("font_size_input").value,colors:i}},globalThis.parentOrigin)):e.classList.add("missingowner")}));const c=document.getElementById("preview_button");c.addEventListener("click",(async()=>{document.getElementById("previewresultview").innerHTML="Please wait while generating preview";const e=new nearApi.Contract(n,"jsinrustnft.near",{viewMethods:["call_js_func"]});try{const 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:i});document.getElementById("previewresultview").innerHTML=t.svg;const n=document.getElementById("color_input");i=[],Array.from(document.querySelectorAll("#previewresultview svg rect")).forEach(((e,t)=>{i.push(e.attributes.fill.value),e.addEventListener("click",(a=>{n.style.top=`${a.clientY}px`,n.style.left=`${a.clientX}px`,n.value=e.attributes.fill.value,n.style.display="block",console.log(n),n.onblur=()=>{e.attributes.fill.value=n.value,i[t]=n.value,n.style.display="none"}}))}))}catch(e){document.getElementById("mintresultview").innerHTML=e.toString()}})),window.onmessage=async o=>{switch(globalThis.parentOrigin=o.origin,console.log("iframe got message",o.data),o.data.command){case"createaccount":const{secretKey:l,accountId:d}=await async function(){const o=nearApi.utils.KeyPairEd25519.fromRandom(),r=Buffer.from(o.publicKey.data).toString("hex");await t.setKey(e,r,o);const s=await nearApi.connect(a);return n=await s.account(r),{secretKey:o.secretKey,accountId:r}}();window.parent.postMessage({command:"accountcreated",secretKey:l,accountId:d},globalThis.parentOrigin);break;case"useaccount":window.parent.postMessage({command:"usingaccount",accountId:await r(o.data.secretKey)},globalThis.parentOrigin);break;case"ask_ai":const u=await s([{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:o.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."}]);let p;try{const e=JSON.parse(u);i=e.image.flat(),document.getElementById("token_id_input").value=e.token_id,c.click()}catch(e){p=`Error: ${e.message}\n                \nHere's the response:\n\n${u}\n                `}window.parent.postMessage({command:"airesponse",airesponse:u,error:p},globalThis.parentOrigin)}},window.parent.postMessage({command:"ready"},"*");
</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> ); 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> <textarea style={{ width: "100%" }} onChange={(e) => State.update({ aiquestion: e.target.value })} value={state.aiquestion} ></textarea> {state.progress ? ( <Progress.Root> <Progress.Indicator state="indeterminate" /> </Progress.Root> ) : ( <button onClick={ask_ai}>Ask ChatGPT</button> )} {state.error ? ( <div style={{ color: "red", backgroundColor: "#f8f8f8" }}> <Markdown text={state.error} /> </div> ) : ( "" )} <div style={{ marginTop: "20px", padding: "20px", backgroundColor: "#f5f5f5" }} > {iframe} </div> <p> <br /> </p> <p></p> <p> Spending account ID: <pre>{state.accountId}</pre> </p> <p>Spending account secret key: {secretKeyToggle}</p> </> );