', {
class: [config.moduleClass, config.sigTableModuleClass].join(' ')
});
$(this).append(moduleElement);
// set default values for all signature "datatables"
$.extend( $.fn.dataTable.defaults, {
pageLength: -1,
lengthMenu: [[5, 10, 25, 50, -1], [5, 10, 25, 50, 'All']],
autoWidth: false,
language: {
emptyTable: 'No signatures added',
zeroRecords: 'No signatures found',
lengthMenu: 'Show _MENU_ signatures',
info: 'Showing _START_ to _END_ of _TOTAL_ signatures'
},
columnDefs: [
{
targets: 0,
//"orderData": 0,
orderable: true,
title: 'sig',
width: '30px'
},{
targets: 1,
orderable: false,
title: 'group',
width: '50px'
},{
targets: 2,
orderable: false,
title: "name/description"
},{
targets: 3,
title: 'created',
width: '90px',
className: config.sigTableCounterClass
},{
targets: 4,
title: "updated",
width: '90px',
className: config.sigTableCounterClass
},{
targets: 5,
title: '',
orderable: false,
width: '10px',
class: 'text-center'
}
]
} );
// draw signature table
moduleElement.drawSignatureTable(signatureData, systemInfoData);
// draw toolbar for signature table
moduleElement.drawSignatureTableToolbar(systemData, emptySignatureData);
};
/**
* update systeminfo
*/
$.fn.drawSystemInfo = function(systemInfoData){
// TODO replace by AJAX
if(systemInfoData.systemId === 30002979){
var system = {
id: 30002979,
//name: 'J150020',
name: 'Tararan',
alias: '',
effect: '',
security: 'L',
trueSec: 0.3,
region: {
id: '10000036',
name: 'Devoid'
},
constellation: {
id: '20000436',
name: 'Jayai'
},
type: 'k-space'
};
}else if(systemInfoData.systemId === 30000142){
var system = {
id: 30000142,
//name: 'J150020',
name: 'Jita',
alias: '',
effect: '',
security: 'H',
trueSec: 0.9,
region: {
id: '10000002',
name: 'The Forge'
},
constellation: {
id: '20000020',
name: 'Kimotoro'
},
type: 'k-space'
};
}else{
var system = {
id: 2,
name: 'J150020',
alias: 'Polaris',
effect: 'magnetar',
security: 'C6',
trueSec: -1,
region: {
id: '12345',
name: 'F-R00030'
},
constellation: {
id: '678990',
name: 'F-C00298'
},
static: [{
security: 'C6',
name: ' W237',
lifetime: 24
}],
type: 'wh'
};
}
// create new module container
var moduleElement = $('
', {
class: [config.moduleClass, config.systemInfoModuleClass].join(' ')
});
$(this).prepend(moduleElement);
var effectName = Util.getEffectInfoForSystem(system.effect, 'name');
var effectClass = Util.getEffectInfoForSystem(system.effect, 'class');
// confirm dialog
var moduleConfig = {
name: 'modules/system_info',
position: moduleElement,
link: 'append',
functions: {
after: function(){
// init tooltips
var tooltipElements = $('.' + config.systemInfoModuleClass + ' [data-toggle="tooltip"]');
tooltipElements.tooltip();
// load trade routes
if(system.type !== 'wh'){
$(moduleElement).find('.' + config.systemInfoRoutesClass).updateSystemInfoRoutes(system.name, ['Jita', 'Amarr', 'Rens', 'Dodixie']);
}
// init system effect popover
var systemEffectData = Util.getSystemEffectData( system.security, system.effect);
if(systemEffectData !== false){
var systemInfoTable = $(moduleElement).find('.' + config.systemInfoTableClass);
// transform data into table
var systemEffectTable = Util.getSystemEffectTable( systemEffectData );
systemInfoTable.popover({
html: true,
trigger: 'hover',
placement: 'top',
delay: 200,
title: 'System effects',
content: systemEffectTable
});
}
// load kill statistic chart
$(moduleElement).find('.' + config.systemInfoGraphsClass).updateSystemInfoGraphs(system.id);
}
}
};
// add security class for statics
if(system.static){
$.each(system.static, function(i, staticWH){
system['static'][i]['class'] = Util.getSecurityClassForSystem( staticWH.security );
});
}
var moduleData = {
system: system,
tableClass: config.systemInfoTableClass,
securityClass: Util.getSecurityClassForSystem( system.security ),
trueSecClass: Util.getTrueSecClassForSystem( system.trueSec ),
effectName: effectName,
effectClass: effectClass
};
Render.showModule(moduleConfig, moduleData);
};
/**
* get label element with given content
* @param text
* @returns {*|XMLList}
*/
var getLabel = function(text, options){
var label = $('
', {
class: ['label', options.type, options.align].join(' ')
}).text( text );
return label;
};
/**
* updates the system info graph
* @param systemId
*/
$.fn.updateSystemInfoGraphs = function(systemId){
var parentElement = $(this);
parentElement.empty();
var graphElement = $('', {
class: config.systemInfoGraphKillsClass
});
parentElement.append(graphElement);
var showHours = 24;
var maxKillmailCount = 200; // limited by API
var labelOptions = {
align: 'center-block'
};
// private function draws a "system kills" graph
var drawGraph = function(data){
var tableData = data.tableData;
var label = '';
if(data.count === 0){
labelOptions.type = 'label-success';
label = getLabel( 'No kills found within 24h', labelOptions );
graphElement.prepend( label );
// reduce height
graphElement.animate({
height: '30px'
}, 200);
return;
}
// draw chart
Morris.Bar({
element: graphElement,
resize: true,
gridTextSize: 10,
gridTextColor: '#3c3f41',
gridTextFamily: 'Oxygen Bold',
hideHover: true,
data: tableData,
xkey: 'label',
ykeys: ['kills'],
labels: ['kills'],
xLabelMargin: 10,
padding: 10,
parseTime: false,
barColors: function (row, series, type) {
if (type === 'bar') {
// highlight last row -> recent kills found
if(this.xmax === row.x){
return '#c2760c';
}
}
return '#63676a';
}
});
// show hint for recent kills
if(tableData[tableData.length - 1].kills > 0){
labelOptions.type = 'label-warning';
label = getLabel( tableData[tableData.length - 1].kills + ' kills within the last hour!', labelOptions );
graphElement.prepend( label );
}
};
// get recent KB stats (last 24h))
var localDate = new Date();
// cache result for 5min
var cacheKey = systemId + '_' + localDate.getHours() + '_' + ( Math.ceil( localDate.getMinutes() / 5 ) * 5);
if(cache.systemKillsGraphData.hasOwnProperty(cacheKey) ){
drawGraph( cache.systemKillsGraphData[cacheKey] );
}else{
// chart data
var chartData = [];
for(var i = 0; i < showHours; i++){
var tempData = {
label: i + 'h',
kills: 0
};
chartData.push(tempData);
}
var serverDate= new Date(
localDate.getUTCFullYear(),
localDate.getUTCMonth(),
localDate.getUTCDate(),
localDate.getUTCHours(),
localDate.getUTCMinutes(),
localDate.getUTCSeconds()
);
// get all kills until current server time
var dateStringEnd = String( serverDate.getFullYear() );
dateStringEnd += String( ('0' + (serverDate.getMonth() + 1)).slice(-2) );
dateStringEnd += String( ('0' + serverDate.getDate()).slice(-2) );
dateStringEnd += String( ('0' + serverDate.getHours()).slice(-2) );
dateStringEnd += String( ('0' + serverDate.getMinutes()).slice(-2) );
// get start Date for kills API request (last 24h)
var startDate = new Date( serverDate.getTime() );
startDate.setDate( startDate.getDate() - 1);
var dateStringStart = String( startDate.getFullYear() );
dateStringStart += String( ('0' + (startDate.getMonth() + 1)).slice(-2) );
dateStringStart += String( ('0' + startDate.getDate()).slice(-2) );
dateStringStart += String( ('0' + startDate.getHours()).slice(-2) );
dateStringStart += String( ('0' + startDate.getMinutes()).slice(-2) );
var url = Config.url.zKillboard;
url += '/no-items/no-attackers/solarSystemID/' + systemId + '/startTime/' + dateStringStart + '/endTime/' + dateStringEnd + '/';
graphElement.showLoadingAnimation();
$.getJSON(url, function(kbData){
// the API wont return more than 200KMs ! - remember last bar block with complete KM information
var lastCompleteDiffHourData = 0;
// loop kills and count kills by hour
for(var i = 0; i < kbData.length; i++){
var match = kbData[i].killTime.match(/^(\d+)-(\d+)-(\d+) (\d+)\:(\d+)\:(\d+)$/);
var killDate = new Date(match[1], match[2] - 1, match[3], match[4], match[5], match[6]);
// get time diff
var timeDiffMin = Math.round( ( serverDate - killDate ) / 1000 / 60 );
var timeDiffHour = Math.round( timeDiffMin / 60 );
// update chart data
if(chartData[timeDiffHour]){
chartData[timeDiffHour].kills++;
if(timeDiffHour > lastCompleteDiffHourData){
lastCompleteDiffHourData = timeDiffHour;
}
}
}
// remove empty chart Data
if(kbData.length >= maxKillmailCount){
chartData = chartData.splice(0, lastCompleteDiffHourData + 1);
}
// change order
chartData.reverse();
// fill cache
cache.systemKillsGraphData[cacheKey] = {};
cache.systemKillsGraphData[cacheKey].tableData = chartData;
cache.systemKillsGraphData[cacheKey].count = kbData.length;
drawGraph( cache.systemKillsGraphData[cacheKey] );
parentElement.hideLoadingAnimation();
}).error(function(e){
Util.showNotify({title: e.status + ': Get system kills', text: 'Loading failed', type: 'error'});
});
}
};
$.fn.updateSystemInfoRoutes = function(systemFrom, systemsTo){
// TODO get cached routes from backend
var parentElement = $(this);
// crate new route table
var table = $('
', {
class: ['compact', 'stripe', 'order-column', 'row-border', config.systemInfoRoutesTableClass].join(' ')
});
parentElement.append( $(table) );
// init empty table
var routesTable = table.DataTable( {
paging: false,
ordering: true,
order: [ 1, 'asc' ],
info: false,
searching: false,
hover: false,
//autoWidth: false,
columnDefs: [
{
targets: 0,
//"orderData": 0,
orderable: true,
title: 'system'
},{
targets: 1,
orderable: true,
title: 'jumps  ',
width: '40px',
class: 'text-right'
},{
targets: 2,
orderable: false,
title: 'route'
}
],
data: [] // will be added dynamic
} );
$.each(systemsTo, function(i, systemTo){
if(systemFrom !== systemTo){
var cacheKey = systemFrom + '_' + systemTo;
// row class
var rowClass = config.systemInfoRoutesTableRowPrefix + i;
if(cache.systemRoutes.hasOwnProperty(cacheKey)){
// add new row from cache
routesTable.row.add( cache.systemRoutes[cacheKey] ).draw().nodes().to$().addClass( rowClass );
// init tooltips for each jump system
var tooltipElements = parentElement.find('.' + rowClass + ' [data-toggle="tooltip"]');
$(tooltipElements).tooltip();
}else{
// get route from API
var baseUrl = Config.url.eveCentral + 'route/from/';
var url = baseUrl + systemFrom + '/to/' + systemTo;
$.getJSON(url, function(routeData){
// add row Data
var rowData = [systemTo, routeData.length];
var jumpData = [];
// loop all systems on a rout
$.each(routeData, function(j, systemData){
var systemSecClass = config.systemSecurityClassPrefix;
var systemSec = systemData.to.security.toFixed(1).toString();
systemSecClass += systemSec.replace('.', '-');
var system = '';
jumpData.push( system );
});
rowData.push( jumpData.join(' ') );
cache.systemRoutes[cacheKey] = rowData;
// add new row
routesTable.row.add( cache.systemRoutes[cacheKey] ).draw().nodes().to$().addClass( rowClass );
// init tooltips for each jump system
var tooltipElements = parentElement.find('.' + rowClass + ' [data-toggle="tooltip"]');
$(tooltipElements).tooltip();
});
}
}
});
};
/**
* update Progressbar for all scanned signatures in a system
*/
$.fn.updateScannedSignaturesBar = function(options){
var moduleElement = $(this);
// get progress bar
var progressBarWrapper = $(this).find('.' + config.systemInfoProgressScannedClass);
var progressBar = $(progressBarWrapper).find('.progress-bar');
var progressBarLabel = $(progressBarWrapper).find('.progress-label-right');
var tableData = moduleElement.getSignatureTableData();
var percent = 0;
var progressBarType = 'progress-bar-danger';
if(tableData){
var sigCount = tableData.length;
var sigIncompleteCount = 0;
// check for signatures without "type" -> these are un scanned signatures
$.each(tableData, function(i, data){
var typeId = parseInt(data.typeId);
if(typeId === 0){
sigIncompleteCount++;
}
});
if(sigCount > 0){
percent = 100 - Math.round( 100 / sigCount * sigIncompleteCount );
}
if(percent < 30){
progressBarType = 'progress-bar-danger' ;
}else if(percent < 100){
progressBarType = 'progress-bar-warning';
}else{
progressBarType = 'progress-bar-success';
}
}
setTimeout(
function() {
progressBarLabel.text(percent + '%');
progressBar.removeClass().addClass('progress-bar').addClass(progressBarType);
progressBar.attr('aria-valuenow', percent);
progressBar.css({width: percent + '%'});
var notification = (sigCount - sigIncompleteCount) + ' / ' + sigCount + ' (' + percent + '%) signatures scanned';
// show notifications
if(options.showNotice !== false){
if(percent < 100){
Util.showNotify({title: 'Unknown signatures', text: notification, type: 'info'});
}else{
Util.showNotify({title: 'System is scanned', text: notification, type: 'success'});
}
}
}, 100);
};
/**
* make a table editable (in-line-editor)
* @param systemInfoData
*/
$.fn.makeEditable = function(systemInfoData){
var table = $(this);
// default x-editable options for each field
//$.fn.editable.defaults.url = '/post';
$.fn.editable.defaults.ajaxOptions = {
type: 'post',
dataType: 'json'
};
// find editable fields
var sigIdFields = table.find('.' + config.sigTableEditSigNameInput);
var sigTypeFields = table.find('.' + config.sigTableEditSigTypeSelect);
var sigNameFields = table.find('.' + config.sigTableEditSigNameSelect);
// jump to "next" editable field on save
var openNextEditDialogOnSave = function(fields, updateProgressBar){
fields.on('save', {test: 1}, function(e, data){
var that = this;
setTimeout(function() {
var nextField = getNextEditableField(that);
$(nextField).editable('show');
// update scanning progressbar if sig "type" has changed
if($(e.target).hasClass(config.sigTableEditSigTypeSelect)){
$(that).parents('.' + config.moduleClass).updateScannedSignaturesBar({showNotice: true});
}
}, 200);
});
};
// get the next editable field
var getNextEditableField = function(field){
return $(field).closest('td').next().find('.editable');
};
/**
* add map/system specific data for each editable field in the sig-table
* @param params
* @returns {*}
*/
var modifyFieldParamsOnSend = function(params){
params.systemId = systemInfoData.systemId;
params.mapId = systemInfoData.mapId;
return params;
};
// sigTableEditSigNameInput
sigIdFields.editable({
mode: 'popup',
type: 'text',
title: 'signature id',
name: 'name',
emptytext: '? ? ?',
validate: function(value) {
if($.trim(value) === '') {
return 'Signature id cant be empty';
}
},
params: modifyFieldParamsOnSend
});
// cache signature groups -----------------------------------------------------
var sigTypeCache = {};
// Select sig type (master)
sigTypeFields.editable({
mode: 'popup',
type: 'select',
title: 'signature type',
name: 'typeId',
emptytext: 'unknown',
params: modifyFieldParamsOnSend,
source: function(){
var systemType = $(this).attr('data-systemtype');
var areaId = $(this).attr('data-areaid');
var cacheKey = [systemType, areaId].join('_');
// check for cached signature names
if(sigTypeCache.hasOwnProperty( cacheKey )){
return sigTypeCache[cacheKey];
}
var availableTypes = {};
// get all available Signature Types
if(
Config.signatureTypes[systemType] &&
Config.signatureTypes[systemType][areaId]
){
// json object -> "translate" keys to names
var tempTypes = Config.signatureTypes[systemType][areaId];
availableTypes = Util.getSignatureGroupInfo('label');
// add empty option
availableTypes[0] = '';
availableTypes = sigTypeCache[cacheKey] = availableTypes;
}
return availableTypes;
},
success: function(response, newValue){
// find related "name" select (same row) and change options
var nameSelect = getNextEditableField(this);
var systemType = $(this).attr('data-systemtype');
var areaId = $(this).attr('data-areaid');
// set new Options
var newSelectOptions = Util.getAllSignatureNames(systemType, areaId, newValue);
$(nameSelect).editable('option', 'source', newSelectOptions);
$(nameSelect).editable('setValue', null);
if(newValue > 0){
$(nameSelect).editable('enable');
}else{
$(nameSelect).editable('disable');
}
}
});
// cache signature names -----------------------------------------------------------
var sigNameCache = {};
// Select sig name (slave: depends on sig type)
sigNameFields.editable({
mode: 'popup',
type: 'select',
title: 'signature name',
name: 'groupId',
emptytext: 'unknown',
params: modifyFieldParamsOnSend,
source: function(){
var systemType = $(this).attr('data-systemtype');
var areaId = $(this).attr('data-areaid');
var groupId = $(this).attr('data-groupId');
var cacheKey = [systemType, areaId, groupId].join('_');
// check for cached signature names
if(sigNameCache.hasOwnProperty( cacheKey )){
return sigNameCache[cacheKey];
}
var signatureNames = Util.getAllSignatureNames(systemType, areaId, groupId);
// add empty option
signatureNames[0] = '';
// get all available Signature Names
var availableSigs = sigNameCache[cacheKey] = signatureNames;
return availableSigs;
}
});
// open next field dialog
openNextEditDialogOnSave(sigIdFields, true);
openNextEditDialogOnSave(sigTypeFields, false);
// set button observer (delete sig)
$(this).find('.btn-danger').on('click', function(e){
e.preventDefault();
bootbox.confirm('Delete signature?', function(result) {
if(result){
// get module
var moduleElement = $(e.target).parents('.' + config.moduleClass);
// get clicked dataTable object
var currentTable = moduleElement.find('.' + config.sigTableMainClass);
currentTable = $(currentTable).dataTable();
// delete signature row
currentTable.fnDeleteRow($(e.target).parents('tr'));
// update signature bar
moduleElement.updateScannedSignaturesBar({showNotice: false});
Util.showNotify({title: 'Signature deleted', type: 'success'});
}
});
});
// init signature counter
$(this).find('.' + config.sigTableCounterClass + '[data-counter!="init"]').initSignatureCounter();
};
/**
* formats all signature data for table
* @param systemData
* @returns {Array}
*/
var formatSignatureData = function(systemData, options){
var formattedData = [];
if(
systemData &&
systemData.config &&
systemData.config.id &&
systemData.config.type === 'wh'
){
// system data "should" be valid :)
var systemSecurity = systemData.config.security;
var systemType = systemData.config.type;
var areaId = Util.getAreaIdBySecurity(systemSecurity);
// areaId is required as a key for signature names
if(areaId){
$.each(systemData.signatures, function(i, data){
if(! data.created){
data.created = Math.round( new Date().getTime() / 1000 );
}
if(! data.updated){
data.updated = Math.round( new Date().getTime() / 1000 );
}
var tempData = [];
// set signature name
var sigName = ' 0){
sigName += 'data-pk="' + data.id + '" ';
}
sigName += '>' + data.name + '';
tempData.push(sigName);
var sigType = ' 0){
sigType += 'data-pk="' + data.id + '" ';
}
sigType += 'data-systemType="' + systemType + '" ';
sigType += 'data-areaId="' + areaId + '" ';
sigType += 'data-value="' + data.groupId + '" ';
sigType += '>';
// set signature group id -----------------------------------------------
tempData.push( sigType );
var sigElement = ' 0){
sigElement += 'data-pk="' + data.id + '" ';
}
// set disabled if sig type is not selected
if(data.groupId < 1){
sigElement += 'data-disabled="1" ';
}
sigElement += 'data-systemType="' + systemType + '" ';
sigElement += 'data-areaId="' + areaId + '" ';
sigElement += 'data-groupId="' + data.groupId + '" ';
sigElement += 'data-value="' + data.typeId + '" ';
sigElement += '>';
// set signature type id ---------------------------------------------
tempData.push( sigElement );
// set Sig created
tempData.push( data.created );
// set Sig updated
tempData.push( data.updated );
// action icon
var actionButtonClass = 'btn-danger';
var actionButtonIcon = 'fa-close';
if(options.action){
actionButtonClass = options.action.buttonClass;
actionButtonIcon = options.action.buttonIcon;
}
var deleteButton = '';
deleteButton += ' ';
deleteButton += '';
tempData.push( deleteButton );
formattedData.push(tempData);
});
}
}
return formattedData;
};
/**
* TODO delete function !!!!!!!
*/
var tempFunctionGetSystemData = function(){
var data = {
config: {
id: 2,
name: 'J150020',
alias: 'Polaris',
effect: 'magnetar',
security: 'C6',
static: [{
lifetime: 24
}],
type: 'wh'
},
signatures: [
{
id: 2,
name: 'gdf',
typeId: 1,
groupId: 2,
created: 1325376000,
updated: 1415215936,
createdBy: 'Exodus4D',
updatedBy: 'Exodus4D'
},{
id: 6,
name: 'hfs',
typeId: 0,
groupId: 1,
created: 1415989953,
updated: 1415215936,
createdBy: 'Exodus4D',
updatedBy: 'Exodus4D'
},{
id: 8,
name: 'hfg',
typeId: 1,
groupId: 1,
created: 1415215936,
updated: 1415215936,
createdBy: 'Exodus4D',
updatedBy: 'Exodus4D'
},{
id: 12,
name: 'lld',
typeId: 1,
groupId: 1,
created: 1415215936,
updated: 1415215936,
createdBy: 'Exodus4D',
updatedBy: 'Exodus4D'
},{
id: 13,
name: 'dge',
typeId: 1,
groupId: 1,
created: 1394613252,
updated: 1415215936,
updatedBy: 'Exodus4D'
},{
id: 14,
name: 'exs',
typeId: 1,
groupId: 1,
created: 1415215936,
updated: 1415215936,
createdBy: 'Exodus4D',
updatedBy: 'Exodus4D'
},{
id: 15,
name: 'cvs',
typeId: 3,
groupId: 1,
created: 1415215936,
updated: 1386934983,
createdBy: 'Exodus4D',
updatedBy: 'Exodus4D'
},{
id: 16,
name: 'ggd',
typeId: 0,
groupId: 0,
created: 1415215936,
updated: 1415215936,
createdBy: 'Exodus4D',
updatedBy: 'Exodus4D'
},{
id: 18,
name: 'okd',
typeId: 1,
groupId: 1,
created: 1415215936,
updated: 1394613252,
createdBy: 'Exodus4D',
updatedBy: 'Exodus4D'
},{
id: 8,
name: 'dbe',
typeId: 3,
groupId: 1,
created: 1415215936,
updated: 1415215936,
createdBy: 'Exodus4D',
updatedBy: 'Exodus4D'
},{
id: 20,
name: 'asw',
typeId: 0,
groupId: 3,
created: 1415215936,
updated: 1386934983,
createdBy: 'Exodus4D',
updatedBy: 'Exodus4D'
},{
id: 22,
name: 'nfg',
typeId: 2,
groupId: 2,
created: 1415215936,
updated: 1415215936,
createdBy: 'Exodus4D',
updatedBy: 'Exodus4D'
}
]
};
return data;
};
/**
* updates complete map module (all maps)
* @param userData
* @returns {boolean}
*/
$.fn.updateMapModuleData = function(userData){
// get all active map elements for module
var mapElements = $(this).getMaps();
var currentUserData = null;
// current user data
if(userData.currentUserData){
currentUserData = userData.currentUserData;
}
// get map Data
for(var i = 0; i < mapElements.length; i++){
var mapElement = $(mapElements[i]);
var mapId = mapElement.data('id');
var mapUserData = null;
// get user data for each active map
for(var j = 0; j < userData.mapUserData.length; j++){
var tempMapData = userData.mapUserData[j];
if(tempMapData.config.id === mapId){
// map userData found
mapUserData = tempMapData;
break;
}
}
// update map
if(mapUserData){
mapElement.updateUserData(mapUserData, currentUserData);
}
}
return true;
};
/**
* load all structure elements into a TabsContent div (tab body)
*/
$.fn.initContentStructure = function(){
return this.each(function(){
// init bootstrap Grid
var contentStructure = $('', {
class: ['row', config.mapTabContentRow].join(' ')
}).append(
$('
', {
class: ['col-xs-12', 'col-md-8', config.mapTabContentCellFirst, config.mapTabContentCell].join(' ')
})
).append(
$('
', {
class: ['col-xs-6', 'col-md-4', config.mapTabContentCellSecond, config.mapTabContentCell].join(' ')
})
);
// append grid structure
$(this).append(contentStructure);
});
};
var getTabElementh = function(options){
var tabElement = $('
', {
id: config.mapTabElementId
});
var tabBar = $('
', {
class: ['nav', 'nav-tabs'].join(' '),
id: options.barId
}).attr('role', 'tablist');
var tabContent = $('', {
class: 'tab-content'
}).attr('data-map-tabs', options.barId);
tabElement.append(tabBar);
tabElement.append(tabContent);
return tabElement;
};
/**
* add a new tab to tab-map-module end returns the new objects
* @param options
* @returns {{listElement: (*|void), contentElement: (*|HTMLElement)}}
*/
$.fn.addTab = function(options){
var tabElement = $(this);
var tabBar = tabElement.find('ul.nav-tabs');
var tabContent = tabElement.find('div.tab-content');
var listElement = $('
- ', {
class: options.tabClasses.join(' ')
}).attr('role', 'presentation');
if(options.active === true){
listElement.addClass('active');
}
if(options.right === true){
listElement.addClass('pull-right');
}
// link element -------
var linkElement = $('', {
href: '#' + config.mapTabIdPrefix + options.id
}).attr('role', 'tab').attr('data-toggle', 'tab').data('map-id', options.id);
// icon element ------
var iconElement = $('', {
class: ['fa', 'fa-fw', options.icon].join(' ')
});
// text element -----
var textElement = $('', {
class: config.mapTabLinkTextClass,
text: options.name
});
var newListElement = listElement.append(
linkElement.append(iconElement).append(textElement)
);
tabBar.append( newListElement );
// tabs content ====================================
var contentElement = $('
', {
id: config.mapTabIdPrefix + options.id,
class: options.contentClasses.join(' ')
});
contentElement.addClass('tab-pane');
if(options.active === true){
contentElement.addClass('active');
}
tabContent.append(contentElement);
return {
listElement: newListElement,
contentElement: contentElement
};
};
/**
* deletes a map tab for a given map id
* @param mapId
*/
$.fn.deleteTab = function(mapId){
var tabElement = $(this);
var linkElement = tabElement.find('a[href="#' + config.mapTabIdPrefix + mapId + '"]');
var deletedTabName = '';
if(linkElement.length > 0){
deletedTabName = linkElement.find('.' + config.mapTabLinkTextClass).text();
var liElement = linkElement.parent();
var contentElement = tabElement.find('div[id="' + config.mapTabIdPrefix + mapId + '"]');
var findNewActiveTab = false;
// check if liElement was active
if(liElement.hasClass('active')){
// search any remaining li element and set active
findNewActiveTab = true;
}
liElement.remove();
contentElement.remove();
if(findNewActiveTab === true){
tabElement.find('a:first').tab('show');
}
}
return deletedTabName;
};
/**
* get current map data for a map id
* @param mapId
* @returns {boolean}
*/
var getMapDataById = function(mapId){
var mapData = false;
for(var i = 0; i < currentMapData.length; i++){
if(currentMapData[i].config.id === mapId){
mapData = currentMapData[i];
break;
}
}
return mapData;
};
/**
* load/update map module into element (all maps)
* @param mapData
* @returns {boolean}
*/
$.fn.updateMapModule = function(mapData){
// update current map data
currentMapData = mapData;
// temp store current map data to prevent data-change while function execution!
var tempMapData = currentMapData;
var mapModuleElement = $(this);
// check if tabs module is already loaded
var tabMapElement = $('#' + config.mapTabElementId);
// check if tabs have changed
var tabsChanged = false;
if(tabMapElement.length > 0){
// tab element already exists
var tabElements = getTabElements();
// mapIds that are currently active
var activeMapIds = [];
// check whether a tab/map is still active
for(var i = 0; i < tabElements.length; i++){
var mapId = $(tabElements[i]).data('map-id');
if(mapId > 0){
var tabMapData = getMapDataById(mapId);
if(tabMapData !== false){
// map data available -> update map
activeMapIds.push(mapId);
}else{
tabsChanged = true;
// map data not available -> remove tab
var deletedTabName = tabMapElement.deleteTab(mapId);
if(deletedTabName.length > 0){
Util.showNotify({title: 'Map removed', text: deletedTabName + ' deleted', type: 'warning'});
}
}
}
}
// add new tabs for new maps
$.each(tempMapData, function(i, data){
if( activeMapIds.indexOf( data.config.id ) === -1 ){
// add new map tab
var tabOptions = {
id: parseInt( data.config.id ),
tabClasses: [config.mapTabClass, Util.getInfoForMap( data.config.type, 'classTab') ],
contentClasses: [config.mapTabContentClass],
active: false,
icon: data.config.icon,
name: data.config.name,
right: false
};
var newTabElements = tabMapElement.addTab(tabOptions);
// set observer for manually triggered map events
newTabElements.contentElement.setTabContentObserver();
// load all the structure elements for the new tab
newTabElements.contentElement.initContentStructure();
Util.showNotify({title: 'Map added', text: data.config.name + ' added', type: 'success'});
}
});
}else{
// create Tab Element
tabsChanged = true;
var options = {
barId: config.mapTabBarId
};
tabMapElement = getTabElementh(options);
// add new tab for each map
for(var j = 0; j < tempMapData.length; j++){
var data = tempMapData[j];
var activeTab = false;
if(j === 0){
activeTab = true;
}
var tabOptions = {
id: parseInt( data.config.id ),
tabClasses: [config.mapTabClass, Util.getInfoForMap( data.config.type, 'classTab') ],
contentClasses: [config.mapTabContentClass],
active: activeTab,
icon: data.config.icon,
name: data.config.name,
right: false
};
tabMapElement.addTab(tabOptions);
}
// add "add" button
var tabAddOptions = {
id: 0,
tabClasses: [config.mapTabClass, Util.getInfoForMap( 'standard', 'classTab') ],
contentClasses: [config.mapTabContentClass],
icon: 'fa-plus',
name: 'add',
right: true
};
tabMapElement.addTab(tabAddOptions);
mapModuleElement.prepend(tabMapElement);
// ==============================================================
// this new created module
var tabContentElements = tabMapElement.find('.' + config.mapTabContentClass);
// set observer for manually triggered map events
tabContentElements.setTabContentObserver();
// load all the structure elements for ALL Tab Content Body
tabContentElements.initContentStructure();
// load first map i in first tab content container
$( tabContentElements[0] ).updateMapData(tempMapData[0]);
}
if(tabsChanged === true){
// remove previous event handlers
var allTabElements = getTabElements();
allTabElements.off('show.bs.tab');
allTabElements.off('shown.bs.tab');
allTabElements.off('hide.bs.tab');
// check for "new map" action before tap-change
allTabElements.on('show.bs.tab', function (e) {
var mapId = $(e.target).data('map-id');
if(mapId === 0){
// add new Tab selected
$(document).trigger('pf:menuEditMap', {newMap: true});
e.preventDefault();
}
});
// load new map right after tab-change
allTabElements.on('shown.bs.tab', function (e) {
var mapId = $(e.target).data('map-id');
var tabMapData = getMapDataById(mapId);
if(tabMapData !== false){
// load map
var currentTabContentElement = $('#' + config.mapTabIdPrefix + mapId);
$( currentTabContentElement).updateMapData( tabMapData);
// "wake up" scrollbar for map and get previous state back
var scrollableElement = currentTabContentElement.find('.' + config.mapWrapperClass);
$(scrollableElement).mCustomScrollbar( 'update');
}else{
// no map data found -> remove tab
tabMapElement.deleteTab(mapId);
}
});
allTabElements.on('hide.bs.tab', function (e, a) {
var newMapId = $(e.relatedTarget).data('map-id');
var oldMapId = $(e.target).data('map-id');
// disable map if new map is selected -> not "add button"
if(newMapId > 0){
var currentTabContentElement = $('#' + config.mapTabIdPrefix + oldMapId);
// disable scrollbar for map that will be hidden. "freeze" current state
var scrollableElement = currentTabContentElement.find('.' + config.mapWrapperClass);
$(scrollableElement).mCustomScrollbar( 'disable' );
}
});
}
return true;
};
/**
* collect all data for export/save from each active map in the map module
* @returns {Array}
*/
$.fn.getMapModuleData = function(){
// get all active map elements for module
var mapElements = $(this).getMaps();
var data = [];
for(var i = 0; i < mapElements.length; i++){
var mapData = $(mapElements[i]).getMapData();
if(mapData !== false){
data.push(mapData);
}
}
return data;
};
/**
* load OR updates a map module with its data
* @param mapData
* @returns {*}
*/
$.fn.updateMapData = function(mapData){
return this.each(function(){
$(this).loadMap(mapData);
});
};
});