date ?? Carbon::now()->toDateString(); $date = Carbon::rawCreateFromFormat('Y-m-d', $date); // Get entries and nutrient sums for the day. $entries = JournalEntry::where([ 'user_id' => Auth::user()->id, 'date' => $date->toDateString(), ])->get(); $sums = []; foreach (Nutrients::all()->pluck('value') as $nutrient) { $sums[$nutrient] = round($entries->sum($nutrient)); } // Get daily goals data for user. $goals = Auth::user()->getGoalsByTime($date); $dailyGoals = []; foreach (Nutrients::all()->pluck('value') as $nutrient) { $goal = $goals['present'] ->where('frequency', 'daily') ->where('name', $nutrient) ->first(); if ($goal) { $dailyGoals[$goal->name] = round($sums[$goal->name] / $goal->goal * 100); if ($dailyGoals[$goal->name] > 0) { $dailyGoals[$goal->name] .= '%'; } } } return view('journal-entries.index') ->with('entries', $entries) ->with('sums', $sums) ->with('dailyGoals', $dailyGoals) ->with('date', $date); } /** * Show the form for creating a new resource. */ public function create(Request $request): View { $date = $request->date ?? Carbon::now()->toDateString(); $ingredients = []; if ($old = old('ingredients')) { foreach ($old['amount'] as $key => $amount) { if ( empty($old['date'][$key]) && empty($old['meal'][$key]) && empty($amount) && empty($old['unit'][$key]) && empty($old['id'][$key]) ) { continue; } $ingredients[$key] = [ 'date' => $old['date'][$key], 'meal' => $old['meal'][$key], 'amount' => $amount, 'unit' => $old['unit'][$key], 'id' => $old['id'][$key], 'type' => $old['type'][$key], 'name' => $old['name'][$key], ]; // Add supported units for the ingredient. $ingredient = NULL; if ($ingredients[$key]['type'] === Food::class) { $ingredient = Food::whereId($ingredients[$key]['id'])->first(); } elseif ($ingredients[$key]['type'] === Recipe::class) { $ingredient = Recipe::whereId($ingredients[$key]['id'])->first(); } if ($ingredient) { $ingredients[$key]['units'] = $ingredient->units_supported; } } } return view('journal-entries.create') ->with('ingredients', $ingredients) ->with('meals', JournalEntry::meals()->toArray()) ->with('units', Nutrients::units()->toArray()) ->with('default_date', Carbon::createFromFormat('Y-m-d', $date)); } /** * Show the form for creating a journal entry from nutrients directly. */ public function createFromNutrients(Request $request): View { $date = $request->date ?? Carbon::now()->toDateString(); return view('journal-entries.create-from-nutrients') ->with('meals', JournalEntry::meals()->toArray()) ->with('units', Nutrients::units()->toArray()) ->with('default_date', Carbon::createFromFormat('Y-m-d', $date)); } /** * Store a newly created resource in storage. */ public function store(Request $request): RedirectResponse { $input = $request->validate([ 'ingredients.date' => ['required', 'array', new ArrayNotEmpty], 'ingredients.date.*' => ['nullable', 'date', 'required_with:ingredients.id.*'], 'ingredients.meal' => ['required', 'array', new ArrayNotEmpty], 'ingredients.meal.*' => [ 'nullable', 'string', 'required_with:ingredients.id.*', new InArray(JournalEntry::meals()->pluck('value')->toArray()) ], '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.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()], 'group_entries' => ['nullable', 'boolean'], ]); $ingredients = ArrayFormat::flipTwoDimensionalKeys($input['ingredients']); /** @var \App\Models\JournalEntry[] $entries */ $entries = []; $entry_key = 0; // TODO: Improve efficiency. Potential for lots of queries here... foreach ($ingredients as $ingredient) { // Set entry key (combined date and meal or individual entries). if (isset($input['group_entries']) && (bool) $input['group_entries']) { $entry_key = "{$ingredient['date']}{$ingredient['meal']}"; } else { $entry_key++; } // Prepare entry values. $entries[$entry_key] = $entries[$entry_key] ?? JournalEntry::make([ 'date' => $ingredient['date'], 'meal' => $ingredient['meal'], ])->user()->associate(Auth::user()); // Calculate amounts based on ingredient type. if ($ingredient['type'] == Food::class) { $item = Food::whereId($ingredient['id'])->first(); $nutrient_multiplier = Nutrients::calculateFoodNutrientMultiplier( $item, Number::floatFromString($ingredient['amount']), $ingredient['unit'] ); foreach (Nutrients::all()->pluck('value') as $nutrient) { $entries[$entry_key]->{$nutrient} += $item->{$nutrient} * $nutrient_multiplier; } $entries[$entry_key]->foods->add($item); } elseif ($ingredient['type'] == Recipe::class) { $item = Recipe::whereId($ingredient['id'])->first(); foreach (Nutrients::all()->pluck('value') as $nutrient) { $entries[$entry_key]->{$nutrient} += Nutrients::calculateRecipeNutrientAmount( $item, $nutrient, Number::floatFromString($ingredient['amount']), $ingredient['unit'] ); } $entries[$entry_key]->recipes->add($item); } else { return back()->withInput()->withErrors("Invalid ingredient type {$ingredient['type']}."); } // Set entry summary. $unit = $ingredient['unit']; if ($item instanceof Food) { if ($unit === 'serving') { if (empty($item->serving_unit) && empty($item->serving_unit_name)) { $unit = null; } elseif (!empty($item->serving_unit_name)) { $unit = $item->serving_unit_formatted; } } } $entries[$entry_key]->summary .= (!empty($entries[$entry_key]->summary) ? ', ' : null); $entries[$entry_key]->summary .= "{$ingredient['amount']} {$unit} {$item->name}"; if (isset($item->detail) && !empty($item->detail)) { $entries[$entry_key]->summary .= ", {$item->detail}"; } } foreach ($entries as $entry) { $entry->save(); $entry->user->save(); $entry->foods()->saveMany($entry->foods); $entry->recipes()->saveMany($entry->recipes); } $count = count($entries); session()->flash('message', "Added {$count} journal entries!"); // Redirect to the date if only one date is used. $parameters = []; $unique_dates = array_unique($input['ingredients']['date']); if (count($unique_dates) === 1) { $parameters['date'] = reset($unique_dates); } return redirect()->route('journal-entries.index', $parameters); } /** * Store an entry from nutrients. */ public function storeFromNutrients(Request $request): RedirectResponse { $attributes = $request->validate([ 'date' => ['required', 'date'], 'meal' => [ 'required', 'string', new InArray(JournalEntry::meals()->pluck('value')->toArray()) ], 'summary' => ['required', 'string'], 'calories' => ['nullable', 'required_without_all:fat,cholesterol,sodium,carbohydrates,protein', 'numeric'], 'fat' => ['nullable', 'required_without_all:calories,cholesterol,sodium,carbohydrates,protein', 'numeric'], 'cholesterol' => ['nullable', 'required_without_all:calories,fat,sodium,carbohydrates,protein', 'numeric'], 'sodium' => ['nullable', 'required_without_all:calories,fat,cholesterol,carbohydrates,protein', 'numeric'], 'carbohydrates' => ['nullable', 'required_without_all:calories,fat,cholesterol,sodium,protein', 'numeric'], 'protein' => ['nullable', 'required_without_all:calories,fat,cholesterol,sodium,carbohydrates', 'numeric'], ]); $entry = JournalEntry::make(array_filter($attributes)) ->user()->associate(Auth::user()); $entry->save(); session()->flash('message', "Journal entry added!"); return redirect()->route( 'journal-entries.index', ['date' => $entry->date->format('Y-m-d')] ); } /** * Confirm removal of the specified resource. */ public function delete(JournalEntry $journal_entry): View { return view('journal-entries.delete') ->with('journal_entry', $journal_entry); } /** * Remove the specified resource from storage. */ public function destroy(JournalEntry $journal_entry): RedirectResponse { $journal_entry->delete(); session()->flash('message', 'Journal entry deleted!'); return redirect(route('journal-entries.index', [ 'date' => $journal_entry->date->toDateString() ])); } }