Инфраструктура Yii: использование данных из связанных моделей Active Record для поиска - PullRequest
3 голосов
/ 27 января 2012

Yii 1.1 Разработка приложений Поваренная книга объясняет метод использования данных из связанных моделей Active Record для поиска соответствующих моделей. Этот метод описан на страницах № 193 и 194. Я пытался интегрировать этот метод в свое приложение, но он не работает. Может кто-нибудь объяснить мне, доступна ли эта функция в Yii Framework версии 1.1.8

В этом месте я также могу найти комментарии для поиска моделей активных записей, связанных с формой данных. Но это тоже не работает. http://www.yiiframework.com/doc/api/1.1/CDbCriteria

У меня есть таблица заказов и таблица пользователей

Таблица заказов и таблица пользователей имеют отношение один ко многим.

У пользователя много заказов, и у заказа ровно один пользователь.

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

Таблица заказов имеет следующие отношения

public function relations()
{
    // NOTE: you may need to adjust the relation name and the related
    // class name for the relations automatically generated below.
    return array(
        'comments' => array(self::HAS_MANY, 'Comment', 'order_id'),
        'user' => array(self::BELONGS_TO, 'User', 'user_id'),
        'orderstatus' => array(self::BELONGS_TO, 'Orderstatus', 'orderstatus_id'),
        'commentCount' => array(self::STAT, 'Comment' , 'order_id')
    );
}

Это условия поиска / фильтрации с включенным именем таблицы пользователя

public function search()
    {
        // Warning: Please modify the following code to remove attributes that
        // should not be searched.

        $criteria=new CDbCriteria;

        $criteria->compare('id',$this->id);
        $criteria->compare('order_create_date',$this->order_create_date,true);
        $criteria->compare('price',$this->price,true);
        $criteria->compare('bank_account_number',$this->bank_account_number,true);
        $criteria->compare('hardwaredetail_Id',$this->hardwaredetail_Id);
        $criteria->compare('user_id',$this->user_id);
        $criteria->compare('order_update_date',$this->order_update_date,true);
        $criteria->compare('is_received',$this->is_received);
        $criteria->compare('order_received_date',$this->order_received_date,true);
        $criteria->compare('is_notify_by_email',$this->is_notify_by_email);
        $criteria->compare('warehouse_offered_price',$this->warehouse_offered_price,true);
        $criteria->compare('warehouse_offered_price_date',$this->warehouse_offered_price_date,true);
        $criteria->compare('orderstatus_id',$this->orderstatus_id);
        $criteria->together = true; 
        $criteria->with = array('user');
        $criteria->compare('user.name',$this->user,true);
        //$criteria->compare('user.name',$this->user->name);
        return new CActiveDataProvider($this, array(
            'criteria'=>$criteria,
        ));
    }

и страница администрирования заказа редактируется для отображения имени, поданного следующим образом

<?php $this->widget('zii.widgets.grid.CGridView', array(
    'id'=>'order-grid',
    'dataProvider'=>$model->search(),
    //'filter'=>$model,
    'columns'=>array(
        'id',
        'order_create_date',
        'price',
        'bank_account_number',
        array(
            'name'=>'user',
            'value'=>'$data->user->name'
        ),
       ),
));

Возвращено сообщение об ошибке

enter image description here

После решения проблемы неоднозначности столбца id путем применения решения, которое дал thaddeusmt, я столкнулся со следующим сообщением об ошибке.

enter image description here

Заранее спасибо за любую помощь

Ответы [ 2 ]

11 голосов
/ 27 января 2012

Я мог найти ответ.Это все, что я сделал.

Ниже приведены две таблицы, которые у меня есть.Заказ и пользователь

enter image description here

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

Это начальный вид страницы поиска.

enter image description here

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

enter image description here

, так что это шаги, которые я сделал.

сначала в модели заказа, у меня есть следующие отношения

    public function relations()
        {
            // NOTE: you may need to adjust the relation name and the related
            // class name for the relations automatically generated below.
            return array(

                'user' => array(self::BELONGS_TO, 'User', 'user_id'),


            );
        }
This is generated by Yii framework , i did nothing :)

Then , i changed the search method as follows

