@solidxai/core-ui 0.1.8 → 0.1.9-beta.1

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 (62) hide show
  1. package/dist/components/core/common/SolidAccountSettings/SolidAccountSettings.module.css +149 -0
  2. package/dist/components/core/common/SolidAccountSettings/SolidVersionInfo.d.ts.map +1 -1
  3. package/dist/components/core/common/SolidAccountSettings/SolidVersionInfo.js +21 -4
  4. package/dist/components/core/common/SolidAccountSettings/SolidVersionInfo.js.map +1 -1
  5. package/dist/components/core/common/SolidAccountSettings/SolidVersionInfo.tsx +128 -1
  6. package/dist/components/core/form/SolidFormActionHeader.d.ts.map +1 -1
  7. package/dist/components/core/form/SolidFormActionHeader.js +4 -2
  8. package/dist/components/core/form/SolidFormActionHeader.js.map +1 -1
  9. package/dist/components/core/form/SolidFormActionHeader.tsx +4 -2
  10. package/dist/components/core/form/fields/relations/SolidRelationManyToOneField.js +2 -2
  11. package/dist/components/core/form/fields/relations/SolidRelationManyToOneField.js.map +1 -1
  12. package/dist/components/core/form/fields/relations/SolidRelationManyToOneField.tsx +2 -2
  13. package/dist/components/core/list/SolidListViewConfigure.d.ts.map +1 -1
  14. package/dist/components/core/list/SolidListViewConfigure.js +31 -2
  15. package/dist/components/core/list/SolidListViewConfigure.js.map +1 -1
  16. package/dist/components/core/list/SolidListViewConfigure.tsx +34 -2
  17. package/dist/components/layout/user-profile-menu.d.ts.map +1 -1
  18. package/dist/components/layout/user-profile-menu.js +2 -1
  19. package/dist/components/layout/user-profile-menu.js.map +1 -1
  20. package/dist/components/layout/user-profile-menu.tsx +22 -27
  21. package/dist/components/shad-cn-ui/SolidButton.d.ts +14 -15
  22. package/dist/components/shad-cn-ui/SolidButton.d.ts.map +1 -1
  23. package/dist/components/shad-cn-ui/SolidButton.js +13 -3
  24. package/dist/components/shad-cn-ui/SolidButton.js.map +1 -1
  25. package/dist/components/shad-cn-ui/SolidButton.tsx +24 -4
  26. package/dist/index.d.ts +1 -0
  27. package/dist/index.d.ts.map +1 -1
  28. package/dist/index.js +1 -0
  29. package/dist/index.js.map +1 -1
  30. package/dist/index.ts +1 -0
  31. package/dist/redux/api/solidEntityApi.d.ts.map +1 -1
  32. package/dist/redux/api/solidEntityApi.js +13 -0
  33. package/dist/redux/api/solidEntityApi.js.map +1 -1
  34. package/dist/redux/api/solidEntityApi.tsx +16 -0
  35. package/dist/redux/store/createSolidStore.d.ts +1 -1
  36. package/dist/redux/store/createSolidStore.d.ts.map +1 -1
  37. package/dist/redux/store/createSolidStore.js +20 -11
  38. package/dist/redux/store/createSolidStore.js.map +1 -1
  39. package/dist/redux/store/createSolidStore.ts +20 -11
  40. package/dist/redux/store/dynamicEntityApiMiddleware.d.ts +10 -0
  41. package/dist/redux/store/dynamicEntityApiMiddleware.d.ts.map +1 -0
  42. package/dist/redux/store/dynamicEntityApiMiddleware.js +44 -0
  43. package/dist/redux/store/dynamicEntityApiMiddleware.js.map +1 -0
  44. package/dist/redux/store/dynamicEntityApiMiddleware.ts +58 -0
  45. package/dist/redux/store/dynamicReducerManager.d.ts +10 -0
  46. package/dist/redux/store/dynamicReducerManager.d.ts.map +1 -0
  47. package/dist/redux/store/dynamicReducerManager.js +51 -0
  48. package/dist/redux/store/dynamicReducerManager.js.map +1 -0
  49. package/dist/redux/store/dynamicReducerManager.ts +51 -0
  50. package/dist/redux/store/solidEntityApiPool.d.ts +31 -0
  51. package/dist/redux/store/solidEntityApiPool.d.ts.map +1 -0
  52. package/dist/redux/store/solidEntityApiPool.js +123 -0
  53. package/dist/redux/store/solidEntityApiPool.js.map +1 -0
  54. package/dist/redux/store/solidEntityApiPool.ts +167 -0
  55. package/dist/resources/globals.css +7 -1
  56. package/dist/resources/shadcn-base.css +8 -0
  57. package/dist/routes/pages/admin/core/DiagnosticsPage.d.ts +2 -0
  58. package/dist/routes/pages/admin/core/DiagnosticsPage.d.ts.map +1 -0
  59. package/dist/routes/pages/admin/core/DiagnosticsPage.js +48 -0
  60. package/dist/routes/pages/admin/core/DiagnosticsPage.js.map +1 -0
  61. package/dist/routes/pages/admin/core/DiagnosticsPage.tsx +167 -0
  62. package/package.json +1 -1
@@ -486,6 +486,137 @@
486
486
  font-size: 13px;
487
487
  }
488
488
 
