mirror of https://github.com/kcal-app/kcal.git
				
				
				
			Refactor on servings-based food data (WIP)
This commit is contained in:
		
							parent
							
								
									1ec4439d8e
								
							
						
					
					
						commit
						c33776155a
					
				|  | @ -9,14 +9,16 @@ use Illuminate\Database\Eloquent\Model; | ||||||
|  * @property int id |  * @property int id | ||||||
|  * @property string name Food base name. |  * @property string name Food base name. | ||||||
|  * @property ?string detail Some additional detail about the food (e.g. "small" with the name "onion"). |  * @property ?string detail Some additional detail about the food (e.g. "small" with the name "onion"). | ||||||
|  * @property float carbohydrates (per 100g). |  * @property ?string brand Brand name. | ||||||
|  * @property float calories (per 100g). |  * @property float carbohydrates per serving (g). | ||||||
|  * @property float cholesterol (per 100g). |  * @property float calories per serving (g). | ||||||
|  * @property float fat (per 100g). |  * @property float cholesterol per serving (g). | ||||||
|  * @property float protein (per 100g). |  * @property float fat per serving (g). | ||||||
|  * @property float sodium (per 100g). |  * @property float protein per serving (g). | ||||||
|  * @property ?float unit_weight Weight of one cup of the food. |  * @property float sodium per serving (g). | ||||||
|  * @property ?float cup_weight Weight of one "unit" (e.g. an egg, onion, etc.) of the food. |  * @property float serving_size Size of one serving of the food. | ||||||
|  |  * @property ?string serving_unit Unit for serving weight (tsp, tbsp, cup, or null). | ||||||
|  |  * @property float serving_weight per serving (g). | ||||||
|  * @property \Illuminate\Support\Carbon created_at |  * @property \Illuminate\Support\Carbon created_at | ||||||
|  * @property \Illuminate\Support\Carbon updated_at |  * @property \Illuminate\Support\Carbon updated_at | ||||||
|  */ |  */ | ||||||
|  | @ -35,14 +37,16 @@ class Food extends Model | ||||||
|     protected $fillable = [ |     protected $fillable = [ | ||||||
|         'name', |         'name', | ||||||
|         'detail', |         'detail', | ||||||
|  |         'brand', | ||||||
|         'calories', |         'calories', | ||||||
|         'carbohydrates', |         'carbohydrates', | ||||||
|         'cholesterol', |         'cholesterol', | ||||||
|         'fat', |         'fat', | ||||||
|         'protein', |         'protein', | ||||||
|         'sodium', |         'sodium', | ||||||
|         'unit_weight', |         'serving_size', | ||||||
|         'cup_weight', |         'serving_unit', | ||||||
|  |         'serving_weight', | ||||||
|     ]; |     ]; | ||||||
| 
 | 
 | ||||||
