NOTICE: By continued use of this site you understand and agree to the binding Terms of Service and Privacy Policy.
// ==UserScript== // @name FA Hotkeys // @namespace Artex // @description hotkeys for furaffinity // @author Artex // @homepageURL http://www.furaffinity.net/user/artex./ // @homeURL https://openuserjs.org/scripts/Artex/FA_Hotkeys // @icon  // @include http://www.furaffinity.net/* // @include https://www.furaffinity.net/* // @run-at document-end // @version 1.6.4 // @grant none // ==/UserScript== /*jshint esnext: true */ var settingsOpen = false; var settingsMenu = null; var hotkeys = new Map(); var storage = window.localStorage; var isResized = false; //for fit to screen var smallImage = ""; //small submission view var debugEnabled = false; function log() { if (debugEnabled === true) { console.log.apply(console, arguments); } } //get the name of the key that was pressed function keyFromKeyboardEvent(event) { var key = null; if (event.key) { key = event.key; } else { var identifier = event.keyIdentifier; if (identifier.search(/^U\+/) === -1) { key = identifier; } else { //not unicode, likely the key name key = String.fromCharCode(parseInt(identifier.replace(/^U\+/,""), 16)).toLowerCase(); } } return key; } //page url contains the given string function windowUrlContains(url) { var windowUrl = window.location.href; if (windowUrl.search(url) != -1) { return true; } else { return false; } } function nextElement(n) { var next = n.nextSibling; if (next.nodeType == 1 || next === null) { return next; } else { return nextElement(next); } } function previousElement(n) { var previous = n.previousSibling; if (previous.nodeType == 1 || previous === null) { return previous; } else { return previousElement(previous); } } function isTyping() { // is a text field in focus? var active = document.activeElement; if (active.tagName == "TEXTAREA" || active.tagName == "INPUT") { return true; } else { return false; } } function getSubmissionSource(){ var controlBar = document.getElementsByClassName("button submission"); for (var i=0; i < controlBar.length; i++) { var link = controlBar[i].getElementsByTagName("a")[0]; if (link !== undefined && link.textContent == "Download") { return link.href; } } } //////////////////////////////////////// // hotkey actions /////////////////////////////////////// function fullView() { //sets image source to loading gif and then full image if (windowUrlContains(/www.furaffinity.net\/(view|full)/)) { var image = document.getElementById("submissionImg"); var largeImage = getSubmissionSource(); if (largeImage != image.src) { smallImage = image.src; image.src = loadingGif.src; image.src = largeImage; } else { image.src = smallImage; } /* old full view code var url = window.location.href; var newUrl = ""; if (url.search("view") != -1) { newUrl = url.replace("view", "full"); } else { newUrl = url.replace("full", "view"); } window.location.href = newUrl; */ } } function fitToScreen() { function ResizeImage(img) { var viewportX = window.innerWidth; var viewportY = window.innerHeight; var imageX = img.naturalWidth; var imageY = img.naturalHeight; if (isResized) { //if resized, return normal dimensions. img.width = imageX; img.height = imageY; isResized = false; } else { if (imageX > viewportX) { img.width = viewportX; img.height = imageY * (viewportX/imageX); } if (imageY > viewportY) { //image is larger then window img.width = imageX * (viewportY/imageY); img.height = viewportY; } isResized = true; } } var image = document.getElementById('submissionImg'); if (image) { window.location.hash = '#submissionImg'; ResizeImage(image); } } function favorite() { if (windowUrlContains(/www.furaffinity.net\/(view|full)/)) { var favButton = document.getElementsByClassName("button fav")[0]; if (favButton) { favButton.click(); } } /* old favorite code document.querySelector('a.p20r').click(); */ } function navigateLeft() { if (windowUrlContains(/www.furaffinity.net\/(view|full)/)) { var center = document.getElementsByClassName("minigallery-title")[0]; previousElement(center).firstChild.firstChild.firstChild.click(); //Prone to breaking if structure changes } else if (windowUrlContains(/www.furaffinity.net\/(gallery|scraps|favorites)/)) { var url = window.location.href; var reg = /(favorites|gallery|scraps)(\/.+\/)([^/]+)(\/$|$)/; var pageNum = url.match(reg); pageNum = pageNum === null ? 2 : pageNum[3]; pageNum = pageNum > 1 ? pageNum : 2; var nextPage = url.replace(reg, "$1$2" + (pageNum - 1) + "$4"); window.location.href = nextPage; } else if (windowUrlContains(/www.furaffinity.net\/browse/)) { var url = window.location.href; var reg = /(browse\/)(\d+)(?=\/$|$)/; var pageNum = url.match(reg); pageNum = pageNum === null ? 2 : pageNum[2]; pageNum = pageNum > 1 ? pageNum : 2; var nextPage = url.replace(reg, "$1" + (pageNum - 1)); window.location.href = nextPage; } } function navigateRight() { if (windowUrlContains(/www.furaffinity.net\/(view|full)/)) { var center = document.getElementsByClassName("minigallery-title")[0]; nextElement(center).firstChild.firstChild.firstChild.click(); } else if (windowUrlContains(/www.furaffinity.net\/(gallery|scraps|favorites)/)) { var url = window.location.href; var reg = /(favorites|gallery|scraps)(\/.+\/)([^/]+)(\/$|$)/; var pageNum = url.match(reg); var nextPage; if (pageNum === null) { //console.log(pageNum); nextPage = url + "2/"; } else { //console.log(pageNum); nextPage = url.replace(reg, "$1$2" + (+pageNum[3] + 1) + "$4"); } window.location.href = nextPage; } else if (windowUrlContains(/www.furaffinity.net\/browse/)) { console.log("TEST"); var url = window.location.href; var reg = /(browse\/)(\d+)(?=\/$|$)/; var pageNum = url.match(reg); console.log(pageNum); var nextPage; if (pageNum === null) { //console.log(pageNum); nextPage = url + "2/"; } else { //console.log(pageNum); nextPage = url.replace(reg, "$1" + (+pageNum[2] + 1)); } window.location.href = nextPage; } } function checkAll() { var button = document.getElementsByClassName("section-button invert-selection")[0]; if (button !== undefined) { button.click(); } } function removeChecked() { var button = document.getElementsByClassName("section-button remove-checked")[0]; if (button !== undefined) { button.click(); } } //////////////////////////////////////// // setup hotkeys /////////////////////////////////////// function saveHotkeys() { var save = ""; hotkeys.forEach(function(func, key) { save = save + key + ":" + func.name + "|"; }); storage.setItem("FA_HotkeySettings", save); log("hotkeys saved"); } function loadDefaultHotkeys() { //if no keys are saved or data to reset keys. hotkeys = new Map(); hotkeys.set("f", favorite); hotkeys.set("g", fitToScreen); hotkeys.set("ArrowLeft", navigateLeft); hotkeys.set("ArrowRight", navigateRight); hotkeys.set("q", checkAll); hotkeys.set("e", removeChecked); hotkeys.set("1", openSettings); log("hotkeys set, saving"); saveHotkeys(); } function loadHotkeys() {//load saved hotkey configuration. //format "key:functionName|key:functionName" var data = storage.getItem("FA_HotkeySettings"); var dataSuccess = false; log("loading hotkey data: ", data); if (data !== null) { var matches = data.match(/[^\|]+/g);//get key:value - note: would probably break if key was ':' or '|' if (matches !== null) { //bad save data for (var i in matches ) { if (typeof matches[i] === "string") { var key = matches[i].match(/^[^\:]+/g)[0]; var func = matches[i].match(/[^\:]+$/g)[0]; //a more direct connection between the function name and the function would be ideal.. if (func.search(/[\.\(\)]/g) === -1) { //hopefully filter out malacious modifications to localstorage hotkeys.set(key, eval(func)); //eval is my last resort to get the function pointer //Idea: if I assign the functions to a map I can properly get the functions with a string. } } } } else { dataSuccess = false; } } if (dataSuccess === false) { log("loading default hotkeys"); loadDefaultHotkeys(); } } function initiateHotkeys() { log("iniating hotkey script"); loadHotkeys(); //loadDefaultHotkeys(); //take key input and watch for hotkeys to call. window.addEventListener("keydown", function(event) { var key = keyFromKeyboardEvent(event); log("key:", key); log("unicode:", event.keyIdentifier); if (hotkeys.has(key) === true && isTyping() === false) { hotkeys.get(key)(); } }, true); } //////////////////////////////////////// // Hotkey settings /////////////////////////////////////// function openSettings() { if (settingsOpen === false) { settingsOpen = true; var inputs = []; //array of the text inputs //This isn't going to be pretty //create style sheet for settings menu var style = document.createElement("style"); style.innerHTML = "" + "#Hotkey-Settings { " + " position: fixed;" + " z-index: 99999;" + " width: 250px;" + " left: 50%;" + " right: 50%;" + " top: 50%;" + " transform: translate(-50%, -50%);" + " background-color: #DDD;" + " color: #2F2F2F;" + " padding: 5px;" + "} "; //generate settings menu var container = document.createElement("table"); var header = document.createElement("thead"); var tr1 = document.createElement("tr"); var th1 = document.createElement("th"); var titleLeft = document.createTextNode("Key"); var th2 = document.createElement("th"); var titleRight = document.createTextNode("Action"); container.appendChild(style); container.appendChild(header); header.appendChild(tr1); tr1.appendChild(th1); th1.appendChild(titleLeft); tr1.appendChild(th2); th2.appendChild(titleRight); //generate input fields var n = 0; hotkeys.forEach(function(func, key) { var row = document.createElement("tr"); var td1 = document.createElement("td"); var input = document.createElement("input"); var td2 = document.createElement("td"); var name = document.createTextNode(func.name);//hopefully func.name has enough support. container.appendChild(row); row.appendChild(td1); //append input later row.appendChild(td2); td2.appendChild(name); input.setAttribute("type","text"); input.setAttribute("name",func.name); input.setAttribute("value",key); td1.appendChild(input); inputs[n++] = input; //hotkey input text field input.addEventListener("keydown", function(event) { var key = keyFromKeyboardEvent(event); log("input: ", key); if (key !== "Enter") { event.preventDefault(); var oldKey = this.getAttribute("value"); var newKey = key; var action = hotkeys.get(oldKey); if (hotkeys.get(newKey) === undefined) { hotkeys.delete(oldKey); hotkeys.set(newKey, action); this.setAttribute("value", newKey); saveHotkeys(); } else { //hotkey already in use, alert user var alertStyle = document.createElement("style"); alertStyle.innerHTML = "" + "#Hotkey-Settings input[value='" + newKey + "'] {" + " -webkit-transition: background-color 0.2s;" + " transition: background-color 0.2s;" + " background-color: #FFA3A3;" + "} "; container.appendChild(alertStyle); window.setTimeout(function() { alertStyle.innerHTML = "" + "#Hotkey-Settings input[value='" + newKey + "'] {" + " -webkit-transition: background-color 0.5s;" + " transition: background-color 0.5s;" + " background-color: white;" + "} "; window.setTimeout(function() { container.removeChild(alertStyle); },600); }, 800); } } }, true); }); //reset settings button var bottomRow = document.createElement("tr"); var bottomData = document.createElement("td"); var reset = document.createElement("input"); container.appendChild(bottomRow); bottomRow.appendChild(bottomData); reset.setAttribute("type","button"); reset.setAttribute("value","Reset"); bottomData.appendChild(reset); reset.addEventListener("click", function(event) { loadDefaultHotkeys(); var i = 0; hotkeys.forEach(function(func, key) { inputs[i++].value = key; }); }); container.setAttribute("id", "Hotkey-Settings"); document.body.appendChild(container); settingsMenu = container; } else { settingsOpen = false; document.body.removeChild(settingsMenu); } } initiateHotkeys();