@@ -1,9 +1,8 @@
|
|||||||
# GitBlog.md
|
|
||||||
|
|
||||||
|
|
||||||
[](https://travis-ci.org/Klemek/GitBlog.md)
|
[](https://travis-ci.org/Klemek/GitBlog.md)
|
||||||
[](https://coveralls.io/github/Klemek/GitBlog.md?branch=master)
|
[](https://coveralls.io/github/Klemek/GitBlog.md?branch=master)
|
||||||
|
|
||||||
|
# GitBlog.md
|
||||||
|
|
||||||
A static blog using Markdown pulled from your git repository.
|
A static blog using Markdown pulled from your git repository.
|
||||||
|
|
||||||
> Step 1 : ```$ vi 2019/06/21/index.md```
|
> Step 1 : ```$ vi 2019/06/21/index.md```
|
||||||
@@ -245,6 +244,9 @@ Any URL like `/year/month/day/anything/` will redirect to this article (and link
|
|||||||
It allows you to add math equations to your articles by simply writing LaTeX between `$$` for full size (and between $ for inline) (more info [here](https://www.mathjax.org/))
|
It allows you to add math equations to your articles by simply writing LaTeX between `$$` for full size (and between $ for inline) (more info [here](https://www.mathjax.org/))
|
||||||
* **PlantUML**
|
* **PlantUML**
|
||||||
It allows you to add UML diagrams with PlantUML Syntax between `@startuml` and `@enduml` (more info [here](http://www.plantuml.com))
|
It allows you to add UML diagrams with PlantUML Syntax between `@startuml` and `@enduml` (more info [here](http://www.plantuml.com))
|
||||||
|
* **fa-diagrams**
|
||||||
|
It allows you to define SVG diagrams with Font-Awesome icons in YAML between `@startfad` and `@endfad` (more info [here](https://github.com/Klemek/fa-diagrams))
|
||||||
|
|
||||||
|
|
||||||
## Configuration
|
## Configuration
|
||||||
[back to top](#gitblog-md)
|
[back to top](#gitblog-md)
|
||||||
@@ -273,6 +275,8 @@ Any URL like `/year/month/day/anything/` will redirect to this article (and link
|
|||||||
activate MathJax equations formatting
|
activate MathJax equations formatting
|
||||||
* `plantuml` (default: true)
|
* `plantuml` (default: true)
|
||||||
activate PlantUML diagram rendering
|
activate PlantUML diagram rendering
|
||||||
|
* `fa-diagrams` (default: true)
|
||||||
|
activate fa-diagrams rendering
|
||||||
* `home`
|
* `home`
|
||||||
* `title` (default: GitBlog.md)
|
* `title` (default: GitBlog.md)
|
||||||
the title of your blog, **strongly advised to be changed**
|
the title of your blog, **strongly advised to be changed**
|
||||||
|
|||||||
Generated
+19
-7
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "gitblog.md",
|
"name": "gitblog.md",
|
||||||
"version": "1.1.5",
|
"version": "1.2.5",
|
||||||
"lockfileVersion": 1,
|
"lockfileVersion": 1,
|
||||||
"requires": true,
|
"requires": true,
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
@@ -1337,7 +1337,6 @@
|
|||||||
"version": "1.0.10",
|
"version": "1.0.10",
|
||||||
"resolved": "https://registry.npmjs.org/argparse/-/argparse-1.0.10.tgz",
|
"resolved": "https://registry.npmjs.org/argparse/-/argparse-1.0.10.tgz",
|
||||||
"integrity": "sha512-o5Roy6tNG4SL/FOkCAN6RzjiakZS25RLYFrcMttJqbdd8BWrnA+fGz57iN5Pb06pvBGvl5gQ0B48dJlslXvoTg==",
|
"integrity": "sha512-o5Roy6tNG4SL/FOkCAN6RzjiakZS25RLYFrcMttJqbdd8BWrnA+fGz57iN5Pb06pvBGvl5gQ0B48dJlslXvoTg==",
|
||||||
"dev": true,
|
|
||||||
"requires": {
|
"requires": {
|
||||||
"sprintf-js": "~1.0.2"
|
"sprintf-js": "~1.0.2"
|
||||||
}
|
}
|
||||||
@@ -3190,6 +3189,14 @@
|
|||||||
"resolved": "https://registry.npmjs.org/extsprintf/-/extsprintf-1.3.0.tgz",
|
"resolved": "https://registry.npmjs.org/extsprintf/-/extsprintf-1.3.0.tgz",
|
||||||
"integrity": "sha1-lpGEQOMEGnpBT4xS48V06zw+HgU="
|
"integrity": "sha1-lpGEQOMEGnpBT4xS48V06zw+HgU="
|
||||||
},
|
},
|
||||||
|
"fa-diagrams": {
|
||||||
|
"version": "1.0.3",
|
||||||
|
"resolved": "https://registry.npmjs.org/fa-diagrams/-/fa-diagrams-1.0.3.tgz",
|
||||||
|
"integrity": "sha512-z9pFYYnc6s1sUs4lq+DY2m5ddLCKN3jdo5O6hv450WwjiQ7otk1RoGQsvpMcwSU1kMEzPJUI6AeTVvzyIElhwQ==",
|
||||||
|
"requires": {
|
||||||
|
"xml-js": "^1.6.11"
|
||||||
|
}
|
||||||
|
},
|
||||||
"fast-deep-equal": {
|
"fast-deep-equal": {
|
||||||
"version": "2.0.1",
|
"version": "2.0.1",
|
||||||
"resolved": "https://registry.npmjs.org/fast-deep-equal/-/fast-deep-equal-2.0.1.tgz",
|
"resolved": "https://registry.npmjs.org/fast-deep-equal/-/fast-deep-equal-2.0.1.tgz",
|
||||||
@@ -6617,7 +6624,6 @@
|
|||||||
"version": "3.13.1",
|
"version": "3.13.1",
|
||||||
"resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-3.13.1.tgz",
|
"resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-3.13.1.tgz",
|
||||||
"integrity": "sha512-YfbcO7jXDdyj0DGxYVSlSeQNHbD7XPWvrVWeVUujrQEoZzWJIRrCPoyk6kL6IAjAG2IolMK4T0hNUe0HOUs5Jw==",
|
"integrity": "sha512-YfbcO7jXDdyj0DGxYVSlSeQNHbD7XPWvrVWeVUujrQEoZzWJIRrCPoyk6kL6IAjAG2IolMK4T0hNUe0HOUs5Jw==",
|
||||||
"dev": true,
|
|
||||||
"requires": {
|
"requires": {
|
||||||
"argparse": "^1.0.7",
|
"argparse": "^1.0.7",
|
||||||
"esprima": "^4.0.0"
|
"esprima": "^4.0.0"
|
||||||
@@ -6626,8 +6632,7 @@
|
|||||||
"esprima": {
|
"esprima": {
|
||||||
"version": "4.0.1",
|
"version": "4.0.1",
|
||||||
"resolved": "https://registry.npmjs.org/esprima/-/esprima-4.0.1.tgz",
|
"resolved": "https://registry.npmjs.org/esprima/-/esprima-4.0.1.tgz",
|
||||||
"integrity": "sha512-eGuFFw7Upda+g4p+QHvnW0RyTX/SVeJBDM/gCtMARO0cLuT2HcEKnTPvhjV6aGeqrCB/sbNop0Kszm0jsaWU4A==",
|
"integrity": "sha512-eGuFFw7Upda+g4p+QHvnW0RyTX/SVeJBDM/gCtMARO0cLuT2HcEKnTPvhjV6aGeqrCB/sbNop0Kszm0jsaWU4A=="
|
||||||
"dev": true
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
@@ -9087,8 +9092,7 @@
|
|||||||
"sprintf-js": {
|
"sprintf-js": {
|
||||||
"version": "1.0.3",
|
"version": "1.0.3",
|
||||||
"resolved": "https://registry.npmjs.org/sprintf-js/-/sprintf-js-1.0.3.tgz",
|
"resolved": "https://registry.npmjs.org/sprintf-js/-/sprintf-js-1.0.3.tgz",
|
||||||
"integrity": "sha1-BOaSb2YolTVPPdAVIDYzuFcpfiw=",
|
"integrity": "sha1-BOaSb2YolTVPPdAVIDYzuFcpfiw="
|
||||||
"dev": true
|
|
||||||
},
|
},
|
||||||
"sshpk": {
|
"sshpk": {
|
||||||
"version": "1.16.1",
|
"version": "1.16.1",
|
||||||
@@ -9777,6 +9781,14 @@
|
|||||||
"resolved": "https://registry.npmjs.org/xml/-/xml-1.0.1.tgz",
|
"resolved": "https://registry.npmjs.org/xml/-/xml-1.0.1.tgz",
|
||||||
"integrity": "sha1-eLpyAgApxbyHuKgaPPzXS0ovweU="
|
"integrity": "sha1-eLpyAgApxbyHuKgaPPzXS0ovweU="
|
||||||
},
|
},
|
||||||
|
"xml-js": {
|
||||||
|
"version": "1.6.11",
|
||||||
|
"resolved": "https://registry.npmjs.org/xml-js/-/xml-js-1.6.11.tgz",
|
||||||
|
"integrity": "sha512-7rVi2KMfwfWFl+GpPg6m80IVMWXLRjO+PxTq7V2CDhoGak0wzYzFgUY2m4XJ47OGdXd8eLE8EmwfAmdjw7lC1g==",
|
||||||
|
"requires": {
|
||||||
|
"sax": "^1.2.4"
|
||||||
|
}
|
||||||
|
},
|
||||||
"xml-name-validator": {
|
"xml-name-validator": {
|
||||||
"version": "3.0.0",
|
"version": "3.0.0",
|
||||||
"resolved": "https://registry.npmjs.org/xml-name-validator/-/xml-name-validator-3.0.0.tgz",
|
"resolved": "https://registry.npmjs.org/xml-name-validator/-/xml-name-validator-3.0.0.tgz",
|
||||||
|
|||||||
+3
-1
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "gitblog.md",
|
"name": "gitblog.md",
|
||||||
"version": "1.2.5",
|
"version": "1.2.6",
|
||||||
"description": "A static blog using Markdown pulled from your git repository.",
|
"description": "A static blog using Markdown pulled from your git repository.",
|
||||||
"main": "src/server.js",
|
"main": "src/server.js",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
@@ -8,6 +8,8 @@
|
|||||||
"crypto": "^1.0.1",
|
"crypto": "^1.0.1",
|
||||||
"ejs": "^2.6.2",
|
"ejs": "^2.6.2",
|
||||||
"express": "^4.17.1",
|
"express": "^4.17.1",
|
||||||
|
"fa-diagrams": "^1.0.3",
|
||||||
|
"js-yaml": "^3.13.1",
|
||||||
"mathjax-node": "^2.1.1",
|
"mathjax-node": "^2.1.1",
|
||||||
"ncp": "^2.0.0",
|
"ncp": "^2.0.0",
|
||||||
"node-prismjs": "^0.1.2",
|
"node-prismjs": "^0.1.2",
|
||||||
|
|||||||
@@ -19,6 +19,7 @@ If you see this page, that means it's working
|
|||||||
* [Spoilers](#spoilers)
|
* [Spoilers](#spoilers)
|
||||||
* [Math Equations](#mathequations)
|
* [Math Equations](#mathequations)
|
||||||
* [UML](#uml)
|
* [UML](#uml)
|
||||||
|
* [Diagrams](#diagrams)
|
||||||
* [Youtube Videos](#youtubevideos)
|
* [Youtube Videos](#youtubevideos)
|
||||||
|
|
||||||
### Headers
|
### Headers
|
||||||
@@ -253,6 +254,30 @@ showdown -left-> express : 4. html
|
|||||||
express -up-> web : 5. html
|
express -up-> web : 5. html
|
||||||
@enduml
|
@enduml
|
||||||
|
|
||||||
|
### Diagrams
|
||||||
|
[Back to top](#top)
|
||||||
|
|
||||||
|
You can use [fa-diagrams](https://github.com/Klemek/fa-diagrams) with `@startfad` and `@endfad` tags and using YAML inside
|
||||||
|
|
||||||
|
@startuml
|
||||||
|
nodes:
|
||||||
|
- name: node1
|
||||||
|
icon: laptop-code
|
||||||
|
color: '#4E342E'
|
||||||
|
bottom: my app
|
||||||
|
- name: node2
|
||||||
|
icon: globe
|
||||||
|
color: '#455A64'
|
||||||
|
bottom: world
|
||||||
|
links:
|
||||||
|
- from: node1
|
||||||
|
to: node2
|
||||||
|
color: '#333333'
|
||||||
|
top:
|
||||||
|
icon: envelope
|
||||||
|
bottom: '"hello"'
|
||||||
|
@enduml
|
||||||
|
|
||||||
### Youtube Videos
|
### Youtube Videos
|
||||||
[Back to top](#top)
|
[Back to top](#top)
|
||||||
|
|
||||||
|
|||||||
@@ -6,16 +6,18 @@
|
|||||||
</head>
|
</head>
|
||||||
<body>
|
<body>
|
||||||
<main>
|
<main>
|
||||||
<h1><%= info.title %></h1>
|
<h1 class="title"><%= info.title %></h1>
|
||||||
<%= info.description %>
|
<%= info.description %>
|
||||||
<h2>Articles in this blog :</h2>
|
<h2>Articles in this blog :</h2>
|
||||||
<% articles.forEach((article) => { %>
|
<% articles.forEach((article) => { %>
|
||||||
<div class="article">
|
<div class="article">
|
||||||
<h3><%- `<a href="${article.url}">${article.title}</a>` %></h3>
|
<%- `<a href="${article.url}">` %>
|
||||||
<span class="time"><span>Published on</span> <%= article.year + '-' + article.month + '-' + article.day %></span>
|
<h3><%- `${article.title}` %></h3>
|
||||||
|
<span class="time"><span>Published on</span> <%= article.year + '-' + ('0' + article.month).slice(-2) + '-' + ('0' + article.day).slice(-2) %></span>
|
||||||
<% if(article.thumbnail){ %>
|
<% if(article.thumbnail){ %>
|
||||||
<%- `<img alt="thumbnail" src=${article.thumbnail}>` %>
|
<%- `<img alt="thumbnail" src=${article.thumbnail}>` %>
|
||||||
<% } %>
|
<% } %>
|
||||||
|
<%- `</a>` %>
|
||||||
</div>
|
</div>
|
||||||
<% }); %>
|
<% }); %>
|
||||||
<%- include('footer'); %>
|
<%- include('footer'); %>
|
||||||
|
|||||||
@@ -16,7 +16,7 @@ body {
|
|||||||
}
|
}
|
||||||
|
|
||||||
main {
|
main {
|
||||||
max-width: 42rem;
|
max-width: 45rem;
|
||||||
padding: 2rem;
|
padding: 2rem;
|
||||||
margin: auto;
|
margin: auto;
|
||||||
background-color: #F0F0F0;
|
background-color: #F0F0F0;
|
||||||
@@ -54,6 +54,13 @@ pre {
|
|||||||
padding: 10px 16px;
|
padding: 10px 16px;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
:not(pre) > code {
|
||||||
|
padding: 0.25em 0.5em;
|
||||||
|
border-radius: 0.25em;
|
||||||
|
background: #DDD;
|
||||||
|
font-size: 90%;
|
||||||
|
}
|
||||||
|
|
||||||
blockquote {
|
blockquote {
|
||||||
border-left: 0.5em solid #ccc;
|
border-left: 0.5em solid #ccc;
|
||||||
padding-left: 1em;
|
padding-left: 1em;
|
||||||
@@ -108,10 +115,11 @@ main.article div.header a.link-home {
|
|||||||
line-height: 2.4;
|
line-height: 2.4;
|
||||||
}
|
}
|
||||||
|
|
||||||
main.article div.header h1, main.article div.header h2 {
|
main.article div.header h1, main.article div.header h2, .title {
|
||||||
margin-top: 0.85em;
|
margin-top: 0.85em;
|
||||||
margin-bottom: 0.25em;
|
margin-bottom: 0.25em;
|
||||||
font-size: 2em;
|
font-size: 2em;
|
||||||
|
font-weight: bold;
|
||||||
}
|
}
|
||||||
|
|
||||||
main.article div.header h1 a, main.article div.header h2 a, div.article h3 a {
|
main.article div.header h1 a, main.article div.header h2 a, div.article h3 a {
|
||||||
@@ -129,6 +137,12 @@ div.article {
|
|||||||
div.article h3 {
|
div.article h3 {
|
||||||
font-size: 1.3em;
|
font-size: 1.3em;
|
||||||
margin:0;
|
margin:0;
|
||||||
|
color: #3C3CA1;
|
||||||
|
}
|
||||||
|
|
||||||
|
div.article a {
|
||||||
|
text-decoration: none;
|
||||||
|
color: inherit;
|
||||||
}
|
}
|
||||||
|
|
||||||
div.article img{
|
div.article img{
|
||||||
@@ -138,6 +152,15 @@ div.article img{
|
|||||||
margin-top:0.25em;
|
margin-top:0.25em;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
div.article:hover {
|
||||||
|
opacity: 0.9;
|
||||||
|
}
|
||||||
|
|
||||||
|
div.article:active {
|
||||||
|
opacity: 0.8;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
#text {
|
#text {
|
||||||
text-align: justify;
|
text-align: justify;
|
||||||
hyphens: auto;
|
hyphens: auto;
|
||||||
@@ -147,7 +170,7 @@ div.article img{
|
|||||||
text-align: left;
|
text-align: left;
|
||||||
}
|
}
|
||||||
|
|
||||||
#text img {
|
#text img, #text svg {
|
||||||
max-width: 100%;
|
max-width: 100%;
|
||||||
height: auto;
|
height: auto;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -9,7 +9,7 @@
|
|||||||
<div class="header">
|
<div class="header">
|
||||||
<a class="link-home" href="/">↑</a>
|
<a class="link-home" href="/">↑</a>
|
||||||
<h1><%= article.title %></h1>
|
<h1><%= article.title %></h1>
|
||||||
<span class="time"><span>Published on</span> <%= article.year + '-' + article.month + '-' + article.day %></span>
|
<span class="time"><span><%= article.draft ? 'Drafted on' : 'Published on' %></span> <%= article.year + '-' + ('0' + article.month).slice(-2) + '-' + ('0' + article.day).slice(-2) %></span>
|
||||||
</div>
|
</div>
|
||||||
<div id="text"><%- article.content %></div>
|
<div id="text"><%- article.content %></div>
|
||||||
<br>
|
<br>
|
||||||
|
|||||||
@@ -10,7 +10,8 @@
|
|||||||
"webhook": true,
|
"webhook": true,
|
||||||
"prism": true,
|
"prism": true,
|
||||||
"mathjax": true,
|
"mathjax": true,
|
||||||
"plantuml": true
|
"plantuml": true,
|
||||||
|
"fa-diagrams": true
|
||||||
},
|
},
|
||||||
"home": {
|
"home": {
|
||||||
"title": "GitBlog.md",
|
"title": "GitBlog.md",
|
||||||
|
|||||||
+118
-20
@@ -5,6 +5,54 @@ const showdown = require('showdown');
|
|||||||
module.exports = (config) => {
|
module.exports = (config) => {
|
||||||
const converter = new showdown.Converter(config['showdown']);
|
const converter = new showdown.Converter(config['showdown']);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* get parts outside of codes/scripts
|
||||||
|
* @param {string} data
|
||||||
|
* @returns {{index:number, end:number, text:string}[]} parts
|
||||||
|
*/
|
||||||
|
const getParts = (data) => {
|
||||||
|
let parts = [];
|
||||||
|
let match;
|
||||||
|
let i = 0;
|
||||||
|
while ((match = /```/m.exec(data.slice(i)))) {
|
||||||
|
parts.push({
|
||||||
|
index: i,
|
||||||
|
text: data.slice(i, i + match.index),
|
||||||
|
});
|
||||||
|
i += match.index + match[0].length;
|
||||||
|
}
|
||||||
|
if (i < data.length)
|
||||||
|
parts.push({
|
||||||
|
index: i,
|
||||||
|
text: data.slice(i, data.length),
|
||||||
|
});
|
||||||
|
|
||||||
|
parts = parts.filter((p, i) => i % 2 === 0); //filter out code parts
|
||||||
|
|
||||||
|
// detect scripts outside of code
|
||||||
|
parts.forEach((p, pi) => {
|
||||||
|
let i = 0;
|
||||||
|
const subParts = [];
|
||||||
|
while ((match = /(<script>((?:(?!<\/script>)[\s\S])*)<\/script>)/gm.exec(p.text.slice(i)))) {
|
||||||
|
subParts.push({
|
||||||
|
index: p.index + i,
|
||||||
|
text: p.text.slice(i, i + match.index),
|
||||||
|
});
|
||||||
|
i += match.index + match[0].length;
|
||||||
|
}
|
||||||
|
if (i < p.text.length)
|
||||||
|
subParts.push({
|
||||||
|
index: p.index + i,
|
||||||
|
text: p.text.slice(i, p.text.length),
|
||||||
|
});
|
||||||
|
parts.splice(pi, 1, ...subParts);
|
||||||
|
});
|
||||||
|
|
||||||
|
parts.forEach(part => part.end = part.index + part.text.length);
|
||||||
|
|
||||||
|
return parts;
|
||||||
|
};
|
||||||
|
|
||||||
const renderShowDown = (data, cb) => {
|
const renderShowDown = (data, cb) => {
|
||||||
const html = converter.makeHtml(data);
|
const html = converter.makeHtml(data);
|
||||||
cb(html);
|
cb(html);
|
||||||
@@ -35,15 +83,19 @@ module.exports = (config) => {
|
|||||||
const renderPlantUML = (data, cb) => {
|
const renderPlantUML = (data, cb) => {
|
||||||
if (!config['modules']['plantuml'])
|
if (!config['modules']['plantuml'])
|
||||||
return cb(data);
|
return cb(data);
|
||||||
|
const parts = getParts(data);
|
||||||
const umlRegex = /@startuml\r?\n((?:(?!@enduml)[\s\S])*)\r?\n@enduml/m;
|
const umlRegex = /@startuml\r?\n((?:(?!@enduml)[\s\S])*)\r?\n@enduml/m;
|
||||||
let match;
|
let match;
|
||||||
while ((match = umlRegex.exec(data))) {
|
parts.forEach(part => {
|
||||||
const code = match[1].trim();
|
while ((match = umlRegex.exec(part.text))) {
|
||||||
const s = unescape(encodeURIComponent(code)); // jshint ignore:line
|
const code = match[1].trim();
|
||||||
const compressed = global['zip_deflate'](s);
|
const s = unescape(encodeURIComponent(code)); // jshint ignore:line
|
||||||
const url = `http://www.plantuml.com/plantuml/${config['plantuml']['output_format']}/${encode64(compressed)}`;// jshint ignore:line
|
const compressed = global['zip_deflate'](s);
|
||||||
data = data.slice(0, match.index) + `<img alt="generated PlantUML diagram" src="${url}">` + data.slice(match.index + match[0].length);
|
const url = `http://www.plantuml.com/plantuml/${config['plantuml']['output_format']}/${encode64(compressed)}`;// jshint ignore:line
|
||||||
}
|
part.text = part.text.slice(0, match.index) + `<img alt="generated PlantUML diagram" src="${url}">` + part.text.slice(match.index + match[0].length);
|
||||||
|
}
|
||||||
|
data = data.slice(0, part.index) + part.text + data.slice(part.end);
|
||||||
|
});
|
||||||
cb(data);
|
cb(data);
|
||||||
};
|
};
|
||||||
|
|
||||||
@@ -64,7 +116,9 @@ module.exports = (config) => {
|
|||||||
if (!config['modules']['mathjax'])
|
if (!config['modules']['mathjax'])
|
||||||
return cb(data);
|
return cb(data);
|
||||||
|
|
||||||
const doMJ = (match, format) => {
|
const parts = getParts(data);
|
||||||
|
|
||||||
|
const doMJ = (match, format, i) => {
|
||||||
const eq = match[1].trim();
|
const eq = match[1].trim();
|
||||||
const output = config['mathjax']['output_format'];
|
const output = config['mathjax']['output_format'];
|
||||||
const mjConf = {
|
const mjConf = {
|
||||||
@@ -74,7 +128,7 @@ module.exports = (config) => {
|
|||||||
};
|
};
|
||||||
mjConf[output] = true;
|
mjConf[output] = true;
|
||||||
mjAPI.typeset(mjConf, (res) => {
|
mjAPI.typeset(mjConf, (res) => {
|
||||||
data = data.slice(0, match.index) + res[output] + data.slice(match.index + match[0].length);
|
data = data.slice(0, parts[i].index + match.index) + res[output] + data.slice(parts[i].index + match.index + match[0].length);
|
||||||
renderMathJax(data, (data2) => {
|
renderMathJax(data, (data2) => {
|
||||||
cb(data2);
|
cb(data2);
|
||||||
});
|
});
|
||||||
@@ -84,31 +138,75 @@ module.exports = (config) => {
|
|||||||
const eqRegex = /\$\$((?:(?!\$\$)[\s\S])*)\$\$/m;
|
const eqRegex = /\$\$((?:(?!\$\$)[\s\S])*)\$\$/m;
|
||||||
const inlineEqRegex = /\$([^$\n]*)\$/;
|
const inlineEqRegex = /\$([^$\n]*)\$/;
|
||||||
|
|
||||||
let match;
|
for (let i = 0; i < parts.length; i++) {
|
||||||
if ((match = eqRegex.exec(data))) {
|
let match;
|
||||||
doMJ(match, 'TeX');
|
if ((match = eqRegex.exec(parts[i].text))) {
|
||||||
} else if ((match = inlineEqRegex.exec(data))) {
|
return doMJ(match, 'TeX', i);
|
||||||
doMJ(match, 'inline-TeX');
|
} else if ((match = inlineEqRegex.exec(parts[i].text))) {
|
||||||
} else {
|
return doMJ(match, 'inline-TeX', i);
|
||||||
cb(data);
|
}
|
||||||
}
|
}
|
||||||
|
cb(data);
|
||||||
|
};
|
||||||
|
|
||||||
|
let faDiagrams;
|
||||||
|
let yaml;
|
||||||
|
if (config['modules']['fa-diagrams']) {
|
||||||
|
faDiagrams = require('fa-diagrams');
|
||||||
|
yaml = require('js-yaml');
|
||||||
|
}
|
||||||
|
|
||||||
|
const renderFaDiagrams = (data, cb) => {
|
||||||
|
if (!config['modules']['fa-diagrams'])
|
||||||
|
return cb(data);
|
||||||
|
const parts = getParts(data);
|
||||||
|
const diagramsRegex = /@startfad\r?\n((?:(?!@endfad)[\s\S])*)\r?\n@endfad/m;
|
||||||
|
let match;
|
||||||
|
parts.forEach(part => {
|
||||||
|
while ((match = diagramsRegex.exec(part.text))) {
|
||||||
|
const code = match[1].trim();
|
||||||
|
let output;
|
||||||
|
try {
|
||||||
|
const diagData = yaml.safeLoad(code);
|
||||||
|
const findLineBreaks = (data) => {
|
||||||
|
Object.keys(data).forEach(key => {
|
||||||
|
if (typeof data[key] === 'object')
|
||||||
|
findLineBreaks(data[key]);
|
||||||
|
else if (typeof data[key] === 'string')
|
||||||
|
data[key] = data[key].replace(/\\n/gm, '\n');
|
||||||
|
});
|
||||||
|
};
|
||||||
|
findLineBreaks(diagData);
|
||||||
|
output = faDiagrams.compute(diagData);
|
||||||
|
} catch (err) {
|
||||||
|
output = `<b style="color:red">${err.toString()}</b>`;
|
||||||
|
}
|
||||||
|
part.text = part.text.slice(0, match.index) + output + part.text.slice(match.index + match[0].length);
|
||||||
|
}
|
||||||
|
data = data.slice(0, part.index) + part.text + data.slice(part.end);
|
||||||
|
});
|
||||||
|
cb(data);
|
||||||
};
|
};
|
||||||
|
|
||||||
return {
|
return {
|
||||||
|
getParts: config['test'] ? getParts : undefined,
|
||||||
renderShowDown: config['test'] ? renderShowDown : undefined,
|
renderShowDown: config['test'] ? renderShowDown : undefined,
|
||||||
renderPrism: config['test'] ? renderPrism : undefined,
|
renderPrism: config['test'] ? renderPrism : undefined,
|
||||||
renderPlantUML: config['test'] ? renderPlantUML : undefined,
|
renderPlantUML: config['test'] ? renderPlantUML : undefined,
|
||||||
renderMathJax: config['test'] ? renderMathJax : undefined,
|
renderMathJax: config['test'] ? renderMathJax : undefined,
|
||||||
|
renderFaDiagrams: config['test'] ? renderFaDiagrams : undefined,
|
||||||
render: (file, cb) => {
|
render: (file, cb) => {
|
||||||
fs.readFile(file, {encoding: 'UTF-8'}, (err, data) => {
|
fs.readFile(file, {encoding: 'UTF-8'}, (err, data) => {
|
||||||
if (err)
|
if (err)
|
||||||
return cb(err);
|
return cb(err);
|
||||||
|
|
||||||
renderPrism(data, (data) => {
|
renderPlantUML(data, (data) => {
|
||||||
renderPlantUML(data, (data) => {
|
renderFaDiagrams(data, (data) => {
|
||||||
renderMathJax(data, (data) => {
|
renderMathJax(data, (data) => {
|
||||||
renderShowDown(data, (html) => {
|
renderPrism(data, (data) => {
|
||||||
cb(null, html);
|
renderShowDown(data, (html) => {
|
||||||
|
cb(null, html);
|
||||||
|
});
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|||||||
+85
-1
@@ -11,7 +11,8 @@ const config = {
|
|||||||
'modules': {
|
'modules': {
|
||||||
'prism': true,
|
'prism': true,
|
||||||
'mathjax': true,
|
'mathjax': true,
|
||||||
'plantuml': true
|
'plantuml': true,
|
||||||
|
'fa-diagrams': true,
|
||||||
},
|
},
|
||||||
'showdown': {
|
'showdown': {
|
||||||
'simplifiedAutoLink': true,
|
'simplifiedAutoLink': true,
|
||||||
@@ -32,6 +33,7 @@ beforeEach(() => {
|
|||||||
config['modules']['prism'] = true;
|
config['modules']['prism'] = true;
|
||||||
config['modules']['mathjax'] = true;
|
config['modules']['mathjax'] = true;
|
||||||
config['modules']['plantuml'] = true;
|
config['modules']['plantuml'] = true;
|
||||||
|
config['modules']['fa-diagrams'] = true;
|
||||||
utils.deleteFolderSync(dataDir);
|
utils.deleteFolderSync(dataDir);
|
||||||
fs.mkdirSync(dataDir);
|
fs.mkdirSync(dataDir);
|
||||||
});
|
});
|
||||||
@@ -42,6 +44,47 @@ afterAll(() => {
|
|||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
|
describe('get parts', () => {
|
||||||
|
test('normal', () => {
|
||||||
|
const data = 'Hello\nthere\ngeneral\nkenobi';
|
||||||
|
const parts = renderer.getParts(data);
|
||||||
|
expect(parts.map(p => p.text)).toEqual([
|
||||||
|
'Hello\nthere\ngeneral\nkenobi'
|
||||||
|
]);
|
||||||
|
});
|
||||||
|
test('lot of stuff', () => {
|
||||||
|
const data = 'Hello\nthere\n```code```\ngeneral<script>script</script>\n<script>script2</script>\n```<script>script3</script>```kenobi';
|
||||||
|
const parts = renderer.getParts(data);
|
||||||
|
expect(parts).toEqual([
|
||||||
|
{
|
||||||
|
index: 0,
|
||||||
|
end: 12,
|
||||||
|
text: 'Hello\nthere\n'
|
||||||
|
},
|
||||||
|
{
|
||||||
|
index: 22,
|
||||||
|
end: 30,
|
||||||
|
text: '\ngeneral'
|
||||||
|
},
|
||||||
|
{
|
||||||
|
index: 53,
|
||||||
|
end: 54,
|
||||||
|
text: '\n'
|
||||||
|
},
|
||||||
|
{
|
||||||
|
index: 78,
|
||||||
|
end: 79,
|
||||||
|
text: '\n'
|
||||||
|
},
|
||||||
|
{
|
||||||
|
index: 109,
|
||||||
|
end: 115,
|
||||||
|
text: 'kenobi'
|
||||||
|
},
|
||||||
|
]);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
describe('Test Showdown', () => {
|
describe('Test Showdown', () => {
|
||||||
test('normal', (done) => {
|
test('normal', (done) => {
|
||||||
renderer.renderShowDown('# Hello', (html) => {
|
renderer.renderShowDown('# Hello', (html) => {
|
||||||
@@ -112,6 +155,13 @@ describe('Test PlantUML', () => {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
test('plantuml ignored in code', (done) => {
|
||||||
|
renderer.renderPlantUML('code:\n```@startuml\nBob -> Alice : hello\n@enduml```\n ```@startuml``` @enduml', (data) => {
|
||||||
|
expect(data).toBe('code:\n```@startuml\nBob -> Alice : hello\n@enduml```\n ```@startuml``` @enduml');
|
||||||
|
done();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
test('plantuml multiple uml', (done) => {
|
test('plantuml multiple uml', (done) => {
|
||||||
renderer.renderPlantUML('@startuml\nBob -> Alice : hello\n@enduml\n@startuml\nBob -> Alice : hello\n@enduml', (data) => {
|
renderer.renderPlantUML('@startuml\nBob -> Alice : hello\n@enduml\n@startuml\nBob -> Alice : hello\n@enduml', (data) => {
|
||||||
expect(data).toBe('<img alt="generated PlantUML diagram" src="http://www.plantuml.com/plantuml/svg/SyfFKj2rKt3CoKnELR1Io4ZDoSa70000">\n<img alt="generated PlantUML diagram" src="http://www.plantuml.com/plantuml/svg/SyfFKj2rKt3CoKnELR1Io4ZDoSa70000">');
|
expect(data).toBe('<img alt="generated PlantUML diagram" src="http://www.plantuml.com/plantuml/svg/SyfFKj2rKt3CoKnELR1Io4ZDoSa70000">\n<img alt="generated PlantUML diagram" src="http://www.plantuml.com/plantuml/svg/SyfFKj2rKt3CoKnELR1Io4ZDoSa70000">');
|
||||||
@@ -157,6 +207,12 @@ describe('Test MathJax', () => {
|
|||||||
done();
|
done();
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
test('no eq in code / script', (done) => {
|
||||||
|
renderer.renderMathJax('this code is ```start $a$ end $$hello$$``` beautiful <script>$A$</script>\n```$no eq$```', (data) => {
|
||||||
|
expect(data).toBe('this code is ```start $a$ end $$hello$$``` beautiful <script>$A$</script>\n```$no eq$```');
|
||||||
|
done();
|
||||||
|
});
|
||||||
|
});
|
||||||
test('multiple eq', (done) => {
|
test('multiple eq', (done) => {
|
||||||
renderer.renderMathJax('$$\n\nA\n\n$$\nstart $a$ end\n$$\n\nA\n\n$$', (data) => {
|
renderer.renderMathJax('$$\n\nA\n\n$$\nstart $a$ end\n$$\n\nA\n\n$$', (data) => {
|
||||||
expect(data).toBe('' +
|
expect(data).toBe('' +
|
||||||
@@ -182,6 +238,34 @@ describe('Test MathJax', () => {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
describe('Test fa-diagrams', () => {
|
||||||
|
test('no fa-diagrams', (done) => {
|
||||||
|
config['modules']['fa-diagrams'] = false;
|
||||||
|
renderer.renderFaDiagrams('@startfad\noptions:\n\trendering:\t\tcolor:red\n\n@endfad', (data) => {
|
||||||
|
expect(data).toBe('@startfad\noptions:\n\trendering:\t\tcolor:red\n\n@endfad');
|
||||||
|
done();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
test('no fa-diagrams in code', (done) => {
|
||||||
|
renderer.renderFaDiagrams('code:\n```\n@startfad\noptions:\n\trendering:\t\tcolor:red\n\n@endfad\n```', (data) => {
|
||||||
|
expect(data).toBe('code:\n```\n@startfad\noptions:\n\trendering:\t\tcolor:red\n\n@endfad\n```');
|
||||||
|
done();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
test('valid fa-diagrams', (done) => {
|
||||||
|
renderer.renderFaDiagrams('before\n@startfad\noptions:\n rendering:\n color: red\n@endfad\nafter', (data) => {
|
||||||
|
expect(data).toBe('before\n<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 0 0" width="0" height="0" font-family="Arial" font-size="15" fill="red" stroke-width="0"></svg>\nafter');
|
||||||
|
done();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
test('invalid yaml', (done) => {
|
||||||
|
renderer.renderFaDiagrams('before\n@startfad\noptions:\n@endfad\nafter', (data) => {
|
||||||
|
expect(data).toBe('before\n<b style="color:red">TypeError: Cannot convert undefined or null to object</b>\nafter');
|
||||||
|
done();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
describe('Test render', () => {
|
describe('Test render', () => {
|
||||||
test('invalid file', (done) => {
|
test('invalid file', (done) => {
|
||||||
renderer.render('invalid file', (err, html) => {
|
renderer.render('invalid file', (err, html) => {
|
||||||
|
|||||||
Reference in New Issue
Block a user