|     /** |     /** | ||||||
|  | @ -52,10 +56,10 @@ class Food extends Model | ||||||
|         'calories' => 'float', |         'calories' => 'float', | ||||||
|         'carbohydrates' => 'float', |         'carbohydrates' => 'float', | ||||||
|         'cholesterol' => 'float', |         'cholesterol' => 'float', | ||||||
|         'cup_weight' => 'float', |  | ||||||
|         'fat' => 'float', |         'fat' => 'float', | ||||||
|         'protein' => 'float', |         'protein' => 'float', | ||||||
|  |         'serving_size' => 'float', | ||||||
|  |         'serving_weight' => 'float', | ||||||
|         'sodium' => 'float', |         'sodium' => 'float', | ||||||
|         'unit_weight' => 'float', |  | ||||||
|     ]; |     ]; | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -94,19 +94,38 @@ class FoodAmount extends Model | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     /** |     /** | ||||||
|      * Get the multiplier for the food unit based on weight. |      * Get the multiplier for nutrient calculations based on serving data. | ||||||
|      * |  | ||||||
|      * Unit weight will be specified for foods that are added by unit |  | ||||||
|      * (e.g. eggs, vegetables, etc.) and cup weight (the weight of the |  | ||||||
|      * food equal to one cup) will be specified for foods that are |  | ||||||
|      * measured (e.g. flour, milk, etc.). |  | ||||||
|      */ |      */ | ||||||
|     private function unitMultiplier(): float { |     private function unitMultiplier(): float { | ||||||
|         return match ($this->unit) { |         if ($this->unit === 'oz') { | ||||||
|             null => $this->food->unit_weight, |             return $this->amount * 28.349523125 / $this->food->serving_weight; | ||||||
|             'tsp' => 1/48, |         } | ||||||
|             'tbsp' => 1/16, | 
 | ||||||
|             default => 1 |         if ($this->food->serving_unit === $this->unit) { | ||||||
|         } * $this->amount * ($this->food->cup_weight ?? 1) / 100; |             $multiplier = 1; | ||||||
|  |         } | ||||||
|  |         elseif ($this->unit === 'tsp') { | ||||||
|  |             $multiplier = match ($this->food->serving_unit) { | ||||||
|  |                 'tbsp' => 1/3, | ||||||
|  |                 'cup' => 1/48, | ||||||
|  |             }; | ||||||
|  |         } | ||||||
|  |         elseif ($this->unit === 'tbsp') { | ||||||
|  |             $multiplier = match ($this->food->serving_unit) { | ||||||
|  |                 'tsp' => 3, | ||||||
|  |                 'cup' => 1/16, | ||||||
|  |             }; | ||||||
|  |         } | ||||||
|  |         elseif ($this->unit === 'cup') { | ||||||
|  |             $multiplier = match ($this->food->serving_unit) { | ||||||
|  |                 'tsp' => 48, | ||||||
|  |                 'tbsp' => 16, | ||||||
|  |             }; | ||||||
|  |         } | ||||||
|  |         else { | ||||||
|  |             throw new \DomainException("Unhandled unit combination: {$this->unit}, {$this->food->serving_unit}"); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         return $multiplier / $this->food->serving_size * $this->amount; | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -15,14 +15,16 @@ class CreateFoodsTable extends Migration | ||||||
|             $table->id(); |             $table->id(); | ||||||
|             $table->string('name'); |             $table->string('name'); | ||||||
|             $table->string('detail')->nullable(); |             $table->string('detail')->nullable(); | ||||||
|  |             $table->string('brand')->nullable(); | ||||||
|  |             $table->unsignedFloat('serving_size'); | ||||||
|  |             $table->enum('serving_unit', ['tsp', 'tbsp', 'cup', 'oz'])->nullable(); | ||||||
|  |             $table->unsignedFloat('serving_weight'); | ||||||
|             $table->unsignedFloat('calories')->default(0); |             $table->unsignedFloat('calories')->default(0); | ||||||
|             $table->unsignedFloat('carbohydrates')->default(0); |  | ||||||
|             $table->unsignedFloat('cholesterol')->default(0); |  | ||||||
|             $table->unsignedFloat('fat')->default(0); |             $table->unsignedFloat('fat')->default(0); | ||||||
|             $table->unsignedFloat('protein')->default(0); |             $table->unsignedFloat('cholesterol')->default(0); | ||||||
|             $table->unsignedFloat('sodium')->default(0); |             $table->unsignedFloat('sodium')->default(0); | ||||||
|             $table->unsignedFloat('unit_weight')->nullable(); |             $table->unsignedFloat('carbohydrates')->default(0); | ||||||
|             $table->unsignedFloat('cup_weight')->nullable(); |             $table->unsignedFloat('protein')->default(0); | ||||||
|             $table->timestamps(); |             $table->timestamps(); | ||||||
|         }); |         }); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  | @ -19,7 +19,7 @@ class CreateFoodAmountsTable extends Migration | ||||||
|             $table->id(); |             $table->id(); | ||||||
|             $table->foreignIdFor(Food::class); |             $table->foreignIdFor(Food::class); | ||||||
|             $table->unsignedFloat('amount'); |             $table->unsignedFloat('amount'); | ||||||
|             $table->enum('unit', ['tsp', 'tbsp', 'cup', 'grams'])->nullable(); |             $table->enum('unit', ['tsp', 'tbsp', 'cup', 'oz'])->nullable(); | ||||||
|             $table->foreignIdFor(Recipe::class); |             $table->foreignIdFor(Recipe::class); | ||||||
|             $table->unsignedInteger('weight'); |             $table->unsignedInteger('weight'); | ||||||
|             $table->timestamps(); |             $table->timestamps(); | ||||||
|  |  | ||||||
|  | @ -15,74 +15,146 @@ class FoodSeeder extends Seeder | ||||||
|         $default_foods = [ |         $default_foods = [ | ||||||
|             [ |             [ | ||||||
|                 'name' => 'baking powder', |                 'name' => 'baking powder', | ||||||
|                 'calories' => 53, |                 'serving_size' => 1, | ||||||
|                 'carbohydrates' => 27.7, |                 'serving_unit' => 'tsp', | ||||||
|                 'sodium' => 10.6, |                 'serving_weight' => 4.6, | ||||||
|                 'cup_weight' => 220.8, |                 'calories' => 2.44, | ||||||
|  |                 'fat' => 0, | ||||||
|  |                 'cholesterol' => 0, | ||||||
|  |                 'sodium' => 0.488, | ||||||
|  |                 'carbohydrates' => 1.27, | ||||||
|  |                 'protein' => 0, | ||||||
|             ], |             ], | ||||||
|             [ |             [ | ||||||
|                 'name' => 'egg', |                 'name' => 'egg', | ||||||
|                 'detail' => 'large', |                 'detail' => 'large', | ||||||
|                 'calories' => 147, |                 'serving_size' => 1, | ||||||
|                 'carbohydrates' => 0.96, |                 'serving_weight' => 50.3, | ||||||
|                 'cholesterol' => 0.411, |                 'calories' => 71.9, | ||||||
|                 'fat' => 9.96, |                 'fat' => 5.01, | ||||||
|                 'protein' => 12.4, |                 'cholesterol' => 0.207, | ||||||
|                 'sodium' => 0.129, |                 'sodium' => 0.0649, | ||||||
|                 'unit_weight' => 50.3, |                 'carbohydrates' => 0.483, | ||||||
|  |                 'protein' => 6.24, | ||||||
|             ], |             ], | ||||||
|             [ |             [ | ||||||
|                 'name' => 'flour', |                 'name' => 'flour', | ||||||
|                 'detail' => 'all-purpose', |                 'detail' => 'all-purpose', | ||||||
|                 'calories' => 364, |                 'serving_size' => 1, | ||||||
|                 'carbohydrates' => 76.31, |                 'serving_unit' => 'cup', | ||||||
|                 'fat' => 0.98, |                 'serving_weight' => 125, | ||||||
|                 'protein' => 10.33, |                 'calories' => 455, | ||||||
|                 'sodium' => 0.004, |                 'fat' => 1.22, | ||||||
|                 'cup_weight' => 125, |                 'cholesterol' => 0, | ||||||
|  |                 'sodium' => 0.0025, | ||||||
|  |                 'carbohydrates' => 95.4, | ||||||
|  |                 'protein' => 12.9, | ||||||
|             ], |             ], | ||||||
|             [ |             [ | ||||||
|                 'name' => 'milk', |                 'name' => 'milk', | ||||||
|                 'detail' => 'whole', |                 'detail' => 'whole', | ||||||
|                 'calories' => 60, |                 'serving_size' => 1, | ||||||
|                 'carbohydrates' => 4.67, |                 'serving_unit' => 'cup', | ||||||
|                 'cholesterol' => 0.012, |                 'serving_weight' => 244, | ||||||
|                 'fat' => 3.2, |                 'calories' => 146, | ||||||
|                 'protein' => 3.28, |                 'fat' => 7.81, | ||||||
|                 'sodium' => 0.038, |                 'cholesterol' => 0.0293, | ||||||
|                 'cup_weight' => 244, |                 'sodium' => 0.0927, | ||||||
|  |                 'carbohydrates' => 11.4, | ||||||
|  |                 'protein' => 8, | ||||||
|             ], |             ], | ||||||
|             [ |             [ | ||||||
|                 'name' => 'salt', |                 'name' => 'salt', | ||||||
|                 'detail' => 'table', |                 'detail' => 'table', | ||||||
|                 'sodium' => 38.758, |                 'serving_size' => 1, | ||||||
|                 'cup_weight' => 292, |                 'serving_unit' => 'tsp', | ||||||
|  |                 'serving_weight' => 6, | ||||||
|  |                 'calories' => 0, | ||||||
|  |                 'fat' => 0, | ||||||
|  |                 'cholesterol' => 0, | ||||||
|  |                 'sodium' => 2.33, | ||||||
|  |                 'carbohydrates' => 0, | ||||||
|  |                 'protein' => 0, | ||||||
|  | 
 | ||||||
|             ], |             ], | ||||||
|             [ |             [ | ||||||
|                 'name' => 'sugar', |                 'name' => 'sugar', | ||||||
|                 'detail' => 'white', |                 'detail' => 'white', | ||||||
|                 'calories' => 385, |                 'serving_size' => 1, | ||||||
|                 'carbohydrates' => 99.6, |                 'serving_unit' => 'cup', | ||||||
|                 'fat' => 0.32, |                 'serving_weight' => 200, | ||||||
|  |                 'calories' => 770, | ||||||
|  |                 'fat' => 0.64, | ||||||
|  |                 'cholesterol' => 0, | ||||||
|  |                 'sodium' => 0.002, | ||||||
|  |                 'carbohydrates' => 199, | ||||||
|                 'protein' => 0, |                 'protein' => 0, | ||||||
|                 'sodium' => 0.001, |  | ||||||
|                 'cup_weight' => 200, |  | ||||||
|             ], |             ], | ||||||
|             [ |             [ | ||||||
|                 'name' => 'vegetable oil', |                 'name' => 'vegetable oil', | ||||||
|                 'calories' => 886, |                 'serving_size' => 1, | ||||||
|                 'fat' => 100, |                 'serving_unit' => 'tbsp', | ||||||
|                 'cup_weight' => 224, |                 'serving_weight' => 14, | ||||||
|  |                 'calories' => 124, | ||||||
|  |                 'fat' => 14, | ||||||
|  |                 'cholesterol' => 0, | ||||||
|  |                 'sodium' => 0, | ||||||
|  |                 'carbohydrates' => 0, | ||||||
|  |                 'protein' => 0, | ||||||
|             ], |             ], | ||||||
|             [ |             [ | ||||||
|                 'name' => 'peanut butter', |                 'name' => 'peanut butter', | ||||||
|                 'detail' => 'Kirkland organic creamy', |                 'detail' => 'organic creamy', | ||||||
|                 'calories' => 562.5, |                 'brand' => 'Kirkland', | ||||||
|                 'fat' => 46.875, |                 'serving_size' => 2, | ||||||
|                 'sodium' => 0.203125, |                 'serving_unit' => 'tbsp', | ||||||
|                 'carbohydrates' => 21.875, |                 'serving_weight' => 32, | ||||||
|                 'protein' => 25, |                 'calories' => 180, | ||||||
|                 'cup_weight' => 256, |                 'fat' => 15, | ||||||
|  |                 'cholesterol' => 0, | ||||||
|  |                 'sodium' => 0.065, | ||||||
|  |                 'carbohydrates' => 7, | ||||||
|  |                 'protein' => 8, | ||||||
|  |             ], | ||||||
|  |             [ | ||||||
|  |                 'name' => 'raisins', | ||||||
|  |                 'brand' => 'Kroger', | ||||||
|  |                 'serving_size' => 0.25, | ||||||
|  |                 'serving_unit' => 'cup', | ||||||
|  |                 'serving_weight' => 40, | ||||||
|  |                 'calories' => 140, | ||||||
|  |                 'fat' => 0, | ||||||
|  |                 'cholesterol' => 0, | ||||||
|  |                 'sodium' => 0.010, | ||||||
|  |                 'carbohydrates' => 33, | ||||||
|  |                 'protein' => 1, | ||||||
|  |             ], | ||||||
|  |             [ | ||||||
|  |                 'name' => 'peanuts', | ||||||
|  |                 'detail' => 'dry roasted, unsalted', | ||||||
|  |                 'brand' => 'Kroger', | ||||||
|  |                 'serving_size' => 0.25, | ||||||
|  |                 'serving_unit' => 'cup', | ||||||
|  |                 'serving_weight' => 28, | ||||||
|  |                 'calories' => 160, | ||||||
|  |                 'fat' => 14, | ||||||
|  |                 'cholesterol' => 0, | ||||||
|  |                 'sodium' => 0, | ||||||
|  |                 'carbohydrates' => 6, | ||||||
|  |                 'protein' => 7, | ||||||
|  |             ], | ||||||
|  |             [ | ||||||
|  |                 'name' => 'canned corn', | ||||||
|  |                 'detail' => 'golden sweet', | ||||||
|  |                 'brand' => 'WinCo', | ||||||
|  |                 'serving_size' => 0.5, | ||||||
|  |                 'serving_unit' => 'cup', | ||||||
|  |                 'serving_weight' => 125, | ||||||
|  |                 'calories' => 60, | ||||||
|  |                 'fat' => 0.5, | ||||||
|  |                 'sodium' => 0.2, | ||||||
|  |                 'carbohydrates' => 9, | ||||||
|  |                 'protein' => 1, | ||||||
|             ], |             ], | ||||||
|         ]; |         ]; | ||||||
|         Food::factory()->createMany($default_foods); |         Food::factory()->createMany($default_foods); | ||||||
|  |  | ||||||
|  | @ -27,8 +27,8 @@ class RecipeSeeder extends Seeder | ||||||
|             [ |             [ | ||||||
|                 'food_id' => Food::where('name', 'flour') |                 'food_id' => Food::where('name', 'flour') | ||||||
|                     ->first()->id, |                     ->first()->id, | ||||||
|                 'amount' => 1, |                 'amount' => 4.25, | ||||||
|                 'unit' => 'cup', |                 'unit' => 'oz', | ||||||
|                 'recipe_id' => $recipe->id, |                 'recipe_id' => $recipe->id, | ||||||
|                 'weight' => $weight++, |                 'weight' => $weight++, | ||||||
|             ], |             ], | ||||||
|  | @ -95,5 +95,47 @@ class RecipeSeeder extends Seeder | ||||||
|             ] |             ] | ||||||
|         ]; |         ]; | ||||||
|         RecipeStep::factory()->createMany($steps); |         RecipeStep::factory()->createMany($steps); | ||||||
|  | 
 | ||||||
