Реализация AI-ассистента для подбора рецептов в мобильном приложении
«Что приготовить из того, что есть в холодильнике» — классическая задача на ограниченный инвентарь. AI-ассистент здесь не просто поисковик по базе рецептов, а генератор рецептов под конкретный набор ингредиентов, диетические ограничения и время приготовления.
Ввод ингредиентов: текст, фото, голос
Три канала ввода, все нужны. Текстовый — очевидный. Голосовой — через SFSpeechRecognizer/Android SpeechRecognizer. Самый полезный — фото содержимого холодильника с распознаванием ингредиентов.
Для распознавания ингредиентов на фото: CoreML с моделью, дообученной на food-датасете (Food-101 или OpenFoodFacts), или Google Cloud Vision API с label detection.
// iOS - распознавание ингредиентов через Vision + CoreML
func recognizeIngredients(in image: UIImage) async throws -> [String] {
guard let cgImage = image.cgImage else { return [] }
let model = try FoodClassifier(configuration: .init())
let vnModel = try VNCoreMLModel(for: model.model)
let request = VNCoreMLRequest(model: vnModel)
request.imageCropAndScaleOption = .centerCrop
let handler = VNImageRequestHandler(cgImage: cgImage)
try handler.perform([request])
guard let results = request.results as? [VNClassificationObservation] else { return [] }
return results
.filter { $0.confidence > 0.6 }
.prefix(10)
.map { $0.identifier }
}
На Android — ML Kit ImageLabeler с localModel или remoteModel (скачивается при первом использовании).
Генерация рецепта с учётом ограничений
Промпт — ключевая часть. Ограничений может быть много: аллергии, диеты, порции, время приготовления, сложность.
struct RecipeRequest: Encodable {
let ingredients: [String]
let servings: Int
let maxCookingMinutes: Int
let dietaryRestrictions: [String] // "vegan", "gluten-free", "nut-allergy", ...
let difficulty: String // "easy", "medium", "hard"
let cuisinePreferences: [String] // опционально
}
func buildRecipePrompt(_ req: RecipeRequest) -> String {
"""
Create a recipe using ONLY these ingredients (you may add basic pantry staples: salt, oil, water, common spices):
Available: \(req.ingredients.joined(separator: ", "))
Requirements:
- Servings: \(req.servings)
- Max cooking time: \(req.maxCookingMinutes) minutes
- Dietary: \(req.dietaryRestrictions.isEmpty ? "none" : req.dietaryRestrictions.joined(separator: ", "))
- Difficulty: \(req.difficulty)
Return JSON: {name, cookingTime, servings, ingredients: [{name, amount, unit}], steps: [{number, instruction, duration}], nutrition: {calories, protein, carbs, fat}}
"""
}
response_format: json_object обязателен — парсить markdown-обёртку вокруг JSON в продакшне не стоит.
Карточка рецепта в UI
// Android Compose
@Composable
fun RecipeCard(recipe: Recipe) {
LazyColumn(
modifier = Modifier.fillMaxSize(),
contentPadding = PaddingValues(16.dp),
verticalArrangement = Arrangement.spacedBy(12.dp)
) {
item {
Text(recipe.name, style = MaterialTheme.typography.headlineSmall)
Row(horizontalArrangement = Arrangement.spacedBy(16.dp)) {
InfoChip(Icons.Default.Timer, "${recipe.cookingTime} мин")
InfoChip(Icons.Default.People, "${recipe.servings} порц.")
InfoChip(Icons.Default.LocalFireDepartment, "${recipe.nutrition.calories} ккал")
}
}
item { Text("Ингредиенты", style = MaterialTheme.typography.titleMedium) }
items(recipe.ingredients) { ing ->
Text("• ${ing.amount} ${ing.unit} ${ing.name}")
}
item { Text("Приготовление", style = MaterialTheme.typography.titleMedium) }
itemsIndexed(recipe.steps) { index, step ->
StepCard(number = index + 1, instruction = step.instruction, duration = step.duration)
}
}
}
Таймер для шагов приготовления
Каждый шаг с указанным временем должен запускать таймер прямо из карточки. Это не AI-фича, но то, что делает ассистент реально полезным в процессе готовки.
class StepTimerManager: ObservableObject {
@Published var activeTimers = [Int: TimeInterval]()
private var timers = [Int: Timer]()
func startTimer(for stepIndex: Int, duration: TimeInterval) {
activeTimers[stepIndex] = duration
timers[stepIndex] = Timer.scheduledTimer(withTimeInterval: 1.0, repeats: true) { [weak self] _ in
guard let self else { return }
if let remaining = self.activeTimers[stepIndex], remaining > 0 {
self.activeTimers[stepIndex] = remaining - 1
} else {
self.timers[stepIndex]?.invalidate()
self.notifyStepComplete(stepIndex)
}
}
}
private func notifyStepComplete(_ step: Int) {
let content = UNMutableNotificationContent()
content.title = "Шаг \(step + 1) готов"
content.sound = .default
UNUserNotificationCenter.current().add(
UNNotificationRequest(identifier: "step-\(step)", content: content, trigger: nil)
)
}
}
Сохранение и персонализация
Рецепты, которые пользователь сохранил и приготовил, формируют профиль предпочтений. При следующем запросе добавляем в промпт «Previously liked: [список блюд]» — это улучшает релевантность без fine-tuning.
Хранение рецептов: SwiftData (iOS 17+) или Core Data — JSON-сериализация структуры рецепта в атрибут. На Android — Room с TypeConverter для List<Ingredient> и List<Step>.
Ориентиры по срокам
Базовый ассистент (текстовый ввод + генерация рецепта) — 3–5 дней. Полная реализация с распознаванием ингредиентов из фото, таймерами шагов, профилем предпочтений и офлайн-хранением — 3–4 недели.







