432 lines
11 KiB
Svelte
432 lines
11 KiB
Svelte
<script>
|
|
import Icon from '../ui/Icon.svelte';
|
|
export let p = {};
|
|
export let capabilities = [];
|
|
export let permissions = [];
|
|
export let contributions = {};
|
|
export let vaultOpen = false;
|
|
export let settingsPanels = [];
|
|
export let onEnable = () => {};
|
|
export let onDisable = () => {};
|
|
|
|
$: m = p.manifest || {};
|
|
$: pluginId = m.id || 'unknown';
|
|
$: hasSettingsPanel = settingsPanels.length > 0;
|
|
$: hasUIPermission = (m.permissions || []).includes('ui.register');
|
|
$: hasStoragePermission = (m.permissions || []).includes('storage.namespace');
|
|
$: hasCommandsPermission = (m.permissions || []).includes('commands.register');
|
|
|
|
$: statusColor = ({
|
|
loaded: '#4ecca3',
|
|
degraded: '#ffc857',
|
|
disabled: '#a0a0b8',
|
|
failed: '#e94560',
|
|
incompatible: '#e94560',
|
|
'missing-required-capability': '#e94560',
|
|
loading: '#ffc857',
|
|
discovered: '#a0a0b8',
|
|
}[p.status] || '#a0a0b8');
|
|
|
|
$: contribCounts = {
|
|
views: (contributions.views || []).filter(v => v.pluginId === pluginId).length,
|
|
commands: (contributions.commands || []).filter(c => c.pluginId === pluginId).length,
|
|
sidebar: (contributions.sidebarItems || []).filter(s => s.pluginId === pluginId).length,
|
|
statusbar: (contributions.statusBarItems || []).filter(s => s.pluginId === pluginId).length,
|
|
};
|
|
|
|
$: contribSummary = (() => {
|
|
const parts = [];
|
|
if (contribCounts.views > 0) parts.push(contribCounts.views + ' view' + (contribCounts.views !== 1 ? 's' : ''));
|
|
if (contribCounts.commands > 0) parts.push(contribCounts.commands + ' command' + (contribCounts.commands !== 1 ? 's' : ''));
|
|
if (contribCounts.sidebar > 0) parts.push(contribCounts.sidebar + ' sidebar' + (contribCounts.sidebar !== 1 ? 's' : ''));
|
|
if (contribCounts.statusbar > 0) parts.push(contribCounts.statusbar + ' statusbar' + (contribCounts.statusbar !== 1 ? 's' : ''));
|
|
return parts.length > 0 ? parts.join(', ') : 'none';
|
|
})();
|
|
|
|
$: dangerousPermissions = (m.permissions || []).filter(name => {
|
|
let perm = permissions.find(p => p.name === name);
|
|
return perm && perm.dangerous;
|
|
});
|
|
|
|
$: missingRequired = (m.requires || []).filter(req =>
|
|
!capabilities.some(c => c.name === req)
|
|
);
|
|
|
|
$: availableOptional = (m.optionalRequires || []).filter(opt =>
|
|
capabilities.some(c => c.name === opt)
|
|
);
|
|
|
|
$: missingOptional = (m.optionalRequires || []).filter(opt =>
|
|
!capabilities.some(c => c.name === opt)
|
|
);
|
|
|
|
$: isDisabled = p.status === 'disabled' || !p.enabled;
|
|
$: canToggle = p.status !== 'failed' && p.status !== 'incompatible' && p.status !== 'missing-required-capability' && p.status !== 'discovered';
|
|
</script>
|
|
|
|
<div class="plugin-card" class:disabled={isDisabled} class:failed={p.status === 'failed'}>
|
|
<div class="card-header">
|
|
<div class="plugin-id">
|
|
<span class="status-dot" style="background: {statusColor}"></span>
|
|
<strong>{pluginId}</strong>
|
|
<span class="version">v{m.version || '?'}</span>
|
|
</div>
|
|
<span class="status-badge" style="color: {statusColor}">{p.status}</span>
|
|
</div>
|
|
|
|
{#if p.status === 'degraded'}
|
|
<p class="degraded-text">Plugin is usable, but some optional capabilities are unavailable.</p>
|
|
{/if}
|
|
|
|
{#if m.description}
|
|
<p class="description">{m.description}</p>
|
|
{/if}
|
|
|
|
<div class="card-meta">
|
|
<div class="meta-row">
|
|
<span class="label">Name:</span>
|
|
<span>{m.name || '-'}</span>
|
|
</div>
|
|
<div class="meta-row">
|
|
<span class="label">API Version:</span>
|
|
<span>{m.apiVersion || '-'}</span>
|
|
</div>
|
|
<div class="meta-row">
|
|
<span class="label">Source:</span>
|
|
<span>{m.source || 'unknown'}</span>
|
|
</div>
|
|
<div class="meta-row">
|
|
<span class="label">Root:</span>
|
|
<span class="path">{p.rootPath || '-'}</span>
|
|
</div>
|
|
<div class="meta-row">
|
|
<span class="label">Contributions:</span>
|
|
<span>{contribSummary}</span>
|
|
</div>
|
|
</div>
|
|
|
|
<!-- Capabilities -->
|
|
<div class="section">
|
|
<span class="section-title">Provides</span>
|
|
<div class="tags">
|
|
{#each m.provides || [] as cap}
|
|
<span class="tag provides">{cap}</span>
|
|
{/each}
|
|
</div>
|
|
</div>
|
|
|
|
{#if m.requires && m.requires.length > 0}
|
|
<div class="section">
|
|
<span class="section-title">Requires</span>
|
|
<div class="tags">
|
|
{#each m.requires as req}
|
|
{@const found = capabilities.some(c => c.name === req)}
|
|
<span class="tag" class:required-ok={found} class:required-missing={!found}>
|
|
{req}
|
|
{#if found}<span class="check">✓</span>{/if}
|
|
</span>
|
|
{/each}
|
|
</div>
|
|
{#if missingRequired.length > 0}
|
|
<p class="warning"><Icon name="warning" size={12} /> Missing required capabilities: {missingRequired.join(', ')}</p>
|
|
{/if}
|
|
</div>
|
|
{/if}
|
|
|
|
{#if m.optionalRequires && m.optionalRequires.length > 0}
|
|
<div class="section">
|
|
<span class="section-title">Optional Requires</span>
|
|
<div class="tags">
|
|
{#each m.optionalRequires as opt}
|
|
{@const found = capabilities.some(c => c.name === opt)}
|
|
<span class="tag" class:optional-ok={found} class:optional-missing={!found}>
|
|
{opt}
|
|
{#if found}<span class="check">✓</span>{/if}
|
|
</span>
|
|
{/each}
|
|
</div>
|
|
{#if missingOptional.length > 0}
|
|
<p class="info"><Icon name="warning" size={12} /> Optional capabilities not available — plugin running in degraded mode</p>
|
|
{/if}
|
|
</div>
|
|
{/if}
|
|
|
|
<!-- Permissions -->
|
|
{#if m.permissions && m.permissions.length > 0}
|
|
<div class="section">
|
|
<span class="section-title">Permissions</span>
|
|
<div class="tags">
|
|
{#each m.permissions as perm}
|
|
{@const isDangerous = dangerousPermissions.includes(perm)}
|
|
<span class="tag" class:dangerous={isDangerous}>
|
|
{perm}
|
|
{#if isDangerous}<Icon name="warning" size={12} className="danger-icon" />{/if}
|
|
</span>
|
|
{/each}
|
|
</div>
|
|
</div>
|
|
{/if}
|
|
|
|
<!-- Error -->
|
|
{#if p.error}
|
|
<div class="error-box">{p.error}</div>
|
|
{/if}
|
|
|
|
<!-- Actions -->
|
|
<div class="card-actions">
|
|
{#if hasSettingsPanel}
|
|
<button class="btn-settings" on:click={() => window.dispatchEvent(new CustomEvent('verstak:open-settings', { detail: { pluginId: m.id, panelId: settingsPanels[0]?.id } }))} type="button" disabled={isDisabled || p.status === 'failed'}>
|
|
<Icon name="gear" size={14} /> Settings
|
|
</button>
|
|
{/if}
|
|
{#if vaultOpen && canToggle}
|
|
{#if isDisabled}
|
|
<button class="btn-enable" on:click={() => onEnable(m.id)} type="button">
|
|
▶ Enable
|
|
</button>
|
|
{:else}
|
|
<button class="btn-disable" on:click={() => onDisable(m.id)} type="button">
|
|
⏸ Disable
|
|
</button>
|
|
{/if}
|
|
{/if}
|
|
{#if !vaultOpen && canToggle}
|
|
<span class="vault-hint">Open a vault to manage plugin state</span>
|
|
{/if}
|
|
</div>
|
|
|
|
<!-- Permission warnings -->
|
|
{#if !hasUIPermission && (m.contributes && (m.contributes.views || m.contributes.sidebarItems || m.contributes.settingsPanels).length > 0)}
|
|
<p class="warning"><Icon name="warning" size={12} /> Plugin has UI contributions but lacks ui.register permission</p>
|
|
{/if}
|
|
</div>
|
|
|
|
<style>
|
|
.plugin-card {
|
|
background: #16213e;
|
|
border: 1px solid #0f3460;
|
|
border-radius: 8px;
|
|
padding: 1rem;
|
|
}
|
|
|
|
.plugin-card.disabled {
|
|
opacity: 0.6;
|
|
}
|
|
|
|
.plugin-card.failed {
|
|
border-color: #e94560;
|
|
}
|
|
|
|
.card-header {
|
|
display: flex;
|
|
align-items: center;
|
|
justify-content: space-between;
|
|
margin-bottom: 0.5rem;
|
|
}
|
|
|
|
.plugin-id {
|
|
display: flex;
|
|
align-items: center;
|
|
gap: 0.5rem;
|
|
}
|
|
|
|
.status-dot {
|
|
width: 8px;
|
|
height: 8px;
|
|
border-radius: 50%;
|
|
display: inline-block;
|
|
}
|
|
|
|
.version {
|
|
color: #a0a0b8;
|
|
font-size: 0.8rem;
|
|
}
|
|
|
|
.status-badge {
|
|
font-size: 0.75rem;
|
|
text-transform: uppercase;
|
|
letter-spacing: 0.05em;
|
|
font-weight: 600;
|
|
}
|
|
|
|
.description {
|
|
color: #a0a0b8;
|
|
font-size: 0.85rem;
|
|
margin-bottom: 0.75rem;
|
|
}
|
|
|
|
.degraded-text {
|
|
color: #ffc857;
|
|
font-size: 0.8rem;
|
|
margin-bottom: 0.5rem;
|
|
font-style: italic;
|
|
}
|
|
|
|
.card-meta {
|
|
display: grid;
|
|
grid-template-columns: 1fr 1fr;
|
|
gap: 0.3rem;
|
|
margin-bottom: 0.75rem;
|
|
font-size: 0.8rem;
|
|
}
|
|
|
|
.meta-row {
|
|
display: flex;
|
|
gap: 0.5rem;
|
|
}
|
|
|
|
.label {
|
|
color: #a0a0b8;
|
|
min-width: 80px;
|
|
}
|
|
|
|
.path {
|
|
font-family: monospace;
|
|
font-size: 0.75rem;
|
|
color: #a0a0b8;
|
|
}
|
|
|
|
.section {
|
|
margin-bottom: 0.5rem;
|
|
}
|
|
|
|
.section-title {
|
|
display: block;
|
|
font-size: 0.75rem;
|
|
color: #a0a0b8;
|
|
text-transform: uppercase;
|
|
letter-spacing: 0.05em;
|
|
margin-bottom: 0.3rem;
|
|
}
|
|
|
|
.tags {
|
|
display: flex;
|
|
flex-wrap: wrap;
|
|
gap: 0.3rem;
|
|
}
|
|
|
|
.tag {
|
|
background: #0f3460;
|
|
padding: 0.15rem 0.5rem;
|
|
border-radius: 4px;
|
|
font-size: 0.75rem;
|
|
font-family: monospace;
|
|
color: #e0e0e0;
|
|
}
|
|
|
|
.tag.provides {
|
|
background: #1a3a5c;
|
|
border: 1px solid #533483;
|
|
}
|
|
|
|
.tag.required-ok {
|
|
border: 1px solid #4ecca3;
|
|
}
|
|
|
|
.tag.required-missing {
|
|
border: 1px solid #e94560;
|
|
color: #e94560;
|
|
}
|
|
|
|
.tag.optional-ok {
|
|
border: 1px solid #4ecca3;
|
|
}
|
|
|
|
.tag.optional-missing {
|
|
border: 1px solid #ffc857;
|
|
color: #ffc857;
|
|
}
|
|
|
|
.tag.dangerous {
|
|
border: 1px solid #e94560;
|
|
}
|
|
|
|
.check { color: #4ecca3; margin-left: 2px; }
|
|
.danger-icon { color: #e94560; margin-left: 2px; vertical-align: middle; }
|
|
|
|
.info {
|
|
color: #ffc857;
|
|
font-size: 0.8rem;
|
|
margin-top: 0.3rem;
|
|
}
|
|
|
|
.warning {
|
|
color: #ffc857;
|
|
font-size: 0.8rem;
|
|
margin-top: 0.3rem;
|
|
}
|
|
|
|
.error-box {
|
|
background: rgba(233, 69, 96, 0.1);
|
|
border: 1px solid #e94560;
|
|
border-radius: 4px;
|
|
padding: 0.5rem;
|
|
margin-top: 0.5rem;
|
|
font-size: 0.8rem;
|
|
color: #e94560;
|
|
font-family: monospace;
|
|
}
|
|
|
|
.card-actions {
|
|
display: flex;
|
|
align-items: center;
|
|
gap: 0.5rem;
|
|
margin-top: 0.75rem;
|
|
padding-top: 0.5rem;
|
|
border-top: 1px solid #0f3460;
|
|
}
|
|
|
|
.btn-settings {
|
|
display: inline-flex;
|
|
align-items: center;
|
|
gap: 0.3rem;
|
|
background: #0f3460;
|
|
border: 1px solid #1a3a5c;
|
|
color: #e0e0f0;
|
|
padding: 0.3rem 0.75rem;
|
|
border-radius: 4px;
|
|
cursor: pointer;
|
|
font-size: 0.8rem;
|
|
}
|
|
|
|
.btn-settings:hover {
|
|
background: #1a3a5c;
|
|
}
|
|
|
|
.btn-enable {
|
|
background: #4ecca3;
|
|
color: #1a1a2e;
|
|
border: none;
|
|
padding: 0.3rem 0.75rem;
|
|
border-radius: 4px;
|
|
cursor: pointer;
|
|
font-size: 0.8rem;
|
|
font-weight: 600;
|
|
}
|
|
|
|
.btn-enable:hover {
|
|
background: #3dbb92;
|
|
}
|
|
|
|
.btn-disable {
|
|
background: #533483;
|
|
color: #e0e0f0;
|
|
border: none;
|
|
padding: 0.3rem 0.75rem;
|
|
border-radius: 4px;
|
|
cursor: pointer;
|
|
font-size: 0.8rem;
|
|
font-weight: 600;
|
|
}
|
|
|
|
.btn-disable:hover {
|
|
background: #6b44a0;
|
|
}
|
|
|
|
.vault-hint {
|
|
color: #666;
|
|
font-size: 0.75rem;
|
|
font-style: italic;
|
|
}
|
|
</style>
|