智能问答学习助手
// ==UserScript==
// @name 智能问答学习助手
// @namespace http://tampermonkey.net/
// @version 1.2
// @description 考试学习必备,在任意网页嵌入一个聊天交互界面,用户可以通过问问题然后得到回复,支持拖拽和悬浮小球展开,支持Tab切换和API Key加密显示
// @match *://*/*
// @grant GM_xmlhttpRequest
// @grant GM_setValue
// @grant GM_getValue
// @connect open.bigmodel.cn
// @author 不会使用使用过程遇到问题请联系q:2430486030 如果获取apikey:https://sa6z03i7sec.feishu.cn/docx/FemzdZ1mnosEgVxRVNocwKEnneY?from=from_copylink
// ==/UserScript==
(function() {
'use strict';
const API_URL = 'https://open.bigmodel.cn/api/paas/v4/chat/completions';
let apiKey = GM_getValue('apiKey', ''); // 从油猴存储获取 API Key
// 创建悬浮小球
function createFloatingBall() {
const ball = document.createElement('div');
ball.id = 'floating-ball';
ball.style.position = 'fixed';
ball.style.bottom = '20px';
ball.style.right = '20px';
ball.style.width = '50px';
ball.style.height = '50px';
ball.style.backgroundColor = '#007bff';
ball.style.borderRadius = '50%';
ball.style.boxShadow = '0 0 10px rgba(0, 0, 0, 0.1)';
ball.style.zIndex = '2147483647';
ball.style.cursor = 'pointer';
ball.style.display = 'flex';
ball.style.alignItems = 'center';
ball.style.justifyContent = 'center';
ball.style.color = 'white';
ball.style.fontSize = '24px';
ball.textContent = '+';
document.body.appendChild(ball);
makeDraggable(ball);
ball.onclick = toggleChatContainer;
return ball;
}
// 创建聊天界面容器
function createChatContainer() {
const container = document.createElement('div');
container.id = 'chat-container';
container.style.position = 'fixed';
container.style.bottom = '80px';
container.style.right = '20px';
container.style.width = '300px';
container.style.height = '400px';
container.style.backgroundColor = 'white';
container.style.border = '1px solid #ccc';
container.style.borderRadius = '5px';
container.style.boxShadow = '0 0 10px rgba(0, 0, 0, 0.1)';
container.style.zIndex = '2147483647';
container.style.display = 'none';
container.style.flexDirection = 'column';
container.style.overflow = 'hidden';
document.body.appendChild(container);
return container;
}
// 切换聊天界面显示状态
function toggleChatContainer() {
const container = document.getElementById('chat-container');
if (container.style.display === 'none') {
container.style.display = 'flex';
} else {
container.style.display = 'none';
}
}
// 创建聊天界面
function createChatInterface(container) {
const tabHeader = document.createElement('div');
tabHeader.style.display = 'flex';
tabHeader.style.borderBottom = '1px solid #ccc';
const chatTab = document.createElement('div');
chatTab.textContent = '聊天';
chatTab.style.flex = '1';
chatTab.style.padding = '10px';
chatTab.style.cursor = 'pointer';
chatTab.style.textAlign = 'center';
chatTab.style.backgroundColor = '#007bff';
chatTab.style.color = 'white';
chatTab.onclick = () => switchTab('chat');
tabHeader.appendChild(chatTab);
const aboutTab = document.createElement('div');
aboutTab.textContent = '关于';
aboutTab.style.flex = '1';
aboutTab.style.padding = '10px';
aboutTab.style.cursor = 'pointer';
aboutTab.style.textAlign = 'center';
aboutTab.style.backgroundColor = '#f1f1f1';
aboutTab.style.color = 'black';
aboutTab.onclick = () => switchTab('about');
tabHeader.appendChild(aboutTab);
container.appendChild(tabHeader);
const chatContent = document.createElement('div');
chatContent.id = 'chat-content';
chatContent.style.flex = '1';
chatContent.style.padding = '10px';
chatContent.style.overflowY = 'auto';
chatContent.style.backgroundColor = '#f9f9f9';
container.appendChild(chatContent);
const aboutContent = document.createElement('div');
aboutContent.id = 'about-content';
aboutContent.style.flex = '1';
aboutContent.style.padding = '10px';
aboutContent.style.overflowY = 'auto';
aboutContent.style.backgroundColor = '#f9f9f9';
aboutContent.style.display = 'none';
container.appendChild(aboutContent);
createChatContent(chatContent);
createAboutContent(aboutContent);
}
// 切换Tab
function switchTab(tab) {
const chatTab = document.querySelector('#chat-container div:nth-child(1) div:nth-child(1)');
const aboutTab = document.querySelector('#chat-container div:nth-child(1) div:nth-child(2)');
const chatContent = document.getElementById('chat-content');
const aboutContent = document.getElementById('about-content');
if (tab === 'chat') {
chatTab.style.backgroundColor = '#007bff';
chatTab.style.color = 'white';
aboutTab.style.backgroundColor = '#f1f1f1';
aboutTab.style.color = 'black';
chatContent.style.display = 'block';
aboutContent.style.display = 'none';
} else {
chatTab.style.backgroundColor = '#f1f1f1';
chatTab.style.color = 'black';
aboutTab.style.backgroundColor = '#007bff';
aboutTab.style.color = 'white';
chatContent.style.display = 'none';
aboutContent.style.display = 'block';
}
}
// 创建聊天内容
function createChatContent(container) {
const chatBody = document.createElement('div');
chatBody.id = 'chat-body';
chatBody.style.flex = '1';
chatBody.style.padding = '10px';
chatBody.style.overflowY = 'auto';
chatBody.style.backgroundColor = '#f9f9f9';
container.appendChild(chatBody);
const chatFooter = document.createElement('div');
chatFooter.style.display = 'flex';
chatFooter.style.padding = '10px';
chatFooter.style.borderTop = '1px solid #ccc';
container.appendChild(chatFooter);
const input = document.createElement('input');
input.type = 'text';
input.id = 'chat-input';
input.style.flex = '1';
input.style.padding = '10px';
input.style.border = '1px solid #ccc';
input.style.borderRadius = '5px';
input.addEventListener('keydown', function(event) {
if (event.key === 'Enter') {
sendMessage();
}
});
chatFooter.appendChild(input);
const sendButton = document.createElement('button');
sendButton.textContent = '发送';
sendButton.style.marginLeft = '10px';
sendButton.style.padding = '10px';
sendButton.style.border = 'none';
sendButton.style.borderRadius = '5px';
sendButton.style.backgroundColor = '#007bff';
sendButton.style.color = 'white';
sendButton.style.cursor = 'pointer';
sendButton.onclick = sendMessage;
chatFooter.appendChild(sendButton);
}
// 创建关于内容
function createAboutContent(container) {
const helpText = document.createElement('div');
helpText.textContent = '帮助文档:在聊天界面中输入问题并点击发送按钮,您会看到回复显示在聊天界面中。';
helpText.style.marginBottom = '10px';
container.appendChild(helpText);
const apiKeyContainer = document.createElement('div');
apiKeyContainer.style.display = 'flex';
apiKeyContainer.style.alignItems = 'center';
const apiKeyInput = document.createElement('input');
apiKeyInput.type = 'text';
apiKeyInput.id = 'api-key-input';
apiKeyInput.placeholder = '输入API Key';
apiKeyInput.style.flex = '1';
apiKeyInput.style.padding = '10px';
apiKeyInput.style.border = '1px solid #ccc';
apiKeyInput.style.borderRadius = '5px';
apiKeyInput.value = apiKey;
apiKeyInput.onchange = function() {
apiKey = apiKeyInput.value;
GM_setValue('apiKey', apiKey); // 保存 API Key 到油猴存储
apiKeyInput.type = 'password'; // 将输入框类型改为密码
};
apiKeyContainer.appendChild(apiKeyInput);
const toggleButton = document.createElement('button');
toggleButton.textContent = '👁️';
toggleButton.style.marginLeft = '10px';
toggleButton.style.padding = '10px';
toggleButton.style.border = 'none';
toggleButton.style.borderRadius = '5px';
toggleButton.style.backgroundColor = '#007bff';
toggleButton.style.color = 'white';
toggleButton.style.cursor = 'pointer';
toggleButton.onclick = function() {
if (apiKeyInput.type === 'password') {
apiKeyInput.type = 'text';
} else {
apiKeyInput.type = 'password';
}
};
apiKeyContainer.appendChild(toggleButton);
container.appendChild(apiKeyContainer);
if (apiKey) {
apiKeyInput.type = 'password'; // 如果已有 API Key,将输入框类型改为密码
}
// const contactImage = document.createElement('img');
// contactImage.src = '';
// contactImage.alt = '联系方式';
// contactImage.style.width = '100%';
// contactImage.style.marginTop = '10px';
// container.appendChild(contactImage);
}
// 发送消息
function sendMessage() {
const input = document.getElementById('chat-input');
const message = input.value.trim();
if (message) {
appendMessage('user', message);
input.value = '';
getReply(message);
}
}
// 获取回复
function getReply(message) {
const chatBody = document.getElementById('chat-body');
const loadingMessage = appendMessage('bot', '正在思考...');
GM_xmlhttpRequest({
method: 'POST',
url: API_URL,
headers: {
'Authorization': `Bearer ${apiKey}`,
'Content-Type': 'application/json'
},
data: JSON.stringify({
model: 'glm-4',
messages: [
{
role: 'user',
content: message
}
]
}),
onload: function(response) {
console.log('API response:', response.responseText); // 调试信息
const data = JSON.parse(response.responseText);
if (data && data.choices && data.choices.length > 0) {
loadingMessage.textContent = data.choices[0].message.content;
} else {
loadingMessage.textContent = '抱歉,我无法理解您的问题。';
}
},
onerror: function() {
loadingMessage.textContent = '请求失败,请稍后再试。';
}
});
}
// 添加消息到聊天界面
function appendMessage(role, content) {
const chatBody = document.getElementById('chat-body');
const message = document.createElement('div');
message.style.marginBottom = '10px';
message.style.padding = '10px';
message.style.borderRadius = '5px';
message.style.backgroundColor = role === 'user' ? '#007bff' : '#f1f1f1';
message.style.color = role === 'user' ? 'white' : 'black';
message.style.alignSelf = role === 'user' ? 'flex-end' : 'flex-start';
message.textContent = content;
chatBody.appendChild(message);
chatBody.scrollTop = chatBody.scrollHeight;
return message;
}
// 添加拖动功能
function makeDraggable(element) {
let pos1 = 0, pos2 = 0, pos3 = 0, pos4 = 0;
element.onmousedown = dragMouseDown;
function dragMouseDown(e) {
e = e || window.event;
e.preventDefault();
pos3 = e.clientX;
pos4 = e.clientY;
document.onmouseup = closeDragElement;
document.onmousemove = elementDrag;
}
function elementDrag(e) {
e = e || window.event;
e.preventDefault();
pos1 = pos3 - e.clientX;
pos2 = pos4 - e.clientY;
pos3 = e.clientX;
pos4 = e.clientY;
element.style.top = (element.offsetTop - pos2) + 'px';
element.style.left = (element.offsetLeft - pos1) + 'px';
}
function closeDragElement() {
document.onmouseup = null;
document.onmousemove = null;
}
}
// 初始化悬浮小球和聊天界面
const floatingBall = createFloatingBall();
const chatContainer = createChatContainer();
createChatInterface(chatContainer);
// 监听鼠标选中事件
document.addEventListener('mouseup', function() {
const selectedText = window.getSelection().toString().trim();
if (selectedText) {
const input = document.getElementById('chat-input');
input.value = selectedText;
}
});
})();