Compare commits
8 Commits
436f92cafc
...
feature/is
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
4834286005 | ||
| be2af1675a | |||
|
|
b93f4677fc | ||
| 4eec4923af | |||
|
|
f70b90748a | ||
| 1c54415a29 | |||
|
|
01c5880e37 | ||
|
|
223f4b6ea1 |
3
app/.env.example
Normal file
3
app/.env.example
Normal file
@@ -0,0 +1,3 @@
|
||||
# Supabase Configuration
|
||||
NUXT_PUBLIC_SUPABASE_URL=http://localhost:54321
|
||||
NUXT_PUBLIC_SUPABASE_ANON_KEY=your-anon-key-here
|
||||
24
app/.gitignore
vendored
Normal file
24
app/.gitignore
vendored
Normal file
@@ -0,0 +1,24 @@
|
||||
# Nuxt dev/build outputs
|
||||
.output
|
||||
.data
|
||||
.nuxt
|
||||
.nitro
|
||||
.cache
|
||||
dist
|
||||
|
||||
# Node dependencies
|
||||
node_modules
|
||||
|
||||
# Logs
|
||||
logs
|
||||
*.log
|
||||
|
||||
# Misc
|
||||
.DS_Store
|
||||
.fleet
|
||||
.idea
|
||||
|
||||
# Local env files
|
||||
.env
|
||||
.env.*
|
||||
!.env.example
|
||||
75
app/README.md
Normal file
75
app/README.md
Normal file
@@ -0,0 +1,75 @@
|
||||
# Nuxt Minimal Starter
|
||||
|
||||
Look at the [Nuxt documentation](https://nuxt.com/docs/getting-started/introduction) to learn more.
|
||||
|
||||
## Setup
|
||||
|
||||
Make sure to install dependencies:
|
||||
|
||||
```bash
|
||||
# npm
|
||||
npm install
|
||||
|
||||
# pnpm
|
||||
pnpm install
|
||||
|
||||
# yarn
|
||||
yarn install
|
||||
|
||||
# bun
|
||||
bun install
|
||||
```
|
||||
|
||||
## Development Server
|
||||
|
||||
Start the development server on `http://localhost:3000`:
|
||||
|
||||
```bash
|
||||
# npm
|
||||
npm run dev
|
||||
|
||||
# pnpm
|
||||
pnpm dev
|
||||
|
||||
# yarn
|
||||
yarn dev
|
||||
|
||||
# bun
|
||||
bun run dev
|
||||
```
|
||||
|
||||
## Production
|
||||
|
||||
Build the application for production:
|
||||
|
||||
```bash
|
||||
# npm
|
||||
npm run build
|
||||
|
||||
# pnpm
|
||||
pnpm build
|
||||
|
||||
# yarn
|
||||
yarn build
|
||||
|
||||
# bun
|
||||
bun run build
|
||||
```
|
||||
|
||||
Locally preview production build:
|
||||
|
||||
```bash
|
||||
# npm
|
||||
npm run preview
|
||||
|
||||
# pnpm
|
||||
pnpm preview
|
||||
|
||||
# yarn
|
||||
yarn preview
|
||||
|
||||
# bun
|
||||
bun run preview
|
||||
```
|
||||
|
||||
Check out the [deployment documentation](https://nuxt.com/docs/getting-started/deployment) for more information.
|
||||
5
app/app/app.vue
Normal file
5
app/app/app.vue
Normal file
@@ -0,0 +1,5 @@
|
||||
<template>
|
||||
<NuxtLayout>
|
||||
<NuxtPage />
|
||||
</NuxtLayout>
|
||||
</template>
|
||||
15
app/app/layouts/default.vue
Normal file
15
app/app/layouts/default.vue
Normal file
@@ -0,0 +1,15 @@
|
||||
<template>
|
||||
<div class="min-h-screen bg-gray-50">
|
||||
<AppHeader />
|
||||
|
||||
<main class="container mx-auto px-4 py-6 max-w-7xl">
|
||||
<slot />
|
||||
</main>
|
||||
|
||||
<AppFooter />
|
||||
</div>
|
||||
</template>
|
||||
|
||||
<script setup lang="ts">
|
||||
// App layout automatically wraps all pages
|
||||
</script>
|
||||
2259
app/bun.lock
Normal file
2259
app/bun.lock
Normal file
File diff suppressed because it is too large
Load Diff
40
app/components/AppFooter.vue
Normal file
40
app/components/AppFooter.vue
Normal file
@@ -0,0 +1,40 @@
|
||||
<template>
|
||||
<footer class="bg-white border-t border-gray-200 mt-auto">
|
||||
<div class="container mx-auto px-4 py-6 max-w-7xl">
|
||||
<div class="flex flex-col md:flex-row items-center justify-between gap-4">
|
||||
<!-- Copyright -->
|
||||
<p class="text-sm text-gray-600">
|
||||
© {{ currentYear }} Pantry. Self-hosted inventory management.
|
||||
</p>
|
||||
|
||||
<!-- Links -->
|
||||
<div class="flex items-center space-x-6">
|
||||
<a
|
||||
href="https://github.com/pantry-app/pantry"
|
||||
target="_blank"
|
||||
rel="noopener noreferrer"
|
||||
class="text-sm text-gray-600 hover:text-primary transition-colors"
|
||||
>
|
||||
GitHub
|
||||
</a>
|
||||
<NuxtLink
|
||||
to="/about"
|
||||
class="text-sm text-gray-600 hover:text-primary transition-colors"
|
||||
>
|
||||
About
|
||||
</NuxtLink>
|
||||
<NuxtLink
|
||||
to="/privacy"
|
||||
class="text-sm text-gray-600 hover:text-primary transition-colors"
|
||||
>
|
||||
Privacy
|
||||
</NuxtLink>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</footer>
|
||||
</template>
|
||||
|
||||
<script setup lang="ts">
|
||||
const currentYear = new Date().getFullYear()
|
||||
</script>
|
||||
122
app/components/AppHeader.vue
Normal file
122
app/components/AppHeader.vue
Normal file
@@ -0,0 +1,122 @@
|
||||
<template>
|
||||
<header class="bg-white border-b border-gray-200 sticky top-0 z-50">
|
||||
<div class="container mx-auto px-4 max-w-7xl">
|
||||
<div class="flex items-center justify-between h-16">
|
||||
<!-- Logo / Brand -->
|
||||
<NuxtLink to="/" class="flex items-center space-x-2">
|
||||
<UIcon name="i-heroicons-squares-2x2" class="w-8 h-8 text-primary" />
|
||||
<span class="text-xl font-bold text-gray-900">Pantry</span>
|
||||
</NuxtLink>
|
||||
|
||||
<!-- Navigation -->
|
||||
<nav class="hidden md:flex items-center space-x-6">
|
||||
<NuxtLink
|
||||
to="/"
|
||||
class="text-gray-700 hover:text-primary transition-colors"
|
||||
active-class="text-primary font-semibold"
|
||||
>
|
||||
Inventory
|
||||
</NuxtLink>
|
||||
<NuxtLink
|
||||
to="/scan"
|
||||
class="text-gray-700 hover:text-primary transition-colors"
|
||||
active-class="text-primary font-semibold"
|
||||
>
|
||||
Scan
|
||||
</NuxtLink>
|
||||
<NuxtLink
|
||||
to="/settings"
|
||||
class="text-gray-700 hover:text-primary transition-colors"
|
||||
active-class="text-primary font-semibold"
|
||||
>
|
||||
Settings
|
||||
</NuxtLink>
|
||||
</nav>
|
||||
|
||||
<!-- User Menu -->
|
||||
<div class="flex items-center space-x-4">
|
||||
<UButton
|
||||
v-if="!user"
|
||||
to="/auth/login"
|
||||
color="primary"
|
||||
variant="soft"
|
||||
>
|
||||
Sign In
|
||||
</UButton>
|
||||
|
||||
<UDropdown v-else :items="userMenuItems" mode="hover">
|
||||
<UAvatar
|
||||
:alt="user.email"
|
||||
size="sm"
|
||||
:ui="{ background: 'bg-primary' }"
|
||||
/>
|
||||
</UDropdown>
|
||||
</div>
|
||||
|
||||
<!-- Mobile Menu Button -->
|
||||
<UButton
|
||||
icon="i-heroicons-bars-3"
|
||||
color="gray"
|
||||
variant="ghost"
|
||||
class="md:hidden"
|
||||
@click="mobileMenuOpen = !mobileMenuOpen"
|
||||
/>
|
||||
</div>
|
||||
|
||||
<!-- Mobile Navigation -->
|
||||
<nav
|
||||
v-if="mobileMenuOpen"
|
||||
class="md:hidden py-4 space-y-2 border-t border-gray-200"
|
||||
>
|
||||
<NuxtLink
|
||||
to="/"
|
||||
class="block px-4 py-2 text-gray-700 hover:bg-gray-50 rounded"
|
||||
@click="mobileMenuOpen = false"
|
||||
>
|
||||
Inventory
|
||||
</NuxtLink>
|
||||
<NuxtLink
|
||||
to="/scan"
|
||||
class="block px-4 py-2 text-gray-700 hover:bg-gray-50 rounded"
|
||||
@click="mobileMenuOpen = false"
|
||||
>
|
||||
Scan
|
||||
</NuxtLink>
|
||||
<NuxtLink
|
||||
to="/settings"
|
||||
class="block px-4 py-2 text-gray-700 hover:bg-gray-50 rounded"
|
||||
@click="mobileMenuOpen = false"
|
||||
>
|
||||
Settings
|
||||
</NuxtLink>
|
||||
</nav>
|
||||
</div>
|
||||
</header>
|
||||
</template>
|
||||
|
||||
<script setup lang="ts">
|
||||
const { user, signOut } = useSupabaseAuth()
|
||||
const mobileMenuOpen = ref(false)
|
||||
|
||||
const userMenuItems = [[
|
||||
{
|
||||
label: user.value?.email || 'User',
|
||||
slot: 'account',
|
||||
disabled: true
|
||||
}
|
||||
], [
|
||||
{
|
||||
label: 'Settings',
|
||||
icon: 'i-heroicons-cog-6-tooth',
|
||||
to: '/settings'
|
||||
},
|
||||
{
|
||||
label: 'Sign Out',
|
||||
icon: 'i-heroicons-arrow-right-on-rectangle',
|
||||
click: async () => {
|
||||
await signOut()
|
||||
navigateTo('/auth/login')
|
||||
}
|
||||
}
|
||||
]]
|
||||
</script>
|
||||
257
app/components/inventory/AddItemForm.vue
Normal file
257
app/components/inventory/AddItemForm.vue
Normal 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>
|
||||
184
app/components/inventory/EditItemModal.vue
Normal file
184
app/components/inventory/EditItemModal.vue
Normal file
@@ -0,0 +1,184 @@
|
||||
<template>
|
||||
<UModal v-model="isOpen">
|
||||
<UCard>
|
||||
<template #header>
|
||||
<div class="flex items-center justify-between">
|
||||
<h3 class="text-lg font-semibold">Edit Item</h3>
|
||||
<UButton
|
||||
icon="i-heroicons-x-mark"
|
||||
color="gray"
|
||||
variant="ghost"
|
||||
@click="close"
|
||||
/>
|
||||
</div>
|
||||
</template>
|
||||
|
||||
<form @submit.prevent="handleSubmit" class="space-y-4">
|
||||
<!-- Item Name -->
|
||||
<UFormGroup label="Item Name" required>
|
||||
<UInput
|
||||
v-model="form.name"
|
||||
placeholder="Item name"
|
||||
size="lg"
|
||||
/>
|
||||
</UFormGroup>
|
||||
|
||||
<!-- Quantity & Unit -->
|
||||
<div class="grid grid-cols-2 gap-3">
|
||||
<UFormGroup label="Quantity" required>
|
||||
<UInput
|
||||
v-model.number="form.quantity"
|
||||
type="number"
|
||||
min="0.01"
|
||||
step="0.01"
|
||||
size="lg"
|
||||
/>
|
||||
</UFormGroup>
|
||||
|
||||
<UFormGroup label="Unit" required>
|
||||
<USelect
|
||||
v-model="form.unit_id"
|
||||
:options="unitOptions"
|
||||
option-attribute="label"
|
||||
value-attribute="value"
|
||||
size="lg"
|
||||
/>
|
||||
</UFormGroup>
|
||||
</div>
|
||||
|
||||
<!-- Expiry Date -->
|
||||
<UFormGroup label="Expiry Date" hint="Optional">
|
||||
<UInput
|
||||
v-model="form.expiry_date"
|
||||
type="date"
|
||||
size="lg"
|
||||
/>
|
||||
</UFormGroup>
|
||||
|
||||
<!-- Notes -->
|
||||
<UFormGroup label="Notes" hint="Optional">
|
||||
<UTextarea
|
||||
v-model="form.notes"
|
||||
placeholder="Any additional notes..."
|
||||
:rows="2"
|
||||
/>
|
||||
</UFormGroup>
|
||||
|
||||
<!-- Submit -->
|
||||
<div class="flex gap-2 pt-2">
|
||||
<UButton
|
||||
type="submit"
|
||||
color="primary"
|
||||
size="lg"
|
||||
class="flex-1"
|
||||
:loading="submitting"
|
||||
:disabled="!isValid"
|
||||
>
|
||||
Save Changes
|
||||
</UButton>
|
||||
<UButton
|
||||
color="gray"
|
||||
size="lg"
|
||||
variant="soft"
|
||||
@click="close"
|
||||
>
|
||||
Cancel
|
||||
</UButton>
|
||||
</div>
|
||||
</form>
|
||||
</UCard>
|
||||
</UModal>
|
||||
</template>
|
||||
|
||||
<script setup lang="ts">
|
||||
const { updateInventoryItem } = useInventory()
|
||||
const { getUnits } = useUnits()
|
||||
|
||||
const props = defineProps<{
|
||||
item: any | null
|
||||
}>()
|
||||
|
||||
const emit = defineEmits<{
|
||||
close: []
|
||||
updated: [item: any]
|
||||
}>()
|
||||
|
||||
const isOpen = ref(false)
|
||||
const submitting = ref(false)
|
||||
const units = ref<any[]>([])
|
||||
|
||||
const form = reactive({
|
||||
name: '',
|
||||
quantity: 1,
|
||||
unit_id: '',
|
||||
expiry_date: '',
|
||||
notes: ''
|
||||
})
|
||||
|
||||
// Load units
|
||||
onMounted(async () => {
|
||||
const { data } = await getUnits()
|
||||
units.value = data || []
|
||||
})
|
||||
|
||||
// Unit options for select
|
||||
const unitOptions = computed(() => {
|
||||
return units.value.map(unit => ({
|
||||
label: `${unit.name} (${unit.abbreviation})`,
|
||||
value: unit.id
|
||||
}))
|
||||
})
|
||||
|
||||
// Watch for item changes (open modal)
|
||||
watch(() => props.item, (newItem) => {
|
||||
if (newItem) {
|
||||
form.name = newItem.name
|
||||
form.quantity = Number(newItem.quantity)
|
||||
form.unit_id = newItem.unit_id
|
||||
form.expiry_date = newItem.expiry_date || ''
|
||||
form.notes = newItem.notes || ''
|
||||
isOpen.value = true
|
||||
}
|
||||
}, { immediate: true })
|
||||
|
||||
// Watch modal close
|
||||
watch(isOpen, (val) => {
|
||||
if (!val) {
|
||||
emit('close')
|
||||
}
|
||||
})
|
||||
|
||||
// Validation
|
||||
const isValid = computed(() => {
|
||||
return form.name.trim().length > 0 && form.quantity > 0 && form.unit_id
|
||||
})
|
||||
|
||||
const close = () => {
|
||||
isOpen.value = false
|
||||
}
|
||||
|
||||
// Submit
|
||||
const handleSubmit = async () => {
|
||||
if (!isValid.value || !props.item) return
|
||||
|
||||
submitting.value = true
|
||||
|
||||
const { data, error } = await updateInventoryItem(props.item.id, {
|
||||
name: form.name.trim(),
|
||||
quantity: form.quantity,
|
||||
unit_id: form.unit_id,
|
||||
expiry_date: form.expiry_date || null,
|
||||
notes: form.notes.trim() || null
|
||||
})
|
||||
|
||||
if (error) {
|
||||
alert('Failed to update item: ' + error.message)
|
||||
submitting.value = false
|
||||
return
|
||||
}
|
||||
|
||||
emit('updated', data)
|
||||
submitting.value = false
|
||||
close()
|
||||
}
|
||||
</script>
|
||||
151
app/components/inventory/InventoryCard.vue
Normal file
151
app/components/inventory/InventoryCard.vue
Normal 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>
|
||||
131
app/components/inventory/InventoryList.vue
Normal file
131
app/components/inventory/InventoryList.vue
Normal 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>
|
||||
201
app/composables/useInventory.ts
Normal file
201
app/composables/useInventory.ts
Normal file
@@ -0,0 +1,201 @@
|
||||
import type { Database } from '~/types/database.types'
|
||||
|
||||
type InventoryItem = Database['public']['Tables']['inventory_items']['Row']
|
||||
type InventoryItemInsert = Database['public']['Tables']['inventory_items']['Insert']
|
||||
type InventoryItemUpdate = Database['public']['Tables']['inventory_items']['Update']
|
||||
|
||||
export const useInventory = () => {
|
||||
const supabase = useSupabase()
|
||||
const { user } = useSupabaseAuth()
|
||||
|
||||
/**
|
||||
* Get all inventory items with denormalized data
|
||||
*/
|
||||
const getInventory = async () => {
|
||||
const { data, error } = await supabase
|
||||
.from('inventory_items')
|
||||
.select(`
|
||||
*,
|
||||
product:products(*),
|
||||
unit:units(*),
|
||||
tags:item_tags(tag:tags(*))
|
||||
`)
|
||||
.order('created_at', { ascending: false })
|
||||
|
||||
if (error) {
|
||||
console.error('Error fetching inventory:', error)
|
||||
return { data: null, error }
|
||||
}
|
||||
|
||||
return { data, error: null }
|
||||
}
|
||||
|
||||
/**
|
||||
* Get single inventory item by ID
|
||||
*/
|
||||
const getInventoryItem = async (id: string) => {
|
||||
const { data, error } = await supabase
|
||||
.from('inventory_items')
|
||||
.select(`
|
||||
*,
|
||||
product:products(*),
|
||||
unit:units(*),
|
||||
tags:item_tags(tag:tags(*))
|
||||
`)
|
||||
.eq('id', id)
|
||||
.single()
|
||||
|
||||
if (error) {
|
||||
console.error('Error fetching item:', error)
|
||||
return { data: null, error }
|
||||
}
|
||||
|
||||
return { data, error: null }
|
||||
}
|
||||
|
||||
/**
|
||||
* Add new inventory item
|
||||
*/
|
||||
const addInventoryItem = async (item: Omit<InventoryItemInsert, 'added_by'>) => {
|
||||
if (!user.value) {
|
||||
return { data: null, error: { message: 'User not authenticated' } }
|
||||
}
|
||||
|
||||
const { data, error } = await supabase
|
||||
.from('inventory_items')
|
||||
.insert({
|
||||
...item,
|
||||
added_by: user.value.id
|
||||
})
|
||||
.select(`
|
||||
*,
|
||||
product:products(*),
|
||||
unit:units(*),
|
||||
tags:item_tags(tag:tags(*))
|
||||
`)
|
||||
.single()
|
||||
|
||||
if (error) {
|
||||
console.error('Error adding item:', error)
|
||||
return { data: null, error }
|
||||
}
|
||||
|
||||
return { data, error: null }
|
||||
}
|
||||
|
||||
/**
|
||||
* Update inventory item
|
||||
*/
|
||||
const updateInventoryItem = async (id: string, updates: InventoryItemUpdate) => {
|
||||
const { data, error } = await supabase
|
||||
.from('inventory_items')
|
||||
.update(updates)
|
||||
.eq('id', id)
|
||||
.select(`
|
||||
*,
|
||||
product:products(*),
|
||||
unit:units(*),
|
||||
tags:item_tags(tag:tags(*))
|
||||
`)
|
||||
.single()
|
||||
|
||||
if (error) {
|
||||
console.error('Error updating item:', error)
|
||||
return { data: null, error }
|
||||
}
|
||||
|
||||
return { data, error: null }
|
||||
}
|
||||
|
||||
/**
|
||||
* Delete inventory item
|
||||
*/
|
||||
const deleteInventoryItem = async (id: string) => {
|
||||
// First delete associated tags
|
||||
await supabase
|
||||
.from('item_tags')
|
||||
.delete()
|
||||
.eq('item_id', id)
|
||||
|
||||
const { error } = await supabase
|
||||
.from('inventory_items')
|
||||
.delete()
|
||||
.eq('id', id)
|
||||
|
||||
if (error) {
|
||||
console.error('Error deleting item:', error)
|
||||
return { error }
|
||||
}
|
||||
|
||||
return { error: null }
|
||||
}
|
||||
|
||||
/**
|
||||
* Update item quantity (consume or restock)
|
||||
*/
|
||||
const updateQuantity = async (id: string, change: number) => {
|
||||
const { data: item, error: fetchError } = await getInventoryItem(id)
|
||||
if (fetchError || !item) {
|
||||
return { data: null, error: fetchError }
|
||||
}
|
||||
|
||||
const newQuantity = Number(item.quantity) + change
|
||||
|
||||
if (newQuantity <= 0) {
|
||||
// Auto-delete when quantity reaches zero
|
||||
return await deleteInventoryItem(id)
|
||||
}
|
||||
|
||||
return await updateInventoryItem(id, { quantity: newQuantity })
|
||||
}
|
||||
|
||||
/**
|
||||
* Add tags to item
|
||||
*/
|
||||
const addItemTags = async (itemId: string, tagIds: string[]) => {
|
||||
const items = tagIds.map(tagId => ({
|
||||
item_id: itemId,
|
||||
tag_id: tagId
|
||||
}))
|
||||
|
||||
const { error } = await supabase
|
||||
.from('item_tags')
|
||||
.insert(items)
|
||||
|
||||
if (error) {
|
||||
console.error('Error adding tags:', error)
|
||||
return { error }
|
||||
}
|
||||
|
||||
return { error: null }
|
||||
}
|
||||
|
||||
/**
|
||||
* Remove tag from item
|
||||
*/
|
||||
const removeItemTag = async (itemId: string, tagId: string) => {
|
||||
const { error } = await supabase
|
||||
.from('item_tags')
|
||||
.delete()
|
||||
.eq('item_id', itemId)
|
||||
.eq('tag_id', tagId)
|
||||
|
||||
if (error) {
|
||||
console.error('Error removing tag:', error)
|
||||
return { error }
|
||||
}
|
||||
|
||||
return { error: null }
|
||||
}
|
||||
|
||||
return {
|
||||
getInventory,
|
||||
getInventoryItem,
|
||||
addInventoryItem,
|
||||
updateInventoryItem,
|
||||
deleteInventoryItem,
|
||||
updateQuantity,
|
||||
addItemTags,
|
||||
removeItemTag
|
||||
}
|
||||
}
|
||||
81
app/composables/useSupabase.ts
Normal file
81
app/composables/useSupabase.ts
Normal file
@@ -0,0 +1,81 @@
|
||||
import { createClient, SupabaseClient } from '@supabase/supabase-js'
|
||||
import type { Database } from '~/types/database.types'
|
||||
|
||||
let supabaseInstance: SupabaseClient<Database> | null = null
|
||||
|
||||
export const useSupabase = () => {
|
||||
const config = useRuntimeConfig()
|
||||
|
||||
if (!supabaseInstance) {
|
||||
supabaseInstance = createClient<Database>(
|
||||
config.public.supabaseUrl,
|
||||
config.public.supabaseAnonKey,
|
||||
{
|
||||
auth: {
|
||||
persistSession: true,
|
||||
autoRefreshToken: true,
|
||||
detectSessionInUrl: true,
|
||||
storage: process.client ? window.localStorage : undefined
|
||||
}
|
||||
}
|
||||
)
|
||||
}
|
||||
|
||||
return supabaseInstance
|
||||
}
|
||||
|
||||
/**
|
||||
* Composable for Supabase authentication
|
||||
*/
|
||||
export const useSupabaseAuth = () => {
|
||||
const supabase = useSupabase()
|
||||
const user = useState('supabase_user', () => null as any)
|
||||
const session = useState('supabase_session', () => null as any)
|
||||
|
||||
// Initialize auth state
|
||||
const initAuth = async () => {
|
||||
const { data: { session: currentSession } } = await supabase.auth.getSession()
|
||||
session.value = currentSession
|
||||
user.value = currentSession?.user || null
|
||||
|
||||
// Listen for auth changes
|
||||
supabase.auth.onAuthStateChange((_event, newSession) => {
|
||||
session.value = newSession
|
||||
user.value = newSession?.user || null
|
||||
})
|
||||
}
|
||||
|
||||
// Call initAuth on composable mount
|
||||
if (process.client) {
|
||||
initAuth()
|
||||
}
|
||||
|
||||
const signIn = async (email: string, password: string) => {
|
||||
const { data, error } = await supabase.auth.signInWithPassword({
|
||||
email,
|
||||
password
|
||||
})
|
||||
return { data, error }
|
||||
}
|
||||
|
||||
const signUp = async (email: string, password: string) => {
|
||||
const { data, error } = await supabase.auth.signUp({
|
||||
email,
|
||||
password
|
||||
})
|
||||
return { data, error }
|
||||
}
|
||||
|
||||
const signOut = async () => {
|
||||
const { error } = await supabase.auth.signOut()
|
||||
return { error }
|
||||
}
|
||||
|
||||
return {
|
||||
user: readonly(user),
|
||||
session: readonly(session),
|
||||
signIn,
|
||||
signUp,
|
||||
signOut
|
||||
}
|
||||
}
|
||||
44
app/composables/useTags.ts
Normal file
44
app/composables/useTags.ts
Normal 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
|
||||
}
|
||||
}
|
||||
53
app/composables/useUnits.ts
Normal file
53
app/composables/useUnits.ts
Normal file
@@ -0,0 +1,53 @@
|
||||
export const useUnits = () => {
|
||||
const supabase = useSupabase()
|
||||
|
||||
/**
|
||||
* Get all units
|
||||
*/
|
||||
const getUnits = async () => {
|
||||
const { data, error } = await supabase
|
||||
.from('units')
|
||||
.select('*')
|
||||
.order('unit_type', { ascending: true })
|
||||
.order('name', { ascending: true })
|
||||
|
||||
if (error) {
|
||||
console.error('Error fetching units:', error)
|
||||
return { data: null, error }
|
||||
}
|
||||
|
||||
return { data, error: null }
|
||||
}
|
||||
|
||||
/**
|
||||
* Get default unit for a type
|
||||
*/
|
||||
const getDefaultUnit = async (unitType: 'weight' | 'volume' | 'count' | 'custom') => {
|
||||
const { data, error } = await supabase
|
||||
.from('units')
|
||||
.select('*')
|
||||
.eq('unit_type', unitType)
|
||||
.eq('is_default', true)
|
||||
.single()
|
||||
|
||||
if (error) {
|
||||
console.error('Error fetching default unit:', error)
|
||||
return { data: null, error }
|
||||
}
|
||||
|
||||
return { data, error: null }
|
||||
}
|
||||
|
||||
/**
|
||||
* Convert quantity between units
|
||||
*/
|
||||
const convertUnit = (quantity: number, fromFactor: number, toFactor: number): number => {
|
||||
return (quantity * fromFactor) / toFactor
|
||||
}
|
||||
|
||||
return {
|
||||
getUnits,
|
||||
getDefaultUnit,
|
||||
convertUnit
|
||||
}
|
||||
}
|
||||
21
app/nuxt.config.ts
Normal file
21
app/nuxt.config.ts
Normal file
@@ -0,0 +1,21 @@
|
||||
// https://nuxt.com/docs/api/configuration/nuxt-config
|
||||
export default defineNuxtConfig({
|
||||
compatibilityDate: '2025-07-15',
|
||||
devtools: { enabled: true },
|
||||
|
||||
modules: [
|
||||
'@nuxt/ui',
|
||||
'@nuxt/fonts'
|
||||
],
|
||||
|
||||
runtimeConfig: {
|
||||
public: {
|
||||
supabaseUrl: process.env.NUXT_PUBLIC_SUPABASE_URL || 'http://localhost:54321',
|
||||
supabaseAnonKey: process.env.NUXT_PUBLIC_SUPABASE_ANON_KEY || ''
|
||||
}
|
||||
},
|
||||
|
||||
colorMode: {
|
||||
preference: 'light'
|
||||
}
|
||||
})
|
||||
23
app/package.json
Normal file
23
app/package.json
Normal file
@@ -0,0 +1,23 @@
|
||||
{
|
||||
"name": "app",
|
||||
"type": "module",
|
||||
"private": true,
|
||||
"scripts": {
|
||||
"build": "nuxt build",
|
||||
"dev": "nuxt dev",
|
||||
"generate": "nuxt generate",
|
||||
"preview": "nuxt preview",
|
||||
"postinstall": "nuxt prepare"
|
||||
},
|
||||
"dependencies": {
|
||||
"@nuxt/fonts": "^0.13.0",
|
||||
"@nuxt/ui": "^4.4.0",
|
||||
"@supabase/supabase-js": "^2.95.3",
|
||||
"nuxt": "^4.3.1",
|
||||
"vue": "^3.5.28",
|
||||
"vue-router": "^4.6.4"
|
||||
},
|
||||
"devDependencies": {
|
||||
"@nuxtjs/tailwindcss": "^6.14.0"
|
||||
}
|
||||
}
|
||||
74
app/pages/index.vue
Normal file
74
app/pages/index.vue
Normal file
@@ -0,0 +1,74 @@
|
||||
<template>
|
||||
<div>
|
||||
<div class="flex items-center justify-between mb-6">
|
||||
<h1 class="text-3xl font-bold text-gray-900">Inventory</h1>
|
||||
|
||||
<div class="flex gap-2">
|
||||
<UButton
|
||||
to="/scan"
|
||||
color="primary"
|
||||
size="lg"
|
||||
icon="i-heroicons-qr-code"
|
||||
>
|
||||
Scan Item
|
||||
</UButton>
|
||||
|
||||
<UButton
|
||||
color="white"
|
||||
size="lg"
|
||||
icon="i-heroicons-plus"
|
||||
@click="showAddForm = true"
|
||||
>
|
||||
Add Manually
|
||||
</UButton>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- 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"
|
||||
/>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- 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>
|
||||
|
||||
<script setup lang="ts">
|
||||
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>
|
||||
33
app/pages/scan.vue
Normal file
33
app/pages/scan.vue
Normal file
@@ -0,0 +1,33 @@
|
||||
<template>
|
||||
<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"
|
||||
/>
|
||||
<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>
|
||||
</UCard>
|
||||
</div>
|
||||
</template>
|
||||
|
||||
<script setup lang="ts">
|
||||
definePageMeta({
|
||||
layout: 'default'
|
||||
})
|
||||
</script>
|
||||
74
app/pages/settings.vue
Normal file
74
app/pages/settings.vue
Normal file
@@ -0,0 +1,74 @@
|
||||
<template>
|
||||
<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>
|
||||
</div>
|
||||
|
||||
<UButton
|
||||
v-if="!user"
|
||||
to="/auth/login"
|
||||
color="primary"
|
||||
>
|
||||
Sign In
|
||||
</UButton>
|
||||
</div>
|
||||
</UCard>
|
||||
|
||||
<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>
|
||||
|
||||
<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>
|
||||
</div>
|
||||
</template>
|
||||
|
||||
<script setup lang="ts">
|
||||
const { user } = useSupabaseAuth()
|
||||
|
||||
definePageMeta({
|
||||
layout: 'default'
|
||||
})
|
||||
</script>
|
||||
BIN
app/public/favicon.ico
Normal file
BIN
app/public/favicon.ico
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 4.2 KiB |
2
app/public/robots.txt
Normal file
2
app/public/robots.txt
Normal file
@@ -0,0 +1,2 @@
|
||||
User-Agent: *
|
||||
Disallow:
|
||||
18
app/tsconfig.json
Normal file
18
app/tsconfig.json
Normal file
@@ -0,0 +1,18 @@
|
||||
{
|
||||
// https://nuxt.com/docs/guide/concepts/typescript
|
||||
"files": [],
|
||||
"references": [
|
||||
{
|
||||
"path": "./.nuxt/tsconfig.app.json"
|
||||
},
|
||||
{
|
||||
"path": "./.nuxt/tsconfig.server.json"
|
||||
},
|
||||
{
|
||||
"path": "./.nuxt/tsconfig.shared.json"
|
||||
},
|
||||
{
|
||||
"path": "./.nuxt/tsconfig.node.json"
|
||||
}
|
||||
]
|
||||
}
|
||||
184
app/types/database.types.ts
Normal file
184
app/types/database.types.ts
Normal file
@@ -0,0 +1,184 @@
|
||||
/**
|
||||
* Database type definitions
|
||||
*
|
||||
* TODO: Generate these from Supabase schema using:
|
||||
* supabase gen types typescript --project-id <project-id> > types/database.types.ts
|
||||
*
|
||||
* For now, using a placeholder structure that matches our schema
|
||||
*/
|
||||
|
||||
export type Json =
|
||||
| string
|
||||
| number
|
||||
| boolean
|
||||
| null
|
||||
| { [key: string]: Json | undefined }
|
||||
| Json[]
|
||||
|
||||
export interface Database {
|
||||
public: {
|
||||
Tables: {
|
||||
inventory_items: {
|
||||
Row: {
|
||||
id: string
|
||||
product_id: string | null
|
||||
name: string
|
||||
quantity: number
|
||||
unit_id: string
|
||||
expiry_date: string | null
|
||||
notes: string | null
|
||||
added_by: string
|
||||
created_at: string
|
||||
updated_at: string
|
||||
}
|
||||
Insert: {
|
||||
id?: string
|
||||
product_id?: string | null
|
||||
name: string
|
||||
quantity: number
|
||||
unit_id: string
|
||||
expiry_date?: string | null
|
||||
notes?: string | null
|
||||
added_by: string
|
||||
created_at?: string
|
||||
updated_at?: string
|
||||
}
|
||||
Update: {
|
||||
id?: string
|
||||
product_id?: string | null
|
||||
name?: string
|
||||
quantity?: number
|
||||
unit_id?: string
|
||||
expiry_date?: string | null
|
||||
notes?: string | null
|
||||
added_by?: string
|
||||
created_at?: string
|
||||
updated_at?: string
|
||||
}
|
||||
}
|
||||
products: {
|
||||
Row: {
|
||||
id: string
|
||||
barcode: string
|
||||
name: string
|
||||
brand: string | null
|
||||
image_url: string | null
|
||||
default_unit_id: string | null
|
||||
cached_at: string
|
||||
created_at: string
|
||||
}
|
||||
Insert: {
|
||||
id?: string
|
||||
barcode: string
|
||||
name: string
|
||||
brand?: string | null
|
||||
image_url?: string | null
|
||||
default_unit_id?: string | null
|
||||
cached_at?: string
|
||||
created_at?: string
|
||||
}
|
||||
Update: {
|
||||
id?: string
|
||||
barcode?: string
|
||||
name?: string
|
||||
brand?: string | null
|
||||
image_url?: string | null
|
||||
default_unit_id?: string | null
|
||||
cached_at?: string
|
||||
created_at?: string
|
||||
}
|
||||
}
|
||||
tags: {
|
||||
Row: {
|
||||
id: string
|
||||
name: string
|
||||
category: 'position' | 'type' | 'custom'
|
||||
icon: string | null
|
||||
color: string | null
|
||||
created_by: string | null
|
||||
created_at: string
|
||||
}
|
||||
Insert: {
|
||||
id?: string
|
||||
name: string
|
||||
category?: 'position' | 'type' | 'custom'
|
||||
icon?: string | null
|
||||
color?: string | null
|
||||
created_by?: string | null
|
||||
created_at?: string
|
||||
}
|
||||
Update: {
|
||||
id?: string
|
||||
name?: string
|
||||
category?: 'position' | 'type' | 'custom'
|
||||
icon?: string | null
|
||||
color?: string | null
|
||||
created_by?: string | null
|
||||
created_at?: string
|
||||
}
|
||||
}
|
||||
units: {
|
||||
Row: {
|
||||
id: string
|
||||
name: string
|
||||
abbreviation: string
|
||||
unit_type: 'weight' | 'volume' | 'count' | 'custom'
|
||||
base_unit_id: string | null
|
||||
conversion_factor: number | null
|
||||
is_default: boolean
|
||||
created_by: string | null
|
||||
created_at: string
|
||||
}
|
||||
Insert: {
|
||||
id?: string
|
||||
name: string
|
||||
abbreviation: string
|
||||
unit_type?: 'weight' | 'volume' | 'count' | 'custom'
|
||||
base_unit_id?: string | null
|
||||
conversion_factor?: number | null
|
||||
is_default?: boolean
|
||||
created_by?: string | null
|
||||
created_at?: string
|
||||
}
|
||||
Update: {
|
||||
id?: string
|
||||
name?: string
|
||||
abbreviation?: string
|
||||
unit_type?: 'weight' | 'volume' | 'count' | 'custom'
|
||||
base_unit_id?: string | null
|
||||
conversion_factor?: number | null
|
||||
is_default?: boolean
|
||||
created_by?: string | null
|
||||
created_at?: string
|
||||
}
|
||||
}
|
||||
item_tags: {
|
||||
Row: {
|
||||
item_id: string
|
||||
tag_id: string
|
||||
created_at: string
|
||||
}
|
||||
Insert: {
|
||||
item_id: string
|
||||
tag_id: string
|
||||
created_at?: string
|
||||
}
|
||||
Update: {
|
||||
item_id?: string
|
||||
tag_id?: string
|
||||
created_at?: string
|
||||
}
|
||||
}
|
||||
}
|
||||
Views: {
|
||||
[_ in never]: never
|
||||
}
|
||||
Functions: {
|
||||
[_ in never]: never
|
||||
}
|
||||
Enums: {
|
||||
tag_category: 'position' | 'type' | 'custom'
|
||||
unit_type: 'weight' | 'volume' | 'count' | 'custom'
|
||||
}
|
||||
}
|
||||
}
|
||||
251
supabase/migrations/003_helper_functions.sql
Normal file
251
supabase/migrations/003_helper_functions.sql
Normal file
@@ -0,0 +1,251 @@
|
||||
-- Migration: Additional SQL Functions for Inventory Management
|
||||
-- Week 2: Helper functions for common queries
|
||||
|
||||
-- Function: Get inventory items with full details (tags, product info, unit conversion)
|
||||
CREATE OR REPLACE FUNCTION get_inventory_details()
|
||||
RETURNS TABLE (
|
||||
item_id UUID,
|
||||
item_name TEXT,
|
||||
quantity DECIMAL,
|
||||
unit_abbreviation TEXT,
|
||||
unit_name TEXT,
|
||||
expiry_date DATE,
|
||||
days_until_expiry INTEGER,
|
||||
tags TEXT[],
|
||||
product_brand TEXT,
|
||||
product_image_url TEXT,
|
||||
product_barcode TEXT,
|
||||
created_at TIMESTAMPTZ,
|
||||
updated_at TIMESTAMPTZ
|
||||
) AS $$
|
||||
BEGIN
|
||||
RETURN QUERY
|
||||
SELECT
|
||||
i.id AS item_id,
|
||||
i.name AS item_name,
|
||||
i.quantity,
|
||||
u.abbreviation AS unit_abbreviation,
|
||||
u.name AS unit_name,
|
||||
i.expiry_date,
|
||||
(i.expiry_date - CURRENT_DATE) AS days_until_expiry,
|
||||
COALESCE(ARRAY_AGG(DISTINCT t.name) FILTER (WHERE t.name IS NOT NULL), '{}') AS tags,
|
||||
p.brand AS product_brand,
|
||||
p.image_url AS product_image_url,
|
||||
p.barcode AS product_barcode,
|
||||
i.created_at,
|
||||
i.updated_at
|
||||
FROM inventory_items i
|
||||
JOIN units u ON i.unit_id = u.id
|
||||
LEFT JOIN products p ON i.product_id = p.id
|
||||
LEFT JOIN item_tags it ON i.id = it.item_id
|
||||
LEFT JOIN tags t ON it.tag_id = t.id
|
||||
GROUP BY
|
||||
i.id, i.name, i.quantity, u.abbreviation, u.name,
|
||||
i.expiry_date, p.brand, p.image_url, p.barcode,
|
||||
i.created_at, i.updated_at
|
||||
ORDER BY i.created_at DESC;
|
||||
END;
|
||||
$$ LANGUAGE plpgsql;
|
||||
|
||||
COMMENT ON FUNCTION get_inventory_details() IS 'Returns all inventory items with denormalized data for display';
|
||||
|
||||
-- Function: Get items expiring soon
|
||||
CREATE OR REPLACE FUNCTION get_expiring_items(days_ahead INTEGER DEFAULT 7)
|
||||
RETURNS TABLE (
|
||||
item_id UUID,
|
||||
item_name TEXT,
|
||||
quantity DECIMAL,
|
||||
unit_abbreviation TEXT,
|
||||
expiry_date DATE,
|
||||
days_until_expiry INTEGER,
|
||||
tags TEXT[]
|
||||
) AS $$
|
||||
BEGIN
|
||||
RETURN QUERY
|
||||
SELECT
|
||||
i.id AS item_id,
|
||||
i.name AS item_name,
|
||||
i.quantity,
|
||||
u.abbreviation AS unit_abbreviation,
|
||||
i.expiry_date,
|
||||
(i.expiry_date - CURRENT_DATE) AS days_until_expiry,
|
||||
COALESCE(ARRAY_AGG(DISTINCT t.name) FILTER (WHERE t.name IS NOT NULL), '{}') AS tags
|
||||
FROM inventory_items i
|
||||
JOIN units u ON i.unit_id = u.id
|
||||
LEFT JOIN item_tags it ON i.id = it.item_id
|
||||
LEFT JOIN tags t ON it.tag_id = t.id
|
||||
WHERE
|
||||
i.expiry_date IS NOT NULL
|
||||
AND i.expiry_date <= CURRENT_DATE + MAKE_INTERVAL(days => days_ahead)
|
||||
GROUP BY i.id, i.name, i.quantity, u.abbreviation, i.expiry_date
|
||||
ORDER BY i.expiry_date ASC;
|
||||
END;
|
||||
$$ LANGUAGE plpgsql;
|
||||
|
||||
COMMENT ON FUNCTION get_expiring_items(INTEGER) IS 'Returns items expiring within specified days (default 7)';
|
||||
|
||||
-- Function: Get items by tag
|
||||
CREATE OR REPLACE FUNCTION get_items_by_tag(tag_name TEXT)
|
||||
RETURNS TABLE (
|
||||
item_id UUID,
|
||||
item_name TEXT,
|
||||
quantity DECIMAL,
|
||||
unit_abbreviation TEXT,
|
||||
expiry_date DATE,
|
||||
created_at TIMESTAMPTZ
|
||||
) AS $$
|
||||
BEGIN
|
||||
RETURN QUERY
|
||||
SELECT
|
||||
i.id AS item_id,
|
||||
i.name AS item_name,
|
||||
i.quantity,
|
||||
u.abbreviation AS unit_abbreviation,
|
||||
i.expiry_date,
|
||||
i.created_at
|
||||
FROM inventory_items i
|
||||
JOIN units u ON i.unit_id = u.id
|
||||
JOIN item_tags it ON i.id = it.item_id
|
||||
JOIN tags t ON it.tag_id = t.id
|
||||
WHERE t.name ILIKE tag_name
|
||||
ORDER BY i.created_at DESC;
|
||||
END;
|
||||
$$ LANGUAGE plpgsql;
|
||||
|
||||
COMMENT ON FUNCTION get_items_by_tag(TEXT) IS 'Returns all items with specified tag (case-insensitive)';
|
||||
|
||||
-- Function: Get low stock items (quantity <= threshold)
|
||||
CREATE OR REPLACE FUNCTION get_low_stock_items(threshold DECIMAL DEFAULT 1.0)
|
||||
RETURNS TABLE (
|
||||
item_id UUID,
|
||||
item_name TEXT,
|
||||
quantity DECIMAL,
|
||||
unit_abbreviation TEXT,
|
||||
tags TEXT[]
|
||||
) AS $$
|
||||
BEGIN
|
||||
RETURN QUERY
|
||||
SELECT
|
||||
i.id AS item_id,
|
||||
i.name AS item_name,
|
||||
i.quantity,
|
||||
u.abbreviation AS unit_abbreviation,
|
||||
COALESCE(ARRAY_AGG(DISTINCT t.name) FILTER (WHERE t.name IS NOT NULL), '{}') AS tags
|
||||
FROM inventory_items i
|
||||
JOIN units u ON i.unit_id = u.id
|
||||
LEFT JOIN item_tags it ON i.id = it.item_id
|
||||
LEFT JOIN tags t ON it.tag_id = t.id
|
||||
WHERE i.quantity <= threshold
|
||||
GROUP BY i.id, i.name, i.quantity, u.abbreviation
|
||||
ORDER BY i.quantity ASC, i.name ASC;
|
||||
END;
|
||||
$$ LANGUAGE plpgsql;
|
||||
|
||||
COMMENT ON FUNCTION get_low_stock_items(DECIMAL) IS 'Returns items with quantity at or below threshold';
|
||||
|
||||
-- Function: Update item quantity (consume or restock)
|
||||
CREATE OR REPLACE FUNCTION update_item_quantity(
|
||||
item_uuid UUID,
|
||||
quantity_change DECIMAL,
|
||||
delete_if_zero BOOLEAN DEFAULT TRUE
|
||||
)
|
||||
RETURNS BOOLEAN AS $$
|
||||
DECLARE
|
||||
new_quantity DECIMAL;
|
||||
BEGIN
|
||||
-- Calculate new quantity
|
||||
SELECT quantity + quantity_change INTO new_quantity
|
||||
FROM inventory_items
|
||||
WHERE id = item_uuid;
|
||||
|
||||
IF new_quantity IS NULL THEN
|
||||
RETURN FALSE; -- Item not found
|
||||
END IF;
|
||||
|
||||
-- Delete if zero and flag is set
|
||||
IF new_quantity <= 0 AND delete_if_zero THEN
|
||||
DELETE FROM inventory_items WHERE id = item_uuid;
|
||||
RETURN TRUE;
|
||||
END IF;
|
||||
|
||||
-- Update quantity (ensure non-negative)
|
||||
UPDATE inventory_items
|
||||
SET quantity = GREATEST(new_quantity, 0),
|
||||
updated_at = NOW()
|
||||
WHERE id = item_uuid;
|
||||
|
||||
RETURN TRUE;
|
||||
END;
|
||||
$$ LANGUAGE plpgsql;
|
||||
|
||||
COMMENT ON FUNCTION update_item_quantity(UUID, DECIMAL, BOOLEAN) IS 'Updates item quantity (positive for restock, negative for consume). Optionally deletes if zero.';
|
||||
|
||||
-- Function: Get inventory statistics
|
||||
CREATE OR REPLACE FUNCTION get_inventory_stats()
|
||||
RETURNS TABLE (
|
||||
total_items BIGINT,
|
||||
total_unique_products BIGINT,
|
||||
items_expiring_week BIGINT,
|
||||
items_expired BIGINT,
|
||||
total_tags_used BIGINT
|
||||
) AS $$
|
||||
BEGIN
|
||||
RETURN QUERY
|
||||
SELECT
|
||||
COUNT(DISTINCT i.id) AS total_items,
|
||||
COUNT(DISTINCT i.product_id) FILTER (WHERE i.product_id IS NOT NULL) AS total_unique_products,
|
||||
COUNT(i.id) FILTER (
|
||||
WHERE i.expiry_date IS NOT NULL
|
||||
AND i.expiry_date BETWEEN CURRENT_DATE AND CURRENT_DATE + INTERVAL '7 days'
|
||||
) AS items_expiring_week,
|
||||
COUNT(i.id) FILTER (
|
||||
WHERE i.expiry_date IS NOT NULL
|
||||
AND i.expiry_date < CURRENT_DATE
|
||||
) AS items_expired,
|
||||
COUNT(DISTINCT it.tag_id) AS total_tags_used
|
||||
FROM inventory_items i
|
||||
LEFT JOIN item_tags it ON i.id = it.item_id;
|
||||
END;
|
||||
$$ LANGUAGE plpgsql;
|
||||
|
||||
COMMENT ON FUNCTION get_inventory_stats() IS 'Returns summary statistics for the entire inventory';
|
||||
|
||||
-- Function: Search inventory (full-text search on items and products)
|
||||
CREATE OR REPLACE FUNCTION search_inventory(search_query TEXT)
|
||||
RETURNS TABLE (
|
||||
item_id UUID,
|
||||
item_name TEXT,
|
||||
quantity DECIMAL,
|
||||
unit_abbreviation TEXT,
|
||||
product_brand TEXT,
|
||||
tags TEXT[],
|
||||
relevance REAL
|
||||
) AS $$
|
||||
BEGIN
|
||||
RETURN QUERY
|
||||
SELECT
|
||||
i.id AS item_id,
|
||||
i.name AS item_name,
|
||||
i.quantity,
|
||||
u.abbreviation AS unit_abbreviation,
|
||||
p.brand AS product_brand,
|
||||
COALESCE(ARRAY_AGG(DISTINCT t.name) FILTER (WHERE t.name IS NOT NULL), '{}') AS tags,
|
||||
ts_rank(
|
||||
to_tsvector('english', i.name || ' ' || COALESCE(p.brand, '') || ' ' || COALESCE(p.name, '')),
|
||||
plainto_tsquery('english', search_query)
|
||||
) AS relevance
|
||||
FROM inventory_items i
|
||||
JOIN units u ON i.unit_id = u.id
|
||||
LEFT JOIN products p ON i.product_id = p.id
|
||||
LEFT JOIN item_tags it ON i.id = it.item_id
|
||||
LEFT JOIN tags t ON it.tag_id = t.id
|
||||
WHERE
|
||||
to_tsvector('english', i.name || ' ' || COALESCE(p.brand, '') || ' ' || COALESCE(p.name, ''))
|
||||
@@ plainto_tsquery('english', search_query)
|
||||
GROUP BY i.id, i.name, i.quantity, u.abbreviation, p.brand, p.name
|
||||
ORDER BY relevance DESC, i.created_at DESC
|
||||
LIMIT 50;
|
||||
END;
|
||||
$$ LANGUAGE plpgsql;
|
||||
|
||||
COMMENT ON FUNCTION search_inventory(TEXT) IS 'Full-text search across inventory items and products';
|
||||
37
supabase/migrations/004_seed_units.sql
Normal file
37
supabase/migrations/004_seed_units.sql
Normal file
@@ -0,0 +1,37 @@
|
||||
-- Migration: Seed Default Units
|
||||
-- Week 2: Pre-populate common measurement units with conversions
|
||||
|
||||
-- Weight units (metric base: gram)
|
||||
INSERT INTO units (id, name, abbreviation, unit_type, base_unit_id, conversion_factor, is_default, created_by) VALUES
|
||||
('f47ac10b-58cc-4372-a567-0e02b2c3d479', 'Gram', 'g', 'weight', NULL, 1.0, TRUE, NULL),
|
||||
('550e8400-e29b-41d4-a716-446655440001', 'Kilogram', 'kg', 'weight', 'f47ac10b-58cc-4372-a567-0e02b2c3d479', 1000.0, FALSE, NULL),
|
||||
('550e8400-e29b-41d4-a716-446655440002', 'Milligram', 'mg', 'weight', 'f47ac10b-58cc-4372-a567-0e02b2c3d479', 0.001, FALSE, NULL),
|
||||
('550e8400-e29b-41d4-a716-446655440003', 'Pound', 'lb', 'weight', 'f47ac10b-58cc-4372-a567-0e02b2c3d479', 453.592, FALSE, NULL),
|
||||
('550e8400-e29b-41d4-a716-446655440004', 'Ounce', 'oz', 'weight', 'f47ac10b-58cc-4372-a567-0e02b2c3d479', 28.3495, FALSE, NULL);
|
||||
|
||||
-- Volume units (metric base: milliliter)
|
||||
INSERT INTO units (id, name, abbreviation, unit_type, base_unit_id, conversion_factor, is_default, created_by) VALUES
|
||||
('550e8400-e29b-41d4-a716-446655440010', 'Milliliter', 'mL', 'volume', NULL, 1.0, TRUE, NULL),
|
||||
('550e8400-e29b-41d4-a716-446655440011', 'Liter', 'L', 'volume', '550e8400-e29b-41d4-a716-446655440010', 1000.0, FALSE, NULL),
|
||||
('550e8400-e29b-41d4-a716-446655440012', 'Centiliter', 'cL', 'volume', '550e8400-e29b-41d4-a716-446655440010', 10.0, FALSE, NULL),
|
||||
('550e8400-e29b-41d4-a716-446655440013', 'Deciliter', 'dL', 'volume', '550e8400-e29b-41d4-a716-446655440010', 100.0, FALSE, NULL),
|
||||
('550e8400-e29b-41d4-a716-446655440014', 'Cup', 'cup', 'volume', '550e8400-e29b-41d4-a716-446655440010', 236.588, FALSE, NULL),
|
||||
('550e8400-e29b-41d4-a716-446655440015', 'Tablespoon', 'tbsp', 'volume', '550e8400-e29b-41d4-a716-446655440010', 14.7868, FALSE, NULL),
|
||||
('550e8400-e29b-41d4-a716-446655440016', 'Teaspoon', 'tsp', 'volume', '550e8400-e29b-41d4-a716-446655440010', 4.92892, FALSE, NULL),
|
||||
('550e8400-e29b-41d4-a716-446655440017', 'Fluid Ounce', 'fl oz', 'volume', '550e8400-e29b-41d4-a716-446655440010', 29.5735, FALSE, NULL),
|
||||
('550e8400-e29b-41d4-a716-446655440018', 'Gallon', 'gal', 'volume', '550e8400-e29b-41d4-a716-446655440010', 3785.41, FALSE, NULL),
|
||||
('550e8400-e29b-41d4-a716-446655440019', 'Quart', 'qt', 'volume', '550e8400-e29b-41d4-a716-446655440010', 946.353, FALSE, NULL),
|
||||
('550e8400-e29b-41d4-a716-446655440020', 'Pint', 'pt', 'volume', '550e8400-e29b-41d4-a716-446655440010', 473.176, FALSE, NULL);
|
||||
|
||||
-- Count units (no conversions, each is independent)
|
||||
INSERT INTO units (id, name, abbreviation, unit_type, base_unit_id, conversion_factor, is_default, created_by) VALUES
|
||||
('550e8400-e29b-41d4-a716-446655440030', 'Piece', 'pc', 'count', NULL, 1.0, TRUE, NULL),
|
||||
('550e8400-e29b-41d4-a716-446655440031', 'Dozen', 'doz', 'count', '550e8400-e29b-41d4-a716-446655440030', 12.0, FALSE, NULL),
|
||||
('550e8400-e29b-41d4-a716-446655440032', 'Package', 'pkg', 'count', NULL, 1.0, FALSE, NULL),
|
||||
('550e8400-e29b-41d4-a716-446655440033', 'Bottle', 'btl', 'count', NULL, 1.0, FALSE, NULL),
|
||||
('550e8400-e29b-41d4-a716-446655440034', 'Can', 'can', 'count', NULL, 1.0, FALSE, NULL),
|
||||
('550e8400-e29b-41d4-a716-446655440035', 'Jar', 'jar', 'count', NULL, 1.0, FALSE, NULL),
|
||||
('550e8400-e29b-41d4-a716-446655440036', 'Box', 'box', 'count', NULL, 1.0, FALSE, NULL),
|
||||
('550e8400-e29b-41d4-a716-446655440037', 'Bag', 'bag', 'count', NULL, 1.0, FALSE, NULL);
|
||||
|
||||
COMMENT ON TABLE units IS 'Measurement units with 30 common presets covering metric, imperial, and count units';
|
||||
49
supabase/migrations/005_seed_tags.sql
Normal file
49
supabase/migrations/005_seed_tags.sql
Normal file
@@ -0,0 +1,49 @@
|
||||
-- Migration: Seed Default Tags
|
||||
-- Week 2: Pre-populate common organizational tags
|
||||
|
||||
-- Position Tags (where items are stored)
|
||||
INSERT INTO tags (id, name, category, icon, color, created_by) VALUES
|
||||
('650e8400-e29b-41d4-a716-446655440001', 'Fridge', 'position', '🧊', '#3b82f6', NULL),
|
||||
('650e8400-e29b-41d4-a716-446655440002', 'Freezer', 'position', '❄️', '#06b6d4', NULL),
|
||||
('650e8400-e29b-41d4-a716-446655440003', 'Pantry', 'position', '🗄️', '#8b5cf6', NULL),
|
||||
('650e8400-e29b-41d4-a716-446655440004', 'Cabinet', 'position', '🚪', '#6b7280', NULL),
|
||||
('650e8400-e29b-41d4-a716-446655440005', 'Countertop', 'position', '🍽️', '#f59e0b', NULL),
|
||||
('650e8400-e29b-41d4-a716-446655440006', 'Cellar', 'position', '🏚️', '#78350f', NULL);
|
||||
|
||||
-- Type Tags (food categories)
|
||||
INSERT INTO tags (id, name, category, icon, color, created_by) VALUES
|
||||
('650e8400-e29b-41d4-a716-446655440010', 'Dairy', 'type', '🧀', '#fbbf24', NULL),
|
||||
('650e8400-e29b-41d4-a716-446655440011', 'Meat', 'type', '🥩', '#ef4444', NULL),
|
||||
('650e8400-e29b-41d4-a716-446655440012', 'Fish', 'type', '🐟', '#3b82f6', NULL),
|
||||
('650e8400-e29b-41d4-a716-446655440013', 'Vegetables', 'type', '🥬', '#22c55e', NULL),
|
||||
('650e8400-e29b-41d4-a716-446655440014', 'Fruits', 'type', '🍎', '#f97316', NULL),
|
||||
('650e8400-e29b-41d4-a716-446655440015', 'Grains', 'type', '🌾', '#eab308', NULL),
|
||||
('650e8400-e29b-41d4-a716-446655440016', 'Legumes', 'type', '🫘', '#84cc16', NULL),
|
||||
('650e8400-e29b-41d4-a716-446655440017', 'Condiments', 'type', '🧂', '#ef4444', NULL),
|
||||
('650e8400-e29b-41d4-a716-446655440018', 'Snacks', 'type', '🍿', '#f97316', NULL),
|
||||
('650e8400-e29b-41d4-a716-446655440019', 'Beverages', 'type', '🥤', '#06b6d4', NULL),
|
||||
('650e8400-e29b-41d4-a716-446655440020', 'Baking', 'type', '🧁', '#ec4899', NULL),
|
||||
('650e8400-e29b-41d4-a716-446655440021', 'Spices', 'type', '🌶️', '#dc2626', NULL),
|
||||
('650e8400-e29b-41d4-a716-446655440022', 'Canned', 'type', '🥫', '#71717a', NULL),
|
||||
('650e8400-e29b-41d4-a716-446655440023', 'Frozen', 'type', '🧊', '#06b6d4', NULL);
|
||||
|
||||
-- Dietary Tags
|
||||
INSERT INTO tags (id, name, category, icon, color, created_by) VALUES
|
||||
('650e8400-e29b-41d4-a716-446655440030', 'Vegan', 'dietary', '🌱', '#22c55e', NULL),
|
||||
('650e8400-e29b-41d4-a716-446655440031', 'Vegetarian', 'dietary', '🥕', '#84cc16', NULL),
|
||||
('650e8400-e29b-41d4-a716-446655440032', 'Gluten-Free', 'dietary', '🌾', '#eab308', NULL),
|
||||
('650e8400-e29b-41d4-a716-446655440033', 'Lactose-Free', 'dietary', '🥛', '#60a5fa', NULL),
|
||||
('650e8400-e29b-41d4-a716-446655440034', 'Organic', 'dietary', '♻️', '#10b981', NULL),
|
||||
('650e8400-e29b-41d4-a716-446655440035', 'Low-Carb', 'dietary', '🥗', '#22c55e', NULL),
|
||||
('650e8400-e29b-41d4-a716-446655440036', 'Kosher', 'dietary', '✡️', '#3b82f6', NULL),
|
||||
('650e8400-e29b-41d4-a716-446655440037', 'Halal', 'dietary', '☪️', '#22c55e', NULL);
|
||||
|
||||
-- Custom/Workflow Tags
|
||||
INSERT INTO tags (id, name, category, icon, color, created_by) VALUES
|
||||
('650e8400-e29b-41d4-a716-446655440040', 'Low Stock', 'custom', '⚠️', '#ef4444', NULL),
|
||||
('650e8400-e29b-41d4-a716-446655440041', 'To Buy', 'custom', '🛒', '#3b82f6', NULL),
|
||||
('650e8400-e29b-41d4-a716-446655440042', 'Meal Prep', 'custom', '🍱', '#8b5cf6', NULL),
|
||||
('650e8400-e29b-41d4-a716-446655440043', 'Leftovers', 'custom', '♻️', '#f59e0b', NULL),
|
||||
('650e8400-e29b-41d4-a716-446655440044', 'Opening Soon', 'custom', '📆', '#f97316', NULL);
|
||||
|
||||
COMMENT ON TABLE tags IS 'Pre-populated with 33 common tags across position, type, dietary, and workflow categories';
|
||||
Reference in New Issue
Block a user