489
+ .versionActions {
490
+ margin-top: 12px;
491
+ display: flex;
492
+ justify-content: flex-start;
493
+ }
494
+
495
+ .diagnosticsBackdrop {
496
+ position: fixed;
497
+ inset: 0;
498
+ z-index: 1300;
499
+ background: rgba(2, 6, 23, 0.5);
500
+ display: grid;
501
+ place-items: center;
502
+ padding: 16px;
503
+ }
504
+
505
+ .diagnosticsModal {
506
+ width: min(1100px, calc(100vw - 24px));
507
+ max-height: calc(100vh - 28px);
508
+ overflow: auto;
509
+ background: var(--card);
510
+ color: var(--card-foreground);
511
+ border: 1px solid var(--border);
512
+ border-radius: 12px;
513
+ box-shadow: 0 22px 50px rgba(2, 6, 23, 0.35);
514
+ padding: 16px;
515
+ }
516
+
517
+ .diagnosticsHeader {
518
+ display: flex;
519
+ align-items: flex-start;
520
+ justify-content: space-between;
521
+ gap: 12px;
522
+ }
523
+
524
+ .diagnosticsHeaderActions {
525
+ display: flex;
526
+ align-items: center;
527
+ gap: 8px;
528
+ }
529
+
530
+ .diagnosticsSummary {
531
+ margin-top: 12px;
532
+ padding: 10px 12px;
533
+ border: 1px solid var(--border);
534
+ border-radius: 10px;
535
+ background: color-mix(in srgb, var(--muted) 18%, transparent);
536
+ display: grid;
537
+ grid-template-columns: repeat(auto-fit, minmax(170px, 1fr));
538
+ gap: 8px 12px;
539
+ font-size: 13px;
540
+ }
541
+
542
+ .diagnosticsTableWrap {
543
+ margin-top: 12px;
544
+ border: 1px solid var(--border);
545
+ border-radius: 10px;
546
+ overflow: auto;
547
+ }
548
+
549
+ .diagnosticsTabs {
550
+ margin-top: 12px;
551
+ display: flex;
552
+ gap: 8px;
553
+ }
554
+
555
+ .diagnosticsTab {
556
+ border: 1px solid var(--border);
557
+ border-radius: 999px;
558
+ background: transparent;
559
+ color: var(--muted-foreground);
560
+ padding: 5px 12px;
561
+ font-size: 12px;
562
+ font-weight: 600;
563
+ cursor: pointer;
564
+ }
565
+
566
+ .diagnosticsTab:hover {
567
+ color: var(--foreground);
568
+ }
569
+
570
+ .diagnosticsTabActive {
571
+ background: color-mix(in srgb, var(--accent) 42%, transparent);
572
+ color: var(--foreground);
573
+ }
574
+
575
+ .diagnosticsTable {
576
+ width: 100%;
577
+ border-collapse: collapse;
578
+ font-size: 13px;
579
+ }
580
+
581
+ .diagnosticsTable thead tr {
582
+ border-bottom: 1px solid var(--border);
583
+ }
584
+
585
+ .diagnosticsTable tbody tr {
586
+ border-bottom: 1px solid color-mix(in srgb, var(--border) 85%, transparent);
587
+ }
588
+
589
+ .diagnosticsTable th,
590
+ .diagnosticsTable td {
591
+ text-align: left;
592
+ padding: 8px 10px;
593
+ vertical-align: top;
594
+ }
595
+
596
+ .diagnosticsEmpty {
597
+ opacity: 0.75;
598
+ }
599
+
600
+ .diagnosticsMono {
601
+ font-family: ui-monospace, SFMono-Regular, Menlo, Monaco, Consolas, "Liberation Mono", "Courier New", monospace;
602
+ }
603
+
604
+ .diagnosticsRuntimeGrid {
605
+ margin-top: 12px;
606
+ border: 1px solid var(--border);
607
+ border-radius: 10px;
608
+ background: color-mix(in srgb, var(--muted) 15%, transparent);
609
+ padding: 12px;
610
+ display: grid;
611
+ grid-template-columns: repeat(2, minmax(0, 1fr));
612
+ gap: 10px 14px;
613
+ font-size: 13px;
614
+ }
615
+
616
+ .diagnosticsSpan2 {
617
+ grid-column: span 2;
618
+ }
619
+
489
620
  .switchRow {
490
621
  border: 1px solid var(--border);
491
622
  border-radius: 10px;
@@ -544,4 +675,22 @@
544
675
  .modal {
545
676
  width: min(98vw, 540px);
546
677
  }
678
+
679
+ .diagnosticsModal {
680
+ width: min(98vw, 1100px);
681
+ padding: 12px;
682
+ }
683
+
684
+ .diagnosticsHeader {
685
+ flex-direction: column;
686
+ align-items: stretch;
687
+ }
688
+
689
+ .diagnosticsRuntimeGrid {
690
+ grid-template-columns: 1fr;
691
+ }
692
+
693
+ .diagnosticsSpan2 {
694
+ grid-column: span 1;
695
+ }
547
696
  }
@@ -1 +1 @@
1
- {"version":3,"file":"SolidVersionInfo.d.ts","sourceRoot":"","sources":["../../../../../src/components/core/common/SolidAccountSettings/SolidVersionInfo.tsx"],"names":[],"mappings":"AAqBA,eAAO,MAAM,gBAAgB,+CA4C5B,CAAC"}
1
+ {"version":3,"file":"SolidVersionInfo.d.ts","sourceRoot":"","sources":["../../../../../src/components/core/common/SolidAccountSettings/SolidVersionInfo.tsx"],"names":[],"mappings":"AA0BA,eAAO,MAAM,gBAAgB,+CAsK5B,CAAC"}
@@ -1,6 +1,11 @@
1
1
  import { jsx as _jsx, jsxs as _jsxs } from "react/jsx-runtime";
2
- import { useEffect } from "react";
2
+ import { useEffect, useMemo, useState } from "react";
3
+ import { env } from "../../../../adapters/env";
4
+ import { hasAnyRole } from "../../../../helpers/rolesHelper";
5
+ import { useSession } from "../../../../hooks/useSession";
3
6
  import { useLazyGetSolidVersionInfoQuery } from "../../../../redux/api/solidSettingsApi";
