web-dev-qa-db-ja.com

Eloquentを使用してピボットテーブルをフィルタリングする方法

私は、ユーザーの作品を取得するために作業しているプロジェクトでピボットテーブルを使用しています。

例: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();

生のテーブル名を入力する必要がありました。しかし、使用せずにこれを取得するより良い方法はありますか?

27
Arda

Laravel 4.1はネイティブのwherePivotorWherePivotmethods をもたらします。これは私の問題の直接的な解決策です。

40
Arda

withPivot('foo')を呼び出すたびに、Laravelを実行します:

SELECT ... `table`.`foo` AS `pivot_foo` FROM `table` ...

修正回答:

特にMySQLでは、HAVING句、GROUP BY句、およびORDER BY句で列エイリアスを使用できますが、WHERE句では使用できません。

HAVINGWHEREは両方ともクエリのフィルタリングに使用されますが、動作が少し異なります。GROUP BYHAVINGの前にWHEREが適用されます。

一般的なSQLルールとして、列のエイリアス(その場合はpivot_foo)をグループ化、フィルター処理などに使用しないでください。他のSQLデータベースでは機能しない場合があります。

推奨されていませんが、以下を使用することができます。

return User::find(1)->works()->having('pivot_active','=','1')->get();
3
vFragosop

$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/

2
omar j