179 lines
6.9 KiB
JavaScript
179 lines
6.9 KiB
JavaScript
|
||
// PORTAL RENDER LIMIT HANDLER ///////////////////////////////////////
|
||
// Functions to handle hiding low level portal when portal render
|
||
// limit is reached.
|
||
//
|
||
// On initialization, previous minLevel will preserve to previousMinLevel
|
||
// and modify with zoom level difference.
|
||
//
|
||
// After initialized and reset in window.requestData(), "processPortals"
|
||
// intercept all portals data in "handleDataResponse". Put the count of
|
||
// new portals to newPortalsPerLevel[portal level]. Portals with level >=
|
||
// previousMinLevel and already on map will return as result and continue
|
||
// to render. Others will save to portalsLowerThanPrevMinLv. If there is
|
||
// no more active request of map data, portalsLowerThanPrevMinLv will add
|
||
// back to result and render base on current minLevel.
|
||
//
|
||
// "window.handleFailRequest" is added to handle the case when the last request
|
||
// failed and "processPortals" didn't get called. It will get
|
||
// portalsLowerThanPrevMinLv base on current minLevel and render them.
|
||
//
|
||
// "getMinLevel" will be called by "getMinPortalLevel" in utils_misc.js
|
||
// to determine min portal level to draw on map.
|
||
//
|
||
// "getMinLevel" will return minLevel and call "setMinLevel" if
|
||
// minLevel hasn't set yet.
|
||
//
|
||
// In "setMinLevel", it will loop through all portal level from
|
||
// high to low, and sum total portal count (old + new) to check
|
||
// minLevel.
|
||
//
|
||
// minLevel is preserved and only replaced when render limit reached in
|
||
// higher level, until next window.requestData() called and reset.
|
||
//
|
||
|
||
window.portalRenderLimit = function() {}
|
||
|
||
window.portalRenderLimit.initialized = false;
|
||
window.portalRenderLimit.minLevelSet = false;
|
||
window.portalRenderLimit.minLevel = -1;
|
||
window.portalRenderLimit.previousMinLevel = -1;
|
||
window.portalRenderLimit.previousZoomLevel = null;
|
||
window.portalRenderLimit.newPortalsPerLevel = new Array(MAX_PORTAL_LEVEL + 1);
|
||
window.portalRenderLimit.portalsLowerThanPrevMinLv = new Array(MAX_PORTAL_LEVEL + 1);
|
||
window.portalRenderLimit.portalsLowerThanPrevMinLvCnt = new Array(MAX_PORTAL_LEVEL + 1);
|
||
|
||
window.portalRenderLimit.init = function () {
|
||
var currentZoomLevel = map.getZoom();
|
||
// previousZoomLevel set to current zoom level on the first run
|
||
portalRenderLimit.previousZoomLevel = portalRenderLimit.previousZoomLevel || currentZoomLevel;
|
||
|
||
// If there is a minLevel set in previous run, calculate previousMinLevel with it.
|
||
if(portalRenderLimit.minLevelSet) {
|
||
var zoomDiff = currentZoomLevel - portalRenderLimit.previousZoomLevel;
|
||
portalRenderLimit.previousMinLevel = Math.max(portalRenderLimit.minLevel - zoomDiff, -1);
|
||
portalRenderLimit.previousMinLevel = Math.min(portalRenderLimit.previousMinLevel, MAX_PORTAL_LEVEL);
|
||
}
|
||
|
||
portalRenderLimit.previousZoomLevel = currentZoomLevel;
|
||
|
||
portalRenderLimit.minLevel = -1;
|
||
portalRenderLimit.resetCounting();
|
||
portalRenderLimit.resetPortalsLowerThanPrevMinLv();
|
||
portalRenderLimit.initialized = true;
|
||
}
|
||
|
||
window.portalRenderLimit.resetCounting = function() {
|
||
portalRenderLimit.minLevelSet = false;
|
||
for(var i = 0; i <= MAX_PORTAL_LEVEL; i++) {
|
||
portalRenderLimit.newPortalsPerLevel[i] = 0;
|
||
}
|
||
}
|
||
|
||
window.portalRenderLimit.resetPortalsLowerThanPrevMinLv = function() {
|
||
for(var i = 0; i <= MAX_PORTAL_LEVEL; i++) {
|
||
portalRenderLimit.portalsLowerThanPrevMinLv[i] = {};
|
||
portalRenderLimit.portalsLowerThanPrevMinLvCnt[i] = 0;
|
||
}
|
||
}
|
||
|
||
// Use to clean up level of portals which is over render limit after counting new portals
|
||
window.portalRenderLimit.cleanUpOverLimitPortalLevel = function() {
|
||
var currentMinLevel = window.getMinPortalLevel();
|
||
for(var i = 0; i < currentMinLevel; i++) {
|
||
portalsLayers[i].eachLayer(function(item) {
|
||
var itemGuid = item.options.guid;
|
||
// check if 'item' is a portal
|
||
if(getTypeByGuid(itemGuid) != TYPE_PORTAL) return true;
|
||
// Don’t remove if it is selected.
|
||
if(itemGuid == window.selectedPortal) return true;
|
||
portalsLayers[i].removeLayer(item);
|
||
});
|
||
}
|
||
}
|
||
|
||
// Count new portals. Then split lower level portal if it's not last request.
|
||
// And Merge back if it's last request and render limit not yet hit
|
||
window.portalRenderLimit.splitOrMergeLowLevelPortals = function(originPortals) {
|
||
portalRenderLimit.resetCounting();
|
||
portalRenderLimit.countingPortals(originPortals);
|
||
|
||
var resultPortals = requests.isLastRequest('getThinnedEntitiesV4')
|
||
? portalRenderLimit.mergeLowLevelPortals(originPortals)
|
||
: portalRenderLimit.splitLowLevelPortals(originPortals);
|
||
|
||
return resultPortals;
|
||
}
|
||
|
||
window.portalRenderLimit.countingPortals = function(portals) {
|
||
$.each(portals, function(ind, portal) {
|
||
var portalGuid = portal[0];
|
||
var portalLevel = parseInt(getPortalLevel(portal[2]));
|
||
var layerGroup = portalsLayers[portalLevel];
|
||
|
||
if(findEntityInLeaflet(layerGroup, window.portals, portalGuid)) return true;
|
||
|
||
portalRenderLimit.newPortalsPerLevel[portalLevel]++;
|
||
});
|
||
}
|
||
|
||
// Split the portal if it's lower level and not on map
|
||
window.portalRenderLimit.splitLowLevelPortals = function(portals) {
|
||
var resultPortals = {};
|
||
|
||
$.each(portals || {}, function(guid, portal) {
|
||
var portalLevel = parseInt(getPortalLevel(portal[2]));
|
||
var portalOnMap = window.portals[guid];
|
||
|
||
if(!portalOnMap && portalLevel < portalRenderLimit.previousMinLevel) {
|
||
portalRenderLimit.portalsLowerThanPrevMinLv[portalLevel][guid] = portal;
|
||
portalRenderLimit.portalsLowerThanPrevMinLvCnt[portalLevel]++;
|
||
} else {
|
||
resultPortals[guid] = portal;
|
||
}
|
||
});
|
||
return resultPortals;
|
||
}
|
||
|
||
window.portalRenderLimit.mergeLowLevelPortals = function(appendTo) {
|
||
var resultPortals = appendTo ? appendTo : {};
|
||
for(var i = portalRenderLimit.getMinLevel();
|
||
i < portalRenderLimit.previousMinLevel;
|
||
i++) {
|
||
$.extend(resultPortals, portalRenderLimit.portalsLowerThanPrevMinLv[i]);
|
||
}
|
||
|
||
// Reset portalsLowerThanPrevMinLv, ensure they return only once
|
||
portalRenderLimit.resetPortalsLowerThanPrevMinLv();
|
||
return resultPortals;
|
||
}
|
||
|
||
window.portalRenderLimit.getMinLevel = function() {
|
||
if(!portalRenderLimit.initialized) return -1;
|
||
if(!portalRenderLimit.minLevelSet) portalRenderLimit.setMinLevel();
|
||
return portalRenderLimit.minLevel;
|
||
}
|
||
|
||
window.portalRenderLimit.setMinLevel = function() {
|
||
var totalPortalsCount = 0;
|
||
var newMinLevel = MAX_PORTAL_LEVEL + 1;
|
||
|
||
// Find the min portal level under render limit
|
||
while(newMinLevel > 0) {
|
||
var oldPortalCount = layerGroupLength(portalsLayers[newMinLevel - 1]);
|
||
var storedPortalCount = portalRenderLimit.portalsLowerThanPrevMinLvCnt[newMinLevel - 1];
|
||
var newPortalCount = Math.max(storedPortalCount, portalRenderLimit.newPortalsPerLevel[newMinLevel - 1]);
|
||
|
||
totalPortalsCount += oldPortalCount + newPortalCount;
|
||
if(totalPortalsCount >= MAX_DRAWN_PORTALS)
|
||
break;
|
||
newMinLevel--;
|
||
}
|
||
|
||
// If render limit reached at max portal level, still let portal at max level render
|
||
newMinLevel = Math.min(newMinLevel, MAX_PORTAL_LEVEL);
|
||
|
||
portalRenderLimit.minLevel = Math.max(newMinLevel, portalRenderLimit.minLevel);
|
||
portalRenderLimit.minLevelSet = true;
|
||
}
|