Fanqie Novel Free Reading

番茄小说免费网页阅读 不用客户端 可下载小说

2025/03/27のページです。最新版はこちら。

このスクリプトの質問や評価の投稿はこちら通報はこちらへお寄せください。
// ==UserScript==
// @name              Fanqie Novel Free Reading
// @namespace         https://github.com/SmashPhoenix272
// @version           5.1
// @description       番茄小说免费网页阅读 不用客户端 可下载小说
// @description:zh-cn 番茄小说免费网页阅读 不用客户端 可下载小说
// @description:en    Fanqie Novel Reading, No Need for a Client, Novels Available for Download
// @author            ibxff, SmashPhoenix272
// @license           MIT License
// @match             https://fanqienovel.com/*
// @require           https://cdnjs.cloudflare.com/ajax/libs/FileSaver.js/2.0.5/FileSaver.min.js
// @icon              
// @grant             GM_xmlhttpRequest
// @updateURL
// ==/UserScript==

const styleElement = document.createElement("style");
const cssRule = `
    @keyframes hideAnimation {
      0% {
        opacity: 1;
      }
      50% {
        opacity: 0.75;
      }
      100% {
        opacity: 0;
        display: none;
      }
    }

    option:checked {
        background-color: #ffb144;
        color: white;
    }
    `;

styleElement.innerHTML = cssRule;
document.head.appendChild(styleElement);

function hideElement(ele) {
  if (!ele) return;
  ele.style.animation = "hideAnimation 1.5s ease";
  ele.addEventListener("animationend", function () {
    ele.style.display = "none";
  });
}

function sleep(ms) {
  return new Promise(resolve => setTimeout(resolve, ms));
}

const mark = (ele) => {
  if (ele) ele.style.boxShadow = "0px 0px 50px rgba(0, 0, 0, 0.2)";
};

// Function to extract content from different response formats
function extractContent(data) {
    if (data.code === 200 && data.data && data.data.content) {
        return data.data.content;
    }
    throw new Error('Unexpected response format');
}

