@knitli/docs-components 1.3.2 → 1.3.5

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.
@@ -0,0 +1,289 @@
1
+ ---
2
+ // SPDX-FileCopyrightText: 2025 Knitli Inc.
3
+ // SPDX-FileContributor: Adam Poulemanos <adam@knit.li>
4
+ //
5
+ // SPDX-License-Identifier: MIT OR Apache-2.0
6
+ //
7
+ // Product card for the documentation homepage. Renders a single product
8
+ // from the product catalog with status-driven behavior (clickable link
9
+ // vs static card with "coming soon" stamp).
10
+
11
+ import { Image } from "astro:assets";
12
+ // @ts-expect-error: 2604
13
+ import { Icon } from "@astrojs/starlight/components";
14
+ import type { StarlightIcon } from "@astrojs/starlight/types";
15
+ import * as logos from "@knitli/shared-layouts";
16
+ import type { ImageMetadata } from "astro";
17
+ import type { Product } from "../data/products.js";
18
+
19
+ interface Props {
20
+ /** Product data from the catalog */
21
+ product: Product;
22
+ /** Optional Starlight icon (overrides logo) */
23
+ icon?: StarlightIcon;
24
+ }
25
+
26
+ const { product, icon } = Astro.props;
27
+
28
+ /** Resolve a logo key string to an actual ImageMetadata asset from shared-layouts */
29
+ const resolveLogoAsset = (key: string): ImageMetadata | undefined =>
30
+ (logos as Record<string, ImageMetadata>)[key];
31
+
32
+ const logo = product.logo ? resolveLogoAsset(product.logo) : undefined;
33
+ const darkLogo = product.darkLogo
34
+ ? resolveLogoAsset(product.darkLogo)
35
+ : undefined;
36
+ const isClickable = product.docsUrl && product.status !== "coming-soon";
37
+
38
+ const imageAttrs = {
39
+ loading: "eager" as const,
40
+ decoding: "async" as const,
41
+ width: 48,
42
+ height: 48,
43
+ alt: `${product.name} logo`,
44
+ };
45
+ ---
46
+
47
+ {isClickable ? (
48
+ <a
49
+ href={product.docsUrl}
50
+ target={product.external ? "_blank" : "_self"}
51
+ rel={product.external ? "noopener noreferrer" : undefined}
52
+ >
53
+ <article class="card sl-flex linkcard" aria-label="button" tabindex="0">
54
+ <div class="title-row sl-flex">
55
+ {(logo || darkLogo) && (
56
+ <div class="product-logo">
57
+ {darkLogo && (
58
+ <Image
59
+ src={darkLogo}
60
+ {...imageAttrs}
61
+ class:list={{ "light:sl-hidden": Boolean(logo) }}
62
+ />
63
+ )}
64
+ {logo && (
65
+ <Image
66
+ src={logo}
67
+ {...imageAttrs}
68
+ class="dark:sl-hidden"
69
+ />
70
+ )}
71
+ </div>
72
+ )}
73
+ <p class="title sl-flex">
74
+ {icon && <Icon name={icon} class="icon" size="1.333em" />}
75
+ <span>{product.name}</span>
76
+ </p>
77
+ </div>
78
+ <div class="body">
79
+ <p>
80
+ {product.description}
81
+ {product.highlight && (
82
+ <>
83
+ <br /><br />
84
+ <strong>{product.highlight}</strong>
85
+ </>
86
+ )}
87
+ {product.footnote && (
88
+ <>
89
+ <br /><br />
90
+ <em>{product.footnote}</em>
91
+ </>
92
+ )}
93
+ </p>
94
+ <slot />
95
+ </div>
96
+ </article>
97
+ </a>
98
+ ) : (
99
+ <article class="card sl-flex">
100
+ <div class="title-row sl-flex">
101
+ {(logo || darkLogo) && (
102
+ <div class="product-logo">
103
+ {darkLogo && (
104
+ <Image
105
+ src={darkLogo}
106
+ {...imageAttrs}
107
+ class:list={{ "light:sl-hidden": Boolean(logo) }}
108
+ />
109
+ )}
110
+ {logo && (
111
+ <Image
112
+ src={logo}
113
+ {...imageAttrs}
114
+ class="dark:sl-hidden"
115
+ />
116
+ )}
117
+ </div>
118
+ )}
119
+ <p class="title sl-flex">
120
+ {icon && <Icon name={icon} class="icon" size="1.333em" />}
121
+ <span>{product.name}</span>
122
+ </p>
123
+ </div>
124
+ <div class="body">
125
+ <p>
126
+ {product.description}
127
+ {product.highlight && (
128
+ <>
129
+ <br /><br />
130
+ <strong>{product.highlight}</strong>
131
+ </>
132
+ )}
133
+ {product.footnote && (
134
+ <>
135
+ <br /><br />
136
+ <em>{product.footnote}</em>
137
+ </>
138
+ )}
139
+ </p>
140
+ <slot />
141
+ </div>
142
+ {product.status === "coming-soon" && (
143
+ <span class="status-badge">Docs coming soon</span>
144
+ )}
145
+ </article>
146
+ )}
147
+
148
+ <style>
149
+ @layer starlight.components {
150
+ .card {
151
+ --sl-card-border: var(--knitli-aubergine);
152
+ --sl-card-bg: var(--sl-color-purple-low);
153
+ border: 1px solid var(--sl-color-gray-5);
154
+ background-color: var(--sl-color-black);
155
+ padding: clamp(1rem, calc(0.125rem + 3vw), 2.5rem);
156
+ flex-direction: column;
157
+ gap: clamp(0.5rem, calc(0.125rem + 1vw), 1rem);
158
+ height: 100%;
159
+ position: relative;
160
+ overflow: hidden;
161
+ }
162
+ .title {
163
+ font-weight: 600;
164
+ font-size: var(--sl-text-h4);
165
+ color: var(--sl-color-white);
166
+ line-height: var(--sl-line-height-headings);
167
+ text-decoration: none;
168
+ gap: 1rem;
169
+ align-items: center;
170
+ margin: 0;
171
+ }
172
+ .card .body {
173
+ margin: 0;
174
+ text-decoration: none;
175
+ font-size: clamp(
176
+ var(--sl-text-sm),
177
+ calc(0.5rem + 1vw),
178
+ var(--sl-text-body)
179
+ );
180
+ display: flex;
181
+ flex-direction: column;
182
+ }
183
+ .card .body > p {
184
+ flex: 1;
185
+ line-height: 1.6;
186
+ }
187
+ /* Clickable card styles */
188
+ a {
189
+ text-decoration: none !important;
190
+ color: var(--sl-color-text) !important;
191
+ }
192
+ a:hover {
193
+ text-decoration: none !important;
194
+ color: var(--sl-color-text) !important;
195
+ cursor: pointer;
196
+ }
197
+ a .linkcard {
198
+ transition:
199
+ background-color 0.3s,
200
+ border-color 0.3s;
201
+ cursor: pointer;
202
+ }
203
+ a .linkcard:hover,
204
+ a:hover .linkcard {
205
+ background-color: var(--sl-color-bg-inline-code);
206
+ border: 0.1rem solid var(--knitli-rust);
207
+ scale: 1.02;
208
+ transition: transform 0.3s ease;
209
+ }
210
+
211
+ /* Remove text decoration from all elements inside linked cards */
212
+ a .linkcard *,
213
+ a .linkcard .title,
214
+ a .linkcard .body,
215
+ a .linkcard .body p {
216
+ text-decoration: none !important;
217
+ color: inherit;
218
+ }
219
+
220
+ a .linkcard .title {
221
+ color: var(--sl-color-white);
222
+ }
223
+ .title-row {
224
+ gap: 1rem;
225
+ align-items: center;
226
+ margin-bottom: 1rem;
227
+ }
228
+
229
+ .card .icon {
230
+ border: 1px solid var(--sl-card-border);
231
+ background-color: var(--sl-card-bg);
232
+ padding: 0.2em;
233
+ border-radius: 0.25rem;
234
+ flex-shrink: 0;
235
+ }
236
+
237
+ /* Add padding when stamp is present so description doesn't overlap */
238
+ .card:has(.status-badge) .body {
239
+ padding-bottom: 3rem;
240
+ }
241
+
242
+ .product-logo {
243
+ width: 60px;
244
+ height: 60px;
245
+ display: flex;
246
+ align-items: center;
247
+ justify-content: center;
248
+ flex-shrink: 0;
249
+ }
250
+ .product-logo img {
251
+ width: 100%;
252
+ height: 100%;
253
+ object-fit: contain;
254
+ }
255
+ .product-logo img:hover {
256
+ scale: 1.05;
257
+ transition: transform 0.3s ease;
258
+ }
259
+ .status-badge {
260
+ position: absolute;
261
+ bottom: 1.25rem;
262
+ right: 1.75rem;
263
+ transform: rotate(-8deg);
264
+ transform-origin: center;
265
+
266
+ /* Double-border rubber stamp effect */
267
+ border: 2px solid var(--sl-color-accent);
268
+ outline: 1px solid var(--sl-color-accent);
269
+ outline-offset: 3px;
270
+ border-radius: 1px;
271
+
272
+ color: var(--sl-color-accent);
273
+ background: transparent;
274
+
275
+ font-size: 0.6rem;
276
+ font-weight: 800;
277
+ text-transform: uppercase;
278
+ letter-spacing: 0.18em;
279
+ padding: 0.3em 0.75em;
280
+
281
+ /* Worn ink look */
282
+ opacity: 0.75;
283
+ white-space: nowrap;
284
+ z-index: 10;
285
+ pointer-events: none;
286
+ user-select: none;
287
+ }
288
+ }
289
+ </style>
@@ -0,0 +1,54 @@
1
+ ---
2
+ // SPDX-FileCopyrightText: 2025 Knitli Inc.
3
+ // SPDX-FileContributor: Adam Poulemanos <adam@knit.li>
4
+ //
5
+ // SPDX-License-Identifier: MIT OR Apache-2.0
6
+ //
7
+ // Product grid layout — 3-column on desktop with optional staggered offset.
8
+
9
+ interface Props {
10
+ staggered?: boolean;
11
+ stagger?: boolean;
12
+ }
13
+
14
+ const { staggered, stagger = staggered ?? false } = Astro.props;
15
+ ---
16
+
17
+ <div class:list={["card-grid", { stagger }]}><slot /></div>
18
+
19
+ <style>
20
+ /** biome-ignore: lint/complexity/noImportantStyles */
21
+ @layer starlight.components {
22
+ .card-grid {
23
+ display: grid;
24
+ grid-template-columns: 100%;
25
+ gap: 1rem;
26
+ }
27
+
28
+ .card-grid > :global(*) {
29
+ margin-top: 0 !important;
30
+ }
31
+
32
+ @media (min-width: 50rem) {
33
+ .card-grid {
34
+ grid-template-columns: 1fr 1fr 1fr;
35
+ gap: 1.5rem;
36
+ }
37
+ .stagger {
38
+ --stagger-height: 2.5rem;
39
+ padding-top: var(--stagger-height);
40
+ }
41
+
42
+ /* Pattern: Up, Baseline, Up */
43
+ .stagger > :global(*):nth-child(3n + 1) {
44
+ transform: translateY(calc(-1 * var(--stagger-height)));
45
+ }
46
+ .stagger > :global(*):nth-child(3n + 2) {
47
+ transform: translateY(0);
48
+ }
49
+ .stagger > :global(*):nth-child(3n + 3) {
50
+ transform: translateY(calc(-1 * var(--stagger-height)));
51
+ }
52
+ }
53
+ }
54
+ </style>
@@ -0,0 +1,22 @@
1
+ ---
2
+ // SPDX-FileCopyrightText: 2025 Knitli Inc.
3
+ // SPDX-FileContributor: Adam Poulemanos <adam@knit.li>
4
+ //
5
+ // SPDX-License-Identifier: MIT OR Apache-2.0
6
+ //
7
+ // Composite component that renders the full product showcase grid
8
+ // from the product catalog. Requires zero configuration — just
9
+ // import and use: <ProductShowcase />
10
+
11
+ import { featuredProducts } from "../data/products.js";
12
+ import ProductCard from "./ProductCard.astro";
13
+ import ProductGrid from "./ProductGrid.astro";
14
+
15
+ const products = featuredProducts();
16
+ ---
17
+
18
+ <ProductGrid staggered>
19
+ {products.map((product) => (
20
+ <ProductCard key={product.name} product={product} />
21
+ ))}
22
+ </ProductGrid>
@@ -0,0 +1,53 @@
1
+ ---
2
+ // SPDX-FileCopyrightText: 2026 Knitli Inc.
3
+ // SPDX-FileContributor: Adam Poulemanos <adam@knit.li>
4
+ //
5
+ // SPDX-License-Identifier: MIT OR Apache-2.0
6
+ //
7
+ // Knitli-branded wrapper around astro-seo-meta Seo component.
8
+ // Pre-applies brand theme color, color scheme, and OG defaults.
9
+ //
10
+ // Usage:
11
+ // <SeoMeta title="Getting Started" description="..." />
12
+ // <SeoMeta title="API Reference" twitter={{ card: "summary" }} />
13
+
14
+ import { Seo } from "astro-seo-meta";
15
+
16
+ type ColorScheme = "normal" | "light dark" | "dark light" | "only light";
17
+
18
+ export interface Props {
19
+ title?: string;
20
+ description?: string;
21
+ keywords?: string[];
22
+ icon?: string;
23
+ /** Theme color for browser chrome. Defaults to Knitli aubergine. */
24
+ themeColor?: string;
25
+ colorScheme?: ColorScheme;
26
+ facebook?: {
27
+ image?: string;
28
+ type?: string;
29
+ url?: string;
30
+ };
31
+ twitter?: {
32
+ image?: string;
33
+ card?: string;
34
+ site?: string;
35
+ };
36
+ robots?: string;
37
+ }
38
+
39
+ const {
40
+ themeColor = "#1e061b",
41
+ colorScheme = "light dark",
42
+ twitter = {},
43
+ ...rest
44
+ } = Astro.props;
45
+
46
+ const mergedTwitter = {
47
+ card: "summary_large_image",
48
+ site: "@knitli_inc",
49
+ ...twitter,
50
+ };
51
+ ---
52
+
53
+ <Seo themeColor={themeColor} colorScheme={colorScheme} twitter={mergedTwitter} {...rest} />
package/dist/index.js CHANGED
@@ -1,10 +1,177 @@
1
- export { availableProducts, featuredProducts, getProduct, getProductByName, getProductSitemaps, productNames, products } from './data/products.js';
2
- import { faviconSvg, faviconIco } from './assets/favicon/index.js';
3
- import { knitliWordmark, headlineLogoLight, headlineLogoDark, threadLogoLight, threadLogoDark, codeweaverReverse, codeweaverPrimary, recocoLogoXl, recocoLogoXs, recocoLogoLg, recocoLogoSm, recocoLogoMed, knitliLogo } from './assets/logos/index.js';
4
- import { variables, codeweaverTheme } from './assets/styles/index.js';
5
- import { docsStyle } from './styles/index.js';
1
+ var __create = Object.create;
2
+ var __getProtoOf = Object.getPrototypeOf;
3
+ var __defProp = Object.defineProperty;
4
+ var __getOwnPropNames = Object.getOwnPropertyNames;
5
+ var __hasOwnProp = Object.prototype.hasOwnProperty;
6
+ var __toESM = (mod, isNodeMode, target) => {
7
+ target = mod != null ? __create(__getProtoOf(mod)) : {};
8
+ const to = isNodeMode || !mod || !mod.__esModule ? __defProp(target, "default", { value: mod, enumerable: true }) : target;
9
+ for (let key of __getOwnPropNames(mod))
10
+ if (!__hasOwnProp.call(to, key))
11
+ __defProp(to, key, {
12
+ get: () => mod[key],
13
+ enumerable: true
14
+ });
15
+ return to;
16
+ };
17
+ var __commonJS = (cb, mod) => () => (mod || cb((mod = { exports: {} }).exports, mod), mod.exports);
18
+
19
+ // src/components/DocsBreadcrumb.astro
20
+ var require_DocsBreadcrumb = __commonJS((exports, module) => {
21
+ module.exports = "./DocsBreadcrumb-120137a7.astro";
22
+ });
23
+
24
+ // src/components/Footer.astro
25
+ var require_Footer = __commonJS((exports, module) => {
26
+ module.exports = "./Footer-eayrt0r3.astro";
27
+ });
28
+
29
+ // src/components/PageFrame.astro
30
+ var require_PageFrame = __commonJS((exports, module) => {
31
+ module.exports = "./PageFrame-y7qynxws.astro";
32
+ });
33
+
34
+ // src/components/Contributors.astro
35
+ var require_Contributors = __commonJS((exports, module) => {
36
+ module.exports = "./Contributors-he0tpj6t.astro";
37
+ });
38
+
39
+ // src/components/AllContributors.astro
40
+ var require_AllContributors = __commonJS((exports, module) => {
41
+ module.exports = "./AllContributors-7n8pafkm.astro";
42
+ });
43
+
44
+ // src/components/SeoMeta.astro
45
+ var require_SeoMeta = __commonJS((exports, module) => {
46
+ module.exports = "./SeoMeta-h32x587e.astro";
47
+ });
48
+
49
+ // src/components/ProductCard.astro
50
+ var require_ProductCard = __commonJS((exports, module) => {
51
+ module.exports = "./ProductCard-gebg6mpp.astro";
52
+ });
53
+
54
+ // src/components/ProductGrid.astro
55
+ var require_ProductGrid = __commonJS((exports, module) => {
56
+ module.exports = "./ProductGrid-j9rxskvp.astro";
57
+ });
6
58
 
