私は、ユーザーの作品を取得するために作業しているプロジェクトでピボットテーブルを使用しています。
例:User::find(1)->works
は、IDが1のユーザーの作品を提供します。
問題は、この結果を追加のピボットデータでフィルター処理することです。
何かのようなもの:
User::find(1)->works->pivot->where('active',1)->get();
ここで、アクティブはuser_worksピボットテーブルに設定した列です。
これは、User.phpモデルの関連部分です。
<?php
class User extends Cartalyst\Sentry\Users\Eloquent\User {
public function works() {
return $this->belongsToMany('Work','user_works')->withPivot('active')->withTimestamps();
}
}
これは、Work.phpモデルの関連部分です。
<?php
class Work extends Eloquent {
public function users() {
return $this->belongsToMany('User','user_works')->withPivot('active')->withTimestamps();
}
}
これは私のピボットテーブルスキーマです。
<?php
use Illuminate\Database\Migrations\Migration;
use Illuminate\Database\Schema\Blueprint;
class CreateUserWorksTable extends Migration {
/**
* Run the migrations.
*
* @return void
*/
public function up()
{
Schema::create('user_works', function(Blueprint $table) {
$table->increments('id');
$table->integer('user_id')->unsigned()->default(0);
$table->integer('work_id')->unsigned()->default(0);
$table->enum('active',array('0','1'))->default(1);
$table->foreign('user_id')->references('id')->on('users')->onDelete('cascade');
$table->foreign('work_id')->references('id')->on('works')->onDelete('cascade');
$table->timestamps();
});
}
/**
* Reverse the migrations.
*
* @return void
*/
public function down()
{
Schema::drop('user_works');
}
}
ピボットテーブルの新しいモデルを作成せずにデータを取得する方法はありますか?
前もって感謝します、
編集:この方法でフィルタリングできます:
return User::find(1)->works()->where('user_works.active','=','1')->get();
生のテーブル名を入力する必要がありました。しかし、使用せずにこれを取得するより良い方法はありますか?
Laravel 4.1はネイティブのwherePivot
とorWherePivot
methods をもたらします。これは私の問題の直接的な解決策です。
withPivot('foo')
を呼び出すたびに、Laravelを実行します:
SELECT ... `table`.`foo` AS `pivot_foo` FROM `table` ...
特にMySQLでは、HAVING
句、GROUP BY
句、およびORDER BY
句で列エイリアスを使用できますが、WHERE
句では使用できません。
HAVING
とWHERE
は両方ともクエリのフィルタリングに使用されますが、動作が少し異なります。GROUP BY
とHAVING
の前にWHERE
が適用されます。
一般的なSQLルールとして、列のエイリアス(その場合はpivot_foo
)をグループ化、フィルター処理などに使用しないでください。他のSQLデータベースでは機能しない場合があります。
推奨されていませんが、以下を使用することができます。
return User::find(1)->works()->having('pivot_active','=','1')->get();
$user->works()
などの動的プロパティを使用できるため、両方向のすべての関係を設定しようとします。
class Collection extends Eloquent {
public function contents()
{
return $this->belongsToMany('Content', 'collection_content', 'collection_id', 'content_id')->withPivot('collection_id', 'group_id', 'field_identifier');
}
}
class Content extends Eloquent {
public function collections()
{
return $this->belongsToMany('Collection', 'collection_content', 'collection_id', 'content_id')->withPivot('collection_id', 'group_id', 'field_identifier');
}
}
class CollectionContent extends Eloquent {
public function content()
{
return $this->belongsTo('Content');
}
public function collection()
{
return $this->belongsTo('Collection');
}
}
次にクエリ:
$works = User::find(1)->works()->where('active', 1)->get();
Eloquentのドキュメントは、ピボットテーブルの使用に関してはひどいものです。これは素晴らしいチュートリアルです: http://www.developed.be/2013/08/30/laravel-4-pivot-table-example-attach-and-detach/