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.

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