// ==UserScript==
// @name Reworks Oracle for Epicmafia as script
// @namespace https://greasyfork.dpdns.org/en/users/159342-cleresd
// @description Reworks Oracle for Epicmafia as script. Originally created by lailai. Add rehost button, autorefresh ability and game search ability.
// @version 1.06
// @match https://epicmafia.com/*
// @exclude https://epicmafia.com/game*
// @grant GM_addStyle
// @grant GM_setValue
// @grant GM_getValue
// @run-at document-end
// ==/UserScript==
GM_addStyle(`
/* head additional */
._orcDropdownContent {
display: none;
position: absolute;
background-color: #eee;
min-width: 160px;
box-shadow: 0 8px 16px 0 rgba(0, 0, 0, 0.2);
border-radius: 6px;
z-index: 5;
font-size: 85%;
border: 1px solid #cd88d3;
}
._orcDropdownContent a {
color: black;
padding: 12px 16px;
text-decoration: none;
float: left;
}
.sel:hover ._orcDropdownContent {
display: none;
}
._orcDropdownContent a:hover {
background-color: #ccc;
}
li:hover ._orcDropdownContent {
display: block;
}
/* auto refresh button */
._oracle_icon {
background-image: url();
height: 19px;
width: 19px;
display: inline-block;
position: relative;
top: 1px;
}
#_oAutoRefreshWrap {
display: inline-block;
border: 1px solid #cd88d3;
padding: 0 2px 2px 2px;
margin-left: 2px;
}
#_oAutoRefreshWrap::before {
content: '';
position: absolute;
right: 48px;
top: 9px;
width: 0;
height: 0;
border-top: 6px solid transparent;
border-bottom: 6px solid transparent;
border-right: 6px solid #c788d3;
}
#_oAutoRefreshWrap ._oracle_icon {
top: 3px;
left: -4px;
}
/* auto rehost */
._oRehost {
margin-left: 5px;
border: 1px solid #c788d3;
padding: 5px 5px 4px 5px;
color: #c788d3;
font-size: 12px;
}
._oRehost:hover {
background-color: #c788d3;
color: white;
}
/* other */
select[name='siterule_id'] {
margin: 10px 0;
}
@keyframes fadeIn {
from {
opacity: 0;
}
to {
opacity: 1;
}
}`);
// It needs to auto refresh games' list
let lastActivity = Date.now();
$('body').mousemove(e => lastActivity = Date.now());
$("#header_inner").find("ul li:eq(0)").append ( `
<div class="_orcDropdownContent">
<a href="https://epicmafia.com/game/new">Create game</a>
<a href="https://epicmafia.com/setup">Find setup</a>
<a href="https://epicmafia.com/lobby/rules">Rules</a>
` );
$("#header_inner").find("ul li:eq(1)").append ( `
<div class="_orcDropdownContent">
<a href="https://epicmafia.com/message">Inbox</a>
<a href="https://epicmafia.com/family">Family</a>
<a href="https://epicmafia.com/addon">Buy</a>
` );
$("#header_inner").find("ul li:eq(2)").append ( `
<div class="_orcDropdownContent">
<a href="https://epicmafia.com/moderator">Moderators</a>
<a href="https://epicmafia.com/report">Reports</a>
` );
if ($(".ll-refresh").length) {
insertAutoRefresh();
insertRehost();
insertSearch();
}
if ($("#nav").find("li.sel a").text() === "Round") {
$("#subnav").append(`<li><a href="/lobby/rules" style="color: #c788d3">Rules</a></li>`);
}
function insertRehost() {
let $body = $("body");
let rehostButtonHtml = `<div class="_oRehost" data-name="mafia_gamerow">Re</div>`;
$body.on("mouseenter", ".gamerow.mafia_gamerow, .gamerow.dicewars_gamerow, .gamerow.snakes_gamerow", e => {
$(e.currentTarget).find('.gamesetup').parent().before(rehostButtonHtml);
//$(e.currentTarget).find(".gamesetup").parent().before();
});
$body.on("mouseleave", ".gamerow", e => {
$(e.currentTarget).find("._oRehost").remove();
});
$body.on("click", "._oRehost", e => {
$(e.currentTarget).text("Rehosting");
const $gameRow = $(e.currentTarget).parents(".gamerow");
const type = $gameRow[0].classList[2].split('_')[0];
const numPlayers = $gameRow.find('.players').children().length;
const isGoldHeart = $gameRow.find("img[src='/images/goldlives.png'],img[src='/images/broken_goldlives.png']").length !== 0;
const isRedHeart = $gameRow.find("img[src='/images/lives.png'],img[src='/images/broken_lives.png']").length !== 0;
let isPassword = $gameRow.find("img[src='/images/lock.png']").length !== 0 ? 1 : 0;
let password = '';
const rank = isGoldHeart ? 2 : (isRedHeart ? 1 : 0);
const gid = $gameRow.attr("data-gid");
if (isPassword) {
$('#pop').removeClass('hide');
$('#pop_').removeClass('hide');
$('#pop_warn').css('display', 'none');
$('#password').attr('placeholder', 'Leave empty to no password');
$('#password').next().click(() => {
$('#pop_').addClass('hide');
$('#pop').addClass('hide');
$('#pop_warn').css('display', '');
password = $('#password').val();
if (!password) {
isPassword = 0;
}
rehost(gid, type, rank, isPassword, password, numPlayers);
return true;
});
}
else {
rehost(gid, type, rank, isPassword, password, numPlayers);
}
});
}
function rehost(gid, type, rank, isPassword, password, numPlayers) {
if (type === 'mafia') {
rehostMafiaGame(gid, rank, isPassword, password)
} else if (type === 'dicewars' || type === 'snakes') {
isPassword = !!isPassword;
rehostCustomGame(type, numPlayers, isPassword, password)
}
}
function rehostMafiaGame(gid, rank, isPassword, password) {
$.get(`https://epicmafia.com/game/${gid}/info`, data => {
const setup =data[1].data.match(/\/setup\/[0-9]+/)[0].split("/")[2];
$.get(`https://epicmafia.com/game/add/mafia?` +
`setupid=${setup}&ranked=${rank}&add_password=${isPassword}&password=${password}`,
d => {
if (d[1].table) {
let location = `game/${d[1].table}`;
location += isPassword ? `?password=${d[1].password}` : ``;
GM_setValue('_oGameSrcFirstConf', location);
GM_setValue('_oGameSrcSecondConf', location);
GM_setValue('_oGameSrcCh', location);
document.location = location;
} else {
alert(d[1].msg || d[1]);
}
})
});
}
function rehostCustomGame(type, numPlayers, isPassword, password) {
$.get(`https://epicmafia.com/game/add/${type}?` +
`numplayers=${numPlayers}&add_password=${isPassword}&password=${password}`,
d => {
if (d[1].game_id) {
let location = `game/${d[1].game_id}`;
location += isPassword ? `?password=${d[1].password}` : ``;
GM_setValue('_oGameSrcFirstConf', location);
GM_setValue('_oGameSrcSecondConf', location);
GM_setValue('_oGameSrcCh', location);
document.location = location;
} else {
alert(d[1].msg || d[1]);
}
})
}
function insertAutoRefresh() {
let autoRefresh = GM_getValue('_oAutoRefreshBox') === undefined ? true : GM_getValue('_oAutoRefreshBox');
$(".ll-refresh").after(`<div id="_oAutoRefreshWrap" class="tt" data-title="Auto-refresh">
<input type="checkbox" id="_oAutoRefreshBox" ${autoRefresh ? "checked" : ""}/>
<label for="_oAutoRefreshBox"><i class="_oracle_icon"></i></label>
</div>`);
$("#_oAutoRefreshBox").change(e => {
autoRefresh = e.target.checked;
GM_setValue('_oAutoRefreshBox', autoRefresh);
});
setTimeout(function run() {
let isNeedRefresh = autoRefresh
&& $('.pagenav .grey.smallfont').text() === "Page 1";
let isJustRefresh = document.hasFocus()
&& (Date.now() - lastActivity < 5 * 60 * 1000)
&& $('.ll-gamelist.sel').length === 1;
let isGameSearchEnable = autoRefresh
&& (Date.now() - lastActivity < 30 * 60 * 1000)
&& !($("#_oGameSearchNone")[0].checked);
if (isNeedRefresh) {
if (isGameSearchEnable || isJustRefresh)
$('.icon-refresh').click();
if (isGameSearchEnable) {
searchGame();
}
}
setTimeout(run, 2200);
}, 10);
$(document).focus(() => {
if (autoRefresh) {
$('.icon-refresh').click();
}
});
}
function getOpenGames($_oGameSearchOption) {
let $openGames = [];
let isRankedGamesChecked = $('#_oGameRankedRedHeart')[0].checked;
let isCompetitionGamesChecked = $('#_oGameRankedGoldHeart')[0].checked;
let doNotSearch = !$_oGameSearchOption[0].checked;
if (doNotSearch) { // prepare
if (isRankedGamesChecked) {
$openGames = $('.gamerow.join > .gameoptions > .randimg > img[src="/images/lives.png"]').parent().parent().parent();
} else if (isCompetitionGamesChecked) {
let $goldLives = $('.gamerow.join > .gameoptions > .randimg > img[src="/images/goldlives.png"]').parent().parent().parent();
$openGames = isRankedGamesChecked ? $openGames.add($goldLives) : $goldLives;
} else {
$openGames = $('.gamerow.join > .gameoptions > .randimg > img[src="/images/lock.png"], .gamerow.join > .gameoptions > .randimg > img[src="/images/cam.png"]')
.parent().parent().parent();
$openGames = $openGames.add($('.gamerow.join > .gameoptions > .randimg:empty').parent().parent());
}
}
return $openGames;
}
function searchGame() {
let $_oGameSearchOption = $("input[name=_oGameSearchOption]:radio");
let $openGames = getOpenGames($_oGameSearchOption);
if ($openGames.length) {
searchGameBy();
}
function searchGameBy() {
let searchByGameType = $_oGameSearchOption[1].checked;
let searchByGameSetup = $_oGameSearchOption[2].checked;
let searchByHostId = $_oGameSearchOption[3].checked;
if (searchByGameType) {
let _oGameType = $('#_oGameTypeSelect').val();
openGameByType($openGames, _oGameType);
} else if (searchByGameSetup) {
let _oGameSetups = $('#_oGameSetupId').val().replace(/\s/g, '').split(',');
openGameBySetup($openGames, _oGameSetups);
} else if (searchByHostId) {
let _oGameCreators = $('#_oGameCreatorId').val().replace(/\s/g, '').split(',');
openGameByCreator($openGames, _oGameCreators);
}
}
}
function insertSearch() {
let insertHtml = `
<div>
<!--Header-->
<h2 class="separator vm">Search game when auto-refresh is on</h2>
<!--Settings-->
<input type="radio" id="_oGameSearchNone" name="_oGameSearchOption" checked>Don't search game<br>
<input type="radio" name="_oGameSearchOption" value="_oGameSearchByType">
<label for="_oGameTypeSelect">Game type:</label><br>
<select id="_oGameTypeSelect">
<option value="mafia">mafia</option>
<option value="dicewars">dicewars</option>
<option value="liarsdice">liarsdice</option>
<option value="acrotopia">acrotopia</option>
<option value="battlegrounds">battlegrounds</option>
<option value="battleships">battleships</option>
<option value="boggle">boggle</option>
<option value="cheat">cheat</option>
<option value="checkers">checkers</option>
<option value="chess">chess</option>
<option value="chinese">chinese</option>
<option value="concentration">concentration</option>
<option value="connect4">connect4</option>
<option value="crackcode">crackcode</option>
<option value="crazyeights">crazyeights</option>
<option value="dotsboxes">dotsboxes</option>
<option value="drawit">drawit</option>
<option value="ghost">ghost</option>
<option value="go">go</option>
<option value="hearts">hearts</option>
<option value="hex">hex</option>
<option value="jotto">jotto</option>
<option value="mancala">mancala</option>
<option value="ratscrew">ratscrew</option>
<option value="reversi">reversi</option>
<option value="snakes">snakes</option>
<option value="stackit">stackit</option>
<option value="texas">texas</option>
<option value="tira">tira</option>
</select><br>
<label for="_oGameRankedRedHeart"><img src="/images/lives.png">(Ranked)</label>
<input type="checkbox" id="_oGameRankedRedHeart" name="redHeart" value="redHeart"><br>
<label for="_oGameRankedGoldHeart"><img src="/images/goldlives.png">(Competition)</label>
<input type="checkbox" id="_oGameRankedGoldHeart" name="goldHeart" value="goldHeart"><br>
<input type="radio" name="_oGameSearchOption" value="_oGameSearchBySetup">
<label for="_oGameSetupId">Setup ids:</label><br>
<input id="_oGameSetupId" type="text" value="1375880"><br>
<input type="radio" name="_oGameSearchOption" value="_oGameSearchByCreator">
<label for="_oGameCreatorId">Creator ids:</label><br>
<input id="_oGameCreatorId" type="text" value="656967"><br>
</div>`;
$('#lobbyinfo_box').before(insertHtml);
$( "#_oGameSetupId" ).val(GM_getValue('_oGameSetupId') || '');
$( "#_oGameCreatorId" ).val(GM_getValue('_oGameCreatorId') || '');
$("#_oGameRankedRedHeart").prop("checked", GM_getValue('_oGameRankedRedHeart') || false);
$("#_oGameRankedGoldHeart").prop("checked", GM_getValue('_oGameRankedGoldHeart') || false);
$( "#_oGameSetupId" ).change((e) => {
GM_setValue('_oGameSetupId', e.target.value);
});
$( "#_oGameCreatorId" ).change((e) => {
GM_setValue('_oGameCreatorId', e.target.value);
});
$( "#_oGameRankedRedHeart" ).change((e) => {
GM_setValue('_oGameRankedRedHeart', e.target.checked);
});
$( "#_oGameRankedGoldHeart" ).change((e) => {
GM_setValue('_oGameRankedGoldHeart', e.target.checked);
});
}
function openGameByType($openGames, _oGameType) {
$openGames.each((_, row) => {
if (row.classList.contains(_oGameType + '_gamerow')) {
// let gameId = row.getAttribute('data-gid');
// document.location = `game/${gameId}`;
$(row).find('.sitdown > a')[0].click();
$('#_oGameSearchNone')[0].click();
return false;
}
});
}
async function openGameBySetup($gameRows, _oGameSetups) {
let rowsLength = $gameRows.length;
for (let i = 0; i < rowsLength; i++) {
let data = await _openGameBySetup($gameRows[i]);
let setup = data[1].data !== undefined ? data[1].data.match(/\/setup\/[0-9]+/)[0].split("/")[2] : '';
console.log(_oGameSetups, setup);
if (_oGameSetups.indexOf(setup) !== -1) {
console.log("found game = " + setup);
let gameId = $gameRows.eq(i).attr('data-gid');
document.location = `game/${gameId}`;
// $gameRows[i].querySelector('a').click();
// $gameRows[i].querySelector('.sitdown > a').click();
/*$gameRows.eq(i).find('.sitdown > a').each((_, e) => {
if (e.textContent === "Join game")
e.click();
});
$gameRows.eq(i).find('.sitdown > a').click();*/
$('#_oGameSearchNone')[0].click();
break;
}
}
}
function _openGameBySetup(gameRow) {
const gid = gameRow.getAttribute("data-gid");
return Promise.resolve($.get(`https://epicmafia.com/game/${gid}/info`));
}
function openGameByCreator($openGames, _oGameCreators) {
$openGames.find('.pull_right > .creator > a').each((_, creator) => {
let creatorId = creator.getAttribute('data-uid');
if (_oGameCreators.indexOf(creatorId) !== -1) {
$(creator).parents('.gamerow').find('.sitdown > a')[0].click();
$('#_oGameSearchNone')[0].click();
}
});
}