(function() {
    'use strict';
    const path = window.location.href.match(/\/([^/]+)\/\d/)?.[1];

    switch(path){
        case 'reader':
            const toolbar = document.querySelector("#app > div > div > div > div.reader-toolbar > div > div.reader-toolbar-item.reader-toolbar-item-download");
            const text = toolbar?.querySelector('div:nth-child(2)');
            
            if (toolbar && text) {
                mark(toolbar);
                text.innerHTML = 'Processing';
            }

            document.title = document.title.replace(/在线免费阅读_番茄小说官网$/, '')

            var currentURL = window.location.href
            setInterval(() => window.location.href !== currentURL ? location.reload() : null, 1000);

            // Get content div for both PC and mobile layouts
            let cdiv;

            // Try PC layout first
            cdiv = document.getElementsByClassName('muye-reader-content noselect')[0];
            if (cdiv) {
                console.log('Found PC layout content div');
                cdiv.classList = cdiv.classList[0];
            } else {
                // Try mobile layout
                console.log('Trying mobile layout');
                const html0 = document.getElementById('html_0');
                if (!html0) {
                    console.log('Could not find html_0 for mobile layout');
                    return;
                }
                cdiv = html0.children[2];
                if (!cdiv) {
                    console.log('Could not find content div in mobile layout');
                    return;
                }
                console.log('Found mobile layout content div');
            }

            console.log('Content div found:', cdiv);

            console.log('Getting chapter ID...');
            const url = window.location.href;
            const regex = /\/(\d+)/;
            const match = url.match(regex);
            if (!match) {
                console.log('Could not extract chapter ID');
                return;
            }

            const extractedId = match[1];
            console.log('Chapter ID:', extractedId);
            const apiUrl = `https://api.cenguigui.cn/api/tomato/content.php?item_id=${extractedId}`;

            console.log('Making API request...');
            GM_xmlhttpRequest({
                method: "GET",
                url: apiUrl,
                onload: function(response) {
                    if (response.status === 200) {
                        try {
                            const data = JSON.parse(response.responseText);
                            const content = extractContent(data);
                            console.log(content);
                            document.getElementsByClassName('muye-to-fanqie')[0]?.remove();
                            document.getElementsByClassName('pay-page')[0]?.remove();
                            cdiv.innerHTML = content.replace(/\n/g, "</p><p>").replace(/  /g,"").replace(/(.*?)\n/, '');

                            document.getElementsByClassName('muye-to-fanqie')[0]?.remove();
                            document.getElementsByClassName('pay-page')[0]?.remove();
                            cdiv.innerHTML = content.replace(/\n/g, "</p><p>").replace(/  /g,"").replace(/(.*?)\n/, '');

                            // Clean up any remaining pay-related elements
                            document.getElementById('html_0')?.classList.remove('pay-page-html');

                            if (toolbar && text) {
                                toolbar.style.backgroundColor = '#B0E57C'
                                text.innerHTML = 'Successed'
                                hideElement(toolbar)
                            }
                        } catch (error) {
                            console.error('Error processing content:', error);
                            if (toolbar && text) {
                                toolbar.style.backgroundColor = 'pink'
                                text.innerHTML = 'Failed'
                                hideElement(toolbar)
                            }
                        }
                    }
                },
                onerror: function(error) {
                    if (toolbar && text) {
                        toolbar.style.backgroundColor = 'pink'
                        text.innerHTML = 'Failed'
                        hideElement(toolbar)
                    }
                    console.error(`Fetch error: ${error}`);
                }
            });
     break;

     case 'page':
            const infoName = document.querySelector("#app > div > div.muye.muye-page > div > div.page-wrap > div > div.page-header-info > div.info > div.info-name > h1")?.innerHTML;
            const authorName = document.querySelector(".author-name-text")?.innerHTML;
            const totalChapters = document.querySelector(".page-directory-header h3")?.textContent.match(/(\d+)章/)?.[1] || '';
            const infoLabels = Array.from(document.querySelectorAll('.info-label span')).map(span => span.textContent).join(' ');
            const wordCount = document.querySelector('.info-count-word')?.textContent.trim();
            const lastUpdate = document.querySelector('.info-last')?.textContent.trim();
            const abstract = document.querySelector("#app > div > div.muye.muye-page > div > div.page-body-wrap > div > div.page-abstract-content > p")?.innerHTML;

            var content = 'Using Free Fanqie script download\n\n' +
                         '作者:' + authorName + '\n' +
                         '书名:' + infoName + '\n' +
                         '标签:' + infoLabels + '\n' +
                         '字数:' + wordCount + '\n' +
                         '更新:' + lastUpdate + '\n\n' +
                         '简介:' + abstract + '\n'
            content = content.replace(/undefined|null|NaN/g,'')

            const processContentForDownload = (content) => {
                return content
                    .replace(/<[^>]+>/g, '')  // Remove all HTML tags
                    .replace(/&nbsp;/g, ' ')  // Replace HTML spaces
                    .replace(/&lt;/g, '<')    // Replace HTML entities
                    .replace(/&gt;/g, '>')
                    .replace(/&amp;/g, '&')
                    .replace(/&quot;/g, '"')
                    .replace(/\n\s*\n/g, '\n')  // Remove multiple consecutive newlines
                    .split('\n')  // Split into paragraphs
                    .map(para => '  ' + para)  // Add 2 spaces to each paragraph
                    .join('\n')  // Join back together
                    .trim();
            };

            sleep(1500).then(()=>{
                document.querySelector("#app > div > div.muye.muye-page > div > div.page-wrap > div > div.page-header-info > div.info > div.download-icon.muyeicon-tomato")?.remove()
                const download = document.querySelector("#app > div > div.muye.muye-page > div > div.page-wrap > div > div.page-header-info > div.info > a")
                if (!download) return;
                
                const downloadSpan = download.querySelector('button > span')
                if (downloadSpan) downloadSpan.innerHTML = '*Download Novel'
                download.href = 'javascript:void 0'

                const parentElement = document.querySelector("#app > div > div.muye.muye-page > div > div.page-wrap > div > div.page-header-info > div.info");
                if (!parentElement) return;

                const selectElement = document.createElement("select");
                selectElement.className = "byte-btn byte-btn-primary byte-btn-size-large byte-btn-shape-square muye-button";
                const options = [
                    {text: "UTF-8" },
                    {text: "GBK" },
                    {text: "UNICODE" },
                    {text:'UTF-16'},
                    {text:'ASCII'}
                ];
                options.forEach(function(optionData) {
                    var option = document.createElement("option");
                    option.text = optionData.text;
                    option.value = optionData.text
                    selectElement.appendChild(option);
                });
                selectElement.style.position = "absolute";
                selectElement.style.left = "320px";
                selectElement.style.bottom = "0px";
                selectElement.style.height = "32px";
                selectElement.style.width = "80px";
                selectElement.style.fontSize = "15px";
                parentElement.appendChild(selectElement);

                const books = Array.from(document.getElementsByClassName('chapter-item'))
                var accomplish = false

                function next(){
                    if (!books.length) return;
                    const ele = books[0].querySelector('a')
                    if (!ele) return;

                    ele.style.border = "3px solid navajowhite"
                    ele.style.borderRadius = "5px"
                    ele.style.backgroundColor = "navajowhite"
                    const url = ele.href;
                    console.log(url)
                    const regex = /\/(\d+)/;
                    const match = url.match(regex);
                    if (!match) return;

                    const extractedId = match[1];
                    const apiURL = `https://api.cenguigui.cn/api/tomato/content.php?item_id=${extractedId}`;
                    const charset = selectElement.value

                    content += '\n\n'
                    GM_xmlhttpRequest({
                        method: "GET",
                        url: apiURL,
                        'Content-Type': "application/json; charset="+charset,
                        onload: function(response) {
                            if (response.status === 200) {
                                try{
                                    const data = JSON.parse(response.responseText);
                                    const chapterContent = extractContent(data);
                                    const processedContent = processContentForDownload(chapterContent);
                                    content += processedContent;
                                    ele.style.backgroundColor = '#D2F9D1'
                                    ele.style.border = "2px solid #D2F9D1"
                                    books.shift()
                                    console.log(books)
                                    if(!books.length){
                                        console.log('Download succesfully, saving')
                                        console.log(charset)
                                        const blob = new Blob([new TextEncoder(charset).encode(content)], { type: `text/plain;charset=`+charset });
                                        const fileName = totalChapters ? `${infoName}_${totalChapters}章.txt` : `${infoName}.txt`;
                                        saveAs(blob, fileName);
                                        return
                                    }
                                    else{
                                        next()
                                    }
                                }
                                catch(e){
                                    ele.style.backgroundColor = 'pink'
                                    ele.style.border = "2px solid pink"
                                    next()
                                }
                            }
                        },
                        onerror: function(error) {
                            console.error(`Fetch error: ${error}`);
                            ele.style.backgroundColor = 'pink'
                            ele.style.border = "2px solid pink"
                            next()
                        },
                        ontimeout: function(error) {
                            console.error(`Fetch error: ${error}`);
                            ele.style.backgroundColor = 'pink'
                            ele.style.border = "2px solid pink"
                            next()
                        }
                    });
                }
                download.addEventListener('click', next)
                download.addEventListener('click', ()=>{
                    download.style.display = 'none';
                    selectElement.style.display = 'none'
                })
            })
      break;
    }
})();