|  |         /** @var \App\Models\Recipe $recipe */ | ||||||
|  |         $recipe = Recipe::factory()->create([ | ||||||
|  |             'name' => 'peanut butter corn', | ||||||
|  |             'description' => 'Peanut butter and corn -- YUM', | ||||||
|  |             'servings' => 4, | ||||||
|  |         ]); | ||||||
|  | 
 | ||||||
|  |         $weight = 0; | ||||||
|  |         $amounts = [ | ||||||
|  |             [ | ||||||
|  |                 'food_id' => Food::where('name', 'peanut butter') | ||||||
|  |                     ->first()->id, | ||||||
|  |                 'amount' => 2, | ||||||
|  |                 'unit' => 'cup', | ||||||
|  |                 'recipe_id' => $recipe->id, | ||||||
|  |                 'weight' => $weight++, | ||||||
|  |             ], | ||||||
|  |             [ | ||||||
|  |                 'food_id' => Food::where('name', 'canned corn') | ||||||
|  |                     ->first()->id, | ||||||
|  |                 'amount' => 15.25, | ||||||
|  |                 'unit' => 'oz', | ||||||
|  |                 'recipe_id' => $recipe->id, | ||||||
|  |                 'weight' => $weight++, | ||||||
|  |             ], | ||||||
|  |         ]; | ||||||
|  |         FoodAmount::factory()->createMany($amounts); | ||||||
|  | 
 | ||||||
|  |         $steps = [ | ||||||
|  |             [ | ||||||
|  |                 'recipe_id' => $recipe->id, | ||||||
|  |                 'number' => 1, | ||||||
|  |                 'step' => 'Mix it together.', | ||||||
|  |             ], | ||||||
|  |             [ | ||||||
|  |                 'recipe_id' => $recipe->id, | ||||||
|  |                 'number' => 2, | ||||||
|  |                 'step' => 'Eat it.', | ||||||
|  |             ] | ||||||
|  |         ]; | ||||||
|  |         RecipeStep::factory()->createMany($steps); | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
		Loading…
	
		Reference in New Issue