@ifc-lite/viewer 1.27.0 → 1.28.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/.turbo/turbo-build.log +35 -42
- package/CHANGELOG.md +74 -0
- package/dist/assets/{basketViewActivator-B3CdrLsb.js → basketViewActivator-Ce38DhXd.js} +8 -8
- package/dist/assets/{bcf-QeHK_Aud.js → bcf-Cv_O3JfD.js} +56 -56
- package/dist/assets/{decode-worker-CgM1iNSK.js → decode-worker-Cjign7Zh.js} +1 -1
- package/dist/assets/{deflate-B-d0SYQM.js → deflate-HbyMq59o.js} +1 -1
- package/dist/assets/drawing-2d-DW98umlt.js +257 -0
- package/dist/assets/e57-source-2wI9jkCA.js +1 -0
- package/dist/assets/{exporters-B4LbZFeT.js → exporters-BuD3XRzB.js} +1309 -1153
- package/dist/assets/geometry.worker-TH3fCCoY.js +1 -0
- package/dist/assets/{geotiff-CrVtDRFq.js → geotiff-B2HA8Bwm.js} +10 -10
- package/dist/assets/{ids-DjsGFN10.js → ids-DYUFMd5f.js} +952 -945
- package/dist/assets/{ifc-lite_bg-DsYUIHm3.wasm → ifc-lite_bg-BEA5DLmg.wasm} +0 -0
- package/dist/assets/index-E9wB0zWt.css +1 -0
- package/dist/assets/{index-COYokSKc.js → index-n5O1QJMM.js} +37877 -38126
- package/dist/assets/{index.es-CY202jA3.js → index.es-BKVIpZgL.js} +9 -9
- package/dist/assets/{jpeg-D4wOkf5h.js → jpeg-C7hjKjPX.js} +1 -1
- package/dist/assets/{jspdf.es.min-DIGb9BHN.js → jspdf.es.min-oWlFc42Y.js} +4 -4
- package/dist/assets/lens-C4p1kQ0p.js +1 -0
- package/dist/assets/{lerc-DmW0_tgf.js → lerc-BfIOGhQz.js} +1 -1
- package/dist/assets/{lzw-oWetY-d6.js → lzw-B0jRuuW5.js} +1 -1
- package/dist/assets/{native-bridge-BX8_tHXE.js → native-bridge-DpB-dtEn.js} +6 -3
- package/dist/assets/{packbits-F8Nkp4NY.js → packbits-DVvBTC39.js} +1 -1
- package/dist/assets/parser.worker-BDsWQ6rc.js +182 -0
- package/dist/assets/{pdf-Dsh3HPZB.js → pdf-dVIqI5ac.js} +10 -10
- package/dist/assets/raw-C0ZJYGmN.js +1 -0
- package/dist/assets/{sandbox-BAC3a-eN.js → sandbox-qpJlrNN0.js} +2962 -2554
- package/dist/assets/server-client-DVZ2huNS.js +719 -0
- package/dist/assets/{webimage-BLV1dgmd.js → webimage-B394g0Tw.js} +1 -1
- package/dist/assets/{xlsx-Bc2HTrjC.js → xlsx-D-oHO76J.js} +8 -8
- package/dist/assets/{zstd-C_1HxVrA.js → zstd-Bf38MwV2.js} +1 -1
- package/dist/index.html +9 -9
- package/package.json +24 -23
- package/src/App.tsx +1 -3
- package/src/components/mcp/playground-dispatcher.ts +3 -0
- package/src/components/mcp/playground-files.ts +33 -1
- package/src/components/viewer/BCFPanel.tsx +1 -16
- package/src/components/viewer/ChatPanel.tsx +11 -46
- package/src/components/viewer/CommandPalette.tsx +6 -1
- package/src/components/viewer/ComparePanel.tsx +420 -0
- package/src/components/viewer/HierarchyPanel.tsx +48 -183
- package/src/components/viewer/IDSPanel.tsx +1 -26
- package/src/components/viewer/MainToolbar.tsx +94 -187
- package/src/components/viewer/MobileToolbar.tsx +1 -9
- package/src/components/viewer/PropertiesPanel.tsx +98 -127
- package/src/components/viewer/ScriptPanel.tsx +8 -34
- package/src/components/viewer/Section2DPanel.tsx +32 -1
- package/src/components/viewer/ViewerLayout.tsx +5 -2
- package/src/components/viewer/Viewport.tsx +3 -0
- package/src/components/viewer/ViewportContainer.tsx +24 -42
- package/src/components/viewer/ViewportOverlays.tsx +1 -4
- package/src/components/viewer/hierarchy/HierarchyNode.tsx +3 -3
- package/src/components/viewer/hierarchy/ifc-icons.ts +9 -0
- package/src/components/viewer/hierarchy/treeDataBuilder.ts +87 -0
- package/src/components/viewer/hierarchy/types.ts +1 -0
- package/src/components/viewer/hierarchy/useHierarchyTree.ts +6 -2
- package/src/components/viewer/properties/MaterialTotalsPanel.tsx +283 -0
- package/src/components/viewer/useGeometryStreaming.ts +0 -2
- package/src/hooks/federationLoadGate.test.ts +12 -2
- package/src/hooks/federationLoadGate.ts +9 -2
- package/src/hooks/ingest/federationAlign.ts +488 -0
- package/src/hooks/ingest/viewerModelIngest.ts +3 -212
- package/src/hooks/useCompare.ts +0 -0
- package/src/hooks/useCompareOverlay.ts +119 -0
- package/src/hooks/useDrawingGeneration.ts +234 -14
- package/src/hooks/useIfc.ts +1 -1
- package/src/hooks/useIfcCache.ts +100 -24
- package/src/hooks/useIfcFederation.ts +42 -811
- package/src/hooks/useIfcLoader.ts +349 -1517
- package/src/hooks/useIfcServer.ts +3 -0
- package/src/hooks/useLens.ts +5 -1
- package/src/hooks/useSymbolicAnnotations.ts +70 -38
- package/src/lib/compare/buildFingerprints.ts +173 -0
- package/src/lib/compare/describeChange.ts +0 -0
- package/src/lib/compare/geometricData.test.ts +54 -0
- package/src/lib/compare/geometricData.ts +37 -0
- package/src/lib/compare/overlay.test.ts +99 -0
- package/src/lib/compare/overlay.ts +91 -0
- package/src/lib/geo/cesium-placement.ts +1 -1
- package/src/lib/geo/reproject.ts +4 -1
- package/src/lib/llm/script-edit-ops.ts +23 -0
- package/src/lib/llm/stream-client.ts +8 -1
- package/src/lib/search/result-export.ts +7 -1
- package/src/sdk/adapters/export-adapter.ts +6 -1
- package/src/services/cacheService.ts +9 -25
- package/src/services/desktop-export.ts +2 -59
- package/src/services/file-dialog.ts +8 -142
- package/src/store/constants.ts +23 -0
- package/src/store/globalId.ts +15 -13
- package/src/store/index.ts +19 -6
- package/src/store/slices/cesiumSlice.ts +8 -1
- package/src/store/slices/compareSlice.ts +96 -0
- package/src/store/slices/drawing2DSlice.ts +8 -0
- package/src/store/slices/lensSlice.ts +8 -0
- package/src/store/slices/visibilitySlice.ts +22 -1
- package/src/store/types.ts +1 -71
- package/src/utils/acquireFileBuffer.test.ts +12 -4
- package/src/utils/ifcConfig.ts +0 -12
- package/src/utils/loadingUtils.ts +32 -0
- package/src/utils/spatialHierarchy.test.ts +53 -1
- package/src/utils/spatialHierarchy.ts +42 -2
- package/src/vite-env.d.ts +2 -0
- package/vite.config.ts +6 -3
- package/DESKTOP_CONTRACT_VERSION +0 -1
- package/dist/assets/drawing-2d-C71b8Ugx.js +0 -257
- package/dist/assets/e57-source-CQHxE8n3.js +0 -1
- package/dist/assets/event-B0kAzHa-.js +0 -1
- package/dist/assets/geometry.worker-BdH-E6NB.js +0 -1
- package/dist/assets/index-ajK6D32J.css +0 -1
- package/dist/assets/lens-PYsLu_MA.js +0 -1
- package/dist/assets/parser.worker-D591Zu_-.js +0 -182
- package/dist/assets/raw-D9iw0tmc.js +0 -1
- package/dist/assets/server-client-Cjwnm7il.js +0 -706
- package/dist/assets/tauri-core-stub-D8Fa-u43.js +0 -1
- package/dist/assets/tauri-dialog-stub-r7Wksg7o.js +0 -1
- package/dist/assets/tauri-fs-stub-BdeRC7aK.js +0 -1
- package/src/components/viewer/DesktopEntitlementBanner.tsx +0 -74
- package/src/components/viewer/SettingsPage.tsx +0 -581
- package/src/hooks/ingest/resolveDataStoreOrAbort.test.ts +0 -61
- package/src/hooks/ingest/resolveDataStoreOrAbort.ts +0 -28
- package/src/hooks/ingest/watchedGeometryStream.test.ts +0 -78
- package/src/hooks/ingest/watchedGeometryStream.ts +0 -76
- package/src/lib/desktop/desktopEntitlementEvents.ts +0 -39
- package/src/lib/desktop-entitlement.ts +0 -43
- package/src/lib/desktop-product.ts +0 -130
- package/src/lib/platform.ts +0 -23
- package/src/services/desktop-cache.ts +0 -186
- package/src/services/desktop-harness.ts +0 -196
- package/src/services/desktop-logger.ts +0 -20
- package/src/services/desktop-native-metadata.ts +0 -230
- package/src/services/desktop-panel-actions.ts +0 -43
- package/src/services/desktop-preferences.ts +0 -44
- package/src/services/fs-cache.ts +0 -212
- package/src/services/tauri-core-stub.ts +0 -7
- package/src/services/tauri-dialog-stub.ts +0 -7
- package/src/services/tauri-fs-stub.ts +0 -7
- package/src/services/tauri-modules.d.ts +0 -50
- package/src/store/slices/desktopEntitlementSlice.ts +0 -86
- package/src/utils/desktopModelSnapshot.ts +0 -358
- package/src/utils/nativeSpatialDataStore.ts +0 -277
- package/src-tauri/Cargo.toml +0 -29
- package/src-tauri/build.rs +0 -7
- package/src-tauri/capabilities/default.json +0 -18
- package/src-tauri/icons/128x128.png +0 -0
- package/src-tauri/icons/128x128@2x.png +0 -0
- package/src-tauri/icons/32x32.png +0 -0
- package/src-tauri/icons/Square107x107Logo.png +0 -0
- package/src-tauri/icons/Square142x142Logo.png +0 -0
- package/src-tauri/icons/Square150x150Logo.png +0 -0
- package/src-tauri/icons/Square284x284Logo.png +0 -0
- package/src-tauri/icons/Square30x30Logo.png +0 -0
- package/src-tauri/icons/Square310x310Logo.png +0 -0
- package/src-tauri/icons/Square44x44Logo.png +0 -0
- package/src-tauri/icons/Square71x71Logo.png +0 -0
- package/src-tauri/icons/Square89x89Logo.png +0 -0
- package/src-tauri/icons/StoreLogo.png +0 -0
- package/src-tauri/icons/icon.icns +0 -0
- package/src-tauri/icons/icon.ico +0 -0
- package/src-tauri/icons/icon.png +0 -0
- package/src-tauri/src/lib.rs +0 -21
- package/src-tauri/src/main.rs +0 -10
- package/src-tauri/tauri.conf.json +0 -39
|
@@ -1 +0,0 @@
|
|
|
1
|
-
async function r(){throw new Error("Tauri core API is unavailable in the browser build")}export{r as invoke};
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
async function n(){return null}export{n as open};
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
async function r(e){throw new Error("Tauri file system API is unavailable in the browser benchmark build")}export{r as readFile};
|
|
@@ -1,74 +0,0 @@
|
|
|
1
|
-
/* This Source Code Form is subject to the terms of the Mozilla Public
|
|
2
|
-
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
3
|
-
* file, You can obtain one at https://mozilla.org/MPL/2.0/. */
|
|
4
|
-
|
|
5
|
-
import { AlertTriangle, Clock3, WifiOff } from 'lucide-react';
|
|
6
|
-
import { Button } from '@/components/ui/button';
|
|
7
|
-
import { useViewerStore } from '@/store';
|
|
8
|
-
import { buildDesktopUpgradeUrl, hasDesktopPro } from '@/lib/desktop-product';
|
|
9
|
-
import { navigateToPath } from '@/services/app-navigation';
|
|
10
|
-
|
|
11
|
-
function formatDate(timestamp: number | null): string | null {
|
|
12
|
-
if (!timestamp || !Number.isFinite(timestamp)) {
|
|
13
|
-
return null;
|
|
14
|
-
}
|
|
15
|
-
return new Date(timestamp).toLocaleDateString();
|
|
16
|
-
}
|
|
17
|
-
|
|
18
|
-
export function DesktopEntitlementBanner() {
|
|
19
|
-
const entitlement = useViewerStore((s) => s.desktopEntitlement);
|
|
20
|
-
|
|
21
|
-
if (entitlement.status === 'active' || entitlement.status === 'anonymous') {
|
|
22
|
-
return null;
|
|
23
|
-
}
|
|
24
|
-
|
|
25
|
-
if (entitlement.status === 'trial') {
|
|
26
|
-
return (
|
|
27
|
-
<div className="flex items-center justify-between gap-3 border-b bg-amber-50 px-4 py-2 text-sm text-amber-900 dark:bg-amber-950/60 dark:text-amber-200">
|
|
28
|
-
<div className="flex items-center gap-2 min-w-0">
|
|
29
|
-
<Clock3 className="h-4 w-4 shrink-0" />
|
|
30
|
-
<span className="truncate">
|
|
31
|
-
Desktop Pro trial active{formatDate(entitlement.trialEndsAt) ? ` until ${formatDate(entitlement.trialEndsAt)}` : ''}.
|
|
32
|
-
</span>
|
|
33
|
-
</div>
|
|
34
|
-
<Button variant="outline" size="sm" onClick={() => navigateToPath('/settings')}>
|
|
35
|
-
View Plan
|
|
36
|
-
</Button>
|
|
37
|
-
</div>
|
|
38
|
-
);
|
|
39
|
-
}
|
|
40
|
-
|
|
41
|
-
if (entitlement.status === 'grace_offline') {
|
|
42
|
-
return (
|
|
43
|
-
<div className="flex items-center justify-between gap-3 border-b bg-blue-50 px-4 py-2 text-sm text-blue-900 dark:bg-blue-950/60 dark:text-blue-200">
|
|
44
|
-
<div className="flex items-center gap-2 min-w-0">
|
|
45
|
-
<WifiOff className="h-4 w-4 shrink-0" />
|
|
46
|
-
<span className="truncate">
|
|
47
|
-
Offline grace active{formatDate(entitlement.graceUntil) ? ` until ${formatDate(entitlement.graceUntil)}` : ''}. Pro features remain available from the last validated plan.
|
|
48
|
-
</span>
|
|
49
|
-
</div>
|
|
50
|
-
<Button variant="outline" size="sm" onClick={() => navigateToPath('/settings')}>
|
|
51
|
-
View Plan
|
|
52
|
-
</Button>
|
|
53
|
-
</div>
|
|
54
|
-
);
|
|
55
|
-
}
|
|
56
|
-
|
|
57
|
-
if (!hasDesktopPro(entitlement)) {
|
|
58
|
-
return (
|
|
59
|
-
<div className="flex items-center justify-between gap-3 border-b bg-muted/60 px-4 py-2 text-sm text-foreground">
|
|
60
|
-
<div className="flex items-center gap-2 min-w-0">
|
|
61
|
-
<AlertTriangle className="h-4 w-4 shrink-0 text-amber-600 dark:text-amber-400" />
|
|
62
|
-
<span className="truncate">
|
|
63
|
-
Desktop Pro is not active. Core viewing stays available; advanced features are locked.
|
|
64
|
-
</span>
|
|
65
|
-
</div>
|
|
66
|
-
<Button variant="default" size="sm" onClick={() => navigateToPath(buildDesktopUpgradeUrl())}>
|
|
67
|
-
Upgrade
|
|
68
|
-
</Button>
|
|
69
|
-
</div>
|
|
70
|
-
);
|
|
71
|
-
}
|
|
72
|
-
|
|
73
|
-
return null;
|
|
74
|
-
}
|
|
@@ -1,581 +0,0 @@
|
|
|
1
|
-
/* This Source Code Form is subject to the terms of the Mozilla Public
|
|
2
|
-
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
3
|
-
* file, You can obtain one at https://mozilla.org/MPL/2.0/. */
|
|
4
|
-
|
|
5
|
-
import {
|
|
6
|
-
ArrowLeft,
|
|
7
|
-
Bot,
|
|
8
|
-
Check,
|
|
9
|
-
ChevronDown,
|
|
10
|
-
ChevronUp,
|
|
11
|
-
Clock3,
|
|
12
|
-
Cloud,
|
|
13
|
-
CreditCard,
|
|
14
|
-
ExternalLink,
|
|
15
|
-
Eye,
|
|
16
|
-
EyeOff,
|
|
17
|
-
FolderOpen,
|
|
18
|
-
Key,
|
|
19
|
-
LayoutPanelTop,
|
|
20
|
-
Lock,
|
|
21
|
-
Settings2,
|
|
22
|
-
ShieldCheck,
|
|
23
|
-
Trash2,
|
|
24
|
-
WifiOff,
|
|
25
|
-
} from 'lucide-react';
|
|
26
|
-
import { useCallback, useEffect, useMemo, useState } from 'react';
|
|
27
|
-
import { Badge } from '@/components/ui/badge';
|
|
28
|
-
import { Button } from '@/components/ui/button';
|
|
29
|
-
import { Switch } from '@/components/ui/switch';
|
|
30
|
-
import { toast } from '@/components/ui/toast';
|
|
31
|
-
import { useViewerStore } from '@/store';
|
|
32
|
-
import {
|
|
33
|
-
buildDesktopUpgradeUrl,
|
|
34
|
-
getDesktopFeatureCatalog,
|
|
35
|
-
getDesktopPlanSummary,
|
|
36
|
-
getDesktopPlanTier,
|
|
37
|
-
hasDesktopPro,
|
|
38
|
-
isDesktopBillingEnforced,
|
|
39
|
-
type DesktopEntitlement,
|
|
40
|
-
} from '@/lib/desktop-product';
|
|
41
|
-
import { navigateToPath } from '@/services/app-navigation';
|
|
42
|
-
import {
|
|
43
|
-
getDesktopPreferences,
|
|
44
|
-
subscribeDesktopPreferences,
|
|
45
|
-
updateDesktopPreferences,
|
|
46
|
-
} from '@/services/desktop-preferences';
|
|
47
|
-
import {
|
|
48
|
-
getApiKeys,
|
|
49
|
-
updateApiKeys,
|
|
50
|
-
clearApiKeys,
|
|
51
|
-
subscribeApiKeys,
|
|
52
|
-
type ApiKeyConfig,
|
|
53
|
-
} from '@/services/api-keys';
|
|
54
|
-
|
|
55
|
-
export function SettingsPage() {
|
|
56
|
-
const desktopEntitlement = useViewerStore((s) => s.desktopEntitlement);
|
|
57
|
-
const chatUsage = useViewerStore((s) => s.chatUsage);
|
|
58
|
-
const [preferences, setPreferences] = useState(() => getDesktopPreferences());
|
|
59
|
-
const [apiKeys, setApiKeys] = useState(() => getApiKeys());
|
|
60
|
-
const returnTo = (() => {
|
|
61
|
-
const params = new URLSearchParams(window.location.search);
|
|
62
|
-
const candidate = params.get('returnTo');
|
|
63
|
-
return candidate && candidate.startsWith('/') ? candidate : '/';
|
|
64
|
-
})();
|
|
65
|
-
useEffect(() => subscribeDesktopPreferences(() => {
|
|
66
|
-
setPreferences(getDesktopPreferences());
|
|
67
|
-
}), []);
|
|
68
|
-
useEffect(() => subscribeApiKeys(() => {
|
|
69
|
-
setApiKeys(getApiKeys());
|
|
70
|
-
}), []);
|
|
71
|
-
|
|
72
|
-
const updatePreference = (updates: Partial<typeof preferences>) => {
|
|
73
|
-
setPreferences(updateDesktopPreferences(updates));
|
|
74
|
-
};
|
|
75
|
-
const planTier = getDesktopPlanTier(desktopEntitlement);
|
|
76
|
-
const planSummary = getDesktopPlanSummary(desktopEntitlement, chatUsage);
|
|
77
|
-
const featureCatalog = getDesktopFeatureCatalog(desktopEntitlement);
|
|
78
|
-
const canRestoreWorkspace = hasDesktopPro(desktopEntitlement);
|
|
79
|
-
const usageSummary = useMemo(() => {
|
|
80
|
-
if (!chatUsage) {
|
|
81
|
-
return null;
|
|
82
|
-
}
|
|
83
|
-
const resetLabel = chatUsage.resetAt
|
|
84
|
-
? new Date(chatUsage.resetAt * 1000).toLocaleDateString()
|
|
85
|
-
: 'Unknown';
|
|
86
|
-
const unit = chatUsage.type === 'credits' ? 'credits' : 'requests';
|
|
87
|
-
return `${chatUsage.used}/${chatUsage.limit} ${unit} used. Resets ${resetLabel}.`;
|
|
88
|
-
}, [chatUsage]);
|
|
89
|
-
|
|
90
|
-
return (
|
|
91
|
-
<div className="min-h-screen bg-background text-foreground">
|
|
92
|
-
<div className="mx-auto w-full max-w-4xl px-6 py-8">
|
|
93
|
-
<div className="mb-6 flex items-center justify-between">
|
|
94
|
-
<Button variant="ghost" size="sm" onClick={() => navigateToPath(returnTo, { replace: true })}>
|
|
95
|
-
<ArrowLeft className="mr-2 h-4 w-4" />
|
|
96
|
-
Back to Viewer
|
|
97
|
-
</Button>
|
|
98
|
-
</div>
|
|
99
|
-
|
|
100
|
-
<div className="space-y-6">
|
|
101
|
-
{/* API Keys Section */}
|
|
102
|
-
<ApiKeysSection apiKeys={apiKeys} />
|
|
103
|
-
|
|
104
|
-
<section className="rounded-lg border bg-card p-6 shadow-sm">
|
|
105
|
-
<div className="mb-5 flex items-center gap-3">
|
|
106
|
-
<ShieldCheck className="h-5 w-5" />
|
|
107
|
-
<div>
|
|
108
|
-
<h1 className="text-2xl font-semibold">Desktop Account</h1>
|
|
109
|
-
<p className="text-sm text-muted-foreground">
|
|
110
|
-
App-wide entitlement, trial state, offline grace, and AI usage limits.
|
|
111
|
-
</p>
|
|
112
|
-
</div>
|
|
113
|
-
</div>
|
|
114
|
-
|
|
115
|
-
<div className="mb-5 flex flex-wrap items-center gap-2">
|
|
116
|
-
<StatusBadge entitlement={desktopEntitlement} />
|
|
117
|
-
<Badge variant={hasDesktopPro(desktopEntitlement) ? 'default' : 'secondary'}>
|
|
118
|
-
{planTier === 'pro' ? 'Desktop Pro' : 'Desktop Free'}
|
|
119
|
-
</Badge>
|
|
120
|
-
<Badge variant="outline">
|
|
121
|
-
Source: {desktopEntitlement.source.replace('_', ' ')}
|
|
122
|
-
</Badge>
|
|
123
|
-
</div>
|
|
124
|
-
|
|
125
|
-
<div className="mb-5 grid gap-3 md:grid-cols-2">
|
|
126
|
-
<InfoCard
|
|
127
|
-
title="Plan Summary"
|
|
128
|
-
body={planSummary}
|
|
129
|
-
icon={<CreditCard className="h-4 w-4" />}
|
|
130
|
-
/>
|
|
131
|
-
<InfoCard
|
|
132
|
-
title="AI Usage (Free Tier)"
|
|
133
|
-
body={usageSummary ?? 'No AI usage data yet. Usage appears after the first chat message through the proxy.'}
|
|
134
|
-
icon={<Bot className="h-4 w-4" />}
|
|
135
|
-
/>
|
|
136
|
-
<InfoCard
|
|
137
|
-
title="Last Validated"
|
|
138
|
-
body={formatTimestamp(desktopEntitlement.validatedAt)}
|
|
139
|
-
icon={<Clock3 className="h-4 w-4" />}
|
|
140
|
-
/>
|
|
141
|
-
<InfoCard
|
|
142
|
-
title="Offline Grace"
|
|
143
|
-
body={formatOfflineGrace(desktopEntitlement)}
|
|
144
|
-
icon={<WifiOff className="h-4 w-4" />}
|
|
145
|
-
/>
|
|
146
|
-
</div>
|
|
147
|
-
</section>
|
|
148
|
-
|
|
149
|
-
<section className="rounded-lg border bg-card p-6 shadow-sm">
|
|
150
|
-
<div className="mb-5 flex items-center gap-3">
|
|
151
|
-
<Settings2 className="h-5 w-5" />
|
|
152
|
-
<div>
|
|
153
|
-
<h1 className="text-2xl font-semibold">Desktop Settings</h1>
|
|
154
|
-
<p className="text-sm text-muted-foreground">
|
|
155
|
-
Local preferences for startup behavior and account access.
|
|
156
|
-
</p>
|
|
157
|
-
</div>
|
|
158
|
-
</div>
|
|
159
|
-
|
|
160
|
-
<div className="space-y-5">
|
|
161
|
-
<div className="flex items-start justify-between gap-4 rounded-md border p-4">
|
|
162
|
-
<div className="space-y-1">
|
|
163
|
-
<div className="flex items-center gap-2 font-medium">
|
|
164
|
-
<FolderOpen className="h-4 w-4" />
|
|
165
|
-
Reopen last model on launch
|
|
166
|
-
</div>
|
|
167
|
-
<p className="text-sm text-muted-foreground">
|
|
168
|
-
Automatically load the most recently used IFC file when the desktop app starts.
|
|
169
|
-
</p>
|
|
170
|
-
</div>
|
|
171
|
-
<Switch
|
|
172
|
-
checked={preferences.reopenLastModelOnLaunch}
|
|
173
|
-
onCheckedChange={(checked) => updatePreference({ reopenLastModelOnLaunch: checked })}
|
|
174
|
-
/>
|
|
175
|
-
</div>
|
|
176
|
-
|
|
177
|
-
<div className="flex items-start justify-between gap-4 rounded-md border p-4">
|
|
178
|
-
<div className="space-y-1">
|
|
179
|
-
<div className="flex items-center gap-2 font-medium">
|
|
180
|
-
<LayoutPanelTop className="h-4 w-4" />
|
|
181
|
-
Restore workspace layout
|
|
182
|
-
</div>
|
|
183
|
-
<p className="text-sm text-muted-foreground">
|
|
184
|
-
Restore panel visibility, camera view, sectioning, and other saved workspace state on launch. Desktop Pro feature.
|
|
185
|
-
</p>
|
|
186
|
-
</div>
|
|
187
|
-
<Switch
|
|
188
|
-
checked={preferences.restoreWorkspaceLayoutOnLaunch}
|
|
189
|
-
disabled={!canRestoreWorkspace}
|
|
190
|
-
onCheckedChange={(checked) => updatePreference({ restoreWorkspaceLayoutOnLaunch: checked })}
|
|
191
|
-
/>
|
|
192
|
-
</div>
|
|
193
|
-
{!canRestoreWorkspace && (
|
|
194
|
-
<div className="rounded-md border border-dashed p-4 text-sm text-muted-foreground">
|
|
195
|
-
Workspace restore is included with Desktop Pro. Reopening the last model remains available on Free.
|
|
196
|
-
<div className="mt-3">
|
|
197
|
-
<Button size="sm" onClick={() => navigateToPath(buildDesktopUpgradeUrl('/settings'))}>
|
|
198
|
-
Upgrade to Desktop Pro
|
|
199
|
-
</Button>
|
|
200
|
-
</div>
|
|
201
|
-
</div>
|
|
202
|
-
)}
|
|
203
|
-
</div>
|
|
204
|
-
</section>
|
|
205
|
-
|
|
206
|
-
<section className="rounded-lg border bg-card p-6 shadow-sm">
|
|
207
|
-
<div className="mb-5 flex items-center gap-3">
|
|
208
|
-
<CreditCard className="h-5 w-5" />
|
|
209
|
-
<div>
|
|
210
|
-
<h2 className="text-xl font-semibold">Billing & Features</h2>
|
|
211
|
-
<p className="text-sm text-muted-foreground">
|
|
212
|
-
Desktop billing is app-wide. The viewer stays available on Free, while Pro unlocks advanced desktop features.
|
|
213
|
-
</p>
|
|
214
|
-
</div>
|
|
215
|
-
</div>
|
|
216
|
-
|
|
217
|
-
<div className="mb-5 rounded-md border p-4">
|
|
218
|
-
<div className="flex items-center justify-between gap-4">
|
|
219
|
-
<div>
|
|
220
|
-
<div className="font-medium capitalize">{planTier} plan</div>
|
|
221
|
-
<p className="text-sm text-muted-foreground">{planSummary}</p>
|
|
222
|
-
</div>
|
|
223
|
-
{isDesktopBillingEnforced() && !hasDesktopPro(desktopEntitlement) && (
|
|
224
|
-
<Button onClick={() => navigateToPath(buildDesktopUpgradeUrl('/settings'))}>
|
|
225
|
-
Upgrade to Pro
|
|
226
|
-
</Button>
|
|
227
|
-
)}
|
|
228
|
-
</div>
|
|
229
|
-
</div>
|
|
230
|
-
|
|
231
|
-
<div className="mb-5 grid gap-3">
|
|
232
|
-
{featureCatalog.map((feature) => (
|
|
233
|
-
<div key={feature.key} className="flex items-start justify-between gap-4 rounded-md border p-4">
|
|
234
|
-
<div>
|
|
235
|
-
<div className="font-medium">{feature.label}</div>
|
|
236
|
-
<p className="text-sm text-muted-foreground">{feature.description}</p>
|
|
237
|
-
</div>
|
|
238
|
-
<div className="flex items-center gap-2 text-sm">
|
|
239
|
-
{feature.enabled ? (
|
|
240
|
-
<>
|
|
241
|
-
<Check className="h-4 w-4 text-emerald-500" />
|
|
242
|
-
Included
|
|
243
|
-
</>
|
|
244
|
-
) : (
|
|
245
|
-
<>
|
|
246
|
-
<Lock className="h-4 w-4 text-amber-500" />
|
|
247
|
-
Pro
|
|
248
|
-
</>
|
|
249
|
-
)}
|
|
250
|
-
</div>
|
|
251
|
-
</div>
|
|
252
|
-
))}
|
|
253
|
-
</div>
|
|
254
|
-
</section>
|
|
255
|
-
|
|
256
|
-
<section className="rounded-lg border bg-card p-6 shadow-sm">
|
|
257
|
-
<div className="mb-5 flex items-center gap-3">
|
|
258
|
-
<Cloud className="h-5 w-5" />
|
|
259
|
-
<div>
|
|
260
|
-
<h2 className="text-xl font-semibold">Privacy & Network</h2>
|
|
261
|
-
<p className="text-sm text-muted-foreground">
|
|
262
|
-
Local IFC viewing remains available offline. Connected services degrade individually instead of blocking the desktop viewer.
|
|
263
|
-
</p>
|
|
264
|
-
</div>
|
|
265
|
-
</div>
|
|
266
|
-
|
|
267
|
-
<div className="grid gap-3 md:grid-cols-2">
|
|
268
|
-
<InfoCard
|
|
269
|
-
title="Always Local"
|
|
270
|
-
body="Model loading, hierarchy, properties, navigation, measurement, and core viewing stay on your machine."
|
|
271
|
-
icon={<ShieldCheck className="h-4 w-4" />}
|
|
272
|
-
/>
|
|
273
|
-
<InfoCard
|
|
274
|
-
title="Needs Network"
|
|
275
|
-
body="AI assistant (free via proxy, or direct with your API key), live bSDD lookups, and billing sync require network access."
|
|
276
|
-
icon={<Cloud className="h-4 w-4" />}
|
|
277
|
-
/>
|
|
278
|
-
</div>
|
|
279
|
-
</section>
|
|
280
|
-
</div>
|
|
281
|
-
</div>
|
|
282
|
-
</div>
|
|
283
|
-
);
|
|
284
|
-
}
|
|
285
|
-
|
|
286
|
-
// ── API Keys Section ──────────────────────────────────────────────────────
|
|
287
|
-
|
|
288
|
-
function ApiKeyInput({
|
|
289
|
-
id,
|
|
290
|
-
value,
|
|
291
|
-
onChange,
|
|
292
|
-
onSave,
|
|
293
|
-
placeholder,
|
|
294
|
-
isSaved,
|
|
295
|
-
}: {
|
|
296
|
-
id: string;
|
|
297
|
-
value: string;
|
|
298
|
-
onChange: (v: string) => void;
|
|
299
|
-
onSave: () => void;
|
|
300
|
-
placeholder: string;
|
|
301
|
-
isSaved: boolean;
|
|
302
|
-
}) {
|
|
303
|
-
const [show, setShow] = useState(false);
|
|
304
|
-
return (
|
|
305
|
-
<div className="flex gap-2">
|
|
306
|
-
<div className="relative flex-1">
|
|
307
|
-
<input
|
|
308
|
-
id={id}
|
|
309
|
-
type={show ? 'text' : 'password'}
|
|
310
|
-
value={value}
|
|
311
|
-
onChange={(e) => onChange(e.target.value)}
|
|
312
|
-
onKeyDown={(e) => { if (e.key === 'Enter' && !isSaved) onSave(); }}
|
|
313
|
-
placeholder={placeholder}
|
|
314
|
-
autoComplete="off"
|
|
315
|
-
spellCheck={false}
|
|
316
|
-
className="w-full rounded-md border border-input bg-background px-3 py-1.5 text-sm font-mono focus:outline-none focus:ring-1 focus:ring-ring pr-8"
|
|
317
|
-
/>
|
|
318
|
-
<button
|
|
319
|
-
type="button"
|
|
320
|
-
onClick={() => setShow(!show)}
|
|
321
|
-
className="absolute right-2 top-1/2 -translate-y-1/2 text-muted-foreground hover:text-foreground"
|
|
322
|
-
aria-label={show ? 'Hide key' : 'Show key'}
|
|
323
|
-
>
|
|
324
|
-
{show ? <EyeOff className="h-3.5 w-3.5" /> : <Eye className="h-3.5 w-3.5" />}
|
|
325
|
-
</button>
|
|
326
|
-
</div>
|
|
327
|
-
<Button size="sm" onClick={onSave} disabled={isSaved}>
|
|
328
|
-
Save
|
|
329
|
-
</Button>
|
|
330
|
-
</div>
|
|
331
|
-
);
|
|
332
|
-
}
|
|
333
|
-
|
|
334
|
-
function ApiKeysSection({ apiKeys }: { apiKeys: ApiKeyConfig }) {
|
|
335
|
-
const [anthropicKey, setAnthropicKey] = useState(apiKeys.anthropicKey);
|
|
336
|
-
const [openaiKey, setOpenaiKey] = useState(apiKeys.openaiKey);
|
|
337
|
-
const [showHowTo, setShowHowTo] = useState<'anthropic' | 'openai' | null>(null);
|
|
338
|
-
|
|
339
|
-
useEffect(() => {
|
|
340
|
-
setAnthropicKey(apiKeys.anthropicKey);
|
|
341
|
-
setOpenaiKey(apiKeys.openaiKey);
|
|
342
|
-
}, [apiKeys.anthropicKey, apiKeys.openaiKey]);
|
|
343
|
-
|
|
344
|
-
const saveAnthropicKey = useCallback(() => {
|
|
345
|
-
updateApiKeys({ anthropicKey: anthropicKey.trim() });
|
|
346
|
-
toast.success(anthropicKey.trim() ? 'Anthropic API key saved' : 'Anthropic API key removed');
|
|
347
|
-
}, [anthropicKey]);
|
|
348
|
-
|
|
349
|
-
const saveOpenaiKey = useCallback(() => {
|
|
350
|
-
updateApiKeys({ openaiKey: openaiKey.trim() });
|
|
351
|
-
toast.success(openaiKey.trim() ? 'OpenAI API key saved' : 'OpenAI API key removed');
|
|
352
|
-
}, [openaiKey]);
|
|
353
|
-
|
|
354
|
-
const handleClearAll = useCallback(() => {
|
|
355
|
-
clearApiKeys();
|
|
356
|
-
setAnthropicKey('');
|
|
357
|
-
setOpenaiKey('');
|
|
358
|
-
toast.success('All API keys removed');
|
|
359
|
-
}, []);
|
|
360
|
-
|
|
361
|
-
const hasAnyKey = apiKeys.anthropicKey.length > 0 || apiKeys.openaiKey.length > 0;
|
|
362
|
-
|
|
363
|
-
return (
|
|
364
|
-
<section className="rounded-lg border bg-card p-6 shadow-sm">
|
|
365
|
-
<div className="mb-5 flex items-center gap-3">
|
|
366
|
-
<Key className="h-5 w-5" />
|
|
367
|
-
<div>
|
|
368
|
-
<h1 className="text-2xl font-semibold">API Keys</h1>
|
|
369
|
-
<p className="text-sm text-muted-foreground">
|
|
370
|
-
Bring your own Anthropic or OpenAI API key to unlock additional models.
|
|
371
|
-
You can configure one or both providers independently.
|
|
372
|
-
</p>
|
|
373
|
-
</div>
|
|
374
|
-
</div>
|
|
375
|
-
|
|
376
|
-
<div className="space-y-4">
|
|
377
|
-
{/* Anthropic */}
|
|
378
|
-
<div className="rounded-md border p-4">
|
|
379
|
-
<div className="mb-2 flex items-center justify-between">
|
|
380
|
-
<label className="text-sm font-medium" htmlFor="anthropic-key">Anthropic API Key</label>
|
|
381
|
-
{apiKeys.anthropicKey ? (
|
|
382
|
-
<Badge variant="default" className="text-[10px]">
|
|
383
|
-
<Check className="mr-1 h-3 w-3" />
|
|
384
|
-
Configured
|
|
385
|
-
</Badge>
|
|
386
|
-
) : (
|
|
387
|
-
<Badge variant="outline" className="text-[10px]">Not set</Badge>
|
|
388
|
-
)}
|
|
389
|
-
</div>
|
|
390
|
-
<p className="mb-2 text-xs text-muted-foreground">
|
|
391
|
-
Unlocks <strong>Claude Opus 4.6</strong>, <strong>Claude Sonnet 4.6</strong>, and <strong>Claude Haiku 4.5</strong>.
|
|
392
|
-
</p>
|
|
393
|
-
<button
|
|
394
|
-
type="button"
|
|
395
|
-
onClick={() => setShowHowTo(showHowTo === 'anthropic' ? null : 'anthropic')}
|
|
396
|
-
className="mb-3 flex items-center gap-1 text-xs text-muted-foreground hover:text-foreground transition-colors"
|
|
397
|
-
>
|
|
398
|
-
{showHowTo === 'anthropic' ? <ChevronUp className="h-3 w-3" /> : <ChevronDown className="h-3 w-3" />}
|
|
399
|
-
How to get an Anthropic API key
|
|
400
|
-
</button>
|
|
401
|
-
{showHowTo === 'anthropic' && (
|
|
402
|
-
<div className="mb-3 rounded-md bg-muted/50 p-3 text-xs text-muted-foreground space-y-1.5">
|
|
403
|
-
<p>1. Go to{' '}
|
|
404
|
-
<a href="https://console.anthropic.com/settings/keys" target="_blank" rel="noopener noreferrer" className="underline inline-flex items-center gap-0.5">
|
|
405
|
-
console.anthropic.com/settings/keys <ExternalLink className="h-2.5 w-2.5" />
|
|
406
|
-
</a>
|
|
407
|
-
</p>
|
|
408
|
-
<p>2. Sign in or create an Anthropic account</p>
|
|
409
|
-
<p>3. Click <strong>Create Key</strong>, name it (e.g. "ifc-lite")</p>
|
|
410
|
-
<p>4. Copy the key (starts with <code className="bg-muted px-1 rounded">sk-ant-api03-...</code>)</p>
|
|
411
|
-
<p>5. Paste it below and click Save</p>
|
|
412
|
-
<p className="pt-1 text-muted-foreground/70">Anthropic offers $5 free credit on new accounts. After that, usage is pay-as-you-go on your Anthropic billing.</p>
|
|
413
|
-
</div>
|
|
414
|
-
)}
|
|
415
|
-
<ApiKeyInput
|
|
416
|
-
id="anthropic-key"
|
|
417
|
-
value={anthropicKey}
|
|
418
|
-
onChange={setAnthropicKey}
|
|
419
|
-
onSave={saveAnthropicKey}
|
|
420
|
-
placeholder="sk-ant-api03-..."
|
|
421
|
-
isSaved={anthropicKey.trim() === apiKeys.anthropicKey}
|
|
422
|
-
/>
|
|
423
|
-
</div>
|
|
424
|
-
|
|
425
|
-
{/* OpenAI */}
|
|
426
|
-
<div className="rounded-md border p-4">
|
|
427
|
-
<div className="mb-2 flex items-center justify-between">
|
|
428
|
-
<label className="text-sm font-medium" htmlFor="openai-key">OpenAI API Key</label>
|
|
429
|
-
{apiKeys.openaiKey ? (
|
|
430
|
-
<Badge variant="default" className="text-[10px]">
|
|
431
|
-
<Check className="mr-1 h-3 w-3" />
|
|
432
|
-
Configured
|
|
433
|
-
</Badge>
|
|
434
|
-
) : (
|
|
435
|
-
<Badge variant="outline" className="text-[10px]">Not set</Badge>
|
|
436
|
-
)}
|
|
437
|
-
</div>
|
|
438
|
-
<p className="mb-2 text-xs text-muted-foreground">
|
|
439
|
-
Unlocks <strong>GPT-5.4</strong>, <strong>GPT-5.3 Codex</strong>, and <strong>GPT-5.4 Mini</strong>.
|
|
440
|
-
</p>
|
|
441
|
-
<button
|
|
442
|
-
type="button"
|
|
443
|
-
onClick={() => setShowHowTo(showHowTo === 'openai' ? null : 'openai')}
|
|
444
|
-
className="mb-3 flex items-center gap-1 text-xs text-muted-foreground hover:text-foreground transition-colors"
|
|
445
|
-
>
|
|
446
|
-
{showHowTo === 'openai' ? <ChevronUp className="h-3 w-3" /> : <ChevronDown className="h-3 w-3" />}
|
|
447
|
-
How to get an OpenAI API key
|
|
448
|
-
</button>
|
|
449
|
-
{showHowTo === 'openai' && (
|
|
450
|
-
<div className="mb-3 rounded-md bg-muted/50 p-3 text-xs text-muted-foreground space-y-1.5">
|
|
451
|
-
<p>1. Go to{' '}
|
|
452
|
-
<a href="https://platform.openai.com/api-keys" target="_blank" rel="noopener noreferrer" className="underline inline-flex items-center gap-0.5">
|
|
453
|
-
platform.openai.com/api-keys <ExternalLink className="h-2.5 w-2.5" />
|
|
454
|
-
</a>
|
|
455
|
-
</p>
|
|
456
|
-
<p>2. Sign in or create an OpenAI account</p>
|
|
457
|
-
<p>3. Click <strong>Create new secret key</strong>, name it (e.g. "ifc-lite")</p>
|
|
458
|
-
<p>4. Copy the key (starts with <code className="bg-muted px-1 rounded">sk-...</code>)</p>
|
|
459
|
-
<p>5. Paste it below and click Save</p>
|
|
460
|
-
<p className="pt-1 text-muted-foreground/70">OpenAI requires prepaid credits or a payment method. Usage is billed to your OpenAI account.</p>
|
|
461
|
-
</div>
|
|
462
|
-
)}
|
|
463
|
-
<ApiKeyInput
|
|
464
|
-
id="openai-key"
|
|
465
|
-
value={openaiKey}
|
|
466
|
-
onChange={setOpenaiKey}
|
|
467
|
-
onSave={saveOpenaiKey}
|
|
468
|
-
placeholder="sk-..."
|
|
469
|
-
isSaved={openaiKey.trim() === apiKeys.openaiKey}
|
|
470
|
-
/>
|
|
471
|
-
</div>
|
|
472
|
-
|
|
473
|
-
{hasAnyKey && (
|
|
474
|
-
<div className="flex justify-end">
|
|
475
|
-
<Button variant="outline" size="sm" onClick={handleClearAll}>
|
|
476
|
-
<Trash2 className="mr-2 h-3.5 w-3.5" />
|
|
477
|
-
Remove all keys
|
|
478
|
-
</Button>
|
|
479
|
-
</div>
|
|
480
|
-
)}
|
|
481
|
-
|
|
482
|
-
{/* Security & Privacy notice */}
|
|
483
|
-
<div className="rounded-md border border-dashed p-4 text-xs space-y-2">
|
|
484
|
-
<p className="font-medium text-foreground">Your API keys never leave your machine.</p>
|
|
485
|
-
<ul className="list-disc pl-4 space-y-1 text-muted-foreground">
|
|
486
|
-
<li>Keys are stored in your browser's <code className="bg-muted px-1 rounded">localStorage</code> and persist across page reloads.</li>
|
|
487
|
-
<li>When you use a BYOK model, requests go <strong>directly from your browser to the provider</strong> (Anthropic or OpenAI). They never pass through our servers.</li>
|
|
488
|
-
<li>Free models use our server proxy and do not require any API key.</li>
|
|
489
|
-
<li>Clearing your browser data or clicking "Remove all keys" above permanently deletes them.</li>
|
|
490
|
-
</ul>
|
|
491
|
-
<p className="text-muted-foreground pt-1">
|
|
492
|
-
<strong>Verify in your browser console:</strong> open DevTools (F12), go to the Console tab, and run:
|
|
493
|
-
</p>
|
|
494
|
-
<pre className="bg-muted rounded px-2 py-1.5 font-mono text-[11px] overflow-x-auto text-foreground">
|
|
495
|
-
{`JSON.parse(localStorage.getItem('ifc-lite:api-keys:v1') ?? '{}')`}
|
|
496
|
-
</pre>
|
|
497
|
-
<p className="text-muted-foreground">
|
|
498
|
-
This shows exactly what is stored. Only you can see it. To delete everything:
|
|
499
|
-
</p>
|
|
500
|
-
<pre className="bg-muted rounded px-2 py-1.5 font-mono text-[11px] overflow-x-auto text-foreground">
|
|
501
|
-
{`localStorage.removeItem('ifc-lite:api-keys:v1')`}
|
|
502
|
-
</pre>
|
|
503
|
-
</div>
|
|
504
|
-
</div>
|
|
505
|
-
</section>
|
|
506
|
-
);
|
|
507
|
-
}
|
|
508
|
-
|
|
509
|
-
// ── Helpers ──────────────────────────────────────────────────────────────
|
|
510
|
-
|
|
511
|
-
function formatTimestamp(value: number | null): string {
|
|
512
|
-
if (!value) {
|
|
513
|
-
return 'Not validated yet';
|
|
514
|
-
}
|
|
515
|
-
return new Date(value).toLocaleString();
|
|
516
|
-
}
|
|
517
|
-
|
|
518
|
-
function formatOfflineGrace(entitlement: DesktopEntitlement): string {
|
|
519
|
-
if (!entitlement.graceUntil) {
|
|
520
|
-
return 'No offline grace cached yet';
|
|
521
|
-
}
|
|
522
|
-
const remainingDays = Math.max(0, Math.ceil((entitlement.graceUntil - Date.now()) / (24 * 60 * 60 * 1000)));
|
|
523
|
-
return `${new Date(entitlement.graceUntil).toLocaleString()}${remainingDays > 0 ? ` (${remainingDays} day${remainingDays === 1 ? '' : 's'} left)` : ''}`;
|
|
524
|
-
}
|
|
525
|
-
|
|
526
|
-
function describeDesktopStatus(entitlement: DesktopEntitlement): string {
|
|
527
|
-
switch (entitlement.status) {
|
|
528
|
-
case 'trial':
|
|
529
|
-
return entitlement.trialEndsAt
|
|
530
|
-
? `Trial active until ${new Date(entitlement.trialEndsAt).toLocaleDateString()}`
|
|
531
|
-
: 'Trial active';
|
|
532
|
-
case 'grace_offline':
|
|
533
|
-
return entitlement.graceUntil
|
|
534
|
-
? `Offline grace until ${new Date(entitlement.graceUntil).toLocaleDateString()}`
|
|
535
|
-
: 'Offline grace active';
|
|
536
|
-
case 'expired':
|
|
537
|
-
return 'Subscription expired';
|
|
538
|
-
case 'active':
|
|
539
|
-
return 'Subscription active';
|
|
540
|
-
case 'signed_out':
|
|
541
|
-
return 'Signed out';
|
|
542
|
-
case 'anonymous':
|
|
543
|
-
return 'No account configured';
|
|
544
|
-
default:
|
|
545
|
-
return entitlement.status;
|
|
546
|
-
}
|
|
547
|
-
}
|
|
548
|
-
|
|
549
|
-
function getStatusBadgeVariant(entitlement: DesktopEntitlement): 'default' | 'secondary' | 'destructive' | 'outline' {
|
|
550
|
-
switch (entitlement.status) {
|
|
551
|
-
case 'active':
|
|
552
|
-
case 'trial':
|
|
553
|
-
return 'default';
|
|
554
|
-
case 'grace_offline':
|
|
555
|
-
return 'secondary';
|
|
556
|
-
case 'expired':
|
|
557
|
-
return 'destructive';
|
|
558
|
-
default:
|
|
559
|
-
return 'outline';
|
|
560
|
-
}
|
|
561
|
-
}
|
|
562
|
-
|
|
563
|
-
function StatusBadge({ entitlement }: { entitlement: DesktopEntitlement }) {
|
|
564
|
-
return (
|
|
565
|
-
<Badge variant={getStatusBadgeVariant(entitlement)}>
|
|
566
|
-
{describeDesktopStatus(entitlement)}
|
|
567
|
-
</Badge>
|
|
568
|
-
);
|
|
569
|
-
}
|
|
570
|
-
|
|
571
|
-
function InfoCard({ title, body, icon }: { title: string; body: string; icon: React.ReactNode }) {
|
|
572
|
-
return (
|
|
573
|
-
<div className="rounded-md border p-4">
|
|
574
|
-
<div className="mb-2 flex items-center gap-2 text-sm font-medium">
|
|
575
|
-
{icon}
|
|
576
|
-
{title}
|
|
577
|
-
</div>
|
|
578
|
-
<p className="text-sm text-muted-foreground">{body}</p>
|
|
579
|
-
</div>
|
|
580
|
-
);
|
|
581
|
-
}
|