MiloticScale / Flight Rising Helper

// ==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