@oat-sa/tao-core-ui 1.60.2 → 1.62.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (783) hide show
  1. package/LICENSE +339 -339
  2. package/README.md +23 -18
  3. package/dist/actionbar.js +393 -410
  4. package/dist/adder.js +143 -156
  5. package/dist/animable/absorbable/absorbable.js +208 -211
  6. package/dist/animable/absorbable/css/absorb.css +7 -8
  7. package/dist/animable/absorbable/css/absorb.css.map +1 -1
  8. package/dist/animable/pulsable/css/pulse.css +7 -7
  9. package/dist/animable/pulsable/css/pulse.css.map +1 -1
  10. package/dist/animable/pulsable/pulsable.js +174 -174
  11. package/dist/areaBroker.js +59 -72
  12. package/dist/autocomplete/css/autocomplete.css +7 -8
  13. package/dist/autocomplete/css/autocomplete.css.map +1 -1
  14. package/dist/autocomplete.js +494 -604
  15. package/dist/autoscroll.js +23 -25
  16. package/dist/badge/badge.js +201 -199
  17. package/dist/badge/css/badge.css +7 -8
  18. package/dist/badge/css/badge.css.map +1 -1
  19. package/dist/breadcrumbs/css/breadcrumbs.css +7 -7
  20. package/dist/breadcrumbs/css/breadcrumbs.css.map +1 -1
  21. package/dist/breadcrumbs.js +286 -285
  22. package/dist/btngrouper.js +83 -95
  23. package/dist/bulkActionPopup/css/bulkActionPopup.css +7 -7
  24. package/dist/bulkActionPopup/css/bulkActionPopup.css.map +1 -1
  25. package/dist/bulkActionPopup.js +503 -503
  26. package/dist/button.js +300 -293
  27. package/dist/calculator/css/calculator.css +10 -10
  28. package/dist/calculator/css/calculator.css.map +1 -1
  29. package/dist/calculator.js +59 -72
  30. package/dist/cascadingComboBox.js +257 -256
  31. package/dist/ckeditor/ckConfigurator.js +130 -154
  32. package/dist/ckeditor/dtdHandler.js +305 -345
  33. package/dist/class/css/selector.css +7 -7
  34. package/dist/class/css/selector.css.map +1 -1
  35. package/dist/class/selector.js +437 -470
  36. package/dist/component/alignable.js +97 -121
  37. package/dist/component/containable.js +53 -58
  38. package/dist/component/css/components.css +7 -7
  39. package/dist/component/css/components.css.map +1 -1
  40. package/dist/component/css/windowComponent.css +7 -7
  41. package/dist/component/css/windowComponent.css.map +1 -1
  42. package/dist/component/draggable.js +34 -36
  43. package/dist/component/placeable.js +84 -102
  44. package/dist/component/resizable.js +79 -89
  45. package/dist/component/stackable.js +21 -27
  46. package/dist/component/windowed.js +284 -297
  47. package/dist/component.js +439 -467
  48. package/dist/container.js +79 -98
  49. package/dist/contextualPopup/css/contextualPopup.css +7 -7
  50. package/dist/contextualPopup/css/contextualPopup.css.map +1 -1
  51. package/dist/contextualPopup.js +417 -436
  52. package/dist/dashboard/css/dashboard.css +7 -7
  53. package/dist/dashboard/css/dashboard.css.map +1 -1
  54. package/dist/dashboard.js +302 -305
  55. package/dist/datalist/css/datalist.css +7 -7
  56. package/dist/datalist/css/datalist.css.map +1 -1
  57. package/dist/datalist.js +783 -791
  58. package/dist/datatable/css/datatable.css +11 -7
  59. package/dist/datatable/css/datatable.css.map +1 -1
  60. package/dist/datatable/filterStrategy/filterStrategy.js +28 -30
  61. package/dist/datatable/filterStrategy/multiple.js +26 -45
  62. package/dist/datatable/filterStrategy/single.js +28 -40
  63. package/dist/datatable.js +1591 -1616
  64. package/dist/dateRange/css/dateRange.css +7 -7
  65. package/dist/dateRange/css/dateRange.css.map +1 -1
  66. package/dist/dateRange/dateRange.js +398 -412
  67. package/dist/datetime/css/picker.css +7 -7
  68. package/dist/datetime/css/picker.css.map +1 -1
  69. package/dist/datetime/picker.js +670 -703
  70. package/dist/deleter.js +383 -398
  71. package/dist/destination/css/selector.css +7 -7
  72. package/dist/destination/css/selector.css.map +1 -1
  73. package/dist/destination/selector.js +295 -293
  74. package/dist/dialog/alert.js +30 -32
  75. package/dist/dialog/confirm.js +29 -32
  76. package/dist/dialog/confirmDelete.js +223 -221
  77. package/dist/dialog.js +651 -695
  78. package/dist/disabler.js +108 -119
  79. package/dist/documentViewer/css/documentViewer.css +7 -7
  80. package/dist/documentViewer/css/documentViewer.css.map +1 -1
  81. package/dist/documentViewer/providers/pdfViewer/fallback/viewer.js +172 -169
  82. package/dist/documentViewer/providers/pdfViewer/pdfjs/areaBroker.js +32 -29
  83. package/dist/documentViewer/providers/pdfViewer/pdfjs/findBar.js +523 -564
  84. package/dist/documentViewer/providers/pdfViewer/pdfjs/pageView.js +376 -393
  85. package/dist/documentViewer/providers/pdfViewer/pdfjs/pagesManager.js +53 -68
  86. package/dist/documentViewer/providers/pdfViewer/pdfjs/searchEngine.js +517 -567
  87. package/dist/documentViewer/providers/pdfViewer/pdfjs/textManager.js +73 -96
  88. package/dist/documentViewer/providers/pdfViewer/pdfjs/viewer.js +381 -392
  89. package/dist/documentViewer/providers/pdfViewer/pdfjs/wrapper.js +125 -152
  90. package/dist/documentViewer/providers/pdfViewer.js +189 -191
  91. package/dist/documentViewer/viewerFactory.js +76 -81
  92. package/dist/documentViewer.js +303 -311
  93. package/dist/dropdown/css/dropdown.css +7 -7
  94. package/dist/dropdown/css/dropdown.css.map +1 -1
  95. package/dist/dropdown.js +392 -412
  96. package/dist/durationer.js +77 -85
  97. package/dist/dynamicComponent/css/dynamicComponent.css +7 -7
  98. package/dist/dynamicComponent/css/dynamicComponent.css.map +1 -1
  99. package/dist/dynamicComponent.js +595 -630
  100. package/dist/feedback.js +365 -375
  101. package/dist/figure/FigureStateActive.js +246 -296
  102. package/dist/filesender.js +44 -54
  103. package/dist/filter.js +244 -240
  104. package/dist/form/css/dropdownForm.css +7 -7
  105. package/dist/form/css/dropdownForm.css.map +1 -1
  106. package/dist/form/css/form.css +7 -7
  107. package/dist/form/css/form.css.map +1 -1
  108. package/dist/form/dropdownForm.js +357 -362
  109. package/dist/form/form.js +703 -984
  110. package/dist/form/simpleForm.js +74 -77
  111. package/dist/form/validator/css/validator.css +7 -7
  112. package/dist/form/validator/css/validator.css.map +1 -1
  113. package/dist/form/validator/renderer.js +235 -241
  114. package/dist/form/validator/validator.js +181 -267
  115. package/dist/form/widget/css/widget.css +7 -7
  116. package/dist/form/widget/css/widget.css.map +1 -1
  117. package/dist/form/widget/definitions.js +27 -28
  118. package/dist/form/widget/loader.js +16 -16
  119. package/dist/form/widget/providers/checkBox.js +257 -264
  120. package/dist/form/widget/providers/comboBox.js +194 -193
  121. package/dist/form/widget/providers/default.js +44 -50
  122. package/dist/form/widget/providers/hidden.js +179 -176
  123. package/dist/form/widget/providers/hiddenBox.js +262 -272
  124. package/dist/form/widget/providers/radioBox.js +222 -223
  125. package/dist/form/widget/providers/textArea.js +197 -193
  126. package/dist/form/widget/providers/textBox.js +36 -37
  127. package/dist/form/widget/widget.js +480 -508
  128. package/dist/form.js +12 -17
  129. package/dist/formValidator/formValidator.js +74 -91
  130. package/dist/formValidator/highlighters/highlighter.js +43 -51
  131. package/dist/formValidator/highlighters/message.js +31 -32
  132. package/dist/formValidator/highlighters/tooltip.js +33 -35
  133. package/dist/generis/form/css/form.css +7 -7
  134. package/dist/generis/form/css/form.css.map +1 -1
  135. package/dist/generis/form/form.js +323 -331
  136. package/dist/generis/validator/css/validator.css +7 -7
  137. package/dist/generis/validator/css/validator.css.map +1 -1
  138. package/dist/generis/validator/validator.js +215 -216
  139. package/dist/generis/widget/checkBox/checkBox.js +227 -225
  140. package/dist/generis/widget/comboBox/comboBox.js +189 -183
  141. package/dist/generis/widget/css/widget.css +7 -7
  142. package/dist/generis/widget/css/widget.css.map +1 -1
  143. package/dist/generis/widget/hiddenBox/hiddenBox.js +231 -232
  144. package/dist/generis/widget/loader.js +21 -23
  145. package/dist/generis/widget/textBox/textBox.js +178 -172
  146. package/dist/generis/widget/widget.js +255 -266
  147. package/dist/groupedComboBox.js +231 -226
  148. package/dist/groupvalidator.js +30 -34
  149. package/dist/hider.js +43 -50
  150. package/dist/highlighter.js +946 -1057
  151. package/dist/image/ImgStateActive/extractLabel.js +20 -21
  152. package/dist/image/ImgStateActive/helper.js +21 -25
  153. package/dist/image/ImgStateActive/initHelper.js +133 -162
  154. package/dist/image/ImgStateActive/initMediaEditor.js +72 -92
  155. package/dist/image/ImgStateActive/mediaSizer.js +48 -58
  156. package/dist/image/ImgStateActive.js +185 -215
  157. package/dist/incrementer.js +91 -110
  158. package/dist/inplacer.js +108 -132
  159. package/dist/interactUtils.js +56 -67
  160. package/dist/itemButtonList/css/item-button-list.css +23 -24
  161. package/dist/itemButtonList/css/item-button-list.css.map +1 -1
  162. package/dist/itemButtonList.js +444 -469
  163. package/dist/keyNavigation/navigableDomElement.js +124 -158
  164. package/dist/keyNavigation/navigator.js +235 -312
  165. package/dist/listbox/css/listbox.css +7 -7
  166. package/dist/listbox/css/listbox.css.map +1 -1
  167. package/dist/listbox.js +463 -476
  168. package/dist/liststyler.js +75 -76
  169. package/dist/loadingButton/css/button.css +7 -7
  170. package/dist/loadingButton/css/button.css.map +1 -1
  171. package/dist/loadingButton/loadingButton.js +219 -218
  172. package/dist/lock.js +484 -509
  173. package/dist/login/login.js +471 -486
  174. package/dist/maths/calculator/basicCalculator.js +247 -242
  175. package/dist/maths/calculator/calculatorComponent.js +34 -36
  176. package/dist/maths/calculator/core/areaBroker.js +30 -28
  177. package/dist/maths/calculator/core/board.js +805 -872
  178. package/dist/maths/calculator/core/expression.js +500 -518
  179. package/dist/maths/calculator/core/labels.js +240 -235
  180. package/dist/maths/calculator/core/plugin.js +20 -20
  181. package/dist/maths/calculator/core/terms.js +27 -27
  182. package/dist/maths/calculator/core/tokenizer.js +119 -132
  183. package/dist/maths/calculator/core/tokens.js +157 -168
  184. package/dist/maths/calculator/css/calculator.css +7 -7
  185. package/dist/maths/calculator/css/calculator.css.map +1 -1
  186. package/dist/maths/calculator/defaultCalculator.js +24 -27
  187. package/dist/maths/calculator/plugins/core/degrad.js +22 -30
  188. package/dist/maths/calculator/plugins/core/history.js +55 -67
  189. package/dist/maths/calculator/plugins/core/remind.js +22 -25
  190. package/dist/maths/calculator/plugins/core/stepNavigation.js +33 -49
  191. package/dist/maths/calculator/plugins/keyboard/templateKeyboard/templateKeyboard.js +255 -258
  192. package/dist/maths/calculator/plugins/modifiers/pow10.js +47 -52
  193. package/dist/maths/calculator/plugins/modifiers/sign.js +103 -130
  194. package/dist/maths/calculator/plugins/screen/simpleScreen/simpleScreen.js +288 -293
  195. package/dist/maths/calculator/pluginsLoader.js +22 -22
  196. package/dist/maths/calculator/scientificCalculator.js +339 -334
  197. package/dist/mediaEditor/mediaEditorComponent.js +244 -249
  198. package/dist/mediaEditor/plugins/mediaAlignment/helper.js +28 -33
  199. package/dist/mediaEditor/plugins/mediaAlignment/mediaAlignmentComponent.js +228 -236
  200. package/dist/mediaEditor/plugins/mediaAlignment/style.css +7 -7
  201. package/dist/mediaEditor/plugins/mediaDimension/helper.js +81 -106
  202. package/dist/mediaEditor/plugins/mediaDimension/mediaDimensionComponent.js +597 -635
  203. package/dist/mediaEditor/plugins/mediaDimension/style.css +141 -141
  204. package/dist/mediaplayer/css/player.css +7 -7
  205. package/dist/mediaplayer/css/player.css.map +1 -1
  206. package/dist/mediaplayer/players/html5.js +646 -706
  207. package/dist/mediaplayer/players/youtube.js +393 -438
  208. package/dist/mediaplayer/players.js +19 -19
  209. package/dist/mediaplayer/support.js +66 -79
  210. package/dist/mediaplayer/utils/reminder.js +117 -128
  211. package/dist/mediaplayer/utils/timeObserver.js +106 -109
  212. package/dist/mediaplayer/youtubeManager.js +142 -170
  213. package/dist/mediaplayer.js +1484 -1774
  214. package/dist/mediasizer.js +633 -676
  215. package/dist/modal.js +137 -153
  216. package/dist/movableComponent.js +37 -37
  217. package/dist/pageSizeSelector.js +231 -224
  218. package/dist/pageStatus.js +43 -46
  219. package/dist/pagination/css/pagination.css +7 -7
  220. package/dist/pagination/css/pagination.css.map +1 -1
  221. package/dist/pagination/paginationStrategy.js +24 -25
  222. package/dist/pagination/providers/pages.js +281 -291
  223. package/dist/pagination/providers/simple.js +202 -196
  224. package/dist/pagination.js +49 -72
  225. package/dist/previewer.js +113 -138
  226. package/dist/progressbar.js +70 -80
  227. package/dist/propertySelector/css/propertySelector.css +74 -0
  228. package/dist/propertySelector/css/propertySelector.css.map +1 -0
  229. package/dist/propertySelector/propertySelector.js +489 -0
  230. package/dist/report.js +356 -359
  231. package/dist/resource/css/selector.css +7 -7
  232. package/dist/resource/css/selector.css.map +1 -1
  233. package/dist/resource/filters.js +275 -289
  234. package/dist/resource/list.js +1589 -1371
  235. package/dist/resource/selectable.js +101 -136
  236. package/dist/resource/selector.js +855 -914
  237. package/dist/resource/tree.js +1796 -1612
  238. package/dist/resourcemgr/css/resourcemgr.css +7 -7
  239. package/dist/resourcemgr/css/resourcemgr.css.map +1 -1
  240. package/dist/resourcemgr/fileBrowser.js +540 -587
  241. package/dist/resourcemgr/filePreview.js +16 -25
  242. package/dist/resourcemgr/fileSelector.js +520 -547
  243. package/dist/resourcemgr/util/updatePermissions.js +10 -18
  244. package/dist/resourcemgr.js +325 -332
  245. package/dist/scroller.js +29 -33
  246. package/dist/searchModal/advancedSearch.js +949 -833
  247. package/dist/searchModal/css/advancedSearch.css +7 -14
  248. package/dist/searchModal/css/advancedSearch.css.map +1 -1
  249. package/dist/searchModal/css/searchModal.css +28 -8
  250. package/dist/searchModal/css/searchModal.css.map +1 -1
  251. package/dist/searchModal.js +616 -335
  252. package/dist/selecter.js +21 -28
  253. package/dist/stacker.js +52 -60
  254. package/dist/switch/css/switch.css +7 -7
  255. package/dist/switch/css/switch.css.map +1 -1
  256. package/dist/switch/switch.js +305 -311
  257. package/dist/tableModel.js +39 -49
  258. package/dist/tabs/css/tabs.css +12 -12
  259. package/dist/tabs/css/tabs.css.map +1 -1
  260. package/dist/tabs.js +588 -639
  261. package/dist/taskQueue/css/taskQueue.css +7 -7
  262. package/dist/taskQueue/css/taskQueue.css.map +1 -1
  263. package/dist/taskQueue/status.js +317 -322
  264. package/dist/taskQueue/table.js +383 -394
  265. package/dist/taskQueue/taskQueue.js +19 -19
  266. package/dist/taskQueue/taskQueueModel.js +459 -538
  267. package/dist/taskQueue.js +49 -67
  268. package/dist/taskQueueButton/css/taskable.css +7 -7
  269. package/dist/taskQueueButton/css/taskable.css.map +1 -1
  270. package/dist/taskQueueButton/css/treeButton.css +7 -7
  271. package/dist/taskQueueButton/css/treeButton.css.map +1 -1
  272. package/dist/taskQueueButton/standardButton.js +45 -43
  273. package/dist/taskQueueButton/taskable.js +267 -272
  274. package/dist/taskQueueButton/treeButton.js +200 -198
  275. package/dist/themeLoader.js +109 -129
  276. package/dist/themes.js +86 -100
  277. package/dist/toggler.js +76 -88
  278. package/dist/tooltip.js +296 -311
  279. package/dist/tooltipster.js +17 -17
  280. package/dist/transformer.js +140 -169
  281. package/dist/tristateCheckboxGroup/css/tristateCheckboxGroup.css +7 -7
  282. package/dist/tristateCheckboxGroup/css/tristateCheckboxGroup.css.map +1 -1
  283. package/dist/tristateCheckboxGroup.js +319 -325
  284. package/dist/uploader.js +691 -724
  285. package/dist/validator/Report.js +1 -2
  286. package/dist/validator/Validator.js +6 -25
  287. package/dist/validator/validators.js +66 -80
  288. package/dist/validator.js +218 -254
  289. package/dist/waitForMedia.js +45 -51
  290. package/dist/waitingDialog/css/waitingDialog.css +7 -7
  291. package/dist/waitingDialog/css/waitingDialog.css.map +1 -1
  292. package/dist/waitingDialog/waitingDialog.js +59 -77
  293. package/package.json +109 -107
  294. package/scss/basic.scss +16 -16
  295. package/scss/ckeditor/skins/tao/scss/inc/_ck-icons.scss +59 -59
  296. package/scss/ckeditor/skins/tao/scss/inc/_tao.scss +59 -59
  297. package/scss/font/tao/tao.svg +234 -234
  298. package/scss/inc/_base.scss +495 -495
  299. package/scss/inc/_bootstrap.scss +6 -6
  300. package/scss/inc/_buttons.scss +114 -114
  301. package/scss/inc/_colors.scss +93 -88
  302. package/scss/inc/_feedback.scss +150 -150
  303. package/scss/inc/_flex-grid.scss +15 -15
  304. package/scss/inc/_fonts.scss +4 -4
  305. package/scss/inc/_forms.scss +832 -832
  306. package/scss/inc/_functions.scss +283 -283
  307. package/scss/inc/_jquery.nouislider.scss +254 -254
  308. package/scss/inc/_normalize.scss +528 -528
  309. package/scss/inc/_report.scss +67 -67
  310. package/scss/inc/_secondary-properties.scss +89 -89
  311. package/scss/inc/_select2.scss +634 -634
  312. package/scss/inc/_toolbars.scss +155 -155
  313. package/scss/inc/_tooltip.scss +312 -312
  314. package/scss/inc/_variables.scss +21 -21
  315. package/scss/inc/base/_highlight.scss +5 -5
  316. package/scss/inc/base/_list-style.scss +58 -58
  317. package/scss/inc/base/_svg.scss +3 -3
  318. package/scss/inc/base/_table.scss +62 -62
  319. package/scss/inc/fonts/_source-sans-pro.scss +29 -29
  320. package/scss/inc/fonts/_tao-icon-classes.scss +226 -226
  321. package/scss/inc/fonts/_tao-icon-def.scss +12 -12
  322. package/scss/inc/fonts/_tao-icon-vars.scss +240 -240
  323. package/src/actionbar/tpl/main.tpl +8 -8
  324. package/src/actionbar.js +251 -251
  325. package/src/adder.js +250 -250
  326. package/src/animable/absorbable/absorbable.js +134 -134
  327. package/src/animable/absorbable/css/absorb.css +7 -8
  328. package/src/animable/absorbable/css/absorb.css.map +1 -1
  329. package/src/animable/absorbable/scss/absorb.scss +37 -37
  330. package/src/animable/pulsable/css/pulse.css +7 -7
  331. package/src/animable/pulsable/css/pulse.css.map +1 -1
  332. package/src/animable/pulsable/pulsable.js +90 -90
  333. package/src/animable/pulsable/scss/pulse.scss +22 -22
  334. package/src/areaBroker.js +160 -160
  335. package/src/autocomplete/css/autocomplete.css +7 -8
  336. package/src/autocomplete/css/autocomplete.css.map +1 -1
  337. package/src/autocomplete/scss/autocomplete.scss +37 -37
  338. package/src/autocomplete.js +1029 -1029
  339. package/src/autoscroll.js +57 -57
  340. package/src/badge/badge.js +119 -119
  341. package/src/badge/css/badge.css +7 -8
  342. package/src/badge/css/badge.css.map +1 -1
  343. package/src/badge/scss/badge.scss +92 -92
  344. package/src/badge/tpl/badge.tpl +4 -4
  345. package/src/breadcrumbs/css/breadcrumbs.css +7 -7
  346. package/src/breadcrumbs/css/breadcrumbs.css.map +1 -1
  347. package/src/breadcrumbs/scss/breadcrumbs.scss +52 -52
  348. package/src/breadcrumbs/tpl/breadcrumbs.tpl +20 -20
  349. package/src/breadcrumbs.js +99 -99
  350. package/src/btngrouper.js +213 -213
  351. package/src/bulkActionPopup/css/bulkActionPopup.css +7 -7
  352. package/src/bulkActionPopup/css/bulkActionPopup.css.map +1 -1
  353. package/src/bulkActionPopup/scss/bulkActionPopup.scss +63 -63
  354. package/src/bulkActionPopup/tpl/layout.tpl +76 -76
  355. package/src/bulkActionPopup/tpl/select.tpl +8 -8
  356. package/src/bulkActionPopup.js +274 -274
  357. package/src/button/tpl/button.tpl +4 -4
  358. package/src/button.js +135 -135
  359. package/src/calculator/css/calculator.css +10 -10
  360. package/src/calculator/css/calculator.css.map +1 -1
  361. package/src/calculator/scss/calculator.scss +139 -139
  362. package/src/calculator.js +188 -188
  363. package/src/cascadingComboBox.js +126 -126
  364. package/src/ckeditor/ckConfigurator.js +736 -736
  365. package/src/ckeditor/dtdHandler.js +1030 -1030
  366. package/src/class/css/selector.css +7 -7
  367. package/src/class/css/selector.css.map +1 -1
  368. package/src/class/scss/selector.scss +101 -101
  369. package/src/class/selector.js +329 -329
  370. package/src/class/tpl/listItem.tpl +9 -9
  371. package/src/class/tpl/selector.tpl +10 -10
  372. package/src/component/alignable.js +274 -274
  373. package/src/component/containable.js +122 -122
  374. package/src/component/css/components.css +7 -7
  375. package/src/component/css/components.css.map +1 -1
  376. package/src/component/css/windowComponent.css +7 -7
  377. package/src/component/css/windowComponent.css.map +1 -1
  378. package/src/component/draggable.js +104 -104
  379. package/src/component/placeable.js +233 -233
  380. package/src/component/resizable.js +195 -195
  381. package/src/component/scss/components.scss +507 -507
  382. package/src/component/scss/windowComponent.scss +62 -62
  383. package/src/component/stackable.js +67 -67
  384. package/src/component/tpl/window.tpl +7 -7
  385. package/src/component/windowed.js +206 -206
  386. package/src/component.js +401 -401
  387. package/src/container.js +200 -200
  388. package/src/contextualPopup/css/contextualPopup.css +7 -7
  389. package/src/contextualPopup/css/contextualPopup.css.map +1 -1
  390. package/src/contextualPopup/scss/contextualPopup.scss +78 -78
  391. package/src/contextualPopup/tpl/popup.tpl +10 -10
  392. package/src/contextualPopup.js +297 -297
  393. package/src/css/basic.css +103 -106
  394. package/src/css/basic.css.map +1 -1
  395. package/src/dashboard/css/dashboard.css +7 -7
  396. package/src/dashboard/css/dashboard.css.map +1 -1
  397. package/src/dashboard/scss/dashboard.scss +93 -93
  398. package/src/dashboard/tpl/dashboard.tpl +16 -16
  399. package/src/dashboard/tpl/dashboardMetricsList.tpl +15 -15
  400. package/src/dashboard.js +184 -184
  401. package/src/datalist/css/datalist.css +7 -7
  402. package/src/datalist/css/datalist.css.map +1 -1
  403. package/src/datalist/scss/datalist.scss +116 -116
  404. package/src/datalist/tpl/list.tpl +24 -24
  405. package/src/datalist/tpl/main.tpl +44 -44
  406. package/src/datalist.js +500 -500
  407. package/src/datatable/css/datatable.css +11 -7
  408. package/src/datatable/css/datatable.css.map +1 -1
  409. package/src/datatable/filterStrategy/filterStrategy.js +70 -70
  410. package/src/datatable/filterStrategy/multiple.js +126 -126
  411. package/src/datatable/filterStrategy/single.js +108 -108
  412. package/src/datatable/scss/datatable.scss +149 -146
  413. package/src/datatable/tpl/button.tpl +6 -6
  414. package/src/datatable/tpl/layout.tpl +158 -158
  415. package/src/datatable.js +1056 -1056
  416. package/src/dateRange/css/dateRange.css +7 -7
  417. package/src/dateRange/css/dateRange.css.map +1 -1
  418. package/src/dateRange/dateRange.js +341 -341
  419. package/src/dateRange/scss/dateRange.scss +7 -7
  420. package/src/dateRange/tpl/select.tpl +18 -18
  421. package/src/datetime/css/picker.css +7 -7
  422. package/src/datetime/css/picker.css.map +1 -1
  423. package/src/datetime/picker.js +576 -576
  424. package/src/datetime/scss/picker.scss +192 -192
  425. package/src/datetime/tpl/picker.tpl +18 -18
  426. package/src/deleter/undo.tpl +6 -6
  427. package/src/deleter.js +296 -296
  428. package/src/destination/css/selector.css +7 -7
  429. package/src/destination/css/selector.css.map +1 -1
  430. package/src/destination/scss/selector.scss +36 -36
  431. package/src/destination/selector.js +195 -195
  432. package/src/destination/tpl/selector.tpl +13 -13
  433. package/src/dialog/alert.js +70 -70
  434. package/src/dialog/confirm.js +85 -85
  435. package/src/dialog/confirmDelete.js +95 -95
  436. package/src/dialog/tpl/body.tpl +24 -24
  437. package/src/dialog/tpl/buttons.tpl +6 -6
  438. package/src/dialog/tpl/checkbox.tpl +5 -5
  439. package/src/dialog.js +517 -517
  440. package/src/disabler.js +230 -230
  441. package/src/documentViewer/css/documentViewer.css +7 -7
  442. package/src/documentViewer/css/documentViewer.css.map +1 -1
  443. package/src/documentViewer/providers/pdfViewer/fallback/viewer.js +69 -69
  444. package/src/documentViewer/providers/pdfViewer/pdfjs/areaBroker.js +41 -41
  445. package/src/documentViewer/providers/pdfViewer/pdfjs/findBar.js +475 -475
  446. package/src/documentViewer/providers/pdfViewer/pdfjs/findBar.tpl +20 -20
  447. package/src/documentViewer/providers/pdfViewer/pdfjs/match.tpl +1 -1
  448. package/src/documentViewer/providers/pdfViewer/pdfjs/page.tpl +4 -4
  449. package/src/documentViewer/providers/pdfViewer/pdfjs/pageView.js +318 -318
  450. package/src/documentViewer/providers/pdfViewer/pdfjs/pagesManager.js +167 -167
  451. package/src/documentViewer/providers/pdfViewer/pdfjs/searchEngine.js +451 -451
  452. package/src/documentViewer/providers/pdfViewer/pdfjs/textManager.js +252 -252
  453. package/src/documentViewer/providers/pdfViewer/pdfjs/viewer.js +299 -299
  454. package/src/documentViewer/providers/pdfViewer/pdfjs/viewer.tpl +16 -16
  455. package/src/documentViewer/providers/pdfViewer/pdfjs/wrapper.js +351 -351
  456. package/src/documentViewer/providers/pdfViewer.js +93 -93
  457. package/src/documentViewer/scss/documentViewer.scss +184 -184
  458. package/src/documentViewer/viewerFactory.js +191 -191
  459. package/src/documentViewer.js +238 -238
  460. package/src/dropdown/css/dropdown.css +7 -7
  461. package/src/dropdown/css/dropdown.css.map +1 -1
  462. package/src/dropdown/scss/dropdown.scss +99 -99
  463. package/src/dropdown/tpl/dropdown.tpl +8 -8
  464. package/src/dropdown/tpl/list-item.tpl +4 -4
  465. package/src/dropdown.js +255 -255
  466. package/src/durationer.js +222 -222
  467. package/src/dynamicComponent/css/dynamicComponent.css +7 -7
  468. package/src/dynamicComponent/css/dynamicComponent.css.map +1 -1
  469. package/src/dynamicComponent/scss/dynamicComponent.scss +98 -98
  470. package/src/dynamicComponent/tpl/layout.tpl +17 -17
  471. package/src/dynamicComponent.js +554 -554
  472. package/src/feedback/feedback.tpl +7 -7
  473. package/src/feedback.js +295 -295
  474. package/src/figure/FigureStateActive.js +174 -174
  475. package/src/filesender.js +114 -114
  476. package/src/filter/template.tpl +5 -5
  477. package/src/filter.js +135 -135
  478. package/src/form/css/dropdownForm.css +7 -7
  479. package/src/form/css/dropdownForm.css.map +1 -1
  480. package/src/form/css/form.css +7 -7
  481. package/src/form/css/form.css.map +1 -1
  482. package/src/form/dropdownForm.js +281 -281
  483. package/src/form/form.js +688 -688
  484. package/src/form/scss/dropdownForm.scss +60 -60
  485. package/src/form/scss/form.scss +25 -25
  486. package/src/form/simpleForm.js +125 -125
  487. package/src/form/tpl/dropdownForm.tpl +4 -4
  488. package/src/form/tpl/form.tpl +7 -7
  489. package/src/form/validator/css/validator.css +7 -7
  490. package/src/form/validator/css/validator.css.map +1 -1
  491. package/src/form/validator/renderer.js +118 -118
  492. package/src/form/validator/scss/validator.scss +14 -14
  493. package/src/form/validator/tpl/message.tpl +1 -1
  494. package/src/form/validator/tpl/validator.tpl +1 -1
  495. package/src/form/validator/validator.js +220 -220
  496. package/src/form/widget/css/widget.css +7 -7
  497. package/src/form/widget/css/widget.css.map +1 -1
  498. package/src/form/widget/definitions.js +51 -51
  499. package/src/form/widget/loader.js +40 -40
  500. package/src/form/widget/providers/checkBox.js +138 -138
  501. package/src/form/widget/providers/comboBox.js +63 -63
  502. package/src/form/widget/providers/default.js +90 -90
  503. package/src/form/widget/providers/hidden.js +62 -62
  504. package/src/form/widget/providers/hiddenBox.js +152 -152
  505. package/src/form/widget/providers/radioBox.js +99 -99
  506. package/src/form/widget/providers/textArea.js +52 -52
  507. package/src/form/widget/providers/textBox.js +48 -48
  508. package/src/form/widget/scss/widget.scss +55 -55
  509. package/src/form/widget/tpl/checkBox.tpl +25 -25
  510. package/src/form/widget/tpl/comboBox.tpl +13 -13
  511. package/src/form/widget/tpl/hidden.tpl +1 -1
  512. package/src/form/widget/tpl/hiddenBox.tpl +17 -17
  513. package/src/form/widget/tpl/label.tpl +6 -6
  514. package/src/form/widget/tpl/radioBox.tpl +25 -25
  515. package/src/form/widget/tpl/textArea.tpl +8 -8
  516. package/src/form/widget/tpl/widget.tpl +8 -8
  517. package/src/form/widget/widget.js +372 -372
  518. package/src/form.js +53 -53
  519. package/src/formValidator/formValidator.js +253 -253
  520. package/src/formValidator/highlighters/highlighter.js +102 -102
  521. package/src/formValidator/highlighters/message.js +70 -70
  522. package/src/formValidator/highlighters/tooltip.js +78 -78
  523. package/src/generis/form/css/form.css +7 -7
  524. package/src/generis/form/css/form.css.map +1 -1
  525. package/src/generis/form/form.js +239 -239
  526. package/src/generis/form/readme.md +70 -70
  527. package/src/generis/form/scss/form.scss +23 -23
  528. package/src/generis/form/tpl/form.tpl +16 -16
  529. package/src/generis/validator/css/validator.css +7 -7
  530. package/src/generis/validator/css/validator.css.map +1 -1
  531. package/src/generis/validator/readme.md +46 -46
  532. package/src/generis/validator/scss/validator.scss +13 -13
  533. package/src/generis/validator/validator.js +128 -128
  534. package/src/generis/widget/checkBox/checkBox.js +112 -112
  535. package/src/generis/widget/checkBox/checkBox.tpl +18 -18
  536. package/src/generis/widget/comboBox/comboBox.js +67 -67
  537. package/src/generis/widget/comboBox/comboBox.tpl +12 -12
  538. package/src/generis/widget/css/widget.css +7 -7
  539. package/src/generis/widget/css/widget.css.map +1 -1
  540. package/src/generis/widget/hiddenBox/hiddenBox.js +132 -132
  541. package/src/generis/widget/hiddenBox/hiddenBox.tpl +16 -16
  542. package/src/generis/widget/loader.js +49 -49
  543. package/src/generis/widget/readme.md +59 -59
  544. package/src/generis/widget/scss/widget.scss +61 -61
  545. package/src/generis/widget/textBox/textBox.js +65 -65
  546. package/src/generis/widget/textBox/textBox.tpl +7 -7
  547. package/src/generis/widget/widget.js +164 -164
  548. package/src/generis/widget/widget.tpl +5 -5
  549. package/src/groupedComboBox.js +99 -99
  550. package/src/groupvalidator.js +84 -84
  551. package/src/hider.js +88 -88
  552. package/src/highlighter.js +1166 -1166
  553. package/src/image/ImgStateActive/extractLabel.js +29 -29
  554. package/src/image/ImgStateActive/helper.js +36 -36
  555. package/src/image/ImgStateActive/initHelper.js +137 -137
  556. package/src/image/ImgStateActive/initMediaEditor.js +92 -92
  557. package/src/image/ImgStateActive/mediaSizer.js +63 -63
  558. package/src/image/ImgStateActive.js +115 -115
  559. package/src/incrementer.js +319 -319
  560. package/src/inplacer.js +316 -316
  561. package/src/interactUtils.js +140 -140
  562. package/src/itemButtonList/css/item-button-list.css +23 -24
  563. package/src/itemButtonList/css/item-button-list.css.map +1 -1
  564. package/src/itemButtonList/scss/item-button-list.scss +236 -236
  565. package/src/itemButtonList/tpl/itemButtonList.tpl +21 -21
  566. package/src/itemButtonList.js +274 -274
  567. package/src/keyNavigation/navigableDomElement.js +282 -282
  568. package/src/keyNavigation/navigator.js +543 -543
  569. package/src/listbox/css/listbox.css +7 -7
  570. package/src/listbox/css/listbox.css.map +1 -1
  571. package/src/listbox/scss/listbox.scss +116 -116
  572. package/src/listbox/tpl/list.tpl +14 -14
  573. package/src/listbox/tpl/main.tpl +9 -9
  574. package/src/listbox.js +251 -251
  575. package/src/liststyler.js +155 -155
  576. package/src/loadingButton/css/button.css +7 -7
  577. package/src/loadingButton/css/button.css.map +1 -1
  578. package/src/loadingButton/loadingButton.js +110 -110
  579. package/src/loadingButton/scss/button.scss +41 -41
  580. package/src/loadingButton/tpl/button.tpl +5 -5
  581. package/src/lock/lock.tpl +16 -16
  582. package/src/lock.js +395 -395
  583. package/src/login/login.js +322 -322
  584. package/src/login/tpl/login.tpl +29 -29
  585. package/src/login/tpl/passwordReveal.tpl +7 -7
  586. package/src/maths/calculator/basicCalculator.js +55 -55
  587. package/src/maths/calculator/calculatorComponent.js +128 -128
  588. package/src/maths/calculator/core/areaBroker.js +38 -38
  589. package/src/maths/calculator/core/board.js +841 -841
  590. package/src/maths/calculator/core/expression.js +430 -430
  591. package/src/maths/calculator/core/labels.js +116 -116
  592. package/src/maths/calculator/core/plugin.js +40 -40
  593. package/src/maths/calculator/core/terms.js +459 -459
  594. package/src/maths/calculator/core/tokenizer.js +245 -245
  595. package/src/maths/calculator/core/tokens.js +178 -178
  596. package/src/maths/calculator/core/tpl/board.tpl +4 -4
  597. package/src/maths/calculator/css/calculator.css +7 -7
  598. package/src/maths/calculator/css/calculator.css.map +1 -1
  599. package/src/maths/calculator/defaultCalculator.js +66 -66
  600. package/src/maths/calculator/plugins/core/degrad.js +90 -90
  601. package/src/maths/calculator/plugins/core/history.js +166 -166
  602. package/src/maths/calculator/plugins/core/remind.js +96 -96
  603. package/src/maths/calculator/plugins/core/stepNavigation.js +175 -175
  604. package/src/maths/calculator/plugins/keyboard/templateKeyboard/defaultTemplate.tpl +36 -36
  605. package/src/maths/calculator/plugins/keyboard/templateKeyboard/templateKeyboard.js +91 -91
  606. package/src/maths/calculator/plugins/modifiers/pow10.js +143 -143
  607. package/src/maths/calculator/plugins/modifiers/sign.js +339 -339
  608. package/src/maths/calculator/plugins/screen/simpleScreen/defaultTemplate.tpl +3 -3
  609. package/src/maths/calculator/plugins/screen/simpleScreen/history.tpl +3 -3
  610. package/src/maths/calculator/plugins/screen/simpleScreen/simpleScreen.js +191 -191
  611. package/src/maths/calculator/pluginsLoader.js +46 -46
  612. package/src/maths/calculator/scientificCalculator.js +74 -74
  613. package/src/maths/calculator/scss/calculator.scss +396 -396
  614. package/src/maths/calculator/tpl/basicKeyboard.tpl +37 -37
  615. package/src/maths/calculator/tpl/basicScreen.tpl +2 -2
  616. package/src/maths/calculator/tpl/scientificKeyboard.tpl +61 -61
  617. package/src/maths/calculator/tpl/scientificScreen.tpl +3 -3
  618. package/src/mediaEditor/mediaEditorComponent.js +141 -141
  619. package/src/mediaEditor/plugins/mediaAlignment/helper.js +62 -62
  620. package/src/mediaEditor/plugins/mediaAlignment/mediaAlignmentComponent.js +99 -99
  621. package/src/mediaEditor/plugins/mediaAlignment/style.css +7 -7
  622. package/src/mediaEditor/plugins/mediaAlignment/tpl/mediaAlignment.tpl +25 -25
  623. package/src/mediaEditor/plugins/mediaDimension/helper.js +189 -189
  624. package/src/mediaEditor/plugins/mediaDimension/mediaDimensionComponent.js +561 -561
  625. package/src/mediaEditor/plugins/mediaDimension/style.css +141 -141
  626. package/src/mediaEditor/plugins/mediaDimension/tpl/mediaDimension.tpl +55 -55
  627. package/src/mediaEditor/tpl/editor.tpl +4 -4
  628. package/src/mediaplayer/css/player.css +7 -7
  629. package/src/mediaplayer/css/player.css.map +1 -1
  630. package/src/mediaplayer/players/html5.js +564 -564
  631. package/src/mediaplayer/players/youtube.js +323 -323
  632. package/src/mediaplayer/players.js +29 -29
  633. package/src/mediaplayer/readme.md +305 -305
  634. package/src/mediaplayer/scss/player.scss +569 -569
  635. package/src/mediaplayer/support.js +126 -126
  636. package/src/mediaplayer/tpl/audio.tpl +6 -6
  637. package/src/mediaplayer/tpl/player.tpl +37 -37
  638. package/src/mediaplayer/tpl/source.tpl +1 -1
  639. package/src/mediaplayer/tpl/video.tpl +6 -6
  640. package/src/mediaplayer/tpl/youtube.tpl +1 -1
  641. package/src/mediaplayer/utils/reminder.js +184 -184
  642. package/src/mediaplayer/utils/timeObserver.js +143 -143
  643. package/src/mediaplayer/youtubeManager.js +161 -161
  644. package/src/mediaplayer.js +1606 -1606
  645. package/src/mediasizer/mediasizer.tpl +55 -55
  646. package/src/mediasizer.js +635 -635
  647. package/src/modal.js +365 -365
  648. package/src/movableComponent.js +78 -78
  649. package/src/pageSizeSelector/tpl/pageSizeSelector.tpl +9 -9
  650. package/src/pageSizeSelector.js +107 -107
  651. package/src/pageStatus.js +147 -147
  652. package/src/pagination/css/pagination.css +7 -7
  653. package/src/pagination/css/pagination.css.map +1 -1
  654. package/src/pagination/paginationStrategy.js +53 -53
  655. package/src/pagination/providers/pages.js +161 -161
  656. package/src/pagination/providers/simple.js +74 -74
  657. package/src/pagination/providers/tpl/pages/page.tpl +1 -1
  658. package/src/pagination/providers/tpl/pages.tpl +8 -8
  659. package/src/pagination/providers/tpl/simple.tpl +7 -7
  660. package/src/pagination/scss/pagination.scss +111 -111
  661. package/src/pagination.js +237 -237
  662. package/src/previewer.js +300 -300
  663. package/src/progressbar.js +165 -165
  664. package/src/propertySelector/css/propertySelector.css +74 -0
  665. package/src/propertySelector/css/propertySelector.css.map +1 -0
  666. package/src/propertySelector/propertySelector.js +286 -0
  667. package/src/propertySelector/scss/propertySelector.scss +66 -0
  668. package/src/propertySelector/tpl/highlighted-text.tpl +1 -0
  669. package/src/propertySelector/tpl/property-description.tpl +13 -0
  670. package/src/propertySelector/tpl/property-selector.tpl +7 -0
  671. package/src/report/feedback.tpl +11 -11
  672. package/src/report/layout.tpl +10 -10
  673. package/src/report.js +184 -184
  674. package/src/resource/css/selector.css +7 -7
  675. package/src/resource/css/selector.css.map +1 -1
  676. package/src/resource/filters.js +208 -208
  677. package/src/resource/list.js +200 -200
  678. package/src/resource/scss/_filters.scss +26 -26
  679. package/src/resource/scss/_resource-list.scss +107 -107
  680. package/src/resource/scss/_resource-tree.scss +205 -205
  681. package/src/resource/scss/selector.scss +187 -187
  682. package/src/resource/selectable.js +322 -322
  683. package/src/resource/selector.js +871 -871
  684. package/src/resource/tpl/filters.tpl +2 -2
  685. package/src/resource/tpl/list.tpl +7 -7
  686. package/src/resource/tpl/listNode.tpl +4 -4
  687. package/src/resource/tpl/selector.tpl +46 -46
  688. package/src/resource/tpl/tree.tpl +4 -4
  689. package/src/resource/tpl/treeNode.tpl +30 -30
  690. package/src/resource/tree.js +400 -400
  691. package/src/resourcemgr/css/resourcemgr.css +7 -7
  692. package/src/resourcemgr/css/resourcemgr.css.map +1 -1
  693. package/src/resourcemgr/fileBrowser.js +381 -381
  694. package/src/resourcemgr/filePreview.js +73 -73
  695. package/src/resourcemgr/fileSelector.js +348 -348
  696. package/src/resourcemgr/scss/resourcemgr.scss +254 -254
  697. package/src/resourcemgr/tpl/fileSelect.tpl +39 -39
  698. package/src/resourcemgr/tpl/folder.tpl +11 -11
  699. package/src/resourcemgr/tpl/layout.tpl +84 -84
  700. package/src/resourcemgr/tpl/rootFolder.tpl +13 -13
  701. package/src/resourcemgr/util/updatePermissions.js +53 -53
  702. package/src/resourcemgr.js +216 -216
  703. package/src/scroller.js +94 -94
  704. package/src/scss/basic.scss +16 -16
  705. package/src/searchModal/advancedSearch.js +638 -601
  706. package/src/searchModal/css/advancedSearch.css +7 -14
  707. package/src/searchModal/css/advancedSearch.css.map +1 -1
  708. package/src/searchModal/css/searchModal.css +28 -8
  709. package/src/searchModal/css/searchModal.css.map +1 -1
  710. package/src/searchModal/scss/advancedSearch.scss +171 -177
  711. package/src/searchModal/scss/searchModal.scss +393 -375
  712. package/src/searchModal/tpl/advanced-search.tpl +9 -9
  713. package/src/searchModal/tpl/criteria-alias.tpl +1 -0
  714. package/src/searchModal/tpl/criteria-class-label.tpl +1 -0
  715. package/src/searchModal/tpl/criteria-label.tpl +1 -0
  716. package/src/searchModal/tpl/highlighted-text.tpl +1 -0
  717. package/src/searchModal/tpl/info-message.tpl +3 -3
  718. package/src/searchModal/tpl/invalid-criteria-warning.tpl +10 -10
  719. package/src/searchModal/tpl/layout.tpl +27 -25
  720. package/src/searchModal/tpl/list-checkbox-criterion.tpl +17 -12
  721. package/src/searchModal/tpl/list-select-criterion.tpl +12 -6
  722. package/src/searchModal/tpl/property-select-button.tpl +1 -0
  723. package/src/searchModal/tpl/results-container.tpl +1 -0
  724. package/src/searchModal/tpl/text-criterion.tpl +11 -6
  725. package/src/searchModal.js +761 -496
  726. package/src/selecter.js +43 -43
  727. package/src/stacker.js +133 -133
  728. package/src/switch/css/switch.css +7 -7
  729. package/src/switch/css/switch.css.map +1 -1
  730. package/src/switch/scss/switch.scss +83 -83
  731. package/src/switch/switch.js +195 -195
  732. package/src/switch/tpl/switch.tpl +7 -7
  733. package/src/tableModel.js +112 -112
  734. package/src/tabs/css/tabs.css +12 -12
  735. package/src/tabs/css/tabs.css.map +1 -1
  736. package/src/tabs/scss/tabs.scss +50 -50
  737. package/src/tabs/tpl/panel.tpl +3 -3
  738. package/src/tabs/tpl/tabs.tpl +10 -10
  739. package/src/tabs.js +528 -528
  740. package/src/taskQueue/css/taskQueue.css +7 -7
  741. package/src/taskQueue/css/taskQueue.css.map +1 -1
  742. package/src/taskQueue/scss/taskQueue.scss +47 -47
  743. package/src/taskQueue/status.js +228 -228
  744. package/src/taskQueue/table.js +350 -350
  745. package/src/taskQueue/taskQueue.js +33 -33
  746. package/src/taskQueue/taskQueueModel.js +548 -548
  747. package/src/taskQueue/tpl/statusMessage.tpl +7 -7
  748. package/src/taskQueue.js +218 -218
  749. package/src/taskQueueButton/css/taskable.css +7 -7
  750. package/src/taskQueueButton/css/taskable.css.map +1 -1
  751. package/src/taskQueueButton/css/treeButton.css +7 -7
  752. package/src/taskQueueButton/css/treeButton.css.map +1 -1
  753. package/src/taskQueueButton/scss/taskable.scss +4 -4
  754. package/src/taskQueueButton/scss/treeButton.scss +34 -34
  755. package/src/taskQueueButton/standardButton.js +108 -108
  756. package/src/taskQueueButton/taskable.js +202 -202
  757. package/src/taskQueueButton/tpl/report.tpl +5 -5
  758. package/src/taskQueueButton/tpl/treeButton.tpl +6 -6
  759. package/src/taskQueueButton/treeButton.js +109 -109
  760. package/src/themeLoader.js +252 -252
  761. package/src/themes.js +162 -162
  762. package/src/toggler.js +200 -200
  763. package/src/tooltip/default.tpl +3 -3
  764. package/src/tooltip.js +160 -160
  765. package/src/tooltipster.js +25 -25
  766. package/src/transformer.js +327 -327
  767. package/src/tristateCheckboxGroup/css/tristateCheckboxGroup.css +7 -7
  768. package/src/tristateCheckboxGroup/css/tristateCheckboxGroup.css.map +1 -1
  769. package/src/tristateCheckboxGroup/scss/tristateCheckboxGroup.scss +15 -15
  770. package/src/tristateCheckboxGroup/tpl/li.tpl +6 -6
  771. package/src/tristateCheckboxGroup.js +207 -207
  772. package/src/uploader/fileEntry.tpl +6 -6
  773. package/src/uploader/uploader.tpl +32 -32
  774. package/src/uploader.js +594 -594
  775. package/src/validator/Report.js +10 -10
  776. package/src/validator/Validator.js +108 -108
  777. package/src/validator/validators.js +220 -220
  778. package/src/validator.js +264 -264
  779. package/src/waitForMedia.js +82 -82
  780. package/src/waitingDialog/css/waitingDialog.css +7 -7
  781. package/src/waitingDialog/css/waitingDialog.css.map +1 -1
  782. package/src/waitingDialog/scss/waitingDialog.scss +34 -34
  783. package/src/waitingDialog/waitingDialog.js +240 -240
