3

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 = 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\']))',
                ),
            ),
        ),
    ),
));

ありがとう

4

2 に答える 2

6

データプロバイダーをとしてグリッドビューで(列のヘッダーをクリックして)並べ替えを有効にするにはCSqlDataProvider、少なくとも2つのものが必要です。

  1. CSortデータプロバイダーのオブジェクトを定義する必要があります。attributesこれは並べ替え可能です。
  2. name列の を定義する必要がありますが、グリッドビューのプロパティを指定する場合に限りcolumnsます。そうでない場合、columnsプロパティを空白のままにすると、CSortオブジェクトに記載されている属性はすべて並べ替え可能になります。

とはいえ、SQLが単純で、1つのテーブルからのものである場合、もう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;',
        ),
    ),
    
  • 競合する列名:

    1. まず、SQLで競合するすべての名前のエイリアスを指定します。
    2. 次に、これらのエイリアスをオブジェクトのソート可能として指定attributesします。CSort

      'attributes'=>array(
          'some_alias, some_other_alias'
      )
      
    3. 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場合を除いて、を使用する必要はありません。

于 2012-10-24T10:33:11.133 に答える
1

以下のようにしてみてください

    $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
));
于 2012-10-24T07:45:53.267 に答える