2015-10-28 21:51:30 +00:00
|
|
|
/*****************************************************************************
|
|
|
|
* Open MCT Web, Copyright (c) 2014-2015, United States Government
|
|
|
|
* as represented by the Administrator of the National Aeronautics and Space
|
|
|
|
* Administration. All rights reserved.
|
|
|
|
*
|
|
|
|
* Open MCT Web is licensed under the Apache License, Version 2.0 (the
|
|
|
|
* "License"); you may not use this file except in compliance with the License.
|
|
|
|
* You may obtain a copy of the License at
|
|
|
|
* http://www.apache.org/licenses/LICENSE-2.0.
|
|
|
|
*
|
|
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
|
|
* distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
|
|
|
|
* WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
|
|
|
|
* License for the specific language governing permissions and limitations
|
|
|
|
* under the License.
|
|
|
|
*
|
|
|
|
* Open MCT Web includes source code licensed under additional open source
|
|
|
|
* licenses. See the Open Source Licenses file (LICENSES.md) included with
|
|
|
|
* this source code distribution or the Licensing information page available
|
|
|
|
* at runtime from the About dialog for additional information.
|
|
|
|
*****************************************************************************/
|
|
|
|
/*global define,Promise*/
|
|
|
|
|
|
|
|
define(
|
|
|
|
[],
|
|
|
|
function () {
|
|
|
|
"use strict";
|
|
|
|
|
|
|
|
/**
|
2015-10-29 14:58:32 +00:00
|
|
|
* The `templateLinker` service is intended for internal use by
|
|
|
|
* the `mct-include` and `mct-representation` directives. It is
|
|
|
|
* used to support common behavior of directives; specifically,
|
|
|
|
* loading templates and inserting them into a specified element,
|
|
|
|
* and/or removing that element from the DOM when there is no
|
|
|
|
* template to populate it with.
|
2015-10-28 21:51:30 +00:00
|
|
|
*
|
2015-10-30 20:34:13 +00:00
|
|
|
* @param {Function} $templateRequest Angular's `$templateRequest`
|
|
|
|
* service
|
|
|
|
* @param $sce Angular's `$sce` service
|
2015-10-28 21:51:30 +00:00
|
|
|
* @param {Function} $compile Angular's `$compile` service
|
|
|
|
* @param $log Angular's `$log` service
|
2015-10-29 14:58:32 +00:00
|
|
|
* @private
|
2015-10-28 21:51:30 +00:00
|
|
|
*/
|
2015-10-30 19:22:43 +00:00
|
|
|
function TemplateLinker($templateRequest, $sce, $compile, $log) {
|
|
|
|
this.$templateRequest = $templateRequest;
|
|
|
|
this.$sce = $sce;
|
2015-10-28 21:51:30 +00:00
|
|
|
this.$compile = $compile;
|
|
|
|
this.$log = $log;
|
|
|
|
}
|
|
|
|
|
2015-10-30 19:38:32 +00:00
|
|
|
/**
|
|
|
|
* Load a template from the given URL. This request will be handled
|
|
|
|
* via `$templateRequest` to ensure caching et cetera.
|
|
|
|
* @param {string} the URL for the template
|
|
|
|
* @returns {Promise.<string>} a promise for the HTML content of
|
|
|
|
* the template
|
|
|
|
* @private
|
|
|
|
*/
|
2015-10-28 21:51:30 +00:00
|
|
|
TemplateLinker.prototype.load = function (templateUrl) {
|
2015-10-30 19:22:43 +00:00
|
|
|
return this.$templateRequest(
|
|
|
|
this.$sce.trustAsResourceUrl(templateUrl),
|
|
|
|
false
|
|
|
|
);
|
2015-10-28 21:51:30 +00:00
|
|
|
};
|
|
|
|
|
2015-10-30 18:02:13 +00:00
|
|
|
/**
|
|
|
|
* Get a path to a template from an extension definition fo
|
|
|
|
* a template, representation, or view.
|
|
|
|
* @param {TemplateDefinition} extensionDefinition the definition
|
|
|
|
* of the template/representation/view to resolve
|
|
|
|
*/
|
|
|
|
TemplateLinker.prototype.getPath = function (extensionDefinition) {
|
|
|
|
return [
|
|
|
|
extensionDefinition.bundle.path,
|
|
|
|
extensionDefinition.bundle.resources,
|
|
|
|
extensionDefinition.templateUrl
|
|
|
|
].join('/');
|
|
|
|
};
|
|
|
|
|
2015-10-28 21:51:30 +00:00
|
|
|
/**
|
2015-10-29 14:58:32 +00:00
|
|
|
* Populate the given element with templates, within the given scope;
|
|
|
|
* intended to support the `link` function of the supported directives.
|
|
|
|
*
|
|
|
|
* @param {Scope} scope the Angular scope to use when rendering
|
|
|
|
* templates
|
|
|
|
* @param element the jqLite-wrapped element into which templates
|
|
|
|
* should be inserted
|
2015-10-28 21:51:30 +00:00
|
|
|
* @returns {Function} a function which can be called with a template
|
|
|
|
* URL to switch templates, or `undefined` to remove.
|
|
|
|
*/
|
2015-10-29 15:17:25 +00:00
|
|
|
TemplateLinker.prototype.link = function (scope, element, templateUrl) {
|
2015-10-28 22:29:57 +00:00
|
|
|
var activeElement = element,
|
2015-10-28 22:17:36 +00:00
|
|
|
activeTemplateUrl,
|
2015-10-28 22:29:57 +00:00
|
|
|
comment = this.$compile('<!-- hidden mct element -->')(scope),
|
2015-10-30 22:06:51 +00:00
|
|
|
activeScope,
|
2015-10-28 21:51:30 +00:00
|
|
|
self = this;
|
|
|
|
|
2015-10-30 22:06:51 +00:00
|
|
|
function destroyScope() {
|
|
|
|
if (activeScope) {
|
|
|
|
activeScope.$destroy();
|
|
|
|
activeScope = undefined;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-10-28 21:51:30 +00:00
|
|
|
function removeElement() {
|
2015-10-28 22:29:57 +00:00
|
|
|
if (activeElement !== comment) {
|
2015-10-30 22:06:51 +00:00
|
|
|
destroyScope();
|
2015-10-28 22:29:57 +00:00
|
|
|
activeElement.replaceWith(comment);
|
|
|
|
activeElement = comment;
|
2015-10-28 21:51:30 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-10-28 22:32:50 +00:00
|
|
|
function addElement() {
|
|
|
|
if (activeElement !== element) {
|
|
|
|
activeElement.replaceWith(element);
|
|
|
|
activeElement = element;
|
|
|
|
activeElement.empty();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
function populateElement(template) {
|
2015-10-30 22:06:51 +00:00
|
|
|
destroyScope();
|
|
|
|
activeScope = scope.$new(false);
|
2015-10-30 19:22:43 +00:00
|
|
|
element.empty();
|
2015-10-30 22:06:51 +00:00
|
|
|
element.append(self.$compile(template)(activeScope));
|
2015-10-28 21:51:30 +00:00
|
|
|
}
|
|
|
|
|
2015-10-30 19:22:43 +00:00
|
|
|
function badTemplate(templateUrl) {
|
|
|
|
self.$log.warn("Couldn't load template at " + templateUrl);
|
|
|
|
removeElement();
|
2015-10-28 21:51:30 +00:00
|
|
|
}
|
|
|
|
|
2015-10-28 22:17:36 +00:00
|
|
|
function changeTemplate(templateUrl) {
|
2015-10-30 22:32:20 +00:00
|
|
|
if (templateUrl) {
|
2015-10-30 23:18:01 +00:00
|
|
|
destroyScope();
|
2015-10-30 22:32:20 +00:00
|
|
|
addElement();
|
|
|
|
self.load(templateUrl).then(function (template) {
|
|
|
|
// Avoid race conditions
|
|
|
|
if (templateUrl === activeTemplateUrl) {
|
|
|
|
populateElement(template);
|
|
|
|
}
|
|
|
|
}, function () {
|
|
|
|
badTemplate(templateUrl);
|
|
|
|
});
|
|
|
|
} else {
|
|
|
|
removeElement();
|
2015-10-28 21:51:30 +00:00
|
|
|
}
|
2015-10-30 22:32:20 +00:00
|
|
|
activeTemplateUrl = templateUrl;
|
2015-10-28 22:17:36 +00:00
|
|
|
}
|
|
|
|
|
2015-10-29 15:17:25 +00:00
|
|
|
if (templateUrl) {
|
|
|
|
changeTemplate(templateUrl);
|
|
|
|
} else {
|
|
|
|
removeElement();
|
|
|
|
}
|
2015-10-28 22:29:57 +00:00
|
|
|
|
2015-10-28 22:17:36 +00:00
|
|
|
return changeTemplate;
|
2015-10-28 21:51:30 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
return TemplateLinker;
|
|
|
|
}
|
|
|
|
);
|
|
|
|
|