Files
insightflow/chrome-extension/options.js
OpenClaw Bot 797ca58e8e Phase 7 Task 7: 插件与集成系统
- 创建 plugin_manager.py 模块
  - PluginManager: 插件管理主类
  - ChromeExtensionHandler: Chrome 插件处理
  - BotHandler: 飞书/钉钉/Slack 机器人处理
  - WebhookIntegration: Zapier/Make Webhook 集成
  - WebDAVSync: WebDAV 同步管理

- 创建完整的 Chrome 扩展代码
  - manifest.json, background.js, content.js, content.css
  - popup.html/js: 弹出窗口界面
  - options.html/js: 设置页面
  - 支持网页剪藏、选中文本保存、项目选择

- 更新 schema.sql 添加插件相关数据库表
  - plugins: 插件配置表
  - bot_sessions: 机器人会话表
  - webhook_endpoints: Webhook 端点表
  - webdav_syncs: WebDAV 同步配置表
  - plugin_activity_logs: 插件活动日志表

- 更新 main.py 添加插件相关 API 端点
  - GET/POST /api/v1/plugins - 插件管理
  - POST /api/v1/plugins/chrome/clip - Chrome 插件保存网页
  - POST /api/v1/bots/webhook/{platform} - 接收机器人消息
  - GET /api/v1/bots/sessions - 机器人会话列表
  - POST /api/v1/webhook-endpoints - 创建 Webhook 端点
  - POST /webhook/{type}/{token} - 接收外部 Webhook
  - POST /api/v1/webdav-syncs - WebDAV 同步配置
  - POST /api/v1/webdav-syncs/{id}/test - 测试 WebDAV 连接
  - POST /api/v1/webdav-syncs/{id}/sync - 触发 WebDAV 同步

- 更新 requirements.txt 添加插件依赖
  - beautifulsoup4: HTML 解析
  - webdavclient3: WebDAV 客户端

- 更新 STATUS.md 和 README.md 开发进度
2026-02-23 12:09:15 +08:00

175 lines
5.1 KiB
JavaScript
Raw Blame History

This file contains ambiguous Unicode characters
This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.
// InsightFlow Chrome Extension - Options Script
document.addEventListener('DOMContentLoaded', () => {
const serverUrlInput = document.getElementById('serverUrl');
const apiKeyInput = document.getElementById('apiKey');
const defaultProjectSelect = document.getElementById('defaultProject');
const testBtn = document.getElementById('testBtn');
const testResult = document.getElementById('testResult');
const saveBtn = document.getElementById('saveBtn');
const resetBtn = document.getElementById('resetBtn');
const openConsole = document.getElementById('openConsole');
const helpLink = document.getElementById('helpLink');
// 加载配置
loadConfig();
// 测试连接
testBtn.addEventListener('click', async () => {
testBtn.disabled = true;
testBtn.textContent = '测试中...';
testResult.className = '';
testResult.style.display = 'none';
const serverUrl = serverUrlInput.value.trim();
const apiKey = apiKeyInput.value.trim();
if (!serverUrl || !apiKey) {
showTestResult('请填写服务器地址和 API Key', 'error');
testBtn.disabled = false;
testBtn.textContent = '测试连接';
return;
}
try {
const response = await fetch(`${serverUrl}/api/v1/projects`, {
headers: { 'X-API-Key': apiKey }
});
if (response.ok) {
const data = await response.json();
showTestResult(`连接成功!找到 ${data.projects?.length || 0} 个项目`, 'success');
// 更新项目列表
updateProjectList(data.projects || []);
} else if (response.status === 401) {
showTestResult('API Key 无效,请检查', 'error');
} else {
showTestResult(`连接失败: HTTP ${response.status}`, 'error');
}
} catch (error) {
showTestResult(`连接错误: ${error.message}`, 'error');
}
testBtn.disabled = false;
testBtn.textContent = '测试连接';
});
// 保存设置
saveBtn.addEventListener('click', async () => {
const config = {
serverUrl: serverUrlInput.value.trim(),
apiKey: apiKeyInput.value.trim(),
defaultProjectId: defaultProjectSelect.value
};
if (!config.serverUrl) {
alert('请填写服务器地址');
return;
}
await chrome.storage.sync.set({ insightflowConfig: config });
// 显示保存成功
saveBtn.textContent = '已保存 ✓';
saveBtn.classList.add('btn-success');
setTimeout(() => {
saveBtn.textContent = '保存设置';
saveBtn.classList.remove('btn-success');
}, 2000);
});
// 重置设置
resetBtn.addEventListener('click', () => {
if (confirm('确定要重置所有设置吗?')) {
const defaultConfig = {
serverUrl: 'http://122.51.127.111:18000',
apiKey: '',
defaultProjectId: ''
};
chrome.storage.sync.set({ insightflowConfig: defaultConfig }, () => {
loadConfig();
showTestResult('设置已重置', 'success');
});
}
});
// 打开控制台
openConsole.addEventListener('click', (e) => {
e.preventDefault();
const serverUrl = serverUrlInput.value.trim();
if (serverUrl) {
chrome.tabs.create({ url: serverUrl });
}
});
// 帮助链接
helpLink.addEventListener('click', (e) => {
e.preventDefault();
const serverUrl = serverUrlInput.value.trim();
if (serverUrl) {
chrome.tabs.create({ url: `${serverUrl}/docs` });
}
});
// 加载配置
async function loadConfig() {
const result = await chrome.storage.sync.get(['insightflowConfig']);
const config = result.insightflowConfig || {
serverUrl: 'http://122.51.127.111:18000',
apiKey: '',
defaultProjectId: ''
};
serverUrlInput.value = config.serverUrl;
apiKeyInput.value = config.apiKey;
// 如果有 API Key加载项目列表
if (config.apiKey) {
loadProjects(config);
}
}
// 加载项目列表
async function loadProjects(config) {
try {
const response = await fetch(`${config.serverUrl}/api/v1/projects`, {
headers: { 'X-API-Key': config.apiKey }
});
if (response.ok) {
const data = await response.json();
updateProjectList(data.projects || [], config.defaultProjectId);
}
} catch (error) {
console.error('Failed to load projects:', error);
}
}
// 更新项目列表
function updateProjectList(projects, selectedId = '') {
let html = '<option value="">不设置默认项目</option>';
projects.forEach(project => {
const selected = project.id === selectedId ? 'selected' : '';
html += `<option value="${project.id}" ${selected}>${escapeHtml(project.name)}</option>`;
});
defaultProjectSelect.innerHTML = html;
}
// 显示测试结果
function showTestResult(message, type) {
testResult.textContent = message;
testResult.className = type;
}
// HTML 转义
function escapeHtml(text) {
const div = document.createElement('div');
div.textContent = text;
return div.innerHTML;
}
});