@oat-sa/tao-core-ui 1.64.1 → 1.64.2
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.
- package/LICENSE +339 -339
- package/README.md +22 -22
- package/dist/actionbar.js +92 -92
- package/dist/adder.js +109 -109
- package/dist/animable/absorbable/absorbable.js +29 -29
- package/dist/animable/absorbable/css/absorb.css +7 -7
- package/dist/animable/absorbable/css/absorb.css.map +1 -1
- package/dist/animable/pulsable/css/pulse.css +7 -7
- package/dist/animable/pulsable/css/pulse.css.map +1 -1
- package/dist/animable/pulsable/pulsable.js +23 -23
- package/dist/areaBroker.js +51 -51
- package/dist/autocomplete/css/autocomplete.css +7 -7
- package/dist/autocomplete/css/autocomplete.css.map +1 -1
- package/dist/autocomplete.js +384 -384
- package/dist/autoscroll.js +22 -22
- package/dist/badge/badge.js +35 -35
- package/dist/badge/css/badge.css +7 -7
- package/dist/badge/css/badge.css.map +1 -1
- package/dist/breadcrumbs/css/breadcrumbs.css +7 -7
- package/dist/breadcrumbs/css/breadcrumbs.css.map +1 -1
- package/dist/breadcrumbs.js +57 -57
- package/dist/btngrouper.js +64 -64
- package/dist/bulkActionPopup/css/bulkActionPopup.css +7 -7
- package/dist/bulkActionPopup/css/bulkActionPopup.css.map +1 -1
- package/dist/bulkActionPopup.js +91 -91
- package/dist/button.js +89 -89
- package/dist/calculator/css/calculator.css +10 -10
- package/dist/calculator/css/calculator.css.map +1 -1
- package/dist/calculator.js +51 -51
- package/dist/cascadingComboBox.js +34 -34
- package/dist/ckeditor/ckConfigurator.js +49 -49
- package/dist/ckeditor/dtdHandler.js +110 -110
- package/dist/class/css/selector.css +7 -7
- package/dist/class/css/selector.css.map +1 -1
- package/dist/class/selector.js +98 -98
- package/dist/component/alignable.js +81 -81
- package/dist/component/containable.js +37 -37
- package/dist/component/css/components.css +7 -7
- package/dist/component/css/components.css.map +1 -1
- package/dist/component/css/windowComponent.css +7 -7
- package/dist/component/css/windowComponent.css.map +1 -1
- package/dist/component/draggable.js +25 -25
- package/dist/component/placeable.js +70 -70
- package/dist/component/resizable.js +61 -61
- package/dist/component/stackable.js +20 -20
- package/dist/component/windowed.js +46 -46
- package/dist/component.js +140 -140
- package/dist/container.js +76 -76
- package/dist/contextualPopup/css/contextualPopup.css +7 -7
- package/dist/contextualPopup/css/contextualPopup.css.map +1 -1
- package/dist/contextualPopup.js +100 -100
- package/dist/dashboard/css/dashboard.css +7 -7
- package/dist/dashboard/css/dashboard.css.map +1 -1
- package/dist/dashboard.js +62 -62
- package/dist/datalist/css/datalist.css +7 -7
- package/dist/datalist/css/datalist.css.map +1 -1
- package/dist/datalist.js +147 -147
- package/dist/datatable/css/datatable.css +7 -7
- package/dist/datatable/css/datatable.css.map +1 -1
- package/dist/datatable/filterStrategy/filterStrategy.js +27 -27
- package/dist/datatable/filterStrategy/multiple.js +25 -25
- package/dist/datatable/filterStrategy/single.js +25 -25
- package/dist/datatable.js +377 -377
- package/dist/dateRange/css/dateRange.css +7 -7
- package/dist/dateRange/css/dateRange.css.map +1 -1
- package/dist/dateRange/dateRange.js +97 -97
- package/dist/datetime/css/picker.css +7 -7
- package/dist/datetime/css/picker.css.map +1 -1
- package/dist/datetime/picker.js +161 -161
- package/dist/deleter.js +79 -79
- package/dist/destination/css/selector.css +7 -7
- package/dist/destination/css/selector.css.map +1 -1
- package/dist/destination/selector.js +52 -52
- package/dist/dialog/alert.js +26 -26
- package/dist/dialog/confirm.js +27 -27
- package/dist/dialog/confirmDelete.js +31 -31
- package/dist/dialog.js +141 -141
- package/dist/disabler.js +90 -90
- package/dist/documentViewer/css/documentViewer.css +7 -7
- package/dist/documentViewer/css/documentViewer.css.map +1 -1
- package/dist/documentViewer/providers/pdfViewer/fallback/viewer.js +30 -30
- package/dist/documentViewer/providers/pdfViewer/pdfjs/areaBroker.js +25 -25
- package/dist/documentViewer/providers/pdfViewer/pdfjs/findBar.js +120 -120
- package/dist/documentViewer/providers/pdfViewer/pdfjs/pageView.js +89 -89
- package/dist/documentViewer/providers/pdfViewer/pdfjs/pagesManager.js +52 -52
- package/dist/documentViewer/providers/pdfViewer/pdfjs/searchEngine.js +112 -112
- package/dist/documentViewer/providers/pdfViewer/pdfjs/textManager.js +67 -67
- package/dist/documentViewer/providers/pdfViewer/pdfjs/viewer.js +81 -81
- package/dist/documentViewer/providers/pdfViewer/pdfjs/wrapper.js +111 -111
- package/dist/documentViewer/providers/pdfViewer.js +29 -29
- package/dist/documentViewer/viewerFactory.js +71 -71
- package/dist/documentViewer.js +86 -86
- package/dist/dropdown/css/dropdown.css +7 -7
- package/dist/dropdown/css/dropdown.css.map +1 -1
- package/dist/dropdown.js +84 -84
- package/dist/durationer.js +58 -58
- package/dist/dynamicComponent/css/dynamicComponent.css +7 -7
- package/dist/dynamicComponent/css/dynamicComponent.css.map +1 -1
- package/dist/dynamicComponent.js +103 -103
- package/dist/feedback.js +84 -84
- package/dist/figure/FigureStateActive.js +111 -111
- package/dist/filesender.js +26 -26
- package/dist/filter.js +34 -34
- package/dist/form/css/dropdownForm.css +7 -7
- package/dist/form/css/dropdownForm.css.map +1 -1
- package/dist/form/css/form.css +7 -7
- package/dist/form/css/form.css.map +1 -1
- package/dist/form/dropdownForm.js +99 -99
- package/dist/form/form.js +248 -248
- package/dist/form/simpleForm.js +71 -71
- package/dist/form/validator/css/validator.css +7 -7
- package/dist/form/validator/css/validator.css.map +1 -1
- package/dist/form/validator/renderer.js +52 -52
- package/dist/form/validator/validator.js +103 -103
- package/dist/form/widget/css/widget.css +7 -7
- package/dist/form/widget/css/widget.css.map +1 -1
- package/dist/form/widget/definitions.js +24 -24
- package/dist/form/widget/loader.js +16 -16
- package/dist/form/widget/providers/checkBox.js +62 -62
- package/dist/form/widget/providers/comboBox.js +46 -46
- package/dist/form/widget/providers/default.js +35 -35
- package/dist/form/widget/providers/hidden.js +37 -37
- package/dist/form/widget/providers/hiddenBox.js +58 -58
- package/dist/form/widget/providers/radioBox.js +57 -57
- package/dist/form/widget/providers/textArea.js +35 -35
- package/dist/form/widget/providers/textBox.js +34 -34
- package/dist/form/widget/widget.js +141 -141
- package/dist/form.js +10 -10
- package/dist/formValidator/formValidator.js +61 -61
- package/dist/formValidator/highlighters/highlighter.js +41 -41
- package/dist/formValidator/highlighters/message.js +29 -29
- package/dist/formValidator/highlighters/tooltip.js +32 -32
- package/dist/generis/form/css/form.css +7 -7
- package/dist/generis/form/css/form.css.map +1 -1
- package/dist/generis/form/form.js +73 -73
- package/dist/generis/validator/css/validator.css +7 -7
- package/dist/generis/validator/css/validator.css.map +1 -1
- package/dist/generis/validator/validator.js +38 -38
- package/dist/generis/widget/checkBox/checkBox.js +39 -39
- package/dist/generis/widget/comboBox/comboBox.js +32 -32
- package/dist/generis/widget/css/widget.css +7 -7
- package/dist/generis/widget/css/widget.css.map +1 -1
- package/dist/generis/widget/hiddenBox/hiddenBox.js +40 -40
- package/dist/generis/widget/loader.js +20 -20
- package/dist/generis/widget/textBox/textBox.js +27 -27
- package/dist/generis/widget/widget.js +47 -47
- package/dist/groupedComboBox.js +36 -36
- package/dist/groupvalidator.js +19 -19
- package/dist/hider.js +41 -41
- package/dist/highlighter.js +278 -278
- package/dist/image/ImgStateActive/extractLabel.js +20 -20
- package/dist/image/ImgStateActive/helper.js +16 -16
- package/dist/image/ImgStateActive/initHelper.js +85 -85
- package/dist/image/ImgStateActive/initMediaEditor.js +48 -48
- package/dist/image/ImgStateActive/mediaSizer.js +32 -32
- package/dist/image/ImgStateActive.js +104 -104
- package/dist/incrementer.js +58 -58
- package/dist/inplacer.js +87 -87
- package/dist/interactUtils.js +42 -42
- package/dist/itemButtonList/css/item-button-list.css +23 -23
- package/dist/itemButtonList/css/item-button-list.css.map +1 -1
- package/dist/itemButtonList.js +102 -102
- package/dist/keyNavigation/navigableDomElement.js +76 -76
- package/dist/keyNavigation/navigator.js +158 -158
- package/dist/listbox/css/listbox.css +7 -7
- package/dist/listbox/css/listbox.css.map +1 -1
- package/dist/listbox.js +84 -84
- package/dist/liststyler.js +57 -57
- package/dist/loadingButton/css/button.css +7 -7
- package/dist/loadingButton/css/button.css.map +1 -1
- package/dist/loadingButton/loadingButton.js +35 -35
- package/dist/lock.js +112 -112
- package/dist/login/login.js +87 -87
- package/dist/maths/calculator/basicCalculator.js +50 -50
- package/dist/maths/calculator/calculatorComponent.js +29 -29
- package/dist/maths/calculator/core/areaBroker.js +25 -25
- package/dist/maths/calculator/core/board.js +300 -300
- package/dist/maths/calculator/core/expression.js +98 -98
- package/dist/maths/calculator/core/labels.js +16 -16
- package/dist/maths/calculator/core/plugin.js +19 -19
- package/dist/maths/calculator/core/terms.js +26 -26
- package/dist/maths/calculator/core/tokenizer.js +98 -98
- package/dist/maths/calculator/core/tokens.js +75 -75
- package/dist/maths/calculator/css/calculator.css +7 -7
- package/dist/maths/calculator/css/calculator.css.map +1 -1
- package/dist/maths/calculator/defaultCalculator.js +23 -23
- package/dist/maths/calculator/plugins/core/degrad.js +22 -22
- package/dist/maths/calculator/plugins/core/history.js +45 -45
- package/dist/maths/calculator/plugins/core/remind.js +22 -22
- package/dist/maths/calculator/plugins/core/stepNavigation.js +31 -31
- package/dist/maths/calculator/plugins/keyboard/templateKeyboard/templateKeyboard.js +48 -48
- package/dist/maths/calculator/plugins/modifiers/pow10.js +35 -35
- package/dist/maths/calculator/plugins/modifiers/sign.js +90 -90
- package/dist/maths/calculator/plugins/screen/simpleScreen/simpleScreen.js +47 -47
- package/dist/maths/calculator/pluginsLoader.js +21 -21
- package/dist/maths/calculator/scientificCalculator.js +78 -78
- package/dist/mediaEditor/mediaEditorComponent.js +51 -51
- package/dist/mediaEditor/plugins/mediaAlignment/helper.js +16 -16
- package/dist/mediaEditor/plugins/mediaAlignment/mediaAlignmentComponent.js +47 -47
- package/dist/mediaEditor/plugins/mediaAlignment/style.css +7 -7
- package/dist/mediaEditor/plugins/mediaDimension/helper.js +70 -70
- package/dist/mediaEditor/plugins/mediaDimension/mediaDimensionComponent.js +140 -140
- package/dist/mediaEditor/plugins/mediaDimension/style.css +141 -141
- package/dist/mediaplayer/css/player.css +7 -7
- package/dist/mediaplayer/css/player.css.map +1 -1
- package/dist/mediaplayer/players/html5.js +52 -52
- package/dist/mediaplayer/players/youtube.js +39 -39
- package/dist/mediaplayer/players.js +18 -18
- package/dist/mediaplayer/support.js +55 -55
- package/dist/mediaplayer/utils/reminder.js +100 -100
- package/dist/mediaplayer/utils/timeObserver.js +92 -92
- package/dist/mediaplayer/youtubeManager.js +51 -51
- package/dist/mediaplayer.js +447 -447
- package/dist/mediasizer.js +122 -122
- package/dist/modal.js +87 -87
- package/dist/movableComponent.js +35 -35
- package/dist/pageSizeSelector.js +31 -31
- package/dist/pageStatus.js +33 -33
- package/dist/pagination/css/pagination.css +7 -7
- package/dist/pagination/css/pagination.css.map +1 -1
- package/dist/pagination/paginationStrategy.js +23 -23
- package/dist/pagination/providers/pages.js +24 -24
- package/dist/pagination/providers/simple.js +22 -22
- package/dist/pagination.js +45 -45
- package/dist/previewer.js +67 -67
- package/dist/progressbar.js +58 -58
- package/dist/propertySelector/css/propertySelector.css +7 -7
- package/dist/propertySelector/css/propertySelector.css.map +1 -1
- package/dist/propertySelector/propertySelector.js +86 -86
- package/dist/report.js +73 -73
- package/dist/resource/css/selector.css +7 -7
- package/dist/resource/css/selector.css.map +1 -1
- package/dist/resource/filters.js +60 -60
- package/dist/resource/list.js +53 -53
- package/dist/resource/selectable.js +92 -92
- package/dist/resource/selector.js +182 -182
- package/dist/resource/tree.js +91 -91
- package/dist/resourcemgr/css/resourcemgr.css +7 -7
- package/dist/resourcemgr/css/resourcemgr.css.map +1 -1
- package/dist/resourcemgr/fileBrowser.js +75 -75
- package/dist/resourcemgr/fileSelector.js +45 -45
- package/dist/resourcemgr/util/updatePermissions.js +4 -4
- package/dist/resourcemgr.js +49 -49
- package/dist/scroller.js +26 -26
- package/dist/searchModal/advancedSearch.js +155 -155
- package/dist/searchModal/css/advancedSearch.css +7 -7
- package/dist/searchModal/css/advancedSearch.css.map +1 -1
- package/dist/searchModal/css/searchModal.css +7 -7
- package/dist/searchModal/css/searchModal.css.map +1 -1
- package/dist/searchModal.js +134 -134
- package/dist/selecter.js +6 -6
- package/dist/stacker.js +43 -43
- package/dist/switch/css/switch.css +7 -7
- package/dist/switch/css/switch.css.map +1 -1
- package/dist/switch/switch.js +62 -62
- package/dist/tableModel.js +33 -33
- package/dist/tabs/css/tabs.css +12 -12
- package/dist/tabs/css/tabs.css.map +1 -1
- package/dist/tabs.js +220 -220
- package/dist/taskQueue/css/taskQueue.css +7 -7
- package/dist/taskQueue/css/taskQueue.css.map +1 -1
- package/dist/taskQueue/status.js +59 -59
- package/dist/taskQueue/table.js +54 -54
- package/dist/taskQueue/taskQueue.js +18 -18
- package/dist/taskQueue/taskQueueModel.js +102 -102
- package/dist/taskQueue.js +47 -47
- package/dist/taskQueueButton/css/taskable.css +7 -7
- package/dist/taskQueueButton/css/taskable.css.map +1 -1
- package/dist/taskQueueButton/css/treeButton.css +7 -7
- package/dist/taskQueueButton/css/treeButton.css.map +1 -1
- package/dist/taskQueueButton/standardButton.js +39 -39
- package/dist/taskQueueButton/taskable.js +41 -41
- package/dist/taskQueueButton/treeButton.js +43 -43
- package/dist/themeLoader.js +75 -75
- package/dist/themes.js +84 -84
- package/dist/toggler.js +57 -57
- package/dist/tooltip.js +41 -41
- package/dist/tooltipster.js +17 -17
- package/dist/transformer.js +117 -117
- package/dist/tristateCheckboxGroup/css/tristateCheckboxGroup.css +7 -7
- package/dist/tristateCheckboxGroup/css/tristateCheckboxGroup.css.map +1 -1
- package/dist/tristateCheckboxGroup.js +62 -62
- package/dist/uploader.js +145 -145
- package/dist/validator/validators.js +48 -48
- package/dist/validator.js +23 -23
- package/dist/waitForMedia.js +33 -33
- package/dist/waitingDialog/css/waitingDialog.css +7 -7
- package/dist/waitingDialog/css/waitingDialog.css.map +1 -1
- package/dist/waitingDialog/waitingDialog.js +54 -54
- package/package.json +110 -110
- package/scss/basic.scss +16 -16
- package/scss/ckeditor/skins/tao/scss/inc/_ck-icons.scss +60 -60
- package/scss/ckeditor/skins/tao/scss/inc/_tao.scss +59 -59
- package/scss/font/tao/tao.svg +235 -235
- package/scss/inc/_base.scss +495 -495
- package/scss/inc/_bootstrap.scss +6 -6
- package/scss/inc/_buttons.scss +114 -114
- package/scss/inc/_colors.scss +93 -93
- package/scss/inc/_feedback.scss +154 -154
- package/scss/inc/_flex-grid.scss +15 -15
- package/scss/inc/_fonts.scss +4 -4
- package/scss/inc/_forms.scss +832 -832
- package/scss/inc/_functions.scss +283 -283
- package/scss/inc/_jquery.nouislider.scss +254 -254
- package/scss/inc/_normalize.scss +528 -528
- package/scss/inc/_report.scss +67 -67
- package/scss/inc/_secondary-properties.scss +89 -89
- package/scss/inc/_select2.scss +634 -634
- package/scss/inc/_toolbars.scss +155 -155
- package/scss/inc/_tooltip.scss +312 -312
- package/scss/inc/_variables.scss +21 -21
- package/scss/inc/base/_highlight.scss +5 -5
- package/scss/inc/base/_list-style.scss +58 -58
- package/scss/inc/base/_svg.scss +3 -3
- package/scss/inc/base/_table.scss +62 -62
- package/scss/inc/fonts/_source-sans-pro.scss +29 -29
- package/scss/inc/fonts/_tao-icon-classes.scss +227 -227
- package/scss/inc/fonts/_tao-icon-def.scss +12 -12
- package/scss/inc/fonts/_tao-icon-vars.scss +241 -241
- package/src/actionbar/tpl/main.tpl +8 -8
- package/src/actionbar.js +251 -251
- package/src/adder.js +250 -250
- package/src/animable/absorbable/absorbable.js +134 -134
- package/src/animable/absorbable/css/absorb.css +7 -7
- package/src/animable/absorbable/css/absorb.css.map +1 -1
- package/src/animable/absorbable/scss/absorb.scss +37 -37
- package/src/animable/pulsable/css/pulse.css +7 -7
- package/src/animable/pulsable/css/pulse.css.map +1 -1
- package/src/animable/pulsable/pulsable.js +90 -90
- package/src/animable/pulsable/scss/pulse.scss +22 -22
- package/src/areaBroker.js +160 -160
- package/src/autocomplete/css/autocomplete.css +7 -7
- package/src/autocomplete/css/autocomplete.css.map +1 -1
- package/src/autocomplete/scss/autocomplete.scss +37 -37
- package/src/autocomplete.js +1027 -1027
- package/src/autoscroll.js +57 -57
- package/src/badge/badge.js +119 -119
- package/src/badge/css/badge.css +7 -7
- package/src/badge/css/badge.css.map +1 -1
- package/src/badge/scss/badge.scss +92 -92
- package/src/badge/tpl/badge.tpl +4 -4
- package/src/breadcrumbs/css/breadcrumbs.css +7 -7
- package/src/breadcrumbs/css/breadcrumbs.css.map +1 -1
- package/src/breadcrumbs/scss/breadcrumbs.scss +52 -52
- package/src/breadcrumbs/tpl/breadcrumbs.tpl +20 -20
- package/src/breadcrumbs.js +99 -99
- package/src/btngrouper.js +213 -213
- package/src/bulkActionPopup/css/bulkActionPopup.css +7 -7
- package/src/bulkActionPopup/css/bulkActionPopup.css.map +1 -1
- package/src/bulkActionPopup/scss/bulkActionPopup.scss +63 -63
- package/src/bulkActionPopup/tpl/layout.tpl +76 -76
- package/src/bulkActionPopup/tpl/select.tpl +8 -8
- package/src/bulkActionPopup.js +274 -274
- package/src/button/tpl/button.tpl +4 -4
- package/src/button.js +135 -135
- package/src/calculator/css/calculator.css +10 -10
- package/src/calculator/css/calculator.css.map +1 -1
- package/src/calculator/scss/calculator.scss +139 -139
- package/src/calculator.js +188 -188
- package/src/cascadingComboBox.js +126 -126
- package/src/ckeditor/ckConfigurator.js +742 -742
- package/src/ckeditor/dtdHandler.js +1030 -1030
- package/src/class/css/selector.css +7 -7
- package/src/class/css/selector.css.map +1 -1
- package/src/class/scss/selector.scss +101 -101
- package/src/class/selector.js +329 -329
- package/src/class/tpl/listItem.tpl +9 -9
- package/src/class/tpl/selector.tpl +10 -10
- package/src/component/alignable.js +274 -274
- package/src/component/containable.js +122 -122
- package/src/component/css/components.css +7 -7
- package/src/component/css/components.css.map +1 -1
- package/src/component/css/windowComponent.css +7 -7
- package/src/component/css/windowComponent.css.map +1 -1
- package/src/component/draggable.js +104 -104
- package/src/component/placeable.js +233 -233
- package/src/component/resizable.js +195 -195
- package/src/component/scss/components.scss +507 -507
- package/src/component/scss/windowComponent.scss +62 -62
- package/src/component/stackable.js +67 -67
- package/src/component/tpl/window.tpl +7 -7
- package/src/component/windowed.js +206 -206
- package/src/component.js +401 -401
- package/src/container.js +200 -200
- package/src/contextualPopup/css/contextualPopup.css +7 -7
- package/src/contextualPopup/css/contextualPopup.css.map +1 -1
- package/src/contextualPopup/scss/contextualPopup.scss +78 -78
- package/src/contextualPopup/tpl/popup.tpl +10 -10
- package/src/contextualPopup.js +294 -294
- package/src/css/basic.css +104 -104
- package/src/css/basic.css.map +1 -1
- package/src/dashboard/css/dashboard.css +7 -7
- package/src/dashboard/css/dashboard.css.map +1 -1
- package/src/dashboard/scss/dashboard.scss +93 -93
- package/src/dashboard/tpl/dashboard.tpl +16 -16
- package/src/dashboard/tpl/dashboardMetricsList.tpl +15 -15
- package/src/dashboard.js +179 -179
- package/src/datalist/css/datalist.css +7 -7
- package/src/datalist/css/datalist.css.map +1 -1
- package/src/datalist/scss/datalist.scss +116 -116
- package/src/datalist/tpl/list.tpl +24 -24
- package/src/datalist/tpl/main.tpl +44 -44
- package/src/datalist.js +500 -500
- package/src/datatable/css/datatable.css +7 -7
- package/src/datatable/css/datatable.css.map +1 -1
- package/src/datatable/filterStrategy/filterStrategy.js +70 -70
- package/src/datatable/filterStrategy/multiple.js +126 -126
- package/src/datatable/filterStrategy/single.js +108 -108
- package/src/datatable/scss/datatable.scss +149 -149
- package/src/datatable/tpl/button.tpl +6 -6
- package/src/datatable/tpl/layout.tpl +158 -158
- package/src/datatable.js +1056 -1056
- package/src/dateRange/css/dateRange.css +7 -7
- package/src/dateRange/css/dateRange.css.map +1 -1
- package/src/dateRange/dateRange.js +341 -341
- package/src/dateRange/scss/dateRange.scss +7 -7
- package/src/dateRange/tpl/select.tpl +18 -18
- package/src/datetime/css/picker.css +7 -7
- package/src/datetime/css/picker.css.map +1 -1
- package/src/datetime/picker.js +576 -576
- package/src/datetime/scss/picker.scss +192 -192
- package/src/datetime/tpl/picker.tpl +18 -18
- package/src/deleter/undo.tpl +6 -6
- package/src/deleter.js +296 -296
- package/src/destination/css/selector.css +7 -7
- package/src/destination/css/selector.css.map +1 -1
- package/src/destination/scss/selector.scss +36 -36
- package/src/destination/selector.js +195 -195
- package/src/destination/tpl/selector.tpl +13 -13
- package/src/dialog/alert.js +70 -70
- package/src/dialog/confirm.js +85 -85
- package/src/dialog/confirmDelete.js +95 -95
- package/src/dialog/tpl/body.tpl +24 -24
- package/src/dialog/tpl/buttons.tpl +6 -6
- package/src/dialog/tpl/checkbox.tpl +5 -5
- package/src/dialog.js +517 -517
- package/src/disabler.js +230 -230
- package/src/documentViewer/css/documentViewer.css +7 -7
- package/src/documentViewer/css/documentViewer.css.map +1 -1
- package/src/documentViewer/providers/pdfViewer/fallback/viewer.js +69 -69
- package/src/documentViewer/providers/pdfViewer/pdfjs/areaBroker.js +41 -41
- package/src/documentViewer/providers/pdfViewer/pdfjs/findBar.js +475 -475
- package/src/documentViewer/providers/pdfViewer/pdfjs/findBar.tpl +20 -20
- package/src/documentViewer/providers/pdfViewer/pdfjs/match.tpl +1 -1
- package/src/documentViewer/providers/pdfViewer/pdfjs/page.tpl +4 -4
- package/src/documentViewer/providers/pdfViewer/pdfjs/pageView.js +318 -318
- package/src/documentViewer/providers/pdfViewer/pdfjs/pagesManager.js +167 -167
- package/src/documentViewer/providers/pdfViewer/pdfjs/searchEngine.js +451 -451
- package/src/documentViewer/providers/pdfViewer/pdfjs/textManager.js +252 -252
- package/src/documentViewer/providers/pdfViewer/pdfjs/viewer.js +299 -299
- package/src/documentViewer/providers/pdfViewer/pdfjs/viewer.tpl +16 -16
- package/src/documentViewer/providers/pdfViewer/pdfjs/wrapper.js +351 -351
- package/src/documentViewer/providers/pdfViewer.js +93 -93
- package/src/documentViewer/scss/documentViewer.scss +184 -184
- package/src/documentViewer/viewerFactory.js +191 -191
- package/src/documentViewer.js +238 -238
- package/src/dropdown/css/dropdown.css +7 -7
- package/src/dropdown/css/dropdown.css.map +1 -1
- package/src/dropdown/scss/dropdown.scss +99 -99
- package/src/dropdown/tpl/dropdown.tpl +8 -8
- package/src/dropdown/tpl/list-item.tpl +4 -4
- package/src/dropdown.js +255 -255
- package/src/durationer.js +220 -220
- package/src/dynamicComponent/css/dynamicComponent.css +7 -7
- package/src/dynamicComponent/css/dynamicComponent.css.map +1 -1
- package/src/dynamicComponent/scss/dynamicComponent.scss +98 -98
- package/src/dynamicComponent/tpl/layout.tpl +17 -17
- package/src/dynamicComponent.js +553 -553
- package/src/feedback/feedback.tpl +7 -7
- package/src/feedback.js +295 -295
- package/src/figure/FigureStateActive.js +184 -184
- package/src/filesender.js +112 -112
- package/src/filter/template.tpl +5 -5
- package/src/filter.js +129 -129
- package/src/form/css/dropdownForm.css +7 -7
- package/src/form/css/dropdownForm.css.map +1 -1
- package/src/form/css/form.css +7 -7
- package/src/form/css/form.css.map +1 -1
- package/src/form/dropdownForm.js +281 -281
- package/src/form/form.js +688 -688
- package/src/form/scss/dropdownForm.scss +60 -60
- package/src/form/scss/form.scss +25 -25
- package/src/form/simpleForm.js +125 -125
- package/src/form/tpl/dropdownForm.tpl +4 -4
- package/src/form/tpl/form.tpl +7 -7
- package/src/form/validator/css/validator.css +7 -7
- package/src/form/validator/css/validator.css.map +1 -1
- package/src/form/validator/renderer.js +118 -118
- package/src/form/validator/scss/validator.scss +14 -14
- package/src/form/validator/tpl/message.tpl +1 -1
- package/src/form/validator/tpl/validator.tpl +1 -1
- package/src/form/validator/validator.js +220 -220
- package/src/form/widget/css/widget.css +7 -7
- package/src/form/widget/css/widget.css.map +1 -1
- package/src/form/widget/definitions.js +51 -51
- package/src/form/widget/loader.js +40 -40
- package/src/form/widget/providers/checkBox.js +138 -138
- package/src/form/widget/providers/comboBox.js +63 -63
- package/src/form/widget/providers/default.js +90 -90
- package/src/form/widget/providers/hidden.js +62 -62
- package/src/form/widget/providers/hiddenBox.js +152 -152
- package/src/form/widget/providers/radioBox.js +99 -99
- package/src/form/widget/providers/textArea.js +52 -52
- package/src/form/widget/providers/textBox.js +48 -48
- package/src/form/widget/scss/widget.scss +55 -55
- package/src/form/widget/tpl/checkBox.tpl +25 -25
- package/src/form/widget/tpl/comboBox.tpl +13 -13
- package/src/form/widget/tpl/hidden.tpl +1 -1
- package/src/form/widget/tpl/hiddenBox.tpl +17 -17
- package/src/form/widget/tpl/label.tpl +6 -6
- package/src/form/widget/tpl/radioBox.tpl +25 -25
- package/src/form/widget/tpl/textArea.tpl +8 -8
- package/src/form/widget/tpl/widget.tpl +8 -8
- package/src/form/widget/widget.js +372 -372
- package/src/form.js +47 -47
- package/src/formValidator/formValidator.js +253 -253
- package/src/formValidator/highlighters/highlighter.js +103 -103
- package/src/formValidator/highlighters/message.js +68 -68
- package/src/formValidator/highlighters/tooltip.js +78 -78
- package/src/generis/form/css/form.css +7 -7
- package/src/generis/form/css/form.css.map +1 -1
- package/src/generis/form/form.js +239 -239
- package/src/generis/form/readme.md +70 -70
- package/src/generis/form/scss/form.scss +23 -23
- package/src/generis/form/tpl/form.tpl +16 -16
- package/src/generis/validator/css/validator.css +7 -7
- package/src/generis/validator/css/validator.css.map +1 -1
- package/src/generis/validator/readme.md +46 -46
- package/src/generis/validator/scss/validator.scss +13 -13
- package/src/generis/validator/validator.js +128 -128
- package/src/generis/widget/checkBox/checkBox.js +112 -112
- package/src/generis/widget/checkBox/checkBox.tpl +18 -18
- package/src/generis/widget/comboBox/comboBox.js +66 -66
- package/src/generis/widget/comboBox/comboBox.tpl +12 -12
- package/src/generis/widget/css/widget.css +7 -7
- package/src/generis/widget/css/widget.css.map +1 -1
- package/src/generis/widget/hiddenBox/hiddenBox.js +131 -131
- package/src/generis/widget/hiddenBox/hiddenBox.tpl +16 -16
- package/src/generis/widget/loader.js +49 -49
- package/src/generis/widget/readme.md +59 -59
- package/src/generis/widget/scss/widget.scss +61 -61
- package/src/generis/widget/textBox/textBox.js +64 -64
- package/src/generis/widget/textBox/textBox.tpl +7 -7
- package/src/generis/widget/widget.js +164 -164
- package/src/generis/widget/widget.tpl +5 -5
- package/src/groupedComboBox.js +99 -99
- package/src/groupvalidator.js +84 -84
- package/src/hider.js +88 -88
- package/src/highlighter.js +1192 -1192
- package/src/image/ImgStateActive/extractLabel.js +29 -29
- package/src/image/ImgStateActive/helper.js +36 -36
- package/src/image/ImgStateActive/initHelper.js +137 -137
- package/src/image/ImgStateActive/initMediaEditor.js +92 -92
- package/src/image/ImgStateActive/mediaSizer.js +63 -63
- package/src/image/ImgStateActive.js +115 -115
- package/src/incrementer.js +309 -309
- package/src/inplacer.js +315 -315
- package/src/interactUtils.js +140 -140
- package/src/itemButtonList/css/item-button-list.css +23 -23
- package/src/itemButtonList/css/item-button-list.css.map +1 -1
- package/src/itemButtonList/scss/item-button-list.scss +236 -236
- package/src/itemButtonList/tpl/itemButtonList.tpl +21 -21
- package/src/itemButtonList.js +274 -274
- package/src/keyNavigation/navigableDomElement.js +285 -285
- package/src/keyNavigation/navigator.js +535 -535
- package/src/listbox/css/listbox.css +7 -7
- package/src/listbox/css/listbox.css.map +1 -1
- package/src/listbox/scss/listbox.scss +116 -116
- package/src/listbox/tpl/list.tpl +14 -14
- package/src/listbox/tpl/main.tpl +9 -9
- package/src/listbox.js +252 -252
- package/src/liststyler.js +155 -155
- package/src/loadingButton/css/button.css +7 -7
- package/src/loadingButton/css/button.css.map +1 -1
- package/src/loadingButton/loadingButton.js +110 -110
- package/src/loadingButton/scss/button.scss +41 -41
- package/src/loadingButton/tpl/button.tpl +5 -5
- package/src/lock/lock.tpl +16 -16
- package/src/lock.js +393 -393
- package/src/login/login.js +317 -317
- package/src/login/tpl/login.tpl +29 -29
- package/src/login/tpl/passwordReveal.tpl +7 -7
- package/src/maths/calculator/basicCalculator.js +55 -55
- package/src/maths/calculator/calculatorComponent.js +128 -128
- package/src/maths/calculator/core/areaBroker.js +38 -38
- package/src/maths/calculator/core/board.js +841 -841
- package/src/maths/calculator/core/expression.js +430 -430
- package/src/maths/calculator/core/labels.js +116 -116
- package/src/maths/calculator/core/plugin.js +40 -40
- package/src/maths/calculator/core/terms.js +459 -459
- package/src/maths/calculator/core/tokenizer.js +245 -245
- package/src/maths/calculator/core/tokens.js +178 -178
- package/src/maths/calculator/core/tpl/board.tpl +4 -4
- package/src/maths/calculator/css/calculator.css +7 -7
- package/src/maths/calculator/css/calculator.css.map +1 -1
- package/src/maths/calculator/defaultCalculator.js +66 -66
- package/src/maths/calculator/plugins/core/degrad.js +90 -90
- package/src/maths/calculator/plugins/core/history.js +166 -166
- package/src/maths/calculator/plugins/core/remind.js +96 -96
- package/src/maths/calculator/plugins/core/stepNavigation.js +175 -175
- package/src/maths/calculator/plugins/keyboard/templateKeyboard/defaultTemplate.tpl +36 -36
- package/src/maths/calculator/plugins/keyboard/templateKeyboard/templateKeyboard.js +91 -91
- package/src/maths/calculator/plugins/modifiers/pow10.js +143 -143
- package/src/maths/calculator/plugins/modifiers/sign.js +339 -339
- package/src/maths/calculator/plugins/screen/simpleScreen/defaultTemplate.tpl +3 -3
- package/src/maths/calculator/plugins/screen/simpleScreen/history.tpl +3 -3
- package/src/maths/calculator/plugins/screen/simpleScreen/simpleScreen.js +190 -190
- package/src/maths/calculator/pluginsLoader.js +46 -46
- package/src/maths/calculator/scientificCalculator.js +74 -74
- package/src/maths/calculator/scss/calculator.scss +396 -396
- package/src/maths/calculator/tpl/basicKeyboard.tpl +37 -37
- package/src/maths/calculator/tpl/basicScreen.tpl +2 -2
- package/src/maths/calculator/tpl/scientificKeyboard.tpl +61 -61
- package/src/maths/calculator/tpl/scientificScreen.tpl +3 -3
- package/src/mediaEditor/mediaEditorComponent.js +141 -141
- package/src/mediaEditor/plugins/mediaAlignment/helper.js +110 -110
- package/src/mediaEditor/plugins/mediaAlignment/mediaAlignmentComponent.js +99 -99
- package/src/mediaEditor/plugins/mediaAlignment/style.css +7 -7
- package/src/mediaEditor/plugins/mediaAlignment/tpl/mediaAlignment.tpl +25 -25
- package/src/mediaEditor/plugins/mediaDimension/helper.js +190 -190
- package/src/mediaEditor/plugins/mediaDimension/mediaDimensionComponent.js +561 -561
- package/src/mediaEditor/plugins/mediaDimension/style.css +141 -141
- package/src/mediaEditor/plugins/mediaDimension/tpl/mediaDimension.tpl +55 -55
- package/src/mediaEditor/tpl/editor.tpl +4 -4
- package/src/mediaplayer/css/player.css +7 -7
- package/src/mediaplayer/css/player.css.map +1 -1
- package/src/mediaplayer/players/html5.js +564 -564
- package/src/mediaplayer/players/youtube.js +323 -323
- package/src/mediaplayer/players.js +29 -29
- package/src/mediaplayer/readme.md +305 -305
- package/src/mediaplayer/scss/player.scss +569 -569
- package/src/mediaplayer/support.js +126 -126
- package/src/mediaplayer/tpl/audio.tpl +6 -6
- package/src/mediaplayer/tpl/player.tpl +37 -37
- package/src/mediaplayer/tpl/source.tpl +1 -1
- package/src/mediaplayer/tpl/video.tpl +6 -6
- package/src/mediaplayer/tpl/youtube.tpl +1 -1
- package/src/mediaplayer/utils/reminder.js +184 -184
- package/src/mediaplayer/utils/timeObserver.js +143 -143
- package/src/mediaplayer/youtubeManager.js +161 -161
- package/src/mediaplayer.js +1606 -1606
- package/src/mediasizer/mediasizer.tpl +55 -55
- package/src/mediasizer.js +635 -635
- package/src/modal.js +365 -365
- package/src/movableComponent.js +78 -78
- package/src/pageSizeSelector/tpl/pageSizeSelector.tpl +9 -9
- package/src/pageSizeSelector.js +106 -106
- package/src/pageStatus.js +147 -147
- package/src/pagination/css/pagination.css +7 -7
- package/src/pagination/css/pagination.css.map +1 -1
- package/src/pagination/paginationStrategy.js +53 -53
- package/src/pagination/providers/pages.js +161 -161
- package/src/pagination/providers/simple.js +74 -74
- package/src/pagination/providers/tpl/pages/page.tpl +1 -1
- package/src/pagination/providers/tpl/pages.tpl +8 -8
- package/src/pagination/providers/tpl/simple.tpl +7 -7
- package/src/pagination/scss/pagination.scss +111 -111
- package/src/pagination.js +237 -237
- package/src/previewer.js +300 -300
- package/src/progressbar.js +165 -165
- package/src/propertySelector/css/propertySelector.css +7 -7
- package/src/propertySelector/css/propertySelector.css.map +1 -1
- package/src/propertySelector/propertySelector.js +286 -286
- package/src/propertySelector/scss/propertySelector.scss +66 -66
- package/src/propertySelector/tpl/property-description.tpl +12 -12
- package/src/propertySelector/tpl/property-selector.tpl +6 -6
- package/src/report/feedback.tpl +11 -11
- package/src/report/layout.tpl +10 -10
- package/src/report.js +184 -184
- package/src/resource/css/selector.css +7 -7
- package/src/resource/css/selector.css.map +1 -1
- package/src/resource/filters.js +208 -208
- package/src/resource/list.js +200 -200
- package/src/resource/scss/_filters.scss +26 -26
- package/src/resource/scss/_resource-list.scss +107 -107
- package/src/resource/scss/_resource-tree.scss +205 -205
- package/src/resource/scss/selector.scss +187 -187
- package/src/resource/selectable.js +322 -322
- package/src/resource/selector.js +871 -871
- package/src/resource/tpl/filters.tpl +2 -2
- package/src/resource/tpl/list.tpl +7 -7
- package/src/resource/tpl/listNode.tpl +4 -4
- package/src/resource/tpl/selector.tpl +46 -46
- package/src/resource/tpl/tree.tpl +4 -4
- package/src/resource/tpl/treeNode.tpl +30 -30
- package/src/resource/tree.js +398 -398
- package/src/resourcemgr/css/resourcemgr.css +7 -7
- package/src/resourcemgr/css/resourcemgr.css.map +1 -1
- package/src/resourcemgr/fileBrowser.js +381 -381
- package/src/resourcemgr/filePreview.js +73 -73
- package/src/resourcemgr/fileSelector.js +322 -322
- package/src/resourcemgr/scss/resourcemgr.scss +254 -254
- package/src/resourcemgr/tpl/fileSelect.tpl +39 -39
- package/src/resourcemgr/tpl/folder.tpl +11 -11
- package/src/resourcemgr/tpl/layout.tpl +84 -84
- package/src/resourcemgr/tpl/rootFolder.tpl +13 -13
- package/src/resourcemgr/util/updatePermissions.js +53 -53
- package/src/resourcemgr.js +216 -216
- package/src/scroller.js +94 -94
- package/src/scss/basic.scss +16 -16
- package/src/searchModal/advancedSearch.js +638 -638
- package/src/searchModal/css/advancedSearch.css +7 -7
- package/src/searchModal/css/advancedSearch.css.map +1 -1
- package/src/searchModal/css/searchModal.css +7 -7
- package/src/searchModal/css/searchModal.css.map +1 -1
- package/src/searchModal/scss/advancedSearch.scss +171 -171
- package/src/searchModal/scss/searchModal.scss +393 -393
- package/src/searchModal/tpl/advanced-search.tpl +9 -9
- package/src/searchModal/tpl/info-message.tpl +3 -3
- package/src/searchModal/tpl/invalid-criteria-warning.tpl +10 -10
- package/src/searchModal/tpl/layout.tpl +27 -27
- package/src/searchModal/tpl/list-checkbox-criterion.tpl +17 -17
- package/src/searchModal/tpl/list-select-criterion.tpl +12 -12
- package/src/searchModal/tpl/text-criterion.tpl +11 -11
- package/src/searchModal.js +760 -760
- package/src/selecter.js +43 -43
- package/src/stacker.js +133 -133
- package/src/switch/css/switch.css +7 -7
- package/src/switch/css/switch.css.map +1 -1
- package/src/switch/scss/switch.scss +83 -83
- package/src/switch/switch.js +195 -195
- package/src/switch/tpl/switch.tpl +7 -7
- package/src/tableModel.js +112 -112
- package/src/tabs/css/tabs.css +12 -12
- package/src/tabs/css/tabs.css.map +1 -1
- package/src/tabs/scss/tabs.scss +50 -50
- package/src/tabs/tpl/panel.tpl +3 -3
- package/src/tabs/tpl/tabs.tpl +10 -10
- package/src/tabs.js +528 -528
- package/src/taskQueue/css/taskQueue.css +7 -7
- package/src/taskQueue/css/taskQueue.css.map +1 -1
- package/src/taskQueue/scss/taskQueue.scss +47 -47
- package/src/taskQueue/status.js +228 -228
- package/src/taskQueue/table.js +350 -350
- package/src/taskQueue/taskQueue.js +33 -33
- package/src/taskQueue/taskQueueModel.js +548 -548
- package/src/taskQueue/tpl/statusMessage.tpl +7 -7
- package/src/taskQueue.js +216 -216
- package/src/taskQueueButton/css/taskable.css +7 -7
- package/src/taskQueueButton/css/taskable.css.map +1 -1
- package/src/taskQueueButton/css/treeButton.css +7 -7
- package/src/taskQueueButton/css/treeButton.css.map +1 -1
- package/src/taskQueueButton/scss/taskable.scss +4 -4
- package/src/taskQueueButton/scss/treeButton.scss +34 -34
- package/src/taskQueueButton/standardButton.js +108 -108
- package/src/taskQueueButton/taskable.js +202 -202
- package/src/taskQueueButton/tpl/report.tpl +5 -5
- package/src/taskQueueButton/tpl/treeButton.tpl +6 -6
- package/src/taskQueueButton/treeButton.js +108 -108
- package/src/themeLoader.js +252 -252
- package/src/themes.js +162 -162
- package/src/toggler.js +200 -200
- package/src/tooltip/default.tpl +3 -3
- package/src/tooltip.js +160 -160
- package/src/tooltipster.js +25 -25
- package/src/transformer.js +327 -327
- package/src/tristateCheckboxGroup/css/tristateCheckboxGroup.css +7 -7
- package/src/tristateCheckboxGroup/css/tristateCheckboxGroup.css.map +1 -1
- package/src/tristateCheckboxGroup/scss/tristateCheckboxGroup.scss +15 -15
- package/src/tristateCheckboxGroup/tpl/li.tpl +6 -6
- package/src/tristateCheckboxGroup.js +207 -207
- package/src/uploader/fileEntry.tpl +6 -6
- package/src/uploader/uploader.tpl +32 -32
- package/src/uploader.js +594 -594
- package/src/validator/Report.js +10 -10
- package/src/validator/Validator.js +108 -108
- package/src/validator/validators.js +217 -217
- package/src/validator.js +262 -262
- package/src/waitForMedia.js +82 -82
- package/src/waitingDialog/css/waitingDialog.css +7 -7
- package/src/waitingDialog/css/waitingDialog.css.map +1 -1
- package/src/waitingDialog/scss/waitingDialog.scss +34 -34
- package/src/waitingDialog/waitingDialog.js +240 -240
|
@@ -1,638 +1,638 @@
|
|
|
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) 2021 (original work) Open Assessment Technologies SA ;
|
|
17
|
-
*/
|
|
18
|
-
|
|
19
|
-
import $ from 'jquery';
|
|
20
|
-
import _ from 'lodash';
|
|
21
|
-
import advancedSearchTpl from 'ui/searchModal/tpl/advanced-search';
|
|
22
|
-
import textCriterionTpl from 'ui/searchModal/tpl/text-criterion';
|
|
23
|
-
import invalidCriteriaWarningTpl from 'ui/searchModal/tpl/invalid-criteria-warning';
|
|
24
|
-
import listCheckboxCriterionTpl from 'ui/searchModal/tpl/list-checkbox-criterion';
|
|
25
|
-
import listSelectCriterionTpl from 'ui/searchModal/tpl/list-select-criterion';
|
|
26
|
-
import highlightedTextTpl from 'ui/searchModal/tpl/highlighted-text';
|
|
27
|
-
import classLabelTpl from 'ui/searchModal/tpl/criteria-class-label';
|
|
28
|
-
import aliasTpl from 'ui/searchModal/tpl/criteria-alias';
|
|
29
|
-
import labelTpl from 'ui/searchModal/tpl/criteria-label';
|
|
30
|
-
import 'ui/searchModal/css/advancedSearch.css';
|
|
31
|
-
import component from 'ui/component';
|
|
32
|
-
import 'ui/modal';
|
|
33
|
-
import 'ui/datatable';
|
|
34
|
-
import 'select2';
|
|
35
|
-
import request from 'core/dataProvider/request';
|
|
36
|
-
|
|
37
|
-
/**
|
|
38
|
-
* Sort an array by a particular property.
|
|
39
|
-
* @param {Array} iter - The array to sort.
|
|
40
|
-
* @param {string} prop - The name of the sorting property.
|
|
41
|
-
* @returns {Array} - Returns a sorted copy of the array.
|
|
42
|
-
* @private
|
|
43
|
-
*/
|
|
44
|
-
function sortBy(iter, prop) {
|
|
45
|
-
return Array.from(iter).sort((a, b) => {
|
|
46
|
-
const textA = (a && a[prop]) || '';
|
|
47
|
-
const textB = (b && b[prop]) || '';
|
|
48
|
-
return textA.localeCompare(textB);
|
|
49
|
-
});
|
|
50
|
-
}
|
|
51
|
-
|
|
52
|
-
/**
|
|
53
|
-
* Creates advanced search component
|
|
54
|
-
*
|
|
55
|
-
* @param {object} config
|
|
56
|
-
* @param {object} config.renderTo - DOM element where component will be rendered to
|
|
57
|
-
* @param {string} config.advancedCriteria - advanced criteria to be set on component creation
|
|
58
|
-
* @param {bool} config.hideCriteria - if the criteria must be hidden
|
|
59
|
-
* @param {string} config.rootClassUri - rootClassUri to check for whitelist sections
|
|
60
|
-
* @param {string} config.statusUrl - the URL to the status API (usually '/tao/AdvancedSearch/status')
|
|
61
|
-
* @returns {advancedSearch}
|
|
62
|
-
*/
|
|
63
|
-
export default function advancedSearchFactory(config) {
|
|
64
|
-
// Private properties to be easily accessible by instance methods
|
|
65
|
-
let $container = null;
|
|
66
|
-
let $addCriteria = null;
|
|
67
|
-
let $addCriteriaInput = null;
|
|
68
|
-
let $criteriaSelect = null;
|
|
69
|
-
let $advancedCriteriaContainer = null;
|
|
70
|
-
let criteriaState = null;
|
|
71
|
-
let criteriaMapping = {};
|
|
72
|
-
const criteriaTypes = {
|
|
73
|
-
text: 'text',
|
|
74
|
-
list: 'list'
|
|
75
|
-
};
|
|
76
|
-
|
|
77
|
-
let isAdvancedSearchStatusEnabled;
|
|
78
|
-
let isCriteriaListUpdated = false;
|
|
79
|
-
|
|
80
|
-
// Creates new component
|
|
81
|
-
const instance = component({
|
|
82
|
-
/**
|
|
83
|
-
* Tells if the advanced search is enabled.
|
|
84
|
-
* @returns {boolean}
|
|
85
|
-
*/
|
|
86
|
-
isEnabled() {
|
|
87
|
-
return !!isAdvancedSearchStatusEnabled;
|
|
88
|
-
},
|
|
89
|
-
|
|
90
|
-
/**
|
|
91
|
-
* Request metadata (criteria) for the given uri
|
|
92
|
-
* @param {string} classUri - url to make the reques to
|
|
93
|
-
* @returns {Promise} - Request promise
|
|
94
|
-
*/
|
|
95
|
-
updateCriteria(route) {
|
|
96
|
-
if (!isAdvancedSearchStatusEnabled) {
|
|
97
|
-
return Promise.resolve();
|
|
98
|
-
}
|
|
99
|
-
|
|
100
|
-
const $criteriaIcon = $('.add-criteria-container a span').eq(0);
|
|
101
|
-
$criteriaIcon.toggleClass('icon-add').toggleClass('icon-loop');
|
|
102
|
-
return request(route)
|
|
103
|
-
.then(response => {
|
|
104
|
-
criteriaMapping = response.criteriaMapping || {};
|
|
105
|
-
const classTree = response.classDefinition ? response.classDefinition : response;
|
|
106
|
-
const criteria = formatCriteria(classTree);
|
|
107
|
-
updateCriteria(criteria);
|
|
108
|
-
isCriteriaListUpdated = true;
|
|
109
|
-
$criteriaIcon.toggleClass('icon-add').toggleClass('icon-loop');
|
|
110
|
-
})
|
|
111
|
-
.catch(e => instance.trigger('error', e));
|
|
112
|
-
},
|
|
113
|
-
/**
|
|
114
|
-
* Access to component state
|
|
115
|
-
* @returns {Object} - criteria state
|
|
116
|
-
*/
|
|
117
|
-
getState() {
|
|
118
|
-
return criteriaState;
|
|
119
|
-
},
|
|
120
|
-
/**
|
|
121
|
-
* Removes every rendered criterion, updates criteria state accordingly
|
|
122
|
-
* and removes classes applied to scrollable list of criteria
|
|
123
|
-
*/
|
|
124
|
-
clear() {
|
|
125
|
-
$advancedCriteriaContainer.removeClass(['scrollable', 'scroll-separator-top', 'scroll-separator-bottom']);
|
|
126
|
-
$advancedCriteriaContainer.empty();
|
|
127
|
-
_.forEach(criteriaState, criterion => {
|
|
128
|
-
criterion.rendered = false;
|
|
129
|
-
criterion.value = null;
|
|
130
|
-
});
|
|
131
|
-
},
|
|
132
|
-
/**
|
|
133
|
-
* Builds substring of search query with the advanced criteria conditions
|
|
134
|
-
*/
|
|
135
|
-
getAdvancedCriteriaQuery(hasSearchInput) {
|
|
136
|
-
const advancedSearchCriteria = _.filter(criteriaState, criterion => criterion.rendered === true);
|
|
137
|
-
let query = '';
|
|
138
|
-
|
|
139
|
-
advancedSearchCriteria.forEach(renderedCriterion => {
|
|
140
|
-
const queryParam = renderedCriterion.propertyUri;
|
|
141
|
-
if ((hasSearchInput || query.trim().length !== 0) && renderedCriterion.value) {
|
|
142
|
-
query += ' AND ';
|
|
143
|
-
}
|
|
144
|
-
if (renderedCriterion.type === criteriaTypes.text) {
|
|
145
|
-
if (renderedCriterion.value && renderedCriterion.value.trim() !== '') {
|
|
146
|
-
query += `${queryParam}:${renderedCriterion.value.trim()}`;
|
|
147
|
-
}
|
|
148
|
-
} else if (renderedCriterion.type === criteriaTypes.list) {
|
|
149
|
-
if (renderedCriterion.value && renderedCriterion.value.length > 0) {
|
|
150
|
-
/* Temp replaced OR with AND. See ADF-7 for details */
|
|
151
|
-
query += `${queryParam}:${renderedCriterion.value.join(' AND ')}`;
|
|
152
|
-
}
|
|
153
|
-
}
|
|
154
|
-
});
|
|
155
|
-
|
|
156
|
-
return query;
|
|
157
|
-
}
|
|
158
|
-
})
|
|
159
|
-
.setTemplate(advancedSearchTpl)
|
|
160
|
-
.on('render', () => {
|
|
161
|
-
initUiSelectors();
|
|
162
|
-
initAddCriteriaSelector()
|
|
163
|
-
.then(() => {
|
|
164
|
-
initCriteriaState();
|
|
165
|
-
instance.trigger('ready');
|
|
166
|
-
})
|
|
167
|
-
.catch(e => instance.trigger('error', e));
|
|
168
|
-
});
|
|
169
|
-
|
|
170
|
-
/**
|
|
171
|
-
* Inits template selectors and scroll animation
|
|
172
|
-
*/
|
|
173
|
-
function initUiSelectors() {
|
|
174
|
-
$container = instance.getElement();
|
|
175
|
-
$addCriteria = $('.add-criteria-container', $container);
|
|
176
|
-
$addCriteria.addClass('disabled');
|
|
177
|
-
$addCriteriaInput = $('.add-criteria-container a', $container);
|
|
178
|
-
$criteriaSelect = $('.add-criteria-container select', $container);
|
|
179
|
-
$advancedCriteriaContainer = $('.advanced-criteria-container', $container);
|
|
180
|
-
|
|
181
|
-
$advancedCriteriaContainer.on('scroll', _.throttle(animateScroll, 100));
|
|
182
|
-
}
|
|
183
|
-
|
|
184
|
-
/**
|
|
185
|
-
* Lookup for characters in text to highlight
|
|
186
|
-
* @param {String} text - text to lookup
|
|
187
|
-
* @param {String} searchString - match to be applied in the text
|
|
188
|
-
* @returns {String} - highlighted text
|
|
189
|
-
*/
|
|
190
|
-
function highlightCharacter(text, searchString) {
|
|
191
|
-
if (!searchString) {
|
|
192
|
-
return text;
|
|
193
|
-
}
|
|
194
|
-
const reg = new RegExp(searchString, 'gi');
|
|
195
|
-
return text.replace(reg, str => highlightedTextTpl({ text: str }));
|
|
196
|
-
}
|
|
197
|
-
|
|
198
|
-
/**
|
|
199
|
-
* Inits select2 on criteria select and its UX logic
|
|
200
|
-
*/
|
|
201
|
-
function initAddCriteriaSelector() {
|
|
202
|
-
return request(instance.config.statusUrl)
|
|
203
|
-
.then(function (response) {
|
|
204
|
-
if (
|
|
205
|
-
config.hideCriteria ||
|
|
206
|
-
!response.enabled ||
|
|
207
|
-
(response.whitelist && response.whitelist.includes(config.rootClassUri))
|
|
208
|
-
) {
|
|
209
|
-
isAdvancedSearchStatusEnabled = false;
|
|
210
|
-
return;
|
|
211
|
-
}
|
|
212
|
-
isAdvancedSearchStatusEnabled = true;
|
|
213
|
-
$addCriteria.removeClass('disabled');
|
|
214
|
-
$criteriaSelect.select2({
|
|
215
|
-
containerCssClass: 'criteria-select2',
|
|
216
|
-
dropdownCssClass: 'criteria-dropdown-select2',
|
|
217
|
-
sortResults: results => sortBy(results, 'text'),
|
|
218
|
-
escapeMarkup: function (markup) {
|
|
219
|
-
return markup;
|
|
220
|
-
},
|
|
221
|
-
formatResult: function formatResult(result, container, query) {
|
|
222
|
-
const label = result.element[0].getAttribute('label');
|
|
223
|
-
const alias = result.element[0].getAttribute('alias');
|
|
224
|
-
const classLabel = result.element[0].getAttribute('class-label');
|
|
225
|
-
|
|
226
|
-
let html = labelTpl({ text: highlightCharacter(label, query.term) });
|
|
227
|
-
|
|
228
|
-
if (alias) {
|
|
229
|
-
html += aliasTpl({ text: alias });
|
|
230
|
-
}
|
|
231
|
-
|
|
232
|
-
if (classLabel) {
|
|
233
|
-
html += classLabelTpl({ text: classLabel });
|
|
234
|
-
}
|
|
235
|
-
|
|
236
|
-
return html;
|
|
237
|
-
}
|
|
238
|
-
});
|
|
239
|
-
|
|
240
|
-
// open dropdown when user clicks on add criteria input
|
|
241
|
-
$addCriteriaInput.on('click', () => {
|
|
242
|
-
if (isCriteriaListUpdated) {
|
|
243
|
-
$criteriaSelect.select2('open');
|
|
244
|
-
// if dropdown is opened above addCriteria input, top property is slightly decreased to avoid overlapping with addCriteria icon
|
|
245
|
-
if ($('.criteria-dropdown-select2').hasClass('select2-drop-above')) {
|
|
246
|
-
$('.criteria-dropdown-select2').css(
|
|
247
|
-
'top',
|
|
248
|
-
$('.criteria-dropdown-select2').css('top').split('px')[0] - 10 + 'px'
|
|
249
|
-
);
|
|
250
|
-
}
|
|
251
|
-
}
|
|
252
|
-
});
|
|
253
|
-
|
|
254
|
-
// when a criterion is selected add it to criteria container, remove it from dropdown options and reset select
|
|
255
|
-
$criteriaSelect.on('change', () => {
|
|
256
|
-
const criterionToAdd = $criteriaSelect.children('option:selected').val();
|
|
257
|
-
addNewCriterion(criterionToAdd);
|
|
258
|
-
$criteriaSelect.children('option:selected').remove();
|
|
259
|
-
$criteriaSelect.select2('val', '');
|
|
260
|
-
});
|
|
261
|
-
})
|
|
262
|
-
.catch(function (e) {
|
|
263
|
-
return instance.trigger('error', e);
|
|
264
|
-
});
|
|
265
|
-
}
|
|
266
|
-
|
|
267
|
-
/**
|
|
268
|
-
* inits criteriaState loading it from the store (if present) or empty object.
|
|
269
|
-
* If there is a stored criteriaState, those criteria that were rendered
|
|
270
|
-
* but with null value are updated to not being rendered
|
|
271
|
-
*/
|
|
272
|
-
function initCriteriaState() {
|
|
273
|
-
if (instance.config.advancedCriteria) {
|
|
274
|
-
_.forEach(instance.config.advancedCriteria, criterion => {
|
|
275
|
-
if (criterion.rendered === true && criterion.value === null) {
|
|
276
|
-
criterion.rendered = false;
|
|
277
|
-
}
|
|
278
|
-
});
|
|
279
|
-
}
|
|
280
|
-
criteriaState = instance.config.advancedCriteria || {};
|
|
281
|
-
}
|
|
282
|
-
|
|
283
|
-
/**
|
|
284
|
-
* Styles scrolling on $advancedCriteriaContainer
|
|
285
|
-
*/
|
|
286
|
-
function animateScroll() {
|
|
287
|
-
const scrollPercentage =
|
|
288
|
-
$advancedCriteriaContainer.get(0).scrollTop /
|
|
289
|
-
($advancedCriteriaContainer.get(0).scrollHeight - $advancedCriteriaContainer.get(0).clientHeight);
|
|
290
|
-
if (scrollPercentage > 0.1) {
|
|
291
|
-
$advancedCriteriaContainer.addClass('scroll-separator-top');
|
|
292
|
-
} else {
|
|
293
|
-
$advancedCriteriaContainer.removeClass('scroll-separator-top');
|
|
294
|
-
}
|
|
295
|
-
if (scrollPercentage < 0.9) {
|
|
296
|
-
$advancedCriteriaContainer.addClass('scroll-separator-bottom');
|
|
297
|
-
} else {
|
|
298
|
-
$advancedCriteriaContainer.removeClass('scroll-separator-bottom');
|
|
299
|
-
}
|
|
300
|
-
}
|
|
301
|
-
|
|
302
|
-
/**
|
|
303
|
-
* Renders new criterion to criteria container so it can be used on advanced search filtering
|
|
304
|
-
* @param {string} criterionToAdd - new criterion to be added
|
|
305
|
-
*/
|
|
306
|
-
function addNewCriterion(criterionToAdd) {
|
|
307
|
-
// remove deprecated warning message
|
|
308
|
-
$('.invalid-criteria-warning-container').remove();
|
|
309
|
-
|
|
310
|
-
// render new criterion
|
|
311
|
-
const criterion = criteriaState[criterionToAdd];
|
|
312
|
-
const $criterionContainer = renderCriterion(criterion);
|
|
313
|
-
|
|
314
|
-
// set logic to remove criterion
|
|
315
|
-
$('.icon-result-nok', $criterionContainer).on('click', { criterion }, removeCriterion);
|
|
316
|
-
|
|
317
|
-
// set initial value and manage value changes
|
|
318
|
-
bindCriterionValue(criterion, $criterionContainer);
|
|
319
|
-
|
|
320
|
-
// update styles if scroll is enabled
|
|
321
|
-
if ($advancedCriteriaContainer.get(0).scrollHeight > $advancedCriteriaContainer.outerHeight()) {
|
|
322
|
-
$advancedCriteriaContainer.addClass('scrollable');
|
|
323
|
-
}
|
|
324
|
-
|
|
325
|
-
criterion.rendered = true;
|
|
326
|
-
}
|
|
327
|
-
|
|
328
|
-
/**
|
|
329
|
-
* Renders the new criterion selecting the appropiate handlebars template and prepending to advanced criteria container.
|
|
330
|
-
* If criterion is of type list with a uri endpoint to request the options, select2 is also init
|
|
331
|
-
* @param {object} criterion - criterion to render
|
|
332
|
-
* @returns - the rendered container
|
|
333
|
-
*/
|
|
334
|
-
function renderCriterion(criterion) {
|
|
335
|
-
let templateToUse = null;
|
|
336
|
-
if (criterion.type === criteriaTypes.text) {
|
|
337
|
-
templateToUse = textCriterionTpl;
|
|
338
|
-
} else if (criterion.type === criteriaTypes.list && criterion.uri) {
|
|
339
|
-
templateToUse = listSelectCriterionTpl;
|
|
340
|
-
} else {
|
|
341
|
-
templateToUse = listCheckboxCriterionTpl;
|
|
342
|
-
}
|
|
343
|
-
|
|
344
|
-
$advancedCriteriaContainer.append(templateToUse({ criterion }));
|
|
345
|
-
|
|
346
|
-
const $criterionContainer = $(`.${criterion.id}-filter`, $container);
|
|
347
|
-
const valueMapping = criteriaMapping[criterion.type];
|
|
348
|
-
|
|
349
|
-
/**
|
|
350
|
-
* On criterion of type list with a uri endpoint to retrieve options, template includes a select
|
|
351
|
-
* that is managed with select2, so we init it here
|
|
352
|
-
*/
|
|
353
|
-
if (criterion.type === criteriaTypes.list && criterion.uri) {
|
|
354
|
-
$(`input[name=${criterion.id}-select]`, $criterionContainer).select2({
|
|
355
|
-
multiple: true,
|
|
356
|
-
ajax: {
|
|
357
|
-
url: criterion.uri,
|
|
358
|
-
dataType: 'json',
|
|
359
|
-
data: function (term) {
|
|
360
|
-
return {
|
|
361
|
-
subject: term
|
|
362
|
-
};
|
|
363
|
-
},
|
|
364
|
-
results: response => ({
|
|
365
|
-
results: response.data.map(option => ({
|
|
366
|
-
id: valueMapping === 'uri' ? option.uri : option.label,
|
|
367
|
-
text: option.label
|
|
368
|
-
}))
|
|
369
|
-
})
|
|
370
|
-
},
|
|
371
|
-
initSelection: function (element, callback) {
|
|
372
|
-
const data = [];
|
|
373
|
-
$(element.val().split(',')).each(function () {
|
|
374
|
-
data.push({ id: this, text: this });
|
|
375
|
-
});
|
|
376
|
-
callback(data);
|
|
377
|
-
}
|
|
378
|
-
});
|
|
379
|
-
}
|
|
380
|
-
|
|
381
|
-
return $criterionContainer;
|
|
382
|
-
}
|
|
383
|
-
|
|
384
|
-
/**
|
|
385
|
-
* Fetches initial criterion label from api in case the value mapping is uri
|
|
386
|
-
* @param {object} criterion - criterion to be managed
|
|
387
|
-
*/
|
|
388
|
-
function getInitialCriterionLabel(criterion) {
|
|
389
|
-
const valueMapping = criteriaMapping[criterion.type];
|
|
390
|
-
if (valueMapping !== 'uri' || !criterion.value) {
|
|
391
|
-
return Promise.resolve({
|
|
392
|
-
id: criterion.value,
|
|
393
|
-
text: criterion.value
|
|
394
|
-
});
|
|
395
|
-
}
|
|
396
|
-
return $.ajax({
|
|
397
|
-
type: 'GET',
|
|
398
|
-
url: criterion.uri,
|
|
399
|
-
dataType: 'json'
|
|
400
|
-
}).then(({ data }) => {
|
|
401
|
-
if (Array.isArray(criterion.value)) {
|
|
402
|
-
return criterion.value.map(v => ({
|
|
403
|
-
id: v,
|
|
404
|
-
text: (data.find(d => d.uri === v) || {}).label
|
|
405
|
-
}));
|
|
406
|
-
}
|
|
407
|
-
let c = data.find(d => d.uri === criterion.value) || {};
|
|
408
|
-
return {
|
|
409
|
-
text: c.label,
|
|
410
|
-
id: criterion.value
|
|
411
|
-
};
|
|
412
|
-
});
|
|
413
|
-
}
|
|
414
|
-
|
|
415
|
-
/**
|
|
416
|
-
* Sets initial value for rendered criterion and sets binding between view and state
|
|
417
|
-
* @param {object} criterion - criterion to be managed
|
|
418
|
-
* @param {object} $criterionContainer - rendered criterion
|
|
419
|
-
*/
|
|
420
|
-
function bindCriterionValue(criterion, $criterionContainer) {
|
|
421
|
-
getInitialCriterionLabel(criterion).then(initialCriterion => {
|
|
422
|
-
if (criterion.type === criteriaTypes.text) {
|
|
423
|
-
// set initial value
|
|
424
|
-
$('input', $criterionContainer).val(criterion.value);
|
|
425
|
-
// set event to bind input value to critariaState
|
|
426
|
-
$('input', $criterionContainer).on('change', function () {
|
|
427
|
-
criterion.value = $(this).val() || null;
|
|
428
|
-
});
|
|
429
|
-
} else if (criterion.type === criteriaTypes.list && criterion.uri) {
|
|
430
|
-
// set initial value
|
|
431
|
-
if (criterion.value) {
|
|
432
|
-
$(`input[name=${criterion.id}-select]`, $criterionContainer).select2('data', initialCriterion);
|
|
433
|
-
}
|
|
434
|
-
// set event to bind input value to critariaState
|
|
435
|
-
$(`input[name=${criterion.id}-select]`, $criterionContainer).on('change', event => {
|
|
436
|
-
criterion.value = event.val;
|
|
437
|
-
});
|
|
438
|
-
} else {
|
|
439
|
-
// set initial value
|
|
440
|
-
if (criterion.value) {
|
|
441
|
-
criterion.value.forEach(selectedValue => {
|
|
442
|
-
$(`input[value=${selectedValue}]`, $criterionContainer).prop('checked', true);
|
|
443
|
-
});
|
|
444
|
-
}
|
|
445
|
-
// set event to bind input value to critariaState
|
|
446
|
-
$('input[type="checkbox"]', $criterionContainer).on('change', function () {
|
|
447
|
-
criterion.value = $(this)
|
|
448
|
-
.closest('.filter-container')
|
|
449
|
-
.find('input[type=checkbox]:checked')
|
|
450
|
-
.get()
|
|
451
|
-
.map(element => element.value);
|
|
452
|
-
});
|
|
453
|
-
}
|
|
454
|
-
});
|
|
455
|
-
}
|
|
456
|
-
|
|
457
|
-
/**
|
|
458
|
-
* Removes a criterion from advanced criteria container when user clicks on the criterion close icon.
|
|
459
|
-
* It also adds the option element to criteria select so removed criterion can be rendered again
|
|
460
|
-
* @param {object} event - click event triggered on closing icon
|
|
461
|
-
*/
|
|
462
|
-
function removeCriterion(event) {
|
|
463
|
-
const criterion = event.data.criterion;
|
|
464
|
-
const newOption = createCriteriaOption(criterion);
|
|
465
|
-
const criterionKey = getCriterionStateId(criterion);
|
|
466
|
-
|
|
467
|
-
// remove criterion and append new criterion to select options
|
|
468
|
-
$(this).parent().remove();
|
|
469
|
-
$criteriaSelect.append(newOption);
|
|
470
|
-
|
|
471
|
-
// reset criterion values on criteriaState
|
|
472
|
-
criteriaState[criterionKey].rendered = false;
|
|
473
|
-
criteriaState[criterionKey].value = null;
|
|
474
|
-
|
|
475
|
-
// check if advanced criteria container is no longer scrollable
|
|
476
|
-
if ($advancedCriteriaContainer.get(0).scrollHeight <= $advancedCriteriaContainer.outerHeight()) {
|
|
477
|
-
$advancedCriteriaContainer.removeClass('scrollable');
|
|
478
|
-
}
|
|
479
|
-
}
|
|
480
|
-
|
|
481
|
-
/**
|
|
482
|
-
* Parses received criteria from BE to the data structure required for criteria selector. To do so,
|
|
483
|
-
* appends every criterion into criteria array and then returns a duplicate-free version of it
|
|
484
|
-
* considering label property as uniqueness criterion
|
|
485
|
-
* @param {Array} classes - array of classes with the metadata (aka criteria) for each one of them
|
|
486
|
-
* @returns {Array} - criteria array
|
|
487
|
-
*/
|
|
488
|
-
function formatCriteria(classTree) {
|
|
489
|
-
let criteria = [];
|
|
490
|
-
|
|
491
|
-
_.forEach(classTree, classInstance => {
|
|
492
|
-
criteria.push(...classInstance.metadata);
|
|
493
|
-
});
|
|
494
|
-
|
|
495
|
-
// extends each criterion with an id that can be use as a valid css class
|
|
496
|
-
_.forEach(criteria, criterion => {
|
|
497
|
-
criterion.label = getCriterionLabel(criterion);
|
|
498
|
-
criterion.id = criterion.propertyUri.replace(/^[^a-zA-Z]*|[^a-zA-Z0-9]*/g, '');
|
|
499
|
-
});
|
|
500
|
-
|
|
501
|
-
return criteria;
|
|
502
|
-
}
|
|
503
|
-
|
|
504
|
-
/**
|
|
505
|
-
* Manages the new set of available criteria. To do so, removes warning container and every
|
|
506
|
-
* selectable criterion. Then removes from criteriaState and view every deprecated criterion,
|
|
507
|
-
* updates criteriaState with the new available criteria set, and renders new warning
|
|
508
|
-
* message if required
|
|
509
|
-
* @param {array} criteria - array of class properties
|
|
510
|
-
*/
|
|
511
|
-
function updateCriteria(criteria) {
|
|
512
|
-
$('.invalid-criteria-warning-container').remove();
|
|
513
|
-
$criteriaSelect.find('option:not(:first-child)').remove();
|
|
514
|
-
const invalidCriteria = deleteDeprecatedCriteria(criteria);
|
|
515
|
-
extendCriteria(criteria);
|
|
516
|
-
renderWarningMessage(invalidCriteria);
|
|
517
|
-
}
|
|
518
|
-
|
|
519
|
-
/**
|
|
520
|
-
* If there is any invalid criterion, renders an explanatory warning message
|
|
521
|
-
* @param {array} invalidCriteria - array of string containing the label of every invalid criterion (those that were rendered but are no longer available)
|
|
522
|
-
*/
|
|
523
|
-
function renderWarningMessage(invalidCriteria) {
|
|
524
|
-
if (invalidCriteria.length > 0) {
|
|
525
|
-
const invalidCriteriaWarning = invalidCriteriaWarningTpl({ invalidCriteria });
|
|
526
|
-
$advancedCriteriaContainer.prepend(invalidCriteriaWarning);
|
|
527
|
-
$('.invalid-criteria-warning-container .select2-search-choice-close', $advancedCriteriaContainer).on(
|
|
528
|
-
'click',
|
|
529
|
-
function () {
|
|
530
|
-
$(this).parent().remove();
|
|
531
|
-
}
|
|
532
|
-
);
|
|
533
|
-
}
|
|
534
|
-
}
|
|
535
|
-
|
|
536
|
-
/**
|
|
537
|
-
* Loops through current criteriaState so every criterion that is no longer available in the new
|
|
538
|
-
* available criteria set is removed from criteriaState and from view, in case it had been
|
|
539
|
-
* rendered. In that case it is also pushed into invalidCriteria array to be latter
|
|
540
|
-
* included in the warning message
|
|
541
|
-
* @param {array} criteria - array containing new set of criteria for current class
|
|
542
|
-
* @returns {array} - array of strings with each deprecated criterion that was being displayed
|
|
543
|
-
*/
|
|
544
|
-
function deleteDeprecatedCriteria(criteria) {
|
|
545
|
-
const invalidCriteria = [];
|
|
546
|
-
|
|
547
|
-
_.forEach(criteriaState, oldCriterion => {
|
|
548
|
-
const deprecatedCriterion = !criteria.find(newCriterion => newCriterion.label === oldCriterion.label);
|
|
549
|
-
const oldCriterionKey = getCriterionStateId(oldCriterion);
|
|
550
|
-
|
|
551
|
-
if (deprecatedCriterion) {
|
|
552
|
-
if (criteriaState[oldCriterionKey].rendered) {
|
|
553
|
-
$advancedCriteriaContainer.find(`.${oldCriterion.id}-filter`).remove();
|
|
554
|
-
invalidCriteria.push(oldCriterion.label);
|
|
555
|
-
}
|
|
556
|
-
delete criteriaState[oldCriterionKey];
|
|
557
|
-
}
|
|
558
|
-
});
|
|
559
|
-
|
|
560
|
-
return invalidCriteria;
|
|
561
|
-
}
|
|
562
|
-
|
|
563
|
-
/**
|
|
564
|
-
* Loops through new criteria set and checks if each new criterion was already present or not on criteriaState
|
|
565
|
-
* and updates view and selectable criteria list accordingly
|
|
566
|
-
* @param {array} criteria - array containing new set of criteria for current class
|
|
567
|
-
*/
|
|
568
|
-
function extendCriteria(criteria) {
|
|
569
|
-
criteria.forEach(criterion => {
|
|
570
|
-
let createOption = true;
|
|
571
|
-
const criteriaStateId = getCriterionStateId(criterion);
|
|
572
|
-
|
|
573
|
-
// if new criterion was already on criteriaState and had to be rendered, we avoid creating an option for it and render it if it was not
|
|
574
|
-
if (criteriaState[criteriaStateId] && criteriaState[criteriaStateId].rendered === true) {
|
|
575
|
-
createOption = false;
|
|
576
|
-
|
|
577
|
-
if ($advancedCriteriaContainer.find(`.${criterion.id}-filter`).length === 0) {
|
|
578
|
-
addNewCriterion(criteriaStateId);
|
|
579
|
-
}
|
|
580
|
-
} else {
|
|
581
|
-
// if new criterion was not on criteriaState we add it
|
|
582
|
-
criteriaState[criteriaStateId] = criterion;
|
|
583
|
-
criteriaState[criteriaStateId].rendered = false;
|
|
584
|
-
criteriaState[criteriaStateId].value = null;
|
|
585
|
-
}
|
|
586
|
-
|
|
587
|
-
// create new option element to criteria select
|
|
588
|
-
if (createOption) {
|
|
589
|
-
$criteriaSelect.append(createCriteriaOption(criterion));
|
|
590
|
-
}
|
|
591
|
-
});
|
|
592
|
-
}
|
|
593
|
-
|
|
594
|
-
/**
|
|
595
|
-
* Creates a new option element
|
|
596
|
-
* with attributes to use in select2 markup
|
|
597
|
-
* @param {Object} criterion
|
|
598
|
-
* @returns {HTMLOptionElement} Single option criteria
|
|
599
|
-
*/
|
|
600
|
-
function createCriteriaOption(criterion) {
|
|
601
|
-
const label = criterion.label;
|
|
602
|
-
let classLabel = '';
|
|
603
|
-
let alias = '';
|
|
604
|
-
let option;
|
|
605
|
-
|
|
606
|
-
if (criterion.isDuplicated) {
|
|
607
|
-
classLabel = criterion.class.label || '';
|
|
608
|
-
alias = criterion.alias || '';
|
|
609
|
-
}
|
|
610
|
-
|
|
611
|
-
option = new Option(label, getCriterionStateId(criterion), false, false);
|
|
612
|
-
|
|
613
|
-
option.setAttribute('label', label);
|
|
614
|
-
option.setAttribute('alias', alias);
|
|
615
|
-
option.setAttribute('class-label', classLabel);
|
|
616
|
-
|
|
617
|
-
return option;
|
|
618
|
-
}
|
|
619
|
-
|
|
620
|
-
/**
|
|
621
|
-
* @param {Object} criterion
|
|
622
|
-
* @returns String
|
|
623
|
-
*/
|
|
624
|
-
function getCriterionStateId(criterion) {
|
|
625
|
-
return criterion.propertyUri;
|
|
626
|
-
}
|
|
627
|
-
|
|
628
|
-
/**
|
|
629
|
-
* @param {Object} criterion
|
|
630
|
-
* @returns String
|
|
631
|
-
*/
|
|
632
|
-
function getCriterionLabel(criterion) {
|
|
633
|
-
return criterion.label;
|
|
634
|
-
}
|
|
635
|
-
|
|
636
|
-
// return initialized instance of searchModal
|
|
637
|
-
return instance.init(config);
|
|
638
|
-
}
|
|
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) 2021 (original work) Open Assessment Technologies SA ;
|
|
17
|
+
*/
|
|
18
|
+
|
|
19
|
+
import $ from 'jquery';
|
|
20
|
+
import _ from 'lodash';
|
|
21
|
+
import advancedSearchTpl from 'ui/searchModal/tpl/advanced-search';
|
|
22
|
+
import textCriterionTpl from 'ui/searchModal/tpl/text-criterion';
|
|
23
|
+
import invalidCriteriaWarningTpl from 'ui/searchModal/tpl/invalid-criteria-warning';
|
|
24
|
+
import listCheckboxCriterionTpl from 'ui/searchModal/tpl/list-checkbox-criterion';
|
|
25
|
+
import listSelectCriterionTpl from 'ui/searchModal/tpl/list-select-criterion';
|
|
26
|
+
import highlightedTextTpl from 'ui/searchModal/tpl/highlighted-text';
|
|
27
|
+
import classLabelTpl from 'ui/searchModal/tpl/criteria-class-label';
|
|
28
|
+
import aliasTpl from 'ui/searchModal/tpl/criteria-alias';
|
|
29
|
+
import labelTpl from 'ui/searchModal/tpl/criteria-label';
|
|
30
|
+
import 'ui/searchModal/css/advancedSearch.css';
|
|
31
|
+
import component from 'ui/component';
|
|
32
|
+
import 'ui/modal';
|
|
33
|
+
import 'ui/datatable';
|
|
34
|
+
import 'select2';
|
|
35
|
+
import request from 'core/dataProvider/request';
|
|
36
|
+
|
|
37
|
+
/**
|
|
38
|
+
* Sort an array by a particular property.
|
|
39
|
+
* @param {Array} iter - The array to sort.
|
|
40
|
+
* @param {string} prop - The name of the sorting property.
|
|
41
|
+
* @returns {Array} - Returns a sorted copy of the array.
|
|
42
|
+
* @private
|
|
43
|
+
*/
|
|
44
|
+
function sortBy(iter, prop) {
|
|
45
|
+
return Array.from(iter).sort((a, b) => {
|
|
46
|
+
const textA = (a && a[prop]) || '';
|
|
47
|
+
const textB = (b && b[prop]) || '';
|
|
48
|
+
return textA.localeCompare(textB);
|
|
49
|
+
});
|
|
50
|
+
}
|
|
51
|
+
|
|
52
|
+
/**
|
|
53
|
+
* Creates advanced search component
|
|
54
|
+
*
|
|
55
|
+
* @param {object} config
|
|
56
|
+
* @param {object} config.renderTo - DOM element where component will be rendered to
|
|
57
|
+
* @param {string} config.advancedCriteria - advanced criteria to be set on component creation
|
|
58
|
+
* @param {bool} config.hideCriteria - if the criteria must be hidden
|
|
59
|
+
* @param {string} config.rootClassUri - rootClassUri to check for whitelist sections
|
|
60
|
+
* @param {string} config.statusUrl - the URL to the status API (usually '/tao/AdvancedSearch/status')
|
|
61
|
+
* @returns {advancedSearch}
|
|
62
|
+
*/
|
|
63
|
+
export default function advancedSearchFactory(config) {
|
|
64
|
+
// Private properties to be easily accessible by instance methods
|
|
65
|
+
let $container = null;
|
|
66
|
+
let $addCriteria = null;
|
|
67
|
+
let $addCriteriaInput = null;
|
|
68
|
+
let $criteriaSelect = null;
|
|
69
|
+
let $advancedCriteriaContainer = null;
|
|
70
|
+
let criteriaState = null;
|
|
71
|
+
let criteriaMapping = {};
|
|
72
|
+
const criteriaTypes = {
|
|
73
|
+
text: 'text',
|
|
74
|
+
list: 'list'
|
|
75
|
+
};
|
|
76
|
+
|
|
77
|
+
let isAdvancedSearchStatusEnabled;
|
|
78
|
+
let isCriteriaListUpdated = false;
|
|
79
|
+
|
|
80
|
+
// Creates new component
|
|
81
|
+
const instance = component({
|
|
82
|
+
/**
|
|
83
|
+
* Tells if the advanced search is enabled.
|
|
84
|
+
* @returns {boolean}
|
|
85
|
+
*/
|
|
86
|
+
isEnabled() {
|
|
87
|
+
return !!isAdvancedSearchStatusEnabled;
|
|
88
|
+
},
|
|
89
|
+
|
|
90
|
+
/**
|
|
91
|
+
* Request metadata (criteria) for the given uri
|
|
92
|
+
* @param {string} classUri - url to make the reques to
|
|
93
|
+
* @returns {Promise} - Request promise
|
|
94
|
+
*/
|
|
95
|
+
updateCriteria(route) {
|
|
96
|
+
if (!isAdvancedSearchStatusEnabled) {
|
|
97
|
+
return Promise.resolve();
|
|
98
|
+
}
|
|
99
|
+
|
|
100
|
+
const $criteriaIcon = $('.add-criteria-container a span').eq(0);
|
|
101
|
+
$criteriaIcon.toggleClass('icon-add').toggleClass('icon-loop');
|
|
102
|
+
return request(route)
|
|
103
|
+
.then(response => {
|
|
104
|
+
criteriaMapping = response.criteriaMapping || {};
|
|
105
|
+
const classTree = response.classDefinition ? response.classDefinition : response;
|
|
106
|
+
const criteria = formatCriteria(classTree);
|
|
107
|
+
updateCriteria(criteria);
|
|
108
|
+
isCriteriaListUpdated = true;
|
|
109
|
+
$criteriaIcon.toggleClass('icon-add').toggleClass('icon-loop');
|
|
110
|
+
})
|
|
111
|
+
.catch(e => instance.trigger('error', e));
|
|
112
|
+
},
|
|
113
|
+
/**
|
|
114
|
+
* Access to component state
|
|
115
|
+
* @returns {Object} - criteria state
|
|
116
|
+
*/
|
|
117
|
+
getState() {
|
|
118
|
+
return criteriaState;
|
|
119
|
+
},
|
|
120
|
+
/**
|
|
121
|
+
* Removes every rendered criterion, updates criteria state accordingly
|
|
122
|
+
* and removes classes applied to scrollable list of criteria
|
|
123
|
+
*/
|
|
124
|
+
clear() {
|
|
125
|
+
$advancedCriteriaContainer.removeClass(['scrollable', 'scroll-separator-top', 'scroll-separator-bottom']);
|
|
126
|
+
$advancedCriteriaContainer.empty();
|
|
127
|
+
_.forEach(criteriaState, criterion => {
|
|
128
|
+
criterion.rendered = false;
|
|
129
|
+
criterion.value = null;
|
|
130
|
+
});
|
|
131
|
+
},
|
|
132
|
+
/**
|
|
133
|
+
* Builds substring of search query with the advanced criteria conditions
|
|
134
|
+
*/
|
|
135
|
+
getAdvancedCriteriaQuery(hasSearchInput) {
|
|
136
|
+
const advancedSearchCriteria = _.filter(criteriaState, criterion => criterion.rendered === true);
|
|
137
|
+
let query = '';
|
|
138
|
+
|
|
139
|
+
advancedSearchCriteria.forEach(renderedCriterion => {
|
|
140
|
+
const queryParam = renderedCriterion.propertyUri;
|
|
141
|
+
if ((hasSearchInput || query.trim().length !== 0) && renderedCriterion.value) {
|
|
142
|
+
query += ' AND ';
|
|
143
|
+
}
|
|
144
|
+
if (renderedCriterion.type === criteriaTypes.text) {
|
|
145
|
+
if (renderedCriterion.value && renderedCriterion.value.trim() !== '') {
|
|
146
|
+
query += `${queryParam}:${renderedCriterion.value.trim()}`;
|
|
147
|
+
}
|
|
148
|
+
} else if (renderedCriterion.type === criteriaTypes.list) {
|
|
149
|
+
if (renderedCriterion.value && renderedCriterion.value.length > 0) {
|
|
150
|
+
/* Temp replaced OR with AND. See ADF-7 for details */
|
|
151
|
+
query += `${queryParam}:${renderedCriterion.value.join(' AND ')}`;
|
|
152
|
+
}
|
|
153
|
+
}
|
|
154
|
+
});
|
|
155
|
+
|
|
156
|
+
return query;
|
|
157
|
+
}
|
|
158
|
+
})
|
|
159
|
+
.setTemplate(advancedSearchTpl)
|
|
160
|
+
.on('render', () => {
|
|
161
|
+
initUiSelectors();
|
|
162
|
+
initAddCriteriaSelector()
|
|
163
|
+
.then(() => {
|
|
164
|
+
initCriteriaState();
|
|
165
|
+
instance.trigger('ready');
|
|
166
|
+
})
|
|
167
|
+
.catch(e => instance.trigger('error', e));
|
|
168
|
+
});
|
|
169
|
+
|
|
170
|
+
/**
|
|
171
|
+
* Inits template selectors and scroll animation
|
|
172
|
+
*/
|
|
173
|
+
function initUiSelectors() {
|
|
174
|
+
$container = instance.getElement();
|
|
175
|
+
$addCriteria = $('.add-criteria-container', $container);
|
|
176
|
+
$addCriteria.addClass('disabled');
|
|
177
|
+
$addCriteriaInput = $('.add-criteria-container a', $container);
|
|
178
|
+
$criteriaSelect = $('.add-criteria-container select', $container);
|
|
179
|
+
$advancedCriteriaContainer = $('.advanced-criteria-container', $container);
|
|
180
|
+
|
|
181
|
+
$advancedCriteriaContainer.on('scroll', _.throttle(animateScroll, 100));
|
|
182
|
+
}
|
|
183
|
+
|
|
184
|
+
/**
|
|
185
|
+
* Lookup for characters in text to highlight
|
|
186
|
+
* @param {String} text - text to lookup
|
|
187
|
+
* @param {String} searchString - match to be applied in the text
|
|
188
|
+
* @returns {String} - highlighted text
|
|
189
|
+
*/
|
|
190
|
+
function highlightCharacter(text, searchString) {
|
|
191
|
+
if (!searchString) {
|
|
192
|
+
return text;
|
|
193
|
+
}
|
|
194
|
+
const reg = new RegExp(searchString, 'gi');
|
|
195
|
+
return text.replace(reg, str => highlightedTextTpl({ text: str }));
|
|
196
|
+
}
|
|
197
|
+
|
|
198
|
+
/**
|
|
199
|
+
* Inits select2 on criteria select and its UX logic
|
|
200
|
+
*/
|
|
201
|
+
function initAddCriteriaSelector() {
|
|
202
|
+
return request(instance.config.statusUrl)
|
|
203
|
+
.then(function (response) {
|
|
204
|
+
if (
|
|
205
|
+
config.hideCriteria ||
|
|
206
|
+
!response.enabled ||
|
|
207
|
+
(response.whitelist && response.whitelist.includes(config.rootClassUri))
|
|
208
|
+
) {
|
|
209
|
+
isAdvancedSearchStatusEnabled = false;
|
|
210
|
+
return;
|
|
211
|
+
}
|
|
212
|
+
isAdvancedSearchStatusEnabled = true;
|
|
213
|
+
$addCriteria.removeClass('disabled');
|
|
214
|
+
$criteriaSelect.select2({
|
|
215
|
+
containerCssClass: 'criteria-select2',
|
|
216
|
+
dropdownCssClass: 'criteria-dropdown-select2',
|
|
217
|
+
sortResults: results => sortBy(results, 'text'),
|
|
218
|
+
escapeMarkup: function (markup) {
|
|
219
|
+
return markup;
|
|
220
|
+
},
|
|
221
|
+
formatResult: function formatResult(result, container, query) {
|
|
222
|
+
const label = result.element[0].getAttribute('label');
|
|
223
|
+
const alias = result.element[0].getAttribute('alias');
|
|
224
|
+
const classLabel = result.element[0].getAttribute('class-label');
|
|
225
|
+
|
|
226
|
+
let html = labelTpl({ text: highlightCharacter(label, query.term) });
|
|
227
|
+
|
|
228
|
+
if (alias) {
|
|
229
|
+
html += aliasTpl({ text: alias });
|
|
230
|
+
}
|
|
231
|
+
|
|
232
|
+
if (classLabel) {
|
|
233
|
+
html += classLabelTpl({ text: classLabel });
|
|
234
|
+
}
|
|
235
|
+
|
|
236
|
+
return html;
|
|
237
|
+
}
|
|
238
|
+
});
|
|
239
|
+
|
|
240
|
+
// open dropdown when user clicks on add criteria input
|
|
241
|
+
$addCriteriaInput.on('click', () => {
|
|
242
|
+
if (isCriteriaListUpdated) {
|
|
243
|
+
$criteriaSelect.select2('open');
|
|
244
|
+
// if dropdown is opened above addCriteria input, top property is slightly decreased to avoid overlapping with addCriteria icon
|
|
245
|
+
if ($('.criteria-dropdown-select2').hasClass('select2-drop-above')) {
|
|
246
|
+
$('.criteria-dropdown-select2').css(
|
|
247
|
+
'top',
|
|
248
|
+
$('.criteria-dropdown-select2').css('top').split('px')[0] - 10 + 'px'
|
|
249
|
+
);
|
|
250
|
+
}
|
|
251
|
+
}
|
|
252
|
+
});
|
|
253
|
+
|
|
254
|
+
// when a criterion is selected add it to criteria container, remove it from dropdown options and reset select
|
|
255
|
+
$criteriaSelect.on('change', () => {
|
|
256
|
+
const criterionToAdd = $criteriaSelect.children('option:selected').val();
|
|
257
|
+
addNewCriterion(criterionToAdd);
|
|
258
|
+
$criteriaSelect.children('option:selected').remove();
|
|
259
|
+
$criteriaSelect.select2('val', '');
|
|
260
|
+
});
|
|
261
|
+
})
|
|
262
|
+
.catch(function (e) {
|
|
263
|
+
return instance.trigger('error', e);
|
|
264
|
+
});
|
|
265
|
+
}
|
|
266
|
+
|
|
267
|
+
/**
|
|
268
|
+
* inits criteriaState loading it from the store (if present) or empty object.
|
|
269
|
+
* If there is a stored criteriaState, those criteria that were rendered
|
|
270
|
+
* but with null value are updated to not being rendered
|
|
271
|
+
*/
|
|
272
|
+
function initCriteriaState() {
|
|
273
|
+
if (instance.config.advancedCriteria) {
|
|
274
|
+
_.forEach(instance.config.advancedCriteria, criterion => {
|
|
275
|
+
if (criterion.rendered === true && criterion.value === null) {
|
|
276
|
+
criterion.rendered = false;
|
|
277
|
+
}
|
|
278
|
+
});
|
|
279
|
+
}
|
|
280
|
+
criteriaState = instance.config.advancedCriteria || {};
|
|
281
|
+
}
|
|
282
|
+
|
|
283
|
+
/**
|
|
284
|
+
* Styles scrolling on $advancedCriteriaContainer
|
|
285
|
+
*/
|
|
286
|
+
function animateScroll() {
|
|
287
|
+
const scrollPercentage =
|
|
288
|
+
$advancedCriteriaContainer.get(0).scrollTop /
|
|
289
|
+
($advancedCriteriaContainer.get(0).scrollHeight - $advancedCriteriaContainer.get(0).clientHeight);
|
|
290
|
+
if (scrollPercentage > 0.1) {
|
|
291
|
+
$advancedCriteriaContainer.addClass('scroll-separator-top');
|
|
292
|
+
} else {
|
|
293
|
+
$advancedCriteriaContainer.removeClass('scroll-separator-top');
|
|
294
|
+
}
|
|
295
|
+
if (scrollPercentage < 0.9) {
|
|
296
|
+
$advancedCriteriaContainer.addClass('scroll-separator-bottom');
|
|
297
|
+
} else {
|
|
298
|
+
$advancedCriteriaContainer.removeClass('scroll-separator-bottom');
|
|
299
|
+
}
|
|
300
|
+
}
|
|
301
|
+
|
|
302
|
+
/**
|
|
303
|
+
* Renders new criterion to criteria container so it can be used on advanced search filtering
|
|
304
|
+
* @param {string} criterionToAdd - new criterion to be added
|
|
305
|
+
*/
|
|
306
|
+
function addNewCriterion(criterionToAdd) {
|
|
307
|
+
// remove deprecated warning message
|
|
308
|
+
$('.invalid-criteria-warning-container').remove();
|
|
309
|
+
|
|
310
|
+
// render new criterion
|
|
311
|
+
const criterion = criteriaState[criterionToAdd];
|
|
312
|
+
const $criterionContainer = renderCriterion(criterion);
|
|
313
|
+
|
|
314
|
+
// set logic to remove criterion
|
|
315
|
+
$('.icon-result-nok', $criterionContainer).on('click', { criterion }, removeCriterion);
|
|
316
|
+
|
|
317
|
+
// set initial value and manage value changes
|
|
318
|
+
bindCriterionValue(criterion, $criterionContainer);
|
|
319
|
+
|
|
320
|
+
// update styles if scroll is enabled
|
|
321
|
+
if ($advancedCriteriaContainer.get(0).scrollHeight > $advancedCriteriaContainer.outerHeight()) {
|
|
322
|
+
$advancedCriteriaContainer.addClass('scrollable');
|
|
323
|
+
}
|
|
324
|
+
|
|
325
|
+
criterion.rendered = true;
|
|
326
|
+
}
|
|
327
|
+
|
|
328
|
+
/**
|
|
329
|
+
* Renders the new criterion selecting the appropiate handlebars template and prepending to advanced criteria container.
|
|
330
|
+
* If criterion is of type list with a uri endpoint to request the options, select2 is also init
|
|
331
|
+
* @param {object} criterion - criterion to render
|
|
332
|
+
* @returns - the rendered container
|
|
333
|
+
*/
|
|
334
|
+
function renderCriterion(criterion) {
|
|
335
|
+
let templateToUse = null;
|
|
336
|
+
if (criterion.type === criteriaTypes.text) {
|
|
337
|
+
templateToUse = textCriterionTpl;
|
|
338
|
+
} else if (criterion.type === criteriaTypes.list && criterion.uri) {
|
|
339
|
+
templateToUse = listSelectCriterionTpl;
|
|
340
|
+
} else {
|
|
341
|
+
templateToUse = listCheckboxCriterionTpl;
|
|
342
|
+
}
|
|
343
|
+
|
|
344
|
+
$advancedCriteriaContainer.append(templateToUse({ criterion }));
|
|
345
|
+
|
|
346
|
+
const $criterionContainer = $(`.${criterion.id}-filter`, $container);
|
|
347
|
+
const valueMapping = criteriaMapping[criterion.type];
|
|
348
|
+
|
|
349
|
+
/**
|
|
350
|
+
* On criterion of type list with a uri endpoint to retrieve options, template includes a select
|
|
351
|
+
* that is managed with select2, so we init it here
|
|
352
|
+
*/
|
|
353
|
+
if (criterion.type === criteriaTypes.list && criterion.uri) {
|
|
354
|
+
$(`input[name=${criterion.id}-select]`, $criterionContainer).select2({
|
|
355
|
+
multiple: true,
|
|
356
|
+
ajax: {
|
|
357
|
+
url: criterion.uri,
|
|
358
|
+
dataType: 'json',
|
|
359
|
+
data: function (term) {
|
|
360
|
+
return {
|
|
361
|
+
subject: term
|
|
362
|
+
};
|
|
363
|
+
},
|
|
364
|
+
results: response => ({
|
|
365
|
+
results: response.data.map(option => ({
|
|
366
|
+
id: valueMapping === 'uri' ? option.uri : option.label,
|
|
367
|
+
text: option.label
|
|
368
|
+
}))
|
|
369
|
+
})
|
|
370
|
+
},
|
|
371
|
+
initSelection: function (element, callback) {
|
|
372
|
+
const data = [];
|
|
373
|
+
$(element.val().split(',')).each(function () {
|
|
374
|
+
data.push({ id: this, text: this });
|
|
375
|
+
});
|
|
376
|
+
callback(data);
|
|
377
|
+
}
|
|
378
|
+
});
|
|
379
|
+
}
|
|
380
|
+
|
|
381
|
+
return $criterionContainer;
|
|
382
|
+
}
|
|
383
|
+
|
|
384
|
+
/**
|
|
385
|
+
* Fetches initial criterion label from api in case the value mapping is uri
|
|
386
|
+
* @param {object} criterion - criterion to be managed
|
|
387
|
+
*/
|
|
388
|
+
function getInitialCriterionLabel(criterion) {
|
|
389
|
+
const valueMapping = criteriaMapping[criterion.type];
|
|
390
|
+
if (valueMapping !== 'uri' || !criterion.value) {
|
|
391
|
+
return Promise.resolve({
|
|
392
|
+
id: criterion.value,
|
|
393
|
+
text: criterion.value
|
|
394
|
+
});
|
|
395
|
+
}
|
|
396
|
+
return $.ajax({
|
|
397
|
+
type: 'GET',
|
|
398
|
+
url: criterion.uri,
|
|
399
|
+
dataType: 'json'
|
|
400
|
+
}).then(({ data }) => {
|
|
401
|
+
if (Array.isArray(criterion.value)) {
|
|
402
|
+
return criterion.value.map(v => ({
|
|
403
|
+
id: v,
|
|
404
|
+
text: (data.find(d => d.uri === v) || {}).label
|
|
405
|
+
}));
|
|
406
|
+
}
|
|
407
|
+
let c = data.find(d => d.uri === criterion.value) || {};
|
|
408
|
+
return {
|
|
409
|
+
text: c.label,
|
|
410
|
+
id: criterion.value
|
|
411
|
+
};
|
|
412
|
+
});
|
|
413
|
+
}
|
|
414
|
+
|
|
415
|
+
/**
|
|
416
|
+
* Sets initial value for rendered criterion and sets binding between view and state
|
|
417
|
+
* @param {object} criterion - criterion to be managed
|
|
418
|
+
* @param {object} $criterionContainer - rendered criterion
|
|
419
|
+
*/
|
|
420
|
+
function bindCriterionValue(criterion, $criterionContainer) {
|
|
421
|
+
getInitialCriterionLabel(criterion).then(initialCriterion => {
|
|
422
|
+
if (criterion.type === criteriaTypes.text) {
|
|
423
|
+
// set initial value
|
|
424
|
+
$('input', $criterionContainer).val(criterion.value);
|
|
425
|
+
// set event to bind input value to critariaState
|
|
426
|
+
$('input', $criterionContainer).on('change', function () {
|
|
427
|
+
criterion.value = $(this).val() || null;
|
|
428
|
+
});
|
|
429
|
+
} else if (criterion.type === criteriaTypes.list && criterion.uri) {
|
|
430
|
+
// set initial value
|
|
431
|
+
if (criterion.value) {
|
|
432
|
+
$(`input[name=${criterion.id}-select]`, $criterionContainer).select2('data', initialCriterion);
|
|
433
|
+
}
|
|
434
|
+
// set event to bind input value to critariaState
|
|
435
|
+
$(`input[name=${criterion.id}-select]`, $criterionContainer).on('change', event => {
|
|
436
|
+
criterion.value = event.val;
|
|
437
|
+
});
|
|
438
|
+
} else {
|
|
439
|
+
// set initial value
|
|
440
|
+
if (criterion.value) {
|
|
441
|
+
criterion.value.forEach(selectedValue => {
|
|
442
|
+
$(`input[value=${selectedValue}]`, $criterionContainer).prop('checked', true);
|
|
443
|
+
});
|
|
444
|
+
}
|
|
445
|
+
// set event to bind input value to critariaState
|
|
446
|
+
$('input[type="checkbox"]', $criterionContainer).on('change', function () {
|
|
447
|
+
criterion.value = $(this)
|
|
448
|
+
.closest('.filter-container')
|
|
449
|
+
.find('input[type=checkbox]:checked')
|
|
450
|
+
.get()
|
|
451
|
+
.map(element => element.value);
|
|
452
|
+
});
|
|
453
|
+
}
|
|
454
|
+
});
|
|
455
|
+
}
|
|
456
|
+
|
|
457
|
+
/**
|
|
458
|
+
* Removes a criterion from advanced criteria container when user clicks on the criterion close icon.
|
|
459
|
+
* It also adds the option element to criteria select so removed criterion can be rendered again
|
|
460
|
+
* @param {object} event - click event triggered on closing icon
|
|
461
|
+
*/
|
|
462
|
+
function removeCriterion(event) {
|
|
463
|
+
const criterion = event.data.criterion;
|
|
464
|
+
const newOption = createCriteriaOption(criterion);
|
|
465
|
+
const criterionKey = getCriterionStateId(criterion);
|
|
466
|
+
|
|
467
|
+
// remove criterion and append new criterion to select options
|
|
468
|
+
$(this).parent().remove();
|
|
469
|
+
$criteriaSelect.append(newOption);
|
|
470
|
+
|
|
471
|
+
// reset criterion values on criteriaState
|
|
472
|
+
criteriaState[criterionKey].rendered = false;
|
|
473
|
+
criteriaState[criterionKey].value = null;
|
|
474
|
+
|
|
475
|
+
// check if advanced criteria container is no longer scrollable
|
|
476
|
+
if ($advancedCriteriaContainer.get(0).scrollHeight <= $advancedCriteriaContainer.outerHeight()) {
|
|
477
|
+
$advancedCriteriaContainer.removeClass('scrollable');
|
|
478
|
+
}
|
|
479
|
+
}
|
|
480
|
+
|
|
481
|
+
/**
|
|
482
|
+
* Parses received criteria from BE to the data structure required for criteria selector. To do so,
|
|
483
|
+
* appends every criterion into criteria array and then returns a duplicate-free version of it
|
|
484
|
+
* considering label property as uniqueness criterion
|
|
485
|
+
* @param {Array} classes - array of classes with the metadata (aka criteria) for each one of them
|
|
486
|
+
* @returns {Array} - criteria array
|
|
487
|
+
*/
|
|
488
|
+
function formatCriteria(classTree) {
|
|
489
|
+
let criteria = [];
|
|
490
|
+
|
|
491
|
+
_.forEach(classTree, classInstance => {
|
|
492
|
+
criteria.push(...classInstance.metadata);
|
|
493
|
+
});
|
|
494
|
+
|
|
495
|
+
// extends each criterion with an id that can be use as a valid css class
|
|
496
|
+
_.forEach(criteria, criterion => {
|
|
497
|
+
criterion.label = getCriterionLabel(criterion);
|
|
498
|
+
criterion.id = criterion.propertyUri.replace(/^[^a-zA-Z]*|[^a-zA-Z0-9]*/g, '');
|
|
499
|
+
});
|
|
500
|
+
|
|
501
|
+
return criteria;
|
|
502
|
+
}
|
|
503
|
+
|
|
504
|
+
/**
|
|
505
|
+
* Manages the new set of available criteria. To do so, removes warning container and every
|
|
506
|
+
* selectable criterion. Then removes from criteriaState and view every deprecated criterion,
|
|
507
|
+
* updates criteriaState with the new available criteria set, and renders new warning
|
|
508
|
+
* message if required
|
|
509
|
+
* @param {array} criteria - array of class properties
|
|
510
|
+
*/
|
|
511
|
+
function updateCriteria(criteria) {
|
|
512
|
+
$('.invalid-criteria-warning-container').remove();
|
|
513
|
+
$criteriaSelect.find('option:not(:first-child)').remove();
|
|
514
|
+
const invalidCriteria = deleteDeprecatedCriteria(criteria);
|
|
515
|
+
extendCriteria(criteria);
|
|
516
|
+
renderWarningMessage(invalidCriteria);
|
|
517
|
+
}
|
|
518
|
+
|
|
519
|
+
/**
|
|
520
|
+
* If there is any invalid criterion, renders an explanatory warning message
|
|
521
|
+
* @param {array} invalidCriteria - array of string containing the label of every invalid criterion (those that were rendered but are no longer available)
|
|
522
|
+
*/
|
|
523
|
+
function renderWarningMessage(invalidCriteria) {
|
|
524
|
+
if (invalidCriteria.length > 0) {
|
|
525
|
+
const invalidCriteriaWarning = invalidCriteriaWarningTpl({ invalidCriteria });
|
|
526
|
+
$advancedCriteriaContainer.prepend(invalidCriteriaWarning);
|
|
527
|
+
$('.invalid-criteria-warning-container .select2-search-choice-close', $advancedCriteriaContainer).on(
|
|
528
|
+
'click',
|
|
529
|
+
function () {
|
|
530
|
+
$(this).parent().remove();
|
|
531
|
+
}
|
|
532
|
+
);
|
|
533
|
+
}
|
|
534
|
+
}
|
|
535
|
+
|
|
536
|
+
/**
|
|
537
|
+
* Loops through current criteriaState so every criterion that is no longer available in the new
|
|
538
|
+
* available criteria set is removed from criteriaState and from view, in case it had been
|
|
539
|
+
* rendered. In that case it is also pushed into invalidCriteria array to be latter
|
|
540
|
+
* included in the warning message
|
|
541
|
+
* @param {array} criteria - array containing new set of criteria for current class
|
|
542
|
+
* @returns {array} - array of strings with each deprecated criterion that was being displayed
|
|
543
|
+
*/
|
|
544
|
+
function deleteDeprecatedCriteria(criteria) {
|
|
545
|
+
const invalidCriteria = [];
|
|
546
|
+
|
|
547
|
+
_.forEach(criteriaState, oldCriterion => {
|
|
548
|
+
const deprecatedCriterion = !criteria.find(newCriterion => newCriterion.label === oldCriterion.label);
|
|
549
|
+
const oldCriterionKey = getCriterionStateId(oldCriterion);
|
|
550
|
+
|
|
551
|
+
if (deprecatedCriterion) {
|
|
552
|
+
if (criteriaState[oldCriterionKey].rendered) {
|
|
553
|
+
$advancedCriteriaContainer.find(`.${oldCriterion.id}-filter`).remove();
|
|
554
|
+
invalidCriteria.push(oldCriterion.label);
|
|
555
|
+
}
|
|
556
|
+
delete criteriaState[oldCriterionKey];
|
|
557
|
+
}
|
|
558
|
+
});
|
|
559
|
+
|
|
560
|
+
return invalidCriteria;
|
|
561
|
+
}
|
|
562
|
+
|
|
563
|
+
/**
|
|
564
|
+
* Loops through new criteria set and checks if each new criterion was already present or not on criteriaState
|
|
565
|
+
* and updates view and selectable criteria list accordingly
|
|
566
|
+
* @param {array} criteria - array containing new set of criteria for current class
|
|
567
|
+
*/
|
|
568
|
+
function extendCriteria(criteria) {
|
|
569
|
+
criteria.forEach(criterion => {
|
|
570
|
+
let createOption = true;
|
|
571
|
+
const criteriaStateId = getCriterionStateId(criterion);
|
|
572
|
+
|
|
573
|
+
// if new criterion was already on criteriaState and had to be rendered, we avoid creating an option for it and render it if it was not
|
|
574
|
+
if (criteriaState[criteriaStateId] && criteriaState[criteriaStateId].rendered === true) {
|
|
575
|
+
createOption = false;
|
|
576
|
+
|
|
577
|
+
if ($advancedCriteriaContainer.find(`.${criterion.id}-filter`).length === 0) {
|
|
578
|
+
addNewCriterion(criteriaStateId);
|
|
579
|
+
}
|
|
580
|
+
} else {
|
|
581
|
+
// if new criterion was not on criteriaState we add it
|
|
582
|
+
criteriaState[criteriaStateId] = criterion;
|
|
583
|
+
criteriaState[criteriaStateId].rendered = false;
|
|
584
|
+
criteriaState[criteriaStateId].value = null;
|
|
585
|
+
}
|
|
586
|
+
|
|
587
|
+
// create new option element to criteria select
|
|
588
|
+
if (createOption) {
|
|
589
|
+
$criteriaSelect.append(createCriteriaOption(criterion));
|
|
590
|
+
}
|
|
591
|
+
});
|
|
592
|
+
}
|
|
593
|
+
|
|
594
|
+
/**
|
|
595
|
+
* Creates a new option element
|
|
596
|
+
* with attributes to use in select2 markup
|
|
597
|
+
* @param {Object} criterion
|
|
598
|
+
* @returns {HTMLOptionElement} Single option criteria
|
|
599
|
+
*/
|
|
600
|
+
function createCriteriaOption(criterion) {
|
|
601
|
+
const label = criterion.label;
|
|
602
|
+
let classLabel = '';
|
|
603
|
+
let alias = '';
|
|
604
|
+
let option;
|
|
605
|
+
|
|
606
|
+
if (criterion.isDuplicated) {
|
|
607
|
+
classLabel = criterion.class.label || '';
|
|
608
|
+
alias = criterion.alias || '';
|
|
609
|
+
}
|
|
610
|
+
|
|
611
|
+
option = new Option(label, getCriterionStateId(criterion), false, false);
|
|
612
|
+
|
|
613
|
+
option.setAttribute('label', label);
|
|
614
|
+
option.setAttribute('alias', alias);
|
|
615
|
+
option.setAttribute('class-label', classLabel);
|
|
616
|
+
|
|
617
|
+
return option;
|
|
618
|
+
}
|
|
619
|
+
|
|
620
|
+
/**
|
|
621
|
+
* @param {Object} criterion
|
|
622
|
+
* @returns String
|
|
623
|
+
*/
|
|
624
|
+
function getCriterionStateId(criterion) {
|
|
625
|
+
return criterion.propertyUri;
|
|
626
|
+
}
|
|
627
|
+
|
|
628
|
+
/**
|
|
629
|
+
* @param {Object} criterion
|
|
630
|
+
* @returns String
|
|
631
|
+
*/
|
|
632
|
+
function getCriterionLabel(criterion) {
|
|
633
|
+
return criterion.label;
|
|
634
|
+
}
|
|
635
|
+
|
|
636
|
+
// return initialized instance of searchModal
|
|
637
|
+
return instance.init(config);
|
|
638
|
+
}
|