relatedsearchbehavior Behavior making searching relations in CGridView easier

  1. Requirements
  2. Forum
  3. Usage
  4. History
  5. Resources

Creating CGridViews with related tables or getting fields from related tables is simplified with this behavior. It magically adds the needed 'with' clauses and provides aliases to fields of records in the relations.

Live demo

Uses the KeenActiveDataProvider extension to limit the number of requests to the database.

Requirements

Developped with Yii 1.1.12 - surely useable with earlier and later versions.

Forum

A Forum thread is available for this extension.

Usage

To use this extension you must do 5 steps:

  1. "Install" the RelatedSearchBehavior;
  2. Update the Model's search() method;
  3. Add/Update the 'behaviors()' method;
  4. Add the search fields to the Model rules as safe fields;
  5. Use the search field as any other field in the CGridView.

Here are the details for these steps:

1. "Install" the RelatedSearchBehavior;

Installation is as easy as adding the location to the 'import' configuration of the Yii configuration file (main.php):

return array(
         [...]
	'import'=>array(
            [...]
            'ext.KeenActiveDataProvider', // Alias to right location.
	    'ext.RelatedSearchBehavior',  // Alias to right location.
	),

In the above the RelatedSearchBehavior file was placed in the extensions directory.

2. Update the Model's search() method;

Update the overload of ActiveRecord::search(). The following example shows how you can set the default search order. You can set any parameter accepted by CActiveDataProvider (such as pagination) in the second array. So, replace code like this:

return new CActiveDataProvider($this, array(
  'criteria'=>$criteria,
  'sort'=>array(
    'defaultOrder'=>'title ASC',
  )
));

by code like this (I prefer using intermediate variables like '$sort' to improve readability and increase flexibility):

$sort=array(
    'defaultOrder'=>'title ASC',
);
return $this->relatedSearch(
    $criteria,
    array('sort'=>$sort)
);

In the above example, 'title' can be one of the related fields that you defined.

3. Add/Update the 'behaviors()' method

Attach the behavior to the CActiveRecord and specifies the related fields.

function behaviors() {
    return array(
        'relatedsearch'=>array(
             'class'=>'RelatedSearchBehavior',
             'relations'=>array(
                  'serial'=>'device.device_identifier',
                  'location'=>'device.location.description',
                   // Field where search value is different($this->deviceid)
                 'fieldwithdifferentsearchvalue'=>array(
                     'field'=>'device.displayname',
                     'searchvalue'=>'deviceid'
                  ),
                 // Next line describes a field we do not search,
                 // but we define it here for convienience
                 'mylocalreference'=>'field.very.far.away.in.the.relation.tree',
             ),
         ),
    );
}
4. Add the search fields to the Model rules as safe fields;
public function rules()
	{
	    return array(
	        [...]
			array('serial,location,deviceid','safe','on'=>'search'),
		);
	}
5. Use the search field as any other field in the CGridView.

For the CGridView column specification, you can then just put 'serial' for the column (no need to do 'name'=>..., 'filter'=>..., 'value'=>... .

Example:

$this->widget('zii.widgets.grid.CGridView', array(
  [...]
  'columns'=>array(
      [...]
      'serial',
      'location',
  )
));
6. To use "autoScope"

Autoscope allows you to search a field using a scope without declaring the scope yourself.

For instance, you can use this:

MyModel::model()->location('Belgium')->findAll();

Before, you have to:

  1. Add RelatedSearchBehavior to the behaviors of your CActiveRecord (already done in the preceding steps,
  2. Add the following generic code to your CActiveRecord Model.
/**
	 * Add automatic scopes for attributes (uses RelatedSearchBehavior).
	 */
	public function __call($name,$parameters) {
	    try {
	        return parent::__call($name,$parameters);
	    } catch (CException $e) {
	        if(preg_match(
	                '/'.Yii::t(
	                        'yii',
	                        quotemeta(
	                                Yii::t(
	                                        'yii',
	                                        '{class} and its behaviors do not have a method or closure named "{name}".'
	                                        )
	                                ),
	                                array('{class}'=>'.*','{name}'=>'.*')
	                        )
	                .'/',$e->getMessage())) {
	            return $this->autoScope($name, $parameters);
	        } else {
	            throw $e;
	        }
	    }
	}

You are allowed to provide all the regular compare parameters:

MyModel::model()->location($searchvalue,$partialMatch,$operator,$escape)->findAll();

defaults are the same as compare: autoscope(,false,"AND",false).

This is usefull in complex nested conditions, not so much for simple searches like the above.

7. Using relations in CSort's attributes for sorting.

'CSort' allows you to specify 'virtual attributes' for sorting as mentioned in the Yii documentation. Without RelatedSearchBehavior, you must make sure that you include the relations used in the search condition. With RelatedSearchBehavior, you do not need to take care about that - the extension takes care about it for you (since 1.16).

$sort=array(
    'defaultOrder'=>'title ASC',
    'attributes'=>
	    array(
		    'price'=>array(
			    'asc'=>'item.price',
			    'desc'=>'item.price DESC',
			    'label'=>'Item Price'
		    ),
	    ),
    );
return $this->relatedSearch(
    $criteria,
    array('sort'=>$sort)
);

The preferred approach is that you'ld use attributes defined for RelatedSearchBehavior, but this might be usefull in combined sort conditions:

$sort=array(
    'defaultOrder'=>'title ASC',
    'attributes'=>
	    array(
		    'groupprice'=>array(
			    'asc'=>'item.group, item.price',
			    'desc'=>'item.group DESC, item.price DESC',
			    'label'=>'Item Price'
		    ),
	    ),
    );
return $this->relatedSearch(
    $criteria,
    array('sort'=>$sort)
);
8. Tips & notes
  • If you like RelatedSearchBehavior, you can create or update your Gii template to generate it automatically.
  • If you use 'ERememberFiltersBehavior', you must set the rememberScenario before getting the dataProvider - otherwise the relations will not be resolved in the sort clause. So you write:
$model->rememberScenario="admin"; // Must be before ->search().
$dataProvider=$model->search(); // Uses RelatedSearchBehavior

History

 * 1.03  Quoting relations in database.
 * 1.04  Added autoScope.
 *       Added option 'partialMatch' for relation.
 * 1.05  Enable multiple attributes in default sort.
 * 1.06  Fix to autoScope - return owner (chaining) + correct example in comment.
 * 1.07  Improved compatibility with ERememberFiltersBehavior ("standard" sort key).
 * 1.08  Fix in KeenActiveDataProvider for postgresql + updates to demo for postgresql.
 * 1.09  Allow array as search value.
 * 1.10  Use alias defined in model's relation (Svobik7)
 * 1.11  Corrected test of owner class type (lower case 'c') and improved error message 
 * 1.12  Autoscope for relations and 'addRelatedCondition' method as a complement to 'addCondition'.
 * 1.13  Handle 'getter' in autoscope call.
 * 1.14  Look recursively for relations for autoscope.
 * 1.15  Added 'getDataProvider'.
 * 1.16  Added relations used in sort "attributes" provided as a parameter.
 * 1.17  Improved error messages. Fix for relations that are defined through option array.
 *       Fix for local fields (aliases/virtual attributes with modified search field)
 * 1.18  Renamed 'addRelationCondition' in 'addRelatedCompare'.
 * 1.19  Reactivated 'together'.

Resources

10 0
26 followers
2 305 downloads
Yii Version: 1.1
License: MIT
Category: Database
Developed by: le_top
Created on: Feb 3, 2013
Last updated: 9 years ago

Downloads

show all

Related Extensions