No Description
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

comboTreePlugin.js 15KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439
  1. /*!
  2. * jQuery ComboTree Plugin
  3. * Author: Erhan FIRAT
  4. * Mail: erhanfirat@gmail.com
  5. * Licensed under the MIT license
  6. * Version: 1.1
  7. */
  8. ;(function ( $, window, document, undefined ) {
  9. // Create the defaults once
  10. var comboTreePlugin = 'comboTree',
  11. defaults = {
  12. source: [],
  13. isMultiple: false
  14. };
  15. // The actual plugin constructor
  16. function ComboTree( element, options ) {
  17. this.elemInput = element;
  18. this._elemInput = $(element);
  19. this.options = $.extend( {}, defaults, options) ;
  20. this._defaults = defaults;
  21. this._name = comboTreePlugin;
  22. this.init();
  23. }
  24. ComboTree.prototype.init = function () {
  25. // Setting Doms
  26. this.comboTreeId = 'comboTree' + Math.floor(Math.random() * 999999);
  27. this._elemInput.addClass('comboTreeInputBox');
  28. if (this._elemInput.attr('id') === undefined)
  29. this._elemInput.attr('id', this.comboTreeId + 'Input');
  30. this.elemInputId = this._elemInput.attr('id');
  31. this._elemInput.wrap('<div id="'+ this.comboTreeId + 'Wrapper" class="comboTreeWrapper"></div>');
  32. this._elemInput.wrap('<div id="'+ this.comboTreeId + 'InputWrapper" class="comboTreeInputWrapper"></div>');
  33. this._elemWrapper = $('#' + this.comboTreeId + 'Wrapper');
  34. this._elemArrowBtn = $('<button id="' + this.comboTreeId + 'ArrowBtn" class="comboTreeArrowBtn"><span class="comboTreeArrowBtnImg">▼</span></button>');
  35. this._elemInput.after(this._elemArrowBtn);
  36. this._elemWrapper.append('<div id="' + this.comboTreeId + 'DropDownContainer" class="comboTreeDropDownContainer"><div class="comboTreeDropDownContent"></div>');
  37. // DORP DOWN AREA
  38. this._elemDropDownContainer = $('#' + this.comboTreeId + 'DropDownContainer');
  39. this._elemDropDownContainer.html(this.createSourceHTML());
  40. this._elemItems = this._elemDropDownContainer.find('li');
  41. this._elemItemsTitle = this._elemDropDownContainer.find('span.comboTreeItemTitle');
  42. // VARIABLES
  43. this._selectedItem = {};
  44. this._selectedItems = [];
  45. this.bindings();
  46. };
  47. // *********************************
  48. // SOURCES CODES
  49. // *********************************
  50. ComboTree.prototype.removeSourceHTML = function () {
  51. this._elemDropDownContainer.html('');
  52. };
  53. ComboTree.prototype.createSourceHTML = function () {
  54. var sourceHTML = '';
  55. if (this.options.isMultiple)
  56. sourceHTML = this.createFilterHTMLForMultiSelect();
  57. sourceHTML += this.createSourceSubItemsHTML(this.options.source);
  58. return sourceHTML;
  59. };
  60. ComboTree.prototype.createFilterHTMLForMultiSelect = function (){
  61. return '<input id="' + this.comboTreeId + 'MultiFilter" type="text" class="multiplesFilter" placeholder="搜索某个会员"/>';
  62. }
  63. ComboTree.prototype.createSourceSubItemsHTML = function (subItems) {
  64. var subItemsHtml = '<UL>';
  65. for (var i=0; i<subItems.length; i++){
  66. subItemsHtml += this.createSourceItemHTML(subItems[i]);
  67. }
  68. subItemsHtml += '</UL>'
  69. return subItemsHtml;
  70. }
  71. ComboTree.prototype.createSourceItemHTML = function (sourceItem) {
  72. var itemHtml = "",
  73. isThereSubs = sourceItem.hasOwnProperty("subs");
  74. itemHtml = '<LI class="ComboTreeItem' + (isThereSubs?'Parent':'Chlid') + '"> ';
  75. if (isThereSubs)
  76. itemHtml += '<span class="comboTreeParentPlus">&minus;</span>';
  77. if (this.options.isMultiple)
  78. itemHtml += '<span data-id="' + sourceItem.id + '" class="comboTreeItemTitle"><input type="checkbox">' + sourceItem.title + '</span>';
  79. else
  80. itemHtml += '<span data-id="' + sourceItem.id + '" class="comboTreeItemTitle">' + sourceItem.title + '</span>';
  81. if (isThereSubs)
  82. itemHtml += this.createSourceSubItemsHTML(sourceItem.subs);
  83. itemHtml += '</LI>';
  84. return itemHtml;
  85. };
  86. // BINDINGS
  87. // *****************************
  88. ComboTree.prototype.bindings = function () {
  89. var _this = this;
  90. this._elemArrowBtn.on('click', function(e){
  91. e.stopPropagation();
  92. _this.toggleDropDown();
  93. });
  94. this._elemInput.on('click', function(e){
  95. e.stopPropagation();
  96. if (!_this._elemDropDownContainer.is(':visible'))
  97. _this.toggleDropDown();
  98. });
  99. this._elemItems.on('click', function(e){
  100. e.stopPropagation();
  101. if ($(this).hasClass('ComboTreeItemParent')){
  102. _this.toggleSelectionTree(this);
  103. }
  104. });
  105. this._elemItemsTitle.on('click', function(e){
  106. e.stopPropagation();
  107. if (_this.options.isMultiple)
  108. _this.multiItemClick(this);
  109. else
  110. _this.singleItemClick(this);
  111. });
  112. this._elemItemsTitle.on("mousemove", function (e) {
  113. e.stopPropagation();
  114. _this.dropDownMenuHover(this);
  115. });
  116. // KEY BINDINGS
  117. this._elemInput.on('keyup', function(e) {
  118. e.stopPropagation();
  119. switch (e.keyCode) {
  120. case 27:
  121. _this.closeDropDownMenu(); break;
  122. case 13:
  123. case 39: case 37: case 40: case 38:
  124. e.preventDefault();
  125. break;
  126. default:
  127. if (!_this.options.isMultiple)
  128. _this.filterDropDownMenu();
  129. break;
  130. }
  131. });
  132. if (_this.options.isMultiple) {
  133. $("#" + _this.comboTreeId + "MultiFilter").on('keyup', function(e) {
  134. e.stopPropagation();
  135. switch (e.keyCode) {
  136. case 27:
  137. $(this).val(''); _this.filterDropDownMenu(); break;
  138. default:
  139. _this.filterDropDownMenu();
  140. break;
  141. }
  142. });
  143. }
  144. this._elemInput.on('keydown', function(e) {
  145. e.stopPropagation();
  146. switch (e.keyCode) {
  147. case 9:
  148. _this.closeDropDownMenu(); break;
  149. case 40: case 38:
  150. e.preventDefault();
  151. _this.dropDownInputKeyControl(e.keyCode - 39); break;
  152. case 37: case 39:
  153. e.preventDefault();
  154. _this.dropDownInputKeyToggleTreeControl(e.keyCode - 38);
  155. break;
  156. case 13:
  157. if (_this.options.isMultiple)
  158. _this.multiItemClick(_this._elemHoveredItem);
  159. else
  160. _this.singleItemClick(_this._elemHoveredItem);
  161. e.preventDefault();
  162. break;
  163. default:
  164. if (_this.options.isMultiple)
  165. e.preventDefault();
  166. }
  167. });
  168. // ON FOCUS OUT CLOSE DROPDOWN
  169. $(document).on('mouseup.' + _this.comboTreeId, function (e){
  170. if (!_this._elemWrapper.is(e.target) && _this._elemWrapper.has(e.target).length === 0 && _this._elemDropDownContainer.is(':visible'))
  171. _this.closeDropDownMenu();
  172. });
  173. };
  174. // EVENTS HERE
  175. // ****************************
  176. // DropDown Menu Open/Close
  177. ComboTree.prototype.toggleDropDown = function () {
  178. this._elemDropDownContainer.slideToggle(50);
  179. this._elemInput.focus();
  180. };
  181. ComboTree.prototype.closeDropDownMenu = function () {
  182. this._elemDropDownContainer.slideUp(50);
  183. };
  184. // Selection Tree Open/Close
  185. ComboTree.prototype.toggleSelectionTree = function (item, direction) {
  186. var subMenu = $(item).children('ul')[0];
  187. if (direction === undefined){
  188. if ($(subMenu).is(':visible'))
  189. $(item).children('span.comboTreeParentPlus').html("+");
  190. else
  191. $(item).children('span.comboTreeParentPlus').html("&minus;");
  192. $(subMenu).slideToggle(50);
  193. }
  194. else if (direction == 1 && !$(subMenu).is(':visible')){
  195. $(item).children('span.comboTreeParentPlus').html("&minus;");
  196. $(subMenu).slideDown(50);
  197. }
  198. else if (direction == -1){
  199. if ($(subMenu).is(':visible')){
  200. $(item).children('span.comboTreeParentPlus').html("+");
  201. $(subMenu).slideUp(50);
  202. }
  203. else {
  204. this.dropDownMenuHoverToParentItem(item);
  205. }
  206. }
  207. };
  208. // SELECTION FUNCTIONS
  209. // *****************************
  210. ComboTree.prototype.singleItemClick = function (ctItem) {
  211. this._selectedItem = {
  212. id: $(ctItem).attr("data-id"),
  213. title: $(ctItem).text()
  214. };
  215. this.refreshInputVal();
  216. this.closeDropDownMenu();
  217. };
  218. ComboTree.prototype.multiItemClick = function (ctItem) {
  219. this._selectedItem = {
  220. id: $(ctItem).attr("data-id"),
  221. title: $(ctItem).text()
  222. };
  223. var index = this.isItemInArray(this._selectedItem, this._selectedItems);
  224. if (index){
  225. this._selectedItems.splice(parseInt(index), 1);
  226. $(ctItem).find("input").prop('checked', false);
  227. }
  228. else {
  229. this._selectedItems.push(this._selectedItem);
  230. $(ctItem).find("input").prop('checked', true);
  231. }
  232. this.refreshInputVal();
  233. };
  234. ComboTree.prototype.isItemInArray = function (item, arr) {
  235. for (var i=0; i<arr.length; i++)
  236. if (item.id == arr[i].id && item.title == arr[i].title)
  237. return i + "";
  238. return false;
  239. }
  240. ComboTree.prototype.refreshInputVal = function () {
  241. var tmpTitle = "";
  242. if (this.options.isMultiple) {
  243. for (var i=0; i<this._selectedItems.length; i++){
  244. tmpTitle += this._selectedItems[i].title;
  245. if (i<this._selectedItems.length-1)
  246. tmpTitle += ", ";
  247. }
  248. }
  249. else {
  250. tmpTitle = this._selectedItem.title;
  251. }
  252. this._elemInput.val(tmpTitle);
  253. }
  254. ComboTree.prototype.dropDownMenuHover = function (itemSpan, withScroll) {
  255. this._elemItems.find('span.comboTreeItemHover').removeClass('comboTreeItemHover');
  256. $(itemSpan).addClass('comboTreeItemHover');
  257. this._elemHoveredItem = $(itemSpan);
  258. if (withScroll)
  259. this.dropDownScrollToHoveredItem(this._elemHoveredItem);
  260. }
  261. ComboTree.prototype.dropDownScrollToHoveredItem = function (itemSpan) {
  262. var curScroll = this._elemDropDownContainer.scrollTop();
  263. this._elemDropDownContainer.scrollTop(curScroll + $(itemSpan).parent().position().top - 80);
  264. }
  265. ComboTree.prototype.dropDownMenuHoverToParentItem = function (item) {
  266. var parentSpanItem = $($(item).parents('li.ComboTreeItemParent')[0]).children("span.comboTreeItemTitle");
  267. if (parentSpanItem.length)
  268. this.dropDownMenuHover(parentSpanItem, true);
  269. else
  270. this.dropDownMenuHover(this._elemItemsTitle[0], true);
  271. }
  272. ComboTree.prototype.dropDownInputKeyToggleTreeControl = function (direction) {
  273. var item = this._elemHoveredItem;
  274. if ($(item).parent('li').hasClass('ComboTreeItemParent'))
  275. this.toggleSelectionTree($(item).parent('li'), direction);
  276. else if (direction == -1)
  277. this.dropDownMenuHoverToParentItem(item);
  278. }
  279. ComboTree.prototype.dropDownInputKeyControl = function (step) {
  280. if (!this._elemDropDownContainer.is(":visible"))
  281. this.toggleDropDown();
  282. var list = this._elemItems.find("span.comboTreeItemTitle:visible");
  283. i = this._elemHoveredItem?list.index(this._elemHoveredItem) + step:0;
  284. i = (list.length + i) % list.length;
  285. this.dropDownMenuHover(list[i], true);
  286. },
  287. ComboTree.prototype.filterDropDownMenu = function () {
  288. var searchText = '';
  289. if (!this.options.isMultiple)
  290. searchText = this._elemInput.val();
  291. else
  292. searchText = $("#" + this.comboTreeId + "MultiFilter").val();
  293. if (searchText != ""){
  294. this._elemItemsTitle.hide();
  295. this._elemItemsTitle.siblings("span.comboTreeParentPlus").hide();
  296. list = this._elemItems.find("span:icontains('" + searchText + "')").each(function (i, elem) {
  297. $(this).show();
  298. $(this).siblings("span.comboTreeParentPlus").show();
  299. });
  300. }
  301. else{
  302. this._elemItemsTitle.show();
  303. this._elemItemsTitle.siblings("span.comboTreeParentPlus").show();
  304. }
  305. }
  306. // Retuns Array (multiple), Integer (single), or False (No choice)
  307. ComboTree.prototype.getSelectedItemsId = function () {
  308. if (this.options.isMultiple && this._selectedItems.length>0){
  309. var tmpArr = [];
  310. for (i=0; i<this._selectedItems.length; i++)
  311. tmpArr.push(this._selectedItems[i].id);
  312. return tmpArr;
  313. }
  314. else if (!this.options.isMultiple && this._selectedItem.hasOwnProperty('id')){
  315. return this._selectedItem.id;
  316. }
  317. return false;
  318. }
  319. // Retuns Array (multiple), Integer (single), or False (No choice)
  320. ComboTree.prototype.getSelectedItemsTitle = function () {
  321. if (this.options.isMultiple && this._selectedItems.length>0){
  322. var tmpArr = [];
  323. for (i=0; i<this._selectedItems.length; i++)
  324. tmpArr.push(this._selectedItems[i].title);
  325. return tmpArr;
  326. }
  327. else if (!this.options.isMultiple && this._selectedItem.hasOwnProperty('id')){
  328. return this._selectedItem.title;
  329. }
  330. return false;
  331. }
  332. ComboTree.prototype.unbind = function () {
  333. this._elemArrowBtn.off('click');
  334. this._elemInput.off('click');
  335. this._elemItems.off('click');
  336. this._elemItemsTitle.off('click');
  337. this._elemItemsTitle.off("mousemove");
  338. this._elemInput.off('keyup');
  339. this._elemInput.off('keydown');
  340. this._elemInput.off('mouseup.' + this.comboTreeId);
  341. $(document).off('mouseup.' + this.comboTreeId);
  342. }
  343. ComboTree.prototype.destroy = function () {
  344. this.unbind();
  345. this._elemWrapper.before(this._elemInput);
  346. this._elemWrapper.remove();
  347. this._elemInput.removeData('plugin_' + comboTreePlugin);
  348. }
  349. $.fn[comboTreePlugin] = function (options) {
  350. var ctArr = [];
  351. this.each(function () {
  352. if (!$.data(this, 'plugin_' + comboTreePlugin)) {
  353. $.data(this, 'plugin_' + comboTreePlugin, new ComboTree( this, options));
  354. ctArr.push($(this).data()['plugin_' + comboTreePlugin]);
  355. }
  356. });
  357. if (this.length == 1)
  358. return ctArr[0];
  359. else
  360. return ctArr;
  361. }
  362. })( jQuery, window, document );