小部件
=======
小部件是在[视图](structure-views.md)中使用的可重用单元,
使用面向对象方式创建复杂和可配置用户界面单元。
例如,日期选择器小部件可生成一个精致的允许用户选择日期的日期选择器,
你只需要在视图中插入如下代码:
```php
< ?php
use yii\jui\DatePicker;
?>
<?= DatePicker::widget(['name' => 'date']) ?>
```
Yii提供许多优秀的小部件,比如 [[yii\widgets\ActiveForm|active form]],[[yii\widgets\Menu|menu]],
[jQuery UI widgets ](widget-jui.md ),
[Twitter Bootstrap widgets ](widget-bootstrap.md )。
接下来介绍小部件的基本知识,如果你想了解某个小部件请参考对应的类 API 文档。
## 使用小部件 <span id="using-widgets"></span>
小部件基本上在 [views ](structure-views.md ) 中使用,
在视图中可调用 [[yii\base\Widget::widget()]] 方法使用小部件。
该方法使用 [配置 ](concept-configurations.md ) 数组初始化小部件并返回小部件渲染后的结果。
例如如下代码插入一个日期选择器小部件,它配置为使用俄罗斯语,
输入框内容为 `$model` 的 `from_date` 属性值。
```php
< ?php
use yii\jui\DatePicker;
?>
< ?= DatePicker::widget([
'model' => $model,
'attribute' => 'from_date',
'language' => 'ru',
'dateFormat' => 'php:Y-m-d',
]) ?>
```
一些小部件可在 [[yii\base\Widget::begin()]]
和 [[yii\base\Widget::end()]] 调用中使用数据内容。
例如如下代码使用 [[yii\widgets\ActiveForm]] 小部件生成一个登录表单,
小部件会在 `begin()` 和 `end()` 执行处分别生成 `<form>` 的开始标签和结束标签,
中间的任何代码也会被渲染。
```php
< ?php
use yii\widgets\ActiveForm;
use yii\helpers\Html;
?>
<?php $form = ActiveForm::begin(['id' => 'login-form']); ?>
<?= $form->field($model, 'username') ?>
<?= $form->field($model, 'password')->passwordInput() ?>
< div class = "form-group" >
<?= Html::submitButton('Login') ?>
< / div >
<?php ActiveForm::end(); ?>
```
注意和调用 [[yii\base\Widget::widget()]] 返回渲染结果不同,
调用 [[yii\base\Widget::begin()]] 方法返回一个可组建小部件内容的小部件实例。
> Note: 当调用 [[yii\base\Widget::end()]] 的时候,一些小部件将使用 [输出缓冲](http://php.net/manual/en/book.outcontrol.php)
> 来调整封闭的内容。因此,当调用 [[yii\base\Widget::begin()]] 和
> [[yii\base\Widget::end()]] 时,最好在同一个视图文件里。
> 不遵循这个规则可能会导致意外的输出。
### 配置全局默认值
小部件的全局默认值可以通过 DI 容器配置:
```php
\Yii::$container->set('yii\widgets\LinkPager', ['maxButtonCount' => 5]);
```
有关详细信息,请参阅
[依赖注入容器“实践中的应用”一节 ](concept-di-container.md#practical-usage ) 。
## 创建小部件 <span id="creating-widgets"></span>
可以根据需要以两种不同方式创建小部件。
### 1: 使用 `widget()` 方法
继承 [[yii\base\Widget]] 类并覆盖 [[yii\base\Widget::init()]] 和/或
[[yii\base\Widget::run()]] 方法可创建小部件。通常 `init()` 方法处理小部件属性,
`run()` 方法包含小部件生成渲染结果的代码。
渲染结果可以直接“输出”或通过 `run()` 方法作为字符串返回。
如下代码中 `HelloWidget` 编码并显示赋给 `message` 属性的值,
如果属性没有被赋值,默认会显示 "Hello World"。
```php
namespace app\components;
use yii\base\Widget;
use yii\helpers\Html;
class HelloWidget extends Widget
{
public $message;
public function init()
{
parent::init();
if ($this->message === null) {
$this->message = 'Hello World';
}
}
public function run()
{
return Html::encode($this->message);
}
}
```
使用这个小部件只需在视图中简单使用如下代码:
```php
< ?php
use app\components\HelloWidget;
?>
<?= HelloWidget::widget(['message' => 'Good morning']) ?>
```
有时小部件需要渲染很多内容,虽然你可以在 `run()` 方法中嵌入内容,但更好的方法是将内容放入一个[视图](structure-views.md)文件,
然后调用 [[yii\base\Widget::render()]] 方法渲染该视图文件,
例如:
```php
public function run()
{
return $this->render('hello');
}
```
### 2: 使用 `begin()` 和 `end()` 方法
这类似于上面的有细微差别。
以下是另一种可在 `begin()` 和 `end()` 调用中使用的 `HelloWidget` ,
HTML 编码内容然后显示。
```php
namespace app\components;
use yii\base\Widget;
use yii\helpers\Html;
class HelloWidget extends Widget
{
public function init()
{
parent::init();
ob_start();
}
public function run()
{
$content = ob_get_clean();
return Html::encode($content);
}
}
```
如上所示,PHP 输出缓冲在 `init()` 启动,所有在 `init()`
和 `run()` 方法之间的输出内容都会被获取,并在 `run()` 处理和返回。
> Info: 当你调用 [[yii\base\Widget::begin()]] 时会创建一个新的小部件
实例并在构造结束时调用 `init()` 方法,
在 `end()` 时会调用 `run()` 方法并输出返回结果。
如下代码显示如何使用这种 `HelloWidget` :
```php
< ?php
use app\components\HelloWidget;
?>
<?php HelloWidget::begin(); ?>
sample content that may contain one or more < strong > HTML< / strong > < pre > tags< / pre >
If this content grows too big, use sub views
For e.g.
<?php echo $this->render('viewfile'); // Note: here render() method is of class \yii \base \View as this part of code is within view file and not in Widget class file ?>
<?php HelloWidget::end(); ?>
```
默认情况下,小部件的视图应存储在 `WidgetPath/views` 目录,`WidgetPath` 代表小部件类文件所在的目录。
假如小部件类文件在 `@app/components` 下,
上述示例会渲染 `@app/components/views/hello.php` 视图文件。
您可以覆盖 [[yii\base\Widget::getViewPath()]] 方法自定义视图文件所在路径。
## 最佳实践 <span id="best-practices"></span>
小部件是面向对象方式来重用视图代码。
创建小部件时仍需要遵循MVC模式,通常逻辑代码在小部件类,
展示内容在[视图](structure-views.md)中。
小部件设计时应是独立的,也就是说使用一个小部件时候,
可以直接丢弃它而不需要额外的处理。
但是当小部件需要外部资源如 CSS,JavaScript,图片等会比较棘手,
幸运的时候Yii提供 [资源包 ](structure-asset-bundles.md ) 来解决这个问题。
当一个小部件只包含视图代码,它和[视图](structure-views.md)很相似,
实际上,在这种情况下,唯一的区别是小部件是可以重用类,
视图只是应用中使用的普通 PHP 脚本。