7
+ import { getSolidEntityApiPoolSnapshot, SOLID_ENTITY_API_POOL_LIMIT } from "../../../../redux/store/solidEntityApiPool";
8
+ import { SolidButton } from "../../../shad-cn-ui/SolidButton";
4
9
  import styles from "./SolidAccountSettings.module.css";
5
10
  var PACKAGE_LABELS = {
6
11
  "solid-core": "Solid Core",
@@ -13,8 +18,13 @@ var PACKAGE_HINTS = {
13
18
  "solid-code-builder": "@solidxai/code-builder",
14
19
  };
15
20
  export var SolidVersionInfo = function () {
16
- var _a;
17
- var _b = useLazyGetSolidVersionInfoQuery(), trigger = _b[0], _c = _b[1], data = _c.data, isLoading = _c.isLoading, isError = _c.isError;
21
+ var _a, _b, _c, _d, _e;
22
+ var session = useSession().data;
23
+ var _f = useLazyGetSolidVersionInfoQuery(), trigger = _f[0], _g = _f[1], data = _g.data, isLoading = _g.isLoading, isError = _g.isError;
24
+ var _h = useState(false), showDiagnostics = _h[0], setShowDiagnostics = _h[1];
25
+ var _j = useState(0), refreshTick = _j[0], setRefreshTick = _j[1];
26
+ var _k = useState("pool"), activeTab = _k[0], setActiveTab = _k[1];
27
+ var poolSnapshot = useMemo(function () { return getSolidEntityApiPoolSnapshot(); }, [refreshTick, showDiagnostics]);
18
28
  useEffect(function () {
19
29
  trigger("");
20
30
  }, [trigger]);
@@ -25,9 +35,16 @@ export var SolidVersionInfo = function () {
25
35
  return _jsx("p", { className: styles.versionError, children: "Unable to load version information." });
26
36
  }
27
37
  var packages = ((_a = data === null || data === void 0 ? void 0 : data.data) !== null && _a !== void 0 ? _a : data);
38
+ var envName = (env("VITE_SOLIDX_ENV") || "").toLowerCase();
39
+ var isDevLikeEnvironment = ["dev", "development", "staging", "stage", "uat", "test", "local"].includes(envName);
40
+ var canOpenDiagnostics = hasAnyRole((_b = session === null || session === void 0 ? void 0 : session.user) === null || _b === void 0 ? void 0 : _b.roles, ["Admin"]) && isDevLikeEnvironment;
41
+ var loginRedirect = env("NEXT_PUBLIC_LOGIN_REDIRECT_URL") || "/admin";
42
+ var backendApi = env("NEXT_PUBLIC_BACKEND_API_URL") || env("API_URL") || "(not set)";
43
+ var roles = (((_c = session === null || session === void 0 ? void 0 : session.user) === null || _c === void 0 ? void 0 : _c.roles) || []).map(function (role) { return (role === null || role === void 0 ? void 0 : role.name) || role; }).join(", ") || "(none)";
44
+ var activeCount = poolSnapshot.filter(function (entry) { return entry.active; }).length;
28
45
  return (_jsxs("div", { className: styles.versionPanel, children: [_jsx("h3", { className: styles.sectionTitle, children: "Version Information" }), _jsx("p", { className: styles.versionCaption, children: "Packages powering this application" }), _jsx("div", { className: styles.versionList, children: Object.entries(packages).map(function (_a) {
29
46
  var key = _a[0], info = _a[1];
30
47
  return (_jsxs("div", { className: styles.versionRow, children: [_jsxs("div", { children: [_jsx("div", { className: styles.versionName, children: PACKAGE_LABELS[key] || key }), _jsx("div", { className: styles.versionHint, children: PACKAGE_HINTS[key] || key })] }), _jsxs("div", { className: styles.versionTags, children: [_jsx("span", { className: styles.versionTag, children: info.version }), _jsx("span", { className: "".concat(styles.versionTag, " ").concat(info.repo === "local" ? styles.repoLocal : styles.repoNpm), children: info.repo })] })] }, key));
31
- }) })] }));
48
+ }) }), canOpenDiagnostics && (_jsx("div", { className: styles.versionActions, children: _jsx(SolidButton, { type: "button", size: "sm", variant: "outline", onClick: function () { return setShowDiagnostics(true); }, children: "Diagnostics" }) })), canOpenDiagnostics && showDiagnostics && (_jsx("div", { className: styles.diagnosticsBackdrop, role: "presentation", onClick: function () { return setShowDiagnostics(false); }, children: _jsxs("section", { className: styles.diagnosticsModal, role: "dialog", "aria-modal": "true", "aria-labelledby": "solid-diagnostics-title", onClick: function (event) { return event.stopPropagation(); }, children: [_jsxs("header", { className: styles.diagnosticsHeader, children: [_jsxs("div", { children: [_jsx("h3", { id: "solid-diagnostics-title", className: styles.sectionTitle, children: "Diagnostics" }), _jsx("p", { className: styles.versionCaption, children: "Current tab runtime entity API pool snapshot." })] }), _jsxs("div", { className: styles.diagnosticsHeaderActions, children: [_jsx(SolidButton, { type: "button", size: "sm", variant: "outline", onClick: function () { return setRefreshTick(function (value) { return value + 1; }); }, children: "Refresh" }), _jsx("button", { type: "button", className: styles.closeButton, onClick: function () { return setShowDiagnostics(false); }, "aria-label": "Close diagnostics", children: "\u00D7" })] })] }), _jsxs("div", { className: styles.diagnosticsSummary, children: [_jsxs("span", { children: [_jsx("strong", { children: "Environment:" }), " ", envName || "unknown"] }), _jsxs("span", { children: [_jsx("strong", { children: "Pool Limit:" }), " ", SOLID_ENTITY_API_POOL_LIMIT] }), _jsxs("span", { children: [_jsx("strong", { children: "Active:" }), " ", activeCount] }), _jsxs("span", { children: [_jsx("strong", { children: "Cached:" }), " ", poolSnapshot.length] })] }), _jsxs("div", { className: styles.diagnosticsTabs, children: [_jsx("button", { type: "button", className: "".concat(styles.diagnosticsTab, " ").concat(activeTab === "pool" ? styles.diagnosticsTabActive : ""), onClick: function () { return setActiveTab("pool"); }, children: "Pool" }), _jsx("button", { type: "button", className: "".concat(styles.diagnosticsTab, " ").concat(activeTab === "runtime" ? styles.diagnosticsTabActive : ""), onClick: function () { return setActiveTab("runtime"); }, children: "Runtime" })] }), activeTab === "pool" ? (_jsx("div", { className: styles.diagnosticsTableWrap, children: _jsxs("table", { className: styles.diagnosticsTable, children: [_jsx("thead", { children: _jsxs("tr", { children: [_jsx("th", { children: "Entity" }), _jsx("th", { children: "Reducer Path" }), _jsx("th", { children: "Registered" }), _jsx("th", { children: "Last Accessed" }), _jsx("th", { children: "Active" })] }) }), _jsx("tbody", { children: poolSnapshot.length === 0 ? (_jsx("tr", { children: _jsx("td", { colSpan: 5, className: styles.diagnosticsEmpty, children: "No entity APIs created in this session yet." }) })) : (poolSnapshot.map(function (entry) { return (_jsxs("tr", { children: [_jsx("td", { children: entry.entityName }), _jsx("td", { className: styles.diagnosticsMono, children: entry.reducerPath }), _jsxs("td", { children: ["#", entry.registeredAt] }), _jsxs("td", { children: ["#", entry.lastAccessedAt] }), _jsx("td", { children: entry.active ? "yes" : "no" })] }, entry.reducerPath)); })) })] }) })) : (_jsxs("div", { className: styles.diagnosticsRuntimeGrid, children: [_jsxs("div", { children: [_jsx("strong", { children: "Environment:" }), " ", envName || "unknown"] }), _jsxs("div", { children: [_jsx("strong", { children: "Backend API:" }), " ", backendApi] }), _jsxs("div", { children: [_jsx("strong", { children: "Login Redirect:" }), " ", loginRedirect] }), _jsxs("div", { children: [_jsx("strong", { children: "User:" }), " ", ((_d = session === null || session === void 0 ? void 0 : session.user) === null || _d === void 0 ? void 0 : _d.email) || ((_e = session === null || session === void 0 ? void 0 : session.user) === null || _e === void 0 ? void 0 : _e.name) || "(unknown)"] }), _jsxs("div", { className: styles.diagnosticsSpan2, children: [_jsx("strong", { children: "Roles:" }), " ", roles] })] }))] }) }))] }));
32
49
  };
