// ==UserScript== // @name osu! PP Save // @version 1.2 // @description A Simple script to save users pp to compare later. // @author Rafael Moreira Fonseca // @match https://osu.ppy.sh/* // @icon http://osu.ppy.sh/favicon.ico // @namespace https://twitter.com/RafaelMoreiraFo // @grant none // @downloadURL none // ==/UserScript== /*jslint browser: true*/ /*jslint jquery: true*/ /* * jQuery Hotkeys Plugin * Copyright 2010, John Resig * Dual licensed under the MIT or GPL Version 2 licenses. * * Based upon the plugin by Tzury Bar Yochay: * https://github.com/tzuryby/jquery.hotkeys * * Original idea by: * Binny V A, http://www.openjs.com/scripts/events/keyboard_shortcuts/ */ /* * One small change is: now keys are passed by object { keys: '...' } * Might be useful, when you want to pass some other data to your handler */ (function(jQuery) { jQuery.hotkeys = { version: "0.2.0", specialKeys: { 8: "backspace", 9: "tab", 10: "return", 13: "return", 16: "shift", 17: "ctrl", 18: "alt", 19: "pause", 20: "capslock", 27: "esc", 32: "space", 33: "pageup", 34: "pagedown", 35: "end", 36: "home", 37: "left", 38: "up", 39: "right", 40: "down", 45: "insert", 46: "del", 59: ";", 61: "=", 96: "0", 97: "1", 98: "2", 99: "3", 100: "4", 101: "5", 102: "6", 103: "7", 104: "8", 105: "9", 106: "*", 107: "+", 109: "-", 110: ".", 111: "/", 112: "f1", 113: "f2", 114: "f3", 115: "f4", 116: "f5", 117: "f6", 118: "f7", 119: "f8", 120: "f9", 121: "f10", 122: "f11", 123: "f12", 144: "numlock", 145: "scroll", 173: "-", 186: ";", 187: "=", 188: ",", 189: "-", 190: ".", 191: "/", 192: "`", 219: "[", 220: "\\", 221: "]", 222: "'" }, shiftNums: { "`": "~", "1": "!", "2": "@", "3": "#", "4": "$", "5": "%", "6": "^", "7": "&", "8": "*", "9": "(", "0": ")", "-": "_", "=": "+", ";": ": ", "'": "\"", ",": "<", ".": ">", "/": "?", "\\": "|" }, // excludes: button, checkbox, file, hidden, image, password, radio, reset, search, submit, url textAcceptingInputTypes: [ "text", "password", "number", "email", "url", "range", "date", "month", "week", "time", "datetime", "datetime-local", "search", "color", "tel"], // default input types not to bind to unless bound directly textInputTypes: /textarea|input|select/i, options: { filterInputAcceptingElements: true, filterTextInputs: true, filterContentEditable: true } }; function keyHandler(handleObj) { if (typeof handleObj.data === "string") { handleObj.data = { keys: handleObj.data }; } // Only care when a possible input has been specified if (!handleObj.data || !handleObj.data.keys || typeof handleObj.data.keys !== "string") { return; } var origHandler = handleObj.handler, keys = handleObj.data.keys.toLowerCase().split(" "); handleObj.handler = function(event) { // Don't fire in text-accepting inputs that we didn't directly bind to if (this !== event.target && (jQuery.hotkeys.options.filterInputAcceptingElements && jQuery.hotkeys.textInputTypes.test(event.target.nodeName) || (jQuery.hotkeys.options.filterContentEditable && jQuery(event.target).attr('contenteditable')) || (jQuery.hotkeys.options.filterTextInputs && jQuery.inArray(event.target.type, jQuery.hotkeys.textAcceptingInputTypes) > -1))) { return; } var special = event.type !== "keypress" && jQuery.hotkeys.specialKeys[event.which], character = String.fromCharCode(event.which).toLowerCase(), modif = "", possible = {}; jQuery.each(["alt", "ctrl", "shift"], function(index, specialKey) { if (event[specialKey + 'Key'] && special !== specialKey) { modif += specialKey + '+'; } }); // metaKey is triggered off ctrlKey erronously if (event.metaKey && !event.ctrlKey && special !== "meta") { modif += "meta+"; } if (event.metaKey && special !== "meta" && modif.indexOf("alt+ctrl+shift+") > -1) { modif = modif.replace("alt+ctrl+shift+", "hyper+"); } if (special) { possible[modif + special] = true; } else { possible[modif + character] = true; possible[modif + jQuery.hotkeys.shiftNums[character]] = true; // "$" can be triggered as "Shift+4" or "Shift+$" or just "$" if (modif === "shift+") { possible[jQuery.hotkeys.shiftNums[character]] = true; } } for (var i = 0, l = keys.length; i < l; i++) { if (possible[keys[i]]) { return origHandler.apply(this, arguments); } } }; } jQuery.each(["keydown", "keyup", "keypress"], function() { jQuery.event.special[this] = { add: keyHandler }; }); })(jQuery || this.jQuery || window.jQuery); // Script $(window).ready(function() { var pathname = $.trim(window.location.pathname); // Images var configImage = ""; var configImageMini = ""; // Create styles var styles = ".ppButton{background:rgba(0,0,0,0.07);border-radius:5px;border:1px solid rgba(0,0,0,0.07);padding: 1px 14px;font-size:13px;float:right;cursor:pointer;}"; styles += ".deleteppButton{background: #ffe2fa;border-radius:5px;border: solid 1px #ffbff1;padding: 1px 14px;font-size:13px;float:right;cursor:pointer;}"; styles += ".oldPP{font-size:80%;color:green;float:right;margin:2px 10px;}"; styles += ".alertPP{font-size:80%;color:red;float:right;margin:2px 5px;}"; styles += ".morePP{color: #3843a6;}"; styles += ".arrow{font-family: Verdana,sans-serif;}"; styles += ".saveAllPP{background:rgba(0,0,0,0.07);margin:3px;border-radius:5px;border:1px solid rgba(0,0,0,0.07);padding: 1px 14px;font-size:13px;float:right;cursor:pointer;}"; styles += ".oldPPTable{color:green;margin-left:10px;float:right;}"; styles += ".oldPPTableUpdated{color: #3843a6;margin-left:10px;float:right;}"; styles += ".noPPTable{color:black;margin-left:10px;float:right;}"; styles += ".alertReload{font-size:14px;color:red;float:right;margin:4px 5px;}"; styles += ".updateAllButton{margin-top: 3px;}"; styles += ".configIcon{background: rgba(0, 0, 0, 0.2) url('"+configImage+"') no-repeat 5px 10px;border:1px solid rgba(0, 0, 0, 0.3);border-top:0px;position:fixed;width:32px;height:32px;padding: 10px 5px 5px 5px;opacity:0.5;margin-right:15px;top:0;right:0;cursor:pointer;}"; styles += ".configIcon:hover{opacity:1;}"; styles += ".configIconMini{background: rgba(0, 0, 0, 0.2) url('"+configImageMini+"') no-repeat 5px 10px;border:1px solid rgba(0, 0, 0, 0.3);border-top:0px;position:fixed;width:16px;height:16px;padding: 10px 5px 5px 5px;opacity:0.5;margin-right:15px;top:0;right:0;cursor:pointer;}"; styles += ".configIconMini:hover{opacity:1;}"; styles += ".nanoModal {width:580px;position: absolute;top: 0px;display: none;z-index: 10000;padding: 15px 20px 10px;-webkit-border-radius: 10px;-moz-border-radius: 10px;border-radius: 10px;background: #fff;}"; styles += ".nanoModalOverlay {position: fixed;top: 0;left: 0;bottom:0;right:0;width: 100%;height: 100%;z-index: 9999;background: #000;display: none;-ms-filter: 'alpha(Opacity=50)';-moz-opacity: .5;-khtml-opacity: .5;opacity: .5;}"; styles += ".nanoModalButtons {margin-top: 15px;text-align: right;position: absolute;right: 10px;top: -10px;}"; styles += ".nanoModalButtons .nanoModalBtnPrimary {background-color: #efefef;color: #333;border: 1px solid silver;cursor: pointer;display: inline-block;font-size: 14px;margin: 8px 4px 0;padding: 6px 12px;text-align: center;vertical-align: middle;white-space: nowrap;-webkit-user-select: none;-moz-user-select: none;-ms-user-select: none;user-select: none;-khtml-border-radius: 4px;border-radius: 4px;}"; styles += ".nanoModalButtons .nanoModalBtnPrimary:hover {background-color:#e2e2e2;}"; styles += ".copyrights{text-align:right;display:block;padding:5px 0px;font-size:11px;color: #777;font-weight:normal;}"; styles += "#nanoModal .btnmain {cursor:pointer;padding:5px 10px;}"; // Some JSON useful functions Storage.prototype.getObject = function(key) { var value = this.getItem(key); return value && JSON.parse(value); } Storage.prototype.setObject = function(key, userName, gameMode, userValue) { var getValue = localStorage.getObject('usersPP'); if(getValue == null){ var getValue = {}; getValue[userName] = {"gm-0":"", "gm-1":"", "gm-2":"", "gm-3":""}; }else{ if(getValue[userName] == null){ getValue[userName] = {"gm-0":"", "gm-1":"", "gm-2":"", "gm-3":""}; } } getValue[userName][gameMode] = userValue; this.setItem(key, JSON.stringify(getValue)); } //Another Useful Functions $.urlParam = function(name){ var results = new RegExp('[\?&]' + name + '=([^]*)').exec(window.location.href); if (results==null){ return null; } else{ return results[1] || 0; } } //Osu! website useful functions function saveCurrentGameMode(userPP){ var userName = $.trim($(".profile-username").html()); var userCurrentMode = $(".profileGameModes").find(".active").attr("id"); localStorage.setObject('usersPP', userName, userCurrentMode, userPP); $("#saveUserPP").css("border","1px solid green"); if(getConfigValue("reloadPage") == "yes"){ location.reload(); }else{ showReloadAlert(); } } function deleteCurrentUser(userName){ var getValue = localStorage.getObject('usersPP'); delete getValue[userName]; localStorage.setItem('usersPP', JSON.stringify(getValue)); $("#deleteUserPP").css("border","1px solid red"); showReloadAlert(); } function showReloadAlert(){ var oldPP = $('.oldPP'); if(oldPP.length > 0){ oldPP.addClass("alertPP"); oldPP.html("Reload the page."); }else{ $(".profileStatLine").first().append("Reload the page."); } } function showOldPP(){ var userName = $.trim($(".profile-username").html()); var userCurrentMode = $(".profileGameModes").find(".active").attr("id"); var getValue = localStorage.getObject('usersPP'); if(getValue != null){ var oldPP = getValue[userName][userCurrentMode]; $(".profileStatLine").first().append(""+oldPP+""); } } function countProgress(currentPP, currentPerformance){ var userName = $.trim($(".profile-username").html()); var userCurrentMode = $(".profileGameModes").find(".active").attr("id"); var getValue = localStorage.getObject('usersPP'); if(getValue != null){ var oldPP = getValue[userName][userCurrentMode]; if(oldPP != ""){ var justOldPP = parsePP(oldPP); var justNewPP = parsePP(currentPP); var countPP = justNewPP - justOldPP; var splitPP = currentPP.split("("); var arrow = "▲"; if(countPP < 0){ arrow = "▼"; } var newCurrentPP = ": " +splitPP[0] + " (" + arrow + "" + countPP + "pp) " + splitPP[1].replace(")",""); $(".profileStatLine b").first().html(currentPerformance+newCurrentPP); } } } function parsePP(numberPP){ return parseInt($.trim($.trim(numberPP.split("(")[0]).replace("pp","").replace(",",""))); } // Some Config useful functions function Initialize(){ var getConfig = localStorage.getObject('config'); if(getConfig == null){ var config = { "changeTableColor" : "yes", "showConfigButton" : "yes", "reloadPage" : "no", "autoLeaveSave" : "no", "oddcolor" : "#A9A9FF", "evencolor" : "#b2b2ff", "hovercolor" : "#F0ECFA", "iconSize" : "32px", "configKey" : "shift+y" }; localStorage.setItem('config', JSON.stringify(config)); } } Initialize(); function setConfigValue(configName, configValue){ var getConfig = localStorage.getObject('config'); getConfig[configName] = configValue; localStorage.setItem('config', JSON.stringify(getConfig)); } function getConfigValue(configName){ var getConfig = localStorage.getObject('config'); return getConfig[configName]; } // Table styles styles += ".oddTable{background-color:"+getConfigValue("oddcolor")+";}"; styles += ".evenTable{background-color:"+getConfigValue("evencolor")+";}"; styles += ".oddTable:hover, .evenTable:hover{background:"+getConfigValue("hovercolor")+"!important;}"; $('').appendTo($('head')); // Configuration if(getConfigValue("showConfigButton") == "yes"){ if(getConfigValue("iconSize") == "32px"){ $("body").prepend('
'); }else if(getConfigValue("iconSize") == "16px"){ $("body").prepend(''); } } var windowWidth = $(window).width(); var modalLeft = (windowWidth / 2) - 290; $("body").append(''); $("body").append(''); $("#nanoModal").prepend(''); // Fields types var FieldTypes = { Text : 1, YesNo : 2, Select : 3, Textarea: 4 }; // Configuration fields function addConfig(fieldType, fieldText, fieldComment, fieldName, fieldValue){ var fieldContent; switch(fieldType){ case FieldTypes.Text: fieldContent = '