1: <?php declare(strict_types=1);
2:
3: namespace Salient\Contract\Collection;
4:
5: use Salient\Contract\Core\Arrayable;
6: use Salient\Contract\Core\Jsonable;
7: use ArrayAccess;
8: use Countable;
9: use JsonSerializable;
10: use OutOfRangeException;
11: use Traversable;
12:
13: /**
14: * @api
15: *
16: * @template TKey of array-key
17: * @template TValue
18: *
19: * @extends ArrayAccess<TKey,TValue>
20: * @extends Traversable<TKey,TValue>
21: * @extends Arrayable<TKey,TValue|mixed[]>
22: */
23: interface CollectionInterface extends
24: ArrayAccess,
25: Countable,
26: JsonSerializable,
27: Traversable,
28: Arrayable,
29: Jsonable
30: {
31: /**
32: * Pass the value of each item to the callback
33: */
34: public const CALLBACK_USE_VALUE = 1;
35:
36: /**
37: * Pass the key of each item to the callback
38: */
39: public const CALLBACK_USE_KEY = 2;
40:
41: /**
42: * Pass a key-value pair to the callback for each item
43: */
44: public const CALLBACK_USE_BOTH = 3;
45:
46: /**
47: * Return the first item that satisfies the callback
48: */
49: public const FIND_VALUE = 8;
50:
51: /**
52: * Return the key of the first item that satisfies the callback
53: */
54: public const FIND_KEY = 16;
55:
56: /**
57: * @param Arrayable<TKey,TValue>|iterable<TKey,TValue> $items
58: */
59: public function __construct($items = []);
60:
61: /**
62: * Check if the collection is empty
63: *
64: * @phpstan-assert-if-false non-empty-array<TKey,TValue> $this->all()
65: * @phpstan-assert-if-false !null $this->first()
66: * @phpstan-assert-if-false !null $this->last()
67: */
68: public function isEmpty(): bool;
69:
70: /**
71: * Add or replace an item with a given key
72: *
73: * @param TKey $key
74: * @param TValue $value
75: * @return static
76: */
77: public function set($key, $value);
78:
79: /**
80: * Check if an item with a given key exists
81: *
82: * @param TKey $key
83: * @phpstan-assert-if-true non-empty-array<TKey,TValue> $this->all()
84: * @phpstan-assert-if-true !null $this->first()
85: * @phpstan-assert-if-true !null $this->last()
86: */
87: public function has($key): bool;
88:
89: /**
90: * Get the item with the given key
91: *
92: * @param TKey $key
93: * @return TValue
94: * @throws OutOfRangeException if there is no such item in the collection.
95: */
96: public function get($key);
97:
98: /**
99: * Remove an item with a given key
100: *
101: * @param TKey $key
102: * @return static
103: */
104: public function unset($key);
105:
106: /**
107: * Add an item
108: *
109: * @param TValue $value
110: * @return static<TKey|int,TValue>
111: */
112: public function add($value);
113:
114: /**
115: * Merge the collection with the given items
116: *
117: * @param Arrayable<TKey,TValue>|iterable<TKey,TValue> $items
118: * @return static
119: */
120: public function merge($items);
121:
122: /**
123: * Sort items in the collection
124: *
125: * @return static
126: */
127: public function sort();
128:
129: /**
130: * Reverse the order of items in the collection
131: *
132: * @return static
133: */
134: public function reverse();
135:
136: /**
137: * Pass each item in the collection to a callback
138: *
139: * The callback's return values are discarded.
140: *
141: * @template TMode of int-mask-of<CollectionInterface::*>
142: *
143: * @param (callable(TValue, TValue|null $next, TValue|null $prev): mixed)|(callable(TKey, TKey|null $next, TKey|null $prev): mixed)|(callable(array{TKey,TValue}, array{TKey,TValue}|null $next, array{TKey,TValue}|null $prev): mixed) $callback
144: * @phpstan-param (
145: * TMode is 3|11|19
146: * ? (callable(array{TKey,TValue}, array{TKey,TValue}|null $next, array{TKey,TValue}|null $prev): mixed)
147: * : (TMode is 2|10|18
148: * ? (callable(TKey, TKey|null $next, TKey|null $prev): mixed)
149: * : (callable(TValue, TValue|null $next, TValue|null $prev): mixed)
150: * )
151: * ) $callback
152: * @param TMode $mode
153: * @return $this
154: */
155: public function forEach(callable $callback, int $mode = CollectionInterface::CALLBACK_USE_VALUE);
156:
157: /**
158: * Pass each item in the collection to a callback and populate a new
159: * collection with its return values
160: *
161: * @template TMode of int-mask-of<CollectionInterface::*>
162: * @template TReturn of TValue
163: *
164: * @param (callable(TValue, TValue|null $next, TValue|null $prev): TReturn)|(callable(TKey, TKey|null $next, TKey|null $prev): TReturn)|(callable(array{TKey,TValue}, array{TKey,TValue}|null $next, array{TKey,TValue}|null $prev): TReturn) $callback
165: * @phpstan-param (
166: * TMode is 3|11|19
167: * ? (callable(array{TKey,TValue}, array{TKey,TValue}|null $next, array{TKey,TValue}|null $prev): TReturn)
168: * : (TMode is 2|10|18
169: * ? (callable(TKey, TKey|null $next, TKey|null $prev): TReturn)
170: * : (callable(TValue, TValue|null $next, TValue|null $prev): TReturn)
171: * )
172: * ) $callback
173: * @param TMode $mode
174: * @return static<TKey,TReturn>
175: */
176: public function map(callable $callback, int $mode = CollectionInterface::CALLBACK_USE_VALUE);
177:
178: /**
179: * Reduce the collection to items that satisfy a callback
180: *
181: * @template TMode of int-mask-of<CollectionInterface::*>
182: *
183: * @param (callable(TValue, TValue|null $next, TValue|null $prev): bool)|(callable(TKey, TKey|null $next, TKey|null $prev): bool)|(callable(array{TKey,TValue}, array{TKey,TValue}|null $next, array{TKey,TValue}|null $prev): bool) $callback
184: * @phpstan-param (
185: * TMode is 3|11|19
186: * ? (callable(array{TKey,TValue}, array{TKey,TValue}|null $next, array{TKey,TValue}|null $prev): bool)
187: * : (TMode is 2|10|18
188: * ? (callable(TKey, TKey|null $next, TKey|null $prev): bool)
189: * : (callable(TValue, TValue|null $next, TValue|null $prev): bool)
190: * )
191: * ) $callback
192: * @param TMode $mode
193: * @return static
194: */
195: public function filter(callable $callback, int $mode = CollectionInterface::CALLBACK_USE_VALUE);
196:
197: /**
198: * Get the first item that satisfies a callback, or null if there is no such
199: * item in the collection
200: *
201: * @template TMode of int-mask-of<CollectionInterface::*>
202: *
203: * @param (callable(TValue, TValue|null $next, TValue|null $prev): bool)|(callable(TKey, TKey|null $next, TKey|null $prev): bool)|(callable(array{TKey,TValue}, array{TKey,TValue}|null $next, array{TKey,TValue}|null $prev): bool) $callback
204: * @phpstan-param (
205: * TMode is 3|11|19
206: * ? (callable(array{TKey,TValue}, array{TKey,TValue}|null $next, array{TKey,TValue}|null $prev): bool)
207: * : (TMode is 2|10|18
208: * ? (callable(TKey, TKey|null $next, TKey|null $prev): bool)
209: * : (callable(TValue, TValue|null $next, TValue|null $prev): bool)
210: * )
211: * ) $callback
212: * @param TMode $mode
213: * @return (TMode is 16|17|18|19 ? TKey : TValue)|null
214: */
215: public function find(callable $callback, int $mode = CollectionInterface::CALLBACK_USE_VALUE | CollectionInterface::FIND_VALUE);
216:
217: /**
218: * Reduce the collection to items with keys in an array
219: *
220: * @param TKey[] $keys
221: * @return static
222: */
223: public function only(array $keys);
224:
225: /**
226: * Reduce the collection to items with keys in an index
227: *
228: * @param array<TKey,true> $index
229: * @return static
230: */
231: public function onlyIn(array $index);
232:
233: /**
234: * Reduce the collection to items with keys not in an array
235: *
236: * @param TKey[] $keys
237: * @return static
238: */
239: public function except(array $keys);
240:
241: /**
242: * Reduce the collection to items with keys not in an index
243: *
244: * @param array<TKey,true> $index
245: * @return static
246: */
247: public function exceptIn(array $index);
248:
249: /**
250: * Extract a slice of the collection
251: *
252: * @return static
253: */
254: public function slice(int $offset, ?int $length = null);
255:
256: /**
257: * Check if a value is in the collection
258: *
259: * @param TValue $value
260: * @phpstan-assert-if-true non-empty-array<TKey,TValue> $this->all()
261: * @phpstan-assert-if-true !null $this->first()
262: * @phpstan-assert-if-true !null $this->last()
263: */
264: public function hasValue($value, bool $strict = false): bool;
265:
266: /**
267: * Get the first key at which a value is found, or null if it's not in the
268: * collection
269: *
270: * @param TValue $value
271: * @return TKey|null
272: */
273: public function keyOf($value, bool $strict = false);
274:
275: /**
276: * Get the first item equal but not necessarily identical to a value, or
277: * null if it's not in the collection
278: *
279: * @param TValue $value
280: * @return TValue|null
281: */
282: public function firstOf($value);
283:
284: /**
285: * Get all items in the collection
286: *
287: * @return array<TKey,TValue>
288: */
289: public function all(): array;
290:
291: /**
292: * Get the first item, or null if the collection is empty
293: *
294: * @return TValue|null
295: */
296: public function first();
297:
298: /**
299: * Get the last item, or null if the collection is empty
300: *
301: * @return TValue|null
302: */
303: public function last();
304:
305: /**
306: * Get the nth item (1-based), or null if there is no such item in the
307: * collection
308: *
309: * If `$n` is negative, the nth item from the end of the collection is
310: * returned.
311: *
312: * @return TValue|null
313: */
314: public function nth(int $n);
315:
316: /**
317: * Push items onto the end of the collection
318: *
319: * @param TValue ...$items
320: * @return static<TKey|int,TValue>
321: */
322: public function push(...$items);
323:
324: /**
325: * Pop an item off the end of the collection
326: *
327: * @param TValue|null $last Receives the value removed from the collection,
328: * or `null` if the collection is empty.
329: * @return static
330: */
331: public function pop(&$last = null);
332:
333: /**
334: * Shift an item off the beginning of the collection
335: *
336: * @param TValue|null $first Receives the value removed from the collection,
337: * or `null` if the collection is empty.
338: * @return static
339: */
340: public function shift(&$first = null);
341:
342: /**
343: * Add items to the beginning of the collection
344: *
345: * Items are added in one operation and stay in the given order.
346: *
347: * @param TValue ...$items
348: * @return static<TKey|int,TValue>
349: */
350: public function unshift(...$items);
351: }
352: