@geenius/feedback 0.1.0 → 0.3.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (104) hide show
  1. package/package.json +16 -3
  2. package/packages/convex/dist/index.d.ts +192 -0
  3. package/packages/convex/dist/index.js +239 -0
  4. package/packages/convex/dist/index.js.map +1 -0
  5. package/packages/react/README.md +1 -1
  6. package/packages/react/dist/index.d.ts +146 -0
  7. package/packages/react/dist/index.js +545 -0
  8. package/packages/react/dist/index.js.map +1 -0
  9. package/packages/react-css/README.md +1 -1
  10. package/packages/react-css/dist/index.css +965 -0
  11. package/packages/react-css/dist/index.css.map +1 -0
  12. package/packages/react-css/dist/index.d.ts +49 -0
  13. package/packages/react-css/dist/index.js +228 -0
  14. package/packages/react-css/dist/index.js.map +1 -0
  15. package/packages/shared/README.md +1 -1
  16. package/packages/shared/dist/index.d.ts +115 -0
  17. package/packages/shared/dist/index.js +112 -0
  18. package/packages/shared/dist/index.js.map +1 -0
  19. package/packages/solidjs/README.md +1 -1
  20. package/packages/solidjs/dist/index.d.ts +128 -0
  21. package/packages/solidjs/dist/index.js +289 -0
  22. package/packages/solidjs/dist/index.js.map +1 -0
  23. package/packages/solidjs-css/README.md +1 -1
  24. package/packages/solidjs-css/dist/index.css +965 -0
  25. package/packages/solidjs-css/dist/index.css.map +1 -0
  26. package/packages/solidjs-css/dist/index.d.ts +2 -0
  27. package/packages/solidjs-css/dist/index.js +29 -0
  28. package/packages/solidjs-css/dist/index.js.map +1 -0
  29. package/.changeset/config.json +0 -11
  30. package/.github/CODEOWNERS +0 -1
  31. package/.github/ISSUE_TEMPLATE/bug_report.md +0 -16
  32. package/.github/ISSUE_TEMPLATE/feature_request.md +0 -11
  33. package/.github/PULL_REQUEST_TEMPLATE.md +0 -10
  34. package/.github/dependabot.yml +0 -11
  35. package/.github/workflows/ci.yml +0 -23
  36. package/.github/workflows/release.yml +0 -29
  37. package/.nvmrc +0 -1
  38. package/.project/ACCOUNT.yaml +0 -4
  39. package/.project/IDEAS.yaml +0 -7
  40. package/.project/PROJECT.yaml +0 -11
  41. package/.project/ROADMAP.yaml +0 -15
  42. package/CODE_OF_CONDUCT.md +0 -16
  43. package/CONTRIBUTING.md +0 -26
  44. package/SECURITY.md +0 -15
  45. package/SUPPORT.md +0 -8
  46. package/packages/convex/package.json +0 -42
  47. package/packages/convex/src/index.ts +0 -3
  48. package/packages/convex/src/mutations.ts +0 -88
  49. package/packages/convex/src/queries.ts +0 -78
  50. package/packages/convex/src/schema.ts +0 -47
  51. package/packages/convex/tsconfig.json +0 -18
  52. package/packages/convex/tsup.config.ts +0 -17
  53. package/packages/react/package.json +0 -49
  54. package/packages/react/src/components/FeedbackCard.tsx +0 -51
  55. package/packages/react/src/components/FeedbackForm.tsx +0 -43
  56. package/packages/react/src/components/FeedbackWidget.tsx +0 -32
  57. package/packages/react/src/components/NPSSurvey.tsx +0 -62
  58. package/packages/react/src/components/index.ts +0 -4
  59. package/packages/react/src/hooks/index.ts +0 -5
  60. package/packages/react/src/hooks/useFeedback.ts +0 -23
  61. package/packages/react/src/hooks/useFeedbackAdmin.ts +0 -24
  62. package/packages/react/src/hooks/useFeedbackForm.ts +0 -35
  63. package/packages/react/src/hooks/useNPS.ts +0 -26
  64. package/packages/react/src/index.tsx +0 -13
  65. package/packages/react/src/pages/FeedbackAdminPage.tsx +0 -71
  66. package/packages/react/src/pages/FeedbackPublicPage.tsx +0 -42
  67. package/packages/react/src/pages/FeedbackWidgetPage.tsx +0 -25
  68. package/packages/react/src/pages/index.ts +0 -3
  69. package/packages/react/tsconfig.json +0 -19
  70. package/packages/react/tsup.config.ts +0 -12
  71. package/packages/react-css/package.json +0 -36
  72. package/packages/react-css/src/components/index.ts +0 -5
  73. package/packages/react-css/src/components/index.tsx +0 -107
  74. package/packages/react-css/src/hooks/index.ts +0 -2
  75. package/packages/react-css/src/index.tsx +0 -5
  76. package/packages/react-css/src/pages/FeedbackAdminPage.tsx +0 -112
  77. package/packages/react-css/src/pages/FeedbackPage.tsx +0 -76
  78. package/packages/react-css/src/styles.css +0 -281
  79. package/packages/react-css/tsconfig.json +0 -19
  80. package/packages/react-css/tsup.config.ts +0 -10
  81. package/packages/shared/package.json +0 -44
  82. package/packages/shared/src/__tests__/feedback.test.ts +0 -72
  83. package/packages/shared/src/config.ts +0 -49
  84. package/packages/shared/src/index.ts +0 -111
  85. package/packages/shared/src/types.ts +0 -59
  86. package/packages/shared/tsconfig.json +0 -18
  87. package/packages/shared/tsup.config.ts +0 -11
  88. package/packages/shared/vitest.config.ts +0 -4
  89. package/packages/solidjs/package.json +0 -45
  90. package/packages/solidjs/src/components.tsx +0 -72
  91. package/packages/solidjs/src/index.tsx +0 -3
  92. package/packages/solidjs/src/primitives.ts +0 -49
  93. package/packages/solidjs/tsconfig.json +0 -20
  94. package/packages/solidjs/tsup.config.ts +0 -12
  95. package/packages/solidjs-css/package.json +0 -32
  96. package/packages/solidjs-css/src/index.tsx +0 -4
  97. package/packages/solidjs-css/src/pages/FeedbackAdminPage.tsx +0 -78
  98. package/packages/solidjs-css/src/pages/FeedbackPage.tsx +0 -65
  99. package/packages/solidjs-css/src/primitives/index.ts +0 -1
  100. package/packages/solidjs-css/src/styles.css +0 -281
  101. package/packages/solidjs-css/tsconfig.json +0 -20
  102. package/packages/solidjs-css/tsup.config.ts +0 -10
  103. package/pnpm-workspace.yaml +0 -2
  104. package/tsconfig.json +0 -23
