Extract mention-aware link node helper and add tests

Co-Authored-By: Paperclip <noreply@paperclip.ing>
This commit is contained in:
dotta 2026-03-23 20:41:50 -05:00
parent 334e7e61b5
commit 52dab938cb
3 changed files with 119 additions and 42 deletions

View file

@ -26,53 +26,13 @@ import {
thematicBreakPlugin,
type RealmPlugin,
} from "@mdxeditor/editor";
import { LinkNode, type LinkAttributes } from "@lexical/link";
import { buildAgentMentionHref, buildProjectMentionHref } from "@paperclipai/shared";
import { AgentIcon } from "./AgentIconPicker";
import { applyMentionChipDecoration, clearMentionChipDecoration, parseMentionChipHref } from "../lib/mention-chips";
import { MentionAwareLinkNode, mentionAwareLinkNodeReplacement } from "../lib/mention-aware-link-node";
import { mentionDeletionPlugin } from "../lib/mention-deletion";
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 ---- */
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",
contentClassName,
)}
additionalLexicalNodes={[mentionAwareLinkNodeReplacement]}
additionalLexicalNodes={[MentionAwareLinkNode, mentionAwareLinkNodeReplacement]}
plugins={plugins}
/>

View 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);
});
});

View 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;