"use strict";
MetacatUI.recaptchaURL =
"https://www.google.com/recaptcha/api/js/recaptcha_ajax";
if (MetacatUI.mapKey) {
var gmapsURL =
"https://maps.googleapis.com/maps/api/js?v=3&libraries=places&key=" +
MetacatUI.mapKey;
define("gmaps", ["async!" + gmapsURL], function () {
return google.maps;
});
} else {
define("gmaps", null);
}
MetacatUI.d3URL = "../components/d3.v3.min";
/* Configure the app to use requirejs, and map dependency aliases to their
directory location (.js is ommitted). Shim libraries that don't natively
support requirejs. */
require.config({
baseUrl: MetacatUI.root + "/js/",
waitSeconds: 180, //wait 3 minutes before throwing a timeout error
map: MetacatUI.themeMap,
urlArgs:
"v=" + (MetacatUI.AppConfig.cachebuster || MetacatUI.metacatUIVersion),
paths: {
jquery: MetacatUI.root + "/components/jquery-1.9.1.min",
jqueryui: MetacatUI.root + "/components/jquery-ui.min",
jqueryform: MetacatUI.root + "/components/jquery.form",
underscore: MetacatUI.root + "/components/underscore-min",
backbone: MetacatUI.root + "/components/backbone-min",
localforage: MetacatUI.root + "/components/localforage.min",
bootstrap: MetacatUI.root + "/components/bootstrap.min",
text: MetacatUI.root + "/components/require-text",
jws: MetacatUI.root + "/components/jws-3.2.min",
jsrasign: MetacatUI.root + "/components/jsrsasign-4.9.0.min",
async: MetacatUI.root + "/components/async",
recaptcha: [MetacatUI.recaptchaURL, "scripts/placeholder"],
nGeohash: MetacatUI.root + "/components/geohash/main",
fancybox: MetacatUI.root + "/components/fancybox/jquery.fancybox.pack", //v. 2.1.5
annotator: MetacatUI.root + "/components/annotator/v1.2.10/annotator-full",
bioportal: MetacatUI.root + "/components/bioportal/jquery.ncbo.tree-2.0.2",
clipboard: MetacatUI.root + "/components/clipboard.min",
uuid: MetacatUI.root + "/components/uuid",
md5: MetacatUI.root + "/components/md5",
rdflib: MetacatUI.root + "/components/rdflib.min",
x2js: MetacatUI.root + "/components/xml2json",
he: MetacatUI.root + "/components/he",
citation: MetacatUI.root + "/components/citation.min",
promise: MetacatUI.root + "/components/es6-promise.min",
metacatuiConnectors: MetacatUI.root + "/js/connectors/Filters-Search",
// showdown + extensions (used in the MarkdownView to convert markdown to html)
showdown: MetacatUI.root + "/components/showdown/showdown.min",
showdownHighlight:
MetacatUI.root +
"/components/showdown/extensions/showdown-highlight/showdown-highlight",
highlight:
MetacatUI.root +
"/components/showdown/extensions/showdown-highlight/highlight.pack",
showdownFootnotes:
MetacatUI.root + "/components/showdown/extensions/showdown-footnotes",
showdownBootstrap:
MetacatUI.root + "/components/showdown/extensions/showdown-bootstrap",
showdownDocbook:
MetacatUI.root + "/components/showdown/extensions/showdown-docbook",
showdownKatex:
MetacatUI.root +
"/components/showdown/extensions/showdown-katex/showdown-katex.min",
showdownCitation:
MetacatUI.root +
"/components/showdown/extensions/showdown-citation/showdown-citation",
showdownImages:
MetacatUI.root + "/components/showdown/extensions/showdown-images",
showdownXssFilter:
MetacatUI.root +
"/components/showdown/extensions/showdown-xss-filter/showdown-xss-filter",
xss:
MetacatUI.root +
"/components/showdown/extensions/showdown-xss-filter/xss.min",
showdownHtags:
MetacatUI.root + "/components/showdown/extensions/showdown-htags",
// woofmark - markdown editor
woofmark: MetacatUI.root + "/components/woofmark.min",
// drop zone creates drag and drop areas
Dropzone: MetacatUI.root + "/components/dropzone-amd-module",
// Packages that convert between json data to markdown table
markdownTableFromJson:
MetacatUI.root + "/components/markdown-table-from-json.min",
markdownTableToJson: MetacatUI.root + "/components/markdown-table-to-json",
// Polyfill required for using dropzone with older browsers
corejs: MetacatUI.root + "/components/core-js",
// Custom semantic bundle used for searchable multi-select dropdown component
semantic: `${MetacatUI.root}/js/common/Semantic`,
// To make elements drag and drop, sortable
sortable: MetacatUI.root + "/components/sortable.min",
//Cesium
cesium:
"https://cesium.com/downloads/cesiumjs/releases/1.91/Build/Cesium/Cesium",
//Have a null fallback for our d3 components for browsers that don't support SVG
d3: MetacatUI.d3URL,
LineChart: ["views/LineChartView", null],
BarChart: ["views/BarChartView", null],
CircleBadge: ["views/CircleBadgeView", null],
DonutChart: ["views/DonutChartView", null],
MetricsChart: ["views/MetricsChartView", null],
},
shim: {
/* used for libraries without native AMD support */
underscore: {
exports: "_",
},
backbone: {
deps: ["underscore", "jquery"],
exports: "Backbone",
},
bootstrap: {
deps: ["jquery"],
exports: "Bootstrap",
},
annotator: {
exports: "Annotator",
},
bioportal: {
exports: "Bioportal",
},
jws: {
exports: "JWS",
deps: ["jsrasign"],
},
nGeohash: {
exports: "geohash",
},
fancybox: {
deps: ["jquery"],
},
uuid: {
exports: "uuid",
},
rdflib: {
exports: "rdf",
},
xss: {
exports: "filterXSS",
},
citation: {
exports: "citationRequire",
},
promise: {
exports: "Promise",
},
metacatuiConnectors: {
exports: "FiltersSearchConnector",
},
},
});
MetacatUI.appModel = MetacatUI.appModel || {};
MetacatUI.appView = MetacatUI.appView || {};
MetacatUI.uiRouter = MetacatUI.uiRouter || {};
MetacatUI.appSearchResults = MetacatUI.appSearchResults || {};
MetacatUI.appSearchModel = MetacatUI.appSearchModel || {};
/**
* @name MetacatUI.rootDataPackage
* @type {string}
* @description The top-level {@link DataPackage} that is currently being viewed or edited in MetacatUI.
*/
MetacatUI.rootDataPackage = MetacatUI.rootDataPackage || {};
MetacatUI.statsModel = MetacatUI.statsModel || {};
MetacatUI.mapModel = MetacatUI.mapModel || {};
MetacatUI.appLookupModel = MetacatUI.appLookupModel || {};
MetacatUI.nodeModel = MetacatUI.nodeModel || {};
MetacatUI.appUserModel = MetacatUI.appUserModel || {};
MetacatUI.analytics = MetacatUI.analytics || {};
/* Setup the application scaffolding first */
require(["bootstrap", "views/AppView", "models/AppModel"], function (
Bootstrap,
AppView,
AppModel,
) {
"use strict";
// Create an AppModel, which controls the global app configuration and app states
// To be compatible with MetacatUI 2.11.X and earlier, we need to set the metacat context attribute here.
// This supports the old way tof configuring the app via the index.html file.
// As of MetacatUI 2.12.0, it is recommended that you configure MetacatUI via an AppConfig file.
MetacatUI.appModel = new AppModel({
context: MetacatUI.AppConfig.metacatContext,
});
//Check for custom settings in the theme config file
if (typeof MetacatUI.customAppConfig == "function")
MetacatUI.customAppConfig();
/* Now require the rest of the libraries for the application */
require([
"underscore",
"backbone",
"routers/router",
"collections/SolrResults",
"models/Search",
"models/Stats",
"models/Map",
"models/LookupModel",
"models/NodeModel",
"models/UserModel",
"models/DataONEObject",
"collections/DataPackage",
], function (
_,
Backbone,
UIRouter,
SolrResultList,
Search,
Stats,
MapModel,
LookupModel,
NodeModel,
UserModel,
DataONEObject,
DataPackage,
) {
"use strict";
//Create all the other models and collections first
MetacatUI.appSearchResults = new SolrResultList([], {});
MetacatUI.appSearchModel = new Search();
MetacatUI.statsModel = new Stats();
MetacatUI.mapModel =
typeof customMapModelOptions == "object"
? new MapModel(customMapModelOptions)
: new MapModel();
MetacatUI.appLookupModel = new LookupModel();
MetacatUI.nodeModel = new NodeModel();
MetacatUI.appUserModel = new UserModel();
require(["models/analytics/GoogleAnalytics"], function (Analytics) {
MetacatUI.analytics = new Analytics();
});
/* Create a general event dispatcher to enable
communication across app components
*/
MetacatUI.eventDispatcher = _.clone(Backbone.Events);
//Load the App View now
MetacatUI.appView = new AppView();
MetacatUI.appView.render();
// Initialize routing and start Backbone.history()
(function () {
/**
* Backbone.routeNotFound
*
* Simple plugin that listens for false returns on Backbone.history.loadURL and fires an event
* to let the application know that no routes matched.
*
* @author STRML
*/
var oldLoadUrl = Backbone.History.prototype.loadUrl;
_.extend(Backbone.History.prototype, {
/*
* Override loadUrl & watch return value. Trigger event if no route was matched.
* @extends Backbone.History
* @return {Boolean} True if a route was matched
*/
loadUrl: function (fragment) {
if (!this.matchRoot()) return false;
fragment = this.fragment = this.getFragment(fragment);
var match = _.some(this.handlers, function (handler) {
if (handler.route.test(fragment)) {
handler.callback(fragment);
return true;
}
});
if (!match) this.trigger("routeNotFound");
return match;
},
matchRoot: function () {
var path = this.decodeFragment(this.location.pathname);
var rootPath = path.slice(0, this.root.length - 1) + "/";
return rootPath === this.root;
},
decodeFragment: function (fragment) {
return decodeURI(fragment.replace(/%25/g, "%2525"));
},
});
}).call(this);
//Make the router and begin the Backbone history
//The router will figure out which view to load first based on window location
MetacatUI.uiRouter = new UIRouter();
//Take the protocol and origin out of the root URL when sending it to Backbone.history.
// The root URL sent to Backbone.history should be either `/` or `/directory/...`
var historyRoot = MetacatUI.root;
//If there is a protocol
if (historyRoot.indexOf("://") > -1) {
//Get the substring after the ``://``
historyRoot = historyRoot.substring(historyRoot.indexOf("://") + 3);
//If there is no `/`, this must be the root directory
if (historyRoot.indexOf("/") == -1) historyRoot = "/";
//Otherwise get the substring after the first /
else historyRoot = historyRoot.substring(historyRoot.indexOf("/"));
}
//If there are no colons, periods, or slashes, this is a directory name
else if (
historyRoot.indexOf(":") == -1 &&
historyRoot.indexOf(".") == -1 &&
historyRoot.indexOf("/") == -1
) {
//So the root is a leading slash and the directory name
historyRoot = "/" + historyRoot;
}
//If there is a slash, get the path name starting with the slash
else if (historyRoot.indexOf("/") > -1) {
historyRoot = historyRoot.substring(historyRoot.indexOf("/"));
}
//All other strings are the root directory
else {
historyRoot = "/";
}
Backbone.history.start({
pushState: true,
root: historyRoot,
});
$(document).on("click", "a:not([data-toggle],[target])", function (evt) {
// Don't hijack the event if the user had Control or Command held down
if (evt.ctrlKey || evt.metaKey) {
return;
}
var href = { prop: $(this).prop("href"), attr: $(this).attr("href") };
// Stop if the click happened on an a w/o an href
// This is kind of a weird edge case where. This could be removed if
// we remove these instances from the codebase
if (
typeof href === "undefined" ||
typeof href.attr === "undefined" ||
href.attr === ""
) {
return;
}
//Don't route to URLs with the DataONE API, which are sometimes proxied
// via Apache ProxyPass so start with the MetacatUI origin
if (
href.attr.indexOf("/cn/v2/") > 0 ||
href.attr.indexOf("/mn/v2/") > 0
) {
return;
}
var root =
location.protocol +
"//" +
location.host +
Backbone.history.options.root;
// Remove the MetacatUI (plus a trailing /) from the value in the 'href'
// attribute of the clicked element so Backbone.history.navigate works.
// Note that a RegExp was used here to anchor the .replace call to the
// front of the string so that this code works when MetacatUI.root is "".
var route = href.attr.replace(new RegExp("^" + MetacatUI.root + "/"), "");
// Catch routes hrefs that start with # and don't do anything with them
if (href.attr.indexOf("#") == 0) {
return;
}
//If the URL is not a route defined in the app router, then follow the link
//If the URL is not at the MetacatUI root, then follow the link
if (
href.prop &&
href.prop.slice(0, root.length) === root &&
_.contains(
MetacatUI.uiRouter.getRouteNames(),
MetacatUI.uiRouter.getRouteName(route),
)
) {
evt.preventDefault();
Backbone.history.navigate(route, true);
}
});
MetacatUI.appModel.trigger("appInitialized");
});
});