package/src/datatable.js CHANGED
@@ -1,1056 +1,1056 @@
1
- /**
2
- * This program is free software; you can redistribute it and/or
3
- * modify it under the terms of the GNU General Public License
4
- * as published by the Free Software Foundation; under version 2
5
- * of the License (non-upgradable).
6
- *
7
- * This program is distributed in the hope that it will be useful,
8
- * but WITHOUT ANY WARRANTY; without even the implied warranty of
9
- * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
10
- * GNU General Public License for more details.
11
- *
12
- * You should have received a copy of the GNU General Public License
13
- * along with this program; if not, write to the Free Software
14
- * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
15
- *
16
- * Copyright (c) 2015-2020 (original work) Open Assessment Technologies SA;
17
- */
18
-
19
- import $ from 'jquery';
20
- import _ from 'lodash';
21
- import __ from 'i18n';
22
- import Pluginifier from 'core/pluginifier';
23
- import layout from 'ui/datatable/tpl/layout';
24
- import buttonTpl from 'ui/datatable/tpl/button';
25
- import filterStrategyFactory from 'ui/datatable/filterStrategy/filterStrategy';
26
- import paginationComponent from 'ui/pagination';
27
- import loadingBar from 'layout/loading-bar';
28
- import loggerFactory from 'core/logger';
29
- import httpErrorParser from 'util/httpErrorParser';
30
- import pageSizeSelector from 'ui/pageSizeSelector';
31
- import 'ui/datatable/css/datatable.css';
32
- import DOMPurify from 'lib/dompurify/purify';
33
-
34
- const ns = 'datatable';
35
-
36
- const dataNs = 'ui.' + ns;
37
-
38
- const defaults = {
39
- atomicUpdate: false,
40
- start: 0,
41
- rows: 25,
42
- page: 1,
43
- sortby: 'id',
44
- sortorder: 'asc',
45
- sorttype: 'string',
46
- paginationStrategyTop: 'none',
47
- paginationStrategyBottom: 'simple',
48
- labels: {
49
- filter: __('Filter'),
50
- empty: __('Nothing to list!'),
51
- available: __('Available'),
52
- loading: __('Loading'),
53
- actions: __('Actions')
54
- },
55
- pageSizeSelector: false
56
- };
57
-
58
- const logger = loggerFactory('ui/datatable');
59
-
60
- /**
61
- * The CSS class used to hide an element
62
- * @type {String}
63
- */
64
- const hiddenCls = 'hidden';
65
-
66
- /**
67
- * Deactivate pagination's
68
- * @param {Array} pagination
69
- */
70
- const disablePagination = pagination => {
71
- if (pagination && pagination.length) {
72
- pagination.forEach(step => {
73
- step.disable();
74
- });
75
- }
76
- };
77
-
78
- /**
79
- * Activate pagination's
80
- * @param {Array} pagination
81
- */
82
- const enablePagination = pagination => {
83
- if (pagination && pagination.length) {
84
- pagination.forEach(step => {
85
- step.enable();
86
- });
87
- }
88
- };
89
-
90
- /**
91
- * Helper for reading actions value in context
92
- *
93
- * @param {String} property
94
- * @param {Object} action
95
- * @param {Object} context
96
- */
97
- const getPropertyValue = (property, action, context) => {
98
- const value = action[property];
99
-
100
- return _.isFunction(value) ? value.apply(context) : value;
101
- };
102
-
103
- /**
104
- * Update the data table status in the header
105
- *
106
- * @param {Object} options
107
- * @param {jQueryElement} $container
108
- * @param {Object} dataset
109
- */
110
- const updateHeaderStatus = (options, $container, dataset) => {
111
- if (!options.status) {
112
- return;
113
- }
114
-
115
- const $statusEmpty = $container.find('.empty-list');
116
- const $statusAvailable = $container.find('.available-list');
117
- const $statusCount = $statusAvailable.find('.count');
118
-
119
- $container.find('.loading').addClass(hiddenCls);
120
-
121
- // when the status is enabled, the response must contain the total amount of records
122
- const amount = dataset.amount || dataset.length;
123
-
124
- if (amount) {
125
- $statusCount.text(amount);
126
- $statusAvailable.removeClass(hiddenCls);
127
- $statusEmpty.addClass(hiddenCls);
128
- } else {
129
- $statusEmpty.removeClass(hiddenCls);
130
- $statusAvailable.addClass(hiddenCls);
131
- }
132
- };
133
-
134
- /**
135
- * The dataTable component makes you able to browse items and bind specific
136
- * actions to undertake for edition and removal of them.
137
- *
138
- * Parameters that will be send to backend by component:
139
- *
140
- * Pagination
141
- * @param {Number} rows - count of rows, that should be returned from backend, in other words limit.
142
- * @param {Number} page - number of page, that should be requested.
143
- *
144
- * Sorting
145
- * @param {String} sortby - name of column
146
- * @param {String} sortorder - order of sorting, can be 'asc' or 'desc' for ascending sorting and descending sorting respectively.
147
- * @param {String} sorttype - type of sorting, can be 'string' or 'numeric' for proper sorting numeric and string values.
148
- *
149
- * Filtering
150
- * @param {String} filterstrategy - filtering strategy. Default is single (see ui/datatable/filterStrategy/single.js).
151
- * @param {String} filterquery - query string for filtering of rows.
152
- * @param {String[]} filtercolumns[] - array of columns, in which will be implemented search during filtering process.
153
- * For column filter it will be only one item with column name, but component has ability define list of columns for default filter (in top toolbar).
154
- * Backend should correctly receive this list of columns and do search in accordance with this parameters.
155
- * By default, columns are not defined, so this parameter not will be sent. If filtercolumns[] not exists, backend should search by all columns.
156
- *
157
- * @example of query (GET): rows=25&page=1&sortby=login&sortorder=asc&filterquery=loginame&filtercolumns[]=login
158
- *
159
- * @exports ui/datatable
160
- */
161
- const dataTable = {
162
- /**
163
- * Used for generating action button action button
164
- * @typedef Action
165
- * @type {Object}
166
- * @property {String} id ID is added to the button class
167
- * @property {String} [title] Button title
168
- * @property {Boolean} [disabled] When present, button should be disabled
169
- * @property {String} [icon] Generate button icon
170
- * @property {Boolean} [hidden] When present, button is hidden
171
- * @property {Function} [action] Handler on button click
172
- */
173
-
174
- /**
175
- * Used for generating action button from Object
176
- * @deprecated
177
- * @typedef {{
178
- * [key: Action.id & Action.icon & Action.title ]: Action.action,
179
- * }} ActionsObject
180
- *
181
- * @example
182
- * {
183
- * actions: {
184
- * edit: editUser,
185
- * remove: removeUser,
186
- * }
187
- * }
188
- *
189
- * ! IMPORTANT USE INSTEAD:
190
- * {
191
- * actions: [
192
- * {
193
- * id: "edit",
194
- * title: __("Edit"),
195
- * icon: "edit",
196
- * action: editUser
197
- * },
198
- * {
199
- * id: "edit",
200
- * title: __("Edit"),
201
- * icon: "edit",
202
- * action: editUser
203
- * }
204
- * ]
205
- * }
206
- */
207
-
208
- /**
209
- * Initialize the plugin.
210
- *
211
- * Called the jQuery way once registered by the Pluginifier.
212
- * @example $('selector').datatable([], {});
213
- *
214
- * @constructor
215
- * @param {Object} options - the plugin options.
216
- * @param {String} options.url - the URL of the service used to retrieve the resources.
217
- * @param {Object[]} options.model - the model definition.
218
- * @param {ActionsObject | Action[]} options.actions - Generates action buttons
219
- * @param {Function} options.listeners.xxx - the callback function for event xxx, parameters depends to event trigger call.
220
- * @param {Boolean} options.selectable - enables the selection of rows using checkboxes.
221
- * @param {Boolean} options.rowSelection - enables the selection of rows by clicking on them.
222
- * @param {Object} options.tools - a list of tool buttons to display above the table.
223
- * @param {Object|Boolean} options.status - allow to display a status bar.
224
- * @param {Object|Boolean} options.filter - allow to display a filter bar.
225
- * @param {String} options.filterStrategy - 'multiple' | 'single' -- filtered by all filters together or filtering allowed only by one field at the moment (default 'single'),
226
- * @param {String} options.filterSelector - css selector for search of filter inputs, by defaul 'select, input'
227
- * @param {String} options.filterTransform - transform filter value before send to server.
228
- * @param {String[]} options.filter.columns - a list of columns that will be used for default filter. Can be overridden by column filter.
229
- * @param {String} options.filterquery - a query string for filtering, using only in runtime.
230
- * @param {String[]} options.filtercolumns - a list of columns, in that should be done search, using only in runtime.
231
- * @param {String} options.paginationStrategyTop - 'none' | 'pages' | 'simple' -- 'none' by default (next/prev), 'pages' show pages and extended control for pagination
232
- * @param {String} options.paginationStrategyBottom - 'none' | 'pages' | 'simple' -- 'simple' by default (next/prev), 'pages' show pages and extended control for pagination
233
- * @param {Object} options.labels - list of labels in datatable interface, that can be overridden by incoming options
234
- * @param {String} options.emptyText - text that will be shown when no data found for showing in the grid.
235
- * @param {Boolean} options.pageSizeSelector - flag that indicates if control for changing page size should be displayed
236
- * @param {Boolean} options.atomicUpdate - allowed to keep the datatable state to be able on "render" event, compare with new state and atomically update the table cells.
237
- * @param {Function} options.requestInterceptor - Intercept sending AJAX request. The call of function must returns promise with provided data.
238
- * @param {Object} [data] - inject predefined data to avoid the first query.
239
- * @fires dataTable#create.datatable
240
- * @returns {jQueryElement} for chaining
241
- */
242
- init(options, data) {
243
- options = _.defaults(options, defaults);
244
-
245
- return this.each(function () {
246
- const $elt = $(this);
247
- const currentOptions = $elt.data(dataNs);
248
-
249
- if (options.atomicUpdate && data) {
250
- $elt.data(`${dataNs}state`, data.data);
251
- }
252
- // implement encapsulated pages for the datatable
253
- $elt.paginations = [];
254
-
255
- if (!currentOptions) {
256
- //add data to the element
257
- $elt.data(dataNs, options);
258
-
259
- $elt.one('load.' + ns, function () {
260
- /**
261
- * @event dataTable#create.datatable
262
- */
263
- $elt.trigger('create.' + ns);
264
- });
265
-
266
- if (data) {
267
- dataTable._render($elt, data);
268
- } else {
269
- dataTable._query($elt);
270
- }
271
- } else {
272
- // update existing options
273
- $elt.data(dataNs, _.merge(currentOptions, options));
274
-
275
- dataTable._refresh($elt, data);
276
- }
277
- });
278
- },
279
-
280
- /**
281
- * Refresh the data table using current options
282
- *
283
- * Called the jQuery way once registered by the Pluginifier.
284
- * @example $('selector').datatable('refresh');
285
- *
286
- * @param {jQueryElement} $elt - plugin's element
287
- * @param {Object} [data] - Data to render immediately, prevents the query to be made.
288
- */
289
- _refresh($elt, data) {
290
- if (data) {
291
- this._render($elt, data);
292
- } else {
293
- this._query($elt);
294
- }
295
- },
296
-
297
- /**
298
- * Query the server for data and load the table.
299
- *
300
- * @private
301
- * @param {jQueryElement} $elt - plugin's element
302
- * @param $filter
303
- * @fires dataTable#query.datatable
304
- */
305
- _query($elt, $filter) {
306
- const self = this;
307
- let options = $elt.data(dataNs);
308
-
309
- loadingBar.start();
310
-
311
- if (!$filter) {
312
- $filter = $('.filter', $elt);
313
- }
314
-
315
- options = _.assign({}, options, this._getFilterStrategy($elt).getQueryData($elt, $filter, options));
316
-
317
- const parameters = _.merge(
318
- {},
319
- _.pick(options, ['rows', 'page', 'sortby', 'sortorder', 'sorttype', 'filterquery', 'filtercolumns']),
320
- options.params || {}
321
- );
322
-
323
- const ajaxConfig = {
324
- url: options.url,
325
- data: parameters,
326
- dataType: 'json',
327
- type: options.querytype || 'GET'
328
- };
329
-
330
- // disable pagination to not press multiple on it
331
- disablePagination($elt.paginations);
332
-
333
- /**
334
- * @event dataTable#query.datatable
335
- * @param {Object} ajaxConfig - The config object used to setup the AJAX request
336
- */
337
- $elt.trigger('query.' + ns, [ajaxConfig]);
338
-
339
- // display the loading state
340
- if (options.status) {
341
- $elt.find('.loading').removeClass(hiddenCls);
342
- }
343
-
344
- if (typeof options.requestInterceptor === 'function') {
345
- Promise.resolve(options.requestInterceptor(parameters))
346
- .then(data => {
347
- self._render($elt, data);
348
- })
349
- .catch(error => {
350
- $elt.trigger('error.' + ns, [error]);
351
- self._render($elt, {});
352
- });
353
- return;
354
- }
355
-
356
- $.ajax(ajaxConfig)
357
- .done(function (response) {
358
- if($elt && typeof $elt.data(dataNs) !== 'object'){
359
- return $elt.trigger(`error.${ns}`, [new Error(`Unable to load data attached to the element`)]);
360
- }
361
- self._render($elt, response);
362
- })
363
- .fail(function (response, option, err) {
364
- const requestErr = httpErrorParser.parse(response, option, err);
365
- logger.error(requestErr.message);
366
- requestErr.code = response.status;
367
- enablePagination(this.paginations);
368
- $elt.trigger('error.' + ns, [requestErr]);
369
-
370
- self._render($elt, {});
371
- });
372
- },
373
-
374
- /**
375
- * Renders the table using the provided data set
376
- *
377
- * @param {jQueryElement} $elt - plugin's element
378
- * @param {Object} dataset - the data set to render
379
- * @private
380
- * @fires dataTable#beforeload.datatable
381
- * @fires dataTable#load.datatable
382
- */
383
- _render($elt, dataset = {}) {
384
- const self = this;
385
- let options = _.cloneDeep($elt.data(dataNs));
386
- if(typeof options === 'undefined') {
387
- return $elt.trigger(`error.${ns}`, [new Error(`Unable to load data attached to the element`)]);
388
- }
389
-
390
- const model = [];
391
- let $massActionBtns = $();
392
-
393
- /**
394
- * @event dataTable#beforeload.datatable
395
- * @param {Object} dataset - The data set object used to render the table
396
- */
397
- $elt.trigger('beforeload.' + ns, [_.cloneDeep(dataset)]);
398
-
399
- // overrides column options
400
- _.forEach(options.model, function (field, key) {
401
- if (!options.filter) {
402
- field.filterable = false;
403
- }
404
-
405
- if (_.isUndefined(field.order)) {
406
- field.order = key + 1;
407
- }
408
-
409
- if (field.filterable && typeof field.filterable !== 'object') {
410
- field.filterable = { placeholder: __('Filter') };
411
- }
412
-
413
- if (typeof field.visible === 'undefined') {
414
- model.push(field);
415
- } else if (typeof field.visible === 'function' && field.visible()) {
416
- model.push(field);
417
- } else if (field.visible === true) {
418
- model.push(field);
419
- }
420
- });
421
-
422
- model.sort(function (a, b) {
423
- return a.order - b.order;
424
- });
425
-
426
- if (options.sortby) {
427
- options = this._sortOptions($elt, options.sortby, options.sortorder, options.sorttype);
428
- }
429
-
430
- // process data by model rules
431
- if (_.some(model, 'transform')) {
432
- const transforms = _.where(model, 'transform');
433
-
434
- _.forEach(dataset.data, (row, index) => {
435
- _.forEach(transforms, field => {
436
- if (_.isFunction(field.transform)) {
437
- row[field.id] = field.transform(row[field.id], row, field, index, dataset.data);
438
- }
439
- });
440
- });
441
- }
442
-
443
- options.model = model;
444
-
445
- if (options.atomicUpdate) {
446
- const skipForceUpdate = this._shallowUpdate($elt, dataset, options);
447
-
448
- if (skipForceUpdate) {
449
- updateHeaderStatus(options, $elt, dataset);
450
- loadingBar.stop();
451
- $elt.trigger(`load.${ns}`, [dataset]);
452
- return;
453
- }
454
- }
455
-
456
- // Call the rendering
457
- const $rendering = $(layout({ options: options, dataset: dataset }));
458
-
459
- // the readonly property contains an associative array where keys are the ids of the items (lines)
460
- // the value can be a boolean (true for disable buttons, false to enable)
461
- // it can also bo an array that let you disable/enable the action you want
462
- // readonly = {
463
- // id1 : {'view':true, 'delete':false},
464
- // id2 : true
465
- //}
466
- _.forEach(dataset.readonly, function (values, id) {
467
- if (values === true) {
468
- $('[data-item-identifier="' + id + '"] button', $rendering).addClass('disabled');
469
- } else if (values && typeof values === 'object') {
470
- for (const action in values) {
471
- if (values.hasOwnProperty(action) && values[action] === true) {
472
- $(`[data-item-identifier="${id}"] button.${action}`, $rendering).addClass('disabled');
473
- }
474
- }
475
- }
476
- });
477
-
478
- /**
479
- * Attach handlers on the action buttons
480
- * @param {ActionsObject | Action[]} actions
481
- */
482
- const attachActionListeners = actions => {
483
- // Attach a listener to every action button created
484
- _.forEach(actions, (action, name) => {
485
- if (!_.isFunction(action)) {
486
- name = action.id || name;
487
- action = action.action || function () {};
488
- }
489
-
490
- const css = `.${name}`;
491
-
492
- $rendering.off('click', css).on('click', css, function (e) {
493
- e.preventDefault();
494
-
495
- const $btn = $(this);
496
-
497
- if (!$btn.hasClass('disabled')) {
498
- const identifier = $btn.closest('[data-item-identifier]').data('item-identifier');
499
-
500
- action.apply($btn, [
501
- identifier,
502
- _.first(
503
- _.where(dataset.data, {
504
- id: identifier
505
- })
506
- )
507
- ]);
508
- }
509
- });
510
- });
511
- };
512
-
513
- if (options.actions) {
514
- attachActionListeners(options.actions);
515
- }
516
-
517
- // Attach listeners to model.type = action
518
- if (_.some(options.model, 'type')) {
519
- const types = _.where(options.model, 'type');
520
- _.forEach(types, field => {
521
- if (field.type === 'actions' && field.actions) {
522
- attachActionListeners(field.actions);
523
- }
524
- });
525
- }
526
-
527
- // Attach a listener to every tool button created
528
- _.forEach(options.tools, (action, name) => {
529
- let isMassAction = true;
530
-
531
- if (!_.isFunction(action)) {
532
- name = action.id || name;
533
- isMassAction = action.massAction;
534
- action = action.action || function () {};
535
- }
536
-
537
- const css = `.tool-${name}`;
538
-
539
- if (isMassAction) {
540
- $massActionBtns = $massActionBtns.add($rendering.find(css));
541
- }
542
-
543
- $rendering.off('click', css).on('click', css, function (e) {
544
- e.preventDefault();
545
-
546
- const $btn = $(this);
547
-
548
- if (!$btn.hasClass('disabled')) {
549
- action.apply($btn, [self._selection($elt)]);
550
- }
551
- });
552
- });
553
-
554
- // bind listeners to events
555
- _.forEach(options.listeners, (callback, event) => {
556
- const ev = [event, ns].join('.');
557
- $elt.off(ev).on(ev, callback);
558
- });
559
-
560
- function renderPagination($container, mode) {
561
- return paginationComponent({
562
- mode: mode,
563
- activePage: dataset.page,
564
- totalPages: dataset.total
565
- })
566
- .on('change', function () {
567
- self._setPage($elt, this.getActivePage());
568
- })
569
- .on('prev', function () {
570
- /**
571
- * @event dataTable#backward.dataTable
572
- */
573
- $elt.trigger('backward.' + ns);
574
- })
575
- .on('next', function () {
576
- /**
577
- * @event dataTable#forward.dataTable
578
- */
579
- $elt.trigger('forward.' + ns);
580
- })
581
- .render($container);
582
- }
583
-
584
- $elt.paginations = [];
585
-
586
- if (options.paginationStrategyTop !== 'none') {
587
- // bind pagination component to the datatable
588
- $elt.paginations.push(
589
- renderPagination($('.datatable-pagination-top', $rendering), options.paginationStrategyTop)
590
- );
591
- }
592
- if (options.paginationStrategyBottom !== 'none') {
593
- // bind pagination component to the datatable
594
- $elt.paginations.push(
595
- renderPagination($('.datatable-pagination-bottom', $rendering), options.paginationStrategyBottom)
596
- );
597
- }
598
- disablePagination($elt.paginations);
599
-
600
- // Now $rendering takes the place of $elt...
601
- const $rows = $rendering.find('tbody tr');
602
- const $sortBy = $rendering.find('th [data-sort-by]');
603
- const $sortElement = $rendering.find('[data-sort-by="' + options.sortby + '"]');
604
- const $checkAll = $rendering.find('th.checkboxes input');
605
- const $checkboxes = $rendering.find('td.checkboxes input');
606
-
607
- if (options.rowSelection) {
608
- $('table.datatable', $rendering).addClass('hoverable');
609
- $rendering.on('click', 'tbody td', function (e) {
610
- // exclude from processing columns with actions
611
- if ($(e.target).hasClass('checkboxes') || $(e.target).hasClass('actions')) {
612
- return false;
613
- }
614
-
615
- const currentRow = $(this).parent();
616
-
617
- $rows.removeClass('selected');
618
- currentRow.toggleClass('selected');
619
-
620
- $elt.trigger('selected.' + ns, _.where(dataset.data, { id: currentRow.data('item-identifier') }));
621
- });
622
- }
623
-
624
- $sortBy.on('click keyup', function (e) {
625
- if (e.type === 'keyup' && e.keyCode !== 13) {
626
- return;
627
- }
628
- e.preventDefault();
629
- const column = $(this).data('sort-by');
630
- const type = $(this).data('sort-type');
631
-
632
- self._sort($elt, column, undefined, type);
633
- });
634
-
635
- // Add the filter behavior
636
- if (options.filter) {
637
- self._getFilterStrategy($elt).render($rendering, options);
638
- _.forEach($('.filter', $rendering), function (filter) {
639
- const $filter = $(filter);
640
- const $filterBtn = $('button', $filter);
641
- const $filterInput = $('select, input', $filter);
642
-
643
- if ($filterInput.is('select')) {
644
- $filterInput.on('change', function () {
645
- self._filter($elt, $filter);
646
- });
647
- } else {
648
- // clicking the button trigger the request
649
- $filterBtn.off('click').on('click', function (e) {
650
- e.preventDefault();
651
- self._filter($elt, $filter);
652
- });
653
-
654
- // or press ENTER
655
- $filterInput.off('keypress').on('keypress', function (e) {
656
- if (e.which === 13) {
657
- e.preventDefault();
658
- self._filter($elt, $filter);
659
- }
660
- });
661
- }
662
- });
663
- }
664
-
665
- // check/uncheck all checkboxes
666
- $checkAll.click(function () {
667
- if (this.checked) {
668
- $checkAll.prop('checked', true);
669
- $checkboxes.prop('checked', true);
670
- } else {
671
- $checkAll.prop('checked', false);
672
- $checkboxes.prop('checked', false);
673
- }
674
-
675
- if ($massActionBtns.length) {
676
- $massActionBtns.toggleClass('invisible', !$checkboxes.filter(':checked').length);
677
- }
678
-
679
- /**
680
- * @event dataTable#select.dataTable
681
- */
682
- $elt.trigger('select.' + ns);
683
- });
684
-
685
- // when check/uncheck a box, toggle the check/uncheck all
686
- $checkboxes.click(function () {
687
- const $checked = $checkboxes.filter(':checked');
688
-
689
- if ($checked.length === $checkboxes.length) {
690
- $checkAll.prop('checked', true);
691
- } else {
692
- $checkAll.prop('checked', false);
693
- }
694
-
695
- if ($massActionBtns.length) {
696
- $massActionBtns.toggleClass('invisible', !$checkboxes.filter(':checked').length);
697
- }
698
-
699
- /**
700
- * @event dataTable#select.dataTable
701
- */
702
- $elt.trigger('select.' + ns);
703
- });
704
-
705
- // Remove sorted class from all th
706
- $('th.sorted', $rendering).removeClass('sorted');
707
- // Add the sorted class to the sorted element and the order class
708
- $sortElement.addClass('sorted').addClass('sorted_' + options.sortorder);
709
-
710
- // Update the status
711
-
712
- updateHeaderStatus(options, $rendering, dataset);
713
-
714
- $elt.html($rendering);
715
-
716
- // if the filter is enabled and a value is present, set the focus on the input field
717
- if (options.filter && options.filterquery) {
718
- $rendering.find('[name=filter].focused').focus();
719
- }
720
-
721
- // restore pagination's after data loaded
722
- enablePagination($elt.paginations);
723
-
724
- if (options.pageSizeSelector) {
725
- pageSizeSelector({
726
- renderTo: $('.toolbox-container', $rendering),
727
- defaultSize: options.rows
728
- }).on('change', function (val) {
729
- self._setRows($elt, val);
730
- });
731
- }
732
-
733
- loadingBar.stop();
734
-
735
- /**
736
- * @event dataTable#load.dataTable
737
- * @param {Object} dataset - The data set used to render the table
738
- */
739
- $elt.trigger('load.' + ns, [dataset]);
740
- },
741
-
742
- /**
743
- * Query set new page
744
- *
745
- * @param $elt
746
- * @param page
747
- * @fires dataTable#setpage.datatable
748
- */
749
- _setPage($elt, page) {
750
- const options = $elt.data(dataNs);
751
-
752
- if (options.page !== page) {
753
- // set new page value
754
- options.page = page;
755
-
756
- //rebind options to the elt
757
- $elt.data(dataNs, options);
758
-
759
- /**
760
- * @event dataTable#setpage.dataTable
761
- */
762
- $elt.trigger('setpage.' + ns);
763
-
764
- // Call the query
765
- this._query($elt);
766
- }
767
- },
768
-
769
- /**
770
- * Query filtered list of items
771
- *
772
- * @param {jQueryElement} $elt - plugin's element
773
- * @param {jQueryElement} $filter - the filter input
774
- * @fires dataTable#filter.datatable
775
- * @fires dataTable#sort.datatable
776
- * @private
777
- */
778
- _filter($elt, $filter) {
779
- const options = $elt.data(dataNs);
780
- const filtersData = this._getFilterStrategy($elt).getFiltersData($elt, $filter, options);
781
- options.page = 1;
782
- $elt.data(dataNs, _.assign(options, filtersData));
783
-
784
- /**
785
- * @event dataTable#filter.datatable
786
- * @param {Object} options - The options list
787
- */
788
- $elt.trigger('filter.' + ns, [options]);
789
-
790
- // Call the query
791
- this._query($elt, $filter);
792
- },
793
-
794
- _getFilterStrategy($elt) {
795
- const options = $elt.data(dataNs);
796
-
797
- return filterStrategyFactory(options);
798
- },
799
-
800
- /**
801
- * Query the previous page
802
- *
803
- * Called the jQuery way once registered by the Pluginifier.
804
- * @example $('selector').datatable('sort', 'firstname', false);
805
- *
806
- * @param {jQueryElement} $elt - plugin's element
807
- * @param {String} sortBy - the model id of the col to sort
808
- * @param {Boolean} [asc] - sort direction true for asc of deduced
809
- * @param {String} sortType - type of sorting, numeric or string
810
- * @fires dataTable#sort.datatable
811
- */
812
- _sort($elt, sortBy, asc, sortType) {
813
- const options = this._sortOptions($elt, sortBy, asc, sortType);
814
-
815
- /**
816
- * @event dataTable#sort.datatable
817
- * @param {String} column - The name of the column to sort
818
- * @param {String} direction - The sort direction
819
- * @param {String} type - The type of sorting field, string or numeric
820
- */
821
- $elt.trigger('sort.' + ns, [options.sortby, options.sortorder, options.sorttype]);
822
-
823
- this._query($elt);
824
- },
825
-
826
- /**
827
- * Compared current and next number of row and identifiers order of rows.
828
- *
829
- * @param {dataset} currentState
830
- * @param {dataset} nextState
831
- * @returns {Boolean}
832
- */
833
- _canApplyShallowUpdate(currentState, nextState) {
834
- const isStatesHasData = _.has(currentState, 'data') && _.has(nextState, 'data');
835
-
836
- if (!isStatesHasData) {
837
- return false;
838
- }
839
-
840
- const currentData = currentState.data;
841
- const nextData = nextState.data;
842
-
843
- if (currentData.length !== nextData.length) {
844
- return false;
845
- }
846
-
847
- return _.isEqual(
848
- currentData.map(data => data.id),
849
- nextData.map(data => data.id)
850
- );
851
- },
852
-
853
- /**
854
- * Check possibility of atomic update data in datatable.
855
- *
856
- * @param {jQueryElement} $container Data table container element
857
- * @param {Object} nextState Data to be set to the data table
858
- * @param {Object} options Data table options
859
- * @returns {Boolean} Return true when data in table can be atomically updated
860
- */
861
- _shallowUpdate($container, nextState, options) {
862
- const currentState = $container.data(`${dataNs}state`);
863
-
864
- // Always update data state
865
- $container.data(`${dataNs}state`, nextState);
866
-
867
- if (!this._canApplyShallowUpdate(currentState, nextState)) {
868
- return;
869
- }
870
-
871
- // NOTE: The code above generate the table cell. With updating handlebars to the version > 2.*, please move it to the dedicated template to reuse it in layout.tpl as well
872
- nextState.data.forEach((nextData, index) => {
873
- const $row = $container.find(`tr[data-item-identifier="${nextData.id}"]`);
874
-
875
- options.model.forEach(model => {
876
- const cellId = model.id;
877
-
878
- if (model.type) {
879
- const $actionCell = $row.find(`td.actions.${cellId}`);
880
-
881
- $actionCell.html('');
882
-
883
- model.actions.forEach(action => {
884
- const id = action.id;
885
- const hidden = getPropertyValue('hidden', action, nextData);
886
- const title = getPropertyValue('title', action, nextData);
887
- const disabled = getPropertyValue('disabled', action, nextData);
888
- const icon = getPropertyValue('icon', action, nextData);
889
- const label = getPropertyValue('label', action, nextData);
890
- const $actionButton = $(buttonTpl({ id, icon, label, title, disabled }));
891
-
892
- if (!hidden) {
893
- $actionCell.append('\n').append($actionButton);
894
- }
895
- });
896
- } else {
897
- const nextContent = nextData[cellId];
898
-
899
- $row.find(`td.${cellId}`).html(DOMPurify.sanitize(nextContent));
900
- }
901
- });
902
- });
903
-
904
- return true;
905
- },
906
-
907
- /**
908
- * Set the sort options.
909
- *
910
- * @param {jQueryElement} $elt - plugin's element
911
- * @param {String} sortBy - the model id of the col to sort
912
- * @param {Boolean|String} [asc] - sort direction true for asc of deduced
913
- * @param {String} sortType - sorting type, numeric or string sorting
914
- * @returns {Object} - returns the options
915
- * @private
916
- */
917
- _sortOptions($elt, sortBy, asc, sortType) {
918
- const options = $elt.data(dataNs);
919
-
920
- if (typeof asc !== 'undefined') {
921
- if ('asc' !== asc && 'desc' !== asc) {
922
- asc = !!asc ? 'asc' : 'desc';
923
- }
924
- options.sortorder = asc;
925
- } else if (options.sortorder === 'asc' && options.sortby === sortBy) {
926
- // If I already sort asc this element
927
- options.sortorder = 'desc';
928
- } else {
929
- // If I never sort by this element or
930
- // I sort by this element & the order was desc
931
- options.sortorder = 'asc';
932
- }
933
-
934
- // Change the sorting element anyway.
935
- options.sortby = sortBy;
936
-
937
- // define sorting type
938
- options.sorttype = sortType;
939
-
940
- //rebind options to the elt
941
- $elt.data(dataNs, options);
942
-
943
- return _.cloneDeep(options);
944
- },
945
-
946
- /**
947
- * Gets the selected items. Returns an array of identifiers.
948
- *
949
- * @param {jQueryElement} $elt - plugin's element
950
- * @returns {Array} - Returns an array of identifiers.
951
- */
952
- _selection($elt) {
953
- const $selected = $elt.find('[data-item-identifier]').has('td.checkboxes input:checked');
954
- const selection = [];
955
-
956
- $selected.each(function () {
957
- selection.push($(this).data('item-identifier'));
958
- });
959
-
960
- return selection;
961
- },
962
-
963
- _highlightRows($elt, rowIds) {
964
- $elt.find('[data-item-identifier]').removeClass('highlight');
965
-
966
- rowIds.forEach(rowId => {
967
- this._highlightRow($elt, rowId);
968
- });
969
- },
970
- /**
971
- * Highlight the row with identifier
972
- *
973
- * @param $elt
974
- * @param rowId
975
- * @deprecated Use highlightRows instead
976
- */
977
- _highlightRow($elt, rowId) {
978
- this._addRowClass($elt, rowId, 'highlight');
979
- },
980
-
981
- /**
982
- * Css class add to the row with id
983
- *
984
- * @param $elt
985
- * @param rowId
986
- * @param className
987
- * @private
988
- */
989
- _addRowClass($elt, rowId, className) {
990
- const $row = $elt.find('[data-item-identifier="' + rowId + '"]');
991
-
992
- if (!$row.hasClass(className)) {
993
- $row.addClass(className);
994
- }
995
- },
996
-
997
- /**
998
- * Css class remove from the row with id
999
- *
1000
- * @param $elt
1001
- * @param rowId
1002
- * @param className
1003
- * @private
1004
- */
1005
- _removeRowClass($elt, rowId, className) {
1006
- const $row = $elt.find('[data-item-identifier="' + rowId + '"]');
1007
-
1008
- if ($row.hasClass(className)) {
1009
- $row.removeClass(className);
1010
- }
1011
- },
1012
-
1013
- /**
1014
- * Update amount items per page
1015
- *
1016
- * @param $elt
1017
- * @param rows
1018
- * @fires dataTable#setpage.datatable
1019
- */
1020
- _setRows($elt, rows) {
1021
- const options = $elt.data(dataNs);
1022
-
1023
- if (options.rows !== rows) {
1024
- // set new amount of items per page
1025
- options.rows = rows;
1026
-
1027
- // set page to the first one
1028
- options.page = 1;
1029
-
1030
- //rebind options to the elt
1031
- $elt.data(dataNs, options);
1032
-
1033
- /**
1034
- * @event dataTable#setpage.dataTable
1035
- */
1036
- $elt.trigger('setpage.' + ns);
1037
-
1038
- // Call the query
1039
- this._query($elt);
1040
- }
1041
- }
1042
- };
1043
-
1044
- Pluginifier.register(ns, dataTable, {
1045
- expose: [
1046
- 'refresh',
1047
- 'sort',
1048
- 'filter',
1049
- 'selection',
1050
- 'render',
1051
- 'highlightRow',
1052
- 'highlightRows',
1053
- 'addRowClass',
1054
- 'removeRowClass'
1055
- ]
1056
- });
1
+ /**
2
+ * This program is free software; you can redistribute it and/or
3
+ * modify it under the terms of the GNU General Public License
4
+ * as published by the Free Software Foundation; under version 2
5
+ * of the License (non-upgradable).
6
+ *
7
+ * This program is distributed in the hope that it will be useful,
8
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
9
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
10
+ * GNU General Public License for more details.
11
+ *
12
+ * You should have received a copy of the GNU General Public License
13
+ * along with this program; if not, write to the Free Software
14
+ * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
15
+ *
16
+ * Copyright (c) 2015-2020 (original work) Open Assessment Technologies SA;
17
+ */
18
+
19
+ import $ from 'jquery';
20
+ import _ from 'lodash';
21
+ import __ from 'i18n';
22
+ import Pluginifier from 'core/pluginifier';
23
+ import layout from 'ui/datatable/tpl/layout';
24
+ import buttonTpl from 'ui/datatable/tpl/button';
25
+ import filterStrategyFactory from 'ui/datatable/filterStrategy/filterStrategy';
26
+ import paginationComponent from 'ui/pagination';
27
+ import loadingBar from 'layout/loading-bar';
28
+ import loggerFactory from 'core/logger';
29
+ import httpErrorParser from 'util/httpErrorParser';
30
+ import pageSizeSelector from 'ui/pageSizeSelector';
31
+ import 'ui/datatable/css/datatable.css';
32
+ import DOMPurify from 'lib/dompurify/purify';
33
+
34
+ const ns = 'datatable';
35
+
36
+ const dataNs = 'ui.' + ns;
37
+
38
+ const defaults = {
39
+ atomicUpdate: false,
40
+ start: 0,
41
+ rows: 25,
42
+ page: 1,
43
+ sortby: 'id',
44
+ sortorder: 'asc',
45
+ sorttype: 'string',
46
+ paginationStrategyTop: 'none',
47
+ paginationStrategyBottom: 'simple',
48
+ labels: {
49
+ filter: __('Filter'),
50
+ empty: __('Nothing to list!'),
51
+ available: __('Available'),
52
+ loading: __('Loading'),
53
+ actions: __('Actions')
54
+ },
55
+ pageSizeSelector: false
56
+ };
57
+
58
+ const logger = loggerFactory('ui/datatable');
59
+
60
+ /**
61
+ * The CSS class used to hide an element
62
+ * @type {String}
63
+ */
64
+ const hiddenCls = 'hidden';
65
+
66
+ /**
67
+ * Deactivate pagination's
68
+ * @param {Array} pagination
69
+ */
70
+ const disablePagination = pagination => {
71
+ if (pagination && pagination.length) {
72
+ pagination.forEach(step => {
73
+ step.disable();
74
+ });
75
+ }
76
+ };
77
+
78
+ /**
79
+ * Activate pagination's
80
+ * @param {Array} pagination
81
+ */
82
+ const enablePagination = pagination => {
83
+ if (pagination && pagination.length) {
84
+ pagination.forEach(step => {
85
+ step.enable();
86
+ });
87
+ }
88
+ };
89
+
90
+ /**
91
+ * Helper for reading actions value in context
92
+ *
93
+ * @param {String} property
94
+ * @param {Object} action
95
+ * @param {Object} context
96
+ */
97
+ const getPropertyValue = (property, action, context) => {
98
+ const value = action[property];
99
+
100
+ return _.isFunction(value) ? value.apply(context) : value;
101
+ };
102
+
103
+ /**
104
+ * Update the data table status in the header
105
+ *
106
+ * @param {Object} options
107
+ * @param {jQueryElement} $container
108
+ * @param {Object} dataset
109
+ */
110
+ const updateHeaderStatus = (options, $container, dataset) => {
111
+ if (!options.status) {
112
+ return;
113
+ }
114
+
115
+ const $statusEmpty = $container.find('.empty-list');
116
+ const $statusAvailable = $container.find('.available-list');
117
+ const $statusCount = $statusAvailable.find('.count');
118
+
119
+ $container.find('.loading').addClass(hiddenCls);
120
+
121
+ // when the status is enabled, the response must contain the total amount of records
122
+ const amount = dataset.amount || dataset.length;
123
+
124
+ if (amount) {
125
+ $statusCount.text(amount);
126
+ $statusAvailable.removeClass(hiddenCls);
127
+ $statusEmpty.addClass(hiddenCls);
128
+ } else {
129
+ $statusEmpty.removeClass(hiddenCls);
130
+ $statusAvailable.addClass(hiddenCls);
131
+ }
132
+ };
133
+
134
+ /**
135
+ * The dataTable component makes you able to browse items and bind specific
136
+ * actions to undertake for edition and removal of them.
137
+ *
138
+ * Parameters that will be send to backend by component:
139
+ *
140
+ * Pagination
141
+ * @param {Number} rows - count of rows, that should be returned from backend, in other words limit.
142
+ * @param {Number} page - number of page, that should be requested.
143
+ *
144
+ * Sorting
145
+ * @param {String} sortby - name of column
146
+ * @param {String} sortorder - order of sorting, can be 'asc' or 'desc' for ascending sorting and descending sorting respectively.
147
+ * @param {String} sorttype - type of sorting, can be 'string' or 'numeric' for proper sorting numeric and string values.
148
+ *
149
+ * Filtering
150
+ * @param {String} filterstrategy - filtering strategy. Default is single (see ui/datatable/filterStrategy/single.js).
151
+ * @param {String} filterquery - query string for filtering of rows.
152
+ * @param {String[]} filtercolumns[] - array of columns, in which will be implemented search during filtering process.
153
+ * For column filter it will be only one item with column name, but component has ability define list of columns for default filter (in top toolbar).
154
+ * Backend should correctly receive this list of columns and do search in accordance with this parameters.
155
+ * By default, columns are not defined, so this parameter not will be sent. If filtercolumns[] not exists, backend should search by all columns.
156
+ *
157
+ * @example of query (GET): rows=25&page=1&sortby=login&sortorder=asc&filterquery=loginame&filtercolumns[]=login
158
+ *
159
+ * @exports ui/datatable
160
+ */
161
+ const dataTable = {
162
+ /**
163
+ * Used for generating action button action button
164
+ * @typedef Action
165
+ * @type {Object}
166
+ * @property {String} id ID is added to the button class
167
+ * @property {String} [title] Button title
168
+ * @property {Boolean} [disabled] When present, button should be disabled
169
+ * @property {String} [icon] Generate button icon
170
+ * @property {Boolean} [hidden] When present, button is hidden
171
+ * @property {Function} [action] Handler on button click
172
+ */
173
+
174
+ /**
175
+ * Used for generating action button from Object
176
+ * @deprecated
177
+ * @typedef {{
178
+ * [key: Action.id & Action.icon & Action.title ]: Action.action,
179
+ * }} ActionsObject
180
+ *
181
+ * @example
182
+ * {
183
+ * actions: {
184
+ * edit: editUser,
185
+ * remove: removeUser,
186
+ * }
187
+ * }
188
+ *
189
+ * ! IMPORTANT USE INSTEAD:
190
+ * {
191
+ * actions: [
192
+ * {
193
+ * id: "edit",
194
+ * title: __("Edit"),
195
+ * icon: "edit",
196
+ * action: editUser
197
+ * },
198
+ * {
199
+ * id: "edit",
200
+ * title: __("Edit"),
201
+ * icon: "edit",
202
+ * action: editUser
203
+ * }
204
+ * ]
205
+ * }
206
+ */
207
+
208
+ /**
209
+ * Initialize the plugin.
210
+ *
211
+ * Called the jQuery way once registered by the Pluginifier.
212
+ * @example $('selector').datatable([], {});
213
+ *
214
+ * @constructor
215
+ * @param {Object} options - the plugin options.
216
+ * @param {String} options.url - the URL of the service used to retrieve the resources.
217
+ * @param {Object[]} options.model - the model definition.
218
+ * @param {ActionsObject | Action[]} options.actions - Generates action buttons
219
+ * @param {Function} options.listeners.xxx - the callback function for event xxx, parameters depends to event trigger call.
220
+ * @param {Boolean} options.selectable - enables the selection of rows using checkboxes.
221
+ * @param {Boolean} options.rowSelection - enables the selection of rows by clicking on them.
222
+ * @param {Object} options.tools - a list of tool buttons to display above the table.
223
+ * @param {Object|Boolean} options.status - allow to display a status bar.
224
+ * @param {Object|Boolean} options.filter - allow to display a filter bar.
225
+ * @param {String} options.filterStrategy - 'multiple' | 'single' -- filtered by all filters together or filtering allowed only by one field at the moment (default 'single'),
226
+ * @param {String} options.filterSelector - css selector for search of filter inputs, by defaul 'select, input'
227
+ * @param {String} options.filterTransform - transform filter value before send to server.
228
+ * @param {String[]} options.filter.columns - a list of columns that will be used for default filter. Can be overridden by column filter.
229
+ * @param {String} options.filterquery - a query string for filtering, using only in runtime.
230
+ * @param {String[]} options.filtercolumns - a list of columns, in that should be done search, using only in runtime.
231
+ * @param {String} options.paginationStrategyTop - 'none' | 'pages' | 'simple' -- 'none' by default (next/prev), 'pages' show pages and extended control for pagination
232
+ * @param {String} options.paginationStrategyBottom - 'none' | 'pages' | 'simple' -- 'simple' by default (next/prev), 'pages' show pages and extended control for pagination
233
+ * @param {Object} options.labels - list of labels in datatable interface, that can be overridden by incoming options
234
+ * @param {String} options.emptyText - text that will be shown when no data found for showing in the grid.
235
+ * @param {Boolean} options.pageSizeSelector - flag that indicates if control for changing page size should be displayed
236
+ * @param {Boolean} options.atomicUpdate - allowed to keep the datatable state to be able on "render" event, compare with new state and atomically update the table cells.
237
+ * @param {Function} options.requestInterceptor - Intercept sending AJAX request. The call of function must returns promise with provided data.
238
+ * @param {Object} [data] - inject predefined data to avoid the first query.
239
+ * @fires dataTable#create.datatable
240
+ * @returns {jQueryElement} for chaining
241
+ */
242
+ init(options, data) {
243
+ options = _.defaults(options, defaults);
244
+
245
+ return this.each(function () {
246
+ const $elt = $(this);
247
+ const currentOptions = $elt.data(dataNs);
248
+
249
+ if (options.atomicUpdate && data) {
250
+ $elt.data(`${dataNs}state`, data.data);
251
+ }
252
+ // implement encapsulated pages for the datatable
253
+ $elt.paginations = [];
254
+
255
+ if (!currentOptions) {
256
+ //add data to the element
257
+ $elt.data(dataNs, options);
258
+
259
+ $elt.one('load.' + ns, function () {
260
+ /**
261
+ * @event dataTable#create.datatable
262
+ */
263
+ $elt.trigger('create.' + ns);
264
+ });
265
+
266
+ if (data) {
267
+ dataTable._render($elt, data);
268
+ } else {
269
+ dataTable._query($elt);
270
+ }
271
+ } else {
272
+ // update existing options
273
+ $elt.data(dataNs, _.merge(currentOptions, options));
274
+
275
+ dataTable._refresh($elt, data);
276
+ }
277
+ });
278
+ },
279
+
280
+ /**
281
+ * Refresh the data table using current options
282
+ *
283
+ * Called the jQuery way once registered by the Pluginifier.
284
+ * @example $('selector').datatable('refresh');
285
+ *
286
+ * @param {jQueryElement} $elt - plugin's element
287
+ * @param {Object} [data] - Data to render immediately, prevents the query to be made.
288
+ */
289
+ _refresh($elt, data) {
290
+ if (data) {
291
+ this._render($elt, data);
292
+ } else {
293
+ this._query($elt);
294
+ }
295
+ },
296
+
297
+ /**
298
+ * Query the server for data and load the table.
299
+ *
300
+ * @private
301
+ * @param {jQueryElement} $elt - plugin's element
302
+ * @param $filter
303
+ * @fires dataTable#query.datatable
304
+ */
305
+ _query($elt, $filter) {
306
+ const self = this;
307
+ let options = $elt.data(dataNs);
308
+
309
+ loadingBar.start();
310
+
311
+ if (!$filter) {
312
+ $filter = $('.filter', $elt);
313
+ }
314
+
315
+ options = _.assign({}, options, this._getFilterStrategy($elt).getQueryData($elt, $filter, options));
316
+
317
+ const parameters = _.merge(
318
+ {},
319
+ _.pick(options, ['rows', 'page', 'sortby', 'sortorder', 'sorttype', 'filterquery', 'filtercolumns']),
320
+ options.params || {}
321
+ );
322
+
323
+ const ajaxConfig = {
324
+ url: options.url,
325
+ data: parameters,
326
+ dataType: 'json',
327
+ type: options.querytype || 'GET'
328
+ };
329
+
330
+ // disable pagination to not press multiple on it
331
+ disablePagination($elt.paginations);
332
+
333
+ /**
334
+ * @event dataTable#query.datatable
335
+ * @param {Object} ajaxConfig - The config object used to setup the AJAX request
336
+ */
337
+ $elt.trigger('query.' + ns, [ajaxConfig]);
338
+
339
+ // display the loading state
340
+ if (options.status) {
341
+ $elt.find('.loading').removeClass(hiddenCls);
342
+ }
343
+
344
+ if (typeof options.requestInterceptor === 'function') {
345
+ Promise.resolve(options.requestInterceptor(parameters))
346
+ .then(data => {
347
+ self._render($elt, data);
348
+ })
349
+ .catch(error => {
350
+ $elt.trigger('error.' + ns, [error]);
351
+ self._render($elt, {});
352
+ });
353
+ return;
354
+ }
355
+
356
+ $.ajax(ajaxConfig)
357
+ .done(function (response) {
358
+ if($elt && typeof $elt.data(dataNs) !== 'object'){
359
+ return $elt.trigger(`error.${ns}`, [new Error(`Unable to load data attached to the element`)]);
360
+ }
361
+ self._render($elt, response);
362
+ })
363
+ .fail(function (response, option, err) {
364
+ const requestErr = httpErrorParser.parse(response, option, err);
365
+ logger.error(requestErr.message);
366
+ requestErr.code = response.status;
367
+ enablePagination(this.paginations);
368
+ $elt.trigger('error.' + ns, [requestErr]);
369
+
370
+ self._render($elt, {});
371
+ });
372
+ },
373
+
374
+ /**
375
+ * Renders the table using the provided data set
376
+ *
377
+ * @param {jQueryElement} $elt - plugin's element
378
+ * @param {Object} dataset - the data set to render
379
+ * @private
380
+ * @fires dataTable#beforeload.datatable
381
+ * @fires dataTable#load.datatable
382
+ */
383
+ _render($elt, dataset = {}) {
384
+ const self = this;
385
+ let options = _.cloneDeep($elt.data(dataNs));
386
+ if(typeof options === 'undefined') {
387
+ return $elt.trigger(`error.${ns}`, [new Error(`Unable to load data attached to the element`)]);
388
+ }
389
+
390
+ const model = [];
391
+ let $massActionBtns = $();
392
+
393
+ /**
394
+ * @event dataTable#beforeload.datatable
395
+ * @param {Object} dataset - The data set object used to render the table
396
+ */
397
+ $elt.trigger('beforeload.' + ns, [_.cloneDeep(dataset)]);
398
+
399
+ // overrides column options
400
+ _.forEach(options.model, function (field, key) {
401
+ if (!options.filter) {
402
+ field.filterable = false;
403
+ }
404
+
405
+ if (_.isUndefined(field.order)) {
406
+ field.order = key + 1;
407
+ }
408
+
409
+ if (field.filterable && typeof field.filterable !== 'object') {
410
+ field.filterable = { placeholder: __('Filter') };
411
+ }
412
+
413
+ if (typeof field.visible === 'undefined') {
414
+ model.push(field);
415
+ } else if (typeof field.visible === 'function' && field.visible()) {
416
+ model.push(field);
417
+ } else if (field.visible === true) {
418
+ model.push(field);
419
+ }
420
+ });
421
+
422
+ model.sort(function (a, b) {
423
+ return a.order - b.order;
424
+ });
425
+
426
+ if (options.sortby) {
427
+ options = this._sortOptions($elt, options.sortby, options.sortorder, options.sorttype);
428
+ }
429
+
430
+ // process data by model rules
431
+ if (_.some(model, 'transform')) {
432
+ const transforms = _.where(model, 'transform');
433
+
434
+ _.forEach(dataset.data, (row, index) => {
435
+ _.forEach(transforms, field => {
436
+ if (_.isFunction(field.transform)) {
437
+ row[field.id] = field.transform(row[field.id], row, field, index, dataset.data);
438
+ }
439
+ });
440
+ });
441
+ }
442
+
443
+ options.model = model;
444
+
445
+ if (options.atomicUpdate) {
446
+ const skipForceUpdate = this._shallowUpdate($elt, dataset, options);
447
+
448
+ if (skipForceUpdate) {
449
+ updateHeaderStatus(options, $elt, dataset);
450
+ loadingBar.stop();
451
+ $elt.trigger(`load.${ns}`, [dataset]);
452
+ return;
453
+ }
454
+ }
455
+
456
+ // Call the rendering
457
+ const $rendering = $(layout({ options: options, dataset: dataset }));
458
+
459
+ // the readonly property contains an associative array where keys are the ids of the items (lines)
460
+ // the value can be a boolean (true for disable buttons, false to enable)
461
+ // it can also bo an array that let you disable/enable the action you want
462
+ // readonly = {
463
+ // id1 : {'view':true, 'delete':false},
464
+ // id2 : true
465
+ //}
466
+ _.forEach(dataset.readonly, function (values, id) {
467
+ if (values === true) {
468
+ $('[data-item-identifier="' + id + '"] button', $rendering).addClass('disabled');
469
+ } else if (values && typeof values === 'object') {
470
+ for (const action in values) {
471
+ if (values.hasOwnProperty(action) && values[action] === true) {
472
+ $(`[data-item-identifier="${id}"] button.${action}`, $rendering).addClass('disabled');
473
+ }
474
+ }
475
+ }
476
+ });
477
+
478
+ /**
479
+ * Attach handlers on the action buttons
480
+ * @param {ActionsObject | Action[]} actions
481
+ */
482
+ const attachActionListeners = actions => {
483
+ // Attach a listener to every action button created
484
+ _.forEach(actions, (action, name) => {
485
+ if (!_.isFunction(action)) {
486
+ name = action.id || name;
487
+ action = action.action || function () {};
488
+ }
489
+
490
+ const css = `.${name}`;
491
+
492
+ $rendering.off('click', css).on('click', css, function (e) {
493
+ e.preventDefault();
494
+
495
+ const $btn = $(this);
496
+
497
+ if (!$btn.hasClass('disabled')) {
498
+ const identifier = $btn.closest('[data-item-identifier]').data('item-identifier');
499
+
500
+ action.apply($btn, [
501
+ identifier,
502
+ _.first(
503
+ _.where(dataset.data, {
504
+ id: identifier
505
+ })
506
+ )
507
+ ]);
508
+ }
509
+ });
510
+ });
511
+ };
512
+
513
+ if (options.actions) {
514
+ attachActionListeners(options.actions);
515
+ }
516
+
517
+ // Attach listeners to model.type = action
518
+ if (_.some(options.model, 'type')) {
519
+ const types = _.where(options.model, 'type');
520
+ _.forEach(types, field => {
521
+ if (field.type === 'actions' && field.actions) {
522
+ attachActionListeners(field.actions);
523
+ }
524
+ });
525
+ }
526
+
527
+ // Attach a listener to every tool button created
528
+ _.forEach(options.tools, (action, name) => {
529
+ let isMassAction = true;
530
+
531
+ if (!_.isFunction(action)) {
532
+ name = action.id || name;
533
+ isMassAction = action.massAction;
534
+ action = action.action || function () {};
535
+ }
536
+
537
+ const css = `.tool-${name}`;
538
+
539
+ if (isMassAction) {
540
+ $massActionBtns = $massActionBtns.add($rendering.find(css));
541
+ }
542
+
543
+ $rendering.off('click', css).on('click', css, function (e) {
544
+ e.preventDefault();
545
+
546
+ const $btn = $(this);
547
+
548
+ if (!$btn.hasClass('disabled')) {
549
+ action.apply($btn, [self._selection($elt)]);
550
+ }
551
+ });
552
+ });
553
+
554
+ // bind listeners to events
555
+ _.forEach(options.listeners, (callback, event) => {
556
+ const ev = [event, ns].join('.');
557
+ $elt.off(ev).on(ev, callback);
558
+ });
559
+
560
+ function renderPagination($container, mode) {
561
+ return paginationComponent({
562
+ mode: mode,
563
+ activePage: dataset.page,
564
+ totalPages: dataset.total
565
+ })
566
+ .on('change', function () {
567
+ self._setPage($elt, this.getActivePage());
568
+ })
569
+ .on('prev', function () {
570
+ /**
571
+ * @event dataTable#backward.dataTable
572
+ */
573
+ $elt.trigger('backward.' + ns);
574
+ })
575
+ .on('next', function () {
576
+ /**
577
+ * @event dataTable#forward.dataTable
578
+ */
579
+ $elt.trigger('forward.' + ns);
580
+ })
581
+ .render($container);
582
+ }
583
+
584
+ $elt.paginations = [];
585
+
586
+ if (options.paginationStrategyTop !== 'none') {
587
+ // bind pagination component to the datatable
588
+ $elt.paginations.push(
589
+ renderPagination($('.datatable-pagination-top', $rendering), options.paginationStrategyTop)
590
+ );
591
+ }
592
+ if (options.paginationStrategyBottom !== 'none') {
593
+ // bind pagination component to the datatable
594
+ $elt.paginations.push(
595
+ renderPagination($('.datatable-pagination-bottom', $rendering), options.paginationStrategyBottom)
596
+ );
597
+ }
598
+ disablePagination($elt.paginations);
599
+
600
+ // Now $rendering takes the place of $elt...
601
+ const $rows = $rendering.find('tbody tr');
602
+ const $sortBy = $rendering.find('th [data-sort-by]');
603
+ const $sortElement = $rendering.find('[data-sort-by="' + options.sortby + '"]');
604
+ const $checkAll = $rendering.find('th.checkboxes input');
605
+ const $checkboxes = $rendering.find('td.checkboxes input');
606
+
607
+ if (options.rowSelection) {
608
+ $('table.datatable', $rendering).addClass('hoverable');
609
+ $rendering.on('click', 'tbody td', function (e) {
610
+ // exclude from processing columns with actions
611
+ if ($(e.target).hasClass('checkboxes') || $(e.target).hasClass('actions')) {
612
+ return false;
613
+ }
614
+
615
+ const currentRow = $(this).parent();
616
+
617
+ $rows.removeClass('selected');
618
+ currentRow.toggleClass('selected');
619
+
620
+ $elt.trigger('selected.' + ns, _.where(dataset.data, { id: currentRow.data('item-identifier') }));
621
+ });
622
+ }
623
+
624
+ $sortBy.on('click keyup', function (e) {
625
+ if (e.type === 'keyup' && e.keyCode !== 13) {
626
+ return;
627
+ }
628
+ e.preventDefault();
629
+ const column = $(this).data('sort-by');
630
+ const type = $(this).data('sort-type');
631
+
632
+ self._sort($elt, column, undefined, type);
633
+ });
634
+
635
+ // Add the filter behavior
636
+ if (options.filter) {
637
+ self._getFilterStrategy($elt).render($rendering, options);
638
+ _.forEach($('.filter', $rendering), function (filter) {
639
+ const $filter = $(filter);
640
+ const $filterBtn = $('button', $filter);
641
+ const $filterInput = $('select, input', $filter);
642
+
643
+ if ($filterInput.is('select')) {
644
+ $filterInput.on('change', function () {
645
+ self._filter($elt, $filter);
646
+ });
647
+ } else {
648
+ // clicking the button trigger the request
649
+ $filterBtn.off('click').on('click', function (e) {
650
+ e.preventDefault();
651
+ self._filter($elt, $filter);
652
+ });
653
+
654
+ // or press ENTER
655
+ $filterInput.off('keypress').on('keypress', function (e) {
656
+ if (e.which === 13) {
657
+ e.preventDefault();
658
+ self._filter($elt, $filter);
659
+ }
660
+ });
661
+ }
662
+ });
663
+ }
664
+
665
+ // check/uncheck all checkboxes
666
+ $checkAll.click(function () {
667
+ if (this.checked) {
668
+ $checkAll.prop('checked', true);
669
+ $checkboxes.prop('checked', true);
670
+ } else {
671
+ $checkAll.prop('checked', false);
672
+ $checkboxes.prop('checked', false);
673
+ }
674
+
675
+ if ($massActionBtns.length) {
676
+ $massActionBtns.toggleClass('invisible', !$checkboxes.filter(':checked').length);
677
+ }
678
+
679
+ /**
680
+ * @event dataTable#select.dataTable
681
+ */
682
+ $elt.trigger('select.' + ns);
683
+ });
684
+
685
+ // when check/uncheck a box, toggle the check/uncheck all
686
+ $checkboxes.click(function () {
687
+ const $checked = $checkboxes.filter(':checked');
688
+
689
+ if ($checked.length === $checkboxes.length) {
690
+ $checkAll.prop('checked', true);
691
+ } else {
692
+ $checkAll.prop('checked', false);
693
+ }
694
+
695
+ if ($massActionBtns.length) {
696
+ $massActionBtns.toggleClass('invisible', !$checkboxes.filter(':checked').length);
697
+ }
698
+
699
+ /**
700
+ * @event dataTable#select.dataTable
701
+ */
702
+ $elt.trigger('select.' + ns);
703
+ });
704
+
705
+ // Remove sorted class from all th
706
+ $('th.sorted', $rendering).removeClass('sorted');
707
+ // Add the sorted class to the sorted element and the order class
708
+ $sortElement.addClass('sorted').addClass('sorted_' + options.sortorder);
709
+
710
+ // Update the status
711
+
712
+ updateHeaderStatus(options, $rendering, dataset);
713
+
714
+ $elt.html($rendering);
715
+
716
+ // if the filter is enabled and a value is present, set the focus on the input field
717
+ if (options.filter && options.filterquery) {
718
+ $rendering.find('[name=filter].focused').focus();
719
+ }
720
+
721
+ // restore pagination's after data loaded
722
+ enablePagination($elt.paginations);
723
+
724
+ if (options.pageSizeSelector) {
725
+ pageSizeSelector({
726
+ renderTo: $('.toolbox-container', $rendering),
727
+ defaultSize: options.rows
728
+ }).on('change', function (val) {
729
+ self._setRows($elt, val);
730
+ });
731
+ }
732
+
733
+ loadingBar.stop();
734
+
735
+ /**
736
+ * @event dataTable#load.dataTable
737
+ * @param {Object} dataset - The data set used to render the table
738
+ */
739
+ $elt.trigger('load.' + ns, [dataset]);
740
+ },
741
+
742
+ /**
743
+ * Query set new page
744
+ *
745
+ * @param $elt
746
+ * @param page
747
+ * @fires dataTable#setpage.datatable
748
+ */
749
+ _setPage($elt, page) {
750
+ const options = $elt.data(dataNs);
751
+
752
+ if (options.page !== page) {
753
+ // set new page value
754
+ options.page = page;
755
+
756
+ //rebind options to the elt
757
+ $elt.data(dataNs, options);
758
+
759
+ /**
760
+ * @event dataTable#setpage.dataTable
761
+ */
762
+ $elt.trigger('setpage.' + ns);
763
+
764
+ // Call the query
765
+ this._query($elt);
766
+ }
767
+ },
768
+
769
+ /**
770
+ * Query filtered list of items
771
+ *
772
+ * @param {jQueryElement} $elt - plugin's element
773
+ * @param {jQueryElement} $filter - the filter input
774
+ * @fires dataTable#filter.datatable
775
+ * @fires dataTable#sort.datatable
776
+ * @private
777
+ */
778
+ _filter($elt, $filter) {
779
+ const options = $elt.data(dataNs);
780
+ const filtersData = this._getFilterStrategy($elt).getFiltersData($elt, $filter, options);
781
+ options.page = 1;
782
+ $elt.data(dataNs, _.assign(options, filtersData));
783
+
784
+ /**
785
+ * @event dataTable#filter.datatable
786
+ * @param {Object} options - The options list
787
+ */
788
+ $elt.trigger('filter.' + ns, [options]);
789
+
790
+ // Call the query
791
+ this._query($elt, $filter);
792
+ },
793
+
794
+ _getFilterStrategy($elt) {
795
+ const options = $elt.data(dataNs);
796
+
797
+ return filterStrategyFactory(options);
798
+ },
799
+
800
+ /**
801
+ * Query the previous page
802
+ *
803
+ * Called the jQuery way once registered by the Pluginifier.
804
+ * @example $('selector').datatable('sort', 'firstname', false);
805
+ *
806
+ * @param {jQueryElement} $elt - plugin's element
807
+ * @param {String} sortBy - the model id of the col to sort
808
+ * @param {Boolean} [asc] - sort direction true for asc of deduced
809
+ * @param {String} sortType - type of sorting, numeric or string
810
+ * @fires dataTable#sort.datatable
811
+ */
812
+ _sort($elt, sortBy, asc, sortType) {
813
+ const options = this._sortOptions($elt, sortBy, asc, sortType);
814
+
815
+ /**
816
+ * @event dataTable#sort.datatable
817
+ * @param {String} column - The name of the column to sort
818
+ * @param {String} direction - The sort direction
819
+ * @param {String} type - The type of sorting field, string or numeric
820
+ */
821
+ $elt.trigger('sort.' + ns, [options.sortby, options.sortorder, options.sorttype]);
822
+
823
+ this._query($elt);
824
+ },
825
+
826
+ /**
827
+ * Compared current and next number of row and identifiers order of rows.
828
+ *
829
+ * @param {dataset} currentState
830
+ * @param {dataset} nextState
831
+ * @returns {Boolean}
832
+ */
833
+ _canApplyShallowUpdate(currentState, nextState) {
834
+ const isStatesHasData = _.has(currentState, 'data') && _.has(nextState, 'data');
835
+
836
+ if (!isStatesHasData) {
837
+ return false;
838
+ }
839
+
840
+ const currentData = currentState.data;
841
+ const nextData = nextState.data;
842
+
843
+ if (currentData.length !== nextData.length) {
844
+ return false;
845
+ }
846
+
847
+ return _.isEqual(
848
+ currentData.map(data => data.id),
849
+ nextData.map(data => data.id)
850
+ );
851
+ },
852
+
853
+ /**
854
+ * Check possibility of atomic update data in datatable.
855
+ *
856
+ * @param {jQueryElement} $container Data table container element
857
+ * @param {Object} nextState Data to be set to the data table
858
+ * @param {Object} options Data table options
859
+ * @returns {Boolean} Return true when data in table can be atomically updated
860
+ */
861
+ _shallowUpdate($container, nextState, options) {
862
+ const currentState = $container.data(`${dataNs}state`);
863
+
864
+ // Always update data state
865
+ $container.data(`${dataNs}state`, nextState);
866
+
867
+ if (!this._canApplyShallowUpdate(currentState, nextState)) {
868
+ return;
869
+ }
870
+
871
+ // NOTE: The code above generate the table cell. With updating handlebars to the version > 2.*, please move it to the dedicated template to reuse it in layout.tpl as well
872
+ nextState.data.forEach((nextData, index) => {
873
+ const $row = $container.find(`tr[data-item-identifier="${nextData.id}"]`);
874
+
875
+ options.model.forEach(model => {
876
+ const cellId = model.id;
877
+
878
+ if (model.type) {
879
+ const $actionCell = $row.find(`td.actions.${cellId}`);
880
+
881
+ $actionCell.html('');
882
+
883
+ model.actions.forEach(action => {
884
+ const id = action.id;
885
+ const hidden = getPropertyValue('hidden', action, nextData);
886
+ const title = getPropertyValue('title', action, nextData);
887
+ const disabled = getPropertyValue('disabled', action, nextData);
888
+ const icon = getPropertyValue('icon', action, nextData);
889
+ const label = getPropertyValue('label', action, nextData);
890
+ const $actionButton = $(buttonTpl({ id, icon, label, title, disabled }));
891
+
892
+ if (!hidden) {
893
+ $actionCell.append('\n').append($actionButton);
894
+ }
895
+ });
896
+ } else {
897
+ const nextContent = nextData[cellId];
898
+
899
+ $row.find(`td.${cellId}`).html(DOMPurify.sanitize(nextContent));
900
+ }
901
+ });
902
+ });
903
+
904
+ return true;
905
+ },
906
+
907
+ /**
908
+ * Set the sort options.
909
+ *
910
+ * @param {jQueryElement} $elt - plugin's element
911
+ * @param {String} sortBy - the model id of the col to sort
912
+ * @param {Boolean|String} [asc] - sort direction true for asc of deduced
913
+ * @param {String} sortType - sorting type, numeric or string sorting
914
+ * @returns {Object} - returns the options
915
+ * @private
916
+ */
917
+ _sortOptions($elt, sortBy, asc, sortType) {
918
+ const options = $elt.data(dataNs);
919
+
920
+ if (typeof asc !== 'undefined') {
921
+ if ('asc' !== asc && 'desc' !== asc) {
922
+ asc = !!asc ? 'asc' : 'desc';
923
+ }
924
+ options.sortorder = asc;
925
+ } else if (options.sortorder === 'asc' && options.sortby === sortBy) {
926
+ // If I already sort asc this element
927
+ options.sortorder = 'desc';
928
+ } else {
929
+ // If I never sort by this element or
930
+ // I sort by this element & the order was desc
931
+ options.sortorder = 'asc';
932
+ }
933
+
934
+ // Change the sorting element anyway.
935
+ options.sortby = sortBy;
936
+
937
+ // define sorting type
938
+ options.sorttype = sortType;
939
+
940
+ //rebind options to the elt
941
+ $elt.data(dataNs, options);
942
+
943
+ return _.cloneDeep(options);
944
+ },
945
+
946
+ /**
947
+ * Gets the selected items. Returns an array of identifiers.
948
+ *
949
+ * @param {jQueryElement} $elt - plugin's element
950
+ * @returns {Array} - Returns an array of identifiers.
951
+ */
952
+ _selection($elt) {
953
+ const $selected = $elt.find('[data-item-identifier]').has('td.checkboxes input:checked');
954
+ const selection = [];
955
+
956
+ $selected.each(function () {
957
+ selection.push($(this).data('item-identifier'));
958
+ });
959
+
960
+ return selection;
961
+ },
962
+
963
+ _highlightRows($elt, rowIds) {
964
+ $elt.find('[data-item-identifier]').removeClass('highlight');
965
+
966
+ rowIds.forEach(rowId => {
967
+ this._highlightRow($elt, rowId);
968
+ });
969
+ },
970
+ /**
971
+ * Highlight the row with identifier
972
+ *
973
+ * @param $elt
974
+ * @param rowId
975
+ * @deprecated Use highlightRows instead
976
+ */
977
+ _highlightRow($elt, rowId) {
978
+ this._addRowClass($elt, rowId, 'highlight');
979
+ },
980
+
981
+ /**
982
+ * Css class add to the row with id
983
+ *
984
+ * @param $elt
985
+ * @param rowId
986
+ * @param className
987
+ * @private
988
+ */
989
+ _addRowClass($elt, rowId, className) {
990
+ const $row = $elt.find('[data-item-identifier="' + rowId + '"]');
991
+
992
+ if (!$row.hasClass(className)) {
993
+ $row.addClass(className);
994
+ }
995
+ },
996
+
997
+ /**
998
+ * Css class remove from the row with id
999
+ *
1000
+ * @param $elt
1001
+ * @param rowId
1002
+ * @param className
1003
+ * @private
1004
+ */
1005
+ _removeRowClass($elt, rowId, className) {
1006
+ const $row = $elt.find('[data-item-identifier="' + rowId + '"]');
1007
+
1008
+ if ($row.hasClass(className)) {
1009
+ $row.removeClass(className);
1010
+ }
1011
+ },
1012
+
1013
+ /**
1014
+ * Update amount items per page
1015
+ *
1016
+ * @param $elt
1017
+ * @param rows
1018
+ * @fires dataTable#setpage.datatable
1019
+ */
1020
+ _setRows($elt, rows) {
1021
+ const options = $elt.data(dataNs);
1022
+
1023
+ if (options.rows !== rows) {
1024
+ // set new amount of items per page
1025
+ options.rows = rows;
1026
+
1027
+ // set page to the first one
1028
+ options.page = 1;
1029
+
1030
+ //rebind options to the elt
1031
+ $elt.data(dataNs, options);
1032
+
1033
+ /**
1034
+ * @event dataTable#setpage.dataTable
1035
+ */
1036
+ $elt.trigger('setpage.' + ns);
1037
+
1038
+ // Call the query
1039
+ this._query($elt);
1040
+ }
1041
+ }
1042
+ };
1043
+
1044
+ Pluginifier.register(ns, dataTable, {
1045
+ expose: [
1046
+ 'refresh',
1047
+ 'sort',
1048
+ 'filter',
1049
+ 'selection',
1050
+ 'render',
1051
+ 'highlightRow',
1052
+ 'highlightRows',
1053
+ 'addRowClass',
1054
+ 'removeRowClass'
1055
+ ]
1056
+ });