@agent-relay/dashboard 2.0.82 → 2.0.84

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 (228) hide show
  1. package/out/404.html +1 -1
  2. package/out/_next/static/chunks/1028-da5d75e35d1420f1.js +1 -0
  3. package/out/_next/static/chunks/1528-78b17000a7e10bc6.js +2 -0
  4. package/out/_next/static/chunks/1695-4a5d33ba715e09b4.js +1 -0
  5. package/out/_next/static/chunks/1705-36c2180d00a4a569.js +1 -0
  6. package/out/_next/static/chunks/1dd3208c-e1f87c7b3dc1a820.js +1 -0
  7. package/out/_next/static/chunks/3663-47290254b8f6f5dd.js +1 -0
  8. package/out/_next/static/chunks/3677-4b225baf4801d9b9.js +73 -0
  9. package/out/_next/static/chunks/5118-7e8ada2df38eef07.js +1 -0
  10. package/out/_next/static/chunks/5888-15cbe97c90ed5fae.js +1 -0
  11. package/out/_next/static/chunks/6773-a45343a98df3abb5.js +1 -0
  12. package/out/_next/static/chunks/6940-b824612b605e79b3.js +9 -0
  13. package/out/_next/static/chunks/7894-f4a15249082a680d.js +1 -0
  14. package/out/_next/static/chunks/9175-b3617c1e5cbfed0e.js +1 -0
  15. package/out/_next/static/chunks/9372-1a804b8d08c7a236.js +1 -0
  16. package/out/_next/static/chunks/{ab6c8a12-0a58072fbb505134.js → ab6c8a12-91438a812d94ecf0.js} +1 -1
  17. package/out/_next/static/chunks/app/_not-found/page-8e8842f82d204726.js +1 -0
  18. package/out/_next/static/chunks/app/about/page-b78577a7da8fa459.js +1 -0
  19. package/out/_next/static/chunks/app/app/[[...slug]]/page-3dffd65b6344f53e.js +1 -0
  20. package/out/_next/static/chunks/app/app/onboarding/page-b89be9aa6264a5e1.js +1 -0
  21. package/out/_next/static/chunks/app/blog/go-to-bed-wake-up-to-a-finished-product/page-fbd00893ef69e499.js +1 -0
  22. package/out/_next/static/chunks/app/blog/let-them-cook-multi-agent-orchestration/page-de2ea13649d0b6d3.js +1 -0
  23. package/out/_next/static/chunks/app/blog/page-a08e263c57a156fa.js +1 -0
  24. package/out/_next/static/chunks/app/careers/page-02228e1d6969b232.js +1 -0
  25. package/out/_next/static/chunks/app/changelog/page-1b5c1d79efc6e53a.js +1 -0
  26. package/out/_next/static/chunks/app/cloud/link/page-99654edffffb3af2.js +1 -0
  27. package/out/_next/static/chunks/app/complete-profile/page-59d146e5ddeafc5c.js +1 -0
  28. package/out/_next/static/chunks/app/connect-repos/page-995e16a976a6632c.js +1 -0
  29. package/out/_next/static/chunks/app/contact/page-273396a5ad57bcee.js +1 -0
  30. package/out/_next/static/chunks/app/dev/cli-tools/page-a71b80dcb2d5fc8d.js +1 -0
  31. package/out/_next/static/chunks/app/dev/log-viewer/page-46a6151ae1be0796.js +1 -0
  32. package/out/_next/static/chunks/app/docs/page-7c7cb603b24b7c40.js +1 -0
  33. package/out/_next/static/chunks/app/history/page-0c5cab1dab4e8886.js +1 -0
  34. package/out/_next/static/chunks/app/layout-96d72ba8ef8a43a0.js +1 -0
  35. package/out/_next/static/chunks/app/login/page-0ccbab34213df842.js +1 -0
  36. package/out/_next/static/chunks/app/metrics/page-8616272aeab9c8b0.js +1 -0
  37. package/out/_next/static/chunks/app/page-09ce10603ad9a251.js +1 -0
  38. package/out/_next/static/chunks/app/pricing/page-91c975079120c941.js +1 -0
  39. package/out/_next/static/chunks/app/privacy/{page-c21d51ac2dee3a88.js → page-a49ab271cc686644.js} +1 -1
  40. package/out/_next/static/chunks/app/providers/{page-59114505f4353512.js → page-d775d6eb5bc29e96.js} +1 -1
  41. package/out/_next/static/chunks/app/providers/setup/[provider]/page-ec4ef3cd80de807e.js +1 -0
  42. package/out/_next/static/chunks/app/security/page-d9da9bd9191e8f95.js +1 -0
  43. package/out/_next/static/chunks/app/signup/page-930eca0bf5fd299d.js +1 -0
  44. package/out/_next/static/chunks/app/terms/page-3e4827620b98613c.js +1 -0
  45. package/out/_next/static/chunks/framework-648e1ae7da590300.js +1 -0
  46. package/out/_next/static/chunks/{main-acb1b24265295d6a.js → main-2b1990080c292d92.js} +1 -1
  47. package/out/_next/static/chunks/main-app-9f6b7ff9e754a8f5.js +1 -0
  48. package/out/_next/static/chunks/pages/_app-a077b72e02273ab1.js +1 -0
  49. package/out/_next/static/chunks/pages/_error-84001666436a04e4.js +1 -0
  50. package/out/_next/static/chunks/{webpack-dd93b81e2659669c.js → webpack-7586035f1585f2db.js} +1 -1
  51. package/out/_next/static/css/eb9fc69d1e3d2bed.css +1 -0
  52. package/out/_next/static/{IxfA6RZu4trcsEMYlkQra → g3G0LMdB7lxcrU5mdM54m}/_buildManifest.js +1 -1
  53. package/out/about.html +2 -2
  54. package/out/about.txt +2 -2
  55. package/out/app/onboarding.html +1 -1
  56. package/out/app/onboarding.txt +2 -2
  57. package/out/app.html +1 -1
  58. package/out/app.txt +2 -2
  59. package/out/blog/go-to-bed-wake-up-to-a-finished-product.html +3 -3
  60. package/out/blog/go-to-bed-wake-up-to-a-finished-product.txt +1 -1
  61. package/out/blog/let-them-cook-multi-agent-orchestration.html +2 -2
  62. package/out/blog/let-them-cook-multi-agent-orchestration.txt +2 -2
  63. package/out/blog.html +2 -2
  64. package/out/blog.txt +1 -1
  65. package/out/careers.html +2 -2
  66. package/out/careers.txt +2 -2
  67. package/out/changelog.html +2 -2
  68. package/out/changelog.txt +2 -2
  69. package/out/cloud/link.html +1 -1
  70. package/out/cloud/link.txt +2 -2
  71. package/out/complete-profile.html +2 -2
  72. package/out/complete-profile.txt +2 -2
  73. package/out/connect-repos.html +1 -1
  74. package/out/connect-repos.txt +2 -2
  75. package/out/contact.html +2 -2
  76. package/out/contact.txt +2 -2
  77. package/out/dev/cli-tools.html +1 -0
  78. package/out/dev/cli-tools.txt +7 -0
  79. package/out/dev/log-viewer.html +23 -0
  80. package/out/dev/log-viewer.txt +7 -0
  81. package/out/docs.html +2 -2
  82. package/out/docs.txt +2 -2
  83. package/out/history.html +1 -1
  84. package/out/history.txt +2 -2
  85. package/out/index.html +1 -1
  86. package/out/index.txt +2 -2
  87. package/out/login.html +2 -2
  88. package/out/login.txt +2 -2
  89. package/out/metrics.html +1 -1
  90. package/out/metrics.txt +2 -2
  91. package/out/pricing.html +2 -2
  92. package/out/pricing.txt +2 -2
  93. package/out/privacy.html +2 -2
  94. package/out/privacy.txt +2 -2
  95. package/out/providers/setup/claude.html +1 -1
  96. package/out/providers/setup/claude.txt +2 -2
  97. package/out/providers/setup/codex.html +1 -1
  98. package/out/providers/setup/codex.txt +2 -2
  99. package/out/providers/setup/cursor.html +1 -1
  100. package/out/providers/setup/cursor.txt +2 -2
  101. package/out/providers.html +1 -1
  102. package/out/providers.txt +2 -2
  103. package/out/security.html +2 -2
  104. package/out/security.txt +2 -2
  105. package/out/signup.html +2 -2
  106. package/out/signup.txt +2 -2
  107. package/out/terms.html +2 -2
  108. package/out/terms.txt +2 -2
  109. package/package.json +5 -1
  110. package/src/adapters/DashboardConfigProvider.tsx +56 -0
  111. package/src/adapters/cloudFetchAdapter.ts +278 -0
  112. package/src/adapters/index.ts +3 -0
  113. package/src/adapters/types.ts +508 -0
  114. package/src/app/app/[[...slug]]/DashboardPageClient.tsx +67 -18
  115. package/src/app/app/onboarding/page.tsx +870 -170
  116. package/src/app/cloud/link/page.tsx +14 -6
  117. package/src/app/connect-repos/page.tsx +9 -3
  118. package/src/app/dev/cli-tools/page.tsx +130 -0
  119. package/src/app/dev/log-viewer/MockLogViewer.tsx +132 -0
  120. package/src/app/dev/log-viewer/fixtures.ts +110 -0
  121. package/src/app/dev/log-viewer/page.tsx +288 -0
  122. package/src/app/history/page.tsx +28 -12
  123. package/src/app/page.tsx +1 -1
  124. package/src/app/providers/setup/[provider]/ProviderSetupClient.tsx +209 -59
  125. package/src/components/AgentCard.tsx +4 -4
  126. package/src/components/AgentLogPreview.tsx +2 -38
  127. package/src/components/App.tsx +441 -2624
  128. package/src/components/CliToolHarness.test.tsx +83 -0
  129. package/src/components/CliToolHarness.tsx +292 -0
  130. package/src/components/CoordinatorPanel.tsx +13 -6
  131. package/src/components/LogViewer.tsx +2 -42
  132. package/src/components/ProviderAuthFlow.tsx +201 -81
  133. package/src/components/ProvisioningProgress.tsx +1 -1
  134. package/src/components/ReactionChips.tsx +2 -1
  135. package/src/components/SpawnModal.test.tsx +51 -18
  136. package/src/components/SpawnModal.tsx +175 -207
  137. package/src/components/TerminalProviderSetup.tsx +1 -1
  138. package/src/components/ThreadPanel.tsx +2 -0
  139. package/src/components/WorkspaceContext.tsx +7 -19
  140. package/src/components/XTermLogViewer.tsx +190 -27
  141. package/src/components/channels/ChannelMessageList.tsx +94 -4
  142. package/src/components/channels/ChannelViewV1.tsx +35 -11
  143. package/src/components/channels/api.ts +21 -20
  144. package/src/components/channels/types.ts +16 -0
  145. package/src/components/hooks/index.ts +0 -19
  146. package/src/components/hooks/useMessages.test.ts +80 -0
  147. package/src/components/hooks/useMessages.ts +13 -4
  148. package/src/components/hooks/useOrchestrator.ts +1 -1
  149. package/src/components/hooks/usePresence.ts +45 -6
  150. package/src/components/hooks/useThread.ts +83 -46
  151. package/src/components/hooks/useTrajectory.ts +62 -5
  152. package/src/components/hooks/useWebSocket.test.ts +358 -0
  153. package/src/components/hooks/useWebSocket.ts +243 -5
  154. package/src/components/index.ts +2 -14
  155. package/src/components/layout/Header.tsx +9 -15
  156. package/src/components/layout/Sidebar.tsx +1 -8
  157. package/src/components/settings/SettingsPage.tsx +108 -47
  158. package/src/components/settings/index.ts +0 -3
  159. package/src/landing/blogData.ts +1 -1
  160. package/src/lib/agent-merge.test.ts +2 -2
  161. package/src/lib/api.ts +8 -38
  162. package/src/lib/identity.test.ts +139 -0
  163. package/src/lib/identity.ts +48 -0
  164. package/src/lib/relaycastMessageAdapters.test.ts +182 -0
  165. package/src/lib/relaycastMessageAdapters.ts +105 -0
  166. package/src/lib/sanitize-logs.test.ts +227 -0
  167. package/src/lib/sanitize-logs.ts +202 -0
  168. package/src/providers/AgentProvider.tsx +799 -0
  169. package/src/providers/ChannelProvider.tsx +528 -0
  170. package/src/providers/CloudWorkspaceProvider.tsx +402 -0
  171. package/src/providers/MessageProvider.tsx +875 -0
  172. package/src/providers/RelayConfigProvider.tsx +94 -0
  173. package/src/providers/SendProvider.tsx +497 -0
  174. package/src/providers/SettingsProvider.tsx +247 -0
  175. package/src/providers/index.ts +26 -0
  176. package/src/types/index.ts +10 -10
  177. package/out/_next/static/chunks/11-9a2993a37266dcb3.js +0 -9
  178. package/out/_next/static/chunks/118-ae2b650136a5a5fc.js +0 -1
  179. package/out/_next/static/chunks/1dd3208c-40ab0fc0f60392b8.js +0 -1
  180. package/out/_next/static/chunks/202-fc0763dd7488e58f.js +0 -1
  181. package/out/_next/static/chunks/259-83b77fa1b91ba5aa.js +0 -1
  182. package/out/_next/static/chunks/407-0c82986cf79c8ecb.js +0 -1
  183. package/out/_next/static/chunks/528-f5f676996d613c25.js +0 -2
  184. package/out/_next/static/chunks/663-ddb04081febc3678.js +0 -1
  185. package/out/_next/static/chunks/687-88b6b139a6bb0e2e.js +0 -1
  186. package/out/_next/static/chunks/695-51d25b1988644374.js +0 -1
  187. package/out/_next/static/chunks/773-54a2641043c81e55.js +0 -1
  188. package/out/_next/static/chunks/app/_not-found/page-6da9b72091e5b511.js +0 -1
  189. package/out/_next/static/chunks/app/about/page-fff7c6457683f243.js +0 -1
  190. package/out/_next/static/chunks/app/app/[[...slug]]/page-f7eca1b66fb4249b.js +0 -1
  191. package/out/_next/static/chunks/app/app/onboarding/page-129abc5da2e67971.js +0 -1
  192. package/out/_next/static/chunks/app/blog/go-to-bed-wake-up-to-a-finished-product/page-5d5f28fd126b692f.js +0 -1
  193. package/out/_next/static/chunks/app/blog/let-them-cook-multi-agent-orchestration/page-b194f207fbd91862.js +0 -1
  194. package/out/_next/static/chunks/app/blog/page-b9bd9d8703fca76a.js +0 -1
  195. package/out/_next/static/chunks/app/careers/page-a4bd8d5f4de8f4eb.js +0 -1
  196. package/out/_next/static/chunks/app/changelog/page-9a1f6ad1743d63c5.js +0 -1
  197. package/out/_next/static/chunks/app/cloud/link/page-0844c5699b027c3b.js +0 -1
  198. package/out/_next/static/chunks/app/complete-profile/page-39ed5a67916beb87.js +0 -1
  199. package/out/_next/static/chunks/app/connect-repos/page-297eddee0c39f2a3.js +0 -1
  200. package/out/_next/static/chunks/app/contact/page-3c1dd8690217fade.js +0 -1
  201. package/out/_next/static/chunks/app/docs/page-1875e981f2c3fd13.js +0 -1
  202. package/out/_next/static/chunks/app/history/page-2d5c5695c9e8b40c.js +0 -1
  203. package/out/_next/static/chunks/app/layout-0a4b99656da25511.js +0 -1
  204. package/out/_next/static/chunks/app/login/page-f69c076f5a6fc520.js +0 -1
  205. package/out/_next/static/chunks/app/metrics/page-bebbee055669a17e.js +0 -1
  206. package/out/_next/static/chunks/app/page-0ee604f7070d14c0.js +0 -1
  207. package/out/_next/static/chunks/app/pricing/page-eeae7d594af333b6.js +0 -1
  208. package/out/_next/static/chunks/app/providers/setup/[provider]/page-daf9b3e05e77ae19.js +0 -1
  209. package/out/_next/static/chunks/app/security/page-cd562730fe84a0a2.js +0 -1
  210. package/out/_next/static/chunks/app/signup/page-c242ca08101a84ff.js +0 -1
  211. package/out/_next/static/chunks/app/terms/page-c7001720e7941dc6.js +0 -1
  212. package/out/_next/static/chunks/framework-3664cab31236a9fa.js +0 -1
  213. package/out/_next/static/chunks/main-app-7f73a939a312a228.js +0 -1
  214. package/out/_next/static/chunks/pages/_app-10a93ab5b7c32eb3.js +0 -1
  215. package/out/_next/static/chunks/pages/_error-2d792b2a41857be4.js +0 -1
  216. package/out/_next/static/css/8968d98ed4c4d33f.css +0 -1
  217. package/src/components/BillingResult.tsx +0 -447
  218. package/src/components/CloudSessionProvider.tsx +0 -130
  219. package/src/components/SessionExpiredModal.tsx +0 -128
  220. package/src/components/WorkspaceStatusIndicator.tsx +0 -396
  221. package/src/components/hooks/useSession.ts +0 -209
  222. package/src/components/hooks/useWorkspaceMembers.ts +0 -132
  223. package/src/components/hooks/useWorkspaceStatus.ts +0 -237
  224. package/src/components/settings/BillingSettingsPanel.tsx +0 -564
  225. package/src/components/settings/TeamSettingsPanel.tsx +0 -560
  226. package/src/components/settings/WorkspaceSettingsPanel.tsx +0 -1368
  227. package/src/lib/cloudApi.ts +0 -893
  228. /package/out/_next/static/{IxfA6RZu4trcsEMYlkQra → g3G0LMdB7lxcrU5mdM54m}/_ssgManifest.js +0 -0
