@salesmind-ai/design-system 0.3.1 → 0.3.2

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (197) hide show
  1. package/dist/admin/index.cjs +68 -2928
  2. package/dist/admin/index.cjs.map +1 -1
  3. package/dist/admin/index.js +5 -2915
  4. package/dist/admin/index.js.map +1 -1
  5. package/dist/blog/index.cjs +53 -1064
  6. package/dist/blog/index.cjs.map +1 -1
  7. package/dist/blog/index.js +8 -1054
  8. package/dist/blog/index.js.map +1 -1
  9. package/dist/charts/index.cjs +46 -2694
  10. package/dist/charts/index.cjs.map +1 -1
  11. package/dist/charts/index.js +3 -2680
  12. package/dist/charts/index.js.map +1 -1
  13. package/dist/chunk-2GARWEJK.js +17 -0
  14. package/dist/chunk-2GARWEJK.js.map +1 -0
  15. package/dist/chunk-3NKRFUAR.js +37 -0
  16. package/dist/chunk-3NKRFUAR.js.map +1 -0
  17. package/dist/chunk-3TGSIILM.cjs +201 -0
  18. package/dist/chunk-3TGSIILM.cjs.map +1 -0
  19. package/dist/chunk-4GM5BGBN.cjs +801 -0
  20. package/dist/chunk-4GM5BGBN.cjs.map +1 -0
  21. package/dist/chunk-5LGDEZWY.cjs +2434 -0
  22. package/dist/chunk-5LGDEZWY.cjs.map +1 -0
  23. package/dist/chunk-6H4DSTXR.js +786 -0
  24. package/dist/chunk-6H4DSTXR.js.map +1 -0
  25. package/dist/chunk-6UNG76Y2.js +153 -0
  26. package/dist/chunk-6UNG76Y2.js.map +1 -0
  27. package/dist/chunk-7PX2AZ6Y.js +39 -0
  28. package/dist/chunk-7PX2AZ6Y.js.map +1 -0
  29. package/dist/chunk-B6AVAX4F.js +1415 -0
  30. package/dist/chunk-B6AVAX4F.js.map +1 -0
  31. package/dist/chunk-BILT5KD3.js +264 -0
  32. package/dist/chunk-BILT5KD3.js.map +1 -0
  33. package/dist/chunk-C2BCDNAV.js +24 -0
  34. package/dist/chunk-C2BCDNAV.js.map +1 -0
  35. package/dist/chunk-CH42VPWE.cjs +421 -0
  36. package/dist/chunk-CH42VPWE.cjs.map +1 -0
  37. package/dist/chunk-CJ2MKVAF.cjs +46 -0
  38. package/dist/chunk-CJ2MKVAF.cjs.map +1 -0
  39. package/dist/chunk-DP74LUXG.cjs +98 -0
  40. package/dist/chunk-DP74LUXG.cjs.map +1 -0
  41. package/dist/chunk-E7D6EKJ4.cjs +44 -0
  42. package/dist/chunk-E7D6EKJ4.cjs.map +1 -0
  43. package/dist/chunk-ECXBTUH6.cjs +584 -0
  44. package/dist/chunk-ECXBTUH6.cjs.map +1 -0
  45. package/dist/chunk-EFRAP5ES.js +157 -0
  46. package/dist/chunk-EFRAP5ES.js.map +1 -0
  47. package/dist/chunk-F6YYWMME.js +485 -0
  48. package/dist/chunk-F6YYWMME.js.map +1 -0
  49. package/dist/chunk-FAFAP4L5.js +183 -0
  50. package/dist/chunk-FAFAP4L5.js.map +1 -0
  51. package/dist/chunk-GUZIMHWS.js +1608 -0
  52. package/dist/chunk-GUZIMHWS.js.map +1 -0
  53. package/dist/chunk-H2Y6BSTL.cjs +69 -0
  54. package/dist/chunk-H2Y6BSTL.cjs.map +1 -0
  55. package/dist/chunk-HN4PHABT.js +126 -0
  56. package/dist/chunk-HN4PHABT.js.map +1 -0
  57. package/dist/chunk-HRENHNDJ.js +211 -0
  58. package/dist/chunk-HRENHNDJ.js.map +1 -0
  59. package/dist/chunk-I75BFEYT.cjs +2561 -0
  60. package/dist/chunk-I75BFEYT.cjs.map +1 -0
  61. package/dist/chunk-IFRATNLU.js +562 -0
  62. package/dist/chunk-IFRATNLU.js.map +1 -0
  63. package/dist/chunk-IYPXJ6YC.cjs +69 -0
  64. package/dist/chunk-IYPXJ6YC.cjs.map +1 -0
  65. package/dist/chunk-JPJN4YBC.js +409 -0
  66. package/dist/chunk-JPJN4YBC.js.map +1 -0
  67. package/dist/chunk-KBA2LFBG.js +62 -0
  68. package/dist/chunk-KBA2LFBG.js.map +1 -0
  69. package/dist/chunk-KCKUSU2M.cjs +166 -0
  70. package/dist/chunk-KCKUSU2M.cjs.map +1 -0
  71. package/dist/chunk-KJ2OXQF4.js +287 -0
  72. package/dist/chunk-KJ2OXQF4.js.map +1 -0
  73. package/dist/chunk-KNQEIU7O.cjs +1202 -0
  74. package/dist/chunk-KNQEIU7O.cjs.map +1 -0
  75. package/dist/chunk-KVGSVGRK.cjs +569 -0
  76. package/dist/chunk-KVGSVGRK.cjs.map +1 -0
  77. package/dist/chunk-L352JRV6.cjs +105 -0
  78. package/dist/chunk-L352JRV6.cjs.map +1 -0
  79. package/dist/chunk-LJADZITX.cjs +298 -0
  80. package/dist/chunk-LJADZITX.cjs.map +1 -0
  81. package/dist/chunk-LMJPWXTZ.cjs +194 -0
  82. package/dist/chunk-LMJPWXTZ.cjs.map +1 -0
  83. package/dist/chunk-LOWEAQST.js +701 -0
  84. package/dist/chunk-LOWEAQST.js.map +1 -0
  85. package/dist/chunk-MDB2WCRQ.cjs +137 -0
  86. package/dist/chunk-MDB2WCRQ.cjs.map +1 -0
  87. package/dist/chunk-MQDEE7HC.cjs +283 -0
  88. package/dist/chunk-MQDEE7HC.cjs.map +1 -0
  89. package/dist/chunk-MQRB634A.cjs +34 -0
  90. package/dist/chunk-MQRB634A.cjs.map +1 -0
  91. package/dist/chunk-MTI27RDV.js +185 -0
  92. package/dist/chunk-MTI27RDV.js.map +1 -0
  93. package/dist/chunk-MU6GW5ZV.js +2317 -0
  94. package/dist/chunk-MU6GW5ZV.js.map +1 -0
  95. package/dist/chunk-NN3TUHIH.js +28 -0
  96. package/dist/chunk-NN3TUHIH.js.map +1 -0
  97. package/dist/chunk-NT4LBP7D.cjs +111 -0
  98. package/dist/chunk-NT4LBP7D.cjs.map +1 -0
  99. package/dist/chunk-OLV7OD3X.cjs +502 -0
  100. package/dist/chunk-OLV7OD3X.cjs.map +1 -0
  101. package/dist/chunk-OXNXEQY7.js +2538 -0
  102. package/dist/chunk-OXNXEQY7.js.map +1 -0
  103. package/dist/chunk-P5BOFE5A.js +546 -0
  104. package/dist/chunk-P5BOFE5A.js.map +1 -0
  105. package/dist/chunk-Q2MFGYTE.cjs +1449 -0
  106. package/dist/chunk-Q2MFGYTE.cjs.map +1 -0
  107. package/dist/chunk-Q75DBVDY.cjs +68 -0
  108. package/dist/chunk-Q75DBVDY.cjs.map +1 -0
  109. package/dist/chunk-REQ5Q6ZI.js +1022 -0
  110. package/dist/chunk-REQ5Q6ZI.js.map +1 -0
  111. package/dist/chunk-SICKWUWB.js +62 -0
  112. package/dist/chunk-SICKWUWB.js.map +1 -0
  113. package/dist/chunk-T343CCH5.js +1190 -0
  114. package/dist/chunk-T343CCH5.js.map +1 -0
  115. package/dist/chunk-TEC62D4A.cjs +1624 -0
  116. package/dist/chunk-TEC62D4A.cjs.map +1 -0
  117. package/dist/chunk-TW5JB35D.js +2122 -0
  118. package/dist/chunk-TW5JB35D.js.map +1 -0
  119. package/dist/chunk-VC5LMUVQ.cjs +20 -0
  120. package/dist/chunk-VC5LMUVQ.cjs.map +1 -0
  121. package/dist/chunk-VM7WFMKI.cjs +76 -0
  122. package/dist/chunk-VM7WFMKI.cjs.map +1 -0
  123. package/dist/chunk-W2WTP6HS.cjs +233 -0
  124. package/dist/chunk-W2WTP6HS.cjs.map +1 -0
  125. package/dist/chunk-WH7PYHZY.cjs +35 -0
  126. package/dist/chunk-WH7PYHZY.cjs.map +1 -0
  127. package/dist/chunk-XQZVY7JJ.cjs +717 -0
  128. package/dist/chunk-XQZVY7JJ.cjs.map +1 -0
  129. package/dist/chunk-XU3OMQ7V.js +98 -0
  130. package/dist/chunk-XU3OMQ7V.js.map +1 -0
  131. package/dist/chunk-XWPDRMZG.js +62 -0
  132. package/dist/chunk-XWPDRMZG.js.map +1 -0
  133. package/dist/chunk-Y3CPKNB7.js +67 -0
  134. package/dist/chunk-Y3CPKNB7.js.map +1 -0
  135. package/dist/chunk-YNVRDD2P.js +98 -0
  136. package/dist/chunk-YNVRDD2P.js.map +1 -0
  137. package/dist/chunk-YSYR54XR.js +92 -0
  138. package/dist/chunk-YSYR54XR.js.map +1 -0
  139. package/dist/chunk-YTYDQBVY.cjs +162 -0
  140. package/dist/chunk-YTYDQBVY.cjs.map +1 -0
  141. package/dist/chunk-ZDLOA2UT.cjs +1042 -0
  142. package/dist/chunk-ZDLOA2UT.cjs.map +1 -0
  143. package/dist/chunk-ZWUKRCOJ.cjs +2162 -0
  144. package/dist/chunk-ZWUKRCOJ.cjs.map +1 -0
  145. package/dist/core/index.cjs +807 -4333
  146. package/dist/core/index.cjs.map +1 -1
  147. package/dist/core/index.js +14 -4130
  148. package/dist/core/index.js.map +1 -1
  149. package/dist/i18n/index.cjs +86 -558
  150. package/dist/i18n/index.cjs.map +1 -1
  151. package/dist/i18n/index.js +1 -544
  152. package/dist/i18n/index.js.map +1 -1
  153. package/dist/index.cjs +1432 -17140
  154. package/dist/index.cjs.map +1 -1
  155. package/dist/index.js +31 -16785
  156. package/dist/index.js.map +1 -1
  157. package/dist/marketing/index.cjs +142 -3072
  158. package/dist/marketing/index.cjs.map +1 -1
  159. package/dist/marketing/index.js +11 -3042
  160. package/dist/marketing/index.js.map +1 -1
  161. package/dist/motion/index.cjs +26 -1222
  162. package/dist/motion/index.cjs.map +1 -1
  163. package/dist/motion/index.js +2 -1215
  164. package/dist/motion/index.js.map +1 -1
  165. package/dist/nav/index.cjs +101 -1518
  166. package/dist/nav/index.cjs.map +1 -1
  167. package/dist/nav/index.js +4 -1498
  168. package/dist/nav/index.js.map +1 -1
  169. package/dist/report/index.cjs +171 -2403
  170. package/dist/report/index.cjs.map +1 -1
  171. package/dist/report/index.js +3 -2363
  172. package/dist/report/index.js.map +1 -1
  173. package/dist/sections/index.cjs +28 -378
  174. package/dist/sections/index.cjs.map +1 -1
  175. package/dist/sections/index.js +4 -372
  176. package/dist/sections/index.js.map +1 -1
  177. package/dist/social-proof/index.cjs +53 -1250
  178. package/dist/social-proof/index.cjs.map +1 -1
  179. package/dist/social-proof/index.js +6 -1235
  180. package/dist/social-proof/index.js.map +1 -1
  181. package/dist/theme/index.cjs +38 -565
  182. package/dist/theme/index.cjs.map +1 -1
  183. package/dist/theme/index.js +2 -555
  184. package/dist/theme/index.js.map +1 -1
  185. package/dist/web/client/index.cjs +38 -491
  186. package/dist/web/client/index.cjs.map +1 -1
  187. package/dist/web/client/index.js +4 -483
  188. package/dist/web/client/index.js.map +1 -1
  189. package/dist/web/index.cjs +158 -1346
  190. package/dist/web/index.cjs.map +1 -1
  191. package/dist/web/index.js +9 -1305
  192. package/dist/web/index.js.map +1 -1
  193. package/dist/web/server/index.cjs +26 -563
  194. package/dist/web/server/index.cjs.map +1 -1
  195. package/dist/web/server/index.js +1 -560
  196. package/dist/web/server/index.js.map +1 -1
  197. package/package.json +1 -1
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../src/report-engine/constants.ts","../src/report-engine/confidence.ts","../src/report-engine/context.tsx","../src/report-engine/chart-types.ts","../src/report-engine/validation.ts","../src/components/ConfidenceIndicator/ConfidenceIndicator.tsx","../src/components/DataCoverageBadge/DataCoverageBadge.tsx","../src/components/DataStateIndicator/DataStateIndicator.tsx"],"names":["React","getComponentName","isChart","jsx","clsx","jsxs"],"mappings":"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","sourcesContent":["/* ============================================================================\n REPORT ENGINE — CONSTANTS & CONFIGURATION\n ============================================================================ */\n\nimport type {\n ReportModeConfig,\n ConfidenceThreshold,\n InsightTypeContract,\n InsightType,\n} from './types';\n\n/**\n * Executive mode configuration - minimal detail, boardroom-ready.\n */\nexport const EXECUTIVE_MODE: ReportModeConfig = {\n mode: 'executive',\n features: {\n showMethodology: false,\n showConfidenceScores: false,\n showRawData: false,\n showStatisticalTests: false,\n chartDetailLevel: 'summary',\n maxInsightsShown: 5,\n allowDrilldown: false,\n },\n};\n\n/**\n * Analyst mode configuration - full detail, internal use.\n */\nexport const ANALYST_MODE: ReportModeConfig = {\n mode: 'analyst',\n features: {\n showMethodology: true,\n showConfidenceScores: true,\n showRawData: true,\n showStatisticalTests: true,\n chartDetailLevel: 'detailed',\n maxInsightsShown: 20,\n allowDrilldown: true,\n },\n};\n\n/**\n * Confidence level thresholds.\n */\nexport const CONFIDENCE_THRESHOLDS: ConfidenceThreshold[] = [\n { level: 'high', min: 90, max: 100, label: 'High Confidence', color: 'var(--status-success)' },\n {\n level: 'moderate',\n min: 70,\n max: 89,\n label: 'Moderate Confidence',\n color: 'var(--status-info)',\n },\n { level: 'low', min: 50, max: 69, label: 'Low Confidence', color: 'var(--status-warning)' },\n {\n level: 'insufficient',\n min: 0,\n max: 49,\n label: 'Insufficient Data',\n color: 'var(--status-error)',\n },\n];\n\n/**\n * Minimum sample sizes for different analysis types.\n */\nexport const MINIMUM_SAMPLE_SIZES = {\n overall: 30,\n segment: 30,\n mbtiType: 50,\n correlation: 100,\n timeSeries: 7,\n} as const;\n\n/**\n * Insight type contracts defining allowed charts and requirements.\n */\nexport const INSIGHT_TYPE_CONTRACTS: Record<InsightType, InsightTypeContract> = {\n 'performance-delta': {\n type: 'performance-delta',\n allowedCharts: ['BarChart', 'LineChart', 'KeyMetric'],\n requiredFields: ['metric_name', 'current_value', 'previous_value', 'time_period'],\n minSampleSize: 30,\n headlinePattern: '{metric} {direction} by {delta}% {timeframe}',\n recommendationFormats: ['maintain-strategy', 'investigate-decline', 'scale-success'],\n },\n 'behavioral-correlation': {\n type: 'behavioral-correlation',\n allowedCharts: ['HeatmapChart', 'BarChart'],\n requiredFields: ['variable_a', 'variable_b', 'correlation_coefficient'],\n minSampleSize: 100,\n headlinePattern: '{variable_a} correlates with {variable_b}',\n recommendationFormats: ['leverage-correlation', 'investigate-causation'],\n },\n 'funnel-leakage': {\n type: 'funnel-leakage',\n allowedCharts: ['FunnelChart', 'BarChart'],\n requiredFields: ['stages', 'stage_counts', 'drop_rates'],\n minSampleSize: 50,\n headlinePattern: '{drop_rate}% drop-off at {stage}',\n recommendationFormats: ['optimize-stage', 'remove-friction'],\n },\n 'audience-mismatch': {\n type: 'audience-mismatch',\n allowedCharts: ['BarChart', 'PieChart'],\n requiredFields: ['target_audience', 'actual_audience', 'overlap_percentage'],\n minSampleSize: 50,\n headlinePattern: 'Target audience differs by {mismatch}%',\n recommendationFormats: ['refine-targeting', 'expand-audience'],\n },\n 'message-fatigue': {\n type: 'message-fatigue',\n allowedCharts: ['LineChart', 'BarChart'],\n requiredFields: ['message_sequence', 'engagement_rates', 'fatigue_point'],\n minSampleSize: 100,\n headlinePattern: 'Engagement drops {drop}% after message {n}',\n recommendationFormats: ['reduce-frequency', 'vary-content'],\n },\n 'psychological-resonance': {\n type: 'psychological-resonance',\n allowedCharts: ['MBTIRadar', 'MBTITypeGrid', 'BarChart'],\n requiredFields: ['mbti_types', 'response_rates', 'message_type'],\n minSampleSize: 50,\n headlinePattern: '{mbti_type} responds {rate}% better to {message_type}',\n recommendationFormats: ['personalize-messaging', 'segment-by-type'],\n },\n 'timing-pattern': {\n type: 'timing-pattern',\n allowedCharts: ['HeatmapChart', 'LineChart'],\n requiredFields: ['time_slots', 'engagement_rates'],\n minSampleSize: 100,\n headlinePattern: 'Best engagement on {day} at {time}',\n recommendationFormats: ['optimize-timing', 'schedule-sends'],\n },\n 'segment-outperformance': {\n type: 'segment-outperformance',\n allowedCharts: ['BarChart', 'KeyMetric'],\n requiredFields: ['segments', 'performance_metrics', 'baseline'],\n minSampleSize: 30,\n headlinePattern: '{segment} outperforms by {delta}%',\n recommendationFormats: ['double-down', 'investigate-success'],\n },\n 'anomaly-detection': {\n type: 'anomaly-detection',\n allowedCharts: ['LineChart', 'BarChart'],\n requiredFields: ['metric', 'expected_value', 'actual_value', 'deviation'],\n minSampleSize: 30,\n headlinePattern: 'Unusual {direction} in {metric}',\n recommendationFormats: ['investigate', 'monitor'],\n },\n 'competitive-benchmark': {\n type: 'competitive-benchmark',\n allowedCharts: ['BarChart', 'KeyMetric'],\n requiredFields: ['metric', 'our_value', 'benchmark_value'],\n minSampleSize: 30,\n headlinePattern: '{position} industry benchmark by {delta}%',\n recommendationFormats: ['maintain-lead', 'close-gap'],\n },\n};\n\n/**\n * Analytical state metadata for UI rendering.\n */\nexport const ANALYTICAL_STATE_CONFIG = {\n VALID: {\n label: 'Valid',\n icon: 'check',\n color: 'var(--status-success)',\n action: null,\n },\n INSUFFICIENT_SAMPLE: {\n label: 'Insufficient Sample',\n icon: 'alert-triangle',\n color: 'var(--status-warning)',\n action: 'Add more data',\n },\n LOW_CONFIDENCE: {\n label: 'Low Confidence',\n icon: 'alert-circle',\n color: 'var(--status-warning)',\n action: 'Interpret with caution',\n },\n PARTIAL_TIME_WINDOW: {\n label: 'Partial Data',\n icon: 'clock',\n color: 'var(--status-info)',\n action: 'Expand date range',\n },\n DATA_SKEW_DETECTED: {\n label: 'Data Skew',\n icon: 'bar-chart-2',\n color: 'var(--status-warning)',\n action: 'Review methodology',\n },\n CONFLICTING_SIGNALS: {\n label: 'Conflicting Signals',\n icon: 'git-merge',\n color: 'var(--status-error)',\n action: 'Manual review needed',\n },\n INFERRED_NOT_OBSERVED: {\n label: 'AI Generated',\n icon: 'cpu',\n color: 'var(--status-info)',\n action: 'Verify with data',\n },\n STALE_DATA: {\n label: 'Stale Data',\n icon: 'clock',\n color: 'var(--status-warning)',\n action: 'Refresh data',\n },\n FIELD_MISSING: {\n label: 'Missing Field',\n icon: 'file-x',\n color: 'var(--status-error)',\n action: 'Upload complete data',\n },\n CALCULATION_ERROR: {\n label: 'Calculation Error',\n icon: 'x-circle',\n color: 'var(--status-error)',\n action: 'Contact support',\n },\n} as const;\n","/* ============================================================================\n REPORT ENGINE — CONFIDENCE SCORING\n ============================================================================ */\n\nimport type { ConfidenceLevel } from './types';\nimport { CONFIDENCE_THRESHOLDS } from './constants';\n\nexport interface ConfidenceParams {\n sampleSize: number;\n requiredSize: number;\n completeness: number; // 0-1: percentage of non-null fields\n recency: number; // 0-1: how recent the data is\n variance: number; // 0-1: normalized variance (higher = worse)\n aiInference: boolean; // Whether AI generated the insight\n}\n\n/**\n * Calculate confidence score based on data quality factors.\n * Returns a score from 0-100.\n */\nexport function calculateConfidence(params: ConfidenceParams): number {\n const { sampleSize, requiredSize, completeness, recency, variance, aiInference } = params;\n\n // Base score from sample size (0-40 points)\n const sampleScore = Math.min(40, (sampleSize / requiredSize) * 40);\n\n // Completeness bonus (0-20 points)\n const completenessScore = completeness * 20;\n\n // Recency bonus (0-20 points)\n const recencyScore = recency * 20;\n\n // Variance penalty (0-10 points deducted for high variance)\n const variancePenalty = variance * 10;\n\n // AI inference penalty (10 points if AI-generated)\n const aiPenalty = aiInference ? 10 : 0;\n\n const rawScore = sampleScore + completenessScore + recencyScore - variancePenalty - aiPenalty;\n\n return Math.max(0, Math.min(100, Math.round(rawScore)));\n}\n\n/**\n * Get confidence level from score.\n */\nexport function getConfidenceLevel(score: number): ConfidenceLevel {\n const threshold = CONFIDENCE_THRESHOLDS.find((t) => score >= t.min && score <= t.max);\n return threshold?.level ?? 'insufficient';\n}\n\n/**\n * Get confidence threshold config from score.\n */\nexport function getConfidenceThreshold(score: number) {\n return (\n CONFIDENCE_THRESHOLDS.find((t) => score >= t.min && score <= t.max) ??\n CONFIDENCE_THRESHOLDS[CONFIDENCE_THRESHOLDS.length - 1]\n );\n}\n\n/**\n * Calculate recency score based on data age.\n * @param dataDate - The date of the most recent data point\n * @param maxAgeDays - Maximum acceptable age in days (default 30)\n */\nexport function calculateRecency(dataDate: Date, maxAgeDays = 30): number {\n const now = new Date();\n const ageInDays = (now.getTime() - dataDate.getTime()) / (1000 * 60 * 60 * 24);\n\n if (ageInDays <= 0) return 1;\n if (ageInDays >= maxAgeDays) return 0;\n\n return 1 - ageInDays / maxAgeDays;\n}\n\n/**\n * Calculate data completeness from an array of records.\n * @param data - Array of data objects\n * @param requiredFields - Fields to check for completeness\n */\nexport function calculateCompleteness(\n data: Record<string, unknown>[],\n requiredFields: string[],\n): number {\n if (data.length === 0 || requiredFields.length === 0) return 0;\n\n let totalFields = 0;\n let filledFields = 0;\n\n for (const record of data) {\n for (const field of requiredFields) {\n totalFields++;\n const value = record[field];\n if (value !== null && value !== undefined && value !== '') {\n filledFields++;\n }\n }\n }\n\n return totalFields > 0 ? filledFields / totalFields : 0;\n}\n\n/**\n * Calculate normalized variance for a numeric dataset.\n * Returns 0-1 where 0 is low variance and 1 is high variance.\n */\nexport function calculateNormalizedVariance(values: number[]): number {\n if (values.length < 2) return 0;\n\n const mean = values.reduce((a, b) => a + b, 0) / values.length;\n const squaredDiffs = values.map((v) => Math.pow(v - mean, 2));\n const variance = squaredDiffs.reduce((a, b) => a + b, 0) / values.length;\n const stdDev = Math.sqrt(variance);\n\n // Coefficient of variation (CV) normalized to 0-1\n // CV > 1 is considered high variance\n const cv = mean !== 0 ? stdDev / Math.abs(mean) : 0;\n\n return Math.min(1, cv);\n}\n","/* ============================================================================\n REPORT ENGINE — CONTEXT & HOOKS\n ============================================================================ */\n\n/* eslint-disable react-refresh/only-export-components */\n// This is a library file exporting hooks alongside providers - fast refresh doesn't apply\n\nimport React, { createContext, useContext, useMemo } from 'react';\nimport type { ReportMode, ReportModeConfig, AnalyticalState, DataStateDetails } from './types';\nimport { EXECUTIVE_MODE, ANALYST_MODE, MINIMUM_SAMPLE_SIZES } from './constants';\nimport { calculateConfidence, getConfidenceLevel, type ConfidenceParams } from './confidence';\n\n/* --------------------------------------------------------------------------\n Report Mode Context\n -------------------------------------------------------------------------- */\n\ninterface ReportModeContextValue extends ReportModeConfig {\n setMode?: (mode: ReportMode) => void;\n}\n\nconst ReportModeContext = createContext<ReportModeContextValue | null>(null);\n\nexport interface ReportModeProviderProps {\n mode?: ReportMode;\n children: React.ReactNode;\n onModeChange?: (mode: ReportMode) => void;\n}\n\n/**\n * Provider for report mode context.\n * Wrap your report with this to control executive vs analyst mode.\n */\nexport function ReportModeProvider({\n mode = 'executive',\n children,\n onModeChange,\n}: ReportModeProviderProps) {\n const config = mode === 'executive' ? EXECUTIVE_MODE : ANALYST_MODE;\n\n const value = useMemo(\n () => ({\n ...config,\n setMode: onModeChange,\n }),\n [config, onModeChange],\n );\n\n return <ReportModeContext.Provider value={value}>{children}</ReportModeContext.Provider>;\n}\n\n/**\n * Hook to access report mode configuration.\n */\nexport function useReportMode(): ReportModeContextValue {\n const context = useContext(ReportModeContext);\n\n if (!context) {\n // Return default executive mode if used outside provider\n return EXECUTIVE_MODE;\n }\n\n return context;\n}\n\n/* --------------------------------------------------------------------------\n Data State Hook\n -------------------------------------------------------------------------- */\n\nexport interface UseDataStateParams {\n sampleSize: number;\n requiredSize?: number;\n completeness?: number;\n recency?: number;\n variance?: number;\n aiInferred?: boolean;\n missingFields?: string[];\n staleDays?: number;\n}\n\nexport interface UseDataStateResult {\n state: AnalyticalState;\n confidence: number;\n confidenceLevel: ReturnType<typeof getConfidenceLevel>;\n details: DataStateDetails;\n isValid: boolean;\n}\n\n/**\n * Hook to compute analytical state from data parameters.\n */\nexport function useDataState(params: UseDataStateParams): UseDataStateResult {\n const {\n sampleSize,\n requiredSize = MINIMUM_SAMPLE_SIZES.overall,\n completeness = 1,\n recency = 1,\n variance = 0,\n aiInferred = false,\n missingFields = [],\n staleDays,\n } = params;\n\n return useMemo(() => {\n // Calculate confidence\n const confidenceParams: ConfidenceParams = {\n sampleSize,\n requiredSize,\n completeness,\n recency,\n variance,\n aiInference: aiInferred,\n };\n\n const confidence = calculateConfidence(confidenceParams);\n const confidenceLevel = getConfidenceLevel(confidence);\n\n // Determine analytical state\n let state: AnalyticalState = 'VALID';\n\n if (missingFields.length > 0) {\n state = 'FIELD_MISSING';\n } else if (sampleSize < requiredSize) {\n state = 'INSUFFICIENT_SAMPLE';\n } else if (staleDays && staleDays > 30) {\n state = 'STALE_DATA';\n } else if (aiInferred) {\n state = 'INFERRED_NOT_OBSERVED';\n } else if (confidence < 50) {\n state = 'LOW_CONFIDENCE';\n } else if (variance > 0.7) {\n state = 'DATA_SKEW_DETECTED';\n } else if (recency < 0.3) {\n state = 'PARTIAL_TIME_WINDOW';\n }\n\n const details: DataStateDetails = {\n sampleSize,\n requiredSize,\n confidence,\n missingFields: missingFields.length > 0 ? missingFields : undefined,\n staleDays,\n };\n\n return {\n state,\n confidence,\n confidenceLevel,\n details,\n isValid: state === 'VALID',\n };\n }, [\n sampleSize,\n requiredSize,\n completeness,\n recency,\n variance,\n aiInferred,\n missingFields,\n staleDays,\n ]);\n}\n\n/* --------------------------------------------------------------------------\n Chart Data Validation Hook\n -------------------------------------------------------------------------- */\n\nexport interface UseChartDataParams<T> {\n data: T[];\n requiredFields: string[];\n minSampleSize?: number;\n insightType?: string;\n}\n\nexport interface UseChartDataResult<T> {\n data: T[];\n state: AnalyticalState;\n confidence: number;\n isRenderable: boolean;\n message?: string;\n}\n\n/**\n * Hook to validate chart data against data contracts.\n */\nexport function useChartData<T extends Record<string, unknown>>({\n data,\n requiredFields,\n minSampleSize = MINIMUM_SAMPLE_SIZES.overall,\n}: UseChartDataParams<T>): UseChartDataResult<T> {\n return useMemo(() => {\n // Check for missing required fields\n const missingFields: string[] = [];\n if (data.length > 0) {\n const firstRecord = data[0];\n for (const field of requiredFields) {\n if (!(field in firstRecord)) {\n missingFields.push(field);\n }\n }\n }\n\n // Calculate completeness\n let completeness = 1;\n if (data.length > 0 && requiredFields.length > 0) {\n let filled = 0;\n let total = 0;\n for (const record of data) {\n for (const field of requiredFields) {\n total++;\n const value = record[field];\n if (value !== null && value !== undefined && value !== '') {\n filled++;\n }\n }\n }\n completeness = total > 0 ? filled / total : 0;\n }\n\n // Determine state\n let state: AnalyticalState = 'VALID';\n let message: string | undefined;\n\n if (missingFields.length > 0) {\n state = 'FIELD_MISSING';\n message = `Missing fields: ${missingFields.join(', ')}`;\n } else if (data.length === 0) {\n state = 'INSUFFICIENT_SAMPLE';\n message = 'No data available';\n } else if (data.length < minSampleSize) {\n state = 'INSUFFICIENT_SAMPLE';\n message = `Sample size ${data.length} is below minimum ${minSampleSize}`;\n } else if (completeness < 0.7) {\n state = 'LOW_CONFIDENCE';\n message = `Data completeness is ${Math.round(completeness * 100)}%`;\n }\n\n // Calculate confidence\n const confidence = calculateConfidence({\n sampleSize: data.length,\n requiredSize: minSampleSize,\n completeness,\n recency: 1, // Assume fresh data if not specified\n variance: 0,\n aiInference: false,\n });\n\n const isRenderable = state === 'VALID' || state === 'LOW_CONFIDENCE';\n\n return {\n data,\n state,\n confidence,\n isRenderable,\n message,\n };\n }, [data, requiredFields, minSampleSize]);\n}\n","/* ============================================================================\n CHART ABSTRACTION LAYER — TYPE DEFINITIONS & CONTRACTS\n \n This module defines the governed chart system with:\n - Chart taxonomy (semantic families)\n - Data contracts per chart type\n - Switching rules\n - Guardrails and fallback behaviors\n ============================================================================ */\n\n/* ============================================================================\n CHART INTENT (Semantic Families)\n ============================================================================ */\n\n/**\n * Chart intent defines the semantic purpose of visualization.\n * This is the primary classifier that governs switching rules.\n */\nexport type ChartIntent =\n | 'trend' // Time-based evolution, deltas, momentum\n | 'comparison' // Comparing discrete categories\n | 'composition' // Proportions and distributions\n | 'distribution' // Statistical spread and density\n | 'relationship' // Correlation and clustering\n | 'geo'; // Geographic data\n\n/* ============================================================================\n CHART TYPE TAXONOMY\n ============================================================================ */\n\n/**\n * All supported chart types organized by intent.\n */\n\n// Trend charts (time-based)\nexport type TrendChartType = 'line' | 'smooth_line' | 'area' | 'stacked_area' | 'area_100pct';\n\n// Comparison charts (categorical)\nexport type ComparisonChartType =\n | 'bar'\n | 'column'\n | 'stacked_bar'\n | 'stacked_column'\n | 'column_100pct'\n | 'combo'; // Phase 2\n\n// Composition charts\nexport type CompositionChartType = 'pie' | 'donut' | 'treemap'; // Phase 2\n\n// Distribution charts\nexport type DistributionChartType = 'histogram' | 'radar'; // Phase 2\n\n// Relationship charts\nexport type RelationshipChartType =\n | 'scatter' // Phase 2\n | 'bubble' // Phase 2\n | 'heatmap';\n\n// Geographic charts\nexport type GeoChartType =\n | 'geo' // Phase 2\n | 'choropleth'; // Phase 2\n\n/**\n * Union of all chart types\n */\nexport type ChartTypeId =\n | TrendChartType\n | ComparisonChartType\n | CompositionChartType\n | DistributionChartType\n | RelationshipChartType\n | GeoChartType\n | 'funnel' // Special: conversion flows\n | 'waterfall' // Phase 2: sequential changes\n | 'timeline'; // Phase 2: event sequences\n\n/**\n * Chart phase classification\n */\nexport type ChartPhase = 'mvp' | 'phase2' | 'phase3';\n\n/* ============================================================================\n CHART METADATA & REGISTRY\n ============================================================================ */\n\nexport interface ChartTypeMetadata {\n id: ChartTypeId;\n name: string;\n description: string;\n intent: ChartIntent;\n phase: ChartPhase;\n\n // Data requirements\n contract: ChartDataContract;\n\n // Allowed switches within same intent\n switchableTo: ChartTypeId[];\n\n // Visual configuration\n supportsStacking: boolean;\n supportsMultipleSeries: boolean;\n requiresTimeAxis: boolean;\n maxCategories?: number;\n maxSlices?: number;\n}\n\nexport interface ChartDataContract {\n // Required data shape\n requiredDimensions: number; // e.g., 1 for pie, 2 for scatter\n requiredMeasures: number; // Number of numeric fields required\n\n // Sample size requirements\n minDataPoints: number;\n idealDataPoints: number;\n maxDataPoints?: number;\n\n // Value constraints\n allowNegativeValues: boolean;\n requireNonZeroTotal: boolean;\n\n // Warnings\n warningThresholds: {\n tooManyCategories?: number;\n imbalancedRatio?: number;\n sparseData?: number;\n };\n\n // Fallback behavior\n fallback: ChartFallbackBehavior;\n}\n\nexport type ChartFallbackBehavior =\n | { type: 'hide' }\n | { type: 'show_kpi'; message: string }\n | { type: 'aggregate'; strategy: 'other_bucket' | 'top_n' }\n | { type: 'warn'; message: string };\n\n/* ============================================================================\n CHART REGISTRY (Canonical List)\n ============================================================================ */\n\nexport const CHART_REGISTRY: Record<ChartTypeId, ChartTypeMetadata> = {\n // ══════════════════════════════════════════════════════════════════════════\n // TREND CHARTS (MVP)\n // ══════════════════════════════════════════════════════════════════════════\n line: {\n id: 'line',\n name: 'Line Chart',\n description: 'Shows trends over time with connected data points',\n intent: 'trend',\n phase: 'mvp',\n contract: {\n requiredDimensions: 1,\n requiredMeasures: 1,\n minDataPoints: 3,\n idealDataPoints: 7,\n allowNegativeValues: true,\n requireNonZeroTotal: false,\n warningThresholds: { sparseData: 3 },\n fallback: { type: 'show_kpi', message: 'Insufficient data for trend visualization' },\n },\n switchableTo: ['smooth_line', 'area', 'stacked_area'],\n supportsStacking: false,\n supportsMultipleSeries: true,\n requiresTimeAxis: true,\n },\n\n smooth_line: {\n id: 'smooth_line',\n name: 'Smooth Line Chart',\n description: 'Curved line chart for smoother trend visualization',\n intent: 'trend',\n phase: 'mvp',\n contract: {\n requiredDimensions: 1,\n requiredMeasures: 1,\n minDataPoints: 3,\n idealDataPoints: 7,\n allowNegativeValues: true,\n requireNonZeroTotal: false,\n warningThresholds: { sparseData: 3 },\n fallback: { type: 'show_kpi', message: 'Insufficient data for trend visualization' },\n },\n switchableTo: ['line', 'area', 'stacked_area'],\n supportsStacking: false,\n supportsMultipleSeries: true,\n requiresTimeAxis: true,\n },\n\n area: {\n id: 'area',\n name: 'Area Chart',\n description: 'Filled area under line, emphasizes volume over time',\n intent: 'trend',\n phase: 'mvp',\n contract: {\n requiredDimensions: 1,\n requiredMeasures: 1,\n minDataPoints: 3,\n idealDataPoints: 7,\n allowNegativeValues: false,\n requireNonZeroTotal: false,\n warningThresholds: { sparseData: 3 },\n fallback: { type: 'show_kpi', message: 'Insufficient data for area visualization' },\n },\n switchableTo: ['line', 'smooth_line', 'stacked_area', 'area_100pct'],\n supportsStacking: true,\n supportsMultipleSeries: true,\n requiresTimeAxis: true,\n },\n\n stacked_area: {\n id: 'stacked_area',\n name: 'Stacked Area Chart',\n description: 'Multiple series stacked to show cumulative trends',\n intent: 'trend',\n phase: 'mvp',\n contract: {\n requiredDimensions: 1,\n requiredMeasures: 2,\n minDataPoints: 3,\n idealDataPoints: 7,\n allowNegativeValues: false,\n requireNonZeroTotal: true,\n warningThresholds: { sparseData: 3 },\n fallback: { type: 'warn', message: 'Stacked area requires non-negative values' },\n },\n switchableTo: ['line', 'area', 'area_100pct'],\n supportsStacking: true,\n supportsMultipleSeries: true,\n requiresTimeAxis: true,\n },\n\n area_100pct: {\n id: 'area_100pct',\n name: '100% Stacked Area',\n description: 'Shows proportional contribution over time',\n intent: 'trend',\n phase: 'mvp',\n contract: {\n requiredDimensions: 1,\n requiredMeasures: 2,\n minDataPoints: 3,\n idealDataPoints: 7,\n allowNegativeValues: false,\n requireNonZeroTotal: true,\n warningThresholds: { sparseData: 3 },\n fallback: { type: 'warn', message: '100% stacked requires positive totals' },\n },\n switchableTo: ['stacked_area', 'area'],\n supportsStacking: true,\n supportsMultipleSeries: true,\n requiresTimeAxis: true,\n },\n\n // ══════════════════════════════════════════════════════════════════════════\n // COMPARISON CHARTS (MVP)\n // ══════════════════════════════════════════════════════════════════════════\n bar: {\n id: 'bar',\n name: 'Horizontal Bar Chart',\n description: 'Horizontal bars for comparing categories',\n intent: 'comparison',\n phase: 'mvp',\n contract: {\n requiredDimensions: 1,\n requiredMeasures: 1,\n minDataPoints: 2,\n idealDataPoints: 5,\n maxDataPoints: 15,\n allowNegativeValues: true,\n requireNonZeroTotal: false,\n warningThresholds: { tooManyCategories: 12 },\n fallback: { type: 'aggregate', strategy: 'top_n' },\n },\n switchableTo: ['column', 'stacked_bar', 'stacked_column'],\n supportsStacking: true,\n supportsMultipleSeries: true,\n requiresTimeAxis: false,\n },\n\n column: {\n id: 'column',\n name: 'Vertical Column Chart',\n description: 'Vertical bars for comparing categories',\n intent: 'comparison',\n phase: 'mvp',\n contract: {\n requiredDimensions: 1,\n requiredMeasures: 1,\n minDataPoints: 2,\n idealDataPoints: 5,\n maxDataPoints: 12,\n allowNegativeValues: true,\n requireNonZeroTotal: false,\n warningThresholds: { tooManyCategories: 10 },\n fallback: { type: 'aggregate', strategy: 'top_n' },\n },\n switchableTo: ['bar', 'stacked_column', 'stacked_bar'],\n supportsStacking: true,\n supportsMultipleSeries: true,\n requiresTimeAxis: false,\n },\n\n stacked_bar: {\n id: 'stacked_bar',\n name: 'Stacked Horizontal Bar',\n description: 'Horizontal stacked bars showing composition per category',\n intent: 'comparison',\n phase: 'mvp',\n contract: {\n requiredDimensions: 1,\n requiredMeasures: 2,\n minDataPoints: 2,\n idealDataPoints: 5,\n maxDataPoints: 10,\n allowNegativeValues: false,\n requireNonZeroTotal: true,\n warningThresholds: { tooManyCategories: 8, imbalancedRatio: 0.9 },\n fallback: { type: 'aggregate', strategy: 'other_bucket' },\n },\n switchableTo: ['bar', 'stacked_column', 'column_100pct'],\n supportsStacking: true,\n supportsMultipleSeries: true,\n requiresTimeAxis: false,\n },\n\n stacked_column: {\n id: 'stacked_column',\n name: 'Stacked Vertical Column',\n description: 'Vertical stacked columns showing composition per category',\n intent: 'comparison',\n phase: 'mvp',\n contract: {\n requiredDimensions: 1,\n requiredMeasures: 2,\n minDataPoints: 2,\n idealDataPoints: 5,\n maxDataPoints: 10,\n allowNegativeValues: false,\n requireNonZeroTotal: true,\n warningThresholds: { tooManyCategories: 8, imbalancedRatio: 0.9 },\n fallback: { type: 'aggregate', strategy: 'other_bucket' },\n },\n switchableTo: ['column', 'stacked_bar', 'column_100pct'],\n supportsStacking: true,\n supportsMultipleSeries: true,\n requiresTimeAxis: false,\n },\n\n column_100pct: {\n id: 'column_100pct',\n name: '100% Stacked Column',\n description: 'Shows proportional composition per category',\n intent: 'comparison',\n phase: 'mvp',\n contract: {\n requiredDimensions: 1,\n requiredMeasures: 2,\n minDataPoints: 2,\n idealDataPoints: 5,\n maxDataPoints: 10,\n allowNegativeValues: false,\n requireNonZeroTotal: true,\n warningThresholds: { tooManyCategories: 8 },\n fallback: { type: 'warn', message: '100% stacked requires positive totals' },\n },\n switchableTo: ['stacked_column', 'stacked_bar'],\n supportsStacking: true,\n supportsMultipleSeries: true,\n requiresTimeAxis: false,\n },\n\n combo: {\n id: 'combo',\n name: 'Combo Chart',\n description: 'Combines bars with line overlay (Phase 2)',\n intent: 'comparison',\n phase: 'phase2',\n contract: {\n requiredDimensions: 1,\n requiredMeasures: 2,\n minDataPoints: 3,\n idealDataPoints: 6,\n allowNegativeValues: true,\n requireNonZeroTotal: false,\n warningThresholds: { tooManyCategories: 10 },\n fallback: { type: 'warn', message: 'Combo chart requires dual measures' },\n },\n switchableTo: ['column', 'line'],\n supportsStacking: false,\n supportsMultipleSeries: true,\n requiresTimeAxis: false,\n },\n\n // ══════════════════════════════════════════════════════════════════════════\n // COMPOSITION CHARTS (MVP)\n // ══════════════════════════════════════════════════════════════════════════\n pie: {\n id: 'pie',\n name: 'Pie Chart',\n description: 'Shows proportional composition of a whole',\n intent: 'composition',\n phase: 'mvp',\n contract: {\n requiredDimensions: 1,\n requiredMeasures: 1,\n minDataPoints: 2,\n idealDataPoints: 4,\n maxDataPoints: 6,\n allowNegativeValues: false,\n requireNonZeroTotal: true,\n warningThresholds: { tooManyCategories: 6, imbalancedRatio: 0.95 },\n fallback: { type: 'aggregate', strategy: 'other_bucket' },\n },\n switchableTo: ['donut', 'column_100pct'],\n supportsStacking: false,\n supportsMultipleSeries: false,\n requiresTimeAxis: false,\n maxSlices: 6,\n },\n\n donut: {\n id: 'donut',\n name: 'Donut Chart',\n description: 'Pie chart with center cutout, can display total',\n intent: 'composition',\n phase: 'mvp',\n contract: {\n requiredDimensions: 1,\n requiredMeasures: 1,\n minDataPoints: 2,\n idealDataPoints: 4,\n maxDataPoints: 6,\n allowNegativeValues: false,\n requireNonZeroTotal: true,\n warningThresholds: { tooManyCategories: 6, imbalancedRatio: 0.95 },\n fallback: { type: 'aggregate', strategy: 'other_bucket' },\n },\n switchableTo: ['pie', 'column_100pct'],\n supportsStacking: false,\n supportsMultipleSeries: false,\n requiresTimeAxis: false,\n maxSlices: 6,\n },\n\n treemap: {\n id: 'treemap',\n name: 'Treemap',\n description: 'Nested rectangles showing hierarchical composition (Phase 2)',\n intent: 'composition',\n phase: 'phase2',\n contract: {\n requiredDimensions: 1,\n requiredMeasures: 1,\n minDataPoints: 3,\n idealDataPoints: 10,\n maxDataPoints: 50,\n allowNegativeValues: false,\n requireNonZeroTotal: true,\n warningThresholds: { tooManyCategories: 30 },\n fallback: { type: 'aggregate', strategy: 'top_n' },\n },\n switchableTo: ['pie', 'donut'],\n supportsStacking: false,\n supportsMultipleSeries: false,\n requiresTimeAxis: false,\n },\n\n // ══════════════════════════════════════════════════════════════════════════\n // DISTRIBUTION CHARTS (MVP)\n // ══════════════════════════════════════════════════════════════════════════\n histogram: {\n id: 'histogram',\n name: 'Histogram',\n description: 'Shows distribution of values across bins',\n intent: 'distribution',\n phase: 'mvp',\n contract: {\n requiredDimensions: 0,\n requiredMeasures: 1,\n minDataPoints: 10,\n idealDataPoints: 50,\n allowNegativeValues: true,\n requireNonZeroTotal: false,\n warningThresholds: { sparseData: 10 },\n fallback: { type: 'show_kpi', message: 'Insufficient data for distribution' },\n },\n switchableTo: ['radar'],\n supportsStacking: false,\n supportsMultipleSeries: false,\n requiresTimeAxis: false,\n },\n\n radar: {\n id: 'radar',\n name: 'Radar Chart',\n description: 'Multi-dimensional comparison in radial layout (Phase 2)',\n intent: 'distribution',\n phase: 'phase2',\n contract: {\n requiredDimensions: 1,\n requiredMeasures: 3,\n minDataPoints: 3,\n idealDataPoints: 6,\n maxDataPoints: 8,\n allowNegativeValues: false,\n requireNonZeroTotal: false,\n warningThresholds: { tooManyCategories: 8 },\n fallback: { type: 'warn', message: 'Radar works best with 3-8 dimensions' },\n },\n switchableTo: ['histogram'],\n supportsStacking: false,\n supportsMultipleSeries: true,\n requiresTimeAxis: false,\n },\n\n // ══════════════════════════════════════════════════════════════════════════\n // RELATIONSHIP CHARTS\n // ══════════════════════════════════════════════════════════════════════════\n heatmap: {\n id: 'heatmap',\n name: 'Heatmap',\n description: 'Color-coded matrix showing intensity',\n intent: 'relationship',\n phase: 'mvp',\n contract: {\n requiredDimensions: 2,\n requiredMeasures: 1,\n minDataPoints: 9,\n idealDataPoints: 25,\n allowNegativeValues: true,\n requireNonZeroTotal: false,\n warningThresholds: { sparseData: 9 },\n fallback: { type: 'warn', message: 'Heatmap requires grid data' },\n },\n switchableTo: ['scatter'],\n supportsStacking: false,\n supportsMultipleSeries: false,\n requiresTimeAxis: false,\n },\n\n scatter: {\n id: 'scatter',\n name: 'Scatter Plot',\n description: 'Shows correlation between two variables (Phase 2)',\n intent: 'relationship',\n phase: 'phase2',\n contract: {\n requiredDimensions: 0,\n requiredMeasures: 2,\n minDataPoints: 20,\n idealDataPoints: 100,\n allowNegativeValues: true,\n requireNonZeroTotal: false,\n warningThresholds: { sparseData: 20 },\n fallback: { type: 'show_kpi', message: 'Scatter requires 20+ points' },\n },\n switchableTo: ['bubble', 'heatmap'],\n supportsStacking: false,\n supportsMultipleSeries: true,\n requiresTimeAxis: false,\n },\n\n bubble: {\n id: 'bubble',\n name: 'Bubble Chart',\n description: 'Scatter with sized points for third dimension (Phase 2)',\n intent: 'relationship',\n phase: 'phase2',\n contract: {\n requiredDimensions: 0,\n requiredMeasures: 3,\n minDataPoints: 10,\n idealDataPoints: 50,\n allowNegativeValues: true,\n requireNonZeroTotal: false,\n warningThresholds: { sparseData: 10 },\n fallback: { type: 'show_kpi', message: 'Bubble requires 3 measures' },\n },\n switchableTo: ['scatter'],\n supportsStacking: false,\n supportsMultipleSeries: true,\n requiresTimeAxis: false,\n },\n\n // ══════════════════════════════════════════════════════════════════════════\n // GEOGRAPHIC CHARTS (Phase 2)\n // ══════════════════════════════════════════════════════════════════════════\n geo: {\n id: 'geo',\n name: 'Geographic Map',\n description: 'Map with regional data markers (Phase 2)',\n intent: 'geo',\n phase: 'phase2',\n contract: {\n requiredDimensions: 1, // geo dimension (country/region)\n requiredMeasures: 1,\n minDataPoints: 1,\n idealDataPoints: 10,\n allowNegativeValues: true,\n requireNonZeroTotal: false,\n warningThresholds: {},\n fallback: { type: 'hide' },\n },\n switchableTo: ['choropleth'],\n supportsStacking: false,\n supportsMultipleSeries: false,\n requiresTimeAxis: false,\n },\n\n choropleth: {\n id: 'choropleth',\n name: 'Choropleth Map',\n description: 'Color-shaded regions by value (Phase 2)',\n intent: 'geo',\n phase: 'phase2',\n contract: {\n requiredDimensions: 1,\n requiredMeasures: 1,\n minDataPoints: 3,\n idealDataPoints: 20,\n allowNegativeValues: false,\n requireNonZeroTotal: false,\n warningThresholds: {},\n fallback: { type: 'hide' },\n },\n switchableTo: ['geo'],\n supportsStacking: false,\n supportsMultipleSeries: false,\n requiresTimeAxis: false,\n },\n\n // ══════════════════════════════════════════════════════════════════════════\n // SPECIAL CHARTS\n // ══════════════════════════════════════════════════════════════════════════\n funnel: {\n id: 'funnel',\n name: 'Funnel Chart',\n description: 'Shows conversion through sequential stages',\n intent: 'comparison',\n phase: 'mvp',\n contract: {\n requiredDimensions: 1,\n requiredMeasures: 1,\n minDataPoints: 3,\n idealDataPoints: 5,\n maxDataPoints: 7,\n allowNegativeValues: false,\n requireNonZeroTotal: true,\n warningThresholds: { tooManyCategories: 7 },\n fallback: { type: 'warn', message: 'Funnel requires sequential stages' },\n },\n switchableTo: ['bar'],\n supportsStacking: false,\n supportsMultipleSeries: false,\n requiresTimeAxis: false,\n },\n\n waterfall: {\n id: 'waterfall',\n name: 'Waterfall Chart',\n description: 'Shows cumulative effect of sequential changes (Phase 2)',\n intent: 'comparison',\n phase: 'phase2',\n contract: {\n requiredDimensions: 1,\n requiredMeasures: 1,\n minDataPoints: 3,\n idealDataPoints: 6,\n maxDataPoints: 12,\n allowNegativeValues: true,\n requireNonZeroTotal: false,\n warningThresholds: { tooManyCategories: 10 },\n fallback: { type: 'warn', message: 'Waterfall requires sequential data' },\n },\n switchableTo: ['bar', 'column'],\n supportsStacking: false,\n supportsMultipleSeries: false,\n requiresTimeAxis: false,\n },\n\n timeline: {\n id: 'timeline',\n name: 'Timeline',\n description: 'Shows events along time axis (Phase 2)',\n intent: 'trend',\n phase: 'phase2',\n contract: {\n requiredDimensions: 1,\n requiredMeasures: 0,\n minDataPoints: 2,\n idealDataPoints: 10,\n allowNegativeValues: false,\n requireNonZeroTotal: false,\n warningThresholds: {},\n fallback: { type: 'hide' },\n },\n switchableTo: ['line'],\n supportsStacking: false,\n supportsMultipleSeries: false,\n requiresTimeAxis: true,\n },\n};\n\n/* ============================================================================\n CHART FAMILY SWITCHING RULES\n ============================================================================ */\n\n/**\n * Maps intent to allowed chart types for switching.\n * Users can ONLY switch within the same intent family.\n */\nexport const INTENT_CHART_FAMILIES: Record<ChartIntent, ChartTypeId[]> = {\n trend: ['line', 'smooth_line', 'area', 'stacked_area', 'area_100pct'],\n comparison: ['bar', 'column', 'stacked_bar', 'stacked_column', 'column_100pct', 'funnel'],\n composition: ['pie', 'donut'],\n distribution: ['histogram'],\n relationship: ['heatmap'],\n geo: [],\n};\n\n/**\n * Get allowed chart types for switching from a given type.\n */\nexport function getAllowedSwitchTargets(currentType: ChartTypeId): ChartTypeId[] {\n const metadata = CHART_REGISTRY[currentType];\n if (!metadata) return [];\n\n // Only return MVP charts that are in the same family\n return metadata.switchableTo.filter((type) => {\n const targetMeta = CHART_REGISTRY[type];\n return targetMeta && targetMeta.phase === 'mvp';\n });\n}\n\n/**\n * Check if switching from one chart type to another is allowed.\n */\nexport function canSwitchChart(from: ChartTypeId, to: ChartTypeId): boolean {\n const fromMeta = CHART_REGISTRY[from];\n const toMeta = CHART_REGISTRY[to];\n\n if (!fromMeta || !toMeta) return false;\n\n // Must be same intent\n if (fromMeta.intent !== toMeta.intent) return false;\n\n // Must be in allowed list\n return fromMeta.switchableTo.includes(to);\n}\n\n/* ============================================================================\n DATA VALIDATION\n ============================================================================ */\n\nexport interface ChartValidationResult {\n isValid: boolean;\n warnings: string[];\n errors: string[];\n suggestedType?: ChartTypeId;\n fallbackAction?: ChartFallbackBehavior;\n}\n\n/**\n * Validate data against a chart's contract.\n */\nexport function validateChartData(\n chartType: ChartTypeId,\n data: unknown[],\n dimensions: string[],\n measures: string[],\n): ChartValidationResult {\n const metadata = CHART_REGISTRY[chartType];\n if (!metadata) {\n return {\n isValid: false,\n warnings: [],\n errors: [`Unknown chart type: ${chartType}`],\n };\n }\n\n const contract = metadata.contract;\n const warnings: string[] = [];\n const errors: string[] = [];\n\n // Check data points\n if (data.length < contract.minDataPoints) {\n errors.push(`Requires at least ${contract.minDataPoints} data points, got ${data.length}`);\n }\n\n if (contract.maxDataPoints && data.length > contract.maxDataPoints) {\n warnings.push(`Chart works best with ≤${contract.maxDataPoints} items. Consider aggregating.`);\n }\n\n // Check dimensions\n if (dimensions.length < contract.requiredDimensions) {\n errors.push(`Requires ${contract.requiredDimensions} dimension(s), got ${dimensions.length}`);\n }\n\n // Check measures\n if (measures.length < contract.requiredMeasures) {\n errors.push(`Requires ${contract.requiredMeasures} measure(s), got ${measures.length}`);\n }\n\n // Check for negative values if not allowed\n if (!contract.allowNegativeValues && data.length > 0) {\n const hasNegative = data.some((row: unknown) => {\n if (typeof row !== 'object' || row === null) return false;\n return measures.some((m) => {\n const val = (row as Record<string, unknown>)[m];\n return typeof val === 'number' && val < 0;\n });\n });\n if (hasNegative) {\n errors.push('This chart type does not support negative values');\n }\n }\n\n // Check warnings\n if (\n contract.warningThresholds.tooManyCategories &&\n data.length > contract.warningThresholds.tooManyCategories\n ) {\n warnings.push(`Too many categories (${data.length}). Consider grouping smaller values.`);\n }\n\n return {\n isValid: errors.length === 0,\n warnings,\n errors,\n fallbackAction: errors.length > 0 ? contract.fallback : undefined,\n };\n}\n\n/* ============================================================================\n CHART SELECTION HELPERS\n ============================================================================ */\n\n/**\n * Get MVP charts for a given intent.\n */\nexport function getMVPChartsForIntent(intent: ChartIntent): ChartTypeMetadata[] {\n return Object.values(CHART_REGISTRY).filter(\n (meta) => meta.intent === intent && meta.phase === 'mvp',\n );\n}\n\n/**\n * Get the default chart type for an intent.\n */\nexport function getDefaultChartForIntent(intent: ChartIntent): ChartTypeId {\n const defaults: Record<ChartIntent, ChartTypeId> = {\n trend: 'line',\n comparison: 'column',\n composition: 'donut',\n distribution: 'histogram',\n relationship: 'heatmap',\n geo: 'geo',\n };\n return defaults[intent];\n}\n\n/**\n * Suggest the best chart type for given data characteristics.\n */\nexport function suggestChartType(params: {\n hasTimeAxis: boolean;\n dimensionCount: number;\n measureCount: number;\n dataPointCount: number;\n hasNegativeValues: boolean;\n isSequential: boolean;\n}): ChartTypeId {\n const {\n hasTimeAxis,\n dimensionCount,\n measureCount,\n dataPointCount,\n hasNegativeValues,\n isSequential,\n } = params;\n\n // Time-based → trend charts\n if (hasTimeAxis && measureCount >= 1) {\n if (measureCount > 1 && !hasNegativeValues) return 'stacked_area';\n return 'line';\n }\n\n // Sequential stages → funnel\n if (isSequential && !hasNegativeValues && dataPointCount <= 7) {\n return 'funnel';\n }\n\n // Single measure, few categories → composition\n if (measureCount === 1 && dataPointCount <= 6 && !hasNegativeValues) {\n return 'donut';\n }\n\n // Two dimensions → heatmap\n if (dimensionCount === 2 && measureCount === 1) {\n return 'heatmap';\n }\n\n // Default to column chart\n return 'column';\n}\n","/* ============================================================================\n REPORT ENGINE — COMPOSITION VALIDATION\n ============================================================================ */\n\nimport React from 'react';\n\nexport interface ValidationIssue {\n ruleId: string;\n severity: 'error' | 'warning';\n message: string;\n component?: string;\n}\n\nexport interface ReportValidationResult {\n isValid: boolean;\n issues: ValidationIssue[];\n}\n\n/**\n * Validates the structure of a report against narrative rules.\n *\n * Rules:\n * - R1: ExecutiveThesis MUST appear exactly once, at the top (or very near top)\n * - R2: InsightBlock MUST precede any chart it references (Charts must be inside or after insights)\n * - R3: Charts CANNOT appear without a parent InsightBlock (Orphan charts)\n * - R6: Max 3 charts per InsightBlock\n */\nexport function validateReportStructure(children: React.ReactNode): ReportValidationResult {\n const issues: ValidationIssue[] = [];\n const childArray = React.Children.toArray(children);\n\n let hasThesis = false;\n let thesisIndex = -1;\n\n // Helper to get component name safely\n const getComponentName = (child: React.ReactNode): string | undefined => {\n if (!React.isValidElement(child)) return undefined;\n\n const type = child.type as { displayName?: string; name?: string };\n return type.displayName || type.name;\n };\n\n const isChart = (name?: string) => {\n return (\n name &&\n (name.endsWith('Chart') ||\n name === 'MBTIRadar' ||\n name === 'MBTITypeGrid' ||\n name === 'KeyMetric') // Metric is evidence too\n );\n };\n\n // First pass: Top-level analysis\n childArray.forEach((child, index) => {\n const name = getComponentName(child);\n\n if (name === 'ExecutiveThesis') {\n if (hasThesis) {\n issues.push({\n ruleId: 'R1',\n severity: 'error',\n message: 'Only one ExecutiveThesis is allowed per report.',\n component: 'ExecutiveThesis',\n });\n }\n hasThesis = true;\n thesisIndex = index;\n }\n\n // Check for orphan charts at root level\n if (isChart(name)) {\n issues.push({\n ruleId: 'R3',\n severity: 'error',\n message: `Orphan chart detected: <${name} />. Charts must be placed inside an InsightBlock or EvidenceGroup.`,\n component: name,\n });\n }\n\n // Check InsightBlock constraints\n if (name === 'InsightBlock' && React.isValidElement(child)) {\n validateInsightBlock(child, issues);\n }\n });\n\n // R1: Check Thesis existence and position\n if (!hasThesis) {\n issues.push({\n ruleId: 'R1',\n severity: 'error',\n message: 'Report must contain an ExecutiveThesis.',\n });\n } else if (thesisIndex > 2) {\n // Allow some spacers/headers before it\n issues.push({\n ruleId: 'R1',\n severity: 'warning',\n message: 'ExecutiveThesis should be the first major component in the report.',\n });\n }\n\n return {\n isValid: issues.filter((i) => i.severity === 'error').length === 0,\n issues,\n };\n}\n\nfunction validateInsightBlock(node: React.ReactElement, issues: ValidationIssue[]) {\n const children = node.props.children;\n const childArray = React.Children.toArray(children);\n\n // Count charts inside this block (recursive check not strictly needed if structure is flat)\n let chartsInBlock = 0;\n\n const countCharts = (nodes: React.ReactNode[]) => {\n nodes.forEach((n) => {\n const name = getComponentName(n);\n if (isChart(name)) {\n chartsInBlock++;\n }\n // Check inside EvidenceGroup\n if (name === 'EvidenceGroup' && React.isValidElement(n)) {\n countCharts(React.Children.toArray(n.props.children));\n }\n });\n };\n\n countCharts(childArray);\n\n // R6: Max 3 charts\n if (chartsInBlock > 3) {\n issues.push({\n ruleId: 'R6',\n severity: 'warning',\n message: `InsightBlock contains ${chartsInBlock} charts. Recommended maximum is 3 to prevent data dumping.`,\n component: 'InsightBlock',\n });\n }\n}\n\n// Duplicate helper for local scope\nconst getComponentName = (child: React.ReactNode): string | undefined => {\n if (!React.isValidElement(child)) return undefined;\n const type = child.type as { displayName?: string; name?: string };\n return type.displayName || type.name;\n};\n\nconst isChart = (name?: string) => {\n return (\n name &&\n (name.endsWith('Chart') ||\n name === 'MBTIRadar' ||\n name === 'MBTITypeGrid' ||\n name === 'KeyMetric')\n );\n};\n","import React from 'react';\nimport clsx from 'clsx';\nimport { getConfidenceThreshold } from '../../report-engine';\nimport './ConfidenceIndicator.css';\n\nexport interface ConfidenceIndicatorProps {\n /** Confidence score (0-100) */\n score: number;\n /** Show the numeric score */\n showScore?: boolean;\n /** Show the label */\n showLabel?: boolean;\n /** Size variant */\n size?: 'sm' | 'md' | 'lg';\n /** Additional CSS class */\n className?: string;\n}\n\n/**\n * ConfidenceIndicator displays the reliability score of an insight or data point.\n * This is a guardrail component that helps users understand data quality.\n */\nexport const ConfidenceIndicator = React.forwardRef<HTMLDivElement, ConfidenceIndicatorProps>(\n ({ score, showScore = true, showLabel = true, size = 'md', className, ...props }, ref) => {\n const threshold = getConfidenceThreshold(score);\n const normalizedScore = Math.max(0, Math.min(100, score));\n\n return (\n <div\n ref={ref}\n className={clsx(\n 'ds-confidence-indicator',\n `ds-confidence-indicator--${size}`,\n `ds-confidence-indicator--${threshold.level}`,\n className,\n )}\n role=\"meter\"\n aria-valuenow={normalizedScore}\n aria-valuemin={0}\n aria-valuemax={100}\n aria-label={`Confidence: ${threshold.label} (${normalizedScore}%)`}\n {...props}\n >\n <div className=\"ds-confidence-indicator__bar\">\n <div className=\"ds-confidence-indicator__fill\" style={{ width: `${normalizedScore}%` }} />\n </div>\n <div className=\"ds-confidence-indicator__text\">\n {showScore && <span className=\"ds-confidence-indicator__score\">{normalizedScore}%</span>}\n {showLabel && <span className=\"ds-confidence-indicator__label\">{threshold.label}</span>}\n </div>\n </div>\n );\n },\n);\n\nConfidenceIndicator.displayName = 'ConfidenceIndicator';\n","import React from 'react';\nimport clsx from 'clsx';\nimport './DataCoverageBadge.css';\n\nexport interface DataCoverageBadgeProps {\n /** Current sample size */\n sampleSize: number;\n /** Required minimum sample size */\n requiredSize?: number;\n /** Date range covered */\n dateRange?: {\n start: string;\n end: string;\n };\n /** Show date range */\n showDateRange?: boolean;\n /** Size variant */\n size?: 'sm' | 'md';\n /** Additional CSS class */\n className?: string;\n}\n\n/**\n * DataCoverageBadge shows the sample size and optionally the date range of the data.\n * This is a guardrail component that helps users understand data coverage.\n */\nexport const DataCoverageBadge = React.forwardRef<HTMLDivElement, DataCoverageBadgeProps>(\n (\n { sampleSize, requiredSize, dateRange, showDateRange = true, size = 'md', className, ...props },\n ref,\n ) => {\n const isSufficient = !requiredSize || sampleSize >= requiredSize;\n const percentage = requiredSize\n ? Math.min(100, Math.round((sampleSize / requiredSize) * 100))\n : 100;\n\n return (\n <div\n ref={ref}\n className={clsx(\n 'ds-data-coverage-badge',\n `ds-data-coverage-badge--${size}`,\n !isSufficient && 'ds-data-coverage-badge--insufficient',\n className,\n )}\n {...props}\n >\n <div className=\"ds-data-coverage-badge__sample\">\n <span className=\"ds-data-coverage-badge__icon\">\n <svg\n width=\"14\"\n height=\"14\"\n viewBox=\"0 0 24 24\"\n fill=\"none\"\n stroke=\"currentColor\"\n strokeWidth=\"2\"\n >\n <path d=\"M17 21v-2a4 4 0 0 0-4-4H5a4 4 0 0 0-4 4v2\" />\n <circle cx=\"9\" cy=\"7\" r=\"4\" />\n <path d=\"M23 21v-2a4 4 0 0 0-3-3.87\" />\n <path d=\"M16 3.13a4 4 0 0 1 0 7.75\" />\n </svg>\n </span>\n <span className=\"ds-data-coverage-badge__count\">n={sampleSize.toLocaleString()}</span>\n {requiredSize && (\n <span className=\"ds-data-coverage-badge__required\">\n ({percentage}% of min {requiredSize})\n </span>\n )}\n </div>\n\n {showDateRange && dateRange && (\n <div className=\"ds-data-coverage-badge__date\">\n <span className=\"ds-data-coverage-badge__icon\">\n <svg\n width=\"14\"\n height=\"14\"\n viewBox=\"0 0 24 24\"\n fill=\"none\"\n stroke=\"currentColor\"\n strokeWidth=\"2\"\n >\n <rect x=\"3\" y=\"4\" width=\"18\" height=\"18\" rx=\"2\" ry=\"2\" />\n <line x1=\"16\" y1=\"2\" x2=\"16\" y2=\"6\" />\n <line x1=\"8\" y1=\"2\" x2=\"8\" y2=\"6\" />\n <line x1=\"3\" y1=\"10\" x2=\"21\" y2=\"10\" />\n </svg>\n </span>\n <span>\n {dateRange.start} – {dateRange.end}\n </span>\n </div>\n )}\n </div>\n );\n },\n);\n\nDataCoverageBadge.displayName = 'DataCoverageBadge';\n","import React from 'react';\nimport clsx from 'clsx';\nimport type { AnalyticalState, DataStateDetails } from '../../report-engine';\nimport { ANALYTICAL_STATE_CONFIG } from '../../report-engine';\nimport './DataStateIndicator.css';\n\nexport interface DataStateIndicatorProps {\n /** Current analytical state */\n state: AnalyticalState;\n /** Additional details about the state */\n details?: DataStateDetails;\n /** Show full details or compact */\n variant?: 'compact' | 'full';\n /** Action button handler */\n onAction?: () => void;\n /** Additional CSS class */\n className?: string;\n}\n\n/**\n * DataStateIndicator shows the analytical state of data (valid, insufficient, etc.).\n * This is a guardrail component that prevents misleading outputs.\n */\nexport const DataStateIndicator = React.forwardRef<HTMLDivElement, DataStateIndicatorProps>(\n ({ state, details, variant = 'compact', onAction, className, ...props }, ref) => {\n const config = ANALYTICAL_STATE_CONFIG[state];\n\n // Don't render if valid and compact\n if (state === 'VALID' && variant === 'compact') {\n return null;\n }\n\n const renderIcon = () => {\n switch (config.icon) {\n case 'check':\n return (\n <svg\n width=\"16\"\n height=\"16\"\n viewBox=\"0 0 24 24\"\n fill=\"none\"\n stroke=\"currentColor\"\n strokeWidth=\"2\"\n >\n <polyline points=\"20,6 9,17 4,12\" />\n </svg>\n );\n case 'alert-triangle':\n return (\n <svg\n width=\"16\"\n height=\"16\"\n viewBox=\"0 0 24 24\"\n fill=\"none\"\n stroke=\"currentColor\"\n strokeWidth=\"2\"\n >\n <path d=\"M10.29 3.86L1.82 18a2 2 0 0 0 1.71 3h16.94a2 2 0 0 0 1.71-3L13.71 3.86a2 2 0 0 0-3.42 0z\" />\n <line x1=\"12\" y1=\"9\" x2=\"12\" y2=\"13\" />\n <line x1=\"12\" y1=\"17\" x2=\"12.01\" y2=\"17\" />\n </svg>\n );\n case 'alert-circle':\n return (\n <svg\n width=\"16\"\n height=\"16\"\n viewBox=\"0 0 24 24\"\n fill=\"none\"\n stroke=\"currentColor\"\n strokeWidth=\"2\"\n >\n <circle cx=\"12\" cy=\"12\" r=\"10\" />\n <line x1=\"12\" y1=\"8\" x2=\"12\" y2=\"12\" />\n <line x1=\"12\" y1=\"16\" x2=\"12.01\" y2=\"16\" />\n </svg>\n );\n case 'x-circle':\n return (\n <svg\n width=\"16\"\n height=\"16\"\n viewBox=\"0 0 24 24\"\n fill=\"none\"\n stroke=\"currentColor\"\n strokeWidth=\"2\"\n >\n <circle cx=\"12\" cy=\"12\" r=\"10\" />\n <line x1=\"15\" y1=\"9\" x2=\"9\" y2=\"15\" />\n <line x1=\"9\" y1=\"9\" x2=\"15\" y2=\"15\" />\n </svg>\n );\n case 'cpu':\n return (\n <svg\n width=\"16\"\n height=\"16\"\n viewBox=\"0 0 24 24\"\n fill=\"none\"\n stroke=\"currentColor\"\n strokeWidth=\"2\"\n >\n <rect x=\"4\" y=\"4\" width=\"16\" height=\"16\" rx=\"2\" ry=\"2\" />\n <rect x=\"9\" y=\"9\" width=\"6\" height=\"6\" />\n <line x1=\"9\" y1=\"1\" x2=\"9\" y2=\"4\" />\n <line x1=\"15\" y1=\"1\" x2=\"15\" y2=\"4\" />\n <line x1=\"9\" y1=\"20\" x2=\"9\" y2=\"23\" />\n <line x1=\"15\" y1=\"20\" x2=\"15\" y2=\"23\" />\n <line x1=\"20\" y1=\"9\" x2=\"23\" y2=\"9\" />\n <line x1=\"20\" y1=\"14\" x2=\"23\" y2=\"14\" />\n <line x1=\"1\" y1=\"9\" x2=\"4\" y2=\"9\" />\n <line x1=\"1\" y1=\"14\" x2=\"4\" y2=\"14\" />\n </svg>\n );\n default:\n return (\n <svg\n width=\"16\"\n height=\"16\"\n viewBox=\"0 0 24 24\"\n fill=\"none\"\n stroke=\"currentColor\"\n strokeWidth=\"2\"\n >\n <circle cx=\"12\" cy=\"12\" r=\"10\" />\n <line x1=\"12\" y1=\"16\" x2=\"12\" y2=\"12\" />\n <line x1=\"12\" y1=\"8\" x2=\"12.01\" y2=\"8\" />\n </svg>\n );\n }\n };\n\n return (\n <div\n ref={ref}\n className={clsx(\n 'ds-data-state-indicator',\n `ds-data-state-indicator--${variant}`,\n `ds-data-state-indicator--${state.toLowerCase().replace(/_/g, '-')}`,\n className,\n )}\n role=\"status\"\n aria-live=\"polite\"\n {...props}\n >\n <span className=\"ds-data-state-indicator__icon\" style={{ color: config.color }}>\n {renderIcon()}\n </span>\n\n <div className=\"ds-data-state-indicator__content\">\n <span className=\"ds-data-state-indicator__label\">{config.label}</span>\n\n {variant === 'full' && details && (\n <span className=\"ds-data-state-indicator__details\">\n {details.message}\n {details.sampleSize !== undefined && details.requiredSize !== undefined && (\n <>\n {' '}\n (n={details.sampleSize}/{details.requiredSize})\n </>\n )}\n {details.confidence !== undefined && <> • {details.confidence}% confidence</>}\n {details.missingFields && details.missingFields.length > 0 && (\n <> • Missing: {details.missingFields.join(', ')}</>\n )}\n </span>\n )}\n </div>\n\n {config.action && onAction && (\n <button type=\"button\" className=\"ds-data-state-indicator__action\" onClick={onAction}>\n {config.action}\n </button>\n )}\n </div>\n );\n },\n);\n\nDataStateIndicator.displayName = 'DataStateIndicator';\n"]}
@@ -0,0 +1,264 @@
1
+ import { classifyUrl, requiresUtm } from './chunk-KJ2OXQF4.js';
2
+
3
+ // src/web/utm/constants.ts
4
+ var UTM_SOURCES = [
5
+ "linkedin",
6
+ "whatsapp",
7
+ "intercom",
8
+ "email",
9
+ "website",
10
+ "app",
11
+ "chromeStore",
12
+ "stripe",
13
+ "direct"
14
+ ];
15
+ var UTM_MEDIUMS_MESSAGING = [
16
+ "dm",
17
+ "group",
18
+ "email",
19
+ "inAppChat",
20
+ "organicPost",
21
+ "paidAd",
22
+ "qrCode",
23
+ "redirect",
24
+ "storeListing"
25
+ ];
26
+ var UTM_MEDIUMS_APP = [
27
+ "appHome",
28
+ "appDashboard",
29
+ "appInbox",
30
+ "appContacts",
31
+ "appCampaigns",
32
+ "appSettings",
33
+ "appBilling",
34
+ "appCheckout",
35
+ "appOnboarding",
36
+ "appSupport",
37
+ "appCalendar"
38
+ ];
39
+ var UTM_MEDIUMS_WEB = [
40
+ "webHome",
41
+ "webDemo",
42
+ "webPricing",
43
+ "webFeatures",
44
+ "webUseCase",
45
+ "webSolution",
46
+ "webIndustry",
47
+ "webIntegrations",
48
+ "webBlog",
49
+ "webBlogPost",
50
+ "webLanding",
51
+ "webComparison",
52
+ "webCaseStudy",
53
+ "webAbout",
54
+ "webContact",
55
+ "webCareers",
56
+ "webLegal",
57
+ "webDocs",
58
+ "webAffiliate"
59
+ ];
60
+ var UTM_MEDIUMS_ALL = [
61
+ ...UTM_MEDIUMS_MESSAGING,
62
+ ...UTM_MEDIUMS_APP,
63
+ ...UTM_MEDIUMS_WEB
64
+ ];
65
+ var UTM_CAMPAIGNS = [
66
+ "discoveryCall",
67
+ "demo",
68
+ "trial",
69
+ "onboarding",
70
+ "upgrade",
71
+ "renewal",
72
+ "retention",
73
+ "accountSupport",
74
+ "supportCall",
75
+ "partnerCall",
76
+ "hiring",
77
+ "interviewCall"
78
+ ];
79
+ var UTM_TERMS = [
80
+ "julienGadea",
81
+ "bramSmith",
82
+ "florentDupont",
83
+ "sawLin",
84
+ "evaSupport",
85
+ "team",
86
+ "auto"
87
+ ];
88
+ var UTM_CONTENTS = [
89
+ "ctaPrimary",
90
+ "ctaSecondary",
91
+ "ctaHeader",
92
+ "ctaFooter",
93
+ "ctaInline",
94
+ "buttonPrimary",
95
+ "buttonSecondary",
96
+ "banner",
97
+ "popup",
98
+ "variantA",
99
+ "variantB",
100
+ "variantC"
101
+ ];
102
+ var UTM_SOURCES_REQUIRING_SELLER = [
103
+ "linkedin",
104
+ "whatsapp",
105
+ "intercom",
106
+ "email"
107
+ ];
108
+
109
+ // src/web/utm/validators.ts
110
+ function validateUtmField(field, value) {
111
+ switch (field) {
112
+ case "source":
113
+ return UTM_SOURCES.includes(value);
114
+ case "medium":
115
+ return UTM_MEDIUMS_ALL.includes(value);
116
+ case "campaign":
117
+ return UTM_CAMPAIGNS.includes(value);
118
+ case "term":
119
+ return UTM_TERMS.includes(value);
120
+ case "content":
121
+ return UTM_CONTENTS.includes(value);
122
+ default:
123
+ return false;
124
+ }
125
+ }
126
+ function isValidUtmParams(params) {
127
+ if (!params.source || !params.medium || !params.campaign) {
128
+ return false;
129
+ }
130
+ if (!validateUtmField("source", params.source)) return false;
131
+ if (!validateUtmField("medium", params.medium)) return false;
132
+ if (!validateUtmField("campaign", params.campaign)) return false;
133
+ if (params.term !== void 0 && !validateUtmField("term", params.term)) return false;
134
+ if (params.content !== void 0 && !validateUtmField("content", params.content)) return false;
135
+ return true;
136
+ }
137
+ function validateCompliance(url, params) {
138
+ const errors = [];
139
+ const classification = classifyUrl(url);
140
+ const needsUtm = requiresUtm(url);
141
+ if (!needsUtm) {
142
+ return {
143
+ status: "compliant",
144
+ url,
145
+ params: params ?? null,
146
+ errors: []
147
+ };
148
+ }
149
+ if (!params) {
150
+ return {
151
+ status: "blocked",
152
+ url,
153
+ params: null,
154
+ errors: [`URL classified as '${classification}' requires UTM parameters but none provided`]
155
+ };
156
+ }
157
+ if (!params.source) errors.push("Missing required field: utm_source");
158
+ if (!params.medium) errors.push("Missing required field: utm_medium");
159
+ if (!params.campaign) errors.push("Missing required field: utm_campaign");
160
+ if (params.source && !validateUtmField("source", params.source)) {
161
+ errors.push(`Invalid utm_source: '${params.source}'. Must be one of: ${UTM_SOURCES.join(", ")}`);
162
+ }
163
+ if (params.medium && !validateUtmField("medium", params.medium)) {
164
+ errors.push(`Invalid utm_medium: '${params.medium}'. Must be one of governed enum values`);
165
+ }
166
+ if (params.campaign && !validateUtmField("campaign", params.campaign)) {
167
+ errors.push(`Invalid utm_campaign: '${params.campaign}'. Must be one of: ${UTM_CAMPAIGNS.join(", ")}`);
168
+ }
169
+ if (params.term !== void 0 && params.term && !validateUtmField("term", params.term)) {
170
+ errors.push(`Invalid utm_term: '${params.term}'. Must be one of: ${UTM_TERMS.join(", ")}`);
171
+ }
172
+ if (params.content !== void 0 && params.content && !validateUtmField("content", params.content)) {
173
+ errors.push(`Invalid utm_content: '${params.content}'. Must be one of: ${UTM_CONTENTS.join(", ")}`);
174
+ }
175
+ const allValues = [];
176
+ if (params.source) allValues.push(params.source);
177
+ if (params.medium) allValues.push(params.medium);
178
+ if (params.campaign) allValues.push(params.campaign);
179
+ if (params.term) allValues.push(params.term);
180
+ if (params.content) allValues.push(params.content);
181
+ for (const value of allValues) {
182
+ if (/\s/.test(value)) errors.push(`Value '${value}' contains spaces`);
183
+ if (/_/.test(value)) errors.push(`Value '${value}' contains underscores`);
184
+ if (/-/.test(value)) errors.push(`Value '${value}' contains hyphens`);
185
+ if (/[^\x20-\x7E]/.test(value)) errors.push(`Value '${value}' contains non-ASCII characters`);
186
+ if (value === value.toUpperCase() && value.length > 1) errors.push(`Value '${value}' is ALL CAPS`);
187
+ }
188
+ const status = errors.length === 0 ? "compliant" : "blocked";
189
+ return { status, url, params, errors };
190
+ }
191
+ function classifyAndEnforce(url, params) {
192
+ return validateCompliance(url, params);
193
+ }
194
+ function buildBlockedError(reason, url) {
195
+ return {
196
+ status: "UTM_COMPLIANCE_BLOCKED",
197
+ reason,
198
+ requiredFix: reason,
199
+ correctedExample: url ?? null
200
+ };
201
+ }
202
+
203
+ // src/web/utm/attribution.ts
204
+ function parseUtmParams(search) {
205
+ if (!search) return {};
206
+ const params = new URLSearchParams(search);
207
+ const result = {};
208
+ const source = params.get("utm_source");
209
+ if (source && UTM_SOURCES.includes(source)) {
210
+ result.source = source;
211
+ }
212
+ const medium = params.get("utm_medium");
213
+ if (medium) {
214
+ result.medium = medium;
215
+ }
216
+ const campaign = params.get("utm_campaign");
217
+ if (campaign && UTM_CAMPAIGNS.includes(campaign)) {
218
+ result.campaign = campaign;
219
+ }
220
+ const term = params.get("utm_term");
221
+ if (term && UTM_TERMS.includes(term)) {
222
+ result.term = term;
223
+ }
224
+ const content = params.get("utm_content");
225
+ if (content && UTM_CONTENTS.includes(content)) {
226
+ result.content = content;
227
+ }
228
+ return result;
229
+ }
230
+ function toFirstTouchAttribution(params) {
231
+ return {
232
+ firstTouchSource: params.source,
233
+ firstTouchMedium: params.medium,
234
+ firstTouchCampaign: params.campaign,
235
+ firstTouchSeller: params.term ?? null
236
+ };
237
+ }
238
+ function requiresSellerAttribution(source) {
239
+ return UTM_SOURCES_REQUIRING_SELLER.includes(source);
240
+ }
241
+
242
+ // src/web/utm/audit.ts
243
+ function createAuditEntry(url, params, generatorContext, confidence) {
244
+ const utmParts = [];
245
+ if (params) {
246
+ if (params.source) utmParts.push(`utm_source=${params.source}`);
247
+ if (params.medium) utmParts.push(`utm_medium=${params.medium}`);
248
+ if (params.campaign) utmParts.push(`utm_campaign=${params.campaign}`);
249
+ if (params.term) utmParts.push(`utm_term=${params.term}`);
250
+ if (params.content) utmParts.push(`utm_content=${params.content}`);
251
+ }
252
+ return {
253
+ url,
254
+ utmString: utmParts.join("&"),
255
+ generatorContext,
256
+ sellerAttribution: params?.term ?? null,
257
+ timestamp: (/* @__PURE__ */ new Date()).toISOString(),
258
+ confidence
259
+ };
260
+ }
261
+
262
+ export { UTM_CAMPAIGNS, UTM_CONTENTS, UTM_MEDIUMS_ALL, UTM_MEDIUMS_APP, UTM_MEDIUMS_MESSAGING, UTM_MEDIUMS_WEB, UTM_SOURCES, UTM_SOURCES_REQUIRING_SELLER, UTM_TERMS, buildBlockedError, classifyAndEnforce, createAuditEntry, isValidUtmParams, parseUtmParams, requiresSellerAttribution, toFirstTouchAttribution, validateCompliance, validateUtmField };
263
+ //# sourceMappingURL=out.js.map
264
+ //# sourceMappingURL=chunk-BILT5KD3.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../src/web/utm/constants.ts","../src/web/utm/validators.ts","../src/web/utm/attribution.ts","../src/web/utm/audit.ts"],"names":[],"mappings":"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","sourcesContent":["/* ============================================================================\n UTM Governance Constants — Enum Arrays (Hard Constraints)\n ============================================================================\n These arrays are the canonical enum lists. Values must never be invented\n outside of a governance update.\n ============================================================================ */\n\nimport type {\n UtmSource,\n UtmMediumMessaging,\n UtmMediumAppPage,\n UtmMediumWebPage,\n UtmMedium,\n UtmCampaign,\n UtmTerm,\n UtmContent,\n UtmSourceRequiringSeller,\n} from './types';\n\n// ── Sources ──────────────────────────────────────────────────────────────────\n\nexport const UTM_SOURCES: readonly UtmSource[] = [\n 'linkedin',\n 'whatsapp',\n 'intercom',\n 'email',\n 'website',\n 'app',\n 'chromeStore',\n 'stripe',\n 'direct',\n] as const;\n\n// ── Mediums ──────────────────────────────────────────────────────────────────\n\nexport const UTM_MEDIUMS_MESSAGING: readonly UtmMediumMessaging[] = [\n 'dm',\n 'group',\n 'email',\n 'inAppChat',\n 'organicPost',\n 'paidAd',\n 'qrCode',\n 'redirect',\n 'storeListing',\n] as const;\n\nexport const UTM_MEDIUMS_APP: readonly UtmMediumAppPage[] = [\n 'appHome',\n 'appDashboard',\n 'appInbox',\n 'appContacts',\n 'appCampaigns',\n 'appSettings',\n 'appBilling',\n 'appCheckout',\n 'appOnboarding',\n 'appSupport',\n 'appCalendar',\n] as const;\n\nexport const UTM_MEDIUMS_WEB: readonly UtmMediumWebPage[] = [\n 'webHome',\n 'webDemo',\n 'webPricing',\n 'webFeatures',\n 'webUseCase',\n 'webSolution',\n 'webIndustry',\n 'webIntegrations',\n 'webBlog',\n 'webBlogPost',\n 'webLanding',\n 'webComparison',\n 'webCaseStudy',\n 'webAbout',\n 'webContact',\n 'webCareers',\n 'webLegal',\n 'webDocs',\n 'webAffiliate',\n] as const;\n\nexport const UTM_MEDIUMS_ALL: readonly UtmMedium[] = [\n ...UTM_MEDIUMS_MESSAGING,\n ...UTM_MEDIUMS_APP,\n ...UTM_MEDIUMS_WEB,\n] as const;\n\n// ── Campaigns ────────────────────────────────────────────────────────────────\n\nexport const UTM_CAMPAIGNS: readonly UtmCampaign[] = [\n 'discoveryCall',\n 'demo',\n 'trial',\n 'onboarding',\n 'upgrade',\n 'renewal',\n 'retention',\n 'accountSupport',\n 'supportCall',\n 'partnerCall',\n 'hiring',\n 'interviewCall',\n] as const;\n\n// ── Terms (seller attribution) ───────────────────────────────────────────────\n\nexport const UTM_TERMS: readonly UtmTerm[] = [\n 'julienGadea',\n 'bramSmith',\n 'florentDupont',\n 'sawLin',\n 'evaSupport',\n 'team',\n 'auto',\n] as const;\n\n// ── Contents (CTA variant) ──────────────────────────────────────────────────\n\nexport const UTM_CONTENTS: readonly UtmContent[] = [\n 'ctaPrimary',\n 'ctaSecondary',\n 'ctaHeader',\n 'ctaFooter',\n 'ctaInline',\n 'buttonPrimary',\n 'buttonSecondary',\n 'banner',\n 'popup',\n 'variantA',\n 'variantB',\n 'variantC',\n] as const;\n\n// ── Sources requiring seller attribution ────────────────────────────────────\n\nexport const UTM_SOURCES_REQUIRING_SELLER: readonly UtmSourceRequiringSeller[] = [\n 'linkedin',\n 'whatsapp',\n 'intercom',\n 'email',\n] as const;\n","/* ============================================================================\n UTM Validators — Hard Governance Enforcement\n ============================================================================\n All validation uses enum-only matching. No creative fixes, no fuzzy matching.\n If a value is not in the governed enum, it is invalid.\n ============================================================================ */\n\nimport type {\n UtmParams,\n UtmComplianceResult,\n UtmComplianceStatus,\n UtmBlockedError,\n} from './types';\n\nimport {\n UTM_SOURCES,\n UTM_MEDIUMS_ALL,\n UTM_CAMPAIGNS,\n UTM_TERMS,\n UTM_CONTENTS,\n} from './constants';\n\nimport { classifyUrl, requiresUtm } from './classifiers';\n\n// ── Field-level validation ───────────────────────────────────────────────────\n\n/**\n * Validate a single UTM field name + value pair against the governed enums.\n *\n * @param field - The UTM field name (e.g., 'source', 'medium', 'campaign', 'term', 'content')\n * @param value - The value to validate\n * @returns true if the value is in the corresponding enum\n */\nexport function validateUtmField(field: string, value: string): boolean {\n switch (field) {\n case 'source':\n return (UTM_SOURCES as readonly string[]).includes(value);\n case 'medium':\n return (UTM_MEDIUMS_ALL as readonly string[]).includes(value);\n case 'campaign':\n return (UTM_CAMPAIGNS as readonly string[]).includes(value);\n case 'term':\n return (UTM_TERMS as readonly string[]).includes(value);\n case 'content':\n return (UTM_CONTENTS as readonly string[]).includes(value);\n default:\n return false;\n }\n}\n\n// ── Param-level validation ───────────────────────────────────────────────────\n\n/**\n * Validate a complete or partial UtmParams object.\n *\n * Returns true only if:\n * - source, medium, and campaign are all present\n * - All present values match their respective enums\n * - Optional fields (term, content) match their enums if provided\n */\nexport function isValidUtmParams(params: Partial<UtmParams>): boolean {\n // Required fields must exist and be non-empty\n if (!params.source || !params.medium || !params.campaign) {\n return false;\n }\n\n // Validate required fields\n if (!validateUtmField('source', params.source)) return false;\n if (!validateUtmField('medium', params.medium)) return false;\n if (!validateUtmField('campaign', params.campaign)) return false;\n\n // Validate optional fields if present\n if (params.term !== undefined && !validateUtmField('term', params.term)) return false;\n if (params.content !== undefined && !validateUtmField('content', params.content)) return false;\n\n return true;\n}\n\n// ── Compliance validation ────────────────────────────────────────────────────\n\n/**\n * Full compliance validation for a URL + UTM params combination.\n *\n * Checks:\n * 1. URL classification (does it require UTMs?)\n * 2. All required params present\n * 3. All values match governed enums\n * 4. No illegal characters (spaces, underscores, hyphens, emojis, ALL CAPS)\n * 5. camelCase format enforcement\n */\nexport function validateCompliance(\n url: string,\n params?: Partial<UtmParams> | null,\n): UtmComplianceResult {\n const errors: string[] = [];\n const classification = classifyUrl(url);\n const needsUtm = requiresUtm(url);\n\n // If URL doesn't require UTMs, it's automatically compliant\n if (!needsUtm) {\n return {\n status: 'compliant',\n url,\n params: params ?? null,\n errors: [],\n };\n }\n\n // UTMs are required but not provided\n if (!params) {\n return {\n status: 'blocked',\n url,\n params: null,\n errors: [`URL classified as '${classification}' requires UTM parameters but none provided`],\n };\n }\n\n // Check required fields\n if (!params.source) errors.push('Missing required field: utm_source');\n if (!params.medium) errors.push('Missing required field: utm_medium');\n if (!params.campaign) errors.push('Missing required field: utm_campaign');\n\n // Validate values against enums\n if (params.source && !validateUtmField('source', params.source)) {\n errors.push(`Invalid utm_source: '${params.source}'. Must be one of: ${UTM_SOURCES.join(', ')}`);\n }\n if (params.medium && !validateUtmField('medium', params.medium)) {\n errors.push(`Invalid utm_medium: '${params.medium}'. Must be one of governed enum values`);\n }\n if (params.campaign && !validateUtmField('campaign', params.campaign)) {\n errors.push(`Invalid utm_campaign: '${params.campaign}'. Must be one of: ${UTM_CAMPAIGNS.join(', ')}`);\n }\n if (params.term !== undefined && params.term && !validateUtmField('term', params.term)) {\n errors.push(`Invalid utm_term: '${params.term}'. Must be one of: ${UTM_TERMS.join(', ')}`);\n }\n if (params.content !== undefined && params.content && !validateUtmField('content', params.content)) {\n errors.push(`Invalid utm_content: '${params.content}'. Must be one of: ${UTM_CONTENTS.join(', ')}`);\n }\n\n // Validate value format (camelCase, no illegal chars)\n const allValues: string[] = [];\n if (params.source) allValues.push(params.source);\n if (params.medium) allValues.push(params.medium);\n if (params.campaign) allValues.push(params.campaign);\n if (params.term) allValues.push(params.term);\n if (params.content) allValues.push(params.content);\n\n for (const value of allValues) {\n if (/\\s/.test(value)) errors.push(`Value '${value}' contains spaces`);\n if (/_/.test(value)) errors.push(`Value '${value}' contains underscores`);\n if (/-/.test(value)) errors.push(`Value '${value}' contains hyphens`);\n if (/[^\\x20-\\x7E]/.test(value)) errors.push(`Value '${value}' contains non-ASCII characters`);\n if (value === value.toUpperCase() && value.length > 1) errors.push(`Value '${value}' is ALL CAPS`);\n }\n\n const status: UtmComplianceStatus = errors.length === 0 ? 'compliant' : 'blocked';\n\n return { status, url, params, errors };\n}\n\n/**\n * Classify a URL and enforce UTM compliance in a single call.\n */\nexport function classifyAndEnforce(\n url: string,\n params?: Partial<UtmParams>,\n): UtmComplianceResult {\n return validateCompliance(url, params);\n}\n\n// ── Error builder ────────────────────────────────────────────────────────────\n\n/**\n * Build a structured blocked-error response.\n */\nexport function buildBlockedError(\n reason: string,\n url?: string,\n): UtmBlockedError {\n return {\n status: 'UTM_COMPLIANCE_BLOCKED',\n reason,\n requiredFix: reason,\n correctedExample: url ?? null,\n };\n}\n","/* ============================================================================\n First-Touch Attribution & Seller Logic\n ============================================================================\n Handles UTM parsing from URLs and first-touch CRM attribution mapping.\n Once first-touch fields are set, they must NEVER be overwritten.\n ============================================================================ */\n\nimport type {\n UtmParams,\n UtmSource,\n UtmTerm,\n FirstTouchAttribution,\n} from './types';\n\nimport {\n UTM_SOURCES,\n UTM_CAMPAIGNS,\n UTM_TERMS,\n UTM_CONTENTS,\n UTM_SOURCES_REQUIRING_SELLER,\n} from './constants';\n\n// ── Parse UTM params from URL search string ──────────────────────────────────\n\n/**\n * Parse UTM parameters from a URL search string.\n *\n * Validates each field against governed enums:\n * - source: must be in UTM_SOURCES (invalid → excluded)\n * - medium: accepted as-is (no enum validation — medium is loosely parsed)\n * - campaign: must be in UTM_CAMPAIGNS (invalid → excluded)\n * - term: must be in UTM_TERMS (invalid → excluded)\n * - content: must be in UTM_CONTENTS (invalid → excluded)\n *\n * @param search - URL search string, with or without leading '?'\n * @returns Partial<UtmParams> with only valid fields included\n */\nexport function parseUtmParams(search: string): Partial<UtmParams> {\n if (!search) return {};\n\n const params = new URLSearchParams(search);\n const result: Partial<UtmParams> = {};\n\n // Source — validate against enum\n const source = params.get('utm_source');\n if (source && (UTM_SOURCES as readonly string[]).includes(source)) {\n result.source = source as UtmSource;\n }\n\n // Medium — loosely parsed (no enum validation per test spec line 674-678)\n const medium = params.get('utm_medium');\n if (medium) {\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n result.medium = medium as any;\n }\n\n // Campaign — validate against enum\n const campaign = params.get('utm_campaign');\n if (campaign && (UTM_CAMPAIGNS as readonly string[]).includes(campaign)) {\n result.campaign = campaign as UtmParams['campaign'];\n }\n\n // Term — validate against enum\n const term = params.get('utm_term');\n if (term && (UTM_TERMS as readonly string[]).includes(term)) {\n result.term = term as UtmTerm;\n }\n\n // Content — validate against enum\n const content = params.get('utm_content');\n if (content && (UTM_CONTENTS as readonly string[]).includes(content)) {\n result.content = content as UtmParams['content'];\n }\n\n return result;\n}\n\n// ── First-touch attribution mapping ──────────────────────────────────────────\n\n/**\n * Map UTM params to CRM first-touch attribution fields.\n *\n * On first lead capture:\n * - utm_source → firstTouchSource\n * - utm_medium → firstTouchMedium\n * - utm_campaign → firstTouchCampaign\n * - utm_term → firstTouchSeller (null if absent)\n *\n * utm_content is intentionally excluded from attribution.\n * These fields must be persisted permanently and never overwritten.\n */\nexport function toFirstTouchAttribution(params: UtmParams): FirstTouchAttribution {\n return {\n firstTouchSource: params.source,\n firstTouchMedium: params.medium,\n firstTouchCampaign: params.campaign,\n firstTouchSeller: params.term ?? null,\n };\n}\n\n// ── Seller attribution check ─────────────────────────────────────────────────\n\n/**\n * Check if a source requires seller attribution (utm_term).\n *\n * Sources requiring seller: linkedin, whatsapp, intercom, email\n * These are manual outbound channels where individual seller tracking matters.\n */\nexport function requiresSellerAttribution(source: UtmSource): boolean {\n return (UTM_SOURCES_REQUIRING_SELLER as readonly string[]).includes(source);\n}\n","/* ============================================================================\n UTM Audit & Observability\n ============================================================================\n Creates structured audit entries for UTM compliance telemetry.\n ============================================================================ */\n\nimport type { UtmParams, UtmConfidence, UtmAuditEntry } from './types';\n\n/**\n * Create a structured audit log entry for a generated/validated link.\n *\n * Every time a link is generated, an audit entry should be logged with:\n * - URL and full UTM string\n * - Generator context (component, agent, or tool that created the link)\n * - Seller attribution (from utm_term)\n * - Timestamp\n * - Confidence flag (valid / corrected / blocked)\n */\nexport function createAuditEntry(\n url: string,\n params: Partial<UtmParams> | null,\n generatorContext: string,\n confidence: UtmConfidence,\n): UtmAuditEntry {\n const utmParts: string[] = [];\n\n if (params) {\n if (params.source) utmParts.push(`utm_source=${params.source}`);\n if (params.medium) utmParts.push(`utm_medium=${params.medium}`);\n if (params.campaign) utmParts.push(`utm_campaign=${params.campaign}`);\n if (params.term) utmParts.push(`utm_term=${params.term}`);\n if (params.content) utmParts.push(`utm_content=${params.content}`);\n }\n\n return {\n url,\n utmString: utmParts.join('&'),\n generatorContext,\n sellerAttribution: params?.term ?? null,\n timestamp: new Date().toISOString(),\n confidence,\n };\n}\n"]}
@@ -0,0 +1,24 @@
1
+ import React from 'react';
2
+ import { Tooltip as Tooltip$1 } from '@base-ui/react/tooltip';
3
+ import clsx from 'clsx';
4
+ import { jsx, jsxs } from 'react/jsx-runtime';
5
+
6
+ // src/components/Tooltip/Tooltip.tsx
7
+ var TooltipProvider = Tooltip$1.Provider;
8
+ var TooltipRoot = Tooltip$1.Root;
9
+ var TooltipTrigger = Tooltip$1.Trigger;
10
+ var TooltipContent = React.forwardRef(({ className, children, ...props }, ref) => /* @__PURE__ */ jsx(Tooltip$1.Portal, { children: /* @__PURE__ */ jsx(Tooltip$1.Positioner, { sideOffset: 5, children: /* @__PURE__ */ jsxs(Tooltip$1.Popup, { ref, className: clsx("ds-tooltip__content", className), ...props, children: [
11
+ children,
12
+ /* @__PURE__ */ jsx(Tooltip$1.Arrow, { className: "ds-tooltip__arrow" })
13
+ ] }) }) }));
14
+ TooltipContent.displayName = "TooltipContent";
15
+ var Tooltip = ({ content, children, delay = 200 }) => {
16
+ return /* @__PURE__ */ jsx(Tooltip$1.Provider, { delay, children: /* @__PURE__ */ jsxs(Tooltip$1.Root, { children: [
17
+ /* @__PURE__ */ jsx(Tooltip$1.Trigger, { render: children }),
18
+ /* @__PURE__ */ jsx(TooltipContent, { children: content })
19
+ ] }) });
20
+ };
21
+
22
+ export { Tooltip, TooltipContent, TooltipProvider, TooltipRoot, TooltipTrigger };
23
+ //# sourceMappingURL=out.js.map
24
+ //# sourceMappingURL=chunk-C2BCDNAV.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../src/components/Tooltip/Tooltip.tsx"],"names":[],"mappings":";AAAA,OAAO,WAAW;AAClB,SAAS,WAAW,mBAAmB;AACvC,OAAO,UAAU;AAaX,SAEE,KAFF;AAVC,IAAM,kBAAkB,YAAY;AACpC,IAAM,cAAc,YAAY;AAChC,IAAM,iBAAiB,YAAY;AAEnC,IAAM,iBAAiB,MAAM,WAGlC,CAAC,EAAE,WAAW,UAAU,GAAG,MAAM,GAAG,QACpC,oBAAC,YAAY,QAAZ,EACC,8BAAC,YAAY,YAAZ,EAAuB,YAAY,GAClC,+BAAC,YAAY,OAAZ,EAAkB,KAAU,WAAW,KAAK,uBAAuB,SAAS,GAAI,GAAG,OACjF;AAAA;AAAA,EACD,oBAAC,YAAY,OAAZ,EAAkB,WAAU,qBAAoB;AAAA,GACnD,GACF,GACF,CACD;AACD,eAAe,cAAc;AAQtB,IAAM,UAAU,CAAC,EAAE,SAAS,UAAU,QAAQ,IAAI,MAAoB;AAC3E,SACE,oBAAC,YAAY,UAAZ,EAAqB,OACpB,+BAAC,YAAY,MAAZ,EACC;AAAA,wBAAC,YAAY,SAAZ,EAAoB,QAAQ,UAAgC;AAAA,IAC7D,oBAAC,kBAAgB,mBAAQ;AAAA,KAC3B,GACF;AAEJ","sourcesContent":["import React from 'react';\nimport { Tooltip as BaseTooltip } from '@base-ui/react/tooltip';\nimport clsx from 'clsx';\nimport './Tooltip.css';\n\nexport const TooltipProvider = BaseTooltip.Provider;\nexport const TooltipRoot = BaseTooltip.Root;\nexport const TooltipTrigger = BaseTooltip.Trigger;\n\nexport const TooltipContent = React.forwardRef<\n HTMLDivElement,\n React.HTMLAttributes<HTMLDivElement>\n>(({ className, children, ...props }, ref) => (\n <BaseTooltip.Portal>\n <BaseTooltip.Positioner sideOffset={5}>\n <BaseTooltip.Popup ref={ref} className={clsx('ds-tooltip__content', className)} {...props}>\n {children}\n <BaseTooltip.Arrow className=\"ds-tooltip__arrow\" />\n </BaseTooltip.Popup>\n </BaseTooltip.Positioner>\n </BaseTooltip.Portal>\n));\nTooltipContent.displayName = 'TooltipContent';\n\nexport interface TooltipProps {\n content: React.ReactNode;\n children: React.ReactNode;\n delay?: number;\n}\n\nexport const Tooltip = ({ content, children, delay = 200 }: TooltipProps) => {\n return (\n <BaseTooltip.Provider delay={delay}>\n <BaseTooltip.Root>\n <BaseTooltip.Trigger render={children as React.ReactElement} />\n <TooltipContent>{content}</TooltipContent>\n </BaseTooltip.Root>\n </BaseTooltip.Provider>\n );\n};\n"]}