diff --git a/packages/mdx/index.js b/packages/mdx/index.js
index d5cdd889b..0feb733f8 100644
--- a/packages/mdx/index.js
+++ b/packages/mdx/index.js
@@ -30,6 +30,7 @@ function createMdxAstCompiler(options) {
const fn = unified()
.use(toMDAST, options)
.use(remarkMdx, options)
+ .use(mdxHastToJsx, options) // Set JSX compiler early so it can be overridden
.use(squeeze, options)
.use(toMDXAST, options)
@@ -81,8 +82,6 @@ function applyHastPluginsAndCompilers(compiler, options) {
}
})
- compiler.use(mdxHastToJsx, options)
-
for (const compilerPlugin of compilers) {
compiler.use(compilerPlugin, options)
}
diff --git a/packages/remark-mdxs/index.js b/packages/remark-mdxs/index.js
new file mode 100644
index 000000000..177f4d5b6
--- /dev/null
+++ b/packages/remark-mdxs/index.js
@@ -0,0 +1,76 @@
+const visit = require('unist-util-visit')
+const remove = require('unist-util-remove')
+const {toJSX} = require('@mdx-js/mdx/mdx-hast-to-jsx')
+
+module.exports = function({delimiter = 'hr'}) {
+ this.Compiler = tree => {
+ const splits = []
+ const documents = []
+
+ const importNodes = tree.children.filter(n => n.type === 'import')
+ const exportNodes = tree.children.filter(n => n.type === 'export')
+
+ const layout = exportNodes.find(node => node.default)
+
+ // We don't care about imports and exports when handling
+ // multiple MDX documents
+ let mdxsTree = remove(remove(tree, 'export'), 'import')
+ const {children} = mdxsTree
+
+ visit(mdxsTree, node => {
+ if (node.tagName === delimiter) {
+ splits.push(children.indexOf(node))
+ }
+ })
+
+ let previousSplit = 0
+ for (let i = 0; i < splits.length; i++) {
+ const split = splits[i]
+ documents.push(children.slice(previousSplit, split))
+ previousSplit = split + 1
+ }
+
+ documents.push(children.slice(previousSplit))
+
+ const jsxFragments = documents
+ .map(nodes => nodes.map(toJSX).join('\n'))
+ .map((jsx, i) =>
+ `
+function MDXSContent${i}({ components, ...props }) {
+ return (
+ ${jsx.trim()}
+ )
+}
+ `.trim()
+ )
+
+ const defaultExport = `
+const MDXSWrapper = props => [
+${jsxFragments.map((_, i) => ` `).join(',\n')}
+]
+
+export default MDXWrapper
+ `.trim()
+
+ return [
+ importNodes.map(n => n.value.trim()).join('\n'),
+ '',
+ exportNodes
+ .filter(n => !n.default)
+ .map(n => n.value.trim())
+ .join('\n'),
+ '',
+ `const MDXSLayout = ${
+ layout
+ ? layout.value
+ .replace(/^export\s+default\s+/, '')
+ .replace(/;\s*$/, '')
+ : '"wrapper"'
+ }`,
+ '',
+ jsxFragments.join('\n\n'),
+ '',
+ defaultExport
+ ].join('\n')
+ }
+}
diff --git a/packages/remark-mdxs/license b/packages/remark-mdxs/license
new file mode 100644
index 000000000..dc4d4beab
--- /dev/null
+++ b/packages/remark-mdxs/license
@@ -0,0 +1,21 @@
+The MIT License (MIT)
+
+Copyright (c) 2019 John Otander and Brent Jackson
+
+Permission is hereby granted, free of charge, to any person obtaining a copy
+of this software and associated documentation files (the "Software"), to deal
+in the Software without restriction, including without limitation the rights
+to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
+copies of the Software, and to permit persons to whom the Software is
+furnished to do so, subject to the following conditions:
+
+The above copyright notice and this permission notice shall be included in
+all copies or substantial portions of the Software.
+
+THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
+IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
+FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
+AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
+LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
+OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
+THE SOFTWARE.
diff --git a/packages/remark-mdxs/package.json b/packages/remark-mdxs/package.json
new file mode 100644
index 000000000..76c653ea4
--- /dev/null
+++ b/packages/remark-mdxs/package.json
@@ -0,0 +1,54 @@
+{
+ "name": "remark-mdxs",
+ "version": "1.0.0-rc.0",
+ "description": "Support for multiple MDX documents in a single file",
+ "license": "MIT",
+ "keywords": [
+ "mdx",
+ "mdxs",
+ "markdown",
+ "react",
+ "jsx",
+ "remark",
+ "mdxast"
+ ],
+ "homepage": "https://mdxjs.com",
+ "repository": "mdx-js/mdx",
+ "bugs": "https://github.com/mdx-js/mdx/issues",
+ "author": "John Otander (https://johno.com)",
+ "contributors": [
+ "John Otander (http://johnotander.com)",
+ "Brent Jackson (https://jxnblk.com)",
+ "Tim Neutkens ",
+ "Matija Marohnić ",
+ "Titus Wormer (https://wooorm.com)"
+ ],
+ "files": [
+ "index.js"
+ ],
+ "dependencies": {
+ "@babel/core": "^7.2.2",
+ "@babel/helper-plugin-utils": "^7.0.0",
+ "@babel/plugin-proposal-object-rest-spread": "^7.3.2",
+ "@babel/plugin-syntax-jsx": "^7.2.0",
+ "is-alphabetical": "^1.0.2",
+ "remark-parse": "^6.0.0",
+ "unified": "^7.0.0",
+ "unist-util-remove": "^1.0.1",
+ "unist-util-visit": "^1.4.0"
+ },
+ "peerDependencies": {
+ "@mdx-js/mdx": "*"
+ },
+ "scripts": {
+ "test": "jest"
+ },
+ "jest": {
+ "testEnvironment": "node"
+ },
+ "devDependencies": {
+ "jest": "^24.0.0",
+ "remark-stringify": "^6.0.4",
+ "vfile": "^4.0.0"
+ }
+}
diff --git a/packages/remark-mdxs/readme.md b/packages/remark-mdxs/readme.md
new file mode 100644
index 000000000..4645b2e03
--- /dev/null
+++ b/packages/remark-mdxs/readme.md
@@ -0,0 +1,55 @@
+# [remark][]-[mdx][]s
+
+[![Build Status][build-badge]][build]
+[![lerna][lerna-badge]][lerna]
+[![Join the community on Spectrum][spectrum-badge]][spectrum]
+
+> :warning: This project is currently in alpha
+
+[MDXs][] syntax support for [remark][].
+
+## Installation
+
+```sh
+npm install --save remark-mdxs
+```
+
+## Contribute
+
+See [`contributing.md` in `mdx-js/mdx`][contributing] for ways to get started.
+
+This organisation has a [Code of Conduct][coc].
+By interacting with this repository, organisation, or community you agree to
+abide by its terms.
+
+## License
+
+[MIT][] © [Titus Wormer][author] and [John Otander][author2]
+
+
+
+[build]: https://travis-ci.org/mdx-js/mdx
+
+[build-badge]: https://travis-ci.org/mdx-js/mdx.svg?branch=master
+
+[lerna]: https://lernajs.io/
+
+[lerna-badge]: https://img.shields.io/badge/maintained%20with-lerna-cc00ff.svg
+
+[spectrum]: https://spectrum.chat/mdx
+
+[spectrum-badge]: https://withspectrum.github.io/badge/badge.svg
+
+[contributing]: https://github.com/mdx-js/mdx/blob/master/contributing.md
+
+[coc]: https://github.com/mdx-js/mdx/blob/master/code-of-conduct.md
+
+[mit]: license
+
+[remark]: https://github.com/remarkjs/remark
+
+[mdx]: https://github.com/mdx-js/mdx
+
+[author]: https://wooorm.com
+
+[author2]: https://johno.com
diff --git a/packages/remark-mdxs/test/__snapshots__/test.js.snap b/packages/remark-mdxs/test/__snapshots__/test.js.snap
new file mode 100644
index 000000000..d528eae18
--- /dev/null
+++ b/packages/remark-mdxs/test/__snapshots__/test.js.snap
@@ -0,0 +1,41 @@
+// Jest Snapshot v1, https://goo.gl/fbAQLP
+
+exports[`correctly transpiles 1`] = `
+"/* @jsx mdx */
+import Foo from './bar'
+
+export const author = 'fred'
+
+const MDXSLayout = Foo
+
+function MDXSContent0({ components, ...props }) {
+ return (
+ {\`Hello, world! \`}
+ )
+}
+
+function MDXSContent1({ components, ...props }) {
+ return (
+
+ Hi!
+
+ )
+}
+
+function MDXSContent2({ components, ...props }) {
+ return (
+ {\`I'm another document\`}
+
+
+{\`over here.\`}
+ )
+}
+
+const MDXSWrapper = props => [
+ ,
+ ,
+
+]
+
+export default MDXWrapper"
+`;
diff --git a/packages/remark-mdxs/test/test.js b/packages/remark-mdxs/test/test.js
new file mode 100644
index 000000000..cb25266c0
--- /dev/null
+++ b/packages/remark-mdxs/test/test.js
@@ -0,0 +1,30 @@
+const mdx = require('../../mdx')
+const remarkMdxs = require('..')
+
+const FIXTURE = `
+import Foo from './bar'
+export const author = 'fred'
+export default Foo
+
+# Hello, world!
+
+---
+
+
+ Hi!
+
+
+---
+
+# I'm another document
+
+over here.
+`
+
+it('correctly transpiles', async () => {
+ const result = await mdx(FIXTURE, {
+ remarkPlugins: [remarkMdxs]
+ })
+
+ expect(result).toMatchSnapshot()
+})