View Plugin Development
In most cases, we need to read the data, render the template and then present it to the user. The framework does not force to use one template engine, but allows developers to select the template by themselves. For details, see Template Rendering.
This article describes the framework's specification constraints on the View plugin, and we can use this to encapsulate the corresponding template engine plugin. The following takes egg-view-ejs as an example.
# Plugin Directory Structure
egg-view-ejs |
# Plugin Naming Convention
- Follow the plugin development specification
- According to the convention, the names of plugins start with
egg-view-
package.json
is configured as follows. Plugins are named after the template engine, such as ejs
{ |
- The configuration item is also named after the template engine
// config/config.default.js |
# View Base Class
The next step is to provide a View base class that will be instantiated on each request.
The base class of the View needs to provide render
and renderString
methods and supports generator and async functions (it can also be a function that returns a Promise). The render
method is used to render files, and the renderString
method is used to render template strings.
The following is a simplified code that can be directly view source
const ejs = require('ejs'); |
# Parameters
The three parameters of the render
method are:
- filename: is the path to the complete file. The framework determines if the file exists when looking for the file. It does not need to be processed here.
- locals: The data needs rendering. It comes from
app.locals
,ctx.locals
and callsrender
methods. The framework also has built inctx
,request
,ctx.helper
objects. - viewOptions: The incoming configuration of the user, which can override the default configuration of the template engine. This can be considered based on the characteristics of the template engine. For example, the cache is enabled by default but a page does not need to be cached.
The three parameters of the renderString
method:
- tpl: template string, not file path.
- locals: same with
render
. - viewOptions: same with
render
.
# Plugin Configuration
According to the naming conventions mentioned above, the configuration name is generally the name of the template engine, such as ejs.
The configuration of the plugin mainly comes from the configuration of the template engine, and the configuration items can be defined according to the specific conditions, such as the configuration of ejs.
// config/config.default.js |
# Helper
The framework provides ctx.helper
for developer use, but in some cases we want to override the helper method and only take effect when the template is rendered.
In template rendering, we often need to output a user-supplied html fragment, in which case, we often use the helper.shtml
provided by the egg-security
plugin.
<div>{{ helper.shtml(data.content) | safe }}</div> |
However, as shown in the code above, we need to use | safe
to tell the template engine that the html is safe and it doesn't need to run escape
again.
This is more cumbersome to use and easy to forget, so we can package it:
- First provide a helper subclass:
// {plugin_root}/lib/helper.js |
- Use a custom helper when rendering:
// {plugin_root}/lib/view.js |
You can view the specific code here
# Security Related
Templates and security are related and egg-security also provides some methods for the template. The template engine can be used according to requirements.
First declare a dependency on egg-security:
{ |
Besides, the framework provides app.injectCsrf and app.injectNonce, for more information on security section.
# Unit Tests
As a high-quality plugin, perfect unit testing is indispensable, and we also provide lots of auxiliary tools to make it painless for plugin developers to write tests with, see unit testing and plugin docs.