// ==UserScript==
// @name Heello Keyboard Shortcuts
// @namespace http://userstyles.org
// @description jkでタイムラインをスクロール
// @author kawarin
// @match http://heello.com/*
// @match https://heello.com/*
// @version 0.7
// ==/UserScript==
// Navigation
//
// j: Next Ping
// k: Previous Ping
// gg: Scroll to top
// shift+g: Scroll to bottom
//
// Actions
//
// e: Echo
/****************************************************************
* http://www.openjs.com/scripts/events/keyboard_shortcuts/
* Version : 2.01.B
* By Binny V A
* License : BSD
*/
shortcut = {
'all_shortcuts':{},//All the shortcuts are stored in this array
'add': function(shortcut_combination,callback,opt) {
//Provide a set of default options
var default_options = {
'type':'keydown',
'propagate':false,
'disable_in_input':true,
'target':document,
'keycode':false
}
if(!opt) opt = default_options;
else {
for(var dfo in default_options) {
if(typeof opt[dfo] == 'undefined') opt[dfo] = default_options[dfo];
}
}
var ele = opt.target;
if(typeof opt.target == 'string') ele = document.getElementById(opt.target);
var ths = this;
shortcut_combination = shortcut_combination.toLowerCase();
//The function to be called at keypress
var func = function(e) {
e = e || window.event;
if(opt['disable_in_input']) { //Don't enable shortcut keys in Input, Textarea fields
var element;
if(e.target) element=e.target;
else if(e.srcElement) element=e.srcElement;
if(element.nodeType==3) element=element.parentNode;
if(element.tagName == 'INPUT' || element.tagName == 'TEXTAREA') return;
}
//Find Which key is pressed
if (e.keyCode) code = e.keyCode;
else if (e.which) code = e.which;
var character = String.fromCharCode(code).toLowerCase();
if(code == 188) character=","; //If the user presses , when the type is onkeydown
if(code == 190) character="."; //If the user presses , when the type is onkeydown
var keys = shortcut_combination.split("+");
//Key Pressed - counts the number of valid keypresses - if it is same as the number of keys, the shortcut function is invoked
var kp = 0;
//Work around for stupid Shift key bug created by using lowercase - as a result the shift+num combination was broken
var shift_nums = {
"`":"~",
"1":"!",
"2":"@",
"3":"#",
"4":"$",
"5":"%",
"6":"^",
"7":"&",
"8":"*",
"9":"(",
"0":")",
"-":"_",
"=":"+",
";":":",
"'":"\"",
",":"<",
".":">",
"/":"?",
"\\":"|"
}
//Special Keys - and their codes
var special_keys = {
'esc':27,
'escape':27,
'tab':9,
'space':32,
'return':13,
'enter':13,
'backspace':8,
'scrolllock':145,
'scroll_lock':145,
'scroll':145,
'capslock':20,
'caps_lock':20,
'caps':20,
'numlock':144,
'num_lock':144,
'num':144,
'pause':19,
'break':19,
'insert':45,
'home':36,
'delete':46,
'end':35,
'pageup':33,
'page_up':33,
'pu':33,
'pagedown':34,
'page_down':34,
'pd':34,
'left':37,
'up':38,
'right':39,
'down':40,
'f1':112,
'f2':113,
'f3':114,
'f4':115,
'f5':116,
'f6':117,
'f7':118,
'f8':119,
'f9':120,
'f10':121,
'f11':122,
'f12':123
}
var modifiers = {
shift: { wanted:false, pressed:false},
ctrl : { wanted:false, pressed:false},
alt : { wanted:false, pressed:false},
meta : { wanted:false, pressed:false} //Meta is Mac specific
};
if(e.ctrlKey) modifiers.ctrl.pressed = true;
if(e.shiftKey) modifiers.shift.pressed = true;
if(e.altKey) modifiers.alt.pressed = true;
if(e.metaKey) modifiers.meta.pressed = true;
for(var i=0; k=keys[i],i<keys.length; i++) {
//Modifiers
if(k == 'ctrl' || k == 'control') {
kp++;
modifiers.ctrl.wanted = true;
} else if(k == 'shift') {
kp++;
modifiers.shift.wanted = true;
} else if(k == 'alt') {
kp++;
modifiers.alt.wanted = true;
} else if(k == 'meta') {
kp++;
modifiers.meta.wanted = true;
} else if(k.length > 1) { //If it is a special key
if(special_keys[k] == code) kp++;
} else if(opt['keycode']) {
if(opt['keycode'] == code) kp++;
} else { //The special keys did not match
if(character == k) kp++;
else {
if(shift_nums[character] && e.shiftKey) { //Stupid Shift key bug created by using lowercase
character = shift_nums[character];
if(character == k) kp++;
}
}
}
}
if(kp == keys.length &&
modifiers.ctrl.pressed == modifiers.ctrl.wanted &&
modifiers.shift.pressed == modifiers.shift.wanted &&
modifiers.alt.pressed == modifiers.alt.wanted &&
modifiers.meta.pressed == modifiers.meta.wanted) {
callback(e);
if(!opt['propagate']) { //Stop the event
//e.cancelBubble is supported by IE - this will kill the bubbling process.
e.cancelBubble = true;
e.returnValue = false;
//e.stopPropagation works in Firefox.
if (e.stopPropagation) {
e.stopPropagation();
e.preventDefault();
}
return false;
}
}
}
this.all_shortcuts[shortcut_combination] = {
'callback':func,
'target':ele,
'event': opt['type']
};
//Attach the function with the event
if(ele.addEventListener) ele.addEventListener(opt['type'], func, false);
else if(ele.attachEvent) ele.attachEvent('on'+opt['type'], func);
else ele['on'+opt['type']] = func;
},
//Remove the shortcut - just specify the shortcut and I will remove the binding
'remove':function(shortcut_combination) {
shortcut_combination = shortcut_combination.toLowerCase();
var binding = this.all_shortcuts[shortcut_combination];
delete(this.all_shortcuts[shortcut_combination])
if(!binding) return;
var type = binding['event'];
var ele = binding['target'];
var callback = binding['callback'];
if(ele.detachEvent) ele.detachEvent('on'+type, callback);
else if(ele.removeEventListener) ele.removeEventListener(type, callback, false);
else ele['on'+type] = false;
}
};
/**********************************************************/
/* End of "shorcut.js" */
/**********************************************************/
// Add class name
function add_class_name(obj,add_classes){
var tmp_hash = new Array();
var new_class_names = new Array();
var class_names = obj.className.split(" ").concat(add_classes.split(" "));
for(var i in class_names){if(class_names[i] != ""){tmp_hash[class_names[i]] = 0;}}
for(var key in tmp_hash){new_class_names.push(key);}
obj.className = new_class_names.join(" ");
}
// Delete class name
function delete_class_name(obj,delete_classes){
var new_class_names = new Array();
var class_names = obj.className.split(" ");
var delete_class_names = delete_classes.split(" ");
for(var i in class_names){
var flag = true;
for(var j in delete_class_names){
if(class_names[i] == delete_class_names[j]){flag = false;break;};
}
if(flag){new_class_names.push(class_names[i])}
}
obj.className = new_class_names.join(" ");
}
/*
// Send Ping
function shortPing(content) {
var obj = new XMLHttpRequest();
obj.open('POST', "https://heello.com/pings.json", true);
obj.setRequestHeader('X-Requested-With', 'XMLHttpRequest');
obj.setRequestHeader('Content-Type', 'application/x-www-form-urlencoded');
obj.send("text="+encodeURIComponent(content));
}
// Echo
function shortEcho(id) {
var obj = new XMLHttpRequest();
obj.open('POST', "https://heello.com/pings/" + id + "/echo", true);
obj.setRequestHeader('X-Requested-With', 'XMLHttpRequest');
obj.setRequestHeader('Content-Type', 'application/x-www-form-urlencoded');
obj.send('');
}
// Like
function shortLike(id) {
var obj = new XMLHttpRequest();
obj.open('POST', "https://heello.com/pings/" + id + "/like.json", true);
obj.setRequestHeader('X-Requested-With', 'XMLHttpRequest');
obj.setRequestHeader('Content-Type', 'application/x-www-form-urlencoded');
obj.send('');
}
// Button
function clickPing(){
var forms = document.getElementsByTagName('form');
if (forms[0].getElementsByTagName('textarea')[0].value){
document.querySelector('div.modal-buttons input[type="button"].grey-button').click();
}
if (forms[1]) {
if (forms[1].getElementsByTagName('textarea')[0].value){
forms[1].querySelector('input[type="button"].grey-button').click();
}
}
document.activeElement.blur();
}
*/
// Scroll window
function scrollStream(elm) {
var body = document.body;
var html = document.documentElement;
var rect = elm.getBoundingClientRect();
var pos = {
"left": rect.left + (body.scrollLeft || html.scrollLeft),
"top" : rect.top + (body.scrollTop || html.scrollTop) - (html.clientHeight)/2 + 90
}
window.scrollTo(body.scrollLeft || html.scrollLeft, pos.top);
}
// Initialization
var userprofile = document.querySelector('ul#nav-buttons.nav-pills li.dropdown ul.dropdown-menu li a').href;
var username = userprofile.replace('https://heello.com/', '');
// Background-color of selected Ping
var sh = document.styleSheets[0];
sh.insertRule("div.hovered-stream-item {background-color: #eff8ff;}", sh.cssRules.length);
var pnt = null;
var cmf = false;
var elements = document.getElementById('timeline');
//var newPing = document.querySelector('span#nav-compose.black-button img');
var click = function(n) {
var e = document.createEvent('MouseEvents');
e.initMouseEvent("click",true,true,window,0,0,0,0,0,false,false,false,false,0,null);
n.dispatchEvent(e);
};
// Keyboard shortcuts
// Navigation
//
// j: Next Ping
shortcut.add('j', function() {
var hov = elements.querySelector('div.hovered-stream-item');
if (pnt === null) {
add_class_name(elements.querySelector('div[data-id]'), 'hovered-stream-item');
pnt = 0;
} else if (pnt >= 0 && pnt < elements.querySelectorAll('div[data-id]').length-1) {
delete_class_name(hov, 'hovered-stream-item');
add_class_name(elements.querySelectorAll('div[data-id]')[pnt+1], 'hovered-stream-item');
pnt++;
}
var hov2 = elements.querySelector('div.hovered-stream-item');
if (hov2) {
scrollStream(hov2);
}
});
// k: Previous Ping
shortcut.add('k', function() {
var hov = elements.querySelector('div.hovered-stream-item');
if (pnt === 0) {
delete_class_name(hov, 'hovered-stream-item');
pnt = null;
} else if (pnt > 0) {
delete_class_name(hov, 'hovered-stream-item');
add_class_name(elements.querySelectorAll('div[data-id]')[pnt-1], 'hovered-stream-item');
pnt--;
}
var hov2 = elements.querySelector('div.hovered-stream-item');
if (hov2) {
scrollStream(hov2);
}
});
// shift+g: scroll to bottom
shortcut.add('shift+g', function() {
var hov = elements.querySelector('div.hovered-stream-item');
var pings = elements.querySelectorAll('div[data-id]');
if (pnt !== null && pnt < pings.length-1) {
delete_class_name(hov, 'hovered-stream-item');
}
pnt = pings.length-1;
add_class_name(pings[pnt], 'hovered-stream-item');
scrollStream(pings[pnt]);
});
// Actions
//
// e: Echo
shortcut.add('e', function() {
// var id;
var echolink;
var hov = elements.querySelector('div.hovered-stream-item');
if (hov) {
echolink = hov.querySelector('a[href$="echo"]');
// id = hov.getAttribute('data-id');
} else {
// id = elements.querySelector('div[data-id]').getAttribute('data-id');
echolink = elements.querySelector('div[data-id] a[href$="echo"]');
}
// shortEcho(id);
click(echolink);
});
/*
// c: Repeat
shortcut.add('c', function() {
var content;
var hov = elements.querySelector('div.hovered-stream-item');
if (hov) {
content = hov.querySelector('div.ping-text').textContent;
} else {
content = elements.querySelector('div.ping-text').textContent;
}
shortPing(content);
});
*/
// enter: Open Ping details
shortcut.add('enter', function() {
window.location.href = elements.querySelector('div.hovered-stream-item h3 small a').href;
});
/*
// n: New Ping
shortcut.add('n', function() {
click(newPing);
});
// shift + enter: Send Ping
// ctrl + enter : Send Ping
shortcut.add('shift+enter', function() {
clickPing();
}, { 'disable_in_input': false});
shortcut.add('ctrl+enter', function() {
clickPing();
}, { 'disable_in_input': false});
*/
// Timelines
//
// g: Timeline Commands Flag ON
// g g: Scroll to top
shortcut.add('g', function() {
if (cmf) {
cmf = false;
var hov = elements.querySelector('div.hovered-stream-item');
if (hov) {
delete_class_name(hov, 'hovered-stream-item');
}
pnt = null;
window.scrollTo(document.body.scrollLeft || document.documentElement.scrollLeft, 0);
} else {
cmf = true;
}
});
/*
// g h: Home
shortcut.add('h', function() {
if (cmf) {
cmf = false;
window.location.href = '/';
}
});
// r: Reply
// g r: Replies
shortcut.add('r', function() {
if (cmf) {
cmf = false;
window.location.href = '/replies';
} else {
click(elements.querySelector('div.hovered-stream-item span.ping-reply img'));
}
});
// g l: What's Happening?
shortcut.add('l', function() {
if (cmf) {
cmf = false;
window.location.href = '/live';
}
});
// g p: Profile
shortcut.add('p', function() {
if (cmf) {
cmf = false;
window.location.href = '/' + username;
}
});
// g s: Settings
shortcut.add('s', function() {
if (cmf) {
cmf = false;
window.location.href = '/account/settings';
}
});
// f: Like
// g f: Likes
shortcut.add('f', function() {
if (cmf) {
cmf = false;
window.location.href = '/' + username + '/likes';
} else {
var hov = elements.querySelector('div.hovered-stream-item');
if (hov) {
id = hov.getAttribute('data-id');
shortLike(id);
}
}
});
//
// esc: Close modal windows
shortcut.add('esc', function() {
cmf = false;
var closeButtons = document.querySelectorAll("div.modal-title div.modal-title-close");
for (var i=0; i<closeButtons.length; i++) {
click(closeButtons[i]);
}
document.activeElement.blur();
}, {'disable_in_input': false});
*/