yo-bug 0.1.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 (140) hide show
  1. package/LICENSE +21 -0
  2. package/README.md +160 -0
  3. package/bin/cli.ts +17 -0
  4. package/bin/install.ts +34 -0
  5. package/bin/mcp.ts +2 -0
  6. package/dist/bin/cli.d.ts +3 -0
  7. package/dist/bin/cli.d.ts.map +1 -0
  8. package/dist/bin/cli.js +19 -0
  9. package/dist/bin/cli.js.map +1 -0
  10. package/dist/bin/install.d.ts +3 -0
  11. package/dist/bin/install.d.ts.map +1 -0
  12. package/dist/bin/install.js +28 -0
  13. package/dist/bin/install.js.map +1 -0
  14. package/dist/bin/mcp.d.ts +3 -0
  15. package/dist/bin/mcp.d.ts.map +1 -0
  16. package/dist/bin/mcp.js +3 -0
  17. package/dist/bin/mcp.js.map +1 -0
  18. package/dist/src/detect/dev-server.d.ts +11 -0
  19. package/dist/src/detect/dev-server.d.ts.map +1 -0
  20. package/dist/src/detect/dev-server.js +92 -0
  21. package/dist/src/detect/dev-server.js.map +1 -0
  22. package/dist/src/detect/spawn-dev.d.ts +4 -0
  23. package/dist/src/detect/spawn-dev.d.ts.map +1 -0
  24. package/dist/src/detect/spawn-dev.js +37 -0
  25. package/dist/src/detect/spawn-dev.js.map +1 -0
  26. package/dist/src/install/detect-ai-tool.d.ts +7 -0
  27. package/dist/src/install/detect-ai-tool.d.ts.map +1 -0
  28. package/dist/src/install/detect-ai-tool.js +38 -0
  29. package/dist/src/install/detect-ai-tool.js.map +1 -0
  30. package/dist/src/install/write-mcp-config.d.ts +6 -0
  31. package/dist/src/install/write-mcp-config.d.ts.map +1 -0
  32. package/dist/src/install/write-mcp-config.js +47 -0
  33. package/dist/src/install/write-mcp-config.js.map +1 -0
  34. package/dist/src/mcp/index.d.ts +2 -0
  35. package/dist/src/mcp/index.d.ts.map +1 -0
  36. package/dist/src/mcp/index.js +263 -0
  37. package/dist/src/mcp/index.js.map +1 -0
  38. package/dist/src/mcp/tools/checklist.d.ts +22 -0
  39. package/dist/src/mcp/tools/checklist.d.ts.map +1 -0
  40. package/dist/src/mcp/tools/checklist.js +58 -0
  41. package/dist/src/mcp/tools/checklist.js.map +1 -0
  42. package/dist/src/mcp/tools/get-feedback.d.ts +13 -0
  43. package/dist/src/mcp/tools/get-feedback.d.ts.map +1 -0
  44. package/dist/src/mcp/tools/get-feedback.js +27 -0
  45. package/dist/src/mcp/tools/get-feedback.js.map +1 -0
  46. package/dist/src/mcp/tools/list-feedbacks.d.ts +20 -0
  47. package/dist/src/mcp/tools/list-feedbacks.d.ts.map +1 -0
  48. package/dist/src/mcp/tools/list-feedbacks.js +29 -0
  49. package/dist/src/mcp/tools/list-feedbacks.js.map +1 -0
  50. package/dist/src/mcp/tools/resolve-feedback.d.ts +8 -0
  51. package/dist/src/mcp/tools/resolve-feedback.d.ts.map +1 -0
  52. package/dist/src/mcp/tools/resolve-feedback.js +28 -0
  53. package/dist/src/mcp/tools/resolve-feedback.js.map +1 -0
  54. package/dist/src/mcp/tools/start-session.d.ts +11 -0
  55. package/dist/src/mcp/tools/start-session.d.ts.map +1 -0
  56. package/dist/src/mcp/tools/start-session.js +56 -0
  57. package/dist/src/mcp/tools/start-session.js.map +1 -0
  58. package/dist/src/mcp/tools/stop-session.d.ts +6 -0
  59. package/dist/src/mcp/tools/stop-session.d.ts.map +1 -0
  60. package/dist/src/mcp/tools/stop-session.js +80 -0
  61. package/dist/src/mcp/tools/stop-session.js.map +1 -0
  62. package/dist/src/mcp/tools/test-history.d.ts +33 -0
  63. package/dist/src/mcp/tools/test-history.d.ts.map +1 -0
  64. package/dist/src/mcp/tools/test-history.js +66 -0
  65. package/dist/src/mcp/tools/test-history.js.map +1 -0
  66. package/dist/src/server/feedback-api.d.ts +4 -0
  67. package/dist/src/server/feedback-api.d.ts.map +1 -0
  68. package/dist/src/server/feedback-api.js +152 -0
  69. package/dist/src/server/feedback-api.js.map +1 -0
  70. package/dist/src/server/html-injector.d.ts +10 -0
  71. package/dist/src/server/html-injector.d.ts.map +1 -0
  72. package/dist/src/server/html-injector.js +34 -0
  73. package/dist/src/server/html-injector.js.map +1 -0
  74. package/dist/src/server/proxy-server.d.ts +8 -0
  75. package/dist/src/server/proxy-server.d.ts.map +1 -0
  76. package/dist/src/server/proxy-server.js +87 -0
  77. package/dist/src/server/proxy-server.js.map +1 -0
  78. package/dist/src/server/sdk-serve.d.ts +3 -0
  79. package/dist/src/server/sdk-serve.d.ts.map +1 -0
  80. package/dist/src/server/sdk-serve.js +20 -0
  81. package/dist/src/server/sdk-serve.js.map +1 -0
  82. package/dist/src/storage/store.d.ts +21 -0
  83. package/dist/src/storage/store.d.ts.map +1 -0
  84. package/dist/src/storage/store.js +138 -0
  85. package/dist/src/storage/store.js.map +1 -0
  86. package/dist/src/storage/types.d.ts +74 -0
  87. package/dist/src/storage/types.d.ts.map +1 -0
  88. package/dist/src/storage/types.js +2 -0
  89. package/dist/src/storage/types.js.map +1 -0
  90. package/dist/vibe-feedback.js +399 -0
  91. package/package.json +67 -0
  92. package/src/client/annotation-mode/canvas-editor.ts +178 -0
  93. package/src/client/annotation-mode/history.ts +32 -0
  94. package/src/client/annotation-mode/region-selector.ts +123 -0
  95. package/src/client/annotation-mode/screenshot.ts +17 -0
  96. package/src/client/annotation-mode/toolbar.ts +139 -0
  97. package/src/client/annotation-mode/tools/arrow.ts +57 -0
  98. package/src/client/annotation-mode/tools/base-tool.ts +25 -0
  99. package/src/client/annotation-mode/tools/circle.ts +37 -0
  100. package/src/client/annotation-mode/tools/freehand.ts +23 -0
  101. package/src/client/annotation-mode/tools/rect.ts +32 -0
  102. package/src/client/annotation-mode/tools/text.ts +93 -0
  103. package/src/client/api/client.ts +48 -0
  104. package/src/client/capture/action-recorder.ts +157 -0
  105. package/src/client/capture/console-interceptor.ts +65 -0
  106. package/src/client/capture/context-buffer.ts +23 -0
  107. package/src/client/capture/error-interceptor.ts +52 -0
  108. package/src/client/capture/network-interceptor.ts +143 -0
  109. package/src/client/core/event-bus.ts +20 -0
  110. package/src/client/core/i18n.ts +83 -0
  111. package/src/client/core/sdk.ts +373 -0
  112. package/src/client/core/shadow-host.ts +27 -0
  113. package/src/client/element-mode/highlighter.ts +79 -0
  114. package/src/client/element-mode/inspector.ts +73 -0
  115. package/src/client/element-mode/selector.ts +22 -0
  116. package/src/client/index.ts +10 -0
  117. package/src/client/styles/sdk.css.ts +222 -0
  118. package/src/client/ui/checklist-panel.ts +279 -0
  119. package/src/client/ui/feedback-panel.ts +149 -0
  120. package/src/client/ui/floating-button.ts +103 -0
  121. package/src/client/ui/toast.ts +17 -0
  122. package/src/client/ui/verify-panel.ts +111 -0
  123. package/src/detect/dev-server.ts +110 -0
  124. package/src/detect/spawn-dev.ts +50 -0
  125. package/src/install/detect-ai-tool.ts +49 -0
  126. package/src/install/write-mcp-config.ts +49 -0
  127. package/src/mcp/index.ts +327 -0
  128. package/src/mcp/tools/checklist.ts +61 -0
  129. package/src/mcp/tools/get-feedback.ts +34 -0
  130. package/src/mcp/tools/list-feedbacks.ts +37 -0
  131. package/src/mcp/tools/resolve-feedback.ts +34 -0
  132. package/src/mcp/tools/start-session.ts +65 -0
  133. package/src/mcp/tools/stop-session.ts +93 -0
  134. package/src/mcp/tools/test-history.ts +97 -0
  135. package/src/server/feedback-api.ts +164 -0
  136. package/src/server/html-injector.ts +41 -0
  137. package/src/server/proxy-server.ts +107 -0
  138. package/src/server/sdk-serve.ts +24 -0
  139. package/src/storage/store.ts +172 -0
  140. package/src/storage/types.ts +68 -0
