Skip to content

Commit 83b872e

Browse files
Document withAttributes (#10148)
* Document `withAttributes` * formatting * formatting --------- Co-authored-by: Taylor Otwell <[email protected]>
1 parent 490a498 commit 83b872e

File tree

2 files changed

+85
-0
lines changed

2 files changed

+85
-0
lines changed

eloquent-relationships.md

+53
Original file line numberDiff line numberDiff line change
@@ -8,6 +8,7 @@
88
- [Has One of Many](#has-one-of-many)
99
- [Has One Through](#has-one-through)
1010
- [Has Many Through](#has-many-through)
11+
- [Scoped Relationships](#scoped-relationships)
1112
- [Many to Many Relationships](#many-to-many)
1213
- [Retrieving Intermediate Table Columns](#retrieving-intermediate-table-columns)
1314
- [Filtering Queries via Intermediate Table Columns](#filtering-queries-via-intermediate-table-columns)
@@ -612,6 +613,53 @@ return $this->through('environments')->has('deployments');
612613
return $this->throughEnvironments()->hasDeployments();
613614
```
614615

616+
<a name="scoped-relationships"></a>
617+
### Scoped Relationships
618+
619+
It's common to add additional methods to models that constrain relationships. For example, you might add a `featuredPosts` method to a `User` model which constrains the broader `posts` relationship with an additional `where` constraint:
620+
621+
<?php
622+
623+
namespace App\Models;
624+
625+
use Illuminate\Database\Eloquent\Model;
626+
use Illuminate\Database\Eloquent\Relations\HasMany;
627+
628+
class User extends Model
629+
{
630+
/**
631+
* Get the user's posts.
632+
*/
633+
public function posts(): HasMany
634+
{
635+
return $this->hasMany(Post::class)->latest();
636+
}
637+
638+
/**
639+
* Get the user's featured posts.
640+
*/
641+
public function featuredPosts(): HasMany
642+
{
643+
return $this->posts()->where('featured', true);
644+
}
645+
}
646+
647+
However, if you attempt to create a model via the `featuredPosts` method, its `featured` attribute would not be set to `true`. If you would like to create models via relationship methods and also specify attributes that should be added to all models created via that relationship, you may use the `withAttributes` method when building the relationship query:
648+
649+
/**
650+
* Get the user's featured posts.
651+
*/
652+
public function featuredPosts(): HasMany
653+
{
654+
return $this->posts()->withAttributes(['featured' => true]);
655+
}
656+
657+
The `withAttributes` method will add `where` clause constraints to the query using the given attributes, and it will also add the given attributes to any models created via the relationship method:
658+
659+
$post = $user->featuredPosts()->create(['title' => 'Featured Post']);
660+
661+
$post->featured; // true
662+
615663
<a name="many-to-many"></a>
616664
## Many to Many Relationships
617665

@@ -781,6 +829,11 @@ You can also filter the results returned by `belongsToMany` relationship queries
781829
->as('subscriptions')
782830
->wherePivotNotNull('expired_at');
783831

832+
The `wherePivot` adds a where clause constraint to the query, but does not add the specified value when creating new models via the defined relationship. If you need to both query and create relationships with a particular pivot value, you may use the `withPivotValue` method:
833+
834+
return $this->belongsToMany(Role::class)
835+
->withPivotValue('approved', 1);
836+
784837
<a name="ordering-queries-via-intermediate-table-columns"></a>
785838
### Ordering Queries via Intermediate Table Columns
786839

eloquent.md

+32
Original file line numberDiff line numberDiff line change
@@ -32,6 +32,7 @@
3232
- [Query Scopes](#query-scopes)
3333
- [Global Scopes](#global-scopes)
3434
- [Local Scopes](#local-scopes)
35+
- [Pending Attributes](#pending-attributes)
3536
- [Comparing Models](#comparing-models)
3637
- [Events](#events)
3738
- [Using Closures](#events-using-closures)
@@ -1390,6 +1391,37 @@ Once the expected arguments have been added to your scope method's signature, yo
13901391

13911392
$users = User::ofType('admin')->get();
13921393

1394+
<a name="pending-attributes"></a>
1395+
### Pending Attributes
1396+
1397+
If you would like to use scopes to create models that have the same attributes as those used to constrain the scope, you may use the `withAttributes` method when building the scope query:
1398+
1399+
<?php
1400+
1401+
namespace App\Models;
1402+
1403+
use Illuminate\Database\Eloquent\Builder;
1404+
use Illuminate\Database\Eloquent\Model;
1405+
1406+
class Post extends Model
1407+
{
1408+
/**
1409+
* Scope the query to only include drafts.
1410+
*/
1411+
public function scopeDraft(Builder $query): void
1412+
{
1413+
$query->withAttributes([
1414+
'hidden' => true,
1415+
]);
1416+
}
1417+
}
1418+
1419+
The `withAttributes` method will add `where` clause constraints to the query using the given attributes, and it will also add the given attributes to any models created via the scope:
1420+
1421+
$draft = Post::draft()->create(['title' => 'In Progress']);
1422+
1423+
$draft->hidden; // true
1424+
13931425
<a name="comparing-models"></a>
13941426
## Comparing Models
13951427

0 commit comments

Comments
 (0)