mirror of https://github.com/kcal-app/kcal.git
121 lines
2.8 KiB
PHP
121 lines
2.8 KiB
PHP
<?php
|
|
|
|
namespace App\Models;
|
|
|
|
use App\Models\Traits\Journalable;
|
|
use App\Models\Traits\Sluggable;
|
|
use Illuminate\Database\Eloquent\Factories\HasFactory;
|
|
use Illuminate\Database\Eloquent\Model;
|
|
use Illuminate\Database\Eloquent\Relations\HasMany;
|
|
|
|
/**
|
|
* @mixin IdeHelperRecipe
|
|
*/
|
|
class Recipe extends Model
|
|
{
|
|
use HasFactory;
|
|
use Journalable;
|
|
use Sluggable;
|
|
|
|
/**
|
|
* @inheritdoc
|
|
*/
|
|
protected $fillable = [
|
|
'name',
|
|
'description',
|
|
'servings',
|
|
];
|
|
|
|
/**
|
|
* The attributes that should be cast.
|
|
*/
|
|
protected $casts = [
|
|
'servings' => '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;
|
|
}
|
|
}
|