term_reference_tree.js 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287
  1. (function($) {
  2. Drupal.behaviors.termReferenceTree = {
  3. attach: function(context, settings) {
  4. // Bind the term expand/contract button to slide toggle the list underneath.
  5. $('.term-reference-tree-button', context).click(function() {
  6. $(this).toggleClass('term-reference-tree-collapsed');
  7. $(this).siblings('ul').slideToggle('fast');
  8. });
  9. // An expand all button (unimplemented)
  10. /*
  11. $('.expandbutton').click(function() {
  12. $(this).siblings('.term-reference-tree-button').trigger('click');
  13. });
  14. */
  15. $('.term-reference-tree', context).each(function() {
  16. // On page load, check whether the maximum number of choices is already selected.
  17. // If so, disable the other options.
  18. var tree = $(this);
  19. checkMaxChoices(tree, false);
  20. $(this).find('input[type=checkbox]').change(function() {
  21. checkMaxChoices(tree, $(this));
  22. });
  23. //On page load, check if the user wants a track list. If so, add the
  24. //currently selected items to it.
  25. if($(this).hasClass('term-reference-tree-track-list-shown')) {
  26. var track_list_container = $(this).find('.term-reference-tree-track-list');
  27. //Var to track whether using checkboxes or radio buttons.
  28. var input_type =
  29. ( $(this).has('input[type=checkbox]').size() > 0 ) ? 'checkbox' : 'radio';
  30. //Find all the checked controls.
  31. var checked_controls = $(this).find('input[type=' + input_type + ']:checked');
  32. //Get their labels.
  33. var labels = checked_controls.next();
  34. var label_element;
  35. //For each label of the checked boxes, add item to the track list.
  36. labels.each(function(index) {
  37. label_element = $(labels[index]);
  38. addItemToTrackList(
  39. track_list_container, //Where to add new item.
  40. label_element.html(), //Text of new item.
  41. $(label_element).attr('for'), //Id of control new item is for.
  42. input_type //checkbox or radio
  43. );
  44. }); //End labels.each
  45. //Show "nothing selected" message, if needed.
  46. showNothingSelectedMessage(track_list_container);
  47. //Event - when an element on the track list is clicked on:
  48. // 1. Delete it.
  49. // 2. Uncheck the associated checkbox.
  50. //The event is bound to the track list container, not each element.
  51. $(track_list_container).click(function(event){
  52. //Remove the "nothing selected" message if showing - add it later if needed.
  53. //removeNothingSelectedMessage(track_list_container);
  54. var event_target = $(event.target);
  55. var control_id = event_target.data('control_id');
  56. if(control_id) {
  57. event_target.remove();
  58. var checkbox = $('#' + control_id);
  59. checkbox.removeAttr('checked');
  60. checkMaxChoices(tree, checkbox);
  61. //Show "nothing selected" message, if needed.
  62. showNothingSelectedMessage(track_list_container);
  63. }
  64. });
  65. //Change track list when controls are clicked.
  66. $(this).find('.form-' + input_type).change(function(event){
  67. //Remove the "nothing selected" message if showing - add it later if needed.
  68. removeNothingSelectedMessage(track_list_container);
  69. var event_target = $(event.target);
  70. var control_id = event_target.attr('id');
  71. if ( event_target.attr('checked') ) {
  72. //Control checked - add item to the track list.
  73. label_element = event_target.next();
  74. addItemToTrackList(
  75. track_list_container, //Where to add new item.
  76. label_element.html(), //Text of new item.
  77. $(label_element).attr('for'), //Id of control new item is for.
  78. input_type //checkbox or radio
  79. );
  80. }
  81. else {
  82. //Checkbox unchecked. Remove from the track list.
  83. $('#' + control_id + '_list').remove();
  84. }
  85. //Show "nothing selected" message, if needed.
  86. showNothingSelectedMessage(track_list_container);
  87. }); //End process checkbox changes.
  88. } //End Want a track list.
  89. //On page load, check if the user wants a cascading selection.
  90. if($(this).hasClass('term-reference-tree-cascading-selection')) {
  91. //Check children when checkboxes are clicked.
  92. $(this).find('.form-checkbox').change(function(event) {
  93. var event_target = $(event.target);
  94. var control_id = event_target.attr('id');
  95. var children = event_target.parent().next().children().children('div.form-type-checkbox').children('input[id^="' + control_id + '-children"]');
  96. if(event_target.attr('checked')) {
  97. //Checkbox checked - check children if none were checked.
  98. if(!$(children).filter(':checked').length) {
  99. $(children).click().trigger('change');
  100. }
  101. }
  102. else {
  103. //Checkbox unchecked. Uncheck children if all were checked.
  104. if(!$(children).not(':checked').length) {
  105. $(children).click().trigger('change');
  106. }
  107. }
  108. });
  109. //End process checkbox changes.
  110. } //End Want a cascading checking.
  111. });
  112. }
  113. };
  114. /**
  115. * Add a new item to the track list.
  116. * If more than one item can be selected, the new item is positioned to
  117. * match the order of the terms in the checkbox tree.
  118. *
  119. * @param track_list_container Container where the new item will be added.
  120. *
  121. * @param item_text Text of the item to add.
  122. *
  123. * @param control_id Id of the checkbox/radio control the item matches.
  124. *
  125. * @param control_type Control type - 'checkbox' or 'radio'.
  126. */
  127. function addItemToTrackList(track_list_container, item_text, control_id, control_type) {
  128. var new_item = $('<li class="track-item">' + item_text + '</li>');
  129. new_item.data('control_id', control_id);
  130. //Add an id for easy finding of the item.
  131. new_item.attr('id', control_id + '_list');
  132. //Process radio controls - only one item can be selected.
  133. if ( control_type == 'radio') {
  134. //Find the existing element on the track list, if there is one.
  135. var current_items = track_list_container.find('li');
  136. //If there are no items on the track list, add the new item.
  137. if ( current_items.size() == 0 ) {
  138. track_list_container.append(new_item);
  139. }
  140. else {
  141. //There is an item on the list.
  142. var current_item = $(current_items.get(0));
  143. //Is the item we want to add different from what is there?
  144. if ( current_item.data('control_id') != control_id ) {
  145. //Remove exiting element from track list, and add the new one.
  146. current_item.remove();
  147. track_list_container.append(new_item);
  148. }
  149. }
  150. return;
  151. }
  152. //Using checkboxes, so there can be more than one selected item.
  153. //Find the right place to put the new item, to match the order of the
  154. //checkboxes.
  155. var list_items = track_list_container.find('li');
  156. var item_comparing_to;
  157. //Flag to tell whether the item was inserted.
  158. var inserted_flag = false;
  159. list_items.each(function(index){
  160. item_comparing_to = $(list_items[index]);
  161. //If item is already on the track list, do nothing.
  162. if ( control_id == item_comparing_to.data('control_id') ) {
  163. inserted_flag = true;
  164. return false; //Returning false stops the loop.
  165. }
  166. else if ( control_id < item_comparing_to.data('control_id') ) {
  167. //Add it here.
  168. item_comparing_to.before(new_item);
  169. inserted_flag = true;
  170. return false; //Returning false stops the loop.
  171. }
  172. });
  173. //If not inserted yet, add new item at the end of the track list.
  174. if ( ! inserted_flag ) {
  175. track_list_container.append(new_item);
  176. }
  177. }
  178. /**
  179. * Show the 'nothing selected' message if it applies.
  180. *
  181. * @param track_list_container Where the message is to be shown.
  182. */
  183. function showNothingSelectedMessage(track_list_container) {
  184. //Is the message there already?
  185. var message_showing =
  186. (track_list_container.find('.term_ref_tree_nothing_message').size() != 0);
  187. //Number of real items showing.
  188. var num_real_items_showing =
  189. message_showing
  190. ? track_list_container.find('li').size() - 1
  191. : track_list_container.find('li').size();
  192. if ( num_real_items_showing == 0 ) {
  193. //No items showing, so show the message.
  194. if ( ! message_showing ) {
  195. track_list_container.append(
  196. '<li class="term_ref_tree_nothing_message">' + termReferenceTreeNothingSelectedText + '</li>'
  197. );
  198. }
  199. }
  200. else { // !(num_real_items_showing == 0)
  201. //There are real items.
  202. if ( message_showing ) {
  203. track_list_container.find('.term_ref_tree_nothing_message').remove();
  204. }
  205. }
  206. }
  207. /**
  208. * Remove the 'nothing selected' message. Makes processing easier.
  209. *
  210. * @param track_list_container Where the message is shown.
  211. */
  212. function removeNothingSelectedMessage(track_list_container) {
  213. track_list_container.find('.term_ref_tree_nothing_message').remove();
  214. }
  215. // This helper function checks if the maximum number of choices is already selected.
  216. // If so, it disables all the other options. If not, it enables them.
  217. function checkMaxChoices(item, checkbox) {
  218. var maxChoices = -1;
  219. try {
  220. maxChoices = parseInt(Drupal.settings.term_reference_tree.trees[item.attr('id')]['max_choices']);
  221. }
  222. catch (e){}
  223. var count = item.find(':checked').length;
  224. if(maxChoices > 0 && count >= maxChoices) {
  225. item.find('input[type=checkbox]:not(:checked)').attr('disabled', 'disabled').parent().addClass('disabled');
  226. } else {
  227. item.find('input[type=checkbox]').removeAttr('disabled').parent().removeClass('disabled');
  228. }
  229. if(checkbox) {
  230. if(item.hasClass('select-parents')) {
  231. var track_list_container = item.find('.term-reference-tree-track-list');
  232. var input_type =
  233. ( item.has('input[type=checkbox]').size() > 0 ) ? 'checkbox' : 'radio';
  234. if(checkbox.attr('checked')) {
  235. checkbox.parents('ul.term-reference-tree-level li').children('div.form-item').children('input[type=checkbox]').each(function() {
  236. $(this).attr('checked', checkbox.attr('checked'));
  237. if(track_list_container) {
  238. label_element = $(this).next();
  239. addItemToTrackList(
  240. track_list_container, //Where to add new item.
  241. label_element.html(), //Text of new item.
  242. $(label_element).attr('for'), //Id of control new item is for.
  243. input_type //checkbox or radio
  244. );
  245. }
  246. });
  247. }
  248. }
  249. }
  250. }
  251. })(jQuery);