diff --git a/blocksuite/affine/block-embed/src/common/adapters/notion-html.ts b/blocksuite/affine/block-embed/src/common/adapters/notion-html.ts new file mode 100644 index 0000000000000..1fd9b6d1b52fa --- /dev/null +++ b/blocksuite/affine/block-embed/src/common/adapters/notion-html.ts @@ -0,0 +1,89 @@ +import { + type BlockNotionHtmlAdapterMatcher, + HastUtils, +} from '@blocksuite/affine-shared/adapters'; +import { nanoid } from '@blocksuite/store'; + +export function createEmbedBlockNotionHtmlAdapterMatcher( + flavour: string, + urlRegex: RegExp, + { + toMatch = o => { + const isFigure = + HastUtils.isElement(o.node) && o.node.tagName === 'figure'; + const embededFigureWrapper = HastUtils.querySelector(o.node, '.source'); + if (!isFigure || !embededFigureWrapper) { + return false; + } + const embededURL = HastUtils.querySelector(embededFigureWrapper, 'a') + ?.properties.href; + + if (!embededURL || typeof embededURL !== 'string') { + return false; + } + // To avoid polynomial regular expression used on uncontrolled data + // https://codeql.github.com/codeql-query-help/javascript/js-polynomial-redos/ + if (embededURL.length > 1000) { + return false; + } + return urlRegex.test(embededURL); + }, + fromMatch = o => o.node.flavour === flavour, + toBlockSnapshot = { + enter: (o, context) => { + if (!HastUtils.isElement(o.node)) { + return; + } + const { assets, walkerContext } = context; + if (!assets) { + return; + } + + const embededFigureWrapper = HastUtils.querySelector(o.node, '.source'); + if (!embededFigureWrapper) { + return; + } + + let embededURL = ''; + const embedA = HastUtils.querySelector(embededFigureWrapper, 'a'); + embededURL = + typeof embedA?.properties.href === 'string' + ? embedA.properties.href + : ''; + if (!embededURL) { + return; + } + + walkerContext + .openNode( + { + type: 'block', + id: nanoid(), + flavour, + props: { + url: embededURL, + }, + children: [], + }, + 'children' + ) + .closeNode(); + walkerContext.skipAllChildren(); + }, + }, + fromBlockSnapshot = {}, + }: { + toMatch?: BlockNotionHtmlAdapterMatcher['toMatch']; + fromMatch?: BlockNotionHtmlAdapterMatcher['fromMatch']; + toBlockSnapshot?: BlockNotionHtmlAdapterMatcher['toBlockSnapshot']; + fromBlockSnapshot?: BlockNotionHtmlAdapterMatcher['fromBlockSnapshot']; + } = Object.create(null) +): BlockNotionHtmlAdapterMatcher { + return { + flavour, + toMatch, + fromMatch, + toBlockSnapshot, + fromBlockSnapshot, + }; +} diff --git a/blocksuite/affine/block-embed/src/embed-figma-block/adapters/extension.ts b/blocksuite/affine/block-embed/src/embed-figma-block/adapters/extension.ts index 1b8017c5a4e8d..831deba680461 100644 --- a/blocksuite/affine/block-embed/src/embed-figma-block/adapters/extension.ts +++ b/blocksuite/affine/block-embed/src/embed-figma-block/adapters/extension.ts @@ -2,10 +2,12 @@ import type { ExtensionType } from '@blocksuite/block-std'; import { EmbedFigmaBlockHtmlAdapterExtension } from './html.js'; import { EmbedFigmaMarkdownAdapterExtension } from './markdown.js'; +import { EmbedFigmaNotionHtmlAdapterExtension } from './notion-html.js'; import { EmbedFigmaBlockPlainTextAdapterExtension } from './plain-text.js'; export const EmbedFigmaBlockAdapterExtensions: ExtensionType[] = [ EmbedFigmaBlockHtmlAdapterExtension, EmbedFigmaMarkdownAdapterExtension, EmbedFigmaBlockPlainTextAdapterExtension, + EmbedFigmaNotionHtmlAdapterExtension, ]; diff --git a/blocksuite/affine/block-embed/src/embed-figma-block/adapters/index.ts b/blocksuite/affine/block-embed/src/embed-figma-block/adapters/index.ts index c1d476903d99b..b4dd5a6d2a7e6 100644 --- a/blocksuite/affine/block-embed/src/embed-figma-block/adapters/index.ts +++ b/blocksuite/affine/block-embed/src/embed-figma-block/adapters/index.ts @@ -1,3 +1,4 @@ export * from './html.js'; export * from './markdown.js'; +export * from './notion-html.js'; export * from './plain-text.js'; diff --git a/blocksuite/affine/block-embed/src/embed-figma-block/adapters/notion-html.ts b/blocksuite/affine/block-embed/src/embed-figma-block/adapters/notion-html.ts new file mode 100644 index 0000000000000..53119d95792d4 --- /dev/null +++ b/blocksuite/affine/block-embed/src/embed-figma-block/adapters/notion-html.ts @@ -0,0 +1,14 @@ +import { EmbedFigmaBlockSchema } from '@blocksuite/affine-model'; +import { BlockNotionHtmlAdapterExtension } from '@blocksuite/affine-shared/adapters'; + +import { createEmbedBlockNotionHtmlAdapterMatcher } from '../../common/adapters/notion-html.js'; +import { figmaUrlRegex } from '../embed-figma-model.js'; + +export const embedFigmaBlockNotionHtmlAdapterMatcher = + createEmbedBlockNotionHtmlAdapterMatcher( + EmbedFigmaBlockSchema.model.flavour, + figmaUrlRegex + ); + +export const EmbedFigmaNotionHtmlAdapterExtension = + BlockNotionHtmlAdapterExtension(embedFigmaBlockNotionHtmlAdapterMatcher); diff --git a/blocksuite/affine/block-embed/src/embed-github-block/adapters/extension.ts b/blocksuite/affine/block-embed/src/embed-github-block/adapters/extension.ts index 5c6fa5cb57be6..3f548aedcb914 100644 --- a/blocksuite/affine/block-embed/src/embed-github-block/adapters/extension.ts +++ b/blocksuite/affine/block-embed/src/embed-github-block/adapters/extension.ts @@ -2,10 +2,12 @@ import type { ExtensionType } from '@blocksuite/block-std'; import { EmbedGithubBlockHtmlAdapterExtension } from './html.js'; import { EmbedGithubMarkdownAdapterExtension } from './markdown.js'; +import { EmbedGithubNotionHtmlAdapterExtension } from './notion-html.js'; import { EmbedGithubBlockPlainTextAdapterExtension } from './plain-text.js'; export const EmbedGithubBlockAdapterExtensions: ExtensionType[] = [ EmbedGithubBlockHtmlAdapterExtension, EmbedGithubMarkdownAdapterExtension, EmbedGithubBlockPlainTextAdapterExtension, + EmbedGithubNotionHtmlAdapterExtension, ]; diff --git a/blocksuite/affine/block-embed/src/embed-github-block/adapters/index.ts b/blocksuite/affine/block-embed/src/embed-github-block/adapters/index.ts index c1d476903d99b..b4dd5a6d2a7e6 100644 --- a/blocksuite/affine/block-embed/src/embed-github-block/adapters/index.ts +++ b/blocksuite/affine/block-embed/src/embed-github-block/adapters/index.ts @@ -1,3 +1,4 @@ export * from './html.js'; export * from './markdown.js'; +export * from './notion-html.js'; export * from './plain-text.js'; diff --git a/blocksuite/affine/block-embed/src/embed-github-block/adapters/notion-html.ts b/blocksuite/affine/block-embed/src/embed-github-block/adapters/notion-html.ts new file mode 100644 index 0000000000000..20022c2345abf --- /dev/null +++ b/blocksuite/affine/block-embed/src/embed-github-block/adapters/notion-html.ts @@ -0,0 +1,14 @@ +import { EmbedGithubBlockSchema } from '@blocksuite/affine-model'; +import { BlockNotionHtmlAdapterExtension } from '@blocksuite/affine-shared/adapters'; + +import { createEmbedBlockNotionHtmlAdapterMatcher } from '../../common/adapters/notion-html.js'; +import { githubUrlRegex } from '../embed-github-model.js'; + +export const embedGithubBlockNotionHtmlAdapterMatcher = + createEmbedBlockNotionHtmlAdapterMatcher( + EmbedGithubBlockSchema.model.flavour, + githubUrlRegex + ); + +export const EmbedGithubNotionHtmlAdapterExtension = + BlockNotionHtmlAdapterExtension(embedGithubBlockNotionHtmlAdapterMatcher); diff --git a/blocksuite/affine/block-embed/src/embed-loom-block/adapters/extension.ts b/blocksuite/affine/block-embed/src/embed-loom-block/adapters/extension.ts index 13afbf167cafd..d742f5c575f91 100644 --- a/blocksuite/affine/block-embed/src/embed-loom-block/adapters/extension.ts +++ b/blocksuite/affine/block-embed/src/embed-loom-block/adapters/extension.ts @@ -2,10 +2,12 @@ import type { ExtensionType } from '@blocksuite/block-std'; import { EmbedLoomBlockHtmlAdapterExtension } from './html.js'; import { EmbedLoomMarkdownAdapterExtension } from './markdown.js'; +import { EmbedLoomNotionHtmlAdapterExtension } from './notion-html.js'; import { EmbedLoomBlockPlainTextAdapterExtension } from './plain-text.js'; export const EmbedLoomBlockAdapterExtensions: ExtensionType[] = [ EmbedLoomBlockHtmlAdapterExtension, EmbedLoomMarkdownAdapterExtension, EmbedLoomBlockPlainTextAdapterExtension, + EmbedLoomNotionHtmlAdapterExtension, ]; diff --git a/blocksuite/affine/block-embed/src/embed-loom-block/adapters/index.ts b/blocksuite/affine/block-embed/src/embed-loom-block/adapters/index.ts index c1d476903d99b..b4dd5a6d2a7e6 100644 --- a/blocksuite/affine/block-embed/src/embed-loom-block/adapters/index.ts +++ b/blocksuite/affine/block-embed/src/embed-loom-block/adapters/index.ts @@ -1,3 +1,4 @@ export * from './html.js'; export * from './markdown.js'; +export * from './notion-html.js'; export * from './plain-text.js'; diff --git a/blocksuite/affine/block-embed/src/embed-loom-block/adapters/notion-html.ts b/blocksuite/affine/block-embed/src/embed-loom-block/adapters/notion-html.ts new file mode 100644 index 0000000000000..db06ca0b57193 --- /dev/null +++ b/blocksuite/affine/block-embed/src/embed-loom-block/adapters/notion-html.ts @@ -0,0 +1,14 @@ +import { EmbedLoomBlockSchema } from '@blocksuite/affine-model'; +import { BlockNotionHtmlAdapterExtension } from '@blocksuite/affine-shared/adapters'; + +import { createEmbedBlockNotionHtmlAdapterMatcher } from '../../common/adapters/notion-html.js'; +import { loomUrlRegex } from '../embed-loom-model.js'; + +export const embedLoomBlockNotionHtmlAdapterMatcher = + createEmbedBlockNotionHtmlAdapterMatcher( + EmbedLoomBlockSchema.model.flavour, + loomUrlRegex + ); + +export const EmbedLoomNotionHtmlAdapterExtension = + BlockNotionHtmlAdapterExtension(embedLoomBlockNotionHtmlAdapterMatcher); diff --git a/blocksuite/affine/block-embed/src/embed-youtube-block/adapters/extension.ts b/blocksuite/affine/block-embed/src/embed-youtube-block/adapters/extension.ts index 8ef9c011bb6e0..f3f6bb2489edd 100644 --- a/blocksuite/affine/block-embed/src/embed-youtube-block/adapters/extension.ts +++ b/blocksuite/affine/block-embed/src/embed-youtube-block/adapters/extension.ts @@ -2,10 +2,12 @@ import type { ExtensionType } from '@blocksuite/block-std'; import { EmbedYoutubeBlockHtmlAdapterExtension } from './html.js'; import { EmbedYoutubeMarkdownAdapterExtension } from './markdown.js'; +import { EmbedYoutubeNotionHtmlAdapterExtension } from './notion-html.js'; import { EmbedYoutubeBlockPlainTextAdapterExtension } from './plain-text.js'; export const EmbedYoutubeBlockAdapterExtensions: ExtensionType[] = [ EmbedYoutubeBlockHtmlAdapterExtension, EmbedYoutubeMarkdownAdapterExtension, EmbedYoutubeBlockPlainTextAdapterExtension, + EmbedYoutubeNotionHtmlAdapterExtension, ]; diff --git a/blocksuite/affine/block-embed/src/embed-youtube-block/adapters/index.ts b/blocksuite/affine/block-embed/src/embed-youtube-block/adapters/index.ts index c1d476903d99b..b4dd5a6d2a7e6 100644 --- a/blocksuite/affine/block-embed/src/embed-youtube-block/adapters/index.ts +++ b/blocksuite/affine/block-embed/src/embed-youtube-block/adapters/index.ts @@ -1,3 +1,4 @@ export * from './html.js'; export * from './markdown.js'; +export * from './notion-html.js'; export * from './plain-text.js'; diff --git a/blocksuite/affine/block-embed/src/embed-youtube-block/adapters/notion-html.ts b/blocksuite/affine/block-embed/src/embed-youtube-block/adapters/notion-html.ts new file mode 100644 index 0000000000000..e4da1a9e3c037 --- /dev/null +++ b/blocksuite/affine/block-embed/src/embed-youtube-block/adapters/notion-html.ts @@ -0,0 +1,14 @@ +import { EmbedYoutubeBlockSchema } from '@blocksuite/affine-model'; +import { BlockNotionHtmlAdapterExtension } from '@blocksuite/affine-shared/adapters'; + +import { createEmbedBlockNotionHtmlAdapterMatcher } from '../../common/adapters/notion-html.js'; +import { youtubeUrlRegex } from '../embed-youtube-model.js'; + +export const embedYoutubeBlockNotionHtmlAdapterMatcher = + createEmbedBlockNotionHtmlAdapterMatcher( + EmbedYoutubeBlockSchema.model.flavour, + youtubeUrlRegex + ); + +export const EmbedYoutubeNotionHtmlAdapterExtension = + BlockNotionHtmlAdapterExtension(embedYoutubeBlockNotionHtmlAdapterMatcher); diff --git a/blocksuite/blocks/src/_common/adapters/notion-html/block-matcher.ts b/blocksuite/blocks/src/_common/adapters/notion-html/block-matcher.ts index 6013a505d2fea..2450c62b8eef6 100644 --- a/blocksuite/blocks/src/_common/adapters/notion-html/block-matcher.ts +++ b/blocksuite/blocks/src/_common/adapters/notion-html/block-matcher.ts @@ -1,3 +1,9 @@ +import { + embedFigmaBlockNotionHtmlAdapterMatcher, + embedGithubBlockNotionHtmlAdapterMatcher, + embedLoomBlockNotionHtmlAdapterMatcher, + embedYoutubeBlockNotionHtmlAdapterMatcher, +} from '@blocksuite/affine-block-embed'; import { listBlockNotionHtmlAdapterMatcher } from '@blocksuite/affine-block-list'; import { paragraphBlockNotionHtmlAdapterMatcher } from '@blocksuite/affine-block-paragraph'; import type { BlockNotionHtmlAdapterMatcher } from '@blocksuite/affine-shared/adapters'; @@ -21,6 +27,10 @@ export const defaultBlockNotionHtmlAdapterMatchers: BlockNotionHtmlAdapterMatche rootBlockNotionHtmlAdapterMatcher, bookmarkBlockNotionHtmlAdapterMatcher, databaseBlockNotionHtmlAdapterMatcher, - attachmentBlockNotionHtmlAdapterMatcher, latexBlockNotionHtmlAdapterMatcher, + embedYoutubeBlockNotionHtmlAdapterMatcher, + embedFigmaBlockNotionHtmlAdapterMatcher, + embedGithubBlockNotionHtmlAdapterMatcher, + embedLoomBlockNotionHtmlAdapterMatcher, + attachmentBlockNotionHtmlAdapterMatcher, ];