Passed
Pull Request — master (#2)
by Kyle
03:13
created

MenuItem::make()   A

Complexity

Conditions 1
Paths 1

Size

Total Lines 5
Code Lines 2

Duplication

Lines 0
Ratio 0 %

Code Coverage

Tests 3
CRAP Score 1

Importance

Changes 0
Metric Value
cc 1
eloc 2
nc 1
nop 1
dl 0
loc 5
ccs 3
cts 3
cp 1
crap 1
rs 10
c 0
b 0
f 0
1
<?php
2
3
namespace KyleMassacre\Menus;
4
5
use AllowDynamicProperties;
0 ignored issues
show
Bug introduced by
The type AllowDynamicProperties was not found. Maybe you did not declare it correctly or list all dependencies?

The issue could also be caused by a filter entry in the build configuration. If the path has been excluded in your configuration, e.g. excluded_paths: ["lib/*"], you can move it to the dependency path list as follows:

filter:
    dependency_paths: ["lib/*"]

For further information see https://scrutinizer-ci.com/docs/tools/php/php-scrutinizer/#list-dependency-paths

Loading history...
6
use Collective\Html\HtmlFacade as HTML;
7
use Illuminate\Support\Arr;
8
use Illuminate\Support\Facades\Request;
9
use KyleMassacre\Menus\Contracts\MenuItemContract;
10
use KyleMassacre\Menus\Traits\CanHide;
11
12
/**
13
 * @property string url
14
 * @property string route
15
 * @property string title
16
 * @property string name
17
 * @property string icon
18
 * @property int parent
19
 * @property array attributes
20
 * @property bool active
21
 * @property int order
22
 * @property array badge
23
 */
24
#[AllowDynamicProperties] class MenuItem extends MenuItemContract
25
{
26
    use CanHide;
27
28
    /**
29
     * Constructor.
30
     *
31
     * @param array $properties
32
     */
33 26
    public function __construct(array $properties = array())
34
    {
35 26
        $this->properties = $properties;
36 26
        $this->fill($properties);
37
    }
38
39
    /**
40
     * Fill the attributes.
41
     *
42
     * @param array $attributes
43
     */
44 26
    public function fill(array $attributes): void
45
    {
46 26
        foreach ($attributes as $key => $value) {
47 25
            if (in_array($key, $this->fillable)) {
48 25
                $this->{$key} = $value;
49
            }
50
        }
51
    }
52
53
    /**
54
     * Create new menu child item using array.
55
     *
56
     * @param array $attributes
57
     *
58
     * @return self
59
     */
60 2
    public function child(array $attributes): self
61
    {
62 2
        $this->childs[] = static::make($attributes);
63
64 2
        return $this;
65
    }
66
67
    /**
68
     * Register new child menu with dropdown.
69
     *
70
     * @param $title
71
     * @param \Closure $callback
72
     * @param int $order
73
     * @param array $attributes
74
     * @return $this
75
     */
76 9
    public function dropdown(string $title, \Closure $callback, int $order = 0, array $attributes = array()): static
77
    {
78 9
        $properties = compact('title', 'order', 'attributes');
79
80 9
        if (func_num_args() === 3) {
81
            $arguments = func_get_args();
82
83
            $title = Arr::get($arguments, 0);
84
            $attributes = Arr::get($arguments, 2);
85
86
            $properties = compact('title', 'attributes');
87
        }
88
89 9
        $child = static::make($properties);
90
91 9
        call_user_func($callback, $child);
92
93 9
        $this->childs[] = $child;
94
95 9
        return $child;
96
    }
97
98
    /**
99
     * Create new menu item and set the action to route.
100
     *
101
     * @param $route
102
     * @param $title
103
     * @param array $parameters
104
     * @param array $attributes
105
     *
106
     * @return MenuItemContract
107
     */
108 3
    public function route($route, $title, array $parameters = array(), $order = 0, array $attributes = array()): static
109
    {
110 3
        if (func_num_args() === 4) {
111
            $arguments = func_get_args();
112
113
            return $this->add([
114
                'route' => [Arr::get($arguments, 0), Arr::get($arguments, 2)],
115
                'title' => Arr::get($arguments, 1),
116
                'attributes' => Arr::get($arguments, 3),
117
            ]);
118
        }
119
120 3
        $route = array($route, $parameters);
121
122 3
        return $this->add(compact('route', 'title', 'order', 'attributes'));
123
    }
124
125
    /**
126
     * Create new menu item  and set the action to url.
127
     *
128
     * @param $url
129
     * @param $title
130
     * @param array $attributes
131
     *
132
     * @return static
133
     */
134 6
    public function url($url, $title, $order = 0, array $attributes = array()): static
135
    {
136 6
        if (func_num_args() === 3) {
137
            $arguments = func_get_args();
138
139
            return $this->add([
140
                'url' => Arr::get($arguments, 0),
141
                'title' => Arr::get($arguments, 1),
142
                'attributes' => Arr::get($arguments, 2),
143
            ]);
144
        }
145
146 6
        return $this->add(compact('url', 'title', 'order', 'attributes'));
147
    }
148
149
    /**
150
     * Add new divider.
151
     *
152
     * @param int $order
153
     *
154
     * @return self
155
     */
156 1
    public function addDivider($order = null): static
157
    {
158 1
        $item = static::make(array('name' => 'divider', 'order' => $order));
159
160 1
        $this->childs[] = $item;
161
162 1
        return $item;
163
    }
164
165
    /**
166
     * Alias method instead "addDivider".
167
     *
168
     * @param int $order
169
     *
170
     * @return MenuItem
171
     */
172 1
    public function divider($order = null): static
173
    {
174 1
        return $this->addDivider($order);
175
    }
176
177
    /**
178
     * Add dropdown header.
179
     *
180
     * @param $title
181
     *
182
     * @return $this
183
     */
184 2
    public function addHeader($title): static
185
    {
186 2
        $item = static::make(array(
187 2
            'name' => 'header',
188 2
            'title' => $title,
189 2
        ));
190
191 2
        $this->childs[] = $item;
192
193 2
        return $item;
194
    }
195
196
    /**
197
     * Same with "addHeader" method.
198
     *
199
     * @param $title
200
     *
201
     * @return $this
202
     */
203 2
    public function header($title): static
204
    {
205 2
        return $this->addHeader($title);
206
    }
207
208
    public function addBadge(string $type, $text): static
209
    {
210
        $properties = array(
211
            'type' => $type,
212
            'text' => $text,
213
            'name' => 'badge',
214
        );
215
        $item = static::make($properties);
216
        $this->badge = $properties;
217
218
        return $item;
219
    }
220
221
    /**
222
     * @deprecated See `getChildren`
223
     * @return array
224
     */
225 8
    public function getChilds(): array
226
    {
227 8
        return $this->getChildren();
228
    }
229
230
    /**
231
     * Get childs.
232
     *
233
     * @return array
234
     */
235 10
    public function getChildren(): array
236
    {
237 10
        if (config('menus.ordering')) {
238 1
            return collect($this->childs)->sortBy('order')->all();
0 ignored issues
show
Bug introduced by
$this->childs of type array is incompatible with the type Illuminate\Contracts\Support\Arrayable expected by parameter $value of collect(). ( Ignorable by Annotation )

If this is a false-positive, you can also ignore this issue in your code via the ignore-type  annotation

238
            return collect(/** @scrutinizer ignore-type */ $this->childs)->sortBy('order')->all();
Loading history...
239
        }
240
241 9
        return $this->childs;
242
    }
243
244
    /**
245
     * Get url.
246
     *
247
     * @return string
248
     */
249 3
    public function getUrl(): string
250
    {
251 3
        if ($this->route !== null) {
252 1
            return route($this->route[0], $this->route[1]);
253
        }
254
255 2
        if (empty($this->url)) {
256
            return url('/#');
257
        }
258
259 2
        return url($this->url);
260
    }
261
262
    /**
263
     * Get request url.
264
     *
265
     * @return string
266
     */
267 1
    public function getRequest(): string
268
    {
269 1
        return ltrim(str_replace(url('/'), '', $this->getUrl()), '/');
270
    }
271
272
    /**
273
     * @return string
274
     */
275
    public function getBadge(): string
276
    {
277
        if($this->hasBadge()) {
278
            extract($this->badge);
279
280
            return '<span class="' . $type . '">' . $text . '</span>';
281
        }
0 ignored issues
show
Bug Best Practice introduced by
The function implicitly returns null when the if condition on line 277 is false. This is incompatible with the type-hinted return string. Consider adding a return statement or allowing null as return value.

For hinted functions/methods where all return statements with the correct type are only reachable via conditions, ?null? gets implicitly returned which may be incompatible with the hinted type. Let?s take a look at an example:

interface ReturnsInt {
    public function returnsIntHinted(): int;
}

class MyClass implements ReturnsInt {
    public function returnsIntHinted(): int
    {
        if (foo()) {
            return 123;
        }
        // here: null is implicitly returned
    }
}
Loading history...
282
    }
283
284
    /**
285
     * Get icon.
286
     *
287
     * @param string|null $default
288
     *
289
     * @return string|null
290
     */
291 3
    public function getIcon(string $default = null): ?string
292
    {
293 3
        if ($this->icon !== null && $this->icon !== '') {
294 1
            return '<i class="' . $this->icon . '"></i>';
295
        }
296 2
        if ($default === null) {
297 1
            return $default;
298
        }
299
300 1
        return '<i class="' . $default . '"></i>';
301
    }
302
303
    /**
304
     * Get properties.
305
     *
306
     * @return array
307
     */
308 4
    public function getProperties(): array
309
    {
310 4
        return $this->properties;
311
    }
312
313
    /**
314
     * Get HTML attribute data.
315
     *
316
     * @return mixed
317
     */
318 1
    public function getAttributes(): mixed
319
    {
320 1
        $attributes = $this->attributes ?: [];
321
322 1
        Arr::forget($attributes, ['active', 'icon']);
323
324 1
        return HTML::attributes($attributes);
325
    }
326
327
    /**
328
     * Check is the current item divider.
329
     *
330
     * @return bool
331
     */
332 1
    public function isDivider(): bool
333
    {
334 1
        return $this->is('divider');
335
    }
336
337
    /**
338
     * Check is the current item divider.
339
     *
340
     * @return bool
341
     */
342 1
    public function isHeader(): bool
343
    {
344 1
        return $this->is('header');
345
    }
346
347
    /**
348
     * Check is the current item divider.
349
     *
350
     * @param $name
351
     *
352
     * @return bool
353
     */
354 2
    public function is($name): bool
355
    {
356 2
        return $this->name == $name;
357
    }
358
359
    /**
360
     * Check is the current item has sub menu .
361
     *
362
     * @return bool
363
     */
364 1
    public function hasSubMenu(): bool
365
    {
366 1
        return !empty($this->childs);
367
    }
368
369
    /**
370
     * @deprecated Same with hasSubMenu.
371
     *
372
     * @return bool
373
     */
374 1
    public function hasChilds(): bool
375
    {
376 1
        return $this->hasSubMenu();
377
    }
378
379
    /**
380
     * Check the active state for current menu.
381
     *
382
     * @return mixed
383
     */
384
    public function hasActiveOnChild(): mixed
385
    {
386
        if ($this->inactive()) {
387
            return false;
388
        }
389
390
        return $this->hasSubMenu() && $this->getActiveStateFromChildren();
391
    }
392
393
    public function getActiveStateFromChildren(): bool
394
    {
395
        foreach ($this->getChildren() as $child) {
396
            if ($child->inactive()) {
397
                continue;
398
            }
399
            if ($child->hasChilds()) {
400
                if ($child->getActiveStateFromChilds()) {
401
                    return true;
402
                }
403
            } elseif ($child->isActive()) {
404
                return true;
405
            } elseif ($child->hasRoute() && $child->getActiveStateFromRoute()) {
406
                return true;
407
            } elseif ($child->getActiveStateFromUrl()) {
408
                return true;
409
            }
410
        }
411
412
        return false;
413
    }
414
415
    /**
416
     * @deprecated See `getActiveStateFromChildren()`
417
     * Get active state from child menu items.
418
     *
419
     * @return bool
420
     */
421
    public function getActiveStateFromChilds(): bool
422
    {
423
        return $this->getActiveStateFromChildren();
424
    }
425
426
    /**
427
     * Get inactive state.
428
     *
429
     * @return bool
430
     */
431
    public function inactive(): bool
432
    {
433
        $inactive = $this->getInactiveAttribute();
434
435
        if (is_bool($inactive)) {
0 ignored issues
show
introduced by
The condition is_bool($inactive) is always false.
Loading history...
436
            return $inactive;
437
        }
438
439
        if ($inactive instanceof \Closure) {
0 ignored issues
show
introduced by
$inactive is never a sub-type of Closure.
Loading history...
440
            return call_user_func($inactive);
441
        }
442
443
        return false;
444
    }
445
446
    /**
447
     * Get active attribute.
448
     *
449
     * @return string
450
     */
451
    public function getActiveAttribute(): string
452
    {
453
        return Arr::get($this->attributes, 'active');
0 ignored issues
show
Bug Best Practice introduced by
The expression return Illuminate\Suppor...->attributes, 'active') could return the type null which is incompatible with the type-hinted return string. Consider adding an additional type-check to rule them out.
Loading history...
454
    }
455
456
    /**
457
     * Get inactive attribute.
458
     *
459
     * @return string
460
     */
461
    public function getInactiveAttribute(): string
462
    {
463
        return Arr::get($this->attributes, 'inactive');
0 ignored issues
show
Bug Best Practice introduced by
The expression return Illuminate\Suppor...attributes, 'inactive') could return the type null which is incompatible with the type-hinted return string. Consider adding an additional type-check to rule them out.
Loading history...
464
    }