33
50
  //# sourceMappingURL=SolidVersionInfo.js.map
@@ -1 +1 @@
1
- {"version":3,"file":"SolidVersionInfo.js","sourceRoot":"","sources":["../../../../../src/components/core/common/SolidAccountSettings/SolidVersionInfo.tsx"],"names":[],"mappings":";AAAA,OAAO,EAAE,SAAS,EAAE,MAAM,OAAO,CAAC;AAClC,OAAO,EAAE,+BAA+B,EAAE,MAAM,wCAAwC,CAAC;AACzF,OAAO,MAAM,MAAM,mCAAmC,CAAC;AAOvD,IAAM,cAAc,GAA2B;IAC7C,YAAY,EAAE,YAAY;IAC1B,eAAe,EAAE,eAAe;IAChC,oBAAoB,EAAE,oBAAoB;CAC3C,CAAC;AAEF,IAAM,aAAa,GAA2B;IAC5C,YAAY,EAAE,gBAAgB;IAC9B,eAAe,EAAE,mBAAmB;IACpC,oBAAoB,EAAE,wBAAwB;CAC/C,CAAC;AAEF,MAAM,CAAC,IAAM,gBAAgB,GAAG;;IACxB,IAAA,KAA0C,+BAA+B,EAAE,EAA1E,OAAO,QAAA,EAAE,UAA4B,EAA1B,IAAI,UAAA,EAAE,SAAS,eAAA,EAAE,OAAO,aAAuC,CAAC;IAElF,SAAS,CAAC;QACR,OAAO,CAAC,EAAE,CAAC,CAAC;IACd,CAAC,EAAE,CAAC,OAAO,CAAC,CAAC,CAAC;IAEd,IAAI,SAAS,EAAE;QACb,OAAO,CACL,cAAK,SAAS,EAAE,MAAM,CAAC,iBAAiB,YACtC,eAAM,SAAS,EAAC,mBAAmB,iBAAa,MAAM,GAAG,GACrD,CACP,CAAC;KACH;IAED,IAAI,OAAO,IAAI,CAAC,IAAI,EAAE;QACpB,OAAO,YAAG,SAAS,EAAE,MAAM,CAAC,YAAY,oDAAyC,CAAC;KACnF;IAED,IAAM,QAAQ,GAAG,CAAC,MAAC,IAAY,aAAZ,IAAI,uBAAJ,IAAI,CAAU,IAAI,mCAAI,IAAI,CAAuC,CAAC;IAErF,OAAO,CACL,eAAK,SAAS,EAAE,MAAM,CAAC,YAAY,aACjC,aAAI,SAAS,EAAE,MAAM,CAAC,YAAY,oCAA0B,EAC5D,YAAG,SAAS,EAAE,MAAM,CAAC,cAAc,mDAAwC,EAE3E,cAAK,SAAS,EAAE,MAAM,CAAC,WAAW,YAC/B,MAAM,CAAC,OAAO,CAAC,QAAQ,CAAC,CAAC,GAAG,CAAC,UAAC,EAAW;wBAAV,GAAG,QAAA,EAAE,IAAI,QAAA;oBAAM,OAAA,CAC7C,eAAe,SAAS,EAAE,MAAM,CAAC,UAAU,aACzC,0BACE,cAAK,SAAS,EAAE,MAAM,CAAC,WAAW,YAAG,cAAc,CAAC,GAAG,CAAC,IAAI,GAAG,GAAO,EACtE,cAAK,SAAS,EAAE,MAAM,CAAC,WAAW,YAAG,aAAa,CAAC,GAAG,CAAC,IAAI,GAAG,GAAO,IACjE,EACN,eAAK,SAAS,EAAE,MAAM,CAAC,WAAW,aAChC,eAAM,SAAS,EAAE,MAAM,CAAC,UAAU,YAAG,IAAI,CAAC,OAAO,GAAQ,EACzD,eAAM,SAAS,EAAE,UAAG,MAAM,CAAC,UAAU,cAAI,IAAI,CAAC,IAAI,KAAK,OAAO,CAAC,CAAC,CAAC,MAAM,CAAC,SAAS,CAAC,CAAC,CAAC,MAAM,CAAC,OAAO,CAAE,YACjG,IAAI,CAAC,IAAI,GACL,IACH,KAVE,GAAG,CAWP,CACP;gBAb8C,CAa9C,CAAC,GACE,IACF,CACP,CAAC;AACJ,CAAC,CAAC","sourcesContent":["import { useEffect } from \"react\";\nimport { useLazyGetSolidVersionInfoQuery } from \"../../../../redux/api/solidSettingsApi\";\nimport styles from \"./SolidAccountSettings.module.css\";\n\ninterface PackageVersionInfo {\n repo: \"local\" | \"npm\";\n version: string;\n}\n\nconst PACKAGE_LABELS: Record<string, string> = {\n \"solid-core\": \"Solid Core\",\n \"solid-core-ui\": \"Solid Core UI\",\n \"solid-code-builder\": \"Solid Code Builder\",\n};\n\nconst PACKAGE_HINTS: Record<string, string> = {\n \"solid-core\": \"@solidxai/core\",\n \"solid-core-ui\": \"@solidxai/core-ui\",\n \"solid-code-builder\": \"@solidxai/code-builder\",\n};\n\nexport const SolidVersionInfo = () => {\n const [trigger, { data, isLoading, isError }] = useLazyGetSolidVersionInfoQuery();\n\n useEffect(() => {\n trigger(\"\");\n }, [trigger]);\n\n if (isLoading) {\n return (\n <div className={styles.versionLoaderWrap}>\n <span className=\"solid-btn-spinner\" aria-hidden=\"true\" />\n </div>\n );\n }\n\n if (isError || !data) {\n return <p className={styles.versionError}>Unable to load version information.</p>;\n }\n\n const packages = ((data as any)?.data ?? data) as Record<string, PackageVersionInfo>;\n\n return (\n <div className={styles.versionPanel}>\n <h3 className={styles.sectionTitle}>Version Information</h3>\n <p className={styles.versionCaption}>Packages powering this application</p>\n\n <div className={styles.versionList}>\n {Object.entries(packages).map(([key, info]) => (\n <div key={key} className={styles.versionRow}>\n <div>\n <div className={styles.versionName}>{PACKAGE_LABELS[key] || key}</div>\n <div className={styles.versionHint}>{PACKAGE_HINTS[key] || key}</div>\n </div>\n <div className={styles.versionTags}>\n <span className={styles.versionTag}>{info.version}</span>\n <span className={`${styles.versionTag} ${info.repo === \"local\" ? styles.repoLocal : styles.repoNpm}`}>\n {info.repo}\n </span>\n </div>\n </div>\n ))}\n </div>\n </div>\n );\n};\n"]}
1
+ {"version":3,"file":"SolidVersionInfo.js","sourceRoot":"","sources":["../../../../../src/components/core/common/SolidAccountSettings/SolidVersionInfo.tsx"],"names":[],"mappings":"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","sourcesContent":["import { useEffect, useMemo, useState } from \"react\";\nimport { env } from \"../../../../adapters/env\";\nimport { hasAnyRole } from \"../../../../helpers/rolesHelper\";\nimport { useSession } from \"../../../../hooks/useSession\";\nimport { useLazyGetSolidVersionInfoQuery } from \"../../../../redux/api/solidSettingsApi\";\nimport { getSolidEntityApiPoolSnapshot, SOLID_ENTITY_API_POOL_LIMIT } from \"../../../../redux/store/solidEntityApiPool\";\nimport { SolidButton } from \"../../../shad-cn-ui/SolidButton\";\nimport styles from \"./SolidAccountSettings.module.css\";\n\ninterface PackageVersionInfo {\n repo: \"local\" | \"npm\";\n version: string;\n}\n\nconst PACKAGE_LABELS: Record<string, string> = {\n \"solid-core\": \"Solid Core\",\n \"solid-core-ui\": \"Solid Core UI\",\n \"solid-code-builder\": \"Solid Code Builder\",\n};\n\nconst PACKAGE_HINTS: Record<string, string> = {\n \"solid-core\": \"@solidxai/core\",\n \"solid-core-ui\": \"@solidxai/core-ui\",\n \"solid-code-builder\": \"@solidxai/code-builder\",\n};\n\nexport const SolidVersionInfo = () => {\n const { data: session } = useSession();\n const [trigger, { data, isLoading, isError }] = useLazyGetSolidVersionInfoQuery();\n const [showDiagnostics, setShowDiagnostics] = useState(false);\n const [refreshTick, setRefreshTick] = useState(0);\n const [activeTab, setActiveTab] = useState<\"pool\" | \"runtime\">(\"pool\");\n const poolSnapshot = useMemo(() => getSolidEntityApiPoolSnapshot(), [refreshTick, showDiagnostics]);\n\n useEffect(() => {\n trigger(\"\");\n }, [trigger]);\n\n if (isLoading) {\n return (\n <div className={styles.versionLoaderWrap}>\n <span className=\"solid-btn-spinner\" aria-hidden=\"true\" />\n </div>\n );\n }\n\n if (isError || !data) {\n return <p className={styles.versionError}>Unable to load version information.</p>;\n }\n\n const packages = ((data as any)?.data ?? data) as Record<string, PackageVersionInfo>;\n const envName = (env(\"VITE_SOLIDX_ENV\") || \"\").toLowerCase();\n const isDevLikeEnvironment = [\"dev\", \"development\", \"staging\", \"stage\", \"uat\", \"test\", \"local\"].includes(envName);\n const canOpenDiagnostics = hasAnyRole(session?.user?.roles, [\"Admin\"]) && isDevLikeEnvironment;\n const loginRedirect = env(\"NEXT_PUBLIC_LOGIN_REDIRECT_URL\") || \"/admin\";\n const backendApi = env(\"NEXT_PUBLIC_BACKEND_API_URL\") || env(\"API_URL\") || \"(not set)\";\n const roles = (session?.user?.roles || []).map((role: any) => role?.name || role).join(\", \") || \"(none)\";\n const activeCount = poolSnapshot.filter((entry) => entry.active).length;\n\n return (\n <div className={styles.versionPanel}>\n <h3 className={styles.sectionTitle}>Version Information</h3>\n <p className={styles.versionCaption}>Packages powering this application</p>\n\n <div className={styles.versionList}>\n {Object.entries(packages).map(([key, info]) => (\n <div key={key} className={styles.versionRow}>\n <div>\n <div className={styles.versionName}>{PACKAGE_LABELS[key] || key}</div>\n <div className={styles.versionHint}>{PACKAGE_HINTS[key] || key}</div>\n </div>\n <div className={styles.versionTags}>\n <span className={styles.versionTag}>{info.version}</span>\n <span className={`${styles.versionTag} ${info.repo === \"local\" ? styles.repoLocal : styles.repoNpm}`}>\n {info.repo}\n </span>\n </div>\n </div>\n ))}\n </div>\n\n {canOpenDiagnostics && (\n <div className={styles.versionActions}>\n <SolidButton\n type=\"button\"\n size=\"sm\"\n variant=\"outline\"\n onClick={() => setShowDiagnostics(true)}\n >\n Diagnostics\n </SolidButton>\n </div>\n )}\n\n {canOpenDiagnostics && showDiagnostics && (\n <div className={styles.diagnosticsBackdrop} role=\"presentation\" onClick={() => setShowDiagnostics(false)}>\n <section\n className={styles.diagnosticsModal}\n role=\"dialog\"\n aria-modal=\"true\"\n aria-labelledby=\"solid-diagnostics-title\"\n onClick={(event) => event.stopPropagation()}\n >\n <header className={styles.diagnosticsHeader}>\n <div>\n <h3 id=\"solid-diagnostics-title\" className={styles.sectionTitle}>Diagnostics</h3>\n <p className={styles.versionCaption}>Current tab runtime entity API pool snapshot.</p>\n </div>\n <div className={styles.diagnosticsHeaderActions}>\n <SolidButton type=\"button\" size=\"sm\" variant=\"outline\" onClick={() => setRefreshTick((value) => value + 1)}>\n Refresh\n </SolidButton>\n <button\n type=\"button\"\n className={styles.closeButton}\n onClick={() => setShowDiagnostics(false)}\n aria-label=\"Close diagnostics\"\n >\n ×\n </button>\n </div>\n </header>\n\n <div className={styles.diagnosticsSummary}>\n <span><strong>Environment:</strong> {envName || \"unknown\"}</span>\n <span><strong>Pool Limit:</strong> {SOLID_ENTITY_API_POOL_LIMIT}</span>\n <span><strong>Active:</strong> {activeCount}</span>\n <span><strong>Cached:</strong> {poolSnapshot.length}</span>\n </div>\n\n <div className={styles.diagnosticsTabs}>\n <button\n type=\"button\"\n className={`${styles.diagnosticsTab} ${activeTab === \"pool\" ? styles.diagnosticsTabActive : \"\"}`}\n onClick={() => setActiveTab(\"pool\")}\n >\n Pool\n </button>\n <button\n type=\"button\"\n className={`${styles.diagnosticsTab} ${activeTab === \"runtime\" ? styles.diagnosticsTabActive : \"\"}`}\n onClick={() => setActiveTab(\"runtime\")}\n >\n Runtime\n </button>\n </div>\n\n {activeTab === \"pool\" ? (\n <div className={styles.diagnosticsTableWrap}>\n <table className={styles.diagnosticsTable}>\n <thead>\n <tr>\n <th>Entity</th>\n <th>Reducer Path</th>\n <th>Registered</th>\n <th>Last Accessed</th>\n <th>Active</th>\n </tr>\n </thead>\n <tbody>\n {poolSnapshot.length === 0 ? (\n <tr>\n <td colSpan={5} className={styles.diagnosticsEmpty}>No entity APIs created in this session yet.</td>\n </tr>\n ) : (\n poolSnapshot.map((entry) => (\n <tr key={entry.reducerPath}>\n <td>{entry.entityName}</td>\n <td className={styles.diagnosticsMono}>{entry.reducerPath}</td>\n <td>#{entry.registeredAt}</td>\n <td>#{entry.lastAccessedAt}</td>\n <td>{entry.active ? \"yes\" : \"no\"}</td>\n </tr>\n ))\n )}\n </tbody>\n </table>\n </div>\n ) : (\n <div className={styles.diagnosticsRuntimeGrid}>\n <div><strong>Environment:</strong> {envName || \"unknown\"}</div>\n <div><strong>Backend API:</strong> {backendApi}</div>\n <div><strong>Login Redirect:</strong> {loginRedirect}</div>\n <div><strong>User:</strong> {session?.user?.email || session?.user?.name || \"(unknown)\"}</div>\n <div className={styles.diagnosticsSpan2}><strong>Roles:</strong> {roles}</div>\n </div>\n )}\n </section>\n </div>\n )}\n </div>\n );\n};\n"]}
@@ -1,5 +1,10 @@
1
- import { useEffect } from "react";
1
+ import { useEffect, useMemo, useState } from "react";
2
+ import { env } from "../../../../adapters/env";
3
+ import { hasAnyRole } from "../../../../helpers/rolesHelper";
4
+ import { useSession } from "../../../../hooks/useSession";
2
5
  import { useLazyGetSolidVersionInfoQuery } from "../../../../redux/api/solidSettingsApi";
