X-Git-Url: http://wagnertech.de/git?a=blobdiff_plain;f=js%2Fkivi.js;h=210020a60480dd6d0ceb0e415337082f4dae82e1;hb=98aa76be5673cb73d70acecd3c34c55567600889;hp=17c0ab0563f428cac15eb2155f83ae2ec245220e;hpb=766f5705ecb9cd56adfbffd94c871959bb64c6fd;p=kivitendo-erp.git diff --git a/js/kivi.js b/js/kivi.js index 17c0ab056..210020a60 100644 --- a/js/kivi.js +++ b/js/kivi.js @@ -1,21 +1,142 @@ namespace("kivi", function(ns) { + "use strict"; + ns._locale = {}; + ns._date_format = { + sep: '.', + y: 2, + m: 1, + d: 0 + }; + ns._number_format = { + decimalSep: ',', + thousandSep: '.' + }; + + ns.setup_formats = function(params) { + var res = (params.dates || "").match(/^([ymd]+)([^a-z])([ymd]+)[^a-z]([ymd]+)$/); + if (res) { + ns._date_format = { sep: res[2] }; + ns._date_format[res[1].substr(0, 1)] = 0; + ns._date_format[res[3].substr(0, 1)] = 1; + ns._date_format[res[4].substr(0, 1)] = 2; + } + + res = (params.numbers || "").match(/^\d*([^\d]?)\d+([^\d])\d+$/); + if (res) + ns._number_format = { + decimalSep: res[2], + thousandSep: res[1] + }; + }; + + ns.parse_date = function(date) { + var parts = date.replace(/\s+/g, "").split(ns._date_format.sep); + date = new Date( + ((parts[ ns._date_format.y ] || 0) * 1) || (new Date()).getFullYear(), + (parts[ ns._date_format.m ] || 0) * 1 - 1, // Months are 0-based. + (parts[ ns._date_format.d ] || 0) * 1 + ); + + return isNaN(date.getTime()) ? undefined : date; + }; + + ns.format_date = function(date) { + if (isNaN(date.getTime())) + return undefined; + + var parts = [ "", "", "" ] + parts[ ns._date_format.y ] = date.getFullYear(); + parts[ ns._date_format.m ] = (date.getMonth() < 9 ? "0" : "") + (date.getMonth() + 1); // Months are 0-based, but days are 1-based. + parts[ ns._date_format.d ] = (date.getDate() < 10 ? "0" : "") + date.getDate(); + return parts.join(ns._date_format.sep); + }; + + ns.parse_amount = function(amount) { + if ((amount === undefined) || (amount === '')) + return 0; + + if (ns._number_format.decimalSep == ',') + amount = amount.replace(/\./g, "").replace(/,/g, "."); + + amount = amount.replace(/[\',]/g, "") + + // Make sure no code wich is not a math expression ends up in eval(). + if (!amount.match(/^[0-9 ()\-+*/.]*$/)) + return 0; + + /* jshint -W061 */ + try { + return eval(amount); + } catch (err) { + return 0; + } + }; + + ns.round_amount = function(amount, places) { + var neg = amount >= 0 ? 1 : -1; + var mult = Math.pow(10, places + 1); + var temp = Math.abs(amount) * mult; + var diff = Math.abs(1 - temp + Math.floor(temp)); + temp = Math.floor(temp) + (diff <= 0.00001 ? 1 : 0); + var dec = temp % 10; + temp += dec >= 5 ? 10 - dec: dec * -1; + + return neg * temp / mult; + }; + + ns.format_amount = function(amount, places) { + amount = amount || 0; + + if ((places !== undefined) && (places >= 0)) + amount = ns.round_amount(amount, Math.abs(places)); + + var parts = ("" + Math.abs(amount)).split(/\./); + var intg = parts[0]; + var dec = parts.length > 1 ? parts[1] : ""; + var sign = amount < 0 ? "-" : ""; + + if (places !== undefined) { + while (dec.length < Math.abs(places)) + dec += "0"; + + if ((places > 0) && (dec.length > Math.abs(places))) + dec = d.substr(0, places); + } + + if ((ns._number_format.thousandSep !== "") && (intg.length > 3)) { + var len = ((intg.length + 2) % 3) + 1, + start = len, + res = intg.substr(0, len); + while (start < intg.length) { + res += ns._number_format.thousandSep + intg.substr(start, 3); + start += 3; + } + + intg = res; + } + + var sep = (places !== 0) && (dec !== "") ? ns._number_format.decimalSep : ""; + + return sign + intg + sep + dec; + }; ns.t8 = function(text, params) { - var text = ns._locale[text] || text; + text = ns._locale[text] || text; + var key, value if( Object.prototype.toString.call( params ) === '[object Array]' ) { var len = params.length; for(var i=0; i 1)) - config[prop] = matches[1]; - }); + config.height = $e.height(); + config.width = $e.width(); - $e.ckeditor(config); + var editor = CKEDITOR.inline($e.get(0), config); + $e.data('ckeditorInstance', editor); if ($e.hasClass('texteditor-autofocus')) - $e.ckeditor(function() { ns.focus_ckeditor($e); }); + editor.on('instanceReady', function() { ns.focus_ckeditor($e); }); }; ns.reinit_widgets = function() { @@ -122,17 +241,36 @@ namespace("kivi", function(ns) { $(elt).datepicker(); }); - if (ns.PartPicker) - ns.run_once_for('input.part_autocomplete', 'part_picker', function(elt) { - kivi.PartPicker($(elt)); + if (ns.Part) ns.Part.reinit_widgets(); + if (ns.CustomerVendor) ns.CustomerVendor.reinit_widgets(); + + if (ns.ProjectPicker) + ns.run_once_for('input.project_autocomplete', 'project_picker', function(elt) { + kivi.ProjectPicker($(elt)); }); + if (ns.ChartPicker) + ns.run_once_for('input.chart_autocomplete', 'chart_picker', function(elt) { + kivi.ChartPicker($(elt)); + }); + + var func = kivi.get_function_by_name('local_reinit_widgets'); if (func) func(); - ns.run_once_for('.tooltip', 'tooltip', function(elt) { - $(elt).tooltip(); + ns.run_once_for('.tooltipster', 'tooltipster', function(elt) { + $(elt).tooltipster({ + contentAsHTML: false, + theme: 'tooltipster-light' + }) + }); + + ns.run_once_for('.tooltipster-html', 'tooltipster-html', function(elt) { + $(elt).tooltipster({ + contentAsHTML: true, + theme: 'tooltipster-light' + }) }); ns.run_once_for('.tabwidget', 'tabwidget', kivi.init_tabwidget); @@ -149,6 +287,31 @@ namespace("kivi", function(ns) { return true; }; + // This function submits an existing form given by "form_selector" + // and sets the "action" input to "action_to_call" before submitting + // it. Any existing input named "action" will be removed prior to + // submitting. + ns.submit_form_with_action = function(form_selector, action_to_call) { + $('[name=action]').remove(); + + var $form = $(form_selector); + var $hidden = $(''); + + $hidden.attr('name', 'action'); + $hidden.attr('value', action_to_call); + $form.append($hidden); + + $form.submit(); + }; + + // This function exists solely so that it can be found with + // kivi.get_functions_by_name() and called later on. Using something + // like "var func = history["back"]" works, but calling it later + // with "func.apply()" doesn't. + ns.history_back = function() { + history.back(); + }; + // Return a function object by its name (a string). Works both with // global functions (e.g. "check_right_date_format") and those in // namespaces (e.g. "kivi.t8"). @@ -163,19 +326,21 @@ namespace("kivi", function(ns) { // Open a modal jQuery UI popup dialog. The content can be either // loaded via AJAX (if the parameter 'url' is given) or simply // displayed if it exists in the DOM already (referenced via - // 'id'). If an existing DOM div should be used then the element - // won't be removed upon closing the dialog which allows re-opening - // it later on. + // 'id') or given via param.html. If an existing DOM div should be used then + // the element won't be removed upon closing the dialog which allows + // re-opening it later on. // // Parameters: // - id: dialog DIV ID (optional; defaults to 'jqueryui_popup_dialog') // - url, data, type: passed as the first three arguments to the $.ajax() call if an AJAX call is made, otherwise ignored. // - dialog: an optional object of options passed to the $.dialog() call + // - load: an optional function that is called after the content has been loaded successfully (only if an AJAX call is made) ns.popup_dialog = function(params) { var dialog; params = params || { }; var id = params.id || 'jqueryui_popup_dialog'; + var custom_close = params.dialog ? params.dialog.close : undefined; var dialog_params = $.extend( { // kivitendo default parameters: width: 800 @@ -185,10 +350,18 @@ namespace("kivi", function(ns) { // User supplied options: params.dialog || { }, { // Options that must not be changed: - close: function(event, ui) { if (params.url) dialog.remove(); else dialog.dialog('close'); } + close: function(event, ui) { + dialog.dialog('close'); + + if (custom_close) + custom_close(); + + if (params.url || params.html) + dialog.remove(); + } }); - if (!params.url) { + if (!params.url && !params.html) { // Use existing DOM element and show it. No AJAX call. dialog = $('#' + id) @@ -204,15 +377,22 @@ namespace("kivi", function(ns) { dialog = $('').appendTo('body'); dialog.dialog(dialog_params); - $.ajax({ - url: params.url, - data: params.data, - type: params.type, - success: function(new_html) { - dialog.html(new_html); - dialog.removeClass('loading'); - } - }); + if (params.html) { + dialog.html(params.html); + } else { + // no html? get it via ajax + $.ajax({ + url: params.url, + data: params.data, + type: params.type, + success: function(new_html) { + dialog.html(new_html); + dialog.removeClass('loading'); + if (params.load) + params.load(); + } + }); + } return true; }; @@ -238,7 +418,7 @@ namespace("kivi", function(ns) { return; } - $(selector).filter(function() { return $(this).data(attr_name) != true; }).each(function(idx, elt) { + $(selector).filter(function() { return $(this).data(attr_name) !== true; }).each(function(idx, elt) { var $elt = $(elt); $elt.data(attr_name, true); fn($elt); @@ -257,11 +437,115 @@ namespace("kivi", function(ns) { ns.run = function(function_name, args) { var fn = ns.get_function_by_name(function_name); if (fn) - return fn.apply({}, args); + return fn.apply({}, args || []); console.error('kivi.run("' + function_name + '"): No function by that name found'); return undefined; }; + + ns.detect_duplicate_ids_in_dom = function() { + var ids = {}, + found = false; + + $('[id]').each(function() { + if (this.id && ids[this.id]) { + found = true; + console.warn('Duplicate ID #' + this.id); + } + ids[this.id] = 1; + }); + + if (!found) + console.log('No duplicate IDs found :)'); + }; + + // Verifies that at least one checkbox matching the + // "checkbox_selector" is actually checked. If not, an error message + // is shown, and false is returned. Otherwise (at least one of them + // is checked) nothing is shown and true returned. + // + // Can be used in checks when clicking buttons. + ns.check_if_entries_selected = function(checkbox_selector) { + if ($(checkbox_selector + ':checked').length > 0) + return true; + + alert(kivi.t8('No entries have been selected.')); + + return false; + }; + + // Performs various validation steps on the descendants of + // 'selector'. Elements that should be validated must have an + // attribute named "data-validate" which is set to a space-separated + // list of tests to perform. Additionally, the attribute + // "data-title" must be set to a human-readable name of the field + // that can be shown as part of an error message. + // + // Supported validation tests are: + // - "required": the field must be set (its .val() must not be empty) + // + // The validation will abort and return "false" as soon as + // validation routine fails. + // + // The function returns "true" if all validations succeed for all + // elements. + ns.validate_form = function(selector) { + var validate_field = function(elt) { + var $elt = $(elt); + var tests = $elt.data('validate').split(/ +/); + var info = { + title: $elt.data('title'), + value: $elt.val(), + }; + + for (var test_idx in tests) { + var test = tests[test_idx]; + + if (test === "required") { + if ($elt.val() === '') { + alert(kivi.t8("The field '#{title}' must be set.", info)); + return false; + } + + } else { + var error = "kivi.validate_form: unknown test '" + test + "' for element ID '" + $elt.prop('id') + "'"; + console.error(error); + alert(error); + + return false; + } + } + + return true; + }; + + selector = selector || '#form'; + var ok = true; + var to_check = $(selector + ' [data-validate]').toArray(); + + for (var to_check_idx in to_check) + if (!validate_field(to_check[to_check_idx])) + return false; + + return true; + }; + + ns.switch_areainput_to_textarea = function(id) { + var $input = $('#' + id); + if (!$input.length) + return; + + var $area = $(''); + + $area.prop('rows', 3); + $area.prop('cols', $input.prop('size') || 40); + $area.prop('name', $input.prop('name')); + $area.prop('id', $input.prop('id')); + $area.val($input.val()); + + $input.parent().replaceWith($area); + $area.focus(); + }; }); kivi = namespace('kivi');