// ==UserScript== // @name Xbanxia Novel Downloader (Background Fetch) // @namespace http://tampermonkey.net/ // @version 1.2 // UI 크기 및 체크박스 미세 조정 // @description Downloads novel text from xbanxia.cc by fetching chapters in the background. Handles rate-limiting (429) and forbidden (403) errors with retries. // @author You // @match https://www.xbanxia.cc/books/* // @grant GM_download // @grant GM_xmlhttpRequest // @require https://code.jquery.com/jquery-3.6.0.min.js // @run-at document-idle // ==/UserScript== (function() { 'use strict'; // ======================================================================= // 1. 선택자 및 상수 // ======================================================================= const episodeListSelector = '.book-list ul li a'; const textToRemove = ''; let isCheckAll = false; let countLabel; // 다운로드 제어 상수 const BASE_DELAY_MS = 1500; const INITIAL_PAGE_DELAY_MS = 500; const MAX_RETRIES = 5; const MIN_FILE_SIZE_KB = 1; function log(message) { console.log(`[Tampermonkey: Xbanxia Downloader] ${message}`); } // ======================================================================= // 2. UI 및 보조 함수 (CSS 수정 포함) // ======================================================================= function addGlobalStyle() { if ($('#tampermonkey-shuba-style').length > 0) return; const style = document.createElement('style'); style.id = 'tampermonkey-shuba-style'; style.innerHTML = ` /* UI 컨트롤 요소 기본 스타일 */ .tampermonkey-ui-element { display: inline-block; margin-left: 5px; vertical-align: middle; } /* 중앙 상단 카운트 레이블 (크기 증가 및 Z-INDEX 강화) */ #tampermonkey-count-label { position: fixed !important; top: 10px !important; left: 50% !important; transform: translateX(-50%) !important; z-index: 99999 !important; background-color: #007bff !important; color: white !important; padding: 10px 15px !important; /* 패딩 증가 */ border-radius: 4px !important; font-family: sans-serif !important; font-size: 16px !important; /* 폰트 크기 증가 */ font-weight: bold !important; /* 폰트 굵게 */ box-shadow: 0 2px 5px rgba(0,0,0,0.2) !important; pointer-events: none !important; min-width: 150px !important; /* 최소 너비 확보 */ text-align: center !important; } /* 컨트롤 컨테이너 */ #tampermonkey-controls-container { margin-bottom: 10px; padding: 10px; background-color: #f0f0f0; border-radius: 5px; display: flex; align-items: center; flex-wrap: wrap; gap: 5px; } /* 하단 다운로드 버튼 */ #tampermonkey-download-button-bottom { position: fixed !important; bottom: 20px !important; left: 50% !important; transform: translateX(-50%) !important; z-index: 9998 !important; background-color: #28a745 !important; border: none !important; color: white !important; padding: 15px 32px !important; text-align: center !important; font-size: 16px !important; cursor: pointer !important; border-radius: 8px !important; } #tampermonkey-download-button-bottom:disabled, #tampermonkey-download-button-top:disabled { background-color: #cccccc !important; cursor: not-allowed !important; } /* 챕터 리스트 레이아웃 수정 */ .book-list ul li { display: flex !important; align-items: center !important; line-height: normal !important; height: auto !important; } /* 체크박스 미세 조정: 아래로 1px 이동 */ .book-list ul li input[type="checkbox"] { flex-shrink: 0 !important; margin-top: 1px !important; /* 1px 아래로 이동 */ } .book-list ul li a { white-space: nowrap !important; overflow: hidden !important; text-overflow: ellipsis !important; } `; document.head.appendChild(style); } function updateCheckedCount() { if (!countLabel) return; const checkedCount = $(episodeListSelector).parent().find('input[type="checkbox"]:checked').length; countLabel.text(`선택된 항목 = ${checkedCount}개`); } function addCheckboxes() { let episodeIndex = 1; $(episodeListSelector).each(function() { const link = $(this); const listItem = link.parent(); if (listItem.find('input[type="checkbox"]').length > 0) return; const checkbox = $('').css('margin-right', '5px'); checkbox.on('change', updateCheckedCount); const chapterUrl = new URL(link.attr('href'), window.location.href).href; link.data({ index: episodeIndex++, title: link.text().trim(), url: chapterUrl }); listItem.prepend(checkbox); }); updateCheckedCount(); } function createListPageUI() { if ($('#tampermonkey-controls-container').length > 0) return; addGlobalStyle(); countLabel = $('').attr('id', 'tampermonkey-count-label'); $('body').append(countLabel); const checkAllButton = $('모두 선택').addClass('tampermonkey-ui-element'); const uncheckAllButton = $('모두 해제').addClass('tampermonkey-ui-element'); const downloadButton = $('선택 다운로드').attr('id', 'tampermonkey-download-button-top').addClass('tampermonkey-ui-element'); const rangeInput = $('').addClass('tampermonkey-ui-element'); const checkTBButton = $('범위 선택').addClass('tampermonkey-ui-element'); const uncheckTBButton = $('범위 해제').addClass('tampermonkey-ui-element'); const betweenChkButton = $('선택 사이').addClass('tampermonkey-ui-element'); checkAllButton.on('click', function() { isCheckAll = !isCheckAll; $(episodeListSelector).parent().find('input[type="checkbox"]').prop('checked', isCheckAll); $(this).text(isCheckAll ? '모두 해제' : '모두 선택'); updateCheckedCount(); }); uncheckAllButton.on('click', function() { $(episodeListSelector).parent().find('input[type="checkbox"]').prop('checked', false); isCheckAll = false; checkAllButton.text('모두 선택'); updateCheckedCount(); }); downloadButton.on('click', startBackgroundDownload); function handleRangeAction(shouldCheck, start, end) { if (start === undefined || end === undefined) { const rangeString = rangeInput.val(); const range = rangeString.split('-').map(s => parseInt(s.trim(), 10)); if (range.length < 1 || isNaN(range[0])) return; start = range.length > 1 && !isNaN(range[1]) ? Math.min(...range) : range[0]; end = range.length > 1 && !isNaN(range[1]) ? Math.max(...range) : range[0]; } $(episodeListSelector).filter((i, el) => { const index = $(el).data('index'); return index >= start && index <= end; }).parent().find('input[type="checkbox"]').prop('checked', shouldCheck); updateCheckedCount(); } checkTBButton.on('click', () => handleRangeAction(true)); uncheckTBButton.on('click', () => handleRangeAction(false)); betweenChkButton.on('click', function() { const checkedIndexes = $(episodeListSelector).parent().find('input:checked').siblings('a').map((i, el) => $(el).data('index')).get(); if (checkedIndexes.length < 2) { alert("범위를 지정하려면 최소 두 개의 체크박스를 선택해주세요."); return; } handleRangeAction(true, Math.min(...checkedIndexes), Math.max(...checkedIndexes)); }); const controlsContainer = $('
아래 요청한 소설,
60년도로 회귀 부자되고 아이키우기 소설...
검색해보니 이 사이트가 나와서 일단 올립니다.
링크:
https://www.xbanxia.cc/books/79393.html
