Modal.php 6.25 KB
Newer Older
1 2 3 4 5 6 7
<?php
/**
 * @link http://www.yiiframework.com/
 * @copyright Copyright (c) 2008 Yii Software LLC
 * @license http://www.yiiframework.com/license/
 */

Antonio Ramirez committed
8
namespace yii\bootstrap;
9 10 11

use Yii;
use yii\helpers\ArrayHelper;
12
use yii\helpers\Html;
13 14

/**
15
 * Modal renders a modal window that can be toggled by clicking on a button.
16
 *
17
 * For example,
18
 *
19 20 21 22 23 24 25
 * ~~~php
 * echo Modal::widget(array(
 *     'header' => '<h2>Hello world</h2>',
 *     'body' => 'Say hello...',
 *     'toggleButton' => array(
 *         'label' => 'click me',
 *     ),
26
 * ));
27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44
 * ~~~
 *
 * The following example will show the content enclosed between the [[begin()]]
 * and [[end()]] calls within the modal window:
 *
 * ~~~php
 * Modal::begin(array(
 *     'header' => '<h2>Hello world</h2>',
 *     'toggleButton' => array(
 *         'label' => 'click me',
 *     ),
 * ));
 *
 * echo 'Say hello...';
 *
 * Modal::end();
 * ~~~
 *
45 46
 * @see http://twitter.github.io/bootstrap/javascript.html#modals
 * @author Antonio Ramirez <amigo.cobos@gmail.com>
47
 * @author Qiang Xue <qiang.xue@gmail.com>
48 49
 * @since 2.0
 */
