@contractspec/example.saas-boilerplate 1.56.1 → 1.58.0

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 (284) hide show
  1. package/.turbo/turbo-build.log +160 -188
  2. package/.turbo/turbo-prebuild.log +1 -0
  3. package/CHANGELOG.md +45 -0
  4. package/dist/billing/billing.entity.d.ts +40 -45
  5. package/dist/billing/billing.entity.d.ts.map +1 -1
  6. package/dist/billing/billing.entity.js +110 -118
  7. package/dist/billing/billing.enum.d.ts +2 -8
  8. package/dist/billing/billing.enum.d.ts.map +1 -1
  9. package/dist/billing/billing.enum.js +17 -24
  10. package/dist/billing/billing.event.d.ts +67 -73
  11. package/dist/billing/billing.event.d.ts.map +1 -1
  12. package/dist/billing/billing.event.js +84 -146
  13. package/dist/billing/billing.handler.d.ts +59 -62
  14. package/dist/billing/billing.handler.d.ts.map +1 -1
  15. package/dist/billing/billing.handler.js +140 -49
  16. package/dist/billing/billing.operations.d.ts +138 -144
  17. package/dist/billing/billing.operations.d.ts.map +1 -1
  18. package/dist/billing/billing.operations.js +273 -175
  19. package/dist/billing/billing.presentation.d.ts +2 -7
  20. package/dist/billing/billing.presentation.d.ts.map +1 -1
  21. package/dist/billing/billing.presentation.js +51 -57
  22. package/dist/billing/billing.schema.d.ts +159 -164
  23. package/dist/billing/billing.schema.d.ts.map +1 -1
  24. package/dist/billing/billing.schema.js +112 -204
  25. package/dist/billing/index.d.ts +11 -8
  26. package/dist/billing/index.d.ts.map +1 -0
  27. package/dist/billing/index.js +689 -9
  28. package/dist/browser/billing/billing.entity.js +113 -0
  29. package/dist/browser/billing/billing.enum.js +19 -0
  30. package/dist/browser/billing/billing.event.js +90 -0
  31. package/dist/browser/billing/billing.handler.js +148 -0
  32. package/dist/browser/billing/billing.operations.js +278 -0
  33. package/dist/browser/billing/billing.presentation.js +52 -0
  34. package/dist/browser/billing/billing.schema.js +121 -0
  35. package/dist/browser/billing/index.js +688 -0
  36. package/dist/browser/dashboard/dashboard.presentation.js +52 -0
  37. package/dist/browser/dashboard/index.js +52 -0
  38. package/dist/browser/docs/index.js +93 -0
  39. package/dist/browser/docs/saas-boilerplate.docblock.js +93 -0
  40. package/dist/browser/example.js +39 -0
  41. package/dist/browser/handlers/index.js +358 -0
  42. package/dist/browser/handlers/saas.handlers.js +134 -0
  43. package/dist/browser/index.js +3340 -0
  44. package/dist/browser/presentations/index.js +290 -0
  45. package/dist/browser/project/index.js +790 -0
  46. package/dist/browser/project/project.entity.js +77 -0
  47. package/dist/browser/project/project.enum.js +18 -0
  48. package/dist/browser/project/project.event.js +103 -0
  49. package/dist/browser/project/project.handler.js +178 -0
  50. package/dist/browser/project/project.operations.js +372 -0
  51. package/dist/browser/project/project.presentation.js +177 -0
  52. package/dist/browser/project/project.schema.js +134 -0
  53. package/dist/browser/saas-boilerplate.feature.js +88 -0
  54. package/dist/browser/seeders/index.js +20 -0
  55. package/dist/browser/settings/index.js +75 -0
  56. package/dist/browser/settings/settings.entity.js +74 -0
  57. package/dist/browser/settings/settings.enum.js +11 -0
  58. package/dist/browser/shared/mock-data.js +104 -0
  59. package/dist/browser/shared/overlay-types.js +0 -0
  60. package/dist/browser/tests/operations.test-spec.js +112 -0
  61. package/dist/browser/ui/SaasDashboard.js +988 -0
  62. package/dist/browser/ui/SaasProjectList.js +162 -0
  63. package/dist/browser/ui/SaasSettingsPanel.js +145 -0
  64. package/dist/browser/ui/hooks/index.js +159 -0
  65. package/dist/browser/ui/hooks/useProjectList.js +66 -0
  66. package/dist/browser/ui/hooks/useProjectMutations.js +91 -0
  67. package/dist/browser/ui/index.js +1808 -0
  68. package/dist/browser/ui/modals/CreateProjectModal.js +153 -0
  69. package/dist/browser/ui/modals/ProjectActionsModal.js +335 -0
  70. package/dist/browser/ui/modals/index.js +487 -0
  71. package/dist/browser/ui/overlays/demo-overlays.js +61 -0
  72. package/dist/browser/ui/overlays/index.js +61 -0
  73. package/dist/browser/ui/renderers/index.js +675 -0
  74. package/dist/browser/ui/renderers/project-list.markdown.js +499 -0
  75. package/dist/browser/ui/renderers/project-list.renderer.js +177 -0
  76. package/dist/dashboard/dashboard.presentation.d.ts +2 -7
  77. package/dist/dashboard/dashboard.presentation.d.ts.map +1 -1
  78. package/dist/dashboard/dashboard.presentation.js +51 -53
  79. package/dist/dashboard/index.d.ts +5 -2
  80. package/dist/dashboard/index.d.ts.map +1 -0
  81. package/dist/dashboard/index.js +53 -3
  82. package/dist/docs/index.d.ts +2 -1
  83. package/dist/docs/index.d.ts.map +1 -0
  84. package/dist/docs/index.js +94 -1
  85. package/dist/docs/saas-boilerplate.docblock.d.ts +2 -1
  86. package/dist/docs/saas-boilerplate.docblock.d.ts.map +1 -0
  87. package/dist/docs/saas-boilerplate.docblock.js +45 -51
  88. package/dist/example.d.ts +2 -6
  89. package/dist/example.d.ts.map +1 -1
  90. package/dist/example.js +37 -50
  91. package/dist/handlers/index.d.ts +7 -4
  92. package/dist/handlers/index.d.ts.map +1 -0
  93. package/dist/handlers/index.js +358 -4
  94. package/dist/handlers/saas.handlers.d.ts +60 -60
  95. package/dist/handlers/saas.handlers.d.ts.map +1 -1
  96. package/dist/handlers/saas.handlers.js +127 -140
  97. package/dist/index.d.ts +15 -45
  98. package/dist/index.d.ts.map +1 -1
  99. package/dist/index.js +3335 -75
  100. package/dist/node/billing/billing.entity.js +113 -0
  101. package/dist/node/billing/billing.enum.js +19 -0
  102. package/dist/node/billing/billing.event.js +90 -0
  103. package/dist/node/billing/billing.handler.js +148 -0
  104. package/dist/node/billing/billing.operations.js +278 -0
  105. package/dist/node/billing/billing.presentation.js +52 -0
  106. package/dist/node/billing/billing.schema.js +121 -0
  107. package/dist/node/billing/index.js +688 -0
  108. package/dist/node/dashboard/dashboard.presentation.js +52 -0
  109. package/dist/node/dashboard/index.js +52 -0
  110. package/dist/node/docs/index.js +93 -0
  111. package/dist/node/docs/saas-boilerplate.docblock.js +93 -0
  112. package/dist/node/example.js +39 -0
  113. package/dist/node/handlers/index.js +358 -0
  114. package/dist/node/handlers/saas.handlers.js +134 -0
  115. package/dist/node/index.js +3340 -0
  116. package/dist/node/presentations/index.js +290 -0
  117. package/dist/node/project/index.js +790 -0
  118. package/dist/node/project/project.entity.js +77 -0
  119. package/dist/node/project/project.enum.js +18 -0
  120. package/dist/node/project/project.event.js +103 -0
  121. package/dist/node/project/project.handler.js +178 -0
  122. package/dist/node/project/project.operations.js +372 -0
  123. package/dist/node/project/project.presentation.js +177 -0
  124. package/dist/node/project/project.schema.js +134 -0
  125. package/dist/node/saas-boilerplate.feature.js +88 -0
  126. package/dist/node/seeders/index.js +20 -0
  127. package/dist/node/settings/index.js +75 -0
  128. package/dist/node/settings/settings.entity.js +74 -0
  129. package/dist/node/settings/settings.enum.js +11 -0
  130. package/dist/node/shared/mock-data.js +104 -0
  131. package/dist/node/shared/overlay-types.js +0 -0
  132. package/dist/node/tests/operations.test-spec.js +112 -0
  133. package/dist/node/ui/SaasDashboard.js +988 -0
  134. package/dist/node/ui/SaasProjectList.js +162 -0
  135. package/dist/node/ui/SaasSettingsPanel.js +145 -0
  136. package/dist/node/ui/hooks/index.js +159 -0
  137. package/dist/node/ui/hooks/useProjectList.js +66 -0
  138. package/dist/node/ui/hooks/useProjectMutations.js +91 -0
  139. package/dist/node/ui/index.js +1808 -0
  140. package/dist/node/ui/modals/CreateProjectModal.js +153 -0
  141. package/dist/node/ui/modals/ProjectActionsModal.js +335 -0
  142. package/dist/node/ui/modals/index.js +487 -0
  143. package/dist/node/ui/overlays/demo-overlays.js +61 -0
  144. package/dist/node/ui/overlays/index.js +61 -0
  145. package/dist/node/ui/renderers/index.js +675 -0
  146. package/dist/node/ui/renderers/project-list.markdown.js +499 -0
  147. package/dist/node/ui/renderers/project-list.renderer.js +177 -0
  148. package/dist/presentations/index.d.ts +13 -15
  149. package/dist/presentations/index.d.ts.map +1 -1
  150. package/dist/presentations/index.js +289 -15
  151. package/dist/project/index.d.ts +11 -8
  152. package/dist/project/index.d.ts.map +1 -0
  153. package/dist/project/index.js +791 -9
  154. package/dist/project/project.entity.d.ts +23 -28
  155. package/dist/project/project.entity.d.ts.map +1 -1
  156. package/dist/project/project.entity.js +75 -82
  157. package/dist/project/project.enum.d.ts +2 -8
  158. package/dist/project/project.enum.d.ts.map +1 -1
  159. package/dist/project/project.enum.js +16 -23
  160. package/dist/project/project.event.d.ts +69 -75
  161. package/dist/project/project.event.d.ts.map +1 -1
  162. package/dist/project/project.event.js +95 -156
  163. package/dist/project/project.handler.d.ts +44 -47
  164. package/dist/project/project.handler.d.ts.map +1 -1
  165. package/dist/project/project.handler.js +168 -71
  166. package/dist/project/project.operations.d.ts +341 -347
  167. package/dist/project/project.operations.d.ts.map +1 -1
  168. package/dist/project/project.operations.js +366 -253
  169. package/dist/project/project.presentation.d.ts +2 -7
  170. package/dist/project/project.presentation.d.ts.map +1 -1
  171. package/dist/project/project.presentation.js +174 -61
  172. package/dist/project/project.schema.d.ts +191 -196
  173. package/dist/project/project.schema.d.ts.map +1 -1
  174. package/dist/project/project.schema.js +125 -205
  175. package/dist/saas-boilerplate.feature.d.ts +1 -7
  176. package/dist/saas-boilerplate.feature.d.ts.map +1 -1
  177. package/dist/saas-boilerplate.feature.js +87 -206
  178. package/dist/seeders/index.d.ts +4 -8
  179. package/dist/seeders/index.d.ts.map +1 -1
  180. package/dist/seeders/index.js +18 -16
  181. package/dist/settings/index.d.ts +6 -3
  182. package/dist/settings/index.d.ts.map +1 -0
  183. package/dist/settings/index.js +75 -3
  184. package/dist/settings/settings.entity.d.ts +23 -28
  185. package/dist/settings/settings.entity.d.ts.map +1 -1
  186. package/dist/settings/settings.entity.js +72 -75
  187. package/dist/settings/settings.enum.d.ts +1 -6
  188. package/dist/settings/settings.enum.d.ts.map +1 -1
  189. package/dist/settings/settings.enum.js +10 -19
  190. package/dist/shared/mock-data.d.ts +74 -77
  191. package/dist/shared/mock-data.d.ts.map +1 -1
  192. package/dist/shared/mock-data.js +102 -135
  193. package/dist/shared/overlay-types.d.ts +25 -28
  194. package/dist/shared/overlay-types.d.ts.map +1 -1
  195. package/dist/shared/overlay-types.js +1 -0
  196. package/dist/tests/operations.test-spec.d.ts +4 -9
  197. package/dist/tests/operations.test-spec.d.ts.map +1 -1
  198. package/dist/tests/operations.test-spec.js +108 -118
  199. package/dist/ui/SaasDashboard.d.ts +1 -6
  200. package/dist/ui/SaasDashboard.d.ts.map +1 -1
  201. package/dist/ui/SaasDashboard.js +977 -286
  202. package/dist/ui/SaasProjectList.d.ts +4 -11
  203. package/dist/ui/SaasProjectList.d.ts.map +1 -1
  204. package/dist/ui/SaasProjectList.js +159 -72
  205. package/dist/ui/SaasSettingsPanel.d.ts +1 -6
  206. package/dist/ui/SaasSettingsPanel.d.ts.map +1 -1
  207. package/dist/ui/SaasSettingsPanel.js +142 -134
  208. package/dist/ui/hooks/index.d.ts +3 -3
  209. package/dist/ui/hooks/index.d.ts.map +1 -0
  210. package/dist/ui/hooks/index.js +158 -4
  211. package/dist/ui/hooks/useProjectList.d.ts +26 -30
  212. package/dist/ui/hooks/useProjectList.d.ts.map +1 -1
  213. package/dist/ui/hooks/useProjectList.js +63 -71
  214. package/dist/ui/hooks/useProjectMutations.d.ts +20 -24
  215. package/dist/ui/hooks/useProjectMutations.d.ts.map +1 -1
  216. package/dist/ui/hooks/useProjectMutations.js +88 -142
  217. package/dist/ui/index.d.ts +8 -14
  218. package/dist/ui/index.d.ts.map +1 -0
  219. package/dist/ui/index.js +1809 -15
  220. package/dist/ui/modals/CreateProjectModal.d.ts +10 -19
  221. package/dist/ui/modals/CreateProjectModal.d.ts.map +1 -1
  222. package/dist/ui/modals/CreateProjectModal.js +150 -135
  223. package/dist/ui/modals/ProjectActionsModal.d.ts +20 -33
  224. package/dist/ui/modals/ProjectActionsModal.d.ts.map +1 -1
  225. package/dist/ui/modals/ProjectActionsModal.js +333 -289
  226. package/dist/ui/modals/index.d.ts +3 -3
  227. package/dist/ui/modals/index.d.ts.map +1 -0
  228. package/dist/ui/modals/index.js +487 -3
  229. package/dist/ui/overlays/demo-overlays.d.ts +10 -9
  230. package/dist/ui/overlays/demo-overlays.d.ts.map +1 -1
  231. package/dist/ui/overlays/demo-overlays.js +60 -68
  232. package/dist/ui/overlays/index.d.ts +2 -2
  233. package/dist/ui/overlays/index.d.ts.map +1 -0
  234. package/dist/ui/overlays/index.js +62 -3
  235. package/dist/ui/renderers/index.d.ts +3 -3
  236. package/dist/ui/renderers/index.d.ts.map +1 -0
  237. package/dist/ui/renderers/index.js +675 -3
  238. package/dist/ui/renderers/project-list.markdown.d.ts +15 -15
  239. package/dist/ui/renderers/project-list.markdown.d.ts.map +1 -1
  240. package/dist/ui/renderers/project-list.markdown.js +496 -144
  241. package/dist/ui/renderers/project-list.renderer.d.ts +6 -8
  242. package/dist/ui/renderers/project-list.renderer.d.ts.map +1 -1
  243. package/dist/ui/renderers/project-list.renderer.js +176 -15
  244. package/package.json +509 -99
  245. package/src/ui/renderers/project-list.markdown.ts +1 -1
  246. package/tsdown.config.js +1 -2
  247. package/.turbo/turbo-build$colon$bundle.log +0 -188
  248. package/dist/billing/billing.entity.js.map +0 -1
  249. package/dist/billing/billing.enum.js.map +0 -1
  250. package/dist/billing/billing.event.js.map +0 -1
  251. package/dist/billing/billing.handler.js.map +0 -1
  252. package/dist/billing/billing.operations.js.map +0 -1
  253. package/dist/billing/billing.presentation.js.map +0 -1
  254. package/dist/billing/billing.schema.js.map +0 -1
  255. package/dist/dashboard/dashboard.presentation.js.map +0 -1
  256. package/dist/docs/saas-boilerplate.docblock.js.map +0 -1
  257. package/dist/example.js.map +0 -1
  258. package/dist/handlers/saas.handlers.js.map +0 -1
  259. package/dist/index.js.map +0 -1
  260. package/dist/presentations/index.js.map +0 -1
  261. package/dist/project/project.entity.js.map +0 -1
  262. package/dist/project/project.enum.js.map +0 -1
  263. package/dist/project/project.event.js.map +0 -1
  264. package/dist/project/project.handler.js.map +0 -1
  265. package/dist/project/project.operations.js.map +0 -1
  266. package/dist/project/project.presentation.js.map +0 -1
  267. package/dist/project/project.schema.js.map +0 -1
  268. package/dist/saas-boilerplate.feature.js.map +0 -1
  269. package/dist/seeders/index.js.map +0 -1
  270. package/dist/settings/settings.entity.js.map +0 -1
  271. package/dist/settings/settings.enum.js.map +0 -1
  272. package/dist/shared/mock-data.js.map +0 -1
  273. package/dist/tests/operations.test-spec.js.map +0 -1
  274. package/dist/ui/SaasDashboard.js.map +0 -1
  275. package/dist/ui/SaasProjectList.js.map +0 -1
  276. package/dist/ui/SaasSettingsPanel.js.map +0 -1
  277. package/dist/ui/hooks/useProjectList.js.map +0 -1
  278. package/dist/ui/hooks/useProjectMutations.js.map +0 -1
  279. package/dist/ui/modals/CreateProjectModal.js.map +0 -1
  280. package/dist/ui/modals/ProjectActionsModal.js.map +0 -1
  281. package/dist/ui/overlays/demo-overlays.js.map +0 -1
  282. package/dist/ui/renderers/project-list.markdown.js.map +0 -1
  283. package/dist/ui/renderers/project-list.renderer.js.map +0 -1
  284. package/tsconfig.tsbuildinfo +0 -1
