Skip to content

Instantly share code, notes, and snippets.

@mscharley
Last active April 9, 2022 11:17
Show Gist options
  • Save mscharley/cb2302fb97454e874db30a6be2f2445c to your computer and use it in GitHub Desktop.
Save mscharley/cb2302fb97454e874db30a6be2f2445c to your computer and use it in GitHub Desktop.
// ==UserScript==
// @name Nuklear Power keyboard advancer
// @namespace http://matt.scharley.me/
// @version 1.0
// @updateURL https://gist.github.com/mscharley/cb2302fb97454e874db30a6be2f2445c/raw/8bit-theater-keyboard.user.js
// @downloadURL https://gist.github.com/mscharley/cb2302fb97454e874db30a6be2f2445c/raw/8bit-theater-keyboard.user.js
// @description Adds keyboard shortcuts for easily navigating the 8-bit Theater webcomic.
// @author Matthew Scharley
// @match *://www.nuklearpower.com/*
// @grant none
// @run-at document-end
// ==/UserScript==
// Click the raw button above to install this script.
(function() {
"use strict";
var backLink = document.querySelectorAll('.navbar-previous a')[0],
nextLink = document.querySelectorAll('.navbar-next a')[0];
document.addEventListener('keyup', function(event) {
switch(event.code) {
case "ArrowRight":
event.preventDefault();
console.log("npka: trying to click", nextLink);
if (nextLink) {
fireEvent(nextLink, 'click');
}
break;
case "ArrowLeft":
event.preventDefault();
console.log("npka: trying to click", backLink);
if (backLink) {
fireEvent(backLink, 'click');
}
break;
}
});
/**
* @param {NodeList} list A nodelist or array of Nodes to search
* @param {String} selector A selector to apply to all elements
*/
function nodeListQuerySelectorAll(list, selector) {
var mappings = [];
for (var i = 0; i < list.length; i++) {
var sublisting = list[i].querySelectorAll(selector);
for (var j = 0; j < sublisting.length; j++) {
mappings.push(sublisting[j]);
}
}
return mappings;
}
/**
* Fire an event handler to the specified node. Event handlers can detect that the event was fired programatically
* by testing for a 'synthetic=true' property on the event object.
*
* @param {HTMLNode} node The node to fire the event handler on.
* @param {String} eventName The name of the event without the "on" (e.g., "focus")
*
* @see http://stackoverflow.com/a/2381862/15537
*/
function fireEvent(node, eventName) {
// Make sure we use the ownerDocument from the provided node to avoid cross-window problems
var doc;
if (node.ownerDocument) {
doc = node.ownerDocument;
} else if (node.nodeType == 9){
// the node may be the document itself, nodeType 9 = DOCUMENT_NODE
doc = node;
} else {
throw new Error("Invalid node passed to fireEvent: " + node.id);
}
if (node.dispatchEvent) {
// Gecko-style approach (now the standard) takes more work
var eventClass = "";
// Different events have different event classes.
// If this switch statement can't map an eventName to an eventClass,
// the event firing is going to fail.
switch (eventName) {
case "click": // Dispatching of 'click' appears to not work correctly in Safari. Use 'mousedown' or 'mouseup' instead.
case "mousedown":
case "mouseup":
eventClass = "MouseEvents";
break;
case "focus":
case "change":
case "blur":
case "select":
eventClass = "HTMLEvents";
break;
default:
throw "fireEvent: Couldn't find an event class for event '" + eventName + "'.";
}
var event = doc.createEvent(eventClass);
var bubbles = eventName == "change" ? false : true;
event.initEvent(eventName, bubbles, true); // All events created as bubbling and cancelable.
event.synthetic = true; // allow detection of synthetic events
// The second parameter says go ahead with the default action
node.dispatchEvent(event, true);
} else if (node.fireEvent) {
// IE-old school style
var event = doc.createEventObject();
event.synthetic = true; // allow detection of synthetic events
node.fireEvent("on" + eventName, event);
}
}
})();
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment