@ekrist1/vulse 0.1.6-alpha.3 → 0.1.7-alpha.4
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/cli/migrate.d.ts.map +1 -1
- package/dist/cli/migrate.js +3 -4
- package/dist/cli/setup.d.ts.map +1 -1
- package/dist/cli/setup.js +11 -10
- package/dist/core/blueprints/compile.d.ts +1 -1
- package/dist/core/blueprints/compile.d.ts.map +1 -1
- package/dist/core/blueprints/compile.js +3 -3
- package/dist/core/blueprints/mutations.js +2 -2
- package/dist/core/forms/rate-limit.d.ts +1 -1
- package/dist/core/forms/rate-limit.d.ts.map +1 -1
- package/dist/core/forms/rate-limit.js +3 -3
- package/dist/core/forms/unique.d.ts +1 -1
- package/dist/core/forms/unique.d.ts.map +1 -1
- package/dist/core/forms/unique.js +4 -4
- package/dist/core/globals/compile.d.ts +1 -1
- package/dist/core/globals/compile.d.ts.map +1 -1
- package/dist/core/globals/compile.js +2 -2
- package/dist/core/globals/definition.d.ts +1 -1
- package/dist/core/globals/definition.d.ts.map +1 -1
- package/dist/core/globals/definition.js +3 -3
- package/dist/core/repos/globals.js +3 -3
- package/dist/core/sha256.d.ts +3 -0
- package/dist/core/sha256.d.ts.map +1 -0
- package/dist/core/sha256.js +9 -0
- package/dist/integration/index.js +1 -1
- package/dist/integration/install-hook.d.ts.map +1 -1
- package/dist/integration/install-hook.js +6 -4
- package/dist/integration/wrangler-config.d.ts +12 -0
- package/dist/integration/wrangler-config.d.ts.map +1 -0
- package/dist/integration/wrangler-config.js +97 -0
- package/dist/integration/wrangler-patch.d.ts +1 -0
- package/dist/integration/wrangler-patch.d.ts.map +1 -1
- package/dist/integration/wrangler-patch.js +2 -1
- package/dist/server/routes/form-submit.js +1 -1
- package/dist/version.d.ts +1 -1
- package/dist/version.js +1 -1
- package/package.json +11 -3
- package/src/admin/assets/logo-mark.svg +5 -0
- package/src/admin/client/active-locale.ts +17 -0
- package/src/admin/client/api.ts +21 -0
- package/src/admin/client/form-from-zod.ts +7 -0
- package/src/admin/client/live-preview-enabled.ts +5 -0
- package/src/admin/components/AdminShell.astro +45 -0
- package/src/admin/components/AuthSettings.vue +60 -0
- package/src/admin/components/BlockEditor.vue +53 -0
- package/src/admin/components/BlueprintEditor.vue +1783 -0
- package/src/admin/components/CollectionKindIcon.vue +26 -0
- package/src/admin/components/CollectionTree.vue +220 -0
- package/src/admin/components/EntryEditorWithPreview.vue +130 -0
- package/src/admin/components/EntryForm.vue +411 -0
- package/src/admin/components/EntryList.vue +121 -0
- package/src/admin/components/EntryStatusBadge.vue +24 -0
- package/src/admin/components/FormEditor.vue +233 -0
- package/src/admin/components/FormList.vue +54 -0
- package/src/admin/components/GlobalSetEditor.vue +272 -0
- package/src/admin/components/GlobalSetList.vue +55 -0
- package/src/admin/components/LivePreviewPanel.vue +171 -0
- package/src/admin/components/LoginForm.vue +53 -0
- package/src/admin/components/MediaLibrary.vue +106 -0
- package/src/admin/components/MediaPicker.vue +49 -0
- package/src/admin/components/RevisionDiff.vue +11 -0
- package/src/admin/components/RevisionList.vue +134 -0
- package/src/admin/components/SeoFields.vue +113 -0
- package/src/admin/components/SetEditor.vue +137 -0
- package/src/admin/components/SetList.vue +32 -0
- package/src/admin/components/SettingsForm.vue +189 -0
- package/src/admin/components/SideNav.vue +152 -0
- package/src/admin/components/SubmissionDetail.vue +45 -0
- package/src/admin/components/SubmissionList.vue +89 -0
- package/src/admin/components/ToastHost.vue +33 -0
- package/src/admin/components/TreeRow.vue +163 -0
- package/src/admin/components/UserEditor.vue +186 -0
- package/src/admin/components/UserList.vue +46 -0
- package/src/admin/components/blocks/BlockItem.vue +32 -0
- package/src/admin/components/blocks/BlockToolbar.vue +12 -0
- package/src/admin/components/blocks/edit/CodeEdit.vue +18 -0
- package/src/admin/components/blocks/edit/EmbedEdit.vue +14 -0
- package/src/admin/components/blocks/edit/HeadingEdit.vue +19 -0
- package/src/admin/components/blocks/edit/ImageEdit.vue +40 -0
- package/src/admin/components/blocks/edit/ListEdit.vue +36 -0
- package/src/admin/components/blocks/edit/ParagraphEdit.vue +14 -0
- package/src/admin/components/blocks/edit/QuoteEdit.vue +18 -0
- package/src/admin/components/fields/BlocksField.vue +123 -0
- package/src/admin/components/fields/BlocksSetsPicker.vue +59 -0
- package/src/admin/components/fields/BoolField.vue +10 -0
- package/src/admin/components/fields/DateField.vue +22 -0
- package/src/admin/components/fields/EntriesField.vue +153 -0
- package/src/admin/components/fields/EntryField.vue +138 -0
- package/src/admin/components/fields/EnumField.vue +81 -0
- package/src/admin/components/fields/FieldRenderer.vue +87 -0
- package/src/admin/components/fields/GridField.vue +173 -0
- package/src/admin/components/fields/LinkField.vue +219 -0
- package/src/admin/components/fields/MediaField.vue +69 -0
- package/src/admin/components/fields/NumberField.vue +12 -0
- package/src/admin/components/fields/ObjectField.vue +18 -0
- package/src/admin/components/fields/RefField.vue +170 -0
- package/src/admin/components/fields/RepeaterField.vue +27 -0
- package/src/admin/components/fields/ReplicatorField.vue +121 -0
- package/src/admin/components/fields/TextField.vue +11 -0
- package/src/admin/components/fields/TextareaField.vue +11 -0
- package/src/admin/components/fields/VulseAccordionGroupNodeView.vue +82 -0
- package/src/admin/components/fields/VulseAccordionNodeView.vue +128 -0
- package/src/admin/components/fields/VulseCalloutNodeView.vue +81 -0
- package/src/admin/components/fields/VulseIframeNodeView.vue +112 -0
- package/src/admin/components/fields/VulseSetNodeView.vue +68 -0
- package/src/admin/components/fields/VulseVideoNodeView.vue +104 -0
- package/src/admin/components/fields/blocks-editor-extensions.ts +26 -0
- package/src/admin/components/fields/emoji-extension.ts +54 -0
- package/src/admin/components/fields/link-extension.ts +48 -0
- package/src/admin/components/fields/set-node-utils.ts +115 -0
- package/src/admin/components/fields/url-utils.ts +85 -0
- package/src/admin/components/fields/vulse-accordion-extension.ts +64 -0
- package/src/admin/components/fields/vulse-accordion-group-extension.ts +49 -0
- package/src/admin/components/fields/vulse-callout-extension.ts +53 -0
- package/src/admin/components/fields/vulse-iframe-extension.ts +96 -0
- package/src/admin/components/fields/vulse-set-extension.ts +66 -0
- package/src/admin/components/fields/vulse-video-extension.ts +65 -0
- package/src/admin/composables/toast.ts +35 -0
- package/src/admin/composables/useEntrySearch.ts +112 -0
- package/src/admin/composables/useSets.ts +31 -0
- package/src/admin/pages/collections/[name]/[id]/revisions.astro +27 -0
- package/src/admin/pages/collections/[name]/[id].astro +90 -0
- package/src/admin/pages/collections/[name]/index.astro +31 -0
- package/src/admin/pages/collections/[name]/new.astro +38 -0
- package/src/admin/pages/forms/[handle]/submissions/[id].astro +9 -0
- package/src/admin/pages/forms/[handle]/submissions/index.astro +9 -0
- package/src/admin/pages/forms/[handle].astro +9 -0
- package/src/admin/pages/forms/index.astro +7 -0
- package/src/admin/pages/forms/new.astro +7 -0
- package/src/admin/pages/index.astro +36 -0
- package/src/admin/pages/login.astro +14 -0
- package/src/admin/pages/media.astro +8 -0
- package/src/admin/pages/schema/[handle].astro +10 -0
- package/src/admin/pages/schema/new.astro +9 -0
- package/src/admin/pages/settings/auth.astro +9 -0
- package/src/admin/pages/settings/globals/[handle].astro +9 -0
- package/src/admin/pages/settings/globals/index.astro +7 -0
- package/src/admin/pages/settings/globals/new.astro +7 -0
- package/src/admin/pages/settings/index.astro +8 -0
- package/src/admin/pages/settings/sets/[handle].astro +9 -0
- package/src/admin/pages/settings/sets/index.astro +7 -0
- package/src/admin/pages/settings/sets/new.astro +7 -0
- package/src/admin/pages/users/[id].astro +10 -0
- package/src/admin/pages/users/index.astro +8 -0
- package/src/admin/styles/admin.css +166 -0
- package/src/core/access.ts +9 -0
- package/src/core/blocks/schema.ts +66 -0
- package/src/core/blueprints/code-to-definition.ts +156 -0
- package/src/core/blueprints/compile.ts +176 -0
- package/src/core/blueprints/define.ts +12 -0
- package/src/core/blueprints/definition.ts +185 -0
- package/src/core/blueprints/load.ts +144 -0
- package/src/core/blueprints/mutations.ts +236 -0
- package/src/core/blueprints/preview-path.ts +33 -0
- package/src/core/blueprints/reflect-fields.ts +305 -0
- package/src/core/blueprints/registry.ts +14 -0
- package/src/core/blueprints/seed.ts +20 -0
- package/src/core/blueprints/select-helpers.ts +30 -0
- package/src/core/blueprints/seo.ts +180 -0
- package/src/core/blueprints/types.ts +59 -0
- package/src/core/blueprints/zod-helpers.ts +86 -0
- package/src/core/db.ts +11 -0
- package/src/core/errors.ts +34 -0
- package/src/core/forms/compile.ts +84 -0
- package/src/core/forms/definition.ts +102 -0
- package/src/core/forms/rate-limit.ts +52 -0
- package/src/core/forms/unique.ts +38 -0
- package/src/core/globals/compile.ts +35 -0
- package/src/core/globals/definition.ts +27 -0
- package/src/core/locales.ts +45 -0
- package/src/core/migrations.ts +48 -0
- package/src/core/parse-content.ts +85 -0
- package/src/core/plugins/definition.ts +150 -0
- package/src/core/preview-content.ts +21 -0
- package/src/core/repos/entries.ts +504 -0
- package/src/core/repos/forms.ts +270 -0
- package/src/core/repos/globals.ts +179 -0
- package/src/core/repos/media.ts +106 -0
- package/src/core/repos/preview-sessions.ts +108 -0
- package/src/core/repos/revisions.ts +60 -0
- package/src/core/repos/settings.ts +23 -0
- package/src/core/schema.ts +244 -0
- package/src/core/sets/compile.ts +12 -0
- package/src/core/sets/definition.ts +10 -0
- package/src/core/sets/service.ts +82 -0
- package/src/core/sets/validate-tree.ts +57 -0
- package/src/core/sha256.ts +10 -0
- package/src/core/slug.ts +30 -0
- package/src/scaffold/collection-write.ts +83 -0
- package/src/scaffold/collection.ts +277 -0
- package/src/server/assets/live-preview-bridge.content.ts +2 -0
- package/src/server/assets/live-preview-bridge.js +535 -0
- package/src/server/better-auth.ts +82 -0
- package/src/server/cf-images.ts +34 -0
- package/src/server/cron.ts +37 -0
- package/src/server/email.ts +17 -0
- package/src/server/endpoints/api-auth.ts +10 -0
- package/src/server/endpoints/api-vulse-blueprints.ts +23 -0
- package/src/server/endpoints/api-vulse-entries-locales.ts +12 -0
- package/src/server/endpoints/api-vulse-entries-move.ts +7 -0
- package/src/server/endpoints/api-vulse-entries-publish.ts +7 -0
- package/src/server/endpoints/api-vulse-entries-tree.ts +7 -0
- package/src/server/endpoints/api-vulse-entries.ts +23 -0
- package/src/server/endpoints/api-vulse-form-handle.ts +30 -0
- package/src/server/endpoints/api-vulse-form-public.ts +7 -0
- package/src/server/endpoints/api-vulse-form-submit.ts +7 -0
- package/src/server/endpoints/api-vulse-form-upload.ts +7 -0
- package/src/server/endpoints/api-vulse-forms.ts +12 -0
- package/src/server/endpoints/api-vulse-globals-handle.ts +20 -0
- package/src/server/endpoints/api-vulse-globals-public-handle.ts +7 -0
- package/src/server/endpoints/api-vulse-globals-public.ts +7 -0
- package/src/server/endpoints/api-vulse-globals-value.ts +8 -0
- package/src/server/endpoints/api-vulse-globals.ts +12 -0
- package/src/server/endpoints/api-vulse-media-file.ts +7 -0
- package/src/server/endpoints/api-vulse-media-id.ts +12 -0
- package/src/server/endpoints/api-vulse-media.ts +12 -0
- package/src/server/endpoints/api-vulse-preview-bridge.ts +11 -0
- package/src/server/endpoints/api-vulse-preview-sessions-id.ts +10 -0
- package/src/server/endpoints/api-vulse-preview-sessions.ts +7 -0
- package/src/server/endpoints/api-vulse-preview-start.ts +7 -0
- package/src/server/endpoints/api-vulse-preview-stop.ts +7 -0
- package/src/server/endpoints/api-vulse-revisions-restore.ts +7 -0
- package/src/server/endpoints/api-vulse-revisions.ts +7 -0
- package/src/server/endpoints/api-vulse-search.ts +7 -0
- package/src/server/endpoints/api-vulse-sets.ts +23 -0
- package/src/server/endpoints/api-vulse-settings.ts +12 -0
- package/src/server/endpoints/api-vulse-users-id.ts +12 -0
- package/src/server/endpoints/api-vulse-users-reset-password.ts +7 -0
- package/src/server/endpoints/api-vulse-users-role.ts +9 -0
- package/src/server/endpoints/api-vulse-users.ts +7 -0
- package/src/server/endpoints/with-runtime.ts +11 -0
- package/src/server/env.ts +23 -0
- package/src/server/envelope.ts +21 -0
- package/src/server/forms/email.ts +11 -0
- package/src/server/forms/process-submission.ts +95 -0
- package/src/server/forms/queue.ts +25 -0
- package/src/server/forms/templates.ts +24 -0
- package/src/server/forms/webhook.ts +19 -0
- package/src/server/handler.ts +66 -0
- package/src/server/image-probe.ts +35 -0
- package/src/server/loader.ts +54 -0
- package/src/server/plugins.ts +214 -0
- package/src/server/preview.ts +25 -0
- package/src/server/r2.ts +13 -0
- package/src/server/routes/blueprints.ts +62 -0
- package/src/server/routes/entries.ts +255 -0
- package/src/server/routes/form-submit.ts +168 -0
- package/src/server/routes/form-upload.ts +100 -0
- package/src/server/routes/forms.ts +88 -0
- package/src/server/routes/globals-public.ts +30 -0
- package/src/server/routes/globals.ts +93 -0
- package/src/server/routes/media.ts +145 -0
- package/src/server/routes/preview-sessions.ts +76 -0
- package/src/server/routes/preview.ts +36 -0
- package/src/server/routes/revisions.ts +29 -0
- package/src/server/routes/search.ts +31 -0
- package/src/server/routes/sets.ts +40 -0
- package/src/server/routes/settings.ts +24 -0
- package/src/server/routes/users.ts +127 -0
- package/src/server/runtime.ts +99 -0
- package/src/server/sdk/collections.ts +98 -0
- package/src/server/sdk/index.ts +25 -0
- package/src/server/sdk/media.ts +11 -0
- package/src/server/sdk/search.ts +90 -0
|
@@ -0,0 +1,104 @@
|
|
|
1
|
+
<script setup lang="ts">
|
|
2
|
+
import { NodeViewWrapper, type NodeViewProps } from '@tiptap/vue-3';
|
|
3
|
+
import { ref, watch } from 'vue';
|
|
4
|
+
import { deleteCurrentNode, insertParagraphAfter, insertParagraphBefore } from './set-node-utils.js';
|
|
5
|
+
import { sanitizeMediaSrc } from './url-utils.js';
|
|
6
|
+
|
|
7
|
+
const props = defineProps<NodeViewProps>();
|
|
8
|
+
|
|
9
|
+
const srcDraft = ref(String(props.node.attrs?.src ?? ''));
|
|
10
|
+
const invalidSrc = ref(false);
|
|
11
|
+
|
|
12
|
+
watch(
|
|
13
|
+
() => props.node.attrs?.src,
|
|
14
|
+
(value) => {
|
|
15
|
+
srcDraft.value = String(value ?? '');
|
|
16
|
+
invalidSrc.value = false;
|
|
17
|
+
},
|
|
18
|
+
);
|
|
19
|
+
|
|
20
|
+
function commitSrc() {
|
|
21
|
+
const sanitized = sanitizeMediaSrc(srcDraft.value);
|
|
22
|
+
if (!srcDraft.value.trim()) {
|
|
23
|
+
props.updateAttributes({ src: null });
|
|
24
|
+
invalidSrc.value = false;
|
|
25
|
+
return;
|
|
26
|
+
}
|
|
27
|
+
if (!sanitized) {
|
|
28
|
+
invalidSrc.value = true;
|
|
29
|
+
return;
|
|
30
|
+
}
|
|
31
|
+
invalidSrc.value = false;
|
|
32
|
+
srcDraft.value = sanitized;
|
|
33
|
+
props.updateAttributes({ src: sanitized });
|
|
34
|
+
}
|
|
35
|
+
|
|
36
|
+
function addAbove() {
|
|
37
|
+
insertParagraphBefore(props);
|
|
38
|
+
}
|
|
39
|
+
|
|
40
|
+
function addBelow() {
|
|
41
|
+
insertParagraphAfter(props);
|
|
42
|
+
}
|
|
43
|
+
|
|
44
|
+
function removeSet() {
|
|
45
|
+
deleteCurrentNode(props);
|
|
46
|
+
}
|
|
47
|
+
</script>
|
|
48
|
+
|
|
49
|
+
<template>
|
|
50
|
+
<NodeViewWrapper
|
|
51
|
+
class="my-3 overflow-hidden rounded-lg border border-zinc-200 bg-zinc-50"
|
|
52
|
+
data-testid="video-node-view"
|
|
53
|
+
>
|
|
54
|
+
<div class="border-b border-zinc-200 px-3 py-2" contenteditable="false">
|
|
55
|
+
<div class="flex items-center justify-between gap-3">
|
|
56
|
+
<div class="text-xs font-medium uppercase tracking-wide text-zinc-500">Video</div>
|
|
57
|
+
<div class="flex items-center gap-2">
|
|
58
|
+
<button
|
|
59
|
+
type="button"
|
|
60
|
+
class="rounded border border-zinc-300 bg-white px-2 py-1 text-xs text-zinc-600 hover:bg-zinc-100"
|
|
61
|
+
data-testid="video-add-above"
|
|
62
|
+
@click="addAbove"
|
|
63
|
+
>
|
|
64
|
+
Add text at top
|
|
65
|
+
</button>
|
|
66
|
+
<button
|
|
67
|
+
type="button"
|
|
68
|
+
class="rounded border border-zinc-300 bg-white px-2 py-1 text-xs text-zinc-600 hover:bg-zinc-100"
|
|
69
|
+
data-testid="video-add-below"
|
|
70
|
+
@click="addBelow"
|
|
71
|
+
>
|
|
72
|
+
Add text below
|
|
73
|
+
</button>
|
|
74
|
+
<button
|
|
75
|
+
type="button"
|
|
76
|
+
class="rounded border border-red-200 bg-white px-2 py-1 text-xs text-red-600 hover:bg-red-50"
|
|
77
|
+
data-testid="video-delete"
|
|
78
|
+
@click="removeSet"
|
|
79
|
+
>
|
|
80
|
+
Delete
|
|
81
|
+
</button>
|
|
82
|
+
</div>
|
|
83
|
+
</div>
|
|
84
|
+
</div>
|
|
85
|
+
<div class="grid gap-3 px-3 py-3" contenteditable="false">
|
|
86
|
+
<label class="grid gap-1 text-xs text-zinc-500">
|
|
87
|
+
<span>Video URL</span>
|
|
88
|
+
<input
|
|
89
|
+
v-model="srcDraft"
|
|
90
|
+
class="rounded border border-zinc-300 bg-white px-3 py-2 text-sm text-zinc-900"
|
|
91
|
+
placeholder="https://example.com/video.mp4"
|
|
92
|
+
data-testid="video-src"
|
|
93
|
+
@blur="commitSrc"
|
|
94
|
+
/>
|
|
95
|
+
</label>
|
|
96
|
+
<div v-if="invalidSrc" class="text-xs text-red-600">
|
|
97
|
+
Enter a valid `http` or `https` URL.
|
|
98
|
+
</div>
|
|
99
|
+
<div class="text-xs text-zinc-500">
|
|
100
|
+
The video renders on the frontend using the saved source URL.
|
|
101
|
+
</div>
|
|
102
|
+
</div>
|
|
103
|
+
</NodeViewWrapper>
|
|
104
|
+
</template>
|
|
@@ -0,0 +1,26 @@
|
|
|
1
|
+
import StarterKit from '@tiptap/starter-kit';
|
|
2
|
+
import { EmojiExtension } from './emoji-extension.js';
|
|
3
|
+
import { VulseAccordionGroupExtension } from './vulse-accordion-group-extension.js';
|
|
4
|
+
import { VulseAccordionExtension } from './vulse-accordion-extension.js';
|
|
5
|
+
import { VulseCalloutExtension } from './vulse-callout-extension.js';
|
|
6
|
+
import { VulseIframeExtension } from './vulse-iframe-extension.js';
|
|
7
|
+
import { VulseLinkExtension } from './link-extension.js';
|
|
8
|
+
import { VulseVideoExtension } from './vulse-video-extension.js';
|
|
9
|
+
import { VulseSetExtension } from './vulse-set-extension.js';
|
|
10
|
+
|
|
11
|
+
export const EMPTY_BLOCKS_DOC = {
|
|
12
|
+
type: 'doc',
|
|
13
|
+
content: [{ type: 'paragraph' }],
|
|
14
|
+
} as const;
|
|
15
|
+
|
|
16
|
+
export const blocksEditorExtensions = [
|
|
17
|
+
StarterKit,
|
|
18
|
+
VulseLinkExtension,
|
|
19
|
+
EmojiExtension,
|
|
20
|
+
VulseCalloutExtension,
|
|
21
|
+
VulseAccordionGroupExtension,
|
|
22
|
+
VulseAccordionExtension,
|
|
23
|
+
VulseIframeExtension,
|
|
24
|
+
VulseVideoExtension,
|
|
25
|
+
VulseSetExtension,
|
|
26
|
+
];
|
|
@@ -0,0 +1,54 @@
|
|
|
1
|
+
import { Node, mergeAttributes } from '@tiptap/core';
|
|
2
|
+
|
|
3
|
+
declare module '@tiptap/core' {
|
|
4
|
+
interface Commands<ReturnType> {
|
|
5
|
+
emoji: {
|
|
6
|
+
insertEmoji: (value: string, label?: string) => ReturnType;
|
|
7
|
+
};
|
|
8
|
+
}
|
|
9
|
+
}
|
|
10
|
+
|
|
11
|
+
export const EmojiExtension = Node.create({
|
|
12
|
+
name: 'emoji',
|
|
13
|
+
group: 'inline',
|
|
14
|
+
inline: true,
|
|
15
|
+
atom: true,
|
|
16
|
+
|
|
17
|
+
addAttributes() {
|
|
18
|
+
return {
|
|
19
|
+
value: {
|
|
20
|
+
default: '🙂',
|
|
21
|
+
parseHTML: (element: HTMLElement) =>
|
|
22
|
+
element.getAttribute('data-value') ?? element.textContent ?? '🙂',
|
|
23
|
+
renderHTML: (attrs: { value: string }) => ({ 'data-value': attrs.value }),
|
|
24
|
+
},
|
|
25
|
+
label: {
|
|
26
|
+
default: null,
|
|
27
|
+
parseHTML: (element: HTMLElement) => element.getAttribute('aria-label'),
|
|
28
|
+
renderHTML: (attrs: { label?: string | null }) =>
|
|
29
|
+
attrs.label ? { 'aria-label': attrs.label } : {},
|
|
30
|
+
},
|
|
31
|
+
};
|
|
32
|
+
},
|
|
33
|
+
|
|
34
|
+
parseHTML() {
|
|
35
|
+
return [{ tag: 'span[data-vulse-emoji]' }];
|
|
36
|
+
},
|
|
37
|
+
|
|
38
|
+
renderHTML({ HTMLAttributes }) {
|
|
39
|
+
const value = (HTMLAttributes['data-value'] as string | undefined) ?? '🙂';
|
|
40
|
+
return ['span', mergeAttributes(HTMLAttributes, { 'data-vulse-emoji': '' }), value];
|
|
41
|
+
},
|
|
42
|
+
|
|
43
|
+
addCommands() {
|
|
44
|
+
return {
|
|
45
|
+
insertEmoji:
|
|
46
|
+
(value: string, label?: string) =>
|
|
47
|
+
({ commands }) =>
|
|
48
|
+
commands.insertContent({
|
|
49
|
+
type: this.name,
|
|
50
|
+
attrs: { value, ...(label ? { label } : {}) },
|
|
51
|
+
}),
|
|
52
|
+
};
|
|
53
|
+
},
|
|
54
|
+
});
|
|
@@ -0,0 +1,48 @@
|
|
|
1
|
+
import { Mark, mergeAttributes } from '@tiptap/core';
|
|
2
|
+
import { sanitizeLinkHref } from './url-utils.js';
|
|
3
|
+
|
|
4
|
+
declare module '@tiptap/core' {
|
|
5
|
+
interface Commands<ReturnType> {
|
|
6
|
+
vulseLink: {
|
|
7
|
+
setVulseLink: (href: string) => ReturnType;
|
|
8
|
+
unsetVulseLink: () => ReturnType;
|
|
9
|
+
};
|
|
10
|
+
}
|
|
11
|
+
}
|
|
12
|
+
|
|
13
|
+
export const VulseLinkExtension = Mark.create({
|
|
14
|
+
name: 'link',
|
|
15
|
+
inclusive: false,
|
|
16
|
+
|
|
17
|
+
addAttributes() {
|
|
18
|
+
return {
|
|
19
|
+
href: {
|
|
20
|
+
default: null,
|
|
21
|
+
parseHTML: (element: HTMLElement) =>
|
|
22
|
+
sanitizeLinkHref(element.getAttribute('href') ?? '') ?? null,
|
|
23
|
+
renderHTML: (attrs: { href?: string | null }) => (attrs.href ? { href: attrs.href } : {}),
|
|
24
|
+
},
|
|
25
|
+
};
|
|
26
|
+
},
|
|
27
|
+
|
|
28
|
+
parseHTML() {
|
|
29
|
+
return [{ tag: 'a[href]' }];
|
|
30
|
+
},
|
|
31
|
+
|
|
32
|
+
renderHTML({ HTMLAttributes }) {
|
|
33
|
+
return ['a', mergeAttributes(HTMLAttributes), 0];
|
|
34
|
+
},
|
|
35
|
+
|
|
36
|
+
addCommands() {
|
|
37
|
+
return {
|
|
38
|
+
setVulseLink:
|
|
39
|
+
(href: string) =>
|
|
40
|
+
({ commands }) =>
|
|
41
|
+
commands.setMark(this.name, { href }),
|
|
42
|
+
unsetVulseLink:
|
|
43
|
+
() =>
|
|
44
|
+
({ commands }) =>
|
|
45
|
+
commands.unsetMark(this.name),
|
|
46
|
+
};
|
|
47
|
+
},
|
|
48
|
+
});
|
|
@@ -0,0 +1,115 @@
|
|
|
1
|
+
import type { JSONContent } from '@tiptap/core';
|
|
2
|
+
import type { NodeViewProps } from '@tiptap/vue-3';
|
|
3
|
+
|
|
4
|
+
function nodePos(props: NodeViewProps): number | null {
|
|
5
|
+
const getPos = props.getPos;
|
|
6
|
+
if (typeof getPos !== 'function') return null;
|
|
7
|
+
const pos = getPos();
|
|
8
|
+
return typeof pos === 'number' ? pos : null;
|
|
9
|
+
}
|
|
10
|
+
|
|
11
|
+
function nodeEndPos(props: NodeViewProps): number | null {
|
|
12
|
+
const pos = nodePos(props);
|
|
13
|
+
return pos === null ? null : pos + props.node.nodeSize;
|
|
14
|
+
}
|
|
15
|
+
|
|
16
|
+
function nodeContentEndPos(props: NodeViewProps): number | null {
|
|
17
|
+
const pos = nodePos(props);
|
|
18
|
+
return pos === null ? null : pos + props.node.nodeSize - 1;
|
|
19
|
+
}
|
|
20
|
+
|
|
21
|
+
function insertParagraphAt(props: NodeViewProps, pos: number): void {
|
|
22
|
+
const paragraphType = props.editor.schema.nodes.paragraph;
|
|
23
|
+
if (!paragraphType) return;
|
|
24
|
+
|
|
25
|
+
const paragraph = paragraphType.create();
|
|
26
|
+
const { view, state } = props.editor;
|
|
27
|
+
view.dispatch(state.tr.insert(pos, paragraph));
|
|
28
|
+
props.editor.commands.focus();
|
|
29
|
+
props.editor.commands.setTextSelection(pos + 1);
|
|
30
|
+
}
|
|
31
|
+
|
|
32
|
+
export function insertContentAfter(
|
|
33
|
+
props: NodeViewProps,
|
|
34
|
+
content: JSONContent | JSONContent[],
|
|
35
|
+
): void {
|
|
36
|
+
const after = nodeEndPos(props);
|
|
37
|
+
if (after === null) return;
|
|
38
|
+
props.editor.chain().focus().insertContentAt(after, content).run();
|
|
39
|
+
}
|
|
40
|
+
|
|
41
|
+
export function appendContentInside(
|
|
42
|
+
props: NodeViewProps,
|
|
43
|
+
content: JSONContent | JSONContent[],
|
|
44
|
+
): void {
|
|
45
|
+
const end = nodeContentEndPos(props);
|
|
46
|
+
if (end === null) return;
|
|
47
|
+
props.editor.chain().focus().insertContentAt(end, content).run();
|
|
48
|
+
}
|
|
49
|
+
|
|
50
|
+
export function insertParagraphBefore(props: NodeViewProps): void {
|
|
51
|
+
const pos = nodePos(props);
|
|
52
|
+
if (pos === null) return;
|
|
53
|
+
insertParagraphAt(props, pos);
|
|
54
|
+
}
|
|
55
|
+
|
|
56
|
+
export function insertParagraphAfter(props: NodeViewProps): void {
|
|
57
|
+
const after = nodeEndPos(props);
|
|
58
|
+
if (after === null) return;
|
|
59
|
+
insertParagraphAt(props, after);
|
|
60
|
+
}
|
|
61
|
+
|
|
62
|
+
export function deleteCurrentNode(props: NodeViewProps): void {
|
|
63
|
+
props.deleteNode();
|
|
64
|
+
}
|
|
65
|
+
|
|
66
|
+
export function parentNodeInfo(props: NodeViewProps): {
|
|
67
|
+
name: string;
|
|
68
|
+
childCount: number;
|
|
69
|
+
index: number;
|
|
70
|
+
pos: number | null;
|
|
71
|
+
} | null {
|
|
72
|
+
const pos = nodePos(props);
|
|
73
|
+
const doc = props.editor.state?.doc;
|
|
74
|
+
if (pos === null || !doc || typeof doc.resolve !== 'function') return null;
|
|
75
|
+
|
|
76
|
+
const $pos = doc.resolve(pos);
|
|
77
|
+
if ($pos.depth < 1) return null;
|
|
78
|
+
|
|
79
|
+
return {
|
|
80
|
+
name: $pos.parent.type.name,
|
|
81
|
+
childCount: $pos.parent.childCount,
|
|
82
|
+
index: $pos.index($pos.depth),
|
|
83
|
+
pos: $pos.before($pos.depth),
|
|
84
|
+
};
|
|
85
|
+
}
|
|
86
|
+
|
|
87
|
+
export function deleteCurrentNodeOrParentIfOnlyChild(
|
|
88
|
+
props: NodeViewProps,
|
|
89
|
+
parentName: string,
|
|
90
|
+
): void {
|
|
91
|
+
const parent = parentNodeInfo(props);
|
|
92
|
+
const doc = props.editor.state?.doc;
|
|
93
|
+
if (
|
|
94
|
+
parent?.name === parentName &&
|
|
95
|
+
parent.childCount === 1 &&
|
|
96
|
+
parent.pos !== null &&
|
|
97
|
+
doc &&
|
|
98
|
+
typeof doc.nodeAt === 'function'
|
|
99
|
+
) {
|
|
100
|
+
const parentNode = doc.nodeAt(parent.pos);
|
|
101
|
+
if (!parentNode) {
|
|
102
|
+
props.deleteNode();
|
|
103
|
+
return;
|
|
104
|
+
}
|
|
105
|
+
|
|
106
|
+
props.editor
|
|
107
|
+
.chain()
|
|
108
|
+
.focus()
|
|
109
|
+
.deleteRange({ from: parent.pos, to: parent.pos + parentNode.nodeSize })
|
|
110
|
+
.run();
|
|
111
|
+
return;
|
|
112
|
+
}
|
|
113
|
+
|
|
114
|
+
props.deleteNode();
|
|
115
|
+
}
|
|
@@ -0,0 +1,85 @@
|
|
|
1
|
+
const LINK_PROTOCOLS = new Set(['http:', 'https:', 'mailto:', 'tel:']);
|
|
2
|
+
const MEDIA_PROTOCOLS = new Set(['http:', 'https:']);
|
|
3
|
+
const RELATIVE_PREFIXES = ['/', './', '../', '#', '?'];
|
|
4
|
+
|
|
5
|
+
export interface ParsedIframeCode {
|
|
6
|
+
code: string;
|
|
7
|
+
src: string;
|
|
8
|
+
title: string;
|
|
9
|
+
width?: string;
|
|
10
|
+
height?: string;
|
|
11
|
+
allow?: string;
|
|
12
|
+
loading?: string;
|
|
13
|
+
referrerpolicy?: string;
|
|
14
|
+
frameborder?: string;
|
|
15
|
+
allowfullscreen?: boolean;
|
|
16
|
+
}
|
|
17
|
+
|
|
18
|
+
function hasRelativePrefix(value: string): boolean {
|
|
19
|
+
return RELATIVE_PREFIXES.some((prefix) => value.startsWith(prefix));
|
|
20
|
+
}
|
|
21
|
+
|
|
22
|
+
function tryParseUrl(value: string): URL | null {
|
|
23
|
+
try {
|
|
24
|
+
return new URL(value);
|
|
25
|
+
} catch {
|
|
26
|
+
return null;
|
|
27
|
+
}
|
|
28
|
+
}
|
|
29
|
+
|
|
30
|
+
export function sanitizeLinkHref(value: string): string | null {
|
|
31
|
+
const trimmed = value.trim();
|
|
32
|
+
if (!trimmed) return null;
|
|
33
|
+
if (hasRelativePrefix(trimmed)) return trimmed;
|
|
34
|
+
const parsed = tryParseUrl(trimmed);
|
|
35
|
+
if (!parsed || !LINK_PROTOCOLS.has(parsed.protocol)) return null;
|
|
36
|
+
return parsed.toString();
|
|
37
|
+
}
|
|
38
|
+
|
|
39
|
+
export function sanitizeMediaSrc(value: string): string | null {
|
|
40
|
+
const trimmed = value.trim();
|
|
41
|
+
if (!trimmed) return null;
|
|
42
|
+
if (hasRelativePrefix(trimmed)) return trimmed;
|
|
43
|
+
const parsed = tryParseUrl(trimmed);
|
|
44
|
+
if (!parsed || !MEDIA_PROTOCOLS.has(parsed.protocol)) return null;
|
|
45
|
+
return parsed.toString();
|
|
46
|
+
}
|
|
47
|
+
|
|
48
|
+
function parseAttributes(source: string): Record<string, string> {
|
|
49
|
+
const attrs: Record<string, string> = {};
|
|
50
|
+
const pattern = /([a-zA-Z_:][-a-zA-Z0-9_:.]*)(?:\s*=\s*(?:"([^"]*)"|'([^']*)'|([^\s"'=<>`]+)))?/g;
|
|
51
|
+
|
|
52
|
+
for (const match of source.matchAll(pattern)) {
|
|
53
|
+
const name = match[1]?.toLowerCase();
|
|
54
|
+
if (!name) continue;
|
|
55
|
+
const value = match[2] ?? match[3] ?? match[4] ?? '';
|
|
56
|
+
attrs[name] = value;
|
|
57
|
+
}
|
|
58
|
+
|
|
59
|
+
return attrs;
|
|
60
|
+
}
|
|
61
|
+
|
|
62
|
+
export function parseIframeCode(raw: string): ParsedIframeCode | null {
|
|
63
|
+
const trimmed = raw.trim();
|
|
64
|
+
if (!trimmed) return null;
|
|
65
|
+
|
|
66
|
+
const match = trimmed.match(/<iframe\b([^>]*)>(?:\s*<\/iframe>)?/i);
|
|
67
|
+
if (!match) return null;
|
|
68
|
+
|
|
69
|
+
const attrs = parseAttributes(match[1] ?? '');
|
|
70
|
+
const src = sanitizeMediaSrc(attrs.src ?? '');
|
|
71
|
+
if (!src) return null;
|
|
72
|
+
|
|
73
|
+
return {
|
|
74
|
+
code: trimmed,
|
|
75
|
+
src,
|
|
76
|
+
title: attrs.title?.trim() || 'Embedded content',
|
|
77
|
+
...(attrs.width ? { width: attrs.width } : {}),
|
|
78
|
+
...(attrs.height ? { height: attrs.height } : {}),
|
|
79
|
+
...(attrs.allow ? { allow: attrs.allow } : {}),
|
|
80
|
+
...(attrs.loading ? { loading: attrs.loading } : {}),
|
|
81
|
+
...(attrs.referrerpolicy ? { referrerpolicy: attrs.referrerpolicy } : {}),
|
|
82
|
+
...(attrs.frameborder ? { frameborder: attrs.frameborder } : {}),
|
|
83
|
+
...(Object.hasOwn(attrs, 'allowfullscreen') ? { allowfullscreen: true } : {}),
|
|
84
|
+
};
|
|
85
|
+
}
|
|
@@ -0,0 +1,64 @@
|
|
|
1
|
+
import { Node, mergeAttributes } from '@tiptap/core';
|
|
2
|
+
import { VueNodeViewRenderer } from '@tiptap/vue-3';
|
|
3
|
+
import VulseAccordionNodeView from './VulseAccordionNodeView.vue';
|
|
4
|
+
|
|
5
|
+
declare module '@tiptap/core' {
|
|
6
|
+
interface Commands<ReturnType> {
|
|
7
|
+
vulseAccordion: {
|
|
8
|
+
insertVulseAccordion: (summary?: string) => ReturnType;
|
|
9
|
+
};
|
|
10
|
+
}
|
|
11
|
+
}
|
|
12
|
+
|
|
13
|
+
export const VulseAccordionExtension = Node.create({
|
|
14
|
+
name: 'vulseAccordion',
|
|
15
|
+
group: 'block',
|
|
16
|
+
content: 'block+',
|
|
17
|
+
defining: true,
|
|
18
|
+
isolating: true,
|
|
19
|
+
|
|
20
|
+
addAttributes() {
|
|
21
|
+
return {
|
|
22
|
+
summary: {
|
|
23
|
+
default: 'Accordion',
|
|
24
|
+
parseHTML: (element: HTMLElement) =>
|
|
25
|
+
element.querySelector('summary')?.textContent?.trim() || 'Accordion',
|
|
26
|
+
renderHTML: (attrs: { summary?: string }) => ({
|
|
27
|
+
'data-summary': attrs.summary ?? 'Accordion',
|
|
28
|
+
}),
|
|
29
|
+
},
|
|
30
|
+
open: {
|
|
31
|
+
default: false,
|
|
32
|
+
parseHTML: (element: HTMLElement) => element.hasAttribute('open'),
|
|
33
|
+
renderHTML: (attrs: { open?: boolean }) => (attrs.open ? { open: 'open' } : {}),
|
|
34
|
+
},
|
|
35
|
+
};
|
|
36
|
+
},
|
|
37
|
+
|
|
38
|
+
parseHTML() {
|
|
39
|
+
return [{ tag: 'details[data-vulse-accordion]' }];
|
|
40
|
+
},
|
|
41
|
+
|
|
42
|
+
renderHTML({ HTMLAttributes }) {
|
|
43
|
+
const { summary = 'Accordion', ...rest } = HTMLAttributes;
|
|
44
|
+
return [
|
|
45
|
+
'details',
|
|
46
|
+
mergeAttributes(rest, { 'data-vulse-accordion': '' }),
|
|
47
|
+
['summary', {}, String(summary)],
|
|
48
|
+
['div', { 'data-vulse-accordion-content': '' }, 0],
|
|
49
|
+
];
|
|
50
|
+
},
|
|
51
|
+
|
|
52
|
+
addNodeView() {
|
|
53
|
+
return VueNodeViewRenderer(VulseAccordionNodeView);
|
|
54
|
+
},
|
|
55
|
+
|
|
56
|
+
addCommands() {
|
|
57
|
+
return {
|
|
58
|
+
insertVulseAccordion:
|
|
59
|
+
(summary = 'Accordion') =>
|
|
60
|
+
({ commands }) =>
|
|
61
|
+
commands.insertVulseAccordionGroup(summary),
|
|
62
|
+
};
|
|
63
|
+
},
|
|
64
|
+
});
|
|
@@ -0,0 +1,49 @@
|
|
|
1
|
+
import { Node, mergeAttributes } from '@tiptap/core';
|
|
2
|
+
import { VueNodeViewRenderer } from '@tiptap/vue-3';
|
|
3
|
+
import VulseAccordionGroupNodeView from './VulseAccordionGroupNodeView.vue';
|
|
4
|
+
|
|
5
|
+
declare module '@tiptap/core' {
|
|
6
|
+
interface Commands<ReturnType> {
|
|
7
|
+
vulseAccordionGroup: {
|
|
8
|
+
insertVulseAccordionGroup: (summary?: string) => ReturnType;
|
|
9
|
+
};
|
|
10
|
+
}
|
|
11
|
+
}
|
|
12
|
+
|
|
13
|
+
export const VulseAccordionGroupExtension = Node.create({
|
|
14
|
+
name: 'vulseAccordionGroup',
|
|
15
|
+
group: 'block',
|
|
16
|
+
content: 'vulseAccordion+',
|
|
17
|
+
defining: true,
|
|
18
|
+
isolating: true,
|
|
19
|
+
|
|
20
|
+
parseHTML() {
|
|
21
|
+
return [{ tag: 'div[data-vulse-accordion-group]' }];
|
|
22
|
+
},
|
|
23
|
+
|
|
24
|
+
renderHTML({ HTMLAttributes }) {
|
|
25
|
+
return ['div', mergeAttributes(HTMLAttributes, { 'data-vulse-accordion-group': '' }), 0];
|
|
26
|
+
},
|
|
27
|
+
|
|
28
|
+
addNodeView() {
|
|
29
|
+
return VueNodeViewRenderer(VulseAccordionGroupNodeView);
|
|
30
|
+
},
|
|
31
|
+
|
|
32
|
+
addCommands() {
|
|
33
|
+
return {
|
|
34
|
+
insertVulseAccordionGroup:
|
|
35
|
+
(summary = 'Accordion') =>
|
|
36
|
+
({ commands }) =>
|
|
37
|
+
commands.insertContent({
|
|
38
|
+
type: this.name,
|
|
39
|
+
content: [
|
|
40
|
+
{
|
|
41
|
+
type: 'vulseAccordion',
|
|
42
|
+
attrs: { summary, open: false },
|
|
43
|
+
content: [{ type: 'paragraph' }],
|
|
44
|
+
},
|
|
45
|
+
],
|
|
46
|
+
}),
|
|
47
|
+
};
|
|
48
|
+
},
|
|
49
|
+
});
|
|
@@ -0,0 +1,53 @@
|
|
|
1
|
+
import { Node, mergeAttributes } from '@tiptap/core';
|
|
2
|
+
import { VueNodeViewRenderer } from '@tiptap/vue-3';
|
|
3
|
+
import VulseCalloutNodeView from './VulseCalloutNodeView.vue';
|
|
4
|
+
|
|
5
|
+
declare module '@tiptap/core' {
|
|
6
|
+
interface Commands<ReturnType> {
|
|
7
|
+
vulseCallout: {
|
|
8
|
+
insertVulseCallout: (tone?: 'info' | 'warn') => ReturnType;
|
|
9
|
+
};
|
|
10
|
+
}
|
|
11
|
+
}
|
|
12
|
+
|
|
13
|
+
export const VulseCalloutExtension = Node.create({
|
|
14
|
+
name: 'vulseCallout',
|
|
15
|
+
group: 'block',
|
|
16
|
+
content: 'block+',
|
|
17
|
+
defining: true,
|
|
18
|
+
|
|
19
|
+
addAttributes() {
|
|
20
|
+
return {
|
|
21
|
+
tone: {
|
|
22
|
+
default: 'info',
|
|
23
|
+
parseHTML: (el: HTMLElement) => el.getAttribute('data-tone') ?? 'info',
|
|
24
|
+
renderHTML: (attrs: { tone: string }) => ({ 'data-tone': attrs.tone }),
|
|
25
|
+
},
|
|
26
|
+
};
|
|
27
|
+
},
|
|
28
|
+
|
|
29
|
+
parseHTML() {
|
|
30
|
+
return [{ tag: 'aside[data-vulse-callout]' }];
|
|
31
|
+
},
|
|
32
|
+
|
|
33
|
+
renderHTML({ HTMLAttributes }) {
|
|
34
|
+
return ['aside', mergeAttributes(HTMLAttributes, { 'data-vulse-callout': '' }), 0];
|
|
35
|
+
},
|
|
36
|
+
|
|
37
|
+
addNodeView() {
|
|
38
|
+
return VueNodeViewRenderer(VulseCalloutNodeView);
|
|
39
|
+
},
|
|
40
|
+
|
|
41
|
+
addCommands() {
|
|
42
|
+
return {
|
|
43
|
+
insertVulseCallout:
|
|
44
|
+
(tone: 'info' | 'warn' = 'info') =>
|
|
45
|
+
({ commands }) =>
|
|
46
|
+
commands.insertContent({
|
|
47
|
+
type: this.name,
|
|
48
|
+
attrs: { tone },
|
|
49
|
+
content: [{ type: 'paragraph' }],
|
|
50
|
+
}),
|
|
51
|
+
};
|
|
52
|
+
},
|
|
53
|
+
});
|
|
@@ -0,0 +1,96 @@
|
|
|
1
|
+
import { Node, mergeAttributes } from '@tiptap/core';
|
|
2
|
+
import { VueNodeViewRenderer } from '@tiptap/vue-3';
|
|
3
|
+
import { parseIframeCode, sanitizeMediaSrc } from './url-utils.js';
|
|
4
|
+
import VulseIframeNodeView from './VulseIframeNodeView.vue';
|
|
5
|
+
|
|
6
|
+
declare module '@tiptap/core' {
|
|
7
|
+
interface Commands<ReturnType> {
|
|
8
|
+
vulseIframe: {
|
|
9
|
+
insertVulseIframe: (src?: string, title?: string) => ReturnType;
|
|
10
|
+
};
|
|
11
|
+
}
|
|
12
|
+
}
|
|
13
|
+
|
|
14
|
+
export const VulseIframeExtension = Node.create({
|
|
15
|
+
name: 'vulseIframe',
|
|
16
|
+
group: 'block',
|
|
17
|
+
atom: true,
|
|
18
|
+
selectable: true,
|
|
19
|
+
draggable: true,
|
|
20
|
+
|
|
21
|
+
addAttributes() {
|
|
22
|
+
return {
|
|
23
|
+
code: {
|
|
24
|
+
default: null,
|
|
25
|
+
parseHTML: (element: HTMLElement) => element.outerHTML,
|
|
26
|
+
renderHTML: () => ({}),
|
|
27
|
+
},
|
|
28
|
+
src: {
|
|
29
|
+
default: null,
|
|
30
|
+
parseHTML: (element: HTMLElement) =>
|
|
31
|
+
sanitizeMediaSrc(element.getAttribute('src') ?? '') ?? null,
|
|
32
|
+
renderHTML: (attrs: { src?: string | null }) => (attrs.src ? { src: attrs.src } : {}),
|
|
33
|
+
},
|
|
34
|
+
title: {
|
|
35
|
+
default: 'Embedded content',
|
|
36
|
+
parseHTML: (element: HTMLElement) => element.getAttribute('title') ?? 'Embedded content',
|
|
37
|
+
renderHTML: (attrs: { title?: string | null }) => ({
|
|
38
|
+
title: attrs.title ?? 'Embedded content',
|
|
39
|
+
}),
|
|
40
|
+
},
|
|
41
|
+
};
|
|
42
|
+
},
|
|
43
|
+
|
|
44
|
+
parseHTML() {
|
|
45
|
+
return [{ tag: 'iframe[data-vulse-embed="iframe"]' }, { tag: 'iframe[src]' }];
|
|
46
|
+
},
|
|
47
|
+
|
|
48
|
+
renderHTML({ HTMLAttributes }) {
|
|
49
|
+
const parsed = parseIframeCode(String(HTMLAttributes.code ?? ''));
|
|
50
|
+
const src = parsed?.src ?? (HTMLAttributes.src as string | null | undefined) ?? null;
|
|
51
|
+
const title =
|
|
52
|
+
parsed?.title ?? (HTMLAttributes.title as string | null | undefined) ?? 'Embedded content';
|
|
53
|
+
if (!src) return ['div', { 'data-vulse-embed': 'iframe' }];
|
|
54
|
+
|
|
55
|
+
return [
|
|
56
|
+
'iframe',
|
|
57
|
+
mergeAttributes(
|
|
58
|
+
{
|
|
59
|
+
'data-vulse-embed': 'iframe',
|
|
60
|
+
loading: parsed?.loading ?? 'lazy',
|
|
61
|
+
allowfullscreen: parsed?.allowfullscreen ? 'true' : 'true',
|
|
62
|
+
frameborder: parsed?.frameborder ?? '0',
|
|
63
|
+
src,
|
|
64
|
+
title,
|
|
65
|
+
},
|
|
66
|
+
parsed?.width ? { width: parsed.width } : {},
|
|
67
|
+
parsed?.height ? { height: parsed.height } : {},
|
|
68
|
+
parsed?.allow ? { allow: parsed.allow } : {},
|
|
69
|
+
parsed?.referrerpolicy ? { referrerpolicy: parsed.referrerpolicy } : {},
|
|
70
|
+
HTMLAttributes,
|
|
71
|
+
),
|
|
72
|
+
];
|
|
73
|
+
},
|
|
74
|
+
|
|
75
|
+
addNodeView() {
|
|
76
|
+
return VueNodeViewRenderer(VulseIframeNodeView);
|
|
77
|
+
},
|
|
78
|
+
|
|
79
|
+
addCommands() {
|
|
80
|
+
return {
|
|
81
|
+
insertVulseIframe:
|
|
82
|
+
(src = '', title = 'Embedded content') =>
|
|
83
|
+
({ commands }) =>
|
|
84
|
+
commands.insertContent({
|
|
85
|
+
type: this.name,
|
|
86
|
+
attrs: {
|
|
87
|
+
code: src
|
|
88
|
+
? `<iframe src="${sanitizeMediaSrc(src) ?? ''}" title="${title}"></iframe>`
|
|
89
|
+
: null,
|
|
90
|
+
src: sanitizeMediaSrc(src),
|
|
91
|
+
title,
|
|
92
|
+
},
|
|
93
|
+
}),
|
|
94
|
+
};
|
|
95
|
+
},
|
|
96
|
+
});
|