tina4-nodejs 3.0.0-rc.2

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 (119) hide show
  1. package/BENCHMARK_REPORT.md +96 -0
  2. package/CARBONAH.md +140 -0
  3. package/CLAUDE.md +599 -0
  4. package/COMPARISON.md +194 -0
  5. package/README.md +595 -0
  6. package/package.json +59 -0
  7. package/packages/cli/src/bin.ts +110 -0
  8. package/packages/cli/src/commands/init.ts +194 -0
  9. package/packages/cli/src/commands/migrate.ts +96 -0
  10. package/packages/cli/src/commands/migrateCreate.ts +59 -0
  11. package/packages/cli/src/commands/routes.ts +61 -0
  12. package/packages/cli/src/commands/serve.ts +58 -0
  13. package/packages/cli/src/commands/test.ts +83 -0
  14. package/packages/core/gallery/auth/meta.json +1 -0
  15. package/packages/core/gallery/auth/src/routes/api/gallery/auth/login/post.ts +22 -0
  16. package/packages/core/gallery/auth/src/routes/api/gallery/auth/verify/get.ts +16 -0
  17. package/packages/core/gallery/auth/src/routes/gallery/auth/get.ts +97 -0
  18. package/packages/core/gallery/database/meta.json +1 -0
  19. package/packages/core/gallery/database/src/routes/api/gallery/db/notes/get.ts +13 -0
  20. package/packages/core/gallery/database/src/routes/api/gallery/db/notes/post.ts +17 -0
  21. package/packages/core/gallery/database/src/routes/api/gallery/db/tables/get.ts +23 -0
  22. package/packages/core/gallery/error-overlay/meta.json +1 -0
  23. package/packages/core/gallery/error-overlay/src/routes/api/gallery/crash/get.ts +17 -0
  24. package/packages/core/gallery/orm/meta.json +1 -0
  25. package/packages/core/gallery/orm/src/routes/api/gallery/products/get.ts +12 -0
  26. package/packages/core/gallery/orm/src/routes/api/gallery/products/post.ts +7 -0
  27. package/packages/core/gallery/queue/meta.json +1 -0
  28. package/packages/core/gallery/queue/src/routes/api/gallery/queue/produce/post.ts +16 -0
  29. package/packages/core/gallery/queue/src/routes/api/gallery/queue/status/get.ts +10 -0
  30. package/packages/core/gallery/rest-api/meta.json +1 -0
  31. package/packages/core/gallery/rest-api/src/routes/api/gallery/hello/get.ts +6 -0
  32. package/packages/core/gallery/rest-api/src/routes/api/gallery/hello/post.ts +7 -0
  33. package/packages/core/gallery/templates/meta.json +1 -0
  34. package/packages/core/gallery/templates/src/routes/gallery/page/get.ts +15 -0
  35. package/packages/core/gallery/templates/src/templates/gallery_page.twig +257 -0
  36. package/packages/core/public/css/tina4.css +2463 -0
  37. package/packages/core/public/css/tina4.min.css +1 -0
  38. package/packages/core/public/favicon.ico +0 -0
  39. package/packages/core/public/images/logo.svg +5 -0
  40. package/packages/core/public/images/tina4-logo-icon.webp +0 -0
  41. package/packages/core/public/js/frond.min.js +420 -0
  42. package/packages/core/public/js/tina4-dev-admin.min.js +327 -0
  43. package/packages/core/public/js/tina4.min.js +93 -0
  44. package/packages/core/public/swagger/index.html +90 -0
  45. package/packages/core/public/swagger/oauth2-redirect.html +63 -0
  46. package/packages/core/src/ai.ts +359 -0
  47. package/packages/core/src/api.ts +248 -0
  48. package/packages/core/src/auth.ts +287 -0
  49. package/packages/core/src/cache.ts +121 -0
  50. package/packages/core/src/constants.ts +48 -0
  51. package/packages/core/src/container.ts +90 -0
  52. package/packages/core/src/devAdmin.ts +2024 -0
  53. package/packages/core/src/devMailbox.ts +316 -0
  54. package/packages/core/src/dotenv.ts +172 -0
  55. package/packages/core/src/errorOverlay.test.ts +122 -0
  56. package/packages/core/src/errorOverlay.ts +278 -0
  57. package/packages/core/src/events.ts +112 -0
  58. package/packages/core/src/fakeData.ts +309 -0
  59. package/packages/core/src/graphql.ts +812 -0
  60. package/packages/core/src/health.ts +31 -0
  61. package/packages/core/src/htmlElement.ts +172 -0
  62. package/packages/core/src/i18n.ts +136 -0
  63. package/packages/core/src/index.ts +88 -0
  64. package/packages/core/src/logger.ts +226 -0
  65. package/packages/core/src/messenger.ts +822 -0
  66. package/packages/core/src/middleware.ts +138 -0
  67. package/packages/core/src/queue.ts +481 -0
  68. package/packages/core/src/queueBackends/kafkaBackend.ts +348 -0
  69. package/packages/core/src/queueBackends/rabbitmqBackend.ts +479 -0
  70. package/packages/core/src/rateLimiter.ts +107 -0
  71. package/packages/core/src/request.ts +189 -0
  72. package/packages/core/src/response.ts +146 -0
  73. package/packages/core/src/routeDiscovery.ts +87 -0
  74. package/packages/core/src/router.ts +398 -0
  75. package/packages/core/src/scss.ts +366 -0
  76. package/packages/core/src/server.ts +610 -0
  77. package/packages/core/src/service.ts +380 -0
  78. package/packages/core/src/session.ts +480 -0
  79. package/packages/core/src/sessionHandlers/mongoHandler.ts +286 -0
  80. package/packages/core/src/sessionHandlers/valkeyHandler.ts +184 -0
  81. package/packages/core/src/static.ts +58 -0
  82. package/packages/core/src/testing.ts +233 -0
  83. package/packages/core/src/types.ts +98 -0
  84. package/packages/core/src/watcher.ts +37 -0
  85. package/packages/core/src/websocket.ts +408 -0
  86. package/packages/core/src/wsdl.ts +546 -0
  87. package/packages/core/templates/errors/302.twig +14 -0
  88. package/packages/core/templates/errors/401.twig +9 -0
  89. package/packages/core/templates/errors/403.twig +29 -0
  90. package/packages/core/templates/errors/404.twig +29 -0
  91. package/packages/core/templates/errors/500.twig +38 -0
  92. package/packages/core/templates/errors/502.twig +9 -0
  93. package/packages/core/templates/errors/503.twig +12 -0
  94. package/packages/core/templates/errors/base.twig +37 -0
  95. package/packages/frond/src/engine.ts +1475 -0
  96. package/packages/frond/src/index.ts +2 -0
  97. package/packages/orm/src/adapters/firebird.ts +455 -0
  98. package/packages/orm/src/adapters/mssql.ts +440 -0
  99. package/packages/orm/src/adapters/mysql.ts +355 -0
  100. package/packages/orm/src/adapters/postgres.ts +362 -0
  101. package/packages/orm/src/adapters/sqlite.ts +270 -0
  102. package/packages/orm/src/autoCrud.ts +231 -0
  103. package/packages/orm/src/baseModel.ts +536 -0
  104. package/packages/orm/src/database.ts +321 -0
  105. package/packages/orm/src/fakeData.ts +118 -0
  106. package/packages/orm/src/index.ts +49 -0
  107. package/packages/orm/src/migration.ts +392 -0
  108. package/packages/orm/src/model.ts +56 -0
  109. package/packages/orm/src/query.ts +113 -0
  110. package/packages/orm/src/seeder.ts +120 -0
  111. package/packages/orm/src/sqlTranslation.ts +272 -0
  112. package/packages/orm/src/types.ts +110 -0
  113. package/packages/orm/src/validation.ts +93 -0
  114. package/packages/swagger/src/generator.ts +189 -0
  115. package/packages/swagger/src/index.ts +2 -0
  116. package/packages/swagger/src/ui.ts +48 -0
  117. package/skills/tina4-developer.skill +0 -0
  118. package/skills/tina4-js.skill +0 -0
  119. package/skills/tina4-maintainer.skill +0 -0
