@agent-relay/dashboard 2.0.80 → 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.
Files changed (244) hide show
  1. package/out/404.html +1 -1
  2. package/out/_next/static/chunks/{118-4c8241b0218335de.js → 118-ae2b650136a5a5fc.js} +1 -1
  3. package/out/_next/static/chunks/407-0c82986cf79c8ecb.js +1 -0
  4. package/out/_next/static/chunks/app/app/[[...slug]]/{page-1e81c047cff17212.js → page-f7eca1b66fb4249b.js} +1 -1
  5. package/out/_next/static/chunks/app/{page-6892fe2dd07fb48b.js → page-0ee604f7070d14c0.js} +1 -1
  6. package/out/_next/static/css/8968d98ed4c4d33f.css +1 -0
  7. package/out/about.html +2 -2
  8. package/out/about.txt +1 -1
  9. package/out/app/onboarding.html +1 -1
  10. package/out/app/onboarding.txt +1 -1
  11. package/out/app.html +1 -1
  12. package/out/app.txt +2 -2
  13. package/out/blog/go-to-bed-wake-up-to-a-finished-product.html +2 -2
  14. package/out/blog/go-to-bed-wake-up-to-a-finished-product.txt +1 -1
  15. package/out/blog/let-them-cook-multi-agent-orchestration.html +2 -2
  16. package/out/blog/let-them-cook-multi-agent-orchestration.txt +2 -2
  17. package/out/blog.html +2 -2
  18. package/out/blog.txt +1 -1
  19. package/out/careers.html +2 -2
  20. package/out/careers.txt +1 -1
  21. package/out/changelog.html +2 -2
  22. package/out/changelog.txt +1 -1
  23. package/out/cloud/link.html +1 -1
  24. package/out/cloud/link.txt +2 -2
  25. package/out/complete-profile.html +2 -2
  26. package/out/complete-profile.txt +1 -1
  27. package/out/connect-repos.html +1 -1
  28. package/out/connect-repos.txt +1 -1
  29. package/out/contact.html +2 -2
  30. package/out/contact.txt +1 -1
  31. package/out/docs.html +2 -2
  32. package/out/docs.txt +1 -1
  33. package/out/history.html +1 -1
  34. package/out/history.txt +2 -2
  35. package/out/index.html +1 -1
  36. package/out/index.txt +2 -2
  37. package/out/login.html +2 -2
  38. package/out/login.txt +1 -1
  39. package/out/metrics.html +1 -1
  40. package/out/metrics.txt +2 -2
  41. package/out/pricing.html +2 -2
  42. package/out/pricing.txt +1 -1
  43. package/out/privacy.html +2 -2
  44. package/out/privacy.txt +1 -1
  45. package/out/providers/setup/claude.html +1 -1
  46. package/out/providers/setup/claude.txt +1 -1
  47. package/out/providers/setup/codex.html +1 -1
  48. package/out/providers/setup/codex.txt +1 -1
  49. package/out/providers/setup/cursor.html +1 -1
  50. package/out/providers/setup/cursor.txt +1 -1
  51. package/out/providers.html +1 -1
  52. package/out/providers.txt +1 -1
  53. package/out/security.html +2 -2
  54. package/out/security.txt +1 -1
  55. package/out/signup.html +2 -2
  56. package/out/signup.txt +1 -1
  57. package/out/terms.html +2 -2
  58. package/out/terms.txt +1 -1
  59. package/package.json +7 -1
  60. package/src/app/about/page.tsx +7 -0
  61. package/src/app/app/[[...slug]]/DashboardPageClient.tsx +853 -0
  62. package/src/app/app/[[...slug]]/page.tsx +23 -0
  63. package/src/app/app/onboarding/page.tsx +394 -0
  64. package/src/app/apple-icon.png +0 -0
  65. package/src/app/blog/go-to-bed-wake-up-to-a-finished-product/page.tsx +88 -0
  66. package/src/app/blog/let-them-cook-multi-agent-orchestration/page.tsx +93 -0
  67. package/src/app/blog/page.tsx +15 -0
  68. package/src/app/careers/page.tsx +7 -0
  69. package/src/app/changelog/page.tsx +7 -0
  70. package/src/app/cloud/link/page.tsx +464 -0
  71. package/src/app/complete-profile/page.tsx +204 -0
  72. package/src/app/connect-repos/page.tsx +410 -0
  73. package/src/app/contact/page.tsx +7 -0
  74. package/src/app/docs/page.tsx +7 -0
  75. package/src/app/favicon.png +0 -0
  76. package/src/app/globals.css +200 -0
  77. package/src/app/history/page.tsx +658 -0
  78. package/src/app/layout.tsx +25 -0
  79. package/src/app/login/page.tsx +424 -0
  80. package/src/app/metrics/page.tsx +781 -0
  81. package/src/app/page.tsx +59 -0
  82. package/src/app/pricing/page.tsx +7 -0
  83. package/src/app/privacy/page.tsx +7 -0
  84. package/src/app/providers/page.tsx +193 -0
  85. package/src/app/providers/setup/[provider]/ProviderSetupClient.tsx +197 -0
  86. package/src/app/providers/setup/[provider]/constants.ts +35 -0
  87. package/src/app/providers/setup/[provider]/page.tsx +42 -0
  88. package/src/app/security/page.tsx +7 -0
  89. package/src/app/signup/page.tsx +533 -0
  90. package/src/app/terms/page.tsx +7 -0
  91. package/src/components/ActivityFeed.tsx +216 -0
  92. package/src/components/AddWorkspaceModal.tsx +170 -0
  93. package/src/components/AgentCard.test.tsx +134 -0
  94. package/src/components/AgentCard.tsx +585 -0
  95. package/src/components/AgentList.test.tsx +147 -0
  96. package/src/components/AgentList.tsx +419 -0
  97. package/src/components/AgentLogPreview.tsx +173 -0
  98. package/src/components/AgentProfilePanel.tsx +569 -0
  99. package/src/components/App.tsx +3424 -0
  100. package/src/components/BillingPanel.tsx +922 -0
  101. package/src/components/BillingResult.tsx +447 -0
  102. package/src/components/BroadcastComposer.tsx +690 -0
  103. package/src/components/ChannelAdminPanel.tsx +773 -0
  104. package/src/components/ChannelBrowser.tsx +385 -0
  105. package/src/components/ChannelChat.tsx +261 -0
  106. package/src/components/ChannelSidebar.tsx +399 -0
  107. package/src/components/CloudSessionProvider.tsx +130 -0
  108. package/src/components/CommandPalette.tsx +815 -0
  109. package/src/components/ConfirmationDialog.tsx +133 -0
  110. package/src/components/ConversationHistory.tsx +518 -0
  111. package/src/components/CoordinatorPanel.tsx +956 -0
  112. package/src/components/DecisionQueue.tsx +717 -0
  113. package/src/components/DirectMessageView.tsx +164 -0
  114. package/src/components/FileAutocomplete.tsx +368 -0
  115. package/src/components/FleetOverview.tsx +278 -0
  116. package/src/components/LogViewer.tsx +310 -0
  117. package/src/components/LogViewerPanel.tsx +482 -0
  118. package/src/components/Logo.tsx +284 -0
  119. package/src/components/MentionAutocomplete.tsx +384 -0
  120. package/src/components/MessageComposer.tsx +473 -0
  121. package/src/components/MessageList.tsx +725 -0
  122. package/src/components/MessageSenderName.tsx +91 -0
  123. package/src/components/MessageStatusIndicator.tsx +142 -0
  124. package/src/components/NewConversationModal.tsx +400 -0
  125. package/src/components/NotificationToast.tsx +488 -0
  126. package/src/components/OnlineUsersIndicator.tsx +164 -0
  127. package/src/components/Pagination.tsx +124 -0
  128. package/src/components/PricingPlans.tsx +386 -0
  129. package/src/components/ProjectList.tsx +711 -0
  130. package/src/components/ProviderAuthFlow.tsx +343 -0
  131. package/src/components/ProviderConnectionList.tsx +375 -0
  132. package/src/components/ProvisioningProgress.tsx +730 -0
  133. package/src/components/ReactionChips.tsx +70 -0
  134. package/src/components/ReactionPicker.tsx +121 -0
  135. package/src/components/RepoAccessPanel.tsx +787 -0
  136. package/src/components/RepositoriesPanel.tsx +901 -0
  137. package/src/components/ServerCard.tsx +202 -0
  138. package/src/components/SessionExpiredModal.tsx +128 -0
  139. package/src/components/SpawnModal.test.tsx +190 -0
  140. package/src/components/SpawnModal.tsx +1001 -0
  141. package/src/components/TaskAssignmentUI.tsx +375 -0
  142. package/src/components/TerminalProviderSetup.tsx +517 -0
  143. package/src/components/ThemeProvider.tsx +159 -0
  144. package/src/components/ThinkingIndicator.tsx +231 -0
  145. package/src/components/ThreadList.tsx +198 -0
  146. package/src/components/ThreadPanel.tsx +405 -0
  147. package/src/components/TrajectoryViewer.tsx +698 -0
  148. package/src/components/TypingIndicator.tsx +69 -0
  149. package/src/components/UsageBanner.tsx +231 -0
  150. package/src/components/UserProfilePanel.tsx +233 -0
  151. package/src/components/WorkspaceContext.tsx +95 -0
  152. package/src/components/WorkspaceSelector.tsx +234 -0
  153. package/src/components/WorkspaceStatusIndicator.tsx +396 -0
  154. package/src/components/XTermInteractive.tsx +516 -0
  155. package/src/components/XTermLogViewer.tsx +719 -0
  156. package/src/components/channels/ChannelDialogs.tsx +1411 -0
  157. package/src/components/channels/ChannelHeader.tsx +317 -0
  158. package/src/components/channels/ChannelMessageList.tsx +463 -0
  159. package/src/components/channels/ChannelViewV1.tsx +146 -0
  160. package/src/components/channels/MessageInput.tsx +302 -0
  161. package/src/components/channels/SearchInput.tsx +172 -0
  162. package/src/components/channels/SearchResults.tsx +336 -0
  163. package/src/components/channels/api.test.ts +1527 -0
  164. package/src/components/channels/api.ts +703 -0
  165. package/src/components/channels/index.ts +76 -0
  166. package/src/components/channels/mockApi.ts +344 -0
  167. package/src/components/channels/types.ts +566 -0
  168. package/src/components/hooks/index.ts +58 -0
  169. package/src/components/hooks/useAgentLogs.ts +504 -0
  170. package/src/components/hooks/useAgents.ts +127 -0
  171. package/src/components/hooks/useBroadcastDedup.test.ts +371 -0
  172. package/src/components/hooks/useBroadcastDedup.ts +86 -0
  173. package/src/components/hooks/useChannelAdmin.ts +329 -0
  174. package/src/components/hooks/useChannelBrowser.ts +239 -0
  175. package/src/components/hooks/useChannelCommands.ts +138 -0
  176. package/src/components/hooks/useChannels.ts +367 -0
  177. package/src/components/hooks/useDebounce.ts +29 -0
  178. package/src/components/hooks/useDirectMessage.test.ts +952 -0
  179. package/src/components/hooks/useDirectMessage.ts +141 -0
  180. package/src/components/hooks/useMessages.ts +310 -0
  181. package/src/components/hooks/useOrchestrator.test.ts +165 -0
  182. package/src/components/hooks/useOrchestrator.ts +424 -0
  183. package/src/components/hooks/usePinnedAgents.test.ts +356 -0
  184. package/src/components/hooks/usePinnedAgents.ts +140 -0
  185. package/src/components/hooks/usePresence.test.ts +245 -0
  186. package/src/components/hooks/usePresence.ts +377 -0
  187. package/src/components/hooks/useRecentRepos.ts +130 -0
  188. package/src/components/hooks/useSession.ts +209 -0
  189. package/src/components/hooks/useThread.ts +138 -0
  190. package/src/components/hooks/useTrajectory.ts +265 -0
  191. package/src/components/hooks/useWebSocket.ts +290 -0
  192. package/src/components/hooks/useWorkspaceMembers.ts +132 -0
  193. package/src/components/hooks/useWorkspaceRepos.ts +73 -0
  194. package/src/components/hooks/useWorkspaceStatus.ts +237 -0
  195. package/src/components/index.ts +81 -0
  196. package/src/components/layout/Header.tsx +311 -0
  197. package/src/components/layout/RepoContextHeader.tsx +361 -0
  198. package/src/components/layout/Sidebar.archive.test.tsx +126 -0
  199. package/src/components/layout/Sidebar.test.tsx +691 -0
  200. package/src/components/layout/Sidebar.tsx +900 -0
  201. package/src/components/layout/index.ts +7 -0
  202. package/src/components/settings/BillingSettingsPanel.tsx +564 -0
  203. package/src/components/settings/SettingsPage.tsx +683 -0
  204. package/src/components/settings/TeamSettingsPanel.tsx +560 -0
  205. package/src/components/settings/WorkspaceSettingsPanel.tsx +1368 -0
  206. package/src/components/settings/index.ts +11 -0
  207. package/src/components/settings/types.ts +79 -0
  208. package/src/components/utils/messageFormatting.test.tsx +331 -0
  209. package/src/components/utils/messageFormatting.tsx +597 -0
  210. package/src/index.ts +63 -0
  211. package/src/landing/AboutPage.tsx +77 -0
  212. package/src/landing/BlogContent.tsx +187 -0
  213. package/src/landing/BlogPage.tsx +47 -0
  214. package/src/landing/CareersPage.tsx +53 -0
  215. package/src/landing/ChangelogPage.tsx +33 -0
  216. package/src/landing/ContactPage.tsx +41 -0
  217. package/src/landing/DocsPage.tsx +43 -0
  218. package/src/landing/LandingPage.tsx +702 -0
  219. package/src/landing/PricingPage.tsx +549 -0
  220. package/src/landing/PrivacyPage.tsx +117 -0
  221. package/src/landing/SecurityPage.tsx +42 -0
  222. package/src/landing/StaticPage.tsx +165 -0
  223. package/src/landing/TermsPage.tsx +125 -0
  224. package/src/landing/blogData.ts +312 -0
  225. package/src/landing/index.ts +18 -0
  226. package/src/landing/styles.css +3673 -0
  227. package/src/lib/agent-merge.test.ts +43 -0
  228. package/src/lib/agent-merge.ts +35 -0
  229. package/src/lib/api.ts +1294 -0
  230. package/src/lib/cloudApi.ts +893 -0
  231. package/src/lib/colors.test.ts +175 -0
  232. package/src/lib/colors.ts +218 -0
  233. package/src/lib/config.ts +109 -0
  234. package/src/lib/hierarchy.ts +242 -0
  235. package/src/lib/stuckDetection.ts +142 -0
  236. package/src/lib/useUrlRouting.ts +190 -0
  237. package/src/types/index.ts +317 -0
  238. package/src/types/threading.ts +7 -0
  239. package/out/_next/static/chunks/285-dc644487a8d6500d.js +0 -1
  240. package/out/_next/static/css/4c58d9cf493aa626.css +0 -1
  241. /package/out/_next/static/{AqelRhy1vr2nBUcU0Iqcp → IxfA6RZu4trcsEMYlkQra}/_buildManifest.js +0 -0
  242. /package/out/_next/static/{AqelRhy1vr2nBUcU0Iqcp → IxfA6RZu4trcsEMYlkQra}/_ssgManifest.js +0 -0
  243. /package/out/_next/static/chunks/{528-d375bc8b46912d2c.js → 528-f5f676996d613c25.js} +0 -0
  244. /package/out/_next/static/chunks/app/blog/let-them-cook-multi-agent-orchestration/{page-a58308f43557b908.js → page-b194f207fbd91862.js} +0 -0
