アイテムリソースとコレクションリソースに異なるデータを定義できるかどうか疑問に思っていました。
コレクションについては、['id', 'title', 'slug']
ただし、アイテムリソースには追加の詳細が含まれます['id', 'title', 'slug', 'user', etc.]
次のようなものを実現したい
class PageResource extends Resource
{
/**
* Transform the resource into an array.
*
* @param \Illuminate\Http\Request
* @return array
*/
public function toArray($request)
{
return [
'id' => $this->id,
'title' => $this->title,
'slug' => $this->slug,
'user' => [
'id' => $this->user->id,
'name' => $this->user->name,
'email' => $this->user->email,
],
];
}
}
class PageResourceCollection extends ResourceCollection
{
/**
* Transform the resource collection into an array.
*
* @param \Illuminate\Http\Request
* @return array
*/
public function toArray($request)
{
return [
'id' => $this->id,
'title' => $this->title,
'slug' => $this->slug,
];
}
}
PageResourceCollectionはPageResourceを使用するため、期待どおりに機能しません。
return [
'data' => $this->collection,
];
リソースをPageFullResource
/PageListResource
とPageFullResourceCollection
/PageListResourceCollection
に複製できますが、同じ結果を達成するためのより良い方法を見つけようとしています。
Resourceクラスにはコレクションメソッドがあります。 ResourceCollectionへのパラメーター入力としてそれを返し、コレクションで変換を指定できます。
コントローラ:
class PageController extends Controller
{
public function index()
{
return new PageResourceCollection(PageResource::collection(Page::all()));
}
public function show(Page $page)
{
return new PageResource($page);
}
}
リソース:
class PageResource extends Resource
{
public function toArray($request)
{
return [
'id' => $this->id,
'title' => $this->title,
'slug' => $this->slug,
'user' => [
'id' => $this->user->id,
'name' => $this->user->name,
'email' => $this->user->email,
],
];
}
}
class PageResourceCollection extends ResourceCollection
{
public function toArray($request)
{
return [
'data' => $this->collection->transform(function($page){
return [
'id' => $page->id,
'title' => $page->title,
'slug' => $page->slug,
];
}),
];
}
}
リソースとコレクションで応答フィールドの値を同じにする場合は、コレクション内でリソースを再利用できます
PersonResource.php
<?php
namespace App\Http\Resources;
use Illuminate\Http\Resources\Json\Resource;
class PersonResource extends Resource
{
/**
* Transform the resource into an array.
*
* @param \Illuminate\Http\Request $request
* @return array
*/
public function toArray($request)
{
// return parent::toArray($request);
return [
'id' => $this->id,
'person_type' => $this->person_type,
'first_name' => $this->first_name,
'last_name' => $this->last_name,
'created_at' => (string) $this->created_at,
'updated_at' => (string) $this->updated_at,
];
}
}
PersonCollection.php
<?php
namespace App\Http\Resources;
use Illuminate\Http\Resources\Json\ResourceCollection;
class PersonCollection extends ResourceCollection
{
/**
* Transform the resource collection into an array.
*
* @param \Illuminate\Http\Request $request
* @return \Illuminate\Http\Resources\Json\AnonymousResourceCollection
*/
public function toArray($request)
{
// return parent::toArray($request);
return PersonResource::collection($this->collection);
}
}
リンクとメタデータの使用に興味がない場合は、受け入れられた答えが機能します。必要に応じて、単に以下を返します。
return new PageResourceCollection(Page::paginate(10));
コントローラーで。また、リソースコレクションに渡す前に、他の依存関係を積極的にロードする必要があります。