// eslint-disable-next-line @typescript-eslint/no-unused-vars async function getAIResponse(contents) { let response; let thinking = document.createElement('div'); thinking.className = 'message-bubble model'; thinking.id = 'thinking'; thinking.innerHTML = '
思考中...
'; document.querySelector('.chat-body').appendChild(thinking); document.querySelector('.chat-body').scrollTop = document.querySelector('.chat-body').scrollHeight; if (MODEL_GROUP === "Gemini") { response = await getGeminiResponse(contents); } else if (MODEL_GROUP === "DeepSeek") { response = await getDeepSeekResponse(contents); } else if (MODEL_GROUP === "QWen") { response = await getQWenResponse(contents); } else if (MODEL_GROUP === "Custom") { response = await getCustomResponse(contents); } else { response = { err_code: 1, error: "模型不被支持" }; } console.log(response); if (response.err_code === 0) { document.getElementById('thinking').remove(); let content = response.response; content = content.replace(/\u003cthink\u003e/, '
').replace(/\u003c\/think\u003e/, '
'); createBubble('model', content, undefined); if (MODEL_GROUP === "Gemini") { chatHistory.push({ role: 'model', parts: [{ text: content }] }); } else if (MODEL_GROUP === "DeepSeek" || MODEL_GROUP === "Qwen" || MODEL_GROUP === "Custom") { chatHistory.push({ role: 'assistant', content: content }); } if (CHAT_TITLE === "新的对话") { let temp = chatHistory.slice(); if (MODEL_GROUP === "Gemini") { temp.push({ role: 'user', parts: [{ text: "用几个字总结这个对话,将其作为对话的标题" }] }); getGeminiResponse(temp).then(response => { if (response.err_code === 0) { const summary = response.response; CHAT_TITLE = summary.replace('*', ''); document.getElementById('title').innerText = summary; } }); } else if (MODEL_GROUP === "DeepSeek") { temp.push({ role: 'user', content: "用几个字总结这个对话,将其作为对话的标题,除此以外不要有任何其他文字" }); getDeepSeekResponse(temp).then(response => { if (response.err_code === 0) { const summary = response.response; CHAT_TITLE = summary.replace(/\u003cthink\u003e.*\u003c\/think\u003e/ms, '').replace('*', ''); document.getElementById('title').innerText = CHAT_TITLE; } }); } else if (MODEL_GROUP === "QWen") { temp.push({ role: 'user', content: "用几个字总结这个对话,将其作为对话的标题,除此以外不要有任何其他文字" }); getQWenResponse(temp).then(response => { if (response.err_code === 0) { const summary = response.response; CHAT_TITLE = summary.replace(/\u003cthink\u003e.*\u003c\/think\u003e/ms, '').replace('*', ''); document.getElementById('title').innerText = CHAT_TITLE; } }); } else if (MODEL_GROUP === "Custom") { temp.push({ role: 'user', content: "用几个字总结这个对话,将其作为对话的标题,除此以外不要有任何其他文字" }); getCustomResponse(temp).then((response) => { if (response.err_code === 0) { const summary = response.response; CHAT_TITLE = summary.replace(/.*<\/think>/ms, '').replaceAll('*', '').replaceAll('\n', ''); document.getElementById('title').innerText = CHAT_TITLE; } }); } } } else { document.getElementById('thinking').remove(); createBubble('model', response.error, "异常"); } } // eslint-disable-next-line @typescript-eslint/no-unused-vars async function setAICache(cache) { if (MODEL_GROUP === "Gemini" && MODEL_NAME !== "gemini-2.0-flash-001") { return await setGeminiCache(cache); } else if (MODEL_GROUP === "QWen" || MODEL_GROUP === "DeepSeek" || MODEL_GROUP === "Custom") { return await setOtherCache(decodeURIComponent(atob(cache))); } else if (MODEL_GROUP === "Gemini" && MODEL_NAME === "gemini-2.0-flash-001") { return await setOtherCache(decodeURIComponent(atob(cache))); } else { createBubble('model', "抱歉,该模型不支持缓存长上下文", "异常"); } } async function getGeminiResponse(contents) { const API = GOOGLE_SETTINGS.HOST + "/v1beta/models/" + MODEL_NAME + ":generateContent?key=" + GOOGLE_SETTINGS.API_KEY; console.log(chatHistory); const data = { contents: [ contents ], safetySettings: [ [ { "category": "HARM_CATEGORY_HARASSMENT", "threshold": "BLOCK_ONLY_HIGH" }, { "category": "HARM_CATEGORY_HATE_SPEECH", "threshold": "BLOCK_ONLY_HIGH" }, { "category": "HARM_CATEGORY_SEXUALLY_EXPLICIT", "threshold": "BLOCK_ONLY_HIGH" }, { "category": "HARM_CATEGORY_DANGEROUS_CONTENT", "threshold": "BLOCK_ONLY_HIGH" } ] ] }; if (CACHE_NAME !== null) { data.cachedContent = CACHE_NAME; } else { data.systemInstruction = { parts: [{ "text": "You are an electronic engineer. Use Chinese to answer." }] } } try { const response = await eda.sys_ClientUrl.request(API, "POST", JSON.stringify(data), { headers: GOOGLE_SETTINGS.HEADER }); if (response.status !== 200) { throw new Error(response.statusText); } const result = await response.json(); return { response: result.candidates[0].content.parts[0].text, err_code: 0 }; } catch (error) { return { error: error.message || "请求失败", err_code: 1 }; } } async function getDeepSeekResponse(contents) { const API = DS_SETTINGS.HOST + "/chat/completions"; let HEADERS = DS_SETTINGS.HEADER; HEADERS['Authorization'] = 'Bearer ' + DS_SETTINGS.API_KEY; const data = { model: MODEL_NAME, messages: contents, stream: false } try { const response = await eda.sys_ClientUrl.request(API, "POST", JSON.stringify(data), { headers: HEADERS }); if (response.status !== 200) { throw new Error(response.statusText); } const result = await response.json(); return { response: result.choices[0].message.content, err_code: 0 }; } catch (error) { return { error: error.message || "请求失败", err_code: 1 }; } } async function getQWenResponse(contents) { const API = QWEN_SETTINGS.HOST + "/compatible-mode/v1/chat/completions"; let HEADERS = QWEN_SETTINGS.HEADER; HEADERS['Authorization'] = 'Bearer ' + QWEN_SETTINGS.API_KEY; const data = { model: MODEL_NAME, messages: contents, } try { const response = await eda.sys_ClientUrl.request(API, "POST", JSON.stringify(data), { headers: HEADERS }); if (response.status !== 200) { throw new Error(response.statusText); } const result = await response.json(); return { response: result.choices[0].message.content, err_code: 0 }; } catch (error) { return { error: error.message || "请求失败", err_code: 1 }; } } async function getCustomResponse(contents) { const API = CUSTOM_SETTINGS.HOST + CUSTOM_SETTINGS.ENDPOINT; const data = { model: CUSTOM_SETTINGS.MODEL, messages: contents, stream: false } try { const response = await eda.sys_ClientUrl.request(API, "POST", JSON.stringify(data), { headers: CUSTOM_SETTINGS.HEADER }); if (response.status !== 200) { throw new Error(response.statusText); } const result = await response.json(); return { response: result.choices[0].message.content, err_code: 0 }; } catch (error) { return { error: error.message || "请求失败", err_code: 1 }; } } async function setGeminiCache(cache) { let API = GOOGLE_SETTINGS.HOST + "/v1beta/cachedContents?key=" + GOOGLE_SETTINGS.API_KEY; const data = { model: "models/" + MODEL_NAME, contents: [ { role: 'user', parts: [{ inline_data: { mime_type: "text/plain", data: cache} }, {text: "Here's a netlist file describing the circuit diagram, and I'm going to ask you questions about it."}] } ], systemInstruction: { parts: [ { "text": "You are an electronic engineer. The text describes a netlist of circuit diagrams. When asked a question about a component, it is displayed if there is a URL in the netlist, otherwise it is not. Use Chinese to answer." } ] }, } let thinking = document.createElement('div'); thinking.className = 'message-bubble model'; thinking.id = 'thinking'; thinking.innerHTML = '
解读中...
'; document.querySelector('.chat-body').appendChild(thinking); document.querySelector('.chat-body').scrollTop = document.querySelector('.chat-body').scrollHeight; eda.sys_ClientUrl.request(API, "POST", JSON.stringify(data), { headers: GOOGLE_SETTINGS.HEADER }).then(response => { if (response.status !== 200) { throw new Error(response.statusText); } return response.json(); }).then(result => { document.getElementById('thinking').remove(); createBubble('model', "好的,接下来你可以围绕这张原理图向我提问", undefined); CACHE_NAME = result.name; }).catch(error => { document.getElementById('thinking').remove(); createBubble('model', error.toString(), "异常"); }); } async function setOtherCache(cache) { chatHistory.push({ role: 'user', content: '这是一个电路图的网表,后续回答依照这个网表回答:' + cache }); chatHistory.push({ role: 'assistant', content: '好的,我了解了'}); } // eslint-disable-next-line @typescript-eslint/no-unused-vars function getNetlist() { let netlist = ''; eda.sch_Netlist.getNetlist().then(data => { netlist = data[0].toString(); const cache = btoa(encodeURIComponent(netlist)); // 获取 cache 的字节数 let size = cache.length; let unit = 'B'; if (size > 1024) { size /= 1024; unit = 'KB'; } if (size > 1024) { size /= 1024; unit = 'MB'; } let fileBubble = document.createElement('div'); fileBubble.classList.add('file-bubble'); fileBubble.innerHTML = `File Icon
网表文件
${size} ${unit}
`; document.querySelector('.chat-body').appendChild(fileBubble); setAICache(cache); }) } // eslint-disable-next-line @typescript-eslint/no-unused-vars async function setAIFile() { if (MODEL_GROUP !== "Gemini") { eda.sys_Message.showToastMessage('当前模型尚未支持该功能', 'warn'); return; } const file = await eda.sys_FileSystem.openReadFileDialog("pdf"); const reader = new FileReader(); reader.onload = function (event) { const base64 = event.target.result; const title = file.name; // 判断是不是 pdf 文件 if (file.type !== 'application/pdf') { eda.sys_Message.showToastMessage('仅可上传 PDF 格式的数据手册', 'warn'); return; } FILE_BASE64 = base64.replaceAll('data:application/pdf;base64,', ''); // 如果 chat-body 中最后一个元素是 file-bubble,替换之 const chatBody = document.querySelector('.chat-body'); const lastElement = chatBody.lastElementChild; if (lastElement && lastElement.classList.contains('file-bubble')) { lastElement.remove(); } // 创建 file bubble let size = FILE_BASE64.length; let unit = 'B'; if (size > 1024) { size /= 1024; unit = 'KB'; } if (size > 1024) { size /= 1024; unit = 'MB'; } let fileBubble = document.createElement('div'); fileBubble.classList.add('file-bubble'); fileBubble.innerHTML = `File Icon
${title}
${size} ${unit}
`; document.querySelector('.chat-body').appendChild(fileBubble); }; reader.readAsDataURL(file); } /* 废弃 */ // eslint-disable-next-line @typescript-eslint/no-unused-vars async function getDatasheetUrl() { try { const selectedPrimitives = await eda.sch_SelectControl.getAllSelectedPrimitives_PrimitiveId(); const primitive = await eda.sch_PrimitiveComponent.get(selectedPrimitives[0]); let url = primitive[0].otherProperty.Datasheet; // atta.szlcsc.com -> atta-szlcsc.mirror.soraharu.com url = url.replace('atta.szlcsc.com', 'atta-szlcsc.mirror.soraharu.com'); console.log(url); return url; } catch (error) { console.error(error); return null; } } // eslint-disable-next-line @typescript-eslint/no-unused-vars async function getDatasheetData(url) { console.log(url); try { const response = await eda.sys_ClientUrl.request(url); const data = await response.text(); console.log(data); return data; } catch (error) { console.error(error); return null; } }