diff --git a/app/assets/javascripts/spotlight/spotlight.esm.js b/app/assets/javascripts/spotlight/spotlight.esm.js
index bc73d2af57..401286b53f 100644
--- a/app/assets/javascripts/spotlight/spotlight.esm.js
+++ b/app/assets/javascripts/spotlight/spotlight.esm.js
@@ -5135,6 +5135,7 @@ jQuery.fn.scrollStop = function(callback) {
};
// Place all the behaviors and hooks related to the matching controller here.
+// All this logic will automatically be available in application.js.
class Pages {
connect(){
@@ -5175,23 +5176,539 @@ class Pages {
}
}
+var adapters = {
+ logger: typeof console !== "undefined" ? console : undefined,
+ WebSocket: typeof WebSocket !== "undefined" ? WebSocket : undefined
+};
+
+var logger = {
+ log(...messages) {
+ if (this.enabled) {
+ messages.push(Date.now());
+ adapters.logger.log("[ActionCable]", ...messages);
+ }
+ }
+};
+
+const now = () => (new Date).getTime();
+
+const secondsSince = time => (now() - time) / 1e3;
+
+class ConnectionMonitor {
+ constructor(connection) {
+ this.visibilityDidChange = this.visibilityDidChange.bind(this);
+ this.connection = connection;
+ this.reconnectAttempts = 0;
+ }
+ start() {
+ if (!this.isRunning()) {
+ this.startedAt = now();
+ delete this.stoppedAt;
+ this.startPolling();
+ addEventListener("visibilitychange", this.visibilityDidChange);
+ logger.log(`ConnectionMonitor started. stale threshold = ${this.constructor.staleThreshold} s`);
+ }
+ }
+ stop() {
+ if (this.isRunning()) {
+ this.stoppedAt = now();
+ this.stopPolling();
+ removeEventListener("visibilitychange", this.visibilityDidChange);
+ logger.log("ConnectionMonitor stopped");
+ }
+ }
+ isRunning() {
+ return this.startedAt && !this.stoppedAt;
+ }
+ recordMessage() {
+ this.pingedAt = now();
+ }
+ recordConnect() {
+ this.reconnectAttempts = 0;
+ delete this.disconnectedAt;
+ logger.log("ConnectionMonitor recorded connect");
+ }
+ recordDisconnect() {
+ this.disconnectedAt = now();
+ logger.log("ConnectionMonitor recorded disconnect");
+ }
+ startPolling() {
+ this.stopPolling();
+ this.poll();
+ }
+ stopPolling() {
+ clearTimeout(this.pollTimeout);
+ }
+ poll() {
+ this.pollTimeout = setTimeout((() => {
+ this.reconnectIfStale();
+ this.poll();
+ }), this.getPollInterval());
+ }
+ getPollInterval() {
+ const {staleThreshold: staleThreshold, reconnectionBackoffRate: reconnectionBackoffRate} = this.constructor;
+ const backoff = Math.pow(1 + reconnectionBackoffRate, Math.min(this.reconnectAttempts, 10));
+ const jitterMax = this.reconnectAttempts === 0 ? 1 : reconnectionBackoffRate;
+ const jitter = jitterMax * Math.random();
+ return staleThreshold * 1e3 * backoff * (1 + jitter);
+ }
+ reconnectIfStale() {
+ if (this.connectionIsStale()) {
+ logger.log(`ConnectionMonitor detected stale connection. reconnectAttempts = ${this.reconnectAttempts}, time stale = ${secondsSince(this.refreshedAt)} s, stale threshold = ${this.constructor.staleThreshold} s`);
+ this.reconnectAttempts++;
+ if (this.disconnectedRecently()) {
+ logger.log(`ConnectionMonitor skipping reopening recent disconnect. time disconnected = ${secondsSince(this.disconnectedAt)} s`);
+ } else {
+ logger.log("ConnectionMonitor reopening");
+ this.connection.reopen();
+ }
+ }
+ }
+ get refreshedAt() {
+ return this.pingedAt ? this.pingedAt : this.startedAt;
+ }
+ connectionIsStale() {
+ return secondsSince(this.refreshedAt) > this.constructor.staleThreshold;
+ }
+ disconnectedRecently() {
+ return this.disconnectedAt && secondsSince(this.disconnectedAt) < this.constructor.staleThreshold;
+ }
+ visibilityDidChange() {
+ if (document.visibilityState === "visible") {
+ setTimeout((() => {
+ if (this.connectionIsStale() || !this.connection.isOpen()) {
+ logger.log(`ConnectionMonitor reopening stale connection on visibilitychange. visibilityState = ${document.visibilityState}`);
+ this.connection.reopen();
+ }
+ }), 200);
+ }
+ }
+}
+
+ConnectionMonitor.staleThreshold = 6;
+
+ConnectionMonitor.reconnectionBackoffRate = .15;
+
+var INTERNAL = {
+ message_types: {
+ welcome: "welcome",
+ disconnect: "disconnect",
+ ping: "ping",
+ confirmation: "confirm_subscription",
+ rejection: "reject_subscription"
+ },
+ disconnect_reasons: {
+ unauthorized: "unauthorized",
+ invalid_request: "invalid_request",
+ server_restart: "server_restart",
+ remote: "remote"
+ },
+ default_mount_path: "/cable",
+ protocols: [ "actioncable-v1-json", "actioncable-unsupported" ]
+};
+
+const {message_types: message_types, protocols: protocols} = INTERNAL;
+
+const supportedProtocols = protocols.slice(0, protocols.length - 1);
+
+const indexOf = [].indexOf;
+
+class Connection {
+ constructor(consumer) {
+ this.open = this.open.bind(this);
+ this.consumer = consumer;
+ this.subscriptions = this.consumer.subscriptions;
+ this.monitor = new ConnectionMonitor(this);
+ this.disconnected = true;
+ }
+ send(data) {
+ if (this.isOpen()) {
+ this.webSocket.send(JSON.stringify(data));
+ return true;
+ } else {
+ return false;
+ }
+ }
+ open() {
+ if (this.isActive()) {
+ logger.log(`Attempted to open WebSocket, but existing socket is ${this.getState()}`);
+ return false;
+ } else {
+ const socketProtocols = [ ...protocols, ...this.consumer.subprotocols || [] ];
+ logger.log(`Opening WebSocket, current state is ${this.getState()}, subprotocols: ${socketProtocols}`);
+ if (this.webSocket) {
+ this.uninstallEventHandlers();
+ }
+ this.webSocket = new adapters.WebSocket(this.consumer.url, socketProtocols);
+ this.installEventHandlers();
+ this.monitor.start();
+ return true;
+ }
+ }
+ close({allowReconnect: allowReconnect} = {
+ allowReconnect: true
+ }) {
+ if (!allowReconnect) {
+ this.monitor.stop();
+ }
+ if (this.isOpen()) {
+ return this.webSocket.close();
+ }
+ }
+ reopen() {
+ logger.log(`Reopening WebSocket, current state is ${this.getState()}`);
+ if (this.isActive()) {
+ try {
+ return this.close();
+ } catch (error) {
+ logger.log("Failed to reopen WebSocket", error);
+ } finally {
+ logger.log(`Reopening WebSocket in ${this.constructor.reopenDelay}ms`);
+ setTimeout(this.open, this.constructor.reopenDelay);
+ }
+ } else {
+ return this.open();
+ }
+ }
+ getProtocol() {
+ if (this.webSocket) {
+ return this.webSocket.protocol;
+ }
+ }
+ isOpen() {
+ return this.isState("open");
+ }
+ isActive() {
+ return this.isState("open", "connecting");
+ }
+ triedToReconnect() {
+ return this.monitor.reconnectAttempts > 0;
+ }
+ isProtocolSupported() {
+ return indexOf.call(supportedProtocols, this.getProtocol()) >= 0;
+ }
+ isState(...states) {
+ return indexOf.call(states, this.getState()) >= 0;
+ }
+ getState() {
+ if (this.webSocket) {
+ for (let state in adapters.WebSocket) {
+ if (adapters.WebSocket[state] === this.webSocket.readyState) {
+ return state.toLowerCase();
+ }
+ }
+ }
+ return null;
+ }
+ installEventHandlers() {
+ for (let eventName in this.events) {
+ const handler = this.events[eventName].bind(this);
+ this.webSocket[`on${eventName}`] = handler;
+ }
+ }
+ uninstallEventHandlers() {
+ for (let eventName in this.events) {
+ this.webSocket[`on${eventName}`] = function() {};
+ }
+ }
+}
+
+Connection.reopenDelay = 500;
+
+Connection.prototype.events = {
+ message(event) {
+ if (!this.isProtocolSupported()) {
+ return;
+ }
+ const {identifier: identifier, message: message, reason: reason, reconnect: reconnect, type: type} = JSON.parse(event.data);
+ this.monitor.recordMessage();
+ switch (type) {
+ case message_types.welcome:
+ if (this.triedToReconnect()) {
+ this.reconnectAttempted = true;
+ }
+ this.monitor.recordConnect();
+ return this.subscriptions.reload();
+
+ case message_types.disconnect:
+ logger.log(`Disconnecting. Reason: ${reason}`);
+ return this.close({
+ allowReconnect: reconnect
+ });
+
+ case message_types.ping:
+ return null;
+
+ case message_types.confirmation:
+ this.subscriptions.confirmSubscription(identifier);
+ if (this.reconnectAttempted) {
+ this.reconnectAttempted = false;
+ return this.subscriptions.notify(identifier, "connected", {
+ reconnected: true
+ });
+ } else {
+ return this.subscriptions.notify(identifier, "connected", {
+ reconnected: false
+ });
+ }
+
+ case message_types.rejection:
+ return this.subscriptions.reject(identifier);
+
+ default:
+ return this.subscriptions.notify(identifier, "received", message);
+ }
+ },
+ open() {
+ logger.log(`WebSocket onopen event, using '${this.getProtocol()}' subprotocol`);
+ this.disconnected = false;
+ if (!this.isProtocolSupported()) {
+ logger.log("Protocol is unsupported. Stopping monitor and disconnecting.");
+ return this.close({
+ allowReconnect: false
+ });
+ }
+ },
+ close(event) {
+ logger.log("WebSocket onclose event");
+ if (this.disconnected) {
+ return;
+ }
+ this.disconnected = true;
+ this.monitor.recordDisconnect();
+ return this.subscriptions.notifyAll("disconnected", {
+ willAttemptReconnect: this.monitor.isRunning()
+ });
+ },
+ error() {
+ logger.log("WebSocket onerror event");
+ }
+};
+
+const extend = function(object, properties) {
+ if (properties != null) {
+ for (let key in properties) {
+ const value = properties[key];
+ object[key] = value;
+ }
+ }
+ return object;
+};
+
+class Subscription {
+ constructor(consumer, params = {}, mixin) {
+ this.consumer = consumer;
+ this.identifier = JSON.stringify(params);
+ extend(this, mixin);
+ }
+ perform(action, data = {}) {
+ data.action = action;
+ return this.send(data);
+ }
+ send(data) {
+ return this.consumer.send({
+ command: "message",
+ identifier: this.identifier,
+ data: JSON.stringify(data)
+ });
+ }
+ unsubscribe() {
+ return this.consumer.subscriptions.remove(this);
+ }
+}
+
+class SubscriptionGuarantor {
+ constructor(subscriptions) {
+ this.subscriptions = subscriptions;
+ this.pendingSubscriptions = [];
+ }
+ guarantee(subscription) {
+ if (this.pendingSubscriptions.indexOf(subscription) == -1) {
+ logger.log(`SubscriptionGuarantor guaranteeing ${subscription.identifier}`);
+ this.pendingSubscriptions.push(subscription);
+ } else {
+ logger.log(`SubscriptionGuarantor already guaranteeing ${subscription.identifier}`);
+ }
+ this.startGuaranteeing();
+ }
+ forget(subscription) {
+ logger.log(`SubscriptionGuarantor forgetting ${subscription.identifier}`);
+ this.pendingSubscriptions = this.pendingSubscriptions.filter((s => s !== subscription));
+ }
+ startGuaranteeing() {
+ this.stopGuaranteeing();
+ this.retrySubscribing();
+ }
+ stopGuaranteeing() {
+ clearTimeout(this.retryTimeout);
+ }
+ retrySubscribing() {
+ this.retryTimeout = setTimeout((() => {
+ if (this.subscriptions && typeof this.subscriptions.subscribe === "function") {
+ this.pendingSubscriptions.map((subscription => {
+ logger.log(`SubscriptionGuarantor resubscribing ${subscription.identifier}`);
+ this.subscriptions.subscribe(subscription);
+ }));
+ }
+ }), 500);
+ }
+}
+
+class Subscriptions {
+ constructor(consumer) {
+ this.consumer = consumer;
+ this.guarantor = new SubscriptionGuarantor(this);
+ this.subscriptions = [];
+ }
+ create(channelName, mixin) {
+ const channel = channelName;
+ const params = typeof channel === "object" ? channel : {
+ channel: channel
+ };
+ const subscription = new Subscription(this.consumer, params, mixin);
+ return this.add(subscription);
+ }
+ add(subscription) {
+ this.subscriptions.push(subscription);
+ this.consumer.ensureActiveConnection();
+ this.notify(subscription, "initialized");
+ this.subscribe(subscription);
+ return subscription;
+ }
+ remove(subscription) {
+ this.forget(subscription);
+ if (!this.findAll(subscription.identifier).length) {
+ this.sendCommand(subscription, "unsubscribe");
+ }
+ return subscription;
+ }
+ reject(identifier) {
+ return this.findAll(identifier).map((subscription => {
+ this.forget(subscription);
+ this.notify(subscription, "rejected");
+ return subscription;
+ }));
+ }
+ forget(subscription) {
+ this.guarantor.forget(subscription);
+ this.subscriptions = this.subscriptions.filter((s => s !== subscription));
+ return subscription;
+ }
+ findAll(identifier) {
+ return this.subscriptions.filter((s => s.identifier === identifier));
+ }
+ reload() {
+ return this.subscriptions.map((subscription => this.subscribe(subscription)));
+ }
+ notifyAll(callbackName, ...args) {
+ return this.subscriptions.map((subscription => this.notify(subscription, callbackName, ...args)));
+ }
+ notify(subscription, callbackName, ...args) {
+ let subscriptions;
+ if (typeof subscription === "string") {
+ subscriptions = this.findAll(subscription);
+ } else {
+ subscriptions = [ subscription ];
+ }
+ return subscriptions.map((subscription => typeof subscription[callbackName] === "function" ? subscription[callbackName](...args) : undefined));
+ }
+ subscribe(subscription) {
+ if (this.sendCommand(subscription, "subscribe")) {
+ this.guarantor.guarantee(subscription);
+ }
+ }
+ confirmSubscription(identifier) {
+ logger.log(`Subscription confirmed ${identifier}`);
+ this.findAll(identifier).map((subscription => this.guarantor.forget(subscription)));
+ }
+ sendCommand(subscription, command) {
+ const {identifier: identifier} = subscription;
+ return this.consumer.send({
+ command: command,
+ identifier: identifier
+ });
+ }
+}
+
+class Consumer {
+ constructor(url) {
+ this._url = url;
+ this.subscriptions = new Subscriptions(this);
+ this.connection = new Connection(this);
+ this.subprotocols = [];
+ }
+ get url() {
+ return createWebSocketURL(this._url);
+ }
+ send(data) {
+ return this.connection.send(data);
+ }
+ connect() {
+ return this.connection.open();
+ }
+ disconnect() {
+ return this.connection.close({
+ allowReconnect: false
+ });
+ }
+ ensureActiveConnection() {
+ if (!this.connection.isActive()) {
+ return this.connection.open();
+ }
+ }
+ addSubProtocol(subprotocol) {
+ this.subprotocols = [ ...this.subprotocols, subprotocol ];
+ }
+}
+
+function createWebSocketURL(url) {
+ if (typeof url === "function") {
+ url = url();
+ }
+ if (url && !/^wss?:/i.test(url)) {
+ const a = document.createElement("a");
+ a.href = url;
+ a.href = a.href;
+ a.protocol = a.protocol.replace("http", "ws");
+ return a.href;
+ } else {
+ return url;
+ }
+}
+
+function createConsumer(url = getConfig("url") || INTERNAL.default_mount_path) {
+ return new Consumer(url);
+}
+
+function getConfig(name) {
+ const element = document.head.querySelector(`meta[name='action-cable-${name}']`);
+ if (element) {
+ return element.getAttribute("content");
+ }
+}
+
+// Action Cable provides the framework to deal with WebSockets in Rails.
+// You can generate new channels where WebSocket features live using the `bin/rails generate channel` command.
+
+
+const consumer = createConsumer();
+
class ProgressMonitor {
connect() {
var monitorElements = $('[data-behavior="progress-panel"]');
- var defaultRefreshRate = 3000;
var panelContainer;
var pollers = [];
$(monitorElements).each(function() {
panelContainer = $(this);
panelContainer.hide();
- var monitorUrl = panelContainer.data('monitorUrl');
- var refreshRate = panelContainer.data('refreshRate') || defaultRefreshRate;
- pollers.push(
- setInterval(function() {
- checkMonitorUrl(monitorUrl);
- }, refreshRate)
- );
+
+ consumer.subscriptions.create({ channel: "ProgressChannel"}, {
+ received(data) {
+ if (data.exhibit_id != panelContainer.data('exhibit-id')) return;
+ updateMonitorPanel(data);
+ }
+ });
});
// Clear the intervals on turbolink:click event (e.g. when the user navigates away from the page)
@@ -5204,21 +5721,6 @@ class ProgressMonitor {
}
});
- function checkMonitorUrl(url) {
- $.ajax(url).done(success).fail(fail);
- }
-
- function success(data) {
- if (data.recently_in_progress) {
- updateMonitorPanel(data);
- monitorPanel().show();
- } else {
- monitorPanel().hide();
- }
- }
-
- function fail() { monitorPanel().hide(); }
-
function updateMonitorPanel(data) {
panelStartDate().text(data.started_at);
panelCurrentDate().text(data.updated_at);
@@ -5601,6 +6103,8 @@ class CheckboxSubmit {
}
// Visibility toggle for items in an exhibit, based on Blacklight's bookmark toggle
+// See: https://github.com/projectblacklight/blacklight/blob/main/app/javascript/blacklight/bookmark_toggle.js
+
const VisibilityToggle = (e) => {
if (e.target.matches('[data-checkboxsubmit-target="checkbox"]')) {
diff --git a/app/assets/javascripts/spotlight/spotlight.esm.js.map b/app/assets/javascripts/spotlight/spotlight.esm.js.map
index dbae104410..201ff34edc 100644
--- a/app/assets/javascripts/spotlight/spotlight.esm.js.map
+++ b/app/assets/javascripts/spotlight/spotlight.esm.js.map
@@ -1 +1 @@
-{"version":3,"file":"spotlight.esm.js","sources":["../../../javascript/spotlight/user/browse_group_categories.js","../../../javascript/spotlight/user/carousel.js","../../../javascript/spotlight/user/clear_form_button.js","../../../javascript/spotlight/user/report_a_problem.js","../../../javascript/spotlight/user/zpr_links.js","../../../javascript/spotlight/user/index.js","../../../../vendor/assets/javascripts/nestable.js","../../../../vendor/assets/javascripts/bootstrap-tagsinput.js","../../../../vendor/assets/javascripts/jquery.serializejson.js","../../../../vendor/assets/javascripts/leaflet-iiif.js","../../../../vendor/assets/javascripts/Leaflet.Editable.js","../../../../vendor/assets/javascripts/Path.Drag.js","../../../javascript/spotlight/admin/add_another.js","../../../javascript/spotlight/admin/add_new_button.js","../../../javascript/spotlight/admin/blacklight_configuration.js","../../../javascript/spotlight/admin/copy_email_addresses.js","../../../javascript/spotlight/admin/iiif.js","../../../javascript/spotlight/admin/add_image_selector.js","../../../javascript/spotlight/core.js","../../../javascript/spotlight/admin/crop.js","../../../javascript/spotlight/admin/croppable.js","../../../javascript/spotlight/admin/edit_in_place.js","../../../javascript/spotlight/admin/exhibit_tag_autocomplete.js","../../../../vendor/assets/javascripts/parameterize.js","../../../javascript/spotlight/admin/exhibits.js","../../../javascript/spotlight/admin/form_observer.js","../../../javascript/spotlight/admin/locks.js","../../../javascript/spotlight/admin/multi_image_selector.js","../../../javascript/spotlight/admin/pages.js","../../../javascript/spotlight/admin/progress_monitor.js","../../../javascript/spotlight/admin/readonly_checkbox.js","../../../javascript/spotlight/admin/search_typeahead.js","../../../javascript/spotlight/admin/select_related_input.js","../../../javascript/spotlight/admin/spotlight_nestable.js","../../../javascript/spotlight/admin/tabs.js","../../../javascript/spotlight/admin/translation_progress.js","../../../javascript/spotlight/admin/checkbox_submit.js","../../../javascript/spotlight/admin/visibility_toggle.js","../../../javascript/spotlight/admin/users.js","../../../javascript/spotlight/admin/block_mixins/autocompleteable.js","../../../javascript/spotlight/admin/block_mixins/formable.js","../../../javascript/spotlight/admin/block_mixins/plustextable.js","../../../javascript/spotlight/admin/blocks/block.js","../../../javascript/spotlight/admin/blocks/resources_block.js","../../../javascript/spotlight/admin/blocks/browse_block.js","../../../javascript/spotlight/admin/blocks/browse_group_categories_block.js","../../../javascript/spotlight/admin/blocks/iframe_block.js","../../../javascript/spotlight/admin/blocks/link_to_search_block.js","../../../javascript/spotlight/admin/blocks/oembed_block.js","../../../javascript/spotlight/admin/blocks/pages_block.js","../../../javascript/spotlight/admin/blocks/rule_block.js","../../../javascript/spotlight/admin/blocks/search_result_block.js","../../../javascript/spotlight/admin/blocks/solr_documents_base_block.js","../../../javascript/spotlight/admin/blocks/solr_documents_block.js","../../../javascript/spotlight/admin/blocks/solr_documents_carousel_block.js","../../../javascript/spotlight/admin/blocks/solr_documents_embed_block.js","../../../javascript/spotlight/admin/blocks/solr_documents_features_block.js","../../../javascript/spotlight/admin/blocks/solr_documents_grid_block.js","../../../javascript/spotlight/admin/blocks/uploaded_items_block.js","../../../javascript/spotlight/admin/sir-trevor/block_controls.js","../../../javascript/spotlight/admin/sir-trevor/block_limits.js","../../../javascript/spotlight/admin/sir-trevor/locales.js","../../../javascript/spotlight/admin/index.js","../../../javascript/spotlight/index.js"],"sourcesContent":["export default class {\n connect() {\n var $container, slider;\n\n function init() {\n var data = $container.data();\n var sidebar = $container.data().sidebar;\n var items = data.browseGroupCategoriesCount;\n var dir = $('html').attr('dir');\n var controls = $container.parent().find('.browse-group-categories-controls')[0];\n\n slider = tns({\n container: $container[0],\n controlsContainer: controls,\n loop: false,\n nav: false,\n items: 1,\n slideBy: 'page',\n textDirection: dir,\n responsive: {\n 576: {\n items: itemCount(items, sidebar)\n }\n }\n });\n }\n\n // Destroy the slider instance, as tns will change the dom elements, causing some issues with turbolinks\n function setupDestroy() {\n document.addEventListener('turbolinks:before-cache', function() {\n if (slider && slider.destroy) {\n slider.destroy();\n }\n });\n }\n\n function itemCount(items, sidebar) {\n if (items < 3) {\n return items;\n }\n return sidebar ? 3 : 4;\n }\n\n return $('[data-browse-group-categories-carousel]').each(function() {\n $container = $(this);\n init();\n setupDestroy();\n });\n }\n}\n","export default class {\n connect() {\n $('.carousel').carousel();\n }\n}\n","export default class {\n connect() {\n var $clearBtn = $('.btn-reset');\n var $input = $clearBtn.parent().prev('input');\n var btnCheck = function(){\n if ($input.val() !== '') {\n $clearBtn.css('display', 'inline-block');\n } else {\n $clearBtn.css('display', 'none');\n }\n };\n\n btnCheck();\n $input.on('keyup', function() {\n btnCheck();\n });\n\n $clearBtn.on('click', function(event) {\n event.preventDefault();\n $input.val('');\n });\n }\n}\n","export default class {\n connect(){\n var container, target;\n\n function init() {\n const target_val = container.attr('data-target') || container.attr('data-bs-target');\n if (!target_val) \n return\n\n target = $(\"#\" + target_val); \n container.on('click', open);\n target.find('[data-behavior=\"cancel-link\"]').on('click', close);\n }\n\n function open(event) {\n event.preventDefault();\n target.slideToggle('slow');\n }\n\n function close(event) {\n event.preventDefault();\n target.slideUp('fast');\n }\n\n return $('[data-behavior=\"contact-link\"]').each(function() { \n container = $(this);\n init();\n });\n }\n}","export default class {\n connect() {\n $('.zpr-link').on('click', function() {\n var modalDialog = $('#blacklight-modal .modal-dialog');\n var modalContent = modalDialog.find('.modal-content')\n modalDialog.removeClass('modal-lg')\n modalDialog.addClass('modal-xl')\n modalContent.html('
');\n var controls = `\n
\n \n
\n
\n \n \n
\n
\n
`\n\n $('#osd-modal-container').append('');\n $('#osd-modal-container').append(controls);\n\n $('#blacklight-modal').modal('show');\n \n $('#blacklight-modal').one('hidden.bs.modal', function (event) {\n modalDialog.removeClass('modal-xl')\n modalDialog.addClass('modal-lg')\n });\n\n OpenSeadragon({\n id: 'osd-div',\n zoomInButton: \"osd-zoom-in\",\n zoomOutButton: \"osd-zoom-out\",\n // This is a hack where OpenSeadragon (if using mapped buttons) requires you\n // to map all of the buttons.\n homeButton: \"empty-div-required-by-osd\",\n fullPageButton: \"empty-div-required-by-osd\",\n nextButton: \"empty-div-required-by-osd\",\n previousButton: \"empty-div-required-by-osd\",\n tileSources: [$(this).data('iiif-tilesource')]\n })\n });\n }\n}\n","import BrowseGroupCateogries from 'spotlight/user/browse_group_categories'\nimport Carousel from 'spotlight/user/carousel'\nimport ClearFormButton from 'spotlight/user/clear_form_button'\nimport ReportProblem from 'spotlight/user/report_a_problem'\nimport ZprLinks from 'spotlight/user/zpr_links'\n\nexport default class {\n connect() {\n new BrowseGroupCateogries().connect()\n new Carousel().connect()\n new ClearFormButton().connect()\n new ReportProblem().connect()\n new ZprLinks().connect()\n }\n}\n","/*!\n * Nestable jQuery Plugin - Copyright (c) 2012 David Bushell - http://dbushell.com/\n * Dual-licensed under the BSD or MIT licenses\n */\n;(function($, window, document, undefined)\n{\n var hasTouch = 'ontouchstart' in window;\n var nestableCopy;\n\n /**\n * Detect CSS pointer-events property\n * events are normally disabled on the dragging element to avoid conflicts\n * https://github.com/ausi/Feature-detection-technique-for-pointer-events/blob/master/modernizr-pointerevents.js\n */\n var hasPointerEvents = (function()\n {\n var el = document.createElement('div'),\n docEl = document.documentElement;\n if (!('pointerEvents' in el.style)) {\n return false;\n }\n el.style.pointerEvents = 'auto';\n el.style.pointerEvents = 'x';\n docEl.appendChild(el);\n var supports = window.getComputedStyle && window.getComputedStyle(el, '').pointerEvents === 'auto';\n docEl.removeChild(el);\n return !!supports;\n })();\n\n var eStart = hasTouch ? 'touchstart' : 'mousedown',\n eMove = hasTouch ? 'touchmove' : 'mousemove',\n eEnd = hasTouch ? 'touchend' : 'mouseup',\n eCancel = hasTouch ? 'touchcancel' : 'mouseup';\n \n var defaults = {\n listNodeName : 'ol',\n itemNodeName : 'li',\n rootClass : 'dd',\n listClass : 'dd-list',\n itemClass : 'dd-item',\n dragClass : 'dd-dragel',\n handleClass : 'dd-handle',\n collapsedClass : 'dd-collapsed',\n placeClass : 'dd-placeholder',\n noDragClass : 'dd-nodrag',\n noChildrenClass : 'dd-nochildren',\n emptyClass : 'dd-empty',\n expandBtnHTML : '',\n collapseBtnHTML : '',\n group : 0,\n maxDepth : 5,\n threshold : 20,\n reject : [],\n //method for call when an item has been successfully dropped\n //method has 1 argument in which sends an object containing all\n //necessary details\n dropCallback : null,\n // When a node is dragged it is moved to its new location.\n // You can set the next option to true to create a copy of the node that is dragged.\n cloneNodeOnDrag : false,\n // When the node is dragged and released outside its list delete it.\n dragOutsideToDelete : false\n };\n\n function Plugin(element, options)\n {\n this.w = $(document);\n this.el = $(element);\n this.options = $.extend({}, defaults, options);\n this.init();\n }\n\n Plugin.prototype = {\n\n init: function()\n {\n var list = this;\n\n list.reset();\n\n list.el.data('nestable-group', this.options.group);\n\n list.placeEl = $('');\n\n $.each(this.el.find(list.options.itemNodeName), function(k, el) {\n list.setParent($(el));\n });\n\n list.el.on('click', 'button', function(e)\n {\n if (list.dragEl || (!hasTouch && e.button !== 0)) {\n return;\n }\n var target = $(e.currentTarget),\n action = target.data('action'),\n item = target.parent(list.options.itemNodeName);\n if (action === 'collapse') {\n list.collapseItem(item);\n }\n if (action === 'expand') {\n list.expandItem(item);\n }\n });\n\n var onStartEvent = function(e)\n {\n var handle = $(e.target);\n\n list.nestableCopy = handle.closest('.'+list.options.rootClass).clone(true);\n\n if (!handle.hasClass(list.options.handleClass)) {\n if (handle.closest('.' + list.options.noDragClass).length) {\n return;\n }\n handle = handle.closest('.' + list.options.handleClass);\n }\n if (!handle.length || list.dragEl || (!hasTouch && e.which !== 1) || (hasTouch && e.touches.length !== 1)) {\n return;\n }\n e.preventDefault();\n list.dragStart(hasTouch ? e.touches[0] : e);\n };\n\n var onMoveEvent = function(e)\n {\n if (list.dragEl) {\n e.preventDefault();\n list.dragMove(hasTouch ? e.touches[0] : e);\n }\n };\n\n var onEndEvent = function(e)\n {\n if (list.dragEl) {\n e.preventDefault();\n list.dragStop(hasTouch ? e.touches[0] : e);\n }\n };\n\n if (hasTouch) {\n list.el[0].addEventListener(eStart, onStartEvent, false);\n window.addEventListener(eMove, onMoveEvent, false);\n window.addEventListener(eEnd, onEndEvent, false);\n window.addEventListener(eCancel, onEndEvent, false);\n } else {\n list.el.on(eStart, onStartEvent);\n list.w.on(eMove, onMoveEvent);\n list.w.on(eEnd, onEndEvent);\n }\n\n var destroyNestable = function()\n {\n if (hasTouch) {\n list.el[0].removeEventListener(eStart, onStartEvent, false);\n window.removeEventListener(eMove, onMoveEvent, false);\n window.removeEventListener(eEnd, onEndEvent, false);\n window.removeEventListener(eCancel, onEndEvent, false);\n } else {\n list.el.off(eStart, onStartEvent);\n list.w.off(eMove, onMoveEvent);\n list.w.off(eEnd, onEndEvent);\n }\n\n list.el.off('click');\n list.el.unbind('destroy-nestable');\n\n list.el.data(\"nestable\", null);\n\n var buttons = list.el[0].getElementsByTagName('button');\n\n $(buttons).remove();\n };\n\n list.el.bind('destroy-nestable', destroyNestable);\n },\n\n destroy: function ()\n {\n this.expandAll();\n this.el.trigger('destroy-nestable');\n },\n\n serialize: function()\n {\n var data,\n depth = 0,\n list = this;\n const step = function(level, depth)\n {\n var array = [ ],\n items = level.children(list.options.itemNodeName);\n items.each(function()\n {\n var li = $(this),\n item = $.extend({}, li.data()),\n sub = li.children(list.options.listNodeName);\n if (sub.length) {\n item.children = step(sub, depth + 1);\n }\n array.push(item);\n });\n return array;\n };\n var el;\n\n if (list.el.is(list.options.listNodeName)) {\n el = list.el;\n } else {\n el = list.el.find(list.options.listNodeName).first();\n }\n data = step(el, depth);\n return data;\n },\n\n reset: function()\n {\n this.mouse = {\n offsetX : 0,\n offsetY : 0,\n startX : 0,\n startY : 0,\n lastX : 0,\n lastY : 0,\n nowX : 0,\n nowY : 0,\n distX : 0,\n distY : 0,\n dirAx : 0,\n dirX : 0,\n dirY : 0,\n lastDirX : 0,\n lastDirY : 0,\n distAxX : 0,\n distAxY : 0\n };\n this.moving = false;\n this.dragEl = null;\n this.dragRootEl = null;\n this.dragDepth = 0;\n this.dragItem = null;\n this.hasNewRoot = false;\n this.pointEl = null;\n this.sourceRoot = null;\n this.isOutsideRoot = false;\n },\n\n expandItem: function(li)\n {\n li.removeClass(this.options.collapsedClass);\n li.children('[data-action=\"expand\"]').hide();\n li.children('[data-action=\"collapse\"]').show();\n li.children(this.options.listNodeName).show();\n this.el.trigger('expand', [li]);\n li.trigger('expand');\n },\n\n collapseItem: function(li)\n {\n var lists = li.children(this.options.listNodeName);\n if (lists.length) {\n li.addClass(this.options.collapsedClass);\n li.children('[data-action=\"collapse\"]').hide();\n li.children('[data-action=\"expand\"]').show();\n li.children(this.options.listNodeName).hide();\n }\n this.el.trigger('collapse', [li]);\n li.trigger('collapse');\n },\n\n expandAll: function()\n {\n var list = this;\n list.el.find(list.options.itemNodeName).each(function() {\n list.expandItem($(this));\n });\n },\n\n collapseAll: function()\n {\n var list = this;\n list.el.find(list.options.itemNodeName).each(function() {\n list.collapseItem($(this));\n });\n },\n\n setParent: function(li)\n {\n if (li.children(this.options.listNodeName).length) {\n li.prepend($(this.options.expandBtnHTML));\n li.prepend($(this.options.collapseBtnHTML));\n }\n if( (' ' + li[0].className + ' ').indexOf(' ' + defaults.collapsedClass + ' ') > -1 )\n {\n li.children('[data-action=\"collapse\"]').hide();\n } else {\n li.children('[data-action=\"expand\"]').hide();\n }\n },\n\n unsetParent: function(li)\n {\n li.removeClass(this.options.collapsedClass);\n li.children('[data-action]').remove();\n li.children(this.options.listNodeName).remove();\n },\n\n dragStart: function(e)\n {\n var mouse = this.mouse,\n target = $(e.target),\n dragItem = target.closest('.' + this.options.handleClass).closest(this.options.itemNodeName);\n\n this.sourceRoot = target.closest('.' + this.options.rootClass);\n\n this.dragItem = dragItem;\n\n this.placeEl.css('height', dragItem.height());\n\n mouse.offsetX = e.offsetX !== undefined ? e.offsetX : e.pageX - target.offset().left;\n mouse.offsetY = e.offsetY !== undefined ? e.offsetY : e.pageY - target.offset().top;\n mouse.startX = mouse.lastX = e.pageX;\n mouse.startY = mouse.lastY = e.pageY;\n\n this.dragRootEl = this.el;\n\n this.dragEl = $(document.createElement(this.options.listNodeName)).addClass(this.options.listClass + ' ' + this.options.dragClass);\n this.dragEl.css('width', dragItem.width());\n\n // fix for zepto.js\n //dragItem.after(this.placeEl).detach().appendTo(this.dragEl);\n if(this.options.cloneNodeOnDrag) {\n dragItem.after(dragItem.clone());\n } else {\n dragItem.after(this.placeEl);\n }\n dragItem[0].parentNode.removeChild(dragItem[0]);\n dragItem.appendTo(this.dragEl);\n\n $(document.body).append(this.dragEl);\n this.dragEl.css({\n 'left' : e.pageX - mouse.offsetX,\n 'top' : e.pageY - mouse.offsetY\n });\n // total depth of dragging item\n var i, depth,\n items = this.dragEl.find(this.options.itemNodeName);\n for (i = 0; i < items.length; i++) {\n depth = $(items[i]).parents(this.options.listNodeName).length;\n if (depth > this.dragDepth) {\n this.dragDepth = depth;\n }\n }\n },\n\n dragStop: function(e)\n {\n // fix for zepto.js\n //this.placeEl.replaceWith(this.dragEl.children(this.options.itemNodeName + ':first').detach());\n var el = this.dragEl.children(this.options.itemNodeName).first();\n el[0].parentNode.removeChild(el[0]);\n\n if(this.isOutsideRoot && this.options.dragOutsideToDelete)\n {\n var parent = this.placeEl.parent();\n this.placeEl.remove();\n if (!parent.children().length) {\n this.unsetParent(parent.parent());\n }\n // If all nodes where deleted, create a placeholder element.\n if (!this.dragRootEl.find(this.options.itemNodeName).length)\n {\n this.dragRootEl.append('');\n }\n } \n else \n {\n this.placeEl.replaceWith(el);\n }\n\n if (!this.moving)\n {\n $(this.dragItem).trigger('click');\n }\n\n var i;\n var isRejected = false;\n for (i = 0; i < this.options.reject.length; i++)\n {\n var reject = this.options.reject[i];\n if (reject.rule.apply(this.dragRootEl))\n {\n var nestableDragEl = el.clone(true);\n this.dragRootEl.html(this.nestableCopy.children().clone(true));\n if (reject.action) {\n reject.action.apply(this.dragRootEl, [nestableDragEl]);\n }\n\n isRejected = true;\n break;\n }\n }\n\n if (!isRejected)\n {\n this.dragEl.remove();\n this.el.trigger('change');\n\n //Let's find out new parent id\n var parentItem = el.parent().parent();\n var parentId = null;\n if(parentItem !== null && !parentItem.is('.' + this.options.rootClass))\n parentId = parentItem.data('id');\n\n if($.isFunction(this.options.dropCallback))\n {\n var details = {\n sourceId : el.data('id'),\n destId : parentId,\n sourceEl : el,\n destParent : parentItem,\n destRoot : el.closest('.' + this.options.rootClass),\n sourceRoot : this.sourceRoot\n };\n this.options.dropCallback.call(this, details);\n }\n\n if (this.hasNewRoot) {\n this.dragRootEl.trigger('change');\n }\n\n this.reset();\n }\n },\n\n dragMove: function(e)\n {\n var list, parent, prev, next, depth,\n opt = this.options,\n mouse = this.mouse;\n\n this.dragEl.css({\n 'left' : e.pageX - mouse.offsetX,\n 'top' : e.pageY - mouse.offsetY\n });\n\n // mouse position last events\n mouse.lastX = mouse.nowX;\n mouse.lastY = mouse.nowY;\n // mouse position this events\n mouse.nowX = e.pageX;\n mouse.nowY = e.pageY;\n // distance mouse moved between events\n mouse.distX = mouse.nowX - mouse.lastX;\n mouse.distY = mouse.nowY - mouse.lastY;\n // direction mouse was moving\n mouse.lastDirX = mouse.dirX;\n mouse.lastDirY = mouse.dirY;\n // direction mouse is now moving (on both axis)\n mouse.dirX = mouse.distX === 0 ? 0 : mouse.distX > 0 ? 1 : -1;\n mouse.dirY = mouse.distY === 0 ? 0 : mouse.distY > 0 ? 1 : -1;\n // axis mouse is now moving on\n var newAx = Math.abs(mouse.distX) > Math.abs(mouse.distY) ? 1 : 0;\n\n // do nothing on first move\n if (!this.moving) {\n mouse.dirAx = newAx;\n this.moving = true;\n return;\n }\n\n // calc distance moved on this axis (and direction)\n if (mouse.dirAx !== newAx) {\n mouse.distAxX = 0;\n mouse.distAxY = 0;\n } else {\n mouse.distAxX += Math.abs(mouse.distX);\n if (mouse.dirX !== 0 && mouse.dirX !== mouse.lastDirX) {\n mouse.distAxX = 0;\n }\n mouse.distAxY += Math.abs(mouse.distY);\n if (mouse.dirY !== 0 && mouse.dirY !== mouse.lastDirY) {\n mouse.distAxY = 0;\n }\n }\n mouse.dirAx = newAx;\n\n /**\n * move horizontal\n */\n if (mouse.dirAx && mouse.distAxX >= opt.threshold) {\n // reset move distance on x-axis for new phase\n mouse.distAxX = 0;\n prev = this.placeEl.prev(opt.itemNodeName);\n // increase horizontal level if previous sibling exists and is not collapsed\n if (mouse.distX > 0 && prev.length && !prev.hasClass(opt.collapsedClass) && !prev.hasClass(opt.noChildrenClass)) {\n // cannot increase level when item above is collapsed\n list = prev.find(opt.listNodeName).last();\n // check if depth limit has reached\n depth = this.placeEl.parents(opt.listNodeName).length;\n if (depth + this.dragDepth <= opt.maxDepth) {\n // create new sub-level if one doesn't exist\n if (!list.length) {\n list = $('<' + opt.listNodeName + '/>').addClass(opt.listClass);\n list.append(this.placeEl);\n prev.append(list);\n this.setParent(prev);\n } else {\n // else append to next level up\n list = prev.children(opt.listNodeName).last();\n list.append(this.placeEl);\n }\n }\n }\n // decrease horizontal level\n if (mouse.distX < 0) {\n // we can't decrease a level if an item preceeds the current one\n next = this.placeEl.next(opt.itemNodeName);\n if (!next.length) {\n parent = this.placeEl.parent();\n this.placeEl.closest(opt.itemNodeName).after(this.placeEl);\n if (!parent.children().length) {\n this.unsetParent(parent.parent());\n }\n }\n }\n }\n\n var isEmpty = false;\n\n // find list item under cursor\n if (!hasPointerEvents) {\n this.dragEl[0].style.visibility = 'hidden';\n }\n \n this.pointEl = $(document.elementFromPoint(e.pageX - document.documentElement.scrollLeft, e.pageY - (window.pageYOffset || document.documentElement.scrollTop)));\n\n // Check if the node is dragged outside of its list.\n if(this.dragRootEl.has(this.pointEl).length) {\n this.isOutsideRoot = false;\n this.dragEl[0].style.opacity = 1;\n } else {\n this.isOutsideRoot = true;\n this.dragEl[0].style.opacity = 0.5;\n }\n\n // find parent list of item under cursor\n var pointElRoot = this.pointEl.closest('.' + opt.rootClass),\n isNewRoot = this.dragRootEl.data('nestable-id') !== pointElRoot.data('nestable-id');\n\n this.isOutsideRoot = !pointElRoot.length;\n\n if (!hasPointerEvents) {\n this.dragEl[0].style.visibility = 'visible';\n }\n if (this.pointEl.hasClass(opt.handleClass)) {\n this.pointEl = this.pointEl.closest( opt.itemNodeName );\n }\n\n if (opt.maxDepth == 1 && !this.pointEl.hasClass(opt.itemClass)) {\n this.pointEl = this.pointEl.closest(\".\" + opt.itemClass);\n }\n\n if (this.pointEl.hasClass(opt.emptyClass)) {\n isEmpty = true;\n }\n else if (!this.pointEl.length || !this.pointEl.hasClass(opt.itemClass)) {\n return;\n }\n\n /**\n * move vertical\n */\n if (!mouse.dirAx || isNewRoot || isEmpty) {\n // check if groups match if dragging over new root\n if (isNewRoot && opt.group !== pointElRoot.data('nestable-group')) {\n return;\n }\n // check depth limit\n depth = this.dragDepth - 1 + this.pointEl.parents(opt.listNodeName).length;\n if (depth > opt.maxDepth) {\n return;\n }\n var before = e.pageY < (this.pointEl.offset().top + this.pointEl.height() / 2);\n parent = this.placeEl.parent();\n // if empty create new list to replace empty placeholder\n if (isEmpty) {\n list = $(document.createElement(opt.listNodeName)).addClass(opt.listClass);\n list.append(this.placeEl);\n this.pointEl.replaceWith(list);\n }\n else if (before) {\n this.pointEl.before(this.placeEl);\n }\n else {\n this.pointEl.after(this.placeEl);\n }\n if (!parent.children().length) {\n this.unsetParent(parent.parent());\n }\n if (!this.dragRootEl.find(opt.itemNodeName).length) {\n this.dragRootEl.append('');\n }\n // parent root list has changed\n this.dragRootEl = pointElRoot;\n if (isNewRoot) {\n this.hasNewRoot = this.el[0] !== this.dragRootEl[0];\n }\n }\n }\n\n };\n\n $.fn.nestable = function(params)\n {\n var lists = this,\n retval = this;\n\n var generateUid = function (separator) {\n var delim = separator || \"-\";\n\n function S4() {\n return (((1 + Math.random()) * 0x10000) | 0).toString(16).substring(1);\n }\n\n return (S4() + S4() + delim + S4() + delim + S4() + delim + S4() + delim + S4() + S4() + S4());\n };\n\n lists.each(function()\n {\n var plugin = $(this).data(\"nestable\");\n\n if (!plugin) {\n $(this).data(\"nestable\", new Plugin(this, params));\n $(this).data(\"nestable-id\", generateUid());\n } else {\n if (typeof params === 'string' && typeof plugin[params] === 'function') {\n retval = plugin[params]();\n }\n }\n });\n\n return retval || lists;\n };\n\n})(window.jQuery || window.Zepto, window, document);\n","/* From https://github.com/TimSchlechter/bootstrap-tagsinput/blob/2661784c2c281d3a69b93897ff3f39e4ffa5cbd1/dist/bootstrap-tagsinput.js */\n\n/* The MIT License (MIT)\n\nCopyright (c) 2013 Tim Schlechter\n\nPermission is hereby granted, free of charge, to any person obtaining a copy of\nthis software and associated documentation files (the \"Software\"), to deal in\nthe Software without restriction, including without limitation the rights to\nuse, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of\nthe Software, and to permit persons to whom the Software is furnished to do so,\nsubject to the following conditions:\n\nThe above copyright notice and this permission notice shall be included in all\ncopies or substantial portions of the Software.\n\nTHE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\nIMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS\nFOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR\nCOPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER\nIN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN\nCONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.\n*/\n\n/* Retrieved 12 February 2014 */\n\n(function ($) {\n \"use strict\";\n\n var defaultOptions = {\n tagClass: function(item) {\n return 'badge badge-info bg-info';\n },\n itemValue: function(item) {\n return item ? item.toString() : item;\n },\n itemText: function(item) {\n return this.itemValue(item);\n },\n freeInput: true,\n maxTags: undefined,\n confirmKeys: [13],\n onTagExists: function(item, $tag) {\n $tag.hide().fadeIn();\n }\n };\n\n /**\n * Constructor function\n */\n function TagsInput(element, options) {\n this.itemsArray = [];\n\n this.$element = $(element);\n this.$element.hide();\n\n this.isSelect = (element.tagName === 'SELECT');\n this.multiple = (this.isSelect && element.hasAttribute('multiple'));\n this.objectItems = options && options.itemValue;\n this.placeholderText = element.hasAttribute('placeholder') ? this.$element.attr('placeholder') : '';\n this.inputSize = Math.max(1, this.placeholderText.length);\n\n this.$container = $('');\n this.$input = $('').appendTo(this.$container);\n\n this.$element.after(this.$container);\n\n this.build(options);\n }\n\n TagsInput.prototype = {\n constructor: TagsInput,\n\n /**\n * Adds the given item as a new tag. Pass true to dontPushVal to prevent\n * updating the elements val()\n */\n add: function(item, dontPushVal) {\n var self = this;\n\n if (self.options.maxTags && self.itemsArray.length >= self.options.maxTags)\n return;\n\n // Ignore falsey values, except false\n if (item !== false && !item)\n return;\n\n // Throw an error when trying to add an object while the itemValue option was not set\n if (typeof item === \"object\" && !self.objectItems)\n throw(\"Can't add objects when itemValue option is not set\");\n\n // Ignore strings only containg whitespace\n if (item.toString().match(/^\\s*$/))\n return;\n\n // If SELECT but not multiple, remove current tag\n if (self.isSelect && !self.multiple && self.itemsArray.length > 0)\n self.remove(self.itemsArray[0]);\n\n if (typeof item === \"string\" && this.$element[0].tagName === 'INPUT') {\n var items = item.split(',');\n if (items.length > 1) {\n for (var i = 0; i < items.length; i++) {\n this.add(items[i], true);\n }\n\n if (!dontPushVal)\n self.pushVal();\n return;\n }\n }\n\n var itemValue = self.options.itemValue(item),\n itemText = self.options.itemText(item),\n tagClass = self.options.tagClass(item);\n\n // Ignore items allready added\n var existing = $.grep(self.itemsArray, function(item) { return self.options.itemValue(item) === itemValue; } )[0];\n if (existing) {\n // Invoke onTagExists\n if (self.options.onTagExists) {\n var $existingTag = $(\".tag\", self.$container).filter(function() { return $(this).data(\"item\") === existing; });\n self.options.onTagExists(item, $existingTag);\n }\n return;\n }\n\n // register item in internal array and map\n self.itemsArray.push(item);\n\n // add a tag element\n var $tag = $('' + htmlEncode(itemText) + '');\n $tag.data('item', item);\n self.findInputWrapper().before($tag);\n $tag.after(' ');\n\n // add if item represents a value not present in one of the 's options\n if (self.isSelect && !$('option[value=\"' + escape(itemValue) + '\"]',self.$element)[0]) {\n var $option = $('');\n $option.data('item', item);\n $option.attr('value', itemValue);\n self.$element.append($option);\n }\n\n if (!dontPushVal)\n self.pushVal();\n\n // Add class when reached maxTags\n if (self.options.maxTags === self.itemsArray.length)\n self.$container.addClass('bootstrap-tagsinput-max');\n\n self.$element.trigger($.Event('itemAdded', { item: item }));\n },\n\n /**\n * Removes the given item. Pass true to dontPushVal to prevent updating the\n * elements val()\n */\n remove: function(item, dontPushVal) {\n var self = this;\n\n if (self.objectItems) {\n if (typeof item === \"object\")\n item = $.grep(self.itemsArray, function(other) { return self.options.itemValue(other) == self.options.itemValue(item); } )[0];\n else\n item = $.grep(self.itemsArray, function(other) { return self.options.itemValue(other) == item; } )[0];\n }\n\n if (item) {\n $('.tag', self.$container).filter(function() { return $(this).data('item') === item; }).remove();\n $('option', self.$element).filter(function() { return $(this).data('item') === item; }).remove();\n self.itemsArray.splice($.inArray(item, self.itemsArray), 1);\n }\n\n if (!dontPushVal)\n self.pushVal();\n\n // Remove class when reached maxTags\n if (self.options.maxTags > self.itemsArray.length)\n self.$container.removeClass('bootstrap-tagsinput-max');\n\n self.$element.trigger($.Event('itemRemoved', { item: item }));\n },\n\n /**\n * Removes all items\n */\n removeAll: function() {\n var self = this;\n\n $('.tag', self.$container).remove();\n $('option', self.$element).remove();\n\n while(self.itemsArray.length > 0)\n self.itemsArray.pop();\n\n self.pushVal();\n\n if (self.options.maxTags && !this.isEnabled())\n this.enable();\n },\n\n /**\n * Refreshes the tags so they match the text/value of their corresponding\n * item.\n */\n refresh: function() {\n var self = this;\n $('.tag', self.$container).each(function() {\n var $tag = $(this),\n item = $tag.data('item'),\n itemValue = self.options.itemValue(item),\n itemText = self.options.itemText(item),\n tagClass = self.options.tagClass(item);\n\n // Update tag's class and inner text\n $tag.attr('class', null);\n $tag.addClass('tag ' + htmlEncode(tagClass));\n $tag.contents().filter(function() {\n return this.nodeType == 3;\n })[0].nodeValue = htmlEncode(itemText);\n\n if (self.isSelect) {\n var option = $('option', self.$element).filter(function() { return $(this).data('item') === item; });\n option.attr('value', itemValue);\n }\n });\n },\n\n /**\n * Returns the items added as tags\n */\n items: function() {\n return this.itemsArray;\n },\n\n /**\n * Assembly value by retrieving the value of each item, and set it on the\n * element. \n */\n pushVal: function() {\n var self = this,\n val = $.map(self.items(), function(item) {\n return self.options.itemValue(item).toString();\n });\n\n self.$element.val(val, true).trigger('change');\n },\n\n /**\n * Initializes the tags input behaviour on the element\n */\n build: function(options) {\n var self = this;\n\n self.options = $.extend({}, defaultOptions, options);\n var typeahead = self.options.typeahead || {};\n\n // When itemValue is set, freeInput should always be false\n if (self.objectItems)\n self.options.freeInput = false;\n\n makeOptionItemFunction(self.options, 'itemValue');\n makeOptionItemFunction(self.options, 'itemText');\n makeOptionItemFunction(self.options, 'tagClass');\n\n // for backwards compatibility, self.options.source is deprecated\n if (self.options.source)\n typeahead.source = self.options.source;\n\n if (typeahead.source && $.fn.typeahead) {\n makeOptionFunction(typeahead, 'source');\n\n self.$input.typeahead({\n source: function (query, process) {\n function processItems(items) {\n var texts = [];\n\n for (var i = 0; i < items.length; i++) {\n var text = self.options.itemText(items[i]);\n map[text] = items[i];\n texts.push(text);\n }\n process(texts);\n }\n\n this.map = {};\n var map = this.map,\n data = typeahead.source(query);\n\n if ($.isFunction(data.success)) {\n // support for Angular promises\n data.success(processItems);\n } else {\n // support for functions and jquery promises\n $.when(data)\n .then(processItems);\n }\n },\n updater: function (text) {\n self.add(this.map[text]);\n },\n matcher: function (text) {\n return (text.toLowerCase().indexOf(this.query.trim().toLowerCase()) !== -1);\n },\n sorter: function (texts) {\n return texts.sort();\n },\n highlighter: function (text) {\n var regex = new RegExp( '(' + this.query + ')', 'gi' );\n return text.replace( regex, \"$1\" );\n }\n });\n }\n\n self.$container.on('click', $.proxy(function(event) {\n self.$input.focus();\n }, self));\n\n self.$container.on('keydown', 'input', $.proxy(function(event) {\n var $input = $(event.target),\n $inputWrapper = self.findInputWrapper();\n\n switch (event.which) {\n // BACKSPACE\n case 8:\n if (doGetCaretPosition($input[0]) === 0) {\n var prev = $inputWrapper.prev();\n if (prev) {\n self.remove(prev.data('item'));\n }\n }\n break;\n\n // DELETE\n case 46:\n if (doGetCaretPosition($input[0]) === 0) {\n var next = $inputWrapper.next();\n if (next) {\n self.remove(next.data('item'));\n }\n }\n break;\n\n // LEFT ARROW\n case 37:\n // Try to move the input before the previous tag\n var $prevTag = $inputWrapper.prev();\n if ($input.val().length === 0 && $prevTag[0]) {\n $prevTag.before($inputWrapper);\n $input.focus();\n }\n break;\n // RIGHT ARROW\n case 39:\n // Try to move the input after the next tag\n var $nextTag = $inputWrapper.next();\n if ($input.val().length === 0 && $nextTag[0]) {\n $nextTag.after($inputWrapper);\n $input.focus();\n }\n break;\n default:\n // When key corresponds one of the confirmKeys, add current input\n // as a new tag\n if (self.options.freeInput && $.inArray(event.which, self.options.confirmKeys) >= 0) {\n self.add($input.val());\n $input.val('');\n event.preventDefault();\n }\n }\n\n // Reset internal input's size\n $input.attr('size', Math.max(this.inputSize, $input.val().length));\n }, self));\n\n // Remove icon clicked\n self.$container.on('click', '[data-role=remove]', $.proxy(function(event) {\n self.remove($(event.target).closest('.tag').data('item'));\n }, self));\n\n // Only add existing value as tags when using strings as tags\n if (self.options.itemValue === defaultOptions.itemValue) {\n if (self.$element[0].tagName === 'INPUT') {\n self.add(self.$element.val());\n } else {\n $('option', self.$element).each(function() {\n self.add($(this).attr('value'), true);\n });\n }\n }\n },\n\n /**\n * Removes all tagsinput behaviour and unregsiter all event handlers\n */\n destroy: function() {\n var self = this;\n\n // Unbind events\n self.$container.off('keypress', 'input');\n self.$container.off('click', '[role=remove]');\n\n self.$container.remove();\n self.$element.removeData('tagsinput');\n self.$element.show();\n },\n\n /**\n * Sets focus on the tagsinput \n */\n focus: function() {\n this.$input.focus();\n },\n\n /**\n * Returns the internal input element\n */\n input: function() {\n return this.$input;\n },\n\n /**\n * Returns the element which is wrapped around the internal input. This\n * is normally the $container, but typeahead.js moves the $input element.\n */\n findInputWrapper: function() {\n var elt = this.$input[0],\n container = this.$container[0];\n while(elt && elt.parentNode !== container)\n elt = elt.parentNode;\n\n return $(elt);\n }\n };\n\n /**\n * Register JQuery plugin\n */\n $.fn.tagsinput = function(arg1, arg2) {\n var results = [];\n\n this.each(function() {\n var tagsinput = $(this).data('tagsinput');\n\n // Initialize a new tags input\n if (!tagsinput) {\n tagsinput = new TagsInput(this, arg1);\n $(this).data('tagsinput', tagsinput);\n results.push(tagsinput);\n\n if (this.tagName === 'SELECT') {\n $('option', $(this)).attr('selected', 'selected');\n }\n\n // Init tags from $(this).val()\n $(this).val($(this).val());\n } else {\n // Invoke function on existing tags input\n var retVal = tagsinput[arg1](arg2);\n if (retVal !== undefined)\n results.push(retVal);\n }\n });\n\n if ( typeof arg1 == 'string') {\n // Return the results from the invoked function calls\n return results.length > 1 ? results : results[0];\n } else {\n return results;\n }\n };\n\n $.fn.tagsinput.Constructor = TagsInput;\n \n /**\n * Most options support both a string or number as well as a function as \n * option value. This function makes sure that the option with the given\n * key in the given options is wrapped in a function\n */\n function makeOptionItemFunction(options, key) {\n if (typeof options[key] !== 'function') {\n var propertyName = options[key];\n options[key] = function(item) { return item[propertyName]; };\n }\n }\n function makeOptionFunction(options, key) {\n if (typeof options[key] !== 'function') {\n var value = options[key];\n options[key] = function() { return value; };\n }\n }\n /**\n * HtmlEncodes the given value\n */\n var htmlEncodeContainer = $('');\n function htmlEncode(value) {\n if (value) {\n return htmlEncodeContainer.text(value).html();\n } else {\n return '';\n }\n }\n\n /**\n * Returns the position of the caret in the given input field\n * http://flightschool.acylt.com/devnotes/caret-position-woes/\n */\n function doGetCaretPosition(oField) {\n var iCaretPos = 0;\n if (document.selection) {\n oField.focus ();\n var oSel = document.selection.createRange();\n oSel.moveStart ('character', -oField.value.length);\n iCaretPos = oSel.text.length;\n } else if (oField.selectionStart || oField.selectionStart == '0') {\n iCaretPos = oField.selectionStart;\n }\n return (iCaretPos);\n }\n\n /**\n * Initialize tagsinput behaviour on inputs and selects which have\n * data-role=tagsinput\n */\n $(function() {\n $(\"input[data-role=tagsinput], select[multiple][data-role=tagsinput]\").tagsinput();\n });\n})(window.jQuery);\n\n","/*!\n SerializeJSON jQuery plugin.\n https://github.com/marioizquierdo/jquery.serializeJSON\n version 2.4.2 (Oct, 2014)\n\n Copyright (c) 2014 Mario Izquierdo\n Dual licensed under the MIT (http://www.opensource.org/licenses/mit-license.php)\n and GPL (http://www.opensource.org/licenses/gpl-license.php) licenses.\n*/\n(function ($) {\n \"use strict\";\n\n // jQuery('form').serializeJSON()\n $.fn.serializeJSON = function (options) {\n var serializedObject, formAsArray, keys, type, value, _ref, f, opts;\n f = $.serializeJSON;\n opts = f.optsWithDefaults(options); // calculate values for options {parseNumbers, parseBoolens, parseNulls}\n f.validateOptions(opts);\n formAsArray = this.serializeArray(); // array of objects {name, value}\n f.readCheckboxUncheckedValues(formAsArray, this, opts); // add {name, value} of unchecked checkboxes if needed\n\n serializedObject = {};\n $.each(formAsArray, function (i, input) {\n keys = f.splitInputNameIntoKeysArray(input.name);\n type = keys.pop(); // the last element is always the type (\"string\" by default)\n if (type !== 'skip') { // easy way to skip a value\n value = f.parseValue(input.value, type, opts); // string, number, boolean or null\n if (opts.parseWithFunction && type === '_') value = opts.parseWithFunction(value, input.name); // allow for custom parsing\n f.deepSet(serializedObject, keys, value, opts);\n }\n });\n return serializedObject;\n };\n\n // Use $.serializeJSON as namespace for the auxiliar functions\n // and to define defaults\n $.serializeJSON = {\n\n defaultOptions: {\n parseNumbers: false, // convert values like \"1\", \"-2.33\" to 1, -2.33\n parseBooleans: false, // convert \"true\", \"false\" to true, false\n parseNulls: false, // convert \"null\" to null\n parseAll: false, // all of the above\n parseWithFunction: null, // to use custom parser, a function like: function(val){ return parsed_val; }\n checkboxUncheckedValue: undefined, // to include that value for unchecked checkboxes (instead of ignoring them)\n useIntKeysAsArrayIndex: false // name=\"foo[2]\" value=\"v\" => {foo: [null, null, \"v\"]}, instead of {foo: [\"2\": \"v\"]}\n },\n\n // Merge options with defaults to get {parseNumbers, parseBoolens, parseNulls, useIntKeysAsArrayIndex}\n optsWithDefaults: function(options) {\n var f, parseAll;\n if (options == null) options = {}; // arg default value = {}\n f = $.serializeJSON;\n parseAll = f.optWithDefaults('parseAll', options);\n return {\n parseNumbers: parseAll || f.optWithDefaults('parseNumbers', options),\n parseBooleans: parseAll || f.optWithDefaults('parseBooleans', options),\n parseNulls: parseAll || f.optWithDefaults('parseNulls', options),\n parseWithFunction: f.optWithDefaults('parseWithFunction', options),\n checkboxUncheckedValue: f.optWithDefaults('checkboxUncheckedValue', options),\n useIntKeysAsArrayIndex: f.optWithDefaults('useIntKeysAsArrayIndex', options)\n }\n },\n\n optWithDefaults: function(key, options) {\n return (options[key] !== false) && (options[key] !== '') && (options[key] || $.serializeJSON.defaultOptions[key]);\n },\n\n validateOptions: function(opts) {\n var opt, validOpts;\n validOpts = ['parseNumbers', 'parseBooleans', 'parseNulls', 'parseAll', 'parseWithFunction', 'checkboxUncheckedValue', 'useIntKeysAsArrayIndex']\n for (opt in opts) {\n if (validOpts.indexOf(opt) === -1) {\n throw new Error(\"serializeJSON ERROR: invalid option '\" + opt + \"'. Please use one of \" + validOpts.join(','));\n }\n }\n },\n\n // Convert the string to a number, boolean or null, depending on the enable option and the string format.\n parseValue: function(str, type, opts) {\n var value, f;\n f = $.serializeJSON;\n if (type == 'string') return str; // force string\n if (type == 'number' || (opts.parseNumbers && f.isNumeric(str))) return Number(str); // number\n if (type == 'boolean' || (opts.parseBooleans && (str === \"true\" || str === \"false\"))) return ([\"false\", \"null\", \"undefined\", \"\", \"0\"].indexOf(str) === -1); // boolean\n if (type == 'null' || (opts.parseNulls && str == \"null\")) return [\"false\", \"null\", \"undefined\", \"\", \"0\"].indexOf(str) !== -1 ? null : str; // null\n if (type == 'array' || type == 'object') return JSON.parse(str); // array or objects require JSON\n if (type == 'auto') return f.parseValue(str, null, {parseNumbers: true, parseBooleans: true, parseNulls: true}); // try again with something like \"parseAll\"\n return str; // otherwise, keep same string\n },\n\n isObject: function(obj) { return obj === Object(obj); }, // is this variable an object?\n isUndefined: function(obj) { return obj === void 0; }, // safe check for undefined values\n isValidArrayIndex: function(val) { return /^[0-9]+$/.test(String(val)); }, // 1,2,3,4 ... are valid array indexes\n isNumeric: function(obj) { return obj - parseFloat(obj) >= 0; }, // taken from jQuery.isNumeric implementation. Not using jQuery.isNumeric to support old jQuery and Zepto versions\n\n // Split the input name in programatically readable keys.\n // The last element is always the type (default \"_\").\n // Examples:\n // \"foo\" => ['foo', '_']\n // \"foo:string\" => ['foo', 'string']\n // \"foo:boolean\" => ['foo', 'boolean']\n // \"[foo]\" => ['foo', '_']\n // \"foo[inn][bar]\" => ['foo', 'inn', 'bar', '_']\n // \"foo[inn[bar]]\" => ['foo', 'inn', 'bar', '_']\n // \"foo[inn][arr][0]\" => ['foo', 'inn', 'arr', '0', '_']\n // \"arr[][val]\" => ['arr', '', 'val', '_']\n // \"arr[][val]:null\" => ['arr', '', 'val', 'null']\n splitInputNameIntoKeysArray: function (name) {\n var keys, nameWithoutType, type, _ref, f;\n f = $.serializeJSON;\n _ref = f.extractTypeFromInputName(name), nameWithoutType = _ref[0], type = _ref[1];\n keys = nameWithoutType.split('['); // split string into array\n keys = $.map(keys, function (key) { return key.replace(/]/g, ''); }); // remove closing brackets\n if (keys[0] === '') { keys.shift(); } // ensure no opening bracket (\"[foo][inn]\" should be same as \"foo[inn]\")\n keys.push(type); // add type at the end\n return keys;\n },\n\n // Returns [name-without-type, type] from name.\n // \"foo\" => [\"foo\", \"_\"]\n // \"foo:boolean\" => [\"foo\", \"boolean\"]\n // \"foo[bar]:null\" => [\"foo[bar]\", \"null\"]\n extractTypeFromInputName: function(name) {\n var match, f;\n f = $.serializeJSON;\n if (match = name.match(/(.*):([^:]+)$/)){\n var validTypes = ['string', 'number', 'boolean', 'null', 'array', 'object', 'skip', 'auto']; // validate type\n if (validTypes.indexOf(match[2]) !== -1) {\n return [match[1], match[2]];\n } else {\n throw new Error(\"serializeJSON ERROR: Invalid type \" + match[2] + \" found in input name '\" + name + \"', please use one of \" + validTypes.join(', '))\n }\n } else {\n return [name, '_']; // no defined type, then use parse options\n }\n },\n\n // Set a value in an object or array, using multiple keys to set in a nested object or array:\n //\n // deepSet(obj, ['foo'], v) // obj['foo'] = v\n // deepSet(obj, ['foo', 'inn'], v) // obj['foo']['inn'] = v // Create the inner obj['foo'] object, if needed\n // deepSet(obj, ['foo', 'inn', '123'], v) // obj['foo']['arr']['123'] = v //\n //\n // deepSet(obj, ['0'], v) // obj['0'] = v\n // deepSet(arr, ['0'], v, {useIntKeysAsArrayIndex: true}) // arr[0] = v\n // deepSet(arr, [''], v) // arr.push(v)\n // deepSet(obj, ['arr', ''], v) // obj['arr'].push(v)\n //\n // arr = [];\n // deepSet(arr, ['', v] // arr => [v]\n // deepSet(arr, ['', 'foo'], v) // arr => [v, {foo: v}]\n // deepSet(arr, ['', 'bar'], v) // arr => [v, {foo: v, bar: v}]\n // deepSet(arr, ['', 'bar'], v) // arr => [v, {foo: v, bar: v}, {bar: v}]\n //\n deepSet: function (o, keys, value, opts) {\n var key, nextKey, tail, lastIdx, lastVal, f;\n if (opts == null) opts = {};\n f = $.serializeJSON;\n if (f.isUndefined(o)) { throw new Error(\"ArgumentError: param 'o' expected to be an object or array, found undefined\"); }\n if (!keys || keys.length === 0) { throw new Error(\"ArgumentError: param 'keys' expected to be an array with least one element\"); }\n\n key = keys[0];\n\n // Only one key, then it's not a deepSet, just assign the value.\n if (keys.length === 1) {\n if (key === '') {\n o.push(value); // '' is used to push values into the array (assume o is an array)\n } else {\n o[key] = value; // other keys can be used as object keys or array indexes\n }\n\n // With more keys is a deepSet. Apply recursively.\n } else {\n\n nextKey = keys[1];\n\n // '' is used to push values into the array,\n // with nextKey, set the value into the same object, in object[nextKey].\n // Covers the case of ['', 'foo'] and ['', 'var'] to push the object {foo, var}, and the case of nested arrays.\n if (key === '') {\n lastIdx = o.length - 1; // asume o is array\n lastVal = o[lastIdx];\n if (f.isObject(lastVal) && (f.isUndefined(lastVal[nextKey]) || keys.length > 2)) { // if nextKey is not present in the last object element, or there are more keys to deep set\n key = lastIdx; // then set the new value in the same object element\n } else {\n key = lastIdx + 1; // otherwise, point to set the next index in the array\n }\n }\n\n // o[key] defaults to object or array, depending if nextKey is an array index (int or '') or an object key (string)\n if (f.isUndefined(o[key])) {\n if (nextKey === '') { // '' is used to push values into the array.\n o[key] = [];\n } else if (opts.useIntKeysAsArrayIndex && f.isValidArrayIndex(nextKey)) { // if 1, 2, 3 ... then use an array, where nextKey is the index\n o[key] = [];\n } else { // for anything else, use an object, where nextKey is going to be the attribute name\n o[key] = {};\n }\n }\n\n // Recursively set the inner object\n tail = keys.slice(1);\n f.deepSet(o[key], tail, value, opts);\n }\n },\n\n // Fill the formAsArray object with values for the unchecked checkbox inputs,\n // using the same format as the jquery.serializeArray function.\n // The value of the unchecked values is determined from the opts.checkboxUncheckedValue\n // and/or the data-unchecked-value attribute of the inputs.\n readCheckboxUncheckedValues: function (formAsArray, $form, opts) {\n var selector, $uncheckedCheckboxes, $el, dataUncheckedValue, f;\n if (opts == null) opts = {};\n f = $.serializeJSON;\n\n selector = 'input[type=checkbox][name]:not(:checked,[disabled])';\n $uncheckedCheckboxes = $form.find(selector).add($form.filter(selector));\n $uncheckedCheckboxes.each(function (i, el) {\n $el = $(el);\n dataUncheckedValue = $el.attr('data-unchecked-value');\n if(dataUncheckedValue) { // data-unchecked-value has precedence over option opts.checkboxUncheckedValue\n formAsArray.push({name: el.name, value: dataUncheckedValue});\n } else {\n if (!f.isUndefined(opts.checkboxUncheckedValue)) {\n formAsArray.push({name: el.name, value: opts.checkboxUncheckedValue});\n }\n }\n });\n }\n\n };\n\n}(window.jQuery || window.Zepto || window.$));","/*\n * Leaflet-IIIF 3.0.0\n * IIIF Viewer for Leaflet\n * by Jack Reed, @mejackreed\n */\n\nL.TileLayer.Iiif = L.TileLayer.extend({\n options: {\n continuousWorld: true,\n tileSize: 256,\n updateWhenIdle: true,\n tileFormat: 'jpg',\n fitBounds: true,\n setMaxBounds: false\n },\n\n initialize: function(url, options) {\n options = typeof options !== 'undefined' ? options : {};\n\n if (options.maxZoom) {\n this._customMaxZoom = true;\n }\n\n // Check for explicit tileSize set\n if (options.tileSize) {\n this._explicitTileSize = true;\n }\n\n // Check for an explicit quality\n if (options.quality) {\n this._explicitQuality = true;\n }\n\n options = L.setOptions(this, options);\n this._infoPromise = null;\n this._infoUrl = url;\n this._baseUrl = this._templateUrl();\n this._getInfo();\n },\n getTileUrl: function(coords) {\n var _this = this,\n x = coords.x,\n y = (coords.y),\n zoom = _this._getZoomForUrl(),\n scale = Math.pow(2, _this.maxNativeZoom - zoom),\n tileBaseSize = _this.options.tileSize * scale,\n minx = (x * tileBaseSize),\n miny = (y * tileBaseSize),\n maxx = Math.min(minx + tileBaseSize, _this.x),\n maxy = Math.min(miny + tileBaseSize, _this.y);\n\n var xDiff = (maxx - minx);\n var yDiff = (maxy - miny);\n\n // Canonical URI Syntax for v2\n var size = Math.ceil(xDiff / scale) + ',';\n if (_this.type === 'ImageService3') {\n // Cannonical URI Syntax for v3\n size = size + Math.ceil(yDiff / scale);\n }\n\n return L.Util.template(this._baseUrl, L.extend({\n format: _this.options.tileFormat,\n quality: _this.quality,\n region: [minx, miny, xDiff, yDiff].join(','),\n rotation: 0,\n size: size\n }, this.options));\n },\n onAdd: function(map) {\n var _this = this;\n\n // Wait for info.json fetch and parse to complete\n Promise.all([_this._infoPromise]).then(function() {\n // Store unmutated imageSizes\n _this._imageSizesOriginal = _this._imageSizes.slice(0);\n\n // Set maxZoom for map\n map._layersMaxZoom = _this.maxZoom;\n\n // Call add TileLayer\n L.TileLayer.prototype.onAdd.call(_this, map);\n\n // Set minZoom and minNativeZoom based on how the imageSizes match up\n var smallestImage = _this._imageSizes[0];\n var mapSize = _this._map.getSize();\n var newMinZoom = 0;\n // Loop back through 5 times to see if a better fit can be found.\n for (var i = 1; i <= 5; i++) {\n if (smallestImage.x > mapSize.x || smallestImage.y > mapSize.y) {\n smallestImage = smallestImage.divideBy(2);\n _this._imageSizes.unshift(smallestImage);\n newMinZoom = -i;\n } else {\n break;\n }\n }\n _this.options.minZoom = newMinZoom;\n _this.options.minNativeZoom = newMinZoom;\n _this._prev_map_layersMinZoom = _this._map._layersMinZoom;\n _this._map._layersMinZoom = newMinZoom;\n\n if (_this.options.fitBounds) {\n _this._fitBounds();\n }\n\n if(_this.options.setMaxBounds) {\n _this._setMaxBounds();\n }\n\n // Reset tile sizes to handle non 256x256 IIIF tiles\n _this.on('tileload', function(tile, url) {\n\n var height = tile.tile.naturalHeight,\n width = tile.tile.naturalWidth;\n\n // No need to resize if tile is 256 x 256\n if (height === 256 && width === 256) return;\n\n tile.tile.style.width = width + 'px';\n tile.tile.style.height = height + 'px';\n\n });\n })\n .catch(function(err){\n console.error(err);\n });\n },\n onRemove: function(map) {\n var _this = this;\n\n map._layersMinZoom = _this._prev_map_layersMinZoom;\n _this._imageSizes = _this._imageSizesOriginal;\n\n // Remove maxBounds set for this image\n if(_this.options.setMaxBounds) {\n map.setMaxBounds(null);\n }\n\n // Call remove TileLayer\n L.TileLayer.prototype.onRemove.call(_this, map);\n\n },\n _fitBounds: function() {\n var _this = this;\n\n // Find best zoom level and center map\n var initialZoom = _this._getInitialZoom(_this._map.getSize());\n var offset = _this._imageSizes.length - 1 - _this.options.maxNativeZoom;\n var imageSize = _this._imageSizes[initialZoom + offset];\n var sw = _this._map.options.crs.pointToLatLng(L.point(0, imageSize.y), initialZoom);\n var ne = _this._map.options.crs.pointToLatLng(L.point(imageSize.x, 0), initialZoom);\n var bounds = L.latLngBounds(sw, ne);\n\n _this._map.fitBounds(bounds, true);\n },\n _setMaxBounds: function() {\n var _this = this;\n\n // Find best zoom level, center map, and constrain viewer\n var initialZoom = _this._getInitialZoom(_this._map.getSize());\n var imageSize = _this._imageSizes[initialZoom];\n var sw = _this._map.options.crs.pointToLatLng(L.point(0, imageSize.y), initialZoom);\n var ne = _this._map.options.crs.pointToLatLng(L.point(imageSize.x, 0), initialZoom);\n var bounds = L.latLngBounds(sw, ne);\n\n _this._map.setMaxBounds(bounds, true);\n },\n _getInfo: function() {\n var _this = this;\n\n _this._infoPromise = fetch(_this._infoUrl)\n .then(function(response) {\n return response.json();\n })\n .catch(function(err){\n console.error(err);\n })\n .then(function(data) {\n _this.y = data.height;\n _this.x = data.width;\n\n var tierSizes = [],\n imageSizes = [],\n scale,\n width_,\n height_,\n tilesX_,\n tilesY_;\n\n // Set quality based off of IIIF version\n if (data.profile instanceof Array) {\n _this.profile = data.profile[0];\n }else {\n _this.profile = data.profile;\n }\n _this.type = data.type;\n\n _this._setQuality();\n\n // Unless an explicit tileSize is set, use a preferred tileSize\n if (!_this._explicitTileSize) {\n // Set the default first\n _this.options.tileSize = 256;\n if (data.tiles) {\n // Image API 2.0 Case\n _this.options.tileSize = data.tiles[0].width;\n } else if (data.tile_width){\n // Image API 1.1 Case\n _this.options.tileSize = data.tile_width;\n }\n }\n\n function ceilLog2(x) {\n return Math.ceil(Math.log(x) / Math.LN2);\n };\n\n // Calculates maximum native zoom for the layer\n _this.maxNativeZoom = Math.max(\n ceilLog2(_this.x / _this.options.tileSize),\n ceilLog2(_this.y / _this.options.tileSize),\n 0\n );\n _this.options.maxNativeZoom = _this.maxNativeZoom;\n\n // Enable zooming further than native if maxZoom option supplied\n if (_this._customMaxZoom && _this.options.maxZoom > _this.maxNativeZoom) {\n _this.maxZoom = _this.options.maxZoom;\n }\n else {\n _this.maxZoom = _this.maxNativeZoom;\n }\n\n for (var i = 0; i <= _this.maxZoom; i++) {\n scale = Math.pow(2, _this.maxNativeZoom - i);\n width_ = Math.ceil(_this.x / scale);\n height_ = Math.ceil(_this.y / scale);\n tilesX_ = Math.ceil(width_ / _this.options.tileSize);\n tilesY_ = Math.ceil(height_ / _this.options.tileSize);\n tierSizes.push([tilesX_, tilesY_]);\n imageSizes.push(L.point(width_,height_));\n }\n\n _this._tierSizes = tierSizes;\n _this._imageSizes = imageSizes;\n })\n .catch(function(err){\n console.error(err);\n });\n\n },\n\n _setQuality: function() {\n var _this = this;\n var profileToCheck = _this.profile;\n\n if (_this._explicitQuality) {\n return;\n }\n\n // If profile is an object\n if (typeof(profileToCheck) === 'object') {\n profileToCheck = profileToCheck['@id'];\n }\n\n // Set the quality based on the IIIF compliance level\n switch (true) {\n case /^http:\\/\\/library.stanford.edu\\/iiif\\/image-api\\/1.1\\/compliance.html.*$/.test(profileToCheck):\n _this.options.quality = 'native';\n break;\n // Assume later profiles and set to default\n default:\n _this.options.quality = 'default';\n break;\n }\n },\n\n _infoToBaseUrl: function() {\n return this._infoUrl.replace('info.json', '');\n },\n _templateUrl: function() {\n return this._infoToBaseUrl() + '{region}/{size}/{rotation}/{quality}.{format}';\n },\n _isValidTile: function(coords) {\n var _this = this;\n var zoom = _this._getZoomForUrl();\n var sizes = _this._tierSizes[zoom];\n var x = coords.x;\n var y = coords.y;\n if (zoom < 0 && x >= 0 && y >= 0) {\n return true;\n }\n\n if (!sizes) return false;\n if (x < 0 || sizes[0] <= x || y < 0 || sizes[1] <= y) {\n return false;\n }else {\n return true;\n }\n },\n _tileShouldBeLoaded: function(coords) {\n return this._isValidTile(coords);\n },\n _getInitialZoom: function (mapSize) {\n var _this = this;\n var tolerance = 0.8;\n var imageSize;\n // Calculate an offset between the zoom levels and the array accessors\n var offset = _this._imageSizes.length - 1 - _this.options.maxNativeZoom;\n for (var i = _this._imageSizes.length - 1; i >= 0; i--) {\n imageSize = _this._imageSizes[i];\n if (imageSize.x * tolerance < mapSize.x && imageSize.y * tolerance < mapSize.y) {\n return i - offset;\n }\n }\n // return a default zoom\n return 2;\n }\n});\n\nL.tileLayer.iiif = function(url, options) {\n return new L.TileLayer.Iiif(url, options);\n};\n","'use strict';\n(function (factory, window) {\n /*globals define, module, require*/\n\n // define an AMD module that relies on 'leaflet'\n if (typeof define === 'function' && define.amd) {\n define(['leaflet'], factory);\n\n\n // define a Common JS module that relies on 'leaflet'\n } else if (typeof exports === 'object') {\n module.exports = factory(require('leaflet'));\n }\n\n // attach your plugin to the global 'L' variable\n if(typeof window !== 'undefined' && window.L){\n factory(window.L);\n }\n\n}(function (L) {\n // 🍂miniclass CancelableEvent (Event objects)\n // 🍂method cancel()\n // Cancel any subsequent action.\n\n // 🍂miniclass VertexEvent (Event objects)\n // 🍂property vertex: VertexMarker\n // The vertex that fires the event.\n\n // 🍂miniclass ShapeEvent (Event objects)\n // 🍂property shape: Array\n // The shape (LatLngs array) subject of the action.\n\n // 🍂miniclass CancelableVertexEvent (Event objects)\n // 🍂inherits VertexEvent\n // 🍂inherits CancelableEvent\n\n // 🍂miniclass CancelableShapeEvent (Event objects)\n // 🍂inherits ShapeEvent\n // 🍂inherits CancelableEvent\n\n // 🍂miniclass LayerEvent (Event objects)\n // 🍂property layer: object\n // The Layer (Marker, Polyline…) subject of the action.\n\n // 🍂namespace Editable; 🍂class Editable; 🍂aka L.Editable\n // Main edition handler. By default, it is attached to the map\n // as `map.editTools` property.\n // Leaflet.Editable is made to be fully extendable. You have three ways to customize\n // the behaviour: using options, listening to events, or extending.\n L.Editable = L.Evented.extend({\n\n statics: {\n FORWARD: 1,\n BACKWARD: -1\n },\n\n options: {\n\n // You can pass them when creating a map using the `editOptions` key.\n // 🍂option zIndex: int = 1000\n // The default zIndex of the editing tools.\n zIndex: 1000,\n\n // 🍂option polygonClass: class = L.Polygon\n // Class to be used when creating a new Polygon.\n polygonClass: L.Polygon,\n\n // 🍂option polylineClass: class = L.Polyline\n // Class to be used when creating a new Polyline.\n polylineClass: L.Polyline,\n\n // 🍂option markerClass: class = L.Marker\n // Class to be used when creating a new Marker.\n markerClass: L.Marker,\n\n // 🍂option rectangleClass: class = L.Rectangle\n // Class to be used when creating a new Rectangle.\n rectangleClass: L.Rectangle,\n\n // 🍂option circleClass: class = L.Circle\n // Class to be used when creating a new Circle.\n circleClass: L.Circle,\n\n // 🍂option drawingCSSClass: string = 'leaflet-editable-drawing'\n // CSS class to be added to the map container while drawing.\n drawingCSSClass: 'leaflet-editable-drawing',\n\n // 🍂option drawingCursor: const = 'crosshair'\n // Cursor mode set to the map while drawing.\n drawingCursor: 'crosshair',\n\n // 🍂option editLayer: Layer = new L.LayerGroup()\n // Layer used to store edit tools (vertex, line guide…).\n editLayer: undefined,\n\n // 🍂option featuresLayer: Layer = new L.LayerGroup()\n // Default layer used to store drawn features (Marker, Polyline…).\n featuresLayer: undefined,\n\n // 🍂option polylineEditorClass: class = PolylineEditor\n // Class to be used as Polyline editor.\n polylineEditorClass: undefined,\n\n // 🍂option polygonEditorClass: class = PolygonEditor\n // Class to be used as Polygon editor.\n polygonEditorClass: undefined,\n\n // 🍂option markerEditorClass: class = MarkerEditor\n // Class to be used as Marker editor.\n markerEditorClass: undefined,\n\n // 🍂option rectangleEditorClass: class = RectangleEditor\n // Class to be used as Rectangle editor.\n rectangleEditorClass: undefined,\n\n // 🍂option circleEditorClass: class = CircleEditor\n // Class to be used as Circle editor.\n circleEditorClass: undefined,\n\n // 🍂option lineGuideOptions: hash = {}\n // Options to be passed to the line guides.\n lineGuideOptions: {},\n\n // 🍂option skipMiddleMarkers: boolean = false\n // Set this to true if you don't want middle markers.\n skipMiddleMarkers: false\n\n },\n\n initialize: function (map, options) {\n L.setOptions(this, options);\n this._lastZIndex = this.options.zIndex;\n this.map = map;\n this.editLayer = this.createEditLayer();\n this.featuresLayer = this.createFeaturesLayer();\n this.forwardLineGuide = this.createLineGuide();\n this.backwardLineGuide = this.createLineGuide();\n },\n\n fireAndForward: function (type, e) {\n e = e || {};\n e.editTools = this;\n this.fire(type, e);\n this.map.fire(type, e);\n },\n\n createLineGuide: function () {\n var options = L.extend({dashArray: '5,10', weight: 1, interactive: false}, this.options.lineGuideOptions);\n return L.polyline([], options);\n },\n\n createVertexIcon: function (options) {\n return L.Browser.touch ? new L.Editable.TouchVertexIcon(options) : new L.Editable.VertexIcon(options);\n },\n\n createEditLayer: function () {\n return this.options.editLayer || new L.LayerGroup().addTo(this.map);\n },\n\n createFeaturesLayer: function () {\n return this.options.featuresLayer || new L.LayerGroup().addTo(this.map);\n },\n\n moveForwardLineGuide: function (latlng) {\n if (this.forwardLineGuide._latlngs.length) {\n this.forwardLineGuide._latlngs[1] = latlng;\n this.forwardLineGuide._bounds.extend(latlng);\n this.forwardLineGuide.redraw();\n }\n },\n\n moveBackwardLineGuide: function (latlng) {\n if (this.backwardLineGuide._latlngs.length) {\n this.backwardLineGuide._latlngs[1] = latlng;\n this.backwardLineGuide._bounds.extend(latlng);\n this.backwardLineGuide.redraw();\n }\n },\n\n anchorForwardLineGuide: function (latlng) {\n this.forwardLineGuide._latlngs[0] = latlng;\n this.forwardLineGuide._bounds.extend(latlng);\n this.forwardLineGuide.redraw();\n },\n\n anchorBackwardLineGuide: function (latlng) {\n this.backwardLineGuide._latlngs[0] = latlng;\n this.backwardLineGuide._bounds.extend(latlng);\n this.backwardLineGuide.redraw();\n },\n\n attachForwardLineGuide: function () {\n this.editLayer.addLayer(this.forwardLineGuide);\n },\n\n attachBackwardLineGuide: function () {\n this.editLayer.addLayer(this.backwardLineGuide);\n },\n\n detachForwardLineGuide: function () {\n this.forwardLineGuide.setLatLngs([]);\n this.editLayer.removeLayer(this.forwardLineGuide);\n },\n\n detachBackwardLineGuide: function () {\n this.backwardLineGuide.setLatLngs([]);\n this.editLayer.removeLayer(this.backwardLineGuide);\n },\n\n blockEvents: function () {\n // Hack: force map not to listen to other layers events while drawing.\n if (!this._oldTargets) {\n this._oldTargets = this.map._targets;\n this.map._targets = {};\n }\n },\n\n unblockEvents: function () {\n if (this._oldTargets) {\n // Reset, but keep targets created while drawing.\n this.map._targets = L.extend(this.map._targets, this._oldTargets);\n delete this._oldTargets;\n }\n },\n\n registerForDrawing: function (editor) {\n if (this._drawingEditor) this.unregisterForDrawing(this._drawingEditor);\n this.blockEvents();\n editor.reset(); // Make sure editor tools still receive events.\n this._drawingEditor = editor;\n this.map.on('mousemove touchmove', editor.onDrawingMouseMove, editor);\n this.map.on('mousedown', this.onMousedown, this);\n this.map.on('mouseup', this.onMouseup, this);\n L.DomUtil.addClass(this.map._container, this.options.drawingCSSClass);\n this.defaultMapCursor = this.map._container.style.cursor;\n this.map._container.style.cursor = this.options.drawingCursor;\n },\n\n unregisterForDrawing: function (editor) {\n this.unblockEvents();\n L.DomUtil.removeClass(this.map._container, this.options.drawingCSSClass);\n this.map._container.style.cursor = this.defaultMapCursor;\n editor = editor || this._drawingEditor;\n if (!editor) return;\n this.map.off('mousemove touchmove', editor.onDrawingMouseMove, editor);\n this.map.off('mousedown', this.onMousedown, this);\n this.map.off('mouseup', this.onMouseup, this);\n if (editor !== this._drawingEditor) return;\n delete this._drawingEditor;\n if (editor._drawing) editor.cancelDrawing();\n },\n\n onMousedown: function (e) {\n this._mouseDown = e;\n this._drawingEditor.onDrawingMouseDown(e);\n },\n\n onMouseup: function (e) {\n if (this._mouseDown) {\n var editor = this._drawingEditor,\n mouseDown = this._mouseDown;\n this._mouseDown = null;\n editor.onDrawingMouseUp(e);\n if (this._drawingEditor !== editor) return; // onDrawingMouseUp may call unregisterFromDrawing.\n var origin = L.point(mouseDown.originalEvent.clientX, mouseDown.originalEvent.clientY);\n var distance = L.point(e.originalEvent.clientX, e.originalEvent.clientY).distanceTo(origin);\n if (Math.abs(distance) < 9 * (window.devicePixelRatio || 1)) this._drawingEditor.onDrawingClick(e);\n }\n },\n\n // 🍂section Public methods\n // You will generally access them by the `map.editTools`\n // instance:\n //\n // `map.editTools.startPolyline();`\n\n // 🍂method drawing(): boolean\n // Return true if any drawing action is ongoing.\n drawing: function () {\n return this._drawingEditor && this._drawingEditor.drawing();\n },\n\n // 🍂method stopDrawing()\n // When you need to stop any ongoing drawing, without needing to know which editor is active.\n stopDrawing: function () {\n this.unregisterForDrawing();\n },\n\n // 🍂method commitDrawing()\n // When you need to commit any ongoing drawing, without needing to know which editor is active.\n commitDrawing: function (e) {\n if (!this._drawingEditor) return;\n this._drawingEditor.commitDrawing(e);\n },\n\n connectCreatedToMap: function (layer) {\n return this.featuresLayer.addLayer(layer);\n },\n\n // 🍂method startPolyline(latlng: L.LatLng, options: hash): L.Polyline\n // Start drawing a Polyline. If `latlng` is given, a first point will be added. In any case, continuing on user click.\n // If `options` is given, it will be passed to the Polyline class constructor.\n startPolyline: function (latlng, options) {\n var line = this.createPolyline([], options);\n line.enableEdit(this.map).newShape(latlng);\n return line;\n },\n\n // 🍂method startPolygon(latlng: L.LatLng, options: hash): L.Polygon\n // Start drawing a Polygon. If `latlng` is given, a first point will be added. In any case, continuing on user click.\n // If `options` is given, it will be passed to the Polygon class constructor.\n startPolygon: function (latlng, options) {\n var polygon = this.createPolygon([], options);\n polygon.enableEdit(this.map).newShape(latlng);\n return polygon;\n },\n\n // 🍂method startMarker(latlng: L.LatLng, options: hash): L.Marker\n // Start adding a Marker. If `latlng` is given, the Marker will be shown first at this point.\n // In any case, it will follow the user mouse, and will have a final `latlng` on next click (or touch).\n // If `options` is given, it will be passed to the Marker class constructor.\n startMarker: function (latlng, options) {\n latlng = latlng || this.map.getCenter().clone();\n var marker = this.createMarker(latlng, options);\n marker.enableEdit(this.map).startDrawing();\n return marker;\n },\n\n // 🍂method startRectangle(latlng: L.LatLng, options: hash): L.Rectangle\n // Start drawing a Rectangle. If `latlng` is given, the Rectangle anchor will be added. In any case, continuing on user drag.\n // If `options` is given, it will be passed to the Rectangle class constructor.\n startRectangle: function(latlng, options) {\n var corner = latlng || L.latLng([0, 0]);\n var bounds = new L.LatLngBounds(corner, corner);\n var rectangle = this.createRectangle(bounds, options);\n rectangle.enableEdit(this.map).startDrawing();\n return rectangle;\n },\n\n // 🍂method startCircle(latlng: L.LatLng, options: hash): L.Circle\n // Start drawing a Circle. If `latlng` is given, the Circle anchor will be added. In any case, continuing on user drag.\n // If `options` is given, it will be passed to the Circle class constructor.\n startCircle: function (latlng, options) {\n latlng = latlng || this.map.getCenter().clone();\n var circle = this.createCircle(latlng, options);\n circle.enableEdit(this.map).startDrawing();\n return circle;\n },\n\n startHole: function (editor, latlng) {\n editor.newHole(latlng);\n },\n\n createLayer: function (klass, latlngs, options) {\n options = L.Util.extend({editOptions: {editTools: this}}, options);\n var layer = new klass(latlngs, options);\n // 🍂namespace Editable\n // 🍂event editable:created: LayerEvent\n // Fired when a new feature (Marker, Polyline…) is created.\n this.fireAndForward('editable:created', {layer: layer});\n return layer;\n },\n\n createPolyline: function (latlngs, options) {\n return this.createLayer(options && options.polylineClass || this.options.polylineClass, latlngs, options);\n },\n\n createPolygon: function (latlngs, options) {\n return this.createLayer(options && options.polygonClass || this.options.polygonClass, latlngs, options);\n },\n\n createMarker: function (latlng, options) {\n return this.createLayer(options && options.markerClass || this.options.markerClass, latlng, options);\n },\n\n createRectangle: function (bounds, options) {\n return this.createLayer(options && options.rectangleClass || this.options.rectangleClass, bounds, options);\n },\n\n createCircle: function (latlng, options) {\n return this.createLayer(options && options.circleClass || this.options.circleClass, latlng, options);\n }\n\n });\n\n L.extend(L.Editable, {\n\n makeCancellable: function (e) {\n e.cancel = function () {\n e._cancelled = true;\n };\n }\n\n });\n\n // 🍂namespace Map; 🍂class Map\n // Leaflet.Editable add options and events to the `L.Map` object.\n // See `Editable` events for the list of events fired on the Map.\n // 🍂example\n //\n // ```js\n // var map = L.map('map', {\n // editable: true,\n // editOptions: {\n // …\n // }\n // });\n // ```\n // 🍂section Editable Map Options\n L.Map.mergeOptions({\n\n // 🍂namespace Map\n // 🍂section Map Options\n // 🍂option editToolsClass: class = L.Editable\n // Class to be used as vertex, for path editing.\n editToolsClass: L.Editable,\n\n // 🍂option editable: boolean = false\n // Whether to create a L.Editable instance at map init.\n editable: false,\n\n // 🍂option editOptions: hash = {}\n // Options to pass to L.Editable when instanciating.\n editOptions: {}\n\n });\n\n L.Map.addInitHook(function () {\n\n this.whenReady(function () {\n if (this.options.editable) {\n this.editTools = new this.options.editToolsClass(this, this.options.editOptions);\n }\n });\n\n });\n\n L.Editable.VertexIcon = L.DivIcon.extend({\n\n options: {\n iconSize: new L.Point(8, 8)\n }\n\n });\n\n L.Editable.TouchVertexIcon = L.Editable.VertexIcon.extend({\n\n options: {\n iconSize: new L.Point(20, 20)\n }\n\n });\n\n\n // 🍂namespace Editable; 🍂class VertexMarker; Handler for dragging path vertices.\n L.Editable.VertexMarker = L.Marker.extend({\n\n options: {\n draggable: true,\n className: 'leaflet-div-icon leaflet-vertex-icon'\n },\n\n\n // 🍂section Public methods\n // The marker used to handle path vertex. You will usually interact with a `VertexMarker`\n // instance when listening for events like `editable:vertex:ctrlclick`.\n\n initialize: function (latlng, latlngs, editor, options) {\n // We don't use this._latlng, because on drag Leaflet replace it while\n // we want to keep reference.\n this.latlng = latlng;\n this.latlngs = latlngs;\n this.editor = editor;\n L.Marker.prototype.initialize.call(this, latlng, options);\n this.options.icon = this.editor.tools.createVertexIcon({className: this.options.className});\n this.latlng.__vertex = this;\n this.editor.editLayer.addLayer(this);\n this.setZIndexOffset(editor.tools._lastZIndex + 1);\n },\n\n onAdd: function (map) {\n L.Marker.prototype.onAdd.call(this, map);\n this.on('drag', this.onDrag);\n this.on('dragstart', this.onDragStart);\n this.on('dragend', this.onDragEnd);\n this.on('mouseup', this.onMouseup);\n this.on('click', this.onClick);\n this.on('contextmenu', this.onContextMenu);\n this.on('mousedown touchstart', this.onMouseDown);\n this.addMiddleMarkers();\n },\n\n onRemove: function (map) {\n if (this.middleMarker) this.middleMarker.delete();\n delete this.latlng.__vertex;\n this.off('drag', this.onDrag);\n this.off('dragstart', this.onDragStart);\n this.off('dragend', this.onDragEnd);\n this.off('mouseup', this.onMouseup);\n this.off('click', this.onClick);\n this.off('contextmenu', this.onContextMenu);\n this.off('mousedown touchstart', this.onMouseDown);\n L.Marker.prototype.onRemove.call(this, map);\n },\n\n onDrag: function (e) {\n e.vertex = this;\n this.editor.onVertexMarkerDrag(e);\n var iconPos = L.DomUtil.getPosition(this._icon),\n latlng = this._map.layerPointToLatLng(iconPos);\n this.latlng.update(latlng);\n this._latlng = this.latlng; // Push back to Leaflet our reference.\n this.editor.refresh();\n if (this.middleMarker) this.middleMarker.updateLatLng();\n var next = this.getNext();\n if (next && next.middleMarker) next.middleMarker.updateLatLng();\n },\n\n onDragStart: function (e) {\n e.vertex = this;\n this.editor.onVertexMarkerDragStart(e);\n },\n\n onDragEnd: function (e) {\n e.vertex = this;\n this.editor.onVertexMarkerDragEnd(e);\n },\n\n onClick: function (e) {\n e.vertex = this;\n this.editor.onVertexMarkerClick(e);\n },\n\n onMouseup: function (e) {\n L.DomEvent.stop(e);\n e.vertex = this;\n this.editor.map.fire('mouseup', e);\n },\n\n onContextMenu: function (e) {\n e.vertex = this;\n this.editor.onVertexMarkerContextMenu(e);\n },\n\n onMouseDown: function (e) {\n e.vertex = this;\n this.editor.onVertexMarkerMouseDown(e);\n },\n\n // 🍂method delete()\n // Delete a vertex and the related LatLng.\n delete: function () {\n var next = this.getNext(); // Compute before changing latlng\n this.latlngs.splice(this.getIndex(), 1);\n this.editor.editLayer.removeLayer(this);\n this.editor.onVertexDeleted({latlng: this.latlng, vertex: this});\n if (!this.latlngs.length) this.editor.deleteShape(this.latlngs);\n if (next) next.resetMiddleMarker();\n this.editor.refresh();\n },\n\n // 🍂method getIndex(): int\n // Get the index of the current vertex among others of the same LatLngs group.\n getIndex: function () {\n return this.latlngs.indexOf(this.latlng);\n },\n\n // 🍂method getLastIndex(): int\n // Get last vertex index of the LatLngs group of the current vertex.\n getLastIndex: function () {\n return this.latlngs.length - 1;\n },\n\n // 🍂method getPrevious(): VertexMarker\n // Get the previous VertexMarker in the same LatLngs group.\n getPrevious: function () {\n if (this.latlngs.length < 2) return;\n var index = this.getIndex(),\n previousIndex = index - 1;\n if (index === 0 && this.editor.CLOSED) previousIndex = this.getLastIndex();\n var previous = this.latlngs[previousIndex];\n if (previous) return previous.__vertex;\n },\n\n // 🍂method getNext(): VertexMarker\n // Get the next VertexMarker in the same LatLngs group.\n getNext: function () {\n if (this.latlngs.length < 2) return;\n var index = this.getIndex(),\n nextIndex = index + 1;\n if (index === this.getLastIndex() && this.editor.CLOSED) nextIndex = 0;\n var next = this.latlngs[nextIndex];\n if (next) return next.__vertex;\n },\n\n addMiddleMarker: function (previous) {\n if (!this.editor.hasMiddleMarkers()) return;\n previous = previous || this.getPrevious();\n if (previous && !this.middleMarker) this.middleMarker = this.editor.addMiddleMarker(previous, this, this.latlngs, this.editor);\n },\n\n addMiddleMarkers: function () {\n if (!this.editor.hasMiddleMarkers()) return;\n var previous = this.getPrevious();\n if (previous) this.addMiddleMarker(previous);\n var next = this.getNext();\n if (next) next.resetMiddleMarker();\n },\n\n resetMiddleMarker: function () {\n if (this.middleMarker) this.middleMarker.delete();\n this.addMiddleMarker();\n },\n\n // 🍂method split()\n // Split the vertex LatLngs group at its index, if possible.\n split: function () {\n if (!this.editor.splitShape) return; // Only for PolylineEditor\n this.editor.splitShape(this.latlngs, this.getIndex());\n },\n\n // 🍂method continue()\n // Continue the vertex LatLngs from this vertex. Only active for first and last vertices of a Polyline.\n continue: function () {\n if (!this.editor.continueBackward) return; // Only for PolylineEditor\n var index = this.getIndex();\n if (index === 0) this.editor.continueBackward(this.latlngs);\n else if (index === this.getLastIndex()) this.editor.continueForward(this.latlngs);\n }\n\n });\n\n L.Editable.mergeOptions({\n\n // 🍂namespace Editable\n // 🍂option vertexMarkerClass: class = VertexMarker\n // Class to be used as vertex, for path editing.\n vertexMarkerClass: L.Editable.VertexMarker\n\n });\n\n L.Editable.MiddleMarker = L.Marker.extend({\n\n options: {\n opacity: 0.5,\n className: 'leaflet-div-icon leaflet-middle-icon',\n draggable: true\n },\n\n initialize: function (left, right, latlngs, editor, options) {\n this.left = left;\n this.right = right;\n this.editor = editor;\n this.latlngs = latlngs;\n L.Marker.prototype.initialize.call(this, this.computeLatLng(), options);\n this._opacity = this.options.opacity;\n this.options.icon = this.editor.tools.createVertexIcon({className: this.options.className});\n this.editor.editLayer.addLayer(this);\n this.setVisibility();\n },\n\n setVisibility: function () {\n var leftPoint = this._map.latLngToContainerPoint(this.left.latlng),\n rightPoint = this._map.latLngToContainerPoint(this.right.latlng),\n size = L.point(this.options.icon.options.iconSize);\n if (leftPoint.distanceTo(rightPoint) < size.x * 3) this.hide();\n else this.show();\n },\n\n show: function () {\n this.setOpacity(this._opacity);\n },\n\n hide: function () {\n this.setOpacity(0);\n },\n\n updateLatLng: function () {\n this.setLatLng(this.computeLatLng());\n this.setVisibility();\n },\n\n computeLatLng: function () {\n var leftPoint = this.editor.map.latLngToContainerPoint(this.left.latlng),\n rightPoint = this.editor.map.latLngToContainerPoint(this.right.latlng),\n y = (leftPoint.y + rightPoint.y) / 2,\n x = (leftPoint.x + rightPoint.x) / 2;\n return this.editor.map.containerPointToLatLng([x, y]);\n },\n\n onAdd: function (map) {\n L.Marker.prototype.onAdd.call(this, map);\n L.DomEvent.on(this._icon, 'mousedown touchstart', this.onMouseDown, this);\n map.on('zoomend', this.setVisibility, this);\n },\n\n onRemove: function (map) {\n delete this.right.middleMarker;\n L.DomEvent.off(this._icon, 'mousedown touchstart', this.onMouseDown, this);\n map.off('zoomend', this.setVisibility, this);\n L.Marker.prototype.onRemove.call(this, map);\n },\n\n onMouseDown: function (e) {\n var iconPos = L.DomUtil.getPosition(this._icon),\n latlng = this.editor.map.layerPointToLatLng(iconPos);\n e = {\n originalEvent: e,\n latlng: latlng\n };\n if (this.options.opacity === 0) return;\n L.Editable.makeCancellable(e);\n this.editor.onMiddleMarkerMouseDown(e);\n if (e._cancelled) return;\n this.latlngs.splice(this.index(), 0, e.latlng);\n this.editor.refresh();\n var icon = this._icon;\n var marker = this.editor.addVertexMarker(e.latlng, this.latlngs);\n this.editor.onNewVertex(marker);\n /* Hack to workaround browser not firing touchend when element is no more on DOM */\n var parent = marker._icon.parentNode;\n parent.removeChild(marker._icon);\n marker._icon = icon;\n parent.appendChild(marker._icon);\n marker._initIcon();\n marker._initInteraction();\n marker.setOpacity(1);\n /* End hack */\n // Transfer ongoing dragging to real marker\n L.Draggable._dragging = false;\n marker.dragging._draggable._onDown(e.originalEvent);\n this.delete();\n },\n\n delete: function () {\n this.editor.editLayer.removeLayer(this);\n },\n\n index: function () {\n return this.latlngs.indexOf(this.right.latlng);\n }\n\n });\n\n L.Editable.mergeOptions({\n\n // 🍂namespace Editable\n // 🍂option middleMarkerClass: class = VertexMarker\n // Class to be used as middle vertex, pulled by the user to create a new point in the middle of a path.\n middleMarkerClass: L.Editable.MiddleMarker\n\n });\n\n // 🍂namespace Editable; 🍂class BaseEditor; 🍂aka L.Editable.BaseEditor\n // When editing a feature (Marker, Polyline…), an editor is attached to it. This\n // editor basically knows how to handle the edition.\n L.Editable.BaseEditor = L.Handler.extend({\n\n initialize: function (map, feature, options) {\n L.setOptions(this, options);\n this.map = map;\n this.feature = feature;\n this.feature.editor = this;\n this.editLayer = new L.LayerGroup();\n this.tools = this.options.editTools || map.editTools;\n },\n\n // 🍂method enable(): this\n // Set up the drawing tools for the feature to be editable.\n addHooks: function () {\n if (this.isConnected()) this.onFeatureAdd();\n else this.feature.once('add', this.onFeatureAdd, this);\n this.onEnable();\n this.feature.on(this._getEvents(), this);\n return;\n },\n\n // 🍂method disable(): this\n // Remove the drawing tools for the feature.\n removeHooks: function () {\n this.feature.off(this._getEvents(), this);\n if (this.feature.dragging) this.feature.dragging.disable();\n this.editLayer.clearLayers();\n this.tools.editLayer.removeLayer(this.editLayer);\n this.onDisable();\n if (this._drawing) this.cancelDrawing();\n return;\n },\n\n // 🍂method drawing(): boolean\n // Return true if any drawing action is ongoing with this editor.\n drawing: function () {\n return !!this._drawing;\n },\n\n reset: function () {},\n\n onFeatureAdd: function () {\n this.tools.editLayer.addLayer(this.editLayer);\n if (this.feature.dragging) this.feature.dragging.enable();\n },\n\n hasMiddleMarkers: function () {\n return !this.options.skipMiddleMarkers && !this.tools.options.skipMiddleMarkers;\n },\n\n fireAndForward: function (type, e) {\n e = e || {};\n e.layer = this.feature;\n this.feature.fire(type, e);\n this.tools.fireAndForward(type, e);\n },\n\n onEnable: function () {\n // 🍂namespace Editable\n // 🍂event editable:enable: Event\n // Fired when an existing feature is ready to be edited.\n this.fireAndForward('editable:enable');\n },\n\n onDisable: function () {\n // 🍂namespace Editable\n // 🍂event editable:disable: Event\n // Fired when an existing feature is not ready anymore to be edited.\n this.fireAndForward('editable:disable');\n },\n\n onEditing: function () {\n // 🍂namespace Editable\n // 🍂event editable:editing: Event\n // Fired as soon as any change is made to the feature geometry.\n this.fireAndForward('editable:editing');\n },\n\n onStartDrawing: function () {\n // 🍂namespace Editable\n // 🍂section Drawing events\n // 🍂event editable:drawing:start: Event\n // Fired when a feature is to be drawn.\n this.fireAndForward('editable:drawing:start');\n },\n\n onEndDrawing: function () {\n // 🍂namespace Editable\n // 🍂section Drawing events\n // 🍂event editable:drawing:end: Event\n // Fired when a feature is not drawn anymore.\n this.fireAndForward('editable:drawing:end');\n },\n\n onCancelDrawing: function () {\n // 🍂namespace Editable\n // 🍂section Drawing events\n // 🍂event editable:drawing:cancel: Event\n // Fired when user cancel drawing while a feature is being drawn.\n this.fireAndForward('editable:drawing:cancel');\n },\n\n onCommitDrawing: function (e) {\n // 🍂namespace Editable\n // 🍂section Drawing events\n // 🍂event editable:drawing:commit: Event\n // Fired when user finish drawing a feature.\n this.fireAndForward('editable:drawing:commit', e);\n },\n\n onDrawingMouseDown: function (e) {\n // 🍂namespace Editable\n // 🍂section Drawing events\n // 🍂event editable:drawing:mousedown: Event\n // Fired when user `mousedown` while drawing.\n this.fireAndForward('editable:drawing:mousedown', e);\n },\n\n onDrawingMouseUp: function (e) {\n // 🍂namespace Editable\n // 🍂section Drawing events\n // 🍂event editable:drawing:mouseup: Event\n // Fired when user `mouseup` while drawing.\n this.fireAndForward('editable:drawing:mouseup', e);\n },\n\n startDrawing: function () {\n if (!this._drawing) this._drawing = L.Editable.FORWARD;\n this.tools.registerForDrawing(this);\n this.onStartDrawing();\n },\n\n commitDrawing: function (e) {\n this.onCommitDrawing(e);\n this.endDrawing();\n },\n\n cancelDrawing: function () {\n // If called during a vertex drag, the vertex will be removed before\n // the mouseup fires on it. This is a workaround. Maybe better fix is\n // To have L.Draggable reset it's status on disable (Leaflet side).\n L.Draggable._dragging = false;\n this.onCancelDrawing();\n this.endDrawing();\n },\n\n endDrawing: function () {\n this._drawing = false;\n this.tools.unregisterForDrawing(this);\n this.onEndDrawing();\n },\n\n onDrawingClick: function (e) {\n if (!this.drawing()) return;\n L.Editable.makeCancellable(e);\n // 🍂namespace Editable\n // 🍂section Drawing events\n // 🍂event editable:drawing:click: CancelableEvent\n // Fired when user `click` while drawing, before any internal action is being processed.\n this.fireAndForward('editable:drawing:click', e);\n if (e._cancelled) return;\n if (!this.isConnected()) this.connect(e);\n this.processDrawingClick(e);\n },\n\n isConnected: function () {\n return this.map.hasLayer(this.feature);\n },\n\n connect: function (e) {\n this.tools.connectCreatedToMap(this.feature);\n this.tools.editLayer.addLayer(this.editLayer);\n },\n\n onMove: function (e) {\n // 🍂namespace Editable\n // 🍂section Drawing events\n // 🍂event editable:drawing:move: Event\n // Fired when `move` mouse while drawing, while dragging a marker, and while dragging a vertex.\n this.fireAndForward('editable:drawing:move', e);\n },\n\n onDrawingMouseMove: function (e) {\n this.onMove(e);\n },\n\n _getEvents: function () {\n return {\n dragstart: this.onDragStart,\n drag: this.onDrag,\n dragend: this.onDragEnd,\n remove: this.disable\n };\n },\n\n onDragStart: function (e) {\n this.onEditing();\n // 🍂namespace Editable\n // 🍂event editable:dragstart: Event\n // Fired before a path feature is dragged.\n this.fireAndForward('editable:dragstart', e);\n },\n\n onDrag: function (e) {\n this.onMove(e);\n // 🍂namespace Editable\n // 🍂event editable:drag: Event\n // Fired when a path feature is being dragged.\n this.fireAndForward('editable:drag', e);\n },\n\n onDragEnd: function (e) {\n // 🍂namespace Editable\n // 🍂event editable:dragend: Event\n // Fired after a path feature has been dragged.\n this.fireAndForward('editable:dragend', e);\n }\n\n });\n\n // 🍂namespace Editable; 🍂class MarkerEditor; 🍂aka L.Editable.MarkerEditor\n // 🍂inherits BaseEditor\n // Editor for Marker.\n L.Editable.MarkerEditor = L.Editable.BaseEditor.extend({\n\n onDrawingMouseMove: function (e) {\n L.Editable.BaseEditor.prototype.onDrawingMouseMove.call(this, e);\n if (this._drawing) this.feature.setLatLng(e.latlng);\n },\n\n processDrawingClick: function (e) {\n // 🍂namespace Editable\n // 🍂section Drawing events\n // 🍂event editable:drawing:clicked: Event\n // Fired when user `click` while drawing, after all internal actions.\n this.fireAndForward('editable:drawing:clicked', e);\n this.commitDrawing(e);\n },\n\n connect: function (e) {\n // On touch, the latlng has not been updated because there is\n // no mousemove.\n if (e) this.feature._latlng = e.latlng;\n L.Editable.BaseEditor.prototype.connect.call(this, e);\n }\n\n });\n\n // 🍂namespace Editable; 🍂class PathEditor; 🍂aka L.Editable.PathEditor\n // 🍂inherits BaseEditor\n // Base class for all path editors.\n L.Editable.PathEditor = L.Editable.BaseEditor.extend({\n\n CLOSED: false,\n MIN_VERTEX: 2,\n\n addHooks: function () {\n L.Editable.BaseEditor.prototype.addHooks.call(this);\n if (this.feature) this.initVertexMarkers();\n return this;\n },\n\n initVertexMarkers: function (latlngs) {\n if (!this.enabled()) return;\n latlngs = latlngs || this.getLatLngs();\n if (isFlat(latlngs)) this.addVertexMarkers(latlngs);\n else for (var i = 0; i < latlngs.length; i++) this.initVertexMarkers(latlngs[i]);\n },\n\n getLatLngs: function () {\n return this.feature.getLatLngs();\n },\n\n // 🍂method reset()\n // Rebuild edit elements (Vertex, MiddleMarker, etc.).\n reset: function () {\n this.editLayer.clearLayers();\n this.initVertexMarkers();\n },\n\n addVertexMarker: function (latlng, latlngs) {\n return new this.tools.options.vertexMarkerClass(latlng, latlngs, this);\n },\n\n onNewVertex: function (vertex) {\n // 🍂namespace Editable\n // 🍂section Vertex events\n // 🍂event editable:vertex:new: VertexEvent\n // Fired when a new vertex is created.\n this.fireAndForward('editable:vertex:new', {latlng: vertex.latlng, vertex: vertex});\n },\n\n addVertexMarkers: function (latlngs) {\n for (var i = 0; i < latlngs.length; i++) {\n this.addVertexMarker(latlngs[i], latlngs);\n }\n },\n\n refreshVertexMarkers: function (latlngs) {\n latlngs = latlngs || this.getDefaultLatLngs();\n for (var i = 0; i < latlngs.length; i++) {\n latlngs[i].__vertex.update();\n }\n },\n\n addMiddleMarker: function (left, right, latlngs) {\n return new this.tools.options.middleMarkerClass(left, right, latlngs, this);\n },\n\n onVertexMarkerClick: function (e) {\n L.Editable.makeCancellable(e);\n // 🍂namespace Editable\n // 🍂section Vertex events\n // 🍂event editable:vertex:click: CancelableVertexEvent\n // Fired when a `click` is issued on a vertex, before any internal action is being processed.\n this.fireAndForward('editable:vertex:click', e);\n if (e._cancelled) return;\n if (this.tools.drawing() && this.tools._drawingEditor !== this) return;\n var index = e.vertex.getIndex(), commit;\n if (e.originalEvent.ctrlKey) {\n this.onVertexMarkerCtrlClick(e);\n } else if (e.originalEvent.altKey) {\n this.onVertexMarkerAltClick(e);\n } else if (e.originalEvent.shiftKey) {\n this.onVertexMarkerShiftClick(e);\n } else if (e.originalEvent.metaKey) {\n this.onVertexMarkerMetaKeyClick(e);\n } else if (index === e.vertex.getLastIndex() && this._drawing === L.Editable.FORWARD) {\n if (index >= this.MIN_VERTEX - 1) commit = true;\n } else if (index === 0 && this._drawing === L.Editable.BACKWARD && this._drawnLatLngs.length >= this.MIN_VERTEX) {\n commit = true;\n } else if (index === 0 && this._drawing === L.Editable.FORWARD && this._drawnLatLngs.length >= this.MIN_VERTEX && this.CLOSED) {\n commit = true; // Allow to close on first point also for polygons\n } else {\n this.onVertexRawMarkerClick(e);\n }\n // 🍂namespace Editable\n // 🍂section Vertex events\n // 🍂event editable:vertex:clicked: VertexEvent\n // Fired when a `click` is issued on a vertex, after all internal actions.\n this.fireAndForward('editable:vertex:clicked', e);\n if (commit) this.commitDrawing(e);\n },\n\n onVertexRawMarkerClick: function (e) {\n // 🍂namespace Editable\n // 🍂section Vertex events\n // 🍂event editable:vertex:rawclick: CancelableVertexEvent\n // Fired when a `click` is issued on a vertex without any special key and without being in drawing mode.\n this.fireAndForward('editable:vertex:rawclick', e);\n if (e._cancelled) return;\n if (!this.vertexCanBeDeleted(e.vertex)) return;\n e.vertex.delete();\n },\n\n vertexCanBeDeleted: function (vertex) {\n return vertex.latlngs.length > this.MIN_VERTEX;\n },\n\n onVertexDeleted: function (e) {\n // 🍂namespace Editable\n // 🍂section Vertex events\n // 🍂event editable:vertex:deleted: VertexEvent\n // Fired after a vertex has been deleted by user.\n this.fireAndForward('editable:vertex:deleted', e);\n },\n\n onVertexMarkerCtrlClick: function (e) {\n // 🍂namespace Editable\n // 🍂section Vertex events\n // 🍂event editable:vertex:ctrlclick: VertexEvent\n // Fired when a `click` with `ctrlKey` is issued on a vertex.\n this.fireAndForward('editable:vertex:ctrlclick', e);\n },\n\n onVertexMarkerShiftClick: function (e) {\n // 🍂namespace Editable\n // 🍂section Vertex events\n // 🍂event editable:vertex:shiftclick: VertexEvent\n // Fired when a `click` with `shiftKey` is issued on a vertex.\n this.fireAndForward('editable:vertex:shiftclick', e);\n },\n\n onVertexMarkerMetaKeyClick: function (e) {\n // 🍂namespace Editable\n // 🍂section Vertex events\n // 🍂event editable:vertex:metakeyclick: VertexEvent\n // Fired when a `click` with `metaKey` is issued on a vertex.\n this.fireAndForward('editable:vertex:metakeyclick', e);\n },\n\n onVertexMarkerAltClick: function (e) {\n // 🍂namespace Editable\n // 🍂section Vertex events\n // 🍂event editable:vertex:altclick: VertexEvent\n // Fired when a `click` with `altKey` is issued on a vertex.\n this.fireAndForward('editable:vertex:altclick', e);\n },\n\n onVertexMarkerContextMenu: function (e) {\n // 🍂namespace Editable\n // 🍂section Vertex events\n // 🍂event editable:vertex:contextmenu: VertexEvent\n // Fired when a `contextmenu` is issued on a vertex.\n this.fireAndForward('editable:vertex:contextmenu', e);\n },\n\n onVertexMarkerMouseDown: function (e) {\n // 🍂namespace Editable\n // 🍂section Vertex events\n // 🍂event editable:vertex:mousedown: VertexEvent\n // Fired when user `mousedown` a vertex.\n this.fireAndForward('editable:vertex:mousedown', e);\n },\n\n onMiddleMarkerMouseDown: function (e) {\n // 🍂namespace Editable\n // 🍂section MiddleMarker events\n // 🍂event editable:middlemarker:mousedown: VertexEvent\n // Fired when user `mousedown` a middle marker.\n this.fireAndForward('editable:middlemarker:mousedown', e);\n },\n\n onVertexMarkerDrag: function (e) {\n this.onMove(e);\n if (this.feature._bounds) this.extendBounds(e);\n // 🍂namespace Editable\n // 🍂section Vertex events\n // 🍂event editable:vertex:drag: VertexEvent\n // Fired when a vertex is dragged by user.\n this.fireAndForward('editable:vertex:drag', e);\n },\n\n onVertexMarkerDragStart: function (e) {\n // 🍂namespace Editable\n // 🍂section Vertex events\n // 🍂event editable:vertex:dragstart: VertexEvent\n // Fired before a vertex is dragged by user.\n this.fireAndForward('editable:vertex:dragstart', e);\n },\n\n onVertexMarkerDragEnd: function (e) {\n // 🍂namespace Editable\n // 🍂section Vertex events\n // 🍂event editable:vertex:dragend: VertexEvent\n // Fired after a vertex is dragged by user.\n this.fireAndForward('editable:vertex:dragend', e);\n },\n\n setDrawnLatLngs: function (latlngs) {\n this._drawnLatLngs = latlngs || this.getDefaultLatLngs();\n },\n\n startDrawing: function () {\n if (!this._drawnLatLngs) this.setDrawnLatLngs();\n L.Editable.BaseEditor.prototype.startDrawing.call(this);\n },\n\n startDrawingForward: function () {\n this.startDrawing();\n },\n\n endDrawing: function () {\n this.tools.detachForwardLineGuide();\n this.tools.detachBackwardLineGuide();\n if (this._drawnLatLngs && this._drawnLatLngs.length < this.MIN_VERTEX) this.deleteShape(this._drawnLatLngs);\n L.Editable.BaseEditor.prototype.endDrawing.call(this);\n delete this._drawnLatLngs;\n },\n\n addLatLng: function (latlng) {\n if (this._drawing === L.Editable.FORWARD) this._drawnLatLngs.push(latlng);\n else this._drawnLatLngs.unshift(latlng);\n this.feature._bounds.extend(latlng);\n var vertex = this.addVertexMarker(latlng, this._drawnLatLngs);\n this.onNewVertex(vertex);\n this.refresh();\n },\n\n newPointForward: function (latlng) {\n this.addLatLng(latlng);\n this.tools.attachForwardLineGuide();\n this.tools.anchorForwardLineGuide(latlng);\n },\n\n newPointBackward: function (latlng) {\n this.addLatLng(latlng);\n this.tools.anchorBackwardLineGuide(latlng);\n },\n\n // 🍂namespace PathEditor\n // 🍂method push()\n // Programmatically add a point while drawing.\n push: function (latlng) {\n if (!latlng) return console.error('L.Editable.PathEditor.push expect a vaild latlng as parameter');\n if (this._drawing === L.Editable.FORWARD) this.newPointForward(latlng);\n else this.newPointBackward(latlng);\n },\n\n removeLatLng: function (latlng) {\n latlng.__vertex.delete();\n this.refresh();\n },\n\n // 🍂method pop(): L.LatLng or null\n // Programmatically remove last point (if any) while drawing.\n pop: function () {\n if (this._drawnLatLngs.length <= 1) return;\n var latlng;\n if (this._drawing === L.Editable.FORWARD) latlng = this._drawnLatLngs[this._drawnLatLngs.length - 1];\n else latlng = this._drawnLatLngs[0];\n this.removeLatLng(latlng);\n if (this._drawing === L.Editable.FORWARD) this.tools.anchorForwardLineGuide(this._drawnLatLngs[this._drawnLatLngs.length - 1]);\n else this.tools.anchorForwardLineGuide(this._drawnLatLngs[0]);\n return latlng;\n },\n\n processDrawingClick: function (e) {\n if (e.vertex && e.vertex.editor === this) return;\n if (this._drawing === L.Editable.FORWARD) this.newPointForward(e.latlng);\n else this.newPointBackward(e.latlng);\n this.fireAndForward('editable:drawing:clicked', e);\n },\n\n onDrawingMouseMove: function (e) {\n L.Editable.BaseEditor.prototype.onDrawingMouseMove.call(this, e);\n if (this._drawing) {\n this.tools.moveForwardLineGuide(e.latlng);\n this.tools.moveBackwardLineGuide(e.latlng);\n }\n },\n\n refresh: function () {\n this.feature.redraw();\n this.onEditing();\n },\n\n // 🍂namespace PathEditor\n // 🍂method newShape(latlng?: L.LatLng)\n // Add a new shape (Polyline, Polygon) in a multi, and setup up drawing tools to draw it;\n // if optional `latlng` is given, start a path at this point.\n newShape: function (latlng) {\n var shape = this.addNewEmptyShape();\n if (!shape) return;\n this.setDrawnLatLngs(shape[0] || shape); // Polygon or polyline\n this.startDrawingForward();\n // 🍂namespace Editable\n // 🍂section Shape events\n // 🍂event editable:shape:new: ShapeEvent\n // Fired when a new shape is created in a multi (Polygon or Polyline).\n this.fireAndForward('editable:shape:new', {shape: shape});\n if (latlng) this.newPointForward(latlng);\n },\n\n deleteShape: function (shape, latlngs) {\n var e = {shape: shape};\n L.Editable.makeCancellable(e);\n // 🍂namespace Editable\n // 🍂section Shape events\n // 🍂event editable:shape:delete: CancelableShapeEvent\n // Fired before a new shape is deleted in a multi (Polygon or Polyline).\n this.fireAndForward('editable:shape:delete', e);\n if (e._cancelled) return;\n shape = this._deleteShape(shape, latlngs);\n if (this.ensureNotFlat) this.ensureNotFlat(); // Polygon.\n this.feature.setLatLngs(this.getLatLngs()); // Force bounds reset.\n this.refresh();\n this.reset();\n // 🍂namespace Editable\n // 🍂section Shape events\n // 🍂event editable:shape:deleted: ShapeEvent\n // Fired after a new shape is deleted in a multi (Polygon or Polyline).\n this.fireAndForward('editable:shape:deleted', {shape: shape});\n return shape;\n },\n\n _deleteShape: function (shape, latlngs) {\n latlngs = latlngs || this.getLatLngs();\n if (!latlngs.length) return;\n var self = this,\n inplaceDelete = function (latlngs, shape) {\n // Called when deleting a flat latlngs\n shape = latlngs.splice(0, Number.MAX_VALUE);\n return shape;\n },\n spliceDelete = function (latlngs, shape) {\n // Called when removing a latlngs inside an array\n latlngs.splice(latlngs.indexOf(shape), 1);\n if (!latlngs.length) self._deleteShape(latlngs);\n return shape;\n };\n if (latlngs === shape) return inplaceDelete(latlngs, shape);\n for (var i = 0; i < latlngs.length; i++) {\n if (latlngs[i] === shape) return spliceDelete(latlngs, shape);\n else if (latlngs[i].indexOf(shape) !== -1) return spliceDelete(latlngs[i], shape);\n }\n },\n\n // 🍂namespace PathEditor\n // 🍂method deleteShapeAt(latlng: L.LatLng): Array\n // Remove a path shape at the given `latlng`.\n deleteShapeAt: function (latlng) {\n var shape = this.feature.shapeAt(latlng);\n if (shape) return this.deleteShape(shape);\n },\n\n // 🍂method appendShape(shape: Array)\n // Append a new shape to the Polygon or Polyline.\n appendShape: function (shape) {\n this.insertShape(shape);\n },\n\n // 🍂method prependShape(shape: Array)\n // Prepend a new shape to the Polygon or Polyline.\n prependShape: function (shape) {\n this.insertShape(shape, 0);\n },\n\n // 🍂method insertShape(shape: Array, index: int)\n // Insert a new shape to the Polygon or Polyline at given index (default is to append).\n insertShape: function (shape, index) {\n this.ensureMulti();\n shape = this.formatShape(shape);\n if (typeof index === 'undefined') index = this.feature._latlngs.length;\n this.feature._latlngs.splice(index, 0, shape);\n this.feature.redraw();\n if (this._enabled) this.reset();\n },\n\n extendBounds: function (e) {\n this.feature._bounds.extend(e.vertex.latlng);\n },\n\n onDragStart: function (e) {\n this.editLayer.clearLayers();\n L.Editable.BaseEditor.prototype.onDragStart.call(this, e);\n },\n\n onDragEnd: function (e) {\n this.initVertexMarkers();\n L.Editable.BaseEditor.prototype.onDragEnd.call(this, e);\n }\n\n });\n\n // 🍂namespace Editable; 🍂class PolylineEditor; 🍂aka L.Editable.PolylineEditor\n // 🍂inherits PathEditor\n L.Editable.PolylineEditor = L.Editable.PathEditor.extend({\n\n startDrawingBackward: function () {\n this._drawing = L.Editable.BACKWARD;\n this.startDrawing();\n },\n\n // 🍂method continueBackward(latlngs?: Array)\n // Set up drawing tools to continue the line backward.\n continueBackward: function (latlngs) {\n if (this.drawing()) return;\n latlngs = latlngs || this.getDefaultLatLngs();\n this.setDrawnLatLngs(latlngs);\n if (latlngs.length > 0) {\n this.tools.attachBackwardLineGuide();\n this.tools.anchorBackwardLineGuide(latlngs[0]);\n }\n this.startDrawingBackward();\n },\n\n // 🍂method continueForward(latlngs?: Array)\n // Set up drawing tools to continue the line forward.\n continueForward: function (latlngs) {\n if (this.drawing()) return;\n latlngs = latlngs || this.getDefaultLatLngs();\n this.setDrawnLatLngs(latlngs);\n if (latlngs.length > 0) {\n this.tools.attachForwardLineGuide();\n this.tools.anchorForwardLineGuide(latlngs[latlngs.length - 1]);\n }\n this.startDrawingForward();\n },\n\n getDefaultLatLngs: function (latlngs) {\n latlngs = latlngs || this.feature._latlngs;\n if (!latlngs.length || latlngs[0] instanceof L.LatLng) return latlngs;\n else return this.getDefaultLatLngs(latlngs[0]);\n },\n\n ensureMulti: function () {\n if (this.feature._latlngs.length && isFlat(this.feature._latlngs)) {\n this.feature._latlngs = [this.feature._latlngs];\n }\n },\n\n addNewEmptyShape: function () {\n if (this.feature._latlngs.length) {\n var shape = [];\n this.appendShape(shape);\n return shape;\n } else {\n return this.feature._latlngs;\n }\n },\n\n formatShape: function (shape) {\n if (isFlat(shape)) return shape;\n else if (shape[0]) return this.formatShape(shape[0]);\n },\n\n // 🍂method splitShape(latlngs?: Array, index: int)\n // Split the given `latlngs` shape at index `index` and integrate new shape in instance `latlngs`.\n splitShape: function (shape, index) {\n if (!index || index >= shape.length - 1) return;\n this.ensureMulti();\n var shapeIndex = this.feature._latlngs.indexOf(shape);\n if (shapeIndex === -1) return;\n var first = shape.slice(0, index + 1),\n second = shape.slice(index);\n // We deal with reference, we don't want twice the same latlng around.\n second[0] = L.latLng(second[0].lat, second[0].lng, second[0].alt);\n this.feature._latlngs.splice(shapeIndex, 1, first, second);\n this.refresh();\n this.reset();\n }\n\n });\n\n // 🍂namespace Editable; 🍂class PolygonEditor; 🍂aka L.Editable.PolygonEditor\n // 🍂inherits PathEditor\n L.Editable.PolygonEditor = L.Editable.PathEditor.extend({\n\n CLOSED: true,\n MIN_VERTEX: 3,\n\n newPointForward: function (latlng) {\n L.Editable.PathEditor.prototype.newPointForward.call(this, latlng);\n if (!this.tools.backwardLineGuide._latlngs.length) this.tools.anchorBackwardLineGuide(latlng);\n if (this._drawnLatLngs.length === 2) this.tools.attachBackwardLineGuide();\n },\n\n addNewEmptyHole: function (latlng) {\n this.ensureNotFlat();\n var latlngs = this.feature.shapeAt(latlng);\n if (!latlngs) return;\n var holes = [];\n latlngs.push(holes);\n return holes;\n },\n\n // 🍂method newHole(latlng?: L.LatLng, index: int)\n // Set up drawing tools for creating a new hole on the Polygon. If the `latlng` param is given, a first point is created.\n newHole: function (latlng) {\n var holes = this.addNewEmptyHole(latlng);\n if (!holes) return;\n this.setDrawnLatLngs(holes);\n this.startDrawingForward();\n if (latlng) this.newPointForward(latlng);\n },\n\n addNewEmptyShape: function () {\n if (this.feature._latlngs.length && this.feature._latlngs[0].length) {\n var shape = [];\n this.appendShape(shape);\n return shape;\n } else {\n return this.feature._latlngs;\n }\n },\n\n ensureMulti: function () {\n if (this.feature._latlngs.length && isFlat(this.feature._latlngs[0])) {\n this.feature._latlngs = [this.feature._latlngs];\n }\n },\n\n ensureNotFlat: function () {\n if (!this.feature._latlngs.length || isFlat(this.feature._latlngs)) this.feature._latlngs = [this.feature._latlngs];\n },\n\n vertexCanBeDeleted: function (vertex) {\n var parent = this.feature.parentShape(vertex.latlngs),\n idx = L.Util.indexOf(parent, vertex.latlngs);\n if (idx > 0) return true; // Holes can be totally deleted without removing the layer itself.\n return L.Editable.PathEditor.prototype.vertexCanBeDeleted.call(this, vertex);\n },\n\n getDefaultLatLngs: function () {\n if (!this.feature._latlngs.length) this.feature._latlngs.push([]);\n return this.feature._latlngs[0];\n },\n\n formatShape: function (shape) {\n // [[1, 2], [3, 4]] => must be nested\n // [] => must be nested\n // [[]] => is already nested\n if (isFlat(shape) && (!shape[0] || shape[0].length !== 0)) return [shape];\n else return shape;\n }\n\n });\n\n // 🍂namespace Editable; 🍂class RectangleEditor; 🍂aka L.Editable.RectangleEditor\n // 🍂inherits PathEditor\n L.Editable.RectangleEditor = L.Editable.PathEditor.extend({\n\n CLOSED: true,\n MIN_VERTEX: 4,\n\n options: {\n skipMiddleMarkers: true\n },\n\n extendBounds: function (e) {\n var index = e.vertex.getIndex(),\n next = e.vertex.getNext(),\n previous = e.vertex.getPrevious(),\n oppositeIndex = (index + 2) % 4,\n opposite = e.vertex.latlngs[oppositeIndex],\n bounds = new L.LatLngBounds(e.latlng, opposite);\n // Update latlngs by hand to preserve order.\n previous.latlng.update([e.latlng.lat, opposite.lng]);\n next.latlng.update([opposite.lat, e.latlng.lng]);\n this.updateBounds(bounds);\n this.refreshVertexMarkers();\n },\n\n onDrawingMouseDown: function (e) {\n L.Editable.PathEditor.prototype.onDrawingMouseDown.call(this, e);\n this.connect();\n var latlngs = this.getDefaultLatLngs();\n // L.Polygon._convertLatLngs removes last latlng if it equals first point,\n // which is the case here as all latlngs are [0, 0]\n if (latlngs.length === 3) latlngs.push(e.latlng);\n var bounds = new L.LatLngBounds(e.latlng, e.latlng);\n this.updateBounds(bounds);\n this.updateLatLngs(bounds);\n this.refresh();\n this.reset();\n // Stop dragging map.\n // L.Draggable has two workflows:\n // - mousedown => mousemove => mouseup\n // - touchstart => touchmove => touchend\n // Problem: L.Map.Tap does not allow us to listen to touchstart, so we only\n // can deal with mousedown, but then when in a touch device, we are dealing with\n // simulated events (actually simulated by L.Map.Tap), which are no more taken\n // into account by L.Draggable.\n // Ref.: https://github.com/Leaflet/Leaflet.Editable/issues/103\n e.originalEvent._simulated = false;\n this.map.dragging._draggable._onUp(e.originalEvent);\n // Now transfer ongoing drag action to the bottom right corner.\n // Should we refine which corne will handle the drag according to\n // drag direction?\n latlngs[3].__vertex.dragging._draggable._onDown(e.originalEvent);\n },\n\n onDrawingMouseUp: function (e) {\n this.commitDrawing(e);\n e.originalEvent._simulated = false;\n L.Editable.PathEditor.prototype.onDrawingMouseUp.call(this, e);\n },\n\n onDrawingMouseMove: function (e) {\n e.originalEvent._simulated = false;\n L.Editable.PathEditor.prototype.onDrawingMouseMove.call(this, e);\n },\n\n\n getDefaultLatLngs: function (latlngs) {\n return latlngs || this.feature._latlngs[0];\n },\n\n updateBounds: function (bounds) {\n this.feature._bounds = bounds;\n },\n\n updateLatLngs: function (bounds) {\n var latlngs = this.getDefaultLatLngs(),\n newLatlngs = this.feature._boundsToLatLngs(bounds);\n // Keep references.\n for (var i = 0; i < latlngs.length; i++) {\n latlngs[i].update(newLatlngs[i]);\n };\n }\n\n });\n\n // 🍂namespace Editable; 🍂class CircleEditor; 🍂aka L.Editable.CircleEditor\n // 🍂inherits PathEditor\n L.Editable.CircleEditor = L.Editable.PathEditor.extend({\n\n MIN_VERTEX: 2,\n\n options: {\n skipMiddleMarkers: true\n },\n\n initialize: function (map, feature, options) {\n L.Editable.PathEditor.prototype.initialize.call(this, map, feature, options);\n this._resizeLatLng = this.computeResizeLatLng();\n },\n\n computeResizeLatLng: function () {\n // While circle is not added to the map, _radius is not set.\n var delta = (this.feature._radius || this.feature._mRadius) * Math.cos(Math.PI / 4),\n point = this.map.project(this.feature._latlng);\n return this.map.unproject([point.x + delta, point.y - delta]);\n },\n\n updateResizeLatLng: function () {\n this._resizeLatLng.update(this.computeResizeLatLng());\n this._resizeLatLng.__vertex.update();\n },\n\n getLatLngs: function () {\n return [this.feature._latlng, this._resizeLatLng];\n },\n\n getDefaultLatLngs: function () {\n return this.getLatLngs();\n },\n\n onVertexMarkerDrag: function (e) {\n if (e.vertex.getIndex() === 1) this.resize(e);\n else this.updateResizeLatLng(e);\n L.Editable.PathEditor.prototype.onVertexMarkerDrag.call(this, e);\n },\n\n resize: function (e) {\n var radius = this.feature._latlng.distanceTo(e.latlng)\n this.feature.setRadius(radius);\n },\n\n onDrawingMouseDown: function (e) {\n L.Editable.PathEditor.prototype.onDrawingMouseDown.call(this, e);\n this._resizeLatLng.update(e.latlng);\n this.feature._latlng.update(e.latlng);\n this.connect();\n // Stop dragging map.\n e.originalEvent._simulated = false;\n this.map.dragging._draggable._onUp(e.originalEvent);\n // Now transfer ongoing drag action to the radius handler.\n this._resizeLatLng.__vertex.dragging._draggable._onDown(e.originalEvent);\n },\n\n onDrawingMouseUp: function (e) {\n this.commitDrawing(e);\n e.originalEvent._simulated = false;\n L.Editable.PathEditor.prototype.onDrawingMouseUp.call(this, e);\n },\n\n onDrawingMouseMove: function (e) {\n e.originalEvent._simulated = false;\n L.Editable.PathEditor.prototype.onDrawingMouseMove.call(this, e);\n },\n\n onDrag: function (e) {\n L.Editable.PathEditor.prototype.onDrag.call(this, e);\n this.feature.dragging.updateLatLng(this._resizeLatLng);\n }\n\n });\n\n // 🍂namespace Editable; 🍂class EditableMixin\n // `EditableMixin` is included to `L.Polyline`, `L.Polygon`, `L.Rectangle`, `L.Circle`\n // and `L.Marker`. It adds some methods to them.\n // *When editing is enabled, the editor is accessible on the instance with the\n // `editor` property.*\n var EditableMixin = {\n\n createEditor: function (map) {\n map = map || this._map;\n var tools = (this.options.editOptions || {}).editTools || map.editTools;\n if (!tools) throw Error('Unable to detect Editable instance.')\n var Klass = this.options.editorClass || this.getEditorClass(tools);\n return new Klass(map, this, this.options.editOptions);\n },\n\n // 🍂method enableEdit(map?: L.Map): this.editor\n // Enable editing, by creating an editor if not existing, and then calling `enable` on it.\n enableEdit: function (map) {\n if (!this.editor) this.createEditor(map);\n this.editor.enable();\n return this.editor;\n },\n\n // 🍂method editEnabled(): boolean\n // Return true if current instance has an editor attached, and this editor is enabled.\n editEnabled: function () {\n return this.editor && this.editor.enabled();\n },\n\n // 🍂method disableEdit()\n // Disable editing, also remove the editor property reference.\n disableEdit: function () {\n if (this.editor) {\n this.editor.disable();\n delete this.editor;\n }\n },\n\n // 🍂method toggleEdit()\n // Enable or disable editing, according to current status.\n toggleEdit: function () {\n if (this.editEnabled()) this.disableEdit();\n else this.enableEdit();\n },\n\n _onEditableAdd: function () {\n if (this.editor) this.enableEdit();\n }\n\n };\n\n var PolylineMixin = {\n\n getEditorClass: function (tools) {\n return (tools && tools.options.polylineEditorClass) ? tools.options.polylineEditorClass : L.Editable.PolylineEditor;\n },\n\n shapeAt: function (latlng, latlngs) {\n // We can have those cases:\n // - latlngs are just a flat array of latlngs, use this\n // - latlngs is an array of arrays of latlngs, loop over\n var shape = null;\n latlngs = latlngs || this._latlngs;\n if (!latlngs.length) return shape;\n else if (isFlat(latlngs) && this.isInLatLngs(latlng, latlngs)) shape = latlngs;\n else for (var i = 0; i < latlngs.length; i++) if (this.isInLatLngs(latlng, latlngs[i])) return latlngs[i];\n return shape;\n },\n\n isInLatLngs: function (l, latlngs) {\n if (!latlngs) return false;\n var i, k, len, part = [], p,\n w = this._clickTolerance();\n this._projectLatlngs(latlngs, part, this._pxBounds);\n part = part[0];\n p = this._map.latLngToLayerPoint(l);\n\n if (!this._pxBounds.contains(p)) { return false; }\n for (i = 1, len = part.length, k = 0; i < len; k = i++) {\n\n if (L.LineUtil.pointToSegmentDistance(p, part[k], part[i]) <= w) {\n return true;\n }\n }\n return false;\n }\n\n };\n\n var PolygonMixin = {\n\n getEditorClass: function (tools) {\n return (tools && tools.options.polygonEditorClass) ? tools.options.polygonEditorClass : L.Editable.PolygonEditor;\n },\n\n shapeAt: function (latlng, latlngs) {\n // We can have those cases:\n // - latlngs are just a flat array of latlngs, use this\n // - latlngs is an array of arrays of latlngs, this is a simple polygon (maybe with holes), use the first\n // - latlngs is an array of arrays of arrays, this is a multi, loop over\n var shape = null;\n latlngs = latlngs || this._latlngs;\n if (!latlngs.length) return shape;\n else if (isFlat(latlngs) && this.isInLatLngs(latlng, latlngs)) shape = latlngs;\n else if (isFlat(latlngs[0]) && this.isInLatLngs(latlng, latlngs[0])) shape = latlngs;\n else for (var i = 0; i < latlngs.length; i++) if (this.isInLatLngs(latlng, latlngs[i][0])) return latlngs[i];\n return shape;\n },\n\n isInLatLngs: function (l, latlngs) {\n var inside = false, l1, l2, j, k, len2;\n\n for (j = 0, len2 = latlngs.length, k = len2 - 1; j < len2; k = j++) {\n l1 = latlngs[j];\n l2 = latlngs[k];\n\n if (((l1.lat > l.lat) !== (l2.lat > l.lat)) &&\n (l.lng < (l2.lng - l1.lng) * (l.lat - l1.lat) / (l2.lat - l1.lat) + l1.lng)) {\n inside = !inside;\n }\n }\n\n return inside;\n },\n\n parentShape: function (shape, latlngs) {\n latlngs = latlngs || this._latlngs;\n if (!latlngs) return;\n var idx = L.Util.indexOf(latlngs, shape);\n if (idx !== -1) return latlngs;\n for (var i = 0; i < latlngs.length; i++) {\n idx = L.Util.indexOf(latlngs[i], shape);\n if (idx !== -1) return latlngs[i];\n }\n }\n\n };\n\n\n var MarkerMixin = {\n\n getEditorClass: function (tools) {\n return (tools && tools.options.markerEditorClass) ? tools.options.markerEditorClass : L.Editable.MarkerEditor;\n }\n\n };\n\n var RectangleMixin = {\n\n getEditorClass: function (tools) {\n return (tools && tools.options.rectangleEditorClass) ? tools.options.rectangleEditorClass : L.Editable.RectangleEditor;\n }\n\n };\n\n var CircleMixin = {\n\n getEditorClass: function (tools) {\n return (tools && tools.options.circleEditorClass) ? tools.options.circleEditorClass : L.Editable.CircleEditor;\n }\n\n };\n\n var keepEditable = function () {\n // Make sure you can remove/readd an editable layer.\n this.on('add', this._onEditableAdd);\n };\n\n var isFlat = L.LineUtil.isFlat || L.LineUtil._flat || L.Polyline._flat; // <=> 1.1 compat.\n\n\n if (L.Polyline) {\n L.Polyline.include(EditableMixin);\n L.Polyline.include(PolylineMixin);\n L.Polyline.addInitHook(keepEditable);\n }\n if (L.Polygon) {\n L.Polygon.include(EditableMixin);\n L.Polygon.include(PolygonMixin);\n }\n if (L.Marker) {\n L.Marker.include(EditableMixin);\n L.Marker.include(MarkerMixin);\n L.Marker.addInitHook(keepEditable);\n }\n if (L.Rectangle) {\n L.Rectangle.include(EditableMixin);\n L.Rectangle.include(RectangleMixin);\n }\n if (L.Circle) {\n L.Circle.include(EditableMixin);\n L.Circle.include(CircleMixin);\n }\n\n L.LatLng.prototype.update = function (latlng) {\n latlng = L.latLng(latlng);\n this.lat = latlng.lat;\n this.lng = latlng.lng;\n }\n\n}, window));\n","'use strict';\n\n/* A Draggable that does not update the element position\nand takes care of only bubbling to targetted path in Canvas mode. */\nL.PathDraggable = L.Draggable.extend({\n\n initialize: function (path) {\n this._path = path;\n this._canvas = (path._map.getRenderer(path) instanceof L.Canvas);\n var element = this._canvas ? this._path._map.getRenderer(this._path)._container : this._path._path;\n L.Draggable.prototype.initialize.call(this, element, element, true);\n },\n\n _updatePosition: function () {\n var e = {originalEvent: this._lastEvent};\n this.fire('drag', e);\n },\n\n _onDown: function (e) {\n var first = e.touches ? e.touches[0] : e;\n this._startPoint = new L.Point(first.clientX, first.clientY);\n if (this._canvas && !this._path._containsPoint(this._path._map.mouseEventToLayerPoint(first))) { return; }\n L.Draggable.prototype._onDown.call(this, e);\n }\n\n});\n\n\nL.Handler.PathDrag = L.Handler.extend({\n\n initialize: function (path) {\n this._path = path;\n },\n\n getEvents: function () {\n return {\n dragstart: this._onDragStart,\n drag: this._onDrag,\n dragend: this._onDragEnd\n };\n },\n\n addHooks: function () {\n if (!this._draggable) { this._draggable = new L.PathDraggable(this._path); }\n this._draggable.on(this.getEvents(), this).enable();\n L.DomUtil.addClass(this._draggable._element, 'leaflet-path-draggable');\n },\n\n removeHooks: function () {\n this._draggable.off(this.getEvents(), this).disable();\n L.DomUtil.removeClass(this._draggable._element, 'leaflet-path-draggable');\n },\n\n moved: function () {\n return this._draggable && this._draggable._moved;\n },\n\n _onDragStart: function () {\n this._startPoint = this._draggable._startPoint;\n this._path\n .closePopup()\n .fire('movestart')\n .fire('dragstart');\n },\n\n _onDrag: function (e) {\n var path = this._path,\n event = (e.originalEvent.touches && e.originalEvent.touches.length === 1 ? e.originalEvent.touches[0] : e.originalEvent),\n newPoint = L.point(event.clientX, event.clientY),\n latlng = path._map.layerPointToLatLng(newPoint);\n\n this._offset = newPoint.subtract(this._startPoint);\n this._startPoint = newPoint;\n\n this._path.eachLatLng(this.updateLatLng, this);\n path.redraw();\n\n e.latlng = latlng;\n e.offset = this._offset;\n path.fire('move', e)\n .fire('drag', e);\n },\n\n _onDragEnd: function (e) {\n if (this._path._bounds) this.resetBounds();\n this._path.fire('moveend')\n .fire('dragend', e);\n },\n\n latLngToLayerPoint: function (latlng) {\n // Same as map.latLngToLayerPoint, but without the round().\n var projectedPoint = this._path._map.project(L.latLng(latlng));\n return projectedPoint._subtract(this._path._map.getPixelOrigin());\n },\n\n updateLatLng: function (latlng) {\n var oldPoint = this.latLngToLayerPoint(latlng);\n oldPoint._add(this._offset);\n var newLatLng = this._path._map.layerPointToLatLng(oldPoint);\n latlng.lat = newLatLng.lat;\n latlng.lng = newLatLng.lng;\n },\n\n resetBounds: function () {\n this._path._bounds = new L.LatLngBounds();\n this._path.eachLatLng(function (latlng) {\n this._bounds.extend(latlng);\n });\n }\n\n});\n\nL.Path.include({\n\n eachLatLng: function (callback, context) {\n context = context || this;\n var loop = function (latlngs) {\n for (var i = 0; i < latlngs.length; i++) {\n if (L.Util.isArray(latlngs[i])) loop(latlngs[i]);\n else callback.call(context, latlngs[i]);\n }\n };\n loop(this.getLatLngs ? this.getLatLngs() : [this.getLatLng()]);\n }\n\n});\n\nL.Path.addInitHook(function () {\n\n this.dragging = new L.Handler.PathDrag(this);\n if (this.options.draggable) {\n this.once('add', function () {\n this.dragging.enable();\n });\n }\n\n});\n","export default class {\n connect() {\n $(\"[data-action='add-another']\").on(\"click\", function(event) {\n event.preventDefault();\n\n var templateId = $(this).data('template-id');\n\n var template = document.querySelector('#' + templateId);\n var clone = document.importNode(template.content, true);\n\n var count = $(this).closest('.form-group').find('[name=\"' + $(clone).find('[name]').attr('name') + '\"]').length + 1;\n $(clone).find('[id]').each(function(index, el) {\n $(el).attr('id', $(el).attr('id') + '_' + String(count));\n });\n\n $(clone).find('[for]').each(function(index, el) {\n $(el).attr('for', $(el).attr('for') + '_' + String(count));\n });\n\n\n $(clone).insertBefore(this);\n });\n }\n}\n","export default class {\n connect() {\n $(\"[data-expanded-add-button]\").each((_i, el) => this.addExpandBehaviorToButton($(el)))\n }\n\n addExpandBehaviorToButton(button){\n var settings = {\n speed: (button.data('speed') || 450),\n animate_width: (button.data('animate_width') || 425)\n }\n var target = $(button.data('field-target'));\n var save = $(\"input[data-behavior='save']\", target);\n var cancel = $(\"input[data-behavior='cancel']\", target);\n var input = $(\"input[type='text']\", target);\n var original_width = button.outerWidth();\n var expanded = false;\n\n // Animate button open when the mouse enters or\n // the button is given focus (i.e. clicked/tabbed)\n button.on(\"mouseenter focus\", function(){\n expandButton();\n });\n\n // Don't allow blank titles\n save.on('click', function(){\n if ( inputEmpty() ) {\n return false;\n }\n });\n\n // Empty input and collapse\n // button on cancel click\n cancel.on('click', function(e){\n e.preventDefault();\n input.val('');\n collapseButton();\n });\n\n // Collapse the button on when\n // an empty input loses focus\n input.on(\"blur\", function(){\n if ( inputEmpty() ) {\n collapseButton();\n }\n });\n function expandButton(){\n // If this has not yet been expanded, recalculate original_width to \n // handle things that may have been originally hidden.\n if (!expanded) {\n original_width = button.outerWidth();\n }\n if(button.outerWidth() <= (original_width + 5)) {\n expanded = true;\n button.animate(\n {width: settings.animate_width + 'px'}, settings.speed, function(){\n target.show(0, function(){\n input.focus();\n // Set the button to auto width to make\n // sure it has room for any inputs\n button.width(\"auto\");\n // Explicitly set the width of the button\n // so the close animation works properly\n button.width(button.width());\n });\n }\n )\n }\n }\n function collapseButton(){\n target.hide();\n button.animate({width: original_width + 'px'}, settings.speed);\n }\n function inputEmpty(){\n return $.trim(input.val()) == \"\";\n }\n }\n}\n","export default class {\n connect() {\n // Add Select/Deselect all button behavior\n this.addCheckboxToggleBehavior();\n this.addEnableToggleBehavior();\n }\n \n // Add Select/Deselect all button behavior\n addCheckboxToggleBehavior() {\n $(\"[data-behavior='metadata-select']\").each(function(){\n var button = $(this)\n var parentCell = button.parents(\"th\");\n var table = parentCell.closest(\"table\");\n var columnRows = $(\"tr td:nth-child(\" + (parentCell.index() + 1) + \")\", table);\n var checkboxes = $(\"input[type='checkbox']\", columnRows);\n swapSelectAllButtonText(button, columnRows);\n // Add the check/uncheck behavior to the button\n // and swap the button text if necessary\n button.on('click', function(e){\n e.preventDefault();\n var allChecked = allCheckboxesChecked(columnRows);\n columnRows.each(function(){\n $(\"input[type='checkbox']\", $(this)).prop('checked', !allChecked);\n swapSelectAllButtonText(button, columnRows);\n });\n });\n // Swap button text when a checkbox value changes\n checkboxes.each(function(){\n $(this).on('change', function(){\n swapSelectAllButtonText(button, columnRows);\n });\n });\n });\n // Check number of checkboxes against the number of checked\n // checkboxes to determine if all of them are checked or not\n function allCheckboxesChecked(elements) {\n return ($(\"input[type='checkbox']\", elements).length == $(\"input[type='checkbox']:checked\", elements).length)\n }\n // Swap the button text to \"Deselect all\"\n // when all the checkboxes are checked and\n // \"Select all\" when any are unchecked\n function swapSelectAllButtonText(button, elements) {\n if ( allCheckboxesChecked(elements) ) {\n button.text(button.data('deselect-text'));\n } else {\n button.text(button.data('select-text'));\n }\n }\n }\n\n addEnableToggleBehavior() {\n $(\"[data-behavior='enable-feature']\").each(function(){\n var checkbox = $(this);\n var target = $($(this).data('target'));\n\n checkbox.on('change', function() {\n if ($(this).is(':checked')) {\n target.find('input:checkbox').not(\"[data-behavior='enable-feature']\").prop('checked', true).attr('disabled', false);\n } else {\n target.find('input:checkbox').not(\"[data-behavior='enable-feature']\").prop('checked', false).attr('disabled', true);\n }\n });\n });\n }\n}","export default class {\n connect() {\n new Clipboard('.copy-email-addresses');\n }\n}\n","export default class Iiif {\n constructor(manifestUrl, manifest) {\n this.manifestUrl = manifestUrl;\n this.manifest = manifest;\n }\n\n sequences() {\n var it = {};\n var context = this;\n it[Symbol.iterator] = function*() {\n for (let sequence of context.manifest.sequences) {\n yield sequence;\n };\n }\n return it;\n }\n\n canvases() {\n var it = {};\n var context = this;\n it[Symbol.iterator] = function*() {\n for (let sequence of context.sequences()) {\n for (let canvas of sequence.canvases) {\n yield canvas;\n }\n }\n }\n return it;\n }\n\n images() {\n var it = {};\n var context = this;\n it[Symbol.iterator] = function*() {\n for (let canvas of context.canvases()) {\n for (let image of canvas.images) {\n var iiifService = image.resource.service['@id'];\n yield {\n 'thumb': iiifService + '/full/!100,100/0/default.jpg',\n 'tilesource': iiifService + '/info.json',\n 'manifest': context.manifestUrl,\n 'canvasId': canvas['@id'],\n 'imageId': image['@id']\n };\n }\n }\n }\n return it;\n }\n\n imagesArray() {\n return Array.from(this.images())\n }\n}\n","import Iiif from 'spotlight/admin/iiif'\n\nexport function addImageSelector(input, panel, manifestUrl, initialize) {\n if (!manifestUrl) {\n showNonIiifAlert(input);\n return;\n }\n var cropper = input.data('iiifCropper');\n $.ajax(manifestUrl).done(\n function(manifest) {\n var iiifManifest = new Iiif(manifestUrl, manifest);\n\n var thumbs = iiifManifest.imagesArray();\n\n hideNonIiifAlert(input);\n\n if (initialize) {\n cropper.setIiifFields(thumbs[0]);\n panel.multiImageSelector(); // Clears out existing selector\n }\n\n if(thumbs.length > 1) {\n panel.show();\n panel.multiImageSelector(thumbs, function(selectorImage) {\n cropper.setIiifFields(selectorImage);\n }, cropper.iiifImageField.val());\n }\n }\n );\n}\n\nfunction showNonIiifAlert(input){\n input.parent().prev('[data-behavior=\"non-iiif-alert\"]').show();\n}\n\nfunction hideNonIiifAlert(input){\n input.parent().prev('[data-behavior=\"non-iiif-alert\"]').hide();\n}","const Spotlight = function() {\n var buffer = [];\n return {\n onLoad: function(func) {\n buffer.push(func);\n },\n\n activate: function() {\n for(var i = 0; i < buffer.length; i++) {\n buffer[i].call();\n }\n },\n csrfToken: function () {\n return document.querySelector('meta[name=csrf-token]')?.content\n },\n ZprLinks: {\n close: \"\",\n zoomIn: \"\\n\",\n zoomOut: \"\\n\"\n }\n };\n}();\n\n// This allows us to configure Spotlight in app/views/layouts/base.html.erb\nwindow.Spotlight = Spotlight\n\nexport default Spotlight\n\nBlacklight.onLoad(function() {\n Spotlight.activate();\n});\n\n","import { addImageSelector } from 'spotlight/admin/add_image_selector'\nimport Core from 'spotlight/core'\n\nexport default class Crop {\n constructor(cropArea) {\n this.cropArea = cropArea;\n this.cropArea.data('iiifCropper', this);\n this.cropSelector = '[data-cropper=\"' + cropArea.data('cropperKey') + '\"]';\n this.cropTool = $(this.cropSelector);\n this.formPrefix = this.cropTool.data('form-prefix');\n this.iiifUrlField = $('#' + this.formPrefix + '_iiif_tilesource');\n this.iiifRegionField = $('#' + this.formPrefix + '_iiif_region');\n this.iiifManifestField = $('#' + this.formPrefix + '_iiif_manifest_url');\n this.iiifCanvasField = $('#' + this.formPrefix + '_iiif_canvas_id');\n this.iiifImageField = $('#' + this.formPrefix + '_iiif_image_id');\n\n this.form = cropArea.closest('form');\n this.tileSource = null;\n }\n\n // Render the cropper environment and add hooks into the autocomplete and upload forms\n render() {\n this.setupAutoCompletes();\n this.setupAjaxFileUpload();\n this.setupExistingIiifCropper();\n }\n\n // Setup the cropper on page load if the field\n // that holds the IIIF url is populated\n setupExistingIiifCropper() {\n if(this.iiifUrlField.val() === '') {\n return;\n }\n\n this.addImageSelectorToExistingCropTool();\n this.setTileSource(this.iiifUrlField.val());\n }\n\n // Display the IIIF Cropper map with the current IIIF Layer (and cropbox, once the layer is available)\n setupIiifCropper() {\n this.loaded = false;\n\n this.renderCropperMap();\n\n if (this.imageLayer) {\n // Force a broken layer's container to be an element before removing.\n // Code in leaflet-iiif land calls delete on the image layer's container when removing,\n // which errors if there is an issue fetching the info.json and stops further necessary steps to execute.\n if(!this.imageLayer._container) {\n this.imageLayer._container = $('');\n }\n this.cropperMap.removeLayer(this.imageLayer);\n }\n\n this.imageLayer = L.tileLayer.iiif(this.tileSource).addTo(this.cropperMap);\n\n var self = this;\n this.imageLayer.on('load', function() {\n if (!self.loaded) {\n var region = self.getCropRegion();\n self.positionIiifCropBox(region);\n self.loaded = true;\n }\n });\n\n this.cropArea.data('initiallyVisible', this.cropArea.is(':visible'));\n }\n\n // Get (or initialize) the current crop region from the form data\n getCropRegion() {\n var regionFieldValue = this.iiifRegionField.val();\n if(!regionFieldValue || regionFieldValue === '') {\n var region = this.defaultCropRegion();\n this.iiifRegionField.val(region);\n return region;\n } else {\n return regionFieldValue.split(',');\n }\n }\n\n // Calculate a default crop region in the center of the image using the correct aspect ratio\n defaultCropRegion() {\n var imageWidth = this.imageLayer.x;\n var imageHeight = this.imageLayer.y;\n\n var boxWidth = Math.floor(imageWidth / 2);\n var boxHeight = Math.floor(boxWidth / this.aspectRatio());\n\n return [\n Math.floor((imageWidth - boxWidth) / 2),\n Math.floor((imageHeight - boxHeight) / 2),\n boxWidth,\n boxHeight\n ];\n }\n\n // Calculate the required aspect ratio for the crop area\n aspectRatio() {\n var cropWidth = parseInt(this.cropArea.data('crop-width'));\n var cropHeight = parseInt(this.cropArea.data('crop-height'));\n return cropWidth / cropHeight;\n }\n\n // Position the IIIF Crop Box at the given IIIF region\n positionIiifCropBox(region) {\n var bounds = this.unprojectIIIFRegionToBounds(region);\n\n if (!this.cropBox) {\n this.renderCropBox(bounds);\n }\n\n this.cropBox.setBounds(bounds);\n this.cropperMap.invalidateSize();\n this.cropperMap.fitBounds(bounds);\n\n this.cropBox.editor.editLayer.clearLayers();\n this.cropBox.editor.refresh();\n this.cropBox.editor.initVertexMarkers();\n }\n\n // Set all of the various input fields to\n // the appropriate IIIF URL or identifier\n setIiifFields(iiifObject) {\n this.setTileSource(iiifObject.tilesource);\n this.iiifManifestField.val(iiifObject.manifest);\n this.iiifCanvasField.val(iiifObject.canvasId);\n this.iiifImageField.val(iiifObject.imageId);\n }\n\n // Set the Crop tileSource and setup the cropper\n setTileSource(source) {\n if (source == this.tileSource) {\n return;\n }\n\n if (source === null || source === undefined) {\n console.error('No tilesource provided when setting up IIIF Cropper');\n return;\n }\n\n if (this.cropBox) {\n this.iiifRegionField.val(\"\");\n }\n\n this.tileSource = source;\n this.iiifUrlField.val(source);\n this.setupIiifCropper();\n }\n\n // Render the Leaflet Map into the crop area\n renderCropperMap() {\n if (this.cropperMap) {\n return;\n }\n this.cropperMap = L.map(this.cropArea.attr('id'), {\n editable: true,\n center: [0, 0],\n crs: L.CRS.Simple,\n zoom: 0,\n editOptions: {\n rectangleEditorClass: this.aspectRatioPreservingRectangleEditor(this.aspectRatio())\n }\n });\n this.invalidateMapSizeOnTabToggle();\n }\n\n // Render the crop box (a Leaflet editable rectangle) onto the canvas\n renderCropBox(initialBounds) {\n this.cropBox = L.rectangle(initialBounds);\n this.cropBox.addTo(this.cropperMap);\n this.cropBox.enableEdit();\n this.cropBox.on('dblclick', L.DomEvent.stop).on('dblclick', this.cropBox.toggleEdit);\n\n var self = this;\n this.cropperMap.on('editable:dragend editable:vertex:dragend', function(e) {\n var bounds = e.layer.getBounds();\n var region = self.projectBoundsToIIIFRegion(bounds);\n\n self.iiifRegionField.val(region.join(','));\n });\n }\n\n // Get the maximum zoom level for the IIIF Layer (always 1:1 image pixel to canvas?)\n maxZoom() {\n if(this.imageLayer) {\n return this.imageLayer.maxZoom;\n }\n }\n\n // Take a Leaflet LatLngBounds object and transform it into a IIIF [x, y, w, h] region\n projectBoundsToIIIFRegion(bounds) {\n var min = this.cropperMap.project(bounds.getNorthWest(), this.maxZoom());\n var max = this.cropperMap.project(bounds.getSouthEast(), this.maxZoom());\n return [\n Math.max(Math.floor(min.x), 0),\n Math.max(Math.floor(min.y), 0),\n Math.floor(max.x - min.x),\n Math.floor(max.y - min.y)\n ];\n }\n\n // Take a IIIF [x, y, w, h] region and transform it into a Leaflet LatLngBounds\n unprojectIIIFRegionToBounds(region) {\n var minPoint = L.point(parseInt(region[0]), parseInt(region[1]));\n var maxPoint = L.point(parseInt(region[0]) + parseInt(region[2]), parseInt(region[1]) + parseInt(region[3]));\n\n var min = this.cropperMap.unproject(minPoint, this.maxZoom());\n var max = this.cropperMap.unproject(maxPoint, this.maxZoom());\n return L.latLngBounds(min, max);\n }\n\n // TODO: Add accessors to update hidden inputs with IIIF uri/ids?\n\n // Setup autocomplete inputs to have the iiif_cropper context\n setupAutoCompletes() {\n var input = $('[data-behavior=\"autocomplete\"]', this.cropTool);\n input.data('iiifCropper', this);\n }\n\n setupAjaxFileUpload() {\n this.fileInput = $('input[type=\"file\"]', this.cropTool);\n this.fileInput.change(() => this.uploadFile());\n }\n\n addImageSelectorToExistingCropTool() {\n if(this.iiifManifestField.val() === '') {\n return;\n }\n\n var input = $('[data-behavior=\"autocomplete\"]', this.cropTool);\n var panel = $(input.data('target-panel'));\n\n addImageSelector(input, panel, this.iiifManifestField.val(), !this.iiifImageField.val());\n }\n\n invalidateMapSizeOnTabToggle() {\n var tabs = $('[role=\"tablist\"]', this.form);\n var self = this;\n tabs.on('shown.bs.tab', function() {\n if(self.cropArea.data('initiallyVisible') === false && self.cropArea.is(':visible')) {\n self.cropperMap.invalidateSize();\n // Because the map size is 0,0 when image is loading (not visible) we need to refit the bounds of the layer\n self.imageLayer._fitBounds();\n self.cropArea.data('initiallyVisible', null);\n }\n });\n }\n\n // Get all the form data with the exception of the _method field.\n getData() {\n var data = new FormData(this.form[0]);\n data.append('_method', null);\n return data;\n }\n\n uploadFile() {\n var url = this.fileInput.data('endpoint')\n // Every post creates a new image/masthead.\n // Because they create IIIF urls which are heavily cached.\n $.ajax({\n url: url, //Server script to process data\n type: 'POST',\n success: (data, stat, xhr) => this.successHandler(data, stat, xhr),\n // error: errorHandler,\n // Form data\n data: this.getData(),\n headers: {\n 'X-CSRF-Token': Core.csrfToken() || ''\n },\n //Options to tell jQuery not to process data or worry about content-type.\n cache: false,\n contentType: false,\n processData: false\n });\n }\n\n successHandler(data, stat, xhr) {\n this.setIiifFields({ tilesource: data.tilesource });\n this.setUploadId(data.id);\n }\n\n setUploadId(id) {\n $('#' + this.formPrefix + \"_upload_id\").val(id);\n }\n\n aspectRatioPreservingRectangleEditor(aspect) {\n return L.Editable.RectangleEditor.extend({\n extendBounds: function (e) {\n var index = e.vertex.getIndex(),\n next = e.vertex.getNext(),\n previous = e.vertex.getPrevious(),\n oppositeIndex = (index + 2) % 4,\n opposite = e.vertex.latlngs[oppositeIndex];\n\n if ((index % 2) == 1) {\n // calculate horiz. displacement\n e.latlng.update([opposite.lat + ((1 / aspect) * (opposite.lng - e.latlng.lng)), e.latlng.lng]);\n } else {\n // calculate vert. displacement\n e.latlng.update([e.latlng.lat, (opposite.lng - (aspect * (opposite.lat - e.latlng.lat)))]);\n }\n var bounds = new L.LatLngBounds(e.latlng, opposite);\n // Update latlngs by hand to preserve order.\n previous.latlng.update([e.latlng.lat, opposite.lng]);\n next.latlng.update([opposite.lat, e.latlng.lng]);\n this.updateBounds(bounds);\n this.refreshVertexMarkers();\n }\n });\n }\n}\n","import Crop from 'spotlight/admin/crop';\n\nexport default class {\n connect() {\n $('[data-behavior=\"iiif-cropper\"]').each(function() {\n var cropElement = $(this)\n new Crop(cropElement).render()\n })\n }\n}\n","/*\n Simple plugin add edit-in-place behavior\n*/\nexport default class {\n connect() {\n $('[data-in-place-edit-target]').each(function() {\n $(this).on('click.inplaceedit', function() {\n var $label = $(this).find($(this).data('in-place-edit-target'));\n var $input = $(this).find($(this).data('in-place-edit-field-target'));\n\n // hide the edit-in-place affordance icon while in edit mode\n $(this).addClass('hide-edit-icon');\n $label.hide();\n $input.val($label.text());\n $input.attr('type', 'text');\n $input.select();\n $input.focus();\n\n $input.on('keypress', function(e) {\n if(e.which == 13) {\n $input.trigger('blur.inplaceedit');\n return false;\n }\n });\n\n $input.on('blur.inplaceedit', function() {\n var value = $input.val();\n\n if ($.trim(value).length == 0) {\n $input.val($label.text());\n } else {\n $label.text(value);\n }\n\n $label.show();\n $input.attr('type', 'hidden');\n // when leaving edit mode, should no longer hide edit-in-place affordance icon\n $(\"[data-in-place-edit-target]\").removeClass('hide-edit-icon');\n\n return false;\n });\n\n return false;\n });\n })\n\n $(\"[data-behavior='restore-default']\").each(function(){\n var hidden = $(\"[data-default-value]\", $(this));\n var value = $($(\"[data-in-place-edit-target]\", $(this)).data('in-place-edit-target'), $(this));\n var button = $(\"[data-restore-default]\", $(this));\n\n hidden.on('keypress', function(e) {\n if(e.which == 13) {\n hidden.trigger('blur');\n return false;\n }\n });\n\n hidden.on('blur', function(){\n if( $(this).val() == $(this).data('default-value') ) {\n button.addClass('d-none');\n } else {\n button.removeClass('d-none');\n }\n });\n button.on('click', function(e){\n e.preventDefault();\n hidden.val(hidden.data('default-value'));\n value.text(hidden.data('default-value'));\n button.hide();\n });\n });\n }\n}\n","export default class {\n connect() {\n $('[data-autocomplete-tag=\"true\"]').each(function(_i, el) {\n var $el = $(el);\n // By default tags input binds on page ready to [data-role=tagsinput],\n // however, that doesn't work with Turbolinks. So we init manually:\n $el.tagsinput();\n\n var tags = new Bloodhound({\n datumTokenizer: function(d) { return Bloodhound.tokenizers.whitespace(d.name); },\n queryTokenizer: Bloodhound.tokenizers.whitespace,\n limit: 100,\n prefetch: {\n url: $el.data('autocomplete-url'),\n ttl: 1,\n filter: function(list) {\n // Let the dom know that the response has been returned\n $el.attr('data-autocomplete-fetched', true);\n return $.map(list, function(tag) { return { name: tag }; });\n }\n }\n });\n\n tags.initialize();\n\n $el.tagsinput('input').typeahead({highlight: true, hint: false}, {\n name: 'tags',\n displayKey: 'name',\n source: tags.ttAdapter()\n }).bind('typeahead:selected', $.proxy(function (obj, datum) {\n $el.tagsinput('add', datum.name);\n $el.tagsinput('input').typeahead('val', '');\n })).bind('blur', function() {\n $el.tagsinput('add', $el.tagsinput('input').typeahead('val'));\n $el.tagsinput('input').typeahead('val', '');\n })\n })\n }\n}\n","/*\nhttps://gist.github.com/pjambet/3710461\n*/\nvar LATIN_MAP = {\n 'À': 'A', 'Á': 'A', 'Â': 'A', 'Ã': 'A', 'Ä': 'A', 'Å': 'A', 'Æ': 'AE', 'Ç':\n 'C', 'È': 'E', 'É': 'E', 'Ê': 'E', 'Ë': 'E', 'Ì': 'I', 'Í': 'I', 'Î': 'I',\n 'Ï': 'I', 'Ð': 'D', 'Ñ': 'N', 'Ò': 'O', 'Ó': 'O', 'Ô': 'O', 'Õ': 'O', 'Ö':\n 'O', 'Ő': 'O', 'Ø': 'O', 'Ù': 'U', 'Ú': 'U', 'Û': 'U', 'Ü': 'U', 'Ű': 'U',\n 'Ý': 'Y', 'Þ': 'TH', 'ß': 'ss', 'à':'a', 'á':'a', 'â': 'a', 'ã': 'a', 'ä':\n 'a', 'å': 'a', 'æ': 'ae', 'ç': 'c', 'è': 'e', 'é': 'e', 'ê': 'e', 'ë': 'e',\n 'ì': 'i', 'í': 'i', 'î': 'i', 'ï': 'i', 'ð': 'd', 'ñ': 'n', 'ò': 'o', 'ó':\n 'o', 'ô': 'o', 'õ': 'o', 'ö': 'o', 'ő': 'o', 'ø': 'o', 'ù': 'u', 'ú': 'u',\n 'û': 'u', 'ü': 'u', 'ű': 'u', 'ý': 'y', 'þ': 'th', 'ÿ': 'y'\n}\nvar LATIN_SYMBOLS_MAP = {\n '©':'(c)'\n}\nvar GREEK_MAP = {\n 'α':'a', 'β':'b', 'γ':'g', 'δ':'d', 'ε':'e', 'ζ':'z', 'η':'h', 'θ':'8',\n 'ι':'i', 'κ':'k', 'λ':'l', 'μ':'m', 'ν':'n', 'ξ':'3', 'ο':'o', 'π':'p',\n 'ρ':'r', 'σ':'s', 'τ':'t', 'υ':'y', 'φ':'f', 'χ':'x', 'ψ':'ps', 'ω':'w',\n 'ά':'a', 'έ':'e', 'ί':'i', 'ό':'o', 'ύ':'y', 'ή':'h', 'ώ':'w', 'ς':'s',\n 'ϊ':'i', 'ΰ':'y', 'ϋ':'y', 'ΐ':'i',\n 'Α':'A', 'Β':'B', 'Γ':'G', 'Δ':'D', 'Ε':'E', 'Ζ':'Z', 'Η':'H', 'Θ':'8',\n 'Ι':'I', 'Κ':'K', 'Λ':'L', 'Μ':'M', 'Ν':'N', 'Ξ':'3', 'Ο':'O', 'Π':'P',\n 'Ρ':'R', 'Σ':'S', 'Τ':'T', 'Υ':'Y', 'Φ':'F', 'Χ':'X', 'Ψ':'PS', 'Ω':'W',\n 'Ά':'A', 'Έ':'E', 'Ί':'I', 'Ό':'O', 'Ύ':'Y', 'Ή':'H', 'Ώ':'W', 'Ϊ':'I',\n 'Ϋ':'Y'\n}\nvar TURKISH_MAP = {\n 'ş':'s', 'Ş':'S', 'ı':'i', 'İ':'I', 'ç':'c', 'Ç':'C', 'ü':'u', 'Ü':'U',\n 'ö':'o', 'Ö':'O', 'ğ':'g', 'Ğ':'G'\n}\nvar RUSSIAN_MAP = {\n 'а':'a', 'б':'b', 'в':'v', 'г':'g', 'д':'d', 'е':'e', 'ё':'yo', 'ж':'zh',\n 'з':'z', 'и':'i', 'й':'j', 'к':'k', 'л':'l', 'м':'m', 'н':'n', 'о':'o',\n 'п':'p', 'р':'r', 'с':'s', 'т':'t', 'у':'u', 'ф':'f', 'х':'h', 'ц':'c',\n 'ч':'ch', 'ш':'sh', 'щ':'sh', 'ъ':'', 'ы':'y', 'ь':'', 'э':'e', 'ю':'yu',\n 'я':'ya',\n 'А':'A', 'Б':'B', 'В':'V', 'Г':'G', 'Д':'D', 'Е':'E', 'Ё':'Yo', 'Ж':'Zh',\n 'З':'Z', 'И':'I', 'Й':'J', 'К':'K', 'Л':'L', 'М':'M', 'Н':'N', 'О':'O',\n 'П':'P', 'Р':'R', 'С':'S', 'Т':'T', 'У':'U', 'Ф':'F', 'Х':'H', 'Ц':'C',\n 'Ч':'Ch', 'Ш':'Sh', 'Щ':'Sh', 'Ъ':'', 'Ы':'Y', 'Ь':'', 'Э':'E', 'Ю':'Yu',\n 'Я':'Ya'\n}\nvar UKRAINIAN_MAP = {\n 'Є':'Ye', 'І':'I', 'Ї':'Yi', 'Ґ':'G', 'є':'ye', 'і':'i', 'ї':'yi', 'ґ':'g'\n}\nvar CZECH_MAP = {\n 'č':'c', 'ď':'d', 'ě':'e', 'ň': 'n', 'ř':'r', 'š':'s', 'ť':'t', 'ů':'u',\n 'ž':'z', 'Č':'C', 'Ď':'D', 'Ě':'E', 'Ň': 'N', 'Ř':'R', 'Š':'S', 'Ť':'T',\n 'Ů':'U', 'Ž':'Z'\n}\n\nvar POLISH_MAP = {\n 'ą':'a', 'ć':'c', 'ę':'e', 'ł':'l', 'ń':'n', 'ó':'o', 'ś':'s', 'ź':'z',\n 'ż':'z', 'Ą':'A', 'Ć':'C', 'Ę':'e', 'Ł':'L', 'Ń':'N', 'Ó':'o', 'Ś':'S',\n 'Ź':'Z', 'Ż':'Z'\n}\n\nvar LATVIAN_MAP = {\n 'ā':'a', 'č':'c', 'ē':'e', 'ģ':'g', 'ī':'i', 'ķ':'k', 'ļ':'l', 'ņ':'n',\n 'š':'s', 'ū':'u', 'ž':'z', 'Ā':'A', 'Č':'C', 'Ē':'E', 'Ģ':'G', 'Ī':'i',\n 'Ķ':'k', 'Ļ':'L', 'Ņ':'N', 'Š':'S', 'Ū':'u', 'Ž':'Z'\n}\n\nvar ALL_DOWNCODE_MAPS=new Array()\nALL_DOWNCODE_MAPS[0]=LATIN_MAP\nALL_DOWNCODE_MAPS[1]=LATIN_SYMBOLS_MAP\nALL_DOWNCODE_MAPS[2]=GREEK_MAP\nALL_DOWNCODE_MAPS[3]=TURKISH_MAP\nALL_DOWNCODE_MAPS[4]=RUSSIAN_MAP\nALL_DOWNCODE_MAPS[5]=UKRAINIAN_MAP\nALL_DOWNCODE_MAPS[6]=CZECH_MAP\nALL_DOWNCODE_MAPS[7]=POLISH_MAP\nALL_DOWNCODE_MAPS[8]=LATVIAN_MAP\n\nvar Downcoder = new Object();\nDowncoder.Initialize = function()\n{\n if (Downcoder.map) // already made\n return ;\n Downcoder.map ={}\n Downcoder.chars = '' ;\n for(var i in ALL_DOWNCODE_MAPS)\n {\n var lookup = ALL_DOWNCODE_MAPS[i]\n for (var c in lookup)\n {\n Downcoder.map[c] = lookup[c] ;\n Downcoder.chars += c ;\n }\n }\n Downcoder.regex = new RegExp('[' + Downcoder.chars + ']|[^' + Downcoder.chars + ']+','g') ;\n }\n \nconst downcode = function( slug )\n{\n Downcoder.Initialize() ;\n var downcoded =\"\"\n var pieces = slug.match(Downcoder.regex);\n if(pieces)\n {\n for (var i = 0 ; i < pieces.length ; i++)\n {\n if (pieces[i].length == 1)\n {\n var mapped = Downcoder.map[pieces[i]] ;\n if (mapped != null)\n {\n downcoded+=mapped;\n continue ;\n }\n }\n downcoded+=pieces[i];\n }\n }\n else\n {\n downcoded = slug;\n }\n return downcoded;\n}\n\n\nfunction URLify(s, num_chars) {\n // changes, e.g., \"Petty theft\" to \"petty_theft\"\n // remove all these words from the string before urlifying\n s = downcode(s);\n //\n // if downcode doesn't hit, the char will be stripped here\n s = s.replace(/[^-\\w\\s]/g, ' '); // remove unneeded chars\n s = s.replace(/^\\s+|\\s+$/g, ''); // trim leading/trailing spaces\n s = s.replace(/[-\\s]+/g, '-'); // convert spaces to hyphens\n s = s.toLowerCase(); // convert to lowercase\n return s.substring(0, num_chars);// trim to first num_chars chars\n}\n\nexport { URLify };","import { URLify } from 'parameterize';\n\nexport default class {\n connect() {\n // auto-fill the exhibit slug on the new exhibit form\n $('#new_exhibit').each(function() {\n $('#exhibit_title').on('change keyup', function() {\n $('#exhibit_slug').attr('placeholder', URLify($(this).val(), $(this).val().length));\n });\n\n $('#exhibit_slug').on('focus', function() {\n if ($(this).val() === '') {\n $(this).val($(this).attr('placeholder'));\n }\n });\n });\n\n $(\"#another-email\").on(\"click\", function(e) {\n e.preventDefault();\n\n var container = $(this).closest('.form-group');\n var contacts = container.find('.contact');\n var inputContainer = contacts.first().clone();\n\n // wipe out any values from the inputs\n inputContainer.find('input').each(function() {\n $(this).val('');\n $(this).attr('id', $(this).attr('id').replace('0', contacts.length));\n $(this).attr('name', $(this).attr('name').replace('0', contacts.length));\n if ($(this).attr('aria-label')) {\n $(this).attr('aria-label', $(this).attr('aria-label').replace('1', contacts.length + 1));\n }\n });\n\n inputContainer.find('.contact-email-delete-wrapper').remove();\n inputContainer.find('.confirmation-status').remove();\n\n // bootstrap does not render input-groups with only one value in them correctly.\n inputContainer.find('.input-group input:only-child').closest('.input-group').removeClass('input-group');\n\n $(inputContainer).insertAfter(contacts.last());\n });\n\n $('.contact-email-delete').on('ajax:success', function() {\n $(this).closest('.contact').fadeOut(250, function() { $(this).remove(); });\n });\n\n $('.contact-email-delete').on('ajax:error', function(event, _xhr, _status, error) {\n var errSpan = $(this).closest('.contact').find('.contact-email-delete-error');\n errSpan.show();\n errSpan.find('.error-msg').first().text(error || event.detail[1]);\n });\n\n $('.btn-with-tooltip').tooltip();\n\n // Put focus in saved search title input when Save this search modal is shown\n $('#save-modal').on('shown.bs.modal', function () {\n $('#search_title').focus();\n });\n }\n}","\n(function($, _) {\n 'use strict';\n\n /*\n * SerializedForm is built as a singleton jQuery plugin. It needs to be able to\n * handle instantiation from multiple sources, and use the [data-form-observer]\n * as global state object.\n */\n $.SerializedForm = function () {\n var $serializedForm;\n var plugin = this;\n\n // Store form serialization in data attribute\n function serializeFormStatus () {\n $serializedForm.data('serialized-form', formSerialization($serializedForm));\n }\n\n // Do custom serialization of the sir-trevor form data. This needs to be a\n // passed in argument for comparison later on.\n function formSerialization (form) {\n var content_editable = [];\n var i = 0;\n $(\"[contenteditable='true']\", form).each(function(){\n content_editable.push('&contenteditable_' + i + '=' + $(this).text());\n });\n return form.serialize() + content_editable.join();\n }\n\n // Unbind observing form on submit (which we have to do because of turbolinks)\n function unbindObservedFormSubmit () {\n $serializedForm.on('submit', function () {\n $(this).data('being-submitted', true);\n });\n }\n\n // Get the stored serialized form status\n function serializedFormStatus () {\n return $serializedForm.data('serialized-form');\n }\n\n // Check all observed forms on page for status change\n plugin.observedFormsStatusHasChanged = function () {\n var unsaved_changes = false;\n $('[data-form-observer]').each(function (){\n if ( !$(this).data(\"being-submitted\") ) {\n if (serializedFormStatus() != formSerialization($(this))) {\n unsaved_changes = true;\n }\n }\n });\n return unsaved_changes;\n }\n\n function init () {\n $serializedForm = $('[data-form-observer]');\n serializeFormStatus();\n unbindObservedFormSubmit();\n }\n\n init();\n\n return plugin;\n };\n})(jQuery);\n\nexport default class {\n connect() {\n // Instantiate the singleton SerializedForm plugin\n var serializedForm = $.SerializedForm();\n $(window).on('beforeunload page:before-change turbolinks:before-visit', function(event) {\n // Don't handle the same event twice #turbolinks\n if (event.handled !== true) {\n if ( serializedForm.observedFormsStatusHasChanged() ) {\n event.handled = true;\n var message = \"You have unsaved changes. Are you sure you want to leave this page?\";\n // There are variations in how Webkit browsers may handle this:\n // https://developer.mozilla.org/en-US/docs/Web/Events/beforeunload\n if ( event.type == \"beforeunload\" ) {\n return message;\n } else {\n return confirm(message)\n }\n }\n }\n });\n }\n}\n","export default class {\n delete_lock(el) {\n $.ajax({ url: $(el).data('lock'), type: 'POST', data: { _method: \"delete\" }, async: false});\n $(el).removeAttr('data-lock');\n }\n\n connect() {\n $('[data-lock]').on('click', (e) => {\n this.delete_lock(e.target);\n })\n }\n}","// Module to add multi-image selector to widget panels\n\n(function(){\n $.fn.multiImageSelector = function(image_versions, clickCallback, activeImageId) {\n var changeLink = $(\"Change\"),\n thumbsListContainer = $(\"\"),\n thumbList = $(\"\"),\n panel;\n\n var imageIds = $.map(image_versions, function(e) { return e['imageId']; });\n\n return init(this);\n\n function init(el) {\n panel = el;\n\n destroyExistingImageSelector();\n if(image_versions && image_versions.length > 1) {\n addChangeLink();\n addThumbsList();\n }\n }\n function addChangeLink() {\n $('[data-panel-image-pagination]', panel)\n .html(\"Image \" + indexOf(activeImageId) + \" of \" + image_versions.length)\n .show()\n .append(\" \")\n .append(changeLink);\n addChangeLinkBehavior();\n }\n\n function destroyExistingImageSelector() {\n var pagination = $('[data-panel-image-pagination]', panel);\n pagination.html('');\n pagination.next('.' + thumbsListContainer.attr('class')).remove();\n }\n\n function indexOf(thumb){\n const index = imageIds.indexOf(thumb)\n if (index > -1) {\n return index + 1;\n } else {\n return 1;\n }\n }\n function addChangeLinkBehavior() {\n changeLink.on('click', function(){\n thumbsListContainer.slideToggle();\n updateThumbListWidth();\n addScrollBehavior();\n scrollToActiveThumb();\n loadVisibleThumbs();\n swapChangeLinkText($(this));\n });\n }\n function updateThumbListWidth() {\n var width = 0;\n $('li', thumbList).each(function(){\n width += $(this).outerWidth();\n });\n thumbList.width(width + 5);\n }\n function loadVisibleThumbs(){\n var viewportWidth = thumbsListContainer.width();\n var width = 0;\n $('li', thumbList).each(function(){\n var thisThumb = $(this),\n image = $('img', thisThumb),\n totalWidth = width += thisThumb.width(),\n position = (thumbList.position().left + totalWidth) - thisThumb.width();\n\n if(position >= 0 && position < viewportWidth) {\n image.prop('src', image.data('src'));\n }\n });\n }\n function addScrollBehavior(){\n thumbsListContainer.scrollStop(function(){\n loadVisibleThumbs();\n });\n }\n function scrollToActiveThumb(){\n var halfContainerWidth = (thumbsListContainer.width() / 2),\n activeThumbLeftPosition = ($('.active', thumbList).position() || $('li', thumbList).first().position()).left,\n halfActiveThumbWidth = ($('.active', thumbList).width() / 2);\n thumbsListContainer.scrollLeft(\n (activeThumbLeftPosition - halfContainerWidth) + halfActiveThumbWidth\n );\n }\n function addThumbsList() {\n addThumbsToList();\n updateActiveThumb();\n $('.card-header', panel).append(\n thumbsListContainer.append(\n thumbList\n )\n );\n }\n function updateActiveThumb(){\n $('li', thumbList).each(function(){\n var item = $(this);\n if($('img', item).data('image-id') == activeImageId){\n item.addClass('active');\n }\n });\n }\n function swapChangeLinkText(link){\n link.text(\n link.text() == 'Change' ? 'Close' : 'Change'\n )\n }\n\n function addThumbsToList(){\n $.each(image_versions, function(i){\n var listItem = $('');\n listItem.on('click', function(){\n // get the current image id\n var imageid = $('img', $(this)).data('image-id');\n var src = $('img', $(this)).attr('src');\n\n if (typeof clickCallback === 'function' ) {\n clickCallback(image_versions[i]);\n }\n\n // mark the current selection as active\n $('li.active', thumbList).removeClass('active');\n $(this).addClass('active');\n\n // update the multi-image selector image\n $(\".pic img.img-thumbnail\", panel).attr(\"src\", src);\n\n $('[data-panel-image-pagination] [data-current-image]', panel).text(\n $('li', thumbList).index($(this)) + 1\n );\n scrollToActiveThumb();\n });\n $(\"img\", listItem).on('load', function() {\n updateThumbListWidth();\n });\n thumbList.append(listItem);\n });\n }\n };\n\n})(jQuery);\n\n// source: http://stackoverflow.com/questions/14035083/jquery-bind-event-on-scroll-stops\njQuery.fn.scrollStop = function(callback) {\n $(this).scroll(function() {\n var self = this,\n $this = $(self);\n\n if ($this.data('scrollTimeout')) {\n clearTimeout($this.data('scrollTimeout'));\n }\n\n $this.data('scrollTimeout', setTimeout(callback, 250, self));\n });\n};\n","// Place all the behaviors and hooks related to the matching controller here.\n// All this logic will automatically be available in application.js.\nimport Core from 'spotlight/core'\n\nexport default class {\n connect(){\n SirTrevor.setDefaults({\n iconUrl: Spotlight.sirTrevorIcon,\n uploadUrl: $('[data-attachment-endpoint]').data('attachment-endpoint'),\n ajaxOptions: {\n headers: {\n 'X-CSRF-Token': Core.csrfToken() || ''\n },\n credentials: 'same-origin'\n }\n });\n\n SirTrevor.Blocks.Heading.prototype.toolbarEnabled = true;\n SirTrevor.Blocks.Quote.prototype.toolbarEnabled = true;\n SirTrevor.Blocks.Text.prototype.toolbarEnabled = true;\n\n var instance = $('.js-st-instance').first();\n\n if (instance.length) {\n var editor = new SirTrevor.Editor({\n el: instance[0],\n blockTypes: instance.data('blockTypes'),\n defaultType:[\"Text\"],\n onEditorRender: function() {\n $.SerializedForm();\n },\n blockTypeLimits: {\n \"SearchResults\": 1\n }\n });\n\n editor.blockControls = Core.BlockControls.create(editor);\n\n new Core.BlockLimits(editor).enforceLimits(editor);\n }\n }\n}\n","export default class {\n connect() {\n var monitorElements = $('[data-behavior=\"progress-panel\"]');\n var defaultRefreshRate = 3000;\n var panelContainer;\n var pollers = [];\n\n $(monitorElements).each(function() {\n panelContainer = $(this);\n panelContainer.hide();\n var monitorUrl = panelContainer.data('monitorUrl');\n var refreshRate = panelContainer.data('refreshRate') || defaultRefreshRate;\n pollers.push(\n setInterval(function() {\n checkMonitorUrl(monitorUrl);\n }, refreshRate)\n );\n });\n\n // Clear the intervals on turbolink:click event (e.g. when the user navigates away from the page)\n $(document).on('turbolinks:click', function() {\n if (pollers.length > 0) {\n $.each(pollers, function() {\n clearInterval(this);\n });\n pollers = [];\n }\n });\n\n function checkMonitorUrl(url) {\n $.ajax(url).done(success).fail(fail);\n }\n\n function success(data) {\n if (data.recently_in_progress) {\n updateMonitorPanel(data);\n monitorPanel().show();\n } else {\n monitorPanel().hide();\n }\n }\n\n function fail() { monitorPanel().hide(); }\n\n function updateMonitorPanel(data) {\n panelStartDate().text(data.started_at);\n panelCurrentDate().text(data.updated_at);\n panelCompletedDate().text(data.updated_at);\n panelCurrent().text(data.completed);\n setPanelCompleted(data.finished);\n updatePanelTotals(data);\n updatePanelErrorMessage(data);\n updateProgressBar(data);\n\n panelContainer.show();\n }\n\n function updateProgressBar(data) {\n var percentage = calculatePercentage(data);\n progressBar()\n .attr('aria-valuemax', data.total)\n .attr('aria-valuenow', percentage)\n .css('width', percentage + '%')\n .text(percentage + '%');\n\n if (data.finished) {\n progressBar().removeClass('active').removeClass('progress-bar-striped');\n }\n }\n\n function updatePanelErrorMessage(data) {\n // We currently do not store this state,\n // but with this code we can in the future.\n if ( data.errored ) {\n panelErrorMessage().show();\n } else {\n panelErrorMessage().hide();\n }\n }\n\n function updatePanelTotals(data) {\n panelTotals().each(function() {\n $(this).text(data.total);\n });\n }\n\n function calculatePercentage(data) {\n if (data.total == 0) return 0;\n return Math.floor((data.completed / data.total) * 100);\n }\n\n function monitorPanel() {\n return panelContainer.find('.index-status');\n }\n\n function panelStartDate() {\n return monitorPanel()\n .find('[data-behavior=\"monitor-start\"]')\n .find('[data-behavior=\"date\"]');\n }\n\n function panelCurrentDate() {\n return monitorPanel()\n .find('[data-behavior=\"monitor-current\"]')\n .find('[data-behavior=\"date\"]');\n }\n\n function panelCompletedDate() {\n return monitorPanel()\n .find('[data-behavior=\"monitor-completed\"]')\n .find('[data-behavior=\"date\"]');\n }\n\n function panelTotals() {\n return monitorPanel().find('[data-behavior=\"total\"]');\n }\n\n function panelCurrent() {\n return monitorPanel()\n .find('[data-behavior=\"monitor-current\"]')\n .find('[data-behavior=\"completed\"]');\n }\n\n function progressBar() {\n return monitorPanel().find('.progress-bar');\n }\n\n function panelErrorMessage() {\n return monitorPanel().find('[data-behavior=\"monitor-error\"]');\n }\n\n function setPanelCompleted(finished) {\n var panel = monitorPanel().find('[data-behavior=\"monitor-completed\"]');\n\n if (finished) {\n panel.show();\n } else {\n panel.hide();\n }\n }\n\n return this;\n }\n}\n","export default class {\n connect() {\n // Don't allow unchecking of checkboxes with the data-readonly attribute \n $(\"input[type='checkbox'][data-readonly]\").on(\"click\", function(event) {\n event.preventDefault();\n })\n }\n}\n","import { addImageSelector } from 'spotlight/admin/add_image_selector'\n\n(function($){\n $.fn.spotlightSearchTypeAhead = function( options ) {\n $.each(this, function(){\n addAutocompleteBehavior($(this));\n });\n\n function addAutocompleteBehavior( typeAheadInput, _ ) {\n var settings = $.extend({\n displayKey: 'title',\n minLength: 0,\n highlight: (typeAheadInput.data('autocomplete-highlight') || true),\n hint: (typeAheadInput.data('autocomplete-hint') || false),\n autoselect: (typeAheadInput.data('autocomplete-autoselect') || true)\n }, options);\n typeAheadInput.typeahead(settings, {\n displayKey: settings.displayKey,\n source: settings.bloodhound.ttAdapter(),\n templates: {\n suggestion: settings.template\n }\n })\n }\n return this;\n }\n})( jQuery );\n\nfunction itemsBloodhound() {\n var results = new Bloodhound({\n datumTokenizer: function(d) {\n return Bloodhound.tokenizers.whitespace(d.title);\n },\n queryTokenizer: Bloodhound.tokenizers.whitespace,\n limit: 100,\n remote: {\n url: $('form[data-autocomplete-exhibit-catalog-path]').data('autocomplete-exhibit-catalog-path').replace(\"%25QUERY\", \"%QUERY\"),\n filter: function(response) {\n return $.map(response['docs'], function(doc) {\n return doc;\n })\n }\n }\n });\n results.initialize();\n return results;\n};\n\nfunction templateFunc(obj) {\n const thumbnail = obj.thumbnail ? `` : ''\n return $(`${thumbnail}\n ${obj.title}
${obj.description}
`)\n}\n\nexport function addAutocompletetoFeaturedImage(){\n if($('[data-featured-image-typeahead]').length > 0) {\n $('[data-featured-image-typeahead]').spotlightSearchTypeAhead({bloodhound: itemsBloodhound(), template: templateFunc}).on('click', function() {\n $(this).select();\n }).on('typeahead:selected typeahead:autocompleted', function(e, data) {\n var panel = $($(this).data('target-panel'));\n addImageSelector($(this), panel, data.iiif_manifest, true);\n $($(this).data('id-field')).val(data['global_id']);\n $(this).attr('type', 'text');\n });\n }\n}\n","/*\n Simple plugin to select form elements\n when other elements are clicked.\n*/\n(function($) {\n $.fn.selectRelatedInput = function() {\n var clickElements = this;\n\n $(clickElements).each(function() {\n var target = $($(this).data('input-select-target'));\n\n var event;\n\n if ($(this).is(\"select\")) {\n event = 'change';\n } else {\n event = 'click';\n }\n\n $(this).on(event, function() {\n if (target.is(\":checkbox\") || target.is(\":radio\")) {\n target.prop('checked', true);\n } else {\n target.focus();\n }\n });\n });\n\n return this;\n };\n})(jQuery);\n\nexport default class {\n connect() {\n $('[data-input-select-target]').selectRelatedInput();\n }\n}\n","const Module = (function() {\n var nestableSelector = '[data-behavior=\"nestable\"]';\n return {\n init: function(selector){\n\n $(selector || nestableSelector).each(function(){\n // Because the Rails helper will not maintain the case that Nestable\n // expects, we just need to do this manual conversion. :(\n var data = $(this).data();\n data.expandBtnHTML = data.expandBtnHtml;\n data.collapseBtnHTML = data.collapseBtnHtml;\n $(this).nestable(data);\n updateWeightsAndRelationships($(this));\n });\n }\n };\n function updateWeightsAndRelationships(nestedList){\n nestedList.on('change', function(event){\n var container = $(event.currentTarget);\n var data = $(this).nestable('serialize');\n var weight = 0;\n for(var i in data){\n var parent_id = data[i]['id'];\n const parent_node = findNode(parent_id, container);\n setWeight(parent_node, weight++);\n if(data[i]['children']){\n var children = data[i]['children'];\n for(var child in children){\n var id = children[child]['id']\n var child_node = findNode(id, container);\n setWeight(child_node, weight++);\n setParent(child_node, parent_id);\n }\n } else {\n setParent(parent_node, \"\");\n }\n }\n });\n\n }\n function findNode(id, container) {\n return container.find(\"[data-id=\"+id+\"]\");\n }\n\n function setWeight(node, weight) {\n weight_field(node).val(weight);\n }\n\n function setParent(node, parent_id) {\n parent_page_field(node).val(parent_id);\n }\n\n /* find the input element with data-property=\"weight\" that is nested under the given node */\n function weight_field(node) {\n return find_property(node, \"weight\");\n }\n\n /* find the input element with data-property=\"parent_page\" that is nested under the given node */\n function parent_page_field(node){\n return find_property(node, \"parent_page\");\n }\n\n function find_property(node, property) {\n return node.find(\"input[data-property=\" + property + \"]\");\n }\n})();\n\nexport default Module\n","export default class {\n connect() {\n if ($('[role=tabpanel]').length > 0 && window.location.hash) {\n var tabpanel = $(window.location.hash).closest('[role=tabpanel]');\n $('a[role=tab][href=\"#'+tabpanel.attr('id')+'\"]').tab('show');\n }\n }\n}\n","// translationProgress is a plugin that updates the \"3/14\" progress\n// counters in the tabs of the translation adminstration dashboard.\n// This works by counting the number of progress items and translations\n// present (indicated by data attributes) in each tab's content\nexport default class {\n connect() {\n $('[data-behavior=\"translation-progress\"]').each(function(){\n var currentTab = $(this);\n var tabName = $(this).attr('aria-controls');\n var translationFields = $('#' + tabName).find('[data-translation-progress-item=\"true\"]');\n var completedTranslations = $('#' + tabName).find('[data-translation-present=\"true\"]');\n\n currentTab.find('span').text(completedTranslations.length + '/' + translationFields.length);\n })\n }\n}","/*\nNOTE: this is copied & adapted from BL8's checkbox_submit.js in order to have\nit accessible in a BL7-based spotlight. Once we drop support for BL7, this file\ncan be deleted and we can change visibility_toggle.es6 to import CheckboxSubmit\nfrom Blacklight.\n\nSee https://github.com/projectblacklight/blacklight/blob/main/app/javascript/blacklight/checkbox_submit.js\n*/\nexport default class CheckboxSubmit {\n constructor(form) {\n this.form = form\n }\n\n async clicked(evt) {\n this.spanTarget.innerHTML = this.form.getAttribute('data-inprogress')\n this.labelTarget.setAttribute('disabled', 'disabled');\n this.checkboxTarget.setAttribute('disabled', 'disabled');\n const csrfMeta = document.querySelector('meta[name=csrf-token]')\n const response = await fetch(this.formTarget.getAttribute('action'), {\n body: new FormData(this.formTarget),\n method: this.formTarget.getAttribute('method').toUpperCase(),\n headers: {\n 'Accept': 'application/json',\n 'X-Requested-With': 'XMLHttpRequest',\n 'X-CSRF-Token': csrfMeta ? csrfMeta.content : ''\n }\n })\n this.labelTarget.removeAttribute('disabled')\n this.checkboxTarget.removeAttribute('disabled')\n if (response.ok) {\n this.updateStateFor(!this.checked)\n // Not used for our case in Spotlight (visibility toggle)\n // const json = await response.json()\n // document.querySelector('[data-role=bookmark-counter]').innerHTML = json.bookmarks.count\n } else {\n alert('Error')\n }\n }\n\n get checked() {\n return (this.form.querySelectorAll('input[name=_method][value=delete]').length != 0)\n }\n\n get formTarget() {\n return this.form\n }\n\n get labelTarget() {\n return this.form.querySelector('[data-checkboxsubmit-target=\"label\"]')\n }\n\n get checkboxTarget() {\n return this.form.querySelector('[data-checkboxsubmit-target=\"checkbox\"]')\n }\n\n get spanTarget() {\n return this.form.querySelector('[data-checkboxsubmit-target=\"span\"]')\n }\n\n updateStateFor(state) {\n this.checkboxTarget.checked = state\n\n if (state) {\n this.labelTarget.classList.add('checked')\n //Set the Rails hidden field that fakes an HTTP verb\n //properly for current state action.\n this.formTarget.querySelector('input[name=_method]').value = 'delete'\n this.spanTarget.innerHTML = this.form.getAttribute('data-present')\n } else {\n this.labelTarget.classList.remove('checked')\n this.formTarget.querySelector('input[name=_method]').value = 'put'\n this.spanTarget.innerHTML = this.form.getAttribute('data-absent')\n }\n }\n}\n","// Visibility toggle for items in an exhibit, based on Blacklight's bookmark toggle\n// See: https://github.com/projectblacklight/blacklight/blob/main/app/javascript/blacklight/bookmark_toggle.js\n\nimport CheckboxSubmit from 'spotlight/admin/checkbox_submit'\n\nconst VisibilityToggle = (e) => {\n if (e.target.matches('[data-checkboxsubmit-target=\"checkbox\"]')) {\n const form = e.target.closest('form')\n if (form) {\n new CheckboxSubmit(form).clicked(e)\n\n // Add/remove the \"private\" label to the document row when visibility is toggled\n const docRow = form.closest('tr')\n if (docRow) docRow.classList.toggle('blacklight-private')\n }\n }\n}\n\nVisibilityToggle.selector = 'form.visibility-toggle'\n\ndocument.addEventListener('click', VisibilityToggle)\n\nexport default VisibilityToggle\n","export default class {\n connect() {\n var container;\n function edit_user(event) {\n event.preventDefault();\n $(this).closest('tr').hide();\n const id = $(this).attr('data-target') || $(this).attr('data-bs-target');\n const edit_view = $(\"[data-edit-for='\"+id+\"']\", container).show();\n $.each(edit_view.find('input[type=\"text\"], select'), function() {\n // Cache original values incase editing is canceled\n $(this).data('orig', $(this).val());\n });\n }\n\n function cancel_edit(event) {\n event.preventDefault();\n const id = $(this).closest('tr').attr('data-edit-for');\n const edit_view = $(\"[data-edit-for='\"+id+\"']\", container).hide();\n clear_errors(edit_view);\n rollback_changes(edit_view);\n $(\"[data-show-for='\"+id+\"']\", container).show();\n }\n\n function clear_errors(element) {\n element.find('.has-error')\n .removeClass('has-error')\n .find('.form-text')\n .remove(); // Remove the error messages\n }\n\n function rollback_changes(element) {\n $.each(element.find('input[type=\"text\"], select'), function() {\n $(this).val($(this).data('orig')).trigger('change');\n });\n }\n\n function destroy_user(event) {\n const id = $(this).attr('data-target') || $(this).attr('data-bs-target');\n $(\"[data-destroy-for='\"+id+\"']\", container).val('1');\n }\n\n function new_user(event) {\n event.preventDefault();\n const edit_view = $(\"[data-edit-for='new']\", container).show();\n $.each(edit_view.find('input[type=\"text\"], select'), function() {\n // Cache original values incase editing is canceled\n $(this).data('orig', $(this).val());\n });\n }\n\n function open_errors() {\n const edit_row = container.find('.has-error').closest('[data-edit-for]');\n edit_row.show();\n // The following row has the controls, so show it too.\n edit_row.next().show();\n }\n\n $('.edit_exhibit, .admin-users').each(function() {\n\n container = $(this);\n $('[data-edit-for]', container).hide();\n open_errors();\n $(\"[data-behavior='edit-user']\", container).on('click', edit_user);\n $(\"[data-behavior='cancel-edit']\", container).on('click', cancel_edit);\n $(\"[data-behavior='destroy-user']\", container).on('click', destroy_user);\n $(\"[data-behavior='new-user']\", container).on('click', new_user);\n })\n }\n}\n","(function ($){\n SirTrevor.BlockMixins.Autocompleteable = {\n mixinName: \"Autocompleteable\",\n preload: true,\n\n initializeAutocompleteable: function() {\n this.on(\"onRender\", this.addAutocompletetoSirTrevorForm);\n\n if (this['autocomplete_url'] === undefined) {\n this.autocomplete_url = function() { return $('form[data-autocomplete-url]').data('autocomplete-url').replace(\"%25QUERY\", \"%QUERY\"); };\n }\n\n if (this['transform_autocomplete_results'] === undefined) {\n this.transform_autocomplete_results = (val) => val\n }\n\n if (this['autocomplete_control'] === undefined) {\n this.autocomplete_control = function() { return `` };\n }\n\n if (this['bloodhoundOptions'] === undefined) {\n this.bloodhoundOptions = function() {\n return {\n remote: {\n url: this.autocomplete_url(),\n filter: this.transform_autocomplete_results\n }\n };\n };\n }\n },\n\n addAutocompletetoSirTrevorForm: function() {\n $('[data-twitter-typeahead]', this.inner).spotlightSearchTypeAhead({bloodhound: this.bloodhound(), template: this.autocomplete_template}).on('typeahead:selected typeahead:autocompleted', this.autocompletedHandler()).on( 'focus', function() {\n if($(this).val() === '') {\n $(this).data().ttTypeahead.input.trigger('queryChanged', '');\n }\n });\n },\n\n autocompletedHandler: function(e, data) {\n var context = this;\n\n return function(e, data) {\n $(this).typeahead(\"val\", \"\");\n $(this).val(\"\");\n\n context.createItemPanel($.extend(data, {display: \"true\"}));\n }\n },\n\n bloodhound: function() {\n var block = this;\n var results = new Bloodhound(Object.assign({\n datumTokenizer: function(d) {\n return Bloodhound.tokenizers.whitespace(d.title);\n },\n queryTokenizer: Bloodhound.tokenizers.whitespace,\n limit: 100,\n }, block.bloodhoundOptions()));\n results.initialize();\n return results;\n },\n },\n\n\n SirTrevor.Block.prototype.availableMixins.push(\"autocompleteable\");\n})(jQuery);\n","(function ($){\n SirTrevor.BlockMixins.Formable = {\n mixinName: \"Formable\",\n preload: true,\n\n initializeFormable: function() {\n\n if (this['afterLoadData'] === undefined) {\n this['afterLoadData'] = function(data) { };\n }\n },\n\n formId: function(id) {\n return this.blockID + \"_\" + id;\n },\n\n _serializeData: function() {\n\n var data = $(\":input,textarea,select\", this.inner).not(':input:radio').serializeJSON();\n\n $(':input:radio:checked', this.inner).each(function(index, input) {\n var key = $(input).data('key') || input.getAttribute('name');\n\n if (!key.match(\"\\\\[\")) {\n data[key] = $(input).val();\n }\n });\n\n /* Simple to start. Add conditions later */\n if (this.hasTextBlock()) {\n data.text = this.getTextBlockHTML();\n data.format = 'html';\n if (data.text && data.text.length > 0 && this.options.convertToMarkdown) {\n data.text = stToMarkdown(data.text, this.type);\n data.format = 'markdown';\n }\n }\n\n return data;\n },\n\n loadData: function(data){\n if (this.hasTextBlock()) {\n if (data.text && data.text.length > 0 && this.options.convertFromMarkdown && data.format !== \"html\") {\n this.setTextBlockHTML(SirTrevor.toHTML(data.text, this.type));\n } else {\n this.setTextBlockHTML(data.text);\n }\n }\n this.loadFormDataByKey(data);\n this.afterLoadData(data);\n },\n\n loadFormDataByKey: function(data) {\n $(':input', this.inner).not('button,:input[type=hidden]').each(function(index, input) {\n var key = $(input).data('key') || input.getAttribute('name');\n\n if (key) {\n\n if (key.match(\"\\\\[\\\\]$\")) {\n key = key.replace(\"[]\", \"\");\n }\n\n // by wrapping it in an array, this'll \"just work\" for radio and checkbox fields too\n var input_data = data[key];\n\n if (!(input_data instanceof Array)) {\n input_data = [input_data];\n }\n $(this).val(input_data);\n }\n });\n },\n },\n\n\n SirTrevor.Block.prototype.availableMixins.push(\"formable\");\n})(jQuery);\n","(function ($){\n SirTrevor.BlockMixins.Plustextable = {\n mixinName: \"Textable\",\n preload: true,\n\n initializeTextable: function() {\n if (this['formId'] === undefined) {\n this.withMixin(SirTrevor.BlockMixins.Formable);\n }\n \n if (this['show_heading'] === undefined) {\n this.show_heading = true;\n }\n },\n \n align_key:\"text-align\",\n text_key:\"item-text\",\n heading_key: \"title\",\n \n text_area: function() { \n return `\n \n
\n
\n
\n
${i18n.t(\"blocks:textable:align:title\")}
\n
\n
\n
\n
\n
\n
\n
`\n },\n \n heading: function() {\n if(this.show_heading) {\n return `\n \n \n
`\n } else {\n return \"\";\n }\n },\n };\n \n\n SirTrevor.Block.prototype.availableMixins.push(\"plustextable\");\n})(jQuery);\n","import Core from 'spotlight/core'\n(function ($){\n Core.Block = SirTrevor.Block.extend({\n scribeOptions: {\n allowBlockElements: true,\n tags: { p: true }\n },\n formable: true,\n editorHTML: function() {\n return '';\n },\n beforeBlockRender: function() {\n this.availableMixins.forEach(function(mixin) {\n if (this[mixin] && SirTrevor.BlockMixins[this.capitalize(mixin)].preload) {\n this.withMixin(SirTrevor.BlockMixins[this.capitalize(mixin)]);\n }\n }, this);\n },\n $instance: function() { return $('#' + this.instanceID); },\n capitalize: function(string) {\n return string.charAt(0).toUpperCase() + string.substring(1).toLowerCase();\n }\n })\n})(jQuery);\n","import Core from 'spotlight/core'\nimport SpotlightNestable from 'spotlight/admin/spotlight_nestable'\n\nCore.Block.Resources = (function(){\n\n return Core.Block.extend({\n type: \"resources\",\n formable: true,\n autocompleteable: true,\n show_heading: true,\n show_alt_text: true,\n\n title: function() { return i18n.t(\"blocks:\" + this.type + \":title\"); },\n description: function() { return i18n.t(\"blocks:\" + this.type + \":description\"); },\n\n icon_name: \"resources\",\n blockGroup: function() { return i18n.t(\"blocks:group:items\") },\n\n primary_field_key: \"primary-caption-field\",\n show_primary_field_key: \"show-primary-caption\",\n secondary_field_key: \"secondary-caption-field\",\n show_secondary_field_key: \"show-secondary-caption\",\n\n display_checkbox: \"display-checkbox\",\n decorative_checkbox: \"decorative-checkbox\",\n alt_text_textarea: \"alt-text-textarea\",\n\n globalIndex: 0,\n\n _itemPanelIiifFields: function(index, data) {\n return [];\n },\n\n _altTextFieldsHTML: function(index, data) {\n if (this.show_alt_text) {\n return this.altTextHTML(index, data);\n }\n return \"\";\n },\n\n _itemPanel: function(data) {\n var index = \"item_\" + this.globalIndex++;\n var checked;\n if (data.display == \"true\") {\n checked = \"checked='checked'\"\n } else {\n checked = \"\";\n }\n var resource_id = data.slug || data.id;\n var markup = `\n \n \n \n ${this._itemPanelIiifFields(index, data)}\n \n \n
${i18n.t(\"blocks:resources:panel:drag\")}
\n \n
\n \n `\n\n const panel = $(markup);\n var context = this;\n\n $('.remove a', panel).on('click', function(e) {\n e.preventDefault();\n $(this).closest('.field').remove();\n context.afterPanelDelete();\n\n });\n\n this.afterPanelRender(data, panel);\n\n return panel;\n },\n\n afterPanelRender: function(data, panel) {\n\n },\n\n afterPanelDelete: function() {\n\n },\n\n createItemPanel: function(data) {\n var panel = this._itemPanel(data);\n this.attachAltTextHandlers(panel);\n $(panel).appendTo($('.panels > ol', this.inner));\n $('[data-behavior=\"nestable\"]', this.inner).trigger('change');\n },\n\n item_options: function() { return \"\"; },\n\n content: function() {\n var templates = [this.items_selector()];\n if (this.plustextable) {\n templates.push(this.text_area());\n }\n return templates.join(\"
\\n\");\n },\n\n items_selector: function() { return [\n '',\n '
',\n '
',\n this.item_options(),\n '
',\n '
'].join(\"\\n\")\n },\n\n editorHTML: function() {\n return `\n \n ${this.content()}\n
`\n },\n\n _altTextData: function(data) {\n const isDecorative = data.decorative;\n const altText = isDecorative ? '' : (data.alt_text || '');\n const altTextBackup = data.alt_text_backup || '';\n const placeholderAttr = isDecorative ? '' : `placeholder=\"${i18n.t(\"blocks:resources:alt_text:placeholder\")}\"`;\n const disabledAttr = isDecorative ? 'disabled' : '';\n\n return { isDecorative, altText, altTextBackup, placeholderAttr, disabledAttr };\n },\n\n altTextHTML: function(index, data) {\n const { isDecorative, altText, altTextBackup, placeholderAttr, disabledAttr } = this._altTextData(data);\n return `\n
\n
\n
\n \n \n
\n
\n
\n \n \n
\n
`\n },\n\n attachAltTextHandlers: function(panel) {\n if (this.show_alt_text) {\n const decorativeCheckbox = $('input[name$=\"[decorative]\"]', panel);\n const altTextInput = $('textarea[name$=\"[alt_text]\"]', panel);\n const altTextBackupInput = $('input[name$=\"[alt_text_backup]\"]', panel);\n\n decorativeCheckbox.on('change', function() {\n const isDecorative = this.checked;\n if (isDecorative) {\n altTextBackupInput.val(altTextInput.val());\n altTextInput.val('');\n } else {\n altTextInput.val(altTextBackupInput.val());\n }\n altTextInput\n .prop('disabled', isDecorative)\n .attr('placeholder', isDecorative ? '' : i18n.t(\"blocks:resources:alt_text:placeholder\"));\n });\n\n altTextInput.on('input', function() {\n $(this).data('lastValue', $(this).val());\n });\n }\n },\n\n onBlockRender: function() {\n SpotlightNestable.init($('[data-behavior=\"nestable\"]', this.inner));\n\n $('[data-input-select-target]', this.inner).selectRelatedInput();\n },\n\n afterLoadData: function(data) {\n var context = this;\n $.each(Object.keys(data.item || {}).map(function(k) { return data.item[k]}).sort(function(a,b) { return a.weight - b.weight; }), function(index, item) {\n context.createItemPanel(item);\n });\n },\n });\n\n})();\n","import Core from 'spotlight/core'\n\nSirTrevor.Blocks.Browse = (function(){\n\n return Core.Block.Resources.extend({\n type: \"browse\",\n\n icon_name: \"browse\",\n\n autocomplete_url: function() {\n return $(this.inner).closest('form[data-autocomplete-exhibit-searches-path]').data('autocomplete-exhibit-searches-path').replace(\"%25QUERY\", \"%QUERY\");\n },\n\n autocomplete_template: function(obj) {\n const thumbnail = obj.thumbnail_image_url ? `` : ''\n return `${thumbnail}\n ${obj.full_title}
${obj.description}
`\n },\n\n bloodhoundOptions: function() {\n return {\n prefetch: {\n url: this.autocomplete_url(),\n ttl: 0\n }\n };\n },\n\n _itemPanel: function(data) {\n var index = \"item_\" + this.globalIndex++;\n var checked;\n if (data.display == \"true\") {\n checked = \"checked='checked'\"\n } else {\n checked = \"\";\n }\n var resource_id = data.slug || data.id;\n var markup = `\n \n \n \n \n \n
${i18n.t(\"blocks:resources:panel:drag\")}
\n \n
\n `\n\n var panel = $(markup);\n var context = this;\n\n $('.remove a', panel).on('click', function(e) {\n e.preventDefault();\n $(this).closest('.field').remove();\n context.afterPanelDelete();\n\n });\n\n this.afterPanelRender(data, panel);\n\n return panel;\n },\n\n item_options: function() { return `\n `\n },\n });\n\n})();\n","/*\n Sir Trevor BrowseGroupCategories\n*/\nimport Core from 'spotlight/core'\n\nSirTrevor.Blocks.BrowseGroupCategories = (function(){\n\n return Core.Block.Resources.extend({\n type: \"browse_group_categories\",\n icon_name: \"browse\",\n bloodhoundOptions: function() {\n var that = this;\n return {\n prefetch: {\n url: this.autocomplete_url(),\n ttl: 0,\n filter: function(response) {\n // Let the dom know that the response has been returned\n $(that.inner).attr('data-browse-groups-fetched', true);\n return response;\n }\n }\n };\n },\n\n autocomplete_control: function() {\n return ``\n },\n autocomplete_template: function(obj) {\n return `\n ${obj.title}
`\n },\n\n autocomplete_url: function() { return $(this.inner).closest('form[data-autocomplete-exhibit-browse-groups-path]').data('autocomplete-exhibit-browse-groups-path').replace(\"%25QUERY\", \"%QUERY\"); },\n _itemPanel: function(data) {\n var index = \"item_\" + this.globalIndex++;\n var checked;\n if (data.display == \"true\") {\n checked = \"checked='checked'\"\n } else {\n checked = \"\";\n }\n var resource_id = data.slug || data.id;\n var markup = `\n \n \n \n \n \n
${i18n.t(\"blocks:resources:panel:drag\")}
\n
\n
\n `\n\n const panel = $(markup);\n var context = this;\n\n $('a[data-item-grid-panel-remove]', panel).on('click', function(e) {\n e.preventDefault();\n $(this).closest('.field').remove();\n context.afterPanelDelete();\n\n });\n\n this.afterPanelRender(data, panel);\n\n return panel;\n },\n\n item_options: function() { return `\n '`\n },\n });\n})();\n","/*\n Sir Trevor ItemText Block.\n This block takes an ID,\n fetches the record from solr,\n displays the image, title, \n and any provided text\n and displays them.\n*/\n\nSirTrevor.Blocks.Iframe = (function(){\n\n return SirTrevor.Block.extend({\n type: \"Iframe\",\n formable: true,\n \n title: function() { return i18n.t('blocks:iframe:title'); },\n description: function() { return i18n.t('blocks:iframe:description'); },\n\n icon_name: \"iframe\",\n \n editorHTML: function() {\n return `\n \n \n
`;\n }\n });\n})();","SirTrevor.Blocks.LinkToSearch = (function(){\n\n return SirTrevor.Blocks.Browse.extend({\n\n type: \"link_to_search\",\n\n icon_name: 'search_results',\n\n searches_key: \"slug\",\n view_key: \"view\",\n plustextable: false,\n\n });\n})();\n","/*\n Sir Trevor ItemText Block.\n This block takes an ID,\n fetches the record from solr,\n displays the image, title, \n and any provided text\n and displays them.\n*/\nimport Core from 'spotlight/core'\n\nSirTrevor.Blocks.Oembed = (function(){\n\n return Core.Block.extend({\n plustextable: true,\n\n id_key:\"url\",\n\n type: \"oembed\",\n \n title: function() { return i18n.t('blocks:oembed:title'); },\n description: function() { return i18n.t('blocks:oembed:description'); },\n\n icon_name: \"oembed\",\n show_heading: false,\n\n editorHTML: function () {\n return ``\n }\n });\n})();","import Core from 'spotlight/core'\n\nSirTrevor.Blocks.FeaturedPages = (function(){\n\n return Core.Block.Resources.extend({\n type: \"featured_pages\",\n\n icon_name: \"pages\",\n\n autocomplete_url: function() { return $(this.inner).closest('form[data-autocomplete-exhibit-pages-path]').data('autocomplete-exhibit-pages-path').replace(\"%25QUERY\", \"%QUERY\"); },\n autocomplete_template: function(obj) {\n const thumbnail = obj.thumbnail_image_url ? `` : ''\n return `${thumbnail}\n ${obj.title}
${obj.description}
`\n },\n bloodhoundOptions: function() {\n return {\n prefetch: {\n url: this.autocomplete_url(),\n ttl: 0\n }\n };\n }\n });\n\n})();\n","/*\n Sir Trevor ItemText Block.\n This block takes an ID,\n fetches the record from solr,\n displays the image, title, \n and any provided text\n and displays them.\n*/\n\nSirTrevor.Blocks.Rule = (function(){\n\n return SirTrevor.Block.extend({\n type: \"rule\",\n \n title: function() { return i18n.t('blocks:rule:title'); },\n\n icon_name: \"rule\",\n \n editorHTML: function() {\n return '
'\n }\n });\n})();","//= require spotlight/admin/blocks/browse_block\n\nSirTrevor.Blocks.SearchResults = (function(){\n\n return SirTrevor.Blocks.Browse.extend({\n\n type: \"search_results\",\n\n icon_name: 'search_results',\n\n searches_key: \"slug\",\n view_key: \"view\",\n plustextable: false,\n\n content: function() {\n return this.items_selector()\n },\n\n item_options: function() {\n var block = this;\n var fields = $('[data-blacklight-configuration-search-views]').data('blacklight-configuration-search-views');\n\n return $.map(fields, function(field) {\n return `\n \n
`\n }).join(\"\\n\");\n },\n\n afterPanelRender: function(data, panel) {\n $(this.inner).find('.item-input-field').attr(\"disabled\", \"disabled\");\n },\n\n afterPanelDelete: function() {\n $(this.inner).find('.item-input-field').removeAttr(\"disabled\");\n },\n\n });\n})();\n","import Iiif from 'spotlight/admin/iiif'\nimport Core from 'spotlight/core'\n\nSirTrevor.Blocks.SolrDocumentsBase = (function(){\n\n return Core.Block.Resources.extend({\n plustextable: true,\n autocomplete_url: function() { return this.$instance().closest('form[data-autocomplete-exhibit-catalog-path]').data('autocomplete-exhibit-catalog-path').replace(\"%25QUERY\", \"%QUERY\"); },\n autocomplete_template: function(obj) {\n const thumbnail = obj.thumbnail ? `` : ''\n return `${thumbnail}\n ${obj.title}
${obj.description}
`\n },\n transform_autocomplete_results: function(response) {\n return $.map(response['docs'], function(doc) {\n return doc;\n })\n },\n\n caption_option_values: function() {\n var fields = $('[data-blacklight-configuration-index-fields]').data('blacklight-configuration-index-fields');\n\n return $.map(fields, function(field) {\n return $('').val(field.key).text(field.label)[0].outerHTML;\n }).join(\"\\n\");\n },\n\n item_options: function() { return this.caption_options(); },\n\n caption_options: function() { return `\n \n \n \n \n \n
\n \n \n \n \n \n
\n `},\n\n // Sets the first version of the IIIF information from autocomplete data.\n _itemPanelIiifFields: function(index, autocomplete_data) {\n return [\n // '',\n // for legacy compatiblity:\n '',\n '',\n '',\n '',\n '',\n '',\n ].join(\"\\n\");\n },\n // Overwrites the hidden inputs from _itemPanelIiifFields with data from the\n // manifest. Called by afterPanelRender - the manifest_data here is built\n // from canvases in the manifest, transformed by spotlight/admin/iiif.js in\n // the #images method.\n setIiifFields: function(panel, manifest_data, initialize) {\n var legacyThumbnailField = $(panel).find('[name$=\"[thumbnail_image_url]\"]')\n var legacyFullField = $(panel).find('[name$=\"[full_image_url]\"]')\n\n if (initialize && legacyThumbnailField.val().length > 0) {\n return;\n }\n\n legacyThumbnailField.val(\"\");\n legacyFullField.val(\"\");\n $(panel).find('[name$=\"[iiif_image_id]\"]').val(manifest_data.imageId);\n $(panel).find('[name$=\"[iiif_tilesource]\"]').val(manifest_data.tilesource);\n $(panel).find('[name$=\"[iiif_manifest_url]\"]').val(manifest_data.manifest);\n $(panel).find('[name$=\"[iiif_canvas_id]\"]').val(manifest_data.canvasId);\n $(panel).find('img.img-thumbnail').attr('src', manifest_data.thumbnail_image_url || manifest_data.tilesource.replace(\"/info.json\", \"/full/100,100/0/default.jpg\"));\n },\n afterPanelRender: function(data, panel) {\n var context = this;\n var manifestUrl = data.iiif_manifest || data.iiif_manifest_url;\n\n if (!manifestUrl) {\n $(panel).find('[name$=\"[thumbnail_image_url]\"]').val(data.thumbnail_image_url || data.thumbnail);\n $(panel).find('[name$=\"[full_image_url]\"]').val(data.full_image_url);\n\n return;\n }\n\n $.ajax(manifestUrl).done(\n function(manifest) {\n var iiifManifest = new Iiif(manifestUrl, manifest);\n\n var thumbs = iiifManifest.imagesArray();\n\n if (!data.iiif_image_id) {\n context.setIiifFields(panel, thumbs[0], !!data.iiif_manifest_url);\n }\n\n\n if(thumbs.length > 1) {\n panel.multiImageSelector(thumbs, function(selectorImage) {\n context.setIiifFields(panel, selectorImage, false);\n }, data.iiif_image_id);\n }\n }\n );\n }\n });\n\n})();\n","//= require spotlight/admin/blocks/solr_documents_base_block\n\nSirTrevor.Blocks.SolrDocuments = (function(){\n\n return SirTrevor.Blocks.SolrDocumentsBase.extend({\n type: \"solr_documents\",\n\n icon_name: \"items\",\n\n item_options: function() { return this.caption_options() + this.zpr_option(); },\n\n zpr_option: function() {\n return `\n \n \n \n \n
\n `\n },\n\n zpr_key: 'zpr_link'\n });\n\n})();\n","//= require spotlight/admin/blocks/solr_documents_base_block\n\nSirTrevor.Blocks.SolrDocumentsCarousel = (function(){\n\n return SirTrevor.Blocks.SolrDocumentsBase.extend({\n plustextable: false,\n type: \"solr_documents_carousel\",\n\n icon_name: \"item_carousel\",\n\n auto_play_images_key: \"auto-play-images\",\n auto_play_images_interval_key: \"auto-play-images-interval\",\n max_height_key: \"max-height\",\n\n carouselCycleTimesInSeconds: {\n values: [ 3, 5, 8, 12, 20 ],\n selected: 5\n },\n\n carouselMaxHeights: {\n values: { 'Small': 'small', 'Medium': 'medium', 'Large': 'large' },\n selected: 'Medium'\n },\n\n item_options: function() {\n return `${this.caption_options()}\n \n \n \n \n \n
\n \n
\n ${this.addCarouselMaxHeightOptions(this.carouselMaxHeights)}\n
`\n },\n\n addCarouselCycleOptions: function(options) {\n var html = '';\n\n $.each(options.values, function(index, interval) {\n var selected = (interval === options.selected) ? 'selected' : '',\n intervalInMilliSeconds = parseInt(interval, 10) * 1000;\n\n html += '';\n });\n\n return html;\n },\n\n addCarouselMaxHeightOptions: function(options) {\n var html = '',\n _this = this;\n\n $.each(options.values, function(size, px) {\n var checked = (size === options.selected) ? 'checked' : '',\n id = _this.formId(_this.max_height_key)\n\n html += '';\n html += '';\n });\n\n return html;\n },\n\n afterPreviewLoad: function(options) {\n $(this.inner).find('.carousel').carousel();\n\n // the bootstrap carousel only initializes data-bs-slide widgets on page load, so we need\n // to initialize them ourselves..\n var clickHandler = function (e) {\n var href\n var $this = $(this)\n var $target = $($this.attr('data-target') || $this.attr('data-bs-target') || (href = $this.attr('href')) && href.replace(/.*(?=#[^\\s]+$)/, '')) // strip for ie7\n if (!$target.hasClass('carousel')) return\n var options = $.extend({}, $target.data(), $this.data())\n var slideIndex = $this.attr('data-slide-to') || $this.attr('data-bs-slide-to')\n if (slideIndex) options.interval = false\n\n $.fn.carousel.call($target, options)\n\n if (slideIndex) {\n $target.data('bs.carousel').to(slideIndex)\n }\n\n e.preventDefault()\n }\n\n $(this.inner).find('.carousel')\n .on('click.bs.carousel.data-api', '[data-slide], [data-bs-slide]', clickHandler)\n .on('click.bs.carousel.data-api', '[data-slide-to], [data-bs-slide-to]', clickHandler)\n }\n\n });\n\n})();\n","//= require spotlight/admin/blocks/solr_documents_base_block\n\nSirTrevor.Blocks.SolrDocumentsEmbed = (function(){\n\n return SirTrevor.Blocks.SolrDocumentsBase.extend({\n type: \"solr_documents_embed\",\n show_alt_text: false,\n icon_name: \"item_embed\",\n\n item_options: function() { return \"\" },\n\n afterPreviewLoad: function(options) {\n $(this.inner).find('picture[data-openseadragon]').openseadragon();\n }\n });\n\n})();\n","//= require spotlight/admin/blocks/solr_documents_base_block\n\nSirTrevor.Blocks.SolrDocumentsFeatures = (function(){\n\n return SirTrevor.Blocks.SolrDocumentsBase.extend({\n plustextable: false,\n type: \"solr_documents_features\",\n\n icon_name: \"item_features\",\n\n afterPreviewLoad: function(options) {\n $(this.inner).find('.carousel').carousel();\n\n // the bootstrap carousel only initializes data-bs-slide widgets on page load, so we need\n // to initialize them ourselves..\n var clickHandler = function (e) {\n var href\n var $this = $(this)\n var $target = $($this.attr('data-target') || $this.attr('data-bs-target') || (href = $this.attr('href')) && href.replace(/.*(?=#[^\\s]+$)/, '')) // strip for ie7\n if (!$target.hasClass('carousel')) return\n var options = $.extend({}, $target.data(), $this.data())\n var slideIndex = $this.attr('data-slide-to') || $this.attr('data-bs-slide-to')\n if (slideIndex) options.interval = false\n\n $.fn.carousel.call($target, options)\n\n if (slideIndex) {\n $target.data('bs.carousel').to(slideIndex)\n }\n\n e.preventDefault()\n }\n\n $(this.inner).find('.carousel')\n .on('click.bs.carousel.data-api', '[data-slide], [data-bs-slide]', clickHandler)\n .on('click.bs.carousel.data-api', '[data-slide-to], [data-bs-slide-to]', clickHandler)\n }\n\n });\n\n})();\n","//= require spotlight/admin/blocks/solr_documents_base_block\n\nSirTrevor.Blocks.SolrDocumentsGrid = (function(){\n\n return SirTrevor.Blocks.SolrDocumentsBase.extend({\n type: \"solr_documents_grid\",\n\n icon_name: \"item_grid\",\n\n\n item_options: function() { return \"\" }\n });\n\n})();\n","import SpotlightNestable from 'spotlight/admin/spotlight_nestable'\nimport Core from 'spotlight/core'\n\nSirTrevor.Blocks.UploadedItems = (function(){\n return Core.Block.Resources.extend({\n plustextable: true,\n uploadable: true,\n autocompleteable: false,\n\n id_key: 'file',\n\n type: 'uploaded_items',\n\n icon_name: 'items',\n\n blockGroup: 'undefined',\n\n // Clear out the default Uploadable upload options\n // since we will be using our own custom controls\n upload_options: { html: '' },\n\n fileInput: function() { return $(this.inner).find('input[type=\"file\"]'); },\n\n onBlockRender: function(){\n SpotlightNestable.init($(this.inner).find('[data-behavior=\"nestable\"]'));\n\n this.fileInput().on('change', (function(ev) {\n this.onDrop(ev.currentTarget);\n }).bind(this));\n },\n\n onDrop: function(transferData){\n var file = transferData.files[0],\n urlAPI = (typeof URL !== \"undefined\") ? URL : (typeof webkitURL !== \"undefined\") ? webkitURL : null;\n\n // Handle one upload at a time\n if (/image/.test(file.type)) {\n this.loading();\n\n this.uploader(\n file,\n function(data) {\n this.createItemPanel(data);\n this.fileInput().val('');\n this.ready();\n },\n function(error) {\n this.addMessage(i18n.t('blocks:image:upload_error'));\n this.ready();\n }\n );\n }\n },\n\n title: function() { return i18n.t('blocks:uploaded_items:title'); },\n description: function() { return i18n.t('blocks:uploaded_items:description'); },\n\n globalIndex: 0,\n\n _itemPanel: function(data) {\n var index = \"file_\" + this.globalIndex++;\n var checked = 'checked=\"checked\"';\n\n if (data.display == 'false') {\n checked = '';\n }\n\n var dataId = data.id || data.uid;\n var dataTitle = data.title || data.name;\n var dataUrl = data.url || data.file.url;\n\n var markup = `\n \n \n \n \n \n \n
${i18n.t(\"blocks:resources:panel:drag\")}
\n \n `\n\n const panel = $(markup);\n panel.find('[data-field=\"caption\"]').val(data.caption);\n panel.find('[data-field=\"link\"]').val(data.link);\n var context = this;\n\n $('.remove a', panel).on('click', function(e) {\n e.preventDefault();\n $(this).closest('.field').remove();\n context.afterPanelDelete();\n });\n\n this.afterPanelRender(data, panel);\n\n return panel;\n },\n\n editorHTML: function() {\n return `
`\n },\n\n altTextHTML: function(index, data) {\n const { isDecorative, altText, altTextBackup, placeholderAttr, disabledAttr } = this._altTextData(data);\n return `\n
\n
\n
\n
\n \n \n
\n
\n
\n
\n
`\n },\n\n zpr_key: 'zpr_link'\n });\n})();\n","import Core from 'spotlight/core'\n\n(function() {\n var BLOCK_REPLACER_CONTROL_TEMPLATE = function(block) {\n var el = document.createElement('button');\n el.className = \"st-block-controls__button\";\n el.setAttribute('data-type', block.type);\n el.type = \"button\";\n\n var img = document.createElement('svg');\n img.className = \"st-icon\";\n img.setAttribute('role', 'img');\n\n var use = document.createElement('use');\n use.setAttributeNS('https://www.w3.org/1999/xlink', 'href', SirTrevor.config.defaults.iconUrl + \"#\" + block.icon_name);\n img.appendChild(use);\n el.appendChild(img);\n el.appendChild(document.createTextNode(block.title()));\n\n return el.outerHTML;\n };\n\n function generateBlocksHTML(Blocks, availableTypes) {\n var groups = {};\n for(var i in availableTypes) {\n var type = availableTypes[i];\n if (Blocks.hasOwnProperty(type) && Blocks[type].prototype.toolbarEnabled) {\n var blockGroup;\n\n if ($.isFunction(Blocks[type].prototype.blockGroup)) {\n blockGroup = Blocks[type].prototype.blockGroup();\n } else {\n blockGroup = Blocks[type].prototype.blockGroup;\n }\n\n if (blockGroup == 'undefined' || blockGroup === undefined) {\n blockGroup = i18n.t(\"blocks:group:undefined\");\n }\n\n groups[blockGroup] = groups[blockGroup] || [];\n groups[blockGroup].push(BLOCK_REPLACER_CONTROL_TEMPLATE(Blocks[type].prototype));\n }\n }\n\n function generateBlock(groups, key) {\n var group = groups[key];\n var groupEl = $(\"
\");\n var buttons = group.reduce(function(memo, btn) {\n return memo += btn;\n }, \"\");\n groupEl.append(buttons);\n return groupEl[0].outerHTML;\n }\n\n var standardWidgets = generateBlock(groups, i18n.t(\"blocks:group:undefined\"));\n\n var exhibitWidgets = Object.keys(groups).map(function(key) {\n if (key !== i18n.t(\"blocks:group:undefined\")) {\n return generateBlock(groups, key);\n }\n }).filter(function (element) {\n return element != null;\n });\n\n var blocks = [standardWidgets].concat(exhibitWidgets).join(\"
\");\n return blocks;\n }\n\n function render(Blocks, availableTypes) {\n var el = document.createElement('div');\n el.className = \"st-block-controls__buttons\";\n el.innerHTML = generateBlocksHTML.apply(null, arguments);\n\n var elButtons = document.createElement('div');\n elButtons.className = \"spotlight-block-controls\";\n elButtons.appendChild(el);\n return elButtons;\n }\n\n Core.BlockControls = function() { };\n Core.BlockControls.create = function(editor) {\n // REFACTOR - should probably not know about blockManager\n var el = render(SirTrevor.Blocks, editor.blockManager.blockTypes);\n\n function hide() {\n var parent = el.parentNode;\n if (!parent) { return; }\n parent.removeChild(el);\n parent.classList.remove(\"st-block--controls-active\");\n return parent;\n }\n\n function destroy() {\n SirTrevor = null;\n el = null;\n }\n\n function insert(e) {\n e.stopPropagation();\n\n var parent = this.parentNode;\n if (!parent || hide() === parent) { return; }\n $('.st-block__inner', parent).after(el);\n parent.classList.add(\"st-block--controls-active\");\n }\n\n function replaceBlock() {\n SirTrevor.mediator.trigger(\n \"block:replace\", el.parentNode, this.getAttribute('data-type')\n );\n }\n\n $(editor.wrapper).delegate(\".st-block-replacer\", \"click\", insert);\n $(editor.wrapper).delegate(\".st-block-controls__button\", \"click\", insert);\n\n return {\n el: el,\n hide: hide,\n destroy: destroy\n };\n };\n})();\n","import Core from 'spotlight/core'\n\nCore.BlockLimits = function(editor) {\n this.editor = editor;\n};\n\nCore.BlockLimits.prototype.enforceLimits = function(editor) {\n this.addEditorCallbacks(editor);\n this.checkGlobalBlockTypeLimit()();\n};\n\nCore.BlockLimits.prototype.addEditorCallbacks = function(editor) {\n SirTrevor.EventBus.on('block:create:new', this.checkBlockTypeLimitOnAdd());\n SirTrevor.EventBus.on('block:remove', this.checkGlobalBlockTypeLimit());\n};\n\nCore.BlockLimits.prototype.checkBlockTypeLimitOnAdd = function() {\n var editor = this.editor;\n\n return function(block) {\n var control = $(\".st-block-controls__button[data-type='\" + block.type + \"']\", editor.blockControls.el);\n\n control.prop(\"disabled\", !editor.blockManager.canCreateBlock(block.class()));\n };\n};\n\nCore.BlockLimits.prototype.checkGlobalBlockTypeLimit = function() {\n // we don't know what type of block was created or removed.. So, try them all.\n var editor = this.editor;\n\n return function() {\n $.each(editor.blockManager.blockTypes, function(i, type) {\n var block_type = SirTrevor.Blocks[type].prototype;\n\n var control = $(editor.blockControls.el).find(\".st-block-controls__button[data-type='\" + block_type.type + \"']\");\n control.prop(\"disabled\", !editor.blockManager.canCreateBlock(type));\n });\n };\n};\n","SirTrevor.Locales.en.blocks = $.extend(SirTrevor.Locales.en.blocks, {\n autocompleteable: {\n placeholder: \"Enter a title...\"\n },\n\n browse: {\n title: \"Browse Categories\",\n description: \"This widget highlights browse categories. Each highlighted category links to the corresponding browse category results page.\",\n item_counts: \"Include item counts?\"\n },\n\n browse_group_categories: {\n autocomplete: \"Enter a browse group title...\",\n title: \"Browse Group Categories\",\n description: \"This widget displays all browse categories associated with a selected browse group as a horizontally-scrolling row. Each selected browse group is displayed as a separate row. Each displayed category in a group links to the corresponding browse category results page.\",\n item_counts: \"Include category item counts?\"\n },\n\n link_to_search: {\n title: \"Saved Searches\",\n description: \"This widget highlights saved searches. Each highlighted saved search links to the search results page generated by the saved search parameters. Any saved search listed on the Curation > Browse categories page, whether published or not, can be highlighted as a saved search.\",\n item_counts: \"Include item counts?\"\n },\n\n iframe: {\n title: \"IFrame\",\n description: \"This widget embeds iframe-based embed code into pages\",\n placeholder: \"Enter embed code here. It should begin with e.g. '