NOTICE: By continued use of this site you understand and agree to the binding Terms of Service and Privacy Policy.
// ==UserScript== // @name Flight Rising Helper // @namespace Flight Rising // @author Jonathon Braswell // @description A powerful, feature-rich toolbox for Flight Rising. // @include http://flightrising.com/* // @version 2.4 // @license MIT // ==/UserScript== /** * MIT License * Copyright (c) 2017 Jonathon Braswell * * Permission is hereby granted, free of charge, to any person obtaining a copy * of this software and associated documentation files (the "Software"), to deal * in the Software without restriction, including without limitation the rights * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell * copies of the Software, and to permit persons to whom the Software is * furnished to do so, subject to the following conditions: * * The above copyright notice and this permission notice shall be included in all * copies or substantial portions of the Software. * * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE * SOFTWARE. */ /* * Flight Rising Helper * @namespace FlightRisingHelper * @dependency jQuery (included on site) * @author Jonathon Braswell */ var FlightRisingHelper = (($) => { /** UTILITY FUNCTIONS **/ let /* * Send async AJAX POST request. * @function _request * @private * @param payload {object} Data payload to [path]. * @param path {string} Directory or URL path to send [payload] to. */ _request = (payload, path) => { // Simply return AJAX promise. return $.ajax({ type: 'POST' ,data: payload || {} ,url: path ,cache: false }); } /* * Extract a dragon's unique identifier from a given [card] element. * @function _extractDragonIdFromCard * @private * @param card {HTMLDomElement|jQuery Object} Element from which to extract a dragon's unique identifier. */ ,_extractDragonIdFromCard = (card) => { // Find the icon's anchor element and extract the unique identifier from its URL. let anchor = $(card).find('a[rel]') ,link = $(anchor).attr('rel') || '' ,id = (link.split('id=')[1] || '').trim(); return id; } /* * Extract a query string variable from the given [searchString]. * @function _getUrlVar * @private * @param key {string} Variable name to return value for. * @param (searchString) {string} Optional. Source query string for search. */ ,_getUrlVar = (key, searchString) => { // Determine whether we're trying to query the parameter or the current URL and grab the variables. let query = (typeof searchString !== 'undefined' ? searchString.split('?')[1] : window.location.search.substring(1)) ,vars = query.split('&'); // Iterate the variables array until we find what we're looking for, then return the found value. for (var i = 0; i < vars.length; i++) { let pair = vars[i].split("="); if (pair[0] == key) return pair[1]; } } /* * Invoke the site's "assay dragons" functionality and create representation in the current DOM. * @function _assayDragons * @private * @param firstDragonId {int} A dragon's unique identifier. * @param secondDragonId {int} A dragon's unique identifier. */ ,_assayDragons = (firstDragonId, secondDragonId) => { // Invoke the site's native [showNewbies] function if it exists. if (typeof showNewbies === 'function') showNewbies(); else return; // Construct a payload and send a request to the site's scrying handler. let payload = { id1: firstDragonId ,id2: secondDragonId }; _request(payload, 'includes/ol/scryer_bloodlines.php').done((result) => { // Upon request completion determine if the given dragons are capable of // "breeding", construct a box in the DOM to display the determination, and // cache this element's intended parent. let canMate = (result.toLowerCase().indexOf('success') > -1) ,canBreedBoxCss = { padding: '20px' ,margin: '20px' ,textAlign: 'center' ,fontSize: '15px' ,background: (canMate ? 'lightgreen' : 'lightpink') } ,nestBoxParent = $('.nestbox').parent(); // Apply style the box and prepend it to the intended parent. $('<div />').attr('id', 'can-breed').css(canBreedBoxCss).prependTo(nestBoxParent); // Place situational text in the display box. $('#can-breed').html('These dragons <b>are' + (canMate ? '</b>' : ' not</b>') + ' eligible to breed as assayed.'); // Place "Breed" button. if (canMate) { let breedLink = $('<a href="" class="redbutton anybutton" target="_blank"></a>').text('Breed'); $(breedLink).attr('href', 'http://flightrising.com/main.php?p=lair&tab=nest&dids=' + firstDragonId + ',' + secondDragonId); $('[onclick="showNewbies()"]').parent().css('width', '210px').append(breedLink); } }); } /* * Mark all visible checkboxes as the parameter value. * @function _markAllCheckboxes * @private * @param checkValue {bool} Value to mark all visible checkboxes. */ ,_markAllCheckboxes = (checkValue) => { // Mark all checkboxes as the boolean value of the parameter. $('input[type="checkbox"]:visible').prop('checked', checkValue); } /* * Recurrently perform the site's "exalt" functionality on an array of dragons. * @function _multiExalt * @private * @param index {int} Index of current iteration over the given [dragonIds] array. * @param dragonIds {array} Array of dragons' unique identifiers to "exalt." * @param callback {function} Callback function to invoke after the last array element's invocation. */ ,_multiExalt = (index, dragonIds, callback) => { // Determine the current iteration's dragon, whether this is the last iteration, and construct a payload. let currentDragonId = dragonIds[index] ,isLast = (currentDragonId === dragonIds[dragonIds.length - 1]) ,payload = { dragon: currentDragonId }; // Send a request to the site's dragon "exile" handler. _request(payload, 'includes/ol/exiledragon.php').done(() => { // Upon request completion invoke this function if the [currentDragonId] isn't the last element of [dragonIds]. // Othwerise invoke the given [callback] function. if (!isLast) _multiExalt(++index, dragonIds, callback); else callback(); }); } /* * Return stored "offspring" records from [localStorage]. * @function _loadOffspring * @private */ ,_loadOffspring = () => { // Ensure the data exists in [localStorage] and return JSON parsed data. if (typeof localStorage.fhr_offspring === 'undefined') localStorage.fhr_offspring = '[]'; return JSON.parse(localStorage.fhr_offspring); } /* * Update stored "offspring" in [localStorage] with the given [offpsring] array. * @function _updateOffspring * @private * @param offspring {array} Updated array of "offspring" records for the current [localStorage] instance. */ ,_updateOffspring = (offspring) => { // Simply save the given stringified [offspring] onto [localStorage]. localStorage.fhr_offspring = JSON.stringify(offspring); }; /** UTILITY CONSTANTS **/ const // Minified CSS for various functionalities. CSS = '#bonding{min-height:280px!important;}/* safari and chrome */ @-webkit-keyframes wiggle { 0% {-webkit-transform:rotate(4deg);} 50% {-webkit-transform:rotate(-4deg);} 100% {-webkit-transform:rotate(4deg);} } /* firefox */ @-moz-keyframes wiggle { 0% {-moz-transform:rotate(4deg);} 50% {-moz-transform:rotate(-4deg);} 100% {-moz-transform:rotate(4deg);} } /* anyone brave enough to implement the ideal method */ @keyframes wiggle { 0% {transform:rotate(4deg);} 50% {transform:rotate(-4deg);} 100% {transform:rotate(4deg);} }' // Icon data or location URLs for various functionalities. ,ICONS = { NEST: 'http://puu.sh/wJ9Rr/b889eca68f.png' ,HOARD: 'http://flightrising.com/images/layout/vault/vault_active.png' ,WRENCH: '' ,COATHANGER: '' ,SELECT_ALL: 'http://flightrising.com/images/layout/vault/select_all.png' ,SELECT_NONE: 'http://flightrising.com/images/layout/vault/select_none.png' } // Query variables. ,QUERY = { DID: _getUrlVar('did') ,DIDS: _getUrlVar('dids') ,DRAGON: _getUrlVar('dragon') ,ID: _getUrlVar('id') ,PAGE: _getUrlVar('p') ,TAB: _getUrlVar('tab') ,VIEW: _getUrlVar('view') }; /** SETUP/FUNCTIONALITY FUNCTIONS **/ let /* * Initialize functionality for the currency management system. * @function _setupCurrencyManagement * @private */ _setupCurrencyManagement = () => { // Apply CSS to the page. $('<style type="text/css" />').text(CSS).appendTo('head'); // Grab the "gem" and "treasure" currency containers, counts, and create deposit buttons for each. let gemContainer = $('#usertab').children().last() ,treasureContainer = $(gemContainer).prev() ,gemCount = parseInt($('#user_gems').text()) || 0 ,treasureCount = parseInt($('#user_treasure').text()) || 0 ,gemDepositBtn = $('<img />') .attr('id', 'gem-deposit') .attr('src', ICONS.HOARD) .attr('width', '15px') .attr('height', '15px') .click((event) => { // On "gem" deposit button click. // Construct a payload and send a request to the vault transaction handler. let payload = { trans: 'deposit' ,curr: 1 ,amount: gemCount } ,button = event.target; _request(payload, 'includes/ol/vault_withdep.php').done(() => { // Upon request completion show a temporary message to the user for validation. $('#user_gems').text('Deposited!'); setTimeout(() => { $('#user_gems').text('0'); $(button).remove(); }, 2000); }); } ) ,treasureDepositBtn = $('<img />') .attr('id', 'treasure-deposit') .attr('src', ICONS.HOARD) .attr('width', '15px') .attr('height', '15px') .click((event) => { // On "treasure" deposit button click. // Construct a payload and send a request to the vault transaction handler. let payload = { trans: 'deposit' ,curr: 0 ,amount: treasureCount } ,button = event.target; _request(payload, 'includes/ol/vault_withdep.php').done(() => { // Upon request completion show a temporary message to the user for validation. $('#user_treasure').text('Deposited!'); setTimeout(() => { $('#user_treasure').text('0'); $(button).remove(); }, 2000); }); } ); // If the user has "treasure" or "gem" currency available, add these buttons to the DOM. if (treasureCount) $(treasureDepositBtn).appendTo(treasureContainer); if (gemCount) $(gemDepositBtn).appendTo(gemContainer); } /* * Initialize functionality for familiar icon replacement with full art. * @function _setupFamiliarIconReplacement * @private */ ,_setupFamiliarIconReplacement = () => { $('img[src^="/images/cms/familiar"]').each((i, familiar) => { if (!familiar.src.includes('art')) { familiar.src = familiar.src.replace('familiar', 'familiar/art'); } }); } /* * Initialize functionality for bestiary familiar art coloration. * @function _setupBestiaryColoration * @private */ ,_setupBestiaryColoration = () => { $('img[src^="/images/cms/familiar"]').each((i, familiar) => { if (familiar.src.includes('gray')) { $(familiar).addClass('frh-locked'); } }); $('.frh-locked').hover(function () { let familiar = $(this); if ($(familiar).attr('src').includes('gray')) { $(familiar).attr('src', $(familiar).attr('src').replace('_gray', '')); } }, function () { let familiar = $(this); if (!$(familiar).attr('src').includes('gray')) { $(familiar).attr('src', $(familiar).attr('src').replace('.png', '_gray.png')); } }); } /* * Initialize functionality for the "progeny" functionality shortcuts. * @function _setupProgenyShortcuts * @private */ ,_setupProgenyShortcuts = () => { // New up an array to track selected dragon unique identifiers. let selectedDragonIds = []; $('.dragoncard .miniclue').click((event) => { // On dragon card's "gender" icon click. // Prevent bubble. event.preventDefault(), event.stopPropagation(); // Cache the clicked element and extract this dragon's unique identifier. let button = event.target ,id = _extractDragonIdFromCard($(button).closest('.dragoncard')); // Add this dragon's unique identifier to the list of selected dragons. selectedDragonIds.push(id); // Change the border appearance to show that this is a selected entry. $(button).closest('.dragoncard').css('border', '3px dashed #9f9a8f'); // If this is the second dragon selected, open a new window to the "progeny" page, then reset all the cards' border appearances. if(selectedDragonIds.length > 1) { let dIds = selectedDragonIds.join(','); window.open('http://flightrising.com/main.php?p=scrying&view=progeny&dids=' + dIds, '_blank'); selectedDragonIds = []; $('.dragoncard').each((i, card) => { $(card).css('border', '1px solid #9f9a8f'); }); } }); } /* * Initialize functionality for the "morphology" shortcuts. * @function _setupMorphologyShortcuts * @private */ ,_setupMorphologyShortcuts = () => { // Begin "dragon" tab functionality suite. if (QUERY.TAB === 'dragon') { // Cache the "morph" button container, create a button for the user, and append it to the container. let btnContainer = $('.main'); $('<a />') .addClass('morph-btn beigebutton thingbutton') .text('Morph') .css({ position: 'absolute' ,top: '46px' ,right: '360px' ,zIndex: '2' }) .attr('href', 'http://flightrising.com/main.php?p=scrying&view=morphintime&did=' + QUERY.DID) .attr('target', '_blank') .prependTo(btnContainer); } // Iterate all the dragon cards on the page. $('.dragoncard').each((i, card) => { // Extract this dragon's unique identifier, "morph" URL, and container. Create a wrench tool image. let id = _extractDragonIdFromCard(card) ,img = $('<img />') .attr('src', ICONS.WRENCH) .height('25') .width('25') .css('float', 'left') ,anchor = $('<a />') .attr('href', 'http://flightrising.com/main.php?p=scrying&view=morphintime&did=' + id) .attr('target', '_blank') ,container = $(card).children().last().children().last(); // Add the wrench tool image to the container. $(img).appendTo($(anchor).prependTo(container)); // Add CSS animation effects to this card on hover. $(card).hover( () => { $(card).find('.dragonthmb').css({ '-webkit-animation': 'wiggle 0.5s infinite' ,'-moz-animation': 'wiggle 0.5s infinite' ,'animation': 'wiggle 0.5s infinite' }); } ,() => { $(card).find('.dragonthmb').css({ '-webkit-animation': 'none' ,'-moz-animation': 'none' ,'animation': 'none' }); } ); }); } /* * Initialize functionality for the "bonding" shortcuts. * @function _setupBondingShortcuts * @private */ ,_setupBondingShortcuts = () => { // Create an invisible DOM element for storing a "bonding" action frame element. let bondBin = $('<div />') .attr('id', 'bond-bin') .css('display', 'none') .appendTo('body'); // Change the cursor style on the "famicon" button image. $('img[src$="famicon.png"]') .css('cursor', 'pointer') .click((event) => { // On "famicon" button click. // Cache this button and extract this dragon's URL. let button = event.target ,url = $(button).closest('.dragoncard').find('a[rel]').attr('href'); // Load the dragon's URL into the bond bin and activate the "bond" functionality. // Yes this ugly page scraping is the result of this site having no sense of APIs. :P $(bondBin).load(url, () => { $(bondBin).find('img[src$="button_bond.png"]').click(); $(bondBin).empty(); }); }); } /* * Initialize functionality for the "dressing room" shortcuts. * @function _setupDressingRoomShortcuts * @private */ ,_setupDressingRoomShortcuts = () => { // Define a scoped function for submitting a dragon's unique identifier to the "dressing room" functionality page. let dressingRoomFormSubmit = (dragonId) => { let input = $('<input id="dragon" name="dragon" type="text" />') ,form = $('<form action="http://www1.flightrising.com/dressing/outfit" method="POST" target="_blank"></form>'); $(form).hide().appendTo('body').append($(input).val(dragonId)).submit().remove(); }; // Begin "dragon" tab functionality suite. if (QUERY.TAB === 'dragon') { // Cache button container, create and append an action button to this container. let btnContainer = $('.main'); $('<a />') .addClass('dressing-room-btn beigebutton thingbutton') .text('Dress') .css({ position: 'absolute' ,top: '46px' ,right: '270px' ,zIndex: '2' }).prependTo(btnContainer) .click(() => { dressingRoomFormSubmit(QUERY.DID); }); // On "dress" button click: invoke [dressingRoomformSubmit] with the current page's dragon's unique identifier. } else if (QUERY.ID && !QUERY.TAB) { // All other tab/id combo suite. // Iterate each dragon card on the page. $('.dragoncard').each((i, card) => { // Create a "dress" button using this dragon's unique identifier. let id = _extractDragonIdFromCard(card) ,img = $('<img />') .attr('src', ICONS.COATHANGER) .css({ height: '15px' ,width: '25px' ,float: 'left' ,marginTop: '5px' ,marginLeft: '-8px' ,cursor: 'pointer' }) .click(() => { dressingRoomFormSubmit(id); }) ,container = $(card).children().last().children().last(); // Append it to this iteration's container. $(img).appendTo(container); }); } } /* * Initialize functionality for the "progeny" functionality page. * @function _setupInitializedProgeny * @private */ ,_setupInitializedProgeny = () => { // Only act if the query has unique identifiers. if (QUERY.DIDS) { // Grab the two query string identifier values. let firstId = QUERY.DIDS.split(',')[0] ,secondId = QUERY.DIDS.split(',')[1]; // Place them into the "progeny" form. $('#id10t1').val(firstId), $('#id10t2').val(secondId); // Invoke our [_assayDragons] function with these values. _assayDragons(firstId, secondId); } } /* * Initialize functionality for the "morphology" functionality page. * @function _setupInitializedMorphology * @private */ ,_setupInitializedMorphology = () => { // Set the form's input value to the current query's unique identifier and invoke the site's [AEGBAGA] (???) function to submit. let id = QUERY.DID; $('#greenranger').val(id), AEGBAGA(); setTimeout(() => { // Wait a bit and flip the age of the dragon to adult, then refres the current view by invoking the site's [lolRedRanger] (again, ???) function. $('#setage').val(1), lolRedRanger(); }, 1000); } /* * Initialize functionality for the " */ ,_setupInitializedNest = () => { if ((QUERY.TAB === 'nest') && QUERY.DIDS) { let firstId = QUERY.DIDS.split(',')[0] ,secondId = QUERY.DIDS.split(',')[1]; if ($('#dad').find('option[value="' + firstId + '"]').length) { console.log('first id is dad'); $('#dad').val(firstId).change(); } else if ($('#mom').find('option[value="' + firstId + '"]').length) { console.log('first id is mom'); $('#mom').val(firstId).change(); } if ($('#mom').find('option[value="' + secondId + '"]').length) { console.log('second id is mom'); $('#mom').val(secondId).change(); } else if ($('#dad').find('option[value="' + secondId + '"]').length) { console.log('second id is dad'); $('#dad').val(secondId).change(); } if (typeof showBabies === 'function') setTimeout(showBabies, 1000); } } /* * Initialize "hoard" functionality page. * @function _setupHoard * @private */ ,_setupHoard = () => { // Remove functionality we're replacing. $('#food').parent().remove(); // Redefine the site's [uncheckAll] and [checkAll] functions with our versions. uncheckAll = () => { _markAllCheckboxes(false); } ,checkAll = () => { _markAllCheckboxes(true); }; // Create check and uncheck buttons for pages that don't have them. let container = $('#invent').children().last() ,checkLeft = (QUERY.PAGE === 'hoard' ? '0' : '415px') ,uncheckLeft = (QUERY.PAGE === 'hoard' ? '53px' : '470px'); // Only act if the check and uncheck buttons we need don't exist. if (!$('img[onclick="checkAll(invent)"]:visible').length) { // Create an uncheck all button, add a click event handler, and append it to the DOM. $('<img />') .attr('src', ICONS.SELECT_NONE) .prependTo(container) .click(() => { _markAllCheckboxes(false); }) .css({ position: 'absolute' ,left: uncheckLeft ,top: '0' }) // Create a check all button, add a click event handler, and append it to the DOM. ,$('<img />') .attr('src', ICONS.SELECT_ALL) .prependTo(container) .click(() => { _markAllCheckboxes(true); }) .css({ position: 'absolute' ,left: checkLeft ,top: '0' }); } // Begin "vault" page functionality suite. if (QUERY.PAGE === 'vault') { // Grab "treausure" and "gem" currency counts. let treasureCount = $('#nodiving').text().trim() ,gemCount = $('#bling').text().trim(); // Create "treasure" withdraw button. $('img[src="../images/layout/icon_treasure.png"]') .css('cursor', 'pointer') .unbind() .click(() => { // On "treasure" withdraw button click. // Construct a payload and send a request to the vault transaction handler. let payload = { trans: 'withdraw' ,curr: 0 ,amount: treasureCount }; _request(payload, 'includes/ol/vault_withdep.php').done(() => { // Upon request completion show a temporary message to the user for validation. $('#nodiving').text('Withdrawn!'); setTimeout(() => { $('#nodiving').text('0'); }, 2000); }); }) // Create "gem" withdraw button. ,$('img[src="../images/layout/icon_gems.png"]') .css('cursor', 'pointer') .unbind() .click(() => { // On "gem" withdraw button click. // Construct a payload and send a request to the vault transaction handler. let payload = { trans: 'withdraw' ,curr: 1 ,amount: gemCount }; _request(payload, 'includes/ol/vault_withdep.php').done(() => { // Upon request completion show a temporary message to the user for validation. $('#bling').text('Withdrawn!'); setTimeout(() => { $('#bling').text('0'); }, 2000); }); }); } } /* * Initialize the "multiple-exalt" tool and custom functionality. * @function _setupMultiExaltTool * @private */ ,_setupMultiExaltTool = () => { // Init a selecting flag, an array to hold dragon identifiers. let isSelecting = false ,exaltees = [] ,btnContainer = $('#tunak').prev().children().last() ,marquee = $('<div />') // Create an element to act as a marquee to relay info to the user. .css('text-align', 'center') .html('<h2>Select the dragons that you wish to exalt and then click the Exalt button above.</h2>') .hide() ,marqueePrev = $('form').prev().after(marquee) // Apply it to the DOM. ,handleExalteeSelection = (event) => { // Scoped function to handle dragon selection. // Prevent bubble. event.preventDefault(), event.stopPropagation(); // Grab the current card and extract its dragon's unique identifier. let card = $(event.target).closest('.dragoncard') ,id = _extractDragonIdFromCard(card); // If selected id is in the [exaltees] array, remove it. Otherwise add it. if (exaltees.indexOf(id) > -1) { exaltees.splice(exaltees.indexOf(id), 1); $(card).css('border', '1px solid #9f9a8f'); } else { exaltees.push(id); $(card).css('border', '3px dashed #9f9a8f'); } }; // Create an action button to toggle selection status/submission for "multiple-exalt" custom functionality. $('<button />') .addClass('multi-exalt-btn beigebutton thingbutton') .text('Multi-Exalt') .css({ position: 'absolute' ,top: '7px' ,left: '-125px' }).prependTo(btnContainer) .click((event) => { // On "Multi-Exalt" button click. // Cache action button. let button = event.target; // If not already selecting, toggle selecting and init click event on all dragon cards. if (!isSelecting) { isSelecting = true; $(button).removeClass('thingbutton beigebutton').addClass('redbutton anybutton').text('Exalt'); $('.dragoncard').click(handleExalteeSelection); $(marquee).show(); } else { // If selecting, remove click event handler and reset selection form. isSelecting = false; $('.dragoncard').unbind('click', handleExalteeSelection); $(marquee).hide(); $('.dragoncard').each((i, card) => { $(card).css('border', '1px solid #9f9a8f'); }); $(button).removeClass('redbutton anybutton').addClass('beigebutton thingbutton').text('Multi-Exalt'); // If submitting and there are dragons to "exalt". if (exaltees.length) { // Confirm the user wants to "exalt" all the selected dragons. $(button).text('Exalting...'); let isSingle = (exaltees.length === 1) ,confirmation = confirm('Are you sure you wish to exalt ' + (isSingle ? 'this' : 'these') + ' ' + exaltees.length + ' dragon' + (isSingle ? '': 's') + '?'); if (confirmation) { // If user confirms, invoke our [_multiExalt] function starting at the 0th index of [exaltees] with callback of reloading the page. _multiExalt(0, exaltees, () => { window.location.reload(); }); } else { // If the user cancels, reset the [exaltees] array, and reset the GUI. exaltees = []; $(button).removeClass('redbutton anybutton').addClass('beigebutton thingbutton').text('Multi-Exalt'); } } } }); } /* * Initialize the "saved offspring" tool and custom functionality. * @function _setupSavedOffspringTool * @private */ ,_setupSavedOffspringTool = () => { // Gather saved offspring from [localStorage] and create a box to show the offspring in. let savedOffspring = _loadOffspring() ,savedBox = $('<div class="saved-box" />') .css({ background: '#dedacf', borderRadius: '5px', border: '1px solid #000', textAlign: 'center', width: '100%', marginTop: '20px' }).html('<h2>Saved Offspring</h2>') // Scoped function to draw "saved offspring" functionality display box in the DOM. ,drawSavedOffspring = (saveBox) => { // Ensure any previous instance is removed--gracefully. $('.saved-box').empty().html('<h2>Saved Offspring</h2>').remove(); // Iterate each saved "offspring." $(savedOffspring).each((i, offspring) => { // Find the offspring URL that matches current URL. if (offspring.url === window.location.search) { // Append the drawn box to the DOM. $('input[value="Preview"]').parent().after(saveBox); // Only act if there are offspring. if (offspring.offspring.length) { // Iterate each offspring. $(offspring.offspring).each((i, image) => { // Create an image of the saved offspring. let img = $('<img src="' + image + '" />') .click(() => { // On offspring icon click. // Confirm the user wants to remove saved offspring. let confirmation = confirm('Are you sure you wish to remove this saved offspring?'); // If confirmed, remove this offspring and update the save. Then redraw the box. if (confirmation) { offspring.offspring.splice(offspring.offspring.indexOf(image), 1); _updateOffspring(savedOffspring); drawSavedOffspring(saveBox); } }); // Append the image to the saved box. $(savedBox).append(img); }); } else { // No offspring detected. $(saveBox).append('<b>No offspring saved for this pair.</b><br/><br/>'); } return; } }); } /* * Save clicked dragon as offspring for pair on the current page. * @function offspringSavingHandler * @private * @param event {object} Click event object. */ ,offspringSavingHandler = (event) => { // Prevent bubble. event.preventDefault(), event.stopPropagation(); // Cache clicked element and confirmation from user. let image = event.target ,confirmation = confirm('Do you wish to save this offspring possibility for future reference?'); // Only act if the user confirmed. if (confirmation) { // Declare flag. let thisPageExists = false; // Iterate each saved "offspring." $(savedOffspring).each((i, offspring) => { // If this offspring matches the current page, flip the flag. if (offspring.url === window.location.search) { thisPageExists = true; return; } }); // If this page doesn't exist in the offspring, save it to the offspring array. if (!thisPageExists) { let offspring = { url: window.location.search, offspring: [$(image).attr('src')] }; savedOffspring.push(offspring); _updateOffspring(savedOffspring); } else { // Otherwise update the existing entry for this offspring. $(savedOffspring).each((i, offspring) => { if (offspring.url === window.location.search) { if (typeof offspring.offspring === 'undefined') offspring.offspring = []; if (offspring.offspring.indexOf($(image).attr('src')) === -1) { offspring.offspring.push($(image).attr('src')); _updateOffspring(savedOffspring); } return; } }); } // Redraw the "saved offspring" display box. drawSavedOffspring(savedBox); } } // If there are "saved offspring" entries, draw the "saved offspring" display box. if (savedOffspring.length) drawSavedOffspring(savedBox); // Scoped function to initialize the click event handler on the preview button on this page. let initializer = () => { $('#preview img') .unbind('click') .click(offspringSavingHandler); }; // Register [initializer] to doc ready and AJAX.oncomplete. $(document).ready(initializer).ajaxComplete(initializer); } /* * Initialize the "setup lineage" tool and custom functionality. */ ,_setupLineagePreviewTool = () => { // If there is no "lineage preview" display box, draw one in the DOM. if (!$('#lineage-preview').length) { $('<div id="lineage-preview" style="position: fixed; display: none; border: 1px solid #000; background: #dad6c8; border-radius: 1em; padding:20px;"></div>').appendTo('body'); } // Scoped function to handle "ancestor" name mouse hover. let hoverHandler = (event) => { // Cache clicked name. let link = event.target; // If the "lineage preview" display box is visible, hide it. if ($('#lineage-preview').is(':visible')) { $('#lineage-preview').hide(); } else { // Otherwise grab the link's source and grab the dragon's unique identifier from this source. let href = $(link).attr('href') || $(link).attr('src') ,dId = _getUrlVar('did', href) || _getUrlVar('dragon', href); // Load dragon image from their page into the "lineage preview" display box. $('#lineage-preview') .load('http://flightrising.com/main.php?dragon=' + dId + ' #dragbuttons', () => { if (!$('#lineage-preview').children().length) { $('#lineage-preview') .load('http://flightrising.com/main.php?dragon=' + dId + ' span[style="width:350px; height:350px; margin-bottom:20px; margin-top:28px; margin-left:15px; display:inline-block;"]', () => { let left = (event.clientX - 420) + 'px' ,top = (event.clientY - 420) + 'px' ,div = document.getElementById('lineage-preview'); div.style.left = left; div.style.top = top; $('#lineage-preview').show(); }); } else { let left = (event.clientX - 420) + 'px' ,top = (event.clientY - 420) + 'px' ,div = document.getElementById('lineage-preview'); div.style.left = left; div.style.top = top; $('#lineage-preview').show(); } }); } }; // Apply the [hoverHandler] function as hover event handlers to "ancestor" names on the page. $('a[style="color:#000; font-weight:bold; text-decoration:underline;"]').on('mouseenter mouseleave', hoverHandler); $('.itemicon.ah-listing-icon img').on('mouseenter mouseleave', hoverHandler); $('#lineage-preview').hover((event) => { $(event.target).hide(); }); } /* * Replaces breeding cooldown icons with their numerical values. * @function _replaceBreedingCooldownIcons * @private */ ,_replaceBreedingCooldownIcons = () => { let eggElements = $('a[title^="This dragon will not be available to breed"]'); $(eggElements).each((i, element) => { let cooldown = $(element).attr('title').replace(/\D/g, ''); $(element).find('img').replaceWith('<span style="font-size: 18px; color: #731d08; font-weight: bold;">' + cooldown + '</span>'); }); }; /** INITIALIZATION **/ /* * Initialize script functionality! * @function _init * @private */ let _init = () => { // Initialize currency management on all pages. _setupCurrencyManagement(); // Switch on current page. switch (QUERY.PAGE) { // Initialize hoard custom functionalities and set AJAX.oncomplete handler. case 'hoard': case 'vault': _setupHoard(), $(document).ajaxComplete(_setupHoard), $(document).ajaxComplete(_setupFamiliarIconReplacement); break; // Initialize "multiple-exalt" and "bonding" shortcut tools. case 'lair': _setupMultiExaltTool(), _setupBondingShortcuts(), _setupInitializedNest(), _replaceBreedingCooldownIcons(); // Initialize "progeny" and "morphology" shortcut tools. case 'view': _setupProgenyShortcuts(), _setupMorphologyShortcuts(); // Begin "dragon" tab, "" dragon, and AH Dragons functionality suite. // Initialize the "lineage preview" tool and the "dressing room" shortcuts tool. if (QUERY.TAB === 'dragon' || QUERY.DRAGON !== '' || window.location.pathname === '/auction-house/buy/realm/dragons') _setupLineagePreviewTool(); _setupDressingRoomShortcuts(); break; // Initialize "initialized progeny, morphology" and "saved offspring" tools. case 'scrying': switch (QUERY.VIEW) { case 'progeny': _setupInitializedProgeny(), _setupSavedOffspringTool(); break; case 'morphintime': _setupInitializedMorphology(); break; }; break; // Initialize "bestiary familiar coloration" tool. case 'bestiary': _setupBestiaryColoration(); break; }; }; return { /* * Initialize! * @method init * @memberof FlightRisingHelper * @public */ init: _init }; })($); // Attach FRH.init as doc.ready event handler. $(document).ready(FlightRisingHelper.init); // EOF