Yii CGridview сортировка с помощью CSqlDataProvider
Я использую CSqlDataProvider для создания CGridview Я не могу использовать CActiveRecord, потому что набор результатов огромен и выбрасывает ошибки памяти. Столбцы должны быть сортируемыми. Как мне этого добиться?
Образец sql
$orders_query_raw = 'select o.order_id, o.customer_name, o.customer_email, o.customer_advertiser, o.payment_method, o.created, o.last_updated, o.currency, o.currency_value, o.status, o.blinking, s.name, ot.text order_total, o.customer_id, op.product_id, o.phonebooking
from `order` o, `order_total` ot, `order_status` s , order_product op
where o.order_id = op.order_id and o.status = s.order_status_id and ot.order_id = o.order_id and s.language_id = '1' and ot.class = 'ot_total' group by o.order_id'
SQL DataProvider
$dataProvider = new CSqlDataProvider($orders_query_raw, array(
'totalItemCount'=>$count, // get from a count query
'pagination'=>array(
'pageSize'=>50,
),
));
И gridview
$this->widget('zii.widgets.grid.CGridView', array(
'dataProvider' => $dataProvider,
'id'=>'order-grid',
'columns' => array(
array(
'header'=>'Order ID',
'value'=>array($this, 'gridOrderId'),
'type'=>'raw',
),
array(
'header'=>Yii::t('order', 'Customers'),
'name'=>'customer.fullName',
'value'=>'CHtml::link($data[\'customer_name\'], \'mailto:\'.$data[\'customer_email\'])',
'type'=>'raw',
),
array(
'header'=>Yii::t('order', 'Order total'),
'value'=>'strip_tags($data[\'order_total\'])',
'type'=>'raw',
'htmlOptions'=>array(
'style'=>'text-align:right;',
),
),
array(
'header' => Yii::t('order', 'Date Purchased'),
'name' => 'created',
),
array(
'header'=> Yii::t('order', 'Last modify date'),
'value'=>array($this, 'gridLastModified'),
),
array(
'header' => Yii::t('order', 'Status changed by'),
'value' => array($this, 'gridLastModifiedUserFirstName'),
),
array(
'header' => Yii::t('provider', 'Provider\'s code'),
'value' => array($this, 'gridProviderCode'),
'type' => 'raw',
'htmlOptions'=>array(
'class'=>'nobr',
),
),
array(
'header' => Yii::t('order', 'Follow up'),
'value' => array($this, 'gridFollowUp'),
'type' => 'raw',
),
array(
'header' => Yii::t('order', 'Order status'),
'value' => '$data[\'name\']',
),
array(
'class'=>'CButtonColumn',
'template'=>'{update}',
'header'=>'Action',
'buttons'=>array(
'update'=>array(
'url'=>'Yii::app()->createUrl(\'order/update\', array(\'order_id\'=>$data[\'order_id\']))',
),
),
),
),
));
Спасибо
2 ответа
Чтобы включить сортировку (нажав на заголовок столбца) в виде сетки с поставщиком данных как CSqlDataProvider
Вам понадобятся как минимум 2 вещи:
- Должны определить
CSort
объект для поставщика данных, сattributes
это было бы сортируемо. - Должны определить
name
столбца,но только в том случае, если вы указываетеcolumns
свойство grid-view, в противном случае, если свойство columns оставлено пустым, любые атрибуты, упомянутые в объекте CSort, будут сортируемыми.
Тем не менее, другой ответ должен работать в тех случаях, когда sql прост и приходит из 1 таблицы, но в вашем случае, когда sql немного сложнее, т.е. данные поступают из нескольких таблиц, решение будет немного меняться.
В таких случаях вам придется учитывать конфликтующие имена столбцов (если они есть) и правильную спецификацию CSort. attributes
массив.
Примеры:
Нет конфликтующих имен столбцов ни в одной из таблиц (так же, как в другом ответе):
$dataProvider=new CSqlDataProvider($sql, array( 'totalItemCount'=>$count, 'sort'=>array( 'attributes'=>array( 'order_id, order_total' // csv of sortable column names ) ) ));
Тогда в вашей сетке:
array( 'header'=>Yii::t('order', 'Order total'), 'name'=>'order_total',// to make header clickable to sort 'value'=>'strip_tags($data[\'order_total\'])', 'type'=>'raw', 'htmlOptions'=>array( 'style'=>'text-align:right;', ), ),
Конфликтующие имена столбцов:
- Во-первых, дайте всем конфликтующим именам псевдонимы в вашем sql.
Во-вторых, укажите эти псевдонимы как сортируемые
attributes
вCSort
объект:'attributes'=>array( 'some_alias, some_other_alias' )
Укажите
name
для столбца вcolumns
:array( 'header'=>'Foo', 'name'=>'some_alias', 'value'=>'$data[\'some_alias\']' // this is actually redundant in this // case, because the name will itself pick up the value, and we don't // need to specify value explicitly if we are not applying any function to it )
Обратите внимание, что сортировка по вызову URL-адреса включается только указанием объекта сортировки, нет необходимости использовать name
, если вы не хотите, чтобы отсортировать заголовки.
Попробуйте как ниже
$sort = new CSort();
$sort->defaultOrder = 'order_id'; // for initial order
$sort->attributes = array(
'created'
);
$dataProvider = new CSqlDataProvider($orders_query_raw, array(
'totalItemCount'=>$count, // get from a count query
'pagination'=>array(
'pageSize'=>50,
),
'sort'=>$sort
));