public function search()
    {
        // Warning: Please modify the following code to remove attributes that
        // should not be searched.

        $criteria=new CDbCriteria;


        $criteria->compare('t.order_create_date',$this->order_create_date,true);
        $criteria->compare('t.price',$this->price,true);
        $criteria->compare('t.bank_account_number',$this->bank_account_number,true);
        $criteria->compare('t.hardwaredetail_Id',$this->hardwaredetail_Id);
        //$criteria->compare('user_id',$this->user_id);

        $criteria->compare('t.order_update_date',$this->order_update_date,true);
        $criteria->compare('t.is_received',$this->is_received);
        $criteria->compare('t.order_received_date',$this->order_received_date,true);
        $criteria->compare('t.is_notify_by_email',$this->is_notify_by_email);
        $criteria->compare('t.warehouse_offered_price',$this->warehouse_offered_price,true);
        $criteria->compare('t.warehouse_offered_price_date',$this->warehouse_offered_price_date,true);
        $criteria->compare('t.orderstatus_id',$this->orderstatus_id);
        $criteria->together = true; 
        $criteria->compare('t.id',$this->id,true);
        $criteria->with = array('user');
        $criteria->compare('name',$this->user,true,"OR");
        return new CActiveDataProvider($this, array(
            'criteria'=>$criteria,
        ));
    }

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

Другое дело, порядок элементов

$ criterial-> togeter = true;должен предшествовать реляционным элементам.

Затем вы обновили метод правил в таблице Order.я добавил пользователя и имя файла в безопасные атрибуты.

public function rules()
    {
        // NOTE: you should only define rules for those attributes that
        // will receive user inputs.
        return array(
            //array(' orderstatus_id', 'required'),
            array('hardwaredetail_Id, user_id, is_received, is_notify_by_email, orderstatus_id', 'numerical', 'integerOnly'=>true),
            array('price, warehouse_offered_price', 'length', 'max'=>10),
            array('bank_account_number', 'length', 'max'=>100),
            array('order_create_date, order_update_date, order_received_date, warehouse_offered_price_date, user,name', 'safe'),
            // The following rule is used by search().
            // Please remove those attributes that should not be searched.
            array('id, order_create_date, price, bank_account_number, hardwaredetail_Id, user_id, order_update_date, is_received, order_received_date, is_notify_by_email, warehouse_offered_price, warehouse_offered_price_date, orderstatus_id', 'safe', 'on'=>'search'),
        );
    }

Наконец-то обновите свой код интерфейса.

<?php $this->widget('zii.widgets.grid.CGridView', array(
    'id'=>'order-grid',
    'dataProvider'=>$model->search(),
    'filter'=>$model,
    'columns'=>array(
        'id',
        'order_create_date',
        'price',
        'bank_account_number',
        array(
            'name'=>'user',
            'value'=>'$data->user->name'
        )
)); ?>

Я обновил администратор заказа с помощью

array(
                'name'=>'user',
                'value'=>'$data->user->name'
            )

Это то, что я сделал, и это сработало для меня.спросите меня, если вам нужна помощь.Спасибо всем, кто интересуется этой проблемой.

2 голосов
/ 27 января 2012

Похоже, что и user, и order имеют столбцы с именем id. И ваши критерии используют их оба в предложении WHERE, которое выдает «неоднозначное» сообщение об ошибке mySql.

При использовании критерия with (который выполняет SQL JOIN для таблиц), если в двух ваших таблицах есть столбцы с одинаковыми именами, вам необходимо использовать префикс mySql "точка" в таких условиях, например:

$criteria->compare('t.id',$this->id); // the default table prefix in Yii is "t"

Когда я присоединяюсь к таблице, используя $criteria->with, я начинаю с префикса всех имен столбцов (в моих критериях compare и condition и т. Д.), Например:

$criteria->compare('t.id',$this->id); // t
$criteria->compare('t.order_create_date',$this->order_create_date,true); // t
$criteria->with = array('user');
$criteria->compare('user.name',$this->user_id,true); // user (the filter will set user_id)

Ваше сеточное представление должно выглядеть так:

array(
  'name'=>'user_id',
  'header'=>'User',
  'sortable'=>false,
  'value'=>'$data->user->name'
),

Кроме того, я думаю, что есть более серьезная проблема, как вы указали в своем редактировании:

Ваша поисковая функция настроена так: user.name',$this->user - но $this->user будет возвращать текущий пользовательский объект через отношение, а не критерии поиска. Фильтр столбца установит свойство user_id, свойство.

РЕДАКТИРОВАТЬ : Нет, вы можете $this->user в качестве имени столбца, если вы установите его в качестве атрибута safe.

Способ, которым я обхожу это, показан более подробно здесь:

Поиск в столбце модели BELONGS_TO с CGridView, Yii

Сортировка не будет работать, хотя - только фильтрация.

Вот хороший пост на форуме Yii, который также может дать вам больше подсказок:

http://www.yiiframework.com/forum/index.php?/topic/9083-search-filter-of-a-relations-field-through-cgridview/

К сожалению, сортировка и фильтрация CGridViews по отношениям на самом деле не являются функциональностью по умолчанию. Вы можете легко отобразить связанную информацию с именем столбца, например user.name, но она не будет сортироваться или фильтроваться. Удачи!

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...