Completed
Pull Request — master (#15)
by James
04:04
created
src/Axolotl/Model.php 1 patch
Indentation   +730 added lines, -730 removed lines patch added patch discarded remove patch
@@ -21,734 +21,734 @@
 block discarded – undo
21 21
  * @since      0.1.0
22 22
  */
23 23
 abstract class Model implements Serializes {
24
-	/**
25
-	 * Table attribute key.
26
-	 */
27
-	const TABLE_KEY = '@@table';
28
-
29
-	/**
30
-	 * Object attribute key.
31
-	 */
32
-	const OBJECT_KEY = '@@object';
33
-
34
-	/**
35
-	 * Memoized values for class methods.
36
-	 *
37
-	 * @var array
38
-	 */
39
-	private static $memo = array();
40
-
41
-	/**
42
-	 * Model attributes.
43
-	 *
44
-	 * @var array
45
-	 */
46
-	private $attributes = array(
47
-		self::TABLE_KEY  => array(),
48
-		self::OBJECT_KEY => null,
49
-	);
50
-
51
-	/**
52
-	 * Model's original attributes.
53
-	 *
54
-	 * @var array
55
-	 */
56
-	private $original = array(
57
-		self::TABLE_KEY  => array(),
58
-		self::OBJECT_KEY => null,
59
-	);
60
-
61
-	/**
62
-	 * Default attribute values.
63
-	 *
64
-	 * @var array
65
-	 */
66
-	protected $defaults = array();
67
-
68
-	/**
69
-	 * Properties which are allowed to be set on the model.
70
-	 *
71
-	 * If this array is empty, any attributes can be set on the model.
72
-	 *
73
-	 * @var string[]
74
-	 */
75
-	protected $fillable = array();
76
-
77
-	/**
78
-	 * Properties which cannot be automatically filled on the model.
79
-	 *
80
-	 * If the model is unguarded, these properties can be filled.
81
-	 *
82
-	 * @var array
83
-	 */
84
-	protected $guarded = array();
85
-
86
-	/**
87
-	 * Properties which should not be serialized.
88
-	 *
89
-	 * @var array
90
-	 */
91
-	protected $hidden = array();
92
-
93
-	/**
94
-	 * Properties which should be serialized.
95
-	 *
96
-	 * @var array
97
-	 */
98
-	protected $visible = array();
99
-
100
-	/**
101
-	 * Whether the model's properties are guarded.
102
-	 *
103
-	 * When false, allows guarded properties to be filled.
104
-	 *
105
-	 * @var bool
106
-	 */
107
-	protected $is_guarded = true;
108
-
109
-	/**
110
-	 * Constructs a new model with provided attributes.
111
-	 *
112
-	 * If self::OBJECT_KEY is passed as one of the attributes, the underlying post
113
-	 * will be overwritten.
114
-	 *
115
-	 * @param array <string, mixed> $attributes
116
-	 */
117
-	public function __construct( array $attributes = array() ) {
118
-		$this->maybe_boot();
119
-		$this->sync_original();
120
-
121
-		if ( $this->uses_wp_object() ) {
122
-			$this->create_wp_object();
123
-		}
124
-
125
-		$this->unguard();
126
-		$this->refresh( $attributes );
127
-		$this->reguard();
128
-	}
129
-
130
-	/**
131
-	 * Refreshes the model's current attributes with the provided array.
132
-	 *
133
-	 * The model's attributes will match what was provided in the array,
134
-	 * and any attributes not passed
135
-	 *
136
-	 * @param array $attributes
137
-	 *
138
-	 * @return $this
139
-	 */
140
-	public function refresh( array $attributes ) {
141
-		$this->clear();
142
-
143
-		return $this->merge( $attributes );
144
-	}
145
-
146
-	/**
147
-	 * Merges the provided attributes with the provided array.
148
-	 *
149
-	 * @param array $attributes
150
-	 *
151
-	 * @return $this
152
-	 */
153
-	public function merge( array $attributes ) {
154
-		foreach ( $attributes as $name => $value ) {
155
-			$this->set_attribute( $name, $value );
156
-		}
157
-
158
-		return $this;
159
-	}
160
-
161
-	/**
162
-	 * Get the model's table attributes.
163
-	 *
164
-	 * Returns the array of for the model that will either need to be
165
-	 * saved in postmeta or a separate table.
166
-	 *
167
-	 * @return array
168
-	 */
169
-	public function get_table_attributes() {
170
-		return $this->attributes[ self::TABLE_KEY ];
171
-	}
172
-
173
-	/**
174
-	 * Get the model's original attributes.
175
-	 *
176
-	 * @return array
177
-	 */
178
-	public function get_original_table_attributes() {
179
-		return $this->original[ self::TABLE_KEY ];
180
-	}
181
-
182
-	/**
183
-	 * Retrieve an array of the attributes on the model
184
-	 * that have changed compared to the model's
185
-	 * original data.
186
-	 *
187
-	 * @return array
188
-	 */
189
-	public function get_changed_table_attributes() {
190
-		$changed = array();
191
-
192
-		foreach ( $this->get_table_attributes() as $key => $value ) {
193
-			if ( $value !==
194
-				 $this->get_original_attribute( $key )
195
-			) {
196
-				$changed[ $key ] = $value;
197
-			}
198
-		}
199
-
200
-		return $changed;
201
-	}
202
-
203
-	/**
204
-	 * Get the model's underlying post.
205
-	 *
206
-	 * Returns the underlying WP_Post object for the model, representing
207
-	 * the data that will be save in the wp_posts table.
208
-	 *
209
-	 * @return false|WP_Post|WP_Term
210
-	 */
211
-	public function get_underlying_wp_object() {
212
-		if ( isset( $this->attributes[ self::OBJECT_KEY ] ) ) {
213
-			return $this->attributes[ self::OBJECT_KEY ];
214
-		}
215
-
216
-		return false;
217
-	}
218
-
219
-	/**
220
-	 * Get the model's original underlying post.
221
-	 *
222
-	 * @return WP_Post
223
-	 */
224
-	public function get_original_underlying_wp_object() {
225
-		return $this->original[ self::OBJECT_KEY ];
226
-	}
227
-
228
-	/**
229
-	 * Get the model attributes on the WordPress object
230
-	 * that have changed compared to the model's
231
-	 * original attributes.
232
-	 *
233
-	 * @return array
234
-	 */
235
-	public function get_changed_wp_object_attributes() {
236
-		$changed = array();
237
-
238
-		foreach ( $this->get_wp_object_keys() as $key ) {
239
-			if ( $this->get_attribute( $key ) !==
240
-				 $this->get_original_attribute( $key )
241
-			) {
242
-				$changed[ $key ] = $this->get_attribute( $key );
243
-			}
244
-		}
245
-
246
-		return $changed;
247
-	}
248
-
249
-	/**
250
-	 * Magic __set method.
251
-	 *
252
-	 * Passes the name and value to set_attribute, which is where the magic happens.
253
-	 *
254
-	 * @param string $name
255
-	 * @param mixed  $value
256
-	 */
257
-	public function __set( $name, $value ) {
258
-		$this->set_attribute( $name, $value );
259
-	}
260
-
261
-	/**
262
-	 * Sets the model attributes.
263
-	 *
264
-	 * Checks whether the model attribute can be set, check if it
265
-	 * maps to the WP_Post property, otherwise, assigns it to the
266
-	 * table attribute array.
267
-	 *
268
-	 * @param string $name
269
-	 * @param mixed  $value
270
-	 *
271
-	 * @return $this
272
-	 *
273
-	 * @throws GuardedPropertyException
274
-	 */
275
-	public function set_attribute( $name, $value ) {
276
-		if ( self::OBJECT_KEY === $name ) {
277
-			return $this->override_wp_object( $value );
278
-		}
279
-
280
-		if ( self::TABLE_KEY === $name ) {
281
-			return $this->override_table( $value );
282
-		}
283
-
284
-		if ( ! $this->is_fillable( $name ) ) {
285
-			throw new GuardedPropertyException;
286
-		}
287
-
288
-		if ( $method = $this->has_map_method( $name ) ) {
289
-			$this->attributes[ self::OBJECT_KEY ]->{$this->{$method}()} = $value;
290
-		} else {
291
-			$this->attributes[ self::TABLE_KEY ][ $name ] = $value;
292
-		}
293
-
294
-		return $this;
295
-	}
296
-
297
-	/**
298
-	 * Retrieves all the attribute keys for the model.
299
-	 *
300
-	 * @return array
301
-	 */
302
-	public function get_attribute_keys() {
303
-		if ( isset( self::$memo[ get_called_class() ][ __METHOD__ ] ) ) {
304
-			return self::$memo[ get_called_class() ][ __METHOD__ ];
305
-		}
306
-
307
-		return self::$memo[ get_called_class() ][ __METHOD__ ]
308
-			= array_merge(
309
-				$this->fillable,
310
-				$this->guarded,
311
-				$this->get_compute_methods()
312
-			);
313
-	}
314
-
315
-	/**
316
-	 * Retrieves the attribute keys that aren't mapped to a post.
317
-	 *
318
-	 * @return array
319
-	 */
320
-	public function get_table_keys() {
321
-		if ( isset( self::$memo[ get_called_class() ][ __METHOD__ ] ) ) {
322
-			return self::$memo[ get_called_class() ][ __METHOD__ ];
323
-		}
324
-
325
-		$keys = array();
326
-
327
-		foreach ( $this->get_attribute_keys() as $key ) {
328
-			if ( ! $this->has_map_method( $key ) &&
329
-				 ! $this->has_compute_method( $key )
330
-			) {
331
-				$keys[] = $key;
332
-			}
333
-		}
334
-
335
-		return self::$memo[ get_called_class() ][ __METHOD__ ] = $keys;
336
-	}
337
-
338
-	/**
339
-	 * Retrieves the attribute keys that are mapped to a post.
340
-	 *
341
-	 * @return array
342
-	 */
343
-	public function get_wp_object_keys() {
344
-		if ( isset( self::$memo[ get_called_class() ][ __METHOD__ ] ) ) {
345
-			return self::$memo[ get_called_class() ][ __METHOD__ ];
346
-		}
347
-
348
-		$keys = array();
349
-
350
-		foreach ( $this->get_attribute_keys() as $key ) {
351
-			if ( $this->has_map_method( $key ) ) {
352
-				$keys[] = $key;
353
-			}
354
-		}
355
-
356
-		return self::$memo[ get_called_class() ][ __METHOD__ ] = $keys;
357
-	}
358
-
359
-	/**
360
-	 * Returns the model's keys that are computed at call time.
361
-	 *
362
-	 * @return array
363
-	 */
364
-	public function get_computed_keys() {
365
-		if ( isset( self::$memo[ get_called_class() ][ __METHOD__ ] ) ) {
366
-			return self::$memo[ get_called_class() ][ __METHOD__ ];
367
-		}
368
-
369
-		$keys = array();
370
-
371
-		foreach ( $this->get_attribute_keys() as $key ) {
372
-			if ( $this->has_compute_method( $key ) ) {
373
-				$keys[] = $key;
374
-			}
375
-		}
376
-
377
-		return self::$memo[ get_called_class() ][ __METHOD__ ] = $keys;
378
-	}
379
-
380
-	/**
381
-	 * Serializes the model's public data into an array.
382
-	 *
383
-	 * @return array
384
-	 */
385
-	public function serialize() {
386
-		$attributes = array();
387
-
388
-		if ( $this->visible ) {
389
-			// If visible attributes are set, we'll only reveal those.
390
-			foreach ( $this->visible as $key ) {
391
-				$attributes[ $key ] = $this->get_attribute( $key );
392
-			}
393
-		} elseif ( $this->hidden ) {
394
-			// If hidden attributes are set, we'll grab everything and hide those.
395
-			foreach ( $this->get_attribute_keys() as $key ) {
396
-				if ( ! in_array( $key, $this->hidden ) ) {
397
-					$attributes[ $key ] = $this->get_attribute( $key );
398
-				}
399
-			}
400
-		} else {
401
-			// If nothing is hidden/visible, we'll grab and reveal everything.
402
-			foreach ( $this->get_attribute_keys() as $key ) {
403
-				$attributes[ $key ] = $this->get_attribute( $key );
404
-			}
405
-		}
406
-
407
-		return array_map( function ( $attribute ) {
408
-			if ( $attribute instanceof Serializes ) {
409
-				return $attribute->serialize();
410
-			}
411
-
412
-			return $attribute;
413
-		}, $attributes );
414
-	}
415
-
416
-	/**
417
-	 * Syncs the current attributes to the model's original.
418
-	 *
419
-	 * @return $this
420
-	 */
421
-	public function sync_original() {
422
-		$this->original = $this->attributes;
423
-
424
-		if ( $this->attributes[ self::OBJECT_KEY ] ) {
425
-			$this->original[ self::OBJECT_KEY ] = clone $this->attributes[ self::OBJECT_KEY ];
426
-		}
427
-
428
-		foreach ( $this->original[ self::TABLE_KEY ] as $key => $item ) {
429
-			if ( is_object( $item ) ) {
430
-				$this->original[ $key ] = clone $item;
431
-			}
432
-		}
433
-
434
-		return $this;
435
-	}
436
-
437
-	/**
438
-	 * Checks if a given attribute is mass-fillable.
439
-	 *
440
-	 * Returns true if the attribute can be filled, false if it can't.
441
-	 *
442
-	 * @param string $name
443
-	 *
444
-	 * @return bool
445
-	 */
446
-	private function is_fillable( $name ) {
447
-		// If this model isn't guarded, everything is fillable.
448
-		if ( ! $this->is_guarded ) {
449
-			return true;
450
-		}
451
-
452
-		// If it's in the fillable array, then it's fillable.
453
-		if ( in_array( $name, $this->fillable ) ) {
454
-			return true;
455
-		}
456
-
457
-		// If it's explicitly guarded, then it's not fillable.
458
-		if ( in_array( $name, $this->guarded ) ) {
459
-			return false;
460
-		}
461
-
462
-		// If fillable hasn't been defined, then everything else fillable.
463
-		return ! $this->fillable;
464
-	}
465
-
466
-	/**
467
-	 * Overrides the current WordPress object with a provided one.
468
-	 *
469
-	 * Resets the post's default values and stores it in the attributes.
470
-	 *
471
-	 * @param WP_Post|WP_Term|null $value
472
-	 *
473
-	 * @return $this
474
-	 */
475
-	private function override_wp_object( $value ) {
476
-		if ( is_object( $value ) ) {
477
-			$this->attributes[ self::OBJECT_KEY ] = $this->set_wp_object_constants( $value );
478
-		} else {
479
-			$this->attributes[ self::OBJECT_KEY ] = null;
480
-
481
-			if ( $this->uses_wp_object() ) {
482
-				$this->create_wp_object();
483
-			}
484
-		}
485
-
486
-		return $this;
487
-	}
488
-
489
-	/**
490
-	 * Overrides the current table attributes array with a provided one.
491
-	 *
492
-	 * @param array $value
493
-	 *
494
-	 * @return $this
495
-	 */
496
-	private function override_table( array $value ) {
497
-		$this->attributes[ self::TABLE_KEY ] = $value;
498
-
499
-		return $this;
500
-	}
501
-
502
-	/**
503
-	 * Create and set with a new blank post.
504
-	 *
505
-	 * Creates a new WP_Post object, assigns it the default attributes,
506
-	 * and stores it in the attributes.
507
-	 *
508
-	 * @throws LogicException
509
-	 */
510
-	private function create_wp_object() {
511
-		switch ( true ) {
512
-			case $this instanceof UsesWordPressPost:
513
-				$object = new WP_Post( (object) array() );
514
-				break;
515
-			case $this instanceof UsesWordPressTerm:
516
-				$object = new WP_Term( (object) array() );
517
-				break;
518
-			default:
519
-				throw new LogicException;
520
-				break;
521
-		}
522
-
523
-		$this->attributes[ self::OBJECT_KEY ] = $this->set_wp_object_constants( $object );
524
-	}
525
-
526
-	/**
527
-	 * Enforces values on the post that can't change.
528
-	 *
529
-	 * Primarily, this is used to make sure the post_type always maps
530
-	 * to the model's "$type" property, but this can all be overridden
531
-	 * by the developer to enforce other values in the model.
532
-	 *
533
-	 * @param object $object
534
-	 *
535
-	 * @return object
536
-	 */
537
-	protected function set_wp_object_constants( $object ) {
538
-		if ( $this instanceof UsesWordPressPost ) {
539
-			$object->post_type = static::get_post_type();
540
-		}
541
-
542
-		if ( $this instanceof UsesWordPressTerm ) {
543
-			$object->taxonomy = static::get_taxonomy();
544
-		}
545
-
546
-		return $object;
547
-	}
548
-
549
-	/**
550
-	 * Magic __get method.
551
-	 *
552
-	 * Passes the name and value to get_attribute, which is where the magic happens.
553
-	 *
554
-	 * @param string $name
555
-	 *
556
-	 * @return mixed
557
-	 */
558
-	public function __get( $name ) {
559
-		return $this->get_attribute( $name );
560
-	}
561
-
562
-	/**
563
-	 * Retrieves the model attribute.
564
-	 *
565
-	 * @param string $name
566
-	 *
567
-	 * @return mixed
568
-	 *
569
-	 * @throws PropertyDoesNotExistException If property isn't found.
570
-	 */
571
-	public function get_attribute( $name ) {
572
-		if ( $method = $this->has_map_method( $name ) ) {
573
-			return $this->attributes[ self::OBJECT_KEY ]->{$this->{$method}()};
574
-		}
575
-
576
-		if ( $method = $this->has_compute_method( $name ) ) {
577
-			return $this->{$method}();
578
-		}
579
-
580
-		if ( isset( $this->attributes[ self::TABLE_KEY ][ $name ] ) ) {
581
-			return $this->attributes[ self::TABLE_KEY ][ $name ];
582
-		}
583
-
584
-		if ( isset( $this->defaults[ $name ] ) ) {
585
-			return $this->defaults[ $name ];
586
-		}
587
-
588
-		throw new PropertyDoesNotExistException( $name );
589
-	}
590
-
591
-	/**
592
-	 * Retrieve the model's original attribute value.
593
-	 *
594
-	 * @param string $name
595
-	 *
596
-	 * @return mixed
597
-	 *
598
-	 * @throws PropertyDoesNotExistException If property isn't found.
599
-	 */
600
-	public function get_original_attribute( $name ) {
601
-		$original_attributes = $this->original;
602
-
603
-		if ( ! is_object( $original_attributes[ static::OBJECT_KEY ] ) ) {
604
-			unset( $original_attributes[ static::OBJECT_KEY ] );
605
-		}
606
-
607
-		$original = new static( $original_attributes );
608
-
609
-		try {
610
-			return $original->get_attribute( $name );
611
-		} catch ( Exception $exception ) {
612
-			return null;
613
-		}
614
-	}
615
-
616
-	/**
617
-	 * Fetches the Model's primary ID, depending on the model
618
-	 * implementation.
619
-	 *
620
-	 * @return int
621
-	 *
622
-	 * @throws LogicException
623
-	 */
624
-	public function get_primary_id() {
625
-		if ( $this instanceof UsesWordPressPost ) {
626
-			return $this->get_underlying_wp_object()->ID;
627
-		}
628
-
629
-		if ( $this instanceof UsesWordPressTerm ) {
630
-			return $this->get_underlying_wp_object()->term_id;
631
-		}
632
-
633
-		// Model w/o wp_object not yet supported.
634
-		throw new LogicException;
635
-	}
636
-
637
-	/**
638
-	 * Checks whether the attribute has a map method.
639
-	 *
640
-	 * This is used to determine whether the attribute maps to a
641
-	 * property on the underlying WP_Post object. Returns the
642
-	 * method if one exists, returns false if it doesn't.
643
-	 *
644
-	 * @param string $name
645
-	 *
646
-	 * @return false|string
647
-	 */
648
-	protected function has_map_method( $name ) {
649
-		if ( method_exists( $this, $method = "map_{$name}" ) ) {
650
-			return $method;
651
-		}
652
-
653
-		return false;
654
-	}
655
-
656
-	/**
657
-	 * Checks whether the attribute has a compute method.
658
-	 *
659
-	 * This is used to determine if the attribute should be computed
660
-	 * from other attributes.
661
-	 *
662
-	 * @param string $name
663
-	 *
664
-	 * @return false|string
665
-	 */
666
-	protected function has_compute_method( $name ) {
667
-		if ( method_exists( $this, $method = "compute_{$name}" ) ) {
668
-			return $method;
669
-		}
670
-
671
-		return false;
672
-	}
673
-
674
-	/**
675
-	 * Clears all the current attributes from the model.
676
-	 *
677
-	 * This does not touch the model's original attributes, and will
678
-	 * only clear fillable attributes, unless the model is unguarded.
679
-	 *
680
-	 * @return $this
681
-	 */
682
-	public function clear() {
683
-		$keys = array_merge(
684
-			$this->get_table_keys(),
685
-			$this->get_wp_object_keys()
686
-		);
687
-
688
-		foreach ( $keys as $key ) {
689
-			try {
690
-				$this->set_attribute( $key, null );
691
-			} catch ( GuardedPropertyException $e ) {
692
-				// We won't clear out guarded attributes.
693
-			}
694
-		}
695
-
696
-		return $this;
697
-	}
698
-
699
-	/**
700
-	 * Unguards the model.
701
-	 *
702
-	 * Sets the model to be unguarded, allowing the filling of
703
-	 * guarded attributes.
704
-	 */
705
-	public function unguard() {
706
-		$this->is_guarded = false;
707
-	}
708
-
709
-	/**
710
-	 * Reguards the model.
711
-	 *
712
-	 * Sets the model to be guarded, preventing filling of
713
-	 * guarded attributes.
714
-	 */
715
-	public function reguard() {
716
-		$this->is_guarded = true;
717
-	}
718
-
719
-	/**
720
-	 * Retrieves all the compute methods on the model.
721
-	 *
722
-	 * @return array
723
-	 */
724
-	protected function get_compute_methods() {
725
-		$methods = get_class_methods( get_called_class() );
726
-		$methods = array_filter( $methods, function ( $method ) {
727
-			return strrpos( $method, 'compute_', - strlen( $method ) ) !== false;
728
-		} );
729
-		$methods = array_map( function ( $method ) {
730
-			return substr( $method, strlen( 'compute_' ) );
731
-		}, $methods );
732
-
733
-		return $methods;
734
-	}
735
-
736
-	/**
737
-	 * Sets up the memo array for the creating model.
738
-	 */
739
-	private function maybe_boot() {
740
-		if ( ! isset( self::$memo[ get_called_class() ] ) ) {
741
-			self::$memo[ get_called_class() ] = array();
742
-		}
743
-	}
744
-
745
-	/**
746
-	 * Whether this Model uses an underlying WordPress object.
747
-	 *
748
-	 * @return bool
749
-	 */
750
-	protected function uses_wp_object() {
751
-		return $this instanceof UsesWordPressPost ||
752
-			$this instanceof UsesWordPressTerm;
753
-	}
24
+    /**
25
+     * Table attribute key.
26
+     */
27
+    const TABLE_KEY = '@@table';
28
+
29
+    /**
30
+     * Object attribute key.
31
+     */
32
+    const OBJECT_KEY = '@@object';
33
+
34
+    /**
35
+     * Memoized values for class methods.
36
+     *
37
+     * @var array
38
+     */
39
+    private static $memo = array();
40
+
41
+    /**
42
+     * Model attributes.
43
+     *
44
+     * @var array
45
+     */
46
+    private $attributes = array(
47
+        self::TABLE_KEY  => array(),
48
+        self::OBJECT_KEY => null,
49
+    );
50
+
51
+    /**
52
+     * Model's original attributes.
53
+     *
54
+     * @var array
55
+     */
56
+    private $original = array(
57
+        self::TABLE_KEY  => array(),
58
+        self::OBJECT_KEY => null,
59
+    );
60
+
61
+    /**
62
+     * Default attribute values.
63
+     *
64
+     * @var array
65
+     */
66
+    protected $defaults = array();
67
+
68
+    /**
69
+     * Properties which are allowed to be set on the model.
70
+     *
71
+     * If this array is empty, any attributes can be set on the model.
72
+     *
73
+     * @var string[]
74
+     */
75
+    protected $fillable = array();
76
+
77
+    /**
78
+     * Properties which cannot be automatically filled on the model.
79
+     *
80
+     * If the model is unguarded, these properties can be filled.
81
+     *
82
+     * @var array
83
+     */
84
+    protected $guarded = array();
85
+
86
+    /**
87
+     * Properties which should not be serialized.
88
+     *
89
+     * @var array
90
+     */
91
+    protected $hidden = array();
92
+
93
+    /**
94
+     * Properties which should be serialized.
95
+     *
96
+     * @var array
97
+     */
98
+    protected $visible = array();
99
+
100
+    /**
101
+     * Whether the model's properties are guarded.
102
+     *
103
+     * When false, allows guarded properties to be filled.
104
+     *
105
+     * @var bool
106
+     */
107
+    protected $is_guarded = true;
108
+
109
+    /**
110
+     * Constructs a new model with provided attributes.
111
+     *
112
+     * If self::OBJECT_KEY is passed as one of the attributes, the underlying post
113
+     * will be overwritten.
114
+     *
115
+     * @param array <string, mixed> $attributes
116
+     */
117
+    public function __construct( array $attributes = array() ) {
118
+        $this->maybe_boot();
119
+        $this->sync_original();
120
+
121
+        if ( $this->uses_wp_object() ) {
122
+            $this->create_wp_object();
123
+        }
124
+
125
+        $this->unguard();
126
+        $this->refresh( $attributes );
127
+        $this->reguard();
128
+    }
129
+
130
+    /**
131
+     * Refreshes the model's current attributes with the provided array.
132
+     *
133
+     * The model's attributes will match what was provided in the array,
134
+     * and any attributes not passed
135
+     *
136
+     * @param array $attributes
137
+     *
138
+     * @return $this
139
+     */
140
+    public function refresh( array $attributes ) {
141
+        $this->clear();
142
+
143
+        return $this->merge( $attributes );
144
+    }
145
+
146
+    /**
147
+     * Merges the provided attributes with the provided array.
148
+     *
149
+     * @param array $attributes
150
+     *
151
+     * @return $this
152
+     */
153
+    public function merge( array $attributes ) {
154
+        foreach ( $attributes as $name => $value ) {
155
+            $this->set_attribute( $name, $value );
156
+        }
157
+
158
+        return $this;
159
+    }
160
+
161
+    /**
162
+     * Get the model's table attributes.
163
+     *
164
+     * Returns the array of for the model that will either need to be
165
+     * saved in postmeta or a separate table.
166
+     *
167
+     * @return array
168
+     */
169
+    public function get_table_attributes() {
170
+        return $this->attributes[ self::TABLE_KEY ];
171
+    }
172
+
173
+    /**
174
+     * Get the model's original attributes.
175
+     *
176
+     * @return array
177
+     */
178
+    public function get_original_table_attributes() {
179
+        return $this->original[ self::TABLE_KEY ];
180
+    }
181
+
182
+    /**
183
+     * Retrieve an array of the attributes on the model
184
+     * that have changed compared to the model's
185
+     * original data.
186
+     *
187
+     * @return array
188
+     */
189
+    public function get_changed_table_attributes() {
190
+        $changed = array();
191
+
192
+        foreach ( $this->get_table_attributes() as $key => $value ) {
193
+            if ( $value !==
194
+                    $this->get_original_attribute( $key )
195
+            ) {
196
+                $changed[ $key ] = $value;
197
+            }
198
+        }
199
+
200
+        return $changed;
201
+    }
202
+
203
+    /**
204
+     * Get the model's underlying post.
205
+     *
206
+     * Returns the underlying WP_Post object for the model, representing
207
+     * the data that will be save in the wp_posts table.
208
+     *
209
+     * @return false|WP_Post|WP_Term
210
+     */
211
+    public function get_underlying_wp_object() {
212
+        if ( isset( $this->attributes[ self::OBJECT_KEY ] ) ) {
213
+            return $this->attributes[ self::OBJECT_KEY ];
214
+        }
215
+
216
+        return false;
217
+    }
218
+
219
+    /**
220
+     * Get the model's original underlying post.
221
+     *
222
+     * @return WP_Post
223
+     */
224
+    public function get_original_underlying_wp_object() {
225
+        return $this->original[ self::OBJECT_KEY ];
226
+    }
227
+
228
+    /**
229
+     * Get the model attributes on the WordPress object
230
+     * that have changed compared to the model's
231
+     * original attributes.
232
+     *
233
+     * @return array
234
+     */
235
+    public function get_changed_wp_object_attributes() {
236
+        $changed = array();
237
+
238
+        foreach ( $this->get_wp_object_keys() as $key ) {
239
+            if ( $this->get_attribute( $key ) !==
240
+                    $this->get_original_attribute( $key )
241
+            ) {
242
+                $changed[ $key ] = $this->get_attribute( $key );
243
+            }
244
+        }
245
+
246
+        return $changed;
247
+    }
248
+
249
+    /**
250
+     * Magic __set method.
251
+     *
252
+     * Passes the name and value to set_attribute, which is where the magic happens.
253
+     *
254
+     * @param string $name
255
+     * @param mixed  $value
256
+     */
257
+    public function __set( $name, $value ) {
258
+        $this->set_attribute( $name, $value );
259
+    }
260
+
261
+    /**
262
+     * Sets the model attributes.
263
+     *
264
+     * Checks whether the model attribute can be set, check if it
265
+     * maps to the WP_Post property, otherwise, assigns it to the
266
+     * table attribute array.
267
+     *
268
+     * @param string $name
269
+     * @param mixed  $value
270
+     *
271
+     * @return $this
272
+     *
273
+     * @throws GuardedPropertyException
274
+     */
275
+    public function set_attribute( $name, $value ) {
276
+        if ( self::OBJECT_KEY === $name ) {
277
+            return $this->override_wp_object( $value );
278
+        }
279
+
280
+        if ( self::TABLE_KEY === $name ) {
281
+            return $this->override_table( $value );
282
+        }
283
+
284
+        if ( ! $this->is_fillable( $name ) ) {
285
+            throw new GuardedPropertyException;
286
+        }
287
+
288
+        if ( $method = $this->has_map_method( $name ) ) {
289
+            $this->attributes[ self::OBJECT_KEY ]->{$this->{$method}()} = $value;
290
+        } else {
291
+            $this->attributes[ self::TABLE_KEY ][ $name ] = $value;
292
+        }
293
+
294
+        return $this;
295
+    }
296
+
297
+    /**
298
+     * Retrieves all the attribute keys for the model.
299
+     *
300
+     * @return array
301
+     */
302
+    public function get_attribute_keys() {
303
+        if ( isset( self::$memo[ get_called_class() ][ __METHOD__ ] ) ) {
304
+            return self::$memo[ get_called_class() ][ __METHOD__ ];
305
+        }
306
+
307
+        return self::$memo[ get_called_class() ][ __METHOD__ ]
308
+            = array_merge(
309
+                $this->fillable,
310
+                $this->guarded,
311
+                $this->get_compute_methods()
312
+            );
313
+    }
314
+
315
+    /**
316
+     * Retrieves the attribute keys that aren't mapped to a post.
317
+     *
318
+     * @return array
319
+     */
320
+    public function get_table_keys() {
321
+        if ( isset( self::$memo[ get_called_class() ][ __METHOD__ ] ) ) {
322
+            return self::$memo[ get_called_class() ][ __METHOD__ ];
323
+        }
324
+
325
+        $keys = array();
326
+
327
+        foreach ( $this->get_attribute_keys() as $key ) {
328
+            if ( ! $this->has_map_method( $key ) &&
329
+                 ! $this->has_compute_method( $key )
330
+            ) {
331
+                $keys[] = $key;
332
+            }
333
+        }
334
+
335
+        return self::$memo[ get_called_class() ][ __METHOD__ ] = $keys;
336
+    }
337
+
338
+    /**
339
+     * Retrieves the attribute keys that are mapped to a post.
340
+     *
341
+     * @return array
342
+     */
343
+    public function get_wp_object_keys() {
344
+        if ( isset( self::$memo[ get_called_class() ][ __METHOD__ ] ) ) {
345
+            return self::$memo[ get_called_class() ][ __METHOD__ ];
346
+        }
347
+
348
+        $keys = array();
349
+
350
+        foreach ( $this->get_attribute_keys() as $key ) {
351
+            if ( $this->has_map_method( $key ) ) {
352
+                $keys[] = $key;
353
+            }
354
+        }
355
+
356
+        return self::$memo[ get_called_class() ][ __METHOD__ ] = $keys;
357
+    }
358
+
359
+    /**
360
+     * Returns the model's keys that are computed at call time.
361
+     *
362
+     * @return array
363
+     */
364
+    public function get_computed_keys() {
365
+        if ( isset( self::$memo[ get_called_class() ][ __METHOD__ ] ) ) {
366
+            return self::$memo[ get_called_class() ][ __METHOD__ ];
367
+        }
368
+
369
+        $keys = array();
370
+
371
+        foreach ( $this->get_attribute_keys() as $key ) {
372
+            if ( $this->has_compute_method( $key ) ) {
373
+                $keys[] = $key;
374
+            }
375
+        }
376
+
377
+        return self::$memo[ get_called_class() ][ __METHOD__ ] = $keys;
378
+    }
379
+
380
+    /**
381
+     * Serializes the model's public data into an array.
382
+     *
383
+     * @return array
384
+     */
385
+    public function serialize() {
386
+        $attributes = array();
387
+
388
+        if ( $this->visible ) {
389
+            // If visible attributes are set, we'll only reveal those.
390
+            foreach ( $this->visible as $key ) {
391
+                $attributes[ $key ] = $this->get_attribute( $key );
392
+            }
393
+        } elseif ( $this->hidden ) {
394
+            // If hidden attributes are set, we'll grab everything and hide those.
395
+            foreach ( $this->get_attribute_keys() as $key ) {
396
+                if ( ! in_array( $key, $this->hidden ) ) {
397
+                    $attributes[ $key ] = $this->get_attribute( $key );
398
+                }
399
+            }
400
+        } else {
401
+            // If nothing is hidden/visible, we'll grab and reveal everything.
402
+            foreach ( $this->get_attribute_keys() as $key ) {
403
+                $attributes[ $key ] = $this->get_attribute( $key );
404
+            }
405
+        }
406
+
407
+        return array_map( function ( $attribute ) {
408
+            if ( $attribute instanceof Serializes ) {
409
+                return $attribute->serialize();
410
+            }
411
+
412
+            return $attribute;
413
+        }, $attributes );
414
+    }
415
+
416
+    /**
417
+     * Syncs the current attributes to the model's original.
418
+     *
419
+     * @return $this
420
+     */
421
+    public function sync_original() {
422
+        $this->original = $this->attributes;
423
+
424
+        if ( $this->attributes[ self::OBJECT_KEY ] ) {
425
+            $this->original[ self::OBJECT_KEY ] = clone $this->attributes[ self::OBJECT_KEY ];
426
+        }
427
+
428
+        foreach ( $this->original[ self::TABLE_KEY ] as $key => $item ) {
429
+            if ( is_object( $item ) ) {
430
+                $this->original[ $key ] = clone $item;
431
+            }
432
+        }
433
+
434
+        return $this;
435
+    }
436
+
437
+    /**
438
+     * Checks if a given attribute is mass-fillable.
439
+     *
440
+     * Returns true if the attribute can be filled, false if it can't.
441
+     *
442
+     * @param string $name
443
+     *
444
+     * @return bool
445
+     */
446
+    private function is_fillable( $name ) {
447
+        // If this model isn't guarded, everything is fillable.
448
+        if ( ! $this->is_guarded ) {
449
+            return true;
450
+        }
451
+
452
+        // If it's in the fillable array, then it's fillable.
453
+        if ( in_array( $name, $this->fillable ) ) {
454
+            return true;
455
+        }
456
+
457
+        // If it's explicitly guarded, then it's not fillable.
458
+        if ( in_array( $name, $this->guarded ) ) {
459
+            return false;
460
+        }
461
+
462
+        // If fillable hasn't been defined, then everything else fillable.
463
+        return ! $this->fillable;
464
+    }
465
+
466
+    /**
467
+     * Overrides the current WordPress object with a provided one.
468
+     *
469
+     * Resets the post's default values and stores it in the attributes.
470
+     *
471
+     * @param WP_Post|WP_Term|null $value
472
+     *
473
+     * @return $this
474
+     */
475
+    private function override_wp_object( $value ) {
476
+        if ( is_object( $value ) ) {
477
+            $this->attributes[ self::OBJECT_KEY ] = $this->set_wp_object_constants( $value );
478
+        } else {
479
+            $this->attributes[ self::OBJECT_KEY ] = null;
480
+
481
+            if ( $this->uses_wp_object() ) {
482
+                $this->create_wp_object();
483
+            }
484
+        }
485
+
486
+        return $this;
487
+    }
488
+
489
+    /**
490
+     * Overrides the current table attributes array with a provided one.
491
+     *
492
+     * @param array $value
493
+     *
494
+     * @return $this
495
+     */
496
+    private function override_table( array $value ) {
497
+        $this->attributes[ self::TABLE_KEY ] = $value;
498
+
499
+        return $this;
500
+    }
501
+
502
+    /**
503
+     * Create and set with a new blank post.
504
+     *
505
+     * Creates a new WP_Post object, assigns it the default attributes,
506
+     * and stores it in the attributes.
507
+     *
508
+     * @throws LogicException
509
+     */
510
+    private function create_wp_object() {
511
+        switch ( true ) {
512
+            case $this instanceof UsesWordPressPost:
513
+                $object = new WP_Post( (object) array() );
514
+                break;
515
+            case $this instanceof UsesWordPressTerm:
516
+                $object = new WP_Term( (object) array() );
517
+                break;
518
+            default:
519
+                throw new LogicException;
520
+                break;
521
+        }
522
+
523
+        $this->attributes[ self::OBJECT_KEY ] = $this->set_wp_object_constants( $object );
524
+    }
525
+
526
+    /**
527
+     * Enforces values on the post that can't change.
528
+     *
529
+     * Primarily, this is used to make sure the post_type always maps
530
+     * to the model's "$type" property, but this can all be overridden
531
+     * by the developer to enforce other values in the model.
532
+     *
533
+     * @param object $object
534
+     *
535
+     * @return object
536
+     */
537
+    protected function set_wp_object_constants( $object ) {
538
+        if ( $this instanceof UsesWordPressPost ) {
539
+            $object->post_type = static::get_post_type();
540
+        }
541
+
542
+        if ( $this instanceof UsesWordPressTerm ) {
543
+            $object->taxonomy = static::get_taxonomy();
544
+        }
545
+
546
+        return $object;
547
+    }
548
+
549
+    /**
550
+     * Magic __get method.
551
+     *
552
+     * Passes the name and value to get_attribute, which is where the magic happens.
553
+     *
554
+     * @param string $name
555
+     *
556
+     * @return mixed
557
+     */
558
+    public function __get( $name ) {
559
+        return $this->get_attribute( $name );
560
+    }
561
+
562
+    /**
563
+     * Retrieves the model attribute.
564
+     *
565
+     * @param string $name
566
+     *
567
+     * @return mixed
568
+     *
569
+     * @throws PropertyDoesNotExistException If property isn't found.
570
+     */
571
+    public function get_attribute( $name ) {
572
+        if ( $method = $this->has_map_method( $name ) ) {
573
+            return $this->attributes[ self::OBJECT_KEY ]->{$this->{$method}()};
574
+        }
575
+
576
+        if ( $method = $this->has_compute_method( $name ) ) {
577
+            return $this->{$method}();
578
+        }
579
+
580
+        if ( isset( $this->attributes[ self::TABLE_KEY ][ $name ] ) ) {
581
+            return $this->attributes[ self::TABLE_KEY ][ $name ];
582
+        }
583
+
584
+        if ( isset( $this->defaults[ $name ] ) ) {
585
+            return $this->defaults[ $name ];
586
+        }
587
+
588
+        throw new PropertyDoesNotExistException( $name );
589
+    }
590
+
591
+    /**
592
+     * Retrieve the model's original attribute value.
593
+     *
594
+     * @param string $name
595
+     *
596
+     * @return mixed
597
+     *
598
+     * @throws PropertyDoesNotExistException If property isn't found.
599
+     */
600
+    public function get_original_attribute( $name ) {
601
+        $original_attributes = $this->original;
602
+
603
+        if ( ! is_object( $original_attributes[ static::OBJECT_KEY ] ) ) {
604
+            unset( $original_attributes[ static::OBJECT_KEY ] );
605
+        }
606
+
607
+        $original = new static( $original_attributes );
608
+
609
+        try {
610
+            return $original->get_attribute( $name );
611
+        } catch ( Exception $exception ) {
612
+            return null;
613
+        }
614
+    }
615
+
616
+    /**
617
+     * Fetches the Model's primary ID, depending on the model
618
+     * implementation.
619
+     *
620
+     * @return int
621
+     *
622
+     * @throws LogicException
623
+     */
624
+    public function get_primary_id() {
625
+        if ( $this instanceof UsesWordPressPost ) {
626
+            return $this->get_underlying_wp_object()->ID;
627
+        }
628
+
629
+        if ( $this instanceof UsesWordPressTerm ) {
630
+            return $this->get_underlying_wp_object()->term_id;
631
+        }
632
+
633
+        // Model w/o wp_object not yet supported.
634
+        throw new LogicException;
635
+    }
636
+
637
+    /**
638
+     * Checks whether the attribute has a map method.
639
+     *
640
+     * This is used to determine whether the attribute maps to a
641
+     * property on the underlying WP_Post object. Returns the
642
+     * method if one exists, returns false if it doesn't.
643
+     *
644
+     * @param string $name
645
+     *
646
+     * @return false|string
647
+     */
648
+    protected function has_map_method( $name ) {
649
+        if ( method_exists( $this, $method = "map_{$name}" ) ) {
650
+            return $method;
651
+        }
652
+
653
+        return false;
654
+    }
655
+
656
+    /**
657
+     * Checks whether the attribute has a compute method.
658
+     *
659
+     * This is used to determine if the attribute should be computed
660
+     * from other attributes.
661
+     *
662
+     * @param string $name
663
+     *
664
+     * @return false|string
665
+     */
666
+    protected function has_compute_method( $name ) {
667
+        if ( method_exists( $this, $method = "compute_{$name}" ) ) {
668
+            return $method;
669
+        }
670
+
671
+        return false;
672
+    }
673
+
674
+    /**
675
+     * Clears all the current attributes from the model.
676
+     *
677
+     * This does not touch the model's original attributes, and will
678
+     * only clear fillable attributes, unless the model is unguarded.
679
+     *
680
+     * @return $this
681
+     */
682
+    public function clear() {
683
+        $keys = array_merge(
684
+            $this->get_table_keys(),
685
+            $this->get_wp_object_keys()
686
+        );
687
+
688
+        foreach ( $keys as $key ) {
689
+            try {
690
+                $this->set_attribute( $key, null );
691
+            } catch ( GuardedPropertyException $e ) {
692
+                // We won't clear out guarded attributes.
693
+            }
694
+        }
695
+
696
+        return $this;
697
+    }
698
+
699
+    /**
700
+     * Unguards the model.
701
+     *
702
+     * Sets the model to be unguarded, allowing the filling of
703
+     * guarded attributes.
704
+     */
705
+    public function unguard() {
706
+        $this->is_guarded = false;
707
+    }
708
+
709
+    /**
710
+     * Reguards the model.
711
+     *
712
+     * Sets the model to be guarded, preventing filling of
713
+     * guarded attributes.
714
+     */
715
+    public function reguard() {
716
+        $this->is_guarded = true;
717
+    }
718
+
719
+    /**
720
+     * Retrieves all the compute methods on the model.
721
+     *
722
+     * @return array
723
+     */
724
+    protected function get_compute_methods() {
725
+        $methods = get_class_methods( get_called_class() );
726
+        $methods = array_filter( $methods, function ( $method ) {
727
+            return strrpos( $method, 'compute_', - strlen( $method ) ) !== false;
728
+        } );
729
+        $methods = array_map( function ( $method ) {
730
+            return substr( $method, strlen( 'compute_' ) );
731
+        }, $methods );
732
+
733
+        return $methods;
734
+    }
735
+
736
+    /**
737
+     * Sets up the memo array for the creating model.
738
+     */
739
+    private function maybe_boot() {
740
+        if ( ! isset( self::$memo[ get_called_class() ] ) ) {
741
+            self::$memo[ get_called_class() ] = array();
742
+        }
743
+    }
744
+
745
+    /**
746
+     * Whether this Model uses an underlying WordPress object.
747
+     *
748
+     * @return bool
749
+     */
750
+    protected function uses_wp_object() {
751
+        return $this instanceof UsesWordPressPost ||
752
+            $this instanceof UsesWordPressTerm;
753
+    }
754 754
 }
Please login to merge, or discard this patch.
src/Axolotl/Type.php 1 patch
Indentation   +153 added lines, -153 removed lines patch added patch discarded remove patch
@@ -15,157 +15,157 @@
 block discarded – undo
15 15
  */
16 16
 class Type {
17 17
 
18
-	/**
19
-	 * Type to validate against.
20
-	 *
21
-	 * @var string
22
-	 */
23
-	private $type;
24
-
25
-	/**
26
-	 * Type constructor.
27
-	 *
28
-	 * @param string $type
29
-	 */
30
-	public function __construct( $type ) {
31
-		$this->type = $this->determine( $type );
32
-	}
33
-
34
-	/**
35
-	 * Get validation type.
36
-	 *
37
-	 * @return string
38
-	 */
39
-	public function get_type() {
40
-		return $this->type;
41
-	}
42
-
43
-	/**
44
-	 * Returns whether the type is an Axolotl model.
45
-	 *
46
-	 * @return bool
47
-	 */
48
-	public function is_model() {
49
-		if ( ! class_exists( $this->type ) ) {
50
-			return false;
51
-		}
52
-
53
-		$reflection = new ReflectionClass( $this->type );
54
-		return $reflection->isSubclassOf( 'Intraxia\Jaxion\Axolotl\Model' );
55
-	}
56
-
57
-	/**
58
-	 * Create a new model from the given data.
59
-	 *
60
-	 * @param array $data Data for the model.
61
-	 *
62
-	 * @return Model
63
-	 */
64
-	public function create_model( array $data ) {
65
-		return new $this->type( $data );
66
-	}
67
-
68
-	/**
69
-	 * Validates an array of element.
70
-	 *
71
-	 * @param array $elements Elements to be validated.
72
-	 *
73
-	 * @throws InvalidArgumentException
74
-	 */
75
-	public function validate_elements( array $elements ) {
76
-		foreach ( $elements as $element ) {
77
-			$this->validate_element( $element );
78
-		}
79
-	}
80
-
81
-	/**
82
-	 * Validate whether the
83
-	 *
84
-	 * @param mixed $element Element to validate.
85
-	 *
86
-	 * @throws InvalidArgumentException
87
-	 */
88
-	public function validate_element( $element ) {
89
-		$type = gettype( $element );
90
-		$callable = $this->type === 'callable';
91
-		$is_object = 'object' === $type;
92
-		$loose_check = $this->type === 'object';
93
-
94
-		// callable must be callable
95
-		if ( $callable && ! is_callable( $element ) ) {
96
-			throw new InvalidArgumentException( 'Item must be callable' );
97
-		}
98
-
99
-		// target isn't callable, object must be an instance of target
100
-		if ( ! $loose_check && ! $callable && $is_object && ! is_a( $element, $this->type ) ) {
101
-			throw new InvalidArgumentException( "Item is not type or subtype of $this->type" );
102
-		}
103
-
104
-		// a non callable, non object type should match the target string
105
-		if ( ! $callable && ! $is_object && $type !== $this->type ) {
106
-			throw new InvalidArgumentException( "Item is not of type: $this->type" );
107
-		}
108
-	}
109
-
110
-	/**
111
-	 * Determine the type to validate against.
112
-	 *
113
-	 * @param string $type     Type to determine.
114
-	 * @param bool   $key_type Whether the type is for keys.
115
-	 *
116
-	 * @return string
117
-	 *
118
-	 * @throws InvalidArgumentException
119
-	 */
120
-	private function determine( $type, $key_type = false ) {
121
-		if ( ! $key_type && $this->non_scalar_type_exists( $type ) ) {
122
-			return $type;
123
-		}
124
-
125
-		if ( $scalar_type = $this->determine_scalar( $type ) ) {
126
-			if ( $key_type && (in_array( $scalar_type, [ 'double', 'boolean' ] )) ) {
127
-				throw new InvalidArgumentException( 'This type is not supported as a key.' );
128
-			}
129
-
130
-			return $scalar_type;
131
-		}
132
-
133
-		throw new InvalidArgumentException( 'This type does not exist.' );
134
-	}
135
-
136
-	/**
137
-	 * Determines whether the given type exists.
138
-	 *
139
-	 * @param string $type Type to check.
140
-	 *
141
-	 * @return bool
142
-	 */
143
-	private function non_scalar_type_exists( $type ) {
144
-		return class_exists( $type )
145
-				|| interface_exists( $type )
146
-				|| in_array( $type, [ 'array', 'object', 'callable' ] );
147
-	}
148
-
149
-	/**
150
-	 * Returns the type if it's scalar, otherwise, returns null.
151
-	 *
152
-	 * @param string $type Type to check.
153
-	 *
154
-	 * @return string|null
155
-	 */
156
-	private function determine_scalar( $type ) {
157
-		$synonyms = array(
158
-			'int' => 'integer',
159
-			'float' => 'double',
160
-			'bool' => 'boolean',
161
-		);
162
-
163
-		if ( array_key_exists( $type, $synonyms ) ) {
164
-			$type = $synonyms[ $type ];
165
-		}
166
-
167
-		return in_array( $type, array( 'string', 'integer', 'double', 'boolean' ) ) ?
168
-			$type :
169
-			null;
170
-	}
18
+    /**
19
+     * Type to validate against.
20
+     *
21
+     * @var string
22
+     */
23
+    private $type;
24
+
25
+    /**
26
+     * Type constructor.
27
+     *
28
+     * @param string $type
29
+     */
30
+    public function __construct( $type ) {
31
+        $this->type = $this->determine( $type );
32
+    }
33
+
34
+    /**
35
+     * Get validation type.
36
+     *
37
+     * @return string
38
+     */
39
+    public function get_type() {
40
+        return $this->type;
41
+    }
42
+
43
+    /**
44
+     * Returns whether the type is an Axolotl model.
45
+     *
46
+     * @return bool
47
+     */
48
+    public function is_model() {
49
+        if ( ! class_exists( $this->type ) ) {
50
+            return false;
51
+        }
52
+
53
+        $reflection = new ReflectionClass( $this->type );
54
+        return $reflection->isSubclassOf( 'Intraxia\Jaxion\Axolotl\Model' );
55
+    }
56
+
57
+    /**
58
+     * Create a new model from the given data.
59
+     *
60
+     * @param array $data Data for the model.
61
+     *
62
+     * @return Model
63
+     */
64
+    public function create_model( array $data ) {
65
+        return new $this->type( $data );
66
+    }
67
+
68
+    /**
69
+     * Validates an array of element.
70
+     *
71
+     * @param array $elements Elements to be validated.
72
+     *
73
+     * @throws InvalidArgumentException
74
+     */
75
+    public function validate_elements( array $elements ) {
76
+        foreach ( $elements as $element ) {
77
+            $this->validate_element( $element );
78
+        }
79
+    }
80
+
81
+    /**
82
+     * Validate whether the
83
+     *
84
+     * @param mixed $element Element to validate.
85
+     *
86
+     * @throws InvalidArgumentException
87
+     */
88
+    public function validate_element( $element ) {
89
+        $type = gettype( $element );
90
+        $callable = $this->type === 'callable';
91
+        $is_object = 'object' === $type;
92
+        $loose_check = $this->type === 'object';
93
+
94
+        // callable must be callable
95
+        if ( $callable && ! is_callable( $element ) ) {
96
+            throw new InvalidArgumentException( 'Item must be callable' );
97
+        }
98
+
99
+        // target isn't callable, object must be an instance of target
100
+        if ( ! $loose_check && ! $callable && $is_object && ! is_a( $element, $this->type ) ) {
101
+            throw new InvalidArgumentException( "Item is not type or subtype of $this->type" );
102
+        }
103
+
104
+        // a non callable, non object type should match the target string
105
+        if ( ! $callable && ! $is_object && $type !== $this->type ) {
106
+            throw new InvalidArgumentException( "Item is not of type: $this->type" );
107
+        }
108
+    }
109
+
110
+    /**
111
+     * Determine the type to validate against.
112
+     *
113
+     * @param string $type     Type to determine.
114
+     * @param bool   $key_type Whether the type is for keys.
115
+     *
116
+     * @return string
117
+     *
118
+     * @throws InvalidArgumentException
119
+     */
120
+    private function determine( $type, $key_type = false ) {
121
+        if ( ! $key_type && $this->non_scalar_type_exists( $type ) ) {
122
+            return $type;
123
+        }
124
+
125
+        if ( $scalar_type = $this->determine_scalar( $type ) ) {
126
+            if ( $key_type && (in_array( $scalar_type, [ 'double', 'boolean' ] )) ) {
127
+                throw new InvalidArgumentException( 'This type is not supported as a key.' );
128
+            }
129
+
130
+            return $scalar_type;
131
+        }
132
+
133
+        throw new InvalidArgumentException( 'This type does not exist.' );
134
+    }
135
+
136
+    /**
137
+     * Determines whether the given type exists.
138
+     *
139
+     * @param string $type Type to check.
140
+     *
141
+     * @return bool
142
+     */
143
+    private function non_scalar_type_exists( $type ) {
144
+        return class_exists( $type )
145
+                || interface_exists( $type )
146
+                || in_array( $type, [ 'array', 'object', 'callable' ] );
147
+    }
148
+
149
+    /**
150
+     * Returns the type if it's scalar, otherwise, returns null.
151
+     *
152
+     * @param string $type Type to check.
153
+     *
154
+     * @return string|null
155
+     */
156
+    private function determine_scalar( $type ) {
157
+        $synonyms = array(
158
+            'int' => 'integer',
159
+            'float' => 'double',
160
+            'bool' => 'boolean',
161
+        );
162
+
163
+        if ( array_key_exists( $type, $synonyms ) ) {
164
+            $type = $synonyms[ $type ];
165
+        }
166
+
167
+        return in_array( $type, array( 'string', 'integer', 'double', 'boolean' ) ) ?
168
+            $type :
169
+            null;
170
+    }
171 171
 }
Please login to merge, or discard this patch.
src/Contract/Axolotl/Collection.php 1 patch
Indentation   +314 added lines, -314 removed lines patch added patch discarded remove patch
@@ -9,318 +9,318 @@
 block discarded – undo
9 9
 
10 10
 interface Collection extends Iterator, Countable, Serializes {
11 11
 
12
-	/**
13
-	 * Returns the type of the collection.
14
-	 *
15
-	 * @return string
16
-	 */
17
-	public function get_type();
18
-
19
-	/**
20
-	 * Returns a collection with $element added.
21
-	 *
22
-	 * @param  mixed $element
23
-	 * @return Collection
24
-	 * @throws InvalidArgumentException
25
-	 */
26
-	public function add( $element );
27
-
28
-	/**
29
-	 * Removes every element from the collection.
30
-	 *
31
-	 * @return Collection
32
-	 */
33
-	public function clear();
34
-
35
-	/**
36
-	 * Returns true if the collection contains any elements that satisfy
37
-	 * $condition, returns false if it contains none.
38
-	 *
39
-	 * @param callable $condition
40
-	 * @return bool
41
-	 */
42
-	public function contains( callable $condition );
43
-
44
-	/**
45
-	 * Returns the first element in the collection that satisfies
46
-	 * $condition, returns false if no such element exists.
47
-	 *
48
-	 * @param callable $condition
49
-	 * @return mixed
50
-	 */
51
-	public function find( callable $condition );
52
-
53
-	/**
54
-	 * Returns the index of the first element in the collection that satisfies
55
-	 * $condition, returns -1 if no such element exists.
56
-	 *
57
-	 * @param callable $condition
58
-	 * @return int
59
-	 */
60
-	public function find_index( callable $condition );
61
-
62
-	/**
63
-	 * Returns the element in the collection at $index.
64
-	 *
65
-	 * @param  int $index Index to get element from.
66
-	 * @return mixed
67
-	 * @throws OutOfRangeException
68
-	 */
69
-	public function at( $index );
70
-
71
-	/**
72
-	 * Returns true if $index is within the collection's range and returns false
73
-	 * if it is not.
74
-	 *
75
-	 * @param  int $index Index to check for existence.
76
-	 * @return bool
77
-	 * @throws InvalidArgumentException
78
-	 */
79
-	public function index_exists( $index );
80
-
81
-	/**
82
-	 * Returns the number of elements in the collection.
83
-	 *
84
-	 * @return int
85
-	 */
86
-	public function count();
87
-
88
-	/**
89
-	 * Returns a collection that only contains the elements which satisfy
90
-	 * $condition.
91
-	 *
92
-	 * @param callable $condition
93
-	 * @return Collection
94
-	 */
95
-	public function filter( callable $condition );
96
-
97
-	/**
98
-	 * Returns the last element in the collection that satisfies $condition,
99
-	 * returns false if no such element exists.
100
-	 *
101
-	 * @param callable $condition
102
-	 * @return mixed
103
-	 */
104
-	public function find_last( callable $condition );
105
-
106
-	/**
107
-	 * Returns the index of the last element in the collection that satisfies
108
-	 * $condition, returns -1 if no such element exists.
109
-	 *
110
-	 * @param callable $condition
111
-	 * @return int
112
-	 */
113
-	public function find_last_index( callable $condition );
114
-
115
-	/**
116
-	 * Returns a collection that contains the subset of elements ranging from the
117
-	 * index $start to $end.
118
-	 *
119
-	 * @param  int $start Begining index to slice from.
120
-	 * @param  int $end   End index to slice to.
121
-	 * @return Collection
122
-	 * @throws InvalidArgumentException
123
-	 */
124
-	public function slice( $start, $end );
125
-
126
-	/**
127
-	 * Inserts $element at $index.
128
-	 *
129
-	 * @param int   $index     Index to start at.
130
-	 * @param mixed $element Element to insert.
131
-	 * @return Collection
132
-	 * @throws InvalidArgumentException
133
-	 * @throws OutOfRangeException
134
-	 */
135
-	public function insert( $index, $element );
136
-
137
-	/**
138
-	 * Inserts the range $elements at $index.
139
-	 *
140
-	 * @param int   $index
141
-	 * @param array $elements
142
-	 * @return Collection
143
-	 * @throws OutOfRangeException
144
-	 */
145
-	public function insert_range( $index, array $elements );
146
-
147
-	/**
148
-	 * Removes all of the elements that satisfy $condition.
149
-	 *
150
-	 * @param  callable $condition
151
-	 * @return Collection
152
-	 */
153
-	public function without( callable $condition );
154
-
155
-	/**
156
-	 * Removes the element at $index.
157
-	 *
158
-	 * @param  int $index Index to remove.
159
-	 * @return Collection
160
-	 * @throws OutOfRangeException
161
-	 */
162
-	public function remove_at( $index );
163
-
164
-	/**
165
-	 * Reverses the order of the elements in the collection.
166
-	 *
167
-	 * @return Collection
168
-	 */
169
-	public function reverse();
170
-
171
-	/**
172
-	 * Sorts the elements in the collection using the user supplied comparison
173
-	 * function $callback.
174
-	 *
175
-	 * @param callable $callback
176
-	 * @return Collection
177
-	 */
178
-	public function sort( callable $callback );
179
-
180
-	/**
181
-	 * Returns an array containing the elements in the collection.
182
-	 *
183
-	 * @return array
184
-	 */
185
-	public function to_array();
186
-
187
-	/**
188
-	 * Iteratively reduces the collection to a single value using the callback
189
-	 * function $callable.
190
-	 *
191
-	 * @param callable $callable
192
-	 * @param null     $initial
193
-	 * @return mixed
194
-	 */
195
-	public function reduce( callable $callable, $initial = null );
196
-
197
-	/**
198
-	 * Returns true if every element in the collection satisfies $condition,
199
-	 * returns false if not.
200
-	 *
201
-	 * @param callable $condition
202
-	 * @return bool
203
-	 */
204
-	public function every( callable $condition );
205
-
206
-	/**
207
-	 * Removes all of the elements in the collection starting at index $num.
208
-	 *
209
-	 * @param  int $num Number of elements to drop.
210
-	 * @return Collection
211
-	 * @throws InvalidArgumentException
212
-	 */
213
-	public function drop( $num );
214
-
215
-	/**
216
-	 * Removes all of the elements in the collectioin between index 0 and $num.
217
-	 *
218
-	 * @param int $num Number of elements to drop.
219
-	 * @return Collection
220
-	 * @throws InvalidArgumentException
221
-	 */
222
-	public function drop_right( $num );
223
-
224
-	/**
225
-	 * Iteratively drops elements in the collection that satisfy $condition until
226
-	 * an element is encountered that does not satisfy $condition.
227
-	 *
228
-	 * @param callable $condition
229
-	 * @return Collection
230
-	 */
231
-	public function drop_while( callable $condition );
232
-
233
-	/**
234
-	 * Removes the first element in the collection.
235
-	 *
236
-	 * @return Collection
237
-	 * @throws InvalidArgumentException
238
-	 */
239
-	public function tail();
240
-
241
-	/**
242
-	 * Removes all of the elements in the collection starting at index $num.
243
-	 *
244
-	 * @param  int $num Number of elements to take.
245
-	 * @return Collection
246
-	 * @throws InvalidArgumentException
247
-	 */
248
-	public function take( $num );
249
-
250
-	/**
251
-	 * Removes all of the elements in the collection before index $num.
252
-	 *
253
-	 * @param int $num Number of elements to take.
254
-	 * @return Collection
255
-	 * @throws InvalidArgumentException
256
-	 */
257
-	public function take_right( $num );
258
-
259
-	/**
260
-	 * Iterates through the collection until an element is encountered that does
261
-	 * not satisfy $condition, then drops all of the elements starting at that
262
-	 * index.
263
-	 *
264
-	 * @param callable $condition
265
-	 * @return Collection
266
-	 */
267
-	public function take_while( callable $condition );
268
-
269
-	/**
270
-	 * Applies the callback function $callable to each element in the collection.
271
-	 *
272
-	 * @param callable $callable
273
-	 */
274
-	public function each( callable $callable );
275
-
276
-	/**
277
-	 * Returns a new instance of the collection with the callback function
278
-	 * $callable applied to each element.
279
-	 *
280
-	 * @param callable $callable
281
-	 * @return Collection
282
-	 */
283
-	public function map( callable $callable );
284
-
285
-	/**
286
-	 * Iteratively reduces the collection to a single value using the callback
287
-	 * function $callable starting at the rightmost index.
288
-	 *
289
-	 * @param callable $callable
290
-	 * @param null     $initial
291
-	 * @return mixed
292
-	 */
293
-	public function reduce_right( callable $callable, $initial = null );
294
-
295
-	/**
296
-	 * Randomly reorders the elements in the collection.
297
-	 *
298
-	 * @return Collection
299
-	 */
300
-	public function shuffle();
301
-
302
-	/**
303
-	 * Adds every member of $elements to the collection.
304
-	 *
305
-	 * @param array|Collection $elements Array of elements to merge.
306
-	 * @return Collection
307
-	 * @throws InvalidArgumentException
308
-	 */
309
-	public function merge( $elements );
310
-
311
-	/**
312
-	 * Get first element of the collection
313
-	 *
314
-	 * @return mixed
315
-	 * @throws OutOfBoundsException
316
-	 */
317
-	public function first();
318
-
319
-	/**
320
-	 * Get last element of the collection
321
-	 *
322
-	 * @return mixed
323
-	 * @throws OutOfBoundsException
324
-	 */
325
-	public function last();
12
+    /**
13
+     * Returns the type of the collection.
14
+     *
15
+     * @return string
16
+     */
17
+    public function get_type();
18
+
19
+    /**
20
+     * Returns a collection with $element added.
21
+     *
22
+     * @param  mixed $element
23
+     * @return Collection
24
+     * @throws InvalidArgumentException
25
+     */
26
+    public function add( $element );
27
+
28
+    /**
29
+     * Removes every element from the collection.
30
+     *
31
+     * @return Collection
32
+     */
33
+    public function clear();
34
+
35
+    /**
36
+     * Returns true if the collection contains any elements that satisfy
37
+     * $condition, returns false if it contains none.
38
+     *
39
+     * @param callable $condition
40
+     * @return bool
41
+     */
42
+    public function contains( callable $condition );
43
+
44
+    /**
45
+     * Returns the first element in the collection that satisfies
46
+     * $condition, returns false if no such element exists.
47
+     *
48
+     * @param callable $condition
49
+     * @return mixed
50
+     */
51
+    public function find( callable $condition );
52
+
53
+    /**
54
+     * Returns the index of the first element in the collection that satisfies
55
+     * $condition, returns -1 if no such element exists.
56
+     *
57
+     * @param callable $condition
58
+     * @return int
59
+     */
60
+    public function find_index( callable $condition );
61
+
62
+    /**
63
+     * Returns the element in the collection at $index.
64
+     *
65
+     * @param  int $index Index to get element from.
66
+     * @return mixed
67
+     * @throws OutOfRangeException
68
+     */
69
+    public function at( $index );
70
+
71
+    /**
72
+     * Returns true if $index is within the collection's range and returns false
73
+     * if it is not.
74
+     *
75
+     * @param  int $index Index to check for existence.
76
+     * @return bool
77
+     * @throws InvalidArgumentException
78
+     */
79
+    public function index_exists( $index );
80
+
81
+    /**
82
+     * Returns the number of elements in the collection.
83
+     *
84
+     * @return int
85
+     */
86
+    public function count();
87
+
88
+    /**
89
+     * Returns a collection that only contains the elements which satisfy
90
+     * $condition.
91
+     *
92
+     * @param callable $condition
93
+     * @return Collection
94
+     */
95
+    public function filter( callable $condition );
96
+
97
+    /**
98
+     * Returns the last element in the collection that satisfies $condition,
99
+     * returns false if no such element exists.
100
+     *
101
+     * @param callable $condition
102
+     * @return mixed
103
+     */
104
+    public function find_last( callable $condition );
105
+
106
+    /**
107
+     * Returns the index of the last element in the collection that satisfies
108
+     * $condition, returns -1 if no such element exists.
109
+     *
110
+     * @param callable $condition
111
+     * @return int
112
+     */
113
+    public function find_last_index( callable $condition );
114
+
115
+    /**
116
+     * Returns a collection that contains the subset of elements ranging from the
117
+     * index $start to $end.
118
+     *
119
+     * @param  int $start Begining index to slice from.
120
+     * @param  int $end   End index to slice to.
121
+     * @return Collection
122
+     * @throws InvalidArgumentException
123
+     */
124
+    public function slice( $start, $end );
125
+
126
+    /**
127
+     * Inserts $element at $index.
128
+     *
129
+     * @param int   $index     Index to start at.
130
+     * @param mixed $element Element to insert.
131
+     * @return Collection
132
+     * @throws InvalidArgumentException
133
+     * @throws OutOfRangeException
134
+     */
135
+    public function insert( $index, $element );
136
+
137
+    /**
138
+     * Inserts the range $elements at $index.
139
+     *
140
+     * @param int   $index
141
+     * @param array $elements
142
+     * @return Collection
143
+     * @throws OutOfRangeException
144
+     */
145
+    public function insert_range( $index, array $elements );
146
+
147
+    /**
148
+     * Removes all of the elements that satisfy $condition.
149
+     *
150
+     * @param  callable $condition
151
+     * @return Collection
152
+     */
153
+    public function without( callable $condition );
154
+
155
+    /**
156
+     * Removes the element at $index.
157
+     *
158
+     * @param  int $index Index to remove.
159
+     * @return Collection
160
+     * @throws OutOfRangeException
161
+     */
162
+    public function remove_at( $index );
163
+
164
+    /**
165
+     * Reverses the order of the elements in the collection.
166
+     *
167
+     * @return Collection
168
+     */
169
+    public function reverse();
170
+
171
+    /**
172
+     * Sorts the elements in the collection using the user supplied comparison
173
+     * function $callback.
174
+     *
175
+     * @param callable $callback
176
+     * @return Collection
177
+     */
178
+    public function sort( callable $callback );
179
+
180
+    /**
181
+     * Returns an array containing the elements in the collection.
182
+     *
183
+     * @return array
184
+     */
185
+    public function to_array();
186
+
187
+    /**
188
+     * Iteratively reduces the collection to a single value using the callback
189
+     * function $callable.
190
+     *
191
+     * @param callable $callable
192
+     * @param null     $initial
193
+     * @return mixed
194
+     */
195
+    public function reduce( callable $callable, $initial = null );
196
+
197
+    /**
198
+     * Returns true if every element in the collection satisfies $condition,
199
+     * returns false if not.
200
+     *
201
+     * @param callable $condition
202
+     * @return bool
203
+     */
204
+    public function every( callable $condition );
205
+
206
+    /**
207
+     * Removes all of the elements in the collection starting at index $num.
208
+     *
209
+     * @param  int $num Number of elements to drop.
210
+     * @return Collection
211
+     * @throws InvalidArgumentException
212
+     */
213
+    public function drop( $num );
214
+
215
+    /**
216
+     * Removes all of the elements in the collectioin between index 0 and $num.
217
+     *
218
+     * @param int $num Number of elements to drop.
219
+     * @return Collection
220
+     * @throws InvalidArgumentException
221
+     */
222
+    public function drop_right( $num );
223
+
224
+    /**
225
+     * Iteratively drops elements in the collection that satisfy $condition until
226
+     * an element is encountered that does not satisfy $condition.
227
+     *
228
+     * @param callable $condition
229
+     * @return Collection
230
+     */
231
+    public function drop_while( callable $condition );
232
+
233
+    /**
234
+     * Removes the first element in the collection.
235
+     *
236
+     * @return Collection
237
+     * @throws InvalidArgumentException
238
+     */
239
+    public function tail();
240
+
241
+    /**
242
+     * Removes all of the elements in the collection starting at index $num.
243
+     *
244
+     * @param  int $num Number of elements to take.
245
+     * @return Collection
246
+     * @throws InvalidArgumentException
247
+     */
248
+    public function take( $num );
249
+
250
+    /**
251
+     * Removes all of the elements in the collection before index $num.
252
+     *
253
+     * @param int $num Number of elements to take.
254
+     * @return Collection
255
+     * @throws InvalidArgumentException
256
+     */
257
+    public function take_right( $num );
258
+
259
+    /**
260
+     * Iterates through the collection until an element is encountered that does
261
+     * not satisfy $condition, then drops all of the elements starting at that
262
+     * index.
263
+     *
264
+     * @param callable $condition
265
+     * @return Collection
266
+     */
267
+    public function take_while( callable $condition );
268
+
269
+    /**
270
+     * Applies the callback function $callable to each element in the collection.
271
+     *
272
+     * @param callable $callable
273
+     */
274
+    public function each( callable $callable );
275
+
276
+    /**
277
+     * Returns a new instance of the collection with the callback function
278
+     * $callable applied to each element.
279
+     *
280
+     * @param callable $callable
281
+     * @return Collection
282
+     */
283
+    public function map( callable $callable );
284
+
285
+    /**
286
+     * Iteratively reduces the collection to a single value using the callback
287
+     * function $callable starting at the rightmost index.
288
+     *
289
+     * @param callable $callable
290
+     * @param null     $initial
291
+     * @return mixed
292
+     */
293
+    public function reduce_right( callable $callable, $initial = null );
294
+
295
+    /**
296
+     * Randomly reorders the elements in the collection.
297
+     *
298
+     * @return Collection
299
+     */
300
+    public function shuffle();
301
+
302
+    /**
303
+     * Adds every member of $elements to the collection.
304
+     *
305
+     * @param array|Collection $elements Array of elements to merge.
306
+     * @return Collection
307
+     * @throws InvalidArgumentException
308
+     */
309
+    public function merge( $elements );
310
+
311
+    /**
312
+     * Get first element of the collection
313
+     *
314
+     * @return mixed
315
+     * @throws OutOfBoundsException
316
+     */
317
+    public function first();
318
+
319
+    /**
320
+     * Get last element of the collection
321
+     *
322
+     * @return mixed
323
+     * @throws OutOfBoundsException
324
+     */
325
+    public function last();
326 326
 }
Please login to merge, or discard this patch.
src/Axolotl/Collection.php 1 patch
Indentation   +737 added lines, -737 removed lines patch added patch discarded remove patch
@@ -15,741 +15,741 @@
 block discarded – undo
15 15
  */
16 16
 class Collection implements CollectionContract {
17 17
 
18
-	/**
19
-	 * Collection elements.
20
-	 *
21
-	 * @var array
22
-	 */
23
-	protected $elements = array();
24
-
25
-	/**
26
-	 * Collection type to enforce.
27
-	 *
28
-	 * @var Type
29
-	 */
30
-	private $type;
31
-
32
-	/**
33
-	 * Where Collection is in loop.
34
-	 *
35
-	 * @var int
36
-	 */
37
-	protected $position = 0;
38
-
39
-	/**
40
-	 * Collection constructor.
41
-	 *
42
-	 * @param string $type
43
-	 * @param array  $elements
44
-	 */
45
-	public function __construct( $type, array $elements = array() ) {
46
-		$this->type = new Type( $type );
47
-
48
-		if ( $this->type->is_model() ) {
49
-			foreach ( $elements as $idx => $element ) {
50
-				if ( is_array( $element ) ) {
51
-					$elements[ $idx ] = $this->type->create_model( $element );
52
-				}
53
-			}
54
-		}
55
-
56
-		if ( $elements ) {
57
-			$this->type->validate_elements( $elements );
58
-		}
59
-
60
-		$this->elements = $elements;
61
-	}
62
-
63
-	/**
64
-	 * {@inheritdoc}
65
-	 *
66
-	 * @return string
67
-	 */
68
-	public function get_type() {
69
-		return $this->type->get_type();
70
-	}
71
-
72
-	/**
73
-	 * {@inheritdoc}
74
-	 *
75
-	 * @param mixed $element
76
-	 *
77
-	 * @return Collection
78
-	 *
79
-	 * @throws InvalidArgumentException
80
-	 */
81
-	public function add( $element ) {
82
-		if ( $this->type->is_model() && is_array( $element ) ) {
83
-			$element = $this->type->create_model( $element );
84
-		}
85
-
86
-		$this->type->validate_element( $element );
87
-
88
-		$elements   = $this->elements;
89
-		$elements[] = $element;
90
-
91
-		$collection = new static( $this->get_type() );
92
-		$collection->set_from_trusted( $elements );
93
-
94
-		return $collection;
95
-	}
96
-
97
-	/**
98
-	 * {@inheritdoc}
99
-	 *
100
-	 * @return Collection
101
-	 */
102
-	public function clear() {
103
-		return new static( $this->get_type() );
104
-	}
105
-
106
-	/**
107
-	 * {@inheritdoc}
108
-	 *
109
-	 * @param  callable $condition Condition to satisfy.
110
-	 *
111
-	 * @return bool
112
-	 */
113
-	public function contains( callable $condition ) {
114
-		return (bool) $this->find( $condition );
115
-	}
116
-
117
-	/**
118
-	 * {@inheritdoc}
119
-	 *
120
-	 * @param  callable $condition Condition to satisfy.
121
-	 *
122
-	 * @return mixed
123
-	 */
124
-	public function find( callable $condition ) {
125
-		$index = $this->find_index( $condition );
126
-
127
-		return -1 === $index ? false : $this->elements[ $index ];
128
-	}
129
-
130
-	/**
131
-	 * {@inheritdoc}
132
-	 *
133
-	 * @param  callable $condition Condition to satisfy.
134
-	 *
135
-	 * @return int
136
-	 */
137
-	public function find_index( callable $condition ) {
138
-		$index = -1;
139
-
140
-		for ( $i = 0, $count = count( $this->elements ); $i < $count; $i++ ) {
141
-			if ( call_user_func( $condition, ($this->at( $i ) ) ) ) {
142
-				$index = $i;
143
-				break;
144
-			}
145
-		}
146
-
147
-		return $index;
148
-	}
149
-
150
-	/**
151
-	 * Fetches the element at the provided index.
152
-	 *
153
-	 * @param int $index Index to get element from.
154
-	 *
155
-	 * @return mixed
156
-	 *
157
-	 * @throws OutOfRangeException
158
-	 */
159
-	public function at( $index ) {
160
-		$this->validate_index( $index );
161
-
162
-		return $this->elements[ $index ];
163
-	}
164
-
165
-	/**
166
-	 * {@inheritdoc}
167
-	 *
168
-	 * @param  int $index Index to check for existence.
169
-	 *
170
-	 * @return bool
171
-	 *
172
-	 * @throws InvalidArgumentException
173
-	 */
174
-	public function index_exists( $index ) {
175
-		if ( ! is_int( $index ) ) {
176
-			throw new InvalidArgumentException( 'Index must be an integer' );
177
-		}
178
-
179
-		if ( $index < 0 ) {
180
-			throw new InvalidArgumentException( 'Index must be a non-negative integer' );
181
-		}
182
-
183
-		return $index < $this->count();
184
-	}
185
-
186
-	/**
187
-	 * {@inheritdoc}
188
-	 *
189
-	 * @param  callable $condition Condition to satisfy.
190
-	 *
191
-	 * @return mixed
192
-	 */
193
-	public function filter( callable $condition ) {
194
-		$elements = array();
195
-
196
-		foreach ( $this->elements as $element ) {
197
-			if ( call_user_func( $condition, $element ) ) {
198
-				$elements[] = $element;
199
-			}
200
-		}
201
-
202
-		return $this->new_from_trusted( $elements );
203
-	}
204
-	/**
205
-	 * {@inheritdoc}
206
-	 *
207
-	 * @param  callable $condition Condition to satisfy.
208
-	 *
209
-	 * @return mixed
210
-	 */
211
-	public function find_last( callable $condition ) {
212
-		$index = $this->find_last_index( $condition );
213
-
214
-		return -1 === $index ? null : $this->elements[ $index ];
215
-	}
216
-
217
-	/**
218
-	 * {@inheritdoc}
219
-	 *
220
-	 * @param  callable $condition
221
-	 * @return int
222
-	 */
223
-	public function find_last_index( callable $condition ) {
224
-		$index = -1;
225
-
226
-		for ( $i = count( $this->elements ) - 1; $i >= 0; $i-- ) {
227
-			if ( call_user_func( $condition, $this->elements[ $i ] ) ) {
228
-				$index = $i;
229
-				break;
230
-			}
231
-		}
232
-
233
-		return $index;
234
-	}
235
-
236
-	/**
237
-	 * {@inheritdoc}
238
-	 *
239
-	 * @param  int $start Begining index to slice from.
240
-	 * @param  int $end   End index to slice to.
241
-	 *
242
-	 * @return Collection
243
-	 *
244
-	 * @throws InvalidArgumentException
245
-	 */
246
-	public function slice( $start, $end ) {
247
-		if ( $start < 0 || ! is_int( $start ) ) {
248
-			throw new InvalidArgumentException( 'Start must be a non-negative integer' );
249
-		}
250
-
251
-		if ( $end < 0 || ! is_int( $end ) ) {
252
-			throw new InvalidArgumentException( 'End must be a positive integer' );
253
-		}
254
-
255
-		if ( $start > $end ) {
256
-			throw new InvalidArgumentException( 'End must be greater than start' );
257
-		}
258
-
259
-		if ( $end > $this->count() + 1 ) {
260
-			throw new InvalidArgumentException( 'End must be less than the count of the items in the Collection' );
261
-		}
262
-
263
-		$length = $end - $start + 1;
264
-
265
-		return $this->new_from_trusted( array_slice( $this->elements, $start, $length ) );
266
-	}
267
-
268
-	/**
269
-	 * {@inheritdoc}
270
-	 *
271
-	 * @param int   $index     Index to start at.
272
-	 * @param mixed $element Element to insert.
273
-	 *
274
-	 * @return Collection
275
-	 *
276
-	 * @throws InvalidArgumentException
277
-	 * @throws OutOfRangeException
278
-	 */
279
-	public function insert( $index, $element ) {
280
-		$this->validate_index( $index );
281
-		$this->type->validate_element( $element );
282
-
283
-		$a = array_slice( $this->elements, 0, $index );
284
-		$b = array_slice( $this->elements, $index, count( $this->elements ) );
285
-
286
-		$a[] = $element;
287
-
288
-		return $this->new_from_trusted( array_merge( $a, $b ) );
289
-	}
290
-
291
-	/**
292
-	 * {@inheritdoc}
293
-	 *
294
-	 * @param int   $index    Index to start insertion at.
295
-	 * @param array $elements Elements in insert.
296
-	 *
297
-	 * @return Collection
298
-	 *
299
-	 * @throws OutOfRangeException
300
-	 */
301
-	public function insert_range( $index, array $elements ) {
302
-		$this->validate_index( $index );
303
-		$this->type->validate_elements( $elements );
304
-
305
-		if ( $index < 0 ) {
306
-			$index = $this->count() + $index + 1;
307
-		}
308
-
309
-		return $this->new_from_trusted(
310
-			array_merge(
311
-				array_slice( $this->elements, 0, $index ),
312
-				$elements,
313
-				array_slice( $this->elements, $index, count( $this->elements ) )
314
-			)
315
-		);
316
-	}
317
-
318
-	/**
319
-	 * {@inheritdoc}
320
-	 *
321
-	 * @param  callable $condition Condition to satisfy.
322
-	 *
323
-	 * @return Collection
324
-	 */
325
-	public function without( callable $condition ) {
326
-		$inverse = function ( $element ) use ( $condition ) {
327
-			return ! call_user_func( $condition, $element );
328
-		};
329
-
330
-		return $this->filter( $inverse );
331
-	}
332
-
333
-	/**
334
-	 * {@inheritdoc}
335
-	 *
336
-	 * @param  int $index Index to remove.
337
-	 *
338
-	 * @return Collection
339
-	 *
340
-	 * @throws OutOfRangeException
341
-	 */
342
-	public function remove_at( $index ) {
343
-		$this->validate_index( $index );
344
-
345
-		$elements = $this->elements;
346
-
347
-		return $this->new_from_trusted(
348
-			array_merge(
349
-				array_slice( $elements, 0, $index ),
350
-				array_slice( $elements, $index + 1, count( $elements ) )
351
-			)
352
-		);
353
-	}
354
-	/**
355
-	 * {@inheritdoc}
356
-	 *
357
-	 * @return Collection
358
-	 */
359
-	public function reverse() {
360
-		return $this->new_from_trusted(
361
-			array_reverse( $this->elements )
362
-		);
363
-	}
364
-
365
-	/**
366
-	 * {@inheritdoc}
367
-	 *
368
-	 * @param callable $callback Sort callback.
369
-	 *
370
-	 * @return Collection
371
-	 */
372
-	public function sort( callable $callback ) {
373
-		$elements = $this->elements;
374
-		usort( $elements, $callback );
375
-		return $this->new_from_trusted( $elements );
376
-	}
377
-
378
-	/**
379
-	 * {@inheritdoc}
380
-	 *
381
-	 * @return array
382
-	 */
383
-	public function to_array() {
384
-		return $this->elements;
385
-	}
386
-
387
-	/**
388
-	 * {@inheritdoc}
389
-	 *
390
-	 * @param callable $callable Reducer function.
391
-	 *
392
-	 * @param null     $initial  Initial reducer value.
393
-	 *
394
-	 * @return mixed
395
-	 */
396
-	public function reduce( callable $callable, $initial = null ) {
397
-		return array_reduce( $this->elements, $callable, $initial );
398
-	}
399
-
400
-	/**
401
-	 * {@inheritdoc}
402
-	 *
403
-	 * @param callable $condition Condition callback.
404
-	 *
405
-	 * @return bool
406
-	 */
407
-	public function every( callable $condition ) {
408
-		$response = true;
409
-
410
-		foreach ( $this->elements as $element ) {
411
-			$result = call_user_func( $condition, $element );
412
-
413
-			if ( false === $result ) {
414
-				$response = false;
415
-				break;
416
-			}
417
-		}
418
-
419
-		return $response;
420
-	}
421
-
422
-	/**
423
-	 * {@inheritdoc}
424
-	 *
425
-	 * @param  int $num Number of elements to drop.
426
-	 *
427
-	 * @return Collection
428
-	 *
429
-	 * @throws InvalidArgumentException
430
-	 */
431
-	public function drop( $num ) {
432
-		return $this->slice( $num, $this->count() );
433
-	}
434
-
435
-	/**
436
-	 * {@inheritdoc}
437
-	 *
438
-	 * @param int $num Number of elements to drop.
439
-	 *
440
-	 * @return Collection
441
-	 *
442
-	 * @throws InvalidArgumentException
443
-	 */
444
-	public function drop_right( $num ) {
445
-		return $num !== $this->count()
446
-			? $this->slice( 0, $this->count() - $num - 1 )
447
-			: $this->clear();
448
-	}
449
-
450
-	/**
451
-	 * {@inheritdoc}
452
-	 *
453
-	 * @param callable $condition Condition callback.
454
-	 *
455
-	 * @return Collection
456
-	 */
457
-	public function drop_while( callable $condition ) {
458
-		$count = $this->count_while_true( $condition );
459
-		return $count ? $this->drop( $count ) : $this;
460
-	}
461
-	/**
462
-	 * {@inheritdoc}
463
-	 *
464
-	 * @return Collection
465
-	 *
466
-	 * @throws InvalidArgumentException
467
-	 */
468
-	public function tail() {
469
-		return $this->slice( 1, $this->count() );
470
-	}
471
-
472
-	/**
473
-	 * {@inheritdoc}
474
-	 *
475
-	 * @param  int $num Number of elements to take.
476
-	 *
477
-	 * @return Collection
478
-	 *
479
-	 * @throws InvalidArgumentException
480
-	 */
481
-	public function take( $num ) {
482
-		return $this->slice( 0, $num - 1 );
483
-	}
484
-
485
-	/**
486
-	 * {@inheritdoc}
487
-	 *
488
-	 * @param int $num Number of elements to take.
489
-	 *
490
-	 * @return Collection
491
-	 *
492
-	 * @throws InvalidArgumentException
493
-	 */
494
-	public function take_right( $num ) {
495
-		return $this->slice( $this->count() - $num, $this->count() );
496
-	}
497
-
498
-	/**
499
-	 * {@inheritdoc}
500
-	 *
501
-	 * @param callable $condition Callback function.
502
-	 *
503
-	 * @return Collection
504
-	 */
505
-	public function take_while( callable $condition ) {
506
-		$count = $this->count_while_true( $condition );
507
-
508
-		return $count ? $this->take( $count ) : $this->clear();
509
-	}
510
-
511
-	/**
512
-	 * {@inheritdoc}
513
-	 *
514
-	 * @param callable $callable Callback function.
515
-	 */
516
-	public function each( callable $callable ) {
517
-		foreach ( $this->elements as $element ) {
518
-			call_user_func( $callable, $element );
519
-		}
520
-	}
521
-
522
-	/**
523
-	 * {@inheritdoc}
524
-	 *
525
-	 * @param callable $callable Callback function.
526
-	 *
527
-	 * @return Collection
528
-	 */
529
-	public function map( callable $callable ) {
530
-		$elements = array();
531
-		$type = null;
532
-		foreach ( $this->elements as $element ) {
533
-			$result = call_user_func( $callable, $element );
534
-
535
-			if ( null === $type ) {
536
-				$type = gettype( $result );
537
-
538
-				if ( 'object' === $type ) {
539
-					$type = get_class( $result );
540
-				}
541
-			}
542
-
543
-			$elements[] = $result;
544
-		}
545
-
546
-		return $this->new_from_trusted( $elements, $type ? : $this->get_type() );
547
-	}
548
-
549
-	/**
550
-	 * {@inheritdoc}
551
-	 *
552
-	 * @param callable $callable Reducer function.
553
-	 * @param null     $initial  Initial value.
554
-	 *
555
-	 * @return mixed
556
-	 */
557
-	public function reduce_right( callable $callable, $initial = null ) {
558
-		return array_reduce(
559
-			array_reverse( $this->elements ),
560
-			$callable,
561
-			$initial
562
-		);
563
-	}
564
-
565
-	/**
566
-	 * {@inheritdoc}
567
-	 *
568
-	 * @return Collection
569
-	 */
570
-	public function shuffle() {
571
-		$elements = $this->elements;
572
-		shuffle( $elements );
573
-
574
-		return $this->new_from_trusted( $elements );
575
-	}
576
-
577
-	/**
578
-	 * {@inheritdoc}
579
-	 *
580
-	 * @param array|Collection $elements Array of elements to merge.
581
-	 *
582
-	 * @return Collection
583
-	 *
584
-	 * @throws InvalidArgumentException
585
-	 */
586
-	public function merge( $elements ) {
587
-		if ( $elements instanceof static ) {
588
-			$elements = $elements->to_array();
589
-		}
590
-
591
-		if ( ! is_array( $elements ) ) {
592
-			throw new InvalidArgumentException( 'Merge must be given array or Collection' );
593
-		}
594
-
595
-		$this->type->validate_elements( $elements );
596
-
597
-		return $this->new_from_trusted(
598
-			array_merge( $this->elements, $elements )
599
-		);
600
-	}
601
-
602
-	/**
603
-	 * {@inheritdoc}
604
-	 *
605
-	 * @return mixed
606
-	 *
607
-	 * @throws OutOfBoundsException
608
-	 */
609
-	public function first() {
610
-		if ( empty( $this->elements ) ) {
611
-			throw new OutOfBoundsException( 'Cannot get first element of empty Collection' );
612
-		}
613
-
614
-		return reset( $this->elements );
615
-	}
616
-
617
-	/**
618
-	 * {@inheritdoc}
619
-	 *
620
-	 * @return mixed
621
-	 *
622
-	 * @throws OutOfBoundsException
623
-	 */
624
-	public function last() {
625
-		if ( empty( $this->elements ) ) {
626
-			throw new OutOfBoundsException( 'Cannot get last element of empty Collection' );
627
-		}
628
-
629
-		return end( $this->elements );
630
-	}
631
-
632
-	/**
633
-	 * {@inheritdoc}
634
-	 *
635
-	 * @return int
636
-	 */
637
-	public function count() {
638
-		return count( $this->elements );
639
-	}
640
-
641
-	/**
642
-	 * {@inheritDoc}
643
-	 *
644
-	 * @return array
645
-	 */
646
-	public function serialize() {
647
-		return $this->map(function( $element ) {
648
-			if ( $element instanceof Serializes ) {
649
-				return $element->serialize();
650
-			}
651
-
652
-			return $element;
653
-		} )->to_array();
654
-	}
655
-
656
-	/**
657
-	 * Return the current element.
658
-	 *
659
-	 * @return mixed
660
-	 */
661
-	public function current() {
662
-		return $this->at( $this->position );
663
-	}
664
-
665
-	/**
666
-	 * Move forward to next element.
667
-	 */
668
-	public function next() {
669
-		$this->position ++;
670
-	}
671
-
672
-	/**
673
-	 * Return the key of the current element.
674
-	 *
675
-	 * @return mixed
676
-	 */
677
-	public function key() {
678
-		return $this->position;
679
-	}
680
-
681
-	/**
682
-	 * Checks if current position is valid.
683
-	 *
684
-	 * @return bool
685
-	 */
686
-	public function valid() {
687
-		return isset( $this->elements[ $this->position ] );
688
-	}
689
-
690
-	/**
691
-	 * Rewind the Iterator to the first element.
692
-	 */
693
-	public function rewind() {
694
-		$this->position = 0;
695
-	}
696
-
697
-	/**
698
-	 * Creates a new instance of the Collection
699
-	 * from a trusted set of elements.
700
-	 *
701
-	 * @param array      $elements Array of elements to pass into new collection.
702
-	 * @param null|mixed $type
703
-	 *
704
-	 * @return static
705
-	 */
706
-	protected function new_from_trusted( array $elements, $type = null ) {
707
-		$collection = new static( null !== $type ? $type : $this->get_type() );
708
-		$collection->set_from_trusted( $elements );
709
-
710
-		return $collection;
711
-	}
712
-
713
-	/**
714
-	 * Sets the elements without validating them.
715
-	 *
716
-	 * @param array $elements Pre-validated elements to set.
717
-	 */
718
-	protected function set_from_trusted( array $elements ) {
719
-		$this->elements = $elements;
720
-	}
721
-
722
-	/**
723
-	 * Number of elements true for the condition.
724
-	 *
725
-	 * @param callable $condition Condition to check.
726
-	 * @return int
727
-	 */
728
-	protected function count_while_true( callable $condition ) {
729
-		$count = 0;
730
-
731
-		foreach ( $this->elements as $element ) {
732
-			if ( ! $condition($element) ) {
733
-				break;
734
-			}
735
-			$count++;
736
-		}
737
-
738
-		return $count;
739
-	}
740
-
741
-	/**
742
-	 * Validates a number to be used as an index.
743
-	 *
744
-	 * @param  integer $index The number to be validated as an index.
745
-	 *
746
-	 * @throws OutOfRangeException
747
-	 */
748
-	protected function validate_index( $index ) {
749
-		$exists = $this->index_exists( $index );
750
-
751
-		if ( ! $exists ) {
752
-			throw new OutOfRangeException( 'Index out of bounds of collection' );
753
-		}
754
-	}
18
+    /**
19
+     * Collection elements.
20
+     *
21
+     * @var array
22
+     */
23
+    protected $elements = array();
24
+
25
+    /**
26
+     * Collection type to enforce.
27
+     *
28
+     * @var Type
29
+     */
30
+    private $type;
31
+
32
+    /**
33
+     * Where Collection is in loop.
34
+     *
35
+     * @var int
36
+     */
37
+    protected $position = 0;
38
+
39
+    /**
40
+     * Collection constructor.
41
+     *
42
+     * @param string $type
43
+     * @param array  $elements
44
+     */
45
+    public function __construct( $type, array $elements = array() ) {
46
+        $this->type = new Type( $type );
47
+
48
+        if ( $this->type->is_model() ) {
49
+            foreach ( $elements as $idx => $element ) {
50
+                if ( is_array( $element ) ) {
51
+                    $elements[ $idx ] = $this->type->create_model( $element );
52
+                }
53
+            }
54
+        }
55
+
56
+        if ( $elements ) {
57
+            $this->type->validate_elements( $elements );
58
+        }
59
+
60
+        $this->elements = $elements;
61
+    }
62
+
63
+    /**
64
+     * {@inheritdoc}
65
+     *
66
+     * @return string
67
+     */
68
+    public function get_type() {
69
+        return $this->type->get_type();
70
+    }
71
+
72
+    /**
73
+     * {@inheritdoc}
74
+     *
75
+     * @param mixed $element
76
+     *
77
+     * @return Collection
78
+     *
79
+     * @throws InvalidArgumentException
80
+     */
81
+    public function add( $element ) {
82
+        if ( $this->type->is_model() && is_array( $element ) ) {
83
+            $element = $this->type->create_model( $element );
84
+        }
85
+
86
+        $this->type->validate_element( $element );
87
+
88
+        $elements   = $this->elements;
89
+        $elements[] = $element;
90
+
91
+        $collection = new static( $this->get_type() );
92
+        $collection->set_from_trusted( $elements );
93
+
94
+        return $collection;
95
+    }
96
+
97
+    /**
98
+     * {@inheritdoc}
99
+     *
100
+     * @return Collection
101
+     */
102
+    public function clear() {
103
+        return new static( $this->get_type() );
104
+    }
105
+
106
+    /**
107
+     * {@inheritdoc}
108
+     *
109
+     * @param  callable $condition Condition to satisfy.
110
+     *
111
+     * @return bool
112
+     */
113
+    public function contains( callable $condition ) {
114
+        return (bool) $this->find( $condition );
115
+    }
116
+
117
+    /**
118
+     * {@inheritdoc}
119
+     *
120
+     * @param  callable $condition Condition to satisfy.
121
+     *
122
+     * @return mixed
123
+     */
124
+    public function find( callable $condition ) {
125
+        $index = $this->find_index( $condition );
126
+
127
+        return -1 === $index ? false : $this->elements[ $index ];
128
+    }
129
+
130
+    /**
131
+     * {@inheritdoc}
132
+     *
133
+     * @param  callable $condition Condition to satisfy.
134
+     *
135
+     * @return int
136
+     */
137
+    public function find_index( callable $condition ) {
138
+        $index = -1;
139
+
140
+        for ( $i = 0, $count = count( $this->elements ); $i < $count; $i++ ) {
141
+            if ( call_user_func( $condition, ($this->at( $i ) ) ) ) {
142
+                $index = $i;
143
+                break;
144
+            }
145
+        }
146
+
147
+        return $index;
148
+    }
149
+
150
+    /**
151
+     * Fetches the element at the provided index.
152
+     *
153
+     * @param int $index Index to get element from.
154
+     *
155
+     * @return mixed
156
+     *
157
+     * @throws OutOfRangeException
158
+     */
159
+    public function at( $index ) {
160
+        $this->validate_index( $index );
161
+
162
+        return $this->elements[ $index ];
163
+    }
164
+
165
+    /**
166
+     * {@inheritdoc}
167
+     *
168
+     * @param  int $index Index to check for existence.
169
+     *
170
+     * @return bool
171
+     *
172
+     * @throws InvalidArgumentException
173
+     */
174
+    public function index_exists( $index ) {
175
+        if ( ! is_int( $index ) ) {
176
+            throw new InvalidArgumentException( 'Index must be an integer' );
177
+        }
178
+
179
+        if ( $index < 0 ) {
180
+            throw new InvalidArgumentException( 'Index must be a non-negative integer' );
181
+        }
182
+
183
+        return $index < $this->count();
184
+    }
185
+
186
+    /**
187
+     * {@inheritdoc}
188
+     *
189
+     * @param  callable $condition Condition to satisfy.
190
+     *
191
+     * @return mixed
192
+     */
193
+    public function filter( callable $condition ) {
194
+        $elements = array();
195
+
196
+        foreach ( $this->elements as $element ) {
197
+            if ( call_user_func( $condition, $element ) ) {
198
+                $elements[] = $element;
199
+            }
200
+        }
201
+
202
+        return $this->new_from_trusted( $elements );
203
+    }
204
+    /**
205
+     * {@inheritdoc}
206
+     *
207
+     * @param  callable $condition Condition to satisfy.
208
+     *
209
+     * @return mixed
210
+     */
211
+    public function find_last( callable $condition ) {
212
+        $index = $this->find_last_index( $condition );
213
+
214
+        return -1 === $index ? null : $this->elements[ $index ];
215
+    }
216
+
217
+    /**
218
+     * {@inheritdoc}
219
+     *
220
+     * @param  callable $condition
221
+     * @return int
222
+     */
223
+    public function find_last_index( callable $condition ) {
224
+        $index = -1;
225
+
226
+        for ( $i = count( $this->elements ) - 1; $i >= 0; $i-- ) {
227
+            if ( call_user_func( $condition, $this->elements[ $i ] ) ) {
228
+                $index = $i;
229
+                break;
230
+            }
231
+        }
232
+
233
+        return $index;
234
+    }
235
+
236
+    /**
237
+     * {@inheritdoc}
238
+     *
239
+     * @param  int $start Begining index to slice from.
240
+     * @param  int $end   End index to slice to.
241
+     *
242
+     * @return Collection
243
+     *
244
+     * @throws InvalidArgumentException
245
+     */
246
+    public function slice( $start, $end ) {
247
+        if ( $start < 0 || ! is_int( $start ) ) {
248
+            throw new InvalidArgumentException( 'Start must be a non-negative integer' );
249
+        }
250
+
251
+        if ( $end < 0 || ! is_int( $end ) ) {
252
+            throw new InvalidArgumentException( 'End must be a positive integer' );
253
+        }
254
+
255
+        if ( $start > $end ) {
256
+            throw new InvalidArgumentException( 'End must be greater than start' );
257
+        }
258
+
259
+        if ( $end > $this->count() + 1 ) {
260
+            throw new InvalidArgumentException( 'End must be less than the count of the items in the Collection' );
261
+        }
262
+
263
+        $length = $end - $start + 1;
264
+
265
+        return $this->new_from_trusted( array_slice( $this->elements, $start, $length ) );
266
+    }
267
+
268
+    /**
269
+     * {@inheritdoc}
270
+     *
271
+     * @param int   $index     Index to start at.
272
+     * @param mixed $element Element to insert.
273
+     *
274
+     * @return Collection
275
+     *
276
+     * @throws InvalidArgumentException
277
+     * @throws OutOfRangeException
278
+     */
279
+    public function insert( $index, $element ) {
280
+        $this->validate_index( $index );
281
+        $this->type->validate_element( $element );
282
+
283
+        $a = array_slice( $this->elements, 0, $index );
284
+        $b = array_slice( $this->elements, $index, count( $this->elements ) );
285
+
286
+        $a[] = $element;
287
+
288
+        return $this->new_from_trusted( array_merge( $a, $b ) );
289
+    }
290
+
291
+    /**
292
+     * {@inheritdoc}
293
+     *
294
+     * @param int   $index    Index to start insertion at.
295
+     * @param array $elements Elements in insert.
296
+     *
297
+     * @return Collection
298
+     *
299
+     * @throws OutOfRangeException
300
+     */
301
+    public function insert_range( $index, array $elements ) {
302
+        $this->validate_index( $index );
303
+        $this->type->validate_elements( $elements );
304
+
305
+        if ( $index < 0 ) {
306
+            $index = $this->count() + $index + 1;
307
+        }
308
+
309
+        return $this->new_from_trusted(
310
+            array_merge(
311
+                array_slice( $this->elements, 0, $index ),
312
+                $elements,
313
+                array_slice( $this->elements, $index, count( $this->elements ) )
314
+            )
315
+        );
316
+    }
317
+
318
+    /**
319
+     * {@inheritdoc}
320
+     *
321
+     * @param  callable $condition Condition to satisfy.
322
+     *
323
+     * @return Collection
324
+     */
325
+    public function without( callable $condition ) {
326
+        $inverse = function ( $element ) use ( $condition ) {
327
+            return ! call_user_func( $condition, $element );
328
+        };
329
+
330
+        return $this->filter( $inverse );
331
+    }
332
+
333
+    /**
334
+     * {@inheritdoc}
335
+     *
336
+     * @param  int $index Index to remove.
337
+     *
338
+     * @return Collection
339
+     *
340
+     * @throws OutOfRangeException
341
+     */
342
+    public function remove_at( $index ) {
343
+        $this->validate_index( $index );
344
+
345
+        $elements = $this->elements;
346
+
347
+        return $this->new_from_trusted(
348
+            array_merge(
349
+                array_slice( $elements, 0, $index ),
350
+                array_slice( $elements, $index + 1, count( $elements ) )
351
+            )
352
+        );
353
+    }
354
+    /**
355
+     * {@inheritdoc}
356
+     *
357
+     * @return Collection
358
+     */
359
+    public function reverse() {
360
+        return $this->new_from_trusted(
361
+            array_reverse( $this->elements )
362
+        );
363
+    }
364
+
365
+    /**
366
+     * {@inheritdoc}
367
+     *
368
+     * @param callable $callback Sort callback.
369
+     *
370
+     * @return Collection
371
+     */
372
+    public function sort( callable $callback ) {
373
+        $elements = $this->elements;
374
+        usort( $elements, $callback );
375
+        return $this->new_from_trusted( $elements );
376
+    }
377
+
378
+    /**
379
+     * {@inheritdoc}
380
+     *
381
+     * @return array
382
+     */
383
+    public function to_array() {
384
+        return $this->elements;
385
+    }
386
+
387
+    /**
388
+     * {@inheritdoc}
389
+     *
390
+     * @param callable $callable Reducer function.
391
+     *
392
+     * @param null     $initial  Initial reducer value.
393
+     *
394
+     * @return mixed
395
+     */
396
+    public function reduce( callable $callable, $initial = null ) {
397
+        return array_reduce( $this->elements, $callable, $initial );
398
+    }
399
+
400
+    /**
401
+     * {@inheritdoc}
402
+     *
403
+     * @param callable $condition Condition callback.
404
+     *
405
+     * @return bool
406
+     */
407
+    public function every( callable $condition ) {
408
+        $response = true;
409
+
410
+        foreach ( $this->elements as $element ) {
411
+            $result = call_user_func( $condition, $element );
412
+
413
+            if ( false === $result ) {
414
+                $response = false;
415
+                break;
416
+            }
417
+        }
418
+
419
+        return $response;
420
+    }
421
+
422
+    /**
423
+     * {@inheritdoc}
424
+     *
425
+     * @param  int $num Number of elements to drop.
426
+     *
427
+     * @return Collection
428
+     *
429
+     * @throws InvalidArgumentException
430
+     */
431
+    public function drop( $num ) {
432
+        return $this->slice( $num, $this->count() );
433
+    }
434
+
435
+    /**
436
+     * {@inheritdoc}
437
+     *
438
+     * @param int $num Number of elements to drop.
439
+     *
440
+     * @return Collection
441
+     *
442
+     * @throws InvalidArgumentException
443
+     */
444
+    public function drop_right( $num ) {
445
+        return $num !== $this->count()
446
+            ? $this->slice( 0, $this->count() - $num - 1 )
447
+            : $this->clear();
448
+    }
449
+
450
+    /**
451
+     * {@inheritdoc}
452
+     *
453
+     * @param callable $condition Condition callback.
454
+     *
455
+     * @return Collection
456
+     */
457
+    public function drop_while( callable $condition ) {
458
+        $count = $this->count_while_true( $condition );
459
+        return $count ? $this->drop( $count ) : $this;
460
+    }
461
+    /**
462
+     * {@inheritdoc}
463
+     *
464
+     * @return Collection
465
+     *
466
+     * @throws InvalidArgumentException
467
+     */
468
+    public function tail() {
469
+        return $this->slice( 1, $this->count() );
470
+    }
471
+
472
+    /**
473
+     * {@inheritdoc}
474
+     *
475
+     * @param  int $num Number of elements to take.
476
+     *
477
+     * @return Collection
478
+     *
479
+     * @throws InvalidArgumentException
480
+     */
481
+    public function take( $num ) {
482
+        return $this->slice( 0, $num - 1 );
483
+    }
484
+
485
+    /**
486
+     * {@inheritdoc}
487
+     *
488
+     * @param int $num Number of elements to take.
489
+     *
490
+     * @return Collection
491
+     *
492
+     * @throws InvalidArgumentException
493
+     */
494
+    public function take_right( $num ) {
495
+        return $this->slice( $this->count() - $num, $this->count() );
496
+    }
497
+
498
+    /**
499
+     * {@inheritdoc}
500
+     *
501
+     * @param callable $condition Callback function.
502
+     *
503
+     * @return Collection
504
+     */
505
+    public function take_while( callable $condition ) {
506
+        $count = $this->count_while_true( $condition );
507
+
508
+        return $count ? $this->take( $count ) : $this->clear();
509
+    }
510
+
511
+    /**
512
+     * {@inheritdoc}
513
+     *
514
+     * @param callable $callable Callback function.
515
+     */
516
+    public function each( callable $callable ) {
517
+        foreach ( $this->elements as $element ) {
518
+            call_user_func( $callable, $element );
519
+        }
520
+    }
521
+
522
+    /**
523
+     * {@inheritdoc}
524
+     *
525
+     * @param callable $callable Callback function.
526
+     *
527
+     * @return Collection
528
+     */
529
+    public function map( callable $callable ) {
530
+        $elements = array();
531
+        $type = null;
532
+        foreach ( $this->elements as $element ) {
533
+            $result = call_user_func( $callable, $element );
534
+
535
+            if ( null === $type ) {
536
+                $type = gettype( $result );
537
+
538
+                if ( 'object' === $type ) {
539
+                    $type = get_class( $result );
540
+                }
541
+            }
542
+
543
+            $elements[] = $result;
544
+        }
545
+
546
+        return $this->new_from_trusted( $elements, $type ? : $this->get_type() );
547
+    }
548
+
549
+    /**
550
+     * {@inheritdoc}
551
+     *
552
+     * @param callable $callable Reducer function.
553
+     * @param null     $initial  Initial value.
554
+     *
555
+     * @return mixed
556
+     */
557
+    public function reduce_right( callable $callable, $initial = null ) {
558
+        return array_reduce(
559
+            array_reverse( $this->elements ),
560
+            $callable,
561
+            $initial
562
+        );
563
+    }
564
+
565
+    /**
566
+     * {@inheritdoc}
567
+     *
568
+     * @return Collection
569
+     */
570
+    public function shuffle() {
571
+        $elements = $this->elements;
572
+        shuffle( $elements );
573
+
574
+        return $this->new_from_trusted( $elements );
575
+    }
576
+
577
+    /**
578
+     * {@inheritdoc}
579
+     *
580
+     * @param array|Collection $elements Array of elements to merge.
581
+     *
582
+     * @return Collection
583
+     *
584
+     * @throws InvalidArgumentException
585
+     */
586
+    public function merge( $elements ) {
587
+        if ( $elements instanceof static ) {
588
+            $elements = $elements->to_array();
589
+        }
590
+
591
+        if ( ! is_array( $elements ) ) {
592
+            throw new InvalidArgumentException( 'Merge must be given array or Collection' );
593
+        }
594
+
595
+        $this->type->validate_elements( $elements );
596
+
597
+        return $this->new_from_trusted(
598
+            array_merge( $this->elements, $elements )
599
+        );
600
+    }
601
+
602
+    /**
603
+     * {@inheritdoc}
604
+     *
605
+     * @return mixed
606
+     *
607
+     * @throws OutOfBoundsException
608
+     */
609
+    public function first() {
610
+        if ( empty( $this->elements ) ) {
611
+            throw new OutOfBoundsException( 'Cannot get first element of empty Collection' );
612
+        }
613
+
614
+        return reset( $this->elements );
615
+    }
616
+
617
+    /**
618
+     * {@inheritdoc}
619
+     *
620
+     * @return mixed
621
+     *
622
+     * @throws OutOfBoundsException
623
+     */
624
+    public function last() {
625
+        if ( empty( $this->elements ) ) {
626
+            throw new OutOfBoundsException( 'Cannot get last element of empty Collection' );
627
+        }
628
+
629
+        return end( $this->elements );
630
+    }
631
+
632
+    /**
633
+     * {@inheritdoc}
634
+     *
635
+     * @return int
636
+     */
637
+    public function count() {
638
+        return count( $this->elements );
639
+    }
640
+
641
+    /**
642
+     * {@inheritDoc}
643
+     *
644
+     * @return array
645
+     */
646
+    public function serialize() {
647
+        return $this->map(function( $element ) {
648
+            if ( $element instanceof Serializes ) {
649
+                return $element->serialize();
650
+            }
651
+
652
+            return $element;
653
+        } )->to_array();
654
+    }
655
+
656
+    /**
657
+     * Return the current element.
658
+     *
659
+     * @return mixed
660
+     */
661
+    public function current() {
662
+        return $this->at( $this->position );
663
+    }
664
+
665
+    /**
666
+     * Move forward to next element.
667
+     */
668
+    public function next() {
669
+        $this->position ++;
670
+    }
671
+
672
+    /**
673
+     * Return the key of the current element.
674
+     *
675
+     * @return mixed
676
+     */
677
+    public function key() {
678
+        return $this->position;
679
+    }
680
+
681
+    /**
682
+     * Checks if current position is valid.
683
+     *
684
+     * @return bool
685
+     */
686
+    public function valid() {
687
+        return isset( $this->elements[ $this->position ] );
688
+    }
689
+
690
+    /**
691
+     * Rewind the Iterator to the first element.
692
+     */
693
+    public function rewind() {
694
+        $this->position = 0;
695
+    }
696
+
697
+    /**
698
+     * Creates a new instance of the Collection
699
+     * from a trusted set of elements.
700
+     *
701
+     * @param array      $elements Array of elements to pass into new collection.
702
+     * @param null|mixed $type
703
+     *
704
+     * @return static
705
+     */
706
+    protected function new_from_trusted( array $elements, $type = null ) {
707
+        $collection = new static( null !== $type ? $type : $this->get_type() );
708
+        $collection->set_from_trusted( $elements );
709
+
710
+        return $collection;
711
+    }
712
+
713
+    /**
714
+     * Sets the elements without validating them.
715
+     *
716
+     * @param array $elements Pre-validated elements to set.
717
+     */
718
+    protected function set_from_trusted( array $elements ) {
719
+        $this->elements = $elements;
720
+    }
721
+
722
+    /**
723
+     * Number of elements true for the condition.
724
+     *
725
+     * @param callable $condition Condition to check.
726
+     * @return int
727
+     */
728
+    protected function count_while_true( callable $condition ) {
729
+        $count = 0;
730
+
731
+        foreach ( $this->elements as $element ) {
732
+            if ( ! $condition($element) ) {
733
+                break;
734
+            }
735
+            $count++;
736
+        }
737
+
738
+        return $count;
739
+    }
740
+
741
+    /**
742
+     * Validates a number to be used as an index.
743
+     *
744
+     * @param  integer $index The number to be validated as an index.
745
+     *
746
+     * @throws OutOfRangeException
747
+     */
748
+    protected function validate_index( $index ) {
749
+        $exists = $this->index_exists( $index );
750
+
751
+        if ( ! $exists ) {
752
+            throw new OutOfRangeException( 'Index out of bounds of collection' );
753
+        }
754
+    }
755 755
 }
Please login to merge, or discard this patch.