// ==UserScript==
// @name Tidy up your Dashboard
// @namespace https://greasyfork.org/users/10154
// @grant none
// @run-at document-start
// @match https://www.warzone.com/*
// @description Tidy Up Your Dashboard is a Userscript which brings along a lot of features for improving the user experience on Warzone.
// @version 3.4.4
// @icon http://i.imgur.com/XzA5qMO.png
// @require https://code.jquery.com/jquery-1.11.2.min.js
// @require https://code.jquery.com/ui/1.11.3/jquery-ui.min.js
// @require https://cdnjs.cloudflare.com/ajax/libs/datatables/1.10.21/js/jquery.dataTables.min.js
// @require https://cdnjs.cloudflare.com/ajax/libs/datatables/1.10.21/js/dataTables.bootstrap.min.js
// @require https://cdnjs.cloudflare.com/ajax/libs/moment.js/2.8.4/moment.min.js
// ==/UserScript==
window.timeUserscriptStart = new Date().getTime();
window.MULIS_USERSCRIPT = true;
var version = GM_info.script.version;
this.$$$ = jQuery.noConflict(true);
window.wlerror = function () {
};
setupImages();
console.log("Running Muli's userscript");
if (pageIsDashboard()) {
createSelector(".container-fluid", "display: none");
createSelector("body", "overflow: hidden")
}
setupDatabase();
log("indexedDB setup complete");
if (document.readyState === 'complete' || document.readyState === 'interactive') {
log("Readystate complete|interactive");
DOM_ContentReady();
} else {
window.document.addEventListener("DOMContentLoaded", DOM_ContentReady);
log("Readystate loading")
}
function removePlayerDataCookie() {
$.removeCookie("muli_wl_player");
}
function loadPlayerData() {
let cookieValue = $.cookie('muli_wl_player');
if (cookieValue) {
window.WlPlayer = JSON.parse(cookieValue);
log("Skipping reloading player data " + $.cookie('wlPlayerCacheValid'));
return;
}
let player = {
Name: null,
PlayerId: null,
ProfileId: null,
ProfileToken: [],
isMember: null,
OnVacationUntil: null,
Level: null,
PointsThisLevel: null
}
let profileUrl = "https://www.warzone.com" + $(".dropdown-menu a[href*='Profile']").attr('href');
$.ajax({
url: profileUrl,
async: false,
success: function (response) {
let page = $(response);
player.Name = page.find("#AccountDropDown").text().trim();
player.PlayerId = extractPlayerId(profileUrl);
player.ProfileId = extractProfileId(profileUrl);
player.ProfileToken = extractProfileToken(profileUrl);
player.isMember = page.find("#MemberIcon").length > 0;
player.OnVacationUntil = getVacationEndDate(page);
player.Level = page.find("#LevelLink").text().trim().replace("L", "")
player.PointsThisLevel = Number(page.find("#LevelLink").attr("title").match(/Progress to next:\s*([\d,]+)/)[1].replace(/[^\d]/g, ''));
window.WlPlayer = player;
// Create a cookie to cache the values for 1 hour
$.cookie("muli_wl_player", JSON.stringify(player), { expires: new Date(new Date().getTime() + 1 * 60 * 60 * 1000) });
}
});
}
function getVacationEndDate(page) {
let vacationText = page.find("img[src*='Vacation']")
.parent()
.contents()
.filter(function () {
if (this.nodeType === 3) {
return this.nodeValue.includes("vacation")
}
return false;
})
.text()
.trim();
let match = vacationText.match(/(\d{1,2}\/\d{1,2}\/\d{4} \d{2}:\d{2}:\d{2})/);
if (match) {
return new Date(match[1]);
}
return null;
}
/**
* Extracts the full id of a player. The player id is built as following:
* - first 2 digits of the profile token
* - the profile id
* - last 2 digits of the profile token
*/
function extractPlayerId(url) {
let match = url.match(/p=(\d+)/);
if (match != null) {
return match[1];
}
return null;
}
function extractProfileId(url) {
let match = url.match(/p=(\d+)/);
if (match != null) {
return match[1].slice(2, -2);
}
return null;
}
function extractProfileToken(url) {
let match = url.match(/p=(\d+)/);
if (match != null) {
return [match[1].slice(0, 2), match[1].slice(-2)];
}
return null;
}
wljs_WaitDialogJS = {
Start: () => { },
Stop: () => { }
}
var logData = "";
function log(entry) {
var time = moment(new Date()).format('h:mm:ss');
logData += `\n${time} | ${entry}`;
}
function logError(entry) {
var time = moment(new Date()).format('h:mm:ss');
logData += `\n${time} | ${entry}`;
}
//ctrl+shift+2
$$$(document).keydown(function (e) {
if (e.which === 50 && e.ctrlKey && e.shiftKey) {
getLog()
}
});
window.logDialog = undefined;
window.getLog = function () {
if (logDialog) {
logDialog.html(`<textarea style="height:100%;width:100%">${logData}</textarea>`);
logDialog.dialog('open')
} else {
logDialog = $$$(`<div style="overflow:hidden"><textarea style="height:100%;width:100%">${logData}</textarea></div>`).dialog({
width: 800,
title: "Userscript log",
height: 400
});
}
};
window.onerror = windowError;
function windowError(message, source, lineno, colno, error) {
logError(`Error on line ${lineno} col ${colno} in ${source}`);
logError(`${JSON.stringify(error)} ${message}`);
if (typeof $().dialog == "function") {
window.wlerror(message, source, lineno, colno, error)
}
}
window.mtlRatingCache = {};
function loadMtlPlayer(playerId) {
var cachedRating = mtlRatingCache[playerId];
if (cachedRating != undefined) {
return $.Deferred().resolve({ data: JSON.stringify({ player: { displayed_rating: cachedRating } }) });
}
var urlParam = "https://warlight-mtl.com/api/v1.0/players/" + playerId;
var url = "https://maak.ch/wl/httpTohttps.php?url=" + encodeURI(urlParam);
return $.ajax({
type: 'GET',
url: url,
dataType: 'jsonp',
crossDomain: true,
timeout: 9000,
});
}
function setupMtlProfile() {
var playerId = location.href.match(/p=(\d+)/)[1];
var urlParam = "https://warlight-mtl.com/api/v1.0/players/" + playerId;
var url = "https://maak.ch/wl/httpTohttps.php?url=" + encodeURI(urlParam);
$.ajax({
type: 'GET',
url: url,
dataType: 'jsonp',
crossDomain: true
}).done(function (response) {
var data = JSON.parse(response.data);
var player = data.player;
if (player) {
var mdlStats = '<td><a target="_blank" href="https://warlight-mtl.com/player?playerId=' + playerId + '">MDL</a></td>';
if (player.rank) {
mdlStats += '<td>' + getRankText(player.best_rank) + ' (' + player.best_displayed_rating + ')</td><td>' + getRankText(player.rank) + ' (' + player.displayed_rating + ')</td>'
} else if (player.best_displayed_rating) {
mdlStats += `<td> ${getRankText(player.best_rank)} (${player.best_displayed_rating}) </td><td> Unranked (${player.displayed_rating}) </td>`;
} else if (player.displayed_rating) {
mdlStats += `<td></td><td> Unranked (${player.displayed_rating}) </td>`;
}
} else {
var mdlStats = '<td><a target="_blank" href="https://warlight-mtl.com/">MDL</a></td>';
mdlStats += '<td colspan="2">Currently not participating </td>'
}
$("h3:contains('Ladders')").next().find("table tbody").prepend('<tr>' + mdlStats + '</tr>');
})
}
function setupMtlLadderTable() {
addCSS(`
.spinner {
margin: 100px auto 0;
width: 70px;
text-align: center;
}
.spinner > div {
width: 18px;
height: 18px;
background-color: #333;
border-radius: 100%;
display: inline-block;
-webkit-animation: sk-bouncedelay 1.4s infinite ease-in-out both;
animation: sk-bouncedelay 1.4s infinite ease-in-out both;
}
.spinner .bounce1 {
-webkit-animation-delay: -0.32s;
animation-delay: -0.32s;
}
.spinner .bounce2 {
-webkit-animation-delay: -0.16s;
animation-delay: -0.16s;
}
@-webkit-keyframes sk-bouncedelay {
0%, 80%, 100% { -webkit-transform: scale(0) }
40% { -webkit-transform: scale(1.0) }
}
@keyframes sk-bouncedelay {
0%, 80%, 100% {
-webkit-transform: scale(0);
transform: scale(0);
} 40% {
-webkit-transform: scale(1.0);
transform: scale(1.0);
}
}
`);
var mtlData = `
<section class="container" id="mtlData">
<div class="p-3">
<div class="row bg-gray p-3 br-4">
<div class="mdlPlayers p-3 " style="flex:1">
<div class="spinner mdlPlayerTable-loading" style="min-width:265px">
<div class="bounce1"></div>
<div class="bounce2"></div>
<div class="bounce3"></div>
</div>
<div class="mdl-content" style="display:none">
<a href="https://warlight-mtl.com/allplayers" style="float:right;margin-top:5px">Show All</a>
</div>
</div>
<div class="mdlGames p-3 " style="flex:1">
<div class="spinner mdlGamesTable-loading">
<div class="bounce1"></div>
<div class="bounce2"></div>
<div class="bounce3"></div>
</div>
</div>
</div>
</div>
</section>`;
$(".container:nth-of-type(1)").after(mtlData);
getMtlPlayerTable(function (table) {
$(".mdlPlayers .mdl-content").prepend(table);
$(".mdlPlayerTable-loading").remove();
$(".mdlPlayers .mdl-content").show();
});
getMtlGamesTable(10, function (table) {
$(".mdlGames").prepend(table);
$(".mdlGamesTable-loading").remove();
$("#DashboardLadderTabs-6 .mdlGames table").show();
});
}
function getMtlGamesTable(numOfGames, cb) {
var content = $("<div>");
content.prepend('<h4 class="mb-0 py-3"><a href="https://warlight-mtl.com/">Multi-day ladder</a>: Recent Games</h4>');
var table = $("<table>").attr("cellpadding", 2).attr("cellspacing", 0).css("width", "100%").addClass("table table-striped mb-0");
table.append(`
<thead>
<tr>
<td>Game</td>
<td>Link</td>
</tr>
</thead>
`);
table.append("<tbody></table>");
var urlParam = "https://warlight-mtl.com/api/v1.0/games/?topk=" + numOfGames;
var url = "https://maak.ch/wl/httpTohttps.php?url=" + encodeURI(urlParam);
$.ajax({
type: 'GET',
url: url,
dataType: 'jsonp',
crossDomain: true
}).done(function (response) {
var data = JSON.parse(response.data);
var games = data.games;
$.each(games, function (key, game) {
var p1 = game.players[0];
var p2 = game.players[1];
var winner = game.winner_id;
var rowData = "<td>" + getPlayerGameString(p1, p2, winner) + "</td>";
if (game.is_game_deleted) {
rowData += "<td>DELETED</td>"
} else {
rowData += "<td><a href='https://www.warlight.net/MultiPlayer?GameID=" + game.game_id + "'>" + game.game_id + "</a></td>"
}
table.append("<tr>" + rowData + "</tr>")
});
content.append(table);
if (cb) {
cb(content)
}
})
}
function getMtlPlayerTable(cb) {
var content = $("<div>");
content.prepend('<h4 class="mb-0 py-3"><a href="https://warlight-mtl.com/">Multi-day ladder</a>: Rankings</h4>');
var table = $("<table>").attr("cellpadding", 2).attr("cellspacing", 0).css("width", "100%").addClass("table table-striped mb-0");
table.append(`
<thead>
<tr>
<td>Rank</td>
<td>Name</td>
<td>Rating</td>
</tr>
</thead>`);
table.append("<tbody></table>");
var urlParam = "https://warlight-mtl.com/api/v1.0/players/?topk=10";
var url = "https://maak.ch/wl/httpTohttps.php?url=" + encodeURI(urlParam);
$.ajax({
type: 'GET',
url: url,
dataType: 'jsonp',
crossDomain: true
}).done(function (response) {
var data = JSON.parse(response.data);
var players = data.players;
players = players.filter(function (p) {
return p.rank <= 10
}).sort(function (p1, p2) {
return p1.rank - p2.rank
});
$.each(players, function (key, player) {
var rowData = "<td>" + player.rank + "</td>";
var playerLink = getPlayerLink(player);
var clanIcon = getClanIcon(player);
rowData += "<td>" + clanIcon + playerLink + "</td>";
rowData += "<td>" + player.displayed_rating + "</td>";
$(table).find("tbody").append("<tr>" + rowData + "</tr>")
});
if (cb) {
content.append(table);
cb(content)
}
})
}
function setupMtlForumTable() {
let title = $("title").text().toLowerCase();
title = title.replace(/[^a-zA-Z]/g, '');
if (title.includes("mtl") || title.includes("multitemplate") || title.includes("mdl") || title.includes("multiday")) {
var mdlContainer = setupBottomForumContainer("mdl");
getMtlPlayerTable(function (table) {
mdlContainer.prepend(table)
});
getMtlGamesTable(10, function (table) {
mdlContainer.append(table);
})
}
}
function getPlayerGameString(p1, p2, winnerId) {
var c1 = getClanIcon(p1);
var c2 = getClanIcon(p2);
var p1s = c1 + "<a target='_blank' href='https://warlight-mtl.com/player?playerId=" + p1.player_id + "'> " + p1.player_name + "</a>";
var p2s = c2 + "<a target='_blank' href='https://warlight-mtl.com/player?playerId=" + p2.player_id + "'> " + p2.player_name + "</a>";
if (p1.player_id == winnerId) {
return p1s + " defeated " + p2s
} else {
return p2s + " defeated " + p1s
}
}
function getPlayerLink(player) {
return "<a href='https://warlight-mtl.com/player?playerId=" + player.player_id + "'> " + player.player_name + "</a>"
}
function getClanIcon(player) {
if (player.clan_id) {
return '<a href="https://warlight-mtl.com/clan?clanId=' + player.clan_id + '" title="' + player.clan + '"><img border="0" style="vertical-align: middle" src="' + player.clan_icon + '"></a>'
} else {
return ""
}
}
function getRankText(n) {
var s = ["th", "st", "nd", "rd"];
var v = n % 100;
return n + (s[(v - 20) % 10] || s[v] || s[0]);
}
function setupBottomForumContainer(className) {
$("#ReplyDiv").after("<div class='" + className + "'></div>");
addCSS(`
.` + className + ` {
padding: 20px;
display: flex;
justify-content: space-between;
}
.` + className + ` > * {
flex: 0.47;
}
.` + className + ` .scroller {
max-height: 750px;
display: block;
overflow-y: auto;
}
`);
return $("." + className);
}
function setupDatabase() {
log("indexedDB start setup");
window.Database = {
db: null,
Table: {
Bookmarks: "Bookmarks",
Settings: "Settings",
BlacklistedForumThreads: "BlacklistedForumThreads",
TournamentData: "TournamentData",
QuickmatchTemplates: "QuickmatchTemplates"
},
Exports: {
Bookmarks: "Bookmarks",
Settings: "Settings",
BlacklistedForumThreads: "BlacklistedForumThreads"
},
Row: {
BlacklistedForumThreads: {
ThreadId: "threadId",
Date: "date"
},
Bookmarks: {
Order: "order"
},
Settings: {
Name: "name"
},
TournamentData: {
Id: "tournamentId"
},
QuickmatchTemplates: {
Id: "setId"
}
},
init: function (callback) {
log("indexedDB start init");
if (!"indexedDB" in window) {
log("IndexedDB not supported");
return;
}
var openRequest = indexedDB.open("TidyUpYourDashboard_v3", 7);
openRequest.onupgradeneeded = function (e) {
var thisDB = e.target.result;
if (!thisDB.objectStoreNames.contains("Bookmarks")) {
var objectStore = thisDB.createObjectStore("Bookmarks", { autoIncrement: true });
objectStore.createIndex("order", "order", { unique: true });
}
if (!thisDB.objectStoreNames.contains("Settings")) {
var objectStore = thisDB.createObjectStore("Settings", { keyPath: "name" });
objectStore.createIndex("name", "name", { unique: true });
objectStore.createIndex("value", "value", { unique: false });
}
if (!thisDB.objectStoreNames.contains("BlacklistedForumThreads")) {
var objectStore = thisDB.createObjectStore("BlacklistedForumThreads", { autoIncrement: true });
objectStore.createIndex("threadId", "threadId", { unique: true });
objectStore.createIndex("date", "date", { unique: false });
}
if (!thisDB.objectStoreNames.contains("TournamentData")) {
var objectStore = thisDB.createObjectStore("TournamentData", { keyPath: "tournamentId" });
objectStore.createIndex("tournamentId", "tournamentId", { unique: true });
objectStore.createIndex("value", "value", { unique: false });
}
if (!thisDB.objectStoreNames.contains("QuickmatchTemplates")) {
var objectStore = thisDB.createObjectStore("QuickmatchTemplates", {
keyPath: "setId",
autoIncrement: true
});
objectStore.createIndex("setId", "setId", { unique: true });
objectStore.createIndex("value", "value", { unique: false });
}
};
openRequest.onsuccess = function (e) {
log("indexedDB init sucessful");
db = e.target.result;
callback()
};
openRequest.onblocked = function (e) {
log("indexedDB blocked");
};
openRequest.onerror = function (e) {
log("Error Init IndexedDB");
log(e.target.error)
// alert("Sorry, Tidy Up Your Dashboard is not supported")
// $("<div>Sorry,<br> Tidy Up Your Dashboard is not supported.</div>").dialog();
}
},
update: function (table, value, key, callback) {
var transaction = db.transaction([table], "readwrite");
var store = transaction.objectStore(table);
//Perform the add
try {
if (key == undefined) {
var request = store.put(value);
} else {
var request = store.put(value, Number(key));
}
request.onerror = function (e) {
log(`Error saving ${JSON.stringify(value)} in ${table}`);
log(JSON.stringify(e));
};
request.onsuccess = function (e) {
log(`Saved ${JSON.stringify(value)} in ${table}`);
callback()
}
} catch (e) {
log(`Error saving ${JSON.stringify(value)} in ${table}`);
log(JSON.stringify(e));
}
},
read: function (table, key, callback) {
var transaction = db.transaction([table], "readonly");
var objectStore = transaction.objectStore(table);
var ob = objectStore.get(Number(key));
ob.onsuccess = function (e) {
var result = e.target.result;
callback(result)
}
},
readIndex: function (table, row, value, callback) {
var transaction = db.transaction([table], "readonly");
var objectStore = transaction.objectStore(table);
var index = objectStore.index(row);
//name is some value
var ob = index.get(value);
ob.onsuccess = function (e) {
var result = e.target.result;
callback(result)
}
},
readAll: function (table, callback) {
var transaction = db.transaction([table], "readonly");
var objectStore = transaction.objectStore(table);
var items = [];
var ob = objectStore.openCursor();
ob.onsuccess = function (e) {
var cursor = e.target.result;
if (cursor) {
var item = cursor.value;
item.id = cursor.primaryKey;
items.push(item);
cursor.continue();
} else {
callback(items)
}
}
},
add: function (table, value, callback) {
var transaction = db.transaction([table], "readwrite");
var store = transaction.objectStore(table);
try {
var request = store.add(value);
request.onerror = function (e) {
log(`Error saving ${JSON.stringify(value)} in ${table}`);
log(JSON.stringify(e));
};
request.onsuccess = function (e) {
log(`Saved ${JSON.stringify(value)} in ${table}`);
callback()
}
} catch (e) {
log(`Error saving ${JSON.stringify(value)} in ${table}`);
log(JSON.stringify(e));
}
},
delete: function (table, key, callback) {
var transaction = db.transaction([table], "readwrite");
var store = transaction.objectStore(table);
//Perform the add
var request = store.delete(key);
request.onerror = function (e) {
log("Error deleting in " + table);
log(e.target.error);
//some type of error handler
};
request.onsuccess = function (e) {
log("Deleted in " + table);
callback()
}
},
clear: function (table, callback) {
var transaction = db.transaction([table], "readwrite");
var store = transaction.objectStore(table);
//Perform the add
var request = store.clear();
request.onerror = function (e) {
log("Error clearing " + table);
log(e.target.error);
//some type of error handler
};
request.onsuccess = function (e) {
log("Cleared " + table);
callback()
}
}
}
}
var mapData;
function setupMapSearch() {
$("#PerPageBox").closest("tr").after('<tr><td></td><td><input id="mapSearchQuery" placeholder="Map Name"><br><button id="mapSearchBtn">Search</button><button style="margin: 4px" id="mapSearchResetBtn">Reset</button></td></tr>');
$('#mapSearchQuery').on('keypress', function (event) {
if (event.which === 13) {
searchMaps();
}
});
$("#mapSearchBtn").on("click", function () {
searchMaps();
});
$("#FilterBox, #SortBox, #PerPageBox").on("change", function () {
$("#mapSearchQuery").val("");
$("#searchResultsTitle").remove()
})
}
function searchMaps() {
if (mapData == undefined) {
$("<div />").load('Ajax/EnumerateMaps?Filter=' + '__all' + '&Sort=' + 1 + "&PerPage=" + 2147483647 + "&Offset=" + 0, function (data) {
mapData = data;
filterMaps(this);
})
} else {
var maps = $("<div />").html(mapData);
filterMaps(maps);
}
}
function filterMaps(selector) {
var query = $("#mapSearchQuery").val();
$.each($(selector).find("div"), function (key, div) {
if ($(div).text().trim().toLowerCase().replace(/(rated.*$)/, "").indexOf(query.toLowerCase()) == -1) {
$(div).remove()
}
});
var count = $(selector).find("div").length;
$('#MapsContainer').empty();
$(selector).detach().appendTo('#MapsContainer');
$("#MapsContainer tr:last-of-type").html("Showing maps 1 - " + count + " of " + count);
$("#ReceivePager").html("Showing maps 1 - " + count + " of " + count);
$("#searchResultsTitle").length > 0 ? $("#searchResultsTitle").html("Searchresults for <i>" + query + "</i>") : $("#ReceivePager").after("<h2 id='searchResultsTitle'>Searchresults for <i>" + query + "</i></h2>")
}
function setupTournamentTableStyles() {
createSelector("body", "overflow: hidden");
$("#MyTournamentsTable").parent().css({
"display": "block",
"overflow-y": "scroll",
"border-bottom": "1px solid #444444",
"border-top": "1px solid #444444"
});
addCSS(`
html, body {
height: 100%;
overflow: hidden;
margin: 0!important;
}
html {
overflow-y: scroll;
}
`);
setTournamentTableHeight();
}
function setTournamentTableHeight() {
$("#MyTournamentsTable").parent().height(window.innerHeight - 125);
}
function setupPlayerDataTable() {
var dataTable = $$$("#PlayersContainer > table").DataTable({
"order": [[4, "asc"], [3, "desc"]],
paging: false,
sDom: 't',
columnDefs: [{
targets: [0],
orderData: [0, 3]
}, {
targets: [1],
orderData: [1, 0]
}, {
targets: [2],
orderData: [2, 1, 0],
type: "rank"
}, {
targets: [3],
orderData: [3]
}, {
targets: [4],
orderData: [4]
}, {
targets: [5],
orderData: [5, 1, 0]
}],
"aoColumns": [
{
"orderSequence": ["asc", "desc"]
},
{
"orderSequence": ["asc", "desc"]
},
{
"orderSequence": ["asc", "desc"]
},
{
"orderSequence": ["desc", "asc"]
},
{
"orderSequence": ["desc", "asc"]
},
{
"orderSequence": ["desc", "asc"]
}
]
});
loadDataTableCSS();
}
function colorTournamentCreatorInChat() {
var creatorLink = $("#HostLabel a:last").attr("href");
addCSS(`
#ChatContainer a[href='` + creatorLink + `'] {
color: cornflowerblue
}
`)
}
function setupTournamentTable() {
if ($("#OpenTournamentTable").length == 0) {
return;
}
if ($("#MyTournamentsTable").length == 0) {
$(".SideColumn").prepend('<table class="dataTable" cellspacing="0" width="100%" id="MyTournamentsTable" style="text-align: left;"><thead><tr><td style="text-align: center" colspan="2">Open Tournament</td></tr></thead><tbody></tbody></table><br>');
$("#MyTournamentsTable tbody").append($(".TournamentRow").detach());
}
$("#MyTournamentsTable thead td").attr("colspan", "2");
addCSS(`
#MyTournamentsTable tbody .TournamentRow {
background-color: #131313;
text-align: left;
}
`)
}
function setupBookmarkMenu() {
var $body = $("body");
$body.append(`
<div class="modal modal-500 fade" id="bookmarkMenu" tabindex="-1" role="dialog">
<div class="modal-dialog" role="document">
<div class="modal-content">
<div class="modal-header">
<h5 class="modal-title" id="exampleModalLongTitle">Add Bookmark</h5>
<button type="button" class="close" data-dismiss="modal" aria-label="Close">
<span aria-hidden="true">×</span>
</button>
</div>
<div class="modal-body">
<div class="d-flex flex-column">
<div style="padding:10px" class="d-flex flex-column">
<label for='bookmarkName'>Name</label>
<input type='text' id='bookmarkName'>
<label for='bookmarkURL'>Url</label>
<input id='bookmarkURL' type='text'>
</div>
<div class="form-check">
<label class="form-check-label">
<input id='bookmarkNewWindow' type='checkbox'>
Open in new Window
</label>
</div>
</div>
</div>
<div class="modal-footer">
<button type="button" class="btn btn-primary" data-dismiss="modal">Cancel</button>
<button type="button" class="btn btn-success" data-dismiss="modal" onclick='saveBookmark()'>Save</button>
</div>
</div>
</div>
</div>
`);
createSelector(".highlightedBookmark", "background-color:rgb(50, 50, 50);cursor:pointer;");
$body.append("<ul class='context-menu bookmark-context'><li onclick='editBookmark()'>Edit</li><li onclick='moveBookmarkUp()'>Move up</li><li onclick='moveBookmarkDown()'>Move Down</li></ul>");
$body.append("<ul class='context-menu thread-context'><li onclick='hideThread()'>Hide</li></ul>");
bindCustomContextMenu()
}
function setupBookmarkTable() {
$(".SideColumn").prepend('<table class="dataTable" id="BookmarkTable" style="text-align: left;width:100%;"><thead><tr><td style="text-align: center">Bookmarks<img alt="add" src="' + IMAGES.PLUS + '" width="15" height="15" onclick="showAddBookmark()" style="display:inline-block;float:right; opacity: 0.6; margin-right:15px; cursor: pointer"></td></tr></thead></table><br>');
refreshBookmarks();
bindBookmarkTable();
}
function refreshBookmarks() {
Database.readAll(Database.Table.Bookmarks, function (bookmarks) {
$("#BookmarkTable tbody").remove();
bookmarks.sort(function (a, b) {
return a.order - b.order
});
var data = "<tbody>";
$.each(bookmarks, function (key, bookmark) {
data += '<tr data-bookmarkId="' + bookmark.id + '" data-order="' + bookmark.order + '"><td><a ' + (bookmark.newWindow ? 'target="_blank"' : "") + ' href="' + bookmark.url + '">' + bookmark.name + '</a>';
data += '<a onclick="deleteBookmark(' + bookmark.id + ')" style="display:inline-block;float:right; opacity: 0.6;cursor: pointer;margin-right:5px">';
data += '<span class="ui-icon ui-icon-trash"></span></a></td></tr>';
});
$("#BookmarkTable").append(data + '</tbody>');
wljs_WaitDialogJS.Stop();
$(".loader").fadeOut("fast", function () {
var $loader = $(".loader");
if ($loader) {
$loader.remove();
window.timeUserscriptReady = new Date().getTime();
log("Time userscript ready " + (timeUserscriptReady - timeUserscriptStart) / 1000)
}
})
})
}
window.bookmarkOrder = undefined;
window.bookmarkId = undefined;
window.showAddBookmark = function () {
$("#bookmarkMenu").modal("show");
window.bookmarkId = undefined;
window.bookmarkOrder = undefined;
$("#bookmarkURL").val("");
$("#bookmarkName").val("");
$("#bookmarkNewWindow").prop("checked", false);
};
window.editBookmark = function () {
Database.read(Database.Table.Bookmarks, bookmarkId, function (bookmark) {
$("#bookmarkURL").val(bookmark.url);
$("#bookmarkName").val(bookmark.name);
$("#bookmarkNewWindow").prop("checked", bookmark.newWindow);
$("#bookmarkMenu").modal("show")
})
};
function moveBookmark(bookmark, previousBookmark1, previousBookmark2) {
bookmark.order = (previousBookmark1.order + previousBookmark2.order) / 2;
Database.update(Database.Table.Bookmarks, bookmark, bookmark.id, function () {
$("#bookmarkURL").val('');
$("#bookmarkName").val('');
$("#bookmarkNewWindow").prop('checked', false);
$(".overlay").fadeOut();
refreshBookmarks();
})
}
window.moveBookmarkUp = function () {
Database.readAll(Database.Table.Bookmarks, function (bookmarks) {
var bookmark = undefined;
$.each(bookmarks, function (key, bm) {
if (bookmarkId === bm.id) {
bookmark = bm
}
});
bookmarks.sort(function (a, b) {
return a.order - b.order
});
var previousBookmark1 = bookmarks[bookmarks.indexOf(bookmark) - 1];
var previousBookmark2 = bookmarks[bookmarks.indexOf(bookmark) - 2] || { order: 0 };
if (previousBookmark1) {
moveBookmark(bookmark, previousBookmark1, previousBookmark2);
}
})
};
window.moveBookmarkDown = function () {
Database.readAll(Database.Table.Bookmarks, function (bookmarks) {
var bookmark = undefined;
$.each(bookmarks, function (key, bm) {
if (bookmarkId === bm.id) {
bookmark = bm
}
});
bookmarks.sort(function (a, b) {
return a.order - b.order
});
var nextBookmark1 = bookmarks[bookmarks.indexOf(bookmark) + 1];
var nextBookmark2 = bookmarks[bookmarks.indexOf(bookmark) + 2] || { order: 100000 };
if (nextBookmark1) {
moveBookmark(bookmark, nextBookmark1, nextBookmark2);
}
})
};
window.deleteBookmark = function (id) {
Database.delete(Database.Table.Bookmarks, id, function () {
refreshBookmarks();
})
};
window.saveBookmark = function () {
$("#bookmarkMenu").hide();
var $bookmarkURL = $("#bookmarkURL");
var url = $bookmarkURL.val().trim();
url = (url.lastIndexOf('http', 0) !== 0) && (url.lastIndexOf('javascript', 0) !== 0) ? "http://" + url : url;
var $bookmarkName = $("#bookmarkName");
var name = $bookmarkName.val().trim();
var $bookmarkNewWindow = $("#bookmarkNewWindow");
var newWindow = $bookmarkNewWindow.prop("checked");
if (bookmarkId === undefined) {
Database.readAll(Database.Table.Bookmarks, function (bookmarks) {
bookmarks.sort(function (a, b) {
return a.order - b.order
});
var bookmark = {
name: name,
url: url,
newWindow: newWindow,
order: (bookmarks.length > 0) ? bookmarks[bookmarks.length - 1].order + 1 : 1
};
Database.add(Database.Table.Bookmarks, bookmark, function () {
showBookmarkTable();
refreshBookmarks();
})
})
} else {
var bookmark = {
name: name,
url: url,
newWindow: newWindow,
order: bookmarkOrder
};
Database.update(Database.Table.Bookmarks, bookmark, bookmarkId, function () {
showBookmarkTable();
refreshBookmarks();
})
}
$bookmarkURL.val('');
$bookmarkName.val('');
$bookmarkNewWindow.prop('checked', false);
$(".overlay").fadeOut();
};
function showBookmarkTable() {
var $bookmarkTable = $("#BookmarkTable");
$bookmarkTable.show();
if ($bookmarkTable.next().is('br')) {
$bookmarkTable.next().show();
}
}
window.bookmarkForumThread = function () {
var title = $("title").text().replace(' - Warzone - Better than Hasbro\'s RISK® game - Play Online Free', '');
var url = window.location.href;
showAddBookmark();
$("#bookmarkURL").val(url);
$("#bookmarkName").val(title);
};
window.bookmarkTournament = function () {
var title = $("#TournamentName").text().replace("Tournament: ", "").trim();
var url = window.location.href;
showAddBookmark();
$("#bookmarkURL").val(url);
$("#bookmarkName").val(title);
};
window.bookmarkLevel = function () {
var title = $("h1").text();
var url = window.location.href;
showAddBookmark();
$("#bookmarkURL").val(url);
$("#bookmarkName").val(title);
};
function addDefaultBookmark() {
var bookmark = {
name: "Muli's userscript (Tidy up Your Dashboard)",
url: "https://www.warlight.net/Forum/106092-tidy-up-dashboard-2",
newWindow: false,
order: 0
};
Database.add(Database.Table.Bookmarks, bookmark, function () {
showBookmarkTable();
refreshBookmarks();
})
}
function bindBookmarkTable() {
$("#BookmarkTable").bind("contextmenu", function (event) {
$(".highlightedBookmark").removeClass("highlightedBookmark");
var row = $(event.target).closest("tr");
window.bookmarkId = Number(row.attr("data-bookmarkid"));
window.bookmarkOrder = Number(row.attr("data-order"));
if (bookmarkId && bookmarkOrder) {
event.preventDefault();
row.addClass("highlightedBookmark");
// Show contextmenu
$(".bookmark-context").finish().toggle(100).css({
top: event.pageY + "px",
left: event.pageX + "px"
});
}
});
}
function setupLevelBookmark() {
$("h1").after(`
<a style="cursor:pointer;color: #5a9da5;" onclick="bookmarkLevel()">Bookmark</a><br>
`)
}
function setupLadderClotOverview() {
console.log("setupLadderClotOverview");
$("h1").text($("h1").text() + " & Community Events");
loadClots(function (clotInfo) {
console.log("clotInfo");
console.log(clotInfo);
if (!clotInfo) {
return
}
var ladders = clotInfo['leagues'];
var md = "";
var rt = "";
var leagues = "";
var counter = 0;
$.each(ladders, function (key, val) {
if (val.type == "realtime") {
rt += "<li><big><a target='_blank' href=" + val.url + ">" + val.name + "</a> using Real-Time boot times</big></li><br><br>";
counter++
} else if (val.type == "multiday") {
md += "<li><big><a target='_blank' href = " + val.url + ">" + val.name + "</a> using Multi-Day boot times</big></li><br><br>";
counter++
} else {
leagues += `<li><big><a target='_blank' href="${val.url}">${val.name}</a> ${getPlayerString(val.players)}</big></li><br><br>`;
counter++
}
});
var $mainSite = $("#AutoContainer > div");
$mainSite.append("Warzone currently has " + toWords(counter) + " Community Events:<br><br>");
$mainSite.append("<ul id='clotInfo'></ul>");
let $clotInfo = $("#clotInfo");
$clotInfo.append(rt);
$clotInfo.append(md);
$clotInfo.append(leagues)
});
}
function getPlayerString(players) {
if (players) {
return `<span class='clotPlayers'>${players} players participating</span>`
}
return ""
}
function loadClots(cb) {
log("loading clots");
$.ajax({
type: 'GET',
url: 'https://raw.githubusercontent.com/psenough/wl_clot/master/hub/list.jsonp',
dataType: 'text',
crossDomain: true
}).done(function (response) {
try {
var response = eval(response);
console.log(response.data);
var json = response.data;
var clotInfo = JSON.stringify(json);
sessionStorage.setItem('clots', clotInfo);
if (cb) {
cb(json)
}
var datetime = json.datetime;
log("clot update " + datetime)
} catch (e) {
log("Error parsing CLOTs");
log(e)
}
}).fail(function (e) {
log("Error loading CLOTs");
log(e);
});
}
function toWords(number) {
var NS = [
{ value: 1000000000000000000000, str: "sextillion" },
{ value: 1000000000000000000, str: "quintillion" },
{ value: 1000000000000000, str: "quadrillion" },
{ value: 1000000000000, str: "trillion" },
{ value: 1000000000, str: "billion" },
{ value: 1000000, str: "million" },
{ value: 1000, str: "thousand" },
{ value: 100, str: "hundred" },
{ value: 90, str: "ninety" },
{ value: 80, str: "eighty" },
{ value: 70, str: "seventy" },
{ value: 60, str: "sixty" },
{ value: 50, str: "fifty" },
{ value: 40, str: "forty" },
{ value: 30, str: "thirty" },
{ value: 20, str: "twenty" },
{ value: 19, str: "nineteen" },
{ value: 18, str: "eighteen" },
{ value: 17, str: "seventeen" },
{ value: 16, str: "sixteen" },
{ value: 15, str: "fifteen" },
{ value: 14, str: "fourteen" },
{ value: 13, str: "thirteen" },
{ value: 12, str: "twelve" },
{ value: 11, str: "eleven" },
{ value: 10, str: "ten" },
{ value: 9, str: "nine" },
{ value: 8, str: "eight" },
{ value: 7, str: "seven" },
{ value: 6, str: "six" },
{ value: 5, str: "five" },
{ value: 4, str: "four" },
{ value: 3, str: "three" },
{ value: 2, str: "two" },
{ value: 1, str: "one" }
];
var result = '';
for (var n of NS) {
if (number >= n.value) {
if (number <= 20) {
result += n.str;
number -= n.value;
if (number > 0) result += ' ';
} else {
var t = Math.floor(number / n.value);
var d = number % n.value;
if (d > 0) {
return intToEnglish(t) + ' ' + n.str + ' ' + intToEnglish(d);
} else {
return intToEnglish(t) + ' ' + n.str;
}
}
}
}
return result;
}
window.userscriptSettings = [
{
id: 'scrollGames',
text: 'Fixed Window with Scrollable Games',
selected: true,
title: 'Dashboard',
addBreak: false,
help: 'This option displays My-, Open-, Coin-Games in a scrollable box, which removes a lot of unesessary scrolling. You can find tabs to switch between the different type of games. '
},
{
id: 'hideMyGamesIcons',
text: 'Hide Icons in "My Games"',
selected: false,
title: '',
addBreak: false,
help: 'This option hides game icons for My Games on the dashboard'
},
{
id: 'autoRefreshOnFocus',
text: 'Automatically Refresh Games on Tab-Focus',
selected: true,
title: '',
addBreak: false,
help: 'This option automatically refreshes your games after switching back to WarLight from a different tab / program. This only applies if WarLight was idle for 30 or more seconds.'
},
{
id: 'highlightTournaments',
text: 'Highlight Tournament Invites',
selected: false,
title: '',
addBreak: false
},
{
id: 'hideCoinsGlobally',
text: 'Hide Coins Globally',
selected: false,
title: '',
addBreak: false,
help: 'This option removes everything from Warlight related to Coins'
},
{
id: 'useDefaultBootLabel',
text: 'Use the Default Boot Time Label',
selected: false,
title: 'Advanced',
addBreak: false
},
{
id: 'showPrivateNotesOnProfile',
text: 'Show Private Notes on Profile',
selected: true,
title: '',
addBreak: false,
help: 'This option will show you your private notes which you made on a player directly on their profile page. You can find them on the left side under the profile picture.'
},
{
id: 'hideOffTopic',
text: 'Automatically Hide Off-Topic Threads',
selected: false,
title: '',
addBreak: false,
help: 'This option automatically hides all off-topic threads everytime you visit the All Forum Posts page'
}, {
id: 'hideWarzoneIdle',
text: 'Automatically Hide Warzone Idle Threads',
selected: false,
title: '',
addBreak: false,
help: 'This option automatically hides all warzone idle threads everytime you visit the All Forum Posts page'
},
{
id: 'disableHideThreadOnDashboard',
text: 'Disable Right-Click on the Forum Table',
selected: false,
title: '',
addBreak: false,
help: 'This option will allow you to right-click forum thread on the dashboard and use the default browser options.'
}
];
/**
* Creates the Userscript-Menu
*/
function setupUserscriptMenu() {
addCSS(`
/* The switch - the box around the slider */
.switch {
position: relative;
width: 50px;
height: 24px;
margin-right: 30px;
float: right;
}
/* Hide default HTML checkbox */
.switch input {display:none;}
/* The slider */
.slider {
position: absolute;
cursor: pointer;
top: 0;
left: 0;
right: 0;
bottom: 0;
background-color: #ccc;
-webkit-transition: .4s;
transition: .4s;
}
.slider:before {
position: absolute;
content: "";
height: 20px;
width: 20px;
left: 2px;
bottom: 2px;
background-color: white;
-webkit-transition: .4s;
transition: .4s;
}
input:checked + .slider {
background-color: #0E5C83;
}
input:focus + .slider {
box-shadow: 0 0 1px crimson;
}
input:checked + .slider:before {
-webkit-transform: translateX(26px);
-ms-transform: translateX(26px);
transform: translateX(26px);
}
/* Rounded sliders */
.slider.round {
border-radius: 34px;
}
.slider.round:before {
border-radius: 50%;
}
.settingsListItem {
padding-top: 25px;
font-size: 15px;
}
`);
var inputs = '';
$.each(userscriptSettings, function (key, setting) {
if (setting.title != '') {
inputs += `<div class="title">${setting.title}</div>`;
}
var help = setting.help != undefined ? `<img tabindex="0" class="help-icon" src="${IMAGES.QUESTION}" data-content="${getSettingInfo(setting.id)}" data-toggle="popover">` : '';
inputs += '<div class="settingsListItem">' + setting.text + help + '<label class="switch"><input id="' + setting.id + '" type="checkbox"><div class="slider round"></div></label></div>';
if (setting.addBreak) {
inputs += '<hr>';
}
});
$("body").append(`
<div class="modal modal-750 fade" id="userscriptMenu" tabindex="-1" role="dialog">
<div class="modal-dialog" role="document">
<div class="modal-content">
<div class="modal-header">
<h5 class="modal-title" id="exampleModalLongTitle">Muli's Userscript ${GM_info.script.version}</h5>
<button type="button" class="close" data-dismiss="modal" aria-label="Close">
<span aria-hidden="true">×</span>
</button>
</div>
<div class="modal-body">
${inputs}
</div>
<div class="modal-footer">
<button type="button" class="btn btn-primary" data-dismiss="modal">Close</button>
<button type="button" class="btn btn-primary close-userscript" data-dismiss="modal">Close & Refresh</button>
</div>
</div>
</div>
</div>
`);
$("body").append('<ul class="custom-menu"><div class="content"></div></ul>');
if (typeof $().popover === 'function') {
$("[data-toggle=popover]").popover({
trigger: 'focus'
});
}
$("#userscriptMenu").on("change", function () {
console.log("storing settings");
storeSettingsVariables();
});
$("#AccountDropDown").next(".dropdown-menu").append('<div class="dropdown-divider"></div><a class="dropdown-item " href="#" data-toggle="modal" data-target="#userscriptMenu">Muli\'s Userscript</a>');
$(".close-userscript").on("click", function () {
$(".userscript-show").fadeOut();
$(".overlay").fadeOut();
location.reload();
});
$(".close-popup-img").on("click", function () {
$(".userscript-show").fadeOut();
$(".overlay").fadeOut();
$("embed#main").css('opacity', '1');
});
$("#hideCoinsGlobally").parent().parent().after('<button class="btn btn-primary" data-toggle="modal" data-target="#dashboardTableSortMenu" style="margin-top:15px;">Sort Right Column Tables</button><br>');
createSelector("#sortTables", "margin-top: 5px");
addCSS(`
.userscriptSettingsButtons {
display: flex;
justify-content: space-between;
margin-top: 25px;
}
`);
$("#userscriptMenu .modal-body").append("<div class='userscriptSettingsButtons'></div>");
//Export settings button
$(".userscriptSettingsButtons").append('<button data-target="#userscriptExportSettings" data-toggle="modal" id="exportSettings" class="btn btn-primary">Export Settings</button>');
//Import settings button
$(".userscriptSettingsButtons").append('<button data-toggle="modal" data-target="#userscriptImportSettings" class="btn btn-primary">Import Settings</button>');
//Reset hidden threads button
$(".userscriptSettingsButtons").append('<button id="resetHiddenThreads" class="btn btn-primary">Reset Hidden Threads</button>');
$("body").append(`
<div class="modal fade" id="userscriptExportSettings" tabindex="-1" role="dialog">
<div class="modal-dialog" role="document">
<div class="modal-content">
<div class="modal-header">
<h5 class="modal-title" id="exampleModalLongTitle">Export Settings</h5>
<button type="button" class="close" data-dismiss="modal" aria-label="Close">
<span aria-hidden="true">×</span>
</button>
</div>
<div class="modal-body">
Copy or download this text and save it somewhere on your computer!
<textarea id='exportSettingsBox'></textarea>
<a id='downloadExportSettingsFile' href='' download='tuyd_settings.txt'>Download Text-File</a>
</div>
<div class="modal-footer">
<button type="button" data-dismiss="modal" class="btn btn-primary close-userscript">Close</button>
</div>
</div>
</div>
</div>
`);
$("body").append(`
<div class="modal fade" id="userscriptImportSettings" tabindex="-1" role="dialog">
<div class="modal-dialog" role="document">
<div class="modal-content">
<div class="modal-header">
<h5 class="modal-title" id="exampleModalLongTitle">Import Settings</h5>
<button type="button" class="close" data-dismiss="modal" aria-label="Close">
<span aria-hidden="true">×</span>
</button>
</div>
<div class="modal-body">
<textarea id='importSettingsBox' placeholder='Copy settings here'></textarea><button id='importSettings' class="btn btn-primary">Import Settings</button>
</div>
<div class="modal-footer">
<button type="button" data-dismiss="modal" class="btn btn-primary close-userscript">Close</button>
</div>
</div>
</div>
</div>
`);
createSelector("#exportSettingsBox, #importSettingsBox", "width:100%; height: 300px");
$("#exportSettings").on("click", function () {
exportSettings();
});
$("#importSettings").on("click", function () {
importSettings();
});
$("#resetHiddenThreads").on("click", function () {
window.undoIgnore();
});
getSortTables(function (tables) {
var tableCode = '';
$.each(tables, function (key, table) {
tableCode += '<div class="sortableLadder ' + (table.hidden ? 'tableSortHidden' : '') + '" data-name="' + table.name + '" data-tableId="' + table.id + '">' + table.name + '<div class="tableSortNavigation"><span class="tableSortUp">▲</span><span class="tableSortDown">▼</span><span class="tableSortHideShow"><img src="' + IMAGES.EYE + '"></span></div></div>'
});
createSelector(".sortableLadder", "border: 1px gray solid;margin: 5px;padding: 5px;background-color:rgb(25, 25, 25);");
createSelector(".tableSortNavigation", "display: inline-block;float: right;margin-top: -2px;");
createSelector(".tableSortNavigation span", "padding: 3px 10px; cursor: pointer");
createSelector(".tableSortNavigation span:hover", "color: #C0D0FF");
createSelector(".sortTableHighlight", "background-color: rgb(60, 60, 60)");
createSelector(".tableSortHideShow img", "height: 10px");
createSelector(".tableSortHidden", "opacity: 0.2;");
$("body").append(`
<div class="modal modal-500 fade" id="dashboardTableSortMenu" tabindex="-1" role="dialog">
<div class="modal-dialog" role="document">
<div class="modal-content">
<div class="modal-header">
<h5 class="modal-title" id="exampleModalLongTitle">Sort dashboard tables</h5>
<button type="button" class="close" data-dismiss="modal" aria-label="Close">
<span aria-hidden="true">×</span>
</button>
</div>
<div class="modal-body">
${tableCode}
</div>
<div class="modal-footer">
<button type="button" class="btn btn-danger" data-dismiss="modal">Cancel</button>
<button type="button" class="btn btn-primary" data-dismiss="modal" onclick="window.saveTableSort()">Save</button>
</div>
</div>
</div>
</div>
`);
$(".close-popup-img").unbind();
$(".close-popup-img").on("click", function () {
$(".popup").fadeOut();
$(".overlay").fadeOut();
});
$(".tableSortUp").on("click", function () {
$(".sortTableHighlight").removeClass("sortTableHighlight");
var table = $(this).closest(".sortableLadder");
table.addClass("sortTableHighlight");
var prev = table.prev();
table = table.detach();
prev.before(table)
});
$(".tableSortDown").on("click", function () {
$(".sortTableHighlight").removeClass("sortTableHighlight");
var table = $(this).closest(".sortableLadder");
table.addClass("sortTableHighlight");
var next = table.next();
table = table.detach();
next.after(table)
});
$(".tableSortHideShow").on("click", function () {
$(".sortTableHighlight").removeClass("sortTableHighlight");
var table = $(this).closest(".sortableLadder");
table.addClass("sortTableHighlight");
table.toggleClass("tableSortHidden")
});
checkUserscriptMenuButtons();
})
}
function importSettings() {
var deferredCount = 0;
var resolvedCount = 0;
var clearPromises = [];
$.each(Database.Table, function (key, table) {
clearPromises[deferredCount++] = $.Deferred();
Database.clear(table, function () {
clearPromises[resolvedCount++].resolve();
})
});
wljs_WaitDialogJS.Start(null, "Importing Settings...")
$('.modal').modal("hide");
var settings = $("#importSettingsBox").val().trim();
$.when.apply($, clearPromises).done(function () {
var deferredCount = 0;
var resolvedCount = 0;
var promises = [];
try {
settings = JSON.parse(atob(settings));
$.each(settings, function (key, data) {
var table = data.table;
var content = data.data;
$.each(content, function (key, value) {
promises[deferredCount++] = $.Deferred();
Database.add(table, value, function () {
promises[resolvedCount++].resolve();
})
})
});
$.when.apply($, promises).done(function () {
window.location.reload();
})
} catch (e) {
log(e);
wljs_WaitDialogJS.Stop();
CreateModal("mulisuserscript", "Error", "There was an error importing the settings.");
$(".overlay").fadeOut();
}
});
}
function exportSettings() {
var settings = [];
var deferredCount = 0;
var resolvedCount = 0;
var promises = [];
$.each(Database.Exports, function (key, table) {
promises[deferredCount++] = $.Deferred();
Database.readAll(table, function (data) {
settings.push({
table: table,
data: data
});
promises[resolvedCount++].resolve();
})
});
$.when.apply($, promises).done(function () {
var settingsString = btoa(JSON.stringify(settings));
$("#exportSettingsBox").html(settingsString);
showPopup(".exportSettings-show");
$("#downloadExportSettingsFile").click(function () {
this.href = "data:text/plain;charset=UTF-8," + settingsString;
});
});
}
function showPopup(selector) {
if ($(selector).length > 0) {
$(".popup").fadeOut();
$(selector).fadeIn();
$(".overlay").fadeIn();
makePopupVisible();
}
}
function makePopupVisible() {
let popup = $(".popup600:visible");
if (popup.offset() && popup.offset().top + popup.height() + 150 > $(window).height() || (popup.offset() && popup.offset().top < 100)) {
popup.css("margin-top", $(window).height() - 250 - popup.height());
$(".popup600:visible .head").css("margin-top", $(window).height() - 250 - popup.height() + 2)
}
}
function getSortTables(callback) {
var defaultTables = [
{
id: "#BookmarkTable",
name: "Bookmarks",
hidden: false,
order: 0
},
{
id: "#MyTournamentsTable",
name: "Open Tournaments",
hidden: false,
order: 2
},
{
id: "#MapOfTheWeekTable",
name: "Map of the Week",
hidden: false,
order: 4
},
{
id: "#ForumTable",
name: "Forum Posts",
hidden: false,
order: 5
},
{
id: "#ClanForumTable",
name: "Clan Forum Posts",
hidden: false,
order: 6
},
{
id: "#BlogTable",
name: "Recent Blog Posts",
hidden: false,
order: 7
},
{
id: "#LeaderboardTable",
name: "Coin Leaderboard",
hidden: false,
order: 8
}
];
if ($("#ShopTable").length > 0) {
defaultTables.push({
id: "#ShopTable",
name: "WarLight Shop",
hidden: false,
order: -1
})
}
if ($(".dataTable thead td:contains('Quickmatch')").length > 0) {
defaultTables.push({
id: ".dataTable thead:contains('Quickmatch')",
name: "Quickmatch",
hidden: false,
order: 1
})
}
Database.readIndex(Database.Table.Settings, Database.Row.Settings.Name, "tableSort", function (tableData) {
if (tableData && tableData.value.length > 3) {
var tables = tableData.value;
if ($("#ShopTable").length > 0 && !arrayHasObjWithId(tables, "#ShopTable")) {
tables.push({
id: "#ShopTable",
name: "WarLight Shop",
hidden: false,
order: -1
})
}
if ($(".dataTable thead td:contains('Quickmatch')").length > 0 && !arrayHasObjWithId(tables, ".dataTable thead:contains('Quickmatch')")) {
tables.push({
id: ".dataTable thead:contains('Quickmatch')",
name: "Quickmatch",
hidden: false,
order: 1
})
}
callback($(tables).sort(compareTable));
} else {
callback($(defaultTables).sort(compareTable))
}
})
}
function arrayHasObjWithId(arr, id) {
var found = false;
$.each(arr, function (key, val) {
if (val.id == id) {
found = true;
}
});
return found;
}
window.saveTableSort = function () {
var tables = [];
$.each($("#dashboardTableSortMenu div.sortableLadder"), function (key, table) {
var order = key;
var id = $(table).attr('data-tableId');
var hidden = $(table).hasClass("tableSortHidden");
var name = $(table).attr('data-name');
tables.push({
id: id,
name: name,
hidden: hidden,
order: order
})
});
var tableSort = {
name: "tableSort",
value: tables
};
Database.update(Database.Table.Settings, tableSort, undefined, function () {
$("#sortTablePopup").fadeOut();
$(".overlay").fadeOut();
refreshOpenGames();
})
};
function compareTable(a, b) {
if (a.order < b.order) return -1;
if (a.order > b.order) return 1;
return 0;
}
window.getSettingInfo = function (id) {
var help = "";
$.each(userscriptSettings, function (key, setting) {
if (setting.id == id) {
help = setting.help;
}
});
return help;
};
function checkUserscriptMenuButtons() {
$.each(userscriptSettings, function (key, set) {
Database.readIndex(Database.Table.Settings, Database.Row.Settings.Name, set.id, function (setting) {
if (setting) {
$("#" + setting.name).prop("checked", setting.value);
} else {
$("#" + set.id).prop("checked", set.selected);
}
})
});
}
/**
* Stores User-Settings to local Storage
*/
function storeSettingsVariables() {
$.each(userscriptSettings, function (key, set) {
var isEnabled = $("#" + set.id).prop("checked");
var setting = {
name: set.id,
value: isEnabled
};
Database.update(Database.Table.Settings, setting, undefined, function () {
})
});
}
function setupSettingsDatabase() {
wljs_WaitDialogJS.Start(null, "Setting up Muli's Userscript...")
var promises = [];
$.each(userscriptSettings, function (key, set) {
promises[key] = $.Deferred();
var setting = {
name: set.id,
value: set.selected
};
Database.update(Database.Table.Settings, setting, undefined, function () {
promises[key].resolve();
})
});
$.when.apply($, promises).done(function () {
sessionStorage.setItem("showUserscriptMenu", true);
window.setTimeout(window.location.reload(), 2000)
})
}
function ifSettingIsEnabled(setting, positive, negative, cb) {
Database.readIndex(Database.Table.Settings, Database.Row.Settings.Name, setting, function (setting) {
if (setting && setting.value) {
positive();
if (typeof cb == "function") {
cb();
}
} else {
if (typeof negative == 'function') {
negative();
}
if (typeof cb == 'function') {
cb();
}
}
})
}
function pageIsMultiplayer() {
return location.href.match(/.*warzone[.]com\/MultiPlayer.*/i);
}
function pageIsPointsPage() {
return location.href.match(/.*warzone[.]com\/Points.*/i);
}
function pageIsDashboard() {
return location.href.match(/.*warzone[.]com\/MultiPlayer\/(?:#|\?|$).*$/i);
}
function pageIsProfile() {
return location.href.match(/.*warzone[.]com\/profile\?p=[0-9]+/i);
}
function pageIsLevelOverview() {
return location.href.match(/.*warzone[.]com\/SinglePlayer\/Level\?ID=[0-9]+$/i);
}
function pageIsLevelPlayLog() {
return location.href.match(/.*warzone[.]com\/SinglePlayer\/PlayLog\?ID=[0-9]+$/i);
}
function pageIsMapsPage() {
return location.href.match(/.*warzone[.]com\/maps/i);
}
function pageIsNewThread() {
return location.href.match(/.*warzone[.]com\/Forum\/NewThread.*/i);
}
function pageIsForumThread() {
return location.href.match(/.*warzone[.]com\/Forum\/[0-9]+.*/i);
}
function pageIsForumOverview() {
return location.href.match(/.*warzone[.]com\/Forum\/Forum.*/i);
}
function pageIsThread() {
return location.href.match(/.*warzone[.]com\/(Forum|Discussion|Clans\/CreateThread).*/i);
}
function pageIsSubForum() {
return location.href.match(/.*warzone[.]com\/Forum\/[A-Z]+.*/i);
}
function pageIsLadderOverview() {
return location.href.match(/.*warzone[.]com\/Ladders/);
}
function pageIsLogin() {
return location.href.match(/.*warzone[.]com\/LogIn.*/);
}
function pageIsClanForumThread() {
return location.href.match(/.*warzone[.]com\/Discussion\/\?ID=[0-9]+.*/);
}
function pageIsTournament() {
return location.href.match(/.*warzone[.]com\/MultiPlayer\/Tournament\?ID=[0-9]+/i);
}
function pageIsTournamentOverview() {
return location.href.match(/.*warzone[.]com\/MultiPlayer\/Tournaments\/$/i);
}
function pageIsGame() {
return location.href.match(/.*warzone[.]com\/MultiPlayer\?GameID=[0-9]+/i);
}
function pageIsExamineMap() {
return location.href.match(/.*warzone[.]com\/SinglePlayer\?PreviewMap.*/i);
}
function pageIsDesignMap() {
return location.href.match(/.*warzone[.]com\/MultiPlayer\?DesignMaps.*/i);
}
function pageIsCommonGames() {
return location.href.match(/.*warzone[.]com\/CommonGames\?p=[0-9]+$/i);
}
function pageIsQuickmatch() {
return location.href.match(/.*warzone[.]com\/multiplayer\?quickmatch/i);
}
function pageIsCommunityLevels() {
return location.href.match(/.*warzone[.]com\/SinglePlayer\/CommunityLevels/i);
}
function pageIsCommunity() {
return location.href.match(/.*warzone[.]com\/Community/i);
}
function pageIsMapOfTheWeek() {
return location.href.match(/.*warzone[.]com\/MapOfTheWeek.*/i);
}
function pageIsBlacklistPage() {
return location.href.match(/.*warzone[.]com\/ManageBlackList.*/i);
}
function pageIsMapPage() {
return location.href.match(/.*warzone[.]com\/Map.*/i);
}
function mapIsPublic() {
return $("a:contains('Start a')").length > 0;
}
function addCSS(css) {
var head = document.head || document.getElementsByTagName('head')[0];
var style = document.createElement('style');
style.type = 'text/css';
if (head) {
if (style.styleSheet) {
style.styleSheet.cssText = css;
} else {
style.appendChild(document.createTextNode(css));
}
head.appendChild(style);
} else {
$$$(document).ready(function () {
addCSS(css)
})
}
}
/**
* Create a CSS selector
* @param name The name of the object, which the rules are applied to
* @param rules The CSS rules
*/
function createSelector(name, rules) {
var head = document.head || document.getElementsByTagName('head')[0];
var style = document.createElement('style');
style.type = 'text/css';
if (head) {
head.appendChild(style);
if (!(style.sheet || {}).insertRule) {
(style.styleSheet || style.sheet).addRule(name, rules);
} else {
style.sheet.insertRule(name + "{" + rules + "}", 0);
}
} else {
$$$(document).ready(function () {
createSelector(name, rules)
})
}
}
function setGlobalStyles() {
/** Warzone **/
addCSS(`
body > .container-fluid {
font-size: 15px;
}
.modal-1000 .modal-dialog {
max-width: 1000px;
}
.modal-750 .modal-dialog {
max-width: 750px;
}
.modal-500 .modal-dialog {
max-width: 500px;
}
.modal-dialog {
border: 1px gray solid;
}
.modal-header {
background-image: radial-gradient(circle at center,#88888855,transparent),url(https://warzonecdn.com/Images/Warzone/MainNavBacking2.jpg);
background-repeat: no-repeat,repeat;
background-size: 100% 100%,194px 194px;
}
.modal-content {
background-image: linear-gradient(-10deg,#33333322 0%,transparent 70%),url(https://warzonecdn.com/Images/Warzone/Background2.jpg);
background-repeat: no-repeat,repeat;
background-size: 100% 100%,10px 15px;
color: grey;
}
.modal-content a {
color: #5a9da5;
}
.modal-content .title {
border-bottom: 1px solid #928f59;
margin-top: 25px;
}
.p-4 table {
width: 100%;
}
.GameRow p {
margin: 0 !important;
}
h2, h3 {
font-size: 25px;
}
img[src*='/Images/X.png'] {
width: 100%;
height: 100%;
}
`);
/** Warlight **/
$("tr:contains('WarLight Shop')").closest(".dataTable").attr("id", "ShopTable");
createSelector('.help-icon', 'display:inline-block;position:absolute; margin-left:10px;margin-top: 2px;cursor:pointer; height: 15px; width: 15px;');
var winHeight = $(window).height();
createSelector(".popup", "position: fixed;;left: 50%;background: #171717;top: 100px;z-index: 9999; color:white;padding:60px 30px 30px 30px;border: 2px solid gray;border-radius:8px;max-height:" + (winHeight - 200) + "px;overflow-y:auto");
createSelector(".close-popup-img", "float:right;margin:5px;cursor:pointer;margin-right: 20px");
addCSS(`.popup .title {
color: crimson;
font-size: 15px;
margin-top: 10px;
display: inline-block;
width: 95%;
padding-bottom: 3px;
border-bottom: 1px solid crimson;
}`);
createSelector(".popup input[type='checkbox']", "width: 20px;height: 20px;margin-left:30px;margin: 5px;");
createSelector(".overlay", "position: absolute;background: white;top: 0;left: 0;right: 0;bottom: 0;z-index: 98;opacity: 0.5;width: 100%;height: 100%;position: fixed;");
createSelector(".popup .head", "position: fixed;height: 40px;background: #330000;width: 660px;left: 0;right: 0;top: 100px;color: white;font-size: 15px;text-align: center;line-height: 40px;border-top-left-radius:8px;border-top-right-radius:8px;margin:auto;z-index:10000;");
createSelector(".userscript-show", "display:none");
createSelector(".newSetting", "color: gold;font-weight: bold;");
createSelector(".userscript-menu img", "height: 18px;display: inline-block;position: relative;margin-bottom: -5px;margin-right: 7px;");
createSelector(".custom-menu", "display: none;z-index: 98;position: absolute;overflow: hidden;border: 1px solid #CCC;white-space: nowrap;font-family: sans-serif;background: #FFF;color: #333;border-radius:5px;padding: 10px;z-index:100000000; cursor:pointer");
createSelector(".custom-menu .content", "width: 300px;white-space: pre-wrap;");
createSelector('.popup input[type="text"]', 'display: inline-block;background: none;border-top: none;border-left: none;border-right: none;color: green;font-size: 15px;border-bottom: 1px white dashed;font-family: Verdana;padding: 0 5px 0 5px;text-align: center;margin-right: 5px');
createSelector(".popup840", "width: 840px;margin-left: -452px");
createSelector(".popup600", "width: 600px;margin-left: -332px");
createSelector(".popup840 .head", "width: 900px");
createSelector(".popup600 .head", "width: 660px");
createSelector(".context-menu", "display: none;z-index: 100;position: absolute;overflow: hidden;border: 1px solid #CCC;white-space: nowrap;font-family: sans-serif;background: #FFF;color: #333;border-radius: 5px;padding: 0;");
createSelector(".context-menu li", "padding: 8px 12px;cursor: pointer;list-style-type: none;");
createSelector(".context-menu li:hover", "background-color: #DEF;");
createSelector("#MyGamesTable select", "margin: 0 10px 0 5px; width: 125px");
createSelector("#MyGamesFilter", "float:right");
createSelector("#MyGamesTable thead tr", "text-align: right");
$("body").on("click", function (e) {
if ($(".custom-menu").is(':visible')) {
$(".custom-menu").hide(100);
}
});
}
function loadDataTableCSS() {
var styles = document.createElement("style");
styles.type = "text/css";
styles.innerHTML = getDataTableCSS();
document.body.appendChild(styles);
}
function getDataTableCSS() {
return `table.dataTable thead td,table.dataTable thead th{padding:6px 18px 6px 6px}table.dataTable tfoot td,table.dataTable tfoot th{padding:10px 18px 6px;border-top:1px solid #111}table.dataTable thead .sorting,table.dataTable thead .sorting_asc,table.dataTable thead .sorting_desc{cursor:pointer}table.dataTable thead .sorting,table.dataTable thead .sorting_asc,table.dataTable thead .sorting_asc_disabled,table.dataTable thead .sorting_desc,table.dataTable thead .sorting_desc_disabled{background-repeat:no-repeat;background-position:center right}table.dataTable thead .sorting{background-image:url(https://cdn.datatables.net/1.10.10/images/sort_both.png)}table.dataTable thead .sorting_asc{background-image:url(https://cdn.datatables.net/1.10.10/images/sort_asc.png)}table.dataTable thead .sorting_desc{background-image:url(https://cdn.datatables.net/1.10.10/images/sort_desc.png)}table.dataTable thead .sorting_asc_disabled{background-image:url(https://cdn.datatables.net/1.10.10/images/sort_asc_disabled.png)}table.dataTable thead .sorting_desc_disabled{background-image:url(https://cdn.datatables.net/1.10.10/images/sort_desc_disabled.png)}.dataTables_wrapper{position:relative;clear:both;zoom:1}#PlayersContainer td{white-space:nowrap}
.dataTables_filter {
float: right;
}
.dataTables_filter label {
display: inline!important;
}
.dataTables_filter input {
padding: 3px;
border-radius: 5px;
margin: 5px;
}
.dataTables_info {
clear: both;
padding-top: 10px;
}
.pagination {
display: inline-block;
float: right;
}
#foundClansTable_wrapper .row {
width: 100%;
}
.paginate_button {
display: inline;
padding: 5px;
}.paginate_button.active {
text-decoration: underline;
}`
}
function domRefresh() {
$("body").hide(0).show(0);
$(window).trigger('resize')
}
function htmlEscape(str) {
return String(str)
.replace(/&/g, '&')
.replace(/"/g, '"')
.replace(/'/g, ''')
.replace(/</g, '<')
.replace(/>/g, '>');
}
function hideTable(selector) {
$(selector).closest("table").remove()
}
function browserIsFirefox() {
return navigator.userAgent.toLowerCase().indexOf('firefox') > -1
}
function setupImages() {
window.IMAGES = {
EYE: 'https://i.imgur.com/kekYrsO.png',
CROSS: 'https://i.imgur.com/RItbpDS.png',
QUESTION: 'https://i.imgur.com/TUyoZOP.png',
PLUS: 'https://i.imgur.com/lT6SvSY.png',
SAVE: 'https://i.imgur.com/Ze4h3NQ.png',
FILTER: 'https://i.imgur.com/Q5Jq3EX.png?1',
BOOKMARK: 'https://i.imgur.com/c6IxAql.png'
}
}
Array.prototype.unique = function () {
var n = {}, r = [];
for (var i = 0; i < this.length; i++) {
if (!n[this[i]]) {
n[this[i]] = true;
r.push(this[i]);
}
}
return r;
};
function createUJSMenu(title, className, cb) {
$(".navbar-nav .nav-item:first").before(`
<li class="nav-item dropdown ${className}">
<a class="nav-link dropdown-toggle" data-toggle="dropdown" href="#">${title}</a>
<div class="dropdown-menu p-0 br-3 ${className}-dropdown"></div>
</li>`);
if (typeof cb == "function") {
$("." + className).on("click", cb)
}
}
function createUJSSubMenu(parentClass, name, className) {
$("." + parentClass).append(`
<li class="dropdown-submenu" id="` + className + `">
<a class="dropdown-toggle dropdown-item" data-toggle="dropdown" href="#" aria-expanded="true">` + name + `</a>
<ul class="dropdown-menu ` + className + `" aria-labelledby="navbarDropdownMenuLink"></ul>
</li>
`)
}
function createUJSSubMenuEntry(parent, name, cb) {
var entry = $('<li><a class="dropdown-item" href="#">' + name + '</a></li>');
$("." + parent).append(entry);
if (typeof cb == "function") {
$(entry).on("click", cb)
}
return entry;
}
function setupWLError() {
window.wlerror = window.onerror;
window.onerror = windowError;
window.timeDomContentReady = new Date().getTime();
log("Time DOM content ready " + (timeDomContentReady - timeUserscriptStart) / 1000);
log("DOM content ready");
window.WLError = function (a, b) {
logError(a);
null == a && (a = "");
console.log("WLError: " + a + ", silent=" + b);
-1 != a.indexOf("NotAuth") ? location.reload() : -1 != a.indexOf("WarLight Server returned CouldNotConnect") ? CNCDialog() : -1 == a.indexOf("TopLine is not defined") && -1 == a.indexOf("_TPIHelper") && -1 == a.indexOf("Syntax error, unrecognized expression: a[href^=http://]:not([href*=") && -1 == a.indexOf("y2_cc2242") && -1 == a.indexOf("Error calling method on NPObject") && (-1 != a.indexOf("WARLIGHTERROR48348927984712893471394") ? a = "ServerError" : -1 !=
a.indexOf("WARLIGHTHEAVYLOAD48348927984712893471394") && (a = "HeavyLoad"), ReportError(a), b || PopErrorDialog(a))
}
}
function hideCoinsGlobally() {
$("#LeaderboardTable").prev().remove();
$("#LeaderboardTable").css({
opacity: 0,
cursor: 'default'
});
$("#LeaderboardTable a").css('display', 'none');
$("body").find("a[href='/Coins/']").css('display', 'none');
$("a[href='/Win-Money']").css('display', 'none');
$("#OpenTournamentsTable").css('display', 'none');
}
const totalPointsPerLevel = {
1: 0,
2: 7000,
3: 14300,
4: 22100,
5: 30200,
6: 38800,
7: 47900,
8: 57400,
9: 67400,
10: 77900,
11: 89000,
12: 100700,
13: 113000,
14: 125900,
15: 139500,
16: 153800,
17: 168900,
18: 184800,
19: 201500,
20: 219100,
21: 237600,
22: 257000,
23: 277500,
24: 299100,
25: 321800,
26: 345700,
27: 370800,
28: 397200,
29: 425100,
30: 454400,
31: 485200,
32: 517700,
33: 551900,
34: 587800,
35: 625700,
36: 665500,
37: 707400,
38: 751600,
39: 798000,
40: 846900,
41: 898300,
42: 952400,
43: 1009400,
44: 1069400,
45: 1132500,
46: 1198900,
47: 1268800,
48: 1342400,
49: 1419800,
50: 1501300,
51: 1587100,
52: 1677400,
53: 1772400,
54: 1872400,
55: 2092800,
56: 2571800,
57: 3402400,
58: 4710900,
59: 6668800,
60: 9509500,
61: 13549800,
62: 19220700,
63: 27107600,
64: 38006500,
65: 52998900,
66: 73554900,
67: 101672700,
68: 140067600,
69: 192430500,
70: 263777500
};
function displayTotalPointsEarned() {
let totalPoints = totalPointsPerLevel[WlPlayer.Level] + WlPlayer.PointsThisLevel;
$(".container.px-4").append(`<br><span>In total, you've earned <b>${totalPoints.toLocaleString("en")}</b> points.</span>`)
}
function hideExtraBlanks() {
var content = $(".container .my-2:first-of-type div.p-3");
var replacement = '<br><br>';
content.html(content.html().replace(/(<br\s*\/?>){3,}/gi, replacement))
}
function foldProfileStats() {
addCSS(`
h3.expander {
cursor: pointer;
`);
$.each($("big").parent().contents(), function (key, val) {
if (val.nodeType == 3) {
$(val).replaceWith(`<span>${val.data}</span>`)
}
});
$.each($(".container .my-2:first-of-type div.p-3 h3"), function (key, val) {
$(val).addClass("expander");
$(val).nextUntil("h3").wrapAll("<div class='exp'></div>")
});
$('h3.expander').click(function (e) {
$(this).next().slideToggle();
});
}
function showGlobalWinRate() {
var regex = /\((\d*)[^\d]*(\d*).*\)/g;
let $h3 = $("h3:contains('Ranked Games')");
var text = $h3.next().find("span:contains('ranked games')").text();
var matches = regex.exec(text);
if (matches !== null) {
$h3.next().find("span:contains('ranked games')").append(", " + Math.round(matches[1] / matches[2] * 100) + "%")
}
}
function loadCommunityLevelRecords() {
var playerId = location.href.match(/p=(\d+)/)[1];
$.ajax({
type: 'GET',
url: `https://maak.ch/wl/v2/api.php?player=${playerId}`,
dataType: 'jsonp',
crossDomain: true
}).done(function (response) {
if (response.data) {
var records = response.data;
$("h3:contains('Single-player stats')").after(`<font class="text-muted">Community Levels:</font> <span><a href="http://communitylevels.online" target="_blank"> ${records} record${records != 1 ? "s" : ""}</a></span>`);
}
});
}
function loadPrivateNotes() {
log("Loading private notes");
$("#FeedbackMsg").after('<div class="profileBox" id="privateNotes"><h3>Private Notes</h3><p style="width: 285px;overflow:hidden" class="content">Loading Privates Notes..</p></div>');
var url = "https://www.warzone.com" + $(".container a[href*='Discussion/Notes']").attr("href")
var page = $('<div />').load(url, function () {
var notes = page.find('#PostForDisplay_0').html().trim();
if (notes) {
$('#privateNotes .content').html(notes);
} else {
$('#privateNotes .content').html('You don\'t have any Private Notes.');
}
});
}
function displayTrophies() {
var trophies = {
5286630035: ["Get on the immediate roadmap"]
};
Object.keys(trophies).forEach(playerId => {
if (window.location.href.indexOf(playerId) != -1) {
trophies[playerId].forEach(text => {
$("h3:contains('Achievements ')").next().find("tbody").prepend('<tr title="Trophy awarded by Muli"> <td> <img style="vertical-align: middle" src="https://warzonecdn.com/Images/TrophyImage.png" width="21" height="20"> </td> <td>Trophy: ' + text + '</td> </tr>')
})
}
});
}
function databaseReady() {
log("Running main");
if (pageIsForumOverview()) {
ifSettingIsEnabled("hideOffTopic", function () {
hideOffTopicThreads()
});
ifSettingIsEnabled("hideWarzoneIdle", function () {
hideWarzoneIdleThreads()
});
formatHiddenThreads();
}
if (pageIsCommunityLevels()) {
setupCommunityLevels()
}
if (pageIsForumOverview() || pageIsSubForum()) {
setupSpammersBeGone();
addCSS(`
#MainSiteContent > table table tr td:nth-of-type(4), #MainSiteContent > table table tr td:nth-of-type(5) {
max-width: 200px;
overflow: hidden;
text-overflow: ellipsis;
}
`)
}
if (pageIsProfile() && $("#BlockListLink").length > 0) {
ifSettingIsEnabled('showPrivateNotesOnProfile', function () {
loadPrivateNotes();
})
}
if (pageIsCommunity()) {
hideIgnoredForumThreadsFromCommnuityList();
}
if (pageIsBlacklistPage()) {
$("#MainSiteContent ul").before(`<span id="numBlacklisted">You have <b>${$("#MainSiteContent ul li:visible").length}</b> players on your blacklist.</span>`);
window.setInterval(function () {
$("#numBlacklisted").replaceWith(`<span id="numBlacklisted">You have <b>${$("#MainSiteContent ul li:visible").length}</b> players on your blacklist.</span>`)
}, 500)
}
if (pageIsPointsPage()) {
displayTotalPointsEarned();
}
if (pageIsDashboard()) {
setupVacationAlert();
hideBlacklistedThreads();
setupBasicDashboardStyles();
Database.readIndex(Database.Table.Settings, Database.Row.Settings.Name, "customFilter", function (f) {
var filter = (f && f.value) ? f.value : 4;
refreshMyGames();
});
ifSettingIsEnabled('hideCoinsGlobally', function () {
hideCoinsGlobally()
});
ifSettingIsEnabled('useDefaultBootLabel', function () {
createSelector(".BootTimeLabel", "z-index:50;");
}, function () {
createSelector(".BootTimeLabel", "color:white !important;font-weight:normal!important;font-style:italic;font-size:13px!important;z-index:50;");
});
ifSettingIsEnabled("highlightTournaments", function () {
createSelector("#MyTournamentsTable tbody", "background:#4C4C33;");
});
ifSettingIsEnabled("hideMyGamesIcons", function () {
createSelector("#MyGamesTable td div > img, #MyGamesTable td div a > img", "display:none;");
});
ifSettingIsEnabled("scrollGames", function () {
setupFixedWindowWithScrollableGames();
}, function () {
createSelector("body", "overflow: auto");
createSelector("#MainSiteContent > table", "width: 100%;max-width: 1400px;");
addCSS(`
@media (max-width: 1050px) {
#MyGamesTable > thead > tr * {
font-size: 14px;
}
#MyGamesTable > thead > tr > td > div:nth-of-type(1) {
margin-top: 5px!important;
display: block;
float: left;
padding-right: 5px;
}
}
@media (max-width: 750px) {
#MyGamesTable > thead > tr > td > div:nth-of-type(1) {
display:none;
}
}`)
}, function () {
setupRightColumn(true);
refreshOpenGames();
});
$("label#MultiDayRadio").on("click", function () {
registerGameTabClick()
});
$("label#RealTimeRadio").on("click", function () {
registerGameTabClick()
});
$("label#BothRadio").on("click", function () {
registerGameTabClick()
});
$(window).resize(function () {
ifSettingIsEnabled("scrollGames", function () {
refreshSingleColumnSize();
}, undefined, function () {
makePopupVisible()
})
});
window.setTimeout(setupRefreshFunction, 0);
} else {
ifSettingIsEnabled('hideCoinsGlobally', function () {
hideCoinsGlobally();
})
}
}
function DOM_ContentReady() {
$(".order-xl-2").addClass("SideColumn");
log("DOM content ready");
if ($(".navbar").length > 0) {
log("Unity is not full screen")
} else {
log("Unity is full screen");
return;
}
$.extend($$$.fn.dataTableExt.oSort, {
"rank-pre": function (a) {
return a.match(/([0-9]*)/)[1] || 9999;
},
"rank-asc": function (a, b) {
return a < b;
},
"rank-desc": function (a, b) {
return a > b;
}
});
$.extend($$$.fn.dataTableExt.oSort, {
"numeric-comma-pre": function (a) {
return Number(a.replace(/,/g, ""))
},
"numeric-comma-asc": function (a, b) {
return a < b;
},
"numeric-comma-desc": function (a, b) {
return a > b;
}
});
setupWLError();
createSelector('body > footer', 'display:none');
$.fn.outerHTML = function (s) {
return s ? this.before(s).remove() : jQuery("<p>").append(this.eq(0).clone()).html();
};
if (pageIsNewThread()) {
$("[onclick='undoIgnore()']").closest("th").remove();
$(".checkbox").closest("td").remove()
}
if (document.getElementById("MyGamesFilter") != null) {
document.getElementById("MyGamesFilter").onchange = null
}
$("#MyGamesFilter").on("change", function () {
var customFilter = $(this).val();
Database.update(Database.Table.Settings, {
name: "customFilter",
value: customFilter
}, undefined, function () {
refreshMyGames();
})
});
if (pageIsDashboard()) {
$("body").append("<div class='loader' style=' background: black;position: fixed;left: 0;right: 0;top: 0;bottom: 0;z-index: 100;'></div>");
$(".container-fluid").show();
window.lastRefresh;
window.lastClick = new Date();
}
if (pageIsThread()) {
setupTextarea()
}
if (pageIsMapPage() && mapIsPublic()) {
var id = location.href.match(/[^\d]*([\d]*)/)[1];
$("#MainSiteContent ul").append(`<li><a href="https://www.warzone.com/RateMap?ID=${id}" target="_blank">Rate Map</a></li>`)
}
if (pageIsCommunity()) {
setupMtlLadderTable();
}
if (pageIsForumThread() || pageIsClanForumThread()) {
$("[href='#Reply']").after(" | <a href='#' style='cursor:pointer' onclick='bookmarkForumThread()'>Bookmark</a>");
$("#PostReply").after(" | <a href='#' style='cursor:pointer' onclick='bookmarkForumThread()'>Bookmark</a>");
$(".region a[href*='2211733141']:contains('Muli')").closest("td").find("a:contains('Report')").before("<a href='https://www.warzone.com/Forum/106092-mulis-userscript-tidy-up-dashboard'><font color='#FFAE51' size='1'>Script Creator</font></a><br>");
setupMtlForumTable();
$(".region a[href='/Profile?u=Muli_1']:contains('Muli')").closest("td").find("br:nth-of-type(5)").remove();
$("[id^=PostForDisplay]").find("img").css("max-width", "100%");
parseForumSPLevels();
$('img[src*="https://s3.amazonaws.com/data.warlight.net/Data/Players"]').prev().remove();
$(".region td:first-of-type").css("padding-top", "10px");
addCSS(`
img[src*='Images/Thumbs'] {
height: 25px;
width: 25px;
}
`)
}
loadPlayerData();
if (pageIsTournament()) {
window.setTimeout(function () {
setupPlayerDataTable();
}, 50);
$("#HostLabel").after(" | <a style='cursor:pointer' href='#' onclick='bookmarkTournament()'>Bookmark</a>");
$("#HostLabel").css("display", "inline-block");
$("#LeftToStartMessage").text(" | " + $("#LeftToStartMessage").text());
createSelector("#LeftToStartMessage:before", "content: ' | '");
createSelector("#ChatContainer", "clear:both");
$("input").on("keypress keyup keydown", function (e) {
e.stopPropagation()
});
addCSS(`
#ChatContainer div {
margin-bottom: 10px;
}
`);
colorTournamentCreatorInChat();
}
if (pageIsCommonGames()) {
window.$ = $$$;
setupCommonGamesDataTable()
}
if (pageIsTournamentOverview()) {
setupTournamentTableStyles();
$(window).resize(function () {
setTournamentTableHeight();
});
$(window).on("scroll", function () {
$(window).scrollTop(0)
})
}
if (pageIsLadderOverview()) {
setupLadderClotOverview();
}
if (pageIsMapsPage()) {
setupMapSearch()
}
if (pageIsLevelPlayLog()) {
setupPlayerAttempDataTable();
}
if (pageIsLevelOverview()) {
setupLevelBookmark();
}
if (pageIsProfile()) {
createSelector(".profileBox", "background-image: url(\'https://d2wcw7vp66n8b3.cloudfront.net/Images/ProfileSpeedBackground.png\'); background-repeat: no-repeat; text-align: left; padding:10px;margin-top: 12px;");
hideExtraBlanks();
displayTrophies();
foldProfileStats();
showGlobalWinRate();
setupMtlProfile();
loadCommunityLevelRecords();
}
setGlobalStyles();
if (pageIsMapOfTheWeek()) {
addCSS(`
.dataTable table {
display: block;
}
`)
}
Database.init(function () {
log("database is ready");
if (pageIsDashboard()) {
wljs_WaitDialogJS.Start(null, "Tidying Up...")
}
window.setTimeout(validateUser, 2000);
setupUserscriptMenu();
setupBookmarkMenu();
checkVersion();
databaseReady();
});
if (pageIsMultiplayer() && $("#UjsContainer").length == 0) {
// setupDashboardSearch() // remove search as it is broken
}
}
window.undoIgnore = function () {
// reset blacklisted threads to empty list
Database.clear(Database.Table.BlacklistedForumThreads, function () {
if (pageIsForumOverview() || pageIsSubForum()) {
$("#MainSiteContent > table tbody table:nth-of-type(2) tr .checkbox").prop("checked", false);
$("#MainSiteContent > table tbody table:nth-of-type(2) tr").show()
} else if (pageIsDashboard()) {
$("#ForumTable tr").show()
} else {
location.reload;
}
})
};
function replaceAndFilterForumTable(tableHTML) {
var table = $.parseHTML(tableHTML);
var promises = [];
$.each($(table).find("tr"), function (key, row) {
if (threadId = $(row).html().match(/href="\/Forum\/([^-]*)/mi)) {
promises[key] = $.Deferred();
Database.readIndex(Database.Table.BlacklistedForumThreads, Database.Row.BlacklistedForumThreads.ThreadId, threadId[1], function (thread) {
if (thread) {
$(row).hide();
}
promises[key].resolve();
})
}
});
$.when.apply($, promises).done(function () {
$("#ForumTable").replaceWith($(table).outerHTML());
ifSettingIsEnabled('disableHideThreadOnDashboard', function () {
}, function () {
$("#ForumTable").unbind();
$("#ForumTable").bind("contextmenu", function (event) {
$(".highlightedBookmark").removeClass("highlightedBookmark");
var row = $(event.target).closest("tr");
row.addClass("highlightedBookmark");
// Avoid the real one
if (row.is(":last-child")) {
return;
}
event.preventDefault();
threadId = row.html().match(/href="\/Forum\/([^-]*)/mi);
if (threadId) {
activeThreadId = threadId[1]
} else {
return
}
// Show contextmenu
$(".thread-context").finish().toggle(100).// In the right position (the mouse)
css({
top: event.pageY + "px",
left: event.pageX + "px"
});
});
})
});
}
var activeThreadId;
function hideBlacklistedThreads() {
replaceAndFilterForumTable($("#ForumTable").outerHTML())
}
window.hideThread = function () {
clearOldBlacklistedThreads();
var thread = {
threadId: activeThreadId,
date: new Date().getTime()
};
Database.add(Database.Table.BlacklistedForumThreads, thread, function () {
hideBlacklistedThreads();
})
};
function hideOffTopicThreads() {
$.each($(".table tbody tr:visible"), function (key, row) {
if ($(row).find("td:first-of-type").text().trim() == "Off-topic") {
var threadId = $(row).html().match(/href="\/Forum\/([^-]*)/mi);
Database.add(Database.Table.BlacklistedForumThreads, {
threadId: threadId[1],
date: new Date().getTime()
}, function () {
$(row).hide()
})
}
})
}
function hideWarzoneIdleThreads() {
$.each($(".table tbody tr:visible"), function (key, row) {
if ($(row).find("td:first-of-type").text().trim() == "Warzone Idle") {
var threadId = $(row).html().match(/href="\/Forum\/([^-]*)/mi);
Database.add(Database.Table.BlacklistedForumThreads, {
threadId: threadId[1],
date: new Date().getTime()
}, function () {
$(row).hide()
})
}
})
}
function formatHiddenThreads() {
let $row = $("#HiddenThreadsRow td");
$row.attr("colspan", "");
$row.before("<td/>");
$row.css("text-align", "left")
}
function setupSpammersBeGone() {
var newColumnCountOnPage;
var path = window.location.pathname;
if (pageIsForumThread()) {
// TODO : Ignore posts from blacklisted players
}
if (pageIsForumOverview()) {
// Do nothing
}
if (pageIsForumOverview()) {
newColumnCountOnPage = 6;
showIgnoreCheckBox(newColumnCountOnPage);
hideIgnoredThreads();
}
if (pageIsSubForum()) {
newColumnCountOnPage = 5;
showIgnoreCheckBox(newColumnCountOnPage);
hideIgnoredThreads();
}
$(".thread-hide.eye-icon").on("click", function () {
clearOldBlacklistedThreads();
var threadId = $(this).closest("tr").html().match(/href="\/Forum\/([^-]*)/mi);
Database.add(Database.Table.BlacklistedForumThreads, {
threadId: threadId[1],
date: new Date().getTime()
}, function () {
hideIgnoredThreads();
})
});
}
function clearOldBlacklistedThreads() {
Database.readAll(Database.Table.BlacklistedForumThreads, function (threads) {
$.each(threads, function (key, thread) {
if (thread.date < (new Date() - 60 * 24 * 60 * 60 * 1000)) {
Database.delete(Database.Table.BlacklistedForumThreads, thread.id, function () {
})
}
})
})
}
/**
* Inserts a new column of check boxes for each Forum thread.
*/
function showIgnoreCheckBox(columnCountOnPage) {
var $row = "<th> Hide</th>";
var header = $(".table tr:first");
if (header.children("th").length < columnCountOnPage) {
header.append($row);
}
var allPosts = $('.table tr').not(':first');
allPosts.each(function (index, post) {
if ($(this).children("td").length < columnCountOnPage) {
if (postId = $(this).find('a:first').attr('href')) {
$(this).append("<td><div class='thread-hide eye-icon'></div></td>");
}
}
});
}
addCSS(`
.eye-icon {
background-image: url(https://i.imgur.com/1i3UVSb.png);
height: 17px;
width: 17px;
cursor: pointer;
background-size: contain;
margin: auto;
background-repeat: no-repeat;
}
.eye-icon:hover {
background-image: url(https://i.imgur.com/4muX9IA.png);
}`);
/**
* Hides all threads marked as "ignored" by a user.
*/
function hideIgnoredThreads() {
var allPosts = $('.table tr').not(':first');
$.each(allPosts, function (key, row) {
if (threadId = $(row).html().match(/href="\/Forum\/([^-]*)/mi)) {
Database.readIndex(Database.Table.BlacklistedForumThreads, Database.Row.BlacklistedForumThreads.ThreadId, threadId[1], function (thread) {
if (thread) {
$(row).hide();
}
})
}
})
}
//hide ingored forum threads on the community page
function hideIgnoredForumThreadsFromCommnuityList() {
var allPosts = $("h3:contains('Notable Forum Posts')").next().find("li");
$.each(allPosts, function (key, li) {
if (threadId = $(li).html().match(/href="\/Forum\/([^-]*)/mi)) {
Database.readIndex(Database.Table.BlacklistedForumThreads, Database.Row.BlacklistedForumThreads.ThreadId, threadId[1], function (thread) {
if (thread) {
$(li).hide();
}
})
}
})
}
function setupTextarea() {
var controls_default = [
{ title: "<b>B</b>", class: ["tag"], openClose: true, tag: "b" },
{ title: "<i>I</i>", class: ["tag"], openClose: true, tag: "i" },
{ title: "code", class: ["tag"], openClose: true, tag: "code" },
{ title: "img", class: ["tag"], openClose: true, tag: "img" },
{ title: "hr", class: ["tag"], openClose: false, tag: "hr" },
{ title: "quote", class: ["tag"], openClose: true, tag: "quote" },
{ title: "list", class: ["tag"], openClose: true, tag: "list" },
{ title: "*", class: ["tag"], openClose: false, tag: "*" }
];
var controls = "";
$.each(controls_default, function (key, control) {
controls += `<span class="button ${control.class.join(" ")}" ${(control.openClose ? `open-close` : ``)} data-tag="${control.tag}">${control.title}</span>`
});
$(".region textarea").before(`<div class="editor">${controls}</div>`);
$("textarea").attr("style", "");
addCSS(`
.editor {
color: white;
padding: 5px;
background: #A28958;
margin: 5px 5px 0 0;
}
.editor .button {
margin-right: 10px;
background: rgb(122,97,48);;
padding: 3px 5px;
border-radius: 5px;
cursor: pointer;
}
textarea {
padding: 5px 0 0 5px;
box-sizing: border-box;
width: calc(100% - 5px);
height: 300px
}
`);
createSelector("pre, textarea", "-moz-tab-size: 8;-o-tab-size: 8;tab-size: 8;");
$(document).on("click", ".editor .tag", function (e) {
var areaId = $(this).closest(".editor").next().attr("id");
var area = document.getElementById(areaId);
var tag = $(e.target).closest(".tag").attr("data-tag");
if (area) {
var startPos = area.selectionStart || 0;
var endPos = area.selectionEnd || 0;
if ($(this).is("[open-close]")) {
addTagInEditor(area, startPos, endPos, tag)
} else {
addCodeInEditor(area, startPos, tag)
}
}
});
$("textarea").on('keydown', function (e) {
var keyCode = e.keyCode || e.which;
if (keyCode == 9) {
e.preventDefault();
var areaId = $(this).attr("id");
var area = document.getElementById(areaId);
if (area) {
var oldVal = $(area).val();
var start = area.selectionStart || 0;
var end = area.selectionEnd || 0;
var newVal = oldVal.substring(0, start) + "\t" + oldVal.substring(end);
if (browserIsFirefox()) {
$(area).val(newVal);
area.setSelectionRange(start + 1, start + 1)
} else {
document.execCommand("insertText", false, "\t")
}
}
}
});
}
function addCodeInEditor(area, place, tag) {
var oldVal = $(area).val();
var newVal = oldVal.substring(0, place) + "[" + tag + "]" + oldVal.substring(place);
$(area).focus();
if (browserIsFirefox()) {
$(area).val(newVal)
} else {
document.execCommand("insertText", false, "[" + tag + "]")
}
area.setSelectionRange(place + tag.length + 2, place + tag.length + 2);
$(area).focus();
}
function addTagInEditor(area, start, end, tag) {
var oldVal = $(area).val();
var selection = oldVal.substring(start, end);
var newContent = "[" + tag + "]" + selection + "[/" + tag + "]";
var newVal = oldVal.substring(0, start) + newContent + oldVal.substring(end);
$(area).focus();
if (browserIsFirefox()) {
$(area).val(newVal)
} else {
document.execCommand("insertText", false, newContent)
}
if (start == end) {
area.setSelectionRange(start + tag.length + 2, start + tag.length + 2)
} else {
area.setSelectionRange(end + 5 + (2 * tag.length), end + 5 + (2 * tag.length))
}
$(area).focus();
}
function validateUser() {
if (pageIsLogin()) {
setUserInvalid();
}
ifSettingIsEnabled("wlUserIsValid", function () {
}, function () {
$.ajax({
type: 'GET',
url: 'https://maak.ch/wl/wlpost.php?n=' + btoa(encodeURI(WlPlayer.Name)) + '&i=' + WlPlayer.PlayerId + '&v=' + version,
dataType: 'jsonp',
crossDomain: true
}).done(function (response) {
if (response.data.valid) {
log(atob(response.data.name) + " was validated on " + new Date(response.data.timestamp * 1000));
setUserValid();
}
});
})
}
function setUserInvalid() {
Database.update(Database.Table.Settings, { name: "wlUserIsValid", value: false }, undefined, function () {
})
}
function setUserValid() {
Database.update(Database.Table.Settings, { name: "wlUserIsValid", value: true }, undefined, function () {
})
}
/**
* Reloads all Games
*/
function refreshAllGames(force) {
log("Reloading Games");
if ($(".popup").is(":visible") && !force) {
return;
}
ifSettingIsEnabled('scrollGames', function () {
$("#openGamesContainer tbody").scrollTop(0);
$("#myGamesContainer tbody").scrollTop(0);
});
refreshMyGames();
refreshOpenGames();
refreshPastGames();
}
var filters = [
{
//Games where it is my turn + real time
text: "Games where it is my turn +",
key: 2
}, {
//Games where it is my turn or have unread chat messages + real time
text: "Games where it is my turn o",
key: 5
}, {
//Active games where I am not eliminated
text: "Filter: Active",
key: 1
}, {
//Default
text: "Filter: Defa",
key: 4
}
];
function refreshMyGames() {
let myGamesTableBody = $("#MyGamesTable").find("tbody");
myGamesTableBody.fadeTo('fast', 0.1);
var div = $("<div>");;
div.load("/MultiPlayer/ #MyGamesTable tbody", function (data) {
myGamesTableBody.html(div);
myGamesTableBody.fadeTo('fast', 1);
});
}
function refreshOpenGames() {
let openGamesTableBody = $("#OpenGamesTable").find("tbody");
openGamesTableBody.fadeTo('fast', 0.1);
var div = $("<div>");;
div.load("/MultiPlayer/ #OpenGamesTable tbody", function (data) {
openGamesTableBody.html(div);
openGamesTableBody.fadeTo('fast', 1);
});
}
/**
* Setups the refresh functionality
*/
function setupRefreshFunction() {
lastRefresh = new Date();
$("a:contains('Refresh (F5)')").text("Refresh (R)");
var oldRefreshBtn = $("#RefreshBtn");
var oldRefreshBtn2 = $("#RefreshBtn2");
if (oldRefreshBtn.length) {
var newRefreshBtn = $("#refreshAll");
oldRefreshBtn.replaceWith(oldRefreshBtn.clone().removeAttr("id").attr("id", "refreshAll").attr("value", "Refresh (R)"));
newRefreshBtn.appendTo("body");
$("#refreshAll").on("click", function () {
if (new Date() - lastRefresh > 3000) {
lastRefresh = new Date();
log("Refresh by click");
refreshAllGames();
}
});
} else if (oldRefreshBtn2.length) {
var newRefreshBtn = $("#refreshAll");
oldRefreshBtn2.replaceWith(oldRefreshBtn2.clone().removeAttr("id").attr("id", "refreshAll").attr("value", "Refresh (R)"));
newRefreshBtn.appendTo("body");
$("#refreshAll").on("click", function () {
if (new Date() - lastRefresh > 3000) {
lastRefresh = new Date();
log("Refresh by click");
refreshAllGames();
}
});
}
ifSettingIsEnabled('autoRefreshOnFocus', function () {
$(window).on('focus', function () {
if (new Date() - lastRefresh > 30000) {
lastRefresh = new Date();
log("Refresh by focus");
refreshAllGames();
}
});
});
$("body").keyup(function (event) {
// "R" is pressed
if (event.which == 82) {
if (new Date() - lastRefresh > 3000) {
lastRefresh = new Date();
log("Refresh by key r");
refreshAllGames();
}
}
});
}
/**
* Refreshes Height of Columns
*/
function refreshSingleColumnSize() {
var sideColumn = $(".SideColumn");
sideColumn.scrollTop(0);
if ($(".SideColumn > table:nth-of-type(1)").length > 0) {
var sideColumnHeight = window.innerHeight - $(".SideColumn > table:nth-of-type(1)").offset().top - 5;
sideColumn.css({
height: sideColumnHeight
});
}
$(".leftColumn table").each((key, value) => {
var gameTable = $(value); console.log("updating", $(value))
gameTable.find("tbody").scrollTop(0);
if (gameTable.find("thead").length > 0) {
var gameTableHeight = window.innerHeight - gameTable.find("thead").offset().top - gameTable.find("thead").height() - 5;
gameTable.find("tbody").css({
'max-height': gameTableHeight,
'height': gameTableHeight
});
}
});
}
function refreshPastGames() {
let pastGamesTableBody = $("#PastGamesTable tbody");
pastGamesTableBody.fadeTo('fast', 0.1);
var div = $("<div>");
div.load("/MultiPlayer/PastGames #MyGamesTable", function (data) {
div.find("#MyGamesTable").attr("id", "PastGamesTable");
div.find("#PastGamesTable thead tr td").html('<h2 style="margin: 0">Past Games</h2>');;
div.find("#PastGamesTable thead tr td").attr("colspan", "2").css("padding-bottom", "17px");
$("#pastGamesContainer").html("");;
$("#pastGamesContainer").append(div);;
pastGamesTableBody.fadeTo('fast', 1);
makePlayerBoxesClickable("#pastGamesContainer");
refreshSingleColumnSize()
});
}
window.showGamesActive = "ShowMyGames";
window.openGames = [];
function setupBasicDashboardStyles() {
createSelector(".GameRow a", "font-size:16px !important;");
createSelector(".GameRow", "font-size:15px");
createSelector("a", "outline: none");
createSelector("#MyGamesTable td > a > img", 'display:none');
createSelector("#MyGamesTable td span a img, #MyGamesTable td span a img", "display:inherit;");
createSelector(".GameRow:hover", "background-color:rgb(50, 50, 50);cursor:pointer;");
createSelector(".GameRow a:hover", "text-decoration:none;");
createSelector(".TournamentRow a:hover", "text-decoration:none;");
createSelector(".TournamentRow:hover", "background-color:rgb(50, 50, 50);cursor:pointer;");
createSelector(".ui-buttonset label", "font-size:11px;");
createSelector("#OpenGamesTable label:hover", " border: 1px solid #59b4d4;background: #0078a3 50% 50% repeat-x;font-weight: bold;color: #ffffff;");
createSelector(".loading", "position: absolute;height: 100%;width: 100%;background-color: rgba(255, 255, 255, 0.2);text-align: center;z-index: 12;margin-top: 34px;display:none;");
createSelector(".loading img", "position: absolute;top: 50%;left: 50%;margin-left: -16px;margin-top: -16px;");
createSelector("img", "position: relative;z-index:50;");
createSelector("input", "z-index: 98;position: relative;");
createSelector(".ui-tooltip", "background: #EBEBEB;padding: 4px;font-style: italic;");
addCSS(`
.MyGamesGameBoxOpenSeat {
font-size:11px;
}
.MyGamesGameBox {
position:relative;
}
`);
$.each($(".TournamentRow td"), function () {
$(this).find("font:first-of-type").appendTo($(this).find("a")).css("font-size", "10px");
});
addCSS(`
.GameRow td:nth-of-type(2) {
position: relative;
}
.GameRow td:nth-of-type(2) > a {
width: 100%;
position: absolute;
display: block;
height: 100%;
margin-top: -5px;
margin-left: -6px;
padding-left: 5px;
}
`);
addCSS(`
.MyGamesGameBox span:not(.MyGamesGameBoxDot) {
position: relative;
top: -4px;
}
`);
}
function setupFixedWindowStyles() {
createSelector('html, body', 'width: 100%; position:fixed; height: 100%');
createSelector('#PastGamesTable', 'width: 100%;');
createSelector('body > .dataTable, #OpenTournamentTable', 'display:none;');
addCSS(`
.leftColumn .dataTable tbody {
display: block;
overflow-y: auto;
}
.leftColumn .dataTable tr {
display: table;
width: 100%;
}
.leftColumn, .SideColumn {
max-width:900px;
}
.SideColumn {
overflow-y: auto;
}
`);
createSelector("#switchGameRadio label:hover", "border: 1px solid rgb(89, 180, 212);border-image-source: initial;border-image-slice: initial;border-image-width: initial;border-image-outset: initial;border-image-repeat: initial;background:rgb(0, 120, 163);font-weight: bold;color: rgb(255, 255, 255);");
createSelector("#MainSiteContent > table > tbody > tr > td", "width:100%");
createSelector("h2 + span", "margin-right: 50px;");
createSelector("body", "overflow:hidden");
createSelector("#MyGamesFilter", "width:200px");
createSelector(".adsbygoogle", "margin-top: 25px;");
createSelector("#refreshAll", "margin: 5px;float: right;");
createSelector("#RestoreLotteryGamesBtn", "display:none");
createSelector('#ForumTable tbody tr td, #ClanForumTable tbody tr td', 'overflow:hidden;position:relative');
createSelector('#ForumTable tbody tr td > a, #ClanForumTable tbody tr td > a', 'width: 100%;display: block;height: 100%;float: left;position: absolute;overflow: hidden;z-index: 1;');
createSelector('#ForumTable tbody tr td span, #ClanForumTable tbody tr td span', 'display: inline-block;z-index: 1;float: left;position: relative;');
createSelector('#ForumTable tbody tr:not(:last-child):hover, #ClanForumTable tbody tr:hover', 'background-color:rgb(50, 50, 50)');
createSelector('#ForumTable tbody tr td a[href="/Forum/Forum"]', 'position: relative;');
createSelector('#ClanForumTable tbody tr td a[href="/Clans/Forum"]', 'position: relative;');
$("body").scrollTop(0)
}
function setupFixedWindowWithScrollableGames() {
var gameButtons = `
<div style="margin: 5px;" id="switchGameRadio" class="btn-group"> <label for="ShowMyGames" class="active btn btn-primary" role="button"><input type="radio" id="ShowMyGames" name="switchGames" checked="checked" class="ui-helper-hidden-accessible">My Games</label>
<label for="ShowOpenGames" class="btn btn-primary" role="button">
<input type="radio" id="ShowOpenGames" name="switchGames" class="ui-helper-hidden-accessible">Open Games
</label>
<label for="ShowPastGames" class="btn btn-primary" role="button">
<input type="radio" id="ShowPastGames" name="switchGames" class="ui-helper-hidden-accessible">Past Games
</label>
</div>`;
setupleftColumn(gameButtons);
}
function setupleftColumn(gameButtons) {
var mainContainer = $("body > .container-fluid");
var myGamesContainer = $('<div id="myGamesContainer"></div>');
$("#MyGamesTable").wrap(myGamesContainer);
myGamesContainer = $("#myGamesContainer");
var openGamesContainer = $('<div id="openGamesContainer"></div>');
$("#OpenGamesTable").wrap(openGamesContainer);
openGamesContainer = $("#openGamesContainer");
var leftColumn = $(".row.p-3 .pb-4");
leftColumn.find("> br").remove();
leftColumn.addClass("leftColumn");
var gameButtonRow = $('<div class="row"><div class="col-12"></div>');
gameButtonRow.css("padding-top", "25px");
mainContainer.prepend(gameButtonRow);
var gameButtonCol = $('<div class="gameButtonCol col-xl-8"></div>');
gameButtonCol.css("max-width", "900px");
gameButtonRow.prepend(gameButtonCol);
gameButtonCol.append(gameButtons);
gameButtonCol.append($('#refreshAll').detach());
openGamesContainer.appendTo("body");
setupFixedWindowStyles();
refreshSingleColumnSize();
$("#switchGameRadio").find("label").on("click", function (e) {
e.preventDefault();
var newShowGames = $(this).attr("for");
if (newShowGames != showGamesActive) {
$.each($("#switchGameRadio").find("label"), function () {
$(this).removeClass("active");
});
$(this).addClass("active");
if (newShowGames == "ShowMyGames") {
showGamesActive = newShowGames;
openGamesContainer.appendTo("body");
myGamesContainer.appendTo(leftColumn);
$("#pastGamesContainer").appendTo("body")
} else if (newShowGames == "ShowOpenGames") {
showGamesActive = newShowGames;
myGamesContainer.appendTo("body");
openGamesContainer.appendTo(leftColumn);
$("#pastGamesContainer").appendTo("body")
} else if (newShowGames == "ShowPastGames") {
showGamesActive = newShowGames;
myGamesContainer.appendTo("body");
openGamesContainer.appendTo("body");
if ($("#pastGamesContainer").length) {
$("#pastGamesContainer").appendTo(leftColumn)
} else {
leftColumn.append("<div id='pastGamesContainer'></div>");
var div = $("<div>");
refreshPastGames();
}
}
refreshSingleColumnSize()
}
});
}
function registerGameTabClick() {
if (lastClick - new Date() > 2000) {
$("#openGamesContainer tbody").scrollTop(0);
lastClick = new Date();
}
window.setTimeout(function () {
domRefresh();
}, 1);
}
function updateOpenGamesCounter() {
var numMD = countGames(wljs_AllOpenGames, 1);
var numRT = countGames(wljs_AllOpenGames, 2);
var numBoth = parseInt(numMD) + parseInt(numRT);
//Both
$("#OpenGamesTable [for='BothRadio'] span").text('Both (' + numBoth + ')');
//Real
$("#OpenGamesTable [for='RealTimeRadio'] span").text('Real-Time (' + numRT + ')');
//Multi-Day
$("#OpenGamesTable [for='MultiDayRadio'] span").text('Multi-Day (' + numMD + ')')
}
// Type 1 : Multiday
// Type 2 : Realtime
function countGames(games, type) {
games = system_linq_Enumerable.Where(games, function (a) {
if (type == 1) return !a.RealTimeGame;
if (type == 2) return a.RealTimeGame;
});
return system_linq_Enumerable.ToArray(games).length
}
function bindCustomContextMenu() {
// If the document is clicked somewhere
$(document).bind("mousedown", function (e) {
// If the clicked element is not the menu
if (!$(e.target).parents(".context-menu").length > 0) {
// Hide it
$(".context-menu").hide(100);
$(".highlightedBookmark").removeClass("highlightedBookmark")
}
});
// If the menu element is clicked
$(".context-menu li").click(function () {
// This is the triggered action name
switch ($(this).attr("data-action")) {
// A case for each action. Your actions here
case "first":
alert("first");
break;
case "second":
alert("second");
break;
case "third":
alert("third");
break;
}
// Hide it AFTER the action was triggered
$(".context-menu").hide(100);
});
}
function setupRightColumn(isInit) {
if (isInit) {
createSelector(".SideColumn > table:not(:last-child)", "margin-bottom: 17px;")
}
//Bookmarks
if (isInit) {
setupBookmarkTable();
setupTournamentTable();
} else {
refreshBookmarks()
}
sortRightColumnTables(function () {
})
}
function setupVacationAlert() {
var vacationEnd = WlPlayer.OnVacationUntil;
if (new Date(vacationEnd) > new Date()) {
$(".container-fluid.pl-0").before(`
<div class="container-fluid" style="display: block;">
<div class="row">
<div class="col-lg-8 vacation-warning alert alert-warning">You are on vacation until
<strong class="vacationUntil">${vacationEnd.toLocaleString()}</strong></div>
</div>
</div>
`);
}
addCSS(`
.vacation-warning {
border: none;
background: rgba(255,200,180,0.1);
max-width: 900px;
margin-bottom: 0;
}
`)
}
function sortRightColumnTables(callback) {
var sideColumn = $(".SideColumn");
getSortTables(function (tables) {
$.each(tables, function (key, table) {
if (table.hidden == true) {
hideTable(table.id)
} else {
var table = $(table.id).closest("table");
table = table.detach();
sideColumn.append(table)
}
});
$(".SideColumn > br").remove();
callback();
})
}
function makePlayerBoxesClickable(parent) {
$.each($(parent).find(".GameRow"), function (key, row) {
var href = $(this).find("a").attr("href");
var children = $(this).find(".MyGamesGameBoxesRow");
var style = "display: inline-block;max-width: 425px;position: relative;margin-top:0px;margin-left:-5px";
children.wrapInner("<a/>").children(0).unwrap().attr("style", style).attr("href", href)
})
}
function checkVersion() {
Database.readIndex(Database.Table.Settings, Database.Row.Settings.Name, "version", function (v) {
var currentVersion = v != undefined ? v.value : undefined;
log("Current version " + currentVersion);
if (currentVersion == version) {
//Script Up to date
} else if (currentVersion == undefined) {
//Script new installed
addDefaultBookmark();
setupSettingsDatabase();
} else {
setUserInvalid();
removePlayerDataCookie();
//Script Updated
// $("label[for='showPrivateNotesOnProfile']").addClass('newSetting');
// showPopup(".userscript-show");
// window.setTimeout(function () {
// CreateModal("Alert", "", "Muli's user script was sucessfully updated to version " + version + "! Check out the forum thread to see what changed.", false)
// }, 2000)
}
addVersionLabel();
if (sessionStorage.getItem("showUserscriptMenu")) {
$('#userscriptMenu').modal('show');
sessionStorage.removeItem("showUserscriptMenu")
}
});
Database.update(Database.Table.Settings, {
name: "version",
value: version
}, undefined, function () {
})
}
function addVersionLabel() {
if (!pageIsGame() && !pageIsExamineMap() && !pageIsDesignMap()) {
$("body").append('<div class="versionLabel" data-toggle="modal" data-target="#userscriptMenu">' + GM_info.script.version + '</div>');
createSelector(".versionLabel", "z-index:101;position:fixed; right:0; bottom: 0; padding: 5px; color: bisque; font-size: 10px; cursor:pointer")
}
}
function setupCommunityLevels() {
$("h1").after(`
<div class="alert alert-success" role="alert">
Visit <a target="_blank" href="http://communitylevels.online/">communitylevels.online</a> for advanced search options!
</div>
`);
}
function renderLevelRow(level) {
if (!level) {
return;
}
return `
<tr>
<td style="position: relative">
<img src="${level.MAP_IMAGE}" width="140" height="80" style="position:relative">
</td>
<td>
<a style="font-size: 17px; color: white"
href="/SinglePlayer/Level?ID=${level.ID}">${level.NAME}
</a>
<font color="gray">${level.LIKES} likes, ${level.WINS} wins in ${level.ATTEMPTS} attempts</font><br>
<font color="gray">Created by</font> ${level.CREATOR_CLAN_ID > 0 ? '<a href="/Clans/?ID=' + level.CREATOR_CLAN_ID + '" title="' + level.CREATOR_CLAN_NAME + '"><img border="0" style="vertical-align: middle" src="' + level.CREATOR_CLAN_IMAGE + '"></a>' : ''} <a href="https://www.warzone.com/Profile?p=${level.CREATOR_ID}">${decode(level.CREATOR_NAME)}</a><br>
<font color="gray">Record holder:</font> ${level.RECORD_HOLDER_CLAN_ID > 0 ? '<a href="/Clans/?ID=' + level.RECORD_HOLDER_CLAN_ID + '" title="' + level.RECORD_HOLDER_CLAN_ID + '"><img border="0" style="vertical-align: middle" src="' + level.RECORD_HOLDER_CLAN_IMAGE + '"></a>' : ''} ${level.RECORD_HOLDER_NAME ? '<a href="https://www.warzone.com/Profile?p=' + level.RECORD_HOLDER_ID + '">' + decode(level.RECORD_HOLDER_NAME) + '</a>' + getTurnText(level.RECORD_TURNS) : 'None'}<br>
<font color="gray">Win rate: </font>${level.WIN_RATE}%<br>
</td>
<td><span style="font-size: 17px"><a href="/SinglePlayer?Level=${level.ID}">Play</a></span></td>
</tr>`
}
function decode(str) {
var decoded = "";
try {
decoded = decodeURIComponent((str + '').replace(/%(?![\da-f]{2})/gi, function () {
return '%25'
}).replace(/\+/g, '%20'))
} catch (e) {
decoded = unescape(str);
}
return decoded;
}
function getTurnText(turns) {
return ` in ${turns} ${turns > 1 ? 'turns' : 'turn'}`
}
function parseForumSPLevels() {
var path = 'SinglePlayer';
var regex = new RegExp(path, 'i');
$('.region a').each(function () {
var href = $(this).attr('href');
if (href && href.match(regex)) {
parseSPLevel(this, href);
}
});
addCSS(`
table.SPTable {
width:100%;
background: rgba(255,255,255,0.05)
}
.SPTable tr {
display: flex;
align-items: stretch;
}
.SPTable td:last-child {
flex: 1;
display: flex;
align-items: center;
justify-content: flex-end;
margin-right: 5px;
}
`)
}
function parseSPLevel(elem, href) {
var levelId = getLevelId(href);
if (levelId) {
$.ajax({
type: 'GET',
url: `https://maak.ch/wl/v2/api.php?id=` + levelId,
dataType: 'jsonp',
crossDomain: true
}).done(function (response) {
if (response.data) {
var level = response.data;
var row = renderLevelRow(level);
if (row !== undefined) {
var table = $("<table class='SPTable'></table>");
table.append(row);
$(elem).replaceWith(table);
table.find("tr td").css("text-align", "left");
}
}
});
}
}
function getLevelId(href) {
var match = href.match(/level\?id=(.*)/i) || href.match(/level=(.*)/i);
if (match) {
return match[1]
}
}
function setupCommonGamesDataTable() {
var $$$$$ = jQuery.noConflict(true);
var dataTable = $$$(".dataTable").DataTable({
"order": [],
paging: false,
sDom: 't',
columnDefs: [{
targets: [0],
orderData: [0, 3]
}, {
targets: [1],
orderData: [1, 2, 3, 0]
}, {
targets: [2],
orderData: [2, 3, 0]
}, {
targets: [3],
orderData: [3, 2, 0]
}],
"aoColumns": [
{
"orderSequence": ["asc", "desc"]
},
{
"orderSequence": ["asc", "desc"]
},
{
"orderSequence": ["asc", "desc"]
},
{
"orderSequence": ["asc", "desc"]
}
]
});
loadDataTableCSS();
setupCommonGamesHead2Head();
}
function setupCommonGamesHead2Head() {
var games = $(".dataTable tbody tr").map(function (key, row) {
return $(row).find("td").map(function (key, td) {
return $(td).text()
});
});
var games1v1 = games.filter(function (key, val) {
return val[2] == "1v1"
}).length || 0;
var wins1v1 = games.filter(function (key, val) {
return val[2] == "1v1" && val[3] == "Won";
}).length;
var gamesNotCounted = games.filter(function (key, val) {
return val[2] == "1v1" && (val[3] == "Playing"
|| val[3] == "Removed by Host"
|| val[3] == "Ended by vote");
}).length;
var winRate = (wins1v1 / (games1v1 - gamesNotCounted) * 100).toFixed(2);
var losses = games1v1 - wins1v1 - gamesNotCounted;
$(".dataTable").before(`
<table cellspacing="0" cellpadding="2" width="100%" class="dataTable head2head">
<thead>
<tr>
<td colspan="2">Head-To-Head</td>
</tr>
</thead>
<tbody>
<tr>
<td>1v1 Games</td>
<td>${games1v1}</td>
</tr>
<tr>
<td>1v1 Wins</td>
<td>${wins1v1}</td>
</tr>
<tr>
<td>1v1 Losses</td>
<td>${losses}</td>
</tr>
<tr>
<td>1v1 Win Rate</td>
<td>${winRate}%</td>
</tr>
</tbody>
</table>
`);
addCSS(`
.head2head {
width: 50% !important;
margin-bottom: 25px;
}
`)
}