commit 920f892ca74a9983d7e8a7e81c9abf54c629bff7 Author: default Date: Thu Dec 25 07:23:24 2025 +0000 Initial Code Commit diff --git a/.gitignore b/.gitignore new file mode 100644 index 0000000..c0bb667 --- /dev/null +++ b/.gitignore @@ -0,0 +1,26 @@ +node_modules + +# Output +.output +.vercel +.netlify +.wrangler +/.svelte-kit +/build + +# OS +.DS_Store +Thumbs.db + +# Env +.env +.env.* +!.env.example +!.env.test + +# Vite +vite.config.js.timestamp-* +vite.config.ts.timestamp-* + +.vscode/ + diff --git a/.npmrc b/.npmrc new file mode 100644 index 0000000..b6f27f1 --- /dev/null +++ b/.npmrc @@ -0,0 +1 @@ +engine-strict=true diff --git a/.prettierrc b/.prettierrc new file mode 100644 index 0000000..dfcc0e9 --- /dev/null +++ b/.prettierrc @@ -0,0 +1,4 @@ +{ + "tabWidth": 4, + "useTabs": true +} diff --git a/README.md b/README.md new file mode 100644 index 0000000..0cf41be --- /dev/null +++ b/README.md @@ -0,0 +1,198 @@ +# Filaprint + +Filaprint is a modern, premium web application designed to help 3D printing enthusiasts manage their filament inventory, track print jobs, and calculate costs and energy usage. + +## 🛠️ Technology Stack + +- **Framework:** SvelteKit (Svelte 5) +- **Language:** TypeScript +- **Styling:** Tailwind CSS v4 (Cerberus Theme) +- **State Management:** Svelte 5 Runes +- **Build Tool:** Vite +- **Data Visualization:** Chart.js +- **Icons:** Iconify (@iconify/svelte) +- **Database:** MongoDB with Mongoose +- **Authentication:** JWT with bcrypt password hashing + +## ✨ Features + +### 1. Dashboard + +- **Overview Stats:** Active spools, filament on hand, printers, estimated value, and total spent. +- **Recent Activity:** Quick view of the 5 most recent prints with status indicators. +- **Printer Status:** Shows active print job with real-time countdown and progress bar. +- **Browser Notifications:** Get notified when a print job completes. + +### 2. Filament Inventory Management + +- **Spool Tracking:** + - Brand, Material (PLA, PETG, ABS, ASA, TPU, Other), Color (with hex preview). + - Initial Weight vs. Remaining Weight. + - Cost per spool and automatic cost-per-gram calculation. + - Purchase date tracking. +- **Edit/Delete:** Full CRUD operations for spool management. +- **Visual Indicators:** Color preview badges and remaining weight display. + +### 3. Print Job Logging + +- **Log Prints:** + - Link to specific Printer and Filament Spool. + - Duration (minutes) and Weight used (g). + - Calculated Cost (auto-calculated or manual override). + - Status: Success, Fail, Cancelled, **In Progress**. +- **In Progress Tracking:** + - Assign printer and spool to active jobs. + - Specify elapsed time for accurate dashboard countdown. + - Real-time progress display on dashboard. +- **Edit/Delete:** Full CRUD operations for print history. +- **History:** Clickable entries with detailed information. + +### 4. Printer Configuration + +- **Profiles:** Manage multiple printers with custom names. +- **Specs:** Model name, Power consumption (Watts), Nozzle diameter (mm). +- **Configure Button:** Edit or delete printer profiles. + +### 5. Analytics + +- **Daily Filament Usage:** Line chart showing filament consumption over time. +- **Daily Electricity Usage:** Bar chart showing power consumption in kWh. +- **Success Rate:** Visual ring chart with percentage. +- **Material Distribution:** Doughnut chart showing material breakdown. +- **Stats Summary:** Total prints, success rate, total electricity used. + +### 6. User Management + +- **Authentication:** Secure login/registration with JWT tokens. +- **User Settings:** Profile editing and password change. +- **Admin Panel:** Manage users (Admin role only). +- **Role-Based Access:** Admin and User roles with appropriate permissions. + +## 🗂️ Data Models (Mongoose Schemas) + +### User Schema + +- `_id`: ObjectId +- `username`: String (Required, Unique) +- `email`: String +- `password`: String (Hashed with bcrypt) +- `role`: String (Enum: User, Admin) +- `createdAt`: Date + +### Spool Schema + +- `_id`: ObjectId +- `user_id`: ObjectId (Ref: User) +- `brand`: String (Required) +- `material`: String (Required, Enum: PLA, PETG, ABS, ASA, TPU, Other) +- `color_hex`: String (Default: #ffffff) +- `weight_initial_g`: Number (Required) +- `weight_remaining_g`: Number (Required) +- `price`: Number +- `purchased_at`: Date +- `is_active`: Boolean (Default: true) + +### Printer Schema + +- `_id`: ObjectId +- `user_id`: ObjectId (Ref: User) +- `name`: String (Required) +- `model`: String +- `nozzle_diameter_mm`: Number (Default: 0.4) +- `power_consumption_watts`: Number (Default: 0) + +### PrintJob Schema + +- `_id`: ObjectId +- `user_id`: ObjectId (Ref: User) +- `printer_id`: ObjectId (Ref: Printer) +- `spool_id`: ObjectId (Ref: Spool) +- `name`: String +- `duration_minutes`: Number +- `filament_used_g`: Number +- `calculated_cost_filament`: Number +- `status`: String (Enum: Success, Fail, Cancelled, In Progress) +- `started_at`: Date (For In Progress jobs) +- `date`: Date (Default: Date.now) + +## 🚀 Getting Started + +### Prerequisites + +- Node.js 18+ or Bun +- MongoDB instance (local or Atlas) + +### Installation + +```bash +# Clone the repository +git clone https://github.com/yourusername/filaprint.git +cd filaprint + +# Install dependencies +bun install + +# Set up environment variables +cp .env.example .env +# Edit .env with your MongoDB URI and JWT secret + +# Run development server +bun run dev +``` + +### Environment Variables + +```env +MONGODB_URI=mongodb://localhost:27017/filaprint +JWT_SECRET=your-super-secret-jwt-key +``` + +## 📁 Project Structure + +``` +src/ +├── lib/ +│ ├── components/ +│ │ ├── ui/ # Base UI components (Button, Card, Input, Modal) +│ │ ├── prints/ # Print-specific components (LogPrintModal, EditPrintModal) +│ │ └── Navbar.svelte +│ ├── models/ # Mongoose schemas +│ └── server/ # Server utilities (db connection, auth) +├── routes/ +│ ├── admin/users/ # Admin user management +│ ├── analytics/ # Analytics dashboard +│ ├── login/ # Authentication +│ ├── printers/ # Printer management +│ ├── prints/ # Print job logging +│ ├── register/ # User registration +│ ├── settings/ # User settings +│ └── spools/ # Filament inventory +└── app.css # Global styles (Cerberus theme) +``` + +## ✅ Completed Features + +- [x] User authentication (Login/Register) +- [x] Dashboard with live stats and active print tracking +- [x] Spool management (CRUD) +- [x] Printer management (CRUD) +- [x] Print job logging with "In Progress" support +- [x] Cost calculation (auto and manual) +- [x] Filament deduction on print completion +- [x] Analytics with Chart.js (filament, electricity, materials) +- [x] User settings (profile, password change) +- [x] Admin user management panel +- [x] Browser notifications for completed prints +- [x] Iconify icon library integration +- [x] Responsive design + +## 🔮 Future Enhancements + +- [ ] 3D spool visualization with Threlte +- [ ] QR/Barcode scanning for quick spool lookup +- [ ] Photo uploads for print jobs +- [ ] Export data (CSV/PDF reports) +- [ ] Multi-language support +- [ ] Dark/Light theme toggle +- [ ] Email notifications +- [ ] Print job templates diff --git a/bun.lock b/bun.lock new file mode 100644 index 0000000..96dbd55 --- /dev/null +++ b/bun.lock @@ -0,0 +1,636 @@ +{ + "lockfileVersion": 1, + "configVersion": 1, + "workspaces": { + "": { + "name": "filaprint", + "dependencies": { + "@iconify/svelte": "^5.1.0", + "@sveltejs/adapter-node": "^5.4.0", + "@threlte/core": "^8.3.1", + "@threlte/extras": "^9.7.1", + "@types/three": "^0.182.0", + "bcryptjs": "^3.0.3", + "chart.js": "^4.5.1", + "cookie": "^1.1.1", + "cors": "^2.8.5", + "dotenv": "^17.2.3", + "express": "^5.2.1", + "jsonwebtoken": "^9.0.3", + "mongoose": "^9.0.2", + "three": "^0.182.0", + }, + "devDependencies": { + "@sveltejs/adapter-auto": "^7.0.0", + "@sveltejs/kit": "^2.49.1", + "@sveltejs/vite-plugin-svelte": "^6.2.1", + "@tailwindcss/forms": "^0.5.10", + "@tailwindcss/typography": "^0.5.19", + "@tailwindcss/vite": "^4.1.17", + "@types/bcryptjs": "^3.0.0", + "@types/cookie": "^1.0.0", + "@types/jsonwebtoken": "^9.0.10", + "svelte": "^5.45.6", + "svelte-check": "^4.3.4", + "tailwindcss": "^4.1.17", + "typescript": "^5.9.3", + "vite": "^7.2.6", + }, + }, + }, + "packages": { + "@dimforge/rapier3d-compat": ["@dimforge/rapier3d-compat@0.12.0", "", {}, "sha512-uekIGetywIgopfD97oDL5PfeezkFpNhwlzlaEYNOA0N6ghdsOvh/HYjSMek5Q2O1PYvRSDFcqFVJl4r4ZBwOow=="], + + "@esbuild/aix-ppc64": ["@esbuild/aix-ppc64@0.27.2", "", { "os": "aix", "cpu": "ppc64" }, "sha512-GZMB+a0mOMZs4MpDbj8RJp4cw+w1WV5NYD6xzgvzUJ5Ek2jerwfO2eADyI6ExDSUED+1X8aMbegahsJi+8mgpw=="], + + "@esbuild/android-arm": ["@esbuild/android-arm@0.27.2", "", { "os": "android", "cpu": "arm" }, "sha512-DVNI8jlPa7Ujbr1yjU2PfUSRtAUZPG9I1RwW4F4xFB1Imiu2on0ADiI/c3td+KmDtVKNbi+nffGDQMfcIMkwIA=="], + + "@esbuild/android-arm64": ["@esbuild/android-arm64@0.27.2", "", { "os": "android", "cpu": "arm64" }, "sha512-pvz8ZZ7ot/RBphf8fv60ljmaoydPU12VuXHImtAs0XhLLw+EXBi2BLe3OYSBslR4rryHvweW5gmkKFwTiFy6KA=="], + + "@esbuild/android-x64": ["@esbuild/android-x64@0.27.2", "", { "os": "android", "cpu": "x64" }, "sha512-z8Ank4Byh4TJJOh4wpz8g2vDy75zFL0TlZlkUkEwYXuPSgX8yzep596n6mT7905kA9uHZsf/o2OJZubl2l3M7A=="], + + "@esbuild/darwin-arm64": ["@esbuild/darwin-arm64@0.27.2", "", { "os": "darwin", "cpu": "arm64" }, "sha512-davCD2Zc80nzDVRwXTcQP/28fiJbcOwvdolL0sOiOsbwBa72kegmVU0Wrh1MYrbuCL98Omp5dVhQFWRKR2ZAlg=="], + + "@esbuild/darwin-x64": ["@esbuild/darwin-x64@0.27.2", "", { "os": "darwin", "cpu": "x64" }, "sha512-ZxtijOmlQCBWGwbVmwOF/UCzuGIbUkqB1faQRf5akQmxRJ1ujusWsb3CVfk/9iZKr2L5SMU5wPBi1UWbvL+VQA=="], + + "@esbuild/freebsd-arm64": ["@esbuild/freebsd-arm64@0.27.2", "", { "os": "freebsd", "cpu": "arm64" }, "sha512-lS/9CN+rgqQ9czogxlMcBMGd+l8Q3Nj1MFQwBZJyoEKI50XGxwuzznYdwcav6lpOGv5BqaZXqvBSiB/kJ5op+g=="], + + "@esbuild/freebsd-x64": ["@esbuild/freebsd-x64@0.27.2", "", { "os": "freebsd", "cpu": "x64" }, "sha512-tAfqtNYb4YgPnJlEFu4c212HYjQWSO/w/h/lQaBK7RbwGIkBOuNKQI9tqWzx7Wtp7bTPaGC6MJvWI608P3wXYA=="], + + "@esbuild/linux-arm": ["@esbuild/linux-arm@0.27.2", "", { "os": "linux", "cpu": "arm" }, "sha512-vWfq4GaIMP9AIe4yj1ZUW18RDhx6EPQKjwe7n8BbIecFtCQG4CfHGaHuh7fdfq+y3LIA2vGS/o9ZBGVxIDi9hw=="], + + "@esbuild/linux-arm64": ["@esbuild/linux-arm64@0.27.2", "", { "os": "linux", "cpu": "arm64" }, "sha512-hYxN8pr66NsCCiRFkHUAsxylNOcAQaxSSkHMMjcpx0si13t1LHFphxJZUiGwojB1a/Hd5OiPIqDdXONia6bhTw=="], + + "@esbuild/linux-ia32": ["@esbuild/linux-ia32@0.27.2", "", { "os": "linux", "cpu": "ia32" }, "sha512-MJt5BRRSScPDwG2hLelYhAAKh9imjHK5+NE/tvnRLbIqUWa+0E9N4WNMjmp/kXXPHZGqPLxggwVhz7QP8CTR8w=="], + + "@esbuild/linux-loong64": ["@esbuild/linux-loong64@0.27.2", "", { "os": "linux", "cpu": "none" }, "sha512-lugyF1atnAT463aO6KPshVCJK5NgRnU4yb3FUumyVz+cGvZbontBgzeGFO1nF+dPueHD367a2ZXe1NtUkAjOtg=="], + + "@esbuild/linux-mips64el": ["@esbuild/linux-mips64el@0.27.2", "", { "os": "linux", "cpu": "none" }, "sha512-nlP2I6ArEBewvJ2gjrrkESEZkB5mIoaTswuqNFRv/WYd+ATtUpe9Y09RnJvgvdag7he0OWgEZWhviS1OTOKixw=="], + + "@esbuild/linux-ppc64": ["@esbuild/linux-ppc64@0.27.2", "", { "os": "linux", "cpu": "ppc64" }, "sha512-C92gnpey7tUQONqg1n6dKVbx3vphKtTHJaNG2Ok9lGwbZil6DrfyecMsp9CrmXGQJmZ7iiVXvvZH6Ml5hL6XdQ=="], + + "@esbuild/linux-riscv64": ["@esbuild/linux-riscv64@0.27.2", "", { "os": "linux", "cpu": "none" }, "sha512-B5BOmojNtUyN8AXlK0QJyvjEZkWwy/FKvakkTDCziX95AowLZKR6aCDhG7LeF7uMCXEJqwa8Bejz5LTPYm8AvA=="], + + "@esbuild/linux-s390x": ["@esbuild/linux-s390x@0.27.2", "", { "os": "linux", "cpu": "s390x" }, "sha512-p4bm9+wsPwup5Z8f4EpfN63qNagQ47Ua2znaqGH6bqLlmJ4bx97Y9JdqxgGZ6Y8xVTixUnEkoKSHcpRlDnNr5w=="], + + "@esbuild/linux-x64": ["@esbuild/linux-x64@0.27.2", "", { "os": "linux", "cpu": "x64" }, "sha512-uwp2Tip5aPmH+NRUwTcfLb+W32WXjpFejTIOWZFw/v7/KnpCDKG66u4DLcurQpiYTiYwQ9B7KOeMJvLCu/OvbA=="], + + "@esbuild/netbsd-arm64": ["@esbuild/netbsd-arm64@0.27.2", "", { "os": "none", "cpu": "arm64" }, "sha512-Kj6DiBlwXrPsCRDeRvGAUb/LNrBASrfqAIok+xB0LxK8CHqxZ037viF13ugfsIpePH93mX7xfJp97cyDuTZ3cw=="], + + "@esbuild/netbsd-x64": ["@esbuild/netbsd-x64@0.27.2", "", { "os": "none", "cpu": "x64" }, "sha512-HwGDZ0VLVBY3Y+Nw0JexZy9o/nUAWq9MlV7cahpaXKW6TOzfVno3y3/M8Ga8u8Yr7GldLOov27xiCnqRZf0tCA=="], + + "@esbuild/openbsd-arm64": ["@esbuild/openbsd-arm64@0.27.2", "", { "os": "openbsd", "cpu": "arm64" }, "sha512-DNIHH2BPQ5551A7oSHD0CKbwIA/Ox7+78/AWkbS5QoRzaqlev2uFayfSxq68EkonB+IKjiuxBFoV8ESJy8bOHA=="], + + "@esbuild/openbsd-x64": ["@esbuild/openbsd-x64@0.27.2", "", { "os": "openbsd", "cpu": "x64" }, "sha512-/it7w9Nb7+0KFIzjalNJVR5bOzA9Vay+yIPLVHfIQYG/j+j9VTH84aNB8ExGKPU4AzfaEvN9/V4HV+F+vo8OEg=="], + + "@esbuild/openharmony-arm64": ["@esbuild/openharmony-arm64@0.27.2", "", { "os": "none", "cpu": "arm64" }, "sha512-LRBbCmiU51IXfeXk59csuX/aSaToeG7w48nMwA6049Y4J4+VbWALAuXcs+qcD04rHDuSCSRKdmY63sruDS5qag=="], + + "@esbuild/sunos-x64": ["@esbuild/sunos-x64@0.27.2", "", { "os": "sunos", "cpu": "x64" }, "sha512-kMtx1yqJHTmqaqHPAzKCAkDaKsffmXkPHThSfRwZGyuqyIeBvf08KSsYXl+abf5HDAPMJIPnbBfXvP2ZC2TfHg=="], + + "@esbuild/win32-arm64": ["@esbuild/win32-arm64@0.27.2", "", { "os": "win32", "cpu": "arm64" }, "sha512-Yaf78O/B3Kkh+nKABUF++bvJv5Ijoy9AN1ww904rOXZFLWVc5OLOfL56W+C8F9xn5JQZa3UX6m+IktJnIb1Jjg=="], + + "@esbuild/win32-ia32": ["@esbuild/win32-ia32@0.27.2", "", { "os": "win32", "cpu": "ia32" }, "sha512-Iuws0kxo4yusk7sw70Xa2E2imZU5HoixzxfGCdxwBdhiDgt9vX9VUCBhqcwY7/uh//78A1hMkkROMJq9l27oLQ=="], + + "@esbuild/win32-x64": ["@esbuild/win32-x64@0.27.2", "", { "os": "win32", "cpu": "x64" }, "sha512-sRdU18mcKf7F+YgheI/zGf5alZatMUTKj/jNS6l744f9u3WFu4v7twcUI9vu4mknF4Y9aDlblIie0IM+5xxaqQ=="], + + "@iconify/svelte": ["@iconify/svelte@5.1.0", "", { "dependencies": { "@iconify/types": "^2.0.0" }, "peerDependencies": { "svelte": ">4.0.0" } }, "sha512-I14nSqo0pNXO5OKsT61ZO3XIPF4yRHA2ErgPsaZ1sPJdKXn80o7o8jOe1xpWphbb9FihdX6by9zlKKBss61mFw=="], + + "@iconify/types": ["@iconify/types@2.0.0", "", {}, "sha512-+wluvCrRhXrhyOmRDJ3q8mux9JkKy5SJ/v8ol2tu4FVjyYvtEzkc/3pK15ET6RKg4b4w4BmTk1+gsCUhf21Ykg=="], + + "@jridgewell/gen-mapping": ["@jridgewell/gen-mapping@0.3.13", "", { "dependencies": { "@jridgewell/sourcemap-codec": "^1.5.0", "@jridgewell/trace-mapping": "^0.3.24" } }, "sha512-2kkt/7niJ6MgEPxF0bYdQ6etZaA+fQvDcLKckhy1yIQOzaoKjBBjSj63/aLVjYE3qhRt5dvM+uUyfCg6UKCBbA=="], + + "@jridgewell/remapping": ["@jridgewell/remapping@2.3.5", "", { "dependencies": { "@jridgewell/gen-mapping": "^0.3.5", "@jridgewell/trace-mapping": "^0.3.24" } }, "sha512-LI9u/+laYG4Ds1TDKSJW2YPrIlcVYOwi2fUC6xB43lueCjgxV4lffOCZCtYFiH6TNOX+tQKXx97T4IKHbhyHEQ=="], + + "@jridgewell/resolve-uri": ["@jridgewell/resolve-uri@3.1.2", "", {}, "sha512-bRISgCIjP20/tbWSPWMEi54QVPRZExkuD9lJL+UIxUKtwVJA8wW1Trb1jMs1RFXo1CBTNZ/5hpC9QvmKWdopKw=="], + + "@jridgewell/sourcemap-codec": ["@jridgewell/sourcemap-codec@1.5.5", "", {}, "sha512-cYQ9310grqxueWbl+WuIUIaiUaDcj7WOq5fVhEljNVgRfOUhY9fy2zTvfoqWsnebh8Sl70VScFbICvJnLKB0Og=="], + + "@jridgewell/trace-mapping": ["@jridgewell/trace-mapping@0.3.31", "", { "dependencies": { "@jridgewell/resolve-uri": "^3.1.0", "@jridgewell/sourcemap-codec": "^1.4.14" } }, "sha512-zzNR+SdQSDJzc8joaeP8QQoCQr8NuYx2dIIytl1QeBEZHJ9uW6hebsrYgbz8hJwUQao3TWCMtmfV8Nu1twOLAw=="], + + "@kurkle/color": ["@kurkle/color@0.3.4", "", {}, "sha512-M5UknZPHRu3DEDWoipU6sE8PdkZ6Z/S+v4dD+Ke8IaNlpdSQah50lz1KtcFBa2vsdOnwbbnxJwVM4wty6udA5w=="], + + "@mongodb-js/saslprep": ["@mongodb-js/saslprep@1.4.4", "", { "dependencies": { "sparse-bitfield": "^3.0.3" } }, "sha512-p7X/ytJDIdwUfFL/CLOhKgdfJe1Fa8uw9seJYvdOmnP9JBWGWHW69HkOixXS6Wy9yvGf1MbhcS6lVmrhy4jm2g=="], + + "@polka/url": ["@polka/url@1.0.0-next.29", "", {}, "sha512-wwQAWhWSuHaag8c4q/KN/vCoeOJYshAIvMQwD4GpSb3OiZklFfvAgmj0VCBBImRpuF/aFgIRzllXlVX93Jevww=="], + + "@rollup/plugin-commonjs": ["@rollup/plugin-commonjs@28.0.9", "", { "dependencies": { "@rollup/pluginutils": "^5.0.1", "commondir": "^1.0.1", "estree-walker": "^2.0.2", "fdir": "^6.2.0", "is-reference": "1.2.1", "magic-string": "^0.30.3", "picomatch": "^4.0.2" }, "peerDependencies": { "rollup": "^2.68.0||^3.0.0||^4.0.0" }, "optionalPeers": ["rollup"] }, "sha512-PIR4/OHZ79romx0BVVll/PkwWpJ7e5lsqFa3gFfcrFPWwLXLV39JVUzQV9RKjWerE7B845Hqjj9VYlQeieZ2dA=="], + + "@rollup/plugin-json": ["@rollup/plugin-json@6.1.0", "", { "dependencies": { "@rollup/pluginutils": "^5.1.0" }, "peerDependencies": { "rollup": "^1.20.0||^2.0.0||^3.0.0||^4.0.0" }, "optionalPeers": ["rollup"] }, "sha512-EGI2te5ENk1coGeADSIwZ7G2Q8CJS2sF120T7jLw4xFw9n7wIOXHo+kIYRAoVpJAN+kmqZSoO3Fp4JtoNF4ReA=="], + + "@rollup/plugin-node-resolve": ["@rollup/plugin-node-resolve@16.0.3", "", { "dependencies": { "@rollup/pluginutils": "^5.0.1", "@types/resolve": "1.20.2", "deepmerge": "^4.2.2", "is-module": "^1.0.0", "resolve": "^1.22.1" }, "peerDependencies": { "rollup": "^2.78.0||^3.0.0||^4.0.0" }, "optionalPeers": ["rollup"] }, "sha512-lUYM3UBGuM93CnMPG1YocWu7X802BrNF3jW2zny5gQyLQgRFJhV1Sq0Zi74+dh/6NBx1DxFC4b4GXg9wUCG5Qg=="], + + "@rollup/pluginutils": ["@rollup/pluginutils@5.3.0", "", { "dependencies": { "@types/estree": "^1.0.0", "estree-walker": "^2.0.2", "picomatch": "^4.0.2" }, "peerDependencies": { "rollup": "^1.20.0||^2.0.0||^3.0.0||^4.0.0" }, "optionalPeers": ["rollup"] }, "sha512-5EdhGZtnu3V88ces7s53hhfK5KSASnJZv8Lulpc04cWO3REESroJXg73DFsOmgbU2BhwV0E20bu2IDZb3VKW4Q=="], + + "@rollup/rollup-android-arm-eabi": ["@rollup/rollup-android-arm-eabi@4.54.0", "", { "os": "android", "cpu": "arm" }, "sha512-OywsdRHrFvCdvsewAInDKCNyR3laPA2mc9bRYJ6LBp5IyvF3fvXbbNR0bSzHlZVFtn6E0xw2oZlyjg4rKCVcng=="], + + "@rollup/rollup-android-arm64": ["@rollup/rollup-android-arm64@4.54.0", "", { "os": "android", "cpu": "arm64" }, "sha512-Skx39Uv+u7H224Af+bDgNinitlmHyQX1K/atIA32JP3JQw6hVODX5tkbi2zof/E69M1qH2UoN3Xdxgs90mmNYw=="], + + "@rollup/rollup-darwin-arm64": ["@rollup/rollup-darwin-arm64@4.54.0", "", { "os": "darwin", "cpu": "arm64" }, "sha512-k43D4qta/+6Fq+nCDhhv9yP2HdeKeP56QrUUTW7E6PhZP1US6NDqpJj4MY0jBHlJivVJD5P8NxrjuobZBJTCRw=="], + + "@rollup/rollup-darwin-x64": ["@rollup/rollup-darwin-x64@4.54.0", "", { "os": "darwin", "cpu": "x64" }, "sha512-cOo7biqwkpawslEfox5Vs8/qj83M/aZCSSNIWpVzfU2CYHa2G3P1UN5WF01RdTHSgCkri7XOlTdtk17BezlV3A=="], + + "@rollup/rollup-freebsd-arm64": ["@rollup/rollup-freebsd-arm64@4.54.0", "", { "os": "freebsd", "cpu": "arm64" }, "sha512-miSvuFkmvFbgJ1BevMa4CPCFt5MPGw094knM64W9I0giUIMMmRYcGW/JWZDriaw/k1kOBtsWh1z6nIFV1vPNtA=="], + + "@rollup/rollup-freebsd-x64": ["@rollup/rollup-freebsd-x64@4.54.0", "", { "os": "freebsd", "cpu": "x64" }, "sha512-KGXIs55+b/ZfZsq9aR026tmr/+7tq6VG6MsnrvF4H8VhwflTIuYh+LFUlIsRdQSgrgmtM3fVATzEAj4hBQlaqQ=="], + + "@rollup/rollup-linux-arm-gnueabihf": ["@rollup/rollup-linux-arm-gnueabihf@4.54.0", "", { "os": "linux", "cpu": "arm" }, "sha512-EHMUcDwhtdRGlXZsGSIuXSYwD5kOT9NVnx9sqzYiwAc91wfYOE1g1djOEDseZJKKqtHAHGwnGPQu3kytmfaXLQ=="], + + "@rollup/rollup-linux-arm-musleabihf": ["@rollup/rollup-linux-arm-musleabihf@4.54.0", "", { "os": "linux", "cpu": "arm" }, "sha512-+pBrqEjaakN2ySv5RVrj/qLytYhPKEUwk+e3SFU5jTLHIcAtqh2rLrd/OkbNuHJpsBgxsD8ccJt5ga/SeG0JmA=="], + + "@rollup/rollup-linux-arm64-gnu": ["@rollup/rollup-linux-arm64-gnu@4.54.0", "", { "os": "linux", "cpu": "arm64" }, "sha512-NSqc7rE9wuUaRBsBp5ckQ5CVz5aIRKCwsoa6WMF7G01sX3/qHUw/z4pv+D+ahL1EIKy6Enpcnz1RY8pf7bjwng=="], + + "@rollup/rollup-linux-arm64-musl": ["@rollup/rollup-linux-arm64-musl@4.54.0", "", { "os": "linux", "cpu": "arm64" }, "sha512-gr5vDbg3Bakga5kbdpqx81m2n9IX8M6gIMlQQIXiLTNeQW6CucvuInJ91EuCJ/JYvc+rcLLsDFcfAD1K7fMofg=="], + + "@rollup/rollup-linux-loong64-gnu": ["@rollup/rollup-linux-loong64-gnu@4.54.0", "", { "os": "linux", "cpu": "none" }, "sha512-gsrtB1NA3ZYj2vq0Rzkylo9ylCtW/PhpLEivlgWe0bpgtX5+9j9EZa0wtZiCjgu6zmSeZWyI/e2YRX1URozpIw=="], + + "@rollup/rollup-linux-ppc64-gnu": ["@rollup/rollup-linux-ppc64-gnu@4.54.0", "", { "os": "linux", "cpu": "ppc64" }, "sha512-y3qNOfTBStmFNq+t4s7Tmc9hW2ENtPg8FeUD/VShI7rKxNW7O4fFeaYbMsd3tpFlIg1Q8IapFgy7Q9i2BqeBvA=="], + + "@rollup/rollup-linux-riscv64-gnu": ["@rollup/rollup-linux-riscv64-gnu@4.54.0", "", { "os": "linux", "cpu": "none" }, "sha512-89sepv7h2lIVPsFma8iwmccN7Yjjtgz0Rj/Ou6fEqg3HDhpCa+Et+YSufy27i6b0Wav69Qv4WBNl3Rs6pwhebQ=="], + + "@rollup/rollup-linux-riscv64-musl": ["@rollup/rollup-linux-riscv64-musl@4.54.0", "", { "os": "linux", "cpu": "none" }, "sha512-ZcU77ieh0M2Q8Ur7D5X7KvK+UxbXeDHwiOt/CPSBTI1fBmeDMivW0dPkdqkT4rOgDjrDDBUed9x4EgraIKoR2A=="], + + "@rollup/rollup-linux-s390x-gnu": ["@rollup/rollup-linux-s390x-gnu@4.54.0", "", { "os": "linux", "cpu": "s390x" }, "sha512-2AdWy5RdDF5+4YfG/YesGDDtbyJlC9LHmL6rZw6FurBJ5n4vFGupsOBGfwMRjBYH7qRQowT8D/U4LoSvVwOhSQ=="], + + "@rollup/rollup-linux-x64-gnu": ["@rollup/rollup-linux-x64-gnu@4.54.0", "", { "os": "linux", "cpu": "x64" }, "sha512-WGt5J8Ij/rvyqpFexxk3ffKqqbLf9AqrTBbWDk7ApGUzaIs6V+s2s84kAxklFwmMF/vBNGrVdYgbblCOFFezMQ=="], + + "@rollup/rollup-linux-x64-musl": ["@rollup/rollup-linux-x64-musl@4.54.0", "", { "os": "linux", "cpu": "x64" }, "sha512-JzQmb38ATzHjxlPHuTH6tE7ojnMKM2kYNzt44LO/jJi8BpceEC8QuXYA908n8r3CNuG/B3BV8VR3Hi1rYtmPiw=="], + + "@rollup/rollup-openharmony-arm64": ["@rollup/rollup-openharmony-arm64@4.54.0", "", { "os": "none", "cpu": "arm64" }, "sha512-huT3fd0iC7jigGh7n3q/+lfPcXxBi+om/Rs3yiFxjvSxbSB6aohDFXbWvlspaqjeOh+hx7DDHS+5Es5qRkWkZg=="], + + "@rollup/rollup-win32-arm64-msvc": ["@rollup/rollup-win32-arm64-msvc@4.54.0", "", { "os": "win32", "cpu": "arm64" }, "sha512-c2V0W1bsKIKfbLMBu/WGBz6Yci8nJ/ZJdheE0EwB73N3MvHYKiKGs3mVilX4Gs70eGeDaMqEob25Tw2Gb9Nqyw=="], + + "@rollup/rollup-win32-ia32-msvc": ["@rollup/rollup-win32-ia32-msvc@4.54.0", "", { "os": "win32", "cpu": "ia32" }, "sha512-woEHgqQqDCkAzrDhvDipnSirm5vxUXtSKDYTVpZG3nUdW/VVB5VdCYA2iReSj/u3yCZzXID4kuKG7OynPnB3WQ=="], + + "@rollup/rollup-win32-x64-gnu": ["@rollup/rollup-win32-x64-gnu@4.54.0", "", { "os": "win32", "cpu": "x64" }, "sha512-dzAc53LOuFvHwbCEOS0rPbXp6SIhAf2txMP5p6mGyOXXw5mWY8NGGbPMPrs4P1WItkfApDathBj/NzMLUZ9rtQ=="], + + "@rollup/rollup-win32-x64-msvc": ["@rollup/rollup-win32-x64-msvc@4.54.0", "", { "os": "win32", "cpu": "x64" }, "sha512-hYT5d3YNdSh3mbCU1gwQyPgQd3T2ne0A3KG8KSBdav5TiBg6eInVmV+TeR5uHufiIgSFg0XsOWGW5/RhNcSvPg=="], + + "@standard-schema/spec": ["@standard-schema/spec@1.1.0", "", {}, "sha512-l2aFy5jALhniG5HgqrD6jXLi/rUWrKvqN/qJx6yoJsgKhblVd+iqqU4RCXavm/jPityDo5TCvKMnpjKnOriy0w=="], + + "@sveltejs/acorn-typescript": ["@sveltejs/acorn-typescript@1.0.8", "", { "peerDependencies": { "acorn": "^8.9.0" } }, "sha512-esgN+54+q0NjB0Y/4BomT9samII7jGwNy/2a3wNZbT2A2RpmXsXwUt24LvLhx6jUq2gVk4cWEvcRO6MFQbOfNA=="], + + "@sveltejs/adapter-auto": ["@sveltejs/adapter-auto@7.0.0", "", { "peerDependencies": { "@sveltejs/kit": "^2.0.0" } }, "sha512-ImDWaErTOCkRS4Gt+5gZuymKFBobnhChXUZ9lhUZLahUgvA4OOvRzi3sahzYgbxGj5nkA6OV0GAW378+dl/gyw=="], + + "@sveltejs/adapter-node": ["@sveltejs/adapter-node@5.4.0", "", { "dependencies": { "@rollup/plugin-commonjs": "^28.0.1", "@rollup/plugin-json": "^6.1.0", "@rollup/plugin-node-resolve": "^16.0.0", "rollup": "^4.9.5" }, "peerDependencies": { "@sveltejs/kit": "^2.4.0" } }, "sha512-NMsrwGVPEn+J73zH83Uhss/hYYZN6zT3u31R3IHAn3MiKC3h8fjmIAhLfTSOeNHr5wPYfjjMg8E+1gyFgyrEcQ=="], + + "@sveltejs/kit": ["@sveltejs/kit@2.49.2", "", { "dependencies": { "@standard-schema/spec": "^1.0.0", "@sveltejs/acorn-typescript": "^1.0.5", "@types/cookie": "^0.6.0", "acorn": "^8.14.1", "cookie": "^0.6.0", "devalue": "^5.3.2", "esm-env": "^1.2.2", "kleur": "^4.1.5", "magic-string": "^0.30.5", "mrmime": "^2.0.0", "sade": "^1.8.1", "set-cookie-parser": "^2.6.0", "sirv": "^3.0.0" }, "peerDependencies": { "@opentelemetry/api": "^1.0.0", "@sveltejs/vite-plugin-svelte": "^3.0.0 || ^4.0.0-next.1 || ^5.0.0 || ^6.0.0-next.0", "svelte": "^4.0.0 || ^5.0.0-next.0", "vite": "^5.0.3 || ^6.0.0 || ^7.0.0-beta.0" }, "optionalPeers": ["@opentelemetry/api"], "bin": { "svelte-kit": "svelte-kit.js" } }, "sha512-Vp3zX/qlwerQmHMP6x0Ry1oY7eKKRcOWGc2P59srOp4zcqyn+etJyQpELgOi4+ZSUgteX8Y387NuwruLgGXLUQ=="], + + "@sveltejs/vite-plugin-svelte": ["@sveltejs/vite-plugin-svelte@6.2.1", "", { "dependencies": { "@sveltejs/vite-plugin-svelte-inspector": "^5.0.0", "debug": "^4.4.1", "deepmerge": "^4.3.1", "magic-string": "^0.30.17", "vitefu": "^1.1.1" }, "peerDependencies": { "svelte": "^5.0.0", "vite": "^6.3.0 || ^7.0.0" } }, "sha512-YZs/OSKOQAQCnJvM/P+F1URotNnYNeU3P2s4oIpzm1uFaqUEqRxUB0g5ejMjEb5Gjb9/PiBI5Ktrq4rUUF8UVQ=="], + + "@sveltejs/vite-plugin-svelte-inspector": ["@sveltejs/vite-plugin-svelte-inspector@5.0.1", "", { "dependencies": { "debug": "^4.4.1" }, "peerDependencies": { "@sveltejs/vite-plugin-svelte": "^6.0.0-next.0", "svelte": "^5.0.0", "vite": "^6.3.0 || ^7.0.0" } }, "sha512-ubWshlMk4bc8mkwWbg6vNvCeT7lGQojE3ijDh3QTR6Zr/R+GXxsGbyH4PExEPpiFmqPhYiVSVmHBjUcVc1JIrA=="], + + "@tailwindcss/forms": ["@tailwindcss/forms@0.5.11", "", { "dependencies": { "mini-svg-data-uri": "^1.2.3" }, "peerDependencies": { "tailwindcss": ">=3.0.0 || >= 3.0.0-alpha.1 || >= 4.0.0-alpha.20 || >= 4.0.0-beta.1" } }, "sha512-h9wegbZDPurxG22xZSoWtdzc41/OlNEUQERNqI/0fOwa2aVlWGu7C35E/x6LDyD3lgtztFSSjKZyuVM0hxhbgA=="], + + "@tailwindcss/node": ["@tailwindcss/node@4.1.18", "", { "dependencies": { "@jridgewell/remapping": "^2.3.4", "enhanced-resolve": "^5.18.3", "jiti": "^2.6.1", "lightningcss": "1.30.2", "magic-string": "^0.30.21", "source-map-js": "^1.2.1", "tailwindcss": "4.1.18" } }, "sha512-DoR7U1P7iYhw16qJ49fgXUlry1t4CpXeErJHnQ44JgTSKMaZUdf17cfn5mHchfJ4KRBZRFA/Coo+MUF5+gOaCQ=="], + + "@tailwindcss/oxide": ["@tailwindcss/oxide@4.1.18", "", { "optionalDependencies": { "@tailwindcss/oxide-android-arm64": "4.1.18", "@tailwindcss/oxide-darwin-arm64": "4.1.18", "@tailwindcss/oxide-darwin-x64": "4.1.18", "@tailwindcss/oxide-freebsd-x64": "4.1.18", "@tailwindcss/oxide-linux-arm-gnueabihf": "4.1.18", "@tailwindcss/oxide-linux-arm64-gnu": "4.1.18", "@tailwindcss/oxide-linux-arm64-musl": "4.1.18", "@tailwindcss/oxide-linux-x64-gnu": "4.1.18", "@tailwindcss/oxide-linux-x64-musl": "4.1.18", "@tailwindcss/oxide-wasm32-wasi": "4.1.18", "@tailwindcss/oxide-win32-arm64-msvc": "4.1.18", "@tailwindcss/oxide-win32-x64-msvc": "4.1.18" } }, "sha512-EgCR5tTS5bUSKQgzeMClT6iCY3ToqE1y+ZB0AKldj809QXk1Y+3jB0upOYZrn9aGIzPtUsP7sX4QQ4XtjBB95A=="], + + "@tailwindcss/oxide-android-arm64": ["@tailwindcss/oxide-android-arm64@4.1.18", "", { "os": "android", "cpu": "arm64" }, "sha512-dJHz7+Ugr9U/diKJA0W6N/6/cjI+ZTAoxPf9Iz9BFRF2GzEX8IvXxFIi/dZBloVJX/MZGvRuFA9rqwdiIEZQ0Q=="], + + "@tailwindcss/oxide-darwin-arm64": ["@tailwindcss/oxide-darwin-arm64@4.1.18", "", { "os": "darwin", "cpu": "arm64" }, "sha512-Gc2q4Qhs660bhjyBSKgq6BYvwDz4G+BuyJ5H1xfhmDR3D8HnHCmT/BSkvSL0vQLy/nkMLY20PQ2OoYMO15Jd0A=="], + + "@tailwindcss/oxide-darwin-x64": ["@tailwindcss/oxide-darwin-x64@4.1.18", "", { "os": "darwin", "cpu": "x64" }, "sha512-FL5oxr2xQsFrc3X9o1fjHKBYBMD1QZNyc1Xzw/h5Qu4XnEBi3dZn96HcHm41c/euGV+GRiXFfh2hUCyKi/e+yw=="], + + "@tailwindcss/oxide-freebsd-x64": ["@tailwindcss/oxide-freebsd-x64@4.1.18", "", { "os": "freebsd", "cpu": "x64" }, "sha512-Fj+RHgu5bDodmV1dM9yAxlfJwkkWvLiRjbhuO2LEtwtlYlBgiAT4x/j5wQr1tC3SANAgD+0YcmWVrj8R9trVMA=="], + + "@tailwindcss/oxide-linux-arm-gnueabihf": ["@tailwindcss/oxide-linux-arm-gnueabihf@4.1.18", "", { "os": "linux", "cpu": "arm" }, "sha512-Fp+Wzk/Ws4dZn+LV2Nqx3IilnhH51YZoRaYHQsVq3RQvEl+71VGKFpkfHrLM/Li+kt5c0DJe/bHXK1eHgDmdiA=="], + + "@tailwindcss/oxide-linux-arm64-gnu": ["@tailwindcss/oxide-linux-arm64-gnu@4.1.18", "", { "os": "linux", "cpu": "arm64" }, "sha512-S0n3jboLysNbh55Vrt7pk9wgpyTTPD0fdQeh7wQfMqLPM/Hrxi+dVsLsPrycQjGKEQk85Kgbx+6+QnYNiHalnw=="], + + "@tailwindcss/oxide-linux-arm64-musl": ["@tailwindcss/oxide-linux-arm64-musl@4.1.18", "", { "os": "linux", "cpu": "arm64" }, "sha512-1px92582HkPQlaaCkdRcio71p8bc8i/ap5807tPRDK/uw953cauQBT8c5tVGkOwrHMfc2Yh6UuxaH4vtTjGvHg=="], + + "@tailwindcss/oxide-linux-x64-gnu": ["@tailwindcss/oxide-linux-x64-gnu@4.1.18", "", { "os": "linux", "cpu": "x64" }, "sha512-v3gyT0ivkfBLoZGF9LyHmts0Isc8jHZyVcbzio6Wpzifg/+5ZJpDiRiUhDLkcr7f/r38SWNe7ucxmGW3j3Kb/g=="], + + "@tailwindcss/oxide-linux-x64-musl": ["@tailwindcss/oxide-linux-x64-musl@4.1.18", "", { "os": "linux", "cpu": "x64" }, "sha512-bhJ2y2OQNlcRwwgOAGMY0xTFStt4/wyU6pvI6LSuZpRgKQwxTec0/3Scu91O8ir7qCR3AuepQKLU/kX99FouqQ=="], + + "@tailwindcss/oxide-wasm32-wasi": ["@tailwindcss/oxide-wasm32-wasi@4.1.18", "", { "dependencies": { "@emnapi/core": "^1.7.1", "@emnapi/runtime": "^1.7.1", "@emnapi/wasi-threads": "^1.1.0", "@napi-rs/wasm-runtime": "^1.1.0", "@tybys/wasm-util": "^0.10.1", "tslib": "^2.4.0" }, "cpu": "none" }, "sha512-LffYTvPjODiP6PT16oNeUQJzNVyJl1cjIebq/rWWBF+3eDst5JGEFSc5cWxyRCJ0Mxl+KyIkqRxk1XPEs9x8TA=="], + + "@tailwindcss/oxide-win32-arm64-msvc": ["@tailwindcss/oxide-win32-arm64-msvc@4.1.18", "", { "os": "win32", "cpu": "arm64" }, "sha512-HjSA7mr9HmC8fu6bdsZvZ+dhjyGCLdotjVOgLA2vEqxEBZaQo9YTX4kwgEvPCpRh8o4uWc4J/wEoFzhEmjvPbA=="], + + "@tailwindcss/oxide-win32-x64-msvc": ["@tailwindcss/oxide-win32-x64-msvc@4.1.18", "", { "os": "win32", "cpu": "x64" }, "sha512-bJWbyYpUlqamC8dpR7pfjA0I7vdF6t5VpUGMWRkXVE3AXgIZjYUYAK7II1GNaxR8J1SSrSrppRar8G++JekE3Q=="], + + "@tailwindcss/typography": ["@tailwindcss/typography@0.5.19", "", { "dependencies": { "postcss-selector-parser": "6.0.10" }, "peerDependencies": { "tailwindcss": ">=3.0.0 || insiders || >=4.0.0-alpha.20 || >=4.0.0-beta.1" } }, "sha512-w31dd8HOx3k9vPtcQh5QHP9GwKcgbMp87j58qi6xgiBnFFtKEAgCWnDw4qUT8aHwkCp8bKvb/KGKWWHedP0AAg=="], + + "@tailwindcss/vite": ["@tailwindcss/vite@4.1.18", "", { "dependencies": { "@tailwindcss/node": "4.1.18", "@tailwindcss/oxide": "4.1.18", "tailwindcss": "4.1.18" }, "peerDependencies": { "vite": "^5.2.0 || ^6 || ^7" } }, "sha512-jVA+/UpKL1vRLg6Hkao5jldawNmRo7mQYrZtNHMIVpLfLhDml5nMRUo/8MwoX2vNXvnaXNNMedrMfMugAVX1nA=="], + + "@threejs-kit/instanced-sprite-mesh": ["@threejs-kit/instanced-sprite-mesh@2.5.1", "", { "dependencies": { "diet-sprite": "^0.0.1", "earcut": "^2.2.4", "maath": "^0.10.7", "three-instanced-uniforms-mesh": "^0.52.4", "troika-three-utils": "^0.52.4" }, "peerDependencies": { "three": ">=0.170.0" } }, "sha512-pmt1ALRhbHhCJQTj2FuthH6PeLIeaM4hOuS2JO3kWSwlnvx/9xuUkjFR3JOi/myMqsH7pSsLIROSaBxDfttjeA=="], + + "@threlte/core": ["@threlte/core@8.3.1", "", { "dependencies": { "mitt": "^3.0.1" }, "peerDependencies": { "svelte": ">=5", "three": ">=0.160" } }, "sha512-qKjjNCQ+40hyeBcfOMh/8ef5x/j5PG5Wmo/L9Ye0aDCcdD6fCewWxfp7tV/J3CxPzX1dEp1JGK7sjyc7ntZSrg=="], + + "@threlte/extras": ["@threlte/extras@9.7.1", "", { "dependencies": { "@threejs-kit/instanced-sprite-mesh": "^2.5.1", "camera-controls": "^3.1.2", "three-mesh-bvh": "^0.9.1", "three-perf": "^1.0.11", "three-viewport-gizmo": "^2.2.0", "troika-three-text": "^0.52.4" }, "peerDependencies": { "svelte": ">=5", "three": ">=0.160" } }, "sha512-SGm59HDCdHxADFHuweHfFDknwubkCPodyK0pbfsVtOWWOX26gE2xfK7aKolh6YFDiPAjWjGxN0jIgkNbbr1ohg=="], + + "@tweenjs/tween.js": ["@tweenjs/tween.js@23.1.3", "", {}, "sha512-vJmvvwFxYuGnF2axRtPYocag6Clbb5YS7kLL+SO/TeVFzHqDIWrNKYtcsPMibjDx9O+bu+psAy9NKfWklassUA=="], + + "@types/bcryptjs": ["@types/bcryptjs@3.0.0", "", { "dependencies": { "bcryptjs": "*" } }, "sha512-WRZOuCuaz8UcZZE4R5HXTco2goQSI2XxjGY3hbM/xDvwmqFWd4ivooImsMx65OKM6CtNKbnZ5YL+YwAwK7c1dg=="], + + "@types/cookie": ["@types/cookie@1.0.0", "", { "dependencies": { "cookie": "*" } }, "sha512-mGFXbkDQJ6kAXByHS7QAggRXgols0mAdP4MuXgloGY1tXokvzaFFM4SMqWvf7AH0oafI7zlFJwoGWzmhDqTZ9w=="], + + "@types/estree": ["@types/estree@1.0.8", "", {}, "sha512-dWHzHa2WqEXI/O1E9OjrocMTKJl2mSrEolh1Iomrv6U+JuNwaHXsXx9bLu5gG7BUWFIN0skIQJQ/L1rIex4X6w=="], + + "@types/jsonwebtoken": ["@types/jsonwebtoken@9.0.10", "", { "dependencies": { "@types/ms": "*", "@types/node": "*" } }, "sha512-asx5hIG9Qmf/1oStypjanR7iKTv0gXQ1Ov/jfrX6kS/EO0OFni8orbmGCn0672NHR3kXHwpAwR+B368ZGN/2rA=="], + + "@types/ms": ["@types/ms@2.1.0", "", {}, "sha512-GsCCIZDE/p3i96vtEqx+7dBUGXrc7zeSK3wwPHIaRThS+9OhWIXRqzs4d6k1SVU8g91DrNRWxWUGhp5KXQb2VA=="], + + "@types/node": ["@types/node@25.0.3", "", { "dependencies": { "undici-types": "~7.16.0" } }, "sha512-W609buLVRVmeW693xKfzHeIV6nJGGz98uCPfeXI1ELMLXVeKYZ9m15fAMSaUPBHYLGFsVRcMmSCksQOrZV9BYA=="], + + "@types/resolve": ["@types/resolve@1.20.2", "", {}, "sha512-60BCwRFOZCQhDncwQdxxeOEEkbc5dIMccYLwbxsS4TUNeVECQ/pBJ0j09mrHOl/JJvpRPGwO9SvE4nR2Nb/a4Q=="], + + "@types/stats.js": ["@types/stats.js@0.17.4", "", {}, "sha512-jIBvWWShCvlBqBNIZt0KAshWpvSjhkwkEu4ZUcASoAvhmrgAUI2t1dXrjSL4xXVLB4FznPrIsX3nKXFl/Dt4vA=="], + + "@types/three": ["@types/three@0.182.0", "", { "dependencies": { "@dimforge/rapier3d-compat": "~0.12.0", "@tweenjs/tween.js": "~23.1.3", "@types/stats.js": "*", "@types/webxr": ">=0.5.17", "@webgpu/types": "*", "fflate": "~0.8.2", "meshoptimizer": "~0.22.0" } }, "sha512-WByN9V3Sbwbe2OkWuSGyoqQO8Du6yhYaXtXLoA5FkKTUJorZ+yOHBZ35zUUPQXlAKABZmbYp5oAqpA4RBjtJ/Q=="], + + "@types/webidl-conversions": ["@types/webidl-conversions@7.0.3", "", {}, "sha512-CiJJvcRtIgzadHCYXw7dqEnMNRjhGZlYK05Mj9OyktqV8uVT8fD2BFOB7S1uwBE3Kj2Z+4UyPmFw/Ixgw/LAlA=="], + + "@types/webxr": ["@types/webxr@0.5.24", "", {}, "sha512-h8fgEd/DpoS9CBrjEQXR+dIDraopAEfu4wYVNY2tEPwk60stPWhvZMf4Foo5FakuQ7HFZoa8WceaWFervK2Ovg=="], + + "@types/whatwg-url": ["@types/whatwg-url@13.0.0", "", { "dependencies": { "@types/webidl-conversions": "*" } }, "sha512-N8WXpbE6Wgri7KUSvrmQcqrMllKZ9uxkYWMt+mCSGwNc0Hsw9VQTW7ApqI4XNrx6/SaM2QQJCzMPDEXE058s+Q=="], + + "@webgpu/types": ["@webgpu/types@0.1.68", "", {}, "sha512-3ab1B59Ojb6RwjOspYLsTpCzbNB3ZaamIAxBMmvnNkiDoLTZUOBXZ9p5nAYVEkQlDdf6qAZWi1pqj9+ypiqznA=="], + + "accepts": ["accepts@2.0.0", "", { "dependencies": { "mime-types": "^3.0.0", "negotiator": "^1.0.0" } }, "sha512-5cvg6CtKwfgdmVqY1WIiXKc3Q1bkRqGLi+2W/6ao+6Y7gu/RCwRuAhGEzh5B4KlszSuTLgZYuqFqo5bImjNKng=="], + + "acorn": ["acorn@8.15.0", "", { "bin": { "acorn": "bin/acorn" } }, "sha512-NZyJarBfL7nWwIq+FDL6Zp/yHEhePMNnnJ0y3qfieCrmNvYct8uvtiV41UvlSe6apAfk0fY1FbWx+NwfmpvtTg=="], + + "aria-query": ["aria-query@5.3.2", "", {}, "sha512-COROpnaoap1E2F000S62r6A60uHZnmlvomhfyT2DlTcrY1OrBKn2UhH7qn5wTC9zMvD0AY7csdPSNwKP+7WiQw=="], + + "axobject-query": ["axobject-query@4.1.0", "", {}, "sha512-qIj0G9wZbMGNLjLmg1PT6v2mE9AH2zlnADJD/2tC6E00hgmhUOfEB6greHPAfLRSufHqROIUTkw6E+M3lH0PTQ=="], + + "bcryptjs": ["bcryptjs@3.0.3", "", { "bin": { "bcrypt": "bin/bcrypt" } }, "sha512-GlF5wPWnSa/X5LKM1o0wz0suXIINz1iHRLvTS+sLyi7XPbe5ycmYI3DlZqVGZZtDgl4DmasFg7gOB3JYbphV5g=="], + + "bidi-js": ["bidi-js@1.0.3", "", { "dependencies": { "require-from-string": "^2.0.2" } }, "sha512-RKshQI1R3YQ+n9YJz2QQ147P66ELpa1FQEg20Dk8oW9t2KgLbpDLLp9aGZ7y8WHSshDknG0bknqGw5/tyCs5tw=="], + + "body-parser": ["body-parser@2.2.1", "", { "dependencies": { "bytes": "^3.1.2", "content-type": "^1.0.5", "debug": "^4.4.3", "http-errors": "^2.0.0", "iconv-lite": "^0.7.0", "on-finished": "^2.4.1", "qs": "^6.14.0", "raw-body": "^3.0.1", "type-is": "^2.0.1" } }, "sha512-nfDwkulwiZYQIGwxdy0RUmowMhKcFVcYXUU7m4QlKYim1rUtg83xm2yjZ40QjDuc291AJjjeSc9b++AWHSgSHw=="], + + "bson": ["bson@7.0.0", "", {}, "sha512-Kwc6Wh4lQ5OmkqqKhYGKIuELXl+EPYSCObVE6bWsp1T/cGkOCBN0I8wF/T44BiuhHyNi1mmKVPXk60d41xZ7kw=="], + + "buffer-equal-constant-time": ["buffer-equal-constant-time@1.0.1", "", {}, "sha512-zRpUiDwd/xk6ADqPMATG8vc9VPrkck7T07OIx0gnjmJAnHnTVXNQG3vfvWNuiZIkwu9KrKdA1iJKfsfTVxE6NA=="], + + "bytes": ["bytes@3.1.2", "", {}, "sha512-/Nf7TyzTx6S3yRJObOAV7956r8cr2+Oj8AC5dt8wSP3BQAoeX58NoHyCU8P8zGkNXStjTSi6fzO6F0pBdcYbEg=="], + + "call-bind-apply-helpers": ["call-bind-apply-helpers@1.0.2", "", { "dependencies": { "es-errors": "^1.3.0", "function-bind": "^1.1.2" } }, "sha512-Sp1ablJ0ivDkSzjcaJdxEunN5/XvksFJ2sMBFfq6x0ryhQV/2b/KwFe21cMpmHtPOSij8K99/wSfoEuTObmuMQ=="], + + "call-bound": ["call-bound@1.0.4", "", { "dependencies": { "call-bind-apply-helpers": "^1.0.2", "get-intrinsic": "^1.3.0" } }, "sha512-+ys997U96po4Kx/ABpBCqhA9EuxJaQWDQg7295H4hBphv3IZg0boBKuwYpt4YXp6MZ5AmZQnU/tyMTlRpaSejg=="], + + "camera-controls": ["camera-controls@3.1.2", "", { "peerDependencies": { "three": ">=0.126.1" } }, "sha512-xkxfpG2ECZ6Ww5/9+kf4mfg1VEYAoe9aDSY+IwF0UEs7qEzwy0aVRfs2grImIECs/PoBtWFrh7RXsQkwG922JA=="], + + "chart.js": ["chart.js@4.5.1", "", { "dependencies": { "@kurkle/color": "^0.3.0" } }, "sha512-GIjfiT9dbmHRiYi6Nl2yFCq7kkwdkp1W/lp2J99rX0yo9tgJGn3lKQATztIjb5tVtevcBtIdICNWqlq5+E8/Pw=="], + + "chokidar": ["chokidar@4.0.3", "", { "dependencies": { "readdirp": "^4.0.1" } }, "sha512-Qgzu8kfBvo+cA4962jnP1KkS6Dop5NS6g7R5LFYJr4b8Ub94PPQXUksCw9PvXoeXPRRddRNC5C1JQUR2SMGtnA=="], + + "clsx": ["clsx@2.1.1", "", {}, "sha512-eYm0QWBtUrBWZWG0d386OGAw16Z995PiOVo2B7bjWSbHedGl5e0ZWaq65kOGgUSNesEIDkB9ISbTg/JK9dhCZA=="], + + "commondir": ["commondir@1.0.1", "", {}, "sha512-W9pAhw0ja1Edb5GVdIF1mjZw/ASI0AlShXM83UUGe2DVr5TdAPEA1OA8m/g8zWp9x6On7gqufY+FatDbC3MDQg=="], + + "content-disposition": ["content-disposition@1.0.1", "", {}, "sha512-oIXISMynqSqm241k6kcQ5UwttDILMK4BiurCfGEREw6+X9jkkpEe5T9FZaApyLGGOnFuyMWZpdolTXMtvEJ08Q=="], + + "content-type": ["content-type@1.0.5", "", {}, "sha512-nTjqfcBFEipKdXCv4YDQWCfmcLZKm81ldF0pAopTvyrFGVbcR6P/VAAd5G7N+0tTr8QqiU0tFadD6FK4NtJwOA=="], + + "cookie": ["cookie@1.1.1", "", {}, "sha512-ei8Aos7ja0weRpFzJnEA9UHJ/7XQmqglbRwnf2ATjcB9Wq874VKH9kfjjirM6UhU2/E5fFYadylyhFldcqSidQ=="], + + "cookie-signature": ["cookie-signature@1.2.2", "", {}, "sha512-D76uU73ulSXrD1UXF4KE2TMxVVwhsnCgfAyTg9k8P6KGZjlXKrOLe4dJQKI3Bxi5wjesZoFXJWElNWBjPZMbhg=="], + + "cors": ["cors@2.8.5", "", { "dependencies": { "object-assign": "^4", "vary": "^1" } }, "sha512-KIHbLJqu73RGr/hnbrO9uBeixNGuvSQjul/jdFvS/KFSIH1hWVd1ng7zOHx+YrEfInLG7q4n6GHQ9cDtxv/P6g=="], + + "cssesc": ["cssesc@3.0.0", "", { "bin": { "cssesc": "bin/cssesc" } }, "sha512-/Tb/JcjK111nNScGob5MNtsntNM1aCNUDipB/TkwZFhyDrrE47SOx/18wF2bbjgc3ZzCSKW1T5nt5EbFoAz/Vg=="], + + "debug": ["debug@4.4.3", "", { "dependencies": { "ms": "^2.1.3" } }, "sha512-RGwwWnwQvkVfavKVt22FGLw+xYSdzARwm0ru6DhTVA3umU5hZc28V3kO4stgYryrTlLpuvgI9GiijltAjNbcqA=="], + + "deepmerge": ["deepmerge@4.3.1", "", {}, "sha512-3sUqbMEc77XqpdNO7FRyRog+eW3ph+GYCbj+rK+uYyRMuwsVy0rMiVtPn+QJlKFvWP/1PYpapqYn0Me2knFn+A=="], + + "depd": ["depd@2.0.0", "", {}, "sha512-g7nH6P6dyDioJogAAGprGpCtVImJhpPk/roCzdb3fIh61/s/nPsfR6onyMwkCAR/OlC3yBC0lESvUoQEAssIrw=="], + + "detect-libc": ["detect-libc@2.1.2", "", {}, "sha512-Btj2BOOO83o3WyH59e8MgXsxEQVcarkUOpEYrubB0urwnN10yQ364rsiByU11nZlqWYZm05i/of7io4mzihBtQ=="], + + "devalue": ["devalue@5.6.1", "", {}, "sha512-jDwizj+IlEZBunHcOuuFVBnIMPAEHvTsJj0BcIp94xYguLRVBcXO853px/MyIJvbVzWdsGvrRweIUWJw8hBP7A=="], + + "diet-sprite": ["diet-sprite@0.0.1", "", {}, "sha512-zSHI2WDAn1wJqJYxcmjWfJv3Iw8oL9reQIbEyx2x2/EZ4/qmUTIo8/5qOCurnAcq61EwtJJaZ0XTy2NRYqpB5A=="], + + "dotenv": ["dotenv@17.2.3", "", {}, "sha512-JVUnt+DUIzu87TABbhPmNfVdBDt18BLOWjMUFJMSi/Qqg7NTYtabbvSNJGOJ7afbRuv9D/lngizHtP7QyLQ+9w=="], + + "dunder-proto": ["dunder-proto@1.0.1", "", { "dependencies": { "call-bind-apply-helpers": "^1.0.1", "es-errors": "^1.3.0", "gopd": "^1.2.0" } }, "sha512-KIN/nDJBQRcXw0MLVhZE9iQHmG68qAVIBg9CqmUYjmQIhgij9U5MFvrqkUL5FbtyyzZuOeOt0zdeRe4UY7ct+A=="], + + "earcut": ["earcut@2.2.4", "", {}, "sha512-/pjZsA1b4RPHbeWZQn66SWS8nZZWLQQ23oE3Eam7aroEFGEvwKAsJfZ9ytiEMycfzXWpca4FA9QIOehf7PocBQ=="], + + "ecdsa-sig-formatter": ["ecdsa-sig-formatter@1.0.11", "", { "dependencies": { "safe-buffer": "^5.0.1" } }, "sha512-nagl3RYrbNv6kQkeJIpt6NJZy8twLB/2vtz6yN9Z4vRKHN4/QZJIEbqohALSgwKdnksuY3k5Addp5lg8sVoVcQ=="], + + "ee-first": ["ee-first@1.1.1", "", {}, "sha512-WMwm9LhRUo+WUaRN+vRuETqG89IgZphVSNkdFgeb6sS/E4OrDIN7t48CAewSHXc6C8lefD8KKfr5vY61brQlow=="], + + "encodeurl": ["encodeurl@2.0.0", "", {}, "sha512-Q0n9HRi4m6JuGIV1eFlmvJB7ZEVxu93IrMyiMsGC0lrMJMWzRgx6WGquyfQgZVb31vhGgXnfmPNNXmxnOkRBrg=="], + + "enhanced-resolve": ["enhanced-resolve@5.18.4", "", { "dependencies": { "graceful-fs": "^4.2.4", "tapable": "^2.2.0" } }, "sha512-LgQMM4WXU3QI+SYgEc2liRgznaD5ojbmY3sb8LxyguVkIg5FxdpTkvk72te2R38/TGKxH634oLxXRGY6d7AP+Q=="], + + "es-define-property": ["es-define-property@1.0.1", "", {}, "sha512-e3nRfgfUZ4rNGL232gUgX06QNyyez04KdjFrF+LTRoOXmrOgFKDg4BCdsjW8EnT69eqdYGmRpJwiPVYNrCaW3g=="], + + "es-errors": ["es-errors@1.3.0", "", {}, "sha512-Zf5H2Kxt2xjTvbJvP2ZWLEICxA6j+hAmMzIlypy4xcBg1vKVnx89Wy0GbS+kf5cwCVFFzdCFh2XSCFNULS6csw=="], + + "es-object-atoms": ["es-object-atoms@1.1.1", "", { "dependencies": { "es-errors": "^1.3.0" } }, "sha512-FGgH2h8zKNim9ljj7dankFPcICIK9Cp5bm+c2gQSYePhpaG5+esrLODihIorn+Pe6FGJzWhXQotPv73jTaldXA=="], + + "esbuild": ["esbuild@0.27.2", "", { "optionalDependencies": { "@esbuild/aix-ppc64": "0.27.2", "@esbuild/android-arm": "0.27.2", "@esbuild/android-arm64": "0.27.2", "@esbuild/android-x64": "0.27.2", "@esbuild/darwin-arm64": "0.27.2", "@esbuild/darwin-x64": "0.27.2", "@esbuild/freebsd-arm64": "0.27.2", "@esbuild/freebsd-x64": "0.27.2", "@esbuild/linux-arm": "0.27.2", "@esbuild/linux-arm64": "0.27.2", "@esbuild/linux-ia32": "0.27.2", "@esbuild/linux-loong64": "0.27.2", "@esbuild/linux-mips64el": "0.27.2", "@esbuild/linux-ppc64": "0.27.2", "@esbuild/linux-riscv64": "0.27.2", "@esbuild/linux-s390x": "0.27.2", "@esbuild/linux-x64": "0.27.2", "@esbuild/netbsd-arm64": "0.27.2", "@esbuild/netbsd-x64": "0.27.2", "@esbuild/openbsd-arm64": "0.27.2", "@esbuild/openbsd-x64": "0.27.2", "@esbuild/openharmony-arm64": "0.27.2", "@esbuild/sunos-x64": "0.27.2", "@esbuild/win32-arm64": "0.27.2", "@esbuild/win32-ia32": "0.27.2", "@esbuild/win32-x64": "0.27.2" }, "bin": { "esbuild": "bin/esbuild" } }, "sha512-HyNQImnsOC7X9PMNaCIeAm4ISCQXs5a5YasTXVliKv4uuBo1dKrG0A+uQS8M5eXjVMnLg3WgXaKvprHlFJQffw=="], + + "escape-html": ["escape-html@1.0.3", "", {}, "sha512-NiSupZ4OeuGwr68lGIeym/ksIZMJodUGOSCZ/FSnTxcrekbvqrgdUxlJOMpijaKZVjAJrWrGs/6Jy8OMuyj9ow=="], + + "esm-env": ["esm-env@1.2.2", "", {}, "sha512-Epxrv+Nr/CaL4ZcFGPJIYLWFom+YeV1DqMLHJoEd9SYRxNbaFruBwfEX/kkHUJf55j2+TUbmDcmuilbP1TmXHA=="], + + "esrap": ["esrap@2.2.1", "", { "dependencies": { "@jridgewell/sourcemap-codec": "^1.4.15" } }, "sha512-GiYWG34AN/4CUyaWAgunGt0Rxvr1PTMlGC0vvEov/uOQYWne2bpN03Um+k8jT+q3op33mKouP2zeJ6OlM+qeUg=="], + + "estree-walker": ["estree-walker@2.0.2", "", {}, "sha512-Rfkk/Mp/DL7JVje3u18FxFujQlTNR2q6QfMSMB7AvCBx91NGj/ba3kCfza0f6dVDbw7YlRf/nDrn7pQrCCyQ/w=="], + + "etag": ["etag@1.8.1", "", {}, "sha512-aIL5Fx7mawVa300al2BnEE4iNvo1qETxLrPI/o05L7z6go7fCw1J6EQmbK4FmJ2AS7kgVF/KEZWufBfdClMcPg=="], + + "express": ["express@5.2.1", "", { "dependencies": { "accepts": "^2.0.0", "body-parser": "^2.2.1", "content-disposition": "^1.0.0", "content-type": "^1.0.5", "cookie": "^0.7.1", "cookie-signature": "^1.2.1", "debug": "^4.4.0", "depd": "^2.0.0", "encodeurl": "^2.0.0", "escape-html": "^1.0.3", "etag": "^1.8.1", "finalhandler": "^2.1.0", "fresh": "^2.0.0", "http-errors": "^2.0.0", "merge-descriptors": "^2.0.0", "mime-types": "^3.0.0", "on-finished": "^2.4.1", "once": "^1.4.0", "parseurl": "^1.3.3", "proxy-addr": "^2.0.7", "qs": "^6.14.0", "range-parser": "^1.2.1", "router": "^2.2.0", "send": "^1.1.0", "serve-static": "^2.2.0", "statuses": "^2.0.1", "type-is": "^2.0.1", "vary": "^1.1.2" } }, "sha512-hIS4idWWai69NezIdRt2xFVofaF4j+6INOpJlVOLDO8zXGpUVEVzIYk12UUi2JzjEzWL3IOAxcTubgz9Po0yXw=="], + + "fdir": ["fdir@6.5.0", "", { "peerDependencies": { "picomatch": "^3 || ^4" }, "optionalPeers": ["picomatch"] }, "sha512-tIbYtZbucOs0BRGqPJkshJUYdL+SDH7dVM8gjy+ERp3WAUjLEFJE+02kanyHtwjWOnwrKYBiwAmM0p4kLJAnXg=="], + + "fflate": ["fflate@0.8.2", "", {}, "sha512-cPJU47OaAoCbg0pBvzsgpTPhmhqI5eJjh/JIu8tPj5q+T7iLvW/JAYUqmE7KOB4R1ZyEhzBaIQpQpardBF5z8A=="], + + "finalhandler": ["finalhandler@2.1.1", "", { "dependencies": { "debug": "^4.4.0", "encodeurl": "^2.0.0", "escape-html": "^1.0.3", "on-finished": "^2.4.1", "parseurl": "^1.3.3", "statuses": "^2.0.1" } }, "sha512-S8KoZgRZN+a5rNwqTxlZZePjT/4cnm0ROV70LedRHZ0p8u9fRID0hJUZQpkKLzro8LfmC8sx23bY6tVNxv8pQA=="], + + "forwarded": ["forwarded@0.2.0", "", {}, "sha512-buRG0fpBtRHSTCOASe6hD258tEubFoRLb4ZNA6NxMVHNw2gOcwHo9wyablzMzOA5z9xA9L1KNjk/Nt6MT9aYow=="], + + "fresh": ["fresh@2.0.0", "", {}, "sha512-Rx/WycZ60HOaqLKAi6cHRKKI7zxWbJ31MhntmtwMoaTeF7XFH9hhBp8vITaMidfljRQ6eYWCKkaTK+ykVJHP2A=="], + + "fsevents": ["fsevents@2.3.3", "", { "os": "darwin" }, "sha512-5xoDfX+fL7faATnagmWPpbFtwh/R77WmMMqqHGS65C3vvB0YHrgF+B1YmZ3441tMj5n63k0212XNoJwzlhffQw=="], + + "function-bind": ["function-bind@1.1.2", "", {}, "sha512-7XHNxH7qX9xG5mIwxkhumTox/MIRNcOgDrxWsMt2pAr23WHp6MrRlN7FBSFpCpr+oVO0F744iUgR82nJMfG2SA=="], + + "get-intrinsic": ["get-intrinsic@1.3.0", "", { "dependencies": { "call-bind-apply-helpers": "^1.0.2", "es-define-property": "^1.0.1", "es-errors": "^1.3.0", "es-object-atoms": "^1.1.1", "function-bind": "^1.1.2", "get-proto": "^1.0.1", "gopd": "^1.2.0", "has-symbols": "^1.1.0", "hasown": "^2.0.2", "math-intrinsics": "^1.1.0" } }, "sha512-9fSjSaos/fRIVIp+xSJlE6lfwhES7LNtKaCBIamHsjr2na1BiABJPo0mOjjz8GJDURarmCPGqaiVg5mfjb98CQ=="], + + "get-proto": ["get-proto@1.0.1", "", { "dependencies": { "dunder-proto": "^1.0.1", "es-object-atoms": "^1.0.0" } }, "sha512-sTSfBjoXBp89JvIKIefqw7U2CCebsc74kiY6awiGogKtoSGbgjYE/G/+l9sF3MWFPNc9IcoOC4ODfKHfxFmp0g=="], + + "gopd": ["gopd@1.2.0", "", {}, "sha512-ZUKRh6/kUFoAiTAtTYPZJ3hw9wNxx+BIBOijnlG9PnrJsCcSjs1wyyD6vJpaYtgnzDrKYRSqf3OO6Rfa93xsRg=="], + + "graceful-fs": ["graceful-fs@4.2.11", "", {}, "sha512-RbJ5/jmFcNNCcDV5o9eTnBLJ/HszWV0P73bc+Ff4nS/rJj+YaS6IGyiOL0VoBYX+l1Wrl3k63h/KrH+nhJ0XvQ=="], + + "has-symbols": ["has-symbols@1.1.0", "", {}, "sha512-1cDNdwJ2Jaohmb3sg4OmKaMBwuC48sYni5HUw2DvsC8LjGTLK9h+eb1X6RyuOHe4hT0ULCW68iomhjUoKUqlPQ=="], + + "hasown": ["hasown@2.0.2", "", { "dependencies": { "function-bind": "^1.1.2" } }, "sha512-0hJU9SCPvmMzIBdZFqNPXWa6dqh7WdH0cII9y+CyS8rG3nL48Bclra9HmKhVVUHyPWNH5Y7xDwAB7bfgSjkUMQ=="], + + "http-errors": ["http-errors@2.0.1", "", { "dependencies": { "depd": "~2.0.0", "inherits": "~2.0.4", "setprototypeof": "~1.2.0", "statuses": "~2.0.2", "toidentifier": "~1.0.1" } }, "sha512-4FbRdAX+bSdmo4AUFuS0WNiPz8NgFt+r8ThgNWmlrjQjt1Q7ZR9+zTlce2859x4KSXrwIsaeTqDoKQmtP8pLmQ=="], + + "iconv-lite": ["iconv-lite@0.7.1", "", { "dependencies": { "safer-buffer": ">= 2.1.2 < 3.0.0" } }, "sha512-2Tth85cXwGFHfvRgZWszZSvdo+0Xsqmw8k8ZwxScfcBneNUraK+dxRxRm24nszx80Y0TVio8kKLt5sLE7ZCLlw=="], + + "inherits": ["inherits@2.0.4", "", {}, "sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ=="], + + "ipaddr.js": ["ipaddr.js@1.9.1", "", {}, "sha512-0KI/607xoxSToH7GjN1FfSbLoU0+btTicjsQSWQlh/hZykN8KpmMf7uYwPW3R+akZ6R/w18ZlXSHBYXiYUPO3g=="], + + "is-core-module": ["is-core-module@2.16.1", "", { "dependencies": { "hasown": "^2.0.2" } }, "sha512-UfoeMA6fIJ8wTYFEUjelnaGI67v6+N7qXJEvQuIGa99l4xsCruSYOVSQ0uPANn4dAzm8lkYPaKLrrijLq7x23w=="], + + "is-module": ["is-module@1.0.0", "", {}, "sha512-51ypPSPCoTEIN9dy5Oy+h4pShgJmPCygKfyRCISBI+JoWT/2oJvK8QPxmwv7b/p239jXrm9M1mlQbyKJ5A152g=="], + + "is-promise": ["is-promise@4.0.0", "", {}, "sha512-hvpoI6korhJMnej285dSg6nu1+e6uxs7zG3BYAm5byqDsgJNWwxzM6z6iZiAgQR4TJ30JmBTOwqZUw3WlyH3AQ=="], + + "is-reference": ["is-reference@3.0.3", "", { "dependencies": { "@types/estree": "^1.0.6" } }, "sha512-ixkJoqQvAP88E6wLydLGGqCJsrFUnqoH6HnaczB8XmDH1oaWU+xxdptvikTgaEhtZ53Ky6YXiBuUI2WXLMCwjw=="], + + "jiti": ["jiti@2.6.1", "", { "bin": { "jiti": "lib/jiti-cli.mjs" } }, "sha512-ekilCSN1jwRvIbgeg/57YFh8qQDNbwDb9xT/qu2DAHbFFZUicIl4ygVaAvzveMhMVr3LnpSKTNnwt8PoOfmKhQ=="], + + "jsonwebtoken": ["jsonwebtoken@9.0.3", "", { "dependencies": { "jws": "^4.0.1", "lodash.includes": "^4.3.0", "lodash.isboolean": "^3.0.3", "lodash.isinteger": "^4.0.4", "lodash.isnumber": "^3.0.3", "lodash.isplainobject": "^4.0.6", "lodash.isstring": "^4.0.1", "lodash.once": "^4.0.0", "ms": "^2.1.1", "semver": "^7.5.4" } }, "sha512-MT/xP0CrubFRNLNKvxJ2BYfy53Zkm++5bX9dtuPbqAeQpTVe0MQTFhao8+Cp//EmJp244xt6Drw/GVEGCUj40g=="], + + "jwa": ["jwa@2.0.1", "", { "dependencies": { "buffer-equal-constant-time": "^1.0.1", "ecdsa-sig-formatter": "1.0.11", "safe-buffer": "^5.0.1" } }, "sha512-hRF04fqJIP8Abbkq5NKGN0Bbr3JxlQ+qhZufXVr0DvujKy93ZCbXZMHDL4EOtodSbCWxOqR8MS1tXA5hwqCXDg=="], + + "jws": ["jws@4.0.1", "", { "dependencies": { "jwa": "^2.0.1", "safe-buffer": "^5.0.1" } }, "sha512-EKI/M/yqPncGUUh44xz0PxSidXFr/+r0pA70+gIYhjv+et7yxM+s29Y+VGDkovRofQem0fs7Uvf4+YmAdyRduA=="], + + "kareem": ["kareem@3.0.0", "", {}, "sha512-RKhaOBSPN8L7y4yAgNhDT2602G5FD6QbOIISbjN9D6mjHPeqeg7K+EB5IGSU5o81/X2Gzm3ICnAvQW3x3OP8HA=="], + + "kleur": ["kleur@4.1.5", "", {}, "sha512-o+NO+8WrRiQEE4/7nwRJhN1HWpVmJm511pBHUxPLtp0BUISzlBplORYSmTclCnJvQq2tKu/sgl3xVpkc7ZWuQQ=="], + + "lightningcss": ["lightningcss@1.30.2", "", { "dependencies": { "detect-libc": "^2.0.3" }, "optionalDependencies": { "lightningcss-android-arm64": "1.30.2", "lightningcss-darwin-arm64": "1.30.2", "lightningcss-darwin-x64": "1.30.2", "lightningcss-freebsd-x64": "1.30.2", "lightningcss-linux-arm-gnueabihf": "1.30.2", "lightningcss-linux-arm64-gnu": "1.30.2", "lightningcss-linux-arm64-musl": "1.30.2", "lightningcss-linux-x64-gnu": "1.30.2", "lightningcss-linux-x64-musl": "1.30.2", "lightningcss-win32-arm64-msvc": "1.30.2", "lightningcss-win32-x64-msvc": "1.30.2" } }, "sha512-utfs7Pr5uJyyvDETitgsaqSyjCb2qNRAtuqUeWIAKztsOYdcACf2KtARYXg2pSvhkt+9NfoaNY7fxjl6nuMjIQ=="], + + "lightningcss-android-arm64": ["lightningcss-android-arm64@1.30.2", "", { "os": "android", "cpu": "arm64" }, "sha512-BH9sEdOCahSgmkVhBLeU7Hc9DWeZ1Eb6wNS6Da8igvUwAe0sqROHddIlvU06q3WyXVEOYDZ6ykBZQnjTbmo4+A=="], + + "lightningcss-darwin-arm64": ["lightningcss-darwin-arm64@1.30.2", "", { "os": "darwin", "cpu": "arm64" }, "sha512-ylTcDJBN3Hp21TdhRT5zBOIi73P6/W0qwvlFEk22fkdXchtNTOU4Qc37SkzV+EKYxLouZ6M4LG9NfZ1qkhhBWA=="], + + "lightningcss-darwin-x64": ["lightningcss-darwin-x64@1.30.2", "", { "os": "darwin", "cpu": "x64" }, "sha512-oBZgKchomuDYxr7ilwLcyms6BCyLn0z8J0+ZZmfpjwg9fRVZIR5/GMXd7r9RH94iDhld3UmSjBM6nXWM2TfZTQ=="], + + "lightningcss-freebsd-x64": ["lightningcss-freebsd-x64@1.30.2", "", { "os": "freebsd", "cpu": "x64" }, "sha512-c2bH6xTrf4BDpK8MoGG4Bd6zAMZDAXS569UxCAGcA7IKbHNMlhGQ89eRmvpIUGfKWNVdbhSbkQaWhEoMGmGslA=="], + + "lightningcss-linux-arm-gnueabihf": ["lightningcss-linux-arm-gnueabihf@1.30.2", "", { "os": "linux", "cpu": "arm" }, "sha512-eVdpxh4wYcm0PofJIZVuYuLiqBIakQ9uFZmipf6LF/HRj5Bgm0eb3qL/mr1smyXIS1twwOxNWndd8z0E374hiA=="], + + "lightningcss-linux-arm64-gnu": ["lightningcss-linux-arm64-gnu@1.30.2", "", { "os": "linux", "cpu": "arm64" }, "sha512-UK65WJAbwIJbiBFXpxrbTNArtfuznvxAJw4Q2ZGlU8kPeDIWEX1dg3rn2veBVUylA2Ezg89ktszWbaQnxD/e3A=="], + + "lightningcss-linux-arm64-musl": ["lightningcss-linux-arm64-musl@1.30.2", "", { "os": "linux", "cpu": "arm64" }, "sha512-5Vh9dGeblpTxWHpOx8iauV02popZDsCYMPIgiuw97OJ5uaDsL86cnqSFs5LZkG3ghHoX5isLgWzMs+eD1YzrnA=="], + + "lightningcss-linux-x64-gnu": ["lightningcss-linux-x64-gnu@1.30.2", "", { "os": "linux", "cpu": "x64" }, "sha512-Cfd46gdmj1vQ+lR6VRTTadNHu6ALuw2pKR9lYq4FnhvgBc4zWY1EtZcAc6EffShbb1MFrIPfLDXD6Xprbnni4w=="], + + "lightningcss-linux-x64-musl": ["lightningcss-linux-x64-musl@1.30.2", "", { "os": "linux", "cpu": "x64" }, "sha512-XJaLUUFXb6/QG2lGIW6aIk6jKdtjtcffUT0NKvIqhSBY3hh9Ch+1LCeH80dR9q9LBjG3ewbDjnumefsLsP6aiA=="], + + "lightningcss-win32-arm64-msvc": ["lightningcss-win32-arm64-msvc@1.30.2", "", { "os": "win32", "cpu": "arm64" }, "sha512-FZn+vaj7zLv//D/192WFFVA0RgHawIcHqLX9xuWiQt7P0PtdFEVaxgF9rjM/IRYHQXNnk61/H/gb2Ei+kUQ4xQ=="], + + "lightningcss-win32-x64-msvc": ["lightningcss-win32-x64-msvc@1.30.2", "", { "os": "win32", "cpu": "x64" }, "sha512-5g1yc73p+iAkid5phb4oVFMB45417DkRevRbt/El/gKXJk4jid+vPFF/AXbxn05Aky8PapwzZrdJShv5C0avjw=="], + + "locate-character": ["locate-character@3.0.0", "", {}, "sha512-SW13ws7BjaeJ6p7Q6CO2nchbYEc3X3J6WrmTTDto7yMPqVSZTUyY5Tjbid+Ab8gLnATtygYtiDIJGQRRn2ZOiA=="], + + "lodash.includes": ["lodash.includes@4.3.0", "", {}, "sha512-W3Bx6mdkRTGtlJISOvVD/lbqjTlPPUDTMnlXZFnVwi9NKJ6tiAk6LVdlhZMm17VZisqhKcgzpO5Wz91PCt5b0w=="], + + "lodash.isboolean": ["lodash.isboolean@3.0.3", "", {}, "sha512-Bz5mupy2SVbPHURB98VAcw+aHh4vRV5IPNhILUCsOzRmsTmSQ17jIuqopAentWoehktxGd9e/hbIXq980/1QJg=="], + + "lodash.isinteger": ["lodash.isinteger@4.0.4", "", {}, "sha512-DBwtEWN2caHQ9/imiNeEA5ys1JoRtRfY3d7V9wkqtbycnAmTvRRmbHKDV4a0EYc678/dia0jrte4tjYwVBaZUA=="], + + "lodash.isnumber": ["lodash.isnumber@3.0.3", "", {}, "sha512-QYqzpfwO3/CWf3XP+Z+tkQsfaLL/EnUlXWVkIk5FUPc4sBdTehEqZONuyRt2P67PXAk+NXmTBcc97zw9t1FQrw=="], + + "lodash.isplainobject": ["lodash.isplainobject@4.0.6", "", {}, "sha512-oSXzaWypCMHkPC3NvBEaPHf0KsA5mvPrOPgQWDsbg8n7orZ290M0BmC/jgRZ4vcJ6DTAhjrsSYgdsW/F+MFOBA=="], + + "lodash.isstring": ["lodash.isstring@4.0.1", "", {}, "sha512-0wJxfxH1wgO3GrbuP+dTTk7op+6L41QCXbGINEmD+ny/G/eCqGzxyCsh7159S+mgDDcoarnBw6PC1PS5+wUGgw=="], + + "lodash.once": ["lodash.once@4.1.1", "", {}, "sha512-Sb487aTOCr9drQVL8pIxOzVhafOjZN9UU54hiN8PU3uAiSV7lx1yYNpbNmex2PK6dSJoNTSJUUswT651yww3Mg=="], + + "maath": ["maath@0.10.8", "", { "peerDependencies": { "@types/three": ">=0.134.0", "three": ">=0.134.0" } }, "sha512-tRvbDF0Pgqz+9XUa4jjfgAQ8/aPKmQdWXilFu2tMy4GWj4NOsx99HlULO4IeREfbO3a0sA145DZYyvXPkybm0g=="], + + "magic-string": ["magic-string@0.30.21", "", { "dependencies": { "@jridgewell/sourcemap-codec": "^1.5.5" } }, "sha512-vd2F4YUyEXKGcLHoq+TEyCjxueSeHnFxyyjNp80yg0XV4vUhnDer/lvvlqM/arB5bXQN5K2/3oinyCRyx8T2CQ=="], + + "math-intrinsics": ["math-intrinsics@1.1.0", "", {}, "sha512-/IXtbwEk5HTPyEwyKX6hGkYXxM9nbj64B+ilVJnC/R6B0pH5G4V3b0pVbL7DBj4tkhBAppbQUlf6F6Xl9LHu1g=="], + + "media-typer": ["media-typer@1.1.0", "", {}, "sha512-aisnrDP4GNe06UcKFnV5bfMNPBUw4jsLGaWwWfnH3v02GnBuXX2MCVn5RbrWo0j3pczUilYblq7fQ7Nw2t5XKw=="], + + "memory-pager": ["memory-pager@1.5.0", "", {}, "sha512-ZS4Bp4r/Zoeq6+NLJpP+0Zzm0pR8whtGPf1XExKLJBAczGMnSi3It14OiNCStjQjM6NU1okjQGSxgEZN8eBYKg=="], + + "merge-descriptors": ["merge-descriptors@2.0.0", "", {}, "sha512-Snk314V5ayFLhp3fkUREub6WtjBfPdCPY1Ln8/8munuLuiYhsABgBVWsozAG+MWMbVEvcdcpbi9R7ww22l9Q3g=="], + + "meshoptimizer": ["meshoptimizer@0.22.0", "", {}, "sha512-IebiK79sqIy+E4EgOr+CAw+Ke8hAspXKzBd0JdgEmPHiAwmvEj2S4h1rfvo+o/BnfEYd/jAOg5IeeIjzlzSnDg=="], + + "mime-db": ["mime-db@1.54.0", "", {}, "sha512-aU5EJuIN2WDemCcAp2vFBfp/m4EAhWJnUNSSw0ixs7/kXbd6Pg64EmwJkNdFhB8aWt1sH2CTXrLxo/iAGV3oPQ=="], + + "mime-types": ["mime-types@3.0.2", "", { "dependencies": { "mime-db": "^1.54.0" } }, "sha512-Lbgzdk0h4juoQ9fCKXW4by0UJqj+nOOrI9MJ1sSj4nI8aI2eo1qmvQEie4VD1glsS250n15LsWsYtCugiStS5A=="], + + "mini-svg-data-uri": ["mini-svg-data-uri@1.4.4", "", { "bin": { "mini-svg-data-uri": "cli.js" } }, "sha512-r9deDe9p5FJUPZAk3A59wGH7Ii9YrjjWw0jmw/liSbHl2CHiyXj6FcDXDu2K3TjVAXqiJdaw3xxwlZZr9E6nHg=="], + + "mitt": ["mitt@3.0.1", "", {}, "sha512-vKivATfr97l2/QBCYAkXYDbrIWPM2IIKEl7YPhjCvKlG3kE2gm+uBo6nEXK3M5/Ffh/FLpKExzOQ3JJoJGFKBw=="], + + "mongodb": ["mongodb@7.0.0", "", { "dependencies": { "@mongodb-js/saslprep": "^1.3.0", "bson": "^7.0.0", "mongodb-connection-string-url": "^7.0.0" }, "peerDependencies": { "@aws-sdk/credential-providers": "^3.806.0", "@mongodb-js/zstd": "^7.0.0", "gcp-metadata": "^7.0.1", "kerberos": "^7.0.0", "mongodb-client-encryption": ">=7.0.0 <7.1.0", "snappy": "^7.3.2", "socks": "^2.8.6" }, "optionalPeers": ["@aws-sdk/credential-providers", "@mongodb-js/zstd", "gcp-metadata", "kerberos", "mongodb-client-encryption", "snappy", "socks"] }, "sha512-vG/A5cQrvGGvZm2mTnCSz1LUcbOPl83hfB6bxULKQ8oFZauyox/2xbZOoGNl+64m8VBrETkdGCDBdOsCr3F3jg=="], + + "mongodb-connection-string-url": ["mongodb-connection-string-url@7.0.0", "", { "dependencies": { "@types/whatwg-url": "^13.0.0", "whatwg-url": "^14.1.0" } }, "sha512-irhhjRVLE20hbkRl4zpAYLnDMM+zIZnp0IDB9akAFFUZp/3XdOfwwddc7y6cNvF2WCEtfTYRwYbIfYa2kVY0og=="], + + "mongoose": ["mongoose@9.0.2", "", { "dependencies": { "kareem": "3.0.0", "mongodb": "~7.0", "mpath": "0.9.0", "mquery": "6.0.0", "ms": "2.1.3", "sift": "17.1.3" } }, "sha512-+GCaqwE+X//yN9eo2M2L/n+mVti9J6vH5iQKbhD+2AArZd5iaZqK/DkmkE4S6/iYYMyVQPTXsRk7jyVOYEtJzA=="], + + "mpath": ["mpath@0.9.0", "", {}, "sha512-ikJRQTk8hw5DEoFVxHG1Gn9T/xcjtdnOKIU1JTmGjZZlg9LST2mBLmcX3/ICIbgJydT2GOc15RnNy5mHmzfSew=="], + + "mquery": ["mquery@6.0.0", "", {}, "sha512-b2KQNsmgtkscfeDgkYMcWGn9vZI9YoXh802VDEwE6qc50zxBFQ0Oo8ROkawbPAsXCY1/Z1yp0MagqsZStPWJjw=="], + + "mri": ["mri@1.2.0", "", {}, "sha512-tzzskb3bG8LvYGFF/mDTpq3jpI6Q9wc3LEmBaghu+DdCssd1FakN7Bc0hVNmEyGq1bq3RgfkCb3cmQLpNPOroA=="], + + "mrmime": ["mrmime@2.0.1", "", {}, "sha512-Y3wQdFg2Va6etvQ5I82yUhGdsKrcYox6p7FfL1LbK2J4V01F9TGlepTIhnK24t7koZibmg82KGglhA1XK5IsLQ=="], + + "ms": ["ms@2.1.3", "", {}, "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA=="], + + "nanoid": ["nanoid@3.3.11", "", { "bin": { "nanoid": "bin/nanoid.cjs" } }, "sha512-N8SpfPUnUp1bK+PMYW8qSWdl9U+wwNWI4QKxOYDy9JAro3WMX7p2OeVRF9v+347pnakNevPmiHhNmZ2HbFA76w=="], + + "negotiator": ["negotiator@1.0.0", "", {}, "sha512-8Ofs/AUQh8MaEcrlq5xOX0CQ9ypTF5dl78mjlMNfOK08fzpgTHQRQPBxcPlEtIw0yRpws+Zo/3r+5WRby7u3Gg=="], + + "object-assign": ["object-assign@4.1.1", "", {}, "sha512-rJgTQnkUnH1sFw8yT6VSU3zD3sWmu6sZhIseY8VX+GRu3P6F7Fu+JNDoXfklElbLJSnc3FUQHVe4cU5hj+BcUg=="], + + "object-inspect": ["object-inspect@1.13.4", "", {}, "sha512-W67iLl4J2EXEGTbfeHCffrjDfitvLANg0UlX3wFUUSTx92KXRFegMHUVgSqE+wvhAbi4WqjGg9czysTV2Epbew=="], + + "on-finished": ["on-finished@2.4.1", "", { "dependencies": { "ee-first": "1.1.1" } }, "sha512-oVlzkg3ENAhCk2zdv7IJwd/QUD4z2RxRwpkcGY8psCVcCYZNq4wYnVWALHM+brtuJjePWiYF/ClmuDr8Ch5+kg=="], + + "once": ["once@1.4.0", "", { "dependencies": { "wrappy": "1" } }, "sha512-lNaJgI+2Q5URQBkccEKHTQOPaXdUxnZZElQTZY0MFUAuaEqe1E+Nyvgdz/aIyNi6Z9MzO5dv1H8n58/GELp3+w=="], + + "parseurl": ["parseurl@1.3.3", "", {}, "sha512-CiyeOxFT/JZyN5m0z9PfXw4SCBJ6Sygz1Dpl0wqjlhDEGGBP1GnsUVEL0p63hoG1fcj3fHynXi9NYO4nWOL+qQ=="], + + "path-parse": ["path-parse@1.0.7", "", {}, "sha512-LDJzPVEEEPR+y48z93A0Ed0yXb8pAByGWo/k5YYdYgpY2/2EsOsksJrq7lOHxryrVOn1ejG6oAp8ahvOIQD8sw=="], + + "path-to-regexp": ["path-to-regexp@8.3.0", "", {}, "sha512-7jdwVIRtsP8MYpdXSwOS0YdD0Du+qOoF/AEPIt88PcCFrZCzx41oxku1jD88hZBwbNUIEfpqvuhjFaMAqMTWnA=="], + + "picocolors": ["picocolors@1.1.1", "", {}, "sha512-xceH2snhtb5M9liqDsmEw56le376mTZkEX/jEb/RxNFyegNul7eNslCXP9FDj/Lcu0X8KEyMceP2ntpaHrDEVA=="], + + "picomatch": ["picomatch@4.0.3", "", {}, "sha512-5gTmgEY/sqK6gFXLIsQNH19lWb4ebPDLA4SdLP7dsWkIXHWlG66oPuVvXSGFPppYZz8ZDZq0dYYrbHfBCVUb1Q=="], + + "postcss": ["postcss@8.5.6", "", { "dependencies": { "nanoid": "^3.3.11", "picocolors": "^1.1.1", "source-map-js": "^1.2.1" } }, "sha512-3Ybi1tAuwAP9s0r1UQ2J4n5Y0G05bJkpUIO0/bI9MhwmD70S5aTWbXGBwxHrelT+XM1k6dM0pk+SwNkpTRN7Pg=="], + + "postcss-selector-parser": ["postcss-selector-parser@6.0.10", "", { "dependencies": { "cssesc": "^3.0.0", "util-deprecate": "^1.0.2" } }, "sha512-IQ7TZdoaqbT+LCpShg46jnZVlhWD2w6iQYAcYXfHARZ7X1t/UGhhceQDs5X0cGqKvYlHNOuv7Oa1xmb0oQuA3w=="], + + "proxy-addr": ["proxy-addr@2.0.7", "", { "dependencies": { "forwarded": "0.2.0", "ipaddr.js": "1.9.1" } }, "sha512-llQsMLSUDUPT44jdrU/O37qlnifitDP+ZwrmmZcoSKyLKvtZxpyV0n2/bD/N4tBAAZ/gJEdZU7KMraoK1+XYAg=="], + + "punycode": ["punycode@2.3.1", "", {}, "sha512-vYt7UD1U9Wg6138shLtLOvdAu+8DsC/ilFtEVHcH+wydcSpNE20AfSOduf6MkRFahL5FY7X1oU7nKVZFtfq8Fg=="], + + "qs": ["qs@6.14.0", "", { "dependencies": { "side-channel": "^1.1.0" } }, "sha512-YWWTjgABSKcvs/nWBi9PycY/JiPJqOD4JA6o9Sej2AtvSGarXxKC3OQSk4pAarbdQlKAh5D4FCQkJNkW+GAn3w=="], + + "range-parser": ["range-parser@1.2.1", "", {}, "sha512-Hrgsx+orqoygnmhFbKaHE6c296J+HTAQXoxEF6gNupROmmGJRoyzfG3ccAveqCBrwr/2yxQ5BVd/GTl5agOwSg=="], + + "raw-body": ["raw-body@3.0.2", "", { "dependencies": { "bytes": "~3.1.2", "http-errors": "~2.0.1", "iconv-lite": "~0.7.0", "unpipe": "~1.0.0" } }, "sha512-K5zQjDllxWkf7Z5xJdV0/B0WTNqx6vxG70zJE4N0kBs4LovmEYWJzQGxC9bS9RAKu3bgM40lrd5zoLJ12MQ5BA=="], + + "readdirp": ["readdirp@4.1.2", "", {}, "sha512-GDhwkLfywWL2s6vEjyhri+eXmfH6j1L7JE27WhqLeYzoh/A3DBaYGEj2H/HFZCn/kMfim73FXxEJTw06WtxQwg=="], + + "require-from-string": ["require-from-string@2.0.2", "", {}, "sha512-Xf0nWe6RseziFMu+Ap9biiUbmplq6S9/p+7w7YXP/JBHhrUDDUhwa+vANyubuqfZWTveU//DYVGsDG7RKL/vEw=="], + + "resolve": ["resolve@1.22.11", "", { "dependencies": { "is-core-module": "^2.16.1", "path-parse": "^1.0.7", "supports-preserve-symlinks-flag": "^1.0.0" }, "bin": { "resolve": "bin/resolve" } }, "sha512-RfqAvLnMl313r7c9oclB1HhUEAezcpLjz95wFH4LVuhk9JF/r22qmVP9AMmOU4vMX7Q8pN8jwNg/CSpdFnMjTQ=="], + + "rollup": ["rollup@4.54.0", "", { "dependencies": { "@types/estree": "1.0.8" }, "optionalDependencies": { "@rollup/rollup-android-arm-eabi": "4.54.0", "@rollup/rollup-android-arm64": "4.54.0", "@rollup/rollup-darwin-arm64": "4.54.0", "@rollup/rollup-darwin-x64": "4.54.0", "@rollup/rollup-freebsd-arm64": "4.54.0", "@rollup/rollup-freebsd-x64": "4.54.0", "@rollup/rollup-linux-arm-gnueabihf": "4.54.0", "@rollup/rollup-linux-arm-musleabihf": "4.54.0", "@rollup/rollup-linux-arm64-gnu": "4.54.0", "@rollup/rollup-linux-arm64-musl": "4.54.0", "@rollup/rollup-linux-loong64-gnu": "4.54.0", "@rollup/rollup-linux-ppc64-gnu": "4.54.0", "@rollup/rollup-linux-riscv64-gnu": "4.54.0", "@rollup/rollup-linux-riscv64-musl": "4.54.0", "@rollup/rollup-linux-s390x-gnu": "4.54.0", "@rollup/rollup-linux-x64-gnu": "4.54.0", "@rollup/rollup-linux-x64-musl": "4.54.0", "@rollup/rollup-openharmony-arm64": "4.54.0", "@rollup/rollup-win32-arm64-msvc": "4.54.0", "@rollup/rollup-win32-ia32-msvc": "4.54.0", "@rollup/rollup-win32-x64-gnu": "4.54.0", "@rollup/rollup-win32-x64-msvc": "4.54.0", "fsevents": "~2.3.2" }, "bin": { "rollup": "dist/bin/rollup" } }, "sha512-3nk8Y3a9Ea8szgKhinMlGMhGMw89mqule3KWczxhIzqudyHdCIOHw8WJlj/r329fACjKLEh13ZSk7oE22kyeIw=="], + + "router": ["router@2.2.0", "", { "dependencies": { "debug": "^4.4.0", "depd": "^2.0.0", "is-promise": "^4.0.0", "parseurl": "^1.3.3", "path-to-regexp": "^8.0.0" } }, "sha512-nLTrUKm2UyiL7rlhapu/Zl45FwNgkZGaCpZbIHajDYgwlJCOzLSk+cIPAnsEqV955GjILJnKbdQC1nVPz+gAYQ=="], + + "sade": ["sade@1.8.1", "", { "dependencies": { "mri": "^1.1.0" } }, "sha512-xal3CZX1Xlo/k4ApwCFrHVACi9fBqJ7V+mwhBsuf/1IOKbBy098Fex+Wa/5QMubw09pSZ/u8EY8PWgevJsXp1A=="], + + "safe-buffer": ["safe-buffer@5.2.1", "", {}, "sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ=="], + + "safer-buffer": ["safer-buffer@2.1.2", "", {}, "sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg=="], + + "semver": ["semver@7.7.3", "", { "bin": { "semver": "bin/semver.js" } }, "sha512-SdsKMrI9TdgjdweUSR9MweHA4EJ8YxHn8DFaDisvhVlUOe4BF1tLD7GAj0lIqWVl+dPb/rExr0Btby5loQm20Q=="], + + "send": ["send@1.2.1", "", { "dependencies": { "debug": "^4.4.3", "encodeurl": "^2.0.0", "escape-html": "^1.0.3", "etag": "^1.8.1", "fresh": "^2.0.0", "http-errors": "^2.0.1", "mime-types": "^3.0.2", "ms": "^2.1.3", "on-finished": "^2.4.1", "range-parser": "^1.2.1", "statuses": "^2.0.2" } }, "sha512-1gnZf7DFcoIcajTjTwjwuDjzuz4PPcY2StKPlsGAQ1+YH20IRVrBaXSWmdjowTJ6u8Rc01PoYOGHXfP1mYcZNQ=="], + + "serve-static": ["serve-static@2.2.1", "", { "dependencies": { "encodeurl": "^2.0.0", "escape-html": "^1.0.3", "parseurl": "^1.3.3", "send": "^1.2.0" } }, "sha512-xRXBn0pPqQTVQiC8wyQrKs2MOlX24zQ0POGaj0kultvoOCstBQM5yvOhAVSUwOMjQtTvsPWoNCHfPGwaaQJhTw=="], + + "set-cookie-parser": ["set-cookie-parser@2.7.2", "", {}, "sha512-oeM1lpU/UvhTxw+g3cIfxXHyJRc/uidd3yK1P242gzHds0udQBYzs3y8j4gCCW+ZJ7ad0yctld8RYO+bdurlvw=="], + + "setprototypeof": ["setprototypeof@1.2.0", "", {}, "sha512-E5LDX7Wrp85Kil5bhZv46j8jOeboKq5JMmYM3gVGdGH8xFpPWXUMsNrlODCrkoxMEeNi/XZIwuRvY4XNwYMJpw=="], + + "side-channel": ["side-channel@1.1.0", "", { "dependencies": { "es-errors": "^1.3.0", "object-inspect": "^1.13.3", "side-channel-list": "^1.0.0", "side-channel-map": "^1.0.1", "side-channel-weakmap": "^1.0.2" } }, "sha512-ZX99e6tRweoUXqR+VBrslhda51Nh5MTQwou5tnUDgbtyM0dBgmhEDtWGP/xbKn6hqfPRHujUNwz5fy/wbbhnpw=="], + + "side-channel-list": ["side-channel-list@1.0.0", "", { "dependencies": { "es-errors": "^1.3.0", "object-inspect": "^1.13.3" } }, "sha512-FCLHtRD/gnpCiCHEiJLOwdmFP+wzCmDEkc9y7NsYxeF4u7Btsn1ZuwgwJGxImImHicJArLP4R0yX4c2KCrMrTA=="], + + "side-channel-map": ["side-channel-map@1.0.1", "", { "dependencies": { "call-bound": "^1.0.2", "es-errors": "^1.3.0", "get-intrinsic": "^1.2.5", "object-inspect": "^1.13.3" } }, "sha512-VCjCNfgMsby3tTdo02nbjtM/ewra6jPHmpThenkTYh8pG9ucZ/1P8So4u4FGBek/BjpOVsDCMoLA/iuBKIFXRA=="], + + "side-channel-weakmap": ["side-channel-weakmap@1.0.2", "", { "dependencies": { "call-bound": "^1.0.2", "es-errors": "^1.3.0", "get-intrinsic": "^1.2.5", "object-inspect": "^1.13.3", "side-channel-map": "^1.0.1" } }, "sha512-WPS/HvHQTYnHisLo9McqBHOJk2FkHO/tlpvldyrnem4aeQp4hai3gythswg6p01oSoTl58rcpiFAjF2br2Ak2A=="], + + "sift": ["sift@17.1.3", "", {}, "sha512-Rtlj66/b0ICeFzYTuNvX/EF1igRbbnGSvEyT79McoZa/DeGhMyC5pWKOEsZKnpkqtSeovd5FL/bjHWC3CIIvCQ=="], + + "sirv": ["sirv@3.0.2", "", { "dependencies": { "@polka/url": "^1.0.0-next.24", "mrmime": "^2.0.0", "totalist": "^3.0.0" } }, "sha512-2wcC/oGxHis/BoHkkPwldgiPSYcpZK3JU28WoMVv55yHJgcZ8rlXvuG9iZggz+sU1d4bRgIGASwyWqjxu3FM0g=="], + + "source-map-js": ["source-map-js@1.2.1", "", {}, "sha512-UXWMKhLOwVKb728IUtQPXxfYU+usdybtUrK/8uGE8CQMvrhOpwvzDBwj0QhSL7MQc7vIsISBG8VQ8+IDQxpfQA=="], + + "sparse-bitfield": ["sparse-bitfield@3.0.3", "", { "dependencies": { "memory-pager": "^1.0.2" } }, "sha512-kvzhi7vqKTfkh0PZU+2D2PIllw2ymqJKujUcyPMd9Y75Nv4nPbGJZXNhxsgdQab2BmlDct1YnfQCguEvHr7VsQ=="], + + "statuses": ["statuses@2.0.2", "", {}, "sha512-DvEy55V3DB7uknRo+4iOGT5fP1slR8wQohVdknigZPMpMstaKJQWhwiYBACJE3Ul2pTnATihhBYnRhZQHGBiRw=="], + + "supports-preserve-symlinks-flag": ["supports-preserve-symlinks-flag@1.0.0", "", {}, "sha512-ot0WnXS9fgdkgIcePe6RHNk1WA8+muPa6cSjeR3V8K27q9BB1rTE3R1p7Hv0z1ZyAc8s6Vvv8DIyWf681MAt0w=="], + + "svelte": ["svelte@5.46.1", "", { "dependencies": { "@jridgewell/remapping": "^2.3.4", "@jridgewell/sourcemap-codec": "^1.5.0", "@sveltejs/acorn-typescript": "^1.0.5", "@types/estree": "^1.0.5", "acorn": "^8.12.1", "aria-query": "^5.3.1", "axobject-query": "^4.1.0", "clsx": "^2.1.1", "devalue": "^5.5.0", "esm-env": "^1.2.1", "esrap": "^2.2.1", "is-reference": "^3.0.3", "locate-character": "^3.0.0", "magic-string": "^0.30.11", "zimmerframe": "^1.1.2" } }, "sha512-ynjfCHD3nP2el70kN5Pmg37sSi0EjOm9FgHYQdC4giWG/hzO3AatzXXJJgP305uIhGQxSufJLuYWtkY8uK/8RA=="], + + "svelte-check": ["svelte-check@4.3.5", "", { "dependencies": { "@jridgewell/trace-mapping": "^0.3.25", "chokidar": "^4.0.1", "fdir": "^6.2.0", "picocolors": "^1.0.0", "sade": "^1.7.4" }, "peerDependencies": { "svelte": "^4.0.0 || ^5.0.0-next.0", "typescript": ">=5.0.0" }, "bin": { "svelte-check": "bin/svelte-check" } }, "sha512-e4VWZETyXaKGhpkxOXP+B/d0Fp/zKViZoJmneZWe/05Y2aqSKj3YN2nLfYPJBQ87WEiY4BQCQ9hWGu9mPT1a1Q=="], + + "tailwindcss": ["tailwindcss@4.1.18", "", {}, "sha512-4+Z+0yiYyEtUVCScyfHCxOYP06L5Ne+JiHhY2IjR2KWMIWhJOYZKLSGZaP5HkZ8+bY0cxfzwDE5uOmzFXyIwxw=="], + + "tapable": ["tapable@2.3.0", "", {}, "sha512-g9ljZiwki/LfxmQADO3dEY1CbpmXT5Hm2fJ+QaGKwSXUylMybePR7/67YW7jOrrvjEgL1Fmz5kzyAjWVWLlucg=="], + + "three": ["three@0.182.0", "", {}, "sha512-GbHabT+Irv+ihI1/f5kIIsZ+Ef9Sl5A1Y7imvS5RQjWgtTPfPnZ43JmlYI7NtCRDK9zir20lQpfg8/9Yd02OvQ=="], + + "three-instanced-uniforms-mesh": ["three-instanced-uniforms-mesh@0.52.4", "", { "dependencies": { "troika-three-utils": "^0.52.4" }, "peerDependencies": { "three": ">=0.125.0" } }, "sha512-YwDBy05hfKZQtU+Rp0KyDf9yH4GxfhxMbVt9OYruxdgLfPwmDG5oAbGoW0DrKtKZSM3BfFcCiejiOHCjFBTeng=="], + + "three-mesh-bvh": ["three-mesh-bvh@0.9.4", "", { "peerDependencies": { "three": ">= 0.159.0" } }, "sha512-+y6xLS6k5LWkNNhYsTgKXBC2D9r/z0swiehVHYhZZ8AOhaKDRCWKsN94ctV5Xy7xA4Xbnv4LKYzf7epRLPT6oQ=="], + + "three-perf": ["three-perf@1.0.11", "", { "dependencies": { "troika-three-text": "^0.52.0", "tweakpane": "^3.1.10" }, "peerDependencies": { "three": ">=0.170" } }, "sha512-OgBpZjwL+csQKGKZjpkH/QHdbGFMxqngMbSEJeSnVNfXDYd6On7WHNv/GhUZH4YxIpNMwMahBWrNnsJvnbSJHQ=="], + + "three-viewport-gizmo": ["three-viewport-gizmo@2.2.0", "", { "peerDependencies": { "three": ">=0.162.0 <1.0.0" } }, "sha512-Jo9Liur1rUmdKk75FZumLU/+hbF+RtJHi1qsKZpntjKlCYScK6tjbYoqvJ9M+IJphrlQJF5oReFW7Sambh0N4Q=="], + + "tinyglobby": ["tinyglobby@0.2.15", "", { "dependencies": { "fdir": "^6.5.0", "picomatch": "^4.0.3" } }, "sha512-j2Zq4NyQYG5XMST4cbs02Ak8iJUdxRM0XI5QyxXuZOzKOINmWurp3smXu3y5wDcJrptwpSjgXHzIQxR0omXljQ=="], + + "toidentifier": ["toidentifier@1.0.1", "", {}, "sha512-o5sSPKEkg/DIQNmH43V0/uerLrpzVedkUh8tGNvaeXpfpuwjKenlSox/2O/BTlZUtEe+JG7s5YhEz608PlAHRA=="], + + "totalist": ["totalist@3.0.1", "", {}, "sha512-sf4i37nQ2LBx4m3wB74y+ubopq6W/dIzXg0FDGjsYnZHVa1Da8FH853wlL2gtUhg+xJXjfk3kUZS3BRoQeoQBQ=="], + + "tr46": ["tr46@5.1.1", "", { "dependencies": { "punycode": "^2.3.1" } }, "sha512-hdF5ZgjTqgAntKkklYw0R03MG2x/bSzTtkxmIRw/sTNV8YXsCJ1tfLAX23lhxhHJlEf3CRCOCGGWw3vI3GaSPw=="], + + "troika-three-text": ["troika-three-text@0.52.4", "", { "dependencies": { "bidi-js": "^1.0.2", "troika-three-utils": "^0.52.4", "troika-worker-utils": "^0.52.0", "webgl-sdf-generator": "1.1.1" }, "peerDependencies": { "three": ">=0.125.0" } }, "sha512-V50EwcYGruV5rUZ9F4aNsrytGdKcXKALjEtQXIOBfhVoZU9VAqZNIoGQ3TMiooVqFAbR1w15T+f+8gkzoFzawg=="], + + "troika-three-utils": ["troika-three-utils@0.52.4", "", { "peerDependencies": { "three": ">=0.125.0" } }, "sha512-NORAStSVa/BDiG52Mfudk4j1FG4jC4ILutB3foPnfGbOeIs9+G5vZLa0pnmnaftZUGm4UwSoqEpWdqvC7zms3A=="], + + "troika-worker-utils": ["troika-worker-utils@0.52.0", "", {}, "sha512-W1CpvTHykaPH5brv5VHLfQo9D1OYuo0cSBEUQFFT/nBUzM8iD6Lq2/tgG/f1OelbAS1WtaTPQzE5uM49egnngw=="], + + "tweakpane": ["tweakpane@3.1.10", "", {}, "sha512-rqwnl/pUa7+inhI2E9ayGTqqP0EPOOn/wVvSWjZsRbZUItzNShny7pzwL3hVlaN4m9t/aZhsP0aFQ9U5VVR2VQ=="], + + "type-is": ["type-is@2.0.1", "", { "dependencies": { "content-type": "^1.0.5", "media-typer": "^1.1.0", "mime-types": "^3.0.0" } }, "sha512-OZs6gsjF4vMp32qrCbiVSkrFmXtG/AZhY3t0iAMrMBiAZyV9oALtXO8hsrHbMXF9x6L3grlFuwW2oAz7cav+Gw=="], + + "typescript": ["typescript@5.9.3", "", { "bin": { "tsc": "bin/tsc", "tsserver": "bin/tsserver" } }, "sha512-jl1vZzPDinLr9eUt3J/t7V6FgNEw9QjvBPdysz9KfQDD41fQrC2Y4vKQdiaUpFT4bXlb1RHhLpp8wtm6M5TgSw=="], + + "undici-types": ["undici-types@7.16.0", "", {}, "sha512-Zz+aZWSj8LE6zoxD+xrjh4VfkIG8Ya6LvYkZqtUQGJPZjYl53ypCaUwWqo7eI0x66KBGeRo+mlBEkMSeSZ38Nw=="], + + "unpipe": ["unpipe@1.0.0", "", {}, "sha512-pjy2bYhSsufwWlKwPc+l3cN7+wuJlK6uz0YdJEOlQDbl6jo/YlPi4mb8agUkVC8BF7V8NuzeyPNqRksA3hztKQ=="], + + "util-deprecate": ["util-deprecate@1.0.2", "", {}, "sha512-EPD5q1uXyFxJpCrLnCc1nHnq3gOa6DZBocAIiI2TaSCA7VCJ1UJDMagCzIkXNsUYfD1daK//LTEQ8xiIbrHtcw=="], + + "vary": ["vary@1.1.2", "", {}, "sha512-BNGbWLfd0eUPabhkXUVm0j8uuvREyTh5ovRa/dyow/BqAbZJyC+5fU+IzQOzmAKzYqYRAISoRhdQr3eIZ/PXqg=="], + + "vite": ["vite@7.3.0", "", { "dependencies": { "esbuild": "^0.27.0", "fdir": "^6.5.0", "picomatch": "^4.0.3", "postcss": "^8.5.6", "rollup": "^4.43.0", "tinyglobby": "^0.2.15" }, "optionalDependencies": { "fsevents": "~2.3.3" }, "peerDependencies": { "@types/node": "^20.19.0 || >=22.12.0", "jiti": ">=1.21.0", "less": "^4.0.0", "lightningcss": "^1.21.0", "sass": "^1.70.0", "sass-embedded": "^1.70.0", "stylus": ">=0.54.8", "sugarss": "^5.0.0", "terser": "^5.16.0", "tsx": "^4.8.1", "yaml": "^2.4.2" }, "optionalPeers": ["@types/node", "jiti", "less", "lightningcss", "sass", "sass-embedded", "stylus", "sugarss", "terser", "tsx", "yaml"], "bin": { "vite": "bin/vite.js" } }, "sha512-dZwN5L1VlUBewiP6H9s2+B3e3Jg96D0vzN+Ry73sOefebhYr9f94wwkMNN/9ouoU8pV1BqA1d1zGk8928cx0rg=="], + + "vitefu": ["vitefu@1.1.1", "", { "peerDependencies": { "vite": "^3.0.0 || ^4.0.0 || ^5.0.0 || ^6.0.0 || ^7.0.0-beta.0" }, "optionalPeers": ["vite"] }, "sha512-B/Fegf3i8zh0yFbpzZ21amWzHmuNlLlmJT6n7bu5e+pCHUKQIfXSYokrqOBGEMMe9UG2sostKQF9mml/vYaWJQ=="], + + "webgl-sdf-generator": ["webgl-sdf-generator@1.1.1", "", {}, "sha512-9Z0JcMTFxeE+b2x1LJTdnaT8rT8aEp7MVxkNwoycNmJWwPdzoXzMh0BjJSh/AEFP+KPYZUli814h8bJZFIZ2jA=="], + + "webidl-conversions": ["webidl-conversions@7.0.0", "", {}, "sha512-VwddBukDzu71offAQR975unBIGqfKZpM+8ZX6ySk8nYhVoo5CYaZyzt3YBvYtRtO+aoGlqxPg/B87NGVZ/fu6g=="], + + "whatwg-url": ["whatwg-url@14.2.0", "", { "dependencies": { "tr46": "^5.1.0", "webidl-conversions": "^7.0.0" } }, "sha512-De72GdQZzNTUBBChsXueQUnPKDkg/5A5zp7pFDuQAj5UFoENpiACU0wlCvzpAGnTkj++ihpKwKyYewn/XNUbKw=="], + + "wrappy": ["wrappy@1.0.2", "", {}, "sha512-l4Sp/DRseor9wL6EvV2+TuQn63dMkPjZ/sp9XkghTEbV9KlPS1xUsZ3u7/IQO4wxtcFB4bgpQPRcR3QCvezPcQ=="], + + "zimmerframe": ["zimmerframe@1.1.4", "", {}, "sha512-B58NGBEoc8Y9MWWCQGl/gq9xBCe4IiKM0a2x7GZdQKOW5Exr8S1W24J6OgM1njK8xCRGvAJIL/MxXHf6SkmQKQ=="], + + "@rollup/plugin-commonjs/is-reference": ["is-reference@1.2.1", "", { "dependencies": { "@types/estree": "*" } }, "sha512-U82MsXXiFIrjCK4otLT+o2NA2Cd2g5MLoOVXUZjIOhLurrRxpEXzI8O0KZHr3IjLvlAH1kTPYSuqer5T9ZVBKQ=="], + + "@sveltejs/kit/@types/cookie": ["@types/cookie@0.6.0", "", {}, "sha512-4Kh9a6B2bQciAhf7FSuMRRkUWecJgJu9nPnx3yzpsfXX/c50REIqpHY4C82bXP90qrLtXtkDxTZosYO3UpOwlA=="], + + "@sveltejs/kit/cookie": ["cookie@0.6.0", "", {}, "sha512-U71cyTamuh1CRNCfpGY6to28lxvNwPG4Guz/EVjgf3Jmzv0vlDp1atT9eS5dDjMYHucpHbWns6Lwf3BKz6svdw=="], + + "@tailwindcss/oxide-wasm32-wasi/@emnapi/core": ["@emnapi/core@1.7.1", "", { "dependencies": { "@emnapi/wasi-threads": "1.1.0", "tslib": "^2.4.0" }, "bundled": true }, "sha512-o1uhUASyo921r2XtHYOHy7gdkGLge8ghBEQHMWmyJFoXlpU58kIrhhN3w26lpQb6dspetweapMn2CSNwQ8I4wg=="], + + "@tailwindcss/oxide-wasm32-wasi/@emnapi/runtime": ["@emnapi/runtime@1.7.1", "", { "dependencies": { "tslib": "^2.4.0" }, "bundled": true }, "sha512-PVtJr5CmLwYAU9PZDMITZoR5iAOShYREoR45EyyLrbntV50mdePTgUn4AmOw90Ifcj+x2kRjdzr1HP3RrNiHGA=="], + + "@tailwindcss/oxide-wasm32-wasi/@emnapi/wasi-threads": ["@emnapi/wasi-threads@1.1.0", "", { "dependencies": { "tslib": "^2.4.0" }, "bundled": true }, "sha512-WI0DdZ8xFSbgMjR1sFsKABJ/C5OnRrjT06JXbZKexJGrDuPTzZdDYfFlsgcCXCyf+suG5QU2e/y1Wo2V/OapLQ=="], + + "@tailwindcss/oxide-wasm32-wasi/@napi-rs/wasm-runtime": ["@napi-rs/wasm-runtime@1.1.0", "", { "dependencies": { "@emnapi/core": "^1.7.1", "@emnapi/runtime": "^1.7.1", "@tybys/wasm-util": "^0.10.1" }, "bundled": true }, "sha512-Fq6DJW+Bb5jaWE69/qOE0D1TUN9+6uWhCeZpdnSBk14pjLcCWR7Q8n49PTSPHazM37JqrsdpEthXy2xn6jWWiA=="], + + "@tailwindcss/oxide-wasm32-wasi/@tybys/wasm-util": ["@tybys/wasm-util@0.10.1", "", { "dependencies": { "tslib": "^2.4.0" }, "bundled": true }, "sha512-9tTaPJLSiejZKx+Bmog4uSubteqTvFrVrURwkmHixBo0G4seD0zUxp98E1DzUBJxLQ3NPwXrGKDiVjwx/DpPsg=="], + + "@tailwindcss/oxide-wasm32-wasi/tslib": ["tslib@2.8.1", "", { "bundled": true }, "sha512-oJFu94HQb+KVduSUQL7wnpmqnfmLsOA/nAh6b6EH0wCEoK0/mPeXU6c3wKDV83MkOuHPRHtSXKKU99IBazS/2w=="], + + "express/cookie": ["cookie@0.7.2", "", {}, "sha512-yki5XnKuf750l50uGTllt6kKILY4nQ1eNIQatoXEByZ5dWgnKqbnqmTrBE5B4N7lrMJKQ2ytWMiTO2o0v6Ew/w=="], + } +} diff --git a/package.json b/package.json new file mode 100644 index 0000000..7573e78 --- /dev/null +++ b/package.json @@ -0,0 +1,43 @@ +{ + "name": "filaprint", + "private": true, + "version": "0.0.1", + "type": "module", + "scripts": { + "dev": "vite dev --host", + "build": "vite build", + "start": "node server/server.js" + }, + "devDependencies": { + "@sveltejs/adapter-auto": "^7.0.0", + "@sveltejs/kit": "^2.49.1", + "@sveltejs/vite-plugin-svelte": "^6.2.1", + "@tailwindcss/forms": "^0.5.10", + "@tailwindcss/typography": "^0.5.19", + "@tailwindcss/vite": "^4.1.17", + "@types/bcryptjs": "^3.0.0", + "@types/cookie": "^1.0.0", + "@types/jsonwebtoken": "^9.0.10", + "svelte": "^5.45.6", + "svelte-check": "^4.3.4", + "tailwindcss": "^4.1.17", + "typescript": "^5.9.3", + "vite": "^7.2.6" + }, + "dependencies": { + "@iconify/svelte": "^5.1.0", + "@sveltejs/adapter-node": "^5.4.0", + "@threlte/core": "^8.3.1", + "@threlte/extras": "^9.7.1", + "@types/three": "^0.182.0", + "bcryptjs": "^3.0.3", + "chart.js": "^4.5.1", + "cookie": "^1.1.1", + "cors": "^2.8.5", + "dotenv": "^17.2.3", + "express": "^5.2.1", + "jsonwebtoken": "^9.0.3", + "mongoose": "^9.0.2", + "three": "^0.182.0" + } +} diff --git a/server/db.js b/server/db.js new file mode 100644 index 0000000..e962f93 --- /dev/null +++ b/server/db.js @@ -0,0 +1,16 @@ +import mongoose from 'mongoose'; +import dotenv from 'dotenv'; + +dotenv.config(); + +const MONGODB_URI = process.env.MONGODB_URI || 'mongodb://localhost:27017/filaprint'; + +export const connectDB = async () => { + try { + await mongoose.connect(MONGODB_URI); + console.log('MongoDB connected successfully'); + } catch (error) { + console.error('MongoDB connection error:', error); + process.exit(1); + } +}; diff --git a/server/server.js b/server/server.js new file mode 100644 index 0000000..7f56981 --- /dev/null +++ b/server/server.js @@ -0,0 +1,23 @@ +import { handler } from '../build/handler.js'; +import dotenv from 'dotenv'; +import http from 'http'; +import express from 'express'; +import cors from 'cors'; +import { connectDB } from './db.js'; + +dotenv.config(); + +const app = express(); +const server = http.Server(app); + +app.use(cors()); + +// Connect to Database +connectDB(); + +app.use(handler); + +const PORT = process.env.PORT || 3000; +server.listen(PORT, () => { + console.log('listening on port http://localhost:' + PORT); +}); \ No newline at end of file diff --git a/src/app.css b/src/app.css new file mode 100644 index 0000000..4148d54 --- /dev/null +++ b/src/app.css @@ -0,0 +1,137 @@ +@import "tailwindcss"; + +/* + Cerberus Theme - Exact Implementation + Using the official Cerberus theme variables +*/ + +:root { + /* Spacing & Typography */ + --spacing: 0.25rem; + --text-scaling: 1.067; + + /* Font Family - JetBrains Mono */ + --base-font-family: "JetBrains Mono", system-ui, sans-serif; + + /* Primary Colors */ + --color-primary-50: oklch(0.92 0.04 257.51); + --color-primary-100: oklch(0.84 0.08 254.62); + --color-primary-200: oklch(0.77 0.11 254.28); + --color-primary-300: oklch(0.7 0.15 254.36); + --color-primary-400: oklch(0.63 0.19 255.71); + --color-primary-500: oklch(0.57 0.21 258.29); + --color-primary-600: oklch(0.52 0.19 258.15); + --color-primary-700: oklch(0.46 0.17 257.78); + --color-primary-800: oklch(0.4 0.14 257.62); + --color-primary-900: oklch(0.34 0.11 257.14); + --color-primary-950: oklch(0.28 0.08 257.49); + + /* Secondary Colors */ + --color-secondary-50: oklch(0.87 0.05 300.12); + --color-secondary-100: oklch(0.79 0.09 303.55); + --color-secondary-200: oklch(0.7 0.13 304.43); + --color-secondary-300: oklch(0.63 0.17 303.8); + --color-secondary-400: oklch(0.55 0.2 302.74); + --color-secondary-500: oklch(0.49 0.23 300.45); + --color-secondary-600: oklch(0.45 0.21 299.59); + --color-secondary-700: oklch(0.42 0.19 298.25); + --color-secondary-800: oklch(0.38 0.17 296.27); + --color-secondary-900: oklch(0.34 0.15 293.96); + --color-secondary-950: oklch(0.3 0.13 291.15); + + /* Tertiary Colors */ + --color-tertiary-50: oklch(0.91 0.08 328.89); + --color-tertiary-100: oklch(0.83 0.13 339.66); + --color-tertiary-200: oklch(0.76 0.18 345.54); + --color-tertiary-300: oklch(0.7 0.23 350.67); + --color-tertiary-400: oklch(0.66 0.25 355.84); + --color-tertiary-500: oklch(0.65 0.26 2.47); + --color-tertiary-600: oklch(0.59 0.24 1.69); + --color-tertiary-700: oklch(0.54 0.22 0.5); + --color-tertiary-800: oklch(0.48 0.2 359.65); + --color-tertiary-900: oklch(0.43 0.17 357.7); + --color-tertiary-950: oklch(0.37 0.15 355.33); + + /* Success Colors */ + --color-success-50: oklch(0.94 0.09 178.68); + --color-success-500: oklch(0.83 0.13 174.96); + --color-success-950: oklch(0.27 0.04 185.3); + + /* Warning Colors */ + --color-warning-50: oklch(0.96 0.05 84.57); + --color-warning-500: oklch(0.82 0.14 76.72); + --color-warning-950: oklch(0.52 0.13 51.44); + + /* Error Colors */ + --color-error-50: oklch(0.9 0.04 14); + --color-error-500: oklch(0.64 0.22 28.71); + --color-error-950: oklch(0.42 0.17 29.23); + + /* Surface Colors - Cerberus Exact */ + --color-surface-50: oklch(0.99 0 0); + --color-surface-100: oklch(0.91 0 0); + --color-surface-200: oklch(0.81 0 0); + --color-surface-300: oklch(0.72 0 0); + --color-surface-400: oklch(0.62 0 0); + --color-surface-500: oklch(0.51 0 0); + --color-surface-600: oklch(0.45 0 0); + --color-surface-700: oklch(0.39 0 0); + --color-surface-800: oklch(0.32 0 0); + --color-surface-900: oklch(0.25 0 0); + --color-surface-950: oklch(0.18 0 0); + + /* Semantic Colors */ + --base-font-color: var(--color-surface-950); + --base-font-color-dark: var(--color-surface-50); + --body-background-color: var(--color-surface-50); + --body-background-color-dark: var(--color-surface-950); +} + +@font-face { + font-family: "JetBrains Mono"; + src: url("/font/JetBrainsMono-Regular.ttf") format("truetype"); + font-weight: normal; + font-style: normal; +} + +@theme { + --font-mono: "JetBrains Mono", monospace; + --font-display: "JetBrains Mono", monospace; + --font-body: "JetBrains Mono", monospace; + + /* Register Theme Colors for Tailwind */ + --color-background: var(--body-background-color-dark); + --color-surface: var(--color-surface-900); + + --color-primary: var(--color-primary-500); + --color-secondary: var(--color-secondary-500); + --color-accent: var(--color-tertiary-500); + + --color-success: var(--color-success-500); + --color-warning: var(--color-warning-500); + --color-danger: var(--color-error-500); + + --color-text-main: var(--base-font-color-dark); + --color-text-muted: var(--color-surface-400); +} + +body { + background-color: var(--body-background-color-dark); + color: var(--base-font-color-dark); + font-family: var(--base-font-family); + @apply antialiased min-h-screen selection:bg-primary selection:text-white; +} + +/* Card Utilities */ +.glass-card { + background-color: var(--color-surface-900); + border: 1px solid var(--color-surface-700); + @apply rounded-xl shadow-lg transition-all duration-300; +} + +.glass-card:hover { + background-color: var(--color-surface-800); + border-color: var(--color-primary-500); + transform: translateY(-2px); + box-shadow: 0 10px 30px -10px rgba(0, 0, 0, 0.5); +} diff --git a/src/app.d.ts b/src/app.d.ts new file mode 100644 index 0000000..e678134 --- /dev/null +++ b/src/app.d.ts @@ -0,0 +1,13 @@ +declare global { + namespace App { + // interface Error {} + interface Locals { + user: { id: string; username: string; role: string } | null; + } + // interface PageData {} + // interface PageState {} + // interface Platform {} + } +} + +export {}; diff --git a/src/app.html b/src/app.html new file mode 100644 index 0000000..f273cc5 --- /dev/null +++ b/src/app.html @@ -0,0 +1,11 @@ + + + + + + %sveltekit.head% + + +
%sveltekit.body%
+ + diff --git a/src/hooks.server.ts b/src/hooks.server.ts new file mode 100644 index 0000000..f311aba --- /dev/null +++ b/src/hooks.server.ts @@ -0,0 +1,43 @@ +import { connectDB } from '$lib/server/db'; +import { User } from '$lib/models/User'; +import type { Handle } from '@sveltejs/kit'; +import jwt from 'jsonwebtoken'; + +const JWT_SECRET = process.env.JWT_SECRET || 'fallback_secret_change_me'; + +export const handle: Handle = async ({ event, resolve }) => { + // 1. Connect to DB on every request (ensure connection) + await connectDB(); + + // 2. Auth Check + const token = event.cookies.get('session'); + + if (token) { + try { + const decoded = jwt.verify(token, JWT_SECRET) as { id: string; username: string; role: string }; + // Optional: Fetch full user if needed, but token payload is faster + event.locals.user = { id: decoded.id, username: decoded.username, role: decoded.role }; + } catch (err) { + // Invalid token + event.cookies.delete('session', { path: '/' }); + event.locals.user = null; + } + } else { + event.locals.user = null; + } + + // 3. Route Protection + if (!event.locals.user) { + if (!event.url.pathname.startsWith('/login') && !event.url.pathname.startsWith('/register')) { + return new Response('Redirect', { status: 303, headers: { Location: '/login' } }); + } + } else { + // If logged in and trying to go to login/register, redirect to dashboard + if (event.url.pathname.startsWith('/login') || event.url.pathname.startsWith('/register')) { + return new Response('Redirect', { status: 303, headers: { Location: '/' } }); + } + } + + const response = await resolve(event); + return response; +}; diff --git a/src/lib/assets/favicon.svg b/src/lib/assets/favicon.svg new file mode 100644 index 0000000..cc5dc66 --- /dev/null +++ b/src/lib/assets/favicon.svg @@ -0,0 +1 @@ +svelte-logo \ No newline at end of file diff --git a/src/lib/components/3d/Spool3D.svelte b/src/lib/components/3d/Spool3D.svelte new file mode 100644 index 0000000..66cc636 --- /dev/null +++ b/src/lib/components/3d/Spool3D.svelte @@ -0,0 +1,56 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/src/lib/components/Navbar.svelte b/src/lib/components/Navbar.svelte new file mode 100644 index 0000000..807b7fb --- /dev/null +++ b/src/lib/components/Navbar.svelte @@ -0,0 +1,201 @@ + + + + + + +
+ + Filaprint + + +
+ + +{#if mobileMenuOpen} + + +
+ +
+ {#each links as link} + + + {link.name} + + {/each} + +
+ {#if $page.data.user} + +
+
+

{$page.data.user.username}

+

+ {$page.data.user.role || "Maker"} +

+
+
+
+ +
+ {:else} + + + Sign In + + {/if} +
+
+{/if} + + diff --git a/src/lib/components/prints/EditPrintModal.svelte b/src/lib/components/prints/EditPrintModal.svelte new file mode 100644 index 0000000..79578b9 --- /dev/null +++ b/src/lib/components/prints/EditPrintModal.svelte @@ -0,0 +1,259 @@ + + + + {#if print} +
{ + isSubmitting = true; + return async ({ update }) => { + await update(); + isSubmitting = false; + handleClose(); + }; + }} + class="space-y-4" + > + + + +
+ + +
+ + + + +
+
+ + + + {#if editStatus === "In Progress"} + +
+
+ + + +
+
+ + + +
+
+ +
+

+ + Update the total print time and how long it's been running. +

+
+ + +
+
+ + +
+
+ {:else} + +
+ + + +
+ {/if} + +
+ +
+ + +
+
+
+ {/if} +
diff --git a/src/lib/components/prints/LogPrintModal.svelte b/src/lib/components/prints/LogPrintModal.svelte new file mode 100644 index 0000000..5fae20d --- /dev/null +++ b/src/lib/components/prints/LogPrintModal.svelte @@ -0,0 +1,227 @@ + + + +
{ + isSubmitting = true; + return async ({ update }) => { + await update(); + isSubmitting = false; + handleClose(); + }; + }} + class="space-y-4" + > + +
+ + +
+ + + + +
+
+ + + +
+
+ + + +
+
+ + + +
+
+ + {#if selectedStatus === "In Progress"} + +
+

+ + Enter the expected total print time and how long it's been running. +

+
+ + +
+
+ + +
+
+ {:else} + +
+ + + +
+ {/if} + +
+ + +
+
+
diff --git a/src/lib/components/ui/Button.svelte b/src/lib/components/ui/Button.svelte new file mode 100644 index 0000000..965d61e --- /dev/null +++ b/src/lib/components/ui/Button.svelte @@ -0,0 +1,50 @@ + + + diff --git a/src/lib/components/ui/Card.svelte b/src/lib/components/ui/Card.svelte new file mode 100644 index 0000000..c83c0fb --- /dev/null +++ b/src/lib/components/ui/Card.svelte @@ -0,0 +1,12 @@ + + +
+ {@render children()} +
diff --git a/src/lib/components/ui/Input.svelte b/src/lib/components/ui/Input.svelte new file mode 100644 index 0000000..40d752e --- /dev/null +++ b/src/lib/components/ui/Input.svelte @@ -0,0 +1,39 @@ + + +
+ + + {#if error} +

{error}

+ {/if} +
diff --git a/src/lib/components/ui/Modal.svelte b/src/lib/components/ui/Modal.svelte new file mode 100644 index 0000000..fa70f6c --- /dev/null +++ b/src/lib/components/ui/Modal.svelte @@ -0,0 +1,54 @@ + + +{#if open} +
+ + + +
+ + +
+ +
+

{title}

+ +
+ + +
+ {@render children()} +
+
+
+{/if} diff --git a/src/lib/index.ts b/src/lib/index.ts new file mode 100644 index 0000000..856f2b6 --- /dev/null +++ b/src/lib/index.ts @@ -0,0 +1 @@ +// place files you want to import through the `$lib` alias in this folder. diff --git a/src/lib/models/PrintJob.ts b/src/lib/models/PrintJob.ts new file mode 100644 index 0000000..a25f68e --- /dev/null +++ b/src/lib/models/PrintJob.ts @@ -0,0 +1,58 @@ +import mongoose from 'mongoose'; + +const printJobSchema = new mongoose.Schema({ + user_id: { + type: mongoose.Schema.Types.ObjectId, + ref: 'User', + required: true + }, + spool_id: { + type: mongoose.Schema.Types.ObjectId, + ref: 'Spool', + required: true + }, + printer_id: { + type: mongoose.Schema.Types.ObjectId, + ref: 'Printer', + required: true + }, + name: { + type: String, + required: true, + default: 'Untitled Print' + }, + duration_minutes: { + type: Number, + required: true + }, + filament_used_g: { + type: Number, + required: true + }, + filament_used_m: { + type: Number + }, + calculated_cost_filament: { + type: Number + }, + calculated_cost_energy: { + type: Number + }, + status: { + type: String, + enum: ['In Progress', 'Success', 'Fail', 'Cancelled'], + default: 'Success' + }, + started_at: { + type: Date + }, + notes: String, + date: { + type: Date, + default: Date.now + } +}, { + timestamps: true +}); + +export const PrintJob = mongoose.models.PrintJob || mongoose.model('PrintJob', printJobSchema); diff --git a/src/lib/models/Printer.ts b/src/lib/models/Printer.ts new file mode 100644 index 0000000..789be4a --- /dev/null +++ b/src/lib/models/Printer.ts @@ -0,0 +1,34 @@ +import mongoose from 'mongoose'; + +const printerSchema = new mongoose.Schema({ + user_id: { + type: mongoose.Schema.Types.ObjectId, + ref: 'User', + required: true + }, + name: { + type: String, + required: true, + trim: true + }, + model: { + type: String, + trim: true + }, + nozzle_diameter_mm: { + type: Number, + default: 0.4 + }, + power_consumption_watts: { + type: Number, + default: 0 // Used for energy calculation + }, + bed_size_x_mm: Number, + bed_size_y_mm: Number, + bed_size_z_mm: Number, + image_url: String, // Optional URL for printer image +}, { + timestamps: true +}); + +export const Printer = mongoose.models.Printer || mongoose.model('Printer', printerSchema); diff --git a/src/lib/models/Spool.ts b/src/lib/models/Spool.ts new file mode 100644 index 0000000..2b23430 --- /dev/null +++ b/src/lib/models/Spool.ts @@ -0,0 +1,51 @@ +import mongoose from 'mongoose'; + +const spoolSchema = new mongoose.Schema({ + user_id: { + type: mongoose.Schema.Types.ObjectId, + ref: 'User', + required: true + }, + brand: { + type: String, + required: true, + trim: true + }, + material: { + type: String, + required: true, + enum: ['PLA', 'PETG', 'ABS', 'ASA', 'TPU', 'Nylon', 'PC', 'Other'], + default: 'PLA' + }, + color_hex: { + type: String, + default: '#ffffff' + }, + weight_initial_g: { + type: Number, + required: true, + min: 0 + }, + weight_remaining_g: { + type: Number, + required: true, + min: 0 + }, + price: { + type: Number, + min: 0 + }, + purchased_at: { + type: Date, + default: Date.now + }, + is_active: { + type: Boolean, + default: true + } +}, { + timestamps: true +}); + +// Check if model already exists to prevent overwrite error in HMR/dev mode +export const Spool = mongoose.models.Spool || mongoose.model('Spool', spoolSchema); diff --git a/src/lib/models/User.ts b/src/lib/models/User.ts new file mode 100644 index 0000000..e5e93d0 --- /dev/null +++ b/src/lib/models/User.ts @@ -0,0 +1,37 @@ +import mongoose from 'mongoose'; + +const userSchema = new mongoose.Schema({ + username: { + type: String, + required: true, + unique: true, + trim: true, + minlength: 3 + }, + password: { + type: String, + required: true + }, + role: { + type: String, + default: 'Maker' + }, + location: { + type: String, + default: '' + }, + electricity_rate: { + type: Number, + default: 0.12 // Default rate in $/kWh + }, + currency: { + type: String, + default: 'USD' + }, + createdAt: { + type: Date, + default: Date.now + } +}); + +export const User = mongoose.models.User || mongoose.model('User', userSchema); diff --git a/src/lib/server/db.ts b/src/lib/server/db.ts new file mode 100644 index 0000000..d21a61b --- /dev/null +++ b/src/lib/server/db.ts @@ -0,0 +1,17 @@ +import mongoose from 'mongoose'; +import { env } from '$env/dynamic/private'; + +const MONGODB_URI = env.MONGODB_URI || 'mongodb://localhost:27017/filaprint'; + +export const connectDB = async () => { + if (mongoose.connection.readyState >= 1) { + return; + } + + try { + await mongoose.connect(MONGODB_URI); + console.log('MongoDB connected successfully via SvelteKit'); + } catch (error) { + console.error('MongoDB connection error:', error); + } +}; diff --git a/src/routes/+layout.server.ts b/src/routes/+layout.server.ts new file mode 100644 index 0000000..01c4b28 --- /dev/null +++ b/src/routes/+layout.server.ts @@ -0,0 +1,7 @@ +import type { LayoutServerLoad } from './$types'; + +export const load: LayoutServerLoad = async ({ locals }) => { + return { + user: locals.user + }; +}; diff --git a/src/routes/+layout.svelte b/src/routes/+layout.svelte new file mode 100644 index 0000000..ace5697 --- /dev/null +++ b/src/routes/+layout.svelte @@ -0,0 +1,21 @@ + + + + +
+ + +
+
+ {@render children()} +
+
+
diff --git a/src/routes/+page.server.ts b/src/routes/+page.server.ts new file mode 100644 index 0000000..b2d9cf4 --- /dev/null +++ b/src/routes/+page.server.ts @@ -0,0 +1,65 @@ +import { Spool } from '$lib/models/Spool'; +import { Printer } from '$lib/models/Printer'; +import { PrintJob } from '$lib/models/PrintJob'; +import { connectDB } from '$lib/server/db'; +import type { PageServerLoad } from './$types'; +import { redirect } from '@sveltejs/kit'; + +export const load: PageServerLoad = async ({ locals }) => { + if (!locals.user) throw redirect(303, '/login'); + + await connectDB(); + + const userId = locals.user.id; + + // Run in parallel - all filtered by user + const [spoolCount, spools, printerCount, recentPrints, activePrinter, activePrintJob, completedPrints] = await Promise.all([ + Spool.countDocuments({ user_id: userId, is_active: true }), + Spool.find({ user_id: userId, is_active: true }).lean(), + Printer.countDocuments({ user_id: userId }), + PrintJob.find({ user_id: userId }).sort({ date: -1 }).limit(5).populate('printer_id', 'name').populate('spool_id', 'brand color_hex').lean(), + Printer.findOne({ user_id: userId }).lean(), + PrintJob.findOne({ user_id: userId, status: 'In Progress' }).populate('printer_id', 'name').populate('spool_id', 'brand color_hex material').lean(), + PrintJob.find({ user_id: userId, status: { $ne: 'In Progress' } }).select('calculated_cost_filament').lean() + ]); + + // Calculate totals + let totalWeightG = 0; + let totalValue = 0; + + spools.forEach(spool => { + totalWeightG += (spool.weight_remaining_g || 0); + + // Value = (Remaining / Initial) * Price + if (spool.weight_initial_g > 0 && spool.price > 0) { + const ratio = (spool.weight_remaining_g || 0) / spool.weight_initial_g; + totalValue += (ratio * spool.price); + } + }); + + // Calculate total spent on prints + let totalSpent = 0; + completedPrints.forEach(print => { + totalSpent += (print.calculated_cost_filament || 0); + }); + + // Keep grams for precision, format for display + const totalWeightKg = totalWeightG >= 1000 + ? (totalWeightG / 1000).toFixed(2) + : (totalWeightG / 1000).toFixed(3); + const estimatedValue = totalValue.toFixed(2); + + return { + stats: { + spoolCount, + totalWeightKg, + totalWeightG, + printerCount, + estimatedValue, + totalSpent: totalSpent.toFixed(2) + }, + recentPrints: JSON.parse(JSON.stringify(recentPrints)), + activePrinter: activePrinter ? JSON.parse(JSON.stringify(activePrinter)) : null, + activePrintJob: activePrintJob ? JSON.parse(JSON.stringify(activePrintJob)) : null + }; +}; diff --git a/src/routes/+page.svelte b/src/routes/+page.svelte new file mode 100644 index 0000000..8c1431b --- /dev/null +++ b/src/routes/+page.svelte @@ -0,0 +1,426 @@ + + +
+ +
+
+

Dashboard

+

Welcome back to your printing hub.

+
+
+ + + + +
+
+ + +
+ +
+
+

+ Active Spools +

+
+ {stats.spoolCount} +
+
+
+ + +
+
+

+ Filament On Hand +

+
+ {#if stats.totalWeightG >= 1000} + {stats.totalWeightKg}kg + {:else} + {stats.totalWeightG}g + {/if} +
+
+
+ + +
+
+

+ Printers +

+
+ {stats.printerCount} +
+
+
+ + +
+
+

+ Est. Value +

+
+ ${stats.estimatedValue} +
+
+
+ + +
+
+

+ Total Spent +

+
+ ${stats.totalSpent} +
+
+
+
+ + +
+ +
+
+

Recent Activity

+ View All +
+ + {#if recentPrints.length === 0} + +
+
+ +
+

No recent prints found

+ + + +
+
+ {:else} +
+ {#each recentPrints as print} + +
+
+ {#if print.status === "Success"} + + {:else if print.status === "Fail"} + + {:else if print.status === "In Progress"} + + {:else} + + {/if} +
+
+

{print.name}

+

+ {print.printer_id?.name || "Unknown Printer"} • {print.filament_used_g}g +

+
+
+
+ + {print.status} + +

+ {new Date(print.date).toLocaleDateString()} +

+
+
+ {/each} +
+ {/if} +
+ + +
+

Printer Status

+ + {#if activePrintJob} + +
+ {activePrintJob.printer_id?.name || "Unknown Printer"} + Printing +
+ +
+ +
+

Currently Printing

+

+ {activePrintJob.name} +

+
+ + +
+
+ Progress + {Math.round( + getProgress( + activePrintJob.started_at, + activePrintJob.duration_minutes + ) + )}% +
+
+
+
+
+ + +
+
+

Time Remaining

+

+ {getTimeRemaining( + activePrintJob.started_at, + activePrintJob.duration_minutes + )} +

+
+
+

Filament

+

+ {activePrintJob.filament_used_g}g +

+
+
+ + + {#if activePrintJob.spool_id} +
+
+ {activePrintJob.spool_id.brand} + {activePrintJob.spool_id.material || ""} +
+ {/if} +
+ + + {:else if activePrinter} + +
+ {activePrinter.name} + Idle +
+
+

No active print job

+ + + +
+ + {:else} +
+

No printers configured

+ + + +
+ {/if} +
+
+
+
+ + diff --git a/src/routes/admin/+layout.server.ts b/src/routes/admin/+layout.server.ts new file mode 100644 index 0000000..1bc6ec5 --- /dev/null +++ b/src/routes/admin/+layout.server.ts @@ -0,0 +1,9 @@ +import { redirect } from '@sveltejs/kit'; +import type { LayoutServerLoad } from './$types'; + +export const load: LayoutServerLoad = async ({ locals }) => { + if (!locals.user || locals.user.role !== 'Admin') { + throw redirect(303, '/'); + } + return {}; +}; diff --git a/src/routes/admin/users/+page.server.ts b/src/routes/admin/users/+page.server.ts new file mode 100644 index 0000000..25a9241 --- /dev/null +++ b/src/routes/admin/users/+page.server.ts @@ -0,0 +1,11 @@ +import { User } from '$lib/models/User'; +import { connectDB } from '$lib/server/db'; +import type { PageServerLoad } from './$types'; + +export const load: PageServerLoad = async () => { + await connectDB(); + const users = await User.find({}).sort({ createdAt: -1 }).lean(); + return { + users: JSON.parse(JSON.stringify(users)) + }; +}; diff --git a/src/routes/admin/users/+page.svelte b/src/routes/admin/users/+page.svelte new file mode 100644 index 0000000..344f9e9 --- /dev/null +++ b/src/routes/admin/users/+page.svelte @@ -0,0 +1,59 @@ + + +
+
+

User Management

+

Manage platform access

+
+ + + + + + + + + + + + + {#each users as user} + + + + + + + {/each} + +
UsernameRoleJoinedActions
{user.username} + + {user.role} + + {new Date(user.createdAt).toLocaleDateString()} + {#if user.role !== "Admin"} + + {/if} +
+
+
diff --git a/src/routes/analytics/+page.server.ts b/src/routes/analytics/+page.server.ts new file mode 100644 index 0000000..95228cd --- /dev/null +++ b/src/routes/analytics/+page.server.ts @@ -0,0 +1,65 @@ +import { PrintJob } from '$lib/models/PrintJob'; +import { connectDB } from '$lib/server/db'; +import type { PageServerLoad } from './$types'; +import { redirect } from '@sveltejs/kit'; + +export const load: PageServerLoad = async ({ locals }) => { + if (!locals.user) throw redirect(303, '/login'); + + await connectDB(); + + // Fetch all prints for aggregation - filtered by user + const prints = await PrintJob.find({ user_id: locals.user.id }) + .populate('spool_id', 'color_hex material') + .populate('printer_id', 'power_consumption_watts') + .sort({ date: 1 }) + .lean(); + + // 1. Success vs Fail + let successCount = 0; + let failCount = 0; + + // 2. Material Usage (Map: Material -> Weight) + const materialUsage: Record = {}; + + // 3. Usage Over Time (Last 30 days) + const usageByDate: Record = {}; + + // 4. Electricity Usage Over Time (Wh) + const electricityByDate: Record = {}; + let totalElectricity = 0; + + prints.forEach(print => { + // Status + if (print.status === 'Success') successCount++; + else if (print.status === 'Fail') failCount++; + + // Material + if (print.spool_id?.material) { + const mat = print.spool_id.material; + materialUsage[mat] = (materialUsage[mat] || 0) + print.filament_used_g; + } + + // Timeline - Filament + const dateKey = new Date(print.date).toISOString().split('T')[0]; + usageByDate[dateKey] = (usageByDate[dateKey] || 0) + print.filament_used_g; + + // Electricity: Power (W) × Duration (hours) = Wh + const powerWatts = print.printer_id?.power_consumption_watts || 0; + const durationHours = (print.duration_minutes || 0) / 60; + const wattHours = powerWatts * durationHours; + + electricityByDate[dateKey] = (electricityByDate[dateKey] || 0) + wattHours; + totalElectricity += wattHours; + }); + + return { + analytics: { + successRate: { success: successCount, fail: failCount }, + materialUsage, + usageByDate, + electricityByDate, + totalElectricity: (totalElectricity / 1000).toFixed(2) // Convert to kWh + } + }; +}; diff --git a/src/routes/analytics/+page.svelte b/src/routes/analytics/+page.svelte new file mode 100644 index 0000000..d82bb8e --- /dev/null +++ b/src/routes/analytics/+page.svelte @@ -0,0 +1,282 @@ + + +
+
+

Analytics

+

Insights into your printing habits

+
+ + +
+ +

+ Total Prints +

+

{totalPrints}

+
+ +

+ Success Rate +

+

+ {successRate}% +

+
+ +

+ Electricity Used +

+

+ {analytics.totalElectricity}kWh +

+
+ +

+ Materials +

+

+ {Object.keys(analytics.materialUsage).length} +

+
+
+ +
+ + +
+ +

+ Daily Filament Usage (g) +

+
+
+ +
+
+ + + +
+ +

+ Daily Electricity Usage (kWh) +

+
+
+ +
+
+ + + +
+ + + + +
+ {successRate}% + Success +
+
+

+ {analytics.successRate.success} Success / {analytics.successRate + .fail} Fail +

+
+ + + +
+ +

+ Material Distribution +

+
+
+ +
+
+
+
+ + diff --git a/src/routes/api/spools/+server.ts b/src/routes/api/spools/+server.ts new file mode 100644 index 0000000..945b127 --- /dev/null +++ b/src/routes/api/spools/+server.ts @@ -0,0 +1,26 @@ +import { json } from '@sveltejs/kit'; +import { Spool } from '$lib/models/Spool'; +import type { RequestHandler } from './$types'; + +export const GET: RequestHandler = async () => { + try { + const spools = await Spool.find({ is_active: true }).sort({ createdAt: -1 }); + return json(spools); + } catch (error) { + return json({ error: 'Failed to fetch spools' }, { status: 500 }); + } +}; + +export const POST: RequestHandler = async ({ request }) => { + try { + const data = await request.json(); + + // Basic validation could go here, but Mongoose validation handles most + const newSpool = await Spool.create(data); + + return json(newSpool, { status: 201 }); + } catch (error) { + console.error('Create Spool Error:', error); + return json({ error: 'Failed to create spool' }, { status: 400 }); + } +}; diff --git a/src/routes/layout.css b/src/routes/layout.css new file mode 100644 index 0000000..cd67023 --- /dev/null +++ b/src/routes/layout.css @@ -0,0 +1,3 @@ +@import 'tailwindcss'; +@plugin '@tailwindcss/forms'; +@plugin '@tailwindcss/typography'; diff --git a/src/routes/login/+page.server.ts b/src/routes/login/+page.server.ts new file mode 100644 index 0000000..7bc054e --- /dev/null +++ b/src/routes/login/+page.server.ts @@ -0,0 +1,49 @@ +import { fail, redirect } from '@sveltejs/kit'; +import type { Actions, PageServerLoad } from './$types'; +import { User } from '$lib/models/User'; +import { connectDB } from '$lib/server/db'; +import bcrypt from 'bcryptjs'; +import jwt from 'jsonwebtoken'; + +const JWT_SECRET = process.env.JWT_SECRET || 'fallback_secret_change_me'; + +export const load: PageServerLoad = async ({ locals }) => { + if (locals.user) throw redirect(303, '/'); + return {}; +}; + +export const actions: Actions = { + login: async ({ request, cookies }) => { + const data = await request.formData(); + const username = data.get('username') as string; + const password = data.get('password') as string; + + if (!username || !password) { + return fail(400, { missing: true }); + } + + await connectDB(); + + const user = await User.findOne({ username }); + if (!user) { + return fail(400, { invalid: true }); + } + + const validPass = await bcrypt.compare(password, user.password); + if (!validPass) { + return fail(400, { invalid: true }); + } + + const token = jwt.sign({ id: user._id, username: user.username, role: user.role }, JWT_SECRET, { expiresIn: '7d' }); + + cookies.set('session', token, { + path: '/', + httpOnly: true, + sameSite: 'strict', + secure: process.env.NODE_ENV === 'production', + maxAge: 60 * 60 * 24 * 7 // 1 week + }); + + throw redirect(303, '/'); + } +}; diff --git a/src/routes/login/+page.svelte b/src/routes/login/+page.svelte new file mode 100644 index 0000000..7142b02 --- /dev/null +++ b/src/routes/login/+page.svelte @@ -0,0 +1,70 @@ + + +
+ +
+

+ Filaprint +

+

Sign in to your dashboard

+
+ + {#if form?.invalid} +
+ Invalid credentials. Please try again. +
+ {/if} + +
{ + loading = true; + return async ({ update }) => { + await update(); + loading = false; + }; + }} + class="space-y-6" + > + + + + +
+ +
+ Don't have an account? + Create one +
+
+
diff --git a/src/routes/logout/+server.ts b/src/routes/logout/+server.ts new file mode 100644 index 0000000..4b34b75 --- /dev/null +++ b/src/routes/logout/+server.ts @@ -0,0 +1,7 @@ +import { redirect } from '@sveltejs/kit'; +import type { RequestHandler } from './$types'; + +export const POST: RequestHandler = async ({ cookies }) => { + cookies.delete('session', { path: '/' }); + throw redirect(303, '/login'); +}; diff --git a/src/routes/printers/+page.server.ts b/src/routes/printers/+page.server.ts new file mode 100644 index 0000000..b986d95 --- /dev/null +++ b/src/routes/printers/+page.server.ts @@ -0,0 +1,94 @@ +import { Printer } from '$lib/models/Printer'; +import { connectDB } from '$lib/server/db'; +import type { PageServerLoad, Actions } from './$types'; +import { fail, redirect } from '@sveltejs/kit'; + +export const load: PageServerLoad = async ({ locals }) => { + if (!locals.user) throw redirect(303, '/login'); + + await connectDB(); + const printers = await Printer.find({ user_id: locals.user.id }).sort({ name: 1 }).lean(); + return { + printers: JSON.parse(JSON.stringify(printers)) + }; +}; + +export const actions: Actions = { + create: async ({ request, locals }) => { + if (!locals.user) return fail(401, { unauthorized: true }); + + const formData = await request.formData(); + const name = formData.get('name'); + const model = formData.get('model'); + const watts = formData.get('power_consumption_watts'); + const nozzle = formData.get('nozzle_diameter_mm'); + + if (!name) return fail(400, { missing: true }); + + await connectDB(); + + try { + await Printer.create({ + user_id: locals.user.id, + name, + model, + power_consumption_watts: watts ? Number(watts) : 0, + nozzle_diameter_mm: nozzle ? Number(nozzle) : 0.4 + }); + return { success: true }; + } catch (error) { + console.error(error); + return fail(500, { dbError: true }); + } + }, + + edit: async ({ request, locals }) => { + if (!locals.user) return fail(401, { unauthorized: true }); + + const formData = await request.formData(); + const id = formData.get('id'); + const name = formData.get('name'); + const model = formData.get('model'); + const watts = formData.get('power_consumption_watts'); + const nozzle = formData.get('nozzle_diameter_mm'); + + if (!id || !name) return fail(400, { missing: true }); + + await connectDB(); + + try { + await Printer.findOneAndUpdate( + { _id: id, user_id: locals.user.id }, + { + name, + model, + power_consumption_watts: watts ? Number(watts) : 0, + nozzle_diameter_mm: nozzle ? Number(nozzle) : 0.4 + } + ); + return { success: true }; + } catch (error) { + console.error(error); + return fail(500, { dbError: true }); + } + }, + + delete: async ({ request, locals }) => { + if (!locals.user) return fail(401, { unauthorized: true }); + + const formData = await request.formData(); + const id = formData.get('id'); + + if (!id) return fail(400, { missing: true }); + + await connectDB(); + + try { + await Printer.findOneAndDelete({ _id: id, user_id: locals.user.id }); + return { success: true }; + } catch (error) { + console.error(error); + return fail(500, { dbError: true }); + } + } +}; diff --git a/src/routes/printers/+page.svelte b/src/routes/printers/+page.svelte new file mode 100644 index 0000000..a4003c8 --- /dev/null +++ b/src/routes/printers/+page.svelte @@ -0,0 +1,236 @@ + + +
+
+
+

Printers

+

Configure your machines

+
+ +
+ +
+ {#each printers as printer} + +
+

{printer.name}

+ + {printer.model || "Unknown Model"} + +
+ +
+
+ Power + {printer.power_consumption_watts}W +
+
+ Nozzle + {printer.nozzle_diameter_mm}mm +
+
+ +
+ +
+
+ {/each} + + {#if printers.length === 0} + +
+ +
+

No printers yet

+

+ Add your first 3D printer to start tracking prints. +

+ +
+ {/if} +
+
+ + + (showAddModal = false)} +> +
{ + isSubmitting = true; + return async ({ update }) => { + await update(); + isSubmitting = false; + showAddModal = false; + }; + }} + class="space-y-4" + > + + + +
+ + +
+ +
+ + +
+
+
+ + + + {#if editingPrinter} +
{ + isSubmitting = true; + return async ({ update }) => { + await update(); + isSubmitting = false; + closeEditModal(); + }; + }} + class="space-y-4" + > + + + + + +
+ + +
+ +
+ +
+ + +
+
+
+ {/if} +
+ + diff --git a/src/routes/prints/+page.server.ts b/src/routes/prints/+page.server.ts new file mode 100644 index 0000000..3924e21 --- /dev/null +++ b/src/routes/prints/+page.server.ts @@ -0,0 +1,203 @@ +import { PrintJob } from '$lib/models/PrintJob'; +import { Spool } from '$lib/models/Spool'; +import { Printer } from '$lib/models/Printer'; +import { connectDB } from '$lib/server/db'; +import type { PageServerLoad, Actions } from './$types'; +import { fail, redirect } from '@sveltejs/kit'; + +export const load: PageServerLoad = async ({ locals }) => { + if (!locals.user) throw redirect(303, '/login'); + + await connectDB(); + + // Fetch prints with populated fields (Spool and Printer) - filtered by user + const prints = await PrintJob.find({ user_id: locals.user.id }) + .populate('spool_id', 'brand material color_hex') + .populate('printer_id', 'name model') + .sort({ date: -1 }) + .lean(); + + // Fetch active spools and printers for the "Log Print" form - filtered by user + const [spools, printers] = await Promise.all([ + Spool.find({ user_id: locals.user.id, is_active: true }).lean(), + Printer.find({ user_id: locals.user.id }).lean() + ]); + + return { + prints: JSON.parse(JSON.stringify(prints)), + spools: JSON.parse(JSON.stringify(spools)), + printers: JSON.parse(JSON.stringify(printers)) + }; +}; + +export const actions: Actions = { + log: async ({ request, locals }) => { + if (!locals.user) return fail(401, { unauthorized: true }); + + const formData = await request.formData(); + const name = formData.get('name'); + const spool_id = formData.get('spool_id'); + const printer_id = formData.get('printer_id'); + const duration_minutes = formData.get('duration_minutes'); + const filament_used_g = formData.get('filament_used_g'); + const status = formData.get('status'); + const manual_cost = formData.get('manual_cost'); + const elapsed_minutes = formData.get('elapsed_minutes'); + + if (!spool_id || !printer_id || !filament_used_g) { + return fail(400, { missing: true }); + } + + await connectDB(); + + try { + // 1. Get the spool to calculate cost and deduct weight + const spool = await Spool.findOne({ _id: spool_id, user_id: locals.user.id }); + if (!spool) return fail(404, { spoolNotFound: true }); + + // Verify printer belongs to user + const printer = await Printer.findOne({ _id: printer_id, user_id: locals.user.id }); + if (!printer) return fail(404, { printerNotFound: true }); + + const weightUsed = Number(filament_used_g); + + // Calculate Cost: use manual if provided, otherwise calculate + let costFilament: number; + if (manual_cost && String(manual_cost).trim() !== '') { + costFilament = Number(manual_cost); + } else { + costFilament = (spool.price / spool.weight_initial_g) * weightUsed; + } + + // 2. Create Print Job + const isInProgress = status === 'In Progress'; + + // Calculate started_at based on elapsed time + let startedAt: Date | null = null; + if (isInProgress) { + const elapsedMs = Number(elapsed_minutes || 0) * 60 * 1000; + startedAt = new Date(Date.now() - elapsedMs); + } + + await PrintJob.create({ + user_id: locals.user.id, + name: name || 'Untitled Print', + spool_id, + printer_id, + duration_minutes: Number(duration_minutes), + filament_used_g: weightUsed, + calculated_cost_filament: Number(costFilament.toFixed(2)), + status, + started_at: startedAt, + date: new Date() + }); + + // 3. Deduct Filament from Spool (only if not In Progress) + if (!isInProgress) { + spool.weight_remaining_g = Math.max(0, spool.weight_remaining_g - weightUsed); + await spool.save(); + } + + return { success: true }; + } catch (error) { + console.error(error); + return fail(500, { dbError: true }); + } + }, + + edit: async ({ request, locals }) => { + if (!locals.user) return fail(401, { unauthorized: true }); + + const formData = await request.formData(); + const id = formData.get('id'); + const name = formData.get('name'); + const duration_minutes = formData.get('duration_minutes'); + const filament_used_g = formData.get('filament_used_g'); + const status = formData.get('status'); + const manual_cost = formData.get('manual_cost'); + const elapsed_minutes = formData.get('elapsed_minutes'); + const printer_id = formData.get('printer_id'); + const spool_id = formData.get('spool_id'); + + if (!id || !name) { + return fail(400, { missing: true }); + } + + await connectDB(); + + try { + const printJob = await PrintJob.findOne({ _id: id, user_id: locals.user.id }).populate('spool_id'); + if (!printJob) return fail(404, { notFound: true }); + + const weightUsed = Number(filament_used_g); + + // Calculate Cost: use manual if provided, otherwise calculate + let costFilament: number; + if (manual_cost && String(manual_cost).trim() !== '') { + costFilament = Number(manual_cost); + } else if (printJob.spool_id?.price && printJob.spool_id?.weight_initial_g) { + costFilament = (printJob.spool_id.price / printJob.spool_id.weight_initial_g) * weightUsed; + } else { + costFilament = printJob.calculated_cost_filament || 0; + } + + // Calculate started_at based on elapsed time for In Progress + const isInProgress = status === 'In Progress'; + let startedAt = printJob.started_at; + + if (isInProgress && elapsed_minutes) { + const elapsedMs = Number(elapsed_minutes) * 60 * 1000; + startedAt = new Date(Date.now() - elapsedMs); + } else if (!isInProgress) { + startedAt = null; + } + + // Build update object + const updateData: any = { + name, + duration_minutes: Number(duration_minutes), + filament_used_g: weightUsed, + calculated_cost_filament: Number(costFilament.toFixed(2)), + status, + started_at: startedAt + }; + + // Update printer/spool if provided (for In Progress) + if (printer_id) { + updateData.printer_id = printer_id; + } + if (spool_id) { + updateData.spool_id = spool_id; + } + + await PrintJob.findOneAndUpdate( + { _id: id, user_id: locals.user.id }, + updateData + ); + + return { success: true }; + } catch (error) { + console.error(error); + return fail(500, { dbError: true }); + } + }, + + delete: async ({ request, locals }) => { + if (!locals.user) return fail(401, { unauthorized: true }); + + const formData = await request.formData(); + const id = formData.get('id'); + + if (!id) return fail(400, { missing: true }); + + await connectDB(); + + try { + await PrintJob.findOneAndDelete({ _id: id, user_id: locals.user.id }); + return { success: true }; + } catch (error) { + console.error(error); + return fail(500, { dbError: true }); + } + } +}; diff --git a/src/routes/prints/+page.svelte b/src/routes/prints/+page.svelte new file mode 100644 index 0000000..4d7b6b9 --- /dev/null +++ b/src/routes/prints/+page.svelte @@ -0,0 +1,177 @@ + + +
+
+
+

Print History

+

+ Track your usage and successful prints +

+
+ +
+ +
+ {#each prints as print} + + +
openEditModal(print)} class="cursor-pointer"> + + +
+ {#if print.status === "Success"} + + {:else if print.status === "Fail"} + + {:else if print.status === "In Progress"} + + {:else} + + {/if} +
+ +
+

+ {print.name} +

+
+ + + {print.spool_id?.brand || "Unknown Spool"} + + + {print.printer_id?.name || + "Unknown Printer"} + + {new Date( + print.date, + ).toLocaleDateString()} +
+
+ +
+
+

+ Duration +

+

{print.duration_minutes}m

+
+
+

+ Weight +

+

{print.filament_used_g}g

+
+
+

+ Cost +

+

+ ${print.calculated_cost_filament?.toFixed(2) || + "0.00"} +

+
+
+
+
+ {/each} + + {#if prints.length === 0} +
+ No prints logged yet. Start printing! +
+ {/if} +
+
+ + + (showLogModal = false)} +/> + + + + diff --git a/src/routes/register/+page.server.ts b/src/routes/register/+page.server.ts new file mode 100644 index 0000000..5f73fba --- /dev/null +++ b/src/routes/register/+page.server.ts @@ -0,0 +1,71 @@ +import { fail, redirect } from '@sveltejs/kit'; +import type { Actions, PageServerLoad } from './$types'; +import { User } from '$lib/models/User'; +import { connectDB } from '$lib/server/db'; +import bcrypt from 'bcryptjs'; +import jwt from 'jsonwebtoken'; + +const JWT_SECRET = process.env.JWT_SECRET || 'fallback_secret_change_me'; + +export const load: PageServerLoad = async ({ locals }) => { + if (locals.user) throw redirect(303, '/'); + return {}; +}; + +export const actions: Actions = { + register: async ({ request, cookies }) => { + const data = await request.formData(); + const username = data.get('username') as string; + const password = data.get('password') as string; + const confirmPassword = data.get('confirmPassword') as string; + + if (!username || !password || !confirmPassword) { + return fail(400, { missing: true }); + } + + if (password !== confirmPassword) { + return fail(400, { mismatch: true }); + } + + if (password.length < 6) { + return fail(400, { weak: true }); + } + + await connectDB(); + + const existing = await User.findOne({ username }); + if (existing) { + return fail(400, { exists: true }); + } + + const hashedPassword = await bcrypt.hash(password, 10); + + try { + // First user becomes Admin + const userCount = await User.countDocuments({}); + const role = userCount === 0 ? 'Admin' : 'Maker'; + + const user = await User.create({ + username, + password: hashedPassword, + role + }); + + const token = jwt.sign({ id: user._id, username: user.username, role: user.role }, JWT_SECRET, { expiresIn: '7d' }); + + cookies.set('session', token, { + path: '/', + httpOnly: true, + sameSite: 'strict', + secure: process.env.NODE_ENV === 'production', + maxAge: 60 * 60 * 24 * 7 + }); + + } catch (error) { + console.error(error); + return fail(500, { error: true }); + } + + throw redirect(303, '/'); + } +}; diff --git a/src/routes/register/+page.svelte b/src/routes/register/+page.svelte new file mode 100644 index 0000000..a5b2428 --- /dev/null +++ b/src/routes/register/+page.svelte @@ -0,0 +1,88 @@ + + +
+ +
+

+ Filaprint +

+

Create your account

+
+ + {#if form?.mismatch} +
+ Passwords do not match. +
+ {:else if form?.exists} +
+ Username already taken. +
+ {:else if form?.weak} +
+ Password must be at least 6 characters. +
+ {/if} + +
{ + loading = true; + return async ({ update }) => { + await update(); + loading = false; + }; + }} + class="space-y-6" + > + + + + + +
+ +
+ Already have an account? + Log In +
+
+
diff --git a/src/routes/settings/+page.server.ts b/src/routes/settings/+page.server.ts new file mode 100644 index 0000000..5d23925 --- /dev/null +++ b/src/routes/settings/+page.server.ts @@ -0,0 +1,99 @@ +import { connectDB } from '$lib/server/db'; +import { User } from '$lib/models/User'; +import type { PageServerLoad, Actions } from './$types'; +import { fail, redirect } from '@sveltejs/kit'; +import bcrypt from 'bcryptjs'; + +export const load: PageServerLoad = async ({ locals }) => { + if (!locals.user) throw redirect(303, '/login'); + + await connectDB(); + + const user = await User.findById(locals.user.id).select('-password').lean(); + + return { + userProfile: JSON.parse(JSON.stringify(user)) + }; +}; + +export const actions: Actions = { + updateProfile: async ({ request, locals }) => { + if (!locals.user) return fail(401, { unauthorized: true }); + + const formData = await request.formData(); + const username = formData.get('username'); + const location = formData.get('location'); + const electricity_rate = formData.get('electricity_rate'); + + if (!username) { + return fail(400, { missing: true, message: 'Username is required' }); + } + + await connectDB(); + + try { + // Check if username is taken by another user + const existingUser = await User.findOne({ + username, + _id: { $ne: locals.user.id } + }); + + if (existingUser) { + return fail(400, { taken: true, message: 'Username already taken' }); + } + + await User.findByIdAndUpdate(locals.user.id, { + username, + location: location || '', + electricity_rate: electricity_rate ? Number(electricity_rate) : 0.12 + }); + + return { success: true, message: 'Profile updated successfully' }; + } catch (error) { + console.error(error); + return fail(500, { dbError: true, message: 'Failed to update profile' }); + } + }, + + changePassword: async ({ request, locals }) => { + if (!locals.user) return fail(401, { unauthorized: true }); + + const formData = await request.formData(); + const currentPassword = formData.get('currentPassword') as string; + const newPassword = formData.get('newPassword') as string; + const confirmPassword = formData.get('confirmPassword') as string; + + if (!currentPassword || !newPassword || !confirmPassword) { + return fail(400, { missing: true, message: 'All password fields are required' }); + } + + if (newPassword !== confirmPassword) { + return fail(400, { mismatch: true, message: 'New passwords do not match' }); + } + + if (newPassword.length < 6) { + return fail(400, { weak: true, message: 'Password must be at least 6 characters' }); + } + + await connectDB(); + + try { + const user = await User.findById(locals.user.id); + if (!user) return fail(404, { notFound: true }); + + const validPassword = await bcrypt.compare(currentPassword, user.password); + if (!validPassword) { + return fail(400, { invalid: true, message: 'Current password is incorrect' }); + } + + const hashedPassword = await bcrypt.hash(newPassword, 10); + user.password = hashedPassword; + await user.save(); + + return { passwordChanged: true, message: 'Password changed successfully' }; + } catch (error) { + console.error(error); + return fail(500, { dbError: true, message: 'Failed to change password' }); + } + } +}; diff --git a/src/routes/settings/+page.svelte b/src/routes/settings/+page.svelte new file mode 100644 index 0000000..df43e2b --- /dev/null +++ b/src/routes/settings/+page.svelte @@ -0,0 +1,213 @@ + + +
+
+

Settings

+

Manage your account preferences

+
+ + {#if showSuccess} +
+ + {successMessage} +
+ {/if} + + {#if form?.message && !form?.success && !form?.passwordChanged} +
+ + {form.message} +
+ {/if} + + + +
+ +

Profile

+
+ +
{ + isSubmitting = true; + return async ({ update }) => { + await update(); + isSubmitting = false; + }; + }} + class="space-y-4" + > +
+
+ {userProfile?.username?.charAt(0).toUpperCase() || "U"} +
+
+

+ {userProfile?.username} +

+

{userProfile?.role}

+
+
+ + + + + +
+ +
+

+ Rate Info +

+

+ Used to calculate electricity costs for your prints. + Check your utility bill for your rate. +

+
+
+ +
+ +
+
+
+ + + +
+ +

Change Password

+
+ +
{ + isSubmitting = true; + return async ({ update }) => { + await update(); + isSubmitting = false; + }; + }} + class="space-y-4" + > + +
+ + +
+ +
+ +
+
+
+ + + +
+ +

Danger Zone

+
+ +

+ Once you delete your account, there is no going back. Please be + certain. +

+ + +
+
+ + diff --git a/src/routes/spools/+page.server.ts b/src/routes/spools/+page.server.ts new file mode 100644 index 0000000..e3747b0 --- /dev/null +++ b/src/routes/spools/+page.server.ts @@ -0,0 +1,114 @@ +import { Spool } from '$lib/models/Spool'; +import { connectDB } from '$lib/server/db'; +import type { PageServerLoad, Actions } from './$types'; +import { fail, redirect } from '@sveltejs/kit'; + +export const load: PageServerLoad = async ({ locals }) => { + if (!locals.user) throw redirect(303, '/login'); + + await connectDB(); + // Filter by user_id + const spools = await Spool.find({ user_id: locals.user.id, is_active: true }).sort({ createdAt: -1 }).lean(); + + return { + spools: JSON.parse(JSON.stringify(spools)) + }; +}; + +export const actions: Actions = { + create: async ({ request, locals }) => { + if (!locals.user) return fail(401, { unauthorized: true }); + + const formData = await request.formData(); + const brand = formData.get('brand'); + const material = formData.get('material'); + const color_hex = formData.get('color_hex'); + const weight_initial_g = formData.get('weight_initial_g'); + const weight_remaining_g = formData.get('weight_remaining_g'); + const price = formData.get('price'); + + if (!brand || !material || !weight_initial_g) { + return fail(400, { missing: true }); + } + + await connectDB(); + + try { + await Spool.create({ + user_id: locals.user.id, + brand, + material, + color_hex, + weight_initial_g: Number(weight_initial_g), + weight_remaining_g: Number(weight_remaining_g || weight_initial_g), + price: price ? Number(price) : 0 + }); + return { success: true }; + } catch (error) { + console.error(error); + return fail(500, { dbError: true }); + } + }, + + edit: async ({ request, locals }) => { + if (!locals.user) return fail(401, { unauthorized: true }); + + const formData = await request.formData(); + const id = formData.get('id'); + const brand = formData.get('brand'); + const material = formData.get('material'); + const color_hex = formData.get('color_hex'); + const weight_initial_g = formData.get('weight_initial_g'); + const weight_remaining_g = formData.get('weight_remaining_g'); + const price = formData.get('price'); + + if (!id || !brand || !material || !weight_initial_g) { + return fail(400, { missing: true }); + } + + await connectDB(); + + try { + const result = await Spool.findOneAndUpdate( + { _id: id, user_id: locals.user.id }, + { + brand, + material, + color_hex, + weight_initial_g: Number(weight_initial_g), + weight_remaining_g: Number(weight_remaining_g), + price: price ? Number(price) : 0 + } + ); + if (!result) return fail(404, { notFound: true }); + return { success: true }; + } catch (error) { + console.error(error); + return fail(500, { dbError: true }); + } + }, + + delete: async ({ request, locals }) => { + if (!locals.user) return fail(401, { unauthorized: true }); + + const formData = await request.formData(); + const id = formData.get('id'); + + if (!id) return fail(400, { missing: true }); + + await connectDB(); + + try { + // Soft delete by setting is_active to false + const result = await Spool.findOneAndUpdate( + { _id: id, user_id: locals.user.id }, + { is_active: false } + ); + if (!result) return fail(404, { notFound: true }); + return { success: true }; + } catch (error) { + console.error(error); + return fail(500, { dbError: true }); + } + } +}; diff --git a/src/routes/spools/+page.svelte b/src/routes/spools/+page.svelte new file mode 100644 index 0000000..47d807a --- /dev/null +++ b/src/routes/spools/+page.svelte @@ -0,0 +1,362 @@ + + +
+
+
+

My Spools

+

Manage your filament inventory

+
+ +
+ + {#if spools.length === 0} + +
+ +
+

No spools yet

+

+ Add your first filament spool to start tracking usage and costs. +

+ +
+ {:else} +
+ {#each spools as spool} + + +
openEditModal(spool)} class="cursor-pointer"> + + +
+ +
+
+
+ {spool.material} +

+ {spool.brand} +

+
+
+
+ +
+
+ Remaining + {spool.weight_remaining_g}g +
+ +
+
+
+
+ {spool.weight_initial_g}g Initial + ${spool.price} +
+
+
+
+
+ {/each} +
+ {/if} +
+ + + (showAddModal = false)} +> +
{ + isSubmitting = true; + return async ({ update }) => { + await update(); + isSubmitting = false; + showAddModal = false; + }; + }} + class="space-y-4" + > +
+ + + +
+ +
+
+ + +
+ +
+
+
+ +
+ +
+ + +
+ +
+ + +
+
+
+ + + + {#if editingSpool} +
{ + isSubmitting = true; + return async ({ update }) => { + await update(); + isSubmitting = false; + closeEditModal(); + }; + }} + class="space-y-4" + > + + +
+ + + +
+ +
+
+ + +
+ +
+
+
+ +
+ +
+ + +
+ +
+ + +
+ + +
+
+
+ {/if} +
+ + diff --git a/static/font/JetBrainsMono-Regular.ttf b/static/font/JetBrainsMono-Regular.ttf new file mode 100644 index 0000000..dff66cc Binary files /dev/null and b/static/font/JetBrainsMono-Regular.ttf differ diff --git a/static/robots.txt b/static/robots.txt new file mode 100644 index 0000000..b6dd667 --- /dev/null +++ b/static/robots.txt @@ -0,0 +1,3 @@ +# allow crawling everything by default +User-agent: * +Disallow: diff --git a/svelte.config.js b/svelte.config.js new file mode 100644 index 0000000..fb67358 --- /dev/null +++ b/svelte.config.js @@ -0,0 +1,15 @@ +import adapter from '@sveltejs/adapter-node'; +import { vitePreprocess } from '@sveltejs/vite-plugin-svelte'; + +/** @type {import('@sveltejs/kit').Config} */ +const config = { + preprocess: vitePreprocess(), + kit: { + adapter: adapter(), + csrf: { + checkOrigin: true + } + } +}; + +export default config; diff --git a/tsconfig.json b/tsconfig.json new file mode 100644 index 0000000..feea18b --- /dev/null +++ b/tsconfig.json @@ -0,0 +1,15 @@ +{ + "extends": "./.svelte-kit/tsconfig.json", + "compilerOptions": { + "rewriteRelativeImportExtensions": true, + "allowJs": true, + "checkJs": true, + "esModuleInterop": true, + "forceConsistentCasingInFileNames": true, + "resolveJsonModule": true, + "skipLibCheck": true, + "sourceMap": true, + "strict": true, + "moduleResolution": "bundler" + } +} diff --git a/vite.config.ts b/vite.config.ts new file mode 100644 index 0000000..56f40c7 --- /dev/null +++ b/vite.config.ts @@ -0,0 +1,5 @@ +import tailwindcss from '@tailwindcss/vite'; +import { sveltekit } from '@sveltejs/kit/vite'; +import { defineConfig } from 'vite'; + +export default defineConfig({ plugins: [tailwindcss(), sveltekit()] });