// ==UserScript== // @name HDrezka Helper // @version 4.3.3 // @description Adds a «Download» button below the video. Export favorites and more. // @author Super Zombi // @match https://hdrezka.cm/* // @match https://rezka.ag/* // @match https://hdrezka.me/* // @match https://hdrezka.ag/* // @match https://hdrezka.co/* // @icon  // @homepageURL https://github.com/SuperZombi/HDrezka-Helper // @supportURL https://github.com/SuperZombi/HDrezka-Helper/issues // @updateURL https://raw.githubusercontent.com/SuperZombi/HDrezka-Helper/main/hdrezka_helper.user.js // @downloadURL https://raw.githubusercontent.com/SuperZombi/HDrezka-Helper/main/hdrezka_helper.user.js // @grant GM_setValue // @grant GM_getValue // @grant GM_deleteValue // @grant GM_listValues // @grant GM_registerMenuCommand // ==/UserScript== (function() { 'use strict'; const locale = { 'en': { "settings": "Settings", "saveBut": "Save", "resetBut": "Reset", "closeBut": "Close", "downloadBut": "«Download» button", "exportStr": "Export", "importStr": "Import", "exportBut": "Export Favorites", "animationBut": "Animation", "hideVkBut": "Hide VK", "downloadStr": "Download", "downloadLinkDesc": "Alt click on the link to save the file", "subtitles": "Subtitles", "cancelDownload": "Cancel", "fileNamePattern": "File name pattern", "insertVariable": "Insert Variable", "help": "Help", "errorStr": "Error!", "movieTitle": "Movie title", "season": "Season", "episode": "Episode", "translation": "Translation", "resolution": "Resolution", "vpnErrorMsg": "Try using a VPN!", "donateTitle": "Enjoying this extension?", "donateButton": "Donate" }, 'ru': { "settings": "Настройки", "saveBut": "Сохранить", "resetBut": "Сбросить", "closeBut": "Закрыть", "downloadBut": "Кнопка «Скачать»", "exportStr": "Экспорт", "importStr": "Импорт", "exportBut": "Экспорт закладок", "animationBut": "Анимация", "hideVkBut": "Скрывать VK", "downloadStr": "Скачать", "downloadLinkDesc": "Нажмите на ссылку, удерживая клавишу Alt, чтобы сохранить файл", "subtitles": "Субтитры", "cancelDownload": "Отменить", "fileNamePattern": "Шаблон имени файла", "insertVariable": "Вставить переменную", "help": "Помощь", "errorStr": "Ошибка!", "movieTitle": "Название фильма", "season": "Сезон", "episode": "Эпизод", "translation": "Перевод", "resolution": "Качество", "vpnErrorMsg": "Попробуйте использовать VPN!", "donateTitle": "Нравится расширение?", "donateButton": "Пожертвовать" }, 'uk': { "settings": "Налаштування", "saveBut": "Зберегти", "resetBut": "Скинути", "closeBut": "Закрити", "downloadBut": "Кнопка «Завантажити»", "exportStr": "Експорт", "importStr": "Імпорт", "exportBut": "Експорт закладок", "animationBut": "Анімація", "hideVkBut": "Приховувати VK", "downloadStr": "Завантажити", "downloadLinkDesc": "Натисніть на посилання, утримуючи клавішу Alt, щоб зберегти файл", "subtitles": "Субтитри", "cancelDownload": "Скасувати", "fileNamePattern": "Шаблон імені файлу", "insertVariable": "Вставити змінну", "help": "Допомога", "errorStr": "Помилка!", "movieTitle": "Назва фільму", "season": "Сезон", "episode": "Епізод", "translation": "Переклад", "resolution": "Якість", "vpnErrorMsg": "Спробуйте скористатися VPN!", "donateTitle": "Подобається розширення?", "donateButton": "Пожертвувати" } } function get_message(name, default_="en"){ var userLang = (navigator.language || navigator.userLanguage).slice(0,2).toLowerCase(); if (Object.keys(locale).includes(userLang) && Object.keys(locale[userLang]).includes(name)){ return locale[userLang][name] } return locale[default_][name] } function db_get(value, default_=undefined){ let val = GM_getValue(value) if (val != undefined){ return val } else{ return default_ } } function db_save(array){ for (let el in array) { GM_setValue(el, array[el]) } } GM_registerMenuCommand(get_message('settings'), ()=>{ let div = document.createElement("div") div.style.position = 'fixed' div.style.zIndex = 10000 div.style.top = 0 div.style.right = "-50%" div.style.willChange = "right" div.style.transition = "0.5s" div.style.borderRadius = "0 0 0 12px"; div.style.background = "white"; div.style.filter = "drop-shadow(0px 0px 2px black)"; div.style.fontSize = "14pt" div.style.minWidth = "300px" if (document.body.classList.contains("b-theme__template__night")){ div.style.backgroundColor = "#181818"; div.style.color = "white"; div.style.filter = "drop-shadow(0px 0px 2px white)"; } let close = document.createElement("div") close.style.textAlign = "right" close.style.cursor = "pointer" close.style.margin = "10px" let img = document.createElement("img") img.title = get_message("closeBut") img.style.height = '20px' img.src = '' img.onclick = _ => { div.style.right = "-50%" setTimeout(()=>{ div.remove() }, 500) } close.appendChild(img) div.appendChild(close) let content = document.createElement("div") content.style.textAlign = "center" content.style.padding = "20px" content.style.paddingTop = 0 content.innerHTML = `

