Ответ 1
В Laravel 4.1 приведены методы wherePivot
и orWherePivot
, что является прямым решением моей проблемы.
Я использую сводную таблицу в проекте, над которым я работаю, чтобы работать с пользователями.
Например: User::find(1)->works
дает мне работы пользователя с идентификатором 1.
Дело в том, что я хочу отфильтровать эти результаты с дополнительными данными Pivot.
Что-то вроде:
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');
}
}
Есть ли способ получить данные без создания новой модели для сводной таблицы?
Спасибо заранее,
Edit: Я могу фильтровать этот путь:
return User::find(1)->works()->where('user_works.active','=','1')->get();
Мне пришлось набирать имя таблицы raw. Но есть ли лучший способ получить это, не используя его?
В Laravel 4.1 приведены методы wherePivot
и orWherePivot
, что является прямым решением моей проблемы.
Всякий раз, когда вы вызываете withPivot('foo')
, Laravel вы делаете:
SELECT ... `table`.`foo` AS `pivot_foo` FROM `table` ...
MySQL, в частности, позволяет использовать псевдонимы столбцов в разделах HAVING
, GROUP BY
и ORDER BY
, но не в предложениях WHERE
.
Оба параметра HAVING
и WHERE
используются для фильтрации запросов, но они ведут себя немного иначе: HAVING
применяется после GROUP BY
и 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 = Пользователь:: find (1) → works() → где ('active', 1) → get();
Яркая документация ужасна, когда речь заходит о использовании сводных таблиц. Это отличный учебник: http://www.developed.be/2013/08/30/laravel-4-pivot-table-example-attach-and-detach/