kcal/app/Http/Controllers/RecipeController.php

418 lines
15 KiB
PHP

<?php
namespace App\Http\Controllers;
use App\Models\IngredientAmount;
use App\Models\Recipe;
use App\Models\RecipeSeparator;
use App\Models\RecipeStep;
use App\Rules\ArrayNotEmpty;
use App\Rules\StringIsDecimalOrFraction;
use App\Rules\UsesIngredientTrait;
use App\Support\Number;
use App\Support\Nutrients;
use Illuminate\Contracts\View\View;
use Illuminate\Http\RedirectResponse;
use Illuminate\Http\Request;
use Illuminate\Support\Collection;
use Illuminate\Support\Facades\DB;
use Illuminate\Support\Str;
class RecipeController extends Controller
{
/**
* Display a listing of the resource.
*
* @return \Illuminate\Contracts\View\View
*/
public function index(): View
{
return view('recipes.index')->with('tags', Recipe::getTagTotals());
}
/**
* Show the form for creating a new resource.
*
* @return \Illuminate\Contracts\View\View
*/
public function create(): View
{
return $this->edit(new Recipe());
}
/**
* Store a newly created resource in storage.
*
* @param \Illuminate\Http\Request $request
* @return \Illuminate\Http\RedirectResponse
*
* @throws \Throwable
*/
public function store(Request $request): RedirectResponse
{
return $this->update($request, new Recipe());
}
/**
* Display the specified resource.
*
* @param \App\Models\Recipe $recipe
* @return \Illuminate\Contracts\View\View
*/
public function show(Recipe $recipe): View
{
// Set feature image if media has been added.
$feature_image = NULL;
if ($recipe->hasMedia() && $recipe->getFirstMedia()->hasGeneratedConversion('header')) {
$feature_image = $recipe->getFirstMediaUrl('default', 'header');
}
return view('recipes.show')
->with('recipe', $recipe)
->with('feature_image', $feature_image);
}
/**
* Show the form for editing the specified resource.
*
* @param \App\Models\Recipe $recipe
* @return \Illuminate\Contracts\View\View
*/
public function edit(Recipe $recipe): View
{
// Pre-populate relationships from form data or current recipe.
$ingredients = [];
if ($old = old('ingredients')) {
foreach ($old['id'] as $key => $ingredient_id) {
$ingredients[] = [
'type' => 'ingredient',
'key' => $old['key'][$key],
'weight' => $old['weight'][$key],
'amount' => $old['amount'][$key],
'unit' => $old['unit'][$key],
'ingredient_id' => $ingredient_id,
'ingredient_type' => $old['type'][$key],
'ingredient_name' => $old['name'][$key],
'detail' => $old['detail'][$key],
];
}
}
else {
foreach ($recipe->ingredientAmounts as $key => $ingredientAmount) {
$ingredients[] = [
'type' => 'ingredient',
'key' => $key,
'weight' => $ingredientAmount->weight,
'amount' => $ingredientAmount->amount_formatted,
'unit' => $ingredientAmount->unit,
'ingredient_id' => $ingredientAmount->ingredient_id,
'ingredient_type' => $ingredientAmount->ingredient_type,
'ingredient_name' => $ingredientAmount->ingredient->name,
'detail' => $ingredientAmount->detail,
];
}
}
$separators = [];
if ($old = old('separators')) {
foreach ($old['key'] as $index => $key) {
$separators[] = [
'type' => 'separator',
'key' => $old['key'][$index],
'weight' => $old['weight'][$index],
'text' => $old['text'][$index],
];
}
}
else {
foreach ($recipe->ingredientSeparators as $key => $ingredientSeparator) {
$separators[] = [
'type' => 'separator',
'key' => $key,
'weight' => $ingredientSeparator->weight,
'text' => $ingredientSeparator->text,
];
}
}
$steps = [];
if ($old = old('steps')) {
foreach ($old['step'] as $key => $step) {
if (empty($step)) {
continue;
}
$steps[] = [
'key' => $old['key'][$key],
'step_default' => $step,
];
}
}
else {
foreach ($recipe->steps as $key => $step) {
$steps[] = [
'key' => $key,
'step_default' => $step->step,
];
}
}
// Convert string tags (from old form data) to a Collection.
$recipe_tags = old('tags', $recipe->tags->pluck('name'));
if (is_string($recipe_tags)) {
$recipe_tags = new Collection(explode(',', $recipe_tags));
}
return view('recipes.edit')
->with('recipe', $recipe)
->with('recipe_tags', $recipe_tags)
->with('ingredients_list', new Collection([...$ingredients, ...$separators]))
->with('steps', $steps)
->with('ingredients_units', Nutrients::$units);
}
/**
* Update the specified resource in storage.
*
* @param \Illuminate\Http\Request $request
* @param \App\Models\Recipe $recipe
*
* @return \Illuminate\Http\RedirectResponse
*
* @throws \Throwable
*/
public function update(Request $request, Recipe $recipe): RedirectResponse
{
$input = $request->validate([
'name' => ['required', 'string'],
'description' => ['nullable', 'string'],
'description_delta' => ['nullable', 'string'],
'image' => ['nullable', 'file', 'mimes:jpg,png,gif'],
'remove_image' => ['nullable', 'boolean'],
'servings' => ['required', 'numeric'],
'time_prep' => ['nullable', 'numeric'],
'time_cook' => ['nullable', 'numeric'],
'weight' => ['nullable', 'numeric'],
'source' => ['nullable', 'string'],
'ingredients.amount' => ['required', 'array', new ArrayNotEmpty],
'ingredients.amount.*' => ['required_with:ingredients.id.*', 'nullable', new StringIsDecimalOrFraction],
'ingredients.unit' => ['required', 'array'],
'ingredients.unit.*' => ['required_with:ingredients.id.*'],
'ingredients.detail' => ['required', 'array'],
'ingredients.detail.*' => ['nullable', 'string'],
'ingredients.id' => ['required', 'array', new ArrayNotEmpty],
'ingredients.id.*' => 'required_with:ingredients.amount.*|nullable',
'ingredients.type' => ['required', 'array', new ArrayNotEmpty],
'ingredients.type.*' => ['required_with:ingredients.id.*', 'nullable', new UsesIngredientTrait()],
'ingredients.key' => ['required', 'array', new ArrayNotEmpty],
'ingredients.key.*' => ['nullable', 'int'],
'ingredients.weight' => ['required', 'array', new ArrayNotEmpty],
'ingredients.weight.*' => ['required', 'int'],
'separators.key' => ['nullable', 'array'],
'separators.key.*' => ['nullable', 'int'],
'separators.weight' => ['nullable', 'array'],
'separators.weight.*' => ['required', 'int'],
'separators.text' => ['nullable', 'array'],
'separators.text.*' => ['nullable', 'string'],
'steps.step' => ['required', 'array', new ArrayNotEmpty],
'steps.step.*' => ['nullable', 'string'],
'steps.key' => ['nullable', 'array'],
]);
// Validate that no ingredients are recursive.
// TODO: refactor as custom validator.
foreach (array_filter($input['ingredients']['id']) as $key => $id) {
if ($input['ingredients']['type'][$key] == Recipe::class && $id == $recipe->id) {
return back()->withInput()->withErrors('To understand recursion, you must understand recursion. Remove this recipe from this recipe.');
}
}
$recipe->fill([
'name' => Str::lower($input['name']),
'description' => $input['description'],
'description_delta' => $input['description_delta'],
'servings' => (int) $input['servings'],
'weight' => $input['weight'],
'time_prep' => (int) $input['time_prep'],
'time_cook' => (int) $input['time_cook'],
'source' => $input['source'],
]);
try {
DB::transaction(function () use ($recipe, $input) {
$recipe->saveOrFail();
$this->updateIngredients($recipe, $input);
$this->updateIngredientSeparators($recipe, $input);
$this->updateSteps($recipe, $input);
});
} catch (\Exception $e) {
DB::rollBack();
return back()->withInput()->withErrors($e->getMessage());
}
// Sync tags.
$tags = $request->get('tags');
if (!empty($tags)) {
$recipe->syncTags(explode(',', $tags));
}
elseif ($recipe->tags->isNotEmpty()) {
$recipe->detachTags($recipe->tags);
}
// Handle recipe image.
if (!empty($input['image'])) {
/** @var \Illuminate\Http\UploadedFile $file */
$file = $input['image'];
$recipe->clearMediaCollection();
$recipe
->addMediaFromRequest('image')
->usingName($recipe->name)
->usingFileName("{$recipe->slug}.{$file->extension()}")
->toMediaCollection();
}
elseif (isset($input['remove_image']) && (bool) $input['remove_image']) {
$recipe->clearMediaCollection();
}
session()->flash('message', "Recipe {$recipe->name} updated!");
return redirect()->route('recipes.show', $recipe);
}
/**
* Updates recipe ingredients data based on input.
*
* @param \App\Models\Recipe $recipe
* @param array $input
*
* @throws \Exception
*/
private function updateIngredients(Recipe $recipe, array $input): void {
// Delete any removed ingredients.
$removed = array_diff($recipe->ingredientAmounts->keys()->all(), $input['ingredients']['key']);
foreach ($removed as $removed_key) {
$recipe->ingredientAmounts[$removed_key]->delete();
}
// Add/update current ingredients.
$ingredient_amounts = [];
foreach (array_filter($input['ingredients']['id']) as $key => $ingredient_id) {
if (!is_null($input['ingredients']['key'][$key])) {
$ingredient_amounts[$key] = $recipe->ingredientAmounts[$input['ingredients']['key'][$key]];
}
else {
$ingredient_amounts[$key] = new IngredientAmount();
}
$ingredient_amounts[$key]->fill([
'amount' => Number::floatFromString($input['ingredients']['amount'][$key]),
'unit' => $input['ingredients']['unit'][$key],
'detail' => $input['ingredients']['detail'][$key],
'weight' => (int) $input['ingredients']['weight'][$key],
]);
$ingredient_amounts[$key]->ingredient()
->associate($input['ingredients']['type'][$key]::where('id', $ingredient_id)->first());
}
$recipe->ingredientAmounts()->saveMany($ingredient_amounts);
}
/**
* Updates recipe steps data based on input.
*
* @param \App\Models\Recipe $recipe
* @param array $input
*
* @throws \Exception
*/
private function updateSteps(Recipe $recipe, array $input): void {
$steps = [];
$number = 1;
// Delete any removed steps.
$removed = array_diff($recipe->steps->keys()->all(), $input['steps']['key']);
foreach ($removed as $removed_key) {
$recipe->steps[$removed_key]->delete();
}
foreach (array_filter($input['steps']['step']) as $key => $step) {
if (!is_null($input['steps']['key'][$key])) {
$steps[$key] = $recipe->steps[$input['steps']['key'][$key]];
}
else {
$steps[$key] = new RecipeStep();
}
$steps[$key]->fill(['number' => $number++, 'step' => $step]);
}
$recipe->steps()->saveMany($steps);
}
/**
* Updates recipe ingredient separators data based on input.
*
* @param \App\Models\Recipe $recipe
* @param array $input
*
* @throws \Exception
*/
private function updateIngredientSeparators(Recipe $recipe, array $input): void {
// Take no action of remove all separators
if (!isset($input['separators']) || empty($input['separators'])) {
if ($recipe->ingredientSeparators->isNotEmpty()) {
$recipe->ingredientSeparators()->delete();
}
return;
}
// Delete any removed separators.
$removed = array_diff($recipe->ingredientSeparators->keys()->all(), $input['separators']['key']);
foreach ($removed as $removed_key) {
$recipe->ingredientSeparators[$removed_key]->delete();
}
// Add/update current separators.
$ingredient_separators = [];
foreach ($input['separators']['key'] as $index => $key) {
if (!is_null($key)) {
$ingredient_separators[$index] = $recipe->ingredientSeparators[$key];
}
else {
$ingredient_separators[$index] = new RecipeSeparator();
}
$ingredient_separators[$index]->fill([
'container' => 'ingredients',
'text' => $input['separators']['text'][$index],
'weight' => (int) $input['separators']['weight'][$index],
]);
}
$recipe->ingredientSeparators()->saveMany($ingredient_separators);
}
/**
* Confirm removal of specified resource.
*/
public function delete(Recipe $recipe): View
{
return view('recipes.delete')->with('recipe', $recipe);
}
/**
* Remove the specified resource from storage.
*/
public function destroy(Recipe $recipe): RedirectResponse
{
// Remove recipe ingredients.
foreach ($recipe->ingredientAmounts as $ia) {
$ia->delete();
}
// Remove the recipe from any recipes.
foreach ($recipe->ingredientAmountRelationships as $iar) {
$iar->delete();
}
// Remove the recipe.
$recipe->delete();
return redirect(route('recipes.index'))
->with('message', "Recipe {$recipe->name} deleted!");
}
}