Как применить фильтр к реляционной таблице с помощью инструкции with

#php #laravel #filter #laravel-query-builder

#php #laravel #Фильтр #laravel-конструктор запросов

Вопрос:

Я написал код для фильтрации с использованием реляционных данных, но я не получаю хорошего результата. Вот код. Родительский запрос и с запросом работают независимо.

 public function get(Request $request, Company $company, Survey $survey)
{
    $this->authorize('update', $survey->company);
    $search = $request->get('search');
    $query = EsAnswer::where(['company_id' => $company->id, 'survey_id' => $survey->id])
        ->orderByDesc('created_at')
        ->with(['employee'=> function ($q) use($search) {
            $q->where('first_name', 'LIKE', "%{$search}%");
            $q->orwhere('last_name', 'LIKE', "%{$search}%");
            $q->select('id', 'first_name', 'last_name');   
        }]);
    if ($request->has('start_date')) {
        $query->whereDate('created_at', '>=', $request->start_date);
    }
    if ($request->has('end_date')) {
        $query->whereDate('created_at', '<=', $request->end_date);
    }

    $answers = $query->get()->groupBy('submission_id');
        // ->paginate('100');
    return $answers;
}
  

когда я использовал это

 {
    "search":""
}
  

Я получаю этот ответ:

 {
    "id": 2,
    "company_id": 1,
    "employee_id": 1,
    "survey_id": 2,
    "es_question_id": 1,
    "answer": "done",
    "submission_id": "1",
    "mark_as_read": 1,
    "created_at": "2020-08-19 12:18:25",
    "updated_at": "2020-08-26 06:55:21",
    "employee": {
        "id": 1,
        "first_name": "Baseapp",
        "last_name": "saw"
    }
}
  

когда я попытался выполнить поиск по имени сотрудника, которое присутствует в другой таблице, если данные присутствуют, то данные должны быть отфильтрованы и выдать ответ, но если имя сотрудника или данные поиска отсутствуют, то данные не должны давать никакого ответа. Как с этим бороться?

 {
    "search":"sure"
}


{
    "id": 2,
    "company_id": 1,
    "employee_id": 1,
    "survey_id": 2,
    "es_question_id": 1,
    "answer": "done",
    "submission_id": "1",
    "mark_as_read": 1,
    "created_at": "2020-08-19 12:18:25",
    "updated_at": "2020-08-26 06:55:21",
    "employee": null
}
  

здесь данные равны нулю, но почему отображаются родительские данные. пожалуйста, помогите и дайте другую идею для фильтрации с помощью реляционных данных.

Ответ №1:

мы используем whereHas для фильтрации в связанной таблице. так что используйте это как

 $query = EsAnswer::with('employee:id,first_name,last_name')
        ->where(['company_id' => $company->id, 'survey_id' => $survey->id])
        ->whereHas('employee', function($query) use($search) {
            $query->where('first_name', 'LIKE', "%{$search}%")
                ->orwhere('last_name', 'LIKE', "%{$search}%");
        })
        ->orderByDesc('created_at');
  

это приведет к поиску в вашей связанной таблице и вернет null, если ничего не найдено. и я добавил элегантный способ быстрой загрузки данных. вы также должны проверить это.

Ответ №2:

он использует whereHas, затем с :

 $query = EsAnswer::where(['company_id' => $company->id, 'survey_id' => $survey->id])
    ->orderByDesc('created_at')
    ->whereHas('employee', function ($q) use($search) {
      return $q->where('first_name', 'LIKE', "%{$search}%")
      ->orwhere('last_name', 'LIKE', "%{$search}%");   
    })->with(['employee' => function ($q){
        $q->select('id', 'first_name', 'last_name');
    }]);
  

Ответ №3:

используйте whereHas() функцию

   $query = EsAnswer::with('employee:id,first_name,last_name')
        ->where(['company_id' => $company->id, 'survey_id' => $survey->id])
        ->whereHas('employee', function($query) use($search) {
            $query->where('first_name', 'LIKE', "%$search%")
                ->orwhere('last_name', 'LIKE', "%$search%");
        })
        ->orderByDesc('created_at');
  

ссылка на ссылкуhttps://laravel.com/docs/7.x/eloquent-relationships#querying-relationship-existence

Комментарии:

1. Ответ бесполезен, оба они одинаковы. Я хочу, чтобы, если я хочу выполнить поиск по имени сотрудника, данные отношения должны фильтроваться, иначе результат будет нулевым. но здесь только запрос фильтрует модель employee.