// 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.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; } }