Greasy Fork

Greasy Fork is available in English.

🔥🔥番茄小说网页版,免费阅读,支持解锁VIP🔥🔥

番茄小说网页版,免费阅读,支持解锁VIP,支持一键下载小说

当前为 2024-11-24 提交的版本,查看 最新版本

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴Greasemonkey 油猴子Violentmonkey 暴力猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴Violentmonkey 暴力猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴Userscripts ,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展后才能安装此脚本。

(我已经安装了用户脚本管理器,让我安装!)

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

(我已经安装了用户样式管理器,让我安装!)

// ==UserScript==
// @name         🔥🔥番茄小说网页版,免费阅读,支持解锁VIP🔥🔥
// @namespace    https://www.softrr.cn/
// @version      1.1.8
// @author       hackhase
// @description  番茄小说网页版,免费阅读,支持解锁VIP,支持一键下载小说
// @license      MIT
// @icon         https://p1-tt.byteimg.com/origin/novel-static/a3621391ca2e537045168afda6722ee9
// @match        *://fanqienovel.com/*
// @require      https://cdn.jsdelivr.net/npm/[email protected]/dist/vue.global.prod.js
// @require      data:application/javascript,%3Bwindow.Vue%3DVue%3B
// @connect      www.softrr.cn
// @connect      novel.snssdk.com
// @connect      novel.snssdk.com
// @connect      www.dushuge.com
// @connect      www.b5200.net
// @connect      fq.travacocro.com
// @connect      localhost
// @connect      api.softrr.cn
// @grant        GM_addStyle
// @grant        GM_xmlhttpRequest
// @grant        unsafeWindow
// ==/UserScript==

(o=>{if(typeof GM_addStyle=="function"){GM_addStyle(o);return}const t=document.createElement("style");t.textContent=o,document.head.append(t)})(" :root{font-family:Inter,Avenir,Helvetica,Arial,sans-serif;font-size:16px;line-height:24px;font-weight:400;color-scheme:light dark;color:#ffffffde;background-color:#242424;font-synthesis:none;text-rendering:optimizeLegibility;-webkit-font-smoothing:antialiased;-moz-osx-font-smoothing:grayscale;-webkit-text-size-adjust:100%}a{font-weight:500;color:#646cff;text-decoration:inherit}a:hover{color:#535bf2}body{margin:0;place-items:center;min-width:320px;min-height:100vh}h1{font-size:3.2em;line-height:1.1}button{border-radius:8px;border:1px solid transparent;padding:.6em 1.2em;font-size:1em;font-weight:500;font-family:inherit;background-color:#1a1a1a;cursor:pointer;transition:border-color .25s}button:hover{border-color:#646cff}button:focus,button:focus-visible{outline:4px auto -webkit-focus-ring-color}.card{padding:2em}#app{height:100px}@media (prefers-color-scheme: light){:root{color:#213547;background-color:#fff}a:hover{color:#747bff}button{background-color:#f9f9f9}}.modal-wrapper[data-v-6f8ba791]{position:fixed;top:0;left:0;width:100%;height:100%;background-color:#00000080;display:flex;justify-content:center;align-items:center;z-index:9999}.modal[data-v-6f8ba791]{background-color:#fff;padding:20px;border-radius:5px}.header[data-v-6f8ba791]{display:flex;justify-content:space-between;align-items:center;margin-bottom:10px}.header h2[data-v-6f8ba791]{margin:0;font-size:20px;font-weight:700}.header button[data-v-6f8ba791]{border:none;background-color:transparent;font-size:20px;cursor:pointer}.contentBox[data-v-6f8ba791]{max-height:400px;overflow:auto;font-size:16px;display:flex;justify-content:space-between}.contentBox .produce p[data-v-6f8ba791]{margin-top:15px}.contentBox .produce .ipt[data-v-6f8ba791]{margin-top:15px;height:30px;border-radius:5px;padding-left:10px}.contentBox .img[data-v-6f8ba791]{display:flex;align-items:center;justify-content:center}.contentBox .img img[data-v-6f8ba791]{width:180px}input[data-v-6f8ba791]::-webkit-input-placeholder{color:#aab2bd;font-size:14px;padding-left:5px}.copy[data-v-ad8f765f]{width:160px;position:fixed;right:10px;top:80px;color:#111;z-index:999;display:flex;flex-direction:column}.copy .prase[data-v-ad8f765f],.copy .down[data-v-ad8f765f]{width:100px;height:30px;font-size:14px;background-color:red;color:#fff;border-radius:10%;z-index:999}.copy .prase[data-v-ad8f765f]:hover,.copy .down[data-v-ad8f765f]:hover{background-color:#87ceeb;color:#fff} ");

