自定义地图平铺叠加问题

时间:2015-01-22 22:14:11

标签: javascript asp.net google-maps google-maps-api-3

我需要在谷歌地图提供的标准卫星图像上显示几百到一千张高分辨率航拍照片。图像在地理上是分散的,所以我决定将tile服务器实现为通用的asp.net处理程序(* .ashx文件)。我将根据以下网址在Google开发者网站上显示的地图上发布我的问题说明:

https://developers.google.com/maps/documentation/javascript/examples/maptype-overlay

一切都或多或少有效,但我有以下两个问题:

1)选择“卫星”地图类型后,将鼠标悬停在该按钮上会产生一个带有“标签”复选框的下拉列表。如何在标题为“航空照片”的下拉列表中添加另一个复选框,以打开/关闭我的叠加层?我是否需要对使用Google Maps实施细节的JQuery黑客进行硬编码,还是可以通过API实现这一目标?

2)如果指定的图块不存在,则我的* .ashx处理程序返回图像或状态204(无内容)。问题是204结果没有缓存,因此每次我缩小并返回到同一位置时,我的服务器会重新命中客户端应该已知的所有磁贴都不存在。我没有看到它记录了一个tile服务器应该为这样一个“空”tile返回什么,因此客户端可以缓存结果。如果特定位置没有地图图块,我应该返回什么?

感谢。

1 个答案:

答案 0 :(得分:1)

鉴于对此问题缺乏回应,很明显稀疏磁贴服务器是一种不常见的做法。以下是两个问题的解决方案(但可能是hacky):

1)如何在“卫星”下拉列表中添加一个复选框以切换我的地图图层?不幸的是,没有支持的方法来做到这一点,所以我提出了以下令人难以置信的hacky代码:

// Create a function to select the "Labels" checkbox
var getLabelButton = function() {
    return $('.gm-style-mtc:last > div:last > div:last');
};

// Clone the "Labels" checkbox used to show/hide the hybrid map overlay
var labelBtn   = getLabelButton();
var labelClone = labelBtn.clone();

// Change the display and hover text for the new button
labelClone.prop('title', Localizer.GetString('CustomImagery.Description'));
labelClone.find('label').html(Localizer.GetString('CustomImagery.Name'));

// Highlight the button when the client hovers the mouse over it
var checkbox        = labelClone.children('span');
var prevBackColor   = labelClone.css('background-color');
var prevBorderColor = checkbox  .css('border-color');
labelClone.hover(function() {
    labelClone.css('background-color', '#EBEBEB');
    checkbox  .css('border-color'    , '#666');
}, function() {
    labelClone.css('background-color', prevBackColor);
    checkbox  .css('border-color'    , prevBorderColor);
});

// Set the checkmark image source to be the correct value, instead of transparent
var checkmark    = checkbox .children('div');
var checkmarkImg = checkmark.children('img');
checkmarkImg.attr('src', 'https://maps.gstatic.com/mapfiles/mv/imgs8.png');

// Attach the new checkbox after the Labels checkbox
labelBtn.after(labelClone);

// Create a method to determine if the selected map type supports custom imagery
var mapTypesSupportingCustomImagery = [
    google.maps.MapTypeId.SATELLITE,
    google.maps.MapTypeId.HYBRID
];
var isImagerySupportedOnSelectedMapType = function() {
    var mapTypeId = googleMap.getMapTypeId();
    return (0 <= mapTypesSupportingCustomImagery.indexOf(mapTypeId));
};

// Show the checkmark and imagery if the initial map type supports it
if (isImagerySupportedOnSelectedMapType()) {
    checkmark.css('display', '');
    googleMap.overlayMapTypes.push(tileServer);
}

// Show/hide the checkmark and imagery when the user clicks on the checkbox
labelClone.on('click', function() {
    var showImagery = (checkmark.css('display') === 'none');
    if (showImagery) {
        checkmark.css('display', '');
        googleMap.overlayMapTypes.push(tileServer);
    } else {
        checkmark.css('display', 'none');
        var tileServerIndex = googleMap.overlayMapTypes.indexOf(tileServer);
        googleMap.overlayMapTypes.removeAt(tileServerIndex);
    }
});

// Create a function that returns whether the custom imagery should be displayed
var displayCustomImagery = function() {
    return (isImagerySupportedOnSelectedMapType() && checkmark.css('display') != 'none');
};

// Add an event listener to add the tile server when displaying satellite view
google.maps.event.addListener(googleMap, 'maptypeid_changed', function() {
    var tileServerIndex = googleMap.overlayMapTypes.indexOf(tileServer);
    if (displayCustomImagery()) {
        if (tileServerIndex < 0) {
            googleMap.overlayMapTypes.push(tileServer);
        }
    } else if (0 <= tileServerIndex) {
        googleMap.overlayMapTypes.removeAt(tileServerIndex);
    }
});

在上面的代码中,googleMap是地图对象,而tileServer是我对google.maps.ImageMapType对象的实现。

2)我应该返回什么来代表一个空的瓷砖?

我对这个问题的解决方案相当简洁。我只列出服务器上所有磁贴的文件名,这是所请求磁贴的Morton编号的base-4编码。然后我将此列表作为字典从字符串发送到客户端到bool(始终为true)。客户端在发出请求之前只是检查服务器是否包含地图图块,因此服务器不必担心返回什么(如果发出无效请求,我将其保留为返回204错误)。在getTileUrl方法中获取tile名称的javascript如下:

function(coord, zoom) {
    // Return null if the zoom level is not supported
    // NOTE: This should not be necessary, but minZoom and
    //       maxZoom parameters are ignored for image map types
    if (zoom < minZoom || maxZoom < zoom) {
        return null;
    }

    // Get the name of the map tile being requested
    var tileName = '';
    var y = coord.y << 1;
    for (var shift = zoom - 1; 0 <= shift; --shift) {
        var digit = (coord.x >>> shift) & 1;
        digit    |= (      y >>> shift) & 2;
        tileName += digit;
    }

    // Return if the map tile being requested does not exist
    if (!mapTiles[tileName]) {
        return null;
    }

    // Return the url to the tile server
    ...
}