UNPKG

remark-oembed

Version:

Converts URLs surrounded by newlines into embeds that are loaded asynchronously

385 lines (321 loc) 10.7 kB
const Got = require('got'); const h = require('hastscript'); const Intercept = require('apr-intercept'); const { getType } = require('mime'); const { paramCase: ParamCase } = require('param-case'); const toHtml = require('hast-util-to-html'); const { format, parse } = require('url'); const IsUrl = require('is-url'); const OEMBED_PROVIDERS_URL = 'https://oembed.com/providers.json'; const EMPTY_CANVAS = ''; const ATTRS = { template: { 'data-oembed-template': true, }, inline: (provider = '') => ({ class: `remark-oembed-inline remark-oembed-${ParamCase(provider)}`, 'data-oembed': true, }), image: { class: 'remark-oembed-photo', 'data-oembed': true, }, anchor: { class: 'remark-oembed-anchor', 'data-oembed': true, }, }; const Script = () => { const script = ` document.querySelectorAll('div[data-oembed]').forEach((el) => { const template = el.querySelector('[data-oembed-template]').content.cloneNode(true); el.innerHTML = ''; el.attachShadow({ mode: 'closed' }).appendChild(template); }); document.querySelectorAll('img[data-oembed][data-src]').forEach((img) => { img.setAttribute('src', img.getAttribute('data-src')); img.removeAttribute('data-src'); }); document.querySelectorAll('[data-oembed]').forEach((el) => { el.removeAttribute('data-oembed'); }); `; return ` const isDocumentReady = () => { if (document.readyState !== 'complete') { document.addEventListener('readystatechange', isDocumentReady); return false; } requestAnimationFrame(() => { ${script}; }); return true; }; if (!isDocumentReady()) { document.addEventListener('readystatechange', isDocumentReady); } `; }; const RequestAnimationFrame = ({ script: __html }) => { return h('script', { async: true, defer: true }, [__html]); }; const JsxScript = ({ script: __html }) => { return `<script defer async dangerouslySetInnerHTML={{ __html: \`${__html}\` }} />`; }; const Img = ({ id, url, title, width, height, dataSrc }) => { return h('img', { id, src: url, 'data-src': dataSrc, title, width, height, ...ATTRS.image, }); }; const Anchor = (props, children = []) => { const { rel = 'noopener noreferrer nofollow', target = '_blank', ...rest } = props; return h( 'a', Object.assign(rest, ATTRS.anchor, { rel, target }), children.filter(Boolean), ); }; const StaticPhotoOembed = ({ emptyUrl, url, href, jsx = false, ...rest }) => { const src = emptyUrl || url; const isImage = /^image\//.test(getType(href) || ''); const img = Img({ ...rest, url: src, dataSrc: url }); const anchor = isImage ? img : Anchor({ href }, [img]); if (!jsx) { return anchor; } const html = toHtml(anchor, { allowDangerousHtml: true, }); return { type: 'jsx', properties: { dataOembed: true }, value: `<wrapper dangerouslySetInnerHTML={{ __html: \`${html}\` }} />`, }; }; const resolvePreview = ({ isImage, thumbnail, url, emptyUrl }) => { if (isImage) { return emptyUrl || url; } if (thumbnail.isImage) { const { url, emptyUrl } = thumbnail; return emptyUrl || url; } return ''; }; const Oembed = ({ type, source, jsx = false, provider_name, ...rest }) => { if (type === 'photo' && !source) { return StaticPhotoOembed({ jsx, ...rest }); } const imgSrc = resolvePreview(rest); const img = imgSrc ? Img({ ...rest, url: imgSrc }) : ''; const anchor = img ? Anchor({ href: rest.href }, [img]) : img; const node = h( 'div', ATTRS.inline(provider_name), [anchor, h('template', ATTRS.template, [source])].filter(Boolean), ); if (!jsx) { return node; } const html = toHtml(node, { allowDangerousHtml: true, }); return { type: 'jsx', properties: { dataOembed: true }, value: `<wrapper dangerouslySetInnerHTML={{ __html: \`${html}\` }} />`, }; }; const transformImage = async (ctx = {}) => { const { url, width, height, asyncImg = false } = ctx; if (!url) { return {}; } const isImage = /^image\//.test(getType(parse(url).pathname) || ''); if (!isImage) { return {}; } return { isImage: true, width, height, emptyUrl: asyncImg ? EMPTY_CANVAS : '', url, }; }; const processOembed = async (oembed) => { const { source, url, syncWidget = false, jsx = false } = oembed; const isImage = /^image\//.test(getType(url) || ''); if (!isImage && !source) { return; } if (syncWidget && source) { const { provider_name } = oembed; const newNode = h('div', ATTRS.inline(provider_name), [source]); if (!jsx) { return newNode; } const html = toHtml(newNode, { allowDangerousHtml: true, }); return { type: 'jsx', value: `<wrapper dangerouslySetInnerHTML={{ __html: \`${html}\` }} />`, }; } return Oembed({ ...oembed, ...(await transformImage(oembed)), thumbnail: await transformImage({ ...oembed, url: oembed.thumbnail_url, height: oembed.height || oembed.thumbnail_height, width: oembed.width || oembed.thumbnail_width, }), }); }; const fetchOembed = async (endpoint) => { const href = format( Object.assign(parse(endpoint.url), { query: { format: 'json', ...endpoint.query, }, }), ); return Got(href, { responseType: 'json', resolveBodyOnly: true, }); }; const getProviderEndpoint = (linkUrl, providers = []) => { for (const provider of providers || []) { for (const endpoint of provider.endpoints || []) { for (let schema of endpoint.schemes || []) { schema = schema.replace('*', '.*'); const regExp = new RegExp(schema); if (regExp.test(linkUrl)) { return { url: endpoint.url, query: { url: linkUrl, ...endpoint.params, }, }; } } } } }; const processNode = async (node, { providers = [], ...options }) => { const child = node.children.slice(0, 1)[0]; if (!child || !child.url) { return node; } const endpoint = getProviderEndpoint(child.url, providers); if (!endpoint) { return node; } const { html, ...rest } = await fetchOembed(endpoint); const source = html ? { type: 'raw', value: html } : undefined; const oembed = { ...rest, ...options, source, href: child.url }; const newNode = await processOembed(oembed); return newNode ? newNode : node; }; const fetchOembedProviders = async () => { return Got(OEMBED_PROVIDERS_URL, { responseType: 'json', resolveBodyOnly: true, }); }; // https://github.com/syntax-tree/unist-util-map/blob/bb0567f651517b2d521af711d7376475b3d8446a/index.js const map = async (tree, iteratee) => { const bound = (node) => async (child, index) => { return preorder(child, index, node); }; const preorder = async (node, index, parent) => { const [, newNode = {}] = await Intercept(iteratee(node, index, parent)); const { children = [] } = newNode || node; return { ...node, ...newNode, children: await Promise.all(children.map(bound(node))), }; }; return preorder(tree, null, null); }; module.exports = (opts = {}) => { const { jsx = false } = opts; return async (tree) => { const providers = await fetchOembedProviders(); const ctx = { ...opts, providers, }; const newTree = await map(tree, async (node) => { const isParagraph = node.type === 'paragraph'; const hasChildren = Array.isArray(node.children); if (!isParagraph || !hasChildren) { return node; } const children = node.children.map((node) => { if (node.type === 'link') { return node; } return IsUrl(node.value) ? Object.assign(node, { url: node.value }) : node; }); const onlyChild = node.children.length === 1; const hasLinkChild = children.every(({ url }) => url); if (!onlyChild || !hasLinkChild) { return node; } return processNode(Object.assign(node, { children }), ctx); }); let count = 0; const newSerializedTree = await map(newTree, async (node) => { const isOembed = ((node || {}).properties || {}).dataOembed; if (isOembed) { count += 1; } if (!isOembed || jsx) { return node; } return { type: 'html', value: toHtml(node, { allowDangerousHtml: true, }), }; }); if (ctx.syncWidget || !count) { return newSerializedTree; } const script = Script(); const { children = [] } = newSerializedTree; return Object.assign(newSerializedTree, { children: children.concat([ { type: jsx ? 'jsx' : 'html', value: jsx ? JsxScript({ script }) : toHtml(RequestAnimationFrame({ script }), { allowDangerousHtml: true, }), }, ]), }); }; };