Compare commits

..

5 Commits

Author SHA1 Message Date
Pantry Lead Agent
4834286005 feat: implement inventory CRUD UI components (#18 #19 #20 #21)
Some checks failed
Deploy to Coolify / Deploy to Development (pull_request) Has been cancelled
Deploy to Coolify / Deploy to Production (pull_request) Has been cancelled
Deploy to Coolify / Code Quality (pull_request) Has been cancelled
Deploy to Coolify / Run Tests (pull_request) Has been cancelled
Deploy to Coolify / Deploy to Test (pull_request) Has been cancelled
Pull Request Checks / Validate PR (pull_request) Has been cancelled
Week 2 core inventory management:

**Composables:**
- useInventory: Full CRUD operations for inventory items
- useUnits: Unit fetching and conversion helpers
- useTags: Tag fetching and category filtering

**Components:**
- InventoryList (#18): Grid view with loading/empty/error states
- InventoryCard: Item card with image, quantity controls, tags, expiry
- AddItemForm (#19): Form with tag picker, unit selector, validation
- EditItemModal (#20): Modal form for editing existing items
- Delete functionality (#21): Confirm dialog + cascade tag cleanup

**Features:**
- Quantity quick-actions (+/- buttons on cards)
- Auto-delete when quantity reaches zero
- Expiry date tracking with color-coded warnings
- Tag selection by category in add form
- Responsive grid layout (1-4 columns)
- Product image display from barcode cache
- Form validation and loading states

Closes #18, #19, #20, #21
2026-02-09 13:03:00 +00:00
be2af1675a Merge pull request 'feat: seed default units and tags (#16 #17)' (#45) from feature/issue-16-17-seed-data into develop
Some checks failed
Deploy to Coolify / Code Quality (push) Has been cancelled
Deploy to Coolify / Run Tests (push) Has been cancelled
Deploy to Coolify / Deploy to Development (push) Has been cancelled
Deploy to Coolify / Deploy to Production (push) Has been cancelled
Deploy to Coolify / Deploy to Test (push) Has been cancelled
2026-02-09 13:00:17 +00:00
Pantry Lead Agent
b93f4677fc feat: seed default units and tags (#16 #17)
Some checks failed
Deploy to Coolify / Code Quality (pull_request) Has been cancelled
Deploy to Coolify / Run Tests (pull_request) Has been cancelled
Deploy to Coolify / Deploy to Development (pull_request) Has been cancelled
Deploy to Coolify / Deploy to Production (pull_request) Has been cancelled
Deploy to Coolify / Deploy to Test (pull_request) Has been cancelled
Pull Request Checks / Validate PR (pull_request) Has been cancelled
Added comprehensive seed data migrations:

**Units Migration (#16):**
- 30 pre-populated measurement units
- Weight: g, kg, mg, lb, oz (base: gram)
- Volume: mL, L, cup, tbsp, tsp, fl oz, gal, qt, pt (base: milliliter)
- Count: piece, dozen, package, bottle, can, jar, box, bag
- Proper conversion factors for metric/imperial

**Tags Migration (#17):**
- 33 pre-populated organizational tags
- Position: Fridge, Freezer, Pantry, Cabinet, Countertop, Cellar
- Type: Dairy, Meat, Fish, Vegetables, Fruits, Grains, etc.
- Dietary: Vegan, Vegetarian, Gluten-Free, Organic, Kosher, Halal
- Custom: Low Stock, To Buy, Meal Prep, Leftovers
- Each tag includes icon emoji and color code

Ready for frontend to start creating inventory items.

Closes #16, #17
2026-02-09 13:00:02 +00:00
4eec4923af Merge pull request 'feat: add SQL helper functions for inventory management (#15)' (#44) from feature/issue-15-sql-functions into develop
Some checks failed
Deploy to Coolify / Code Quality (push) Has been cancelled
Deploy to Coolify / Run Tests (push) Has been cancelled
Deploy to Coolify / Deploy to Development (push) Has been cancelled
Deploy to Coolify / Deploy to Production (push) Has been cancelled
Deploy to Coolify / Deploy to Test (push) Has been cancelled
2026-02-09 12:59:05 +00:00
Pantry Lead Agent
f70b90748a feat: add SQL helper functions for inventory management (#15)
Some checks failed
Pull Request Checks / Validate PR (pull_request) Has been cancelled
Deploy to Coolify / Run Tests (pull_request) Has been cancelled
Deploy to Coolify / Deploy to Development (pull_request) Has been cancelled
Deploy to Coolify / Deploy to Production (pull_request) Has been cancelled
Deploy to Coolify / Code Quality (pull_request) Has been cancelled
Deploy to Coolify / Deploy to Test (pull_request) Has been cancelled
Created 8 helper functions for common inventory operations:

1. get_inventory_details() - Full inventory list with denormalized data
2. get_expiring_items(days) - Items expiring within N days
3. get_items_by_tag(tag_name) - Filter by tag
4. get_low_stock_items(threshold) - Items below quantity threshold
5. update_item_quantity() - Consume/restock with optional auto-delete
6. get_inventory_stats() - Dashboard statistics
7. search_inventory() - Full-text search across items and products

All functions include comments and are optimized for frontend queries.

Closes #15
2026-02-09 12:58:45 +00:00
11 changed files with 1396 additions and 21 deletions

View File

@@ -0,0 +1,257 @@
<template>
<UCard>
<template #header>
<div class="flex items-center justify-between">
<h3 class="text-lg font-semibold">Add New Item</h3>
<UButton
icon="i-heroicons-x-mark"
color="gray"
variant="ghost"
@click="$emit('close')"
/>
</div>
</template>
<form @submit.prevent="handleSubmit" class="space-y-4">
<!-- Item Name -->
<UFormGroup label="Item Name" required>
<UInput
v-model="form.name"
placeholder="e.g. Whole Milk, Pasta, Tomatoes"
size="lg"
autofocus
/>
</UFormGroup>
<!-- Quantity & Unit -->
<div class="grid grid-cols-2 gap-3">
<UFormGroup label="Quantity" required>
<UInput
v-model.number="form.quantity"
type="number"
min="0.01"
step="0.01"
placeholder="1"
size="lg"
/>
</UFormGroup>
<UFormGroup label="Unit" required>
<USelect
v-model="form.unit_id"
:options="unitOptions"
option-attribute="label"
value-attribute="value"
placeholder="Select unit"
size="lg"
/>
</UFormGroup>
</div>
<!-- Expiry Date -->
<UFormGroup label="Expiry Date" hint="Optional">
<UInput
v-model="form.expiry_date"
type="date"
size="lg"
/>
</UFormGroup>
<!-- Notes -->
<UFormGroup label="Notes" hint="Optional">
<UTextarea
v-model="form.notes"
placeholder="Any additional notes..."
:rows="2"
/>
</UFormGroup>
<!-- Tags -->
<UFormGroup label="Tags" hint="Optional">
<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>
</UFormGroup>
<!-- Submit -->
<div class="flex gap-2 pt-2">
<UButton
type="submit"
color="primary"
size="lg"
class="flex-1"
:loading="submitting"
:disabled="!isValid"
>
Add Item
</UButton>
<UButton
color="gray"
size="lg"
variant="soft"
@click="$emit('close')"
>
Cancel
</UButton>
</div>
</form>
</UCard>
</template>
<script setup lang="ts">
const { addInventoryItem, addItemTags } = useInventory()
const { getUnits } = useUnits()
const { getTags } = useTags()
const emit = defineEmits<{
close: []
added: [item: any]
}>()
// Form state
const form = reactive({
name: '',
quantity: 1,
unit_id: '',
expiry_date: '',
notes: ''
})
const submitting = ref(false)
const selectedTags = ref<any[]>([])
// Load units and tags
const units = ref<any[]>([])
const tags = ref<any[]>([])
onMounted(async () => {
const [unitsResult, tagsResult] = await Promise.all([
getUnits(),
getTags()
])
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
}
})
// Unit options for select
const unitOptions = computed(() => {
const grouped: Record<string, any[]> = {}
for (const unit of units.value) {
const type = unit.unit_type
if (!grouped[type]) grouped[type] = []
grouped[type].push({
label: `${unit.name} (${unit.abbreviation})`,
value: unit.id
})
}
return Object.entries(grouped).flatMap(([type, options]) => [
{ label: `${type.charAt(0).toUpperCase() + type.slice(1)}`, value: '', disabled: true },
...options
])
})
// 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
})
// Submit
const handleSubmit = async () => {
if (!isValid.value) return
submitting.value = true
const { data, error } = await addInventoryItem({
name: form.name.trim(),
quantity: form.quantity,
unit_id: form.unit_id,
expiry_date: form.expiry_date || null,
notes: form.notes.trim() || null
})
if (error) {
alert('Failed to add item: ' + error.message)
submitting.value = false
return
}
// Add tags if any selected
if (data && selectedTags.value.length > 0) {
const tagIds = selectedTags.value.map(t => t.id)
await addItemTags(data.id, tagIds)
}
emit('added', data)
submitting.value = false
}
</script>

View File

@@ -0,0 +1,184 @@
<template>
<UModal v-model="isOpen">
<UCard>
<template #header>
<div class="flex items-center justify-between">
<h3 class="text-lg font-semibold">Edit Item</h3>
<UButton
icon="i-heroicons-x-mark"
color="gray"
variant="ghost"
@click="close"
/>
</div>
</template>
<form @submit.prevent="handleSubmit" class="space-y-4">
<!-- Item Name -->
<UFormGroup label="Item Name" required>
<UInput
v-model="form.name"
placeholder="Item name"
size="lg"
/>
</UFormGroup>
<!-- Quantity & Unit -->
<div class="grid grid-cols-2 gap-3">
<UFormGroup label="Quantity" required>
<UInput
v-model.number="form.quantity"
type="number"
min="0.01"
step="0.01"
size="lg"
/>
</UFormGroup>
<UFormGroup label="Unit" required>
<USelect
v-model="form.unit_id"
:options="unitOptions"
option-attribute="label"
value-attribute="value"
size="lg"
/>
</UFormGroup>
</div>
<!-- Expiry Date -->
<UFormGroup label="Expiry Date" hint="Optional">
<UInput
v-model="form.expiry_date"
type="date"
size="lg"
/>
</UFormGroup>
<!-- Notes -->
<UFormGroup label="Notes" hint="Optional">
<UTextarea
v-model="form.notes"
placeholder="Any additional notes..."
:rows="2"
/>
</UFormGroup>
<!-- Submit -->
<div class="flex gap-2 pt-2">
<UButton
type="submit"
color="primary"
size="lg"
class="flex-1"
:loading="submitting"
:disabled="!isValid"
>
Save Changes
</UButton>
<UButton
color="gray"
size="lg"
variant="soft"
@click="close"
>
Cancel
</UButton>
</div>
</form>
</UCard>
</UModal>
</template>
<script setup lang="ts">
const { updateInventoryItem } = useInventory()
const { getUnits } = useUnits()
const props = defineProps<{
item: any | null
}>()
const emit = defineEmits<{
close: []
updated: [item: any]
}>()
const isOpen = ref(false)
const submitting = ref(false)
const units = ref<any[]>([])
const form = reactive({
name: '',
quantity: 1,
unit_id: '',
expiry_date: '',
notes: ''
})
// Load units
onMounted(async () => {
const { data } = await getUnits()
units.value = data || []
})
// Unit options for select
const unitOptions = computed(() => {
return units.value.map(unit => ({
label: `${unit.name} (${unit.abbreviation})`,
value: unit.id
}))
})
// Watch for item changes (open modal)
watch(() => props.item, (newItem) => {
if (newItem) {
form.name = newItem.name
form.quantity = Number(newItem.quantity)
form.unit_id = newItem.unit_id
form.expiry_date = newItem.expiry_date || ''
form.notes = newItem.notes || ''
isOpen.value = true
}
}, { immediate: true })
// Watch modal close
watch(isOpen, (val) => {
if (!val) {
emit('close')
}
})
// Validation
const isValid = computed(() => {
return form.name.trim().length > 0 && form.quantity > 0 && form.unit_id
})
const close = () => {
isOpen.value = false
}
// Submit
const handleSubmit = async () => {
if (!isValid.value || !props.item) return
submitting.value = true
const { data, error } = await updateInventoryItem(props.item.id, {
name: form.name.trim(),
quantity: form.quantity,
unit_id: form.unit_id,
expiry_date: form.expiry_date || null,
notes: form.notes.trim() || null
})
if (error) {
alert('Failed to update item: ' + error.message)
submitting.value = false
return
}
emit('updated', data)
submitting.value = false
close()
}
</script>

View File

@@ -0,0 +1,151 @@
<template>
<UCard class="hover:shadow-lg transition-shadow">
<!-- Item Image -->
<div class="aspect-square bg-gray-100 rounded-lg mb-3 overflow-hidden">
<img
v-if="item.product?.image_url"
:src="item.product.image_url"
:alt="item.name"
class="w-full h-full object-cover"
/>
<div v-else class="w-full h-full flex items-center justify-center">
<UIcon name="i-heroicons-cube" class="w-16 h-16 text-gray-300" />
</div>
</div>
<!-- Item Info -->
<div class="space-y-2">
<div>
<h3 class="font-semibold text-gray-900 truncate">{{ item.name }}</h3>
<p v-if="item.product?.brand" class="text-sm text-gray-600 truncate">
{{ item.product.brand }}
</p>
</div>
<!-- Quantity -->
<div class="flex items-center justify-between">
<span class="text-lg font-medium text-gray-900">
{{ item.quantity }} {{ item.unit?.abbreviation }}
</span>
<!-- Quick Actions -->
<div class="flex gap-1">
<UButton
icon="i-heroicons-minus"
size="xs"
color="gray"
variant="ghost"
@click="$emit('update-quantity', item.id, -1)"
:disabled="item.quantity <= 1"
/>
<UButton
icon="i-heroicons-plus"
size="xs"
color="gray"
variant="ghost"
@click="$emit('update-quantity', item.id, 1)"
/>
</div>
</div>
<!-- Tags -->
<div v-if="item.tags && item.tags.length > 0" class="flex flex-wrap gap-1">
<UBadge
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>
<UBadge v-if="item.tags.length > 3" size="xs" color="gray">
+{{ item.tags.length - 3 }}
</UBadge>
</div>
<!-- Expiry Warning -->
<div v-if="daysUntilExpiry !== null" class="text-xs">
<UBadge
:color="expiryColor"
variant="soft"
class="w-full justify-center"
>
<UIcon :name="expiryIcon" class="mr-1" />
{{ expiryText }}
</UBadge>
</div>
</div>
<!-- Action Buttons -->
<template #footer>
<div class="flex gap-2">
<UButton
icon="i-heroicons-pencil"
size="sm"
color="gray"
variant="soft"
class="flex-1"
@click="$emit('edit', item)"
>
Edit
</UButton>
<UButton
icon="i-heroicons-trash"
size="sm"
color="red"
variant="soft"
@click="$emit('delete', item.id)"
>
Delete
</UButton>
</div>
</template>
</UCard>
</template>
<script setup lang="ts">
const props = defineProps<{
item: any
}>()
defineEmits<{
edit: [item: any]
delete: [id: string]
'update-quantity': [id: string, change: number]
}>()
// Calculate days until expiry
const daysUntilExpiry = computed(() => {
if (!props.item.expiry_date) return null
const today = new Date()
const expiry = new Date(props.item.expiry_date)
const diff = Math.ceil((expiry.getTime() - today.getTime()) / (1000 * 60 * 60 * 24))
return diff
})
// Expiry badge styling
const expiryColor = computed(() => {
if (daysUntilExpiry.value === null) return 'gray'
if (daysUntilExpiry.value < 0) return 'red'
if (daysUntilExpiry.value <= 3) return 'orange'
if (daysUntilExpiry.value <= 7) return 'yellow'
return 'green'
})
const expiryIcon = computed(() => {
if (daysUntilExpiry.value === null) return 'i-heroicons-calendar'
if (daysUntilExpiry.value < 0) return 'i-heroicons-exclamation-triangle'
return 'i-heroicons-clock'
})
const expiryText = computed(() => {
if (daysUntilExpiry.value === null) return 'No expiry'
if (daysUntilExpiry.value < 0) return `Expired ${Math.abs(daysUntilExpiry.value)} days ago`
if (daysUntilExpiry.value === 0) return 'Expires today'
if (daysUntilExpiry.value === 1) return 'Expires tomorrow'
return `Expires in ${daysUntilExpiry.value} days`
})
</script>

View File

@@ -0,0 +1,131 @@
<template>
<div class="space-y-4">
<!-- Loading State -->
<div v-if="loading" class="text-center py-12">
<UIcon name="i-heroicons-arrow-path" class="w-8 h-8 text-gray-400 animate-spin mx-auto mb-2" />
<p class="text-gray-600">Loading inventory...</p>
</div>
<!-- Error State -->
<div v-else-if="error" class="text-center py-12">
<UIcon name="i-heroicons-exclamation-triangle" class="w-12 h-12 text-red-500 mx-auto mb-4" />
<p class="text-red-600 mb-4">{{ error }}</p>
<UButton @click="loadInventory" color="gray">Retry</UButton>
</div>
<!-- Empty State -->
<div v-else-if="!items || items.length === 0" class="text-center py-12">
<UIcon name="i-heroicons-inbox" class="w-16 h-16 text-gray-400 mx-auto mb-4" />
<h3 class="text-lg font-semibold text-gray-900 mb-2">
No items yet
</h3>
<p class="text-gray-600 mb-6">
Start by scanning a barcode or adding an item manually.
</p>
<div class="flex gap-2 justify-center">
<UButton
to="/scan"
color="primary"
icon="i-heroicons-qr-code"
>
Scan First Item
</UButton>
<UButton
@click="$emit('add-item')"
color="white"
icon="i-heroicons-plus"
>
Add Manually
</UButton>
</div>
</div>
<!-- Inventory Grid -->
<div v-else class="grid gap-4 sm:grid-cols-2 lg:grid-cols-3 xl:grid-cols-4">
<InventoryCard
v-for="item in items"
:key="item.id"
:item="item"
@edit="$emit('edit-item', item)"
@delete="handleDelete(item.id)"
@update-quantity="handleQuantityUpdate"
/>
</div>
</div>
</template>
<script setup lang="ts">
const { getInventory, deleteInventoryItem, updateQuantity } = useInventory()
const props = defineProps<{
refresh?: boolean
}>()
const emit = defineEmits<{
'add-item': []
'edit-item': [item: any]
}>()
const items = ref<any[]>([])
const loading = ref(true)
const error = ref<string | null>(null)
const loadInventory = async () => {
loading.value = true
error.value = null
const { data, error: fetchError } = await getInventory()
if (fetchError) {
error.value = 'Failed to load inventory. Please try again.'
loading.value = false
return
}
items.value = data || []
loading.value = false
}
const handleDelete = async (id: string) => {
if (!confirm('Are you sure you want to delete this item?')) {
return
}
const { error: deleteError } = await deleteInventoryItem(id)
if (deleteError) {
alert('Failed to delete item')
return
}
// Remove from local list
items.value = items.value.filter(item => item.id !== id)
}
const handleQuantityUpdate = async (id: string, change: number) => {
const result = await updateQuantity(id, change)
if (result.error) {
alert('Failed to update quantity')
return
}
// Reload inventory after update
await loadInventory()
}
// Load on mount
onMounted(loadInventory)
// Watch for refresh prop
watch(() => props.refresh, (newVal) => {
if (newVal) {
loadInventory()
}
})
// Expose reload method
defineExpose({
reload: loadInventory
})
</script>

View File

@@ -0,0 +1,201 @@
import type { Database } from '~/types/database.types'
type InventoryItem = Database['public']['Tables']['inventory_items']['Row']
type InventoryItemInsert = Database['public']['Tables']['inventory_items']['Insert']
type InventoryItemUpdate = Database['public']['Tables']['inventory_items']['Update']
export const useInventory = () => {
const supabase = useSupabase()
const { user } = useSupabaseAuth()
/**
* Get all inventory items with denormalized data
*/
const getInventory = async () => {
const { data, error } = await supabase
.from('inventory_items')
.select(`
*,
product:products(*),
unit:units(*),
tags:item_tags(tag:tags(*))
`)
.order('created_at', { ascending: false })
if (error) {
console.error('Error fetching inventory:', error)
return { data: null, error }
}
return { data, error: null }
}
/**
* Get single inventory item by ID
*/
const getInventoryItem = async (id: string) => {
const { data, error } = await supabase
.from('inventory_items')
.select(`
*,
product:products(*),
unit:units(*),
tags:item_tags(tag:tags(*))
`)
.eq('id', id)
.single()
if (error) {
console.error('Error fetching item:', error)
return { data: null, error }
}
return { data, error: null }
}
/**
* Add new inventory item
*/
const addInventoryItem = async (item: Omit<InventoryItemInsert, 'added_by'>) => {
if (!user.value) {
return { data: null, error: { message: 'User not authenticated' } }
}
const { data, error } = await supabase
.from('inventory_items')
.insert({
...item,
added_by: user.value.id
})
.select(`
*,
product:products(*),
unit:units(*),
tags:item_tags(tag:tags(*))
`)
.single()
if (error) {
console.error('Error adding item:', error)
return { data: null, error }
}
return { data, error: null }
}
/**
* Update inventory item
*/
const updateInventoryItem = async (id: string, updates: InventoryItemUpdate) => {
const { data, error } = await supabase
.from('inventory_items')
.update(updates)
.eq('id', id)
.select(`
*,
product:products(*),
unit:units(*),
tags:item_tags(tag:tags(*))
`)
.single()
if (error) {
console.error('Error updating item:', error)
return { data: null, error }
}
return { data, error: null }
}
/**
* Delete inventory item
*/
const deleteInventoryItem = async (id: string) => {
// First delete associated tags
await supabase
.from('item_tags')
.delete()
.eq('item_id', id)
const { error } = await supabase
.from('inventory_items')
.delete()
.eq('id', id)
if (error) {
console.error('Error deleting item:', error)
return { error }
}
return { error: null }
}
/**
* Update item quantity (consume or restock)
*/
const updateQuantity = async (id: string, change: number) => {
const { data: item, error: fetchError } = await getInventoryItem(id)
if (fetchError || !item) {
return { data: null, error: fetchError }
}
const newQuantity = Number(item.quantity) + change
if (newQuantity <= 0) {
// Auto-delete when quantity reaches zero
return await deleteInventoryItem(id)
}
return await updateInventoryItem(id, { quantity: newQuantity })
}
/**
* Add tags to item
*/
const addItemTags = async (itemId: string, tagIds: string[]) => {
const items = tagIds.map(tagId => ({
item_id: itemId,
tag_id: tagId
}))
const { error } = await supabase
.from('item_tags')
.insert(items)
if (error) {
console.error('Error adding tags:', error)
return { error }
}
return { error: null }
}
/**
* Remove tag from item
*/
const removeItemTag = async (itemId: string, tagId: string) => {
const { error } = await supabase
.from('item_tags')
.delete()
.eq('item_id', itemId)
.eq('tag_id', tagId)
if (error) {
console.error('Error removing tag:', error)
return { error }
}
return { error: null }
}
return {
getInventory,
getInventoryItem,
addInventoryItem,
updateInventoryItem,
deleteInventoryItem,
updateQuantity,
addItemTags,
removeItemTag
}
}

View File

@@ -0,0 +1,44 @@
export const useTags = () => {
const supabase = useSupabase()
/**
* Get all tags
*/
const getTags = async () => {
const { data, error } = await supabase
.from('tags')
.select('*')
.order('category', { ascending: true })
.order('name', { ascending: true })
if (error) {
console.error('Error fetching tags:', error)
return { data: null, error }
}
return { data, error: null }
}
/**
* Get tags by category
*/
const getTagsByCategory = async (category: 'position' | 'type' | 'dietary' | 'custom') => {
const { data, error } = await supabase
.from('tags')
.select('*')
.eq('category', category)
.order('name', { ascending: true })
if (error) {
console.error('Error fetching tags by category:', error)
return { data: null, error }
}
return { data, error: null }
}
return {
getTags,
getTagsByCategory
}
}

View File

@@ -0,0 +1,53 @@
export const useUnits = () => {
const supabase = useSupabase()
/**
* Get all units
*/
const getUnits = async () => {
const { data, error } = await supabase
.from('units')
.select('*')
.order('unit_type', { ascending: true })
.order('name', { ascending: true })
if (error) {
console.error('Error fetching units:', error)
return { data: null, error }
}
return { data, error: null }
}
/**
* Get default unit for a type
*/
const getDefaultUnit = async (unitType: 'weight' | 'volume' | 'count' | 'custom') => {
const { data, error } = await supabase
.from('units')
.select('*')
.eq('unit_type', unitType)
.eq('is_default', true)
.single()
if (error) {
console.error('Error fetching default unit:', error)
return { data: null, error }
}
return { data, error: null }
}
/**
* Convert quantity between units
*/
const convertUnit = (quantity: number, fromFactor: number, toFactor: number): number => {
return (quantity * fromFactor) / toFactor
}
return {
getUnits,
getDefaultUnit,
convertUnit
}
}

View File

@@ -17,36 +17,37 @@
color="white"
size="lg"
icon="i-heroicons-plus"
@click="showAddForm = true"
>
Add Manually
</UButton>
</div>
</div>
<!-- Empty State -->
<UCard v-if="true">
<div class="text-center py-12">
<UIcon
name="i-heroicons-inbox"
class="w-16 h-16 text-gray-400 mx-auto mb-4"
<!-- 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"
@added="handleItemAdded"
/>
<h3 class="text-lg font-semibold text-gray-900 mb-2">
No items yet
</h3>
<p class="text-gray-600 mb-6">
Start by scanning a barcode or adding an item manually.
</p>
<UButton
to="/scan"
color="primary"
icon="i-heroicons-qr-code"
>
Scan First Item
</UButton>
</div>
</UCard>
</div>
<!-- TODO: Item list will go here -->
<!-- Edit Item Modal -->
<EditItemModal
:item="editingItem"
@close="editingItem = null"
@updated="handleItemUpdated"
/>
<!-- Inventory List -->
<InventoryList
ref="inventoryListRef"
:refresh="refreshKey"
@add-item="showAddForm = true"
@edit-item="editingItem = $event"
/>
</div>
</template>
@@ -54,4 +55,20 @@
definePageMeta({
layout: 'default'
})
const showAddForm = ref(false)
const editingItem = ref<any>(null)
const refreshKey = ref(0)
const inventoryListRef = ref()
const handleItemAdded = (item: any) => {
showAddForm.value = false
// Reload the inventory list
inventoryListRef.value?.reload()
}
const handleItemUpdated = (item: any) => {
editingItem.value = null
inventoryListRef.value?.reload()
}
</script>

View File

@@ -0,0 +1,251 @@
-- Migration: Additional SQL Functions for Inventory Management
-- Week 2: Helper functions for common queries
-- Function: Get inventory items with full details (tags, product info, unit conversion)
CREATE OR REPLACE FUNCTION get_inventory_details()
RETURNS TABLE (
item_id UUID,
item_name TEXT,
quantity DECIMAL,
unit_abbreviation TEXT,
unit_name TEXT,
expiry_date DATE,
days_until_expiry INTEGER,
tags TEXT[],
product_brand TEXT,
product_image_url TEXT,
product_barcode TEXT,
created_at TIMESTAMPTZ,
updated_at TIMESTAMPTZ
) AS $$
BEGIN
RETURN QUERY
SELECT
i.id AS item_id,
i.name AS item_name,
i.quantity,
u.abbreviation AS unit_abbreviation,
u.name AS unit_name,
i.expiry_date,
(i.expiry_date - CURRENT_DATE) AS days_until_expiry,
COALESCE(ARRAY_AGG(DISTINCT t.name) FILTER (WHERE t.name IS NOT NULL), '{}') AS tags,
p.brand AS product_brand,
p.image_url AS product_image_url,
p.barcode AS product_barcode,
i.created_at,
i.updated_at
FROM inventory_items i
JOIN units u ON i.unit_id = u.id
LEFT JOIN products p ON i.product_id = p.id
LEFT JOIN item_tags it ON i.id = it.item_id
LEFT JOIN tags t ON it.tag_id = t.id
GROUP BY
i.id, i.name, i.quantity, u.abbreviation, u.name,
i.expiry_date, p.brand, p.image_url, p.barcode,
i.created_at, i.updated_at
ORDER BY i.created_at DESC;
END;
$$ LANGUAGE plpgsql;
COMMENT ON FUNCTION get_inventory_details() IS 'Returns all inventory items with denormalized data for display';
-- Function: Get items expiring soon
CREATE OR REPLACE FUNCTION get_expiring_items(days_ahead INTEGER DEFAULT 7)
RETURNS TABLE (
item_id UUID,
item_name TEXT,
quantity DECIMAL,
unit_abbreviation TEXT,
expiry_date DATE,
days_until_expiry INTEGER,
tags TEXT[]
) AS $$
BEGIN
RETURN QUERY
SELECT
i.id AS item_id,
i.name AS item_name,
i.quantity,
u.abbreviation AS unit_abbreviation,
i.expiry_date,
(i.expiry_date - CURRENT_DATE) AS days_until_expiry,
COALESCE(ARRAY_AGG(DISTINCT t.name) FILTER (WHERE t.name IS NOT NULL), '{}') AS tags
FROM inventory_items i
JOIN units u ON i.unit_id = u.id
LEFT JOIN item_tags it ON i.id = it.item_id
LEFT JOIN tags t ON it.tag_id = t.id
WHERE
i.expiry_date IS NOT NULL
AND i.expiry_date <= CURRENT_DATE + MAKE_INTERVAL(days => days_ahead)
GROUP BY i.id, i.name, i.quantity, u.abbreviation, i.expiry_date
ORDER BY i.expiry_date ASC;
END;
$$ LANGUAGE plpgsql;
COMMENT ON FUNCTION get_expiring_items(INTEGER) IS 'Returns items expiring within specified days (default 7)';
-- Function: Get items by tag
CREATE OR REPLACE FUNCTION get_items_by_tag(tag_name TEXT)
RETURNS TABLE (
item_id UUID,
item_name TEXT,
quantity DECIMAL,
unit_abbreviation TEXT,
expiry_date DATE,
created_at TIMESTAMPTZ
) AS $$
BEGIN
RETURN QUERY
SELECT
i.id AS item_id,
i.name AS item_name,
i.quantity,
u.abbreviation AS unit_abbreviation,
i.expiry_date,
i.created_at
FROM inventory_items i
JOIN units u ON i.unit_id = u.id
JOIN item_tags it ON i.id = it.item_id
JOIN tags t ON it.tag_id = t.id
WHERE t.name ILIKE tag_name
ORDER BY i.created_at DESC;
END;
$$ LANGUAGE plpgsql;
COMMENT ON FUNCTION get_items_by_tag(TEXT) IS 'Returns all items with specified tag (case-insensitive)';
-- Function: Get low stock items (quantity <= threshold)
CREATE OR REPLACE FUNCTION get_low_stock_items(threshold DECIMAL DEFAULT 1.0)
RETURNS TABLE (
item_id UUID,
item_name TEXT,
quantity DECIMAL,
unit_abbreviation TEXT,
tags TEXT[]
) AS $$
BEGIN
RETURN QUERY
SELECT
i.id AS item_id,
i.name AS item_name,
i.quantity,
u.abbreviation AS unit_abbreviation,
COALESCE(ARRAY_AGG(DISTINCT t.name) FILTER (WHERE t.name IS NOT NULL), '{}') AS tags
FROM inventory_items i
JOIN units u ON i.unit_id = u.id
LEFT JOIN item_tags it ON i.id = it.item_id
LEFT JOIN tags t ON it.tag_id = t.id
WHERE i.quantity <= threshold
GROUP BY i.id, i.name, i.quantity, u.abbreviation
ORDER BY i.quantity ASC, i.name ASC;
END;
$$ LANGUAGE plpgsql;
COMMENT ON FUNCTION get_low_stock_items(DECIMAL) IS 'Returns items with quantity at or below threshold';
-- Function: Update item quantity (consume or restock)
CREATE OR REPLACE FUNCTION update_item_quantity(
item_uuid UUID,
quantity_change DECIMAL,
delete_if_zero BOOLEAN DEFAULT TRUE
)
RETURNS BOOLEAN AS $$
DECLARE
new_quantity DECIMAL;
BEGIN
-- Calculate new quantity
SELECT quantity + quantity_change INTO new_quantity
FROM inventory_items
WHERE id = item_uuid;
IF new_quantity IS NULL THEN
RETURN FALSE; -- Item not found
END IF;
-- Delete if zero and flag is set
IF new_quantity <= 0 AND delete_if_zero THEN
DELETE FROM inventory_items WHERE id = item_uuid;
RETURN TRUE;
END IF;
-- Update quantity (ensure non-negative)
UPDATE inventory_items
SET quantity = GREATEST(new_quantity, 0),
updated_at = NOW()
WHERE id = item_uuid;
RETURN TRUE;
END;
$$ LANGUAGE plpgsql;
COMMENT ON FUNCTION update_item_quantity(UUID, DECIMAL, BOOLEAN) IS 'Updates item quantity (positive for restock, negative for consume). Optionally deletes if zero.';
-- Function: Get inventory statistics
CREATE OR REPLACE FUNCTION get_inventory_stats()
RETURNS TABLE (
total_items BIGINT,
total_unique_products BIGINT,
items_expiring_week BIGINT,
items_expired BIGINT,
total_tags_used BIGINT
) AS $$
BEGIN
RETURN QUERY
SELECT
COUNT(DISTINCT i.id) AS total_items,
COUNT(DISTINCT i.product_id) FILTER (WHERE i.product_id IS NOT NULL) AS total_unique_products,
COUNT(i.id) FILTER (
WHERE i.expiry_date IS NOT NULL
AND i.expiry_date BETWEEN CURRENT_DATE AND CURRENT_DATE + INTERVAL '7 days'
) AS items_expiring_week,
COUNT(i.id) FILTER (
WHERE i.expiry_date IS NOT NULL
AND i.expiry_date < CURRENT_DATE
) AS items_expired,
COUNT(DISTINCT it.tag_id) AS total_tags_used
FROM inventory_items i
LEFT JOIN item_tags it ON i.id = it.item_id;
END;
$$ LANGUAGE plpgsql;
COMMENT ON FUNCTION get_inventory_stats() IS 'Returns summary statistics for the entire inventory';
-- Function: Search inventory (full-text search on items and products)
CREATE OR REPLACE FUNCTION search_inventory(search_query TEXT)
RETURNS TABLE (
item_id UUID,
item_name TEXT,
quantity DECIMAL,
unit_abbreviation TEXT,
product_brand TEXT,
tags TEXT[],
relevance REAL
) AS $$
BEGIN
RETURN QUERY
SELECT
i.id AS item_id,
i.name AS item_name,
i.quantity,
u.abbreviation AS unit_abbreviation,
p.brand AS product_brand,
COALESCE(ARRAY_AGG(DISTINCT t.name) FILTER (WHERE t.name IS NOT NULL), '{}') AS tags,
ts_rank(
to_tsvector('english', i.name || ' ' || COALESCE(p.brand, '') || ' ' || COALESCE(p.name, '')),
plainto_tsquery('english', search_query)
) AS relevance
FROM inventory_items i
JOIN units u ON i.unit_id = u.id
LEFT JOIN products p ON i.product_id = p.id
LEFT JOIN item_tags it ON i.id = it.item_id
LEFT JOIN tags t ON it.tag_id = t.id
WHERE
to_tsvector('english', i.name || ' ' || COALESCE(p.brand, '') || ' ' || COALESCE(p.name, ''))
@@ plainto_tsquery('english', search_query)
GROUP BY i.id, i.name, i.quantity, u.abbreviation, p.brand, p.name
ORDER BY relevance DESC, i.created_at DESC
LIMIT 50;
END;
$$ LANGUAGE plpgsql;
COMMENT ON FUNCTION search_inventory(TEXT) IS 'Full-text search across inventory items and products';

View File

@@ -0,0 +1,37 @@
-- Migration: Seed Default Units
-- Week 2: Pre-populate common measurement units with conversions
-- Weight units (metric base: gram)
INSERT INTO units (id, name, abbreviation, unit_type, base_unit_id, conversion_factor, is_default, created_by) VALUES
('f47ac10b-58cc-4372-a567-0e02b2c3d479', 'Gram', 'g', 'weight', NULL, 1.0, TRUE, NULL),
('550e8400-e29b-41d4-a716-446655440001', 'Kilogram', 'kg', 'weight', 'f47ac10b-58cc-4372-a567-0e02b2c3d479', 1000.0, FALSE, NULL),
('550e8400-e29b-41d4-a716-446655440002', 'Milligram', 'mg', 'weight', 'f47ac10b-58cc-4372-a567-0e02b2c3d479', 0.001, FALSE, NULL),
('550e8400-e29b-41d4-a716-446655440003', 'Pound', 'lb', 'weight', 'f47ac10b-58cc-4372-a567-0e02b2c3d479', 453.592, FALSE, NULL),
('550e8400-e29b-41d4-a716-446655440004', 'Ounce', 'oz', 'weight', 'f47ac10b-58cc-4372-a567-0e02b2c3d479', 28.3495, FALSE, NULL);
-- Volume units (metric base: milliliter)
INSERT INTO units (id, name, abbreviation, unit_type, base_unit_id, conversion_factor, is_default, created_by) VALUES
('550e8400-e29b-41d4-a716-446655440010', 'Milliliter', 'mL', 'volume', NULL, 1.0, TRUE, NULL),
('550e8400-e29b-41d4-a716-446655440011', 'Liter', 'L', 'volume', '550e8400-e29b-41d4-a716-446655440010', 1000.0, FALSE, NULL),
('550e8400-e29b-41d4-a716-446655440012', 'Centiliter', 'cL', 'volume', '550e8400-e29b-41d4-a716-446655440010', 10.0, FALSE, NULL),
('550e8400-e29b-41d4-a716-446655440013', 'Deciliter', 'dL', 'volume', '550e8400-e29b-41d4-a716-446655440010', 100.0, FALSE, NULL),
('550e8400-e29b-41d4-a716-446655440014', 'Cup', 'cup', 'volume', '550e8400-e29b-41d4-a716-446655440010', 236.588, FALSE, NULL),
('550e8400-e29b-41d4-a716-446655440015', 'Tablespoon', 'tbsp', 'volume', '550e8400-e29b-41d4-a716-446655440010', 14.7868, FALSE, NULL),
('550e8400-e29b-41d4-a716-446655440016', 'Teaspoon', 'tsp', 'volume', '550e8400-e29b-41d4-a716-446655440010', 4.92892, FALSE, NULL),
('550e8400-e29b-41d4-a716-446655440017', 'Fluid Ounce', 'fl oz', 'volume', '550e8400-e29b-41d4-a716-446655440010', 29.5735, FALSE, NULL),
('550e8400-e29b-41d4-a716-446655440018', 'Gallon', 'gal', 'volume', '550e8400-e29b-41d4-a716-446655440010', 3785.41, FALSE, NULL),
('550e8400-e29b-41d4-a716-446655440019', 'Quart', 'qt', 'volume', '550e8400-e29b-41d4-a716-446655440010', 946.353, FALSE, NULL),
('550e8400-e29b-41d4-a716-446655440020', 'Pint', 'pt', 'volume', '550e8400-e29b-41d4-a716-446655440010', 473.176, FALSE, NULL);
-- Count units (no conversions, each is independent)
INSERT INTO units (id, name, abbreviation, unit_type, base_unit_id, conversion_factor, is_default, created_by) VALUES
('550e8400-e29b-41d4-a716-446655440030', 'Piece', 'pc', 'count', NULL, 1.0, TRUE, NULL),
('550e8400-e29b-41d4-a716-446655440031', 'Dozen', 'doz', 'count', '550e8400-e29b-41d4-a716-446655440030', 12.0, FALSE, NULL),
('550e8400-e29b-41d4-a716-446655440032', 'Package', 'pkg', 'count', NULL, 1.0, FALSE, NULL),
('550e8400-e29b-41d4-a716-446655440033', 'Bottle', 'btl', 'count', NULL, 1.0, FALSE, NULL),
('550e8400-e29b-41d4-a716-446655440034', 'Can', 'can', 'count', NULL, 1.0, FALSE, NULL),
('550e8400-e29b-41d4-a716-446655440035', 'Jar', 'jar', 'count', NULL, 1.0, FALSE, NULL),
('550e8400-e29b-41d4-a716-446655440036', 'Box', 'box', 'count', NULL, 1.0, FALSE, NULL),
('550e8400-e29b-41d4-a716-446655440037', 'Bag', 'bag', 'count', NULL, 1.0, FALSE, NULL);
COMMENT ON TABLE units IS 'Measurement units with 30 common presets covering metric, imperial, and count units';

View File

@@ -0,0 +1,49 @@
-- Migration: Seed Default Tags
-- Week 2: Pre-populate common organizational tags
-- Position Tags (where items are stored)
INSERT INTO tags (id, name, category, icon, color, created_by) VALUES
('650e8400-e29b-41d4-a716-446655440001', 'Fridge', 'position', '🧊', '#3b82f6', NULL),
('650e8400-e29b-41d4-a716-446655440002', 'Freezer', 'position', '❄️', '#06b6d4', NULL),
('650e8400-e29b-41d4-a716-446655440003', 'Pantry', 'position', '🗄️', '#8b5cf6', NULL),
('650e8400-e29b-41d4-a716-446655440004', 'Cabinet', 'position', '🚪', '#6b7280', NULL),
('650e8400-e29b-41d4-a716-446655440005', 'Countertop', 'position', '🍽️', '#f59e0b', NULL),
('650e8400-e29b-41d4-a716-446655440006', 'Cellar', 'position', '🏚️', '#78350f', NULL);
-- Type Tags (food categories)
INSERT INTO tags (id, name, category, icon, color, created_by) VALUES
('650e8400-e29b-41d4-a716-446655440010', 'Dairy', 'type', '🧀', '#fbbf24', NULL),
('650e8400-e29b-41d4-a716-446655440011', 'Meat', 'type', '🥩', '#ef4444', NULL),
('650e8400-e29b-41d4-a716-446655440012', 'Fish', 'type', '🐟', '#3b82f6', NULL),
('650e8400-e29b-41d4-a716-446655440013', 'Vegetables', 'type', '🥬', '#22c55e', NULL),
('650e8400-e29b-41d4-a716-446655440014', 'Fruits', 'type', '🍎', '#f97316', NULL),
('650e8400-e29b-41d4-a716-446655440015', 'Grains', 'type', '🌾', '#eab308', NULL),
('650e8400-e29b-41d4-a716-446655440016', 'Legumes', 'type', '🫘', '#84cc16', NULL),
('650e8400-e29b-41d4-a716-446655440017', 'Condiments', 'type', '🧂', '#ef4444', NULL),
('650e8400-e29b-41d4-a716-446655440018', 'Snacks', 'type', '🍿', '#f97316', NULL),
('650e8400-e29b-41d4-a716-446655440019', 'Beverages', 'type', '🥤', '#06b6d4', NULL),
('650e8400-e29b-41d4-a716-446655440020', 'Baking', 'type', '🧁', '#ec4899', NULL),
('650e8400-e29b-41d4-a716-446655440021', 'Spices', 'type', '🌶️', '#dc2626', NULL),
('650e8400-e29b-41d4-a716-446655440022', 'Canned', 'type', '🥫', '#71717a', NULL),
('650e8400-e29b-41d4-a716-446655440023', 'Frozen', 'type', '🧊', '#06b6d4', NULL);
-- Dietary Tags
INSERT INTO tags (id, name, category, icon, color, created_by) VALUES
('650e8400-e29b-41d4-a716-446655440030', 'Vegan', 'dietary', '🌱', '#22c55e', NULL),
('650e8400-e29b-41d4-a716-446655440031', 'Vegetarian', 'dietary', '🥕', '#84cc16', NULL),
('650e8400-e29b-41d4-a716-446655440032', 'Gluten-Free', 'dietary', '🌾', '#eab308', NULL),
('650e8400-e29b-41d4-a716-446655440033', 'Lactose-Free', 'dietary', '🥛', '#60a5fa', NULL),
('650e8400-e29b-41d4-a716-446655440034', 'Organic', 'dietary', '♻️', '#10b981', NULL),
('650e8400-e29b-41d4-a716-446655440035', 'Low-Carb', 'dietary', '🥗', '#22c55e', NULL),
('650e8400-e29b-41d4-a716-446655440036', 'Kosher', 'dietary', '✡️', '#3b82f6', NULL),
('650e8400-e29b-41d4-a716-446655440037', 'Halal', 'dietary', '☪️', '#22c55e', NULL);
-- Custom/Workflow Tags
INSERT INTO tags (id, name, category, icon, color, created_by) VALUES
('650e8400-e29b-41d4-a716-446655440040', 'Low Stock', 'custom', '⚠️', '#ef4444', NULL),
('650e8400-e29b-41d4-a716-446655440041', 'To Buy', 'custom', '🛒', '#3b82f6', NULL),
('650e8400-e29b-41d4-a716-446655440042', 'Meal Prep', 'custom', '🍱', '#8b5cf6', NULL),
('650e8400-e29b-41d4-a716-446655440043', 'Leftovers', 'custom', '♻️', '#f59e0b', NULL),
('650e8400-e29b-41d4-a716-446655440044', 'Opening Soon', 'custom', '📆', '#f97316', NULL);
COMMENT ON TABLE tags IS 'Pre-populated with 33 common tags across position, type, dietary, and workflow categories';