mirror of
https://github.com/docmost/docmost.git
synced 2026-05-08 23:33:09 +08:00
Compare commits
10 Commits
feature-flag
...
fix-245
| Author | SHA1 | Date | |
|---|---|---|---|
| cc5c800238 | |||
| cfaee93af9 | |||
| 74eddb0638 | |||
| 7c83a9d4f0 | |||
| 2678c4e279 | |||
| b0bde4b375 | |||
| 724e37d5b7 | |||
| 33184e9d8d | |||
| 7520c329d0 | |||
| d7a5fda53c |
@@ -1,7 +1,7 @@
|
|||||||
{
|
{
|
||||||
"name": "client",
|
"name": "client",
|
||||||
"private": true,
|
"private": true,
|
||||||
"version": "0.70.1",
|
"version": "0.70.3",
|
||||||
"scripts": {
|
"scripts": {
|
||||||
"dev": "vite",
|
"dev": "vite",
|
||||||
"build": "tsc && vite build",
|
"build": "tsc && vite build",
|
||||||
|
|||||||
@@ -444,7 +444,6 @@
|
|||||||
"Toggle space public sharing": "Toggle space public sharing",
|
"Toggle space public sharing": "Toggle space public sharing",
|
||||||
"Public sharing is disabled at the workspace level": "Public sharing is disabled at the workspace level",
|
"Public sharing is disabled at the workspace level": "Public sharing is disabled at the workspace level",
|
||||||
"Prevent pages in this space from being shared publicly.": "Prevent pages in this space from being shared publicly.",
|
"Prevent pages in this space from being shared publicly.": "Prevent pages in this space from being shared publicly.",
|
||||||
"Requires an enterprise license": "Requires an enterprise license",
|
|
||||||
"Page permissions": "Page permissions",
|
"Page permissions": "Page permissions",
|
||||||
"Control who can view and edit individual pages. Available with an enterprise license.": "Control who can view and edit individual pages. Available with an enterprise license.",
|
"Control who can view and edit individual pages. Available with an enterprise license.": "Control who can view and edit individual pages. Available with an enterprise license.",
|
||||||
"Enable public sharing": "Enable public sharing",
|
"Enable public sharing": "Enable public sharing",
|
||||||
@@ -626,7 +625,9 @@
|
|||||||
"Generative AI (Ask AI)": "Generative AI (Ask AI)",
|
"Generative AI (Ask AI)": "Generative AI (Ask AI)",
|
||||||
"Enable AI-powered content generation in the editor. Allows users to generate, improve, translate and transform text.": "Enable AI-powered content generation in the editor. Allows users to generate, improve, translate and transform text.",
|
"Enable AI-powered content generation in the editor. Allows users to generate, improve, translate and transform text.": "Enable AI-powered content generation in the editor. Allows users to generate, improve, translate and transform text.",
|
||||||
"Toggle generative AI": "Toggle generative AI",
|
"Toggle generative AI": "Toggle generative AI",
|
||||||
"Enterprise feature": "Enterprise feature",
|
"Upgrade your plan": "Upgrade your plan",
|
||||||
|
"Available with a paid license": "Available with a paid license",
|
||||||
|
"Upgrade your license tier.": "Upgrade your license tier.",
|
||||||
"AI is only available in the Docmost enterprise edition. Contact sales@docmost.com.": "AI is only available in the Docmost enterprise edition. Contact sales@docmost.com.",
|
"AI is only available in the Docmost enterprise edition. Contact sales@docmost.com.": "AI is only available in the Docmost enterprise edition. Contact sales@docmost.com.",
|
||||||
"AI & MCP": "AI & MCP",
|
"AI & MCP": "AI & MCP",
|
||||||
"AI": "AI",
|
"AI": "AI",
|
||||||
@@ -634,17 +635,15 @@
|
|||||||
"Model Context Protocol (MCP)": "Model Context Protocol (MCP)",
|
"Model Context Protocol (MCP)": "Model Context Protocol (MCP)",
|
||||||
"Enable the MCP server to allow AI assistants and tools to interact with your workspace content.": "Enable the MCP server to allow AI assistants and tools to interact with your workspace content.",
|
"Enable the MCP server to allow AI assistants and tools to interact with your workspace content.": "Enable the MCP server to allow AI assistants and tools to interact with your workspace content.",
|
||||||
"MCP is only available in the Docmost enterprise edition. Contact sales@docmost.com.": "MCP is only available in the Docmost enterprise edition. Contact sales@docmost.com.",
|
"MCP is only available in the Docmost enterprise edition. Contact sales@docmost.com.": "MCP is only available in the Docmost enterprise edition. Contact sales@docmost.com.",
|
||||||
"MCP documentation": "MCP documentation",
|
|
||||||
"MCP Server URL": "MCP Server URL",
|
"MCP Server URL": "MCP Server URL",
|
||||||
"Use your API key for authentication. You can manage API keys in your account settings.": "Use your API key for authentication. You can manage API keys in your account settings.",
|
"Use your API key for authentication. You can manage API keys in your account settings.": "Use your API key for authentication. You can manage API keys in your account settings.",
|
||||||
"Supported tools": "Supported tools",
|
"Supported tools": "Supported tools",
|
||||||
"Your workspace has MCP enabled. Use your API key to connect AI assistants.": "Your workspace has MCP enabled. Use your API key to connect AI assistants.",
|
"Your workspace has MCP enabled. Use your API key to connect AI assistants.": "Your workspace has MCP enabled. Use your API key to connect AI assistants.",
|
||||||
"MCP server URL:": "MCP server URL:",
|
"MCP server URL:": "MCP server URL:",
|
||||||
"Learn more": "Learn more",
|
"Learn more": "Learn more",
|
||||||
"View the": "View the",
|
"Manage API keys for all users in the workspace. View the <anchor>API documentation</anchor> for usage details.": "Manage API keys for all users in the workspace. View the <anchor>API documentation</anchor> for usage details.",
|
||||||
"for usage details.": "for usage details.",
|
"View the <anchor>API documentation</anchor> for usage details.": "View the <anchor>API documentation</anchor> for usage details.",
|
||||||
"for setup instructions.": "for setup instructions.",
|
"View the <anchor>MCP documentation</anchor>.": "View the <anchor>MCP documentation</anchor>.",
|
||||||
"API documentation": "API documentation",
|
|
||||||
"Sources": "Sources",
|
"Sources": "Sources",
|
||||||
"AI Answers not available for attachments": "AI Answers not available for attachments",
|
"AI Answers not available for attachments": "AI Answers not available for attachments",
|
||||||
"No answer available": "No answer available",
|
"No answer available": "No answer available",
|
||||||
@@ -659,12 +658,12 @@
|
|||||||
"Mark all as read": "Mark all as read",
|
"Mark all as read": "Mark all as read",
|
||||||
"Mark as read": "Mark as read",
|
"Mark as read": "Mark as read",
|
||||||
"More options": "More options",
|
"More options": "More options",
|
||||||
"mentioned you in a comment": "mentioned you in a comment",
|
"<bold>{{name}}</bold> mentioned you in a comment": "<bold>{{name}}</bold> mentioned you in a comment",
|
||||||
"commented on a page": "commented on a page",
|
"<bold>{{name}}</bold> commented on a page": "<bold>{{name}}</bold> commented on a page",
|
||||||
"resolved a comment": "resolved a comment",
|
"<bold>{{name}}</bold> resolved a comment": "<bold>{{name}}</bold> resolved a comment",
|
||||||
"mentioned you on a page": "mentioned you on a page",
|
"<bold>{{name}}</bold> mentioned you on a page": "<bold>{{name}}</bold> mentioned you on a page",
|
||||||
"gave you edit access to a page": "gave you edit access to a page",
|
"<bold>{{name}}</bold> gave you edit access to a page": "<bold>{{name}}</bold> gave you edit access to a page",
|
||||||
"gave you view access to a page": "gave you view access to a page",
|
"<bold>{{name}}</bold> gave you view access to a page": "<bold>{{name}}</bold> gave you view access to a page",
|
||||||
"Today": "Today",
|
"Today": "Today",
|
||||||
"Yesterday": "Yesterday",
|
"Yesterday": "Yesterday",
|
||||||
"This week": "This week",
|
"This week": "This week",
|
||||||
|
|||||||
@@ -21,7 +21,9 @@ import { useTranslation } from "react-i18next";
|
|||||||
import { isCloud } from "@/lib/config.ts";
|
import { isCloud } from "@/lib/config.ts";
|
||||||
import useUserRole from "@/hooks/use-user-role.tsx";
|
import useUserRole from "@/hooks/use-user-role.tsx";
|
||||||
import { useAtom } from "jotai";
|
import { useAtom } from "jotai";
|
||||||
import { workspaceAtom } from "@/features/user/atoms/current-user-atom.ts";
|
import { entitlementAtom } from "@/ee/entitlement/entitlement-atom";
|
||||||
|
import { Feature } from "@/ee/features";
|
||||||
|
import { useUpgradeLabel } from "@/ee/hooks/use-upgrade-label";
|
||||||
import {
|
import {
|
||||||
prefetchApiKeyManagement,
|
prefetchApiKeyManagement,
|
||||||
prefetchApiKeys,
|
prefetchApiKeys,
|
||||||
@@ -39,22 +41,19 @@ import { mobileSidebarAtom } from "@/components/layouts/global/hooks/atoms/sideb
|
|||||||
import { useToggleSidebar } from "@/components/layouts/global/hooks/hooks/use-toggle-sidebar.ts";
|
import { useToggleSidebar } from "@/components/layouts/global/hooks/hooks/use-toggle-sidebar.ts";
|
||||||
import { useSettingsNavigation } from "@/hooks/use-settings-navigation";
|
import { useSettingsNavigation } from "@/hooks/use-settings-navigation";
|
||||||
|
|
||||||
interface DataItem {
|
type DataItem = {
|
||||||
label: string;
|
label: string;
|
||||||
icon: React.ElementType;
|
icon: React.ElementType;
|
||||||
path: string;
|
path: string;
|
||||||
isCloud?: boolean;
|
feature?: string;
|
||||||
isEnterprise?: boolean;
|
role?: "admin" | "owner";
|
||||||
isAdmin?: boolean;
|
env?: "cloud" | "selfhosted";
|
||||||
isOwner?: boolean;
|
};
|
||||||
isSelfhosted?: boolean;
|
|
||||||
showDisabledInNonEE?: boolean;
|
|
||||||
}
|
|
||||||
|
|
||||||
interface DataGroup {
|
type DataGroup = {
|
||||||
heading: string;
|
heading: string;
|
||||||
items: DataItem[];
|
items: DataItem[];
|
||||||
}
|
};
|
||||||
|
|
||||||
const groupedData: DataGroup[] = [
|
const groupedData: DataGroup[] = [
|
||||||
{
|
{
|
||||||
@@ -70,9 +69,7 @@ const groupedData: DataGroup[] = [
|
|||||||
label: "API keys",
|
label: "API keys",
|
||||||
icon: IconKey,
|
icon: IconKey,
|
||||||
path: "/settings/account/api-keys",
|
path: "/settings/account/api-keys",
|
||||||
isCloud: true,
|
feature: Feature.API_KEYS,
|
||||||
isEnterprise: true,
|
|
||||||
showDisabledInNonEE: true,
|
|
||||||
},
|
},
|
||||||
],
|
],
|
||||||
},
|
},
|
||||||
@@ -80,26 +77,20 @@ const groupedData: DataGroup[] = [
|
|||||||
heading: "Workspace",
|
heading: "Workspace",
|
||||||
items: [
|
items: [
|
||||||
{ label: "General", icon: IconSettings, path: "/settings/workspace" },
|
{ label: "General", icon: IconSettings, path: "/settings/workspace" },
|
||||||
{
|
{ label: "Members", icon: IconUsers, path: "/settings/members" },
|
||||||
label: "Members",
|
|
||||||
icon: IconUsers,
|
|
||||||
path: "/settings/members",
|
|
||||||
},
|
|
||||||
{
|
{
|
||||||
label: "Billing",
|
label: "Billing",
|
||||||
icon: IconCoin,
|
icon: IconCoin,
|
||||||
path: "/settings/billing",
|
path: "/settings/billing",
|
||||||
isCloud: true,
|
role: "admin",
|
||||||
isAdmin: true,
|
env: "cloud",
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
label: "Security & SSO",
|
label: "Security & SSO",
|
||||||
icon: IconLock,
|
icon: IconLock,
|
||||||
path: "/settings/security",
|
path: "/settings/security",
|
||||||
isCloud: true,
|
feature: Feature.SECURITY_SETTINGS,
|
||||||
isEnterprise: true,
|
role: "admin",
|
||||||
isAdmin: true,
|
|
||||||
showDisabledInNonEE: true,
|
|
||||||
},
|
},
|
||||||
{ label: "Groups", icon: IconUsersGroup, path: "/settings/groups" },
|
{ label: "Groups", icon: IconUsersGroup, path: "/settings/groups" },
|
||||||
{ label: "Spaces", icon: IconSpaces, path: "/settings/spaces" },
|
{ label: "Spaces", icon: IconSpaces, path: "/settings/spaces" },
|
||||||
@@ -108,25 +99,22 @@ const groupedData: DataGroup[] = [
|
|||||||
label: "API management",
|
label: "API management",
|
||||||
icon: IconKey,
|
icon: IconKey,
|
||||||
path: "/settings/api-keys",
|
path: "/settings/api-keys",
|
||||||
isCloud: true,
|
feature: Feature.API_KEYS,
|
||||||
isEnterprise: true,
|
role: "admin",
|
||||||
isAdmin: true,
|
|
||||||
showDisabledInNonEE: true,
|
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
label: "AI settings",
|
label: "AI settings",
|
||||||
icon: IconSparkles,
|
icon: IconSparkles,
|
||||||
path: "/settings/ai",
|
path: "/settings/ai",
|
||||||
isAdmin: true,
|
role: "admin",
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
label: "Audit log",
|
label: "Audit log",
|
||||||
icon: IconHistory,
|
icon: IconHistory,
|
||||||
path: "/settings/audit",
|
path: "/settings/audit",
|
||||||
isEnterprise: true,
|
feature: Feature.AUDIT_LOGS,
|
||||||
isOwner: true,
|
role: "owner",
|
||||||
isSelfhosted: true,
|
env: "selfhosted",
|
||||||
showDisabledInNonEE: true,
|
|
||||||
},
|
},
|
||||||
],
|
],
|
||||||
},
|
},
|
||||||
@@ -148,7 +136,8 @@ export default function SettingsSidebar() {
|
|||||||
const [active, setActive] = useState(location.pathname);
|
const [active, setActive] = useState(location.pathname);
|
||||||
const { goBack } = useSettingsNavigation();
|
const { goBack } = useSettingsNavigation();
|
||||||
const { isAdmin, isOwner } = useUserRole();
|
const { isAdmin, isOwner } = useUserRole();
|
||||||
const [workspace] = useAtom(workspaceAtom);
|
const [entitlements] = useAtom(entitlementAtom);
|
||||||
|
const upgradeLabel = useUpgradeLabel();
|
||||||
const [mobileSidebarOpened] = useAtom(mobileSidebarAtom);
|
const [mobileSidebarOpened] = useAtom(mobileSidebarAtom);
|
||||||
const toggleMobileSidebar = useToggleSidebar(mobileSidebarAtom);
|
const toggleMobileSidebar = useToggleSidebar(mobileSidebarAtom);
|
||||||
|
|
||||||
@@ -156,43 +145,20 @@ export default function SettingsSidebar() {
|
|||||||
setActive(location.pathname);
|
setActive(location.pathname);
|
||||||
}, [location.pathname]);
|
}, [location.pathname]);
|
||||||
|
|
||||||
const hasRoleAccess = (item: DataItem) => {
|
const hasFeature = (f: string) =>
|
||||||
if (item.isOwner) return isOwner;
|
entitlements?.features?.includes(f) ?? false;
|
||||||
if (item.isAdmin) return isAdmin;
|
|
||||||
|
const canShowItem = (item: DataItem) => {
|
||||||
|
if (item.env === "cloud" && !isCloud()) return false;
|
||||||
|
if (item.env === "selfhosted" && isCloud()) return false;
|
||||||
|
if (item.role === "admin" && !isAdmin) return false;
|
||||||
|
if (item.role === "owner" && !isOwner) return false;
|
||||||
return true;
|
return true;
|
||||||
};
|
};
|
||||||
|
|
||||||
const canShowItem = (item: DataItem) => {
|
|
||||||
if (item.showDisabledInNonEE && item.isEnterprise) {
|
|
||||||
if (item.isSelfhosted && isCloud()) return false;
|
|
||||||
return hasRoleAccess(item);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (item.isCloud && item.isEnterprise) {
|
|
||||||
if (!(isCloud() || workspace?.hasLicenseKey)) return false;
|
|
||||||
return hasRoleAccess(item);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (item.isCloud) {
|
|
||||||
return isCloud() ? hasRoleAccess(item) : false;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (item.isSelfhosted) {
|
|
||||||
return !isCloud() ? hasRoleAccess(item) : false;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (item.isEnterprise) {
|
|
||||||
return workspace?.hasLicenseKey ? hasRoleAccess(item) : false;
|
|
||||||
}
|
|
||||||
|
|
||||||
return hasRoleAccess(item);
|
|
||||||
};
|
|
||||||
|
|
||||||
const isItemDisabled = (item: DataItem) => {
|
const isItemDisabled = (item: DataItem) => {
|
||||||
if (item.showDisabledInNonEE && item.isEnterprise) {
|
if (!item.feature) return false;
|
||||||
return !(isCloud() || workspace?.hasLicenseKey);
|
return !hasFeature(item.feature);
|
||||||
}
|
|
||||||
return false;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
const menuItems = groupedData.map((group) => {
|
const menuItems = groupedData.map((group) => {
|
||||||
@@ -225,7 +191,7 @@ export default function SettingsSidebar() {
|
|||||||
prefetchHandler = prefetchBilling;
|
prefetchHandler = prefetchBilling;
|
||||||
break;
|
break;
|
||||||
case "License & Edition":
|
case "License & Edition":
|
||||||
if (workspace?.hasLicenseKey) {
|
if (entitlements?.tier !== "free") {
|
||||||
prefetchHandler = prefetchLicense;
|
prefetchHandler = prefetchLicense;
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
@@ -280,7 +246,7 @@ export default function SettingsSidebar() {
|
|||||||
return (
|
return (
|
||||||
<Tooltip
|
<Tooltip
|
||||||
key={item.label}
|
key={item.label}
|
||||||
label={t("Available in enterprise edition")}
|
label={upgradeLabel}
|
||||||
position="right"
|
position="right"
|
||||||
withArrow
|
withArrow
|
||||||
>
|
>
|
||||||
|
|||||||
@@ -1,12 +1,13 @@
|
|||||||
import { Group, Text, Switch, MantineSize, Title } from "@mantine/core";
|
import { Group, Text, Switch, MantineSize, Tooltip } from "@mantine/core";
|
||||||
import { useAtom } from "jotai";
|
import { useAtom } from "jotai";
|
||||||
import { workspaceAtom } from "@/features/user/atoms/current-user-atom.ts";
|
import { workspaceAtom } from "@/features/user/atoms/current-user-atom.ts";
|
||||||
import React, { useState } from "react";
|
import React, { useState } from "react";
|
||||||
import { useTranslation } from "react-i18next";
|
import { useTranslation } from "react-i18next";
|
||||||
import { updateWorkspace } from "@/features/workspace/services/workspace-service.ts";
|
import { updateWorkspace } from "@/features/workspace/services/workspace-service.ts";
|
||||||
import { notifications } from "@mantine/notifications";
|
import { notifications } from "@mantine/notifications";
|
||||||
import { isCloud } from "@/lib/config.ts";
|
import { useHasFeature } from "@/ee/hooks/use-feature";
|
||||||
import useLicense from "@/ee/hooks/use-license.tsx";
|
import { Feature } from "@/ee/features";
|
||||||
|
import { useUpgradeLabel } from "@/ee/hooks/use-upgrade-label";
|
||||||
|
|
||||||
export default function EnableAiSearch() {
|
export default function EnableAiSearch() {
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
@@ -37,9 +38,8 @@ export function AiSearchToggle({ size, label }: AiSearchToggleProps) {
|
|||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
const [workspace, setWorkspace] = useAtom(workspaceAtom);
|
const [workspace, setWorkspace] = useAtom(workspaceAtom);
|
||||||
const [checked, setChecked] = useState(workspace?.settings?.ai?.search);
|
const [checked, setChecked] = useState(workspace?.settings?.ai?.search);
|
||||||
const { hasLicenseKey } = useLicense();
|
const hasAccess = useHasFeature(Feature.AI);
|
||||||
|
const upgradeLabel = useUpgradeLabel();
|
||||||
const hasAccess = isCloud() || (!isCloud() && hasLicenseKey);
|
|
||||||
|
|
||||||
const handleChange = async (event: React.ChangeEvent<HTMLInputElement>) => {
|
const handleChange = async (event: React.ChangeEvent<HTMLInputElement>) => {
|
||||||
const value = event.currentTarget.checked;
|
const value = event.currentTarget.checked;
|
||||||
@@ -56,14 +56,16 @@ export function AiSearchToggle({ size, label }: AiSearchToggleProps) {
|
|||||||
};
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Switch
|
<Tooltip label={upgradeLabel} disabled={hasAccess} refProp="rootRef">
|
||||||
size={size}
|
<Switch
|
||||||
label={label}
|
size={size}
|
||||||
labelPosition="left"
|
label={label}
|
||||||
defaultChecked={checked}
|
labelPosition="left"
|
||||||
onChange={handleChange}
|
defaultChecked={checked}
|
||||||
disabled={!hasAccess}
|
onChange={handleChange}
|
||||||
aria-label={t("Toggle AI search")}
|
disabled={!hasAccess}
|
||||||
/>
|
aria-label={t("Toggle AI search")}
|
||||||
|
/>
|
||||||
|
</Tooltip>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,17 +1,20 @@
|
|||||||
import { Group, Text, Switch } from "@mantine/core";
|
import { Group, Text, Switch, Tooltip } from "@mantine/core";
|
||||||
import { useAtom } from "jotai";
|
import { useAtom } from "jotai";
|
||||||
import { workspaceAtom } from "@/features/user/atoms/current-user-atom.ts";
|
import { workspaceAtom } from "@/features/user/atoms/current-user-atom.ts";
|
||||||
import React, { useState } from "react";
|
import React, { useState } from "react";
|
||||||
import { useTranslation } from "react-i18next";
|
import { useTranslation } from "react-i18next";
|
||||||
import { updateWorkspace } from "@/features/workspace/services/workspace-service.ts";
|
import { updateWorkspace } from "@/features/workspace/services/workspace-service.ts";
|
||||||
import { notifications } from "@mantine/notifications";
|
import { notifications } from "@mantine/notifications";
|
||||||
import { useIsCloudEE } from "@/hooks/use-is-cloud-ee.tsx";
|
import { useHasFeature } from "@/ee/hooks/use-feature";
|
||||||
|
import { Feature } from "@/ee/features";
|
||||||
|
import { useUpgradeLabel } from "@/ee/hooks/use-upgrade-label";
|
||||||
|
|
||||||
export default function EnableGenerativeAi() {
|
export default function EnableGenerativeAi() {
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
const [workspace, setWorkspace] = useAtom(workspaceAtom);
|
const [workspace, setWorkspace] = useAtom(workspaceAtom);
|
||||||
const [checked, setChecked] = useState(workspace?.settings?.ai?.generative);
|
const [checked, setChecked] = useState(workspace?.settings?.ai?.generative);
|
||||||
const hasAccess = useIsCloudEE();
|
const hasAccess = useHasFeature(Feature.AI);
|
||||||
|
const upgradeLabel = useUpgradeLabel();
|
||||||
|
|
||||||
const handleChange = async (event: React.ChangeEvent<HTMLInputElement>) => {
|
const handleChange = async (event: React.ChangeEvent<HTMLInputElement>) => {
|
||||||
const value = event.currentTarget.checked;
|
const value = event.currentTarget.checked;
|
||||||
@@ -38,11 +41,13 @@ export default function EnableGenerativeAi() {
|
|||||||
</Text>
|
</Text>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<Switch
|
<Tooltip label={upgradeLabel} disabled={hasAccess} refProp="rootRef">
|
||||||
defaultChecked={checked}
|
<Switch
|
||||||
onChange={handleChange}
|
defaultChecked={checked}
|
||||||
disabled={!hasAccess}
|
onChange={handleChange}
|
||||||
/>
|
disabled={!hasAccess}
|
||||||
|
/>
|
||||||
|
</Tooltip>
|
||||||
</Group>
|
</Group>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -13,10 +13,12 @@ import {
|
|||||||
import { useAtom } from "jotai";
|
import { useAtom } from "jotai";
|
||||||
import { workspaceAtom } from "@/features/user/atoms/current-user-atom.ts";
|
import { workspaceAtom } from "@/features/user/atoms/current-user-atom.ts";
|
||||||
import React, { useState } from "react";
|
import React, { useState } from "react";
|
||||||
import { useTranslation } from "react-i18next";
|
import { Trans, useTranslation } from "react-i18next";
|
||||||
import { updateWorkspace } from "@/features/workspace/services/workspace-service.ts";
|
import { updateWorkspace } from "@/features/workspace/services/workspace-service.ts";
|
||||||
import { notifications } from "@mantine/notifications";
|
import { notifications } from "@mantine/notifications";
|
||||||
import { useIsCloudEE } from "@/hooks/use-is-cloud-ee.tsx";
|
import { useHasFeature } from "@/ee/hooks/use-feature";
|
||||||
|
import { Feature } from "@/ee/features";
|
||||||
|
import { useUpgradeLabel } from "@/ee/hooks/use-upgrade-label";
|
||||||
import { getAppUrl } from "@/lib/config.ts";
|
import { getAppUrl } from "@/lib/config.ts";
|
||||||
import { IconCheck, IconCopy, IconInfoCircle } from "@tabler/icons-react";
|
import { IconCheck, IconCopy, IconInfoCircle } from "@tabler/icons-react";
|
||||||
import { CopyButton } from "@/components/common/copy-button.tsx";
|
import { CopyButton } from "@/components/common/copy-button.tsx";
|
||||||
@@ -25,7 +27,8 @@ export default function McpSettings() {
|
|||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
const [workspace, setWorkspace] = useAtom(workspaceAtom);
|
const [workspace, setWorkspace] = useAtom(workspaceAtom);
|
||||||
const [checked, setChecked] = useState(workspace?.settings?.ai?.mcp);
|
const [checked, setChecked] = useState(workspace?.settings?.ai?.mcp);
|
||||||
const hasAccess = useIsCloudEE();
|
const hasAccess = useHasFeature(Feature.MCP);
|
||||||
|
const upgradeLabel = useUpgradeLabel();
|
||||||
|
|
||||||
const mcpUrl = `${getAppUrl()}/mcp`;
|
const mcpUrl = `${getAppUrl()}/mcp`;
|
||||||
|
|
||||||
@@ -46,11 +49,7 @@ export default function McpSettings() {
|
|||||||
return (
|
return (
|
||||||
<Stack gap="lg">
|
<Stack gap="lg">
|
||||||
{!hasAccess && (
|
{!hasAccess && (
|
||||||
<Alert
|
<Alert icon={<IconInfoCircle />} title={upgradeLabel} color="blue">
|
||||||
icon={<IconInfoCircle />}
|
|
||||||
title={t("Enterprise feature")}
|
|
||||||
color="blue"
|
|
||||||
>
|
|
||||||
{t(
|
{t(
|
||||||
"MCP is only available in the Docmost enterprise edition. Contact sales@docmost.com.",
|
"MCP is only available in the Docmost enterprise edition. Contact sales@docmost.com.",
|
||||||
)}
|
)}
|
||||||
@@ -64,23 +63,22 @@ export default function McpSettings() {
|
|||||||
{t(
|
{t(
|
||||||
"Enable the MCP server to allow AI assistants and tools to interact with your workspace content.",
|
"Enable the MCP server to allow AI assistants and tools to interact with your workspace content.",
|
||||||
)}{" "}
|
)}{" "}
|
||||||
{t("View the")}{" "}
|
<Trans
|
||||||
<Anchor
|
i18nKey="View the <anchor>MCP documentation</anchor>."
|
||||||
href="https://docmost.com/docs/user-guide/mcp"
|
components={{
|
||||||
target="_blank"
|
anchor: <Anchor href="https://docmost.com/docs/user-guide/mcp" target="_blank" size="sm" />,
|
||||||
size="sm"
|
}}
|
||||||
>
|
/>
|
||||||
{t("MCP documentation")}
|
|
||||||
</Anchor>
|
|
||||||
.
|
|
||||||
</Text>
|
</Text>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<Switch
|
<Tooltip label={upgradeLabel} disabled={hasAccess} refProp="rootRef">
|
||||||
defaultChecked={checked}
|
<Switch
|
||||||
onChange={handleChange}
|
defaultChecked={checked}
|
||||||
disabled={!hasAccess}
|
onChange={handleChange}
|
||||||
/>
|
disabled={!hasAccess}
|
||||||
|
/>
|
||||||
|
</Tooltip>
|
||||||
</Group>
|
</Group>
|
||||||
|
|
||||||
{checked && (
|
{checked && (
|
||||||
@@ -89,11 +87,7 @@ export default function McpSettings() {
|
|||||||
{t("MCP Server URL")}
|
{t("MCP Server URL")}
|
||||||
</Text>
|
</Text>
|
||||||
<Group gap="xs">
|
<Group gap="xs">
|
||||||
<TextInput
|
<TextInput value={mcpUrl} readOnly style={{ flex: 1 }} />
|
||||||
value={mcpUrl}
|
|
||||||
readOnly
|
|
||||||
style={{ flex: 1 }}
|
|
||||||
/>
|
|
||||||
<CopyButton value={mcpUrl} timeout={2000}>
|
<CopyButton value={mcpUrl} timeout={2000}>
|
||||||
{({ copied, copy }) => (
|
{({ copied, copy }) => (
|
||||||
<Tooltip
|
<Tooltip
|
||||||
@@ -123,12 +117,36 @@ export default function McpSettings() {
|
|||||||
{t("Supported tools")}
|
{t("Supported tools")}
|
||||||
</Text>
|
</Text>
|
||||||
<List size="sm" spacing={2}>
|
<List size="sm" spacing={2}>
|
||||||
<List.Item><Text size="sm" c="dimmed" span>search_pages, get_page, create_page, update_page</Text></List.Item>
|
<List.Item>
|
||||||
<List.Item><Text size="sm" c="dimmed" span>list_pages, list_child_pages, duplicate_page</Text></List.Item>
|
<Text size="sm" c="dimmed" span>
|
||||||
<List.Item><Text size="sm" c="dimmed" span>copy_page_to_space, move_page, move_page_to_space</Text></List.Item>
|
search_pages, get_page, create_page, update_page
|
||||||
<List.Item><Text size="sm" c="dimmed" span>get_space, list_spaces, create_space, update_space</Text></List.Item>
|
</Text>
|
||||||
<List.Item><Text size="sm" c="dimmed" span>get_comments, create_comment, update_comment</Text></List.Item>
|
</List.Item>
|
||||||
<List.Item><Text size="sm" c="dimmed" span>search_attachments, list_workspace_members, get_current_user</Text></List.Item>
|
<List.Item>
|
||||||
|
<Text size="sm" c="dimmed" span>
|
||||||
|
list_pages, list_child_pages, duplicate_page
|
||||||
|
</Text>
|
||||||
|
</List.Item>
|
||||||
|
<List.Item>
|
||||||
|
<Text size="sm" c="dimmed" span>
|
||||||
|
copy_page_to_space, move_page, move_page_to_space
|
||||||
|
</Text>
|
||||||
|
</List.Item>
|
||||||
|
<List.Item>
|
||||||
|
<Text size="sm" c="dimmed" span>
|
||||||
|
get_space, list_spaces, create_space, update_space
|
||||||
|
</Text>
|
||||||
|
</List.Item>
|
||||||
|
<List.Item>
|
||||||
|
<Text size="sm" c="dimmed" span>
|
||||||
|
get_comments, create_comment, update_comment
|
||||||
|
</Text>
|
||||||
|
</List.Item>
|
||||||
|
<List.Item>
|
||||||
|
<Text size="sm" c="dimmed" span>
|
||||||
|
search_attachments, list_workspace_members, get_current_user
|
||||||
|
</Text>
|
||||||
|
</List.Item>
|
||||||
</List>
|
</List>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
@@ -9,14 +9,17 @@ import EnableGenerativeAi from "@/ee/ai/components/enable-generative-ai.tsx";
|
|||||||
import McpSettings from "@/ee/ai/components/mcp-settings.tsx";
|
import McpSettings from "@/ee/ai/components/mcp-settings.tsx";
|
||||||
import { Alert, Stack, Tabs } from "@mantine/core";
|
import { Alert, Stack, Tabs } from "@mantine/core";
|
||||||
import { IconInfoCircle } from "@tabler/icons-react";
|
import { IconInfoCircle } from "@tabler/icons-react";
|
||||||
import { useIsCloudEE } from "@/hooks/use-is-cloud-ee.tsx";
|
import { useHasFeature } from "@/ee/hooks/use-feature";
|
||||||
|
import { Feature } from "@/ee/features";
|
||||||
|
import { useUpgradeLabel } from "@/ee/hooks/use-upgrade-label";
|
||||||
import { isCloud } from "@/lib/config.ts";
|
import { isCloud } from "@/lib/config.ts";
|
||||||
import { useLocation, useNavigate } from "react-router-dom";
|
import { useLocation, useNavigate } from "react-router-dom";
|
||||||
|
|
||||||
export default function AiSettings() {
|
export default function AiSettings() {
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
const { isAdmin } = useUserRole();
|
const { isAdmin } = useUserRole();
|
||||||
const hasAccess = useIsCloudEE();
|
const hasAccess = useHasFeature(Feature.AI);
|
||||||
|
const upgradeLabel = useUpgradeLabel();
|
||||||
const location = useLocation();
|
const location = useLocation();
|
||||||
const navigate = useNavigate();
|
const navigate = useNavigate();
|
||||||
|
|
||||||
@@ -55,7 +58,7 @@ export default function AiSettings() {
|
|||||||
{!hasAccess && (
|
{!hasAccess && (
|
||||||
<Alert
|
<Alert
|
||||||
icon={<IconInfoCircle />}
|
icon={<IconInfoCircle />}
|
||||||
title={t("Enterprise feature")}
|
title={upgradeLabel}
|
||||||
color="blue"
|
color="blue"
|
||||||
mb="lg"
|
mb="lg"
|
||||||
>
|
>
|
||||||
|
|||||||
@@ -5,12 +5,14 @@ import React, { useState } from "react";
|
|||||||
import { useTranslation } from "react-i18next";
|
import { useTranslation } from "react-i18next";
|
||||||
import { updateWorkspace } from "@/features/workspace/services/workspace-service.ts";
|
import { updateWorkspace } from "@/features/workspace/services/workspace-service.ts";
|
||||||
import { notifications } from "@mantine/notifications";
|
import { notifications } from "@mantine/notifications";
|
||||||
import useEnterpriseAccess from "@/ee/hooks/use-enterprise-access.tsx";
|
import { useHasFeature } from "@/ee/hooks/use-feature";
|
||||||
|
import { Feature } from "@/ee/features";
|
||||||
import {
|
import {
|
||||||
ResponsiveSettingsRow,
|
ResponsiveSettingsRow,
|
||||||
ResponsiveSettingsContent,
|
ResponsiveSettingsContent,
|
||||||
ResponsiveSettingsControl,
|
ResponsiveSettingsControl,
|
||||||
} from "@/components/ui/responsive-settings-row";
|
} from "@/components/ui/responsive-settings-row";
|
||||||
|
import { useUpgradeLabel } from "@/ee/hooks/use-upgrade-label.ts";
|
||||||
|
|
||||||
export default function RestrictApiToAdmins() {
|
export default function RestrictApiToAdmins() {
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
@@ -18,7 +20,8 @@ export default function RestrictApiToAdmins() {
|
|||||||
const [checked, setChecked] = useState(
|
const [checked, setChecked] = useState(
|
||||||
workspace?.settings?.api?.restrictToAdmins === true,
|
workspace?.settings?.api?.restrictToAdmins === true,
|
||||||
);
|
);
|
||||||
const hasAccess = useEnterpriseAccess();
|
const hasAccess = useHasFeature(Feature.API_KEYS);
|
||||||
|
const upgradeLabel = useUpgradeLabel();
|
||||||
|
|
||||||
const handleChange = async (event: React.ChangeEvent<HTMLInputElement>) => {
|
const handleChange = async (event: React.ChangeEvent<HTMLInputElement>) => {
|
||||||
const value = event.currentTarget.checked;
|
const value = event.currentTarget.checked;
|
||||||
@@ -51,7 +54,7 @@ export default function RestrictApiToAdmins() {
|
|||||||
|
|
||||||
<ResponsiveSettingsControl>
|
<ResponsiveSettingsControl>
|
||||||
<Tooltip
|
<Tooltip
|
||||||
label={t("Requires an enterprise license")}
|
label={upgradeLabel}
|
||||||
disabled={hasAccess}
|
disabled={hasAccess}
|
||||||
refProp="rootRef"
|
refProp="rootRef"
|
||||||
>
|
>
|
||||||
|
|||||||
@@ -2,7 +2,7 @@ import React, { useState } from "react";
|
|||||||
import { Anchor, Alert, Button, Group, Space, Text } from "@mantine/core";
|
import { Anchor, Alert, Button, Group, Space, Text } from "@mantine/core";
|
||||||
import { IconInfoCircle } from "@tabler/icons-react";
|
import { IconInfoCircle } from "@tabler/icons-react";
|
||||||
import { Helmet } from "react-helmet-async";
|
import { Helmet } from "react-helmet-async";
|
||||||
import { useTranslation } from "react-i18next";
|
import { Trans, useTranslation } from "react-i18next";
|
||||||
import SettingsTitle from "@/components/settings/settings-title";
|
import SettingsTitle from "@/components/settings/settings-title";
|
||||||
import { getAppName, getAppUrl } from "@/lib/config";
|
import { getAppName, getAppUrl } from "@/lib/config";
|
||||||
import { ApiKeyTable } from "@/ee/api-key/components/api-key-table";
|
import { ApiKeyTable } from "@/ee/api-key/components/api-key-table";
|
||||||
@@ -58,11 +58,12 @@ export default function UserApiKeys() {
|
|||||||
<SettingsTitle title={t("API keys")} />
|
<SettingsTitle title={t("API keys")} />
|
||||||
|
|
||||||
<Text size="sm" c="dimmed" mb="md">
|
<Text size="sm" c="dimmed" mb="md">
|
||||||
{t("View the")}{" "}
|
<Trans
|
||||||
<Anchor href="https://docmost.com/api-docs" target="_blank" size="sm">
|
i18nKey="View the <anchor>API documentation</anchor> for usage details."
|
||||||
{t("API documentation")}
|
components={{
|
||||||
</Anchor>{" "}
|
anchor: <Anchor href="https://docmost.com/api-docs" target="_blank" size="sm" />,
|
||||||
{t("for usage details.")}
|
}}
|
||||||
|
/>
|
||||||
</Text>
|
</Text>
|
||||||
|
|
||||||
{mcpEnabled && canCreate && (
|
{mcpEnabled && canCreate && (
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
import React, { useState } from "react";
|
import React, { useState } from "react";
|
||||||
import { Anchor, Button, Divider, Group, Space, Text } from "@mantine/core";
|
import { Anchor, Button, Divider, Group, Space, Text } from "@mantine/core";
|
||||||
import { Helmet } from "react-helmet-async";
|
import { Helmet } from "react-helmet-async";
|
||||||
import { useTranslation } from "react-i18next";
|
import { Trans, useTranslation } from "react-i18next";
|
||||||
import SettingsTitle from "@/components/settings/settings-title";
|
import SettingsTitle from "@/components/settings/settings-title";
|
||||||
import { getAppName } from "@/lib/config";
|
import { getAppName } from "@/lib/config";
|
||||||
import { ApiKeyTable } from "@/ee/api-key/components/api-key-table";
|
import { ApiKeyTable } from "@/ee/api-key/components/api-key-table";
|
||||||
@@ -56,12 +56,12 @@ export default function WorkspaceApiKeys() {
|
|||||||
<SettingsTitle title={t("API management")} />
|
<SettingsTitle title={t("API management")} />
|
||||||
|
|
||||||
<Text size="sm" c="dimmed" mb="md">
|
<Text size="sm" c="dimmed" mb="md">
|
||||||
{t("Manage API keys for all users in the workspace.")}{" "}
|
<Trans
|
||||||
{t("View the")}{" "}
|
i18nKey="Manage API keys for all users in the workspace. View the <anchor>API documentation</anchor> for usage details."
|
||||||
<Anchor href="https://docmost.com/api-docs" target="_blank" size="sm">
|
components={{
|
||||||
{t("API documentation")}
|
anchor: <Anchor href="https://docmost.com/api-docs" target="_blank" size="sm" />,
|
||||||
</Anchor>{" "}
|
}}
|
||||||
{t("for usage details.")}
|
/>
|
||||||
</Text>
|
</Text>
|
||||||
|
|
||||||
<RestrictApiToAdmins />
|
<RestrictApiToAdmins />
|
||||||
|
|||||||
@@ -6,7 +6,6 @@ import { IAuthProvider } from "@/ee/security/types/security.types.ts";
|
|||||||
import { buildSsoLoginUrl } from "@/ee/security/sso.utils.ts";
|
import { buildSsoLoginUrl } from "@/ee/security/sso.utils.ts";
|
||||||
import { SSO_PROVIDER } from "@/ee/security/contants.ts";
|
import { SSO_PROVIDER } from "@/ee/security/contants.ts";
|
||||||
import { GoogleIcon } from "@/components/icons/google-icon.tsx";
|
import { GoogleIcon } from "@/components/icons/google-icon.tsx";
|
||||||
import { isCloud } from "@/lib/config.ts";
|
|
||||||
import { LdapLoginModal } from "@/ee/components/ldap-login-modal.tsx";
|
import { LdapLoginModal } from "@/ee/components/ldap-login-modal.tsx";
|
||||||
|
|
||||||
export default function SsoLogin() {
|
export default function SsoLogin() {
|
||||||
@@ -57,7 +56,7 @@ export default function SsoLogin() {
|
|||||||
/>
|
/>
|
||||||
)}
|
)}
|
||||||
|
|
||||||
{(isCloud() || data.hasLicenseKey) && (
|
{data.authProviders.length > 0 && (
|
||||||
<>
|
<>
|
||||||
<Stack align="stretch" justify="center" gap="sm">
|
<Stack align="stretch" justify="center" gap="sm">
|
||||||
{data.authProviders.map((provider) => (
|
{data.authProviders.map((provider) => (
|
||||||
|
|||||||
@@ -0,0 +1,7 @@
|
|||||||
|
import { atomWithStorage } from "jotai/utils";
|
||||||
|
import type { Entitlements } from "./entitlement.types";
|
||||||
|
|
||||||
|
export const entitlementAtom = atomWithStorage<Entitlements | null>(
|
||||||
|
"entitlements",
|
||||||
|
null,
|
||||||
|
);
|
||||||
@@ -0,0 +1,7 @@
|
|||||||
|
import api from "@/lib/api-client";
|
||||||
|
import { Entitlements } from "./entitlement.types";
|
||||||
|
|
||||||
|
export async function getEntitlements(): Promise<Entitlements> {
|
||||||
|
const req = await api.post<Entitlements>("/workspace/entitlements");
|
||||||
|
return req.data as Entitlements;
|
||||||
|
}
|
||||||
@@ -0,0 +1,7 @@
|
|||||||
|
export type Tier = "free" | "standard" | "business" | "enterprise";
|
||||||
|
|
||||||
|
export type Entitlements = {
|
||||||
|
cloud: boolean;
|
||||||
|
tier: Tier;
|
||||||
|
features: string[];
|
||||||
|
};
|
||||||
@@ -0,0 +1,11 @@
|
|||||||
|
import { useQuery, UseQueryResult } from "@tanstack/react-query";
|
||||||
|
import { getEntitlements } from "./entitlement-service";
|
||||||
|
import { Entitlements } from "./entitlement.types";
|
||||||
|
|
||||||
|
export function useEntitlements(): UseQueryResult<Entitlements> {
|
||||||
|
return useQuery({
|
||||||
|
queryKey: ["entitlements"],
|
||||||
|
queryFn: getEntitlements,
|
||||||
|
staleTime: 5 * 60 * 1000,
|
||||||
|
});
|
||||||
|
}
|
||||||
@@ -0,0 +1,19 @@
|
|||||||
|
export const Feature = {
|
||||||
|
SSO_CUSTOM: 'sso:custom',
|
||||||
|
SSO_GOOGLE: 'sso:google',
|
||||||
|
MFA: 'mfa',
|
||||||
|
API_KEYS: 'api:keys',
|
||||||
|
COMMENT_RESOLUTION: 'comment:resolution',
|
||||||
|
PAGE_PERMISSIONS: 'page:permissions',
|
||||||
|
AI: 'ai',
|
||||||
|
CONFLUENCE_IMPORT: 'import:confluence',
|
||||||
|
DOCX_IMPORT: 'import:docx',
|
||||||
|
ATTACHMENT_INDEXING: 'attachment:indexing',
|
||||||
|
SECURITY_SETTINGS: 'security:settings',
|
||||||
|
MCP: 'mcp',
|
||||||
|
SCIM: 'scim',
|
||||||
|
PAGE_VERIFICATION: 'page:verification',
|
||||||
|
AUDIT_LOGS: 'audit:logs',
|
||||||
|
RETENTION: 'retention',
|
||||||
|
SHARING_CONTROLS: 'sharing:controls',
|
||||||
|
} as const;
|
||||||
@@ -1,12 +0,0 @@
|
|||||||
import { isCloud } from "@/lib/config";
|
|
||||||
import useLicense from "@/ee/hooks/use-license";
|
|
||||||
import usePlan from "@/ee/hooks/use-plan";
|
|
||||||
|
|
||||||
const useEnterpriseAccess = () => {
|
|
||||||
const { hasLicenseKey } = useLicense();
|
|
||||||
const { isBusiness } = usePlan();
|
|
||||||
|
|
||||||
return (isCloud() && isBusiness) || (!isCloud() && hasLicenseKey);
|
|
||||||
};
|
|
||||||
|
|
||||||
export default useEnterpriseAccess;
|
|
||||||
@@ -0,0 +1,7 @@
|
|||||||
|
import { useAtom } from "jotai";
|
||||||
|
import { entitlementAtom } from "@/ee/entitlement/entitlement-atom";
|
||||||
|
|
||||||
|
export const useHasFeature = (feature: string): boolean => {
|
||||||
|
const [entitlements] = useAtom(entitlementAtom);
|
||||||
|
return entitlements?.features?.includes(feature) ?? false;
|
||||||
|
};
|
||||||
@@ -1,9 +0,0 @@
|
|||||||
import { useAtom } from "jotai";
|
|
||||||
import { currentUserAtom } from "@/features/user/atoms/current-user-atom.ts";
|
|
||||||
|
|
||||||
export const useLicense = () => {
|
|
||||||
const [currentUser] = useAtom(currentUserAtom);
|
|
||||||
return { hasLicenseKey: currentUser?.workspace?.hasLicenseKey };
|
|
||||||
};
|
|
||||||
|
|
||||||
export default useLicense;
|
|
||||||
@@ -0,0 +1,16 @@
|
|||||||
|
import { useAtom } from "jotai";
|
||||||
|
import { useTranslation } from "react-i18next";
|
||||||
|
import { entitlementAtom } from "@/ee/entitlement/entitlement-atom";
|
||||||
|
import { isCloud } from "@/lib/config";
|
||||||
|
|
||||||
|
export function useUpgradeLabel(): string {
|
||||||
|
const { t } = useTranslation();
|
||||||
|
const [entitlements] = useAtom(entitlementAtom);
|
||||||
|
|
||||||
|
if (!isCloud()) {
|
||||||
|
return entitlements != null && entitlements.tier !== "free"
|
||||||
|
? t("Upgrade your license tier.")
|
||||||
|
: t("Available with a paid license");
|
||||||
|
}
|
||||||
|
return t("Upgrade your plan");
|
||||||
|
}
|
||||||
@@ -7,21 +7,22 @@ import { useTranslation } from "react-i18next";
|
|||||||
import { useActivateMutation } from "@/ee/licence/queries/license-query.ts";
|
import { useActivateMutation } from "@/ee/licence/queries/license-query.ts";
|
||||||
import { useDisclosure } from "@mantine/hooks";
|
import { useDisclosure } from "@mantine/hooks";
|
||||||
import { useAtom } from "jotai";
|
import { useAtom } from "jotai";
|
||||||
import { workspaceAtom } from "@/features/user/atoms/current-user-atom.ts";
|
import { entitlementAtom } from "@/ee/entitlement/entitlement-atom";
|
||||||
import RemoveLicense from "@/ee/licence/components/remove-license.tsx";
|
import RemoveLicense from "@/ee/licence/components/remove-license.tsx";
|
||||||
|
|
||||||
export default function ActivateLicense() {
|
export default function ActivateLicense() {
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
const [opened, { open, close }] = useDisclosure(false);
|
const [opened, { open, close }] = useDisclosure(false);
|
||||||
const [workspace] = useAtom(workspaceAtom);
|
const [entitlements] = useAtom(entitlementAtom);
|
||||||
|
const hasLicense = entitlements != null && entitlements.tier !== "free";
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Group justify="flex-end" wrap="nowrap" mb="sm">
|
<Group justify="flex-end" wrap="nowrap" mb="sm">
|
||||||
<Button onClick={open}>
|
<Button onClick={open}>
|
||||||
{workspace?.hasLicenseKey ? t("Update license") : t("Add license")}
|
{hasLicense ? t("Update license") : t("Add license")}
|
||||||
</Button>
|
</Button>
|
||||||
|
|
||||||
{workspace?.hasLicenseKey && <RemoveLicense />}
|
{hasLicense && <RemoveLicense />}
|
||||||
|
|
||||||
<Modal
|
<Modal
|
||||||
size="550"
|
size="550"
|
||||||
@@ -59,7 +60,7 @@ export function ActivateLicenseForm({ onClose }: ActivateLicenseFormProps) {
|
|||||||
async function handleSubmit(data: { licenseKey: string }) {
|
async function handleSubmit(data: { licenseKey: string }) {
|
||||||
await activateLicenseMutation.mutateAsync(data.licenseKey);
|
await activateLicenseMutation.mutateAsync(data.licenseKey);
|
||||||
form.reset();
|
form.reset();
|
||||||
onClose();
|
onClose?.();
|
||||||
}
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
|
|||||||
@@ -31,7 +31,8 @@ export default function LicenseDetails() {
|
|||||||
<Table.Tr>
|
<Table.Tr>
|
||||||
<Table.Th w={160}>Edition</Table.Th>
|
<Table.Th w={160}>Edition</Table.Th>
|
||||||
<Table.Td>
|
<Table.Td>
|
||||||
Enterprise {license.trial && <Badge color="green">Trial</Badge>}
|
{license.licenseType === "business" ? "Business" : "Enterprise"}{" "}
|
||||||
|
{license.trial && <Badge color="green">Trial</Badge>}
|
||||||
</Table.Td>
|
</Table.Td>
|
||||||
</Table.Tr>
|
</Table.Tr>
|
||||||
|
|
||||||
|
|||||||
@@ -8,10 +8,11 @@ import ActivateLicenseForm from "@/ee/licence/components/activate-license-modal.
|
|||||||
import InstallationDetails from "@/ee/licence/components/installation-details.tsx";
|
import InstallationDetails from "@/ee/licence/components/installation-details.tsx";
|
||||||
import OssDetails from "@/ee/licence/components/oss-details.tsx";
|
import OssDetails from "@/ee/licence/components/oss-details.tsx";
|
||||||
import { useAtom } from "jotai/index";
|
import { useAtom } from "jotai/index";
|
||||||
import { workspaceAtom } from "@/features/user/atoms/current-user-atom.ts";
|
import { entitlementAtom } from "@/ee/entitlement/entitlement-atom";
|
||||||
|
|
||||||
export default function License() {
|
export default function License() {
|
||||||
const [workspace] = useAtom(workspaceAtom);
|
const [entitlements] = useAtom(entitlementAtom);
|
||||||
|
const hasLicense = entitlements != null && entitlements.tier !== "free";
|
||||||
const { isAdmin } = useUserRole();
|
const { isAdmin } = useUserRole();
|
||||||
|
|
||||||
if (!isAdmin) {
|
if (!isAdmin) {
|
||||||
@@ -29,7 +30,7 @@ export default function License() {
|
|||||||
|
|
||||||
<InstallationDetails />
|
<InstallationDetails />
|
||||||
|
|
||||||
{workspace?.hasLicenseKey ? <LicenseDetails /> : <OssDetails />}
|
{hasLicense ? <LicenseDetails /> : <OssDetails />}
|
||||||
</>
|
</>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -31,6 +31,7 @@ export function useActivateMutation() {
|
|||||||
queryKey: ["license"],
|
queryKey: ["license"],
|
||||||
});
|
});
|
||||||
queryClient.refetchQueries({ queryKey: ["currentUser"] });
|
queryClient.refetchQueries({ queryKey: ["currentUser"] });
|
||||||
|
queryClient.refetchQueries({ queryKey: ["entitlements"] });
|
||||||
},
|
},
|
||||||
onError: (error) => {
|
onError: (error) => {
|
||||||
const errorMessage = error["response"]?.data?.message;
|
const errorMessage = error["response"]?.data?.message;
|
||||||
@@ -47,6 +48,7 @@ export function useRemoveLicenseMutation() {
|
|||||||
onSuccess: () => {
|
onSuccess: () => {
|
||||||
queryClient.refetchQueries({ queryKey: ["license"] });
|
queryClient.refetchQueries({ queryKey: ["license"] });
|
||||||
queryClient.refetchQueries({ queryKey: ["currentUser"] });
|
queryClient.refetchQueries({ queryKey: ["currentUser"] });
|
||||||
|
queryClient.refetchQueries({ queryKey: ["entitlements"] });
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,7 +1,10 @@
|
|||||||
|
export type LicenseType = 'business' | 'enterprise';
|
||||||
|
|
||||||
export interface ILicenseInfo {
|
export interface ILicenseInfo {
|
||||||
id: string;
|
id: string;
|
||||||
customerName: string;
|
customerName: string;
|
||||||
seatCount: number;
|
seatCount: number;
|
||||||
|
licenseType: LicenseType;
|
||||||
issuedAt: Date;
|
issuedAt: Date;
|
||||||
expiresAt: Date;
|
expiresAt: Date;
|
||||||
trial: boolean;
|
trial: boolean;
|
||||||
|
|||||||
@@ -7,8 +7,9 @@ import { getMfaStatus } from "@/ee/mfa";
|
|||||||
import { MfaSetupModal } from "@/ee/mfa";
|
import { MfaSetupModal } from "@/ee/mfa";
|
||||||
import { MfaDisableModal } from "@/ee/mfa";
|
import { MfaDisableModal } from "@/ee/mfa";
|
||||||
import { MfaBackupCodesModal } from "@/ee/mfa";
|
import { MfaBackupCodesModal } from "@/ee/mfa";
|
||||||
import { isCloud } from "@/lib/config.ts";
|
import { useHasFeature } from "@/ee/hooks/use-feature";
|
||||||
import useLicense from "@/ee/hooks/use-license.tsx";
|
import { Feature } from "@/ee/features";
|
||||||
|
import { useUpgradeLabel } from "@/ee/hooks/use-upgrade-label";
|
||||||
import { ResponsiveSettingsRow, ResponsiveSettingsContent, ResponsiveSettingsControl } from "@/components/ui/responsive-settings-row";
|
import { ResponsiveSettingsRow, ResponsiveSettingsContent, ResponsiveSettingsControl } from "@/components/ui/responsive-settings-row";
|
||||||
|
|
||||||
export function MfaSettings() {
|
export function MfaSettings() {
|
||||||
@@ -17,7 +18,8 @@ export function MfaSettings() {
|
|||||||
const [setupModalOpen, setSetupModalOpen] = useState(false);
|
const [setupModalOpen, setSetupModalOpen] = useState(false);
|
||||||
const [disableModalOpen, setDisableModalOpen] = useState(false);
|
const [disableModalOpen, setDisableModalOpen] = useState(false);
|
||||||
const [backupCodesModalOpen, setBackupCodesModalOpen] = useState(false);
|
const [backupCodesModalOpen, setBackupCodesModalOpen] = useState(false);
|
||||||
const { hasLicenseKey } = useLicense();
|
const canUseMfa = useHasFeature(Feature.MFA);
|
||||||
|
const upgradeLabel = useUpgradeLabel();
|
||||||
|
|
||||||
const { data: mfaStatus, isLoading } = useQuery({
|
const { data: mfaStatus, isLoading } = useQuery({
|
||||||
queryKey: ["mfa-status"],
|
queryKey: ["mfa-status"],
|
||||||
@@ -28,8 +30,6 @@ export function MfaSettings() {
|
|||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
const canUseMfa = isCloud() || hasLicenseKey;
|
|
||||||
|
|
||||||
// Check if MFA is truly enabled
|
// Check if MFA is truly enabled
|
||||||
const isMfaEnabled = mfaStatus?.isEnabled === true;
|
const isMfaEnabled = mfaStatus?.isEnabled === true;
|
||||||
|
|
||||||
@@ -69,7 +69,7 @@ export function MfaSettings() {
|
|||||||
<ResponsiveSettingsControl>
|
<ResponsiveSettingsControl>
|
||||||
{!isMfaEnabled ? (
|
{!isMfaEnabled ? (
|
||||||
<Tooltip
|
<Tooltip
|
||||||
label={t("Available in enterprise edition")}
|
label={upgradeLabel}
|
||||||
disabled={canUseMfa}
|
disabled={canUseMfa}
|
||||||
>
|
>
|
||||||
<Button
|
<Button
|
||||||
|
|||||||
@@ -19,7 +19,8 @@ import { usePageRestrictionInfoQuery } from "@/ee/page-permission/queries/page-p
|
|||||||
import { PagePermissionTab } from "@/ee/page-permission";
|
import { PagePermissionTab } from "@/ee/page-permission";
|
||||||
import { PublishTab } from "./publish-tab";
|
import { PublishTab } from "./publish-tab";
|
||||||
import { useShareForPageQuery } from "@/features/share/queries/share-query";
|
import { useShareForPageQuery } from "@/features/share/queries/share-query";
|
||||||
import { useIsCloudEE } from "@/hooks/use-is-cloud-ee";
|
import { useHasFeature } from "@/ee/hooks/use-feature";
|
||||||
|
import { Feature } from "@/ee/features";
|
||||||
import { useAtom } from "jotai";
|
import { useAtom } from "jotai";
|
||||||
import { workspaceAtom } from "@/features/user/atoms/current-user-atom";
|
import { workspaceAtom } from "@/features/user/atoms/current-user-atom";
|
||||||
import { useSpaceQuery } from "@/features/space/queries/space-query";
|
import { useSpaceQuery } from "@/features/space/queries/space-query";
|
||||||
@@ -33,9 +34,9 @@ export function PageShareModal({ readOnly }: PageShareModalProps) {
|
|||||||
const { pageSlug, spaceSlug } = useParams();
|
const { pageSlug, spaceSlug } = useParams();
|
||||||
const pageSlugId = extractPageSlugId(pageSlug);
|
const pageSlugId = extractPageSlugId(pageSlug);
|
||||||
const [opened, { open, close }] = useDisclosure(false);
|
const [opened, { open, close }] = useDisclosure(false);
|
||||||
const isCloudEE = useIsCloudEE();
|
const hasPagePermissions = useHasFeature(Feature.PAGE_PERMISSIONS);
|
||||||
const [activeTab, setActiveTab] = useState<string | null>(
|
const [activeTab, setActiveTab] = useState<string | null>(
|
||||||
isCloudEE ? "access" : "publish",
|
hasPagePermissions ? "access" : "publish",
|
||||||
);
|
);
|
||||||
|
|
||||||
const [workspace] = useAtom(workspaceAtom);
|
const [workspace] = useAtom(workspaceAtom);
|
||||||
@@ -51,7 +52,7 @@ export function PageShareModal({ readOnly }: PageShareModalProps) {
|
|||||||
const isPubliclyShared = !!share;
|
const isPubliclyShared = !!share;
|
||||||
|
|
||||||
const { data: restrictionInfo, isLoading: restrictionLoading } =
|
const { data: restrictionInfo, isLoading: restrictionLoading } =
|
||||||
usePageRestrictionInfoQuery(opened && isCloudEE ? pageId : undefined);
|
usePageRestrictionInfoQuery(opened && hasPagePermissions ? pageId : undefined);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
@@ -92,7 +93,7 @@ export function PageShareModal({ readOnly }: PageShareModalProps) {
|
|||||||
</Tabs.List>
|
</Tabs.List>
|
||||||
|
|
||||||
<Tabs.Panel value="access">
|
<Tabs.Panel value="access">
|
||||||
{!isCloudEE ? (
|
{!hasPagePermissions ? (
|
||||||
<Stack align="center" py="md">
|
<Stack align="center" py="md">
|
||||||
<IconLock size={20} stroke={1.5} />
|
<IconLock size={20} stroke={1.5} />
|
||||||
<Text size="sm" ta="center" fw={500}>
|
<Text size="sm" ta="center" fw={500}>
|
||||||
|
|||||||
@@ -6,21 +6,23 @@ import React, { useState } from "react";
|
|||||||
import { useTranslation } from "react-i18next";
|
import { useTranslation } from "react-i18next";
|
||||||
import { updateWorkspace } from "@/features/workspace/services/workspace-service.ts";
|
import { updateWorkspace } from "@/features/workspace/services/workspace-service.ts";
|
||||||
import { notifications } from "@mantine/notifications";
|
import { notifications } from "@mantine/notifications";
|
||||||
import useEnterpriseAccess from "@/ee/hooks/use-enterprise-access.tsx";
|
import { useHasFeature } from "@/ee/hooks/use-feature";
|
||||||
|
import { Feature } from "@/ee/features";
|
||||||
|
import { useUpgradeLabel } from "@/ee/hooks/use-upgrade-label.ts";
|
||||||
|
|
||||||
export default function DisablePublicSharing() {
|
export default function DisablePublicSharing() {
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Group justify="space-between" wrap="nowrap" gap="xl">
|
<Group justify="space-between" wrap="nowrap" gap="xl">
|
||||||
<div>
|
<div>
|
||||||
<Text size="md">{t("Disable public sharing")}</Text>
|
<Text size="md">{t("Disable public sharing")}</Text>
|
||||||
<Text size="sm" c="dimmed">
|
<Text size="sm" c="dimmed">
|
||||||
{t("Prevent members from sharing pages publicly.")}
|
{t("Prevent members from sharing pages publicly.")}
|
||||||
</Text>
|
</Text>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<DisablePublicSharingToggle />
|
<DisablePublicSharingToggle />
|
||||||
</Group>
|
</Group>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
@@ -31,7 +33,8 @@ function DisablePublicSharingToggle() {
|
|||||||
const [checked, setChecked] = useState(
|
const [checked, setChecked] = useState(
|
||||||
workspace?.settings?.sharing?.disabled === true,
|
workspace?.settings?.sharing?.disabled === true,
|
||||||
);
|
);
|
||||||
const hasAccess = useEnterpriseAccess();
|
const hasSharingControls = useHasFeature(Feature.SHARING_CONTROLS);
|
||||||
|
const upgradeLabel = useUpgradeLabel();
|
||||||
|
|
||||||
const applyChange = async (value: boolean) => {
|
const applyChange = async (value: boolean) => {
|
||||||
try {
|
try {
|
||||||
@@ -72,15 +75,11 @@ function DisablePublicSharingToggle() {
|
|||||||
};
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Tooltip
|
<Tooltip label={upgradeLabel} disabled={hasSharingControls} refProp="rootRef">
|
||||||
label={t("Requires an enterprise license")}
|
|
||||||
disabled={hasAccess}
|
|
||||||
refProp="rootRef"
|
|
||||||
>
|
|
||||||
<Switch
|
<Switch
|
||||||
checked={checked}
|
checked={checked}
|
||||||
onChange={handleChange}
|
onChange={handleChange}
|
||||||
disabled={!hasAccess}
|
disabled={!hasSharingControls}
|
||||||
aria-label={t("Toggle public sharing")}
|
aria-label={t("Toggle public sharing")}
|
||||||
/>
|
/>
|
||||||
</Tooltip>
|
</Tooltip>
|
||||||
|
|||||||
@@ -1,10 +1,20 @@
|
|||||||
import { Group, Text, Switch, MantineSize, Title } from "@mantine/core";
|
import {
|
||||||
|
Group,
|
||||||
|
Text,
|
||||||
|
Switch,
|
||||||
|
MantineSize,
|
||||||
|
Title,
|
||||||
|
Tooltip,
|
||||||
|
} from "@mantine/core";
|
||||||
import { useAtom } from "jotai";
|
import { useAtom } from "jotai";
|
||||||
import { workspaceAtom } from "@/features/user/atoms/current-user-atom.ts";
|
import { workspaceAtom } from "@/features/user/atoms/current-user-atom.ts";
|
||||||
import React, { useState } from "react";
|
import React, { useState } from "react";
|
||||||
import { useTranslation } from "react-i18next";
|
import { useTranslation } from "react-i18next";
|
||||||
import { updateWorkspace } from "@/features/workspace/services/workspace-service.ts";
|
import { updateWorkspace } from "@/features/workspace/services/workspace-service.ts";
|
||||||
import { notifications } from "@mantine/notifications";
|
import { notifications } from "@mantine/notifications";
|
||||||
|
import { useHasFeature } from "@/ee/hooks/use-feature.ts";
|
||||||
|
import { Feature } from "@/ee/features.ts";
|
||||||
|
import { useUpgradeLabel } from "@/ee/hooks/use-upgrade-label.ts";
|
||||||
|
|
||||||
export default function EnforceMfa() {
|
export default function EnforceMfa() {
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
@@ -33,6 +43,8 @@ export function EnforceMfaToggle({ size, label }: EnforceMfaToggleProps) {
|
|||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
const [workspace, setWorkspace] = useAtom(workspaceAtom);
|
const [workspace, setWorkspace] = useAtom(workspaceAtom);
|
||||||
const [checked, setChecked] = useState(workspace?.enforceMfa);
|
const [checked, setChecked] = useState(workspace?.enforceMfa);
|
||||||
|
const hasAccess = useHasFeature(Feature.MFA);
|
||||||
|
const upgradeLabel = useUpgradeLabel();
|
||||||
|
|
||||||
const handleChange = async (event: React.ChangeEvent<HTMLInputElement>) => {
|
const handleChange = async (event: React.ChangeEvent<HTMLInputElement>) => {
|
||||||
const value = event.currentTarget.checked;
|
const value = event.currentTarget.checked;
|
||||||
@@ -49,13 +61,16 @@ export function EnforceMfaToggle({ size, label }: EnforceMfaToggleProps) {
|
|||||||
};
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Switch
|
<Tooltip label={upgradeLabel} disabled={hasAccess} refProp="rootRef">
|
||||||
size={size}
|
<Switch
|
||||||
label={label}
|
size={size}
|
||||||
labelPosition="left"
|
label={label}
|
||||||
defaultChecked={checked}
|
labelPosition="left"
|
||||||
onChange={handleChange}
|
defaultChecked={checked}
|
||||||
aria-label={t("Toggle MFA enforcement")}
|
onChange={handleChange}
|
||||||
/>
|
disabled={!hasAccess}
|
||||||
|
aria-label={t("Toggle MFA enforcement")}
|
||||||
|
/>
|
||||||
|
</Tooltip>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,10 +1,13 @@
|
|||||||
import { Group, Text, Switch, MantineSize } from "@mantine/core";
|
import { Group, Text, Switch, MantineSize, Tooltip } from "@mantine/core";
|
||||||
import { useAtom } from "jotai";
|
import { useAtom } from "jotai";
|
||||||
import { workspaceAtom } from "@/features/user/atoms/current-user-atom.ts";
|
import { workspaceAtom } from "@/features/user/atoms/current-user-atom.ts";
|
||||||
import React, { useState } from "react";
|
import React, { useState } from "react";
|
||||||
import { useTranslation } from "react-i18next";
|
import { useTranslation } from "react-i18next";
|
||||||
import { updateWorkspace } from "@/features/workspace/services/workspace-service.ts";
|
import { updateWorkspace } from "@/features/workspace/services/workspace-service.ts";
|
||||||
import { notifications } from "@mantine/notifications";
|
import { notifications } from "@mantine/notifications";
|
||||||
|
import { useHasFeature } from "@/ee/hooks/use-feature.ts";
|
||||||
|
import { Feature } from "@/ee/features.ts";
|
||||||
|
import { useUpgradeLabel } from "@/ee/hooks/use-upgrade-label.ts";
|
||||||
|
|
||||||
export default function EnforceSso() {
|
export default function EnforceSso() {
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
@@ -33,6 +36,8 @@ export function EnforceSsoToggle({ size, label }: EnforceSsoToggleProps) {
|
|||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
const [workspace, setWorkspace] = useAtom(workspaceAtom);
|
const [workspace, setWorkspace] = useAtom(workspaceAtom);
|
||||||
const [checked, setChecked] = useState(workspace?.enforceSso);
|
const [checked, setChecked] = useState(workspace?.enforceSso);
|
||||||
|
const hasAccess = useHasFeature(Feature.SSO_CUSTOM);
|
||||||
|
const upgradeLabel = useUpgradeLabel();
|
||||||
|
|
||||||
const handleChange = async (event: React.ChangeEvent<HTMLInputElement>) => {
|
const handleChange = async (event: React.ChangeEvent<HTMLInputElement>) => {
|
||||||
const value = event.currentTarget.checked;
|
const value = event.currentTarget.checked;
|
||||||
@@ -49,13 +54,16 @@ export function EnforceSsoToggle({ size, label }: EnforceSsoToggleProps) {
|
|||||||
};
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Switch
|
<Tooltip label={upgradeLabel} disabled={hasAccess} refProp="rootRef">
|
||||||
size={size}
|
<Switch
|
||||||
label={label}
|
size={size}
|
||||||
labelPosition="left"
|
label={label}
|
||||||
defaultChecked={checked}
|
labelPosition="left"
|
||||||
onChange={handleChange}
|
defaultChecked={checked}
|
||||||
aria-label={t("Toggle sso enforcement")}
|
onChange={handleChange}
|
||||||
/>
|
disabled={!hasAccess}
|
||||||
|
aria-label={t("Toggle sso enforcement")}
|
||||||
|
/>
|
||||||
|
</Tooltip>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -6,6 +6,9 @@ import React, { useState } from "react";
|
|||||||
import { useTranslation } from "react-i18next";
|
import { useTranslation } from "react-i18next";
|
||||||
import { ISpace } from "@/features/space/types/space.types.ts";
|
import { ISpace } from "@/features/space/types/space.types.ts";
|
||||||
import { useUpdateSpaceMutation } from "@/features/space/queries/space-query.ts";
|
import { useUpdateSpaceMutation } from "@/features/space/queries/space-query.ts";
|
||||||
|
import { useHasFeature } from "@/ee/hooks/use-feature.ts";
|
||||||
|
import { Feature } from "@/ee/features.ts";
|
||||||
|
import { useUpgradeLabel } from "@/ee/hooks/use-upgrade-label.ts";
|
||||||
|
|
||||||
type SpacePublicSharingToggleProps = {
|
type SpacePublicSharingToggleProps = {
|
||||||
space: ISpace;
|
space: ISpace;
|
||||||
@@ -17,6 +20,9 @@ export default function SpacePublicSharingToggle({
|
|||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
const [workspace] = useAtom(workspaceAtom);
|
const [workspace] = useAtom(workspaceAtom);
|
||||||
const workspaceDisabled = workspace?.settings?.sharing?.disabled === true;
|
const workspaceDisabled = workspace?.settings?.sharing?.disabled === true;
|
||||||
|
const hasSharingControls = useHasFeature(Feature.SHARING_CONTROLS);
|
||||||
|
const upgradeLabel = useUpgradeLabel();
|
||||||
|
const isDisabled = !hasSharingControls || workspaceDisabled;
|
||||||
const [checked, setChecked] = useState(
|
const [checked, setChecked] = useState(
|
||||||
space.settings?.sharing?.disabled === true,
|
space.settings?.sharing?.disabled === true,
|
||||||
);
|
);
|
||||||
@@ -68,14 +74,14 @@ export default function SpacePublicSharingToggle({
|
|||||||
</Text>
|
</Text>
|
||||||
</div>
|
</div>
|
||||||
<Tooltip
|
<Tooltip
|
||||||
label={t("Public sharing is disabled at the workspace level")}
|
label={!hasSharingControls ? upgradeLabel : t("Public sharing is disabled at the workspace level")}
|
||||||
disabled={!workspaceDisabled}
|
disabled={!isDisabled}
|
||||||
refProp="rootRef"
|
refProp="rootRef"
|
||||||
>
|
>
|
||||||
<Switch
|
<Switch
|
||||||
checked={checked}
|
checked={checked}
|
||||||
onChange={handleChange}
|
onChange={handleChange}
|
||||||
disabled={workspaceDisabled}
|
disabled={isDisabled}
|
||||||
aria-label={t("Toggle space public sharing")}
|
aria-label={t("Toggle space public sharing")}
|
||||||
/>
|
/>
|
||||||
</Tooltip>
|
</Tooltip>
|
||||||
|
|||||||
@@ -12,13 +12,18 @@ import { workspaceAtom } from "@/features/user/atoms/current-user-atom.ts";
|
|||||||
import { useTranslation } from "react-i18next";
|
import { useTranslation } from "react-i18next";
|
||||||
import { updateWorkspace } from "@/features/workspace/services/workspace-service.ts";
|
import { updateWorkspace } from "@/features/workspace/services/workspace-service.ts";
|
||||||
import { notifications } from "@mantine/notifications";
|
import { notifications } from "@mantine/notifications";
|
||||||
import useEnterpriseAccess from "@/ee/hooks/use-enterprise-access.tsx";
|
import { useHasFeature } from "@/ee/hooks/use-feature";
|
||||||
|
import { Feature } from "@/ee/features";
|
||||||
|
import { useUpgradeLabel } from "@/ee/hooks/use-upgrade-label.ts";
|
||||||
|
|
||||||
type RetentionUnit = "days" | "months" | "years";
|
type RetentionUnit = "days" | "months" | "years";
|
||||||
|
|
||||||
const DEFAULT_RETENTION_DAYS = 30;
|
const DEFAULT_RETENTION_DAYS = 30;
|
||||||
|
|
||||||
function daysToRetention(days: number): { amount: number; unit: RetentionUnit } {
|
function daysToRetention(days: number): {
|
||||||
|
amount: number;
|
||||||
|
unit: RetentionUnit;
|
||||||
|
} {
|
||||||
if (days >= 365 && days % 365 === 0) {
|
if (days >= 365 && days % 365 === 0) {
|
||||||
return { amount: days / 365, unit: "years" };
|
return { amount: days / 365, unit: "years" };
|
||||||
}
|
}
|
||||||
@@ -36,14 +41,19 @@ function retentionToDays(amount: number, unit: RetentionUnit): number {
|
|||||||
|
|
||||||
export default function TrashRetention() {
|
export default function TrashRetention() {
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
const hasAccess = useEnterpriseAccess();
|
const hasRetention = useHasFeature(Feature.RETENTION);
|
||||||
|
const upgradeLabel = useUpgradeLabel();
|
||||||
const [workspace, setWorkspace] = useAtom(workspaceAtom);
|
const [workspace, setWorkspace] = useAtom(workspaceAtom);
|
||||||
|
|
||||||
const currentDays = workspace?.trashRetentionDays ?? DEFAULT_RETENTION_DAYS;
|
const currentDays = workspace?.trashRetentionDays ?? DEFAULT_RETENTION_DAYS;
|
||||||
const parsed = daysToRetention(currentDays);
|
const parsed = daysToRetention(currentDays);
|
||||||
|
|
||||||
const [retentionAmount, setRetentionAmount] = useState<number | string>(parsed.amount);
|
const [retentionAmount, setRetentionAmount] = useState<number | string>(
|
||||||
const [retentionUnit, setRetentionUnit] = useState<RetentionUnit>(parsed.unit);
|
parsed.amount,
|
||||||
|
);
|
||||||
|
const [retentionUnit, setRetentionUnit] = useState<RetentionUnit>(
|
||||||
|
parsed.unit,
|
||||||
|
);
|
||||||
const [saving, setSaving] = useState(false);
|
const [saving, setSaving] = useState(false);
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
@@ -63,14 +73,17 @@ export default function TrashRetention() {
|
|||||||
|
|
||||||
setSaving(true);
|
setSaving(true);
|
||||||
try {
|
try {
|
||||||
const updatedWorkspace = await updateWorkspace({ trashRetentionDays: days });
|
const updatedWorkspace = await updateWorkspace({
|
||||||
|
trashRetentionDays: days,
|
||||||
|
});
|
||||||
setWorkspace(updatedWorkspace);
|
setWorkspace(updatedWorkspace);
|
||||||
notifications.show({
|
notifications.show({
|
||||||
message: t("Trash retention updated"),
|
message: t("Trash retention updated"),
|
||||||
});
|
});
|
||||||
} catch (err: any) {
|
} catch (err: any) {
|
||||||
notifications.show({
|
notifications.show({
|
||||||
message: err?.response?.data?.message || t("Failed to update trash retention"),
|
message:
|
||||||
|
err?.response?.data?.message || t("Failed to update trash retention"),
|
||||||
color: "red",
|
color: "red",
|
||||||
});
|
});
|
||||||
const { amount, unit } = daysToRetention(currentDays);
|
const { amount, unit } = daysToRetention(currentDays);
|
||||||
@@ -81,10 +94,11 @@ export default function TrashRetention() {
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
const isDirty = retentionToDays(
|
const isDirty =
|
||||||
typeof retentionAmount === "number" ? retentionAmount : 1,
|
retentionToDays(
|
||||||
retentionUnit,
|
typeof retentionAmount === "number" ? retentionAmount : 1,
|
||||||
) !== currentDays;
|
retentionUnit,
|
||||||
|
) !== currentDays;
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div>
|
<div>
|
||||||
@@ -93,10 +107,7 @@ export default function TrashRetention() {
|
|||||||
{t("Pages in trash will be permanently deleted after this period.")}
|
{t("Pages in trash will be permanently deleted after this period.")}
|
||||||
</Text>
|
</Text>
|
||||||
|
|
||||||
<Tooltip
|
<Tooltip label={upgradeLabel} disabled={hasRetention}>
|
||||||
label={t("Requires an enterprise license")}
|
|
||||||
disabled={hasAccess}
|
|
||||||
>
|
|
||||||
<Group gap="xs" wrap="nowrap" maw={320}>
|
<Group gap="xs" wrap="nowrap" maw={320}>
|
||||||
<NumberInput
|
<NumberInput
|
||||||
value={retentionAmount}
|
value={retentionAmount}
|
||||||
@@ -105,7 +116,7 @@ export default function TrashRetention() {
|
|||||||
hideControls
|
hideControls
|
||||||
size="sm"
|
size="sm"
|
||||||
w={60}
|
w={60}
|
||||||
disabled={!hasAccess}
|
disabled={!hasRetention}
|
||||||
/>
|
/>
|
||||||
<Select
|
<Select
|
||||||
data={[
|
data={[
|
||||||
@@ -121,13 +132,13 @@ export default function TrashRetention() {
|
|||||||
}}
|
}}
|
||||||
size="sm"
|
size="sm"
|
||||||
style={{ flex: 1 }}
|
style={{ flex: 1 }}
|
||||||
disabled={!hasAccess}
|
disabled={!hasRetention}
|
||||||
/>
|
/>
|
||||||
<Button
|
<Button
|
||||||
size="sm"
|
size="sm"
|
||||||
onClick={handleSave}
|
onClick={handleSave}
|
||||||
loading={saving}
|
loading={saving}
|
||||||
disabled={!hasAccess || !isDirty}
|
disabled={!hasRetention || !isDirty}
|
||||||
>
|
>
|
||||||
{t("Save")}
|
{t("Save")}
|
||||||
</Button>
|
</Button>
|
||||||
|
|||||||
@@ -12,14 +12,15 @@ import { useTranslation } from "react-i18next";
|
|||||||
import EnforceMfa from "@/ee/security/components/enforce-mfa.tsx";
|
import EnforceMfa from "@/ee/security/components/enforce-mfa.tsx";
|
||||||
import DisablePublicSharing from "@/ee/security/components/disable-public-sharing.tsx";
|
import DisablePublicSharing from "@/ee/security/components/disable-public-sharing.tsx";
|
||||||
import TrashRetention from "@/ee/security/components/trash-retention.tsx";
|
import TrashRetention from "@/ee/security/components/trash-retention.tsx";
|
||||||
import useEnterpriseAccess from "@/ee/hooks/use-enterprise-access.tsx";
|
import { useHasFeature } from "@/ee/hooks/use-feature";
|
||||||
import { useIsCloudEE } from "@/hooks/use-is-cloud-ee.tsx";
|
import { Feature } from "@/ee/features";
|
||||||
|
|
||||||
export default function Security() {
|
export default function Security() {
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
const { isAdmin } = useUserRole();
|
const { isAdmin } = useUserRole();
|
||||||
const hasEnterpriseAccess = useEnterpriseAccess();
|
const hasCustomSso = useHasFeature(Feature.SSO_CUSTOM);
|
||||||
const isCloudEE = useIsCloudEE();
|
const hasRetention = useHasFeature(Feature.RETENTION);
|
||||||
|
const hasSharingControls = useHasFeature(Feature.SHARING_CONTROLS);
|
||||||
|
|
||||||
if (!isAdmin) {
|
if (!isAdmin) {
|
||||||
return null;
|
return null;
|
||||||
@@ -36,39 +37,27 @@ export default function Security() {
|
|||||||
|
|
||||||
<Divider my="lg" />
|
<Divider my="lg" />
|
||||||
|
|
||||||
{(!isCloud() || hasEnterpriseAccess) && (
|
<DisablePublicSharing />
|
||||||
<>
|
<Divider my="lg" />
|
||||||
<DisablePublicSharing />
|
|
||||||
<Divider my="lg" />
|
|
||||||
</>
|
|
||||||
)}
|
|
||||||
|
|
||||||
{!isCloud() && (
|
<TrashRetention />
|
||||||
<>
|
<Divider my="lg" />
|
||||||
<TrashRetention />
|
|
||||||
<Divider my="lg" />
|
|
||||||
</>
|
|
||||||
)}
|
|
||||||
|
|
||||||
<Title order={4} my="lg">
|
<Title order={4} my="lg">
|
||||||
Single sign-on (SSO)
|
Single sign-on (SSO)
|
||||||
</Title>
|
</Title>
|
||||||
|
|
||||||
{hasEnterpriseAccess && (
|
<EnforceSso />
|
||||||
<>
|
<Divider my="lg" />
|
||||||
<EnforceSso />
|
|
||||||
<Divider my="lg" />
|
|
||||||
</>
|
|
||||||
)}
|
|
||||||
|
|
||||||
{isCloudEE && (
|
{(isCloud() || hasCustomSso) && (
|
||||||
<>
|
<>
|
||||||
<AllowedDomains />
|
<AllowedDomains />
|
||||||
<Divider my="lg" />
|
<Divider my="lg" />
|
||||||
</>
|
</>
|
||||||
)}
|
)}
|
||||||
|
|
||||||
{hasEnterpriseAccess && (
|
{hasCustomSso && (
|
||||||
<>
|
<>
|
||||||
<CreateSsoProvider />
|
<CreateSsoProvider />
|
||||||
<Divider size={0} my="lg" />
|
<Divider size={0} my="lg" />
|
||||||
|
|||||||
@@ -7,7 +7,8 @@ import CommentEditor from "@/features/comment/components/comment-editor";
|
|||||||
import { pageEditorAtom } from "@/features/editor/atoms/editor-atoms";
|
import { pageEditorAtom } from "@/features/editor/atoms/editor-atoms";
|
||||||
import CommentActions from "@/features/comment/components/comment-actions";
|
import CommentActions from "@/features/comment/components/comment-actions";
|
||||||
import CommentMenu from "@/features/comment/components/comment-menu";
|
import CommentMenu from "@/features/comment/components/comment-menu";
|
||||||
import { useIsCloudEE } from "@/hooks/use-is-cloud-ee";
|
import { useHasFeature } from "@/ee/hooks/use-feature";
|
||||||
|
import { Feature } from "@/ee/features";
|
||||||
import ResolveComment from "@/ee/comment/components/resolve-comment";
|
import ResolveComment from "@/ee/comment/components/resolve-comment";
|
||||||
import { useHover } from "@mantine/hooks";
|
import { useHover } from "@mantine/hooks";
|
||||||
import {
|
import {
|
||||||
@@ -44,7 +45,7 @@ function CommentListItem({
|
|||||||
const deleteCommentMutation = useDeleteCommentMutation(comment.pageId);
|
const deleteCommentMutation = useDeleteCommentMutation(comment.pageId);
|
||||||
const resolveCommentMutation = useResolveCommentMutation();
|
const resolveCommentMutation = useResolveCommentMutation();
|
||||||
const [currentUser] = useAtom(currentUserAtom);
|
const [currentUser] = useAtom(currentUserAtom);
|
||||||
const isCloudEE = useIsCloudEE();
|
const canResolve = useHasFeature(Feature.COMMENT_RESOLUTION);
|
||||||
const createdAtAgo = useTimeAgo(comment.createdAt);
|
const createdAtAgo = useTimeAgo(comment.createdAt);
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
@@ -81,7 +82,7 @@ function CommentListItem({
|
|||||||
}
|
}
|
||||||
|
|
||||||
async function handleResolveComment() {
|
async function handleResolveComment() {
|
||||||
if (!isCloudEE) return;
|
if (!canResolve) return;
|
||||||
|
|
||||||
try {
|
try {
|
||||||
const isResolved = comment.resolvedAt != null;
|
const isResolved = comment.resolvedAt != null;
|
||||||
@@ -137,7 +138,7 @@ function CommentListItem({
|
|||||||
</Text>
|
</Text>
|
||||||
|
|
||||||
<div style={{ visibility: hovered ? "visible" : "hidden" }}>
|
<div style={{ visibility: hovered ? "visible" : "hidden" }}>
|
||||||
{!comment.parentCommentId && canComment && isCloudEE && (
|
{!comment.parentCommentId && canComment && canResolve && (
|
||||||
<ResolveComment
|
<ResolveComment
|
||||||
editor={editor}
|
editor={editor}
|
||||||
commentId={comment.id}
|
commentId={comment.id}
|
||||||
|
|||||||
@@ -1,8 +1,16 @@
|
|||||||
import { ActionIcon, Menu, Tooltip } from "@mantine/core";
|
import { ActionIcon, Menu, Tooltip } from "@mantine/core";
|
||||||
import { IconDots, IconEdit, IconTrash, IconCircleCheck, IconCircleCheckFilled } from "@tabler/icons-react";
|
import {
|
||||||
|
IconDots,
|
||||||
|
IconEdit,
|
||||||
|
IconTrash,
|
||||||
|
IconCircleCheck,
|
||||||
|
IconCircleCheckFilled,
|
||||||
|
} from "@tabler/icons-react";
|
||||||
import { modals } from "@mantine/modals";
|
import { modals } from "@mantine/modals";
|
||||||
import { useTranslation } from "react-i18next";
|
import { useTranslation } from "react-i18next";
|
||||||
import { useIsCloudEE } from "@/hooks/use-is-cloud-ee";
|
import { useHasFeature } from "@/ee/hooks/use-feature";
|
||||||
|
import { Feature } from "@/ee/features";
|
||||||
|
import { useUpgradeLabel } from "@/ee/hooks/use-upgrade-label";
|
||||||
|
|
||||||
type CommentMenuProps = {
|
type CommentMenuProps = {
|
||||||
onEditComment: () => void;
|
onEditComment: () => void;
|
||||||
@@ -13,16 +21,17 @@ type CommentMenuProps = {
|
|||||||
isParentComment?: boolean;
|
isParentComment?: boolean;
|
||||||
};
|
};
|
||||||
|
|
||||||
function CommentMenu({
|
function CommentMenu({
|
||||||
onEditComment,
|
onEditComment,
|
||||||
onDeleteComment,
|
onDeleteComment,
|
||||||
onResolveComment,
|
onResolveComment,
|
||||||
canEdit = true,
|
canEdit = true,
|
||||||
isResolved = false,
|
isResolved = false,
|
||||||
isParentComment = false
|
isParentComment = false,
|
||||||
}: CommentMenuProps) {
|
}: CommentMenuProps) {
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
const isCloudEE = useIsCloudEE();
|
const canResolve = useHasFeature(Feature.COMMENT_RESOLUTION);
|
||||||
|
const upgradeLabel = useUpgradeLabel();
|
||||||
|
|
||||||
//@ts-ignore
|
//@ts-ignore
|
||||||
const openDeleteModal = () =>
|
const openDeleteModal = () =>
|
||||||
@@ -44,33 +53,34 @@ function CommentMenu({
|
|||||||
|
|
||||||
<Menu.Dropdown>
|
<Menu.Dropdown>
|
||||||
{canEdit && (
|
{canEdit && (
|
||||||
<Menu.Item onClick={onEditComment} leftSection={<IconEdit size={14} />}>
|
<Menu.Item
|
||||||
|
onClick={onEditComment}
|
||||||
|
leftSection={<IconEdit size={14} />}
|
||||||
|
>
|
||||||
{t("Edit comment")}
|
{t("Edit comment")}
|
||||||
</Menu.Item>
|
</Menu.Item>
|
||||||
)}
|
)}
|
||||||
{isParentComment && (
|
{isParentComment &&
|
||||||
isCloudEE ? (
|
(canResolve ? (
|
||||||
<Menu.Item
|
<Menu.Item
|
||||||
onClick={onResolveComment}
|
onClick={onResolveComment}
|
||||||
leftSection={
|
leftSection={
|
||||||
isResolved ?
|
isResolved ? (
|
||||||
<IconCircleCheckFilled size={14} /> :
|
<IconCircleCheckFilled size={14} />
|
||||||
|
) : (
|
||||||
<IconCircleCheck size={14} />
|
<IconCircleCheck size={14} />
|
||||||
|
)
|
||||||
}
|
}
|
||||||
>
|
>
|
||||||
{isResolved ? t("Re-open comment") : t("Resolve comment")}
|
{isResolved ? t("Re-open comment") : t("Resolve comment")}
|
||||||
</Menu.Item>
|
</Menu.Item>
|
||||||
) : (
|
) : (
|
||||||
<Tooltip label={t("Available in enterprise edition")} position="left">
|
<Tooltip label={upgradeLabel} position="left">
|
||||||
<Menu.Item
|
<Menu.Item disabled leftSection={<IconCircleCheck size={14} />}>
|
||||||
disabled
|
|
||||||
leftSection={<IconCircleCheck size={14} />}
|
|
||||||
>
|
|
||||||
{t("Resolve comment")}
|
{t("Resolve comment")}
|
||||||
</Menu.Item>
|
</Menu.Item>
|
||||||
</Tooltip>
|
</Tooltip>
|
||||||
)
|
))}
|
||||||
)}
|
|
||||||
<Menu.Item
|
<Menu.Item
|
||||||
leftSection={<IconTrash size={14} />}
|
leftSection={<IconTrash size={14} />}
|
||||||
onClick={openDeleteModal}
|
onClick={openDeleteModal}
|
||||||
|
|||||||
@@ -10,3 +10,5 @@ export const readOnlyEditorAtom = atom<Editor | null>(null);
|
|||||||
export const yjsConnectionStatusAtom = atom<string>("");
|
export const yjsConnectionStatusAtom = atom<string>("");
|
||||||
|
|
||||||
export const showAiMenuAtom = atom(false);
|
export const showAiMenuAtom = atom(false);
|
||||||
|
|
||||||
|
export const showLinkMenuAtom = atom(false);
|
||||||
|
|||||||
@@ -26,7 +26,7 @@ import { v7 as uuid7 } from "uuid";
|
|||||||
import { isCellSelection, isTextSelected } from "@docmost/editor-ext";
|
import { isCellSelection, isTextSelected } from "@docmost/editor-ext";
|
||||||
import { LinkSelector } from "@/features/editor/components/bubble-menu/link-selector.tsx";
|
import { LinkSelector } from "@/features/editor/components/bubble-menu/link-selector.tsx";
|
||||||
import { useTranslation } from "react-i18next";
|
import { useTranslation } from "react-i18next";
|
||||||
import { showAiMenuAtom } from "@/features/editor/atoms/editor-atoms";
|
import { showAiMenuAtom, showLinkMenuAtom } from "@/features/editor/atoms/editor-atoms";
|
||||||
import { workspaceAtom } from "@/features/user/atoms/current-user-atom";
|
import { workspaceAtom } from "@/features/user/atoms/current-user-atom";
|
||||||
|
|
||||||
export interface BubbleMenuItem {
|
export interface BubbleMenuItem {
|
||||||
@@ -49,6 +49,8 @@ export const EditorBubbleMenu: FC<EditorBubbleMenuProps> = (props) => {
|
|||||||
const [, setDraftCommentId] = useAtom(draftCommentIdAtom);
|
const [, setDraftCommentId] = useAtom(draftCommentIdAtom);
|
||||||
const showCommentPopupRef = useRef(showCommentPopup);
|
const showCommentPopupRef = useRef(showCommentPopup);
|
||||||
const showAiMenuRef = useRef(showAiMenu);
|
const showAiMenuRef = useRef(showAiMenu);
|
||||||
|
const [showLinkMenu] = useAtom(showLinkMenuAtom);
|
||||||
|
const showLinkMenuRef = useRef(showLinkMenu);
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
showCommentPopupRef.current = showCommentPopup;
|
showCommentPopupRef.current = showCommentPopup;
|
||||||
@@ -58,6 +60,10 @@ export const EditorBubbleMenu: FC<EditorBubbleMenuProps> = (props) => {
|
|||||||
showAiMenuRef.current = showAiMenu;
|
showAiMenuRef.current = showAiMenu;
|
||||||
}, [showAiMenu]);
|
}, [showAiMenu]);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
showLinkMenuRef.current = showLinkMenu;
|
||||||
|
}, [showLinkMenu]);
|
||||||
|
|
||||||
const editorState = useEditorState({
|
const editorState = useEditorState({
|
||||||
editor: props.editor,
|
editor: props.editor,
|
||||||
selector: (ctx) => {
|
selector: (ctx) => {
|
||||||
@@ -135,6 +141,7 @@ export const EditorBubbleMenu: FC<EditorBubbleMenuProps> = (props) => {
|
|||||||
isNodeSelection(selection) ||
|
isNodeSelection(selection) ||
|
||||||
isCellSelection(selection) ||
|
isCellSelection(selection) ||
|
||||||
showAiMenuRef.current ||
|
showAiMenuRef.current ||
|
||||||
|
showLinkMenuRef.current ||
|
||||||
showCommentPopupRef?.current
|
showCommentPopupRef?.current
|
||||||
) {
|
) {
|
||||||
return false;
|
return false;
|
||||||
@@ -147,7 +154,6 @@ export const EditorBubbleMenu: FC<EditorBubbleMenuProps> = (props) => {
|
|||||||
onHide: () => {
|
onHide: () => {
|
||||||
setIsNodeSelectorOpen(false);
|
setIsNodeSelectorOpen(false);
|
||||||
setIsTextAlignmentOpen(false);
|
setIsTextAlignmentOpen(false);
|
||||||
setIsLinkSelectorOpen(false);
|
|
||||||
setIsColorSelectorOpen(false);
|
setIsColorSelectorOpen(false);
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
@@ -155,11 +161,10 @@ export const EditorBubbleMenu: FC<EditorBubbleMenuProps> = (props) => {
|
|||||||
|
|
||||||
const [isNodeSelectorOpen, setIsNodeSelectorOpen] = useState(false);
|
const [isNodeSelectorOpen, setIsNodeSelectorOpen] = useState(false);
|
||||||
const [isTextAlignmentSelectorOpen, setIsTextAlignmentOpen] = useState(false);
|
const [isTextAlignmentSelectorOpen, setIsTextAlignmentOpen] = useState(false);
|
||||||
const [isLinkSelectorOpen, setIsLinkSelectorOpen] = useState(false);
|
|
||||||
const [isColorSelectorOpen, setIsColorSelectorOpen] = useState(false);
|
const [isColorSelectorOpen, setIsColorSelectorOpen] = useState(false);
|
||||||
|
|
||||||
// Hide the bubble menu immediately when AI menu is shown
|
// Hide the bubble menu immediately when AI menu is shown
|
||||||
if (showAiMenu) return;
|
if (showAiMenu || showLinkMenu) return;
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<BubbleMenu
|
<BubbleMenu
|
||||||
@@ -189,7 +194,6 @@ export const EditorBubbleMenu: FC<EditorBubbleMenuProps> = (props) => {
|
|||||||
setIsOpen={() => {
|
setIsOpen={() => {
|
||||||
setIsNodeSelectorOpen(!isNodeSelectorOpen);
|
setIsNodeSelectorOpen(!isNodeSelectorOpen);
|
||||||
setIsTextAlignmentOpen(false);
|
setIsTextAlignmentOpen(false);
|
||||||
setIsLinkSelectorOpen(false);
|
|
||||||
setIsColorSelectorOpen(false);
|
setIsColorSelectorOpen(false);
|
||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
@@ -200,7 +204,6 @@ export const EditorBubbleMenu: FC<EditorBubbleMenuProps> = (props) => {
|
|||||||
setIsOpen={() => {
|
setIsOpen={() => {
|
||||||
setIsTextAlignmentOpen(!isTextAlignmentSelectorOpen);
|
setIsTextAlignmentOpen(!isTextAlignmentSelectorOpen);
|
||||||
setIsNodeSelectorOpen(false);
|
setIsNodeSelectorOpen(false);
|
||||||
setIsLinkSelectorOpen(false);
|
|
||||||
setIsColorSelectorOpen(false);
|
setIsColorSelectorOpen(false);
|
||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
@@ -224,16 +227,7 @@ export const EditorBubbleMenu: FC<EditorBubbleMenuProps> = (props) => {
|
|||||||
))}
|
))}
|
||||||
</ActionIcon.Group>
|
</ActionIcon.Group>
|
||||||
|
|
||||||
<LinkSelector
|
<LinkSelector />
|
||||||
editor={props.editor}
|
|
||||||
isOpen={isLinkSelectorOpen}
|
|
||||||
setIsOpen={(value) => {
|
|
||||||
setIsLinkSelectorOpen(value);
|
|
||||||
setIsNodeSelectorOpen(false);
|
|
||||||
setIsTextAlignmentOpen(false);
|
|
||||||
setIsColorSelectorOpen(false);
|
|
||||||
}}
|
|
||||||
/>
|
|
||||||
|
|
||||||
<ColorSelector
|
<ColorSelector
|
||||||
editor={props.editor}
|
editor={props.editor}
|
||||||
@@ -242,7 +236,6 @@ export const EditorBubbleMenu: FC<EditorBubbleMenuProps> = (props) => {
|
|||||||
setIsColorSelectorOpen(!isColorSelectorOpen);
|
setIsColorSelectorOpen(!isColorSelectorOpen);
|
||||||
setIsNodeSelectorOpen(false);
|
setIsNodeSelectorOpen(false);
|
||||||
setIsTextAlignmentOpen(false);
|
setIsTextAlignmentOpen(false);
|
||||||
setIsLinkSelectorOpen(false);
|
|
||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
|
|
||||||
|
|||||||
@@ -1,68 +1,25 @@
|
|||||||
import { Dispatch, FC, SetStateAction, useCallback } from "react";
|
import { FC } from "react";
|
||||||
import { IconLink } from "@tabler/icons-react";
|
import { IconLink } from "@tabler/icons-react";
|
||||||
import { ActionIcon, Popover, Tooltip } from "@mantine/core";
|
import { ActionIcon, Tooltip } from "@mantine/core";
|
||||||
import { useEditor } from "@tiptap/react";
|
import { useSetAtom } from "jotai";
|
||||||
import { TextSelection } from "@tiptap/pm/state";
|
|
||||||
import { LinkEditorPanel } from "@/features/editor/components/link/link-editor-panel.tsx";
|
|
||||||
import { normalizeUrl } from "@/features/editor/components/link/link-view";
|
|
||||||
import { useTranslation } from "react-i18next";
|
import { useTranslation } from "react-i18next";
|
||||||
|
import { showLinkMenuAtom } from "@/features/editor/atoms/editor-atoms";
|
||||||
|
|
||||||
interface LinkSelectorProps {
|
export const LinkSelector: FC = () => {
|
||||||
editor: ReturnType<typeof useEditor>;
|
|
||||||
isOpen: boolean;
|
|
||||||
setIsOpen: Dispatch<SetStateAction<boolean>>;
|
|
||||||
}
|
|
||||||
|
|
||||||
export const LinkSelector: FC<LinkSelectorProps> = ({
|
|
||||||
editor,
|
|
||||||
isOpen,
|
|
||||||
setIsOpen,
|
|
||||||
}) => {
|
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
const onLink = useCallback(
|
const setShowLinkMenu = useSetAtom(showLinkMenuAtom);
|
||||||
(url: string, internal?: boolean) => {
|
|
||||||
setIsOpen(false);
|
|
||||||
editor
|
|
||||||
.chain()
|
|
||||||
.focus()
|
|
||||||
.setLink({ href: internal ? url : normalizeUrl(url), internal: !!internal } as any)
|
|
||||||
.command(({ tr }) => {
|
|
||||||
tr.setSelection(TextSelection.create(tr.doc, tr.selection.to));
|
|
||||||
return true;
|
|
||||||
})
|
|
||||||
.run();
|
|
||||||
},
|
|
||||||
[editor, setIsOpen],
|
|
||||||
);
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Popover
|
<Tooltip label={t("Add link")} withArrow>
|
||||||
width={320}
|
<ActionIcon
|
||||||
opened={isOpen}
|
variant="default"
|
||||||
trapFocus
|
size="lg"
|
||||||
offset={{ mainAxis: 35, crossAxis: 0 }}
|
radius="0"
|
||||||
withArrow
|
style={{ border: "none" }}
|
||||||
shadow="md"
|
onClick={() => setShowLinkMenu(true)}
|
||||||
>
|
>
|
||||||
<Popover.Target>
|
<IconLink size={16} />
|
||||||
<Tooltip label={t("Add link")} withArrow>
|
</ActionIcon>
|
||||||
<ActionIcon
|
</Tooltip>
|
||||||
variant="default"
|
|
||||||
size="lg"
|
|
||||||
radius="0"
|
|
||||||
style={{
|
|
||||||
border: "none",
|
|
||||||
}}
|
|
||||||
onClick={() => setIsOpen(!isOpen)}
|
|
||||||
>
|
|
||||||
<IconLink size={16} />
|
|
||||||
</ActionIcon>
|
|
||||||
</Tooltip>
|
|
||||||
</Popover.Target>
|
|
||||||
|
|
||||||
<Popover.Dropdown p="sm">
|
|
||||||
<LinkEditorPanel onSetLink={onLink} />
|
|
||||||
</Popover.Dropdown>
|
|
||||||
</Popover>
|
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -36,7 +36,7 @@ export const LinkEditorPanel = ({
|
|||||||
includeUsers: false,
|
includeUsers: false,
|
||||||
includePages: true,
|
includePages: true,
|
||||||
spaceId: space?.id,
|
spaceId: space?.id,
|
||||||
limit: state.isSearchQuery ? 10 : 5,
|
limit: state.isSearchQuery ? 10 : 3,
|
||||||
preload: true,
|
preload: true,
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -105,6 +105,7 @@ export const LinkEditorPanel = ({
|
|||||||
value={state.url}
|
value={state.url}
|
||||||
onChange={state.onChange}
|
onChange={state.onChange}
|
||||||
onKeyDown={handleKeyDown}
|
onKeyDown={handleKeyDown}
|
||||||
|
data-autofocus
|
||||||
autoFocus
|
autoFocus
|
||||||
/>
|
/>
|
||||||
</form>
|
</form>
|
||||||
|
|||||||
@@ -0,0 +1,114 @@
|
|||||||
|
import { FC, useCallback, useEffect, useRef } from "react";
|
||||||
|
import { BubbleMenu } from "@tiptap/react/menus";
|
||||||
|
import type { Editor } from "@tiptap/react";
|
||||||
|
import { useAtom } from "jotai";
|
||||||
|
import { isTextSelected } from "@docmost/editor-ext";
|
||||||
|
import { showLinkMenuAtom } from "@/features/editor/atoms/editor-atoms";
|
||||||
|
import { LinkEditorPanel } from "@/features/editor/components/link/link-editor-panel";
|
||||||
|
import { normalizeUrl } from "@/features/editor/components/link/link-view";
|
||||||
|
import { TextSelection } from "@tiptap/pm/state";
|
||||||
|
import { Paper } from "@mantine/core";
|
||||||
|
|
||||||
|
type EditorLinkMenuProps = {
|
||||||
|
editor: Editor;
|
||||||
|
};
|
||||||
|
|
||||||
|
export const EditorLinkMenu: FC<EditorLinkMenuProps> = ({ editor }) => {
|
||||||
|
const [showLinkMenu, setShowLinkMenu] = useAtom(showLinkMenuAtom);
|
||||||
|
const showLinkMenuRef = useRef(showLinkMenu);
|
||||||
|
|
||||||
|
const containerRef = useRef<HTMLDivElement>(null);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
showLinkMenuRef.current = showLinkMenu;
|
||||||
|
if (showLinkMenu) {
|
||||||
|
editor.commands.focus();
|
||||||
|
}
|
||||||
|
}, [showLinkMenu, editor]);
|
||||||
|
|
||||||
|
const focusInput = useCallback(() => {
|
||||||
|
requestAnimationFrame(() => {
|
||||||
|
containerRef.current
|
||||||
|
?.querySelector<HTMLInputElement>("input")
|
||||||
|
?.focus({ preventScroll: true });
|
||||||
|
});
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
const onSetLink = useCallback(
|
||||||
|
(url: string, internal?: boolean) => {
|
||||||
|
editor
|
||||||
|
.chain()
|
||||||
|
.focus()
|
||||||
|
.setLink({
|
||||||
|
href: internal ? url : normalizeUrl(url),
|
||||||
|
internal: !!internal,
|
||||||
|
} as any)
|
||||||
|
.command(({ tr }) => {
|
||||||
|
tr.setSelection(TextSelection.create(tr.doc, tr.selection.to));
|
||||||
|
return true;
|
||||||
|
})
|
||||||
|
.run();
|
||||||
|
setShowLinkMenu(false);
|
||||||
|
},
|
||||||
|
[editor, setShowLinkMenu],
|
||||||
|
);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
if (!showLinkMenu) return;
|
||||||
|
|
||||||
|
const dismiss = () => {
|
||||||
|
setShowLinkMenu(false);
|
||||||
|
editor.commands.focus();
|
||||||
|
editor.commands.setTextSelection(editor.state.selection.to);
|
||||||
|
};
|
||||||
|
|
||||||
|
const handleKeyDown = (e: KeyboardEvent) => {
|
||||||
|
if (e.key === "Escape") {
|
||||||
|
dismiss();
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const handleMouseDown = (e: MouseEvent) => {
|
||||||
|
if (containerRef.current && !containerRef.current.contains(e.target as Node)) {
|
||||||
|
dismiss();
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
document.addEventListener("keydown", handleKeyDown);
|
||||||
|
document.addEventListener("mousedown", handleMouseDown);
|
||||||
|
return () => {
|
||||||
|
document.removeEventListener("keydown", handleKeyDown);
|
||||||
|
document.removeEventListener("mousedown", handleMouseDown);
|
||||||
|
};
|
||||||
|
}, [showLinkMenu, setShowLinkMenu]);
|
||||||
|
|
||||||
|
if (!showLinkMenu) return null;
|
||||||
|
|
||||||
|
return (
|
||||||
|
<BubbleMenu
|
||||||
|
editor={editor}
|
||||||
|
shouldShow={({ editor, state }) => {
|
||||||
|
const { empty } = state.selection;
|
||||||
|
return (
|
||||||
|
showLinkMenuRef.current &&
|
||||||
|
editor.isEditable &&
|
||||||
|
!empty &&
|
||||||
|
isTextSelected(editor)
|
||||||
|
);
|
||||||
|
}}
|
||||||
|
options={{
|
||||||
|
placement: "bottom",
|
||||||
|
offset: 8,
|
||||||
|
onShow: focusInput,
|
||||||
|
onHide: () => {
|
||||||
|
setShowLinkMenu(false);
|
||||||
|
},
|
||||||
|
}}
|
||||||
|
style={{ zIndex: 198, position: "relative" }}
|
||||||
|
>
|
||||||
|
<Paper ref={containerRef} w={320} p="sm" shadow="md" radius={6} withBorder>
|
||||||
|
<LinkEditorPanel onSetLink={onSetLink} />
|
||||||
|
</Paper>
|
||||||
|
</BubbleMenu>
|
||||||
|
);
|
||||||
|
};
|
||||||
@@ -66,6 +66,7 @@ import { jwtDecode } from "jwt-decode";
|
|||||||
import { searchSpotlight } from "@/features/search/constants.ts";
|
import { searchSpotlight } from "@/features/search/constants.ts";
|
||||||
import { useEditorScroll } from "./hooks/use-editor-scroll";
|
import { useEditorScroll } from "./hooks/use-editor-scroll";
|
||||||
import { EditorAiMenu } from "@/ee/ai/components/editor/ai-menu/ai-menu";
|
import { EditorAiMenu } from "@/ee/ai/components/editor/ai-menu/ai-menu";
|
||||||
|
import { EditorLinkMenu } from "@/features/editor/components/link/link-menu";
|
||||||
import ColumnsMenu from "@/features/editor/components/columns/columns-menu.tsx";
|
import ColumnsMenu from "@/features/editor/components/columns/columns-menu.tsx";
|
||||||
|
|
||||||
interface PageEditorProps {
|
interface PageEditorProps {
|
||||||
@@ -407,6 +408,7 @@ export default function PageEditor({
|
|||||||
{editor && editorIsEditable && (
|
{editor && editorIsEditable && (
|
||||||
<div>
|
<div>
|
||||||
<EditorAiMenu editor={editor} />
|
<EditorAiMenu editor={editor} />
|
||||||
|
<EditorLinkMenu editor={editor} />
|
||||||
<EditorBubbleMenu editor={editor} />
|
<EditorBubbleMenu editor={editor} />
|
||||||
<TableMenu editor={editor} />
|
<TableMenu editor={editor} />
|
||||||
<TableCellMenu editor={editor} appendTo={menuContainerRef} />
|
<TableCellMenu editor={editor} appendTo={menuContainerRef} />
|
||||||
|
|||||||
@@ -12,7 +12,7 @@ import {
|
|||||||
} from "@tabler/icons-react";
|
} from "@tabler/icons-react";
|
||||||
import { CustomAvatar } from "@/components/ui/custom-avatar";
|
import { CustomAvatar } from "@/components/ui/custom-avatar";
|
||||||
import { INotification } from "../types/notification.types";
|
import { INotification } from "../types/notification.types";
|
||||||
import { useTranslation } from "react-i18next";
|
import { Trans, useTranslation } from "react-i18next";
|
||||||
import { useNavigate } from "react-router-dom";
|
import { useNavigate } from "react-router-dom";
|
||||||
import { useState } from "react";
|
import { useState } from "react";
|
||||||
import { useMarkReadMutation } from "../queries/notification-query";
|
import { useMarkReadMutation } from "../queries/notification-query";
|
||||||
@@ -36,20 +36,20 @@ export function NotificationItem({
|
|||||||
|
|
||||||
const isUnread = !notification.readAt;
|
const isUnread = !notification.readAt;
|
||||||
|
|
||||||
const getNotificationMessage = (): string => {
|
const getNotificationMessageKey = (): string => {
|
||||||
switch (notification.type) {
|
switch (notification.type) {
|
||||||
case "comment.user_mention":
|
case "comment.user_mention":
|
||||||
return t("mentioned you in a comment");
|
return "<bold>{{name}}</bold> mentioned you in a comment";
|
||||||
case "comment.created":
|
case "comment.created":
|
||||||
return t("commented on a page");
|
return "<bold>{{name}}</bold> commented on a page";
|
||||||
case "comment.resolved":
|
case "comment.resolved":
|
||||||
return t("resolved a comment");
|
return "<bold>{{name}}</bold> resolved a comment";
|
||||||
case "page.user_mention":
|
case "page.user_mention":
|
||||||
return t("mentioned you on a page");
|
return "<bold>{{name}}</bold> mentioned you on a page";
|
||||||
case "page.permission_granted":
|
case "page.permission_granted":
|
||||||
return notification.data?.role === "writer"
|
return notification.data?.role === "writer"
|
||||||
? t("gave you edit access to a page")
|
? "<bold>{{name}}</bold> gave you edit access to a page"
|
||||||
: t("gave you view access to a page");
|
: "<bold>{{name}}</bold> gave you view access to a page";
|
||||||
default:
|
default:
|
||||||
return "";
|
return "";
|
||||||
}
|
}
|
||||||
@@ -95,10 +95,11 @@ export function NotificationItem({
|
|||||||
|
|
||||||
<div style={{ flex: 1, minWidth: 0 }}>
|
<div style={{ flex: 1, minWidth: 0 }}>
|
||||||
<Text size="sm" lineClamp={2}>
|
<Text size="sm" lineClamp={2}>
|
||||||
<Text span fw={600}>
|
<Trans
|
||||||
{notification.actor?.name}
|
i18nKey={getNotificationMessageKey()}
|
||||||
</Text>{" "}
|
values={{ name: notification.actor?.name }}
|
||||||
{getNotificationMessage()}
|
components={{ bold: <Text span fw={600} /> }}
|
||||||
|
/>
|
||||||
</Text>
|
</Text>
|
||||||
|
|
||||||
{notification.page && (
|
{notification.page && (
|
||||||
|
|||||||
@@ -28,9 +28,11 @@ import { IPage } from "@/features/page/types/page.types.ts";
|
|||||||
import React, { useEffect, useRef, useState } from "react";
|
import React, { useEffect, useRef, useState } from "react";
|
||||||
import { useTranslation } from "react-i18next";
|
import { useTranslation } from "react-i18next";
|
||||||
import { ConfluenceIcon } from "@/components/icons/confluence-icon.tsx";
|
import { ConfluenceIcon } from "@/components/icons/confluence-icon.tsx";
|
||||||
import { getFileImportSizeLimit, isCloud } from "@/lib/config.ts";
|
import { getFileImportSizeLimit } from "@/lib/config.ts";
|
||||||
import { formatBytes } from "@/lib";
|
import { formatBytes } from "@/lib";
|
||||||
import { workspaceAtom } from "@/features/user/atoms/current-user-atom.ts";
|
import { useHasFeature } from "@/ee/hooks/use-feature";
|
||||||
|
import { Feature } from "@/ee/features";
|
||||||
|
import { useUpgradeLabel } from "@/ee/hooks/use-upgrade-label";
|
||||||
import { getFileTaskById } from "@/features/file-task/services/file-task-service.ts";
|
import { getFileTaskById } from "@/features/file-task/services/file-task-service.ts";
|
||||||
import { queryClient } from "@/main.tsx";
|
import { queryClient } from "@/main.tsx";
|
||||||
import { useQueryEmit } from "@/features/websocket/use-query-emit.ts";
|
import { useQueryEmit } from "@/features/websocket/use-query-emit.ts";
|
||||||
@@ -82,7 +84,6 @@ interface ImportFormatSelection {
|
|||||||
function ImportFormatSelection({ spaceId, onClose }: ImportFormatSelection) {
|
function ImportFormatSelection({ spaceId, onClose }: ImportFormatSelection) {
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
const [treeData, setTreeData] = useAtom(treeDataAtom);
|
const [treeData, setTreeData] = useAtom(treeDataAtom);
|
||||||
const [workspace] = useAtom(workspaceAtom);
|
|
||||||
const [fileTaskId, setFileTaskId] = useState<string | null>(null);
|
const [fileTaskId, setFileTaskId] = useState<string | null>(null);
|
||||||
const emit = useQueryEmit();
|
const emit = useQueryEmit();
|
||||||
|
|
||||||
@@ -93,8 +94,9 @@ function ImportFormatSelection({ spaceId, onClose }: ImportFormatSelection) {
|
|||||||
const confluenceFileRef = useRef<() => void>(null);
|
const confluenceFileRef = useRef<() => void>(null);
|
||||||
const zipFileRef = useRef<() => void>(null);
|
const zipFileRef = useRef<() => void>(null);
|
||||||
|
|
||||||
const canUseConfluence = isCloud() || workspace?.hasLicenseKey;
|
const canUseConfluence = useHasFeature(Feature.CONFLUENCE_IMPORT);
|
||||||
const canUseDocx = isCloud() || workspace?.hasLicenseKey;
|
const canUseDocx = useHasFeature(Feature.DOCX_IMPORT);
|
||||||
|
const upgradeLabel = useUpgradeLabel();
|
||||||
|
|
||||||
const handleZipUpload = async (selectedFile: File, source: string) => {
|
const handleZipUpload = async (selectedFile: File, source: string) => {
|
||||||
if (!selectedFile) {
|
if (!selectedFile) {
|
||||||
@@ -360,7 +362,7 @@ function ImportFormatSelection({ spaceId, onClose }: ImportFormatSelection) {
|
|||||||
>
|
>
|
||||||
{(props) => (
|
{(props) => (
|
||||||
<Tooltip
|
<Tooltip
|
||||||
label={t("Available in enterprise edition")}
|
label={upgradeLabel}
|
||||||
disabled={canUseDocx}
|
disabled={canUseDocx}
|
||||||
>
|
>
|
||||||
<Button
|
<Button
|
||||||
@@ -399,7 +401,7 @@ function ImportFormatSelection({ spaceId, onClose }: ImportFormatSelection) {
|
|||||||
>
|
>
|
||||||
{(props) => (
|
{(props) => (
|
||||||
<Tooltip
|
<Tooltip
|
||||||
label={t("Available in enterprise edition")}
|
label={upgradeLabel}
|
||||||
disabled={canUseConfluence}
|
disabled={canUseConfluence}
|
||||||
>
|
>
|
||||||
<Button
|
<Button
|
||||||
|
|||||||
@@ -22,9 +22,9 @@ import {
|
|||||||
import { useTranslation } from "react-i18next";
|
import { useTranslation } from "react-i18next";
|
||||||
import { useDebouncedValue } from "@mantine/hooks";
|
import { useDebouncedValue } from "@mantine/hooks";
|
||||||
import { useGetSpacesQuery } from "@/features/space/queries/space-query";
|
import { useGetSpacesQuery } from "@/features/space/queries/space-query";
|
||||||
import { useLicense } from "@/ee/hooks/use-license";
|
import { useHasFeature } from "@/ee/hooks/use-feature";
|
||||||
|
import { Feature } from "@/ee/features";
|
||||||
import classes from "./search-spotlight-filters.module.css";
|
import classes from "./search-spotlight-filters.module.css";
|
||||||
import { isCloud } from "@/lib/config.ts";
|
|
||||||
import { useAtom } from "jotai";
|
import { useAtom } from "jotai";
|
||||||
import { workspaceAtom } from "@/features/user/atoms/current-user-atom.ts";
|
import { workspaceAtom } from "@/features/user/atoms/current-user-atom.ts";
|
||||||
|
|
||||||
@@ -42,7 +42,7 @@ export function SearchSpotlightFilters({
|
|||||||
isAiMode = false,
|
isAiMode = false,
|
||||||
}: SearchSpotlightFiltersProps) {
|
}: SearchSpotlightFiltersProps) {
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
const { hasLicenseKey } = useLicense();
|
const hasAttachmentIndexing = useHasFeature(Feature.ATTACHMENT_INDEXING);
|
||||||
const [selectedSpaceId, setSelectedSpaceId] = useState<string | null>(
|
const [selectedSpaceId, setSelectedSpaceId] = useState<string | null>(
|
||||||
spaceId || null,
|
spaceId || null,
|
||||||
);
|
);
|
||||||
@@ -87,7 +87,7 @@ export function SearchSpotlightFilters({
|
|||||||
{
|
{
|
||||||
value: "attachment",
|
value: "attachment",
|
||||||
label: t("Attachments"),
|
label: t("Attachments"),
|
||||||
disabled: !isCloud() && !hasLicenseKey,
|
disabled: !hasAttachmentIndexing,
|
||||||
},
|
},
|
||||||
];
|
];
|
||||||
|
|
||||||
|
|||||||
@@ -11,15 +11,16 @@ import { useUnifiedSearch } from "../hooks/use-unified-search.ts";
|
|||||||
import { useAiSearch } from "../../../ee/ai/hooks/use-ai-search.ts";
|
import { useAiSearch } from "../../../ee/ai/hooks/use-ai-search.ts";
|
||||||
import { SearchResultItem } from "./search-result-item.tsx";
|
import { SearchResultItem } from "./search-result-item.tsx";
|
||||||
import { AiSearchResult } from "../../../ee/ai/components/ai-search-result.tsx";
|
import { AiSearchResult } from "../../../ee/ai/components/ai-search-result.tsx";
|
||||||
import { useLicense } from "@/ee/hooks/use-license.tsx";
|
import { useHasFeature } from "@/ee/hooks/use-feature";
|
||||||
import { isCloud } from "@/lib/config.ts";
|
import { Feature } from "@/ee/features";
|
||||||
|
|
||||||
interface SearchSpotlightProps {
|
interface SearchSpotlightProps {
|
||||||
spaceId?: string;
|
spaceId?: string;
|
||||||
}
|
}
|
||||||
export function SearchSpotlight({ spaceId }: SearchSpotlightProps) {
|
export function SearchSpotlight({ spaceId }: SearchSpotlightProps) {
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
const { hasLicenseKey } = useLicense();
|
const hasAiFeature = useHasFeature(Feature.AI);
|
||||||
|
const hasAttachmentIndexing = useHasFeature(Feature.ATTACHMENT_INDEXING);
|
||||||
const [query, setQuery] = useState("");
|
const [query, setQuery] = useState("");
|
||||||
const [debouncedSearchQuery] = useDebouncedValue(query, 300);
|
const [debouncedSearchQuery] = useDebouncedValue(query, 300);
|
||||||
const [filters, setFilters] = useState<{
|
const [filters, setFilters] = useState<{
|
||||||
@@ -84,7 +85,7 @@ export function SearchSpotlight({ spaceId }: SearchSpotlightProps) {
|
|||||||
|
|
||||||
// Determine result type for rendering
|
// Determine result type for rendering
|
||||||
const isAttachmentSearch =
|
const isAttachmentSearch =
|
||||||
filters.contentType === "attachment" && (hasLicenseKey || isCloud());
|
filters.contentType === "attachment" && hasAttachmentIndexing;
|
||||||
|
|
||||||
const resultItems = (searchResults || []).map((result) => (
|
const resultItems = (searchResults || []).map((result) => (
|
||||||
<SearchResultItem
|
<SearchResultItem
|
||||||
@@ -134,7 +135,7 @@ export function SearchSpotlight({ spaceId }: SearchSpotlightProps) {
|
|||||||
}
|
}
|
||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
{isAiMode && hasLicenseKey && (
|
{isAiMode && hasAiFeature && (
|
||||||
<Button
|
<Button
|
||||||
size="xs"
|
size="xs"
|
||||||
leftSection={<IconSparkles size={16} />}
|
leftSection={<IconSparkles size={16} />}
|
||||||
|
|||||||
@@ -8,8 +8,8 @@ import {
|
|||||||
IPageSearch,
|
IPageSearch,
|
||||||
IPageSearchParams,
|
IPageSearchParams,
|
||||||
} from "@/features/search/types/search.types";
|
} from "@/features/search/types/search.types";
|
||||||
import { useLicense } from "@/ee/hooks/use-license";
|
import { useHasFeature } from "@/ee/hooks/use-feature";
|
||||||
import { isCloud } from "@/lib/config";
|
import { Feature } from "@/ee/features";
|
||||||
|
|
||||||
export type UnifiedSearchResult = IPageSearch | IAttachmentSearch;
|
export type UnifiedSearchResult = IPageSearch | IAttachmentSearch;
|
||||||
|
|
||||||
@@ -21,10 +21,10 @@ export function useUnifiedSearch(
|
|||||||
params: UseUnifiedSearchParams,
|
params: UseUnifiedSearchParams,
|
||||||
enabled: boolean = true,
|
enabled: boolean = true,
|
||||||
): UseQueryResult<UnifiedSearchResult[], Error> {
|
): UseQueryResult<UnifiedSearchResult[], Error> {
|
||||||
const { hasLicenseKey } = useLicense();
|
const hasAttachmentIndexing = useHasFeature(Feature.ATTACHMENT_INDEXING);
|
||||||
|
|
||||||
const isAttachmentSearch =
|
const isAttachmentSearch =
|
||||||
params.contentType === "attachment" && (isCloud() || hasLicenseKey);
|
params.contentType === "attachment" && hasAttachmentIndexing;
|
||||||
const searchType = isAttachmentSearch ? "attachment" : "page";
|
const searchType = isAttachmentSearch ? "attachment" : "page";
|
||||||
|
|
||||||
return useQuery({
|
return useQuery({
|
||||||
|
|||||||
@@ -180,7 +180,7 @@ export default function ShareShell({
|
|||||||
<AppShell.Main>
|
<AppShell.Main>
|
||||||
{children}
|
{children}
|
||||||
|
|
||||||
{data && shareId && !data.hasLicenseKey && <ShareBranding />}
|
{data && shareId && !(data.features?.length > 0) && <ShareBranding />}
|
||||||
</AppShell.Main>
|
</AppShell.Main>
|
||||||
|
|
||||||
<AppShell.Aside
|
<AppShell.Aside
|
||||||
|
|||||||
@@ -41,7 +41,7 @@ export interface ISharedPage extends IShare {
|
|||||||
level: number;
|
level: number;
|
||||||
sharedPage: { id: string; slugId: string; title: string; icon: string };
|
sharedPage: { id: string; slugId: string; title: string; icon: string };
|
||||||
};
|
};
|
||||||
hasLicenseKey: boolean;
|
features?: string[];
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface IShareForPage extends IShare {
|
export interface IShareForPage extends IShare {
|
||||||
@@ -71,5 +71,5 @@ export interface IShareInfoInput {
|
|||||||
export interface ISharedPageTree {
|
export interface ISharedPageTree {
|
||||||
share: IShare;
|
share: IShare;
|
||||||
pageTree: Partial<IPage[]>;
|
pageTree: Partial<IPage[]>;
|
||||||
hasLicenseKey: boolean;
|
features?: string[];
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -19,7 +19,6 @@ import {
|
|||||||
ResponsiveSettingsRow,
|
ResponsiveSettingsRow,
|
||||||
} from "@/components/ui/responsive-settings-row.tsx";
|
} from "@/components/ui/responsive-settings-row.tsx";
|
||||||
import SpacePublicSharingToggle from "@/ee/security/components/space-public-sharing-toggle.tsx";
|
import SpacePublicSharingToggle from "@/ee/security/components/space-public-sharing-toggle.tsx";
|
||||||
import useEnterpriseAccess from "@/ee/hooks/use-enterprise-access.tsx";
|
|
||||||
|
|
||||||
interface SpaceDetailsProps {
|
interface SpaceDetailsProps {
|
||||||
spaceId: string;
|
spaceId: string;
|
||||||
@@ -28,8 +27,7 @@ interface SpaceDetailsProps {
|
|||||||
export default function SpaceDetails({ spaceId, readOnly }: SpaceDetailsProps) {
|
export default function SpaceDetails({ spaceId, readOnly }: SpaceDetailsProps) {
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
const { data: space, isLoading, refetch } = useSpaceQuery(spaceId);
|
const { data: space, isLoading, refetch } = useSpaceQuery(spaceId);
|
||||||
const hasEnterpriseAccess = useEnterpriseAccess();
|
const showSharingToggle = !readOnly;
|
||||||
const showSharingToggle = !readOnly && hasEnterpriseAccess;
|
|
||||||
const [exportOpened, { open: openExportModal, close: closeExportModal }] =
|
const [exportOpened, { open: openExportModal, close: closeExportModal }] =
|
||||||
useDisclosure(false);
|
useDisclosure(false);
|
||||||
const [isIconUploading, setIsIconUploading] = useState(false);
|
const [isIconUploading, setIsIconUploading] = useState(false);
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import { useAtom } from "jotai";
|
import { useAtom, useSetAtom } from "jotai";
|
||||||
import { currentUserAtom } from "@/features/user/atoms/current-user-atom";
|
import { currentUserAtom } from "@/features/user/atoms/current-user-atom";
|
||||||
import React, { useEffect } from "react";
|
import React, { useEffect } from "react";
|
||||||
import useCurrentUser from "@/features/user/hooks/use-current-user";
|
import useCurrentUser from "@/features/user/hooks/use-current-user";
|
||||||
@@ -11,10 +11,14 @@ import { useTreeSocket } from "@/features/websocket/use-tree-socket.ts";
|
|||||||
import { useNotificationSocket } from "@/features/notification/hooks/use-notification-socket.ts";
|
import { useNotificationSocket } from "@/features/notification/hooks/use-notification-socket.ts";
|
||||||
import { useCollabToken } from "@/features/auth/queries/auth-query.tsx";
|
import { useCollabToken } from "@/features/auth/queries/auth-query.tsx";
|
||||||
import { Error404 } from "@/components/ui/error-404.tsx";
|
import { Error404 } from "@/components/ui/error-404.tsx";
|
||||||
|
import { useEntitlements } from "@/ee/entitlement/use-entitlements";
|
||||||
|
import { entitlementAtom } from "@/ee/entitlement/entitlement-atom";
|
||||||
|
|
||||||
export function UserProvider({ children }: React.PropsWithChildren) {
|
export function UserProvider({ children }: React.PropsWithChildren) {
|
||||||
const [, setCurrentUser] = useAtom(currentUserAtom);
|
const [, setCurrentUser] = useAtom(currentUserAtom);
|
||||||
|
const setEntitlements = useSetAtom(entitlementAtom);
|
||||||
const { data, isLoading, error, isError } = useCurrentUser();
|
const { data, isLoading, error, isError } = useCurrentUser();
|
||||||
|
const { data: entitlements } = useEntitlements();
|
||||||
const { i18n } = useTranslation();
|
const { i18n } = useTranslation();
|
||||||
const [, setSocket] = useAtom(socketAtom);
|
const [, setSocket] = useAtom(socketAtom);
|
||||||
// fetch collab token on load
|
// fetch collab token on load
|
||||||
@@ -56,6 +60,12 @@ export function UserProvider({ children }: React.PropsWithChildren) {
|
|||||||
}
|
}
|
||||||
}, [data, isLoading]);
|
}, [data, isLoading]);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
if (entitlements) {
|
||||||
|
setEntitlements(entitlements);
|
||||||
|
}
|
||||||
|
}, [entitlements]);
|
||||||
|
|
||||||
if (isLoading) return <></>;
|
if (isLoading) return <></>;
|
||||||
|
|
||||||
if (isError && error?.["response"]?.status === 404) {
|
if (isError && error?.["response"]?.status === 404) {
|
||||||
|
|||||||
@@ -20,7 +20,6 @@ export interface IWorkspace {
|
|||||||
emailDomains: string[];
|
emailDomains: string[];
|
||||||
memberCount?: number;
|
memberCount?: number;
|
||||||
plan?: string;
|
plan?: string;
|
||||||
hasLicenseKey?: boolean;
|
|
||||||
enforceMfa?: boolean;
|
enforceMfa?: boolean;
|
||||||
aiSearch?: boolean;
|
aiSearch?: boolean;
|
||||||
generativeAi?: boolean;
|
generativeAi?: boolean;
|
||||||
@@ -84,7 +83,6 @@ export interface IPublicWorkspace {
|
|||||||
hostname: string;
|
hostname: string;
|
||||||
enforceSso: boolean;
|
enforceSso: boolean;
|
||||||
authProviders: IAuthProvider[];
|
authProviders: IAuthProvider[];
|
||||||
hasLicenseKey?: boolean;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface IVersion {
|
export interface IVersion {
|
||||||
|
|||||||
@@ -1,7 +0,0 @@
|
|||||||
import { isCloud } from "@/lib/config";
|
|
||||||
import { useLicense } from "@/ee/hooks/use-license";
|
|
||||||
|
|
||||||
export const useIsCloudEE = () => {
|
|
||||||
const { hasLicenseKey } = useLicense();
|
|
||||||
return isCloud() || !!hasLicenseKey;
|
|
||||||
};
|
|
||||||
@@ -68,7 +68,7 @@ export default function SharedPage() {
|
|||||||
/>
|
/>
|
||||||
</Container>
|
</Container>
|
||||||
|
|
||||||
{data && !shareId && !data.hasLicenseKey && <ShareBranding />}
|
{data && !shareId && !(data.features?.length > 0) && <ShareBranding />}
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "server",
|
"name": "server",
|
||||||
"version": "0.70.1",
|
"version": "0.70.3",
|
||||||
"description": "",
|
"description": "",
|
||||||
"author": "",
|
"author": "",
|
||||||
"private": true,
|
"private": true,
|
||||||
|
|||||||
@@ -91,15 +91,6 @@ export function extractBearerTokenFromHeader(
|
|||||||
return type === 'Bearer' ? token : undefined;
|
return type === 'Bearer' ? token : undefined;
|
||||||
}
|
}
|
||||||
|
|
||||||
export function hasLicenseOrEE(opts: {
|
|
||||||
licenseKey: string;
|
|
||||||
plan: string;
|
|
||||||
isCloud: boolean;
|
|
||||||
}): boolean {
|
|
||||||
const { licenseKey, plan, isCloud } = opts;
|
|
||||||
return Boolean(licenseKey) || (isCloud && plan === 'business');
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Normalizes a database URL for postgres.js compatibility.
|
* Normalizes a database URL for postgres.js compatibility.
|
||||||
* - Removes `sslmode=no-verify` (not supported by postgres.js), keeps other sslmode values
|
* - Removes `sslmode=no-verify` (not supported by postgres.js), keeps other sslmode values
|
||||||
|
|||||||
@@ -91,9 +91,15 @@ export class SearchService {
|
|||||||
return { items: [] };
|
return { items: [] };
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const isRestricted =
|
||||||
|
await this.pagePermissionRepo.hasRestrictedAncestor(share.pageId);
|
||||||
|
if (isRestricted) {
|
||||||
|
return { items: [] };
|
||||||
|
}
|
||||||
|
|
||||||
const pageIdsToSearch = [];
|
const pageIdsToSearch = [];
|
||||||
if (share.includeSubPages) {
|
if (share.includeSubPages) {
|
||||||
const pageList = await this.pageRepo.getPageAndDescendants(
|
const pageList = await this.pageRepo.getPageAndDescendantsExcludingRestricted(
|
||||||
share.pageId,
|
share.pageId,
|
||||||
{
|
{
|
||||||
includeContent: false,
|
includeContent: false,
|
||||||
|
|||||||
@@ -28,8 +28,7 @@ import { JwtAuthGuard } from '../../common/guards/jwt-auth.guard';
|
|||||||
import { Public } from '../../common/decorators/public.decorator';
|
import { Public } from '../../common/decorators/public.decorator';
|
||||||
import { ShareRepo } from '@docmost/db/repos/share/share.repo';
|
import { ShareRepo } from '@docmost/db/repos/share/share.repo';
|
||||||
import { PaginationOptions } from '@docmost/db/pagination/pagination-options';
|
import { PaginationOptions } from '@docmost/db/pagination/pagination-options';
|
||||||
import { EnvironmentService } from '../../integrations/environment/environment.service';
|
import { LicenseCheckService } from '../../integrations/environment/license-check.service';
|
||||||
import { hasLicenseOrEE } from '../../common/helpers';
|
|
||||||
import { AuditEvent, AuditResource } from '../../common/events/audit-events';
|
import { AuditEvent, AuditResource } from '../../common/events/audit-events';
|
||||||
import {
|
import {
|
||||||
AUDIT_SERVICE,
|
AUDIT_SERVICE,
|
||||||
@@ -45,7 +44,7 @@ export class ShareController {
|
|||||||
private readonly pageRepo: PageRepo,
|
private readonly pageRepo: PageRepo,
|
||||||
private readonly pagePermissionRepo: PagePermissionRepo,
|
private readonly pagePermissionRepo: PagePermissionRepo,
|
||||||
private readonly pageAccessService: PageAccessService,
|
private readonly pageAccessService: PageAccessService,
|
||||||
private readonly environmentService: EnvironmentService,
|
private readonly licenseCheckService: LicenseCheckService,
|
||||||
@Inject(AUDIT_SERVICE) private readonly auditService: IAuditService,
|
@Inject(AUDIT_SERVICE) private readonly auditService: IAuditService,
|
||||||
) {}
|
) {}
|
||||||
|
|
||||||
@@ -81,11 +80,10 @@ export class ShareController {
|
|||||||
|
|
||||||
return {
|
return {
|
||||||
...shareData,
|
...shareData,
|
||||||
hasLicenseKey: hasLicenseOrEE({
|
features: this.licenseCheckService.resolveFeatures(
|
||||||
licenseKey: workspace.licenseKey,
|
workspace.licenseKey,
|
||||||
isCloud: this.environmentService.isCloud(),
|
workspace.plan,
|
||||||
plan: workspace.plan,
|
),
|
||||||
}),
|
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -259,11 +257,10 @@ export class ShareController {
|
|||||||
|
|
||||||
return {
|
return {
|
||||||
...treeData,
|
...treeData,
|
||||||
hasLicenseKey: hasLicenseOrEE({
|
features: this.licenseCheckService.resolveFeatures(
|
||||||
licenseKey: workspace.licenseKey,
|
workspace.licenseKey,
|
||||||
isCloud: this.environmentService.isCloud(),
|
workspace.plan,
|
||||||
plan: workspace.plan,
|
),
|
||||||
}),
|
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -139,10 +139,10 @@ export class SpaceService {
|
|||||||
});
|
});
|
||||||
|
|
||||||
if (
|
if (
|
||||||
!this.licenseCheckService.isValidEELicense(workspace.licenseKey)
|
!this.licenseCheckService.hasFeature(workspace.licenseKey, 'security:settings', workspace.plan)
|
||||||
) {
|
) {
|
||||||
throw new ForbiddenException(
|
throw new ForbiddenException(
|
||||||
'This feature requires a valid enterprise license',
|
'This feature requires a valid license',
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -37,7 +37,6 @@ export class UserController {
|
|||||||
const workspaceInfo = {
|
const workspaceInfo = {
|
||||||
...rest,
|
...rest,
|
||||||
memberCount,
|
memberCount,
|
||||||
hasLicenseKey: Boolean(licenseKey),
|
|
||||||
};
|
};
|
||||||
|
|
||||||
return { user: authUser, workspace: workspaceInfo };
|
return { user: authUser, workspace: workspaceInfo };
|
||||||
|
|||||||
@@ -32,8 +32,10 @@ import {
|
|||||||
} from '../../casl/interfaces/workspace-ability.type';
|
} from '../../casl/interfaces/workspace-ability.type';
|
||||||
import { FastifyReply } from 'fastify';
|
import { FastifyReply } from 'fastify';
|
||||||
import { EnvironmentService } from '../../../integrations/environment/environment.service';
|
import { EnvironmentService } from '../../../integrations/environment/environment.service';
|
||||||
|
import { LicenseCheckService } from '../../../integrations/environment/license-check.service';
|
||||||
import { CheckHostnameDto } from '../dto/check-hostname.dto';
|
import { CheckHostnameDto } from '../dto/check-hostname.dto';
|
||||||
import { RemoveWorkspaceUserDto } from '../dto/remove-workspace-user.dto';
|
import { RemoveWorkspaceUserDto } from '../dto/remove-workspace-user.dto';
|
||||||
|
import { WorkspaceRepo } from '@docmost/db/repos/workspace/workspace.repo';
|
||||||
|
|
||||||
@UseGuards(JwtAuthGuard)
|
@UseGuards(JwtAuthGuard)
|
||||||
@Controller('workspace')
|
@Controller('workspace')
|
||||||
@@ -42,7 +44,9 @@ export class WorkspaceController {
|
|||||||
private readonly workspaceService: WorkspaceService,
|
private readonly workspaceService: WorkspaceService,
|
||||||
private readonly workspaceInvitationService: WorkspaceInvitationService,
|
private readonly workspaceInvitationService: WorkspaceInvitationService,
|
||||||
private readonly workspaceAbility: WorkspaceAbilityFactory,
|
private readonly workspaceAbility: WorkspaceAbilityFactory,
|
||||||
|
private readonly workspaceRepo: WorkspaceRepo,
|
||||||
private environmentService: EnvironmentService,
|
private environmentService: EnvironmentService,
|
||||||
|
private licenseCheckService: LicenseCheckService,
|
||||||
) {}
|
) {}
|
||||||
|
|
||||||
@Public()
|
@Public()
|
||||||
@@ -58,6 +62,23 @@ export class WorkspaceController {
|
|||||||
return this.workspaceService.getWorkspaceInfo(workspace.id);
|
return this.workspaceService.getWorkspaceInfo(workspace.id);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@HttpCode(HttpStatus.OK)
|
||||||
|
@Post('entitlements')
|
||||||
|
async getEntitlements(@AuthWorkspace() workspace: Workspace) {
|
||||||
|
let { licenseKey } = workspace;
|
||||||
|
const { plan } = workspace;
|
||||||
|
|
||||||
|
if (!licenseKey) {
|
||||||
|
licenseKey = await this.workspaceRepo.findLicenseKeyById(workspace.id);
|
||||||
|
}
|
||||||
|
|
||||||
|
return {
|
||||||
|
cloud: this.environmentService.isCloud(),
|
||||||
|
tier: this.licenseCheckService.resolveTier(licenseKey, plan),
|
||||||
|
features: this.licenseCheckService.resolveFeatures(licenseKey, plan),
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
@HttpCode(HttpStatus.OK)
|
@HttpCode(HttpStatus.OK)
|
||||||
@Post('update')
|
@Post('update')
|
||||||
async updateWorkspace(
|
async updateWorkspace(
|
||||||
|
|||||||
@@ -85,7 +85,7 @@ export class WorkspaceService {
|
|||||||
async getWorkspacePublicData(workspaceId: string) {
|
async getWorkspacePublicData(workspaceId: string) {
|
||||||
const workspace = await this.db
|
const workspace = await this.db
|
||||||
.selectFrom('workspaces')
|
.selectFrom('workspaces')
|
||||||
.select(['id', 'name', 'logo', 'hostname', 'enforceSso', 'licenseKey'])
|
.select(['id', 'name', 'logo', 'hostname', 'enforceSso', 'licenseKey', 'plan'])
|
||||||
.select((eb) =>
|
.select((eb) =>
|
||||||
jsonArrayFrom(
|
jsonArrayFrom(
|
||||||
eb
|
eb
|
||||||
@@ -106,12 +106,9 @@ export class WorkspaceService {
|
|||||||
throw new NotFoundException('Workspace not found');
|
throw new NotFoundException('Workspace not found');
|
||||||
}
|
}
|
||||||
|
|
||||||
const { licenseKey, ...rest } = workspace;
|
const { licenseKey, plan, ...rest } = workspace;
|
||||||
|
|
||||||
return {
|
return rest;
|
||||||
...rest,
|
|
||||||
hasLicenseKey: Boolean(licenseKey),
|
|
||||||
};
|
|
||||||
}
|
}
|
||||||
|
|
||||||
async create(
|
async create(
|
||||||
@@ -332,14 +329,32 @@ export class WorkspaceService {
|
|||||||
) {
|
) {
|
||||||
const ws = await this.db
|
const ws = await this.db
|
||||||
.selectFrom('workspaces')
|
.selectFrom('workspaces')
|
||||||
.select(['id', 'licenseKey', 'trashRetentionDays'])
|
.select(['id', 'licenseKey', 'plan', 'trashRetentionDays'])
|
||||||
.where('id', '=', workspaceId)
|
.where('id', '=', workspaceId)
|
||||||
.executeTakeFirst();
|
.executeTakeFirst();
|
||||||
|
|
||||||
if (!this.licenseCheckService.isValidEELicense(ws.licenseKey)) {
|
if (!ws) {
|
||||||
throw new ForbiddenException(
|
throw new NotFoundException('Workspace not found');
|
||||||
'This feature requires a valid enterprise license',
|
}
|
||||||
);
|
|
||||||
|
if (typeof updateWorkspaceDto.mcpEnabled !== 'undefined') {
|
||||||
|
if (!this.licenseCheckService.hasFeature(ws.licenseKey, 'mcp', ws.plan)) {
|
||||||
|
throw new ForbiddenException(
|
||||||
|
'This feature requires a valid license',
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (
|
||||||
|
typeof updateWorkspaceDto.disablePublicSharing !== 'undefined' ||
|
||||||
|
typeof updateWorkspaceDto.trashRetentionDays !== 'undefined' ||
|
||||||
|
typeof updateWorkspaceDto.restrictApiToAdmins !== 'undefined'
|
||||||
|
) {
|
||||||
|
if (!this.licenseCheckService.hasFeature(ws.licenseKey, 'security:settings', ws.plan)) {
|
||||||
|
throw new ForbiddenException(
|
||||||
|
'This feature requires a valid license',
|
||||||
|
);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (
|
if (
|
||||||
@@ -503,10 +518,7 @@ export class WorkspaceService {
|
|||||||
}
|
}
|
||||||
|
|
||||||
const { licenseKey, ...rest } = workspace;
|
const { licenseKey, ...rest } = workspace;
|
||||||
return {
|
return rest;
|
||||||
...rest,
|
|
||||||
hasLicenseKey: Boolean(licenseKey),
|
|
||||||
};
|
|
||||||
}
|
}
|
||||||
|
|
||||||
async getWorkspaceUsers(
|
async getWorkspaceUsers(
|
||||||
|
|||||||
+1
-1
Submodule apps/server/src/ee updated: 47e76280fd...0b5c8646e6
@@ -25,4 +25,75 @@ export class LicenseCheckService {
|
|||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
hasFeature(licenseKey: string, feature: string, plan?: string): boolean {
|
||||||
|
if (this.environmentService.isCloud()) {
|
||||||
|
try {
|
||||||
|
// eslint-disable-next-line @typescript-eslint/no-require-imports
|
||||||
|
const { getFeaturesForCloudPlan } = require('../../ee/licence/feature-registry');
|
||||||
|
return getFeaturesForCloudPlan(plan).has(feature);
|
||||||
|
} catch {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
// eslint-disable-next-line @typescript-eslint/no-require-imports
|
||||||
|
const LicenseModule = require('../../ee/licence/license.service');
|
||||||
|
const licenseService = this.moduleRef.get(LicenseModule.LicenseService, {
|
||||||
|
strict: false,
|
||||||
|
});
|
||||||
|
return licenseService.hasFeature(licenseKey, feature);
|
||||||
|
} catch {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
getFeatures(licenseKey: string): string[] {
|
||||||
|
try {
|
||||||
|
// eslint-disable-next-line @typescript-eslint/no-require-imports
|
||||||
|
const LicenseModule = require('../../ee/licence/license.service');
|
||||||
|
const licenseService = this.moduleRef.get(LicenseModule.LicenseService, {
|
||||||
|
strict: false,
|
||||||
|
});
|
||||||
|
return licenseService.getFeatures(licenseKey);
|
||||||
|
} catch {
|
||||||
|
return [];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
resolveFeatures(licenseKey: string, plan: string): string[] {
|
||||||
|
if (this.environmentService.isCloud()) {
|
||||||
|
try {
|
||||||
|
// eslint-disable-next-line @typescript-eslint/no-require-imports
|
||||||
|
const { getFeaturesForCloudPlan } = require('../../ee/licence/feature-registry');
|
||||||
|
return [...getFeaturesForCloudPlan(plan)];
|
||||||
|
} catch {
|
||||||
|
return [];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return this.getFeatures(licenseKey);
|
||||||
|
}
|
||||||
|
|
||||||
|
resolveTier(licenseKey: string, plan: string): string {
|
||||||
|
if (this.environmentService.isCloud()) {
|
||||||
|
return plan ?? 'standard';
|
||||||
|
}
|
||||||
|
|
||||||
|
return this.getLicenseType(licenseKey) ?? 'free';
|
||||||
|
}
|
||||||
|
|
||||||
|
private getLicenseType(licenseKey: string): string | null {
|
||||||
|
try {
|
||||||
|
// eslint-disable-next-line @typescript-eslint/no-require-imports
|
||||||
|
const LicenseModule = require('../../ee/licence/license.service');
|
||||||
|
const licenseService = this.moduleRef.get(LicenseModule.LicenseService, {
|
||||||
|
strict: false,
|
||||||
|
});
|
||||||
|
return licenseService.getLicenseType(licenseKey);
|
||||||
|
} catch {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -25,6 +25,7 @@ import {
|
|||||||
buildAttachmentCandidates,
|
buildAttachmentCandidates,
|
||||||
collectMarkdownAndHtmlFiles,
|
collectMarkdownAndHtmlFiles,
|
||||||
encodeFilePath,
|
encodeFilePath,
|
||||||
|
extractNotionPartialId,
|
||||||
readDocmostMetadata,
|
readDocmostMetadata,
|
||||||
stripNotionID,
|
stripNotionID,
|
||||||
} from '../utils/import.utils';
|
} from '../utils/import.utils';
|
||||||
@@ -160,6 +161,7 @@ export class FileImportTaskService {
|
|||||||
fileTask: FileTask;
|
fileTask: FileTask;
|
||||||
}): Promise<void> {
|
}): Promise<void> {
|
||||||
const { extractDir, fileTask } = opts;
|
const { extractDir, fileTask } = opts;
|
||||||
|
const isNotion = fileTask.source === FileImportSource.Notion;
|
||||||
const allFiles = await collectMarkdownAndHtmlFiles(extractDir);
|
const allFiles = await collectMarkdownAndHtmlFiles(extractDir);
|
||||||
const attachmentCandidates = await buildAttachmentCandidates(extractDir);
|
const attachmentCandidates = await buildAttachmentCandidates(extractDir);
|
||||||
const docmostMetadata = await readDocmostMetadata(extractDir);
|
const docmostMetadata = await readDocmostMetadata(extractDir);
|
||||||
@@ -230,7 +232,17 @@ export class FileImportTaskService {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// For each folder with content, create a placeholder page if no corresponding .md or .html exists
|
// For each folder with content, create a placeholder page if no corresponding .md or .html exists
|
||||||
foldersWithContent.forEach((folderPath) => {
|
// Process folders with partial UUIDs first so they claim their specific files
|
||||||
|
// before plain folders (without partial UUIDs) take whatever remains.
|
||||||
|
const sortedFolders = isNotion
|
||||||
|
? [...foldersWithContent].sort((a, b) => {
|
||||||
|
const aHasPartial = extractNotionPartialId(path.basename(a)) ? 0 : 1;
|
||||||
|
const bHasPartial = extractNotionPartialId(path.basename(b)) ? 0 : 1;
|
||||||
|
return aHasPartial - bHasPartial;
|
||||||
|
})
|
||||||
|
: [...foldersWithContent];
|
||||||
|
|
||||||
|
sortedFolders.forEach((folderPath) => {
|
||||||
if (
|
if (
|
||||||
skipRootFolder &&
|
skipRootFolder &&
|
||||||
folderPath?.toLowerCase() === skipRootFolder?.toLowerCase()
|
folderPath?.toLowerCase() === skipRootFolder?.toLowerCase()
|
||||||
@@ -243,18 +255,54 @@ export class FileImportTaskService {
|
|||||||
|
|
||||||
if (!pagesMap.has(mdPath) && !pagesMap.has(htmlPath)) {
|
if (!pagesMap.has(mdPath) && !pagesMap.has(htmlPath)) {
|
||||||
const folderName = path.basename(folderPath);
|
const folderName = path.basename(folderPath);
|
||||||
const encodedMdPath = encodeFilePath(mdPath);
|
const parentDir = path.dirname(folderPath);
|
||||||
const placeholderMetadata = docmostMetadata?.pages[encodedMdPath];
|
|
||||||
pagesMap.set(mdPath, {
|
// Notion no longer adds UUIDs to folder names, but still adds them to files.
|
||||||
id: v7(),
|
// For duplicate names, Notion adds a partial UUID "{first4}-{last4}" to the folder.
|
||||||
slugId: generateSlugId(),
|
let matched = false;
|
||||||
name: stripNotionID(folderName),
|
if (isNotion) {
|
||||||
content: '',
|
const partialId = extractNotionPartialId(folderName);
|
||||||
parentPageId: null,
|
const strippedFolderName = stripNotionID(folderName);
|
||||||
fileExtension: '.md',
|
const isSameDir = (fileDir: string) =>
|
||||||
filePath: mdPath,
|
fileDir === parentDir || (parentDir === '.' && !fileDir.includes('/'));
|
||||||
icon: placeholderMetadata?.icon ?? null,
|
|
||||||
});
|
for (const [filePath, page] of pagesMap.entries()) {
|
||||||
|
if (!isSameDir(path.dirname(filePath))) continue;
|
||||||
|
if (page.name !== strippedFolderName) continue;
|
||||||
|
|
||||||
|
if (partialId) {
|
||||||
|
// Match partial UUID against the full UUID in the filename
|
||||||
|
const fileBase = path.basename(filePath, path.extname(filePath));
|
||||||
|
const fullIdMatch = fileBase.match(/[a-f0-9]{32}$/i);
|
||||||
|
if (!fullIdMatch) continue;
|
||||||
|
const fullId = fullIdMatch[0].toLowerCase();
|
||||||
|
if (!fullId.startsWith(partialId.prefix) || !fullId.endsWith(partialId.suffix)) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pagesMap.delete(filePath);
|
||||||
|
page.filePath = mdPath;
|
||||||
|
pagesMap.set(mdPath, page);
|
||||||
|
matched = true;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!matched) {
|
||||||
|
const encodedMdPath = encodeFilePath(mdPath);
|
||||||
|
const placeholderMetadata = docmostMetadata?.pages[encodedMdPath];
|
||||||
|
pagesMap.set(mdPath, {
|
||||||
|
id: v7(),
|
||||||
|
slugId: generateSlugId(),
|
||||||
|
name: stripNotionID(folderName),
|
||||||
|
content: '',
|
||||||
|
parentPageId: null,
|
||||||
|
fileExtension: '.md',
|
||||||
|
filePath: mdPath,
|
||||||
|
icon: placeholderMetadata?.icon ?? null,
|
||||||
|
});
|
||||||
|
}
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|||||||
@@ -1,6 +1,7 @@
|
|||||||
import { getEmbedUrlAndProvider } from '@docmost/editor-ext';
|
import { getEmbedUrlAndProvider } from '@docmost/editor-ext';
|
||||||
import { Logger } from '@nestjs/common';
|
import { Logger } from '@nestjs/common';
|
||||||
import * as path from 'path';
|
import * as path from 'path';
|
||||||
|
import { v7 } from 'uuid';
|
||||||
import { InsertableBacklink } from '@docmost/db/types/entity.types';
|
import { InsertableBacklink } from '@docmost/db/types/entity.types';
|
||||||
import { Cheerio, CheerioAPI, load } from 'cheerio';
|
import { Cheerio, CheerioAPI, load } from 'cheerio';
|
||||||
// eslint-disable-next-line @typescript-eslint/no-require-imports
|
// eslint-disable-next-line @typescript-eslint/no-require-imports
|
||||||
@@ -344,14 +345,35 @@ export async function rewriteInternalLinksToMentionHtml(
|
|||||||
const meta = filePathToPageMetaMap.get(resolved);
|
const meta = filePathToPageMetaMap.get(resolved);
|
||||||
if (!meta) return;
|
if (!meta) return;
|
||||||
|
|
||||||
const titleSlug = slugify(meta.title?.substring(0, 70) || 'untitled');
|
const linkText = $a.text().trim();
|
||||||
const pageSlug = `${titleSlug}-${meta.slugId}`;
|
const titleMatch =
|
||||||
const internalHref = spaceSlug
|
linkText === meta.title ||
|
||||||
? `/s/${spaceSlug}/p/${pageSlug}`
|
linkText === meta.title?.trim();
|
||||||
: `/p/${pageSlug}`;
|
|
||||||
|
|
||||||
$a.attr('href', internalHref);
|
if (titleMatch) {
|
||||||
$a.attr('data-internal', 'true');
|
const mentionId = v7();
|
||||||
|
const $mention = $('<span>')
|
||||||
|
.attr({
|
||||||
|
'data-type': 'mention',
|
||||||
|
'data-id': mentionId,
|
||||||
|
'data-entity-type': 'page',
|
||||||
|
'data-entity-id': meta.id,
|
||||||
|
'data-label': meta.title,
|
||||||
|
'data-slug-id': meta.slugId,
|
||||||
|
'data-creator-id': creatorId,
|
||||||
|
})
|
||||||
|
.text(meta.title);
|
||||||
|
$a.replaceWith($mention);
|
||||||
|
} else {
|
||||||
|
const titleSlug = slugify(meta.title?.substring(0, 70) || 'untitled');
|
||||||
|
const pageSlug = `${titleSlug}-${meta.slugId}`;
|
||||||
|
const internalHref = spaceSlug
|
||||||
|
? `/s/${spaceSlug}/p/${pageSlug}`
|
||||||
|
: `/p/${pageSlug}`;
|
||||||
|
|
||||||
|
$a.attr('href', internalHref);
|
||||||
|
$a.attr('data-internal', 'true');
|
||||||
|
}
|
||||||
|
|
||||||
backlinks.push({ sourcePageId, targetPageId: meta.id, workspaceId });
|
backlinks.push({ sourcePageId, targetPageId: meta.id, workspaceId });
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -81,7 +81,25 @@ export async function collectMarkdownAndHtmlFiles(
|
|||||||
export function stripNotionID(fileName: string): string {
|
export function stripNotionID(fileName: string): string {
|
||||||
// Handle optional separator (space or dash) + 32 alphanumeric chars at end
|
// Handle optional separator (space or dash) + 32 alphanumeric chars at end
|
||||||
const notionIdPattern = /[ -]?[a-z0-9]{32}$/i;
|
const notionIdPattern = /[ -]?[a-z0-9]{32}$/i;
|
||||||
return fileName.replace(notionIdPattern, '').trim();
|
// Handle partial UUID format used for duplicate names: "Name abcd-ef12"
|
||||||
|
const partialIdPattern = / [a-f0-9]{4}-[a-f0-9]{4}$/i;
|
||||||
|
return fileName
|
||||||
|
.replace(notionIdPattern, '')
|
||||||
|
.replace(partialIdPattern, '')
|
||||||
|
.trim();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Extract a partial Notion UUID suffix from a folder name.
|
||||||
|
* Notion adds "{first4}-{last4}" when multiple pages share the same title.
|
||||||
|
* e.g. "Cool 324d-35ab" → { prefix: "324d", suffix: "35ab" }
|
||||||
|
*/
|
||||||
|
export function extractNotionPartialId(
|
||||||
|
folderName: string,
|
||||||
|
): { prefix: string; suffix: string } | null {
|
||||||
|
const match = folderName.match(/ ([a-f0-9]{4})-([a-f0-9]{4})$/i);
|
||||||
|
if (!match) return null;
|
||||||
|
return { prefix: match[1].toLowerCase(), suffix: match[2].toLowerCase() };
|
||||||
}
|
}
|
||||||
|
|
||||||
export function encodeFilePath(filePath: string): string {
|
export function encodeFilePath(filePath: string): string {
|
||||||
|
|||||||
+3
-3
@@ -1,7 +1,7 @@
|
|||||||
{
|
{
|
||||||
"name": "docmost",
|
"name": "docmost",
|
||||||
"homepage": "https://docmost.com",
|
"homepage": "https://docmost.com",
|
||||||
"version": "0.70.1",
|
"version": "0.70.3",
|
||||||
"private": true,
|
"private": true,
|
||||||
"scripts": {
|
"scripts": {
|
||||||
"build": "nx run-many -t build",
|
"build": "nx run-many -t build",
|
||||||
@@ -62,7 +62,7 @@
|
|||||||
"cross-env": "^7.0.3",
|
"cross-env": "^7.0.3",
|
||||||
"date-fns": "^4.1.0",
|
"date-fns": "^4.1.0",
|
||||||
"diff": "8.0.3",
|
"diff": "8.0.3",
|
||||||
"dompurify": "^3.3.1",
|
"dompurify": "^3.3.3",
|
||||||
"fractional-indexing-jittered": "^1.0.0",
|
"fractional-indexing-jittered": "^1.0.0",
|
||||||
"highlight.js": "^11.11.1",
|
"highlight.js": "^11.11.1",
|
||||||
"image-dimensions": "^2.5.0",
|
"image-dimensions": "^2.5.0",
|
||||||
@@ -109,7 +109,7 @@
|
|||||||
"lodash": "4.17.23",
|
"lodash": "4.17.23",
|
||||||
"ws": "8.19.0",
|
"ws": "8.19.0",
|
||||||
"cross-spawn": "7.0.5",
|
"cross-spawn": "7.0.5",
|
||||||
"dompurify": "3.3.1",
|
"dompurify": "3.3.3",
|
||||||
"tmp": "0.2.5",
|
"tmp": "0.2.5",
|
||||||
"lodash-es": "4.17.23",
|
"lodash-es": "4.17.23",
|
||||||
"markdown-it": "14.1.1",
|
"markdown-it": "14.1.1",
|
||||||
|
|||||||
Generated
+8
-8
@@ -14,7 +14,7 @@ overrides:
|
|||||||
lodash: 4.17.23
|
lodash: 4.17.23
|
||||||
ws: 8.19.0
|
ws: 8.19.0
|
||||||
cross-spawn: 7.0.5
|
cross-spawn: 7.0.5
|
||||||
dompurify: 3.3.1
|
dompurify: 3.3.3
|
||||||
tmp: 0.2.5
|
tmp: 0.2.5
|
||||||
lodash-es: 4.17.23
|
lodash-es: 4.17.23
|
||||||
markdown-it: 14.1.1
|
markdown-it: 14.1.1
|
||||||
@@ -170,8 +170,8 @@ importers:
|
|||||||
specifier: 8.0.3
|
specifier: 8.0.3
|
||||||
version: 8.0.3
|
version: 8.0.3
|
||||||
dompurify:
|
dompurify:
|
||||||
specifier: 3.3.1
|
specifier: 3.3.3
|
||||||
version: 3.3.1
|
version: 3.3.3
|
||||||
fractional-indexing-jittered:
|
fractional-indexing-jittered:
|
||||||
specifier: ^1.0.0
|
specifier: ^1.0.0
|
||||||
version: 1.0.0
|
version: 1.0.0
|
||||||
@@ -6510,8 +6510,8 @@ packages:
|
|||||||
resolution: {integrity: sha512-cgwlv/1iFQiFnU96XXgROh8xTeetsnJiDsTc7TYCLFd9+/WNkIqPTxiM/8pSd8VIrhXGTf1Ny1q1hquVqDJB5w==}
|
resolution: {integrity: sha512-cgwlv/1iFQiFnU96XXgROh8xTeetsnJiDsTc7TYCLFd9+/WNkIqPTxiM/8pSd8VIrhXGTf1Ny1q1hquVqDJB5w==}
|
||||||
engines: {node: '>= 4'}
|
engines: {node: '>= 4'}
|
||||||
|
|
||||||
dompurify@3.3.1:
|
dompurify@3.3.3:
|
||||||
resolution: {integrity: sha512-qkdCKzLNtrgPFP1Vo+98FRzJnBRGe4ffyCea9IwHB1fyxPOeNTHpLKYGd4Uk9xvNoH0ZoOjwZxNptyMwqrId1Q==}
|
resolution: {integrity: sha512-Oj6pzI2+RqBfFG+qOaOLbFXLQ90ARpcGG6UePL82bJLtdsa6CYJD7nmiU8MW9nQNOtCHV3lZ/Bzq1X0QYbBZCA==}
|
||||||
|
|
||||||
domutils@3.2.2:
|
domutils@3.2.2:
|
||||||
resolution: {integrity: sha512-6kZKyUajlDuqlHKVX1w7gyslj9MPIXzIFiz/rGu35uC1wMi+kMhQwGhl4lt9unC9Vb9INnY9Z3/ZA3+FhASLaw==}
|
resolution: {integrity: sha512-6kZKyUajlDuqlHKVX1w7gyslj9MPIXzIFiz/rGu35uC1wMi+kMhQwGhl4lt9unC9Vb9INnY9Z3/ZA3+FhASLaw==}
|
||||||
@@ -17205,7 +17205,7 @@ snapshots:
|
|||||||
dependencies:
|
dependencies:
|
||||||
domelementtype: 2.3.0
|
domelementtype: 2.3.0
|
||||||
|
|
||||||
dompurify@3.3.1:
|
dompurify@3.3.3:
|
||||||
optionalDependencies:
|
optionalDependencies:
|
||||||
'@types/trusted-types': 2.0.7
|
'@types/trusted-types': 2.0.7
|
||||||
|
|
||||||
@@ -19251,7 +19251,7 @@ snapshots:
|
|||||||
d3-sankey: 0.12.3
|
d3-sankey: 0.12.3
|
||||||
dagre-d3-es: 7.0.13
|
dagre-d3-es: 7.0.13
|
||||||
dayjs: 1.11.19
|
dayjs: 1.11.19
|
||||||
dompurify: 3.3.1
|
dompurify: 3.3.3
|
||||||
katex: 0.16.27
|
katex: 0.16.27
|
||||||
khroma: 2.1.0
|
khroma: 2.1.0
|
||||||
lodash-es: 4.17.23
|
lodash-es: 4.17.23
|
||||||
@@ -20004,7 +20004,7 @@ snapshots:
|
|||||||
'@posthog/core': 1.22.0
|
'@posthog/core': 1.22.0
|
||||||
'@posthog/types': 1.345.5
|
'@posthog/types': 1.345.5
|
||||||
core-js: 3.43.0
|
core-js: 3.43.0
|
||||||
dompurify: 3.3.1
|
dompurify: 3.3.3
|
||||||
fflate: 0.4.8
|
fflate: 0.4.8
|
||||||
preact: 10.28.3
|
preact: 10.28.3
|
||||||
query-selector-shadow-dom: 1.0.1
|
query-selector-shadow-dom: 1.0.1
|
||||||
|
|||||||
Reference in New Issue
Block a user