@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.
Files changed (162) hide show
  1. package/.turbo/turbo-build.log +35 -42
  2. package/CHANGELOG.md +74 -0
  3. package/dist/assets/{basketViewActivator-B3CdrLsb.js → basketViewActivator-Ce38DhXd.js} +8 -8
  4. package/dist/assets/{bcf-QeHK_Aud.js → bcf-Cv_O3JfD.js} +56 -56
  5. package/dist/assets/{decode-worker-CgM1iNSK.js → decode-worker-Cjign7Zh.js} +1 -1
  6. package/dist/assets/{deflate-B-d0SYQM.js → deflate-HbyMq59o.js} +1 -1
  7. package/dist/assets/drawing-2d-DW98umlt.js +257 -0
  8. package/dist/assets/e57-source-2wI9jkCA.js +1 -0
  9. package/dist/assets/{exporters-B4LbZFeT.js → exporters-BuD3XRzB.js} +1309 -1153
  10. package/dist/assets/geometry.worker-TH3fCCoY.js +1 -0
  11. package/dist/assets/{geotiff-CrVtDRFq.js → geotiff-B2HA8Bwm.js} +10 -10
  12. package/dist/assets/{ids-DjsGFN10.js → ids-DYUFMd5f.js} +952 -945
  13. package/dist/assets/{ifc-lite_bg-DsYUIHm3.wasm → ifc-lite_bg-BEA5DLmg.wasm} +0 -0
  14. package/dist/assets/index-E9wB0zWt.css +1 -0
  15. package/dist/assets/{index-COYokSKc.js → index-n5O1QJMM.js} +37877 -38126
  16. package/dist/assets/{index.es-CY202jA3.js → index.es-BKVIpZgL.js} +9 -9
  17. package/dist/assets/{jpeg-D4wOkf5h.js → jpeg-C7hjKjPX.js} +1 -1
  18. package/dist/assets/{jspdf.es.min-DIGb9BHN.js → jspdf.es.min-oWlFc42Y.js} +4 -4
  19. package/dist/assets/lens-C4p1kQ0p.js +1 -0
  20. package/dist/assets/{lerc-DmW0_tgf.js → lerc-BfIOGhQz.js} +1 -1
  21. package/dist/assets/{lzw-oWetY-d6.js → lzw-B0jRuuW5.js} +1 -1
  22. package/dist/assets/{native-bridge-BX8_tHXE.js → native-bridge-DpB-dtEn.js} +6 -3
  23. package/dist/assets/{packbits-F8Nkp4NY.js → packbits-DVvBTC39.js} +1 -1
  24. package/dist/assets/parser.worker-BDsWQ6rc.js +182 -0
  25. package/dist/assets/{pdf-Dsh3HPZB.js → pdf-dVIqI5ac.js} +10 -10
  26. package/dist/assets/raw-C0ZJYGmN.js +1 -0
  27. package/dist/assets/{sandbox-BAC3a-eN.js → sandbox-qpJlrNN0.js} +2962 -2554
  28. package/dist/assets/server-client-DVZ2huNS.js +719 -0
  29. package/dist/assets/{webimage-BLV1dgmd.js → webimage-B394g0Tw.js} +1 -1
  30. package/dist/assets/{xlsx-Bc2HTrjC.js → xlsx-D-oHO76J.js} +8 -8
  31. package/dist/assets/{zstd-C_1HxVrA.js → zstd-Bf38MwV2.js} +1 -1
  32. package/dist/index.html +9 -9
  33. package/package.json +24 -23
  34. package/src/App.tsx +1 -3
  35. package/src/components/mcp/playground-dispatcher.ts +3 -0
  36. package/src/components/mcp/playground-files.ts +33 -1
  37. package/src/components/viewer/BCFPanel.tsx +1 -16
  38. package/src/components/viewer/ChatPanel.tsx +11 -46
  39. package/src/components/viewer/CommandPalette.tsx +6 -1
  40. package/src/components/viewer/ComparePanel.tsx +420 -0
  41. package/src/components/viewer/HierarchyPanel.tsx +48 -183
  42. package/src/components/viewer/IDSPanel.tsx +1 -26
  43. package/src/components/viewer/MainToolbar.tsx +94 -187
  44. package/src/components/viewer/MobileToolbar.tsx +1 -9
  45. package/src/components/viewer/PropertiesPanel.tsx +98 -127
  46. package/src/components/viewer/ScriptPanel.tsx +8 -34
  47. package/src/components/viewer/Section2DPanel.tsx +32 -1
  48. package/src/components/viewer/ViewerLayout.tsx +5 -2
  49. package/src/components/viewer/Viewport.tsx +3 -0
  50. package/src/components/viewer/ViewportContainer.tsx +24 -42
  51. package/src/components/viewer/ViewportOverlays.tsx +1 -4
  52. package/src/components/viewer/hierarchy/HierarchyNode.tsx +3 -3
  53. package/src/components/viewer/hierarchy/ifc-icons.ts +9 -0
  54. package/src/components/viewer/hierarchy/treeDataBuilder.ts +87 -0
  55. package/src/components/viewer/hierarchy/types.ts +1 -0
  56. package/src/components/viewer/hierarchy/useHierarchyTree.ts +6 -2
  57. package/src/components/viewer/properties/MaterialTotalsPanel.tsx +283 -0
  58. package/src/components/viewer/useGeometryStreaming.ts +0 -2
  59. package/src/hooks/federationLoadGate.test.ts +12 -2
  60. package/src/hooks/federationLoadGate.ts +9 -2
  61. package/src/hooks/ingest/federationAlign.ts +488 -0
  62. package/src/hooks/ingest/viewerModelIngest.ts +3 -212
  63. package/src/hooks/useCompare.ts +0 -0
  64. package/src/hooks/useCompareOverlay.ts +119 -0
  65. package/src/hooks/useDrawingGeneration.ts +234 -14
  66. package/src/hooks/useIfc.ts +1 -1
  67. package/src/hooks/useIfcCache.ts +100 -24
  68. package/src/hooks/useIfcFederation.ts +42 -811
  69. package/src/hooks/useIfcLoader.ts +349 -1517
  70. package/src/hooks/useIfcServer.ts +3 -0
  71. package/src/hooks/useLens.ts +5 -1
  72. package/src/hooks/useSymbolicAnnotations.ts +70 -38
  73. package/src/lib/compare/buildFingerprints.ts +173 -0
  74. package/src/lib/compare/describeChange.ts +0 -0
  75. package/src/lib/compare/geometricData.test.ts +54 -0
  76. package/src/lib/compare/geometricData.ts +37 -0
  77. package/src/lib/compare/overlay.test.ts +99 -0
  78. package/src/lib/compare/overlay.ts +91 -0
  79. package/src/lib/geo/cesium-placement.ts +1 -1
  80. package/src/lib/geo/reproject.ts +4 -1
  81. package/src/lib/llm/script-edit-ops.ts +23 -0
  82. package/src/lib/llm/stream-client.ts +8 -1
  83. package/src/lib/search/result-export.ts +7 -1
  84. package/src/sdk/adapters/export-adapter.ts +6 -1
  85. package/src/services/cacheService.ts +9 -25
  86. package/src/services/desktop-export.ts +2 -59
  87. package/src/services/file-dialog.ts +8 -142
  88. package/src/store/constants.ts +23 -0
  89. package/src/store/globalId.ts +15 -13
  90. package/src/store/index.ts +19 -6
  91. package/src/store/slices/cesiumSlice.ts +8 -1
  92. package/src/store/slices/compareSlice.ts +96 -0
  93. package/src/store/slices/drawing2DSlice.ts +8 -0
  94. package/src/store/slices/lensSlice.ts +8 -0
  95. package/src/store/slices/visibilitySlice.ts +22 -1
  96. package/src/store/types.ts +1 -71
  97. package/src/utils/acquireFileBuffer.test.ts +12 -4
  98. package/src/utils/ifcConfig.ts +0 -12
  99. package/src/utils/loadingUtils.ts +32 -0
  100. package/src/utils/spatialHierarchy.test.ts +53 -1
  101. package/src/utils/spatialHierarchy.ts +42 -2
  102. package/src/vite-env.d.ts +2 -0
  103. package/vite.config.ts +6 -3
  104. package/DESKTOP_CONTRACT_VERSION +0 -1
  105. package/dist/assets/drawing-2d-C71b8Ugx.js +0 -257
  106. package/dist/assets/e57-source-CQHxE8n3.js +0 -1
  107. package/dist/assets/event-B0kAzHa-.js +0 -1
  108. package/dist/assets/geometry.worker-BdH-E6NB.js +0 -1
  109. package/dist/assets/index-ajK6D32J.css +0 -1
  110. package/dist/assets/lens-PYsLu_MA.js +0 -1
  111. package/dist/assets/parser.worker-D591Zu_-.js +0 -182
  112. package/dist/assets/raw-D9iw0tmc.js +0 -1
  113. package/dist/assets/server-client-Cjwnm7il.js +0 -706
  114. package/dist/assets/tauri-core-stub-D8Fa-u43.js +0 -1
  115. package/dist/assets/tauri-dialog-stub-r7Wksg7o.js +0 -1
  116. package/dist/assets/tauri-fs-stub-BdeRC7aK.js +0 -1
  117. package/src/components/viewer/DesktopEntitlementBanner.tsx +0 -74
  118. package/src/components/viewer/SettingsPage.tsx +0 -581
  119. package/src/hooks/ingest/resolveDataStoreOrAbort.test.ts +0 -61
  120. package/src/hooks/ingest/resolveDataStoreOrAbort.ts +0 -28
  121. package/src/hooks/ingest/watchedGeometryStream.test.ts +0 -78
  122. package/src/hooks/ingest/watchedGeometryStream.ts +0 -76
  123. package/src/lib/desktop/desktopEntitlementEvents.ts +0 -39
  124. package/src/lib/desktop-entitlement.ts +0 -43
  125. package/src/lib/desktop-product.ts +0 -130
  126. package/src/lib/platform.ts +0 -23
  127. package/src/services/desktop-cache.ts +0 -186
  128. package/src/services/desktop-harness.ts +0 -196
  129. package/src/services/desktop-logger.ts +0 -20
  130. package/src/services/desktop-native-metadata.ts +0 -230
  131. package/src/services/desktop-panel-actions.ts +0 -43
  132. package/src/services/desktop-preferences.ts +0 -44
  133. package/src/services/fs-cache.ts +0 -212
  134. package/src/services/tauri-core-stub.ts +0 -7
  135. package/src/services/tauri-dialog-stub.ts +0 -7
  136. package/src/services/tauri-fs-stub.ts +0 -7
  137. package/src/services/tauri-modules.d.ts +0 -50
  138. package/src/store/slices/desktopEntitlementSlice.ts +0 -86
  139. package/src/utils/desktopModelSnapshot.ts +0 -358
  140. package/src/utils/nativeSpatialDataStore.ts +0 -277
  141. package/src-tauri/Cargo.toml +0 -29
  142. package/src-tauri/build.rs +0 -7
  143. package/src-tauri/capabilities/default.json +0 -18
  144. package/src-tauri/icons/128x128.png +0 -0
  145. package/src-tauri/icons/128x128@2x.png +0 -0
  146. package/src-tauri/icons/32x32.png +0 -0
  147. package/src-tauri/icons/Square107x107Logo.png +0 -0
  148. package/src-tauri/icons/Square142x142Logo.png +0 -0
  149. package/src-tauri/icons/Square150x150Logo.png +0 -0
  150. package/src-tauri/icons/Square284x284Logo.png +0 -0
  151. package/src-tauri/icons/Square30x30Logo.png +0 -0
  152. package/src-tauri/icons/Square310x310Logo.png +0 -0
  153. package/src-tauri/icons/Square44x44Logo.png +0 -0
  154. package/src-tauri/icons/Square71x71Logo.png +0 -0
  155. package/src-tauri/icons/Square89x89Logo.png +0 -0
  156. package/src-tauri/icons/StoreLogo.png +0 -0
  157. package/src-tauri/icons/icon.icns +0 -0
  158. package/src-tauri/icons/icon.ico +0 -0
  159. package/src-tauri/icons/icon.png +0 -0
  160. package/src-tauri/src/lib.rs +0 -21
  161. package/src-tauri/src/main.rs +0 -10
  162. 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. &quot;ifc-lite&quot;)</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. &quot;ifc-lite&quot;)</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&apos;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 &quot;Remove all keys&quot; 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
- }