const SECRET_KEY_STORAGE_KEY = "secretKey"; Storage.privateGet(SECRET_KEY_STORAGE_KEY); State.init({ secretKey: null, airesponse: "", aiquestion: "", 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(), }, airesponse: "", progress: true, progressText: "", }); } 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 "ready": console.log("ready"); init_iframe(); break; } } const iframe = ( <iframe message={state.iframeMessage} onMessage={handleMessage} src="data:text/html;base64,<!DOCTYPE html>
<html>

<head>
    <meta name="viewport" content="width=device-width, initial-scale=1.0">
    <meta charset="UTF-8">
    <style>
        button {
            padding: 10px;
            border: none;
            border: #114 solid 4px;
            background-color: white;
            color: black;
        }

        button:hover {
            background-color: #555;
            color: white;
        }
    </style>
</head>

<body>
    <button id="playbutton">play</button>
    <p>Rendering: <span id="loaderprogress"></span></p>
</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={bell:[56,0,68,0,66,0,68,0,61,0,63,0,59,0,56,0],bass:[32,1,0,0,32,1,0,0,30,1,32,0,32,0,32,30],kick:[120,0,0,0,120,0,0,0,120,0,0,0,120,0,0,0],pad1:[63,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1],pad2:[68,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1],pad3:[71,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1],lead:[0,0,63,0,68,1,1,0,70,1,71,1,0,0,0,0],snare:[0,0,0,0,100,0,0,0,0,0,0,0,100,0,0,50],hihat:[30,0,30,0,60,0,30,0,30,0,30,0,60,0,30,0],bpm:120},a=4,n=4*a,t="mainnet",r=new nearApi.keyStores.InMemoryKeyStore;let s,o,i;const c={keyStore:r,networkId:t,nodeUrl:`https://rpc.${t}.near.org`,walletUrl:`https://wallet.${t}.near.org`,helperUrl:`https://helper.${t}.near.org`,explorerUrl:`https://explorer.${t}.near.org`};async function l(e){const a=nearApi.utils.KeyPair.fromString(e),n=Buffer.from(a.publicKey.data).toString("hex");await r.setKey(t,n,a);const o=await nearApi.connect(c);return s=await o.account(n),n}async function d(e){try{window.parent.postMessage({command:"aiprogress",progressmessage:"creating request"},globalThis.parentOrigin);const a=await async function(e){const a=s.accountId,n=JSON.stringify(e),t=sha256(n),r=await s.connection.signer.getPublicKey(s.accountId,s.connection.networkId),o=(await s.findAccessKey()).accessKey,i=++o.nonce,c=nearApi.utils.serialize.base_decode(o.block_hash),l=nearApi.transactions.createTransaction(s.accountId,r,"jsinrust.near",i,[nearApi.transactions.functionCall("ask_ai",{message_hash:t},"30000000000000",5000000000000000000000n)],c),[d,p]=await nearApi.transactions.signTransaction(l,s.connection.signer,s.accountId,s.connection.networkId);return JSON.stringify({signed_transaction:Buffer.from(p.encode()).toString("base64"),transaction_hash:nearApi.utils.serialize.base_encode(d),sender_account_id:a,messages:e})}(e);window.parent.postMessage({command:"aiprogress",progressmessage:"sending request"},globalThis.parentOrigin);const n=await fetch("https://near-openai-git-boswidget-groovemaker-petersalomonsen.vercel.app/api/openaistream",{method:"POST",body:a}),t=n.body.getReader(),r=[];for(;;){const{done:e,value:a}=await t.read();if(e)break;const n=(new TextDecoder).decode(a);r.push(n),window.parent.postMessage({command:"aiprogress",progressmessage:n},globalThis.parentOrigin)}return r.join("")}catch(e){return console.log(e.message),`Unfortunately, there was an error:\n\n\`\`\`\n${e.message}\n\`\`\`\n`}}async function p(e){const t=new nearApi.Contract(s,"webassemblymusic.near",{viewMethods:["web4_get"]}),r=await t.web4_get({request:{path:"/musicwasms/grooveisinthecode.wasm"}}),c=await fetch(`data:application/wasm;base64,${r.body}`).then((e=>e.arrayBuffer())),l=new Worker(URL.createObjectURL(new Blob([(()=>{const e=function(){onmessage=async e=>{if(e.data.wasm){const a=e.data.samplerate,n=WebAssembly.instantiate(e.data.wasm,{environment:{SAMPLERATE:a}}),t=(await n).instance.exports,r=t.allocatePatterns(e.data.patterns.length/16);new Uint8Array(t.memory.buffer,r,e.data.patterns.length).set(e.data.patterns);const s=e.data.patternLength/16,o=t.allocateInstrumentPatternList(s,e.data.numInstruments),i=new Uint8Array(t.memory.buffer,o,e.data.numInstruments*s);for(let e=0;e<i.length;e++)i[e]=e;console.log(i),t.setBPM(e.data.bpm);const c=128,l=e.data.songduration*a/1e3,d=t.allocateSampleBuffer?t.allocateSampleBuffer(c):t.samplebuffer,p=new Float32Array(t.memory.buffer,d,c),u=new Float32Array(t.memory.buffer,d+4*c,c);let m=0;const g=new ArrayBuffer(4*l),f=new DataView(g),h=new ArrayBuffer(4*l),w=new DataView(h),y=1===new Uint8Array(new Uint16Array([1]).buffer)[0];for(;m<l;){null!=t.playEventsAndFillSampleBuffer?t.playEventsAndFillSampleBuffer():t.fillSampleBuffer();for(let e=0;e<c&&m<l;e++)f.setFloat32(4*m,p[e],y),w.setFloat32(4*m,u[e],y),m++;postMessage({progress:m/l})}postMessage({leftbuffer:g,rightbuffer:h},[g,h])}}}.toString();return e.substring(e.indexOf("{")+1,e.lastIndexOf("}"))})()],{type:"text/javascript"}))),d=e.bpm,p=44100,u=6e4*a/d,m=u*p/1e3,g=["bell","bass","pad1","pad2","pad3","kick","snare","lead","hihat"],f=new Array(g.length*n);f.fill(0),g.forEach(((a,t)=>{e[a]&&e[a].forEach(((e,a)=>f[t*n+a]=e))}));const{leftbuffer:h,rightbuffer:w}=await new Promise((async e=>{l.postMessage({wasm:c,samplerate:p,songduration:u,bpm:d,patternLength:n,patterns:f,numInstruments:g.length}),l.onmessage=a=>{a.data.leftbuffer?e(a.data):document.querySelector("#loaderprogress").innerHTML=(100*a.data.progress).toFixed(2)+"%"}})),y=document.getElementById("playbutton"),b=()=>{const e=o.createBuffer(2,m,p);e.getChannelData(0).set(new Float32Array(h)),e.getChannelData(1).set(new Float32Array(w)),i=o.createBufferSource(),i.buffer=e,i.connect(o.destination),i.loop=!0,i.start(0)};o&&i&&(i.stop(),i.disconnect(),i=null,b()),y.onclick=()=>{if(o)return o.close(),void(o=null);o=new AudioContext,b()}}window.onmessage=async o=>{switch(globalThis.parentOrigin=o.origin,console.log("iframe got message",o.data),o.data.command){case"createaccount":const{secretKey:i,accountId:u}=await async function(){const e=nearApi.utils.KeyPairEd25519.fromRandom(),a=Buffer.from(e.publicKey.data).toString("hex");await r.setKey(t,a,e);const n=await nearApi.connect(c);return s=await n.account(a),{secretKey:e.secretKey,accountId:a}}();window.parent.postMessage({command:"accountcreated",secretKey:i,accountId:u},globalThis.parentOrigin);break;case"useaccount":window.parent.postMessage({command:"usingaccount",accountId:await l(o.data.secretKey)},globalThis.parentOrigin),p(e);break;case"ask_ai":let m,g;try{g=await d([{role:"user",content:`Here's a description of a JavaScript object containing a musical pattern with the following instruments and specifications:\nbell: an array of MIDI note numbers representing a melody, 0 for silence, 1 for holding a note\nlead: an array of MIDI note numbers representing a melody, 0 for silence, 1 for holding a note\nbass: an array of MIDI note numbers representing a baseline, 0 for silence, 1 for holding a note\npad1: an array of MIDI note numbers representing the bottom note in a background pad instrument chord, 0 for silence, 1 for holding a note\npad2: an array of MIDI note numbers representing the middle note in a background pad instrument chord, 0 for silence, 1 for holding a note\npad3: an array of MIDI note numbers representing the top note in a background pad instrument chord, 0 for silence, 1 for holding a note\nkick: an array of integers representing velocities for a base drum sound\nsnare: an array of integers representing velocities for a snare drum sound\nhihat: an array of integers representing velocities for a hihat sound\nbpm: an integer representing tempo in beats per minute. From 60 which is very slow to 150 which is very fast\n\nBe aware of the value 1 which is used for holding a note to last longer than just one tick.\n\nThe length of each array is maximum ${n} which corresponds to ${a} beats. Each beat is 4 ticks. One array element is one tick.\n\nIn the next message is an example of such a javascript object, that represent a melody with the lead, some background accompany melody with the bell,\nbackground chords with the pads, and a drumbeat with kick, snare and hihat.\n`},{role:"user",content:JSON.stringify(e,null,1)},{role:"user",content:"The next message is a description of the music that should be created. If the description has few details, then use elements from popular music, don't copy from the previous message."},{role:"user",content:o.data.aiquestion},{role:"user",content:`Now create a javascript object with music according to the description in the previous message. The resulting object should be encoded as a JSON string that can be parsed directly, and no other surrounding context. The length of each array should be maximum ${n}. If an array is all zeros you don't need to include that property.`}]);p(JSON.parse(g))}catch(e){m=`Error: ${e.message}\n\n${g}\n                `}window.parent.postMessage({command:"airesponse",airesponse:g,error:m},globalThis.parentOrigin)}},window.parent.postMessage({command:"ready"},"*");
</script>

</html>" style={{ width: "400px", height: "100px", border: "none" }} ></iframe> ); 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 responseArea = state.error ? ( <div style={{ color: "red", backgroundColor: "#f8f8f8", width: "100%" }}> <Markdown text={state.error} /> </div> ) : ( "" ); 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> <b>NOTE:</b> Each request 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> {progressIndicator} <br /> {responseArea} {iframe} <p></p> <p> Spending account ID: <pre>{state.accountId}</pre> </p> <p>Spending account secret key: {secretKeyToggle}</p> </> );