forked from farhoodlabs/paperclip
Expand plugin host surface (#5205)
## Thinking Path > - Paperclip orchestrates AI agents for zero-human companies > - The plugin system is the extension boundary for optional product capabilities > - Rich plugins need more than a worker entrypoint: they need scoped database storage, local project folders, managed agents/routines, host navigation, and reusable UI components > - The LLM Wiki work exposed those missing host surfaces while keeping plugin code outside the core control plane > - This pull request expands the core plugin host, SDK, server APIs, and UI bridge so plugins can declare and use those surfaces > - The benefit is that future plugins can integrate with Paperclip through documented, validated contracts instead of bespoke server or UI imports ## What Changed - Added plugin-managed database namespaces and migration tracking, including Drizzle schema/migration files and SQL validation for namespace isolation. - Added server support for plugin local folders, managed agents, managed routines, scoped plugin APIs, and plugin operation visibility. - Expanded shared plugin manifest/types/validators and SDK host/testing/UI exports for richer plugin surfaces. - Added reusable UI pieces for file trees, managed routines, resizable sidebars, route sidebars, and plugin bridge initialization. - Updated plugin docs and example plugins to use the expanded host and SDK surface. ## Verification - `pnpm install --frozen-lockfile` - `pnpm run preflight:workspace-links && pnpm exec vitest run packages/shared/src/validators/plugin.test.ts server/src/__tests__/plugin-database.test.ts server/src/__tests__/plugin-local-folders.test.ts server/src/__tests__/plugin-managed-agents.test.ts server/src/__tests__/plugin-managed-routines.test.ts server/src/__tests__/plugin-orchestration-apis.test.ts ui/src/api/plugins.test.ts ui/src/components/FileTree.test.tsx ui/src/components/ResizableSidebarPane.test.tsx ui/src/pages/PluginPage.test.tsx ui/src/plugins/bridge.test.ts` passed: 11 files, 67 tests. - Confirmed this PR changes 89 files and does not include `pnpm-lock.yaml` or `.github/workflows/*`. ## Risks - Medium: this expands plugin host contracts across db/shared/server/ui and includes a new core migration (`0076_useful_elektra.sql`). - The plugin database namespace validator is intentionally restrictive; plugin authors may need follow-up affordances for SQL patterns that remain blocked. - Merge this before the LLM Wiki plugin PR so the plugin can resolve the new SDK and host APIs. > For core feature work, check [`ROADMAP.md`](ROADMAP.md) first and discuss it in `#dev` before opening the PR. Feature PRs that overlap with planned core work may need to be redirected — check the roadmap first. See `CONTRIBUTING.md`. ## Model Used - OpenAI Codex, GPT-5 coding agent, tool-enabled shell/git/GitHub workflow. Context window size was not exposed by the runtime. ## Checklist - [x] I have included a thinking path that traces from project context to this change - [x] I have specified the model used (with version and capability details) - [x] I have checked ROADMAP.md and confirmed this PR does not duplicate planned core work - [x] I have run tests locally and they pass - [x] I have added or updated tests where applicable - [x] If this change affects the UI, I have included before/after screenshots - [x] I have updated relevant documentation to reflect my changes - [x] I have considered and documented any risks above - [x] I will address all Greptile and reviewer comments before requesting merge --------- Co-authored-by: Paperclip <noreply@paperclip.ing>
This commit is contained in:
@@ -16,9 +16,47 @@ import {
|
||||
usePluginData,
|
||||
usePluginAction,
|
||||
useHostContext,
|
||||
useHostLocation,
|
||||
useHostNavigation,
|
||||
usePluginStream,
|
||||
usePluginToast,
|
||||
} from "./bridge.js";
|
||||
import { createElement, useEffect, useMemo, useState, type ComponentType, type ReactNode } from "react";
|
||||
import { useMutation, useQuery, useQueryClient } from "@tanstack/react-query";
|
||||
import { User } from "lucide-react";
|
||||
import {
|
||||
FileTree,
|
||||
type FileTreeProps as HostFileTreeProps,
|
||||
} from "@/components/FileTree";
|
||||
import { AgentIcon } from "@/components/AgentIconPicker";
|
||||
import { InlineEntitySelector, type InlineEntityOption } from "@/components/InlineEntitySelector";
|
||||
import { IssuesList as HostIssuesList } from "@/components/IssuesList";
|
||||
import { ManagedRoutinesList as HostManagedRoutinesList } from "@/components/ManagedRoutinesList";
|
||||
import { MarkdownBody } from "@/components/MarkdownBody";
|
||||
import { accessApi } from "@/api/access";
|
||||
import { agentsApi } from "@/api/agents";
|
||||
import { authApi } from "@/api/auth";
|
||||
import { heartbeatsApi } from "@/api/heartbeats";
|
||||
import { issuesApi } from "@/api/issues";
|
||||
import { projectsApi } from "@/api/projects";
|
||||
import {
|
||||
buildCompanyUserInlineOptions,
|
||||
} from "@/lib/company-members";
|
||||
import { collectLiveIssueIds } from "@/lib/liveIssueIds";
|
||||
import { useProjectOrder } from "@/hooks/useProjectOrder";
|
||||
import {
|
||||
assigneeValueFromSelection,
|
||||
currentUserAssigneeOption,
|
||||
parseAssigneeValue,
|
||||
} from "@/lib/assignees";
|
||||
import { queryKeys } from "@/lib/queryKeys";
|
||||
import {
|
||||
getRecentAssigneeSelectionIds,
|
||||
sortAgentsByRecency,
|
||||
trackRecentAssignee,
|
||||
trackRecentAssigneeUser,
|
||||
} from "@/lib/recent-assignees";
|
||||
import { getRecentProjectIds, trackRecentProject } from "@/lib/recent-projects";
|
||||
|
||||
// ---------------------------------------------------------------------------
|
||||
// Global bridge registry
|
||||
@@ -41,6 +79,451 @@ declare global {
|
||||
var __paperclipPluginBridge__: PluginBridgeRegistry | undefined;
|
||||
}
|
||||
|
||||
type PluginFileTreePathCollection = ReadonlySet<string> | readonly string[];
|
||||
|
||||
type PluginFileTreeProps = Omit<
|
||||
HostFileTreeProps,
|
||||
| "expandedDirs"
|
||||
| "checkedFiles"
|
||||
| "renderFileExtra"
|
||||
| "fileRowClassName"
|
||||
| "selectedFile"
|
||||
| "showCheckboxes"
|
||||
| "onToggleDir"
|
||||
| "onSelectFile"
|
||||
> & {
|
||||
selectedFile?: string | null;
|
||||
expandedPaths?: PluginFileTreePathCollection;
|
||||
checkedPaths?: PluginFileTreePathCollection;
|
||||
showCheckboxes?: boolean;
|
||||
onToggleDir?: (path: string) => void;
|
||||
onSelectFile?: (path: string) => void;
|
||||
};
|
||||
|
||||
function toPathSet(paths?: PluginFileTreePathCollection | null): Set<string> {
|
||||
return new Set(paths ?? []);
|
||||
}
|
||||
|
||||
function PluginSdkFileTree({
|
||||
expandedPaths,
|
||||
checkedPaths,
|
||||
selectedFile = null,
|
||||
showCheckboxes = false,
|
||||
onToggleDir,
|
||||
onSelectFile,
|
||||
...props
|
||||
}: PluginFileTreeProps) {
|
||||
return createElement(FileTree, {
|
||||
...props,
|
||||
selectedFile,
|
||||
expandedDirs: toPathSet(expandedPaths),
|
||||
checkedFiles: checkedPaths ? toPathSet(checkedPaths) : undefined,
|
||||
showCheckboxes,
|
||||
onToggleDir: onToggleDir ?? (() => undefined),
|
||||
onSelectFile: onSelectFile ?? (() => undefined),
|
||||
});
|
||||
}
|
||||
|
||||
type PluginMarkdownBlockProps = {
|
||||
content: string;
|
||||
className?: string;
|
||||
enableWikiLinks?: boolean;
|
||||
wikiLinkRoot?: string;
|
||||
resolveWikiLinkHref?: (target: string, label: string) => string | null | undefined;
|
||||
};
|
||||
|
||||
type PluginMarkdownEditorProps = {
|
||||
value: string;
|
||||
onChange: (value: string) => void;
|
||||
placeholder?: string;
|
||||
className?: string;
|
||||
contentClassName?: string;
|
||||
onBlur?: () => void;
|
||||
bordered?: boolean;
|
||||
readOnly?: boolean;
|
||||
onSubmit?: () => void;
|
||||
};
|
||||
|
||||
type PluginIssuesListFilters = {
|
||||
status?: string;
|
||||
projectId?: string;
|
||||
parentId?: string;
|
||||
assigneeAgentId?: string;
|
||||
participantAgentId?: string;
|
||||
assigneeUserId?: string;
|
||||
labelId?: string;
|
||||
workspaceId?: string;
|
||||
executionWorkspaceId?: string;
|
||||
originKind?: string;
|
||||
originKindPrefix?: string;
|
||||
originId?: string;
|
||||
descendantOf?: string;
|
||||
includeRoutineExecutions?: boolean;
|
||||
};
|
||||
|
||||
type PluginIssuesListProps = {
|
||||
companyId: string | null;
|
||||
projectId?: string | null;
|
||||
filters?: PluginIssuesListFilters;
|
||||
viewStateKey?: string;
|
||||
initialSearch?: string;
|
||||
createIssueLabel?: string;
|
||||
searchWithinLoadedIssues?: boolean;
|
||||
};
|
||||
|
||||
type PluginAssigneePickerSelection = {
|
||||
assigneeAgentId: string | null;
|
||||
assigneeUserId: string | null;
|
||||
};
|
||||
|
||||
type PluginAssigneePickerProps = {
|
||||
companyId?: string | null;
|
||||
value: string;
|
||||
onChange: (value: string, selection: PluginAssigneePickerSelection) => void;
|
||||
placeholder?: string;
|
||||
noneLabel?: string;
|
||||
searchPlaceholder?: string;
|
||||
emptyMessage?: string;
|
||||
includeUsers?: boolean;
|
||||
includeTerminatedAgents?: boolean;
|
||||
className?: string;
|
||||
onConfirm?: () => void;
|
||||
};
|
||||
|
||||
type PluginProjectPickerProps = {
|
||||
companyId?: string | null;
|
||||
value: string;
|
||||
onChange: (projectId: string) => void;
|
||||
placeholder?: string;
|
||||
noneLabel?: string;
|
||||
searchPlaceholder?: string;
|
||||
emptyMessage?: string;
|
||||
includeArchived?: boolean;
|
||||
className?: string;
|
||||
onConfirm?: () => void;
|
||||
};
|
||||
|
||||
function PluginSdkMarkdownEditor(props: PluginMarkdownEditorProps) {
|
||||
const [Editor, setEditor] = useState<ComponentType<PluginMarkdownEditorProps> | null>(null);
|
||||
|
||||
useEffect(() => {
|
||||
let cancelled = false;
|
||||
import("@/components/MarkdownEditor").then((module) => {
|
||||
if (!cancelled) setEditor(() => module.MarkdownEditor as ComponentType<PluginMarkdownEditorProps>);
|
||||
});
|
||||
return () => {
|
||||
cancelled = true;
|
||||
};
|
||||
}, []);
|
||||
|
||||
if (Editor) return createElement(Editor, props);
|
||||
|
||||
return createElement("textarea", {
|
||||
className: props.className,
|
||||
value: props.value,
|
||||
placeholder: props.placeholder,
|
||||
readOnly: props.readOnly,
|
||||
onBlur: props.onBlur,
|
||||
onChange: (event) => props.onChange((event.currentTarget as HTMLTextAreaElement).value),
|
||||
});
|
||||
}
|
||||
|
||||
function compactIssueFilters(filters: PluginIssuesListFilters): PluginIssuesListFilters {
|
||||
return Object.fromEntries(
|
||||
Object.entries(filters).filter(([, value]) =>
|
||||
value !== undefined && value !== null && value !== "" && value !== false,
|
||||
),
|
||||
) as PluginIssuesListFilters;
|
||||
}
|
||||
|
||||
function PluginSdkIssuesList({
|
||||
companyId,
|
||||
projectId = null,
|
||||
filters,
|
||||
viewStateKey = "paperclip:plugin-issues-view",
|
||||
initialSearch,
|
||||
createIssueLabel,
|
||||
searchWithinLoadedIssues = true,
|
||||
}: PluginIssuesListProps) {
|
||||
const queryClient = useQueryClient();
|
||||
const issueFilters = useMemo(
|
||||
() => compactIssueFilters({
|
||||
...(filters ?? {}),
|
||||
projectId: filters?.projectId ?? projectId ?? undefined,
|
||||
}),
|
||||
[filters, projectId],
|
||||
);
|
||||
const originKindPrefix = issueFilters.originKindPrefix ?? null;
|
||||
const resolvedProjectId = issueFilters.projectId ?? projectId ?? null;
|
||||
const issuesQueryKey = useMemo(
|
||||
() => ["plugins", "sdk-ui", "issues-list", companyId ?? "__no-company__", issueFilters] as const,
|
||||
[companyId, issueFilters],
|
||||
);
|
||||
|
||||
const { data: agents } = useQuery({
|
||||
queryKey: queryKeys.agents.list(companyId ?? "__no-company__"),
|
||||
queryFn: () => agentsApi.list(companyId!),
|
||||
enabled: !!companyId,
|
||||
});
|
||||
const { data: projects } = useQuery({
|
||||
queryKey: queryKeys.projects.list(companyId ?? "__no-company__"),
|
||||
queryFn: () => projectsApi.list(companyId!),
|
||||
enabled: !!companyId,
|
||||
});
|
||||
const { data: liveRuns } = useQuery({
|
||||
queryKey: queryKeys.liveRuns(companyId ?? "__no-company__"),
|
||||
queryFn: () => heartbeatsApi.liveRunsForCompany(companyId!),
|
||||
enabled: !!companyId,
|
||||
refetchInterval: 5000,
|
||||
});
|
||||
const liveIssueIds = useMemo(() => collectLiveIssueIds(liveRuns), [liveRuns]);
|
||||
|
||||
const { data: issues, isLoading, error } = useQuery({
|
||||
queryKey: issuesQueryKey,
|
||||
queryFn: () => issuesApi.list(companyId!, issueFilters),
|
||||
enabled: !!companyId,
|
||||
});
|
||||
|
||||
const updateIssue = useMutation({
|
||||
mutationFn: ({ id, data }: { id: string; data: Record<string, unknown> }) =>
|
||||
issuesApi.update(id, data),
|
||||
onSuccess: () => {
|
||||
if (!companyId) return;
|
||||
queryClient.invalidateQueries({ queryKey: ["plugins", "sdk-ui", "issues-list", companyId] });
|
||||
queryClient.invalidateQueries({ queryKey: queryKeys.issues.list(companyId) });
|
||||
if (resolvedProjectId) {
|
||||
queryClient.invalidateQueries({ queryKey: queryKeys.issues.listByProject(companyId, resolvedProjectId) });
|
||||
if (originKindPrefix) {
|
||||
queryClient.invalidateQueries({
|
||||
queryKey: queryKeys.issues.listPluginOperationsByProject(companyId, resolvedProjectId, originKindPrefix),
|
||||
});
|
||||
}
|
||||
}
|
||||
},
|
||||
});
|
||||
|
||||
if (!companyId) {
|
||||
return createElement("div", { className: "text-sm text-muted-foreground" }, "Select a company to view issues.");
|
||||
}
|
||||
|
||||
return createElement(HostIssuesList, {
|
||||
issues: issues ?? [],
|
||||
isLoading,
|
||||
error: error as Error | null,
|
||||
agents,
|
||||
projects,
|
||||
liveIssueIds,
|
||||
projectId: resolvedProjectId ?? undefined,
|
||||
viewStateKey,
|
||||
initialSearch,
|
||||
createIssueLabel,
|
||||
searchWithinLoadedIssues,
|
||||
onUpdateIssue: (id: string, data: Record<string, unknown>) => updateIssue.mutate({ id, data }),
|
||||
});
|
||||
}
|
||||
|
||||
function PluginSdkAssigneePicker({
|
||||
companyId,
|
||||
value,
|
||||
onChange,
|
||||
placeholder = "Assignee",
|
||||
noneLabel = "No assignee",
|
||||
searchPlaceholder = "Search assignees...",
|
||||
emptyMessage = "No assignees found.",
|
||||
includeUsers = true,
|
||||
includeTerminatedAgents = false,
|
||||
className,
|
||||
onConfirm,
|
||||
}: PluginAssigneePickerProps) {
|
||||
const hostContext = useHostContext();
|
||||
const resolvedCompanyId = companyId ?? hostContext.companyId ?? null;
|
||||
const { data: session } = useQuery({
|
||||
queryKey: queryKeys.auth.session,
|
||||
queryFn: () => authApi.getSession(),
|
||||
enabled: includeUsers,
|
||||
});
|
||||
const currentUserId = session?.user?.id ?? session?.session?.userId ?? null;
|
||||
const { data: agents } = useQuery({
|
||||
queryKey: queryKeys.agents.list(resolvedCompanyId ?? "__no-company__"),
|
||||
queryFn: () => agentsApi.list(resolvedCompanyId!),
|
||||
enabled: !!resolvedCompanyId,
|
||||
});
|
||||
const { data: companyMembers } = useQuery({
|
||||
queryKey: queryKeys.access.companyUserDirectory(resolvedCompanyId ?? "__no-company__"),
|
||||
queryFn: () => accessApi.listUserDirectory(resolvedCompanyId!),
|
||||
enabled: !!resolvedCompanyId && includeUsers,
|
||||
});
|
||||
const recentAssigneeSelectionIds = useMemo(() => getRecentAssigneeSelectionIds(), []);
|
||||
const recentAssigneeIds = useMemo(
|
||||
() => recentAssigneeSelectionIds
|
||||
.map((id) => id.startsWith("agent:") ? id.slice("agent:".length) : null)
|
||||
.filter((id): id is string => Boolean(id)),
|
||||
[recentAssigneeSelectionIds],
|
||||
);
|
||||
const sortedAgents = useMemo(
|
||||
() => sortAgentsByRecency(
|
||||
(agents ?? []).filter((agent) => includeTerminatedAgents || agent.status !== "terminated"),
|
||||
recentAssigneeIds,
|
||||
),
|
||||
[agents, includeTerminatedAgents, recentAssigneeIds],
|
||||
);
|
||||
const options = useMemo<InlineEntityOption[]>(
|
||||
() => [
|
||||
...(includeUsers ? currentUserAssigneeOption(currentUserId) : []),
|
||||
...(includeUsers
|
||||
? buildCompanyUserInlineOptions(companyMembers?.users, { excludeUserIds: [currentUserId] })
|
||||
: []),
|
||||
...sortedAgents.map((agent) => ({
|
||||
id: assigneeValueFromSelection({ assigneeAgentId: agent.id }),
|
||||
label: agent.name,
|
||||
searchText: `${agent.name} ${agent.role} ${agent.title ?? ""}`,
|
||||
})),
|
||||
],
|
||||
[companyMembers?.users, currentUserId, includeUsers, sortedAgents],
|
||||
);
|
||||
const selectedAssignee = parseAssigneeValue(value);
|
||||
const selectedAgent = selectedAssignee.assigneeAgentId
|
||||
? sortedAgents.find((agent) => agent.id === selectedAssignee.assigneeAgentId)
|
||||
: null;
|
||||
|
||||
return createElement(InlineEntitySelector, {
|
||||
value,
|
||||
options,
|
||||
recentOptionIds: recentAssigneeSelectionIds,
|
||||
placeholder,
|
||||
noneLabel,
|
||||
searchPlaceholder,
|
||||
emptyMessage,
|
||||
className,
|
||||
onConfirm,
|
||||
onChange: (nextValue: string) => {
|
||||
const selection = parseAssigneeValue(nextValue);
|
||||
if (selection.assigneeAgentId) trackRecentAssignee(selection.assigneeAgentId);
|
||||
if (selection.assigneeUserId) trackRecentAssigneeUser(selection.assigneeUserId);
|
||||
onChange(nextValue, selection);
|
||||
},
|
||||
renderTriggerValue: (option: InlineEntityOption | null) => {
|
||||
if (!option) return createElement("span", { className: "text-muted-foreground" }, placeholder);
|
||||
if (selectedAgent) {
|
||||
return createElement(
|
||||
FragmentSafe,
|
||||
null,
|
||||
createElement(AgentIcon, { icon: selectedAgent.icon, className: "h-3.5 w-3.5 shrink-0 text-muted-foreground" }),
|
||||
createElement("span", { className: "truncate" }, option.label),
|
||||
);
|
||||
}
|
||||
return createElement("span", { className: "truncate" }, option.label);
|
||||
},
|
||||
renderOption: (option: InlineEntityOption) => {
|
||||
if (!option.id) return createElement("span", { className: "truncate" }, option.label);
|
||||
const selection = parseAssigneeValue(option.id);
|
||||
const agent = selection.assigneeAgentId
|
||||
? sortedAgents.find((entry) => entry.id === selection.assigneeAgentId)
|
||||
: null;
|
||||
return createElement(
|
||||
FragmentSafe,
|
||||
null,
|
||||
agent
|
||||
? createElement(AgentIcon, { icon: agent.icon, className: "h-3.5 w-3.5 shrink-0 text-muted-foreground" })
|
||||
: createElement(User, { className: "h-3.5 w-3.5 shrink-0 text-muted-foreground" }),
|
||||
createElement("span", { className: "truncate" }, option.label),
|
||||
);
|
||||
},
|
||||
});
|
||||
}
|
||||
|
||||
function PluginSdkProjectPicker({
|
||||
companyId,
|
||||
value,
|
||||
onChange,
|
||||
placeholder = "Project",
|
||||
noneLabel = "No project",
|
||||
searchPlaceholder = "Search projects...",
|
||||
emptyMessage = "No projects found.",
|
||||
includeArchived = false,
|
||||
className,
|
||||
onConfirm,
|
||||
}: PluginProjectPickerProps) {
|
||||
const hostContext = useHostContext();
|
||||
const resolvedCompanyId = companyId ?? hostContext.companyId ?? null;
|
||||
const { data: session } = useQuery({
|
||||
queryKey: queryKeys.auth.session,
|
||||
queryFn: () => authApi.getSession(),
|
||||
});
|
||||
const currentUserId = session?.user?.id ?? session?.session?.userId ?? null;
|
||||
const { data: projects } = useQuery({
|
||||
queryKey: queryKeys.projects.list(resolvedCompanyId ?? "__no-company__"),
|
||||
queryFn: () => projectsApi.list(resolvedCompanyId!),
|
||||
enabled: !!resolvedCompanyId,
|
||||
});
|
||||
const visibleProjects = useMemo(
|
||||
() => (projects ?? []).filter((project) => includeArchived || !project.archivedAt),
|
||||
[includeArchived, projects],
|
||||
);
|
||||
const { orderedProjects } = useProjectOrder({
|
||||
projects: visibleProjects,
|
||||
companyId: resolvedCompanyId,
|
||||
userId: currentUserId,
|
||||
});
|
||||
const recentProjectIds = useMemo(() => getRecentProjectIds(), []);
|
||||
const options = useMemo<InlineEntityOption[]>(
|
||||
() => orderedProjects.map((project) => ({
|
||||
id: project.id,
|
||||
label: project.name,
|
||||
searchText: project.description ?? "",
|
||||
})),
|
||||
[orderedProjects],
|
||||
);
|
||||
const selectedProject = orderedProjects.find((project) => project.id === value) ?? null;
|
||||
|
||||
return createElement(InlineEntitySelector, {
|
||||
value,
|
||||
options,
|
||||
recentOptionIds: recentProjectIds,
|
||||
placeholder,
|
||||
noneLabel,
|
||||
searchPlaceholder,
|
||||
emptyMessage,
|
||||
className,
|
||||
onConfirm,
|
||||
onChange: (nextProjectId: string) => {
|
||||
if (nextProjectId) trackRecentProject(nextProjectId);
|
||||
onChange(nextProjectId);
|
||||
},
|
||||
renderTriggerValue: (option: InlineEntityOption | null) => {
|
||||
if (!option || !selectedProject) {
|
||||
return createElement("span", { className: "text-muted-foreground" }, placeholder);
|
||||
}
|
||||
return createElement(
|
||||
FragmentSafe,
|
||||
null,
|
||||
createElement("span", {
|
||||
className: "h-3.5 w-3.5 shrink-0 rounded-sm",
|
||||
style: { backgroundColor: selectedProject.color ?? "#6366f1" },
|
||||
}),
|
||||
createElement("span", { className: "truncate" }, option.label),
|
||||
);
|
||||
},
|
||||
renderOption: (option: InlineEntityOption) => {
|
||||
if (!option.id) return createElement("span", { className: "truncate" }, option.label);
|
||||
const project = orderedProjects.find((entry) => entry.id === option.id);
|
||||
return createElement(
|
||||
FragmentSafe,
|
||||
null,
|
||||
createElement("span", {
|
||||
className: "h-3.5 w-3.5 shrink-0 rounded-sm",
|
||||
style: { backgroundColor: project?.color ?? "#6366f1" },
|
||||
}),
|
||||
createElement("span", { className: "truncate" }, option.label),
|
||||
);
|
||||
},
|
||||
});
|
||||
}
|
||||
|
||||
function FragmentSafe({ children }: { children?: ReactNode }) {
|
||||
return createElement("span", { className: "contents" }, children);
|
||||
}
|
||||
|
||||
/**
|
||||
* Initialize the plugin bridge global registry.
|
||||
*
|
||||
@@ -62,8 +545,31 @@ export function initPluginBridge(
|
||||
usePluginData,
|
||||
usePluginAction,
|
||||
useHostContext,
|
||||
useHostLocation,
|
||||
useHostNavigation,
|
||||
usePluginStream,
|
||||
usePluginToast,
|
||||
MarkdownBlock: ({
|
||||
content,
|
||||
className,
|
||||
enableWikiLinks,
|
||||
wikiLinkRoot,
|
||||
resolveWikiLinkHref,
|
||||
}: PluginMarkdownBlockProps) =>
|
||||
createElement(MarkdownBody, {
|
||||
className,
|
||||
softBreaks: false,
|
||||
enableWikiLinks,
|
||||
wikiLinkRoot,
|
||||
resolveWikiLinkHref,
|
||||
children: content,
|
||||
}),
|
||||
MarkdownEditor: PluginSdkMarkdownEditor,
|
||||
FileTree: PluginSdkFileTree,
|
||||
IssuesList: PluginSdkIssuesList,
|
||||
AssigneePicker: PluginSdkAssigneePicker,
|
||||
ProjectPicker: PluginSdkProjectPicker,
|
||||
ManagedRoutinesList: HostManagedRoutinesList,
|
||||
},
|
||||
};
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user