diff --git a/registration-system/admin/pages_cost.php b/registration-system/admin/pages_cost.php index a4e15fe53210fa4f3c8b6cacdb796775f584735e..c3d365b08385e6f23156fef5feae80536d5b2179 100644 --- a/registration-system/admin/pages_cost.php +++ b/registration-system/admin/pages_cost.php @@ -8,6 +8,12 @@ global $config_studitypen, $config_reisearten, $config_essen, $admin_db, $config_current_fahrt_id, $config_admin_verbose_level, $config_verbose_level, $text, $headers, $ajax; + +$headers .= '<script type="text/javascript" src="../view/js/jquery-1.11.1.min.js"></script> + <script src="../view/js/jquery.tabletojson.js"></script> + <script src="../view/js/jquery.jeditable.js"></script>'; + + $text .= "<h3>Kalkulation</h3>"; $h1 = array("Position", "Anzahl (normal)", "Satz", "Summe"); $s1[3] = array(1,2); @@ -18,7 +24,9 @@ $d1 = array(array("Reisekosten", 2, 1.9), array("Bettwäsche", 1, 4), array("Grillen", 1, 0.3), array("Kurzreisezuschlag", 1, 2)); -$text .= html_table($h1, $d1, $s1, $t1); +$text .= html_table($h1, $d1, $s1, $t1, "test"); +$text .= '<span id="anc_add">add</span> - <span id="anc_rem">rem</span>'; + $text .= "<h2>Einkaufen</h2>"; $h2 = array("Position", "Anzahl", "Satz", "Summe"); @@ -71,7 +79,46 @@ $text .= '</div><div style="float:left">'; $text .= html_table($h4, $d4_in, $s4, $t4); $text .= '</div><div style="clear:both"></div>'; +$text .="<script type='text/javascript'> +$('#testy').click( function() { + var table = $('#test').tableToJSON(); // Convert the table into a javascript object + console.log(table); + alert(JSON.stringify(table)); +}); + +function ref_editable(){ + $('.edita').editable(function(value, settings){return(value);}, + { + indicator : '<img src=\'img/indicator.gif\'>', + tooltip : 'Click to edit...', + style : 'inherit', + callback : function(value, settings){ + var table = $('#test').tableToJSON(); // Convert the table into a javascript object + console.log(JSON.stringify(table)); + } + } + ); +} +$(function(){ + ref_editable(); + + var cnt = 2; + $('#anc_add').click(function(){ + $('#test>tbody tr').last().after('<tr><td class=\'edita\'>Static Content ['+cnt+']</td><td class=\'edita\'></td><td class=\'edita\'></td><td class=\'edita\'></td></tr>'); + cnt++; + ref_editable(); + }); + + $('#anc_rem').click(function(){ + if($('#test>tbody tr').size()>1){ + $('#test>tbody tr:last-child').remove(); + }else{ + alert('One row should be present in table'); + } + }); +}); +</script>"; /** * $headers @@ -86,10 +133,10 @@ $text .= '</div><div style="clear:both"></div>'; * * return value: variable containing the html code for echo */ -function html_table($header, $data, $sum = array(), $type = array()){ +function html_table($header, $data, $sum = array(), $type = array(), $id=""){ $summy = array(); - $ret = "<table class=\"cost-table\"> + $ret = "<table class=\"cost-table\" id=\"".$id."\"> <thead> <tr>\n"; foreach($header as $h) @@ -97,10 +144,12 @@ function html_table($header, $data, $sum = array(), $type = array()){ $ret.=" </tr> </thead> <tbody>\n"; + $cnt = 0; foreach($data as $row){ $ret.="<tr>"; for($i = 0; $i < count($header); $i++){ - $ret.= "<td".numeric_class($row, $sum, $i).">"; + $cnt++; + $ret.= "<td".numeric_class($row, $sum, $i)." id='cell".$id.$cnt."'>"; if(isset($row[$i])){ $ret .= prepval($row[$i],(isset($type[$i]) ? $type[$i] : "")); if(isset($sum[$i])){ @@ -155,11 +204,12 @@ function prepval($val, $post){ } function numeric_class($a,$b,$c){ - $d = ' class="cost-table-numeric"'; + $d = ' class="cost-table-numeric edita"'; if(isset($a[$c])){ if(is_numeric($a[$c])) return $d; } if(isset($b[$c])) return $d; + return ' class="edita"'; } diff --git a/registration-system/view/js/jquery.jeditable.js b/registration-system/view/js/jquery.jeditable.js new file mode 100644 index 0000000000000000000000000000000000000000..1d066d45b1ac99b47fbac1e5eacd5b79463884d5 --- /dev/null +++ b/registration-system/view/js/jquery.jeditable.js @@ -0,0 +1,546 @@ +/* + * Jeditable - jQuery in place edit plugin + * + * Copyright (c) 2006-2013 Mika Tuupola, Dylan Verheul + * + * Licensed under the MIT license: + * http://www.opensource.org/licenses/mit-license.php + * + * Project home: + * http://www.appelsiini.net/projects/jeditable + * + * Based on editable by Dylan Verheul <dylan_at_dyve.net>: + * http://www.dyve.net/jquery/?editable + * + */ + +/** + * Version 1.7.3 + * + * ** means there is basic unit tests for this parameter. + * + * @name Jeditable + * @type jQuery + * @param String target (POST) URL or function to send edited content to ** + * @param Hash options additional options + * @param String options[method] method to use to send edited content (POST or PUT) ** + * @param Function options[callback] Function to run after submitting edited content ** + * @param String options[name] POST parameter name of edited content + * @param String options[id] POST parameter name of edited div id + * @param Hash options[submitdata] Extra parameters to send when submitting edited content. + * @param String options[type] text, textarea or select (or any 3rd party input type) ** + * @param Integer options[rows] number of rows if using textarea ** + * @param Integer options[cols] number of columns if using textarea ** + * @param Mixed options[height] 'auto', 'none' or height in pixels ** + * @param Mixed options[width] 'auto', 'none' or width in pixels ** + * @param String options[loadurl] URL to fetch input content before editing ** + * @param String options[loadtype] Request type for load url. Should be GET or POST. + * @param String options[loadtext] Text to display while loading external content. + * @param Mixed options[loaddata] Extra parameters to pass when fetching content before editing. + * @param Mixed options[data] Or content given as paramameter. String or function.** + * @param String options[indicator] indicator html to show when saving + * @param String options[tooltip] optional tooltip text via title attribute ** + * @param String options[event] jQuery event such as 'click' of 'dblclick' ** + * @param String options[submit] submit button value, empty means no button ** + * @param String options[cancel] cancel button value, empty means no button ** + * @param String options[cssclass] CSS class to apply to input form. 'inherit' to copy from parent. ** + * @param String options[style] Style to apply to input form 'inherit' to copy from parent. ** + * @param String options[select] true or false, when true text is highlighted ?? + * @param String options[placeholder] Placeholder text or html to insert when element is empty. ** + * @param String options[onblur] 'cancel', 'submit', 'ignore' or function ?? + * + * @param Function options[onsubmit] function(settings, original) { ... } called before submit + * @param Function options[onreset] function(settings, original) { ... } called before reset + * @param Function options[onerror] function(settings, original, xhr) { ... } called on error + * + * @param Hash options[ajaxoptions] jQuery Ajax options. See docs.jquery.com. + * + */ + +(function($) { + + $.fn.editable = function(target, options) { + + if ('disable' == target) { + $(this).data('disabled.editable', true); + return; + } + if ('enable' == target) { + $(this).data('disabled.editable', false); + return; + } + if ('destroy' == target) { + $(this) + .unbind($(this).data('event.editable')) + .removeData('disabled.editable') + .removeData('event.editable'); + return; + } + + var settings = $.extend({}, $.fn.editable.defaults, {target:target}, options); + + /* setup some functions */ + var plugin = $.editable.types[settings.type].plugin || function() { }; + var submit = $.editable.types[settings.type].submit || function() { }; + var buttons = $.editable.types[settings.type].buttons + || $.editable.types['defaults'].buttons; + var content = $.editable.types[settings.type].content + || $.editable.types['defaults'].content; + var element = $.editable.types[settings.type].element + || $.editable.types['defaults'].element; + var reset = $.editable.types[settings.type].reset + || $.editable.types['defaults'].reset; + var callback = settings.callback || function() { }; + var onedit = settings.onedit || function() { }; + var onsubmit = settings.onsubmit || function() { }; + var onreset = settings.onreset || function() { }; + var onerror = settings.onerror || reset; + + /* Show tooltip. */ + if (settings.tooltip) { + $(this).attr('title', settings.tooltip); + } + + settings.autowidth = 'auto' == settings.width; + settings.autoheight = 'auto' == settings.height; + + return this.each(function() { + + /* Save this to self because this changes when scope changes. */ + var self = this; + + /* Inlined block elements lose their width and height after first edit. */ + /* Save them for later use as workaround. */ + var savedwidth = $(self).width(); + var savedheight = $(self).height(); + + /* Save so it can be later used by $.editable('destroy') */ + $(this).data('event.editable', settings.event); + + /* If element is empty add something clickable (if requested) */ + if (!$.trim($(this).html())) { + $(this).html(settings.placeholder); + } + + $(this).bind(settings.event, function(e) { + + /* Abort if element is disabled. */ + if (true === $(this).data('disabled.editable')) { + return; + } + + /* Prevent throwing an exeption if edit field is clicked again. */ + if (self.editing) { + return; + } + + /* Abort if onedit hook returns false. */ + if (false === onedit.apply(this, [settings, self])) { + return; + } + + /* Prevent default action and bubbling. */ + e.preventDefault(); + e.stopPropagation(); + + /* Remove tooltip. */ + if (settings.tooltip) { + $(self).removeAttr('title'); + } + + /* Figure out how wide and tall we are, saved width and height. */ + /* Workaround for http://dev.jquery.com/ticket/2190 */ + if (0 == $(self).width()) { + settings.width = savedwidth; + settings.height = savedheight; + } else { + if (settings.width != 'none') { + settings.width = + settings.autowidth ? $(self).width() : settings.width; + } + if (settings.height != 'none') { + settings.height = + settings.autoheight ? $(self).height() : settings.height; + } + } + + /* Remove placeholder text, replace is here because of IE. */ + if ($(this).html().toLowerCase().replace(/(;|"|\/)/g, '') == + settings.placeholder.toLowerCase().replace(/(;|"|\/)/g, '')) { + $(this).html(''); + } + + self.editing = true; + self.revert = $(self).html(); + $(self).html(''); + + /* Create the form object. */ + var form = $('<form />'); + + /* Apply css or style or both. */ + if (settings.cssclass) { + if ('inherit' == settings.cssclass) { + form.attr('class', $(self).attr('class')); + } else { + form.attr('class', settings.cssclass); + } + } + + if (settings.style) { + if ('inherit' == settings.style) { + form.attr('style', $(self).attr('style')); + /* IE needs the second line or display wont be inherited. */ + form.css('display', $(self).css('display')); + } else { + form.attr('style', settings.style); + } + } + + /* Add main input element to form and store it in input. */ + var input = element.apply(form, [settings, self]); + + /* Set input content via POST, GET, given data or existing value. */ + var input_content; + + if (settings.loadurl) { + var t = setTimeout(function() { + input.disabled = true; + content.apply(form, [settings.loadtext, settings, self]); + }, 100); + + var loaddata = {}; + loaddata[settings.id] = self.id; + if ($.isFunction(settings.loaddata)) { + $.extend(loaddata, settings.loaddata.apply(self, [self.revert, settings])); + } else { + $.extend(loaddata, settings.loaddata); + } + $.ajax({ + type : settings.loadtype, + url : settings.loadurl, + data : loaddata, + async : false, + success: function(result) { + window.clearTimeout(t); + input_content = result; + input.disabled = false; + } + }); + } else if (settings.data) { + input_content = settings.data; + if ($.isFunction(settings.data)) { + input_content = settings.data.apply(self, [self.revert, settings]); + } + } else { + input_content = self.revert; + } + content.apply(form, [input_content, settings, self]); + + input.attr('name', settings.name); + + /* Add buttons to the form. */ + buttons.apply(form, [settings, self]); + + /* Add created form to self. */ + $(self).append(form); + + /* Attach 3rd party plugin if requested. */ + plugin.apply(form, [settings, self]); + + /* Focus to first visible form element. */ + $(':input:visible:enabled:first', form).focus(); + + /* Highlight input contents when requested. */ + if (settings.select) { + input.select(); + } + + /* discard changes if pressing esc */ + input.keydown(function(e) { + if (e.keyCode == 27) { + e.preventDefault(); + reset.apply(form, [settings, self]); + } + }); + + /* Discard, submit or nothing with changes when clicking outside. */ + /* Do nothing is usable when navigating with tab. */ + var t; + if ('cancel' == settings.onblur) { + input.blur(function(e) { + /* Prevent canceling if submit was clicked. */ + t = setTimeout(function() { + reset.apply(form, [settings, self]); + }, 500); + }); + } else if ('submit' == settings.onblur) { + input.blur(function(e) { + /* Prevent double submit if submit was clicked. */ + t = setTimeout(function() { + form.submit(); + }, 200); + }); + } else if ($.isFunction(settings.onblur)) { + input.blur(function(e) { + settings.onblur.apply(self, [input.val(), settings]); + }); + } else { + input.blur(function(e) { + /* TODO: maybe something here */ + }); + } + + form.submit(function(e) { + + if (t) { + clearTimeout(t); + } + + /* Do no submit. */ + e.preventDefault(); + + /* Call before submit hook. */ + /* If it returns false abort submitting. */ + if (false !== onsubmit.apply(form, [settings, self])) { + /* Custom inputs call before submit hook. */ + /* If it returns false abort submitting. */ + if (false !== submit.apply(form, [settings, self])) { + + /* Check if given target is function */ + if ($.isFunction(settings.target)) { + var str = settings.target.apply(self, [input.val(), settings]); + $(self).html(str); + self.editing = false; + callback.apply(self, [self.innerHTML, settings]); + /* TODO: this is not dry */ + if (!$.trim($(self).html())) { + $(self).html(settings.placeholder); + } + } else { + /* Add edited content and id of edited element to POST. */ + var submitdata = {}; + submitdata[settings.name] = input.val(); + submitdata[settings.id] = self.id; + /* Add extra data to be POST:ed. */ + if ($.isFunction(settings.submitdata)) { + $.extend(submitdata, settings.submitdata.apply(self, [self.revert, settings])); + } else { + $.extend(submitdata, settings.submitdata); + } + + /* Quick and dirty PUT support. */ + if ('PUT' == settings.method) { + submitdata['_method'] = 'put'; + } + + /* Show the saving indicator. */ + $(self).html(settings.indicator); + + /* Defaults for ajaxoptions. */ + var ajaxoptions = { + type : 'POST', + data : submitdata, + dataType: 'html', + url : settings.target, + success : function(result, status) { + if (ajaxoptions.dataType == 'html') { + $(self).html(result); + } + self.editing = false; + callback.apply(self, [result, settings]); + if (!$.trim($(self).html())) { + $(self).html(settings.placeholder); + } + }, + error : function(xhr, status, error) { + onerror.apply(form, [settings, self, xhr]); + } + }; + + /* Override with what is given in settings.ajaxoptions. */ + $.extend(ajaxoptions, settings.ajaxoptions); + $.ajax(ajaxoptions); + + } + } + } + + /* Show tooltip again. */ + $(self).attr('title', settings.tooltip); + + return false; + }); + }); + + /* Privileged methods */ + this.reset = function(form) { + /* Prevent calling reset twice when blurring. */ + if (this.editing) { + /* Before reset hook, if it returns false abort reseting. */ + if (false !== onreset.apply(form, [settings, self])) { + $(self).html(self.revert); + self.editing = false; + if (!$.trim($(self).html())) { + $(self).html(settings.placeholder); + } + /* Show tooltip again. */ + if (settings.tooltip) { + $(self).attr('title', settings.tooltip); + } + } + } + }; + }); + + }; + + + $.editable = { + types: { + defaults: { + element : function(settings, original) { + var input = $('<input type="hidden"></input>'); + $(this).append(input); + return(input); + }, + content : function(string, settings, original) { + $(':input:first', this).val(string); + }, + reset : function(settings, original) { + original.reset(this); + }, + buttons : function(settings, original) { + var form = this; + if (settings.submit) { + /* If given html string use that. */ + if (settings.submit.match(/>$/)) { + var submit = $(settings.submit).click(function() { + if (submit.attr("type") != "submit") { + form.submit(); + } + }); + /* Otherwise use button with given string as text. */ + } else { + var submit = $('<button type="submit" />'); + submit.html(settings.submit); + } + $(this).append(submit); + } + if (settings.cancel) { + /* If given html string use that. */ + if (settings.cancel.match(/>$/)) { + var cancel = $(settings.cancel); + /* otherwise use button with given string as text */ + } else { + var cancel = $('<button type="cancel" />'); + cancel.html(settings.cancel); + } + $(this).append(cancel); + + $(cancel).click(function(event) { + if ($.isFunction($.editable.types[settings.type].reset)) { + var reset = $.editable.types[settings.type].reset; + } else { + var reset = $.editable.types['defaults'].reset; + } + reset.apply(form, [settings, original]); + return false; + }); + } + } + }, + text: { + element : function(settings, original) { + var input = $('<input />'); + if (settings.width != 'none') { input.attr('width', settings.width); } + if (settings.height != 'none') { input.attr('height', settings.height); } + /* https://bugzilla.mozilla.org/show_bug.cgi?id=236791 */ + //input[0].setAttribute('autocomplete','off'); + input.attr('autocomplete','off'); + $(this).append(input); + return(input); + } + }, + textarea: { + element : function(settings, original) { + var textarea = $('<textarea />'); + if (settings.rows) { + textarea.attr('rows', settings.rows); + } else if (settings.height != "none") { + textarea.height(settings.height); + } + if (settings.cols) { + textarea.attr('cols', settings.cols); + } else if (settings.width != "none") { + textarea.width(settings.width); + } + $(this).append(textarea); + return(textarea); + } + }, + select: { + element : function(settings, original) { + var select = $('<select />'); + $(this).append(select); + return(select); + }, + content : function(data, settings, original) { + /* If it is string assume it is json. */ + if (String == data.constructor) { + eval ('var json = ' + data); + } else { + /* Otherwise assume it is a hash already. */ + var json = data; + } + for (var key in json) { + if (!json.hasOwnProperty(key)) { + continue; + } + if ('selected' == key) { + continue; + } + var option = $('<option />').val(key).append(json[key]); + $('select', this).append(option); + } + /* Loop option again to set selected. IE needed this... */ + $('select', this).children().each(function() { + if ($(this).val() == json['selected'] || + $(this).text() == $.trim(original.revert)) { + $(this).attr('selected', 'selected'); + } + }); + /* Submit on change if no submit button defined. */ + if (!settings.submit) { + var form = this; + $('select', this).change(function() { + form.submit(); + }); + } + } + } + }, + + /* Add new input type */ + addInputType: function(name, input) { + $.editable.types[name] = input; + } + }; + + /* Publicly accessible defaults. */ + $.fn.editable.defaults = { + name : 'value', + id : 'id', + type : 'text', + width : 'auto', + height : 'auto', + event : 'click.editable', + onblur : 'cancel', + loadtype : 'GET', + loadtext : 'Loading...', + placeholder: 'Click to edit', + loaddata : {}, + submitdata : {}, + ajaxoptions: {} + }; + +})(jQuery); diff --git a/registration-system/view/js/jquery.tabletojson.js b/registration-system/view/js/jquery.tabletojson.js new file mode 100644 index 0000000000000000000000000000000000000000..b060edbceef402e6315eb054b86ed2546d5ea065 --- /dev/null +++ b/registration-system/view/js/jquery.tabletojson.js @@ -0,0 +1,136 @@ +(function( $ ) { + 'use strict'; + + $.fn.tableToJSON = function(opts) { + + // Set options + var defaults = { + ignoreColumns: [], + onlyColumns: null, + ignoreHiddenRows: true, + headings: null, + allowHTML: false + }; + opts = $.extend(defaults, opts); + + var notNull = function(value) { + return value !== undefined && value !== null; + }; + + var ignoredColumn = function(index) { + if( notNull(opts.onlyColumns) ) { + return $.inArray(index, opts.onlyColumns) === -1; + } + return $.inArray(index, opts.ignoreColumns) !== -1; + }; + + var arraysToHash = function(keys, values) { + var result = {}, index = 0; + $.each(values, function(i, value) { + // when ignoring columns, the header option still starts + // with the first defined column + if ( index < keys.length && notNull(value) ) { + result[ keys[index] ] = value; + index++; + } + }); + return result; + }; + + var cellValues = function(cellIndex, cell) { + var value, result; + var override = $(cell).data('override'); + if ( opts.allowHTML ) { + value = $.trim($(cell).html()); + } else { + value = $.trim($(cell).text()); + } + result = notNull(override) ? override : value; + return result; + }; + + var rowValues = function(row) { + var result = []; + $(row).children('td,th').each(function(cellIndex, cell) { + result.push( cellValues(cellIndex, cell) ); + }); + return result; + }; + + var getHeadings = function(table) { + var firstRow = table.find('tr:first').first(); + return notNull(opts.headings) ? opts.headings : rowValues(firstRow); + }; + + var construct = function(table, headings) { + var i, j, len, len2, txt, $row, $cell, + tmpArray = [], cellIndex = 0, result = []; + table.children('tbody,*').children('tr').each(function(rowIndex, row) { + if( rowIndex > 0 || notNull(opts.headings) ) { + $row = $(row); + if( $row.is(':visible') || !opts.ignoreHiddenRows ) { + if (!tmpArray[rowIndex]) { + tmpArray[rowIndex] = []; + } + cellIndex = 0; + $row.children().each(function(){ + $cell = $(this); + + // process rowspans + if ($cell.filter('[rowspan]').length) { + len = parseInt( $cell.attr('rowspan'), 10) - 1; + txt = cellValues(cellIndex, $cell, []); + for (i = 1; i <= len; i++) { + if (!tmpArray[rowIndex + i]) { tmpArray[rowIndex + i] = []; } + tmpArray[rowIndex + i][cellIndex] = txt; + } + } + // process colspans + if ($cell.filter('[colspan]').length) { + len = parseInt( $cell.attr('colspan'), 10) - 1; + txt = cellValues(cellIndex, $cell, []); + for (i = 1; i <= len; i++) { + // cell has both col and row spans + if ($cell.filter('[rowspan]').length) { + len2 = parseInt( $cell.attr('rowspan'), 10); + for (j = 0; j < len2; j++) { + tmpArray[rowIndex + j][cellIndex + i] = txt; + } + } else { + tmpArray[rowIndex][cellIndex + i] = txt; + } + } + } + // skip column if already defined + while (tmpArray[rowIndex][cellIndex]) { cellIndex++; } + txt = tmpArray[rowIndex][cellIndex] || cellValues(cellIndex, $cell, []); + if (notNull(txt)) { + tmpArray[rowIndex][cellIndex] = txt; + } + cellIndex++; + }); + } + } + }); + $.each(tmpArray, function( i, row ){ + if (notNull(row)) { + // remove ignoredColumns / add onlyColumns + var newRow = notNull(opts.onlyColumns) || opts.ignoreColumns.length ? + $.grep(row, function(v, index){ return !ignoredColumn(index); }) : row, + + // remove ignoredColumns / add onlyColumns if headings is not defined + newHeadings = notNull(opts.headings) ? headings : + $.grep(headings, function(v, index){ return !ignoredColumn(index); }); + + txt = arraysToHash(newHeadings, newRow); + result[result.length] = txt; + } + }); + return result; + }; + + // Run + var headings = getHeadings(this); + return construct(this, headings); + }; +})( jQuery ); \ No newline at end of file