1
|
|
|
<?php |
2
|
|
|
/** |
3
|
|
|
* Recipe plugin for Craft CMS 3.x |
4
|
|
|
* |
5
|
|
|
* A comprehensive recipe FieldType for Craft CMS that includes metric/imperial |
6
|
|
|
* conversion, portion calculation, and JSON-LD microdata support |
7
|
|
|
* |
8
|
|
|
* @link https://nystudio107.com |
|
|
|
|
9
|
|
|
* @copyright Copyright (c) 2017 nystudio107 |
|
|
|
|
10
|
|
|
*/ |
|
|
|
|
11
|
|
|
|
12
|
|
|
namespace nystudio107\recipe\models; |
13
|
|
|
|
14
|
|
|
use Craft; |
15
|
|
|
use craft\base\Model; |
16
|
|
|
use craft\helpers\StringHelper; |
17
|
|
|
use craft\helpers\Template; |
18
|
|
|
use craft\validators\ArrayValidator; |
19
|
|
|
use Exception; |
20
|
|
|
use nystudio107\recipe\helpers\Json; |
21
|
|
|
use nystudio107\recipe\helpers\PluginTemplate; |
22
|
|
|
use nystudio107\seomatic\models\MetaJsonLd; |
23
|
|
|
use nystudio107\seomatic\Seomatic; |
24
|
|
|
use Twig\Markup; |
25
|
|
|
|
26
|
|
|
/** |
|
|
|
|
27
|
|
|
* @author nystudio107 |
|
|
|
|
28
|
|
|
* @package Recipe |
|
|
|
|
29
|
|
|
* @since 1.0.0 |
|
|
|
|
30
|
|
|
*/ |
|
|
|
|
31
|
|
|
class Recipe extends Model |
32
|
|
|
{ |
33
|
|
|
// Constants |
34
|
|
|
// ========================================================================= |
35
|
|
|
|
36
|
|
|
public const SEOMATIC_PLUGIN_HANDLE = 'seomatic'; |
37
|
|
|
|
38
|
|
|
public const MAIN_ENTITY_KEY = 'mainEntityOfPage'; |
39
|
|
|
|
40
|
|
|
public const US_RDA = [ |
41
|
|
|
'calories' => 2000, |
42
|
|
|
'carbohydrateContent' => 275, |
43
|
|
|
'cholesterolContent' => 300, |
44
|
|
|
'fatContent' => 78, |
45
|
|
|
'fiberContent' => 28, |
46
|
|
|
'proteinContent' => 50, |
47
|
|
|
'saturatedFatContent' => 20, |
48
|
|
|
'sodiumContent' => 2300, |
49
|
|
|
'sugarContent' => 50, |
50
|
|
|
]; |
51
|
|
|
|
52
|
|
|
// Mapping to convert any of the incorrect plural values |
53
|
|
|
public const NORMALIZE_PLURALS = [ |
54
|
|
|
'tsps' => 'tsp', |
55
|
|
|
'tbsps' => 'tbsp', |
56
|
|
|
'flozs' => 'floz', |
57
|
|
|
'cups' => 'cups', |
58
|
|
|
'ozs' => 'oz', |
59
|
|
|
'lbs' => 'lb', |
60
|
|
|
'mls' => 'ml', |
61
|
|
|
'ls' => 'l', |
62
|
|
|
'mgs' => 'mg', |
63
|
|
|
'gs' => 'g', |
64
|
|
|
'kg' => 'kg', |
65
|
|
|
]; |
66
|
|
|
|
67
|
|
|
// Public Properties |
68
|
|
|
// ========================================================================= |
69
|
|
|
|
70
|
|
|
/** |
|
|
|
|
71
|
|
|
* @var string |
72
|
|
|
*/ |
73
|
|
|
public string $name = ''; |
74
|
|
|
|
75
|
|
|
/** |
|
|
|
|
76
|
|
|
* @var string |
77
|
|
|
*/ |
78
|
|
|
public string $author = ''; |
79
|
|
|
|
80
|
|
|
/** |
|
|
|
|
81
|
|
|
* @var string |
82
|
|
|
*/ |
83
|
|
|
public string $description = ''; |
84
|
|
|
|
85
|
|
|
/** |
|
|
|
|
86
|
|
|
* @var string |
87
|
|
|
*/ |
88
|
|
|
public string $keywords = ''; |
89
|
|
|
|
90
|
|
|
/** |
|
|
|
|
91
|
|
|
* @var string |
92
|
|
|
*/ |
93
|
|
|
public string $recipeCategory = ''; |
94
|
|
|
|
95
|
|
|
/** |
|
|
|
|
96
|
|
|
* @var string |
97
|
|
|
*/ |
98
|
|
|
public string $recipeCuisine = ''; |
99
|
|
|
|
100
|
|
|
/** |
|
|
|
|
101
|
|
|
* @var string |
102
|
|
|
*/ |
103
|
|
|
public string $skill = 'intermediate'; |
104
|
|
|
|
105
|
|
|
/** |
|
|
|
|
106
|
|
|
* @var int |
107
|
|
|
*/ |
108
|
|
|
public int $serves = 1; |
109
|
|
|
|
110
|
|
|
/** |
|
|
|
|
111
|
|
|
* @var string |
112
|
|
|
*/ |
113
|
|
|
public string $servesUnit = ''; |
114
|
|
|
|
115
|
|
|
/** |
|
|
|
|
116
|
|
|
* @var array |
117
|
|
|
*/ |
118
|
|
|
public array $ingredients = []; |
119
|
|
|
|
120
|
|
|
/** |
|
|
|
|
121
|
|
|
* @var array |
122
|
|
|
*/ |
123
|
|
|
public array $directions = []; |
124
|
|
|
|
125
|
|
|
/** |
|
|
|
|
126
|
|
|
* @var array |
127
|
|
|
*/ |
128
|
|
|
public array $equipment = []; |
129
|
|
|
|
130
|
|
|
/** |
|
|
|
|
131
|
|
|
* @var ?int |
132
|
|
|
*/ |
133
|
|
|
public ?int $imageId = null; |
134
|
|
|
|
135
|
|
|
/** |
|
|
|
|
136
|
|
|
* @var ?int |
137
|
|
|
*/ |
138
|
|
|
public ?int $videoId = null; |
139
|
|
|
|
140
|
|
|
/** |
|
|
|
|
141
|
|
|
* @var int |
142
|
|
|
*/ |
143
|
|
|
public int $prepTime = 0; |
144
|
|
|
|
145
|
|
|
/** |
|
|
|
|
146
|
|
|
* @var int |
147
|
|
|
*/ |
148
|
|
|
public int $cookTime = 0; |
149
|
|
|
|
150
|
|
|
/** |
|
|
|
|
151
|
|
|
* @var int |
152
|
|
|
*/ |
153
|
|
|
public int $totalTime = 0; |
154
|
|
|
|
155
|
|
|
/** |
|
|
|
|
156
|
|
|
* @var ?array |
157
|
|
|
*/ |
158
|
|
|
public ?array $ratings = null; |
159
|
|
|
|
160
|
|
|
/** |
|
|
|
|
161
|
|
|
* @var string |
162
|
|
|
*/ |
163
|
|
|
public string $servingSize = ''; |
164
|
|
|
|
165
|
|
|
/** |
|
|
|
|
166
|
|
|
* @var int |
167
|
|
|
*/ |
168
|
|
|
public int $calories = 0; |
169
|
|
|
|
170
|
|
|
/** |
|
|
|
|
171
|
|
|
* @var int |
172
|
|
|
*/ |
173
|
|
|
public int $carbohydrateContent = 0; |
174
|
|
|
|
175
|
|
|
/** |
|
|
|
|
176
|
|
|
* @var int |
177
|
|
|
*/ |
178
|
|
|
public int $cholesterolContent = 0; |
179
|
|
|
|
180
|
|
|
/** |
|
|
|
|
181
|
|
|
* @var int |
182
|
|
|
*/ |
183
|
|
|
public int $fatContent = 0; |
184
|
|
|
|
185
|
|
|
/** |
|
|
|
|
186
|
|
|
* @var int |
187
|
|
|
*/ |
188
|
|
|
public int $fiberContent = 0; |
189
|
|
|
|
190
|
|
|
/** |
|
|
|
|
191
|
|
|
* @var int |
192
|
|
|
*/ |
193
|
|
|
public int $proteinContent = 0; |
194
|
|
|
|
195
|
|
|
/** |
|
|
|
|
196
|
|
|
* @var int |
197
|
|
|
*/ |
198
|
|
|
public int $saturatedFatContent = 0; |
199
|
|
|
|
200
|
|
|
/** |
|
|
|
|
201
|
|
|
* @var int |
202
|
|
|
*/ |
203
|
|
|
public int $sodiumContent = 0; |
204
|
|
|
|
205
|
|
|
/** |
|
|
|
|
206
|
|
|
* @var int |
207
|
|
|
*/ |
208
|
|
|
public int $sugarContent = 0; |
209
|
|
|
|
210
|
|
|
/** |
|
|
|
|
211
|
|
|
* @var int |
212
|
|
|
*/ |
213
|
|
|
public int $transFatContent = 0; |
214
|
|
|
|
215
|
|
|
/** |
|
|
|
|
216
|
|
|
* @var int |
217
|
|
|
*/ |
218
|
|
|
public int $unsaturatedFatContent = 0; |
219
|
|
|
|
220
|
|
|
// Public Methods |
221
|
|
|
// ========================================================================= |
222
|
|
|
|
223
|
|
|
/** |
|
|
|
|
224
|
|
|
* @inheritdoc |
225
|
|
|
*/ |
|
|
|
|
226
|
|
|
public function init(): void |
227
|
|
|
{ |
228
|
|
|
parent::init(); |
229
|
|
|
// Fix any of the incorrect plural values |
230
|
|
|
if (!empty($this->ingredients)) { |
231
|
|
|
foreach ($this->ingredients as &$row) { |
232
|
|
|
if (!empty($row['units']) && !empty(self::NORMALIZE_PLURALS[$row['units']])) { |
233
|
|
|
$row['units'] = self::NORMALIZE_PLURALS[$row['units']]; |
234
|
|
|
} |
235
|
|
|
} |
236
|
|
|
|
237
|
|
|
unset($row); |
238
|
|
|
} |
239
|
|
|
} |
240
|
|
|
|
241
|
|
|
/** |
|
|
|
|
242
|
|
|
* @inheritdoc |
243
|
|
|
*/ |
|
|
|
|
244
|
|
|
public function rules(): array |
245
|
|
|
{ |
246
|
|
|
return [ |
247
|
|
|
['name', 'string'], |
248
|
|
|
['author', 'string'], |
249
|
|
|
['name', 'default', 'value' => ''], |
250
|
|
|
['description', 'string'], |
251
|
|
|
['keywords', 'string'], |
252
|
|
|
['recipeCategory', 'string'], |
253
|
|
|
['recipeCuisine', 'string'], |
254
|
|
|
['skill', 'string'], |
255
|
|
|
['serves', 'integer'], |
256
|
|
|
['imageId', 'integer'], |
257
|
|
|
['videoId', 'integer'], |
258
|
|
|
['prepTime', 'integer'], |
259
|
|
|
['cookTime', 'integer'], |
260
|
|
|
['totalTime', 'integer'], |
261
|
|
|
['servingSize', 'string'], |
262
|
|
|
['calories', 'integer'], |
263
|
|
|
['carbohydrateContent', 'integer'], |
264
|
|
|
['cholesterolContent', 'integer'], |
265
|
|
|
['fatContent', 'integer'], |
266
|
|
|
['fiberContent', 'integer'], |
267
|
|
|
['proteinContent', 'integer'], |
268
|
|
|
['saturatedFatContent', 'integer'], |
269
|
|
|
['sodiumContent', 'integer'], |
270
|
|
|
['sugarContent', 'integer'], |
271
|
|
|
['transFatContent', 'integer'], |
272
|
|
|
['unsaturatedFatContent', 'integer'], |
273
|
|
|
[ |
274
|
|
|
[ |
275
|
|
|
'ingredients', |
276
|
|
|
'directions', |
277
|
|
|
'equipment', |
278
|
|
|
], |
279
|
|
|
ArrayValidator::class, |
280
|
|
|
], |
281
|
|
|
|
282
|
|
|
]; |
283
|
|
|
} |
284
|
|
|
|
285
|
|
|
/** |
286
|
|
|
* Return the JSON-LD Structured Data for this recipe |
287
|
|
|
*/ |
|
|
|
|
288
|
|
|
public function getRecipeJSONLD(): array |
289
|
|
|
{ |
290
|
|
|
$recipeJSONLD = [ |
291
|
|
|
'context' => 'https://schema.org', |
292
|
|
|
'type' => 'Recipe', |
293
|
|
|
'name' => $this->name, |
294
|
|
|
'image' => $this->getImageUrl(), |
295
|
|
|
'description' => $this->description, |
296
|
|
|
'keywords' => $this->keywords, |
297
|
|
|
'recipeCategory' => $this->recipeCategory, |
298
|
|
|
'recipeCuisine' => $this->recipeCuisine, |
299
|
|
|
'recipeYield' => $this->getServes(), |
300
|
|
|
'recipeIngredient' => $this->getIngredients('imperial', 0, false), |
301
|
|
|
'recipeInstructions' => $this->getDirections(false), |
302
|
|
|
'tool' => $this->getEquipment(false), |
303
|
|
|
]; |
304
|
|
|
$recipeJSONLD = array_filter($recipeJSONLD); |
305
|
|
|
|
306
|
|
|
if (!empty($this->author)) { |
307
|
|
|
$author = [ |
308
|
|
|
'type' => 'Person', |
309
|
|
|
'name' => $this->author, |
310
|
|
|
]; |
311
|
|
|
$author = array_filter($author); |
312
|
|
|
$recipeJSONLD['author'] = $author; |
313
|
|
|
} |
314
|
|
|
|
315
|
|
|
$videoUrl = $this->getVideoUrl(); |
316
|
|
|
if (!empty($videoUrl)) { |
317
|
|
|
$video = [ |
318
|
|
|
'type' => 'VideoObject', |
319
|
|
|
'name' => $this->name, |
320
|
|
|
'description' => $this->description, |
321
|
|
|
'contentUrl' => $videoUrl, |
322
|
|
|
'thumbnailUrl' => $this->getImageUrl(), |
323
|
|
|
'uploadDate' => $this->getVideoUploadedDate(), |
324
|
|
|
]; |
325
|
|
|
$video = array_filter($video); |
326
|
|
|
$recipeJSONLD['video'] = $video; |
327
|
|
|
} |
328
|
|
|
|
329
|
|
|
$nutrition = [ |
330
|
|
|
'type' => 'NutritionInformation', |
331
|
|
|
'servingSize' => $this->servingSize, |
332
|
|
|
'calories' => $this->calories, |
333
|
|
|
'carbohydrateContent' => $this->carbohydrateContent, |
334
|
|
|
'cholesterolContent' => $this->cholesterolContent, |
335
|
|
|
'fatContent' => $this->fatContent, |
336
|
|
|
'fiberContent' => $this->fiberContent, |
337
|
|
|
'proteinContent' => $this->proteinContent, |
338
|
|
|
'saturatedFatContent' => $this->saturatedFatContent, |
339
|
|
|
'sodiumContent' => $this->sodiumContent, |
340
|
|
|
'sugarContent' => $this->sugarContent, |
341
|
|
|
'transFatContent' => $this->transFatContent, |
342
|
|
|
'unsaturatedFatContent' => $this->unsaturatedFatContent, |
343
|
|
|
]; |
344
|
|
|
$nutrition = array_filter($nutrition); |
345
|
|
|
$recipeJSONLD['nutrition'] = $nutrition; |
346
|
|
|
if (count($recipeJSONLD['nutrition']) === 1) { |
347
|
|
|
unset($recipeJSONLD['nutrition']); |
348
|
|
|
} |
349
|
|
|
|
350
|
|
|
$aggregateRating = $this->getAggregateRating(); |
351
|
|
|
if ($aggregateRating) { |
352
|
|
|
$aggregateRatings = [ |
353
|
|
|
'type' => 'AggregateRating', |
354
|
|
|
'ratingCount' => $this->getRatingsCount(), |
355
|
|
|
'bestRating' => '5', |
356
|
|
|
'worstRating' => '1', |
357
|
|
|
'ratingValue' => $aggregateRating, |
358
|
|
|
]; |
359
|
|
|
$aggregateRatings = array_filter($aggregateRatings); |
360
|
|
|
$recipeJSONLD['aggregateRating'] = $aggregateRatings; |
361
|
|
|
|
362
|
|
|
$reviews = []; |
363
|
|
|
foreach ($this->ratings as $rating) { |
364
|
|
|
$review = [ |
365
|
|
|
'type' => 'Review', |
366
|
|
|
'author' => $rating['author'], |
367
|
|
|
'name' => $this->name . ' ' . Craft::t('recipe', 'Review'), |
368
|
|
|
'description' => $rating['review'], |
369
|
|
|
'reviewRating' => [ |
370
|
|
|
'type' => 'Rating', |
371
|
|
|
'bestRating' => '5', |
372
|
|
|
'worstRating' => '1', |
373
|
|
|
'ratingValue' => $rating['rating'], |
374
|
|
|
], |
375
|
|
|
]; |
376
|
|
|
$reviews[] = $review; |
377
|
|
|
} |
378
|
|
|
|
379
|
|
|
$reviews = array_filter($reviews); |
380
|
|
|
$recipeJSONLD['review'] = $reviews; |
381
|
|
|
} |
382
|
|
|
|
383
|
|
|
if ($this->prepTime !== 0) { |
384
|
|
|
$recipeJSONLD['prepTime'] = 'PT' . $this->prepTime . 'M'; |
385
|
|
|
} |
386
|
|
|
|
387
|
|
|
if ($this->cookTime !== 0) { |
388
|
|
|
$recipeJSONLD['cookTime'] = 'PT' . $this->cookTime . 'M'; |
389
|
|
|
} |
390
|
|
|
|
391
|
|
|
if ($this->totalTime !== 0) { |
392
|
|
|
$recipeJSONLD['totalTime'] = 'PT' . $this->totalTime . 'M'; |
393
|
|
|
} |
394
|
|
|
|
395
|
|
|
return $recipeJSONLD; |
396
|
|
|
} |
397
|
|
|
|
398
|
|
|
/** |
|
|
|
|
399
|
|
|
* Create the SEOmatic MetaJsonLd object for this recipe |
400
|
|
|
* |
401
|
|
|
* @param null $key |
|
|
|
|
402
|
|
|
*/ |
|
|
|
|
403
|
|
|
public function createRecipeMetaJsonLd($key = null, bool $add = true): ?MetaJsonLd |
404
|
|
|
{ |
405
|
|
|
$result = null; |
406
|
|
|
if (Craft::$app->getPlugins()->getPlugin(self::SEOMATIC_PLUGIN_HANDLE)) { |
407
|
|
|
$seomatic = Seomatic::getInstance(); |
408
|
|
|
if ($seomatic !== null) { |
409
|
|
|
$recipeJson = $this->getRecipeJSONLD(); |
410
|
|
|
// If we're adding the MetaJsonLd to the container, and no key is provided, give it a random key |
411
|
|
|
if ($add && $key === null) { |
412
|
|
|
try { |
413
|
|
|
$key = StringHelper::UUID(); |
414
|
|
|
} catch (Exception) { |
415
|
|
|
// That's okay |
416
|
|
|
} |
417
|
|
|
} |
418
|
|
|
|
419
|
|
|
if ($key !== null) { |
420
|
|
|
$recipeJson['key'] = $key; |
421
|
|
|
} |
422
|
|
|
|
423
|
|
|
// If the key is `mainEntityOfPage` add in the URL |
424
|
|
|
if ($key === self::MAIN_ENTITY_KEY) { |
425
|
|
|
$mainEntity = Seomatic::$plugin->jsonLd->get(self::MAIN_ENTITY_KEY); |
426
|
|
|
if ($mainEntity !== null) { |
427
|
|
|
$recipeJson[self::MAIN_ENTITY_KEY] = $mainEntity[self::MAIN_ENTITY_KEY]; |
428
|
|
|
} |
429
|
|
|
} |
430
|
|
|
|
431
|
|
|
$result = Seomatic::$plugin->jsonLd->create( |
432
|
|
|
$recipeJson, |
433
|
|
|
$add |
434
|
|
|
); |
435
|
|
|
} |
436
|
|
|
} |
437
|
|
|
|
438
|
|
|
return $result; |
439
|
|
|
} |
440
|
|
|
|
441
|
|
|
/** |
|
|
|
|
442
|
|
|
* Render the JSON-LD Structured Data for this recipe |
443
|
|
|
* |
444
|
|
|
* |
445
|
|
|
*/ |
|
|
|
|
446
|
|
|
public function renderRecipeJSONLD(bool $raw = true): string|Markup |
447
|
|
|
{ |
448
|
|
|
return $this->renderJsonLd($this->getRecipeJSONLD(), $raw); |
449
|
|
|
} |
450
|
|
|
|
451
|
|
|
/** |
452
|
|
|
* Get the URL to the recipe's image |
453
|
|
|
* |
454
|
|
|
* @param null $transform |
|
|
|
|
455
|
|
|
*/ |
|
|
|
|
456
|
|
|
public function getImageUrl($transform = null): ?string |
457
|
|
|
{ |
458
|
|
|
$result = ''; |
459
|
|
|
if ($this->imageId) { |
|
|
|
|
460
|
|
|
$image = Craft::$app->getAssets()->getAssetById($this->imageId); |
461
|
|
|
if ($image) { |
462
|
|
|
$result = $image->getUrl($transform); |
463
|
|
|
} |
464
|
|
|
} |
465
|
|
|
|
466
|
|
|
return $result; |
467
|
|
|
} |
468
|
|
|
|
469
|
|
|
/** |
470
|
|
|
* Get the URL to the recipe's video |
471
|
|
|
*/ |
|
|
|
|
472
|
|
|
public function getVideoUrl(): ?string |
473
|
|
|
{ |
474
|
|
|
$result = ''; |
475
|
|
|
if ($this->videoId) { |
|
|
|
|
476
|
|
|
$video = Craft::$app->getAssets()->getAssetById($this->videoId); |
477
|
|
|
if ($video) { |
478
|
|
|
$result = $video->getUrl(); |
479
|
|
|
} |
480
|
|
|
} |
481
|
|
|
|
482
|
|
|
return $result; |
483
|
|
|
} |
484
|
|
|
|
485
|
|
|
/** |
486
|
|
|
* Get the URL to the recipe's uploaded date |
487
|
|
|
*/ |
|
|
|
|
488
|
|
|
public function getVideoUploadedDate(): ?string |
489
|
|
|
{ |
490
|
|
|
$result = ''; |
491
|
|
|
if ($this->videoId) { |
|
|
|
|
492
|
|
|
$video = Craft::$app->getAssets()->getAssetById($this->videoId); |
493
|
|
|
if ($video) { |
494
|
|
|
$result = $video->dateCreated->format('c'); |
495
|
|
|
} |
496
|
|
|
} |
497
|
|
|
|
498
|
|
|
return $result; |
499
|
|
|
} |
500
|
|
|
|
501
|
|
|
/** |
|
|
|
|
502
|
|
|
* Render the Nutrition Facts template |
503
|
|
|
*/ |
|
|
|
|
504
|
|
|
public function renderNutritionFacts(array $rda = self::US_RDA): Markup |
505
|
|
|
{ |
506
|
|
|
return PluginTemplate::renderPluginTemplate( |
507
|
|
|
'recipe-nutrition-facts', |
508
|
|
|
[ |
509
|
|
|
'value' => $this, |
510
|
|
|
'rda' => $rda, |
511
|
|
|
] |
512
|
|
|
); |
513
|
|
|
} |
514
|
|
|
|
515
|
|
|
/** |
|
|
|
|
516
|
|
|
* Get all the ingredients for this recipe |
517
|
|
|
*/ |
|
|
|
|
518
|
|
|
public function getIngredients(string $outputUnits = 'imperial', int $serving = 0, bool $raw = true): array |
519
|
|
|
{ |
520
|
|
|
$result = []; |
521
|
|
|
|
522
|
|
|
foreach ($this->ingredients as $row) { |
523
|
|
|
$convertedUnits = ''; |
524
|
|
|
$ingredient = ''; |
525
|
|
|
if ($row['quantity']) { |
526
|
|
|
// Multiply the quantity by how many servings we want |
527
|
|
|
$multiplier = 1; |
528
|
|
|
if ($serving > 0) { |
529
|
|
|
$multiplier = $serving / $this->serves; |
530
|
|
|
} |
531
|
|
|
|
532
|
|
|
$quantity = $row['quantity'] * $multiplier; |
533
|
|
|
$originalQuantity = $quantity; |
534
|
|
|
|
535
|
|
|
// Do the imperial->metric units conversion |
536
|
|
|
if ($outputUnits === 'imperial') { |
537
|
|
|
switch ($row['units']) { |
538
|
|
|
case 'ml': |
|
|
|
|
539
|
|
|
$convertedUnits = 'tsp'; |
540
|
|
|
$quantity *= 0.2; |
541
|
|
|
break; |
542
|
|
|
case 'l': |
|
|
|
|
543
|
|
|
$convertedUnits = 'cups'; |
544
|
|
|
$quantity *= 4.2; |
545
|
|
|
break; |
546
|
|
|
case 'mg': |
|
|
|
|
547
|
|
|
$convertedUnits = 'oz'; |
548
|
|
|
$quantity *= 0.000035274; |
549
|
|
|
break; |
550
|
|
|
case 'g': |
|
|
|
|
551
|
|
|
$convertedUnits = 'oz'; |
552
|
|
|
$quantity *= 0.035274; |
553
|
|
|
break; |
554
|
|
|
case 'kg': |
|
|
|
|
555
|
|
|
$convertedUnits = 'lb'; |
556
|
|
|
$quantity *= 2.2046226218; |
557
|
|
|
break; |
558
|
|
|
} |
559
|
|
|
} |
560
|
|
|
|
561
|
|
|
// Do the metric->imperial units conversion |
562
|
|
|
if ($outputUnits === 'metric') { |
563
|
|
|
switch ($row['units']) { |
564
|
|
|
case 'tsp': |
|
|
|
|
565
|
|
|
$convertedUnits = 'ml'; |
566
|
|
|
$quantity *= 4.929; |
567
|
|
|
break; |
568
|
|
|
case 'tbsp': |
|
|
|
|
569
|
|
|
$convertedUnits = 'ml'; |
570
|
|
|
$quantity *= 14.787; |
571
|
|
|
break; |
572
|
|
|
case 'floz': |
|
|
|
|
573
|
|
|
$convertedUnits = 'ml'; |
574
|
|
|
$quantity *= 29.574; |
575
|
|
|
break; |
576
|
|
|
case 'cups': |
|
|
|
|
577
|
|
|
$convertedUnits = 'l'; |
578
|
|
|
$quantity *= 0.236588; |
579
|
|
|
break; |
580
|
|
|
case 'oz': |
|
|
|
|
581
|
|
|
$convertedUnits = 'g'; |
582
|
|
|
$quantity *= 28.3495; |
583
|
|
|
break; |
584
|
|
|
case 'lb': |
|
|
|
|
585
|
|
|
$convertedUnits = 'kg'; |
586
|
|
|
$quantity *= 0.45359237; |
587
|
|
|
break; |
588
|
|
|
} |
589
|
|
|
|
590
|
|
|
$quantity = round($quantity, 1); |
591
|
|
|
} |
592
|
|
|
|
593
|
|
|
// Convert units to nice fractions |
594
|
|
|
$quantity = $this->convertToFractions($quantity); |
595
|
|
|
|
596
|
|
|
$ingredient .= $quantity; |
597
|
|
|
|
598
|
|
|
if ($row['units']) { |
599
|
|
|
$units = $row['units']; |
600
|
|
|
if ($convertedUnits !== '' && $convertedUnits !== '0') { |
601
|
|
|
$units = $convertedUnits; |
602
|
|
|
} |
603
|
|
|
|
604
|
|
|
if ($originalQuantity <= 1) { |
605
|
|
|
$units = rtrim($units); |
606
|
|
|
$units = rtrim($units, 's'); |
607
|
|
|
} |
608
|
|
|
|
609
|
|
|
$ingredient .= ' ' . $units; |
610
|
|
|
} |
611
|
|
|
} |
612
|
|
|
|
613
|
|
|
if ($row['ingredient']) { |
614
|
|
|
$ingredient .= ' ' . $row['ingredient']; |
615
|
|
|
} |
616
|
|
|
|
617
|
|
|
if ($raw) { |
618
|
|
|
$ingredient = Template::raw($ingredient); |
619
|
|
|
} |
620
|
|
|
|
621
|
|
|
$result[] = $ingredient; |
622
|
|
|
} |
623
|
|
|
|
624
|
|
|
return $result; |
625
|
|
|
} |
626
|
|
|
|
627
|
|
|
/** |
|
|
|
|
628
|
|
|
* Get all the directions for this recipe |
629
|
|
|
* |
630
|
|
|
* |
631
|
|
|
* @return array |
|
|
|
|
632
|
|
|
*/ |
633
|
|
|
public function getDirections(bool $raw = true): array |
634
|
|
|
{ |
635
|
|
|
$result = []; |
636
|
|
|
foreach ($this->directions as $row) { |
637
|
|
|
$direction = $row['direction']; |
638
|
|
|
if ($raw) { |
639
|
|
|
$direction = Template::raw($direction); |
640
|
|
|
} |
641
|
|
|
|
642
|
|
|
$result[] = $direction; |
643
|
|
|
} |
644
|
|
|
|
645
|
|
|
return $result; |
646
|
|
|
} |
647
|
|
|
|
648
|
|
|
/** |
|
|
|
|
649
|
|
|
* Get all the equipment for this recipe |
650
|
|
|
*/ |
|
|
|
|
651
|
|
|
public function getEquipment(bool $raw = true): array |
652
|
|
|
{ |
653
|
|
|
$result = []; |
654
|
|
|
foreach ($this->equipment as $row) { |
655
|
|
|
$equipment = $row['equipment']; |
656
|
|
|
if ($raw) { |
657
|
|
|
$equipment = Template::raw($equipment); |
658
|
|
|
} |
659
|
|
|
|
660
|
|
|
$result[] = $equipment; |
661
|
|
|
} |
662
|
|
|
|
663
|
|
|
return $result; |
664
|
|
|
} |
665
|
|
|
|
666
|
|
|
/** |
667
|
|
|
* Get the aggregate rating from all the ratings |
668
|
|
|
*/ |
|
|
|
|
669
|
|
|
public function getAggregateRating(): float|int|string |
670
|
|
|
{ |
671
|
|
|
$result = 0; |
672
|
|
|
$total = 0; |
673
|
|
|
if (!empty($this->ratings)) { |
674
|
|
|
foreach ($this->ratings as $row) { |
675
|
|
|
$result += $row['rating']; |
676
|
|
|
++$total; |
677
|
|
|
} |
678
|
|
|
|
679
|
|
|
$result /= $total; |
680
|
|
|
} else { |
681
|
|
|
$result = ''; |
682
|
|
|
} |
683
|
|
|
|
684
|
|
|
return $result; |
685
|
|
|
} |
686
|
|
|
|
687
|
|
|
/** |
688
|
|
|
* Get the total number of ratings |
689
|
|
|
*/ |
|
|
|
|
690
|
|
|
public function getRatingsCount(): int |
691
|
|
|
{ |
692
|
|
|
return count($this->ratings); |
|
|
|
|
693
|
|
|
} |
694
|
|
|
|
695
|
|
|
/** |
696
|
|
|
* Returns concatenated serves with its unit |
697
|
|
|
*/ |
|
|
|
|
698
|
|
|
public function getServes(): int|string |
699
|
|
|
{ |
700
|
|
|
if (!empty($this->servesUnit)) { |
701
|
|
|
return $this->serves . ' ' . $this->servesUnit; |
702
|
|
|
} |
703
|
|
|
|
704
|
|
|
return $this->serves; |
705
|
|
|
} |
706
|
|
|
|
707
|
|
|
/** |
708
|
|
|
* Convert decimal numbers into fractions |
709
|
|
|
* |
710
|
|
|
* @param $quantity |
|
|
|
|
711
|
|
|
*/ |
|
|
|
|
712
|
|
|
private function convertToFractions($quantity): string |
|
|
|
|
713
|
|
|
{ |
714
|
|
|
$whole = floor($quantity); |
715
|
|
|
// Round the mantissa so we can do a floating point comparison without |
716
|
|
|
// weirdness, per: https://www.php.net/manual/en/language.types.float.php#113703 |
717
|
|
|
$fraction = round($quantity - $whole, 3); |
718
|
|
|
switch ($fraction) { |
719
|
|
|
case 0: |
|
|
|
|
720
|
|
|
$fraction = ''; |
721
|
|
|
break; |
722
|
|
|
case 0.25: |
|
|
|
|
723
|
|
|
$fraction = ' ¼'; |
724
|
|
|
break; |
725
|
|
|
case 0.33: |
|
|
|
|
726
|
|
|
$fraction = ' ⅓'; |
727
|
|
|
break; |
728
|
|
|
case 0.66: |
|
|
|
|
729
|
|
|
$fraction = ' ⅔'; |
730
|
|
|
break; |
731
|
|
|
case 0.165: |
|
|
|
|
732
|
|
|
$fraction = ' ⅙'; |
733
|
|
|
break; |
734
|
|
|
case 0.5: |
|
|
|
|
735
|
|
|
$fraction = ' ½'; |
736
|
|
|
break; |
737
|
|
|
case 0.75: |
|
|
|
|
738
|
|
|
$fraction = ' ¾'; |
739
|
|
|
break; |
740
|
|
|
case 0.125: |
|
|
|
|
741
|
|
|
$fraction = ' ⅛'; |
742
|
|
|
break; |
743
|
|
|
case 0.375: |
|
|
|
|
744
|
|
|
$fraction = ' ⅜'; |
745
|
|
|
break; |
746
|
|
|
case 0.625: |
|
|
|
|
747
|
|
|
$fraction = ' ⅝'; |
748
|
|
|
break; |
749
|
|
|
case 0.875: |
|
|
|
|
750
|
|
|
$fraction = ' ⅞'; |
751
|
|
|
break; |
752
|
|
|
default: |
|
|
|
|
753
|
|
|
$precision = 1; |
754
|
|
|
$pnum = round($fraction, $precision); |
755
|
|
|
$denominator = 10 ** $precision; |
756
|
|
|
$numerator = $pnum * $denominator; |
757
|
|
|
$fraction = ' <sup>' |
758
|
|
|
. $numerator |
759
|
|
|
. '</sup>⁄<sub>' |
760
|
|
|
. $denominator |
761
|
|
|
. '</sub>'; |
762
|
|
|
break; |
763
|
|
|
} |
764
|
|
|
|
765
|
|
|
if ($whole == 0) { |
766
|
|
|
$whole = ''; |
767
|
|
|
} |
768
|
|
|
|
769
|
|
|
return $whole . $fraction; |
770
|
|
|
} |
771
|
|
|
|
772
|
|
|
// Private Methods |
773
|
|
|
// ========================================================================= |
774
|
|
|
|
775
|
|
|
/** |
|
|
|
|
776
|
|
|
* Renders a JSON-LD representation of the schema |
777
|
|
|
* |
778
|
|
|
* @param $json |
|
|
|
|
779
|
|
|
*/ |
|
|
|
|
780
|
|
|
private function renderJsonLd($json, bool $raw = true): string|Markup |
|
|
|
|
781
|
|
|
{ |
782
|
|
|
$linebreak = ''; |
783
|
|
|
|
784
|
|
|
// If `devMode` is enabled, make the JSON-LD human-readable |
785
|
|
|
if (Craft::$app->getConfig()->getGeneral()->devMode) { |
786
|
|
|
$linebreak = PHP_EOL; |
787
|
|
|
} |
788
|
|
|
|
789
|
|
|
// Render the resulting JSON-LD |
790
|
|
|
$result = '<script type="application/ld+json">' |
791
|
|
|
. $linebreak |
792
|
|
|
. Json::encode($json) |
793
|
|
|
. $linebreak |
794
|
|
|
. '</script>'; |
795
|
|
|
|
796
|
|
|
if ($raw) { |
797
|
|
|
$result = Template::raw($result); |
798
|
|
|
} |
799
|
|
|
|
800
|
|
|
return $result; |
801
|
|
|
} |
802
|
|
|
} |
803
|
|
|
|