Greasy Fork is available in English.
知网文献、硕博论文PDF批量下载,下载硕博论文章节目录
// ==UserScript==
// @name 中国知网CNKI硕博论文PDF下载
// @version 4.0.1
// @namespace http://greasyfork.icu/users/244539
// @icon https://www.cnki.net/favicon.ico
// @description 知网文献、硕博论文PDF批量下载,下载硕博论文章节目录
// @author @zoglmk
// @match *://*.cnki.net/*
// @run-at document-idle
// @grant unsafeWindow
// @grant GM_getValue
// @grant GM_setValue
// @grant GM_xmlhttpRequest
// @grant GM_download
// @grant GM_registerMenuCommand
// ==/UserScript==
// 如果没有显示批量下载的按钮,大概率是因为你的知网地址没有匹配到。
// 解决办法:将你的域名按照上面的 @match 的格式,添加到后面,保存脚本,刷新网页,应该就可以了。(最后要加个*)
// 例:// @match https://webvpn.cueb.edu.cn/*
(function() {
'use strict';
let useWebVPN = GM_getValue('useWebVPN', false);
let fetchLevels = GM_getValue('fetchLevels', true);
const config = {"buttonText":"批量下载 PDF","buttonColor":"#3b82f6","buttonTextColor":"#ffffff","showIcon":true,"position":"floating-bottom-right","borderRadius":"9999px","autoDownload":false};
// Global Caches
const journalLevelCache = new Map();
const journalRequestCache = new Map();
// Sort State
let sortState = {
field: null,
direction: 'desc'
};
// --- INIT & CSS ---
function init() {
const style = document.createElement('style');
style.textContent = `
/* --- CNKI Beautified UI Styles --- */
:root {
--cnki-primary: #3b82f6;
--cnki-text-on-primary: #ffffff;
--cnki-radius: 20px;
--cnki-primary-light: color-mix(in srgb, var(--cnki-primary), transparent 90%);
--cnki-primary-border: color-mix(in srgb, var(--cnki-primary), transparent 80%);
}
.cnki-ui-overlay {
position: fixed;
top: 0;
left: 0;
width: 100%;
height: 100%;
background-color: rgba(0, 0, 0, 0.4);
backdrop-filter: blur(2px);
z-index: 2147483647; /* Max z-index */
display: flex;
justify-content: center;
align-items: center;
font-family: -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, Helvetica, Arial, sans-serif;
opacity: 0;
animation: cnkiFadeIn 0.2s forwards;
}
@keyframes cnkiFadeIn {
to { opacity: 1; }
}
.cnki-ui-modal {
background: white;
width: 90vw; /* Responsive width */
max-width: 1450px;
height: 85vh;
border-radius: 12px;
box-shadow: 0 20px 25px -5px rgba(0, 0, 0, 0.1), 0 10px 10px -5px rgba(0, 0, 0, 0.04);
display: flex;
flex-direction: column;
overflow: hidden;
transform: scale(0.95);
animation: cnkiScaleIn 0.2s forwards;
}
@keyframes cnkiScaleIn {
to { transform: scale(1); }
}
.cnki-ui-header {
padding: 16px 24px;
border-bottom: 1px solid #e5e7eb;
display: flex;
justify-content: space-between;
align-items: center;
background: #ffffff;
}
.cnki-ui-title {
font-size: 18px;
font-weight: 600;
color: #111827;
display: flex;
align-items: center;
gap: 10px;
}
.cnki-version-tag {
font-size: 12px;
padding: 2px 6px;
border-radius: 4px;
background-color: #f3f4f6;
color: #6b7280;
border: 1px solid #e5e7eb;
font-weight: normal;
margin-left: 6px;
}
.cnki-ui-close {
background: transparent;
border: none;
color: #9ca3af;
cursor: pointer;
font-size: 24px;
line-height: 1;
padding: 4px;
border-radius: 4px;
transition: color 0.2s;
}
.cnki-ui-close:hover { color: #374151; background: #f3f4f6; }
.cnki-ui-content {
flex: 1;
overflow: hidden;
display: flex;
flex-direction: column;
background: #f9fafb;
}
.cnki-ui-toolbar {
padding: 12px 24px;
background: white;
border-bottom: 1px solid #f3f4f6;
display: flex;
gap: 10px;
flex-wrap: wrap;
align-items: center;
}
.cnki-toolbar-tips {
margin-left: auto;
font-size: 12px;
color: #b91c1c;
background-color: #fef2f2;
border: 1px solid #fecaca;
padding: 8px 16px;
border-radius: 6px;
line-height: 1.6;
text-align: left;
max-width: 650px;
box-shadow: 0 1px 2px rgba(0,0,0,0.05);
}
.cnki-ui-btn {
padding: 8px 16px;
border-radius: 6px;
font-size: 13px;
font-weight: 500;
cursor: pointer;
border: 1px solid transparent;
transition: all 0.2s;
display: inline-flex;
align-items: center;
gap: 6px;
white-space: nowrap;
text-decoration: none;
}
.cnki-btn-primary {
background-color: var(--cnki-primary);
color: var(--cnki-text-on-primary);
box-shadow: 0 1px 2px rgba(0,0,0,0.05);
}
.cnki-btn-primary:hover { filter: brightness(110%); transform: translateY(-1px); }
.cnki-btn-secondary {
background-color: white;
border-color: #d1d5db;
color: #374151;
}
.cnki-btn-secondary:hover { background-color: #f9fafb; border-color: #9ca3af; color: #111827; }
.cnki-btn-danger {
background-color: #fff1f2;
color: #be123c;
border-color: #fecdd3;
}
.cnki-btn-danger:hover { background-color: #ffe4e6; border-color: #fda4af; }
.cnki-btn-sm {
padding: 4px 10px;
font-size: 12px;
border: 1px solid var(--cnki-primary);
color: var(--cnki-primary);
background: #f0f7ff;
border-radius: 4px;
cursor: pointer;
transition: all 0.2s;
min-width: 70px;
text-align: center;
display: inline-block;
}
.cnki-btn-sm:hover {
background: var(--cnki-primary);
color: #fff;
}
.cnki-btn-sm:disabled {
opacity: 0.6;
cursor: not-allowed;
}
.cnki-btn-icon {
padding: 2px;
border-radius: 4px;
color: #9ca3af;
cursor: pointer;
background: transparent;
border: none;
display: inline-flex;
align-items: center;
vertical-align: middle;
margin-left: 6px;
opacity: 0;
transition: all 0.2s;
}
.cnki-title-wrapper:hover .cnki-btn-icon { opacity: 1; }
.cnki-btn-icon:hover { color: var(--cnki-primary); background: #f3f4f6; }
.cnki-table-container {
flex: 1;
overflow-y: auto;
padding: 0;
position: relative;
}
.cnki-data-table {
width: 100%;
border-collapse: separate;
border-spacing: 0;
font-size: 14px; /* Comfort Size */
table-layout: fixed;
}
.cnki-data-table th {
text-align: left;
padding: 10px 12px; /* Comfort Padding */
background: #f9fafb;
color: #6b7280;
font-weight: 600;
border-bottom: 1px solid #e5e7eb;
position: sticky;
top: 0;
z-index: 10;
text-transform: uppercase;
font-size: 13px;
letter-spacing: 0.05em;
white-space: nowrap;
overflow: hidden;
text-overflow: ellipsis;
}
.cnki-sortable {
cursor: pointer;
user-select: none;
transition: background-color 0.2s;
}
.cnki-sortable:hover {
background-color: #f3f4f6;
color: var(--cnki-primary);
}
.cnki-sort-icon {
display: inline-block;
margin-left: 4px;
font-size: 10px;
width: 10px;
color: #6b7280;
}
.cnki-sort-active .cnki-sort-icon {
color: var(--cnki-primary);
}
/* Center Alignment */
.cnki-col-center { text-align: center !important; vertical-align: middle !important; }
.cnki-item-check {
margin: 0 auto;
display: block;
width: 16px;
height: 16px;
cursor: pointer;
}
.cnki-data-table td {
padding: 10px 12px; /* Comfort Padding */
border-bottom: 1px solid #f3f4f6;
color: #374151;
vertical-align: middle;
background: white;
word-break: break-all;
}
.cnki-data-table tr:hover td { background-color: #f8fafc; }
.cnki-title-wrapper {
display: flex;
align-items: center;
font-weight: 600;
color: #1f2937;
margin-bottom: 4px;
font-size: 14px;
}
.cnki-highlight {
font-weight: bold;
color: #e50012;
}
.cnki-author {
font-size: 12px;
color: #6b7280;
line-height: 1.4;
max-height: 2.8em;
overflow: hidden;
text-overflow: ellipsis;
display: -webkit-box;
-webkit-line-clamp: 2;
-webkit-box-orient: vertical;
}
/* Standard Keyword Tags */
.cnki-tag {
display: inline-block;
padding: 2px 8px;
border-radius: 99px;
background: var(--cnki-primary-light);
color: var(--cnki-primary);
border: 1px solid var(--cnki-primary-border);
font-size: 12px; /* Comfort Size */
margin-right: 4px;
margin-bottom: 4px;
font-weight: 500;
white-space: nowrap;
}
/* Distinct Level Tags (Red) */
.cnki-level-tag {
display: inline-block;
padding: 2px 8px;
border-radius: 4px;
background: #fff1f2;
color: #be123c;
border: 1px solid #fecdd3;
font-size: 12px; /* Comfort Size */
margin-right: 4px;
margin-bottom: 4px;
white-space: nowrap;
line-height: 1.2;
}
.cnki-level-loading { color: #9ca3af; font-size: 11px; }
.cnki-meta-text {
color: #6b7280;
font-size: 12px;
}
.cnki-status-success { color: #16a34a; font-size: 12px; font-weight: 600; }
.cnki-status-error { color: #dc2626; font-size: 12px; font-weight: 600; }
.cnki-status-loading { color: #ea580c; font-size: 12px; font-weight: 600; }
.cnki-status-pending { display: none; }
.cnki-footer {
padding: 12px 24px;
background: white;
border-top: 1px solid #e5e7eb;
font-size: 13px;
color: #6b7280;
display: flex;
justify-content: space-between;
align-items: center;
flex-wrap: wrap;
gap: 10px;
}
.cnki-footer a { color: var(--cnki-primary); text-decoration: none; cursor: pointer; }
.cnki-footer a:hover { text-decoration: underline; }
/* QR Code Hover Styles */
.cnki-qr-link {
position: relative;
display: inline-block;
cursor: pointer;
color: var(--cnki-primary);
}
/* Bridge to prevent hover loss */
.cnki-qr-link::before {
content: '';
position: absolute;
bottom: 100%;
left: 0;
width: 100%;
height: 20px;
}
.cnki-qr-box {
display: none;
position: absolute;
bottom: 100%;
left: 50%;
transform: translateX(-50%);
background: white;
padding: 10px;
box-shadow: 0 4px 15px rgba(0,0,0,0.15);
border-radius: 8px;
border: 1px solid #f3f4f6;
z-index: 100;
text-align: center;
width: 130px;
margin-bottom: 10px;
}
.cnki-qr-link:hover .cnki-qr-box {
display: block;
}
.cnki-qr-box img {
width: 110px;
height: 110px;
display: block;
margin: 0 auto 5px auto;
background: #f9fafb;
}
.cnki-qr-box div {
font-size: 11px;
color: #6b7280;
}
/* Arrow for QR Box */
.cnki-qr-box::after {
content: '';
position: absolute;
top: 100%;
left: 50%;
margin-left: -6px;
border-width: 6px;
border-style: solid;
border-color: white transparent transparent transparent;
}
.cnki-main-btn {
position: fixed;
z-index: 9999;
background-color: var(--cnki-primary) !important;
color: var(--cnki-text-on-primary) !important;
border: none;
padding: 10px 20px;
border-radius: var(--cnki-radius);
font-size: 14px;
font-weight: 600;
cursor: pointer;
box-shadow: 0 4px 12px rgba(0, 0, 0, 0.15);
transition: transform 0.2s, box-shadow 0.2s;
display: flex;
align-items: center;
gap: 8px;
}
.cnki-main-btn:hover {
transform: translateY(-2px);
box-shadow: 0 8px 16px rgba(0, 0, 0, 0.2);
}
.cnki-hidden {
display: none !important;
}
/* Positions */
.cnki-pos-floating-bottom-right { bottom: 40px; right: 40px; }
.cnki-pos-floating-top-right { top: 120px; right: 40px; }
.cnki-pos-floating-center-right { top: 50%; right: 20px; transform: translateY(-50%); }
.cnki-pos-inline-title { position: static; display: inline-flex; margin-left: 10px; padding: 6px 12px; font-size: 12px; }
.cnki-pos-inline-toolbar { position: static; display: inline-flex; margin-left: 0; padding: 6px 12px; font-size: 12px; }
/* Utilities */
.cnki-loading-toast {
position: fixed;
top: 50%;
left: 50%;
transform: translate(-50%, -50%);
background: #374151;
color: white;
padding: 12px 30px;
border-radius: 8px;
font-size: 15px;
font-weight: 500;
z-index: 2147483648;
box-shadow: 0 10px 25px rgba(0,0,0,0.2);
display: flex;
align-items: center;
gap: 10px;
text-align: center;
animation: fadeIn 0.2s ease-out;
border: 1px solid rgba(255,255,255,0.1);
}
@keyframes fadeIn { from { opacity: 0; transform: translate(-50%, -45%); } to { opacity: 1; transform: translate(-50%, -50%); } }
`;
document.head.appendChild(style);
const url = window.location.href.toLowerCase();
// Add main button to search pages
if (url.includes('defaultresult') || url.includes('advsearch') || url.includes('search')) {
createMainButton();
}
// Add chapter download to abstract pages
if (url.includes('abstract') || url.includes('kns8s')) {
addCategoryDownloadButton();
}
// Register Menu
if (typeof GM_registerMenuCommand !== 'undefined') {
GM_registerMenuCommand("打开批量下载助手", () => {
if (!localStorage.getItem('cnkiFirstTimePopupShown_v3_9')) {
showHelpModal();
} else {
openDashboard();
loadSavedData();
}
});
}
// Keydown for Escape
document.addEventListener('keydown', (e) => {
if (e.key === 'Escape') {
closeDashboard();
}
});
}
// --- HELPER: GM Fetch ---
function gmFetch(url) {
return new Promise((resolve, reject) => {
GM_xmlhttpRequest({
method: 'GET',
url: url,
onload: (res) => resolve(res.responseText),
onerror: (err) => reject(err)
});
});
}
// --- HELPER: Journal Level Fetcher ---
async function fetchJournalLevel(url) {
if (!url) return '无';
if (journalLevelCache.has(url)) return journalLevelCache.get(url);
if (journalRequestCache.has(url)) return journalRequestCache.get(url);
const promise = gmFetch(url).then(html => {
if (!html) return '无';
const doc = new DOMParser().parseFromString(html, 'text/html');
const spans = Array.from(doc.querySelectorAll('.journalType.journalType2 span'));
const res = spans.map(s => s.textContent.trim()).filter(Boolean).join('/');
return res || '无';
}).catch((e) => {
console.warn('Journal fetch failed', e);
return '无';
}).then(res => {
journalLevelCache.set(url, res);
journalRequestCache.delete(url);
return res;
});
journalRequestCache.set(url, promise);
return promise;
}
// --- HELPER: Keyword Highlighting ---
function getSearchWords() {
const input = document.getElementById('txt_search');
if (input && input.value) {
return input.value.trim().split(/\s+/).filter(Boolean);
}
return [];
}
function highlightText(text) {
if (!text) return text;
const words = getSearchWords();
if (words.length === 0) return text;
// Simple and robust string replacement logic
let processedText = text;
words.forEach(word => {
// Escape special regex chars in keyword
const escapedWord = word.replace(/[.*+?^$${}()|[\]\\]/g, '\\$&');
const regex = new RegExp('(' + escapedWord + ')', 'gi');
processedText = processedText.replace(regex, '<span class="cnki-highlight">$1</span>');
});
return processedText;
}
// --- UI COMPONENTS ---
function createMainButton() {
if (document.getElementById('cnki-main-btn')) return;
const btn = document.createElement('button');
btn.id = 'cnki-main-btn';
btn.className = 'cnki-main-btn cnki-pos-' + config.position;
btn.innerHTML = `<svg xmlns="http://www.w3.org/2000/svg" width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"><path d="M21 15v4a2 2 0 0 1-2 2H5a2 2 0 0 1-2-2v-4"/><polyline points="7 10 12 15 17 10"/><line x1="12" y1="15" x2="12" y2="3"/></svg> <span>${config.buttonText}</span>`;
btn.onclick = () => {
if (!localStorage.getItem('cnkiFirstTimePopupShown_v3_9')) {
showHelpModal();
} else {
openDashboard();
loadSavedData();
}
};
// Handle inline injection
if (config.position === 'inline-title') {
const target = document.querySelector('.result-count');
if (target) target.parentNode.insertBefore(btn, target.nextSibling);
else document.body.appendChild(btn);
} else {
document.body.appendChild(btn);
}
}
function createLoading(text, duration = 2000) {
// remove existing
const existing = document.getElementById('cnki-toast');
if (existing) existing.remove();
const toast = document.createElement('div');
toast.id = 'cnki-toast';
toast.className = 'cnki-loading-toast';
toast.innerHTML = text;
document.body.appendChild(toast);
if (duration > 0) {
setTimeout(() => {
if (toast.parentElement) toast.parentElement.removeChild(toast);
}, duration);
}
return toast;
}
// --- DASHBOARD LOGIC ---
function openDashboard() {
if (document.getElementById('cnki-modal-overlay')) return;
const overlay = document.createElement('div');
overlay.id = 'cnki-modal-overlay';
overlay.className = 'cnki-ui-overlay';
overlay.innerHTML = `
<div class="cnki-ui-modal">
<div class="cnki-ui-header">
<div class="cnki-ui-title">
<svg xmlns="http://www.w3.org/2000/svg" width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"><path d="M21 15v4a2 2 0 0 1-2 2H5a2 2 0 0 1-2-2v-4"/><polyline points="7 10 12 15 17 10"/><line x1="12" y1="15" x2="12" y2="3"/></svg> 批量下载助手 <span class="cnki-version-tag">v4.0</span>
</div>
<button id="cnki-close-btn" class="cnki-ui-close" title="关闭 (Esc)">×</button>
</div>
<div class="cnki-ui-toolbar">
<button id="cnki-get-links" class="cnki-ui-btn cnki-btn-primary" title="获取当前页面的所有论文链接">获取本页链接</button>
<button id="cnki-batch-dl" class="cnki-ui-btn cnki-btn-primary" title="下载所有勾选的论文">批量下载</button>
<button id="cnki-clear" class="cnki-ui-btn cnki-btn-danger" title="清空所有已获取的数据">清空数据</button>
<button id="cnki-webvpn" class="cnki-ui-btn ${useWebVPN ? 'cnki-btn-primary' : 'cnki-btn-secondary'}" title="切换WebVPN模式">WebVPN: ${useWebVPN ? '开启' : '关闭'}</button>
<button id="cnki-level-toggle" class="cnki-ui-btn ${fetchLevels ? 'cnki-btn-primary' : 'cnki-btn-secondary'}" title="切换是否显示期刊等级">期刊等级: ${fetchLevels ? '开启' : '关闭'}</button>
<div class="cnki-toolbar-tips">
1、关闭窗口数据不会消失,如需获取新数据,先点击清除数据。<br/>
2、通过WebVPN远程登录的用户获取出错请尝试开启开关。<br/>
3、问题反馈请提供控制台截图到留言区或公众号私信。
</div>
</div>
<div class="cnki-ui-content">
<div class="cnki-table-container">
<table class="cnki-data-table">
<thead>
<tr>
<th class="cnki-col-center" style="width: 40px"><input type="checkbox" id="cnki-select-all" class="cnki-item-check"></th>
<th class="cnki-col-center" style="width: 50px">No.</th>
<th>论文标题 / 作者</th>
<th style="width: 110px">期刊</th>
<th class="cnki-col-center ${fetchLevels ? '' : 'cnki-hidden'}" style="width: 90px" id="cnki-th-level">期刊等级</th>
<th class="cnki-col-center cnki-sortable" style="width: 85px" data-sort="date">发表时间 <span class="cnki-sort-icon">▲▼</span></th>
<th class="cnki-col-center cnki-sortable" style="width: 70px" data-sort="quote">被引 <span class="cnki-sort-icon">▲▼</span></th>
<th class="cnki-col-center cnki-sortable" style="width: 70px" data-sort="download">下载 <span class="cnki-sort-icon">▲▼</span></th>
<th class="cnki-col-center" style="width: 100px">状态</th>
<th style="width: 170px">关键词</th>
</tr>
</thead>
<tbody id="cnki-table-body"></tbody>
</table>
</div>
</div>
<div class="cnki-footer">
<span id="cnki-status-text">暂无数据</span>
<div style="display:flex; gap:10px; align-items:center;">
<span>欢迎关注:</span>
<div class="cnki-qr-link">
公众号: zgmgm
<div class="cnki-qr-box">
<img src="" />
<div>扫码关注</div>
</div>
</div>
<span>|</span>
<a href="https://www.youtube.com/channel/UCC1ExQh99BVTaPbGbGTcUzg/" target="_blank">YouTube</a>
<span> </span>
<a href="https://space.bilibili.com/7241318" target="_blank">B站</a>
</div>
</div>
</div>
`;
document.body.appendChild(overlay);
// Event Listeners
document.getElementById('cnki-close-btn').onclick = closeDashboard;
overlay.onclick = (e) => { if(e.target === overlay) closeDashboard(); };
document.getElementById('cnki-get-links').onclick = getLinks;
document.getElementById('cnki-batch-dl').onclick = downloadSelected;
document.getElementById('cnki-clear').onclick = clearData;
document.getElementById('cnki-select-all').onclick = (e) => {
const checked = e.target.checked;
document.querySelectorAll('.cnki-item-check:not(#cnki-select-all)').forEach(cb => {
if (!cb.disabled) cb.checked = checked;
});
updateStatus();
};
document.getElementById('cnki-webvpn').onclick = function() {
useWebVPN = !useWebVPN;
GM_setValue('useWebVPN', useWebVPN);
this.textContent = `WebVPN: ${useWebVPN ? '开启' : '关闭'}`;
this.className = `cnki-ui-btn ${useWebVPN ? 'cnki-btn-primary' : 'cnki-btn-secondary'}`;
createLoading(`WebVPN ${useWebVPN ? '已开启' : '已关闭'}`);
};
document.getElementById('cnki-level-toggle').onclick = function() {
fetchLevels = !fetchLevels;
GM_setValue('fetchLevels', fetchLevels);
this.textContent = `期刊等级: ${fetchLevels ? '开启' : '关闭'}`;
this.className = `cnki-ui-btn ${fetchLevels ? 'cnki-btn-primary' : 'cnki-btn-secondary'}`;
const th = document.getElementById('cnki-th-level');
const cells = document.querySelectorAll('.cnki-td-level');
if (fetchLevels) {
th.classList.remove('cnki-hidden');
cells.forEach(c => c.classList.remove('cnki-hidden'));
// Trigger auto-fetch if enabled and data exists
triggerLevelLoad();
} else {
th.classList.add('cnki-hidden');
cells.forEach(c => c.classList.add('cnki-hidden'));
}
};
// Header Sort Listeners
document.querySelectorAll('.cnki-sortable').forEach(th => {
th.onclick = () => handleSort(th.dataset.sort);
});
}
function closeDashboard() {
const overlay = document.getElementById('cnki-modal-overlay');
if (overlay) overlay.remove();
}
function showHelpModal() {
if(document.getElementById('cnki-help-overlay')) return;
const overlay = document.createElement('div');
overlay.id = 'cnki-help-overlay';
overlay.className = 'cnki-ui-overlay';
overlay.innerHTML = `
<div class="cnki-ui-modal" style="width: 600px; height: auto; max-height: 80vh;">
<div class="cnki-ui-header">
<div class="cnki-ui-title">使用说明 v4.0</div>
<button id="cnki-help-close" class="cnki-ui-close">×</button>
</div>
<div class="cnki-ui-content" style="padding: 20px; overflow-y: auto;">
<ul style="list-style: none; padding: 0; line-height: 1.8; color: #374151;">
<li style="margin-bottom: 8px;"><b>首先请确保脚本为最新版本。</b></li>
<li style="margin-bottom: 8px;">1. 脚本只能获取当前页的文献,知网默认每页20篇,如果想更多,可将每页数量设置为50。</li>
<li style="margin-bottom: 8px;">2. 如需清除已获取的数据 / 获取新数据,请先点击"<b>清除数据</b>"按钮。</li>
<li style="margin-bottom: 8px;">3. <b>如果只能下载一个,可能是浏览器拦截,允许弹出多窗口即可。</b></li>
<li style="margin-bottom: 8px;">4. 脚本只支持<b>新版知网</b>,不能在<b>隐私模式、无痕窗口</b>运行。</li>
<li style="margin-bottom: 8px;">5. 增加了批量下载延迟(2-5秒),以防止IP被封。</li>
<li style="margin-bottom: 8px;">6. 期刊等级加载为异步加载,不会卡顿页面。</li>
</ul>
</div>
<div class="cnki-footer" style="justify-content: flex-end;">
<button id="cnki-help-ok" class="cnki-ui-btn cnki-btn-primary">我知道了</button>
</div>
</div>
`;
document.body.appendChild(overlay);
const close = () => {
overlay.remove();
localStorage.setItem('cnkiFirstTimePopupShown_v3_9', 'true');
openDashboard();
loadSavedData();
};
document.getElementById('cnki-help-close').onclick = close;
document.getElementById('cnki-help-ok').onclick = close;
}
// --- DATA LOGIC ---
function convertToWebVPNLink(originalLink) {
if (!useWebVPN) return originalLink;
const webVPNDomain = window.location.origin;
const path = originalLink.replace(/^(https?:\/\/)?(www\.)?[^\/]+/, '');
return `${webVPNDomain}${path}`;
}
async function getLinks() {
const tbody = document.getElementById('cnki-table-body');
if (tbody && tbody.children.length > 0) {
createLoading('已有数据!请先清除数据。');
return;
}
// 1. Scan List
const rows = Array.from(document.querySelectorAll('.fz14'));
if (rows.length === 0) {
createLoading('未找到文献链接,请确保在搜索结果页。');
return;
}
const toast = createLoading('正在获取链接...', 0);
const results = [];
// Batch Fetch Logic (Native Fetch for speed)
const batchSize = 5;
for (let i = 0; i < rows.length; i += batchSize) {
const batch = rows.slice(i, i + batchSize);
toast.textContent = `正在获取链接... (${Math.round((i / rows.length) * 100)}%)`;
await Promise.all(batch.map(async (link, batchIdx) => {
try {
const searchRow = link.closest('tr') || link.closest('.list-item'); // Adapt to different views
let date = '', quote = '', download = '', source = '', sourceUrl = '';
if (searchRow) {
date = searchRow.querySelector('.date')?.textContent?.trim() || '';
quote = searchRow.querySelector('.quote')?.textContent?.trim() || '0';
download = searchRow.querySelector('.download')?.textContent?.trim() || '0';
const sourceElem = searchRow.querySelector('.source a');
if (sourceElem) {
source = sourceElem.textContent.trim();
sourceUrl = sourceElem.href;
}
}
const res = await fetch(convertToWebVPNLink(link.href));
const html = await res.text();
const doc = new DOMParser().parseFromString(html, 'text/html');
// Clean spans in h1
const h1 = doc.querySelector('.wx-tit h1');
if (h1) h1.querySelectorAll('span').forEach(s => s.remove());
const title = h1?.textContent.trim() || '无标题';
const author = Array.from(doc.querySelectorAll('.author')).map(a => a.textContent.trim().replace(/;/g, '')).join('; ');
const keywords = Array.from(doc.querySelectorAll('.keywords a')).map(k => k.textContent.replace(/;/g, '').trim()).join(',');
// Find PDF Link
let pdfLink = '';
const operateBtn = doc.querySelector('.operate-btn');
if (operateBtn) {
// Try standard PDF download
const linkElem = Array.from(operateBtn.querySelectorAll('a')).find(a => a.textContent.includes('PDF下载') || a.textContent.includes('整本下载'));
if (linkElem) pdfLink = linkElem.href;
}
results.push({
id: i + batchIdx + 1,
title, author, pdfLink, keywords,
date, quote, download, source, sourceUrl,
level: 'Wait' // placeholder
});
} catch (err) {
console.error('Row fetch error', err);
}
}));
// Random delay to be safe
await new Promise(r => setTimeout(r, 500));
}
// Render Table
toast.remove();
createLoading('获取完毕!', 1000);
// Save & Render
const finalData = results.sort((a,b) => a.id - b.id);
localStorage.setItem('cnkiTableData', JSON.stringify(finalData));
loadSavedData();
// Auto select all
document.getElementById('cnki-select-all').click();
}
// Lazy Load Journal Levels
async function triggerLevelLoad() {
if (!fetchLevels) return;
const data = JSON.parse(localStorage.getItem('cnkiTableData') || '[]');
if (data.length === 0) return;
let changed = false;
// Process sequentially to avoid overwhelming server with cross-origin requests
for (const item of data) {
if (item.sourceUrl && (!item.level || item.level === 'Wait' || item.level === 'Pending')) {
// Update UI to loading
const cell = document.getElementById(`level-cell-${item.id}`);
if (cell && cell.textContent !== '...') {
cell.innerHTML = '<span class="cnki-level-loading">...</span>';
}
const level = await fetchJournalLevel(item.sourceUrl);
item.level = level;
changed = true;
// Update UI immediately
if (cell) {
cell.innerHTML = level === '无' ? '<span class="cnki-meta-text">-</span>' :
level.split('/').map(l => `<span class="cnki-level-tag">${l}</span>`).join('');
}
}
}
if (changed) {
localStorage.setItem('cnkiTableData', JSON.stringify(data));
}
}
function loadSavedData() {
const saved = localStorage.getItem('cnkiTableData');
const tbody = document.getElementById('cnki-table-body');
if (!tbody) return;
tbody.innerHTML = '';
if (!saved) {
updateStatus();
return;
}
let data = JSON.parse(saved);
// Apply Sort
if (sortState.field) {
data.sort((a, b) => {
let va = a[sortState.field];
let vb = b[sortState.field];
// Parse numbers
if (sortState.field === 'quote' || sortState.field === 'download') {
va = parseInt(va) || 0;
vb = parseInt(vb) || 0;
}
if (va < vb) return sortState.direction === 'asc' ? -1 : 1;
if (va > vb) return sortState.direction === 'asc' ? 1 : -1;
return 0;
});
// Update Headers Icons
document.querySelectorAll('.cnki-sortable').forEach(th => {
th.classList.remove('cnki-sort-active');
th.querySelector('.cnki-sort-icon').textContent = '▲▼';
if (th.dataset.sort === sortState.field) {
th.classList.add('cnki-sort-active');
th.querySelector('.cnki-sort-icon').textContent = sortState.direction === 'asc' ? '▲' : '▼';
}
});
}
data.forEach((item, index) => addTableRow(item, tbody, index));
updateStatus();
// Trigger background level load
if (fetchLevels) setTimeout(triggerLevelLoad, 100);
}
function handleSort(field) {
if (sortState.field === field) {
sortState.direction = sortState.direction === 'asc' ? 'desc' : 'asc';
} else {
sortState.field = field;
sortState.direction = (field === 'date') ? 'asc' : 'desc'; // Date asc by default makes sense? actually desc usually. Let's stick to desc default for stats.
if (field === 'date') sortState.direction = 'desc';
}
loadSavedData();
}
function addTableRow(item, tbody, index) {
const tr = document.createElement('tr');
// Title Highlight
const safeTitle = highlightText(item.title);
// Level Render
let levelHtml = '<span class="cnki-meta-text">-</span>';
if (item.level && item.level !== 'Wait' && item.level !== '无') {
levelHtml = item.level.split('/').map(l => `<span class="cnki-level-tag">${l}</span>`).join('');
} else if (item.level === 'Wait') {
levelHtml = ''; // Will be loaded
}
// Keywords
const tags = item.keywords ? item.keywords.split(',').slice(0, 3).map(k => `<span class="cnki-tag">${k}</span>`).join('') : '';
// Single Download Button
const dlBtnId = `btn-dl-${item.id}`;
const statusId = `status-${item.id}`;
const checkboxId = `cb-${item.id}`;
tr.innerHTML = `
<td class="cnki-col-center"><input type="checkbox" id="${checkboxId}" class="cnki-item-check" ${item.pdfLink ? '' : 'disabled'}></td>
<td class="cnki-col-center" style="color:#9ca3af">${index + 1}</td>
<td>
<div class="cnki-title-wrapper">
<span title="${item.title}">${safeTitle}</span>
<button class="cnki-btn-icon" title="复制标题" onclick="navigator.clipboard.writeText('${item.title.replace(/'/g, "\\'")}')">
<svg xmlns="http://www.w3.org/2000/svg" width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"><rect width="14" height="14" x="8" y="8" rx="2" ry="2"/><path d="M4 16c-1.1 0-2-.9-2-2V4c0-1.1.9-2 2-2h10c1.1 0 2 .9 2 2"/></svg>
</button>
</div>
<div class="cnki-author" title="${item.author}">${item.author}</div>
</td>
<td style="font-size:12px">${item.source || '-'}</td>
<td class="cnki-col-center cnki-td-level ${fetchLevels ? '' : 'cnki-hidden'}" id="level-cell-${item.id}">${levelHtml}</td>
<td class="cnki-col-center cnki-meta-text">${item.date}</td>
<td class="cnki-col-center cnki-meta-text">${item.quote}</td>
<td class="cnki-col-center cnki-meta-text">${item.download}</td>
<td class="cnki-col-center">
${item.pdfLink ?
`<button id="${dlBtnId}" class="cnki-btn-sm">PDF下载</button>` :
'<span class="cnki-link-disabled">无链接</span>'}
<span id="${statusId}" class="cnki-status-pending" style="display:none"></span>
</td>
<td><div style="overflow:hidden; height:24px;">${tags}</div></td>
`;
tbody.appendChild(tr);
// Attach Event Listeners
const dlBtn = document.getElementById(dlBtnId);
if (dlBtn) {
dlBtn.onclick = () => downloadSingleFile(item.pdfLink, item.title, dlBtn, document.getElementById(statusId));
}
const cb = document.getElementById(checkboxId);
if (cb) {
cb.checked = true; // Auto select by default on render if needed, or based on state? Defaulting to true as per user flow.
cb.onchange = updateStatus;
}
}
// --- DOWNLOAD LOGIC ---
function downloadSingleFile(url, filename, btn, statusSpan) {
if (!url) return;
console.log(`[CNKI-Helper] 开始下载: ${filename}, URL: ${url}`);
btn.style.display = 'none';
statusSpan.style.display = 'flex';
statusSpan.className = 'cnki-status-loading';
statusSpan.textContent = '准备中...';
// 1. Fetch Blob with Referer
GM_xmlhttpRequest({
method: 'GET',
url: url,
responseType: 'blob',
headers: { 'Referer': window.location.href },
onload: (res) => {
const blob = res.response;
// Check Content-Type header if available
const contentType = res.responseHeaders.match(/content-type:\s*(.*)/i)?.[1];
console.log(`[CNKI-Helper] 请求成功: ${filename}, Status: ${res.status}, Content-Type: ${contentType}, Size: ${blob.size}`);
// If content type is explicitly HTML, it's definitely an error
if (contentType && contentType.includes('text/html')) {
const errorMsg = '下载失败(HTML)';
console.error(`[CNKI-Helper] 错误: 返回了HTML而非PDF. URL: ${url}`);
statusSpan.className = 'cnki-status-error';
statusSpan.textContent = errorMsg;
btn.style.display = 'inline-block';
btn.textContent = '重试';
return;
}
if (blob.size < 1000) { // Check for HTML error page (fallback)
const errorMsg = '文件无效(过小)';
console.error(`[CNKI-Helper] 错误: 文件过小 (${blob.size} bytes). 可能不是有效的PDF. URL: ${url}`);
statusSpan.className = 'cnki-status-error';
statusSpan.textContent = '文件无效';
btn.style.display = 'inline-block';
btn.textContent = '重试';
return;
}
// 2. Save via GM_download
const finalName = filename.replace(/[\\/:*?"<>|]/g, '_') + '.pdf';
const blobUrl = URL.createObjectURL(blob);
console.log(`[CNKI-Helper] 准备保存: ${finalName}, BlobURL: ${blobUrl}`);
GM_download({
url: blobUrl,
name: finalName,
saveAs: false,
onload: () => {
console.log(`[CNKI-Helper] 下载完成: ${finalName}`);
statusSpan.className = 'cnki-status-success';
statusSpan.textContent = '✔ 完成';
URL.revokeObjectURL(blobUrl);
},
onerror: (e) => {
console.error(`[CNKI-Helper] GM_download 失败: ${finalName}`, e);
statusSpan.className = 'cnki-status-error';
statusSpan.textContent = '✘ 失败';
btn.style.display = 'inline-block';
URL.revokeObjectURL(blobUrl);
},
onprogress: (p) => {
if (p.total > 0) {
const pct = Math.round(p.loaded / p.total * 100);
statusSpan.textContent = `⬇ ${pct}%`;
}
}
});
},
onerror: (e) => {
console.error(`[CNKI-Helper] 请求失败: ${url}`, e);
statusSpan.className = 'cnki-status-error';
statusSpan.textContent = '请求失败';
btn.style.display = 'inline-block';
}
});
}
async function downloadSelected() {
const checkboxes = document.querySelectorAll('.cnki-item-check:checked:not(#cnki-select-all)');
if (checkboxes.length === 0) {
createLoading('请选择要下载的项目!');
return;
}
console.log(`[CNKI-Helper] 开始批量下载, 共选 ${checkboxes.length} 个项目`);
const data = JSON.parse(localStorage.getItem('cnkiTableData'));
for (const cb of checkboxes) {
const row = cb.closest('tr');
// Get original ID via hidden attribute or data attribute would be safer, but mapping via original sorted index in localstorage is risky if sorted.
// Better to use the ID we stored in checkbox ID?
const id = parseInt(cb.id.replace('cb-', ''));
const item = data.find(d => d.id === id);
if (item && item.pdfLink) {
const btn = row.querySelector('.cnki-btn-sm');
const status = row.querySelector('span[id^="status-"]');
if (btn && btn.style.display !== 'none') {
console.log(`[CNKI-Helper] 正在处理: ${item.title} (ID: ${id})`);
downloadSingleFile(item.pdfLink, item.title, btn, status);
// Delay between starts
await new Promise(r => setTimeout(r, 2000 + Math.random() * 2000));
}
} else {
console.warn(`[CNKI-Helper] 找不到项目数据或没有PDF链接. ID: ${id}`, item);
}
}
}
function clearData() {
localStorage.removeItem('cnkiTableData');
document.getElementById('cnki-table-body').innerHTML = '';
updateStatus();
createLoading('数据已清除');
}
function updateStatus() {
const count = document.querySelectorAll('.cnki-item-check:checked:not(#cnki-select-all)').length;
const total = document.querySelectorAll('#cnki-table-body tr').length;
document.getElementById('cnki-status-text').textContent = `共 ${total} 条,已选 ${count} 条`;
const btn = document.getElementById('cnki-batch-dl');
if (btn) btn.textContent = `批量下载 (${count})`;
}
function addCategoryDownloadButton() {
const otherBtns = document.querySelector('.other-btns');
if (!otherBtns) return;
// Check exist
if (document.getElementById('diy-cat-dl')) return;
const li = document.createElement('li');
li.id = 'diy-cat-dl';
li.className = 'btn-diy';
li.style.cssText = 'width:auto;height:23px;line-height:22px;background-color:#3f8af0;border-radius:3px;margin-left:10px;display:inline-block;';
const a = document.createElement('a');
a.textContent = '目录下载';
a.style.cssText = 'color:#ffffff;padding:0 10px;text-decoration:none;font-size:12px;cursor:pointer;display:block;';
a.onclick = downloadCategory;
li.appendChild(a);
otherBtns.appendChild(li);
}
async function downloadCategory() {
createLoading('正在获取目录...');
// Logic for category download... (Simplified for brevity, same as original logic)
const operateBtn = document.querySelector('.operate-btn');
let hrefLink = '';
if(operateBtn) {
const link = Array.from(operateBtn.querySelectorAll('a')).find(a => a.innerText.includes('章节下载'));
if(link) hrefLink = link.href;
}
if (!hrefLink) { createLoading('无法获取目录链接'); return; }
try {
const html = await gmFetch(hrefLink);
const doc = new DOMParser().parseFromString(html, 'text/html');
const title = doc.querySelector('.wx-tit h1')?.textContent.trim() || '目录';
const chapters = Array.from(doc.querySelectorAll('.ls-chapters li'))
.map(c => c.textContent.trim().split('-')[0].replace(/\n/g, '\t'))
.join('\n');
const blob = new Blob([chapters], { type: 'text/plain' });
const url = URL.createObjectURL(blob);
const a = document.createElement('a');
a.href = url;
a.download = title + '_目录.txt';
a.click();
URL.revokeObjectURL(url);
createLoading('下载完成');
} catch(e) {
createLoading('下载目录失败');
}
}
// Run
window.addEventListener('load', init);
})();