@@ -1,281 +0,0 @@
1
- /* ─── Feedback Design Tokens (OKLCH) ──────────────────── */
2
- :root {
3
- --feedback-bg: oklch(0.12 0.01 250);
4
- --feedback-surface: oklch(0.16 0.01 250);
5
- --feedback-border: oklch(0.22 0.01 250);
6
- --feedback-text: oklch(0.95 0.01 250);
7
- --feedback-text-muted: oklch(0.58 0.02 250);
8
- --feedback-accent: oklch(0.65 0.20 265);
9
- --feedback-open: oklch(0.65 0.20 245);
10
- --feedback-review: oklch(0.72 0.18 60);
11
- --feedback-planned: oklch(0.70 0.22 280);
12
- --feedback-progress: oklch(0.65 0.20 265);
13
- --feedback-done: oklch(0.72 0.18 155);
14
- --feedback-declined: oklch(0.50 0.10 250);
15
- --feedback-bug: oklch(0.60 0.25 25);
16
- --feedback-feature: oklch(0.70 0.22 280);
17
- --feedback-suggestion: oklch(0.72 0.18 60);
18
- --feedback-general: oklch(0.65 0.20 245);
19
- --feedback-radius: 0.75rem;
20
- }
21
-
22
- /* ─── Feedback Card ──────────────────────────────────── */
23
- .feedback__card {
24
- display: flex;
25
- gap: 0.75rem;
26
- padding: 1rem;
27
- border: 1px solid var(--feedback-border);
28
- border-radius: var(--feedback-radius);
29
- background: oklch(1 0 0 / 0.02);
30
- transition: all 0.2s;
31
- cursor: pointer;
32
- }
33
- .feedback__card:hover { border-color: oklch(0.65 0.20 265 / 0.2); background: oklch(1 0 0 / 0.04); }
34
- .feedback__card-info { flex: 1; min-width: 0; }
35
- .feedback__card-badges { display: flex; align-items: center; gap: 0.5rem; flex-wrap: wrap; margin-bottom: 0.25rem; }
36
- .feedback__card-title { font-size: 0.875rem; font-weight: 600; color: oklch(1 0 0 / 0.9); overflow: hidden; text-overflow: ellipsis; white-space: nowrap; margin-bottom: 0.125rem; }
37
- .feedback__card-desc { font-size: 0.6875rem; color: oklch(1 0 0 / 0.4); display: -webkit-box; -webkit-line-clamp: 2; -webkit-box-orient: vertical; overflow: hidden; }
38
- .feedback__card-meta { display: flex; align-items: center; gap: 0.75rem; margin-top: 0.5rem; font-size: 0.625rem; color: oklch(1 0 0 / 0.25); }
39
- .feedback__card--bug { border-left: 3px solid var(--feedback-bug); }
40
- .feedback__card--feature { border-left: 3px solid var(--feedback-feature); }
41
- .feedback__card--suggestion { border-left: 3px solid var(--feedback-suggestion); }
42
- .feedback__card--general { border-left: 3px solid var(--feedback-general); }
43
-
44
- /* ─── Vote Button ────────────────────────────────────── */
45
- .feedback__vote-btn {
46
- display: flex;
47
- flex-direction: column;
48
- align-items: center;
49
- gap: 0.125rem;
50
- padding: 0.375rem 0.5rem;
51
- border-radius: calc(var(--feedback-radius) * 0.8);
52
- border: none;
53
- font-size: 0.6875rem;
54
- font-weight: 700;
55
- cursor: pointer;
56
- transition: all 0.15s;
57
- font-variant-numeric: tabular-nums;
58
- }
59
- .feedback__vote-btn--inactive { background: oklch(1 0 0 / 0.05); color: oklch(1 0 0 / 0.3); }
60
- .feedback__vote-btn--inactive:hover { background: oklch(1 0 0 / 0.1); color: oklch(1 0 0 / 0.5); }
61
- .feedback__vote-btn--active { background: oklch(0.65 0.20 265 / 0.15); color: oklch(0.65 0.20 265); }
62
- .feedback__vote-btn-arrow { font-size: 0.875rem; }
63
-
64
- /* ─── Status Badge ───────────────────────────────────── */
65
- .feedback__status-badge { display: inline-flex; align-items: center; gap: 0.25rem; padding: 0.125rem 0.625rem; border-radius: 9999px; font-size: 0.625rem; font-weight: 500; }
66
- .feedback__status-badge--open { background: oklch(0.65 0.20 245 / 0.15); color: oklch(0.65 0.20 245); }
67
- .feedback__status-badge--under-review { background: oklch(0.72 0.18 60 / 0.15); color: oklch(0.72 0.18 60); }
68
- .feedback__status-badge--planned { background: oklch(0.70 0.22 280 / 0.15); color: oklch(0.70 0.22 280); }
69
- .feedback__status-badge--in-progress { background: oklch(0.65 0.20 265 / 0.15); color: oklch(0.65 0.20 265); }
70
- .feedback__status-badge--done { background: oklch(0.72 0.18 155 / 0.15); color: oklch(0.72 0.18 155); }
71
- .feedback__status-badge--declined { background: oklch(0.50 0.10 250 / 0.15); color: oklch(0.50 0.10 250); }
72
-
73
- /* ─── Priority Badge ─────────────────────────────────── */
74
- .feedback__priority-badge { padding: 0.125rem 0.5rem; border-radius: 9999px; font-size: 0.625rem; font-weight: 500; }
75
- .feedback__priority-badge--low { background: oklch(0.58 0.10 250 / 0.15); color: oklch(0.58 0.10 250); }
76
- .feedback__priority-badge--medium { background: oklch(0.72 0.18 60 / 0.15); color: oklch(0.72 0.18 60); }
77
- .feedback__priority-badge--high { background: oklch(0.68 0.22 35 / 0.15); color: oklch(0.68 0.22 35); }
78
- .feedback__priority-badge--critical { background: oklch(0.60 0.25 25 / 0.15); color: oklch(0.60 0.25 25); }
79
-
80
- /* ─── Type Badge ─────────────────────────────────────── */
81
- .feedback__type-badge { display: inline-flex; align-items: center; gap: 0.25rem; padding: 0.125rem 0.5rem; border-radius: 9999px; font-size: 0.625rem; font-weight: 500; }
82
- .feedback__type-badge--bug { background: oklch(0.60 0.25 25 / 0.15); color: oklch(0.60 0.25 25); }
83
- .feedback__type-badge--feature { background: oklch(0.70 0.22 280 / 0.15); color: oklch(0.70 0.22 280); }
84
- .feedback__type-badge--suggestion { background: oklch(0.72 0.18 60 / 0.15); color: oklch(0.72 0.18 60); }
85
- .feedback__type-badge--general { background: oklch(0.65 0.20 245 / 0.15); color: oklch(0.65 0.20 245); }
86
-
87
- /* ─── Widget ─────────────────────────────────────────── */
88
- .feedback__widget-trigger {
89
- position: fixed; top: 50%; right: 0; transform: translateY(-50%);
90
- z-index: 40; padding: 0.5rem 1rem; border-radius: var(--feedback-radius) 0 0 var(--feedback-radius);
91
- background: var(--feedback-accent); color: white; border: none; cursor: pointer;
92
- writing-mode: vertical-rl; font-size: 0.6875rem; font-weight: 500; letter-spacing: 0.05em;
93
- box-shadow: -2px 0 12px oklch(0.65 0.20 265 / 0.3);
94
- }
95
- .feedback__widget-trigger:hover { background: oklch(0.70 0.22 265); }
96
- .feedback__widget-overlay { position: fixed; inset: 0; z-index: 50; display: flex; justify-content: flex-end; }
97
- .feedback__widget-backdrop { position: absolute; inset: 0; background: oklch(0 0 0 / 0.5); backdrop-filter: blur(4px); }
98
- .feedback__widget-panel { position: relative; width: 100%; max-width: 28rem; background: oklch(0.06 0.01 250); border-left: 1px solid var(--feedback-border); padding: 1.5rem; overflow-y: auto; }
99
- .feedback__widget-header { display: flex; align-items: center; justify-content: space-between; margin-bottom: 1.5rem; }
100
- .feedback__widget-title { font-size: 1.125rem; font-weight: 700; color: var(--feedback-text); }
101
-
102
- /* ─── Form ───────────────────────────────────────────── */
103
- .feedback__form { display: flex; flex-direction: column; gap: 1rem; }
104
- .feedback__form-input { width: 100%; padding: 0.75rem 1rem; border: 1px solid var(--feedback-border); border-radius: var(--feedback-radius); background: oklch(1 0 0 / 0.03); font-size: 0.875rem; color: var(--feedback-text); outline: none; }
105
- .feedback__form-input::placeholder { color: oklch(1 0 0 / 0.3); }
106
- .feedback__form-input:focus { border-color: oklch(0.65 0.20 265 / 0.4); }
107
- .feedback__form-textarea { resize: none; min-height: 6rem; }
108
- .feedback__form-error { font-size: 0.6875rem; color: oklch(0.60 0.25 25); }
109
- .feedback__form-actions { display: flex; justify-content: flex-end; gap: 0.5rem; }
110
-
111
- /* ─── Type Selector ──────────────────────────────────── */
112
- .feedback__type-selector { display: flex; gap: 0.375rem; }
113
- .feedback__type-tab { display: flex; align-items: center; gap: 0.375rem; padding: 0.5rem 0.75rem; border-radius: calc(var(--feedback-radius) * 0.8); border: none; font-size: 0.6875rem; font-weight: 500; cursor: pointer; transition: all 0.15s; }
114
- .feedback__type-tab--inactive { background: oklch(1 0 0 / 0.05); color: oklch(1 0 0 / 0.5); }
115
- .feedback__type-tab--inactive:hover { background: oklch(1 0 0 / 0.1); }
116
-
117
- /* ─── NPS ────────────────────────────────────────────── */
118
- .feedback__nps { position: fixed; bottom: 1.5rem; right: 1.5rem; z-index: 50; width: 24rem; border: 1px solid oklch(1 0 0 / 0.1); border-radius: calc(var(--feedback-radius) * 1.5); background: oklch(0.08 0.01 250); padding: 1.5rem; box-shadow: 0 8px 32px oklch(0 0 0 / 0.5); }
119
- .feedback__nps-header { display: flex; align-items: center; justify-content: space-between; margin-bottom: 0.25rem; }
120
- .feedback__nps-title { font-size: 0.875rem; font-weight: 600; color: oklch(1 0 0 / 0.9); }
121
- .feedback__nps-subtitle { font-size: 0.6875rem; color: oklch(1 0 0 / 0.4); margin-bottom: 1rem; }
122
- .feedback__nps-scores { display: flex; gap: 0.25rem; }
123
- .feedback__nps-score-btn { flex: 1; padding: 0.625rem 0; border: none; border-radius: calc(var(--feedback-radius) * 0.8); background: oklch(1 0 0 / 0.05); color: oklch(1 0 0 / 0.5); font-size: 0.6875rem; font-weight: 700; cursor: pointer; transition: all 0.15s; }
124
- .feedback__nps-score-btn:hover { background: oklch(1 0 0 / 0.1); }
125
- .feedback__nps-labels { display: flex; justify-content: space-between; margin-top: 0.5rem; font-size: 0.625rem; color: oklch(1 0 0 / 0.2); }
126
- .feedback__nps-results { padding: 1.25rem; border: 1px solid var(--feedback-border); border-radius: var(--feedback-radius); background: oklch(1 0 0 / 0.02); }
127
- .feedback__nps-results-score { font-size: 1.875rem; font-weight: 900; font-variant-numeric: tabular-nums; }
128
- .feedback__nps-bar { display: flex; height: 1.25rem; overflow: hidden; border-radius: 9999px; margin: 1rem 0; }
129
- .feedback__nps-bar-detractors { background: oklch(0.60 0.25 25 / 0.7); }
130
- .feedback__nps-bar-passives { background: oklch(0.72 0.18 60 / 0.7); }
131
- .feedback__nps-bar-promoters { background: oklch(0.72 0.18 155 / 0.7); }
132
- .feedback__nps-breakdown { display: grid; grid-template-columns: repeat(3, 1fr); gap: 0.5rem; text-align: center; font-size: 0.6875rem; }
133
-
134
- /* ─── Board ──────────────────────────────────────────── */
135
- .feedback__board { display: flex; flex-direction: column; gap: 0.625rem; }
136
-
137
- /* ─── Admin Table ────────────────────────────────────── */
138
- .feedback__admin-table { width: 100%; font-size: 0.875rem; border-collapse: collapse; }
139
- .feedback__admin-table th { padding: 0.75rem 1rem; text-align: left; font-size: 0.625rem; font-weight: 500; text-transform: uppercase; letter-spacing: 0.05em; color: oklch(1 0 0 / 0.4); border-bottom: 1px solid oklch(1 0 0 / 0.08); }
140
- .feedback__admin-table td { padding: 0.75rem 1rem; border-bottom: 1px solid oklch(1 0 0 / 0.05); }
141
- .feedback__admin-table tr:hover { background: oklch(1 0 0 / 0.02); }
142
- .feedback__admin-select { padding: 0.25rem 0.5rem; border: 1px solid var(--feedback-border); border-radius: calc(var(--feedback-radius) * 0.6); background: oklch(1 0 0 / 0.05); font-size: 0.6875rem; color: var(--feedback-text); outline: none; }
143
-
144
- /* ─── Stats Grid ─────────────────────────────────────── */
145
- .feedback__stats-grid { display: grid; grid-template-columns: repeat(auto-fit, minmax(140px, 1fr)); gap: 1rem; margin-bottom: 2rem; }
146
- .feedback__stat-card { padding: 1rem; border: 1px solid var(--feedback-border); border-radius: var(--feedback-radius); background: oklch(1 0 0 / 0.02); }
147
- .feedback__stat-icon { font-size: 1.125rem; }
148
- .feedback__stat-value { font-size: 1.5rem; font-weight: 700; font-variant-numeric: tabular-nums; color: oklch(1 0 0 / 0.9); }
149
- .feedback__stat-label { font-size: 0.6875rem; color: oklch(1 0 0 / 0.4); }
150
-
151
- /* ─── Search ─────────────────────────────────────────── */
152
- .feedback__search { width: 100%; max-width: 16rem; padding: 0.625rem 1rem; border: 1px solid var(--feedback-border); border-radius: var(--feedback-radius); background: oklch(1 0 0 / 0.03); font-size: 0.875rem; color: var(--feedback-text); outline: none; }
153
- .feedback__search::placeholder { color: oklch(1 0 0 / 0.3); }
154
- .feedback__search:focus { border-color: oklch(0.65 0.20 265 / 0.4); }
155
-
156
- /* ─── Filter Tabs ────────────────────────────────────── */
157
- .feedback__filter-tabs { display: flex; flex-wrap: wrap; gap: 0.375rem; }
158
- .feedback__filter-tab { padding: 0.375rem 0.75rem; border: none; border-radius: calc(var(--feedback-radius) * 0.8); font-size: 0.6875rem; font-weight: 500; cursor: pointer; }
159
- .feedback__filter-tab--active { background: var(--feedback-accent); color: white; }
160
- .feedback__filter-tab--inactive { background: oklch(1 0 0 / 0.05); color: oklch(1 0 0 / 0.5); }
161
- .feedback__filter-tab--inactive:hover { background: oklch(1 0 0 / 0.1); }
162
-
163
- /* ─── Btn ────────────────────────────────────────────── */
164
- .feedback__btn { padding: 0.5rem 1rem; border-radius: calc(var(--feedback-radius) * 0.8); font-size: 0.6875rem; font-weight: 500; cursor: pointer; border: none; transition: all 0.15s; }
165
- .feedback__btn--primary { background: var(--feedback-accent); color: white; }
166
- .feedback__btn--primary:hover { background: oklch(0.70 0.22 265); }
167
- .feedback__btn--outline { border: 1px solid var(--feedback-border); background: transparent; color: oklch(1 0 0 / 0.6); }
168
- .feedback__btn--danger { background: oklch(0.60 0.25 25 / 0.2); color: oklch(0.60 0.25 25); }
169
-
170
- /* ─── Skeleton ───────────────────────────────────────── */
171
- .feedback__skeleton { background: oklch(1 0 0 / 0.05); border-radius: var(--feedback-radius); animation: feedback-pulse 1.5s ease-in-out infinite; }
172
- @keyframes feedback-pulse { 0%, 100% { opacity: 1; } 50% { opacity: 0.4; } }
173
-
174
- /* ─── Empty ──────────────────────────────────────────── */
175
- .feedback__empty { display: flex; flex-direction: column; align-items: center; justify-content: center; padding: 4rem 1rem; text-align: center; }
176
- .feedback__empty-icon { font-size: 3rem; opacity: 0.2; margin-bottom: 0.75rem; }
177
- .feedback__empty-text { font-size: 0.875rem; color: oklch(1 0 0 / 0.4); }
178
-
179
- /* ─── Success ────────────────────────────────────────── */
180
- .feedback__success { display: flex; flex-direction: column; align-items: center; padding: 2rem; text-align: center; }
181
- .feedback__success-icon { font-size: 2.5rem; margin-bottom: 0.75rem; }
182
- .feedback__success-text { font-size: 1rem; font-weight: 600; color: oklch(1 0 0 / 0.9); margin-bottom: 0.25rem; }
183
- .feedback__success-sub { font-size: 0.875rem; color: oklch(1 0 0 / 0.5); }
184
-
185
- /* ─── Tag ────────────────────────────────────────────── */
186
- .feedback__tag { padding: 0.125rem 0.375rem; border-radius: 0.25rem; background: oklch(1 0 0 / 0.05); font-size: 0.625rem; color: oklch(1 0 0 / 0.3); }
187
-
188
- /* ─── Close Button ───────────────────────────────────── */
189
- .feedback__close-btn { padding: 0.375rem; border: none; background: transparent; color: oklch(1 0 0 / 0.3); cursor: pointer; border-radius: calc(var(--feedback-radius) * 0.6); }
190
- .feedback__close-btn:hover { color: oklch(1 0 0 / 0.5); background: oklch(1 0 0 / 0.05); }
191
-
192
- /* ─── Comment Thread ─────────────────────────────────── */
193
- .feedback__comment-thread { display: flex; flex-direction: column; gap: 1rem; padding: 1rem; border: 1px solid var(--feedback-border); border-radius: var(--feedback-radius); background: oklch(1 0 0 / 0.01); }
194
- .feedback__comment { display: flex; gap: 0.75rem; }
195
- .feedback__comment-avatar { width: 2rem; height: 2rem; border-radius: 50%; background: oklch(0.65 0.20 265 / 0.2); display: flex; align-items: center; justify-content: center; font-size: 0.875rem; flex-shrink: 0; }
196
- .feedback__comment-body { flex: 1; }
197
- .feedback__comment-header { display: flex; align-items: center; gap: 0.5rem; margin-bottom: 0.25rem; }
198
- .feedback__comment-author { font-size: 0.75rem; font-weight: 600; color: oklch(1 0 0 / 0.8); }
199
- .feedback__comment-time { font-size: 0.625rem; color: oklch(1 0 0 / 0.3); }
200
- .feedback__comment-text { font-size: 0.6875rem; color: oklch(1 0 0 / 0.7); line-height: 1.4; }
201
- .feedback__comment-reply-btn { margin-top: 0.5rem; padding: 0.25rem 0.5rem; font-size: 0.625rem; border: none; background: transparent; color: oklch(0.65 0.20 265); cursor: pointer; transition: all 0.1s; }
202
- .feedback__comment-reply-btn:hover { color: oklch(0.70 0.22 265); }
203
-
204
- /* ─── Rating Component ───────────────────────────────── */
205
- .feedback__rating { display: flex; gap: 0.25rem; }
206
- .feedback__rating-star { font-size: 1.5rem; cursor: pointer; opacity: 0.3; transition: all 0.15s; }
207
- .feedback__rating-star--filled { opacity: 1; color: oklch(0.72 0.18 60); }
208
- .feedback__rating-star:hover { opacity: 1; transform: scale(1.1); }
209
-
210
- /* ─── Sentiment Indicator ────────────────────────────── */
211
- .feedback__sentiment-card { padding: 1rem; border: 1px solid var(--feedback-border); border-radius: var(--feedback-radius); background: oklch(1 0 0 / 0.02); text-align: center; }
212
- .feedback__sentiment-icon { font-size: 2rem; margin-bottom: 0.5rem; }
213
- .feedback__sentiment-label { font-size: 0.875rem; font-weight: 600; color: oklch(1 0 0 / 0.8); margin-bottom: 0.25rem; }
214
- .feedback__sentiment-count { font-size: 0.6875rem; color: oklch(1 0 0 / 0.4); }
215
-
216
- /* ─── Response Panel ─────────────────────────────────── */
217
- .feedback__response-panel { border: 1px solid var(--feedback-border); border-radius: var(--feedback-radius); background: oklch(1 0 0 / 0.02); padding: 1rem; }
218
- .feedback__response-header { display: flex; align-items: center; justify-content: space-between; margin-bottom: 0.75rem; }
219
- .feedback__response-title { font-size: 0.875rem; font-weight: 600; color: oklch(1 0 0 / 0.8); }
220
- .feedback__response-time { font-size: 0.625rem; color: oklch(1 0 0 / 0.3); }
221
- .feedback__response-body { font-size: 0.6875rem; color: oklch(1 0 0 / 0.7); line-height: 1.5; }
222
- .feedback__response-actions { display: flex; gap: 0.5rem; margin-top: 0.75rem; }
223
- .feedback__response-btn { padding: 0.25rem 0.5rem; font-size: 0.625rem; border: none; background: oklch(1 0 0 / 0.05); color: oklch(1 0 0 / 0.5); cursor: pointer; border-radius: calc(var(--feedback-radius) * 0.6); transition: all 0.1s; }
224
- .feedback__response-btn:hover { background: oklch(1 0 0 / 0.1); color: oklch(1 0 0 / 0.6); }
225
-
226
- /* ─── Analytics Chart ────────────────────────────────── */
227
- .feedback__analytics { border: 1px solid var(--feedback-border); border-radius: var(--feedback-radius); background: oklch(1 0 0 / 0.02); padding: 1rem; }
228
- .feedback__analytics-title { font-size: 0.875rem; font-weight: 600; color: oklch(1 0 0 / 0.8); margin-bottom: 1rem; }
229
- .feedback__analytics-chart { display: flex; align-items: flex-end; gap: 0.5rem; height: 10rem; }
230
- .feedback__analytics-bar { flex: 1; border-radius: 0.25rem 0.25rem 0 0; background: linear-gradient(to top, oklch(0.65 0.20 265), oklch(0.65 0.20 265 / 0.4)); transition: all 0.2s; position: relative; }
231
- .feedback__analytics-bar:hover { background: linear-gradient(to top, oklch(0.70 0.22 265), oklch(0.70 0.22 265)); }
232
- .feedback__analytics-label { position: absolute; bottom: -1.5rem; left: 50%; transform: translateX(-50%); font-size: 0.625rem; color: oklch(1 0 0 / 0.4); white-space: nowrap; }
233
-
234
- /* ─── Filter Control Panel ───────────────────────────── */
235
- .feedback__filter-panel { border: 1px solid var(--feedback-border); border-radius: var(--feedback-radius); background: oklch(1 0 0 / 0.02); padding: 1rem; }
236
- .feedback__filter-section { margin-bottom: 1rem; }
237
- .feedback__filter-section:last-child { margin-bottom: 0; }
238
- .feedback__filter-section-title { font-size: 0.6875rem; font-weight: 600; text-transform: uppercase; color: oklch(1 0 0 / 0.4); margin-bottom: 0.5rem; letter-spacing: 0.05em; }
239
- .feedback__filter-options { display: flex; flex-direction: column; gap: 0.25rem; }
240
- .feedback__filter-option { display: flex; align-items: center; gap: 0.5rem; padding: 0.25rem 0; font-size: 0.6875rem; }
241
- .feedback__filter-option input { cursor: pointer; }
242
- .feedback__filter-option-label { color: oklch(1 0 0 / 0.7); cursor: pointer; }
243
-
244
- /* ─── Tag Selector ───────────────────────────────────── */
245
- .feedback__tag-selector { padding: 1rem; border: 1px solid var(--feedback-border); border-radius: var(--feedback-radius); background: oklch(1 0 0 / 0.02); }
246
- .feedback__tag-selector-label { display: block; font-size: 0.6875rem; font-weight: 500; color: oklch(1 0 0 / 0.6); margin-bottom: 0.5rem; text-transform: uppercase; }
247
- .feedback__tag-list { display: flex; flex-wrap: wrap; gap: 0.5rem; }
248
- .feedback__tag-item { display: inline-flex; align-items: center; gap: 0.25rem; padding: 0.375rem 0.75rem; border-radius: 9999px; background: oklch(0.65 0.20 265 / 0.15); color: oklch(0.65 0.20 265); font-size: 0.6875rem; font-weight: 500; }
249
- .feedback__tag-remove { margin-left: 0.25rem; cursor: pointer; font-weight: 700; opacity: 0.6; transition: opacity 0.1s; }
250
- .feedback__tag-remove:hover { opacity: 1; }
251
-
252
- /* ─── Feedback Form Actions ──────────────────────────── */
253
- .feedback__form-footer { display: flex; justify-content: space-between; align-items: center; padding-top: 1rem; border-top: 1px solid var(--feedback-border); }
254
- .feedback__form-counter { font-size: 0.625rem; color: oklch(1 0 0 / 0.3); }
255
- .feedback__form-actions-group { display: flex; gap: 0.5rem; }
256
-
257
- /* ─── Feedback Status Indicator ──────────────────────── */
258
- .feedback__status-indicator { display: inline-flex; align-items: center; gap: 0.375rem; padding: 0.375rem 0.75rem; border-radius: 9999px; font-size: 0.625rem; font-weight: 500; }
259
- .feedback__status-indicator--new { background: oklch(0.65 0.20 245 / 0.15); color: oklch(0.65 0.20 245); }
260
- .feedback__status-indicator--in-review { background: oklch(0.72 0.18 60 / 0.15); color: oklch(0.72 0.18 60); }
261
- .feedback__status-indicator--implemented { background: oklch(0.72 0.18 155 / 0.15); color: oklch(0.72 0.18 155); }
262
-
263
- /* ─── Detailed Stats Row ─────────────────────────────── */
264
- .feedback__detailed-stats { display: grid; grid-template-columns: repeat(auto-fit, minmax(120px, 1fr)); gap: 0.5rem; }
265
- .feedback__detailed-stat-item { padding: 0.75rem; border: 1px solid var(--feedback-border); border-radius: var(--feedback-radius); background: oklch(1 0 0 / 0.01); text-align: center; }
266
- .feedback__detailed-stat-icon { font-size: 1.25rem; margin-bottom: 0.25rem; }
267
- .feedback__detailed-stat-value { font-size: 1rem; font-weight: 700; color: oklch(1 0 0 / 0.9); }
268
- .feedback__detailed-stat-label { font-size: 0.625rem; color: oklch(1 0 0 / 0.4); margin-top: 0.25rem; }
269
-
270
- /* ─── Breadcrumb Navigation ──────────────────────────── */
271
- .feedback__breadcrumb { display: flex; align-items: center; gap: 0.5rem; font-size: 0.875rem; margin-bottom: 1rem; }
272
- .feedback__breadcrumb-item { color: oklch(1 0 0 / 0.5); }
273
- .feedback__breadcrumb-item--active { color: oklch(1 0 0 / 0.8); font-weight: 500; }
274
- .feedback__breadcrumb-sep { color: oklch(1 0 0 / 0.2); margin: 0 0.25rem; }
275
-
276
- /* ─── Pagination Controls ────────────────────────────── */
277
- .feedback__pagination { display: flex; align-items: center; gap: 0.5rem; justify-content: center; padding: 1rem; }
278
- .feedback__pagination-btn { padding: 0.375rem 0.75rem; border: 1px solid var(--feedback-border); border-radius: calc(var(--feedback-radius) * 0.6); background: oklch(1 0 0 / 0.05); color: oklch(1 0 0 / 0.6); font-size: 0.6875rem; cursor: pointer; transition: all 0.1s; }
279
- .feedback__pagination-btn:hover:not(:disabled) { background: oklch(1 0 0 / 0.1); color: oklch(1 0 0 / 0.8); }
280
- .feedback__pagination-btn:disabled { opacity: 0.5; cursor: not-allowed; }
281
- .feedback__pagination-info { font-size: 0.6875rem; color: oklch(1 0 0 / 0.5); }
@@ -1,19 +0,0 @@
1
- {
2
- "extends": "../../tsconfig.json",
3
- "compilerOptions": {
4
- "outDir": "./dist",
5
- "rootDir": "./src",
6
- "jsx": "react-jsx",
7
- "strict": true,
8
- "skipLibCheck": true,
9
- "forceConsistentCasingInFileNames": true,
10
- "resolveJsonModule": true,
11
- "isolatedModules": true,
12
- "target": "ES2022",
13
- "module": "ESNext",
14
- "moduleResolution": "bundler"
15
- },
16
- "include": [
17
- "src"
18
- ]
19
- }
@@ -1,10 +0,0 @@
1
- import { defineConfig } from 'tsup'
2
-
3
- export default defineConfig({
4
- entry: ['src/index.tsx'],
5
- format: ['cjs', 'esm'],
6
- dts: true,
7
- clean: true,
8
- sourcemap: true,
9
- external: ['react'],
10
- })
@@ -1,44 +0,0 @@
1
- {
2
- "name": "@geenius-feedback/shared",
3
- "version": "0.1.0",
4
- "private": false,
5
- "type": "module",
6
- "description": "Geenius Feedback \u2014 Shared types & Convex schema",
7
- "author": "Antigravity HQ",
8
- "license": "MIT",
9
- "publishConfig": {
10
- "access": "public"
11
- },
12
- "main": "./dist/index.js",
13
- "module": "./dist/index.js",
14
- "types": "./dist/index.d.ts",
15
- "exports": {
16
- ".": {
17
- "types": "./dist/index.d.ts",
18
- "import": "./dist/index.js"
19
- },
20
- "./convex": "./src/convex.ts"
21
- },
22
- "files": [
23
- "dist",
24
- "src"
25
- ],
26
- "scripts": {
27
- "build": "tsup",
28
- "clean": "rm -rf dist",
29
- "type-check": "tsc --noEmit",
30
- "prepublishOnly": "pnpm clean && pnpm build",
31
- "test": "vitest run",
32
- "test:watch": "vitest",
33
- "test:coverage": "vitest run --coverage"
34
- },
35
- "devDependencies": {
36
- "convex": "^1.34.0",
37
- "tsup": "^8.5.1",
38
- "typescript": "~6.0.2",
39
- "vitest": "^4.0.0"
40
- },
41
- "engines": {
42
- "node": ">=20.0.0"
43
- }
44
- }
@@ -1,72 +0,0 @@
1
- import { describe, it, expect } from 'vitest'
2
- import {
3
- STATUS_CONFIG, PRIORITY_CONFIG, TYPE_CONFIG,
4
- FEEDBACK_TYPES, FEEDBACK_STATUSES, FEEDBACK_PRIORITIES,
5
- getStatusTransitions, getNPSCategory, calcNPSScore,
6
- } from '../index'
7
-
8
- describe('Feedback Constants', () => {
9
- it('FEEDBACK_TYPES has 4 types', () => {
10
- expect(FEEDBACK_TYPES).toHaveLength(4)
11
- expect(FEEDBACK_TYPES).toContain('bug')
12
- expect(FEEDBACK_TYPES).toContain('feature')
13
- })
14
-
15
- it('FEEDBACK_STATUSES has 6 statuses', () => {
16
- expect(FEEDBACK_STATUSES).toHaveLength(6)
17
- expect(FEEDBACK_STATUSES).toContain('open')
18
- expect(FEEDBACK_STATUSES).toContain('done')
19
- })
20
-
21
- it('STATUS_CONFIG maps every status to label+color', () => {
22
- FEEDBACK_STATUSES.forEach(s => {
23
- expect(STATUS_CONFIG[s]).toBeDefined()
24
- expect(STATUS_CONFIG[s].label).toBeDefined()
25
- })
26
- })
27
-
28
- it('PRIORITY_CONFIG has low through critical', () => {
29
- FEEDBACK_PRIORITIES.forEach(p => {
30
- expect(PRIORITY_CONFIG[p]).toBeDefined()
31
- })
32
- })
33
- })
34
-
35
- describe('Status Transitions', () => {
36
- it('open can transition to under-review', () => {
37
- const transitions = getStatusTransitions('open')
38
- expect(transitions).toContain('under-review')
39
- })
40
-
41
- it('done has limited transitions', () => {
42
- const transitions = getStatusTransitions('done')
43
- expect(Array.isArray(transitions)).toBe(true)
44
- })
45
- })
46
-
47
- describe('NPS Scoring', () => {
48
- it('getNPSCategory detractor for 0-6', () => {
49
- expect(getNPSCategory(0)).toBe('detractor')
50
- expect(getNPSCategory(6)).toBe('detractor')
51
- })
52
-
53
- it('getNPSCategory passive for 7-8', () => {
54
- expect(getNPSCategory(7)).toBe('passive')
55
- expect(getNPSCategory(8)).toBe('passive')
56
- })
57
-
58
- it('getNPSCategory promoter for 9-10', () => {
59
- expect(getNPSCategory(9)).toBe('promoter')
60
- expect(getNPSCategory(10)).toBe('promoter')
61
- })
62
-
63
- it('calcNPSScore computes score from responses', () => {
64
- const responses = [
65
- { score: 10 }, { score: 9 }, { score: 5 }, { score: 3 },
66
- ] as any[]
67
- const nps = calcNPSScore(responses)
68
- expect(typeof nps).toBe('number')
69
- expect(nps).toBeGreaterThanOrEqual(-100)
70
- expect(nps).toBeLessThanOrEqual(100)
71
- })
72
- })
@@ -1,49 +0,0 @@
1
- /**
2
- * @fileoverview Configuration factory for Geenius Feedback
3
- */
4
-
5
- import type { FeedbackConfig } from './types'
6
-
7
- export interface ConfigureFeedbackOptions {
8
- floatingWidget?: boolean
9
- npsEnabled?: boolean
10
- npsInterval?: number
11
- categories?: Array<'bug' | 'feature' | 'general' | 'suggestion'>
12
- allowAnonymous?: boolean
13
- }
14
-
15
- /**
16
- * Configure the feedback system with custom options
17
- * @param options Configuration options for the feedback system
18
- * @returns FeedbackConfig object ready for use
19
- * @example
20
- * ```ts
21
- * const config = configureFeedback({
22
- * floatingWidget: true,
23
- * npsEnabled: true,
24
- * npsInterval: 7,
25
- * categories: ['bug', 'feature', 'general'],
26
- * allowAnonymous: true,
27
- * })
28
- * ```
29
- */
30
- export function configureFeedback(options: ConfigureFeedbackOptions = {}): FeedbackConfig {
31
- return {
32
- floatingWidget: options.floatingWidget ?? true,
33
- npsEnabled: options.npsEnabled ?? true,
34
- npsInterval: options.npsInterval ?? 7,
35
- categories: options.categories ?? ['bug', 'feature', 'general', 'suggestion'],
36
- allowAnonymous: options.allowAnonymous ?? true,
37
- }
38
- }
39
-
40
- /**
41
- * Default feedback configuration
42
- */
43
- export const defaultFeedbackConfig: FeedbackConfig = {
44
- floatingWidget: true,
45
- npsEnabled: true,
46
- npsInterval: 7,
47
- categories: ['bug', 'feature', 'general', 'suggestion'],
48
- allowAnonymous: true,
49
- }
@@ -1,111 +0,0 @@
1
- export type {
2
- FeedbackType, FeedbackStatus, FeedbackPriority, NPSCategory,
3
- Attachment, FeedbackItem, NPSResponse, FeedbackConfig,
4
- FeedbackStats, NPSStats,
5
- } from './types'
6
-
7
- export { configureFeedback, defaultFeedbackConfig } from './config'
8
- export type { ConfigureFeedbackOptions } from './config'
9
-
10
- import type { FeedbackStatus, FeedbackPriority, FeedbackType, NPSResponse, FeedbackStats, NPSStats, FeedbackItem } from './types'
11
-
12
- // ─── Status Config ───────────────────────────────────
13
- export const STATUS_CONFIG: Record<FeedbackStatus, { label: string; color: string; emoji: string }> = {
14
- open: { label: 'Open', color: 'oklch(0.65 0.20 245)', emoji: '📬' },
15
- 'under-review': { label: 'Under Review', color: 'oklch(0.72 0.18 60)', emoji: '🔍' },
16
- planned: { label: 'Planned', color: 'oklch(0.70 0.22 280)', emoji: '📋' },
17
- 'in-progress': { label: 'In Progress', color: 'oklch(0.65 0.20 265)', emoji: '🚧' },
18
- done: { label: 'Done', color: 'oklch(0.72 0.18 155)', emoji: '✅' },
19
- declined: { label: 'Declined', color: 'oklch(0.50 0.10 250)', emoji: '🚫' },
20
- }
21
-
22
- export const PRIORITY_CONFIG: Record<FeedbackPriority, { label: string; color: string }> = {
23
- low: { label: 'Low', color: 'oklch(0.58 0.10 250)' },
24
- medium: { label: 'Medium', color: 'oklch(0.72 0.18 60)' },
25
- high: { label: 'High', color: 'oklch(0.68 0.22 35)' },
26
- critical: { label: 'Critical', color: 'oklch(0.60 0.25 25)' },
27
- }
28
-
29
- export const TYPE_CONFIG: Record<FeedbackType, { label: string; icon: string; color: string }> = {
30
- bug: { label: 'Bug', icon: '🐛', color: 'oklch(0.60 0.25 25)' },
31
- feature: { label: 'Feature', icon: '✨', color: 'oklch(0.70 0.22 280)' },
32
- suggestion: { label: 'Suggestion', icon: '💡', color: 'oklch(0.72 0.18 60)' },
33
- general: { label: 'General', icon: '💬', color: 'oklch(0.65 0.20 245)' },
34
- }
35
-
36
- export const FEEDBACK_TYPES: FeedbackType[] = ['bug', 'feature', 'suggestion', 'general']
37
- export const FEEDBACK_STATUSES: FeedbackStatus[] = ['open', 'under-review', 'planned', 'in-progress', 'done', 'declined']
38
- export const FEEDBACK_PRIORITIES: FeedbackPriority[] = ['low', 'medium', 'high', 'critical']
39
-
40
- // ─── Status Transitions ─────────────────────────────
41
- const STATUS_TRANSITIONS: Record<FeedbackStatus, FeedbackStatus[]> = {
42
- open: ['under-review', 'planned', 'declined'],
43
- 'under-review': ['planned', 'in-progress', 'declined'],
44
- planned: ['in-progress', 'declined'],
45
- 'in-progress': ['done', 'planned'],
46
- done: ['open'],
47
- declined: ['open'],
48
- }
49
-
50
- export function getStatusTransitions(status: FeedbackStatus): FeedbackStatus[] {
51
- return STATUS_TRANSITIONS[status] ?? []
52
- }
53
-
54
- // ─── NPS Utilities ───────────────────────────────────
55
- export function getNPSCategory(score: number): 'detractor' | 'passive' | 'promoter' {
56
- if (score <= 6) return 'detractor'
57
- if (score <= 8) return 'passive'
58
- return 'promoter'
59
- }
60
-
61
- export function calcNPSScore(responses: NPSResponse[]): number {
62
- if (responses.length === 0) return 0
63
- let promoters = 0, detractors = 0
64
- for (const r of responses) {
65
- const cat = getNPSCategory(r.score)
66
- if (cat === 'promoter') promoters++
67
- else if (cat === 'detractor') detractors++
68
- }
69
- return Math.round(((promoters - detractors) / responses.length) * 100)
70
- }
71
-
72
- export function calcNPSStats(responses: NPSResponse[]): NPSStats {
73
- let promoters = 0, passives = 0, detractors = 0, totalScore = 0
74
- for (const r of responses) {
75
- totalScore += r.score
76
- const cat = getNPSCategory(r.score)
77
- if (cat === 'promoter') promoters++
78
- else if (cat === 'passive') passives++
79
- else detractors++
80
- }
81
- return {
82
- averageScore: responses.length > 0 ? Math.round((totalScore / responses.length) * 10) / 10 : 0,
83
- totalResponses: responses.length,
84
- promoters, passives, detractors,
85
- npsScore: calcNPSScore(responses),
86
- }
87
- }
88
-
89
- export function calcFeedbackStats(items: FeedbackItem[]): FeedbackStats {
90
- const byType = { bug: 0, feature: 0, suggestion: 0, general: 0 }
91
- const byStatus = { open: 0, 'under-review': 0, planned: 0, 'in-progress': 0, done: 0, declined: 0 }
92
- const byPriority = { low: 0, medium: 0, high: 0, critical: 0 }
93
- for (const item of items) {
94
- byType[item.type]++
95
- byStatus[item.status]++
96
- byPriority[item.priority]++
97
- }
98
- return { total: items.length, byType, byStatus, byPriority }
99
- }
100
-
101
- export function formatRelativeTime(dateStr: string): string {
102
- const diff = Date.now() - new Date(dateStr).getTime()
103
- const mins = Math.floor(diff / 60000)
104
- if (mins < 1) return 'just now'
105
- if (mins < 60) return `${mins}m ago`
106
- const hrs = Math.floor(mins / 60)
107
- if (hrs < 24) return `${hrs}h ago`
108
- const days = Math.floor(hrs / 24)
109
- if (days < 30) return `${days}d ago`
110
- return new Date(dateStr).toLocaleDateString()
111
- }
@@ -1,59 +0,0 @@
1
- export type FeedbackType = 'bug' | 'feature' | 'general' | 'suggestion'
2
- export type FeedbackStatus = 'open' | 'under-review' | 'planned' | 'in-progress' | 'done' | 'declined'
3
- export type FeedbackPriority = 'low' | 'medium' | 'high' | 'critical'
4
- export type NPSCategory = 'detractor' | 'passive' | 'promoter'
5
-
6
- export interface Attachment { url: string; name: string; size: number }
7
-
8
- export interface FeedbackItem {
9
- id: string
10
- type: FeedbackType
11
- title: string
12
- description: string
13
- status: FeedbackStatus
14
- priority: FeedbackPriority
15
- userId?: string
16
- userEmail?: string
17
- userName?: string
18
- url?: string
19
- browser?: string
20
- os?: string
21
- tags: string[]
22
- votes: number
23
- createdAt: string
24
- updatedAt: string
25
- attachments?: Attachment[]
26
- adminNote?: string
27
- }
28
-
29
- export interface NPSResponse {
30
- id: string
31
- score: number
32
- comment?: string
33
- userId?: string
34
- submittedAt: string
35
- }
36
-
37
- export interface FeedbackConfig {
38
- floatingWidget: boolean
39
- npsEnabled: boolean
40
- npsInterval: number
41
- categories: FeedbackType[]
42
- allowAnonymous: boolean
43
- }
44
-
45
- export interface FeedbackStats {
46
- total: number
47
- byType: Record<FeedbackType, number>
48
- byStatus: Record<FeedbackStatus, number>
49
- byPriority: Record<FeedbackPriority, number>
50
- }
51
-
52
- export interface NPSStats {
53
- averageScore: number
54
- totalResponses: number
55
- promoters: number
56
- passives: number
57
- detractors: number
58
- npsScore: number
59
- }