mirror of
https://github.com/sune-org/store.git
synced 2026-01-13 16:17:58 +00:00
1 line
18 KiB
JSON
1 line
18 KiB
JSON
[{"id":"e1yibwd","name":"Marketplace","pinned":false,"avatar":"","url":"gh://sune-org/store@main/marketplace.sune","updatedAt":1757269974397,"settings":{"model":"openai/gpt-5","temperature":"1","top_p":"0.97","top_k":"0","frequency_penalty":"0","presence_penalty":"0","repetition_penalty":"1","min_p":"0","top_a":"0","max_tokens":"","verbosity":"","reasoning_effort":"default","system_prompt":"","html":"<!--\nSune: Marketplace\nVersion: 2.3.4\nDescription: Discover and install new sunes from a configurable catalog URL. URL is saved in SUNE.storage.\n-->\n<section id=\"suneMarketplace\" class=\"p-4 md:p-6 bg-slate-50\">\n <header class=\"mb-4 pb-3 border-b border-slate-200\">\n <div class=\"flex items-center gap-2\">\n <i data-lucide=\"store\" class=\"h-5 w-5 text-slate-700\"></i>\n <h1 class=\"text-xl font-bold text-slate-800\">Sune Store</h1>\n <span class=\"text-[10px] font-mono text-slate-400\">v2.3.3</span>\n </div>\n </header>\n\n <div id=\"mpContent\" class=\"min-h-[120px]\">\n <div class=\"flex justify-center items-center py-12\">\n <div class=\"animate-spin rounded-full h-10 w-10 border-b-2 border-slate-500\"></div>\n </div>\n </div>\n\n <div id=\"mpLogWrap\" class=\"mt-8 hidden\">\n <div class=\"flex justify-between items-center mb-2\">\n <h3 class=\"text-sm font-semibold text-slate-600\">Terminal</h3>\n <div class=\"text-xs text-slate-500\">Avatar diagnostics only</div>\n </div>\n <div id=\"mpLog\" class=\"bg-slate-800 text-white font-mono text-[11px] rounded-lg p-3 h-40 overflow-y-auto\"></div>\n </div>\n\n <div class=\"mt-8 pt-6 border-t border-slate-200\">\n <div class=\"text-center bg-white p-5 rounded-lg border border-slate-200 shadow-sm\">\n <div class=\"flex items-center justify-center gap-2\">\n <i data-lucide=\"git-pull-request-arrow\" class=\"h-5 w-5 text-slate-600\"></i>\n <h3 class=\"text-sm font-semibold text-slate-800\">Contribute your Sune</h3>\n </div>\n <p class=\"mt-2 text-xs text-slate-600 max-w-lg mx-auto\">\n Share your sune with everyone. Open a pull request in\n <a href=\"https://github.com/sune-org/store\" target=\"_blank\" rel=\"noopener noreferrer\" class=\"font-medium text-sky-600 hover:underline\">sune-org/store</a>.\n </p>\n <div class=\"mt-3 flex items-center justify-center gap-2\">\n <a href=\"https://github.com/sune-org/store\" target=\"_blank\" rel=\"noopener noreferrer\" class=\"inline-flex items-center gap-1 px-3 py-1.5 rounded-md bg-slate-900 text-white text-xs hover:bg-slate-800\">\n <i data-lucide=\"external-link\" class=\"h-3.5 w-3.5\"></i><span>Open Repository</span>\n </a>\n </div>\n </div>\n </div>\n\n <div class=\"mt-6 pt-4 border-t border-slate-200\">\n <label for=\"mpCat\" class=\"block text-xs font-medium text-slate-600 mb-1.5\">Catalog URL</label>\n <div class=\"flex flex-col sm:flex-row gap-2\">\n <div class=\"relative flex-1\">\n <i data-lucide=\"link-2\" class=\"pointer-events-none absolute left-3 top-1/2 -translate-y-1/2 h-4 w-4 text-slate-400\"></i>\n <input id=\"mpCat\" type=\"url\" inputmode=\"url\" spellcheck=\"false\"\n class=\"peer w-full h-11 pl-9 pr-10 rounded-lg border border-slate-300 bg-white text-sm placeholder:text-slate-400 shadow-sm focus:outline-none focus:ring-4 focus:ring-slate-200/70 focus:border-slate-400 transition\"\n placeholder=\"https://raw.githubusercontent.com/sune-org/store/main/catalog.json (or gh://user/repo@branch/path.json)\"/>\n <span class=\"pointer-events-none hidden sm:flex items-center gap-1 absolute right-2 top-1/2 -translate-y-1/2 text-[10px] text-slate-400\">\n <i data-lucide=\"keyboard\" class=\"h-3.5 w-3.5\"></i>Enter to save\n </span>\n </div>\n <button id=\"mpRefresh\" class=\"h-11 px-3 rounded-lg bg-slate-900 text-white text-xs hover:bg-slate-800 active:scale-[.99] transition inline-flex items-center justify-center gap-1 shadow-sm\">\n <i data-lucide=\"refresh-cw\" class=\"h-3.5 w-3.5\"></i><span>Refresh</span>\n </button>\n </div>\n <p class=\"mt-1 text-[11px] text-slate-500\">Accepts raw https://… or gh://user/repo@branch/path.json. Value is saved per active sune (SUNE.storage).</p>\n </div>\n</section>\n\n<script>\n(()=>{const $=s=>document.querySelector(s),ico=()=>window.lucide?.createIcons(),esc=s=>String(s??'').replace(/[&<>\"]/g,m=>({'&':'&','<':'<','>':'>','\"':'"'}[m])),\nSID=()=>window.SUNE?.id||'default',LSK=k=>`sune-mp-v233:${SID()}:${k}`,\nroot=$(\"#suneMarketplace\"),content=$(\"#mpContent\"),logWrap=$(\"#mpLogWrap\"),logBox=$(\"#mpLog\"),catIn=$(\"#mpCat\"); if(!root)return;\n\n/* Terminal (optional) */\nconst LOG_VISIBLE=false; logWrap.classList.toggle('hidden',!LOG_VISIBLE);\nconst log=(m,t='i')=>{if(!LOG_VISIBLE||!logBox)return;const d=document.createElement('div'),map={i:['»','text-slate-300'],s:['✔','text-green-400'],e:['✖','text-red-400'],w:['⚠','text-yellow-300']},[p,c]=map[t]||map.i;d.className=`flex items-start gap-2 ${c}`;d.innerHTML=`<span class=\"pt-0.5\">${p}</span><span>${esc(m)}</span>`;logBox.appendChild(d);logBox.scrollTop=logBox.scrollHeight};\n\n/* Helpers */\nconst b64url2b64=s=>{let x=(s||'').replace(/-/g,'+').replace(/_/g,'/');while(x&&x.length%4)x+='=';return x};\nconst sniff=b=>b.startsWith('iVBOR')?'image/png':(b.startsWith('/9j/')?'image/jpeg':(b.startsWith('UklGR')?'image/webp':(b.startsWith('R0lGOD')?'image/gif':'application/octet-stream')));\nconst avatarSrc=v=>{if(!v){log('[avatar] none','w');return ''} if(/^data:/.test(v)){const med=v.slice(5,v.indexOf(';'))||'unknown';log(`[avatar] data-uri (type=${med}) len=${v.length}`,'s');return v} const b64=b64url2b64(v),mime=sniff(b64),head=b64.slice(0,12);log(`[avatar] b64url -> mime=${mime} head=${head} len=${b64.length}`,(mime==='application/octet-stream')?'w':'s');return `data:${mime};base64,${b64}`};\nconst ghRaw=u=>{try{if(!u)return u;if(u.startsWith('gh://')){const s=u.slice(5),[hd,...rest]=s.split('/'),[ur,br]=hd.split('@'),[user,repo]=ur.split('/');return `https://raw.githubusercontent.com/${user}/${repo}/${br}/${rest.join('/')}`}return u}catch{return u}};\nconst ghUrlFromRaw=u=>{try{const r=new URL(u);if(r.hostname!=='raw.githubusercontent.com')return'';const[,u1,r1,b,...p]=r.pathname.split('/');return `gh://${u1}/${r1}@${b}/${p.join('/')}`}catch{return''}};\nconst T=s=>(s||'Untitled').replace(/\\.sune$/,'').replace(/[-_]/g,' ').replace(/\\b\\w/g,m=>m.toUpperCase());\nconst installed=()=>new Set((window.SUNE?.list||[]).map(x=>x?.id).filter(Boolean));\n\n/* UI builders */\nconst btn=(it,ins)=>ins?`<button class=\"mt-3 w-full px-3 py-1.5 text-xs font-medium rounded-md bg-slate-100 text-slate-500 cursor-default flex items-center justify-center gap-1\" disabled><i data-lucide=\"check-circle\" class=\"h-4 w-4\"></i><span>Installed</span></button>`:`<button class=\"mpInstall mt-3 w-full px-3 py-1.5 text-xs font-medium rounded-md bg-slate-700 text-white hover:bg-slate-800 active:scale-[.99] transition flex items-center justify-center gap-1\" data-id=\"${esc(it.id||'')}\" data-name=\"${esc(T(it.name))}\" data-raw=\"${esc(it.raw||'')}\"><i data-lucide=\"download-cloud\" class=\"h-4 w-4\"></i><span>Install</span></button>`;\nconst fallbackNode=()=>{const d=document.createElement('div');d.className='h-9 w-9 rounded-md bg-slate-100 border border-slate-200 text-slate-500 flex items-center justify-center';d.innerHTML='<i data-lucide=\"image-off\" class=\"h-4 w-4\"></i>';return d};\nconst card=(it,ins)=>{const src=avatarSrc(it.avatar);return `<article class=\"bg-white border border-slate-200 rounded-lg shadow-sm p-4 flex flex-col\">\n <div class=\"flex items-center gap-3\">\n ${src?`<img alt=\"\" src=\"${src}\" loading=\"lazy\" data-name=\"${esc(T(it.name))}\" class=\"mpAvatar h-9 w-9 rounded-md object-cover bg-slate-100 border border-slate-200\">`:`<div class=\"h-9 w-9 rounded-md bg-slate-100 border border-slate-200 text-slate-500 flex items-center justify-center\"><i data-lucide=\"image-off\" class=\"h-4 w-4\"></i></div>`}\n <div class=\"min-w-0\">\n <h3 class=\"text-sm font-semibold text-slate-800 truncate\">${esc(T(it.name))}</h3>\n <p class=\"text-[11px] text-slate-500 truncate\">${esc(it.description||'Official utility sune.')}</p>\n </div>\n </div>\n <div class=\"mt-2 text-[10px] text-slate-400\">id: <span class=\"font-mono\">${esc(it.id||'(none)')}</span></div>\n ${btn(it,ins)}\n</article>`};\nconst bindAvatars=()=>{ico(); if(!LOG_VISIBLE)return;\n document.querySelectorAll('.mpAvatar').forEach(img=>{\n const nm=img.dataset.name||'(unknown)';\n img.addEventListener('load',()=>log(`[avatar] ${nm}: loaded ${img.naturalWidth}x${img.naturalHeight}`,'s'),{once:true});\n img.addEventListener('error',()=>{const s=String(img.getAttribute('src')||'');log(`[avatar] ${nm}: load error (src head=\"${s.slice(0,32)}\")`,'e');const fb=fallbackNode();img.replaceWith(fb);ico()},{once:true});\n })\n};\nconst render=items=>{const ids=installed(),html=items.map(i=>card(i,i.id?ids.has(i.id):false)).join('');content.innerHTML=`<div class=\"grid gap-3 grid-cols-1 sm:grid-cols-2 lg:grid-cols-3 xl:grid-cols-4\">${html}</div>`;bindAvatars()};\n\n/* Per-sune storage for Catalog URL (SUNE.storage) */\nconst DEF='https://raw.githubusercontent.com/sune-org/store/main/catalog.json';\nconst getStorage=()=>Object.assign({},window.SUNE?.storage||{});\nconst getUrl=()=>{try{return (window.SUNE?.storage?.marketplace?.url)||DEF}catch{return DEF}};\nconst setUrl=v=>{const url=(v||'').trim()||DEF,st=getStorage();st.marketplace=Object.assign({},st.marketplace,{url});window.SUNE&&(window.SUNE.storage=st);catIn.value=url};\n\n/* One-time migration from old localStorage cfg -> SUNE.storage */\nconst migrate=()=>{try{const k=LSK('cfg'),raw=localStorage.getItem(k);if(!raw)return;const cfg=JSON.parse(raw||'null');if(cfg?.url&&!window.SUNE?.storage?.marketplace?.url)setUrl(cfg.url);localStorage.removeItem(k)}catch{}};\n\n/* Per-sune cache of fetched catalog list in localStorage (ok to keep) */\nconst loadCat=()=>{try{return JSON.parse(localStorage.getItem(LSK('catalog'))||'null')}catch{return null}};\nconst saveCat=d=>{try{localStorage.setItem(LSK('catalog'),JSON.stringify({t:Date.now(),items:d}))}catch{}};\n\n/* Fetch */\nconst fetchCatalog=async(url)=>{const U=ghRaw(url||DEF),r=await fetch(U,{cache:'no-store'});if(!r.ok)throw new Error('Catalog '+r.status);const raw=await r.json(),items=(Array.isArray(raw)?raw:[]).filter(x=>x&&x.name&&x.raw);saveCat(items);return items};\n\n/* Boot + events */\nconst boot=async(force=false)=>{try{\n const src=getUrl();catIn.value=src;\n const cache=!force&&loadCat(); if(cache?.items)render(cache.items);\n const items=await fetchCatalog(src);render(items)\n}catch(e){content.innerHTML=`<div class=\"text-center py-12 bg-red-50 text-red-700 rounded-lg\"><p>Could not load Marketplace.</p><p class=\"text-xs mt-1\">${esc(e?.message||e)}</p></div>`}finally{ico()}};\n\n$(\"#mpRefresh\")?.addEventListener('click',e=>{e.preventDefault();boot(true)});\ncatIn.addEventListener('keydown',e=>{if(e.key==='Enter'){e.preventDefault();setUrl(catIn.value);boot(true)}});\ncatIn.addEventListener('blur',()=>{setUrl(catIn.value)});\n\ncontent.addEventListener('click',async e=>{const b=e.target.closest('.mpInstall');if(!b)return;\n try{const id=b.dataset.id||'',name=b.dataset.name||'Unknown Sune',raw=b.dataset.raw||'';if(!raw)throw new Error('Missing raw URL');\n b.disabled=true;b.innerHTML=`<i data-lucide=\"loader-circle\" class=\"h-4 w-4 animate-spin\"></i><span>Installing...</span>`;ico();\n const url=ghRaw(raw),res=await fetch(url);if(!res.ok)throw new Error('Fetch '+res.status);\n const arr=await res.json();if(!Array.isArray(arr)||!arr.length)throw new Error('Invalid .sune payload');\n let upd=0;for(const s of arr){if(!s?.id)continue;s.url=s.url||ghUrlFromRaw(url);const ex=(window.SUNE?.list||[]).find(x=>x.id===s.id);if(!ex)window.SUNE.create(s);else if(+s.updatedAt>+ex.updatedAt){Object.assign(ex,s);upd++}}\n if(upd>0)window.SUNE.save?.();window.renderSidebar?.();\n b.outerHTML=`<button class=\"mt-3 w-full px-3 py-1.5 text-xs font-medium rounded-md bg-slate-100 text-slate-500 cursor-default flex items-center justify-center gap-1\" disabled><i data-lucide=\"check-circle\" class=\"h-4 w-4\"></i><span>Installed</span></button>`;ico();\n }catch(err){b.disabled=false;b.innerHTML=`<i data-lucide=\"alert-triangle\" class=\"h-4 w-4\"></i><span>Retry</span>`;ico()}\n});\n\n/* Init */\nmigrate(); catIn.value=getUrl(); boot(); ico();\n})();\n</script>\n","extension_html":"<sune src='https://raw.githubusercontent.com/sune-org/store/refs/heads/main/sync.sune' private />","hide_composer":true},"storage":{"marketplace":{"url":"https://raw.githubusercontent.com/sune-org/store/main/catalog.json"}}}] |