6
+ import { getSolidEntityApiPoolSnapshot, SOLID_ENTITY_API_POOL_LIMIT } from "../../../../redux/store/solidEntityApiPool";
7
+ import { SolidButton } from "../../../shad-cn-ui/SolidButton";
3
8
  import styles from "./SolidAccountSettings.module.css";
4
9
 
5
10
  interface PackageVersionInfo {
@@ -20,7 +25,12 @@ const PACKAGE_HINTS: Record<string, string> = {
20
25
  };
21
26
 
22
27
  export const SolidVersionInfo = () => {
28
+ const { data: session } = useSession();
23
29
  const [trigger, { data, isLoading, isError }] = useLazyGetSolidVersionInfoQuery();
30
+ const [showDiagnostics, setShowDiagnostics] = useState(false);
31
+ const [refreshTick, setRefreshTick] = useState(0);
32
+ const [activeTab, setActiveTab] = useState<"pool" | "runtime">("pool");
33
+ const poolSnapshot = useMemo(() => getSolidEntityApiPoolSnapshot(), [refreshTick, showDiagnostics]);
24
34
 
25
35
  useEffect(() => {
26
36
  trigger("");
@@ -39,6 +49,13 @@ export const SolidVersionInfo = () => {
39
49
  }
40
50
 
41
51
  const packages = ((data as any)?.data ?? data) as Record<string, PackageVersionInfo>;
52
+ const envName = (env("VITE_SOLIDX_ENV") || "").toLowerCase();
53
+ const isDevLikeEnvironment = ["dev", "development", "staging", "stage", "uat", "test", "local"].includes(envName);
54
+ const canOpenDiagnostics = hasAnyRole(session?.user?.roles, ["Admin"]) && isDevLikeEnvironment;
55
+ const loginRedirect = env("NEXT_PUBLIC_LOGIN_REDIRECT_URL") || "/admin";
56
+ const backendApi = env("NEXT_PUBLIC_BACKEND_API_URL") || env("API_URL") || "(not set)";
57
+ const roles = (session?.user?.roles || []).map((role: any) => role?.name || role).join(", ") || "(none)";
58
+ const activeCount = poolSnapshot.filter((entry) => entry.active).length;
42
59
 
43
60
  return (
44
61
  <div className={styles.versionPanel}>
@@ -61,6 +78,116 @@ export const SolidVersionInfo = () => {
61
78
  </div>
62
79
  ))}
63
80
  </div>
81
+
82
+ {canOpenDiagnostics && (
83
+ <div className={styles.versionActions}>
84
+ <SolidButton
85
+ type="button"
86
+ size="sm"
87
+ variant="outline"
88
+ onClick={() => setShowDiagnostics(true)}
89
+ >
90
+ Diagnostics
91
+ </SolidButton>
92
+ </div>
93
+ )}
94
+
95
+ {canOpenDiagnostics && showDiagnostics && (
96
+ <div className={styles.diagnosticsBackdrop} role="presentation" onClick={() => setShowDiagnostics(false)}>
97
+ <section
98
+ className={styles.diagnosticsModal}
99
+ role="dialog"
100
+ aria-modal="true"
101
+ aria-labelledby="solid-diagnostics-title"
102
+ onClick={(event) => event.stopPropagation()}
103
+ >
104
+ <header className={styles.diagnosticsHeader}>
105
+ <div>
106
+ <h3 id="solid-diagnostics-title" className={styles.sectionTitle}>Diagnostics</h3>
107
+ <p className={styles.versionCaption}>Current tab runtime entity API pool snapshot.</p>
108
+ </div>
109
+ <div className={styles.diagnosticsHeaderActions}>
110
+ <SolidButton type="button" size="sm" variant="outline" onClick={() => setRefreshTick((value) => value + 1)}>
111
+ Refresh
112
+ </SolidButton>
113
+ <button
114
+ type="button"
115
+ className={styles.closeButton}
116
+ onClick={() => setShowDiagnostics(false)}
117
+ aria-label="Close diagnostics"
118
+ >
119
+ ×
120
+ </button>
121
+ </div>
122
+ </header>
123
+
124
+ <div className={styles.diagnosticsSummary}>
125
+ <span><strong>Environment:</strong> {envName || "unknown"}</span>
126
+ <span><strong>Pool Limit:</strong> {SOLID_ENTITY_API_POOL_LIMIT}</span>
127
+ <span><strong>Active:</strong> {activeCount}</span>
128
+ <span><strong>Cached:</strong> {poolSnapshot.length}</span>
129
+ </div>
130
+
131
+ <div className={styles.diagnosticsTabs}>
132
+ <button
133
+ type="button"
134
+ className={`${styles.diagnosticsTab} ${activeTab === "pool" ? styles.diagnosticsTabActive : ""}`}
135
+ onClick={() => setActiveTab("pool")}
136
+ >
137
+ Pool
138
+ </button>
139
+ <button
140
+ type="button"
141
+ className={`${styles.diagnosticsTab} ${activeTab === "runtime" ? styles.diagnosticsTabActive : ""}`}
142
+ onClick={() => setActiveTab("runtime")}
143
+ >
144
+ Runtime
145
+ </button>
146
+ </div>
147
+
148
+ {activeTab === "pool" ? (
149
+ <div className={styles.diagnosticsTableWrap}>
150
+ <table className={styles.diagnosticsTable}>
151
+ <thead>
152
+ <tr>
153
+ <th>Entity</th>
154
+ <th>Reducer Path</th>
155
+ <th>Registered</th>
156
+ <th>Last Accessed</th>
157
+ <th>Active</th>
158
+ </tr>
159
+ </thead>
160
+ <tbody>
161
+ {poolSnapshot.length === 0 ? (
162
+ <tr>
163
+ <td colSpan={5} className={styles.diagnosticsEmpty}>No entity APIs created in this session yet.</td>
164
+ </tr>
165
+ ) : (
166
+ poolSnapshot.map((entry) => (
167
+ <tr key={entry.reducerPath}>
168
+ <td>{entry.entityName}</td>
169
+ <td className={styles.diagnosticsMono}>{entry.reducerPath}</td>
170
+ <td>#{entry.registeredAt}</td>
171
+ <td>#{entry.lastAccessedAt}</td>
172
+ <td>{entry.active ? "yes" : "no"}</td>
173
+ </tr>
174
+ ))
175
+ )}
176
+ </tbody>
177
+ </table>
178
+ </div>
179
+ ) : (
180
+ <div className={styles.diagnosticsRuntimeGrid}>
181
+ <div><strong>Environment:</strong> {envName || "unknown"}</div>
182
+ <div><strong>Backend API:</strong> {backendApi}</div>
183
+ <div><strong>Login Redirect:</strong> {loginRedirect}</div>
184
+ <div><strong>User:</strong> {session?.user?.email || session?.user?.name || "(unknown)"}</div>
185
+ <div className={styles.diagnosticsSpan2}><strong>Roles:</strong> {roles}</div>
186
+ </div>
187
+ )}
188
+ </section>
189
+ </div>
190
+ )}
64
191
  </div>
65
192
  );
66
193
  };
@@ -1 +1 @@
1
- {"version":3,"file":"SolidFormActionHeader.d.ts","sourceRoot":"","sources":["../../../../src/components/core/form/SolidFormActionHeader.tsx"],"names":[],"mappings":"AAgBA,eAAO,MAAM,qBAAqB,0bAA2b,GAAG,4CA+iB/d,CAAA"}
1
+ {"version":3,"file":"SolidFormActionHeader.d.ts","sourceRoot":"","sources":["../../../../src/components/core/form/SolidFormActionHeader.tsx"],"names":[],"mappings":"AAgBA,eAAO,MAAM,qBAAqB,0bAA2b,GAAG,4CAijB/d,CAAA"}
@@ -31,6 +31,8 @@ export var SolidFormActionHeader = function (_a) {
31
31
  var user = session === null || session === void 0 ? void 0 : session.user;
32
32
  var isPublished = publish && publish !== 'null'; // record is published if publish has value
33
33
  var activeHeaderRequestStatusLabel = headerRequestStatusLabel || (isNavigating ? "Loading..." : null);
34
+ var hasUserInteraction = Object.keys((formik === null || formik === void 0 ? void 0 : formik.touched) || {}).length > 0;
35
+ var shouldShowSaveForExistingRecord = viewMode === "edit" && formik.dirty && hasUserInteraction;
34
36
  useEffect(function () {
35
37
  var _a;
36
38
  if (solidView) {
@@ -194,11 +196,11 @@ export var SolidFormActionHeader = function (_a) {
194
196
  _jsxs(_Fragment, { children: [_jsx("div", { className: "hidden lg:flex", children: _jsx("div", { children: _jsx(SolidButton, { label: "Edit", size: "sm", onClick: function () { return updateViewMode("edit"); }, type: "button" }) }) }), _jsx("div", { className: "lg:hidden", children: _jsx(SolidButton, { icon: "si si-pencil", size: "sm", onClick: function () { return updateViewMode("edit"); }, type: "button", className: "solid-icon-button" }) })] }), params.embeded !== true &&
195
197
  actionsAllowed.includes("".concat(permissionExpression(params.modelName, 'update'))) &&
196
198
  !formViewLayout.attrs.readonly &&
197
- formik.dirty &&
199
+ shouldShowSaveForExistingRecord &&
198
200
  _jsxs("div", { children: [_jsx(SolidButton, { label: "Save", size: "sm", type: "submit", className: "hidden lg:flex", loading: isSubmitting, disabled: isSubmitting }), _jsx(SolidButton, { size: "sm", type: "submit", className: "lg:hidden solid-icon-button", icon: "si si-check", loading: isSubmitting, disabled: isSubmitting })] }), params.embeded == true &&
199
201
  actionsAllowed.includes("".concat(permissionExpression(params.modelName, 'update'))) &&
200
202
  !formViewLayout.attrs.readonly &&
201
- formik.dirty &&
203
+ shouldShowSaveForExistingRecord &&
202
204
  _jsxs("div", { children: [_jsx(SolidButton, { label: "Save", size: "sm", type: "submit", className: "hidden lg:flex", loading: isSubmitting, disabled: isSubmitting }), _jsx(SolidButton, { size: "sm", type: "submit", className: "lg:hidden solid-icon-button", icon: "si si-check", loading: isSubmitting, disabled: isSubmitting })] }), params.embeded == true &&
203
205
  actionsAllowed.includes("".concat(permissionExpression(params.modelName, 'delete'))) &&
204
206
  ((_e = formViewLayout.attrs) === null || _e === void 0 ? void 0 : _e.showDeleteFormButton) !== false &&