@grigorov-it/mazer
Version:
Free and Open-source Bootstrap 5 Admin Dashboard Template and Landing Page
1,419 lines (1,362 loc) • 76.9 kB
JavaScript
(function (global, factory) {
typeof exports === 'object' && typeof module !== 'undefined' ? module.exports = factory() :
typeof define === 'function' && define.amd ? define(factory) :
(global = typeof globalThis !== 'undefined' ? globalThis : global || self, global.jsVectorMap = factory());
})(this, (function () { 'use strict';
function _inheritsLoose(subClass, superClass) {
subClass.prototype = Object.create(superClass.prototype);
subClass.prototype.constructor = subClass;
_setPrototypeOf(subClass, superClass);
}
function _setPrototypeOf(o, p) {
_setPrototypeOf = Object.setPrototypeOf ? Object.setPrototypeOf.bind() : function _setPrototypeOf(o, p) {
o.__proto__ = p;
return o;
};
return _setPrototypeOf(o, p);
}
function _assertThisInitialized(self) {
if (self === void 0) {
throw new ReferenceError("this hasn't been initialised - super() hasn't been called");
}
return self;
}
function _readOnlyError(name) {
throw new TypeError("\"" + name + "\" is read-only");
}
function _unsupportedIterableToArray(o, minLen) {
if (!o) return;
if (typeof o === "string") return _arrayLikeToArray(o, minLen);
var n = Object.prototype.toString.call(o).slice(8, -1);
if (n === "Object" && o.constructor) n = o.constructor.name;
if (n === "Map" || n === "Set") return Array.from(o);
if (n === "Arguments" || /^(?:Ui|I)nt(?:8|16|32)(?:Clamped)?Array$/.test(n)) return _arrayLikeToArray(o, minLen);
}
function _arrayLikeToArray(arr, len) {
if (len == null || len > arr.length) len = arr.length;
for (var i = 0, arr2 = new Array(len); i < len; i++) arr2[i] = arr[i];
return arr2;
}
function _createForOfIteratorHelperLoose(o, allowArrayLike) {
var it = typeof Symbol !== "undefined" && o[Symbol.iterator] || o["@@iterator"];
if (it) return (it = it.call(o)).next.bind(it);
if (Array.isArray(o) || (it = _unsupportedIterableToArray(o)) || allowArrayLike && o && typeof o.length === "number") {
if (it) o = it;
var i = 0;
return function () {
if (i >= o.length) return {
done: true
};
return {
done: false,
value: o[i++]
};
};
}
throw new TypeError("Invalid attempt to iterate non-iterable instance.\nIn order to be iterable, non-array objects must have a [Symbol.iterator]() method.");
}
// Matches polyfill
// https://developer.mozilla.org/en-US/docs/Web/API/Element/matches
if (!Element.prototype.matches) {
Element.prototype.matches = Element.prototype.matchesSelector || Element.prototype.mozMatchesSelector || Element.prototype.msMatchesSelector || Element.prototype.oMatchesSelector || Element.prototype.webkitMatchesSelector || function (s) {
var matches = (this.document || this.ownerDocument).querySelectorAll(s);
var i = matches.length;
while ((_readOnlyError("i")) >= 0 && matches.item(i) !== this) {}
return i > -1;
};
}
// Object.assign polyfill
// https://gist.github.com/spiralx/68cf40d7010d829340cb
if (!Object.assign) {
Object.defineProperty(Object, 'assign', {
enumerable: false,
configurable: true,
writable: true,
value: function value(target) {
if (target === undefined || target === null) {
throw new TypeError('Cannot convert first argument to object');
}
var to = Object(target);
for (var i = 1; i < arguments.length; i++) {
var nextSource = arguments[i];
if (nextSource === undefined || nextSource === null) {
continue;
}
nextSource = Object(nextSource);
var keysArray = Object.keys(Object(nextSource));
for (var nextIndex = 0, len = keysArray.length; nextIndex < len; nextIndex++) {
var nextKey = keysArray[nextIndex];
var desc = Object.getOwnPropertyDescriptor(nextSource, nextKey);
if (desc !== undefined && desc.enumerable) {
to[nextKey] = nextSource[nextKey];
}
}
}
return to;
}
});
}
/**
* By https://github.com/TehShrike/deepmerge
*/
var isMergeableObject = function isMergeableObject(value) {
return isNonNullObject(value) && !isSpecial(value);
};
function isNonNullObject(value) {
return !!value && typeof value === 'object';
}
function isSpecial(value) {
var stringValue = Object.prototype.toString.call(value);
return stringValue === '[object RegExp]' || stringValue === '[object Date]' || isNode(value) || isReactElement(value);
}
// see https://github.com/facebook/react/blob/b5ac963fb791d1298e7f396236383bc955f916c1/src/isomorphic/classic/element/ReactElement.js#L21-L25
var canUseSymbol = typeof Symbol === 'function' && Symbol.for;
var REACT_ELEMENT_TYPE = canUseSymbol ? Symbol.for('react.element') : 0xeac7;
function isReactElement(value) {
return value.$$typeof === REACT_ELEMENT_TYPE;
}
function isNode(value) {
return value instanceof Node;
}
function emptyTarget(val) {
return Array.isArray(val) ? [] : {};
}
function cloneUnlessOtherwiseSpecified(value, options) {
return options.clone !== false && options.isMergeableObject(value) ? deepmerge(emptyTarget(value), value, options) : value;
}
function defaultArrayMerge(target, source, options) {
return target.concat(source).map(function (element) {
return cloneUnlessOtherwiseSpecified(element, options);
});
}
function getMergeFunction(key, options) {
if (!options.customMerge) {
return deepmerge;
}
var customMerge = options.customMerge(key);
return typeof customMerge === 'function' ? customMerge : deepmerge;
}
function getEnumerableOwnPropertySymbols(target) {
return Object.getOwnPropertySymbols ? Object.getOwnPropertySymbols(target).filter(function (symbol) {
return target.propertyIsEnumerable(symbol);
}) : [];
}
function getKeys(target) {
return Object.keys(target).concat(getEnumerableOwnPropertySymbols(target));
}
function propertyIsOnObject(object, property) {
try {
return property in object;
} catch (_) {
return false;
}
}
// Protects from prototype poisoning and unexpected merging up the prototype chain.
function propertyIsUnsafe(target, key) {
return propertyIsOnObject(target, key) // Properties are safe to merge if they don't exist in the target yet,
&& !(Object.hasOwnProperty.call(target, key) // unsafe if they exist up the prototype chain,
&& Object.propertyIsEnumerable.call(target, key)); // and also unsafe if they're nonenumerable.
}
function mergeObject(target, source, options) {
var destination = {};
if (options.isMergeableObject(target)) {
getKeys(target).forEach(function (key) {
destination[key] = cloneUnlessOtherwiseSpecified(target[key], options);
});
}
getKeys(source).forEach(function (key) {
if (propertyIsUnsafe(target, key)) {
return;
}
if (propertyIsOnObject(target, key) && options.isMergeableObject(source[key])) {
destination[key] = getMergeFunction(key, options)(target[key], source[key], options);
} else {
destination[key] = cloneUnlessOtherwiseSpecified(source[key], options);
}
});
return destination;
}
var deepmerge = function deepmerge(target, source, options) {
options = options || {};
options.arrayMerge = options.arrayMerge || defaultArrayMerge;
options.isMergeableObject = options.isMergeableObject || isMergeableObject;
// cloneUnlessOtherwiseSpecified is added to `options` so that custom arrayMerge()
// implementations can use it. The caller may not replace it.
options.cloneUnlessOtherwiseSpecified = cloneUnlessOtherwiseSpecified;
var sourceIsArray = Array.isArray(source);
var targetIsArray = Array.isArray(target);
var sourceAndTargetTypesMatch = sourceIsArray === targetIsArray;
if (!sourceAndTargetTypesMatch) {
return cloneUnlessOtherwiseSpecified(source, options);
} else if (sourceIsArray) {
return options.arrayMerge(target, source, options);
} else {
return mergeObject(target, source, options);
}
};
/**
* --------------------------------------------------------------------------
* Public Util Api
* --------------------------------------------------------------------------
*/
var getElement = function getElement(selector) {
if (typeof selector === 'object' && typeof selector.nodeType !== 'undefined') {
return selector;
}
if (typeof selector === 'string') {
return document.querySelector(selector);
}
return null;
};
var createElement = function createElement(type, classes, content, html) {
if (html === void 0) {
html = false;
}
var el = document.createElement(type);
if (content) {
el[!html ? 'textContent' : 'innerHTML'] = content;
}
if (classes) {
el.className = classes;
}
return el;
};
var findElement = function findElement(parentElement, selector) {
return Element.prototype.querySelector.call(parentElement, selector);
};
var removeElement = function removeElement(target) {
target.parentNode.removeChild(target);
};
var isImageUrl = function isImageUrl(url) {
return /\.(jpg|gif|png)$/.test(url);
};
var hyphenate = function hyphenate(string) {
return string.replace(/[\w]([A-Z])/g, function (m) {
return m[0] + "-" + m[1];
}).toLowerCase();
};
var merge = function merge(target, source, deep) {
if (deep === void 0) {
deep = false;
}
if (deep) {
return deepmerge(target, source);
}
return Object.assign(target, source);
};
var getLineUid = function getLineUid(from, to) {
return from.toLowerCase() + ":to:" + to.toLowerCase();
};
var inherit = function inherit(target, source) {
Object.assign(target.prototype, source);
};
var eventRegistry = {};
var eventUid = 1;
/**
* ------------------------------------------------------------------------
* Event Handler
* ------------------------------------------------------------------------
*/
var EventHandler = {
on: function on(element, event, handler, options) {
if (options === void 0) {
options = {};
}
var uid = "jvm:" + event + "::" + eventUid++;
eventRegistry[uid] = {
selector: element,
handler: handler
};
handler._uid = uid;
element.addEventListener(event, handler, options);
},
delegate: function delegate(element, event, selector, handler) {
event = event.split(' ');
event.forEach(function (eventName) {
EventHandler.on(element, eventName, function (e) {
var target = e.target;
if (target.matches(selector)) {
handler.call(target, e);
}
});
});
},
off: function off(element, event, handler) {
var eventType = event.split(':')[1];
element.removeEventListener(eventType, handler);
delete eventRegistry[handler._uid];
},
flush: function flush() {
Object.keys(eventRegistry).forEach(function (event) {
EventHandler.off(eventRegistry[event].selector, event, eventRegistry[event].handler);
});
},
getEventRegistry: function getEventRegistry() {
return eventRegistry;
}
};
function setupContainerEvents() {
var _this = this;
var map = this;
var mouseDown = false;
var oldPageX;
var oldPageY;
if (this.params.draggable) {
EventHandler.on(this.container, 'mousemove', function (e) {
if (!mouseDown) {
return false;
}
map.transX -= (oldPageX - e.pageX) / map.scale;
map.transY -= (oldPageY - e.pageY) / map.scale;
map._applyTransform();
oldPageX = e.pageX;
oldPageY = e.pageY;
});
EventHandler.on(this.container, 'mousedown', function (e) {
mouseDown = true;
oldPageX = e.pageX;
oldPageY = e.pageY;
return false;
});
EventHandler.on(document.body, 'mouseup', function () {
mouseDown = false;
});
}
if (this.params.zoomOnScroll) {
EventHandler.on(this.container, 'wheel', function (event) {
var deltaY = ((event.deltaY || -event.wheelDelta || event.detail) >> 10 || 1) * 75;
var rect = _this.container.getBoundingClientRect();
var offsetX = event.pageX - rect.left - window.pageXOffset;
var offsetY = event.pageY - rect.top - window.pageYOffset;
var zoomStep = Math.pow(1 + map.params.zoomOnScrollSpeed / 1000, -1.5 * deltaY);
if (map.tooltip) {
map._tooltip.hide();
}
map._setScale(map.scale * zoomStep, offsetX, offsetY);
event.preventDefault();
});
}
}
var Events = {
onLoaded: 'map:loaded',
onViewportChange: 'viewport:changed',
onRegionClick: 'region:clicked',
onMarkerClick: 'marker:clicked',
onRegionSelected: 'region:selected',
onMarkerSelected: 'marker:selected',
onRegionTooltipShow: 'region.tooltip:show',
onMarkerTooltipShow: 'marker.tooltip:show',
onDestroyed: 'map:destroyed'
};
var parseEvent = function parseEvent(map, selector, isTooltip) {
var element = getElement(selector);
var type = element.getAttribute('class').indexOf('jvm-region') === -1 ? 'marker' : 'region';
var isRegion = type === 'region';
var code = isRegion ? element.getAttribute('data-code') : element.getAttribute('data-index');
var event = isRegion ? Events.onRegionSelected : Events.onMarkerSelected;
// Init tooltip event
if (isTooltip) {
event = isRegion ? Events.onRegionTooltipShow : Events.onMarkerTooltipShow;
}
return {
type: type,
code: code,
event: event,
element: isRegion ? map.regions[code].element : map._markers[code].element,
tooltipText: isRegion ? map._mapData.paths[code].name || '' : map._markers[code].config.name || ''
};
};
function setupElementEvents() {
var map = this;
var container = this.container;
var pageX, pageY, mouseMoved;
EventHandler.on(container, 'mousemove', function (event) {
if (Math.abs(pageX - event.pageX) + Math.abs(pageY - event.pageY) > 2) {
mouseMoved = true;
}
});
// When the mouse is pressed
EventHandler.delegate(container, 'mousedown', '.jvm-element', function (event) {
pageX = event.pageX;
pageY = event.pageY;
mouseMoved = false;
});
// When the mouse is over the region/marker | When the mouse is out the region/marker
EventHandler.delegate(container, 'mouseover mouseout', '.jvm-element', function (event) {
var data = parseEvent(map, this, true);
var showTooltip = map.params.showTooltip;
if (event.type === 'mouseover') {
data.element.hover(true);
if (showTooltip) {
map._tooltip.text(data.tooltipText);
map._tooltip.show();
map._emit(data.event, [event, map._tooltip, data.code]);
}
} else {
data.element.hover(false);
if (showTooltip) {
map._tooltip.hide();
}
}
});
// When the click is released
EventHandler.delegate(container, 'mouseup', '.jvm-element', function (event) {
var data = parseEvent(map, this);
if (mouseMoved) {
return;
}
if (data.type === 'region' && map.params.regionsSelectable || data.type === 'marker' && map.params.markersSelectable) {
var element = data.element;
// We're checking if regions/markers|SelectableOne option is presented
if (map.params[data.type + "sSelectableOne"]) {
map._clearSelected(data.type + "s");
}
if (data.element.isSelected) {
element.select(false);
} else {
element.select(true);
}
map._emit(data.event, [data.code, element.isSelected, map._getSelected(data.type + "s")]);
}
});
// When region/marker is clicked
EventHandler.delegate(container, 'click', '.jvm-element', function (event) {
var _parseEvent = parseEvent(map, this),
type = _parseEvent.type,
code = _parseEvent.code;
map._emit(type === 'region' ? Events.onRegionClick : Events.onMarkerClick, [event, code]);
});
}
function setupZoomButtons() {
var _this = this;
var zoomin = createElement('div', 'jvm-zoom-btn jvm-zoomin', '+', true);
var zoomout = createElement('div', 'jvm-zoom-btn jvm-zoomout', '−', true);
this.container.appendChild(zoomin);
this.container.appendChild(zoomout);
var handler = function handler(zoomin) {
if (zoomin === void 0) {
zoomin = true;
}
return function () {
return _this._setScale(zoomin ? _this.scale * _this.params.zoomStep : _this.scale / _this.params.zoomStep, _this._width / 2, _this._height / 2, false, _this.params.zoomAnimate);
};
};
EventHandler.on(zoomin, 'click', handler());
EventHandler.on(zoomout, 'click', handler(false));
}
function setupContainerTouchEvents() {
var map = this,
touchStartScale,
touchStartDistance,
touchX,
touchY,
centerTouchX,
centerTouchY,
lastTouchesLength;
var handleTouchEvent = function handleTouchEvent(e) {
var touches = e.touches;
var offset, scale, transXOld, transYOld;
if (e.type == 'touchstart') {
lastTouchesLength = 0;
}
if (touches.length == 1) {
if (lastTouchesLength == 1) {
transXOld = map.transX;
transYOld = map.transY;
map.transX -= (touchX - touches[0].pageX) / map.scale;
map.transY -= (touchY - touches[0].pageY) / map.scale;
map._tooltip.hide();
map._applyTransform();
if (transXOld != map.transX || transYOld != map.transY) {
e.preventDefault();
}
}
touchX = touches[0].pageX;
touchY = touches[0].pageY;
} else if (touches.length == 2) {
if (lastTouchesLength == 2) {
scale = Math.sqrt(Math.pow(touches[0].pageX - touches[1].pageX, 2) + Math.pow(touches[0].pageY - touches[1].pageY, 2)) / touchStartDistance;
map._setScale(touchStartScale * scale, centerTouchX, centerTouchY);
map._tooltip.hide();
e.preventDefault();
} else {
var rect = map.container.getBoundingClientRect();
offset = {
top: rect.top + window.scrollY,
left: rect.left + window.scrollX
};
if (touches[0].pageX > touches[1].pageX) {
centerTouchX = touches[1].pageX + (touches[0].pageX - touches[1].pageX) / 2;
} else {
centerTouchX = touches[0].pageX + (touches[1].pageX - touches[0].pageX) / 2;
}
if (touches[0].pageY > touches[1].pageY) {
centerTouchY = touches[1].pageY + (touches[0].pageY - touches[1].pageY) / 2;
} else {
centerTouchY = touches[0].pageY + (touches[1].pageY - touches[0].pageY) / 2;
}
centerTouchX -= offset.left;
centerTouchY -= offset.top;
touchStartScale = map.scale;
touchStartDistance = Math.sqrt(Math.pow(touches[0].pageX - touches[1].pageX, 2) + Math.pow(touches[0].pageY - touches[1].pageY, 2));
}
}
lastTouchesLength = touches.length;
};
EventHandler.on(map.container, 'touchstart', handleTouchEvent);
EventHandler.on(map.container, 'touchmove', handleTouchEvent);
}
/**
* ------------------------------------------------------------------------
* Class Definition
* ------------------------------------------------------------------------
*/
var BaseComponent = /*#__PURE__*/function () {
function BaseComponent() {}
var _proto = BaseComponent.prototype;
_proto.dispose = function dispose() {
if (this._tooltip) {
removeElement(this._tooltip);
} else {
// @todo: move shape in base component in v2
this.shape.remove();
}
for (var _iterator = _createForOfIteratorHelperLoose(Object.getOwnPropertyNames(this)), _step; !(_step = _iterator()).done;) {
var propertyName = _step.value;
this[propertyName] = null;
}
};
return BaseComponent;
}();
/**
* ------------------------------------------------------------------------
* Interactable
* ------------------------------------------------------------------------
*/
var Interactable = {
getLabelText: function getLabelText(key, label) {
if (!label) {
return;
}
if (typeof label.render === 'function') {
var params = [];
// Pass additional paramater (Marker config object) in case it's a Marker.
if (this.config && this.config.marker) {
params.push(this.config.marker);
}
// Becuase we need to add the key always at the end
params.push(key);
return label.render.apply(this, params);
}
return key;
},
getLabelOffsets: function getLabelOffsets(key, label) {
if (typeof label.offsets === 'function') {
return label.offsets(key);
}
// If offsets are an array of offsets e.g offsets: [ [0, 25], [10, 15] ]
if (Array.isArray(label.offsets)) {
return label.offsets[key];
}
return [0, 0];
},
setStyle: function setStyle(property, value) {
this.shape.setStyle(property, value);
},
remove: function remove() {
this.shape.remove();
if (this.label) this.label.remove();
},
hover: function hover(state) {
this._setStatus('isHovered', state);
},
select: function select(state) {
this._setStatus('isSelected', state);
},
// Private
_setStatus: function _setStatus(property, state) {
this.shape[property] = state;
this.shape.updateStyle();
this[property] = state;
if (this.label) {
this.label[property] = state;
this.label.updateStyle();
}
}
};
/**
* ------------------------------------------------------------------------
* Class Definition
* ------------------------------------------------------------------------
*/
var Region = /*#__PURE__*/function (_BaseComponent) {
_inheritsLoose(Region, _BaseComponent);
function Region(_ref) {
var _this;
var map = _ref.map,
code = _ref.code,
path = _ref.path,
style = _ref.style,
label = _ref.label,
labelStyle = _ref.labelStyle,
labelsGroup = _ref.labelsGroup;
_this = _BaseComponent.call(this) || this;
_this._map = map;
_this.shape = _this._createRegion(path, code, style);
var text = _this.getLabelText(code, label);
// If label is passed and render function returns something
if (label && text) {
var bbox = _this.shape.getBBox();
var offsets = _this.getLabelOffsets(code, label);
_this.labelX = bbox.x + bbox.width / 2 + offsets[0];
_this.labelY = bbox.y + bbox.height / 2 + offsets[1];
_this.label = _this._map.canvas.createText({
text: text,
textAnchor: 'middle',
alignmentBaseline: 'central',
dataCode: code,
x: _this.labelX,
y: _this.labelY
}, labelStyle, labelsGroup);
_this.label.addClass('jvm-region jvm-element');
}
return _this;
}
var _proto = Region.prototype;
_proto._createRegion = function _createRegion(path, code, style) {
path = this._map.canvas.createPath({
d: path,
dataCode: code
}, style);
path.addClass('jvm-region jvm-element');
return path;
};
_proto.updateLabelPosition = function updateLabelPosition() {
if (this.label) {
this.label.set({
x: this.labelX * this._map.scale + this._map.transX * this._map.scale,
y: this.labelY * this._map.scale + this._map.transY * this._map.scale
});
}
};
return Region;
}(BaseComponent);
inherit(Region, Interactable);
function createRegions() {
this._regionLabelsGroup = this._regionLabelsGroup || this.canvas.createGroup('jvm-regions-labels-group');
for (var code in this._mapData.paths) {
var region = new Region({
map: this,
code: code,
path: this._mapData.paths[code].path,
style: merge({}, this.params.regionStyle),
labelStyle: this.params.regionLabelStyle,
labelsGroup: this._regionLabelsGroup,
label: this.params.labels && this.params.labels.regions
});
this.regions[code] = {
config: this._mapData.paths[code],
element: region
};
}
}
/**
* ------------------------------------------------------------------------
* Class Definition
* ------------------------------------------------------------------------
*/
var Line = /*#__PURE__*/function (_BaseComponent) {
_inheritsLoose(Line, _BaseComponent);
function Line(_ref) {
var _this;
var index = _ref.index,
map = _ref.map,
style = _ref.style,
x1 = _ref.x1,
y1 = _ref.y1,
x2 = _ref.x2,
y2 = _ref.y2,
group = _ref.group,
config = _ref.config;
_this = _BaseComponent.call(this) || this;
_this.config = config;
_this.shape = map.canvas.createLine({
x1: x1,
y1: y1,
x2: x2,
y2: y2,
dataIndex: index
}, style, group);
_this.shape.addClass('jvm-line');
return _this;
}
var _proto = Line.prototype;
_proto.setStyle = function setStyle(property, value) {
this.shape.setStyle(property, value);
};
return Line;
}(BaseComponent);
function createLines(lines, markers, isRecentlyCreated) {
if (isRecentlyCreated === void 0) {
isRecentlyCreated = false;
}
var point1 = false,
point2 = false;
// Create group for holding lines
// we're checking if `linesGroup` exists or not becuase we may add lines
// after the map has loaded so we will append the futured lines to this group as well.
this.linesGroup = this.linesGroup || this.canvas.createGroup('jvm-lines-group');
for (var index in lines) {
var config = lines[index];
for (var mindex in markers) {
var markerConfig = isRecentlyCreated ? markers[mindex].config : markers[mindex];
if (markerConfig.name === config.from) {
point1 = this.getMarkerPosition(markerConfig);
}
if (markerConfig.name === config.to) {
point2 = this.getMarkerPosition(markerConfig);
}
}
if (point1 !== false && point2 !== false) {
// Register lines with unique keys
this._lines[getLineUid(config.from, config.to)] = new Line({
index: index,
map: this,
// Merge the default `lineStyle` object with the custom `line` config style
style: merge({
initial: this.params.lineStyle
}, {
initial: config.style || {}
}, true),
x1: point1.x,
y1: point1.y,
x2: point2.x,
y2: point2.y,
group: this.linesGroup,
config: config
});
}
}
}
/**
* ------------------------------------------------------------------------
* Class Definition
* ------------------------------------------------------------------------
*/
var Marker = /*#__PURE__*/function (_BaseComponent) {
_inheritsLoose(Marker, _BaseComponent);
function Marker(_ref) {
var _this;
var index = _ref.index,
style = _ref.style,
label = _ref.label,
cx = _ref.cx,
cy = _ref.cy,
map = _ref.map,
group = _ref.group;
_this = _BaseComponent.call(this) || this;
// Private
_this._map = map;
_this._isImage = !!style.initial.image;
// Protected
_this.config = arguments[0];
_this.shape = map.canvas[_this._isImage ? 'createImage' : 'createCircle']({
dataIndex: index,
cx: cx,
cy: cy
}, style, group);
_this.shape.addClass('jvm-marker jvm-element');
if (_this._isImage) {
_this.updateLabelPosition();
}
if (label) {
_this._createLabel(_this.config);
}
return _this;
}
var _proto = Marker.prototype;
_proto.updateLabelPosition = function updateLabelPosition() {
if (this.label) {
this.label.set({
x: this._labelX * this._map.scale + this._offsets[0] + this._map.transX * this._map.scale + 5 + (this._isImage ? (this.shape.width || 0) / 2 : this.shape.node.r.baseVal.value),
y: this._labelY * this._map.scale + this._map.transY * this._map.scale + this._offsets[1]
});
}
};
_proto._createLabel = function _createLabel(_ref2) {
var index = _ref2.index,
map = _ref2.map,
label = _ref2.label,
labelsGroup = _ref2.labelsGroup,
cx = _ref2.cx,
cy = _ref2.cy,
marker = _ref2.marker,
isRecentlyCreated = _ref2.isRecentlyCreated;
var labelText = this.getLabelText(index, label);
this._labelX = cx / map.scale - map.transX;
this._labelY = cy / map.scale - map.transY;
this._offsets = isRecentlyCreated && marker.offsets ? marker.offsets : this.getLabelOffsets(index, label);
this.label = map.canvas.createText({
text: labelText,
dataIndex: index,
x: this._labelX,
y: this._labelY,
dy: '0.6ex'
}, map.params.markerLabelStyle, labelsGroup);
this.label.addClass('jvm-marker jvm-element');
if (isRecentlyCreated) {
this.updateLabelPosition();
}
};
return Marker;
}(BaseComponent);
inherit(Marker, Interactable);
function createMarkers(markers, isRecentlyCreated) {
var _this = this;
if (markers === void 0) {
markers = {};
}
if (isRecentlyCreated === void 0) {
isRecentlyCreated = false;
}
// Create groups for holding markers and markers labels
// We're checking if `markersGroup` exists or not becuase we may add markers after the map has loaded
// So we will append the futured markers to this group as well.
this._markersGroup = this._markersGroup || this.canvas.createGroup('jvm-markers-group');
this._markerLabelsGroup = this._markerLabelsGroup || this.canvas.createGroup('jvm-markers-labels-group');
var _loop = function _loop() {
var config = markers[index];
var point = _this.getMarkerPosition(config);
var uid = config.coords.join(':');
if (!point) {
return "continue";
}
// We're checking if recently created marker does already exist
// If it does we don't need to create it again, so we'll continue
// Becuase we may have more than one marker submitted via `addMarkers` method.
if (isRecentlyCreated) {
if (Object.keys(_this._markers).filter(function (i) {
return _this._markers[i]._uid === uid;
}).length) {
return "continue";
}
index = Object.keys(_this._markers).length;
}
var marker = new Marker({
index: index,
map: _this,
// Merge the `markerStyle` object with the marker config `style` if presented.
style: merge(_this.params.markerStyle, {
initial: config.style || {}
}, true),
label: _this.params.labels && _this.params.labels.markers,
labelsGroup: _this._markerLabelsGroup,
cx: point.x,
cy: point.y,
group: _this._markersGroup,
marker: config,
isRecentlyCreated: isRecentlyCreated
});
// Check for marker duplication
// this is useful when for example: a user clicks a button for creating marker two times
// so it will remove the old one and the new one will take its place.
if (_this._markers[index]) {
_this.removeMarkers([index]);
}
_this._markers[index] = {
_uid: uid,
config: config,
element: marker
};
};
for (var index in markers) {
var _ret = _loop();
if (_ret === "continue") continue;
}
}
/**
* ------------------------------------------------------------------------
* Class Definition
* ------------------------------------------------------------------------
*/
var Legend = /*#__PURE__*/function () {
function Legend(options) {
if (options === void 0) {
options = {};
}
this._options = options;
this._map = this._options.map;
this._series = this._options.series;
this._body = createElement('div', 'jvm-legend');
if (this._options.cssClass) {
this._body.setAttribute('class', this._options.cssClass);
}
if (options.vertical) {
this._map.legendVertical.appendChild(this._body);
} else {
this._map.legendHorizontal.appendChild(this._body);
}
this.render();
}
var _proto = Legend.prototype;
_proto.render = function render() {
var ticks = this._series.scale.getTicks();
var inner = createElement('div', 'jvm-legend-inner');
this._body.innderHTML = '';
if (this._options.title) {
var legendTitle = createElement('div', 'jvm-legend-title', this._options.title);
this._body.appendChild(legendTitle);
}
this._body.appendChild(inner);
for (var i = 0; i < ticks.length; i++) {
var tick = createElement('div', 'jvm-legend-tick');
var sample = createElement('div', 'jvm-legend-tick-sample');
switch (this._series.config.attribute) {
case 'fill':
if (isImageUrl(ticks[i].value)) {
sample.style.background = "url(" + ticks[i].value + ")";
} else {
sample.style.background = ticks[i].value;
}
break;
case 'stroke':
sample.style.background = ticks[i].value;
break;
case 'image':
sample.style.background = "url(" + (typeof ticks[i].value === 'object' ? ticks[i].value.url : ticks[i].value) + ") no-repeat center center";
sample.style.backgroundSize = 'cover';
break;
}
tick.appendChild(sample);
var label = ticks[i].label;
if (this._options.labelRender) {
label = this._options.labelRender(label);
}
var tickText = createElement('div', 'jvm-legend-tick-text', label);
tick.appendChild(tickText);
inner.appendChild(tick);
}
};
return Legend;
}();
/**
* ------------------------------------------------------------------------
* Class Definition
* ------------------------------------------------------------------------
*/
var OrdinalScale = /*#__PURE__*/function () {
function OrdinalScale(scale) {
this._scale = scale;
}
var _proto = OrdinalScale.prototype;
_proto.getValue = function getValue(value) {
return this._scale[value];
};
_proto.getTicks = function getTicks() {
var ticks = [];
for (var key in this._scale) {
ticks.push({
label: key,
value: this._scale[key]
});
}
return ticks;
};
return OrdinalScale;
}();
/**
* ------------------------------------------------------------------------
* Class Definition
* ------------------------------------------------------------------------
*/
var Series = /*#__PURE__*/function () {
function Series(config, elements, map) {
if (config === void 0) {
config = {};
}
// Private
this._map = map;
this._elements = elements; // Could be markers or regions
this._values = config.values || {};
// Protected
this.config = config;
this.config.attribute = config.attribute || 'fill';
// Set initial attributes
if (config.attributes) {
this.setAttributes(config.attributes);
}
if (typeof config.scale === 'object') {
this.scale = new OrdinalScale(config.scale);
}
if (this.config.legend) {
this.legend = new Legend(merge({
map: this._map,
series: this
}, this.config.legend));
}
this.setValues(this._values);
}
var _proto = Series.prototype;
_proto.setValues = function setValues(values) {
var attrs = {};
for (var key in values) {
if (values[key]) {
attrs[key] = this.scale.getValue(values[key]);
}
}
this.setAttributes(attrs);
};
_proto.setAttributes = function setAttributes(attrs) {
for (var code in attrs) {
if (this._elements[code]) {
this._elements[code].element.setStyle(this.config.attribute, attrs[code]);
}
}
};
_proto.clear = function clear() {
var key,
attrs = {};
for (key in this._values) {
if (this._elements[key]) {
attrs[key] = this._elements[key].element.shape.style.initial[this.config.attribute];
}
}
this.setAttributes(attrs);
this._values = {};
};
return Series;
}();
function createSeries() {
this.series = {
markers: [],
regions: []
};
for (var key in this.params.series) {
for (var i = 0; i < this.params.series[key].length; i++) {
this.series[key][i] = new Series(this.params.series[key][i], key === 'markers' ? this._markers : this.regions, this);
}
}
}
function applyTransform() {
var maxTransX, maxTransY, minTransX, minTransY;
if (this._defaultWidth * this.scale <= this._width) {
maxTransX = (this._width - this._defaultWidth * this.scale) / (2 * this.scale);
minTransX = (this._width - this._defaultWidth * this.scale) / (2 * this.scale);
} else {
maxTransX = 0;
minTransX = (this._width - this._defaultWidth * this.scale) / this.scale;
}
if (this._defaultHeight * this.scale <= this._height) {
maxTransY = (this._height - this._defaultHeight * this.scale) / (2 * this.scale);
minTransY = (this._height - this._defaultHeight * this.scale) / (2 * this.scale);
} else {
maxTransY = 0;
minTransY = (this._height - this._defaultHeight * this.scale) / this.scale;
}
if (this.transY > maxTransY) {
this.transY = maxTransY;
} else if (this.transY < minTransY) {
this.transY = minTransY;
}
if (this.transX > maxTransX) {
this.transX = maxTransX;
} else if (this.transX < minTransX) {
this.transX = minTransX;
}
this.canvas.applyTransformParams(this.scale, this.transX, this.transY);
if (this._markers) {
this._repositionMarkers();
}
if (this._lines) {
this._repositionLines();
}
this._repositionLabels();
}
function resize() {
var curBaseScale = this._baseScale;
if (this._width / this._height > this._defaultWidth / this._defaultHeight) {
this._baseScale = this._height / this._defaultHeight;
this._baseTransX = Math.abs(this._width - this._defaultWidth * this._baseScale) / (2 * this._baseScale);
} else {
this._baseScale = this._width / this._defaultWidth;
this._baseTransY = Math.abs(this._height - this._defaultHeight * this._baseScale) / (2 * this._baseScale);
}
this.scale *= this._baseScale / curBaseScale;
this.transX *= this._baseScale / curBaseScale;
this.transY *= this._baseScale / curBaseScale;
}
function setScale(scale, anchorX, anchorY, isCentered, animate) {
var _this = this;
var zoomStep,
interval,
i = 0,
count = Math.abs(Math.round((scale - this.scale) * 60 / Math.max(scale, this.scale))),
scaleStart,
scaleDiff,
transXStart,
transXDiff,
transYStart,
transYDiff,
transX,
transY;
if (scale > this.params.zoomMax * this._baseScale) {
scale = this.params.zoomMax * this._baseScale;
} else if (scale < this.params.zoomMin * this._baseScale) {
scale = this.params.zoomMin * this._baseScale;
}
if (typeof anchorX != 'undefined' && typeof anchorY != 'undefined') {
zoomStep = scale / this.scale;
if (isCentered) {
transX = anchorX + this._defaultWidth * (this._width / (this._defaultWidth * scale)) / 2;
transY = anchorY + this._defaultHeight * (this._height / (this._defaultHeight * scale)) / 2;
} else {
transX = this.transX - (zoomStep - 1) / scale * anchorX;
transY = this.transY - (zoomStep - 1) / scale * anchorY;
}
}
if (animate && count > 0) {
scaleStart = this.scale;
scaleDiff = (scale - scaleStart) / count;
transXStart = this.transX * this.scale;
transYStart = this.transY * this.scale;
transXDiff = (transX * scale - transXStart) / count;
transYDiff = (transY * scale - transYStart) / count;
interval = setInterval(function () {
i += 1;
_this.scale = scaleStart + scaleDiff * i;
_this.transX = (transXStart + transXDiff * i) / _this.scale;
_this.transY = (transYStart + transYDiff * i) / _this.scale;
_this._applyTransform();
if (i == count) {
clearInterval(interval);
_this._emit(Events.onViewportChange, [_this.scale, _this.transX, _this.transY]);
}
}, 10);
} else {
this.transX = transX;
this.transY = transY;
this.scale = scale;
this._applyTransform();
this._emit(Events.onViewportChange, [this.scale, this.transX, this.transY]);
}
}
function setFocus(config) {
var _this = this;
if (config === void 0) {
config = {};
}
var bbox,
codes = [];
if (config.region) {
codes.push(config.region);
} else if (config.regions) {
codes = config.regions;
}
if (codes.length) {
codes.forEach(function (code) {
if (_this.regions[code]) {
var itemBbox = _this.regions[code].element.shape.getBBox();
if (itemBbox) {
// Handle the first loop
if (typeof bbox == 'undefined') {
bbox = itemBbox;
} else {
// get the old bbox properties plus the current
// this kinda incrementing the old values and the new values
bbox = {
x: Math.min(bbox.x, itemBbox.x),
y: Math.min(bbox.y, itemBbox.y),
width: Math.max(bbox.x + bbox.width, itemBbox.x + itemBbox.width) - Math.min(bbox.x, itemBbox.x),
height: Math.max(bbox.y + bbox.height, itemBbox.y + itemBbox.height) - Math.min(bbox.y, itemBbox.y)
};
}
}
}
});
return this._setScale(Math.min(this._width / bbox.width, this._height / bbox.height), -(bbox.x + bbox.width / 2), -(bbox.y + bbox.height / 2), true, config.animate);
} else if (config.coords) {
var point = this.coordsToPoint(config.coords[0], config.coords[1]);
var x = this.transX - point.x / this.scale;
var y = this.transY - point.y / this.scale;
return this._setScale(config.scale * this._baseScale, x, y, true, config.animate);
}
}
function updateSize() {
this._width = this.container.offsetWidth;
this._height = this.container.offsetHeight;
this._resize();
this.canvas.setSize(this._width, this._height);
this._applyTransform();
}
/**
* ------------------------------------------------------------------------
* Object
* ------------------------------------------------------------------------
*/
var Proj = {
/* sgn(n){
if (n > 0) {
return 1;
} else if (n < 0) {
return -1;
} else {
return n;
}
}, */
mill: function mill(lat, lng, c) {
return {
x: this.radius * (lng - c) * this.radDeg,
y: -this.radius * Math.log(Math.tan((45 + 0.4 * lat) * this.radDeg)) / 0.8
};
},
/* mill_inv(x, y, c) {
return {
lat: (2.5 * Math.atan(Math.exp(0.8 * y / this.radius)) - 5 * Math.PI / 8) * this.degRad,
lng: (c * this.radDeg + x / this.radius) * this.degRad
};
}, */
merc: function merc(lat, lng, c) {
return {
x: this.radius * (lng - c) * this.radDeg,
y: -this.radius * Math.log(Math.tan(Math.PI / 4 + lat * Math.PI / 360))
};
},
/* merc_inv(x, y, c) {
return {
lat: (2 * Math.atan(Math.exp(y / this.radius)) - Math.PI / 2) * this.degRad,
lng: (c * this.radDeg + x / this.radius) * this.degRad
};
}, */
aea: function aea(lat, lng, c) {
var fi0 = 0,
lambda0 = c * this.radDeg,
fi1 = 29.5 * this.radDeg,
fi2 = 45.5 * this.radDeg,
fi = lat * this.radDeg,
lambda = lng * this.radDeg,
n = (Math.sin(fi1) + Math.sin(fi2)) / 2,
C = Math.cos(fi1) * Math.cos(fi1) + 2 * n * Math.sin(fi1),
theta = n * (lambda - lambda0),
ro = Math.sqrt(C - 2 * n * Math.sin(fi)) / n,
ro0 = Math.sqrt(C - 2 * n * Math.sin(fi0)) / n;
return {
x: ro * Math.sin(theta) * this.radius,
y: -(ro0 - ro * Math.cos(theta)) * this.radius
};
},
/* aea_inv(xCoord, yCoord, c) {
var x = xCoord / this.radius,
y = yCoord / this.radius,
fi0 = 0,
lambda0 = c * this.radDeg,
fi1 = 29.5 * this.radDeg,
fi2 = 45.5 * this.radDeg,
n = (Math.sin(fi1)+Math.sin(fi2)) / 2,
C = Math.cos(fi1)*Math.cos(fi1)+2*n*Math.sin(fi1),
ro0 = Math.sqrt(C-2*n*Math.sin(fi0))/n,
ro = Math.sqrt(x*x+(ro0-y)*(ro0-y)),
theta = Math.atan( x / (ro0 - y) );
return {
lat: (Math.asin((C - ro * ro * n * n) / (2 * n))) * this.degRad,
lng: (lambda0 + theta / n) * this.degRad
};
}, */
lcc: function lcc(lat, lng, c) {
var fi0 = 0,
lambda0 = c * this.radDeg,
lambda = lng * this.radDeg,
fi1 = 33 * this.radDeg,
fi2 = 45 * this.radDeg,
fi = lat * this.radDeg,
n = Math.log(Math.cos(fi1) * (1 / Math.cos(fi2))) / Math.log(Math.tan(Math.PI / 4 + fi2 / 2) * (1 / Math.tan(Math.PI / 4 + fi1 / 2))),
F = Math.cos(fi1) * Math.pow(Math.tan(Math.PI / 4 + fi1 / 2), n) / n,
ro = F * Math.pow(1 / Math.tan(Math.PI / 4 + fi / 2), n),
ro0 = F * Math.pow(1 / Math.tan(Math.PI / 4 + fi0 / 2), n);
return {
x: ro * Math.sin(n * (lambda - lambda0)) * this.radius,
y: -(ro0 - ro * Math.cos(n * (lambda - lambda0))) * this.radius
};
}
/* lcc_inv(xCoord, yCoord, c) {
var x = xCoord / this.radius,
y = yCoord / this.radius,
fi0 = 0,
lambda0 = c * this.radDeg,
fi1 = 33 * this.radDeg,
fi2 = 45 * this.radDeg,
n = Math.log( Math.cos(fi1) * (1 / Math.cos(fi2)) ) / Math.log( Math.tan( Math.PI / 4 + fi2 / 2) * (1 / Math.tan( Math.PI / 4 + fi1 / 2) ) ),
F = ( Math.cos(fi1) * Math.pow( Math.tan( Math.PI / 4 + fi1 / 2 ), n ) ) / n,
ro0 = F * Math.pow( 1 / Math.tan( Math.PI / 4 + fi0 / 2 ), n ),
ro = this.sgn(n) * Math.sqrt(x*x+(ro0-y)*(ro0-y)),
theta = Math.atan( x / (ro0 - y) );
return {
lat: (2 * Math.atan(Math.pow(F/ro, 1/n)) - Math.PI / 2) * this.degRad,
lng: (lambda0 + theta / n) * this.degRad
};
} */
};
Proj.degRad = 180 / Math.PI;
Proj.radDeg = Math.PI / 180;
Proj.radius = 6381372;
function coordsToPoint(lat, lng) {
var projection = Map.maps[this.params.map].projection;
var _Proj$projection$type = Proj[projection.type](lat, lng, projection.centralMeridian),
x = _Proj$projection$type.x,
y = _Proj$projection$type.y;
var inset = this.getInsetForPoint(x, y);
if (!inset) {
return false;
}
var bbox = inset.bbox;
x = (x - bbox[0].x) / (bbox[1].x - bbox[0].x) * inset.width * this.scale;
y = (y - bbox[0].y) / (bbox[1].y - bbox[0].y) * inset.height * this.scale;
return {
x: x + this.transX * this.scale + inset.left * this.scale,
y: y + this.transY * this.scale + inset.top * this.scale
};
}
function getInsetForPoint(x, y) {
var insets = Map.maps[this.params.map].insets;
for (var index = 0; index < insets.length; index++) {
var _insets$index$bbox = insets[index].bbox,
start = _insets$index$bbox[0],
end = _insets$index$bbox[1];
if (x > start.x && x < end.x && y > start.y && y < end.y) {
return insets[index];
}
}
}
function getMarkerPosition(_ref) {
var coords = _ref.coords;
if (Map.maps[this.params.map].projection) {
return this.coordsToPoint.apply(this, coords);
}
return {
x: coords[0] * this.scale + this.transX * this.scale,
y: coords[1] * this.scale + this.transY * this.scale
};
}
function repositionLines() {
var point1 = false,
point2 = false;
for (var index in this._lines) {
for (var mindex in this._markers) {
var marker = this._markers[mindex];
if (marker.config.name === this._lines[index].config.from) {
point1 = this.getMarkerPosition(marker.config);
}
if (marker.config.name === this._lines[index].config.to) {
point2 = this.getMarkerPosition(marker.config);
}
}
if (point1 !== false && point2 !== false) {
this._lines[index].setStyle({
x1: point1.x,
y1: point1.y,
x2: point2.x,
y2: point2.y
});
}
}
}
funct