@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.
- package/out/404.html +1 -1
- package/out/_next/static/chunks/1028-da5d75e35d1420f1.js +1 -0
- package/out/_next/static/chunks/1528-78b17000a7e10bc6.js +2 -0
- package/out/_next/static/chunks/1695-4a5d33ba715e09b4.js +1 -0
- package/out/_next/static/chunks/1705-36c2180d00a4a569.js +1 -0
- package/out/_next/static/chunks/1dd3208c-e1f87c7b3dc1a820.js +1 -0
- package/out/_next/static/chunks/3663-47290254b8f6f5dd.js +1 -0
- package/out/_next/static/chunks/3677-4b225baf4801d9b9.js +73 -0
- package/out/_next/static/chunks/5118-7e8ada2df38eef07.js +1 -0
- package/out/_next/static/chunks/5888-15cbe97c90ed5fae.js +1 -0
- package/out/_next/static/chunks/6773-a45343a98df3abb5.js +1 -0
- package/out/_next/static/chunks/6940-b824612b605e79b3.js +9 -0
- package/out/_next/static/chunks/7894-f4a15249082a680d.js +1 -0
- package/out/_next/static/chunks/9175-b3617c1e5cbfed0e.js +1 -0
- package/out/_next/static/chunks/9372-1a804b8d08c7a236.js +1 -0
- package/out/_next/static/chunks/{ab6c8a12-0a58072fbb505134.js → ab6c8a12-91438a812d94ecf0.js} +1 -1
- package/out/_next/static/chunks/app/_not-found/page-8e8842f82d204726.js +1 -0
- package/out/_next/static/chunks/app/about/page-b78577a7da8fa459.js +1 -0
- package/out/_next/static/chunks/app/app/[[...slug]]/page-3dffd65b6344f53e.js +1 -0
- package/out/_next/static/chunks/app/app/onboarding/page-b89be9aa6264a5e1.js +1 -0
- package/out/_next/static/chunks/app/blog/go-to-bed-wake-up-to-a-finished-product/page-fbd00893ef69e499.js +1 -0
- package/out/_next/static/chunks/app/blog/let-them-cook-multi-agent-orchestration/page-de2ea13649d0b6d3.js +1 -0
- package/out/_next/static/chunks/app/blog/page-a08e263c57a156fa.js +1 -0
- package/out/_next/static/chunks/app/careers/page-02228e1d6969b232.js +1 -0
- package/out/_next/static/chunks/app/changelog/page-1b5c1d79efc6e53a.js +1 -0
- package/out/_next/static/chunks/app/cloud/link/page-99654edffffb3af2.js +1 -0
- package/out/_next/static/chunks/app/complete-profile/page-59d146e5ddeafc5c.js +1 -0
- package/out/_next/static/chunks/app/connect-repos/page-995e16a976a6632c.js +1 -0
- package/out/_next/static/chunks/app/contact/page-273396a5ad57bcee.js +1 -0
- package/out/_next/static/chunks/app/dev/cli-tools/page-a71b80dcb2d5fc8d.js +1 -0
- package/out/_next/static/chunks/app/dev/log-viewer/page-46a6151ae1be0796.js +1 -0
- package/out/_next/static/chunks/app/docs/page-7c7cb603b24b7c40.js +1 -0
- package/out/_next/static/chunks/app/history/page-0c5cab1dab4e8886.js +1 -0
- package/out/_next/static/chunks/app/layout-96d72ba8ef8a43a0.js +1 -0
- package/out/_next/static/chunks/app/login/page-0ccbab34213df842.js +1 -0
- package/out/_next/static/chunks/app/metrics/page-8616272aeab9c8b0.js +1 -0
- package/out/_next/static/chunks/app/page-09ce10603ad9a251.js +1 -0
- package/out/_next/static/chunks/app/pricing/page-91c975079120c941.js +1 -0
- package/out/_next/static/chunks/app/privacy/{page-c21d51ac2dee3a88.js → page-a49ab271cc686644.js} +1 -1
- package/out/_next/static/chunks/app/providers/{page-59114505f4353512.js → page-d775d6eb5bc29e96.js} +1 -1
- package/out/_next/static/chunks/app/providers/setup/[provider]/page-ec4ef3cd80de807e.js +1 -0
- package/out/_next/static/chunks/app/security/page-d9da9bd9191e8f95.js +1 -0
- package/out/_next/static/chunks/app/signup/page-930eca0bf5fd299d.js +1 -0
- package/out/_next/static/chunks/app/terms/page-3e4827620b98613c.js +1 -0
- package/out/_next/static/chunks/framework-648e1ae7da590300.js +1 -0
- package/out/_next/static/chunks/{main-acb1b24265295d6a.js → main-2b1990080c292d92.js} +1 -1
- package/out/_next/static/chunks/main-app-9f6b7ff9e754a8f5.js +1 -0
- package/out/_next/static/chunks/pages/_app-a077b72e02273ab1.js +1 -0
- package/out/_next/static/chunks/pages/_error-84001666436a04e4.js +1 -0
- package/out/_next/static/chunks/{webpack-dd93b81e2659669c.js → webpack-7586035f1585f2db.js} +1 -1
- package/out/_next/static/css/eb9fc69d1e3d2bed.css +1 -0
- package/out/_next/static/{IxfA6RZu4trcsEMYlkQra → g3G0LMdB7lxcrU5mdM54m}/_buildManifest.js +1 -1
- package/out/about.html +2 -2
- package/out/about.txt +2 -2
- package/out/app/onboarding.html +1 -1
- package/out/app/onboarding.txt +2 -2
- 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 +3 -3
- 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 +2 -2
- package/out/changelog.html +2 -2
- package/out/changelog.txt +2 -2
- 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 +2 -2
- package/out/connect-repos.html +1 -1
- package/out/connect-repos.txt +2 -2
- package/out/contact.html +2 -2
- package/out/contact.txt +2 -2
- package/out/dev/cli-tools.html +1 -0
- package/out/dev/cli-tools.txt +7 -0
- package/out/dev/log-viewer.html +23 -0
- package/out/dev/log-viewer.txt +7 -0
- package/out/docs.html +2 -2
- package/out/docs.txt +2 -2
- 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 +2 -2
- package/out/metrics.html +1 -1
- package/out/metrics.txt +2 -2
- package/out/pricing.html +2 -2
- package/out/pricing.txt +2 -2
- package/out/privacy.html +2 -2
- package/out/privacy.txt +2 -2
- package/out/providers/setup/claude.html +1 -1
- package/out/providers/setup/claude.txt +2 -2
- package/out/providers/setup/codex.html +1 -1
- package/out/providers/setup/codex.txt +2 -2
- package/out/providers/setup/cursor.html +1 -1
- package/out/providers/setup/cursor.txt +2 -2
- package/out/providers.html +1 -1
- package/out/providers.txt +2 -2
- package/out/security.html +2 -2
- package/out/security.txt +2 -2
- package/out/signup.html +2 -2
- package/out/signup.txt +2 -2
- package/out/terms.html +2 -2
- package/out/terms.txt +2 -2
- package/package.json +5 -1
- package/src/adapters/DashboardConfigProvider.tsx +56 -0
- package/src/adapters/cloudFetchAdapter.ts +278 -0
- package/src/adapters/index.ts +3 -0
- package/src/adapters/types.ts +508 -0
- package/src/app/app/[[...slug]]/DashboardPageClient.tsx +67 -18
- package/src/app/app/onboarding/page.tsx +870 -170
- package/src/app/cloud/link/page.tsx +14 -6
- package/src/app/connect-repos/page.tsx +9 -3
- package/src/app/dev/cli-tools/page.tsx +130 -0
- package/src/app/dev/log-viewer/MockLogViewer.tsx +132 -0
- package/src/app/dev/log-viewer/fixtures.ts +110 -0
- package/src/app/dev/log-viewer/page.tsx +288 -0
- package/src/app/history/page.tsx +28 -12
- package/src/app/page.tsx +1 -1
- package/src/app/providers/setup/[provider]/ProviderSetupClient.tsx +209 -59
- package/src/components/AgentCard.tsx +4 -4
- package/src/components/AgentLogPreview.tsx +2 -38
- package/src/components/App.tsx +441 -2624
- package/src/components/CliToolHarness.test.tsx +83 -0
- package/src/components/CliToolHarness.tsx +292 -0
- package/src/components/CoordinatorPanel.tsx +13 -6
- package/src/components/LogViewer.tsx +2 -42
- package/src/components/ProviderAuthFlow.tsx +201 -81
- package/src/components/ProvisioningProgress.tsx +1 -1
- package/src/components/ReactionChips.tsx +2 -1
- package/src/components/SpawnModal.test.tsx +51 -18
- package/src/components/SpawnModal.tsx +175 -207
- package/src/components/TerminalProviderSetup.tsx +1 -1
- package/src/components/ThreadPanel.tsx +2 -0
- package/src/components/WorkspaceContext.tsx +7 -19
- package/src/components/XTermLogViewer.tsx +190 -27
- package/src/components/channels/ChannelMessageList.tsx +94 -4
- package/src/components/channels/ChannelViewV1.tsx +35 -11
- package/src/components/channels/api.ts +21 -20
- package/src/components/channels/types.ts +16 -0
- package/src/components/hooks/index.ts +0 -19
- package/src/components/hooks/useMessages.test.ts +80 -0
- package/src/components/hooks/useMessages.ts +13 -4
- package/src/components/hooks/useOrchestrator.ts +1 -1
- package/src/components/hooks/usePresence.ts +45 -6
- package/src/components/hooks/useThread.ts +83 -46
- package/src/components/hooks/useTrajectory.ts +62 -5
- package/src/components/hooks/useWebSocket.test.ts +358 -0
- package/src/components/hooks/useWebSocket.ts +243 -5
- package/src/components/index.ts +2 -14
- package/src/components/layout/Header.tsx +9 -15
- package/src/components/layout/Sidebar.tsx +1 -8
- package/src/components/settings/SettingsPage.tsx +108 -47
- package/src/components/settings/index.ts +0 -3
- package/src/landing/blogData.ts +1 -1
- package/src/lib/agent-merge.test.ts +2 -2
- package/src/lib/api.ts +8 -38
- package/src/lib/identity.test.ts +139 -0
- package/src/lib/identity.ts +48 -0
- package/src/lib/relaycastMessageAdapters.test.ts +182 -0
- package/src/lib/relaycastMessageAdapters.ts +105 -0
- package/src/lib/sanitize-logs.test.ts +227 -0
- package/src/lib/sanitize-logs.ts +202 -0
- package/src/providers/AgentProvider.tsx +799 -0
- package/src/providers/ChannelProvider.tsx +528 -0
- package/src/providers/CloudWorkspaceProvider.tsx +402 -0
- package/src/providers/MessageProvider.tsx +875 -0
- package/src/providers/RelayConfigProvider.tsx +94 -0
- package/src/providers/SendProvider.tsx +497 -0
- package/src/providers/SettingsProvider.tsx +247 -0
- package/src/providers/index.ts +26 -0
- package/src/types/index.ts +10 -10
- package/out/_next/static/chunks/11-9a2993a37266dcb3.js +0 -9
- package/out/_next/static/chunks/118-ae2b650136a5a5fc.js +0 -1
- package/out/_next/static/chunks/1dd3208c-40ab0fc0f60392b8.js +0 -1
- package/out/_next/static/chunks/202-fc0763dd7488e58f.js +0 -1
- package/out/_next/static/chunks/259-83b77fa1b91ba5aa.js +0 -1
- package/out/_next/static/chunks/407-0c82986cf79c8ecb.js +0 -1
- package/out/_next/static/chunks/528-f5f676996d613c25.js +0 -2
- package/out/_next/static/chunks/663-ddb04081febc3678.js +0 -1
- package/out/_next/static/chunks/687-88b6b139a6bb0e2e.js +0 -1
- package/out/_next/static/chunks/695-51d25b1988644374.js +0 -1
- package/out/_next/static/chunks/773-54a2641043c81e55.js +0 -1
- package/out/_next/static/chunks/app/_not-found/page-6da9b72091e5b511.js +0 -1
- package/out/_next/static/chunks/app/about/page-fff7c6457683f243.js +0 -1
- package/out/_next/static/chunks/app/app/[[...slug]]/page-f7eca1b66fb4249b.js +0 -1
- package/out/_next/static/chunks/app/app/onboarding/page-129abc5da2e67971.js +0 -1
- package/out/_next/static/chunks/app/blog/go-to-bed-wake-up-to-a-finished-product/page-5d5f28fd126b692f.js +0 -1
- package/out/_next/static/chunks/app/blog/let-them-cook-multi-agent-orchestration/page-b194f207fbd91862.js +0 -1
- package/out/_next/static/chunks/app/blog/page-b9bd9d8703fca76a.js +0 -1
- package/out/_next/static/chunks/app/careers/page-a4bd8d5f4de8f4eb.js +0 -1
- package/out/_next/static/chunks/app/changelog/page-9a1f6ad1743d63c5.js +0 -1
- package/out/_next/static/chunks/app/cloud/link/page-0844c5699b027c3b.js +0 -1
- package/out/_next/static/chunks/app/complete-profile/page-39ed5a67916beb87.js +0 -1
- package/out/_next/static/chunks/app/connect-repos/page-297eddee0c39f2a3.js +0 -1
- package/out/_next/static/chunks/app/contact/page-3c1dd8690217fade.js +0 -1
- package/out/_next/static/chunks/app/docs/page-1875e981f2c3fd13.js +0 -1
- package/out/_next/static/chunks/app/history/page-2d5c5695c9e8b40c.js +0 -1
- package/out/_next/static/chunks/app/layout-0a4b99656da25511.js +0 -1
- package/out/_next/static/chunks/app/login/page-f69c076f5a6fc520.js +0 -1
- package/out/_next/static/chunks/app/metrics/page-bebbee055669a17e.js +0 -1
- package/out/_next/static/chunks/app/page-0ee604f7070d14c0.js +0 -1
- package/out/_next/static/chunks/app/pricing/page-eeae7d594af333b6.js +0 -1
- package/out/_next/static/chunks/app/providers/setup/[provider]/page-daf9b3e05e77ae19.js +0 -1
- package/out/_next/static/chunks/app/security/page-cd562730fe84a0a2.js +0 -1
- package/out/_next/static/chunks/app/signup/page-c242ca08101a84ff.js +0 -1
- package/out/_next/static/chunks/app/terms/page-c7001720e7941dc6.js +0 -1
- package/out/_next/static/chunks/framework-3664cab31236a9fa.js +0 -1
- package/out/_next/static/chunks/main-app-7f73a939a312a228.js +0 -1
- package/out/_next/static/chunks/pages/_app-10a93ab5b7c32eb3.js +0 -1
- package/out/_next/static/chunks/pages/_error-2d792b2a41857be4.js +0 -1
- package/out/_next/static/css/8968d98ed4c4d33f.css +0 -1
- package/src/components/BillingResult.tsx +0 -447
- package/src/components/CloudSessionProvider.tsx +0 -130
- package/src/components/SessionExpiredModal.tsx +0 -128
- package/src/components/WorkspaceStatusIndicator.tsx +0 -396
- package/src/components/hooks/useSession.ts +0 -209
- package/src/components/hooks/useWorkspaceMembers.ts +0 -132
- package/src/components/hooks/useWorkspaceStatus.ts +0 -237
- package/src/components/settings/BillingSettingsPanel.tsx +0 -564
- package/src/components/settings/TeamSettingsPanel.tsx +0 -560
- package/src/components/settings/WorkspaceSettingsPanel.tsx +0 -1368
- package/src/lib/cloudApi.ts +0 -893
- /package/out/_next/static/{IxfA6RZu4trcsEMYlkQra → g3G0LMdB7lxcrU5mdM54m}/_ssgManifest.js +0 -0
|
@@ -1,560 +0,0 @@
|
|
|
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
|
-
×
|
|
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
|
-
}
|