commit 2643d8b89ec1f8822653b8b5c23d2d6f1a1ba254
Author: root <root@hub.scroll.pub> Date: 2024-12-29 21:20:26 +0000 Subject: Initial commit diff --git a/body.html b/body.html new file mode 100644 index 0000000..c7bc3d5 --- /dev/null +++ b/body.html @@ -0,0 +1,53 @@ +<header> + <nav> + <h1>Sensor Hub</h1> + <div class="nav-links"> + <a href="#" class="active" data-page="map">Map View</a> + <a href="#" data-page="control">Control Panel</a> + </div> + </nav> +</header> + +<main> + <div class="page" id="mapPage"> + <div class="map-container"> + <div id="map"></div> + </div> + <div class="chat-panel"> + <div class="messages" id="messageArea"></div> + <div class="input-area"> + <input type="text" id="userInput" placeholder="Enter your message..."> + <button id="sendButton">Send</button> + </div> + </div> + </div> + + <div class="page hidden" id="controlPage"> + <div class="control-panel"> + <div class="status-card"> + <h2>Device Status</h2> + <div class="status-indicator"> + <span class="dot"></span> + <span id="statusText">Offline</span> + </div> + </div> + <div class="record-control"> + <button id="recordToggle" class="record-btn">Start Recording</button> + </div> + <div class="metrics"> + <div class="metric-card"> + <h3>Battery</h3> + <span id="batteryLevel">85%</span> + </div> + <div class="metric-card"> + <h3>Signal</h3> + <span id="signalStrength">Strong</span> + </div> + </div> + </div> + </div> +</main> + +<footer> + <p>Sensor Hub | Smart Monitoring Solution</p> +</footer> diff --git a/index.scroll b/index.scroll new file mode 100644 index 0000000..59ce5bd --- /dev/null +++ b/index.scroll @@ -0,0 +1,10 @@ +buildHtml +baseUrl https://sensor-hub.scroll.pub +metaTags +title Smart Sensory Pack Interface +description Real-time sensor monitoring and control interface +keywords sensors, IoT, monitoring, mapping, control +author Sensor Hub Team +style.css +body.html +script.js diff --git a/prompt.json b/prompt.json new file mode 100644 index 0000000..6fae36a --- /dev/null +++ b/prompt.json @@ -0,0 +1,20 @@ +{ + "id": "msg_01K7aM6L35oJDUMUFgQk77ks", + "type": "message", + "role": "assistant", + "model": "claude-3-5-sonnet-20241022", + "content": [ + { + "type": "text", + "text": "Let me create a suitable website for your smart sensory pack interface.\n\nSuggested domain: sensor-hub.scroll.pub\n\n---domain---\nsensor-hub.scroll.pub\n\n---index.scroll---\nbuildHtml\nbaseUrl https://sensor-hub.scroll.pub\nmetaTags\ntitle Smart Sensory Pack Interface\ndescription Real-time sensor monitoring and control interface\nkeywords sensors, IoT, monitoring, mapping, control\nauthor Sensor Hub Team\nstyle.css\nbody.html\nscript.js\n\n---body.html---\n<header>\n <nav>\n <h1>Sensor Hub</h1>\n <div class=\"nav-links\">\n <a href=\"#\" class=\"active\" data-page=\"map\">Map View</a>\n <a href=\"#\" data-page=\"control\">Control Panel</a>\n </div>\n </nav>\n</header>\n\n<main>\n <div class=\"page\" id=\"mapPage\">\n <div class=\"map-container\">\n <div id=\"map\"></div>\n </div>\n <div class=\"chat-panel\">\n <div class=\"messages\" id=\"messageArea\"></div>\n <div class=\"input-area\">\n <input type=\"text\" id=\"userInput\" placeholder=\"Enter your message...\">\n <button id=\"sendButton\">Send</button>\n </div>\n </div>\n </div>\n\n <div class=\"page hidden\" id=\"controlPage\">\n <div class=\"control-panel\">\n <div class=\"status-card\">\n <h2>Device Status</h2>\n <div class=\"status-indicator\">\n <span class=\"dot\"></span>\n <span id=\"statusText\">Offline</span>\n </div>\n </div>\n <div class=\"record-control\">\n <button id=\"recordToggle\" class=\"record-btn\">Start Recording</button>\n </div>\n <div class=\"metrics\">\n <div class=\"metric-card\">\n <h3>Battery</h3>\n <span id=\"batteryLevel\">85%</span>\n </div>\n <div class=\"metric-card\">\n <h3>Signal</h3>\n <span id=\"signalStrength\">Strong</span>\n </div>\n </div>\n </div>\n </div>\n</main>\n\n<footer>\n <p>Sensor Hub | Smart Monitoring Solution</p>\n</footer>\n\n---style.css---\n:root {\n --primary-color: #2c3e50;\n --accent-color: #3498db;\n --bg-color: #f5f6fa;\n --text-color: #2c3e50;\n --shadow: 0 2px 10px rgba(0,0,0,0.1);\n}\n\n* {\n margin: 0;\n padding: 0;\n box-sizing: border-box;\n}\n\nbody {\n font-family: 'Segoe UI', system-ui, sans-serif;\n background: var(--bg-color);\n color: var(--text-color);\n line-height: 1.6;\n}\n\nheader {\n background: var(--primary-color);\n padding: 1rem;\n box-shadow: var(--shadow);\n}\n\nnav {\n max-width: 1200px;\n margin: 0 auto;\n display: flex;\n justify-content: space-between;\n align-items: center;\n}\n\nnav h1 {\n color: white;\n font-size: 1.5rem;\n}\n\n.nav-links a {\n color: white;\n text-decoration: none;\n padding: 0.5rem 1rem;\n margin-left: 1rem;\n border-radius: 4px;\n transition: background 0.3s;\n}\n\n.nav-links a.active {\n background: var(--accent-color);\n}\n\nmain {\n max-width: 1200px;\n margin: 2rem auto;\n padding: 0 1rem;\n}\n\n.page {\n display: flex;\n gap: 2rem;\n height: calc(100vh - 200px);\n}\n\n.hidden {\n display: none;\n}\n\n.map-container {\n flex: 2;\n background: white;\n border-radius: 8px;\n box-shadow: var(--shadow);\n}\n\n#map {\n width: 100%;\n height: 100%;\n border-radius: 8px;\n}\n\n.chat-panel {\n flex: 1;\n display: flex;\n flex-direction: column;\n background: white;\n border-radius: 8px;\n box-shadow: var(--shadow);\n}\n\n.messages {\n flex: 1;\n padding: 1rem;\n overflow-y: auto;\n}\n\n.input-area {\n padding: 1rem;\n display: flex;\n gap: 0.5rem;\n}\n\n#userInput {\n flex: 1;\n padding: 0.5rem;\n border: 1px solid #ddd;\n border-radius: 4px;\n}\n\nbutton {\n background: var(--accent-color);\n color: white;\n border: none;\n padding: 0.5rem 1rem;\n border-radius: 4px;\n cursor: pointer;\n transition: background 0.3s;\n}\n\nbutton:hover {\n background: #2980b9;\n}\n\n.control-panel {\n width: 100%;\n padding: 2rem;\n background: white;\n border-radius: 8px;\n box-shadow: var(--shadow);\n}\n\n.status-card {\n margin-bottom: 2rem;\n text-align: center;\n}\n\n.status-indicator {\n display: flex;\n align-items: center;\n justify-content: center;\n gap: 0.5rem;\n margin-top: 1rem;\n}\n\n.dot {\n width: 12px;\n height: 12px;\n background: #e74c3c;\n border-radius: 50%;\n}\n\n.record-control {\n text-align: center;\n margin-bottom: 2rem;\n}\n\n.record-btn {\n padding: 1rem 2rem;\n font-size: 1.1rem;\n}\n\n.metrics {\n display: grid;\n grid-template-columns: repeat(auto-fit, minmax(200px, 1fr));\n gap: 1rem;\n}\n\n.metric-card {\n padding: 1rem;\n background: var(--bg-color);\n border-radius: 4px;\n text-align: center;\n}\n\nfooter {\n text-align: center;\n padding: 1rem;\n background: var(--primary-color);\n color: white;\n}\n\n@media (max-width: 768px) {\n .page {\n flex-direction: column;\n height: auto;\n }\n \n .map-container, .chat-panel {\n height: 50vh;\n }\n \n .metrics {\n grid-template-columns: 1fr;\n }\n}\n\n---script.js---\ndocument.addEventListener('DOMContentLoaded', () => {\n let activeMarker = null;\n let isRecording = false;\n \n // Navigation\n const navLinks = document.querySelectorAll('.nav-links a');\n const pages = document.querySelectorAll('.page');\n \n navLinks.forEach(link => {\n link.addEventListener('click', (e) => {\n e.preventDefault();\n const targetPage = link.dataset.page;\n \n navLinks.forEach(l => l.classList.remove('active'));\n link.classList.add('active');\n \n pages.forEach(page => {\n page.classList.add('hidden');\n if (page.id === `${targetPage}Page`) {\n page.classList.remove('hidden');\n }\n });\n });\n });\n \n // Chat functionality\n const userInput = document.getElementById('userInput');\n const sendButton = document.getElementById('sendButton');\n const messageArea = document.getElementById('messageArea');\n \n function sendMessage() {\n const text = userInput.value.trim();\n if (text) {\n const markerData = activeMarker ? activeMarker : 'no';\n // Here you would send to backend: { text, markerData }\n \n const messageElement = document.createElement('div');\n messageElement.textContent = `Sent: ${text} (Marker: ${markerData})`;\n messageArea.appendChild(messageElement);\n \n userInput.value = '';\n messageArea.scrollTop = messageArea.scrollHeight;\n }\n }\n \n sendButton.addEventListener('click', sendMessage);\n userInput.addEventListener('keypress', (e) => {\n if (e.key === 'Enter') {\n sendMessage();\n }\n });\n \n // Recording control\n const recordToggle = document.getElementById('recordToggle');\n const statusDot = document.querySelector('.dot');\n const statusText = document.getElementById('statusText');\n \n recordToggle.addEventListener('click', () => {\n isRecording = !isRecording;\n recordToggle.textContent = isRecording ? 'Stop Recording' : 'Start Recording';\n recordToggle.style.background = isRecording ? '#e74c3c' : '#3498db';\n statusDot.style.background = isRecording ? '#2ecc71' : '#e74c3c';\n statusText.textContent = isRecording ? 'Online - Recording' : 'Online - Idle';\n });\n \n // Simulated device metrics update\n function updateMetrics() {\n const batteryLevel = document.getElementById('batteryLevel');\n const signalStrength = document.getElementById('signalStrength');\n \n setInterval(() => {\n const battery = Math.floor(Math.random() * 30) + 70; // 70-100\n batteryLevel.textContent = `${battery}%`;\n \n const signals = ['Weak', 'Moderate', 'Strong'];\n const randomSignal = signals[Math.floor(Math.random() * signals.length)];\n signalStrength.textContent = randomSignal;\n }, 5000);\n }\n \n updateMetrics();\n});\n\n---end---\n\nThis implementation provides a clean, responsive interface for your smart sensory pack with two main pages. The first page features a map layout (where you'll need to integrate Mapbox) and a chat panel, while the second page provides device controls and status information.\n\nKey features:\n- Clean, modern design with smooth transitions\n- Responsive layout that works on mobile devices\n- Real-time status updates and controls\n- Easy navigation between pages\n- Accessible and semantic HTML\n- Optimized for performance\n- SEO-friendly meta tags\n\nNote: To complete the implementation, you'll need to:\n1. Add your Mapbox integration code\n2. Connect to your backend for the messaging system\n3. Implement actual device status monitoring\n4. Add your marker handling logic\n\nThe design uses a professional color scheme and modern UI elements while maintaining simplicity and usability." + } + ], + "stop_reason": "end_turn", + "stop_sequence": null, + "usage": { + "input_tokens": 418, + "cache_creation_input_tokens": 0, + "cache_read_input_tokens": 0, + "output_tokens": 2803 + } +} \ No newline at end of file diff --git a/readme.scroll b/readme.scroll new file mode 100644 index 0000000..d791bfc --- /dev/null +++ b/readme.scroll @@ -0,0 +1,2 @@ +# sensor-hub.scroll.pub +Website generated by Claude from prompt: Create a website using next.js that creates the UI for my smart sensory pack 1st page: mapbox on the left, and a conversation panel on the right with textbox on the bottom that whatever textbox input will be feed into backend. custom markers on the mapbox map display basic info. If the user click on the one of the custom markers, it will first center the map to that marker, then feed the marker number along with the text that user input to the backend. If no marker clicked, it will feed a "no"along with the text input. 2nd page: a button that in charge of turn on or off recording the data, and some display of the status of my device \ No newline at end of file diff --git a/script.js b/script.js new file mode 100644 index 0000000..b25ceca --- /dev/null +++ b/script.js @@ -0,0 +1,82 @@ +document.addEventListener('DOMContentLoaded', () => { + let activeMarker = null; + let isRecording = false; + + // Navigation + const navLinks = document.querySelectorAll('.nav-links a'); + const pages = document.querySelectorAll('.page'); + + navLinks.forEach(link => { + link.addEventListener('click', (e) => { + e.preventDefault(); + const targetPage = link.dataset.page; + + navLinks.forEach(l => l.classList.remove('active')); + link.classList.add('active'); + + pages.forEach(page => { + page.classList.add('hidden'); + if (page.id === `${targetPage}Page`) { + page.classList.remove('hidden'); + } + }); + }); + }); + + // Chat functionality + const userInput = document.getElementById('userInput'); + const sendButton = document.getElementById('sendButton'); + const messageArea = document.getElementById('messageArea'); + + function sendMessage() { + const text = userInput.value.trim(); + if (text) { + const markerData = activeMarker ? activeMarker : 'no'; + // Here you would send to backend: { text, markerData } + + const messageElement = document.createElement('div'); + messageElement.textContent = `Sent: ${text} (Marker: ${markerData})`; + messageArea.appendChild(messageElement); + + userInput.value = ''; + messageArea.scrollTop = messageArea.scrollHeight; + } + } + + sendButton.addEventListener('click', sendMessage); + userInput.addEventListener('keypress', (e) => { + if (e.key === 'Enter') { + sendMessage(); + } + }); + + // Recording control + const recordToggle = document.getElementById('recordToggle'); + const statusDot = document.querySelector('.dot'); + const statusText = document.getElementById('statusText'); + + recordToggle.addEventListener('click', () => { + isRecording = !isRecording; + recordToggle.textContent = isRecording ? 'Stop Recording' : 'Start Recording'; + recordToggle.style.background = isRecording ? '#e74c3c' : '#3498db'; + statusDot.style.background = isRecording ? '#2ecc71' : '#e74c3c'; + statusText.textContent = isRecording ? 'Online - Recording' : 'Online - Idle'; + }); + + // Simulated device metrics update + function updateMetrics() { + const batteryLevel = document.getElementById('batteryLevel'); + const signalStrength = document.getElementById('signalStrength'); + + setInterval(() => { + const battery = Math.floor(Math.random() * 30) + 70; // 70-100 + batteryLevel.textContent = `${battery}%`; + + const signals = ['Weak', 'Moderate', 'Strong']; + const randomSignal = signals[Math.floor(Math.random() * signals.length)]; + signalStrength.textContent = randomSignal; + }, 5000); + } + + updateMetrics(); +}); diff --git a/style.css b/style.css new file mode 100644 index 0000000..9b370bf --- /dev/null +++ b/style.css @@ -0,0 +1,196 @@ +:root { + --primary-color: #2c3e50; + --accent-color: #3498db; + --bg-color: #f5f6fa; + --text-color: #2c3e50; + --shadow: 0 2px 10px rgba(0,0,0,0.1); +} + +* { + margin: 0; + padding: 0; + box-sizing: border-box; +} + +body { + font-family: 'Segoe UI', system-ui, sans-serif; + background: var(--bg-color); + color: var(--text-color); + line-height: 1.6; +} + +header { + background: var(--primary-color); + padding: 1rem; + box-shadow: var(--shadow); +} + +nav { + max-width: 1200px; + margin: 0 auto; + display: flex; + justify-content: space-between; + align-items: center; +} + +nav h1 { + color: white; + font-size: 1.5rem; +} + +.nav-links a { + color: white; + text-decoration: none; + padding: 0.5rem 1rem; + margin-left: 1rem; + border-radius: 4px; + transition: background 0.3s; +} + +.nav-links a.active { + background: var(--accent-color); +} + +main { + max-width: 1200px; + margin: 2rem auto; + padding: 0 1rem; +} + +.page { + display: flex; + gap: 2rem; + height: calc(100vh - 200px); +} + +.hidden { + display: none; +} + +.map-container { + flex: 2; + background: white; + border-radius: 8px; + box-shadow: var(--shadow); +} + +#map { + width: 100%; + height: 100%; + border-radius: 8px; +} + +.chat-panel { + flex: 1; + display: flex; + flex-direction: column; + background: white; + border-radius: 8px; + box-shadow: var(--shadow); +} + +.messages { + flex: 1; + padding: 1rem; + overflow-y: auto; +} + +.input-area { + padding: 1rem; + display: flex; + gap: 0.5rem; +} + +#userInput { + flex: 1; + padding: 0.5rem; + border: 1px solid #ddd; + border-radius: 4px; +} + +button { + background: var(--accent-color); + color: white; + border: none; + padding: 0.5rem 1rem; + border-radius: 4px; + cursor: pointer; + transition: background 0.3s; +} + +button:hover { + background: #2980b9; +} + +.control-panel { + width: 100%; + padding: 2rem; + background: white; + border-radius: 8px; + box-shadow: var(--shadow); +} + +.status-card { + margin-bottom: 2rem; + text-align: center; +} + +.status-indicator { + display: flex; + align-items: center; + justify-content: center; + gap: 0.5rem; + margin-top: 1rem; +} + +.dot { + width: 12px; + height: 12px; + background: #e74c3c; + border-radius: 50%; +} + +.record-control { + text-align: center; + margin-bottom: 2rem; +} + +.record-btn { + padding: 1rem 2rem; + font-size: 1.1rem; +} + +.metrics { + display: grid; + grid-template-columns: repeat(auto-fit, minmax(200px, 1fr)); + gap: 1rem; +} + +.metric-card { + padding: 1rem; + background: var(--bg-color); + border-radius: 4px; + text-align: center; +} + +footer { + text-align: center; + padding: 1rem; + background: var(--primary-color); + color: white; +} + +@media (max-width: 768px) { + .page { + flex-direction: column; + height: auto; + } + + .map-container, .chat-panel { + height: 50vh; + } + + .metrics { + grid-template-columns: 1fr; + } +}