mirror of
https://github.com/espocrm/espocrm.git
synced 2026-07-01 08:26:04 +00:00
469 lines
17 KiB
JavaScript
469 lines
17 KiB
JavaScript
/************************************************************************
|
|
* This file is part of EspoCRM.
|
|
*
|
|
* EspoCRM - Open Source CRM application.
|
|
* Copyright (C) 2014-2021 Yurii Kuznietsov, Taras Machyshyn, Oleksii Avramenko
|
|
* Website: https://www.espocrm.com
|
|
*
|
|
* EspoCRM is free software: you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation, either version 3 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* EspoCRM is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with EspoCRM. If not, see http://www.gnu.org/licenses/.
|
|
*
|
|
* The interactive user interfaces in modified source and object code versions
|
|
* of this program must display Appropriate Legal Notices, as required under
|
|
* Section 5 of the GNU General Public License version 3.
|
|
*
|
|
* In accordance with Section 7(b) of the GNU General Public License version 3,
|
|
* these Appropriate Legal Notices must retain the display of the "EspoCRM" word.
|
|
************************************************************************/
|
|
|
|
define('view-helper', ['lib!client/lib/purify.min.js'], function () {
|
|
|
|
var ViewHelper = function (options) {
|
|
this._registerHandlebarsHelpers();
|
|
|
|
this.mdBeforeList = [
|
|
{
|
|
regex: /\&\#x60;\&\#x60;\&\#x60;\n?([\s\S]*?)\&\#x60;\&\#x60;\&\#x60;/g,
|
|
value: function (s, string) {
|
|
return '<pre><code>' + string.replace(/\*/g, '*').replace(/\~/g, '~') + '</code></pre>';
|
|
}
|
|
},
|
|
{
|
|
regex: /\&\#x60;([\s\S]*?)\&\#x60;/g,
|
|
value: function (s, string) {
|
|
return '<code>' + string.replace(/\*/g, '*').replace(/\~/g, '~') + '</code>';
|
|
}
|
|
}
|
|
];
|
|
|
|
marked.setOptions({
|
|
breaks: true,
|
|
tables: false
|
|
});
|
|
|
|
DOMPurify.addHook('beforeSanitizeAttributes', function (node) {
|
|
if (node instanceof HTMLAnchorElement) {
|
|
if (node.getAttribute('target')) {
|
|
node.targetBlack = true;
|
|
} else {
|
|
node.targetBlack = false;
|
|
}
|
|
}
|
|
});
|
|
|
|
DOMPurify.addHook('afterSanitizeAttributes', function (node) {
|
|
if (node instanceof HTMLAnchorElement) {
|
|
if (node.targetBlack) {
|
|
node.setAttribute('target', '_blank');
|
|
node.setAttribute('rel', 'noopener noreferrer');
|
|
} else {
|
|
node.removeAttribute('rel');
|
|
}
|
|
}
|
|
});
|
|
}
|
|
|
|
_.extend(ViewHelper.prototype, {
|
|
|
|
layoutManager: null,
|
|
|
|
settings: null,
|
|
|
|
user: null,
|
|
|
|
preferences: null,
|
|
|
|
language: null,
|
|
|
|
getAppParam: function (name) {
|
|
return (this.appParams || {})[name];
|
|
},
|
|
|
|
stripTags: function (text) {
|
|
text = text || '';
|
|
if (typeof text === 'string' || text instanceof String) {
|
|
return text.replace(/<\/?[^>]+(>|$)/g, '');
|
|
}
|
|
return text;
|
|
},
|
|
|
|
escapeString: function (text) {
|
|
return Handlebars.Utils.escapeExpression(text);
|
|
},
|
|
|
|
getAvatarHtml: function (id, size, width, additionalClassName) {
|
|
if (this.config.get('avatarsDisabled')) {
|
|
return '';
|
|
}
|
|
|
|
var t;
|
|
var cache = this.cache;
|
|
if (cache) {
|
|
t = cache.get('app', 'timestamp');
|
|
} else {
|
|
t = Date.now();
|
|
}
|
|
|
|
var basePath = this.basePath || '';
|
|
var size = size || 'small';
|
|
|
|
var width = (width || 16).toString();
|
|
|
|
var className = 'avatar';
|
|
if (additionalClassName) {
|
|
className += ' ' + additionalClassName;
|
|
}
|
|
|
|
return '<img class="'+className+'" width="'+width+'" src="'+basePath+
|
|
'?entryPoint=avatar&size='+size+'&id=' + id + '&t='+t+'">';
|
|
},
|
|
|
|
_registerHandlebarsHelpers: function () {
|
|
var self = this;
|
|
|
|
Handlebars.registerHelper('img', function (img) {
|
|
return new Handlebars.SafeString("<img src=\"img/" + img + "\"></img>");
|
|
});
|
|
|
|
Handlebars.registerHelper('prop', function (object, name) {
|
|
if (name in object) {
|
|
return object[name];
|
|
}
|
|
});
|
|
|
|
Handlebars.registerHelper('var', function (name, context, options) {
|
|
return new Handlebars.SafeString(context[name]);
|
|
});
|
|
|
|
Handlebars.registerHelper('concat', function (left, right) {
|
|
return left + right;
|
|
});
|
|
|
|
Handlebars.registerHelper('ifEqual', function (left, right, options) {
|
|
if (left == right) {
|
|
return options.fn(this);
|
|
}
|
|
return options.inverse(this);
|
|
});
|
|
|
|
Handlebars.registerHelper('ifNotEqual', function (left, right, options) {
|
|
if (left != right) {
|
|
return options.fn(this);
|
|
}
|
|
return options.inverse(this);
|
|
});
|
|
|
|
Handlebars.registerHelper('ifPropEquals', function (object, property, value, options) {
|
|
if (object[property] == value) {
|
|
return options.fn(this);
|
|
}
|
|
return options.inverse(this);
|
|
});
|
|
|
|
Handlebars.registerHelper('ifAttrEquals', function (model, attr, value, options) {
|
|
if (model.get(attr) == value) {
|
|
return options.fn(this);
|
|
}
|
|
return options.inverse(this);
|
|
});
|
|
|
|
Handlebars.registerHelper('ifAttrNotEmpty', function (model, attr, options) {
|
|
var value = model.get(attr);
|
|
if (value !== null && typeof value !== 'undefined') {
|
|
return options.fn(this);
|
|
}
|
|
return options.inverse(this);
|
|
});
|
|
|
|
Handlebars.registerHelper('get', function (model, name) {
|
|
return model.get(name);
|
|
});
|
|
|
|
Handlebars.registerHelper('length', function (arr) {
|
|
return arr.length;
|
|
});
|
|
|
|
Handlebars.registerHelper('translate', function (name, options) {
|
|
var scope = options.hash.scope || null;
|
|
var category = options.hash.category || null;
|
|
if (name === 'null') {
|
|
return '';
|
|
}
|
|
return self.language.translate(name, category, scope);
|
|
});
|
|
|
|
Handlebars.registerHelper('button', function (name, options) {
|
|
var style = options.hash.style || 'default';
|
|
var scope = options.hash.scope || null;
|
|
var label = options.hash.label || name;
|
|
|
|
var html = options.hash.html || options.hash.text || self.language.translate(label, 'labels', scope);
|
|
return new Handlebars.SafeString('<button class="btn btn-'+style+' action'+ (options.hash.hidden ? ' hidden' : '')+'" data-action="'+name+'" type="button">'+html+'</button>');
|
|
});
|
|
|
|
Handlebars.registerHelper('hyphen', function (string) {
|
|
return Espo.Utils.convert(string, 'c-h');
|
|
});
|
|
|
|
Handlebars.registerHelper('toDom', function (string) {
|
|
return Espo.Utils.toDom(string);
|
|
});
|
|
|
|
Handlebars.registerHelper('breaklines', function (text) {
|
|
text = Handlebars.Utils.escapeExpression(text || '');
|
|
text = text.replace(/(\r\n|\n|\r)/gm, '<br>');
|
|
return new Handlebars.SafeString(text);
|
|
});
|
|
|
|
Handlebars.registerHelper('complexText', function (text, options) {
|
|
return this.transfromMarkdownText(text, options.hash);
|
|
}.bind(this));
|
|
|
|
Handlebars.registerHelper('translateOption', function (name, options) {
|
|
var scope = options.hash.scope || null;
|
|
var field = options.hash.field || null;
|
|
if (!field) {
|
|
return '';
|
|
}
|
|
var translationHash = options.hash.translatedOptions || null;
|
|
if (translationHash === null) {
|
|
translationHash = self.language.translate(field, 'options', scope) || {};
|
|
if (typeof translationHash !== 'object') {
|
|
translationHash = {};
|
|
}
|
|
}
|
|
return translationHash[name] || name;
|
|
});
|
|
|
|
Handlebars.registerHelper('options', function (list, value, options) {
|
|
if (typeof value === 'undefined') {
|
|
value = false;
|
|
}
|
|
list = list || [];
|
|
var html = '';
|
|
|
|
var multiple = (Object.prototype.toString.call(value) === '[object Array]');
|
|
var checkOption = function (name) {
|
|
if (multiple) {
|
|
return value.indexOf(name) !== -1;
|
|
} else {
|
|
return value === name;
|
|
}
|
|
};
|
|
|
|
options.hash = options.hash || {};
|
|
|
|
var scope = options.hash.scope || false;
|
|
var category = options.hash.category || false;
|
|
var field = options.hash.field || false;
|
|
|
|
if (!multiple && options.hash.includeMissingOption && (value || value === '')) {
|
|
if (!~list.indexOf(value)) {
|
|
list = Espo.Utils.clone(list);
|
|
list.push(value);
|
|
}
|
|
}
|
|
|
|
var translationHash = options.hash.translationHash || options.hash.translatedOptions || null;
|
|
|
|
if (translationHash === null) {
|
|
if (!category && field) {
|
|
translationHash = self.language.translate(field, 'options', scope) || {};
|
|
if (typeof translationHash !== 'object') {
|
|
translationHash = {};
|
|
}
|
|
} else {
|
|
translationHash = {};
|
|
}
|
|
}
|
|
var translate = function (name) {
|
|
if (!category) {
|
|
return translationHash[name] || name;
|
|
}
|
|
return self.language.translate(name, category, scope);
|
|
};
|
|
|
|
for (var key in list) {
|
|
var keyVal = list[key];
|
|
var label = translate(list[key]);
|
|
keyVal = self.escapeString(keyVal);
|
|
label = self.escapeString(label);
|
|
html += "<option value=\"" + keyVal + "\" " + (checkOption(list[key]) ? 'selected' : '') + ">" + label + "</option>"
|
|
}
|
|
return new Handlebars.SafeString(html);
|
|
});
|
|
|
|
Handlebars.registerHelper('basePath', function () {
|
|
return self.basePath || '';
|
|
});
|
|
},
|
|
|
|
transfromMarkdownInlineText: function (text) {
|
|
return this.transfromMarkdownText(text, {inline: true});
|
|
},
|
|
|
|
transfromMarkdownText: function (text, options) {
|
|
text = text || '';
|
|
|
|
text = Handlebars.Utils.escapeExpression(text).replace(/>+/g, '>');
|
|
|
|
this.mdBeforeList.forEach(function (item) {
|
|
text = text.replace(item.regex, item.value);
|
|
});
|
|
|
|
options = options || {};
|
|
|
|
if (options.inline) {
|
|
text = marked.inlineLexer(text, []);
|
|
} else {
|
|
text = marked(text);
|
|
}
|
|
|
|
text = DOMPurify.sanitize(text).toString();
|
|
|
|
if (options.linksInNewTab) {
|
|
text = text.replace(/<a href=/gm, '<a target="_blank" rel="noopener noreferrer" href=');
|
|
}
|
|
|
|
text = text.replace(/<a href="mailto:(.*)"/gm, '<a href="javascript:" data-email-address="$1" data-action="mailTo"');
|
|
|
|
return new Handlebars.SafeString(text);
|
|
},
|
|
|
|
getScopeColorIconHtml: function (scope, noWhiteSpace, additionalClassName) {
|
|
if (this.config.get('scopeColorsDisabled') || this.preferences.get('scopeColorsDisabled')) {
|
|
return '';
|
|
}
|
|
|
|
var color = this.metadata.get(['clientDefs', scope, 'color']);
|
|
var html = '';
|
|
|
|
if (color) {
|
|
var $span = $('<span class="color-icon fas fa-square-full">');
|
|
$span.css('color', color);
|
|
if (additionalClassName) {
|
|
$span.addClass(additionalClassName);
|
|
}
|
|
html = $span.get(0).outerHTML;
|
|
}
|
|
|
|
if (!noWhiteSpace) {
|
|
if (html) html += ' ';
|
|
}
|
|
|
|
return html;
|
|
},
|
|
|
|
sanitizeHtml: function (text, options) {
|
|
return DOMPurify.sanitize(text, options);
|
|
},
|
|
|
|
moderateSanitizeHtml: function (value) {
|
|
value = value || '';
|
|
value = value.replace(/<[\/]{0,1}(base)[^><]*>/gi, '');
|
|
value = value.replace(/<[\/]{0,1}(object)[^><]*>/gi, '');
|
|
value = value.replace(/<[\/]{0,1}(embed)[^><]*>/gi, '');
|
|
value = value.replace(/<[\/]{0,1}(applet)[^><]*>/gi, '');
|
|
value = value.replace(/<[\/]{0,1}(iframe)[^><]*>/gi, '');
|
|
value = value.replace(/<[\/]{0,1}(script)[^><]*>/gi, '');
|
|
value = value.replace(/<[^><]*([^a-z]{1}on[a-z]+)=[^><]*>/gi, function (match) {
|
|
return match.replace(/[^a-z]{1}on[a-z]+=/gi, ' data-handler-stripped=');
|
|
});
|
|
|
|
value = this.stripEventHandlersInHtml(value);
|
|
|
|
value = value.replace(/href=" *javascript\:(.*?)"/gi, function(m, $1) {
|
|
return 'removed=""';
|
|
});
|
|
value = value.replace(/href=' *javascript\:(.*?)'/gi, function(m, $1) {
|
|
return 'removed=""';
|
|
});
|
|
value = value.replace(/src=" *javascript\:(.*?)"/gi, function(m, $1) {
|
|
return 'removed=""';
|
|
});
|
|
value = value.replace(/src=' *javascript\:(.*?)'/gi, function(m, $1) {
|
|
return 'removed=""';
|
|
});
|
|
|
|
return value;
|
|
},
|
|
|
|
stripEventHandlersInHtml: function (html) {
|
|
function stripHTML(){
|
|
html = html.slice(0, strip) + html.slice(j);
|
|
j = strip;
|
|
strip = false;
|
|
}
|
|
function isValidTagChar(str) {
|
|
return str.match(/[a-z?\\\/!]/i);
|
|
}
|
|
var strip = false;
|
|
var lastQuote = false;
|
|
for (var i = 0; i<html.length; i++){
|
|
if (html[i] === "<" && html[i+1] && isValidTagChar(html[i+1])) {
|
|
i++;
|
|
for (var j = i; j<html.length; j++){
|
|
if (!lastQuote && html[j] === ">"){
|
|
if (strip) {
|
|
stripHTML();
|
|
}
|
|
i = j;
|
|
break;
|
|
}
|
|
if (lastQuote === html[j]){
|
|
lastQuote = false;
|
|
continue;
|
|
}
|
|
if (!lastQuote && html[j-1] === "=" && (html[j] === "'" || html[j] === '"')){
|
|
lastQuote = html[j];
|
|
}
|
|
if (!lastQuote && html[j-2] === " " && html[j-1] === "o" && html[j] === "n"){
|
|
strip = j-2;
|
|
}
|
|
if (strip && html[j] === " " && !lastQuote){
|
|
stripHTML();
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return html;
|
|
},
|
|
|
|
calculateContentContainerHeight: function ($el) {
|
|
var smallScreenWidth = this.themeManager.getParam('screenWidthXs');
|
|
var $window = $(window);
|
|
|
|
var footerHeight = $('#footer').height() || 26;
|
|
var top = 0;
|
|
var element = $el.get(0);
|
|
|
|
if (element) {
|
|
top = element.getBoundingClientRect().top;
|
|
|
|
if ($window.width() < smallScreenWidth) {
|
|
var $navbarCollapse = $('#navbar .navbar-body');
|
|
if ($navbarCollapse.hasClass('in') || $navbarCollapse.hasClass('collapsing')) {
|
|
top -= $navbarCollapse.height();
|
|
}
|
|
}
|
|
}
|
|
|
|
var spaceHeight = top + footerHeight;
|
|
|
|
return $window.height() - spaceHeight - 20;
|
|
},
|
|
});
|
|
|
|
return ViewHelper;
|
|
});
|