Skip to content

Commit b65f6d7

Browse files
DominusVilicusyyx990803
DominusVilicus
authored andcommitted
feat(ssr): allow template option to be function in renderToString (#9324)
Also allows return Promise for async handling
1 parent 4fca045 commit b65f6d7

File tree

2 files changed

+34
-11
lines changed

2 files changed

+34
-11
lines changed

src/server/create-renderer.js

+19-5
Original file line numberDiff line numberDiff line change
@@ -23,7 +23,7 @@ export type RenderOptions = {
2323
directives?: Object;
2424
isUnaryTag?: Function;
2525
cache?: RenderCache;
26-
template?: string;
26+
template?: string | (content: string, context: any) => string;
2727
inject?: boolean;
2828
basedir?: string;
2929
shouldPreload?: Function;
@@ -82,14 +82,26 @@ export function createRenderer ({
8282
}, cb)
8383
try {
8484
render(component, write, context, err => {
85+
if (err) {
86+
return cb(err)
87+
}
8588
if (context && context.rendered) {
8689
context.rendered(context)
8790
}
8891
if (template) {
89-
result = templateRenderer.renderSync(result, context)
90-
}
91-
if (err) {
92-
cb(err)
92+
try {
93+
const res = templateRenderer.render(result, context)
94+
if (typeof res !== 'string') {
95+
// function template returning promise
96+
res
97+
.then(html => cb(null, html))
98+
.catch(cb)
99+
} else {
100+
cb(null, res)
101+
}
102+
} catch (e) {
103+
cb(e)
104+
}
93105
} else {
94106
cb(null, result)
95107
}
@@ -119,6 +131,8 @@ export function createRenderer ({
119131
})
120132
}
121133
return renderStream
134+
} else if (typeof template === 'function') {
135+
throw new Error(`function template is only supported in renderToString.`)
122136
} else {
123137
const templateStream = templateRenderer.createStream(context)
124138
renderStream.on('error', err => {

src/server/template-renderer/index.js

+15-6
Original file line numberDiff line numberDiff line change
@@ -11,7 +11,7 @@ import type { ParsedTemplate } from './parse-template'
1111
import type { AsyncFileMapper } from './create-async-file-mapper'
1212

1313
type TemplateRendererOptions = {
14-
template: ?string;
14+
template?: string | (content: string, context: any) => string;
1515
inject?: boolean;
1616
clientManifest?: ClientManifest;
1717
shouldPreload?: (file: string, type: string) => boolean;
@@ -42,7 +42,7 @@ type Resource = {
4242
export default class TemplateRenderer {
4343
options: TemplateRendererOptions;
4444
inject: boolean;
45-
parsedTemplate: ParsedTemplate | null;
45+
parsedTemplate: ParsedTemplate | Function | null;
4646
publicPath: string;
4747
clientManifest: ClientManifest;
4848
preloadFiles: Array<Resource>;
@@ -55,8 +55,12 @@ export default class TemplateRenderer {
5555
this.inject = options.inject !== false
5656
// if no template option is provided, the renderer is created
5757
// as a utility object for rendering assets like preload links and scripts.
58-
this.parsedTemplate = options.template
59-
? parseTemplate(options.template)
58+
59+
const { template } = options
60+
this.parsedTemplate = template
61+
? typeof template === 'string'
62+
? parseTemplate(template)
63+
: template
6064
: null
6165

6266
// function used to serialize initial state JSON
@@ -89,12 +93,17 @@ export default class TemplateRenderer {
8993
}
9094

9195
// render synchronously given rendered app content and render context
92-
renderSync (content: string, context: ?Object) {
96+
render (content: string, context: ?Object): string | Promise<string> {
9397
const template = this.parsedTemplate
9498
if (!template) {
95-
throw new Error('renderSync cannot be called without a template.')
99+
throw new Error('render cannot be called without a template.')
96100
}
97101
context = context || {}
102+
103+
if (typeof template === 'function') {
104+
return template(content, context)
105+
}
106+
98107
if (this.inject) {
99108
return (
100109
template.head(context) +

0 commit comments

Comments
 (0)