@morscherlab/mint-sdk 1.0.0-alpha.9 → 1.0.0-beta.2

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 (81) hide show
  1. package/dist/__tests__/components/PluginIcon.test.d.ts +1 -0
  2. package/dist/components/AppTopBar.vue.d.ts +2 -0
  3. package/dist/components/BaseButton.vue.d.ts +1 -1
  4. package/dist/components/BaseCheckbox.vue.d.ts +1 -1
  5. package/dist/components/BaseInput.vue.d.ts +1 -1
  6. package/dist/components/BasePill.vue.d.ts +1 -1
  7. package/dist/components/BaseRadioGroup.vue.d.ts +1 -1
  8. package/dist/components/BaseSelect.vue.d.ts +1 -1
  9. package/dist/components/BaseSlider.vue.d.ts +1 -1
  10. package/dist/components/BaseTextarea.vue.d.ts +1 -1
  11. package/dist/components/BaseToggle.vue.d.ts +1 -1
  12. package/dist/components/ColorSlider.vue.d.ts +1 -1
  13. package/dist/components/ConcentrationInput.vue.d.ts +1 -1
  14. package/dist/components/DatePicker.vue.d.ts +1 -1
  15. package/dist/components/DateTimePicker.vue.d.ts +1 -1
  16. package/dist/components/Divider.vue.d.ts +1 -1
  17. package/dist/components/DropdownButton.vue.d.ts +1 -1
  18. package/dist/components/FileUploader.vue.d.ts +1 -1
  19. package/dist/components/FormulaInput.vue.d.ts +1 -1
  20. package/dist/components/IconButton.vue.d.ts +1 -1
  21. package/dist/components/LoadingSpinner.vue.d.ts +1 -1
  22. package/dist/components/MultiSelect.vue.d.ts +1 -1
  23. package/dist/components/NumberInput.vue.d.ts +1 -1
  24. package/dist/components/PluginIcon.vue.d.ts +11 -0
  25. package/dist/components/ProgressBar.vue.d.ts +1 -1
  26. package/dist/components/ReagentEditor.vue.d.ts +1 -1
  27. package/dist/components/ResourceCard.vue.d.ts +1 -1
  28. package/dist/components/SampleSelector.vue.d.ts +1 -1
  29. package/dist/components/ScientificNumber.vue.d.ts +1 -1
  30. package/dist/components/SegmentedControl.vue.d.ts +1 -1
  31. package/dist/components/SettingsModal.vue.d.ts +22 -2
  32. package/dist/components/TagsInput.vue.d.ts +1 -1
  33. package/dist/components/TimePicker.vue.d.ts +1 -1
  34. package/dist/components/TimeRangeInput.vue.d.ts +1 -1
  35. package/dist/components/UnitInput.vue.d.ts +1 -1
  36. package/dist/components/WellPlate.vue.d.ts +1 -1
  37. package/dist/components/index.d.ts +1 -0
  38. package/dist/components/index.js +3 -3
  39. package/dist/{components-CzbQQPCb.js → components-_XqPEhP9.js} +572 -362
  40. package/dist/components-_XqPEhP9.js.map +1 -0
  41. package/dist/composables/index.js +2 -2
  42. package/dist/composables/usePlatformContext.d.ts +3 -0
  43. package/dist/{composables-BXklV5ii.js → composables-tiZqLu1M.js} +2 -2
  44. package/dist/{composables-BXklV5ii.js.map → composables-tiZqLu1M.js.map} +1 -1
  45. package/dist/index.d.ts +2 -2
  46. package/dist/index.js +4 -4
  47. package/dist/install.js +2 -2
  48. package/dist/stores/auth.d.ts +1 -1
  49. package/dist/styles.css +896 -553
  50. package/dist/types/components.d.ts +39 -0
  51. package/dist/types/index.d.ts +1 -1
  52. package/dist/types/platform.d.ts +1 -0
  53. package/dist/{useScheduleDrag-CxBeqYcu.js → useScheduleDrag-CA9sGNJG.js} +4000 -4000
  54. package/dist/useScheduleDrag-CA9sGNJG.js.map +1 -0
  55. package/package.json +1 -1
  56. package/src/__tests__/components/AppTopBar.test.ts +31 -13
  57. package/src/__tests__/components/PluginIcon.test.ts +119 -0
  58. package/src/components/AppTopBar.vue +32 -27
  59. package/src/components/PluginIcon.story.vue +71 -0
  60. package/src/components/PluginIcon.vue +88 -0
  61. package/src/components/SettingsModal.story.vue +337 -45
  62. package/src/components/SettingsModal.vue +251 -64
  63. package/src/components/index.ts +1 -0
  64. package/src/index.ts +4 -0
  65. package/src/styles/components/app-pill-nav.css +1 -2
  66. package/src/styles/components/app-top-bar.css +1 -2
  67. package/src/styles/components/button.css +3 -7
  68. package/src/styles/components/dropdown-button.css +4 -4
  69. package/src/styles/components/input.css +4 -5
  70. package/src/styles/components/number-input.css +3 -3
  71. package/src/styles/components/plugin-icon.css +38 -0
  72. package/src/styles/components/segmented-control.css +4 -7
  73. package/src/styles/components/settings-modal.css +184 -0
  74. package/src/styles/components/tabs.css +1 -2
  75. package/src/styles/components/textarea.css +4 -5
  76. package/src/styles/components/unit-input.css +3 -3
  77. package/src/types/components.ts +42 -0
  78. package/src/types/index.ts +3 -0
  79. package/src/types/platform.ts +1 -0
  80. package/dist/components-CzbQQPCb.js.map +0 -1
  81. package/dist/useScheduleDrag-CxBeqYcu.js.map +0 -1
@@ -1,3 +1,3 @@
1
- import { $ as formatTime, A as APP_EXPERIMENT_KEY, B as formatExperimentDate, D as useAutoGroup, F as EXPERIMENT_STATUS_LABELS, H as useTheme, I as EXPERIMENT_STATUS_OPTIONS, J as addMinutes, L as EXPERIMENT_STATUS_VARIANT_MAP, M as usePlatformContext, N as useExperimentSelector, O as useWellPlateEditor, P as DATE_PRESET_OPTIONS, Q as formatDuration, R as SORT_OPTIONS, S as useConcentrationUnits, T as extractSamplesFromDesignData, U as useToast, V as useApi, X as durationMinutes, Y as compareTime, Z as findAvailableSlots, _ as DEFAULT_UNITS, a as getFieldRegistryEntry, at as useTimeUtils, b as useDoseCalculator, c as useForm, d as useSequenceUtils, et as generateTimeSlots, g as DEFAULT_PRESETS, h as useProtocolTemplates, i as useFormBuilder, it as snapToSlot, j as useAppExperiment, k as useRackEditor, m as useChemicalFormula, n as useExperimentData, nt as parseTime, o as getTypeDefault, p as ATOMIC_WEIGHTS, r as evaluateCondition, rt as rangesOverlap, t as useScheduleDrag, tt as isTimeInRange, v as generateDilutionSeries, w as DEFAULT_COLORS, y as useReagentSeries, z as datePresetToISO } from "../useScheduleDrag-CxBeqYcu.js";
2
- import { a as useAsyncBatch, i as useAsync, n as useExperimentSave, o as usePasskey, r as usePluginConfig, s as useAuth, t as usePluginApi } from "../composables-BXklV5ii.js";
1
+ import { $ as formatTime, A as getFieldRegistryEntry, B as useChemicalFormula, C as SORT_OPTIONS, D as evaluateCondition, E as useApi, H as useTheme, J as addMinutes, L as useSequenceUtils, O as useFormBuilder, P as useConcentrationUnits, Q as formatDuration, S as EXPERIMENT_STATUS_VARIANT_MAP, T as formatExperimentDate, U as useToast, X as durationMinutes, Y as compareTime, Z as findAvailableSlots, _ as usePlatformContext, a as DEFAULT_UNITS, at as useTimeUtils, b as EXPERIMENT_STATUS_LABELS, c as useDoseCalculator, et as generateTimeSlots, f as useAutoGroup, g as useAppExperiment, h as APP_EXPERIMENT_KEY, i as DEFAULT_PRESETS, it as snapToSlot, j as getTypeDefault, k as useForm, l as DEFAULT_COLORS, m as useRackEditor, n as useExperimentData, nt as parseTime, o as generateDilutionSeries, p as useWellPlateEditor, r as useProtocolTemplates, rt as rangesOverlap, s as useReagentSeries, t as useScheduleDrag, tt as isTimeInRange, u as extractSamplesFromDesignData, v as useExperimentSelector, w as datePresetToISO, x as EXPERIMENT_STATUS_OPTIONS, y as DATE_PRESET_OPTIONS, z as ATOMIC_WEIGHTS } from "../useScheduleDrag-CA9sGNJG.js";
2
+ import { a as useAsyncBatch, i as useAsync, n as useExperimentSave, o as usePasskey, r as usePluginConfig, s as useAuth, t as usePluginApi } from "../composables-tiZqLu1M.js";
3
3
  export { APP_EXPERIMENT_KEY, ATOMIC_WEIGHTS, DATE_PRESET_OPTIONS, DEFAULT_COLORS, DEFAULT_PRESETS, DEFAULT_UNITS, EXPERIMENT_STATUS_LABELS, EXPERIMENT_STATUS_OPTIONS, EXPERIMENT_STATUS_VARIANT_MAP, SORT_OPTIONS, addMinutes, compareTime, datePresetToISO, durationMinutes, evaluateCondition, extractSamplesFromDesignData, findAvailableSlots, formatDuration, formatExperimentDate, formatTime, generateDilutionSeries, generateTimeSlots, getFieldRegistryEntry, getTypeDefault, isTimeInRange, parseTime, rangesOverlap, snapToSlot, useApi, useAppExperiment, useAsync, useAsyncBatch, useAuth, useAutoGroup, useChemicalFormula, useConcentrationUnits, useDoseCalculator, useExperimentData, useExperimentSave, useExperimentSelector, useForm, useFormBuilder, usePasskey, usePlatformContext, usePluginApi, usePluginConfig, useProtocolTemplates, useRackEditor, useReagentSeries, useScheduleDrag, useSequenceUtils, useTheme, useTimeUtils, useToast, useWellPlateEditor };
