@contractspec/example.learning-journey-ui-coaching 1.44.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 (71) hide show
  1. package/.turbo/turbo-build$colon$bundle.log +57 -0
  2. package/.turbo/turbo-build.log +58 -0
  3. package/CHANGELOG.md +262 -0
  4. package/LICENSE +21 -0
  5. package/README.md +32 -0
  6. package/dist/CoachingMiniApp.d.ts +17 -0
  7. package/dist/CoachingMiniApp.d.ts.map +1 -0
  8. package/dist/CoachingMiniApp.js +63 -0
  9. package/dist/CoachingMiniApp.js.map +1 -0
  10. package/dist/components/EngagementMeter.d.ts +18 -0
  11. package/dist/components/EngagementMeter.d.ts.map +1 -0
  12. package/dist/components/EngagementMeter.js +108 -0
  13. package/dist/components/EngagementMeter.js.map +1 -0
  14. package/dist/components/TipCard.d.ts +21 -0
  15. package/dist/components/TipCard.d.ts.map +1 -0
  16. package/dist/components/TipCard.js +76 -0
  17. package/dist/components/TipCard.js.map +1 -0
  18. package/dist/components/TipFeed.d.ts +18 -0
  19. package/dist/components/TipFeed.d.ts.map +1 -0
  20. package/dist/components/TipFeed.js +66 -0
  21. package/dist/components/TipFeed.js.map +1 -0
  22. package/dist/components/index.d.ts +4 -0
  23. package/dist/components/index.js +5 -0
  24. package/dist/docs/index.d.ts +1 -0
  25. package/dist/docs/index.js +1 -0
  26. package/dist/docs/learning-journey-ui-coaching.docblock.d.ts +1 -0
  27. package/dist/docs/learning-journey-ui-coaching.docblock.js +20 -0
  28. package/dist/docs/learning-journey-ui-coaching.docblock.js.map +1 -0
  29. package/dist/example.d.ts +33 -0
  30. package/dist/example.d.ts.map +1 -0
  31. package/dist/example.js +35 -0
  32. package/dist/example.js.map +1 -0
  33. package/dist/index.d.ts +12 -0
  34. package/dist/index.js +14 -0
  35. package/dist/views/Overview.d.ts +16 -0
  36. package/dist/views/Overview.d.ts.map +1 -0
  37. package/dist/views/Overview.js +151 -0
  38. package/dist/views/Overview.js.map +1 -0
  39. package/dist/views/Progress.d.ts +11 -0
  40. package/dist/views/Progress.d.ts.map +1 -0
  41. package/dist/views/Progress.js +115 -0
  42. package/dist/views/Progress.js.map +1 -0
  43. package/dist/views/Steps.d.ts +12 -0
  44. package/dist/views/Steps.d.ts.map +1 -0
  45. package/dist/views/Steps.js +69 -0
  46. package/dist/views/Steps.js.map +1 -0
  47. package/dist/views/Timeline.d.ts +11 -0
  48. package/dist/views/Timeline.d.ts.map +1 -0
  49. package/dist/views/Timeline.js +113 -0
  50. package/dist/views/Timeline.js.map +1 -0
  51. package/dist/views/index.d.ts +5 -0
  52. package/dist/views/index.js +6 -0
  53. package/example.ts +1 -0
  54. package/package.json +79 -0
  55. package/src/CoachingMiniApp.tsx +93 -0
  56. package/src/components/EngagementMeter.tsx +93 -0
  57. package/src/components/TipCard.tsx +101 -0
  58. package/src/components/TipFeed.tsx +101 -0
  59. package/src/components/index.ts +3 -0
  60. package/src/docs/index.ts +1 -0
  61. package/src/docs/learning-journey-ui-coaching.docblock.ts +17 -0
  62. package/src/example.ts +24 -0
  63. package/src/index.ts +10 -0
  64. package/src/views/Overview.tsx +157 -0
  65. package/src/views/Progress.tsx +164 -0
  66. package/src/views/Steps.tsx +63 -0
  67. package/src/views/Timeline.tsx +114 -0
  68. package/src/views/index.ts +4 -0
  69. package/tsconfig.json +10 -0
  70. package/tsconfig.tsbuildinfo +1 -0
  71. package/tsdown.config.js +17 -0
