@mmapp/react-compiler 0.1.0-alpha.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 (314) hide show
  1. package/README.md +107 -0
  2. package/compile-blueprint-chat.mjs +99 -0
  3. package/compile-blueprint-glass-console.mjs +98 -0
  4. package/compile-chat-defs.mjs +92 -0
  5. package/dist/babel/index.d.mts +3 -0
  6. package/dist/babel/index.d.ts +3 -0
  7. package/dist/babel/index.js +4851 -0
  8. package/dist/babel/index.mjs +7 -0
  9. package/dist/chunk-26U577GB.mjs +3465 -0
  10. package/dist/chunk-2FBDFAX6.mjs +2362 -0
  11. package/dist/chunk-2L4QSMXG.mjs +175 -0
  12. package/dist/chunk-2REDFOER.mjs +931 -0
  13. package/dist/chunk-46YKSHQR.mjs +175 -0
  14. package/dist/chunk-4XHK6FWL.mjs +2058 -0
  15. package/dist/chunk-5M7DKKBC.mjs +215 -0
  16. package/dist/chunk-5VNJ7C6N.mjs +154 -0
  17. package/dist/chunk-6CQOAAMV.mjs +1803 -0
  18. package/dist/chunk-6SEVAAVT.mjs +3516 -0
  19. package/dist/chunk-6YLR5ZDA.mjs +2829 -0
  20. package/dist/chunk-AOGY2GK6.mjs +3292 -0
  21. package/dist/chunk-AXXUXRNA.mjs +1434 -0
  22. package/dist/chunk-CHLVKMQW.mjs +175 -0
  23. package/dist/chunk-CKGOZAB7.mjs +939 -0
  24. package/dist/chunk-D34RAZUX.mjs +2223 -0
  25. package/dist/chunk-EQGA6A6D.mjs +121 -0
  26. package/dist/chunk-EY2CSXYA.mjs +822 -0
  27. package/dist/chunk-FIQ65CDR.mjs +925 -0
  28. package/dist/chunk-FOZXJFAR.mjs +186 -0
  29. package/dist/chunk-FX6URXWN.mjs +186 -0
  30. package/dist/chunk-G7SMOWOL.mjs +828 -0
  31. package/dist/chunk-GGB4G5YY.mjs +175 -0
  32. package/dist/chunk-HLRGCCIL.mjs +4839 -0
  33. package/dist/chunk-HOIUP6IF.mjs +690 -0
  34. package/dist/chunk-I3AU7GRD.mjs +120 -0
  35. package/dist/chunk-ILFGMUVD.mjs +1933 -0
  36. package/dist/chunk-IPTX5MJU.mjs +3223 -0
  37. package/dist/chunk-ITGUSH2Z.mjs +2783 -0
  38. package/dist/chunk-IXHBCAMF.mjs +3306 -0
  39. package/dist/chunk-J7TWJ3TM.mjs +2784 -0
  40. package/dist/chunk-JDPLDGVF.mjs +4810 -0
  41. package/dist/chunk-K53XP2DL.mjs +148 -0
  42. package/dist/chunk-K5HX2SVL.mjs +1902 -0
  43. package/dist/chunk-KFGYOOVS.mjs +214 -0
  44. package/dist/chunk-KFVVOS5N.mjs +925 -0
  45. package/dist/chunk-L2OZ4CDV.mjs +113 -0
  46. package/dist/chunk-MIZV3TAN.mjs +3293 -0
  47. package/dist/chunk-NKKLQE5V.mjs +148 -0
  48. package/dist/chunk-NOW23XFZ.mjs +186 -0
  49. package/dist/chunk-NRXQKQ74.mjs +148 -0
  50. package/dist/chunk-OWI6XWCD.mjs +3375 -0
  51. package/dist/chunk-PRUMNNDI.mjs +3192 -0
  52. package/dist/chunk-QTBD5B3F.mjs +148 -0
  53. package/dist/chunk-SKSDPPNT.mjs +3788 -0
  54. package/dist/chunk-SP2YUS33.mjs +186 -0
  55. package/dist/chunk-SU4E6E7B.mjs +3153 -0
  56. package/dist/chunk-SYUUKW5A.mjs +3379 -0
  57. package/dist/chunk-UL2XZEMA.mjs +3128 -0
  58. package/dist/chunk-XMWUHQVV.mjs +939 -0
  59. package/dist/chunk-XZNEDRGN.mjs +3876 -0
  60. package/dist/chunk-Y6FXYEAI.mjs +10 -0
  61. package/dist/chunk-YFS6JMYO.mjs +3342 -0
  62. package/dist/chunk-Z6AIQ4KL.mjs +113 -0
  63. package/dist/cli/index.d.mts +1 -0
  64. package/dist/cli/index.d.ts +1 -0
  65. package/dist/cli/index.js +11585 -0
  66. package/dist/cli/index.mjs +701 -0
  67. package/dist/codemod/cli.d.mts +1 -0
  68. package/dist/codemod/cli.d.ts +1 -0
  69. package/dist/codemod/cli.js +1104 -0
  70. package/dist/codemod/cli.mjs +157 -0
  71. package/dist/codemod/index.d.mts +148 -0
  72. package/dist/codemod/index.d.ts +148 -0
  73. package/dist/codemod/index.js +981 -0
  74. package/dist/codemod/index.mjs +25 -0
  75. package/dist/dev-server-Bs_sz2DG.d.mts +111 -0
  76. package/dist/dev-server-Bs_sz2DG.d.ts +111 -0
  77. package/dist/dev-server-CjoufJ-u.d.mts +109 -0
  78. package/dist/dev-server-CjoufJ-u.d.ts +109 -0
  79. package/dist/dev-server.d.mts +3 -0
  80. package/dist/dev-server.d.ts +3 -0
  81. package/dist/dev-server.js +7603 -0
  82. package/dist/dev-server.mjs +11 -0
  83. package/dist/envelope-DD7v0v6E.d.mts +265 -0
  84. package/dist/envelope-DD7v0v6E.d.ts +265 -0
  85. package/dist/envelope-vCVjrHlo.d.mts +265 -0
  86. package/dist/envelope-vCVjrHlo.d.ts +265 -0
  87. package/dist/envelope.d.mts +2 -0
  88. package/dist/envelope.d.ts +2 -0
  89. package/dist/envelope.js +5184 -0
  90. package/dist/envelope.mjs +9 -0
  91. package/dist/index-B5gSgvnd.d.mts +44 -0
  92. package/dist/index-B5gSgvnd.d.ts +44 -0
  93. package/dist/index-Bs0MnR54.d.mts +103 -0
  94. package/dist/index-Bs0MnR54.d.ts +103 -0
  95. package/dist/index-DR0nNc_f.d.mts +101 -0
  96. package/dist/index-DR0nNc_f.d.ts +101 -0
  97. package/dist/index-revho_gS.d.mts +104 -0
  98. package/dist/index-revho_gS.d.ts +104 -0
  99. package/dist/index.d.mts +1099 -0
  100. package/dist/index.d.ts +1099 -0
  101. package/dist/index.js +10162 -0
  102. package/dist/index.mjs +372 -0
  103. package/dist/init-IXEE2RCF.mjs +340 -0
  104. package/dist/project-compiler-EGJUTAJU.mjs +10 -0
  105. package/dist/project-compiler-VFR6CSDX.mjs +10 -0
  106. package/dist/project-decompiler-5GY2KSG4.mjs +7 -0
  107. package/dist/pull-A2QUHW4K.mjs +109 -0
  108. package/dist/pull-JBEQWVPE.mjs +109 -0
  109. package/dist/testing/index.d.mts +211 -0
  110. package/dist/testing/index.d.ts +211 -0
  111. package/dist/testing/index.js +5106 -0
  112. package/dist/testing/index.mjs +247 -0
  113. package/dist/vite/index.d.mts +59 -0
  114. package/dist/vite/index.d.ts +59 -0
  115. package/dist/vite/index.js +5023 -0
  116. package/dist/vite/index.mjs +8 -0
  117. package/examples/README.md +72 -0
  118. package/examples/authentication/main.workflow.tsx +139 -0
  119. package/examples/authentication/mm.config.ts +22 -0
  120. package/examples/authentication/models/auth.ts +45 -0
  121. package/examples/authentication/pages/LoginPage.tsx +79 -0
  122. package/examples/authentication/pages/SignupPage.tsx +87 -0
  123. package/examples/counter.workflow.tsx +65 -0
  124. package/examples/dashboard.workflow.tsx +419 -0
  125. package/examples/invoice-approval/actions/invoice.server.ts +72 -0
  126. package/examples/invoice-approval/main.workflow.tsx +168 -0
  127. package/examples/invoice-approval/mm.config.ts +18 -0
  128. package/examples/invoice-approval/models/invoice.ts +46 -0
  129. package/examples/invoice-approval/pages/InvoiceDetailPage.tsx +175 -0
  130. package/examples/invoice-approval/pages/InvoiceFormPage.tsx +198 -0
  131. package/examples/invoice-approval/pages/InvoiceListPage.tsx +141 -0
  132. package/examples/todo-app.workflow.tsx +131 -0
  133. package/examples/uber-app/actions/matching.server.ts +177 -0
  134. package/examples/uber-app/actions/notifications.server.ts +176 -0
  135. package/examples/uber-app/actions/payments.server.ts +184 -0
  136. package/examples/uber-app/actions/pricing.server.ts +176 -0
  137. package/examples/uber-app/app/admin/analytics.tsx +102 -0
  138. package/examples/uber-app/app/admin/fleet.tsx +102 -0
  139. package/examples/uber-app/app/admin/surge-pricing.tsx +95 -0
  140. package/examples/uber-app/app/driver/dashboard.tsx +87 -0
  141. package/examples/uber-app/app/driver/earnings.tsx +101 -0
  142. package/examples/uber-app/app/driver/navigation.tsx +94 -0
  143. package/examples/uber-app/app/driver/ride-acceptance.tsx +103 -0
  144. package/examples/uber-app/app/rider/home.tsx +109 -0
  145. package/examples/uber-app/app/rider/payment-methods.tsx +134 -0
  146. package/examples/uber-app/app/rider/ride-history.tsx +90 -0
  147. package/examples/uber-app/app/rider/ride-tracking.tsx +108 -0
  148. package/examples/uber-app/components/DriverCard.tsx +176 -0
  149. package/examples/uber-app/components/MapView.tsx +216 -0
  150. package/examples/uber-app/components/RatingStars.tsx +227 -0
  151. package/examples/uber-app/components/RideCard.tsx +167 -0
  152. package/examples/uber-app/mm.config.ts +30 -0
  153. package/examples/uber-app/models/location.model.ts +70 -0
  154. package/examples/uber-app/models/payment.model.ts +87 -0
  155. package/examples/uber-app/models/rating.model.ts +54 -0
  156. package/examples/uber-app/models/ride.model.ts +118 -0
  157. package/examples/uber-app/models/user.model.ts +66 -0
  158. package/examples/uber-app/models/vehicle.model.ts +63 -0
  159. package/examples/uber-app/tests/payment.test.tsx +129 -0
  160. package/examples/uber-app/tests/ride-flow.test.tsx +123 -0
  161. package/examples/uber-app/workflows/dispute-resolution.workflow.tsx +205 -0
  162. package/examples/uber-app/workflows/driver-onboarding.workflow.tsx +227 -0
  163. package/examples/uber-app/workflows/payment-processing.workflow.tsx +223 -0
  164. package/examples/uber-app/workflows/ride-request.workflow.tsx +194 -0
  165. package/package.json +77 -0
  166. package/package.json.backup +86 -0
  167. package/scripts/decompile.ts +226 -0
  168. package/scripts/seed-auth.ts +267 -0
  169. package/scripts/seed-uber.ts +248 -0
  170. package/scripts/validate-uber.ts +119 -0
  171. package/seed-blueprint-chat.mjs +444 -0
  172. package/seed-blueprint-glass-console.mjs +445 -0
  173. package/seed-compiled.mjs +318 -0
  174. package/src/RoundTripValidator.ts +400 -0
  175. package/src/__tests__/atom-rendering-coverage.test.ts +680 -0
  176. package/src/__tests__/auth-module-compilation.test.ts +247 -0
  177. package/src/__tests__/auth-template-compilation.test.ts +589 -0
  178. package/src/__tests__/change-extractor.test.ts +142 -0
  179. package/src/__tests__/cli-pull.test.ts +73 -0
  180. package/src/__tests__/cli-test.test.ts +72 -0
  181. package/src/__tests__/component-extractor.test.ts +331 -0
  182. package/src/__tests__/context-extractor.test.ts +145 -0
  183. package/src/__tests__/decompiler.test.ts +718 -0
  184. package/src/__tests__/define-blueprint.test.ts +133 -0
  185. package/src/__tests__/definition-validator.test.ts +519 -0
  186. package/src/__tests__/during-extractor.test.ts +152 -0
  187. package/src/__tests__/effect-extractor.test.ts +107 -0
  188. package/src/__tests__/event-emission.test.ts +127 -0
  189. package/src/__tests__/examples.test.ts +236 -0
  190. package/src/__tests__/full-blueprint-coverage.test.ts +1221 -0
  191. package/src/__tests__/golden-suite.test.ts +403 -0
  192. package/src/__tests__/grammar-island-extractor.test.ts +289 -0
  193. package/src/__tests__/instance-key.test.ts +82 -0
  194. package/src/__tests__/ir-migration.test.ts +255 -0
  195. package/src/__tests__/lock-file.test.ts +117 -0
  196. package/src/__tests__/model-extractor.test.ts +195 -0
  197. package/src/__tests__/model-field-acl.test.ts +237 -0
  198. package/src/__tests__/model-hooks.test.ts +130 -0
  199. package/src/__tests__/model-ref-resolution.test.ts +268 -0
  200. package/src/__tests__/model-roundtrip.test.ts +502 -0
  201. package/src/__tests__/model-runtime.test.ts +112 -0
  202. package/src/__tests__/model-transitions.test.ts +183 -0
  203. package/src/__tests__/nrt-action-trace.test.ts +391 -0
  204. package/src/__tests__/pipeline-hardening.test.ts +413 -0
  205. package/src/__tests__/project-compiler.test.ts +546 -0
  206. package/src/__tests__/project-decompiler.test.ts +343 -0
  207. package/src/__tests__/query-compilation.test.ts +145 -0
  208. package/src/__tests__/round-trip/PLAN.md +158 -0
  209. package/src/__tests__/round-trip/README.md +52 -0
  210. package/src/__tests__/round-trip/RESULTS.md +86 -0
  211. package/src/__tests__/round-trip/fixtures/data-heavy/main.workflow.tsx +55 -0
  212. package/src/__tests__/round-trip/fixtures/data-heavy/mm.config.ts +11 -0
  213. package/src/__tests__/round-trip/fixtures/data-heavy/models/contact.ts +54 -0
  214. package/src/__tests__/round-trip/fixtures/full-workflow/main.workflow.tsx +79 -0
  215. package/src/__tests__/round-trip/fixtures/full-workflow/mm.config.ts +12 -0
  216. package/src/__tests__/round-trip/fixtures/full-workflow/models/order.ts +50 -0
  217. package/src/__tests__/round-trip/fixtures/simple-crud/main.workflow.tsx +25 -0
  218. package/src/__tests__/round-trip/fixtures/simple-crud/mm.config.ts +11 -0
  219. package/src/__tests__/round-trip/fixtures/simple-crud/models/task.ts +32 -0
  220. package/src/__tests__/round-trip/fixtures/view-heavy/main.workflow.tsx +79 -0
  221. package/src/__tests__/round-trip/fixtures/view-heavy/mm.config.ts +10 -0
  222. package/src/__tests__/round-trip/round-trip.test.ts +2598 -0
  223. package/src/__tests__/round-trip-ir.test.ts +300 -0
  224. package/src/__tests__/round-trip.test.ts +1212 -0
  225. package/src/__tests__/route-merging.test.ts +372 -0
  226. package/src/__tests__/router-composition.test.ts +489 -0
  227. package/src/__tests__/router-extractor.test.ts +176 -0
  228. package/src/__tests__/server-action-extractor.test.ts +128 -0
  229. package/src/__tests__/smart-type-inference.test.ts +365 -0
  230. package/src/__tests__/source-envelope.test.ts +284 -0
  231. package/src/__tests__/source-fidelity.test.ts +516 -0
  232. package/src/__tests__/state-extractor.test.ts +115 -0
  233. package/src/__tests__/strict-mode.test.ts +227 -0
  234. package/src/__tests__/transition-effect-extractor.test.ts +119 -0
  235. package/src/__tests__/transition-extractor.test.ts +68 -0
  236. package/src/__tests__/ts-to-expression.test.ts +462 -0
  237. package/src/__tests__/type-generator.test.ts +201 -0
  238. package/src/__tests__/uber-validation.test.ts +502 -0
  239. package/src/action-compiler.ts +361 -0
  240. package/src/babel/emitters/experience-transform.ts +199 -0
  241. package/src/babel/emitters/ir-to-tsx-emitter.ts +110 -0
  242. package/src/babel/emitters/pure-form-emitter.ts +1023 -0
  243. package/src/babel/emitters/runtime-glue-emitter.ts +39 -0
  244. package/src/babel/extractors/change-extractor.ts +199 -0
  245. package/src/babel/extractors/component-extractor.ts +907 -0
  246. package/src/babel/extractors/computed-extractor.ts +262 -0
  247. package/src/babel/extractors/context-extractor.ts +277 -0
  248. package/src/babel/extractors/during-extractor.ts +295 -0
  249. package/src/babel/extractors/effect-extractor.ts +340 -0
  250. package/src/babel/extractors/event-extractor.ts +235 -0
  251. package/src/babel/extractors/grammar-island-extractor.ts +302 -0
  252. package/src/babel/extractors/model-extractor.ts +1018 -0
  253. package/src/babel/extractors/router-extractor.ts +303 -0
  254. package/src/babel/extractors/server-action-extractor.ts +173 -0
  255. package/src/babel/extractors/server-action-hook-extractor.ts +72 -0
  256. package/src/babel/extractors/server-state-extractor.ts +88 -0
  257. package/src/babel/extractors/state-extractor.ts +214 -0
  258. package/src/babel/extractors/transition-effect-extractor.ts +176 -0
  259. package/src/babel/extractors/transition-extractor.ts +143 -0
  260. package/src/babel/index.ts +24 -0
  261. package/src/babel/transpilers/ts-to-expression.ts +674 -0
  262. package/src/babel/visitor.ts +807 -0
  263. package/src/cli/auth.ts +255 -0
  264. package/src/cli/build.ts +288 -0
  265. package/src/cli/deploy.ts +206 -0
  266. package/src/cli/index.ts +328 -0
  267. package/src/cli/init.ts +388 -0
  268. package/src/cli/installer.ts +261 -0
  269. package/src/cli/lock-file.ts +94 -0
  270. package/src/cli/mmrc.ts +22 -0
  271. package/src/cli/pull.ts +172 -0
  272. package/src/cli/registry-client.ts +175 -0
  273. package/src/cli/test.ts +397 -0
  274. package/src/cli/type-generator.ts +243 -0
  275. package/src/codemod/__tests__/forward.test.ts +239 -0
  276. package/src/codemod/__tests__/reverse.test.ts +145 -0
  277. package/src/codemod/__tests__/round-trip.test.ts +137 -0
  278. package/src/codemod/annotation.ts +97 -0
  279. package/src/codemod/classify.ts +197 -0
  280. package/src/codemod/cli.ts +207 -0
  281. package/src/codemod/control-flow.ts +409 -0
  282. package/src/codemod/forward.ts +244 -0
  283. package/src/codemod/import-manager.ts +171 -0
  284. package/src/codemod/index.ts +120 -0
  285. package/src/codemod/reverse.ts +197 -0
  286. package/src/codemod/rules.ts +174 -0
  287. package/src/codemod/state-transform.ts +126 -0
  288. package/src/decompiler/ast-builder.ts +538 -0
  289. package/src/decompiler/config-generator.ts +151 -0
  290. package/src/decompiler/index.ts +315 -0
  291. package/src/decompiler/project-decompiler.ts +1776 -0
  292. package/src/decompiler/project.ts +862 -0
  293. package/src/decompiler/split-strategy.ts +140 -0
  294. package/src/decompiler/state-emitter.ts +1053 -0
  295. package/src/decompiler/sx-emitter.ts +318 -0
  296. package/src/decompiler/workspace-hydrator.ts +189 -0
  297. package/src/dev-server.ts +238 -0
  298. package/src/envelope/fs-tree.ts +217 -0
  299. package/src/envelope/source-envelope.ts +264 -0
  300. package/src/envelope.ts +315 -0
  301. package/src/incremental-compiler.ts +401 -0
  302. package/src/index.ts +99 -0
  303. package/src/model-compiler.ts +277 -0
  304. package/src/project-compiler.ts +1629 -0
  305. package/src/route-extractor.ts +333 -0
  306. package/src/testing/index.ts +32 -0
  307. package/src/testing/snapshot.ts +252 -0
  308. package/src/testing/test-utils.ts +226 -0
  309. package/src/types.ts +68 -0
  310. package/src/vite/index.ts +288 -0
  311. package/test-compile.mjs +142 -0
  312. package/tsconfig.json +25 -0
  313. package/tsup.config.ts +23 -0
  314. package/vitest.config.ts +9 -0