@@ -0,0 +1,153 @@
1
+ // src/ui/modals/CreateProjectModal.tsx
2
+ import { useState } from "react";
3
+ import { Button, Input } from "@contractspec/lib.design-system";
4
+ import { jsxDEV } from "react/jsx-dev-runtime";
5
+ "use client";
6
+ var TIERS = [
7
+ { value: "FREE", label: "Free" },
8
+ { value: "PRO", label: "Pro" },
9
+ { value: "ENTERPRISE", label: "Enterprise" }
10
+ ];
11
+ function CreateProjectModal({
12
+ isOpen,
13
+ onClose,
14
+ onSubmit,
15
+ isLoading = false
16
+ }) {
17
+ const [name, setName] = useState("");
18
+ const [description, setDescription] = useState("");
19
+ const [tier, setTier] = useState("FREE");
20
+ const [error, setError] = useState(null);
21
+ const handleSubmit = async (e) => {
22
+ e.preventDefault();
23
+ setError(null);
24
+ if (!name.trim()) {
25
+ setError("Project name is required");
26
+ return;
27
+ }
28
+ try {
29
+ await onSubmit({
30
+ name: name.trim(),
31
+ description: description.trim() || undefined,
32
+ tier
33
+ });
34
+ setName("");
35
+ setDescription("");
36
+ setTier("FREE");
37
+ onClose();
38
+ } catch (err) {
39
+ setError(err instanceof Error ? err.message : "Failed to create project");
40
+ }
41
+ };
42
+ if (!isOpen)
43
+ return null;
44
+ return /* @__PURE__ */ jsxDEV("div", {
45
+ className: "fixed inset-0 z-50 flex items-center justify-center",
46
+ children: [
47
+ /* @__PURE__ */ jsxDEV("div", {
48
+ className: "bg-background/80 absolute inset-0 backdrop-blur-sm",
49
+ onClick: onClose,
50
+ role: "button",
51
+ tabIndex: 0,
52
+ onKeyDown: (e) => {
53
+ if (e.key === "Enter" || e.key === " ")
54
+ onClose();
55
+ },
56
+ "aria-label": "Close modal"
57
+ }, undefined, false, undefined, this),
58
+ /* @__PURE__ */ jsxDEV("div", {
59
+ className: "bg-card border-border relative z-10 w-full max-w-md rounded-xl border p-6 shadow-xl",
60
+ children: [
61
+ /* @__PURE__ */ jsxDEV("h2", {
62
+ className: "mb-4 text-xl font-semibold",
63
+ children: "Create New Project"
64
+ }, undefined, false, undefined, this),
65
+ /* @__PURE__ */ jsxDEV("form", {
66
+ onSubmit: handleSubmit,
67
+ className: "space-y-4",
68
+ children: [
69
+ /* @__PURE__ */ jsxDEV("div", {
70
+ children: [
71
+ /* @__PURE__ */ jsxDEV("label", {
72
+ htmlFor: "project-name",
73
+ className: "text-muted-foreground mb-1 block text-sm font-medium",
74
+ children: "Project Name *"
75
+ }, undefined, false, undefined, this),
76
+ /* @__PURE__ */ jsxDEV(Input, {
77
+ id: "project-name",
78
+ value: name,
79
+ onChange: (e) => setName(e.target.value),
80
+ placeholder: "e.g., My Awesome Project",
81
+ disabled: isLoading
82
+ }, undefined, false, undefined, this)
83
+ ]
84
+ }, undefined, true, undefined, this),
85
+ /* @__PURE__ */ jsxDEV("div", {
86
+ children: [
87
+ /* @__PURE__ */ jsxDEV("label", {
88
+ htmlFor: "project-description",
89
+ className: "text-muted-foreground mb-1 block text-sm font-medium",
90
+ children: "Description"
91
+ }, undefined, false, undefined, this),
92
+ /* @__PURE__ */ jsxDEV("textarea", {
93
+ id: "project-description",
94
+ value: description,
95
+ onChange: (e) => setDescription(e.target.value),
96
+ placeholder: "Describe what this project is about...",
97
+ rows: 3,
98
+ disabled: isLoading,
99
+ className: "border-input bg-background focus:ring-ring w-full rounded-md border px-3 py-2 text-sm focus:ring-2 focus:outline-none disabled:opacity-50"
100
+ }, undefined, false, undefined, this)
101
+ ]
102
+ }, undefined, true, undefined, this),
103
+ /* @__PURE__ */ jsxDEV("div", {
104
+ children: [
105
+ /* @__PURE__ */ jsxDEV("label", {
106
+ htmlFor: "project-tier",
107
+ className: "text-muted-foreground mb-1 block text-sm font-medium",
108
+ children: "Tier"
109
+ }, undefined, false, undefined, this),
110
+ /* @__PURE__ */ jsxDEV("select", {
111
+ id: "project-tier",
112
+ value: tier,
113
+ onChange: (e) => setTier(e.target.value),
114
+ disabled: isLoading,
115
+ className: "border-input bg-background focus:ring-ring h-10 w-full rounded-md border px-3 py-2 text-sm focus:ring-2 focus:outline-none disabled:opacity-50",
116
+ children: TIERS.map((t) => /* @__PURE__ */ jsxDEV("option", {
117
+ value: t.value,
118
+ children: t.label
119
+ }, t.value, false, undefined, this))
120
+ }, undefined, false, undefined, this)
121
+ ]
122
+ }, undefined, true, undefined, this),
123
+ error && /* @__PURE__ */ jsxDEV("div", {
124
+ className: "bg-destructive/10 text-destructive rounded-md p-3 text-sm",
125
+ children: error
126
+ }, undefined, false, undefined, this),
127
+ /* @__PURE__ */ jsxDEV("div", {
128
+ className: "flex justify-end gap-3 pt-2",
129
+ children: [
130
+ /* @__PURE__ */ jsxDEV(Button, {
131
+ type: "button",
132
+ variant: "ghost",
133
+ onPress: onClose,
134
+ disabled: isLoading,
135
+ children: "Cancel"
136
+ }, undefined, false, undefined, this),
137
+ /* @__PURE__ */ jsxDEV(Button, {
138
+ type: "submit",
139
+ disabled: isLoading,
140
+ children: isLoading ? "Creating..." : "Create Project"
141
+ }, undefined, false, undefined, this)
142
+ ]
143
+ }, undefined, true, undefined, this)
144
+ ]
145
+ }, undefined, true, undefined, this)
146
+ ]
147
+ }, undefined, true, undefined, this)
148
+ ]
149
+ }, undefined, true, undefined, this);
150
+ }
151
+ export {
152
+ CreateProjectModal
153
+ };
@@ -0,0 +1,335 @@
1
+ // src/ui/modals/ProjectActionsModal.tsx
2
+ import { useEffect, useState } from "react";
3
+ import { Button, Input } from "@contractspec/lib.design-system";
4
+ import { jsxDEV } from "react/jsx-dev-runtime";
5
+ "use client";
6
+ function ProjectActionsModal({
7
+ isOpen,
8
+ project,
9
+ onClose,
10
+ onUpdate,
11
+ onArchive,
12
+ onActivate,
13
+ onDelete,
14
+ isLoading = false
15
+ }) {
16
+ const [mode, setMode] = useState("menu");
17
+ const [name, setName] = useState("");
18
+ const [description, setDescription] = useState("");
19
+ const [error, setError] = useState(null);
20
+ const resetForm = () => {
21
+ setMode("menu");
22
+ setError(null);
23
+ if (project) {
24
+ setName(project.name);
25
+ setDescription(project.description ?? "");
26
+ }
27
+ };
28
+ const handleClose = () => {
29
+ resetForm();
30
+ onClose();
31
+ };
32
+ useEffect(() => {
33
+ if (project) {
34
+ setName(project.name);
35
+ setDescription(project.description ?? "");
36
+ }
37
+ }, [project]);
38
+ const handleEdit = async () => {
39
+ if (!project)
40
+ return;
41
+ setError(null);
42
+ if (!name.trim()) {
43
+ setError("Project name is required");
44
+ return;
45
+ }
46
+ try {
47
+ await onUpdate({
48
+ id: project.id,
49
+ name: name.trim(),
50
+ description: description.trim() || undefined
51
+ });
52
+ handleClose();
53
+ } catch (err) {
54
+ setError(err instanceof Error ? err.message : "Failed to update project");
55
+ }
56
+ };
57
+ const handleArchive = async () => {
58
+ if (!project)
59
+ return;
60
+ setError(null);
61
+ try {
62
+ await onArchive(project.id);
63
+ handleClose();
64
+ } catch (err) {
65
+ setError(err instanceof Error ? err.message : "Failed to archive project");
66
+ }
67
+ };
68
+ const handleActivate = async () => {
69
+ if (!project)
70
+ return;
71
+ setError(null);
72
+ try {
73
+ await onActivate(project.id);
74
+ handleClose();
75
+ } catch (err) {
76
+ setError(err instanceof Error ? err.message : "Failed to activate project");
77
+ }
78
+ };
79
+ const handleDelete = async () => {
80
+ if (!project)
81
+ return;
82
+ setError(null);
83
+ try {
84
+ await onDelete(project.id);
85
+ handleClose();
86
+ } catch (err) {
87
+ setError(err instanceof Error ? err.message : "Failed to delete project");
88
+ }
89
+ };
90
+ if (!isOpen || !project)
91
+ return null;
92
+ return /* @__PURE__ */ jsxDEV("div", {
93
+ className: "fixed inset-0 z-50 flex items-center justify-center",
94
+ children: [
95
+ /* @__PURE__ */ jsxDEV("div", {
96
+ className: "bg-background/80 absolute inset-0 backdrop-blur-sm",
97
+ onClick: handleClose,
98
+ role: "button",
99
+ tabIndex: 0,
100
+ onKeyDown: (e) => {
101
+ if (e.key === "Enter" || e.key === " ")
102
+ handleClose();
103
+ },
104
+ "aria-label": "Close modal"
105
+ }, undefined, false, undefined, this),
106
+ /* @__PURE__ */ jsxDEV("div", {
107
+ className: "bg-card border-border relative z-10 w-full max-w-md rounded-xl border p-6 shadow-xl",
108
+ children: [
109
+ /* @__PURE__ */ jsxDEV("div", {
110
+ className: "border-border mb-4 border-b pb-4",
111
+ children: [
112
+ /* @__PURE__ */ jsxDEV("h2", {
113
+ className: "text-xl font-semibold",
114
+ children: project.name
115
+ }, undefined, false, undefined, this),
116
+ /* @__PURE__ */ jsxDEV("p", {
117
+ className: "text-muted-foreground text-sm",
118
+ children: [
119
+ project.tier,
120
+ " · ",
121
+ project.status
122
+ ]
123
+ }, undefined, true, undefined, this)
124
+ ]
125
+ }, undefined, true, undefined, this),
126
+ mode === "menu" && /* @__PURE__ */ jsxDEV("div", {
127
+ className: "space-y-3",
128
+ children: [
129
+ /* @__PURE__ */ jsxDEV(Button, {
130
+ className: "w-full justify-start",
131
+ variant: "ghost",
132
+ onPress: () => setMode("edit"),
133
+ children: [
134
+ /* @__PURE__ */ jsxDEV("span", {
135
+ className: "mr-2",
136
+ children: "✏️"
137
+ }, undefined, false, undefined, this),
138
+ " Edit Project"
139
+ ]
140
+ }, undefined, true, undefined, this),
141
+ project.status === "ACTIVE" || project.status === "DRAFT" ? /* @__PURE__ */ jsxDEV(Button, {
142
+ className: "w-full justify-start",
143
+ variant: "ghost",
144
+ onPress: () => setMode("archive"),
145
+ children: [
146
+ /* @__PURE__ */ jsxDEV("span", {
147
+ className: "mr-2",
148
+ children: "\uD83D\uDCE6"
149
+ }, undefined, false, undefined, this),
150
+ " Archive Project"
151
+ ]
152
+ }, undefined, true, undefined, this) : project.status === "ARCHIVED" ? /* @__PURE__ */ jsxDEV(Button, {
153
+ className: "w-full justify-start",
154
+ variant: "ghost",
155
+ onPress: handleActivate,
156
+ disabled: isLoading,
157
+ children: [
158
+ /* @__PURE__ */ jsxDEV("span", {
159
+ className: "mr-2",
160
+ children: "\uD83D\uDD04"
161
+ }, undefined, false, undefined, this),
162
+ " Restore Project"
163
+ ]
164
+ }, undefined, true, undefined, this) : null,
165
+ /* @__PURE__ */ jsxDEV(Button, {
166
+ className: "w-full justify-start text-red-500 hover:text-red-600",
167
+ variant: "ghost",
168
+ onPress: () => setMode("delete"),
169
+ children: [
170
+ /* @__PURE__ */ jsxDEV("span", {
171
+ className: "mr-2",
172
+ children: "\uD83D\uDDD1️"
173
+ }, undefined, false, undefined, this),
174
+ " Delete Project"
175
+ ]
176
+ }, undefined, true, undefined, this),
177
+ /* @__PURE__ */ jsxDEV("div", {
178
+ className: "border-border border-t pt-3",
179
+ children: /* @__PURE__ */ jsxDEV(Button, {
180
+ className: "w-full",
181
+ variant: "outline",
182
+ onPress: handleClose,
183
+ children: "Close"
184
+ }, undefined, false, undefined, this)
185
+ }, undefined, false, undefined, this)
186
+ ]
187
+ }, undefined, true, undefined, this),
188
+ mode === "edit" && /* @__PURE__ */ jsxDEV("div", {
189
+ className: "space-y-4",
190
+ children: [
191
+ /* @__PURE__ */ jsxDEV("div", {
192
+ children: [
193
+ /* @__PURE__ */ jsxDEV("label", {
194
+ htmlFor: "edit-name",
195
+ className: "text-muted-foreground mb-1 block text-sm font-medium",
196
+ children: "Project Name *"
197
+ }, undefined, false, undefined, this),
198
+ /* @__PURE__ */ jsxDEV(Input, {
199
+ id: "edit-name",
200
+ value: name,
201
+ onChange: (e) => setName(e.target.value),
202
+ disabled: isLoading
203
+ }, undefined, false, undefined, this)
204
+ ]
205
+ }, undefined, true, undefined, this),
206
+ /* @__PURE__ */ jsxDEV("div", {
207
+ children: [
208
+ /* @__PURE__ */ jsxDEV("label", {
209
+ htmlFor: "edit-description",
210
+ className: "text-muted-foreground mb-1 block text-sm font-medium",
211
+ children: "Description"
212
+ }, undefined, false, undefined, this),
213
+ /* @__PURE__ */ jsxDEV("textarea", {
214
+ id: "edit-description",
215
+ value: description,
216
+ onChange: (e) => setDescription(e.target.value),
217
+ rows: 3,
218
+ disabled: isLoading,
219
+ className: "border-input bg-background focus:ring-ring w-full rounded-md border px-3 py-2 text-sm focus:ring-2 focus:outline-none disabled:opacity-50"
220
+ }, undefined, false, undefined, this)
221
+ ]
222
+ }, undefined, true, undefined, this),
223
+ error && /* @__PURE__ */ jsxDEV("div", {
224
+ className: "bg-destructive/10 text-destructive rounded-md p-3 text-sm",
225
+ children: error
226
+ }, undefined, false, undefined, this),
227
+ /* @__PURE__ */ jsxDEV("div", {
228
+ className: "flex justify-end gap-3 pt-2",
229
+ children: [
230
+ /* @__PURE__ */ jsxDEV(Button, {
231
+ variant: "ghost",
232
+ onPress: () => setMode("menu"),
233
+ disabled: isLoading,
234
+ children: "Back"
235
+ }, undefined, false, undefined, this),
236
+ /* @__PURE__ */ jsxDEV(Button, {
237
+ onPress: handleEdit,
238
+ disabled: isLoading,
239
+ children: isLoading ? "Saving..." : "Save Changes"
240
+ }, undefined, false, undefined, this)
241
+ ]
242
+ }, undefined, true, undefined, this)
243
+ ]
244
+ }, undefined, true, undefined, this),
245
+ mode === "archive" && /* @__PURE__ */ jsxDEV("div", {
246
+ className: "space-y-4",
247
+ children: [
248
+ /* @__PURE__ */ jsxDEV("p", {
249
+ className: "text-muted-foreground",
250
+ children: [
251
+ "Are you sure you want to archive",
252
+ " ",
253
+ /* @__PURE__ */ jsxDEV("span", {
254
+ className: "text-foreground font-medium",
255
+ children: project.name
256
+ }, undefined, false, undefined, this),
257
+ "?"
258
+ ]
259
+ }, undefined, true, undefined, this),
260
+ /* @__PURE__ */ jsxDEV("p", {
261
+ className: "text-muted-foreground text-sm",
262
+ children: "Archived projects can be restored later."
263
+ }, undefined, false, undefined, this),
264
+ error && /* @__PURE__ */ jsxDEV("div", {
265
+ className: "bg-destructive/10 text-destructive rounded-md p-3 text-sm",
266
+ children: error
267
+ }, undefined, false, undefined, this),
268
+ /* @__PURE__ */ jsxDEV("div", {
269
+ className: "flex justify-end gap-3 pt-2",
270
+ children: [
271
+ /* @__PURE__ */ jsxDEV(Button, {
272
+ variant: "ghost",
273
+ onPress: () => setMode("menu"),
274
+ disabled: isLoading,
275
+ children: "Cancel"
276
+ }, undefined, false, undefined, this),
277
+ /* @__PURE__ */ jsxDEV(Button, {
278
+ onPress: handleArchive,
279
+ disabled: isLoading,
280
+ children: isLoading ? "Archiving..." : "\uD83D\uDCE6 Archive"
281
+ }, undefined, false, undefined, this)
282
+ ]
283
+ }, undefined, true, undefined, this)
284
+ ]
285
+ }, undefined, true, undefined, this),
286
+ mode === "delete" && /* @__PURE__ */ jsxDEV("div", {
287
+ className: "space-y-4",
288
+ children: [
289
+ /* @__PURE__ */ jsxDEV("p", {
290
+ className: "text-muted-foreground",
291
+ children: [
292
+ "Are you sure you want to delete",
293
+ " ",
294
+ /* @__PURE__ */ jsxDEV("span", {
295
+ className: "text-foreground font-medium",
296
+ children: project.name
297
+ }, undefined, false, undefined, this),
298
+ "?"
299
+ ]
300
+ }, undefined, true, undefined, this),
301
+ /* @__PURE__ */ jsxDEV("p", {
302
+ className: "text-destructive text-sm",
303
+ children: "This action cannot be undone."
304
+ }, undefined, false, undefined, this),
305
+ error && /* @__PURE__ */ jsxDEV("div", {
306
+ className: "bg-destructive/10 text-destructive rounded-md p-3 text-sm",
307
+ children: error
308
+ }, undefined, false, undefined, this),
309
+ /* @__PURE__ */ jsxDEV("div", {
310
+ className: "flex justify-end gap-3 pt-2",
311
+ children: [
312
+ /* @__PURE__ */ jsxDEV(Button, {
313
+ variant: "ghost",
314
+ onPress: () => setMode("menu"),
315
+ disabled: isLoading,
316
+ children: "Cancel"
317
+ }, undefined, false, undefined, this),
318
+ /* @__PURE__ */ jsxDEV(Button, {
319
+ variant: "destructive",
320
+ onPress: handleDelete,
321
+ disabled: isLoading,
322
+ children: isLoading ? "Deleting..." : "\uD83D\uDDD1️ Delete"
323
+ }, undefined, false, undefined, this)
324
+ ]
325
+ }, undefined, true, undefined, this)
326
+ ]
327
+ }, undefined, true, undefined, this)
328
+ ]
329
+ }, undefined, true, undefined, this)
330
+ ]
331
+ }, undefined, true, undefined, this);
332
+ }
333
+ export {
334
+ ProjectActionsModal
335
+ };