vite-plugin-opencode-assistant 1.0.15 → 1.0.16
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/es/client/App.vue.d.ts +6 -0
- package/es/client/App.vue.js +300 -0
- package/es/client/components/ChromeWarmupError-sfc.css +1 -0
- package/es/client/components/ChromeWarmupError.vue.d.ts +11 -0
- package/es/client/components/ChromeWarmupError.vue.js +196 -0
- package/es/client/components/LoadingContent.vue.d.ts +5 -0
- package/es/client/components/LoadingContent.vue.js +39 -0
- package/es/client/composables/useContext.d.ts +8 -0
- package/es/client/composables/useContext.js +63 -0
- package/es/client/composables/useHotkey.d.ts +12 -0
- package/es/client/composables/useHotkey.js +41 -0
- package/es/client/composables/useSSE.d.ts +20 -0
- package/es/client/composables/useSSE.js +61 -0
- package/es/client/composables/useSelectedElements.d.ts +19 -0
- package/es/client/composables/useSelectedElements.js +43 -0
- package/es/client/composables/useServiceStatus.d.ts +13 -0
- package/es/client/composables/useServiceStatus.js +53 -0
- package/es/client/composables/useSessions.d.ts +26 -0
- package/es/client/composables/useSessions.js +127 -0
- package/es/client/composables/useTheme.d.ts +12 -0
- package/es/client/composables/useTheme.js +42 -0
- package/es/client/index.d.ts +1 -1
- package/es/client/index.js +5 -675
- package/es/client/styles.css +1 -0
- package/es/core/api.d.ts +18 -6
- package/es/core/api.js +324 -69
- package/es/core/proxy-server.js +127 -2
- package/es/core/service.d.ts +9 -2
- package/es/core/service.js +35 -31
- package/es/endpoints/index.js +1 -1
- package/es/endpoints/sse.js +0 -3
- package/es/endpoints/start.d.ts +1 -2
- package/es/endpoints/start.js +2 -2
- package/es/endpoints/types.d.ts +5 -2
- package/es/endpoints/warmup.js +15 -3
- package/es/index.js +8 -12
- package/es/utils/system.d.ts +1 -0
- package/es/utils/system.js +28 -0
- package/lib/client/App.vue.d.ts +6 -0
- package/lib/client/App.vue.js +329 -0
- package/lib/client/components/ChromeWarmupError-sfc.css +1 -0
- package/lib/client/components/ChromeWarmupError.vue.d.ts +11 -0
- package/lib/client/components/ChromeWarmupError.vue.js +215 -0
- package/lib/client/components/LoadingContent.vue.d.ts +5 -0
- package/lib/client/components/LoadingContent.vue.js +58 -0
- package/lib/client/composables/useContext.d.ts +8 -0
- package/lib/client/composables/useContext.js +86 -0
- package/lib/client/composables/useHotkey.d.ts +12 -0
- package/lib/client/composables/useHotkey.js +66 -0
- package/lib/client/composables/useSSE.d.ts +20 -0
- package/lib/client/composables/useSSE.js +84 -0
- package/lib/client/composables/useSelectedElements.d.ts +19 -0
- package/lib/client/composables/useSelectedElements.js +66 -0
- package/lib/client/composables/useServiceStatus.d.ts +13 -0
- package/lib/client/composables/useServiceStatus.js +76 -0
- package/lib/client/composables/useSessions.d.ts +26 -0
- package/lib/client/composables/useSessions.js +148 -0
- package/lib/client/composables/useTheme.d.ts +12 -0
- package/lib/client/composables/useTheme.js +65 -0
- package/lib/client/index.d.ts +1 -1
- package/lib/client/index.js +22 -667
- package/lib/client/styles.css +1 -0
- package/lib/client.js +2988 -2973
- package/lib/core/api.d.ts +18 -6
- package/lib/core/api.js +321 -74
- package/lib/core/proxy-server.js +127 -2
- package/lib/core/service.d.ts +9 -2
- package/lib/core/service.js +31 -30
- package/lib/endpoints/index.js +1 -1
- package/lib/endpoints/sse.js +0 -3
- package/lib/endpoints/start.d.ts +1 -2
- package/lib/endpoints/start.js +2 -2
- package/lib/endpoints/types.d.ts +5 -2
- package/lib/endpoints/warmup.js +15 -3
- package/lib/index.js +8 -12
- package/lib/style.css +1 -1
- package/lib/utils/system.d.ts +1 -0
- package/lib/utils/system.js +29 -0
- package/package.json +4 -4
package/lib/core/proxy-server.js
CHANGED
|
@@ -130,11 +130,127 @@ function generateBridgeScript(options) {
|
|
|
130
130
|
}
|
|
131
131
|
});
|
|
132
132
|
|
|
133
|
+
// === \u601D\u8003\u72B6\u6001\u76D1\u542C (\u5B8C\u5168\u590D\u523B OpenCode Web \u5B9E\u73B0) ===
|
|
134
|
+
// OpenCode Web \u6838\u5FC3\u903B\u8F91:
|
|
135
|
+
// working = !!pending() || sessionStatus().type !== "idle"
|
|
136
|
+
// pending = \u6700\u540E\u4E00\u6761\u672A\u5B8C\u6210\u7684 assistant \u6D88\u606F (time.completed \u4E0D\u662F\u6570\u5B57)
|
|
137
|
+
// sessionStatus = sync.data.session_status[sessionID]
|
|
138
|
+
|
|
139
|
+
let eventSource = null;
|
|
140
|
+
const sessionStatus = {};
|
|
141
|
+
const pendingMessages = {};
|
|
142
|
+
|
|
143
|
+
function getCurrentSessionID() {
|
|
144
|
+
const match = window.location.pathname.match(/\\/session\\/([^\\/]+)/);
|
|
145
|
+
return match ? match[1] : null;
|
|
146
|
+
}
|
|
147
|
+
|
|
148
|
+
function isPending(message) {
|
|
149
|
+
return message.role === 'assistant' && typeof message.time?.completed !== 'number';
|
|
150
|
+
}
|
|
151
|
+
|
|
152
|
+
function updateThinkingState(sessionID) {
|
|
153
|
+
const status = sessionStatus[sessionID];
|
|
154
|
+
const pending = pendingMessages[sessionID];
|
|
155
|
+
|
|
156
|
+
const isThinking = !!pending || (status && status.type !== 'idle');
|
|
157
|
+
|
|
158
|
+
if (window.parent !== window) {
|
|
159
|
+
window.parent.postMessage({
|
|
160
|
+
type: 'OPENCODE_THINKING_STATE',
|
|
161
|
+
thinking: isThinking,
|
|
162
|
+
sessionID: sessionID,
|
|
163
|
+
statusType: status?.type || 'idle',
|
|
164
|
+
hasPending: !!pending
|
|
165
|
+
}, '*');
|
|
166
|
+
}
|
|
167
|
+
}
|
|
168
|
+
|
|
169
|
+
function handleEvent(payload) {
|
|
170
|
+
const type = payload.type;
|
|
171
|
+
const props = payload.properties;
|
|
172
|
+
|
|
173
|
+
switch (type) {
|
|
174
|
+
case 'session.status': {
|
|
175
|
+
const sessionID = props.sessionID;
|
|
176
|
+
sessionStatus[sessionID] = props.status;
|
|
177
|
+
updateThinkingState(sessionID);
|
|
178
|
+
break;
|
|
179
|
+
}
|
|
180
|
+
|
|
181
|
+
case 'message.updated': {
|
|
182
|
+
const info = props.info;
|
|
183
|
+
if (!info || !info.sessionID) break;
|
|
184
|
+
const sessionID = info.sessionID;
|
|
185
|
+
|
|
186
|
+
if (info.role === 'assistant') {
|
|
187
|
+
if (isPending(info)) {
|
|
188
|
+
pendingMessages[sessionID] = info;
|
|
189
|
+
} else {
|
|
190
|
+
delete pendingMessages[sessionID];
|
|
191
|
+
}
|
|
192
|
+
updateThinkingState(sessionID);
|
|
193
|
+
}
|
|
194
|
+
break;
|
|
195
|
+
}
|
|
196
|
+
|
|
197
|
+
case 'message.part.delta': {
|
|
198
|
+
const sessionID = props.sessionID;
|
|
199
|
+
if (sessionID && !pendingMessages[sessionID]) {
|
|
200
|
+
pendingMessages[sessionID] = { role: 'assistant', time: {} };
|
|
201
|
+
updateThinkingState(sessionID);
|
|
202
|
+
}
|
|
203
|
+
break;
|
|
204
|
+
}
|
|
205
|
+
}
|
|
206
|
+
}
|
|
207
|
+
|
|
208
|
+
function setupThinkingListener() {
|
|
209
|
+
if (eventSource) {
|
|
210
|
+
eventSource.close();
|
|
211
|
+
}
|
|
212
|
+
|
|
213
|
+
eventSource = new EventSource('/global/event');
|
|
214
|
+
|
|
215
|
+
eventSource.onmessage = function(event) {
|
|
216
|
+
try {
|
|
217
|
+
const data = JSON.parse(event.data);
|
|
218
|
+
const payload = data.payload;
|
|
219
|
+
if (!payload) return;
|
|
220
|
+
handleEvent(payload);
|
|
221
|
+
} catch (e) {
|
|
222
|
+
// ignore parse errors
|
|
223
|
+
}
|
|
224
|
+
};
|
|
225
|
+
|
|
226
|
+
eventSource.onerror = function(err) {
|
|
227
|
+
console.warn('[OpenCode Bridge] SSE connection error, retrying in 3s...');
|
|
228
|
+
eventSource.close();
|
|
229
|
+
setTimeout(setupThinkingListener, 3000);
|
|
230
|
+
};
|
|
231
|
+
|
|
232
|
+
console.log('[OpenCode Bridge] SSE listener setup complete');
|
|
233
|
+
}
|
|
234
|
+
|
|
133
235
|
// === \u5C31\u7EEA\u901A\u77E5 ===
|
|
134
|
-
|
|
236
|
+
function init() {
|
|
135
237
|
if (window.parent !== window) {
|
|
136
238
|
window.parent.postMessage({ type: "OPENCODE_READY" }, "*");
|
|
137
239
|
}
|
|
240
|
+
setupThinkingListener();
|
|
241
|
+
}
|
|
242
|
+
|
|
243
|
+
if (document.readyState === 'loading') {
|
|
244
|
+
document.addEventListener('DOMContentLoaded', init);
|
|
245
|
+
} else {
|
|
246
|
+
init();
|
|
247
|
+
}
|
|
248
|
+
|
|
249
|
+
window.addEventListener('beforeunload', function() {
|
|
250
|
+
if (eventSource) {
|
|
251
|
+
eventSource.close();
|
|
252
|
+
eventSource = null;
|
|
253
|
+
}
|
|
138
254
|
});
|
|
139
255
|
})();
|
|
140
256
|
`;
|
|
@@ -162,7 +278,8 @@ function startProxyServer(targetUrl, port, options = {}) {
|
|
|
162
278
|
headers: __spreadProps(__spreadValues({}, req.headers), {
|
|
163
279
|
host: target.host,
|
|
164
280
|
"accept-encoding": "identity"
|
|
165
|
-
})
|
|
281
|
+
}),
|
|
282
|
+
timeout: 0
|
|
166
283
|
};
|
|
167
284
|
const proxyReq = import_http.default.request(requestOptions, (proxyRes) => {
|
|
168
285
|
var _a;
|
|
@@ -208,11 +325,19 @@ function startProxyServer(targetUrl, port, options = {}) {
|
|
|
208
325
|
res.writeHead(502);
|
|
209
326
|
res.end("Proxy error");
|
|
210
327
|
});
|
|
328
|
+
proxyReq.on("socket", (socket) => {
|
|
329
|
+
socket.setTimeout(0);
|
|
330
|
+
});
|
|
331
|
+
req.on("socket", (socket) => {
|
|
332
|
+
socket.setTimeout(0);
|
|
333
|
+
});
|
|
211
334
|
req.pipe(proxyReq);
|
|
212
335
|
});
|
|
213
336
|
server.on("error", (err) => {
|
|
214
337
|
reject(err);
|
|
215
338
|
});
|
|
339
|
+
server.timeout = 0;
|
|
340
|
+
server.keepAliveTimeout = 0;
|
|
216
341
|
server.listen(port, () => {
|
|
217
342
|
const address = server.address();
|
|
218
343
|
const actualPort = typeof address === "object" && address ? address.port : port;
|
package/lib/core/service.d.ts
CHANGED
|
@@ -1,6 +1,7 @@
|
|
|
1
1
|
import type { ResultPromise } from "execa";
|
|
2
2
|
import type http from "http";
|
|
3
3
|
import type { OpenCodeOptions, ServiceStartupTask } from "@vite-plugin-opencode-assistant/shared";
|
|
4
|
+
import { ChromeMcpWarmupErrorType } from "@vite-plugin-opencode-assistant/shared";
|
|
4
5
|
import type { OpenCodeAPI } from "./api.js";
|
|
5
6
|
export declare class OpenCodeService {
|
|
6
7
|
private config;
|
|
@@ -13,16 +14,22 @@ export declare class OpenCodeService {
|
|
|
13
14
|
actualProxyPort: number;
|
|
14
15
|
isStarted: boolean;
|
|
15
16
|
private startPromise;
|
|
16
|
-
sessionUrl: string | null;
|
|
17
17
|
private proxyServer;
|
|
18
18
|
chromeMcpWarmupFailed: boolean;
|
|
19
|
+
chromeMcpWarmupErrorType: ChromeMcpWarmupErrorType | null;
|
|
20
|
+
chromeMcpWarmupErrorMessage: string | null;
|
|
19
21
|
currentTask: {
|
|
20
22
|
task: ServiceStartupTask;
|
|
21
23
|
data?: Record<string, unknown>;
|
|
22
24
|
} | null;
|
|
25
|
+
workspaceRoot: string | null;
|
|
23
26
|
constructor(config: Required<OpenCodeOptions>, api: OpenCodeAPI, sseClients: Set<http.ServerResponse>, onPortAllocated: (port: number) => void, onProxyPortAllocated: (port: number) => void);
|
|
24
27
|
private sendTaskUpdate;
|
|
25
28
|
start(corsOrigins?: string[], contextApiUrl?: string, viteOrigin?: string): Promise<void>;
|
|
26
|
-
retryWarmupChromeMcp(viteOrigin?: string): Promise<
|
|
29
|
+
retryWarmupChromeMcp(viteOrigin?: string): Promise<{
|
|
30
|
+
success: boolean;
|
|
31
|
+
errorType?: string;
|
|
32
|
+
errorMessage?: string;
|
|
33
|
+
}>;
|
|
27
34
|
stop(): Promise<void>;
|
|
28
35
|
}
|
package/lib/core/service.js
CHANGED
|
@@ -72,10 +72,12 @@ class OpenCodeService {
|
|
|
72
72
|
__publicField(this, "actualProxyPort");
|
|
73
73
|
__publicField(this, "isStarted", false);
|
|
74
74
|
__publicField(this, "startPromise", null);
|
|
75
|
-
__publicField(this, "sessionUrl", null);
|
|
76
75
|
__publicField(this, "proxyServer", null);
|
|
77
76
|
__publicField(this, "chromeMcpWarmupFailed", false);
|
|
77
|
+
__publicField(this, "chromeMcpWarmupErrorType", null);
|
|
78
|
+
__publicField(this, "chromeMcpWarmupErrorMessage", null);
|
|
78
79
|
__publicField(this, "currentTask", null);
|
|
80
|
+
__publicField(this, "workspaceRoot", null);
|
|
79
81
|
var _a;
|
|
80
82
|
this.actualWebPort = config.webPort;
|
|
81
83
|
this.actualProxyPort = (_a = config.proxyPort) != null ? _a : import_shared.DEFAULT_PROXY_PORT;
|
|
@@ -149,8 +151,10 @@ Please install OpenCode first:
|
|
|
149
151
|
log.debug(`Using port ${this.actualWebPort}`);
|
|
150
152
|
}
|
|
151
153
|
timer.checkpoint("Port allocated");
|
|
154
|
+
this.workspaceRoot = (0, import_system.findGitRoot)(process.cwd());
|
|
155
|
+
log.info(`Using workspace root: ${this.workspaceRoot}`);
|
|
152
156
|
this.sendTaskUpdate("preparing_runtime");
|
|
153
|
-
const configDir = (0, import_opencode.prepareOpenCodeRuntime)(
|
|
157
|
+
const configDir = (0, import_opencode.prepareOpenCodeRuntime)(this.workspaceRoot);
|
|
154
158
|
timer.checkpoint("Plugin setup complete");
|
|
155
159
|
this.sendTaskUpdate("starting_web");
|
|
156
160
|
log.debug("Starting OpenCode Web process...", {
|
|
@@ -162,7 +166,7 @@ Please install OpenCode first:
|
|
|
162
166
|
port: this.actualWebPort,
|
|
163
167
|
hostname: this.config.hostname,
|
|
164
168
|
serverUrl: "",
|
|
165
|
-
cwd:
|
|
169
|
+
cwd: this.workspaceRoot,
|
|
166
170
|
configDir,
|
|
167
171
|
corsOrigins,
|
|
168
172
|
contextApiUrl
|
|
@@ -235,39 +239,30 @@ Please install OpenCode first:
|
|
|
235
239
|
this.sendTaskUpdate("warming_up_chrome");
|
|
236
240
|
let warmupFailed = false;
|
|
237
241
|
try {
|
|
238
|
-
yield this.api.warmupChromeMcp(viteOrigin);
|
|
242
|
+
yield this.api.warmupChromeMcp(this.workspaceRoot, viteOrigin);
|
|
239
243
|
timer.checkpoint("Chrome MCP warmup complete");
|
|
240
244
|
} catch (e) {
|
|
241
245
|
log.warn("Chrome MCP warmup failed", { error: e });
|
|
242
246
|
this.chromeMcpWarmupFailed = true;
|
|
243
247
|
warmupFailed = true;
|
|
248
|
+
if (e instanceof import_shared.ChromeMcpWarmupError) {
|
|
249
|
+
this.chromeMcpWarmupErrorType = e.type;
|
|
250
|
+
this.chromeMcpWarmupErrorMessage = e.message;
|
|
251
|
+
} else {
|
|
252
|
+
this.chromeMcpWarmupErrorType = import_shared.ChromeMcpWarmupErrorType.UNKNOWN;
|
|
253
|
+
this.chromeMcpWarmupErrorMessage = e instanceof Error ? e.message : String(e);
|
|
254
|
+
}
|
|
244
255
|
}
|
|
245
256
|
this.sendTaskUpdate("creating_session");
|
|
246
|
-
|
|
247
|
-
|
|
248
|
-
|
|
249
|
-
|
|
250
|
-
|
|
251
|
-
} catch (e) {
|
|
252
|
-
log.warn("Failed to get/create session", { error: e });
|
|
253
|
-
sessionFailed = true;
|
|
254
|
-
}
|
|
255
|
-
if (sessionFailed) {
|
|
256
|
-
this.sendTaskUpdate("session_creation_failed");
|
|
257
|
-
this.isStarted = false;
|
|
258
|
-
this.startPromise = null;
|
|
259
|
-
} else if (warmupFailed) {
|
|
260
|
-
this.sendTaskUpdate("chrome_mcp_failed", { sessionUrl: this.sessionUrl });
|
|
261
|
-
this.isStarted = true;
|
|
262
|
-
} else {
|
|
263
|
-
this.sendTaskUpdate("ready", { sessionUrl: this.sessionUrl });
|
|
264
|
-
}
|
|
265
|
-
if (!sessionFailed) {
|
|
257
|
+
if (warmupFailed) {
|
|
258
|
+
this.sendTaskUpdate("chrome_mcp_failed", {
|
|
259
|
+
errorType: this.chromeMcpWarmupErrorType,
|
|
260
|
+
errorMessage: this.chromeMcpWarmupErrorMessage
|
|
261
|
+
});
|
|
266
262
|
this.isStarted = true;
|
|
267
263
|
} else {
|
|
268
|
-
this.
|
|
264
|
+
this.sendTaskUpdate("ready");
|
|
269
265
|
}
|
|
270
|
-
log.debug(`OpenCode services started successfully: ${this.sessionUrl || webUrl}`);
|
|
271
266
|
timer.end("\u2713 Services started successfully");
|
|
272
267
|
}))();
|
|
273
268
|
return this.startPromise;
|
|
@@ -275,12 +270,18 @@ Please install OpenCode first:
|
|
|
275
270
|
}
|
|
276
271
|
retryWarmupChromeMcp(viteOrigin) {
|
|
277
272
|
return __async(this, null, function* () {
|
|
278
|
-
const
|
|
279
|
-
if (success) {
|
|
273
|
+
const result = yield this.api.retryWarmupChromeMcp(this.workspaceRoot, viteOrigin);
|
|
274
|
+
if (result.success) {
|
|
280
275
|
this.chromeMcpWarmupFailed = false;
|
|
281
|
-
this.sendTaskUpdate("ready"
|
|
276
|
+
this.sendTaskUpdate("ready");
|
|
277
|
+
return { success: true };
|
|
282
278
|
}
|
|
283
|
-
|
|
279
|
+
const error = result.error;
|
|
280
|
+
return {
|
|
281
|
+
success: false,
|
|
282
|
+
errorType: error == null ? void 0 : error.type,
|
|
283
|
+
errorMessage: (error == null ? void 0 : error.message) || "Unknown error"
|
|
284
|
+
};
|
|
284
285
|
});
|
|
285
286
|
}
|
|
286
287
|
stop() {
|
package/lib/endpoints/index.js
CHANGED
|
@@ -31,7 +31,7 @@ __reExport(endpoints_exports, require("./types.js"), module.exports);
|
|
|
31
31
|
function setupMiddlewares(server, ctx) {
|
|
32
32
|
(0, import_widget.setupWidgetEndpoints)(server, ctx);
|
|
33
33
|
(0, import_context.setupContextEndpoint)(server, ctx);
|
|
34
|
-
(0, import_start.setupStartEndpoint)(server
|
|
34
|
+
(0, import_start.setupStartEndpoint)(server);
|
|
35
35
|
(0, import_sse.setupSseEndpoint)(server, ctx);
|
|
36
36
|
(0, import_sessions.setupSessionsEndpoint)(server, ctx);
|
|
37
37
|
(0, import_warmup.setupWarmupEndpoint)(server, ctx);
|
package/lib/endpoints/sse.js
CHANGED
package/lib/endpoints/start.d.ts
CHANGED
package/lib/endpoints/start.js
CHANGED
|
@@ -42,13 +42,13 @@ __export(start_exports, {
|
|
|
42
42
|
module.exports = __toCommonJS(start_exports);
|
|
43
43
|
var import_shared = require("@vite-plugin-opencode-assistant/shared");
|
|
44
44
|
var import_shared2 = require("@vite-plugin-opencode-assistant/shared");
|
|
45
|
-
function setupStartEndpoint(server
|
|
45
|
+
function setupStartEndpoint(server) {
|
|
46
46
|
server.middlewares.use(import_shared.START_API_PATH, (_req, res) => __async(null, null, function* () {
|
|
47
47
|
const reqCtx = new import_shared2.RequestContext("GET", import_shared.START_API_PATH);
|
|
48
48
|
res.setHeader("Content-Type", "application/json");
|
|
49
49
|
res.setHeader("Access-Control-Allow-Origin", "*");
|
|
50
50
|
res.writeHead(200);
|
|
51
|
-
res.end(JSON.stringify({ success: true
|
|
51
|
+
res.end(JSON.stringify({ success: true }));
|
|
52
52
|
reqCtx.end(200);
|
|
53
53
|
}));
|
|
54
54
|
}
|
package/lib/endpoints/types.d.ts
CHANGED
|
@@ -1,7 +1,6 @@
|
|
|
1
1
|
import type { PageContext, SessionInfo, ServiceStartupTask } from "@vite-plugin-opencode-assistant/shared";
|
|
2
2
|
import type http from "http";
|
|
3
3
|
export interface EndpointContext {
|
|
4
|
-
get sessionUrl(): string | null;
|
|
5
4
|
get webUrl(): string | null;
|
|
6
5
|
get sseClients(): Set<http.ServerResponse>;
|
|
7
6
|
get pageContext(): PageContext;
|
|
@@ -16,5 +15,9 @@ export interface EndpointContext {
|
|
|
16
15
|
deleteSession: (id: string) => Promise<void>;
|
|
17
16
|
resolveWidgetPath: () => string;
|
|
18
17
|
resolveWidgetStylePath: () => string;
|
|
19
|
-
retryWarmupChromeMcp: () => Promise<
|
|
18
|
+
retryWarmupChromeMcp: () => Promise<{
|
|
19
|
+
success: boolean;
|
|
20
|
+
errorType?: string;
|
|
21
|
+
errorMessage?: string;
|
|
22
|
+
}>;
|
|
20
23
|
}
|
package/lib/endpoints/warmup.js
CHANGED
|
@@ -50,15 +50,27 @@ function setupWarmupEndpoint(server, ctx) {
|
|
|
50
50
|
return;
|
|
51
51
|
}
|
|
52
52
|
try {
|
|
53
|
-
const
|
|
53
|
+
const result = yield ctx.retryWarmupChromeMcp();
|
|
54
54
|
res.setHeader("Content-Type", "application/json");
|
|
55
55
|
res.writeHead(200);
|
|
56
|
-
|
|
56
|
+
if (result.success) {
|
|
57
|
+
res.end(JSON.stringify({ success: true }));
|
|
58
|
+
} else {
|
|
59
|
+
res.end(JSON.stringify({
|
|
60
|
+
success: false,
|
|
61
|
+
errorType: result.errorType,
|
|
62
|
+
error: result.errorMessage
|
|
63
|
+
}));
|
|
64
|
+
}
|
|
57
65
|
} catch (e) {
|
|
58
66
|
log.error("Failed to retry warmup", { error: e });
|
|
59
67
|
res.setHeader("Content-Type", "application/json");
|
|
60
68
|
res.writeHead(500);
|
|
61
|
-
res.end(JSON.stringify({
|
|
69
|
+
res.end(JSON.stringify({
|
|
70
|
+
success: false,
|
|
71
|
+
errorType: "UNKNOWN",
|
|
72
|
+
error: String(e)
|
|
73
|
+
}));
|
|
62
74
|
}
|
|
63
75
|
}));
|
|
64
76
|
}
|
package/lib/index.js
CHANGED
|
@@ -92,7 +92,12 @@ function createOpenCodePlugin(options = {}) {
|
|
|
92
92
|
let actualProxyPort = (_a = config.proxyPort) != null ? _a : import_shared.DEFAULT_PROXY_PORT;
|
|
93
93
|
let pageContext = { url: "", title: "" };
|
|
94
94
|
const sseClients = /* @__PURE__ */ new Set();
|
|
95
|
-
const api = new import_api.OpenCodeAPI(
|
|
95
|
+
const api = new import_api.OpenCodeAPI(
|
|
96
|
+
config.hostname,
|
|
97
|
+
() => actualWebPort,
|
|
98
|
+
() => actualProxyPort,
|
|
99
|
+
config.warmupChromeMcp
|
|
100
|
+
);
|
|
96
101
|
const service = new import_service.OpenCodeService(
|
|
97
102
|
config,
|
|
98
103
|
api,
|
|
@@ -117,9 +122,6 @@ function createOpenCodePlugin(options = {}) {
|
|
|
117
122
|
let viteOrigin = "";
|
|
118
123
|
const getViteOrigin = () => viteOrigin;
|
|
119
124
|
(0, import_endpoints.setupMiddlewares)(server, {
|
|
120
|
-
get sessionUrl() {
|
|
121
|
-
return service.sessionUrl;
|
|
122
|
-
},
|
|
123
125
|
get webUrl() {
|
|
124
126
|
return actualWebPort ? `http://${config.hostname}:${actualWebPort}` : null;
|
|
125
127
|
},
|
|
@@ -138,8 +140,8 @@ function createOpenCodePlugin(options = {}) {
|
|
|
138
140
|
get currentTask() {
|
|
139
141
|
return service.currentTask;
|
|
140
142
|
},
|
|
141
|
-
getSessions: () => api.getSessions(),
|
|
142
|
-
createSession: () => api.createSession(),
|
|
143
|
+
getSessions: () => api.getSessions(service.workspaceRoot),
|
|
144
|
+
createSession: () => api.createSession(service.workspaceRoot),
|
|
143
145
|
deleteSession: (id) => api.deleteSession(id),
|
|
144
146
|
resolveWidgetPath: import_paths.resolveWidgetPath,
|
|
145
147
|
resolveWidgetStylePath: import_paths.resolveWidgetStylePath,
|
|
@@ -195,15 +197,9 @@ function createOpenCodePlugin(options = {}) {
|
|
|
195
197
|
transformIndexHtml(html) {
|
|
196
198
|
const timer = log.timer("transformIndexHtml");
|
|
197
199
|
const widget = (0, import_injector.injectWidget)({
|
|
198
|
-
webUrl: `http://${config.hostname}:${actualWebPort}`,
|
|
199
|
-
proxyUrl: `http://${config.hostname}:${actualProxyPort}`,
|
|
200
|
-
serverUrl: `http://${config.hostname}:${actualWebPort}`,
|
|
201
200
|
position: config.position,
|
|
202
201
|
theme: config.theme,
|
|
203
202
|
open: config.open,
|
|
204
|
-
autoReload: config.autoReload,
|
|
205
|
-
cwd: process.cwd(),
|
|
206
|
-
// 不再注入 sessionUrl,客户端完全依赖 SSE 状态同步
|
|
207
203
|
hotkey: config.hotkey
|
|
208
204
|
});
|
|
209
205
|
timer.end();
|
package/lib/style.css
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
.opencode-widget{--oc-bg-main: #ffffff;--oc-bg-secondary: #f8f9fa;--oc-bg-tertiary: #f3f4f6;--oc-overlay-bg: rgba(255, 255, 255, .9);--oc-bg-inverse: #1e1e1e;--oc-text-primary: #282828;--oc-text-secondary: #4b5563;--oc-text-tertiary: #6b7280;--oc-text-placeholder: #9ca3af;--oc-text-inverse: #ffffff;--oc-border-primary: #e5e7eb;--oc-border-secondary: #d1d5db;--oc-primary: #3b82f6;--oc-primary-hover: #2563eb;--oc-primary-bg: rgba(59, 130, 246, .1);--oc-danger: #ef4444;--oc-danger-hover: #dc2626;--oc-danger-active: #b91c1c;--oc-success: #10b981;--oc-overlay: rgba(0, 0, 0, .5);--oc-tooltip-bg: #1e1e1e;--oc-dialog-overlay: rgba(0, 0, 0, .5);--oc-skeleton-bg: #e5e7eb;--oc-skeleton-gradient: linear-gradient(90deg, #e5e7eb 25%, #f3f4f6 50%, #e5e7eb 75%);--oc-shadow-sm: 0 2px 4px rgba(0, 0, 0, .1);--oc-shadow-md: 0 4px 12px rgba(0, 0, 0, .15);--oc-shadow-lg: 0 8px 32px rgba(0, 0, 0, .12);--oc-shadow-xl: 0 20px 60px rgba(0, 0, 0, .3);--oc-shadow-primary: 0 2px 4px rgba(59, 130, 246, .2);--oc-shadow-primary-hover: 0 4px 6px rgba(59, 130, 246, .3);--oc-shadow-danger: 0 4px 12px rgba(239, 68, 68, .3);--oc-trigger-bg: linear-gradient(135deg, #667eea 0%, #764ba2 100%);--oc-trigger-bg-hover: linear-gradient(135deg, #764ba2 0%, #667eea 100%);--oc-trigger-bg-active: linear-gradient(135deg, #f093fb 0%, #f5576c 100%);--oc-trigger-shadow: 0 4px 15px rgba(102, 126, 234, .4);--oc-trigger-shadow-hover: 0 6px 20px rgba(102, 126, 234, .6);--oc-trigger-shadow-active: 0 6px 20px rgba(240, 147, 251, .4);position:fixed;z-index:999999;font-family:-apple-system,BlinkMacSystemFont,Segoe UI,Roboto,sans-serif}.opencode-widget.opencode-theme-dark{--oc-bg-main: #1a1a1a;--oc-bg-secondary: #1e1e1e;--oc-bg-tertiary: #282828;--oc-overlay-bg: rgba(26, 26, 26, .9);--oc-bg-inverse: #ffffff;--oc-text-primary: #f3f4f6;--oc-text-secondary: #d1d5db;--oc-text-tertiary: #9ca3af;--oc-text-placeholder: #6b7280;--oc-text-inverse: #282828;--oc-border-primary: #282828;--oc-border-secondary: #4b5563;--oc-primary: #3b82f6;--oc-primary-hover: #2563eb;--oc-primary-bg: rgba(59, 130, 246, .15);--oc-danger: #ef4444;--oc-danger-hover: #dc2626;--oc-danger-active: #b91c1c;--oc-success: #10b981;--oc-overlay: rgba(26, 26, 26, .9);--oc-tooltip-bg: #282828;--oc-dialog-overlay: rgba(0, 0, 0, .7);--oc-skeleton-bg: #151515;--oc-skeleton-gradient: linear-gradient(90deg, #282828 25%, #4b5563 50%, #282828 75%);--oc-shadow-sm: 0 2px 4px rgba(0, 0, 0, .3);--oc-shadow-md: 0 4px 12px rgba(0, 0, 0, .4);--oc-shadow-lg: 0 8px 32px rgba(0, 0, 0, .4);--oc-shadow-xl: 0 20px 60px rgba(0, 0, 0, .6);--oc-shadow-primary: 0 2px 4px rgba(59, 130, 246, .3);--oc-shadow-primary-hover: 0 4px 6px rgba(59, 130, 246, .4);--oc-shadow-danger: 0 4px 12px rgba(239, 68, 68, .4);--oc-trigger-bg: linear-gradient(135deg, #667eea 0%, #764ba2 100%);--oc-trigger-bg-hover: linear-gradient(135deg, #764ba2 0%, #667eea 100%);--oc-trigger-bg-active: linear-gradient(135deg, #f093fb 0%, #f5576c 100%);--oc-trigger-shadow: 0 4px 15px rgba(102, 126, 234, .5);--oc-trigger-shadow-hover: 0 6px 20px rgba(102, 126, 234, .7);--oc-trigger-shadow-active: 0 6px 20px rgba(240, 147, 251, .5)}.opencode-widget.bottom-right{bottom:20px;right:20px}.opencode-widget.bottom-left{bottom:20px;left:20px}.opencode-widget.top-right{top:20px;right:20px}.opencode-widget.top-left{top:20px;left:20px}.opencode-chat{position:absolute;width:700px;height:86vh;background:var(--oc-bg-main);border-radius:16px;box-shadow:var(--oc-shadow-lg);overflow:hidden;opacity:0;visibility:hidden;transform:translateY(20px) scale(.95);transition:all .3s ease;display:flex;flex-direction:column}.opencode-chat-content{display:flex;flex:1;overflow:hidden}.opencode-widget.bottom-right .opencode-chat{bottom:56px;right:0}.opencode-widget.bottom-left .opencode-chat{bottom:56px;left:0}.opencode-widget.top-right .opencode-chat{top:56px;right:0}.opencode-widget.top-left .opencode-chat{top:56px;left:0}.opencode-widget.bottom-right .opencode-selected-bubbles{bottom:56px;right:0}.opencode-widget.bottom-left .opencode-selected-bubbles{bottom:56px;left:0}.opencode-widget.top-right .opencode-selected-bubbles{top:56px;bottom:auto;right:0}.opencode-widget.top-left .opencode-selected-bubbles{top:56px;bottom:auto;left:0}.opencode-chat.open{opacity:1;visibility:visible;transform:translateY(0) scale(1)}.opencode-notification{position:absolute;top:20px;left:50%;transform:translate(-50%);padding:12px 24px;background:linear-gradient(135deg,#3b82f6,#2563eb);color:#fff;border-radius:10px;font-size:14px;font-weight:500;box-shadow:0 4px 16px #3b82f666,0 0 0 2px #3b82f633;animation:slideDown .3s ease;z-index:10000000;display:flex;align-items:center;gap:10px}.opencode-notification:before{content:"💡";font-size:16px}.opencode-dialog-overlay{position:fixed;inset:0;background:var(--oc-dialog-overlay);display:flex;align-items:center;justify-content:center;z-index:9999999;animation:fadeIn .2s ease}@keyframes fadeIn{0%{opacity:0}to{opacity:1}}.opencode-dialog{background:var(--oc-bg-main);border-radius:12px;padding:24px;min-width:320px;max-width:400px;box-shadow:var(--oc-shadow-xl);animation:scaleIn .2s ease}@keyframes scaleIn{0%{transform:scale(.9);opacity:0}to{transform:scale(1);opacity:1}}.opencode-dialog-content{margin-bottom:20px}.opencode-dialog-message{font-size:15px;color:var(--oc-text-primary);line-height:1.5}.opencode-dialog-actions{display:flex;gap:12px;justify-content:flex-end}.opencode-dialog-btn{padding:10px 20px;border-radius:8px;border:none;font-size:14px;font-weight:500;cursor:pointer;transition:all .2s}.opencode-dialog-btn.cancel{background:var(--oc-bg-tertiary);color:var(--oc-text-primary)}.opencode-dialog-btn.cancel:hover{background:var(--oc-text-primary);color:var(--oc-bg-main)}.opencode-dialog-btn.confirm{background:var(--oc-danger);color:#fff}.opencode-dialog-btn.confirm:hover{background:var(--oc-danger-hover)}@keyframes slideDown{0%{transform:translate(-50%) translateY(-100%);opacity:0}to{transform:translate(-50%) translateY(0);opacity:1}}.opencode-element-highlight{position:fixed;pointer-events:none;z-index:999998;display:none;transition:all .1s ease;border-radius:4px}#vue-inspector-container{display:none!important}.opencode-element-tooltip{position:fixed;background:var(--oc-tooltip-bg);color:#fff;padding:8px 12px;border-radius:6px;font-size:12px;z-index:9999998;display:none;box-shadow:var(--oc-shadow-md);max-width:300px;pointer-events:none}.opencode-tooltip-tag{font-weight:500;margin-bottom:4px;word-break:break-all}.opencode-tooltip-file{font-size:11px;color:var(--oc-text-placeholder);word-break:break-all}.opencode-element-highlight-temp{position:absolute;pointer-events:none;z-index:999998;border-radius:4px;animation:highlight-pulse 2s ease-out forwards}@keyframes highlight-pulse{0%{opacity:1;transform:scale(1)}50%{opacity:.8;transform:scale(1.02)}to{opacity:0;transform:scale(1)}}@media (max-width:768px){.opencode-chat{width:calc(100vw - 40px);height:calc(100vh - 100px)}}.opencode-iframe-container{flex:1;position:relative;overflow:hidden;display:flex;flex-direction:column;margin-top:-42px}.opencode-loading-overlay{position:absolute;inset:0;background:var(--oc-overlay-bg);display:none;flex-direction:column;align-items:center;justify-content:center;z-index:10;transition:opacity .3s ease}.opencode-loading-overlay.visible{display:flex}.opencode-loading-spinner{width:40px;height:40px;border:3px solid var(--oc-border-primary);border-top-color:var(--oc-primary);border-radius:50%;animation:spin .8s linear infinite}@keyframes spin{to{transform:rotate(360deg)}}.opencode-loading-text{margin-top:12px;font-size:14px;color:var(--oc-text-placeholder)}.opencode-error-overlay{position:absolute;inset:0;z-index:15;margin-top:42px;display:none}.opencode-error-overlay.visible{display:flex}.opencode-empty-state-overlay{position:absolute;inset:0;background:var(--oc-bg-secondary);display:none;flex-direction:column;align-items:center;justify-content:center;z-index:5;transition:opacity .3s ease;margin-top:42px}.opencode-empty-state-overlay.visible{display:flex}.opencode-empty-state-icon{color:var(--oc-text-placeholder);margin-bottom:16px}.opencode-empty-state-text{color:var(--oc-text-primary);font-size:16px;font-weight:500;margin-bottom:24px}.opencode-empty-state-btn{padding:10px 24px;border-radius:8px;border:none;background:var(--oc-primary);color:#fff;font-size:14px;font-weight:500;cursor:pointer;transition:all .2s;box-shadow:var(--oc-shadow-primary)}.opencode-empty-state-btn:hover{background:var(--oc-primary-hover);transform:translateY(-1px);box-shadow:var(--oc-shadow-primary-hover)}.opencode-empty-state-btn:active{transform:translateY(0)}.opencode-iframe{width:100%;height:100%;border:none}.opencode-chat-header{position:relative;flex-shrink:0;display:flex;align-items:center;justify-content:space-between;padding:0 12px;height:40px;background:var(--oc-bg-secondary);border-bottom:1px solid var(--oc-border-primary);z-index:5}.opencode-chat-header-left{display:flex;align-items:center;gap:4px}.opencode-chat-header-title{font-size:14px;font-weight:600;color:var(--oc-text-primary);position:absolute;left:50%;transform:translate(-50%)}.opencode-chat-header-actions{display:flex;gap:4px}.opencode-header-btn{width:28px;height:28px;border-radius:6px;border:none;background:transparent;color:var(--oc-text-placeholder);cursor:pointer;display:flex;align-items:center;justify-content:center;transition:all .2s}.opencode-header-btn:hover{background:var(--oc-bg-tertiary);color:var(--oc-text-primary)}.opencode-header-btn.close:hover{background:var(--oc-danger);color:#fff}.opencode-header-btn.select-btn.active,.opencode-header-btn.session-toggle.active{background:var(--oc-primary);color:#fff}.opencode-select-mode-hint{position:fixed;top:20px;left:50%;transform:translate(-50%);padding:10px;background:linear-gradient(135deg,#ef4444,#dc2626);color:#fff;border-radius:12px;font-size:14px;font-weight:500;box-shadow:0 6px 20px #ef444480,0 0 0 3px #ef44444d;z-index:9999999;display:none;align-items:center;gap:12px;border:1px solid rgba(255,255,255,.3)}.opencode-select-mode-hint.visible{display:flex;animation:slideDown .3s ease,pulseHint 2s ease-in-out infinite}.opencode-hint-shortcut{padding:4px 10px;background:#ffffff40;border-radius:6px;font-size:13px;font-weight:600;border:1px solid rgba(255,255,255,.4)}@keyframes pulseHint{0%,to{box-shadow:0 6px 20px #ef444480,0 0 0 3px #ef44444d}50%{box-shadow:0 6px 20px #ef444499,0 0 0 6px #ef444466}}.opencode-selected-bubbles{position:absolute;display:none;flex-direction:column;gap:6px;max-width:220px;max-height:300px;overflow-y:auto}.opencode-selected-bubbles.visible{display:flex}.opencode-selected-bubble{display:flex;flex-direction:column;gap:2px;padding:8px 24px 8px 10px;background:var(--oc-bg-main);border:1px solid var(--oc-border-primary);border-radius:8px;font-size:12px;box-shadow:var(--oc-shadow-sm);position:relative;cursor:pointer;transition:all .2s}.opencode-selected-bubble:hover{border-color:var(--oc-primary);box-shadow:var(--oc-shadow-primary)}.opencode-bubble-text{color:var(--oc-text-primary);font-weight:500;overflow:hidden;text-overflow:ellipsis;white-space:nowrap}.opencode-bubble-file{color:var(--oc-text-placeholder);font-size:11px;overflow:hidden;text-overflow:ellipsis;white-space:nowrap}.opencode-bubble-remove{position:absolute;top:8px;right:6px;width:16px;height:16px;border-radius:50%;border:none;background:transparent;color:var(--oc-text-placeholder);cursor:pointer;display:flex;align-items:center;justify-content:center;font-size:12px;transition:all .2s}.opencode-bubble-remove:hover{background:var(--oc-danger);color:#fff}.opencode-bubble-empty{padding:8px 12px;background:var(--oc-bg-main);border:1px dashed var(--oc-border-secondary);border-radius:8px;color:var(--oc-text-placeholder);font-size:12px;text-align:center}.opencode-right-toolbar{width:140px;background:var(--oc-bg-secondary);border-left:1px solid var(--oc-border-primary);display:flex;flex-direction:column;flex-shrink:0;transition:width .2s ease;overflow:hidden}.opencode-right-toolbar.collapsed{width:0;overflow:hidden}.opencode-right-toolbar.collapsed .opencode-selected-nodes-header,.opencode-right-toolbar.collapsed .opencode-selected-nodes,.opencode-right-toolbar.collapsed .opencode-clear-all-btn{display:none}.opencode-selected-nodes-header{padding:12px 8px 8px;border-bottom:1px solid var(--oc-border-primary)}.opencode-selected-nodes-title{font-size:14px;font-weight:600;color:var(--oc-text-primary);margin-bottom:4px}.opencode-selected-nodes-desc{font-size:11px;color:var(--oc-text-placeholder);line-height:1.4}.opencode-selected-nodes{flex:1;display:flex;flex-direction:column;padding:8px;gap:6px;overflow-y:auto;overflow-x:hidden}.opencode-selected-nodes:empty:before{content:"暂无选中元素";color:var(--oc-text-placeholder);font-size:12px;text-align:center;padding:20px 10px}.opencode-selected-node{display:flex;align-items:center;gap:8px;padding:8px 10px;background:var(--oc-bg-main);border:1px solid var(--oc-border-primary);border-radius:6px;font-size:12px;transition:all .2s}.opencode-selected-node:hover{border-color:var(--oc-primary);box-shadow:var(--oc-shadow-primary)}.opencode-node-content{flex:1;min-width:0;display:flex;flex-direction:column;gap:2px}.opencode-node-text{color:var(--oc-text-primary);font-weight:500;overflow:hidden;text-overflow:ellipsis;white-space:nowrap}.opencode-node-file{color:var(--oc-text-placeholder);font-size:11px;overflow:hidden;text-overflow:ellipsis;white-space:nowrap}.opencode-node-remove{width:18px;height:18px;border-radius:4px;border:none;background:transparent;color:var(--oc-text-placeholder);cursor:pointer;display:flex;align-items:center;justify-content:center;font-size:14px;transition:all .2s;flex-shrink:0}.opencode-node-remove:hover{background:var(--oc-danger);color:#fff}.opencode-clear-all-btn{width:calc(100% - 16px);margin:8px;padding:8px 12px;border-radius:6px;border:none;background:var(--oc-danger);color:#fff;font-size:12px;font-weight:500;cursor:pointer;display:flex;align-items:center;justify-content:center;gap:4px;transition:all .2s}.opencode-clear-all-btn:hover{background:var(--oc-danger-hover);transform:scale(1.02)}.opencode-session-list{width:240px;background:var(--oc-bg-secondary);border-right:1px solid var(--oc-border-primary);display:flex;flex-direction:column;flex-shrink:0;transition:width .2s ease}.opencode-session-list.collapsed{width:0;overflow:hidden}.opencode-session-list.collapsed .opencode-session-list-header,.opencode-session-list.collapsed .opencode-session-list-content{display:none}.opencode-session-list-header{padding:16px;border-bottom:1px solid var(--oc-border-primary);display:flex;justify-content:space-between;align-items:center;font-weight:600;font-size:14px;color:var(--oc-text-primary)}.opencode-new-session-btn{width:28px;height:28px;border-radius:6px;border:none;background:var(--oc-primary);color:#fff;font-size:18px;cursor:pointer;display:flex;align-items:center;justify-content:center;transition:all .2s}.opencode-new-session-btn:hover{background:var(--oc-primary-hover);transform:scale(1.05)}.opencode-session-list-content{flex:1;overflow-y:auto;padding:8px;position:relative}.opencode-session-list-loading-overlay{position:absolute;inset:0;background:var(--oc-overlay-bg);display:flex;align-items:center;justify-content:center;z-index:10;border-radius:8px}.opencode-loading-spinner.small{width:24px;height:24px;border-width:2px}.opencode-session-item{padding:12px;border-radius:8px;cursor:pointer;transition:transform .2s;margin-bottom:4px;color:var(--oc-text-primary)}.opencode-session-item:hover{background:var(--oc-bg-tertiary)}.opencode-session-item.active{background:var(--oc-primary);color:#fff;transition:none}.opencode-session-title{font-size:14px;font-weight:500;margin-bottom:4px;overflow:hidden;text-overflow:ellipsis;white-space:nowrap}.opencode-session-meta{font-size:12px;opacity:.6}.opencode-session-header{display:flex;justify-content:space-between;align-items:center;margin-bottom:4px}.opencode-session-delete-btn{width:20px;height:20px;border-radius:4px;border:none;background:transparent;color:var(--oc-text-placeholder);font-size:16px;cursor:pointer;display:flex;align-items:center;justify-content:center;transition:all .2s;opacity:0;flex-shrink:0}.opencode-session-item:hover .opencode-session-delete-btn{opacity:1}.opencode-session-delete-btn:hover{background:var(--oc-danger);color:#fff}.opencode-session-item.active .opencode-session-delete-btn{color:#ffffffb3}.opencode-session-item.active .opencode-session-delete-btn:hover{background:#fff3;color:#fff}.opencode-session-header-skeleton{padding:16px;border-bottom:1px solid var(--oc-border-primary);display:none;justify-content:space-between;align-items:center}.opencode-session-header-skeleton.visible{display:flex}.opencode-skeleton-header-title{height:18px;width:80px;background:var(--oc-skeleton-gradient);background-size:200% 100%;animation:skeleton-loading 1.5s ease-in-out infinite;border-radius:4px}.opencode-skeleton-header-btn{width:28px;height:28px;background:var(--oc-skeleton-gradient);background-size:200% 100%;animation:skeleton-loading 1.5s ease-in-out infinite;border-radius:6px}.opencode-session-skeleton{flex:1;overflow-y:auto;padding:8px;display:none}.opencode-session-skeleton.visible{display:block}.opencode-skeleton-item{padding:12px;border-radius:8px;margin-bottom:4px;background:var(--oc-skeleton-bg)}.opencode-skeleton-title{height:16px;background:var(--oc-skeleton-gradient);background-size:200% 100%;animation:skeleton-loading 1.5s ease-in-out infinite;border-radius:4px;margin-bottom:8px;width:70%}.opencode-skeleton-meta{height:12px;background:var(--oc-skeleton-gradient);background-size:200% 100%;animation:skeleton-loading 1.5s ease-in-out infinite;border-radius:4px;width:50%}.opencode-session-empty{padding:32px 16px;text-align:center;color:var(--oc-text-placeholder);font-size:13px}@keyframes skeleton-loading{0%{background-position:200% 0}to{background-position:-200% 0}}.opencode-button{width:44px;height:44px;border-radius:50%;background:var(--oc-trigger-bg);border:none;cursor:pointer;box-shadow:var(--oc-trigger-shadow);transition:all .3s ease;display:flex;align-items:center;justify-content:center;color:#fff;padding:0;position:relative}.opencode-button:before{content:"";position:absolute;inset:-8px;border-radius:50%}.opencode-button:hover{transform:scale(1.1);box-shadow:var(--oc-trigger-shadow-hover);background:var(--oc-trigger-bg-hover)}.opencode-button.active{background:var(--oc-trigger-bg-active);box-shadow:var(--oc-trigger-shadow-active)}.opencode-button.active svg{transform:rotate(180deg)}.opencode-button svg{transition:transform .3s ease}.opencode-button.loading{animation:pulse 1s infinite}@keyframes pulse{0%,to{opacity:1}50%{opacity:.5}}
|
|
1
|
+
.opencode-widget{--oc-bg-main: #ffffff;--oc-bg-secondary: #f8f9fa;--oc-bg-tertiary: #f3f4f6;--oc-overlay-bg: rgba(255, 255, 255, .9);--oc-bg-inverse: #1e1e1e;--oc-text-primary: #282828;--oc-text-secondary: #4b5563;--oc-text-tertiary: #6b7280;--oc-text-placeholder: #9ca3af;--oc-text-inverse: #ffffff;--oc-border-primary: #e5e7eb;--oc-border-secondary: #d1d5db;--oc-primary: #3b82f6;--oc-primary-hover: #2563eb;--oc-primary-bg: rgba(59, 130, 246, .1);--oc-danger: #ef4444;--oc-danger-hover: #dc2626;--oc-danger-active: #b91c1c;--oc-success: #10b981;--oc-overlay: rgba(0, 0, 0, .5);--oc-tooltip-bg: #1e1e1e;--oc-dialog-overlay: rgba(0, 0, 0, .5);--oc-thinking-gradient-1: #10b981;--oc-thinking-gradient-2: #059669;--oc-thinking-glow: rgba(16, 185, 129, .3);--oc-thinking-glow-strong: rgba(16, 185, 129, .6);--oc-skeleton-bg: #e5e7eb;--oc-skeleton-gradient: linear-gradient(90deg, #e5e7eb 25%, #f3f4f6 50%, #e5e7eb 75%);--oc-shadow-sm: 0 2px 4px rgba(0, 0, 0, .1);--oc-shadow-md: 0 4px 12px rgba(0, 0, 0, .15);--oc-shadow-lg: 0 8px 32px rgba(0, 0, 0, .12);--oc-shadow-xl: 0 20px 60px rgba(0, 0, 0, .3);--oc-shadow-primary: 0 2px 4px rgba(59, 130, 246, .2);--oc-shadow-primary-hover: 0 4px 6px rgba(59, 130, 246, .3);--oc-shadow-danger: 0 4px 12px rgba(239, 68, 68, .3);--oc-trigger-bg: #3b82f6;--oc-trigger-bg-hover: #2563eb;--oc-trigger-bg-active: #1d4ed8;--oc-trigger-shadow: 0 2px 8px rgba(59, 130, 246, .3);--oc-trigger-shadow-hover: 0 4px 12px rgba(59, 130, 246, .4);--oc-trigger-shadow-active: 0 4px 12px rgba(59, 130, 246, .5);position:fixed;z-index:999999;font-family:-apple-system,BlinkMacSystemFont,Segoe UI,Roboto,sans-serif}.opencode-widget.opencode-theme-dark{--oc-bg-main: #1a1a1a;--oc-bg-secondary: #1e1e1e;--oc-bg-tertiary: #282828;--oc-overlay-bg: rgba(26, 26, 26, .9);--oc-bg-inverse: #ffffff;--oc-text-primary: #f3f4f6;--oc-text-secondary: #d1d5db;--oc-text-tertiary: #9ca3af;--oc-text-placeholder: #6b7280;--oc-text-inverse: #282828;--oc-border-primary: #282828;--oc-border-secondary: #4b5563;--oc-primary: #3b82f6;--oc-primary-hover: #2563eb;--oc-primary-bg: rgba(59, 130, 246, .15);--oc-danger: #ef4444;--oc-danger-hover: #dc2626;--oc-danger-active: #b91c1c;--oc-success: #10b981;--oc-overlay: rgba(26, 26, 26, .9);--oc-tooltip-bg: #282828;--oc-dialog-overlay: rgba(0, 0, 0, .7);--oc-thinking-gradient-1: #34d399;--oc-thinking-gradient-2: #10b981;--oc-thinking-glow: rgba(52, 211, 153, .3);--oc-thinking-glow-strong: rgba(52, 211, 153, .6);--oc-skeleton-bg: #151515;--oc-skeleton-gradient: linear-gradient(90deg, #282828 25%, #4b5563 50%, #282828 75%);--oc-shadow-sm: 0 2px 4px rgba(0, 0, 0, .3);--oc-shadow-md: 0 4px 12px rgba(0, 0, 0, .4);--oc-shadow-lg: 0 8px 32px rgba(0, 0, 0, .4);--oc-shadow-xl: 0 20px 60px rgba(0, 0, 0, .6);--oc-shadow-primary: 0 2px 4px rgba(59, 130, 246, .3);--oc-shadow-primary-hover: 0 4px 6px rgba(59, 130, 246, .4);--oc-shadow-danger: 0 4px 12px rgba(239, 68, 68, .4);--oc-trigger-bg: #60a5fa;--oc-trigger-bg-hover: #3b82f6;--oc-trigger-bg-active: #2563eb;--oc-trigger-shadow: 0 2px 8px rgba(96, 165, 250, .4);--oc-trigger-shadow-hover: 0 4px 12px rgba(96, 165, 250, .5);--oc-trigger-shadow-active: 0 4px 12px rgba(96, 165, 250, .6)}.opencode-widget.bottom-right{bottom:20px;right:20px}.opencode-widget.bottom-left{bottom:20px;left:20px}.opencode-widget.top-right{top:20px;right:20px}.opencode-widget.top-left{top:20px;left:20px}.opencode-chat{position:absolute;width:700px;height:86vh;background:var(--oc-bg-main);border-radius:16px;box-shadow:var(--oc-shadow-lg);overflow:hidden;opacity:0;visibility:hidden;transform:translateY(20px) scale(.95);transition:all .3s ease;display:flex;flex-direction:column}.opencode-chat-content{display:flex;flex:1;overflow:hidden}.opencode-widget.bottom-right .opencode-chat{bottom:56px;right:0}.opencode-widget.bottom-left .opencode-chat{bottom:56px;left:0}.opencode-widget.top-right .opencode-chat{top:56px;right:0}.opencode-widget.top-left .opencode-chat{top:56px;left:0}.opencode-widget.bottom-right .opencode-selected-bubbles{bottom:56px;right:0}.opencode-widget.bottom-left .opencode-selected-bubbles{bottom:56px;left:0}.opencode-widget.top-right .opencode-selected-bubbles{top:56px;bottom:auto;right:0}.opencode-widget.top-left .opencode-selected-bubbles{top:56px;bottom:auto;left:0}.opencode-chat.open{opacity:1;visibility:visible;transform:translateY(0) scale(1)}.opencode-notification{position:absolute;top:20px;left:50%;transform:translate(-50%);padding:12px 24px;background:linear-gradient(135deg,#3b82f6,#2563eb);color:#fff;border-radius:10px;font-size:14px;font-weight:500;box-shadow:0 4px 16px #3b82f666,0 0 0 2px #3b82f633;animation:slideDown .3s ease;z-index:10000000;display:flex;align-items:center;gap:10px}.opencode-notification:before{content:"💡";font-size:16px}.opencode-dialog-overlay{position:fixed;inset:0;background:var(--oc-dialog-overlay);display:flex;align-items:center;justify-content:center;z-index:9999999;animation:fadeIn .2s ease}@keyframes fadeIn{0%{opacity:0}to{opacity:1}}.opencode-dialog{background:var(--oc-bg-main);border-radius:12px;padding:24px;min-width:320px;max-width:400px;box-shadow:var(--oc-shadow-xl);animation:scaleIn .2s ease}@keyframes scaleIn{0%{transform:scale(.9);opacity:0}to{transform:scale(1);opacity:1}}.opencode-dialog-content{margin-bottom:20px}.opencode-dialog-message{font-size:15px;color:var(--oc-text-primary);line-height:1.5}.opencode-dialog-actions{display:flex;gap:12px;justify-content:flex-end}.opencode-dialog-btn{padding:10px 20px;border-radius:8px;border:none;font-size:14px;font-weight:500;cursor:pointer;transition:all .2s}.opencode-dialog-btn.cancel{background:var(--oc-bg-tertiary);color:var(--oc-text-primary)}.opencode-dialog-btn.cancel:hover{background:var(--oc-text-primary);color:var(--oc-bg-main)}.opencode-dialog-btn.confirm{background:var(--oc-danger);color:#fff}.opencode-dialog-btn.confirm:hover{background:var(--oc-danger-hover)}@keyframes slideDown{0%{transform:translate(-50%) translateY(-100%);opacity:0}to{transform:translate(-50%) translateY(0);opacity:1}}.opencode-element-highlight{position:fixed;pointer-events:none;z-index:999998;display:none;transition:all .1s ease;border-radius:4px}#vue-inspector-container{display:none!important}.opencode-element-tooltip{position:fixed;background:var(--oc-tooltip-bg);color:#fff;padding:8px 12px;border-radius:6px;font-size:12px;z-index:9999998;display:none;box-shadow:var(--oc-shadow-md);max-width:300px;pointer-events:none}.opencode-tooltip-tag{font-weight:500;margin-bottom:4px;word-break:break-all}.opencode-tooltip-file{font-size:11px;color:var(--oc-text-placeholder);word-break:break-all}.opencode-element-highlight-temp{position:absolute;pointer-events:none;z-index:999998;border-radius:4px;animation:highlight-pulse 2s ease-out forwards}@keyframes highlight-pulse{0%{opacity:1;transform:scale(1)}50%{opacity:.8;transform:scale(1.02)}to{opacity:0;transform:scale(1)}}@media (max-width:768px){.opencode-chat{width:calc(100vw - 40px);height:calc(100vh - 100px)}}.opencode-iframe-container{flex:1;position:relative;overflow:hidden;display:flex;flex-direction:column;margin-top:-42px}.opencode-loading-overlay{position:absolute;inset:0;background:var(--oc-overlay-bg);display:none;flex-direction:column;align-items:center;justify-content:center;z-index:10;transition:opacity .3s ease}.opencode-loading-overlay.visible{display:flex}.opencode-loading-spinner{width:40px;height:40px;border:3px solid var(--oc-border-primary);border-top-color:var(--oc-primary);border-radius:50%;animation:spin .8s linear infinite}@keyframes spin{to{transform:rotate(360deg)}}.opencode-loading-text{margin-top:12px;font-size:14px;color:var(--oc-text-placeholder)}.opencode-error-overlay{position:absolute;inset:0;z-index:15;margin-top:42px;display:none}.opencode-error-overlay.visible{display:flex}.opencode-empty-state-overlay{position:absolute;inset:0;background:var(--oc-bg-secondary);display:none;flex-direction:column;align-items:center;justify-content:center;z-index:5;transition:opacity .3s ease;margin-top:42px}.opencode-empty-state-overlay.visible{display:flex}.opencode-empty-state-icon{color:var(--oc-text-placeholder);margin-bottom:16px}.opencode-empty-state-text{color:var(--oc-text-primary);font-size:16px;font-weight:500;margin-bottom:24px}.opencode-empty-state-btn{padding:10px 24px;border-radius:8px;border:none;background:var(--oc-primary);color:#fff;font-size:14px;font-weight:500;cursor:pointer;transition:all .2s;box-shadow:var(--oc-shadow-primary)}.opencode-empty-state-btn:hover{background:var(--oc-primary-hover);transform:translateY(-1px);box-shadow:var(--oc-shadow-primary-hover)}.opencode-empty-state-btn:active{transform:translateY(0)}.opencode-iframe{width:100%;height:100%;border:none}.opencode-chat-header{position:relative;flex-shrink:0;display:flex;align-items:center;justify-content:space-between;padding:0 12px;height:40px;background:var(--oc-bg-secondary);border-bottom:1px solid var(--oc-border-primary);z-index:5}.opencode-chat-header-left{display:flex;align-items:center;gap:4px}.opencode-chat-header-title{font-size:14px;font-weight:600;color:var(--oc-text-primary);position:absolute;left:50%;transform:translate(-50%)}.opencode-chat-header-actions{display:flex;gap:4px}.opencode-header-btn{width:28px;height:28px;border-radius:6px;border:none;background:transparent;color:var(--oc-text-placeholder);cursor:pointer;display:flex;align-items:center;justify-content:center;transition:all .2s}.opencode-header-btn:hover{background:var(--oc-bg-tertiary);color:var(--oc-text-primary)}.opencode-header-btn.close:hover{background:var(--oc-danger);color:#fff}.opencode-header-btn.select-btn.active,.opencode-header-btn.session-toggle.active{background:var(--oc-primary);color:#fff}.opencode-select-mode-hint{position:fixed;top:20px;left:50%;transform:translate(-50%);padding:10px;background:linear-gradient(135deg,#ef4444,#dc2626);color:#fff;border-radius:12px;font-size:14px;font-weight:500;box-shadow:0 6px 20px #ef444480,0 0 0 3px #ef44444d;z-index:9999999;display:none;align-items:center;gap:12px;border:1px solid rgba(255,255,255,.3)}.opencode-select-mode-hint.visible{display:flex;animation:slideDown .3s ease,pulseHint 2s ease-in-out infinite}.opencode-hint-shortcut{padding:4px 10px;background:#ffffff40;border-radius:6px;font-size:13px;font-weight:600;border:1px solid rgba(255,255,255,.4)}@keyframes pulseHint{0%,to{box-shadow:0 6px 20px #ef444480,0 0 0 3px #ef44444d}50%{box-shadow:0 6px 20px #ef444499,0 0 0 6px #ef444466}}.opencode-selected-bubbles{position:absolute;display:none;flex-direction:column;gap:6px;max-width:220px;max-height:300px;overflow-y:auto}.opencode-selected-bubbles.visible{display:flex}.opencode-selected-bubble{display:flex;flex-direction:column;gap:2px;padding:8px 24px 8px 10px;background:var(--oc-bg-main);border:1px solid var(--oc-border-primary);border-radius:8px;font-size:12px;box-shadow:var(--oc-shadow-sm);position:relative;cursor:pointer;transition:all .2s}.opencode-selected-bubble:hover{border-color:var(--oc-primary);box-shadow:var(--oc-shadow-primary)}.opencode-bubble-text{color:var(--oc-text-primary);font-weight:500;overflow:hidden;text-overflow:ellipsis;white-space:nowrap}.opencode-bubble-file{color:var(--oc-text-placeholder);font-size:11px;overflow:hidden;text-overflow:ellipsis;white-space:nowrap}.opencode-bubble-remove{position:absolute;top:8px;right:6px;width:16px;height:16px;border-radius:50%;border:none;background:transparent;color:var(--oc-text-placeholder);cursor:pointer;display:flex;align-items:center;justify-content:center;font-size:12px;transition:all .2s}.opencode-bubble-remove:hover{background:var(--oc-danger);color:#fff}.opencode-bubble-empty{padding:8px 12px;background:var(--oc-bg-main);border:1px dashed var(--oc-border-secondary);border-radius:8px;color:var(--oc-text-placeholder);font-size:12px;text-align:center}.opencode-right-toolbar{width:140px;background:var(--oc-bg-secondary);border-left:1px solid var(--oc-border-primary);display:flex;flex-direction:column;flex-shrink:0;transition:width .2s ease;overflow:hidden}.opencode-right-toolbar.collapsed{width:0;overflow:hidden}.opencode-right-toolbar.collapsed .opencode-selected-nodes-header,.opencode-right-toolbar.collapsed .opencode-selected-nodes,.opencode-right-toolbar.collapsed .opencode-clear-all-btn{display:none}.opencode-selected-nodes-header{padding:12px 8px 8px;border-bottom:1px solid var(--oc-border-primary)}.opencode-selected-nodes-title{font-size:14px;font-weight:600;color:var(--oc-text-primary);margin-bottom:4px}.opencode-selected-nodes-desc{font-size:11px;color:var(--oc-text-placeholder);line-height:1.4}.opencode-selected-nodes{flex:1;display:flex;flex-direction:column;padding:8px;gap:6px;overflow-y:auto;overflow-x:hidden}.opencode-selected-nodes:empty:before{content:"暂无选中元素";color:var(--oc-text-placeholder);font-size:12px;text-align:center;padding:20px 10px}.opencode-selected-node{display:flex;align-items:center;gap:8px;padding:8px 10px;background:var(--oc-bg-main);border:1px solid var(--oc-border-primary);border-radius:6px;font-size:12px;transition:all .2s}.opencode-selected-node:hover{border-color:var(--oc-primary);box-shadow:var(--oc-shadow-primary)}.opencode-node-content{flex:1;min-width:0;display:flex;flex-direction:column;gap:2px}.opencode-node-text{color:var(--oc-text-primary);font-weight:500;overflow:hidden;text-overflow:ellipsis;white-space:nowrap}.opencode-node-file{color:var(--oc-text-placeholder);font-size:11px;overflow:hidden;text-overflow:ellipsis;white-space:nowrap}.opencode-node-remove{width:18px;height:18px;border-radius:4px;border:none;background:transparent;color:var(--oc-text-placeholder);cursor:pointer;display:flex;align-items:center;justify-content:center;font-size:14px;transition:all .2s;flex-shrink:0}.opencode-node-remove:hover{background:var(--oc-danger);color:#fff}.opencode-clear-all-btn{width:calc(100% - 16px);margin:8px;padding:8px 12px;border-radius:6px;border:none;background:var(--oc-danger);color:#fff;font-size:12px;font-weight:500;cursor:pointer;display:flex;align-items:center;justify-content:center;gap:4px;transition:all .2s}.opencode-clear-all-btn:hover{background:var(--oc-danger-hover);transform:scale(1.02)}.opencode-session-list{width:240px;background:var(--oc-bg-secondary);border-right:1px solid var(--oc-border-primary);display:flex;flex-direction:column;flex-shrink:0;transition:width .2s ease}.opencode-session-list.collapsed{width:0;overflow:hidden}.opencode-session-list.collapsed .opencode-session-list-header,.opencode-session-list.collapsed .opencode-session-list-content{display:none}.opencode-session-list-header{padding:16px;border-bottom:1px solid var(--oc-border-primary);display:flex;justify-content:space-between;align-items:center;font-weight:600;font-size:14px;color:var(--oc-text-primary)}.opencode-new-session-btn{width:28px;height:28px;border-radius:6px;border:none;background:var(--oc-primary);color:#fff;font-size:18px;cursor:pointer;display:flex;align-items:center;justify-content:center;transition:all .2s}.opencode-new-session-btn:hover{background:var(--oc-primary-hover);transform:scale(1.05)}.opencode-session-list-content{flex:1;overflow-y:auto;padding:8px;position:relative}.opencode-session-list-loading-overlay{position:absolute;inset:0;background:var(--oc-overlay-bg);display:flex;align-items:center;justify-content:center;z-index:10;border-radius:8px}.opencode-loading-spinner.small{width:24px;height:24px;border-width:2px}.opencode-session-item{padding:12px;border-radius:8px;cursor:pointer;transition:transform .2s;margin-bottom:4px;color:var(--oc-text-primary)}.opencode-session-item:hover{background:var(--oc-bg-tertiary)}.opencode-session-item.active{background:var(--oc-primary);color:#fff;transition:none}.opencode-session-title{font-size:14px;font-weight:500;margin-bottom:4px;overflow:hidden;text-overflow:ellipsis;white-space:nowrap}.opencode-session-meta{font-size:12px;opacity:.6}.opencode-session-header{display:flex;justify-content:space-between;align-items:center;margin-bottom:4px}.opencode-session-delete-btn{width:20px;height:20px;border-radius:4px;border:none;background:transparent;color:var(--oc-text-placeholder);font-size:16px;cursor:pointer;display:flex;align-items:center;justify-content:center;transition:all .2s;opacity:0;flex-shrink:0}.opencode-session-item:hover .opencode-session-delete-btn{opacity:1}.opencode-session-delete-btn:hover{background:var(--oc-danger);color:#fff}.opencode-session-item.active .opencode-session-delete-btn{color:#ffffffb3}.opencode-session-item.active .opencode-session-delete-btn:hover{background:#fff3;color:#fff}.opencode-session-header-skeleton{padding:16px;border-bottom:1px solid var(--oc-border-primary);display:none;justify-content:space-between;align-items:center}.opencode-session-header-skeleton.visible{display:flex}.opencode-skeleton-header-title{height:18px;width:80px;background:var(--oc-skeleton-gradient);background-size:200% 100%;animation:skeleton-loading 1.5s ease-in-out infinite;border-radius:4px}.opencode-skeleton-header-btn{width:28px;height:28px;background:var(--oc-skeleton-gradient);background-size:200% 100%;animation:skeleton-loading 1.5s ease-in-out infinite;border-radius:6px}.opencode-session-skeleton{flex:1;overflow-y:auto;padding:8px;display:none}.opencode-session-skeleton.visible{display:block}.opencode-skeleton-item{padding:12px;border-radius:8px;margin-bottom:4px;background:var(--oc-skeleton-bg)}.opencode-skeleton-title{height:16px;background:var(--oc-skeleton-gradient);background-size:200% 100%;animation:skeleton-loading 1.5s ease-in-out infinite;border-radius:4px;margin-bottom:8px;width:70%}.opencode-skeleton-meta{height:12px;background:var(--oc-skeleton-gradient);background-size:200% 100%;animation:skeleton-loading 1.5s ease-in-out infinite;border-radius:4px;width:50%}.opencode-session-empty{padding:32px 16px;text-align:center;color:var(--oc-text-placeholder);font-size:13px}@keyframes skeleton-loading{0%{background-position:200% 0}to{background-position:-200% 0}}.opencode-button{width:44px;height:44px;border-radius:50%;background:var(--oc-trigger-bg);border:none;cursor:pointer;box-shadow:var(--oc-trigger-shadow);transition:all .3s ease;display:flex;align-items:center;justify-content:center;color:#fff;padding:0;position:relative;overflow:visible}.opencode-button svg{transform:rotate(180deg);transition:transform .3s ease}.opencode-button:before{content:"";position:absolute;inset:-8px;border-radius:50%}.opencode-button:hover{transform:scale(1.1);box-shadow:var(--oc-trigger-shadow-hover);background:var(--oc-trigger-bg-hover)}.opencode-button.thinking{background:linear-gradient(135deg,#667eea,#764ba2,#667eea);background-size:200% 200%;animation:thinking-gradient 2s ease infinite,thinking-pulse 2s ease-in-out infinite;box-shadow:0 0 20px #667eea80,0 0 40px #764ba24d}.opencode-button.thinking svg{animation:none}.opencode-button.thinking:before{content:"";position:absolute;top:50%;left:50%;width:100%;height:100%;border-radius:50%;background:radial-gradient(circle,rgba(102,126,234,.7) 0%,rgba(118,75,162,.5) 50%,transparent 70%);transform:translate(-50%,-50%);animation:ripple-wave 2s ease-out infinite;pointer-events:none}.opencode-button.thinking:after{content:"";position:absolute;top:50%;left:50%;width:100%;height:100%;border-radius:50%;background:radial-gradient(circle,rgba(118,75,162,.6) 0%,rgba(102,126,234,.4) 50%,transparent 70%);transform:translate(-50%,-50%);animation:ripple-wave 2s ease-out infinite;animation-delay:1s;pointer-events:none}@keyframes thinking-gradient{0%{background-position:0% 50%}50%{background-position:100% 50%}to{background-position:0% 50%}}@keyframes ripple-wave{0%{transform:translate(-50%,-50%) scale(1);opacity:.8}to{transform:translate(-50%,-50%) scale(2.5);opacity:0}}@keyframes thinking-pulse{0%,to{transform:scale(1)}50%{transform:scale(.9)}}.opencode-button.loading{animation:pulse 1s infinite}@keyframes pulse{0%,to{opacity:1}50%{opacity:.5}}.opencode-chrome-warmup-error-details[data-v-6b3ae44a]{margin-top:8px;padding:12px;background:var(--oc-bg-tertiary);border-radius:6px;font-family:Monaco,Menlo,Ubuntu Mono,monospace;font-size:12px;color:var(--oc-text-secondary);max-height:120px;overflow-y:auto;word-break:break-word}.opencode-custom-loading{display:flex;flex-direction:column;align-items:center;justify-content:center;padding:20px}.opencode-loading-spinner{width:32px;height:32px;border:3px solid var(--oc-border-primary);border-top-color:var(--oc-primary);border-radius:50%;animation:opencode-spin .8s linear infinite}@keyframes opencode-spin{to{transform:rotate(360deg)}}.opencode-loading-text{margin-top:12px;color:var(--oc-text-secondary);font-size:14px}.opencode-chrome-warmup-failed{position:absolute;inset:0;background:var(--oc-bg-secondary);display:flex;flex-direction:column;align-items:center;justify-content:center;z-index:15}.opencode-chrome-warmup-failed-icon{color:var(--oc-warning, #f59e0b);margin-bottom:16px}.opencode-chrome-warmup-failed-title{color:var(--oc-text-primary);font-size:18px;font-weight:600;margin-bottom:8px}.opencode-chrome-warmup-failed-text{color:var(--oc-text-secondary);font-size:14px;margin-bottom:24px;text-align:left;max-width:400px;line-height:1.6;text-align:center}.opencode-chrome-warmup-failed-text p{margin:0 0 12px;font-weight:500;color:var(--oc-text-primary)}.opencode-chrome-warmup-steps{margin:0;padding-left:20px}.opencode-chrome-warmup-steps li{margin-bottom:8px;color:var(--oc-text-secondary);font-size:13px;line-height:1.5}.opencode-chrome-warmup-steps li:last-child{margin-bottom:0}.opencode-chrome-warmup-code{display:inline-block;background:var(--oc-bg-tertiary);color:var(--oc-primary);padding:2px 6px;border-radius:4px;font-family:Monaco,Menlo,Ubuntu Mono,monospace;font-size:12px;font-weight:500;word-break:break-all;margin:0 2px}.opencode-chrome-warmup-failed-actions{display:flex;gap:12px}.opencode-chrome-warmup-failed-btn{padding:10px 24px;border-radius:8px;border:none;font-size:14px;font-weight:500;cursor:pointer;transition:all .2s}.opencode-chrome-warmup-failed-btn.primary{background:var(--oc-primary);color:#fff;box-shadow:var(--oc-shadow-primary)}.opencode-chrome-warmup-failed-btn.primary:hover:not(:disabled){background:var(--oc-primary-hover);transform:translateY(-1px);box-shadow:var(--oc-shadow-primary-hover)}.opencode-chrome-warmup-failed-btn.primary:active:not(:disabled){transform:translateY(0)}.opencode-chrome-warmup-failed-btn.primary:disabled{opacity:.6;cursor:not-allowed}
|
package/lib/utils/system.d.ts
CHANGED
|
@@ -4,3 +4,4 @@ export declare function checkOpenCodeInstalled(): Promise<boolean>;
|
|
|
4
4
|
export declare function isPortAvailable(port: number, hostname?: string): Promise<boolean>;
|
|
5
5
|
export declare function findAvailablePort(startPort: number, hostname?: string, maxTries?: number): Promise<number>;
|
|
6
6
|
export declare function killOrphanOpenCodeProcesses(): Promise<number>;
|
|
7
|
+
export declare function findGitRoot(startDir: string, maxDepth?: number): string;
|
package/lib/utils/system.js
CHANGED
|
@@ -49,14 +49,17 @@ var system_exports = {};
|
|
|
49
49
|
__export(system_exports, {
|
|
50
50
|
checkOpenCodeInstalled: () => checkOpenCodeInstalled,
|
|
51
51
|
findAvailablePort: () => findAvailablePort,
|
|
52
|
+
findGitRoot: () => findGitRoot,
|
|
52
53
|
isPortAvailable: () => isPortAvailable,
|
|
53
54
|
killOrphanOpenCodeProcesses: () => killOrphanOpenCodeProcesses,
|
|
54
55
|
waitForServer: () => waitForServer
|
|
55
56
|
});
|
|
56
57
|
module.exports = __toCommonJS(system_exports);
|
|
57
58
|
var import_child_process = require("child_process");
|
|
59
|
+
var import_fs = __toESM(require("fs"));
|
|
58
60
|
var import_http = __toESM(require("http"));
|
|
59
61
|
var import_net = __toESM(require("net"));
|
|
62
|
+
var import_path = __toESM(require("path"));
|
|
60
63
|
var import_shared = require("@vite-plugin-opencode-assistant/shared");
|
|
61
64
|
var import_shared2 = require("@vite-plugin-opencode-assistant/shared");
|
|
62
65
|
const log = (0, import_shared2.createLogger)("Utils");
|
|
@@ -218,6 +221,31 @@ function killOrphanProcessesOnWindows(resolve, timer) {
|
|
|
218
221
|
resolve(0);
|
|
219
222
|
});
|
|
220
223
|
}
|
|
224
|
+
function findGitRoot(startDir, maxDepth = 10) {
|
|
225
|
+
const timer = log.timer("findGitRoot", { startDir, maxDepth });
|
|
226
|
+
let currentDir = startDir;
|
|
227
|
+
let depth = 0;
|
|
228
|
+
while (depth < maxDepth) {
|
|
229
|
+
const gitDir = import_path.default.join(currentDir, ".git");
|
|
230
|
+
try {
|
|
231
|
+
if (import_fs.default.existsSync(gitDir)) {
|
|
232
|
+
timer.end(`\u2713 Found git root at depth ${depth}: ${currentDir}`);
|
|
233
|
+
return currentDir;
|
|
234
|
+
}
|
|
235
|
+
} catch (err) {
|
|
236
|
+
log.debug(`Error checking .git directory at ${currentDir}`, { error: err.message });
|
|
237
|
+
}
|
|
238
|
+
const parentDir = import_path.default.dirname(currentDir);
|
|
239
|
+
if (parentDir === currentDir) {
|
|
240
|
+
log.debug("Reached filesystem root");
|
|
241
|
+
break;
|
|
242
|
+
}
|
|
243
|
+
currentDir = parentDir;
|
|
244
|
+
depth++;
|
|
245
|
+
}
|
|
246
|
+
timer.end(`\u274C No git root found after ${depth} levels, using start directory`);
|
|
247
|
+
return startDir;
|
|
248
|
+
}
|
|
221
249
|
function killOrphanProcessesOnUnix(resolve, timer) {
|
|
222
250
|
var _a;
|
|
223
251
|
log.debug("Using Unix method to find orphan processes");
|
|
@@ -273,6 +301,7 @@ function killOrphanProcessesOnUnix(resolve, timer) {
|
|
|
273
301
|
0 && (module.exports = {
|
|
274
302
|
checkOpenCodeInstalled,
|
|
275
303
|
findAvailablePort,
|
|
304
|
+
findGitRoot,
|
|
276
305
|
isPortAvailable,
|
|
277
306
|
killOrphanOpenCodeProcesses,
|
|
278
307
|
waitForServer
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "vite-plugin-opencode-assistant",
|
|
3
|
-
"version": "1.0.
|
|
3
|
+
"version": "1.0.16",
|
|
4
4
|
"description": "Embed OpenCode Web UI in your Vite dev server for real-time code modification and preview",
|
|
5
5
|
"type": "module",
|
|
6
6
|
"main": "lib/index.js",
|
|
@@ -36,9 +36,9 @@
|
|
|
36
36
|
"license": "MIT",
|
|
37
37
|
"dependencies": {
|
|
38
38
|
"unplugin-vue-inspector": "^2.4.0",
|
|
39
|
-
"@vite-plugin-opencode-assistant/
|
|
40
|
-
"@vite-plugin-opencode-assistant/
|
|
41
|
-
"@vite-plugin-opencode-assistant/shared": "1.0.
|
|
39
|
+
"@vite-plugin-opencode-assistant/opencode": "1.0.16",
|
|
40
|
+
"@vite-plugin-opencode-assistant/components": "1.0.16",
|
|
41
|
+
"@vite-plugin-opencode-assistant/shared": "1.0.16"
|
|
42
42
|
},
|
|
43
43
|
"peerDependencies": {
|
|
44
44
|
"vite": ">=4.0.0"
|