-
-
Notifications
You must be signed in to change notification settings - Fork 18
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
How to compile MDX string to JS function, rather than MDX file to JS module #23
Comments
Essentially, I think I'm asking: can I compile to a string which has no |
Yes. |
Thanks @ChristianMurphy but |
|
Boom! Thanks @wooorm that's what I was looking for!! |
This is a somewhat interesting use case that’s not yet support in the API, but the tools are there. Take a look at how evaluate works: https://github.com/wooorm/xdm/blob/main/lib/evaluate.js. Line 10 in f9c3108
|
|
Cool! I'll take a deeper look in to this this week. Maybe I can contribute to your README eventually, for other users who have this use-case 😄 ! |
Would appreciate that! The reason that option starts with an underscore is that I do not yet know how this interface should look. The parts are there (compile and run), but I don't know how to make an intuitive api for it (yet). So I'd appreciate your feedback |
@lunelson i have the exact same use case coming up this week. Did you have any luck? |
@wooorm —CC @stevejcox— so for my Next.js use-case I ended up with the following two functions: the first runs in A few questions and observations:
Thoughts? // /lib/markdown.js
import { useMemo } from 'react';
import * as runtime from 'react/jsx-runtime.js';
import { useMDXComponents } from '@mdx-js/react';
import { runSync } from 'xdm/lib/run';
import { compile } from 'xdm';
import remarkGfm from 'remark-gfm';
export function compileMDXFunction(mdx) {
return compile(mdx, {
format: 'mdx',
_contain: true,
providerImportSource: '@mdx-js/react',
remarkPlugins: [remarkGfm],
}).then((buf) => buf.toString());
}
export function useMDXFunction(code) {
return useMemo(() => {
const { default: Component } = runSync(code, {
...runtime,
useMDXComponents,
});
return Component;
}, [code]);
} FYI, the test Next.js page component: // /pages/index.jsx
import { compileMDXFunction, useMDXFunction } from '../lib/markdown';
import { MDXProvider } from '@mdx-js/react';
export default function Page({ code }) {
const MDXContent = useMDXFunction(code);
return (
<div>
<MDXProvider
components={{
Foo({ children }) {
return (
<p>
this is the foo component with <span>{children}</span>
</p>
);
},
wrapper(props) {
return <div style={{ backgroundColor: 'lightblue' }} {...props} />;
},
}}
>
<h1>xdm testing</h1>
<h2>rendered</h2>
<MDXContent />
<h2>function body</h2>
<pre>
<code>{code}</code>
</pre>
</MDXProvider>
</div>
);
}
export async function getStaticProps() {
const code = await compileMDXFunction(
`
h1 hello mdx
This is ~~some GFM content~~
<Foo>content</Foo>
`
);
return {
props: {
code,
},
};
} The resulting view: |
Nice!
// …
<div>
<h1>xdm testing</h1>
<h2>rendered</h2>
<MDXContent
components={{
Foo({ children }) {
return (
<p>
this is the foo component with <span>{children}</span>
</p>
);
},
wrapper(props) {
return <div style={{ backgroundColor: 'lightblue' }} {...props} />;
},
}}
/>
<h2>function body</h2>
<pre>
<code>{code}</code>
</pre>
</div>
// …
export async function compileMDXFunction(mdx) {
return String(await compile(mdx, {
_contain: true,
providerImportSource: #',
remarkPlugins: [remarkGfm],
}))
}
|
Yep, that's what I figured from looking at the output, I just put
Good tip, I missed that one! 😄
That would be a really nice addition. I'm not familiar with how this would work but maybe I'll find time to dig in to it at some point. Anyway thanks again, this worked really well. FWIW I found a couple of caveats with Next.js, because you have to tell it to transpile ESM dependencies specifically (I had to use the As for the API, I think the |
P.S. Let me know, if you'd like me to contribute to the README about this use-case |
You can also probably use terser outside of unified/xdm. Take the string, use terser and probably configure it to support top-level return statements (if possible), and get a minified output.
That’s an issue that Next needs to solve. The ecosystem is moving soon (unifiedjs/unified#121 (comment)), and they don’t support it yet.
RSC, which is far from ready but Next is also working on, solves this.
It definitely needs a better name. I somewhat like I also need to figure out how to make
Aside: I think the function you have now is more complex that needed. You’re including 1kb of JS to get a provider, so you can do Also, I don’t get the Other than these two thought, I think those functions can live in userland! |
How about
Yes I probably don't need it. I guess I was aiming for parity with existing solutions/patterns, Gatsby etc. Maybe I'll make this an option in my compiler function which defaults to
I took this from KCD's README for mdx-bundler, he shows usage of his
That's an interesting thought: so you mean write a hook that uses
For sure. I'm thinking about writing a post on dev.to about this because I know this use-case is a thing for Next.js users, and there's a need for a really up-to-date solution for both file-/(module-) and string/(function-)based MDX sources. |
That’s a great idea, much better! Taking it further, how about The word “program” is used by estree (the JS AST used by Firefox, Babel, ESLint, much more) to represent the whole. The difference between whether such a program is a module or a script, depends on the environment: I also think that Then the next thing to do would be to split
Then
I think so. It could be its own little module. You can publish it, too 😅. It gets such a “function-body” from xdm as a
Nice! Yeah, maybe it’s a small hook. A couple lines. Then you don’t need to publish it, people can just copy-paste it in. |
This exposes the currently internal `_contain` option in the public interface, which is used by `evaluate`, so that users can depend on it too. Related to GH-23
This exposes the currently internal `_contain` option in the public interface, which is used by `evaluate`, so that users can depend on it too. Related to GH-23
Split previously experimental into two stable options: * `useDynamicImport` — compile import statements into dynamic import expressions * `baseUrl` — resolve relative import specifiers from a given URL This additionally fixes two bugs in the `'function-body'` output format: * `export * from 'a'` was not supported * `export {a as b}` was inverted Closes GH-23. Related-to GH-26.
Split previously experimental into two stable options: * `useDynamicImport` — compile import statements into dynamic import expressions * `baseUrl` — resolve relative import specifiers from a given URL This additionally fixes two bugs in the `'function-body'` output format: * `export * from 'a'` was not supported * `export {a as b}` was inverted Closes GH-23. Related-to GH-26. Closes GH-30.
For minification, I landed a PR in terser to add support for accepting and yielding our AST (ESTree). import {compile} from './index.js'
import {minify} from 'terser'
var code = `export var Thing = () => <>World!</>
# Hello, <Thing />
`
console.log(String(await compile(code)))
console.log(String(await compile(code, {recmaPlugins: [recmaMinify]})))
function recmaMinify() {
return transform
async function transform(tree) {
return (
await minify(tree, {
parse: {spidermonkey: true},
format: {spidermonkey: true, code: false}
})
).ast
}
} Yields: /*@jsxRuntime automatic @jsxImportSource react*/
import {Fragment as _Fragment, jsx as _jsx, jsxs as _jsxs} from "react/jsx-runtime";
export var Thing = () => _jsx(_Fragment, {
children: "World!"
});
function MDXContent(props) {
const _components = Object.assign({
h1: "h1"
}, props.components), {wrapper: MDXLayout} = _components;
const _content = _jsx(_Fragment, {
children: _jsxs(_components.h1, {
children: ["Hello, ", _jsx(Thing, {})]
})
});
return MDXLayout ? _jsx(MDXLayout, Object.assign({}, props, {
children: _content
})) : _content;
}
export default MDXContent; import {Fragment as _Fragment, jsx as _jsx, jsxs as _jsxs} from "react/jsx-runtime";
export var Thing = () => {
return _jsx(_Fragment, {
children: "World!"
});
};
function MDXContent(n) {
const t = Object.assign({
h1: "h1"
}, n.components), {wrapper: MDXLayout} = t, s = _jsx(_Fragment, {
children: _jsxs(t.h1, {
children: ["Hello, ", _jsx(Thing, {})]
})
});
return MDXLayout ? _jsx(MDXLayout, Object.assign({}, n, {
children: s
})) : s;
}
export default MDXContent; Note that this minifies props and such. This is not a formatter. If you also want to format, it becomes a bit more complex. A nice alternative is running esbuild after xdm, which is super fast and can do all that too |
@wooorm thanks for this update! Interesting that you mention Otherwise, I was thinking of doing a package specifically for the Next.js use-case (something like |
You can use esbuild both to build xdm into a CJS bundle, and to run it on the results of xdm. |
I just found this project and I'm excited about moving past some of the issues that seem to have stalled at
@mdx-js
, but I'm not sure how to work with MDX source code that doesn't come from a file.I'm working with MDX content from a CMS, so I'm doing a two-step process:
@mdx-js/mdx
and@babel/core
)MDXRenderer
component (which I also wrote myself, but the API is modelled ongatsby-plugin-mdx
), which takes the code (as well as optionalcomponents
prop), to execute it as a function.Can I do this with
xdm
? The documentation seems aimed at.mdx
files which are imported and thus compiled to modules, but I need a component function. My MDX source strings won't contain any import or export rules, but they will contain components that I'll need to pass in through a context provider or to theircomponents
prop.The text was updated successfully, but these errors were encountered: