Skip to content

Instantly share code, notes, and snippets.

@Jed-Giblin
Last active March 3, 2023 15:21
Show Gist options
  • Save Jed-Giblin/8ec30d2d721cca03197bdbb8a5199660 to your computer and use it in GitHub Desktop.
Save Jed-Giblin/8ec30d2d721cca03197bdbb8a5199660 to your computer and use it in GitHub Desktop.
Tampermonkey script to colate and document commander deck colors for your profile in moxfield
// ==UserScript==
// @name New Userscript
// @namespace http://tampermonkey.net/
// @version 0.1
// @description try to take over the world!
// @author You
// @match https://www.moxfield.com/decks/personal
// @icon https://www.google.com/s2/favicons?sz=64&domain=moxfield.com
// @grant GM.xmlHttpRequest
// ==/UserScript==
const CACHE_PREFIX = 'cc-challenge';
(async () => {
'use strict';
async function getCommanderColorIdentity(deckUri) {
console.log(`Checking cache for ${deckUri}`);
let cachedValue = readCache( deckUri );
if ( cachedValue === null ) {
console.log(`No value in cache for ${deckUri}`);
let apiValue = await getColorIdentityFromApi( deckUri );
console.log(`Got a value of ${apiValue}`);
writeCache( deckUri, apiValue );
cachedValue = apiValue;
}
return cachedValue;
}
async function getColorIdentityFromApi(deckUri) {
return new Promise( (resolve, reject) => {
GM.xmlHttpRequest ({
method: "GET",
url: `https://api2.moxfield.com/v2/decks/all/${deckUri}`,
headers: {
"Content-Type": "application/json"
},
onload: function (response) {
let ident = [];
let responseJson = JSON.parse( response.responseText );
if ( responseJson.hasOwnProperty('commanders') ) {
console.log(responseJson);
Object.values(responseJson.commanders).forEach((c) => {
ident.push( c.card.color_identity );
});
resolve(ident.flat());
} else {
resolve( ident );
}
}
});
});
}
function readDecks() {
let rows = [...document.getElementsByTagName('tr')];
let commanderDecks = rows.filter( (row) => {
return row.cells[1].textContent === "Commander / EDH";
});
return commanderDecks.map( r => r.cells[0].querySelector('a') );
}
function readCache(key) {
let existingValue = localStorage.getItem( `${CACHE_PREFIX}-${key}` );
return JSON.parse( existingValue );
}
function writeCache(key, val) {
localStorage.setItem( `${CACHE_PREFIX}-${key}`, JSON.stringify(val))
}
let numAttempts = 0;
let masterDict = {
'W': { 'name': 'Mono white', decks:[], img: 'https://upload.wikimedia.org/wikipedia/commons/thumb/0/0f/MTG_%28W%29.svg/512px-MTG_%28W%29.svg.png?20180803212526'},
'U': { 'name': 'Mono blue', decks: [], img: 'https://upload.wikimedia.org/wikipedia/commons/thumb/3/34/MTG_%28U%29.svg/512px-MTG_%28U%29.svg.png?20180803212522'},
'B': { 'name': 'Mono black', decks: [], img: 'https://upload.wikimedia.org/wikipedia/commons/thumb/e/e4/MTG_%28B%29.svg/512px-MTG_%28B%29.svg.png?20180803212429'},
'R': { 'name': 'Mono red', decks: [], img: 'https://upload.wikimedia.org/wikipedia/commons/thumb/a/ad/MTG_%28R%29.svg/512px-MTG_%28R%29.svg.png?20180803212503'},
'G': { 'name': 'Mono green', decks: [], img: 'https://upload.wikimedia.org/wikipedia/commons/thumb/9/96/MTG_%28G%29.svg/512px-MTG_%28G%29.svg.png?20180803212451'},
// Pairs 10
// White + x (4)
'WU': { 'name': 'Azorius', decks: [], img: 'https://upload.wikimedia.org/wikipedia/commons/thumb/5/54/MTG_%28WU%29.svg/512px-MTG_%28WU%29.svg.png?20180803212536'},
'WB': { 'name': 'Orzhov', decks: [], img: 'https://upload.wikimedia.org/wikipedia/commons/thumb/1/11/MTG_%28WB%29.svg/512px-MTG_%28WB%29.svg.png?20180803212527'},
'WR': { 'name': 'Boros', decks: [], img: 'https://upload.wikimedia.org/wikipedia/commons/thumb/b/bb/MTG_%28RW%29.svg/512px-MTG_%28RW%29.svg.png?20180803212514'},
'WG': { 'name': 'Selesnya', decks: [], img: 'https://upload.wikimedia.org/wikipedia/commons/thumb/8/8b/MTG_%28GW%29.svg/512px-MTG_%28GW%29.svg.png?20180803212451'},
// Blue + x (3)
'UB': { 'name': 'Dimir', decks: [], img: 'https://upload.wikimedia.org/wikipedia/commons/thumb/2/25/MTG_%28UB%29.svg/512px-MTG_%28UB%29.svg.png?20180803212524'},
'UR': { 'name': 'Izzet', decks: [], img: 'https://upload.wikimedia.org/wikipedia/commons/thumb/8/8a/MTG_%28UR%29.svg/512px-MTG_%28UR%29.svg.png?20180803212524'},
'UG': { name: 'Simic', decks: [], img: 'https://upload.wikimedia.org/wikipedia/commons/thumb/e/e4/MTG_(GU).svg/512px-MTG_(GU).svg.png?20180803213312'},
// Black + x (2 )
'BR': { name: 'Rakdos', decks: [], img: 'https://upload.wikimedia.org/wikipedia/commons/thumb/c/c1/MTG_%28BR%29.svg/512px-MTG_%28BR%29.svg.png?20180803212440'},
'BG': { name: 'Golgari', decks: [], img: 'https://upload.wikimedia.org/wikipedia/commons/thumb/2/2a/MTG_%28BG%29.svg/512px-MTG_%28BG%29.svg.png?20180803212430'},
// Red + x ( 1)
'RG': { name: 'Gruul', decks: [], img: 'https://upload.wikimedia.org/wikipedia/commons/thumb/6/6b/MTG_%28RG%29.svg/512px-MTG_%28RG%29.svg.png?20180803212513'},
// Shards / Wedges 10
'WUB': { name: 'Esper', decks: [], img: 'https://cards.scryfall.io/large/front/a/3/a3d70a07-8d91-462c-aa96-901cc9a81531.jpg?1562707353'},
'WUR': { name: 'Jeskai', decks: [], img: 'https://cards.scryfall.io/large/front/d/4/d4c0fcdb-a312-4976-bcfb-73930c0a556c.jpg?1673148790'},
'WUG': { name: 'Bant', decks: [], img: 'https://cards.scryfall.io/large/front/7/9/7906b1e8-3049-4f4e-b1fd-b2547e7079f0.jpg?1673148361'},
'WBR': { name: 'Mardu', decks: [], img: 'https://cards.scryfall.io/large/front/c/3/c35fb7a3-7c7f-4470-b1ad-5b8709a608e6.jpg?1562793176'},
'WBG': { name: 'Abzan', decks: [], img: 'https://cards.scryfall.io/large/front/b/d/bd5b386e-e71e-492d-8456-6cd178ef5b3d.jpg?1673305419'},
'WRG': { name: 'Naya', decks: [], img: 'https://cards.scryfall.io/large/front/c/e/ce1d6f9b-2ca8-48b8-b4a6-46a61c7c3905.jpg?1673305611'},
'UBR': { name: 'Grixis', decks: [], img: 'https://cards.scryfall.io/large/front/2/0/20332354-a294-4448-a66f-533e6b546df6.jpg?1562901766'},
'UBG': { name: 'Sultai', decks: [], img: 'https://cards.scryfall.io/large/front/7/2/72af6c1f-33a0-4e02-95b7-74ecaa6a6d87.jpg?1673305662'},
'URG': { name: 'Temur', decks: [], img: 'https://cards.scryfall.io/large/front/0/0/009c5aac-194f-4b95-8f69-cc74aa15c3a5.jpg?1591321473'},
'BRG': { name: 'Jund', decks: [], img: 'https://cards.scryfall.io/large/front/d/f/df91979c-fb60-433b-8780-366ce1effd9c.jpg?1562942155'},
// 4 colors 5
'WUBR': { name: 'Yore', decks: [], img: 'https://cards.scryfall.io/large/front/0/2/0247d76c-dee4-4d1b-9c25-faa56171541e.jpg?1593272858'},
'UBRG': { name: 'Glint', decks: [], img: 'https://cards.scryfall.io/normal/front/a/4/a4f00d2b-d616-455a-bf9b-5bd82b690121.jpg?1673305536'},
'WBRG': { name: 'Dune', decks: [], img: 'https://cards.scryfall.io/normal/front/1/5/15b4ee44-28c4-4a39-9c06-aca43787954f.jpg?1593272643'},
'WURG': { name: 'Ink', decks: [], img: 'https://cards.scryfall.io/normal/front/d/2/d29bb432-1ca6-4305-b6d3-d7afe06c6c69.jpg?1593272693'},
'WUBG': { name: 'Witch', decks: [], img: 'https://cards.scryfall.io/normal/front/1/8/185d56ab-b7f2-4eb2-9540-db8f2dc9b436.jpg?1593272845'},
// All
'WUBRG': { name: '5c', decks: [], img: ''},
'C': { name: 'Colorless', decks: [], img: 'https://upload.wikimedia.org/wikipedia/commons/thumb/c/cc/MTG_%28C%29.svg/512px-MTG_%28C%29.svg.png?20180803212441'}
}
let sortOrder = ['w','u','b','r','g'];
let buildPage = function() {
let newFolderButton = [...document.getElementsByTagName('a')].find( a => a.text === "New Folder" );
console.log(newFolderButton);
let ccButton = document.createElement('a');
ccButton.href = '#';
ccButton.text = 'Show Commander Challenge';
newFolderButton.parentNode.append( ccButton );
let body = document.getElementsByTagName('body')[0];
let form = document.createElement('form');
form.id = 'frm-cc';
let modal = document.createElement('div');
let modalDialog = document.createElement('div');
let modalContent = document.createElement('div');
let modalHeader = document.createElement('div');
let modalBody = document.createElement('div');
let modalFooter = document.createElement('div');
let modalBackdrop = document.createElement('div');
// Setup the modalBackdrop
modalBackdrop.classList = 'modal-backdrop fade show d-block';
modalBackdrop.id = 'cc-backdrop';
modal.id = 'cc-modal';
modal.role = 'dialog';
// Setup the Header
let headerText = document.createElement('h5');
headerText.textContent = 'Commander Challenge Data'
headerText.classList = 'modal-title'
let headerButton = document.createElement('button');
headerButton.classList = 'btn-close'
headerButton.id = 'btn-close-cc';
headerButton.onclick = function(e) {
e.preventDefault();
document.getElementById('frm-cc').remove();
}
headerButton.type = 'button'
let headerSpan = document.createElement('span');
// Assemble the header
modalHeader.classList = 'modal-header';
modalBody.classList = 'modal-body text-center';
modalFooter.classList = 'modal-footer';
headerButton.append( headerSpan );
modalHeader.append( headerText );
modalHeader.append( headerButton );
// Assemble the body
let sortedKeys = Object.keys(masterDict).sort( (a,b) => {
if ( b.length > a.length ) {
return -1;
}
return sortOrder.indexOf( a.toLowerCase() ) - sortOrder.indexOf( b.toLowerCase() )
});
let dvSwitch = document.createElement('div');
dvSwitch.classList = 'deckview-switchmodal';
sortedKeys.forEach( (color) => {
let def = masterDict[color];
// Create a div for the entire row
let colorRow = document.createElement('div');
colorRow.classList = 'd-flex ms-3 mb-3';
// Create a div for the image
let colorImageHolder = document.createElement('div');
colorImageHolder.classList = 'flex-shrink-0 text-center me-4'
colorImageHolder.style.position = 'realtive;'
// Create an img
let colorImage = document.createElement('img');
colorImage.classList = 'img-card cursor-pointer'
colorImage.width = '130';
colorImage.src = def.img;
colorImageHolder.append(colorImage );
//Create a div to hold the text
let textHolder = document.createElement('div');
textHolder.classList = 'flex-grow-1 text-start';
// Create a title text
let titleText = document.createElement('h5');
titleText.classList = 'h4 mt-2 mb-1 d-flex align-items-center';
titleText.textContent = `${def.name} (${color})`;
textHolder.append( titleText );
def.decks.forEach( (deckName) => {
let deckNameRow = document.createElement('div');
let deckNameSpan = document.createElement('span');
deckNameSpan.classList = 'text-capitalize d-inline-block'
deckNameSpan.textContent = deckName;
deckNameRow.append( deckNameSpan );
textHolder.append(deckNameRow);
});
colorRow.append( colorImageHolder );
colorRow.append( textHolder );
modalBody.append( colorRow );
});
// Assemble the content
modalContent.classList = 'modal-content'
modalContent.append( modalHeader );
modalContent.append( modalBody );
modalContent.append( modalFooter );
// Assmble the dialog
modalDialog.classList = 'modal-dialog'
modalDialog.append( modalContent);
// Assemble the modal
modal.classList = 'modal zoom show d-block'
modal.append(modalDialog);
//Add the backdrop and the modal to the body
ccButton.onclick = function(e) {
e.preventDefault();
form.append( modalBackdrop );
form.append( modal );
body.append( form );
};
}
async function getData() {
var decks = readDecks();
if (decks.length > 0) {
decks.forEach( async (d) => {
let parts = d.href.split('/');
let deckId = parts[ parts.length -1 ];
let colorIdent = await getCommanderColorIdentity(deckId);
if ( colorIdent.length > 0 ) {
let colorSet = new Set(colorIdent);
let sortedColors = Array.from(colorSet).flat().sort( (a,b) => {
return sortOrder.indexOf( a.toLowerCase() ) - sortOrder.indexOf( b.toLowerCase() );
});
let colorId = sortedColors.join('');
masterDict[ colorId ].decks.push( d.innerText );
}
});
setTimeout(() => buildPage(), 2000);
//cc.modal();
//body.append( modalBackdrop );
//body.append( modal );
} else {
numAttempts++;
if (numAttempts >= 15) {
console.warn('Giving up after 34 attempts. Could not find: ');
} else {
setTimeout(getData, 250 * Math.pow(1.1, numAttempts));
}
}
}
await getData();
// Your code here...
})();
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment