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.
- package/LICENSE +21 -0
- package/README.md +160 -0
- package/bin/cli.ts +17 -0
- package/bin/install.ts +34 -0
- package/bin/mcp.ts +2 -0
- package/dist/bin/cli.d.ts +3 -0
- package/dist/bin/cli.d.ts.map +1 -0
- package/dist/bin/cli.js +19 -0
- package/dist/bin/cli.js.map +1 -0
- package/dist/bin/install.d.ts +3 -0
- package/dist/bin/install.d.ts.map +1 -0
- package/dist/bin/install.js +28 -0
- package/dist/bin/install.js.map +1 -0
- package/dist/bin/mcp.d.ts +3 -0
- package/dist/bin/mcp.d.ts.map +1 -0
- package/dist/bin/mcp.js +3 -0
- package/dist/bin/mcp.js.map +1 -0
- package/dist/src/detect/dev-server.d.ts +11 -0
- package/dist/src/detect/dev-server.d.ts.map +1 -0
- package/dist/src/detect/dev-server.js +92 -0
- package/dist/src/detect/dev-server.js.map +1 -0
- package/dist/src/detect/spawn-dev.d.ts +4 -0
- package/dist/src/detect/spawn-dev.d.ts.map +1 -0
- package/dist/src/detect/spawn-dev.js +37 -0
- package/dist/src/detect/spawn-dev.js.map +1 -0
- package/dist/src/install/detect-ai-tool.d.ts +7 -0
- package/dist/src/install/detect-ai-tool.d.ts.map +1 -0
- package/dist/src/install/detect-ai-tool.js +38 -0
- package/dist/src/install/detect-ai-tool.js.map +1 -0
- package/dist/src/install/write-mcp-config.d.ts +6 -0
- package/dist/src/install/write-mcp-config.d.ts.map +1 -0
- package/dist/src/install/write-mcp-config.js +47 -0
- package/dist/src/install/write-mcp-config.js.map +1 -0
- package/dist/src/mcp/index.d.ts +2 -0
- package/dist/src/mcp/index.d.ts.map +1 -0
- package/dist/src/mcp/index.js +263 -0
- package/dist/src/mcp/index.js.map +1 -0
- package/dist/src/mcp/tools/checklist.d.ts +22 -0
- package/dist/src/mcp/tools/checklist.d.ts.map +1 -0
- package/dist/src/mcp/tools/checklist.js +58 -0
- package/dist/src/mcp/tools/checklist.js.map +1 -0
- package/dist/src/mcp/tools/get-feedback.d.ts +13 -0
- package/dist/src/mcp/tools/get-feedback.d.ts.map +1 -0
- package/dist/src/mcp/tools/get-feedback.js +27 -0
- package/dist/src/mcp/tools/get-feedback.js.map +1 -0
- package/dist/src/mcp/tools/list-feedbacks.d.ts +20 -0
- package/dist/src/mcp/tools/list-feedbacks.d.ts.map +1 -0
- package/dist/src/mcp/tools/list-feedbacks.js +29 -0
- package/dist/src/mcp/tools/list-feedbacks.js.map +1 -0
- package/dist/src/mcp/tools/resolve-feedback.d.ts +8 -0
- package/dist/src/mcp/tools/resolve-feedback.d.ts.map +1 -0
- package/dist/src/mcp/tools/resolve-feedback.js +28 -0
- package/dist/src/mcp/tools/resolve-feedback.js.map +1 -0
- package/dist/src/mcp/tools/start-session.d.ts +11 -0
- package/dist/src/mcp/tools/start-session.d.ts.map +1 -0
- package/dist/src/mcp/tools/start-session.js +56 -0
- package/dist/src/mcp/tools/start-session.js.map +1 -0
- package/dist/src/mcp/tools/stop-session.d.ts +6 -0
- package/dist/src/mcp/tools/stop-session.d.ts.map +1 -0
- package/dist/src/mcp/tools/stop-session.js +80 -0
- package/dist/src/mcp/tools/stop-session.js.map +1 -0
- package/dist/src/mcp/tools/test-history.d.ts +33 -0
- package/dist/src/mcp/tools/test-history.d.ts.map +1 -0
- package/dist/src/mcp/tools/test-history.js +66 -0
- package/dist/src/mcp/tools/test-history.js.map +1 -0
- package/dist/src/server/feedback-api.d.ts +4 -0
- package/dist/src/server/feedback-api.d.ts.map +1 -0
- package/dist/src/server/feedback-api.js +152 -0
- package/dist/src/server/feedback-api.js.map +1 -0
- package/dist/src/server/html-injector.d.ts +10 -0
- package/dist/src/server/html-injector.d.ts.map +1 -0
- package/dist/src/server/html-injector.js +34 -0
- package/dist/src/server/html-injector.js.map +1 -0
- package/dist/src/server/proxy-server.d.ts +8 -0
- package/dist/src/server/proxy-server.d.ts.map +1 -0
- package/dist/src/server/proxy-server.js +87 -0
- package/dist/src/server/proxy-server.js.map +1 -0
- package/dist/src/server/sdk-serve.d.ts +3 -0
- package/dist/src/server/sdk-serve.d.ts.map +1 -0
- package/dist/src/server/sdk-serve.js +20 -0
- package/dist/src/server/sdk-serve.js.map +1 -0
- package/dist/src/storage/store.d.ts +21 -0
- package/dist/src/storage/store.d.ts.map +1 -0
- package/dist/src/storage/store.js +138 -0
- package/dist/src/storage/store.js.map +1 -0
- package/dist/src/storage/types.d.ts +74 -0
- package/dist/src/storage/types.d.ts.map +1 -0
- package/dist/src/storage/types.js +2 -0
- package/dist/src/storage/types.js.map +1 -0
- package/dist/vibe-feedback.js +399 -0
- package/package.json +67 -0
- package/src/client/annotation-mode/canvas-editor.ts +178 -0
- package/src/client/annotation-mode/history.ts +32 -0
- package/src/client/annotation-mode/region-selector.ts +123 -0
- package/src/client/annotation-mode/screenshot.ts +17 -0
- package/src/client/annotation-mode/toolbar.ts +139 -0
- package/src/client/annotation-mode/tools/arrow.ts +57 -0
- package/src/client/annotation-mode/tools/base-tool.ts +25 -0
- package/src/client/annotation-mode/tools/circle.ts +37 -0
- package/src/client/annotation-mode/tools/freehand.ts +23 -0
- package/src/client/annotation-mode/tools/rect.ts +32 -0
- package/src/client/annotation-mode/tools/text.ts +93 -0
- package/src/client/api/client.ts +48 -0
- package/src/client/capture/action-recorder.ts +157 -0
- package/src/client/capture/console-interceptor.ts +65 -0
- package/src/client/capture/context-buffer.ts +23 -0
- package/src/client/capture/error-interceptor.ts +52 -0
- package/src/client/capture/network-interceptor.ts +143 -0
- package/src/client/core/event-bus.ts +20 -0
- package/src/client/core/i18n.ts +83 -0
- package/src/client/core/sdk.ts +373 -0
- package/src/client/core/shadow-host.ts +27 -0
- package/src/client/element-mode/highlighter.ts +79 -0
- package/src/client/element-mode/inspector.ts +73 -0
- package/src/client/element-mode/selector.ts +22 -0
- package/src/client/index.ts +10 -0
- package/src/client/styles/sdk.css.ts +222 -0
- package/src/client/ui/checklist-panel.ts +279 -0
- package/src/client/ui/feedback-panel.ts +149 -0
- package/src/client/ui/floating-button.ts +103 -0
- package/src/client/ui/toast.ts +17 -0
- package/src/client/ui/verify-panel.ts +111 -0
- package/src/detect/dev-server.ts +110 -0
- package/src/detect/spawn-dev.ts +50 -0
- package/src/install/detect-ai-tool.ts +49 -0
- package/src/install/write-mcp-config.ts +49 -0
- package/src/mcp/index.ts +327 -0
- package/src/mcp/tools/checklist.ts +61 -0
- package/src/mcp/tools/get-feedback.ts +34 -0
- package/src/mcp/tools/list-feedbacks.ts +37 -0
- package/src/mcp/tools/resolve-feedback.ts +34 -0
- package/src/mcp/tools/start-session.ts +65 -0
- package/src/mcp/tools/stop-session.ts +93 -0
- package/src/mcp/tools/test-history.ts +97 -0
- package/src/server/feedback-api.ts +164 -0
- package/src/server/html-injector.ts +41 -0
- package/src/server/proxy-server.ts +107 -0
- package/src/server/sdk-serve.ts +24 -0
- package/src/storage/store.ts +172 -0
- 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
|
+
}
|