Qiang Xue
12 years ago
2 changed files with 409 additions and 0 deletions
@ -0,0 +1,286 @@ |
|||||||
|
<?php |
||||||
|
/** |
||||||
|
* @link http://www.yiiframework.com/ |
||||||
|
* @copyright Copyright (c) 2008 Yii Software LLC |
||||||
|
* @license http://www.yiiframework.com/license/ |
||||||
|
*/ |
||||||
|
|
||||||
|
namespace yii\bootstrap; |
||||||
|
|
||||||
|
use Yii; |
||||||
|
use yii\helpers\Html; |
||||||
|
use yii\helpers\ArrayHelper; |
||||||
|
|
||||||
|
/** |
||||||
|
* Modal renders a bootstrap modal on the page for its use on your application. |
||||||
|
* |
||||||
|
* Basic usage: |
||||||
|
* |
||||||
|
* ```php |
||||||
|
* $this->widget(Modal::className(), array( |
||||||
|
* 'id' => 'myModal', |
||||||
|
* 'header' => 'Modal Heading', |
||||||
|
* 'content' => '<p>One fine body...</p>', |
||||||
|
* 'footer' => 'Modal Footer', |
||||||
|
* // if we wish to display a modal button |
||||||
|
* 'buttonOptions' => array( |
||||||
|
* 'label' => 'Show Modal', |
||||||
|
* 'class' => 'btn btn-primary' |
||||||
|
* ) |
||||||
|
* )); |
||||||
|
* ``` |
||||||
|
* @see http://twitter.github.io/bootstrap/javascript.html#modals |
||||||
|
* @author Antonio Ramirez <amigo.cobos@gmail.com> |
||||||
|
* @since 2.0 |
||||||
|
*/ |
||||||
|
class Modal extends Widget |
||||||
|
{ |
||||||
|
/** |
||||||
|
* @var array The additional HTML attributes of the button that will show the modal. If empty array, only |
||||||
|
* the markup of the modal will be rendered on the page, so users can easily call the modal manually with their own |
||||||
|
* scripts. The following special attributes are available: |
||||||
|
* <ul> |
||||||
|
* <li>label: string, the label of the button</li> |
||||||
|
* </ul> |
||||||
|
* |
||||||
|
* For available options of the button trigger, see http://twitter.github.com/bootstrap/javascript.html#modals. |
||||||
|
*/ |
||||||
|
public $buttonOptions = array(); |
||||||
|
|
||||||
|
/** |
||||||
|
* @var boolean indicates whether the modal should use transitions. Defaults to 'true'. |
||||||
|
*/ |
||||||
|
public $fade = true; |
||||||
|
|
||||||
|
/** |
||||||
|
* @var bool $keyboard, closes the modal when escape key is pressed. |
||||||
|
*/ |
||||||
|
public $keyboard = true; |
||||||
|
|
||||||
|
/** |
||||||
|
* @var bool $show, shows the modal when initialized. |
||||||
|
*/ |
||||||
|
public $show = false; |
||||||
|
|
||||||
|
/** |
||||||
|
* @var mixed includes a modal-backdrop element. Alternatively, specify `static` for a backdrop which doesn't close |
||||||
|
* the modal on click. |
||||||
|
*/ |
||||||
|
public $backdrop = true; |
||||||
|
|
||||||
|
/** |
||||||
|
* @var mixed the remote url. If a remote url is provided, content will be loaded via jQuery's load method and |
||||||
|
* injected into the .modal-body of the modal. |
||||||
|
*/ |
||||||
|
public $remote; |
||||||
|
|
||||||
|
/** |
||||||
|
* @var string a javascript function that will be invoked immediately when the `show` instance method is called. |
||||||
|
*/ |
||||||
|
public $onShow; |
||||||
|
|
||||||
|
/** |
||||||
|
* @var string a javascript function that will be invoked when the modal has been made visible to the user |
||||||
|
* (will wait for css transitions to complete). |
||||||
|
*/ |
||||||
|
public $onShown; |
||||||
|
|
||||||
|
/** |
||||||
|
* @var string a javascript function that will be invoked immediately when the hide instance method has been called. |
||||||
|
*/ |
||||||
|
public $onHide; |
||||||
|
|
||||||
|
/** |
||||||
|
* @var string a javascript function that will be invoked when the modal has finished being hidden from the user |
||||||
|
* (will wait for css transitions to complete). |
||||||
|
*/ |
||||||
|
public $onHidden; |
||||||
|
|
||||||
|
/** |
||||||
|
* @var string[] the Javascript event handlers. |
||||||
|
*/ |
||||||
|
protected $events = array(); |
||||||
|
|
||||||
|
/** |
||||||
|
* @var array $pluginOptions the plugin options. |
||||||
|
*/ |
||||||
|
protected $pluginOptions = array(); |
||||||
|
|
||||||
|
/** |
||||||
|
* @var string |
||||||
|
*/ |
||||||
|
public $closeText = '×'; |
||||||
|
|
||||||
|
/** |
||||||
|
* @var string header content. Header can also be a path to a view file. |
||||||
|
*/ |
||||||
|
public $header; |
||||||
|
|
||||||
|
/** |
||||||
|
* @var string body of modal. Body can also be a path to a view file. |
||||||
|
*/ |
||||||
|
public $content; |
||||||
|
|
||||||
|
/** |
||||||
|
* @var string footer content. Content can also be a path to a view file. |
||||||
|
*/ |
||||||
|
public $footer; |
||||||
|
|
||||||
|
/** |
||||||
|
* Widget's init method |
||||||
|
*/ |
||||||
|
public function init() |
||||||
|
{ |
||||||
|
parent::init(); |
||||||
|
|
||||||
|
$this->name = 'modal'; |
||||||
|
|
||||||
|
$this->defaultOption('id', $this->getId()); |
||||||
|
|
||||||
|
$this->defaultOption('role', 'dialog'); |
||||||
|
$this->defaultOption('tabindex', '-1'); |
||||||
|
|
||||||
|
$this->addClassName('modal'); |
||||||
|
$this->addClassName('hide'); |
||||||
|
|
||||||
|
if ($this->fade) |
||||||
|
$this->addClassName('fade'); |
||||||
|
|
||||||
|
$this->initPluginOptions(); |
||||||
|
$this->initPluginEvents(); |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* Initialize plugin events if any |
||||||
|
*/ |
||||||
|
public function initPluginEvents() |
||||||
|
{ |
||||||
|
foreach (array('onShow', 'onShown', 'onHide', 'onHidden') as $event) { |
||||||
|
if ($this->{$event} !== null) { |
||||||
|
$modalEvent = strtolower(substr($event, 2)); |
||||||
|
if ($this->{$event} instanceof JsExpression) |
||||||
|
$this->events[$modalEvent] = $this->$event; |
||||||
|
else |
||||||
|
$this->events[$modalEvent] = new JsExpression($this->{$event}); |
||||||
|
} |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* Initialize plugin options. |
||||||
|
* ***Important***: The display of the button overrides the initialization of the modal bootstrap widget. |
||||||
|
*/ |
||||||
|
public function initPluginOptions() |
||||||
|
{ |
||||||
|
if (null !== $this->remote) |
||||||
|
$this->pluginOptions['remote'] = Html::url($this->remote); |
||||||
|
|
||||||
|
foreach (array('backdrop', 'keyboard', 'show') as $option) { |
||||||
|
$this->pluginOptions[$option] = isset($this->pluginOptions[$option]) |
||||||
|
? $this->pluginOptions[$option] |
||||||
|
: $this->{$option}; |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* Widget's run method |
||||||
|
*/ |
||||||
|
public function run() |
||||||
|
{ |
||||||
|
$this->renderModal(); |
||||||
|
$this->renderButton(); |
||||||
|
$this->registerScript(); |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* Renders the button that will open the modal if its options have been configured |
||||||
|
*/ |
||||||
|
public function renderButton() |
||||||
|
{ |
||||||
|
if (!empty($this->buttonOptions)) { |
||||||
|
|
||||||
|
$this->buttonOptions['data-toggle'] = isset($this->buttonOptions['data-toggle']) |
||||||
|
? $this->buttonOptions['data-toggle'] |
||||||
|
: 'modal'; |
||||||
|
|
||||||
|
if ($this->remote !== null && !isset($this->buttonOptions['data-remote'])) |
||||||
|
$this->buttonOptions['data-remote'] = Html::url($this->remote); |
||||||
|
|
||||||
|
$label = ArrayHelper::remove($this->buttonOptions, 'label', 'Button'); |
||||||
|
$name = ArrayHelper::remove($this->buttonOptions, 'name'); |
||||||
|
$value = ArrayHelper::remove($this->buttonOptions, 'value'); |
||||||
|
|
||||||
|
$attr = isset($this->buttonOptions['data-remote']) |
||||||
|
? 'data-target' |
||||||
|
: 'href'; |
||||||
|
|
||||||
|
$this->buttonOptions[$attr] = isset($this->buttonOptions[$attr]) |
||||||
|
? $this->buttonOptions[$attr] |
||||||
|
: '#' . ArrayHelper::getValue($this->options, 'id'); |
||||||
|
|
||||||
|
echo Html::button($label, $name, $value, $this->buttonOptions); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* Renders the modal markup |
||||||
|
*/ |
||||||
|
public function renderModal() |
||||||
|
{ |
||||||
|
echo Html::beginTag('div', $this->options); |
||||||
|
|
||||||
|
$this->renderModalHeader(); |
||||||
|
$this->renderModalBody(); |
||||||
|
$this->renderModalFooter(); |
||||||
|
|
||||||
|
echo Html::endTag('div'); |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* Renders the header HTML markup of the modal |
||||||
|
*/ |
||||||
|
public function renderModalHeader() |
||||||
|
{ |
||||||
|
echo Html::beginTag('div', array('class'=>'modal-header')); |
||||||
|
if ($this->closeText) |
||||||
|
echo Html::button($this->closeText, null, null, array('data-dismiss' => 'modal', 'class'=>'close')); |
||||||
|
echo $this->header; |
||||||
|
echo Html::endTag('div'); |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* Renders the HTML markup for the body of the modal |
||||||
|
*/ |
||||||
|
public function renderModalBody() |
||||||
|
{ |
||||||
|
echo Html::beginTag('div', array('class'=>'modal-body')); |
||||||
|
echo $this->content; |
||||||
|
echo Html::endTag('div'); |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* Renders the HTML markup for the footer of the modal |
||||||
|
*/ |
||||||
|
public function renderModalFooter() |
||||||
|
{ |
||||||
|
|
||||||
|
echo Html::beginTag('div', array('class'=>'modal-footer')); |
||||||
|
echo $this->footer; |
||||||
|
echo Html::endTag('div'); |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* Registers client scripts |
||||||
|
*/ |
||||||
|
public function registerScript() |
||||||
|
{ |
||||||
|
// do we render a button? If so, bootstrap will handle its behavior through its |
||||||
|
// mark-up, otherwise, register the plugin. |
||||||
|
if(empty($this->buttonOptions)) |
||||||
|
$this->registerPlugin('modal', $this->pluginOptions); |
||||||
|
|
||||||
|
// register events |
||||||
|
$this->registerEvents($this->events); |
||||||
|
} |
||||||
|
|
||||||
|
} |
@ -0,0 +1,123 @@ |
|||||||
|
<?php |
||||||
|
/** |
||||||
|
* @link http://www.yiiframework.com/ |
||||||
|
* @copyright Copyright (c) 2008 Yii Software LLC |
||||||
|
* @license http://www.yiiframework.com/license/ |
||||||
|
*/ |
||||||
|
|
||||||
|
namespace yii\bootstrap; |
||||||
|
|
||||||
|
use Yii; |
||||||
|
use yii\base\View; |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Bootstrap is the base class for bootstrap widgets. |
||||||
|
* |
||||||
|
* @author Antonio Ramirez <amigo.cobos@gmail.com> |
||||||
|
* @since 2.0 |
||||||
|
*/ |
||||||
|
class Widget extends \yii\base\Widget |
||||||
|
{ |
||||||
|
|
||||||
|
/** |
||||||
|
* @var bool whether to register the asset |
||||||
|
*/ |
||||||
|
public static $responsive = true; |
||||||
|
|
||||||
|
/** |
||||||
|
* @var array the HTML attributes for the widget container tag. |
||||||
|
*/ |
||||||
|
public $options = array(); |
||||||
|
|
||||||
|
/** |
||||||
|
* Initializes the widget. |
||||||
|
*/ |
||||||
|
public function init() |
||||||
|
{ |
||||||
|
// ensure bundle |
||||||
|
$this->registerBundle(static::$responsive); |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* Registers plugin events with the API. |
||||||
|
* @param string $selector the CSS selector. |
||||||
|
* @param string[] $events the JavaScript event configuration (name=>handler). |
||||||
|
* @return boolean whether the events were registered. |
||||||
|
* @todo To be discussed |
||||||
|
*/ |
||||||
|
protected function registerEvents($selector, $events = array()) |
||||||
|
{ |
||||||
|
if (empty($events)) |
||||||
|
return; |
||||||
|
|
||||||
|
$script = ''; |
||||||
|
foreach ($events as $name => $handler) { |
||||||
|
$handler = ($handler instanceof JsExpression) |
||||||
|
? $handler |
||||||
|
: new JsExpression($handler); |
||||||
|
|
||||||
|
$script .= ";jQuery('{$selector}').on('{$name}', {$handler});"; |
||||||
|
} |
||||||
|
if (!empty($script)) |
||||||
|
$this->view->registerJs($script); |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* Registers a specific Bootstrap plugin using the given selector and options. |
||||||
|
* |
||||||
|
* @param string $name the name of the javascript widget to initialize |
||||||
|
* @param array $options the Javascript options for the plugin |
||||||
|
*/ |
||||||
|
public function registerPlugin($name, $options = array()) |
||||||
|
{ |
||||||
|
$selector = '#' . ArrayHelper::getValue($this->options, 'id'); |
||||||
|
$options = !empty($options) ? Json::encode($options) : ''; |
||||||
|
$script = ";jQuery('{$selector}').{$name}({$options});"; |
||||||
|
$this->view->registerJs($script); |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* Registers bootstrap bundle |
||||||
|
* @param bool $responsive |
||||||
|
*/ |
||||||
|
public function registerBundle($responsive = false) |
||||||
|
{ |
||||||
|
$bundle = $responsive ? 'yii/bootstrap-responsive' : 'yii/bootstrap'; |
||||||
|
$this->view->registerAssetBundle($bundle); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Adds a new class to options. If the class key does not exists, it will create one, if it exists it will append |
||||||
|
* the value and also makes sure the uniqueness of them. |
||||||
|
* |
||||||
|
* @param string $class |
||||||
|
* @return array |
||||||
|
*/ |
||||||
|
protected function addClassName($class) |
||||||
|
{ |
||||||
|
if (isset($this->options['class'])) { |
||||||
|
if (!is_array($this->options['class'])) |
||||||
|
$this->options['class'] = explode(' ', $this->options['class']); |
||||||
|
$this->options['class'][] = $class; |
||||||
|
$this->options['class'] = array_unique($this->options['class']); |
||||||
|
$this->options['class'] = implode(' ', $this->options['class']); |
||||||
|
} else |
||||||
|
$this->options['class'] = $class; |
||||||
|
return $this->options; |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* Sets the default value for an item if not set. |
||||||
|
* @param string $key the name of the item. |
||||||
|
* @param mixed $value the default value. |
||||||
|
* @return array |
||||||
|
*/ |
||||||
|
protected function defaultOption($key, $value) |
||||||
|
{ |
||||||
|
if (!isset($this->options[$key])) |
||||||
|
$this->options[$key] = $value; |
||||||
|
return $this->options; |
||||||
|
} |
||||||
|
} |
Loading…
Reference in new issue