${get_message('settings')}



GitHub

` div.appendChild(content) let keys = GM_listValues() keys.forEach(e=>{ if (e != 'filename_structure'){ try{ let els = div.querySelectorAll(`input[name=${e}]`) if (els.length > 1){ let val = db_get(e) let el = Array.from(els).filter(e=>e.value == val)[0] el.checked = true } else{ els[0].checked = db_get(e) } } catch{ GM_deleteValue(e) } } }) if (keys['filename_structure'] && keys['filename_structure'] != ""){ dinamic_input(keys['filename_structure']) }else{ dinamic_input("%title s-%s ep-%ep [%transl]") } let save = div.querySelector("button[name=save]") save.onclick = () => initSave() window.addEventListener("keydown", function(e){ if (e.keyCode == 17){ // CTRL if (!event.repeat){ save.innerHTML = get_message('resetBut') save.onclick = _ => { let keys = GM_listValues() keys.forEach(key=>{ GM_deleteValue(key) }) window.location.reload() } } } }) window.addEventListener("keyup", function(e){ if (e.keyCode == 17){ // CTRL save.innerHTML = get_message('saveBut') save.onclick = () => initSave() } }) function initSave(){ let settings = {} let inputs = div.querySelectorAll("input.settings") Array.from(inputs).forEach(e=>{ if (e.type == "checkbox"){ settings[e.name] = e.checked } else if (e.type == "radio" && e.checked){ settings[e.name] = e.value } }) settings['filename_structure'] = div.querySelector("#filename_structure").value db_save(settings) window.location.reload() } function dinamic_input(init_value){ var input = div.querySelector("#filename_structure") set_value(init_value) div.querySelector("[name=downloader_2]").onchange = e=>{ if (e.target.checked){ div.querySelector("#filename_structure_block").style.display = "block" } else{ div.querySelector("#filename_structure_block").style.display = "none" } } if (div.querySelector("[name=downloader_2]").checked){ div.querySelector("#filename_structure_block").style.display = "block" } div.querySelectorAll("#variables_list > *").forEach(e=>{ e.onmouseover = _=>{ e.style.background = "#00C0FF" } e.onmouseout = _=>{ e.style.background = "" } e.onclick = _=>{ if (input.value != ""){ input.value += " " } input.value += e.getAttribute("value") div.querySelector("#variables_list").style.display = "none" input.focus() } }) function hideInsertMenu(){ div.querySelector("#variables_list").style.display = "none" div.removeEventListener("click", hideInsertMenu) } div.querySelector("#insert_variable_but").onclick = _=>{ div.querySelector("#variables_list").style.display = "flex" setTimeout(function(){div.addEventListener("click", hideInsertMenu)}, 500) } input.addEventListener("focus", _=>{ input.selectionStart = input.selectionEnd = input.value.length; }) function set_value(text){ input.value = text } } document.body.appendChild(div) setTimeout(()=>{ div.style.right = 0; }, 1) }) var LOADER = ` ` function main(){ let keys = GM_listValues() if (db_get("hideVK", true)){ hideVK() } var player = document.getElementById('player') if (player){ document.title = document.querySelector('.b-content__main .b-post__title').innerText; if (db_get("download", true)){ downloader(); var temp_video_src = player.getElementsByTagName("video")[0].src let observer = new MutationObserver(function(mutations) { mutations.forEach(function(mutation) { if (temp_video_src != document.getElementById('player').getElementsByTagName("video")[0].src) { temp_video_src = document.getElementById('player').getElementsByTagName("video")[0].src downloader(); } }); }); observer.observe(document.querySelector("body"), {childList: true, subtree: true}); } } else if (window.location.pathname.includes("/favorites/") && db_get("exportFavorites", true)){ exportFavorites() } } window.onload = _=>main(); function hideVK(){ let el_ = document.getElementById("vk_groups"); if (el_){ el_.style.display = "none"; } } async function downloader(){ var arr = clearTrash(CDNPlayerInfo.streams).split(",") createButton() await createDownloadMenu(arr) if (db_get("subtitles", true)){ addSubtitles() } function createButton() { if (!document.getElementById("downloadButton")){ let el = document.getElementById("send-video-issue") let div = document.createElement("div"); div.id = "downloadButton" div.title = get_message('downloadStr') div.innerHTML = ` ` div.style.right = "55px" div.style.top = "0" div.style.height = "50px" div.style.width = "50px" div.style.position = "absolute" div.style.cursor = "pointer" div.style.transition = "0.3s" div.style.background = "#2d2d2d" div.onmouseover = function(){ div.style.background = "#4D4D4D" } div.onmouseout = function(){ div.style.background = "#2d2d2d" } div.onclick = show_download_menu el.parentNode.insertBefore(div, el); } } async function createDownloadMenu(array){ if (!document.getElementById("downloadMenu")){ let div = document.createElement("div") div.id = "downloadMenu" div.style.minHeight = "50px" div.style.width = "160px" div.style.background = "rgba(93, 93, 93, 0.5)" div.style.backdropFilter = "blur(5px)" div.style.position = "absolute" div.style.borderRadius = "6px" div.style.padding = "4px" div.style.filter = "drop-shadow(black 2px 4px 6px)" div.style.zIndex = "100" div.style.right = "0" div.style.top = "55px" div.style.display = "none" div.style.opacity = 0 div.style.transform = "scale(0)" div.style.transformOrigin = "top center" div.style.transition = "0.5s" div.style.userSelect = "none" div.style.overflow = "hidden" div.innerHTML = LOADER; document.getElementById("send-video-issue").parentNode.appendChild(div) } else{ document.getElementById("downloadMenu").innerHTML = LOADER; } let div_target = document.getElementById("downloadMenu") let div_ = document.createElement("div") for (const e of array) { var temp = e.split("[")[1].split("]"); var quality = temp[0]; var links = temp[1].split(" or ").filter(x=>x.endsWith('.mp4')) for (let link of links){ let size = await getFileSize(link); if (size){ size = formatBytes(size, 1); let element = makeLink(quality, link, size); div_.appendChild(element); break } else{ console.error({"_": "Error", "name": quality, "url": link}) } } } div_target.innerHTML = "" div_target.appendChild(div_) if (div_.children.length == 0){ vpnAlert(div_target) } else{ donationPopup(div_target) } } function buildFileName(name, season, episode, translation, res){ if (db_get("filename_structure")){ var selectArray = { "title": name, "s": season, "ep": episode, "transl": translation, "res": res } let temp = db_get("filename_structure") Object.keys(selectArray).forEach(function(e){ if (selectArray[e]){ temp = temp.replaceAll("%" + e, selectArray[e]) }else{ temp = temp.replaceAll("%" + e, "") } }) return temp }else{return "video"} } function makeLink(title, href, size){ let filename = href.split('/').pop() let a = document.createElement("a") if (db_get("downloader_2", false)){ a.title = get_message('downloadStr') a.onclick = _=>{ if (!a.getAttribute("blocked")){ a.setAttribute("blocked", true) let season, episode, translation, name; var xhr = new XMLHttpRequest(); let el = document.querySelector("#simple-episodes-tabs .active") if (el){ season = el.getAttribute("data-season_id") episode = el.getAttribute("data-episode_id") } let el2 = document.querySelector("#translators-list .active") if (el2){ translation = el2.innerText } name = document.querySelector('.b-content__main .b-post__title').innerText var targetFileName = buildFileName(name, season, episode, translation, title) let div = document.createElement("span") div.className = "download-area" div.style.display = "flex" div.style.alignItems = "center" div.style.padding = "6px 0" let progress = document.createElement("progress") progress.max = 100; let percentage = document.createElement("span") percentage.style.marginLeft = "5px" percentage.innerHTML = "0%" let close_but = document.createElement("button") close_but.innerHTML = "✖" close_but.style.marginLeft = "5px" close_but.style.borderRadius = "50px" close_but.style.border = "2px solid transparent" close_but.style.height = "20px"; close_but.style.width = "20px"; close_but.style.display = "flex"; close_but.style.alignItems = "center"; close_but.style.justifyContent = "center"; close_but.style.color = "red"; close_but.style.transition = "0.25s" close_but.title = get_message('cancelDownload') close_but.onclick = _=>{ xhr.abort(); a.querySelector(".download-area").remove() a.style.background = null setTimeout(function(){ a.removeAttribute("blocked") }, 100) } close_but.onmouseover = _=>{ close_but.style.borderColor = "red" } close_but.onmouseout = _=>{ close_but.style.borderColor = "transparent" } div.appendChild(progress) div.appendChild(percentage) div.appendChild(close_but) a.appendChild(div) window.URL = window.URL || window.webkitURL; xhr.open('GET', href, true); xhr.responseType = 'blob'; xhr.onprogress = prog=>{ let percentComplete = Math.round((prog.loaded / prog.total) * 100); progress.value = percentComplete; percentage.innerHTML = percentComplete + "%" } xhr.onload = function () { var file = new Blob([xhr.response], { type : 'application/octet-stream' }); let a_el = document.createElement('a') a_el.href = window.URL.createObjectURL(file); let extension = filename.split('.').pop(); a_el.download = `${targetFileName}.${extension}`; a_el.click(); setTimeout(function(){ close_but.click(); }, 1000) }; xhr.send(); } } } else{ a.href = href a.target = '_blank' a.download = filename a.title = get_message('downloadLinkDesc') } a.style.display = "block" a.style.color = "white" a.style.textDecoration = "none" a.style.padding = "4px 5px" a.style.margin = "2px 0" a.style.borderRadius = "6px" a.style.transition = "0.2s" a.style.cursor = "pointer" a.onmouseover = function(){ a.style.background = "rgb(0, 0, 255, 0.75)" } a.onmouseout = function(){ a.style.background = null } let span = document.createElement("span") span.innerHTML = title let span2 = document.createElement("span") span2.style.float = "right" span2.innerHTML = size a.appendChild(span) a.appendChild(span2) return a; } async function getFileSize(url) { return new Promise(async (resolve) => { let controller = new AbortController(); fetch(url, {signal: controller.signal}).then(resp=>{ resolve(resp.headers.get('Content-Length')) controller.abort(); }).catch(_=>{resolve(0)}) }) } function formatBytes(bytes, decimals = 2) { if (bytes == 0) return ''; const k = 1024; const dm = decimals < 0 ? 0 : decimals; const sizes = ['B', 'KB', 'MB', 'GB', 'TB']; const i = Math.floor(Math.log(bytes) / Math.log(k)); return parseFloat((bytes / Math.pow(k, i)).toFixed(dm)) + ' ' + sizes[i]; } var timer; function show_download_menu(){ let div = document.getElementById("downloadMenu") setTimeout(function(){ document.body.onclick = hide_download_menu }, 50) if (div.style.display == "none"){ div.style.display = "block" setTimeout(function(){ div.style.transform = "scale(1)" div.style.opacity = 1 }, 10) } else{ if (timer) { clearTimeout(timer); div.style.transform = "scale(1)" div.style.opacity = 1 } } } function hide_download_menu(event){ let div = document.getElementById("downloadMenu") let path = event.path || (event.composedPath && event.composedPath()); if (!path.includes(div)){ div.style.transform = "scale(0)" div.style.opacity = 0 setTimeout(function(){ document.body.onclick = "" }, 50) timer = setTimeout(function(){ div.style.display = "none" }, 400) } } function addSubtitles(){ const Subtitles = CDNPlayerInfo.subtitle; if (Subtitles){ let div_ = document.getElementById("downloadMenu"); let details = document.createElement("details"); details.style.border = "1px solid white"; details.style.borderRadius = "8px"; details.style.margin = "2px"; details.style.marginTop = "8px"; details.style.cursor = "pointer"; let summary = document.createElement("summary"); summary.innerHTML = get_message('subtitles'); summary.style.color = "aqua"; summary.style.borderRadius = "8px"; summary.style.textAlign = "center"; summary.style.transition = "0.2s"; summary.style.padding = "2px 0"; summary.onmouseover = function(){ summary.style.background = "blueviolet" } summary.onmouseout = function(){ summary.style.background = null } details.appendChild(summary); div_.appendChild(details); let hr = document.createElement("hr"); hr.style.margin = 0; details.appendChild(hr); Subtitles.split(",").forEach(async function(e){ let temp = e.split("[")[1].split("]"); let lang = temp[0]; let link = temp[1]; let size = await getFileSize(link); size = formatBytes(size, 1); let element = makeLink(lang, link, size); details.appendChild(element); }) } } function clearTrash(data){ function product(iterables, repeat) { var argv = Array.prototype.slice.call(arguments), argc = argv.length; if (argc === 2 && !isNaN(argv[argc - 1])) { var copies = []; for (var i = 0; i < argv[argc - 1]; i++) { copies.push(argv[0].slice()); } argv = copies; } return argv.reduce(function tl(accumulator, value) { var tmp = []; accumulator.forEach(function(a0) { value.forEach(function(a1) { tmp.push(a0.concat(a1)); }); }); return tmp; }, [[]]); } function unite(arr){ var final = []; arr.forEach(function(e){ final.push(e.join("")) }) return final; } var trashList = ["@","#","!","^","$"]; var two = unite(product(trashList, 2)); var tree = unite(product(trashList, 3)); var trashCodesSet = two.concat(tree); var arr = data.replace("#h", "").split("//_//"); var trashString = arr.join(''); trashCodesSet.forEach(function(i){ var temp = btoa(i) trashString = trashString.replaceAll(temp, '') }) var final_string = atob(trashString); return final_string; } function makePopup(parrent){ let div = document.createElement("div") div.style.inset = 0 div.style.position = "absolute" div.style.zIndex = 2 div.style.background = "rgb(0, 0, 0, 0.85)" div.style.padding = "5px" div.style.textAlign = "center" div.style.display = "flex"; div.style.flexDirection = "column"; div.style.justifyContent = "center"; div.style.alignItems = "center"; div.style.gap = "5px"; parrent.appendChild(div) return div } function vpnAlert(div_target){ let popup = makePopup(div_target) popup.style.background = "unset" popup.style.position = "unset" popup.style.padding = "10px" popup.innerHTML = `
${get_message('errorStr')}
${get_message('vpnErrorMsg')}
` } function donationPopup(menu_element){ function checkLastNotificationTime(){ let currentTime = Math.floor(Date.now() / 1000); let lastNotificationTime = localStorage.getItem('lastNotificationTime'); if (!lastNotificationTime || (currentTime - lastNotificationTime > 12*60*60)) { return true; } else { return false; } } if (checkLastNotificationTime()){ let popup = makePopup(menu_element) menu_element.style.minWidth = "175px" popup.innerHTML = `
${get_message('donateTitle')}
` let button = popup.querySelector("button") button.onmouseover = _=>{ button.style.color = "white"; button.style.background = "var(--glow-color)"; button.style.boxShadow = "0 0 1.25em .5em var(--glow-color)"; } button.onmouseout = _=>{ button.style.color = "var(--glow-color)"; button.style.background = "black"; button.style.boxShadow = "0 0 1em .25em var(--glow-color), inset 0 0 1em 0 var(--glow-color)"; } button.onclick = _=>{ popup.remove() var currentTime = Math.floor(Date.now() / 1000); localStorage.setItem('lastNotificationTime', currentTime); } } } } function exportFavorites(){ if(window.location.hash){ let num = parseInt(window.location.hash.substring(1).replace("parsing", "")) let arr = JSON.parse(GM_getValue('parseArray')); let el = document.getElementsByClassName("b-content__htitle")[0]; let span = document.createElement("span"); let exportStr = get_message("exportStr"); span.innerHTML = `— ${exportStr} ${num+1}/${arr.length}` span.style.color = "cornflowerblue"; el.appendChild(span) GM_setValue(`parsedArray${num}`, JSON.stringify(parseContent())); if (num+1 >= arr.length){ var finalResult = {}; GM_deleteValue(`parseArray`); for (let i = 0; i < arr.length; i++) { let parsed = JSON.parse(GM_getValue(`parsedArray${i}`)); finalResult[arr[i].name] = parsed GM_deleteValue(`parsedArray${i}`); } var finalFile = {} finalFile["version"] = "1.0" finalFile["data"] = finalResult download('my_favorites.json', JSON.stringify(finalFile, null, 2)); document.location.replace(window.location.origin + "/favorites/") } else{ document.location.replace(arr[num+1].link) } } else{ let el = document.getElementsByClassName("b-content__htitle")[0]; let div = document.createElement("div"); div.innerHTML = ` ` div.style.cursor = "pointer" div.style.verticalAlign = "bottom" div.style.borderRadius = "8px" div.style.transition = "0.3s" div.style.boxShadow = "0 0 4px grey" div.style.position = "relative" div.style.zIndex = 2 div.onmouseover = function(){ div.style.boxShadow = "0 0 8px grey" } div.onmouseout = function(){ div.style.boxShadow = "0 0 4px grey" } let popup = document.createElement("div") popup.style.position = "absolute" popup.style.visibility = "hidden" popup.style.transition = "0.4s" popup.style.background = "white" popup.style.boxShadow = "0 0 4px lightgrey" if (document.body.classList.contains("b-theme__template__night")){ popup.style.background = "#222d33" popup.style.boxShadow = "0 0 4px grey" } popup.style.left = 0 popup.style.borderRadius = "10px" popup.style.marginTop = "5px" popup.style.padding = "10px 5px" popup.style.display = "flex" popup.style.flexDirection = "row" if (db_get('export_animation', true)){ popup.style.opacity = 0 popup.style.transform = "scale(0)" popup.style.transformOrigin = "left top" } popup.style.cursor = "auto" div.onclick = _=>{ if (popup.style.visibility != "visible"){ if (db_get('export_animation', true)){ popup.style.transform = "scale(1)" popup.style.opacity = 1 } popup.style.visibility = "visible" function hide_popup_menu(event){ let path = event.path || (event.composedPath && event.composedPath()); if (!path.includes(popup)){ if (db_get('export_animation', true)){ popup.style.transform = "scale(0)" popup.style.opacity = 0 } popup.style.visibility = "hidden" document.body.removeEventListener("click", hide_popup_menu) } } setTimeout(function(){ document.body.addEventListener("click", hide_popup_menu) }, 50) } } div.appendChild(popup) function makeButton(){ let but = document.createElement("div") but.style.cursor = "pointer" but.style.borderRadius = "8px" but.style.boxShadow = "0 0 4px grey" but.style.height = "35px" but.style.width = "35px" but.style.padding = "5px" but.style.margin = "0 5px" but.style.display = "inline-block" but.style.transition = "0.2s" but.onmouseover = function(){ but.style.background = "rgb(150, 150, 150, 0.25)" but.style.transform = "translateY(-2px)" but.style.filter = "brightness(1.1)" but.style.borderRadius = "14px" } but.onmouseout = function(){ but.style.background = "" but.style.filter = "" but.style.transform = "" but.style.borderRadius = "8px" } return but } let export_ = makeButton() export_.innerHTML = ` ` export_.title = get_message("exportStr") export_.onclick = parseFavorites popup.appendChild(export_) let import_ = makeButton() import_.innerHTML = ` ` import_.title = get_message("importStr") import_.onclick = importing popup.appendChild(import_) if (db_get('export_animation', true)){ div.style.height = "0px" div.style.width = "0px" div.style.margin = "0 0px" div.style.padding = "0px" div.style.transform = "scale(0)" div.style.transformOrigin = "center" div.style.opacity = 0 el.prepend(div) setTimeout(function(){ div.style.height = "35px" div.style.width = "35px" div.style.margin = "0 5px" div.style.padding = "5px" div.style.transform = "scale(1)" div.style.display = "inline-block" div.style.opacity = 1 }, 50) } else{ div.style.height = "35px" div.style.width = "35px" div.style.margin = "0 5px" div.style.padding = "5px" div.style.display = "inline-block" el.prepend(div) } } function getArray(){ var temp = []; var folders = document.getElementById("user-favorites-holder").getElementsByClassName("b-favorites_content__cats_list_link") Object.keys(folders).forEach(function(e){ var el = folders[e] var link = el.getAttribute("href") var name = el.getElementsByClassName("name")[0].innerHTML temp.push({ name: name, link: link + `#parsing${e}` }) }) return temp; } function parseFavorites(){ var array = getArray() GM_setValue('parseArray', JSON.stringify(array)); document.location.replace(array[0].link) setTimeout(function(){ document.location.reload() }, 500) } function parseContent(){ var temp = []; var list = document.getElementsByClassName("b-favorites_content__holder")[0].children Object.keys(list).forEach(function(e){ let el = list[e] let url = el.getAttribute("data-url") let id_ = el.getAttribute("data-id") let name = el.getElementsByClassName("b-content__inline_item-link")[0].children[0].innerHTML temp.push({ name: name, url: url, id: id_ }) }) return temp } function download(filename, text) { var element = document.createElement('a'); element.setAttribute('href', 'data:text/plain;charset=utf-8,' + encodeURIComponent(text)); element.setAttribute('download', filename); element.style.display = 'none'; document.body.appendChild(element); element.click(); document.body.removeChild(element); } function importMsg(msg, clear=false){ let el = document.getElementsByClassName("b-content__htitle")[0]; let span = el.querySelector("[name=importMsg]") if (span && clear){span.innerHTML = "";} if (clear){return} if (!span){ span = document.createElement("span"); span.setAttribute("name", "importMsg") span.style.color = "red"; el.appendChild(span) } span.innerHTML = `— ${msg}` } function importing(){ let input = document.createElement('input'); input.type = 'file'; input.onchange = e => { let file = e.target.files[0]; const reader = new FileReader(); reader.addEventListener('load', (event) => { try{ let file_data = JSON.parse(event.target.result) if (file_data.version && file_data.data){ importMsg("", true) import_all(file_data.data) } else{ importMsg("Invalid File") } } catch (ex){ console.error(ex) importMsg("Invalid File") } }); reader.readAsText(file); } input.click(); } async function import_all(arr){ async function postData(url, data) { const response = await fetch(url, { method: 'POST', headers: { 'Content-Type': 'application/x-www-form-urlencoded'}, body: new URLSearchParams(data) }); return response.json(); } let error_arr = []; await Promise.all( Object.entries(arr).map(async ([k, v]) => { let r = await postData(window.location.origin + "/ajax/favorites/", { "name": k, "action": "add_cat" }) if (r.success == true){ v.forEach(async e => { let r2 = await postData(window.location.origin + "/ajax/favorites/", { "post_id": e.id, "cat_id": r.id, "action": "add_post" }) if (r2.success != true){ error_arr.push({ "action": "add_post", "name": e.name, ...r2 }) } }) } else{ error_arr.push({ "action": "add_catetory", "name": k, ...r }) } }) ); if (error_arr.length == 0){ setTimeout(function(){ window.location.reload(); }, 500) } else{ console.error(error_arr) alert(JSON.stringify(error_arr, null, 2)) } } } })();