@@ -0,0 +1,222 @@
1
+ export const SDK_CSS = `
2
+ * {
3
+ box-sizing: border-box;
4
+ margin: 0;
5
+ padding: 0;
6
+ }
7
+
8
+ .vf-fab {
9
+ position: fixed;
10
+ bottom: 24px;
11
+ right: 24px;
12
+ width: 52px;
13
+ height: 52px;
14
+ border-radius: 50%;
15
+ border: none;
16
+ background: #3b82f6;
17
+ color: white;
18
+ font-size: 22px;
19
+ cursor: pointer;
20
+ box-shadow: 0 4px 12px rgba(0,0,0,0.25);
21
+ display: flex;
22
+ align-items: center;
23
+ justify-content: center;
24
+ pointer-events: auto;
25
+ transition: background 0.2s, transform 0.2s;
26
+ z-index: 2147483647;
27
+ font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', sans-serif;
28
+ }
29
+ .vf-fab:hover {
30
+ transform: scale(1.08);
31
+ }
32
+ .vf-fab.active {
33
+ background: #f59e0b;
34
+ }
35
+
36
+ .vf-mode-bar {
37
+ position: fixed;
38
+ bottom: 84px;
39
+ right: 16px;
40
+ background: #1e293b;
41
+ border-radius: 12px;
42
+ padding: 6px;
43
+ display: flex;
44
+ gap: 4px;
45
+ pointer-events: auto;
46
+ box-shadow: 0 4px 16px rgba(0,0,0,0.3);
47
+ font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', sans-serif;
48
+ }
49
+ .vf-mode-btn {
50
+ padding: 8px 16px;
51
+ border: none;
52
+ border-radius: 8px;
53
+ background: transparent;
54
+ color: #94a3b8;
55
+ font-size: 13px;
56
+ cursor: pointer;
57
+ transition: all 0.2s;
58
+ }
59
+ .vf-mode-btn.active {
60
+ background: #3b82f6;
61
+ color: white;
62
+ }
63
+ .vf-mode-btn:hover:not(.active) {
64
+ color: white;
65
+ }
66
+
67
+ .vf-panel {
68
+ position: fixed;
69
+ bottom: 84px;
70
+ right: 16px;
71
+ width: 340px;
72
+ background: #1e293b;
73
+ border-radius: 12px;
74
+ padding: 20px;
75
+ pointer-events: auto;
76
+ box-shadow: 0 8px 32px rgba(0,0,0,0.4);
77
+ color: #e2e8f0;
78
+ font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', sans-serif;
79
+ font-size: 14px;
80
+ }
81
+ .vf-panel-title {
82
+ font-size: 15px;
83
+ font-weight: 600;
84
+ margin-bottom: 12px;
85
+ color: white;
86
+ }
87
+ .vf-panel-element {
88
+ background: #0f172a;
89
+ border-radius: 8px;
90
+ padding: 10px 12px;
91
+ margin-bottom: 12px;
92
+ font-family: 'SF Mono', 'Consolas', monospace;
93
+ font-size: 12px;
94
+ color: #93c5fd;
95
+ word-break: break-all;
96
+ }
97
+ .vf-panel-component {
98
+ color: #a78bfa;
99
+ margin-top: 4px;
100
+ font-size: 11px;
101
+ }
102
+ .vf-panel-label {
103
+ font-size: 12px;
104
+ color: #94a3b8;
105
+ margin-bottom: 6px;
106
+ }
107
+ .vf-type-group {
108
+ display: flex;
109
+ flex-wrap: wrap;
110
+ gap: 6px;
111
+ margin-bottom: 14px;
112
+ }
113
+ .vf-type-btn {
114
+ padding: 6px 12px;
115
+ border: 1px solid #334155;
116
+ border-radius: 6px;
117
+ background: transparent;
118
+ color: #94a3b8;
119
+ font-size: 12px;
120
+ cursor: pointer;
121
+ transition: all 0.15s;
122
+ }
123
+ .vf-type-btn.selected {
124
+ border-color: #3b82f6;
125
+ background: rgba(59,130,246,0.15);
126
+ color: #60a5fa;
127
+ }
128
+ .vf-textarea {
129
+ width: 100%;
130
+ height: 72px;
131
+ background: #0f172a;
132
+ border: 1px solid #334155;
133
+ border-radius: 8px;
134
+ padding: 10px 12px;
135
+ color: #e2e8f0;
136
+ font-size: 13px;
137
+ font-family: inherit;
138
+ resize: vertical;
139
+ margin-bottom: 14px;
140
+ }
141
+ .vf-textarea:focus {
142
+ outline: none;
143
+ border-color: #3b82f6;
144
+ }
145
+ .vf-btn-row {
146
+ display: flex;
147
+ gap: 8px;
148
+ justify-content: flex-end;
149
+ }
150
+ .vf-btn {
151
+ padding: 8px 18px;
152
+ border-radius: 8px;
153
+ border: none;
154
+ font-size: 13px;
155
+ cursor: pointer;
156
+ font-weight: 500;
157
+ transition: all 0.15s;
158
+ }
159
+ .vf-btn-cancel {
160
+ background: #334155;
161
+ color: #94a3b8;
162
+ }
163
+ .vf-btn-cancel:hover {
164
+ background: #475569;
165
+ }
166
+ .vf-btn-submit {
167
+ background: #3b82f6;
168
+ color: white;
169
+ }
170
+ .vf-btn-submit:hover {
171
+ background: #2563eb;
172
+ }
173
+ .vf-btn-submit:disabled {
174
+ opacity: 0.5;
175
+ cursor: not-allowed;
176
+ }
177
+ .vf-screenshot-btn {
178
+ width: 100%;
179
+ padding: 32px 16px;
180
+ border: 2px dashed #334155;
181
+ border-radius: 8px;
182
+ background: transparent;
183
+ color: #94a3b8;
184
+ font-size: 14px;
185
+ cursor: pointer;
186
+ margin-bottom: 14px;
187
+ transition: all 0.15s;
188
+ }
189
+ .vf-screenshot-btn:hover {
190
+ border-color: #3b82f6;
191
+ color: #60a5fa;
192
+ }
193
+ .vf-thumbnail {
194
+ width: 100%;
195
+ border-radius: 8px;
196
+ margin-bottom: 14px;
197
+ border: 1px solid #334155;
198
+ }
199
+
200
+ .vf-toast {
201
+ position: fixed;
202
+ top: 20px;
203
+ right: 20px;
204
+ background: #1e293b;
205
+ color: #e2e8f0;
206
+ padding: 12px 20px;
207
+ border-radius: 8px;
208
+ font-size: 13px;
209
+ pointer-events: auto;
210
+ box-shadow: 0 4px 16px rgba(0,0,0,0.3);
211
+ font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', sans-serif;
212
+ animation: vf-slide-in 0.3s ease-out;
213
+ border-left: 3px solid #22c55e;
214
+ }
215
+ .vf-toast.error {
216
+ border-left-color: #ef4444;
217
+ }
218
+ @keyframes vf-slide-in {
219
+ from { transform: translateX(100%); opacity: 0; }
220
+ to { transform: translateX(0); opacity: 1; }
221
+ }
222
+ `;
@@ -0,0 +1,279 @@
1
+ export class ChecklistPanel {
2
+ private container: HTMLDivElement;
3
+ private listEl: HTMLDivElement;
4
+ private headerEl: HTMLDivElement | null = null;
5
+ private pollTimer: ReturnType<typeof setInterval> | null = null;
6
+ private visible = false;
7
+ private collapsed = false;
8
+ private baseUrl: string;
9
+ private dragOffsetX = 0;
10
+ private dragOffsetY = 0;
11
+
12
+ private onItemFail: ((itemId: number) => void) | null;
13
+
14
+ constructor(private shadowRoot: ShadowRoot, onItemFail?: (itemId: number) => void) {
15
+ this.onItemFail = onItemFail || null;
16
+ this.baseUrl = window.location.origin;
17
+ this.container = document.createElement('div');
18
+ this.container.style.cssText = `
19
+ position: fixed; top: 16px; right: 80px; width: 300px;
20
+ max-height: 70vh; overflow: hidden;
21
+ background: #1e293b; border-radius: 10px;
22
+ box-shadow: 0 4px 20px rgba(0,0,0,0.35);
23
+ pointer-events: auto; display: none;
24
+ font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', sans-serif;
25
+ font-size: 13px; color: #e2e8f0; z-index: 2147483647;
26
+ transition: height 0.2s;
27
+ `;
28
+
29
+ this.listEl = document.createElement('div');
30
+ this.listEl.style.cssText = `max-height: calc(70vh - 44px); overflow-y: auto;`;
31
+ this.container.appendChild(this.listEl);
32
+ shadowRoot.appendChild(this.container);
33
+ }
34
+
35
+ startPolling(): void {
36
+ this.poll();
37
+ this.pollTimer = setInterval(() => this.poll(), 3000);
38
+ }
39
+
40
+ stopPolling(): void {
41
+ if (this.pollTimer) {
42
+ clearInterval(this.pollTimer);
43
+ this.pollTimer = null;
44
+ }
45
+ }
46
+
47
+ private async poll(): Promise<void> {
48
+ try {
49
+ const res = await fetch(`${this.baseUrl}/api/checklist`);
50
+ const data = await res.json();
51
+ if (data.items && data.items.length > 0) {
52
+ this.render(data);
53
+ if (!this.visible) {
54
+ this.container.style.display = 'block';
55
+ this.visible = true;
56
+ }
57
+ } else if (this.visible) {
58
+ this.container.style.display = 'none';
59
+ this.visible = false;
60
+ }
61
+ } catch {}
62
+ }
63
+
64
+ private render(data: { title: string; items: any[] }): void {
65
+ const passed = data.items.filter((i) => i.status === 'passed').length;
66
+ const failed = data.items.filter((i) => i.status === 'failed').length;
67
+ const total = data.items.length;
68
+
69
+ this.listEl.innerHTML = '';
70
+
71
+ // Header — draggable + collapsible
72
+ const header = document.createElement('div');
73
+ header.style.cssText = `
74
+ padding: 10px 14px; border-bottom: 1px solid #334155;
75
+ display: flex; justify-content: space-between; align-items: center;
76
+ cursor: move; user-select: none;
77
+ `;
78
+
79
+ const titleWrap = document.createElement('div');
80
+ titleWrap.style.cssText = `display:flex;align-items:center;gap:8px;`;
81
+
82
+ const title = document.createElement('span');
83
+ title.style.cssText = `font-weight:600;color:white;font-size:13px;`;
84
+ title.textContent = data.title;
85
+
86
+ const badge = document.createElement('span');
87
+ badge.style.cssText = `font-size:11px;color:#94a3b8;`;
88
+ badge.textContent = `${passed}/${total}`;
89
+ if (failed > 0) {
90
+ const failBadge = document.createElement('span');
91
+ failBadge.style.cssText = `color:#ef4444;margin-left:4px;`;
92
+ failBadge.textContent = `${failed} fail`;
93
+ badge.appendChild(failBadge);
94
+ }
95
+
96
+ titleWrap.appendChild(title);
97
+ titleWrap.appendChild(badge);
98
+ header.appendChild(titleWrap);
99
+
100
+ // Collapse toggle
101
+ const toggleBtn = document.createElement('button');
102
+ toggleBtn.style.cssText = `
103
+ background:none; border:none; color:#94a3b8; cursor:pointer;
104
+ font-size:16px; padding:0 4px; line-height:1;
105
+ `;
106
+ toggleBtn.textContent = this.collapsed ? '\u25BC' : '\u25B2';
107
+ toggleBtn.addEventListener('click', (e) => {
108
+ e.stopPropagation();
109
+ this.collapsed = !this.collapsed;
110
+ this.listEl.querySelector('.vf-checklist-items')?.toggleAttribute('hidden', this.collapsed);
111
+ toggleBtn.textContent = this.collapsed ? '\u25BC' : '\u25B2';
112
+ });
113
+ header.appendChild(toggleBtn);
114
+
115
+ // Drag support
116
+ header.addEventListener('mousedown', (e) => {
117
+ if ((e.target as HTMLElement).tagName === 'BUTTON') return;
118
+ e.preventDefault();
119
+ const rect = this.container.getBoundingClientRect();
120
+ this.dragOffsetX = e.clientX - rect.left;
121
+ this.dragOffsetY = e.clientY - rect.top;
122
+
123
+ const onMove = (ev: MouseEvent) => {
124
+ this.container.style.left = `${ev.clientX - this.dragOffsetX}px`;
125
+ this.container.style.top = `${ev.clientY - this.dragOffsetY}px`;
126
+ this.container.style.right = 'auto';
127
+ };
128
+ const onUp = () => {
129
+ document.removeEventListener('mousemove', onMove);
130
+ document.removeEventListener('mouseup', onUp);
131
+ };
132
+ document.addEventListener('mousemove', onMove);
133
+ document.addEventListener('mouseup', onUp);
134
+ });
135
+
136
+ this.listEl.appendChild(header);
137
+ this.headerEl = header;
138
+
139
+ // Progress bar
140
+ const progressWrap = document.createElement('div');
141
+ progressWrap.style.cssText = `height:2px;background:#334155;`;
142
+ const pct = total > 0 ? (passed / total) * 100 : 0;
143
+ const progressBar = document.createElement('div');
144
+ progressBar.style.cssText = `height:100%;width:${pct}%;background:#22c55e;transition:width 0.3s;`;
145
+ progressWrap.appendChild(progressBar);
146
+ this.listEl.appendChild(progressWrap);
147
+
148
+ // Items container (collapsible)
149
+ const itemsWrap = document.createElement('div');
150
+ itemsWrap.className = 'vf-checklist-items';
151
+ if (this.collapsed) itemsWrap.hidden = true;
152
+
153
+ for (const item of data.items) {
154
+ const row = document.createElement('div');
155
+ row.style.cssText = `
156
+ padding: 8px 14px; border-bottom: 1px solid #0f172a;
157
+ display: flex; align-items: flex-start; gap: 8px;
158
+ `;
159
+
160
+ // Status buttons
161
+ const controls = document.createElement('div');
162
+ controls.style.cssText = `display:flex;gap:3px;flex-shrink:0;margin-top:2px;`;
163
+
164
+ controls.appendChild(this.createStatusBtn('\u2713', '#22c55e', item.status === 'passed', () => {
165
+ this.updateItem(item.id, 'passed');
166
+ }));
167
+ controls.appendChild(this.createStatusBtn('\u2717', '#ef4444', item.status === 'failed', () => {
168
+ this.promptFeedback(item.id);
169
+ }));
170
+ row.appendChild(controls);
171
+
172
+ // Content
173
+ const text = document.createElement('div');
174
+ text.style.cssText = `flex:1;line-height:1.4;font-size:12px;`;
175
+
176
+ // Badges row
177
+ const badges = document.createElement('div');
178
+ badges.style.cssText = `display:flex;gap:4px;margin-bottom:2px;flex-wrap:wrap;`;
179
+
180
+ if (item.priority === 'critical') {
181
+ const badge = document.createElement('span');
182
+ badge.style.cssText = `
183
+ display:inline-block;font-size:9px;padding:1px 5px;border-radius:3px;
184
+ background:rgba(239,68,68,0.15);color:#ef4444;font-weight:600;
185
+ `;
186
+ badge.textContent = 'CRITICAL';
187
+ badges.appendChild(badge);
188
+ }
189
+
190
+ if (item.dimension) {
191
+ const dimBadge = document.createElement('span');
192
+ dimBadge.style.cssText = `
193
+ display:inline-block;font-size:9px;padding:1px 5px;border-radius:3px;
194
+ background:rgba(59,130,246,0.12);color:#60a5fa;
195
+ `;
196
+ dimBadge.textContent = item.dimension;
197
+ badges.appendChild(dimBadge);
198
+ }
199
+
200
+ if (badges.childNodes.length > 0) {
201
+ text.appendChild(badges);
202
+ }
203
+
204
+ // Step (what to do)
205
+ const stepText = document.createElement('span');
206
+ stepText.textContent = item.step || item.text || '';
207
+ text.appendChild(stepText);
208
+
209
+ // Expected result
210
+ if (item.expect) {
211
+ const expectEl = document.createElement('div');
212
+ expectEl.style.cssText = `font-size:11px;color:#94a3b8;margin-top:2px;`;
213
+ expectEl.textContent = `\u2192 ${item.expect}`;
214
+ text.appendChild(expectEl);
215
+ }
216
+
217
+ if (item.status === 'passed') {
218
+ stepText.style.color = '#22c55e';
219
+ stepText.style.textDecoration = 'line-through';
220
+ stepText.style.opacity = '0.6';
221
+ } else if (item.status === 'failed') {
222
+ stepText.style.color = '#ef4444';
223
+ }
224
+
225
+ if (item.feedback) {
226
+ const fb = document.createElement('div');
227
+ fb.style.cssText = `font-size:11px;color:#f59e0b;margin-top:3px;`;
228
+ fb.textContent = item.feedback;
229
+ text.appendChild(fb);
230
+ }
231
+
232
+ row.appendChild(text);
233
+ itemsWrap.appendChild(row);
234
+ }
235
+
236
+ this.listEl.appendChild(itemsWrap);
237
+ }
238
+
239
+ private createStatusBtn(icon: string, color: string, active: boolean, onClick: () => void): HTMLButtonElement {
240
+ const btn = document.createElement('button');
241
+ btn.textContent = icon;
242
+ btn.style.cssText = `
243
+ width: 22px; height: 22px; border-radius: 4px; border: 1px solid ${color};
244
+ background: ${active ? color : 'transparent'};
245
+ color: ${active ? 'white' : color};
246
+ font-size: 12px; cursor: pointer; display: flex;
247
+ align-items: center; justify-content: center; padding: 0;
248
+ `;
249
+ btn.addEventListener('click', onClick);
250
+ return btn;
251
+ }
252
+
253
+ private promptFeedback(itemId: number): void {
254
+ if (this.onItemFail) {
255
+ // Enter feedback mode — SDK will handle element selection/annotation
256
+ this.onItemFail(itemId);
257
+ } else {
258
+ // Fallback: simple prompt
259
+ const feedback = prompt('Describe the issue (optional):') || '';
260
+ this.updateItem(itemId, 'failed', feedback);
261
+ }
262
+ }
263
+
264
+ private async updateItem(itemId: number, status: string, feedback?: string): Promise<void> {
265
+ try {
266
+ await fetch(`${this.baseUrl}/api/checklist/${itemId}`, {
267
+ method: 'PATCH',
268
+ headers: { 'Content-Type': 'application/json' },
269
+ body: JSON.stringify({ status, feedback }),
270
+ });
271
+ this.poll();
272
+ } catch {}
273
+ }
274
+
275
+ destroy(): void {
276
+ this.stopPolling();
277
+ this.container.remove();
278
+ }
279
+ }
@@ -0,0 +1,149 @@
1
+ import type { ElementInfo } from '../element-mode/inspector.js';
2
+ import { t } from '../core/i18n.js';
3
+
4
+ const PROBLEM_TYPES = [
5
+ { value: 'bug', labelKey: 'type.bug' },
6
+ { value: 'ui-issue', labelKey: 'type.ui-issue' },
7
+ { value: 'performance', labelKey: 'type.performance' },
8
+ { value: 'feature-request', labelKey: 'type.feature-request' },
9
+ { value: 'other', labelKey: 'type.other' },
10
+ ];
11
+
12
+ export interface PanelResult {
13
+ problemType: string;
14
+ description: string;
15
+ }
16
+
17
+ export class FeedbackPanel {
18
+ private container: HTMLDivElement;
19
+ private typeButtons: HTMLButtonElement[] = [];
20
+ private textarea!: HTMLTextAreaElement;
21
+ private submitBtn!: HTMLButtonElement;
22
+ private selectedType = '';
23
+ private thumbnailImg: HTMLImageElement | null = null;
24
+
25
+ constructor(
26
+ private shadowRoot: ShadowRoot,
27
+ private onSubmit: (result: PanelResult) => void,
28
+ private onCancel: () => void
29
+ ) {
30
+ this.container = document.createElement('div');
31
+ this.container.className = 'vf-panel';
32
+ this.container.style.display = 'none';
33
+ }
34
+
35
+ showForElement(elementInfo: ElementInfo): void {
36
+ this.render('element', elementInfo);
37
+ }
38
+
39
+ showForAnnotation(thumbnailDataUrl: string): void {
40
+ this.render('annotation', undefined, thumbnailDataUrl);
41
+ }
42
+
43
+ hide(): void {
44
+ this.container.style.display = 'none';
45
+ this.container.remove();
46
+ this.selectedType = '';
47
+ }
48
+
49
+ private render(mode: string, elementInfo?: ElementInfo, thumbnailDataUrl?: string): void {
50
+ this.container.innerHTML = '';
51
+
52
+ // Title
53
+ const title = document.createElement('div');
54
+ title.className = 'vf-panel-title';
55
+ title.textContent = mode === 'element' ? t('panel.element.title') : t('panel.annotation.title');
56
+ this.container.appendChild(title);
57
+
58
+ // Element info
59
+ if (elementInfo) {
60
+ const elDiv = document.createElement('div');
61
+ elDiv.className = 'vf-panel-element';
62
+ elDiv.textContent = `<${elementInfo.tagName}> ${elementInfo.selector}`;
63
+ if (elementInfo.componentName) {
64
+ const comp = document.createElement('div');
65
+ comp.className = 'vf-panel-component';
66
+ comp.textContent = `Component: ${elementInfo.componentName}`;
67
+ elDiv.appendChild(comp);
68
+ }
69
+ this.container.appendChild(elDiv);
70
+ }
71
+
72
+ // Annotation thumbnail
73
+ if (thumbnailDataUrl) {
74
+ this.thumbnailImg = document.createElement('img');
75
+ this.thumbnailImg.className = 'vf-thumbnail';
76
+ this.thumbnailImg.src = thumbnailDataUrl;
77
+ this.container.appendChild(this.thumbnailImg);
78
+ }
79
+
80
+ // Problem type
81
+ const typeLabel = document.createElement('div');
82
+ typeLabel.className = 'vf-panel-label';
83
+ typeLabel.textContent = t('panel.type');
84
+ this.container.appendChild(typeLabel);
85
+
86
+ const typeGroup = document.createElement('div');
87
+ typeGroup.className = 'vf-type-group';
88
+ this.typeButtons = [];
89
+ for (const pt of PROBLEM_TYPES) {
90
+ const btn = document.createElement('button');
91
+ btn.className = 'vf-type-btn';
92
+ btn.textContent = t(pt.labelKey);
93
+ btn.addEventListener('click', () => {
94
+ this.selectedType = pt.value;
95
+ this.typeButtons.forEach((b) => b.classList.remove('selected'));
96
+ btn.classList.add('selected');
97
+ this.updateSubmitState();
98
+ });
99
+ typeGroup.appendChild(btn);
100
+ this.typeButtons.push(btn);
101
+ }
102
+ this.container.appendChild(typeGroup);
103
+
104
+ // Description
105
+ const descLabel = document.createElement('div');
106
+ descLabel.className = 'vf-panel-label';
107
+ descLabel.textContent = t('panel.description');
108
+ this.container.appendChild(descLabel);
109
+
110
+ this.textarea = document.createElement('textarea');
111
+ this.textarea.className = 'vf-textarea';
112
+ this.textarea.placeholder = t('panel.description.placeholder');
113
+ this.textarea.addEventListener('input', () => this.updateSubmitState());
114
+ this.container.appendChild(this.textarea);
115
+
116
+ // Buttons
117
+ const btnRow = document.createElement('div');
118
+ btnRow.className = 'vf-btn-row';
119
+
120
+ const cancelBtn = document.createElement('button');
121
+ cancelBtn.className = 'vf-btn vf-btn-cancel';
122
+ cancelBtn.textContent = t('panel.cancel');
123
+ cancelBtn.addEventListener('click', () => this.onCancel());
124
+ btnRow.appendChild(cancelBtn);
125
+
126
+ this.submitBtn = document.createElement('button');
127
+ this.submitBtn.className = 'vf-btn vf-btn-submit';
128
+ this.submitBtn.textContent = t('panel.submit');
129
+ this.submitBtn.disabled = true;
130
+ this.submitBtn.addEventListener('click', () => {
131
+ this.onSubmit({
132
+ problemType: this.selectedType,
133
+ description: this.textarea.value.trim(),
134
+ });
135
+ });
136
+ btnRow.appendChild(this.submitBtn);
137
+
138
+ this.container.appendChild(btnRow);
139
+
140
+ // Show
141
+ this.shadowRoot.appendChild(this.container);
142
+ this.container.style.display = 'block';
143
+ this.textarea.focus();
144
+ }
145
+
146
+ private updateSubmitState(): void {
147
+ this.submitBtn.disabled = !this.selectedType;
148
+ }
149
+ }