// PORTAL DETAILS MAIN /////////////////////////////////////////////// // main code block that renders the portal details in the sidebar and // methods that highlight the portal in the map view. window.renderPortalDetails = function(guid) { if(!window.portals[guid]) { unselectOldPortal(); urlPortal = guid; return; } var d = window.portals[guid].options.details; selectPortal(guid); // collect some random data that’s not worth to put in an own method var links = {incoming: 0, outgoing: 0}; if(d.portalV2.linkedEdges) $.each(d.portalV2.linkedEdges, function(ind, link) { links[link.isOrigin ? 'outgoing' : 'incoming']++; }); function linkExpl(t) { return ''+t+''; } var linksText = [linkExpl('links'), linkExpl(' ↳ ' + links.incoming+'  •  '+links.outgoing+' ↴')]; var player = d.captured && d.captured.capturingPlayerId ? '' + getPlayerName(d.captured.capturingPlayerId) + '' : null; var playerText = player ? ['owner', player] : null; var time = d.captured ? '' + unixTimeToString(d.captured.capturedTime) + '' : null; var sinceText = time ? ['since', time] : null; var linkedFields = ['fields', d.portalV2.linkedFields ? d.portalV2.linkedFields.length : 0]; // collect and html-ify random data var randDetails = [ playerText, sinceText, getRangeText(d), getEnergyText(d), linksText, getAvgResoDistText(d), linkedFields, getAttackApGainText(d) ]; randDetails = '' + genFourColumnTable(randDetails) + '
'; var resoDetails = '' + getResonatorDetails(d) + '
'; setPortalIndicators(d); var img = getPortalImageUrl(d); var lat = d.locationE6.latE6/1E6; var lng = d.locationE6.lngE6/1E6; var perma = '/intel?ll='+lat+','+lng+'&z=17&pll='+lat+','+lng; var imgTitle = 'title="'+getPortalDescriptionFromDetails(d)+'\n\nClick to show full image."'; var poslinks = 'window.showPortalPosLinks('+lat+','+lng+',\''+escapeJavascriptString(d.portalV2.descriptiveText.TITLE)+'\')'; var portalDetailObj = window.getPortalDescriptionFromDetailsExtended(d); var portalDetailedDescription = ''; if(portalDetailObj) { portalDetailedDescription = ''; // TODO (once the data supports it) - portals can have multiple photos. display all, with navigation between them // (at this time the data isn't returned from the server - although a count of images IS returned!) if(portalDetailObj.submitter.name.length > 0) { if(portalDetailObj.submitter.team) { submitterSpan = ''; } else { submitterSpan = ''; } portalDetailedDescription += ''; } if(portalDetailObj.submitter.link.length > 0) { portalDetailedDescription += ''; } if(portalDetailObj.description) { portalDetailedDescription += ''; } // if(d.portalV2.descriptiveText.ADDRESS) { // portalDetailedDescription += ''; // } portalDetailedDescription += '
Photo by:' + submitterSpan + escapeHtmlSpecialChars(portalDetailObj.submitter.name) + ' (' + portalDetailObj.submitter.voteCount + ' votes)
Photo from:' + escapeHtmlSpecialChars(portalDetailObj.submitter.link) + '
Description:' + escapeHtmlSpecialChars(portalDetailObj.description) + '
Address:' + escapeHtmlSpecialChars(d.portalV2.descriptiveText.ADDRESS) + '
'; } $('#portaldetails') .attr('class', TEAM_TO_CSS[getTeam(d)]) .html('' + '

'+escapeHtmlSpecialChars(d.portalV2.descriptiveText.TITLE)+'

' + 'X' // help cursor via ".imgpreview img" + '
' + ''+Math.floor(getPortalLevel(d))+'' + '
'+ portalDetailedDescription + '
' + '
' + '' + '
'+getModDetails(d)+'
' + randDetails + resoDetails + '
' + ( typeof android !== 'undefined' && android && android.intentPosLink // Android handles both links via a dialog ? '' : '' + '' ) + '' + '
' ); // try to resolve names that were required for above functions, but // weren't available yet. resolvePlayerNames(); runHooks('portalDetailsUpdated', {portalDetails: d}); } // draws link-range and hack-range circles around the portal with the // given details. window.setPortalIndicators = function(d) { if(portalRangeIndicator) map.removeLayer(portalRangeIndicator); var range = getPortalRange(d); var coord = [d.locationE6.latE6/1E6, d.locationE6.lngE6/1E6]; portalRangeIndicator = (range > 0 ? L.geodesicCircle(coord, range, { fill: false, color: RANGE_INDICATOR_COLOR, weight: 3, clickable: false }) : L.circle(coord, range, { fill: false, stroke: false, clickable: false }) ).addTo(map); if(!portalAccessIndicator) portalAccessIndicator = L.circle(coord, HACK_RANGE, { fill: false, color: ACCESS_INDICATOR_COLOR, weight: 2, clickable: false } ).addTo(map); else portalAccessIndicator.setLatLng(coord); } window.clearPortalIndicators = function() { if(portalRangeIndicator) map.removeLayer(portalRangeIndicator); portalRangeIndicator = null; if(portalAccessIndicator) map.removeLayer(portalAccessIndicator); portalAccessIndicator = null; } // highlights portal with given GUID. Automatically clears highlights // on old selection. Returns false if the selected portal changed. // Returns true if it's still the same portal that just needs an // update. window.selectPortal = function(guid) { var update = selectedPortal === guid; var oldPortal = portals[selectedPortal]; if(!update && oldPortal) setMarkerStyle(oldPortal,false); selectedPortal = guid; if(portals[guid]) { // resonatorsSetSelectStyle(guid); portals[guid].bringToFront(); setMarkerStyle(portals[guid], true); } return update; } window.unselectOldPortal = function() { var oldPortal = portals[selectedPortal]; if(oldPortal) setMarkerStyle(oldPortal,false); selectedPortal = null; $('#portaldetails').html(''); if(isSmartphone()) { $('.fullimg').remove(); $('#mobileinfo').html(''); } clearPortalIndicators(); }