Home>Article>Backend Development> Yii Framework Official Guide Series 44 - Topic: Theming (theme)
Theming is a systematic way to customize the appearance of web pages in web applications. By adopting a new theme, the overall look and feel of a web application can be changed instantly and dramatically.
In Yii, each theme is represented by a directory, including view files, layout files and related resource files, such as images, CSS files, JavaScript files, etc. The name of the theme is its directory name. All themes are placed in the same directoryWebRoot/themes
. At any time, only one theme can be active.
Tip: The default theme root directory
WebRoot/themes
can be configured to other values. Just configure the properties basePath and baseUrl of the themeManager application component to the values you want.
To activate a theme, set the Web application's theme property to the name you want. It can be configured in the application configuration or modified in the controller action during execution.
Note: Topic names are case-sensitive. If you try to start a theme that does not exist,
yii:
:app()->theme
will returnnull
.
The content in the theme directory is organized in the same way as the application base path directory. For example, all view files must be located underviews
, layout view files underviews/layouts
, and system view files underviews/system
. For example, if we want to replace thecreate
view file ofPostController
with theclassic
theme, we will save the new view file asWebRoot/themes/classic /views/post/create.php
.
For the controller view files in the module, the corresponding theme view files will be placed in theviews
directory. For example, if the abovePostController
is in a module namedforum
, we should save thecreate
view file asWebRoot/themes/classic/ views/forum/post/create.php
. If theforum
module is nested in another module namedsupport
, then the view file should beWebRoot/themes/classic/views/support/forum/post/create. php
.
Note: Since the
views
directory may contain security-sensitive data, it should be configured to prevent access by network users.
When we call render or renderPartial to display a view, the corresponding view file and layout file will be found in the currently activated theme. If found, these files will be rendered. Otherwise, it will fall back to the default location specified by viewPath and layoutPath.
Below is an example of directory organization for an application with two themesbaseurl attribute, we can generate the following url for this image file,
yii">Tips: In the view of a theme, we often need to link other theme resource files. For example, we might want to display an image file in the
images
directory under the theme. Using the baseurl attribute of the currently activated theme, we can generate the following url for this image file,
##
yii: :app()->theme->baseUrl . '/images/FileName.gif'
basicand
fancy.
WebRoot/ assets protected/ .htaccess components/ controllers/ models/ views/ layouts/ main.php site/ index.php themes/ basic/ views/ .htaccess layouts/ main.php site/ index.php fancy/ views/ .htaccess layouts/ main.php site/ index.phpIn the application configuration, if we configure
return array( 'theme'=>'basic', ...... );then the
basictheme will be in effect, which means the application's layout will use the one under the directory
themes/basic/views/layouts, and the site's index view will use the one under
themes/basic/views/site. In case a view file is not found in the theme, it will fall back to the one under the
protected/viewsdirectory.
xyzfor a widget whose class name is
Foo, we should first create a folder named
Foo(same as the widget class name) under the currently active theme's view folder. If the widget class is namespaced (available in PHP 5.3.0 or above), such as
\app\widgets\Foo, we should create a folder named
app_widgets_Foo. That is, we replace the namespace separators with the underscore characters.
xyz.phpunder the newly created folder. To this end, we should have a file
themes/basic/views/Foo/xyz.php, which will be used by the widget to replace its original view, if the currently active theme is
basic.
Note:this feature has been available since version 1.1.3.
When using a widget provided by third party or Yii, we often need to customize it for specific needs. For example, we may want to change the value of CLinkPager::maxButtonCount from 10 (default) to 5. We can accomplish this by passing the initial property values when calling CBaseController::widget to create a widget. However, it becomes troublesome to do so if we have to repeat the same customization in every place we useCLinkPager.
$this->widget('CLinkPager', array( 'pages'=>$pagination, 'maxButtonCount'=>5, 'cssFile'=>false, ));
Using the global widget customization feature, we only need to specify these initial values in a single place, i.e., the application configuration. This makes the customization of widgets more manageable.
To use the global widget customization feature, we need to configure the widgetFactory as follows:
return array( 'components'=>array( 'widgetFactory'=>array( 'widgets'=>array( 'CLinkPager'=>array( 'maxButtonCount'=>5, 'cssFile'=>false, ), 'CJuiDatePicker'=>array( 'language'=>'ru', ), ), ), ), );
In the above, we specify the global widget customization for both CLinkPager and CJuiDatePicker widgets by configuring the CWidgetFactory::widgets property. Note that the global customization for each widget is represented as a key-value pair in the array, where the key refers to the wiget class name while the value specifies the initial property value array.
Now, whenever we create a CLinkPager widget in a view, the above property values will be assigned to the widget, and we only need to write the following code in the view to create the widget:
$this->widget('CLinkPager', array( 'pages'=>$pagination, ));
We can still override the initial property values when necessary. For example, if in some view we want to setmaxButtonCount
to be 2, we can do the following:
$this->widget('CLinkPager', array( 'pages'=>$pagination, 'maxButtonCount'=>2, ));
Note:The skin feature has been available since version 1.1.0.
While using a theme we can quickly change the outlook of views, we can use skins to systematically customize the outlook of the widgets used in the views.
A skin is an array of name-value pairs that can be used to initialize the properties of a widget. A skin belongs to a widget class, and a widget class can have multiple skins identified by their names. For example, we can have a skin for the CLinkPager widget and the skin is named asclassic
.
In order to use the skin feature, we first need to modify the application configuration by configuring theCWidgetFactory::enableSkin property to be true for thewidgetFactory
application component:
return array( 'components'=>array( 'widgetFactory'=>array( 'enableSkin'=>true, ), ), );
Please note that in versions prior to 1.1.3, you need to use the following configuration to enable widget skinning:
return array( 'components'=>array( 'widgetFactory'=>array( 'class'=>'CWidgetFactory', ), ), );
We then create the needed skins. Skins belonging to the same widget class are stored in a single PHP script file whose name is the widget class name. All these skin files are stored underprotected/views/skins
, by default. If you want to change this to be a different directory, you may configure theskinPath
property of thewidgetFactory
component. As an example, we may create underprotected/views/skins
a file namedCLinkPager.php
whose content is as follows,
array( 'nextPageLabel'=>'>>', 'prevPageLabel'=>'<<', ), 'classic'=>array( 'header'=>'', 'maxButtonCount'=>5, ), );
In the above, we create two skins for the CLinkPager widget:default
andclassic
. The former is the skin that will be applied to any CLinkPager widget that we do not explicitly specify itsskin
property, while the latter is the skin to be applied to a CLinkPager widget whoseskin
property is specified asclassic
. For example, in the following view code, the first pager will use thedefault
skin while the second theclassic
skin:
widget('CLinkPager'); ?> widget('CLinkPager', array('skin'=>'classic')); ?>
If we create a widget with a set of initial property values, they will take precedence and be merged with any applicable skin. For example, the following view code will create a pager whose initial values will bearray('header'=>'', 'maxButtonCount'=>6, 'cssFile'=>false)
, which is the result of merging the initial property values specified in the view and theclassic
skin.
widget('CLinkPager', array( 'skin'=>'classic', 'maxButtonCount'=>6, 'cssFile'=>false, )); ?>
Note that the skin feature does NOT require using themes. However, when a theme is active, Yii will also look for skins under theskins
directory of the theme's view directory (e.g.WebRoot/themes/classic/views/skins
). In case a skin with the same name exists in both the theme and the main application view directories, the theme skin will take precedence.
If a widget is using a skin that does not exist, Yii will still create the widget as usual without any error.
Info:Using skin may degrade the performance because Yii needs to look for the skin file the first time a widget is being created.
Skin is very similar to the global widget customization feature. The main differences are as follows.
Skin is more related with the customization of presentational property values;
A widget can have multiple skins;
Skin is themeable;
Using skin is more expensive than using global widget customization.
以上就是Yii框架官方指南系列44——专题:Theming(主题)的内容,更多相关内容请关注PHP中文网(m.sbmmt.com)!