* wip * stash * fix: make authenticationId nullable fk * fix: apply generics to resolve compile time type errors * fix: loosen integration settings * chore: refactor into functional component * feat: pass integrations all the way to embeds * perf: avoid re-fetching integrations * fix: change attr name to avoid type overlap * feat: use hostname from embed settings in matcher * Revert "feat: use hostname from embed settings in matcher" This reverts commit e7485d9cda4dcf45104e460465ca104a56c67ddc. * feat: refactor into a class * chore: refactor url regex formation as a util * fix: escape regex special chars * fix: remove in-house escapeRegExp in favor of lodash's * fix: sanitize url * perf: memoize embeds * fix: rename hostname to url and allow spreading entire settings instead of just url * fix: replace diagrams with drawio * fix: rename * fix: support self-hosted and saas both * fix: assert on settings url * fix: move embed integrations loading to hook * fix: address review comments * fix: use observer in favor of explicit state setters * fix: refactor useEmbedIntegrations into useEmbeds * fix: use translations for toasts Co-authored-by: Tom Moor <tom.moor@gmail.com>
97 lines
2.5 KiB
TypeScript
97 lines
2.5 KiB
TypeScript
import MarkdownIt from "markdown-it";
|
||
import Token from "markdown-it/lib/token";
|
||
import { EmbedDescriptor } from "../embeds";
|
||
|
||
function isParagraph(token: Token) {
|
||
return token.type === "paragraph_open";
|
||
}
|
||
|
||
function isInline(token: Token) {
|
||
return token.type === "inline" && token.level === 1;
|
||
}
|
||
|
||
function isLinkOpen(token: Token) {
|
||
return token.type === "link_open";
|
||
}
|
||
|
||
function isLinkClose(token: Token) {
|
||
return token.type === "link_close";
|
||
}
|
||
|
||
export default function linksToEmbeds(embeds: EmbedDescriptor[]) {
|
||
function isEmbed(token: Token, link: Token) {
|
||
const href = link.attrs ? link.attrs[0][1] : "";
|
||
const simpleLink = href === token.content;
|
||
|
||
if (!simpleLink) {
|
||
return false;
|
||
}
|
||
if (!embeds) {
|
||
return false;
|
||
}
|
||
|
||
for (const embed of embeds) {
|
||
const matches = embed.matcher(href);
|
||
if (matches) {
|
||
return {
|
||
...embed,
|
||
matches,
|
||
};
|
||
}
|
||
}
|
||
|
||
return false;
|
||
}
|
||
|
||
return function markdownEmbeds(md: MarkdownIt) {
|
||
md.core.ruler.after("inline", "embeds", (state) => {
|
||
const tokens = state.tokens;
|
||
let insideLink;
|
||
|
||
for (let i = 0; i < tokens.length - 1; i++) {
|
||
// once we find an inline token look through it's children for links
|
||
if (isInline(tokens[i]) && isParagraph(tokens[i - 1])) {
|
||
const tokenChildren = tokens[i].children || [];
|
||
|
||
for (let j = 0; j < tokenChildren.length - 1; j++) {
|
||
const current = tokenChildren[j];
|
||
if (!current) {
|
||
continue;
|
||
}
|
||
|
||
if (isLinkOpen(current)) {
|
||
insideLink = current;
|
||
continue;
|
||
}
|
||
|
||
if (isLinkClose(current)) {
|
||
insideLink = null;
|
||
continue;
|
||
}
|
||
|
||
// of hey, we found a link – lets check to see if it should be
|
||
// converted to an embed
|
||
if (insideLink) {
|
||
const result = isEmbed(current, insideLink);
|
||
if (result) {
|
||
const { content } = current;
|
||
|
||
// convert to embed token
|
||
const token = new Token("embed", "iframe", 0);
|
||
token.attrSet("href", content);
|
||
|
||
// delete the inline link – this makes the assumption that the
|
||
// embed is the only thing in the para.
|
||
tokens.splice(i - 1, 3, token);
|
||
break;
|
||
}
|
||
}
|
||
}
|
||
}
|
||
}
|
||
|
||
return false;
|
||
});
|
||
};
|
||
}
|