Skip to content

Commit

Permalink
feat: support Jekyll-like include syntax, see #441
Browse files Browse the repository at this point in the history
  • Loading branch information
harttle committed Jan 19, 2022
1 parent d71d0a0 commit 388d0fb
Show file tree
Hide file tree
Showing 7 changed files with 141 additions and 11 deletions.
33 changes: 31 additions & 2 deletions docs/source/tags/include.md
Original file line number Diff line number Diff line change
Expand Up @@ -53,12 +53,12 @@ When filename is specified as literal string, it supports Liquid output and filt
```

{% note info Escaping %}
In LiquidJS, `"` within quoted string literals need to be escaped. Adding a slash before the quote, e.g. `\"`. Using Jekyll-like filenames can make this easier, see below.
In LiquidJS, `"` within quoted string literals need to be escaped by adding a slash before the quote, e.g. `\"`. Using Jekyll-like filenames can make this easier, see below.
{% endnote %}

## Jekyll-like filenames

Setting [dynamicPartials][dynamicPartials] to `false` will enable Jekyll-like includes, file names are specified as literal string. And it also supports Liquid outputs and filters.
Setting [dynamicPartials][dynamicPartials] to `false` will enable Jekyll-like filenames, where file names are specified as literal string without surrounding quotes. Liquid outputs and filters are also supported within that, for example:

```liquid
{% include prefix/{{ page.my_variable }}/suffix %}
Expand All @@ -70,6 +70,35 @@ This way, you don't need to escape `"` in the filename expression.
{% include prefix/{{name | append: ".html"}} %}
```

## Jekyll include

[jekyllInclude][jekyllInclude] is used to enable Jekyll-like include syntax. Defaults to `false`, when set to `true`:

- Filename will be static: `dynamicPartials` now defaults to `false` (instead of `true`). And you can set `dynamicPartials` back to `true`.
- Use `=` instead of `:` to separate parameter key-values.
- Parameters are under `include` variable instead of current scope.

For example, the following template:

```liquid
{% include name.html header="HEADER" content="CONTENT" %}
```

`name.html` with following content:

```liquid
<header>{{include.header}}</header>
{{include.content}}
```

Note that we're referencing the first parameter by `include.header` instead of `header`. Will output following:

```html
<header>HEADER</header>
CONTENT
```

[extname]: ../api/interfaces/liquid_options_.liquidoptions.html#Optional-extname
[root]: ../api/interfaces/liquid_options_.liquidoptions.html#Optional-root
[dynamicPartials]: ../api/interfaces/liquid_options_.liquidoptions.html#dynamicPartials
[jekyllInclude]: ../api/interfaces/liquid_options_.liquidoptions.html#jekyllInclude
56 changes: 56 additions & 0 deletions docs/source/zh-cn/tags/include.md
Original file line number Diff line number Diff line change
Expand Up @@ -44,5 +44,61 @@ title: Include

上面的例子中,子模板中 `product` 会保有父模板中的 `featured_product` 变量的值。

## 输出和过滤器

文件名为字符串字面量时,支持 Liquid 输出和过滤器。在拼接文件名时很方便:

```liquid
{% include "prefix/{{name | append: \".html\"}}" %}
```

{% note info 转义 %}
字符串字面量里的 `"` 需要转义为 `\"`,使用静态文件名可以避免这个问题,见下面的 Jekyll-like 文件名。
{% endnote %}

## Jekyll-like 文件名

设置 [dynamicPartials][dynamicPartials]`false` 来启用 Jekyll-like 文件名,这时文件名不需要用引号包含,会被当作字面量处理。 这样的字符串里面仍然支持 Liquid 输出和过滤器,例如:

```liquid
{% include prefix/{{ page.my_variable }}/suffix %}
```

这样文件名里的 `"` 就不用转义了。

```liquid
{% include prefix/{{name | append: ".html"}} %}
```

## Jekyll include

[jekyllInclude][jekyllInclude] 用来启用 Jekyll-like include 语法。默认为 `false`,当设置为 `true` 时:

- 默认启用静态文件名:`dynamicPartials` 的默认值变为 `false`(而非 `true`)。但你也可以把它设置回 `true`
- 参数的键和值之间由 `=` 分隔(本来是 `:`)。
- 参数放到了 `include` 变量下,而非当前作用域。

例如下面的模板:

```liquid
{% include name.html header="HEADER" content="CONTENT" %}
```

其中 `name.html` 的内容是:

```liquid
<header>{{include.header}}</header>
{{include.content}}
```

注意我们通过 `include.header` 引用第一个参数,而不是 `header`。输出如下:

```html
<header>HEADER</header>
CONTENT
```

[extname]: ../../api/interfaces/liquid_options_.liquidoptions.html#Optional-extname
[root]: ../../api/interfaces/liquid_options_.liquidoptions.html#Optional-root
[dynamicPartials]: ../../api/interfaces/liquid_options_.liquidoptions.html#dynamicPartials
[jekyllInclude]: ../../api/interfaces/liquid_options_.liquidoptions.html#jekyllInclude
4 changes: 2 additions & 2 deletions src/builtin/tags/include.ts
Original file line number Diff line number Diff line change
Expand Up @@ -20,7 +20,7 @@ export default {
} else tokenizer.p = begin
} else tokenizer.p = begin

