diff --git a/plugins/ap-list.user.js b/plugins/ap-list.user.js new file mode 100644 index 00000000..f2af503a --- /dev/null +++ b/plugins/ap-list.user.js @@ -0,0 +1,399 @@ +// ==UserScript== +// @id iitc-plugin-ap-list@xelio +// @name iitc: AP List +// @version 0.3 +// @namespace https://github.com/breunigs/ingress-intel-total-conversion +// @updateURL https://raw.github.com/breunigs/ingress-intel-total-conversion/gh-pages/plugins/ap-list.user.js +// @downloadURL https://raw.github.com/breunigs/ingress-intel-total-conversion/gh-pages/plugins/ap-list.user.js +// @description List top 10 portals by AP of either faction. Hover over AP will show breakdown of AP. Click on portal name will select the portal. Double click on portal name will zoom to and select portal. Portals before zooming in will be cached and count in top portal. They will be cleared if you click the "↻ R" or move out of the area. +// @include https://www.ingress.com/intel* +// @match https://www.ingress.com/intel* +// ==/UserScript== + +function wrapper() { +// ensure plugin framework is there, even if iitc is not yet loaded +if(typeof window.plugin !== 'function') window.plugin = function() {}; + + +// PLUGIN START //////////////////////////////////////////////////////// + +// use own namespace for plugin +window.plugin.apList = function() { +}; + +window.plugin.apList.cachedPortals = {}; +window.plugin.apList.SIDE_FRIENDLY = 0; +window.plugin.apList.SIDE_ENEMY = 1; +window.plugin.apList.displaySide = window.plugin.apList.SIDE_ENEMY; +window.plugin.apList.sides = new Array(2); +window.plugin.apList.sortedPortals = new Array(2); +window.plugin.apList.playerApGainFunc = new Array(2); + +window.plugin.apList.topMaxCount = 10; +window.plugin.apList.sideLabelClass = {}; + +window.plugin.apList.useCachedPortals = false; +window.plugin.apList.cacheBounds; +window.plugin.apList.cacheActiveZoomLevel; + + +window.plugin.apList.handleUpdate = function() { + if(!requests.isLastRequest('getThinnedEntitiesV2')) return; + plugin.apList.updateSortedPortals(); + plugin.apList.updatePortalTable(plugin.apList.displaySide); +} + +// Generate html table from top portals +window.plugin.apList.updatePortalTable = function(side) { + var content = ''; + for(var i = 0; i < plugin.apList.topMaxCount; i++) { + var portal = plugin.apList.sortedPortals[side][i]; + content += '' + + '' + + '' + + ''; + } + content += "
' + + (portal ? plugin.apList.getPortalLink(portal) : ' ') + + '' + + (portal ? plugin.apList.getPortalApText(portal) : ' ') + + '
"; + $('div#ap-list-table').html(content); +} + +// Combine title and test +window.plugin.apList.getPortalApText = function(portal) { + var title = plugin.apList.getPortalApTitle(portal); + return '
' + digits(portal.playerApGain.totalAp) + '
'; +} + +// Friendly portal will get resonator upgrade list, enemy +// portal will get ap breakdown +window.plugin.apList.getPortalApTitle = function(portal) { + var t; + var playerApGain = portal.playerApGain; + if(plugin.apList.portalSide(portal) === plugin.apList.SIDE_FRIENDLY) { + t = 'Deploy & Upgrade\n'; + for(var i = 0; i < playerApGain.upgradedReso.length; i++) { + var reso = playerApGain.upgradedReso[i]; + var apGain = (reso.level === 0) ? DEPLOY_RESONATOR : UPGRADE_ANOTHERS_RESONATOR; + t += 'Resonator on ' + OCTANTS[reso.slot] + '\t' + reso.level + '->' + + reso.newLevel + '\t= ' + apGain + '\n'; + } + t += 'Sum: ' + digits(playerApGain.totalAp) + ' AP'; + } else { + t = 'Destroy & Capture:\n' + + 'Destroy AP\t=\t' + digits(playerApGain.destroyAp) + '\n' + + 'Capture AP\t=\t' + digits(playerApGain.captureAp) + '\n' + + 'Sum: ' + digits(playerApGain.totalAp) + ' AP'; + } + return t; +} + +// portal link - single click: select portal +// double click: zoom to and select portal +// hover: show address +window.plugin.apList.getPortalLink = function(portal) { + var latlng = [portal.locationE6.latE6/1E6, portal.locationE6.lngE6/1E6].join(); + var jsSingleClick = 'window.renderPortalDetails(\''+portal.guid+'\');return false'; + var jsDoubleClick = 'window.zoomToAndShowPortal(\''+portal.guid+'\', ['+latlng+']);return false'; + var perma = 'https://ingress.com/intel?latE6='+portal.locationE6.latE6 + +'&lngE6='+portal.locationE6.lngE6+'&z=17&pguid='+portal.guid; + //Use Jquery to create the link, which escape characters in TITLE and ADDRESS of portal + var a = $('',{ + "class": 'help', + text: portal.portalV2.descriptiveText.TITLE, + title: portal.portalV2.descriptiveText.ADDRESS, + href: perma, + onClick: jsSingleClick, + onDblClick: jsDoubleClick + })[0].outerHTML; + var div = '
'+a+'
'; + return div; +} + +// Loop through portals and get playerApGain, then put in sortedPortals by side and sort them by AP. +window.plugin.apList.updateSortedPortals = function() { + plugin.apList.sortedPortals[plugin.apList.SIDE_FRIENDLY] = new Array(); + plugin.apList.sortedPortals[plugin.apList.SIDE_ENEMY] = new Array(); + + // Make a backup of cachedPortals + // If cache is not enabled, empty cachedPortals. In following + // "$.each" loop, the backup portal will copy back into + // cachedPortals if it exist in "window.portals"" and didn't change.' + var oldcachedPortal = $.extend({},plugin.apList.cachedPortals); + if(!plugin.apList.useCachedPortals) + plugin.apList.cachedPortals = {}; + + $.each(window.portals, function(key, value) { + if(getTypeByGuid(key) !== TYPE_PORTAL) + return true; + + var portal = value.options.details; + var cachedPortal = oldcachedPortal[key]; + // If portal is changed, update playerApGain with latest + // information + if(!plugin.apList.isSamePortal(portal,cachedPortal)) { + // Copy portal detail to cachedPortal + cachedPortal = $.extend({}, portal); + var side = plugin.apList.portalSide(portal); + var getApGainFunc = plugin.apList.playerApGainFunc[side]; + // Assign playerApGain and guid to cachedPortal + cachedPortal.playerApGain = getApGainFunc(portal); + cachedPortal.guid = value.options.guid; + } + plugin.apList.cachedPortals[key] = cachedPortal; + }); + + // Add all portals to sortedPortals by side and sort sortedPortals by AP + $.each(plugin.apList.cachedPortals, function(key, portal) { + var side = plugin.apList.portalSide(portal); + plugin.apList.sortedPortals[side].push(portal); + }); + $.each(plugin.apList.sides, function(ind, side) { + plugin.apList.sortedPortals[side].sort(function(a, b) { + return b.playerApGain.totalAp - a.playerApGain.totalAp; + }); + }); +} + +window.plugin.apList.enableCache = function() { + plugin.apList.useCachedPortals = true; + plugin.apList.updateSortedPortals(); + plugin.apList.updatePortalTable(plugin.apList.displaySide); +} + +window.plugin.apList.disableCache = function() { + plugin.apList.useCachedPortals = false; + plugin.apList.cachedPortals = {}; + plugin.apList.updateSortedPortals(); + plugin.apList.updatePortalTable(plugin.apList.displaySide); +} + +window.plugin.apList.isSamePortal = function(a,b) { + if(!a || !b) return false; + if(a.team !== b.team) return false; + if(a.level !== b.level) return false; + for(var i = 0; i < 8; i++) { + if(!isSameResonator(a.resonatorArray.resonators[i],b.resonatorArray.resonators[i])) + return false; + } + return true; +} + +window.plugin.apList.portalSide = function(portal) { + return (portal.controllingTeam.team === PLAYER.team) + ? plugin.apList.SIDE_FRIENDLY + : plugin.apList.SIDE_ENEMY; +} + +// Get AP of friendly portal +window.plugin.apList.getDeployOrUpgradeApGain = function(d) { + var playerResoCount = new Array(MAX_PORTAL_LEVEL + 1); + var otherReso = new Array(); + var totalAp = 0; + var upgradedReso = new Array(); + + // loop through reso slot and find empty reso, deployed + // by others(only level lower than player level) or by player. + for(var i = 0; i < 8; i++) { + var reso = d.resonatorArray.resonators[i]; + + if(!reso) { + // Empty reso + reso = {slot: i, level: 0}; + otherReso.push(reso); + continue; + } + + // By player + if(reso.ownerGuid === window.PLAYER.guid) { + if(!playerResoCount[reso.level]) + playerResoCount[reso.level] = 0; + playerResoCount[reso.level]++; + continue; + } + + // By others and level lower than player + if(reso.level < window.PLAYER.level) { + otherReso.push(reso); + } + } + + // Sort others reso low to high, last reso in otherReso get upgrade first. + otherReso.sort(function(a, b) {return a.level - b.level;}); + + // Find out available count of reso for each level + for(var i = window.PLAYER.level; i > 0 && otherReso.length > 0; i--) { + var availableCount = MAX_RESO_PER_PLAYER[i] - (playerResoCount[i] || 0); + // Loop through lower level reso of others and add to result + while(availableCount > 0 && otherReso.length > 0) { + var targetReso = otherReso.pop(); + // Can only upgrade lower level reso + if(targetReso.level >= i) + continue; + // Add upgraded reso to result + targetReso.newLevel = i; + upgradedReso.push(targetReso); + // Add ap + totalAp += (targetReso.level === 0) + ? DEPLOY_RESONATOR + : UPGRADE_ANOTHERS_RESONATOR; + + availableCount--; + } + } + + return { + totalAp: totalAp, + upgradedReso: upgradedReso + }; +} + +window.plugin.apList.getAttackApGain = function(d) { + var resoCount = 0; + $.each(d.resonatorArray.resonators, function(ind, reso) { + if (!reso) + return true; + resoCount += 1; + }); + + var linkCount = d.portalV2.linkedEdges ? d.portalV2.linkedEdges.length : 0; + var fieldCount = d.portalV2.linkedFields ? d.portalV2.linkedFields.length : 0; + + var resoAp = resoCount * DESTROY_RESONATOR; + var linkAp = linkCount * DESTROY_LINK; + var fieldAp = fieldCount * DESTROY_FIELD; + var destroyAp = resoAp + linkAp + fieldAp; + var captureAp = CAPTURE_PORTAL + 8 * DEPLOY_RESONATOR + COMPLETION_BONUS; + var totalAp = destroyAp + captureAp; + + return { + totalAp: totalAp, + destroyAp: destroyAp, + captureAp: captureAp + } +} + +// Change display table to friendly portals +window.plugin.apList.displayFriendly = function() { + plugin.apList.displaySide = plugin.apList.SIDE_FRIENDLY; + plugin.apList.changeDisplaySide(plugin.apList.displaySide); +} + +// Change display table to enemy portals +window.plugin.apList.displayEnemy = function() { + plugin.apList.displaySide = plugin.apList.SIDE_ENEMY; + plugin.apList.changeDisplaySide(plugin.apList.displaySide); +} + +window.plugin.apList.changeDisplaySide = function(side) { + plugin.apList.updatePortalTable(side); + plugin.apList.toggleSideLabel(side); + + var scrollTo = $("#ap-list").position().top + $("#ap-list").outerHeight() + - $("#sidebar").height() + $("#sidebar").scrollTop() + $('#sidebar').scrollTop(scrollTo); +} + +window.plugin.apList.toggleSideLabel = function(side) { + $.each(plugin.apList.sides, function(ind,key) { + var labelClass = plugin.apList.sideLabelClass[key]; + var opacity = (key === side) ? 1.0 : 0.5; + $(labelClass).css("opacity", opacity); + }); +} + +window.plugin.apList.hideReloadLabel = function() { + $('#ap-list-reload').hide(); +} + +window.plugin.apList.showReloadLabel = function() { + $('#ap-list-reload').show(); +} + +window.plugin.apList.setupVar = function() { + plugin.apList.sides[plugin.apList.SIDE_FRIENDLY] = plugin.apList.SIDE_FRIENDLY; + plugin.apList.sides[plugin.apList.SIDE_ENEMY] = plugin.apList.SIDE_ENEMY; + plugin.apList.playerApGainFunc[plugin.apList.SIDE_FRIENDLY] + = plugin.apList.getDeployOrUpgradeApGain; + plugin.apList.playerApGainFunc[plugin.apList.SIDE_ENEMY] + = plugin.apList.getAttackApGain; + plugin.apList.sideLabelClass[plugin.apList.SIDE_FRIENDLY] + = "#ap-list-frd"; + plugin.apList.sideLabelClass[plugin.apList.SIDE_ENEMY] + = "#ap-list-eny"; +} + +window.plugin.apList.setupList = function() { + var content = '
' + + '' + + '' + + 'Enemy' + + '' + + '' + + 'Friendly' + + '' + + '' + + '' + + '↻ R' + + '' + + '
' + + '
'; + + $('#sidebar').append(content); + $('#ap-list-reload').hide(); + $('#ap-list').css({'color':'#ffce00', 'font-size':'90%', 'padding':'4px 2px'}); +} + +window.plugin.apList.setupMapEvent = function() { + map.on('zoomstart', function() { + plugin.apList.setupMapEvent.zoomLevelBefore = map.getZoom(); + // Stop changing cacheBounds if cache enabled + if(!plugin.apList.useCachedPortals) + plugin.apList.cacheBounds = map.getBounds(); + }); + + map.on('zoomend', function() { + // if zooming in and cache not yet enable, enable it + if(!plugin.apList.useCachedPortals + && map.getZoom() > plugin.apList.setupMapEvent.zoomLevelBefore) { + plugin.apList.enableCache(); + plugin.apList.showReloadLabel(); + } + }); + + map.on('moveend zoomend', function() { + // disable cache after out of cache bounds + if(plugin.apList.useCachedPortals) { + var currentBounds = map.getBounds(); + if(!plugin.apList.cacheBounds.contains(currentBounds)) { + plugin.apList.disableCache(); + plugin.apList.hideReloadLabel(); + } + } + }); +} + +var setup = function() { + window.plugin.apList.setupVar(); + window.plugin.apList.setupList(); + window.plugin.apList.setupMapEvent(); + window.addHook('requestFinished', window.plugin.apList.handleUpdate); +} +// PLUGIN END ////////////////////////////////////////////////////////// + +if(window.iitcLoaded && typeof setup === 'function') { + setup(); +} else { + if(window.bootPlugins) + window.bootPlugins.push(setup); + else + window.bootPlugins = [setup]; +} +} // wrapper end +// inject code into site context +var script = document.createElement('script'); +script.appendChild(document.createTextNode('('+ wrapper +')();')); +(document.body || document.head || document.documentElement).appendChild(script); diff --git a/screenshots/plugin-ap-list.png b/screenshots/plugin-ap-list.png new file mode 100644 index 00000000..ba4df9a5 Binary files /dev/null and b/screenshots/plugin-ap-list.png differ