yunkong2.admin/www/js/adapter-settings.js
2018-10-13 17:56:46 +08:00

2057 lines
79 KiB
JavaScript

var path = location.pathname;
var parts = path.split('/');
parts.splice(-3);
if (location.pathname.match(/^\/admin\//)) {
parts = [];
}
var systemConfig;
var socket = io.connect('/', {path: parts.join('/') + '/socket.io'});
var instance = window.location.search.slice(1);
var common = null; // common information of adapter
var host = null; // host object on which the adapter runs
var changed = false;
var certs = [];
var adapter = '';
var onChangeSupported = false;
var isMaterialize = false;
var ___onChange = null;
function preInit () {
'use strict';
var tmp = window.location.pathname.split('/');
adapter = tmp[tmp.length - 2];
var id = 'system.adapter.' + adapter + '.' + instance;
// Extend dictionary with standard words for adapter
if (typeof systemDictionary === 'undefined') systemDictionary = {};
systemDictionary.save = {"en": "Save", "cn": "保存"},
systemDictionary.saveclose = {"en": "Save and close", "cn": "保存并退出"},
systemDictionary.none = {"en": "none", "cn": "none"},
systemDictionary.nonerooms = {"en": "", "cn": ""},
systemDictionary.nonefunctions = {"en": "", "cn": ""},
systemDictionary.all = {"en": "all", "cn": "全部"},
systemDictionary['Device list'] = {"en": "Device list", "cn": "设备列表"},
systemDictionary['new device'] = {"en": "new device", "cn": "新设备"},
systemDictionary.edit = {"en": "edit", "cn": "编辑"},
systemDictionary.delete = {"en": "delete", "cn": "删除"},
systemDictionary.pair = {"en": "pair", "cn": "配对"},
systemDictionary.unpair = {"en": "unpair", "cn": "解除配对"},
systemDictionary.ok = {"en": "Ok", "cn": "确认"},
systemDictionary.cancel = {"en": "Cancel", "cn": "取消"},
systemDictionary.Message = {"en": "Message", "cn": "信息"},
systemDictionary.close = {"en": "Close", "cn": "关闭"},
systemDictionary.htooltip = {"en": "Click for help", "cn": "帮助"},
systemDictionary.saveConfig = {
"en": "Save configuration to file",
"cn": "导出配置信息"
};
systemDictionary.loadConfig = {
"en": "Load configuration from file",
"cn": "从JSON文件导入"
};
systemDictionary.otherConfig = {
"en": "Configuration from other adapter %s",
"cn": "Configuration from other adapter %s"
};
systemDictionary.invalidConfig = {
"en": "Invalid JSON file"
};
systemDictionary.configLoaded = {
"en": "Configuration was successfully loaded",
"cn": "Configuration was successfully loaded"
};
systemDictionary.maxTableRaw = {
"en": "Maximum number of allowed raws",
"cn": "Maximum number of allowed raws"
};
systemDictionary.maxTableRawInfo = {"en": "Warning", "cn": "提示"};
systemDictionary["Main settings"] = {
"en": "Main settings",
"cn": "基本设置"
};
systemDictionary["Let's Encrypt SSL"] = {
"en": "Let's Encrypt Certificates",
"cn": "Let's Encrypt Certificates"
};
systemDictionary["Please activate secure communication"] = {
"en": "Please activate secure communication",
"cn": "Please activate secure communication"
};
//socket.on('connection', function () {
loadSystemConfig(function () {
if (typeof translateAll === 'function') translateAll();
loadSettings(prepareTooltips);
});
//});
var $body = $('body');
$body.wrapInner('<div class="adapter-body"></div>');
/*$body.prepend('<div class="header ui-tabs-nav ui-widget ui-widget-header ui-corner-all" style="padding: 2px" >' +
'<button id="save" class="translateB">save</button>&nbsp;' +
'<button id="saveclose" class="translateB">saveclose</button>&nbsp;' +
'<button id="close" class="translateB" style="float: right;">cancel</button>&nbsp;' +
'</div>');
*/
$body.append(
'<div class="m"><nav class="dialog-config-buttons nav-wrapper footer">' +
' <a class="btn btn-active btn-save"><i class="material-icons left">save</i><span class="translate">save</span></a> ' +
' <a class="btn btn-save-close"><i class="material-icons left">save</i><i class="material-icons left">close</i><span class="translate">saveclose</span></a> ' +
' <a class="btn btn-cancel"><i class="material-icons left">close</i><span class="translate">close</span></a>' +
'</nav></div>');
var $navButtons = $('.dialog-config-buttons');
$navButtons.find('.btn-save').on('click', function () {
if (typeof save === 'undefined') {
alert('Please implement save function in your admin/index.html');
return;
}
save(function (obj, common, redirect) {
if (redirect && parent && parent.adapterRedirect) {
parent.adapterRedirect(redirect);
}
saveSettings(obj, common);
});
});
$navButtons.find('.btn-save-close').on('click', function () {
if (typeof save === 'undefined') {
alert('Please implement save function in your admin/index.html');
return;
}
save(function (obj, common, redirect) {
if (redirect && parent && parent.adapterRedirect) {
parent.adapterRedirect(redirect);
}
saveSettings(obj, common, function () {
// window.close();
if (typeof parent !== 'undefined' && parent && parent.$iframeDialog && typeof parent.$iframeDialog.close === 'function') {
parent.$iframeDialog.close();
}
});
});
});
$navButtons.find('.btn-cancel').on('click', function () {
if (typeof parent !== 'undefined' && parent && parent.$iframeDialog && typeof parent.$iframeDialog.close === 'function') {
parent.$iframeDialog.close();
}
});
function saveSettings(native, common, callback) {
if (typeof common === 'function') {
callback = common;
common = null;
}
socket.emit('getObject', id, function (err, oldObj) {
if (!oldObj) oldObj = {};
for (var a in native) {
if (native.hasOwnProperty(a)) {
oldObj.native[a] = native[a];
}
}
if (common) {
for (var b in common) {
if (common.hasOwnProperty(b)) {
oldObj.common[b] = common[b];
}
}
}
if (onChangeSupported) {
$navButtons.find('.btn-save').addClass('disabled');
$navButtons.find('.btn-save-close').addClass('disabled');
$navButtons.find('.btn-cancel').find('span').html(_('close'));
}
socket.emit('setObject', id, oldObj, function (err) {
if (err) {
showMessage(err, _('Error'), 'alert');
return;
}
changed = false;
if (callback) callback();
});
});
}
// Read language settings
function loadSystemConfig(callback) {
socket.emit('getObject', 'system.config', function (err, res) {
if (!err && res && res.common) {
systemLang = res.common.language || systemLang;
systemConfig = res;
}
socket.emit('getObject', 'system.certificates', function (err, res) {
if (!err && res) {
if (res.native && res.native.certificates) {
certs = [];
for (var c in res.native.certificates) {
if (!res.native.certificates.hasOwnProperty(c) || !res.native.certificates[c]) continue;
// If it is filename, it could be everything
if (res.native.certificates[c].length < 700 && (res.native.certificates[c].indexOf('/') !== -1 || res.native.certificates[c].indexOf('\\') !== -1)) {
var __cert = {
name: c,
type: ''
};
if (c.toLowerCase().indexOf('private') !== -1) {
__cert.type = 'private';
} else if (res.native.certificates[c].toLowerCase().indexOf('private') !== -1) {
__cert.type = 'private';
} else if (c.toLowerCase().indexOf('public') !== -1) {
__cert.type = 'public';
} else if (res.native.certificates[c].toLowerCase().indexOf('public') !== -1) {
__cert.type = 'public';
}
certs.push(__cert);
continue;
}
var _cert = {
name: c,
type: (res.native.certificates[c].substring(0, '-----BEGIN RSA PRIVATE KEY'.length) === '-----BEGIN RSA PRIVATE KEY' || res.native.certificates[c].substring(0, '-----BEGIN PRIVATE KEY'.length) === '-----BEGIN PRIVATE KEY') ? 'private' : 'public'
};
if (_cert.type === 'public') {
var m = res.native.certificates[c].split('-----END CERTIFICATE-----');
var count = 0;
for (var _m = 0; _m < m.length; _m++) {
if (m[_m].replace(/\r\n|\r|\n/, '').trim()) count++;
}
if (count > 1) _cert.type = 'chained';
}
certs.push(_cert);
}
}
}
if (callback) callback();
});
});
}
// callback if something changed
function onChange(isChanged) {
onChangeSupported = true;
if (typeof isChanged === 'boolean' && isChanged === false) {
changed = false;
$navButtons.find('.btn-save').addClass('disabled');
$navButtons.find('.btn-save-close').addClass('disabled');
$navButtons.find('.btn-cancel').find('span').html(_('close'));
} else {
changed = true;
$navButtons.find('.btn-save').removeClass('disabled');
$navButtons.find('.btn-save-close').removeClass('disabled');
$navButtons.find('.btn-cancel').find('span').html(_('cancel'));
}
}
function loadSettings(callback) {
socket.emit('getObject', id, function (err, res) {
if (!err && res && res.native) {
$('.adapter-instance').html(adapter + '.' + instance);
$('.adapter-config').html('system.adapter.' + adapter + '.' + instance);
common = res.common;
if (res.common && res.common.name) $('.adapter-name').html(res.common.name);
if (typeof load === 'undefined') {
alert('Please implement save function in your admin/index.html');
} else {
load(res.native, onChange);
// init selects
if (isMaterialize) {
$('select').select();
M.updateTextFields();
// workaround for materialize checkbox problem
$('input[type="checkbox"]+span').off('click').on('click', function () {
var $input = $(this).prev();
if (!$input.prop('disabled')) {
$input.prop('checked', !$input.prop('checked')).trigger('change');
}
});
}
}
if (typeof callback === 'function') {
callback();
}
} else {
if (typeof callback === 'function') {
callback();
}
alert('error loading settings for ' + id + '\n\n' + err);
}
});
}
___onChange = onChange;
}
$(document).ready(function () {
'use strict';
if (window.location.pathname.indexOf('/index_m.html') === -1) {
// load materialize
var cssLink = document.createElement('link');
cssLink.href = '../../lib/css/materialize.css';
cssLink.type = 'text/css';
cssLink.rel = 'stylesheet';
cssLink.media = 'screen,print';
document.getElementsByTagName('head')[0].appendChild(cssLink);
// load materialize.js
var jsLink = document.createElement('script');
jsLink.onload = preInit;
jsLink.src = '../../lib/js/materialize.js';
document.head.appendChild(jsLink);
} else {
isMaterialize = true;
preInit();
}
});
function handleFileSelect(evt) {
var f = evt.target.files[0];
if (f) {
var r = new FileReader();
r.onload = function(e) {
var contents = e.target.result;
try {
var json = JSON.parse(contents);
if (json.native && json.common) {
if (json.common.name !== common.name) {
showError(_('otherConfig', json.common.name));
} else {
load(json.native, ___onChange);
// init selects
if (isMaterialize) {
$('select').select();
M.updateTextFields();
// workaround for materialize checkbox problem
$('input[type="checkbox"]+span').off('click').on('click', function () {
var $input = $(this).prev();
if (!$input.prop('disabled')) {
$input.prop('checked', !$input.prop('checked')).trigger('change');
}
});
}
___onChange();
showToast(null, _('configLoaded'));
}
} else {
showError(_('invalidConfig'));
}
} catch (e) {
showError(e.toString());
}
};
r.readAsText(f);
} else {
alert('Failed to open JSON File');
}
}
function generateFile(filename, obj) {
var el = document.createElement('a');
el.setAttribute('href', 'data:application/json;charset=utf-8,' + encodeURIComponent(JSON.stringify(obj, null, 2)));
el.setAttribute('download', filename);
el.style.display = 'none';
document.body.appendChild(el);
el.click();
document.body.removeChild(el);
}
function prepareTooltips() {
if (isMaterialize) {
// init tabs
$('.tabs').mtabs();
var buttonsText = '';
if (common && common.readme) {
buttonsText += ' <a class="btn-floating btn-small waves-effect waves-light" href="' + common.readme +'" target="_blank">' +
' <i class="material-icons">live_help</i>' +
' </a>';
}
buttonsText += ' <a class="btn-floating btn-small waves-effect waves-light adapter-config-load" title="' + _('loadConfig') + '">' +
' <i class="material-icons">file_upload</i>' +
' </a>';
buttonsText += ' <a class="btn-floating btn-small waves-effect waves-light adapter-config-save" title="' + _('saveConfig') + '">' +
' <i class="material-icons">file_download</i>' +
' </a>';
if (buttonsText) {
// add help link after logo
var $logo = $('.logo').first().parent();
if ($logo.length) {
$('<div class="col s6 help-link">' + buttonsText + '</div>').insertAfter($logo);
}
$('.adapter-config-load').click(function () {
var input = document.createElement('input');
input.setAttribute('type', 'file');
input.setAttribute('id', 'files');
input.setAttribute('opacity', 0);
input.addEventListener('change', function (e) {
handleFileSelect(e, function () {});
}, false);
(input.click)();
});
$('.adapter-config-save').click(function () {
save(function (native, cmn) {
var result = {
_id: 'system.adapter.' + common.name + '.' + instance,
common: JSON.parse(JSON.stringify(common)),
native: native
};
// remove unimportant information
if (result.common.news) {
delete result.common.news;
}
if (result.common.titleLang) {
delete result.common.titleLang;
}
if (result.common.desc) {
delete result.common.desc;
}
if (cmn) {
for (var b in cmn) {
if (cmn.hasOwnProperty(b)) {
result.common[b] = cmn[b];
}
}
}
//window.open('data:application/yunkong2; content-disposition=attachment; filename=' + result._id + '.json,' + JSON.stringify(result, null, 2));
generateFile(result._id + '.json', result);
});
})
}
$('.value').each(function () {
var $this = $(this);
// replace all labels after checkboxes to span (bug in materialize)
if ($this.attr('type') === 'checkbox') {
$this.addClass('filled-in');
var $label = $this.next();
if ($label.prop('tagName') === 'LABEL') {
$label.replaceWith('<span style="' + ($label.attr('style') || '') + '" class="' + ($label.attr('class') || '') + '">' + $label.html() +'</span>');
$label = $this.next();
}
$label.off('click').on('click', function () {
var $input = $(this).prev();
if (!$input.prop('disabled')) {
$input.prop('checked', !$input.prop('checked')).trigger('change');
}
});
}
var id = $this.data('id');
var tooltip = '';
if (systemDictionary['tooltip_' + id]) {
tooltip = systemDictionary['tooltip_' + id][systemLang] || systemDictionary['tooltip_' + id].en;
}
var link = $this.data('link');
if (link && common) {
if (link === true) {
if (common.readme) {
link = common.readme + '#' + id;
} else {
link = 'https://github.com/yunkong2/yunkong2.' + common.name + '#' + id;
}
}
if (!link.match('^https?:\/\/')) {
if (common.readme) {
link = common.readme + '#' + link;
} else {
link = 'https://github.com/yunkong2/yunkong2.' + common.name + '#' + link;
}
}
}
if (link) {
$('<a class="tooltip" href="' + link + '" title="' + (tooltip || systemDictionary.htooltip[systemLang]) + '" target="_blank"><i class="material-icons tooltip">live_help</i></a>').insertBefore($this);
} else if (tooltip) {
$('<i class="material-icons tooltip" title="' + tooltip + '">help_outline</i>').insertBefore($this);
}
});
} else {
$('.admin-icon').each(function () {
var id = $(this).data('id');
if (!id) {
var $prev = $(this).prev();
var $input = $prev.find('input');
if (!$input.length) $input = $prev.find('select');
if (!$input.length) $input = $prev.find('textarea');
if (!$input.length) {
$prev = $prev.parent();
$input = $prev.find('input');
if (!$input.length) $input = $prev.find('select');
if (!$input.length) $input = $prev.find('textarea');
}
if ($input.length) id = $input.attr('id');
}
if (!id) return;
var tooltip = '';
if (systemDictionary['tooltip_' + id]) {
tooltip = systemDictionary['tooltip_' + id][systemLang] || systemDictionary['tooltip_' + id].en;
}
var icon = '';
var link = $(this).data('link');
if (link && common) {
if (link === true) {
if (common.readme) {
link = common.readme + '#' + id;
} else {
link = 'https://github.com/yunkong2/yunkong2.' + common.name + '#' + id;
}
}
if (!link.match('^https?:\/\/')) {
if (common.readme) {
link = common.readme + '#' + link;
} else {
link = 'https://github.com/yunkong2/yunkong2.' + common.name + '#' + link;
}
}
icon += '<a class="admin-tooltip-link" target="config_help" href="' + link + '" title="' + (tooltip || systemDictionary.htooltip[systemLang]) + '"><img class="admin-tooltip-icon" src="../../img/info.png" /></a>';
} else if (tooltip) {
icon += '<img class="admin-tooltip-icon" title="' + tooltip + '" src="../../img/info.png"/>';
}
if (icon) {
$(this).html(icon);
}
});
$('.admin-text').each(function () {
var id = $(this).data('id');
if (!id) {
var $prev = $(this).prev();
var $input = $prev.find('input');
if (!$input.length) $input = $prev.find('select');
if (!$input.length) $input = $prev.find('textarea');
if (!$input.length) {
$prev = $prev.parent();
$input = $prev.find('input');
if (!$input.length) $input = $prev.find('select');
if (!$input.length) $input = $prev.find('textarea');
}
if ($input.length) id = $input.attr('id');
}
if (!id) return;
// check if translation for this exist
if (systemDictionary['info_' + id]) {
$(this).html('<span class="admin-tooltip-text">' + (systemDictionary['info_' + id][systemLang] || systemDictionary['info_' + id].en) + '</span>');
}
});
}
}
function showMessageJQ(message, title, icon, width) {
var $dialogMessage = $('#dialog-message-settings');
if (!$dialogMessage.length) {
$('body').append('<div id="dialog-message-settings" title="Message" style="display: none">\n' +
'<p>' +
'<span id="dialog-message-icon-settings" class="ui-icon ui-icon-circle-check" style="float :left; margin: 0 7px 50px 0;"></span>\n' +
'<span id="dialog-message-text-settings"></span>\n' +
'</p>\n' +
'</div>');
$dialogMessage = $('#dialog-message-settings');
$dialogMessage.dialog({
autoOpen: false,
modal: true,
buttons: [
{
text: _('Ok'),
click: function () {
$(this).dialog('close');
}
}
]
});
}
$dialogMessage.dialog('option', 'width', width + 500);
if (typeof _ !== 'undefined') {
$dialogMessage.dialog('option', 'title', title || _('Message'));
} else {
$dialogMessage.dialog('option', 'title', title || 'Message');
}
$('#dialog-message-text-settings').html(message);
if (icon) {
$('#dialog-message-icon-settings')
.show()
.attr('class', '')
.addClass('ui-icon ui-icon-' + icon);
} else {
$('#dialog-message-icon-settings').hide();
}
$dialogMessage.dialog('open');
}
function showMessage(message, title, icon) {
if (!isMaterialize) {
return showMessageJQ(message, title, icon);
}
var $dialogMessage;
// noinspection JSJQueryEfficiency
$dialogMessage = $('#dialog-message');
if (!$dialogMessage.length) {
$('body').append(
'<div class="m"><div id="dialog-message" class="modal modal-fixed-footer">' +
' <div class="modal-content">' +
' <h6 class="dialog-title title"></h6>' +
' <p><i class="large material-icons dialog-icon"></i><span class="dialog-text"></span></p>' +
' </div>' +
' <div class="modal-footer">' +
' <a class="modal-action modal-close waves-effect waves-green btn-flat translate">Ok</a>' +
' </div>' +
'</div></div>');
$dialogMessage = $('#dialog-message');
}
if (icon) {
$dialogMessage.find('.dialog-icon')
.show()
.html(icon);
} else {
$dialogMessage.find('.dialog-icon').hide();
}
if (title) {
$dialogMessage.find('.dialog-title').html(title).show();
} else {
$dialogMessage.find('.dialog-title').hide();
}
$dialogMessage.find('.dialog-text').html(message);
$dialogMessage.modal().modal('open');
}
function confirmMessageJQ(message, title, icon, buttons, callback) {
var $dialogConfirm = $('#dialog-confirm-settings');
if (!$dialogConfirm.length) {
$('body').append('<div id="dialog-confirm-settings" title="Message" style="display: none">\n' +
'<p>' +
'<span id="dialog-confirm-icon-settings" class="ui-icon ui-icon-circle-check" style="float :left; margin: 0 7px 50px 0;"></span>\n' +
'<span id="dialog-confirm-text-settings"></span>\n' +
'</p>\n' +
'</div>');
$dialogConfirm = $('#dialog-confirm-settings');
$dialogConfirm.dialog({
autoOpen: false,
modal: true
});
}
if (typeof buttons === 'function') {
callback = buttons;
$dialogConfirm.dialog('option', 'buttons', [
{
text: _('Ok'),
click: function () {
var cb = $(this).data('callback');
$(this).data('callback', null);
$(this).dialog('close');
if (cb) cb(true);
}
},
{
text: _('Cancel'),
click: function () {
var cb = $(this).data('callback');
$(this).data('callback', null);
$(this).dialog('close');
if (cb) cb(false);
}
}
]);
} else if (typeof buttons === 'object') {
for (var b = 0; b < buttons.length; b++) {
buttons[b] = {
text: buttons[b],
id: 'dialog-confirm-button-' + b,
click: function (e) {
var id = parseInt(e.currentTarget.id.substring('dialog-confirm-button-'.length), 10);
var cb = $(this).data('callback');
$(this).data('callback', null);
$(this).dialog('close');
if (cb) cb(id);
}
}
}
$dialogConfirm.dialog('option', 'buttons', buttons);
}
$dialogConfirm.dialog('option', 'title', title || _('Message'));
$('#dialog-confirm-text-settings').html(message);
if (icon) {
$('#dialog-confirm-icon-settings')
.show()
.attr('class', '')
.addClass('ui-icon ui-icon-' + icon);
} else {
$('#dialog-confirm-icon-settings').hide();
}
$dialogConfirm.data('callback', callback);
$dialogConfirm.dialog('open');
}
function confirmMessage(message, title, icon, buttons, callback) {
if (!isMaterialize) {
return confirmMessageJQ(message, title, icon, buttons, callback);
}
var $dialogConfirm;
// noinspection JSJQueryEfficiency
$dialogConfirm = $('#dialog-confirm');
if (!$dialogConfirm.length) {
$('body').append(
'<div class="m"><div id="dialog-confirm" class="modal modal-fixed-footer">' +
' <div class="modal-content">' +
' <h6 class="dialog-title title"></h6>' +
' <p><i class="large material-icons dialog-icon"></i><span class="dialog-text"></span></p>' +
' </div>' +
' <div class="modal-footer">' +
' </div>' +
'</div></div>'
);
$dialogConfirm = $('#dialog-confirm');
}
if (typeof buttons === 'function') {
callback = buttons;
$dialogConfirm.find('.modal-footer').html(
'<a class="modal-action modal-close waves-effect waves-green btn-flat translate" data-result="true">' + _('Ok') + '</a>' +
'<a class="modal-action modal-close waves-effect waves-green btn-flat translate">' + _('Cancel') + '</a>');
$dialogConfirm.find('.modal-footer .modal-action').on('click', function () {
var cb = $dialogConfirm.data('callback');
cb && cb($(this).data('result'));
});
} else if (typeof buttons === 'object') {
var tButtons = '';
for (var b = buttons.length - 1; b >= 0; b--) {
tButtons += '<a class="modal-action modal-close waves-effect waves-green btn-flat translate" data-id="' + b + '">' + buttons[b] + '</a>';
}
$dialogConfirm.find('.modal-footer').html(tButtons);
$dialogConfirm.find('.modal-footer .modal-action').on('click', function () {
var cb = $dialogConfirm.data('callback');
cb && cb($(this).data('id'));
});
}
$dialogConfirm.find('.dialog-title').text(title || _('Please confirm'));
if (icon) {
$dialogConfirm.find('.dialog-icon')
.show()
.html(icon);
} else {
$dialogConfirm.find('.dialog-icon').hide();
}
if (title) {
$dialogConfirm.find('.dialog-title').html(title).show();
} else {
$dialogConfirm.find('.dialog-title').hide();
}
$dialogConfirm.find('.dialog-text').html(message);
$dialogConfirm.data('callback', callback);
$dialogConfirm.modal({
dismissible: false
}).modal('open');
}
function showError(error) {
showMessage(_(error), _('Error'), 'error_outline');
}
function showToast(parent, message, icon, duration, isError, classes) {
if (typeof parent === 'string') {
classes = isError;
isError = duration;
icon = message;
message = parent;
parent = null;
}
if (parent && parent instanceof jQuery) {
parent = parent[0];
}
classes = classes || [];
if (typeof classes === 'string') {
classes = [classes];
}
isError && classes.push('dropZone-error');
M.toast({
parentSelector: parent || $('body')[0],
html: message + (icon ? '<i class="material-icons">' + icon + '</i>' : ''),
displayLength: duration || 3000,
classes: classes
});
}
function getObject(id, callback) {
socket.emit('getObject', id, function (err, res) {
if (!err && res) {
if (callback) callback(err, res);
} else {
if (callback) callback(null);
}
});
}
function getState(id, callback) {
socket.emit('getState', id, function (err, res) {
if (!err && res) {
if (callback) callback(err, res);
} else {
if (callback) callback(null);
}
});
}
function getEnums(_enum, callback) {
socket.emit('getObjectView', 'system', 'enum', {startkey: 'enum.' + _enum, endkey: 'enum.' + _enum + '.\u9999'}, function (err, res) {
if (!err && res) {
var _res = {};
for (var i = 0; i < res.rows.length; i++) {
if (res.rows[i].id === 'enum.' + _enum) continue;
_res[res.rows[i].id] = res.rows[i].value;
}
if (callback) callback(null, _res);
} else {
if (callback) callback(err, []);
}
});
}
function getGroups(callback) {
socket.emit('getObjectView', 'system', 'group', {startkey: 'system.group.', endkey: 'system.group.\u9999'}, function (err, res) {
if (!err && res) {
var _res = {};
for (var i = 0; i < res.rows.length; i++) {
_res[res.rows[i].id] = res.rows[i].value;
}
if (callback) callback(null, _res);
} else {
if (callback) callback(err, []);
}
});
}
function getUsers(callback) {
socket.emit('getObjectView', 'system', 'user', {startkey: 'system.user.', endkey: 'system.user.\u9999'}, function (err, res) {
if (!err && res) {
var _res = {};
for (var i = 0; i < res.rows.length; i++) {
_res[res.rows[i].id] = res.rows[i].value;
}
if (callback) callback(null, _res);
} else {
if (callback) callback(err, []);
}
});
}
function fillUsers(elemId, current, callback) {
getUsers(function (err, users) {
// Answer is like
// {
// "admin": { <<=== This is a common.name!
// "type": "user",
// "common": {
// "name": "admin",
// "password": "aaa",
// "dontDelete": true,
// "enabled": true,
// "icon": "data:image/png;base64,xxx",
// "color": "#ca0808",
// "desc": ""
// },
// "_id": "system.user.admin"
// },
// ...
// }
var text = '';
// Warning u is name of user and not ID.
var len = 'system.user.'.length;
for (var u in users) {
if (users.hasOwnProperty(u)) {
var id = users[u]._id.substring(len);
text += '<option value="' + id + '" ' + (current === id ? 'selected' : '') + ' >' + u[0].toUpperCase() + u.substring(1) + '</option>\n';
}
}
$(elemId).html(text);
if (isMaterialize) {
$(elemId).select();
}
});
}
function getIPs(host, callback) {
if (typeof host === 'function') {
callback = host;
host = null;
}
socket.emit('getHostByIp', host || common.host, function (ip, _host) {
if (_host) {
host = _host;
var IPs4 = [{name: '[IPv4] 0.0.0.0 - ' + _('Listen on all IPs'), address: '0.0.0.0', family: 'ipv4'}];
var IPs6 = [{name: '[IPv6] ::', address: '::', family: 'ipv6'}];
if (host.native.hardware && host.native.hardware.networkInterfaces) {
for (var eth in host.native.hardware.networkInterfaces) {
if (!host.native.hardware.networkInterfaces.hasOwnProperty(eth)) continue;
for (var num = 0; num < host.native.hardware.networkInterfaces[eth].length; num++) {
if (host.native.hardware.networkInterfaces[eth][num].family !== 'IPv6') {
IPs4.push({name: '[' + host.native.hardware.networkInterfaces[eth][num].family + '] ' + host.native.hardware.networkInterfaces[eth][num].address + ' - ' + eth, address: host.native.hardware.networkInterfaces[eth][num].address, family: 'ipv4'});
} else {
IPs6.push({name: '[' + host.native.hardware.networkInterfaces[eth][num].family + '] ' + host.native.hardware.networkInterfaces[eth][num].address + ' - ' + eth, address: host.native.hardware.networkInterfaces[eth][num].address, family: 'ipv6'});
}
}
}
}
for (var i = 0; i < IPs6.length; i++) {
IPs4.push(IPs6[i]);
}
callback(IPs4);
}
});
}
function fillSelectIPs(id, actualAddr, noIPv4, noIPv6, callback) {
getIPs(function (ips) {
var str = '';
for (var i = 0; i < ips.length; i++) {
if (noIPv4 && ips[i].family === 'ipv4') continue;
if (noIPv6 && ips[i].family === 'ipv6') continue;
str += '<option value="' + ips[i].address + '" ' + ((ips[i].address === actualAddr) ? 'selected' : '') + '>' + ips[i].name + '</option>';
}
$(id).html(str);
if (isMaterialize) {
$(id).select();
}
if (typeof callback === 'function') {
callback();
}
});
}
function sendTo(_adapter_instance, command, message, callback) {
socket.emit('sendTo', (_adapter_instance || adapter + '.' + instance), command, message, callback);
}
function sendToHost(host, command, message, callback) {
socket.emit('sendToHost', host || common.host, command, message, callback);
}
function getInterfaces(onlyNames, callback) {
if (typeof onlyNames === 'function') {
callback = onlyNames;
onlyNames = false;
}
socket.emit('sendToHost', common.host, 'getInterfaces', null, function (result) {
if (result && result.result) {
if (onlyNames) {
callback(null, Object.keys(result.result));
} else {
callback(null, result);
}
} else {
callback((result && result.error) || 'cannot read');
}
});
}
// fills select with names of the certificates and preselect it
function fillSelectCertificates(id, type, actualValued) {
var str = '<option value="">' + _('none') + '</option>';
for (var i = 0; i < certs.length; i++) {
if (certs[i].type && certs[i].type !== type) continue;
str += '<option value="' + certs[i].name + '" ' + ((certs[i].name === actualValued) ? 'selected' : '') + '>' + certs[i].name + '</option>';
}
$(id).html(str);
if (isMaterialize) {
$(id).select();
}
}
function getAdapterInstances(_adapter, callback) {
if (typeof _adapter === 'function') {
callback = _adapter;
_adapter = null;
}
socket.emit('getObjectView', 'system', 'instance', {startkey: 'system.adapter.' + (_adapter || adapter), endkey: 'system.adapter.' + (_adapter || adapter) + '.\u9999'}, function (err, doc) {
if (err) {
if (callback) callback ([]);
} else {
if (doc.rows.length === 0) {
if (callback) callback ([]);
} else {
var res = [];
for (var i = 0; i < doc.rows.length; i++) {
res.push(doc.rows[i].value);
}
if (callback) callback(res);
}
}
});
}
function getExtendableInstances(_adapter, callback) {
if (typeof _adapter === 'function') {
callback = _adapter;
_adapter = null;
}
socket.emit('getObjectView', 'system', 'instance', null, function (err, doc) {
if (err) {
if (callback) callback ([]);
} else {
if (doc.rows.length === 0) {
if (callback) callback ([]);
} else {
var res = [];
for (var i = 0; i < doc.rows.length; i++) {
if (doc.rows[i].value.common.webExtendable) {
res.push(doc.rows[i].value);
}
}
if (callback) callback (res);
}
}
});
}
function getIsAdapterAlive(_adapter, callback) {
if (typeof _adapter === 'function') {
callback = _adapter;
_adapter = null;
}
getState('system.adapter.' + (_adapter || adapter) + '.' + instance + '.alive', function (err, obj) {
if (!obj || !obj.val) {
callback(false);
} else {
callback(true);
}
});
}
// Adds one entry to the created with editTable table
// tabId - the id of the table used by editTable
// value - is one object to add in form {ip: '3.3.3.3', room: 'enum.room.bla3', desc: 'Bla3'}
// $grid - [optional] - object returned by editTable to speed up the addition
// _isInitial - [optional] - if it is initial fill of the table. To not trigger the onChange
function addToTable(tabId, value, $grid, _isInitial) {
$grid = $grid || $('#' + tabId);
var obj = {_id: $grid[0]._maxIdx++};
var cols = $grid[0]._cols;
for (var i = 0; i < cols.length; i++) {
if (cols[i] === 'room') {
obj[cols[i]] = ($grid[0]._rooms[value[cols[i]]]) ? $grid[0]._rooms[value[cols[i]]].common.name : value[cols[i]];
} else {
obj[cols[i]] = value[cols[i]];
}
}
obj._commands =
'<button data-' + tabId + '-id="' + obj._id + '" class="' + tabId + '-edit-submit">' + _('edit') + '</button>' +
'<button data-' + tabId + '-id="' + obj._id + '" class="' + tabId + '-delete-submit">' + _('delete') + '</button>' +
'<button data-' + tabId + '-id="' + obj._id + '" class="' + tabId + '-ok-submit" style="display:none">' + _('ok') + '</button>' +
'<button data-' + tabId + '-id="' + obj._id + '" class="' + tabId + '-cancel-submit" style="display:none">' + _('cancel') + '</button>';
$grid.jqGrid('addRowData', tabId + '_' + obj._id, obj);
_editInitButtons($grid, tabId, obj._id);
if (!_isInitial && $grid[0]._onChange) $grid[0]._onChange('add', value);
}
function _editInitButtons($grid, tabId, objId) {
var search = objId ? '[data-' + tabId + '-id="' + objId + '"]' : '';
$('.' + tabId + '-edit-submit' + search).off('click').button({
icons: {primary: 'ui-icon-pencil'},
text: false
}).on('click', function () {
var id = $(this).attr('data-' + tabId + '-id');
$('.' + tabId + '-edit-submit').hide();
$('.' + tabId + '-delete-submit').hide();
$('.' + tabId + '-ok-submit[data-' + tabId + '-id="' + id + '"]').show();
$('.' + tabId + '-cancel-submit[data-' + tabId + '-id="' + id + '"]').show();
$grid.jqGrid('editRow', tabId + '_' + id, {url: 'clientArray'});
if ($grid[0]._edited.indexOf(id) === -1) {
$grid[0]._edited.push(id);
}
changed = true;
var $navButtons = $('.dialog-config-buttons');
$navButtons.find('.btn-save').removeClass('disabled');
$navButtons.find('.btn-save-close').removeClass('disabled');
if (onChangeSupported) {
$navButtons.find('.btn-cancel').find('span').html(_('cancel'));
}
}).css({'height': '18px', width: '22px'});
$('.' + tabId + '-delete-submit' + search).off('click').button({
icons: {primary: 'ui-icon-trash'},
text: false
}).on('click', function () {
var id = $(this).attr('data-' + tabId + '-id');
$grid.jqGrid('delRowData', tabId + '_' + id);
changed = true;
var $navButtons = $('.dialog-config-buttons');
$navButtons.find('.btn-save').removeClass('disabled');
$navButtons.find('.btn-save-close').removeClass('disabled');
if (onChangeSupported) {
$navButtons.find('.btn-cancel').find('span').html(_('cancel'));
}
var pos = $grid[0]._edited.indexOf(id);
if (pos !== -1) {
$grid[0]._edited.splice(pos, 1);
}
if ($grid[0]._onChange) $grid[0]._onChange('del', id);
}).css({'height': '18px', width: '22px'});
$('.' + tabId + '-ok-submit' + search).off('click').button({
icons: {primary: 'ui-icon-check'},
text: false
}).on('click', function () {
var id = $(this).attr('data-' + tabId + '-id');
$('.' + tabId + '-edit-submit').show();
$('.' + tabId + '-delete-submit').show();
$('.' + tabId + '-ok-submit').hide();
$('.' + tabId + '-cancel-submit').hide();
$grid.jqGrid('saveRow', tabId + '_' + id, {url: 'clientArray'});
changed = true;
var $navButtons = $('.dialog-config-buttons');
$navButtons.find('.btn-save').removeClass('disabled');
$navButtons.find('.btn-save-close').removeClass('disabled');
if (onChangeSupported) {
$navButtons.find('.btn-cancel').find('span').html(_('cancel'));
}
var pos = $grid[0]._edited.indexOf(id);
if (pos !== -1) {
$grid[0]._edited.splice(pos, 1);
}
if ($grid[0]._onChange) $grid[0]._onChange('changed', $grid.jqGrid('getRowData', tabId + '_' + id));
}).css({'height': '18px', width: '22px'});
$('.' + tabId + '-cancel-submit' + search).off('click').button({
icons: {primary: 'ui-icon-close'},
text: false
}).on('click', function () {
var id = $(this).attr('data-' + tabId + '-id');
$('.' + tabId + '-edit-submit').show();
$('.' + tabId + '-delete-submit').show();
$('.' + tabId + '-ok-submit').hide();
$('.' + tabId + '-cancel-submit').hide();
$grid.jqGrid('restoreRow', tabId + '_' + id, false);
var pos = $grid[0]._edited.indexOf(id);
if (pos !== -1) {
$grid[0]._edited.splice(pos, 1);
}
}).css({'height': '18px', width: '22px'});
}
function _editTable(tabId, cols, values, rooms, top, onChange) {
var title = 'Device list';
if (typeof tabId === 'object') {
cols = tabId.cols;
values = tabId.values;
rooms = tabId.rooms;
top = tabId.top;
onChange = tabId.onChange;
if (tabId.title) title = tabId.title;
tabId = tabId.tabId;
}
initGridLanguage(systemLang);
var colNames = [];
var colModel = [];
var $grid = $('#' + tabId);
var room;
colNames.push('id');
colModel.push({
name: '_id',
index: '_id',
hidden: true
});
for (var i = 0; i < cols.length; i++) {
var width = null;
var checkbox = null;
if (typeof cols[i] === 'object') {
width = cols[i].width;
if (cols[i].checkbox) checkbox = true;
cols[i] = cols[i].name;
}
colNames.push(_(cols[i]));
var _obj = {
name: cols[i],
index: cols[i],
// width: 160,
editable: true
};
if (width) _obj.width = width;
if (checkbox) {
_obj.edittype = 'checkbox';
_obj.editoptions = {value: 'true:false'};
}
if (cols[i] === 'room') {
var list = {'': _('none')};
for (room in rooms) {
list[room] = _(translateName(rooms[room].common.name));
}
_obj.stype = 'select';
_obj.edittype = 'select';
_obj.editoptions = {value: list};
_obj.searchoptions = {
sopt: ['eq'],
value: ':' + _('all')
};
for (room in rooms) {
_obj.searchoptions.value += ';' + _(translateName(rooms[room].common.name)) + ':' + _(translateName(rooms[room].common.name));
}
}
colModel.push(_obj);
}
colNames.push('');
colModel.push({name: '_commands', index: '_commands', width: 50, editable: false, align: 'center', search: false});
$grid[0]._cols = cols;
$grid[0]._rooms = rooms;
$grid[0]._maxIdx = 0;
$grid[0]._top = top;
$grid[0]._edited = [];
$grid[0]._onChange = onChange;
$grid.jqGrid({
datatype: 'local',
colNames: colNames,
colModel: colModel,
width: 800,
height: 330,
pager: $('#pager-' + tabId),
rowNum: 1000,
rowList: [1000],
ondblClickRow: function (rowid) {
var id = rowid.substring((tabId + '_').length);
$('.' + tabId + '-edit-submit').hide();
$('.' + tabId + '-delete-submit').hide();
$('.' + tabId + '-ok-submit[data-' + tabId + '-id="' + id + '"]').show();
$('.' + tabId + '-cancel-submit[data-' + tabId + '-id="' + id + '"]').show();
$grid.jqGrid('editRow', rowid, {url: 'clientArray'});
if ($grid[0]._edited.indexOf(id) === -1) $grid[0]._edited.push(id);
changed = true;
var $navButtons = $('.dialog-config-buttons');
$navButtons.find('.btn-save').removeClass('disabled');
$navButtons.find('.btn-save-close').removeClass('disabled');
if (onChangeSupported) {
$navButtons.find('.btn-cancel').find('span').html(_('cancel'));
}
},
sortname: 'id',
sortorder: 'desc',
viewrecords: false,
pgbuttons: false,
pginput: false,
pgtext: false,
caption: _(title),
ignoreCase: true,
loadComplete: function () {
_editInitButtons($grid, tabId);
},
onSortCol: function () {
changed = true;
var $navButtons = $('.dialog-config-buttons');
$navButtons.find('.btn-save').removeClass('disabled');
$navButtons.find('.btn-save-close').removeClass('disabled');
if (onChangeSupported) {
$navButtons.find('.btn-cancel').find('span').html(_('cancel'));
}
}
}).jqGrid('filterToolbar', {
defaultSearch: 'cn',
autosearch: true,
searchOnEnter: false,
enableClear: false,
afterSearch: function () {
_editInitButtons($grid, tabId);
}
});
$('#pager-' + tabId + '_center').hide();
if ($('#pager-' + tabId).length) {
$grid.navGrid('#pager-' + tabId, {
search: false,
edit: false,
add: false,
del: false,
refresh: false
}).jqGrid('navButtonAdd', '#pager-' + tabId, {
caption: '',
buttonicon: 'ui-icon-plus',
onClickButton: function () {
// Find new unique name
var found;
var newText = _('New');
var ids = $grid.jqGrid('getDataIDs');
var idx = 1;
var obj;
do {
found = true;
for (var _id = 0; _id < ids.length; _id++) {
obj = $grid.jqGrid('getRowData', ids[_id]);
if (obj && obj[$grid[0]._cols[0]] === newText + idx) {
idx++;
found = false;
break;
}
}
} while (!found);
obj = {};
for (var t = 0; t < $grid[0]._cols.length; t++) {
obj[$grid[0]._cols[t]] = '';
}
obj[$grid[0]._cols[0]] = newText + idx;
changed = true;
var $navButtons = $('.dialog-config-buttons');
$navButtons.find('.btn-save').removeClass('disabled');
$navButtons.find('.btn-save-close').removeClass('disabled');
if (onChangeSupported) {
$navButtons.find('.btn-cancel').find('span').html(_('cancel'));
}
addToTable(tabId, obj, $grid);
},
position: 'first',
id: 'add-cert',
title: _('new device'),
cursor: 'pointer'
});
}
if (values) {
for (var u = 0; u < values.length; u++) {
addToTable(tabId, values[u], $grid, true);
}
}
$(window).on('resize', function () {
$grid.setGridHeight($(this).height() - top).setGridWidth($(this).width() - 10);
});
$(window).trigger('resize');
// hide scrollbar
$('.ui-jqgrid-bdiv').css({'overflow-x': 'hidden'});
return $grid;
}
// converts "enum.room.Sleeping_room" to "Sleeping room"
// As input gets the list from getEnum
function enumName2Id(enums, name) {
name = name.toLowerCase();
for (var enumId in enums) {
if (!enums.hasOwnProperty(enumId)) continue;
if (enums[enumId] && enums[enumId].common && enums[enumId].common.name) {
if (typeof enums[enumId].common.name === 'object') {
for (var lang in enums[enumId].common.name) {
if (enums[enumId].common.name.hasOwnProperty(lang) && enums[enumId].common.name[lang].toLowerCase() === name) {
return enumId;
}
}
} else {
if (enums[enumId].common.name && enums[enumId].common.name.toLowerCase() === name) return enumId;
}
}
if (enums[enumId] && enums[enumId].name) {
if (typeof enums[enumId].name === 'object') {
for (var lang in enums[enumId].name) {
if (enums[enumId].name.hasOwnProperty(lang) && enums[enumId].name[lang].toLowerCase() === name) {
return enumId;
}
}
} else {
if (enums[enumId].name.toLowerCase() === name) return enumId;
}
}
}
return '';
}
// Creates edit table for any configuration array
// tabId - is id of table where the jqGrid must be created. E.g: <table id="devices"></table><div id="pager-devices"></div>
// cols - array with names of the properties of entry. E.g: ['ip', 'room', 'desc']
// if column has name room, for that will be automatically the room enums loaded and shown
// values - array with values in form [{ip: '1.1.1.1', room: 'enum.room.bla1', desc: 'Bla1'}, {ip: '2.2.2.2', room: 'enum.room.bla2', desc: 'Bla2'}
// top - top position of the table to set the height of the table automatically. Table must be always as last on the page.
// onChange - callback called if something is changed in the table
//
// returns the jquery object of $('#tabId')
// To extract data from table
function editTable(tabId, cols, values, top, onChange) {
if (typeof tabId === 'object') {
cols = tabId.cols;
}
if (cols.indexOf('room') !== -1) {
getEnums('rooms', function (err, list) {
return _editTable(tabId, cols, values, list, top, onChange);
});
} else {
return _editTable(tabId, cols, values, null, top, onChange);
}
}
// Extract edited array from table
// tabId - is id of table where the jqGrid must be created. E.g: <table id="devices"></table><div id="pager-devices"></div>
// cols - array with names of the properties of entry. E.g: ['ip', 'room', 'desc']
//
// Returns array with values
function getTableResult(tabId, cols) {
var $grid = $('#' + tabId);
for (var j = 0; j < $grid[0]._edited.length; j++) {
$grid.jqGrid('saveRow', tabId + '_' + $grid[0]._edited[j], {url: 'clientArray'});
}
$('.' + tabId + '-edit-submit').show();
$('.' + tabId + '-delete-submit').show();
$('.' + tabId + '-ok-submit').hide();
$('.' + tabId + '-cancel-submit').hide();
var data = $grid.jqGrid('getRowData');
var res = [];
for (var i = 0; i < data.length; i++) {
var obj = {};
for (var z = 0; z < cols.length; z++) {
if (cols[z] === 'room') {
obj[cols[z]] = enumName2Id($grid[0]._rooms, data[i][cols[z]]);
} else {
obj[cols[z]] = data[i][cols[z]];
}
}
res.push(obj);
}
return res;
}
/*
<div id="values">
<button class="table-button-add" style="margin-left: 10px"></button>
<table class="table-values" style="width: 100%; calc(100% - 200px)">
<thead>
<tr>
<th data-name="regex" style="width: 30%" class="translate">Context</th>
<th data-name="room" class="translate" data-type="select">Room</th>
<th data-name="aaa" class="translate" data-options="1/A;2/B;3/C;4" data-type="select">Room</th>
<th data-name="enabled" class="translate" data-type="checkbox">Enabled</th>
<th data-buttons="delete up down copy" style="width: 32px"></th>
</tr>
</thead>
</table>
</td>
*/
/**
* Create edit table from javascript array.
*
* This function creates a html edit table.
*
* <pre><code>
* <div id="values" style="width: 100%; height: calc(100% - 205px)">
* <button class="table-button-add" style="margin-left: 10px"></button>
* <div style="width: 100%; height: calc(100% - 30px); overflow: auto;">
* <table class="table-values" style="width: 100%;">
* <thead>
* <tr>
* <th data-name="_index" style="width: 30px" data-style="width: 100%; text-align: right">Context</th>
* <th data-name="regex" class="translate" style="width: 30%" data-style="text-align: right">Context</th>
* <th data-name="room" class="translate" data-type="select">Room</th>
* <th data-name="aaa" class="translate" data-options="1/A;2/B;3/C;4" data-type="select">Room</th>
* <th data-name="enabled" class="translate" data-type="checkbox" data-default="true">Enabled</th>
* <th data-buttons="delete up down copy" style="width: 32px"></th>
* </tr>
* </thead>
* </table>
* </div>
* </div>
* <pre><code>
*
* @param {string} divId name of the html element (or empty).
* @param {string} values data array
* @param {function} onChange this function will be called if something changed
* @param {function} onReady called, when the table is ready (may be to modify some elements of it)
* @param {number} maxRaw maximal number of rows
* @return {object} array with values
*/
function values2table(divId, values, onChange, onReady, maxRaw) {
if (typeof values === 'function') {
typeof onChange === 'number' ? maxRaw = onChange : maxRaw = null;
onChange = values;
values = divId;
divId = '';
}
if (typeof onReady === 'number') {
maxRaw = onReady;
onReady = null;
} else if (typeof maxRaw === 'undefined') {
maxRaw = null;
}
values = values || [];
var names = [];
var $div;
if (!divId) {
$div = $('body');
} else {
$div = $('#' + divId);
}
var $add = $div.find('.table-button-add');
$add.data('raw', values.length);
if (maxRaw) {
$add.data('maxraw', maxRaw);
}
if (!$add.data('inited')) {
$add.data('inited', true);
var addText = $add.text();
if (!isMaterialize) {
$add.button({
icons: {primary: 'ui-icon-plus'},
text: !!addText,
label: addText ? _(addText) : undefined
});
}
$add.on('click', function () {
if (!$add.data('maxraw') || ($add.data('raw') < $add.data('maxraw'))) {
var $table = $div.find('.table-values');
var values = $table.data('values');
var names = $table.data('names');
var obj = {};
for (var i = 0; i < names.length; i++) {
if (!names[i]) continue;
obj[names[i].name] = names[i].def;
}
values.push(obj);
onChange && onChange();
setTimeout(function () {
values2table(divId, values, onChange, onReady);
}, 100);
$add.data('raw', $add.data('raw') + 1);
} else {
confirmMessage(_('maxTableRaw') + ': ' + $add.data('maxraw'), _('maxTableRawInfo'), 'alert', ['Ok']);
}
});
}
if (values) {
var buttons = [];
var $table = $div.find('.table-values');
$table.data('values', values);
// load rooms
if (!$table.data('rooms') && $table.find('th[data-name="room"]').length) {
getEnums('rooms', function (err, list) {
var result = {};
var trRooms = _('nonerooms');
if (trRooms !== 'nonerooms') {
result[_('none')] = trRooms;
} else {
result[_('none')] = '';
}
var nnames = [];
for (var n in list) {
if (list.hasOwnProperty(n)) {
nnames.push(n);
}
}
nnames.sort(function (a, b) {
a = a.toLowerCase();
b = b.toLowerCase();
if (a > b) return 1;
if (a < b) return -1;
return 0;
});
for (var l = 0; l < nnames.length; l++) {
result[nnames[l]] = list[nnames[l]].common.name || l;
if (typeof result[nnames[l]] === 'object') {
result[nnames[l]] = result[nnames[l]][systemLang] || result[nnames[l]].en;
}
}
$table.data('rooms', result);
values2table(divId, values, onChange, onReady);
});
return;
}
// load functions
if (!$table.data('functions') && $table.find('th[data-name="func"]').length) {
getEnums('functions', function (err, list) {
var result = {};
var trFuncs = _('nonefunctions');
if (trFuncs !== 'nonefunctions') {
result[_('none')] = trFuncs;
} else {
result[_('none')] = '';
}
var nnames = [];
for (var n in list) {
if (list.hasOwnProperty(n)) {
nnames.push(n);
}
}
nnames.sort(function (a, b) {
a = a.toLowerCase();
b = b.toLowerCase();
if (a > b) return 1;
if (a < b) return -1;
return 0;
});
for (var l = 0; l < nnames.length; l++) {
result[nnames[l]] = list[nnames[l]].common.name || l;
if (typeof result[nnames[l]] === 'object') {
result[nnames[l]] = result[nnames[l]][systemLang] || result[nnames[l]].en;
}
}
$table.data('functions', result);
values2table(divId, values, onChange, onReady);
});
return;
}
$table.find('th').each(function () {
var name = $(this).data('name');
if (name) {
var obj = {
name: name,
type: $(this).data('type') || 'text',
def: $(this).data('default'),
style: $(this).data('style'),
tdstyle: $(this).data('tdstyle')
};
if (obj.type === 'checkbox') {
if (obj.def === 'false') obj.def = false;
if (obj.def === 'true') obj.def = true;
obj.def = !!obj.def;
} else if (obj.type === 'select' || obj.type === 'select multiple') {
var vals = ($(this).data('options') || '').split(';');
obj.options = {};
for (var v = 0; v < vals.length; v++) {
var parts = vals[v].split('/');
obj.options[parts[0]] = _(parts[1] || parts[0]);
if (v === 0) obj.def = (obj.def === undefined) ? parts[0] : obj.def;
}
} else {
obj.def = obj.def || '';
}
names.push(obj);
} else {
names.push(null);
}
name = $(this).data('buttons');
if (name) {
var bs = name.split(' ');
buttons.push(bs);
} else {
buttons.push(null);
}
});
$table.data('names', names);
var text = '';
for (var v = 0; v < values.length; v++) {
var idName = values[v] && values[v].id;
if (!idName && values[v]) {
if (names[0] === '_index') {
idName = values[v][names[1]];
} else {
idName = values[v][names[0]];
}
}
text += '<tr data-id="' + idName + '" data-index="' + v + '">';
for (var i = 0; i < names.length; i++) {
text += '<td';
var line = '';
var style = '';
var tdstyle = '';
if (names[i]) {
if (names[i].name !== '_index') {
tdstyle = names[i].tdstyle || '';
if (tdstyle && tdstyle[0] !== ';') tdstyle = ';' + tdstyle;
}
if (names[i].name === '_index') {
style = (names[i].style ? names[i].style : 'text-align: right;');
line += (v + 1);
} else if (names[i].type === 'checkbox') {
line += '<input style="' + (names[i].style || '') + '" class="values-input filled-in" type="checkbox" data-index="' + v + '" data-name="' + names[i].name + '" ' + (values[v][names[i].name] ? 'checked' : '') + '" data-old-value="' + (values[v][names[i].name] === undefined ? '' : values[v][names[i].name]) + '"/>';
if (isMaterialize) {
line += '<span></span>';
}
} else if (names[i].type.substring(0, 6) === 'select') {
line += (names[i].type.substring(7, 16) === 'multiple' ? '<select multiple style="' : '<select style="') + (names[i].style ? names[i].style : 'width: 100%') + '" class="values-input" data-index="' + v + '" data-name="' + names[i].name + '">';
var options;
if (names[i].name === 'room') {
options = $table.data('rooms');
} else if (names[i].name === 'func') {
options = $table.data('functions');
if (names[i].type === 'select multiple') delete options[_('none')];
} else {
options = names[i].options;
}
var val = (values[v][names[i].name] === undefined ? '' : values[v][names[i].name]);
if (typeof val !== 'object') val = [val];
for (var p in options) {
line += '<option value="' + p + '" ' + (val.indexOf(p) !== -1 ? ' selected' : '') + '>' + options[p] + '</option>';
}
line += '</select>';
} else {
line += '<input class="values-input" style="' + (names[i].style ? names[i].style : 'width: 100%') + '" type="' + names[i].type + '" data-index="' + v + '" data-name="' + names[i].name + '"/>';
}
}
if (buttons[i]) {
style = 'text-align: center; white-space: nowrap;';
for (var b = 0; b < buttons[i].length; b++) {
if ((!v && buttons[i][b] === 'up') || (v === values.length - 1 && buttons[i][b] === 'down')) {
if (isMaterialize) {
line += '<a data-command="' + buttons[i][b] + '" class="values-buttons btn-floating btn-small waves-effect waves-light disabled"><i class="material-icons">add</i></a>';
} else {
line += '<button data-command="' + buttons[i][b] + '" class="values-buttons" disabled>&nbsp;</button>';
}
} else {
if (isMaterialize) {
line += '<a data-index="' + v + '" data-command="' + buttons[i][b] + '" class="values-buttons btn-floating btn-small waves-effect waves-light"><i class="material-icons">add</i></a>';
} else {
line += '<button data-index="' + v + '" data-command="' + buttons[i][b] + '" class="values-buttons"></button>';
}
}
}
}
if (style.length || tdstyle.length) {
text += ' style="' + style + tdstyle + '">' + line + '</td>';
} else {
text += '>' + line + '</td>';
}
}
text += '</tr>';
}
var $lines = $div.find('.table-lines');
if (!$lines.length) {
$table.append('<tbody class="table-lines"></tbody>');
$lines = $div.find('.table-lines');
}
$lines.html(text);
$lines.find('.values-input').each(function () {
var $this = $(this);
var type = $this.attr('type');
var name = $this.data('name');
var id = $this.data('index');
$this.data('old-value', values[id][name]);
if (type === 'checkbox') {
$this.prop('checked', values[id][name]);
} else {
$this.val(values[id][name]);
}
});
$lines.find('.values-buttons').each(function () {
var command = $(this).data('command');
if (command === 'copy') {
if (!isMaterialize) {
$(this).button({
icons: {primary: 'ui-icon-copy'},
text: false
})
.css({width: '1em', height: '1em'});
} else {
$(this).find('i').html('content_copy');
}
$(this).on('click', function () {
if (!$add.data('maxraw') || ($add.data('raw') < $add.data('maxraw'))) {
var id = $(this).data('index');
var elem = JSON.parse(JSON.stringify(values[id]));
values.push(elem);
onChange && onChange();
setTimeout(function () {
if (typeof tableEvents === 'function') {
tableEvents(values.length - 1, elem, 'add');
}
values2table(divId, values, onChange, onReady);
}, 100);
if ($add.data('maxraw')) {
$add.data('raw', $add.data('raw') + 1);
}
}
});
} else
if (command === 'delete') {
if (!isMaterialize) {
$(this).button({
icons: {primary: 'ui-icon-trash'},
text: false
})
.css({width: '1em', height: '1em'});
} else {
$(this).addClass('red').find('i').html('delete');
}
$(this).on('click', function () {
var id = $(this).data('index');
var elem = values[id];
values.splice(id, 1);
onChange && onChange();
setTimeout(function () {
if (typeof tableEvents === 'function') {
tableEvents(id, elem, 'delete');
}
values2table(divId, values, onChange, onReady);
}, 100);
if ($add.data('maxraw')) {
$add.data('raw', $add.data('raw') - 1);
}
});
} else if (command === 'up') {
if (!isMaterialize) {
$(this).button({
icons: {primary: 'ui-icon-triangle-1-n'},
text: false
})
.css({width: '1em', height: '1em'})
} else {
$(this).find('i').html('arrow_upward');
}
$(this).on('click', function () {
var id = $(this).data('index');
var elem = values[id];
values.splice(id, 1);
values.splice(id - 1, 0, elem);
onChange && onChange();
setTimeout(function () {
values2table(divId, values, onChange, onReady);
}, 100);
});
} else if (command === 'down') {
if (!isMaterialize) {
$(this).button({
icons: {primary: 'ui-icon-triangle-1-s'},
text: false
})
.css({width: '1em', height: '1em'});
} else {
$(this).find('i').html('arrow_downward');
}
$(this).on('click', function () {
var id = $(this).data('index');
var elem = values[id];
values.splice(id, 1);
values.splice(id + 1, 0, elem);
onChange && onChange();
setTimeout(function () {
values2table(divId, values, onChange, onReady);
}, 100);
});
} else if (command === 'pair') {
if (!isMaterialize) {
$(this).button({
icons: {primary: 'ui-icon-transferthick-e-w'},
text: false
})
.css({width: '1em', height: '1em'});
} else {
$(this).find('i').html('leak_add');
}
$(this).on('click', function () {
if (typeof tableEvents === 'function') {
var id = $(this).data('index');
var elem = values[id];
tableEvents(id, elem, 'pair');
}
}).attr('title', _('pair'));
} else if (command === 'unpair') {
if (!isMaterialize) {
$(this).button({
icons: {primary: 'ui-icon-scissors'},
text: false
})
.css({width: '1em', height: '1em'});
} else {
$(this).find('i').html('leak_remove');
}
$(this).on('click', function () {
if (typeof tableEvents === 'function') {
var id = $(this).data('index');
var elem = values[id];
tableEvents(id, elem, 'unpair');
}
}).attr('title', _('unpair'));
} else if (command === 'edit') {
if (!isMaterialize) {
$(this).button({
icons: {primary: 'ui-icon-pencil'},
text: false
})
.css({width: '1em', height: '1em'});
} else {
$(this).find('i').html('edit');
}
$(this).on('click', function () {
var id = $(this).data('index');
if (typeof editLine === 'function') {
setTimeout(function () {
editLine(id, JSON.parse(JSON.stringify(values[id])), function (err, id, newValues) {
if (!err) {
if (JSON.stringify(values[id]) !== JSON.stringify(newValues)) {
onChange && onChange();
values[id] = newValues;
values2table(divId, values, onChange, onReady);
}
}
});
}, 100);
}
});
}
});
$lines.find('.values-input').on('change.adaptersettings', function () {
if ($(this).attr('type') === 'checkbox') {
if ($(this).prop('checked').toString() !== $(this).data('old-value')) onChange();
values[$(this).data('index')][$(this).data('name')] = $(this).prop('checked');
} else {
if ($(this).val() !== $(this).data('old-value')) onChange();
values[$(this).data('index')][$(this).data('name')] = $(this).val();
}
}).on('keyup', function () {
$(this).trigger('change.adaptersettings');
});
}
if (isMaterialize) {
if (!divId) {
M.updateTextFields();
$('select').select();
} else {
M.updateTextFields('#' + divId);
$('#' + divId).find('select').select();
}
// workaround for materialize checkbox problem
$div.find('input[type="checkbox"]+span').off('click').on('click', function () {
var $input = $(this).prev();
if (!$input.prop('disabled')) {
$input.prop('checked', !$input.prop('checked')).trigger('change');
}
});
}
if (typeof onReady === 'function') onReady();
}
/**
* Extract the values from table.
*
* This function extracts the values from edit table, that was generated with values2table function.
*
* @param {string} divId name of the html element (or nothing).
* @return {object} array with values
*/
function table2values(divId) {
var $div;
if (!divId) {
$div = $('body');
} else {
$div = $('#' + divId);
}
var names = [];
$div.find('.table-values th').each(function () {
var name = $(this).data('name');
if (name) {
names.push(name);
} else {
names.push('___ignore___');
}
});
var values = [];
var j = 0;
$div.find('.table-lines tr').each(function () {
values[j] = {};
$(this).find('td').each(function () {
var $input = $(this).find('input');
if ($input.length) {
var name = $input.data('name');
if (name) {
if ($input.attr('type') === 'checkbox') {
values[j][name] = $input.prop('checked');
} else {
values[j][name] = $input.val();
}
}
}
var $select = $(this).find('select');
if ($select.length) {
var name = $select.data('name');
values[j][name] = $select.val() || '';
}
});
j++;
});
return values;
}