@@ -34,6 +34,7 @@ export declare function usePlatformContext(options?: PlatformContextOptions): {
34
34
  version: string;
35
35
  description?: string | undefined;
36
36
  icon?: string | undefined;
37
+ color?: string | undefined;
37
38
  route_prefix: string;
38
39
  api_prefix: string;
39
40
  nav_items?: {
@@ -68,6 +69,7 @@ export declare function usePlatformContext(options?: PlatformContextOptions): {
68
69
  version: string;
69
70
  description?: string | undefined;
70
71
  icon?: string | undefined;
72
+ color?: string | undefined;
71
73
  route_prefix: string;
72
74
  api_prefix: string;
73
75
  nav_items?: {
@@ -102,6 +104,7 @@ export declare function usePlatformContext(options?: PlatformContextOptions): {
102
104
  version: string;
103
105
  description?: string | undefined;
104
106
  icon?: string | undefined;
107
+ color?: string | undefined;
105
108
  route_prefix: string;
106
109
  api_prefix: string;
107
110
  nav_items?: {
@@ -1,4 +1,4 @@
1
- import { M as usePlatformContext, V as useApi } from "./useScheduleDrag-CxBeqYcu.js";
1
+ import { E as useApi, _ as usePlatformContext } from "./useScheduleDrag-CA9sGNJG.js";
2
2
  import { r as useSettingsStore, t as useAuthStore } from "./auth-DsI0rQ7_.js";
3
3
  import { computed, getCurrentInstance, onMounted, onUnmounted, ref, watch } from "vue";
4
4
  import axios from "axios";
@@ -802,4 +802,4 @@ function usePluginApi(options = {}) {
802
802
  //#endregion
803
803
  export { useAsyncBatch as a, useAsync as i, useExperimentSave as n, usePasskey as o, usePluginConfig as r, useAuth as s, usePluginApi as t };
804
804
 
805
- //# sourceMappingURL=composables-BXklV5ii.js.map
805
+ //# sourceMappingURL=composables-tiZqLu1M.js.map
@@ -1 +1 @@
1
- {"version":3,"file":"composables-BXklV5ii.js","names":[],"sources":["../src/composables/useAuth.ts","../src/composables/usePasskey.ts","../src/composables/useAsync.ts","../src/composables/usePluginConfig.ts","../src/composables/useExperimentSave.ts","../src/composables/usePluginApi.ts"],"sourcesContent":["import axios from 'axios'\nimport { ref, onMounted, onUnmounted, watch, getCurrentInstance, type Ref } from 'vue'\nimport { useAuthStore } from '../stores/auth'\nimport { useSettingsStore } from '../stores/settings'\nimport type { AuthConfig, UserInfo, LoginResponse, TokenVerifyResponse, UpdateProfileRequest } from '../types'\n\ninterface UserResponse {\n id: string\n username: string\n shortname: string | null\n email: string | null\n role: string\n is_active: boolean\n}\n\ninterface RefreshResponse {\n access_token: string\n expires_in: number\n token_type: string\n}\n\n// Token refresh configuration\nconst TOKEN_REFRESH_MARGIN_MS = 5 * 60 * 1000 // Refresh 5 minutes before expiry\nconst TOKEN_REFRESH_CHECK_INTERVAL_MS = 60 * 1000 // Check every minute\n\n/**\n * Authentication composable with automatic token refresh.\n *\n * Features:\n * - Automatic token refresh before expiration\n * - Login/logout/register functionality\n * - Token verification on startup\n * - User profile management\n *\n * @example\n * ```typescript\n * const { login, logout, isAuthenticated, user } = useAuth()\n *\n * // Login\n * const success = await login('username', 'password')\n *\n * // Automatic refresh is enabled by default\n * // Tokens are refreshed 5 minutes before expiration\n * ```\n */\nexport interface UseAuthReturn {\n login: (username: string, password: string) => Promise<boolean>\n logout: () => void\n register: (username: string, password: string, email?: string) => Promise<boolean>\n verifyToken: () => Promise<boolean>\n fetchAuthConfig: () => Promise<AuthConfig>\n initializeAuth: () => Promise<void>\n getCurrentUser: () => Promise<UserInfo | null>\n getAuthHeader: () => Record<string, string>\n updateProfile: (data: { email?: string; shortname?: string; currentPassword?: string; newPassword?: string }) => Promise<{ success: boolean; error?: string }>\n refreshToken: () => Promise<boolean>\n isRefreshing: Ref<boolean>\n}\n\n// Module-level singletons to prevent duplicate refresh requests and timers\n// across multiple useAuth() instances\nlet _refreshPromise: Promise<boolean> | null = null\nlet _refreshTimerId: number | null = null\nlet _mountedConsumerCount = 0\n\n/** @internal Reset module-level state — only for use in tests. */\nexport function _resetAuthStateForTesting(): void {\n _refreshPromise = null\n if (_refreshTimerId !== null) {\n clearTimeout(_refreshTimerId)\n _refreshTimerId = null\n }\n _mountedConsumerCount = 0\n}\n\n/** Manages authentication state with login/logout/register and automatic JWT token refresh. */\nexport function useAuth(): UseAuthReturn {\n const authStore = useAuthStore()\n const settingsStore = useSettingsStore()\n\n const isRefreshing = ref(false)\n\n function getApiBaseUrl(): string {\n return settingsStore.getApiBaseUrl()\n }\n\n async function fetchAuthConfig(): Promise<AuthConfig> {\n try {\n const response = await axios.get<{\n auth_required: boolean\n passkey_enabled: boolean\n passkey_registered?: boolean\n registration_enabled?: boolean\n database_mode?: string\n }>(`${getApiBaseUrl()}/setup/config/public`)\n\n const config: AuthConfig = {\n authRequired: response.data.auth_required,\n passkeyEnabled: response.data.passkey_enabled,\n passkeyRegistered: response.data.passkey_registered ?? false,\n registrationEnabled: response.data.registration_enabled ?? false,\n databaseMode: response.data.database_mode ?? 'none',\n }\n\n authStore.setAuthConfig(config)\n return config\n } catch (error) {\n console.error('Failed to fetch auth config:', error)\n return {\n authRequired: false,\n passkeyEnabled: false,\n passkeyRegistered: false,\n registrationEnabled: false,\n databaseMode: 'none',\n }\n }\n }\n\n async function login(username: string, password: string): Promise<boolean> {\n authStore.setLoading(true)\n authStore.setError(null)\n\n try {\n const response = await axios.post<LoginResponse>(\n `${getApiBaseUrl()}/auth/login`,\n { username, password }\n )\n\n authStore.setToken(response.data.access_token, response.data.expires_in)\n authStore.setUsername(username)\n\n await getCurrentUser()\n\n // Start auto-refresh after successful login\n scheduleTokenRefresh()\n\n return true\n } catch (error) {\n if (axios.isAxiosError(error) && error.response) {\n authStore.setError(error.response.data.detail || 'Login failed')\n } else {\n authStore.setError('Network error. Please try again.')\n }\n return false\n } finally {\n authStore.setLoading(false)\n }\n }\n\n async function register(username: string, password: string, email?: string): Promise<boolean> {\n authStore.setLoading(true)\n authStore.setError(null)\n\n try {\n await axios.post<UserResponse>(\n `${getApiBaseUrl()}/users/register`,\n { username, password, email }\n )\n\n return await login(username, password)\n } catch (error) {\n if (axios.isAxiosError(error) && error.response) {\n authStore.setError(error.response.data.detail || 'Registration failed')\n } else {\n authStore.setError('Network error. Please try again.')\n }\n return false\n } finally {\n authStore.setLoading(false)\n }\n }\n\n async function getCurrentUser(): Promise<UserInfo | null> {\n if (!authStore.token) {\n return null\n }\n\n try {\n const response = await axios.get<UserResponse>(\n `${getApiBaseUrl()}/users/me`,\n { headers: getAuthHeader() }\n )\n\n const userInfo: UserInfo = {\n id: response.data.id,\n username: response.data.username,\n shortname: response.data.shortname,\n email: response.data.email,\n role: response.data.role,\n isActive: response.data.is_active,\n }\n\n authStore.setUserInfo(userInfo)\n return userInfo\n } catch {\n return null\n }\n }\n\n async function verifyToken(): Promise<boolean> {\n if (!authStore.token) {\n return false\n }\n\n try {\n const response = await axios.get<TokenVerifyResponse>(\n `${getApiBaseUrl()}/auth/verify`,\n {\n headers: {\n Authorization: `Bearer ${authStore.token}`,\n },\n }\n )\n\n if (response.data.valid && response.data.username) {\n authStore.setUsername(response.data.username)\n return true\n }\n\n authStore.clearToken()\n return false\n } catch {\n authStore.clearToken()\n return false\n }\n }\n\n /**\n * Refresh the authentication token.\n * Called automatically before token expiration.\n * Uses promise caching to prevent concurrent refresh requests.\n */\n async function refreshToken(): Promise<boolean> {\n if (!authStore.token) return false\n if (_refreshPromise) return _refreshPromise\n\n _refreshPromise = (async () => {\n isRefreshing.value = true\n\n try {\n const response = await axios.post<RefreshResponse>(\n `${getApiBaseUrl()}/auth/refresh`,\n {},\n { headers: getAuthHeader() }\n )\n\n authStore.setToken(response.data.access_token, response.data.expires_in)\n\n // Reschedule next refresh\n scheduleTokenRefresh()\n\n return true\n } catch (error) {\n // If refresh fails, the token may have been revoked\n // Clear auth state and let user re-login\n if (axios.isAxiosError(error) && error.response?.status === 401) {\n console.warn('[Auth] Token refresh failed - session expired')\n authStore.clearToken()\n stopTokenRefresh()\n }\n return false\n } finally {\n isRefreshing.value = false\n _refreshPromise = null\n }\n })()\n\n return _refreshPromise\n }\n\n /**\n * Schedule automatic token refresh before expiration.\n */\n function scheduleTokenRefresh(): void {\n // Clear any existing timer\n stopTokenRefresh()\n\n if (!authStore.tokenExpires) {\n return\n }\n\n const expiresAt = authStore.tokenExpires.getTime()\n const refreshAt = expiresAt - TOKEN_REFRESH_MARGIN_MS\n const now = Date.now()\n\n if (refreshAt <= now) {\n // Token is already close to expiring or expired, refresh now\n refreshToken()\n return\n }\n\n // Schedule refresh\n const delay = refreshAt - now\n _refreshTimerId = window.setTimeout(() => {\n refreshToken()\n }, delay)\n }\n\n /**\n * Stop automatic token refresh.\n */\n function stopTokenRefresh(): void {\n if (_refreshTimerId !== null) {\n window.clearTimeout(_refreshTimerId)\n _refreshTimerId = null\n }\n }\n\n /**\n * Check if token needs refresh and refresh if necessary.\n * Called periodically as a safety net.\n */\n function checkAndRefreshIfNeeded(): void {\n if (!authStore.token || !authStore.tokenExpires) {\n return\n }\n\n const expiresAt = authStore.tokenExpires.getTime()\n const refreshAt = expiresAt - TOKEN_REFRESH_MARGIN_MS\n const now = Date.now()\n\n if (now >= refreshAt) {\n refreshToken()\n }\n }\n\n async function initializeAuth(): Promise<void> {\n authStore.initialize()\n await fetchAuthConfig()\n\n if (authStore.token) {\n const valid = await verifyToken()\n if (valid) {\n await getCurrentUser()\n // Start auto-refresh for existing session\n scheduleTokenRefresh()\n }\n }\n }\n\n function logout(): void {\n stopTokenRefresh()\n authStore.logout()\n }\n\n function getAuthHeader(): Record<string, string> {\n if (authStore.token) {\n return { Authorization: `Bearer ${authStore.token}` }\n }\n return {}\n }\n\n async function updateProfile(data: {\n email?: string\n shortname?: string\n currentPassword?: string\n newPassword?: string\n }): Promise<{ success: boolean; error?: string }> {\n if (!authStore.token) {\n return { success: false, error: 'Not authenticated' }\n }\n\n try {\n const requestData: UpdateProfileRequest = {}\n if (data.email !== undefined) requestData.email = data.email\n if (data.shortname !== undefined) requestData.shortname = data.shortname\n if (data.currentPassword) requestData.current_password = data.currentPassword\n if (data.newPassword) requestData.new_password = data.newPassword\n\n const response = await axios.put<UserResponse>(\n `${getApiBaseUrl()}/users/me`,\n requestData,\n { headers: getAuthHeader() }\n )\n\n const userInfo: UserInfo = {\n id: response.data.id,\n username: response.data.username,\n shortname: response.data.shortname,\n email: response.data.email,\n role: response.data.role,\n isActive: response.data.is_active,\n }\n authStore.setUserInfo(userInfo)\n\n return { success: true }\n } catch (error) {\n if (axios.isAxiosError(error) && error.response) {\n return { success: false, error: error.response.data.detail || 'Update failed' }\n }\n return { success: false, error: 'Network error. Please try again.' }\n }\n }\n\n // Set up periodic check as safety net (only inside component setup)\n let checkInterval: number | null = null\n\n if (getCurrentInstance()) {\n onMounted(() => {\n _mountedConsumerCount += 1\n checkInterval = window.setInterval(checkAndRefreshIfNeeded, TOKEN_REFRESH_CHECK_INTERVAL_MS)\n })\n\n onUnmounted(() => {\n if (checkInterval !== null) {\n window.clearInterval(checkInterval)\n checkInterval = null\n }\n\n _mountedConsumerCount = Math.max(0, _mountedConsumerCount - 1)\n if (_mountedConsumerCount === 0) {\n stopTokenRefresh()\n }\n })\n\n // Watch for token changes to reschedule refresh\n watch(\n () => authStore.tokenExpires,\n (newExpires) => {\n if (newExpires) {\n scheduleTokenRefresh()\n } else {\n stopTokenRefresh()\n }\n }\n )\n }\n\n return {\n // Core auth methods\n login,\n logout,\n register,\n verifyToken,\n fetchAuthConfig,\n initializeAuth,\n getCurrentUser,\n getAuthHeader,\n updateProfile,\n\n // Token refresh\n refreshToken,\n isRefreshing,\n }\n}\n","import axios from 'axios'\nimport { useAuthStore } from '../stores/auth'\nimport { useSettingsStore } from '../stores/settings'\nimport type { CredentialInfo } from '../types'\n\n// Lazy-load @simplewebauthn/browser so plugins that never call usePasskey\n// don't need the optional peer dep installed (avoids build failures).\nasync function loadWebAuthn() {\n try {\n return await import('@simplewebauthn/browser')\n } catch {\n throw new Error(\n '@simplewebauthn/browser is required for passkey support. Install it: bun add @simplewebauthn/browser',\n )\n }\n}\n\ninterface PasskeyLoginResponse {\n access_token: string\n token_type: string\n expires_in: number\n}\n\n/** Registers and authenticates passkeys using WebAuthn, lazily loading the browser dependency. */\nexport function usePasskey() {\n const authStore = useAuthStore()\n const settingsStore = useSettingsStore()\n\n function getApiBaseUrl(): string {\n return settingsStore.getApiBaseUrl()\n }\n\n async function isSupported(): Promise<boolean> {\n try {\n const { browserSupportsWebAuthn } = await loadWebAuthn()\n return browserSupportsWebAuthn()\n } catch {\n return false\n }\n }\n\n async function registerPasskey(deviceName?: string): Promise<boolean> {\n const webauthn = await loadWebAuthn()\n if (!webauthn.browserSupportsWebAuthn()) {\n authStore.setError('WebAuthn is not supported in this browser')\n return false\n }\n\n authStore.setLoading(true)\n authStore.setError(null)\n\n try {\n const optionsResponse = await axios.get<{ options: string }>(\n `${getApiBaseUrl()}/auth/passkey/register/options`,\n {\n headers: { Authorization: `Bearer ${authStore.token}` },\n withCredentials: true,\n }\n )\n\n const options = JSON.parse(optionsResponse.data.options)\n\n const credential = await webauthn.startRegistration(options)\n\n await axios.post(\n `${getApiBaseUrl()}/auth/passkey/register/verify`,\n {\n credential: JSON.stringify(credential),\n device_name: deviceName,\n },\n {\n headers: { Authorization: `Bearer ${authStore.token}` },\n withCredentials: true,\n }\n )\n\n authStore.setAuthConfig({\n ...authStore.authConfig,\n passkeyRegistered: true,\n })\n\n return true\n } catch (error) {\n if (axios.isAxiosError(error) && error.response) {\n authStore.setError(error.response.data.detail || 'Passkey registration failed')\n } else if (error instanceof Error) {\n if (error.name === 'NotAllowedError') {\n authStore.setError('Registration was cancelled or timed out')\n } else if (error.name === 'InvalidStateError') {\n authStore.setError('This authenticator is already registered')\n } else {\n authStore.setError(error.message)\n }\n } else {\n authStore.setError('Passkey registration failed')\n }\n return false\n } finally {\n authStore.setLoading(false)\n }\n }\n\n async function loginWithPasskey(): Promise<boolean> {\n const webauthn = await loadWebAuthn()\n if (!webauthn.browserSupportsWebAuthn()) {\n authStore.setError('WebAuthn is not supported in this browser')\n return false\n }\n\n authStore.setLoading(true)\n authStore.setError(null)\n\n try {\n const optionsResponse = await axios.get<{ options: string }>(\n `${getApiBaseUrl()}/auth/passkey/login/options`,\n { withCredentials: true }\n )\n\n const options = JSON.parse(optionsResponse.data.options)\n\n const credential = await webauthn.startAuthentication(options)\n\n const response = await axios.post<PasskeyLoginResponse>(\n `${getApiBaseUrl()}/auth/passkey/login/verify`,\n { credential: JSON.stringify(credential) },\n { withCredentials: true }\n )\n\n authStore.setToken(response.data.access_token, response.data.expires_in)\n\n return true\n } catch (error) {\n if (axios.isAxiosError(error) && error.response) {\n if (error.response.status === 404) {\n authStore.setError('No passkeys registered. Please login with password first.')\n } else {\n authStore.setError(error.response.data.detail || 'Passkey login failed')\n }\n } else if (error instanceof Error) {\n if (error.name === 'NotAllowedError') {\n authStore.setError('Authentication was cancelled or timed out')\n } else {\n authStore.setError(error.message)\n }\n } else {\n authStore.setError('Passkey login failed')\n }\n return false\n } finally {\n authStore.setLoading(false)\n }\n }\n\n async function listCredentials(): Promise<CredentialInfo[]> {\n try {\n const response = await axios.get<{ credentials: CredentialInfo[] }>(\n `${getApiBaseUrl()}/auth/passkey/credentials`,\n {\n headers: { Authorization: `Bearer ${authStore.token}` },\n }\n )\n return response.data.credentials\n } catch {\n return []\n }\n }\n\n async function deleteCredential(credentialId: string): Promise<boolean> {\n try {\n await axios.delete(\n `${getApiBaseUrl()}/auth/passkey/credentials/${encodeURIComponent(credentialId)}`,\n {\n headers: { Authorization: `Bearer ${authStore.token}` },\n }\n )\n\n const remaining = await listCredentials()\n if (remaining.length === 0) {\n authStore.setAuthConfig({\n ...authStore.authConfig,\n passkeyRegistered: false,\n })\n }\n\n return true\n } catch {\n return false\n }\n }\n\n async function deleteAllCredentials(): Promise<boolean> {\n try {\n await axios.delete(`${getApiBaseUrl()}/auth/passkey/credentials`, {\n headers: { Authorization: `Bearer ${authStore.token}` },\n })\n\n authStore.setAuthConfig({\n ...authStore.authConfig,\n passkeyRegistered: false,\n })\n\n return true\n } catch {\n return false\n }\n }\n\n return {\n isSupported,\n registerPasskey,\n loginWithPasskey,\n listCredentials,\n deleteCredential,\n deleteAllCredentials,\n }\n}\n","import { ref, computed, type Ref, type ComputedRef } from 'vue'\n\n/**\n * Error type for async operations.\n */\nexport interface AsyncError {\n message: string\n code?: string\n details?: Record<string, unknown>\n originalError?: unknown\n}\n\n/**\n * State of an async operation.\n */\nexport type AsyncState = 'idle' | 'loading' | 'success' | 'error'\n\n/**\n * Return type for useAsync composable.\n */\nexport interface UseAsyncReturn<T> {\n // Data\n data: Ref<T | null>\n error: Ref<AsyncError | null>\n\n // State\n state: Ref<AsyncState>\n isIdle: ComputedRef<boolean>\n isLoading: ComputedRef<boolean>\n isSuccess: ComputedRef<boolean>\n isError: ComputedRef<boolean>\n\n // Methods\n execute: (...args: unknown[]) => Promise<T | null>\n reset: () => void\n setData: (data: T | null) => void\n setError: (error: AsyncError | null) => void\n}\n\n/**\n * Options for useAsync.\n */\nexport interface UseAsyncOptions<T> {\n // Initial data value\n initialData?: T | null\n\n // Whether to run immediately on creation\n immediate?: boolean\n\n // Arguments for immediate execution\n immediateArgs?: unknown[]\n\n // Transform error into AsyncError\n transformError?: (error: unknown) => AsyncError\n\n // Called on success\n onSuccess?: (data: T) => void\n\n // Called on error\n onError?: (error: AsyncError) => void\n\n // Reset data on new execution\n resetOnExecute?: boolean\n}\n\n/**\n * Default error transformer.\n */\nfunction defaultTransformError(error: unknown): AsyncError {\n if (error instanceof Error) {\n return {\n message: error.message,\n originalError: error,\n }\n }\n\n if (typeof error === 'object' && error !== null) {\n const errorObj = error as Record<string, unknown>\n\n // Handle Axios-like errors\n if ('response' in errorObj && errorObj.response) {\n const response = errorObj.response as Record<string, unknown>\n const data = response.data as Record<string, unknown> | undefined\n\n return {\n message: (data?.detail as string) || (data?.message as string) || 'Request failed',\n code: String(response.status),\n details: data,\n originalError: error,\n }\n }\n\n // Handle generic error objects\n return {\n message: (errorObj.message as string) || 'Unknown error',\n code: errorObj.code as string | undefined,\n details: errorObj,\n originalError: error,\n }\n }\n\n return {\n message: String(error),\n originalError: error,\n }\n}\n\n/**\n * Composable for managing async operation state.\n *\n * Provides standardized loading, error, and success state management\n * for any async function.\n *\n * @param asyncFn - The async function to wrap\n * @param options - Configuration options\n *\n * @example\n * ```typescript\n * // Basic usage\n * const { data, isLoading, error, execute } = useAsync(\n * async (id: string) => {\n * const response = await api.get(`/users/${id}`)\n * return response.data\n * }\n * )\n *\n * // Execute the function\n * await execute('user-123')\n *\n * // In template\n * <div v-if=\"isLoading\">Loading...</div>\n * <div v-else-if=\"error\">{{ error.message }}</div>\n * <div v-else-if=\"data\">{{ data.name }}</div>\n * ```\n *\n * @example\n * ```typescript\n * // With options\n * const { data, execute } = useAsync(\n * fetchUser,\n * {\n * immediate: true,\n * immediateArgs: ['default-user'],\n * onSuccess: (user) => console.log('Fetched:', user.name),\n * onError: (error) => toast.error(error.message),\n * }\n * )\n * ```\n *\n * @example\n * ```typescript\n * // Form submission\n * const { isLoading, error, execute: submit } = useAsync(\n * async (formData: FormData) => {\n * await api.post('/submit', formData)\n * },\n * {\n * onSuccess: () => {\n * toast.success('Submitted!')\n * router.push('/success')\n * },\n * }\n * )\n *\n * const handleSubmit = () => submit(new FormData(formRef.value))\n * ```\n */\n/** Wraps an async function with reactive loading, error, and success state plus optional auto-execute. */\nexport function useAsync<T>(\n asyncFn: (...args: unknown[]) => Promise<T>,\n options: UseAsyncOptions<T> = {}\n): UseAsyncReturn<T> {\n const {\n initialData = null,\n immediate = false,\n immediateArgs = [],\n transformError = defaultTransformError,\n onSuccess,\n onError,\n resetOnExecute = false,\n } = options\n\n // State\n const data = ref<T | null>(initialData) as Ref<T | null>\n const error = ref<AsyncError | null>(null)\n const state = ref<AsyncState>('idle')\n\n // Computed state helpers\n const isIdle = computed(() => state.value === 'idle')\n const isLoading = computed(() => state.value === 'loading')\n const isSuccess = computed(() => state.value === 'success')\n const isError = computed(() => state.value === 'error')\n\n // Execute the async function\n async function execute(...args: unknown[]): Promise<T | null> {\n state.value = 'loading'\n error.value = null\n\n if (resetOnExecute) {\n data.value = null\n }\n\n try {\n const result = await asyncFn(...args)\n data.value = result\n state.value = 'success'\n onSuccess?.(result)\n return result\n } catch (e) {\n const asyncError = transformError(e)\n error.value = asyncError\n state.value = 'error'\n onError?.(asyncError)\n return null\n }\n }\n\n // Reset to initial state\n function reset(): void {\n data.value = initialData\n error.value = null\n state.value = 'idle'\n }\n\n // Manual data setter\n function setData(newData: T | null): void {\n data.value = newData\n if (newData !== null) {\n state.value = 'success'\n error.value = null\n }\n }\n\n // Manual error setter\n function setError(newError: AsyncError | null): void {\n error.value = newError\n if (newError !== null) {\n state.value = 'error'\n }\n }\n\n // Execute immediately if requested\n if (immediate) {\n execute(...immediateArgs)\n }\n\n return {\n data,\n error,\n state,\n isIdle,\n isLoading,\n isSuccess,\n isError,\n execute,\n reset,\n setData,\n setError,\n }\n}\n\n/**\n * Create a batch of async operations that can be executed in parallel.\n *\n * @example\n * ```typescript\n * const { results, isLoading, execute } = useAsyncBatch([\n * () => fetchUser(userId),\n * () => fetchPosts(userId),\n * () => fetchComments(userId),\n * ])\n *\n * await execute()\n * // results.value = [user, posts, comments]\n * ```\n */\nexport function useAsyncBatch<T extends readonly (() => Promise<unknown>)[]>(\n asyncFns: T\n): {\n results: Ref<{ [K in keyof T]: Awaited<ReturnType<T[K]>> | null }>\n errors: Ref<(AsyncError | null)[]>\n isLoading: Ref<boolean>\n execute: () => Promise<void>\n reset: () => void\n} {\n type Results = { [K in keyof T]: Awaited<ReturnType<T[K]>> | null }\n\n const results = ref<Results>(asyncFns.map(() => null) as unknown as Results) as Ref<Results>\n const errors = ref<(AsyncError | null)[]>(asyncFns.map(() => null))\n const isLoading = ref(false)\n\n async function execute(): Promise<void> {\n isLoading.value = true\n errors.value = asyncFns.map(() => null)\n\n const promises = asyncFns.map(async (fn, index) => {\n try {\n const result = await fn()\n ;(results.value as unknown[])[index] = result\n } catch (e) {\n errors.value[index] = defaultTransformError(e)\n ;(results.value as unknown[])[index] = null\n }\n })\n\n await Promise.all(promises)\n isLoading.value = false\n }\n\n function reset(): void {\n results.value = asyncFns.map(() => null) as unknown as Results\n errors.value = asyncFns.map(() => null)\n isLoading.value = false\n }\n\n return {\n results,\n errors,\n isLoading,\n execute,\n reset,\n }\n}\n","import { ref, computed, onMounted, type Ref, type ComputedRef } from 'vue'\nimport { useApi } from './useApi'\nimport { usePlatformContext } from './usePlatformContext'\n\nexport interface UsePluginConfigReturn {\n config: Ref<Record<string, unknown>>\n isLoading: Ref<boolean>\n isSaving: Ref<boolean>\n error: Ref<string | null>\n isDirty: ComputedRef<boolean>\n load: () => Promise<void>\n save: () => Promise<boolean>\n reset: () => void\n}\n\n/** Loads, saves, and tracks dirty state for a plugin's persistent configuration via the platform API. */\nexport function usePluginConfig(pluginName?: string): UsePluginConfigReturn {\n const api = useApi()\n const { plugin } = usePlatformContext()\n\n const resolvedName = computed(() => pluginName ?? plugin.value?.name ?? '')\n\n const config = ref<Record<string, unknown>>({})\n const savedConfig = ref<Record<string, unknown>>({})\n const isLoading = ref(false)\n const isSaving = ref(false)\n const error = ref<string | null>(null)\n\n const isDirty = computed(() => {\n return JSON.stringify(config.value) !== JSON.stringify(savedConfig.value)\n })\n\n async function load(): Promise<void> {\n const name = resolvedName.value\n if (!name) return\n\n isLoading.value = true\n error.value = null\n try {\n const response = await api.get<{ plugin_name: string; config: Record<string, unknown> }>(\n `/plugins/${encodeURIComponent(name)}/config`,\n )\n config.value = { ...response.config }\n savedConfig.value = { ...response.config }\n } catch (e) {\n error.value = e instanceof Error ? e.message : 'Failed to load plugin config'\n } finally {\n isLoading.value = false\n }\n }\n\n async function save(): Promise<boolean> {\n const name = resolvedName.value\n if (!name) return false\n\n isSaving.value = true\n error.value = null\n try {\n const response = await api.patch<{ plugin_name: string; config: Record<string, unknown> }>(\n `/plugins/${encodeURIComponent(name)}/config`,\n { config: config.value },\n )\n config.value = { ...response.config }\n savedConfig.value = { ...response.config }\n return true\n } catch (e) {\n error.value = e instanceof Error ? e.message : 'Failed to save plugin config'\n return false\n } finally {\n isSaving.value = false\n }\n }\n\n function reset(): void {\n config.value = { ...savedConfig.value }\n error.value = null\n }\n\n onMounted(() => {\n load()\n })\n\n return {\n config,\n isLoading,\n isSaving,\n error,\n isDirty,\n load,\n save,\n reset,\n }\n}\n","import { ref, type Ref } from 'vue'\nimport { useApi } from './useApi'\n\nexport interface UseExperimentSaveOptions {\n /** Default plugin_id for all save/load calls. */\n pluginId?: string\n /** Default schema version for design saves (defaults to \"1.0\"). */\n schemaVersion?: string\n /** Override API base URL. */\n apiBaseUrl?: string\n}\n\nexport interface UseExperimentSaveReturn {\n /** Whether a save operation is in progress. */\n isSaving: Ref<boolean>\n /** Error message from the last failed operation, or null. */\n error: Ref<string | null>\n /** Timestamp of the last successful save, or null. */\n lastSavedAt: Ref<Date | null>\n /** Save design data for an experiment. */\n saveDesign: (experimentId: number, data: Record<string, unknown>) => Promise<boolean>\n /** Save analysis result for an experiment. */\n saveAnalysis: (experimentId: number, result: Record<string, unknown>) => Promise<boolean>\n /** Save design and/or analysis in one call. */\n save: (experimentId: number, opts: {\n design?: Record<string, unknown>\n analysis?: Record<string, unknown>\n }) => Promise<boolean>\n /** Load design data for an experiment. */\n loadDesign: (experimentId: number) => Promise<Record<string, unknown> | null>\n /** Load analysis result for an experiment. */\n loadAnalysis: (experimentId: number) => Promise<Record<string, unknown> | null>\n /** Delete design data for an experiment. */\n deleteDesign: (experimentId: number) => Promise<boolean>\n /** Delete analysis result for an experiment. */\n deleteAnalysis: (experimentId: number) => Promise<boolean>\n}\n\n/** Persists and loads experiment design and analysis data via the plugin's API endpoint. */\nexport function useExperimentSave(\n options: UseExperimentSaveOptions = {},\n): UseExperimentSaveReturn {\n const api = useApi({ baseUrl: options.apiBaseUrl })\n const pluginId = options.pluginId\n const schemaVersion = options.schemaVersion ?? '1.0'\n\n const isSaving = ref(false)\n const error = ref<string | null>(null)\n const lastSavedAt = ref<Date | null>(null)\n\n function setError(e: unknown): void {\n error.value = e instanceof Error ? e.message : 'Unknown error'\n }\n\n async function saveDesign(\n experimentId: number,\n data: Record<string, unknown>,\n ): Promise<boolean> {\n if (!pluginId) {\n error.value = 'pluginId is required for saveDesign'\n return false\n }\n isSaving.value = true\n error.value = null\n try {\n await api.put(`/experiments/${experimentId}/data`, {\n plugin_id: pluginId,\n data,\n schema_version: schemaVersion,\n })\n lastSavedAt.value = new Date()\n return true\n } catch (e) {\n setError(e)\n return false\n } finally {\n isSaving.value = false\n }\n }\n\n async function saveAnalysis(\n experimentId: number,\n result: Record<string, unknown>,\n ): Promise<boolean> {\n if (!pluginId) {\n error.value = 'pluginId is required for saveAnalysis'\n return false\n }\n isSaving.value = true\n error.value = null\n try {\n await api.put(`/experiments/${experimentId}/results/${pluginId}`, {\n result,\n })\n lastSavedAt.value = new Date()\n return true\n } catch (e) {\n setError(e)\n return false\n } finally {\n isSaving.value = false\n }\n }\n\n async function save(\n experimentId: number,\n opts: { design?: Record<string, unknown>; analysis?: Record<string, unknown> },\n ): Promise<boolean> {\n if ((opts.design || opts.analysis) && !pluginId) {\n error.value = 'pluginId is required for save'\n return false\n }\n isSaving.value = true\n error.value = null\n try {\n if (opts.design) {\n await api.put(`/experiments/${experimentId}/data`, {\n plugin_id: pluginId,\n data: opts.design,\n schema_version: schemaVersion,\n })\n }\n if (opts.analysis) {\n await api.put(`/experiments/${experimentId}/results/${pluginId}`, {\n result: opts.analysis,\n })\n }\n lastSavedAt.value = new Date()\n return true\n } catch (e) {\n setError(e)\n return false\n } finally {\n isSaving.value = false\n }\n }\n\n async function loadDesign(\n experimentId: number,\n ): Promise<Record<string, unknown> | null> {\n try {\n return await api.get<Record<string, unknown>>(`/experiments/${experimentId}/data`)\n } catch {\n return null\n }\n }\n\n async function loadAnalysis(\n experimentId: number,\n ): Promise<Record<string, unknown> | null> {\n if (!pluginId) return null\n try {\n return await api.get<Record<string, unknown>>(\n `/experiments/${experimentId}/results/${pluginId}`,\n )\n } catch {\n return null\n }\n }\n\n async function deleteDesign(experimentId: number): Promise<boolean> {\n try {\n await api.delete(`/experiments/${experimentId}/data`)\n return true\n } catch {\n return false\n }\n }\n\n async function deleteAnalysis(experimentId: number): Promise<boolean> {\n if (!pluginId) return false\n try {\n await api.delete(`/experiments/${experimentId}/results/${pluginId}`)\n return true\n } catch {\n return false\n }\n }\n\n return {\n isSaving,\n error,\n lastSavedAt,\n saveDesign,\n saveAnalysis,\n save,\n loadDesign,\n loadAnalysis,\n deleteDesign,\n deleteAnalysis,\n }\n}\n","import { useApi } from './useApi'\n\nexport interface UsePluginApiOptions {\n /**\n * Fallback API prefix when `VITE_API_PREFIX` is not set.\n * Typically your plugin's route prefix, e.g. `'/api/drp'`.\n */\n fallbackPrefix?: string\n}\n\n/**\n * Pre-configured API client for plugins.\n *\n * Resolves the base URL in priority order:\n * 1. `import.meta.env.VITE_API_PREFIX` (build-time env override)\n * 2. `fallbackPrefix` option (plugin's route prefix)\n * 3. `'/api'` (default)\n *\n * Eliminates the repeated pattern across plugins:\n * ```ts\n * const API_BASE = import.meta.env.VITE_API_PREFIX || '/api/my-plugin'\n * const api = useApi({ baseUrl: API_BASE })\n * ```\n *\n * @example\n * ```ts\n * const api = usePluginApi({ fallbackPrefix: '/api/drp' })\n * const data = await api.get('/sessions')\n * ```\n */\n/** Pre-configured API client resolving the plugin base URL from VITE_API_PREFIX or a fallback prefix. */\nexport function usePluginApi(options: UsePluginApiOptions = {}) {\n const baseUrl =\n (import.meta.env?.VITE_API_PREFIX as string | undefined) ||\n options.fallbackPrefix ||\n '/api'\n\n return useApi({ baseUrl })\n}\n"],"mappings":"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"}
1
+ {"version":3,"file":"composables-tiZqLu1M.js","names":[],"sources":["../src/composables/useAuth.ts","../src/composables/usePasskey.ts","../src/composables/useAsync.ts","../src/composables/usePluginConfig.ts","../src/composables/useExperimentSave.ts","../src/composables/usePluginApi.ts"],"sourcesContent":["import axios from 'axios'\nimport { ref, onMounted, onUnmounted, watch, getCurrentInstance, type Ref } from 'vue'\nimport { useAuthStore } from '../stores/auth'\nimport { useSettingsStore } from '../stores/settings'\nimport type { AuthConfig, UserInfo, LoginResponse, TokenVerifyResponse, UpdateProfileRequest } from '../types'\n\ninterface UserResponse {\n id: string\n username: string\n shortname: string | null\n email: string | null\n role: string\n is_active: boolean\n}\n\ninterface RefreshResponse {\n access_token: string\n expires_in: number\n token_type: string\n}\n\n// Token refresh configuration\nconst TOKEN_REFRESH_MARGIN_MS = 5 * 60 * 1000 // Refresh 5 minutes before expiry\nconst TOKEN_REFRESH_CHECK_INTERVAL_MS = 60 * 1000 // Check every minute\n\n/**\n * Authentication composable with automatic token refresh.\n *\n * Features:\n * - Automatic token refresh before expiration\n * - Login/logout/register functionality\n * - Token verification on startup\n * - User profile management\n *\n * @example\n * ```typescript\n * const { login, logout, isAuthenticated, user } = useAuth()\n *\n * // Login\n * const success = await login('username', 'password')\n *\n * // Automatic refresh is enabled by default\n * // Tokens are refreshed 5 minutes before expiration\n * ```\n */\nexport interface UseAuthReturn {\n login: (username: string, password: string) => Promise<boolean>\n logout: () => void\n register: (username: string, password: string, email?: string) => Promise<boolean>\n verifyToken: () => Promise<boolean>\n fetchAuthConfig: () => Promise<AuthConfig>\n initializeAuth: () => Promise<void>\n getCurrentUser: () => Promise<UserInfo | null>\n getAuthHeader: () => Record<string, string>\n updateProfile: (data: { email?: string; shortname?: string; currentPassword?: string; newPassword?: string }) => Promise<{ success: boolean; error?: string }>\n refreshToken: () => Promise<boolean>\n isRefreshing: Ref<boolean>\n}\n\n// Module-level singletons to prevent duplicate refresh requests and timers\n// across multiple useAuth() instances\nlet _refreshPromise: Promise<boolean> | null = null\nlet _refreshTimerId: number | null = null\nlet _mountedConsumerCount = 0\n\n/** @internal Reset module-level state — only for use in tests. */\nexport function _resetAuthStateForTesting(): void {\n _refreshPromise = null\n if (_refreshTimerId !== null) {\n clearTimeout(_refreshTimerId)\n _refreshTimerId = null\n }\n _mountedConsumerCount = 0\n}\n\n/** Manages authentication state with login/logout/register and automatic JWT token refresh. */\nexport function useAuth(): UseAuthReturn {\n const authStore = useAuthStore()\n const settingsStore = useSettingsStore()\n\n const isRefreshing = ref(false)\n\n function getApiBaseUrl(): string {\n return settingsStore.getApiBaseUrl()\n }\n\n async function fetchAuthConfig(): Promise<AuthConfig> {\n try {\n const response = await axios.get<{\n auth_required: boolean\n passkey_enabled: boolean\n passkey_registered?: boolean\n registration_enabled?: boolean\n database_mode?: string\n }>(`${getApiBaseUrl()}/setup/config/public`)\n\n const config: AuthConfig = {\n authRequired: response.data.auth_required,\n passkeyEnabled: response.data.passkey_enabled,\n passkeyRegistered: response.data.passkey_registered ?? false,\n registrationEnabled: response.data.registration_enabled ?? false,\n databaseMode: response.data.database_mode ?? 'none',\n }\n\n authStore.setAuthConfig(config)\n return config\n } catch (error) {\n console.error('Failed to fetch auth config:', error)\n return {\n authRequired: false,\n passkeyEnabled: false,\n passkeyRegistered: false,\n registrationEnabled: false,\n databaseMode: 'none',\n }\n }\n }\n\n async function login(username: string, password: string): Promise<boolean> {\n authStore.setLoading(true)\n authStore.setError(null)\n\n try {\n const response = await axios.post<LoginResponse>(\n `${getApiBaseUrl()}/auth/login`,\n { username, password }\n )\n\n authStore.setToken(response.data.access_token, response.data.expires_in)\n authStore.setUsername(username)\n\n await getCurrentUser()\n\n // Start auto-refresh after successful login\n scheduleTokenRefresh()\n\n return true\n } catch (error) {\n if (axios.isAxiosError(error) && error.response) {\n authStore.setError(error.response.data.detail || 'Login failed')\n } else {\n authStore.setError('Network error. Please try again.')\n }\n return false\n } finally {\n authStore.setLoading(false)\n }\n }\n\n async function register(username: string, password: string, email?: string): Promise<boolean> {\n authStore.setLoading(true)\n authStore.setError(null)\n\n try {\n await axios.post<UserResponse>(\n `${getApiBaseUrl()}/users/register`,\n { username, password, email }\n )\n\n return await login(username, password)\n } catch (error) {\n if (axios.isAxiosError(error) && error.response) {\n authStore.setError(error.response.data.detail || 'Registration failed')\n } else {\n authStore.setError('Network error. Please try again.')\n }\n return false\n } finally {\n authStore.setLoading(false)\n }\n }\n\n async function getCurrentUser(): Promise<UserInfo | null> {\n if (!authStore.token) {\n return null\n }\n\n try {\n const response = await axios.get<UserResponse>(\n `${getApiBaseUrl()}/users/me`,\n { headers: getAuthHeader() }\n )\n\n const userInfo: UserInfo = {\n id: response.data.id,\n username: response.data.username,\n shortname: response.data.shortname,\n email: response.data.email,\n role: response.data.role,\n isActive: response.data.is_active,\n }\n\n authStore.setUserInfo(userInfo)\n return userInfo\n } catch {\n return null\n }\n }\n\n async function verifyToken(): Promise<boolean> {\n if (!authStore.token) {\n return false\n }\n\n try {\n const response = await axios.get<TokenVerifyResponse>(\n `${getApiBaseUrl()}/auth/verify`,\n {\n headers: {\n Authorization: `Bearer ${authStore.token}`,\n },\n }\n )\n\n if (response.data.valid && response.data.username) {\n authStore.setUsername(response.data.username)\n return true\n }\n\n authStore.clearToken()\n return false\n } catch {\n authStore.clearToken()\n return false\n }\n }\n\n /**\n * Refresh the authentication token.\n * Called automatically before token expiration.\n * Uses promise caching to prevent concurrent refresh requests.\n */\n async function refreshToken(): Promise<boolean> {\n if (!authStore.token) return false\n if (_refreshPromise) return _refreshPromise\n\n _refreshPromise = (async () => {\n isRefreshing.value = true\n\n try {\n const response = await axios.post<RefreshResponse>(\n `${getApiBaseUrl()}/auth/refresh`,\n {},\n { headers: getAuthHeader() }\n )\n\n authStore.setToken(response.data.access_token, response.data.expires_in)\n\n // Reschedule next refresh\n scheduleTokenRefresh()\n\n return true\n } catch (error) {\n // If refresh fails, the token may have been revoked\n // Clear auth state and let user re-login\n if (axios.isAxiosError(error) && error.response?.status === 401) {\n console.warn('[Auth] Token refresh failed - session expired')\n authStore.clearToken()\n stopTokenRefresh()\n }\n return false\n } finally {\n isRefreshing.value = false\n _refreshPromise = null\n }\n })()\n\n return _refreshPromise\n }\n\n /**\n * Schedule automatic token refresh before expiration.\n */\n function scheduleTokenRefresh(): void {\n // Clear any existing timer\n stopTokenRefresh()\n\n if (!authStore.tokenExpires) {\n return\n }\n\n const expiresAt = authStore.tokenExpires.getTime()\n const refreshAt = expiresAt - TOKEN_REFRESH_MARGIN_MS\n const now = Date.now()\n\n if (refreshAt <= now) {\n // Token is already close to expiring or expired, refresh now\n refreshToken()\n return\n }\n\n // Schedule refresh\n const delay = refreshAt - now\n _refreshTimerId = window.setTimeout(() => {\n refreshToken()\n }, delay)\n }\n\n /**\n * Stop automatic token refresh.\n */\n function stopTokenRefresh(): void {\n if (_refreshTimerId !== null) {\n window.clearTimeout(_refreshTimerId)\n _refreshTimerId = null\n }\n }\n\n /**\n * Check if token needs refresh and refresh if necessary.\n * Called periodically as a safety net.\n */\n function checkAndRefreshIfNeeded(): void {\n if (!authStore.token || !authStore.tokenExpires) {\n return\n }\n\n const expiresAt = authStore.tokenExpires.getTime()\n const refreshAt = expiresAt - TOKEN_REFRESH_MARGIN_MS\n const now = Date.now()\n\n if (now >= refreshAt) {\n refreshToken()\n }\n }\n\n async function initializeAuth(): Promise<void> {\n authStore.initialize()\n await fetchAuthConfig()\n\n if (authStore.token) {\n const valid = await verifyToken()\n if (valid) {\n await getCurrentUser()\n // Start auto-refresh for existing session\n scheduleTokenRefresh()\n }\n }\n }\n\n function logout(): void {\n stopTokenRefresh()\n authStore.logout()\n }\n\n function getAuthHeader(): Record<string, string> {\n if (authStore.token) {\n return { Authorization: `Bearer ${authStore.token}` }\n }\n return {}\n }\n\n async function updateProfile(data: {\n email?: string\n shortname?: string\n currentPassword?: string\n newPassword?: string\n }): Promise<{ success: boolean; error?: string }> {\n if (!authStore.token) {\n return { success: false, error: 'Not authenticated' }\n }\n\n try {\n const requestData: UpdateProfileRequest = {}\n if (data.email !== undefined) requestData.email = data.email\n if (data.shortname !== undefined) requestData.shortname = data.shortname\n if (data.currentPassword) requestData.current_password = data.currentPassword\n if (data.newPassword) requestData.new_password = data.newPassword\n\n const response = await axios.put<UserResponse>(\n `${getApiBaseUrl()}/users/me`,\n requestData,\n { headers: getAuthHeader() }\n )\n\n const userInfo: UserInfo = {\n id: response.data.id,\n username: response.data.username,\n shortname: response.data.shortname,\n email: response.data.email,\n role: response.data.role,\n isActive: response.data.is_active,\n }\n authStore.setUserInfo(userInfo)\n\n return { success: true }\n } catch (error) {\n if (axios.isAxiosError(error) && error.response) {\n return { success: false, error: error.response.data.detail || 'Update failed' }\n }\n return { success: false, error: 'Network error. Please try again.' }\n }\n }\n\n // Set up periodic check as safety net (only inside component setup)\n let checkInterval: number | null = null\n\n if (getCurrentInstance()) {\n onMounted(() => {\n _mountedConsumerCount += 1\n checkInterval = window.setInterval(checkAndRefreshIfNeeded, TOKEN_REFRESH_CHECK_INTERVAL_MS)\n })\n\n onUnmounted(() => {\n if (checkInterval !== null) {\n window.clearInterval(checkInterval)\n checkInterval = null\n }\n\n _mountedConsumerCount = Math.max(0, _mountedConsumerCount - 1)\n if (_mountedConsumerCount === 0) {\n stopTokenRefresh()\n }\n })\n\n // Watch for token changes to reschedule refresh\n watch(\n () => authStore.tokenExpires,\n (newExpires) => {\n if (newExpires) {\n scheduleTokenRefresh()\n } else {\n stopTokenRefresh()\n }\n }\n )\n }\n\n return {\n // Core auth methods\n login,\n logout,\n register,\n verifyToken,\n fetchAuthConfig,\n initializeAuth,\n getCurrentUser,\n getAuthHeader,\n updateProfile,\n\n // Token refresh\n refreshToken,\n isRefreshing,\n }\n}\n","import axios from 'axios'\nimport { useAuthStore } from '../stores/auth'\nimport { useSettingsStore } from '../stores/settings'\nimport type { CredentialInfo } from '../types'\n\n// Lazy-load @simplewebauthn/browser so plugins that never call usePasskey\n// don't need the optional peer dep installed (avoids build failures).\nasync function loadWebAuthn() {\n try {\n return await import('@simplewebauthn/browser')\n } catch {\n throw new Error(\n '@simplewebauthn/browser is required for passkey support. Install it: bun add @simplewebauthn/browser',\n )\n }\n}\n\ninterface PasskeyLoginResponse {\n access_token: string\n token_type: string\n expires_in: number\n}\n\n/** Registers and authenticates passkeys using WebAuthn, lazily loading the browser dependency. */\nexport function usePasskey() {\n const authStore = useAuthStore()\n const settingsStore = useSettingsStore()\n\n function getApiBaseUrl(): string {\n return settingsStore.getApiBaseUrl()\n }\n\n async function isSupported(): Promise<boolean> {\n try {\n const { browserSupportsWebAuthn } = await loadWebAuthn()\n return browserSupportsWebAuthn()\n } catch {\n return false\n }\n }\n\n async function registerPasskey(deviceName?: string): Promise<boolean> {\n const webauthn = await loadWebAuthn()\n if (!webauthn.browserSupportsWebAuthn()) {\n authStore.setError('WebAuthn is not supported in this browser')\n return false\n }\n\n authStore.setLoading(true)\n authStore.setError(null)\n\n try {\n const optionsResponse = await axios.get<{ options: string }>(\n `${getApiBaseUrl()}/auth/passkey/register/options`,\n {\n headers: { Authorization: `Bearer ${authStore.token}` },\n withCredentials: true,\n }\n )\n\n const options = JSON.parse(optionsResponse.data.options)\n\n const credential = await webauthn.startRegistration(options)\n\n await axios.post(\n `${getApiBaseUrl()}/auth/passkey/register/verify`,\n {\n credential: JSON.stringify(credential),\n device_name: deviceName,\n },\n {\n headers: { Authorization: `Bearer ${authStore.token}` },\n withCredentials: true,\n }\n )\n\n authStore.setAuthConfig({\n ...authStore.authConfig,\n passkeyRegistered: true,\n })\n\n return true\n } catch (error) {\n if (axios.isAxiosError(error) && error.response) {\n authStore.setError(error.response.data.detail || 'Passkey registration failed')\n } else if (error instanceof Error) {\n if (error.name === 'NotAllowedError') {\n authStore.setError('Registration was cancelled or timed out')\n } else if (error.name === 'InvalidStateError') {\n authStore.setError('This authenticator is already registered')\n } else {\n authStore.setError(error.message)\n }\n } else {\n authStore.setError('Passkey registration failed')\n }\n return false\n } finally {\n authStore.setLoading(false)\n }\n }\n\n async function loginWithPasskey(): Promise<boolean> {\n const webauthn = await loadWebAuthn()\n if (!webauthn.browserSupportsWebAuthn()) {\n authStore.setError('WebAuthn is not supported in this browser')\n return false\n }\n\n authStore.setLoading(true)\n authStore.setError(null)\n\n try {\n const optionsResponse = await axios.get<{ options: string }>(\n `${getApiBaseUrl()}/auth/passkey/login/options`,\n { withCredentials: true }\n )\n\n const options = JSON.parse(optionsResponse.data.options)\n\n const credential = await webauthn.startAuthentication(options)\n\n const response = await axios.post<PasskeyLoginResponse>(\n `${getApiBaseUrl()}/auth/passkey/login/verify`,\n { credential: JSON.stringify(credential) },\n { withCredentials: true }\n )\n\n authStore.setToken(response.data.access_token, response.data.expires_in)\n\n return true\n } catch (error) {\n if (axios.isAxiosError(error) && error.response) {\n if (error.response.status === 404) {\n authStore.setError('No passkeys registered. Please login with password first.')\n } else {\n authStore.setError(error.response.data.detail || 'Passkey login failed')\n }\n } else if (error instanceof Error) {\n if (error.name === 'NotAllowedError') {\n authStore.setError('Authentication was cancelled or timed out')\n } else {\n authStore.setError(error.message)\n }\n } else {\n authStore.setError('Passkey login failed')\n }\n return false\n } finally {\n authStore.setLoading(false)\n }\n }\n\n async function listCredentials(): Promise<CredentialInfo[]> {\n try {\n const response = await axios.get<{ credentials: CredentialInfo[] }>(\n `${getApiBaseUrl()}/auth/passkey/credentials`,\n {\n headers: { Authorization: `Bearer ${authStore.token}` },\n }\n )\n return response.data.credentials\n } catch {\n return []\n }\n }\n\n async function deleteCredential(credentialId: string): Promise<boolean> {\n try {\n await axios.delete(\n `${getApiBaseUrl()}/auth/passkey/credentials/${encodeURIComponent(credentialId)}`,\n {\n headers: { Authorization: `Bearer ${authStore.token}` },\n }\n )\n\n const remaining = await listCredentials()\n if (remaining.length === 0) {\n authStore.setAuthConfig({\n ...authStore.authConfig,\n passkeyRegistered: false,\n })\n }\n\n return true\n } catch {\n return false\n }\n }\n\n async function deleteAllCredentials(): Promise<boolean> {\n try {\n await axios.delete(`${getApiBaseUrl()}/auth/passkey/credentials`, {\n headers: { Authorization: `Bearer ${authStore.token}` },\n })\n\n authStore.setAuthConfig({\n ...authStore.authConfig,\n passkeyRegistered: false,\n })\n\n return true\n } catch {\n return false\n }\n }\n\n return {\n isSupported,\n registerPasskey,\n loginWithPasskey,\n listCredentials,\n deleteCredential,\n deleteAllCredentials,\n }\n}\n","import { ref, computed, type Ref, type ComputedRef } from 'vue'\n\n/**\n * Error type for async operations.\n */\nexport interface AsyncError {\n message: string\n code?: string\n details?: Record<string, unknown>\n originalError?: unknown\n}\n\n/**\n * State of an async operation.\n */\nexport type AsyncState = 'idle' | 'loading' | 'success' | 'error'\n\n/**\n * Return type for useAsync composable.\n */\nexport interface UseAsyncReturn<T> {\n // Data\n data: Ref<T | null>\n error: Ref<AsyncError | null>\n\n // State\n state: Ref<AsyncState>\n isIdle: ComputedRef<boolean>\n isLoading: ComputedRef<boolean>\n isSuccess: ComputedRef<boolean>\n isError: ComputedRef<boolean>\n\n // Methods\n execute: (...args: unknown[]) => Promise<T | null>\n reset: () => void\n setData: (data: T | null) => void\n setError: (error: AsyncError | null) => void\n}\n\n/**\n * Options for useAsync.\n */\nexport interface UseAsyncOptions<T> {\n // Initial data value\n initialData?: T | null\n\n // Whether to run immediately on creation\n immediate?: boolean\n\n // Arguments for immediate execution\n immediateArgs?: unknown[]\n\n // Transform error into AsyncError\n transformError?: (error: unknown) => AsyncError\n\n // Called on success\n onSuccess?: (data: T) => void\n\n // Called on error\n onError?: (error: AsyncError) => void\n\n // Reset data on new execution\n resetOnExecute?: boolean\n}\n\n/**\n * Default error transformer.\n */\nfunction defaultTransformError(error: unknown): AsyncError {\n if (error instanceof Error) {\n return {\n message: error.message,\n originalError: error,\n }\n }\n\n if (typeof error === 'object' && error !== null) {\n const errorObj = error as Record<string, unknown>\n\n // Handle Axios-like errors\n if ('response' in errorObj && errorObj.response) {\n const response = errorObj.response as Record<string, unknown>\n const data = response.data as Record<string, unknown> | undefined\n\n return {\n message: (data?.detail as string) || (data?.message as string) || 'Request failed',\n code: String(response.status),\n details: data,\n originalError: error,\n }\n }\n\n // Handle generic error objects\n return {\n message: (errorObj.message as string) || 'Unknown error',\n code: errorObj.code as string | undefined,\n details: errorObj,\n originalError: error,\n }\n }\n\n return {\n message: String(error),\n originalError: error,\n }\n}\n\n/**\n * Composable for managing async operation state.\n *\n * Provides standardized loading, error, and success state management\n * for any async function.\n *\n * @param asyncFn - The async function to wrap\n * @param options - Configuration options\n *\n * @example\n * ```typescript\n * // Basic usage\n * const { data, isLoading, error, execute } = useAsync(\n * async (id: string) => {\n * const response = await api.get(`/users/${id}`)\n * return response.data\n * }\n * )\n *\n * // Execute the function\n * await execute('user-123')\n *\n * // In template\n * <div v-if=\"isLoading\">Loading...</div>\n * <div v-else-if=\"error\">{{ error.message }}</div>\n * <div v-else-if=\"data\">{{ data.name }}</div>\n * ```\n *\n * @example\n * ```typescript\n * // With options\n * const { data, execute } = useAsync(\n * fetchUser,\n * {\n * immediate: true,\n * immediateArgs: ['default-user'],\n * onSuccess: (user) => console.log('Fetched:', user.name),\n * onError: (error) => toast.error(error.message),\n * }\n * )\n * ```\n *\n * @example\n * ```typescript\n * // Form submission\n * const { isLoading, error, execute: submit } = useAsync(\n * async (formData: FormData) => {\n * await api.post('/submit', formData)\n * },\n * {\n * onSuccess: () => {\n * toast.success('Submitted!')\n * router.push('/success')\n * },\n * }\n * )\n *\n * const handleSubmit = () => submit(new FormData(formRef.value))\n * ```\n */\n/** Wraps an async function with reactive loading, error, and success state plus optional auto-execute. */\nexport function useAsync<T>(\n asyncFn: (...args: unknown[]) => Promise<T>,\n options: UseAsyncOptions<T> = {}\n): UseAsyncReturn<T> {\n const {\n initialData = null,\n immediate = false,\n immediateArgs = [],\n transformError = defaultTransformError,\n onSuccess,\n onError,\n resetOnExecute = false,\n } = options\n\n // State\n const data = ref<T | null>(initialData) as Ref<T | null>\n const error = ref<AsyncError | null>(null)\n const state = ref<AsyncState>('idle')\n\n // Computed state helpers\n const isIdle = computed(() => state.value === 'idle')\n const isLoading = computed(() => state.value === 'loading')\n const isSuccess = computed(() => state.value === 'success')\n const isError = computed(() => state.value === 'error')\n\n // Execute the async function\n async function execute(...args: unknown[]): Promise<T | null> {\n state.value = 'loading'\n error.value = null\n\n if (resetOnExecute) {\n data.value = null\n }\n\n try {\n const result = await asyncFn(...args)\n data.value = result\n state.value = 'success'\n onSuccess?.(result)\n return result\n } catch (e) {\n const asyncError = transformError(e)\n error.value = asyncError\n state.value = 'error'\n onError?.(asyncError)\n return null\n }\n }\n\n // Reset to initial state\n function reset(): void {\n data.value = initialData\n error.value = null\n state.value = 'idle'\n }\n\n // Manual data setter\n function setData(newData: T | null): void {\n data.value = newData\n if (newData !== null) {\n state.value = 'success'\n error.value = null\n }\n }\n\n // Manual error setter\n function setError(newError: AsyncError | null): void {\n error.value = newError\n if (newError !== null) {\n state.value = 'error'\n }\n }\n\n // Execute immediately if requested\n if (immediate) {\n execute(...immediateArgs)\n }\n\n return {\n data,\n error,\n state,\n isIdle,\n isLoading,\n isSuccess,\n isError,\n execute,\n reset,\n setData,\n setError,\n }\n}\n\n/**\n * Create a batch of async operations that can be executed in parallel.\n *\n * @example\n * ```typescript\n * const { results, isLoading, execute } = useAsyncBatch([\n * () => fetchUser(userId),\n * () => fetchPosts(userId),\n * () => fetchComments(userId),\n * ])\n *\n * await execute()\n * // results.value = [user, posts, comments]\n * ```\n */\nexport function useAsyncBatch<T extends readonly (() => Promise<unknown>)[]>(\n asyncFns: T\n): {\n results: Ref<{ [K in keyof T]: Awaited<ReturnType<T[K]>> | null }>\n errors: Ref<(AsyncError | null)[]>\n isLoading: Ref<boolean>\n execute: () => Promise<void>\n reset: () => void\n} {\n type Results = { [K in keyof T]: Awaited<ReturnType<T[K]>> | null }\n\n const results = ref<Results>(asyncFns.map(() => null) as unknown as Results) as Ref<Results>\n const errors = ref<(AsyncError | null)[]>(asyncFns.map(() => null))\n const isLoading = ref(false)\n\n async function execute(): Promise<void> {\n isLoading.value = true\n errors.value = asyncFns.map(() => null)\n\n const promises = asyncFns.map(async (fn, index) => {\n try {\n const result = await fn()\n ;(results.value as unknown[])[index] = result\n } catch (e) {\n errors.value[index] = defaultTransformError(e)\n ;(results.value as unknown[])[index] = null\n }\n })\n\n await Promise.all(promises)\n isLoading.value = false\n }\n\n function reset(): void {\n results.value = asyncFns.map(() => null) as unknown as Results\n errors.value = asyncFns.map(() => null)\n isLoading.value = false\n }\n\n return {\n results,\n errors,\n isLoading,\n execute,\n reset,\n }\n}\n","import { ref, computed, onMounted, type Ref, type ComputedRef } from 'vue'\nimport { useApi } from './useApi'\nimport { usePlatformContext } from './usePlatformContext'\n\nexport interface UsePluginConfigReturn {\n config: Ref<Record<string, unknown>>\n isLoading: Ref<boolean>\n isSaving: Ref<boolean>\n error: Ref<string | null>\n isDirty: ComputedRef<boolean>\n load: () => Promise<void>\n save: () => Promise<boolean>\n reset: () => void\n}\n\n/** Loads, saves, and tracks dirty state for a plugin's persistent configuration via the platform API. */\nexport function usePluginConfig(pluginName?: string): UsePluginConfigReturn {\n const api = useApi()\n const { plugin } = usePlatformContext()\n\n const resolvedName = computed(() => pluginName ?? plugin.value?.name ?? '')\n\n const config = ref<Record<string, unknown>>({})\n const savedConfig = ref<Record<string, unknown>>({})\n const isLoading = ref(false)\n const isSaving = ref(false)\n const error = ref<string | null>(null)\n\n const isDirty = computed(() => {\n return JSON.stringify(config.value) !== JSON.stringify(savedConfig.value)\n })\n\n async function load(): Promise<void> {\n const name = resolvedName.value\n if (!name) return\n\n isLoading.value = true\n error.value = null\n try {\n const response = await api.get<{ plugin_name: string; config: Record<string, unknown> }>(\n `/plugins/${encodeURIComponent(name)}/config`,\n )\n config.value = { ...response.config }\n savedConfig.value = { ...response.config }\n } catch (e) {\n error.value = e instanceof Error ? e.message : 'Failed to load plugin config'\n } finally {\n isLoading.value = false\n }\n }\n\n async function save(): Promise<boolean> {\n const name = resolvedName.value\n if (!name) return false\n\n isSaving.value = true\n error.value = null\n try {\n const response = await api.patch<{ plugin_name: string; config: Record<string, unknown> }>(\n `/plugins/${encodeURIComponent(name)}/config`,\n { config: config.value },\n )\n config.value = { ...response.config }\n savedConfig.value = { ...response.config }\n return true\n } catch (e) {\n error.value = e instanceof Error ? e.message : 'Failed to save plugin config'\n return false\n } finally {\n isSaving.value = false\n }\n }\n\n function reset(): void {\n config.value = { ...savedConfig.value }\n error.value = null\n }\n\n onMounted(() => {\n load()\n })\n\n return {\n config,\n isLoading,\n isSaving,\n error,\n isDirty,\n load,\n save,\n reset,\n }\n}\n","import { ref, type Ref } from 'vue'\nimport { useApi } from './useApi'\n\nexport interface UseExperimentSaveOptions {\n /** Default plugin_id for all save/load calls. */\n pluginId?: string\n /** Default schema version for design saves (defaults to \"1.0\"). */\n schemaVersion?: string\n /** Override API base URL. */\n apiBaseUrl?: string\n}\n\nexport interface UseExperimentSaveReturn {\n /** Whether a save operation is in progress. */\n isSaving: Ref<boolean>\n /** Error message from the last failed operation, or null. */\n error: Ref<string | null>\n /** Timestamp of the last successful save, or null. */\n lastSavedAt: Ref<Date | null>\n /** Save design data for an experiment. */\n saveDesign: (experimentId: number, data: Record<string, unknown>) => Promise<boolean>\n /** Save analysis result for an experiment. */\n saveAnalysis: (experimentId: number, result: Record<string, unknown>) => Promise<boolean>\n /** Save design and/or analysis in one call. */\n save: (experimentId: number, opts: {\n design?: Record<string, unknown>\n analysis?: Record<string, unknown>\n }) => Promise<boolean>\n /** Load design data for an experiment. */\n loadDesign: (experimentId: number) => Promise<Record<string, unknown> | null>\n /** Load analysis result for an experiment. */\n loadAnalysis: (experimentId: number) => Promise<Record<string, unknown> | null>\n /** Delete design data for an experiment. */\n deleteDesign: (experimentId: number) => Promise<boolean>\n /** Delete analysis result for an experiment. */\n deleteAnalysis: (experimentId: number) => Promise<boolean>\n}\n\n/** Persists and loads experiment design and analysis data via the plugin's API endpoint. */\nexport function useExperimentSave(\n options: UseExperimentSaveOptions = {},\n): UseExperimentSaveReturn {\n const api = useApi({ baseUrl: options.apiBaseUrl })\n const pluginId = options.pluginId\n const schemaVersion = options.schemaVersion ?? '1.0'\n\n const isSaving = ref(false)\n const error = ref<string | null>(null)\n const lastSavedAt = ref<Date | null>(null)\n\n function setError(e: unknown): void {\n error.value = e instanceof Error ? e.message : 'Unknown error'\n }\n\n async function saveDesign(\n experimentId: number,\n data: Record<string, unknown>,\n ): Promise<boolean> {\n if (!pluginId) {\n error.value = 'pluginId is required for saveDesign'\n return false\n }\n isSaving.value = true\n error.value = null\n try {\n await api.put(`/experiments/${experimentId}/data`, {\n plugin_id: pluginId,\n data,\n schema_version: schemaVersion,\n })\n lastSavedAt.value = new Date()\n return true\n } catch (e) {\n setError(e)\n return false\n } finally {\n isSaving.value = false\n }\n }\n\n async function saveAnalysis(\n experimentId: number,\n result: Record<string, unknown>,\n ): Promise<boolean> {\n if (!pluginId) {\n error.value = 'pluginId is required for saveAnalysis'\n return false\n }\n isSaving.value = true\n error.value = null\n try {\n await api.put(`/experiments/${experimentId}/results/${pluginId}`, {\n result,\n })\n lastSavedAt.value = new Date()\n return true\n } catch (e) {\n setError(e)\n return false\n } finally {\n isSaving.value = false\n }\n }\n\n async function save(\n experimentId: number,\n opts: { design?: Record<string, unknown>; analysis?: Record<string, unknown> },\n ): Promise<boolean> {\n if ((opts.design || opts.analysis) && !pluginId) {\n error.value = 'pluginId is required for save'\n return false\n }\n isSaving.value = true\n error.value = null\n try {\n if (opts.design) {\n await api.put(`/experiments/${experimentId}/data`, {\n plugin_id: pluginId,\n data: opts.design,\n schema_version: schemaVersion,\n })\n }\n if (opts.analysis) {\n await api.put(`/experiments/${experimentId}/results/${pluginId}`, {\n result: opts.analysis,\n })\n }\n lastSavedAt.value = new Date()\n return true\n } catch (e) {\n setError(e)\n return false\n } finally {\n isSaving.value = false\n }\n }\n\n async function loadDesign(\n experimentId: number,\n ): Promise<Record<string, unknown> | null> {\n try {\n return await api.get<Record<string, unknown>>(`/experiments/${experimentId}/data`)\n } catch {\n return null\n }\n }\n\n async function loadAnalysis(\n experimentId: number,\n ): Promise<Record<string, unknown> | null> {\n if (!pluginId) return null\n try {\n return await api.get<Record<string, unknown>>(\n `/experiments/${experimentId}/results/${pluginId}`,\n )\n } catch {\n return null\n }\n }\n\n async function deleteDesign(experimentId: number): Promise<boolean> {\n try {\n await api.delete(`/experiments/${experimentId}/data`)\n return true\n } catch {\n return false\n }\n }\n\n async function deleteAnalysis(experimentId: number): Promise<boolean> {\n if (!pluginId) return false\n try {\n await api.delete(`/experiments/${experimentId}/results/${pluginId}`)\n return true\n } catch {\n return false\n }\n }\n\n return {\n isSaving,\n error,\n lastSavedAt,\n saveDesign,\n saveAnalysis,\n save,\n loadDesign,\n loadAnalysis,\n deleteDesign,\n deleteAnalysis,\n }\n}\n","import { useApi } from './useApi'\n\nexport interface UsePluginApiOptions {\n /**\n * Fallback API prefix when `VITE_API_PREFIX` is not set.\n * Typically your plugin's route prefix, e.g. `'/api/drp'`.\n */\n fallbackPrefix?: string\n}\n\n/**\n * Pre-configured API client for plugins.\n *\n * Resolves the base URL in priority order:\n * 1. `import.meta.env.VITE_API_PREFIX` (build-time env override)\n * 2. `fallbackPrefix` option (plugin's route prefix)\n * 3. `'/api'` (default)\n *\n * Eliminates the repeated pattern across plugins:\n * ```ts\n * const API_BASE = import.meta.env.VITE_API_PREFIX || '/api/my-plugin'\n * const api = useApi({ baseUrl: API_BASE })\n * ```\n *\n * @example\n * ```ts\n * const api = usePluginApi({ fallbackPrefix: '/api/drp' })\n * const data = await api.get('/sessions')\n * ```\n */\n/** Pre-configured API client resolving the plugin base URL from VITE_API_PREFIX or a fallback prefix. */\nexport function usePluginApi(options: UsePluginApiOptions = {}) {\n const baseUrl =\n (import.meta.env?.VITE_API_PREFIX as string | undefined) ||\n options.fallbackPrefix ||\n '/api'\n\n return useApi({ baseUrl })\n}\n"],"mappings":"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"}
package/dist/index.d.ts CHANGED
@@ -1,6 +1,6 @@
1
1
  export { MINTSdk, default } from './install';
2
- export { BaseButton, BaseInput, BaseTextarea, BaseSelect, BaseCheckbox, BaseToggle, BaseRadioGroup, BaseSlider, ColorSlider, BaseTabs, BaseModal, FormField, DatePicker, TimePicker, TagsInput, NumberInput, FileUploader, AlertBox, ToastNotification, AppToastContainer, IconButton, ThemeToggle, SettingsButton, CollapsibleCard, AppTopBar, AppPageSelector, AppPillNav, AppAvatarMenu, AppPluginSwitcher, AppSidebar, AppLayout, AppContainer, Skeleton, WellPlate, RackEditor, SampleLegend, PlateMapEditor, ExperimentTimeline, SampleSelector, GroupingModal, AutoGroupModal, GroupAssigner, MoleculeInput, ConcentrationInput, DoseCalculator, ReagentList, SampleHierarchyTree, ProtocolStepEditor, SegmentedControl, MultiSelect, BasePill, DropdownButton, Calendar, DataFrame, LoadingSpinner, Divider, StatusIndicator, ProgressBar, Avatar, EmptyState, Breadcrumb, Tooltip, ConfirmDialog, ChartContainer, SettingsModal, ScientificNumber, ChemicalFormula, FormulaInput, SequenceInput, UnitInput, StepWizard, AuditTrail, BatchProgressList, ExperimentDataViewer, ExperimentCodeBadge, DateTimePicker, TimeRangeInput, ScheduleCalendar, ResourceCard, ExperimentSelectorModal, ExperimentPopover, FitPanel, } from './components';
2
+ export { BaseButton, BaseInput, BaseTextarea, BaseSelect, BaseCheckbox, BaseToggle, BaseRadioGroup, BaseSlider, ColorSlider, BaseTabs, BaseModal, FormField, DatePicker, TimePicker, TagsInput, NumberInput, FileUploader, AlertBox, ToastNotification, AppToastContainer, IconButton, ThemeToggle, SettingsButton, CollapsibleCard, AppTopBar, AppPageSelector, AppPillNav, AppAvatarMenu, AppPluginSwitcher, AppSidebar, AppLayout, AppContainer, PluginIcon, Skeleton, WellPlate, RackEditor, SampleLegend, PlateMapEditor, ExperimentTimeline, SampleSelector, GroupingModal, AutoGroupModal, GroupAssigner, MoleculeInput, ConcentrationInput, DoseCalculator, ReagentList, SampleHierarchyTree, ProtocolStepEditor, SegmentedControl, MultiSelect, BasePill, DropdownButton, Calendar, DataFrame, LoadingSpinner, Divider, StatusIndicator, ProgressBar, Avatar, EmptyState, Breadcrumb, Tooltip, ConfirmDialog, ChartContainer, SettingsModal, ScientificNumber, ChemicalFormula, FormulaInput, SequenceInput, UnitInput, StepWizard, AuditTrail, BatchProgressList, ExperimentDataViewer, ExperimentCodeBadge, DateTimePicker, TimeRangeInput, ScheduleCalendar, ResourceCard, ExperimentSelectorModal, ExperimentPopover, FitPanel, } from './components';
3
3
  export { useApi, useAuth, usePasskey, useTheme, useToast, usePlatformContext, useWellPlateEditor, useConcentrationUnits, useDoseCalculator, useProtocolTemplates, useRackEditor, useChemicalFormula, ATOMIC_WEIGHTS, useSequenceUtils, type ApiClientOptions, type UseWellPlateEditorOptions, type UseWellPlateEditorReturn, type UseRackEditorOptions, type UseRackEditorReturn, type ConcentrationValue, type ConcentrationUnit, type VolumeValue, type VolumeUnit, type StepTemplate, type FormulaParseResult, type FormulaPart, type SequenceType, type SequenceStats, parseTime, formatTime, generateTimeSlots, rangesOverlap, durationMinutes, formatDuration, isTimeInRange, findAvailableSlots, snapToSlot, addMinutes, compareTime, useScheduleDrag, usePluginConfig, type UsePluginConfigReturn, useAutoGroup, DEFAULT_COLORS, useExperimentSelector, type UseExperimentSelectorOptions, type UseExperimentSelectorReturn, formatExperimentDate, datePresetToISO, EXPERIMENT_STATUS_OPTIONS, EXPERIMENT_STATUS_VARIANT_MAP, EXPERIMENT_STATUS_LABELS, DATE_PRESET_OPTIONS, SORT_OPTIONS, useExperimentData, type UseExperimentDataOptions, type UseExperimentDataReturn, useAppExperiment, APP_EXPERIMENT_KEY, type UseAppExperimentOptions, type UseAppExperimentReturn, useExperimentSave, type UseExperimentSaveOptions, type UseExperimentSaveReturn, usePluginApi, type UsePluginApiOptions, } from './composables';
4
4
  export { useAuthStore, useSettingsStore, colorPalettes, type SettingsState, } from './stores';
5
5
  export { hexToHsl, hslToHex, deriveShade, type Hsl, } from './utils/color';
6
- export type { ContainerDirection, ButtonVariant, ButtonSize, InputType, ModalSize, AlertType, Toast, TabItem, SelectOption, RadioOption, FormFieldProps, SidebarToolSection, CollapsibleState, TopBarVariant, TopBarPage, TopBarTab, TopBarTabOption, TopBarSettingsConfig, PillNavItem, PageSelectorItem, PluginSwitcherPlugin, PluginSwitcherInfo, AccountMenuItem, WellPlateFormat, WellState, WellPlateSelectionMode, Well, HeatmapColorScale, HeatmapConfig, SlotPosition, WellExtendedData, WellEditData, WellEditField, WellLegendItem, Rack, SampleType, PlateMap, PlateMapEditorState, ProtocolStepType, ProtocolStepStatus, ProtocolStep, SampleGroup, GroupItem, OutlierAction, InputMode, OutlierInfo, ColumnInfo, MetadataRow, AutoGroupResult, ParsedCsvData, FileUploaderMode, SegmentedOption, SegmentedControlVariant, SegmentedControlSize, MultiSelectOption, MultiSelectSize, PillVariant, PillSize, CalendarSelectionMode, CalendarMarker, CalendarDayContext, SortDirection, SortState, DataFrameColumn, PaginationState, SpinnerSize, SpinnerVariant, DividerSpacing, StatusType, ProgressVariant, ProgressSize, AvatarSize, EmptyStateColor, EmptyStateSize, BreadcrumbItem, TooltipPosition, ConfirmVariant, SettingsTab, NumberNotation, TimePickerFormat, TimeRange, ScheduleView, ScheduleEventStatus, ScheduleEvent, ScheduleBlockedSlot, ScheduleSlotContext, ScheduleEventCreateContext, ScheduleEventUpdateContext, ResourceStatus, ResourceSpec, ExperimentStatus, ExperimentSummary, ExperimentListResponse, ExperimentFilters, FitState, FitResultSummary, UnitOption, WizardStep, WizardStepState, AuditEntryType, AuditEntry, BatchItemStatus, BatchItem, BatchSummary, AuthConfig, UserInfo, LoginResponse, TokenVerifyResponse, RegisterRequest, UpdateProfileRequest, CredentialInfo, SummaryData, SummarySection, SummarySectionItem, TreeNode, TreeNodeType, PluginInfo, PluginNavItem, PluginSettings, PluginSettingField, PlatformContext, PlatformEventType, PlatformEvent, ThemeMode, ColorPalette, TableDensity, UserSummary, } from './types';
6
+ export type { ContainerDirection, ButtonVariant, ButtonSize, InputType, ModalSize, AlertType, Toast, TabItem, SelectOption, RadioOption, FormFieldProps, SidebarToolSection, CollapsibleState, TopBarVariant, TopBarPage, TopBarTab, TopBarTabOption, TopBarSettingsConfig, PillNavItem, PageSelectorItem, PluginSwitcherPlugin, PluginSwitcherInfo, AccountMenuItem, WellPlateFormat, WellState, WellPlateSelectionMode, Well, HeatmapColorScale, HeatmapConfig, SlotPosition, WellExtendedData, WellEditData, WellEditField, WellLegendItem, Rack, SampleType, PlateMap, PlateMapEditorState, ProtocolStepType, ProtocolStepStatus, ProtocolStep, SampleGroup, GroupItem, OutlierAction, InputMode, OutlierInfo, ColumnInfo, MetadataRow, AutoGroupResult, ParsedCsvData, FileUploaderMode, SegmentedOption, SegmentedControlVariant, SegmentedControlSize, MultiSelectOption, MultiSelectSize, PillVariant, PillSize, CalendarSelectionMode, CalendarMarker, CalendarDayContext, SortDirection, SortState, DataFrameColumn, PaginationState, SpinnerSize, SpinnerVariant, DividerSpacing, StatusType, ProgressVariant, ProgressSize, AvatarSize, EmptyStateColor, EmptyStateSize, BreadcrumbItem, TooltipPosition, ConfirmVariant, SettingsTab, SettingsModalLayout, SettingsGroup, SettingsModalSchema, NumberNotation, TimePickerFormat, TimeRange, ScheduleView, ScheduleEventStatus, ScheduleEvent, ScheduleBlockedSlot, ScheduleSlotContext, ScheduleEventCreateContext, ScheduleEventUpdateContext, ResourceStatus, ResourceSpec, ExperimentStatus, ExperimentSummary, ExperimentListResponse, ExperimentFilters, FitState, FitResultSummary, UnitOption, WizardStep, WizardStepState, AuditEntryType, AuditEntry, BatchItemStatus, BatchItem, BatchSummary, AuthConfig, UserInfo, LoginResponse, TokenVerifyResponse, RegisterRequest, UpdateProfileRequest, CredentialInfo, SummaryData, SummarySection, SummarySectionItem, TreeNode, TreeNodeType, PluginInfo, PluginNavItem, PluginSettings, PluginSettingField, PlatformContext, PlatformEventType, PlatformEvent, ThemeMode, ColorPalette, TableDensity, UserSummary, } from './types';
package/dist/index.js CHANGED
@@ -1,7 +1,7 @@
1
- import { $ as ExperimentPopover_default, A as SampleLegend_default, B as LoadingSpinner_default, C as deriveShade, D as StepWizard_default, E as AutoGroupModal_default, F as Breadcrumb_default, G as AppPluginSwitcher_default, H as AppLayout_default, I as Avatar_default, J as AppPageSelector_default, K as AppAvatarMenu_default, L as ProgressBar_default, M as WellPlate_default, N as ChartContainer_default, O as ExperimentTimeline_default, P as Tooltip_default, Q as Skeleton_default, R as StatusIndicator_default, S as SampleSelector_default, T as hslToHex, U as AppSidebar_default, V as AppContainer_default, W as AppTopBar_default, X as ExperimentCodeBadge_default, Y as ExperimentSelectorModal_default, Z as EmptyState_default, _ as ReagentList_default, _t as BaseButton_default, a as TimeRangeInput_default, at as IconButton_default, b as GroupAssigner_default, ct as FormField_default, d as BatchProgressList_default, dt as DropdownButton_default, et as ConfirmDialog_default, f as AuditTrail_default, ft as BasePill_default, g as SampleHierarchyTree_default, gt as ColorSlider_default, h as ProtocolStepEditor_default, ht as BaseTabs_default, i as ScheduleCalendar_default, it as ThemeToggle_default, j as RackEditor_default, k as PlateMapEditor_default, lt as DataFrame_default, m as ScientificNumber_default, mt as BaseModal_default, n as FitPanel_default, nt as CollapsibleCard_default, o as ExperimentDataViewer_default, ot as ToastNotification_default, p as ChemicalFormula_default, pt as SegmentedControl_default, q as AppPillNav_default, r as ResourceCard_default, rt as SettingsButton_default, st as AlertBox_default, tt as SettingsModal_default, ut as Calendar_default, w as hexToHsl, x as GroupingModal_default, y as DoseCalculator_default, z as Divider_default } from "./components-CzbQQPCb.js";
2
- import { $ as formatTime, A as APP_EXPERIMENT_KEY, B as formatExperimentDate, C as MoleculeInput_default, D as useAutoGroup, F as EXPERIMENT_STATUS_LABELS, G as NumberInput_default, H as useTheme, I as EXPERIMENT_STATUS_OPTIONS, J as addMinutes, K as TagsInput_default, L as EXPERIMENT_STATUS_VARIANT_MAP, M as usePlatformContext, N as useExperimentSelector, O as useWellPlateEditor, P as DATE_PRESET_OPTIONS, Q as formatDuration, R as SORT_OPTIONS, S as useConcentrationUnits, U as useToast, V as useApi, W as FileUploader_default, X as durationMinutes, Y as compareTime, Z as findAvailableSlots, b as useDoseCalculator, ct as BaseSlider_default, d as useSequenceUtils, dt as BaseCheckbox_default, et as generateTimeSlots, f as FormulaInput_default, ft as BaseSelect_default, h as useProtocolTemplates, it as snapToSlot, j as useAppExperiment, k as useRackEditor, l as UnitInput_default, lt as BaseRadioGroup_default, m as useChemicalFormula, mt as BaseInput_default, n as useExperimentData, nt as parseTime, ot as DatePicker_default, p as ATOMIC_WEIGHTS, pt as BaseTextarea_default, q as TimePicker_default, rt as rangesOverlap, s as DateTimePicker_default, st as MultiSelect_default, t as useScheduleDrag, tt as isTimeInRange, u as SequenceInput_default, ut as BaseToggle_default, w as DEFAULT_COLORS, x as ConcentrationInput_default, z as datePresetToISO } from "./useScheduleDrag-CxBeqYcu.js";
1
+ import { $ as ExperimentPopover_default, A as RackEditor_default, B as AppContainer_default, C as hexToHsl, D as ExperimentTimeline_default, E as StepWizard_default, F as Avatar_default, G as AppPluginSwitcher_default, H as AppSidebar_default, I as ProgressBar_default, J as AppPageSelector_default, K as AppAvatarMenu_default, L as StatusIndicator_default, M as ChartContainer_default, N as Tooltip_default, O as PlateMapEditor_default, P as Breadcrumb_default, Q as Skeleton_default, R as Divider_default, S as deriveShade, T as AutoGroupModal_default, U as AppTopBar_default, V as AppLayout_default, W as PluginIcon_default, X as ExperimentCodeBadge_default, Y as ExperimentSelectorModal_default, Z as EmptyState_default, _t as ColorSlider_default, a as TimeRangeInput_default, at as ThemeToggle_default, b as GroupingModal_default, ct as AlertBox_default, d as AuditTrail_default, dt as Calendar_default, et as ConfirmDialog_default, f as ChemicalFormula_default, ft as DropdownButton_default, g as ReagentList_default, gt as BaseTabs_default, h as SampleHierarchyTree_default, ht as BaseModal_default, i as ScheduleCalendar_default, it as SettingsButton_default, j as WellPlate_default, k as SampleLegend_default, lt as FormField_default, m as ProtocolStepEditor_default, mt as SegmentedControl_default, n as FitPanel_default, o as ExperimentDataViewer_default, ot as IconButton_default, p as ScientificNumber_default, pt as BasePill_default, q as AppPillNav_default, r as ResourceCard_default, rt as CollapsibleCard_default, st as ToastNotification_default, tt as SettingsModal_default, u as BatchProgressList_default, ut as DataFrame_default, v as DoseCalculator_default, vt as BaseButton_default, w as hslToHex, x as SampleSelector_default, y as GroupAssigner_default, z as LoadingSpinner_default } from "./components-_XqPEhP9.js";
2
+ import { $ as formatTime, B as useChemicalFormula, C as SORT_OPTIONS, E as useApi, F as MoleculeInput_default, G as NumberInput_default, H as useTheme, I as SequenceInput_default, J as addMinutes, K as TagsInput_default, L as useSequenceUtils, M as UnitInput_default, N as ConcentrationInput_default, P as useConcentrationUnits, Q as formatDuration, R as FormulaInput_default, S as EXPERIMENT_STATUS_VARIANT_MAP, T as formatExperimentDate, U as useToast, V as DateTimePicker_default, W as FileUploader_default, X as durationMinutes, Y as compareTime, Z as findAvailableSlots, _ as usePlatformContext, b as EXPERIMENT_STATUS_LABELS, c as useDoseCalculator, ct as BaseSlider_default, dt as BaseCheckbox_default, et as generateTimeSlots, f as useAutoGroup, ft as BaseSelect_default, g as useAppExperiment, h as APP_EXPERIMENT_KEY, it as snapToSlot, l as DEFAULT_COLORS, lt as BaseRadioGroup_default, m as useRackEditor, mt as BaseInput_default, n as useExperimentData, nt as parseTime, ot as DatePicker_default, p as useWellPlateEditor, pt as BaseTextarea_default, q as TimePicker_default, r as useProtocolTemplates, rt as rangesOverlap, st as MultiSelect_default, t as useScheduleDrag, tt as isTimeInRange, ut as BaseToggle_default, v as useExperimentSelector, w as datePresetToISO, x as EXPERIMENT_STATUS_OPTIONS, y as DATE_PRESET_OPTIONS, z as ATOMIC_WEIGHTS } from "./useScheduleDrag-CA9sGNJG.js";
3
3
  import { n as colorPalettes, r as useSettingsStore, t as useAuthStore } from "./auth-DsI0rQ7_.js";
4
4
  import { MINTSdk } from "./install.js";
5
- import { n as useExperimentSave, o as usePasskey, r as usePluginConfig, s as useAuth, t as usePluginApi } from "./composables-BXklV5ii.js";
5
+ import { n as useExperimentSave, o as usePasskey, r as usePluginConfig, s as useAuth, t as usePluginApi } from "./composables-tiZqLu1M.js";
6
6
  import "./stores/index.js";
7
- export { APP_EXPERIMENT_KEY, ATOMIC_WEIGHTS, AlertBox_default as AlertBox, AppAvatarMenu_default as AppAvatarMenu, AppContainer_default as AppContainer, AppLayout_default as AppLayout, AppPageSelector_default as AppPageSelector, AppPillNav_default as AppPillNav, AppPluginSwitcher_default as AppPluginSwitcher, AppSidebar_default as AppSidebar, ToastNotification_default as AppToastContainer, ToastNotification_default as ToastNotification, AppTopBar_default as AppTopBar, AuditTrail_default as AuditTrail, AutoGroupModal_default as AutoGroupModal, Avatar_default as Avatar, BaseButton_default as BaseButton, BaseCheckbox_default as BaseCheckbox, BaseInput_default as BaseInput, BaseModal_default as BaseModal, BasePill_default as BasePill, BaseRadioGroup_default as BaseRadioGroup, BaseSelect_default as BaseSelect, BaseSlider_default as BaseSlider, BaseTabs_default as BaseTabs, BaseTextarea_default as BaseTextarea, BaseToggle_default as BaseToggle, BatchProgressList_default as BatchProgressList, Breadcrumb_default as Breadcrumb, Calendar_default as Calendar, ChartContainer_default as ChartContainer, ChemicalFormula_default as ChemicalFormula, CollapsibleCard_default as CollapsibleCard, ColorSlider_default as ColorSlider, ConcentrationInput_default as ConcentrationInput, ConfirmDialog_default as ConfirmDialog, DATE_PRESET_OPTIONS, DEFAULT_COLORS, DataFrame_default as DataFrame, DatePicker_default as DatePicker, DateTimePicker_default as DateTimePicker, Divider_default as Divider, DoseCalculator_default as DoseCalculator, DropdownButton_default as DropdownButton, EXPERIMENT_STATUS_LABELS, EXPERIMENT_STATUS_OPTIONS, EXPERIMENT_STATUS_VARIANT_MAP, EmptyState_default as EmptyState, ExperimentCodeBadge_default as ExperimentCodeBadge, ExperimentDataViewer_default as ExperimentDataViewer, ExperimentPopover_default as ExperimentPopover, ExperimentSelectorModal_default as ExperimentSelectorModal, ExperimentTimeline_default as ExperimentTimeline, FileUploader_default as FileUploader, FitPanel_default as FitPanel, FormField_default as FormField, FormulaInput_default as FormulaInput, GroupAssigner_default as GroupAssigner, GroupingModal_default as GroupingModal, IconButton_default as IconButton, LoadingSpinner_default as LoadingSpinner, MINTSdk, MINTSdk as default, MoleculeInput_default as MoleculeInput, MultiSelect_default as MultiSelect, NumberInput_default as NumberInput, PlateMapEditor_default as PlateMapEditor, ProgressBar_default as ProgressBar, ProtocolStepEditor_default as ProtocolStepEditor, RackEditor_default as RackEditor, ReagentList_default as ReagentList, ResourceCard_default as ResourceCard, SORT_OPTIONS, SampleHierarchyTree_default as SampleHierarchyTree, SampleLegend_default as SampleLegend, SampleSelector_default as SampleSelector, ScheduleCalendar_default as ScheduleCalendar, ScientificNumber_default as ScientificNumber, SegmentedControl_default as SegmentedControl, SequenceInput_default as SequenceInput, SettingsButton_default as SettingsButton, SettingsModal_default as SettingsModal, Skeleton_default as Skeleton, StatusIndicator_default as StatusIndicator, StepWizard_default as StepWizard, TagsInput_default as TagsInput, ThemeToggle_default as ThemeToggle, TimePicker_default as TimePicker, TimeRangeInput_default as TimeRangeInput, Tooltip_default as Tooltip, UnitInput_default as UnitInput, WellPlate_default as WellPlate, addMinutes, colorPalettes, compareTime, datePresetToISO, deriveShade, durationMinutes, findAvailableSlots, formatDuration, formatExperimentDate, formatTime, generateTimeSlots, hexToHsl, hslToHex, isTimeInRange, parseTime, rangesOverlap, snapToSlot, useApi, useAppExperiment, useAuth, useAuthStore, useAutoGroup, useChemicalFormula, useConcentrationUnits, useDoseCalculator, useExperimentData, useExperimentSave, useExperimentSelector, usePasskey, usePlatformContext, usePluginApi, usePluginConfig, useProtocolTemplates, useRackEditor, useScheduleDrag, useSequenceUtils, useSettingsStore, useTheme, useToast, useWellPlateEditor };
7
+ export { APP_EXPERIMENT_KEY, ATOMIC_WEIGHTS, AlertBox_default as AlertBox, AppAvatarMenu_default as AppAvatarMenu, AppContainer_default as AppContainer, AppLayout_default as AppLayout, AppPageSelector_default as AppPageSelector, AppPillNav_default as AppPillNav, AppPluginSwitcher_default as AppPluginSwitcher, AppSidebar_default as AppSidebar, ToastNotification_default as AppToastContainer, ToastNotification_default as ToastNotification, AppTopBar_default as AppTopBar, AuditTrail_default as AuditTrail, AutoGroupModal_default as AutoGroupModal, Avatar_default as Avatar, BaseButton_default as BaseButton, BaseCheckbox_default as BaseCheckbox, BaseInput_default as BaseInput, BaseModal_default as BaseModal, BasePill_default as BasePill, BaseRadioGroup_default as BaseRadioGroup, BaseSelect_default as BaseSelect, BaseSlider_default as BaseSlider, BaseTabs_default as BaseTabs, BaseTextarea_default as BaseTextarea, BaseToggle_default as BaseToggle, BatchProgressList_default as BatchProgressList, Breadcrumb_default as Breadcrumb, Calendar_default as Calendar, ChartContainer_default as ChartContainer, ChemicalFormula_default as ChemicalFormula, CollapsibleCard_default as CollapsibleCard, ColorSlider_default as ColorSlider, ConcentrationInput_default as ConcentrationInput, ConfirmDialog_default as ConfirmDialog, DATE_PRESET_OPTIONS, DEFAULT_COLORS, DataFrame_default as DataFrame, DatePicker_default as DatePicker, DateTimePicker_default as DateTimePicker, Divider_default as Divider, DoseCalculator_default as DoseCalculator, DropdownButton_default as DropdownButton, EXPERIMENT_STATUS_LABELS, EXPERIMENT_STATUS_OPTIONS, EXPERIMENT_STATUS_VARIANT_MAP, EmptyState_default as EmptyState, ExperimentCodeBadge_default as ExperimentCodeBadge, ExperimentDataViewer_default as ExperimentDataViewer, ExperimentPopover_default as ExperimentPopover, ExperimentSelectorModal_default as ExperimentSelectorModal, ExperimentTimeline_default as ExperimentTimeline, FileUploader_default as FileUploader, FitPanel_default as FitPanel, FormField_default as FormField, FormulaInput_default as FormulaInput, GroupAssigner_default as GroupAssigner, GroupingModal_default as GroupingModal, IconButton_default as IconButton, LoadingSpinner_default as LoadingSpinner, MINTSdk, MINTSdk as default, MoleculeInput_default as MoleculeInput, MultiSelect_default as MultiSelect, NumberInput_default as NumberInput, PlateMapEditor_default as PlateMapEditor, PluginIcon_default as PluginIcon, ProgressBar_default as ProgressBar, ProtocolStepEditor_default as ProtocolStepEditor, RackEditor_default as RackEditor, ReagentList_default as ReagentList, ResourceCard_default as ResourceCard, SORT_OPTIONS, SampleHierarchyTree_default as SampleHierarchyTree, SampleLegend_default as SampleLegend, SampleSelector_default as SampleSelector, ScheduleCalendar_default as ScheduleCalendar, ScientificNumber_default as ScientificNumber, SegmentedControl_default as SegmentedControl, SequenceInput_default as SequenceInput, SettingsButton_default as SettingsButton, SettingsModal_default as SettingsModal, Skeleton_default as Skeleton, StatusIndicator_default as StatusIndicator, StepWizard_default as StepWizard, TagsInput_default as TagsInput, ThemeToggle_default as ThemeToggle, TimePicker_default as TimePicker, TimeRangeInput_default as TimeRangeInput, Tooltip_default as Tooltip, UnitInput_default as UnitInput, WellPlate_default as WellPlate, addMinutes, colorPalettes, compareTime, datePresetToISO, deriveShade, durationMinutes, findAvailableSlots, formatDuration, formatExperimentDate, formatTime, generateTimeSlots, hexToHsl, hslToHex, isTimeInRange, parseTime, rangesOverlap, snapToSlot, useApi, useAppExperiment, useAuth, useAuthStore, useAutoGroup, useChemicalFormula, useConcentrationUnits, useDoseCalculator, useExperimentData, useExperimentSave, useExperimentSelector, usePasskey, usePlatformContext, usePluginApi, usePluginConfig, useProtocolTemplates, useRackEditor, useScheduleDrag, useSequenceUtils, useSettingsStore, useTheme, useToast, useWellPlateEditor };
package/dist/install.js CHANGED
@@ -1,5 +1,5 @@
1
- import { t as components_exports } from "./components-CzbQQPCb.js";
2
- import "./useScheduleDrag-CxBeqYcu.js";
1
+ import { t as components_exports } from "./components-_XqPEhP9.js";
2
+ import "./useScheduleDrag-CA9sGNJG.js";
3
3
  //#region src/install.ts
4
4
  /**
5
5
  * Vue plugin that registers all MINT SDK components globally.
@@ -47,7 +47,7 @@ export declare const useAuthStore: import('pinia').StoreDefinition<"mint-auth",
47
47
  setError: (message: string | null) => void;
48
48
  setLoading: (loading: boolean) => void;
49
49
  logout: () => void;
50
- }, "error" | "username" | "token" | "tokenExpires" | "userInfo" | "authConfig" | "isInitialized" | "isLoading">, Pick<{
50
+ }, "error" | "username" | "isLoading" | "token" | "tokenExpires" | "userInfo" | "authConfig" | "isInitialized">, Pick<{
51
51
  token: import('vue').Ref<string | null, string | null>;
52
52
  tokenExpires: import('vue').Ref<Date | null, Date | null>;
53
53
  username: import('vue').Ref<string | null, string | null>;