You signed in with another tab or window. Reload to refresh your session.You signed out in another tab or window. Reload to refresh your session.You switched accounts on another tab or window. Reload to refresh your session.Dismiss alert
Copy file name to clipboardexpand all lines: docs/as-a-resource/lazy-properties.md
+144-13
Original file line number
Diff line number
Diff line change
@@ -19,7 +19,8 @@ class AlbumData extends Data
19
19
}
20
20
```
21
21
22
-
This will always output a collection of songs, which can become quite large. With lazy properties, we can include properties when we want to:
22
+
This will always output a collection of songs, which can become quite large. With lazy properties, we can include
23
+
properties when we want to:
23
24
24
25
```php
25
26
class AlbumData extends Data
@@ -43,7 +44,8 @@ class AlbumData extends Data
43
44
}
44
45
```
45
46
46
-
The `songs` key won't be included in the resource when transforming it from a model. Because the closure that provides the data won't be called when transforming the data object unless we explicitly demand it.
47
+
The `songs` key won't be included in the resource when transforming it from a model. Because the closure that provides
48
+
the data won't be called when transforming the data object unless we explicitly demand it.
The property will only be included when the `is_admin` property of the data object is true. It is not possible to include the property later on with the `include` method when a condition is not accepted.
161
+
The property will only be included when the `is_admin` property of the data object is true. It is not possible to
162
+
include the property later on with the `include` method when a condition is not accepted.
157
163
158
164
### Relational Lazy properties
159
165
@@ -173,15 +179,138 @@ It is possible to mark a lazy property as included by default:
The property will now always be included when the data object is transformed. You can explicitly exclude properties that were default included as such:
182
+
The property will now always be included when the data object is transformed. You can explicitly exclude properties that
The package knows how to get the property from the model and wrap it into a data object, but since we're using a lazy
216
+
property, we need to write our own magic creation method with a lot of repetitive code.
217
+
218
+
In such a situation auto lazy might be a good fit, instead of casting the property directly into the data object, the
219
+
casting process is wrapped in a lazy Closure.
220
+
221
+
This makes it possible to rewrite the example as such:
222
+
223
+
```php
224
+
#[AutoLazy]
225
+
class UserData extends Data
226
+
{
227
+
public function __construct(
228
+
public string $title,
229
+
public Lazy|SongData $favorite_song,
230
+
) {
231
+
}
232
+
}
233
+
```
234
+
235
+
While achieving the same result!
236
+
237
+
Auto Lazy wraps the casting process of a value for every property typed as `Lazy` into a Lazy Closure when the
238
+
`AutoLazy` attribute is present on the class.
239
+
240
+
It is also possible to use the `AutoLazy` attribute on a property level:
241
+
242
+
```php
243
+
class UserData extends Data
244
+
{
245
+
public function __construct(
246
+
public string $title,
247
+
#[AutoLazy]
248
+
public Lazy|SongData $favorite_song,
249
+
) {
250
+
}
251
+
}
252
+
```
253
+
254
+
The auto lazy process won't be applied in the following situations:
255
+
256
+
- When a null value is passed to the property
257
+
- When the property value isn't present in the input payload and the property typed as `Optional`
258
+
- When a Lazy Closure is passed to the property
259
+
260
+
### Auto lazy with model relations
261
+
262
+
When you're constructing a data object from an Eloquent model, it is also possible to automatically create lazy
263
+
properties for model relations which are only resolved when the relation is loaded:
264
+
265
+
```php
266
+
class UserData extends Data
267
+
{
268
+
public function __construct(
269
+
public string $title,
270
+
#[AutoWhenLoadedLazy]
271
+
public Lazy|SongData $favoriteSong,
272
+
) {
273
+
}
274
+
}
275
+
```
276
+
277
+
When the `favoriteSong` relation is loaded on the model, the property will be included in the data object.
278
+
279
+
If the name of the relation doesn't match the property name, you can specify the relation name:
280
+
281
+
```php
282
+
class UserData extends Data
283
+
{
284
+
public function __construct(
285
+
public string $title,
286
+
#[AutoWhenLoadedLazy('favoriteSong')]
287
+
public Lazy|SongData $favorite_song,
288
+
) {
289
+
}
290
+
}
291
+
```
292
+
293
+
The package will use the regular casting process when the relation is loaded, so it is also perfectly possible to create a collection of data objects:
294
+
295
+
```php
296
+
class UserData extends Data
297
+
{
298
+
/**
299
+
* @param Lazy|array<int,SongData> $favoriteSongs
300
+
*/
301
+
public function __construct(
302
+
public string $title,
303
+
#[AutoWhenLoadedLazy]
304
+
public Lazy|array $favoriteSongs,
305
+
) {
306
+
}
307
+
}
308
+
```
309
+
182
310
## Only and Except
183
311
184
-
Lazy properties are great for reducing payloads sent over the wire. However, when you completely want to remove a property Laravel's `only` and `except` methods can be used:
312
+
Lazy properties are great for reducing payloads sent over the wire. However, when you completely want to remove a
313
+
property Laravel's `only` and `except` methods can be used:
185
314
186
315
```php
187
316
AlbumData::from(Album::first())->only('songs'); // will only show `songs`
Only and except always take precedence over include and exclude, which means that when a property is hidden by `only` or `except` it is impossible to show it again using `include`.
334
+
Only and except always take precedence over include and exclude, which means that when a property is hidden by `only` or
335
+
`except` it is impossible to show it again using `include`.
206
336
207
337
### Conditionally
208
338
@@ -306,7 +436,7 @@ Our JSON would look like this when we request `https://spatie.be/my-account`:
@@ -395,7 +525,8 @@ AlbumData::from(Album::first())->include('songs')->toArray(); // will include so
395
525
AlbumData::from(Album::first())->toArray(); // will not include songs
396
526
```
397
527
398
-
If you want to add includes/excludes/only/except to a data object and its nested chain that will be used for all future transformations, you can define them in their respective *properties methods:
528
+
If you want to add includes/excludes/only/except to a data object and its nested chain that will be used for all future
529
+
transformations, you can define them in their respective *properties methods:
0 commit comments