@@ -0,0 +1,16 @@
1
+ /** Gallery: Auth — verify a JWT token. */
2
+ import type { Tina4Request, Tina4Response } from "@tina4/core";
3
+
4
+ export default async function (req: Tina4Request, res: Tina4Response) {
5
+ const url = new URL(req.url ?? "/", "http://localhost");
6
+ const token = url.searchParams.get("token") ?? "";
7
+
8
+ // In a real app: import { Auth } from "@tina4/core";
9
+ // const auth = new Auth();
10
+ // const isValid = auth.validateToken(token);
11
+ // For the gallery demo, just check the token has 3 parts
12
+ const parts = token.split(".");
13
+ const valid = parts.length === 3 && parts.every((p) => p.length > 0);
14
+
15
+ return res.json({ valid });
16
+ }
@@ -0,0 +1,97 @@
1
+ /** Gallery: Auth — JWT login with a visual demo page. */
2
+ import type { Tina4Request, Tina4Response } from "@tina4/core";
3
+
4
+ export default async function (_req: Tina4Request, res: Tina4Response) {
5
+ const html = `<!DOCTYPE html>
6
+ <html lang="en">
7
+ <head>
8
+ <meta charset="utf-8"><meta name="viewport" content="width=device-width,initial-scale=1">
9
+ <title>Auth Demo</title><link rel="stylesheet" href="/css/tina4.min.css">
10
+ </head>
11
+ <body class="bg-dark text-light">
12
+ <div class="container mt-5" style="max-width:600px;">
13
+ <h2 class="mb-4">JWT Authentication Demo</h2>
14
+ <div class="card mb-4">
15
+ <div class="card-header bg-primary text-white">Login</div>
16
+ <div class="card-body">
17
+ <div class="form-group mb-3">
18
+ <label class="form-label">Username</label>
19
+ <input type="text" id="username" class="form-control" placeholder="admin" value="admin">
20
+ </div>
21
+ <div class="form-group mb-3">
22
+ <label class="form-label">Password</label>
23
+ <input type="password" id="password" class="form-control" placeholder="secret" value="secret">
24
+ </div>
25
+ <button class="btn btn-primary" onclick="doLogin()">Login</button>
26
+ </div>
27
+ </div>
28
+ <div id="result" style="display:none;">
29
+ <div class="card mb-3">
30
+ <div class="card-header bg-success text-white">Token Received</div>
31
+ <div class="card-body">
32
+ <pre id="token" style="word-break:break-all;white-space:pre-wrap;color:#4ade80;background:#1e293b;padding:1rem;border-radius:0.5rem;"></pre>
33
+ </div>
34
+ </div>
35
+ <div class="card mb-3">
36
+ <div class="card-header">Token Payload (decoded)</div>
37
+ <div class="card-body">
38
+ <pre id="payload" style="color:#38bdf8;background:#1e293b;padding:1rem;border-radius:0.5rem;"></pre>
39
+ </div>
40
+ </div>
41
+ <button class="btn btn-outline-info" onclick="verifyToken()">Verify Token</button>
42
+ <span id="verify-result" class="ms-2"></span>
43
+ </div>
44
+ <div class="card bg-dark mt-4" style="border:1px solid #334155;">
45
+ <div class="card-body">
46
+ <h6 style="color:#e2e8f0;">How it works</h6>
47
+ <pre style="background:#0f172a;color:#4ade80;padding:1rem;border-radius:0.5rem;font-size:0.8rem;"><code>import { Auth } from "@tina4/core";
48
+ const auth = new Auth();
49
+ const token = auth.createToken({ username: "admin" });
50
+ const payload = auth.getPayload(token);
51
+ const isValid = auth.validateToken(token);</code></pre>
52
+ </div>
53
+ </div>
54
+ </div>
55
+ <script>
56
+ var currentToken = '';
57
+ function doLogin() {
58
+ fetch('/api/gallery/auth/login', {
59
+ method: 'POST',
60
+ headers: {'Content-Type': 'application/json'},
61
+ body: JSON.stringify({
62
+ username: document.getElementById('username').value,
63
+ password: document.getElementById('password').value
64
+ })
65
+ }).then(r => r.json()).then(d => {
66
+ if (d.token) {
67
+ currentToken = d.token;
68
+ document.getElementById('token').textContent = d.token;
69
+ try {
70
+ var parts = d.token.split('.');
71
+ var payload = JSON.parse(atob(parts[1]));
72
+ document.getElementById('payload').textContent = JSON.stringify(payload, null, 2);
73
+ } catch(e) {
74
+ document.getElementById('payload').textContent = 'Could not decode';
75
+ }
76
+ document.getElementById('result').style.display = 'block';
77
+ document.getElementById('verify-result').textContent = '';
78
+ } else {
79
+ alert(d.error || 'Login failed');
80
+ }
81
+ });
82
+ }
83
+ function verifyToken() {
84
+ fetch('/api/gallery/auth/verify?token=' + encodeURIComponent(currentToken))
85
+ .then(r => r.json()).then(d => {
86
+ var el = document.getElementById('verify-result');
87
+ if (d.valid) {
88
+ el.innerHTML = '<span class="badge bg-success">Valid</span>';
89
+ } else {
90
+ el.innerHTML = '<span class="badge bg-danger">Invalid</span>';
91
+ }
92
+ });
93
+ }
94
+ </script>
95
+ </body></html>`;
96
+ return res.html(html);
97
+ }
@@ -0,0 +1 @@
1
+ {"name": "Database", "description": "Raw SQL queries with the Database class", "try_url": "/api/gallery/db/tables"}
@@ -0,0 +1,13 @@
1
+ /** Gallery: Database — list notes from the gallery database. */
2
+ import type { Tina4Request, Tina4Response } from "@tina4/core";
3
+
4
+ export default async function (_req: Tina4Request, res: Tina4Response) {
5
+ try {
6
+ const orm = await import("@tina4/orm");
7
+ const db = orm.initDatabase({ type: "sqlite", path: "./data/gallery.db" });
8
+ const result = await db.fetch("SELECT * FROM gallery_notes ORDER BY id DESC", undefined, 50);
9
+ return res.json(result.records ?? []);
10
+ } catch (e: unknown) {
11
+ return res.json({ error: String(e) }, 500);
12
+ }
13
+ }
@@ -0,0 +1,17 @@
1
+ /** Gallery: Database — create a note in the gallery database. */
2
+ import type { Tina4Request, Tina4Response } from "@tina4/core";
3
+
4
+ export default async function (req: Tina4Request, res: Tina4Response) {
5
+ try {
6
+ const body = (req.body as Record<string, unknown>) ?? {};
7
+ const orm = await import("@tina4/orm");
8
+ const db = orm.initDatabase({ type: "sqlite", path: "./data/gallery.db" });
9
+ await db.insert("gallery_notes", {
10
+ title: (body.title as string) ?? "Untitled",
11
+ body: (body.body as string) ?? "",
12
+ });
13
+ return res.json({ created: true }, 201);
14
+ } catch (e: unknown) {
15
+ return res.json({ error: String(e) }, 500);
16
+ }
17
+ }
@@ -0,0 +1,23 @@
1
+ /** Gallery: Database — list tables in the gallery database. */
2
+ import type { Tina4Request, Tina4Response } from "@tina4/core";
3
+
4
+ export default async function (_req: Tina4Request, res: Tina4Response) {
5
+ try {
6
+ const orm = await import("@tina4/orm");
7
+ const db = orm.initDatabase({ type: "sqlite", path: "./data/gallery.db" });
8
+
9
+ await db.execute(`
10
+ CREATE TABLE IF NOT EXISTS gallery_notes (
11
+ id INTEGER PRIMARY KEY AUTOINCREMENT,
12
+ title TEXT NOT NULL,
13
+ body TEXT,
14
+ created_at DATETIME DEFAULT CURRENT_TIMESTAMP
15
+ )
16
+ `);
17
+
18
+ const tables = await db.getTables();
19
+ return res.json({ tables, engine: "sqlite" });
20
+ } catch (e: unknown) {
21
+ return res.json({ error: String(e) }, 500);
22
+ }
23
+ }
@@ -0,0 +1 @@
1
+ {"name": "Error Overlay", "description": "See the rich debug error page with stack trace and source code", "try_url": "/api/gallery/crash"}
@@ -0,0 +1,17 @@
1
+ /** Gallery: Error Overlay — deliberately crash to demo the debug overlay.
2
+ *
3
+ * In debug mode (TINA4_DEBUG=true), you will see:
4
+ * - Exception type and message
5
+ * - Stack trace with syntax-highlighted source code
6
+ * - The exact line that caused the error (highlighted)
7
+ * - Request details (method, path, headers)
8
+ * - Environment info (framework version, Node.js version)
9
+ */
10
+ import type { Tina4Request, Tina4Response } from "@tina4/core";
11
+
12
+ export default async function (_req: Tina4Request, _res: Tina4Response) {
13
+ // Simulate a realistic error — accessing a missing property
14
+ const user: Record<string, string> = { name: "Alice", email: "alice@example.com" };
15
+ const role = (user as any).role.toUpperCase(); // TypeError — this line will be highlighted in the overlay
16
+ return _res.json({ role });
17
+ }
@@ -0,0 +1 @@
1
+ {"name": "ORM", "description": "Product model with CRUD endpoints", "try_url": "/api/gallery/products"}
@@ -0,0 +1,12 @@
1
+ /** Gallery: ORM — list products (demo data). */
2
+ import type { Tina4Request, Tina4Response } from "@tina4/core";
3
+
4
+ export default async function (_req: Tina4Request, res: Tina4Response) {
5
+ return res.json({
6
+ products: [
7
+ { id: 1, name: "Widget", price: 9.99 },
8
+ { id: 2, name: "Gadget", price: 24.99 },
9
+ ],
10
+ note: "Connect a database and deploy the ORM model for live data",
11
+ });
12
+ }
@@ -0,0 +1,7 @@
1
+ /** Gallery: ORM — create a product (demo). */
2
+ import type { Tina4Request, Tina4Response } from "@tina4/core";
3
+
4
+ export default async function (req: Tina4Request, res: Tina4Response) {
5
+ const data = (req.body as Record<string, unknown>) ?? {};
6
+ return res.json({ created: data, id: 3 }, 201);
7
+ }
@@ -0,0 +1 @@
1
+ {"name": "Queue", "description": "Background job producer and consumer", "try_url": "/api/gallery/queue/produce"}
@@ -0,0 +1,16 @@
1
+ /** Gallery: Queue — produce a background job. */
2
+ import type { Tina4Request, Tina4Response } from "@tina4/core";
3
+
4
+ export default async function (req: Tina4Request, res: Tina4Response) {
5
+ const body = (req.body as Record<string, unknown>) ?? {};
6
+ const task = (body.task as string) ?? "default-task";
7
+ const data = body.data ?? {};
8
+
9
+ // In a real app you would use:
10
+ // import { Queue, Producer } from "@tina4/core";
11
+ // const queue = new Queue(db, "gallery-tasks");
12
+ // const producer = new Producer(queue);
13
+ // producer.produce({ task, data });
14
+
15
+ return res.json({ queued: true, task }, 201);
16
+ }
@@ -0,0 +1,10 @@
1
+ /** Gallery: Queue — check queue status. */
2
+ import type { Tina4Request, Tina4Response } from "@tina4/core";
3
+
4
+ export default async function (_req: Tina4Request, res: Tina4Response) {
5
+ return res.json({
6
+ topic: "gallery-tasks",
7
+ size: 0,
8
+ note: "Connect a queue backend for live data",
9
+ });
10
+ }
@@ -0,0 +1 @@
1
+ {"name": "REST API", "description": "A simple JSON API with GET and POST endpoints", "try_url": "/api/gallery/hello"}
@@ -0,0 +1,6 @@
1
+ /** Gallery: REST API — simple JSON GET endpoint. */
2
+ import type { Tina4Request, Tina4Response } from "@tina4/core";
3
+
4
+ export default async function (req: Tina4Request, res: Tina4Response) {
5
+ return res.json({ message: "Hello from Tina4!", method: "GET" });
6
+ }
@@ -0,0 +1,7 @@
1
+ /** Gallery: REST API — simple JSON POST endpoint. */
2
+ import type { Tina4Request, Tina4Response } from "@tina4/core";
3
+
4
+ export default async function (req: Tina4Request, res: Tina4Response) {
5
+ const data = (req.body as Record<string, unknown>) ?? {};
6
+ return res.json({ echo: data, method: "POST" }, 201);
7
+ }
@@ -0,0 +1 @@
1
+ {"name": "Templates", "description": "Twig template with dynamic data", "try_url": "/gallery/page"}
@@ -0,0 +1,15 @@
1
+ /** Gallery: Templates — render an HTML page with dynamic data via template export. */
2
+ import type { Tina4Request, Tina4Response } from "@tina4/core";
3
+
4
+ export const template = "gallery_page.twig";
5
+
6
+ export default async function (_req: Tina4Request, _res: Tina4Response) {
7
+ return {
8
+ title: "Gallery Demo Page",
9
+ items: [
10
+ { name: "Tina4 Node.js", description: "Zero-dep web framework", badge: "v3.0.0" },
11
+ { name: "Twig Engine", description: "Built-in template rendering", badge: "included" },
12
+ { name: "Auto-Reload", description: "Templates refresh on save", badge: "dev mode" },
13
+ ],
14
+ };
15
+ }
@@ -0,0 +1,257 @@
1
+ <!DOCTYPE html>
2
+ <html lang="en">
3
+ <head>
4
+ <meta charset="utf-8">
5
+ <meta name="viewport" content="width=device-width, initial-scale=1">
6
+ <title>{{ title }}</title>
7
+ <link rel="stylesheet" href="/css/tina4.min.css">
8
+ <style>
9
+ .component-section { margin-bottom: 2.5rem; }
10
+ .component-section h3 { border-bottom: 2px solid var(--bs-primary, #0d6efd); padding-bottom: 0.5rem; margin-bottom: 1rem; }
11
+ .code-preview { background: #1e293b; color: #4ade80; padding: 1rem; border-radius: 0.5rem; font-family: monospace; font-size: 0.85rem; overflow-x: auto; margin-top: 0.75rem; }
12
+ </style>
13
+ </head>
14
+ <body>
15
+
16
+ <!-- Navbar -->
17
+ <nav class="navbar navbar-dark bg-dark navbar-expand-lg">
18
+ <div class="container">
19
+ <a class="navbar-brand" href="/">Tina4 CSS Showcase</a>
20
+ <div class="navbar-nav ms-auto">
21
+ <a class="nav-link active" href="#">Components</a>
22
+ <a class="nav-link" href="/__dev">Dashboard</a>
23
+ <a class="nav-link" href="/swagger">API Docs</a>
24
+ </div>
25
+ </div>
26
+ </nav>
27
+
28
+ <div class="container mt-4">
29
+
30
+ <div class="alert alert-info alert-dismissible mb-4">
31
+ <strong>tina4css</strong> — Zero-dependency CSS framework (~24KB). Bootstrap-compatible class names, dark mode ready. No CDN needed — ships with every Tina4 project.
32
+ <button type="button" class="btn-close" onclick="this.parentElement.remove()"></button>
33
+ </div>
34
+
35
+ <!-- Buttons -->
36
+ <div class="component-section">
37
+ <h3>Buttons</h3>
38
+ <div class="d-flex flex-wrap gap-2 mb-3">
39
+ <button class="btn btn-primary">Primary</button>
40
+ <button class="btn btn-secondary">Secondary</button>
41
+ <button class="btn btn-success">Success</button>
42
+ <button class="btn btn-danger">Danger</button>
43
+ <button class="btn btn-warning">Warning</button>
44
+ <button class="btn btn-info">Info</button>
45
+ <button class="btn btn-dark">Dark</button>
46
+ <button class="btn btn-light">Light</button>
47
+ </div>
48
+ <div class="d-flex flex-wrap gap-2 mb-3">
49
+ <button class="btn btn-outline-primary">Outline</button>
50
+ <button class="btn btn-outline-success">Outline</button>
51
+ <button class="btn btn-outline-danger">Outline</button>
52
+ <button class="btn btn-outline-warning">Outline</button>
53
+ </div>
54
+ <div class="d-flex flex-wrap gap-2">
55
+ <button class="btn btn-primary btn-lg">Large</button>
56
+ <button class="btn btn-primary">Default</button>
57
+ <button class="btn btn-primary btn-sm">Small</button>
58
+ </div>
59
+ </div>
60
+
61
+ <!-- Alerts -->
62
+ <div class="component-section">
63
+ <h3>Alerts</h3>
64
+ <div class="alert alert-success">Success — Record saved successfully!</div>
65
+ <div class="alert alert-danger">Error — Something went wrong.</div>
66
+ <div class="alert alert-warning">Warning — Please check your input.</div>
67
+ <div class="alert alert-info">Info — New version available.</div>
68
+ </div>
69
+
70
+ <!-- Cards -->
71
+ <div class="component-section">
72
+ <h3>Cards</h3>
73
+ <div class="row">
74
+ {% for item in items %}
75
+ <div class="col-md-4 mb-3">
76
+ <div class="card h-100">
77
+ <div class="card-header bg-primary text-white">{{ item.name }}</div>
78
+ <div class="card-body">
79
+ <p class="card-text">{{ item.description }}</p>
80
+ <span class="badge bg-success">{{ item.badge }}</span>
81
+ </div>
82
+ <div class="card-footer text-muted">
83
+ <small>Built with tina4css</small>
84
+ </div>
85
+ </div>
86
+ </div>
87
+ {% endfor %}
88
+ </div>
89
+ </div>
90
+
91
+ <!-- Badges -->
92
+ <div class="component-section">
93
+ <h3>Badges</h3>
94
+ <span class="badge bg-primary me-1">Primary</span>
95
+ <span class="badge bg-secondary me-1">Secondary</span>
96
+ <span class="badge bg-success me-1">Success</span>
97
+ <span class="badge bg-danger me-1">Danger</span>
98
+ <span class="badge bg-warning me-1">Warning</span>
99
+ <span class="badge bg-info me-1">Info</span>
100
+ <span class="badge bg-dark me-1">Dark</span>
101
+ </div>
102
+
103
+ <!-- Forms -->
104
+ <div class="component-section">
105
+ <h3>Forms</h3>
106
+ <div class="card">
107
+ <div class="card-body">
108
+ <form>
109
+ <div class="row mb-3">
110
+ <div class="col-md-6">
111
+ <div class="form-group">
112
+ <label class="form-label">Full Name</label>
113
+ <input type="text" class="form-control" placeholder="Andre van Zuydam">
114
+ </div>
115
+ </div>
116
+ <div class="col-md-6">
117
+ <div class="form-group">
118
+ <label class="form-label">Email</label>
119
+ <input type="email" class="form-control" placeholder="you@example.com">
120
+ </div>
121
+ </div>
122
+ </div>
123
+ <div class="row mb-3">
124
+ <div class="col-md-6">
125
+ <div class="form-group">
126
+ <label class="form-label">Framework</label>
127
+ <select class="form-select">
128
+ <option>Python</option>
129
+ <option>PHP</option>
130
+ <option>Ruby</option>
131
+ <option selected>Node.js</option>
132
+ </select>
133
+ </div>
134
+ </div>
135
+ <div class="col-md-6">
136
+ <div class="form-group">
137
+ <label class="form-label">Database</label>
138
+ <select class="form-select">
139
+ <option>SQLite</option>
140
+ <option>PostgreSQL</option>
141
+ <option>MySQL</option>
142
+ <option>Firebird</option>
143
+ <option>MSSQL</option>
144
+ </select>
145
+ </div>
146
+ </div>
147
+ </div>
148
+ <div class="form-group mb-3">
149
+ <label class="form-label">Message</label>
150
+ <textarea class="form-control" rows="3" placeholder="Tell us about your project..."></textarea>
151
+ </div>
152
+ <div class="form-check mb-3">
153
+ <input class="form-check-input" type="checkbox" checked>
154
+ <label class="form-check-label">I agree to the terms</label>
155
+ </div>
156
+ <button type="button" class="btn btn-primary">Submit</button>
157
+ <button type="button" class="btn btn-outline-secondary ms-2">Cancel</button>
158
+ </form>
159
+ </div>
160
+ </div>
161
+ </div>
162
+
163
+ <!-- Table -->
164
+ <div class="component-section">
165
+ <h3>Tables</h3>
166
+ <div class="table-responsive">
167
+ <table class="table table-striped table-hover">
168
+ <thead>
169
+ <tr>
170
+ <th>#</th>
171
+ <th>Framework</th>
172
+ <th>Port</th>
173
+ <th>Language</th>
174
+ <th>Status</th>
175
+ </tr>
176
+ </thead>
177
+ <tbody>
178
+ <tr><td>1</td><td>tina4-python</td><td>7145</td><td>Python 3.12+</td><td><span class="badge bg-success">Stable</span></td></tr>
179
+ <tr><td>2</td><td>tina4-php</td><td>7146</td><td>PHP 8.2+</td><td><span class="badge bg-success">Stable</span></td></tr>
180
+ <tr><td>3</td><td>tina4-ruby</td><td>7147</td><td>Ruby 3.1+</td><td><span class="badge bg-success">Stable</span></td></tr>
181
+ <tr><td>4</td><td>tina4-nodejs</td><td>7148</td><td>Node 20+</td><td><span class="badge bg-success">Stable</span></td></tr>
182
+ </tbody>
183
+ </table>
184
+ </div>
185
+ </div>
186
+
187
+ <!-- Breadcrumbs -->
188
+ <div class="component-section">
189
+ <h3>Breadcrumbs</h3>
190
+ <nav>
191
+ <ol class="breadcrumb">
192
+ <li class="breadcrumb-item"><a href="/">Home</a></li>
193
+ <li class="breadcrumb-item"><a href="#">Gallery</a></li>
194
+ <li class="breadcrumb-item active">Components</li>
195
+ </ol>
196
+ </nav>
197
+ </div>
198
+
199
+ <!-- Input Groups -->
200
+ <div class="component-section">
201
+ <h3>Input Groups</h3>
202
+ <div class="input-group mb-3">
203
+ <span class="input-group-text">@</span>
204
+ <input type="text" class="form-control" placeholder="Username">
205
+ </div>
206
+ <div class="input-group mb-3">
207
+ <input type="text" class="form-control" placeholder="Search routes...">
208
+ <button class="btn btn-primary">Search</button>
209
+ </div>
210
+ <div class="input-group">
211
+ <span class="input-group-text">https://</span>
212
+ <input type="text" class="form-control" placeholder="tina4.com">
213
+ <span class="input-group-text">/api</span>
214
+ </div>
215
+ </div>
216
+
217
+ <!-- Progress -->
218
+ <div class="component-section">
219
+ <h3>Progress Bars</h3>
220
+ <div class="progress mb-2"><div class="progress-bar bg-primary" style="width: 25%">25%</div></div>
221
+ <div class="progress mb-2"><div class="progress-bar bg-success" style="width: 50%">50%</div></div>
222
+ <div class="progress mb-2"><div class="progress-bar bg-warning" style="width: 75%">75%</div></div>
223
+ <div class="progress"><div class="progress-bar bg-danger" style="width: 100%">100%</div></div>
224
+ </div>
225
+
226
+ <!-- List Group -->
227
+ <div class="component-section">
228
+ <h3>List Group</h3>
229
+ <div class="list-group">
230
+ <a href="#" class="list-group-item list-group-item-action active">Routes — 12 registered</a>
231
+ <a href="#" class="list-group-item list-group-item-action">Queue — 3 pending jobs</a>
232
+ <a href="#" class="list-group-item list-group-item-action">Database — SQLite connected</a>
233
+ <a href="#" class="list-group-item list-group-item-action disabled">Cache — Not configured</a>
234
+ </div>
235
+ </div>
236
+
237
+ <!-- How this page was built -->
238
+ <div class="card mt-4 mb-5" style="background:#0f172a;border:1px solid #334155;">
239
+ <div class="card-body">
240
+ <h5 style="color:#e2e8f0;">How this page was built</h5>
241
+ <pre style="background:#1e293b;border:1px solid #334155;border-radius:0.5rem;padding:1.25rem;margin-top:0.75rem;overflow-x:auto;font-family:'SF Mono',SFMono-Regular,Consolas,monospace;font-size:0.9rem;line-height:1.7;"><code><span style="color:#64748b">// src/routes/gallery/page/get.ts</span>
242
+ <span style="color:#c084fc">export const</span> <span style="color:#fbbf24">template</span> = <span style="color:#4ade80">"gallery_page.twig"</span>;
243
+
244
+ <span style="color:#c084fc">export default</span> <span style="color:#38bdf8">async function</span>(<span style="color:#38bdf8">_req</span>, <span style="color:#38bdf8">_res</span>) {
245
+ <span style="color:#c084fc">return</span> { <span style="color:#4ade80">"title"</span>: <span style="color:#4ade80">"tina4css Component Showcase"</span>, <span style="color:#4ade80">"items"</span>: [...] };
246
+ }</code></pre>
247
+ <p style="color:#94a3b8;margin-top:0.75rem;margin-bottom:0;">
248
+ Rendered via <code style="color:#c084fc;">template</code> export. Styled with <strong style="color:#e2e8f0;">tina4css</strong> — zero external CDN.
249
+ All components above use only <code style="color:#4ade80;">&lt;link href="/css/tina4.min.css"&gt;</code>.
250
+ </p>
251
+ </div>
252
+ </div>
253
+
254
+ </div>
255
+
256
+ </body>
257
+ </html>