@@ -0,0 +1,108 @@
1
+ 'use client';
2
+
3
+ import { jsx, jsxs } from "react/jsx-runtime";
4
+
5
+ //#region src/components/EngagementMeter.tsx
6
+ function EngagementMeter({ acknowledged, actioned, pending, streak = 0 }) {
7
+ const total = acknowledged + actioned + pending;
8
+ const actionedPercent = total > 0 ? actioned / total * 100 : 0;
9
+ const acknowledgedPercent = total > 0 ? acknowledged / total * 100 : 0;
10
+ return /* @__PURE__ */ jsxs("div", {
11
+ className: "space-y-4",
12
+ children: [
13
+ /* @__PURE__ */ jsx("div", {
14
+ className: "flex items-center justify-center",
15
+ children: /* @__PURE__ */ jsxs("div", {
16
+ className: "relative h-32 w-32",
17
+ children: [/* @__PURE__ */ jsxs("svg", {
18
+ className: "h-full w-full -rotate-90",
19
+ viewBox: "0 0 100 100",
20
+ children: [
21
+ /* @__PURE__ */ jsx("circle", {
22
+ cx: "50",
23
+ cy: "50",
24
+ r: "40",
25
+ fill: "none",
26
+ strokeWidth: "12",
27
+ className: "stroke-muted"
28
+ }),
29
+ /* @__PURE__ */ jsx("circle", {
30
+ cx: "50",
31
+ cy: "50",
32
+ r: "40",
33
+ fill: "none",
34
+ strokeWidth: "12",
35
+ strokeLinecap: "round",
36
+ strokeDasharray: `${actionedPercent * 2.51} 251`,
37
+ className: "stroke-green-500 transition-all duration-500"
38
+ }),
39
+ /* @__PURE__ */ jsx("circle", {
40
+ cx: "50",
41
+ cy: "50",
42
+ r: "40",
43
+ fill: "none",
44
+ strokeWidth: "12",
45
+ strokeLinecap: "round",
46
+ strokeDasharray: `${acknowledgedPercent * 2.51} 251`,
47
+ strokeDashoffset: `${-actionedPercent * 2.51}`,
48
+ className: "stroke-amber-500 transition-all duration-500"
49
+ })
50
+ ]
51
+ }), /* @__PURE__ */ jsxs("div", {
52
+ className: "absolute inset-0 flex flex-col items-center justify-center",
53
+ children: [/* @__PURE__ */ jsx("span", {
54
+ className: "text-2xl font-bold",
55
+ children: total
56
+ }), /* @__PURE__ */ jsx("span", {
57
+ className: "text-muted-foreground text-xs",
58
+ children: "tips"
59
+ })]
60
+ })]
61
+ })
62
+ }),
63
+ /* @__PURE__ */ jsxs("div", {
64
+ className: "flex justify-center gap-4 text-sm",
65
+ children: [
66
+ /* @__PURE__ */ jsxs("div", {
67
+ className: "flex items-center gap-1.5",
68
+ children: [/* @__PURE__ */ jsx("div", { className: "h-3 w-3 rounded-full bg-green-500" }), /* @__PURE__ */ jsxs("span", { children: [
69
+ "Actioned (",
70
+ actioned,
71
+ ")"
72
+ ] })]
73
+ }),
74
+ /* @__PURE__ */ jsxs("div", {
75
+ className: "flex items-center gap-1.5",
76
+ children: [/* @__PURE__ */ jsx("div", { className: "h-3 w-3 rounded-full bg-amber-500" }), /* @__PURE__ */ jsxs("span", { children: [
77
+ "Acknowledged (",
78
+ acknowledged,
79
+ ")"
80
+ ] })]
81
+ }),
82
+ /* @__PURE__ */ jsxs("div", {
83
+ className: "flex items-center gap-1.5",
84
+ children: [/* @__PURE__ */ jsx("div", { className: "bg-muted h-3 w-3 rounded-full" }), /* @__PURE__ */ jsxs("span", { children: [
85
+ "Pending (",
86
+ pending,
87
+ ")"
88
+ ] })]
89
+ })
90
+ ]
91
+ }),
92
+ streak > 0 && /* @__PURE__ */ jsxs("div", {
93
+ className: "flex items-center justify-center gap-2 rounded-lg bg-orange-500/10 px-4 py-2",
94
+ children: [/* @__PURE__ */ jsx("span", {
95
+ className: "text-xl",
96
+ children: "🔥"
97
+ }), /* @__PURE__ */ jsxs("span", {
98
+ className: "font-semibold text-orange-500",
99
+ children: [streak, " day engagement streak!"]
100
+ })]
101
+ })
102
+ ]
103
+ });
104
+ }
105
+
106
+ //#endregion
107
+ export { EngagementMeter };
108
+ //# sourceMappingURL=EngagementMeter.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"EngagementMeter.js","names":[],"sources":["../../src/components/EngagementMeter.tsx"],"sourcesContent":["'use client';\n\ninterface EngagementMeterProps {\n acknowledged: number;\n actioned: number;\n pending: number;\n streak?: number;\n}\n\nexport function EngagementMeter({\n acknowledged,\n actioned,\n pending,\n streak = 0,\n}: EngagementMeterProps) {\n const total = acknowledged + actioned + pending;\n const actionedPercent = total > 0 ? (actioned / total) * 100 : 0;\n const acknowledgedPercent = total > 0 ? (acknowledged / total) * 100 : 0;\n\n return (\n <div className=\"space-y-4\">\n {/* Donut chart */}\n <div className=\"flex items-center justify-center\">\n <div className=\"relative h-32 w-32\">\n <svg className=\"h-full w-full -rotate-90\" viewBox=\"0 0 100 100\">\n {/* Background */}\n <circle\n cx=\"50\"\n cy=\"50\"\n r=\"40\"\n fill=\"none\"\n strokeWidth=\"12\"\n className=\"stroke-muted\"\n />\n {/* Actioned (green) */}\n <circle\n cx=\"50\"\n cy=\"50\"\n r=\"40\"\n fill=\"none\"\n strokeWidth=\"12\"\n strokeLinecap=\"round\"\n strokeDasharray={`${actionedPercent * 2.51} 251`}\n className=\"stroke-green-500 transition-all duration-500\"\n />\n {/* Acknowledged (amber) - offset by actioned */}\n <circle\n cx=\"50\"\n cy=\"50\"\n r=\"40\"\n fill=\"none\"\n strokeWidth=\"12\"\n strokeLinecap=\"round\"\n strokeDasharray={`${acknowledgedPercent * 2.51} 251`}\n strokeDashoffset={`${-actionedPercent * 2.51}`}\n className=\"stroke-amber-500 transition-all duration-500\"\n />\n </svg>\n <div className=\"absolute inset-0 flex flex-col items-center justify-center\">\n <span className=\"text-2xl font-bold\">{total}</span>\n <span className=\"text-muted-foreground text-xs\">tips</span>\n </div>\n </div>\n </div>\n\n {/* Legend */}\n <div className=\"flex justify-center gap-4 text-sm\">\n <div className=\"flex items-center gap-1.5\">\n <div className=\"h-3 w-3 rounded-full bg-green-500\" />\n <span>Actioned ({actioned})</span>\n </div>\n <div className=\"flex items-center gap-1.5\">\n <div className=\"h-3 w-3 rounded-full bg-amber-500\" />\n <span>Acknowledged ({acknowledged})</span>\n </div>\n <div className=\"flex items-center gap-1.5\">\n <div className=\"bg-muted h-3 w-3 rounded-full\" />\n <span>Pending ({pending})</span>\n </div>\n </div>\n\n {/* Streak */}\n {streak > 0 && (\n <div className=\"flex items-center justify-center gap-2 rounded-lg bg-orange-500/10 px-4 py-2\">\n <span className=\"text-xl\">🔥</span>\n <span className=\"font-semibold text-orange-500\">\n {streak} day engagement streak!\n </span>\n </div>\n )}\n </div>\n );\n}\n"],"mappings":";;;;;AASA,SAAgB,gBAAgB,EAC9B,cACA,UACA,SACA,SAAS,KACc;CACvB,MAAM,QAAQ,eAAe,WAAW;CACxC,MAAM,kBAAkB,QAAQ,IAAK,WAAW,QAAS,MAAM;CAC/D,MAAM,sBAAsB,QAAQ,IAAK,eAAe,QAAS,MAAM;AAEvE,QACE,qBAAC;EAAI,WAAU;;GAEb,oBAAC;IAAI,WAAU;cACb,qBAAC;KAAI,WAAU;gBACb,qBAAC;MAAI,WAAU;MAA2B,SAAQ;;OAEhD,oBAAC;QACC,IAAG;QACH,IAAG;QACH,GAAE;QACF,MAAK;QACL,aAAY;QACZ,WAAU;SACV;OAEF,oBAAC;QACC,IAAG;QACH,IAAG;QACH,GAAE;QACF,MAAK;QACL,aAAY;QACZ,eAAc;QACd,iBAAiB,GAAG,kBAAkB,KAAK;QAC3C,WAAU;SACV;OAEF,oBAAC;QACC,IAAG;QACH,IAAG;QACH,GAAE;QACF,MAAK;QACL,aAAY;QACZ,eAAc;QACd,iBAAiB,GAAG,sBAAsB,KAAK;QAC/C,kBAAkB,GAAG,CAAC,kBAAkB;QACxC,WAAU;SACV;;OACE,EACN,qBAAC;MAAI,WAAU;iBACb,oBAAC;OAAK,WAAU;iBAAsB;QAAa,EACnD,oBAAC;OAAK,WAAU;iBAAgC;QAAW;OACvD;MACF;KACF;GAGN,qBAAC;IAAI,WAAU;;KACb,qBAAC;MAAI,WAAU;iBACb,oBAAC,SAAI,WAAU,sCAAsC,EACrD,qBAAC;OAAK;OAAW;OAAS;UAAQ;OAC9B;KACN,qBAAC;MAAI,WAAU;iBACb,oBAAC,SAAI,WAAU,sCAAsC,EACrD,qBAAC;OAAK;OAAe;OAAa;UAAQ;OACtC;KACN,qBAAC;MAAI,WAAU;iBACb,oBAAC,SAAI,WAAU,kCAAkC,EACjD,qBAAC;OAAK;OAAU;OAAQ;UAAQ;OAC5B;;KACF;GAGL,SAAS,KACR,qBAAC;IAAI,WAAU;eACb,oBAAC;KAAK,WAAU;eAAU;MAAS,EACnC,qBAAC;KAAK,WAAU;gBACb,QAAO;MACH;KACH;;GAEJ"}
@@ -0,0 +1,21 @@
1
+ import * as react_jsx_runtime1 from "react/jsx-runtime";
2
+ import { LearningJourneyStepSpec } from "@contractspec/module.learning-journey/track-spec";
3
+
4
+ //#region src/components/TipCard.d.ts
5
+ interface TipCardProps {
6
+ step: LearningJourneyStepSpec;
7
+ isCompleted: boolean;
8
+ isCurrent: boolean;
9
+ onComplete?: () => void;
10
+ onDismiss?: () => void;
11
+ }
12
+ declare function TipCard({
13
+ step,
14
+ isCompleted,
15
+ isCurrent,
16
+ onComplete,
17
+ onDismiss
18
+ }: TipCardProps): react_jsx_runtime1.JSX.Element;
19
+ //#endregion
20
+ export { TipCard };
21
+ //# sourceMappingURL=TipCard.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"TipCard.d.ts","names":[],"sources":["../../src/components/TipCard.tsx"],"sourcesContent":[],"mappings":";;;;UAOU,YAAA;QACF;;EADE,SAAA,EAAA,OAAY;EAkBN,UAAO,CAAA,EAAA,GAAA,GAAA,IAAA;EACrB,SAAA,CAAA,EAAA,GAAA,GAAA,IAAA;;AAEA,iBAHc,OAAA,CAGd;EAAA,IAAA;EAAA,WAAA;EAAA,SAAA;EAAA,UAAA;EAAA;AAAA,CAAA,EAGC,YAHD,CAAA,EAGa,kBAAA,CAAA,GAAA,CAAA,OAHb"}
@@ -0,0 +1,76 @@
1
+ 'use client';
2
+
3
+ import { Card, CardContent } from "@contractspec/lib.ui-kit-web/ui/card";
4
+ import { Button } from "@contractspec/lib.design-system";
5
+ import { cn } from "@contractspec/lib.ui-kit-web/ui/utils";
6
+ import { jsx, jsxs } from "react/jsx-runtime";
7
+
8
+ //#region src/components/TipCard.tsx
9
+ const TIP_ICONS = {
10
+ cash_buffer_too_high: "💰",
11
+ no_savings_goal: "🎯",
12
+ irregular_savings: "📅",
13
+ noise_late_evening: "🔇",
14
+ guest_frequency_high: "👥",
15
+ shared_space_conflicts: "🏠",
16
+ default: "💡"
17
+ };
18
+ function TipCard({ step, isCompleted, isCurrent, onComplete, onDismiss }) {
19
+ const icon = TIP_ICONS[step.metadata?.tipId ?? "default"] ?? TIP_ICONS.default;
20
+ return /* @__PURE__ */ jsx(Card, {
21
+ className: cn("transition-all", isCompleted && "opacity-60", isCurrent && "ring-2 ring-amber-500"),
22
+ children: /* @__PURE__ */ jsx(CardContent, {
23
+ className: "p-4",
24
+ children: /* @__PURE__ */ jsxs("div", {
25
+ className: "flex gap-4",
26
+ children: [/* @__PURE__ */ jsx("div", {
27
+ className: cn("flex h-12 w-12 shrink-0 items-center justify-center rounded-xl text-2xl", isCompleted ? "bg-green-500/10" : isCurrent ? "bg-amber-500/10" : "bg-muted"),
28
+ children: isCompleted ? "✓" : icon
29
+ }), /* @__PURE__ */ jsxs("div", {
30
+ className: "min-w-0 flex-1",
31
+ children: [
32
+ /* @__PURE__ */ jsxs("div", {
33
+ className: "flex items-start justify-between gap-2",
34
+ children: [/* @__PURE__ */ jsx("h4", {
35
+ className: "font-semibold",
36
+ children: step.title
37
+ }), step.xpReward && /* @__PURE__ */ jsxs("span", {
38
+ className: cn("shrink-0 rounded-full px-2 py-0.5 text-xs font-semibold", isCompleted ? "bg-green-500/10 text-green-500" : "bg-amber-500/10 text-amber-500"),
39
+ children: [
40
+ "+",
41
+ step.xpReward,
42
+ " XP"
43
+ ]
44
+ })]
45
+ }),
46
+ /* @__PURE__ */ jsx("p", {
47
+ className: "text-muted-foreground mt-1 text-sm",
48
+ children: step.description
49
+ }),
50
+ !isCompleted && /* @__PURE__ */ jsxs("div", {
51
+ className: "mt-3 flex flex-wrap gap-2",
52
+ children: [/* @__PURE__ */ jsx(Button, {
53
+ size: "sm",
54
+ onClick: onComplete,
55
+ children: "Take Action"
56
+ }), /* @__PURE__ */ jsx(Button, {
57
+ variant: "outline",
58
+ size: "sm",
59
+ onClick: onDismiss,
60
+ children: "Dismiss"
61
+ })]
62
+ }),
63
+ isCompleted && /* @__PURE__ */ jsx("p", {
64
+ className: "mt-2 text-sm text-green-500",
65
+ children: "✓ Tip acknowledged"
66
+ })
67
+ ]
68
+ })]
69
+ })
70
+ })
71
+ });
72
+ }
73
+
74
+ //#endregion
75
+ export { TipCard };
76
+ //# sourceMappingURL=TipCard.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"TipCard.js","names":["TIP_ICONS: Record<string, string>"],"sources":["../../src/components/TipCard.tsx"],"sourcesContent":["'use client';\n\nimport { Button } from '@contractspec/lib.design-system';\nimport { Card, CardContent } from '@contractspec/lib.ui-kit-web/ui/card';\nimport { cn } from '@contractspec/lib.ui-kit-web/ui/utils';\nimport type { LearningJourneyStepSpec } from '@contractspec/module.learning-journey/track-spec';\n\ninterface TipCardProps {\n step: LearningJourneyStepSpec;\n isCompleted: boolean;\n isCurrent: boolean;\n onComplete?: () => void;\n onDismiss?: () => void;\n}\n\nconst TIP_ICONS: Record<string, string> = {\n cash_buffer_too_high: '💰',\n no_savings_goal: '🎯',\n irregular_savings: '📅',\n noise_late_evening: '🔇',\n guest_frequency_high: '👥',\n shared_space_conflicts: '🏠',\n default: '💡',\n};\n\nexport function TipCard({\n step,\n isCompleted,\n isCurrent,\n onComplete,\n onDismiss,\n}: TipCardProps) {\n const tipId = (step.metadata?.tipId as string) ?? 'default';\n const icon = TIP_ICONS[tipId] ?? TIP_ICONS.default;\n\n return (\n <Card\n className={cn(\n 'transition-all',\n isCompleted && 'opacity-60',\n isCurrent && 'ring-2 ring-amber-500'\n )}\n >\n <CardContent className=\"p-4\">\n <div className=\"flex gap-4\">\n {/* Icon */}\n <div\n className={cn(\n 'flex h-12 w-12 shrink-0 items-center justify-center rounded-xl text-2xl',\n isCompleted\n ? 'bg-green-500/10'\n : isCurrent\n ? 'bg-amber-500/10'\n : 'bg-muted'\n )}\n >\n {isCompleted ? '✓' : icon}\n </div>\n\n {/* Content */}\n <div className=\"min-w-0 flex-1\">\n <div className=\"flex items-start justify-between gap-2\">\n <h4 className=\"font-semibold\">{step.title}</h4>\n {step.xpReward && (\n <span\n className={cn(\n 'shrink-0 rounded-full px-2 py-0.5 text-xs font-semibold',\n isCompleted\n ? 'bg-green-500/10 text-green-500'\n : 'bg-amber-500/10 text-amber-500'\n )}\n >\n +{step.xpReward} XP\n </span>\n )}\n </div>\n <p className=\"text-muted-foreground mt-1 text-sm\">\n {step.description}\n </p>\n\n {/* Actions */}\n {!isCompleted && (\n <div className=\"mt-3 flex flex-wrap gap-2\">\n <Button size=\"sm\" onClick={onComplete}>\n Take Action\n </Button>\n <Button variant=\"outline\" size=\"sm\" onClick={onDismiss}>\n Dismiss\n </Button>\n </div>\n )}\n\n {isCompleted && (\n <p className=\"mt-2 text-sm text-green-500\">✓ Tip acknowledged</p>\n )}\n </div>\n </div>\n </CardContent>\n </Card>\n );\n}\n"],"mappings":";;;;;;;;AAeA,MAAMA,YAAoC;CACxC,sBAAsB;CACtB,iBAAiB;CACjB,mBAAmB;CACnB,oBAAoB;CACpB,sBAAsB;CACtB,wBAAwB;CACxB,SAAS;CACV;AAED,SAAgB,QAAQ,EACtB,MACA,aACA,WACA,YACA,aACe;CAEf,MAAM,OAAO,UADE,KAAK,UAAU,SAAoB,cACjB,UAAU;AAE3C,QACE,oBAAC;EACC,WAAW,GACT,kBACA,eAAe,cACf,aAAa,wBACd;YAED,oBAAC;GAAY,WAAU;aACrB,qBAAC;IAAI,WAAU;eAEb,oBAAC;KACC,WAAW,GACT,2EACA,cACI,oBACA,YACE,oBACA,WACP;eAEA,cAAc,MAAM;MACjB,EAGN,qBAAC;KAAI,WAAU;;MACb,qBAAC;OAAI,WAAU;kBACb,oBAAC;QAAG,WAAU;kBAAiB,KAAK;SAAW,EAC9C,KAAK,YACJ,qBAAC;QACC,WAAW,GACT,2DACA,cACI,mCACA,iCACL;;SACF;SACG,KAAK;SAAS;;SACX;QAEL;MACN,oBAAC;OAAE,WAAU;iBACV,KAAK;QACJ;MAGH,CAAC,eACA,qBAAC;OAAI,WAAU;kBACb,oBAAC;QAAO,MAAK;QAAK,SAAS;kBAAY;SAE9B,EACT,oBAAC;QAAO,SAAQ;QAAU,MAAK;QAAK,SAAS;kBAAW;SAE/C;QACL;MAGP,eACC,oBAAC;OAAE,WAAU;iBAA8B;QAAsB;;MAE/D;KACF;IACM;GACT"}
@@ -0,0 +1,18 @@
1
+ import * as react_jsx_runtime2 from "react/jsx-runtime";
2
+ import { LearningJourneyStepSpec } from "@contractspec/module.learning-journey/track-spec";
3
+
4
+ //#region src/components/TipFeed.d.ts
5
+ interface TipFeedItem {
6
+ step: LearningJourneyStepSpec;
7
+ isCompleted: boolean;
8
+ completedAt?: string;
9
+ }
10
+ interface TipFeedProps {
11
+ items: TipFeedItem[];
12
+ }
13
+ declare function TipFeed({
14
+ items
15
+ }: TipFeedProps): react_jsx_runtime2.JSX.Element;
16
+ //#endregion
17
+ export { TipFeed };
18
+ //# sourceMappingURL=TipFeed.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"TipFeed.d.ts","names":[],"sources":["../../src/components/TipFeed.tsx"],"sourcesContent":[],"mappings":";;;;UAKU,WAAA;QACF;;EADE,WAAA,CAAA,EAAW,MAAA;AACU;AAmB/B,UAdU,YAAA,CAca;EAAG,KAAA,EAbjB,WAaiB,EAAA;;AAAqB,iBAA/B,OAAA,CAA+B;EAAA;AAAA,CAAA,EAAZ,YAAY,CAAA,EAAA,kBAAA,CAAA,GAAA,CAAA,OAAA"}
@@ -0,0 +1,66 @@
1
+ 'use client';
2
+
3
+ import { cn } from "@contractspec/lib.ui-kit-web/ui/utils";
4
+ import { jsx, jsxs } from "react/jsx-runtime";
5
+
6
+ //#region src/components/TipFeed.tsx
7
+ const TIP_ICONS = {
8
+ cash_buffer_too_high: "💰",
9
+ no_savings_goal: "🎯",
10
+ irregular_savings: "📅",
11
+ noise_late_evening: "🔇",
12
+ guest_frequency_high: "👥",
13
+ shared_space_conflicts: "🏠",
14
+ default: "💡"
15
+ };
16
+ function TipFeed({ items }) {
17
+ if (items.length === 0) return /* @__PURE__ */ jsx("div", {
18
+ className: "text-muted-foreground py-8 text-center",
19
+ children: "No tips yet. Start engaging with coaching tips!"
20
+ });
21
+ return /* @__PURE__ */ jsxs("div", {
22
+ className: "relative",
23
+ children: [/* @__PURE__ */ jsx("div", { className: "bg-border absolute top-0 left-4 h-full w-0.5" }), /* @__PURE__ */ jsx("div", {
24
+ className: "space-y-4",
25
+ children: items.map((item) => {
26
+ const icon = TIP_ICONS[item.step.metadata?.tipId ?? "default"] ?? TIP_ICONS.default;
27
+ return /* @__PURE__ */ jsxs("div", {
28
+ className: "relative flex gap-4 pl-2",
29
+ children: [/* @__PURE__ */ jsx("div", {
30
+ className: cn("relative z-10 flex h-8 w-8 shrink-0 items-center justify-center rounded-full text-sm", item.isCompleted ? "bg-green-500 text-white" : "bg-muted text-muted-foreground"),
31
+ children: item.isCompleted ? "✓" : icon
32
+ }), /* @__PURE__ */ jsxs("div", {
33
+ className: "bg-card flex-1 rounded-lg border p-3",
34
+ children: [/* @__PURE__ */ jsxs("div", {
35
+ className: "flex items-start justify-between gap-2",
36
+ children: [/* @__PURE__ */ jsxs("div", { children: [/* @__PURE__ */ jsx("p", {
37
+ className: "font-medium",
38
+ children: item.step.title
39
+ }), /* @__PURE__ */ jsx("p", {
40
+ className: "text-muted-foreground mt-0.5 text-sm",
41
+ children: item.step.description
42
+ })] }), item.step.xpReward && /* @__PURE__ */ jsxs("span", {
43
+ className: cn("shrink-0 text-xs font-medium", item.isCompleted ? "text-green-500" : "text-muted-foreground"),
44
+ children: [
45
+ "+",
46
+ item.step.xpReward,
47
+ " XP"
48
+ ]
49
+ })]
50
+ }), /* @__PURE__ */ jsx("div", {
51
+ className: "text-muted-foreground mt-2 flex items-center gap-2 text-xs",
52
+ children: item.isCompleted ? /* @__PURE__ */ jsxs("span", {
53
+ className: "text-green-500",
54
+ children: ["✓ Completed", item.completedAt && ` • ${item.completedAt}`]
55
+ }) : /* @__PURE__ */ jsx("span", { children: "Pending action" })
56
+ })]
57
+ })]
58
+ }, item.step.id);
59
+ })
60
+ })]
61
+ });
62
+ }
63
+
64
+ //#endregion
65
+ export { TipFeed };
66
+ //# sourceMappingURL=TipFeed.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"TipFeed.js","names":["TIP_ICONS: Record<string, string>"],"sources":["../../src/components/TipFeed.tsx"],"sourcesContent":["'use client';\n\nimport { cn } from '@contractspec/lib.ui-kit-web/ui/utils';\nimport type { LearningJourneyStepSpec } from '@contractspec/module.learning-journey/track-spec';\n\ninterface TipFeedItem {\n step: LearningJourneyStepSpec;\n isCompleted: boolean;\n completedAt?: string;\n}\n\ninterface TipFeedProps {\n items: TipFeedItem[];\n}\n\nconst TIP_ICONS: Record<string, string> = {\n cash_buffer_too_high: '💰',\n no_savings_goal: '🎯',\n irregular_savings: '📅',\n noise_late_evening: '🔇',\n guest_frequency_high: '👥',\n shared_space_conflicts: '🏠',\n default: '💡',\n};\n\nexport function TipFeed({ items }: TipFeedProps) {\n if (items.length === 0) {\n return (\n <div className=\"text-muted-foreground py-8 text-center\">\n No tips yet. Start engaging with coaching tips!\n </div>\n );\n }\n\n return (\n <div className=\"relative\">\n {/* Timeline line */}\n <div className=\"bg-border absolute top-0 left-4 h-full w-0.5\" />\n\n {/* Feed items */}\n <div className=\"space-y-4\">\n {items.map((item) => {\n const tipId = (item.step.metadata?.tipId as string) ?? 'default';\n const icon = TIP_ICONS[tipId] ?? TIP_ICONS.default;\n\n return (\n <div key={item.step.id} className=\"relative flex gap-4 pl-2\">\n {/* Node */}\n <div\n className={cn(\n 'relative z-10 flex h-8 w-8 shrink-0 items-center justify-center rounded-full text-sm',\n item.isCompleted\n ? 'bg-green-500 text-white'\n : 'bg-muted text-muted-foreground'\n )}\n >\n {item.isCompleted ? '✓' : icon}\n </div>\n\n {/* Content */}\n <div className=\"bg-card flex-1 rounded-lg border p-3\">\n <div className=\"flex items-start justify-between gap-2\">\n <div>\n <p className=\"font-medium\">{item.step.title}</p>\n <p className=\"text-muted-foreground mt-0.5 text-sm\">\n {item.step.description}\n </p>\n </div>\n {item.step.xpReward && (\n <span\n className={cn(\n 'shrink-0 text-xs font-medium',\n item.isCompleted\n ? 'text-green-500'\n : 'text-muted-foreground'\n )}\n >\n +{item.step.xpReward} XP\n </span>\n )}\n </div>\n\n {/* Timestamp */}\n <div className=\"text-muted-foreground mt-2 flex items-center gap-2 text-xs\">\n {item.isCompleted ? (\n <span className=\"text-green-500\">\n ✓ Completed\n {item.completedAt && ` • ${item.completedAt}`}\n </span>\n ) : (\n <span>Pending action</span>\n )}\n </div>\n </div>\n </div>\n );\n })}\n </div>\n </div>\n );\n}\n"],"mappings":";;;;;;AAeA,MAAMA,YAAoC;CACxC,sBAAsB;CACtB,iBAAiB;CACjB,mBAAmB;CACnB,oBAAoB;CACpB,sBAAsB;CACtB,wBAAwB;CACxB,SAAS;CACV;AAED,SAAgB,QAAQ,EAAE,SAAuB;AAC/C,KAAI,MAAM,WAAW,EACnB,QACE,oBAAC;EAAI,WAAU;YAAyC;GAElD;AAIV,QACE,qBAAC;EAAI,WAAU;aAEb,oBAAC,SAAI,WAAU,iDAAiD,EAGhE,oBAAC;GAAI,WAAU;aACZ,MAAM,KAAK,SAAS;IAEnB,MAAM,OAAO,UADE,KAAK,KAAK,UAAU,SAAoB,cACtB,UAAU;AAE3C,WACE,qBAAC;KAAuB,WAAU;gBAEhC,oBAAC;MACC,WAAW,GACT,wFACA,KAAK,cACD,4BACA,iCACL;gBAEA,KAAK,cAAc,MAAM;OACtB,EAGN,qBAAC;MAAI,WAAU;iBACb,qBAAC;OAAI,WAAU;kBACb,qBAAC,oBACC,oBAAC;QAAE,WAAU;kBAAe,KAAK,KAAK;SAAU,EAChD,oBAAC;QAAE,WAAU;kBACV,KAAK,KAAK;SACT,IACA,EACL,KAAK,KAAK,YACT,qBAAC;QACC,WAAW,GACT,gCACA,KAAK,cACD,mBACA,wBACL;;SACF;SACG,KAAK,KAAK;SAAS;;SAChB;QAEL,EAGN,oBAAC;OAAI,WAAU;iBACZ,KAAK,cACJ,qBAAC;QAAK,WAAU;mBAAiB,eAE9B,KAAK,eAAe,MAAM,KAAK;SAC3B,GAEP,oBAAC,oBAAK,mBAAqB;QAEzB;OACF;OA/CE,KAAK,KAAK,GAgDd;KAER;IACE;GACF"}
@@ -0,0 +1,4 @@
1
+ import { EngagementMeter } from "./EngagementMeter.js";
2
+ import { TipCard } from "./TipCard.js";
3
+ import { TipFeed } from "./TipFeed.js";
4
+ export { EngagementMeter, TipCard, TipFeed };
@@ -0,0 +1,5 @@
1
+ import { TipCard } from "./TipCard.js";
2
+ import { EngagementMeter } from "./EngagementMeter.js";
3
+ import { TipFeed } from "./TipFeed.js";
4
+
5
+ export { EngagementMeter, TipCard, TipFeed };
@@ -0,0 +1 @@
1
+ export { };
@@ -0,0 +1 @@
1
+ import "./learning-journey-ui-coaching.docblock.js";
@@ -0,0 +1 @@
1
+ export { };
@@ -0,0 +1,20 @@
1
+ import { registerDocBlocks } from "@contractspec/lib.contracts/docs";
2
+
3
+ //#region src/docs/learning-journey-ui-coaching.docblock.ts
4
+ registerDocBlocks([{
5
+ id: "docs.examples.learning-journey-ui-coaching",
6
+ title: "Learning Journey UI — Coaching",
7
+ summary: "UI mini-app components for coaching: tips, engagement, progress.",
8
+ kind: "reference",
9
+ visibility: "public",
10
+ route: "/docs/examples/learning-journey-ui-coaching",
11
+ tags: [
12
+ "learning",
13
+ "ui",
14
+ "coaching"
15
+ ],
16
+ body: `## Includes\n- Coaching mini-app shell\n- Views: overview, steps, progress, timeline\n- Components: tip card, engagement meter, tip feed\n\n## Notes\n- Compose using design system components.\n- Keep accessibility and mobile-friendly tap targets.`
17
+ }]);
18
+
19
+ //#endregion
20
+ //# sourceMappingURL=learning-journey-ui-coaching.docblock.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"learning-journey-ui-coaching.docblock.js","names":[],"sources":["../../src/docs/learning-journey-ui-coaching.docblock.ts"],"sourcesContent":["import type { DocBlock } from '@contractspec/lib.contracts/docs';\nimport { registerDocBlocks } from '@contractspec/lib.contracts/docs';\n\nconst blocks: DocBlock[] = [\n {\n id: 'docs.examples.learning-journey-ui-coaching',\n title: 'Learning Journey UI — Coaching',\n summary: 'UI mini-app components for coaching: tips, engagement, progress.',\n kind: 'reference',\n visibility: 'public',\n route: '/docs/examples/learning-journey-ui-coaching',\n tags: ['learning', 'ui', 'coaching'],\n body: `## Includes\\n- Coaching mini-app shell\\n- Views: overview, steps, progress, timeline\\n- Components: tip card, engagement meter, tip feed\\n\\n## Notes\\n- Compose using design system components.\\n- Keep accessibility and mobile-friendly tap targets.`,\n },\n];\n\nregisterDocBlocks(blocks);\n"],"mappings":";;;AAgBA,kBAb2B,CACzB;CACE,IAAI;CACJ,OAAO;CACP,SAAS;CACT,MAAM;CACN,YAAY;CACZ,OAAO;CACP,MAAM;EAAC;EAAY;EAAM;EAAW;CACpC,MAAM;CACP,CACF,CAEwB"}
@@ -0,0 +1,33 @@
1
+ //#region src/example.d.ts
2
+ declare const example: {
3
+ readonly id: "learning-journey-ui-coaching";
4
+ readonly title: "Learning Journey UI — Coaching";
5
+ readonly summary: "UI mini-app for coaching patterns: tips, engagement meter, progress.";
6
+ readonly tags: readonly ["learning", "ui", "coaching"];
7
+ readonly kind: "ui";
8
+ readonly visibility: "public";
9
+ readonly docs: {
10
+ readonly rootDocId: "docs.examples.learning-journey-ui-coaching";
11
+ };
12
+ readonly entrypoints: {
13
+ readonly packageName: "@contractspec/example.learning-journey-ui-coaching";
14
+ readonly docs: "./docs";
15
+ };
16
+ readonly surfaces: {
17
+ readonly templates: true;
18
+ readonly sandbox: {
19
+ readonly enabled: true;
20
+ readonly modes: readonly ["playground", "markdown"];
21
+ };
22
+ readonly studio: {
23
+ readonly enabled: true;
24
+ readonly installable: true;
25
+ };
26
+ readonly mcp: {
27
+ readonly enabled: true;
28
+ };
29
+ };
30
+ };
31
+ //#endregion
32
+ export { example as default };
33
+ //# sourceMappingURL=example.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"example.d.ts","names":[],"sources":["../src/example.ts"],"sourcesContent":[],"mappings":";cAAM;EAAA,SAAA,EAqBI,EAAA,8BAAA"}
@@ -0,0 +1,35 @@
1
+ //#region src/example.ts
2
+ const example = {
3
+ id: "learning-journey-ui-coaching",
4
+ title: "Learning Journey UI — Coaching",
5
+ summary: "UI mini-app for coaching patterns: tips, engagement meter, progress.",
6
+ tags: [
7
+ "learning",
8
+ "ui",
9
+ "coaching"
10
+ ],
11
+ kind: "ui",
12
+ visibility: "public",
13
+ docs: { rootDocId: "docs.examples.learning-journey-ui-coaching" },
14
+ entrypoints: {
15
+ packageName: "@contractspec/example.learning-journey-ui-coaching",
16
+ docs: "./docs"
17
+ },
18
+ surfaces: {
19
+ templates: true,
20
+ sandbox: {
21
+ enabled: true,
22
+ modes: ["playground", "markdown"]
23
+ },
24
+ studio: {
25
+ enabled: true,
26
+ installable: true
27
+ },
28
+ mcp: { enabled: true }
29
+ }
30
+ };
31
+ var example_default = example;
32
+
33
+ //#endregion
34
+ export { example_default as default };
35
+ //# sourceMappingURL=example.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"example.js","names":[],"sources":["../src/example.ts"],"sourcesContent":["const example = {\n id: 'learning-journey-ui-coaching',\n title: 'Learning Journey UI — Coaching',\n summary:\n 'UI mini-app for coaching patterns: tips, engagement meter, progress.',\n tags: ['learning', 'ui', 'coaching'],\n kind: 'ui',\n visibility: 'public',\n docs: {\n rootDocId: 'docs.examples.learning-journey-ui-coaching',\n },\n entrypoints: {\n packageName: '@contractspec/example.learning-journey-ui-coaching',\n docs: './docs',\n },\n surfaces: {\n templates: true,\n sandbox: { enabled: true, modes: ['playground', 'markdown'] },\n studio: { enabled: true, installable: true },\n mcp: { enabled: true },\n },\n} as const;\n\nexport default example;\n"],"mappings":";AAAA,MAAM,UAAU;CACd,IAAI;CACJ,OAAO;CACP,SACE;CACF,MAAM;EAAC;EAAY;EAAM;EAAW;CACpC,MAAM;CACN,YAAY;CACZ,MAAM,EACJ,WAAW,8CACZ;CACD,aAAa;EACX,aAAa;EACb,MAAM;EACP;CACD,UAAU;EACR,WAAW;EACX,SAAS;GAAE,SAAS;GAAM,OAAO,CAAC,cAAc,WAAW;GAAE;EAC7D,QAAQ;GAAE,SAAS;GAAM,aAAa;GAAM;EAC5C,KAAK,EAAE,SAAS,MAAM;EACvB;CACF;AAED,sBAAe"}
@@ -0,0 +1,12 @@
1
+ import { CoachingMiniApp } from "./CoachingMiniApp.js";
2
+ import { EngagementMeter } from "./components/EngagementMeter.js";
3
+ import { TipCard } from "./components/TipCard.js";
4
+ import { TipFeed } from "./components/TipFeed.js";
5
+ import "./components/index.js";
6
+ import example from "./example.js";
7
+ import { Overview } from "./views/Overview.js";
8
+ import { Steps } from "./views/Steps.js";
9
+ import { Progress as ProgressView } from "./views/Progress.js";
10
+ import { Timeline } from "./views/Timeline.js";
11
+ import "./views/index.js";
12
+ export { CoachingMiniApp, EngagementMeter, Overview, ProgressView as Progress, Steps, Timeline, TipCard, TipFeed, example };
package/dist/index.js ADDED
@@ -0,0 +1,14 @@
1
+ import { TipCard } from "./components/TipCard.js";
2
+ import { Overview } from "./views/Overview.js";
3
+ import { Steps } from "./views/Steps.js";
4
+ import { EngagementMeter } from "./components/EngagementMeter.js";
5
+ import { Progress as ProgressView } from "./views/Progress.js";
6
+ import { TipFeed } from "./components/TipFeed.js";
7
+ import { Timeline } from "./views/Timeline.js";
8
+ import { CoachingMiniApp } from "./CoachingMiniApp.js";
9
+ import example_default from "./example.js";
10
+ import "./views/index.js";
11
+ import "./components/index.js";
12
+ import "./docs/index.js";
13
+
14
+ export { CoachingMiniApp, EngagementMeter, Overview, ProgressView as Progress, Steps, Timeline, TipCard, TipFeed, example_default as example };
@@ -0,0 +1,16 @@
1
+ import { LearningViewProps } from "@contractspec/example.learning-journey-ui-shared";
2
+ import * as react_jsx_runtime3 from "react/jsx-runtime";
3
+
4
+ //#region src/views/Overview.d.ts
5
+ interface CoachingOverviewProps extends LearningViewProps {
6
+ onStart?: () => void;
7
+ }
8
+ declare function Overview({
9
+ track,
10
+ progress,
11
+ onStepComplete,
12
+ onStart
13
+ }: CoachingOverviewProps): react_jsx_runtime3.JSX.Element;
14
+ //#endregion
15
+ export { Overview };
16
+ //# sourceMappingURL=Overview.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"Overview.d.ts","names":[],"sources":["../../src/views/Overview.tsx"],"sourcesContent":[],"mappings":";;;;UAgBU,qBAAA,SAA8B;;;AAA9B,iBAIM,QAAA,CAJgB;EAAA,KAAQ;EAAA,QAAA;EAAA,cAAiB;EAAA;AAAA,CAAA,EAStD,qBATsD,CAAA,EASjC,kBAAA,CAAA,GAAA,CAAA,OATiC"}