@@ -0,0 +1,176 @@
1
+ /**
2
+ * DriverCard — Driver info display with vehicle details and real-time status
3
+ */
4
+
5
+ import React from 'react';
6
+ import { Stack, Row, Text, Badge, Icon, Image } from '@mindmatrix/react';
7
+
8
+ interface DriverCardProps {
9
+ name: string;
10
+ avatarUrl?: string;
11
+ rating: number;
12
+ totalTrips: number;
13
+ vehicleMake: string;
14
+ vehicleModel: string;
15
+ vehicleColor: string;
16
+ licensePlate: string;
17
+ vehicleType: string;
18
+ etaMinutes?: number;
19
+ isOnline?: boolean;
20
+ phone?: string;
21
+ onCall?: () => void;
22
+ onMessage?: () => void;
23
+ onCancel?: () => void;
24
+ compact?: boolean;
25
+ }
26
+
27
+ export function DriverCard({
28
+ name,
29
+ avatarUrl,
30
+ rating,
31
+ totalTrips,
32
+ vehicleMake,
33
+ vehicleModel,
34
+ vehicleColor,
35
+ licensePlate,
36
+ vehicleType,
37
+ etaMinutes,
38
+ isOnline,
39
+ phone,
40
+ onCall,
41
+ onMessage,
42
+ onCancel,
43
+ compact = false,
44
+ }: DriverCardProps) {
45
+ if (compact) {
46
+ return (
47
+ <Row sx={{ bg: '#1a202c', borderRadius: 'lg', p: 3, gap: 3, alignItems: 'center', border: '1px solid #2d3748' }}>
48
+ <Stack sx={{ position: 'relative' }}>
49
+ <Stack sx={{ width: 12, height: 12, borderRadius: 'full', bg: '#2d3748', alignItems: 'center', justifyContent: 'center', overflow: 'hidden' }}>
50
+ {avatarUrl
51
+ ? <Image src={avatarUrl} sx={{ width: '100%', height: '100%', objectFit: 'cover' }} />
52
+ : <Icon name="user" sx={{ color: '#a0aec0', fontSize: 'xl' }} />
53
+ }
54
+ </Stack>
55
+ {isOnline !== undefined && (
56
+ <Stack sx={{
57
+ position: 'absolute', bottom: 0, right: 0,
58
+ width: 3, height: 3, borderRadius: 'full',
59
+ bg: isOnline ? '#48bb78' : '#718096',
60
+ border: '2px solid #1a202c',
61
+ }} />
62
+ )}
63
+ </Stack>
64
+ <Stack sx={{ flex: 1 }}>
65
+ <Text sx={{ color: '#e2e8f0', fontWeight: 'bold', fontSize: 'sm' }}>{name}</Text>
66
+ <Row sx={{ gap: 1, alignItems: 'center' }}>
67
+ <Icon name="star" sx={{ color: '#ecc94b', fontSize: 'xs' }} />
68
+ <Text sx={{ color: '#a0aec0', fontSize: 'xs' }}>{rating.toFixed(1)}</Text>
69
+ <Text sx={{ color: '#4a5568', fontSize: 'xs' }}>|</Text>
70
+ <Text sx={{ color: '#a0aec0', fontSize: 'xs' }}>{vehicleColor} {vehicleMake} {vehicleModel}</Text>
71
+ </Row>
72
+ </Stack>
73
+ {etaMinutes !== undefined && (
74
+ <Stack sx={{ alignItems: 'center' }}>
75
+ <Text sx={{ color: '#4299e1', fontWeight: 'bold', fontSize: 'lg' }}>{etaMinutes}</Text>
76
+ <Text sx={{ color: '#718096', fontSize: 'xs' }}>min</Text>
77
+ </Stack>
78
+ )}
79
+ </Row>
80
+ );
81
+ }
82
+
83
+ return (
84
+ <Stack sx={{ bg: '#1a202c', borderRadius: 'xl', p: 5, border: '1px solid #2d3748' }}>
85
+ {/* Avatar + name + rating */}
86
+ <Row sx={{ gap: 4, alignItems: 'center', mb: 4 }}>
87
+ <Stack sx={{ position: 'relative' }}>
88
+ <Stack sx={{ width: 16, height: 16, borderRadius: 'full', bg: '#2d3748', alignItems: 'center', justifyContent: 'center', overflow: 'hidden' }}>
89
+ {avatarUrl
90
+ ? <Image src={avatarUrl} sx={{ width: '100%', height: '100%', objectFit: 'cover' }} />
91
+ : <Icon name="user" sx={{ color: '#a0aec0', fontSize: '2xl' }} />
92
+ }
93
+ </Stack>
94
+ {isOnline !== undefined && (
95
+ <Stack sx={{
96
+ position: 'absolute', bottom: 1, right: 1,
97
+ width: 4, height: 4, borderRadius: 'full',
98
+ bg: isOnline ? '#48bb78' : '#718096',
99
+ border: '2px solid #1a202c',
100
+ }} />
101
+ )}
102
+ </Stack>
103
+ <Stack sx={{ flex: 1 }}>
104
+ <Text sx={{ color: '#e2e8f0', fontWeight: 'bold', fontSize: 'xl' }}>{name}</Text>
105
+ <Row sx={{ gap: 3, mt: 1 }}>
106
+ <Row sx={{ gap: 1, alignItems: 'center' }}>
107
+ <Icon name="star" sx={{ color: '#ecc94b' }} />
108
+ <Text sx={{ color: '#e2e8f0', fontWeight: 600 }}>{rating.toFixed(1)}</Text>
109
+ </Row>
110
+ <Text sx={{ color: '#718096' }}>|</Text>
111
+ <Text sx={{ color: '#a0aec0' }}>{totalTrips.toLocaleString()} trips</Text>
112
+ </Row>
113
+ </Stack>
114
+ {etaMinutes !== undefined && (
115
+ <Stack sx={{ alignItems: 'center', bg: '#2d3748', px: 4, py: 2, borderRadius: 'lg' }}>
116
+ <Text sx={{ color: '#4299e1', fontWeight: 'bold', fontSize: '2xl' }}>{etaMinutes}</Text>
117
+ <Text sx={{ color: '#718096', fontSize: 'xs' }}>min away</Text>
118
+ </Stack>
119
+ )}
120
+ </Row>
121
+
122
+ {/* Vehicle info */}
123
+ <Stack sx={{ bg: '#2d3748', borderRadius: 'lg', p: 3, mb: 4 }}>
124
+ <Row sx={{ justifyContent: 'space-between', alignItems: 'center' }}>
125
+ <Stack>
126
+ <Text sx={{ color: '#a0aec0', fontSize: 'xs', mb: 1 }}>Vehicle</Text>
127
+ <Text sx={{ color: '#e2e8f0', fontWeight: 500 }}>
128
+ {vehicleColor} {vehicleMake} {vehicleModel}
129
+ </Text>
130
+ </Stack>
131
+ <Stack sx={{ alignItems: 'flex-end' }}>
132
+ <Badge sx={{ bg: '#4a5568', color: '#e2e8f0', px: 3, py: 1, borderRadius: 'md', fontSize: 'sm', fontWeight: 'bold', letterSpacing: '1px' }}>
133
+ {licensePlate}
134
+ </Badge>
135
+ <Badge sx={{ mt: 1, bg: '#2d3748', border: '1px solid #4a5568', color: '#a0aec0', px: 2, py: 0.5, borderRadius: 'md', fontSize: 'xs', textTransform: 'capitalize' }}>
136
+ {vehicleType}
137
+ </Badge>
138
+ </Stack>
139
+ </Row>
140
+ </Stack>
141
+
142
+ {/* Action buttons */}
143
+ <Row sx={{ gap: 3 }}>
144
+ {onCall && (
145
+ <Stack
146
+ onClick={onCall}
147
+ sx={{ flex: 1, alignItems: 'center', bg: '#2d3748', py: 3, borderRadius: 'lg', cursor: 'pointer', _hover: { bg: '#4a5568' } }}
148
+ >
149
+ <Icon name="phone" sx={{ color: '#48bb78', mb: 1 }} />
150
+ <Text sx={{ color: '#a0aec0', fontSize: 'xs' }}>Call</Text>
151
+ </Stack>
152
+ )}
153
+ {onMessage && (
154
+ <Stack
155
+ onClick={onMessage}
156
+ sx={{ flex: 1, alignItems: 'center', bg: '#2d3748', py: 3, borderRadius: 'lg', cursor: 'pointer', _hover: { bg: '#4a5568' } }}
157
+ >
158
+ <Icon name="message" sx={{ color: '#4299e1', mb: 1 }} />
159
+ <Text sx={{ color: '#a0aec0', fontSize: 'xs' }}>Message</Text>
160
+ </Stack>
161
+ )}
162
+ {onCancel && (
163
+ <Stack
164
+ onClick={onCancel}
165
+ sx={{ flex: 1, alignItems: 'center', bg: '#2d3748', py: 3, borderRadius: 'lg', cursor: 'pointer', _hover: { bg: '#4a5568' } }}
166
+ >
167
+ <Icon name="x" sx={{ color: '#f56565', mb: 1 }} />
168
+ <Text sx={{ color: '#a0aec0', fontSize: 'xs' }}>Cancel</Text>
169
+ </Stack>
170
+ )}
171
+ </Row>
172
+ </Stack>
173
+ );
174
+ }
175
+
176
+ export default DriverCard;
@@ -0,0 +1,216 @@
1
+ /**
2
+ * MapView — Interactive map component with markers, routes, and live tracking
3
+ * Uses useMapView + useGeolocation from @mindmatrix/react
4
+ */
5
+
6
+ import React, { useState } from 'react';
7
+ import { useMapView, useGeolocation, useOnChange } from '@mindmatrix/react';
8
+ import { Stack, Row, Text, Badge, Button, Icon } from '@mindmatrix/react';
9
+
10
+ interface MapMarkerConfig {
11
+ id: string;
12
+ position: { lat: number; lng: number };
13
+ label?: string;
14
+ icon?: 'pickup' | 'dropoff' | 'driver' | 'surge';
15
+ }
16
+
17
+ interface MapViewProps {
18
+ pickupLocation?: { lat: number; lng: number; address: string };
19
+ dropoffLocation?: { lat: number; lng: number; address: string };
20
+ driverLocation?: { lat: number; lng: number; heading: number };
21
+ routePolyline?: string;
22
+ surgeZones?: Array<{ lat: number; lng: number; multiplier: number; radius: number }>;
23
+ showTraffic?: boolean;
24
+ trackUser?: boolean;
25
+ onLocationSelect?: (lat: number, lng: number, address: string) => void;
26
+ onMapReady?: () => void;
27
+ style?: Record<string, unknown>;
28
+ }
29
+
30
+ export function MapView({
31
+ pickupLocation,
32
+ dropoffLocation,
33
+ driverLocation,
34
+ routePolyline,
35
+ surgeZones = [],
36
+ showTraffic = false,
37
+ trackUser = false,
38
+ onLocationSelect,
39
+ style,
40
+ }: MapViewProps) {
41
+ const [mapMode, setMapMode] = useState<'standard' | 'satellite' | 'terrain'>('standard');
42
+
43
+ const map = useMapView({
44
+ initialCenter: pickupLocation || { lat: 40.7128, lng: -74.006 },
45
+ initialZoom: 14,
46
+ minZoom: 10,
47
+ maxZoom: 20,
48
+ });
49
+
50
+ const geo = useGeolocation({
51
+ enableHighAccuracy: true,
52
+ watch: trackUser,
53
+ enabled: trackUser,
54
+ });
55
+
56
+ // Update markers when locations change
57
+ useOnChange('pickupLocation', () => {
58
+ if (pickupLocation) {
59
+ map.addMarker({
60
+ id: 'pickup',
61
+ position: pickupLocation,
62
+ label: 'Pickup',
63
+ icon: 'pickup-pin',
64
+ });
65
+ }
66
+ });
67
+
68
+ useOnChange('dropoffLocation', () => {
69
+ if (dropoffLocation) {
70
+ map.addMarker({
71
+ id: 'dropoff',
72
+ position: dropoffLocation,
73
+ label: 'Drop-off',
74
+ icon: 'dropoff-pin',
75
+ });
76
+ // Fit both markers in view
77
+ if (pickupLocation) {
78
+ map.fitBounds([pickupLocation, dropoffLocation], 60);
79
+ }
80
+ }
81
+ });
82
+
83
+ useOnChange('driverLocation', () => {
84
+ if (driverLocation) {
85
+ map.updateMarker('driver', driverLocation);
86
+ }
87
+ });
88
+
89
+ const distanceText = pickupLocation && dropoffLocation
90
+ ? `${(map.distance(pickupLocation, dropoffLocation) / 1000).toFixed(1)} km`
91
+ : '';
92
+
93
+ return (
94
+ <Stack sx={{ position: 'relative', width: '100%', height: '100%', bg: '#1a1a2e', borderRadius: 'lg', overflow: 'hidden', ...style }}>
95
+ {/* Map canvas placeholder — in production, renders Mapbox/Google Maps */}
96
+ <Stack sx={{ flex: 1, alignItems: 'center', justifyContent: 'center', bg: 'linear-gradient(135deg, #1a1a2e 0%, #16213e 100%)' }}>
97
+ <Icon name="map" sx={{ fontSize: '64px', color: '#4a5568', mb: 2 }} />
98
+ <Text sx={{ color: '#718096', fontSize: 'sm' }}>Map View</Text>
99
+ <Text sx={{ color: '#4a5568', fontSize: 'xs' }}>
100
+ Center: {map.center.lat.toFixed(4)}, {map.center.lng.toFixed(4)} | Zoom: {map.zoom}
101
+ </Text>
102
+ {distanceText && (
103
+ <Badge sx={{ mt: 2, bg: '#2d3748', color: '#e2e8f0', px: 3, py: 1 }}>
104
+ {distanceText} straight-line
105
+ </Badge>
106
+ )}
107
+ </Stack>
108
+
109
+ {/* Pickup marker overlay */}
110
+ {pickupLocation && (
111
+ <Stack sx={{ position: 'absolute', top: '30%', left: '40%', alignItems: 'center' }}>
112
+ <Stack sx={{ width: 12, height: 12, borderRadius: 'full', bg: '#48bb78', border: '3px solid white', boxShadow: 'lg' }} />
113
+ <Text sx={{ fontSize: 'xs', color: 'white', bg: 'rgba(0,0,0,0.7)', px: 2, py: 0.5, borderRadius: 'sm', mt: 1 }}>
114
+ {pickupLocation.address || 'Pickup'}
115
+ </Text>
116
+ </Stack>
117
+ )}
118
+
119
+ {/* Dropoff marker overlay */}
120
+ {dropoffLocation && (
121
+ <Stack sx={{ position: 'absolute', top: '60%', right: '30%', alignItems: 'center' }}>
122
+ <Stack sx={{ width: 12, height: 12, borderRadius: 'full', bg: '#f56565', border: '3px solid white', boxShadow: 'lg' }} />
123
+ <Text sx={{ fontSize: 'xs', color: 'white', bg: 'rgba(0,0,0,0.7)', px: 2, py: 0.5, borderRadius: 'sm', mt: 1 }}>
124
+ {dropoffLocation.address || 'Drop-off'}
125
+ </Text>
126
+ </Stack>
127
+ )}
128
+
129
+ {/* Driver marker overlay */}
130
+ {driverLocation && (
131
+ <Stack sx={{
132
+ position: 'absolute', top: '45%', left: '35%',
133
+ transform: `rotate(${driverLocation.heading}deg)`,
134
+ }}>
135
+ <Stack sx={{ width: 10, height: 10, borderRadius: 'full', bg: '#4299e1', border: '2px solid white', boxShadow: 'lg' }}>
136
+ <Icon name="car" sx={{ color: 'white', fontSize: 'xs' }} />
137
+ </Stack>
138
+ </Stack>
139
+ )}
140
+
141
+ {/* Surge zone overlays */}
142
+ {surgeZones.map((zone, i) => (
143
+ <Stack
144
+ key={`surge-${i}`}
145
+ sx={{
146
+ position: 'absolute',
147
+ width: `${zone.radius / 50}px`,
148
+ height: `${zone.radius / 50}px`,
149
+ borderRadius: 'full',
150
+ bg: 'rgba(255, 165, 0, 0.15)',
151
+ border: '2px solid rgba(255, 165, 0, 0.4)',
152
+ top: `${30 + i * 15}%`,
153
+ left: `${20 + i * 20}%`,
154
+ alignItems: 'center',
155
+ justifyContent: 'center',
156
+ }}
157
+ >
158
+ <Text sx={{ fontSize: 'xs', fontWeight: 'bold', color: '#ed8936' }}>
159
+ {zone.multiplier}x
160
+ </Text>
161
+ </Stack>
162
+ ))}
163
+
164
+ {/* Controls overlay */}
165
+ <Row sx={{ position: 'absolute', top: 3, right: 3, gap: 1 }}>
166
+ <Button
167
+ onClick={() => geo.getCurrentPosition()}
168
+ sx={{ bg: 'rgba(0,0,0,0.6)', color: 'white', p: 2, borderRadius: 'full', fontSize: 'sm' }}
169
+ >
170
+ <Icon name="crosshair" />
171
+ </Button>
172
+ <Button
173
+ onClick={() => map.setZoom(map.zoom + 1)}
174
+ sx={{ bg: 'rgba(0,0,0,0.6)', color: 'white', p: 2, borderRadius: 'full', fontSize: 'sm' }}
175
+ >
176
+ +
177
+ </Button>
178
+ <Button
179
+ onClick={() => map.setZoom(map.zoom - 1)}
180
+ sx={{ bg: 'rgba(0,0,0,0.6)', color: 'white', p: 2, borderRadius: 'full', fontSize: 'sm' }}
181
+ >
182
+ -
183
+ </Button>
184
+ </Row>
185
+
186
+ {/* Map mode toggle */}
187
+ <Row sx={{ position: 'absolute', bottom: 3, left: 3, gap: 1 }}>
188
+ {(['standard', 'satellite', 'terrain'] as const).map((mode) => (
189
+ <Button
190
+ key={mode}
191
+ onClick={() => setMapMode(mode)}
192
+ sx={{
193
+ bg: mapMode === mode ? '#4299e1' : 'rgba(0,0,0,0.6)',
194
+ color: 'white',
195
+ px: 2,
196
+ py: 1,
197
+ borderRadius: 'md',
198
+ fontSize: 'xs',
199
+ textTransform: 'capitalize',
200
+ }}
201
+ >
202
+ {mode}
203
+ </Button>
204
+ ))}
205
+ </Row>
206
+
207
+ {showTraffic && (
208
+ <Badge sx={{ position: 'absolute', bottom: 3, right: 3, bg: '#ed8936', color: 'white', fontSize: 'xs' }}>
209
+ Traffic ON
210
+ </Badge>
211
+ )}
212
+ </Stack>
213
+ );
214
+ }
215
+
216
+ export default MapView;
@@ -0,0 +1,227 @@
1
+ /**
2
+ * RatingStars — Interactive star rating widget with tags, tip selector, and submission
3
+ */
4
+
5
+ import React, { useState } from 'react';
6
+ import { useForm, useToast } from '@mindmatrix/react';
7
+ import { Stack, Row, Text, Button, Icon, Badge, TextInput } from '@mindmatrix/react';
8
+
9
+ interface RatingStarsProps {
10
+ maxStars?: number;
11
+ rideId: string;
12
+ targetName: string;
13
+ targetRole: 'rider' | 'driver';
14
+ fare?: number;
15
+ showTip?: boolean;
16
+ onSubmit: (rating: { stars: number; comment: string; tags: string[]; tipAmount: number }) => void;
17
+ onSkip?: () => void;
18
+ }
19
+
20
+ const RIDER_TAGS = ['Clean car', 'Great conversation', 'Smooth driving', 'Good music', 'Knew the route', 'Professional'];
21
+ const DRIVER_TAGS = ['Respectful', 'On time', 'Good directions', 'Friendly', 'Clean passenger'];
22
+ const TIP_OPTIONS = [0, 2, 5, 10];
23
+
24
+ export function RatingStars({
25
+ maxStars = 5,
26
+ rideId,
27
+ targetName,
28
+ targetRole,
29
+ fare,
30
+ showTip = false,
31
+ onSubmit,
32
+ onSkip,
33
+ }: RatingStarsProps) {
34
+ const [hoveredStar, setHoveredStar] = useState(0);
35
+ const [selectedStars, setSelectedStars] = useState(0);
36
+ const [selectedTags, setSelectedTags] = useState<string[]>([]);
37
+ const [comment, setComment] = useState('');
38
+ const [tipAmount, setTipAmount] = useState(0);
39
+ const [customTip, setCustomTip] = useState('');
40
+ const [showCustomTip, setShowCustomTip] = useState(false);
41
+ const { toast } = useToast();
42
+
43
+ const tags = targetRole === 'driver' ? RIDER_TAGS : DRIVER_TAGS;
44
+
45
+ const handleTagToggle = (tag: string) => {
46
+ setSelectedTags((prev) =>
47
+ prev.includes(tag) ? prev.filter((t) => t !== tag) : [...prev, tag]
48
+ );
49
+ };
50
+
51
+ const handleSubmit = () => {
52
+ if (selectedStars === 0) {
53
+ toast({ title: 'Please select a rating', variant: 'warning', duration: 3000 });
54
+ return;
55
+ }
56
+ const finalTip = showCustomTip ? parseFloat(customTip) || 0 : tipAmount;
57
+ onSubmit({ stars: selectedStars, comment, tags: selectedTags, tipAmount: finalTip });
58
+ toast({ title: 'Rating submitted!', variant: 'success', duration: 3000 });
59
+ };
60
+
61
+ const starLabel = selectedStars === 1 ? 'Poor' : selectedStars === 2 ? 'Fair' : selectedStars === 3 ? 'Good' : selectedStars === 4 ? 'Great' : selectedStars === 5 ? 'Excellent' : '';
62
+
63
+ return (
64
+ <Stack sx={{ bg: '#1a202c', borderRadius: 'xl', p: 6, maxWidth: '400px', width: '100%', border: '1px solid #2d3748' }}>
65
+ {/* Header */}
66
+ <Stack sx={{ alignItems: 'center', mb: 5 }}>
67
+ <Text sx={{ color: '#e2e8f0', fontSize: 'xl', fontWeight: 'bold', mb: 1 }}>
68
+ Rate your {targetRole === 'driver' ? 'ride' : 'passenger'}
69
+ </Text>
70
+ <Text sx={{ color: '#a0aec0', fontSize: 'sm' }}>
71
+ How was your experience with {targetName}?
72
+ </Text>
73
+ </Stack>
74
+
75
+ {/* Stars */}
76
+ <Row sx={{ justifyContent: 'center', gap: 2, mb: 2 }}>
77
+ {Array.from({ length: maxStars }, (_, i) => i + 1).map((star) => (
78
+ <Stack
79
+ key={star}
80
+ onMouseEnter={() => setHoveredStar(star)}
81
+ onMouseLeave={() => setHoveredStar(0)}
82
+ onClick={() => setSelectedStars(star)}
83
+ sx={{
84
+ cursor: 'pointer',
85
+ transition: 'transform 0.15s',
86
+ transform: (hoveredStar >= star || selectedStars >= star) ? 'scale(1.2)' : 'scale(1)',
87
+ }}
88
+ >
89
+ <Icon
90
+ name="star"
91
+ sx={{
92
+ fontSize: '36px',
93
+ color: (hoveredStar >= star || selectedStars >= star) ? '#ecc94b' : '#4a5568',
94
+ transition: 'color 0.15s',
95
+ }}
96
+ />
97
+ </Stack>
98
+ ))}
99
+ </Row>
100
+
101
+ {starLabel && (
102
+ <Text sx={{ textAlign: 'center', color: '#ecc94b', fontWeight: 600, mb: 4 }}>{starLabel}</Text>
103
+ )}
104
+
105
+ {/* Tags */}
106
+ {selectedStars > 0 && (
107
+ <>
108
+ <Text sx={{ color: '#a0aec0', fontSize: 'sm', mb: 2 }}>What stood out?</Text>
109
+ <Row sx={{ flexWrap: 'wrap', gap: 2, mb: 4 }}>
110
+ {tags.map((tag) => (
111
+ <Badge
112
+ key={tag}
113
+ onClick={() => handleTagToggle(tag)}
114
+ sx={{
115
+ px: 3,
116
+ py: 1.5,
117
+ borderRadius: 'full',
118
+ cursor: 'pointer',
119
+ fontSize: 'sm',
120
+ bg: selectedTags.includes(tag) ? '#4299e1' : '#2d3748',
121
+ color: selectedTags.includes(tag) ? 'white' : '#a0aec0',
122
+ border: selectedTags.includes(tag) ? '1px solid #4299e1' : '1px solid #4a5568',
123
+ transition: 'all 0.15s',
124
+ }}
125
+ >
126
+ {tag}
127
+ </Badge>
128
+ ))}
129
+ </Row>
130
+
131
+ {/* Comment */}
132
+ <TextInput
133
+ value={comment}
134
+ onChange={(e: { target: { value: string } }) => setComment(e.target.value)}
135
+ placeholder="Add a comment (optional)"
136
+ sx={{ bg: '#2d3748', color: '#e2e8f0', border: '1px solid #4a5568', borderRadius: 'lg', p: 3, mb: 4, fontSize: 'sm', _placeholder: { color: '#718096' } }}
137
+ />
138
+ </>
139
+ )}
140
+
141
+ {/* Tip selector */}
142
+ {showTip && selectedStars > 0 && (
143
+ <Stack sx={{ mb: 4 }}>
144
+ <Text sx={{ color: '#a0aec0', fontSize: 'sm', mb: 2 }}>Add a tip for {targetName}</Text>
145
+ <Row sx={{ gap: 2, mb: 2 }}>
146
+ {TIP_OPTIONS.map((amount) => (
147
+ <Button
148
+ key={amount}
149
+ onClick={() => { setTipAmount(amount); setShowCustomTip(false); }}
150
+ sx={{
151
+ flex: 1,
152
+ py: 2,
153
+ borderRadius: 'lg',
154
+ fontWeight: 'bold',
155
+ bg: tipAmount === amount && !showCustomTip ? '#48bb78' : '#2d3748',
156
+ color: tipAmount === amount && !showCustomTip ? 'white' : '#a0aec0',
157
+ border: tipAmount === amount && !showCustomTip ? '1px solid #48bb78' : '1px solid #4a5568',
158
+ }}
159
+ >
160
+ {amount === 0 ? 'No tip' : `$${amount}`}
161
+ </Button>
162
+ ))}
163
+ <Button
164
+ onClick={() => setShowCustomTip(!showCustomTip)}
165
+ sx={{
166
+ flex: 1,
167
+ py: 2,
168
+ borderRadius: 'lg',
169
+ bg: showCustomTip ? '#48bb78' : '#2d3748',
170
+ color: showCustomTip ? 'white' : '#a0aec0',
171
+ border: showCustomTip ? '1px solid #48bb78' : '1px solid #4a5568',
172
+ }}
173
+ >
174
+ Custom
175
+ </Button>
176
+ </Row>
177
+ {showCustomTip && (
178
+ <TextInput
179
+ value={customTip}
180
+ onChange={(e: { target: { value: string } }) => setCustomTip(e.target.value)}
181
+ placeholder="Enter amount"
182
+ sx={{ bg: '#2d3748', color: '#e2e8f0', border: '1px solid #4a5568', borderRadius: 'lg', p: 3, fontSize: 'sm' }}
183
+ />
184
+ )}
185
+ </Stack>
186
+ )}
187
+
188
+ {/* Fare display */}
189
+ {fare !== undefined && (
190
+ <Row sx={{ justifyContent: 'space-between', bg: '#2d3748', p: 3, borderRadius: 'lg', mb: 4 }}>
191
+ <Text sx={{ color: '#a0aec0', fontSize: 'sm' }}>Ride fare</Text>
192
+ <Text sx={{ color: '#e2e8f0', fontWeight: 'bold' }}>${fare.toFixed(2)}</Text>
193
+ </Row>
194
+ )}
195
+
196
+ {/* Actions */}
197
+ <Row sx={{ gap: 3 }}>
198
+ {onSkip && (
199
+ <Button
200
+ onClick={onSkip}
201
+ sx={{ flex: 1, py: 3, borderRadius: 'lg', bg: '#2d3748', color: '#a0aec0', fontWeight: 600, _hover: { bg: '#4a5568' } }}
202
+ >
203
+ Skip
204
+ </Button>
205
+ )}
206
+ <Button
207
+ onClick={handleSubmit}
208
+ sx={{
209
+ flex: 2,
210
+ py: 3,
211
+ borderRadius: 'lg',
212
+ bg: selectedStars > 0 ? '#4299e1' : '#2d3748',
213
+ color: selectedStars > 0 ? 'white' : '#718096',
214
+ fontWeight: 'bold',
215
+ fontSize: 'md',
216
+ cursor: selectedStars > 0 ? 'pointer' : 'not-allowed',
217
+ _hover: selectedStars > 0 ? { bg: '#3182ce' } : {},
218
+ }}
219
+ >
220
+ Submit Rating
221
+ </Button>
222
+ </Row>
223
+ </Stack>
224
+ );
225
+ }
226
+
227
+ export default RatingStars;