Extract mention-aware link node helper and add tests
Co-Authored-By: Paperclip <noreply@paperclip.ing>
This commit is contained in:
parent
334e7e61b5
commit
52dab938cb
3 changed files with 119 additions and 42 deletions
|
|
@ -26,53 +26,13 @@ import {
|
||||||
thematicBreakPlugin,
|
thematicBreakPlugin,
|
||||||
type RealmPlugin,
|
type RealmPlugin,
|
||||||
} from "@mdxeditor/editor";
|
} from "@mdxeditor/editor";
|
||||||
import { LinkNode, type LinkAttributes } from "@lexical/link";
|
|
||||||
import { buildAgentMentionHref, buildProjectMentionHref } from "@paperclipai/shared";
|
import { buildAgentMentionHref, buildProjectMentionHref } from "@paperclipai/shared";
|
||||||
import { AgentIcon } from "./AgentIconPicker";
|
import { AgentIcon } from "./AgentIconPicker";
|
||||||
import { applyMentionChipDecoration, clearMentionChipDecoration, parseMentionChipHref } from "../lib/mention-chips";
|
import { applyMentionChipDecoration, clearMentionChipDecoration, parseMentionChipHref } from "../lib/mention-chips";
|
||||||
|
import { MentionAwareLinkNode, mentionAwareLinkNodeReplacement } from "../lib/mention-aware-link-node";
|
||||||
import { mentionDeletionPlugin } from "../lib/mention-deletion";
|
import { mentionDeletionPlugin } from "../lib/mention-deletion";
|
||||||
import { cn } from "../lib/utils";
|
import { cn } from "../lib/utils";
|
||||||
|
|
||||||
const CUSTOM_MENTION_URL_RE = /^(agent|project):\/\//;
|
|
||||||
|
|
||||||
class MentionAwareLinkNode extends LinkNode {
|
|
||||||
static clone(node: MentionAwareLinkNode): MentionAwareLinkNode {
|
|
||||||
return new MentionAwareLinkNode(
|
|
||||||
node.getURL(),
|
|
||||||
{
|
|
||||||
rel: node.getRel(),
|
|
||||||
target: node.getTarget(),
|
|
||||||
title: node.getTitle(),
|
|
||||||
},
|
|
||||||
node.getKey(),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
constructor(url?: string, attributes?: LinkAttributes, key?: string) {
|
|
||||||
super(url, attributes, key);
|
|
||||||
}
|
|
||||||
|
|
||||||
sanitizeUrl(url: string): string {
|
|
||||||
if (CUSTOM_MENTION_URL_RE.test(url)) return url;
|
|
||||||
return super.sanitizeUrl(url);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
const mentionAwareLinkNodeReplacement = {
|
|
||||||
replace: LinkNode,
|
|
||||||
with: (node: LinkNode) =>
|
|
||||||
new MentionAwareLinkNode(
|
|
||||||
node.getURL(),
|
|
||||||
{
|
|
||||||
rel: node.getRel(),
|
|
||||||
target: node.getTarget(),
|
|
||||||
title: node.getTitle(),
|
|
||||||
},
|
|
||||||
node.getKey(),
|
|
||||||
),
|
|
||||||
withKlass: MentionAwareLinkNode,
|
|
||||||
} as const;
|
|
||||||
|
|
||||||
/* ---- Mention types ---- */
|
/* ---- Mention types ---- */
|
||||||
|
|
||||||
export interface MentionOption {
|
export interface MentionOption {
|
||||||
|
|
@ -590,7 +550,7 @@ export const MarkdownEditor = forwardRef<MarkdownEditorRef, MarkdownEditorProps>
|
||||||
"paperclip-mdxeditor-content focus:outline-none [&_ul]:list-disc [&_ul]:pl-5 [&_ol]:list-decimal [&_ol]:pl-5 [&_li]:list-item",
|
"paperclip-mdxeditor-content focus:outline-none [&_ul]:list-disc [&_ul]:pl-5 [&_ol]:list-decimal [&_ol]:pl-5 [&_li]:list-item",
|
||||||
contentClassName,
|
contentClassName,
|
||||||
)}
|
)}
|
||||||
additionalLexicalNodes={[mentionAwareLinkNodeReplacement]}
|
additionalLexicalNodes={[MentionAwareLinkNode, mentionAwareLinkNodeReplacement]}
|
||||||
plugins={plugins}
|
plugins={plugins}
|
||||||
/>
|
/>
|
||||||
|
|
||||||
|
|
|
||||||
50
ui/src/lib/mention-aware-link-node.test.ts
Normal file
50
ui/src/lib/mention-aware-link-node.test.ts
Normal file
|
|
@ -0,0 +1,50 @@
|
||||||
|
import { describe, expect, it } from "vitest";
|
||||||
|
import { $createLinkNode } from "@lexical/link";
|
||||||
|
import { createEditor } from "lexical";
|
||||||
|
import {
|
||||||
|
MentionAwareLinkNode,
|
||||||
|
getMentionAwareLinkNodeInit,
|
||||||
|
mentionAwareLinkNodeReplacement,
|
||||||
|
} from "./mention-aware-link-node";
|
||||||
|
|
||||||
|
function createTestEditor() {
|
||||||
|
return createEditor({
|
||||||
|
namespace: "mention-aware-link-node-test",
|
||||||
|
nodes: [MentionAwareLinkNode, mentionAwareLinkNodeReplacement],
|
||||||
|
onError(error: Error) {
|
||||||
|
throw error;
|
||||||
|
},
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
describe("getMentionAwareLinkNodeInit", () => {
|
||||||
|
it("copies link attributes without carrying over a node key", () => {
|
||||||
|
const init = getMentionAwareLinkNodeInit({
|
||||||
|
getURL: () => "agent://agent-123",
|
||||||
|
getRel: () => "noreferrer",
|
||||||
|
getTarget: () => "_blank",
|
||||||
|
getTitle: () => "Agent mention",
|
||||||
|
});
|
||||||
|
|
||||||
|
expect(Object.keys(init)).toEqual(["url", "attributes"]);
|
||||||
|
expect(init).toEqual({
|
||||||
|
url: "agent://agent-123",
|
||||||
|
attributes: {
|
||||||
|
rel: "noreferrer",
|
||||||
|
target: "_blank",
|
||||||
|
title: "Agent mention",
|
||||||
|
},
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it("replaces LinkNode creation with MentionAwareLinkNode without throwing", () => {
|
||||||
|
const editor = createTestEditor();
|
||||||
|
let created: unknown;
|
||||||
|
|
||||||
|
editor.update(() => {
|
||||||
|
created = $createLinkNode("agent://agent-123");
|
||||||
|
});
|
||||||
|
|
||||||
|
expect(created).toBeInstanceOf(MentionAwareLinkNode);
|
||||||
|
});
|
||||||
|
});
|
||||||
67
ui/src/lib/mention-aware-link-node.ts
Normal file
67
ui/src/lib/mention-aware-link-node.ts
Normal file
|
|
@ -0,0 +1,67 @@
|
||||||
|
import {
|
||||||
|
LinkNode,
|
||||||
|
type LinkAttributes,
|
||||||
|
type SerializedLinkNode,
|
||||||
|
} from "@lexical/link";
|
||||||
|
|
||||||
|
const CUSTOM_MENTION_URL_RE = /^(agent|project):\/\//;
|
||||||
|
|
||||||
|
export class MentionAwareLinkNode extends LinkNode {
|
||||||
|
static getType(): string {
|
||||||
|
return "mention-aware-link";
|
||||||
|
}
|
||||||
|
|
||||||
|
static clone(node: MentionAwareLinkNode): MentionAwareLinkNode {
|
||||||
|
return new MentionAwareLinkNode(
|
||||||
|
node.getURL(),
|
||||||
|
{
|
||||||
|
rel: node.getRel(),
|
||||||
|
target: node.getTarget(),
|
||||||
|
title: node.getTitle(),
|
||||||
|
},
|
||||||
|
node.getKey(),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
static importJSON(serializedNode: SerializedLinkNode): MentionAwareLinkNode {
|
||||||
|
return new MentionAwareLinkNode(
|
||||||
|
serializedNode.url ?? "",
|
||||||
|
{
|
||||||
|
rel: serializedNode.rel ?? null,
|
||||||
|
target: serializedNode.target ?? null,
|
||||||
|
title: serializedNode.title ?? null,
|
||||||
|
},
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
constructor(url?: string, attributes?: LinkAttributes, key?: string) {
|
||||||
|
super(url, attributes, key);
|
||||||
|
}
|
||||||
|
|
||||||
|
sanitizeUrl(url: string): string {
|
||||||
|
if (CUSTOM_MENTION_URL_RE.test(url)) return url;
|
||||||
|
return super.sanitizeUrl(url);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
type MentionAwareLinkSource = Pick<LinkNode, "getURL" | "getRel" | "getTarget" | "getTitle">;
|
||||||
|
|
||||||
|
export function getMentionAwareLinkNodeInit(node: MentionAwareLinkSource) {
|
||||||
|
return {
|
||||||
|
url: node.getURL(),
|
||||||
|
attributes: {
|
||||||
|
rel: node.getRel(),
|
||||||
|
target: node.getTarget(),
|
||||||
|
title: node.getTitle(),
|
||||||
|
},
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
export const mentionAwareLinkNodeReplacement = {
|
||||||
|
replace: LinkNode,
|
||||||
|
with: (node: LinkNode) => {
|
||||||
|
const { url, attributes } = getMentionAwareLinkNodeInit(node);
|
||||||
|
return new MentionAwareLinkNode(url, attributes);
|
||||||
|
},
|
||||||
|
withKlass: MentionAwareLinkNode,
|
||||||
|
} as const;
|
||||||
Loading…
Add table
Reference in a new issue