您需要先安装一个扩展,例如 篡改猴、Greasemonkey 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 Userscripts ,之后才能安装此脚本。
您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。
您需要先安装用户脚本管理器扩展后才能安装此脚本。
自定义 api key 等信息,实现 AI 总结帖子,会保留缓存记录到本地避免大量消耗 token。
// ==UserScript== // @name v2ex AI 总结帖子 // @namespace https://github.com/dlzmoe/scripts // @version 0.0.5 // @description 自定义 api key 等信息,实现 AI 总结帖子,会保留缓存记录到本地避免大量消耗 token。 // @author dlzmoe // @match *://v2ex.com/* // @match *://*.v2ex.com/* // @match *://www.v2ex.com/* // @grant GM_xmlhttpRequest // @grant GM_registerMenuCommand // @grant GM_unregisterMenuCommand // @grant GM_openInTab // @grant GM_getValue // @grant GM_setValue // @grant GM_notification // @grant GM_info // @icon  // @license Apache-2.0 license // ==/UserScript== (function () { 'use strict'; var menu_ALL = [ ['menu_ManualSummary', '是否开启手动总结 / 自动', '是否开启手动总结 / 自动', true], ]; var menu_ID = []; for (let i = 0; i < menu_ALL.length; i++) { // 如果读取到的值为 null 就写入默认值 if (GM_getValue(menu_ALL[i][0]) == null) { GM_setValue(menu_ALL[i][0], menu_ALL[i][3]) }; } registerMenuCommand(); // 注册(不可用)脚本菜单 function registerMenuCommand() { if (menu_ID.length > menu_ALL.length) { // 如果菜单ID数组多于菜单数组,说明不是首次添加菜单,需要卸载所有脚本菜单 for (let i = 0; i < menu_ID.length; i++) { GM_unregisterMenuCommand(menu_ID[i]); } } for (let i = 0; i < menu_ALL.length; i++) { // 循环注册(不可用)脚本菜单 menu_ALL[i][3] = GM_getValue(menu_ALL[i][0]); menu_ID[i] = GM_registerMenuCommand(`${menu_ALL[i][3]?'✅':'❌'} ${menu_ALL[i][1]}`, function () { menu_switch(`${menu_ALL[i][3]}`, `${menu_ALL[i][0]}`, `${menu_ALL[i][2]}`) }); } menu_ID[menu_ID.length] = GM_registerMenuCommand('⚙️ 设置API key配置', function () { setApiConfig(); }); menu_ID[menu_ID.length] = GM_registerMenuCommand('💬 建议与反馈!', function () { window.GM_openInTab("https://github.com/dlzmoe/scripts", { active: true, insert: true, setParent: true }); }); } function setApiConfig() { $('body').append(` <div class="v2exaisummary"> <input type="text" id="v2exaisummary-apikey" placeholder="sk-xxxxxxx"> <input type="text" id="v2exaisummary-baseurl" placeholder="https://api.openai.com" value="https://api.openai.com"> <input type="text" id="v2exaisummary-model" placeholder="gpt-4o-mini" value="gpt-4o-mini"> <button id="v2exaisummary-save">保存</button> </div> `) $('.v2exaisummary').show(); var v2exaisummaryAPI = JSON.parse(localStorage.getItem('v2exaisummaryAPI')); if (!v2exaisummaryAPI) { v2exaisummaryAPI = { apikey: "", baseurl: "", model: "", } } else { $('#v2exaisummary-apikey').val(v2exaisummaryAPI.apikey); $('#v2exaisummary-baseurl').val(v2exaisummaryAPI.baseurl); $('#v2exaisummary-model').val(v2exaisummaryAPI.model); } // 保存 $('#v2exaisummary-save').click(function () { v2exaisummaryAPI = { apikey: $('#v2exaisummary-apikey').val(), baseurl: $('#v2exaisummary-baseurl').val(), model: $('#v2exaisummary-model').val(), } localStorage.setItem('v2exaisummaryAPI', JSON.stringify(v2exaisummaryAPI)); $('.v2exaisummary').remove(); }) } // 菜单开关 function menu_switch(menu_status, Name, Tips) { if (menu_status == 'true') { GM_setValue(`${Name}`, false); GM_notification({ text: `已关闭 [${Tips}] 功能\n(点击刷新网页后生效)`, timeout: 3500, onclick: function () { location.reload(); } }); } else { GM_setValue(`${Name}`, true); GM_notification({ text: `已开启 [${Tips}] 功能\n(点击刷新网页后生效)`, timeout: 3500, onclick: function () { location.reload(); } }); } registerMenuCommand(); // 重新注册(不可用)脚本菜单 }; // 返回菜单值 function menu_value(menuName) { for (let menu of menu_ALL) { if (menu[0] == menuName) { return menu[3] } } } $(function () { }) // 手动总结 function menu_ManualSummary() { isCache(); if (menu_value('menu_ManualSummary')) { // 手动总结 $('.aisummary').click(function () { $('.aisummary').hide(); getPostContent(); }) } else { // 自动总结 $('.aisummary').hide(); // getPostContent(); } } if (window.location.pathname.indexOf('/t/') > -1) { menu_ManualSummary(); } // 获取帖子内容 function getPostContent() { var v2exaisummaryAPI = JSON.parse(localStorage.getItem('v2exaisummaryAPI')); $('.gpt-summary-wrap').show(); return new Promise((resolve, reject) => { const topic_contentdata = $('h1').html() + $('.topic_content').map(function () { return $(this).html(); }).get().join(''); const v2exprompt = `根据以下帖子内容进行总结,请使用 text 文本返回回答,字数限制 200 字以内,越精炼越好,语言要求返回简体中文,并且进行中英文混排优化。 帖子内容如下: ${topic_contentdata}`; fetch(`${v2exaisummaryAPI.baseurl}/v1/chat/completions`, { method: "POST", headers: { "Content-Type": "application/json", Authorization: `Bearer ${v2exaisummaryAPI.apikey}`, }, body: JSON.stringify({ model: v2exaisummaryAPI.model, messages: [{ role: "user", content: v2exprompt, }], temperature: 0.7, }), }) .then(response => { if (!response.ok) { reject(`HTTP error! status: ${response.status}`); } return response.json(); }) .then(gptData => { $(".gpt-summary").html(`AI 总结:${gptData.choices[0].message.content}`); $('.airegenerate').show(); let v2exaisummarydata = JSON.parse(localStorage.getItem("v2exaisummarydata")) || []; const match = window.location.pathname; let existingObject = v2exaisummarydata.find((item) => item.name == match); let newObject = { name: match, value: gptData.choices[0].message.content, }; if (existingObject) { existingObject.value = newObject.value; } else { v2exaisummarydata.push(newObject); } // 将帖子总结的数据缓存 localStorage.setItem("v2exaisummarydata", JSON.stringify(v2exaisummarydata)); resolve(); }) .catch(error => { $(".gpt-summary").html(`抱歉生成失败,请检查配置或者反馈给开发者!`); $('.airegenerate').show(); }); }); } // 先判断是否有缓存 function isCache() { $("#Main .box>.header").after(`<button type="button" class="aisummary">AI总结</button>`); $("#Main .box>.header").after( `<div class="gpt-summary-wrap"> <div class="gpt-summary">AI 总结:正在使用 AI 总结内容中,请稍后...</div> <button type="button" class="airegenerate" style="display:none">重新生成</button> </div>` ); let v2exaisummarydata = JSON.parse(localStorage.getItem("v2exaisummarydata")) || []; const match = window.location.pathname; let existingObject = v2exaisummarydata.find((item) => item.name === match); if (existingObject) { // 存在缓存,拿旧数据 $('.gpt-summary-wrap').show(); $(".gpt-summary").html(`AI 总结:${existingObject.value}`); $('.airegenerate').show(); $('.aisummary').hide(); } else { $('.gpt-summary-wrap').hide(); if (!menu_value('menu_ManualSummary')) { getPostContent(); } } $('.airegenerate').click(() => { $('.gpt-summary').html(`AI 总结:正在使用 AI 总结内容中,请稍后...`) $('.airegenerate').hide(); getPostContent(); }) } $('body').append(`<style>.gpt-summary-wrap{background:#fffbd9;border-radius:5px;padding:10px;font-size:14px;color:#303030;margin:0;line-height:1.6;text-align:left}.aisummary{display:flex;outline:0;border:1px solid #eee;background:#ffe27d;color:#626262;padding:4px 10px;cursor:pointer;border-radius:3px}.gpt-summary-wrap .airegenerate{margin-top:6px;outline:0;border:1px solid #eee;background:#ffe27d;color:#626262;padding:4px 10px;cursor:pointer;border-radius:3px}.v2exaisummary{position:fixed;bottom:20px;right:20px;z-index:99999;max-width:400px;padding:20px;border:1px solid #ddd;border-radius:8px;box-shadow:0 2px 10px rgba(0,0,0,.1);background-color:#f9f9f9;display:none}.v2exaisummary input[type=text]{width:100%;padding:10px;margin:10px 0;border:1px solid #ccc;border-radius:4px;font-size:16px;transition:border-color .3s}.v2exaisummary input[type=text]:focus{border-color:#007bff;outline:0}.v2exaisummary button{width:100%;padding:10px;background-color:#007bff;color:#fff;border:none;border-radius:4px;font-size:16px;cursor:pointer;transition:background-color .3s}.v2exaisummary button:hover{background-color:#0056b3}</style>`) })();
QingJ © 2025
镜像随时可能失效,请加Q群300939539或关注我们的公众号极客氢云获取最新地址