NOTICE: By continued use of this site you understand and agree to the binding Terms of Service and Privacy Policy.
// ==UserScript== // @name Steemit-Sidebar dev // @namespace http://tampermonkey.net/ // @copyright 2018, mwfiae (https://steemit.com/@mwfiae) // @version 0.6.1 // @description try to take over the world! // @author MWFIAE // @match http*://steemit.com/* // @match http*://steemw.ga/* // @license MIT // @grant GM_getValue // @grant GM_setValue // @grant GM_getResourceText // @grant unsafeWindow // @require http://code.jquery.com/jquery-1.12.4.min.js // @require https://cdn.steemjs.com/lib/latest/steem.min.js // @require https://momentjs.com/downloads/moment-with-locales.min.js // @require https://github.com/inuyaksa/jquery.nicescroll/raw/master/jquery.nicescroll.min.js // @require https://code.jquery.com/ui/1.12.1/jquery-ui.min.js // @resource JQUI_CSS https://code.jquery.com/ui/1.12.1/themes/base/jquery-ui.css // @updateURL https://openuserjs.org/meta/mwfiae/Steemit-Sidebar.meta.js // @noframes // ==/UserScript== /* jshint -W097 */ 'use strict'; // At this point I just want to say 'thank you!' to @therealwolf // without his help and example coding I wouldn't have 'finished' the project this fast // Handle multiple languages (only english for now) const I18N = { "en":{ "username": "Username", "vp": "Voting Power", "full": "Full", "bandwidth": "Bandwidth", "sp": "SteemPower", "settings_title": "Steemit Sidebar Settings", "settings_tab1": "Appearance", "settings_tab2": "Links", "settings_barcolor_low": "Bar Color Low", "settings_barcolor_high": "Bar Color High", "settings_warn_refresh": "The following settings require a site-refresh (F5).", "settings_sidebar_side": "Sidebar-Side", "settings_sidebar_side_left": "left", "settings_sidebar_side_right": "right", "settings_language": "Language", "settings_language_en": "English", "settings_restore_links": "Restore Links", "settings_links_icon": "Icon", "settings_links_url": "Url", "settings_links_text": "Text", }, } // This is the html template for the main area of the sidebar, without the user templates(TEMPLATE_WITH_USER) rendered const TEMPLATE_WITHOUT_USER = ` <style> .ui-tabs{ height:100%; } .ui-tabs-vertical .ui-tabs-nav { padding: .2em .1em .2em .2em; float: left; width: 20%; } .ui-tabs-vertical .ui-tabs-nav li { clear: left; width: 100%; border-bottom-width: 1px !important; border-right-width: 0 !important; margin: 0 -1px .2em 0; } .ui-tabs-vertical .ui-tabs-nav li a { display:inline-block;width: 100% } .ui-tabs-vertical .ui-tabs-nav li.ui-tabs-active { padding-bottom: 0; padding-right: .1em; border-right-width: 1px; } .ui-tabs-vertical .ui-tabs-panel { padding: 1em; float: left; width: 80%; } .ui-button .ui-icon.ui-icon-closethick { background: url('') !important; } #mw-script-container{ position: fixed; float: left; padding-top: 2.5rem; padding-left: 2em; padding-right: 2em; height:100vh; overflow-y: auto; z-index: 5; width: 200px; display: grid; grid-template-rows: 1fr 80px; grid-template-columns: 1fr; } .mw-right#mw-script-container{ right: 0; } #mw-main{ grid-row-start:1; grid-row-end:1; } #mw-footer{ grid-row-start:2; grid-row-end:2; } #mw-collapse-button{ float: left; margin-left: 110px; z-index: 100; position: fixed; cursor: pointer; font-weight: bolder; margin-top: -16px; } #mw-settings{ float: left; z-index: 100; cursor: pointer; position: relative; top: -15px; left: -28px; } .theme-dark #mw-script-container, .theme-dark #mw-button{ background-color: #1C252B; } .theme-light #mw-script-container, .theme-light #mw-button{ background-color: #fcfcfc; } #username{display: inline;} .mw-favicon{width:16px; height: 16px} .mw-ul{list-style-type:none;} .mw-button{background-color:transparent; border-width: 1px;} .mw-nowrap{} /* Tooltip container */ .mw-tooltip { position: relative; display: inline-block; } /* Tooltip text */ .mw-tooltip .mw-tooltiptext { visibility: hidden; width: 120px; background-color: #555; color: #fff; text-align: center; padding: 5px 0; border-radius: 6px; /* Position the tooltip text */ position: absolute; z-index: 1; bottom: 125%; left: 50%; margin-left: -60px; /* Fade in tooltip */ opacity: 0; transition: opacity 0.3s; } /* Show the tooltip text when you mouse over the tooltip container */ .mw-tooltip:hover .mw-tooltiptext { visibility: visible; opacity: 1; width: 100%; } </style> <div id="mw-script-container" class="mw-{{settings.side}}"> <div id="mw-main"> <img id="mw-settings" width=32px height=32px src="" ></img> <span id="mw-collapse-button">⟨</span> <p id="mw-username-p"><input id="mw-username" type="text" value="{{settings.username}}" placeholder="{{i18n.username}}"/></p> <div id="mw-script-content"></div> <hr id="mw-divider" /> <div id="mw-script-content-other"></div> </div> <div id="mw-footer"> © by <a href="https://steemit.com/@mwfiae">MWFIAE</a> </div> </div> `; // This is the template for the users that will be rendered inside the main template (TEMPLATE_WITHOUT_USER) const TEMPLATE_WITH_USER = ` <div id="{{temp.target}}"> <p> <span><a href="https://steemit.com/@{{user.name}}">{{user.name}}</a> ({{user.displayRep}})</span> </p> <span>{{i18n.vp}}</span> <div id="mw-votepower-{{temp.target}}" class="mw-tooltip" style="width:100%;background-color: lightgrey;"> <style> #mw-votepower-bar-{{temp.target}} { width: {{user.trueVotePower}}%; height: 23px; background-color: {{temp.vote_color}}; text-align: center; /* To center it horizontally (if you want) */ line-height: 30px; /* To center it vertically */ } #mw-votepower-bar-text-{{temp.target}}{ color: white; font-size: 0.8em; text-align: center; /* To center it horizontally (if you want) */ float: left; width: 100%; } </style> <span id="mw-votepower-bar-text-{{temp.target}}">{{user.trueVotePower}}%</span> <div id="mw-votepower-bar-{{temp.target}}"></div> <div class="mw-tooltiptext">{{i18n.full}}:<br />{{user.voteSpan}}<br />{{user.voteTime}}</div> </div> <span>{{i18n.bandwidth}} <span style="font-size: 0.8em">{{temp.bw_p}}%</span></span> <div id="mw-bandwidth-{{temp.target}}" style="width:100%;background-color: lightgrey;"> <style> #mw-bandwidth-bar-{{temp.target}} { width: {{temp.bw_pno0}}%; height: 23px; background-color: {{temp.bw_color}}; line-height: 30px; /* To center it vertically */ } #mw-bandwidth-bar-text-{{temp.target}}{ color: white; font-size: 0.8em; text-align: center; /* To center it horizontally (if you want) */ float: left; width: 100%; } </style> <span id="mw-bandwidth-bar-text-{{temp.target}}" class="mw-nowrap" nowrap>{{temp.bw_c}} \/ {{temp.bw_m}}</span> <div id="mw-bandwidth-bar-{{temp.target}}"></div> </div> <div class="mw-tooltip"> {{i18n.sp}}: <a href="https://steemit.com/@{{user.name}}/transfers">{{user.sp}}</a> <div class="mw-tooltiptext">Own: {{user.ownSp}}<br>Received: {{user.recSp}}<br>Delegated: {{user.delSp}}<br></div> </div> </p> <ul class="mw-ul"> {{temp.links}} </ul> </div> `; // This is the template for the settings menu const TEMPLATE_SETTINGS_MENU = ` <div id="mw-settings-window" title="{{i18n.settings_title}}"> <div id="tabs"> <ul> <li><a href="#tabs-1">{{i18n.settings_tab1}}</a></li> <li><a href="#tabs-2">{{i18n.settings_tab2}}</a></li> </ul> <div id="tabs-1"> <table> <tr> <td>{{i18n.settings_barcolor_low}}</td> <td><input id="mw-barColorLow" style="min-width:50px" class="mw-inline" type="color" value="{{settings.barColorLow}}"/></td> <td>{{i18n.settings_barcolor_high}}</td> <td><input id="mw-barColorHigh" style="min-width:50px" class="mw-inline" type="color" value="{{settings.barColorHigh}}"/></td> </tr> <tr> <td colspan=4 style="color:red">{{i18n.settings_warn_refresh}}</td> </tr> <tr> <td>{{i18n.settings_sidebar_side}}</td> <td> <select id="mw-sidebarSide" style="min-width:50px" class="mw-inline" value="{{settings.barColorLow}}"> <option value="left">{{i18n.settings_sidebar_side_left}}</option> <option value="right">{{i18n.settings_sidebar_side_right}}</option> </select> </td> <td>{{i18n.settings_language}}</td> <td> <select id="mw-language" style="min-width:50px" class="mw-inline" value="{{settings.language}}"> <option value="en">{{i18n.settings_language_en}}</option> </select> </td> </tr> </table> </div> <div id="tabs-2" style="overflow-y: auto"> <input class="mw-button" type="button" id="mw-restore-links" value="{{i18n.settings_restore_links}}"/> <table id="mw-link-table"> <tr><th>{{i18n.settings_links_icon}}</th><th>{{i18n.settings_links_url}}</th><th>{{i18n.settings_links_text}}</th></tr> <tr> <td><input id="mw-new-link-icon" type="text" value="" /></td> <td><input id="mw-new-link-url" type="text" value="" /></td> <td><input id="mw-new-link-text" type="text" value="" /></td> <td><input class="mw-button mw-button-add-link" style="background-color:lightgreen" type="button" value="+" /></td> </tr> {{links}} </table> </div> </div> </div> `; // This is the template how a single link should be rendered in the main view const TEMPLATE_LINK = `<li><a href="{{link.url}}" target="_blank"><img class="mw-favicon" src="{{link.icon}}" />{{link.text}}</a></li>`; // This is the template how a single link should be rendered in the settings view (editable) const TEMPLATE_LINK_SETTINGS = `<tr><td><img class="mw-favicon" src="{{link.icon}}" /></td><td>{{link.url}}</td><td>{{link.text}}</td><td><input class="mw-button mw-button-delete-link" style="background-color:pink" type="button" value="x" /></td></tr>`; // various default links that are included for the convencience of the users const DEFAULT_LINKS=[ {url: "https://steemd.com/@{{user.name}}", icon: "https://steemd.com/favicon-steem9.png", text: "Steemd"}, {url: "https://steemworld.org/@{{user.name}}", icon: "https://steemworld.org/favicon.png", text: "Steemworld"}, {url: "https://steemnow.com/@{{user.name}}", icon: "https://steemnow.com/favicon.ico", text: "Steemnow"}, {url: "https://zappl.com/@{{user.name}}", icon: "https://zappl.com/1/images/favicon.png", text: "Zappl"}, {url: "https://d.tube/#!/c/{{user.name}}", icon: "https://d.tube/DTube_files/images/dtubefavicon.png", text: "D.Tube"}, {url: "https://dmania.lol/profile/{{user.name}}", icon: "https://dmania.lol/favicon.ico", text: "D.Mania"}, {url: "https://alpha.steepshot.io/@{{user.name}}", icon: "https://alpha.steepshot.io/static/images/faviconNew.ico", text: "Steepshot"}, ]; // use the steemit api node steem.api.setOptions({ url: 'https://api.steemit.com' }); // store the Private Posting Key in the userscript section, so other script can't as easily access it. // NOTE: even without steemit sidebar it's already very easy for other script to do the same and read the private key. // so I don't consider this as an vulnerability that the steemit sidebar opens, malicious scripts do have easier // and more guaranteed ways to steal the key var privPostKey= null; console.log("Starting...."); //Initialize the main object; let MWSidebar ={ otherUsername: null, // the name of the "second" user settingsMenu: null, // a reference to the settings menu globalProps:{ // the globalProps are filled by an api call totalVestingFund: 0, totalVestingShares: 0, maxVirtualBandwidth: 0, }, settings: { // All the settings that are currently saved for the user and can be edited via the settings menu username: null, // The username entered dateTimeFormat: 'DD.MM. HH:mm:ss', collapsed: false, // If the sidebar should be collapsed/minimized barColorLow: "#FF0000", // The color of the various bars if they get low barColorHigh: "#00FF00", // The color of the various bars if they get high links: DEFAULT_LINKS, // The links that will be rendered side: "left", // The side of the sidebar lastSaved: 0, // The moment the settings were saved, is used by other tabs to auto update }, /** * Some helper variables/functions */ helper:{ //Some helper variables/functions // The regex to find a mustache style variable (used by my rendering functions) regexVariable: new RegExp("{{([^}]*)}}", "gim"), /** * Linear interpolates between colors, used by the progress bars. * @param {string} startColor - The color to lerp from. * @param {string} endColor - The color to lerp to. * @param {number} amount - The amount of lerp inbetween (0 - 1). * @return {string} A color that lies between @startColor and @endColor. */ lerpColor: function(startColor, endColor, amount) { var ah = parseInt(startColor.replace(/#/g, ''), 16), ar = ah >> 16, ag = ah >> 8 & 0xff, ab = ah & 0xff, bh = parseInt(endColor.replace(/#/g, ''), 16), br = bh >> 16, bg = bh >> 8 & 0xff, bb = bh & 0xff, rr = ar + amount * (br - ar), rg = ag + amount * (bg - ag), rb = ab + amount * (bb - ab); return '#' + ((1 << 24) + (rr << 16) + (rg << 8) + rb | 0).toString(16).slice(1); }, /** * Make the amount of bytes readable by humans. * @param {number} bytes - The amount of bytes. * @return {string} A string representing a human readable amount of bytes */ prettyPrintBytes: function(bytes){ if(Math.abs(bytes)>1000*1000*1000*1000) return (bytes/(1000*1000*1000*1000)).toFixed(2)+" TB"; if(Math.abs(bytes)>1000*1000*1000) return (bytes/(1000*1000*1000)).toFixed(2)+" GB"; if(Math.abs(bytes)>1000*1000) return (bytes/(1000*1000)).toFixed(2)+" MB"; if(Math.abs(bytes)>1000) return (bytes/1000).toFixed(2)+" KB"; return bytes+" B" }, /** * Make a human readable timespan out of seconds. * @param {number} seconds - The amount of seconds. * @return {string} A string representing a human readable timespan. */ formatSeconds: function(seconds){ let timeSpan = ""; if(seconds>24*60*60){ let days=parseInt(seconds/(24*60*60)); seconds -= days*24*60*60; timeSpan = timeSpan + days+"d "; } if(seconds>60*60){ let hours=parseInt(seconds/(60*60)); seconds -= hours*60*60; timeSpan = timeSpan + hours+"h "; } if(seconds>60){ let minutes=parseInt(seconds/60); seconds -= minutes*60; timeSpan = timeSpan + minutes+"m "; } return timeSpan; }, /** * Appends custom css globally. * @param {string} css - The css to append. * @param {string} id - The id of the generated style block */ addGlobalStyle: function(css, id) { $("head:first").append($('<style' + (id !== undefined ? ' id="' + id + '"' : '') + ' type="text/css">' + css + '</style>')); }, /** * Replaces the variables in a template with a maximum depth of 5. * @param {string} template - The template which should be processed. * @param {object} values - An object holding all the values. * @param {number} depth - The current depth we are at. * @return {string} The processed template. */ fillTemplate: function(template, values, depth){ if(depth==undefined) depth = 1; if(depth > 5) return template; let result = template; let regex = new RegExp(MWSidebar.helper.regexVariable); let matches=null; while(matches = regex.exec(template)){ let match = matches[1]; try{ let object = values; match.split(".").forEach(function(single){ object = object[single]; }); if((typeof object)=="string" && object.match(new RegExp(MWSidebar.helper.regexVariable))){ object = MWSidebar.helper.fillTemplate(object, values, depth+1); } result = result.replace(matches[0], object); } catch(e){ console.log(e); } }; return result; }, /** * Converts a hex string into an ascii string. * @param {string} hexx - The hex string to process. * @return {string} The resulting ascii string. */ hex2ascii: function(hexx){ var hex = hexx.toString();//force conversion var str = ''; for (var i = 0; i < hex.length; i += 2) str += String.fromCharCode(parseInt(hex.substr(i, 2), 16)); return str; }, }, /** * Functions that are called by the ui */ ui: { /** * Toggle the collapse state of the Sidebar, saves the settings and rerenders the sidebar (so the state change takes effect) */ toggleCollapse: function(){ MWSidebar.settings.collapsed = ! MWSidebar.settings.collapsed; MWSidebar.saveSettings(); MWSidebar.refreshCollapse(); }, /** * Opens the settings menu and wires the events */ openSettings: function (){ console.log("opening Settings..."); MWSidebar.settingsMenu.dialog("open"); $( "#tabs" ).tabs().addClass( "ui-tabs-vertical ui-helper-clearfix" ); $( "#tabs li" ).removeClass( "ui-corner-top" ).addClass( "ui-corner-left" ); jQuery('#mw-barColorHigh').change(MWSidebar.ui.changeBarColorHigh); jQuery('#mw-barColorLow').change(MWSidebar.ui.changeBarColorLow); $("#mw-sidebarSide").val(MWSidebar.settings.side) //initialize with the current value. jQuery('#mw-sidebarSide').change(MWSidebar.ui.changeSidebarSide); $("#mw-language").val(MWSidebar.settings.language) //initialize with the current value. jQuery('#mw-language').change(MWSidebar.ui.changeLanguage); jQuery("#mw-restore-links").click(MWSidebar.ui.restoreLinks); jQuery(".mw-button-delete-link").click(MWSidebar.ui.deleteLink); jQuery(".mw-button-add-link").click(MWSidebar.ui.addLink); /*$('.ui-dialog').css({ 'width': $(window).width(), 'height': $(window).height(), 'left': '0px', 'top':'0px' });*/ }, /** * Changes the side where the sidebar is rendered */ changeSidebarSide: function(){ MWSidebar.settings.side = $("#mw-sidebarSide").val(); }, /** * Change the users language. */ changeLanguage: function(){ MWSidebar.settings.language = $("#mw-language").val(); MWSidebar.update(); MWSidebar.updateOther(); }, /** * Change the "low" color of the progress bars. */ changeBarColorLow: function(){ MWSidebar.settings.barColorLow= $("#mw-barColorLow").val(); MWSidebar.update(); MWSidebar.updateOther(); }, /** * Change the "high" color of the progress bars. */ changeBarColorHigh: function(){ MWSidebar.settings.barColorHigh= $("#mw-barColorHigh").val(); MWSidebar.update(); MWSidebar.updateOther(); }, /** * Remove all links from the dom link table */ removeAllLinks: function(){ jQuery("#mw-link-table tr:has(.mw-button-delete-link)").remove(); }, /** * (Re-)Adds all the links to the dom link table */ addAllLinks: function(){ jQuery("#mw-link-table").append(MWSidebar.settings.links.reduce((result,link)=>{ return result + MWSidebar.helper.fillTemplate(TEMPLATE_LINK_SETTINGS, {link: link, i18n: I18N[MWSidebar.settings.language]}, 5); },"")); jQuery(".mw-button-delete-link").click(MWSidebar.ui.deleteLink); MWSidebar.doUpdate(); MWSidebar.doUpdateOther(); }, /** * Add a link to the saved links and rerenders everything */ addLink: function(){ let icon = jQuery("#mw-new-link-icon").val(); let url = jQuery("#mw-new-link-url").val(); let text = jQuery("#mw-new-link-text").val(); if(url==""){ alert("You have to enter an url!"); return; } if(icon==""&&text==""){ alert("You have to enter an icon or text!"); return; } jQuery("#mw-new-link-icon").val(""); jQuery("#mw-new-link-url").val(""); jQuery("#mw-new-link-text").val(""); let link = {icon: icon, url: url, text: text}; MWSidebar.settings.links.push(link); MWSidebar.saveSettings(); MWSidebar.ui.removeAllLinks() MWSidebar.ui.addAllLinks(); }, /** * Permanently deletes a link from the settings and dom */ deleteLink: function(e){ var i= jQuery(".mw-button-delete-link").index(e.target); MWSidebar.settings.links.splice(i, 1); MWSidebar.saveSettings(); MWSidebar.ui.removeAllLinks() MWSidebar.ui.addAllLinks(); }, /** * (Re-)Adds all the links to the dom link table */ restoreLinks: function(){ MWSidebar.settings.links = DEFAULT_LINKS.slice(0); MWSidebar.saveSettings(); MWSidebar.ui.removeAllLinks() MWSidebar.ui.addAllLinks(); } }, /** * Calculates the bandwidth for a user * @param {object} data - The user data. * @return {array} An array containing the new bandwidth, the allocated bandwidth and the remaining bandwidth (in percent). */ calcBandwidth: function(data){ const STEEMIT_BANDWIDTH_AVERAGE_WINDOW_SECONDS = 60 * 60 * 24 * 7; let vestingShares = parseFloat(data.vesting_shares) let receivedVestingShares = parseFloat(data.received_vesting_shares) let average_bandwidth = parseInt(data.average_bandwidth, 10) let delta_time = (new Date - new Date(data.last_bandwidth_update + "Z")) / 1000 let bandwidthAllocated = (MWSidebar.globalProps.maxVirtualBandwidth * (vestingShares + receivedVestingShares) / MWSidebar.globalProps.totalVestingShares) bandwidthAllocated = Math.round(bandwidthAllocated / 1000000); let new_bandwidth = 0 if (delta_time < STEEMIT_BANDWIDTH_AVERAGE_WINDOW_SECONDS) { new_bandwidth = (((STEEMIT_BANDWIDTH_AVERAGE_WINDOW_SECONDS - delta_time)*average_bandwidth)/STEEMIT_BANDWIDTH_AVERAGE_WINDOW_SECONDS) } new_bandwidth = Math.round(new_bandwidth / 1000000) let remaining = 100 - (100 * new_bandwidth / bandwidthAllocated); return [new_bandwidth, bandwidthAllocated, remaining]; }, /** * Processes and formats the raw user data from the api. * @param {object} newData - the raw user data from the api. * @return {object} The processed and formatted data. */ updateUser: function(newData) { if (newData == undefined) { user = null; return; } newData.displayRep = steem.formatter.reputation(newData.reputation); let base_voting_power = newData.voting_power; let last_time = moment.utc(newData.last_vote_time).valueOf(); let now = moment.utc().valueOf(); let delta = (now-last_time) /1000; let updated_voting_power = base_voting_power +(10000*delta/432000); if( updated_voting_power > 10000 ) { updated_voting_power = 10000; } newData.trueVotePower = (updated_voting_power/100).toFixed(2); let timeForVotePower = (10000-updated_voting_power)/2000*24*60*60; let voteTimeStamp = moment.utc(moment.utc().valueOf() + timeForVotePower*1000); newData.voteTime = voteTimeStamp.local().format(MWSidebar.settings.dateTimeFormat); newData.voteSpan = MWSidebar.helper.formatSeconds(timeForVotePower); newData.sp = 0; let effective_vesting_shares =0; newData.vesting_shares = parseFloat(newData.vesting_shares.replace(" VESTS","")); newData.delegated_vesting_shares = parseFloat(newData.delegated_vesting_shares.replace(" VESTS","")); newData.received_vesting_shares = parseFloat(newData.received_vesting_shares.replace(" VESTS","")); effective_vesting_shares= newData.vesting_shares - newData.delegated_vesting_shares + newData.received_vesting_shares; newData.sp= MWSidebar.globalProps.totalVestingFund * (effective_vesting_shares / MWSidebar.globalProps.totalVestingShares) newData.ownSp= (MWSidebar.globalProps.totalVestingFund * ( newData.vesting_shares / MWSidebar.globalProps.totalVestingShares)).toFixed(3) newData.recSp= (MWSidebar.globalProps.totalVestingFund * (newData.received_vesting_shares / MWSidebar.globalProps.totalVestingShares)).toFixed(3) newData.delSp= (MWSidebar.globalProps.totalVestingFund * (newData.delegated_vesting_shares / MWSidebar.globalProps.totalVestingShares)).toFixed(3) newData.sp = newData.sp.toFixed(3); let bandwidthData= MWSidebar.calcBandwidth(newData); newData.bw_a = bandwidthData[0]; newData.bw_m = bandwidthData[1]; newData.bw_p = bandwidthData[2]; return newData; }, /** * Update the dom from the new user data * @param {string} target - The id of the target dom element. * @param {object} user - The new user data. */ updateDisplay :function(target, user) { if (user == undefined) return; let content = TEMPLATE_WITH_USER; let links = MWSidebar.settings.links.reduce((result,link)=>{ return (result=={}?"":result) + MWSidebar.helper.fillTemplate(TEMPLATE_LINK, {link: link, user: user, i18n: I18N[MWSidebar.settings.language]}); },""); temp = { target: target, links: links, bw_c: MWSidebar.helper.prettyPrintBytes(user.bw_m-user.bw_a), bw_m: MWSidebar.helper.prettyPrintBytes(user.bw_m), bw_p: user.bw_p.toFixed(2), vote_color: MWSidebar.helper.lerpColor(MWSidebar.settings.barColorHigh, MWSidebar.settings.barColorLow, (100-user.trueVotePower)/100), bw_color: MWSidebar.helper.lerpColor(MWSidebar.settings.barColorHigh, MWSidebar.settings.barColorLow, (100-user.bw_p)/100), bw_pno0: user.bw_p>0?user.bw_p.toFixed(2):"0" }; content = MWSidebar.helper.fillTemplate(TEMPLATE_WITH_USER, {settings: MWSidebar.settings, user: user, temp:temp, i18n: I18N[MWSidebar.settings.language]}); jQuery("#" + target).replaceWith(content); jQuery('#mw-script-container').niceScroll({cursorcolor:"lightblue"}); MWSidebar.refreshCollapse(); }, /** * Refresh user data and updates the dom * @param {string} account - The (new) accountname. * @param {string} target - The id of the target dom element. */ updateAccountInfo : function updateAccountInfo(account, target) { if (account == null || account == "") { jQuery("#" + target).replaceWith('<div id="{target}"></div>'.replace("{target}", target)); return; } steem.api.getAccounts([account], function (err, result) { if ( ( err != null && err != "") || result.length == 0) { user = null; console.log(err); return; } var user = MWSidebar.updateUser(result[0]); MWSidebar.updateDisplay(target, user); }); }, /** * Refresh user data for the account in the "search" box */ update: function() { MWSidebar.updateAccountInfo( MWSidebar.settings.username, "mw-script-content", false); }, /** * Refresh user data for the account of the current page */ updateOther:function() { var newOtherUser; splits = document.location.pathname.split('/'); for (i = 0; i < splits.length; i++) { if (splits[i].startsWith('@')) { newOtherUser = splits[i].replace("@", ""); break; } newOtherUser = ""; } if (newOtherUser == MWSidebar.settings.username){ newOtherUser = ""; } if (newOtherUser != MWSidebar.otherUsername) { MWSidebar.otherUsername = newOtherUser; MWSidebar.updateAccountInfo( MWSidebar.otherUsername, "mw-script-content-other", true); } }, /** * Updates the username from the ui. * @param {event} e - The change event. */ updateUsername : function(e) { if (e.which == 13) { MWSidebar.settings.username = jQuery('#mw-username').val().toLowerCase(); MWSidebar.update(); MWSidebar.saveSettings(); return false; }; }, /** * Refresh the dom to reflect the collapse state. */ refreshCollapse: function(){ if( MWSidebar.settings.collapsed){ jQuery('#mw-script-content').hide(); jQuery('#mw-script-content-other').hide(); jQuery('#mw-username-p').hide(); jQuery('#mw-divider').hide(); jQuery('#mw-footer').hide(); jQuery('#mw-collapse-button').css('margin-left','2px'); jQuery('#mw-script-container').css('width','2px'); jQuery('#mw-script-container').css('padding-left','9px'); jQuery('#mw-script-container').css('height','auto'); jQuery('#mw-script-container').css('background-color','transparent'); if(MWSidebar.settings.side=="left"){ jQuery('#mw-collapse-button').html("⟩"); }else{ jQuery('#mw-collapse-button').html("⟨"); } }else{ jQuery('#mw-script-content').show(); jQuery('#mw-script-content-other').show(); jQuery('#mw-username-p').show(); jQuery('#mw-divider').show(); jQuery('#mw-footer').show(); jQuery('#mw-script-container').css('width','200px'); jQuery('#mw-script-container').css('padding-left','2em'); jQuery('#mw-script-container').css('height','100vh'); jQuery('#mw-script-container').css('background-color',''); jQuery('#mw-collapse-button').css('margin-left','110px'); if(MWSidebar.settings.side=="left"){ jQuery('#mw-collapse-button').html("⟨"); }else{ jQuery('#mw-collapse-button').html("⟩"); } } }, /** * Load all settings from the saved values. */ loadSettings: function(){ MWSidebar.settings.username = MWSidebar.getSetting("mw-username",""); MWSidebar.settings.barColorHigh = MWSidebar.getSetting("mw-barColorHigh", "#00FF00"); MWSidebar.settings.barColorLow = MWSidebar.getSetting("mw-barColorLow", "#FF0000"); MWSidebar.settings.side = MWSidebar.getSetting("mw-side", "left"); MWSidebar.settings.language = MWSidebar.getSetting("mw-language", "en"); MWSidebar.settings.lastSaved = MWSidebar.getSetting("mw-lastSaved", 0); let collapsed = MWSidebar.getSetting("mw-collapsed", false); MWSidebar.settings.collapsed = collapsed==true || collapsed=="true"; let json = MWSidebar.getSetting("mw-links", null); if(json==null||json=="") MWSidebar.settings.links = DEFAULT_LINKS; else MWSidebar.settings.links = JSON.parse(json); const autopost2 = localStorage.getItem('autopost2'); if (autopost2) { [username, password, memoWif, login_owner_pubkey] = MWSidebar.helper.hex2ascii(autopost2,'hex').toString().split('\t'); privPostKey = password; //This will be used for upcoming voting features if(!MWSidebar.settings.username|| MWSidebar.settings.username=="") MWSidebar.settings.username = username; } }, /** * Save all settings */ saveSettings: function(){ MWSidebar.settings.lastSaved = moment.utc().valueOf(); MWSidebar.setSetting("mw-username", MWSidebar.settings.username); MWSidebar.setSetting("mw-collapsed", MWSidebar.settings.collapsed); MWSidebar.setSetting("mw-barColorHigh", MWSidebar.settings.barColorHigh); MWSidebar.setSetting("mw-barColorLow", MWSidebar.settings.barColorLow); MWSidebar.setSetting("mw-links", JSON.stringify(MWSidebar.settings.links)); MWSidebar.setSetting("mw-side", MWSidebar.settings.side); MWSidebar.setSetting("mw-language", MWSidebar.settings.language); MWSidebar.setSetting("mw-lastSaved", MWSidebar.settings.lastSaved); }, /** * Initial DOM-Setup */ setup: function() { MWSidebar.loadSettings(); jQuery('.App__content').eq(0).before( MWSidebar.helper.fillTemplate(TEMPLATE_WITHOUT_USER, {settings: MWSidebar.settings, i18n: I18N[MWSidebar.settings.language]}) ); MWSidebar.refreshCollapse(); jQuery('#mw-username').keypress(MWSidebar.updateUsername); jQuery('#mw-collapse-button').click(MWSidebar.ui.toggleCollapse); jQuery('#mw-settings').click(MWSidebar.ui.openSettings); let index = 0; let links = MWSidebar.settings.links.reduce((result,link)=>{ return (result=={}?"":result) + MWSidebar.helper.fillTemplate(TEMPLATE_LINK_SETTINGS, {link: link, i18n: I18N[MWSidebar.settings.language]}, 5); },""); jQuery("body").append(MWSidebar.helper.fillTemplate(TEMPLATE_SETTINGS_MENU, {settings: MWSidebar.settings, links:links, i18n: I18N[MWSidebar.settings.language]})); MWSidebar.settingsMenu = jQuery("#mw-settings-window").dialog({ autoOpen: false, resizable: false, height: 600, width: "80vw", modal: true, buttons: { "Save Changes": function(){ MWSidebar.saveSettings(); MWSidebar.settingsMenu.dialog("close"); MWSidebar.update(); MWSidebar.updateOther(); } }, close: function() { MWSidebar.loadSettings(); MWSidebar.update(); MWSidebar.updateOther(); } }); MWSidebar.helper.addGlobalStyle(GM_getResourceText("JQUI_CSS")); }, /** * Save a setting permanently * @param {string} cname - The key to save under. * @param {string} cvalue - The value to save. */ setSetting: function(cname, cvalue) { GM_setValue(cname, cvalue); }, /** * Get a setting or returns the default value * @param {string} cname - The key of the saved value. * @param {string} defaultValue - The default value that is returned if nothing is found. */ getSetting: function(cname, defaultValue) { var val=GM_getValue(cname); if(val!=null) return val; var name = cname + "="; var decodedCookie = decodeURIComponent(document.cookie); var ca = decodedCookie.split(';'); for (var i = 0; i < ca.length; i++) { var c = ca[i]; while (c.charAt(0) == ' ') { c = c.substring(1); } if (c.indexOf(name) == 0) { return c.substring(name.length, c.length); } } if(defaultValue!=null) return defaultValue; return ""; }, /** * Async updates the global properties from the api */ updateGlobalProperties: function(){ steem.api.getDynamicGlobalProperties(function(err, result) { MWSidebar.globalProps.totalVestingFund=parseFloat(result.total_vesting_fund_steem.replace(" STEEM", "")); MWSidebar.globalProps.totalVestingShares = parseFloat(result.total_vesting_shares.replace(" VESTS", "")); MWSidebar.globalProps.maxVirtualBandwidth = parseInt(result.max_virtual_bandwidth, 10); }); }, /** * Update the current user */ doUpdate: function(){ MWSidebar.update(); setTimeout(MWSidebar.doUpdate, 10000); //Alle 10 Sekunden das eigene Profil updaten }, /** * Update the user from the current page */ doUpdateOther: function(){ MWSidebar.updateOther(); setTimeout(MWSidebar.doUpdateOther, 100); //Jede 1/10 Sekunde überprüfen ob man jetzt das Profil eines anderen Users offen hat :) }, /** * Update the global properties and replans itself */ doUpdateGlobalProperties: function(){ MWSidebar.updateGlobalProperties(); setTimeout(MWSidebar.doUpdateGlobalProperties, 60000); // Jede Minute die neuen Properties holen }, /** * Look for setting changes in other tabs replans itself */ doReloadSettings: function(){ //Falls eine neuere Version der Settings existiert Settings neu laden let latest=MWSidebar.getSetting("mw-lastSaved",0) if(latest>MWSidebar.settings.lastSaved) MWSidebar.loadSettings(); setTimeout(MWSidebar.doReloadSettings, 1000); // Jede Sekunde prüfen ob neue Einstellungen vorhanden sind } } /** * Entry-Point */ $(document).ready(function () { steem.api.getDynamicGlobalProperties(function(err, result) { MWSidebar.globalProps.totalVestingFund=parseFloat(result.total_vesting_fund_steem.replace(" STEEM", "")); MWSidebar.globalProps.totalVestingShares = parseFloat(result.total_vesting_shares.replace(" VESTS", "")); MWSidebar.globalProps.maxVirtualBandwidth = parseInt(result.max_virtual_bandwidth, 10); MWSidebar.setup(); MWSidebar.doUpdate(); MWSidebar.doUpdateOther(); MWSidebar.doUpdateGlobalProperties(); MWSidebar.doReloadSettings(); }); }); //Attaching it to the window of the current site to make debugging alot easier :) unsafeWindow.MWSidebar = MWSidebar;