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.

565 lines
16 KiB

  1. /* ******************************
  2. Semantic Module: Checkbox
  3. Author: Jack Lukic
  4. Notes: First Commit MAy 25, 2013
  5. Simple plug-in which maintains the state for ui dropdown
  6. ****************************** */
  7. ;(function ( $, window, document, undefined ) {
  8. $.fn.dropdown = function(parameters) {
  9. var
  10. $allModules = $(this),
  11. $document = $(document),
  12. settings = $.extend(true, {}, $.fn.dropdown.settings, parameters),
  13. eventNamespace = '.' + settings.namespace,
  14. moduleNamespace = 'module-' + settings.namespace,
  15. selector = $allModules.selector || '',
  16. time = new Date().getTime(),
  17. performance = [],
  18. query = arguments[0],
  19. methodInvoked = (typeof query == 'string'),
  20. queryArguments = [].slice.call(arguments, 1),
  21. invokedResponse
  22. ;
  23. $allModules
  24. .each(function() {
  25. var
  26. $module = $(this),
  27. $menu = $(this).find(settings.selector.menu),
  28. $item = $(this).find(settings.selector.item),
  29. $text = $(this).find(settings.selector.text),
  30. $input = $(this).find(settings.selector.input),
  31. isTouchDevice = ('ontouchstart' in document.documentElement),
  32. selector = $module.selector || '',
  33. element = this,
  34. instance = $module.data('module-' + settings.namespace),
  35. className = settings.className,
  36. metadata = settings.metadata,
  37. namespace = settings.namespace,
  38. animation = settings.animation,
  39. errors = settings.errors,
  40. module
  41. ;
  42. module = {
  43. initialize: function() {
  44. module.verbose('Initializing dropdown with bound events', $module);
  45. if(isTouchDevice) {
  46. $module
  47. .on('touchstart' + eventNamespace, module.event.test.toggle)
  48. ;
  49. }
  50. else if(settings.on == 'click') {
  51. $module
  52. .on('click' + eventNamespace, module.event.test.toggle)
  53. ;
  54. }
  55. else if(settings.on == 'hover') {
  56. $module
  57. .on('mouseenter' + eventNamespace, module.show)
  58. .on('mouseleave' + eventNamespace, module.delayedHide)
  59. ;
  60. }
  61. else {
  62. $module
  63. .on(settings.on + eventNamespace, module.toggle)
  64. ;
  65. }
  66. if(settings.action == 'form') {
  67. module.set.selected();
  68. }
  69. $item
  70. .on(module.get.selectEvent() + eventNamespace, module.event.item.click)
  71. ;
  72. $module
  73. .data(moduleNamespace, module)
  74. ;
  75. },
  76. destroy: function() {
  77. module.verbose('Destroying previous module for', $module);
  78. $module
  79. .off(namespace)
  80. ;
  81. },
  82. event: {
  83. stopPropagation: function(event) {
  84. event.stopPropagation();
  85. },
  86. test: {
  87. toggle: function(event) {
  88. module.intent.test(event, module.toggle);
  89. event.stopPropagation();
  90. },
  91. hide: function(event) {
  92. module.intent.test(event, module.hide);
  93. event.stopPropagation();
  94. }
  95. },
  96. item: {
  97. click: function () {
  98. var
  99. $choice = $(this),
  100. text = $choice.data(metadata.text) || $choice.text(),
  101. value = $choice.data(metadata.value) || text
  102. ;
  103. module.verbose('Adding active state to selected item');
  104. $item
  105. .removeClass(className.active)
  106. ;
  107. $choice
  108. .addClass(className.active)
  109. ;
  110. module.action.determine(text, value);
  111. $.proxy(settings.onChange, $menu.get())(text, value);
  112. }
  113. }
  114. },
  115. intent: {
  116. test: function(event, callback) {
  117. module.debug('Determining whether event occurred in dropdown', event.target);
  118. callback = callback || function(){};
  119. if( $(event.target).closest($menu).size() === 0 ) {
  120. module.verbose('Triggering event', callback);
  121. callback();
  122. }
  123. else {
  124. module.verbose('Event occurred in dropdown, canceling callback');
  125. }
  126. },
  127. bind: function() {
  128. module.verbose('Binding hide intent event to document');
  129. $document
  130. .on(module.get.selectEvent(), module.event.test.hide)
  131. ;
  132. },
  133. unbind: function() {
  134. module.verbose('Removing hide intent event from document');
  135. $document
  136. .off(module.get.selectEvent())
  137. ;
  138. }
  139. },
  140. action: {
  141. determine: function(text, value) {
  142. if( $.isFunction( module.action[settings.action] ) ) {
  143. module.verbose('Triggering preset action', settings.action);
  144. module.action[ settings.action ](text, value);
  145. }
  146. else if( $.isFunction(settings.action) ) {
  147. module.verbose('Triggering user action', settings.action);
  148. settings.action(text, value);
  149. }
  150. else {
  151. module.error(errors.action);
  152. }
  153. },
  154. nothing: function() {},
  155. hide: function() {
  156. module.hide();
  157. },
  158. changeText: function(text, value) {
  159. module.set.text(text);
  160. module.hide();
  161. },
  162. form: function(text, value) {
  163. module.set.text(text);
  164. module.set.value(value);
  165. module.hide();
  166. }
  167. },
  168. get: {
  169. selectEvent: function() {
  170. return (isTouchDevice)
  171. ? 'touchstart'
  172. : 'click'
  173. ;
  174. },
  175. text: function() {
  176. return $text.text();
  177. },
  178. value: function() {
  179. return $input.val();
  180. },
  181. item: function(value) {
  182. var
  183. $selectedItem
  184. ;
  185. value = value || $input.val();
  186. $item
  187. .each(function() {
  188. if( $(this).data(metadata.value) == value ) {
  189. $selectedItem = $(this);
  190. }
  191. })
  192. ;
  193. return $selectedItem;
  194. }
  195. },
  196. set: {
  197. text: function(text) {
  198. module.debug('Changing text', text);
  199. $text.text(text);
  200. },
  201. value: function(value) {
  202. module.debug('Adding selected value to hidden input', value);
  203. $input.val(value);
  204. },
  205. selected: function(value) {
  206. var
  207. selectedValue = value || $input.val(),
  208. $selectedItem = module.get.item(value),
  209. selectedText = $selectedItem.data(metadata.text) || $selectedItem.text()
  210. ;
  211. module.debug('Setting selected menu item to', $selectedItem);
  212. $item
  213. .removeClass(className.active)
  214. ;
  215. $selectedItem
  216. .addClass(className.active)
  217. ;
  218. module.set.text(selectedText);
  219. }
  220. },
  221. is: {
  222. visible: function() {
  223. return $menu.is(':visible');
  224. },
  225. hidden: function() {
  226. return $menu.is(':not(:visible)');
  227. }
  228. },
  229. can: {
  230. click: function() {
  231. return (isTouchDevice || settings.on == 'click');
  232. },
  233. show: function() {
  234. return !$module.hasClass(className.disabled);
  235. }
  236. },
  237. animate: {
  238. show: function() {
  239. module.verbose('Doing menu showing animation');
  240. if(animation.show == 'show') {
  241. $menu
  242. .show()
  243. ;
  244. }
  245. else if(animation.show == 'slide') {
  246. $menu
  247. .clearQueue()
  248. .children()
  249. .clearQueue()
  250. .css('opacity', 0)
  251. .delay(100)
  252. .animate({
  253. opacity : 1
  254. }, 300, 'easeOutQuad')
  255. .end()
  256. .slideDown(200, 'easeOutQuad')
  257. ;
  258. }
  259. },
  260. hide: function() {
  261. module.verbose('Doing menu hiding animation');
  262. if(animation.hide == 'hide') {
  263. $menu
  264. .hide()
  265. ;
  266. }
  267. else if(animation.hide == 'slide') {
  268. $menu
  269. .clearQueue()
  270. .children()
  271. .clearQueue()
  272. .css('opacity', 1)
  273. .animate({
  274. opacity : 0
  275. }, 300, 'easeOutQuad')
  276. .end()
  277. .delay(100)
  278. .slideUp(200, 'easeOutQuad')
  279. ;
  280. }
  281. }
  282. },
  283. show: function() {
  284. clearTimeout(module.graceTimer);
  285. if( !module.is.visible() ) {
  286. module.debug('Showing dropdown');
  287. $module
  288. .addClass(className.visible)
  289. ;
  290. module.animate.show();
  291. if( module.can.click() ) {
  292. module.intent.bind();
  293. }
  294. $.proxy(settings.onShow, $menu.get())();
  295. }
  296. },
  297. delayedHide: function() {
  298. module.verbose('User moused away setting timer to hide dropdown');
  299. module.graceTimer = setTimeout(module.hide, settings.gracePeriod);
  300. },
  301. hide: function() {
  302. if( !module.is.hidden() ) {
  303. module.debug('Hiding dropdown');
  304. $module
  305. .removeClass(className.visible)
  306. ;
  307. if( module.can.click() ) {
  308. module.intent.unbind();
  309. }
  310. module.animate.hide();
  311. $.proxy(settings.onHide, $menu.get())();
  312. }
  313. },
  314. toggle: function() {
  315. module.verbose('Toggling menu visibility');
  316. if(module.can.show()) {
  317. module.show();
  318. }
  319. else {
  320. module.hide();
  321. }
  322. },
  323. setting: function(name, value) {
  324. if(value !== undefined) {
  325. if( $.isPlainObject(name) ) {
  326. $.extend(true, settings, name);
  327. }
  328. else {
  329. settings[name] = value;
  330. }
  331. }
  332. else {
  333. return settings[name];
  334. }
  335. },
  336. internal: function(name, value) {
  337. if(value !== undefined) {
  338. if( $.isPlainObject(name) ) {
  339. $.extend(true, module, name);
  340. }
  341. else {
  342. module[name] = value;
  343. }
  344. }
  345. else {
  346. return module[name];
  347. }
  348. },
  349. debug: function() {
  350. if(settings.debug) {
  351. if(settings.performance) {
  352. module.performance.log(arguments);
  353. }
  354. else {
  355. module.debug = Function.prototype.bind.call(console.info, console, settings.moduleName + ':');
  356. }
  357. }
  358. },
  359. verbose: function() {
  360. if(settings.verbose && settings.debug) {
  361. if(settings.performance) {
  362. module.performance.log(arguments);
  363. }
  364. else {
  365. module.verbose = Function.prototype.bind.call(console.info, console, settings.moduleName + ':');
  366. }
  367. }
  368. },
  369. error: function() {
  370. module.error = Function.prototype.bind.call(console.log, console, settings.moduleName + ':');
  371. },
  372. performance: {
  373. log: function(message) {
  374. var
  375. currentTime,
  376. executionTime,
  377. previousTime
  378. ;
  379. if(settings.performance) {
  380. currentTime = new Date().getTime();
  381. previousTime = time || currentTime,
  382. executionTime = currentTime - previousTime;
  383. time = currentTime;
  384. performance.push({
  385. 'Element' : element,
  386. 'Name' : message[0],
  387. 'Arguments' : message[1] || 'None',
  388. 'Execution Time' : executionTime
  389. });
  390. clearTimeout(module.performance.timer);
  391. module.performance.timer = setTimeout(module.performance.display, 100);
  392. }
  393. },
  394. display: function() {
  395. var
  396. title = settings.moduleName,
  397. caption = settings.moduleName + ': ' + selector + '(' + $allModules.size() + ' elements)',
  398. totalExecutionTime = 0
  399. ;
  400. if(selector) {
  401. title += ' Performance (' + selector + ')';
  402. }
  403. if( (console.group !== undefined || console.table !== undefined) && performance.length > 0) {
  404. console.groupCollapsed(title);
  405. if(console.table) {
  406. $.each(performance, function(index, data) {
  407. totalExecutionTime += data['Execution Time'];
  408. });
  409. console.table(performance);
  410. }
  411. else {
  412. $.each(performance, function(index, data) {
  413. totalExecutionTime += data['Execution Time'];
  414. });
  415. }
  416. console.log('Total Execution Time:', totalExecutionTime +'ms');
  417. console.groupEnd();
  418. performance = [];
  419. time = false;
  420. }
  421. }
  422. },
  423. invoke: function(query, passedArguments, context) {
  424. var
  425. maxDepth,
  426. found
  427. ;
  428. passedArguments = passedArguments || queryArguments;
  429. context = element || context;
  430. if(typeof query == 'string' && instance !== undefined) {
  431. query = query.split('.');
  432. maxDepth = query.length - 1;
  433. $.each(query, function(depth, value) {
  434. if( $.isPlainObject( instance[value] ) && (depth != maxDepth) ) {
  435. instance = instance[value];
  436. return true;
  437. }
  438. else if( instance[value] !== undefined ) {
  439. found = instance[value];
  440. return true;
  441. }
  442. module.error(errors.method);
  443. return false;
  444. });
  445. }
  446. if ( $.isFunction( found ) ) {
  447. module.verbose('Executing invoked function', found);
  448. return found.apply(context, passedArguments);
  449. }
  450. return found || false;
  451. }
  452. };
  453. if(methodInvoked) {
  454. if(instance === undefined) {
  455. module.initialize();
  456. }
  457. invokedResponse = module.invoke(query);
  458. }
  459. else {
  460. if(instance !== undefined) {
  461. module.destroy();
  462. }
  463. module.initialize();
  464. }
  465. })
  466. ;
  467. return (invokedResponse)
  468. ? invokedResponse
  469. : this
  470. ;
  471. };
  472. $.fn.dropdown.settings = {
  473. moduleName : 'Dropdown Module',
  474. namespace : 'dropdown',
  475. verbose : true,
  476. debug : true,
  477. performance : true,
  478. on : 'click',
  479. gracePeriod : 300,
  480. action : 'hide',
  481. animation : {
  482. show: 'slide',
  483. hide: 'slide'
  484. },
  485. onChange : function(){},
  486. onShow : function(){},
  487. onHide : function(){},
  488. errors : {
  489. action : 'You called a dropdown action that was not defined',
  490. method : 'The method you called is not defined.'
  491. },
  492. metadata: {
  493. text : 'text',
  494. value : 'value'
  495. },
  496. selector : {
  497. menu : '.menu',
  498. item : '.menu > .item',
  499. text : '> .text',
  500. input : '> input[type="hidden"]'
  501. },
  502. className : {
  503. active : 'active',
  504. disabled : 'disabled',
  505. visible : 'visible'
  506. }
  507. };
  508. })( jQuery, window , document );