commit 8ddef932ac8f7ebeebf34e2cbfb92a951dc15d94
Author: root <root@hub.scroll.pub> Date: 2025-01-10 19:39:35 +0000 Subject: Initial commit diff --git a/body.html b/body.html new file mode 100644 index 0000000..209da1d --- /dev/null +++ b/body.html @@ -0,0 +1,59 @@ +<header> + <h1>SQL Genie</h1> + <p class="tagline">Transform your data requests into perfect SQL queries</p> +</header> + +<main> + <section id="metadata-section"> + <h2>Database Structure</h2> + <div id="metadata-display" class="card"> + <p class="info-text">Loading database metadata...</p> + </div> + </section> + + <section id="selection-section"> + <h2>Select Tables & Columns</h2> + <div class="card"> + <div id="tables-container"> + <h3>Tables</h3> + <div id="table-list" class="checkbox-list"></div> + </div> + <div id="columns-container"> + <h3>Columns</h3> + <div id="column-list" class="checkbox-list"></div> + </div> + </div> + </section> + + <section id="transformation-section"> + <h2>Data Transformations</h2> + <div class="card"> + <div class="transformation-options"> + <button class="transform-btn" data-type="sort">Sort</button> + <button class="transform-btn" data-type="filter">Filter</button> + <button class="transform-btn" data-type="group">Group By</button> + <button class="transform-btn" data-type="aggregate">Aggregate</button> + </div> + <div id="transformation-config"></div> + </div> + </section> + + <section id="custom-section"> + <h2>Custom Request</h2> + <div class="card"> + <textarea id="custom-input" placeholder="Describe your query requirements in plain English..."></textarea> + </div> + </section> + + <section id="result-section"> + <h2>Generated Query</h2> + <div class="card"> + <pre id="query-result"><code>Your SQL query will appear here...</code></pre> + <button id="copy-btn">Copy Query</button> + </div> + </section> +</main> + +<footer> + <p>SQL Genie - Making database queries magical</p> +</footer> diff --git a/index.scroll b/index.scroll new file mode 100644 index 0000000..c746097 --- /dev/null +++ b/index.scroll @@ -0,0 +1,8 @@ +buildHtml +baseUrl https://sqlgenie.scroll.pub +metaTags +editButton /edit.html +title SQL Genie - Intelligent Query Builder +style.css +body.html +script.js diff --git a/prompt.json b/prompt.json new file mode 100644 index 0000000..c140f01 --- /dev/null +++ b/prompt.json @@ -0,0 +1,20 @@ +{ + "id": "msg_016WXsHnwiYSoY1nzSJMMLKk", + "type": "message", + "role": "assistant", + "model": "claude-3-5-sonnet-20241022", + "content": [ + { + "type": "text", + "text": "Let me create a website for your SQL query builder app with GPT integration.\n\n---domain---\nsqlgenie.scroll.pub\n\n---index.scroll---\nbuildHtml\nbaseUrl https://sqlgenie.scroll.pub\nmetaTags\neditButton /edit.html\ntitle SQL Genie - Intelligent Query Builder\nstyle.css\nbody.html\nscript.js\n\n---body.html---\n<header>\n <h1>SQL Genie</h1>\n <p class=\"tagline\">Transform your data requests into perfect SQL queries</p>\n</header>\n\n<main>\n <section id=\"metadata-section\">\n <h2>Database Structure</h2>\n <div id=\"metadata-display\" class=\"card\">\n <p class=\"info-text\">Loading database metadata...</p>\n </div>\n </section>\n\n <section id=\"selection-section\">\n <h2>Select Tables & Columns</h2>\n <div class=\"card\">\n <div id=\"tables-container\">\n <h3>Tables</h3>\n <div id=\"table-list\" class=\"checkbox-list\"></div>\n </div>\n <div id=\"columns-container\">\n <h3>Columns</h3>\n <div id=\"column-list\" class=\"checkbox-list\"></div>\n </div>\n </div>\n </section>\n\n <section id=\"transformation-section\">\n <h2>Data Transformations</h2>\n <div class=\"card\">\n <div class=\"transformation-options\">\n <button class=\"transform-btn\" data-type=\"sort\">Sort</button>\n <button class=\"transform-btn\" data-type=\"filter\">Filter</button>\n <button class=\"transform-btn\" data-type=\"group\">Group By</button>\n <button class=\"transform-btn\" data-type=\"aggregate\">Aggregate</button>\n </div>\n <div id=\"transformation-config\"></div>\n </div>\n </section>\n\n <section id=\"custom-section\">\n <h2>Custom Request</h2>\n <div class=\"card\">\n <textarea id=\"custom-input\" placeholder=\"Describe your query requirements in plain English...\"></textarea>\n </div>\n </section>\n\n <section id=\"result-section\">\n <h2>Generated Query</h2>\n <div class=\"card\">\n <pre id=\"query-result\"><code>Your SQL query will appear here...</code></pre>\n <button id=\"copy-btn\">Copy Query</button>\n </div>\n </section>\n</main>\n\n<footer>\n <p>SQL Genie - Making database queries magical</p>\n</footer>\n\n---style.css---\n:root {\n --primary: #6366f1;\n --secondary: #818cf8;\n --background: #f8fafc;\n --card: #ffffff;\n --text: #1e293b;\n --border: #e2e8f0;\n}\n\n* {\n margin: 0;\n padding: 0;\n box-sizing: border-box;\n}\n\nbody {\n font-family: system-ui, -apple-system, sans-serif;\n background: var(--background);\n color: var(--text);\n line-height: 1.6;\n}\n\nheader {\n background: linear-gradient(135deg, var(--primary), var(--secondary));\n color: white;\n padding: 2rem;\n text-align: center;\n box-shadow: 0 4px 6px -1px rgba(0, 0, 0, 0.1);\n}\n\nh1 {\n font-size: 2.5rem;\n margin-bottom: 0.5rem;\n}\n\n.tagline {\n font-size: 1.1rem;\n opacity: 0.9;\n}\n\nmain {\n max-width: 1200px;\n margin: 2rem auto;\n padding: 0 1rem;\n}\n\nsection {\n margin-bottom: 2rem;\n}\n\nh2 {\n color: var(--primary);\n margin-bottom: 1rem;\n}\n\n.card {\n background: var(--card);\n border-radius: 0.5rem;\n padding: 1.5rem;\n box-shadow: 0 1px 3px rgba(0, 0, 0, 0.1);\n border: 1px solid var(--border);\n}\n\n.checkbox-list {\n display: grid;\n grid-template-columns: repeat(auto-fill, minmax(200px, 1fr));\n gap: 1rem;\n}\n\n.transformation-options {\n display: flex;\n gap: 1rem;\n flex-wrap: wrap;\n margin-bottom: 1rem;\n}\n\n.transform-btn {\n padding: 0.5rem 1rem;\n background: var(--primary);\n color: white;\n border: none;\n border-radius: 0.25rem;\n cursor: pointer;\n transition: opacity 0.2s;\n}\n\n.transform-btn:hover {\n opacity: 0.9;\n}\n\ntextarea {\n width: 100%;\n min-height: 100px;\n padding: 0.75rem;\n border: 1px solid var(--border);\n border-radius: 0.25rem;\n resize: vertical;\n}\n\npre {\n background: #f1f5f9;\n padding: 1rem;\n border-radius: 0.25rem;\n overflow-x: auto;\n}\n\n#copy-btn {\n margin-top: 1rem;\n padding: 0.5rem 1rem;\n background: var(--secondary);\n color: white;\n border: none;\n border-radius: 0.25rem;\n cursor: pointer;\n}\n\nfooter {\n text-align: center;\n padding: 2rem;\n background: var(--primary);\n color: white;\n margin-top: 4rem;\n}\n\n@media (max-width: 768px) {\n header {\n padding: 1.5rem;\n }\n\n h1 {\n font-size: 2rem;\n }\n\n .card {\n padding: 1rem;\n }\n\n .transformation-options {\n flex-direction: column;\n }\n\n .transform-btn {\n width: 100%;\n }\n}\n\n---script.js---\ndocument.addEventListener('DOMContentLoaded', function() {\n // Mock metadata for demonstration\n const mockMetadata = {\n tables: {\n users: ['id', 'name', 'email', 'created_at'],\n orders: ['id', 'user_id', 'total', 'status', 'order_date'],\n products: ['id', 'name', 'price', 'category']\n }\n };\n\n // Initialize the app\n initializeMetadata(mockMetadata);\n setupEventListeners();\n});\n\nfunction initializeMetadata(metadata) {\n const metadataDisplay = document.getElementById('metadata-display');\n const tableList = document.getElementById('table-list');\n \n // Clear loading message\n metadataDisplay.innerHTML = '<h3>Available Tables</h3>';\n \n // Create table selections\n Object.keys(metadata.tables).forEach(tableName => {\n const tableDiv = document.createElement('div');\n tableDiv.innerHTML = `\n <label>\n <input type=\"checkbox\" name=\"table\" value=\"${tableName}\">\n ${tableName}\n </label>\n `;\n tableList.appendChild(tableDiv);\n });\n}\n\nfunction setupEventListeners() {\n // Table selection handler\n document.getElementById('table-list').addEventListener('change', function(e) {\n if (e.target.type === 'checkbox') {\n updateColumnSelection(e.target);\n }\n });\n\n // Transform button handlers\n document.querySelectorAll('.transform-btn').forEach(btn => {\n btn.addEventListener('click', function() {\n handleTransformation(this.dataset.type);\n });\n });\n\n // Copy button handler\n document.getElementById('copy-btn').addEventListener('click', function() {\n const queryText = document.querySelector('#query-result code').textContent;\n navigator.clipboard.writeText(queryText)\n .then(() => alert('Query copied to clipboard!'))\n .catch(err => console.error('Failed to copy:', err));\n });\n\n // Custom input handler\n let timeout;\n document.getElementById('custom-input').addEventListener('input', function() {\n clearTimeout(timeout);\n timeout = setTimeout(() => generateQuery(), 1000);\n });\n}\n\nfunction updateColumnSelection(tableCheckbox) {\n const columnList = document.getElementById('column-list');\n if (tableCheckbox.checked) {\n // Mock columns for demo\n const columns = ['id', 'name', 'email', 'created_at'];\n columns.forEach(col => {\n const colDiv = document.createElement('div');\n colDiv.innerHTML = `\n <label>\n <input type=\"checkbox\" name=\"column\" value=\"${col}\" data-table=\"${tableCheckbox.value}\">\n ${tableCheckbox.value}.${col}\n </label>\n `;\n columnList.appendChild(colDiv);\n });\n } else {\n // Remove columns for unchecked table\n const tableColumns = columnList.querySelectorAll(`[data-table=\"${tableCheckbox.value}\"]`);\n tableColumns.forEach(col => col.parentElement.parentElement.remove());\n }\n}\n\nfunction handleTransformation(type) {\n const configDiv = document.getElementById('transformation-config');\n configDiv.innerHTML = ''; // Clear previous config\n\n switch(type) {\n case 'sort':\n configDiv.innerHTML = `\n <select id=\"sort-column\">\n <option>Select column</option>\n </select>\n <select id=\"sort-direction\">\n <option value=\"ASC\">Ascending</option>\n <option value=\"DESC\">Descending</option>\n </select>\n `;\n break;\n // Add other transformation types as needed\n }\n\n generateQuery();\n}\n\nfunction generateQuery() {\n // Collect all selections and transformations\n const selections = {\n tables: Array.from(document.querySelectorAll('input[name=\"table\"]:checked'))\n .map(cb => cb.value),\n columns: Array.from(document.querySelectorAll('input[name=\"column\"]:checked'))\n .map(cb => `${cb.dataset.table}.${cb.value}`),\n customInput: document.getElementById('custom-input').value\n };\n\n // In a real application, this would call the ChatGPT API\n // For demo, just show a mock query\n const mockQuery = `SELECT ${selections.columns.join(', ')}\nFROM ${selections.tables.join(' JOIN ')}\nWHERE /* conditions based on custom input: ${selections.customInput} */`;\n\n document.querySelector('#query-result code').textContent = mockQuery;\n}\n\n---end---\n\nThis implementation creates a beautiful, responsive SQL query builder interface with the following features:\n\n1. Clean, modern design with smooth gradients and shadows\n2. Responsive layout that works on all devices\n3. Interactive table and column selection\n4. Transformation options with configurable settings\n5. Custom input field for natural language requests\n6. Copy-to-clipboard functionality\n7. Real-time query generation (mock implementation)\n\nThe domain name \"sqlgenie.scroll.pub\" reflects the magical nature of transforming natural language into SQL queries.\n\nTo make this fully functional, you would need to:\n\n1. Connect to a real database to get metadata\n2. Implement the ChatGPT API integration\n3. Add more transformation options\n4. Implement proper error handling\n5. Add input validation\n6. Add loading states\n7. Implement proper security measures\n\nThe current implementation includes mock data and functionality to demonstrate the user interface and interactions." + } + ], + "stop_reason": "end_turn", + "stop_sequence": null, + "usage": { + "input_tokens": 372, + "cache_creation_input_tokens": 0, + "cache_read_input_tokens": 0, + "output_tokens": 2975 + } +} \ No newline at end of file diff --git a/readme.scroll b/readme.scroll new file mode 100644 index 0000000..f4887e2 --- /dev/null +++ b/readme.scroll @@ -0,0 +1,3 @@ +# sqlgenie.scroll.pub + +website generated by Claude from prompt: I want a simple app, that takes a metadata for a database. Assume that this is linked to the webpage data. Then it shows options to select which tables and columns to select. Then it gives options to do various data transformations, and has a text input for custom requests. Then this information is conveyd to chatgpt api, which is tasked to form an SQL query based on the inputs. On the backend side the gpt needs a system prompt to define what it needs to do \ No newline at end of file diff --git a/script.js b/script.js new file mode 100644 index 0000000..64aa617 --- /dev/null +++ b/script.js @@ -0,0 +1,128 @@ +document.addEventListener('DOMContentLoaded', function() { + // Mock metadata for demonstration + const mockMetadata = { + tables: { + users: ['id', 'name', 'email', 'created_at'], + orders: ['id', 'user_id', 'total', 'status', 'order_date'], + products: ['id', 'name', 'price', 'category'] + } + }; + + // Initialize the app + initializeMetadata(mockMetadata); + setupEventListeners(); +}); + +function initializeMetadata(metadata) { + const metadataDisplay = document.getElementById('metadata-display'); + const tableList = document.getElementById('table-list'); + + // Clear loading message + metadataDisplay.innerHTML = '<h3>Available Tables</h3>'; + + // Create table selections + Object.keys(metadata.tables).forEach(tableName => { + const tableDiv = document.createElement('div'); + tableDiv.innerHTML = ` + <label> + <input type="checkbox" name="table" value="${tableName}"> + ${tableName} + </label> + `; + tableList.appendChild(tableDiv); + }); +} + +function setupEventListeners() { + // Table selection handler + document.getElementById('table-list').addEventListener('change', function(e) { + if (e.target.type === 'checkbox') { + updateColumnSelection(e.target); + } + }); + + // Transform button handlers + document.querySelectorAll('.transform-btn').forEach(btn => { + btn.addEventListener('click', function() { + handleTransformation(this.dataset.type); + }); + }); + + // Copy button handler + document.getElementById('copy-btn').addEventListener('click', function() { + const queryText = document.querySelector('#query-result code').textContent; + navigator.clipboard.writeText(queryText) + .then(() => alert('Query copied to clipboard!')) + .catch(err => console.error('Failed to copy:', err)); + }); + + // Custom input handler + let timeout; + document.getElementById('custom-input').addEventListener('input', function() { + clearTimeout(timeout); + timeout = setTimeout(() => generateQuery(), 1000); + }); +} + +function updateColumnSelection(tableCheckbox) { + const columnList = document.getElementById('column-list'); + if (tableCheckbox.checked) { + // Mock columns for demo + const columns = ['id', 'name', 'email', 'created_at']; + columns.forEach(col => { + const colDiv = document.createElement('div'); + colDiv.innerHTML = ` + <label> + <input type="checkbox" name="column" value="${col}" data-table="${tableCheckbox.value}"> + ${tableCheckbox.value}.${col} + </label> + `; + columnList.appendChild(colDiv); + }); + } else { + // Remove columns for unchecked table + const tableColumns = columnList.querySelectorAll(`[data-table="${tableCheckbox.value}"]`); + tableColumns.forEach(col => col.parentElement.parentElement.remove()); + } +} + +function handleTransformation(type) { + const configDiv = document.getElementById('transformation-config'); + configDiv.innerHTML = ''; // Clear previous config + + switch(type) { + case 'sort': + configDiv.innerHTML = ` + <select id="sort-column"> + <option>Select column</option> + </select> + <select id="sort-direction"> + <option value="ASC">Ascending</option> + <option value="DESC">Descending</option> + </select> + `; + break; + // Add other transformation types as needed + } + + generateQuery(); +} + +function generateQuery() { + // Collect all selections and transformations + const selections = { + tables: Array.from(document.querySelectorAll('input[name="table"]:checked')) + .map(cb => cb.value), + columns: Array.from(document.querySelectorAll('input[name="column"]:checked')) + .map(cb => `${cb.dataset.table}.${cb.value}`), + customInput: document.getElementById('custom-input').value + }; + + // In a real application, this would call the ChatGPT API + // For demo, just show a mock query + const mockQuery = `SELECT ${selections.columns.join(', ')} +FROM ${selections.tables.join(' JOIN ')} +WHERE /* conditions based on custom input: ${selections.customInput} */`; + + document.querySelector('#query-result code').textContent = mockQuery; +} diff --git a/style.css b/style.css new file mode 100644 index 0000000..4c3c50a --- /dev/null +++ b/style.css @@ -0,0 +1,145 @@ +:root { + --primary: #6366f1; + --secondary: #818cf8; + --background: #f8fafc; + --card: #ffffff; + --text: #1e293b; + --border: #e2e8f0; +} + +* { + margin: 0; + padding: 0; + box-sizing: border-box; +} + +body { + font-family: system-ui, -apple-system, sans-serif; + background: var(--background); + color: var(--text); + line-height: 1.6; +} + +header { + background: linear-gradient(135deg, var(--primary), var(--secondary)); + color: white; + padding: 2rem; + text-align: center; + box-shadow: 0 4px 6px -1px rgba(0, 0, 0, 0.1); +} + +h1 { + font-size: 2.5rem; + margin-bottom: 0.5rem; +} + +.tagline { + font-size: 1.1rem; + opacity: 0.9; +} + +main { + max-width: 1200px; + margin: 2rem auto; + padding: 0 1rem; +} + +section { + margin-bottom: 2rem; +} + +h2 { + color: var(--primary); + margin-bottom: 1rem; +} + +.card { + background: var(--card); + border-radius: 0.5rem; + padding: 1.5rem; + box-shadow: 0 1px 3px rgba(0, 0, 0, 0.1); + border: 1px solid var(--border); +} + +.checkbox-list { + display: grid; + grid-template-columns: repeat(auto-fill, minmax(200px, 1fr)); + gap: 1rem; +} + +.transformation-options { + display: flex; + gap: 1rem; + flex-wrap: wrap; + margin-bottom: 1rem; +} + +.transform-btn { + padding: 0.5rem 1rem; + background: var(--primary); + color: white; + border: none; + border-radius: 0.25rem; + cursor: pointer; + transition: opacity 0.2s; +} + +.transform-btn:hover { + opacity: 0.9; +} + +textarea { + width: 100%; + min-height: 100px; + padding: 0.75rem; + border: 1px solid var(--border); + border-radius: 0.25rem; + resize: vertical; +} + +pre { + background: #f1f5f9; + padding: 1rem; + border-radius: 0.25rem; + overflow-x: auto; +} + +#copy-btn { + margin-top: 1rem; + padding: 0.5rem 1rem; + background: var(--secondary); + color: white; + border: none; + border-radius: 0.25rem; + cursor: pointer; +} + +footer { + text-align: center; + padding: 2rem; + background: var(--primary); + color: white; + margin-top: 4rem; +} + +@media (max-width: 768px) { + header { + padding: 1.5rem; + } + + h1 { + font-size: 2rem; + } + + .card { + padding: 1rem; + } + + .transformation-options { + flex-direction: column; + } + + .transform-btn { + width: 100%; + } +}