Antonio Ramirez committed
50
class Modal extends Widget
51 52
{
	/**
53
	 * @var string the header content in the modal window.
54
	 */
55
	public $header;
56
	/**
57 58 59
	 * @var string the body content in the modal window. Note that anything between
	 * the [[begin()]] and [[end()]] calls of the Modal widget will also be treated
	 * as the body content, and will be rendered before this.
60
	 */
61
	public $body;
62
	/**
63
	 * @var string the footer content in the modal window.
64
	 */
65
	public $footer;
66
	/**
67 68 69 70 71 72 73 74 75 76 77 78
	 * @var array the options for rendering the close button tag.
	 * The close button is displayed in the header of the modal window. Clicking
	 * on the button will hide the modal window. If this is null, no close button will be rendered.
	 *
	 * The following special options are supported:
	 *
	 * - tag: string, the tag name of the button. Defaults to 'button'.
	 * - label: string, the label of the button. Defaults to '&times;'.
	 *
	 * The rest of the options will be rendered as the HTML attributes of the button tag.
	 * Please refer to the [Modal plugin help](http://twitter.github.com/bootstrap/javascript.html#modals)
	 * for the supported HTML attributes.
79
	 */
80
	public $closeButton = array();
81
	/**
82 83 84 85 86 87 88 89 90 91 92 93
	 * @var array the options for rendering the toggle button tag.
	 * The toggle button is used to toggle the visibility of the modal window.
	 * If this property is null, no toggle button will be rendered.
	 *
	 * The following special options are supported:
	 *
	 * - tag: string, the tag name of the button. Defaults to 'button'.
	 * - label: string, the label of the button. Defaults to 'Show'.
	 *
	 * The rest of the options will be rendered as the HTML attributes of the button tag.
	 * Please refer to the [Modal plugin help](http://twitter.github.com/bootstrap/javascript.html#modals)
	 * for the supported HTML attributes.
94
	 */
95
	public $toggleButton;
96 97 98


	/**
99
	 * Initializes the widget.
100 101 102 103 104
	 */
	public function init()
	{
		parent::init();

Qiang Xue committed
105
		$this->initOptions();
106

Qiang Xue committed
107 108 109 110
		echo $this->renderToggleButton() . "\n";
		echo Html::beginTag('div', $this->options) . "\n";
		echo $this->renderHeader() . "\n";
		echo $this->renderBodyBegin() . "\n";
111 112 113
	}

	/**
114
	 * Renders the widget.
115 116 117
	 */
	public function run()
	{
Qiang Xue committed
118 119 120
		echo "\n" . $this->renderBodyEnd();
		echo "\n" . $this->renderFooter();
		echo "\n" . Html::endTag('div');
121

122
		$this->registerPlugin('modal');
123 124 125
	}

	/**
126 127
	 * Renders the header HTML markup of the modal
	 * @return string the rendering result
128
	 */
129
	protected function renderHeader()
130
	{
131 132 133 134 135 136 137 138 139
		$button = $this->renderCloseButton();
		if ($button !== null) {
			$this->header = $button . "\n" . $this->header;
		}
		if ($this->header !== null) {
			return Html::tag('div', "\n" . $this->header . "\n", array('class' => 'modal-header'));
		} else {
			return null;
		}
140 141 142
	}

	/**
Qiang Xue committed
143
	 * Renders the opening tag of the modal body.
144
	 * @return string the rendering result
145
	 */
Qiang Xue committed
146
	protected function renderBodyBegin()
147
	{
Qiang Xue committed
148 149 150 151 152 153 154 155 156 157
		return Html::beginTag('div', array('class' => 'modal-body'));
	}

	/**
	 * Renders the closing tag of the modal body.
	 * @return string the rendering result
	 */
	protected function renderBodyEnd()
	{
		return $this->body . "\n" . Html::endTag('div');
158 159 160
	}

	/**
161 162
	 * Renders the HTML markup for the footer of the modal
	 * @return string the rendering result
163
	 */
164
	protected function renderFooter()
165
	{
166
		if ($this->footer !== null) {
Qiang Xue committed
167
			return Html::tag('div', "\n" . $this->footer . "\n", array('class' => 'modal-footer'));
168 169 170
		} else {
			return null;
		}
171 172 173
	}

	/**
174 175
	 * Renders the toggle button.
	 * @return string the rendering result
176
	 */
177
	protected function renderToggleButton()
178
	{
179 180 181 182 183 184
		if ($this->toggleButton !== null) {
			$tag = ArrayHelper::remove($this->toggleButton, 'tag', 'button');
			$label = ArrayHelper::remove($this->toggleButton, 'label', 'Show');
			if ($tag === 'button' && !isset($this->toggleButton['type'])) {
				$this->toggleButton['type'] = 'button';
			}
Qiang Xue committed
185
			return Html::tag($tag, $label, $this->toggleButton);
186 187 188
		} else {
			return null;
		}
189 190 191
	}

	/**
192 193
	 * Renders the close button.
	 * @return string the rendering result
194
	 */
195
	protected function renderCloseButton()
196
	{
197 198 199 200 201 202 203 204 205 206
		if ($this->closeButton !== null) {
			$tag = ArrayHelper::remove($this->closeButton, 'tag', 'button');
			$label = ArrayHelper::remove($this->closeButton, 'label', '&times;');
			if ($tag === 'button' && !isset($this->closeButton['type'])) {
				$this->closeButton['type'] = 'button';
			}
			return Html::tag($tag, $label, $this->closeButton);
		} else {
			return null;
		}
207
	}
Qiang Xue committed
208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240

	/**
	 * Initializes the widget options.
	 * This method sets the default values for various options.
	 */
	protected function initOptions()
	{
		$this->options = array_merge(array(
			'class' => 'modal hide',
		), $this->options);
		$this->addCssClass($this->options, 'modal');

		$this->pluginOptions = array_merge(array(
			'show' => false,
		), $this->pluginOptions);

		if ($this->closeButton !== null) {
			$this->closeButton = array_merge(array(
				'data-dismiss' => 'modal',
				'aria-hidden' => 'true',
				'class' => 'close',
			), $this->closeButton);
		}

		if ($this->toggleButton !== null) {
			$this->toggleButton = array_merge(array(
				'data-toggle' => 'modal',
			), $this->toggleButton);
			if (!isset($this->toggleButton['data-target']) && !isset($this->toggleButton['href'])) {
				$this->toggleButton['data-target'] = '#' . $this->options['id'];
			}
		}
	}
241
}