7
- const DocsAssets = {
59
+ // src/components/ProductShowcase.astro
60
+ var require_ProductShowcase = __commonJS((exports, module) => {
61
+ module.exports = "./ProductShowcase-xjwdmd15.astro";
62
+ });
63
+
64
+ // src/data/products.json
65
+ var products_default = [
66
+ {
67
+ slug: "recoco",
68
+ name: "Recoco",
69
+ tagline: "Intelligent data pipelines",
70
+ description: "Build intelligent data pipelines and ETL workflows. Only process changed data.",
71
+ highlight: "Pure Rust, minimal dependencies, maximum efficiency.",
72
+ status: "available",
73
+ docsUrl: "/recoco/",
74
+ external: false,
75
+ logo: "recocoLogoSm",
76
+ darkLogo: "recocoLogoSm",
77
+ order: 1,
78
+ featured: true
79
+ },
80
+ {
81
+ slug: "codeweaver",
82
+ name: "CodeWeaver",
83
+ tagline: "AI code context engine",
84
+ description: "Your AI code assistants can do better.",
85
+ highlight: "CodeWeaver gives AI agents tailored code context and project understanding, stopping hallucinations, cutting token waste, and supercharging your AI coding workflows.",
86
+ footnote: "Now in Alpha.",
87
+ status: "coming-soon",
88
+ external: false,
89
+ logo: "codeweaverPrimary",
90
+ darkLogo: "codeweaverReverse",
91
+ order: 2,
92
+ featured: true
93
+ },
94
+ {
95
+ slug: "thread",
96
+ name: "Thread",
97
+ tagline: "Ultra-fast codebase analysis",
98
+ description: "An ultra-fast, Rust-native, codebase analysis platform.",
99
+ highlight: "Real-time code understanding and navigation for the largest codebases. Delivers 50x+ speed improvements over comparable tools with minimal resource usage. Works locally, on the edge, or in the cloud.",
100
+ footnote: "Limited features available now, full release coming soon.",
101
+ status: "available",
102
+ docsUrl: "https://docs.rs/thread",
103
+ external: true,
104
+ logo: "threadLogoLight",
105
+ darkLogo: "threadLogoDark",
106
+ order: 3,
107
+ featured: true
108
+ }
109
+ ];
110
+
111
+ // src/data/products.ts
112
+ var products = products_default;
113
+ function featuredProducts() {
114
+ return products.filter((p) => p.featured !== false).sort((a, b) => a.order - b.order);
115
+ }
116
+ function availableProducts() {
117
+ return products.filter((p) => p.status !== "coming-soon");
118
+ }
119
+ function getProduct(slug) {
120
+ return products.find((p) => p.slug === slug);
121
+ }
122
+ function getProductByName(name) {
123
+ return products.find((p) => p.name === name);
124
+ }
125
+ function productNames() {
126
+ return products.map((p) => p.name);
127
+ }
128
+ function getProductSitemaps(baseUrl = "https://docs.knitli.com") {
129
+ return products.filter((p) => p.docsUrl && !p.external && p.status !== "coming-soon").map((p) => `${baseUrl}/${p.slug}/sitemap-index.xml`);
130
+ }
131
+ // src/assets/favicon/index.ts
132
+ import { dirname, join } from "node:path";
133
+ import { fileURLToPath } from "node:url";
134
+ var __dirname2 = dirname(fileURLToPath(import.meta.url));
135
+ var faviconIco = join(__dirname2, "favicon.ico");
136
+ var faviconSvg = join(__dirname2, "favicon.svg");
137
+
138
+ // src/assets/logos/index.ts
139
+ import { dirname as dirname2, join as join2 } from "node:path";
140
+ import { fileURLToPath as fileURLToPath2 } from "node:url";
141
+ var __dirname3 = dirname2(fileURLToPath2(import.meta.url));
142
+ var knitliLogo = join2(__dirname3, "knitli_logo.svg");
143
+ var headlineLogoLight = join2(__dirname3, "headline_logo.webp");
144
+ var headlineLogoDark = join2(__dirname3, "headline_logo_dark.webp");
145
+ var codeweaverPrimary = join2(__dirname3, "codeweaver-primary.svg");
146
+ var codeweaverReverse = join2(__dirname3, "codeweaver-reverse.svg");
147
+ var knitliWordmark = join2(__dirname3, "knitli_wordmark.svg");
148
+ var recocoLogoXl = join2(__dirname3, "recoco-xl.webp");
149
+ var recocoLogoLg = join2(__dirname3, "recoco-lg.webp");
150
+ var recocoLogoMed = join2(__dirname3, "recoco-med.webp");
151
+ var recocoLogoSm = join2(__dirname3, "recoco-sm.webp");
152
+ var recocoLogoXs = join2(__dirname3, "recoco-xs.webp");
153
+ var threadLogoDark = join2(__dirname3, "Thread_vector_mono_parchment.svg");
154
+ var threadLogoLight = join2(__dirname3, "Thread_vector_mono_aubergine.svg");
155
+
156
+ // src/assets/styles/index.ts
157
+ import { dirname as dirname3, join as join3 } from "node:path";
158
+ import { fileURLToPath as fileURLToPath3 } from "node:url";
159
+ var __dirname4 = dirname3(fileURLToPath3(import.meta.url));
160
+ var codeweaverTheme = join3(__dirname4, "codeweaver-theme.css");
161
+ var globalStyles = join3(__dirname4, "global.css");
162
+ var knitliTheme = join3(__dirname4, "knitli-theme.css");
163
+ var textures = join3(__dirname4, "textures.css");
164
+ var typography = join3(__dirname4, "typography.css");
165
+ var variables = join3(__dirname4, "variables.css");
166
+
167
+ // src/styles/index.ts
168
+ import { dirname as dirname4, join as join4 } from "node:path";
169
+ import { fileURLToPath as fileURLToPath4 } from "node:url";
170
+ var __dirname5 = dirname4(fileURLToPath4(import.meta.url));
171
+ var docsStyle = join4(__dirname5, "custom.css");
172
+
173
+ // src/index.ts
174
+ var DocsAssets = {
8
175
  knitliLogo,
9
176
  recocoLogoMed,
10
177
  recocoLogoSm,
@@ -24,20 +191,32 @@ const DocsAssets = {
24
191
  faviconSvg,
25
192
  docsStyle
26
193
  };
27
- const components = {
28
- DocsBreadcrumb: async () => await import('./components/DocsBreadcrumb.astro'),
29
- DocsFooter: async () => await import('./components/Footer.astro'),
30
- PageFrame: async () => await import('./components/PageFrame.astro'),
31
- Contributors: async () => await import('./components/Contributors.astro'),
32
- AllContributors: async () => await import('./components/AllContributors.astro'),
33
- SeoMeta: async () => await import('./components/SeoMeta.astro'),
34
- ProductCard: async () => await import('./components/ProductCard.astro'),
35
- ProductGrid: async () => await import('./components/ProductGrid.astro'),
36
- ProductShowcase: async () => await import('./components/ProductShowcase.astro')
194
+ var components = {
195
+ DocsBreadcrumb: async () => await Promise.resolve().then(() => __toESM(require_DocsBreadcrumb(), 1)),
196
+ DocsFooter: async () => await Promise.resolve().then(() => __toESM(require_Footer(), 1)),
197
+ PageFrame: async () => await Promise.resolve().then(() => __toESM(require_PageFrame(), 1)),
198
+ Contributors: async () => await Promise.resolve().then(() => __toESM(require_Contributors(), 1)),
199
+ AllContributors: async () => await Promise.resolve().then(() => __toESM(require_AllContributors(), 1)),
200
+ SeoMeta: async () => await Promise.resolve().then(() => __toESM(require_SeoMeta(), 1)),
201
+ ProductCard: async () => await Promise.resolve().then(() => __toESM(require_ProductCard(), 1)),
202
+ ProductGrid: async () => await Promise.resolve().then(() => __toESM(require_ProductGrid(), 1)),
203
+ ProductShowcase: async () => await Promise.resolve().then(() => __toESM(require_ProductShowcase(), 1))
37
204
  };
38
- const index = {
205
+ var src_default = {
39
206
  ...components,
40
207
  DocsAssets
41
208
  };
209
+ export {
210
+ products,
211
+ productNames,
212
+ getProductSitemaps,
213
+ getProductByName,
214
+ getProduct,
215
+ featuredProducts,
216
+ src_default as default,
217
+ availableProducts,
218
+ DocsAssets
219
+ };
42
220
 
43
- export { DocsAssets, index as default };
221
+ //# debugId=D99B31DD950B055464756E2164756E21
222
+ //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../src/data/products.ts", "../src/assets/favicon/index.ts", "../src/assets/logos/index.ts", "../src/assets/styles/index.ts", "../src/styles/index.ts", "../src/index.ts"],
  "sourcesContent": [
    "// SPDX-FileCopyrightText: 2025 Knitli Inc.\n// SPDX-FileContributor: Adam Poulemanos <adam@knit.li>\n//\n// SPDX-License-Identifier: MIT OR Apache-2.0\n//\n// Product catalog — single source of truth for all Knitli product metadata.\n// Edit products.json to add, remove, reorder, or update products.\n\nimport catalogData from \"./products.json\" with { type: \"json\" };\n\nexport type ProductStatus = \"available\" | \"coming-soon\" | \"alpha\" | \"beta\";\n\nexport interface Product {\n  /** URL slug used in docs paths, e.g. \"recoco\" */\n  slug: string;\n  /** Display name */\n  name: string;\n  /** Short tagline for cards/nav */\n  tagline: string;\n  /** Primary description */\n  description: string;\n  /** Bold/emphasized callout text */\n  highlight?: string;\n  /** Italicized footnote (e.g. \"Now in Alpha.\") */\n  footnote?: string;\n  /** Product status — drives card rendering and footer links */\n  status: ProductStatus;\n  /** Docs URL — relative for hosted docs, absolute for external */\n  docsUrl?: string;\n  /** Whether docs are hosted externally (e.g. docs.rs) */\n  external?: boolean;\n  /** Logo asset key from DocsAssets */\n  logo: string;\n  /** Dark-mode logo asset key from DocsAssets */\n  darkLogo?: string;\n  /** Sort order for display */\n  order: number;\n  /** Set false to hide from docs-home without deleting */\n  featured?: boolean;\n}\n\n/** All products from the catalog */\nexport const products: Product[] = catalogData as Product[];\n\n/** Products marked as featured (or not explicitly unfeatured), sorted by order */\nexport function featuredProducts(): Product[] {\n  return products\n    .filter((p) => p.featured !== false)\n    .sort((a, b) => a.order - b.order);\n}\n\n/** Products with available documentation (not coming-soon) */\nexport function availableProducts(): Product[] {\n  return products.filter((p) => p.status !== \"coming-soon\");\n}\n\n/** Look up a single product by slug */\nexport function getProduct(slug: string): Product | undefined {\n  return products.find((p) => p.slug === slug);\n}\n\n/** Look up a product by display name */\nexport function getProductByName(name: string): Product | undefined {\n  return products.find((p) => p.name === name);\n}\n\n/** All valid product display names, derived from catalog */\nexport function productNames(): string[] {\n  return products.map((p) => p.name);\n}\n\n/** Generate sitemap URLs for products with hosted (non-external) docs */\nexport function getProductSitemaps(\n  baseUrl = \"https://docs.knitli.com\",\n): string[] {\n  return products\n    .filter((p) => p.docsUrl && !p.external && p.status !== \"coming-soon\")\n    .map((p) => `${baseUrl}/${p.slug}/sitemap-index.xml`);\n}\n",
    "// SPDX-FileCopyrightText: 2025 Knitli Inc.\n// SPDX-License-Identifier: MIT OR Apache-2.0\n//\n// Local logo asset exports - files are copied from @knitli/shared-layouts during build.\n// Do not edit paths here; run `bun run copy-assets` to refresh from source.\n\nimport { dirname, join } from \"node:path\";\nimport { fileURLToPath } from \"node:url\";\n\nconst __dirname = dirname(fileURLToPath(import.meta.url));\n\nexport const faviconIco = join(__dirname, \"favicon.ico\");\nexport const faviconSvg = join(__dirname, \"favicon.svg\");\n\nexport type FaviconAssets = {\n  faviconIco: string;\n  faviconSvg: string;\n};\n",
    "// SPDX-FileCopyrightText: 2025 Knitli Inc.\n// SPDX-License-Identifier: MIT OR Apache-2.0\n//\n// Local logo asset exports - files are copied from @knitli/shared-layouts during build.\n// Do not edit paths here; run `bun run copy-assets` to refresh from source.\n\nimport { dirname, join } from \"node:path\";\nimport { fileURLToPath } from \"node:url\";\n\nconst __dirname = dirname(fileURLToPath(import.meta.url));\n\nexport const knitliLogo = join(__dirname, \"knitli_logo.svg\");\nexport const headlineLogoLight = join(__dirname, \"headline_logo.webp\");\nexport const headlineLogoDark = join(__dirname, \"headline_logo_dark.webp\");\nexport const codeweaverPrimary = join(__dirname, \"codeweaver-primary.svg\");\nexport const codeweaverReverse = join(__dirname, \"codeweaver-reverse.svg\");\nexport const knitliWordmark = join(__dirname, \"knitli_wordmark.svg\");\nexport const recocoLogoXl = join(__dirname, \"recoco-xl.webp\");\nexport const recocoLogoLg = join(__dirname, \"recoco-lg.webp\");\nexport const recocoLogoMed = join(__dirname, \"recoco-med.webp\");\nexport const recocoLogoSm = join(__dirname, \"recoco-sm.webp\");\nexport const recocoLogoXs = join(__dirname, \"recoco-xs.webp\");\nexport const threadLogoDark = join(\n  __dirname,\n  \"Thread_vector_mono_parchment.svg\",\n);\nexport const threadLogoLight = join(\n  __dirname,\n  \"Thread_vector_mono_aubergine.svg\",\n);\n\nexport type LogoAssets = {\n  knitliLogo: string;\n  headlineLogoDark: string;\n  headlineLogoLight: string;\n  codeweaverPrimary: string;\n  codeweaverReverse: string;\n  knitliWordmark: string;\n  recocoLogoXl: string;\n  recocoLogoLg: string;\n  recocoLogoMed: string;\n  recocoLogoSm: string;\n  recocoLogoXs: string;\n  threadLogoDark: string;\n  threadLogoLight: string;\n};\n",
    "// SPDX-FileCopyrightText: 2025 Knitli Inc.\n// SPDX-License-Identifier: MIT OR Apache-2.0\n//\n// Local style asset path exports - CSS files are copied from @knitli/shared-layouts during build.\n// Do not edit; run `bun run copy-assets` to refresh from source.\n\nimport { dirname, join } from \"node:path\";\nimport { fileURLToPath } from \"node:url\";\n\nconst __dirname = dirname(fileURLToPath(import.meta.url));\n\nexport const codeweaverTheme = join(__dirname, \"codeweaver-theme.css\");\nexport const globalStyles = join(__dirname, \"global.css\");\nexport const knitliTheme = join(__dirname, \"knitli-theme.css\");\nexport const textures = join(__dirname, \"textures.css\");\nexport const typography = join(__dirname, \"typography.css\");\nexport const variables = join(__dirname, \"variables.css\");\n\nexport type StyleAssets = {\n  codeweaverTheme: string;\n  globalStyles: string;\n  knitliTheme: string;\n  textures: string;\n  typography: string;\n  variables: string;\n};\n",
    "import { dirname, join } from \"node:path\";\nimport { fileURLToPath } from \"node:url\";\n\nconst __dirname = dirname(fileURLToPath(import.meta.url));\n\nexport const docsStyle = join(__dirname, \"custom.css\");\n",
    "// SPDX-FileCopyrightText: 2025 Knitli Inc.\n// SPDX-FileContributor: Adam Poulemanos <adam@knit.li>\n//\n// SPDX-License-Identifier: MIT OR Apache-2.0\n//\n// Main exports for @knitli/docs-components package\n\nexport * from \"./data/products.js\";\n// Export type definitions and product catalog\nexport * from \"./types/index.js\";\n\nimport { faviconIco, faviconSvg } from \"./assets/favicon/index.js\";\nimport {\n  codeweaverPrimary,\n  codeweaverReverse,\n  headlineLogoDark,\n  headlineLogoLight,\n  knitliLogo,\n  knitliWordmark,\n  recocoLogoLg,\n  recocoLogoMed,\n  recocoLogoSm,\n  recocoLogoXl,\n  recocoLogoXs,\n  threadLogoDark,\n  threadLogoLight,\n} from \"./assets/logos/index.js\";\n\nimport { codeweaverTheme, variables } from \"./assets/styles/index.js\";\n\nimport { docsStyle } from \"./styles/index.js\";\n\nexport const DocsAssets = {\n  knitliLogo,\n  recocoLogoMed,\n  recocoLogoSm,\n  recocoLogoLg,\n  recocoLogoXs,\n  recocoLogoXl,\n  codeweaverPrimary,\n  codeweaverReverse,\n  codeweaverTheme,\n  variables,\n  threadLogoDark,\n  threadLogoLight,\n  headlineLogoDark,\n  headlineLogoLight,\n  knitliWordmark,\n  faviconIco,\n  faviconSvg,\n  docsStyle,\n};\n\nconst components = {\n  DocsBreadcrumb: async () => await import(\"./components/DocsBreadcrumb.astro\"),\n  DocsFooter: async () => await import(\"./components/Footer.astro\"),\n  PageFrame: async () => await import(\"./components/PageFrame.astro\"),\n  Contributors: async () => await import(\"./components/Contributors.astro\"),\n  AllContributors: async () =>\n    await import(\"./components/AllContributors.astro\"),\n  SeoMeta: async () => await import(\"./components/SeoMeta.astro\"),\n  ProductCard: async () => await import(\"./components/ProductCard.astro\"),\n  ProductGrid: async () => await import(\"./components/ProductGrid.astro\"),\n  ProductShowcase: async () =>\n    await import(\"./components/ProductShowcase.astro\"),\n};\n\nexport default {\n  ...components,\n  DocsAssets,\n};\n"
  ],
  "mappings": "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",
  "debugId": "D99B31DD950B055464756E2164756E21",
  "names": []
}