Skip to content

Commit daa388a

Browse files
authored
v0.1.3
* task: replace remarkable with remark * task: update tests * task: update deps * fix: lint issue * task: update docs * fix: revert highlightjs to prismjs * fix: update docs for prismjs
1 parent 1fb5667 commit daa388a

14 files changed

+4482
-3734
lines changed

demo/babel.config.js

-6
Original file line numberDiff line numberDiff line change
@@ -5,12 +5,6 @@ module.exports = {
55
plugins: [
66
['babel-plugin-transform-builtin-classes', {
77
globals: ['LitElement']
8-
}],
9-
['prismjs', {
10-
'languages': ['java', 'css', 'markup'],
11-
'plugins': ['line-numbers'],
12-
'theme': 'twilight',
13-
'css': true
148
}]
159
]
1610

demo/package.json

+1-3
Original file line numberDiff line numberDiff line change
@@ -7,10 +7,8 @@
77
"lint": "eslint ."
88
},
99
"dependencies": {
10-
"babel-plugin-prismjs": "^1.0.2",
1110
"eslint-loader": "^2.1.2",
12-
"lit-element": "^2.0.1",
13-
"prismjs": "^1.15.0"
11+
"lit-element": "^2.0.1"
1412
},
1513
"devDependencies": {
1614
"@babel/core": "^7.1.0",

demo/src/code_example.md

+3
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,3 @@
1+
```js
2+
example
3+
```

demo/src/hello-world.md

+1-6
Original file line numberDiff line numberDiff line change
@@ -3,19 +3,14 @@ label: 'hello'
33
imports:
44
Button: './button.js'
55
HelloWorld: './hello-world.js'
6-
CSS: 'prismjs/themes/prism-dark.css'
76
---
87
# Hello World
98

109
This is an example component
1110

12-
```render js
11+
```js
1312
var test = 'test';
1413
```
1514

16-
```render
1715
<hello-world label="world"></hello-world>
1816
<my-button label="Hello World"></my-button>
19-
20-
<!-- Override default prism theme <style>${CSS}</style>
21-
```

demo/yarn.lock

+2,341-2,144
Large diffs are not rendered by default.

package.json

+9-3
Original file line numberDiff line numberDiff line change
@@ -29,13 +29,19 @@
2929
"watch": "watch 'npm run -s build' src/ test/"
3030
},
3131
"dependencies": {
32+
"@mapbox/rehype-prism": "^0.4.0",
3233
"camelize": "^1.0.0",
3334
"except": "^0.1.3",
3435
"front-matter": "^3.0.1",
3536
"loader-utils": "^1.2.3",
36-
"node-prismjs": "^0.1.0",
37-
"remarkable": "^1.7.1",
38-
"schema-utils": "^1.0.0"
37+
"node-prismjs": "^0.1.2",
38+
"rehype-raw": "^4.0.2",
39+
"rehype-stringify": "^7.0.0",
40+
"remark-parse": "^8.0.1",
41+
"remark-rehype": "^6.0.0",
42+
"schema-utils": "^1.0.0",
43+
"unified": "^9.0.0",
44+
"vfile-reporter": "^6.0.1"
3945
},
4046
"devDependencies": {
4147
"eslint": "^5.14.0",

readme.md

+33-16
Original file line numberDiff line numberDiff line change
@@ -7,20 +7,17 @@ Web Component Markdown Loader
77
[![devDependencies Status](https://david-dm.org/hutchgrant/wc-markdown-loader/dev-status.svg)](https://david-dm.org/hutchgrant/wc-markdown-loader?type=dev)
88

99
Webpack loader that parses markdown files and converts them to Web Components.
10-
It will also parse FrontMatter to import dependencies and render components
11-
along with it’s source code.
10+
It will also parse FrontMatter to import dependencies and render components.
11+
Provides support for syntax highlighting via [prismjs](https://prismjs.com/)
1212

13-
This loader is a modified fork from [javiercf/react-markdown-loader](https://github.com/javiercf/react-markdown-loader) and can easily be used in conjunction with [Create-Evergreen-App](https://github.com/ProjectEvergreen/create-evergreen-app). It's still in the early stages and contributions are welcome.
13+
This loader is a modified fork from [javiercf/react-markdown-loader](https://github.com/javiercf/react-markdown-loader) and can easily be used in conjunction with [Create-Evergreen-App](https://github.com/ProjectEvergreen/create-evergreen-app) and [Greenwood](https://github.com/ProjectEvergreen/greenwood). It's still in the early stages and contributions are welcome.
1414

1515
## Install
1616

1717
```bash
1818
npm i --save-dev wc-markdown-loader
19-
npm i prismjs
2019
```
2120

22-
Prism is required within your application for syntax highlighting. By default, we're using the twilight theme. You can override that as well.
23-
2421
## Usage
2522

2623
In the FrontMatter you should import the components you want to render
@@ -65,8 +62,6 @@ class HelloWorld extends LitElement {
6562
customElements.define('hello-world', HelloWorld);
6663

6764
```
68-
In the markdown File add the *render* tag to code fenceblocks you want the
69-
loader to compile as Components this will output the rendered component.
7065

7166
*hello-world.md*
7267

@@ -81,15 +76,11 @@ imports:
8176

8277
This is an example component rendered from markdown
8378

84-
```render
8579
&lt;hello-world label="world"&gt;&lt;/hello-world&gt;
86-
```
8780

8881
This is an example code block rendered with syntax highlighter
8982

90-
```render html
91-
&lt;!-- This will only render in prism syntax highlighter -->
92-
&lt;!-- You can override this style, see ./demo/hello-world.md --&gt;
83+
```html
9384
&lt;hello-world label="world"&gt;&lt;/hello-world&gt;
9485
```
9586

@@ -119,7 +110,7 @@ customElements.define('eve-app', AppComponent);
119110

120111
### LightDOM
121112

122-
If you want to disable shadowRoot and instead heave your markdown component render in the root node, you can add the following to your webpack config.
113+
If you want to disable shadowRoot and instead have your markdown component render in the root node, you can add the following to your webpack config.
123114

124115
*webpack.config.js*
125116
```js
@@ -140,7 +131,7 @@ This is if you need to manipulate this component from a parent component etc.
140131

141132
### Custom Style
142133

143-
If you want to set a global custom style to use for your markdown components, you can do so from your webpack config. Keep in mind that this is relative to the working directory. You may need to use a `path.join(__dirname, 'mypath/mypath.css')`. The example below demonstrates a prism theme from `node_modules/prismjs/themes/`.
134+
If you want to set a global custom style to use for your markdown components, you can do so from your webpack config. Keep in mind that this is relative to the working directory. You may need to use a `path.join(__dirname, 'mypath/mypath.css')`. The example below demonstrates a prismjs theme from `node_modules/prismjs/themes/prism-tomorrow.css`.
144135

145136
*webpack.config.js*
146137
```js
@@ -151,7 +142,7 @@ module: {
151142
loader: 'wc-markdown-loader',
152143
options: {
153144
defaultStyle: false,
154-
customStyle: 'prismjs/themes/prism-funky.css'
145+
customStyle: 'prismjs/themes/prism-tomorrow.css'
155146
}
156147
}
157148
]
@@ -160,6 +151,32 @@ module: {
160151

161152
**note**: You can toggle the defaultStyle as well, it will have a lower specificity than the customStyle.
162153

154+
### Unified Presets
155+
156+
You can utilize [unified presets](https://github.com/unifiedjs/unified#preset) via the **preset** option within the loader. The presets are placed in the unified process array after remark is converted to rehype.
157+
158+
*webpack.config.js*
159+
```js
160+
module: {
161+
rules: [
162+
{
163+
test: /\.md$/,
164+
loader: 'wc-markdown-loader',
165+
options: {
166+
preset: {
167+
settings: {bullet: '*', emphasis: '*', fences: true},
168+
plugins: [
169+
require('rehype-slug'),
170+
require('rehype-autolink-headings')
171+
]
172+
]
173+
}
174+
}
175+
}
176+
]
177+
}
178+
```
179+
163180
### Graph
164181

165182
If you want to pre-scaffold an application with a graph of all md files paths and add your own generated labels(removing the need to cite label in each of the file's front-matter), you can create a graph array, write the serialized json to a cache file, then add the path of that .json file to the options of the loader e.g.

src/build.js

+2-1
Original file line numberDiff line numberDiff line change
@@ -33,7 +33,7 @@ module.exports = function build(markdown, defaults) {
3333
}
3434

3535
if (defaultStyle) {
36-
doImports += 'import css from \'prismjs/themes/prism-twilight.css\';\n';
36+
doImports += 'import css from \'wc-markdown-loader/node_modules/prismjs/themes/prism-tomorrow.css\';\n';
3737
}
3838

3939
const style = `
@@ -60,6 +60,7 @@ export const attributes = ${JSON.stringify(camelize(frontMatterAttributes))};
6060
class Component extends LitElement {
6161
render(){
6262
return html \`
63+
6364
${style}
6465
<div>
6566
${js}

src/index.js

+4-3
Original file line numberDiff line numberDiff line change
@@ -22,7 +22,7 @@ module.exports = async function loader(content) {
2222
let elementLabel = '';
2323

2424
validateOptions(schema, options, 'wc-markdown-loader');
25-
const { graph, shadowRoot, defaultStyle, customStyle } = options;
25+
const { graph, shadowRoot, preset, defaultStyle, customStyle } = options;
2626

2727
if (Object.keys(options).length > 0 && graph) {
2828
// use preset graph
@@ -43,11 +43,12 @@ module.exports = async function loader(content) {
4343
const defaults = {
4444
label: elementLabel,
4545
defaultDOM: shadowRoot,
46-
defaultStyle: defaultStyle || defaultStyle === undefined,
46+
preset,
47+
defaultStyle,
4748
customStyle
4849
};
4950

50-
parser.parse(content)
51+
parser.parse(content, preset)
5152
.then(markdown => build(markdown, defaults))
5253
.then(component => callback(null, component))
5354
.catch(callback);

src/options.json

+3
Original file line numberDiff line numberDiff line change
@@ -12,6 +12,9 @@
1212
},
1313
"shadowRoot": {
1414
"type": [ "boolean" ]
15+
},
16+
"preset": {
17+
"type": [ "object" ]
1518
}
1619
},
1720
"additionalProperties": false

src/parser.js

+26-93
Original file line numberDiff line numberDiff line change
@@ -1,104 +1,39 @@
11
'use strict';
22

33
const frontMatter = require('front-matter');
4-
const Prism = require('node-prismjs');
5-
const Remarkable = require('remarkable');
6-
const { escapeHtml } = require('remarkable/lib/common/utils');
7-
8-
const md = new Remarkable();
9-
10-
/**
11-
* Wraps the code and jsx in an html component
12-
* for styling it later
13-
* @param {string} exampleRun Code to be run in the styleguide
14-
* @param {string} exampleSrc Source that will be shown as example
15-
* @param {string} langClass CSS class for the code block
16-
* @returns {string} Code block with souce and run code
17-
*/
18-
function codeBlockTemplate(exampleRun, exampleSrc, langClass) {
19-
return `
20-
<pre${!langClass ? '' : ` class="${langClass}"`}>
21-
<code${!langClass ? '' : ` class="${langClass}"`}>${exampleSrc}</code></pre>`;
22-
}
23-
24-
/**
25-
* Parse a code block to have a source and a run code
26-
* @param {String} code - Raw html code
27-
* @param {String} lang - Language indicated in the code block
28-
* @param {String} langPrefix - Language prefix
29-
* @param {Function} highlight - Code highlight function
30-
* @returns {String} Code block with souce and run code
31-
*/
32-
function parseCodeBlock(code, lang, langPrefix, highlight) {
33-
let codeBlock = escapeHtml(code);
34-
35-
if (highlight) {
36-
codeBlock = highlight(code, lang);
37-
}
38-
39-
const langClass = !lang ? '' : `${langPrefix}${escape(lang, true)}`;
40-
const js = code;
41-
42-
codeBlock = codeBlock
43-
// .replace(/{/g, '{"{"{')
44-
// .replace(/}/g, '{"}"}')
45-
// .replace(/{"{"{/g, '{"{"}')
46-
.replace(/\t/g, '');
47-
48-
if (lang !== 'render') {
49-
return codeBlockTemplate(js, codeBlock, langClass);
50-
}
51-
return `<div>${js}</div>`;
52-
}
53-
54-
/**
55-
* @typedef MarkdownObject
56-
* @type {Object}
57-
* @property {Object} attributes - Map of properties from the front matter
58-
* @property {String} body - Markdown
59-
*/
4+
const htmlRehype = require('rehype-stringify');
5+
const raw = require('rehype-raw');
6+
const rehypePrism = require('@mapbox/rehype-prism');
7+
const remarkParse = require('remark-parse');
8+
const remark2rehype = require('remark-rehype');
9+
const report = require('vfile-reporter');
10+
const unified = require('unified');
6011

6112
/**
62-
* @typedef HTMLObject
63-
* @type {Object}
64-
* @property {String} html - HTML parsed from markdown
65-
* @property {Object} imports - Map of dependencies
66-
*/
67-
68-
/**
69-
* Parse Markdown to HTML with code blocks
13+
* Parse Markdown to HTML
7014
* @param {MarkdownObject} markdown - Markdown attributes and body
7115
* @returns {HTMLObject} HTML and imports
7216
*/
73-
function parseMarkdown(markdown) {
74-
return new Promise((resolve, reject) => {
75-
let html;
76-
77-
const options = {
78-
highlight(code, lang) {
79-
const language = Prism.languages[lang] || Prism.languages.autoit;
80-
return Prism.highlight(code, language);
81-
},
82-
xhtmlOut: true
83-
};
17+
function parseMarkdown(markdown, preset = {}) {
18+
return new Promise(async (resolve, reject) => {
19+
let convertedHtml;
20+
const settings = preset.settings ? preset.settings : {};
8421

85-
md.set(options);
22+
try {
23+
const parser = unified()
24+
.use(remarkParse, settings)
25+
.use(remark2rehype, { allowDangerousHtml: true })
26+
.use(raw)
27+
.use(rehypePrism)
28+
.use(preset)
29+
.use(htmlRehype);
8630

87-
md.renderer.rules.fence_custom.render = (tokens, idx, opts) => {
88-
// gets tags applied to fence blocks ```react html
89-
const codeTags = tokens[idx].params.split(/\s+/g);
90-
return parseCodeBlock(
91-
tokens[idx].content,
92-
codeTags[codeTags.length - 1],
93-
opts.langPrefix,
94-
opts.highlight
95-
);
96-
};
31+
convertedHtml = String(await parser.process(markdown.body));
9732

98-
try {
99-
html = md.render(markdown.body);
100-
return resolve({ html, attributes: markdown.attributes });
33+
return resolve({ html: convertedHtml, attributes: markdown.attributes });
10134
} catch (err) {
35+
// eslint-disable-next-line no-console
36+
console.log(report(err));
10237
return reject(err);
10338
}
10439
});
@@ -121,14 +56,12 @@ function parseFrontMatter(markdown) {
12156
* @param {String} markdown - Markdown string to be parsed
12257
* @returns {HTMLObject} HTML and imports
12358
*/
124-
function parse(markdown) {
125-
return parseMarkdown(parseFrontMatter(markdown));
59+
function parse(markdown, preset) {
60+
return parseMarkdown(parseFrontMatter(markdown), preset);
12661
}
12762

12863
module.exports = {
129-
codeBlockTemplate,
13064
parse,
131-
parseCodeBlock,
13265
parseFrontMatter,
13366
parseMarkdown
13467
};

test/index.spec.js

+1-1
Original file line numberDiff line numberDiff line change
@@ -4,7 +4,7 @@ const fs = require('fs');
44
const path = require('path');
55
const loader = require('../src/index.js');
66

7-
describe('React Markdown Loader', () => {
7+
describe('Web Component Markdown Loader', () => {
88
let mdExample = '';
99
const mdFile = path.join(__dirname, '../demo/src/hello-world.md');
1010

0 commit comments

Comments
 (0)