// ==UserScript==
// @name Save Vietnamese xenforo forum content to IndexedDB
// @description Save your favorite thread into IndexedDB
// @namespace Save thread to file
// @icon 
// @match https://voz.vn/t/*
// @match https://forum.gocmod.com/threads/*
// @match https://www.otofun.net/threads/*
// @match https://vn-z.vn/threads/*
// @match https://*xamvn*.*/threads/*
// @match https://*xamvn*.*/r/*
// @match https://*rphang*.*/t/*
// @match https://*thiendia*.*/threads/*
// @match https://tve-4u.org/threads/*
// @grant GM_xmlhttpRequest
// @grant GM_registerMenuCommand
// @grant GM_setValue
// @grant GM_getValue
// @version 1.1
// @author kylyte
// @license GPL-3.0
// ==/UserScript==
const config = {
waitTime: 100,
saveWithImages: true,
concurrentRequests: 15,
chunkSize: 15,
showDebugInfo: true,
maxChunkSizeForHtml: 10000 * 1024
};
const dbName = "ThreadSaverDB";
const dbVersion = 1;
const VIEW_SAVED_BUTTON_KEY = "viewSavedButtonEnabled";
const sleep = (ms) => new Promise((resolve) => setTimeout(resolve, ms));
const log = (...args) => config.showDebugInfo && console.log('[Voz Saver]', ...args);
const sanitizeFilename = (name) => (name || 'untitled').replace(/[^a-z0-9_\- ]/gi, '').trim().replace(/\s+/g, '_').substring(0, 50);
async function openDB() {
return new Promise((resolve, reject) => {
const req = indexedDB.open(dbName, dbVersion);
req.onupgradeneeded = (e) => {
const db = e.target.result;
if (!db.objectStoreNames.contains('threads')) {
db.createObjectStore('threads', { keyPath: 'id' });
}
if (!db.objectStoreNames.contains('pages')) {
db.createObjectStore('pages', { keyPath: ['threadId', 'pageNo'] });
}
if (!db.objectStoreNames.contains('images')) {
db.createObjectStore('images', { keyPath: ['threadId', 'key'] });
}
};
req.onsuccess = (e) => resolve(e.target.result);
req.onerror = (e) => reject(e.target.error);
});
}
async function saveThreadMetadata(db, threadId, title, cssStyles, startPage, endPage, maxPage, wrapperUrl) {
const tx = db.transaction('threads', 'readwrite');
const store = tx.objectStore('threads');
return new Promise((res, rej) => {
const req = store.put({id: threadId, title, cssStyles, startPage, endPage, maxPage, wrapperUrl});
req.onsuccess = res;
req.onerror = rej;
});
}
async function savePage(db, threadId, pageNo, data) {
const tx = db.transaction('pages', 'readwrite');
const store = tx.objectStore('pages');
return new Promise((res, rej) => {
const req = store.put({threadId, pageNo, data});
req.onsuccess = res;
req.onerror = rej;
});
}
async function saveImages(db, threadId, images) {
const tx = db.transaction('images', 'readwrite');
const store = tx.objectStore('images');
const promises = [];
for (const [key, data] of Object.entries(images)) {
promises.push(new Promise((res, rej) => {
const req = store.put({threadId, key, data});
req.onsuccess = res;
req.onerror = rej;
}));
}
await Promise.all(promises);
}
async function getAllThreads(db) {
const tx = db.transaction('threads', 'readonly');
const store = tx.objectStore('threads');
return new Promise((res, rej) => {
const req = store.getAll();
req.onsuccess = () => res(req.result);
req.onerror = rej;
});
}
async function getThreadMetadata(db, threadId) {
const tx = db.transaction('threads', 'readonly');
const store = tx.objectStore('threads');
return new Promise((res, rej) => {
const req = store.get(threadId);
req.onsuccess = () => res(req.result);
req.onerror = rej;
});
}
async function getPage(db, threadId, pageNo) {
const tx = db.transaction('pages', 'readonly');
const store = tx.objectStore('pages');
return new Promise((res, rej) => {
const req = store.get([threadId, pageNo]);
req.onsuccess = () => res(req.result);
req.onerror = rej;
});
}
async function getImagesForThread(db, threadId) {
const tx = db.transaction('images', 'readonly');
const store = tx.objectStore('images');
const images = {};
return new Promise((res, rej) => {
const req = store.openCursor(IDBKeyRange.bound([threadId, ''], [threadId, '\uffff']));
req.onsuccess = (e) => {
const cursor = e.target.result;
if (cursor) {
images[cursor.value.key] = cursor.value.data;
cursor.continue();
} else {
res(images);
}
};
req.onerror = rej;
});
}
async function deleteThread(db, threadId) {
const tx = db.transaction(['threads', 'pages', 'images'], 'readwrite');
const threadsStore = tx.objectStore('threads');
threadsStore.delete(threadId);
const pagesStore = tx.objectStore('pages');
const pagesRange = IDBKeyRange.bound([threadId, 0], [threadId, Number.MAX_SAFE_INTEGER]);
const pagesCursorReq = pagesStore.openCursor(pagesRange);
pagesCursorReq.onsuccess = (e) => {
const cursor = e.target.result;
if (cursor) {
cursor.delete();
cursor.continue();
}
};
const imagesStore = tx.objectStore('images');
const imagesRange = IDBKeyRange.bound([threadId, ''], [threadId, '\uffff']);
const imagesCursorReq = imagesStore.openCursor(imagesRange);
imagesCursorReq.onsuccess = (e) => {
const cursor = e.target.result;
if (cursor) {
cursor.delete();
cursor.continue();
}
};
return new Promise((resolve, reject) => {
tx.oncomplete = resolve;
tx.onerror = (e) => reject(e.target.error);
});
}
async function getPagesForThread(db, threadId) {
const tx = db.transaction('pages', 'readonly');
const store = tx.objectStore('pages');
const pages = [];
return new Promise((res, rej) => {
const req = store.openCursor(IDBKeyRange.bound([threadId, 0], [threadId, Number.MAX_SAFE_INTEGER]));
req.onsuccess = (e) => {
const cursor = e.target.result;
if (cursor) {
pages.push(cursor.value);
cursor.continue();
} else {
res(pages);
}
};
req.onerror = rej;
});
}
async function getAllImagesForThread(db, threadId) {
const tx = db.transaction('images', 'readonly');
const store = tx.objectStore('images');
const images = [];
return new Promise((res, rej) => {
const req = store.openCursor(IDBKeyRange.bound([threadId, ''], [threadId, '\uffff']));
req.onsuccess = (e) => {
const cursor = e.target.result;
if (cursor) {
images.push(cursor.value);
cursor.continue();
} else {
res(images);
}
};
req.onerror = rej;
});
}
async function importThreads() {
const input = document.createElement('input');
input.type = 'file';
input.accept = '.txt, application/json';
input.style.display = 'none';
input.onchange = async (e) => {
const file = e.target.files[0];
if (!file) return;
const reader = new FileReader();
reader.onload = async (event) => {
let data;
try {
data = JSON.parse(event.target.result);
if (!data || !data.version || !Array.isArray(data.threads) || !Array.isArray(data.pages) || !Array.isArray(data.images)) {
throw new Error('Định dạng file không hợp lệ.');
}
} catch (err) {
console.error('Import error:', err);
alert(`Lỗi khi đọc file: ${err.message}`);
return;
}
let importCount = data.threads.length;
try {
const db = await openDB();
const tx = db.transaction(['threads', 'pages', 'images'], 'readwrite');
const threadsStore = tx.objectStore('threads');
const pagesStore = tx.objectStore('pages');
const imagesStore = tx.objectStore('images');
let totalOperations = data.threads.length + data.pages.length + data.images.length;
let completedOperations = 0;
const progressDiv = document.createElement('div');
progressDiv.id = 'import-progress';
progressDiv.style.position = 'fixed';
progressDiv.style.top = '50%';
progressDiv.style.left = '50%';
progressDiv.style.transform = 'translate(-50%, -50%)';
progressDiv.style.background = 'rgba(0,0,0,0.8)';
progressDiv.style.color = 'white';
progressDiv.style.padding = '20px';
progressDiv.style.borderRadius = '8px';
progressDiv.style.zIndex = '10001';
progressDiv.textContent = `Đang nhập 0/${totalOperations} mục...`;
document.body.appendChild(progressDiv);
const updateProgress = () => {
completedOperations++;
progressDiv.textContent = `Đang nhập ${completedOperations}/${totalOperations} mục...`;
};
const onError = (e) => {
console.error('Lỗi khi ghi vào DB:', e.target.error);
tx.abort();
};
data.threads.forEach(thread => {
const req = threadsStore.put(thread);
req.onsuccess = updateProgress;
req.onerror = onError;
});
data.pages.forEach(page => {
const req = pagesStore.put(page);
req.onsuccess = updateProgress;
req.onerror = onError;
});
data.images.forEach(image => {
const req = imagesStore.put(image);
req.onsuccess = updateProgress;
req.onerror = onError;
});
tx.oncomplete = () => {
document.body.removeChild(progressDiv);
alert(`Đã nhập thành công ${importCount} thread(s)!`);
const popup = document.getElementById('saved-threads-popup');
if (popup) {
popup.remove();
viewSavedThreads();
}
};
tx.onerror = (e) => {
document.body.removeChild(progressDiv);
throw new Error(e.target.error || 'Transaction failed');
};
} catch (err) {
console.error('Lỗi khi lưu vào IndexedDB:', err);
alert(`Lỗi khi lưu vào cơ sở dữ liệu: ${err.message}`);
const progressDiv = document.getElementById('import-progress');
if (progressDiv) progressDiv.remove();
}
};
reader.onerror = () => {
alert('Không thể đọc file.');
};
reader.readAsText(file);
document.body.removeChild(input);
};
document.body.appendChild(input);
input.click();
}
async function viewSavedThreads() {
const db = await openDB();
const threads = await getAllThreads(db);
const popup = document.createElement('div');
popup.id = 'saved-threads-popup';
popup.style.position = 'fixed';
popup.style.top = '10%';
popup.style.left = '10%';
popup.style.width = '80%';
popup.style.height = '80%';
popup.style.background = '#333';
popup.style.color = '#eee';
popup.style.border = '1px solid #555';
popup.style.overflow = 'auto';
popup.style.zIndex = '9999';
popup.style.padding = '20px';
popup.style.boxShadow = '0 4px 8px rgba(0,0,0,0.2)';
popup.style.borderRadius = '8px';
const titleEl = document.createElement('h2');
titleEl.textContent = 'Các Threads đã lưu:';
titleEl.style.marginBottom = '10px';
popup.appendChild(titleEl);
const closeBtn = document.createElement('button');
closeBtn.textContent = 'Đóng';
closeBtn.style.float = 'right';
closeBtn.style.background = '#555';
closeBtn.style.color = '#fff';
closeBtn.style.border = '1px solid #777';
closeBtn.style.padding = '5px 10px';
closeBtn.style.borderRadius = '4px';
closeBtn.style.cursor = 'pointer';
closeBtn.onclick = () => popup.remove();
popup.appendChild(closeBtn);
const controlsDiv = document.createElement('div');
controlsDiv.style.marginBottom = '15px';
controlsDiv.style.display = 'flex';
controlsDiv.style.flexWrap = 'wrap';
controlsDiv.style.alignItems = 'center';
controlsDiv.style.gap = '10px';
const selectAll = document.createElement('input');
selectAll.type = 'checkbox';
selectAll.id = 'select-all';
controlsDiv.appendChild(selectAll);
const labelAll = document.createElement('label');
labelAll.htmlFor = 'select-all';
labelAll.textContent = 'Chọn tất cả';
labelAll.style.cursor = 'pointer';
controlsDiv.appendChild(labelAll);
const deleteBtn = document.createElement('button');
deleteBtn.textContent = 'Xóa Threads đã chọn';
deleteBtn.style.background = '#ff4444';
deleteBtn.style.color = 'white';
deleteBtn.style.border = 'none';
deleteBtn.style.padding = '5px 10px';
deleteBtn.style.borderRadius = '4px';
deleteBtn.style.cursor = 'pointer';
controlsDiv.appendChild(deleteBtn);
const exportBtn = document.createElement('button');
exportBtn.textContent = 'Xuất Threads đã chọn';
exportBtn.style.background = '#4a86e8';
exportBtn.style.color = 'white';
exportBtn.style.border = 'none';
exportBtn.style.padding = '5px 10px';
exportBtn.style.borderRadius = '4px';
exportBtn.style.cursor = 'pointer';
controlsDiv.appendChild(exportBtn);
const importBtn = document.createElement('button');
importBtn.textContent = 'Nhập Threads';
importBtn.style.background = '#4CAF50';
importBtn.style.color = 'white';
importBtn.style.border = 'none';
importBtn.style.padding = '5px 10px';
importBtn.style.borderRadius = '4px';
importBtn.style.cursor = 'pointer';
importBtn.onclick = importThreads;
controlsDiv.appendChild(importBtn);
popup.appendChild(controlsDiv);
const table = document.createElement('table');
table.style.width = '100%';
table.style.borderCollapse = 'collapse';
const thead = document.createElement('thead');
const headerRow = document.createElement('tr');
headerRow.innerHTML = `
<th style="text-align: left; padding: 8px; border-bottom: 1px solid #555;">Chọn</th>
<th style="text-align: left; padding: 8px; border-bottom: 1px solid #555;">Tiêu đề</th>
`;
thead.appendChild(headerRow);
table.appendChild(thead);
const tbody = document.createElement('tbody');
for (const thread of threads) {
const tr = document.createElement('tr');
tr.style.borderBottom = '1px solid #444';
const tdSelect = document.createElement('td');
tdSelect.style.padding = '8px';
const checkbox = document.createElement('input');
checkbox.type = 'checkbox';
checkbox.value = thread.id;
checkbox.className = 'thread-checkbox';
tdSelect.appendChild(checkbox);
tr.appendChild(tdSelect);
const tdTitle = document.createElement('td');
tdTitle.style.padding = '8px';
const link = document.createElement('a');
link.textContent = `${thread.title} (Pages ${thread.startPage}-${thread.endPage})`;
link.style.cursor = 'pointer';
link.style.color = '#58a6ff';
link.style.textDecoration = 'none';
link.onmouseover = () => link.style.textDecoration = 'underline';
link.onmouseout = () => link.style.textDecoration = 'none';
link.onclick = async () => {
popup.remove();
await showSavedThread(thread.id);
};
tdTitle.appendChild(link);
tr.appendChild(tdTitle);
tbody.appendChild(tr);
}
table.appendChild(tbody);
popup.appendChild(table);
document.body.appendChild(popup);
const allCheckboxes = popup.querySelectorAll('.thread-checkbox');
selectAll.addEventListener('change', (e) => {
allCheckboxes.forEach(cb => cb.checked = e.target.checked);
});
deleteBtn.onclick = async () => {
const checked = popup.querySelectorAll('.thread-checkbox:checked');
if (checked.length === 0) return;
if (!confirm(`Are you sure you want to delete ${checked.length} selected thread(s)?`)) return;
deleteBtn.textContent = 'Đang xóa...';
deleteBtn.disabled = true;
exportBtn.disabled = true;
importBtn.disabled = true;
selectAll.disabled = true;
allCheckboxes.forEach(cb => cb.disabled = true);
const db = await openDB();
try {
for (const cb of checked) {
await deleteThread(db, cb.value);
}
alert('Đã xóa thành công!');
popup.remove();
viewSavedThreads();
} catch (err) {
console.error('Lỗi khi xóa:', err);
alert('Đã có lỗi xảy ra khi xóa thread.');
deleteBtn.textContent = 'Xóa Threads đã chọn';
deleteBtn.disabled = false;
exportBtn.disabled = false;
importBtn.disabled = false;
selectAll.disabled = false;
allCheckboxes.forEach(cb => cb.disabled = false);
}
};
exportBtn.onclick = async () => {
const checked = popup.querySelectorAll('.thread-checkbox:checked');
if (checked.length === 0) return alert('Vui lòng chọn ít nhất một thread để xuất.');
const db = await openDB();
const exportData = {
version: 1.0,
timestamp: new Date().toISOString(),
threads: [],
pages: [],
images: []
};
try {
exportBtn.textContent = 'Đang xuất...';
exportBtn.disabled = true;
for (const cb of checked) {
const threadId = cb.value;
log(`Exporting thread: ${threadId}`);
const metadata = await getThreadMetadata(db, threadId);
if (metadata) {
exportData.threads.push(metadata);
}
const pages = await getPagesForThread(db, threadId);
exportData.pages.push(...pages);
const images = await getAllImagesForThread(db, threadId);
exportData.images.push(...images);
}
log(`Exporting ${exportData.threads.length} threads, ${exportData.pages.length} pages, ${exportData.images.length} images.`);
const jsonString = JSON.stringify(exportData);
const blob = new Blob([jsonString], { type: 'text/plain;charset=utf-8' });
const link = document.createElement('a');
link.href = URL.createObjectURL(blob);
const timestamp = new Date().toISOString().replace(/[:.-]/g, '_');
let filename = `threads_export_${timestamp}.txt`;
if (checked.length === 1) {
const singleThreadMeta = exportData.threads[0];
if (singleThreadMeta && singleThreadMeta.title) {
const safeTitle = sanitizeFilename(singleThreadMeta.title);
filename = `${safeTitle}_${timestamp}.txt`;
}
}
link.download = filename;
document.body.appendChild(link);
link.click();
document.body.removeChild(link);
URL.revokeObjectURL(link.href);
} catch (err) {
console.error('Export failed:', err);
alert('Xuất file thất bại. Vui lòng kiểm tra console.');
} finally {
exportBtn.textContent = 'Xuất Threads đã chọn';
exportBtn.disabled = false;
}
};
}
async function showSavedThread(threadId) {
const db = await openDB();
const metadata = await getThreadMetadata(db, threadId);
if (!metadata) return alert('Thread not found');
const {title, cssStyles, startPage, endPage, maxPage, wrapperUrl} = metadata;
const images = await getImagesForThread(db, threadId);
const viewer = document.createElement('div');
viewer.style.position = 'fixed';
viewer.style.top = '0';
viewer.style.left = '0';
viewer.style.width = '100%';
viewer.style.height = '100%';
viewer.style.background = 'white';
viewer.style.zIndex = '9999';
viewer.style.overflow = 'auto';
viewer.innerHTML = `
<style>
body, html { margin: 0; padding: 0; }
#loading {
position: fixed; top: 0; left: 0; right: 0; bottom: 0;
background: rgba(0,0,0,0.8); color: white; display: flex;
flex-direction: column; justify-content: center; align-items: center;
z-index: 9999;
}
.navigation-controls {
position: fixed;
bottom: 20px;
right: 20px;
background: #f0f0f0;
padding: 10px;
border-radius: 5px;
box-shadow: 0 2px 5px rgba(0,0,0,0.2);
z-index: 1000;
display: flex;
gap: 10px;
}
.nav-button {
background: #4a86e8;
color: white;
border: none;
padding: 5px 10px;
border-radius: 3px;
cursor: pointer;
}
.nav-button:hover {
background: #2a66c8;
}
.nav-button:disabled {
background: #ccc;
cursor: not-allowed;
}
${cssStyles || ''}
</style>
<div id="loading">
<h2>Loading Thread...</h2>
<progress id="loading-progress" value="0" max="100" style="width:80%; max-width:400px"></progress>
<div id="loading-text">Initializing...</div>
</div>
<div id="screen"></div>
<div class="navigation-controls">
<button id="prev-page-btn" class="nav-button" disabled>Previous</button>
<span id="page-display">Page ${startPage}</span>
<button id="next-page-btn" class="nav-button">Next</button>
<button id="goto-page-btn" class="nav-button">Go to Page</button>
<button id="close-viewer" class="nav-button">Close</button>
</div>
`;
document.body.appendChild(viewer);
const closeBtn = viewer.querySelector('#close-viewer');
closeBtn.onclick = () => viewer.remove();
const threadBodyReplacement = "{ThreadBody_PLACEHOLDER}";
let currentPage = startPage;
function updateNavButtons() {
const prevBtn = viewer.querySelector('#prev-page-btn');
const nextBtn = viewer.querySelector('#next-page-btn');
const pageDisplay = viewer.querySelector('#page-display');
prevBtn.disabled = currentPage <= startPage;
nextBtn.disabled = currentPage >= endPage;
pageDisplay.textContent = `Page ${currentPage}`;
}
const screen = viewer.querySelector('#screen');
const loading = viewer.querySelector('#loading');
const loadingProgress = viewer.querySelector('#loading-progress');
const loadingText = viewer.querySelector('#loading-text');
const prevBtn = viewer.querySelector('#prev-page-btn');
const nextBtn = viewer.querySelector('#next-page-btn');
const gotoBtn = viewer.querySelector('#goto-page-btn');
prevBtn.onclick = () => {
if (currentPage > startPage) {
showPage(currentPage - 1);
}
};
nextBtn.onclick = () => {
if (currentPage < endPage) {
showPage(currentPage + 1);
}
};
gotoBtn.onclick = () => {
const pageNo = prompt(`Enter page number (${startPage}-${endPage})`, currentPage);
if (pageNo && !isNaN(pageNo)) {
const page = parseInt(pageNo);
if (page >= startPage && page <= endPage) {
showPage(page);
} else {
alert(`Please enter a number between ${startPage} and ${endPage}`);
}
}
};
async function decompressData(dataUrl) {
const response = await fetch(dataUrl);
const blob = await response.blob();
const decompressedStream = blob.stream().pipeThrough(new DecompressionStream("gzip"));
return new Response(decompressedStream).blob();
}
async function blobToText(blob) {
return new Promise((resolve) => {
const reader = new FileReader();
reader.onload = () => resolve(reader.result);
reader.readAsText(blob);
});
}
async function showPage(pageId) {
try {
loading.style.display = 'flex';
loadingText.textContent = `Loading page ${pageId}...`;
const pageData = await getPage(db, threadId, pageId);
if (!pageData) throw new Error('Page not found');
const decompressedBody = await decompressData(pageData.data);
const threadBody = await blobToText(decompressedBody);
const pageContent = threadWrapper.replace(threadBodyReplacement, threadBody);
screen.innerHTML = pageContent;
const imgs = screen.querySelectorAll('img[image-data]');
if (imgs.length > 0) {
loadingProgress.max = imgs.length;
loadingProgress.value = 0;
let loadedImages = 0;
for (const img of imgs) {
const key = img.getAttribute('image-data');
if (key && images[key]) {
img.src = images[key];
}
loadedImages++;
loadingProgress.value = loadedImages;
loadingText.textContent = `Loading images (${loadedImages}/${imgs.length})...`;
if (loadedImages % 10 === 0) {
await new Promise(r => setTimeout(r, 0));
}
}
}
setupPageNavigation();
currentPage = pageId;
updateNavButtons();
window.scrollTo(0, 0);
loading.style.display = 'none';
} catch (error) {
console.error('Error showing page:', error);
loadingText.textContent = `Error loading page ${pageId}: ${error.message}`;
}
}
function setupPageNavigation() {
screen.querySelectorAll('ul.pageNav-main a:not([id])').forEach(el => {
el.addEventListener('click', e => {
e.preventDefault();
const pageNum = parseInt(e.target.textContent.trim());
if (!isNaN(pageNum) && pageNum >= startPage && pageNum <= endPage) {
showPage(pageNum);
}
});
});
screen.querySelectorAll('ul.pageNav-main a[title="Go to page"]').forEach(el => {
el.addEventListener('click', e => {
e.preventDefault();
const pageNo = prompt(`Enter page number (${startPage}-${endPage})`, currentPage);
if (pageNo && !isNaN(pageNo)) {
const page = parseInt(pageNo);
if (page >= startPage && page <= endPage) {
showPage(page);
}
}
});
});
screen.querySelectorAll('.pageNav-jump.pageNav-jump--next').forEach(el => {
el.addEventListener('click', e => {
e.preventDefault();
if (currentPage < endPage) {
showPage(currentPage + 1);
}
});
});
screen.querySelectorAll('.pageNav-jump.pageNav-jump--prev').forEach(el => {
el.addEventListener('click', e => {
e.preventDefault();
if (currentPage > startPage) {
showPage(currentPage - 1);
}
});
});
}
let threadWrapper;
try {
loadingText.textContent = 'Preparing thread template...';
const decompressedWrapper = await decompressData(wrapperUrl);
threadWrapper = await blobToText(decompressedWrapper);
await showPage(startPage);
} catch (error) {
console.error('Initialization error:', error);
loadingText.textContent = `Error initializing: ${error.message}`;
}
}
async function createHash(message) {
const msgUint8 = new TextEncoder().encode(message);
const hashBuffer = await window.crypto.subtle.digest("SHA-1", msgUint8);
const hashArray = Array.from(new Uint8Array(hashBuffer));
return hashArray.map((b) => b.toString(16).padStart(2, "0")).join("");
}
function xhr(url, detail = {}) {
const nurl = new URL(url);
let options = { url: url, origin: nurl.origin };
if (typeof detail === 'string' && /^(?:blob|text|json|arraybuffer|document)$/.test(detail)) {
options.responseType = detail;
} else if (typeof detail === 'object') {
options = { ...options, ...detail };
}
return new Promise(resolve => {
options.onloadend = res => (res.status === 200) ? resolve(res.response) : resolve(false);
options.onerror = () => resolve(false);
options.ontimeout = () => resolve(false);
GM_xmlhttpRequest(options);
});
}
function toDataURL(data) {
return new Promise((resolve, reject) => {
const reader = new FileReader();
reader.onload = () => resolve(reader.result);
reader.onerror = () => reject(reader.error);
reader.readAsDataURL(new Blob([data]));
});
}
async function compressData(data) {
const blob = new Blob([data]);
const compressedStream = blob.stream().pipeThrough(new CompressionStream("gzip"));
return await new Response(compressedStream).blob();
}
async function extractStyles(doc) {
const links = doc.querySelectorAll('link[rel="stylesheet"]');
const styles = [];
for (let i = 0; i < links.length; i++) {
try {
const href = links[i].href;
if (!href) continue;
const css = await xhr(href, 'text');
if (css) {
styles.push(css);
}
} catch (err) {
log('Failed to process stylesheet:', links[i].href, err);
}
}
doc.querySelectorAll('style').forEach(style => {
styles.push(style.textContent);
});
return styles.join('\n');
}
async function processContent(htmlStr, isFirstPage = false, images = {}, cssStyles = null) {
const parser = new DOMParser();
let html = parser.parseFromString(htmlStr, 'text/html');
html.querySelector('.blockMessage--none')?.remove();
html.querySelectorAll('form').forEach(el => el?.remove());
html.querySelectorAll('div.block').forEach(el => {
if (el.matches('.block--messages')) return;
el.remove();
});
html.querySelectorAll('div.p-body-main.p-body-main--withSidebar>*').forEach(el => {
if (el.matches('.p-body-content')) return;
el.remove();
});
html.querySelector('footer.p-footer')?.remove();
html.querySelectorAll('[href]').forEach(el => {
let href = el.getAttribute('href');
if (href && href.startsWith('/')) el.setAttribute('href', location.origin + href);
});
html.querySelectorAll('[src]').forEach(el => {
let src = el.getAttribute('src');
if (src && src.startsWith('data:image')) el.setAttribute('src', el.getAttribute('data-src') || src);
if (src && src.startsWith('/')) el.setAttribute('src', location.origin + src);
});
html.querySelectorAll('[srcset]').forEach(el => {
let srcset = el.getAttribute('srcset');
if (srcset) {
srcset = srcset.split(',').map(a => a.trim().startsWith('/') ? location.origin + a.trim() : a).join(',');
el.setAttribute('srcset', srcset);
}
});
html.querySelectorAll('div.bbCodeBlock-content>div.bbCodeBlock-expandContent.js-expandContent').forEach(el => el.className = '');
html.querySelectorAll('.bbCodeSpoiler-button,.bbCodeSpoiler-content').forEach(el => el.classList.add('is-active'));
html.querySelectorAll('div.pageNav a').forEach(el => el.removeAttribute('href'));
if (config.saveWithImages) {
const imgElements = html.querySelectorAll('img');
const imgPromises = [];
for (let i = 0; i < imgElements.length; i++) {
const img = imgElements[i];
if (!img.src || img.src.startsWith('data:image')) continue;
imgPromises.push((async () => {
try {
const key = await createHash(img.src);
if (images[key]) {
img.setAttribute('image-data', key);
return;
}
const imgBlob = await xhr(img.src, 'blob');
if (!imgBlob) return;
const dataUrl = await toDataURL(imgBlob);
images[key] = dataUrl;
img.setAttribute('image-data', key);
} catch (err) {
log('Failed to process image:', img.src, err);
}
})());
if (imgPromises.length >= 5) {
await Promise.all(imgPromises);
imgPromises.length = 0;
await sleep(100);
}
}
if (imgPromises.length > 0) {
await Promise.all(imgPromises);
}
}
let extractedCss = null;
if (isFirstPage && !cssStyles) {
extractedCss = await extractStyles(html);
html.querySelectorAll('link[rel="stylesheet"]').forEach(el => el.remove());
const styleEl = html.createElement('style');
styleEl.textContent = extractedCss;
html.head.appendChild(styleEl);
} else if (isFirstPage && cssStyles) {
html.querySelectorAll('link[rel="stylesheet"]').forEach(el => el.remove());
const styleEl = html.createElement('style');
styleEl.textContent = cssStyles;
html.head.appendChild(styleEl);
}
const threadBody = html.querySelector('div.p-body-main');
if (!threadBody) {
throw new Error('Could not find thread body content');
}
const compressedBody = await compressData(threadBody.outerHTML);
const threadBodyUrl = await toDataURL(compressedBody);
let wrapperUrl = '';
if (isFirstPage) {
threadBody.outerHTML = `{ThreadBody_PLACEHOLDER}`;
const serialized = new XMLSerializer().serializeToString(html);
const compressed = await compressData(serialized);
wrapperUrl = await toDataURL(compressed);
}
return { wrapperUrl, threadBodyUrl, extractedCss };
}
async function saveThread(threadId, threadInPath) {
const db = await openDB();
const maxPageEl = document.querySelector("ul.pageNav-main>li:last-of-type>a");
const maxPage = maxPageEl ? parseInt(maxPageEl.textContent) : 1;
let pageRange = prompt(
"Nhập thông số để tải xuống. Ví dụ:\n" +
"Nhập 1-50 sẽ tải từ trang 1 tới trang 50\n" +
"Nhập 5 sẽ tải chỉ trang 5\n" +
"Bỏ trống sẽ tải tất cả các trang"
);
let startPage = 1;
let endPage = maxPage;
if (pageRange) {
if (pageRange.includes('-')) {
const [start, end] = pageRange.split('-').map(p => parseInt(p.trim()));
if (!isNaN(start) && !isNaN(end) && start >= 1 && end <= maxPage) {
startPage = start;
endPage = end;
} else {
alert(`Invalid range. Using full range (1-${maxPage}).`);
}
} else {
const page = parseInt(pageRange.trim());
if (!isNaN(page) && page >= 1 && page <= maxPage) {
startPage = page;
endPage = page;
} else {
alert(`Invalid page number. Using full range (1-${maxPage}).`);
}
}
}
document.body.insertAdjacentHTML("beforeend",
`<div id="voz_saver_progress" style="position:fixed; bottom:0; left:0; right:0; background:rgba(0,0,0,0.8); color:white; padding:10px; z-index:9999; display:flex; flex-direction:column;">
<div style="display:flex; justify-content:space-between; margin-bottom:5px;">
<span>Saving thread (0/${endPage - startPage + 1} pages)</span>
<button id="voz_saver_cancel" style="background:#ff4444; border:none; color:white; padding:2px 8px; cursor:pointer;">Cancel</button>
</div>
<progress id="voz_saver_progress_bar" value="0" max="${endPage - startPage + 1}" style="width:100%; height:20px;"></progress>
<div id="voz_saver_status" style="margin-top:5px; font-size:12px;">Initializing...</div>
</div>`
);
const progressBar = document.getElementById('voz_saver_progress_bar');
const progressText = document.querySelector('#voz_saver_progress span');
const statusText = document.getElementById('voz_saver_status');
const cancelButton = document.getElementById('voz_saver_cancel');
let cancelled = false;
cancelButton.addEventListener('click', () => {
cancelled = true;
statusText.textContent = 'Cancelling...';
});
const images = {};
let cssStyles = null;
let wrapperUrl = null;
const threadBodies = {};
let pageCount = 0;
const updateProgress = () => {
progressBar.value = pageCount;
progressText.textContent = `Saving thread (${pageCount}/${endPage - startPage + 1} pages)`;
};
async function fetchPage(pageNo, retries = 3) {
if (cancelled) return false;
try {
statusText.textContent = `Fetching page ${pageNo}...`;
const pageUrl = `${location.origin}/${threadInPath}/${threadId}/page-${pageNo}`;
const response = await fetch(pageUrl);
if (response.status !== 200) {
if (retries > 0) {
statusText.textContent = `Error fetching page ${pageNo}, retrying (${retries} left)...`;
await sleep(1000);
return fetchPage(pageNo, retries - 1);
}
throw new Error(`Failed to fetch page ${pageNo} (status: ${response.status})`);
}
const html = await response.text();
statusText.textContent = `Processing page ${pageNo}...`;
const { wrapperUrl: pageWrapperUrl, threadBodyUrl, extractedCss } = await processContent(html, pageNo === startPage, images, cssStyles);
if (pageNo === startPage) {
wrapperUrl = pageWrapperUrl;
if (extractedCss) {
cssStyles = extractedCss;
}
}
threadBodies[pageNo] = threadBodyUrl;
pageCount++;
updateProgress();
return true;
} catch (err) {
if (retries > 0) {
statusText.textContent = `Error processing page ${pageNo}, retrying (${retries} left)...`;
await sleep(1000);
return fetchPage(pageNo, retries - 1);
}
log(`Error processing page ${pageNo}:`, err);
statusText.textContent = `Failed to process page ${pageNo}: ${err.message}`;
return false;
}
}
const chunks = [];
for (let i = startPage; i <= endPage; i += config.chunkSize) {
chunks.push(Array.from({ length: Math.min(config.chunkSize, endPage - i + 1) }, (_, j) => i + j));
}
for (let i = 0; i < chunks.length; i++) {
if (cancelled) break;
statusText.textContent = `Processing chunk ${i+1}/${chunks.length}...`;
for (let j = 0; j < chunks[i].length; j += config.concurrentRequests) {
if (cancelled) break;
const batch = chunks[i].slice(j, j + config.concurrentRequests);
const promises = batch.map(pageNo => fetchPage(pageNo));
await Promise.all(promises);
await sleep(config.waitTime);
}
if (i < chunks.length - 1) {
statusText.textContent = `Chunk ${i+1} complete. Taking a short break...`;
await sleep(1000);
}
}
if (cancelled) {
document.getElementById('voz_saver_progress').remove();
return null;
}
statusText.textContent = 'Saving to IndexedDB...';
const title = document.querySelector('title')?.textContent
.split('-').pop()?.split('|')[0].trim() || 'vozThread';
const threadKey = `${location.origin}/${threadInPath}/${threadId}`;
await saveThreadMetadata(db, threadKey, title, cssStyles, startPage, endPage, maxPage, wrapperUrl);
for (const [pageNo, data] of Object.entries(threadBodies)) {
await savePage(db, threadKey, parseInt(pageNo), data);
}
await saveImages(db, threadKey, images);
document.getElementById('voz_saver_progress').remove();
return true;
}
(async function main() {
GM_registerMenuCommand("Xem Threads đã lưu", viewSavedThreads);
let isButtonEnabled = await GM_getValue(VIEW_SAVED_BUTTON_KEY, false);
const commandText = isButtonEnabled
? "Tắt nút 'Xem Thread đã lưu'"
: "Bật nút 'Xem Thread đã lưu'";
GM_registerMenuCommand(commandText, async () => {
const newState = !isButtonEnabled;
await GM_setValue(VIEW_SAVED_BUTTON_KEY, newState);
alert(`Nút 'Xem đã lưu' đã ${newState ? 'BẬT' : 'TẮT'}. \nVui lòng tải lại trang để thấy thay đổi.`);
location.reload();
});
const domain = window.location.hostname;
const threadName = ['t', 'r', 'threads'];
const threadRegx = threadName.join('|');
const reg = new RegExp(`https:\/\/(?:.*\\.)?${domain.replaceAll('.', '\\.')}\/(${threadRegx})\/(?:[^\/]+\\.)?(\\d+)\/?(?:page-(\\d+))?`);
console.log(reg);
const match = location.href.match(reg);
if (!match) {
log('Not on a recognized thread page. Exiting.');
return;
}
const threadInPath = match[1];
const threadId = match[2];
const createSaveButton = () => {
const btn = document.createElement("a");
btn.classList.add("pageNav-jump", "pageNav-jump--next");
btn.textContent = "Lưu Thread";
btn.style.cursor = "pointer";
btn.addEventListener("click", async () => {
await saveThread(threadId, threadInPath);
});
return btn;
};
document.querySelectorAll(".p-description").forEach(desc => {
const ul = desc.querySelector("ul.listInline");
if (ul) {
const buttonContainer = document.createElement("div");
buttonContainer.style.display = "flex";
buttonContainer.style.flexWrap = "wrap";
buttonContainer.style.gap = "5px";
buttonContainer.style.marginTop = "5px";
const btn = createSaveButton();
if (isButtonEnabled) {
const viewBtn = document.createElement("a");
viewBtn.classList.add("pageNav-jump", "pageNav-jump--next");
viewBtn.textContent = "Xem đã lưu";
viewBtn.style.cursor = "pointer";
viewBtn.addEventListener("click", viewSavedThreads);
buttonContainer.appendChild(viewBtn);
}
buttonContainer.appendChild(btn);
ul.after(buttonContainer);
}
});
})();