Compare commits
30 Commits
feature/is
...
feature/is
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
762ec56a3c | ||
| 91a21e274f | |||
|
|
14e5cab7bb | ||
| 229cb2cc90 | |||
|
|
d4d3d9390c | ||
| 12c5304638 | |||
|
|
080d2424c8 | ||
|
|
6b1c34ceff | ||
| 231f594004 | |||
|
|
7d35a3e7b3 | ||
| 670b2f9200 | |||
|
|
521e3f552f | ||
| 627e970986 | |||
|
|
50a0bd9417 | ||
| 097f0f9cee | |||
|
|
b1ef7e43be | ||
|
|
12bda4c08f | ||
|
|
5eb0d04377 | ||
| 5805be698b | |||
|
|
1f21032194 | ||
| f4b870f59c | |||
|
|
4834286005 | ||
| be2af1675a | |||
|
|
b93f4677fc | ||
| 4eec4923af | |||
|
|
f70b90748a | ||
| 1c54415a29 | |||
|
|
01c5880e37 | ||
|
|
223f4b6ea1 | ||
|
|
436f92cafc |
64
.env.example
@@ -1,54 +1,20 @@
|
|||||||
# Pantry - Environment Variables Template
|
# Supabase Local Development Environment
|
||||||
# Copy to .env.development for local development
|
# Copy this file to .env and adjust as needed
|
||||||
# Copy to .env.production for production deployment
|
|
||||||
|
|
||||||
# ==============================================
|
# PostgreSQL
|
||||||
# Supabase Configuration
|
POSTGRES_PASSWORD=postgres
|
||||||
# ==============================================
|
|
||||||
|
|
||||||
# Supabase API URL (from Coolify service)
|
# JWT Secret (generate with: openssl rand -base64 32)
|
||||||
SUPABASE_URL=https://your-supabase-instance.example.com
|
# Default is fine for local dev, change for production
|
||||||
|
JWT_SECRET=super-secret-jwt-token-with-at-least-32-characters-long
|
||||||
|
|
||||||
# Supabase Anon Key (public, safe to expose to frontend)
|
# API Keys
|
||||||
SUPABASE_ANON_KEY=your-anon-key-here
|
# These are Supabase's default demo keys - OK for local development
|
||||||
|
# For production, generate new keys: https://supabase.com/docs/guides/self-hosting#api-keys
|
||||||
|
ANON_KEY=eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJpc3MiOiJzdXBhYmFzZS1kZW1vIiwicm9sZSI6ImFub24iLCJleHAiOjE5ODM4MTI5OTZ9.CRXP1A7WOeoJeXxjNni43kdQwgnWNReilDMblYTn_I0
|
||||||
|
|
||||||
# Supabase Service Role Key (SECRET - server-side only, never expose to frontend)
|
SERVICE_ROLE_KEY=eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJpc3MiOiJzdXBhYmFzZS1kZW1vIiwicm9sZSI6InNlcnZpY2Vfcm9sZSIsImV4cCI6MTk4MzgxMjk5Nn0.EGIM96RAZx35lJzdJsyH-qQwv8Hdp7fsn3W0YpN81IU
|
||||||
SUPABASE_SERVICE_ROLE_KEY=your-service-role-key-here
|
|
||||||
|
|
||||||
# JWT Secret (for Supabase Auth)
|
# Nuxt App Configuration (also copy to app/.env)
|
||||||
SUPABASE_JWT_SECRET=your-jwt-secret-here
|
NUXT_PUBLIC_SUPABASE_URL=http://localhost:54321
|
||||||
|
NUXT_PUBLIC_SUPABASE_ANON_KEY=eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJpc3MiOiJzdXBhYmFzZS1kZW1vIiwicm9sZSI6ImFub24iLCJleHAiOjE5ODM4MTI5OTZ9.CRXP1A7WOeoJeXxjNni43kdQwgnWNReilDMblYTn_I0
|
||||||
# Database Password (for direct PostgreSQL access if needed)
|
|
||||||
POSTGRES_PASSWORD=your-postgres-password-here
|
|
||||||
|
|
||||||
# ==============================================
|
|
||||||
# Application Configuration
|
|
||||||
# ==============================================
|
|
||||||
|
|
||||||
# Public app URL (where the Nuxt app is hosted)
|
|
||||||
PUBLIC_APP_URL=http://localhost:3000
|
|
||||||
|
|
||||||
# Node environment
|
|
||||||
NODE_ENV=development
|
|
||||||
|
|
||||||
# ==============================================
|
|
||||||
# External APIs
|
|
||||||
# ==============================================
|
|
||||||
|
|
||||||
# Open Food Facts API (no auth required)
|
|
||||||
OPENFOODFACTS_API_URL=https://world.openfoodfacts.org
|
|
||||||
|
|
||||||
# ==============================================
|
|
||||||
# Optional: Auth Providers (OIDC)
|
|
||||||
# ==============================================
|
|
||||||
|
|
||||||
# Google OAuth (optional - configure in Supabase if needed)
|
|
||||||
AUTH_GOOGLE_ENABLED=false
|
|
||||||
AUTH_GOOGLE_CLIENT_ID=
|
|
||||||
AUTH_GOOGLE_SECRET=
|
|
||||||
|
|
||||||
# Authentik (optional)
|
|
||||||
AUTH_AUTHENTIK_ENABLED=false
|
|
||||||
AUTH_AUTHENTIK_URL=
|
|
||||||
AUTH_AUTHENTIK_CLIENT_ID=
|
|
||||||
AUTH_AUTHENTIK_SECRET=
|
|
||||||
|
|||||||
54
.env.example.bak
Normal file
@@ -0,0 +1,54 @@
|
|||||||
|
# Pantry - Environment Variables Template
|
||||||
|
# Copy to .env.development for local development
|
||||||
|
# Copy to .env.production for production deployment
|
||||||
|
|
||||||
|
# ==============================================
|
||||||
|
# Supabase Configuration
|
||||||
|
# ==============================================
|
||||||
|
|
||||||
|
# Supabase API URL (from Coolify service)
|
||||||
|
SUPABASE_URL=https://your-supabase-instance.example.com
|
||||||
|
|
||||||
|
# Supabase Anon Key (public, safe to expose to frontend)
|
||||||
|
SUPABASE_ANON_KEY=your-anon-key-here
|
||||||
|
|
||||||
|
# Supabase Service Role Key (SECRET - server-side only, never expose to frontend)
|
||||||
|
SUPABASE_SERVICE_ROLE_KEY=your-service-role-key-here
|
||||||
|
|
||||||
|
# JWT Secret (for Supabase Auth)
|
||||||
|
SUPABASE_JWT_SECRET=your-jwt-secret-here
|
||||||
|
|
||||||
|
# Database Password (for direct PostgreSQL access if needed)
|
||||||
|
POSTGRES_PASSWORD=your-postgres-password-here
|
||||||
|
|
||||||
|
# ==============================================
|
||||||
|
# Application Configuration
|
||||||
|
# ==============================================
|
||||||
|
|
||||||
|
# Public app URL (where the Nuxt app is hosted)
|
||||||
|
PUBLIC_APP_URL=http://localhost:3000
|
||||||
|
|
||||||
|
# Node environment
|
||||||
|
NODE_ENV=development
|
||||||
|
|
||||||
|
# ==============================================
|
||||||
|
# External APIs
|
||||||
|
# ==============================================
|
||||||
|
|
||||||
|
# Open Food Facts API (no auth required)
|
||||||
|
OPENFOODFACTS_API_URL=https://world.openfoodfacts.org
|
||||||
|
|
||||||
|
# ==============================================
|
||||||
|
# Optional: Auth Providers (OIDC)
|
||||||
|
# ==============================================
|
||||||
|
|
||||||
|
# Google OAuth (optional - configure in Supabase if needed)
|
||||||
|
AUTH_GOOGLE_ENABLED=false
|
||||||
|
AUTH_GOOGLE_CLIENT_ID=
|
||||||
|
AUTH_GOOGLE_SECRET=
|
||||||
|
|
||||||
|
# Authentik (optional)
|
||||||
|
AUTH_AUTHENTIK_ENABLED=false
|
||||||
|
AUTH_AUTHENTIK_URL=
|
||||||
|
AUTH_AUTHENTIK_CLIENT_ID=
|
||||||
|
AUTH_AUTHENTIK_SECRET=
|
||||||
65
README.md
@@ -22,26 +22,37 @@ A simple, modern kitchen inventory app that the whole family can actually use. B
|
|||||||
- 🔒 **Self-hosted** — Your data stays yours
|
- 🔒 **Self-hosted** — Your data stays yours
|
||||||
- 🌐 **Open Food Facts** — Auto-fill product data from barcodes
|
- 🌐 **Open Food Facts** — Auto-fill product data from barcodes
|
||||||
|
|
||||||
## 🚀 Quick Start
|
## 🚀 Quick Start (Local Development)
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
# Clone
|
# Clone repository
|
||||||
git clone https://gitea.jeanlucmakiola.de/pantry-app/pantry.git
|
git clone https://gitea.jeanlucmakiola.de/pantry-app/pantry.git
|
||||||
|
cd pantry
|
||||||
|
|
||||||
# Start services (Docker Compose)
|
# One-command startup
|
||||||
docker-compose up -d
|
./dev.sh
|
||||||
|
|
||||||
# Access at http://localhost:3000
|
|
||||||
```
|
```
|
||||||
|
|
||||||
|
**What this does:**
|
||||||
|
1. Starts Supabase (PostgreSQL + API + Auth + Studio)
|
||||||
|
2. Installs frontend dependencies
|
||||||
|
3. Launches Nuxt dev server
|
||||||
|
|
||||||
|
**Access:**
|
||||||
|
- App: `http://localhost:3000`
|
||||||
|
- Supabase Studio: `http://localhost:54323`
|
||||||
|
|
||||||
|
**See [DEV_SETUP.md](DEV_SETUP.md) for detailed setup guide.**
|
||||||
|
|
||||||
## 📚 Documentation
|
## 📚 Documentation
|
||||||
|
|
||||||
- [**Project Plan**](docs/PROJECT_PLAN.md) — Vision, roadmap, phases
|
- **[Getting Started](docs/development/getting-started.md)** — First-time setup (5 minutes)
|
||||||
- [**Architecture**](docs/ARCHITECTURE.md) — Tech stack, data model, design decisions
|
- **[Local Setup Guide](docs/development/local-setup.md)** — Detailed Docker Compose setup
|
||||||
- [**Database Schema**](docs/DATABASE.md) — Tables, relationships, RLS policies
|
- **[Project Plan](docs/PROJECT_PLAN.md)** — Vision, roadmap, MVP phases
|
||||||
- [**API Reference**](docs/API.md) — Endpoints, Supabase functions
|
- **[Architecture](docs/architecture/overview.md)** — Tech stack, design decisions
|
||||||
- [**Development Guide**](docs/DEVELOPMENT.md) — Setup, workflow, conventions
|
- **[Database Schema](docs/architecture/database.md)** — Tables, RLS policies, migrations
|
||||||
- [**Deployment**](docs/DEPLOYMENT.md) — Docker, Coolify, production setup
|
- **[Development Workflow](docs/development/workflow.md)** — Git flow, conventions
|
||||||
|
- **[Full Documentation Index](docs/README.md)** — Complete docs navigation
|
||||||
|
|
||||||
## 🛠️ Tech Stack
|
## 🛠️ Tech Stack
|
||||||
|
|
||||||
@@ -73,18 +84,32 @@ pantry/
|
|||||||
3. **Extendable** — Clean architecture for future features
|
3. **Extendable** — Clean architecture for future features
|
||||||
4. **Self-hosted first** — No SaaS plans, no lock-in
|
4. **Self-hosted first** — No SaaS plans, no lock-in
|
||||||
|
|
||||||
## 📋 MVP Status
|
## 📋 MVP Status (14/34 Complete - 41.2%)
|
||||||
|
|
||||||
**Target:** v0.1 (6-week sprint)
|
**Current Phase:** Week 2 ✅ Complete, Week 3 🔄 In Progress
|
||||||
|
|
||||||
|
✅ **Week 1 - Foundation (6/6)**
|
||||||
|
- Database schema + RLS policies
|
||||||
|
- Nuxt 4 app scaffold
|
||||||
|
- Supabase integration
|
||||||
|
- App layout
|
||||||
|
|
||||||
|
✅ **Week 2 - Core Inventory (8/8)**
|
||||||
|
- SQL helper functions
|
||||||
|
- Seed data (units + tags)
|
||||||
|
- Inventory CRUD UI
|
||||||
|
- Add/Edit/Delete components
|
||||||
|
|
||||||
|
🔄 **Week 3 - Barcode Scanning (1/5)**
|
||||||
|
- BarcodeScanner component
|
||||||
|
- html5-qrcode integration
|
||||||
|
- Product lookup (pending)
|
||||||
|
- Scan-to-add flow (pending)
|
||||||
|
|
||||||
|
⏸️ **Week 4-6** - Tag UI, PWA, Deployment (20 issues)
|
||||||
|
|
||||||
See [PROJECT_PLAN.md](docs/PROJECT_PLAN.md) for detailed roadmap.
|
See [PROJECT_PLAN.md](docs/PROJECT_PLAN.md) for detailed roadmap.
|
||||||
|
|
||||||
- [ ] Foundation (Nuxt + Supabase + Auth)
|
|
||||||
- [ ] Core inventory (CRUD, tags, units)
|
|
||||||
- [ ] Barcode scanning (PWA camera + Open Food Facts)
|
|
||||||
- [ ] Mobile polish (PWA, offline)
|
|
||||||
- [ ] Docker deployment
|
|
||||||
|
|
||||||
## 🤝 Contributing
|
## 🤝 Contributing
|
||||||
|
|
||||||
This is an early-stage project. Contributions welcome once v0.1 ships.
|
This is an early-stage project. Contributions welcome once v0.1 ships.
|
||||||
|
|||||||
214
SETUP.md
@@ -1,214 +0,0 @@
|
|||||||
# Pantry - Setup Guide
|
|
||||||
|
|
||||||
## ✅ Current Status
|
|
||||||
|
|
||||||
### Infrastructure (Complete)
|
|
||||||
|
|
||||||
- **Supabase Dev Instance**: Running on Coolify
|
|
||||||
- URL: `https://supabasekong-ewo8wssk4gs8cgg0c8kosk40.jeanlucmakiola.de`
|
|
||||||
- Status: ✅ Healthy
|
|
||||||
- Services: PostgreSQL, Auth (GoTrue), Realtime, Storage, PostgREST
|
|
||||||
|
|
||||||
### Environment Configuration (Complete)
|
|
||||||
|
|
||||||
- `.env.example` - Template for all environments
|
|
||||||
- `.env.development` - Dev credentials (Coolify Supabase)
|
|
||||||
- `.gitignore` - Protects secrets
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## 🚀 Quick Start (Development)
|
|
||||||
|
|
||||||
### Prerequisites
|
|
||||||
|
|
||||||
- **Bun** 1.0+ (or Node.js 20+)
|
|
||||||
- **Git** access to repository
|
|
||||||
- **Access to Coolify** Supabase instance (credentials in `.env.development`)
|
|
||||||
|
|
||||||
### Setup Steps
|
|
||||||
|
|
||||||
1. **Clone Repository**
|
|
||||||
```bash
|
|
||||||
git clone https://gitea.jeanlucmakiola.de/pantry-app/pantry.git
|
|
||||||
cd pantry
|
|
||||||
```
|
|
||||||
|
|
||||||
2. **Copy Development Environment**
|
|
||||||
```bash
|
|
||||||
cp .env.development .env
|
|
||||||
```
|
|
||||||
|
|
||||||
3. **Install Dependencies** (once `app/` exists)
|
|
||||||
```bash
|
|
||||||
cd app
|
|
||||||
bun install
|
|
||||||
```
|
|
||||||
|
|
||||||
4. **Apply Database Migrations** (once created)
|
|
||||||
```bash
|
|
||||||
cd supabase
|
|
||||||
# TBD: Migration command
|
|
||||||
```
|
|
||||||
|
|
||||||
5. **Start Development Server**
|
|
||||||
```bash
|
|
||||||
cd app
|
|
||||||
bun run dev
|
|
||||||
```
|
|
||||||
|
|
||||||
6. **Access App**
|
|
||||||
- App: `http://localhost:3000`
|
|
||||||
- Supabase API: `https://supabasekong-ewo8wssk4gs8cgg0c8kosk40.jeanlucmakiola.de`
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## 🗄️ Supabase Instance Details
|
|
||||||
|
|
||||||
### Endpoints
|
|
||||||
|
|
||||||
| Service | URL |
|
|
||||||
|---------|-----|
|
|
||||||
| API (PostgREST) | `https://supabasekong-ewo8wssk4gs8cgg0c8kosk40.jeanlucmakiola.de/rest/v1/` |
|
|
||||||
| Auth | `https://supabasekong-ewo8wssk4gs8cgg0c8kosk40.jeanlucmakiola.de/auth/v1/` |
|
|
||||||
| Realtime | `wss://supabasekong-ewo8wssk4gs8cgg0c8kosk40.jeanlucmakiola.de/realtime/v1/` |
|
|
||||||
| Storage | `https://supabasekong-ewo8wssk4gs8cgg0c8kosk40.jeanlucmakiola.de/storage/v1/` |
|
|
||||||
|
|
||||||
### Credentials
|
|
||||||
|
|
||||||
**Public (safe to use in frontend):**
|
|
||||||
- Anon Key: In `.env.development`
|
|
||||||
|
|
||||||
**Secret (server-side only):**
|
|
||||||
- Service Role Key: In `.env.development`
|
|
||||||
- JWT Secret: In `.env.development`
|
|
||||||
- Postgres Password: In `.env.development`
|
|
||||||
|
|
||||||
### Dashboard Access
|
|
||||||
|
|
||||||
**Supabase Studio:**
|
|
||||||
- URL: `https://supabasekong-ewo8wssk4gs8cgg0c8kosk40.jeanlucmakiola.de` (check Coolify for Studio port/subdomain)
|
|
||||||
- Username: `wJZbjs3Yd5P63cs9`
|
|
||||||
- Password: `Qv3byDujNzYe8r7YRxhNwh3DPTvZBWtN`
|
|
||||||
|
|
||||||
**Direct PostgreSQL Access** (for migrations/debugging):
|
|
||||||
- Host: `supabase-db` (or Coolify service FQDN)
|
|
||||||
- Database: `postgres`
|
|
||||||
- User: `postgres`
|
|
||||||
- Password: `55P0NVRUltRqzZYksuXTFli5iXwbQvgu`
|
|
||||||
- Port: `5432`
|
|
||||||
|
|
||||||
**MinIO (Storage):**
|
|
||||||
- Admin User: `EaTXrXvjo1R4hsaI`
|
|
||||||
- Admin Password: `gCZOphxAExNC17GYFwtw60WzTU0P8HW8`
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## 📋 Next Steps
|
|
||||||
|
|
||||||
### Week 1: Foundation (In Progress)
|
|
||||||
|
|
||||||
- [x] ~~Supabase dev environment setup~~ (Complete)
|
|
||||||
- [x] ~~Environment configuration~~ (Complete)
|
|
||||||
- [ ] Create database schema (`supabase/migrations/`)
|
|
||||||
- [ ] Scaffold Nuxt 4 app (`app/`)
|
|
||||||
- [ ] Implement email/password auth
|
|
||||||
- [ ] Deploy first version to Coolify
|
|
||||||
|
|
||||||
### Immediate Tasks
|
|
||||||
|
|
||||||
1. **Database Schema** (#10)
|
|
||||||
- Create migration files in `supabase/migrations/`
|
|
||||||
- Tables: `inventory_items`, `products`, `tags`, `item_tags`, `units`
|
|
||||||
- See: `docs/DATABASE.md`
|
|
||||||
|
|
||||||
2. **Nuxt Scaffold** (#8)
|
|
||||||
- Initialize Nuxt 4 project in `app/`
|
|
||||||
- Install dependencies: `@nuxtjs/supabase`, `@nuxt/ui`, Tailwind
|
|
||||||
- Configure `nuxt.config.ts`
|
|
||||||
|
|
||||||
3. **Auth Implementation** (#11)
|
|
||||||
- Supabase Auth integration
|
|
||||||
- Login/signup pages
|
|
||||||
- Protected routes
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## 🔧 Development Workflow
|
|
||||||
|
|
||||||
### Making Changes
|
|
||||||
|
|
||||||
```bash
|
|
||||||
# 1. Create feature branch
|
|
||||||
git checkout -b feature/your-feature
|
|
||||||
|
|
||||||
# 2. Make changes
|
|
||||||
# Edit files...
|
|
||||||
|
|
||||||
# 3. Test locally
|
|
||||||
bun run dev
|
|
||||||
|
|
||||||
# 4. Commit and push
|
|
||||||
git add .
|
|
||||||
git commit -m "feat: Your feature description"
|
|
||||||
git push origin feature/your-feature
|
|
||||||
|
|
||||||
# 5. Create PR on Gitea
|
|
||||||
```
|
|
||||||
|
|
||||||
### Database Migrations
|
|
||||||
|
|
||||||
```bash
|
|
||||||
# Create new migration
|
|
||||||
cd supabase/migrations
|
|
||||||
touch 001_initial_schema.sql
|
|
||||||
|
|
||||||
# Edit migration file (SQL)
|
|
||||||
# Test locally against Coolify Supabase instance
|
|
||||||
|
|
||||||
# Apply migration (TBD - once we set up migration tooling)
|
|
||||||
```
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## 🚨 Troubleshooting
|
|
||||||
|
|
||||||
### Can't connect to Supabase
|
|
||||||
|
|
||||||
**Test connection:**
|
|
||||||
```bash
|
|
||||||
curl -s "https://supabasekong-ewo8wssk4gs8cgg0c8kosk40.jeanlucmakiola.de/rest/v1/" \
|
|
||||||
-H "apikey: <ANON_KEY>"
|
|
||||||
```
|
|
||||||
|
|
||||||
Should return OpenAPI spec. If not:
|
|
||||||
- Check Coolify service status
|
|
||||||
- Verify URL in `.env`
|
|
||||||
- Check network/firewall
|
|
||||||
|
|
||||||
### Environment variables not loading
|
|
||||||
|
|
||||||
- Ensure `.env` exists in project root
|
|
||||||
- Check `.env` has no syntax errors
|
|
||||||
- Restart dev server after changes
|
|
||||||
|
|
||||||
### Database migration issues
|
|
||||||
|
|
||||||
- Verify `SUPABASE_SERVICE_ROLE_KEY` is set
|
|
||||||
- Check migration SQL syntax
|
|
||||||
- Review Supabase logs in Coolify
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## 📚 Documentation
|
|
||||||
|
|
||||||
- [Project Plan](docs/PROJECT_PLAN.md) - Vision, roadmap
|
|
||||||
- [Architecture](docs/ARCHITECTURE.md) - Tech stack, design
|
|
||||||
- [Database Schema](docs/DATABASE.md) - Tables, RLS, functions
|
|
||||||
- [API Reference](docs/API.md) - Endpoints, usage
|
|
||||||
- [Development Guide](docs/DEVELOPMENT.md) - Conventions, workflow
|
|
||||||
- [Deployment](docs/DEPLOYMENT.md) - Docker, Coolify
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
**Last Updated:** 2026-02-08
|
|
||||||
**Status:** Week 1 - Foundation in progress
|
|
||||||
3
app/.env.example
Normal file
@@ -0,0 +1,3 @@
|
|||||||
|
# Supabase Configuration
|
||||||
|
NUXT_PUBLIC_SUPABASE_URL=http://localhost:54321
|
||||||
|
NUXT_PUBLIC_SUPABASE_ANON_KEY=your-anon-key-here
|
||||||
24
app/.gitignore
vendored
Normal file
@@ -0,0 +1,24 @@
|
|||||||
|
# Nuxt dev/build outputs
|
||||||
|
.output
|
||||||
|
.data
|
||||||
|
.nuxt
|
||||||
|
.nitro
|
||||||
|
.cache
|
||||||
|
dist
|
||||||
|
|
||||||
|
# Node dependencies
|
||||||
|
node_modules
|
||||||
|
|
||||||
|
# Logs
|
||||||
|
logs
|
||||||
|
*.log
|
||||||
|
|
||||||
|
# Misc
|
||||||
|
.DS_Store
|
||||||
|
.fleet
|
||||||
|
.idea
|
||||||
|
|
||||||
|
# Local env files
|
||||||
|
.env
|
||||||
|
.env.*
|
||||||
|
!.env.example
|
||||||
75
app/README.md
Normal file
@@ -0,0 +1,75 @@
|
|||||||
|
# Nuxt Minimal Starter
|
||||||
|
|
||||||
|
Look at the [Nuxt documentation](https://nuxt.com/docs/getting-started/introduction) to learn more.
|
||||||
|
|
||||||
|
## Setup
|
||||||
|
|
||||||
|
Make sure to install dependencies:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# npm
|
||||||
|
npm install
|
||||||
|
|
||||||
|
# pnpm
|
||||||
|
pnpm install
|
||||||
|
|
||||||
|
# yarn
|
||||||
|
yarn install
|
||||||
|
|
||||||
|
# bun
|
||||||
|
bun install
|
||||||
|
```
|
||||||
|
|
||||||
|
## Development Server
|
||||||
|
|
||||||
|
Start the development server on `http://localhost:3000`:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# npm
|
||||||
|
npm run dev
|
||||||
|
|
||||||
|
# pnpm
|
||||||
|
pnpm dev
|
||||||
|
|
||||||
|
# yarn
|
||||||
|
yarn dev
|
||||||
|
|
||||||
|
# bun
|
||||||
|
bun run dev
|
||||||
|
```
|
||||||
|
|
||||||
|
## Production
|
||||||
|
|
||||||
|
Build the application for production:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# npm
|
||||||
|
npm run build
|
||||||
|
|
||||||
|
# pnpm
|
||||||
|
pnpm build
|
||||||
|
|
||||||
|
# yarn
|
||||||
|
yarn build
|
||||||
|
|
||||||
|
# bun
|
||||||
|
bun run build
|
||||||
|
```
|
||||||
|
|
||||||
|
Locally preview production build:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# npm
|
||||||
|
npm run preview
|
||||||
|
|
||||||
|
# pnpm
|
||||||
|
pnpm preview
|
||||||
|
|
||||||
|
# yarn
|
||||||
|
yarn preview
|
||||||
|
|
||||||
|
# bun
|
||||||
|
bun run preview
|
||||||
|
```
|
||||||
|
|
||||||
|
Check out the [deployment documentation](https://nuxt.com/docs/getting-started/deployment) for more information.
|
||||||
5
app/app/app.vue
Normal file
@@ -0,0 +1,5 @@
|
|||||||
|
<template>
|
||||||
|
<NuxtLayout>
|
||||||
|
<NuxtPage />
|
||||||
|
</NuxtLayout>
|
||||||
|
</template>
|
||||||
15
app/app/layouts/default.vue
Normal file
@@ -0,0 +1,15 @@
|
|||||||
|
<template>
|
||||||
|
<div class="min-h-screen bg-gray-50">
|
||||||
|
<AppHeader />
|
||||||
|
|
||||||
|
<main class="container mx-auto px-4 py-6 max-w-7xl">
|
||||||
|
<slot />
|
||||||
|
</main>
|
||||||
|
|
||||||
|
<AppFooter />
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script setup lang="ts">
|
||||||
|
// App layout automatically wraps all pages
|
||||||
|
</script>
|
||||||
2262
app/bun.lock
Normal file
40
app/components/AppFooter.vue
Normal file
@@ -0,0 +1,40 @@
|
|||||||
|
<template>
|
||||||
|
<footer class="bg-white border-t border-gray-200 mt-auto">
|
||||||
|
<div class="container mx-auto px-4 py-6 max-w-7xl">
|
||||||
|
<div class="flex flex-col md:flex-row items-center justify-between gap-4">
|
||||||
|
<!-- Copyright -->
|
||||||
|
<p class="text-sm text-gray-600">
|
||||||
|
© {{ currentYear }} Pantry. Self-hosted inventory management.
|
||||||
|
</p>
|
||||||
|
|
||||||
|
<!-- Links -->
|
||||||
|
<div class="flex items-center space-x-6">
|
||||||
|
<a
|
||||||
|
href="https://github.com/pantry-app/pantry"
|
||||||
|
target="_blank"
|
||||||
|
rel="noopener noreferrer"
|
||||||
|
class="text-sm text-gray-600 hover:text-primary transition-colors"
|
||||||
|
>
|
||||||
|
GitHub
|
||||||
|
</a>
|
||||||
|
<NuxtLink
|
||||||
|
to="/about"
|
||||||
|
class="text-sm text-gray-600 hover:text-primary transition-colors"
|
||||||
|
>
|
||||||
|
About
|
||||||
|
</NuxtLink>
|
||||||
|
<NuxtLink
|
||||||
|
to="/privacy"
|
||||||
|
class="text-sm text-gray-600 hover:text-primary transition-colors"
|
||||||
|
>
|
||||||
|
Privacy
|
||||||
|
</NuxtLink>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</footer>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script setup lang="ts">
|
||||||
|
const currentYear = new Date().getFullYear()
|
||||||
|
</script>
|
||||||
122
app/components/AppHeader.vue
Normal file
@@ -0,0 +1,122 @@
|
|||||||
|
<template>
|
||||||
|
<header class="bg-white border-b border-gray-200 sticky top-0 z-50">
|
||||||
|
<div class="container mx-auto px-4 max-w-7xl">
|
||||||
|
<div class="flex items-center justify-between h-16">
|
||||||
|
<!-- Logo / Brand -->
|
||||||
|
<NuxtLink to="/" class="flex items-center space-x-2">
|
||||||
|
<UIcon name="i-heroicons-squares-2x2" class="w-8 h-8 text-primary" />
|
||||||
|
<span class="text-xl font-bold text-gray-900">Pantry</span>
|
||||||
|
</NuxtLink>
|
||||||
|
|
||||||
|
<!-- Navigation -->
|
||||||
|
<nav class="hidden md:flex items-center space-x-6">
|
||||||
|
<NuxtLink
|
||||||
|
to="/"
|
||||||
|
class="text-gray-700 hover:text-primary transition-colors"
|
||||||
|
active-class="text-primary font-semibold"
|
||||||
|
>
|
||||||
|
Inventory
|
||||||
|
</NuxtLink>
|
||||||
|
<NuxtLink
|
||||||
|
to="/scan"
|
||||||
|
class="text-gray-700 hover:text-primary transition-colors"
|
||||||
|
active-class="text-primary font-semibold"
|
||||||
|
>
|
||||||
|
Scan
|
||||||
|
</NuxtLink>
|
||||||
|
<NuxtLink
|
||||||
|
to="/settings"
|
||||||
|
class="text-gray-700 hover:text-primary transition-colors"
|
||||||
|
active-class="text-primary font-semibold"
|
||||||
|
>
|
||||||
|
Settings
|
||||||
|
</NuxtLink>
|
||||||
|
</nav>
|
||||||
|
|
||||||
|
<!-- User Menu -->
|
||||||
|
<div class="flex items-center space-x-4">
|
||||||
|
<UButton
|
||||||
|
v-if="!user"
|
||||||
|
to="/auth/login"
|
||||||
|
color="primary"
|
||||||
|
variant="soft"
|
||||||
|
>
|
||||||
|
Sign In
|
||||||
|
</UButton>
|
||||||
|
|
||||||
|
<UDropdown v-else :items="userMenuItems" mode="hover">
|
||||||
|
<UAvatar
|
||||||
|
:alt="user.email"
|
||||||
|
size="sm"
|
||||||
|
:ui="{ background: 'bg-primary' }"
|
||||||
|
/>
|
||||||
|
</UDropdown>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- Mobile Menu Button -->
|
||||||
|
<UButton
|
||||||
|
icon="i-heroicons-bars-3"
|
||||||
|
color="gray"
|
||||||
|
variant="ghost"
|
||||||
|
class="md:hidden"
|
||||||
|
@click="mobileMenuOpen = !mobileMenuOpen"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- Mobile Navigation -->
|
||||||
|
<nav
|
||||||
|
v-if="mobileMenuOpen"
|
||||||
|
class="md:hidden py-4 space-y-2 border-t border-gray-200"
|
||||||
|
>
|
||||||
|
<NuxtLink
|
||||||
|
to="/"
|
||||||
|
class="block px-4 py-2 text-gray-700 hover:bg-gray-50 rounded"
|
||||||
|
@click="mobileMenuOpen = false"
|
||||||
|
>
|
||||||
|
Inventory
|
||||||
|
</NuxtLink>
|
||||||
|
<NuxtLink
|
||||||
|
to="/scan"
|
||||||
|
class="block px-4 py-2 text-gray-700 hover:bg-gray-50 rounded"
|
||||||
|
@click="mobileMenuOpen = false"
|
||||||
|
>
|
||||||
|
Scan
|
||||||
|
</NuxtLink>
|
||||||
|
<NuxtLink
|
||||||
|
to="/settings"
|
||||||
|
class="block px-4 py-2 text-gray-700 hover:bg-gray-50 rounded"
|
||||||
|
@click="mobileMenuOpen = false"
|
||||||
|
>
|
||||||
|
Settings
|
||||||
|
</NuxtLink>
|
||||||
|
</nav>
|
||||||
|
</div>
|
||||||
|
</header>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script setup lang="ts">
|
||||||
|
const { user, signOut } = useSupabaseAuth()
|
||||||
|
const mobileMenuOpen = ref(false)
|
||||||
|
|
||||||
|
const userMenuItems = [[
|
||||||
|
{
|
||||||
|
label: user.value?.email || 'User',
|
||||||
|
slot: 'account',
|
||||||
|
disabled: true
|
||||||
|
}
|
||||||
|
], [
|
||||||
|
{
|
||||||
|
label: 'Settings',
|
||||||
|
icon: 'i-heroicons-cog-6-tooth',
|
||||||
|
to: '/settings'
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: 'Sign Out',
|
||||||
|
icon: 'i-heroicons-arrow-right-on-rectangle',
|
||||||
|
click: async () => {
|
||||||
|
await signOut()
|
||||||
|
navigateTo('/auth/login')
|
||||||
|
}
|
||||||
|
}
|
||||||
|
]]
|
||||||
|
</script>
|
||||||
231
app/components/inventory/AddItemForm.vue
Normal file
@@ -0,0 +1,231 @@
|
|||||||
|
<template>
|
||||||
|
<UCard>
|
||||||
|
<template #header>
|
||||||
|
<div class="flex items-center justify-between">
|
||||||
|
<h3 class="text-lg font-semibold">Add New Item</h3>
|
||||||
|
<UButton
|
||||||
|
icon="i-heroicons-x-mark"
|
||||||
|
color="gray"
|
||||||
|
variant="ghost"
|
||||||
|
@click="$emit('close')"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<form @submit.prevent="handleSubmit" class="space-y-4">
|
||||||
|
<!-- Item Name -->
|
||||||
|
<UFormGroup label="Item Name" required>
|
||||||
|
<UInput
|
||||||
|
v-model="form.name"
|
||||||
|
placeholder="e.g. Whole Milk, Pasta, Tomatoes"
|
||||||
|
size="lg"
|
||||||
|
autofocus
|
||||||
|
/>
|
||||||
|
</UFormGroup>
|
||||||
|
|
||||||
|
<!-- Quantity & Unit -->
|
||||||
|
<div class="grid grid-cols-2 gap-3">
|
||||||
|
<UFormGroup label="Quantity" required>
|
||||||
|
<UInput
|
||||||
|
v-model.number="form.quantity"
|
||||||
|
type="number"
|
||||||
|
min="0.01"
|
||||||
|
step="0.01"
|
||||||
|
placeholder="1"
|
||||||
|
size="lg"
|
||||||
|
/>
|
||||||
|
</UFormGroup>
|
||||||
|
|
||||||
|
<UFormGroup label="Unit" required>
|
||||||
|
<USelect
|
||||||
|
v-model="form.unit_id"
|
||||||
|
:options="unitOptions"
|
||||||
|
option-attribute="label"
|
||||||
|
value-attribute="value"
|
||||||
|
placeholder="Select unit"
|
||||||
|
size="lg"
|
||||||
|
/>
|
||||||
|
</UFormGroup>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- Expiry Date -->
|
||||||
|
<UFormGroup label="Expiry Date" hint="Optional">
|
||||||
|
<UInput
|
||||||
|
v-model="form.expiry_date"
|
||||||
|
type="date"
|
||||||
|
size="lg"
|
||||||
|
/>
|
||||||
|
</UFormGroup>
|
||||||
|
|
||||||
|
<!-- Notes -->
|
||||||
|
<UFormGroup label="Notes" hint="Optional">
|
||||||
|
<UTextarea
|
||||||
|
v-model="form.notes"
|
||||||
|
placeholder="Any additional notes..."
|
||||||
|
:rows="2"
|
||||||
|
/>
|
||||||
|
</UFormGroup>
|
||||||
|
|
||||||
|
<!-- Tags -->
|
||||||
|
<UFormGroup label="Tags" hint="Optional">
|
||||||
|
<TagsTagPicker v-model="selectedTags" />
|
||||||
|
</UFormGroup>
|
||||||
|
|
||||||
|
<!-- Submit -->
|
||||||
|
<div class="flex gap-2 pt-2">
|
||||||
|
<UButton
|
||||||
|
type="submit"
|
||||||
|
color="primary"
|
||||||
|
size="lg"
|
||||||
|
class="flex-1"
|
||||||
|
:loading="submitting"
|
||||||
|
:disabled="!isValid"
|
||||||
|
>
|
||||||
|
Add Item
|
||||||
|
</UButton>
|
||||||
|
<UButton
|
||||||
|
color="gray"
|
||||||
|
size="lg"
|
||||||
|
variant="soft"
|
||||||
|
@click="$emit('close')"
|
||||||
|
>
|
||||||
|
Cancel
|
||||||
|
</UButton>
|
||||||
|
</div>
|
||||||
|
</form>
|
||||||
|
</UCard>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script setup lang="ts">
|
||||||
|
const { addInventoryItem, addItemTags } = useInventory()
|
||||||
|
const { getUnits } = useUnits()
|
||||||
|
|
||||||
|
const props = defineProps<{
|
||||||
|
initialData?: {
|
||||||
|
barcode?: string
|
||||||
|
name?: string
|
||||||
|
brand?: string
|
||||||
|
image_url?: string
|
||||||
|
quantity?: string
|
||||||
|
}
|
||||||
|
}>()
|
||||||
|
|
||||||
|
const emit = defineEmits<{
|
||||||
|
close: []
|
||||||
|
added: [item: any]
|
||||||
|
}>()
|
||||||
|
|
||||||
|
// Form state
|
||||||
|
const form = reactive({
|
||||||
|
name: '',
|
||||||
|
quantity: 1,
|
||||||
|
unit_id: '',
|
||||||
|
expiry_date: '',
|
||||||
|
notes: ''
|
||||||
|
})
|
||||||
|
|
||||||
|
const submitting = ref(false)
|
||||||
|
const selectedTags = ref<any[]>([])
|
||||||
|
|
||||||
|
// Load units
|
||||||
|
const units = ref<any[]>([])
|
||||||
|
|
||||||
|
onMounted(async () => {
|
||||||
|
const unitsResult = await getUnits()
|
||||||
|
units.value = unitsResult.data || []
|
||||||
|
|
||||||
|
// Set default unit (Piece)
|
||||||
|
const defaultUnit = units.value.find(u => u.abbreviation === 'pc')
|
||||||
|
if (defaultUnit) {
|
||||||
|
form.unit_id = defaultUnit.id
|
||||||
|
}
|
||||||
|
|
||||||
|
// Pre-fill from initial data (scan-to-add flow)
|
||||||
|
if (props.initialData) {
|
||||||
|
if (props.initialData.name) {
|
||||||
|
form.name = props.initialData.name
|
||||||
|
}
|
||||||
|
|
||||||
|
// Add brand to notes if available
|
||||||
|
if (props.initialData.brand) {
|
||||||
|
form.notes = `Brand: ${props.initialData.brand}`
|
||||||
|
|
||||||
|
if (props.initialData.barcode) {
|
||||||
|
form.notes += `\nBarcode: ${props.initialData.barcode}`
|
||||||
|
}
|
||||||
|
} else if (props.initialData.barcode) {
|
||||||
|
form.notes = `Barcode: ${props.initialData.barcode}`
|
||||||
|
}
|
||||||
|
|
||||||
|
// Parse quantity if available (e.g., "750g")
|
||||||
|
if (props.initialData.quantity) {
|
||||||
|
const quantityMatch = props.initialData.quantity.match(/^([\d.]+)\s*([a-zA-Z]+)$/)
|
||||||
|
if (quantityMatch) {
|
||||||
|
form.quantity = parseFloat(quantityMatch[1])
|
||||||
|
// Try to match unit
|
||||||
|
const unitAbbr = quantityMatch[2].toLowerCase()
|
||||||
|
const matchedUnit = units.value.find(u =>
|
||||||
|
u.abbreviation.toLowerCase() === unitAbbr
|
||||||
|
)
|
||||||
|
if (matchedUnit) {
|
||||||
|
form.unit_id = matchedUnit.id
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
// Unit options for select
|
||||||
|
const unitOptions = computed(() => {
|
||||||
|
const grouped: Record<string, any[]> = {}
|
||||||
|
|
||||||
|
for (const unit of units.value) {
|
||||||
|
const type = unit.unit_type
|
||||||
|
if (!grouped[type]) grouped[type] = []
|
||||||
|
grouped[type].push({
|
||||||
|
label: `${unit.name} (${unit.abbreviation})`,
|
||||||
|
value: unit.id
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
return Object.entries(grouped).flatMap(([type, options]) => [
|
||||||
|
{ label: `— ${type.charAt(0).toUpperCase() + type.slice(1)} —`, value: '', disabled: true },
|
||||||
|
...options
|
||||||
|
])
|
||||||
|
})
|
||||||
|
|
||||||
|
// Validation
|
||||||
|
const isValid = computed(() => {
|
||||||
|
return form.name.trim().length > 0 && form.quantity > 0 && form.unit_id
|
||||||
|
})
|
||||||
|
|
||||||
|
// Submit
|
||||||
|
const handleSubmit = async () => {
|
||||||
|
if (!isValid.value) return
|
||||||
|
|
||||||
|
submitting.value = true
|
||||||
|
|
||||||
|
const { data, error } = await addInventoryItem({
|
||||||
|
name: form.name.trim(),
|
||||||
|
quantity: form.quantity,
|
||||||
|
unit_id: form.unit_id,
|
||||||
|
expiry_date: form.expiry_date || null,
|
||||||
|
notes: form.notes.trim() || null
|
||||||
|
})
|
||||||
|
|
||||||
|
if (error) {
|
||||||
|
alert('Failed to add item: ' + error.message)
|
||||||
|
submitting.value = false
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// Add tags if any selected
|
||||||
|
if (data && selectedTags.value.length > 0) {
|
||||||
|
const tagIds = selectedTags.value.map(t => t.id)
|
||||||
|
await addItemTags(data.id, tagIds)
|
||||||
|
}
|
||||||
|
|
||||||
|
emit('added', data)
|
||||||
|
submitting.value = false
|
||||||
|
}
|
||||||
|
</script>
|
||||||
184
app/components/inventory/EditItemModal.vue
Normal file
@@ -0,0 +1,184 @@
|
|||||||
|
<template>
|
||||||
|
<UModal v-model="isOpen">
|
||||||
|
<UCard>
|
||||||
|
<template #header>
|
||||||
|
<div class="flex items-center justify-between">
|
||||||
|
<h3 class="text-lg font-semibold">Edit Item</h3>
|
||||||
|
<UButton
|
||||||
|
icon="i-heroicons-x-mark"
|
||||||
|
color="gray"
|
||||||
|
variant="ghost"
|
||||||
|
@click="close"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<form @submit.prevent="handleSubmit" class="space-y-4">
|
||||||
|
<!-- Item Name -->
|
||||||
|
<UFormGroup label="Item Name" required>
|
||||||
|
<UInput
|
||||||
|
v-model="form.name"
|
||||||
|
placeholder="Item name"
|
||||||
|
size="lg"
|
||||||
|
/>
|
||||||
|
</UFormGroup>
|
||||||
|
|
||||||
|
<!-- Quantity & Unit -->
|
||||||
|
<div class="grid grid-cols-2 gap-3">
|
||||||
|
<UFormGroup label="Quantity" required>
|
||||||
|
<UInput
|
||||||
|
v-model.number="form.quantity"
|
||||||
|
type="number"
|
||||||
|
min="0.01"
|
||||||
|
step="0.01"
|
||||||
|
size="lg"
|
||||||
|
/>
|
||||||
|
</UFormGroup>
|
||||||
|
|
||||||
|
<UFormGroup label="Unit" required>
|
||||||
|
<USelect
|
||||||
|
v-model="form.unit_id"
|
||||||
|
:options="unitOptions"
|
||||||
|
option-attribute="label"
|
||||||
|
value-attribute="value"
|
||||||
|
size="lg"
|
||||||
|
/>
|
||||||
|
</UFormGroup>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- Expiry Date -->
|
||||||
|
<UFormGroup label="Expiry Date" hint="Optional">
|
||||||
|
<UInput
|
||||||
|
v-model="form.expiry_date"
|
||||||
|
type="date"
|
||||||
|
size="lg"
|
||||||
|
/>
|
||||||
|
</UFormGroup>
|
||||||
|
|
||||||
|
<!-- Notes -->
|
||||||
|
<UFormGroup label="Notes" hint="Optional">
|
||||||
|
<UTextarea
|
||||||
|
v-model="form.notes"
|
||||||
|
placeholder="Any additional notes..."
|
||||||
|
:rows="2"
|
||||||
|
/>
|
||||||
|
</UFormGroup>
|
||||||
|
|
||||||
|
<!-- Submit -->
|
||||||
|
<div class="flex gap-2 pt-2">
|
||||||
|
<UButton
|
||||||
|
type="submit"
|
||||||
|
color="primary"
|
||||||
|
size="lg"
|
||||||
|
class="flex-1"
|
||||||
|
:loading="submitting"
|
||||||
|
:disabled="!isValid"
|
||||||
|
>
|
||||||
|
Save Changes
|
||||||
|
</UButton>
|
||||||
|
<UButton
|
||||||
|
color="gray"
|
||||||
|
size="lg"
|
||||||
|
variant="soft"
|
||||||
|
@click="close"
|
||||||
|
>
|
||||||
|
Cancel
|
||||||
|
</UButton>
|
||||||
|
</div>
|
||||||
|
</form>
|
||||||
|
</UCard>
|
||||||
|
</UModal>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script setup lang="ts">
|
||||||
|
const { updateInventoryItem } = useInventory()
|
||||||
|
const { getUnits } = useUnits()
|
||||||
|
|
||||||
|
const props = defineProps<{
|
||||||
|
item: any | null
|
||||||
|
}>()
|
||||||
|
|
||||||
|
const emit = defineEmits<{
|
||||||
|
close: []
|
||||||
|
updated: [item: any]
|
||||||
|
}>()
|
||||||
|
|
||||||
|
const isOpen = ref(false)
|
||||||
|
const submitting = ref(false)
|
||||||
|
const units = ref<any[]>([])
|
||||||
|
|
||||||
|
const form = reactive({
|
||||||
|
name: '',
|
||||||
|
quantity: 1,
|
||||||
|
unit_id: '',
|
||||||
|
expiry_date: '',
|
||||||
|
notes: ''
|
||||||
|
})
|
||||||
|
|
||||||
|
// Load units
|
||||||
|
onMounted(async () => {
|
||||||
|
const { data } = await getUnits()
|
||||||
|
units.value = data || []
|
||||||
|
})
|
||||||
|
|
||||||
|
// Unit options for select
|
||||||
|
const unitOptions = computed(() => {
|
||||||
|
return units.value.map(unit => ({
|
||||||
|
label: `${unit.name} (${unit.abbreviation})`,
|
||||||
|
value: unit.id
|
||||||
|
}))
|
||||||
|
})
|
||||||
|
|
||||||
|
// Watch for item changes (open modal)
|
||||||
|
watch(() => props.item, (newItem) => {
|
||||||
|
if (newItem) {
|
||||||
|
form.name = newItem.name
|
||||||
|
form.quantity = Number(newItem.quantity)
|
||||||
|
form.unit_id = newItem.unit_id
|
||||||
|
form.expiry_date = newItem.expiry_date || ''
|
||||||
|
form.notes = newItem.notes || ''
|
||||||
|
isOpen.value = true
|
||||||
|
}
|
||||||
|
}, { immediate: true })
|
||||||
|
|
||||||
|
// Watch modal close
|
||||||
|
watch(isOpen, (val) => {
|
||||||
|
if (!val) {
|
||||||
|
emit('close')
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
// Validation
|
||||||
|
const isValid = computed(() => {
|
||||||
|
return form.name.trim().length > 0 && form.quantity > 0 && form.unit_id
|
||||||
|
})
|
||||||
|
|
||||||
|
const close = () => {
|
||||||
|
isOpen.value = false
|
||||||
|
}
|
||||||
|
|
||||||
|
// Submit
|
||||||
|
const handleSubmit = async () => {
|
||||||
|
if (!isValid.value || !props.item) return
|
||||||
|
|
||||||
|
submitting.value = true
|
||||||
|
|
||||||
|
const { data, error } = await updateInventoryItem(props.item.id, {
|
||||||
|
name: form.name.trim(),
|
||||||
|
quantity: form.quantity,
|
||||||
|
unit_id: form.unit_id,
|
||||||
|
expiry_date: form.expiry_date || null,
|
||||||
|
notes: form.notes.trim() || null
|
||||||
|
})
|
||||||
|
|
||||||
|
if (error) {
|
||||||
|
alert('Failed to update item: ' + error.message)
|
||||||
|
submitting.value = false
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
emit('updated', data)
|
||||||
|
submitting.value = false
|
||||||
|
close()
|
||||||
|
}
|
||||||
|
</script>
|
||||||
148
app/components/inventory/InventoryCard.vue
Normal file
@@ -0,0 +1,148 @@
|
|||||||
|
<template>
|
||||||
|
<UCard class="hover:shadow-lg transition-shadow">
|
||||||
|
<!-- Item Image -->
|
||||||
|
<div class="aspect-square bg-gray-100 rounded-lg mb-3 overflow-hidden">
|
||||||
|
<img
|
||||||
|
v-if="item.product?.image_url"
|
||||||
|
:src="item.product.image_url"
|
||||||
|
:alt="item.name"
|
||||||
|
class="w-full h-full object-cover"
|
||||||
|
/>
|
||||||
|
<div v-else class="w-full h-full flex items-center justify-center">
|
||||||
|
<UIcon name="i-heroicons-cube" class="w-16 h-16 text-gray-300" />
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- Item Info -->
|
||||||
|
<div class="space-y-2">
|
||||||
|
<div>
|
||||||
|
<h3 class="font-semibold text-gray-900 truncate">{{ item.name }}</h3>
|
||||||
|
<p v-if="item.product?.brand" class="text-sm text-gray-600 truncate">
|
||||||
|
{{ item.product.brand }}
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- Quantity -->
|
||||||
|
<div class="flex items-center justify-between">
|
||||||
|
<span class="text-lg font-medium text-gray-900">
|
||||||
|
{{ item.quantity }} {{ item.unit?.abbreviation }}
|
||||||
|
</span>
|
||||||
|
|
||||||
|
<!-- Quick Actions -->
|
||||||
|
<div class="flex gap-1">
|
||||||
|
<UButton
|
||||||
|
icon="i-heroicons-minus"
|
||||||
|
size="xs"
|
||||||
|
color="gray"
|
||||||
|
variant="ghost"
|
||||||
|
@click="$emit('update-quantity', item.id, -1)"
|
||||||
|
:disabled="item.quantity <= 1"
|
||||||
|
/>
|
||||||
|
<UButton
|
||||||
|
icon="i-heroicons-plus"
|
||||||
|
size="xs"
|
||||||
|
color="gray"
|
||||||
|
variant="ghost"
|
||||||
|
@click="$emit('update-quantity', item.id, 1)"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- Tags -->
|
||||||
|
<div v-if="item.tags && item.tags.length > 0" class="flex flex-wrap gap-1">
|
||||||
|
<TagsTagBadge
|
||||||
|
v-for="tagItem in item.tags.slice(0, 3)"
|
||||||
|
:key="tagItem.tag.id"
|
||||||
|
:tag="tagItem.tag"
|
||||||
|
size="sm"
|
||||||
|
/>
|
||||||
|
<UBadge v-if="item.tags.length > 3" size="xs" color="gray">
|
||||||
|
+{{ item.tags.length - 3 }}
|
||||||
|
</UBadge>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- Expiry Warning -->
|
||||||
|
<div v-if="daysUntilExpiry !== null" class="text-xs">
|
||||||
|
<UBadge
|
||||||
|
:color="expiryColor"
|
||||||
|
variant="soft"
|
||||||
|
class="w-full justify-center"
|
||||||
|
>
|
||||||
|
<UIcon :name="expiryIcon" class="mr-1" />
|
||||||
|
{{ expiryText }}
|
||||||
|
</UBadge>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- Action Buttons -->
|
||||||
|
<template #footer>
|
||||||
|
<div class="flex gap-2">
|
||||||
|
<UButton
|
||||||
|
icon="i-heroicons-pencil"
|
||||||
|
size="sm"
|
||||||
|
color="gray"
|
||||||
|
variant="soft"
|
||||||
|
class="flex-1"
|
||||||
|
@click="$emit('edit', item)"
|
||||||
|
>
|
||||||
|
Edit
|
||||||
|
</UButton>
|
||||||
|
<UButton
|
||||||
|
icon="i-heroicons-trash"
|
||||||
|
size="sm"
|
||||||
|
color="red"
|
||||||
|
variant="soft"
|
||||||
|
@click="$emit('delete', item.id)"
|
||||||
|
>
|
||||||
|
Delete
|
||||||
|
</UButton>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
</UCard>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script setup lang="ts">
|
||||||
|
const props = defineProps<{
|
||||||
|
item: any
|
||||||
|
}>()
|
||||||
|
|
||||||
|
defineEmits<{
|
||||||
|
edit: [item: any]
|
||||||
|
delete: [id: string]
|
||||||
|
'update-quantity': [id: string, change: number]
|
||||||
|
}>()
|
||||||
|
|
||||||
|
// Calculate days until expiry
|
||||||
|
const daysUntilExpiry = computed(() => {
|
||||||
|
if (!props.item.expiry_date) return null
|
||||||
|
|
||||||
|
const today = new Date()
|
||||||
|
const expiry = new Date(props.item.expiry_date)
|
||||||
|
const diff = Math.ceil((expiry.getTime() - today.getTime()) / (1000 * 60 * 60 * 24))
|
||||||
|
|
||||||
|
return diff
|
||||||
|
})
|
||||||
|
|
||||||
|
// Expiry badge styling
|
||||||
|
const expiryColor = computed(() => {
|
||||||
|
if (daysUntilExpiry.value === null) return 'gray'
|
||||||
|
if (daysUntilExpiry.value < 0) return 'red'
|
||||||
|
if (daysUntilExpiry.value <= 3) return 'orange'
|
||||||
|
if (daysUntilExpiry.value <= 7) return 'yellow'
|
||||||
|
return 'green'
|
||||||
|
})
|
||||||
|
|
||||||
|
const expiryIcon = computed(() => {
|
||||||
|
if (daysUntilExpiry.value === null) return 'i-heroicons-calendar'
|
||||||
|
if (daysUntilExpiry.value < 0) return 'i-heroicons-exclamation-triangle'
|
||||||
|
return 'i-heroicons-clock'
|
||||||
|
})
|
||||||
|
|
||||||
|
const expiryText = computed(() => {
|
||||||
|
if (daysUntilExpiry.value === null) return 'No expiry'
|
||||||
|
if (daysUntilExpiry.value < 0) return `Expired ${Math.abs(daysUntilExpiry.value)} days ago`
|
||||||
|
if (daysUntilExpiry.value === 0) return 'Expires today'
|
||||||
|
if (daysUntilExpiry.value === 1) return 'Expires tomorrow'
|
||||||
|
return `Expires in ${daysUntilExpiry.value} days`
|
||||||
|
})
|
||||||
|
</script>
|
||||||
147
app/components/inventory/InventoryList.vue
Normal file
@@ -0,0 +1,147 @@
|
|||||||
|
<template>
|
||||||
|
<div class="space-y-4">
|
||||||
|
<!-- Loading State -->
|
||||||
|
<div v-if="loading" class="text-center py-12">
|
||||||
|
<UIcon name="i-heroicons-arrow-path" class="w-8 h-8 text-gray-400 animate-spin mx-auto mb-2" />
|
||||||
|
<p class="text-gray-600">Loading inventory...</p>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- Error State -->
|
||||||
|
<div v-else-if="error" class="text-center py-12">
|
||||||
|
<UIcon name="i-heroicons-exclamation-triangle" class="w-12 h-12 text-red-500 mx-auto mb-4" />
|
||||||
|
<p class="text-red-600 mb-4">{{ error }}</p>
|
||||||
|
<UButton @click="loadInventory" color="gray">Retry</UButton>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- Empty State -->
|
||||||
|
<div v-else-if="!items || items.length === 0" class="text-center py-12">
|
||||||
|
<UIcon name="i-heroicons-inbox" class="w-16 h-16 text-gray-400 mx-auto mb-4" />
|
||||||
|
<h3 class="text-lg font-semibold text-gray-900 mb-2">
|
||||||
|
No items yet
|
||||||
|
</h3>
|
||||||
|
<p class="text-gray-600 mb-6">
|
||||||
|
Start by scanning a barcode or adding an item manually.
|
||||||
|
</p>
|
||||||
|
<div class="flex gap-2 justify-center">
|
||||||
|
<UButton
|
||||||
|
to="/scan"
|
||||||
|
color="primary"
|
||||||
|
icon="i-heroicons-qr-code"
|
||||||
|
>
|
||||||
|
Scan First Item
|
||||||
|
</UButton>
|
||||||
|
<UButton
|
||||||
|
@click="$emit('add-item')"
|
||||||
|
color="white"
|
||||||
|
icon="i-heroicons-plus"
|
||||||
|
>
|
||||||
|
Add Manually
|
||||||
|
</UButton>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- Inventory Grid -->
|
||||||
|
<div v-else class="grid gap-4 sm:grid-cols-2 lg:grid-cols-3 xl:grid-cols-4">
|
||||||
|
<InventoryCard
|
||||||
|
v-for="item in filteredItems"
|
||||||
|
:key="item.id"
|
||||||
|
:item="item"
|
||||||
|
@edit="$emit('edit-item', item)"
|
||||||
|
@delete="handleDelete(item.id)"
|
||||||
|
@update-quantity="handleQuantityUpdate"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script setup lang="ts">
|
||||||
|
const { getInventory, deleteInventoryItem, updateQuantity } = useInventory()
|
||||||
|
|
||||||
|
const props = defineProps<{
|
||||||
|
refresh?: boolean
|
||||||
|
tagFilters?: string[]
|
||||||
|
}>()
|
||||||
|
|
||||||
|
const emit = defineEmits<{
|
||||||
|
'add-item': []
|
||||||
|
'edit-item': [item: any]
|
||||||
|
}>()
|
||||||
|
|
||||||
|
const items = ref<any[]>([])
|
||||||
|
const loading = ref(true)
|
||||||
|
const error = ref<string | null>(null)
|
||||||
|
|
||||||
|
const loadInventory = async () => {
|
||||||
|
loading.value = true
|
||||||
|
error.value = null
|
||||||
|
|
||||||
|
const { data, error: fetchError } = await getInventory()
|
||||||
|
|
||||||
|
if (fetchError) {
|
||||||
|
error.value = 'Failed to load inventory. Please try again.'
|
||||||
|
loading.value = false
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
items.value = data || []
|
||||||
|
loading.value = false
|
||||||
|
}
|
||||||
|
|
||||||
|
// Computed filtered items
|
||||||
|
const filteredItems = computed(() => {
|
||||||
|
if (!props.tagFilters || props.tagFilters.length === 0) {
|
||||||
|
return items.value
|
||||||
|
}
|
||||||
|
|
||||||
|
// Filter items that have at least one of the selected tags
|
||||||
|
return items.value.filter(item => {
|
||||||
|
if (!item.tags || item.tags.length === 0) return false
|
||||||
|
|
||||||
|
const itemTagIds = item.tags.map((t: any) => t.tag.id)
|
||||||
|
return props.tagFilters!.some(filterId => itemTagIds.includes(filterId))
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
const handleDelete = async (id: string) => {
|
||||||
|
if (!confirm('Are you sure you want to delete this item?')) {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
const { error: deleteError } = await deleteInventoryItem(id)
|
||||||
|
|
||||||
|
if (deleteError) {
|
||||||
|
alert('Failed to delete item')
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// Remove from local list
|
||||||
|
items.value = items.value.filter(item => item.id !== id)
|
||||||
|
}
|
||||||
|
|
||||||
|
const handleQuantityUpdate = async (id: string, change: number) => {
|
||||||
|
const result = await updateQuantity(id, change)
|
||||||
|
|
||||||
|
if (result.error) {
|
||||||
|
alert('Failed to update quantity')
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// Reload inventory after update
|
||||||
|
await loadInventory()
|
||||||
|
}
|
||||||
|
|
||||||
|
// Load on mount
|
||||||
|
onMounted(loadInventory)
|
||||||
|
|
||||||
|
// Watch for refresh prop
|
||||||
|
watch(() => props.refresh, (newVal) => {
|
||||||
|
if (newVal) {
|
||||||
|
loadInventory()
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
// Expose reload method
|
||||||
|
defineExpose({
|
||||||
|
reload: loadInventory
|
||||||
|
})
|
||||||
|
</script>
|
||||||
161
app/components/scan/BarcodeScanner.vue
Normal file
@@ -0,0 +1,161 @@
|
|||||||
|
<template>
|
||||||
|
<div class="relative">
|
||||||
|
<!-- Scanner Container -->
|
||||||
|
<div
|
||||||
|
:id="scannerId"
|
||||||
|
ref="scannerRef"
|
||||||
|
class="w-full rounded-lg overflow-hidden bg-black"
|
||||||
|
:class="{ 'aspect-video': !isScanning, 'min-h-[300px]': isScanning }"
|
||||||
|
/>
|
||||||
|
|
||||||
|
<!-- Overlay when not scanning -->
|
||||||
|
<div
|
||||||
|
v-if="!isScanning && !error"
|
||||||
|
class="absolute inset-0 flex items-center justify-center bg-gray-900 rounded-lg"
|
||||||
|
>
|
||||||
|
<div class="text-center">
|
||||||
|
<UIcon name="i-heroicons-camera" class="w-16 h-16 text-gray-400 mb-4" />
|
||||||
|
<UButton
|
||||||
|
color="primary"
|
||||||
|
size="lg"
|
||||||
|
icon="i-heroicons-qr-code"
|
||||||
|
@click="startScanning"
|
||||||
|
>
|
||||||
|
Start Camera
|
||||||
|
</UButton>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- Error State -->
|
||||||
|
<div
|
||||||
|
v-if="error"
|
||||||
|
class="absolute inset-0 flex items-center justify-center bg-gray-900 rounded-lg"
|
||||||
|
>
|
||||||
|
<div class="text-center px-4">
|
||||||
|
<UIcon name="i-heroicons-exclamation-triangle" class="w-12 h-12 text-red-400 mb-4" />
|
||||||
|
<p class="text-white mb-4">{{ error }}</p>
|
||||||
|
<div class="flex gap-2 justify-center">
|
||||||
|
<UButton @click="startScanning" color="primary">Try Again</UButton>
|
||||||
|
<UButton @click="$emit('manual-entry')" color="gray">Enter Manually</UButton>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- Manual Barcode Entry -->
|
||||||
|
<div class="mt-4">
|
||||||
|
<UFormGroup label="Or enter barcode manually">
|
||||||
|
<div class="flex gap-2">
|
||||||
|
<UInput
|
||||||
|
v-model="manualBarcode"
|
||||||
|
placeholder="e.g. 8000500310427"
|
||||||
|
size="lg"
|
||||||
|
class="flex-1"
|
||||||
|
@keyup.enter="submitManualBarcode"
|
||||||
|
/>
|
||||||
|
<UButton
|
||||||
|
color="primary"
|
||||||
|
size="lg"
|
||||||
|
:disabled="!manualBarcode.trim()"
|
||||||
|
@click="submitManualBarcode"
|
||||||
|
>
|
||||||
|
Lookup
|
||||||
|
</UButton>
|
||||||
|
</div>
|
||||||
|
</UFormGroup>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script setup lang="ts">
|
||||||
|
import { Html5Qrcode, Html5QrcodeSupportedFormats } from 'html5-qrcode'
|
||||||
|
|
||||||
|
const emit = defineEmits<{
|
||||||
|
'barcode-detected': [barcode: string]
|
||||||
|
'manual-entry': []
|
||||||
|
}>()
|
||||||
|
|
||||||
|
const scannerId = 'barcode-scanner'
|
||||||
|
const scannerRef = ref<HTMLElement | null>(null)
|
||||||
|
const isScanning = ref(false)
|
||||||
|
const error = ref<string | null>(null)
|
||||||
|
const manualBarcode = ref('')
|
||||||
|
|
||||||
|
let html5QrCode: Html5Qrcode | null = null
|
||||||
|
|
||||||
|
const startScanning = async () => {
|
||||||
|
error.value = null
|
||||||
|
|
||||||
|
try {
|
||||||
|
if (!html5QrCode) {
|
||||||
|
html5QrCode = new Html5Qrcode(scannerId, {
|
||||||
|
formatsToSupport: [
|
||||||
|
Html5QrcodeSupportedFormats.EAN_13,
|
||||||
|
Html5QrcodeSupportedFormats.EAN_8,
|
||||||
|
Html5QrcodeSupportedFormats.UPC_A,
|
||||||
|
Html5QrcodeSupportedFormats.UPC_E,
|
||||||
|
Html5QrcodeSupportedFormats.CODE_128,
|
||||||
|
Html5QrcodeSupportedFormats.CODE_39,
|
||||||
|
Html5QrcodeSupportedFormats.QR_CODE
|
||||||
|
],
|
||||||
|
verbose: false
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
await html5QrCode.start(
|
||||||
|
{ facingMode: 'environment' },
|
||||||
|
{
|
||||||
|
fps: 10,
|
||||||
|
qrbox: { width: 250, height: 150 },
|
||||||
|
aspectRatio: 1.777
|
||||||
|
},
|
||||||
|
onScanSuccess,
|
||||||
|
onScanFailure
|
||||||
|
)
|
||||||
|
|
||||||
|
isScanning.value = true
|
||||||
|
} catch (err: any) {
|
||||||
|
console.error('Scanner error:', err)
|
||||||
|
|
||||||
|
if (err.toString().includes('NotAllowedError')) {
|
||||||
|
error.value = 'Camera permission denied. Please allow camera access and try again.'
|
||||||
|
} else if (err.toString().includes('NotFoundError')) {
|
||||||
|
error.value = 'No camera found. Please use a device with a camera or enter the barcode manually.'
|
||||||
|
} else {
|
||||||
|
error.value = 'Could not start camera. Try entering the barcode manually.'
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const stopScanning = async () => {
|
||||||
|
if (html5QrCode && isScanning.value) {
|
||||||
|
try {
|
||||||
|
await html5QrCode.stop()
|
||||||
|
} catch (err) {
|
||||||
|
console.error('Error stopping scanner:', err)
|
||||||
|
}
|
||||||
|
isScanning.value = false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const onScanSuccess = (decodedText: string) => {
|
||||||
|
// Stop scanning after successful read
|
||||||
|
stopScanning()
|
||||||
|
emit('barcode-detected', decodedText)
|
||||||
|
}
|
||||||
|
|
||||||
|
const onScanFailure = (_errorMessage: string) => {
|
||||||
|
// Ignore - this fires continuously when no barcode is detected
|
||||||
|
}
|
||||||
|
|
||||||
|
const submitManualBarcode = () => {
|
||||||
|
if (manualBarcode.value.trim()) {
|
||||||
|
emit('barcode-detected', manualBarcode.value.trim())
|
||||||
|
manualBarcode.value = ''
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Cleanup on unmount
|
||||||
|
onUnmounted(() => {
|
||||||
|
stopScanning()
|
||||||
|
})
|
||||||
|
</script>
|
||||||
71
app/components/tags/TagBadge.vue
Normal file
@@ -0,0 +1,71 @@
|
|||||||
|
<template>
|
||||||
|
<UBadge
|
||||||
|
:style="badgeStyle"
|
||||||
|
:class="badgeClasses"
|
||||||
|
v-bind="$attrs"
|
||||||
|
>
|
||||||
|
<span v-if="tag.icon" class="mr-1">{{ tag.icon }}</span>
|
||||||
|
<span>{{ tag.name }}</span>
|
||||||
|
<UButton
|
||||||
|
v-if="removable"
|
||||||
|
icon="i-heroicons-x-mark"
|
||||||
|
size="2xs"
|
||||||
|
color="white"
|
||||||
|
variant="link"
|
||||||
|
class="ml-1 -mr-1"
|
||||||
|
@click.stop="$emit('remove', tag.id)"
|
||||||
|
/>
|
||||||
|
</UBadge>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script setup lang="ts">
|
||||||
|
interface Tag {
|
||||||
|
id: string
|
||||||
|
name: string
|
||||||
|
color: string
|
||||||
|
icon?: string
|
||||||
|
category: string
|
||||||
|
}
|
||||||
|
|
||||||
|
const props = withDefaults(defineProps<{
|
||||||
|
tag: Tag
|
||||||
|
removable?: boolean
|
||||||
|
size?: 'sm' | 'md' | 'lg'
|
||||||
|
}>(), {
|
||||||
|
removable: false,
|
||||||
|
size: 'md'
|
||||||
|
})
|
||||||
|
|
||||||
|
defineEmits<{
|
||||||
|
remove: [tagId: string]
|
||||||
|
}>()
|
||||||
|
|
||||||
|
const badgeStyle = computed(() => ({
|
||||||
|
backgroundColor: props.tag.color,
|
||||||
|
color: getContrastColor(props.tag.color)
|
||||||
|
}))
|
||||||
|
|
||||||
|
const badgeClasses = computed(() => ({
|
||||||
|
'cursor-pointer': props.removable,
|
||||||
|
'text-xs px-2 py-1': props.size === 'sm',
|
||||||
|
'text-sm px-2.5 py-1': props.size === 'md',
|
||||||
|
'text-base px-3 py-1.5': props.size === 'lg'
|
||||||
|
}))
|
||||||
|
|
||||||
|
// Calculate contrast color for text (black or white)
|
||||||
|
function getContrastColor(hexColor: string): string {
|
||||||
|
// Remove # if present
|
||||||
|
const hex = hexColor.replace('#', '')
|
||||||
|
|
||||||
|
// Convert to RGB
|
||||||
|
const r = parseInt(hex.slice(0, 2), 16)
|
||||||
|
const g = parseInt(hex.slice(2, 4), 16)
|
||||||
|
const b = parseInt(hex.slice(4, 6), 16)
|
||||||
|
|
||||||
|
// Calculate luminance
|
||||||
|
const luminance = (0.299 * r + 0.587 * g + 0.114 * b) / 255
|
||||||
|
|
||||||
|
// Return white for dark colors, black for light colors
|
||||||
|
return luminance > 0.5 ? '#000000' : '#FFFFFF'
|
||||||
|
}
|
||||||
|
</script>
|
||||||
127
app/components/tags/TagFilter.vue
Normal file
@@ -0,0 +1,127 @@
|
|||||||
|
<template>
|
||||||
|
<div class="space-y-3">
|
||||||
|
<div class="flex items-center justify-between">
|
||||||
|
<h4 class="text-sm font-semibold text-gray-700">Filter by Tags</h4>
|
||||||
|
<UButton
|
||||||
|
v-if="selectedTagIds.length > 0"
|
||||||
|
size="xs"
|
||||||
|
color="gray"
|
||||||
|
variant="ghost"
|
||||||
|
@click="clearFilters"
|
||||||
|
>
|
||||||
|
Clear
|
||||||
|
</UButton>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- Selected Tags (active filters) -->
|
||||||
|
<div v-if="selectedTagIds.length > 0" class="flex flex-wrap gap-1">
|
||||||
|
<TagsTagBadge
|
||||||
|
v-for="tagId in selectedTagIds"
|
||||||
|
:key="tagId"
|
||||||
|
:tag="findTag(tagId)!"
|
||||||
|
size="sm"
|
||||||
|
removable
|
||||||
|
@remove="toggleTag"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- Available Tags by Category -->
|
||||||
|
<div v-for="category in tagsByCategory" :key="category.name" class="space-y-2">
|
||||||
|
<p class="text-xs font-medium text-gray-500 uppercase">{{ category.name }}</p>
|
||||||
|
<div class="flex flex-wrap gap-1">
|
||||||
|
<UButton
|
||||||
|
v-for="tag in category.tags"
|
||||||
|
:key="tag.id"
|
||||||
|
size="xs"
|
||||||
|
:color="isSelected(tag.id) ? 'primary' : 'gray'"
|
||||||
|
:variant="isSelected(tag.id) ? 'solid' : 'soft'"
|
||||||
|
@click="toggleTag(tag.id)"
|
||||||
|
>
|
||||||
|
<span v-if="tag.icon">{{ tag.icon }}</span>
|
||||||
|
<span class="ml-1">{{ tag.name }}</span>
|
||||||
|
</UButton>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- Loading State -->
|
||||||
|
<div v-if="loading" class="text-center py-4">
|
||||||
|
<div class="inline-block animate-spin rounded-full h-4 w-4 border-b-2 border-primary-500"></div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script setup lang="ts">
|
||||||
|
interface Tag {
|
||||||
|
id: string
|
||||||
|
name: string
|
||||||
|
color: string
|
||||||
|
icon?: string
|
||||||
|
category: string
|
||||||
|
}
|
||||||
|
|
||||||
|
const props = defineProps<{
|
||||||
|
modelValue: string[]
|
||||||
|
}>()
|
||||||
|
|
||||||
|
const emit = defineEmits<{
|
||||||
|
'update:modelValue': [tagIds: string[]]
|
||||||
|
}>()
|
||||||
|
|
||||||
|
const { getTags } = useTags()
|
||||||
|
|
||||||
|
const availableTags = ref<Tag[]>([])
|
||||||
|
const loading = ref(true)
|
||||||
|
|
||||||
|
// Load tags on mount
|
||||||
|
onMounted(async () => {
|
||||||
|
const { data } = await getTags()
|
||||||
|
if (data) {
|
||||||
|
availableTags.value = data
|
||||||
|
}
|
||||||
|
loading.value = false
|
||||||
|
})
|
||||||
|
|
||||||
|
// Computed
|
||||||
|
const selectedTagIds = computed(() => props.modelValue)
|
||||||
|
|
||||||
|
const tagsByCategory = computed(() => {
|
||||||
|
const grouped: Record<string, Tag[]> = {}
|
||||||
|
|
||||||
|
for (const tag of availableTags.value) {
|
||||||
|
if (!grouped[tag.category]) {
|
||||||
|
grouped[tag.category] = []
|
||||||
|
}
|
||||||
|
grouped[tag.category].push(tag)
|
||||||
|
}
|
||||||
|
|
||||||
|
return Object.entries(grouped).map(([name, tags]) => ({
|
||||||
|
name: name.charAt(0).toUpperCase() + name.slice(1),
|
||||||
|
tags: tags.sort((a, b) => a.name.localeCompare(b.name))
|
||||||
|
})).sort((a, b) => {
|
||||||
|
if (a.name === 'Position') return -1
|
||||||
|
if (b.name === 'Position') return 1
|
||||||
|
return a.name.localeCompare(b.name)
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
// Methods
|
||||||
|
const isSelected = (tagId: string) => {
|
||||||
|
return selectedTagIds.value.includes(tagId)
|
||||||
|
}
|
||||||
|
|
||||||
|
const toggleTag = (tagId: string) => {
|
||||||
|
if (isSelected(tagId)) {
|
||||||
|
emit('update:modelValue', selectedTagIds.value.filter(id => id !== tagId))
|
||||||
|
} else {
|
||||||
|
emit('update:modelValue', [...selectedTagIds.value, tagId])
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const clearFilters = () => {
|
||||||
|
emit('update:modelValue', [])
|
||||||
|
}
|
||||||
|
|
||||||
|
const findTag = (tagId: string) => {
|
||||||
|
return availableTags.value.find(t => t.id === tagId)
|
||||||
|
}
|
||||||
|
</script>
|
||||||
212
app/components/tags/TagManager.vue
Normal file
@@ -0,0 +1,212 @@
|
|||||||
|
<template>
|
||||||
|
<div class="space-y-6">
|
||||||
|
<!-- Add New Tag Form -->
|
||||||
|
<UCard>
|
||||||
|
<template #header>
|
||||||
|
<h3 class="text-lg font-semibold">Create New Tag</h3>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<form @submit.prevent="handleCreate" class="space-y-4">
|
||||||
|
<div class="grid grid-cols-2 gap-3">
|
||||||
|
<UFormGroup label="Name" required>
|
||||||
|
<UInput
|
||||||
|
v-model="newTag.name"
|
||||||
|
placeholder="e.g. Freezer, Vegan"
|
||||||
|
size="md"
|
||||||
|
/>
|
||||||
|
</UFormGroup>
|
||||||
|
|
||||||
|
<UFormGroup label="Category" required>
|
||||||
|
<USelect
|
||||||
|
v-model="newTag.category"
|
||||||
|
:options="categoryOptions"
|
||||||
|
placeholder="Select category"
|
||||||
|
size="md"
|
||||||
|
/>
|
||||||
|
</UFormGroup>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="grid grid-cols-2 gap-3">
|
||||||
|
<UFormGroup label="Icon" hint="Single emoji">
|
||||||
|
<UInput
|
||||||
|
v-model="newTag.icon"
|
||||||
|
placeholder="❄️"
|
||||||
|
maxlength="2"
|
||||||
|
size="md"
|
||||||
|
/>
|
||||||
|
</UFormGroup>
|
||||||
|
|
||||||
|
<UFormGroup label="Color" required>
|
||||||
|
<div class="flex gap-2">
|
||||||
|
<UInput
|
||||||
|
v-model="newTag.color"
|
||||||
|
type="color"
|
||||||
|
class="w-16"
|
||||||
|
/>
|
||||||
|
<UInput
|
||||||
|
v-model="newTag.color"
|
||||||
|
placeholder="#3B82F6"
|
||||||
|
class="flex-1"
|
||||||
|
size="md"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</UFormGroup>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<UButton
|
||||||
|
type="submit"
|
||||||
|
color="primary"
|
||||||
|
:loading="creating"
|
||||||
|
:disabled="!isFormValid"
|
||||||
|
>
|
||||||
|
Create Tag
|
||||||
|
</UButton>
|
||||||
|
</form>
|
||||||
|
</UCard>
|
||||||
|
|
||||||
|
<!-- Existing Tags -->
|
||||||
|
<UCard>
|
||||||
|
<template #header>
|
||||||
|
<h3 class="text-lg font-semibold">Existing Tags</h3>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<div v-if="loading" class="text-center py-8">
|
||||||
|
<div class="inline-block animate-spin rounded-full h-8 w-8 border-b-2 border-primary-500"></div>
|
||||||
|
<p class="text-gray-600 mt-2">Loading tags...</p>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div v-else-if="tagsByCategory.length === 0" class="text-center py-8">
|
||||||
|
<p class="text-gray-500">No tags yet. Create your first tag above!</p>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div v-else class="space-y-4">
|
||||||
|
<div v-for="category in tagsByCategory" :key="category.name">
|
||||||
|
<h4 class="text-sm font-semibold text-gray-600 uppercase mb-2">
|
||||||
|
{{ category.name }}
|
||||||
|
</h4>
|
||||||
|
<div class="space-y-2">
|
||||||
|
<div
|
||||||
|
v-for="tag in category.tags"
|
||||||
|
:key="tag.id"
|
||||||
|
class="flex items-center justify-between p-3 bg-gray-50 rounded-lg hover:bg-gray-100"
|
||||||
|
>
|
||||||
|
<div class="flex items-center gap-3">
|
||||||
|
<TagsTagBadge :tag="tag" size="md" />
|
||||||
|
<span class="text-sm text-gray-500">{{ tag.color }}</span>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<UButton
|
||||||
|
icon="i-heroicons-trash"
|
||||||
|
color="red"
|
||||||
|
variant="ghost"
|
||||||
|
size="sm"
|
||||||
|
@click="handleDelete(tag.id, tag.name)"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</UCard>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script setup lang="ts">
|
||||||
|
const { getTags, createTag, deleteTag } = useTags()
|
||||||
|
|
||||||
|
const tags = ref<any[]>([])
|
||||||
|
const loading = ref(true)
|
||||||
|
const creating = ref(false)
|
||||||
|
|
||||||
|
const newTag = reactive({
|
||||||
|
name: '',
|
||||||
|
category: '',
|
||||||
|
icon: '',
|
||||||
|
color: '#3B82F6'
|
||||||
|
})
|
||||||
|
|
||||||
|
const categoryOptions = [
|
||||||
|
{ label: 'Position', value: 'position' },
|
||||||
|
{ label: 'Type', value: 'type' },
|
||||||
|
{ label: 'Custom', value: 'custom' }
|
||||||
|
]
|
||||||
|
|
||||||
|
// Load tags on mount
|
||||||
|
const loadTags = async () => {
|
||||||
|
loading.value = true
|
||||||
|
const { data } = await getTags()
|
||||||
|
if (data) {
|
||||||
|
tags.value = data
|
||||||
|
}
|
||||||
|
loading.value = false
|
||||||
|
}
|
||||||
|
|
||||||
|
onMounted(loadTags)
|
||||||
|
|
||||||
|
// Computed
|
||||||
|
const tagsByCategory = computed(() => {
|
||||||
|
const grouped: Record<string, any[]> = {}
|
||||||
|
|
||||||
|
for (const tag of tags.value) {
|
||||||
|
if (!grouped[tag.category]) {
|
||||||
|
grouped[tag.category] = []
|
||||||
|
}
|
||||||
|
grouped[tag.category].push(tag)
|
||||||
|
}
|
||||||
|
|
||||||
|
return Object.entries(grouped).map(([name, tags]) => ({
|
||||||
|
name: name.charAt(0).toUpperCase() + name.slice(1),
|
||||||
|
tags: tags.sort((a, b) => a.name.localeCompare(b.name))
|
||||||
|
})).sort((a, b) => {
|
||||||
|
if (a.name === 'Position') return -1
|
||||||
|
if (b.name === 'Position') return 1
|
||||||
|
return a.name.localeCompare(b.name)
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
const isFormValid = computed(() => {
|
||||||
|
return newTag.name.trim() && newTag.category && newTag.color
|
||||||
|
})
|
||||||
|
|
||||||
|
// Methods
|
||||||
|
const handleCreate = async () => {
|
||||||
|
if (!isFormValid.value) return
|
||||||
|
|
||||||
|
creating.value = true
|
||||||
|
|
||||||
|
const { data, error } = await createTag({
|
||||||
|
name: newTag.name.trim(),
|
||||||
|
category: newTag.category,
|
||||||
|
icon: newTag.icon.trim() || null,
|
||||||
|
color: newTag.color
|
||||||
|
})
|
||||||
|
|
||||||
|
if (error) {
|
||||||
|
alert('Failed to create tag: ' + error.message)
|
||||||
|
} else {
|
||||||
|
// Reset form
|
||||||
|
newTag.name = ''
|
||||||
|
newTag.category = ''
|
||||||
|
newTag.icon = ''
|
||||||
|
newTag.color = '#3B82F6'
|
||||||
|
|
||||||
|
// Reload tags
|
||||||
|
await loadTags()
|
||||||
|
}
|
||||||
|
|
||||||
|
creating.value = false
|
||||||
|
}
|
||||||
|
|
||||||
|
const handleDelete = async (tagId: string, tagName: string) => {
|
||||||
|
if (!confirm(`Delete tag "${tagName}"? This will remove it from all items.`)) {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
const { error } = await deleteTag(tagId)
|
||||||
|
|
||||||
|
if (error) {
|
||||||
|
alert('Failed to delete tag: ' + error.message)
|
||||||
|
} else {
|
||||||
|
await loadTags()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</script>
|
||||||
125
app/components/tags/TagPicker.vue
Normal file
@@ -0,0 +1,125 @@
|
|||||||
|
<template>
|
||||||
|
<div class="space-y-4">
|
||||||
|
<!-- Selected Tags -->
|
||||||
|
<div v-if="selectedTags.length > 0" class="flex flex-wrap gap-2">
|
||||||
|
<TagsTagBadge
|
||||||
|
v-for="tag in selectedTags"
|
||||||
|
:key="tag.id"
|
||||||
|
:tag="tag"
|
||||||
|
:removable="true"
|
||||||
|
@remove="removeTag"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- Empty State -->
|
||||||
|
<div v-else class="text-sm text-gray-500 italic">
|
||||||
|
No tags selected
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- Tag Selection by Category -->
|
||||||
|
<div v-for="category in tagsByCategory" :key="category.name" class="space-y-2">
|
||||||
|
<h4 class="text-xs font-semibold text-gray-500 uppercase tracking-wide">
|
||||||
|
{{ category.name }}
|
||||||
|
</h4>
|
||||||
|
<div class="flex flex-wrap gap-2">
|
||||||
|
<UButton
|
||||||
|
v-for="tag in category.tags"
|
||||||
|
:key="tag.id"
|
||||||
|
size="sm"
|
||||||
|
:color="isSelected(tag.id) ? 'primary' : 'gray'"
|
||||||
|
:variant="isSelected(tag.id) ? 'solid' : 'outline'"
|
||||||
|
@click="toggleTag(tag)"
|
||||||
|
>
|
||||||
|
<span v-if="tag.icon" class="mr-1">{{ tag.icon }}</span>
|
||||||
|
{{ tag.name }}
|
||||||
|
</UButton>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- Loading State -->
|
||||||
|
<div v-if="loading" class="text-center py-4">
|
||||||
|
<div class="inline-block animate-spin rounded-full h-6 w-6 border-b-2 border-primary-500"></div>
|
||||||
|
<p class="text-sm text-gray-500 mt-2">Loading tags...</p>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- Empty State (no tags available) -->
|
||||||
|
<div v-if="!loading && availableTags.length === 0" class="text-center py-4">
|
||||||
|
<p class="text-gray-500">No tags available</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script setup lang="ts">
|
||||||
|
interface Tag {
|
||||||
|
id: string
|
||||||
|
name: string
|
||||||
|
color: string
|
||||||
|
icon?: string
|
||||||
|
category: string
|
||||||
|
}
|
||||||
|
|
||||||
|
const props = defineProps<{
|
||||||
|
modelValue: Tag[]
|
||||||
|
}>()
|
||||||
|
|
||||||
|
const emit = defineEmits<{
|
||||||
|
'update:modelValue': [tags: Tag[]]
|
||||||
|
}>()
|
||||||
|
|
||||||
|
const { getTags } = useTags()
|
||||||
|
|
||||||
|
const availableTags = ref<Tag[]>([])
|
||||||
|
const loading = ref(true)
|
||||||
|
|
||||||
|
// Load tags on mount
|
||||||
|
onMounted(async () => {
|
||||||
|
const { data, error } = await getTags()
|
||||||
|
if (data) {
|
||||||
|
availableTags.value = data
|
||||||
|
}
|
||||||
|
loading.value = false
|
||||||
|
})
|
||||||
|
|
||||||
|
// Computed
|
||||||
|
const selectedTags = computed(() => props.modelValue)
|
||||||
|
|
||||||
|
const tagsByCategory = computed(() => {
|
||||||
|
const grouped: Record<string, Tag[]> = {}
|
||||||
|
|
||||||
|
for (const tag of availableTags.value) {
|
||||||
|
if (!grouped[tag.category]) {
|
||||||
|
grouped[tag.category] = []
|
||||||
|
}
|
||||||
|
grouped[tag.category].push(tag)
|
||||||
|
}
|
||||||
|
|
||||||
|
return Object.entries(grouped).map(([name, tags]) => ({
|
||||||
|
name: name.charAt(0).toUpperCase() + name.slice(1),
|
||||||
|
tags: tags.sort((a, b) => a.name.localeCompare(b.name))
|
||||||
|
})).sort((a, b) => {
|
||||||
|
// Position category first, then others alphabetically
|
||||||
|
if (a.name === 'Position') return -1
|
||||||
|
if (b.name === 'Position') return 1
|
||||||
|
return a.name.localeCompare(b.name)
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
// Methods
|
||||||
|
const isSelected = (tagId: string) => {
|
||||||
|
return selectedTags.value.some(t => t.id === tagId)
|
||||||
|
}
|
||||||
|
|
||||||
|
const toggleTag = (tag: Tag) => {
|
||||||
|
const isCurrentlySelected = isSelected(tag.id)
|
||||||
|
|
||||||
|
if (isCurrentlySelected) {
|
||||||
|
removeTag(tag.id)
|
||||||
|
} else {
|
||||||
|
emit('update:modelValue', [...selectedTags.value, tag])
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const removeTag = (tagId: string) => {
|
||||||
|
emit('update:modelValue', selectedTags.value.filter(t => t.id !== tagId))
|
||||||
|
}
|
||||||
|
</script>
|
||||||
201
app/composables/useInventory.ts
Normal file
@@ -0,0 +1,201 @@
|
|||||||
|
import type { Database } from '~/types/database.types'
|
||||||
|
|
||||||
|
type InventoryItem = Database['public']['Tables']['inventory_items']['Row']
|
||||||
|
type InventoryItemInsert = Database['public']['Tables']['inventory_items']['Insert']
|
||||||
|
type InventoryItemUpdate = Database['public']['Tables']['inventory_items']['Update']
|
||||||
|
|
||||||
|
export const useInventory = () => {
|
||||||
|
const supabase = useSupabase()
|
||||||
|
const { user } = useSupabaseAuth()
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get all inventory items with denormalized data
|
||||||
|
*/
|
||||||
|
const getInventory = async () => {
|
||||||
|
const { data, error } = await supabase
|
||||||
|
.from('inventory_items')
|
||||||
|
.select(`
|
||||||
|
*,
|
||||||
|
product:products(*),
|
||||||
|
unit:units(*),
|
||||||
|
tags:item_tags(tag:tags(*))
|
||||||
|
`)
|
||||||
|
.order('created_at', { ascending: false })
|
||||||
|
|
||||||
|
if (error) {
|
||||||
|
console.error('Error fetching inventory:', error)
|
||||||
|
return { data: null, error }
|
||||||
|
}
|
||||||
|
|
||||||
|
return { data, error: null }
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get single inventory item by ID
|
||||||
|
*/
|
||||||
|
const getInventoryItem = async (id: string) => {
|
||||||
|
const { data, error } = await supabase
|
||||||
|
.from('inventory_items')
|
||||||
|
.select(`
|
||||||
|
*,
|
||||||
|
product:products(*),
|
||||||
|
unit:units(*),
|
||||||
|
tags:item_tags(tag:tags(*))
|
||||||
|
`)
|
||||||
|
.eq('id', id)
|
||||||
|
.single()
|
||||||
|
|
||||||
|
if (error) {
|
||||||
|
console.error('Error fetching item:', error)
|
||||||
|
return { data: null, error }
|
||||||
|
}
|
||||||
|
|
||||||
|
return { data, error: null }
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Add new inventory item
|
||||||
|
*/
|
||||||
|
const addInventoryItem = async (item: Omit<InventoryItemInsert, 'added_by'>) => {
|
||||||
|
if (!user.value) {
|
||||||
|
return { data: null, error: { message: 'User not authenticated' } }
|
||||||
|
}
|
||||||
|
|
||||||
|
const { data, error } = await supabase
|
||||||
|
.from('inventory_items')
|
||||||
|
.insert({
|
||||||
|
...item,
|
||||||
|
added_by: user.value.id
|
||||||
|
})
|
||||||
|
.select(`
|
||||||
|
*,
|
||||||
|
product:products(*),
|
||||||
|
unit:units(*),
|
||||||
|
tags:item_tags(tag:tags(*))
|
||||||
|
`)
|
||||||
|
.single()
|
||||||
|
|
||||||
|
if (error) {
|
||||||
|
console.error('Error adding item:', error)
|
||||||
|
return { data: null, error }
|
||||||
|
}
|
||||||
|
|
||||||
|
return { data, error: null }
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Update inventory item
|
||||||
|
*/
|
||||||
|
const updateInventoryItem = async (id: string, updates: InventoryItemUpdate) => {
|
||||||
|
const { data, error } = await supabase
|
||||||
|
.from('inventory_items')
|
||||||
|
.update(updates)
|
||||||
|
.eq('id', id)
|
||||||
|
.select(`
|
||||||
|
*,
|
||||||
|
product:products(*),
|
||||||
|
unit:units(*),
|
||||||
|
tags:item_tags(tag:tags(*))
|
||||||
|
`)
|
||||||
|
.single()
|
||||||
|
|
||||||
|
if (error) {
|
||||||
|
console.error('Error updating item:', error)
|
||||||
|
return { data: null, error }
|
||||||
|
}
|
||||||
|
|
||||||
|
return { data, error: null }
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Delete inventory item
|
||||||
|
*/
|
||||||
|
const deleteInventoryItem = async (id: string) => {
|
||||||
|
// First delete associated tags
|
||||||
|
await supabase
|
||||||
|
.from('item_tags')
|
||||||
|
.delete()
|
||||||
|
.eq('item_id', id)
|
||||||
|
|
||||||
|
const { error } = await supabase
|
||||||
|
.from('inventory_items')
|
||||||
|
.delete()
|
||||||
|
.eq('id', id)
|
||||||
|
|
||||||
|
if (error) {
|
||||||
|
console.error('Error deleting item:', error)
|
||||||
|
return { error }
|
||||||
|
}
|
||||||
|
|
||||||
|
return { error: null }
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Update item quantity (consume or restock)
|
||||||
|
*/
|
||||||
|
const updateQuantity = async (id: string, change: number) => {
|
||||||
|
const { data: item, error: fetchError } = await getInventoryItem(id)
|
||||||
|
if (fetchError || !item) {
|
||||||
|
return { data: null, error: fetchError }
|
||||||
|
}
|
||||||
|
|
||||||
|
const newQuantity = Number(item.quantity) + change
|
||||||
|
|
||||||
|
if (newQuantity <= 0) {
|
||||||
|
// Auto-delete when quantity reaches zero
|
||||||
|
return await deleteInventoryItem(id)
|
||||||
|
}
|
||||||
|
|
||||||
|
return await updateInventoryItem(id, { quantity: newQuantity })
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Add tags to item
|
||||||
|
*/
|
||||||
|
const addItemTags = async (itemId: string, tagIds: string[]) => {
|
||||||
|
const items = tagIds.map(tagId => ({
|
||||||
|
item_id: itemId,
|
||||||
|
tag_id: tagId
|
||||||
|
}))
|
||||||
|
|
||||||
|
const { error } = await supabase
|
||||||
|
.from('item_tags')
|
||||||
|
.insert(items)
|
||||||
|
|
||||||
|
if (error) {
|
||||||
|
console.error('Error adding tags:', error)
|
||||||
|
return { error }
|
||||||
|
}
|
||||||
|
|
||||||
|
return { error: null }
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Remove tag from item
|
||||||
|
*/
|
||||||
|
const removeItemTag = async (itemId: string, tagId: string) => {
|
||||||
|
const { error } = await supabase
|
||||||
|
.from('item_tags')
|
||||||
|
.delete()
|
||||||
|
.eq('item_id', itemId)
|
||||||
|
.eq('tag_id', tagId)
|
||||||
|
|
||||||
|
if (error) {
|
||||||
|
console.error('Error removing tag:', error)
|
||||||
|
return { error }
|
||||||
|
}
|
||||||
|
|
||||||
|
return { error: null }
|
||||||
|
}
|
||||||
|
|
||||||
|
return {
|
||||||
|
getInventory,
|
||||||
|
getInventoryItem,
|
||||||
|
addInventoryItem,
|
||||||
|
updateInventoryItem,
|
||||||
|
deleteInventoryItem,
|
||||||
|
updateQuantity,
|
||||||
|
addItemTags,
|
||||||
|
removeItemTag
|
||||||
|
}
|
||||||
|
}
|
||||||
61
app/composables/useProductLookup.ts
Normal file
@@ -0,0 +1,61 @@
|
|||||||
|
// Composable for product lookup via Edge Function
|
||||||
|
|
||||||
|
export interface ProductData {
|
||||||
|
barcode: string
|
||||||
|
name: string
|
||||||
|
brand?: string
|
||||||
|
quantity?: string
|
||||||
|
image_url?: string
|
||||||
|
category?: string
|
||||||
|
cached?: boolean
|
||||||
|
}
|
||||||
|
|
||||||
|
export const useProductLookup = () => {
|
||||||
|
const supabase = useSupabaseClient()
|
||||||
|
const isLoading = ref(false)
|
||||||
|
const error = ref<string | null>(null)
|
||||||
|
|
||||||
|
const lookupProduct = async (barcode: string): Promise<ProductData | null> => {
|
||||||
|
isLoading.value = true
|
||||||
|
error.value = null
|
||||||
|
|
||||||
|
try {
|
||||||
|
const { data, error: functionError } = await supabase.functions.invoke('product-lookup', {
|
||||||
|
body: { barcode }
|
||||||
|
})
|
||||||
|
|
||||||
|
if (functionError) {
|
||||||
|
console.error('Product lookup error:', functionError)
|
||||||
|
error.value = functionError.message || 'Failed to lookup product'
|
||||||
|
|
||||||
|
// Return basic product data even on error
|
||||||
|
return {
|
||||||
|
barcode,
|
||||||
|
name: `Product ${barcode}`,
|
||||||
|
cached: false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return data as ProductData
|
||||||
|
|
||||||
|
} catch (err) {
|
||||||
|
console.error('Unexpected error during product lookup:', err)
|
||||||
|
error.value = err instanceof Error ? err.message : 'Unknown error'
|
||||||
|
|
||||||
|
// Return basic product data even on error
|
||||||
|
return {
|
||||||
|
barcode,
|
||||||
|
name: `Product ${barcode}`,
|
||||||
|
cached: false
|
||||||
|
}
|
||||||
|
} finally {
|
||||||
|
isLoading.value = false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return {
|
||||||
|
lookupProduct,
|
||||||
|
isLoading: readonly(isLoading),
|
||||||
|
error: readonly(error)
|
||||||
|
}
|
||||||
|
}
|
||||||
81
app/composables/useSupabase.ts
Normal file
@@ -0,0 +1,81 @@
|
|||||||
|
import { createClient, SupabaseClient } from '@supabase/supabase-js'
|
||||||
|
import type { Database } from '~/types/database.types'
|
||||||
|
|
||||||
|
let supabaseInstance: SupabaseClient<Database> | null = null
|
||||||
|
|
||||||
|
export const useSupabase = () => {
|
||||||
|
const config = useRuntimeConfig()
|
||||||
|
|
||||||
|
if (!supabaseInstance) {
|
||||||
|
supabaseInstance = createClient<Database>(
|
||||||
|
config.public.supabaseUrl,
|
||||||
|
config.public.supabaseAnonKey,
|
||||||
|
{
|
||||||
|
auth: {
|
||||||
|
persistSession: true,
|
||||||
|
autoRefreshToken: true,
|
||||||
|
detectSessionInUrl: true,
|
||||||
|
storage: process.client ? window.localStorage : undefined
|
||||||
|
}
|
||||||
|
}
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
return supabaseInstance
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Composable for Supabase authentication
|
||||||
|
*/
|
||||||
|
export const useSupabaseAuth = () => {
|
||||||
|
const supabase = useSupabase()
|
||||||
|
const user = useState('supabase_user', () => null as any)
|
||||||
|
const session = useState('supabase_session', () => null as any)
|
||||||
|
|
||||||
|
// Initialize auth state
|
||||||
|
const initAuth = async () => {
|
||||||
|
const { data: { session: currentSession } } = await supabase.auth.getSession()
|
||||||
|
session.value = currentSession
|
||||||
|
user.value = currentSession?.user || null
|
||||||
|
|
||||||
|
// Listen for auth changes
|
||||||
|
supabase.auth.onAuthStateChange((_event, newSession) => {
|
||||||
|
session.value = newSession
|
||||||
|
user.value = newSession?.user || null
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
// Call initAuth on composable mount
|
||||||
|
if (process.client) {
|
||||||
|
initAuth()
|
||||||
|
}
|
||||||
|
|
||||||
|
const signIn = async (email: string, password: string) => {
|
||||||
|
const { data, error } = await supabase.auth.signInWithPassword({
|
||||||
|
email,
|
||||||
|
password
|
||||||
|
})
|
||||||
|
return { data, error }
|
||||||
|
}
|
||||||
|
|
||||||
|
const signUp = async (email: string, password: string) => {
|
||||||
|
const { data, error } = await supabase.auth.signUp({
|
||||||
|
email,
|
||||||
|
password
|
||||||
|
})
|
||||||
|
return { data, error }
|
||||||
|
}
|
||||||
|
|
||||||
|
const signOut = async () => {
|
||||||
|
const { error } = await supabase.auth.signOut()
|
||||||
|
return { error }
|
||||||
|
}
|
||||||
|
|
||||||
|
return {
|
||||||
|
user: readonly(user),
|
||||||
|
session: readonly(session),
|
||||||
|
signIn,
|
||||||
|
signUp,
|
||||||
|
signOut
|
||||||
|
}
|
||||||
|
}
|
||||||
86
app/composables/useTags.ts
Normal file
@@ -0,0 +1,86 @@
|
|||||||
|
export const useTags = () => {
|
||||||
|
const supabase = useSupabase()
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get all tags
|
||||||
|
*/
|
||||||
|
const getTags = async () => {
|
||||||
|
const { data, error } = await supabase
|
||||||
|
.from('tags')
|
||||||
|
.select('*')
|
||||||
|
.order('category', { ascending: true })
|
||||||
|
.order('name', { ascending: true })
|
||||||
|
|
||||||
|
if (error) {
|
||||||
|
console.error('Error fetching tags:', error)
|
||||||
|
return { data: null, error }
|
||||||
|
}
|
||||||
|
|
||||||
|
return { data, error: null }
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get tags by category
|
||||||
|
*/
|
||||||
|
const getTagsByCategory = async (category: 'position' | 'type' | 'dietary' | 'custom') => {
|
||||||
|
const { data, error } = await supabase
|
||||||
|
.from('tags')
|
||||||
|
.select('*')
|
||||||
|
.eq('category', category)
|
||||||
|
.order('name', { ascending: true })
|
||||||
|
|
||||||
|
if (error) {
|
||||||
|
console.error('Error fetching tags by category:', error)
|
||||||
|
return { data: null, error }
|
||||||
|
}
|
||||||
|
|
||||||
|
return { data, error: null }
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create a new tag
|
||||||
|
*/
|
||||||
|
const createTag = async (tag: {
|
||||||
|
name: string
|
||||||
|
category: string
|
||||||
|
icon?: string | null
|
||||||
|
color: string
|
||||||
|
}) => {
|
||||||
|
const { data, error } = await supabase
|
||||||
|
.from('tags')
|
||||||
|
.insert(tag)
|
||||||
|
.select()
|
||||||
|
.single()
|
||||||
|
|
||||||
|
if (error) {
|
||||||
|
console.error('Error creating tag:', error)
|
||||||
|
return { data: null, error }
|
||||||
|
}
|
||||||
|
|
||||||
|
return { data, error: null }
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Delete a tag
|
||||||
|
*/
|
||||||
|
const deleteTag = async (tagId: string) => {
|
||||||
|
const { error } = await supabase
|
||||||
|
.from('tags')
|
||||||
|
.delete()
|
||||||
|
.eq('id', tagId)
|
||||||
|
|
||||||
|
if (error) {
|
||||||
|
console.error('Error deleting tag:', error)
|
||||||
|
return { error }
|
||||||
|
}
|
||||||
|
|
||||||
|
return { error: null }
|
||||||
|
}
|
||||||
|
|
||||||
|
return {
|
||||||
|
getTags,
|
||||||
|
getTagsByCategory,
|
||||||
|
createTag,
|
||||||
|
deleteTag
|
||||||
|
}
|
||||||
|
}
|
||||||
53
app/composables/useUnits.ts
Normal file
@@ -0,0 +1,53 @@
|
|||||||
|
export const useUnits = () => {
|
||||||
|
const supabase = useSupabase()
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get all units
|
||||||
|
*/
|
||||||
|
const getUnits = async () => {
|
||||||
|
const { data, error } = await supabase
|
||||||
|
.from('units')
|
||||||
|
.select('*')
|
||||||
|
.order('unit_type', { ascending: true })
|
||||||
|
.order('name', { ascending: true })
|
||||||
|
|
||||||
|
if (error) {
|
||||||
|
console.error('Error fetching units:', error)
|
||||||
|
return { data: null, error }
|
||||||
|
}
|
||||||
|
|
||||||
|
return { data, error: null }
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get default unit for a type
|
||||||
|
*/
|
||||||
|
const getDefaultUnit = async (unitType: 'weight' | 'volume' | 'count' | 'custom') => {
|
||||||
|
const { data, error } = await supabase
|
||||||
|
.from('units')
|
||||||
|
.select('*')
|
||||||
|
.eq('unit_type', unitType)
|
||||||
|
.eq('is_default', true)
|
||||||
|
.single()
|
||||||
|
|
||||||
|
if (error) {
|
||||||
|
console.error('Error fetching default unit:', error)
|
||||||
|
return { data: null, error }
|
||||||
|
}
|
||||||
|
|
||||||
|
return { data, error: null }
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Convert quantity between units
|
||||||
|
*/
|
||||||
|
const convertUnit = (quantity: number, fromFactor: number, toFactor: number): number => {
|
||||||
|
return (quantity * fromFactor) / toFactor
|
||||||
|
}
|
||||||
|
|
||||||
|
return {
|
||||||
|
getUnits,
|
||||||
|
getDefaultUnit,
|
||||||
|
convertUnit
|
||||||
|
}
|
||||||
|
}
|
||||||
119
app/nuxt.config.ts
Normal file
@@ -0,0 +1,119 @@
|
|||||||
|
// https://nuxt.com/docs/api/configuration/nuxt-config
|
||||||
|
export default defineNuxtConfig({
|
||||||
|
compatibilityDate: '2025-07-15',
|
||||||
|
devtools: { enabled: true },
|
||||||
|
|
||||||
|
modules: [
|
||||||
|
'@nuxt/ui',
|
||||||
|
'@nuxt/fonts',
|
||||||
|
'@vite-pwa/nuxt'
|
||||||
|
],
|
||||||
|
|
||||||
|
runtimeConfig: {
|
||||||
|
public: {
|
||||||
|
supabaseUrl: process.env.NUXT_PUBLIC_SUPABASE_URL || 'http://localhost:54321',
|
||||||
|
supabaseAnonKey: process.env.NUXT_PUBLIC_SUPABASE_ANON_KEY || ''
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
colorMode: {
|
||||||
|
preference: 'light'
|
||||||
|
},
|
||||||
|
|
||||||
|
pwa: {
|
||||||
|
registerType: 'autoUpdate',
|
||||||
|
manifest: {
|
||||||
|
name: 'Pantry - Smart Inventory Manager',
|
||||||
|
short_name: 'Pantry',
|
||||||
|
description: 'Track your household pantry inventory with ease. Barcode scanning, smart organization, and multi-user support.',
|
||||||
|
theme_color: '#10b981',
|
||||||
|
background_color: '#ffffff',
|
||||||
|
display: 'standalone',
|
||||||
|
orientation: 'portrait',
|
||||||
|
scope: '/',
|
||||||
|
start_url: '/',
|
||||||
|
categories: ['productivity', 'lifestyle'],
|
||||||
|
icons: [
|
||||||
|
{
|
||||||
|
src: '/icon-192x192.png',
|
||||||
|
sizes: '192x192',
|
||||||
|
type: 'image/png',
|
||||||
|
purpose: 'any'
|
||||||
|
},
|
||||||
|
{
|
||||||
|
src: '/icon-512x512.png',
|
||||||
|
sizes: '512x512',
|
||||||
|
type: 'image/png',
|
||||||
|
purpose: 'any'
|
||||||
|
},
|
||||||
|
{
|
||||||
|
src: '/icon-192x192-maskable.png',
|
||||||
|
sizes: '192x192',
|
||||||
|
type: 'image/png',
|
||||||
|
purpose: 'maskable'
|
||||||
|
},
|
||||||
|
{
|
||||||
|
src: '/icon-512x512-maskable.png',
|
||||||
|
sizes: '512x512',
|
||||||
|
type: 'image/png',
|
||||||
|
purpose: 'maskable'
|
||||||
|
}
|
||||||
|
],
|
||||||
|
screenshots: [
|
||||||
|
{
|
||||||
|
src: '/screenshot-mobile.png',
|
||||||
|
sizes: '390x844',
|
||||||
|
type: 'image/png',
|
||||||
|
form_factor: 'narrow',
|
||||||
|
label: 'Pantry inventory view on mobile'
|
||||||
|
},
|
||||||
|
{
|
||||||
|
src: '/screenshot-desktop.png',
|
||||||
|
sizes: '1920x1080',
|
||||||
|
type: 'image/png',
|
||||||
|
form_factor: 'wide',
|
||||||
|
label: 'Pantry inventory view on desktop'
|
||||||
|
}
|
||||||
|
]
|
||||||
|
},
|
||||||
|
workbox: {
|
||||||
|
navigateFallback: '/',
|
||||||
|
globPatterns: ['**/*.{js,css,html,png,svg,ico}'],
|
||||||
|
cleanupOutdatedCaches: true,
|
||||||
|
runtimeCaching: [
|
||||||
|
{
|
||||||
|
urlPattern: /^https:\/\/api\.supabase\.co\/.*/i,
|
||||||
|
handler: 'NetworkFirst',
|
||||||
|
options: {
|
||||||
|
cacheName: 'supabase-api',
|
||||||
|
expiration: {
|
||||||
|
maxEntries: 100,
|
||||||
|
maxAgeSeconds: 60 * 60 * 24 // 24 hours
|
||||||
|
},
|
||||||
|
cacheableResponse: {
|
||||||
|
statuses: [0, 200]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
urlPattern: /^https:\/\/.*\.supabase\.co\/.*/i,
|
||||||
|
handler: 'NetworkFirst',
|
||||||
|
options: {
|
||||||
|
cacheName: 'supabase-data',
|
||||||
|
expiration: {
|
||||||
|
maxEntries: 100,
|
||||||
|
maxAgeSeconds: 60 * 60 * 24 // 24 hours
|
||||||
|
},
|
||||||
|
cacheableResponse: {
|
||||||
|
statuses: [0, 200]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
]
|
||||||
|
},
|
||||||
|
devOptions: {
|
||||||
|
enabled: true,
|
||||||
|
type: 'module'
|
||||||
|
}
|
||||||
|
}
|
||||||
|
})
|
||||||
18031
app/package-lock.json
generated
Normal file
27
app/package.json
Normal file
@@ -0,0 +1,27 @@
|
|||||||
|
{
|
||||||
|
"name": "app",
|
||||||
|
"type": "module",
|
||||||
|
"private": true,
|
||||||
|
"scripts": {
|
||||||
|
"build": "nuxt build",
|
||||||
|
"dev": "nuxt dev",
|
||||||
|
"generate": "nuxt generate",
|
||||||
|
"preview": "nuxt preview",
|
||||||
|
"postinstall": "nuxt prepare",
|
||||||
|
"generate:icons": "node scripts/generate-icons.js && node scripts/generate-screenshots.js"
|
||||||
|
},
|
||||||
|
"dependencies": {
|
||||||
|
"@nuxt/fonts": "^0.13.0",
|
||||||
|
"@nuxt/ui": "^4.4.0",
|
||||||
|
"@supabase/supabase-js": "^2.95.3",
|
||||||
|
"html5-qrcode": "^2.3.8",
|
||||||
|
"nuxt": "^4.3.1",
|
||||||
|
"vue": "^3.5.28",
|
||||||
|
"vue-router": "^4.6.4"
|
||||||
|
},
|
||||||
|
"devDependencies": {
|
||||||
|
"@nuxtjs/tailwindcss": "^6.14.0",
|
||||||
|
"@vite-pwa/nuxt": "^1.1.1",
|
||||||
|
"sharp": "^0.34.5"
|
||||||
|
}
|
||||||
|
}
|
||||||
121
app/pages/index.vue
Normal file
@@ -0,0 +1,121 @@
|
|||||||
|
<template>
|
||||||
|
<div>
|
||||||
|
<div class="flex items-center justify-between mb-6">
|
||||||
|
<h1 class="text-3xl font-bold text-gray-900">Inventory</h1>
|
||||||
|
|
||||||
|
<div class="flex gap-2">
|
||||||
|
<UButton
|
||||||
|
to="/scan"
|
||||||
|
color="primary"
|
||||||
|
size="lg"
|
||||||
|
icon="i-heroicons-qr-code"
|
||||||
|
>
|
||||||
|
Scan Item
|
||||||
|
</UButton>
|
||||||
|
|
||||||
|
<UButton
|
||||||
|
color="white"
|
||||||
|
size="lg"
|
||||||
|
icon="i-heroicons-plus"
|
||||||
|
@click="showAddForm = true"
|
||||||
|
>
|
||||||
|
Add Manually
|
||||||
|
</UButton>
|
||||||
|
|
||||||
|
<UButton
|
||||||
|
color="gray"
|
||||||
|
size="lg"
|
||||||
|
icon="i-heroicons-funnel"
|
||||||
|
@click="showFilters = !showFilters"
|
||||||
|
>
|
||||||
|
Filter
|
||||||
|
</UButton>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- Tag Filters -->
|
||||||
|
<UCard v-if="showFilters" class="mb-6">
|
||||||
|
<TagsTagFilter v-model="selectedTagFilters" />
|
||||||
|
</UCard>
|
||||||
|
|
||||||
|
<!-- Add Item Form (Overlay) -->
|
||||||
|
<div v-if="showAddForm" class="fixed inset-0 z-50 flex items-start justify-center pt-20 px-4 bg-black/50">
|
||||||
|
<div class="w-full max-w-lg">
|
||||||
|
<AddItemForm
|
||||||
|
:initial-data="prefilledData"
|
||||||
|
@close="handleCloseAddForm"
|
||||||
|
@added="handleItemAdded"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- Edit Item Modal -->
|
||||||
|
<EditItemModal
|
||||||
|
:item="editingItem"
|
||||||
|
@close="editingItem = null"
|
||||||
|
@updated="handleItemUpdated"
|
||||||
|
/>
|
||||||
|
|
||||||
|
<!-- Inventory List -->
|
||||||
|
<InventoryList
|
||||||
|
ref="inventoryListRef"
|
||||||
|
:refresh="refreshKey"
|
||||||
|
:tag-filters="selectedTagFilters"
|
||||||
|
@add-item="showAddForm = true"
|
||||||
|
@edit-item="editingItem = $event"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script setup lang="ts">
|
||||||
|
definePageMeta({
|
||||||
|
layout: 'default'
|
||||||
|
})
|
||||||
|
|
||||||
|
const route = useRoute()
|
||||||
|
const router = useRouter()
|
||||||
|
|
||||||
|
const showAddForm = ref(false)
|
||||||
|
const showFilters = ref(false)
|
||||||
|
const editingItem = ref<any>(null)
|
||||||
|
const refreshKey = ref(0)
|
||||||
|
const inventoryListRef = ref()
|
||||||
|
const prefilledData = ref<any>(null)
|
||||||
|
const selectedTagFilters = ref<string[]>([])
|
||||||
|
|
||||||
|
// Handle scan-to-add flow (Issue #25)
|
||||||
|
onMounted(() => {
|
||||||
|
if (route.query.action === 'add') {
|
||||||
|
// Pre-fill data from query params (from scan)
|
||||||
|
prefilledData.value = {
|
||||||
|
barcode: route.query.barcode as string || undefined,
|
||||||
|
name: route.query.name as string || undefined,
|
||||||
|
brand: route.query.brand as string || undefined,
|
||||||
|
image_url: route.query.image_url as string || undefined,
|
||||||
|
quantity: route.query.quantity as string || undefined,
|
||||||
|
}
|
||||||
|
|
||||||
|
showAddForm.value = true
|
||||||
|
|
||||||
|
// Clean up URL
|
||||||
|
router.replace({ query: {} })
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
const handleCloseAddForm = () => {
|
||||||
|
showAddForm.value = false
|
||||||
|
prefilledData.value = null
|
||||||
|
}
|
||||||
|
|
||||||
|
const handleItemAdded = (item: any) => {
|
||||||
|
showAddForm.value = false
|
||||||
|
prefilledData.value = null
|
||||||
|
// Reload the inventory list
|
||||||
|
inventoryListRef.value?.reload()
|
||||||
|
}
|
||||||
|
|
||||||
|
const handleItemUpdated = (item: any) => {
|
||||||
|
editingItem.value = null
|
||||||
|
inventoryListRef.value?.reload()
|
||||||
|
}
|
||||||
|
</script>
|
||||||
112
app/pages/scan.vue
Normal file
@@ -0,0 +1,112 @@
|
|||||||
|
<template>
|
||||||
|
<div>
|
||||||
|
<h1 class="text-3xl font-bold text-gray-900 mb-6">Scan Item</h1>
|
||||||
|
|
||||||
|
<UCard v-if="!scannedBarcode" class="mb-6">
|
||||||
|
<ScanBarcodeScanner
|
||||||
|
@barcode-detected="handleBarcodeDetected"
|
||||||
|
@manual-entry="showManualEntry = true"
|
||||||
|
/>
|
||||||
|
</UCard>
|
||||||
|
|
||||||
|
<!-- Product Lookup Result -->
|
||||||
|
<UCard v-if="productData" class="mb-6">
|
||||||
|
<div class="space-y-4">
|
||||||
|
<div class="flex items-start gap-4">
|
||||||
|
<img
|
||||||
|
v-if="productData.image_url"
|
||||||
|
:src="productData.image_url"
|
||||||
|
:alt="productData.name"
|
||||||
|
class="w-24 h-24 object-cover rounded"
|
||||||
|
/>
|
||||||
|
<div class="flex-1">
|
||||||
|
<h3 class="text-xl font-bold mb-1">{{ productData.name }}</h3>
|
||||||
|
<p v-if="productData.brand" class="text-gray-600">{{ productData.brand }}</p>
|
||||||
|
<p class="text-sm text-gray-500 mt-2">Barcode: {{ scannedBarcode }}</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<UAlert
|
||||||
|
v-if="lookupError"
|
||||||
|
color="yellow"
|
||||||
|
icon="i-heroicons-exclamation-triangle"
|
||||||
|
title="Product not found"
|
||||||
|
:description="lookupError"
|
||||||
|
/>
|
||||||
|
|
||||||
|
<div class="flex gap-2">
|
||||||
|
<UButton
|
||||||
|
color="primary"
|
||||||
|
size="lg"
|
||||||
|
icon="i-heroicons-plus"
|
||||||
|
class="flex-1"
|
||||||
|
@click="addToInventory"
|
||||||
|
>
|
||||||
|
Add to Inventory
|
||||||
|
</UButton>
|
||||||
|
<UButton
|
||||||
|
color="gray"
|
||||||
|
size="lg"
|
||||||
|
@click="resetScanner"
|
||||||
|
>
|
||||||
|
Scan Again
|
||||||
|
</UButton>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</UCard>
|
||||||
|
|
||||||
|
<!-- Loading State -->
|
||||||
|
<UCard v-if="isLookingUp">
|
||||||
|
<div class="text-center py-8">
|
||||||
|
<div class="inline-block animate-spin rounded-full h-12 w-12 border-b-2 border-primary-500 mb-4"></div>
|
||||||
|
<p class="text-gray-600">Looking up product...</p>
|
||||||
|
</div>
|
||||||
|
</UCard>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script setup lang="ts">
|
||||||
|
definePageMeta({
|
||||||
|
layout: 'default'
|
||||||
|
})
|
||||||
|
|
||||||
|
const scannedBarcode = ref<string | null>(null)
|
||||||
|
const productData = ref<any>(null)
|
||||||
|
const showManualEntry = ref(false)
|
||||||
|
|
||||||
|
// Use product lookup composable
|
||||||
|
const { lookupProduct, isLoading: isLookingUp, error: lookupError } = useProductLookup()
|
||||||
|
|
||||||
|
const handleBarcodeDetected = async (barcode: string) => {
|
||||||
|
scannedBarcode.value = barcode
|
||||||
|
|
||||||
|
// Fetch product data from Edge Function
|
||||||
|
const data = await lookupProduct(barcode)
|
||||||
|
|
||||||
|
if (data) {
|
||||||
|
productData.value = data
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const addToInventory = () => {
|
||||||
|
// Navigate to home page with add form open and pre-filled
|
||||||
|
navigateTo({
|
||||||
|
path: '/',
|
||||||
|
query: {
|
||||||
|
action: 'add',
|
||||||
|
barcode: scannedBarcode.value,
|
||||||
|
name: productData.value?.name || undefined,
|
||||||
|
brand: productData.value?.brand || undefined,
|
||||||
|
image_url: productData.value?.image_url || undefined,
|
||||||
|
quantity: productData.value?.quantity || undefined
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
const resetScanner = () => {
|
||||||
|
scannedBarcode.value = null
|
||||||
|
productData.value = null
|
||||||
|
lookupError.value = null
|
||||||
|
isLookingUp.value = false
|
||||||
|
}
|
||||||
|
</script>
|
||||||
66
app/pages/settings.vue
Normal file
@@ -0,0 +1,66 @@
|
|||||||
|
<template>
|
||||||
|
<div>
|
||||||
|
<h1 class="text-3xl font-bold text-gray-900 mb-6">Settings</h1>
|
||||||
|
|
||||||
|
<UTabs :items="tabs" v-model="activeTab">
|
||||||
|
<template #account>
|
||||||
|
<UCard class="mt-4">
|
||||||
|
<div class="space-y-4">
|
||||||
|
<h3 class="text-lg font-semibold">Account Settings</h3>
|
||||||
|
<p class="text-gray-600">Account management will be implemented in future updates.</p>
|
||||||
|
</div>
|
||||||
|
</UCard>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<template #tags>
|
||||||
|
<div class="mt-4">
|
||||||
|
<TagsTagManager />
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<template #about>
|
||||||
|
<UCard class="mt-4">
|
||||||
|
<div class="space-y-4">
|
||||||
|
<h3 class="text-lg font-semibold">About Pantry</h3>
|
||||||
|
<p class="text-gray-600">Version 0.1.0 (MVP)</p>
|
||||||
|
<p class="text-gray-600">Self-hosted pantry management app with barcode scanning.</p>
|
||||||
|
<UButton
|
||||||
|
to="https://github.com/pantry-app/pantry"
|
||||||
|
target="_blank"
|
||||||
|
color="gray"
|
||||||
|
variant="soft"
|
||||||
|
>
|
||||||
|
View on GitHub
|
||||||
|
</UButton>
|
||||||
|
</div>
|
||||||
|
</UCard>
|
||||||
|
</template>
|
||||||
|
</UTabs>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script setup lang="ts">
|
||||||
|
definePageMeta({
|
||||||
|
layout: 'default'
|
||||||
|
})
|
||||||
|
|
||||||
|
const activeTab = ref('tags')
|
||||||
|
|
||||||
|
const tabs = [
|
||||||
|
{
|
||||||
|
key: 'tags',
|
||||||
|
label: 'Tags',
|
||||||
|
icon: 'i-heroicons-tag'
|
||||||
|
},
|
||||||
|
{
|
||||||
|
key: 'account',
|
||||||
|
label: 'Account',
|
||||||
|
icon: 'i-heroicons-user'
|
||||||
|
},
|
||||||
|
{
|
||||||
|
key: 'about',
|
||||||
|
label: 'About',
|
||||||
|
icon: 'i-heroicons-information-circle'
|
||||||
|
}
|
||||||
|
]
|
||||||
|
</script>
|
||||||
BIN
app/public/apple-touch-icon.png
Normal file
|
After Width: | Height: | Size: 8.0 KiB |
BIN
app/public/favicon.ico
Normal file
|
After Width: | Height: | Size: 1.7 KiB |
BIN
app/public/icon-192x192-maskable.png
Normal file
|
After Width: | Height: | Size: 7.0 KiB |
BIN
app/public/icon-192x192.png
Normal file
|
After Width: | Height: | Size: 8.6 KiB |
BIN
app/public/icon-512x512-maskable.png
Normal file
|
After Width: | Height: | Size: 21 KiB |
BIN
app/public/icon-512x512.png
Normal file
|
After Width: | Height: | Size: 24 KiB |
46
app/public/icon.svg
Normal file
@@ -0,0 +1,46 @@
|
|||||||
|
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 512 512" fill="none">
|
||||||
|
<!-- Background circle with emerald gradient -->
|
||||||
|
<defs>
|
||||||
|
<linearGradient id="grad" x1="0%" y1="0%" x2="100%" y2="100%">
|
||||||
|
<stop offset="0%" style="stop-color:#10b981;stop-opacity:1" />
|
||||||
|
<stop offset="100%" style="stop-color:#059669;stop-opacity:1" />
|
||||||
|
</linearGradient>
|
||||||
|
</defs>
|
||||||
|
|
||||||
|
<!-- Background -->
|
||||||
|
<circle cx="256" cy="256" r="256" fill="url(#grad)"/>
|
||||||
|
|
||||||
|
<!-- Pantry shelves icon (simplified cabinet with items) -->
|
||||||
|
<g transform="translate(106, 96)">
|
||||||
|
<!-- Cabinet outline -->
|
||||||
|
<rect x="0" y="0" width="300" height="320" rx="12" fill="none" stroke="white" stroke-width="8"/>
|
||||||
|
|
||||||
|
<!-- Top shelf -->
|
||||||
|
<line x1="0" y1="80" x2="300" y2="80" stroke="white" stroke-width="6"/>
|
||||||
|
|
||||||
|
<!-- Middle shelf -->
|
||||||
|
<line x1="0" y1="160" x2="300" y2="160" stroke="white" stroke-width="6"/>
|
||||||
|
|
||||||
|
<!-- Bottom shelf -->
|
||||||
|
<line x1="0" y1="240" x2="300" y2="240" stroke="white" stroke-width="6"/>
|
||||||
|
|
||||||
|
<!-- Top shelf items - jars -->
|
||||||
|
<circle cx="60" cy="40" r="25" fill="white" opacity="0.9"/>
|
||||||
|
<circle cx="150" cy="40" r="25" fill="white" opacity="0.9"/>
|
||||||
|
<circle cx="240" cy="40" r="25" fill="white" opacity="0.9"/>
|
||||||
|
|
||||||
|
<!-- Middle shelf items - boxes -->
|
||||||
|
<rect x="35" y="105" width="50" height="40" rx="4" fill="white" opacity="0.9"/>
|
||||||
|
<rect x="125" y="105" width="50" height="40" rx="4" fill="white" opacity="0.9"/>
|
||||||
|
<rect x="215" y="105" width="50" height="40" rx="4" fill="white" opacity="0.9"/>
|
||||||
|
|
||||||
|
<!-- Bottom shelf items - cans -->
|
||||||
|
<rect x="35" y="185" width="50" height="45" rx="6" fill="white" opacity="0.9"/>
|
||||||
|
<rect x="125" y="185" width="50" height="45" rx="6" fill="white" opacity="0.9"/>
|
||||||
|
<rect x="215" y="185" width="50" height="45" rx="6" fill="white" opacity="0.9"/>
|
||||||
|
|
||||||
|
<!-- Very bottom items - larger containers -->
|
||||||
|
<rect x="45" y="260" width="70" height="50" rx="6" fill="white" opacity="0.9"/>
|
||||||
|
<rect x="185" y="260" width="70" height="50" rx="6" fill="white" opacity="0.9"/>
|
||||||
|
</g>
|
||||||
|
</svg>
|
||||||
|
After Width: | Height: | Size: 2.0 KiB |
2
app/public/robots.txt
Normal file
@@ -0,0 +1,2 @@
|
|||||||
|
User-Agent: *
|
||||||
|
Disallow:
|
||||||
BIN
app/public/screenshot-desktop.png
Normal file
|
After Width: | Height: | Size: 71 KiB |
BIN
app/public/screenshot-mobile.png
Normal file
|
After Width: | Height: | Size: 21 KiB |
70
app/scripts/generate-icons.js
Normal file
@@ -0,0 +1,70 @@
|
|||||||
|
#!/usr/bin/env node
|
||||||
|
import { readFile, writeFile } from 'fs/promises';
|
||||||
|
import { join, dirname } from 'path';
|
||||||
|
import { fileURLToPath } from 'url';
|
||||||
|
import sharp from 'sharp';
|
||||||
|
|
||||||
|
const __dirname = dirname(fileURLToPath(import.meta.url));
|
||||||
|
const publicDir = join(__dirname, '..', 'public');
|
||||||
|
const svgPath = join(publicDir, 'icon.svg');
|
||||||
|
|
||||||
|
const sizes = [
|
||||||
|
{ size: 192, name: 'icon-192x192.png', maskable: false },
|
||||||
|
{ size: 512, name: 'icon-512x512.png', maskable: false },
|
||||||
|
{ size: 192, name: 'icon-192x192-maskable.png', maskable: true },
|
||||||
|
{ size: 512, name: 'icon-512x512-maskable.png', maskable: true },
|
||||||
|
];
|
||||||
|
|
||||||
|
async function generateIcons() {
|
||||||
|
console.log('Reading SVG icon...');
|
||||||
|
const svgBuffer = await readFile(svgPath);
|
||||||
|
|
||||||
|
for (const { size, name, maskable } of sizes) {
|
||||||
|
console.log(`Generating ${name}...`);
|
||||||
|
|
||||||
|
let buffer;
|
||||||
|
if (maskable) {
|
||||||
|
// Maskable icons need safe zone padding (80% of icon in center)
|
||||||
|
// Create a transparent canvas with padding
|
||||||
|
const paddedSize = size;
|
||||||
|
const iconSize = Math.floor(size * 0.8);
|
||||||
|
const offset = Math.floor((paddedSize - iconSize) / 2);
|
||||||
|
|
||||||
|
// Resize SVG to icon size
|
||||||
|
const iconBuffer = await sharp(svgBuffer)
|
||||||
|
.resize(iconSize, iconSize)
|
||||||
|
.png()
|
||||||
|
.toBuffer();
|
||||||
|
|
||||||
|
// Create transparent background and composite
|
||||||
|
buffer = await sharp({
|
||||||
|
create: {
|
||||||
|
width: paddedSize,
|
||||||
|
height: paddedSize,
|
||||||
|
channels: 4,
|
||||||
|
background: { r: 0, g: 0, b: 0, alpha: 0 }
|
||||||
|
}
|
||||||
|
})
|
||||||
|
.composite([{
|
||||||
|
input: iconBuffer,
|
||||||
|
top: offset,
|
||||||
|
left: offset
|
||||||
|
}])
|
||||||
|
.png()
|
||||||
|
.toBuffer();
|
||||||
|
} else {
|
||||||
|
// Regular icon - full size
|
||||||
|
buffer = await sharp(svgBuffer)
|
||||||
|
.resize(size, size)
|
||||||
|
.png()
|
||||||
|
.toBuffer();
|
||||||
|
}
|
||||||
|
|
||||||
|
await writeFile(join(publicDir, name), buffer);
|
||||||
|
console.log(`✓ ${name}`);
|
||||||
|
}
|
||||||
|
|
||||||
|
console.log('\n✅ All icons generated successfully!');
|
||||||
|
}
|
||||||
|
|
||||||
|
generateIcons().catch(console.error);
|
||||||
124
app/scripts/generate-screenshots.js
Normal file
@@ -0,0 +1,124 @@
|
|||||||
|
#!/usr/bin/env node
|
||||||
|
import { writeFile } from 'fs/promises';
|
||||||
|
import { join, dirname } from 'path';
|
||||||
|
import { fileURLToPath } from 'url';
|
||||||
|
import sharp from 'sharp';
|
||||||
|
|
||||||
|
const __dirname = dirname(fileURLToPath(import.meta.url));
|
||||||
|
const publicDir = join(__dirname, '..', 'public');
|
||||||
|
|
||||||
|
async function generateScreenshots() {
|
||||||
|
// Mobile screenshot (390x844 - iPhone 12/13/14 size)
|
||||||
|
console.log('Generating mobile screenshot placeholder...');
|
||||||
|
const mobileBuffer = await sharp({
|
||||||
|
create: {
|
||||||
|
width: 390,
|
||||||
|
height: 844,
|
||||||
|
channels: 4,
|
||||||
|
background: { r: 249, g: 250, b: 251, alpha: 1 } // Tailwind gray-50
|
||||||
|
}
|
||||||
|
})
|
||||||
|
.composite([
|
||||||
|
{
|
||||||
|
input: Buffer.from(`
|
||||||
|
<svg width="390" height="844">
|
||||||
|
<!-- Header -->
|
||||||
|
<rect width="390" height="64" fill="#10b981"/>
|
||||||
|
<text x="195" y="42" font-family="Arial" font-size="20" fill="white" text-anchor="middle" font-weight="bold">Pantry</text>
|
||||||
|
|
||||||
|
<!-- Content area -->
|
||||||
|
<text x="24" y="104" font-family="Arial" font-size="24" fill="#111827" font-weight="bold">My Pantry</text>
|
||||||
|
|
||||||
|
<!-- Item cards -->
|
||||||
|
<rect x="16" y="130" width="358" height="80" rx="8" fill="white" stroke="#e5e7eb" stroke-width="1"/>
|
||||||
|
<text x="32" y="160" font-family="Arial" font-size="16" fill="#111827" font-weight="600">Milk</text>
|
||||||
|
<text x="32" y="185" font-family="Arial" font-size="14" fill="#6b7280">Fridge • 1L</text>
|
||||||
|
|
||||||
|
<rect x="16" y="226" width="358" height="80" rx="8" fill="white" stroke="#e5e7eb" stroke-width="1"/>
|
||||||
|
<text x="32" y="256" font-family="Arial" font-size="16" fill="#111827" font-weight="600">Pasta</text>
|
||||||
|
<text x="32" y="281" font-family="Arial" font-size="14" fill="#6b7280">Pantry • 500g</text>
|
||||||
|
|
||||||
|
<rect x="16" y="322" width="358" height="80" rx="8" fill="white" stroke="#e5e7eb" stroke-width="1"/>
|
||||||
|
<text x="32" y="352" font-family="Arial" font-size="16" fill="#111827" font-weight="600">Tomato Sauce</text>
|
||||||
|
<text x="32" y="377" font-family="Arial" font-size="14" fill="#6b7280">Pantry • 400ml</text>
|
||||||
|
|
||||||
|
<!-- Bottom navigation -->
|
||||||
|
<rect y="780" width="390" height="64" fill="white" stroke="#e5e7eb" stroke-width="1"/>
|
||||||
|
<text x="78" y="820" font-family="Arial" font-size="12" fill="#6b7280" text-anchor="middle">Home</text>
|
||||||
|
<text x="195" y="820" font-family="Arial" font-size="12" fill="#10b981" text-anchor="middle">Scan</text>
|
||||||
|
<text x="312" y="820" font-family="Arial" font-size="12" fill="#6b7280" text-anchor="middle">Settings</text>
|
||||||
|
</svg>
|
||||||
|
`),
|
||||||
|
top: 0,
|
||||||
|
left: 0
|
||||||
|
}
|
||||||
|
])
|
||||||
|
.png()
|
||||||
|
.toBuffer();
|
||||||
|
|
||||||
|
await writeFile(join(publicDir, 'screenshot-mobile.png'), mobileBuffer);
|
||||||
|
console.log('✓ screenshot-mobile.png');
|
||||||
|
|
||||||
|
// Desktop screenshot (1920x1080)
|
||||||
|
console.log('Generating desktop screenshot placeholder...');
|
||||||
|
const desktopBuffer = await sharp({
|
||||||
|
create: {
|
||||||
|
width: 1920,
|
||||||
|
height: 1080,
|
||||||
|
channels: 4,
|
||||||
|
background: { r: 249, g: 250, b: 251, alpha: 1 } // Tailwind gray-50
|
||||||
|
}
|
||||||
|
})
|
||||||
|
.composite([
|
||||||
|
{
|
||||||
|
input: Buffer.from(`
|
||||||
|
<svg width="1920" height="1080">
|
||||||
|
<!-- Header -->
|
||||||
|
<rect width="1920" height="80" fill="#10b981"/>
|
||||||
|
<text x="960" y="50" font-family="Arial" font-size="32" fill="white" text-anchor="middle" font-weight="bold">Pantry - Smart Inventory Manager</text>
|
||||||
|
|
||||||
|
<!-- Sidebar -->
|
||||||
|
<rect x="0" y="80" width="280" height="1000" fill="white" stroke="#e5e7eb" stroke-width="1"/>
|
||||||
|
<text x="32" y="130" font-family="Arial" font-size="18" fill="#10b981" font-weight="600">Dashboard</text>
|
||||||
|
<text x="32" y="180" font-family="Arial" font-size="18" fill="#6b7280">Scan Item</text>
|
||||||
|
<text x="32" y="230" font-family="Arial" font-size="18" fill="#6b7280">Settings</text>
|
||||||
|
|
||||||
|
<!-- Main content -->
|
||||||
|
<text x="340" y="150" font-family="Arial" font-size="36" fill="#111827" font-weight="bold">My Pantry Items</text>
|
||||||
|
|
||||||
|
<!-- Grid of items -->
|
||||||
|
<rect x="340" y="200" width="480" height="180" rx="12" fill="white" stroke="#e5e7eb" stroke-width="2"/>
|
||||||
|
<text x="370" y="250" font-family="Arial" font-size="24" fill="#111827" font-weight="600">Milk</text>
|
||||||
|
<text x="370" y="290" font-family="Arial" font-size="18" fill="#6b7280">Fridge • 1L • Expires in 5 days</text>
|
||||||
|
|
||||||
|
<rect x="860" y="200" width="480" height="180" rx="12" fill="white" stroke="#e5e7eb" stroke-width="2"/>
|
||||||
|
<text x="890" y="250" font-family="Arial" font-size="24" fill="#111827" font-weight="600">Pasta</text>
|
||||||
|
<text x="890" y="290" font-family="Arial" font-size="18" fill="#6b7280">Pantry • 500g</text>
|
||||||
|
|
||||||
|
<rect x="1380" y="200" width="480" height="180" rx="12" fill="white" stroke="#e5e7eb" stroke-width="2"/>
|
||||||
|
<text x="1410" y="250" font-family="Arial" font-size="24" fill="#111827" font-weight="600">Tomato Sauce</text>
|
||||||
|
<text x="1410" y="290" font-family="Arial" font-size="18" fill="#6b7280">Pantry • 400ml</text>
|
||||||
|
|
||||||
|
<rect x="340" y="420" width="480" height="180" rx="12" fill="white" stroke="#e5e7eb" stroke-width="2"/>
|
||||||
|
<text x="370" y="470" font-family="Arial" font-size="24" fill="#111827" font-weight="600">Rice</text>
|
||||||
|
<text x="370" y="510" font-family="Arial" font-size="18" fill="#6b7280">Pantry • 1kg</text>
|
||||||
|
|
||||||
|
<rect x="860" y="420" width="480" height="180" rx="12" fill="white" stroke="#e5e7eb" stroke-width="2"/>
|
||||||
|
<text x="890" y="470" font-family="Arial" font-size="24" fill="#111827" font-weight="600">Olive Oil</text>
|
||||||
|
<text x="890" y="510" font-family="Arial" font-size="18" fill="#6b7280">Pantry • 750ml</text>
|
||||||
|
</svg>
|
||||||
|
`),
|
||||||
|
top: 0,
|
||||||
|
left: 0
|
||||||
|
}
|
||||||
|
])
|
||||||
|
.png()
|
||||||
|
.toBuffer();
|
||||||
|
|
||||||
|
await writeFile(join(publicDir, 'screenshot-desktop.png'), desktopBuffer);
|
||||||
|
console.log('✓ screenshot-desktop.png');
|
||||||
|
|
||||||
|
console.log('\n✅ All screenshots generated successfully!');
|
||||||
|
}
|
||||||
|
|
||||||
|
generateScreenshots().catch(console.error);
|
||||||
18
app/tsconfig.json
Normal file
@@ -0,0 +1,18 @@
|
|||||||
|
{
|
||||||
|
// https://nuxt.com/docs/guide/concepts/typescript
|
||||||
|
"files": [],
|
||||||
|
"references": [
|
||||||
|
{
|
||||||
|
"path": "./.nuxt/tsconfig.app.json"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"path": "./.nuxt/tsconfig.server.json"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"path": "./.nuxt/tsconfig.shared.json"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"path": "./.nuxt/tsconfig.node.json"
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
184
app/types/database.types.ts
Normal file
@@ -0,0 +1,184 @@
|
|||||||
|
/**
|
||||||
|
* Database type definitions
|
||||||
|
*
|
||||||
|
* TODO: Generate these from Supabase schema using:
|
||||||
|
* supabase gen types typescript --project-id <project-id> > types/database.types.ts
|
||||||
|
*
|
||||||
|
* For now, using a placeholder structure that matches our schema
|
||||||
|
*/
|
||||||
|
|
||||||
|
export type Json =
|
||||||
|
| string
|
||||||
|
| number
|
||||||
|
| boolean
|
||||||
|
| null
|
||||||
|
| { [key: string]: Json | undefined }
|
||||||
|
| Json[]
|
||||||
|
|
||||||
|
export interface Database {
|
||||||
|
public: {
|
||||||
|
Tables: {
|
||||||
|
inventory_items: {
|
||||||
|
Row: {
|
||||||
|
id: string
|
||||||
|
product_id: string | null
|
||||||
|
name: string
|
||||||
|
quantity: number
|
||||||
|
unit_id: string
|
||||||
|
expiry_date: string | null
|
||||||
|
notes: string | null
|
||||||
|
added_by: string
|
||||||
|
created_at: string
|
||||||
|
updated_at: string
|
||||||
|
}
|
||||||
|
Insert: {
|
||||||
|
id?: string
|
||||||
|
product_id?: string | null
|
||||||
|
name: string
|
||||||
|
quantity: number
|
||||||
|
unit_id: string
|
||||||
|
expiry_date?: string | null
|
||||||
|
notes?: string | null
|
||||||
|
added_by: string
|
||||||
|
created_at?: string
|
||||||
|
updated_at?: string
|
||||||
|
}
|
||||||
|
Update: {
|
||||||
|
id?: string
|
||||||
|
product_id?: string | null
|
||||||
|
name?: string
|
||||||
|
quantity?: number
|
||||||
|
unit_id?: string
|
||||||
|
expiry_date?: string | null
|
||||||
|
notes?: string | null
|
||||||
|
added_by?: string
|
||||||
|
created_at?: string
|
||||||
|
updated_at?: string
|
||||||
|
}
|
||||||
|
}
|
||||||
|
products: {
|
||||||
|
Row: {
|
||||||
|
id: string
|
||||||
|
barcode: string
|
||||||
|
name: string
|
||||||
|
brand: string | null
|
||||||
|
image_url: string | null
|
||||||
|
default_unit_id: string | null
|
||||||
|
cached_at: string
|
||||||
|
created_at: string
|
||||||
|
}
|
||||||
|
Insert: {
|
||||||
|
id?: string
|
||||||
|
barcode: string
|
||||||
|
name: string
|
||||||
|
brand?: string | null
|
||||||
|
image_url?: string | null
|
||||||
|
default_unit_id?: string | null
|
||||||
|
cached_at?: string
|
||||||
|
created_at?: string
|
||||||
|
}
|
||||||
|
Update: {
|
||||||
|
id?: string
|
||||||
|
barcode?: string
|
||||||
|
name?: string
|
||||||
|
brand?: string | null
|
||||||
|
image_url?: string | null
|
||||||
|
default_unit_id?: string | null
|
||||||
|
cached_at?: string
|
||||||
|
created_at?: string
|
||||||
|
}
|
||||||
|
}
|
||||||
|
tags: {
|
||||||
|
Row: {
|
||||||
|
id: string
|
||||||
|
name: string
|
||||||
|
category: 'position' | 'type' | 'custom'
|
||||||
|
icon: string | null
|
||||||
|
color: string | null
|
||||||
|
created_by: string | null
|
||||||
|
created_at: string
|
||||||
|
}
|
||||||
|
Insert: {
|
||||||
|
id?: string
|
||||||
|
name: string
|
||||||
|
category?: 'position' | 'type' | 'custom'
|
||||||
|
icon?: string | null
|
||||||
|
color?: string | null
|
||||||
|
created_by?: string | null
|
||||||
|
created_at?: string
|
||||||
|
}
|
||||||
|
Update: {
|
||||||
|
id?: string
|
||||||
|
name?: string
|
||||||
|
category?: 'position' | 'type' | 'custom'
|
||||||
|
icon?: string | null
|
||||||
|
color?: string | null
|
||||||
|
created_by?: string | null
|
||||||
|
created_at?: string
|
||||||
|
}
|
||||||
|
}
|
||||||
|
units: {
|
||||||
|
Row: {
|
||||||
|
id: string
|
||||||
|
name: string
|
||||||
|
abbreviation: string
|
||||||
|
unit_type: 'weight' | 'volume' | 'count' | 'custom'
|
||||||
|
base_unit_id: string | null
|
||||||
|
conversion_factor: number | null
|
||||||
|
is_default: boolean
|
||||||
|
created_by: string | null
|
||||||
|
created_at: string
|
||||||
|
}
|
||||||
|
Insert: {
|
||||||
|
id?: string
|
||||||
|
name: string
|
||||||
|
abbreviation: string
|
||||||
|
unit_type?: 'weight' | 'volume' | 'count' | 'custom'
|
||||||
|
base_unit_id?: string | null
|
||||||
|
conversion_factor?: number | null
|
||||||
|
is_default?: boolean
|
||||||
|
created_by?: string | null
|
||||||
|
created_at?: string
|
||||||
|
}
|
||||||
|
Update: {
|
||||||
|
id?: string
|
||||||
|
name?: string
|
||||||
|
abbreviation?: string
|
||||||
|
unit_type?: 'weight' | 'volume' | 'count' | 'custom'
|
||||||
|
base_unit_id?: string | null
|
||||||
|
conversion_factor?: number | null
|
||||||
|
is_default?: boolean
|
||||||
|
created_by?: string | null
|
||||||
|
created_at?: string
|
||||||
|
}
|
||||||
|
}
|
||||||
|
item_tags: {
|
||||||
|
Row: {
|
||||||
|
item_id: string
|
||||||
|
tag_id: string
|
||||||
|
created_at: string
|
||||||
|
}
|
||||||
|
Insert: {
|
||||||
|
item_id: string
|
||||||
|
tag_id: string
|
||||||
|
created_at?: string
|
||||||
|
}
|
||||||
|
Update: {
|
||||||
|
item_id?: string
|
||||||
|
tag_id?: string
|
||||||
|
created_at?: string
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Views: {
|
||||||
|
[_ in never]: never
|
||||||
|
}
|
||||||
|
Functions: {
|
||||||
|
[_ in never]: never
|
||||||
|
}
|
||||||
|
Enums: {
|
||||||
|
tag_category: 'position' | 'type' | 'custom'
|
||||||
|
unit_type: 'weight' | 'volume' | 'count' | 'custom'
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
42
dev.sh
Executable file
@@ -0,0 +1,42 @@
|
|||||||
|
#!/bin/bash
|
||||||
|
set -e
|
||||||
|
|
||||||
|
echo "🚀 Pantry - Starting Local Development Environment"
|
||||||
|
echo ""
|
||||||
|
|
||||||
|
# Check prerequisites
|
||||||
|
command -v docker >/dev/null 2>&1 || { echo "❌ Docker not found. Install: https://docs.docker.com/get-docker/"; exit 1; }
|
||||||
|
command -v docker-compose >/dev/null 2>&1 || { echo "❌ Docker Compose not found. Install: https://docs.docker.com/compose/install/"; exit 1; }
|
||||||
|
command -v bun >/dev/null 2>&1 || { echo "❌ Bun not found. Install: curl -fsSL https://bun.sh/install | bash"; exit 1; }
|
||||||
|
|
||||||
|
echo "✅ Prerequisites OK"
|
||||||
|
echo ""
|
||||||
|
|
||||||
|
# Step 1: Start Supabase
|
||||||
|
echo "📦 Starting Supabase services..."
|
||||||
|
docker-compose up -d
|
||||||
|
|
||||||
|
echo "⏳ Waiting for services to initialize (15s)..."
|
||||||
|
sleep 15
|
||||||
|
|
||||||
|
echo "✅ Supabase started:"
|
||||||
|
echo " - API: http://localhost:54321"
|
||||||
|
echo " - Studio: http://localhost:54323"
|
||||||
|
echo ""
|
||||||
|
|
||||||
|
# Step 2: Install frontend dependencies
|
||||||
|
echo "📦 Installing frontend dependencies..."
|
||||||
|
cd app
|
||||||
|
bun install
|
||||||
|
|
||||||
|
echo "✅ Dependencies installed"
|
||||||
|
echo ""
|
||||||
|
|
||||||
|
# Step 3: Start dev server
|
||||||
|
echo "🚀 Starting Nuxt dev server..."
|
||||||
|
echo " App will be available at: http://localhost:3000"
|
||||||
|
echo ""
|
||||||
|
echo "Press Ctrl+C to stop"
|
||||||
|
echo ""
|
||||||
|
|
||||||
|
bun run dev
|
||||||
127
docker-compose.yml
Normal file
@@ -0,0 +1,127 @@
|
|||||||
|
version: '3.8'
|
||||||
|
|
||||||
|
services:
|
||||||
|
# PostgreSQL Database
|
||||||
|
db:
|
||||||
|
image: supabase/postgres:15.1.0.147
|
||||||
|
restart: unless-stopped
|
||||||
|
ports:
|
||||||
|
- "5432:5432"
|
||||||
|
environment:
|
||||||
|
POSTGRES_PASSWORD: ${POSTGRES_PASSWORD:-postgres}
|
||||||
|
POSTGRES_DB: postgres
|
||||||
|
volumes:
|
||||||
|
- db-data:/var/lib/postgresql/data
|
||||||
|
- ./supabase/migrations:/docker-entrypoint-initdb.d:ro
|
||||||
|
|
||||||
|
# Supabase Studio (Admin UI)
|
||||||
|
studio:
|
||||||
|
image: supabase/studio:20231123-64a766a
|
||||||
|
restart: unless-stopped
|
||||||
|
ports:
|
||||||
|
- "54323:3000"
|
||||||
|
environment:
|
||||||
|
SUPABASE_URL: http://kong:8000
|
||||||
|
SUPABASE_PUBLIC_URL: http://localhost:54321
|
||||||
|
SUPABASE_ANON_KEY: ${ANON_KEY}
|
||||||
|
SUPABASE_SERVICE_KEY: ${SERVICE_ROLE_KEY}
|
||||||
|
|
||||||
|
# Kong API Gateway
|
||||||
|
kong:
|
||||||
|
image: kong:2.8.1
|
||||||
|
restart: unless-stopped
|
||||||
|
ports:
|
||||||
|
- "54321:8000"
|
||||||
|
- "54320:8443"
|
||||||
|
environment:
|
||||||
|
KONG_DATABASE: "off"
|
||||||
|
KONG_DECLARATIVE_CONFIG: /var/lib/kong/kong.yml
|
||||||
|
KONG_DNS_ORDER: LAST,A,CNAME
|
||||||
|
KONG_PLUGINS: request-transformer,cors,key-auth,acl
|
||||||
|
volumes:
|
||||||
|
- ./docker/kong.yml:/var/lib/kong/kong.yml:ro
|
||||||
|
|
||||||
|
# GoTrue (Auth)
|
||||||
|
auth:
|
||||||
|
image: supabase/gotrue:v2.99.0
|
||||||
|
restart: unless-stopped
|
||||||
|
depends_on:
|
||||||
|
- db
|
||||||
|
environment:
|
||||||
|
GOTRUE_API_HOST: 0.0.0.0
|
||||||
|
GOTRUE_API_PORT: 9999
|
||||||
|
API_EXTERNAL_URL: http://localhost:54321
|
||||||
|
GOTRUE_DB_DRIVER: postgres
|
||||||
|
GOTRUE_DB_DATABASE_URL: postgres://supabase_auth_admin:${POSTGRES_PASSWORD:-postgres}@db:5432/postgres
|
||||||
|
GOTRUE_SITE_URL: http://localhost:3000
|
||||||
|
GOTRUE_URI_ALLOW_LIST: "*"
|
||||||
|
GOTRUE_DISABLE_SIGNUP: false
|
||||||
|
GOTRUE_JWT_ADMIN_ROLES: service_role
|
||||||
|
GOTRUE_JWT_AUD: authenticated
|
||||||
|
GOTRUE_JWT_DEFAULT_GROUP_NAME: authenticated
|
||||||
|
GOTRUE_JWT_EXP: 3600
|
||||||
|
GOTRUE_JWT_SECRET: ${JWT_SECRET}
|
||||||
|
GOTRUE_EXTERNAL_EMAIL_ENABLED: true
|
||||||
|
GOTRUE_MAILER_AUTOCONFIRM: true
|
||||||
|
|
||||||
|
# PostgREST (Auto API)
|
||||||
|
rest:
|
||||||
|
image: postgrest/postgrest:v11.2.0
|
||||||
|
restart: unless-stopped
|
||||||
|
depends_on:
|
||||||
|
- db
|
||||||
|
environment:
|
||||||
|
PGRST_DB_URI: postgres://authenticator:${POSTGRES_PASSWORD:-postgres}@db:5432/postgres
|
||||||
|
PGRST_DB_SCHEMAS: public,storage
|
||||||
|
PGRST_DB_ANON_ROLE: anon
|
||||||
|
PGRST_JWT_SECRET: ${JWT_SECRET}
|
||||||
|
PGRST_DB_USE_LEGACY_GUCS: "false"
|
||||||
|
|
||||||
|
# Realtime
|
||||||
|
realtime:
|
||||||
|
image: supabase/realtime:v2.25.35
|
||||||
|
restart: unless-stopped
|
||||||
|
depends_on:
|
||||||
|
- db
|
||||||
|
environment:
|
||||||
|
PORT: 4000
|
||||||
|
DB_HOST: db
|
||||||
|
DB_PORT: 5432
|
||||||
|
DB_USER: supabase_admin
|
||||||
|
DB_PASSWORD: ${POSTGRES_PASSWORD:-postgres}
|
||||||
|
DB_NAME: postgres
|
||||||
|
DB_AFTER_CONNECT_QUERY: 'SET search_path TO _realtime'
|
||||||
|
DB_ENC_KEY: supabaserealtime
|
||||||
|
API_JWT_SECRET: ${JWT_SECRET}
|
||||||
|
FLY_ALLOC_ID: fly123
|
||||||
|
FLY_APP_NAME: realtime
|
||||||
|
SECRET_KEY_BASE: ${JWT_SECRET}
|
||||||
|
ERL_AFLAGS: -proto_dist inet_tcp
|
||||||
|
ENABLE_TAILSCALE: "false"
|
||||||
|
DNS_NODES: "''"
|
||||||
|
|
||||||
|
# Storage (S3-compatible)
|
||||||
|
storage:
|
||||||
|
image: supabase/storage-api:v0.40.4
|
||||||
|
restart: unless-stopped
|
||||||
|
depends_on:
|
||||||
|
- db
|
||||||
|
- rest
|
||||||
|
environment:
|
||||||
|
ANON_KEY: ${ANON_KEY}
|
||||||
|
SERVICE_KEY: ${SERVICE_ROLE_KEY}
|
||||||
|
POSTGREST_URL: http://rest:3000
|
||||||
|
PGRST_JWT_SECRET: ${JWT_SECRET}
|
||||||
|
DATABASE_URL: postgres://supabase_storage_admin:${POSTGRES_PASSWORD:-postgres}@db:5432/postgres
|
||||||
|
FILE_SIZE_LIMIT: 52428800
|
||||||
|
STORAGE_BACKEND: file
|
||||||
|
FILE_STORAGE_BACKEND_PATH: /var/lib/storage
|
||||||
|
TENANT_ID: stub
|
||||||
|
REGION: stub
|
||||||
|
GLOBAL_S3_BUCKET: stub
|
||||||
|
volumes:
|
||||||
|
- storage-data:/var/lib/storage
|
||||||
|
|
||||||
|
volumes:
|
||||||
|
db-data:
|
||||||
|
storage-data:
|
||||||
85
docker/kong.yml
Normal file
@@ -0,0 +1,85 @@
|
|||||||
|
_format_version: "2.1"
|
||||||
|
|
||||||
|
_transform: true
|
||||||
|
|
||||||
|
services:
|
||||||
|
- name: auth
|
||||||
|
url: http://auth:9999
|
||||||
|
routes:
|
||||||
|
- name: auth-v1
|
||||||
|
strip_path: true
|
||||||
|
paths:
|
||||||
|
- /auth/v1
|
||||||
|
plugins:
|
||||||
|
- name: cors
|
||||||
|
- name: key-auth
|
||||||
|
config:
|
||||||
|
hide_credentials: true
|
||||||
|
|
||||||
|
- name: rest
|
||||||
|
url: http://rest:3000
|
||||||
|
routes:
|
||||||
|
- name: rest-v1
|
||||||
|
strip_path: true
|
||||||
|
paths:
|
||||||
|
- /rest/v1
|
||||||
|
plugins:
|
||||||
|
- name: cors
|
||||||
|
- name: key-auth
|
||||||
|
config:
|
||||||
|
hide_credentials: true
|
||||||
|
|
||||||
|
- name: realtime
|
||||||
|
url: http://realtime:4000/socket
|
||||||
|
routes:
|
||||||
|
- name: realtime-v1
|
||||||
|
strip_path: true
|
||||||
|
paths:
|
||||||
|
- /realtime/v1
|
||||||
|
plugins:
|
||||||
|
- name: cors
|
||||||
|
- name: key-auth
|
||||||
|
config:
|
||||||
|
hide_credentials: true
|
||||||
|
|
||||||
|
- name: storage
|
||||||
|
url: http://storage:5000
|
||||||
|
routes:
|
||||||
|
- name: storage-v1
|
||||||
|
strip_path: true
|
||||||
|
paths:
|
||||||
|
- /storage/v1
|
||||||
|
plugins:
|
||||||
|
- name: cors
|
||||||
|
|
||||||
|
consumers:
|
||||||
|
- username: anon
|
||||||
|
keyauth_credentials:
|
||||||
|
- key: ${ANON_KEY}
|
||||||
|
- username: service_role
|
||||||
|
keyauth_credentials:
|
||||||
|
- key: ${SERVICE_ROLE_KEY}
|
||||||
|
|
||||||
|
plugins:
|
||||||
|
- name: cors
|
||||||
|
config:
|
||||||
|
origins:
|
||||||
|
- "*"
|
||||||
|
methods:
|
||||||
|
- GET
|
||||||
|
- POST
|
||||||
|
- PUT
|
||||||
|
- PATCH
|
||||||
|
- DELETE
|
||||||
|
- OPTIONS
|
||||||
|
headers:
|
||||||
|
- Accept
|
||||||
|
- Accept-Encoding
|
||||||
|
- Authorization
|
||||||
|
- Content-Type
|
||||||
|
- Origin
|
||||||
|
- X-Client-Info
|
||||||
|
exposed_headers:
|
||||||
|
- X-Total-Count
|
||||||
|
credentials: true
|
||||||
|
max_age: 3600
|
||||||
84
docs/README.md
Normal file
@@ -0,0 +1,84 @@
|
|||||||
|
# Pantry Documentation
|
||||||
|
|
||||||
|
Complete documentation for the Pantry household inventory management system.
|
||||||
|
|
||||||
|
## 📚 Documentation Structure
|
||||||
|
|
||||||
|
### 🚀 Getting Started
|
||||||
|
|
||||||
|
- **[Project Overview](PROJECT_PLAN.md)** - Vision, roadmap, and MVP phases
|
||||||
|
- **[Local Setup Guide](development/local-setup.md)** - Docker Compose development environment
|
||||||
|
- **[Getting Started](development/getting-started.md)** - Quick start for new developers
|
||||||
|
|
||||||
|
### 🏗️ Architecture
|
||||||
|
|
||||||
|
- **[Architecture Overview](architecture/overview.md)** - Tech stack, design decisions, data flow
|
||||||
|
- **[Database Schema](architecture/database.md)** - Tables, relationships, RLS policies, migrations
|
||||||
|
- **[API Reference](architecture/api.md)** - Supabase endpoints, Edge Functions, helpers
|
||||||
|
|
||||||
|
### 💻 Development
|
||||||
|
|
||||||
|
- **[Development Workflow](development/workflow.md)** - Daily workflow, conventions, best practices
|
||||||
|
- **[Git Workflow](development/git-workflow.md)** - Branching strategy, PR process, reviews
|
||||||
|
|
||||||
|
### 🚢 Deployment
|
||||||
|
|
||||||
|
- **[Production Deployment](deployment/production.md)** - Docker, Coolify, environment setup
|
||||||
|
- **[CI/CD Pipeline](deployment/ci-cd.md)** - Automated testing, builds, deployments
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 🎯 Quick Links by Role
|
||||||
|
|
||||||
|
### New Developer
|
||||||
|
1. [Local Setup Guide](development/local-setup.md) - Get running in 5 minutes
|
||||||
|
2. [Architecture Overview](architecture/overview.md) - Understand the stack
|
||||||
|
3. [Development Workflow](development/workflow.md) - Daily development process
|
||||||
|
|
||||||
|
### Feature Implementation
|
||||||
|
1. [Database Schema](architecture/database.md) - Table structure and queries
|
||||||
|
2. [API Reference](architecture/api.md) - Available endpoints
|
||||||
|
3. [Git Workflow](development/git-workflow.md) - Branch naming, PR checklist
|
||||||
|
|
||||||
|
### Deployment & Operations
|
||||||
|
1. [Production Deployment](deployment/production.md) - Deploy to production
|
||||||
|
2. [CI/CD Pipeline](deployment/ci-cd.md) - Automated workflows
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 📊 Current Status
|
||||||
|
|
||||||
|
**MVP Progress:** 14/34 issues complete (41.2%)
|
||||||
|
|
||||||
|
- ✅ Week 1: Database + Frontend Foundation (6/6)
|
||||||
|
- ✅ Week 2: Core Inventory Management (8/8)
|
||||||
|
- 🔄 Week 3: Barcode Scanning (1/5)
|
||||||
|
- ⏸️ Week 4-6: Tag UI, PWA, Deployment (20 remaining)
|
||||||
|
|
||||||
|
See [PROJECT_PLAN.md](PROJECT_PLAN.md) for detailed roadmap.
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 🤝 Contributing
|
||||||
|
|
||||||
|
For contribution guidelines, see [Development Workflow](development/workflow.md).
|
||||||
|
|
||||||
|
Key points:
|
||||||
|
- Feature branches off `develop`
|
||||||
|
- PRs require review before merge
|
||||||
|
- Follow conventional commits
|
||||||
|
- Write tests for new features
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 🔗 External Resources
|
||||||
|
|
||||||
|
- **Repository:** https://gitea.jeanlucmakiola.de/pantry-app/pantry
|
||||||
|
- **Supabase Docs:** https://supabase.com/docs
|
||||||
|
- **Nuxt 4 Docs:** https://nuxt.com
|
||||||
|
- **Open Food Facts API:** https://wiki.openfoodfacts.org/API
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
**Last Updated:** 2026-02-09
|
||||||
|
**Version:** 0.1.0-alpha (MVP in progress)
|
||||||
189
docs/development/getting-started.md
Normal file
@@ -0,0 +1,189 @@
|
|||||||
|
# Getting Started with Pantry Development
|
||||||
|
|
||||||
|
Welcome! This guide will get you from zero to running Pantry locally in ~5 minutes.
|
||||||
|
|
||||||
|
## 🎯 What You'll Build
|
||||||
|
|
||||||
|
A self-hosted kitchen inventory app with:
|
||||||
|
- Inventory management (add, edit, delete items)
|
||||||
|
- Tag-based organization (Fridge, Freezer, Dairy, etc.)
|
||||||
|
- Unit conversions (g, kg, L, cups)
|
||||||
|
- Barcode scanning (coming soon)
|
||||||
|
- PWA features (offline, installable)
|
||||||
|
|
||||||
|
## ⚡ Quick Start
|
||||||
|
|
||||||
|
### Prerequisites
|
||||||
|
|
||||||
|
- **Docker** & **Docker Compose** - [Install](https://docs.docker.com/get-docker/)
|
||||||
|
- **Bun** - [Install](https://bun.sh): `curl -fsSL https://bun.sh/install | bash`
|
||||||
|
- **Git**
|
||||||
|
|
||||||
|
### One-Command Setup
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# Clone the repo
|
||||||
|
git clone https://gitea.jeanlucmakiola.de/pantry-app/pantry.git
|
||||||
|
cd pantry
|
||||||
|
|
||||||
|
# Run the setup script
|
||||||
|
./dev.sh
|
||||||
|
```
|
||||||
|
|
||||||
|
That's it! The script will:
|
||||||
|
1. ✅ Start Supabase services (Docker Compose)
|
||||||
|
2. ✅ Wait for services to initialize
|
||||||
|
3. ✅ Install frontend dependencies
|
||||||
|
4. ✅ Launch Nuxt dev server
|
||||||
|
|
||||||
|
### Access the App
|
||||||
|
|
||||||
|
| Service | URL | Purpose |
|
||||||
|
|---------|-----|---------|
|
||||||
|
| **App** | `http://localhost:3000` | Main frontend |
|
||||||
|
| **Supabase Studio** | `http://localhost:54323` | Database admin UI |
|
||||||
|
| **API** | `http://localhost:54321` | Backend API |
|
||||||
|
|
||||||
|
## 🎮 Try It Out
|
||||||
|
|
||||||
|
1. Open `http://localhost:3000`
|
||||||
|
2. Click **"Add Manually"** to create your first item
|
||||||
|
3. Fill in:
|
||||||
|
- Name: "Milk"
|
||||||
|
- Quantity: 1
|
||||||
|
- Unit: Liter
|
||||||
|
- Tags: Fridge, Dairy
|
||||||
|
- Expiry: Set a date
|
||||||
|
4. Click **"Add Item"** and see it in the grid!
|
||||||
|
|
||||||
|
### Explore Features
|
||||||
|
|
||||||
|
- **Edit item:** Click "Edit" on any card
|
||||||
|
- **Adjust quantity:** Use +/- buttons
|
||||||
|
- **Delete item:** Click "Delete" (confirms first)
|
||||||
|
- **View database:** Open Supabase Studio at `:54323`
|
||||||
|
|
||||||
|
## 📁 Project Structure
|
||||||
|
|
||||||
|
```
|
||||||
|
pantry/
|
||||||
|
├── app/ # Nuxt 4 frontend
|
||||||
|
│ ├── components/ # Vue components
|
||||||
|
│ │ └── inventory/ # Inventory UI (List, Card, Forms)
|
||||||
|
│ ├── composables/ # Data hooks (useInventory, useSupabase)
|
||||||
|
│ ├── pages/ # Routes (index, scan, settings)
|
||||||
|
│ └── types/ # TypeScript definitions
|
||||||
|
├── supabase/
|
||||||
|
│ └── migrations/ # Database schema (001-005)
|
||||||
|
├── docker-compose.yml # Supabase services
|
||||||
|
├── docker/
|
||||||
|
│ └── kong.yml # API gateway config
|
||||||
|
└── docs/ # Documentation
|
||||||
|
```
|
||||||
|
|
||||||
|
## 🛠️ Common Tasks
|
||||||
|
|
||||||
|
### View Logs
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# All services
|
||||||
|
docker-compose logs -f
|
||||||
|
|
||||||
|
# Just the database
|
||||||
|
docker-compose logs -f db
|
||||||
|
```
|
||||||
|
|
||||||
|
### Reset Database
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# Stop and remove volumes (fresh start)
|
||||||
|
docker-compose down -v
|
||||||
|
|
||||||
|
# Restart (migrations auto-apply)
|
||||||
|
docker-compose up -d
|
||||||
|
```
|
||||||
|
|
||||||
|
### Access Database Directly
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# psql CLI
|
||||||
|
docker-compose exec db psql -U postgres -d postgres
|
||||||
|
|
||||||
|
# Or use Supabase Studio (GUI)
|
||||||
|
open http://localhost:54323
|
||||||
|
```
|
||||||
|
|
||||||
|
### Stop Everything
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# Stop services (keep data)
|
||||||
|
docker-compose stop
|
||||||
|
|
||||||
|
# Stop and remove everything
|
||||||
|
docker-compose down -v
|
||||||
|
```
|
||||||
|
|
||||||
|
## 🔍 What's Included
|
||||||
|
|
||||||
|
### Database (Pre-seeded)
|
||||||
|
|
||||||
|
**30 Units:**
|
||||||
|
- Weight: g, kg, mg, lb, oz
|
||||||
|
- Volume: mL, L, cup, tbsp, tsp
|
||||||
|
- Count: piece, dozen, bottle, can, jar
|
||||||
|
|
||||||
|
**33 Tags:**
|
||||||
|
- Position: Fridge, Freezer, Pantry
|
||||||
|
- Type: Dairy, Meat, Vegetables, Fruits
|
||||||
|
- Dietary: Vegan, Gluten-Free, Organic
|
||||||
|
- Custom: Low Stock, To Buy, Meal Prep
|
||||||
|
|
||||||
|
### Features (Working Now)
|
||||||
|
|
||||||
|
- ✅ Add/Edit/Delete inventory items
|
||||||
|
- ✅ Tag selection (multi-select)
|
||||||
|
- ✅ Unit conversions
|
||||||
|
- ✅ Expiry date tracking with warnings
|
||||||
|
- ✅ Responsive layout (mobile-ready)
|
||||||
|
- ✅ Quantity quick actions (+/- buttons)
|
||||||
|
|
||||||
|
### Features (Coming Soon)
|
||||||
|
|
||||||
|
- ⏳ Barcode scanning (Week 3)
|
||||||
|
- ⏳ User authentication UI
|
||||||
|
- ⏳ Tag management
|
||||||
|
- ⏳ PWA (offline mode)
|
||||||
|
|
||||||
|
## 📚 Next Steps
|
||||||
|
|
||||||
|
### Learn the Stack
|
||||||
|
|
||||||
|
1. **[Architecture Overview](../architecture/overview.md)** - Tech stack and design decisions
|
||||||
|
2. **[Database Schema](../architecture/database.md)** - Tables and relationships
|
||||||
|
3. **[Development Workflow](workflow.md)** - Daily development process
|
||||||
|
|
||||||
|
### Make Your First Change
|
||||||
|
|
||||||
|
1. Pick an issue from Gitea
|
||||||
|
2. Create a branch: `git checkout -b feature/your-feature`
|
||||||
|
3. Make changes, test locally
|
||||||
|
4. Commit: `git commit -m "feat: your feature"`
|
||||||
|
5. Push and create PR
|
||||||
|
|
||||||
|
### Troubleshooting
|
||||||
|
|
||||||
|
See **[Local Setup Guide](local-setup.md)** for:
|
||||||
|
- Port conflicts
|
||||||
|
- Database connection issues
|
||||||
|
- Frontend errors
|
||||||
|
- Environment variables
|
||||||
|
|
||||||
|
## 🤝 Need Help?
|
||||||
|
|
||||||
|
- **Documentation:** Browse `/docs` folder
|
||||||
|
- **Issues:** Create an issue on Gitea
|
||||||
|
- **Local setup:** See [local-setup.md](local-setup.md)
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
**Ready to code?** Check out the [Development Workflow](workflow.md)!
|
||||||
335
docs/development/local-setup.md
Normal file
@@ -0,0 +1,335 @@
|
|||||||
|
# Pantry - Local Development Setup
|
||||||
|
|
||||||
|
Self-hosted household pantry management app with barcode scanning.
|
||||||
|
|
||||||
|
## 🚀 Quick Start
|
||||||
|
|
||||||
|
### Prerequisites
|
||||||
|
|
||||||
|
- **Docker** & **Docker Compose** (for Supabase backend)
|
||||||
|
- **Bun** (for Nuxt frontend) - Install: `curl -fsSL https://bun.sh/install | bash`
|
||||||
|
- **Git**
|
||||||
|
|
||||||
|
### 1. Clone & Setup
|
||||||
|
|
||||||
|
```bash
|
||||||
|
git clone https://gitea.jeanlucmakiola.de/pantry-app/pantry.git
|
||||||
|
cd pantry
|
||||||
|
```
|
||||||
|
|
||||||
|
### 2. Start Supabase (Backend)
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# Start all Supabase services
|
||||||
|
docker-compose up -d
|
||||||
|
|
||||||
|
# Wait ~10 seconds for services to initialize
|
||||||
|
# Check status
|
||||||
|
docker-compose ps
|
||||||
|
```
|
||||||
|
|
||||||
|
**Services running:**
|
||||||
|
- PostgreSQL: `localhost:5432`
|
||||||
|
- Supabase API: `http://localhost:54321`
|
||||||
|
- Supabase Studio: `http://localhost:54323` (admin UI)
|
||||||
|
|
||||||
|
### 3. Apply Database Migrations
|
||||||
|
|
||||||
|
The migrations are automatically applied on first startup via `/docker-entrypoint-initdb.d`.
|
||||||
|
|
||||||
|
To verify:
|
||||||
|
```bash
|
||||||
|
docker-compose exec db psql -U postgres -d postgres -c "\dt"
|
||||||
|
```
|
||||||
|
|
||||||
|
You should see: `inventory_items`, `products`, `tags`, `units`, `item_tags`
|
||||||
|
|
||||||
|
### 4. Start Nuxt App (Frontend)
|
||||||
|
|
||||||
|
```bash
|
||||||
|
cd app
|
||||||
|
bun install
|
||||||
|
bun run dev
|
||||||
|
```
|
||||||
|
|
||||||
|
**App running:** `http://localhost:3000`
|
||||||
|
|
||||||
|
### 5. Open & Test
|
||||||
|
|
||||||
|
1. Visit `http://localhost:3000`
|
||||||
|
2. Click "Add Manually" to create your first inventory item
|
||||||
|
3. Access Supabase Studio at `http://localhost:54323` to view database
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 📁 Project Structure
|
||||||
|
|
||||||
|
```
|
||||||
|
pantry/
|
||||||
|
├── app/ # Nuxt 4 frontend
|
||||||
|
│ ├── components/ # Vue components
|
||||||
|
│ │ └── inventory/ # Inventory CRUD UI
|
||||||
|
│ ├── composables/ # Supabase client, data hooks
|
||||||
|
│ ├── pages/ # Routes (index, scan, settings)
|
||||||
|
│ └── types/ # TypeScript definitions
|
||||||
|
├── supabase/
|
||||||
|
│ └── migrations/ # SQL schema & seed data
|
||||||
|
├── docker/
|
||||||
|
│ └── kong.yml # API gateway config
|
||||||
|
├── docker-compose.yml # Supabase services
|
||||||
|
└── .env # Environment variables
|
||||||
|
```
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 🔧 Common Tasks
|
||||||
|
|
||||||
|
### View Logs
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# All services
|
||||||
|
docker-compose logs -f
|
||||||
|
|
||||||
|
# Specific service
|
||||||
|
docker-compose logs -f db
|
||||||
|
docker-compose logs -f auth
|
||||||
|
```
|
||||||
|
|
||||||
|
### Reset Database
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# Stop services
|
||||||
|
docker-compose down -v
|
||||||
|
|
||||||
|
# Restart (migrations auto-apply)
|
||||||
|
docker-compose up -d
|
||||||
|
```
|
||||||
|
|
||||||
|
### Access Database
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# psql
|
||||||
|
docker-compose exec db psql -U postgres -d postgres
|
||||||
|
|
||||||
|
# Supabase Studio (GUI)
|
||||||
|
# http://localhost:54323
|
||||||
|
```
|
||||||
|
|
||||||
|
### Run Migrations Manually
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# If you add new migrations after initial setup
|
||||||
|
docker-compose exec db psql -U postgres -d postgres -f /docker-entrypoint-initdb.d/003_helper_functions.sql
|
||||||
|
```
|
||||||
|
|
||||||
|
### Create Test User
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# Via Supabase Studio: http://localhost:54323
|
||||||
|
# → Authentication → Add User
|
||||||
|
# Email: test@example.com
|
||||||
|
# Password: password123
|
||||||
|
|
||||||
|
# Or via curl:
|
||||||
|
curl http://localhost:54321/auth/v1/signup \
|
||||||
|
-H "apikey: eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9..." \
|
||||||
|
-H "Content-Type: application/json" \
|
||||||
|
-d '{"email":"test@example.com","password":"password123"}'
|
||||||
|
```
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 🧪 Testing Features
|
||||||
|
|
||||||
|
### 1. Inventory Management
|
||||||
|
- Add items manually via form
|
||||||
|
- Edit quantities with +/- buttons
|
||||||
|
- Delete items
|
||||||
|
- View expiry warnings
|
||||||
|
|
||||||
|
### 2. Tags & Organization
|
||||||
|
- Pre-seeded tags: Fridge, Freezer, Pantry, Dairy, Vegan, etc.
|
||||||
|
- Multi-select tags when adding items
|
||||||
|
- Color-coded badges
|
||||||
|
|
||||||
|
### 3. Units
|
||||||
|
- 30 pre-seeded units (g, kg, L, cups, pieces, etc.)
|
||||||
|
- Automatic conversion support
|
||||||
|
|
||||||
|
### 4. Barcode Scanning (Week 3 - In Progress)
|
||||||
|
- Camera access (requires HTTPS in production)
|
||||||
|
- Manual barcode entry fallback
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 🐛 Troubleshooting
|
||||||
|
|
||||||
|
### Port Conflicts
|
||||||
|
|
||||||
|
If ports 5432, 54321, or 3000 are in use:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# Check what's using the port
|
||||||
|
sudo lsof -i :5432
|
||||||
|
|
||||||
|
# Option 1: Stop conflicting service
|
||||||
|
# Option 2: Change port in docker-compose.yml
|
||||||
|
```
|
||||||
|
|
||||||
|
### Database Connection Refused
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# Wait for PostgreSQL to fully start
|
||||||
|
docker-compose logs db | grep "ready to accept connections"
|
||||||
|
|
||||||
|
# If stuck, restart
|
||||||
|
docker-compose restart db
|
||||||
|
```
|
||||||
|
|
||||||
|
### Migrations Not Applied
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# Verify migrations directory is mounted
|
||||||
|
docker-compose exec db ls -la /docker-entrypoint-initdb.d
|
||||||
|
|
||||||
|
# Manually apply
|
||||||
|
docker-compose exec db bash
|
||||||
|
cd /docker-entrypoint-initdb.d
|
||||||
|
for f in *.sql; do psql -U postgres -d postgres -f "$f"; done
|
||||||
|
```
|
||||||
|
|
||||||
|
### Frontend: Module Not Found
|
||||||
|
|
||||||
|
```bash
|
||||||
|
cd app
|
||||||
|
rm -rf node_modules bun.lock .nuxt
|
||||||
|
bun install
|
||||||
|
bun run dev
|
||||||
|
```
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 📊 Database Schema
|
||||||
|
|
||||||
|
### Tables
|
||||||
|
|
||||||
|
| Table | Purpose | Rows (Est.) |
|
||||||
|
|-------|---------|-------------|
|
||||||
|
| `inventory_items` | Current inventory | 100-500 |
|
||||||
|
| `products` | Barcode cache (Open Food Facts) | 500-2000 |
|
||||||
|
| `tags` | Organization labels | 50 (33 pre-seeded) |
|
||||||
|
| `units` | Measurement units | 50 (30 pre-seeded) |
|
||||||
|
| `item_tags` | Many-to-many item ↔ tag | 200-1000 |
|
||||||
|
|
||||||
|
### Pre-Seeded Data
|
||||||
|
|
||||||
|
**Units (30):**
|
||||||
|
- Weight: g, kg, mg, lb, oz
|
||||||
|
- Volume: mL, L, cup, tbsp, tsp, gal, qt, pt
|
||||||
|
- Count: piece, dozen, package, bottle, can, jar, box, bag
|
||||||
|
|
||||||
|
**Tags (33):**
|
||||||
|
- Position: Fridge, Freezer, Pantry, Cabinet
|
||||||
|
- Type: Dairy, Meat, Vegetables, Fruits, Snacks
|
||||||
|
- Dietary: Vegan, Gluten-Free, Organic, Kosher, Halal
|
||||||
|
- Custom: Low Stock, To Buy, Meal Prep, Leftovers
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 🔐 Authentication
|
||||||
|
|
||||||
|
**Default Setup (Development):**
|
||||||
|
- Auto-confirm emails (no SMTP needed)
|
||||||
|
- Anyone can sign up
|
||||||
|
- JWT tokens valid for 1 hour
|
||||||
|
|
||||||
|
**Create Admin User:**
|
||||||
|
|
||||||
|
```sql
|
||||||
|
-- Via psql
|
||||||
|
docker-compose exec db psql -U postgres -d postgres
|
||||||
|
|
||||||
|
INSERT INTO auth.users (id, email, encrypted_password, email_confirmed_at)
|
||||||
|
VALUES (
|
||||||
|
gen_random_uuid(),
|
||||||
|
'admin@pantry.local',
|
||||||
|
crypt('admin123', gen_salt('bf')),
|
||||||
|
NOW()
|
||||||
|
);
|
||||||
|
```
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 🌐 Environment Variables
|
||||||
|
|
||||||
|
**.env** (root)
|
||||||
|
```bash
|
||||||
|
POSTGRES_PASSWORD=postgres
|
||||||
|
JWT_SECRET=your-secret-here
|
||||||
|
ANON_KEY=<supabase-anon-key>
|
||||||
|
SERVICE_ROLE_KEY=<supabase-service-role-key>
|
||||||
|
```
|
||||||
|
|
||||||
|
**app/.env** (Nuxt)
|
||||||
|
```bash
|
||||||
|
NUXT_PUBLIC_SUPABASE_URL=http://localhost:54321
|
||||||
|
NUXT_PUBLIC_SUPABASE_ANON_KEY=<same-as-above>
|
||||||
|
```
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 📈 Development Workflow
|
||||||
|
|
||||||
|
1. **Make changes** to Nuxt app → Hot reload at `localhost:3000`
|
||||||
|
2. **Database changes** → Create new migration in `supabase/migrations/`
|
||||||
|
3. **Test** → Add items, scan barcodes, check database
|
||||||
|
4. **Commit** → Feature branch → PR to `develop`
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 🚢 Production Deployment (Coming Soon)
|
||||||
|
|
||||||
|
See `docs/DEPLOYMENT.md` for:
|
||||||
|
- Coolify setup
|
||||||
|
- Environment configuration
|
||||||
|
- SSL/HTTPS setup
|
||||||
|
- Backups
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 📚 Documentation
|
||||||
|
|
||||||
|
- [Architecture](docs/ARCHITECTURE.md)
|
||||||
|
- [Database Schema](docs/DATABASE.md)
|
||||||
|
- [API Reference](docs/API.md)
|
||||||
|
- [Development Guide](docs/DEVELOPMENT.md)
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 🤝 Contributing
|
||||||
|
|
||||||
|
This is a personal project, but issues and PRs welcome!
|
||||||
|
|
||||||
|
1. Fork the repo
|
||||||
|
2. Create feature branch (`git checkout -b feature/amazing-feature`)
|
||||||
|
3. Commit changes (`git commit -m 'Add amazing feature'`)
|
||||||
|
4. Push to branch (`git push origin feature/amazing-feature`)
|
||||||
|
5. Open Pull Request
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 📝 License
|
||||||
|
|
||||||
|
MIT License - See LICENSE file
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 🙋 Support
|
||||||
|
|
||||||
|
- Issues: https://gitea.jeanlucmakiola.de/pantry-app/pantry/issues
|
||||||
|
- Docs: https://gitea.jeanlucmakiola.de/pantry-app/pantry/wiki
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
**Version:** 0.1.0-alpha (MVP in progress)
|
||||||
|
**Status:** Week 2 complete, Week 3 in progress (14/34 issues done)
|
||||||
83
supabase/functions/product-lookup/README.md
Normal file
@@ -0,0 +1,83 @@
|
|||||||
|
# Product Lookup Edge Function
|
||||||
|
|
||||||
|
Fetches product data from Open Food Facts API by barcode and caches results in the database.
|
||||||
|
|
||||||
|
## Endpoint
|
||||||
|
|
||||||
|
`POST /functions/v1/product-lookup`
|
||||||
|
|
||||||
|
## Request
|
||||||
|
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"barcode": "8000500310427"
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
## Response
|
||||||
|
|
||||||
|
### Success (200)
|
||||||
|
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"barcode": "8000500310427",
|
||||||
|
"name": "Nutella",
|
||||||
|
"brand": "Ferrero",
|
||||||
|
"quantity": "750g",
|
||||||
|
"image_url": "https://...",
|
||||||
|
"category": "spreads",
|
||||||
|
"cached": false
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
### Not Found (404)
|
||||||
|
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"barcode": "1234567890123",
|
||||||
|
"name": "Unknown Product (1234567890123)",
|
||||||
|
"cached": false
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
### Error (500)
|
||||||
|
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"error": "Error message",
|
||||||
|
"barcode": null,
|
||||||
|
"name": null
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
## Features
|
||||||
|
|
||||||
|
- ✅ Queries Open Food Facts API
|
||||||
|
- ✅ Caches results in `products` table
|
||||||
|
- ✅ Returns cached data for subsequent requests
|
||||||
|
- ✅ Handles product not found gracefully
|
||||||
|
- ✅ CORS enabled for frontend access
|
||||||
|
|
||||||
|
## Environment Variables
|
||||||
|
|
||||||
|
- `SUPABASE_URL`: Auto-injected by Supabase
|
||||||
|
- `SUPABASE_SERVICE_ROLE_KEY`: Auto-injected by Supabase
|
||||||
|
|
||||||
|
## Testing
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# Local (with Supabase CLI)
|
||||||
|
supabase functions serve product-lookup
|
||||||
|
|
||||||
|
# Test request
|
||||||
|
curl -X POST http://localhost:54321/functions/v1/product-lookup \
|
||||||
|
-H "Content-Type: application/json" \
|
||||||
|
-H "Authorization: Bearer YOUR_ANON_KEY" \
|
||||||
|
-d '{"barcode":"8000500310427"}'
|
||||||
|
```
|
||||||
|
|
||||||
|
## Deployment
|
||||||
|
|
||||||
|
```bash
|
||||||
|
supabase functions deploy product-lookup
|
||||||
|
```
|
||||||
140
supabase/functions/product-lookup/index.ts
Normal file
@@ -0,0 +1,140 @@
|
|||||||
|
// Product Lookup Edge Function
|
||||||
|
// Fetches product data from Open Food Facts API by barcode
|
||||||
|
|
||||||
|
import { serve } from 'https://deno.land/std@0.168.0/http/server.ts'
|
||||||
|
import { createClient } from 'https://esm.sh/@supabase/supabase-js@2'
|
||||||
|
|
||||||
|
const corsHeaders = {
|
||||||
|
'Access-Control-Allow-Origin': '*',
|
||||||
|
'Access-Control-Allow-Headers': 'authorization, x-client-info, apikey, content-type',
|
||||||
|
}
|
||||||
|
|
||||||
|
interface ProductData {
|
||||||
|
barcode: string
|
||||||
|
name: string
|
||||||
|
brand?: string
|
||||||
|
quantity?: string
|
||||||
|
image_url?: string
|
||||||
|
category?: string
|
||||||
|
cached?: boolean
|
||||||
|
}
|
||||||
|
|
||||||
|
serve(async (req) => {
|
||||||
|
// Handle CORS preflight
|
||||||
|
if (req.method === 'OPTIONS') {
|
||||||
|
return new Response('ok', { headers: corsHeaders })
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
const { barcode } = await req.json()
|
||||||
|
|
||||||
|
if (!barcode) {
|
||||||
|
throw new Error('Barcode is required')
|
||||||
|
}
|
||||||
|
|
||||||
|
// Initialize Supabase client
|
||||||
|
const supabaseUrl = Deno.env.get('SUPABASE_URL')!
|
||||||
|
const supabaseKey = Deno.env.get('SUPABASE_SERVICE_ROLE_KEY')!
|
||||||
|
const supabase = createClient(supabaseUrl, supabaseKey)
|
||||||
|
|
||||||
|
// Check cache first (products table can store known products)
|
||||||
|
const { data: cachedProduct } = await supabase
|
||||||
|
.from('products')
|
||||||
|
.select('*')
|
||||||
|
.eq('barcode', barcode)
|
||||||
|
.single()
|
||||||
|
|
||||||
|
if (cachedProduct) {
|
||||||
|
console.log(`Cache HIT for barcode: ${barcode}`)
|
||||||
|
return new Response(
|
||||||
|
JSON.stringify({
|
||||||
|
barcode: cachedProduct.barcode,
|
||||||
|
name: cachedProduct.name,
|
||||||
|
brand: cachedProduct.brand,
|
||||||
|
quantity: cachedProduct.quantity,
|
||||||
|
image_url: cachedProduct.image_url,
|
||||||
|
category: cachedProduct.category,
|
||||||
|
cached: true,
|
||||||
|
} as ProductData),
|
||||||
|
{ headers: { ...corsHeaders, 'Content-Type': 'application/json' } }
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
console.log(`Cache MISS for barcode: ${barcode}, fetching from Open Food Facts...`)
|
||||||
|
|
||||||
|
// Fetch from Open Food Facts
|
||||||
|
const offResponse = await fetch(
|
||||||
|
`https://world.openfoodfacts.org/api/v2/product/${barcode}.json`,
|
||||||
|
{
|
||||||
|
headers: {
|
||||||
|
'User-Agent': 'Pantry/1.0 (https://github.com/pantry-app/pantry)',
|
||||||
|
},
|
||||||
|
}
|
||||||
|
)
|
||||||
|
|
||||||
|
if (!offResponse.ok) {
|
||||||
|
throw new Error(`Open Food Facts API error: ${offResponse.status}`)
|
||||||
|
}
|
||||||
|
|
||||||
|
const offData = await offResponse.json()
|
||||||
|
|
||||||
|
if (offData.status !== 1 || !offData.product) {
|
||||||
|
// Product not found in Open Food Facts
|
||||||
|
return new Response(
|
||||||
|
JSON.stringify({
|
||||||
|
barcode,
|
||||||
|
name: `Unknown Product (${barcode})`,
|
||||||
|
cached: false,
|
||||||
|
} as ProductData),
|
||||||
|
{
|
||||||
|
headers: { ...corsHeaders, 'Content-Type': 'application/json' },
|
||||||
|
status: 404
|
||||||
|
}
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
const product = offData.product
|
||||||
|
|
||||||
|
// Extract relevant data
|
||||||
|
const productData: ProductData = {
|
||||||
|
barcode,
|
||||||
|
name: product.product_name || product.generic_name || `Product ${barcode}`,
|
||||||
|
brand: product.brands || undefined,
|
||||||
|
quantity: product.quantity || undefined,
|
||||||
|
image_url: product.image_url || product.image_front_url || undefined,
|
||||||
|
category: product.categories || undefined,
|
||||||
|
cached: false,
|
||||||
|
}
|
||||||
|
|
||||||
|
// Cache the product in our database (upsert)
|
||||||
|
await supabase.from('products').upsert({
|
||||||
|
barcode: productData.barcode,
|
||||||
|
name: productData.name,
|
||||||
|
brand: productData.brand,
|
||||||
|
quantity: productData.quantity,
|
||||||
|
image_url: productData.image_url,
|
||||||
|
category: productData.category,
|
||||||
|
}, { onConflict: 'barcode' })
|
||||||
|
|
||||||
|
console.log(`Successfully fetched and cached product: ${productData.name}`)
|
||||||
|
|
||||||
|
return new Response(
|
||||||
|
JSON.stringify(productData),
|
||||||
|
{ headers: { ...corsHeaders, 'Content-Type': 'application/json' } }
|
||||||
|
)
|
||||||
|
|
||||||
|
} catch (error) {
|
||||||
|
console.error('Error in product-lookup:', error)
|
||||||
|
return new Response(
|
||||||
|
JSON.stringify({
|
||||||
|
error: error instanceof Error ? error.message : 'Unknown error',
|
||||||
|
barcode: null,
|
||||||
|
name: null,
|
||||||
|
}),
|
||||||
|
{
|
||||||
|
headers: { ...corsHeaders, 'Content-Type': 'application/json' },
|
||||||
|
status: 500
|
||||||
|
}
|
||||||
|
)
|
||||||
|
}
|
||||||
|
})
|
||||||
251
supabase/migrations/003_helper_functions.sql
Normal file
@@ -0,0 +1,251 @@
|
|||||||
|
-- Migration: Additional SQL Functions for Inventory Management
|
||||||
|
-- Week 2: Helper functions for common queries
|
||||||
|
|
||||||
|
-- Function: Get inventory items with full details (tags, product info, unit conversion)
|
||||||
|
CREATE OR REPLACE FUNCTION get_inventory_details()
|
||||||
|
RETURNS TABLE (
|
||||||
|
item_id UUID,
|
||||||
|
item_name TEXT,
|
||||||
|
quantity DECIMAL,
|
||||||
|
unit_abbreviation TEXT,
|
||||||
|
unit_name TEXT,
|
||||||
|
expiry_date DATE,
|
||||||
|
days_until_expiry INTEGER,
|
||||||
|
tags TEXT[],
|
||||||
|
product_brand TEXT,
|
||||||
|
product_image_url TEXT,
|
||||||
|
product_barcode TEXT,
|
||||||
|
created_at TIMESTAMPTZ,
|
||||||
|
updated_at TIMESTAMPTZ
|
||||||
|
) AS $$
|
||||||
|
BEGIN
|
||||||
|
RETURN QUERY
|
||||||
|
SELECT
|
||||||
|
i.id AS item_id,
|
||||||
|
i.name AS item_name,
|
||||||
|
i.quantity,
|
||||||
|
u.abbreviation AS unit_abbreviation,
|
||||||
|
u.name AS unit_name,
|
||||||
|
i.expiry_date,
|
||||||
|
(i.expiry_date - CURRENT_DATE) AS days_until_expiry,
|
||||||
|
COALESCE(ARRAY_AGG(DISTINCT t.name) FILTER (WHERE t.name IS NOT NULL), '{}') AS tags,
|
||||||
|
p.brand AS product_brand,
|
||||||
|
p.image_url AS product_image_url,
|
||||||
|
p.barcode AS product_barcode,
|
||||||
|
i.created_at,
|
||||||
|
i.updated_at
|
||||||
|
FROM inventory_items i
|
||||||
|
JOIN units u ON i.unit_id = u.id
|
||||||
|
LEFT JOIN products p ON i.product_id = p.id
|
||||||
|
LEFT JOIN item_tags it ON i.id = it.item_id
|
||||||
|
LEFT JOIN tags t ON it.tag_id = t.id
|
||||||
|
GROUP BY
|
||||||
|
i.id, i.name, i.quantity, u.abbreviation, u.name,
|
||||||
|
i.expiry_date, p.brand, p.image_url, p.barcode,
|
||||||
|
i.created_at, i.updated_at
|
||||||
|
ORDER BY i.created_at DESC;
|
||||||
|
END;
|
||||||
|
$$ LANGUAGE plpgsql;
|
||||||
|
|
||||||
|
COMMENT ON FUNCTION get_inventory_details() IS 'Returns all inventory items with denormalized data for display';
|
||||||
|
|
||||||
|
-- Function: Get items expiring soon
|
||||||
|
CREATE OR REPLACE FUNCTION get_expiring_items(days_ahead INTEGER DEFAULT 7)
|
||||||
|
RETURNS TABLE (
|
||||||
|
item_id UUID,
|
||||||
|
item_name TEXT,
|
||||||
|
quantity DECIMAL,
|
||||||
|
unit_abbreviation TEXT,
|
||||||
|
expiry_date DATE,
|
||||||
|
days_until_expiry INTEGER,
|
||||||
|
tags TEXT[]
|
||||||
|
) AS $$
|
||||||
|
BEGIN
|
||||||
|
RETURN QUERY
|
||||||
|
SELECT
|
||||||
|
i.id AS item_id,
|
||||||
|
i.name AS item_name,
|
||||||
|
i.quantity,
|
||||||
|
u.abbreviation AS unit_abbreviation,
|
||||||
|
i.expiry_date,
|
||||||
|
(i.expiry_date - CURRENT_DATE) AS days_until_expiry,
|
||||||
|
COALESCE(ARRAY_AGG(DISTINCT t.name) FILTER (WHERE t.name IS NOT NULL), '{}') AS tags
|
||||||
|
FROM inventory_items i
|
||||||
|
JOIN units u ON i.unit_id = u.id
|
||||||
|
LEFT JOIN item_tags it ON i.id = it.item_id
|
||||||
|
LEFT JOIN tags t ON it.tag_id = t.id
|
||||||
|
WHERE
|
||||||
|
i.expiry_date IS NOT NULL
|
||||||
|
AND i.expiry_date <= CURRENT_DATE + MAKE_INTERVAL(days => days_ahead)
|
||||||
|
GROUP BY i.id, i.name, i.quantity, u.abbreviation, i.expiry_date
|
||||||
|
ORDER BY i.expiry_date ASC;
|
||||||
|
END;
|
||||||
|
$$ LANGUAGE plpgsql;
|
||||||
|
|
||||||
|
COMMENT ON FUNCTION get_expiring_items(INTEGER) IS 'Returns items expiring within specified days (default 7)';
|
||||||
|
|
||||||
|
-- Function: Get items by tag
|
||||||
|
CREATE OR REPLACE FUNCTION get_items_by_tag(tag_name TEXT)
|
||||||
|
RETURNS TABLE (
|
||||||
|
item_id UUID,
|
||||||
|
item_name TEXT,
|
||||||
|
quantity DECIMAL,
|
||||||
|
unit_abbreviation TEXT,
|
||||||
|
expiry_date DATE,
|
||||||
|
created_at TIMESTAMPTZ
|
||||||
|
) AS $$
|
||||||
|
BEGIN
|
||||||
|
RETURN QUERY
|
||||||
|
SELECT
|
||||||
|
i.id AS item_id,
|
||||||
|
i.name AS item_name,
|
||||||
|
i.quantity,
|
||||||
|
u.abbreviation AS unit_abbreviation,
|
||||||
|
i.expiry_date,
|
||||||
|
i.created_at
|
||||||
|
FROM inventory_items i
|
||||||
|
JOIN units u ON i.unit_id = u.id
|
||||||
|
JOIN item_tags it ON i.id = it.item_id
|
||||||
|
JOIN tags t ON it.tag_id = t.id
|
||||||
|
WHERE t.name ILIKE tag_name
|
||||||
|
ORDER BY i.created_at DESC;
|
||||||
|
END;
|
||||||
|
$$ LANGUAGE plpgsql;
|
||||||
|
|
||||||
|
COMMENT ON FUNCTION get_items_by_tag(TEXT) IS 'Returns all items with specified tag (case-insensitive)';
|
||||||
|
|
||||||
|
-- Function: Get low stock items (quantity <= threshold)
|
||||||
|
CREATE OR REPLACE FUNCTION get_low_stock_items(threshold DECIMAL DEFAULT 1.0)
|
||||||
|
RETURNS TABLE (
|
||||||
|
item_id UUID,
|
||||||
|
item_name TEXT,
|
||||||
|
quantity DECIMAL,
|
||||||
|
unit_abbreviation TEXT,
|
||||||
|
tags TEXT[]
|
||||||
|
) AS $$
|
||||||
|
BEGIN
|
||||||
|
RETURN QUERY
|
||||||
|
SELECT
|
||||||
|
i.id AS item_id,
|
||||||
|
i.name AS item_name,
|
||||||
|
i.quantity,
|
||||||
|
u.abbreviation AS unit_abbreviation,
|
||||||
|
COALESCE(ARRAY_AGG(DISTINCT t.name) FILTER (WHERE t.name IS NOT NULL), '{}') AS tags
|
||||||
|
FROM inventory_items i
|
||||||
|
JOIN units u ON i.unit_id = u.id
|
||||||
|
LEFT JOIN item_tags it ON i.id = it.item_id
|
||||||
|
LEFT JOIN tags t ON it.tag_id = t.id
|
||||||
|
WHERE i.quantity <= threshold
|
||||||
|
GROUP BY i.id, i.name, i.quantity, u.abbreviation
|
||||||
|
ORDER BY i.quantity ASC, i.name ASC;
|
||||||
|
END;
|
||||||
|
$$ LANGUAGE plpgsql;
|
||||||
|
|
||||||
|
COMMENT ON FUNCTION get_low_stock_items(DECIMAL) IS 'Returns items with quantity at or below threshold';
|
||||||
|
|
||||||
|
-- Function: Update item quantity (consume or restock)
|
||||||
|
CREATE OR REPLACE FUNCTION update_item_quantity(
|
||||||
|
item_uuid UUID,
|
||||||
|
quantity_change DECIMAL,
|
||||||
|
delete_if_zero BOOLEAN DEFAULT TRUE
|
||||||
|
)
|
||||||
|
RETURNS BOOLEAN AS $$
|
||||||
|
DECLARE
|
||||||
|
new_quantity DECIMAL;
|
||||||
|
BEGIN
|
||||||
|
-- Calculate new quantity
|
||||||
|
SELECT quantity + quantity_change INTO new_quantity
|
||||||
|
FROM inventory_items
|
||||||
|
WHERE id = item_uuid;
|
||||||
|
|
||||||
|
IF new_quantity IS NULL THEN
|
||||||
|
RETURN FALSE; -- Item not found
|
||||||
|
END IF;
|
||||||
|
|
||||||
|
-- Delete if zero and flag is set
|
||||||
|
IF new_quantity <= 0 AND delete_if_zero THEN
|
||||||
|
DELETE FROM inventory_items WHERE id = item_uuid;
|
||||||
|
RETURN TRUE;
|
||||||
|
END IF;
|
||||||
|
|
||||||
|
-- Update quantity (ensure non-negative)
|
||||||
|
UPDATE inventory_items
|
||||||
|
SET quantity = GREATEST(new_quantity, 0),
|
||||||
|
updated_at = NOW()
|
||||||
|
WHERE id = item_uuid;
|
||||||
|
|
||||||
|
RETURN TRUE;
|
||||||
|
END;
|
||||||
|
$$ LANGUAGE plpgsql;
|
||||||
|
|
||||||
|
COMMENT ON FUNCTION update_item_quantity(UUID, DECIMAL, BOOLEAN) IS 'Updates item quantity (positive for restock, negative for consume). Optionally deletes if zero.';
|
||||||
|
|
||||||
|
-- Function: Get inventory statistics
|
||||||
|
CREATE OR REPLACE FUNCTION get_inventory_stats()
|
||||||
|
RETURNS TABLE (
|
||||||
|
total_items BIGINT,
|
||||||
|
total_unique_products BIGINT,
|
||||||
|
items_expiring_week BIGINT,
|
||||||
|
items_expired BIGINT,
|
||||||
|
total_tags_used BIGINT
|
||||||
|
) AS $$
|
||||||
|
BEGIN
|
||||||
|
RETURN QUERY
|
||||||
|
SELECT
|
||||||
|
COUNT(DISTINCT i.id) AS total_items,
|
||||||
|
COUNT(DISTINCT i.product_id) FILTER (WHERE i.product_id IS NOT NULL) AS total_unique_products,
|
||||||
|
COUNT(i.id) FILTER (
|
||||||
|
WHERE i.expiry_date IS NOT NULL
|
||||||
|
AND i.expiry_date BETWEEN CURRENT_DATE AND CURRENT_DATE + INTERVAL '7 days'
|
||||||
|
) AS items_expiring_week,
|
||||||
|
COUNT(i.id) FILTER (
|
||||||
|
WHERE i.expiry_date IS NOT NULL
|
||||||
|
AND i.expiry_date < CURRENT_DATE
|
||||||
|
) AS items_expired,
|
||||||
|
COUNT(DISTINCT it.tag_id) AS total_tags_used
|
||||||
|
FROM inventory_items i
|
||||||
|
LEFT JOIN item_tags it ON i.id = it.item_id;
|
||||||
|
END;
|
||||||
|
$$ LANGUAGE plpgsql;
|
||||||
|
|
||||||
|
COMMENT ON FUNCTION get_inventory_stats() IS 'Returns summary statistics for the entire inventory';
|
||||||
|
|
||||||
|
-- Function: Search inventory (full-text search on items and products)
|
||||||
|
CREATE OR REPLACE FUNCTION search_inventory(search_query TEXT)
|
||||||
|
RETURNS TABLE (
|
||||||
|
item_id UUID,
|
||||||
|
item_name TEXT,
|
||||||
|
quantity DECIMAL,
|
||||||
|
unit_abbreviation TEXT,
|
||||||
|
product_brand TEXT,
|
||||||
|
tags TEXT[],
|
||||||
|
relevance REAL
|
||||||
|
) AS $$
|
||||||
|
BEGIN
|
||||||
|
RETURN QUERY
|
||||||
|
SELECT
|
||||||
|
i.id AS item_id,
|
||||||
|
i.name AS item_name,
|
||||||
|
i.quantity,
|
||||||
|
u.abbreviation AS unit_abbreviation,
|
||||||
|
p.brand AS product_brand,
|
||||||
|
COALESCE(ARRAY_AGG(DISTINCT t.name) FILTER (WHERE t.name IS NOT NULL), '{}') AS tags,
|
||||||
|
ts_rank(
|
||||||
|
to_tsvector('english', i.name || ' ' || COALESCE(p.brand, '') || ' ' || COALESCE(p.name, '')),
|
||||||
|
plainto_tsquery('english', search_query)
|
||||||
|
) AS relevance
|
||||||
|
FROM inventory_items i
|
||||||
|
JOIN units u ON i.unit_id = u.id
|
||||||
|
LEFT JOIN products p ON i.product_id = p.id
|
||||||
|
LEFT JOIN item_tags it ON i.id = it.item_id
|
||||||
|
LEFT JOIN tags t ON it.tag_id = t.id
|
||||||
|
WHERE
|
||||||
|
to_tsvector('english', i.name || ' ' || COALESCE(p.brand, '') || ' ' || COALESCE(p.name, ''))
|
||||||
|
@@ plainto_tsquery('english', search_query)
|
||||||
|
GROUP BY i.id, i.name, i.quantity, u.abbreviation, p.brand, p.name
|
||||||
|
ORDER BY relevance DESC, i.created_at DESC
|
||||||
|
LIMIT 50;
|
||||||
|
END;
|
||||||
|
$$ LANGUAGE plpgsql;
|
||||||
|
|
||||||
|
COMMENT ON FUNCTION search_inventory(TEXT) IS 'Full-text search across inventory items and products';
|
||||||
37
supabase/migrations/004_seed_units.sql
Normal file
@@ -0,0 +1,37 @@
|
|||||||
|
-- Migration: Seed Default Units
|
||||||
|
-- Week 2: Pre-populate common measurement units with conversions
|
||||||
|
|
||||||
|
-- Weight units (metric base: gram)
|
||||||
|
INSERT INTO units (id, name, abbreviation, unit_type, base_unit_id, conversion_factor, is_default, created_by) VALUES
|
||||||
|
('f47ac10b-58cc-4372-a567-0e02b2c3d479', 'Gram', 'g', 'weight', NULL, 1.0, TRUE, NULL),
|
||||||
|
('550e8400-e29b-41d4-a716-446655440001', 'Kilogram', 'kg', 'weight', 'f47ac10b-58cc-4372-a567-0e02b2c3d479', 1000.0, FALSE, NULL),
|
||||||
|
('550e8400-e29b-41d4-a716-446655440002', 'Milligram', 'mg', 'weight', 'f47ac10b-58cc-4372-a567-0e02b2c3d479', 0.001, FALSE, NULL),
|
||||||
|
('550e8400-e29b-41d4-a716-446655440003', 'Pound', 'lb', 'weight', 'f47ac10b-58cc-4372-a567-0e02b2c3d479', 453.592, FALSE, NULL),
|
||||||
|
('550e8400-e29b-41d4-a716-446655440004', 'Ounce', 'oz', 'weight', 'f47ac10b-58cc-4372-a567-0e02b2c3d479', 28.3495, FALSE, NULL);
|
||||||
|
|
||||||
|
-- Volume units (metric base: milliliter)
|
||||||
|
INSERT INTO units (id, name, abbreviation, unit_type, base_unit_id, conversion_factor, is_default, created_by) VALUES
|
||||||
|
('550e8400-e29b-41d4-a716-446655440010', 'Milliliter', 'mL', 'volume', NULL, 1.0, TRUE, NULL),
|
||||||
|
('550e8400-e29b-41d4-a716-446655440011', 'Liter', 'L', 'volume', '550e8400-e29b-41d4-a716-446655440010', 1000.0, FALSE, NULL),
|
||||||
|
('550e8400-e29b-41d4-a716-446655440012', 'Centiliter', 'cL', 'volume', '550e8400-e29b-41d4-a716-446655440010', 10.0, FALSE, NULL),
|
||||||
|
('550e8400-e29b-41d4-a716-446655440013', 'Deciliter', 'dL', 'volume', '550e8400-e29b-41d4-a716-446655440010', 100.0, FALSE, NULL),
|
||||||
|
('550e8400-e29b-41d4-a716-446655440014', 'Cup', 'cup', 'volume', '550e8400-e29b-41d4-a716-446655440010', 236.588, FALSE, NULL),
|
||||||
|
('550e8400-e29b-41d4-a716-446655440015', 'Tablespoon', 'tbsp', 'volume', '550e8400-e29b-41d4-a716-446655440010', 14.7868, FALSE, NULL),
|
||||||
|
('550e8400-e29b-41d4-a716-446655440016', 'Teaspoon', 'tsp', 'volume', '550e8400-e29b-41d4-a716-446655440010', 4.92892, FALSE, NULL),
|
||||||
|
('550e8400-e29b-41d4-a716-446655440017', 'Fluid Ounce', 'fl oz', 'volume', '550e8400-e29b-41d4-a716-446655440010', 29.5735, FALSE, NULL),
|
||||||
|
('550e8400-e29b-41d4-a716-446655440018', 'Gallon', 'gal', 'volume', '550e8400-e29b-41d4-a716-446655440010', 3785.41, FALSE, NULL),
|
||||||
|
('550e8400-e29b-41d4-a716-446655440019', 'Quart', 'qt', 'volume', '550e8400-e29b-41d4-a716-446655440010', 946.353, FALSE, NULL),
|
||||||
|
('550e8400-e29b-41d4-a716-446655440020', 'Pint', 'pt', 'volume', '550e8400-e29b-41d4-a716-446655440010', 473.176, FALSE, NULL);
|
||||||
|
|
||||||
|
-- Count units (no conversions, each is independent)
|
||||||
|
INSERT INTO units (id, name, abbreviation, unit_type, base_unit_id, conversion_factor, is_default, created_by) VALUES
|
||||||
|
('550e8400-e29b-41d4-a716-446655440030', 'Piece', 'pc', 'count', NULL, 1.0, TRUE, NULL),
|
||||||
|
('550e8400-e29b-41d4-a716-446655440031', 'Dozen', 'doz', 'count', '550e8400-e29b-41d4-a716-446655440030', 12.0, FALSE, NULL),
|
||||||
|
('550e8400-e29b-41d4-a716-446655440032', 'Package', 'pkg', 'count', NULL, 1.0, FALSE, NULL),
|
||||||
|
('550e8400-e29b-41d4-a716-446655440033', 'Bottle', 'btl', 'count', NULL, 1.0, FALSE, NULL),
|
||||||
|
('550e8400-e29b-41d4-a716-446655440034', 'Can', 'can', 'count', NULL, 1.0, FALSE, NULL),
|
||||||
|
('550e8400-e29b-41d4-a716-446655440035', 'Jar', 'jar', 'count', NULL, 1.0, FALSE, NULL),
|
||||||
|
('550e8400-e29b-41d4-a716-446655440036', 'Box', 'box', 'count', NULL, 1.0, FALSE, NULL),
|
||||||
|
('550e8400-e29b-41d4-a716-446655440037', 'Bag', 'bag', 'count', NULL, 1.0, FALSE, NULL);
|
||||||
|
|
||||||
|
COMMENT ON TABLE units IS 'Measurement units with 30 common presets covering metric, imperial, and count units';
|
||||||
49
supabase/migrations/005_seed_tags.sql
Normal file
@@ -0,0 +1,49 @@
|
|||||||
|
-- Migration: Seed Default Tags
|
||||||
|
-- Week 2: Pre-populate common organizational tags
|
||||||
|
|
||||||
|
-- Position Tags (where items are stored)
|
||||||
|
INSERT INTO tags (id, name, category, icon, color, created_by) VALUES
|
||||||
|
('650e8400-e29b-41d4-a716-446655440001', 'Fridge', 'position', '🧊', '#3b82f6', NULL),
|
||||||
|
('650e8400-e29b-41d4-a716-446655440002', 'Freezer', 'position', '❄️', '#06b6d4', NULL),
|
||||||
|
('650e8400-e29b-41d4-a716-446655440003', 'Pantry', 'position', '🗄️', '#8b5cf6', NULL),
|
||||||
|
('650e8400-e29b-41d4-a716-446655440004', 'Cabinet', 'position', '🚪', '#6b7280', NULL),
|
||||||
|
('650e8400-e29b-41d4-a716-446655440005', 'Countertop', 'position', '🍽️', '#f59e0b', NULL),
|
||||||
|
('650e8400-e29b-41d4-a716-446655440006', 'Cellar', 'position', '🏚️', '#78350f', NULL);
|
||||||
|
|
||||||
|
-- Type Tags (food categories)
|
||||||
|
INSERT INTO tags (id, name, category, icon, color, created_by) VALUES
|
||||||
|
('650e8400-e29b-41d4-a716-446655440010', 'Dairy', 'type', '🧀', '#fbbf24', NULL),
|
||||||
|
('650e8400-e29b-41d4-a716-446655440011', 'Meat', 'type', '🥩', '#ef4444', NULL),
|
||||||
|
('650e8400-e29b-41d4-a716-446655440012', 'Fish', 'type', '🐟', '#3b82f6', NULL),
|
||||||
|
('650e8400-e29b-41d4-a716-446655440013', 'Vegetables', 'type', '🥬', '#22c55e', NULL),
|
||||||
|
('650e8400-e29b-41d4-a716-446655440014', 'Fruits', 'type', '🍎', '#f97316', NULL),
|
||||||
|
('650e8400-e29b-41d4-a716-446655440015', 'Grains', 'type', '🌾', '#eab308', NULL),
|
||||||
|
('650e8400-e29b-41d4-a716-446655440016', 'Legumes', 'type', '🫘', '#84cc16', NULL),
|
||||||
|
('650e8400-e29b-41d4-a716-446655440017', 'Condiments', 'type', '🧂', '#ef4444', NULL),
|
||||||
|
('650e8400-e29b-41d4-a716-446655440018', 'Snacks', 'type', '🍿', '#f97316', NULL),
|
||||||
|
('650e8400-e29b-41d4-a716-446655440019', 'Beverages', 'type', '🥤', '#06b6d4', NULL),
|
||||||
|
('650e8400-e29b-41d4-a716-446655440020', 'Baking', 'type', '🧁', '#ec4899', NULL),
|
||||||
|
('650e8400-e29b-41d4-a716-446655440021', 'Spices', 'type', '🌶️', '#dc2626', NULL),
|
||||||
|
('650e8400-e29b-41d4-a716-446655440022', 'Canned', 'type', '🥫', '#71717a', NULL),
|
||||||
|
('650e8400-e29b-41d4-a716-446655440023', 'Frozen', 'type', '🧊', '#06b6d4', NULL);
|
||||||
|
|
||||||
|
-- Dietary Tags
|
||||||
|
INSERT INTO tags (id, name, category, icon, color, created_by) VALUES
|
||||||
|
('650e8400-e29b-41d4-a716-446655440030', 'Vegan', 'dietary', '🌱', '#22c55e', NULL),
|
||||||
|
('650e8400-e29b-41d4-a716-446655440031', 'Vegetarian', 'dietary', '🥕', '#84cc16', NULL),
|
||||||
|
('650e8400-e29b-41d4-a716-446655440032', 'Gluten-Free', 'dietary', '🌾', '#eab308', NULL),
|
||||||
|
('650e8400-e29b-41d4-a716-446655440033', 'Lactose-Free', 'dietary', '🥛', '#60a5fa', NULL),
|
||||||
|
('650e8400-e29b-41d4-a716-446655440034', 'Organic', 'dietary', '♻️', '#10b981', NULL),
|
||||||
|
('650e8400-e29b-41d4-a716-446655440035', 'Low-Carb', 'dietary', '🥗', '#22c55e', NULL),
|
||||||
|
('650e8400-e29b-41d4-a716-446655440036', 'Kosher', 'dietary', '✡️', '#3b82f6', NULL),
|
||||||
|
('650e8400-e29b-41d4-a716-446655440037', 'Halal', 'dietary', '☪️', '#22c55e', NULL);
|
||||||
|
|
||||||
|
-- Custom/Workflow Tags
|
||||||
|
INSERT INTO tags (id, name, category, icon, color, created_by) VALUES
|
||||||
|
('650e8400-e29b-41d4-a716-446655440040', 'Low Stock', 'custom', '⚠️', '#ef4444', NULL),
|
||||||
|
('650e8400-e29b-41d4-a716-446655440041', 'To Buy', 'custom', '🛒', '#3b82f6', NULL),
|
||||||
|
('650e8400-e29b-41d4-a716-446655440042', 'Meal Prep', 'custom', '🍱', '#8b5cf6', NULL),
|
||||||
|
('650e8400-e29b-41d4-a716-446655440043', 'Leftovers', 'custom', '♻️', '#f59e0b', NULL),
|
||||||
|
('650e8400-e29b-41d4-a716-446655440044', 'Opening Soon', 'custom', '📆', '#f97316', NULL);
|
||||||
|
|
||||||
|
COMMENT ON TABLE tags IS 'Pre-populated with 33 common tags across position, type, dietary, and workflow categories';
|
||||||