'int', ]; /** * Nutrient total calculation methods. */ private array $nutrientTotalMethods = [ 'caloriesTotal', 'carbohydratesTotal', 'cholesterolTotal', 'fatTotal', 'proteinTotal', 'sodiumTotal', ]; /** * Nutrient per serving methods. */ private array $nutrientPerServingMethods = [ 'caloriesPerServing', 'carbohydratesPerServing', 'cholesterolPerServing', 'fatPerServing', 'proteinPerServing', 'sodiumPerServing', ]; /** * Get the steps for this Recipe. */ public function steps(): HasMany { return $this->hasMany(RecipeStep::class)->orderBy('number'); } /** * Get the Food Amounts used for this Recipe. */ public function foodAmounts(): HasMany { return $this->hasMany(FoodAmount::class)->orderBy('weight'); } /** * Add nutrient calculations handling to overloading. * * @param string $method * @param array $parameters * * @return mixed * * @noinspection PhpMissingParamTypeInspection */ public function __call($method, $parameters): mixed { if (in_array($method, $this->nutrientTotalMethods)) { return $this->sumNutrient(substr($method, 0, -5)); } elseif (in_array($method, $this->nutrientPerServingMethods)) { $sum = $this->sumNutrient(substr($method, 0, -10)) / $this->servings; // Per-serving calculations are rounded, though actual food label // rounding standards are more complex. if ($sum > 1) { return round($sum); } else { return round($sum, 2); } } else { return parent::__call($method, $parameters); } } /** * Sum a specific nutrient for all food amounts. * * @param string $nutrient * Nutrient to sum. * * @return float */ private function sumNutrient(string $nutrient): float { $sum = 0; foreach ($this->foodAmounts as $foodAmount) { $sum += $foodAmount->{$nutrient}(); } return $sum; } }