Compare commits
30 Commits
feature/is
...
60d6e03e87
| Author | SHA1 | Date | |
|---|---|---|---|
| 60d6e03e87 | |||
|
|
7209bb06df | ||
| 5b85132114 | |||
|
|
9bdbe9a420 | ||
| 01db4ef8cb | |||
|
|
e47535d0fa | ||
| 28ff53e8cd | |||
|
|
b98b3bf222 | ||
| 7a01aecb34 | |||
|
|
762ec56a3c | ||
| 91a21e274f | |||
|
|
14e5cab7bb | ||
| 229cb2cc90 | |||
|
|
d4d3d9390c | ||
| 12c5304638 | |||
|
|
080d2424c8 | ||
|
|
6b1c34ceff | ||
| 231f594004 | |||
|
|
7d35a3e7b3 | ||
| 670b2f9200 | |||
|
|
521e3f552f | ||
| 627e970986 | |||
|
|
50a0bd9417 | ||
| 097f0f9cee | |||
|
|
b1ef7e43be | ||
|
|
12bda4c08f | ||
|
|
5eb0d04377 | ||
| 5805be698b | |||
|
|
1f21032194 | ||
| f4b870f59c |
68
.dockerignore
Normal file
@@ -0,0 +1,68 @@
|
||||
# Dependencies
|
||||
node_modules
|
||||
app/node_modules
|
||||
supabase/node_modules
|
||||
|
||||
# Build outputs
|
||||
.nuxt
|
||||
app/.nuxt
|
||||
.output
|
||||
app/.output
|
||||
dist
|
||||
app/dist
|
||||
|
||||
# Logs
|
||||
*.log
|
||||
npm-debug.log*
|
||||
yarn-debug.log*
|
||||
yarn-error.log*
|
||||
pnpm-debug.log*
|
||||
lerna-debug.log*
|
||||
|
||||
# Environment files (use docker env vars instead)
|
||||
.env
|
||||
.env.local
|
||||
.env.*.local
|
||||
app/.env
|
||||
app/.env.local
|
||||
|
||||
# Editor directories and files
|
||||
.vscode/
|
||||
.idea/
|
||||
*.suo
|
||||
*.ntvs*
|
||||
*.njsproj
|
||||
*.sln
|
||||
*.sw?
|
||||
.DS_Store
|
||||
|
||||
# Git
|
||||
.git
|
||||
.gitignore
|
||||
.gitattributes
|
||||
|
||||
# CI/CD
|
||||
.github
|
||||
.gitea
|
||||
|
||||
# Documentation
|
||||
*.md
|
||||
docs/
|
||||
!README.md
|
||||
|
||||
# Tests
|
||||
test/
|
||||
tests/
|
||||
*.spec.js
|
||||
*.spec.ts
|
||||
*.test.js
|
||||
*.test.ts
|
||||
|
||||
# Temporary files
|
||||
*.tmp
|
||||
*.bak
|
||||
*.swp
|
||||
.cache
|
||||
|
||||
# Supabase (handled separately)
|
||||
supabase/
|
||||
64
.env.example
@@ -1,54 +1,20 @@
|
||||
# Pantry - Environment Variables Template
|
||||
# Copy to .env.development for local development
|
||||
# Copy to .env.production for production deployment
|
||||
# Supabase Local Development Environment
|
||||
# Copy this file to .env and adjust as needed
|
||||
|
||||
# ==============================================
|
||||
# Supabase Configuration
|
||||
# ==============================================
|
||||
# PostgreSQL
|
||||
POSTGRES_PASSWORD=postgres
|
||||
|
||||
# Supabase API URL (from Coolify service)
|
||||
SUPABASE_URL=https://your-supabase-instance.example.com
|
||||
# JWT Secret (generate with: openssl rand -base64 32)
|
||||
# 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)
|
||||
SUPABASE_ANON_KEY=your-anon-key-here
|
||||
# API Keys
|
||||
# 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)
|
||||
SUPABASE_SERVICE_ROLE_KEY=your-service-role-key-here
|
||||
SERVICE_ROLE_KEY=eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJpc3MiOiJzdXBhYmFzZS1kZW1vIiwicm9sZSI6InNlcnZpY2Vfcm9sZSIsImV4cCI6MTk4MzgxMjk5Nn0.EGIM96RAZx35lJzdJsyH-qQwv8Hdp7fsn3W0YpN81IU
|
||||
|
||||
# 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=
|
||||
# Nuxt App Configuration (also copy to app/.env)
|
||||
NUXT_PUBLIC_SUPABASE_URL=http://localhost:54321
|
||||
NUXT_PUBLIC_SUPABASE_ANON_KEY=eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJpc3MiOiJzdXBhYmFzZS1kZW1vIiwicm9sZSI6ImFub24iLCJleHAiOjE5ODM4MTI5OTZ9.CRXP1A7WOeoJeXxjNni43kdQwgnWNReilDMblYTn_I0
|
||||
|
||||
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=
|
||||
59
Dockerfile
Normal file
@@ -0,0 +1,59 @@
|
||||
# Pantry Production Dockerfile
|
||||
# Multi-stage build for optimized production image
|
||||
|
||||
# Stage 1: Build the Nuxt application
|
||||
FROM node:20-alpine AS builder
|
||||
|
||||
# Set working directory
|
||||
WORKDIR /app
|
||||
|
||||
# Copy package files
|
||||
COPY app/package*.json ./
|
||||
|
||||
# Install dependencies
|
||||
RUN npm ci --only=production && \
|
||||
npm cache clean --force
|
||||
|
||||
# Copy application source
|
||||
COPY app/ ./
|
||||
|
||||
# Build the application
|
||||
RUN npm run build
|
||||
|
||||
# Stage 2: Production runtime
|
||||
FROM node:20-alpine AS runner
|
||||
|
||||
# Install dumb-init for proper signal handling
|
||||
RUN apk add --no-cache dumb-init
|
||||
|
||||
# Create app user
|
||||
RUN addgroup -g 1001 -S nodejs && \
|
||||
adduser -S nodejs -u 1001
|
||||
|
||||
# Set working directory
|
||||
WORKDIR /app
|
||||
|
||||
# Copy built application from builder
|
||||
COPY --from=builder --chown=nodejs:nodejs /app/.output /app/.output
|
||||
COPY --from=builder --chown=nodejs:nodejs /app/package*.json ./
|
||||
|
||||
# Switch to non-root user
|
||||
USER nodejs
|
||||
|
||||
# Expose port
|
||||
EXPOSE 3000
|
||||
|
||||
# Set environment variables
|
||||
ENV NODE_ENV=production \
|
||||
HOST=0.0.0.0 \
|
||||
PORT=3000
|
||||
|
||||
# Health check
|
||||
HEALTHCHECK --interval=30s --timeout=3s --start-period=40s --retries=3 \
|
||||
CMD node -e "require('http').get('http://localhost:3000/api/health', (r) => {process.exit(r.statusCode === 200 ? 0 : 1)})"
|
||||
|
||||
# Use dumb-init to handle signals properly
|
||||
ENTRYPOINT ["dumb-init", "--"]
|
||||
|
||||
# Start the application
|
||||
CMD ["node", ".output/server/index.mjs"]
|
||||
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
|
||||
- 🌐 **Open Food Facts** — Auto-fill product data from barcodes
|
||||
|
||||
## 🚀 Quick Start
|
||||
## 🚀 Quick Start (Local Development)
|
||||
|
||||
```bash
|
||||
# Clone
|
||||
# Clone repository
|
||||
git clone https://gitea.jeanlucmakiola.de/pantry-app/pantry.git
|
||||
cd pantry
|
||||
|
||||
# Start services (Docker Compose)
|
||||
docker-compose up -d
|
||||
|
||||
# Access at http://localhost:3000
|
||||
# One-command startup
|
||||
./dev.sh
|
||||
```
|
||||
|
||||
**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
|
||||
|
||||
- [**Project Plan**](docs/PROJECT_PLAN.md) — Vision, roadmap, phases
|
||||
- [**Architecture**](docs/ARCHITECTURE.md) — Tech stack, data model, design decisions
|
||||
- [**Database Schema**](docs/DATABASE.md) — Tables, relationships, RLS policies
|
||||
- [**API Reference**](docs/API.md) — Endpoints, Supabase functions
|
||||
- [**Development Guide**](docs/DEVELOPMENT.md) — Setup, workflow, conventions
|
||||
- [**Deployment**](docs/DEPLOYMENT.md) — Docker, Coolify, production setup
|
||||
- **[Getting Started](docs/development/getting-started.md)** — First-time setup (5 minutes)
|
||||
- **[Local Setup Guide](docs/development/local-setup.md)** — Detailed Docker Compose setup
|
||||
- **[Project Plan](docs/PROJECT_PLAN.md)** — Vision, roadmap, MVP phases
|
||||
- **[Architecture](docs/architecture/overview.md)** — Tech stack, design decisions
|
||||
- **[Database Schema](docs/architecture/database.md)** — Tables, RLS policies, migrations
|
||||
- **[Development Workflow](docs/development/workflow.md)** — Git flow, conventions
|
||||
- **[Full Documentation Index](docs/README.md)** — Complete docs navigation
|
||||
|
||||
## 🛠️ Tech Stack
|
||||
|
||||
@@ -73,18 +84,32 @@ pantry/
|
||||
3. **Extendable** — Clean architecture for future features
|
||||
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.
|
||||
|
||||
- [ ] Foundation (Nuxt + Supabase + Auth)
|
||||
- [ ] Core inventory (CRUD, tags, units)
|
||||
- [ ] Barcode scanning (PWA camera + Open Food Facts)
|
||||
- [ ] Mobile polish (PWA, offline)
|
||||
- [ ] Docker deployment
|
||||
|
||||
## 🤝 Contributing
|
||||
|
||||
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
|
||||
@@ -1,5 +1,9 @@
|
||||
<template>
|
||||
<NuxtLayout>
|
||||
<NuxtPage />
|
||||
</NuxtLayout>
|
||||
<div>
|
||||
<OfflineBanner />
|
||||
<InstallPrompt />
|
||||
<NuxtLayout>
|
||||
<NuxtPage />
|
||||
</NuxtLayout>
|
||||
</div>
|
||||
</template>
|
||||
|
||||
@@ -8,6 +8,7 @@
|
||||
"@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",
|
||||
@@ -1076,6 +1077,8 @@
|
||||
|
||||
"hookable": ["hookable@5.5.3", "", {}, "sha512-Yc+BQe8SvoXH1643Qez1zqLRmbA5rCL+sSmk6TVos0LWVfNIB7PGncdlId77WzLGSIB5KaWgTaNTs2lNVEI6VQ=="],
|
||||
|
||||
"html5-qrcode": ["html5-qrcode@2.3.8", "", {}, "sha512-jsr4vafJhwoLVEDW3n1KvPnCCXWaQfRng0/EEYk1vNcQGcG/htAdhJX0be8YyqMoSz7+hZvOZSTAepsabiuhiQ=="],
|
||||
|
||||
"http-assert": ["http-assert@1.5.0", "", { "dependencies": { "deep-equal": "~1.0.1", "http-errors": "~1.8.0" } }, "sha512-uPpH7OKX4H25hBmU6G1jWNaqJGpTXxey+YOUizJUAgu0AjLUeC8D73hTrhvDS5D+GJN1DN1+hhc/eF/wpxtp0w=="],
|
||||
|
||||
"http-errors": ["http-errors@2.0.1", "", { "dependencies": { "depd": "~2.0.0", "inherits": "~2.0.4", "setprototypeof": "~1.2.0", "statuses": "~2.0.2", "toidentifier": "~1.0.1" } }, "sha512-4FbRdAX+bSdmo4AUFuS0WNiPz8NgFt+r8ThgNWmlrjQjt1Q7ZR9+zTlce2859x4KSXrwIsaeTqDoKQmtP8pLmQ=="],
|
||||
|
||||
104
app/components/InstallPrompt.vue
Normal file
@@ -0,0 +1,104 @@
|
||||
<template>
|
||||
<Transition
|
||||
enter-active-class="transition ease-out duration-300"
|
||||
enter-from-class="opacity-0 translate-y-2"
|
||||
enter-to-class="opacity-100 translate-y-0"
|
||||
leave-active-class="transition ease-in duration-200"
|
||||
leave-from-class="opacity-100 translate-y-0"
|
||||
leave-to-class="opacity-0 translate-y-2"
|
||||
>
|
||||
<div
|
||||
v-if="showPrompt"
|
||||
class="fixed bottom-4 left-4 right-4 md:left-auto md:right-4 md:max-w-md z-50"
|
||||
>
|
||||
<div class="bg-white dark:bg-gray-800 rounded-xl shadow-2xl border border-gray-200 dark:border-gray-700 p-4">
|
||||
<div class="flex items-start gap-4">
|
||||
<!-- App Icon -->
|
||||
<div class="flex-shrink-0">
|
||||
<img
|
||||
src="/icon-192x192.png"
|
||||
alt="Pantry icon"
|
||||
class="w-16 h-16 rounded-lg shadow-md"
|
||||
/>
|
||||
</div>
|
||||
|
||||
<!-- Content -->
|
||||
<div class="flex-1 min-w-0">
|
||||
<h3 class="text-lg font-bold text-gray-900 dark:text-white mb-1">
|
||||
Install Pantry
|
||||
</h3>
|
||||
<p class="text-sm text-gray-600 dark:text-gray-400 mb-3">
|
||||
Install this app for quick access and offline use
|
||||
</p>
|
||||
|
||||
<!-- Actions -->
|
||||
<div class="flex gap-2">
|
||||
<UButton
|
||||
color="emerald"
|
||||
size="sm"
|
||||
@click="install"
|
||||
:loading="installing"
|
||||
>
|
||||
<template #leading>
|
||||
<UIcon name="i-heroicons-arrow-down-tray" />
|
||||
</template>
|
||||
Install
|
||||
</UButton>
|
||||
|
||||
<UButton
|
||||
color="gray"
|
||||
variant="ghost"
|
||||
size="sm"
|
||||
@click="dismiss"
|
||||
>
|
||||
Not now
|
||||
</UButton>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- Close button -->
|
||||
<button
|
||||
@click="dismiss"
|
||||
class="flex-shrink-0 text-gray-400 hover:text-gray-600 dark:hover:text-gray-300 transition"
|
||||
>
|
||||
<UIcon name="i-heroicons-x-mark" class="w-5 h-5" />
|
||||
</button>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</Transition>
|
||||
</template>
|
||||
|
||||
<script setup lang="ts">
|
||||
const { canInstall, promptInstall, dismissInstall, shouldShowPrompt } = usePWAInstall()
|
||||
const installing = ref(false)
|
||||
const showPrompt = ref(false)
|
||||
|
||||
// Show prompt after a delay if conditions are met
|
||||
onMounted(() => {
|
||||
setTimeout(() => {
|
||||
if (shouldShowPrompt()) {
|
||||
showPrompt.value = true
|
||||
}
|
||||
}, 3000) // Wait 3 seconds after page load
|
||||
})
|
||||
|
||||
async function install() {
|
||||
installing.value = true
|
||||
try {
|
||||
const { outcome } = await promptInstall()
|
||||
if (outcome === 'accepted') {
|
||||
showPrompt.value = false
|
||||
}
|
||||
} catch (error) {
|
||||
console.error('Install failed:', error)
|
||||
} finally {
|
||||
installing.value = false
|
||||
}
|
||||
}
|
||||
|
||||
function dismiss() {
|
||||
dismissInstall()
|
||||
showPrompt.value = false
|
||||
}
|
||||
</script>
|
||||
58
app/components/OfflineBanner.vue
Normal file
@@ -0,0 +1,58 @@
|
||||
<template>
|
||||
<Transition
|
||||
enter-active-class="transition ease-out duration-300"
|
||||
enter-from-class="opacity-0 -translate-y-2"
|
||||
enter-to-class="opacity-100 translate-y-0"
|
||||
leave-active-class="transition ease-in duration-200"
|
||||
leave-from-class="opacity-100 translate-y-0"
|
||||
leave-to-class="opacity-0 -translate-y-2"
|
||||
>
|
||||
<div
|
||||
v-if="!isOnline"
|
||||
class="fixed top-0 left-0 right-0 z-50 bg-amber-500 text-white px-4 py-2 text-center shadow-lg"
|
||||
>
|
||||
<div class="flex items-center justify-center gap-2">
|
||||
<UIcon name="i-heroicons-wifi-slash" class="w-5 h-5" />
|
||||
<span class="font-medium">
|
||||
You're currently offline. Changes will sync when connection is restored.
|
||||
</span>
|
||||
</div>
|
||||
</div>
|
||||
</Transition>
|
||||
|
||||
<Transition
|
||||
enter-active-class="transition ease-out duration-300"
|
||||
enter-from-class="opacity-0 -translate-y-2"
|
||||
enter-to-class="opacity-100 translate-y-0"
|
||||
leave-active-class="transition ease-in duration-200"
|
||||
leave-from-class="opacity-100 translate-y-0"
|
||||
leave-to-class="opacity-0 -translate-y-2"
|
||||
>
|
||||
<div
|
||||
v-if="isOnline && wasOffline && showReconnected"
|
||||
class="fixed top-0 left-0 right-0 z-50 bg-emerald-500 text-white px-4 py-2 text-center shadow-lg"
|
||||
>
|
||||
<div class="flex items-center justify-center gap-2">
|
||||
<UIcon name="i-heroicons-wifi" class="w-5 h-5" />
|
||||
<span class="font-medium">
|
||||
Back online! Syncing your changes...
|
||||
</span>
|
||||
</div>
|
||||
</div>
|
||||
</Transition>
|
||||
</template>
|
||||
|
||||
<script setup lang="ts">
|
||||
const { isOnline, wasOffline } = useOnlineStatus()
|
||||
const showReconnected = ref(false)
|
||||
|
||||
// Show "back online" message for 3 seconds
|
||||
watch(isOnline, (online) => {
|
||||
if (online && wasOffline.value) {
|
||||
showReconnected.value = true
|
||||
setTimeout(() => {
|
||||
showReconnected.value = false
|
||||
}, 3000)
|
||||
}
|
||||
})
|
||||
</script>
|
||||
190
app/components/Settings/AppSettings.vue
Normal file
@@ -0,0 +1,190 @@
|
||||
<template>
|
||||
<UCard class="mt-4">
|
||||
<div class="space-y-6">
|
||||
<div>
|
||||
<h3 class="text-lg font-semibold mb-4">App Installation</h3>
|
||||
|
||||
<!-- Already installed -->
|
||||
<div v-if="isInstalled" class="flex items-start gap-3 p-4 bg-emerald-50 dark:bg-emerald-900/20 rounded-lg">
|
||||
<UIcon name="i-heroicons-check-circle" class="w-6 h-6 text-emerald-600 dark:text-emerald-400 flex-shrink-0 mt-0.5" />
|
||||
<div>
|
||||
<p class="font-medium text-emerald-900 dark:text-emerald-100">
|
||||
App is installed
|
||||
</p>
|
||||
<p class="text-sm text-emerald-700 dark:text-emerald-300 mt-1">
|
||||
You're using Pantry as an installed app with offline support.
|
||||
</p>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- Can install -->
|
||||
<div v-else-if="canInstall" class="space-y-3">
|
||||
<p class="text-gray-600 dark:text-gray-400">
|
||||
Install Pantry as an app for quick access and offline use.
|
||||
</p>
|
||||
<UButton
|
||||
color="emerald"
|
||||
size="lg"
|
||||
@click="install"
|
||||
:loading="installing"
|
||||
block
|
||||
>
|
||||
<template #leading>
|
||||
<UIcon name="i-heroicons-arrow-down-tray" />
|
||||
</template>
|
||||
Install App
|
||||
</UButton>
|
||||
</div>
|
||||
|
||||
<!-- Not supported or already running -->
|
||||
<div v-else class="space-y-4">
|
||||
<div v-if="isStandalone" class="flex items-start gap-3 p-4 bg-blue-50 dark:bg-blue-900/20 rounded-lg">
|
||||
<UIcon name="i-heroicons-information-circle" class="w-6 h-6 text-blue-600 dark:text-blue-400 flex-shrink-0 mt-0.5" />
|
||||
<div>
|
||||
<p class="font-medium text-blue-900 dark:text-blue-100">
|
||||
Running as standalone app
|
||||
</p>
|
||||
<p class="text-sm text-blue-700 dark:text-blue-300 mt-1">
|
||||
You're already using Pantry in standalone mode.
|
||||
</p>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div v-else>
|
||||
<p class="text-gray-600 dark:text-gray-400 mb-4">
|
||||
Installation is not available yet. Try one of these options:
|
||||
</p>
|
||||
|
||||
<!-- iOS Instructions -->
|
||||
<UCard class="mb-4">
|
||||
<div class="flex items-start gap-3">
|
||||
<UIcon name="i-heroicons-device-phone-mobile" class="w-6 h-6 text-gray-400 flex-shrink-0 mt-0.5" />
|
||||
<div class="flex-1">
|
||||
<h4 class="font-semibold mb-2">iOS (Safari)</h4>
|
||||
<ol class="text-sm text-gray-600 dark:text-gray-400 space-y-1 list-decimal list-inside">
|
||||
<li>Tap the Share button</li>
|
||||
<li>Scroll down and tap "Add to Home Screen"</li>
|
||||
<li>Tap "Add" to confirm</li>
|
||||
</ol>
|
||||
</div>
|
||||
</div>
|
||||
</UCard>
|
||||
|
||||
<!-- Android Instructions -->
|
||||
<UCard>
|
||||
<div class="flex items-start gap-3">
|
||||
<UIcon name="i-heroicons-device-phone-mobile" class="w-6 h-6 text-gray-400 flex-shrink-0 mt-0.5" />
|
||||
<div class="flex-1">
|
||||
<h4 class="font-semibold mb-2">Android (Chrome)</h4>
|
||||
<ol class="text-sm text-gray-600 dark:text-gray-400 space-y-1 list-decimal list-inside">
|
||||
<li>Tap the menu (⋮) button</li>
|
||||
<li>Tap "Add to Home screen" or "Install app"</li>
|
||||
<li>Tap "Install" to confirm</li>
|
||||
</ol>
|
||||
</div>
|
||||
</div>
|
||||
</UCard>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- PWA Features -->
|
||||
<div>
|
||||
<h3 class="text-lg font-semibold mb-3">App Features</h3>
|
||||
<div class="space-y-2">
|
||||
<div class="flex items-center gap-2 text-gray-700 dark:text-gray-300">
|
||||
<UIcon name="i-heroicons-check" class="w-5 h-5 text-emerald-500" />
|
||||
<span>Works offline with cached data</span>
|
||||
</div>
|
||||
<div class="flex items-center gap-2 text-gray-700 dark:text-gray-300">
|
||||
<UIcon name="i-heroicons-check" class="w-5 h-5 text-emerald-500" />
|
||||
<span>Quick access from home screen</span>
|
||||
</div>
|
||||
<div class="flex items-center gap-2 text-gray-700 dark:text-gray-300">
|
||||
<UIcon name="i-heroicons-check" class="w-5 h-5 text-emerald-500" />
|
||||
<span>Full-screen experience</span>
|
||||
</div>
|
||||
<div class="flex items-center gap-2 text-gray-700 dark:text-gray-300">
|
||||
<UIcon name="i-heroicons-check" class="w-5 h-5 text-emerald-500" />
|
||||
<span>Automatic updates</span>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- Storage Info -->
|
||||
<div v-if="storageInfo">
|
||||
<h3 class="text-lg font-semibold mb-3">Storage Usage</h3>
|
||||
<div class="space-y-2">
|
||||
<div class="flex justify-between text-sm">
|
||||
<span class="text-gray-600 dark:text-gray-400">Used</span>
|
||||
<span class="font-medium">{{ formatBytes(storageInfo.usage) }}</span>
|
||||
</div>
|
||||
<div class="flex justify-between text-sm">
|
||||
<span class="text-gray-600 dark:text-gray-400">Available</span>
|
||||
<span class="font-medium">{{ formatBytes(storageInfo.quota) }}</span>
|
||||
</div>
|
||||
<div class="w-full bg-gray-200 dark:bg-gray-700 rounded-full h-2">
|
||||
<div
|
||||
class="bg-emerald-500 h-2 rounded-full transition-all"
|
||||
:style="{ width: storagePercent + '%' }"
|
||||
/>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</UCard>
|
||||
</template>
|
||||
|
||||
<script setup lang="ts">
|
||||
const { canInstall, isInstalled, promptInstall } = usePWAInstall()
|
||||
const installing = ref(false)
|
||||
const storageInfo = ref<{ usage: number; quota: number } | null>(null)
|
||||
|
||||
const isStandalone = computed(() => {
|
||||
if (process.client) {
|
||||
return window.matchMedia('(display-mode: standalone)').matches
|
||||
}
|
||||
return false
|
||||
})
|
||||
|
||||
const storagePercent = computed(() => {
|
||||
if (!storageInfo.value) return 0
|
||||
return Math.round((storageInfo.value.usage / storageInfo.value.quota) * 100)
|
||||
})
|
||||
|
||||
async function install() {
|
||||
installing.value = true
|
||||
try {
|
||||
await promptInstall()
|
||||
} catch (error) {
|
||||
console.error('Install failed:', error)
|
||||
} finally {
|
||||
installing.value = false
|
||||
}
|
||||
}
|
||||
|
||||
function formatBytes(bytes: number): string {
|
||||
if (bytes === 0) return '0 Bytes'
|
||||
const k = 1024
|
||||
const sizes = ['Bytes', 'KB', 'MB', 'GB']
|
||||
const i = Math.floor(Math.log(bytes) / Math.log(k))
|
||||
return Math.round((bytes / Math.pow(k, i)) * 100) / 100 + ' ' + sizes[i]
|
||||
}
|
||||
|
||||
onMounted(async () => {
|
||||
// Get storage info
|
||||
if ('storage' in navigator && 'estimate' in navigator.storage) {
|
||||
try {
|
||||
const estimate = await navigator.storage.estimate()
|
||||
if (estimate.usage !== undefined && estimate.quota !== undefined) {
|
||||
storageInfo.value = {
|
||||
usage: estimate.usage,
|
||||
quota: estimate.quota
|
||||
}
|
||||
}
|
||||
} catch (error) {
|
||||
console.error('Failed to get storage estimate:', error)
|
||||
}
|
||||
}
|
||||
})
|
||||
</script>
|
||||
@@ -68,37 +68,7 @@
|
||||
|
||||
<!-- Tags -->
|
||||
<UFormGroup label="Tags" hint="Optional">
|
||||
<div class="space-y-2">
|
||||
<!-- Selected Tags -->
|
||||
<div v-if="selectedTags.length > 0" class="flex flex-wrap gap-1 mb-2">
|
||||
<UBadge
|
||||
v-for="tag in selectedTags"
|
||||
:key="tag.id"
|
||||
:style="{ backgroundColor: tag.color }"
|
||||
class="text-white cursor-pointer"
|
||||
@click="removeTag(tag.id)"
|
||||
>
|
||||
{{ tag.icon }} {{ tag.name }} ✕
|
||||
</UBadge>
|
||||
</div>
|
||||
|
||||
<!-- Tag Selection by Category -->
|
||||
<div v-for="category in tagCategories" :key="category.name" class="space-y-1">
|
||||
<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="isTagSelected(tag.id) ? 'primary' : 'gray'"
|
||||
:variant="isTagSelected(tag.id) ? 'solid' : 'outline'"
|
||||
@click="toggleTag(tag)"
|
||||
>
|
||||
{{ tag.icon }} {{ tag.name }}
|
||||
</UButton>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
<TagsTagPicker v-model="selectedTags" />
|
||||
</UFormGroup>
|
||||
|
||||
<!-- Submit -->
|
||||
@@ -129,7 +99,16 @@
|
||||
<script setup lang="ts">
|
||||
const { addInventoryItem, addItemTags } = useInventory()
|
||||
const { getUnits } = useUnits()
|
||||
const { getTags } = useTags()
|
||||
|
||||
const props = defineProps<{
|
||||
initialData?: {
|
||||
barcode?: string
|
||||
name?: string
|
||||
brand?: string
|
||||
image_url?: string
|
||||
quantity?: string
|
||||
}
|
||||
}>()
|
||||
|
||||
const emit = defineEmits<{
|
||||
close: []
|
||||
@@ -148,24 +127,52 @@ const form = reactive({
|
||||
const submitting = ref(false)
|
||||
const selectedTags = ref<any[]>([])
|
||||
|
||||
// Load units and tags
|
||||
// Load units
|
||||
const units = ref<any[]>([])
|
||||
const tags = ref<any[]>([])
|
||||
|
||||
onMounted(async () => {
|
||||
const [unitsResult, tagsResult] = await Promise.all([
|
||||
getUnits(),
|
||||
getTags()
|
||||
])
|
||||
|
||||
const unitsResult = await getUnits()
|
||||
units.value = unitsResult.data || []
|
||||
tags.value = tagsResult.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
|
||||
@@ -187,39 +194,6 @@ const unitOptions = computed(() => {
|
||||
])
|
||||
})
|
||||
|
||||
// Tag categories for display
|
||||
const tagCategories = computed(() => {
|
||||
const categories: Record<string, any[]> = {}
|
||||
|
||||
for (const tag of tags.value) {
|
||||
const cat = tag.category
|
||||
if (!categories[cat]) categories[cat] = []
|
||||
categories[cat].push(tag)
|
||||
}
|
||||
|
||||
return Object.entries(categories).map(([name, tags]) => ({
|
||||
name,
|
||||
tags
|
||||
}))
|
||||
})
|
||||
|
||||
// Tag selection helpers
|
||||
const isTagSelected = (tagId: string) => {
|
||||
return selectedTags.value.some(t => t.id === tagId)
|
||||
}
|
||||
|
||||
const toggleTag = (tag: any) => {
|
||||
if (isTagSelected(tag.id)) {
|
||||
removeTag(tag.id)
|
||||
} else {
|
||||
selectedTags.value.push(tag)
|
||||
}
|
||||
}
|
||||
|
||||
const removeTag = (tagId: string) => {
|
||||
selectedTags.value = selectedTags.value.filter(t => t.id !== tagId)
|
||||
}
|
||||
|
||||
// Validation
|
||||
const isValid = computed(() => {
|
||||
return form.name.trim().length > 0 && form.quantity > 0 && form.unit_id
|
||||
|
||||
@@ -50,15 +50,12 @@
|
||||
|
||||
<!-- Tags -->
|
||||
<div v-if="item.tags && item.tags.length > 0" class="flex flex-wrap gap-1">
|
||||
<UBadge
|
||||
v-for="tagItem in item.tags.slice(0, 3)"
|
||||
<TagsTagBadge
|
||||
v-for="tagItem in item.tags.slice(0, 3)"
|
||||
:key="tagItem.tag.id"
|
||||
:style="{ backgroundColor: tagItem.tag.color }"
|
||||
size="xs"
|
||||
class="text-white"
|
||||
>
|
||||
{{ tagItem.tag.icon }} {{ tagItem.tag.name }}
|
||||
</UBadge>
|
||||
:tag="tagItem.tag"
|
||||
size="sm"
|
||||
/>
|
||||
<UBadge v-if="item.tags.length > 3" size="xs" color="gray">
|
||||
+{{ item.tags.length - 3 }}
|
||||
</UBadge>
|
||||
|
||||
@@ -43,7 +43,7 @@
|
||||
<!-- 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 items"
|
||||
v-for="item in filteredItems"
|
||||
:key="item.id"
|
||||
:item="item"
|
||||
@edit="$emit('edit-item', item)"
|
||||
@@ -59,6 +59,7 @@ const { getInventory, deleteInventoryItem, updateQuantity } = useInventory()
|
||||
|
||||
const props = defineProps<{
|
||||
refresh?: boolean
|
||||
tagFilters?: string[]
|
||||
}>()
|
||||
|
||||
const emit = defineEmits<{
|
||||
@@ -86,6 +87,21 @@ const loadInventory = async () => {
|
||||
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
|
||||
|
||||
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>
|
||||
47
app/composables/useOnlineStatus.ts
Normal file
@@ -0,0 +1,47 @@
|
||||
/**
|
||||
* Composable to track online/offline status
|
||||
*
|
||||
* Usage:
|
||||
* const { isOnline, wasOffline } = useOnlineStatus()
|
||||
*
|
||||
* watch(isOnline, (online) => {
|
||||
* if (online && wasOffline.value) {
|
||||
* // User came back online, sync data
|
||||
* }
|
||||
* })
|
||||
*/
|
||||
export function useOnlineStatus() {
|
||||
const isOnline = ref(true)
|
||||
const wasOffline = ref(false)
|
||||
|
||||
if (process.client) {
|
||||
// Initial state
|
||||
isOnline.value = navigator.onLine
|
||||
|
||||
// Listen for online/offline events
|
||||
const updateOnlineStatus = () => {
|
||||
const online = navigator.onLine
|
||||
|
||||
if (!online && isOnline.value) {
|
||||
// Just went offline
|
||||
wasOffline.value = true
|
||||
}
|
||||
|
||||
isOnline.value = online
|
||||
}
|
||||
|
||||
window.addEventListener('online', updateOnlineStatus)
|
||||
window.addEventListener('offline', updateOnlineStatus)
|
||||
|
||||
// Cleanup on unmount
|
||||
onUnmounted(() => {
|
||||
window.removeEventListener('online', updateOnlineStatus)
|
||||
window.removeEventListener('offline', updateOnlineStatus)
|
||||
})
|
||||
}
|
||||
|
||||
return {
|
||||
isOnline: readonly(isOnline),
|
||||
wasOffline: readonly(wasOffline)
|
||||
}
|
||||
}
|
||||
93
app/composables/usePWAInstall.ts
Normal file
@@ -0,0 +1,93 @@
|
||||
/**
|
||||
* Composable to handle PWA installation
|
||||
*
|
||||
* Usage:
|
||||
* const { canInstall, isInstalled, promptInstall, dismissInstall } = usePWAInstall()
|
||||
*/
|
||||
export function usePWAInstall() {
|
||||
const canInstall = ref(false)
|
||||
const isInstalled = ref(false)
|
||||
const deferredPrompt = ref<any>(null)
|
||||
|
||||
if (process.client) {
|
||||
// Check if already installed
|
||||
if (window.matchMedia('(display-mode: standalone)').matches) {
|
||||
isInstalled.value = true
|
||||
}
|
||||
|
||||
// Listen for beforeinstallprompt event
|
||||
window.addEventListener('beforeinstallprompt', (e) => {
|
||||
// Prevent the mini-infobar from appearing on mobile
|
||||
e.preventDefault()
|
||||
|
||||
// Stash the event so it can be triggered later
|
||||
deferredPrompt.value = e
|
||||
canInstall.value = true
|
||||
})
|
||||
|
||||
// Listen for appinstalled event
|
||||
window.addEventListener('appinstalled', () => {
|
||||
isInstalled.value = true
|
||||
canInstall.value = false
|
||||
deferredPrompt.value = null
|
||||
})
|
||||
}
|
||||
|
||||
async function promptInstall() {
|
||||
if (!deferredPrompt.value) {
|
||||
return { outcome: 'not-available' }
|
||||
}
|
||||
|
||||
// Show the install prompt
|
||||
deferredPrompt.value.prompt()
|
||||
|
||||
// Wait for the user to respond to the prompt
|
||||
const { outcome } = await deferredPrompt.value.userChoice
|
||||
|
||||
// Clear the deferredPrompt
|
||||
deferredPrompt.value = null
|
||||
|
||||
if (outcome === 'accepted') {
|
||||
canInstall.value = false
|
||||
}
|
||||
|
||||
return { outcome }
|
||||
}
|
||||
|
||||
function dismissInstall() {
|
||||
canInstall.value = false
|
||||
deferredPrompt.value = null
|
||||
|
||||
// Remember dismissal for 7 days
|
||||
if (process.client) {
|
||||
localStorage.setItem('pwa-install-dismissed', Date.now().toString())
|
||||
}
|
||||
}
|
||||
|
||||
function shouldShowPrompt() {
|
||||
if (!canInstall.value || isInstalled.value) {
|
||||
return false
|
||||
}
|
||||
|
||||
if (process.client) {
|
||||
const dismissed = localStorage.getItem('pwa-install-dismissed')
|
||||
if (dismissed) {
|
||||
const dismissedTime = parseInt(dismissed)
|
||||
const sevenDays = 7 * 24 * 60 * 60 * 1000
|
||||
if (Date.now() - dismissedTime < sevenDays) {
|
||||
return false
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
return true
|
||||
}
|
||||
|
||||
return {
|
||||
canInstall: readonly(canInstall),
|
||||
isInstalled: readonly(isInstalled),
|
||||
promptInstall,
|
||||
dismissInstall,
|
||||
shouldShowPrompt
|
||||
}
|
||||
}
|
||||
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)
|
||||
}
|
||||
}
|
||||
@@ -37,8 +37,50 @@ export const useTags = () => {
|
||||
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
|
||||
getTagsByCategory,
|
||||
createTag,
|
||||
deleteTag
|
||||
}
|
||||
}
|
||||
|
||||
@@ -5,7 +5,8 @@ export default defineNuxtConfig({
|
||||
|
||||
modules: [
|
||||
'@nuxt/ui',
|
||||
'@nuxt/fonts'
|
||||
'@nuxt/fonts',
|
||||
'@vite-pwa/nuxt'
|
||||
],
|
||||
|
||||
runtimeConfig: {
|
||||
@@ -17,5 +18,166 @@ export default defineNuxtConfig({
|
||||
|
||||
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: '/offline',
|
||||
navigateFallbackDenylist: [/^\/api\//],
|
||||
globPatterns: ['**/*.{js,css,html,png,svg,ico,woff,woff2}'],
|
||||
cleanupOutdatedCaches: true,
|
||||
skipWaiting: true,
|
||||
clientsClaim: true,
|
||||
runtimeCaching: [
|
||||
// Supabase API - Network first with fallback
|
||||
{
|
||||
urlPattern: /^https:\/\/.*\.supabase\.co\/rest\/.*/i,
|
||||
handler: 'NetworkFirst',
|
||||
options: {
|
||||
cacheName: 'supabase-rest-api',
|
||||
networkTimeoutSeconds: 10,
|
||||
expiration: {
|
||||
maxEntries: 50,
|
||||
maxAgeSeconds: 60 * 60 // 1 hour
|
||||
},
|
||||
cacheableResponse: {
|
||||
statuses: [0, 200]
|
||||
}
|
||||
}
|
||||
},
|
||||
// Supabase Auth - Network only (don't cache auth)
|
||||
{
|
||||
urlPattern: /^https:\/\/.*\.supabase\.co\/auth\/.*/i,
|
||||
handler: 'NetworkOnly'
|
||||
},
|
||||
// Supabase Storage - Cache first for images
|
||||
{
|
||||
urlPattern: /^https:\/\/.*\.supabase\.co\/storage\/.*/i,
|
||||
handler: 'CacheFirst',
|
||||
options: {
|
||||
cacheName: 'supabase-storage',
|
||||
expiration: {
|
||||
maxEntries: 100,
|
||||
maxAgeSeconds: 60 * 60 * 24 * 7 // 1 week
|
||||
},
|
||||
cacheableResponse: {
|
||||
statuses: [0, 200]
|
||||
}
|
||||
}
|
||||
},
|
||||
// Open Food Facts API - Cache first with network fallback
|
||||
{
|
||||
urlPattern: /^https:\/\/world\.openfoodfacts\.org\/.*/i,
|
||||
handler: 'CacheFirst',
|
||||
options: {
|
||||
cacheName: 'openfoodfacts-api',
|
||||
expiration: {
|
||||
maxEntries: 200,
|
||||
maxAgeSeconds: 60 * 60 * 24 * 30 // 30 days
|
||||
},
|
||||
cacheableResponse: {
|
||||
statuses: [0, 200]
|
||||
}
|
||||
}
|
||||
},
|
||||
// External images - Cache first
|
||||
{
|
||||
urlPattern: /^https:\/\/images\.openfoodfacts\.org\/.*/i,
|
||||
handler: 'CacheFirst',
|
||||
options: {
|
||||
cacheName: 'product-images',
|
||||
expiration: {
|
||||
maxEntries: 100,
|
||||
maxAgeSeconds: 60 * 60 * 24 * 30 // 30 days
|
||||
},
|
||||
cacheableResponse: {
|
||||
statuses: [0, 200]
|
||||
}
|
||||
}
|
||||
},
|
||||
// Google Fonts - Cache first
|
||||
{
|
||||
urlPattern: /^https:\/\/fonts\.googleapis\.com\/.*/i,
|
||||
handler: 'CacheFirst',
|
||||
options: {
|
||||
cacheName: 'google-fonts-stylesheets',
|
||||
expiration: {
|
||||
maxEntries: 20,
|
||||
maxAgeSeconds: 60 * 60 * 24 * 365 // 1 year
|
||||
}
|
||||
}
|
||||
},
|
||||
{
|
||||
urlPattern: /^https:\/\/fonts\.gstatic\.com\/.*/i,
|
||||
handler: 'CacheFirst',
|
||||
options: {
|
||||
cacheName: 'google-fonts-webfonts',
|
||||
expiration: {
|
||||
maxEntries: 30,
|
||||
maxAgeSeconds: 60 * 60 * 24 * 365 // 1 year
|
||||
}
|
||||
}
|
||||
}
|
||||
]
|
||||
},
|
||||
devOptions: {
|
||||
enabled: true,
|
||||
type: 'module'
|
||||
}
|
||||
}
|
||||
})
|
||||
|
||||
18031
app/package-lock.json
generated
Normal file
@@ -7,17 +7,22 @@
|
||||
"dev": "nuxt dev",
|
||||
"generate": "nuxt generate",
|
||||
"preview": "nuxt preview",
|
||||
"postinstall": "nuxt prepare"
|
||||
"postinstall": "nuxt prepare",
|
||||
"generate:icons": "node scripts/generate-icons.js && node scripts/generate-screenshots.js",
|
||||
"verify:pwa": "node scripts/verify-pwa.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"
|
||||
"@nuxtjs/tailwindcss": "^6.14.0",
|
||||
"@vite-pwa/nuxt": "^1.1.1",
|
||||
"sharp": "^0.34.5"
|
||||
}
|
||||
}
|
||||
|
||||
@@ -21,14 +21,29 @@
|
||||
>
|
||||
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
|
||||
@close="showAddForm = false"
|
||||
:initial-data="prefilledData"
|
||||
@close="handleCloseAddForm"
|
||||
@added="handleItemAdded"
|
||||
/>
|
||||
</div>
|
||||
@@ -45,6 +60,7 @@
|
||||
<InventoryList
|
||||
ref="inventoryListRef"
|
||||
:refresh="refreshKey"
|
||||
:tag-filters="selectedTagFilters"
|
||||
@add-item="showAddForm = true"
|
||||
@edit-item="editingItem = $event"
|
||||
/>
|
||||
@@ -56,13 +72,44 @@ 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()
|
||||
}
|
||||
|
||||
69
app/pages/offline.vue
Normal file
@@ -0,0 +1,69 @@
|
||||
<template>
|
||||
<div class="flex flex-col items-center justify-center min-h-screen p-8 text-center">
|
||||
<UIcon name="i-heroicons-wifi-slash" class="w-24 h-24 text-gray-400 mb-6" />
|
||||
|
||||
<h1 class="text-3xl font-bold text-gray-900 mb-4">
|
||||
You're Offline
|
||||
</h1>
|
||||
|
||||
<p class="text-gray-600 mb-8 max-w-md">
|
||||
No internet connection detected. Some features may be limited, but you can still:
|
||||
</p>
|
||||
|
||||
<div class="space-y-3 mb-8 text-left max-w-md">
|
||||
<div class="flex items-start gap-3">
|
||||
<UIcon name="i-heroicons-check-circle" class="w-6 h-6 text-emerald-500 mt-0.5" />
|
||||
<span class="text-gray-700">View cached inventory items</span>
|
||||
</div>
|
||||
<div class="flex items-start gap-3">
|
||||
<UIcon name="i-heroicons-check-circle" class="w-6 h-6 text-emerald-500 mt-0.5" />
|
||||
<span class="text-gray-700">Scan barcodes (will sync when online)</span>
|
||||
</div>
|
||||
<div class="flex items-start gap-3">
|
||||
<UIcon name="i-heroicons-check-circle" class="w-6 h-6 text-emerald-500 mt-0.5" />
|
||||
<span class="text-gray-700">Browse previously loaded data</span>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<UButton
|
||||
color="emerald"
|
||||
size="lg"
|
||||
@click="retry"
|
||||
:loading="retrying"
|
||||
>
|
||||
<template #leading>
|
||||
<UIcon name="i-heroicons-arrow-path" />
|
||||
</template>
|
||||
Try Again
|
||||
</UButton>
|
||||
</div>
|
||||
</template>
|
||||
|
||||
<script setup lang="ts">
|
||||
const retrying = ref(false)
|
||||
|
||||
async function retry() {
|
||||
retrying.value = true
|
||||
|
||||
try {
|
||||
// Test if we're back online
|
||||
const response = await fetch('/api/health', { method: 'HEAD' })
|
||||
if (response.ok) {
|
||||
// We're online! Go back
|
||||
window.location.reload()
|
||||
}
|
||||
} catch (error) {
|
||||
// Still offline
|
||||
setTimeout(() => {
|
||||
retrying.value = false
|
||||
}, 1000)
|
||||
}
|
||||
}
|
||||
|
||||
// Auto-retry when online event fires
|
||||
if (typeof window !== 'undefined') {
|
||||
window.addEventListener('online', () => {
|
||||
window.location.reload()
|
||||
})
|
||||
}
|
||||
</script>
|
||||
@@ -2,25 +2,64 @@
|
||||
<div>
|
||||
<h1 class="text-3xl font-bold text-gray-900 mb-6">Scan Item</h1>
|
||||
|
||||
<UCard>
|
||||
<div class="text-center py-12">
|
||||
<UIcon
|
||||
name="i-heroicons-qr-code"
|
||||
class="w-16 h-16 text-gray-400 mx-auto mb-4"
|
||||
<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"
|
||||
/>
|
||||
<h3 class="text-lg font-semibold text-gray-900 mb-2">
|
||||
Barcode Scanner
|
||||
</h3>
|
||||
<p class="text-gray-600 mb-6">
|
||||
This feature will be implemented in Week 3.
|
||||
</p>
|
||||
<UButton
|
||||
to="/"
|
||||
color="gray"
|
||||
variant="soft"
|
||||
>
|
||||
Back to Inventory
|
||||
</UButton>
|
||||
|
||||
<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>
|
||||
@@ -30,4 +69,44 @@
|
||||
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>
|
||||
|
||||
@@ -2,73 +2,74 @@
|
||||
<div>
|
||||
<h1 class="text-3xl font-bold text-gray-900 mb-6">Settings</h1>
|
||||
|
||||
<div class="grid gap-6 md:grid-cols-2">
|
||||
<UCard>
|
||||
<template #header>
|
||||
<h3 class="text-lg font-semibold">Account</h3>
|
||||
</template>
|
||||
|
||||
<div class="space-y-4">
|
||||
<div v-if="user">
|
||||
<label class="text-sm font-medium text-gray-700">Email</label>
|
||||
<p class="text-gray-900">{{ user.email }}</p>
|
||||
<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>
|
||||
|
||||
<UButton
|
||||
v-if="!user"
|
||||
to="/auth/login"
|
||||
color="primary"
|
||||
>
|
||||
Sign In
|
||||
</UButton>
|
||||
</UCard>
|
||||
</template>
|
||||
|
||||
<template #tags>
|
||||
<div class="mt-4">
|
||||
<TagsTagManager />
|
||||
</div>
|
||||
</UCard>
|
||||
</template>
|
||||
|
||||
<UCard>
|
||||
<template #header>
|
||||
<h3 class="text-lg font-semibold">Tags</h3>
|
||||
</template>
|
||||
|
||||
<p class="text-gray-600">
|
||||
Manage your custom tags here (coming in Week 2).
|
||||
</p>
|
||||
</UCard>
|
||||
<template #app>
|
||||
<SettingsAppSettings />
|
||||
</template>
|
||||
|
||||
<UCard>
|
||||
<template #header>
|
||||
<h3 class="text-lg font-semibold">Units</h3>
|
||||
</template>
|
||||
|
||||
<p class="text-gray-600">
|
||||
Manage your custom units here (coming in Week 2).
|
||||
</p>
|
||||
</UCard>
|
||||
|
||||
<UCard>
|
||||
<template #header>
|
||||
<h3 class="text-lg font-semibold">About</h3>
|
||||
</template>
|
||||
|
||||
<div class="space-y-2 text-sm text-gray-600">
|
||||
<p><strong>Pantry</strong> v0.1.0-alpha</p>
|
||||
<p>Self-hosted inventory management</p>
|
||||
<a
|
||||
href="https://github.com/pantry-app/pantry"
|
||||
target="_blank"
|
||||
class="text-primary hover:underline"
|
||||
>
|
||||
View on GitHub →
|
||||
</a>
|
||||
</div>
|
||||
</UCard>
|
||||
</div>
|
||||
<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">
|
||||
const { user } = useSupabaseAuth()
|
||||
|
||||
definePageMeta({
|
||||
layout: 'default'
|
||||
})
|
||||
|
||||
const activeTab = ref('tags')
|
||||
|
||||
const tabs = [
|
||||
{
|
||||
key: 'tags',
|
||||
label: 'Tags',
|
||||
icon: 'i-heroicons-tag'
|
||||
},
|
||||
{
|
||||
key: 'app',
|
||||
label: 'App',
|
||||
icon: 'i-heroicons-device-phone-mobile'
|
||||
},
|
||||
{
|
||||
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 |
|
Before Width: | Height: | Size: 4.2 KiB 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 |
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);
|
||||
141
app/scripts/verify-pwa.js
Normal file
@@ -0,0 +1,141 @@
|
||||
#!/usr/bin/env node
|
||||
/**
|
||||
* Verify PWA Configuration
|
||||
*
|
||||
* Checks that all PWA assets and configuration are present and valid.
|
||||
*/
|
||||
import { readFile, access } from 'fs/promises';
|
||||
import { join, dirname } from 'path';
|
||||
import { fileURLToPath } from 'url';
|
||||
|
||||
const __dirname = dirname(fileURLToPath(import.meta.url));
|
||||
const publicDir = join(__dirname, '..', 'public');
|
||||
const configPath = join(__dirname, '..', 'nuxt.config.ts');
|
||||
|
||||
let errors = [];
|
||||
let warnings = [];
|
||||
|
||||
async function checkFileExists(path, description) {
|
||||
try {
|
||||
await access(path);
|
||||
console.log(`✓ ${description}`);
|
||||
return true;
|
||||
} catch {
|
||||
errors.push(`✗ ${description} - File not found: ${path}`);
|
||||
return false;
|
||||
}
|
||||
}
|
||||
|
||||
async function verifyPWA() {
|
||||
console.log('🔍 Verifying PWA Configuration...\n');
|
||||
|
||||
// Check icons
|
||||
console.log('Icons:');
|
||||
await checkFileExists(join(publicDir, 'icon.svg'), 'Source icon (SVG)');
|
||||
await checkFileExists(join(publicDir, 'icon-192x192.png'), 'Icon 192x192');
|
||||
await checkFileExists(join(publicDir, 'icon-512x512.png'), 'Icon 512x512');
|
||||
await checkFileExists(join(publicDir, 'icon-192x192-maskable.png'), 'Maskable icon 192x192');
|
||||
await checkFileExists(join(publicDir, 'icon-512x512-maskable.png'), 'Maskable icon 512x512');
|
||||
await checkFileExists(join(publicDir, 'favicon.ico'), 'Favicon');
|
||||
await checkFileExists(join(publicDir, 'apple-touch-icon.png'), 'Apple touch icon');
|
||||
|
||||
// Check screenshots
|
||||
console.log('\nScreenshots:');
|
||||
await checkFileExists(join(publicDir, 'screenshot-mobile.png'), 'Mobile screenshot');
|
||||
await checkFileExists(join(publicDir, 'screenshot-desktop.png'), 'Desktop screenshot');
|
||||
|
||||
// Check Nuxt config
|
||||
console.log('\nConfiguration:');
|
||||
const configExists = await checkFileExists(configPath, 'Nuxt config file');
|
||||
|
||||
if (configExists) {
|
||||
const config = await readFile(configPath, 'utf-8');
|
||||
|
||||
// Check for required PWA configuration
|
||||
if (config.includes('@vite-pwa/nuxt')) {
|
||||
console.log('✓ @vite-pwa/nuxt module configured');
|
||||
} else {
|
||||
errors.push('✗ @vite-pwa/nuxt module not found in config');
|
||||
}
|
||||
|
||||
if (config.includes('registerType')) {
|
||||
console.log('✓ Service worker registration configured');
|
||||
} else {
|
||||
warnings.push('⚠ Service worker registration type not set');
|
||||
}
|
||||
|
||||
if (config.includes('manifest')) {
|
||||
console.log('✓ PWA manifest configured');
|
||||
} else {
|
||||
errors.push('✗ PWA manifest configuration missing');
|
||||
}
|
||||
|
||||
if (config.includes('workbox')) {
|
||||
console.log('✓ Workbox configured');
|
||||
} else {
|
||||
warnings.push('⚠ Workbox configuration missing');
|
||||
}
|
||||
|
||||
// Check for important manifest fields
|
||||
if (config.includes('theme_color')) {
|
||||
console.log('✓ Theme color configured');
|
||||
} else {
|
||||
warnings.push('⚠ Theme color not configured');
|
||||
}
|
||||
|
||||
if (config.includes('display')) {
|
||||
console.log('✓ Display mode configured');
|
||||
} else {
|
||||
warnings.push('⚠ Display mode not configured');
|
||||
}
|
||||
}
|
||||
|
||||
// Check composables
|
||||
console.log('\nComposables:');
|
||||
await checkFileExists(join(__dirname, '..', 'composables', 'usePWAInstall.ts'), 'usePWAInstall composable');
|
||||
await checkFileExists(join(__dirname, '..', 'composables', 'useOnlineStatus.ts'), 'useOnlineStatus composable');
|
||||
|
||||
// Check components
|
||||
console.log('\nComponents:');
|
||||
await checkFileExists(join(__dirname, '..', 'components', 'InstallPrompt.vue'), 'InstallPrompt component');
|
||||
await checkFileExists(join(__dirname, '..', 'components', 'OfflineBanner.vue'), 'OfflineBanner component');
|
||||
|
||||
// Check pages
|
||||
console.log('\nPages:');
|
||||
await checkFileExists(join(__dirname, '..', 'pages', 'offline.vue'), 'Offline fallback page');
|
||||
|
||||
// Print summary
|
||||
console.log('\n' + '='.repeat(60));
|
||||
|
||||
if (errors.length === 0 && warnings.length === 0) {
|
||||
console.log('✅ PWA configuration is valid!');
|
||||
console.log('\nNext steps:');
|
||||
console.log('1. Run `npm run dev` and test in browser');
|
||||
console.log('2. Check DevTools → Application → Manifest');
|
||||
console.log('3. Test offline functionality');
|
||||
console.log('4. Run Lighthouse PWA audit');
|
||||
return 0;
|
||||
}
|
||||
|
||||
if (warnings.length > 0) {
|
||||
console.log('\n⚠️ Warnings:');
|
||||
warnings.forEach(w => console.log(w));
|
||||
}
|
||||
|
||||
if (errors.length > 0) {
|
||||
console.log('\n❌ Errors:');
|
||||
errors.forEach(e => console.log(e));
|
||||
console.log('\nPWA configuration is incomplete. Please fix the errors above.');
|
||||
return 1;
|
||||
}
|
||||
|
||||
console.log('\n✅ PWA configuration is mostly valid (with warnings).');
|
||||
return 0;
|
||||
}
|
||||
|
||||
verifyPWA()
|
||||
.then(code => process.exit(code))
|
||||
.catch(error => {
|
||||
console.error('\n❌ Verification failed:', error.message);
|
||||
process.exit(1);
|
||||
});
|
||||
12
app/server/api/health.get.ts
Normal file
@@ -0,0 +1,12 @@
|
||||
/**
|
||||
* Health check endpoint for container monitoring
|
||||
*
|
||||
* Returns 200 OK if the server is running
|
||||
*/
|
||||
export default defineEventHandler(() => {
|
||||
return {
|
||||
status: 'ok',
|
||||
timestamp: new Date().toISOString(),
|
||||
uptime: process.uptime()
|
||||
}
|
||||
})
|
||||
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:
|
||||
90
docker/README.md
Normal file
@@ -0,0 +1,90 @@
|
||||
# Docker Deployment
|
||||
|
||||
## Production Dockerfile
|
||||
|
||||
The production Dockerfile uses a multi-stage build for optimized image size and security.
|
||||
|
||||
### Build the image
|
||||
|
||||
```bash
|
||||
docker build -t pantry:latest -f Dockerfile .
|
||||
```
|
||||
|
||||
### Run the container
|
||||
|
||||
```bash
|
||||
docker run -d \
|
||||
--name pantry \
|
||||
-p 3000:3000 \
|
||||
-e NUXT_PUBLIC_SUPABASE_URL=https://your-project.supabase.co \
|
||||
-e NUXT_PUBLIC_SUPABASE_ANON_KEY=your-anon-key \
|
||||
pantry:latest
|
||||
```
|
||||
|
||||
### Environment Variables
|
||||
|
||||
Required:
|
||||
- `NUXT_PUBLIC_SUPABASE_URL` - Your Supabase project URL
|
||||
- `NUXT_PUBLIC_SUPABASE_ANON_KEY` - Your Supabase anon/public key
|
||||
|
||||
Optional:
|
||||
- `PORT` - Port to listen on (default: 3000)
|
||||
- `HOST` - Host to bind to (default: 0.0.0.0)
|
||||
|
||||
### Health Check
|
||||
|
||||
The container includes a health check endpoint at `/api/health`
|
||||
|
||||
```bash
|
||||
curl http://localhost:3000/api/health
|
||||
```
|
||||
|
||||
Expected response:
|
||||
```json
|
||||
{
|
||||
"status": "ok",
|
||||
"timestamp": "2026-02-25T00:00:00.000Z",
|
||||
"uptime": 123.456
|
||||
}
|
||||
```
|
||||
|
||||
### Image Features
|
||||
|
||||
- **Multi-stage build**: Separate build and runtime stages
|
||||
- **Alpine Linux**: Minimal base image (~50MB base)
|
||||
- **Non-root user**: Runs as unprivileged user (nodejs:1001)
|
||||
- **dumb-init**: Proper signal handling and zombie reaping
|
||||
- **Health checks**: Built-in container health monitoring
|
||||
- **Production-optimized**: Only production dependencies included
|
||||
|
||||
### Image Size
|
||||
|
||||
Approximate sizes:
|
||||
- Base Alpine + Node.js: ~50MB
|
||||
- Dependencies: ~150MB
|
||||
- Built app: ~20MB
|
||||
- **Total**: ~220MB
|
||||
|
||||
### Security
|
||||
|
||||
- Runs as non-root user (nodejs)
|
||||
- No unnecessary packages
|
||||
- Minimal attack surface
|
||||
- Regular security updates via Alpine base
|
||||
|
||||
### Troubleshooting
|
||||
|
||||
View logs:
|
||||
```bash
|
||||
docker logs pantry
|
||||
```
|
||||
|
||||
Interactive shell:
|
||||
```bash
|
||||
docker exec -it pantry sh
|
||||
```
|
||||
|
||||
Check health:
|
||||
```bash
|
||||
docker inspect --format='{{json .State.Health}}' pantry
|
||||
```
|
||||
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
|
||||
283
docs/PWA_TESTING.md
Normal file
@@ -0,0 +1,283 @@
|
||||
# PWA Offline Functionality Testing Guide
|
||||
|
||||
## Overview
|
||||
|
||||
This guide covers testing the Progressive Web App (PWA) features and offline functionality of Pantry.
|
||||
|
||||
## Prerequisites
|
||||
|
||||
- Development server running (`npm run dev` in the `app/` directory)
|
||||
- Modern browser (Chrome, Edge, Safari, or Firefox)
|
||||
- Browser DevTools access
|
||||
|
||||
## Test Categories
|
||||
|
||||
### 1. PWA Manifest & Installation
|
||||
|
||||
#### Test 1.1: Manifest Validation
|
||||
1. Open browser DevTools → Application tab
|
||||
2. Navigate to "Manifest" section
|
||||
3. **Expected Results:**
|
||||
- ✅ Manifest loads without errors
|
||||
- ✅ App name: "Pantry - Smart Inventory Manager"
|
||||
- ✅ Short name: "Pantry"
|
||||
- ✅ Theme color: #10b981 (emerald)
|
||||
- ✅ All icons (192x192, 512x512, maskable) present
|
||||
- ✅ Display mode: standalone
|
||||
- ✅ No manifest warnings
|
||||
|
||||
#### Test 1.2: Install Prompt
|
||||
1. Wait 3 seconds after page load
|
||||
2. **Expected Results:**
|
||||
- ✅ Install prompt card appears (bottom-right on desktop, bottom on mobile)
|
||||
- ✅ Shows app icon and "Install Pantry" title
|
||||
- ✅ "Install" button is clickable
|
||||
- ✅ "Not now" dismisses the prompt
|
||||
- ✅ Close (X) button dismisses the prompt
|
||||
|
||||
#### Test 1.3: Manual Installation from Settings
|
||||
1. Navigate to Settings → App tab
|
||||
2. **Expected Results:**
|
||||
- ✅ Shows "Install App" button
|
||||
- ✅ Clicking installs the app
|
||||
- ✅ After install, shows "App is installed" status
|
||||
- ✅ Storage usage displayed with progress bar
|
||||
|
||||
#### Test 1.4: Platform-Specific Instructions
|
||||
1. View Settings → App tab on device without beforeinstallprompt support
|
||||
2. **Expected Results:**
|
||||
- ✅ Shows iOS installation instructions (if on iOS)
|
||||
- ✅ Shows Android installation instructions (if on Android)
|
||||
- ✅ Instructions are clear and accurate
|
||||
|
||||
### 2. Service Worker
|
||||
|
||||
#### Test 2.1: Service Worker Registration
|
||||
1. Open DevTools → Application → Service Workers
|
||||
2. **Expected Results:**
|
||||
- ✅ Service worker registered
|
||||
- ✅ Status: "activated and running"
|
||||
- ✅ No registration errors
|
||||
- ✅ Update on reload enabled
|
||||
|
||||
#### Test 2.2: Cache Storage
|
||||
1. Open DevTools → Application → Cache Storage
|
||||
2. Navigate through the app (Home, Scan, Settings)
|
||||
3. **Expected Results:**
|
||||
- ✅ Multiple cache buckets created:
|
||||
- workbox-precache (app shell)
|
||||
- supabase-rest-api
|
||||
- supabase-storage
|
||||
- product-images (if products viewed)
|
||||
- google-fonts-stylesheets
|
||||
- google-fonts-webfonts
|
||||
- ✅ App shell assets cached (JS, CSS, HTML)
|
||||
- ✅ Icons and images cached
|
||||
|
||||
#### Test 2.3: Update Behavior
|
||||
1. Make a code change
|
||||
2. Rebuild the app
|
||||
3. Refresh the page
|
||||
4. **Expected Results:**
|
||||
- ✅ Service worker updates in background
|
||||
- ✅ New version activates automatically (skipWaiting)
|
||||
- ✅ No manual refresh required for future visits
|
||||
|
||||
### 3. Offline Functionality
|
||||
|
||||
#### Test 3.1: Complete Offline Mode
|
||||
1. Load the app while online
|
||||
2. Navigate to all pages (Home, Scan, Settings)
|
||||
3. Open DevTools → Network tab
|
||||
4. Enable "Offline" mode
|
||||
5. Try navigating the app
|
||||
6. **Expected Results:**
|
||||
- ✅ App continues to function
|
||||
- ✅ Previously visited pages load instantly
|
||||
- ✅ Offline banner appears at top
|
||||
- ✅ Banner shows "You're currently offline" message
|
||||
- ✅ Navigation between cached pages works
|
||||
- ✅ No white screens or errors
|
||||
|
||||
#### Test 3.2: Offline Fallback Page
|
||||
1. Go offline (DevTools Network → Offline)
|
||||
2. Try navigating to a non-cached page (e.g., type random URL)
|
||||
3. **Expected Results:**
|
||||
- ✅ Redirects to /offline page
|
||||
- ✅ Shows WiFi icon and helpful message
|
||||
- ✅ Lists what you can do offline
|
||||
- ✅ "Try Again" button present
|
||||
- ✅ Auto-redirects when back online
|
||||
|
||||
#### Test 3.3: Online Status Detection
|
||||
1. Start online, go offline, come back online
|
||||
2. **Expected Results:**
|
||||
- ✅ Offline banner appears when offline
|
||||
- ✅ "Back online!" banner shows when reconnected (green)
|
||||
- ✅ Banner auto-hides after 3 seconds
|
||||
- ✅ No false positives
|
||||
|
||||
#### Test 3.4: API Request Caching (Supabase)
|
||||
1. While online, view some inventory items (once implemented)
|
||||
2. Go offline
|
||||
3. Navigate to the items page
|
||||
4. **Expected Results:**
|
||||
- ✅ Previously loaded items still visible
|
||||
- ✅ Network requests fail gracefully
|
||||
- ✅ Cached data is served
|
||||
- ✅ No crashes or white screens
|
||||
|
||||
#### Test 3.5: Image Caching (Product Images)
|
||||
1. While online, view products with images
|
||||
2. Go offline
|
||||
3. View the same products again
|
||||
4. **Expected Results:**
|
||||
- ✅ Product images load from cache
|
||||
- ✅ No broken image placeholders
|
||||
- ✅ Images from Open Food Facts cached
|
||||
|
||||
### 4. Background Sync (Future Enhancement)
|
||||
|
||||
**Note:** Background sync not yet implemented. This section is reserved for future testing.
|
||||
|
||||
### 5. Cross-Platform Testing
|
||||
|
||||
#### Test 5.1: Desktop Browsers
|
||||
Test on:
|
||||
- [ ] Chrome/Edge (Windows/Mac/Linux)
|
||||
- [ ] Firefox (Windows/Mac/Linux)
|
||||
- [ ] Safari (Mac only)
|
||||
|
||||
#### Test 5.2: Mobile Browsers
|
||||
Test on:
|
||||
- [ ] Chrome (Android)
|
||||
- [ ] Safari (iOS)
|
||||
- [ ] Firefox (Android)
|
||||
- [ ] Samsung Internet (Android)
|
||||
|
||||
#### Test 5.3: Installed App vs Browser
|
||||
Compare behavior when:
|
||||
- [ ] Running in browser tab
|
||||
- [ ] Running as installed PWA (standalone mode)
|
||||
|
||||
**Expected Results:**
|
||||
- ✅ Identical functionality
|
||||
- ✅ Installed app shows in app switcher
|
||||
- ✅ Installed app has no browser chrome
|
||||
- ✅ Installed app survives system restart
|
||||
|
||||
### 6. Performance Testing
|
||||
|
||||
#### Test 6.1: First Load Performance
|
||||
1. Clear all caches
|
||||
2. Load the app (online)
|
||||
3. Check DevTools → Lighthouse
|
||||
4. Run PWA audit
|
||||
5. **Expected Results:**
|
||||
- ✅ PWA score: 90+ / 100
|
||||
- ✅ Performance score: 80+ / 100
|
||||
- ✅ "Installable" badge present
|
||||
- ✅ No PWA warnings
|
||||
|
||||
#### Test 6.2: Repeat Visit Performance
|
||||
1. Visit the app
|
||||
2. Navigate around
|
||||
3. Close tab
|
||||
4. Reopen the app
|
||||
5. **Expected Results:**
|
||||
- ✅ Instant load from cache
|
||||
- ✅ No flash of white screen
|
||||
- ✅ Content visible immediately
|
||||
|
||||
### 7. Storage Management
|
||||
|
||||
#### Test 7.1: Storage Quota
|
||||
1. Open Settings → App tab
|
||||
2. **Expected Results:**
|
||||
- ✅ Storage usage displayed
|
||||
- ✅ Storage quota displayed
|
||||
- ✅ Usage percentage shown visually
|
||||
- ✅ Numbers update as cache grows
|
||||
|
||||
#### Test 7.2: Cache Eviction
|
||||
1. Fill cache with many images/data
|
||||
2. Exceed storage quota
|
||||
3. **Expected Results:**
|
||||
- ✅ Oldest cache entries evicted automatically
|
||||
- ✅ No app crashes
|
||||
- ✅ App continues to function
|
||||
|
||||
## Automated Testing (Future)
|
||||
|
||||
### Playwright E2E Tests (Planned)
|
||||
|
||||
```typescript
|
||||
// Example test structure
|
||||
test('PWA installs correctly', async ({ page }) => {
|
||||
// Test installation flow
|
||||
})
|
||||
|
||||
test('App works offline', async ({ page, context }) => {
|
||||
// Load app, go offline, verify functionality
|
||||
})
|
||||
```
|
||||
|
||||
## Known Issues & Limitations
|
||||
|
||||
1. **iOS Safari:**
|
||||
- No beforeinstallprompt event (use manual Add to Home Screen)
|
||||
- Service worker has storage limits
|
||||
- Background sync not supported
|
||||
|
||||
2. **Firefox:**
|
||||
- Install prompt may not show (desktop only)
|
||||
- Use "Add to Home Screen" on mobile
|
||||
|
||||
3. **Development Mode:**
|
||||
- Service worker may behave differently
|
||||
- Always test in production build
|
||||
|
||||
## Troubleshooting
|
||||
|
||||
### Service Worker Not Updating
|
||||
- Hard refresh: Ctrl+Shift+R (Windows/Linux) or Cmd+Shift+R (Mac)
|
||||
- DevTools → Application → Service Workers → Unregister
|
||||
- Clear cache and reload
|
||||
|
||||
### Install Prompt Not Showing
|
||||
- Check if already installed
|
||||
- Check localStorage for `pwa-install-dismissed`
|
||||
- Wait 7 days or clear localStorage
|
||||
- Ensure criteria met (HTTPS, manifest, service worker)
|
||||
|
||||
### Offline Mode Not Working
|
||||
- Verify service worker is active
|
||||
- Check cache storage has content
|
||||
- Ensure you visited pages while online first
|
||||
|
||||
## Success Criteria
|
||||
|
||||
All tests must pass before marking issue #36 complete:
|
||||
|
||||
- [x] PWA manifest loads correctly
|
||||
- [x] Install prompt works
|
||||
- [x] Service worker registers and activates
|
||||
- [x] App works offline
|
||||
- [x] Cached content loads
|
||||
- [x] Offline banner shows/hides correctly
|
||||
- [x] Online status detected accurately
|
||||
- [x] Install instructions provided for unsupported browsers
|
||||
- [x] Storage usage displayed
|
||||
- [x] No console errors during offline usage
|
||||
|
||||
## Sign-off
|
||||
|
||||
**Tested by:** [Name]
|
||||
**Date:** [Date]
|
||||
**Browsers tested:** [List]
|
||||
**Issues found:** [List or "None"]
|
||||
**Status:** ✅ Pass / ❌ Fail
|
||||
|
||||
---
|
||||
|
||||
**Next Steps:** After testing passes, proceed to Week 6 (Deployment & Testing).
|
||||
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
|
||||
}
|
||||
)
|
||||
}
|
||||
})
|
||||