734 lines
19 KiB
JavaScript
734 lines
19 KiB
JavaScript
(function($) {
|
|
var _options;
|
|
var _self;
|
|
|
|
var _boundingBoxes;
|
|
var _zoneCentroids = {};
|
|
var _selectedRegionKey;
|
|
var _selectedPolygon;
|
|
var _mapper;
|
|
var _mapZones = {};
|
|
var _transitions = {};
|
|
|
|
var _currentHoverRegion;
|
|
var _hoverRegions = {};
|
|
var _hoverPolygons = [];
|
|
|
|
var _loader;
|
|
var _loaderGif;
|
|
var _maskPng;
|
|
var _needsLoader = 0;
|
|
|
|
var OpenLayersMapper = function(
|
|
el,
|
|
mouseClickHandler,
|
|
mouseMoveHandler,
|
|
mapOptions
|
|
) {
|
|
var infoWindow;
|
|
|
|
// Create the maps instance
|
|
var map = new OpenLayers.Map(
|
|
OpenLayers.Util.extend(
|
|
{
|
|
div: el,
|
|
projection: "EPSG:900913",
|
|
displayProjection: "EPSG:4326",
|
|
numZoomLevels: 18,
|
|
controls: [
|
|
new OpenLayers.Control.DragPan(),
|
|
new OpenLayers.Control.Navigation({
|
|
mouseWheelOptions: {
|
|
cumulative: false,
|
|
maxDelta: 6,
|
|
interval: 50
|
|
},
|
|
zoomWheelEnabled: false
|
|
}),
|
|
new OpenLayers.Control.Zoom(),
|
|
new OpenLayers.Control.ZoomBox()
|
|
]
|
|
},
|
|
mapOptions
|
|
)
|
|
);
|
|
|
|
var newLayer = new OpenLayers.Layer.OSM(
|
|
"OSM Layer",
|
|
"http://a.tile.openstreetmap.org/${z}/${x}/${y}.png"
|
|
);
|
|
map.addLayer(newLayer);
|
|
|
|
var vectors = new OpenLayers.Layer.Vector("vector");
|
|
|
|
map.addLayer(vectors);
|
|
|
|
$("#timezone").on("change", function() {
|
|
var optionText = $("#timezone option:selected").val();
|
|
selectZone(optionText);
|
|
});
|
|
|
|
OpenLayers.Control.Click = OpenLayers.Class(OpenLayers.Control, {
|
|
defaultHandlerOptions: {
|
|
single: true,
|
|
double: false,
|
|
pixelTolerance: 0,
|
|
stopSingle: false,
|
|
stopDouble: false
|
|
},
|
|
|
|
initialize: function() {
|
|
this.handlerOptions = OpenLayers.Util.extend(
|
|
{},
|
|
this.defaultHandlerOptions
|
|
);
|
|
OpenLayers.Control.prototype.initialize.apply(this, arguments);
|
|
this.handler = new OpenLayers.Handler.Click(
|
|
this,
|
|
{
|
|
click: this.trigger
|
|
},
|
|
this.handlerOptions
|
|
);
|
|
},
|
|
|
|
trigger: function(e) {
|
|
var position = map.getLonLatFromViewPortPx(e.xy);
|
|
position.transform(
|
|
map.getProjectionObject(),
|
|
new OpenLayers.Projection("EPSG:4326")
|
|
);
|
|
mapClickHandler({
|
|
latLng: {
|
|
lat: function() {
|
|
return position.lat;
|
|
},
|
|
lng: function() {
|
|
return position.lon;
|
|
}
|
|
}
|
|
});
|
|
}
|
|
});
|
|
var click = new OpenLayers.Control.Click();
|
|
map.addControl(click);
|
|
click.activate();
|
|
|
|
if (mouseMoveHandler) {
|
|
map.events.register("mousemove", map, function(e) {
|
|
var position = map.getLonLatFromViewPortPx(e.xy);
|
|
position.transform(
|
|
map.getProjectionObject(),
|
|
new OpenLayers.Projection("EPSG:4326")
|
|
);
|
|
mouseMoveHandler({
|
|
latLng: {
|
|
lat: function() {
|
|
return position.lat;
|
|
},
|
|
lng: function() {
|
|
return position.lon;
|
|
}
|
|
}
|
|
});
|
|
});
|
|
}
|
|
|
|
var onPolygonSelect = function(feature) {
|
|
if (feature.clickHandler) {
|
|
var position = map.getLonLatFromPixel(
|
|
new OpenLayers.Pixel(
|
|
polygonSelect.handlers.feature.evt.layerX,
|
|
polygonSelect.handlers.feature.evt.layerY
|
|
)
|
|
);
|
|
position.transform(
|
|
map.getProjectionObject(),
|
|
new OpenLayers.Projection("EPSG:4326")
|
|
);
|
|
feature.clickHandler({
|
|
latLng: {
|
|
lat: function() {
|
|
return position.lat;
|
|
},
|
|
lng: function() {
|
|
return position.lon;
|
|
}
|
|
}
|
|
});
|
|
}
|
|
};
|
|
|
|
var onPolygonHighlight = function(e) {
|
|
if (e.feature.hoverHandler) {
|
|
e.feature.hoverHandler();
|
|
}
|
|
};
|
|
|
|
var polygonHover = new OpenLayers.Control.SelectFeature(vectors, {
|
|
hover: true,
|
|
highlightOnly: true,
|
|
renderIntent: "temporary",
|
|
eventListeners: {
|
|
beforefeaturehighlighted: onPolygonHighlight
|
|
}
|
|
});
|
|
|
|
var polygonSelect = new OpenLayers.Control.SelectFeature(vectors, {
|
|
onSelect: onPolygonSelect
|
|
});
|
|
|
|
map.addControl(polygonHover);
|
|
map.addControl(polygonSelect);
|
|
polygonHover.activate();
|
|
polygonSelect.activate();
|
|
|
|
map.setCenter(new OpenLayers.LonLat(0, 0), mapOptions.zoom);
|
|
|
|
var addPolygon = function(
|
|
coords,
|
|
stroke,
|
|
fill,
|
|
clickHandler,
|
|
mouseMoveHandler
|
|
) {
|
|
for (var i = 0; i < coords.length; i++) {
|
|
coords[i].transform(
|
|
new OpenLayers.Projection("EPSG:4326"), // transform from WGS 1984
|
|
map.getProjectionObject() // to Spherical Mercator Projection
|
|
);
|
|
}
|
|
|
|
var style = {
|
|
strokeColor: stroke.color,
|
|
strokeOpacity: stroke.opacity,
|
|
strokeWidth: stroke.width,
|
|
fillColor: fill.color,
|
|
fillOpacity: fill.opacity
|
|
};
|
|
var linearRing = new OpenLayers.Geometry.LinearRing(coords);
|
|
var feature = new OpenLayers.Feature.Vector(
|
|
new OpenLayers.Geometry.Polygon(linearRing),
|
|
null,
|
|
style
|
|
);
|
|
|
|
vectors.addFeatures([feature]);
|
|
|
|
// NOTE: Stuff our click/mousemove handlers on the object for use in onPolygonSelect
|
|
feature.clickHandler = clickHandler;
|
|
feature.hoverHandler = mouseMoveHandler;
|
|
|
|
return feature;
|
|
};
|
|
|
|
var createPoint = function(lat, lng) {
|
|
return new OpenLayers.Geometry.Point(lng, lat);
|
|
};
|
|
|
|
var hideInfoWindow = function() {
|
|
if (infoWindow) {
|
|
map.removePopup(infoWindow);
|
|
infoWindow.destroy();
|
|
infoWindow = null;
|
|
}
|
|
};
|
|
|
|
var removePolygon = function(mapPolygon) {
|
|
vectors.removeFeatures([mapPolygon]);
|
|
};
|
|
|
|
var showInfoWindow = function(pos, content, callback) {
|
|
if (infoWindow) {
|
|
hideInfoWindow(infoWindow);
|
|
}
|
|
|
|
pos = new OpenLayers.LonLat(pos.x, pos.y);
|
|
pos.transform(
|
|
new OpenLayers.Projection("EPSG:4326"), // transform from WGS 1984
|
|
map.getProjectionObject() // to Spherical Mercator Projection
|
|
);
|
|
|
|
infoWindow = new OpenLayers.Popup.FramedCloud(
|
|
"timezone_picker_infowindow",
|
|
pos,
|
|
new OpenLayers.Size(100, 100),
|
|
content,
|
|
null,
|
|
true,
|
|
null
|
|
);
|
|
map.addPopup(infoWindow);
|
|
|
|
// HACK: callback for popup using a set timeout
|
|
if (callback) {
|
|
setTimeout(function() {
|
|
callback.apply($("#timezone_picker_infowindow"));
|
|
}, 100);
|
|
}
|
|
};
|
|
|
|
return {
|
|
addPolygon: addPolygon,
|
|
createPoint: createPoint,
|
|
hideInfoWindow: hideInfoWindow,
|
|
removePolygon: removePolygon,
|
|
showInfoWindow: showInfoWindow
|
|
};
|
|
};
|
|
|
|
// Forward declarations to satisfy jshint
|
|
var hideLoader,
|
|
hitTestAndConvert,
|
|
selectPolygonZone,
|
|
showInfoWindow,
|
|
slugifyName;
|
|
|
|
var clearHover = function() {
|
|
$.each(_hoverPolygons, function(i, p) {
|
|
_mapper.removePolygon(p);
|
|
});
|
|
|
|
_hoverPolygons = [];
|
|
};
|
|
|
|
var clearZones = function() {
|
|
$.each(_mapZones, function(i, zone) {
|
|
$.each(zone, function(j, polygon) {
|
|
_mapper.removePolygon(polygon);
|
|
});
|
|
});
|
|
|
|
_mapZones = {};
|
|
};
|
|
|
|
var drawZone = function(name, lat, lng, callback) {
|
|
if (_mapZones[name]) {
|
|
return;
|
|
}
|
|
|
|
$.get(_options.jsonRootUrl + "polygons/" + name + ".json", function(data) {
|
|
_needsLoader--;
|
|
if (_needsLoader === 0 && _loader) {
|
|
hideLoader();
|
|
}
|
|
|
|
if (callback) {
|
|
callback();
|
|
}
|
|
|
|
data = typeof data === "string" ? JSON.parse(data) : data;
|
|
|
|
_mapZones[name] = [];
|
|
$.extend(_transitions, data.transitions);
|
|
|
|
var result = hitTestAndConvert(data.polygons, lat, lng);
|
|
|
|
if (result.inZone) {
|
|
_selectedRegionKey = name;
|
|
|
|
$.each(result.allPolygons, function(i, polygonInfo) {
|
|
var mapPolygon = _mapper.addPolygon(
|
|
polygonInfo.coords,
|
|
{
|
|
color: "#fff",
|
|
opacity: 0.7,
|
|
width: 1
|
|
},
|
|
{
|
|
color: "#82b92e",
|
|
opacity: 0.9
|
|
},
|
|
function() {
|
|
selectPolygonZone(polygonInfo.polygon);
|
|
},
|
|
clearHover
|
|
);
|
|
|
|
_mapZones[name].push(mapPolygon);
|
|
});
|
|
|
|
selectPolygonZone(result.selectedPolygon);
|
|
}
|
|
}).fail(function() {
|
|
console.warn(arguments);
|
|
});
|
|
};
|
|
|
|
var getCurrentTransition = function(transitions) {
|
|
if (transitions.length === 1) {
|
|
return transitions[0];
|
|
}
|
|
|
|
var now = _options.date.getTime() / 1000;
|
|
var selected = null;
|
|
$.each(transitions, function(i, transition) {
|
|
if (
|
|
transition[0] < now &&
|
|
i < transitions.length - 1 &&
|
|
transitions[i + 1][0] > now
|
|
) {
|
|
selected = transition;
|
|
}
|
|
});
|
|
|
|
// If we couldn't find a matching transition, just use the first one
|
|
// NOTE: This will sometimes be wrong for events in the past
|
|
if (!selected) {
|
|
selected = transitions[0];
|
|
}
|
|
|
|
return selected;
|
|
};
|
|
|
|
var hideInfoWindow = function() {
|
|
_mapper.hideInfoWindow();
|
|
};
|
|
|
|
hideLoader = function() {
|
|
_loader.remove();
|
|
_loader = null;
|
|
};
|
|
|
|
hitTestAndConvert = function(polygons, lat, lng) {
|
|
var allPolygons = [];
|
|
var inZone = false;
|
|
var selectedPolygon;
|
|
$.each(polygons, function(i, polygon) {
|
|
// Ray casting counter for hit testing.
|
|
var rayTest = 0;
|
|
var lastPoint = polygon.points.slice(-2);
|
|
|
|
var coords = [];
|
|
var j = 0;
|
|
for (j = 0; j < polygon.points.length; j += 2) {
|
|
var point = polygon.points.slice(j, j + 2);
|
|
|
|
coords.push(_mapper.createPoint(point[0], point[1]));
|
|
|
|
// Ray casting test
|
|
if (
|
|
(lastPoint[0] <= lat && point[0] >= lat) ||
|
|
(lastPoint[0] > lat && point[0] < lat)
|
|
) {
|
|
var slope = (point[1] - lastPoint[1]) / (point[0] - lastPoint[0]);
|
|
var testPoint = slope * (lat - lastPoint[0]) + lastPoint[1];
|
|
if (testPoint < lng) {
|
|
rayTest++;
|
|
}
|
|
}
|
|
|
|
lastPoint = point;
|
|
}
|
|
|
|
allPolygons.push({
|
|
polygon: polygon,
|
|
coords: coords
|
|
});
|
|
|
|
// If the count is odd, we are in the polygon
|
|
var odd = rayTest % 2 === 1;
|
|
inZone = inZone || odd;
|
|
if (odd) {
|
|
selectedPolygon = polygon;
|
|
}
|
|
});
|
|
|
|
return {
|
|
allPolygons: allPolygons,
|
|
inZone: inZone,
|
|
selectedPolygon: selectedPolygon
|
|
};
|
|
};
|
|
|
|
var mapClickHandler = function(e) {
|
|
if (_needsLoader > 0) {
|
|
return;
|
|
}
|
|
|
|
hideInfoWindow();
|
|
|
|
var lat = e.latLng.lat();
|
|
var lng = e.latLng.lng();
|
|
|
|
var candidates = [];
|
|
$.each(_boundingBoxes, function(i, v) {
|
|
var bb = v.boundingBox;
|
|
if (lat > bb.ymin && lat < bb.ymax && lng > bb.xmin && lng < bb.xmax) {
|
|
candidates.push(slugifyName(v.name));
|
|
}
|
|
});
|
|
|
|
_needsLoader = candidates.length;
|
|
setTimeout(function() {
|
|
if (_needsLoader > 0) {
|
|
showLoader();
|
|
}
|
|
}, 500);
|
|
|
|
clearZones();
|
|
$.each(candidates, function(i, v) {
|
|
drawZone(v, lat, lng, function() {
|
|
$.each(_hoverPolygons, function(i, p) {
|
|
_mapper.removePolygon(p);
|
|
});
|
|
_hoverPolygons = [];
|
|
_currentHoverRegion = null;
|
|
});
|
|
});
|
|
};
|
|
|
|
var mouseMoveHandler = function(e) {
|
|
var lat = e.latLng.lat();
|
|
var lng = e.latLng.lng();
|
|
|
|
$.each(_boundingBoxes, function(i, v) {
|
|
var bb = v.boundingBox;
|
|
if (lat > bb.ymin && lat < bb.ymax && lng > bb.xmin && lng < bb.xmax) {
|
|
var hoverRegion = _hoverRegions[v.name];
|
|
if (!hoverRegion) {
|
|
return;
|
|
}
|
|
|
|
var result = hitTestAndConvert(hoverRegion.hoverRegion, lat, lng);
|
|
var slugName = slugifyName(v.name);
|
|
if (
|
|
result.inZone &&
|
|
slugName !== _currentHoverRegion &&
|
|
slugName !== _selectedRegionKey
|
|
) {
|
|
clearHover();
|
|
_currentHoverRegion = slugName;
|
|
|
|
$.each(result.allPolygons, function(i, polygonInfo) {
|
|
var mapPolygon = _mapper.addPolygon(
|
|
polygonInfo.coords,
|
|
{
|
|
color: "#848484",
|
|
opacity: 0.7,
|
|
width: 1
|
|
},
|
|
{
|
|
color: "#cbcbcb",
|
|
opacity: 0.5
|
|
},
|
|
mapClickHandler,
|
|
null
|
|
);
|
|
|
|
_hoverPolygons.push(mapPolygon);
|
|
});
|
|
|
|
if (_options.onHover) {
|
|
var transition = getCurrentTransition(hoverRegion.transitions);
|
|
_options.onHover(transition[1], transition[2]);
|
|
}
|
|
}
|
|
}
|
|
});
|
|
};
|
|
|
|
selectPolygonZone = function(polygon) {
|
|
_selectedPolygon = polygon;
|
|
var transition = getCurrentTransition(_transitions[polygon.name]);
|
|
var olsonName = polygon.name; //when you click in the map
|
|
var utcOffset = transition[1];
|
|
var tzName = transition[2];
|
|
|
|
if (_options.onSelected) {
|
|
_options.onSelected(olsonName);
|
|
} else {
|
|
var pad = function(d) {
|
|
if (d < 10) {
|
|
return "0" + d;
|
|
}
|
|
return d.toString();
|
|
};
|
|
|
|
var now = new Date();
|
|
var adjusted = new Date();
|
|
adjusted.setTime(
|
|
adjusted.getTime() +
|
|
(adjusted.getTimezoneOffset() + utcOffset) * 60 * 1000
|
|
);
|
|
|
|
if (olsonName == "America/Montreal") {
|
|
olsonName = "America/Toronto";
|
|
} else if (olsonName == "Asia/Chongqing") {
|
|
olsonName = "Asia/Shanghai";
|
|
}
|
|
|
|
$("#timezone").val(olsonName); //Select this value in the dropdown
|
|
}
|
|
};
|
|
|
|
// Function to draw the timezone in the map, when you select it in the dropdown
|
|
var selectZone = function(olsonName) {
|
|
var centroid = _zoneCentroids[olsonName];
|
|
if (centroid) {
|
|
mapClickHandler({
|
|
latLng: {
|
|
lat: function() {
|
|
return centroid[1];
|
|
},
|
|
lng: function() {
|
|
return centroid[0];
|
|
}
|
|
}
|
|
});
|
|
}
|
|
};
|
|
|
|
showInfoWindow = function(content, callback) {
|
|
// Hack to get the centroid of the largest polygon - we just check
|
|
// which has the most edges
|
|
var centroid;
|
|
var maxPoints = 0;
|
|
if (_selectedPolygon.points.length > maxPoints) {
|
|
centroid = _selectedPolygon.centroid;
|
|
maxPoints = _selectedPolygon.points.length;
|
|
}
|
|
|
|
hideInfoWindow();
|
|
|
|
_mapper.showInfoWindow(
|
|
_mapper.createPoint(centroid[1], centroid[0]),
|
|
content,
|
|
callback
|
|
);
|
|
};
|
|
|
|
var showLoader = function() {
|
|
_loader = $(
|
|
'<div style="background: url(' +
|
|
_maskPng +
|
|
');z-index:10000;">' +
|
|
'<img style="margin-top:-8px;margin-left:-8px" ' +
|
|
'src="' +
|
|
_loaderGif +
|
|
'" /></div>'
|
|
);
|
|
_loader.height(_self.height()).width(_self.width());
|
|
_self.append(_loader);
|
|
};
|
|
|
|
slugifyName = function(name) {
|
|
return name.toLowerCase().replace(/[^a-z0-9]/g, "-");
|
|
};
|
|
|
|
var methods = {
|
|
init: function(options) {
|
|
_self = this;
|
|
|
|
// Populate the options and set defaults
|
|
_options = options || {};
|
|
_options.initialZoom = _options.initialZoom || 2;
|
|
_options.initialLat = _options.initialLat || 0;
|
|
_options.initialLng = _options.initialLng || 0;
|
|
_options.strokeColor = _options.strokeColor || "#fff";
|
|
_options.strokeWeight = _options.strokeWeight || 2;
|
|
_options.strokeOpacity = _options.strokeOpacity || 0.7;
|
|
_options.fillColor = _options.fillColor || "#82b92e";
|
|
_options.fillOpacity = _options.fillOpacity || 0.5;
|
|
_options.jsonRootUrl =
|
|
_options.jsonRootUrl || "include/javascript/tz_json/";
|
|
_options.date = _options.date || new Date();
|
|
|
|
_options.mapOptions = $.extend(
|
|
{
|
|
zoom: _options.initialZoom,
|
|
centerLat: _options.initialLat,
|
|
centerLng: _options.initialLng
|
|
},
|
|
_options.mapOptions
|
|
);
|
|
|
|
if (typeof _options.hoverRegions === "undefined") {
|
|
_options.hoverRegions = true;
|
|
}
|
|
|
|
if (_options.useOpenLayers) {
|
|
_mapper = new OpenLayersMapper(
|
|
_self.get(0),
|
|
mapClickHandler,
|
|
_options.hoverRegions ? mouseMoveHandler : null,
|
|
_options.mapOptions
|
|
);
|
|
}
|
|
|
|
// Load the necessary data files
|
|
var loadCount = _options.hoverRegions ? 2 : 1;
|
|
var checkLoading = function() {
|
|
loadCount--;
|
|
if (loadCount === 0) {
|
|
hideLoader();
|
|
|
|
if (_options.onReady) {
|
|
_options.onReady();
|
|
}
|
|
}
|
|
};
|
|
|
|
showLoader();
|
|
$.get(_options.jsonRootUrl + "bounding_boxes.json", function(data) {
|
|
_boundingBoxes = typeof data === "string" ? JSON.parse(data) : data;
|
|
$.each(_boundingBoxes, function(i, bb) {
|
|
$.extend(_zoneCentroids, bb.zoneCentroids);
|
|
});
|
|
checkLoading();
|
|
});
|
|
|
|
if (_options.hoverRegions) {
|
|
$.get(_options.jsonRootUrl + "hover_regions.json", function(data) {
|
|
var hoverData = typeof data === "string" ? JSON.parse(data) : data;
|
|
$.each(hoverData, function(i, v) {
|
|
_hoverRegions[v.name] = v;
|
|
});
|
|
checkLoading();
|
|
});
|
|
}
|
|
},
|
|
setDate: function(date) {
|
|
hideInfoWindow();
|
|
_options.date = date;
|
|
},
|
|
hideInfoWindow: hideInfoWindow,
|
|
showInfoWindow: function(content, callback) {
|
|
// showInfoWindow(content, callback);
|
|
},
|
|
selectZone: function(olsonName) {
|
|
var centroid = _zoneCentroids[olsonName];
|
|
if (centroid) {
|
|
mapClickHandler({
|
|
latLng: {
|
|
lat: function() {
|
|
return centroid[1];
|
|
},
|
|
lng: function() {
|
|
return centroid[0];
|
|
}
|
|
}
|
|
});
|
|
}
|
|
}
|
|
};
|
|
|
|
$.fn.timezonePicker = function(method) {
|
|
if (methods[method]) {
|
|
return methods[method].apply(
|
|
this,
|
|
Array.prototype.slice.call(arguments, 1)
|
|
);
|
|
} else if (typeof method === "object" || !method) {
|
|
return methods.init.apply(this, arguments);
|
|
} else {
|
|
$.error("Method " + method + " does not exist on jQuery.timezonePicker.");
|
|
}
|
|
};
|
|
|
|
_loaderGif =
|
|
"";
|
|
_maskPng =
|
|
"";
|
|
})(jQuery);
|