import * as plugins from '../../../plugins.js'; import { customElement, DeesElement, property, html, cssManager, css, state, type TemplateResult, } from '@design.estate/dees-element'; import { accountDesignTokens } from '../sharedstyles.js'; import * as accountStateModule from '../../../states/accountstate.js'; import { IdpState } from '../../../states/idp.state.js'; declare global { interface HTMLElementTagNameMap { 'lele-accountview-baseview': BaseView; } } interface ISessionDisplay { id: string; deviceId: string; deviceName: string; browser: string; os: string; ip: string; lastActive: number; createdAt: number; isCurrent: boolean; } interface IActivityDisplay { id: string; data: plugins.idpInterfaces.data.IActivityLog['data']; } @customElement('lele-accountview-baseview') export class BaseView extends DeesElement { @state() accessor loading: boolean = true; @state() accessor sessions: ISessionDisplay[] = []; @state() accessor activities: IActivityDisplay[] = []; @state() accessor user: plugins.idpInterfaces.data.IUser | null = null; @state() accessor organizations: plugins.idpInterfaces.data.IOrganization[] = []; @state() accessor roles: plugins.idpInterfaces.data.IRole[] = []; public static styles = [ cssManager.defaultStyles, accountDesignTokens, css` :host { display: block; min-height: 100%; background: var(--background); color: var(--foreground); } .container { max-width: 1000px; margin: 0 auto; padding: 32px 24px; } .header { margin-bottom: 32px; } h1 { font-size: 32px; font-weight: 600; margin: 0; letter-spacing: -0.02em; } .subtitle { color: #71717a; margin-top: 8px; font-size: 14px; } .dashboard-grid { display: grid; grid-template-columns: 1fr 1fr; gap: 24px; } @media (max-width: 768px) { .dashboard-grid { grid-template-columns: 1fr; } } .card { background: #18181b; border: 1px solid #27272a; border-radius: 12px; overflow: hidden; } .card.full-width { grid-column: 1 / -1; } .card-header { display: flex; justify-content: space-between; align-items: center; padding: 16px 20px; border-bottom: 1px solid #27272a; } .card-title { font-size: 16px; font-weight: 600; display: flex; align-items: center; gap: 8px; } .card-title dees-icon { opacity: 0.7; } .card-body { padding: 16px 20px; } .card-body.no-padding { padding: 0; } /* Profile Card */ .profile-info { display: flex; align-items: center; gap: 16px; } .avatar { width: 64px; height: 64px; border-radius: 50%; background: linear-gradient(135deg, #3b82f6, #8b5cf6); display: flex; align-items: center; justify-content: center; font-size: 24px; font-weight: 600; color: white; flex-shrink: 0; } .profile-details { flex: 1; min-width: 0; } .profile-name { font-size: 18px; font-weight: 600; margin-bottom: 4px; } .profile-email { font-size: 14px; color: #71717a; word-break: break-all; } /* Organizations */ .org-list { display: flex; flex-direction: column; } .org-item { display: flex; align-items: center; gap: 12px; padding: 12px 20px; border-bottom: 1px solid #27272a; cursor: pointer; transition: background 0.15s ease; } .org-item:last-child { border-bottom: none; } .org-item:hover { background: #27272a; } .org-icon { width: 40px; height: 40px; border-radius: 10px; background: #27272a; display: flex; align-items: center; justify-content: center; flex-shrink: 0; } .org-icon dees-icon { opacity: 0.7; } .org-info { flex: 1; min-width: 0; } .org-name { font-size: 14px; font-weight: 600; margin-bottom: 2px; } .org-role { font-size: 12px; color: #71717a; } .role-badge { padding: 4px 10px; border-radius: 9999px; font-size: 11px; font-weight: 500; background: rgba(59, 130, 246, 0.1); color: #3b82f6; } .role-badge.admin { background: rgba(245, 158, 11, 0.1); color: #f59e0b; } .role-badge.owner { background: rgba(139, 92, 246, 0.1); color: #8b5cf6; } /* Sessions */ .session-list { display: flex; flex-direction: column; } .session-item { display: flex; align-items: center; gap: 12px; padding: 12px 20px; border-bottom: 1px solid #27272a; } .session-item:last-child { border-bottom: none; } .session-icon { width: 40px; height: 40px; border-radius: 10px; background: #27272a; display: flex; align-items: center; justify-content: center; flex-shrink: 0; } .session-icon dees-icon { opacity: 0.7; } .session-icon.current { background: rgba(34, 197, 94, 0.1); } .session-icon.current dees-icon { color: #22c55e; opacity: 1; } .session-info { flex: 1; min-width: 0; } .session-device { font-size: 14px; font-weight: 500; margin-bottom: 2px; display: flex; align-items: center; gap: 8px; } .current-badge { padding: 2px 8px; border-radius: 9999px; font-size: 10px; font-weight: 500; background: rgba(34, 197, 94, 0.1); color: #22c55e; } .session-details { font-size: 12px; color: #71717a; } .session-actions { flex-shrink: 0; } .revoke-btn { padding: 6px 12px; border-radius: 6px; font-size: 12px; font-weight: 500; border: 1px solid #27272a; background: transparent; color: #fafafa; cursor: pointer; transition: all 0.15s ease; } .revoke-btn:hover { background: rgba(239, 68, 68, 0.1); border-color: #ef4444; color: #ef4444; } /* Activity */ .activity-list { display: flex; flex-direction: column; } .activity-item { display: flex; align-items: flex-start; gap: 12px; padding: 12px 20px; border-bottom: 1px solid #27272a; } .activity-item:last-child { border-bottom: none; } .activity-icon { width: 32px; height: 32px; border-radius: 8px; background: #27272a; display: flex; align-items: center; justify-content: center; flex-shrink: 0; } .activity-icon dees-icon { font-size: 14px; opacity: 0.7; } .activity-icon.login { background: rgba(34, 197, 94, 0.1); } .activity-icon.login dees-icon { color: #22c55e; opacity: 1; } .activity-icon.logout { background: rgba(239, 68, 68, 0.1); } .activity-icon.logout dees-icon { color: #ef4444; opacity: 1; } .activity-info { flex: 1; min-width: 0; } .activity-description { font-size: 14px; margin-bottom: 2px; } .activity-time { font-size: 12px; color: #71717a; } /* Empty states */ .empty-state { text-align: center; padding: 32px 20px; color: #71717a; } .empty-state dees-icon { font-size: 32px; opacity: 0.5; margin-bottom: 12px; } .empty-state p { margin: 0; font-size: 14px; } /* Loading state */ .loading { display: flex; align-items: center; justify-content: center; padding: 48px; color: #71717a; } /* Create org button */ .create-org-btn { display: flex; align-items: center; gap: 6px; padding: 6px 12px; border-radius: 6px; font-size: 12px; font-weight: 500; border: 1px solid #27272a; background: transparent; color: #fafafa; cursor: pointer; transition: all 0.15s ease; } .create-org-btn:hover { background: #27272a; } .create-org-btn dees-icon { font-size: 14px; } `, ]; public render(): TemplateResult { if (this.loading) { return html`
Loading your account...
`; } const userInitial = this.user?.data?.username?.charAt(0).toUpperCase() || this.user?.data?.email?.charAt(0).toUpperCase() || '?'; return html`

Account Overview

Manage your profile, organizations, and security settings

Profile
${userInitial}
${this.user?.data?.username || 'Unknown User'}
${this.user?.data?.email || 'No email'}
Organizations
${this.renderOrganizations()}
Active Sessions
${this.renderSessions()}
Recent Activity
${this.renderActivity()}
`; } private renderOrganizations(): TemplateResult { if (this.organizations.length === 0) { return html`

You're not a member of any organizations yet.

`; } return html`
${this.organizations.map((org) => { const roleObj = this.roles.find(r => r.data.organizationId === org.id); const roleName = roleObj?.data.roles?.[0] || 'member'; const roleClass = roleName === 'owner' ? 'owner' : roleName === 'admin' ? 'admin' : ''; const roleDisplay = roleName.charAt(0).toUpperCase() + roleName.slice(1); return html`
this.handleSelectOrg(org)}>
${org.data.name}
${org.data.slug}
${roleDisplay}
`; })}
`; } private renderSessions(): TemplateResult { if (this.sessions.length === 0) { return html`

No active sessions found.

`; } return html`
${this.sessions.map((session) => html`
${session.deviceName || 'Unknown Device'} ${session.isCurrent ? html`Current` : ''}
${session.browser} · ${session.os} · Last active ${this.formatTimeAgo(session.lastActive)}
${!session.isCurrent ? html`
` : ''}
`)}
`; } private renderActivity(): TemplateResult { if (this.activities.length === 0) { return html`

No recent activity.

`; } return html`
${this.activities.slice(0, 5).map((activity) => html`
${activity.data.metadata.description}
${this.formatTimeAgo(activity.data.timestamp)}
`)}
`; } private getDeviceIcon(os: string): string { const osLower = os?.toLowerCase() || ''; if (osLower.includes('mac') || osLower.includes('ios')) { return 'lucide:laptop'; } else if (osLower.includes('android')) { return 'lucide:smartphone'; } else if (osLower.includes('windows')) { return 'lucide:monitor'; } else if (osLower.includes('linux')) { return 'lucide:terminal'; } return 'lucide:monitor'; } private getActivityIcon(action: string): string { switch (action) { case 'login': return 'lucide:log-in'; case 'logout': return 'lucide:log-out'; case 'session_created': return 'lucide:key'; case 'session_revoked': return 'lucide:shield-off'; case 'org_created': return 'lucide:building-2'; case 'org_joined': return 'lucide:user-plus'; case 'org_left': return 'lucide:user-minus'; case 'role_changed': return 'lucide:shield'; case 'profile_updated': return 'lucide:user-cog'; case 'app_connected': return 'lucide:plug'; case 'app_disconnected': return 'lucide:unplug'; default: return 'lucide:activity'; } } private getActivityIconClass(action: string): string { if (action === 'login' || action === 'session_created' || action === 'org_joined' || action === 'app_connected') { return 'login'; } if (action === 'logout' || action === 'session_revoked' || action === 'org_left' || action === 'app_disconnected') { return 'logout'; } return ''; } private formatTimeAgo(timestamp: number): string { const now = Date.now(); const diff = now - timestamp; const minutes = Math.floor(diff / 60000); const hours = Math.floor(diff / 3600000); const days = Math.floor(diff / 86400000); if (minutes < 1) return 'Just now'; if (minutes < 60) return `${minutes}m ago`; if (hours < 24) return `${hours}h ago`; if (days < 7) return `${days}d ago`; return new Date(timestamp).toLocaleDateString(); } public async firstUpdated() { await this.loadDashboardData(); } private async loadDashboardData() { this.loading = true; try { const idpState = await IdpState.getSingletonInstance(); // Load organizations and roles from account state await accountStateModule.accountState.dispatchAction(accountStateModule.getOrganizationsAction, null); const state = accountStateModule.accountState.getState(); this.organizations = state.organizations; this.roles = state.roles; this.user = state.user; // Load sessions await this.loadSessions(); // Load activity await this.loadActivity(); } catch (error) { console.error('Error loading dashboard data:', error); } finally { this.loading = false; } } private async loadSessions() { try { const idpState = await IdpState.getSingletonInstance(); const jwt = await idpState.idpClient.getJwt(); const typedRequest = idpState.idpClient.typedsocket.createTypedRequest( 'getUserSessions' ); const response = await typedRequest.fire({ jwt }); this.sessions = response?.sessions ?? []; } catch (error) { console.error('Error loading sessions:', error); this.sessions = []; } } private async loadActivity() { try { const idpState = await IdpState.getSingletonInstance(); const jwt = await idpState.idpClient.getJwt(); const typedRequest = idpState.idpClient.typedsocket.createTypedRequest( 'getUserActivity' ); const response = await typedRequest.fire({ jwt, limit: 10 }); this.activities = response?.activities ?? []; } catch (error) { console.error('Error loading activity:', error); this.activities = []; } } private async handleRevokeSession(sessionId: string) { if (!confirm('Are you sure you want to revoke this session? The device will be logged out.')) { return; } try { const idpState = await IdpState.getSingletonInstance(); const jwt = await idpState.idpClient.getJwt(); const typedRequest = idpState.idpClient.typedsocket.createTypedRequest( 'revokeSession' ); await typedRequest.fire({ jwt, sessionId }); await this.loadSessions(); } catch (error) { console.error('Error revoking session:', error); alert('Failed to revoke session'); } } private handleSelectOrg(org: plugins.idpInterfaces.data.IOrganization) { accountStateModule.accountState.dispatchAction(accountStateModule.setSelectedOrg, org); const parentElement = (this.getRootNode() as any).host; parentElement.subrouter.pushUrl(`/org/${org.data.slug}/billing`); } private handleCreateOrg() { // Dispatch event to open create org modal this.dispatchEvent(new CustomEvent('open-create-org-modal', { bubbles: true, composed: true, })); } }