@agent-relay/dashboard 2.0.81 → 2.0.82
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/out/404.html +1 -1
- package/out/_next/static/chunks/{118-4c8241b0218335de.js → 118-ae2b650136a5a5fc.js} +1 -1
- package/out/_next/static/chunks/407-0c82986cf79c8ecb.js +1 -0
- package/out/_next/static/chunks/app/app/[[...slug]]/{page-1e81c047cff17212.js → page-f7eca1b66fb4249b.js} +1 -1
- package/out/_next/static/chunks/app/{page-6892fe2dd07fb48b.js → page-0ee604f7070d14c0.js} +1 -1
- package/out/_next/static/css/8968d98ed4c4d33f.css +1 -0
- package/out/about.html +2 -2
- package/out/about.txt +1 -1
- package/out/app/onboarding.html +1 -1
- package/out/app/onboarding.txt +1 -1
- package/out/app.html +1 -1
- package/out/app.txt +2 -2
- package/out/blog/go-to-bed-wake-up-to-a-finished-product.html +2 -2
- package/out/blog/go-to-bed-wake-up-to-a-finished-product.txt +1 -1
- package/out/blog/let-them-cook-multi-agent-orchestration.html +2 -2
- package/out/blog/let-them-cook-multi-agent-orchestration.txt +2 -2
- package/out/blog.html +2 -2
- package/out/blog.txt +1 -1
- package/out/careers.html +2 -2
- package/out/careers.txt +1 -1
- package/out/changelog.html +2 -2
- package/out/changelog.txt +1 -1
- package/out/cloud/link.html +1 -1
- package/out/cloud/link.txt +2 -2
- package/out/complete-profile.html +2 -2
- package/out/complete-profile.txt +1 -1
- package/out/connect-repos.html +1 -1
- package/out/connect-repos.txt +1 -1
- package/out/contact.html +2 -2
- package/out/contact.txt +1 -1
- package/out/docs.html +2 -2
- package/out/docs.txt +1 -1
- package/out/history.html +1 -1
- package/out/history.txt +2 -2
- package/out/index.html +1 -1
- package/out/index.txt +2 -2
- package/out/login.html +2 -2
- package/out/login.txt +1 -1
- package/out/metrics.html +1 -1
- package/out/metrics.txt +2 -2
- package/out/pricing.html +2 -2
- package/out/pricing.txt +1 -1
- package/out/privacy.html +2 -2
- package/out/privacy.txt +1 -1
- package/out/providers/setup/claude.html +1 -1
- package/out/providers/setup/claude.txt +1 -1
- package/out/providers/setup/codex.html +1 -1
- package/out/providers/setup/codex.txt +1 -1
- package/out/providers/setup/cursor.html +1 -1
- package/out/providers/setup/cursor.txt +1 -1
- package/out/providers.html +1 -1
- package/out/providers.txt +1 -1
- package/out/security.html +2 -2
- package/out/security.txt +1 -1
- package/out/signup.html +2 -2
- package/out/signup.txt +1 -1
- package/out/terms.html +2 -2
- package/out/terms.txt +1 -1
- package/package.json +7 -1
- package/src/app/about/page.tsx +7 -0
- package/src/app/app/[[...slug]]/DashboardPageClient.tsx +853 -0
- package/src/app/app/[[...slug]]/page.tsx +23 -0
- package/src/app/app/onboarding/page.tsx +394 -0
- package/src/app/apple-icon.png +0 -0
- package/src/app/blog/go-to-bed-wake-up-to-a-finished-product/page.tsx +88 -0
- package/src/app/blog/let-them-cook-multi-agent-orchestration/page.tsx +93 -0
- package/src/app/blog/page.tsx +15 -0
- package/src/app/careers/page.tsx +7 -0
- package/src/app/changelog/page.tsx +7 -0
- package/src/app/cloud/link/page.tsx +464 -0
- package/src/app/complete-profile/page.tsx +204 -0
- package/src/app/connect-repos/page.tsx +410 -0
- package/src/app/contact/page.tsx +7 -0
- package/src/app/docs/page.tsx +7 -0
- package/src/app/favicon.png +0 -0
- package/src/app/globals.css +200 -0
- package/src/app/history/page.tsx +658 -0
- package/src/app/layout.tsx +25 -0
- package/src/app/login/page.tsx +424 -0
- package/src/app/metrics/page.tsx +781 -0
- package/src/app/page.tsx +59 -0
- package/src/app/pricing/page.tsx +7 -0
- package/src/app/privacy/page.tsx +7 -0
- package/src/app/providers/page.tsx +193 -0
- package/src/app/providers/setup/[provider]/ProviderSetupClient.tsx +197 -0
- package/src/app/providers/setup/[provider]/constants.ts +35 -0
- package/src/app/providers/setup/[provider]/page.tsx +42 -0
- package/src/app/security/page.tsx +7 -0
- package/src/app/signup/page.tsx +533 -0
- package/src/app/terms/page.tsx +7 -0
- package/src/components/ActivityFeed.tsx +216 -0
- package/src/components/AddWorkspaceModal.tsx +170 -0
- package/src/components/AgentCard.test.tsx +134 -0
- package/src/components/AgentCard.tsx +585 -0
- package/src/components/AgentList.test.tsx +147 -0
- package/src/components/AgentList.tsx +419 -0
- package/src/components/AgentLogPreview.tsx +173 -0
- package/src/components/AgentProfilePanel.tsx +569 -0
- package/src/components/App.tsx +3424 -0
- package/src/components/BillingPanel.tsx +922 -0
- package/src/components/BillingResult.tsx +447 -0
- package/src/components/BroadcastComposer.tsx +690 -0
- package/src/components/ChannelAdminPanel.tsx +773 -0
- package/src/components/ChannelBrowser.tsx +385 -0
- package/src/components/ChannelChat.tsx +261 -0
- package/src/components/ChannelSidebar.tsx +399 -0
- package/src/components/CloudSessionProvider.tsx +130 -0
- package/src/components/CommandPalette.tsx +815 -0
- package/src/components/ConfirmationDialog.tsx +133 -0
- package/src/components/ConversationHistory.tsx +518 -0
- package/src/components/CoordinatorPanel.tsx +956 -0
- package/src/components/DecisionQueue.tsx +717 -0
- package/src/components/DirectMessageView.tsx +164 -0
- package/src/components/FileAutocomplete.tsx +368 -0
- package/src/components/FleetOverview.tsx +278 -0
- package/src/components/LogViewer.tsx +310 -0
- package/src/components/LogViewerPanel.tsx +482 -0
- package/src/components/Logo.tsx +284 -0
- package/src/components/MentionAutocomplete.tsx +384 -0
- package/src/components/MessageComposer.tsx +473 -0
- package/src/components/MessageList.tsx +725 -0
- package/src/components/MessageSenderName.tsx +91 -0
- package/src/components/MessageStatusIndicator.tsx +142 -0
- package/src/components/NewConversationModal.tsx +400 -0
- package/src/components/NotificationToast.tsx +488 -0
- package/src/components/OnlineUsersIndicator.tsx +164 -0
- package/src/components/Pagination.tsx +124 -0
- package/src/components/PricingPlans.tsx +386 -0
- package/src/components/ProjectList.tsx +711 -0
- package/src/components/ProviderAuthFlow.tsx +343 -0
- package/src/components/ProviderConnectionList.tsx +375 -0
- package/src/components/ProvisioningProgress.tsx +730 -0
- package/src/components/ReactionChips.tsx +70 -0
- package/src/components/ReactionPicker.tsx +121 -0
- package/src/components/RepoAccessPanel.tsx +787 -0
- package/src/components/RepositoriesPanel.tsx +901 -0
- package/src/components/ServerCard.tsx +202 -0
- package/src/components/SessionExpiredModal.tsx +128 -0
- package/src/components/SpawnModal.test.tsx +190 -0
- package/src/components/SpawnModal.tsx +1001 -0
- package/src/components/TaskAssignmentUI.tsx +375 -0
- package/src/components/TerminalProviderSetup.tsx +517 -0
- package/src/components/ThemeProvider.tsx +159 -0
- package/src/components/ThinkingIndicator.tsx +231 -0
- package/src/components/ThreadList.tsx +198 -0
- package/src/components/ThreadPanel.tsx +405 -0
- package/src/components/TrajectoryViewer.tsx +698 -0
- package/src/components/TypingIndicator.tsx +69 -0
- package/src/components/UsageBanner.tsx +231 -0
- package/src/components/UserProfilePanel.tsx +233 -0
- package/src/components/WorkspaceContext.tsx +95 -0
- package/src/components/WorkspaceSelector.tsx +234 -0
- package/src/components/WorkspaceStatusIndicator.tsx +396 -0
- package/src/components/XTermInteractive.tsx +516 -0
- package/src/components/XTermLogViewer.tsx +719 -0
- package/src/components/channels/ChannelDialogs.tsx +1411 -0
- package/src/components/channels/ChannelHeader.tsx +317 -0
- package/src/components/channels/ChannelMessageList.tsx +463 -0
- package/src/components/channels/ChannelViewV1.tsx +146 -0
- package/src/components/channels/MessageInput.tsx +302 -0
- package/src/components/channels/SearchInput.tsx +172 -0
- package/src/components/channels/SearchResults.tsx +336 -0
- package/src/components/channels/api.test.ts +1527 -0
- package/src/components/channels/api.ts +703 -0
- package/src/components/channels/index.ts +76 -0
- package/src/components/channels/mockApi.ts +344 -0
- package/src/components/channels/types.ts +566 -0
- package/src/components/hooks/index.ts +58 -0
- package/src/components/hooks/useAgentLogs.ts +504 -0
- package/src/components/hooks/useAgents.ts +127 -0
- package/src/components/hooks/useBroadcastDedup.test.ts +371 -0
- package/src/components/hooks/useBroadcastDedup.ts +86 -0
- package/src/components/hooks/useChannelAdmin.ts +329 -0
- package/src/components/hooks/useChannelBrowser.ts +239 -0
- package/src/components/hooks/useChannelCommands.ts +138 -0
- package/src/components/hooks/useChannels.ts +367 -0
- package/src/components/hooks/useDebounce.ts +29 -0
- package/src/components/hooks/useDirectMessage.test.ts +952 -0
- package/src/components/hooks/useDirectMessage.ts +141 -0
- package/src/components/hooks/useMessages.ts +310 -0
- package/src/components/hooks/useOrchestrator.test.ts +165 -0
- package/src/components/hooks/useOrchestrator.ts +424 -0
- package/src/components/hooks/usePinnedAgents.test.ts +356 -0
- package/src/components/hooks/usePinnedAgents.ts +140 -0
- package/src/components/hooks/usePresence.test.ts +245 -0
- package/src/components/hooks/usePresence.ts +377 -0
- package/src/components/hooks/useRecentRepos.ts +130 -0
- package/src/components/hooks/useSession.ts +209 -0
- package/src/components/hooks/useThread.ts +138 -0
- package/src/components/hooks/useTrajectory.ts +265 -0
- package/src/components/hooks/useWebSocket.ts +290 -0
- package/src/components/hooks/useWorkspaceMembers.ts +132 -0
- package/src/components/hooks/useWorkspaceRepos.ts +73 -0
- package/src/components/hooks/useWorkspaceStatus.ts +237 -0
- package/src/components/index.ts +81 -0
- package/src/components/layout/Header.tsx +311 -0
- package/src/components/layout/RepoContextHeader.tsx +361 -0
- package/src/components/layout/Sidebar.archive.test.tsx +126 -0
- package/src/components/layout/Sidebar.test.tsx +691 -0
- package/src/components/layout/Sidebar.tsx +900 -0
- package/src/components/layout/index.ts +7 -0
- package/src/components/settings/BillingSettingsPanel.tsx +564 -0
- package/src/components/settings/SettingsPage.tsx +683 -0
- package/src/components/settings/TeamSettingsPanel.tsx +560 -0
- package/src/components/settings/WorkspaceSettingsPanel.tsx +1368 -0
- package/src/components/settings/index.ts +11 -0
- package/src/components/settings/types.ts +79 -0
- package/src/components/utils/messageFormatting.test.tsx +331 -0
- package/src/components/utils/messageFormatting.tsx +597 -0
- package/src/index.ts +63 -0
- package/src/landing/AboutPage.tsx +77 -0
- package/src/landing/BlogContent.tsx +187 -0
- package/src/landing/BlogPage.tsx +47 -0
- package/src/landing/CareersPage.tsx +53 -0
- package/src/landing/ChangelogPage.tsx +33 -0
- package/src/landing/ContactPage.tsx +41 -0
- package/src/landing/DocsPage.tsx +43 -0
- package/src/landing/LandingPage.tsx +702 -0
- package/src/landing/PricingPage.tsx +549 -0
- package/src/landing/PrivacyPage.tsx +117 -0
- package/src/landing/SecurityPage.tsx +42 -0
- package/src/landing/StaticPage.tsx +165 -0
- package/src/landing/TermsPage.tsx +125 -0
- package/src/landing/blogData.ts +312 -0
- package/src/landing/index.ts +18 -0
- package/src/landing/styles.css +3673 -0
- package/src/lib/agent-merge.test.ts +43 -0
- package/src/lib/agent-merge.ts +35 -0
- package/src/lib/api.ts +1294 -0
- package/src/lib/cloudApi.ts +893 -0
- package/src/lib/colors.test.ts +175 -0
- package/src/lib/colors.ts +218 -0
- package/src/lib/config.ts +109 -0
- package/src/lib/hierarchy.ts +242 -0
- package/src/lib/stuckDetection.ts +142 -0
- package/src/lib/useUrlRouting.ts +190 -0
- package/src/types/index.ts +317 -0
- package/src/types/threading.ts +7 -0
- package/out/_next/static/chunks/285-dc644487a8d6500d.js +0 -1
- package/out/_next/static/css/4c58d9cf493aa626.css +0 -1
- /package/out/_next/static/{dYlczDQI12PIQ3tqq3N4Y → IxfA6RZu4trcsEMYlkQra}/_buildManifest.js +0 -0
- /package/out/_next/static/{dYlczDQI12PIQ3tqq3N4Y → IxfA6RZu4trcsEMYlkQra}/_ssgManifest.js +0 -0
- /package/out/_next/static/chunks/{528-d375bc8b46912d2c.js → 528-f5f676996d613c25.js} +0 -0
- /package/out/_next/static/chunks/app/blog/let-them-cook-multi-agent-orchestration/{page-a58308f43557b908.js → page-b194f207fbd91862.js} +0 -0
|
@@ -0,0 +1,564 @@
|
|
|
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
|
+
×
|
|
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
|
+
}
|