feat: core schema redesign — dishes, structured recipes, cuisines, tags (iteration 7)

Replaces the flat JSONB-based recipe schema with a normalized relational model:

Schema (migrations consolidated to 001_initial_schema + 002_seed_data):
- New: dishes, dish_translations, dish_tags — canonical dish catalog
- New: cuisines, tags, dish_categories with _translations tables + full seed data
- New: recipe_ingredients, recipe_steps with _translations (replaces JSONB blobs)
- New: user_saved_recipes thin bookmark (drops saved_recipes + saved_recipe_translations)
- New: product_ingredients M2M table
- recipes: now a cooking variant of a dish (dish_id FK, no title/JSONB columns)
- recipe_translations: repurposed to per-language notes only
- products: mapping_id → primary_ingredient_id
- menu_items: recipe_id FK → recipes; adds dish_id
- meal_diary: adds dish_id, recipe_id → recipes, portion_g

Backend (Go):
- New packages: internal/cuisine, internal/tag, internal/dish (registry + handler + repo)
- New GET /cuisines, GET /tags (public), GET /dishes, GET /dishes/{id}, GET /recipes/{id}
- recipe, savedrecipe, menu, diary, product, ingredient packages updated for new schema

Flutter:
- New models: Cuisine, Tag; new providers: cuisineNamesProvider, tagNamesProvider
- recipe.dart: RecipeIngredient gains unit_code + effectiveUnit getter
- saved_recipe.dart: thin model, manual fromJson, computed nutrition getter
- diary_entry.dart: adds dishId, recipeId, portionG
- recipe_detail_screen.dart: localized cuisine/tag names via providers

Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
This commit is contained in:
dbastrikin
2026-03-15 18:01:24 +02:00
parent 55d01400b0
commit 61feb91bba
52 changed files with 2479 additions and 1492 deletions

View File

@@ -0,0 +1,22 @@
import 'package:flutter_riverpod/flutter_riverpod.dart';
import '../auth/auth_provider.dart';
import 'api_client.dart';
import '../../shared/models/cuisine.dart';
class CuisineRepository {
final ApiClient _api;
CuisineRepository(this._api);
Future<List<Cuisine>> fetchCuisines() async {
final data = await _api.get('/cuisines');
final List<dynamic> items = data['cuisines'] as List;
return items
.map((e) => Cuisine.fromJson(e as Map<String, dynamic>))
.toList();
}
}
final cuisineRepositoryProvider = Provider<CuisineRepository>(
(ref) => CuisineRepository(ref.watch(apiClientProvider)),
);

View File

@@ -0,0 +1,22 @@
import 'package:flutter_riverpod/flutter_riverpod.dart';
import '../auth/auth_provider.dart';
import 'api_client.dart';
import '../../shared/models/tag.dart';
class TagRepository {
final ApiClient _api;
TagRepository(this._api);
Future<List<Tag>> fetchTags() async {
final data = await _api.get('/tags');
final List<dynamic> items = data['tags'] as List;
return items
.map((e) => Tag.fromJson(e as Map<String, dynamic>))
.toList();
}
}
final tagRepositoryProvider = Provider<TagRepository>(
(ref) => TagRepository(ref.watch(apiClientProvider)),
);

View File

@@ -0,0 +1,19 @@
import 'package:flutter_riverpod/flutter_riverpod.dart';
import '../api/cuisine_repository.dart';
import '../../shared/models/cuisine.dart';
import 'language_provider.dart';
/// Fetches and caches cuisines with localized names.
/// Returns list of [Cuisine] objects.
/// Re-fetches automatically when languageProvider changes.
final cuisinesProvider = FutureProvider<List<Cuisine>>((ref) {
ref.watch(languageProvider); // invalidate when language changes
return ref.read(cuisineRepositoryProvider).fetchCuisines();
});
/// Convenience provider that returns a slug → localized name map.
final cuisineNamesProvider = FutureProvider<Map<String, String>>((ref) async {
final cuisines = await ref.watch(cuisinesProvider.future);
return {for (final c in cuisines) c.slug: c.name};
});

View File

@@ -0,0 +1,19 @@
import 'package:flutter_riverpod/flutter_riverpod.dart';
import '../api/tag_repository.dart';
import '../../shared/models/tag.dart';
import 'language_provider.dart';
/// Fetches and caches tags with localized names.
/// Returns list of [Tag] objects.
/// Re-fetches automatically when languageProvider changes.
final tagsProvider = FutureProvider<List<Tag>>((ref) {
ref.watch(languageProvider); // invalidate when language changes
return ref.read(tagRepositoryProvider).fetchTags();
});
/// Convenience provider that returns a slug → localized name map.
final tagNamesProvider = FutureProvider<Map<String, String>>((ref) async {
final tags = await ref.watch(tagsProvider.future);
return {for (final t in tags) t.slug: t.name};
});