// ==UserScript==
// @name 搜题小帮手|超全题库,方便快捷,一步到位
// @version 0.7
// @description 搜题就用小帮手,支持复制、图片搜题,题库超全!
// @author soti
// @match *://*/*
// @require https://code.jquery.com/jquery-3.7.1.min.js
// @require https://cdnjs.cloudflare.com/ajax/libs/vue/2.7.16/vue.min.js
// @require https://cdnjs.cloudflare.com/ajax/libs/axios/0.21.1/axios.min.js
// @license GUN
// @grant GM_info
// @grant GM_xmlhttpRequest
// @grant GM_getResourceText
// @grant GM_addStyle
// @connect studypro.club
// @grant GM_setValue
// @grant GM_getValue
// @antifeature payment 脚本题目搜索,需对接第三方接口付费
// @namespace https://greasyfork.org/users/736514
// @downloadURL https://update.greasyfork.icu/scripts/534185/%E6%90%9C%E9%A2%98%E5%B0%8F%E5%B8%AE%E6%89%8B%EF%BD%9C%E8%B6%85%E5%85%A8%E9%A2%98%E5%BA%93%EF%BC%8C%E6%96%B9%E4%BE%BF%E5%BF%AB%E6%8D%B7%EF%BC%8C%E4%B8%80%E6%AD%A5%E5%88%B0%E4%BD%8D.user.js
// @updateURL https://update.greasyfork.icu/scripts/534185/%E6%90%9C%E9%A2%98%E5%B0%8F%E5%B8%AE%E6%89%8B%EF%BD%9C%E8%B6%85%E5%85%A8%E9%A2%98%E5%BA%93%EF%BC%8C%E6%96%B9%E4%BE%BF%E5%BF%AB%E6%8D%B7%EF%BC%8C%E4%B8%80%E6%AD%A5%E5%88%B0%E4%BD%8D.meta.js
// ==/UserScript==
$(document).ready(function() {
if (window.self !== window.top) {
return;
}
// 添加CSS作用域前缀函数
function addCSSScope(css, scope = '#tikuHelper') {
// 匹配CSS规则
return css.replace(
/([^\r\n,{}]+)(,(?=[^}]*{)|\s*{)/g,
function(match, selector, delimiter) {
// 检查选择器是否已经有作用域或是全局选择器
if (selector.indexOf(scope) !== -1 ||
selector.trim().startsWith('body >') ||
selector.trim().startsWith('@keyframes')) {
return match;
}
// 添加作用域
return scope + ' ' + selector + delimiter;
}
);
}
// 用户数据管理
const UserDataManager = {
_subscribers: { phone: [] }, // Observer pattern: Store subscribers
// 默认用户数据
defaultUserData: {
remainCount: 0, // 剩余使用次数
totalUsed: 0, // 总共已使用次数
lastResetTime: 0, // 上次重置时间(时间戳)
phone: "", // 用户手机号
isPaid: false, // 是否为付费用户
autoClipboardSearch: false // 自动剪贴板搜索开关,默认关闭
},
// 获取用户数据
getUserData() {
const userData = GM_getValue("userData");
if (!userData) {
return this.defaultUserData;
}
try {
return JSON.parse(userData);
} catch (e) {
return this.defaultUserData;
}
},
// 保存用户数据
saveUserData(userData) {
GM_setValue("userData", JSON.stringify(userData));
},
// 获取手机号
getPhone() {
return this.getUserData().phone || "";
},
// 保存手机号
savePhone(phone) {
const userData = this.getUserData();
const oldPhone = userData.phone;
if (oldPhone !== phone) { // Only notify if phone actually changed
userData.phone = phone;
this.saveUserData(userData);
// Notify subscribers
this._notifySubscribers("phone", phone);
}
},
// Observer: Subscribe to phone changes
subscribePhoneChange(callback) {
if (typeof callback === 'function') {
this._subscribers.phone.push(callback);
}
},
// Observer: Notify subscribers
_notifySubscribers(key, value) {
if (this._subscribers[key]) {
this._subscribers[key].forEach(callback => callback(value));
}
},
// 获取剩余次数
getRemainingCount() {
return this.getUserData().remainCount;
},
// 设置剩余次数
setRemainingCount(count) {
const userData = this.getUserData();
userData.remainCount = count;
this.saveUserData(userData);
return count;
},
// 检查剩余次数是否足够
hasRemainingCount() {
return this.getRemainingCount() > 0;
},
// 减少使用次数
decreaseRemainingCount() {
const userData = this.getUserData();
userData.remainCount = Math.max(0, userData.remainCount - 1);
userData.totalUsed += 1;
this.saveUserData(userData);
return userData.remainCount;
},
// 获取自动剪贴板搜索设置
getAutoClipboardSearch() {
return this.getUserData().autoClipboardSearch;
},
// 设置自动剪贴板搜索
setAutoClipboardSearch(enabled) {
const userData = this.getUserData();
userData.autoClipboardSearch = enabled;
this.saveUserData(userData);
return enabled;
},
// 查询剩余次数
async queryRemainingCount(phone) {
if (!phone || phone.trim() === '') {
return 0;
}
return new Promise((resolve, reject) => {
GM_xmlhttpRequest({
method: 'POST',
url: 'https://so.studypro.club/api/getphone',
headers: {
'Content-Type': 'application/x-www-form-urlencoded'
},
data: `phone=${encodeURIComponent(phone)}`,
responseType: "json",
onload: function(r) {
try {
// 直接使用返回的数字
const result = r.response;
// 确保结果是数字类型,如果不是则转换,如果转换失败则默认为0
const count = typeof result === 'number' ? result :
(parseInt(result) || 0);
// 更新本地存储
UserDataManager.setRemainingCount(count);
resolve(count);
} catch (err) {
console.error('解析剩余次数失败:', err);
resolve(0);
}
},
onerror: function(err) {
console.error('查询剩余次数失败:', err);
resolve(0);
}
});
});
}
};
// 购买功能管理
const PurchaseManager = {
// 购买页面URL
purchaseUrl: 'https://so.studypro.club',
// 价格配置
priceConfig: {
"30": 100,
"800": 500,
"3000": 1000,
"1000000": 30000
},
// 显示购买提示弹窗
showPurchaseDialog() {
const remainCount = UserDataManager.getRemainingCount();
const phone = UserDataManager.getPhone();
// 创建弹窗元素
const dialogHtml = `
您的剩余使用次数为 ${remainCount} 次
购买更多次数以继续使用搜题功能
`;
// 添加到页面
$('body').append(dialogHtml);
// 绑定事件
$('.purchase-dialog-close, .cancel-btn').on('click', function() {
// 添加淡出动画
$('.purchase-dialog-content').css({
'animation': 'slideOutRight 0.3s ease forwards'
});
// 动画结束后移除元素
setTimeout(() => {
$('.purchase-dialog').remove();
}, 300);
});
$('.purchase-btn').on('click', function() {
const redirectUrl = `${PurchaseManager.purchaseUrl}?phone=${encodeURIComponent(phone || '')}`;
//window.open(redirectUrl, '_blank');
// 添加淡出动画
$('.purchase-dialog-content').css({
'animation': 'slideOutRight 0.3s ease forwards'
});
// 动画结束后移除元素
setTimeout(() => {
$('.purchase-dialog').remove();
}, 300);
});
},
// 跳转到购买页面
redirectToPurchase() {
const phone = UserDataManager.getPhone();
const redirectUrl = `${this.purchaseUrl}?phone=${encodeURIComponent(phone || '')}`;
//window.open(redirectUrl, '_blank');
// 切换到购买标签页
const tikuHelper = document.getElementById('tikuHelper');
if (tikuHelper) {
const vueInstance = tikuHelper.__vue__;
if (vueInstance) {
vueInstance.switchToTab(1); // Use the new public method
}
}
},
// 购买次数
purchaseCount(phone, value) {
return new Promise((resolve, reject) => {
if (!phone || phone.trim() === '') {
reject(new Error('请输入手机号'));
return;
}
// 修改为使用URLSearchParams格式传递参数
const params = new URLSearchParams();
params.append('phone', phone);
params.append('value', value);
axios({
method: 'post',
url: 'https://so.studypro.club/api/payphone',
headers: {
'Content-Type': 'application/x-www-form-urlencoded'
},
data: params
}).then(function (response) {
console.log(response.data);
resolve(response.data);
}).catch(function (error) {
console.error('购买请求失败:', error);
reject(error);
});
});
}
};
// 简单的消息提示组件
const MessageManager = {
// 添加消息计数和位置调整
_messageCount: 0,
_messages: [],
_createMessage(type, text, duration = 3000) {
const messageDiv = document.createElement('div');
messageDiv.className = `message message-${type}`;
messageDiv.setAttribute('data-soti-message', 'true');
messageDiv.innerHTML = `
${text}
`;
// 计算位置
const messageId = this._messageCount++;
const offset = this._messages.length * 10 + this._messages.reduce((acc, el) => acc + el.offsetHeight + 10, 0);
messageDiv.style.top = `${20 + offset}px`;
document.body.appendChild(messageDiv);
this._messages.push(messageDiv);
// 显示动画
setTimeout(() => {
messageDiv.classList.add('show');
}, 10);
// 自动关闭
setTimeout(() => {
messageDiv.classList.remove('show');
// 移除元素
setTimeout(() => {
const index = this._messages.indexOf(messageDiv);
if (index > -1) {
this._messages.splice(index, 1);
this._adjustMessagePositions();
}
document.body.removeChild(messageDiv);
}, 300);
}, duration);
},
// 调整所有消息的位置
_adjustMessagePositions() {
let offset = 0;
this._messages.forEach(message => {
message.style.top = `${20 + offset}px`;
offset += message.offsetHeight + 10;
});
},
success(text, duration) {
this._createMessage('success', text, duration);
},
error(text, duration) {
this._createMessage('error', text, duration);
},
warning(text, duration) {
this._createMessage('warning', text, duration);
},
info(text, duration) {
this._createMessage('info', text, duration);
}
};
// Tab 组件
Vue.component('soti-tab', {
props: ['activeTabIndex'], // 接收父组件传递的激活索引
data: function () {
return {
tabList: [
{ name: '搜题'},
{ name: '购买'},
{ name: '通知'} // 添加通知标签
]
// currentIndex: 0 // 移除内部状态,依赖父组件传递
}
},
template: `
`,
methods: {
switchTab(idx) {
// this.currentIndex = idx; // 移除内部状态更新
this.$emit('switch-tab', idx); // 通知父组件切换
}
}
});
// Header
Vue.component('soti-header', {
template: `
`,
mounted() {
this.makeDraggable(this.state, document.getElementById(this.draggleId), document.getElementById(this.targetId));
},
data: function() {
return {
draggleId: 'myHeader',
targetId: 'tikuHelper',
state: {
dragging: false,
initX: 0,
initY: 0,
transX: 0,
transY: 0,
moveX: 0,
moveY: 0,
}
}
},
methods: {
closeEv() {
toggleOpenClose();
},
makeDraggable(state, el, target) {
function globalMousemove(event) {
const { clientX, clientY } = event;
const { initX, initY, transX, transY } = state;
const moveX = clientX - initX;
const moveY = clientY - initY;
target.style.transform = `translate(${transX + moveX}px, ${transY + moveY}px)`;
state.moveX = moveX;
state.moveY = moveY;
state.dragging = true;
}
function globalMouseup(_event) {
const { transX, transY, moveX, moveY } = state;
state.transX = transX + moveX;
state.transY = transY + moveY;
state.moveX = 0;
state.moveYY = 0;
document.removeEventListener('mousemove', globalMousemove);
document.removeEventListener('mouseup', globalMouseup);
state.dragging = null;
}
function mousedown(event) {
const {clientX, clientY} = event;
state.initX = clientX;
state.initY = clientY;
document.addEventListener('mousemove', globalMousemove);
document.addEventListener('mouseup', globalMouseup);
}
el.addEventListener('mousedown', mousedown);
}
}
});
// 搜索
Vue.component('soti-content-search', {
template: `
`,
data: function() {
return {
phone: UserDataManager.getPhone(),
remainCount: UserDataManager.getRemainingCount(),
isQuerying: false,
isSearching: false,
phoneInputTimer: null,
autoClipboardSearch: UserDataManager.getAutoClipboardSearch(),
results: [],
fileInput: null,
progressBar: null,
dataQueried: false, // 标记是否已查询过数据
clipboardMonitorInterval: null // 剪贴板监控定时器
}
},
watch: {
// 监听手机号变化
phone(newVal, oldVal) {
// 保存到本地存储
UserDataManager.savePhone(newVal);
// 清除之前的定时器
if (this.phoneInputTimer) {
clearTimeout(this.phoneInputTimer);
}
// 如果输入的是11位数字,延迟500ms后自动查询
if (/^1\d{10}$/.test(newVal) && this.dataQueried) {
this.phoneInputTimer = setTimeout(() => {
this.queryRemainingCount();
}, 500);
}
},
// 监听结果变化,更新滚动指示器
results() {
this.$nextTick(() => {
this.updateScrollIndicator();
this.setupExpandCollapse();
});
}
},
mounted() {
// 自动查询剩余次数(如果有手机号)
if (this.phone && /^1\d{10}$/.test(this.phone)) {
this.queryRemainingCount();
}
// 设置粘贴监听
this.setupPasteListener();
// 设置全局快捷键
this.setupShortcuts();
// 设置结果面板滚动监听
this.setupScrollListener();
// 设置拖放支持
this.setupDragAndDrop();
// 初始化上传进度条
this.initUploadProgress();
// 确保结果面板可以正确滚动
this.ensureScrollable();
// 设置剪贴板监控
this.setupClipboardMonitor();
},
beforeDestroy() {
// 清除剪贴板监控定时器
if (this.clipboardMonitorInterval) {
clearInterval(this.clipboardMonitorInterval);
}
},
methods: {
// 设置剪贴板监控
setupClipboardMonitor() {
// 上一次剪贴板内容
let lastClipboardContent = '';
// 每秒检查一次剪贴板
this.clipboardMonitorInterval = setInterval(() => {
if (this.autoClipboardSearch) {
navigator.clipboard.readText()
.then(text => {
// 限制剪贴板内容不超过300字
if (text && text.length > 300) {
text = text.substring(0, 300);
}
// 如果剪贴板内容不为空,且与上次不同,且长度大于1
if (text && text.trim() !== '' && text !== lastClipboardContent && text.trim().length > 1) {
lastClipboardContent = text;
// 清空编辑器
const editor = document.getElementById('editor-container');
editor.innerHTML = ''; // Clear content
// 插入文本
editor.innerText = text; // Insert text
// 提示粘贴成功
MessageManager.success('已自动读取剪贴板内容,正在搜索...');
// 自动搜索
this.fetchAnswer(text);
}
})
.catch(error => {
// 剪贴板读取错误,静默处理
console.error('剪贴板读取错误:', error);
});
}
}, 1000);
},
// 切换自动剪贴板搜索
toggleAutoClipboardSearch() {
UserDataManager.setAutoClipboardSearch(this.autoClipboardSearch);
if (this.autoClipboardSearch) {
MessageManager.info('已开启自动读取剪贴板并搜索');
} else {
MessageManager.info('已关闭自动读取剪贴板并搜索');
}
},
// 查询剩余次数
async queryRemainingCount() {
if (!this.phone || this.phone.trim() === '') {
return;
}
// 验证手机号格式
if (!/^1\d{10}$/.test(this.phone)) {
MessageManager.error('请输入正确的手机号');
return;
}
this.isQuerying = true;
try {
// 调用API查询剩余次数
const count = await UserDataManager.queryRemainingCount(this.phone);
this.remainCount = count;
this.dataQueried = true; // 标记已查询过数据
// 显示查询结果
if (count > 0) {
//MessageManager.success(`查询成功,剩余${count}次`);
} else {
MessageManager.warning('剩余次数为0,请购买');
}
} catch (error) {
console.error('查询失败:', error);
MessageManager.error('查询失败,请重试');
} finally {
this.isQuerying = false;
}
},
// 处理手机号输入
handlePhoneInput() {
// 限制只能输入数字
this.phone = this.phone.replace(/\D/g, '');
// 限制最大长度为11位
if (this.phone.length > 11) {
this.phone = this.phone.slice(0, 11);
}
},
handleBuy() {
PurchaseManager.redirectToPurchase();
},
// 搜索
search() {
const editor = document.getElementById('editor-container');
const imgElement = editor.querySelector('img');
const textContent = editor.innerText.trim();
let question = '';
if (imgElement && imgElement.src) {
let base64 = imgElement.src;
// Ensure it's a data URL before processing
if (base64.startsWith('data:image')) {
this.processImage(base64);
} else {
MessageManager.warning('无效的图片内容');
}
} else if (textContent) {
question = textContent.replace(/\n|\t/g, '');
// 限制文本长度不超过300字
if (question.length > 300) {
question = question.substring(0, 300);
// 更新编辑器内容
editor.innerText = question; // Update editor content
MessageManager.warning('内容已超过300字,已自动截取前300字');
}
this.fetchAnswer(question);
} else {
MessageManager.warning('请输入文字或粘贴图片');
}
},
// 清空内容
clearContent() {
const editor = document.getElementById("editor-container");
editor.innerHTML = ''; // Clear content
MessageManager.info("内容已清空");
},
// 处理图片进行OCR识别
processImage(imgData) {
// 将base64图片数据转换为Blob对象
const byteString = atob(imgData.split(',')[1]);
const mimeString = imgData.split(',')[0].split(':')[1].split(';')[0];
const ab = new ArrayBuffer(byteString.length);
const ia = new Uint8Array(ab);
for (let i = 0; i < byteString.length; i++) {
ia[i] = byteString.charCodeAt(i);
}
const blob = new Blob([ab], { type: mimeString });
// 调用OCR识别
this.imgHandle(imgData).then(processedBlob => {
this.imgOcr(processedBlob).then(text => {
if (text && text.trim() !== '') {
// 限制文本长度不超过300字
if (text.length > 300) {
text = text.substring(0, 300);
MessageManager.warning('识别内容已超过300字,已自动截取前300字');
}
// 自动搜索识别出的文本
this.fetchAnswer(text);
} else {
MessageManager.warning('未能识别出文字,请手动输入或重新上传图片');
}
}).catch(error => {
console.error('OCR识别失败:', error);
MessageManager.error('图片识别失败,请重试');
});
}).catch(error => {
console.error('图片处理失败:', error);
MessageManager.error('图片处理失败,请重试');
});
},
// 粘贴并搜索
pasteAndSearch() {
// 请求剪贴板权限并获取内容
navigator.clipboard.read()
.then(clipboardItems => {
// 处理剪贴板中的每个项目
for (const clipboardItem of clipboardItems) {
// 检查是否有图片
if (clipboardItem.types.includes('image/png') ||
clipboardItem.types.includes('image/jpeg') ||
clipboardItem.types.includes('image/gif')) {
// 获取图片类型
const imageType = clipboardItem.types.find(type => type.startsWith('image/'));
// 获取图片数据
clipboardItem.getType(imageType)
.then(blob => {
// 将Blob转换为base64
const reader = new FileReader();
reader.onload = (e) => {
const imgData = e.target.result;
// 将图片插入编辑器
this.insertImageToEditor(imgData);
// 提示粘贴成功
MessageManager.success('图片粘贴成功,正在识别...');
// 处理图片并搜索
this.processImage(imgData);
};
reader.readAsDataURL(blob);
})
.catch(error => {
console.error('获取剪贴板图片失败:', error);
MessageManager.error('获取剪贴板图片失败');
});
return; // 找到图片后退出循环
}
}
navigator.clipboard.readText()
.then(text => {
if (text && text.trim() !== '') {
// 限制文本长度不超过300字
if (text.length > 300) {
text = text.substring(0, 300);
MessageManager.warning('剪贴板内容已超过300字,已自动截取前300字');
}
// 清空编辑器
const editor = document.getElementById('editor-container');
editor.innerHTML = ''; // Clear content
// 插入文本
editor.innerText = text; // Insert text
// 提示粘贴成功
MessageManager.success('文本粘贴成功,正在搜索...');
// 自动搜索
this.fetchAnswer(text);
} else {
MessageManager.warning('剪贴板中没有内容');
}
})
.catch(error => {
console.error('获取剪贴板文本失败:', error);
MessageManager.error('获取剪贴板内容失败');
});
})
.catch(error => {
console.error('读取剪贴板失败:', error);
// 降级方案:尝试只读取文本
navigator.clipboard.readText()
.then(text => {
if (text && text.trim() !== '') {
// 限制文本长度不超过300字
if (text.length > 300) {
text = text.substring(0, 300);
MessageManager.warning('剪贴板内容已超过300字,已自动截取前300字');
}
// 清空编辑器
const editor = document.getElementById('editor-container');
editor.innerHTML = ''; // Clear content
// 插入文本
editor.innerText = text; // Insert text
// 提示粘贴成功
MessageManager.success('文本粘贴成功,正在搜索...');
// 自动搜索
this.fetchAnswer(text);
} else {
MessageManager.warning('剪贴板中没有内容');
}
})
.catch(err => {
MessageManager.error('无法访问剪贴板,请检查浏览器权限设置');
});
});
},
// 检查剪贴板API兼容性
checkClipboardSupport() {
if (!navigator.clipboard) {
console.warn('浏览器不支持Clipboard API');
return false;
}
if (!navigator.clipboard.read) {
console.warn('浏览器不支持Clipboard.read API');
return false;
}
return true;
},
// 根据浏览器兼容性选择粘贴方法
pasteAndSearchBest() {
if (this.checkClipboardSupport()) {
this.pasteAndSearch();
} else {
// 降级方案:提示用户手动粘贴
MessageManager.info('请手动粘贴内容到编辑框中');
// 聚焦到编辑器
const editorElement = document.querySelector('#editor-container');
editorElement.focus();
}
},
// 新增:读取剪贴板并搜索
readClipboardAndSearch() {
if (!this.checkClipboardSupport()) {
MessageManager.info('请手动粘贴内容到编辑框中');
const editorElement = document.getElementById('editor-container');
editorElement.focus();
return;
}
navigator.clipboard.read().then(clipboardItems => {
let foundImage = false;
for (const item of clipboardItems) {
const imageType = item.types.find(type => type.startsWith('image/'));
if (imageType) {
foundImage = true;
item.getType(imageType).then(blob => {
const reader = new FileReader();
reader.onload = (e) => {
const imgData = e.target.result;
this.insertImageToEditor(imgData);
MessageManager.success('图片读取成功,正在识别...');
this.processImage(imgData);
};
reader.readAsDataURL(blob);
}).catch(error => {
console.error('读取剪贴板图片失败:', error);
MessageManager.error('读取剪贴板图片失败');
});
break; // Process first image found
}
}
if (!foundImage) {
navigator.clipboard.readText().then(text => {
if (text && text.trim() !== '') {
if (text.length > 300) {
text = text.substring(0, 300);
MessageManager.warning('剪贴板内容已超过300字,已自动截取前300字');
}
const editor = document.getElementById('editor-container');
editor.innerHTML = ''; // Clear editor
editor.innerText = text; // Insert text
MessageManager.success('文本读取成功,正在搜索...');
this.fetchAnswer(text);
} else {
MessageManager.warning('剪贴板中没有可识别的内容');
}
}).catch(error => {
console.error('读取剪贴板文本失败:', error);
MessageManager.error('读取剪贴板文本失败');
});
}
}).catch(error => {
console.error('读取剪贴板失败:', error);
// Fallback for browsers that might not support read() but support readText()
navigator.clipboard.readText().then(text => {
if (text && text.trim() !== '') {
if (text.length > 300) {
text = text.substring(0, 300);
MessageManager.warning('剪贴板内容已超过300字,已自动截取前300字');
}
const editor = document.getElementById('editor-container');
editor.innerHTML = ''; // Clear editor
editor.innerText = text; // Insert text
MessageManager.success('文本读取成功,正在搜索...');
this.fetchAnswer(text);
} else {
MessageManager.warning('剪贴板中没有可识别的内容');
}
}).catch(err => {
MessageManager.error('无法访问剪贴板,请检查浏览器权限设置');
});
});
},
// 监听编辑器粘贴事件
setupPasteListener() {
const editorElement = document.getElementById("editor-container");
// 监听编辑器的粘贴事件
editorElement.addEventListener("paste", (e) => {
// 阻止默认粘贴行为,以便我们可以处理它
e.preventDefault();
const clipboardData = e.clipboardData || window.clipboardData;
let pastedData = "";
let isImage = false;
// 检查是否有图片
if (clipboardData.files && clipboardData.files.length > 0) {
const file = clipboardData.files[0];
if (file.type.startsWith("image/")) {
isImage = true;
const reader = new FileReader();
reader.onload = (event) => {
const imgData = event.target.result;
this.insertImageToEditor(imgData);
MessageManager.success("图片粘贴成功,正在识别...");
this.processImage(imgData);
};
reader.readAsDataURL(file);
}
}
// 如果不是图片,获取文本
if (!isImage) {
pastedData = clipboardData.getData("text/plain");
if (pastedData && pastedData.trim() !== "") {
// 限制文本长度
if (pastedData.length > 300) {
pastedData = pastedData.substring(0, 300);
MessageManager.warning("粘贴内容已超过300字,已自动截取前300字");
}
// 清空并插入文本
editorElement.innerHTML = "";
editorElement.innerText = pastedData;
MessageManager.success("文本粘贴成功");
// 如果自动搜索开启,则搜索
if (this.autoClipboardSearch) {
MessageManager.info("正在自动搜索...");
this.fetchAnswer(pastedData);
}
}
}
});
},
// 设置全局快捷键
setupShortcuts() {
// 监听Ctrl+V或Command+V
document.addEventListener("keydown", (e) => {
// 检查是否是粘贴快捷键
if ((e.ctrlKey || e.metaKey) && e.key === "v") {
// 如果焦点不在编辑器内,且工具面板可见,则触发粘贴并搜索
const editorElement = document.getElementById("editor-container");
const toolPanel = document.getElementById("tikuHelper");
// Check if focus is outside the editor and the panel is visible
if (!editorElement.contains(document.activeElement) &&
toolPanel && toolPanel.classList.contains("show") &&
this.autoClipboardSearch) {
// 阻止默认粘贴行为
e.preventDefault();
// 触发读取剪贴板并搜索 (using the new method)
this.readClipboardAndSearch();
}
}
});
},
// 设置结果面板滚动监听
setupScrollListener() {
const resultsPanel = document.querySelector('.results-panel');
resultsPanel.addEventListener('scroll', () => {
this.updateScrollIndicator();
});
},
// 确保结果面板可以正确滚动
ensureScrollable() {
// 监听窗口大小变化,重新计算高度
window.addEventListener('resize', this.updateResultsPanelHeight);
// 初始计算
this.$nextTick(() => {
this.updateResultsPanelHeight();
});
},
// 更新结果面板高度
updateResultsPanelHeight() {
const resultsPanel = document.querySelector('.results-panel');
if (!resultsPanel) return;
// 获取容器高度
const containerHeight = document.querySelector('.content-search-container').offsetHeight;
// 获取其他元素高度
const userInfoHeight = document.querySelector('.user-info').offsetHeight;
const clipboardToggleHeight = document.querySelector('.clipboard-toggle').offsetHeight;
const editorOuterHeight = document.querySelector('.editor-outer').offsetHeight;
const progressHeight = this.isSearching ? document.querySelector('.progress-container').offsetHeight : 0;
// 计算结果面板可用高度
const availableHeight = containerHeight - userInfoHeight - clipboardToggleHeight - editorOuterHeight - progressHeight - 30; // 30px为边距
// 设置结果面板高度
resultsPanel.style.maxHeight = `${availableHeight}px`;
// 检查是否需要显示滚动指示器
this.updateScrollIndicator();
},
// 更新滚动指示器
updateScrollIndicator() {
const resultsPanel = document.querySelector('.results-panel');
const scrollIndicator = document.querySelector('.scroll-indicator');
if (!resultsPanel || !scrollIndicator) return;
// 如果内容高度大于容器高度,且未滚动到底部,显示指示器
if (resultsPanel.scrollHeight > resultsPanel.clientHeight) {
if (resultsPanel.scrollTop + resultsPanel.clientHeight < resultsPanel.scrollHeight - 50) {
scrollIndicator.style.display = 'block';
} else {
scrollIndicator.style.display = 'none';
}
} else {
scrollIndicator.style.display = 'none';
}
},
// 处理长文本展开/折叠
setupExpandCollapse() {
this.$nextTick(() => {
const longTextElements = document.querySelectorAll('.result-answer-content, .result-analysis-content');
longTextElements.forEach(element => {
// 检查内容高度是否超过300px
if (element.scrollHeight > 300) {
// 如果已经设置了展开/折叠按钮,则跳过
if (element.parentNode.querySelector('.expand-collapse-btn')) {
return;
}
// 设置最大高度
element.style.maxHeight = '300px';
element.style.overflow = 'hidden';
element.style.position = 'relative';
// 添加长文本指示器
const indicator = document.createElement('div');
indicator.className = 'long-text-indicator';
indicator.textContent = '长文本内容,可展开查看';
element.parentNode.appendChild(indicator);
// 添加展开/折叠按钮
const expandBtn = document.createElement('div');
expandBtn.className = 'expand-collapse-btn';
expandBtn.textContent = '点击展开全部';
expandBtn.dataset.expanded = 'false';
expandBtn.addEventListener('click', function() {
if (this.dataset.expanded === 'false') {
// 展开
element.style.maxHeight = element.scrollHeight + 'px';
this.textContent = '点击折叠';
this.dataset.expanded = 'true';
this.style.background = 'none';
} else {
// 折叠
element.style.maxHeight = '300px';
this.textContent = '点击展开全部';
this.dataset.expanded = 'false';
this.style.background = 'linear-gradient(to bottom, rgba(255,255,255,0), rgba(255,255,255,1))';
}
});
element.parentNode.appendChild(expandBtn);
}
});
});
},
// 结果加载完成处理
onResultsLoaded() {
// 滚动到结果顶部
const resultsPanel = document.querySelector('.results-panel');
if (resultsPanel) {
resultsPanel.scrollTop = 0;
}
// 更新结果面板高度
this.updateResultsPanelHeight();
// 设置长文本展开/折叠功能
this.setupExpandCollapse();
// 如果结果过多,添加提示
if (this.results.length > 5) {
MessageManager.info('共找到 ' + this.results.length + ' 条结果,可滚动查看');
}
},
// 打开文件选择对话框
openFileUpload() {
// 创建隐藏的文件输入元素
if (!this.fileInput) {
this.fileInput = document.createElement('input');
this.fileInput.type = 'file';
this.fileInput.accept = 'image/*';
this.fileInput.style.display = 'none';
document.body.appendChild(this.fileInput);
// 文件选择事件
this.fileInput.addEventListener('change', (e) => {
if (e.target.files && e.target.files[0]) {
const file = e.target.files[0];
this.handleImageUpload(file);
}
});
}
// 触发文件选择
this.fileInput.click();
},
// 处理图片上传
handleImageUpload(file) {
// 检查文件类型
if (!file.type.startsWith('image/')) {
MessageManager.warning('请上传图片文件');
return;
}
// 显示上传进度
this.showUploadProgress();
// 读取文件
const reader = new FileReader();
reader.onload = (e) => {
const imgData = e.target.result;
// 将图片插入编辑器
const editor = document.getElementById('editor-container');
editor.innerHTML = ''; // 清空编辑器
// 插入图片
const img = document.createElement('img');
img.src = imgData;
img.style.maxWidth = '100%';
editor.appendChild(img);
// 更新上传进度
this.updateUploadProgress(100);
// 提示上传成功
setTimeout(() => {
this.hideUploadProgress();
MessageManager.success('图片上传成功,正在识别...');
// 处理图片进行OCR识别
this.processImage(imgData);
}, 500);
};
reader.onprogress = (e) => {
if (e.lengthComputable) {
const progress = Math.round((e.loaded / e.total) * 100);
this.updateUploadProgress(progress);
}
};
reader.onerror = () => {
this.hideUploadProgress();
MessageManager.error('图片上传失败,请重试');
};
reader.readAsDataURL(file);
},
// 初始化上传进度条
initUploadProgress() {
this.progressBar = {
container: document.querySelector('.upload-progress'),
inner: document.querySelector('.upload-progress-inner')
};
},
// 显示上传进度
showUploadProgress() {
if (this.progressBar) {
this.progressBar.container.style.display = 'block';
this.progressBar.inner.style.width = '0%';
}
},
// 更新上传进度
updateUploadProgress(progress) {
if (this.progressBar) {
this.progressBar.inner.style.width = progress + '%';
}
},
// 隐藏上传进度
hideUploadProgress() {
if (this.progressBar) {
this.progressBar.container.style.display = 'none';
}
},
// 设置拖放支持
setupDragAndDrop() {
const dropZone = document.querySelector('#editor-container');
// 防止浏览器默认行为
['dragenter', 'dragover', 'dragleave', 'drop'].forEach(eventName => {
dropZone.addEventListener(eventName, (e) => {
e.preventDefault();
e.stopPropagation();
}, false);
});
// 高亮拖放区域
['dragenter', 'dragover'].forEach(eventName => {
dropZone.addEventListener(eventName, () => {
dropZone.classList.add('drag-over');
}, false);
});
// 取消高亮
['dragleave', 'drop'].forEach(eventName => {
dropZone.addEventListener(eventName, () => {
dropZone.classList.remove('drag-over');
}, false);
});
// 处理拖放文件
dropZone.addEventListener('drop', (e) => {
if (e.dataTransfer.files && e.dataTransfer.files[0]) {
const file = e.dataTransfer.files[0];
this.handleImageUpload(file);
}
}, false);
},
// 将图片插入到编辑器
insertImageToEditor(imgData) {
const editor = document.getElementById("editor-container");
// 清空编辑器内容
editor.innerHTML = ''; // Clear content
// 插入图片
const img = document.createElement('img');
img.src = imgData;
img.style.maxWidth = '100%'; // Optional: style the image
editor.appendChild(img);
},
fetchAnswer(question) {
if (!question) {
MessageManager.warning('请输入文字,或者复制截图进来');
return;
}
if (question.length < 2) {
MessageManager.warning(`至少输入2个字,当前值: ${question}`);
return;
}
// 检查是否已查询过用户数据
if (!this.dataQueried) {
// 如果有手机号,自动查询
if (this.phone && /^1\d{10}$/.test(this.phone)) {
this.queryRemainingCount().then(() => {
// 查询完成后继续搜索
this.continueSearch(question);
});
} else {
MessageManager.warning('请先输入手机号并点击查询按钮获取剩余次数');
}
return;
}
this.continueSearch(question);
},
// 继续搜索流程
continueSearch(question) {
// 检查剩余次数
if (!UserDataManager.hasRemainingCount()) {
// 自动跳转到购买tab
this.$parent.handleSwitchTab(1);
PurchaseManager.showPurchaseDialog();
return;
}
this.isSearching = true;
let prodHost = 'http://so.studypro.club/api/search';
httpClient({
method: 'POST',
url: prodHost,
headers: {
"Content-Type": "application/json"
},
data: JSON.stringify({
question: question,
phone: this.phone
}),
onload: (response) => {
// 减少使用次数
this.remainCount = UserDataManager.decreaseRemainingCount();
this.isSearching = false;
try {
let dataList = JSON.parse(response.responseText);
if(dataList && dataList.map){
this.results = dataList.map((item, index) => {
return {
answer: item.answer,
scores: item.scores,
question: item.question,
analysis: item.analysis
}
});
} else {
this.results = [{
answer: dataList.answer,
question: question,
analysis: dataList.analysis
}]
}
// 结果加载完成后处理
this.$nextTick(() => {
this.onResultsLoaded();
});
} catch (error) {
console.error('解析搜索结果失败:', error);
MessageManager.error('解析搜索结果失败');
}
},
onerror: (error) => {
this.isSearching = false;
console.error('搜索请求失败:', error);
MessageManager.error('搜索失败,请重试');
}
});
},
imgHandle(base64) {
return new Promise((resolve, reject) => {
const canvas = document.createElement("canvas");
const context = canvas.getContext("2d");
const image = new Image();
image.setAttribute("crossOrigin", "Anonymous");
image.src = base64;
image.onload = function() {
canvas.width = image.width;
canvas.height = image.height;
context.fillStyle = "#fff";
context.fillRect(0, 0, canvas.width, canvas.height);
context.drawImage(image, 0, 0);
canvas.toBlob(blob => {
resolve(blob);
});
};
image.onerror = function() {
reject(new Error('图片加载失败'));
};
});
},
imgOcr(blob) {
return new Promise((resolve, reject) => {
var formData = new FormData();
formData.append("image", blob, "screenshot.png");
GM_xmlhttpRequest({
url: "https://so.studypro.club/ocr",
method: "POST",
data: formData,
responseType: "json",
onload: function(r) {
try {
// 根据API返回格式调整解析逻辑
const result = r.response;
if (result && result.text) {
resolve(result.text);
} else if (result && result.data) {
resolve(result.data);
} else {
resolve(JSON.stringify(result));
}
} catch (err) {
reject(err);
}
},
onerror: function(err) {
reject(err);
}
});
});
}
}
});
// 通知内容组件
Vue.component("soti-content-notification", {
template: `
正在加载通知...
加载通知失败: {{ error }}
{{ notification.title }}
{{ formatTime(notification.time) }}
`,
data: function() {
return {
notifications: [],
loading: true,
error: null
};
},
mounted() {
this.fetchNotifications();
},
methods: {
fetchNotifications() {
this.loading = true;
this.error = null;
// 使用 GM_xmlhttpRequest 避免跨域问题
GM_xmlhttpRequest({
method: "GET",
url: "https://so.studypro.club/api/notice",
responseType: "json",
onload: (response) => {
if (response.status === 200 && response.response && Array.isArray(response.response)) {
this.notifications = response.response;
} else {
this.notifications = [];
console.warn("通知接口返回数据格式不正确或请求失败:", response);
this.error = "无法加载通知内容";
}
this.loading = false;
},
onerror: (error) => {
console.error("获取通知失败:", error);
this.error = "网络错误,无法加载通知";
this.loading = false;
}
});
},
formatTime(timestamp) {
if (!timestamp) return "";
// Assuming timestamp is in seconds, convert to milliseconds
const date = new Date(timestamp * 1000);
// Format as YYYY-MM-DD HH:MM:SS
const year = date.getFullYear();
const month = (date.getMonth() + 1).toString().padStart(2, '0');
const day = date.getDate().toString().padStart(2, '0');
const hours = date.getHours().toString().padStart(2, '0');
const minutes = date.getMinutes().toString().padStart(2, '0');
const seconds = date.getSeconds().toString().padStart(2, '0');
return `${year}-${month}-${day} ${hours}:${minutes}:${seconds}`;
}
}
});
// 购买组件
Vue.component('soti-content-purchase', {
template: `
请选择购买套餐:
{{price}}次
{{formatPrice(count)}}
`,
data: function() {
return {
phone: UserDataManager.getPhone(),
remainCount: UserDataManager.getRemainingCount(),
priceConfig: PurchaseManager.priceConfig,
selectedOption: null,
isPurchasing: false
}
},
mounted() {
// 使用 Observer 监听手机号变化
UserDataManager.subscribePhoneChange((newPhone) => {
this.phone = newPhone;
});
// 监听剩余次数变化 (这个可以保留,或者也用Observer实现,但当前需求只要求手机号同步)
this.$watch(() => UserDataManager.getRemainingCount(), (newCount) => {
this.remainCount = newCount;
});
},
methods: {
// 选择套餐选项
selectOption(count) {
this.selectedOption = count;
},
// 格式化价格显示
formatPrice(price) {
if (typeof price === 'string') {
return price;
}
return (price / 100).toFixed(2) + '元';
},
// 处理购买
handlePurchase() {
if (!this.selectedOption || !this.phone || this.isPurchasing) {
return;
}
if (!/^1\d{10}$/.test(this.phone)) {
MessageManager.error('请输入正确的手机号');
return;
}
this.isPurchasing = true;
MessageManager.info('正在处理购买请求...');
PurchaseManager.purchaseCount(this.phone, this.selectedOption)
.then(payUrl => {
this.isPurchasing = false;
// 跳转到支付页面
window.location.href = payUrl;
})
.catch(error => {
this.isPurchasing = false;
MessageManager.error('购买请求失败,请重试');
console.error('购买失败:', error);
});
}
}
});
// 个人中心
Vue.component('soti-content-my', {
data: function() {
return {}
},
methods: {}
});
// 设置
Vue.component('soti-content-set', {
template: `
soti-content-set
`,
data: function() {
return {}
},
methods: {}
});
let httpClient = GM.xmlHttpRequest ? GM.xmlHttpRequest : (GM_xmlhttpRequest || null);
// 添加全局样式(不需要作用域)
GM_addStyle(`
/* 全局消息提示样式 */
body > div[data-soti-message="true"] {
position: fixed;
top: 20px;
right: 20px;
transform: translateX(100%);
z-index: 10001 !important;
width: 300px;
padding: 10px 16px;
border-radius: 4px;
box-shadow: 0 4px 12px rgba(0,0,0,0.15);
background-color: white;
transition: all 0.3s ease;
opacity: 0;
}
body > div[data-soti-message="true"].show {
transform: translateX(0);
opacity: 1;
}
body > div[data-soti-message="true"] .message-content {
display: flex;
align-items: center;
}
body > div[data-soti-message="true"] .message-icon {
margin-right: 8px;
width: 16px;
height: 16px;
background-size: contain;
background-repeat: no-repeat;
background-position: center;
}
body > div[data-soti-message="true"] .icon-success {
background-image: url('');
}
body > div[data-soti-message="true"] .icon-error {
background-image: url('');
}
body > div[data-soti-message="true"] .icon-warning {
background-image: url('');
}
body > div[data-soti-message="true"] .icon-info {
background-image: url('');
}
body > div[data-soti-message="true"].message-success {
border-left: 4px solid #2ecc71;
}
body > div[data-soti-message="true"].message-error {
border-left: 4px solid #e74c3c;
}
body > div[data-soti-message="true"].message-warning {
border-left: 4px solid #f39c12;
}
body > div[data-soti-message="true"].message-info {
border-left: 4px solid #3498db;
}
/* 购买弹窗样式 */
body > div[data-soti-dialog="true"].purchase-dialog {
position: fixed;
top: 0;
right: 0;
width: 400px;
height: 100%;
background-color: transparent;
z-index: 10000 !important;
display: flex;
justify-content: flex-end;
align-items: center;
pointer-events: none; /* 允许点击穿透背景 */
}
body > div[data-soti-dialog="true"] .purchase-dialog-content {
width: 350px;
background-color: white;
border-radius: 10px;
overflow: hidden;
box-shadow: 0 10px 25px rgba(0,0,0,0.2);
margin-right: 25px;
pointer-events: auto; /* 确保内容可以点击 */
animation: slideInRight 0.3s ease;
}
@keyframes slideInRight {
from { transform: translateX(100%); }
to { transform: translateX(0); }
}
@keyframes slideOutRight {
from { transform: translateX(0); }
to { transform: translateX(100%); }
}
body > div[data-soti-dialog="true"] .purchase-dialog-header {
background: linear-gradient(135deg, #3498db, #2980b9);
color: white;
padding: 15px 20px;
display: flex;
justify-content: space-between;
align-items: center;
}
body > div[data-soti-dialog="true"] .purchase-dialog-header h3 {
margin: 0;
font-size: 18px;
}
body > div[data-soti-dialog="true"] .purchase-dialog-close {
width: 24px;
height: 24px;
display: flex;
align-items: center;
justify-content: center;
border-radius: 50%;
cursor: pointer;
transition: all 0.2s ease;
background-color: rgba(255,255,255,0.2);
position: relative;
border: none !important;
}
body > div[data-soti-dialog="true"] .purchase-dialog-close:hover {
background-color: rgba(255,255,255,0.3);
transform: rotate(90deg);
}
body > div[data-soti-dialog="true"] .purchase-dialog-close::before,
body > div[data-soti-dialog="true"] .purchase-dialog-close::after {
content: '';
position: absolute;
width: 12px;
height: 2px;
background-color: white;
border-radius: 1px;
}
body > div[data-soti-dialog="true"] .purchase-dialog-close::before {
transform: rotate(45deg);
}
body > div[data-soti-dialog="true"] .purchase-dialog-close::after {
transform: rotate(-45deg);
}
body > div[data-soti-dialog="true"] .purchase-dialog-body {
padding: 20px;
text-align: center;
}
body > div[data-soti-dialog="true"] .purchase-dialog-body p {
margin: 10px 0;
font-size: 16px;
}
body > div[data-soti-dialog="true"] .highlight {
color: #e74c3c;
font-weight: bold;
font-size: 18px;
}
body > div[data-soti-dialog="true"] .purchase-dialog-footer {
padding: 15px;
display: flex;
justify-content: center;
gap: 10px;
}
body > div[data-soti-dialog="true"] .purchase-dialog-footer .btn {
padding: 8px 20px;
border-radius: 4px;
cursor: pointer;
font-size: 14px;
transition: all 0.2s ease;
}
body > div[data-soti-dialog="true"] .purchase-dialog-footer .btn.primary {
background-color: #3498db;
color: white;
}
body > div[data-soti-dialog="true"] .purchase-dialog-footer .btn.primary:hover {
background-color: #2980b9;
}
body > div[data-soti-dialog="true"] .purchase-dialog-footer .btn.bordered {
border: 1px solid #6c757d;
color: #6c757d;
}
body > div[data-soti-dialog="true"] .purchase-dialog-footer .btn.bordered:hover {
background-color: #f8f9fa;
}
/* 触发按钮样式 */
body > div.trigger-btn {
position: fixed;
right: 0;
bottom: 4%;
width: 40px;
height: 56px;
display: flex;
align-items: center;
justify-content: center;
background: linear-gradient(135deg, #3498db, #2980b9);
color: white;
border-radius: 8px 0 0 8px;
cursor: pointer;
box-shadow: -2px 2px 10px rgba(0,0,0,0.2);
z-index: 9998;
transition: all 0.3s ease;
}
body > div.trigger-btn:hover {
width: 45px;
background: linear-gradient(135deg, #2980b9, #1f6aa6);
}
`);
// 对自定义样式应用作用域
GM_addStyle(addCSSScope(`
/* 基础样式优化 */
#tikuHelper {
font-family: 'Segoe UI', Tahoma, Geneva, Verdana, sans-serif;
color: #333333;
position: fixed;
top: 100px;
right: 20px;
width: 420px;
height: 700px;
background-color: #ffffff;
border-radius: 10px;
box-shadow: 0 10px 25px rgba(0,0,0,0.15);
display: flex;
flex-direction: column;
overflow: hidden;
z-index: 9999 !important;
transition: all 0.3s ease;
opacity: 0;
transform: scale(0.95);
pointer-events: none;
}
#tikuHelper.show {
opacity: 1;
transform: scale(1);
pointer-events: auto;
}
/* 头部样式 */
.header {
background: linear-gradient(135deg, #3498db, #2980b9);
color: white;
height: 48px;
display: flex;
justify-content: space-between;
align-items: center;
padding: 0 15px;
cursor: move;
border-radius: 10px 10px 0 0;
}
.header div:first-child {
font-size: 18px;
font-weight: bold;
}
.close-icon {
width: 28px;
height: 28px;
display: flex;
align-items: center;
justify-content: center;
border-radius: 50%;
cursor: pointer;
transition: all 0.2s ease;
background-color: rgba(255,255,255,0.2);
position: relative;
}
.close-icon:hover {
background-color: rgba(255,255,255,0.3);
transform: rotate(90deg);
}
.close-icon::before,
.close-icon::after {
content: '';
position: absolute;
width: 14px;
height: 2px;
background-color: white;
border-radius: 1px;
}
.close-icon::before {
transform: rotate(45deg);
}
.close-icon::after {
transform: rotate(-45deg);
}
/* 确保结果容器有正确的布局结构 */
.router-content {
display: flex;
flex-direction: column;
height: calc(100% - 90px); /* 减去头部和标签的高度 */
overflow: hidden !important;
}
.content-search-container {
display: flex;
flex-direction: column;
height: 100%;
overflow: hidden !important;
}
/* 用户信息区样式 */
.user-info {
padding: 12px 15px;
display: flex;
justify-content: space-between;
align-items: center;
border-bottom: 1px solid #eee;
background-color: #f8f9fa;
}
.phone-container {
display: flex;
align-items: center;
}
.phone-input {
width: 150px;
height: 32px;
padding: 0 10px;
border: 1px solid #ddd;
border-radius: 4px;
font-size: 14px;
transition: border-color 0.2s;
}
.phone-input:focus {
border-color: #3498db;
outline: none;
box-shadow: 0 0 0 2px rgba(52, 152, 219, 0.2);
}
.query-btn {
margin-left: 5px;
height: 32px;
width:56px;
padding: 0 10px;
background-color: #3498db;
color: white;
border: none;
border-radius: 4px;
cursor: pointer;
transition: background-color 0.2s;
}
.query-btn:hover {
background-color: #2980b9;
}
.usage-info {
display: flex;
align-items: center;
}
.count-display {
font-size: 14px;
color: #2ecc71;
margin-right: 10px;
}
.count-display.count-warning {
color: #e74c3c;
}
.buy-btn {
height: 28px;
width: 56px;
padding: 0 12px;
background-color: #e74c3c;
color: white;
border: none;
border-radius: 4px;
cursor: pointer;
transition: background-color 0.2s;
}
.buy-btn:hover {
background-color: #c0392b;
}
/* 剪贴板开关样式 */
.clipboard-toggle {
padding: 8px 15px;
display: flex;
align-items: center;
background-color: #f8f9fa;
border-bottom: 1px solid #eee;
}
.toggle-switch {
position: relative;
display: inline-block;
width: 40px;
height: 20px;
margin-right: 10px;
}
.toggle-switch input {
opacity: 0;
width: 0;
height: 0;
}
.toggle-slider {
position: absolute;
cursor: pointer;
top: 0;
left: 0;
right: 0;
bottom: 0;
background-color: #ccc;
transition: .4s;
border-radius: 20px;
}
.toggle-slider:before {
position: absolute;
content: "";
height: 16px;
width: 16px;
left: 2px;
bottom: 2px;
background-color: white;
transition: .4s;
border-radius: 50%;
}
input:checked + .toggle-slider {
background-color: #3498db;
}
input:checked + .toggle-slider:before {
transform: translateX(20px);
}
.toggle-label {
font-size: 14px;
color: #666;
}
/* 搜索功能区样式 */
.editor-outer {
padding: 15px;
}
#editor-container {
border: 1px solid #ddd;
border-radius: 6px;
min-height: 120px;
max-height: 200px; /* 添加最大高度 */
overflow-y: auto; /* 添加垂直滚动条 */
margin-bottom: 10px;
transition: border-color 0.3s;
padding: 5px; /* 添加内边距 */
}
#editor-container.drag-over {
border-color: #3498db;
background-color: rgba(52, 152, 219, 0.1);
}
.upload-progress {
height: 0px;
background-color: #e9ecef;
border-radius: 2px;
margin-top: 5px;
margin-bottom: 10px;
overflow: hidden;
display: none;
}
.upload-progress-inner {
height: 100%;
background-color: #3498db;
border-radius: 2px;
width: 0%;
transition: width 0.3s ease;
}
.tool-bar {
display: flex;
justify-content: space-between;
align-items: center;
}
.tool-btn {
height: 36px;
padding: 0 12px;
display: flex;
align-items: center;
justify-content: center;
border: none;
border-radius: 4px;
cursor: pointer;
transition: all 0.2s;
background-color: #f8f9fa;
color: #333;
}
.tool-btn:hover {
background-color: #2980b9;
}
.upload-btn, .paste-btn {
width: 100px;
padding: 0;
background-color: #3498db;
color:#fff;
}
.search-btn {
background-color: #3498db;
color: white;
font-weight: bold;
padding: 0 20px;
}
.search-btn:hover {
background-color: #2980b9;
}
.icon-search {
background-image: url('');
margin-right: 5px;
}
/* 进度条样式 */
.progress-container {
padding: 10px 15px;
text-align: center;
}
.progress-bar {
height: 6px;
background-color: #e9ecef;
border-radius: 3px;
overflow: hidden;
margin-bottom: 5px;
}
.progress-inner {
height: 100%;
width: 0;
background-color: #3498db;
border-radius: 3px;
animation: progress 1.5s infinite ease-in-out;
}
.progress-text {
font-size: 12px;
color: #6c757d;
}
@keyframes progress {
0% { width: 0%; }
50% { width: 70%; }
100% { width: 0%; }
}
/* 结果展示区样式 */
.results-panel {
flex: 1;
overflow-y: auto !important; /* 强制显示滚动条 */
/* max-height: calc(100% - 220px); Removed, let flexbox handle height */
margin-top:10px; /* Reduced margin */
scrollbar-width: thin;
scrollbar-color: #c1c1c1 #f1f1f1;
position: relative; /* 确保定位正确 */
}
/* 美化滚动条 - Webkit浏览器 */
.results-panel::-webkit-scrollbar {
width: 8px !important;
height: 8px !important;
}
.results-panel::-webkit-scrollbar-track {
background: #f1f1f1 !important;
border-radius: 4px !important;
}
.results-panel::-webkit-scrollbar-thumb {
background: #c1c1c1 !important;
border-radius: 4px !important;
}
.results-panel::-webkit-scrollbar-thumb:hover {
background: #a8a8a8 !important;
}
/* 滚动指示器 - 修改为粘性定位 */
.scroll-indicator {
text-align: center;
padding: 5px 0;
color: #6c757d;
font-size: 12px;
display: none;
position: sticky;
bottom: 0;
width: 100%;
background-color: rgba(255,255,255,0.8);
border-top: 1px solid #eee;
margin-top: 10px;
z-index: 1;
}
.empty-state {
display: flex;
flex-direction: column;
align-items: center;
justify-content: center;
height: 100%;
color: #999;
}
.empty-icon {
width: 64px;
height: 64px;
background-image: url('');
background-size: contain;
margin-bottom: 10px;
}
/* 确保结果列表正确显示 */
.results-list {
width: 100%;
padding-right: 5px; /* 为滚动条留出空间 */
}
/* 确保结果卡片在容器内正确显示 */
.result-card {
background-color: white;
border-radius: 8px;
box-shadow: 0 2px 8px rgba(0,0,0,0.1);
padding: 15px;
margin-bottom: 15px;
transition: transform 0.2s, box-shadow 0.2s;
animation: slideIn 0.3s ease;
width: calc(100% - 10px); /* 考虑内边距 */
box-sizing: border-box !important;
}
.result-card:hover {
transform: translateY(-2px);
box-shadow: 0 4px 12px rgba(0,0,0,0.15);
}
.result-header {
display: flex;
justify-content: space-between;
margin-bottom: 10px;
}
.result-number {
background-color: #3498db;
color: white;
width: 24px;
height: 24px;
border-radius: 50%;
display: flex;
align-items: center;
justify-content: center;
font-size: 12px;
font-weight: bold;
}
.result-similarity {
font-size: 12px;
color: #666;
background-color: #f8f9fa;
padding: 2px 8px;
border-radius: 10px;
}
.result-question {
font-size: 15px;
line-height: 1.5;
margin-bottom: 10px;
color: #333;
}
.result-answer, .result-analysis {
margin-top: 10px;
position: relative;
}
.answer-label, .analysis-label {
font-size: 13px;
font-weight: bold;
color: #3498db;
margin-bottom: 5px;
}
.result-answer-content {
background-color: #e8f4fc;
padding: 10px;
border-radius: 6px;
font-size: 14px;
line-height: 1.5;
max-height: 300px;
overflow-y: auto;
transition: max-height 0.3s ease;
}
.result-analysis-content {
padding: 10px;
border-radius: 6px;
background-color: #f8f9fa;
font-size: 14px;
line-height: 1.5;
max-height: 300px;
overflow-y: auto;
transition: max-height 0.3s ease;
}
/* 长文本处理 */
.expand-collapse-btn {
position: absolute;
bottom: 0;
left: 0;
right: 0;
text-align: center;
background: linear-gradient(to bottom, rgba(255,255,255,0), rgba(255,255,255,1));
padding: 30px 0 5px 0;
cursor: pointer;
color: #3498db;
font-size: 12px;
transition: all 0.3s ease;
}
/* 长文本指示器 */
.long-text-indicator {
font-size: 11px;
color: #6c757d;
text-align: right;
margin-top: 5px;
font-style: italic;
}
/* 优化滚动条样式 */
.result-answer-content::-webkit-scrollbar,
.result-analysis-content::-webkit-scrollbar {
width: 6px;
}
.result-answer-content::-webkit-scrollbar-track,
.result-analysis-content::-webkit-scrollbar-track {
background: #f1f1f1;
border-radius: 3px;
}
.result-answer-content::-webkit-scrollbar-thumb,
.result-analysis-content::-webkit-scrollbar-thumb {
background: #c1c1c1;
border-radius: 3px;
}
.result-answer-content::-webkit-scrollbar-thumb:hover,
.result-analysis-content::-webkit-scrollbar-thumb:hover {
background: #a8a8a8;
}
/* Tab样式优化 */
.tab-container {
padding-left: 12px;
margin: 6px 0;
display: flex;
height: 36px;
border-bottom: solid #eee 1px;
}
.tab-container .tab {
box-sizing: border-box;
position: relative;
top: 1px;
width: 72px;
height: 36px;
line-height: 36px;
text-align: center;
cursor: pointer;
border-bottom: solid #eee 1px;
border-top: solid #fff 1px;
transition: all 0.2s ease;
}
.tab-container .tab.active {
border: solid #3498db 1px;
border-bottom: solid #fff 1px;
border-top-left-radius: 6px;
border-top-right-radius: 6px;
background-color: #fff;
font-weight: bold;
}
/* 购买页面样式 */
.content-purchase-container {
padding: 20px;
height: 100%;
overflow-y: auto;
}
.purchase-header {
text-align: center;
margin-bottom: 30px;
}
.purchase-header h2 {
color: #333;
margin-bottom: 15px;
}
.purchase-header p {
color: #666;
margin: 5px 0;
}
.purchase-header p span {
font-weight: bold;
color: #3498db;
}
.purchase-options {
margin-bottom: 30px;
}
.purchase-option-title {
font-size: 16px;
font-weight: bold;
margin-bottom: 15px;
color: #333;
}
.purchase-option-list {
display: grid;
grid-template-columns: repeat(2, 1fr);
gap: 15px;
}
.purchase-option-item {
border: 2px solid #eee;
border-radius: 8px;
padding: 15px;
text-align: center;
cursor: pointer;
transition: all 0.3s ease;
}
.purchase-option-item:hover {
border-color: #3498db;
transform: translateY(-2px);
box-shadow: 0 5px 15px rgba(0,0,0,0.1);
}
.purchase-option-item.selected {
border-color: #3498db;
background-color: rgba(52, 152, 219, 0.1);
}
.option-count {
font-size: 18px;
font-weight: bold;
color: #333;
margin-bottom: 5px;
}
.option-price {
color: #e74c3c;
font-weight: bold;
}
.purchase-action {
text-align: center;
}
.purchase-button {
background-color: #3498db;
color: white;
border: none;
border-radius: 4px;
padding: 12px 30px;
font-size: 16px;
font-weight: bold;
cursor: pointer;
transition: background-color 0.3s;
}
.purchase-button:hover {
background-color: #2980b9;
}
.purchase-button:disabled {
background-color: #95a5a6;
cursor: not-allowed;
}
.purchase-tip {
margin-top: 10px;
color: #e74c3c;
font-size: 14px;
}
.content-notification-container {
padding: 20px;
border-radius: 5px;
background-color: #f9f9f9;
}
.loading-state {
display: flex;
align-items: center;
font-size: 16px;
color: #666;
}
.loading-state .loading-spinner {
display: inline-block;
width: 20px;
height: 20px;
border: 2px solid rgba(0, 0, 0, 0.2);
border-top-color: #007BFF;
border-radius: 50%;
animation: spin 1s ease-in-out infinite;
-webkit-animation: spin 1s ease-in-out infinite;
margin-right: 10px;
vertical-align: middle;
}
.error-state {
display: flex;
align-items: center;
font-size: 16px;
color: #ff0000;
}
.error-state .error-icon {
display: inline-block;
width: 20px;
height: 20px;
background-size: contain;
margin-right: 10px;
vertical-align: middle;
}
.empty-state {
text-align: center;
color: #999;
}
.empty-state .empty-icon {
width: 60px;
height: 60px;
background-size: contain;
margin: 0 auto 10px;
}
.empty-state .empty-text {
font-size: 18px;
}
.notification-list {
list-style-type: none;
padding: 0;
}
.notification-list .notification-item {
margin-bottom: 20px;
padding-bottom: 20px;
border-bottom: 1px solid #e0e0e0;
}
.notification-list .notification-item .notification-title {
font-size: 20px;
font-weight: bold;
margin-bottom: 5px;
}
.notification-list .notification-item .notification-content {
font-size: 16px;
line-height: 1.5;
}
.notification-list .notification-item .notification-time {
font-size: 14px;
color: #999;
margin-top: 5px;
}
@keyframes spin {
to {
-webkit-transform: rotate(360deg);
transform: rotate(360deg);
}
}
/* 动画效果 */
@keyframes slideIn {
from { opacity: 0; transform: translateY(10px); }
to { opacity: 1; transform: translateY(0); }
}
/* 响应式设计 */
@media (max-width: 768px) {
#tikuHelper {
width: 90%;
right: 5%;
height: 80vh;
}
.tool-btn {
height: 42px;
}
.phone-input {
width: 120px;
}
}
`));
var $wrap = $(`
搜题小帮手
`);
$('body').prepend($wrap);
new Vue({
el: '#tikuHelper',
data: {
currentTabIdx: 0
},
methods: {
handleSwitchTab(idx) {
this.currentTabIdx = idx;
},
// 添加一个公共方法用于外部调用切换标签
switchToTab(idx) {
this.currentTabIdx = idx;
}
}
});
var show = false;
$('#triggerBtn').on('click', _ => {
toggleOpenClose();
})
function toggleOpenClose() {
if (show) {
show = false;
$('#tikuHelper').removeClass('show');
$('#triggerBtn').css('background', 'linear-gradient(135deg, #3498db, #2980b9)');
$('#triggerBtn').css('width', '100px');
$('#triggerBtn').html('搜题小帮手');
} else {
show = true;
$('#tikuHelper').addClass('show');
$('#triggerBtn').css('background', 'linear-gradient(135deg, #6c757d, #495057)');
$('#triggerBtn').html('搜题小帮手');
}
}
// 默认展开搜题小帮手
setTimeout(() => {
if (show) {
toggleOpenClose();
}
}, 500);
})