<?php /** * CStarRating class file. * * @author Qiang Xue <qiang.xue@gmail.com> * @link http://www.yiiframework.com/ * @copyright 2008-2013 Yii Software LLC * @license http://www.yiiframework.com/license/ */ /** * CStarRating displays a star rating control that can collect user rating input. * * CStarRating is based on {@link http://www.fyneworks.com/jquery/star-rating/ jQuery Star Rating Plugin}. * It displays a list of stars indicating the rating values. Users can toggle these stars * to indicate their rating input. On the server side, when the rating input is submitted, * the value can be retrieved in the same way as working with a normal HTML input. * For example, using * <pre> * $this->widget('CStarRating',array('name'=>'rating')); * </pre> * we can retrieve the rating value via <code>$_POST['rating']</code>. * * CStarRating allows customization of its appearance. It also supports empty rating as well as read-only rating. * * @author Qiang Xue <qiang.xue@gmail.com> * @package system.web.widgets * @since 1.0 */ class CStarRating extends CInputWidget { /** * @var integer the number of stars. Defaults to 5. */ public $starCount=5; /** * @var mixed the minimum rating allowed. This can be either an integer or a float value. Defaults to 1. */ public $minRating=1; /** * @var mixed the maximum rating allowed. This can be either an integer or a float value. Defaults to 10. */ public $maxRating=10; /** * @var mixed the step size of rating. This is the minimum difference between two rating values. Defaults to 1. */ public $ratingStepSize=1; /** * @var mixed the CSS file used for the widget. Defaults to null, meaning * using the default CSS file included together with the widget. * If false, no CSS file will be used. Otherwise, the specified CSS file * will be included when using this widget. */ public $cssFile; /** * @var array the titles associated with the rating options. The keys are ratings and the values are the corresponding titles. * Defaults to null, meaning using the rating value as the title. */ public $titles; /** * @var string the hint text for the reset button. Defaults to null, meaning using the system-defined text (which is 'Cancel Rating'). */ public $resetText; /** * @var string the value taken when the rating is cleared. Defaults to null, meaning using the system-defined value (which is ''). */ public $resetValue; /** * @var boolean whether the rating value can be empty (not set). Defaults to true. * When this is true, a reset button will be displayed in front of stars. */ public $allowEmpty; /** * @var integer the width of star image. Defaults to null, meaning using the system-defined value (which is 16). */ public $starWidth; /** * @var boolean whether the rating value is read-only or not. Defaults to false. * When this is true, the rating cannot be changed. */ public $readOnly; /** * @var string Callback when the stars are focused. */ public $focus; /** * @var string Callback when the stars are not focused. */ public $blur; /** * @var string Callback when the stars are clicked. */ public $callback; /** * Executes the widget. * This method registers all needed client scripts and renders * the text field. */ public function run() { list($name,$id)=$this->resolveNameID(); if(isset($this->htmlOptions['id'])) $id=$this->htmlOptions['id']; else $this->htmlOptions['id']=$id; if(isset($this->htmlOptions['name'])) $name=$this->htmlOptions['name']; $this->registerClientScript($id); echo CHtml::openTag('span',$this->htmlOptions)."\n"; $this->renderStars($id,$name); echo "</span>"; } /** * Registers the necessary javascript and css scripts. * @param string $id the ID of the container */ public function registerClientScript($id) { $jsOptions=$this->getClientOptions(); $jsOptions=empty($jsOptions) ? '' : CJavaScript::encode($jsOptions); $js="jQuery('#{$id} > input').rating({$jsOptions});"; $cs=Yii::app()->getClientScript(); $cs->registerCoreScript('rating'); $cs->registerScript('Yii.CStarRating#'.$id,$js); if($this->cssFile!==false) self::registerCssFile($this->cssFile); } /** * Registers the needed CSS file. * @param string $url the CSS URL. If null, a default CSS URL will be used. */ public static function registerCssFile($url=null) { $cs=Yii::app()->getClientScript(); if($url===null) $url=$cs->getCoreScriptUrl().'/rating/jquery.rating.css'; $cs->registerCssFile($url); } /** * Renders the stars. * @param string $id the ID of the container * @param string $name the name of the input */ protected function renderStars($id,$name) { $inputCount=(int)(($this->maxRating-$this->minRating)/$this->ratingStepSize+1); $starSplit=(int)($inputCount/$this->starCount); if($this->hasModel()) { $attr=$this->attribute; CHtml::resolveName($this->model,$attr); $selection=$this->model->$attr; } else $selection=$this->value; $options=$starSplit>1 ? array('class'=>"{split:{$starSplit}}") : array(); for($value=$this->minRating, $i=0;$i<$inputCount; ++$i, $value+=$this->ratingStepSize) { $options['id']=$id.'_'.$i; $options['value']=$value; if(isset($this->titles[$value])) $options['title']=$this->titles[$value]; else unset($options['title']); echo CHtml::radioButton($name,!strcmp($value,$selection),$options) . "\n"; } } /** * @return array the javascript options for the star rating */ protected function getClientOptions() { $options=array(); if($this->resetText!==null) $options['cancel']=$this->resetText; if($this->resetValue!==null) $options['cancelValue']=$this->resetValue; if($this->allowEmpty===false) $options['required']=true; if($this->starWidth!==null) $options['starWidth']=$this->starWidth; if($this->readOnly===true) $options['readOnly']=true; foreach(array('focus', 'blur', 'callback') as $event) { if($this->$event!==null) { if($this->$event instanceof CJavaScriptExpression) $options[$event]=$this->$event; else $options[$event]=new CJavaScriptExpression($this->$event); } } return $options; } }