Одиночная пагинация на нескольких моделях в cakephp

Мое требование состоит в том, чтобы искать на 2 разных таблицах: Ветеринары и Клиники. Между ними может быть связь, то есть результат, если в выборочных клиниках указано "а", а у ветеринаров - "а". Ветеринары могут быть связаны с клиниками или нет. В настоящее время я делаю следующее. Есть ли способ, чтобы избежать выполнения 2 запросов, которые также могут помочь мне использовать помощника по разбивке страниц по торту?

$this->paginate = array(
        'Vet' => array(
            'conditions' => $conditions,
            'fields' => array('Vet.id', 'Vet.name', 'Vet.professionnal_address', 'phone_number', 'Vet.email', 'Vet.type', 'Vet.latitude', 'Vet.longitude','Vet.city','Vet.clinic_id','Vet.zipcode'),
            'joins' => array( 
                array(
                    'table' => 'vet_appointment_types',
                    'alias' => 'VetAppointmentType',
                    'type' => 'LEFT',
                    'conditions' => array(
                        'Vet.id = VetAppointmentType.vet_id',
                    )
                )
            ),
            'limit' => $limit, 
            'group' => array(
                'Vet.id'
            ), 
            'order' => array(
                'Vet.name' => 'ASC'
            )
        ),
        'Clinic' => array(
            'conditions' => $conditions1,
            'fields' => array('Clinic.*'),
            'limit' => $limit, 
            'order' => array(
                'Clinic.name' => 'ASC'
            )
        )
    );

$results = $this->paginate('Vet');
$results2 = $this->paginate('Clinic');

Я попытался расширить компонент разбиения на страницы по умолчанию в модели без базы данных и использовать объединение, но структуру базы данных немного сложнее, поэтому нельзя использовать объединение. Кроме того, я думаю, что реализация временной модели на основе таблиц была бы вариантом, но так как она будет использоваться для поиска, поэтому, как именно ее реализовать, я не могу думать. Любая помощь будет +1 добавлена;)

1 ответ

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

// \app\Model\Search.php
App::uses('ConnectionManager', 'Model');
App::uses('AppModel', 'Model');
App::uses('CakeLog', 'Log');

class Search extends AppModel {

    public function __construct() {
        parent::__construct();

        $connection = ConnectionManager::getDataSource('default');
        $return = $connection->execute("CREATE OR REPLACE VIEW `search` AS
            SELECT Vet.id, (CONCAT( `Vet`.`fname` , ' ', `Vet`.`lname` )) AS name, 'vet' AS TYPE , Vet.latitude, Vet.longitude, Vet.zipcode, Vet.speciality FROM `db`.`vets` AS `Vet` 
            UNION 
            SELECT Clinic.id, Clinic.name, 'clinic' AS TYPE , Clinic.lat, Clinic.long, Clinic.zipcode, Clinic.address FROM `db`.`clinics` AS `Clinic`");
        $return = ($return == true)?'Search View created successfully on '.date('Y-m-d H:i:s'):$return;
        CakeLog::write('search', $return);
    }

    public $useTable = 'search';// This model does not use a database table
    public $primaryKey = 'id'; // Define primary key
    public $useDbConfig = 'default'; // Define db

}

Это модель. Каждый пользователь ищет что-нибудь, когда модель загружается в контроллер, создается / заменяется представление, чтобы можно было получить обновленные значения. Может использоваться как любая другая модель в cakephp. Он также поддерживает виртуальные поля. Для построения запроса для представления базы данных вы также можете использовать построитель запросов, который я использовал следующим образом.

$joins = array(
        array(
            'table' => 'vet_appointment_types',
            'alias' => 'VetAppointmentType',
            'type' => 'LEFT',
            'conditions' => array(
                'Vet.id = VetAppointmentType.vet_id',
            )
        )
    );

    $dbo = $this->Vet->getDataSource();
    $subQuery = $dbo->buildStatement(
        array(
            'fields' => array('Vet.id', 'Vet.name'),
            'table' => $dbo->fullTableName($this->Vet),
            'alias' => 'Vet',
            'limit' => $limit, 
            'group' => array(
                'Vet.id'
            ), 
            'order' => array(
                'Vet.name' => 'ASC'
            ),
            'offset' => null,
            'joins' => $joins,
            'conditions' => $conditions
        ),
        $this->Vet
    );
    $query = $subQuery;

    $query .= ' UNION ';

    $dbo = $this->Clinic->getDataSource();
    $subQuery = $dbo->buildStatement(
        array(
            'fields' => array('Clinic.id', 'Clinic.name'),
            'table' => $dbo->fullTableName($this->Clinic),
            'alias' => 'Clinic',
            'limit' => $limit,
            'conditions' => $conditions1,
            'limit' => $limit, 
            'order' => array(
                'Clinic.name' => 'ASC'
            ),
            'offset' => null
        ),
        $this->Clinic
    );

    $query .= $subQuery;

    print_r($query);
Другие вопросы по тегам