Bez popisu

jquery.grp_autocomplete_m2m.js 9.0KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204
  1. /**
  2. * GRAPPELLI AUTOCOMPLETE M2M
  3. * many-to-many lookup with autocomplete
  4. */
  5. (function($){
  6. var methods = {
  7. init: function(options) {
  8. options = $.extend({}, $.fn.grp_autocomplete_m2m.defaults, options);
  9. return this.each(function() {
  10. var $this = $(this);
  11. // assign attributes
  12. $this.attr({
  13. "tabindex": "-1",
  14. "readonly": "readonly"
  15. }).addClass("grp-autocomplete-hidden-field");
  16. // build autocomplete wrapper
  17. $this.next().after(loader).after(remove_link($this.attr('id')));
  18. $this.parent().wrapInner("<div class='grp-autocomplete-wrapper-m2m'></div>");
  19. //$this.parent().prepend("<ul class='search'><li class='search'><input id='" + $this.attr("id") + "-autocomplete' type='text' class='vTextField' value='' /></li></ul>").prepend("<ul class='repr'></ul>");
  20. $this.parent().prepend("<ul class='grp-repr'><li class='grp-search'><input id='" + $this.attr("id") + "-autocomplete' type='text' class='vTextField' value='' /></li></ul>");
  21. // defaults
  22. options = $.extend({
  23. wrapper_autocomplete: $this.parent(),
  24. wrapper_repr: $this.parent().find("ul.grp-repr"),
  25. wrapper_search: $this.parent().find("li.grp-search"),
  26. remove_link: $this.next().next().hide(),
  27. loader: $this.next().next().next().hide()
  28. }, $.fn.grp_autocomplete_m2m.defaults, options);
  29. // move errorlist outside the wrapper
  30. if ($this.parent().find("ul.errorlist")) {
  31. $this.parent().find("ul.errorlist").detach().appendTo($this.parent().parent());
  32. }
  33. // move helptext outside the wrapper
  34. if ($this.parent().find("p.grp-help")) {
  35. $this.parent().find("p.grp-help").detach().appendTo($this.parent().parent());
  36. }
  37. // lookup
  38. lookup_id($this, options); // lookup when loading page
  39. lookup_autocomplete($this, options); // autocomplete-handler
  40. $this.on("change focus keyup", function() { // id-handler
  41. lookup_id($this, options);
  42. });
  43. // labels
  44. $("label[for='"+$this.attr('id')+"']").each(function() {
  45. $(this).attr("for", $this.attr("id")+"-autocomplete");
  46. });
  47. // click on div > focus input
  48. options.wrapper_autocomplete.on("click", function(e) {
  49. // prevent focus when clicking on remove/select
  50. if (!$(e.target).hasClass("related-lookup") && !$(e.target).hasClass("grp-related-remove")) {
  51. options.wrapper_search.find("input:first").trigger("focus");
  52. }
  53. });
  54. });
  55. }
  56. };
  57. $.fn.grp_autocomplete_m2m = function(method) {
  58. if (methods[method]) {
  59. return methods[method].apply(this, Array.prototype.slice.call(arguments, 1));
  60. } else if (typeof method === 'object' || ! method) {
  61. return methods.init.apply(this, arguments);
  62. } else {
  63. $.error('Method ' + method + ' does not exist on jQuery.grp_autocomplete_m2m');
  64. }
  65. return false;
  66. };
  67. var value_add = function(elem, value, options) {
  68. var values = [];
  69. if (elem.val()) values = elem.val().split(",");
  70. values.push(value);
  71. elem.val(values.join(","));
  72. elem.trigger('change');
  73. return values.join(",");
  74. };
  75. var value_remove = function(elem, position, options) {
  76. var values = [];
  77. if (elem.val()) values = elem.val().split(",");
  78. values.splice(position,1);
  79. elem.val(values.join(","));
  80. elem.trigger('change');
  81. return values.join(",");
  82. };
  83. var loader = function() {
  84. var loader = $('<div class="grp-loader">loader</div>');
  85. return loader;
  86. };
  87. var remove_link = function(id) {
  88. var removelink = $('<a class="grp-related-remove"></a>');
  89. removelink.attr('id', 'remove_'+id);
  90. removelink.attr('href', 'javascript://');
  91. removelink.attr('onClick', 'return removeRelatedObject(this);');
  92. removelink.hover(function() {
  93. $(this).parent().toggleClass("grp-autocomplete-preremove");
  94. });
  95. return removelink;
  96. };
  97. var repr_add = function(elem, label, options) {
  98. var repr = $('<li class="grp-repr"></li>');
  99. var removelink = $('<a class="grp-m2m-remove" href="javascript://"></a>').text(label);
  100. repr.append(removelink);
  101. repr.insertBefore(options.wrapper_search);
  102. removelink.on("click", function(e) { // remove-handler
  103. var pos = $(this).parent().parent().children("li").index($(this).parent());
  104. value_remove(elem, pos, options);
  105. $(this).parent().remove();
  106. elem.val() ? $(options.remove_link).show() : $(options.remove_link).hide();
  107. e.stopPropagation(); // prevent focus on input
  108. });
  109. removelink.hover(function() {
  110. $(this).parent().toggleClass("grp-autocomplete-preremove");
  111. });
  112. };
  113. var lookup_autocomplete = function(elem, options) {
  114. options.wrapper_search.find("input:first")
  115. .on("keydown", function(event) { // don't navigate away from the field on tab when selecting an item
  116. if (event.keyCode === $.ui.keyCode.TAB && $(this).data("uiAutocomplete").menu.active) {
  117. event.preventDefault();
  118. }
  119. })
  120. .on("focus", function() {
  121. options.wrapper_autocomplete.addClass("grp-state-focus");
  122. })
  123. .on("blur", function() {
  124. options.wrapper_autocomplete.removeClass("grp-state-focus");
  125. })
  126. .autocomplete({
  127. minLength: 1,
  128. autoFocus: true,
  129. delay: 1000,
  130. position: {my: "left top", at: "left bottom", of: options.wrapper_autocomplete},
  131. open: function(event, ui) {
  132. $(".ui-menu").width(options.wrapper_autocomplete.outerWidth()-6);
  133. },
  134. source: function(request, response) {
  135. $.ajax({
  136. url: options.autocomplete_lookup_url,
  137. dataType: 'json',
  138. data: "term=" + encodeURIComponent(request.term) + "&app_label=" + grappelli.get_app_label(elem) + "&model_name=" + grappelli.get_model_name(elem) + "&query_string=" + grappelli.get_query_string(elem),
  139. beforeSend: function (XMLHttpRequest) {
  140. options.loader.show();
  141. },
  142. success: function(data){
  143. response($.map(data, function(item) {
  144. return {label: item.label, value: item.value};
  145. }));
  146. },
  147. complete: function (XMLHttpRequest, textStatus) {
  148. options.loader.hide();
  149. }
  150. });
  151. },
  152. focus: function() { // prevent value inserted on focus
  153. return false;
  154. },
  155. select: function(event, ui) { // add repr, add value
  156. repr_add(elem, ui.item.label, options);
  157. value_add(elem, ui.item.value, options);
  158. elem.val() ? $(options.remove_link).show() : $(options.remove_link).hide();
  159. $(this).val("").trigger("focus");
  160. return false;
  161. }
  162. })
  163. .data("ui-autocomplete")._renderItem = function(ul,item) {
  164. if (!item.value) {
  165. return $("<li class='ui-state-disabled'></li>")
  166. .data( "item.autocomplete", item )
  167. .append($("<span class='error'></span>").text(item.label))
  168. .appendTo(ul);
  169. } else {
  170. return $("<li></li>")
  171. .data( "item.autocomplete", item )
  172. .append($("<a></a>").text(item.label))
  173. .appendTo(ul);
  174. }
  175. };
  176. };
  177. var lookup_id = function(elem, options) {
  178. $.getJSON(options.lookup_url, {
  179. object_id: elem.val(),
  180. app_label: grappelli.get_app_label(elem),
  181. model_name: grappelli.get_model_name(elem)
  182. }, function(data) {
  183. options.wrapper_repr.find("li.grp-repr").remove();
  184. options.wrapper_search.find("input").val("");
  185. $.each(data, function(index) {
  186. if (data[index].value) repr_add(elem, data[index].label, options);
  187. });
  188. elem.val() ? $(options.remove_link).show() : $(options.remove_link).hide();
  189. });
  190. };
  191. })(grp.jQuery);