this.hash = new Hash(tokenizer.remaining())
this.hash = new Hash(tokenizer.remaining(), this.liquid.options.jekyllInclude)
},
render: function * (ctx: Context, emitter: Emitter) {
const { liquid, hash, withVar } = this
Expand All @@ -34,7 +34,7 @@ export default {
const scope = yield hash.render(ctx)
if (withVar) scope[filepath] = evalToken(withVar, ctx)
const templates = yield liquid._parsePartialFile(filepath, ctx.sync, this['currentFile'])
ctx.push(scope)
ctx.push(ctx.opts.jekyllInclude ? { include: scope } : scope)
yield renderer.renderTemplates(templates, ctx, emitter)
ctx.pop()
ctx.restoreRegister(saved)
Expand Down
6 changes: 5 additions & 1 deletion src/liquid-options.ts
Original file line number Diff line number Diff line change
Expand Up @@ -17,6 +17,8 @@ export interface LiquidOptions {
layouts?: string | string[];
/** Allow refer to layouts/partials by relative pathname. To avoid arbitrary filesystem read, paths been referenced also need to be within corresponding root, partials, layouts. Defaults to `true`. */
relativeReference?: boolean;
/** Use jekyll style include, pass parameters to `include` variable of current scope. Defaults to `false`. */
jekyllInclude?: boolean;
/** Add a extname (if filepath doesn't include one) before template file lookup. Eg: setting to `".html"` will allow including file by basename. Defaults to `""`. */
extname?: string;
/** Whether or not to cache resolved templates. Defaults to `false`. */
Expand Down Expand Up @@ -93,6 +95,7 @@ export interface NormalizedFullOptions extends NormalizedOptions {
partials: string[];
layouts: string[];
relativeReference: boolean;
jekyllInclude: boolean;
extname: string;
cache: undefined | Cache<Thenable<Template[]>>;
jsTruthy: boolean;
Expand Down Expand Up @@ -122,6 +125,7 @@ export const defaultOptions: NormalizedFullOptions = {
layouts: ['.'],
partials: ['.'],
relativeReference: true,
jekyllInclude: false,
cache: undefined,
extname: '',
fs: fs,
Expand Down Expand Up @@ -161,7 +165,7 @@ export function normalize (options: LiquidOptions): NormalizedFullOptions {
else cache = options.cache ? new LRU(1024) : undefined
options.cache = cache
}
options = { ...defaultOptions, ...options }
options = { ...defaultOptions, ...(options.jekyllInclude ? { dynamicPartials: false } : {}), ...options }
if (!options.fs!.dirname && options.relativeReference) {
console.warn('[LiquidJS] `fs.dirname` is required for relativeReference, set relativeReference to `false` to suppress this warning, or provide implementation for `fs.dirname`')
options.relativeReference = false
Expand Down
9 changes: 5 additions & 4 deletions src/parser/tokenizer.ts
Original file line number Diff line number Diff line change
Expand Up @@ -233,16 +233,16 @@ export class Tokenizer {
return new IdentifierToken(this.input, begin, this.p, this.file)
}

readHashes () {
readHashes (jekyllStyle?: boolean) {
const hashes = []
while (true) {
const hash = this.readHash()
const hash = this.readHash(jekyllStyle)
if (!hash) return hashes
hashes.push(hash)
}
}

readHash (): HashToken | undefined {
readHash (jekyllStyle?: boolean): HashToken | undefined {
this.skipBlank()
if (this.peek() === ',') ++this.p
const begin = this.p
Expand All @@ -251,7 +251,8 @@ export class Tokenizer {
let value

this.skipBlank()
if (this.peek() === ':') {
const sep = jekyllStyle ? '=' : ':'
if (this.peek() === sep) {
++this.p
value = this.readValue()
}
Expand Down
4 changes: 2 additions & 2 deletions src/template/tag/hash.ts
Original file line number Diff line number Diff line change
Expand Up @@ -16,9 +16,9 @@ export interface HashValue {
*/
export class Hash {
hash: HashValue = {}
constructor (markup: string) {
constructor (markup: string, jekyllStyle?: boolean) {
const tokenizer = new Tokenizer(markup, {})
for (const hash of tokenizer.readHashes()) {
for (const hash of tokenizer.readHashes(jekyllStyle)) {
this.hash[hash.name.content] = hash.value
}
}
Expand Down
40 changes: 40 additions & 0 deletions test/integration/builtin/tags/include.ts
Original file line number Diff line number Diff line change
Expand Up @@ -248,4 +248,44 @@ describe('tags/include', function () {
return expect(html).to.equal('Xchild with redY')
})
})

describe('Jekyll include', function () {
before(function () {
liquid = new Liquid({
root: '/',
extname: '.html',
jekyllInclude: true
})
})
it('should support Jekyll style include', function () {
mock({
'/current.html': '{% include bar/foo.html content="FOO" %}',
'/bar/foo.html': '{{include.content}}-{{content}}'
})
const html = liquid.renderFileSync('/current.html')
return expect(html).to.equal('FOO-')
})
it('should support multiple parameters', function () {
mock({
'/current.html': '{% include bar/foo.html header="HEADER" content="CONTENT" %}',
'/bar/foo.html': '<h2>{{include.header}}</h2>{{include.content}}'
})
const html = liquid.renderFileSync('/current.html')
return expect(html).to.equal('<h2>HEADER</h2>CONTENT')
})
it('should support dynamicPartials=true', function () {
mock({
'/current.html': '{% include "bar/foo.html" content="FOO" %}',
'/bar/foo.html': '{{include.content}}-{{content}}'
})
liquid = new Liquid({
root: '/',
extname: '.html',
jekyllInclude: true,
dynamicPartials: true
})
const html = liquid.renderFileSync('/current.html')
return expect(html).to.equal('FOO-')
})
})
})

0 comments on commit 388d0fb

Please sign in to comment.