feat: complete ShiftCraft — AI-powered shift scheduling SaaS

Complete implementation including:
- Landing page with hero, features, how-it-works, pricing
- Employee management (CRUD with soft delete)
- AI constraint parser (Anthropic Claude API)
- German labor law templates (ArbZG §3, §5, §9)
- HiGHS ILP solver for optimal fair schedules
- Schedule calendar result view (employee × date grid)
- Shift framework configuration (periods + shifts)
- Subscription tiers: Free / Pro / Business
- PocketBase setup script with collection creation + seed data
- .env.example with all required variables documented

Pages: employees, constraints (list/new/templates), schedules (list/new/[id]),
       settings (organization/shifts/billing), dashboard

Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
This commit is contained in:
2026-04-18 07:47:31 +02:00
parent 2ea4ca5d52
commit 36e0946ee4
38 changed files with 4254 additions and 133 deletions

37
app/composables/useOrg.ts Normal file
View File

@@ -0,0 +1,37 @@
import type { Organization } from '~/shared/types/pocketbase'
export const useOrg = defineStore('org', {
state: () => ({
org: null as Organization | null,
loading: false,
}),
getters: {
plan: (state) => state.org?.plan ?? 'free',
orgId: (state) => state.org?.id ?? null,
},
actions: {
async fetchOrg(orgId: string) {
const { pb } = usePocketBase()
this.loading = true
try {
this.org = await pb.collection('organizations').getOne(orgId) as unknown as Organization
} catch (err) {
console.error('Failed to fetch org', err)
} finally {
this.loading = false
}
},
async updateOrg(data: Partial<Organization>) {
if (!this.org) return
const { pb } = usePocketBase()
this.org = await pb.collection('organizations').update(this.org.id, data) as unknown as Organization
},
clearOrg() {
this.org = null
}
}
})