@julianpedro/plugin-dev-ai-hub 0.1.6

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 (30) hide show
  1. package/dist/api/DevAiHubClient.esm.js +76 -0
  2. package/dist/api/DevAiHubClient.esm.js.map +1 -0
  3. package/dist/components/AssetCard/AssetCard.esm.js +200 -0
  4. package/dist/components/AssetCard/AssetCard.esm.js.map +1 -0
  5. package/dist/components/AssetDetailPanel/AssetDetailPanel.esm.js +327 -0
  6. package/dist/components/AssetDetailPanel/AssetDetailPanel.esm.js.map +1 -0
  7. package/dist/components/AssetFilters/AssetFilters.esm.js +216 -0
  8. package/dist/components/AssetFilters/AssetFilters.esm.js.map +1 -0
  9. package/dist/components/AssetInstallDialog/AssetInstallDialog.esm.js +194 -0
  10. package/dist/components/AssetInstallDialog/AssetInstallDialog.esm.js.map +1 -0
  11. package/dist/components/DevAiHubPage/DevAiHubPage.esm.js +320 -0
  12. package/dist/components/DevAiHubPage/DevAiHubPage.esm.js.map +1 -0
  13. package/dist/components/DevAiHubPage/index.esm.js +6 -0
  14. package/dist/components/DevAiHubPage/index.esm.js.map +1 -0
  15. package/dist/components/McpConfigDialog/McpConfigDialog.esm.js +314 -0
  16. package/dist/components/McpConfigDialog/McpConfigDialog.esm.js.map +1 -0
  17. package/dist/components/ToolIcon/ToolIcon.esm.js +41 -0
  18. package/dist/components/ToolIcon/ToolIcon.esm.js.map +1 -0
  19. package/dist/hooks/index.esm.js +114 -0
  20. package/dist/hooks/index.esm.js.map +1 -0
  21. package/dist/index.d.ts +104 -0
  22. package/dist/index.esm.js +5 -0
  23. package/dist/index.esm.js.map +1 -0
  24. package/dist/plugin.esm.js +40 -0
  25. package/dist/plugin.esm.js.map +1 -0
  26. package/dist/pluginLegacy.esm.js +27 -0
  27. package/dist/pluginLegacy.esm.js.map +1 -0
  28. package/dist/routes.esm.js +6 -0
  29. package/dist/routes.esm.js.map +1 -0
  30. package/package.json +89 -0
