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
├── config
│ ├── config.default.js
│ └── config.local.js
├── lib
│ └── view.js
├── app.js
├── test
├── History.md
├── README.md
└── package.json

# 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
{
"name": "egg-view-ejs",
"eggPlugin": {
"name": "ejs"
},
"keywords": ["egg", "egg-plugin", "egg-view", "ejs"]
}
  • The configuration item is also named after the template engine
// config/config.default.js
module.exports = {
ejs: {}
};

# 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');

Mmdule.exports = class EjsView {
render(filename, locals) {
return new Promise((resolve, reject) => {
// Asynchronous API call
ejs.renderFile(filename, locals, (err, result) => {
if (err) {
reject(err);
} else {
resolve(result);
}
});
});
}

renderString(tpl, locals) {
try {
// Synchronous API call
return Promise.resolve(ejs.render(tpl, locals));
} catch (err) {
return Promise.reject(err);
}
}
};

# 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 calls render methods. The framework also has built in ctx, 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
module.exports = {
ejs: {
cache: true
}
};

# 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
module.exports = app => {
return class ViewHelper extends app.Helper {
// safe is injected by [egg-view-nunjucks] and will not be escaped during rendering.
// Otherwise, the template call shtml will be escaped
shtml(str) {
return this.safe(super.shtml(str));
}
};
};
  • Use a custom helper when rendering:
// {plugin_root}/lib/view.js
const ViewHelper = require('./helper');

module.exports = class MyCustomView {
render(filename, locals) {
locals.helper = new ViewHelper(this.ctx); // call Nunjucks render
}
};

You can view the specific code here

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:

{
"name": "egg-view-nunjucks",
"eggPlugin": {
"name": "nunjucks",
"dep": ["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.