Fanqie Novel Free Reading

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

Versão de: 27/03/2025. Veja: a última versão.

// ==UserScript==
// @name              Fanqie Novel Free Reading
// @namespace         https://github.com/SmashPhoenix272
// @version           5.0
// @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';
    switch(window.location.href.match(/\/([^/]+)\/\d/)[1]){
        case 'reader':
            const toolbar = document.querySelector("#app > div > div > div > div.reader-toolbar > div > div.reader-toolbar-item.reader-toolbar-item-download") ||
                           document.querySelector(".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);

            const html0 = document.getElementById('html_0')
            if (!html0) return;

            // Get content div (third div under html_0 for mobile layout)
            const cdiv = html0.children[2] || document.querySelector('.muye-reader-content.noselect')
            if (!cdiv) return;

            // Store original styles and classes
            const parentDiv = cdiv.closest('.muye-reader-box') || html0
            const parentClasses = parentDiv?.className || ''
            const parentStyles = parentDiv?.getAttribute('style') || ''
            const contentClasses = cdiv.className || ''

            const url = window.location.href;
            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}`;

            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/, '');

                            // Remove pay-page-html class from html_0 div
                            const html0Element = document.getElementById('html_0');
                            if (html0Element && html0Element.classList.contains('pay-page-html')) {
                                html0Element.classList.remove('pay-page-html');
                            }
                            
                            // Restore classes and styles (except for html_0)
                            if (contentClasses) cdiv.className = contentClasses
                            if (parentDiv && parentDiv.id !== 'html_0') {
                                if (parentClasses) parentDiv.className = parentClasses
                                if (parentStyles) parentDiv.setAttribute('style', parentStyles)
                            }

                            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,'')
            console.log(content)

            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;
    }
})();