Vivliostyle Flavored Markdown
Vivliostyle Flavored Markdown (VFM) は本の執筆のために最適化された Markdown 記法です。Vivliostyle とその関連プロジェクトのために標準化され、公開されています。VFMは CommonMark および GitHub Flavored Markdown (GFM) をベースにして実装されています。
コード (Code)
VFM
```javascript
function main() {}
```HTML
<pre class="language-javascript">
<code class="language-javascript"><span class="token keyword">function</span> <span class="token function">main</span><span class="token punctuation">(</span><span class="token punctuation">)</span> <span class="token punctuation">{</span><span class="token punctuation">}</span>
</code></pre>CSS
pre {
}
pre code {
}VFM は構文強調に Prism を利用しています。
キャプションをつける (with caption)
VFM
```javascript:app.js
function main() {}
```あるいは
```javascript title=app.js
function main() {}
```HTML
<figure class="language-javascript">
<figcaption>app.js</figcaption>
<pre>
<code class="language-javascript">
function main() {}
</code>
</pre>
</figure>CSS
figure[class^='language-'] {
}
figure[class^='language-'] figcaption {
}
figure[class^='language-'] pre {
}
figure[class^='language-'] pre code {
}脚注 (Footnotes)
脚注の定義は Pandoc のようになります。
VFM
VFM は GitHub リポジトリで開発しています[^1].
イシューは GitHub で管理します[^イシュー].
脚注は行の中に記述することもできます^[この部分が脚注です。].
[^1]: [VFM](https://github.com/vivliostyle/vfm)
[^イシュー]: [イシュー](https://github.com/vivliostyle/vfm/issues)HTML
<p>
VFM は GitHub リポジトリで開発しています<a id="fnref1" href="#fn1" class="footnote-ref" role="doc-noteref"><sup>1</sup></a>.
イシューは GitHub で管理します<a id="fnref2" href="#fn2" class="footnote-ref" role="doc-noteref"><sup>2</sup></a>.
脚注は行の中に記述することもできます<a id="fnref3" href="#fn3" class="footnote-ref" role="doc-noteref"><sup>3</sup></a>.
</p>
<section class="footnotes" role="doc-endnotes">
<hr>
<ol>
<li id="fn1" role="doc-endnote"><a href="https://github.com/vivliostyle/vfm">VFM</a><a href="#fnref1" class="footnote-back" role="doc-backlink">↩</a></li>
<li id="fn2" role="doc-endnote"><a href="https://github.com/vivliostyle/vfm/issues">イシュー</a><a href="#fnref2" class="footnote-back" role="doc-backlink">↩</a></li>
<li id="fn3" role="doc-endnote">この部分が脚注です。<a href="#fnref3" class="footnote-back" role="doc-backlink">↩</a></li>
</ol>
</section>CSS
.footnotes {
}フロントマター/前付け (Frontmatter)
フロントマター/前付けは Markdown ファイル単位でメタデータを定義するための方法です。ファイルの冒頭へ YAML で記述します。
YAML の解析には js-yaml を使用しています。スキーマは JSON_SCHEMA です。
js-yaml の解析は key: を key: null にします。しかし VFM はこれを空の文字列として扱います。属性値のプロパティとして key: または key:"" が指定された場合は key="" を出力します。
VFM
---
id: 'my-page'
lang: 'ja'
dir: 'ltr'
class: 'my-class'
title: 'Title'
html:
data-color-mode: 'dark'
data-light-theme: 'light'
data-dark-theme: 'dark'
body:
id: 'body'
class: 'foo bar'
base:
target: '_top'
href: 'https://www.example.com/'
meta:
- name: 'theme-color'
media: '(prefers-color-scheme: light)'
content: 'red'
- name: 'theme-color'
media: '(prefers-color-scheme: dark)'
content: 'darkred'
link:
- rel: 'stylesheet'
href: 'sample1.css'
- rel: 'stylesheet'
href: 'sample2.css'
script:
- type: 'text/javascript'
src: 'sample1.js'
- type: 'text/javascript'
src: 'sample2.js'
vfm:
math: false
theme: 'theme.css'
partial: false
hardLineBreaks: false
disableFormatHtml: false
author: 'Author'
---
テキストHTML
<!doctype html>
<html data-color-mode="dark" data-light-theme="light" data-dark-theme="dark" id="my-page" lang="ja" dir="ltr" class="my-class">
<head>
<meta charset="utf-8">
<title>Title</title>
<base target="_top" href="https://www.example.com/">
<meta name="viewport" content="width=device-width, initial-scale=1">
<meta name="theme-color" media="(prefers-color-scheme: light)" content="red">
<meta name="theme-color" media="(prefers-color-scheme: dark)" content="darkred">
<meta name="author" content="Author">
<link rel="stylesheet" href="sample1.css">
<link rel="stylesheet" href="sample2.css">
<script type="text/javascript" src="sample1.js"></script>
<script type="text/javascript" src="sample2.js"></script>
</head>
<body id="body" class="my-class foo bar">
<p>テキスト</p>
</body>
</html>CSS
.my-class {
}
.foo.bar {
}定義済みのプロパティ (Defined properties)
| プロパティ | データ型 | 説明 |
|---|---|---|
id | String | <html id="..."> |
lang | String | <html lang="..."> |
dir | String | <html dir="...">、指定可能な値は ltr、rtl または auto。 |
class | String | <html class="..."> と <body class="..."> へ反映される。 |
title | String | <title>...</title> がない場合、コンテンツの最初の見出しがタイトルになる。 |
html | Object | <html key="value">、キーと値のペアが <html> の属性になる。 |
body | Object | <body key="value">、キーと値のペアが <body> の属性になる。 |
base | Object | <base key="value">、キーと値のペアが <base> の属性になる。 |
meta | Object[] | <meta key="value">、キーと値のペアが <meta> の属性となる。 |
link | Object[] | <meta key="value">、キーと値のペアが <link> の属性となる。 |
script | Object[] | <script key="value">、キーと値のペアが <script> の属性となる。 |
vfm | Object | VFM の設定。 |
head | - | 将来の利用に予約済み。 |
style | - | 将来の利用に予約済み。 |
| その他 | String | <meta name="key" content="value">、キーと値のペアは単独の <meta> になる。 |
vfm
| プロパティ | データ型 | 初期値 | 説明 |
|---|---|---|---|
math | Boolean | true | 数式を有効にする。 |
partial | Boolean | false | Markdown 部分だけを HTML 化する。<body> 以上は出力されない。 |
hardLineBreaks | Boolean | false | 空白を必要とせず強制改行の位置へ <br> を追加。 |
disableFormatHtml | Boolean | false | HTML の自動整形を無効にする。 |
theme | String | - | Vivliostyle の theme パッケージか、CSS ファイルをそのまま指定する。 |
プロパティのオプション (Priority with options)
同じ目的の仕様が複数ある場合、優先順位は以下の通りになります。
- フロントマター/前付け
- VFM オプション
フロントマター/前付けにおいて html プロパティでルートの id と重複した id が指定されている場合、ルートで定義された方が優先されます。
---
id: 'sample1'
html:
id: 'sample2'
---この例では sample1 が採用されました。
<html id="sample1">
</html>class プロパティの結合 (Merge class properties)
最上層と html、body の class プロパティはスペース区切りで結合されます。
---
class: 'root'
html:
class: 'html'
body:
class: 'body sample'
---以下は結合された例です。
<html class="root html">
<body class="root body sample">
</body>
</html>強制改行(オプション) (Hard new line (optional))
- 改行すると行末へ
<br/>が付きます - 2 行連続の改行は新しいブロックを生成します
この機能はオプションです。Node.js API はオプションとして hardLineBreaks: true、CLI では --hard-line-breaks を指定することで有効化されます。
VFM
はじめまして。
Vivliostyle Flavored Markdown(略して VFM)の世界へようこそ。
VFM は出版物の執筆に適した Markdown 方言であり、Vivliostyle プロジェクトのために策定・実装されました。HTML
<!-- hardLineBreaks: true -->
<p>はじめまして。</p>
<p>
Vivliostyle Flavored Markdown(略して VFM)の世界へようこそ。<br />
VFM は出版物の執筆に適した Markdown 方言であり、Vivliostyle
プロジェクトのために策定・実装されました。
</p>
<!-- hardLineBreaks: false (Default) -->
<p>はじめまして。</p>
<p>
Vivliostyle Flavored Markdown(略して VFM)の世界へようこそ。 VFM
は出版物の執筆に適した Markdown 方言であり、Vivliostyle
プロジェクトのために策定・実装されました。
</p>CSS
p {
}画像 (Image)
VFM
HTML
<img src="./fig1.png" />CSS
img {
}単一行キャプション (with caption and single line)
単一行で書かれた画像はキャプション付きで <figure> 内へ包み込みます。
画像の属性を指定した場合、id は <figure> へ移動され <img> 固有のもの (src など) を除いたすべてがコピーされます。
VFM

{id="image" data-sample="sample"}
text HTML
<figure>
<img src="./fig1.png" alt="Figure 1" />
<figcaption>Figure 1</figcaption>
</figure>
<figure id="image" title="Figure 2" data-sample="sample">
<img src="./fig2.png" alt="caption" title="Figure 2" data-sample="sample">
<figcaption>Figure 2</figcaption>
</figure>
<p>text <img src="./fig3.png" alt="Figure 3"></p>CSS
figure img {
}
figure figcaption {
}数式 (Math equation)
MathJax により処理したHTMLを出力します。
数式は標準で有効化されています。無効にする場合は以下を指定してください。
stringifyAPI のオプション :math: falseVFMAPI のオプション :math: false- CLI オプション :
--disable-math - フロントマター :
vfm:プロパティへmath: false- 参照 : フロントマター/前付け (Frontmatter)
- これは
stringifyよりも優先されますがVFMではそうなりません
VFM における MathJax のインライン記法は $...$、ディスプレイ記法は $$...$$ となります。
また $x = y\n1 + 1 = 2$ や $$\nx = y\n$$ のような複数行もサポートします。ただし $x = y\n\n1 + 1 = 2$ のように空行 \n\n がある場合は段落として分かれるため数式にはなりません。
OK:
$...$,$$...$$...範囲指定が一致$...\n...$,$$\n...\n$$...同じ段落内$...\$...$,$...\$...\\\$..$,$$...\$...\\\$...$$...奇数個の\により$をエスケープ(無効化)する
NG:
$...$$,$$...$...範囲指定が不一致$...\n\n...$,$$...\n\n...$$...改行によって段落へ分離されてしまう$ ...$...スペース (スペース、タブ文字、改行など)、インライン開始の$直後にがある$... $...スペース (スペース、タブ文字、改行など)、インライン終了の$直前にがある$...$5...インライン終了の$直後に数字0...Nがある
VFM
inline:$x = y$
display: $$1 + 1 = 2$$HTML
math が有効で数式記法か <math> タグが存在する場合は MathJax 処理用の <script> も出力します。
<html>
<head>
<meta charset="utf-8">
<meta name="viewport" content="width=device-width, initial-scale=1">
<script async src="https://cdnjs.cloudflare.com/ajax/libs/mathjax/2.7.9/MathJax.js?config=TeX-MML-AM_CHTML"></script>
</head>
<body>
<p>inline: <span class="math inline" data-math-typeset="true>"\(x = y\)</span></p>
<p>display: <span class="math display" data-math-typeset="true">$$1 + 1 = 2$$</span></p>
</body>
</html>CSS
.math.inline {
}
.math.display {
}そのままのHTML (Raw HTML)
VFM
<div class="custom">
<p>Hey</p>
</div>HTML
<div class="custom">
<p>Hey</p>
</div>Markdownをつける (with Markdown)
VFM
<div class="custom">
# Heading
</div>HTML
<div class="custom">
<section id="heading" class="level1">
<h1>Heading</h1>
</section>
</div>ルビ (Ruby)
VFM
This is {Ruby|ルビ}HTML
This is <ruby>Ruby<rt>ルビ</rt></ruby>CSS
ruby {
}
ruby rt {
}ルビにおけるパイプのエスケープ (Escape pipe in ruby body)
区切り記号となるパイプ | をエスケープ (無効化) したい場合は直前に \ を追加します。
VFM
{a\|b|c}HTML
<p><ruby>a|b<rt>c</rt></ruby></p>セクション分け (Sectionization)
見出しを階層的なセクションにします。
- 親が
blockquoteの場合はセクションを分けません - 見出しの深さへ一致するように、セクションの
levelNクラスを設定します - 見出しの
id属性値をセクションのaria-labelledby属性へ値をコピーします
VFM
# Plain
# Introduction {#intro}
# Welcome {.title}
# Level 1
## Level 2
> # Not SectionizeHTML
<section class="level1" aria-labelledby="plain">
<h1 id="plain">Plain</h1>
</section>
<section class="level1" aria-labelledby="intro">
<h1 id="intro">Introduction</h1>
</section>
<section class="level1" aria-labelledby="welcome">
<h1 class="title" id="welcome">Welcome</h1>
</section>
<section class="level1" aria-labelledby="level-1">
<h1 id="level-1">Level 1</h1>
<section class="level2" aria-labelledby="level-2">
<h2 id="level-2">Level 2</h2>
<blockquote>
<h1 id="not-sectionize">Not Sectionize</h1>
</blockquote>
</section>
</section>CSS
body > section {
}
body > section > h1:first-child {
}
.level1 {
}
.level2 {
}
blockquote > h1 {
}