@@ -1,564 +0,0 @@
1
- /**
2
- * Billing Settings Panel
3
- *
4
- * Manage subscription, view plans, and access billing portal.
5
- */
6
-
7
- import React, { useState, useEffect, useCallback } from 'react';
8
- import { cloudApi } from '../../lib/cloudApi';
9
-
10
- export interface BillingSettingsPanelProps {
11
- onUpgrade?: () => void;
12
- }
13
-
14
- interface Plan {
15
- tier: string;
16
- name: string;
17
- description: string;
18
- price: { monthly: number; yearly: number };
19
- features: string[];
20
- limits: Record<string, number>;
21
- recommended?: boolean;
22
- }
23
-
24
- interface Subscription {
25
- id: string;
26
- tier: string;
27
- status: string;
28
- currentPeriodStart: string;
29
- currentPeriodEnd: string;
30
- cancelAtPeriodEnd: boolean;
31
- interval: 'month' | 'year';
32
- }
33
-
34
- interface Invoice {
35
- id: string;
36
- number: string;
37
- amount: number;
38
- status: string;
39
- date: string;
40
- pdfUrl?: string;
41
- }
42
-
43
- const TIER_COLORS: Record<string, string> = {
44
- free: 'bg-bg-tertiary border-border-subtle text-text-muted',
45
- pro: 'bg-accent-cyan/10 border-accent-cyan/30 text-accent-cyan',
46
- team: 'bg-blue-900/30 border-blue-500/40 text-blue-400',
47
- enterprise: 'bg-amber-400/10 border-amber-400/30 text-amber-400',
48
- };
49
-
50
- const TIER_DESCRIPTIONS: Record<string, string> = {
51
- free: 'Free tier - upgrade to unlock more features',
52
- pro: 'Professional plan with enhanced features',
53
- team: 'Team plan with collaboration features',
54
- enterprise: 'Enterprise plan with dedicated support',
55
- };
56
-
57
- // Tier ranking for comparison (higher = better plan)
58
- const TIER_RANK: Record<string, number> = {
59
- free: 0,
60
- pro: 1,
61
- team: 2,
62
- enterprise: 3,
63
- };
64
-
65
- export function BillingSettingsPanel({ onUpgrade }: BillingSettingsPanelProps) {
66
- const [plans, setPlans] = useState<Plan[]>([]);
67
- const [currentTier, setCurrentTier] = useState<string>('free');
68
- const [subscription, setSubscription] = useState<Subscription | null>(null);
69
- const [invoices, setInvoices] = useState<Invoice[]>([]);
70
- const [isLoading, setIsLoading] = useState(true);
71
- const [error, setError] = useState<string | null>(null);
72
- const [successMessage, setSuccessMessage] = useState<string | null>(null);
73
-
74
- // Billing interval toggle
75
- const [billingInterval, setBillingInterval] = useState<'month' | 'year'>('month');
76
-
77
- // Action loading states
78
- const [checkoutLoading, setCheckoutLoading] = useState<string | null>(null);
79
- const [portalLoading, setPortalLoading] = useState(false);
80
- const [cancelLoading, setCancelLoading] = useState(false);
81
- const [resumeLoading, setResumeLoading] = useState(false);
82
-
83
- // Load billing data
84
- useEffect(() => {
85
- async function loadBillingData() {
86
- setIsLoading(true);
87
- setError(null);
88
-
89
- const [plansResult, subscriptionResult, invoicesResult] = await Promise.all([
90
- cloudApi.getBillingPlans(),
91
- cloudApi.getSubscription(),
92
- cloudApi.getInvoices(),
93
- ]);
94
-
95
- if (plansResult.success) {
96
- setPlans(plansResult.data.plans);
97
- }
98
-
99
- if (subscriptionResult.success) {
100
- setCurrentTier(subscriptionResult.data.tier);
101
- setSubscription(subscriptionResult.data.subscription);
102
- if (subscriptionResult.data.subscription?.interval) {
103
- setBillingInterval(subscriptionResult.data.subscription.interval);
104
- }
105
- }
106
-
107
- if (invoicesResult.success) {
108
- setInvoices(invoicesResult.data.invoices);
109
- }
110
-
111
- if (!plansResult.success) {
112
- setError(plansResult.error);
113
- }
114
-
115
- setIsLoading(false);
116
- }
117
-
118
- loadBillingData();
119
- }, []);
120
-
121
- // Start checkout for plan upgrade
122
- const handleCheckout = useCallback(async (tier: string) => {
123
- setCheckoutLoading(tier);
124
-
125
- const result = await cloudApi.createCheckoutSession(tier, billingInterval);
126
-
127
- if (result.success && result.data.checkoutUrl) {
128
- // Redirect to Stripe checkout
129
- window.location.href = result.data.checkoutUrl;
130
- } else if (!result.success) {
131
- setError(result.error);
132
- setCheckoutLoading(null);
133
- }
134
- }, [billingInterval]);
135
-
136
- // Open billing portal
137
- const handleOpenPortal = useCallback(async () => {
138
- setPortalLoading(true);
139
-
140
- const result = await cloudApi.createBillingPortal();
141
-
142
- if (result.success && result.data.portalUrl) {
143
- window.location.href = result.data.portalUrl;
144
- } else if (!result.success) {
145
- setError(result.error);
146
- }
147
-
148
- setPortalLoading(false);
149
- }, []);
150
-
151
- // Cancel subscription
152
- const handleCancel = useCallback(async () => {
153
- const confirmed = window.confirm(
154
- 'Are you sure you want to cancel your subscription? You will retain access until the end of your billing period.'
155
- );
156
- if (!confirmed) return;
157
-
158
- setCancelLoading(true);
159
-
160
- const result = await cloudApi.cancelSubscription();
161
-
162
- if (result.success) {
163
- setSubscription((prev) =>
164
- prev ? { ...prev, cancelAtPeriodEnd: true } : null
165
- );
166
- setSuccessMessage(result.data.message);
167
- setTimeout(() => setSuccessMessage(null), 5000);
168
- } else {
169
- setError(result.error);
170
- }
171
-
172
- setCancelLoading(false);
173
- }, []);
174
-
175
- // Resume subscription
176
- const handleResume = useCallback(async () => {
177
- setResumeLoading(true);
178
-
179
- const result = await cloudApi.resumeSubscription();
180
-
181
- if (result.success) {
182
- setSubscription((prev) =>
183
- prev ? { ...prev, cancelAtPeriodEnd: false } : null
184
- );
185
- setSuccessMessage(result.data.message);
186
- setTimeout(() => setSuccessMessage(null), 3000);
187
- } else {
188
- setError(result.error);
189
- }
190
-
191
- setResumeLoading(false);
192
- }, []);
193
-
194
- if (isLoading) {
195
- return (
196
- <div className="flex items-center justify-center h-64">
197
- <LoadingSpinner />
198
- <span className="ml-3 text-text-muted">Loading billing information...</span>
199
- </div>
200
- );
201
- }
202
-
203
- return (
204
- <div className="space-y-8">
205
- {/* Messages */}
206
- {error && (
207
- <div className="p-3 bg-error/10 border border-error/30 rounded-lg text-error text-sm">
208
- {error}
209
- <button
210
- onClick={() => setError(null)}
211
- className="ml-2 text-error/70 hover:text-error"
212
- >
213
- &times;
214
- </button>
215
- </div>
216
- )}
217
-
218
- {successMessage && (
219
- <div className="p-3 bg-success/10 border border-success/30 rounded-lg text-success text-sm">
220
- {successMessage}
221
- </div>
222
- )}
223
-
224
- {/* Current Plan */}
225
- <div>
226
- <h3 className="text-sm font-semibold text-text-muted uppercase tracking-wide mb-4">
227
- Current Plan
228
- </h3>
229
- <div className={`p-4 md:p-6 rounded-lg border-2 ${TIER_COLORS[currentTier]}`}>
230
- <div className="flex flex-col sm:flex-row sm:items-center sm:justify-between gap-4">
231
- <div>
232
- <h4 className="text-lg md:text-xl font-bold text-text-primary capitalize">
233
- {currentTier} Plan
234
- </h4>
235
- {subscription ? (
236
- <p className="text-xs md:text-sm text-text-secondary mt-1">
237
- {subscription.cancelAtPeriodEnd ? (
238
- <span className="text-amber-400">
239
- Cancels on {new Date(subscription.currentPeriodEnd).toLocaleDateString()}
240
- </span>
241
- ) : (
242
- <>
243
- Renews on {new Date(subscription.currentPeriodEnd).toLocaleDateString()}
244
- <span className="text-text-muted ml-2">
245
- ({subscription.interval === 'year' ? 'Yearly' : 'Monthly'})
246
- </span>
247
- </>
248
- )}
249
- </p>
250
- ) : (
251
- <p className="text-xs md:text-sm text-text-muted mt-1">
252
- {TIER_DESCRIPTIONS[currentTier] || TIER_DESCRIPTIONS.free}
253
- </p>
254
- )}
255
- </div>
256
-
257
- <div className="flex flex-wrap gap-2">
258
- {subscription && !subscription.cancelAtPeriodEnd && (
259
- <button
260
- onClick={handleCancel}
261
- disabled={cancelLoading}
262
- className="px-3 md:px-4 py-2 bg-bg-hover text-text-secondary rounded-lg text-xs md:text-sm font-medium hover:text-text-primary disabled:opacity-50 transition-colors"
263
- >
264
- {cancelLoading ? 'Canceling...' : 'Cancel Plan'}
265
- </button>
266
- )}
267
- {subscription?.cancelAtPeriodEnd && (
268
- <button
269
- onClick={handleResume}
270
- disabled={resumeLoading}
271
- className="px-3 md:px-4 py-2 bg-success/20 text-success rounded-lg text-xs md:text-sm font-medium hover:bg-success/30 disabled:opacity-50 transition-colors"
272
- >
273
- {resumeLoading ? 'Resuming...' : 'Resume Plan'}
274
- </button>
275
- )}
276
- {subscription && (
277
- <button
278
- onClick={handleOpenPortal}
279
- disabled={portalLoading}
280
- className="px-3 md:px-4 py-2 bg-accent-cyan text-bg-deep rounded-lg text-xs md:text-sm font-medium hover:bg-accent-cyan/90 disabled:opacity-50 transition-colors"
281
- >
282
- {portalLoading ? 'Opening...' : 'Manage Billing'}
283
- </button>
284
- )}
285
- </div>
286
- </div>
287
- </div>
288
- </div>
289
-
290
- {/* Billing Interval Toggle */}
291
- <div className="flex items-center justify-center gap-4">
292
- <span
293
- className={`text-sm font-medium ${
294
- billingInterval === 'month' ? 'text-text-primary' : 'text-text-muted'
295
- }`}
296
- >
297
- Monthly
298
- </span>
299
- <button
300
- onClick={() => setBillingInterval((prev) => (prev === 'month' ? 'year' : 'month'))}
301
- className="relative w-14 h-7 bg-bg-tertiary rounded-full transition-colors"
302
- >
303
- <span
304
- className={`absolute top-1 w-5 h-5 bg-accent-cyan rounded-full transition-transform ${
305
- billingInterval === 'year' ? 'translate-x-8' : 'translate-x-1'
306
- }`}
307
- />
308
- </button>
309
- <span
310
- className={`text-sm font-medium ${
311
- billingInterval === 'year' ? 'text-text-primary' : 'text-text-muted'
312
- }`}
313
- >
314
- Yearly
315
- <span className="ml-1 text-xs text-success">(Save 20%)</span>
316
- </span>
317
- </div>
318
-
319
- {/* Available Plans */}
320
- <div>
321
- <h3 className="text-sm font-semibold text-text-muted uppercase tracking-wide mb-4">
322
- Available Plans
323
- </h3>
324
- <div className="grid grid-cols-1 sm:grid-cols-2 lg:grid-cols-3 gap-4">
325
- {plans
326
- .filter((p) => p.tier !== 'free')
327
- .map((plan) => {
328
- // Only show recommended styling for upgrades, not downgrades
329
- const isUpgrade = (TIER_RANK[plan.tier] || 0) > (TIER_RANK[currentTier] || 0);
330
- const showRecommended = plan.recommended && isUpgrade;
331
- const isDowngrade = (TIER_RANK[plan.tier] || 0) < (TIER_RANK[currentTier] || 0);
332
-
333
- return (
334
- <div
335
- key={plan.tier}
336
- className={`relative p-6 rounded-lg border ${
337
- showRecommended
338
- ? 'border-accent-cyan shadow-glow-cyan'
339
- : 'border-border-subtle'
340
- } bg-bg-tertiary`}
341
- >
342
- {showRecommended && (
343
- <div className="absolute -top-3 left-1/2 -translate-x-1/2 px-3 py-1 bg-accent-cyan text-bg-deep text-xs font-bold rounded-full">
344
- Most Popular
345
- </div>
346
- )}
347
-
348
- <h4 className="text-lg font-bold text-text-primary">{plan.name}</h4>
349
- <p className="text-xs text-text-muted mt-1 mb-4">{plan.description}</p>
350
-
351
- <div className="mb-4">
352
- <span className="text-3xl font-bold text-text-primary">
353
- ${billingInterval === 'year' ? plan.price.yearly : plan.price.monthly}
354
- </span>
355
- <span className="text-text-muted">
356
- /{billingInterval === 'year' ? 'year' : 'month'}
357
- </span>
358
- </div>
359
-
360
- <ul className="space-y-2 mb-6">
361
- {plan.features.slice(0, 5).map((feature, i) => (
362
- <li key={i} className="flex items-start gap-2 text-sm text-text-secondary">
363
- <CheckIcon className="text-success shrink-0 mt-0.5" />
364
- {feature}
365
- </li>
366
- ))}
367
- </ul>
368
-
369
- {currentTier === plan.tier ? (
370
- <button
371
- disabled
372
- className="w-full py-2.5 bg-bg-hover text-text-muted rounded-lg text-sm font-medium cursor-default"
373
- >
374
- Current Plan
375
- </button>
376
- ) : (
377
- <button
378
- onClick={() => handleCheckout(plan.tier)}
379
- disabled={checkoutLoading !== null}
380
- className={`w-full py-2.5 rounded-lg text-sm font-medium transition-colors disabled:opacity-50 ${
381
- showRecommended
382
- ? 'bg-accent-cyan text-bg-deep hover:bg-accent-cyan/90'
383
- : 'bg-bg-hover text-text-primary hover:bg-bg-active'
384
- }`}
385
- >
386
- {checkoutLoading === plan.tier
387
- ? 'Loading...'
388
- : isDowngrade
389
- ? 'Downgrade'
390
- : 'Upgrade'}
391
- </button>
392
- )}
393
- </div>
394
- );
395
- })}
396
- </div>
397
- </div>
398
-
399
- {/* Enterprise CTA */}
400
- <div className="p-4 md:p-6 bg-gradient-to-r from-amber-400/10 to-accent-purple/10 border border-amber-400/20 rounded-lg">
401
- <div className="flex flex-col sm:flex-row sm:items-center sm:justify-between gap-4">
402
- <div>
403
- <h4 className="text-base md:text-lg font-bold text-text-primary">Enterprise</h4>
404
- <p className="text-xs md:text-sm text-text-secondary mt-1">
405
- Custom solutions for large teams with dedicated support, SLA, and custom integrations.
406
- </p>
407
- </div>
408
- <a
409
- href="mailto:enterprise@agent-relay.com"
410
- className="px-4 md:px-6 py-2 md:py-2.5 bg-amber-400 text-bg-deep rounded-lg text-xs md:text-sm font-bold hover:bg-amber-300 transition-colors text-center shrink-0"
411
- >
412
- Contact Sales
413
- </a>
414
- </div>
415
- </div>
416
-
417
- {/* Invoices */}
418
- {invoices.length > 0 && (
419
- <div>
420
- <h3 className="text-sm font-semibold text-text-muted uppercase tracking-wide mb-4">
421
- Billing History
422
- </h3>
423
-
424
- {/* Desktop Table */}
425
- <div className="hidden md:block bg-bg-tertiary rounded-lg overflow-hidden">
426
- <table className="w-full">
427
- <thead>
428
- <tr className="border-b border-border-subtle">
429
- <th className="text-left px-4 py-3 text-xs font-semibold text-text-muted uppercase">
430
- Invoice
431
- </th>
432
- <th className="text-left px-4 py-3 text-xs font-semibold text-text-muted uppercase">
433
- Date
434
- </th>
435
- <th className="text-left px-4 py-3 text-xs font-semibold text-text-muted uppercase">
436
- Amount
437
- </th>
438
- <th className="text-left px-4 py-3 text-xs font-semibold text-text-muted uppercase">
439
- Status
440
- </th>
441
- <th className="text-right px-4 py-3 text-xs font-semibold text-text-muted uppercase">
442
-
443
- </th>
444
- </tr>
445
- </thead>
446
- <tbody>
447
- {invoices.map((invoice) => (
448
- <tr key={invoice.id} className="border-b border-border-subtle last:border-0">
449
- <td className="px-4 py-3 text-sm text-text-primary font-medium">
450
- {invoice.number}
451
- </td>
452
- <td className="px-4 py-3 text-sm text-text-secondary">
453
- {new Date(invoice.date).toLocaleDateString()}
454
- </td>
455
- <td className="px-4 py-3 text-sm text-text-primary">
456
- ${(invoice.amount / 100).toFixed(2)}
457
- </td>
458
- <td className="px-4 py-3">
459
- <span
460
- className={`text-xs px-2 py-1 rounded-full ${
461
- invoice.status === 'paid'
462
- ? 'bg-success/20 text-success'
463
- : invoice.status === 'open'
464
- ? 'bg-amber-400/20 text-amber-400'
465
- : 'bg-error/20 text-error'
466
- }`}
467
- >
468
- {invoice.status}
469
- </span>
470
- </td>
471
- <td className="px-4 py-3 text-right">
472
- {invoice.pdfUrl && (
473
- <a
474
- href={invoice.pdfUrl}
475
- target="_blank"
476
- rel="noopener noreferrer"
477
- className="text-xs text-accent-cyan hover:underline"
478
- >
479
- Download
480
- </a>
481
- )}
482
- </td>
483
- </tr>
484
- ))}
485
- </tbody>
486
- </table>
487
- </div>
488
-
489
- {/* Mobile Card Layout */}
490
- <div className="md:hidden space-y-3">
491
- {invoices.map((invoice) => (
492
- <div key={invoice.id} className="bg-bg-tertiary rounded-lg p-4">
493
- <div className="flex items-center justify-between mb-2">
494
- <span className="text-sm font-medium text-text-primary">{invoice.number}</span>
495
- <span
496
- className={`text-xs px-2 py-1 rounded-full ${
497
- invoice.status === 'paid'
498
- ? 'bg-success/20 text-success'
499
- : invoice.status === 'open'
500
- ? 'bg-amber-400/20 text-amber-400'
501
- : 'bg-error/20 text-error'
502
- }`}
503
- >
504
- {invoice.status}
505
- </span>
506
- </div>
507
- <div className="flex items-center justify-between text-xs">
508
- <span className="text-text-muted">{new Date(invoice.date).toLocaleDateString()}</span>
509
- <span className="text-text-primary font-medium">${(invoice.amount / 100).toFixed(2)}</span>
510
- </div>
511
- {invoice.pdfUrl && (
512
- <a
513
- href={invoice.pdfUrl}
514
- target="_blank"
515
- rel="noopener noreferrer"
516
- className="mt-3 block text-center text-xs text-accent-cyan py-2 border border-accent-cyan/30 rounded-lg hover:bg-accent-cyan/10 transition-colors"
517
- >
518
- Download PDF
519
- </a>
520
- )}
521
- </div>
522
- ))}
523
- </div>
524
- </div>
525
- )}
526
- </div>
527
- );
528
- }
529
-
530
- // Icons
531
- function LoadingSpinner() {
532
- return (
533
- <svg className="animate-spin h-5 w-5 text-accent-cyan" viewBox="0 0 24 24">
534
- <circle
535
- cx="12"
536
- cy="12"
537
- r="10"
538
- stroke="currentColor"
539
- strokeWidth="2"
540
- fill="none"
541
- strokeDasharray="32"
542
- strokeLinecap="round"
543
- />
544
- </svg>
545
- );
546
- }
547
-
548
- function CheckIcon({ className = '' }: { className?: string }) {
549
- return (
550
- <svg
551
- width="14"
552
- height="14"
553
- viewBox="0 0 24 24"
554
- fill="none"
555
- stroke="currentColor"
556
- strokeWidth="3"
557
- strokeLinecap="round"
558
- strokeLinejoin="round"
559
- className={className}
560
- >
561
- <polyline points="20 6 9 17 4 12" />
562
- </svg>
563
- );
564
- }