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,
|
||||
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}
|
||||
/>
|
||||
|
||||
|
|
|
|||
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