@@ -0,0 +1,320 @@
1
+ import { jsxs, jsx } from 'react/jsx-runtime';
2
+ import { useState, useMemo } from 'react';
3
+ import { useSearchParams } from 'react-router-dom';
4
+ import Box from '@mui/material/Box';
5
+ import Button from '@mui/material/Button';
6
+ import Grid from '@mui/material/Grid';
7
+ import Pagination from '@mui/material/Pagination';
8
+ import Skeleton from '@mui/material/Skeleton';
9
+ import Tooltip from '@mui/material/Tooltip';
10
+ import Typography from '@mui/material/Typography';
11
+ import SettingsEthernetIcon from '@mui/icons-material/SettingsEthernet';
12
+ import ArticleIcon from '@mui/icons-material/Article';
13
+ import SmartToyIcon from '@mui/icons-material/SmartToy';
14
+ import BuildIcon from '@mui/icons-material/Build';
15
+ import AccountTreeIcon from '@mui/icons-material/AccountTree';
16
+ import FiberManualRecordIcon from '@mui/icons-material/FiberManualRecord';
17
+ import HubIcon from '@mui/icons-material/Hub';
18
+ import { Page, Header, Content } from '@backstage/core-components';
19
+ import { AssetCard } from '../AssetCard/AssetCard.esm.js';
20
+ import { AssetFilters } from '../AssetFilters/AssetFilters.esm.js';
21
+ import { AssetDetailPanel } from '../AssetDetailPanel/AssetDetailPanel.esm.js';
22
+ import { AssetInstallDialog } from '../AssetInstallDialog/AssetInstallDialog.esm.js';
23
+ import { McpConfigDialog } from '../McpConfigDialog/McpConfigDialog.esm.js';
24
+ import { ToolIcon } from '../ToolIcon/ToolIcon.esm.js';
25
+ import { useStats, useProviders, useAssets } from '../../hooks/index.esm.js';
26
+
27
+ const SUPPORTED_TOOLS = ["claude-code", "github-copilot", "google-gemini", "cursor"];
28
+ const TOOL_LABELS = {
29
+ "all": "Universal",
30
+ "claude-code": "Claude Code",
31
+ "github-copilot": "GitHub Copilot",
32
+ "google-gemini": "Google Gemini",
33
+ "cursor": "Cursor"
34
+ };
35
+ function timeAgo(iso) {
36
+ const diff = Math.floor((Date.now() - new Date(iso).getTime()) / 1e3);
37
+ if (diff < 60) return "just now";
38
+ if (diff < 3600) return `${Math.floor(diff / 60)}m ago`;
39
+ if (diff < 86400) return `${Math.floor(diff / 3600)}h ago`;
40
+ return `${Math.floor(diff / 86400)}d ago`;
41
+ }
42
+ const DEFAULT_FILTERS = {
43
+ types: [],
44
+ tools: [],
45
+ search: "",
46
+ tags: [],
47
+ providerId: void 0
48
+ };
49
+ const PAGE_SIZE = 24;
50
+ const STATS_CONFIG = [
51
+ {
52
+ key: "instruction",
53
+ label: "Instructions",
54
+ Icon: ArticleIcon,
55
+ gradient: "linear-gradient(135deg, #2563EB 0%, #1D4ED8 100%)",
56
+ shadow: "#2563EB40"
57
+ },
58
+ {
59
+ key: "agent",
60
+ label: "Agents",
61
+ Icon: SmartToyIcon,
62
+ gradient: "linear-gradient(135deg, #7C3AED 0%, #6D28D9 100%)",
63
+ shadow: "#7C3AED40"
64
+ },
65
+ {
66
+ key: "skill",
67
+ label: "Skills",
68
+ Icon: BuildIcon,
69
+ gradient: "linear-gradient(135deg, #059669 0%, #047857 100%)",
70
+ shadow: "#05966940"
71
+ },
72
+ {
73
+ key: "workflow",
74
+ label: "Workflows",
75
+ Icon: AccountTreeIcon,
76
+ gradient: "linear-gradient(135deg, #D97706 0%, #B45309 100%)",
77
+ shadow: "#D9770640"
78
+ }
79
+ ];
80
+ function DevAiHubPage() {
81
+ const [filters, setFilters] = useState(DEFAULT_FILTERS);
82
+ const [page, setPage] = useState(1);
83
+ const [mcpDialogOpen, setMcpDialogOpen] = useState(false);
84
+ const [searchParams, setSearchParams] = useSearchParams();
85
+ const selectedAssetId = searchParams.get("assetId");
86
+ const installAssetId = searchParams.get("installId");
87
+ const handleViewAsset = (id) => setSearchParams((p) => {
88
+ const n = new URLSearchParams(p);
89
+ n.set("assetId", id);
90
+ return n;
91
+ });
92
+ const handleCloseDetail = () => setSearchParams((p) => {
93
+ const n = new URLSearchParams(p);
94
+ n.delete("assetId");
95
+ return n;
96
+ });
97
+ const handleInstallAsset = (id) => setSearchParams((p) => {
98
+ const n = new URLSearchParams(p);
99
+ n.set("installId", id);
100
+ return n;
101
+ });
102
+ const handleCloseInstall = () => setSearchParams((p) => {
103
+ const n = new URLSearchParams(p);
104
+ n.delete("installId");
105
+ return n;
106
+ });
107
+ const { stats } = useStats();
108
+ const { providers } = useProviders();
109
+ const apiFilter = useMemo(
110
+ () => ({
111
+ type: filters.types.length === 1 ? filters.types[0] : void 0,
112
+ tool: filters.tools.length === 1 ? filters.tools[0] : void 0,
113
+ search: filters.search || void 0,
114
+ tags: filters.tags.length > 0 ? filters.tags : void 0,
115
+ providerId: filters.providerId || void 0,
116
+ page,
117
+ pageSize: PAGE_SIZE
118
+ }),
119
+ [filters, page]
120
+ );
121
+ const { result, loading } = useAssets(apiFilter);
122
+ const handleFiltersChange = (next) => {
123
+ setFilters(next);
124
+ setPage(1);
125
+ };
126
+ const totalPages = result ? Math.ceil(result.totalCount / PAGE_SIZE) : 0;
127
+ const availableTags = useMemo(() => {
128
+ if (!result) return [];
129
+ const tagSet = /* @__PURE__ */ new Set();
130
+ result.items.forEach((a) => a.tags.forEach((t) => tagSet.add(t)));
131
+ return Array.from(tagSet).sort();
132
+ }, [result]);
133
+ return /* @__PURE__ */ jsxs(Page, { themeId: "tool", children: [
134
+ /* @__PURE__ */ jsx(
135
+ Header,
136
+ {
137
+ title: /* @__PURE__ */ jsxs(Box, { sx: { display: "flex", alignItems: "center", gap: 2 }, children: [
138
+ /* @__PURE__ */ jsx(
139
+ Box,
140
+ {
141
+ sx: {
142
+ display: "flex",
143
+ alignItems: "center",
144
+ justifyContent: "center",
145
+ width: 44,
146
+ height: 44,
147
+ borderRadius: 2.5,
148
+ background: "rgba(255,255,255,0.15)",
149
+ backdropFilter: "blur(8px)",
150
+ border: "1px solid rgba(255,255,255,0.3)",
151
+ boxShadow: "0 2px 12px rgba(0,0,0,0.15)",
152
+ flexShrink: 0
153
+ },
154
+ children: /* @__PURE__ */ jsx(HubIcon, { sx: { fontSize: "1.5rem", color: "#fff" } })
155
+ }
156
+ ),
157
+ /* @__PURE__ */ jsxs(Box, { sx: { display: "flex", flexDirection: "column", gap: 0.3 }, children: [
158
+ /* @__PURE__ */ jsx(Box, { sx: { fontSize: "1.75rem", fontWeight: 700, lineHeight: 1, color: "#fff" }, children: "Dev AI Hub" }),
159
+ /* @__PURE__ */ jsx(Box, { sx: { fontSize: "0.85rem", fontWeight: 400, color: "rgba(255,255,255,0.75)", lineHeight: 1 }, children: stats ? `${stats.totalAssets} assets \xB7 ${Object.keys(stats.byProvider).length} provider${Object.keys(stats.byProvider).length !== 1 ? "s" : ""}` : "Centralized AI assets for your team" })
160
+ ] })
161
+ ] }),
162
+ pageTitleOverride: "Dev AI Hub",
163
+ children: /* @__PURE__ */ jsxs(Box, { sx: { display: "flex", alignItems: "center", gap: 2 }, children: [
164
+ /* @__PURE__ */ jsx(Box, { sx: { display: "flex", alignItems: "center", gap: 0.75 }, children: SUPPORTED_TOOLS.map((tool) => /* @__PURE__ */ jsx(Tooltip, { title: TOOL_LABELS[tool], arrow: true, children: /* @__PURE__ */ jsx(
165
+ Box,
166
+ {
167
+ sx: {
168
+ width: 32,
169
+ height: 32,
170
+ borderRadius: "50%",
171
+ backgroundColor: "rgba(255,255,255,0.15)",
172
+ display: "flex",
173
+ alignItems: "center",
174
+ justifyContent: "center",
175
+ backdropFilter: "blur(4px)",
176
+ transition: "background-color 0.15s",
177
+ "&:hover": { backgroundColor: "rgba(255,255,255,0.25)" }
178
+ },
179
+ children: /* @__PURE__ */ jsx(ToolIcon, { tool, branded: false, sx: { fontSize: "1rem", color: "#fff" } })
180
+ }
181
+ ) }, tool)) }),
182
+ stats?.lastSync && /* @__PURE__ */ jsx(Tooltip, { title: `Last sync: ${new Date(stats.lastSync).toLocaleString()}`, arrow: true, children: /* @__PURE__ */ jsxs(Box, { sx: { display: "flex", alignItems: "center", gap: 0.5, cursor: "default" }, children: [
183
+ /* @__PURE__ */ jsx(FiberManualRecordIcon, { sx: { fontSize: "0.6rem", color: "#4ade80" } }),
184
+ /* @__PURE__ */ jsx(Typography, { variant: "caption", sx: { color: "rgba(255,255,255,0.8)", whiteSpace: "nowrap" }, children: timeAgo(stats.lastSync) })
185
+ ] }) }),
186
+ /* @__PURE__ */ jsx(
187
+ Button,
188
+ {
189
+ variant: "contained",
190
+ size: "small",
191
+ startIcon: /* @__PURE__ */ jsx(SettingsEthernetIcon, {}),
192
+ onClick: () => setMcpDialogOpen(true),
193
+ sx: {
194
+ borderRadius: 2,
195
+ fontWeight: 700,
196
+ whiteSpace: "nowrap",
197
+ background: "rgba(255,255,255,0.15)",
198
+ backdropFilter: "blur(4px)",
199
+ border: "1px solid rgba(255,255,255,0.3)",
200
+ color: "#fff",
201
+ boxShadow: "none",
202
+ "&:hover": {
203
+ background: "rgba(255,255,255,0.25)",
204
+ boxShadow: "none"
205
+ }
206
+ },
207
+ children: "Configure MCP"
208
+ }
209
+ )
210
+ ] })
211
+ }
212
+ ),
213
+ /* @__PURE__ */ jsxs(Content, { children: [
214
+ /* @__PURE__ */ jsx(Grid, { container: true, spacing: 2, sx: { mb: 3 }, children: STATS_CONFIG.map(({ key, label, Icon, gradient, shadow }) => /* @__PURE__ */ jsx(Grid, { item: true, xs: 6, sm: 3, children: /* @__PURE__ */ jsx(
215
+ Box,
216
+ {
217
+ onClick: () => handleFiltersChange({ ...filters, types: filters.types[0] === key ? [] : [key] }),
218
+ sx: {
219
+ background: gradient,
220
+ borderRadius: 3,
221
+ p: 2,
222
+ cursor: "pointer",
223
+ boxShadow: filters.types[0] === key ? `0 8px 24px ${shadow}` : `0 2px 8px ${shadow}`,
224
+ transform: filters.types[0] === key ? "translateY(-2px)" : "none",
225
+ transition: "all 0.2s ease",
226
+ outline: filters.types[0] === key ? "2px solid rgba(255,255,255,0.6)" : "none",
227
+ outlineOffset: 2,
228
+ "&:hover": {
229
+ boxShadow: `0 8px 24px ${shadow}`,
230
+ transform: "translateY(-2px)"
231
+ }
232
+ },
233
+ children: /* @__PURE__ */ jsxs(Box, { sx: { display: "flex", justifyContent: "space-between", alignItems: "flex-start" }, children: [
234
+ /* @__PURE__ */ jsxs(Box, { children: [
235
+ /* @__PURE__ */ jsx(Typography, { variant: "h4", fontWeight: 800, sx: { color: "#fff", lineHeight: 1 }, children: stats ? stats.byType[key] ?? 0 : /* @__PURE__ */ jsx(Skeleton, { width: 32, sx: { bgcolor: "rgba(255,255,255,0.3)" } }) }),
236
+ /* @__PURE__ */ jsx(Typography, { variant: "body2", sx: { color: "rgba(255,255,255,0.85)", fontWeight: 500, mt: 0.5 }, children: label })
237
+ ] }),
238
+ /* @__PURE__ */ jsx(
239
+ Box,
240
+ {
241
+ sx: {
242
+ width: 40,
243
+ height: 40,
244
+ borderRadius: 2,
245
+ backgroundColor: "rgba(255,255,255,0.2)",
246
+ display: "flex",
247
+ alignItems: "center",
248
+ justifyContent: "center"
249
+ },
250
+ children: /* @__PURE__ */ jsx(Icon, { sx: { color: "#fff", fontSize: "1.4rem" } })
251
+ }
252
+ )
253
+ ] })
254
+ }
255
+ ) }, key)) }),
256
+ /* @__PURE__ */ jsx(
257
+ AssetFilters,
258
+ {
259
+ value: filters,
260
+ onChange: handleFiltersChange,
261
+ availableTags,
262
+ providers: providers.length > 1 ? providers : void 0
263
+ }
264
+ ),
265
+ result && !loading && /* @__PURE__ */ jsxs(Typography, { variant: "caption", color: "text.secondary", sx: { mb: 2, display: "block" }, children: [
266
+ result.totalCount,
267
+ " asset",
268
+ result.totalCount !== 1 ? "s" : "",
269
+ " found"
270
+ ] }),
271
+ /* @__PURE__ */ jsx(Grid, { container: true, spacing: 1.5, children: loading ? Array.from({ length: 8 }).map((_, i) => /* @__PURE__ */ jsx(Grid, { item: true, xs: 12, sm: 6, md: 4, lg: 3, children: /* @__PURE__ */ jsx(Skeleton, { variant: "rectangular", height: 150, sx: { borderRadius: 2 } }) }, i)) : result?.items.map((asset) => /* @__PURE__ */ jsx(Grid, { item: true, xs: 12, sm: 6, md: 4, lg: 3, children: /* @__PURE__ */ jsx(
272
+ AssetCard,
273
+ {
274
+ asset,
275
+ onView: handleViewAsset,
276
+ onInstall: handleInstallAsset
277
+ }
278
+ ) }, asset.id)) }),
279
+ !loading && result?.items.length === 0 && /* @__PURE__ */ jsxs(Box, { sx: { py: 12, textAlign: "center" }, children: [
280
+ /* @__PURE__ */ jsx(Typography, { variant: "h1", sx: { mb: 2, opacity: 0.15, fontSize: "5rem" }, children: "\u{1F916}" }),
281
+ /* @__PURE__ */ jsx(Typography, { variant: "h6", color: "text.secondary", fontWeight: 600, children: "No assets found" }),
282
+ /* @__PURE__ */ jsx(Typography, { variant: "body2", color: "text.disabled", sx: { mt: 0.5 }, children: "Try adjusting your filters or search terms." })
283
+ ] }),
284
+ totalPages > 1 && /* @__PURE__ */ jsx(Box, { sx: { display: "flex", justifyContent: "center", mt: 4 }, children: /* @__PURE__ */ jsx(
285
+ Pagination,
286
+ {
287
+ count: totalPages,
288
+ page,
289
+ onChange: (_, p) => setPage(p),
290
+ color: "primary",
291
+ shape: "rounded"
292
+ }
293
+ ) })
294
+ ] }),
295
+ /* @__PURE__ */ jsx(
296
+ AssetDetailPanel,
297
+ {
298
+ assetId: selectedAssetId,
299
+ onClose: handleCloseDetail
300
+ }
301
+ ),
302
+ /* @__PURE__ */ jsx(
303
+ AssetInstallDialog,
304
+ {
305
+ assetId: installAssetId,
306
+ onClose: handleCloseInstall
307
+ }
308
+ ),
309
+ /* @__PURE__ */ jsx(
310
+ McpConfigDialog,
311
+ {
312
+ open: mcpDialogOpen,
313
+ onClose: () => setMcpDialogOpen(false)
314
+ }
315
+ )
316
+ ] });
317
+ }
318
+
319
+ export { DevAiHubPage };
320
+ //# sourceMappingURL=DevAiHubPage.esm.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"DevAiHubPage.esm.js","sources":["../../../src/components/DevAiHubPage/DevAiHubPage.tsx"],"sourcesContent":["import { useState, useMemo } from 'react';\nimport { useSearchParams } from 'react-router-dom';\nimport Box from '@mui/material/Box';\nimport Button from '@mui/material/Button';\nimport Grid from '@mui/material/Grid';\nimport Pagination from '@mui/material/Pagination';\nimport Skeleton from '@mui/material/Skeleton';\nimport Tooltip from '@mui/material/Tooltip';\nimport Typography from '@mui/material/Typography';\nimport SettingsEthernetIcon from '@mui/icons-material/SettingsEthernet';\nimport ArticleIcon from '@mui/icons-material/Article';\nimport SmartToyIcon from '@mui/icons-material/SmartToy';\nimport BuildIcon from '@mui/icons-material/Build';\nimport AccountTreeIcon from '@mui/icons-material/AccountTree';\nimport FiberManualRecordIcon from '@mui/icons-material/FiberManualRecord';\nimport HubIcon from '@mui/icons-material/Hub';\nimport { Content, Header, Page } from '@backstage/core-components';\nimport type { AssetType, AiTool } from '@julianpedro/plugin-dev-ai-hub-common';\nimport { AssetCard } from '../AssetCard';\nimport { AssetFilters } from '../AssetFilters';\nimport type { AssetFiltersValue } from '../AssetFilters';\nimport { AssetDetailPanel } from '../AssetDetailPanel';\nimport { AssetInstallDialog } from '../AssetInstallDialog';\nimport { McpConfigDialog } from '../McpConfigDialog';\nimport { ToolIcon } from '../ToolIcon';\nimport { useAssets, useStats, useProviders } from '../../hooks';\n\nconst SUPPORTED_TOOLS: AiTool[] = ['claude-code', 'github-copilot', 'google-gemini', 'cursor'];\n\nconst TOOL_LABELS: Record<AiTool, string> = {\n 'all': 'Universal',\n 'claude-code': 'Claude Code',\n 'github-copilot': 'GitHub Copilot',\n 'google-gemini': 'Google Gemini',\n 'cursor': 'Cursor',\n};\n\nfunction timeAgo(iso: string): string {\n const diff = Math.floor((Date.now() - new Date(iso).getTime()) / 1000);\n if (diff < 60) return 'just now';\n if (diff < 3600) return `${Math.floor(diff / 60)}m ago`;\n if (diff < 86400) return `${Math.floor(diff / 3600)}h ago`;\n return `${Math.floor(diff / 86400)}d ago`;\n}\n\nconst DEFAULT_FILTERS: AssetFiltersValue = {\n types: [],\n tools: [],\n search: '',\n tags: [],\n providerId: undefined,\n};\n\nconst PAGE_SIZE = 24;\n\nconst STATS_CONFIG = [\n { key: 'instruction' as AssetType, label: 'Instructions', Icon: ArticleIcon,\n gradient: 'linear-gradient(135deg, #2563EB 0%, #1D4ED8 100%)', shadow: '#2563EB40' },\n { key: 'agent' as AssetType, label: 'Agents', Icon: SmartToyIcon,\n gradient: 'linear-gradient(135deg, #7C3AED 0%, #6D28D9 100%)', shadow: '#7C3AED40' },\n { key: 'skill' as AssetType, label: 'Skills', Icon: BuildIcon,\n gradient: 'linear-gradient(135deg, #059669 0%, #047857 100%)', shadow: '#05966940' },\n { key: 'workflow' as AssetType, label: 'Workflows', Icon: AccountTreeIcon,\n gradient: 'linear-gradient(135deg, #D97706 0%, #B45309 100%)', shadow: '#D9770640' },\n];\n\nexport function DevAiHubPage() {\n const [filters, setFilters] = useState<AssetFiltersValue>(DEFAULT_FILTERS);\n const [page, setPage] = useState(1);\n const [mcpDialogOpen, setMcpDialogOpen] = useState(false);\n\n const [searchParams, setSearchParams] = useSearchParams();\n const selectedAssetId = searchParams.get('assetId');\n const installAssetId = searchParams.get('installId');\n\n const handleViewAsset = (id: string) =>\n setSearchParams(p => { const n = new URLSearchParams(p); n.set('assetId', id); return n; });\n\n const handleCloseDetail = () =>\n setSearchParams(p => { const n = new URLSearchParams(p); n.delete('assetId'); return n; });\n\n const handleInstallAsset = (id: string) =>\n setSearchParams(p => { const n = new URLSearchParams(p); n.set('installId', id); return n; });\n\n const handleCloseInstall = () =>\n setSearchParams(p => { const n = new URLSearchParams(p); n.delete('installId'); return n; });\n\n const { stats } = useStats();\n const { providers } = useProviders();\n\n const apiFilter = useMemo(\n () => ({\n type: filters.types.length === 1 ? (filters.types[0] as AssetType) : undefined,\n tool: filters.tools.length === 1 ? (filters.tools[0] as AiTool) : undefined,\n search: filters.search || undefined,\n tags: filters.tags.length > 0 ? filters.tags : undefined,\n providerId: filters.providerId || undefined,\n page,\n pageSize: PAGE_SIZE,\n }),\n [filters, page],\n );\n\n const { result, loading } = useAssets(apiFilter);\n\n const handleFiltersChange = (next: AssetFiltersValue) => {\n setFilters(next);\n setPage(1);\n };\n\n const totalPages = result ? Math.ceil(result.totalCount / PAGE_SIZE) : 0;\n\n const availableTags = useMemo(() => {\n if (!result) return [];\n const tagSet = new Set<string>();\n result.items.forEach(a => a.tags.forEach(t => tagSet.add(t)));\n return Array.from(tagSet).sort();\n }, [result]);\n\n return (\n <Page themeId=\"tool\">\n <Header\n title={\n <Box sx={{ display: 'flex', alignItems: 'center', gap: 2 }}>\n <Box\n sx={{\n display: 'flex',\n alignItems: 'center',\n justifyContent: 'center',\n width: 44,\n height: 44,\n borderRadius: 2.5,\n background: 'rgba(255,255,255,0.15)',\n backdropFilter: 'blur(8px)',\n border: '1px solid rgba(255,255,255,0.3)',\n boxShadow: '0 2px 12px rgba(0,0,0,0.15)',\n flexShrink: 0,\n }}\n >\n <HubIcon sx={{ fontSize: '1.5rem', color: '#fff' }} />\n </Box>\n <Box sx={{ display: 'flex', flexDirection: 'column', gap: 0.3 }}>\n <Box sx={{ fontSize: '1.75rem', fontWeight: 700, lineHeight: 1, color: '#fff' }}>\n Dev AI Hub\n </Box>\n <Box sx={{ fontSize: '0.85rem', fontWeight: 400, color: 'rgba(255,255,255,0.75)', lineHeight: 1 }}>\n {stats\n ? `${stats.totalAssets} assets · ${Object.keys(stats.byProvider).length} provider${Object.keys(stats.byProvider).length !== 1 ? 's' : ''}`\n : 'Centralized AI assets for your team'}\n </Box>\n </Box>\n </Box>\n }\n pageTitleOverride=\"Dev AI Hub\"\n >\n <Box sx={{ display: 'flex', alignItems: 'center', gap: 2 }}>\n {/* Supported tools */}\n <Box sx={{ display: 'flex', alignItems: 'center', gap: 0.75 }}>\n {SUPPORTED_TOOLS.map(tool => (\n <Tooltip key={tool} title={TOOL_LABELS[tool]} arrow>\n <Box\n sx={{\n width: 32,\n height: 32,\n borderRadius: '50%',\n backgroundColor: 'rgba(255,255,255,0.15)',\n display: 'flex',\n alignItems: 'center',\n justifyContent: 'center',\n backdropFilter: 'blur(4px)',\n transition: 'background-color 0.15s',\n '&:hover': { backgroundColor: 'rgba(255,255,255,0.25)' },\n }}\n >\n <ToolIcon tool={tool} branded={false} sx={{ fontSize: '1rem', color: '#fff' }} />\n </Box>\n </Tooltip>\n ))}\n </Box>\n\n {/* Last sync status */}\n {stats?.lastSync && (\n <Tooltip title={`Last sync: ${new Date(stats.lastSync).toLocaleString()}`} arrow>\n <Box sx={{ display: 'flex', alignItems: 'center', gap: 0.5, cursor: 'default' }}>\n <FiberManualRecordIcon sx={{ fontSize: '0.6rem', color: '#4ade80' }} />\n <Typography variant=\"caption\" sx={{ color: 'rgba(255,255,255,0.8)', whiteSpace: 'nowrap' }}>\n {timeAgo(stats.lastSync)}\n </Typography>\n </Box>\n </Tooltip>\n )}\n\n {/* Configure MCP button */}\n <Button\n variant=\"contained\"\n size=\"small\"\n startIcon={<SettingsEthernetIcon />}\n onClick={() => setMcpDialogOpen(true)}\n sx={{\n borderRadius: 2,\n fontWeight: 700,\n whiteSpace: 'nowrap',\n background: 'rgba(255,255,255,0.15)',\n backdropFilter: 'blur(4px)',\n border: '1px solid rgba(255,255,255,0.3)',\n color: '#fff',\n boxShadow: 'none',\n '&:hover': {\n background: 'rgba(255,255,255,0.25)',\n boxShadow: 'none',\n },\n }}\n >\n Configure MCP\n </Button>\n </Box>\n </Header>\n\n <Content>\n\n {/* Stats row */}\n <Grid container spacing={2} sx={{ mb: 3 }}>\n {STATS_CONFIG.map(({ key, label, Icon, gradient, shadow }) => (\n <Grid item xs={6} sm={3} key={key}>\n <Box\n onClick={() => handleFiltersChange({ ...filters, types: filters.types[0] === key ? [] : [key] })}\n sx={{\n background: gradient,\n borderRadius: 3,\n p: 2,\n cursor: 'pointer',\n boxShadow: filters.types[0] === key ? `0 8px 24px ${shadow}` : `0 2px 8px ${shadow}`,\n transform: filters.types[0] === key ? 'translateY(-2px)' : 'none',\n transition: 'all 0.2s ease',\n outline: filters.types[0] === key ? '2px solid rgba(255,255,255,0.6)' : 'none',\n outlineOffset: 2,\n '&:hover': {\n boxShadow: `0 8px 24px ${shadow}`,\n transform: 'translateY(-2px)',\n },\n }}\n >\n <Box sx={{ display: 'flex', justifyContent: 'space-between', alignItems: 'flex-start' }}>\n <Box>\n <Typography variant=\"h4\" fontWeight={800} sx={{ color: '#fff', lineHeight: 1 }}>\n {stats ? (stats.byType[key] ?? 0) : <Skeleton width={32} sx={{ bgcolor: 'rgba(255,255,255,0.3)' }} />}\n </Typography>\n <Typography variant=\"body2\" sx={{ color: 'rgba(255,255,255,0.85)', fontWeight: 500, mt: 0.5 }}>\n {label}\n </Typography>\n </Box>\n <Box\n sx={{\n width: 40,\n height: 40,\n borderRadius: 2,\n backgroundColor: 'rgba(255,255,255,0.2)',\n display: 'flex',\n alignItems: 'center',\n justifyContent: 'center',\n }}\n >\n <Icon sx={{ color: '#fff', fontSize: '1.4rem' }} />\n </Box>\n </Box>\n </Box>\n </Grid>\n ))}\n </Grid>\n\n {/* Filters */}\n <AssetFilters\n value={filters}\n onChange={handleFiltersChange}\n availableTags={availableTags}\n providers={providers.length > 1 ? providers : undefined}\n />\n\n {/* Results summary */}\n {result && !loading && (\n <Typography variant=\"caption\" color=\"text.secondary\" sx={{ mb: 2, display: 'block' }}>\n {result.totalCount} asset{result.totalCount !== 1 ? 's' : ''} found\n </Typography>\n )}\n\n {/* Asset grid — 4 columns on large screens */}\n <Grid container spacing={1.5}>\n {loading\n ? Array.from({ length: 8 }).map((_, i) => (\n <Grid item xs={12} sm={6} md={4} lg={3} key={i}>\n <Skeleton variant=\"rectangular\" height={150} sx={{ borderRadius: 2 }} />\n </Grid>\n ))\n : result?.items.map(asset => (\n <Grid item xs={12} sm={6} md={4} lg={3} key={asset.id}>\n <AssetCard\n asset={asset}\n onView={handleViewAsset}\n onInstall={handleInstallAsset}\n />\n </Grid>\n ))}\n </Grid>\n\n {/* Empty state */}\n {!loading && result?.items.length === 0 && (\n <Box sx={{ py: 12, textAlign: 'center' }}>\n <Typography variant=\"h1\" sx={{ mb: 2, opacity: 0.15, fontSize: '5rem' }}>🤖</Typography>\n <Typography variant=\"h6\" color=\"text.secondary\" fontWeight={600}>No assets found</Typography>\n <Typography variant=\"body2\" color=\"text.disabled\" sx={{ mt: 0.5 }}>\n Try adjusting your filters or search terms.\n </Typography>\n </Box>\n )}\n\n {/* Pagination */}\n {totalPages > 1 && (\n <Box sx={{ display: 'flex', justifyContent: 'center', mt: 4 }}>\n <Pagination\n count={totalPages}\n page={page}\n onChange={(_, p) => setPage(p)}\n color=\"primary\"\n shape=\"rounded\"\n />\n </Box>\n )}\n </Content>\n\n <AssetDetailPanel\n assetId={selectedAssetId}\n onClose={handleCloseDetail}\n />\n\n <AssetInstallDialog\n assetId={installAssetId}\n onClose={handleCloseInstall}\n />\n\n <McpConfigDialog\n open={mcpDialogOpen}\n onClose={() => setMcpDialogOpen(false)}\n />\n </Page>\n );\n}\n"],"names":[],"mappings":"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"}
@@ -0,0 +1,6 @@
1
+ import { DevAiHubPage } from './DevAiHubPage.esm.js';
2
+
3
+
4
+
5
+ export { DevAiHubPage };
6
+ //# sourceMappingURL=index.esm.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.esm.js","sources":[],"sourcesContent":[],"names":[],"mappings":";;;;"}