(async function (vue) {
  'use strict';

  const _export_sfc = (sfc, props) => {
    const target = sfc.__vccOpts || sfc;
    for (const [key, val] of props) {
      target[key] = val;
    }
    return target;
  };
  const _withScopeId$1 = (n) => (vue.pushScopeId("data-v-6f8ba791"), n = n(), vue.popScopeId(), n);
  const _hoisted_1$1 = { class: "modal" };
  const _hoisted_2$1 = { class: "header" };
  const _hoisted_3 = { class: "contentBox" };
  const _hoisted_4 = { class: "produce" };
  const _hoisted_5 = /* @__PURE__ */ _withScopeId$1(() => /* @__PURE__ */ vue.createElementVNode("p", null, "1、扫描右侧公众号,点击关注!", -1));
  const _hoisted_6 = /* @__PURE__ */ _withScopeId$1(() => /* @__PURE__ */ vue.createElementVNode("p", null, "2、在软件爬取者后台回复:验证码", -1));
  const _hoisted_7 = /* @__PURE__ */ _withScopeId$1(() => /* @__PURE__ */ vue.createElementVNode("p", null, "3、在下方输入框输入获取的验证码后回车", -1));
  const _hoisted_8 = /* @__PURE__ */ _withScopeId$1(() => /* @__PURE__ */ vue.createElementVNode("div", { class: "img" }, [
    /* @__PURE__ */ vue.createElementVNode("img", {
      src: "",
      alt: ""
    })
  ], -1));
  const _sfc_main$1 = {
    __name: "Model",
    props: {
      title: {
        type: String,
        required: true
      },
      code: {
        type: Number
      }
    },
    setup(__props, { expose: __expose }) {
      const props = __props;
      const visible = vue.ref(false);
      const openModal = () => {
        visible.value = true;
      };
      const closeModal = () => {
        visible.value = false;
      };
      __expose({
        visible,
        openModal,
        closeModal
      });
      const codeValue = vue.ref();
      const enterCode = () => {
        if (codeValue.value == props.code) {
          localStorage.setItem("code", codeValue.value);
          visible.value = false;
          alert("验证成功,请再次点击解析!");
          codeValue.value = "";
        } else {
          alert("验证码错误,请重新输入!");
          codeValue.value = "";
        }
      };
      return (_ctx, _cache) => {
        return vue.withDirectives((vue.openBlock(), vue.createElementBlock("div", {
          class: "modal-wrapper",
          onClick: vue.withModifiers(closeModal, ["self"])
        }, [
          vue.createElementVNode("div", _hoisted_1$1, [
            vue.createElementVNode("div", _hoisted_2$1, [
              vue.createElementVNode("h2", null, vue.toDisplayString(__props.title), 1),
              vue.createElementVNode("button", { onClick: closeModal }, "X")
            ]),
            vue.createElementVNode("div", _hoisted_3, [
              vue.createElementVNode("div", _hoisted_4, [
                _hoisted_5,
                _hoisted_6,
                _hoisted_7,
                vue.withDirectives(vue.createElementVNode("input", {
                  class: "ipt",
                  type: "text",
                  "onUpdate:modelValue": _cache[0] || (_cache[0] = ($event) => codeValue.value = $event),
                  onKeydown: vue.withKeys(enterCode, ["enter"]),
                  placeholder: "请输入验证码后按回车"
                }, null, 544), [
                  [vue.vModelText, codeValue.value]
                ])
              ]),
              _hoisted_8
            ])
          ])
        ], 512)), [
          [vue.vShow, visible.value]
        ]);
      };
    }
  };
  const Model = /* @__PURE__ */ _export_sfc(_sfc_main$1, [["__scopeId", "data-v-6f8ba791"]]);
  var _GM_xmlhttpRequest = /* @__PURE__ */ (() => typeof GM_xmlhttpRequest != "undefined" ? GM_xmlhttpRequest : void 0)();
  var _unsafeWindow = /* @__PURE__ */ (() => typeof unsafeWindow != "undefined" ? unsafeWindow : void 0)();
  const importScript = (src) => {
    return new Promise((resolve, reject) => {
      const script = document.createElement("script");
      script.src = src;
      script.addEventListener("load", () => {
        var _a;
        resolve();
        (_a = script.parentElement) == null ? void 0 : _a.removeChild(script);
      });
      script.addEventListener("error", (e) => {
        var _a;
        reject(e);
        (_a = script.parentElement) == null ? void 0 : _a.removeChild(script);
      });
      document.body.appendChild(script);
    });
  };
  await( importScript(
    "https://cdn.jsdelivr.net/npm/[email protected]/dist/jszip.min.js"
  ));
  const JSZip = _unsafeWindow == null ? void 0 : _unsafeWindow.JSZip;
  const getCode = () => {
    return new Promise(function(resolve, reject) {
      _GM_xmlhttpRequest({
        method: "GET",
        url: `https://www.softrr.cn/crawler/getCode`,
        headers: {
          Referer: "https://www.softrr.cn/",
          "User-Agent": "Mozilla/5.0 (Windows NT 10.0; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/114.0.5735.289 Safari/537.36"
        },
        onload: function(res) {
          resolve(JSON.parse(res.response).data[0].code);
        },
        onerror: function(error) {
          console.log(error);
        }
      });
    });
  };
  const cookie = "novel_web_id=7357767624615331362;";
  const getContent = (ID, ResCode) => {
    return new Promise(function(resolve, reject) {
      _GM_xmlhttpRequest({
        method: "GET",
        url: `https://api.softrr.cn/api/fanqie?id=${ID}&code=${ResCode}`,
        headers: {
          "Content-Type": "application/json",
          Accept: "application/json, text/plain, */*"
        },
        cookie,
        anonymous: true,
        onload: function(res) {
          if (res.status == "403") {
            resolve({
              code: 403,
              info: "因接口压力,普通用户一天限制解析30次!会员用户一天限制解析1000次,</br></br>需要会员的请到公众号:软件珍藏室,后台加微信"
            });
          } else {
            if (res.response !== void 0) {
              resolve(JSON.parse(res.responseText).data);
            } else {
              resolve("无法解析");
            }
          }
        },
        onerror: function(error) {
          reject(error);
        }
      });
    });
  };
  const downFileTxt = (IDList, chapterTitle) => {
    const zip = new JSZip();
    const cache = {};
    const promises = [];
    IDList.forEach((item) => {
      const promise = getContent(item.ID).then((data) => {
        if (data.includes('<header><div class="tt-title">')) {
          data = data.replace('<header><div class="tt-title">', "").replace("</div></header><article><p></p><p>", "\n").replace(/<\/p><p>/g, "\n");
        }
        let blob = new Blob([data], { type: "text/plain" });
        zip.file(item.title + ".txt", blob, { binary: true });
        cache[item.title] = data;
      });
      promises.push(promise);
    });
    Promise.all(promises).then(() => {
      zip.generateAsync({ type: "blob" }).then((content) => {
        downLoad(content, chapterTitle, "zip");
      });
    }).catch((err) => {
      console.log(err);
    });
  };
  const downLoad = (blob, name, type) => {
    if (!blob || !type)
      return;
    const url = window.URL || window.webkitURL || window.moxURL;
    const downloadHref = url.createObjectURL(blob);
    const link = document.createElement("a");
    link.href = downloadHref;
    link.download = `${name || "导出文件"}.${type}`;
    document.body.appendChild(link);
    link.click();
    document.body.removeChild(link);
    url.revokeObjectURL(downloadHref);
  };
  const getBiQuGe = (url) => {
    return new Promise((resolve, reject) => {
      _GM_xmlhttpRequest({
        method: "GET",
        url,
        headers: {
          "User-Agent": "Mozilla/5.0 (Windows NT 10.0; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/114.0.5735.289 Safari/537.36"
        },
        onload: function(res) {
          if (res.status == 200) {
            if (res.response !== void 0) {
              resolve(res.response);
            } else {
              resolve("无法解析");
            }
          } else {
            resolve("无法解析");
          }
        },
        onerror: function(error) {
          reject("解析失败", error);
        }
      });
    });
  };
  const getBookName = () => {
    var baseUrl = window.location.href;
    let bookName = "";
    let ChartName = "";
    if (baseUrl.includes("reader")) {
      bookName = document.querySelector(".muye-reader-nav-title").innerText.trim();
      if (document.querySelector(".muye-reader-title")) {
        ChartName = document.querySelector(".muye-reader-title").innerText.trim();
      } else {
        ChartName = document.querySelector("#heading_id_2").innerText.trim();
      }
    }
    return [bookName, ChartName];
  };
  const getDuShuGeContent = async () => {
    let bookInfo = getBookName();
    alert(bookInfo);
    let bookUrl = `http://www.dushuge.com/hsdgiohsdigohsog.php?ie=gbk&q=${bookInfo[0]}`;
    let bookdata = await getBiQuGe(bookUrl);
    if (bookdata === "无法解析") {
      return "无法解析";
    }
    let reg = /<a.+?href=\"(.+?)\".*>(.+)<\/a>/g;
    let pattern = /href=[\"|'](.*?)[\"|']/gi;
    let r = /["|'](.*)["|']/;
    let bookdataList = bookdata.match(reg).toString().replace(/<\/li><li>/g, ",").replace(/<div class=".*?"><h4 class=".*?">/g, ",").replace(/<\/h4>/g, ",").split(",");
    let bookhref = "";
    bookdataList.forEach((element) => {
      let result = getACon(element);
      if (result === bookInfo[0]) {
        bookhref = element.match(pattern)[0].match(r)[1];
      }
    });
    if (bookhref === "") {
      return "无法解析";
    }
    let chartUrl = "http://www.dushuge.com" + bookhref;
    let chartdata = await getBiQuGe(chartUrl);
    let reg1 = /<dd>.*?<a [^>]*>(.*?)<\/a>.*?<\/dd>/g;
    let chartdataList = chartdata.match(reg1);
    let chartInfo = [];
    let regex = /<a[^>]*>(.*?)<\/a>/;
    for (var i = 0; i < chartdataList.length; i++) {
      let obj = {};
      if (chartdataList[i].includes(".html")) {
        obj.charthref = "http://www.dushuge.com" + chartdataList[i].split('"')[1];
        obj.chartname = chartdataList[i].match(regex)[1].toString();
        chartInfo.push(JSON.stringify(obj));
      }
    }
    const chartRes = chartInfo.find((item) => {
      return JSON.parse(item).chartname == bookInfo[1];
    });
    if (chartRes === void 0) {
      return "无法解析";
    }
    let contentUrl = JSON.parse(chartRes).charthref;
    let contentdata = await getBiQuGe(contentUrl);
    contentdata = contentdata.split('<div id="content" class="showtxt">')[1].split("<br /><br /></div>")[0];
    contentdata = contentdata.replace(/&nbsp;/gi, "");
    return contentdata;
  };
  const getBiGuGeContent = async () => {
    let bookInfo = getBookName();
    let bookUrl = `http://www.b5200.net/modules/article/search.php?searchkey=${bookInfo[0]}`;
    let bookdata = await getBiQuGe(bookUrl);
    if (bookdata === "无法解析") {
      return "无法解析";
    }
    let reg = /<a.+?href=\"(.+?)\".*>(.+)<\/a>/g;
    let pattern = /href=[\"|'](.*?)[\"|']/gi;
    let r = /["|'](.*)["|']/;
    let bookdataList = bookdata.match(reg).toString().split(",");
    let bookhref = "";
    bookdataList.forEach((element) => {
      let result = getACon(element);
      if (result === bookInfo[0]) {
        bookhref = element.match(pattern)[0].match(r)[1];
      }
    });
    if (bookhref === "") {
      return "无法解析";
    }
    let chartUrl = "http://www.b5200.net/" + bookhref;
    let chartdata = await getBiQuGe(chartUrl);
    let chartdataList = chartdata.match(reg).toString().split(",");
    let chartInfo = [];
    let regex = /<a[^>]*>(.*?)<\/a>/;
    for (var i = 0; i < chartdataList.length; i++) {
      let obj = {};
      if (chartdataList[i].includes(".html")) {
        obj.charthref = "http://www.b5200.net" + chartdataList[i].match(pattern)[0].match(r)[1].toString();
        obj.chartname = chartdataList[i].match(regex)[1].toString();
        chartInfo.push(JSON.stringify(obj));
      }
    }
    const chartRes = chartInfo.find((item) => {
      return JSON.parse(item).chartname == bookInfo[1];
    });
    if (chartRes === void 0) {
      return "无法解析";
    }
    let contentUrl = JSON.parse(chartRes).charthref;
    let contentdata = await getBiQuGe(contentUrl);
    contentdata = contentdata.replace(/<div id='gc1' class='gcontent1'>(.*?)<\/div>/, "");
    var regexCon = /<div id="content" class="(.*?)">(.*?)<\/div>/;
    let contentList = contentdata.match(regexCon)[0];
    contentList = contentList.replace(/<div id="content" class="(.*?)">/, "");
    contentList = contentList.replace(/<div(.*?)><\/div>/, "");
    return contentList;
  };
  const getACon = (html) => {
    const regex = /<a[^>]*>(.*?)<\/a>/gi;
    let match;
    while (match = regex.exec(html)) {
      return match[1];
    }
  };
  const _withScopeId = (n) => (vue.pushScopeId("data-v-ad8f765f"), n = n(), vue.popScopeId(), n);
  const _hoisted_1 = { class: "copy" };
  const _hoisted_2 = /* @__PURE__ */ _withScopeId(() => /* @__PURE__ */ vue.createElementVNode("a", { href: "" }, null, -1));
  const _sfc_main = {
    __name: "App",
    setup(__props) {
      const url = vue.ref("");
      const flag = vue.ref(true);
      const extractedId = vue.ref();
      url.value = window.location.href;
      var array = url.value.split("/");
      extractedId.value = array[array.length - 1].split("?")[0] ? array[array.length - 1].split("?")[0] : extractedId.value;
      const code = vue.ref();
      const model = vue.ref("");
      const auto = vue.ref("");
      const titleList = vue.ref([]);
      const chapterTitle = vue.ref("");
      const ResCode = vue.ref("");
      vue.onMounted(() => {
        if (url.value.includes("reader")) {
          var lastBtn = document.querySelector(".last");
          flag.value = false;
          lastBtn.addEventListener("click", function() {
            onPrase();
          });
          var nextBtn = document.querySelector(".next");
          nextBtn.addEventListener("click", function() {
            onPrase();
          });
          onPrase();
        } else if (url.value.includes("page")) {
          flag.value = true;
          titleList.value = document.querySelectorAll(".chapter-item");
          chapterTitle.value = document.querySelector(".info-name").innerText;
        }
      });
      const onPrase = async () => {
        code.value = await getCode();
        let locaCode = localStorage.getItem("code") || "";
        if (locaCode == code.value) {
          let content = "";
          if (await getContent(extractedId.value) !== "无法解析") {
            url.value = window.location.href;
            var array2 = url.value.split("/");
            extractedId.value = array2[array2.length - 1].split("?")[0] ? array2[array2.length - 1].split("?")[0] : extractedId.value;
            ResCode.value = localStorage.getItem("ResCode") ? localStorage.getItem("ResCode") : "1234";
            let contentList = await getContent(extractedId.value, ResCode.value);
            if (contentList.code == 403) {
              content = "<h1>" + contentList.info + "</h1>";
            } else {
              content = "<p>" + contentList.join("<p/><p>") + "<p/>";
            }
          } else {
            let promise = getDuShuGeContent();
            if (await getDuShuGeContent() !== "无法解析") {
              content = await getDuShuGeContent();
            } else {
              if (await getBiGuGeContent() !== "无法解析") {
                content = await getBiGuGeContent();
              } else {
                content = "解析失败,请稍后再试";
              }
            }
            promise.catch(async (err) => {
              console.log(err);
              if (err == "解析失败") {
                if (await getBiGuGeContent() !== "无法解析") {
                  console.log(5);
                  content = await getBiGuGeContent();
                } else {
                  content = "解析失败,请稍后再试!";
                }
              }
            });
          }
          content = content + '</br><a href="https://www.softrr.cn/" style="color: blue;font-size: 16px;" target="_blank">有问题请到软件珍藏室后台反映</a>';
          let muye = document.querySelector(".muye-reader-content");
          muye.innerHTML = content;
          let vip = document.querySelector(".muye-to-vip");
          vip.style.display = "none";
          document.querySelector(".muye-to-fanqie").style.display = "none";
          url.value = window.location.href;
        } else {
          model.value.openModal();
        }
      };
      const onDown = async () => {
        code.value = await getCode();
        let locaCode = localStorage.getItem("code") || "";
        if (locaCode == code.value) {
          let IDList = [];
          titleList.value.forEach((item) => {
            let tem = item.children[0].href.split("/");
            let ID = tem[tem.length - 1];
            IDList.push({ ID, title: item.innerText });
          });
          url.value = window.location.href;
          downFileTxt(IDList, chapterTitle.value);
        }
      };
      const title = vue.ref("为了减少端口压力,防止滥用,采取必要的验证手段。");
      return (_ctx, _cache) => {
        return vue.openBlock(), vue.createElementBlock("div", _hoisted_1, [
          vue.withDirectives(vue.createElementVNode("button", {
            onClick: onPrase,
            class: "prase",
            ref_key: "auto",
            ref: auto
          }, "解析", 512), [
            [vue.vShow, false]
          ]),
          vue.withDirectives(vue.createElementVNode("button", {
            onClick: onDown,
            class: "down"
          }, "一键下载", 512), [
            [vue.vShow, false]
          ]),
          vue.createVNode(Model, {
            title: title.value,
            code: code.value,
            ref_key: "model",
            ref: model
          }, null, 8, ["title", "code"]),
          _hoisted_2
        ]);
      };
    }
  };
  const App = /* @__PURE__ */ _export_sfc(_sfc_main, [["__scopeId", "data-v-ad8f765f"]]);
  const app = vue.createApp(App);
  app.mount(
    (() => {
      const app2 = document.createElement("div");
      document.body.append(app2);
      return app2;
    })()
  );

})(Vue);