mirror of
https://github.com/sune-org/store.git
synced 2026-01-13 16:17:58 +00:00
1 line
20 KiB
JSON
1 line
20 KiB
JSON
[{"id":"e9r2j2m","name":"Forums","pinned":false,"avatar":"","url":"gh://sune-org/store/forum.sune","updatedAt":1758061083135,"settings":{"model":"","temperature":"","top_p":"","top_k":"","frequency_penalty":"","repetition_penalty":"","min_p":"","top_a":"","verbosity":"","reasoning_effort":"default","system_prompt":"","html":"<div class=\"mx-auto max-w-2xl p-4 sm:p-6 font-sans text-stone-800\" x-data=\"suneForums()\" x-init=\"init()\">\n <!-- Header -->\n <header class=\"flex items-center justify-between mb-4 sm:mb-6 pb-3 border-b border-stone-200\">\n <div class=\"flex items-center gap-3 min-w-0\">\n <button x-show=\"view !== 'list'\" @click.prevent=\"goToList()\" class=\"h-9 w-9 flex-shrink-0 flex items-center justify-center rounded-full hover:bg-stone-100 transition-colors self-start mt-0.5\" aria-label=\"Back\">\n <i data-lucide=\"arrow-left\" class=\"h-5 w-5 text-stone-600\"></i>\n </button>\n <div class=\"truncate\">\n <h1 class=\"text-xl font-bold text-stone-900 truncate\" x-text=\"view === 'list' ? 'Sune Forums' : (view === 'new_post' ? 'Create a Post' : (currentPost ? currentPost.title : '...'))\"></h1>\n <p x-show=\"view === 'post' && currentPost\" class=\"text-xs text-stone-500 mt-0.5\">By <span class=\"font-medium\" x-text=\"currentPost.author_name\"></span> · <span x-text=\"formatDate(currentPost.created_at)\"></span></p>\n </div>\n </div>\n <div class=\"flex items-center gap-2 flex-shrink-0\">\n <span class=\"inline-flex items-center rounded-md bg-stone-100 px-2 py-0.5 text-xs font-medium text-stone-600\">v1.5.0</span>\n <button x-show=\"view === 'list'\" @click.prevent=\"view = 'new_post'\" class=\"px-3 py-1.5 text-sm font-semibold bg-black text-white rounded-lg hover:bg-black/90 transition-colors focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-black focus-visible:ring-offset-2\">\n New Post\n </button>\n </div>\n </header>\n\n <!-- Error State -->\n <div x-show=\"error\" x-transition class=\"p-3 mb-4 bg-red-50 text-red-800 border border-red-200 rounded-lg text-sm\" x-text=\"error\"></div>\n\n <!-- Main Content Area -->\n <main class=\"relative\">\n <!-- Loading Overlay -->\n <div x-show=\"isLoading\" x-transition:enter=\"transition-opacity duration-300\" x-transition:leave=\"transition-opacity duration-300\" class=\"absolute inset-0 z-10 flex items-center justify-center pt-10 bg-white/50 backdrop-blur-sm\">\n <i data-lucide=\"loader-circle\" class=\"h-6 w-6 animate-spin text-stone-400\"></i>\n </div>\n \n <!-- Views -->\n <div :class=\"{'opacity-50 blur-sm pointer-events-none': isLoading && view !== 'list'}\">\n <!-- Posts List -->\n <div x-show=\"view === 'list'\" x-transition>\n <div class=\"space-y-2\">\n <template x-for=\"post in posts\" :key=\"post.id\">\n <a href=\"#\" @click.prevent=\"selectPost(post.id)\" class=\"block p-3 rounded-lg hover:bg-stone-100 transition-colors duration-200\">\n <p class=\"font-semibold text-stone-900 truncate\" x-text=\"post.title\"></p>\n <div class=\"flex items-center justify-between mt-1.5\">\n <p class=\"text-xs text-stone-500\">By <span class=\"font-medium\" x-text=\"post.author_name\"></span> · <span x-text=\"formatDate(post.created_at)\"></span></p>\n <div class=\"flex items-center gap-1 text-xs text-stone-500\">\n <i data-lucide=\"message-square\" class=\"h-3.5 w-3.5\"></i>\n <span x-text=\"post.comment_count\"></span>\n </div>\n </div>\n </a>\n </template>\n </div>\n </div>\n\n <!-- New Post Form -->\n <div x-show=\"view === 'new_post'\" x-transition class=\"p-1\">\n <form @submit.prevent=\"submitPost()\" class=\"space-y-3\">\n <input type=\"text\" x-model=\"newPost.title\" placeholder=\"Post Title\" required class=\"w-full px-3 py-2 text-[14px] leading-6 bg-white border border-stone-200 rounded-2xl transition focus:outline-none focus:ring-2 focus:ring-black\">\n <textarea x-model=\"newPost.content\" placeholder=\"What's on your mind?\" required rows=\"8\" @input=\"autoResize($el)\" class=\"w-full px-3 py-2 text-[14px] leading-6 bg-white border border-stone-200 rounded-2xl transition resize-none focus:outline-none focus:ring-2 focus:ring-black\"></textarea>\n <div class=\"flex justify-end gap-2 pt-1\">\n <button type=\"button\" @click=\"goToList()\" :disabled=\"submitting\" class=\"px-4 py-2 font-semibold bg-stone-200 text-stone-800 rounded-2xl hover:bg-stone-300 transition-colors disabled:opacity-50\">Cancel</button>\n <button type=\"submit\" :disabled=\"submitting\" class=\"px-4 py-2 font-semibold bg-black text-white rounded-2xl hover:bg-black/90 transition-colors focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-black focus-visible:ring-offset-2 disabled:opacity-50 disabled:cursor-not-allowed\">\n <span x-show=\"!submitting\">Post</span>\n <span x-show=\"submitting\">Posting...</span>\n </button>\n </div>\n </form>\n </div>\n\n <!-- Single Post & Comments -->\n <div x-show=\"view === 'post' && currentPost\" x-transition class=\"space-y-6\">\n <article class=\"p-0\">\n <div class=\"markdown-body\" data-post-content x-html=\"render(currentPost.content)\"></div>\n </article>\n <section class=\"space-y-4\">\n <h2 class=\"text-lg font-bold text-stone-900 border-b border-stone-200 pb-2\">Comments</h2>\n <div x-show=\"!isLoading && currentPost.comments.length === 0\" class=\"text-sm text-stone-500 text-center py-4\">No comments yet.</div>\n <div class=\"space-y-3\">\n <template x-for=\"comment in currentPost.comments\" :key=\"comment.id\">\n <div class=\"p-3 bg-stone-50 rounded-lg\">\n <p class=\"text-xs text-stone-500 mb-1.5\"><span class=\"font-semibold text-stone-700\" x-text=\"comment.author_name\"></span> · <span x-text=\"formatDate(comment.created_at)\"></span></p>\n <div class=\"markdown-body bg-transparent\" :data-comment-id=\"comment.id\" x-html=\"render(comment.content)\"></div>\n </div>\n </template>\n </div>\n <form @submit.prevent=\"submitComment()\" class=\"pt-4 flex items-start gap-2\">\n <textarea x-model=\"newComment\" placeholder=\"Add a comment...\" required rows=\"1\" @input=\"autoResize($el)\" class=\"flex-1 resize-none rounded-2xl border border-stone-200 bg-white px-3 py-2 text-[14px] leading-6 placeholder:text-stone-400 focus:outline-none focus:ring-2 focus:ring-black\"></textarea>\n <button type=\"submit\" aria-label=\"Reply\" :disabled=\"submitting\" class=\"shrink-0 rounded-2xl bg-black text-white h-10 w-10 inline-flex items-center justify-center shadow-sm hover:bg-black/90 active:scale-[.98] transition-transform disabled:opacity-50\">\n <i data-lucide=\"send\" class=\"h-5 w-5\"></i>\n </button>\n </form>\n </section>\n </div>\n </div>\n </main>\n</div>\n<script>\nfunction suneForums() {\n const S_KEY = window.SUNE.id+'-forums', L_KEY = S_KEY+'-lock', L_TIMEOUT=30000, D1_API='https://d1p.awww.workers.dev';\n return {\n view:'list', posts:[], currentPost:null, isLoading:!0, error:'', submitting:!1,\n newPost:{title:'', content:''}, newComment:'',\n md:window.markdownit({html:!1, breaks:!0, linkify:!0}),\n init() {\n const lock=JSON.parse(localStorage.getItem(L_KEY));if(lock&&Date.now()-lock.ts>L_TIMEOUT)localStorage.removeItem(L_KEY);\n const c=JSON.parse(localStorage.getItem(S_KEY))||{};this.newPost=c.newPost||{title:'',content:''};this.newComment=c.newComment||'';\n this.$watch('newPost',()=>this.saveState());this.$watch('newComment',()=>this.saveState());\n this.$watch('currentPost',p=>{if(p&&this.view==='post')this.$nextTick(()=>this.enhanceAllMarkdown())});\n this.fetchPosts()\n },\n isLocked(){if(this.submitting)return!0;const lock=JSON.parse(localStorage.getItem(L_KEY));return lock&&Date.now()-lock.ts<L_TIMEOUT},\n setLock(){localStorage.setItem(L_KEY,JSON.stringify({ts:Date.now()}));this.submitting=!0},\n clearLock(){localStorage.removeItem(L_KEY);this.submitting=!1},\n saveState(){localStorage.setItem(S_KEY,JSON.stringify({newPost:this.newPost,newComment:this.newComment}))},\n async query(sql, p=[]){\n this.isLoading=!0; this.error='';\n try {\n const r=await fetch(D1_API,{method:'POST',headers:{'Content-Type':'application/json'},body:JSON.stringify({query:sql,params:p})});\n if(!r.ok){const d=await r.json().catch(()=>null);throw new Error(d?.error||`HTTP Error ${r.status}`)}\n const d=await r.json();if(d.error)throw new Error(d.details||d.error);return d.results\n } catch(e){this.error=e.message;console.error(e);return null} finally {this.isLoading=!1}\n },\n async fetchPosts(){\n const sql=\"SELECT p.id, p.author_name, p.title, p.created_at, (SELECT COUNT(*) FROM comments c WHERE c.post_id = p.id) as comment_count FROM posts p ORDER BY p.created_at DESC\";\n const r=await this.query(sql);\n if(r)this.posts=r\n },\n async selectPost(id){\n this.view='post'; this.currentPost=null;\n const[p,c]=await Promise.all([this.query(\"SELECT * FROM posts WHERE id = ?\", [id]),this.query(\"SELECT * FROM comments WHERE post_id = ? ORDER BY created_at ASC\", [id])]);\n if(p&&p.length)this.currentPost={...p[0],comments:c||[]}\n },\n async submitPost(){\n if(this.isLocked())return;const{title:t,content:c}=this.newPost;if(!t.trim()||!c.trim())return;\n this.setLock();\n try{\n await this.query(\"INSERT INTO posts(author_name,title,content)VALUES(?,?,?)\",[window.USER?.name||'Anonymous',t.trim(),c.trim()]);\n if(!this.error){this.newPost={title:'',content:''};this.goToList()}\n }finally{this.clearLock()}\n },\n async submitComment(){\n if(this.isLocked())return;const c=this.newComment.trim();if(!c||!this.currentPost)return;\n this.setLock();\n try{\n await this.query(\"INSERT INTO comments(post_id,author_name,content)VALUES(?,?,?)\",[this.currentPost.id,window.USER?.name||'Anonymous',c]);\n if(!this.error){this.newComment='';await this.selectPost(this.currentPost.id)}\n }finally{this.clearLock()}\n },\n goToList(){this.view='list';this.currentPost=null;this.error='';this.fetchPosts()},\n formatDate(s){if(!s)return'';try{return new Date(s.replace(' ','T')+'Z').toLocaleDateString(undefined,{month:'short',day:'numeric'})}catch{return s}},\n render(t){return this.md.render(t||'')},\n autoResize(el){el.style.height='auto';el.style.height=(el.scrollHeight+2)+'px'},\n enhanceAllMarkdown(){\n if(!window.enhanceCodeBlocks)return;\n const postEl=this.$root.querySelector('[data-post-content]');if(postEl)window.enhanceCodeBlocks(postEl,!0);\n this.$root.querySelectorAll('[data-comment-id]').forEach(el=>window.enhanceCodeBlocks(el,!0))\n }\n }\n}\n</script>\n","extension_html":"<sune src='https://raw.githubusercontent.com/sune-org/store/refs/heads/main/sync.sune' private></sune>","hide_composer":true,"include_thoughts":false,"json_output":false,"ignore_master_prompt":false,"json_schema":""},"storage":{}}] |