@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.
- package/package.json +16 -3
- package/packages/convex/dist/index.d.ts +192 -0
- package/packages/convex/dist/index.js +239 -0
- package/packages/convex/dist/index.js.map +1 -0
- package/packages/react/README.md +1 -1
- package/packages/react/dist/index.d.ts +146 -0
- package/packages/react/dist/index.js +545 -0
- package/packages/react/dist/index.js.map +1 -0
- package/packages/react-css/README.md +1 -1
- package/packages/react-css/dist/index.css +965 -0
- package/packages/react-css/dist/index.css.map +1 -0
- package/packages/react-css/dist/index.d.ts +49 -0
- package/packages/react-css/dist/index.js +228 -0
- package/packages/react-css/dist/index.js.map +1 -0
- package/packages/shared/README.md +1 -1
- package/packages/shared/dist/index.d.ts +115 -0
- package/packages/shared/dist/index.js +112 -0
- package/packages/shared/dist/index.js.map +1 -0
- package/packages/solidjs/README.md +1 -1
- package/packages/solidjs/dist/index.d.ts +128 -0
- package/packages/solidjs/dist/index.js +289 -0
- package/packages/solidjs/dist/index.js.map +1 -0
- package/packages/solidjs-css/README.md +1 -1
- package/packages/solidjs-css/dist/index.css +965 -0
- package/packages/solidjs-css/dist/index.css.map +1 -0
- package/packages/solidjs-css/dist/index.d.ts +2 -0
- package/packages/solidjs-css/dist/index.js +29 -0
- package/packages/solidjs-css/dist/index.js.map +1 -0
- package/.changeset/config.json +0 -11
- package/.github/CODEOWNERS +0 -1
- package/.github/ISSUE_TEMPLATE/bug_report.md +0 -16
- package/.github/ISSUE_TEMPLATE/feature_request.md +0 -11
- package/.github/PULL_REQUEST_TEMPLATE.md +0 -10
- package/.github/dependabot.yml +0 -11
- package/.github/workflows/ci.yml +0 -23
- package/.github/workflows/release.yml +0 -29
- package/.nvmrc +0 -1
- package/.project/ACCOUNT.yaml +0 -4
- package/.project/IDEAS.yaml +0 -7
- package/.project/PROJECT.yaml +0 -11
- package/.project/ROADMAP.yaml +0 -15
- package/CODE_OF_CONDUCT.md +0 -16
- package/CONTRIBUTING.md +0 -26
- package/SECURITY.md +0 -15
- package/SUPPORT.md +0 -8
- package/packages/convex/package.json +0 -42
- package/packages/convex/src/index.ts +0 -3
- package/packages/convex/src/mutations.ts +0 -88
- package/packages/convex/src/queries.ts +0 -78
- package/packages/convex/src/schema.ts +0 -47
- package/packages/convex/tsconfig.json +0 -18
- package/packages/convex/tsup.config.ts +0 -17
- package/packages/react/package.json +0 -49
- package/packages/react/src/components/FeedbackCard.tsx +0 -51
- package/packages/react/src/components/FeedbackForm.tsx +0 -43
- package/packages/react/src/components/FeedbackWidget.tsx +0 -32
- package/packages/react/src/components/NPSSurvey.tsx +0 -62
- package/packages/react/src/components/index.ts +0 -4
- package/packages/react/src/hooks/index.ts +0 -5
- package/packages/react/src/hooks/useFeedback.ts +0 -23
- package/packages/react/src/hooks/useFeedbackAdmin.ts +0 -24
- package/packages/react/src/hooks/useFeedbackForm.ts +0 -35
- package/packages/react/src/hooks/useNPS.ts +0 -26
- package/packages/react/src/index.tsx +0 -13
- package/packages/react/src/pages/FeedbackAdminPage.tsx +0 -71
- package/packages/react/src/pages/FeedbackPublicPage.tsx +0 -42
- package/packages/react/src/pages/FeedbackWidgetPage.tsx +0 -25
- package/packages/react/src/pages/index.ts +0 -3
- package/packages/react/tsconfig.json +0 -19
- package/packages/react/tsup.config.ts +0 -12
- package/packages/react-css/package.json +0 -36
- package/packages/react-css/src/components/index.ts +0 -5
- package/packages/react-css/src/components/index.tsx +0 -107
- package/packages/react-css/src/hooks/index.ts +0 -2
- package/packages/react-css/src/index.tsx +0 -5
- package/packages/react-css/src/pages/FeedbackAdminPage.tsx +0 -112
- package/packages/react-css/src/pages/FeedbackPage.tsx +0 -76
- package/packages/react-css/src/styles.css +0 -281
- package/packages/react-css/tsconfig.json +0 -19
- package/packages/react-css/tsup.config.ts +0 -10
- package/packages/shared/package.json +0 -44
- package/packages/shared/src/__tests__/feedback.test.ts +0 -72
- package/packages/shared/src/config.ts +0 -49
- package/packages/shared/src/index.ts +0 -111
- package/packages/shared/src/types.ts +0 -59
- package/packages/shared/tsconfig.json +0 -18
- package/packages/shared/tsup.config.ts +0 -11
- package/packages/shared/vitest.config.ts +0 -4
- package/packages/solidjs/package.json +0 -45
- package/packages/solidjs/src/components.tsx +0 -72
- package/packages/solidjs/src/index.tsx +0 -3
- package/packages/solidjs/src/primitives.ts +0 -49
- package/packages/solidjs/tsconfig.json +0 -20
- package/packages/solidjs/tsup.config.ts +0 -12
- package/packages/solidjs-css/package.json +0 -32
- package/packages/solidjs-css/src/index.tsx +0 -4
- package/packages/solidjs-css/src/pages/FeedbackAdminPage.tsx +0 -78
- package/packages/solidjs-css/src/pages/FeedbackPage.tsx +0 -65
- package/packages/solidjs-css/src/primitives/index.ts +0 -1
- package/packages/solidjs-css/src/styles.css +0 -281
- package/packages/solidjs-css/tsconfig.json +0 -20
- package/packages/solidjs-css/tsup.config.ts +0 -10
- package/pnpm-workspace.yaml +0 -2
- 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,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
|
-
}
|