web/static/admin/js/inlines.js
author veltr
Fri, 11 Oct 2013 17:37:25 +0200
changeset 107 6b346cb90c5a
parent 0 ecdfc63274bf
permissions -rw-r--r--
bugfix
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
0
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
     1
/**
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
     2
 * Django admin inlines
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
     3
 *
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
     4
 * Based on jQuery Formset 1.1
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
     5
 * @author Stanislaus Madueke (stan DOT madueke AT gmail DOT com)
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
     6
 * @requires jQuery 1.2.6 or later
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
     7
 *
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
     8
 * Copyright (c) 2009, Stanislaus Madueke
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
     9
 * All rights reserved.
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
    10
 *
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
    11
 * Spiced up with Code from Zain Memon's GSoC project 2009
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
    12
 * and modified for Django by Jannis Leidel
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
    13
 *
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
    14
 * Licensed under the New BSD License
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
    15
 * See: http://www.opensource.org/licenses/bsd-license.php
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
    16
 */
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
    17
(function($) {
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
    18
	$.fn.formset = function(opts) {
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
    19
		var options = $.extend({}, $.fn.formset.defaults, opts);
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
    20
		var updateElementIndex = function(el, prefix, ndx) {
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
    21
			var id_regex = new RegExp("(" + prefix + "-\\d+)");
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
    22
			var replacement = prefix + "-" + ndx;
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
    23
			if ($(el).attr("for")) {
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
    24
				$(el).attr("for", $(el).attr("for").replace(id_regex, replacement));
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
    25
			}
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
    26
			if (el.id) {
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
    27
				el.id = el.id.replace(id_regex, replacement);
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
    28
			}
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
    29
			if (el.name) {
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
    30
				el.name = el.name.replace(id_regex, replacement);
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
    31
			}
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
    32
		};
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
    33
		var totalForms = $("#id_" + options.prefix + "-TOTAL_FORMS").attr("autocomplete", "off");
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
    34
		var maxForms = $("#id_" + options.prefix + "-MAX_NUM_FORMS").attr("autocomplete", "off");
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
    35
		// only show the add button if we are allowed to add more items,
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
    36
        // note that max_num = None translates to a blank string.
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
    37
		var showAddButton = maxForms.val() == '' || (maxForms.val()-totalForms.val()) > 0;
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
    38
		$(this).each(function(i) {
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
    39
			$(this).not("." + options.emptyCssClass).addClass(options.formCssClass);
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
    40
		});
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
    41
		if ($(this).length && showAddButton) {
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
    42
			var addButton;
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
    43
			if ($(this).attr("tagName") == "TR") {
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
    44
				// If forms are laid out as table rows, insert the
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
    45
				// "add" button in a new table row:
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
    46
				var numCols = this.eq(0).children().length;
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
    47
				$(this).parent().append('<tr class="' + options.addCssClass + '"><td colspan="' + numCols + '"><a href="javascript:void(0)">' + options.addText + "</a></tr>");
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
    48
				addButton = $(this).parent().find("tr:last a");
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
    49
			} else {
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
    50
				// Otherwise, insert it immediately after the last form:
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
    51
				$(this).filter(":last").after('<div class="' + options.addCssClass + '"><a href="javascript:void(0)">' + options.addText + "</a></div>");
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
    52
				addButton = $(this).filter(":last").next().find("a");
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
    53
			}
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
    54
			addButton.click(function() {
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
    55
				var totalForms = $("#id_" + options.prefix + "-TOTAL_FORMS");
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
    56
				var nextIndex = parseInt(totalForms.val());
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
    57
				var template = $("#" + options.prefix + "-empty");
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
    58
				var row = template.clone(true);
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
    59
				row.removeClass(options.emptyCssClass)
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
    60
				    .addClass(options.formCssClass)
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
    61
				    .attr("id", options.prefix + "-" + nextIndex)
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
    62
				    .insertBefore($(template));
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
    63
				row.find("*")
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
    64
				    .filter(function() {
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
    65
				        var el = $(this);
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
    66
				        return el.attr("id") && el.attr("id").search(/__prefix__/) >= 0;
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
    67
				    }).each(function() {
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
    68
				        var el = $(this);
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
    69
				        el.attr("id", el.attr("id").replace(/__prefix__/g, nextIndex));
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
    70
				    })
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
    71
				    .end()
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
    72
				    .filter(function() {
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
    73
				        var el = $(this);
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
    74
				        return el.attr("name") && el.attr("name").search(/__prefix__/) >= 0;
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
    75
				    }).each(function() {
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
    76
				        var el = $(this);
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
    77
				        el.attr("name", el.attr("name").replace(/__prefix__/g, nextIndex));
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
    78
				    });
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
    79
				if (row.is("tr")) {
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
    80
					// If the forms are laid out in table rows, insert
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
    81
					// the remove button into the last table cell:
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
    82
					row.children(":last").append('<div><a class="' + options.deleteCssClass +'" href="javascript:void(0)">' + options.deleteText + "</a></div>");
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
    83
				} else if (row.is("ul") || row.is("ol")) {
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
    84
					// If they're laid out as an ordered/unordered list,
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
    85
					// insert an <li> after the last list item:
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
    86
					row.append('<li><a class="' + options.deleteCssClass +'" href="javascript:void(0)">' + options.deleteText + "</a></li>");
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
    87
				} else {
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
    88
					// Otherwise, just insert the remove button as the
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
    89
					// last child element of the form's container:
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
    90
					row.children(":first").append('<span><a class="' + options.deleteCssClass + '" href="javascript:void(0)">' + options.deleteText + "</a></span>");
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
    91
				}
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
    92
				row.find("input,select,textarea,label,a").each(function() {
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
    93
					updateElementIndex(this, options.prefix, totalForms.val());
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
    94
				});
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
    95
				// Update number of total forms
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
    96
				$(totalForms).val(nextIndex + 1);
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
    97
				// Hide add button in case we've hit the max, except we want to add infinitely
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
    98
				if ((maxForms.val() != '') && (maxForms.val()-totalForms.val()) <= 0) {
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
    99
					addButton.parent().hide();
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
   100
				}
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
   101
				// The delete button of each row triggers a bunch of other things
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
   102
				row.find("a." + options.deleteCssClass).click(function() {
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
   103
					// Remove the parent form containing this button:
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
   104
					var row = $(this).parents("." + options.formCssClass);
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
   105
					row.remove();
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
   106
					// If a post-delete callback was provided, call it with the deleted form:
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
   107
					if (options.removed) {
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
   108
						options.removed(row);
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
   109
					}
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
   110
					// Update the TOTAL_FORMS form count.
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
   111
					var forms = $("." + options.formCssClass);
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
   112
					$("#id_" + options.prefix + "-TOTAL_FORMS").val(forms.length);
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
   113
					// Show add button again once we drop below max
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
   114
					if ((maxForms.val() == '') || (maxForms.val()-forms.length) > 0) {
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
   115
						addButton.parent().show();
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
   116
					}
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
   117
					// Also, update names and ids for all remaining form controls
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
   118
					// so they remain in sequence:
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
   119
					for (var i=0, formCount=forms.length; i<formCount; i++)
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
   120
					{
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
   121
						$(forms.get(i)).find("input,select,textarea,label,a").each(function() {
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
   122
							updateElementIndex(this, options.prefix, i);
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
   123
						});
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
   124
					}
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
   125
					return false;
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
   126
				});
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
   127
				// If a post-add callback was supplied, call it with the added form:
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
   128
				if (options.added) {
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
   129
					options.added(row);
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
   130
				}
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
   131
				return false;
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
   132
			});
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
   133
		}
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
   134
		return this;
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
   135
	}
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
   136
	/* Setup plugin defaults */
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
   137
	$.fn.formset.defaults = {
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
   138
		prefix: "form",					// The form prefix for your django formset
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
   139
		addText: "add another",			// Text for the add link
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
   140
		deleteText: "remove",			// Text for the delete link
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
   141
		addCssClass: "add-row",			// CSS class applied to the add link
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
   142
		deleteCssClass: "delete-row",	// CSS class applied to the delete link
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
   143
		emptyCssClass: "empty-row",		// CSS class applied to the empty row
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
   144
		formCssClass: "dynamic-form",	// CSS class applied to each form in a formset
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
   145
		added: null,					// Function called each time a new form is added
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
   146
		removed: null					// Function called each time a form is deleted
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
   147
	}
ecdfc63274bf first import
ymh <ymh.work@gmail.com>
parents:
diff changeset
   148
})(django.jQuery);