465
466
    /**
467
     * Get active state for current item.
468
     *
469
     * @return mixed
470
     */
471
    public function isActive(): mixed
472
    {
473
        if ($this->inactive()) {
474
            return false;
475
        }
476
477
        $active = $this->getActiveAttribute();
478
479
        if (is_bool($active)) {
0 ignored issues
show
introduced by
The condition is_bool($active) is always false.
Loading history...
480
            return $active;
481
        }
482
483
        if ($active instanceof \Closure) {
0 ignored issues
show
introduced by
$active is never a sub-type of Closure.
Loading history...
484
            return call_user_func($active);
485
        }
486
487
        if ($this->hasRoute()) {
488
            return $this->getActiveStateFromRoute();
489
        }
490
491
        return $this->getActiveStateFromUrl();
492
    }
493
494
    /**
495
     * Determine the current item using route.
496
     *
497
     * @return bool
498
     */
499
    protected function hasRoute(): bool
500
    {
501
        return !empty($this->route);
502
    }
503
504
    /**
505
     * Get active status using route.
506
     *
507
     * @return bool
508
     */
509
    protected function getActiveStateFromRoute(): bool
510
    {
511
        return Request::is(str_replace(url('/') . '/', '', $this->getUrl()));
512
    }
513
514
    /**
515
     * Get active status using request url.
516
     *
517
     * @return bool
518
     */
519
    protected function getActiveStateFromUrl(): bool
520
    {
521
        return Request::is($this->url);
522
    }
523
524
    /**
525
     * Set order value.
526
     *
527
     * @param  int $order
528
     * @return self
529
     */
530
    public function order(int $order): self
531
    {
532
        $this->order = $order;
533
534
        return $this;
535
    }
536
537
    public function hasBadge(): bool
538
    {
539
        return !empty($this->badge);
540
    }
541
}
542