@watermarkinsights/ripple 3.13.0 → 3.14.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +3 -3
- package/dist/cjs/{chartFunctions-34fdd3ce.js → chartFunctions-0ea0888f.js} +420 -420
- package/dist/cjs/{functions-120449cf.js → functions-8d3ceccd.js} +1894 -450
- package/dist/cjs/{global-c1940f7b.js → global-39ada944.js} +62 -62
- package/dist/cjs/{http-service-494d81de.js → http-service-9e8c4dd5.js} +49 -49
- package/dist/cjs/{index-acad7ab3.js → index-df6a3611.js} +4 -1
- package/dist/cjs/{interfaces-a3338581.js → interfaces-30a74c1f.js} +29 -29
- package/dist/cjs/loader.cjs.js +4 -4
- package/dist/cjs/priv-chart-popover.cjs.entry.js +87 -87
- package/dist/cjs/priv-datepicker.cjs.entry.js +658 -658
- package/dist/cjs/priv-navigator-button.cjs.entry.js +19 -19
- package/dist/cjs/priv-navigator-item.cjs.entry.js +24 -24
- package/dist/cjs/ripple.cjs.js +4 -4
- package/dist/cjs/wm-action-menu_2.cjs.entry.js +336 -336
- package/dist/cjs/wm-button.cjs.entry.js +242 -242
- package/dist/cjs/wm-chart-slice.cjs.entry.js +14 -14
- package/dist/cjs/wm-chart.cjs.entry.js +111 -111
- package/dist/cjs/wm-datepicker.cjs.entry.js +266 -266
- package/dist/cjs/wm-input.cjs.entry.js +135 -135
- package/dist/cjs/wm-modal-footer.cjs.entry.js +34 -34
- package/dist/cjs/wm-modal-header.cjs.entry.js +37 -37
- package/dist/cjs/wm-modal.cjs.entry.js +164 -164
- package/dist/cjs/wm-navigation_3.cjs.entry.js +228 -228
- package/dist/cjs/wm-navigator.cjs.entry.js +269 -269
- package/dist/cjs/wm-network-uploader.cjs.entry.js +455 -455
- package/dist/cjs/wm-option_2.cjs.entry.js +717 -710
- package/dist/cjs/wm-pagination.cjs.entry.js +168 -168
- package/dist/cjs/wm-progress-indicator_3.cjs.entry.js +130 -130
- package/dist/cjs/wm-search.cjs.entry.js +233 -233
- package/dist/cjs/wm-snackbar.cjs.entry.js +171 -171
- package/dist/cjs/wm-tab-item_3.cjs.entry.js +297 -297
- package/dist/cjs/wm-tag-input.cjs.entry.js +580 -580
- package/dist/cjs/wm-timepicker.cjs.entry.js +384 -384
- package/dist/cjs/wm-toggletip.cjs.entry.js +125 -125
- package/dist/cjs/wm-uploader.cjs.entry.js +630 -632
- package/dist/cjs/wm-wrapper.cjs.entry.js +13 -13
- package/dist/collection/collection-manifest.json +1 -1
- package/dist/collection/components/charts/priv-chart-popover/priv-chart-popover.js +258 -258
- package/dist/collection/components/charts/wm-chart/wm-chart-slice.js +115 -115
- package/dist/collection/components/charts/wm-chart/wm-chart.js +328 -328
- package/dist/collection/components/charts/wm-progress-monitor/wm-progress-indicator.js +195 -195
- package/dist/collection/components/charts/wm-progress-monitor/wm-progress-monitor.js +88 -88
- package/dist/collection/components/charts/wm-progress-monitor/wm-progress-slice.js +111 -111
- package/dist/collection/components/wm-action-menu/wm-action-menu.js +448 -448
- package/dist/collection/components/wm-button/wm-button.js +504 -504
- package/dist/collection/components/wm-datepicker/priv-datepicker/priv-datepicker.js +1024 -1024
- package/dist/collection/components/wm-datepicker/wm-datepicker.js +471 -471
- package/dist/collection/components/wm-input/wm-input.js +423 -423
- package/dist/collection/components/wm-menuitem/wm-menuitem.js +444 -444
- package/dist/collection/components/wm-modal/wm-modal-footer.js +136 -136
- package/dist/collection/components/wm-modal/wm-modal-header.js +87 -87
- package/dist/collection/components/wm-modal/wm-modal.js +461 -461
- package/dist/collection/components/wm-navigation/wm-navigation-hamburger.js +162 -162
- package/dist/collection/components/wm-navigation/wm-navigation-item.js +121 -121
- package/dist/collection/components/wm-navigation/wm-navigation.js +218 -218
- package/dist/collection/components/wm-navigator/priv-navigator-button/priv-navigator-button.js +97 -97
- package/dist/collection/components/wm-navigator/priv-navigator-item/priv-navigator-item.js +114 -114
- package/dist/collection/components/wm-navigator/wm-navigator.js +468 -468
- package/dist/collection/components/wm-option/wm-option.css +13 -0
- package/dist/collection/components/wm-option/wm-option.js +427 -419
- package/dist/collection/components/wm-pagination/wm-pagination.js +368 -368
- package/dist/collection/components/wm-search/wm-search.js +480 -480
- package/dist/collection/components/wm-select/wm-select.css +5 -2
- package/dist/collection/components/wm-select/wm-select.js +978 -978
- package/dist/collection/components/wm-snackbar/wm-snackbar.js +297 -297
- package/dist/collection/components/wm-tabs/wm-tab-item/wm-tab-item.js +216 -216
- package/dist/collection/components/wm-tabs/wm-tab-list/wm-tab-list.js +330 -330
- package/dist/collection/components/wm-tabs/wm-tab-panel/wm-tab-panel.js +104 -104
- package/dist/collection/components/wm-tag-input/wm-tag-input.js +810 -810
- package/dist/collection/components/wm-timepicker/wm-timepicker.js +583 -583
- package/dist/collection/components/wm-toggletip/wm-toggletip.js +241 -241
- package/dist/collection/components/wm-uploader/wm-network-uploader/wm-network-uploader.js +787 -787
- package/dist/collection/components/wm-uploader/wm-uploader.js +1072 -1077
- package/dist/collection/components/wm-wrapper/wm-wrapper.js +27 -27
- package/dist/collection/dev/scripts.js +20 -20
- package/dist/collection/global/__mocks__/functions.js +6 -6
- package/dist/collection/global/chartFunctions.js +421 -421
- package/dist/collection/global/functions.js +441 -441
- package/dist/collection/global/global.js +69 -69
- package/dist/collection/global/interfaces.js +49 -49
- package/dist/collection/global/services/__mocks__/http-service.js +130 -130
- package/dist/collection/global/services/http-service.js +50 -50
- package/dist/collection/lang/lang.js +5 -5
- package/dist/collection/lang/missing.js +43 -43
- package/dist/collection/lang/piglatin.js +93 -93
- package/dist/esm/{chartFunctions-20f05eb5.js → chartFunctions-6f90586d.js} +420 -420
- package/dist/esm/{functions-036af8dc.js → functions-11c0c4e2.js} +1894 -450
- package/dist/esm/{global-e98c740f.js → global-d36a84c5.js} +62 -62
- package/dist/esm/{http-service-3dc3b3e7.js → http-service-5d037e16.js} +49 -49
- package/dist/esm/{index-7603f98e.js → index-17c79cc6.js} +4 -1
- package/dist/esm/{interfaces-2b97fab2.js → interfaces-61c6305b.js} +29 -29
- package/dist/esm/loader.js +4 -4
- package/dist/esm/polyfills/core-js.js +0 -0
- package/dist/esm/polyfills/css-shim.js +1 -1
- package/dist/esm/polyfills/dom.js +0 -0
- package/dist/esm/polyfills/es5-html-element.js +0 -0
- package/dist/esm/polyfills/index.js +0 -0
- package/dist/esm/polyfills/system.js +0 -0
- package/dist/esm/priv-chart-popover.entry.js +87 -87
- package/dist/esm/priv-datepicker.entry.js +658 -658
- package/dist/esm/priv-navigator-button.entry.js +19 -19
- package/dist/esm/priv-navigator-item.entry.js +24 -24
- package/dist/esm/ripple.js +4 -4
- package/dist/esm/wm-action-menu_2.entry.js +336 -336
- package/dist/esm/wm-button.entry.js +242 -242
- package/dist/esm/wm-chart-slice.entry.js +14 -14
- package/dist/esm/wm-chart.entry.js +111 -111
- package/dist/esm/wm-datepicker.entry.js +266 -266
- package/dist/esm/wm-input.entry.js +135 -135
- package/dist/esm/wm-modal-footer.entry.js +34 -34
- package/dist/esm/wm-modal-header.entry.js +37 -37
- package/dist/esm/wm-modal.entry.js +164 -164
- package/dist/esm/wm-navigation_3.entry.js +228 -228
- package/dist/esm/wm-navigator.entry.js +269 -269
- package/dist/esm/wm-network-uploader.entry.js +455 -455
- package/dist/esm/wm-option_2.entry.js +717 -710
- package/dist/esm/wm-pagination.entry.js +168 -168
- package/dist/esm/wm-progress-indicator_3.entry.js +130 -130
- package/dist/esm/wm-search.entry.js +233 -233
- package/dist/esm/wm-snackbar.entry.js +171 -171
- package/dist/esm/wm-tab-item_3.entry.js +297 -297
- package/dist/esm/wm-tag-input.entry.js +580 -580
- package/dist/esm/wm-timepicker.entry.js +384 -384
- package/dist/esm/wm-toggletip.entry.js +125 -125
- package/dist/esm/wm-uploader.entry.js +630 -632
- package/dist/esm/wm-wrapper.entry.js +13 -13
- package/dist/esm-es5/{chartFunctions-20f05eb5.js → chartFunctions-6f90586d.js} +1 -1
- package/dist/esm-es5/functions-11c0c4e2.js +1 -0
- package/dist/esm-es5/{global-e98c740f.js → global-d36a84c5.js} +1 -1
- package/dist/esm-es5/{http-service-3dc3b3e7.js → http-service-5d037e16.js} +0 -0
- package/dist/esm-es5/{index-7603f98e.js → index-17c79cc6.js} +1 -1
- package/dist/esm-es5/{interfaces-2b97fab2.js → interfaces-61c6305b.js} +0 -0
- package/dist/esm-es5/loader.js +1 -1
- package/dist/esm-es5/priv-chart-popover.entry.js +1 -1
- package/dist/esm-es5/priv-datepicker.entry.js +1 -1
- package/dist/esm-es5/priv-navigator-button.entry.js +1 -1
- package/dist/esm-es5/priv-navigator-item.entry.js +1 -1
- package/dist/esm-es5/ripple.js +1 -1
- package/dist/esm-es5/wm-action-menu_2.entry.js +1 -1
- package/dist/esm-es5/wm-button.entry.js +1 -1
- package/dist/esm-es5/wm-chart-slice.entry.js +1 -1
- package/dist/esm-es5/wm-chart.entry.js +1 -1
- package/dist/esm-es5/wm-datepicker.entry.js +1 -1
- package/dist/esm-es5/wm-input.entry.js +1 -1
- package/dist/esm-es5/wm-modal-footer.entry.js +1 -1
- package/dist/esm-es5/wm-modal-header.entry.js +1 -1
- package/dist/esm-es5/wm-modal.entry.js +1 -1
- package/dist/esm-es5/wm-navigation_3.entry.js +1 -1
- package/dist/esm-es5/wm-navigator.entry.js +1 -1
- package/dist/esm-es5/wm-network-uploader.entry.js +1 -1
- package/dist/esm-es5/wm-option_2.entry.js +1 -1
- package/dist/esm-es5/wm-pagination.entry.js +1 -1
- package/dist/esm-es5/wm-progress-indicator_3.entry.js +1 -1
- package/dist/esm-es5/wm-search.entry.js +1 -1
- package/dist/esm-es5/wm-snackbar.entry.js +1 -1
- package/dist/esm-es5/wm-tab-item_3.entry.js +1 -1
- package/dist/esm-es5/wm-tag-input.entry.js +1 -1
- package/dist/esm-es5/wm-timepicker.entry.js +1 -1
- package/dist/esm-es5/wm-toggletip.entry.js +1 -1
- package/dist/esm-es5/wm-uploader.entry.js +1 -1
- package/dist/esm-es5/wm-wrapper.entry.js +1 -1
- package/dist/ripple/{p-1ccd994d.system.entry.js → p-06012eb7.system.entry.js} +1 -1
- package/dist/ripple/{p-06adbeb9.system.entry.js → p-07ebdcd4.system.entry.js} +1 -1
- package/dist/ripple/{p-f5df5903.system.js → p-08b7ec08.system.js} +0 -0
- package/dist/ripple/{p-7173b0a7.system.entry.js → p-08fc668c.system.entry.js} +1 -1
- package/dist/ripple/p-0e22a3a4.entry.js +1 -0
- package/dist/ripple/p-262c75ff.entry.js +1 -0
- package/dist/ripple/{p-2193190b.js → p-26f94f8f.js} +1 -1
- package/dist/ripple/{p-90ed3268.system.entry.js → p-27253ed2.system.entry.js} +1 -1
- package/dist/ripple/{p-0556279c.system.entry.js → p-2af0b657.system.entry.js} +1 -1
- package/dist/ripple/p-2e5cb321.entry.js +1 -0
- package/dist/ripple/{p-9d02957d.system.js → p-313b6073.system.js} +0 -0
- package/dist/ripple/{p-6bf5cbf4.system.entry.js → p-3e875e87.system.entry.js} +1 -1
- package/dist/ripple/p-4073ac6a.entry.js +1 -0
- package/dist/ripple/p-410687ea.entry.js +1 -0
- package/dist/ripple/p-41374812.system.js +1 -0
- package/dist/ripple/p-42dbd978.entry.js +1 -0
- package/dist/ripple/{p-752da0fb.system.entry.js → p-4439bdd1.system.entry.js} +1 -1
- package/dist/ripple/{p-bbcafbd6.system.entry.js → p-4650a83c.system.entry.js} +1 -1
- package/dist/ripple/{p-4ecd3430.js → p-490dafa4.js} +1 -1
- package/dist/ripple/p-4afa52d4.entry.js +1 -0
- package/dist/ripple/p-4f3f7b2b.entry.js +1 -0
- package/dist/ripple/p-5018f8ae.entry.js +1 -0
- package/dist/ripple/{p-cd3d74d1.system.entry.js → p-57a718f0.system.entry.js} +1 -1
- package/dist/ripple/p-57c94258.entry.js +1 -0
- package/dist/ripple/{p-1df2adee.system.entry.js → p-59678b70.system.entry.js} +1 -1
- package/dist/ripple/p-626d5017.entry.js +1 -0
- package/dist/ripple/{p-3a535823.system.js → p-6a8fd452.system.js} +1 -1
- package/dist/ripple/{p-c1443a0e.system.entry.js → p-6ffdab0d.system.entry.js} +1 -1
- package/dist/ripple/p-70d0e993.js +1 -0
- package/dist/ripple/{p-a0c3ef16.system.js → p-726b9452.system.js} +1 -1
- package/dist/ripple/p-758501d7.entry.js +1 -0
- package/dist/ripple/{p-af3ce4fc.system.entry.js → p-7cb0035a.system.entry.js} +1 -1
- package/dist/ripple/p-80a00759.entry.js +1 -0
- package/dist/ripple/p-80b62707.entry.js +1 -0
- package/dist/ripple/{p-995ba16f.system.entry.js → p-84d0d2ff.system.entry.js} +1 -1
- package/dist/ripple/p-86b9bfcf.system.entry.js +1 -0
- package/dist/ripple/{p-43f1298b.js → p-888bec42.js} +0 -0
- package/dist/ripple/p-8a2cabf2.entry.js +1 -0
- package/dist/ripple/{p-62eac2d6.system.entry.js → p-8b54c589.system.entry.js} +1 -1
- package/dist/ripple/{p-7ef6a7cf.system.entry.js → p-910b6dd5.system.entry.js} +1 -1
- package/dist/ripple/p-962f51ff.entry.js +1 -0
- package/dist/ripple/p-97683ddc.entry.js +1 -0
- package/dist/ripple/p-9c60325b.entry.js +1 -0
- package/dist/ripple/{p-fd8070fb.js → p-a6d6eae7.js} +0 -0
- package/dist/ripple/p-b0536ddb.entry.js +1 -0
- package/dist/ripple/{p-d48c56c7.system.entry.js → p-b4837126.system.entry.js} +1 -1
- package/dist/ripple/{p-04d8b674.system.js → p-b5b7c71f.system.js} +1 -1
- package/dist/ripple/{p-8612829b.system.entry.js → p-b793bc6f.system.entry.js} +1 -1
- package/dist/ripple/p-c2abdcba.js +1 -0
- package/dist/ripple/{p-04e44b30.system.entry.js → p-c49ef079.system.entry.js} +1 -1
- package/dist/ripple/{p-895f5ec5.system.entry.js → p-c4c21c82.system.entry.js} +1 -1
- package/dist/ripple/p-c656fcf1.entry.js +1 -0
- package/dist/ripple/p-caef11cc.entry.js +1 -0
- package/dist/ripple/p-cb357a84.entry.js +1 -0
- package/dist/ripple/p-d2a11410.entry.js +1 -0
- package/dist/ripple/{p-49fd7ede.system.entry.js → p-d927de70.system.entry.js} +1 -1
- package/dist/ripple/{p-ae6cfbca.system.entry.js → p-db44eb6f.system.entry.js} +1 -1
- package/dist/ripple/p-db899686.system.js +1 -0
- package/dist/ripple/{p-f23b3986.system.entry.js → p-dc7e5609.system.entry.js} +1 -1
- package/dist/ripple/p-dfa80ed1.entry.js +1 -0
- package/dist/ripple/p-e012273a.entry.js +1 -0
- package/dist/ripple/p-e4cb8aad.entry.js +1 -0
- package/dist/ripple/{p-e4439bc3.system.entry.js → p-f0d3ceb1.system.entry.js} +1 -1
- package/dist/ripple/{p-3b3e847c.system.entry.js → p-f17076c4.system.entry.js} +1 -1
- package/dist/ripple/p-fa632547.entry.js +1 -0
- package/dist/ripple/{p-0e9ccc6f.system.entry.js → p-fe4c1194.system.entry.js} +1 -1
- package/dist/ripple/{p-ca383a43.system.entry.js → p-ffc34819.system.entry.js} +1 -1
- package/dist/ripple/ripple.esm.js +1 -1
- package/dist/ripple/ripple.js +1 -1
- package/dist/types/components/charts/priv-chart-popover/priv-chart-popover.d.ts +27 -27
- package/dist/types/components/charts/wm-chart/wm-chart-slice.d.ts +11 -11
- package/dist/types/components/charts/wm-chart/wm-chart.d.ts +36 -36
- package/dist/types/components/charts/wm-progress-monitor/wm-progress-indicator.d.ts +27 -27
- package/dist/types/components/charts/wm-progress-monitor/wm-progress-monitor.d.ts +17 -17
- package/dist/types/components/charts/wm-progress-monitor/wm-progress-slice.d.ts +10 -10
- package/dist/types/components/wm-action-menu/wm-action-menu.d.ts +47 -47
- package/dist/types/components/wm-button/wm-button.d.ts +43 -43
- package/dist/types/components/wm-datepicker/priv-datepicker/priv-datepicker.d.ts +80 -80
- package/dist/types/components/wm-datepicker/wm-datepicker.d.ts +41 -41
- package/dist/types/components/wm-input/wm-input.d.ts +46 -46
- package/dist/types/components/wm-menuitem/wm-menuitem.d.ts +34 -34
- package/dist/types/components/wm-modal/wm-modal-footer.d.ts +14 -14
- package/dist/types/components/wm-modal/wm-modal-header.d.ts +11 -11
- package/dist/types/components/wm-modal/wm-modal.d.ts +41 -41
- package/dist/types/components/wm-navigation/wm-navigation-hamburger.d.ts +21 -21
- package/dist/types/components/wm-navigation/wm-navigation-item.d.ts +13 -13
- package/dist/types/components/wm-navigation/wm-navigation.d.ts +28 -28
- package/dist/types/components/wm-navigator/priv-navigator-button/priv-navigator-button.d.ts +10 -10
- package/dist/types/components/wm-navigator/priv-navigator-item/priv-navigator-item.d.ts +13 -13
- package/dist/types/components/wm-navigator/wm-navigator.d.ts +61 -61
- package/dist/types/components/wm-option/wm-option.d.ts +35 -35
- package/dist/types/components/wm-pagination/wm-pagination.d.ts +32 -32
- package/dist/types/components/wm-search/wm-search.d.ts +86 -86
- package/dist/types/components/wm-select/wm-select.d.ts +94 -93
- package/dist/types/components/wm-snackbar/wm-snackbar.d.ts +35 -35
- package/dist/types/components/wm-tabs/wm-tab-item/wm-tab-item.d.ts +38 -38
- package/dist/types/components/wm-tabs/wm-tab-list/wm-tab-list.d.ts +53 -53
- package/dist/types/components/wm-tabs/wm-tab-panel/wm-tab-panel.d.ts +20 -20
- package/dist/types/components/wm-tag-input/wm-tag-input.d.ts +92 -92
- package/dist/types/components/wm-timepicker/wm-timepicker.d.ts +61 -61
- package/dist/types/components/wm-toggletip/wm-toggletip.d.ts +27 -27
- package/dist/types/components/wm-uploader/wm-network-uploader/wm-network-uploader.d.ts +85 -85
- package/dist/types/components/wm-uploader/wm-uploader.d.ts +118 -119
- package/dist/types/components/wm-wrapper/wm-wrapper.d.ts +7 -7
- package/dist/types/components.d.ts +26 -26
- package/dist/types/global/__mocks__/functions.d.ts +6 -6
- package/dist/types/global/chartFunctions.d.ts +29 -29
- package/dist/types/global/functions.d.ts +42 -42
- package/dist/types/global/global.d.ts +1 -1
- package/dist/types/global/interfaces.d.ts +44 -44
- package/dist/types/global/services/__mocks__/http-service.d.ts +6 -6
- package/dist/types/global/services/http-service.d.ts +4 -4
- package/dist/types/lang/lang.d.ts +5 -5
- package/package.json +47 -47
- package/dist/esm-es5/functions-036af8dc.js +0 -15
- package/dist/ripple/p-17ceb8c1.entry.js +0 -1
- package/dist/ripple/p-1887286e.entry.js +0 -1
- package/dist/ripple/p-1d795f42.entry.js +0 -1
- package/dist/ripple/p-2996bfe6.entry.js +0 -1
- package/dist/ripple/p-2f860b24.entry.js +0 -1
- package/dist/ripple/p-358bde27.entry.js +0 -1
- package/dist/ripple/p-3680b55d.entry.js +0 -1
- package/dist/ripple/p-3a1d6fc4.entry.js +0 -1
- package/dist/ripple/p-4d59fe86.entry.js +0 -1
- package/dist/ripple/p-6aa6a818.entry.js +0 -1
- package/dist/ripple/p-726c979a.system.js +0 -15
- package/dist/ripple/p-7ae1a630.entry.js +0 -1
- package/dist/ripple/p-7ecbf258.entry.js +0 -1
- package/dist/ripple/p-7fae0cc2.entry.js +0 -1
- package/dist/ripple/p-8de546e8.entry.js +0 -1
- package/dist/ripple/p-93dee724.entry.js +0 -1
- package/dist/ripple/p-97c2b06f.entry.js +0 -1
- package/dist/ripple/p-9ae81a59.system.js +0 -1
- package/dist/ripple/p-9c92c93f.entry.js +0 -1
- package/dist/ripple/p-9e09d7a1.entry.js +0 -1
- package/dist/ripple/p-a04ba6c8.entry.js +0 -1
- package/dist/ripple/p-a9e0ce88.entry.js +0 -1
- package/dist/ripple/p-b3c71506.js +0 -1
- package/dist/ripple/p-bee0b62d.entry.js +0 -1
- package/dist/ripple/p-da403a48.system.entry.js +0 -1
- package/dist/ripple/p-de3367ee.js +0 -16
- package/dist/ripple/p-e083fca6.entry.js +0 -1
- package/dist/ripple/p-e524d462.entry.js +0 -1
- package/dist/ripple/p-e8d39f68.entry.js +0 -1
- package/dist/ripple/p-ed91be1a.entry.js +0 -1
- package/dist/ripple/p-fda61e7e.entry.js +0 -1
- package/dist/ripple/p-fe952112.entry.js +0 -1
|
@@ -2,721 +2,728 @@
|
|
|
2
2
|
|
|
3
3
|
Object.defineProperty(exports, '__esModule', { value: true });
|
|
4
4
|
|
|
5
|
-
const index = require('./index-
|
|
6
|
-
const functions = require('./functions-
|
|
5
|
+
const index = require('./index-df6a3611.js');
|
|
6
|
+
const functions = require('./functions-8d3ceccd.js');
|
|
7
7
|
|
|
8
|
-
const wmOptionCss = ":host(:not(:last-child)),wm-option:not(:last-child){border-bottom:2px solid rgba(46, 27, 70, 0.05)}:host,wm-option{display:block;cursor:pointer;position:relative;padding:1.25rem;background:#fff;font-family:inherit;list-style:none;color:#4a4a4a;-webkit-user-select:none;-moz-user-select:none;-ms-user-select:none;user-select:none}:host:focus,:host.focus,wm-option:focus,wm-option.focus{outline:none;background:#f4f4f4}:host.icon,wm-option.icon{color:#575195}.checkbox:before{display:inline-block;font:normal normal normal 24px/1 \"Material Design Icons\";font-size:inherit;text-rendering:auto;line-height:inherit;-webkit-font-smoothing:antialiased;-moz-osx-font-smoothing:grayscale;content:\"\\f131\";display:inline;margin-right:0.25rem;color:#575195}:host([aria-selected=true]),wm-option[aria-selected=true]{background:rgba(46, 27, 70, 0.05);font-weight:500}:host([aria-selected=true]) .checkbox:before,wm-option[aria-selected=true] .checkbox:before{content:\"\\f132\"}:host([aria-disabled=true]),wm-option[aria-disabled=true]{font-style:italic;color:#6b6b6b;cursor:default}:host([aria-disabled=true]) .checkbox:before,wm-option[aria-disabled=true] .checkbox:before{color:#6b6b6b}:host(.multi-option),wm-option.multi-option{background:unset}:host(.hidden),wm-option.hidden,:host(.filtered-out),wm-option.filtered-out{display:none}:host(.duplicate.last),wm-option.duplicate.last{border-bottom:12px solid rgba(46, 27, 70, 0.05)}:host(.hassubinfo),wm-option.hassubinfo{display:-ms-flexbox;display:flex}:host(.hassubinfo) .option-wrapper,wm-option.hassubinfo .option-wrapper{-ms-flex:1;flex:1}:host(.hassubinfo) .subinfo,wm-option.hassubinfo .subinfo{-ms-flex:none;flex:none}.subinfo{font-style:italic}.option-wrapper{display:inline-block}::slotted{font-family:inherit}::slotted(i){font-size:0.875rem}::slotted(i):before{display:inline-block;font:normal normal normal 24px/1 \"Material Design Icons\";font-size:inherit;text-rendering:auto;line-height:inherit;-webkit-font-smoothing:antialiased;-moz-osx-font-smoothing:grayscale;display:inline-block;margin-right:0.625rem;pointer-events:none}:host(:focus),wm-option:focus{outline:none;background:#f4f4f4}:host(:hover){background:#f4f4f4;outline:none}";
|
|
8
|
+
const wmOptionCss = ":host(:not(:last-child)),wm-option:not(:last-child){border-bottom:2px solid rgba(46, 27, 70, 0.05)}:host,wm-option{display:block;cursor:pointer;position:relative;padding:1.25rem;background:#fff;font-family:inherit;list-style:none;color:#4a4a4a;-webkit-user-select:none;-moz-user-select:none;-ms-user-select:none;user-select:none}:host:focus,:host.focus,wm-option:focus,wm-option.focus{outline:none;background:#f4f4f4}:host.icon,wm-option.icon{color:#575195}:host .sr-only,wm-option .sr-only{position:absolute !important;width:1px !important;height:1px !important;padding:0 !important;border:0 !important;overflow:hidden !important;clip:rect(0, 0, 0, 0) !important;-webkit-clip-path:inset(50%) !important;clip-path:inset(50%) !important;white-space:nowrap !important;margin:-1px !important}.checkbox:before{display:inline-block;font:normal normal normal 24px/1 \"Material Design Icons\";font-size:inherit;text-rendering:auto;line-height:inherit;-webkit-font-smoothing:antialiased;-moz-osx-font-smoothing:grayscale;content:\"\\f131\";display:inline;margin-right:0.25rem;color:#575195}:host([aria-selected=true]),wm-option[aria-selected=true]{background:rgba(46, 27, 70, 0.05);font-weight:500}:host([aria-selected=true]) .checkbox:before,wm-option[aria-selected=true] .checkbox:before{content:\"\\f132\"}:host([aria-disabled=true]),wm-option[aria-disabled=true]{font-style:italic;color:#6b6b6b;cursor:default}:host([aria-disabled=true]) .checkbox:before,wm-option[aria-disabled=true] .checkbox:before{color:#6b6b6b}:host(.multi-option),wm-option.multi-option{background:unset}:host(.hidden),wm-option.hidden,:host(.filtered-out),wm-option.filtered-out{display:none}:host(.duplicate.last),wm-option.duplicate.last{border-bottom:12px solid rgba(46, 27, 70, 0.05)}:host(.hassubinfo),wm-option.hassubinfo{display:-ms-flexbox;display:flex}:host(.hassubinfo) .option-wrapper,wm-option.hassubinfo .option-wrapper{-ms-flex:1;flex:1}:host(.hassubinfo) .subinfo,wm-option.hassubinfo .subinfo{-ms-flex:none;flex:none}.subinfo{font-style:italic}.option-wrapper{display:inline-block}::slotted{font-family:inherit}::slotted(i){font-size:0.875rem}::slotted(i):before{display:inline-block;font:normal normal normal 24px/1 \"Material Design Icons\";font-size:inherit;text-rendering:auto;line-height:inherit;-webkit-font-smoothing:antialiased;-moz-osx-font-smoothing:grayscale;display:inline-block;margin-right:0.625rem;pointer-events:none}:host(:focus),wm-option:focus{outline:none;background:#f4f4f4}:host(:hover){background:#f4f4f4;outline:none}";
|
|
9
9
|
|
|
10
|
-
|
|
11
|
-
constructor(hostRef) {
|
|
12
|
-
index.registerInstance(this, hostRef);
|
|
13
|
-
this.wmOptionSelected = index.createEvent(this, "wmOptionSelected", 7);
|
|
14
|
-
this.wmKeyUpPressed = index.createEvent(this, "wmKeyUpPressed", 7);
|
|
15
|
-
this.wmKeyDownPressed = index.createEvent(this, "wmKeyDownPressed", 7);
|
|
16
|
-
this.wmEscKeyPressed = index.createEvent(this, "wmEscKeyPressed", 7);
|
|
17
|
-
this.wmHomeKeyPressed = index.createEvent(this, "wmHomeKeyPressed", 7);
|
|
18
|
-
this.wmEndKeyPressed = index.createEvent(this, "wmEndKeyPressed", 7);
|
|
19
|
-
this.wmEnterKeyPressed = index.createEvent(this, "wmEnterKeyPressed", 7);
|
|
20
|
-
this.wmLetterPressed = index.createEvent(this, "wmLetterPressed", 7);
|
|
21
|
-
this.wmOptionBlurred = index.createEvent(this, "wmOptionBlurred", 7);
|
|
22
|
-
this.subinfo = undefined;
|
|
23
|
-
this.disabled = false;
|
|
24
|
-
this.selected = false;
|
|
25
|
-
this.focused = false;
|
|
26
|
-
this.searchTerm = "";
|
|
27
|
-
}
|
|
28
|
-
get hostClasses() {
|
|
29
|
-
let classes = [];
|
|
30
|
-
if (this.subinfo) {
|
|
31
|
-
classes.push("hassubinfo");
|
|
32
|
-
}
|
|
33
|
-
if (this.parentSelectEl.multiple) {
|
|
34
|
-
classes.push("multi-option");
|
|
35
|
-
}
|
|
36
|
-
if (!this.el.textContent.toLowerCase().includes(this.searchTerm)) {
|
|
37
|
-
classes.push("filtered-out");
|
|
38
|
-
}
|
|
39
|
-
return classes.join(" ");
|
|
40
|
-
}
|
|
41
|
-
get parentSelectEl() {
|
|
42
|
-
var _a;
|
|
43
|
-
return ((_a = this.el.parentElement) === null || _a === void 0 ? void 0 : _a.nodeName) !== "WM-SELECT"
|
|
44
|
-
? this.el.getRootNode().host
|
|
45
|
-
: this.el.parentElement;
|
|
46
|
-
}
|
|
47
|
-
handleKeydown(ev) {
|
|
48
|
-
switch (ev.key) {
|
|
49
|
-
case "ArrowUp":
|
|
50
|
-
ev.preventDefault();
|
|
51
|
-
this.wmKeyUpPressed.emit(this.el);
|
|
52
|
-
break;
|
|
53
|
-
case "ArrowDown":
|
|
54
|
-
ev.preventDefault();
|
|
55
|
-
this.wmKeyDownPressed.emit(this.el);
|
|
56
|
-
break;
|
|
57
|
-
case "Enter":
|
|
58
|
-
ev.preventDefault();
|
|
59
|
-
if (!this.disabled) {
|
|
60
|
-
this.el.click();
|
|
61
|
-
this.wmEnterKeyPressed.emit(this.el);
|
|
62
|
-
}
|
|
63
|
-
break;
|
|
64
|
-
case " ":
|
|
65
|
-
ev.preventDefault();
|
|
66
|
-
this.el.click();
|
|
67
|
-
break;
|
|
68
|
-
case "Escape":
|
|
69
|
-
ev.preventDefault();
|
|
70
|
-
ev.stopPropagation(); // for instance if select is in a modal, esc should close the select but not the modal
|
|
71
|
-
this.wmEscKeyPressed.emit();
|
|
72
|
-
break;
|
|
73
|
-
case "Home":
|
|
74
|
-
ev.preventDefault();
|
|
75
|
-
this.wmHomeKeyPressed.emit();
|
|
76
|
-
break;
|
|
77
|
-
case "End":
|
|
78
|
-
ev.preventDefault();
|
|
79
|
-
this.wmEndKeyPressed.emit();
|
|
80
|
-
break;
|
|
81
|
-
default:
|
|
82
|
-
this.wmLetterPressed.emit(ev.key);
|
|
83
|
-
}
|
|
84
|
-
}
|
|
85
|
-
handleSelection() {
|
|
86
|
-
if (!this.disabled) {
|
|
87
|
-
this.wmOptionSelected.emit(this.el);
|
|
88
|
-
// the parent wm-select is in charge of the actual selection
|
|
89
|
-
}
|
|
90
|
-
}
|
|
91
|
-
handleBlur(ev) {
|
|
92
|
-
this.wmOptionBlurred.emit({ relatedTarget: ev.relatedTarget });
|
|
93
|
-
}
|
|
94
|
-
syncAriaSelected() {
|
|
95
|
-
// this function only keeps the aria-selected attr in sync with the selected prop
|
|
96
|
-
// all the logic for selecting / deselecting happens in the parent wm-select
|
|
97
|
-
this.selected ? this.el.setAttribute("aria-selected", "true") : this.el.removeAttribute("aria-selected");
|
|
98
|
-
}
|
|
99
|
-
syncAriaDisabled() {
|
|
100
|
-
this.disabled ? this.el.setAttribute("aria-disabled", "true") : this.el.removeAttribute("aria-disabled");
|
|
101
|
-
}
|
|
102
|
-
focusHandler(newValue) {
|
|
103
|
-
if (newValue)
|
|
104
|
-
this.el.focus();
|
|
105
|
-
}
|
|
106
|
-
updateDisabledOnClick() {
|
|
107
|
-
if (this.disabled && this.el.onclick) {
|
|
108
|
-
this.onClickFunc = this.el.onclick;
|
|
109
|
-
this.el.onclick = null;
|
|
110
|
-
}
|
|
111
|
-
else if (!this.disabled && this.onClickFunc) {
|
|
112
|
-
this.el.onclick = this.onClickFunc;
|
|
113
|
-
}
|
|
114
|
-
}
|
|
115
|
-
handleSearch(ev) {
|
|
116
|
-
// filter is case-insensitive, so
|
|
117
|
-
this.searchTerm = ev.detail.searchTerm.toLowerCase();
|
|
118
|
-
|
|
119
|
-
|
|
120
|
-
|
|
121
|
-
|
|
122
|
-
|
|
123
|
-
|
|
124
|
-
|
|
125
|
-
|
|
126
|
-
|
|
127
|
-
|
|
128
|
-
|
|
129
|
-
|
|
130
|
-
|
|
131
|
-
|
|
132
|
-
|
|
133
|
-
"
|
|
134
|
-
|
|
135
|
-
|
|
136
|
-
|
|
137
|
-
}
|
|
10
|
+
const Option = class {
|
|
11
|
+
constructor(hostRef) {
|
|
12
|
+
index.registerInstance(this, hostRef);
|
|
13
|
+
this.wmOptionSelected = index.createEvent(this, "wmOptionSelected", 7);
|
|
14
|
+
this.wmKeyUpPressed = index.createEvent(this, "wmKeyUpPressed", 7);
|
|
15
|
+
this.wmKeyDownPressed = index.createEvent(this, "wmKeyDownPressed", 7);
|
|
16
|
+
this.wmEscKeyPressed = index.createEvent(this, "wmEscKeyPressed", 7);
|
|
17
|
+
this.wmHomeKeyPressed = index.createEvent(this, "wmHomeKeyPressed", 7);
|
|
18
|
+
this.wmEndKeyPressed = index.createEvent(this, "wmEndKeyPressed", 7);
|
|
19
|
+
this.wmEnterKeyPressed = index.createEvent(this, "wmEnterKeyPressed", 7);
|
|
20
|
+
this.wmLetterPressed = index.createEvent(this, "wmLetterPressed", 7);
|
|
21
|
+
this.wmOptionBlurred = index.createEvent(this, "wmOptionBlurred", 7);
|
|
22
|
+
this.subinfo = undefined;
|
|
23
|
+
this.disabled = false;
|
|
24
|
+
this.selected = false;
|
|
25
|
+
this.focused = false;
|
|
26
|
+
this.searchTerm = "";
|
|
27
|
+
}
|
|
28
|
+
get hostClasses() {
|
|
29
|
+
let classes = [];
|
|
30
|
+
if (this.subinfo) {
|
|
31
|
+
classes.push("hassubinfo");
|
|
32
|
+
}
|
|
33
|
+
if (this.parentSelectEl.multiple) {
|
|
34
|
+
classes.push("multi-option");
|
|
35
|
+
}
|
|
36
|
+
if (!this.el.textContent.toLowerCase().includes(this.searchTerm)) {
|
|
37
|
+
classes.push("filtered-out");
|
|
38
|
+
}
|
|
39
|
+
return classes.join(" ");
|
|
40
|
+
}
|
|
41
|
+
get parentSelectEl() {
|
|
42
|
+
var _a;
|
|
43
|
+
return ((_a = this.el.parentElement) === null || _a === void 0 ? void 0 : _a.nodeName) !== "WM-SELECT"
|
|
44
|
+
? this.el.getRootNode().host
|
|
45
|
+
: this.el.parentElement;
|
|
46
|
+
}
|
|
47
|
+
handleKeydown(ev) {
|
|
48
|
+
switch (ev.key) {
|
|
49
|
+
case "ArrowUp":
|
|
50
|
+
ev.preventDefault();
|
|
51
|
+
this.wmKeyUpPressed.emit(this.el);
|
|
52
|
+
break;
|
|
53
|
+
case "ArrowDown":
|
|
54
|
+
ev.preventDefault();
|
|
55
|
+
this.wmKeyDownPressed.emit(this.el);
|
|
56
|
+
break;
|
|
57
|
+
case "Enter":
|
|
58
|
+
ev.preventDefault();
|
|
59
|
+
if (!this.disabled) {
|
|
60
|
+
this.el.click();
|
|
61
|
+
this.wmEnterKeyPressed.emit(this.el);
|
|
62
|
+
}
|
|
63
|
+
break;
|
|
64
|
+
case " ":
|
|
65
|
+
ev.preventDefault();
|
|
66
|
+
this.el.click();
|
|
67
|
+
break;
|
|
68
|
+
case "Escape":
|
|
69
|
+
ev.preventDefault();
|
|
70
|
+
ev.stopPropagation(); // for instance if select is in a modal, esc should close the select but not the modal
|
|
71
|
+
this.wmEscKeyPressed.emit();
|
|
72
|
+
break;
|
|
73
|
+
case "Home":
|
|
74
|
+
ev.preventDefault();
|
|
75
|
+
this.wmHomeKeyPressed.emit();
|
|
76
|
+
break;
|
|
77
|
+
case "End":
|
|
78
|
+
ev.preventDefault();
|
|
79
|
+
this.wmEndKeyPressed.emit();
|
|
80
|
+
break;
|
|
81
|
+
default:
|
|
82
|
+
this.wmLetterPressed.emit(ev.key);
|
|
83
|
+
}
|
|
84
|
+
}
|
|
85
|
+
handleSelection() {
|
|
86
|
+
if (!this.disabled) {
|
|
87
|
+
this.wmOptionSelected.emit(this.el);
|
|
88
|
+
// the parent wm-select is in charge of the actual selection
|
|
89
|
+
}
|
|
90
|
+
}
|
|
91
|
+
handleBlur(ev) {
|
|
92
|
+
this.wmOptionBlurred.emit({ relatedTarget: ev.relatedTarget });
|
|
93
|
+
}
|
|
94
|
+
syncAriaSelected() {
|
|
95
|
+
// this function only keeps the aria-selected attr in sync with the selected prop
|
|
96
|
+
// all the logic for selecting / deselecting happens in the parent wm-select
|
|
97
|
+
this.selected ? this.el.setAttribute("aria-selected", "true") : this.el.removeAttribute("aria-selected");
|
|
98
|
+
}
|
|
99
|
+
syncAriaDisabled() {
|
|
100
|
+
this.disabled ? this.el.setAttribute("aria-disabled", "true") : this.el.removeAttribute("aria-disabled");
|
|
101
|
+
}
|
|
102
|
+
focusHandler(newValue) {
|
|
103
|
+
if (newValue)
|
|
104
|
+
this.el.focus();
|
|
105
|
+
}
|
|
106
|
+
updateDisabledOnClick() {
|
|
107
|
+
if (this.disabled && this.el.onclick) {
|
|
108
|
+
this.onClickFunc = this.el.onclick;
|
|
109
|
+
this.el.onclick = null;
|
|
110
|
+
}
|
|
111
|
+
else if (!this.disabled && this.onClickFunc) {
|
|
112
|
+
this.el.onclick = this.onClickFunc;
|
|
113
|
+
}
|
|
114
|
+
}
|
|
115
|
+
handleSearch(ev) {
|
|
116
|
+
// filter is case-insensitive, so
|
|
117
|
+
this.searchTerm = ev.detail.searchTerm.toLowerCase();
|
|
118
|
+
if (this.searchTerm) {
|
|
119
|
+
const regexp = new RegExp(`${this.searchTerm}`, "gi");
|
|
120
|
+
const boldedText = this.el.textContent.replace(regexp, (match) => `<strong>${match}</strong>`);
|
|
121
|
+
// for voiceover, text splitting is read as separate phrases. i.e. hel<b>lo</b> will be read as "hel lo"
|
|
122
|
+
// aria-hidden and sr-only is used to circumvent this
|
|
123
|
+
this.textEl.innerHTML = boldedText;
|
|
124
|
+
}
|
|
125
|
+
else {
|
|
126
|
+
this.textEl.innerHTML = this.el.textContent;
|
|
127
|
+
}
|
|
128
|
+
}
|
|
129
|
+
componentWillLoad() {
|
|
130
|
+
this.syncAriaSelected();
|
|
131
|
+
this.syncAriaDisabled();
|
|
132
|
+
this.updateDisabledOnClick();
|
|
133
|
+
this.parentSelectEl.addEventListener("wmSelectSearchChanged", (ev) => this.handleSearch(ev));
|
|
134
|
+
}
|
|
135
|
+
render() {
|
|
136
|
+
return (index.h(index.Host, { role: "option", tabindex: this.focused ? 0 : -1, class: this.hostClasses }, index.h("div", { class: `option-wrapper ${this.parentSelectEl.multiple ? "checkbox" : ""}` }, index.h("span", { "aria-hidden": "true", ref: (el) => (this.textEl = el) }, this.el.textContent), index.h("span", { class: "sr-only" }, this.el.textContent)), index.h("div", { class: "subinfo" }, this.subinfo)));
|
|
137
|
+
}
|
|
138
|
+
get el() { return index.getElement(this); }
|
|
139
|
+
static get watchers() { return {
|
|
140
|
+
"selected": ["syncAriaSelected"],
|
|
141
|
+
"disabled": ["syncAriaDisabled", "updateDisabledOnClick"],
|
|
142
|
+
"focused": ["focusHandler"]
|
|
143
|
+
}; }
|
|
144
|
+
};
|
|
138
145
|
Option.style = wmOptionCss;
|
|
139
146
|
|
|
140
|
-
const wmSelectCss = ":host{position:relative;display:block;-webkit-box-sizing:border-box;box-sizing:border-box;font-family:inherit}:host .sr-only{position:absolute !important;width:1px !important;height:1px !important;padding:0 !important;border:0 !important;overflow:hidden !important;clip:rect(0, 0, 0, 0) !important;-webkit-clip-path:inset(50%) !important;clip-path:inset(50%) !important;white-space:nowrap !important;margin:-1px !important}.wrapper{display:-webkit-box;display:-webkit-flex;display:-ms-flexbox;display:flex;-ms-flex-direction:column;flex-direction:column;font-size:0.875rem}.wrapper .label{display:block;line-height:normal;font-weight:600;white-space:nowrap;margin-bottom:0.25rem}.wrapper .label .required{color:#c0392b}.wrapper.label-left{-ms-flex-direction:row;flex-direction:row}.wrapper.label-left .label-wrapper{line-height:2.5rem}.wrapper.label-left .label-wrapper .label{display:-ms-flexbox;display:flex;-ms-flex-align:center;align-items:center;min-height:2.5rem;white-space:normal;margin-bottom:0;margin-right:0.75rem}.wrapper.label-none label{position:absolute !important;width:1px !important;height:1px !important;padding:0 !important;border:0 !important;overflow:hidden !important;clip:rect(0, 0, 0, 0) !important;-webkit-clip-path:inset(50%) !important;clip-path:inset(50%) !important;white-space:nowrap !important;margin:-1px !important}.wrapper.invalid .label{color:#c0392b}.wrapper.invalid .label:after{display:inline-block;font:normal normal normal 24px/1 \"Material Design Icons\";font-size:inherit;text-rendering:auto;line-height:inherit;-webkit-font-smoothing:antialiased;-moz-osx-font-smoothing:grayscale;content:\"\\f026\";margin-left:0.3125rem}[dir=RTL] .wrapper.invalid .label:after{margin-left:0;margin-right:0.3125rem}.wrapper.rtl.label-left .label{margin-right:0;margin-left:0.75rem}.wrapper .button-wrapper{position:relative;-ms-flex:1;flex:1;font-size:1.125rem;color:#575195;min-width:8.75rem}.wrapper .button-wrapper .displayedoption{-webkit-border-radius:3px;-moz-border-radius:3px;-ms-border-radius:3px;border-radius:3px;-webkit-box-shadow:none;-moz-box-shadow:none;box-shadow:none;display:-ms-flexbox;display:flex;-ms-flex-align:center;align-items:center;-ms-flex-wrap:nowrap;flex-wrap:nowrap;background:transparent;width:100%;border:solid 1px rgba(35, 35, 35, 0.6);padding:0 1.875rem 0 0.9375rem;cursor:pointer;height:2.5rem;line-height:normal;font-family:inherit;color:#575195;font-weight:400;font-size:0.875rem;text-transform:none;text-align:left}@media only screen and (max-width: 768px){.wrapper .button-wrapper .displayedoption{height:2.75rem}}.wrapper .button-wrapper .displayedoption:before{display:inline-block;font:normal normal normal 24px/1 \"Material Design Icons\";font-size:inherit;text-rendering:auto;line-height:inherit;-webkit-font-smoothing:antialiased;-moz-osx-font-smoothing:grayscale;content:\"\\f140\";position:absolute;right:0.5625rem;pointer-events:none}.wrapper .button-wrapper .displayedoption:hover:not(:disabled):not(.-disabled):not(.-raised){background:transparent;text-decoration:none}.wrapper .button-wrapper .displayedoption:active{-ms-transform:scale(1, 1) !important;-webkit-transform:scale(1, 1) !important;transform:scale(1, 1) !important}.wrapper .button-wrapper .displayedoption:focus{outline:none}.wrapper .button-wrapper .displayedoption::-moz-focus-inner{border:0}.wrapper .button-wrapper .displayedoption.user-is-tabbing:focus{-webkit-box-shadow:0 2px 2px 0 rgba(244, 243, 246, 0.2), 0 2px 2px 0 rgba(0, 0, 0, 0.1), 0 0 4px 3px #61279e;-moz-box-shadow:0 2px 2px 0 rgba(244, 243, 246, 0.2), 0 2px 2px 0 rgba(0, 0, 0, 0.1), 0 0 4px 3px #61279e;box-shadow:0 2px 2px 0 rgba(244, 243, 246, 0.2), 0 2px 2px 0 rgba(0, 0, 0, 0.1), 0 0 4px 3px #61279e}.wrapper .button-wrapper .displayedoption.user-is-tabbing:focus::-moz-focus-inner{border:0}.wrapper .button-wrapper .displayedoption .overflowcontrol{display:block;white-space:nowrap;text-overflow:ellipsis;overflow:hidden;-ms-flex:1;flex:1}.wrapper .button-wrapper .displayedoption .overflowcontrol.hassubinfo{display:-ms-flexbox;display:flex}.wrapper .button-wrapper .displayedoption .overflowcontrol.hassubinfo .button-text{-ms-flex:1;flex:1;text-overflow:ellipsis;overflow:hidden;min-width:0}.wrapper .button-wrapper .displayedoption .overflowcontrol.hassubinfo .subinfo{-ms-flex:none;flex:none;font-style:italic}.wrapper .button-wrapper .displayedoption .overflow-counter{font-weight:bold;margin-left:0.5rem}.wrapper .button-wrapper>.displayedoption[disabled]{color:#6b6b6b;border-color:#8a8a8a;background:#f0f0f0;cursor:default}.wrapper .button-wrapper>.dropdown{-webkit-overflow-scrolling:touch
|
|
147
|
+
const wmSelectCss = ":host{position:relative;display:block;-webkit-box-sizing:border-box;box-sizing:border-box;font-family:inherit}:host .sr-only{position:absolute !important;width:1px !important;height:1px !important;padding:0 !important;border:0 !important;overflow:hidden !important;clip:rect(0, 0, 0, 0) !important;-webkit-clip-path:inset(50%) !important;clip-path:inset(50%) !important;white-space:nowrap !important;margin:-1px !important}.wrapper{display:-webkit-box;display:-webkit-flex;display:-ms-flexbox;display:flex;-ms-flex-direction:column;flex-direction:column;font-size:0.875rem}.wrapper .label{display:block;line-height:normal;font-weight:600;white-space:nowrap;margin-bottom:0.25rem}.wrapper .label .required{color:#c0392b}.wrapper.label-left{-ms-flex-direction:row;flex-direction:row}.wrapper.label-left .label-wrapper{line-height:2.5rem}.wrapper.label-left .label-wrapper .label{display:-ms-flexbox;display:flex;-ms-flex-align:center;align-items:center;min-height:2.5rem;white-space:normal;margin-bottom:0;margin-right:0.75rem}.wrapper.label-none label{position:absolute !important;width:1px !important;height:1px !important;padding:0 !important;border:0 !important;overflow:hidden !important;clip:rect(0, 0, 0, 0) !important;-webkit-clip-path:inset(50%) !important;clip-path:inset(50%) !important;white-space:nowrap !important;margin:-1px !important}.wrapper.invalid .label{color:#c0392b}.wrapper.invalid .label:after{display:inline-block;font:normal normal normal 24px/1 \"Material Design Icons\";font-size:inherit;text-rendering:auto;line-height:inherit;-webkit-font-smoothing:antialiased;-moz-osx-font-smoothing:grayscale;content:\"\\f026\";margin-left:0.3125rem}[dir=RTL] .wrapper.invalid .label:after{margin-left:0;margin-right:0.3125rem}.wrapper.rtl.label-left .label{margin-right:0;margin-left:0.75rem}.wrapper .button-wrapper{position:relative;-ms-flex:1;flex:1;font-size:1.125rem;color:#575195;min-width:8.75rem}.wrapper .button-wrapper .displayedoption{-webkit-border-radius:3px;-moz-border-radius:3px;-ms-border-radius:3px;border-radius:3px;-webkit-box-shadow:none;-moz-box-shadow:none;box-shadow:none;display:-ms-flexbox;display:flex;-ms-flex-align:center;align-items:center;-ms-flex-wrap:nowrap;flex-wrap:nowrap;background:transparent;width:100%;border:solid 1px rgba(35, 35, 35, 0.6);padding:0 1.875rem 0 0.9375rem;cursor:pointer;height:2.5rem;line-height:normal;font-family:inherit;color:#575195;font-weight:400;font-size:0.875rem;text-transform:none;text-align:left}@media only screen and (max-width: 768px){.wrapper .button-wrapper .displayedoption{height:2.75rem}}.wrapper .button-wrapper .displayedoption:before{display:inline-block;font:normal normal normal 24px/1 \"Material Design Icons\";font-size:inherit;text-rendering:auto;line-height:inherit;-webkit-font-smoothing:antialiased;-moz-osx-font-smoothing:grayscale;content:\"\\f140\";position:absolute;right:0.5625rem;pointer-events:none}.wrapper .button-wrapper .displayedoption:hover:not(:disabled):not(.-disabled):not(.-raised){background:transparent;text-decoration:none}.wrapper .button-wrapper .displayedoption:active{-ms-transform:scale(1, 1) !important;-webkit-transform:scale(1, 1) !important;transform:scale(1, 1) !important}.wrapper .button-wrapper .displayedoption:focus{outline:none}.wrapper .button-wrapper .displayedoption::-moz-focus-inner{border:0}.wrapper .button-wrapper .displayedoption.user-is-tabbing:focus{-webkit-box-shadow:0 2px 2px 0 rgba(244, 243, 246, 0.2), 0 2px 2px 0 rgba(0, 0, 0, 0.1), 0 0 4px 3px #61279e;-moz-box-shadow:0 2px 2px 0 rgba(244, 243, 246, 0.2), 0 2px 2px 0 rgba(0, 0, 0, 0.1), 0 0 4px 3px #61279e;box-shadow:0 2px 2px 0 rgba(244, 243, 246, 0.2), 0 2px 2px 0 rgba(0, 0, 0, 0.1), 0 0 4px 3px #61279e}.wrapper .button-wrapper .displayedoption.user-is-tabbing:focus::-moz-focus-inner{border:0}.wrapper .button-wrapper .displayedoption .overflowcontrol{display:block;white-space:nowrap;text-overflow:ellipsis;overflow:hidden;-ms-flex:1;flex:1}.wrapper .button-wrapper .displayedoption .overflowcontrol.hassubinfo{display:-ms-flexbox;display:flex}.wrapper .button-wrapper .displayedoption .overflowcontrol.hassubinfo .button-text{-ms-flex:1;flex:1;text-overflow:ellipsis;overflow:hidden;min-width:0}.wrapper .button-wrapper .displayedoption .overflowcontrol.hassubinfo .subinfo{-ms-flex:none;flex:none;font-style:italic}.wrapper .button-wrapper .displayedoption .overflow-counter{font-weight:bold;margin-left:0.5rem}.wrapper .button-wrapper>.displayedoption[disabled]{color:#6b6b6b;border-color:#8a8a8a;background:#f0f0f0;cursor:default}.wrapper .button-wrapper>.dropdown{-webkit-overflow-scrolling:touch;-webkit-box-shadow:0 4px 15px 0 rgba(0, 0, 0, 0.2);-moz-box-shadow:0 4px 15px 0 rgba(0, 0, 0, 0.2);box-shadow:0 4px 15px 0 rgba(0, 0, 0, 0.2);-ms-transition:transform 0.25s ease;-webkit-transition:transform 0.25s ease;-moz-transition:transform 0.25s ease;-webkit-transition:-webkit-transform 0.25s ease;transition:-webkit-transform 0.25s ease;transition:transform 0.25s ease;transition:transform 0.25s ease, -webkit-transform 0.25s ease;-ms-transform:scale(1, 0);-webkit-transform:scale(1, 0);-moz-transform:scale(1, 0);transform:scale(1, 0);-ms-transform-origin:center top;-webkit-transform-origin:center top;-moz-transform-origin:center top;transform-origin:center top;-webkit-border-radius:3px;-moz-border-radius:3px;-ms-border-radius:3px;border-radius:3px;position:absolute;top:0;right:0;background:#fff;z-index:100;width:100%;font-size:0.875rem}.wrapper .button-wrapper>.dropdown.upwards{top:unset;bottom:calc(100% - 2.5rem);-ms-transform-origin:center bottom;-webkit-transform-origin:center bottom;-moz-transform-origin:center bottom;transform-origin:center bottom}.wrapper .button-wrapper>.dropdown.hidden{visibility:hidden}.wrapper .button-wrapper>.dropdown.open{-ms-transform:scale(1, 1);-webkit-transform:scale(1, 1);-moz-transform:scale(1, 1);transform:scale(1, 1)}.wrapper.invalid .button-wrapper .displayedoption{-webkit-box-shadow:0 0 0 1px #c0392b;-moz-box-shadow:0 0 0 1px #c0392b;box-shadow:0 0 0 1px #c0392b;border-color:#c0392b}.wrapper .error-message{color:#c0392b;font-size:0.875rem;margin-top:0.25rem;margin-bottom:0.25rem;display:block;top:100%;left:0;font-style:italic}.options-wrapper{max-height:12.5rem;overflow:auto}.search{-webkit-box-sizing:border-box;box-sizing:border-box;border-bottom:2px solid rgba(46, 27, 70, 0.05);padding:1.25rem}.search .searchfield-wrapper{-webkit-box-sizing:border-box;box-sizing:border-box;display:-ms-flexbox;display:flex;-ms-flex-align:center;align-items:center;height:2.75rem;width:100%;padding:0 0.75rem;-webkit-border-radius:3px;-moz-border-radius:3px;-ms-border-radius:3px;border-radius:3px;border:1px solid #4a4a4a}.search .searchfield-wrapper:focus,.search .searchfield-wrapper.focus{-webkit-box-shadow:0 0 0 1px #20cbd4;-moz-box-shadow:0 0 0 1px #20cbd4;box-shadow:0 0 0 1px #20cbd4;outline:none;border-color:#20cbd4}.search .searchfield{width:100%;border:none;outline:none;font-family:inherit;font-size:0.875rem;margin-left:0.25rem}.search .icon:before{display:inline-block;font:normal normal normal 24px/1 \"Material Design Icons\";font-size:inherit;text-rendering:auto;line-height:inherit;-webkit-font-smoothing:antialiased;-moz-osx-font-smoothing:grayscale;content:\"\\f349\";color:#6b6b6b;font-size:0.875rem}.search-results-message{padding:1.25rem;color:#4a4a4a;font-size:0.875rem;font-style:italic}.rtl>.dropdown{-ms-transform-origin:left top;-webkit-transform-origin:left top;-moz-transform-origin:left top;transform-origin:left top;left:0;right:auto}.rtl>.dropdown .option{padding-left:3rem;padding-right:1.25rem}.rtl>.displayedoption{padding:0 0.9375rem 0 1.875rem;text-align:right}.rtl>.displayedoption:before{right:auto;left:0.5625rem}";
|
|
141
148
|
|
|
142
|
-
|
|
143
|
-
constructor(hostRef) {
|
|
144
|
-
index.registerInstance(this, hostRef);
|
|
145
|
-
this.wmSelectDidLoad = index.createEvent(this, "wmSelectDidLoad", 7);
|
|
146
|
-
this.wmSelectBlurred = index.createEvent(this, "wmSelectBlurred", 7);
|
|
147
|
-
this.wmComponentBlurred = index.createEvent(this, "wmComponentBlurred", 7);
|
|
148
|
-
this.wmSelectSearchChanged = index.createEvent(this, "wmSelectSearchChanged", 7);
|
|
149
|
-
this.disabled = false;
|
|
150
|
-
this.invalid = false;
|
|
151
|
-
this.labelPosition = "top";
|
|
152
|
-
this.requiredField = false;
|
|
153
|
-
this.errorMessage = "";
|
|
154
|
-
this.multiple = false;
|
|
155
|
-
this.search = false;
|
|
156
|
-
this.placeholder = functions.intl.formatMessage({
|
|
157
|
-
id: "select.multiPlaceholder",
|
|
158
|
-
defaultMessage: "Make a selection",
|
|
159
|
-
description: "Placeholder text. Use imperative",
|
|
160
|
-
});
|
|
161
|
-
this.searchPlaceholder = functions.intl.formatMessage({
|
|
162
|
-
id: "select.searchPlaceholder",
|
|
163
|
-
defaultMessage: "Search",
|
|
164
|
-
description: "Placeholder text. Use imperative",
|
|
165
|
-
});
|
|
166
|
-
this.allSelectedMessage = functions.intl.formatMessage({
|
|
167
|
-
id: "select.allSelected",
|
|
168
|
-
defaultMessage: "All selected",
|
|
169
|
-
description: "Text displayed when all options are selected",
|
|
170
|
-
});
|
|
171
|
-
this.requiredMessage = functions.intl.formatMessage({
|
|
172
|
-
id: "global.requiredField",
|
|
173
|
-
defaultMessage: "required field",
|
|
174
|
-
});
|
|
175
|
-
this.isTabbing = false;
|
|
176
|
-
this.isExpanded = false;
|
|
177
|
-
this.announcement = "";
|
|
178
|
-
this.keysSoFar = "";
|
|
179
|
-
this.searchIndex = 0;
|
|
180
|
-
this.keyClear = null;
|
|
181
|
-
this.openUp = false;
|
|
182
|
-
//////////////////////////////////////
|
|
183
|
-
//////////////////////////////////////
|
|
184
|
-
// for multiselect button text
|
|
185
|
-
this.overflowCount = 0;
|
|
186
|
-
this.displayedOptions = [];
|
|
187
|
-
this.debouncedSearch = functions.debounce(() => {
|
|
188
|
-
this.wmSelectSearchChanged.emit({ searchTerm: this.searchTerm });
|
|
189
|
-
if (this.filteredOptions.length) {
|
|
190
|
-
this.announce(this.resultsFoundMessage);
|
|
191
|
-
}
|
|
192
|
-
else {
|
|
193
|
-
this.announce(this.noResultsFoundMessage);
|
|
194
|
-
}
|
|
195
|
-
}, 150);
|
|
196
|
-
}
|
|
197
|
-
get childOptions() {
|
|
198
|
-
return Array.from(this.el.querySelectorAll("wm-option"));
|
|
199
|
-
}
|
|
200
|
-
get duplicateOptions() {
|
|
201
|
-
return Array.from(this.el.shadowRoot.querySelectorAll("wm-option"));
|
|
202
|
-
}
|
|
203
|
-
get allOptionEls() {
|
|
204
|
-
// this includes both slotted wm-options and internally created wm-options
|
|
205
|
-
return this.duplicateOptions.concat(this.childOptions);
|
|
206
|
-
}
|
|
207
|
-
get visibleOptionEls() {
|
|
208
|
-
return this.allOptionEls.filter((option) => !option.classList.contains("hidden") && !option.classList.contains("filtered-out"));
|
|
209
|
-
}
|
|
210
|
-
//////////////////////////////////////
|
|
211
|
-
// for search variants
|
|
212
|
-
get searchTerm() {
|
|
213
|
-
return this.searchFieldEl ? this.searchFieldEl.value : "";
|
|
214
|
-
}
|
|
215
|
-
get filteredOptions() {
|
|
216
|
-
return this.childOptions.filter((option) => { var _a; return (_a = option.textContent) === null || _a === void 0 ? void 0 : _a.toLowerCase().includes(this.searchTerm.toLowerCase()); });
|
|
217
|
-
}
|
|
218
|
-
get selectedOptions() {
|
|
219
|
-
return Array.from(this.el.querySelectorAll("wm-option")).filter((x) => x.selected);
|
|
220
|
-
}
|
|
221
|
-
get allSelected() {
|
|
222
|
-
return this.childOptions.filter((option) => option.selected).length === this.childOptions.length;
|
|
223
|
-
}
|
|
224
|
-
//////////////////////////////////////
|
|
225
|
-
get resultsFoundMessage() {
|
|
226
|
-
return functions.intl.formatMessage({
|
|
227
|
-
id: "select.searchResultsFound",
|
|
228
|
-
defaultMessage: "{numResults, plural, one {1 option found} other {# options found}}",
|
|
229
|
-
description: "The message read by the screen reader, indicating how many results a search returned",
|
|
230
|
-
}, { numResults: this.filteredOptions.length });
|
|
231
|
-
}
|
|
232
|
-
get noResultsFoundMessage() {
|
|
233
|
-
return functions.intl.formatMessage({
|
|
234
|
-
id: "select.noSearchResults",
|
|
235
|
-
defaultMessage: "No results found. Please try your search again.",
|
|
236
|
-
description: "The message displayed when no options pass the search filter",
|
|
237
|
-
});
|
|
238
|
-
}
|
|
239
|
-
toggleTabbingOn() {
|
|
240
|
-
this.isTabbing = true;
|
|
241
|
-
}
|
|
242
|
-
toggleTabbingOff() {
|
|
243
|
-
this.isTabbing = false;
|
|
244
|
-
}
|
|
245
|
-
handleOptionSelection(ev) {
|
|
246
|
-
if (!this.multiple) {
|
|
247
|
-
// ensure only one option is selected at a time, unselect all other options
|
|
248
|
-
this.childOptions.forEach((option) => (option.selected = option === ev.detail));
|
|
249
|
-
}
|
|
250
|
-
this.focusOption(ev.detail);
|
|
251
|
-
this.selectOption(ev.detail);
|
|
252
|
-
if (!this.multiple) {
|
|
253
|
-
this.close();
|
|
254
|
-
}
|
|
255
|
-
}
|
|
256
|
-
handleChildEnter() {
|
|
257
|
-
// only occurs for multiselects. handle the click, then close
|
|
258
|
-
this.close();
|
|
259
|
-
}
|
|
260
|
-
handleChildUp(ev) {
|
|
261
|
-
this.moveUp(ev.detail);
|
|
262
|
-
}
|
|
263
|
-
handleChildDown(ev) {
|
|
264
|
-
this.moveDown(ev.detail);
|
|
265
|
-
}
|
|
266
|
-
moveToFirstOption() {
|
|
267
|
-
this.focusOption(this.visibleOptionEls[0]);
|
|
268
|
-
}
|
|
269
|
-
moveToLastOption() {
|
|
270
|
-
this.focusOption(this.visibleOptionEls[this.visibleOptionEls.length - 1]);
|
|
271
|
-
}
|
|
272
|
-
closePopupOnEscape() {
|
|
273
|
-
this.close();
|
|
274
|
-
}
|
|
275
|
-
handleOptionBlur(ev) {
|
|
276
|
-
// if the Option is blurred to something other than the component emit a blur event with the appropriate relatedTarget
|
|
277
|
-
// keeps our component's blur events accurate, and closes when focusing browser address bar
|
|
278
|
-
if (!this.isElOrChild(ev.detail.relatedTarget)) {
|
|
279
|
-
const event = new CustomEvent("blur");
|
|
280
|
-
// @ts-ignore
|
|
281
|
-
event.relatedTarget = ev.detail.relatedTarget;
|
|
282
|
-
this.el.dispatchEvent(event);
|
|
283
|
-
}
|
|
284
|
-
}
|
|
285
|
-
handleClick(ev) {
|
|
286
|
-
if (!this.isElOrChild(ev.target)) {
|
|
287
|
-
this.close();
|
|
288
|
-
}
|
|
289
|
-
}
|
|
290
|
-
handleButtonBlur(ev) {
|
|
291
|
-
if (this.isElOrChild(ev.relatedTarget)) {
|
|
292
|
-
// do not emit a blur event when opening the dropdown and focusing the Options
|
|
293
|
-
ev.stopPropagation();
|
|
294
|
-
}
|
|
295
|
-
}
|
|
296
|
-
handleSearchFieldBlur(ev) {
|
|
297
|
-
this.searchFieldWrapperEl.classList.remove("focus");
|
|
298
|
-
if (this.isElOrChild(ev.relatedTarget)) {
|
|
299
|
-
// do not emit a blur event when moving from searchfield to options
|
|
300
|
-
ev.stopPropagation();
|
|
301
|
-
}
|
|
302
|
-
}
|
|
303
|
-
handleKey(ev) {
|
|
304
|
-
switch (ev.key) {
|
|
305
|
-
case "ArrowDown":
|
|
306
|
-
if (this.isExpanded === false) {
|
|
307
|
-
ev.preventDefault();
|
|
308
|
-
this.open("next");
|
|
309
|
-
}
|
|
310
|
-
break;
|
|
311
|
-
case "ArrowUp":
|
|
312
|
-
if (this.isExpanded === false) {
|
|
313
|
-
ev.preventDefault();
|
|
314
|
-
this.open("previous");
|
|
315
|
-
}
|
|
316
|
-
break;
|
|
317
|
-
}
|
|
318
|
-
}
|
|
319
|
-
componentWillLoad() {
|
|
320
|
-
if (!this.label) {
|
|
321
|
-
console.error("For accessibility purposes, this component requires a label (even if `label-position` is set to `none`).");
|
|
322
|
-
}
|
|
323
|
-
this.el.focus = function () {
|
|
324
|
-
if (!this.disabled) {
|
|
325
|
-
this.shadowRoot.querySelector("button").focus();
|
|
326
|
-
}
|
|
327
|
-
};
|
|
328
|
-
this.uid = this.el.id ? this.el.id : functions.generateId();
|
|
329
|
-
if (document.body.classList.contains("wmcl-user-is-tabbing")) {
|
|
330
|
-
this.toggleTabbingOn();
|
|
331
|
-
}
|
|
332
|
-
// set initial selections
|
|
333
|
-
if (this.selectedOptions.length > 0) {
|
|
334
|
-
this.selectedOptions.forEach((x) => {
|
|
335
|
-
this.displayedOptions.push(x);
|
|
336
|
-
});
|
|
337
|
-
// single Select only, pre-select if no default option from dev
|
|
338
|
-
}
|
|
339
|
-
else if (!this.multiple) {
|
|
340
|
-
this.selectOption(this.allOptionEls[0]);
|
|
341
|
-
}
|
|
342
|
-
}
|
|
343
|
-
componentDidLoad() {
|
|
344
|
-
this.wmSelectDidLoad.emit();
|
|
345
|
-
if (this.multiple && this.allOptionEls.filter((x) => x.subinfo).length > 0) {
|
|
346
|
-
throw new Error("wm-select with the multiple prop cannot have subinfo options");
|
|
347
|
-
}
|
|
348
|
-
if (this.multiple && !this.placeholder) {
|
|
349
|
-
throw new Error("wm-select with the multiple prop needs to also use the placeholder prop.");
|
|
350
|
-
}
|
|
351
|
-
this.dropdownEl.classList.add("hidden");
|
|
352
|
-
index.forceUpdate(this.el);
|
|
353
|
-
// Dev can overwrite the max-height rule set in the Sass file
|
|
354
|
-
if (this.maxHeight) {
|
|
355
|
-
this.
|
|
356
|
-
}
|
|
357
|
-
const mutationObserver = new MutationObserver(() => this.handleSelectedMutation());
|
|
358
|
-
mutationObserver.observe(this.el, {
|
|
359
|
-
attributes: true,
|
|
360
|
-
attributeFilter: ["selected"],
|
|
361
|
-
subtree: true,
|
|
362
|
-
});
|
|
363
|
-
if (this.multiple) {
|
|
364
|
-
this.updateOptionVisibility();
|
|
365
|
-
}
|
|
366
|
-
this.setButtonText();
|
|
367
|
-
}
|
|
368
|
-
componentWillUpdate() {
|
|
369
|
-
if (this.multiple) {
|
|
370
|
-
// find last visible duplicate option and apply .last class
|
|
371
|
-
const visibleDuplicateOptions = this.visibleOptionEls.filter((option) => option.classList.contains("duplicate"));
|
|
372
|
-
visibleDuplicateOptions.forEach((option, idx) => {
|
|
373
|
-
if (idx === visibleDuplicateOptions.length - 1) {
|
|
374
|
-
option.classList.add("last");
|
|
375
|
-
}
|
|
376
|
-
else {
|
|
377
|
-
option.classList.remove("last");
|
|
378
|
-
}
|
|
379
|
-
});
|
|
380
|
-
}
|
|
381
|
-
}
|
|
382
|
-
handleSelectedMutation() {
|
|
383
|
-
// dispatch change event after selected options change
|
|
384
|
-
// mutation observer prevents emitting event when an already selected option is selected again
|
|
385
|
-
const event = new CustomEvent("change");
|
|
386
|
-
// @ts-ignore
|
|
387
|
-
this.el.dispatchEvent(event);
|
|
388
|
-
}
|
|
389
|
-
moveUp(el) {
|
|
390
|
-
const prevEl = this.visibleOptionEls[this.visibleOptionEls.indexOf(el) - 1];
|
|
391
|
-
if (prevEl) {
|
|
392
|
-
// scroll option to top of dropdown if partially obscured / out of view
|
|
393
|
-
if (prevEl.getBoundingClientRect().top < this.
|
|
394
|
-
this.scrollOptionToTop(prevEl);
|
|
395
|
-
}
|
|
396
|
-
this.focusOption(prevEl);
|
|
397
|
-
}
|
|
398
|
-
else if (this.search) {
|
|
399
|
-
// if top of list and search variant, focus search field
|
|
400
|
-
this.searchFieldEl.focus();
|
|
401
|
-
}
|
|
402
|
-
else {
|
|
403
|
-
// if top of list, focus last element
|
|
404
|
-
this.focusOption(this.visibleOptionEls[this.visibleOptionEls.length - 1]);
|
|
405
|
-
}
|
|
406
|
-
}
|
|
407
|
-
moveDown(el) {
|
|
408
|
-
const nextEl = this.visibleOptionEls[this.visibleOptionEls.indexOf(el) + 1];
|
|
409
|
-
if (nextEl) {
|
|
410
|
-
// scroll option to bottom of dropdown if partially obscured / out of view
|
|
411
|
-
if (nextEl.getBoundingClientRect().bottom > this.
|
|
412
|
-
this.scrollOptionToBottom(nextEl);
|
|
413
|
-
}
|
|
414
|
-
this.focusOption(nextEl);
|
|
415
|
-
}
|
|
416
|
-
else {
|
|
417
|
-
// if end of list, focus first option in all variants
|
|
418
|
-
this.focusOption(this.visibleOptionEls[0]);
|
|
419
|
-
}
|
|
420
|
-
}
|
|
421
|
-
scrollOptionToTop(option) {
|
|
422
|
-
this.
|
|
423
|
-
option.getBoundingClientRect().top - this.
|
|
424
|
-
}
|
|
425
|
-
scrollOptionToBottom(option) {
|
|
426
|
-
this.
|
|
427
|
-
option.getBoundingClientRect().bottom -
|
|
428
|
-
this.
|
|
429
|
-
this.
|
|
430
|
-
this.
|
|
431
|
-
}
|
|
432
|
-
open(optionToSelect) {
|
|
433
|
-
if (!this.disabled) {
|
|
434
|
-
const elHeight = this.el.clientHeight;
|
|
435
|
-
const buttonHeight = this.buttonEl.clientHeight;
|
|
436
|
-
this.openUp = functions.shouldOpenUp(this.el, this.dropdownEl,
|
|
437
|
-
// when opening up, dropdown covers both label and button
|
|
438
|
-
elHeight,
|
|
439
|
-
// when opening down, dropdown covers only the button
|
|
440
|
-
buttonHeight);
|
|
441
|
-
this.isExpanded = true;
|
|
442
|
-
this.dropdownEl.classList.remove("hidden");
|
|
443
|
-
window.requestAnimationFrame(() => {
|
|
444
|
-
switch (optionToSelect) {
|
|
445
|
-
case "next":
|
|
446
|
-
// search variant focuses search field
|
|
447
|
-
// all others focus option "after" last selected option (this can be the first option)
|
|
448
|
-
if (this.search) {
|
|
449
|
-
this.searchFieldEl.focus();
|
|
450
|
-
this.
|
|
451
|
-
}
|
|
452
|
-
else {
|
|
453
|
-
this.moveDown(this.visibleOptionEls.filter((x) => x.selected).slice(-1)[0]);
|
|
454
|
-
}
|
|
455
|
-
break;
|
|
456
|
-
case "previous":
|
|
457
|
-
// search variant focuses last option
|
|
458
|
-
// all others focus option "above" first selected option (this can be the last option)
|
|
459
|
-
if (this.search) {
|
|
460
|
-
this.focusOption(this.visibleOptionEls[this.visibleOptionEls.length - 1]);
|
|
461
|
-
}
|
|
462
|
-
else {
|
|
463
|
-
this.moveUp(this.visibleOptionEls.filter((x) => x.selected)[0]);
|
|
464
|
-
}
|
|
465
|
-
break;
|
|
466
|
-
default:
|
|
467
|
-
// search variant focuses search field
|
|
468
|
-
// all others focus the selected option
|
|
469
|
-
// if no option is selected (empty multiselect), focuses first option
|
|
470
|
-
if (this.search) {
|
|
471
|
-
this.searchFieldEl.focus();
|
|
472
|
-
this.
|
|
473
|
-
}
|
|
474
|
-
else if (this.selectedOptions.length > 0) {
|
|
475
|
-
this.focusOption(this.visibleOptionEls.filter((x) => x.selected)[0]);
|
|
476
|
-
}
|
|
477
|
-
else {
|
|
478
|
-
this.focusOption(this.visibleOptionEls[0]);
|
|
479
|
-
}
|
|
480
|
-
break;
|
|
481
|
-
}
|
|
482
|
-
});
|
|
483
|
-
}
|
|
484
|
-
}
|
|
485
|
-
close(returnFocus = true) {
|
|
486
|
-
if (this.isExpanded) {
|
|
487
|
-
this.isExpanded = false;
|
|
488
|
-
this.allOptionEls.map((i) => (i.focused = false));
|
|
489
|
-
window.setTimeout(() => {
|
|
490
|
-
this.dropdownEl.classList.add("hidden");
|
|
491
|
-
if (this.multiple) {
|
|
492
|
-
this.updateOptionVisibility();
|
|
493
|
-
}
|
|
494
|
-
// clear search field, reset filtered / bolded state of wm-options
|
|
495
|
-
if (this.search) {
|
|
496
|
-
this.searchFieldEl.value = "";
|
|
497
|
-
this.wmSelectSearchChanged.emit({ searchTerm: this.searchTerm });
|
|
498
|
-
}
|
|
499
|
-
// Returns focus to button after popup closes (no need if user is tabbing)
|
|
500
|
-
// Delay is necessary for screenreader to get new expanded state before focus
|
|
501
|
-
// window.requestAnimationFrame is probably enough, but since we are already using setTimeout it may as well be here
|
|
502
|
-
// also UX wise, it makes sense for the button to only be focused after the animation is complete
|
|
503
|
-
if (returnFocus) {
|
|
504
|
-
this.el.focus();
|
|
505
|
-
}
|
|
506
|
-
}, 150);
|
|
507
|
-
}
|
|
508
|
-
}
|
|
509
|
-
updateOptionVisibility() {
|
|
510
|
-
// this runs for search multiselects, where selected options are rendered at the top of the dropdown list
|
|
511
|
-
// slotted wm-options are hidden if selected, and duplicate wm-options are made visible if selected
|
|
512
|
-
this.childOptions.forEach((option, idx) => {
|
|
513
|
-
const duplicateOption = this.duplicateOptions[idx];
|
|
514
|
-
if (option.selected) {
|
|
515
|
-
option.classList.add("hidden");
|
|
516
|
-
duplicateOption.classList.remove("hidden");
|
|
517
|
-
}
|
|
518
|
-
else {
|
|
519
|
-
option.classList.remove("hidden");
|
|
520
|
-
duplicateOption.classList.add("hidden");
|
|
521
|
-
}
|
|
522
|
-
});
|
|
523
|
-
}
|
|
524
|
-
focusOption(option) {
|
|
525
|
-
this.allOptionEls.forEach((i) => (i.focused = i === option));
|
|
526
|
-
option.focus();
|
|
527
|
-
}
|
|
528
|
-
selectOption(option) {
|
|
529
|
-
// this function does not necessarily change an options selected property to true
|
|
530
|
-
// in cases of multiselect, it toggles the selected property
|
|
531
|
-
if (option.classList.contains("duplicate")) {
|
|
532
|
-
// if clicking on a duplicate option, toggle selected property of real one, then rerender
|
|
533
|
-
const correspondingOption = this.findCorrespondingOption(option);
|
|
534
|
-
correspondingOption.selected = !correspondingOption.selected;
|
|
535
|
-
index.forceUpdate(this.el);
|
|
536
|
-
}
|
|
537
|
-
else if (this.multiple) {
|
|
538
|
-
option.selected = !option.selected;
|
|
539
|
-
}
|
|
540
|
-
else {
|
|
541
|
-
this.allOptionEls.forEach((x) => (x.selected = x === option));
|
|
542
|
-
}
|
|
543
|
-
this.setButtonText();
|
|
544
|
-
}
|
|
545
|
-
findAndFocusOption(ev) {
|
|
546
|
-
const character = ev.detail.toUpperCase();
|
|
547
|
-
if (!this.keysSoFar) {
|
|
548
|
-
for (var i = 0; i < this.allOptionEls.length; i++) {
|
|
549
|
-
if (this.allOptionEls[i].focused) {
|
|
550
|
-
this.searchIndex = i;
|
|
551
|
-
}
|
|
552
|
-
}
|
|
553
|
-
}
|
|
554
|
-
this.keysSoFar += character;
|
|
555
|
-
this.clearKeysSoFarAfterDelay();
|
|
556
|
-
var nextMatch = this.findMatchInRange(this.allOptionEls, this.searchIndex + 1, this.allOptionEls.length);
|
|
557
|
-
if (!nextMatch) {
|
|
558
|
-
nextMatch = this.findMatchInRange(this.allOptionEls, 0, this.searchIndex);
|
|
559
|
-
}
|
|
560
|
-
if (nextMatch) {
|
|
561
|
-
this.focusOption(nextMatch);
|
|
562
|
-
}
|
|
563
|
-
}
|
|
564
|
-
clearKeysSoFarAfterDelay() {
|
|
565
|
-
if (this.keyClear) {
|
|
566
|
-
window.clearTimeout(this.keyClear);
|
|
567
|
-
this.keyClear = null;
|
|
568
|
-
}
|
|
569
|
-
this.keyClear = window.setTimeout(function () {
|
|
570
|
-
this.keysSoFar = "";
|
|
571
|
-
this.keyClear = null;
|
|
572
|
-
}.bind(this), 500);
|
|
573
|
-
}
|
|
574
|
-
findMatchInRange(list, startIndex, endIndex) {
|
|
575
|
-
// Find the first option starting with the keysSoFar substring, searching in
|
|
576
|
-
// the specified range of options
|
|
577
|
-
for (var n = startIndex; n < endIndex; n++) {
|
|
578
|
-
var label = list[n].textContent;
|
|
579
|
-
if (label && label.toUpperCase().indexOf(this.keysSoFar) === 0) {
|
|
580
|
-
return list[n];
|
|
581
|
-
}
|
|
582
|
-
}
|
|
583
|
-
return null;
|
|
584
|
-
}
|
|
585
|
-
findCorrespondingOption(el) {
|
|
586
|
-
// if duplicate, returns the child wm-option
|
|
587
|
-
// if child wm-option, returns duplicate
|
|
588
|
-
const isDuplicate = el.classList.contains("duplicate");
|
|
589
|
-
return isDuplicate
|
|
590
|
-
? this.childOptions[this.duplicateOptions.indexOf(el)]
|
|
591
|
-
: this.duplicateOptions[this.childOptions.indexOf(el)];
|
|
592
|
-
}
|
|
593
|
-
isElOrChild(el) {
|
|
594
|
-
var _a;
|
|
595
|
-
// determines whether or not the element is the component, a child of the component, or exists within the component's shadowroot
|
|
596
|
-
return el === this.el || this.el.contains(el) || ((_a = this.el.shadowRoot) === null || _a === void 0 ? void 0 : _a.contains(el));
|
|
597
|
-
}
|
|
598
|
-
exposeErrors() {
|
|
599
|
-
// When the error changes, a new id is set for the error container and the button's aria-describedby attribute is updated accordingly. This is to make sure the screen reader announces teh updated errors in Firefox. See this longstanding bug: https://bugzilla.mozilla.org/show_bug.cgi?id=493683
|
|
600
|
-
const newId = functions.generateId();
|
|
601
|
-
this.errorContainer.id = `wm-errors-${newId}`;
|
|
602
|
-
this.buttonEl.setAttribute("aria-describedby", `wm-errors-${newId}`);
|
|
603
|
-
}
|
|
604
|
-
renderErrorContainer() {
|
|
605
|
-
return (index.h("div", { id: "wm-errors", class: this.errorMessage ? "error-message" : "", ref: (el) => (this.errorContainer = el), "aria-live": "assertive", "aria-atomic": "true" }, this.errorMessage));
|
|
606
|
-
}
|
|
607
|
-
handleComponentBlur(ev) {
|
|
608
|
-
// Do not close or emit custom blur event when blurring to an element inside (wm-option)
|
|
609
|
-
if (!this.el.contains(ev.relatedTarget)) {
|
|
610
|
-
this.close(false);
|
|
611
|
-
this.wmSelectBlurred.emit();
|
|
612
|
-
this.wmComponentBlurred.emit(); // deprecated
|
|
613
|
-
}
|
|
614
|
-
}
|
|
615
|
-
handleSearchFieldKeyDown(ev) {
|
|
616
|
-
switch (ev.key) {
|
|
617
|
-
case "ArrowDown":
|
|
618
|
-
ev.preventDefault();
|
|
619
|
-
if (this.visibleOptionEls.length) {
|
|
620
|
-
this.focusOption(this.visibleOptionEls[0]);
|
|
621
|
-
}
|
|
622
|
-
break;
|
|
623
|
-
case "ArrowUp":
|
|
624
|
-
ev.preventDefault();
|
|
625
|
-
if (this.visibleOptionEls.length) {
|
|
626
|
-
this.focusOption(this.visibleOptionEls[this.visibleOptionEls.length - 1]);
|
|
627
|
-
}
|
|
628
|
-
break;
|
|
629
|
-
case "Escape":
|
|
630
|
-
ev.preventDefault();
|
|
631
|
-
this.close();
|
|
632
|
-
break;
|
|
633
|
-
}
|
|
634
|
-
}
|
|
635
|
-
setButtonText() {
|
|
636
|
-
this.displayedOptions = this.childOptions
|
|
637
|
-
.filter((x) => x.selected)
|
|
638
|
-
.map((y) => (!y.classList.contains("hidden") ? y : this.findCorrespondingOption(y)));
|
|
639
|
-
// handle overflow + counter for multiselect
|
|
640
|
-
if (this.multiple) {
|
|
641
|
-
// this is a fixed measurement accounting for the max width of a 3 character overflow counter
|
|
642
|
-
const overflowCounterWidth = 38;
|
|
643
|
-
const computedStyle = window.getComputedStyle(this.buttonEl);
|
|
644
|
-
// there seems to be no quick way to get an elements width without padding, except for subtracting padding manually
|
|
645
|
-
const paddingLeft = parseInt(computedStyle.getPropertyValue("padding-left").slice(0, -2));
|
|
646
|
-
const paddingRight = parseInt(computedStyle.getPropertyValue("padding-right").slice(0, -2));
|
|
647
|
-
const availableSpace = this.buttonEl.clientWidth - paddingLeft - paddingRight - overflowCounterWidth;
|
|
648
|
-
let optionsWidths = this.displayedOptions.map((x) => x.shadowRoot.querySelector(".option-wrapper").clientWidth);
|
|
649
|
-
let optionsTotalWidth = optionsWidths.reduce((acc, x) => acc + x, 0);
|
|
650
|
-
this.overflowCount = 0;
|
|
651
|
-
while (optionsTotalWidth > availableSpace && this.displayedOptions.length > 1) {
|
|
652
|
-
this.overflowCount++;
|
|
653
|
-
optionsTotalWidth -= optionsWidths[optionsWidths.length - 1];
|
|
654
|
-
optionsWidths.pop();
|
|
655
|
-
this.displayedOptions.pop();
|
|
656
|
-
}
|
|
657
|
-
}
|
|
658
|
-
}
|
|
659
|
-
announce(message) {
|
|
660
|
-
// \u00A0 is a non-breaking space character, which causes the message to be read as a new one
|
|
661
|
-
if (this.liveRegionEl.textContent === message) {
|
|
662
|
-
message += "\u00A0";
|
|
663
|
-
}
|
|
664
|
-
this.announcement = message;
|
|
665
|
-
}
|
|
666
|
-
renderButtonText() {
|
|
667
|
-
if (this.multiple && this.displayedOptions.length < 1) {
|
|
668
|
-
return index.h("span", null, this.placeholder);
|
|
669
|
-
}
|
|
670
|
-
else if (this.multiple && this.allSelected && this.overflowCount > 0) {
|
|
671
|
-
return this.allSelectedMessage;
|
|
672
|
-
}
|
|
673
|
-
else {
|
|
674
|
-
return this.displayedOptions.map((x, idx) => (index.h("span", null, idx > 0 ? ", " : "", x.textContent)));
|
|
675
|
-
}
|
|
676
|
-
}
|
|
677
|
-
renderSubinfo() {
|
|
678
|
-
// multiselects cannot have subinfo for options
|
|
679
|
-
if (!this.multiple && this.selectedOptions.length > 0 && this.selectedOptions[0].subinfo) {
|
|
680
|
-
return index.h("span", { class: "subinfo" }, this.selectedOptions[0].subinfo);
|
|
681
|
-
}
|
|
682
|
-
}
|
|
683
|
-
renderOverflowCount() {
|
|
684
|
-
if (this.overflowCount > 0 && !this.allSelected) {
|
|
685
|
-
return (index.h("span", null, index.h("span", { class: "overflow-counter" }, "+", this.overflowCount)));
|
|
686
|
-
}
|
|
687
|
-
}
|
|
688
|
-
renderSearchField() {
|
|
689
|
-
return (index.h("div", { class: "search" }, index.h("div", { class: "searchfield-wrapper", ref: (el) => (this.searchFieldWrapperEl = el) }, index.h("div", { class: "icon" }), index.h("input", { ref: (el) => (this.searchFieldEl = el), class: "searchfield", role: "combobox", "aria-controls": `list-${this.uid}`, "aria-expanded": this.isExpanded ? "true" : "false", onKeyDown: (ev) => this.handleSearchFieldKeyDown(ev), onFocus: () => this.searchFieldWrapperEl.classList.add("focus"), onBlur: (ev) => this.handleSearchFieldBlur(ev), onInput: () => this.debouncedSearch(), placeholder: this.searchPlaceholder }))));
|
|
690
|
-
}
|
|
691
|
-
renderSearchFailedMessage() {
|
|
692
|
-
return index.h("div", { class: "search-results-message" }, this.noResultsFoundMessage);
|
|
693
|
-
}
|
|
694
|
-
renderDuplicateOptions() {
|
|
695
|
-
return Array.from(this.el.children).map((option) => {
|
|
696
|
-
return (index.h("wm-option", { class: "duplicate", selected: option.selected }, option.textContent));
|
|
697
|
-
});
|
|
698
|
-
}
|
|
699
|
-
render() {
|
|
700
|
-
const buttonProps = {
|
|
701
|
-
id: `selectbtn-${this.uid}`,
|
|
702
|
-
["disabled"]: this.disabled,
|
|
703
|
-
["aria-controls"]: `list-${this.uid}`,
|
|
704
|
-
["aria-labelledby"]: `label-${this.uid} selectbtn-${this.uid}`,
|
|
705
|
-
["aria-describedby"]: "wm-errors",
|
|
706
|
-
["aria-expanded"]: this.isExpanded ? "true" : "false",
|
|
707
|
-
onClick: () => (this.isExpanded ? this.close() : this.open()),
|
|
708
|
-
};
|
|
709
|
-
return (index.h(index.Host, { onBlur: (ev) => this.handleComponentBlur(ev) }, index.h("div", { class: `wrapper ${functions.getTextDir()} label-${this.labelPosition} ${this.invalid || this.errorMessage ? "invalid" : ""}` }, index.h("div", { class: "label-wrapper" }, index.h("label", { class: "label", id: `label-${this.uid}`, htmlFor: `selectbtn-${this.uid}` }, this.label,
|
|
710
|
-
// we can't use aria-required or required attributes because it's invalid on the elements we're using (button controlling a listbox)
|
|
711
|
-
this.requiredField ? (index.h("span", { class: "required" }, index.h("span", { class: "sr-only" }, this.requiredMessage), index.h("span", { "aria-hidden": "true" }, "*"))) : (""))), index.h("div", { class: "button-wrapper" }, index.h("button", Object.assign({}, buttonProps, { class: `displayedoption ${this.isTabbing ? "user-is-tabbing" : ""}`, ref: (el) => (this.buttonEl = el), onBlur: (ev) => this.handleButtonBlur(ev), onFocus: () => this.close() }), index.h("span", { class: this.selectedOptions.length > 0 && this.selectedOptions.filter((x) => x.subinfo).length > 0
|
|
712
|
-
? "overflowcontrol hassubinfo"
|
|
713
|
-
: "overflowcontrol" }, index.h("span", { class: "button-text" }, this.renderButtonText()), this.renderSubinfo()), this.renderOverflowCount()), index.h("div", { class: `dropdown ${this.isExpanded ? "open" : ""} ${this.openUp ? "upwards" : ""}`, ref: (el) => (this.dropdownEl = el) }, this.search && this.renderSearchField(), index.h("div", { id: `list-${this.uid}`, tabindex: -1, role: "listbox", "aria-multiselectable": this.multiple ? "true" : null, "aria-labelledby": `label-${this.uid}
|
|
714
|
-
}
|
|
715
|
-
get el() { return index.getElement(this); }
|
|
716
|
-
static get watchers() { return {
|
|
717
|
-
"errorMessage": ["exposeErrors"]
|
|
718
|
-
}; }
|
|
719
|
-
};
|
|
149
|
+
const Select = class {
|
|
150
|
+
constructor(hostRef) {
|
|
151
|
+
index.registerInstance(this, hostRef);
|
|
152
|
+
this.wmSelectDidLoad = index.createEvent(this, "wmSelectDidLoad", 7);
|
|
153
|
+
this.wmSelectBlurred = index.createEvent(this, "wmSelectBlurred", 7);
|
|
154
|
+
this.wmComponentBlurred = index.createEvent(this, "wmComponentBlurred", 7);
|
|
155
|
+
this.wmSelectSearchChanged = index.createEvent(this, "wmSelectSearchChanged", 7);
|
|
156
|
+
this.disabled = false;
|
|
157
|
+
this.invalid = false;
|
|
158
|
+
this.labelPosition = "top";
|
|
159
|
+
this.requiredField = false;
|
|
160
|
+
this.errorMessage = "";
|
|
161
|
+
this.multiple = false;
|
|
162
|
+
this.search = false;
|
|
163
|
+
this.placeholder = functions.intl.formatMessage({
|
|
164
|
+
id: "select.multiPlaceholder",
|
|
165
|
+
defaultMessage: "Make a selection",
|
|
166
|
+
description: "Placeholder text. Use imperative",
|
|
167
|
+
});
|
|
168
|
+
this.searchPlaceholder = functions.intl.formatMessage({
|
|
169
|
+
id: "select.searchPlaceholder",
|
|
170
|
+
defaultMessage: "Search",
|
|
171
|
+
description: "Placeholder text. Use imperative",
|
|
172
|
+
});
|
|
173
|
+
this.allSelectedMessage = functions.intl.formatMessage({
|
|
174
|
+
id: "select.allSelected",
|
|
175
|
+
defaultMessage: "All selected",
|
|
176
|
+
description: "Text displayed when all options are selected",
|
|
177
|
+
});
|
|
178
|
+
this.requiredMessage = functions.intl.formatMessage({
|
|
179
|
+
id: "global.requiredField",
|
|
180
|
+
defaultMessage: "required field",
|
|
181
|
+
});
|
|
182
|
+
this.isTabbing = false;
|
|
183
|
+
this.isExpanded = false;
|
|
184
|
+
this.announcement = "";
|
|
185
|
+
this.keysSoFar = "";
|
|
186
|
+
this.searchIndex = 0;
|
|
187
|
+
this.keyClear = null;
|
|
188
|
+
this.openUp = false;
|
|
189
|
+
//////////////////////////////////////
|
|
190
|
+
//////////////////////////////////////
|
|
191
|
+
// for multiselect button text
|
|
192
|
+
this.overflowCount = 0;
|
|
193
|
+
this.displayedOptions = [];
|
|
194
|
+
this.debouncedSearch = functions.debounce(() => {
|
|
195
|
+
this.wmSelectSearchChanged.emit({ searchTerm: this.searchTerm });
|
|
196
|
+
if (this.filteredOptions.length) {
|
|
197
|
+
this.announce(this.resultsFoundMessage);
|
|
198
|
+
}
|
|
199
|
+
else {
|
|
200
|
+
this.announce(this.noResultsFoundMessage);
|
|
201
|
+
}
|
|
202
|
+
}, 150);
|
|
203
|
+
}
|
|
204
|
+
get childOptions() {
|
|
205
|
+
return Array.from(this.el.querySelectorAll("wm-option"));
|
|
206
|
+
}
|
|
207
|
+
get duplicateOptions() {
|
|
208
|
+
return Array.from(this.el.shadowRoot.querySelectorAll("wm-option"));
|
|
209
|
+
}
|
|
210
|
+
get allOptionEls() {
|
|
211
|
+
// this includes both slotted wm-options and internally created wm-options
|
|
212
|
+
return this.duplicateOptions.concat(this.childOptions);
|
|
213
|
+
}
|
|
214
|
+
get visibleOptionEls() {
|
|
215
|
+
return this.allOptionEls.filter((option) => !option.classList.contains("hidden") && !option.classList.contains("filtered-out"));
|
|
216
|
+
}
|
|
217
|
+
//////////////////////////////////////
|
|
218
|
+
// for search variants
|
|
219
|
+
get searchTerm() {
|
|
220
|
+
return this.searchFieldEl ? this.searchFieldEl.value : "";
|
|
221
|
+
}
|
|
222
|
+
get filteredOptions() {
|
|
223
|
+
return this.childOptions.filter((option) => { var _a; return (_a = option.textContent) === null || _a === void 0 ? void 0 : _a.toLowerCase().includes(this.searchTerm.toLowerCase()); });
|
|
224
|
+
}
|
|
225
|
+
get selectedOptions() {
|
|
226
|
+
return Array.from(this.el.querySelectorAll("wm-option")).filter((x) => x.selected);
|
|
227
|
+
}
|
|
228
|
+
get allSelected() {
|
|
229
|
+
return this.childOptions.filter((option) => option.selected).length === this.childOptions.length;
|
|
230
|
+
}
|
|
231
|
+
//////////////////////////////////////
|
|
232
|
+
get resultsFoundMessage() {
|
|
233
|
+
return functions.intl.formatMessage({
|
|
234
|
+
id: "select.searchResultsFound",
|
|
235
|
+
defaultMessage: "{numResults, plural, one {1 option found} other {# options found}}",
|
|
236
|
+
description: "The message read by the screen reader, indicating how many results a search returned",
|
|
237
|
+
}, { numResults: this.filteredOptions.length });
|
|
238
|
+
}
|
|
239
|
+
get noResultsFoundMessage() {
|
|
240
|
+
return functions.intl.formatMessage({
|
|
241
|
+
id: "select.noSearchResults",
|
|
242
|
+
defaultMessage: "No results found. Please try your search again.",
|
|
243
|
+
description: "The message displayed when no options pass the search filter",
|
|
244
|
+
});
|
|
245
|
+
}
|
|
246
|
+
toggleTabbingOn() {
|
|
247
|
+
this.isTabbing = true;
|
|
248
|
+
}
|
|
249
|
+
toggleTabbingOff() {
|
|
250
|
+
this.isTabbing = false;
|
|
251
|
+
}
|
|
252
|
+
handleOptionSelection(ev) {
|
|
253
|
+
if (!this.multiple) {
|
|
254
|
+
// ensure only one option is selected at a time, unselect all other options
|
|
255
|
+
this.childOptions.forEach((option) => (option.selected = option === ev.detail));
|
|
256
|
+
}
|
|
257
|
+
this.focusOption(ev.detail);
|
|
258
|
+
this.selectOption(ev.detail);
|
|
259
|
+
if (!this.multiple) {
|
|
260
|
+
this.close();
|
|
261
|
+
}
|
|
262
|
+
}
|
|
263
|
+
handleChildEnter() {
|
|
264
|
+
// only occurs for multiselects. handle the click, then close
|
|
265
|
+
this.close();
|
|
266
|
+
}
|
|
267
|
+
handleChildUp(ev) {
|
|
268
|
+
this.moveUp(ev.detail);
|
|
269
|
+
}
|
|
270
|
+
handleChildDown(ev) {
|
|
271
|
+
this.moveDown(ev.detail);
|
|
272
|
+
}
|
|
273
|
+
moveToFirstOption() {
|
|
274
|
+
this.focusOption(this.visibleOptionEls[0]);
|
|
275
|
+
}
|
|
276
|
+
moveToLastOption() {
|
|
277
|
+
this.focusOption(this.visibleOptionEls[this.visibleOptionEls.length - 1]);
|
|
278
|
+
}
|
|
279
|
+
closePopupOnEscape() {
|
|
280
|
+
this.close();
|
|
281
|
+
}
|
|
282
|
+
handleOptionBlur(ev) {
|
|
283
|
+
// if the Option is blurred to something other than the component emit a blur event with the appropriate relatedTarget
|
|
284
|
+
// keeps our component's blur events accurate, and closes when focusing browser address bar
|
|
285
|
+
if (!this.isElOrChild(ev.detail.relatedTarget)) {
|
|
286
|
+
const event = new CustomEvent("blur");
|
|
287
|
+
// @ts-ignore
|
|
288
|
+
event.relatedTarget = ev.detail.relatedTarget;
|
|
289
|
+
this.el.dispatchEvent(event);
|
|
290
|
+
}
|
|
291
|
+
}
|
|
292
|
+
handleClick(ev) {
|
|
293
|
+
if (!this.isElOrChild(ev.target)) {
|
|
294
|
+
this.close();
|
|
295
|
+
}
|
|
296
|
+
}
|
|
297
|
+
handleButtonBlur(ev) {
|
|
298
|
+
if (this.isElOrChild(ev.relatedTarget)) {
|
|
299
|
+
// do not emit a blur event when opening the dropdown and focusing the Options
|
|
300
|
+
ev.stopPropagation();
|
|
301
|
+
}
|
|
302
|
+
}
|
|
303
|
+
handleSearchFieldBlur(ev) {
|
|
304
|
+
this.searchFieldWrapperEl.classList.remove("focus");
|
|
305
|
+
if (this.isElOrChild(ev.relatedTarget)) {
|
|
306
|
+
// do not emit a blur event when moving from searchfield to options
|
|
307
|
+
ev.stopPropagation();
|
|
308
|
+
}
|
|
309
|
+
}
|
|
310
|
+
handleKey(ev) {
|
|
311
|
+
switch (ev.key) {
|
|
312
|
+
case "ArrowDown":
|
|
313
|
+
if (this.isExpanded === false) {
|
|
314
|
+
ev.preventDefault();
|
|
315
|
+
this.open("next");
|
|
316
|
+
}
|
|
317
|
+
break;
|
|
318
|
+
case "ArrowUp":
|
|
319
|
+
if (this.isExpanded === false) {
|
|
320
|
+
ev.preventDefault();
|
|
321
|
+
this.open("previous");
|
|
322
|
+
}
|
|
323
|
+
break;
|
|
324
|
+
}
|
|
325
|
+
}
|
|
326
|
+
componentWillLoad() {
|
|
327
|
+
if (!this.label) {
|
|
328
|
+
console.error("For accessibility purposes, this component requires a label (even if `label-position` is set to `none`).");
|
|
329
|
+
}
|
|
330
|
+
this.el.focus = function () {
|
|
331
|
+
if (!this.disabled) {
|
|
332
|
+
this.shadowRoot.querySelector("button").focus();
|
|
333
|
+
}
|
|
334
|
+
};
|
|
335
|
+
this.uid = this.el.id ? this.el.id : functions.generateId();
|
|
336
|
+
if (document.body.classList.contains("wmcl-user-is-tabbing")) {
|
|
337
|
+
this.toggleTabbingOn();
|
|
338
|
+
}
|
|
339
|
+
// set initial selections
|
|
340
|
+
if (this.selectedOptions.length > 0) {
|
|
341
|
+
this.selectedOptions.forEach((x) => {
|
|
342
|
+
this.displayedOptions.push(x);
|
|
343
|
+
});
|
|
344
|
+
// single Select only, pre-select if no default option from dev
|
|
345
|
+
}
|
|
346
|
+
else if (!this.multiple) {
|
|
347
|
+
this.selectOption(this.allOptionEls[0]);
|
|
348
|
+
}
|
|
349
|
+
}
|
|
350
|
+
componentDidLoad() {
|
|
351
|
+
this.wmSelectDidLoad.emit();
|
|
352
|
+
if (this.multiple && this.allOptionEls.filter((x) => x.subinfo).length > 0) {
|
|
353
|
+
throw new Error("wm-select with the multiple prop cannot have subinfo options");
|
|
354
|
+
}
|
|
355
|
+
if (this.multiple && !this.placeholder) {
|
|
356
|
+
throw new Error("wm-select with the multiple prop needs to also use the placeholder prop.");
|
|
357
|
+
}
|
|
358
|
+
this.dropdownEl.classList.add("hidden");
|
|
359
|
+
index.forceUpdate(this.el);
|
|
360
|
+
// Dev can overwrite the max-height rule set in the Sass file
|
|
361
|
+
if (this.maxHeight) {
|
|
362
|
+
this.listboxEl.style.maxHeight = this.maxHeight;
|
|
363
|
+
}
|
|
364
|
+
const mutationObserver = new MutationObserver(() => this.handleSelectedMutation());
|
|
365
|
+
mutationObserver.observe(this.el, {
|
|
366
|
+
attributes: true,
|
|
367
|
+
attributeFilter: ["selected"],
|
|
368
|
+
subtree: true,
|
|
369
|
+
});
|
|
370
|
+
if (this.multiple) {
|
|
371
|
+
this.updateOptionVisibility();
|
|
372
|
+
}
|
|
373
|
+
this.setButtonText();
|
|
374
|
+
}
|
|
375
|
+
componentWillUpdate() {
|
|
376
|
+
if (this.multiple) {
|
|
377
|
+
// find last visible duplicate option and apply .last class
|
|
378
|
+
const visibleDuplicateOptions = this.visibleOptionEls.filter((option) => option.classList.contains("duplicate"));
|
|
379
|
+
visibleDuplicateOptions.forEach((option, idx) => {
|
|
380
|
+
if (idx === visibleDuplicateOptions.length - 1) {
|
|
381
|
+
option.classList.add("last");
|
|
382
|
+
}
|
|
383
|
+
else {
|
|
384
|
+
option.classList.remove("last");
|
|
385
|
+
}
|
|
386
|
+
});
|
|
387
|
+
}
|
|
388
|
+
}
|
|
389
|
+
handleSelectedMutation() {
|
|
390
|
+
// dispatch change event after selected options change
|
|
391
|
+
// mutation observer prevents emitting event when an already selected option is selected again
|
|
392
|
+
const event = new CustomEvent("change");
|
|
393
|
+
// @ts-ignore
|
|
394
|
+
this.el.dispatchEvent(event);
|
|
395
|
+
}
|
|
396
|
+
moveUp(el) {
|
|
397
|
+
const prevEl = this.visibleOptionEls[this.visibleOptionEls.indexOf(el) - 1];
|
|
398
|
+
if (prevEl) {
|
|
399
|
+
// scroll option to top of dropdown if partially obscured / out of view
|
|
400
|
+
if (prevEl.getBoundingClientRect().top < this.listboxEl.getBoundingClientRect().top) {
|
|
401
|
+
this.scrollOptionToTop(prevEl);
|
|
402
|
+
}
|
|
403
|
+
this.focusOption(prevEl);
|
|
404
|
+
}
|
|
405
|
+
else if (this.search) {
|
|
406
|
+
// if top of list and search variant, focus search field
|
|
407
|
+
this.searchFieldEl.focus();
|
|
408
|
+
}
|
|
409
|
+
else {
|
|
410
|
+
// if top of list, focus last element
|
|
411
|
+
this.focusOption(this.visibleOptionEls[this.visibleOptionEls.length - 1]);
|
|
412
|
+
}
|
|
413
|
+
}
|
|
414
|
+
moveDown(el) {
|
|
415
|
+
const nextEl = this.visibleOptionEls[this.visibleOptionEls.indexOf(el) + 1];
|
|
416
|
+
if (nextEl) {
|
|
417
|
+
// scroll option to bottom of dropdown if partially obscured / out of view
|
|
418
|
+
if (nextEl.getBoundingClientRect().bottom > this.listboxEl.getBoundingClientRect().bottom) {
|
|
419
|
+
this.scrollOptionToBottom(nextEl);
|
|
420
|
+
}
|
|
421
|
+
this.focusOption(nextEl);
|
|
422
|
+
}
|
|
423
|
+
else {
|
|
424
|
+
// if end of list, focus first option in all variants
|
|
425
|
+
this.focusOption(this.visibleOptionEls[0]);
|
|
426
|
+
}
|
|
427
|
+
}
|
|
428
|
+
scrollOptionToTop(option) {
|
|
429
|
+
this.listboxEl.scrollTop =
|
|
430
|
+
option.getBoundingClientRect().top - this.listboxEl.getBoundingClientRect().top + this.listboxEl.scrollTop;
|
|
431
|
+
}
|
|
432
|
+
scrollOptionToBottom(option) {
|
|
433
|
+
this.listboxEl.scrollTop =
|
|
434
|
+
option.getBoundingClientRect().bottom -
|
|
435
|
+
this.listboxEl.getBoundingClientRect().top +
|
|
436
|
+
this.listboxEl.scrollTop -
|
|
437
|
+
this.listboxEl.offsetHeight;
|
|
438
|
+
}
|
|
439
|
+
open(optionToSelect) {
|
|
440
|
+
if (!this.disabled) {
|
|
441
|
+
const elHeight = this.el.clientHeight;
|
|
442
|
+
const buttonHeight = this.buttonEl.clientHeight;
|
|
443
|
+
this.openUp = functions.shouldOpenUp(this.el, this.dropdownEl,
|
|
444
|
+
// when opening up, dropdown covers both label and button
|
|
445
|
+
elHeight,
|
|
446
|
+
// when opening down, dropdown covers only the button
|
|
447
|
+
buttonHeight);
|
|
448
|
+
this.isExpanded = true;
|
|
449
|
+
this.dropdownEl.classList.remove("hidden");
|
|
450
|
+
window.requestAnimationFrame(() => {
|
|
451
|
+
switch (optionToSelect) {
|
|
452
|
+
case "next":
|
|
453
|
+
// search variant focuses search field
|
|
454
|
+
// all others focus option "after" last selected option (this can be the first option)
|
|
455
|
+
if (this.search) {
|
|
456
|
+
this.searchFieldEl.focus();
|
|
457
|
+
this.listboxEl.scrollTop = 0;
|
|
458
|
+
}
|
|
459
|
+
else {
|
|
460
|
+
this.moveDown(this.visibleOptionEls.filter((x) => x.selected).slice(-1)[0]);
|
|
461
|
+
}
|
|
462
|
+
break;
|
|
463
|
+
case "previous":
|
|
464
|
+
// search variant focuses last option
|
|
465
|
+
// all others focus option "above" first selected option (this can be the last option)
|
|
466
|
+
if (this.search) {
|
|
467
|
+
this.focusOption(this.visibleOptionEls[this.visibleOptionEls.length - 1]);
|
|
468
|
+
}
|
|
469
|
+
else {
|
|
470
|
+
this.moveUp(this.visibleOptionEls.filter((x) => x.selected)[0]);
|
|
471
|
+
}
|
|
472
|
+
break;
|
|
473
|
+
default:
|
|
474
|
+
// search variant focuses search field
|
|
475
|
+
// all others focus the selected option
|
|
476
|
+
// if no option is selected (empty multiselect), focuses first option
|
|
477
|
+
if (this.search) {
|
|
478
|
+
this.searchFieldEl.focus();
|
|
479
|
+
this.listboxEl.scrollTop = 0;
|
|
480
|
+
}
|
|
481
|
+
else if (this.selectedOptions.length > 0) {
|
|
482
|
+
this.focusOption(this.visibleOptionEls.filter((x) => x.selected)[0]);
|
|
483
|
+
}
|
|
484
|
+
else {
|
|
485
|
+
this.focusOption(this.visibleOptionEls[0]);
|
|
486
|
+
}
|
|
487
|
+
break;
|
|
488
|
+
}
|
|
489
|
+
});
|
|
490
|
+
}
|
|
491
|
+
}
|
|
492
|
+
close(returnFocus = true) {
|
|
493
|
+
if (this.isExpanded) {
|
|
494
|
+
this.isExpanded = false;
|
|
495
|
+
this.allOptionEls.map((i) => (i.focused = false));
|
|
496
|
+
window.setTimeout(() => {
|
|
497
|
+
this.dropdownEl.classList.add("hidden");
|
|
498
|
+
if (this.multiple) {
|
|
499
|
+
this.updateOptionVisibility();
|
|
500
|
+
}
|
|
501
|
+
// clear search field, reset filtered / bolded state of wm-options
|
|
502
|
+
if (this.search) {
|
|
503
|
+
this.searchFieldEl.value = "";
|
|
504
|
+
this.wmSelectSearchChanged.emit({ searchTerm: this.searchTerm });
|
|
505
|
+
}
|
|
506
|
+
// Returns focus to button after popup closes (no need if user is tabbing)
|
|
507
|
+
// Delay is necessary for screenreader to get new expanded state before focus
|
|
508
|
+
// window.requestAnimationFrame is probably enough, but since we are already using setTimeout it may as well be here
|
|
509
|
+
// also UX wise, it makes sense for the button to only be focused after the animation is complete
|
|
510
|
+
if (returnFocus) {
|
|
511
|
+
this.el.focus();
|
|
512
|
+
}
|
|
513
|
+
}, 150);
|
|
514
|
+
}
|
|
515
|
+
}
|
|
516
|
+
updateOptionVisibility() {
|
|
517
|
+
// this runs for search multiselects, where selected options are rendered at the top of the dropdown list
|
|
518
|
+
// slotted wm-options are hidden if selected, and duplicate wm-options are made visible if selected
|
|
519
|
+
this.childOptions.forEach((option, idx) => {
|
|
520
|
+
const duplicateOption = this.duplicateOptions[idx];
|
|
521
|
+
if (option.selected) {
|
|
522
|
+
option.classList.add("hidden");
|
|
523
|
+
duplicateOption.classList.remove("hidden");
|
|
524
|
+
}
|
|
525
|
+
else {
|
|
526
|
+
option.classList.remove("hidden");
|
|
527
|
+
duplicateOption.classList.add("hidden");
|
|
528
|
+
}
|
|
529
|
+
});
|
|
530
|
+
}
|
|
531
|
+
focusOption(option) {
|
|
532
|
+
this.allOptionEls.forEach((i) => (i.focused = i === option));
|
|
533
|
+
option.focus();
|
|
534
|
+
}
|
|
535
|
+
selectOption(option) {
|
|
536
|
+
// this function does not necessarily change an options selected property to true
|
|
537
|
+
// in cases of multiselect, it toggles the selected property
|
|
538
|
+
if (option.classList.contains("duplicate")) {
|
|
539
|
+
// if clicking on a duplicate option, toggle selected property of real one, then rerender
|
|
540
|
+
const correspondingOption = this.findCorrespondingOption(option);
|
|
541
|
+
correspondingOption.selected = !correspondingOption.selected;
|
|
542
|
+
index.forceUpdate(this.el);
|
|
543
|
+
}
|
|
544
|
+
else if (this.multiple) {
|
|
545
|
+
option.selected = !option.selected;
|
|
546
|
+
}
|
|
547
|
+
else {
|
|
548
|
+
this.allOptionEls.forEach((x) => (x.selected = x === option));
|
|
549
|
+
}
|
|
550
|
+
this.setButtonText();
|
|
551
|
+
}
|
|
552
|
+
findAndFocusOption(ev) {
|
|
553
|
+
const character = ev.detail.toUpperCase();
|
|
554
|
+
if (!this.keysSoFar) {
|
|
555
|
+
for (var i = 0; i < this.allOptionEls.length; i++) {
|
|
556
|
+
if (this.allOptionEls[i].focused) {
|
|
557
|
+
this.searchIndex = i;
|
|
558
|
+
}
|
|
559
|
+
}
|
|
560
|
+
}
|
|
561
|
+
this.keysSoFar += character;
|
|
562
|
+
this.clearKeysSoFarAfterDelay();
|
|
563
|
+
var nextMatch = this.findMatchInRange(this.allOptionEls, this.searchIndex + 1, this.allOptionEls.length);
|
|
564
|
+
if (!nextMatch) {
|
|
565
|
+
nextMatch = this.findMatchInRange(this.allOptionEls, 0, this.searchIndex);
|
|
566
|
+
}
|
|
567
|
+
if (nextMatch) {
|
|
568
|
+
this.focusOption(nextMatch);
|
|
569
|
+
}
|
|
570
|
+
}
|
|
571
|
+
clearKeysSoFarAfterDelay() {
|
|
572
|
+
if (this.keyClear) {
|
|
573
|
+
window.clearTimeout(this.keyClear);
|
|
574
|
+
this.keyClear = null;
|
|
575
|
+
}
|
|
576
|
+
this.keyClear = window.setTimeout(function () {
|
|
577
|
+
this.keysSoFar = "";
|
|
578
|
+
this.keyClear = null;
|
|
579
|
+
}.bind(this), 500);
|
|
580
|
+
}
|
|
581
|
+
findMatchInRange(list, startIndex, endIndex) {
|
|
582
|
+
// Find the first option starting with the keysSoFar substring, searching in
|
|
583
|
+
// the specified range of options
|
|
584
|
+
for (var n = startIndex; n < endIndex; n++) {
|
|
585
|
+
var label = list[n].textContent;
|
|
586
|
+
if (label && label.toUpperCase().indexOf(this.keysSoFar) === 0) {
|
|
587
|
+
return list[n];
|
|
588
|
+
}
|
|
589
|
+
}
|
|
590
|
+
return null;
|
|
591
|
+
}
|
|
592
|
+
findCorrespondingOption(el) {
|
|
593
|
+
// if duplicate, returns the child wm-option
|
|
594
|
+
// if child wm-option, returns duplicate
|
|
595
|
+
const isDuplicate = el.classList.contains("duplicate");
|
|
596
|
+
return isDuplicate
|
|
597
|
+
? this.childOptions[this.duplicateOptions.indexOf(el)]
|
|
598
|
+
: this.duplicateOptions[this.childOptions.indexOf(el)];
|
|
599
|
+
}
|
|
600
|
+
isElOrChild(el) {
|
|
601
|
+
var _a;
|
|
602
|
+
// determines whether or not the element is the component, a child of the component, or exists within the component's shadowroot
|
|
603
|
+
return el === this.el || this.el.contains(el) || ((_a = this.el.shadowRoot) === null || _a === void 0 ? void 0 : _a.contains(el));
|
|
604
|
+
}
|
|
605
|
+
exposeErrors() {
|
|
606
|
+
// When the error changes, a new id is set for the error container and the button's aria-describedby attribute is updated accordingly. This is to make sure the screen reader announces teh updated errors in Firefox. See this longstanding bug: https://bugzilla.mozilla.org/show_bug.cgi?id=493683
|
|
607
|
+
const newId = functions.generateId();
|
|
608
|
+
this.errorContainer.id = `wm-errors-${newId}`;
|
|
609
|
+
this.buttonEl.setAttribute("aria-describedby", `wm-errors-${newId}`);
|
|
610
|
+
}
|
|
611
|
+
renderErrorContainer() {
|
|
612
|
+
return (index.h("div", { id: "wm-errors", class: this.errorMessage ? "error-message" : "", ref: (el) => (this.errorContainer = el), "aria-live": "assertive", "aria-atomic": "true" }, this.errorMessage));
|
|
613
|
+
}
|
|
614
|
+
handleComponentBlur(ev) {
|
|
615
|
+
// Do not close or emit custom blur event when blurring to an element inside (wm-option)
|
|
616
|
+
if (!this.el.contains(ev.relatedTarget)) {
|
|
617
|
+
this.close(false);
|
|
618
|
+
this.wmSelectBlurred.emit();
|
|
619
|
+
this.wmComponentBlurred.emit(); // deprecated
|
|
620
|
+
}
|
|
621
|
+
}
|
|
622
|
+
handleSearchFieldKeyDown(ev) {
|
|
623
|
+
switch (ev.key) {
|
|
624
|
+
case "ArrowDown":
|
|
625
|
+
ev.preventDefault();
|
|
626
|
+
if (this.visibleOptionEls.length) {
|
|
627
|
+
this.focusOption(this.visibleOptionEls[0]);
|
|
628
|
+
}
|
|
629
|
+
break;
|
|
630
|
+
case "ArrowUp":
|
|
631
|
+
ev.preventDefault();
|
|
632
|
+
if (this.visibleOptionEls.length) {
|
|
633
|
+
this.focusOption(this.visibleOptionEls[this.visibleOptionEls.length - 1]);
|
|
634
|
+
}
|
|
635
|
+
break;
|
|
636
|
+
case "Escape":
|
|
637
|
+
ev.preventDefault();
|
|
638
|
+
this.close();
|
|
639
|
+
break;
|
|
640
|
+
}
|
|
641
|
+
}
|
|
642
|
+
setButtonText() {
|
|
643
|
+
this.displayedOptions = this.childOptions
|
|
644
|
+
.filter((x) => x.selected)
|
|
645
|
+
.map((y) => (!y.classList.contains("hidden") ? y : this.findCorrespondingOption(y)));
|
|
646
|
+
// handle overflow + counter for multiselect
|
|
647
|
+
if (this.multiple) {
|
|
648
|
+
// this is a fixed measurement accounting for the max width of a 3 character overflow counter
|
|
649
|
+
const overflowCounterWidth = 38;
|
|
650
|
+
const computedStyle = window.getComputedStyle(this.buttonEl);
|
|
651
|
+
// there seems to be no quick way to get an elements width without padding, except for subtracting padding manually
|
|
652
|
+
const paddingLeft = parseInt(computedStyle.getPropertyValue("padding-left").slice(0, -2));
|
|
653
|
+
const paddingRight = parseInt(computedStyle.getPropertyValue("padding-right").slice(0, -2));
|
|
654
|
+
const availableSpace = this.buttonEl.clientWidth - paddingLeft - paddingRight - overflowCounterWidth;
|
|
655
|
+
let optionsWidths = this.displayedOptions.map((x) => x.shadowRoot.querySelector(".option-wrapper").clientWidth);
|
|
656
|
+
let optionsTotalWidth = optionsWidths.reduce((acc, x) => acc + x, 0);
|
|
657
|
+
this.overflowCount = 0;
|
|
658
|
+
while (optionsTotalWidth > availableSpace && this.displayedOptions.length > 1) {
|
|
659
|
+
this.overflowCount++;
|
|
660
|
+
optionsTotalWidth -= optionsWidths[optionsWidths.length - 1];
|
|
661
|
+
optionsWidths.pop();
|
|
662
|
+
this.displayedOptions.pop();
|
|
663
|
+
}
|
|
664
|
+
}
|
|
665
|
+
}
|
|
666
|
+
announce(message) {
|
|
667
|
+
// \u00A0 is a non-breaking space character, which causes the message to be read as a new one
|
|
668
|
+
if (this.liveRegionEl.textContent === message) {
|
|
669
|
+
message += "\u00A0";
|
|
670
|
+
}
|
|
671
|
+
this.announcement = message;
|
|
672
|
+
}
|
|
673
|
+
renderButtonText() {
|
|
674
|
+
if (this.multiple && this.displayedOptions.length < 1) {
|
|
675
|
+
return index.h("span", null, this.placeholder);
|
|
676
|
+
}
|
|
677
|
+
else if (this.multiple && this.allSelected && this.overflowCount > 0) {
|
|
678
|
+
return this.allSelectedMessage;
|
|
679
|
+
}
|
|
680
|
+
else {
|
|
681
|
+
return this.displayedOptions.map((x, idx) => (index.h("span", null, idx > 0 ? ", " : "", x.textContent)));
|
|
682
|
+
}
|
|
683
|
+
}
|
|
684
|
+
renderSubinfo() {
|
|
685
|
+
// multiselects cannot have subinfo for options
|
|
686
|
+
if (!this.multiple && this.selectedOptions.length > 0 && this.selectedOptions[0].subinfo) {
|
|
687
|
+
return index.h("span", { class: "subinfo" }, this.selectedOptions[0].subinfo);
|
|
688
|
+
}
|
|
689
|
+
}
|
|
690
|
+
renderOverflowCount() {
|
|
691
|
+
if (this.overflowCount > 0 && !this.allSelected) {
|
|
692
|
+
return (index.h("span", null, index.h("span", { class: "overflow-counter" }, "+", this.overflowCount)));
|
|
693
|
+
}
|
|
694
|
+
}
|
|
695
|
+
renderSearchField() {
|
|
696
|
+
return (index.h("div", { class: "search" }, index.h("div", { class: "searchfield-wrapper", ref: (el) => (this.searchFieldWrapperEl = el) }, index.h("div", { class: "icon" }), index.h("input", { ref: (el) => (this.searchFieldEl = el), class: "searchfield", role: "combobox", "aria-controls": `list-${this.uid}`, "aria-expanded": this.isExpanded ? "true" : "false", onKeyDown: (ev) => this.handleSearchFieldKeyDown(ev), onFocus: () => this.searchFieldWrapperEl.classList.add("focus"), onBlur: (ev) => this.handleSearchFieldBlur(ev), onInput: () => this.debouncedSearch(), placeholder: this.searchPlaceholder }))));
|
|
697
|
+
}
|
|
698
|
+
renderSearchFailedMessage() {
|
|
699
|
+
return index.h("div", { class: "search-results-message" }, this.noResultsFoundMessage);
|
|
700
|
+
}
|
|
701
|
+
renderDuplicateOptions() {
|
|
702
|
+
return Array.from(this.el.children).map((option) => {
|
|
703
|
+
return (index.h("wm-option", { class: "duplicate", selected: option.selected }, option.textContent));
|
|
704
|
+
});
|
|
705
|
+
}
|
|
706
|
+
render() {
|
|
707
|
+
const buttonProps = {
|
|
708
|
+
id: `selectbtn-${this.uid}`,
|
|
709
|
+
["disabled"]: this.disabled,
|
|
710
|
+
["aria-controls"]: `list-${this.uid}`,
|
|
711
|
+
["aria-labelledby"]: `label-${this.uid} selectbtn-${this.uid}`,
|
|
712
|
+
["aria-describedby"]: "wm-errors",
|
|
713
|
+
["aria-expanded"]: this.isExpanded ? "true" : "false",
|
|
714
|
+
onClick: () => (this.isExpanded ? this.close() : this.open()),
|
|
715
|
+
};
|
|
716
|
+
return (index.h(index.Host, { onBlur: (ev) => this.handleComponentBlur(ev) }, index.h("div", { class: `wrapper ${functions.getTextDir()} label-${this.labelPosition} ${this.invalid || this.errorMessage ? "invalid" : ""}` }, index.h("div", { class: "label-wrapper" }, index.h("label", { class: "label", id: `label-${this.uid}`, htmlFor: `selectbtn-${this.uid}` }, this.label,
|
|
717
|
+
// we can't use aria-required or required attributes because it's invalid on the elements we're using (button controlling a listbox)
|
|
718
|
+
this.requiredField ? (index.h("span", { class: "required" }, index.h("span", { class: "sr-only" }, this.requiredMessage), index.h("span", { "aria-hidden": "true" }, "*"))) : (""))), index.h("div", { class: "button-wrapper" }, index.h("button", Object.assign({}, buttonProps, { class: `displayedoption ${this.isTabbing ? "user-is-tabbing" : ""}`, ref: (el) => (this.buttonEl = el), onBlur: (ev) => this.handleButtonBlur(ev), onFocus: () => this.close() }), index.h("span", { class: this.selectedOptions.length > 0 && this.selectedOptions.filter((x) => x.subinfo).length > 0
|
|
719
|
+
? "overflowcontrol hassubinfo"
|
|
720
|
+
: "overflowcontrol" }, index.h("span", { class: "button-text" }, this.renderButtonText()), this.renderSubinfo()), this.renderOverflowCount()), index.h("div", { class: `dropdown ${this.isExpanded ? "open" : ""} ${this.openUp ? "upwards" : ""}`, ref: (el) => (this.dropdownEl = el) }, this.search && this.renderSearchField(), index.h("div", { id: `list-${this.uid}`, class: "options-wrapper", tabindex: -1, role: "listbox", "aria-multiselectable": this.multiple ? "true" : null, "aria-labelledby": `label-${this.uid}`, ref: (el) => (this.listboxEl = el) }, this.search && this.filteredOptions.length === 0 && this.renderSearchFailedMessage(), this.multiple && this.renderDuplicateOptions(), index.h("slot", null))), this.renderErrorContainer(), index.h("div", { id: "announcement", "aria-live": "polite", class: "sr-only", ref: (el) => (this.liveRegionEl = el) }, this.announcement)))));
|
|
721
|
+
}
|
|
722
|
+
get el() { return index.getElement(this); }
|
|
723
|
+
static get watchers() { return {
|
|
724
|
+
"errorMessage": ["exposeErrors"]
|
|
725
|
+
}; }
|
|
726
|
+
};
|
|
720
727
|
Select.style = wmSelectCss;
|
|
721
728
|
|
|
722
729
|
exports.wm_option = Option;
|