@@ -0,0 +1,560 @@
1
+ /**
2
+ * Team Settings Panel
3
+ *
4
+ * Manage workspace team members, invitations, and roles.
5
+ */
6
+
7
+ import React, { useState, useEffect, useCallback } from 'react';
8
+ import { cloudApi } from '../../lib/cloudApi';
9
+
10
+ export interface TeamSettingsPanelProps {
11
+ workspaceId: string;
12
+ currentUserId?: string;
13
+ }
14
+
15
+ interface Member {
16
+ id: string;
17
+ userId: string;
18
+ role: 'owner' | 'admin' | 'member' | 'viewer';
19
+ isPending: boolean;
20
+ user?: {
21
+ githubUsername: string;
22
+ email?: string;
23
+ avatarUrl?: string;
24
+ };
25
+ }
26
+
27
+ interface PendingInvite {
28
+ id: string;
29
+ workspaceId: string;
30
+ workspaceName: string;
31
+ role: string;
32
+ invitedAt: string;
33
+ invitedBy: string;
34
+ }
35
+
36
+ interface RepoCollaborator {
37
+ id: number;
38
+ login: string;
39
+ avatarUrl: string;
40
+ permission: 'admin' | 'write' | 'read' | 'none';
41
+ repos: string[];
42
+ }
43
+
44
+ const ROLE_COLORS: Record<string, string> = {
45
+ owner: 'bg-accent-purple/20 text-accent-purple',
46
+ admin: 'bg-accent-cyan/20 text-accent-cyan',
47
+ member: 'bg-success/20 text-success',
48
+ viewer: 'bg-bg-hover text-text-muted',
49
+ };
50
+
51
+ const ROLE_DESCRIPTIONS: Record<string, string> = {
52
+ owner: 'Full access, can delete workspace and transfer ownership',
53
+ admin: 'Can manage members, settings, and all workspace features',
54
+ member: 'Can use workspace, spawn agents, and send messages',
55
+ viewer: 'Read-only access to workspace activity',
56
+ };
57
+
58
+ export function TeamSettingsPanel({
59
+ workspaceId,
60
+ currentUserId,
61
+ }: TeamSettingsPanelProps) {
62
+ const [members, setMembers] = useState<Member[]>([]);
63
+ const [pendingInvites, setPendingInvites] = useState<PendingInvite[]>([]);
64
+ const [repoCollaborators, setRepoCollaborators] = useState<RepoCollaborator[]>([]);
65
+ const [isLoading, setIsLoading] = useState(true);
66
+ const [collaboratorsLoading, setCollaboratorsLoading] = useState(true);
67
+ const [error, setError] = useState<string | null>(null);
68
+ const [successMessage, setSuccessMessage] = useState<string | null>(null);
69
+
70
+ // Invite form
71
+ const [showInviteForm, setShowInviteForm] = useState(false);
72
+ const [inviteUsername, setInviteUsername] = useState('');
73
+ const [inviteRole, setInviteRole] = useState<'admin' | 'member' | 'viewer'>('member');
74
+ const [inviteLoading, setInviteLoading] = useState(false);
75
+ const [inviteError, setInviteError] = useState<string | null>(null);
76
+
77
+ // Role change
78
+ const [changingRoleFor, setChangingRoleFor] = useState<string | null>(null);
79
+
80
+ // Load members
81
+ useEffect(() => {
82
+ async function loadMembers() {
83
+ setIsLoading(true);
84
+ setError(null);
85
+
86
+ const [membersResult, invitesResult] = await Promise.all([
87
+ cloudApi.getWorkspaceMembers(workspaceId),
88
+ cloudApi.getPendingInvites(),
89
+ ]);
90
+
91
+ if (membersResult.success) {
92
+ setMembers(membersResult.data.members as Member[]);
93
+ } else {
94
+ setError(membersResult.error);
95
+ }
96
+
97
+ if (invitesResult.success) {
98
+ // Filter to invites for this workspace
99
+ setPendingInvites(
100
+ invitesResult.data.invites.filter((i) => i.workspaceId === workspaceId)
101
+ );
102
+ }
103
+
104
+ setIsLoading(false);
105
+ }
106
+
107
+ loadMembers();
108
+ }, [workspaceId]);
109
+
110
+ // Load repo collaborators (users with GitHub access who aren't workspace members)
111
+ useEffect(() => {
112
+ async function loadCollaborators() {
113
+ setCollaboratorsLoading(true);
114
+
115
+ const result = await cloudApi.getRepoCollaborators(workspaceId);
116
+
117
+ if (result.success) {
118
+ setRepoCollaborators(result.data.collaborators || []);
119
+ }
120
+
121
+ setCollaboratorsLoading(false);
122
+ }
123
+
124
+ loadCollaborators();
125
+ }, [workspaceId]);
126
+
127
+ // Invite member
128
+ const handleInvite = useCallback(async () => {
129
+ if (!inviteUsername.trim()) {
130
+ setInviteError('Please enter a GitHub username');
131
+ return;
132
+ }
133
+
134
+ setInviteLoading(true);
135
+ setInviteError(null);
136
+
137
+ const result = await cloudApi.inviteMember(workspaceId, inviteUsername.trim(), inviteRole);
138
+
139
+ if (result.success) {
140
+ // Refresh members
141
+ const membersResult = await cloudApi.getWorkspaceMembers(workspaceId);
142
+ if (membersResult.success) {
143
+ setMembers(membersResult.data.members as Member[]);
144
+ }
145
+ setInviteUsername('');
146
+ setShowInviteForm(false);
147
+ setSuccessMessage(`Invitation sent to ${inviteUsername}`);
148
+ setTimeout(() => setSuccessMessage(null), 3000);
149
+ } else {
150
+ setInviteError(result.error);
151
+ }
152
+
153
+ setInviteLoading(false);
154
+ }, [workspaceId, inviteUsername, inviteRole]);
155
+
156
+ // Update member role
157
+ const handleUpdateRole = useCallback(async (memberId: string, newRole: string) => {
158
+ setChangingRoleFor(memberId);
159
+
160
+ const result = await cloudApi.updateMemberRole(workspaceId, memberId, newRole);
161
+
162
+ if (result.success) {
163
+ setMembers((prev) =>
164
+ prev.map((m) => (m.id === memberId ? { ...m, role: newRole as Member['role'] } : m))
165
+ );
166
+ setSuccessMessage('Role updated successfully');
167
+ setTimeout(() => setSuccessMessage(null), 3000);
168
+ } else {
169
+ setError(result.error);
170
+ }
171
+
172
+ setChangingRoleFor(null);
173
+ }, [workspaceId]);
174
+
175
+ // Remove member
176
+ const handleRemoveMember = useCallback(async (member: Member) => {
177
+ const confirmed = window.confirm(
178
+ `Are you sure you want to remove ${member.user?.githubUsername || 'this member'} from the workspace?`
179
+ );
180
+ if (!confirmed) return;
181
+
182
+ const result = await cloudApi.removeMember(workspaceId, member.id);
183
+
184
+ if (result.success) {
185
+ setMembers((prev) => prev.filter((m) => m.id !== member.id));
186
+ setSuccessMessage('Member removed successfully');
187
+ setTimeout(() => setSuccessMessage(null), 3000);
188
+ } else {
189
+ setError(result.error);
190
+ }
191
+ }, [workspaceId]);
192
+
193
+ // Get current user's role
194
+ const currentUserRole = members.find((m) => m.userId === currentUserId)?.role;
195
+ const canManageMembers = currentUserRole === 'owner' || currentUserRole === 'admin';
196
+
197
+ if (isLoading) {
198
+ return (
199
+ <div className="flex items-center justify-center h-64">
200
+ <LoadingSpinner />
201
+ <span className="ml-3 text-text-muted">Loading team members...</span>
202
+ </div>
203
+ );
204
+ }
205
+
206
+ return (
207
+ <div className="space-y-6">
208
+ {/* Header */}
209
+ <div className="flex flex-col sm:flex-row sm:items-center sm:justify-between gap-3">
210
+ <div>
211
+ <h3 className="text-sm font-semibold text-text-muted uppercase tracking-wide">
212
+ Team Members
213
+ </h3>
214
+ <p className="text-xs text-text-muted mt-1">
215
+ {members.length} member{members.length !== 1 ? 's' : ''}
216
+ </p>
217
+ </div>
218
+ {canManageMembers && (
219
+ <button
220
+ onClick={() => setShowInviteForm(!showInviteForm)}
221
+ 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 transition-colors flex items-center justify-center gap-2 w-full sm:w-auto"
222
+ >
223
+ <PlusIcon />
224
+ Invite Member
225
+ </button>
226
+ )}
227
+ </div>
228
+
229
+ {/* Messages */}
230
+ {error && (
231
+ <div className="p-3 bg-error/10 border border-error/30 rounded-lg text-error text-sm">
232
+ {error}
233
+ <button
234
+ onClick={() => setError(null)}
235
+ className="ml-2 text-error/70 hover:text-error"
236
+ >
237
+ &times;
238
+ </button>
239
+ </div>
240
+ )}
241
+
242
+ {successMessage && (
243
+ <div className="p-3 bg-success/10 border border-success/30 rounded-lg text-success text-sm">
244
+ {successMessage}
245
+ </div>
246
+ )}
247
+
248
+ {/* Invite Form */}
249
+ {showInviteForm && (
250
+ <div className="p-4 bg-bg-tertiary rounded-lg border border-border-subtle space-y-4">
251
+ <h4 className="text-sm font-medium text-text-primary">Invite New Member</h4>
252
+
253
+ {inviteError && (
254
+ <div className="p-2 bg-error/10 border border-error/30 rounded text-error text-xs">
255
+ {inviteError}
256
+ </div>
257
+ )}
258
+
259
+ <div className="grid grid-cols-1 sm:grid-cols-2 gap-4">
260
+ <div>
261
+ <label className="text-xs text-text-muted mb-1 block">GitHub Username</label>
262
+ <input
263
+ type="text"
264
+ value={inviteUsername}
265
+ onChange={(e) => setInviteUsername(e.target.value)}
266
+ placeholder="username"
267
+ className="w-full px-3 py-2 bg-bg-card border border-border-subtle rounded-lg text-sm text-text-primary placeholder:text-text-muted focus:outline-none focus:border-accent-cyan"
268
+ />
269
+ </div>
270
+ <div>
271
+ <label className="text-xs text-text-muted mb-1 block">Role</label>
272
+ <select
273
+ value={inviteRole}
274
+ onChange={(e) => setInviteRole(e.target.value as typeof inviteRole)}
275
+ className="w-full px-3 py-2 bg-bg-card border border-border-subtle rounded-lg text-sm text-text-primary focus:outline-none focus:border-accent-cyan"
276
+ >
277
+ <option value="admin">Admin</option>
278
+ <option value="member">Member</option>
279
+ <option value="viewer">Viewer</option>
280
+ </select>
281
+ </div>
282
+ </div>
283
+
284
+ <p className="text-xs text-text-muted">
285
+ {ROLE_DESCRIPTIONS[inviteRole]}
286
+ </p>
287
+
288
+ <div className="flex flex-col sm:flex-row gap-2">
289
+ <button
290
+ onClick={handleInvite}
291
+ disabled={inviteLoading || !inviteUsername.trim()}
292
+ className="px-4 py-2 bg-accent-cyan text-bg-deep rounded-lg text-sm font-medium hover:bg-accent-cyan/90 disabled:opacity-50 transition-colors"
293
+ >
294
+ {inviteLoading ? 'Sending...' : 'Send Invitation'}
295
+ </button>
296
+ <button
297
+ onClick={() => {
298
+ setShowInviteForm(false);
299
+ setInviteUsername('');
300
+ setInviteError(null);
301
+ }}
302
+ className="px-4 py-2 bg-bg-hover text-text-secondary rounded-lg text-sm font-medium hover:text-text-primary transition-colors"
303
+ >
304
+ Cancel
305
+ </button>
306
+ </div>
307
+ </div>
308
+ )}
309
+
310
+ {/* Members List */}
311
+ <div className="space-y-2">
312
+ {members.map((member) => (
313
+ <div
314
+ key={member.id}
315
+ className="flex flex-col sm:flex-row sm:items-center sm:justify-between gap-3 p-3 md:p-4 bg-bg-tertiary rounded-lg"
316
+ >
317
+ <div className="flex items-center gap-3">
318
+ {member.user?.avatarUrl ? (
319
+ <img
320
+ src={member.user.avatarUrl}
321
+ alt={member.user.githubUsername}
322
+ className="w-9 h-9 md:w-10 md:h-10 rounded-full"
323
+ />
324
+ ) : (
325
+ <div className="w-9 h-9 md:w-10 md:h-10 rounded-full bg-accent-cyan/20 flex items-center justify-center text-accent-cyan font-bold text-xs md:text-sm">
326
+ {member.user?.githubUsername?.[0]?.toUpperCase() || '?'}
327
+ </div>
328
+ )}
329
+ <div className="min-w-0 flex-1">
330
+ <div className="flex flex-wrap items-center gap-2">
331
+ <p className="text-sm font-medium text-text-primary truncate">
332
+ {member.user?.githubUsername || 'Unknown User'}
333
+ </p>
334
+ {member.isPending && (
335
+ <span className="text-[10px] px-2 py-0.5 bg-amber-400/20 text-amber-400 rounded-full">
336
+ Pending
337
+ </span>
338
+ )}
339
+ {member.userId === currentUserId && (
340
+ <span className="text-xs text-text-muted">(you)</span>
341
+ )}
342
+ </div>
343
+ {member.user?.email && (
344
+ <p className="text-xs text-text-muted truncate">{member.user.email}</p>
345
+ )}
346
+ </div>
347
+ </div>
348
+
349
+ <div className="flex items-center gap-2 sm:gap-3 ml-12 sm:ml-0">
350
+ {canManageMembers && member.role !== 'owner' && member.userId !== currentUserId ? (
351
+ <select
352
+ value={member.role}
353
+ onChange={(e) => handleUpdateRole(member.id, e.target.value)}
354
+ disabled={changingRoleFor === member.id}
355
+ className={`px-2 md:px-3 py-1 md:py-1.5 rounded-full text-[10px] md:text-xs font-medium border-none cursor-pointer ${ROLE_COLORS[member.role]} focus:outline-none`}
356
+ >
357
+ <option value="admin">Admin</option>
358
+ <option value="member">Member</option>
359
+ <option value="viewer">Viewer</option>
360
+ </select>
361
+ ) : (
362
+ <span className={`px-2 md:px-3 py-1 md:py-1.5 rounded-full text-[10px] md:text-xs font-medium ${ROLE_COLORS[member.role]}`}>
363
+ {member.role.charAt(0).toUpperCase() + member.role.slice(1)}
364
+ </span>
365
+ )}
366
+
367
+ {canManageMembers && member.role !== 'owner' && member.userId !== currentUserId && (
368
+ <button
369
+ onClick={() => handleRemoveMember(member)}
370
+ className="p-1.5 text-text-muted hover:text-error rounded transition-colors"
371
+ title="Remove member"
372
+ >
373
+ <TrashIcon />
374
+ </button>
375
+ )}
376
+ </div>
377
+ </div>
378
+ ))}
379
+ </div>
380
+
381
+ {/* Repo Collaborators (Grandfathered Access via GitHub) */}
382
+ {(repoCollaborators.length > 0 || collaboratorsLoading) && (
383
+ <div className="mt-8">
384
+ <div className="flex items-center gap-2 mb-4">
385
+ <h3 className="text-sm font-semibold text-text-muted uppercase tracking-wide">
386
+ Repo Collaborators
387
+ </h3>
388
+ <span className="px-2 py-0.5 bg-accent-purple/10 text-accent-purple text-[10px] rounded-full">
389
+ GitHub Access
390
+ </span>
391
+ </div>
392
+ <p className="text-xs text-text-muted mb-4">
393
+ These users have access via GitHub repo permissions. They can access this workspace without an explicit invite.
394
+ </p>
395
+ {collaboratorsLoading ? (
396
+ <div className="flex items-center gap-2 py-4">
397
+ <LoadingSpinner />
398
+ <span className="text-sm text-text-muted">Loading collaborators...</span>
399
+ </div>
400
+ ) : (
401
+ <div className="space-y-2">
402
+ {repoCollaborators.map((collab) => (
403
+ <div
404
+ key={collab.id}
405
+ className="flex flex-col sm:flex-row sm:items-center sm:justify-between gap-3 p-3 md:p-4 bg-bg-tertiary rounded-lg border border-accent-purple/20"
406
+ >
407
+ <div className="flex items-center gap-3">
408
+ {collab.avatarUrl ? (
409
+ <img
410
+ src={collab.avatarUrl}
411
+ alt={collab.login}
412
+ className="w-9 h-9 md:w-10 md:h-10 rounded-full"
413
+ />
414
+ ) : (
415
+ <div className="w-9 h-9 md:w-10 md:h-10 rounded-full bg-accent-purple/20 flex items-center justify-center text-accent-purple font-bold text-xs md:text-sm">
416
+ {collab.login[0]?.toUpperCase() || '?'}
417
+ </div>
418
+ )}
419
+ <div className="min-w-0 flex-1">
420
+ <div className="flex flex-wrap items-center gap-2">
421
+ <p className="text-sm font-medium text-text-primary truncate">
422
+ {collab.login}
423
+ </p>
424
+ <span className="text-[10px] px-2 py-0.5 bg-accent-purple/10 text-accent-purple rounded-full">
425
+ via GitHub
426
+ </span>
427
+ </div>
428
+ <p className="text-xs text-text-muted truncate">
429
+ {collab.repos.length === 1
430
+ ? `Access via ${collab.repos[0]}`
431
+ : `Access via ${collab.repos.length} repos`}
432
+ </p>
433
+ </div>
434
+ </div>
435
+
436
+ <div className="flex items-center gap-2 sm:gap-3 ml-12 sm:ml-0">
437
+ <span className={`px-2 md:px-3 py-1 md:py-1.5 rounded-full text-[10px] md:text-xs font-medium ${
438
+ collab.permission === 'admin'
439
+ ? 'bg-accent-cyan/20 text-accent-cyan'
440
+ : collab.permission === 'write'
441
+ ? 'bg-success/20 text-success'
442
+ : 'bg-bg-hover text-text-muted'
443
+ }`}>
444
+ {collab.permission === 'admin' ? 'Admin' : collab.permission === 'write' ? 'Write' : 'Read'}
445
+ </span>
446
+ </div>
447
+ </div>
448
+ ))}
449
+ </div>
450
+ )}
451
+ </div>
452
+ )}
453
+
454
+ {/* Pending Invites for Current User */}
455
+ {pendingInvites.length > 0 && (
456
+ <div className="mt-8">
457
+ <h3 className="text-sm font-semibold text-text-muted uppercase tracking-wide mb-4">
458
+ Your Pending Invitations
459
+ </h3>
460
+ <div className="space-y-2">
461
+ {pendingInvites.map((invite) => (
462
+ <div
463
+ key={invite.id}
464
+ className="flex items-center justify-between p-4 bg-bg-tertiary rounded-lg border border-accent-cyan/30"
465
+ >
466
+ <div>
467
+ <p className="text-sm font-medium text-text-primary">
468
+ {invite.workspaceName}
469
+ </p>
470
+ <p className="text-xs text-text-muted">
471
+ Invited by {invite.invitedBy} as {invite.role}
472
+ </p>
473
+ </div>
474
+ <div className="flex gap-2">
475
+ <button
476
+ onClick={async () => {
477
+ const result = await cloudApi.acceptInvite(invite.id);
478
+ if (result.success) {
479
+ setPendingInvites((prev) => prev.filter((i) => i.id !== invite.id));
480
+ setSuccessMessage('Invitation accepted!');
481
+ setTimeout(() => setSuccessMessage(null), 3000);
482
+ }
483
+ }}
484
+ className="px-3 py-1.5 bg-success/20 text-success rounded text-xs font-medium hover:bg-success/30 transition-colors"
485
+ >
486
+ Accept
487
+ </button>
488
+ <button
489
+ onClick={async () => {
490
+ const result = await cloudApi.declineInvite(invite.id);
491
+ if (result.success) {
492
+ setPendingInvites((prev) => prev.filter((i) => i.id !== invite.id));
493
+ }
494
+ }}
495
+ className="px-3 py-1.5 bg-bg-hover text-text-muted rounded text-xs font-medium hover:text-text-primary transition-colors"
496
+ >
497
+ Decline
498
+ </button>
499
+ </div>
500
+ </div>
501
+ ))}
502
+ </div>
503
+ </div>
504
+ )}
505
+
506
+ {/* Role Permissions Info */}
507
+ <div className="mt-8 p-4 bg-bg-tertiary/50 rounded-lg">
508
+ <h4 className="text-xs font-semibold text-text-muted uppercase tracking-wide mb-3">
509
+ Role Permissions
510
+ </h4>
511
+ <div className="space-y-2">
512
+ {Object.entries(ROLE_DESCRIPTIONS).map(([role, description]) => (
513
+ <div key={role} className="flex items-start gap-2">
514
+ <span className={`px-2 py-0.5 rounded-full text-xs font-medium ${ROLE_COLORS[role]} shrink-0`}>
515
+ {role.charAt(0).toUpperCase() + role.slice(1)}
516
+ </span>
517
+ <p className="text-xs text-text-muted">{description}</p>
518
+ </div>
519
+ ))}
520
+ </div>
521
+ </div>
522
+ </div>
523
+ );
524
+ }
525
+
526
+ // Icons
527
+ function LoadingSpinner() {
528
+ return (
529
+ <svg className="animate-spin h-5 w-5 text-accent-cyan" viewBox="0 0 24 24">
530
+ <circle
531
+ cx="12"
532
+ cy="12"
533
+ r="10"
534
+ stroke="currentColor"
535
+ strokeWidth="2"
536
+ fill="none"
537
+ strokeDasharray="32"
538
+ strokeLinecap="round"
539
+ />
540
+ </svg>
541
+ );
542
+ }
543
+
544
+ function PlusIcon() {
545
+ return (
546
+ <svg width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" strokeWidth="2.5" strokeLinecap="round" strokeLinejoin="round">
547
+ <line x1="12" y1="5" x2="12" y2="19" />
548
+ <line x1="5" y1="12" x2="19" y2="12" />
549
+ </svg>
550
+ );
551
+ }
552
+
553
+ function TrashIcon() {
554
+ return (
555
+ <svg width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" strokeWidth="2" strokeLinecap="round" strokeLinejoin="round">
556
+ <polyline points="3 6 5 6 21 6" />
557
+ <path d="M19 6v14a2 2 0 0 1-2 2H7a2 2 0 0 1-2-2V6m3 0V4a2 2 0 0 1 2-2h4a2 2 0 0 1 2 2v2" />
558
+ </svg>
559
+ );
560
+ }