@danske/sapphire-angular 1.12.4
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/LICENSE +7 -0
- package/README.md +25 -0
- package/esm2020/danske-sapphire-angular.mjs +5 -0
- package/esm2020/lib/badge/public_api.mjs +3 -0
- package/esm2020/lib/badge/src/badge.component.mjs +66 -0
- package/esm2020/lib/badge/src/badge.module.mjs +18 -0
- package/esm2020/lib/button/public_api.mjs +12 -0
- package/esm2020/lib/button/src/button-group.component.mjs +57 -0
- package/esm2020/lib/button/src/button-icon.directive.mjs +13 -0
- package/esm2020/lib/button/src/button.component.mjs +100 -0
- package/esm2020/lib/button/src/button.module.mjs +48 -0
- package/esm2020/lib/button/src/icon-button.component.mjs +95 -0
- package/esm2020/lib/button/src/toggle-button-required-validator.mjs +27 -0
- package/esm2020/lib/button/src/toggle-button.component.mjs +124 -0
- package/esm2020/lib/checkbox/public_api.mjs +8 -0
- package/esm2020/lib/checkbox/src/checkbox-required-validator.mjs +26 -0
- package/esm2020/lib/checkbox/src/checkbox.component.mjs +264 -0
- package/esm2020/lib/checkbox/src/checkbox.module.mjs +21 -0
- package/esm2020/lib/common/auto-id.decorator.mjs +24 -0
- package/esm2020/lib/common/coerce-boolean.decorator.mjs +32 -0
- package/esm2020/lib/common/constructor.mjs +2 -0
- package/esm2020/lib/common/disabled.mjs +16 -0
- package/esm2020/lib/common/focused.directive.mjs +64 -0
- package/esm2020/lib/common/placement.mjs +60 -0
- package/esm2020/lib/common/platform.mjs +45 -0
- package/esm2020/lib/common/pressed.directive.mjs +55 -0
- package/esm2020/lib/common/public_api.mjs +4 -0
- package/esm2020/lib/common/sapphire-view-encapsulation.mjs +83 -0
- package/esm2020/lib/common/scroll-monitor.directive.mjs +63 -0
- package/esm2020/lib/common/tabindex.mjs +19 -0
- package/esm2020/lib/common/visually-hidden.directive.mjs +32 -0
- package/esm2020/lib/contextual-help/public_api.mjs +3 -0
- package/esm2020/lib/contextual-help/src/contextual-help.component.mjs +112 -0
- package/esm2020/lib/contextual-help/src/contextual-help.module.mjs +33 -0
- package/esm2020/lib/core.module.mjs +179 -0
- package/esm2020/lib/field/public_api.mjs +8 -0
- package/esm2020/lib/field/src/field-control.mjs +17 -0
- package/esm2020/lib/field/src/field-error.directive.mjs +20 -0
- package/esm2020/lib/field/src/field-label.directive.mjs +17 -0
- package/esm2020/lib/field/src/field-note-suffix.directive.mjs +28 -0
- package/esm2020/lib/field/src/field-note.directive.mjs +17 -0
- package/esm2020/lib/field/src/field.component.mjs +153 -0
- package/esm2020/lib/field/src/field.module.mjs +49 -0
- package/esm2020/lib/icon/public_api.mjs +4 -0
- package/esm2020/lib/icon/src/icon-size-provider.mjs +18 -0
- package/esm2020/lib/icon/src/icon.component.mjs +67 -0
- package/esm2020/lib/icon/src/icon.module.mjs +18 -0
- package/esm2020/lib/label/public_api.mjs +6 -0
- package/esm2020/lib/label/src/label.component.mjs +61 -0
- package/esm2020/lib/link/public_api.mjs +7 -0
- package/esm2020/lib/link/src/link.component.mjs +62 -0
- package/esm2020/lib/link/src/link.module.mjs +16 -0
- package/esm2020/lib/listbox/public_api.mjs +10 -0
- package/esm2020/lib/listbox/src/cdk-option-scroll-issue-patch.mjs +30 -0
- package/esm2020/lib/listbox/src/listbox-child.mjs +7 -0
- package/esm2020/lib/listbox/src/listbox-input.directive.mjs +42 -0
- package/esm2020/lib/listbox/src/listbox-item.component.mjs +49 -0
- package/esm2020/lib/listbox/src/listbox.component.mjs +273 -0
- package/esm2020/lib/listbox/src/listbox.module.mjs +81 -0
- package/esm2020/lib/listbox/src/option-icon.directive.mjs +22 -0
- package/esm2020/lib/listbox/src/option-primary-text.directive.mjs +25 -0
- package/esm2020/lib/listbox/src/option-secondary-text.directive.mjs +22 -0
- package/esm2020/lib/listbox/src/option.component.mjs +58 -0
- package/esm2020/lib/listbox/src/section.directive.mjs +46 -0
- package/esm2020/lib/menu/public_api.mjs +10 -0
- package/esm2020/lib/menu/src/menu-item.component.mjs +45 -0
- package/esm2020/lib/menu/src/menu-section.component.mjs +27 -0
- package/esm2020/lib/menu/src/menu-trigger.directive.mjs +149 -0
- package/esm2020/lib/menu/src/menu.component.mjs +33 -0
- package/esm2020/lib/menu/src/menu.module.mjs +39 -0
- package/esm2020/lib/modal/public_api.mjs +19 -0
- package/esm2020/lib/modal/src/dialog/confirmation-dialog.component.mjs +61 -0
- package/esm2020/lib/modal/src/dialog/danger-dialog.component.mjs +62 -0
- package/esm2020/lib/modal/src/dialog/dialog-close-button.directive.mjs +18 -0
- package/esm2020/lib/modal/src/dialog/dialog-content.directive.mjs +29 -0
- package/esm2020/lib/modal/src/dialog/dialog-footer.component.mjs +43 -0
- package/esm2020/lib/modal/src/dialog/dialog-header.component.mjs +65 -0
- package/esm2020/lib/modal/src/dialog/dialog-trigger.directive.mjs +37 -0
- package/esm2020/lib/modal/src/dialog/dialog.component.mjs +60 -0
- package/esm2020/lib/modal/src/modal-base.component.mjs +89 -0
- package/esm2020/lib/modal/src/modal-close-button.directive.mjs +28 -0
- package/esm2020/lib/modal/src/modal-trigger.directive.mjs +42 -0
- package/esm2020/lib/modal/src/modal.module.mjs +143 -0
- package/esm2020/lib/modal/src/modal.service.mjs +58 -0
- package/esm2020/lib/modal/src/panel/inline-panel.component.mjs +15 -0
- package/esm2020/lib/modal/src/panel/panel-back-button.directive.mjs +13 -0
- package/esm2020/lib/modal/src/panel/panel-close-button.directive.mjs +16 -0
- package/esm2020/lib/modal/src/panel/panel-content.component.mjs +38 -0
- package/esm2020/lib/modal/src/panel/panel-footer.component.mjs +38 -0
- package/esm2020/lib/modal/src/panel/panel-header.component.mjs +68 -0
- package/esm2020/lib/modal/src/panel/panel-trigger.directive.mjs +37 -0
- package/esm2020/lib/modal/src/panel/panel.component.mjs +37 -0
- package/esm2020/lib/notification-badge/public_api.mjs +3 -0
- package/esm2020/lib/notification-badge/src/notification-badge.component.mjs +28 -0
- package/esm2020/lib/notification-badge/src/notification-badge.module.mjs +18 -0
- package/esm2020/lib/pagination/public_api.mjs +3 -0
- package/esm2020/lib/pagination/src/pagination.component.mjs +85 -0
- package/esm2020/lib/pagination/src/pagination.module.mjs +36 -0
- package/esm2020/lib/popover/public_api.mjs +10 -0
- package/esm2020/lib/popover/src/popover-close-button.directive.mjs +32 -0
- package/esm2020/lib/popover/src/popover-title.directive.mjs +24 -0
- package/esm2020/lib/popover/src/popover-trigger.directive.mjs +301 -0
- package/esm2020/lib/popover/src/popover.component.mjs +88 -0
- package/esm2020/lib/popover/src/popover.module.mjs +38 -0
- package/esm2020/lib/radio/public_api.mjs +8 -0
- package/esm2020/lib/radio/src/radio-group.component.mjs +195 -0
- package/esm2020/lib/radio/src/radio.component.mjs +279 -0
- package/esm2020/lib/radio/src/radio.module.mjs +22 -0
- package/esm2020/lib/segmented-tabs/public_api.mjs +6 -0
- package/esm2020/lib/segmented-tabs/src/segmented-tab-content.directive.mjs +15 -0
- package/esm2020/lib/segmented-tabs/src/segmented-tab-label.directive.mjs +20 -0
- package/esm2020/lib/segmented-tabs/src/segmented-tab.component.mjs +69 -0
- package/esm2020/lib/segmented-tabs/src/segmented-tabs.component.mjs +215 -0
- package/esm2020/lib/segmented-tabs/src/segmented-tabs.module.mjs +39 -0
- package/esm2020/lib/select/public_api.mjs +10 -0
- package/esm2020/lib/select/src/basic-select/basic-select.component.mjs +127 -0
- package/esm2020/lib/select/src/common/hidden-select.component.mjs +33 -0
- package/esm2020/lib/select/src/common/select-component-base.mjs +267 -0
- package/esm2020/lib/select/src/common/select-value-holder.mjs +86 -0
- package/esm2020/lib/select/src/select/searchable-select.directive.mjs +90 -0
- package/esm2020/lib/select/src/select/select.component.mjs +351 -0
- package/esm2020/lib/select/src/select/select.module.mjs +113 -0
- package/esm2020/lib/select/src/select/selection-text.directive.mjs +19 -0
- package/esm2020/lib/skeleton/public_api.mjs +6 -0
- package/esm2020/lib/skeleton/src/skeleton-block.directive.mjs +16 -0
- package/esm2020/lib/skeleton/src/skeleton-circle.directive.mjs +16 -0
- package/esm2020/lib/skeleton/src/skeleton-text.directive.mjs +43 -0
- package/esm2020/lib/skeleton/src/skeleton.component.mjs +29 -0
- package/esm2020/lib/skeleton/src/skeleton.module.mjs +37 -0
- package/esm2020/lib/skeleton/src/util.mjs +2 -0
- package/esm2020/lib/surface/public_api.mjs +7 -0
- package/esm2020/lib/surface/src/surface.component.mjs +25 -0
- package/esm2020/lib/surface/src/surface.module.mjs +16 -0
- package/esm2020/lib/switch/public_api.mjs +4 -0
- package/esm2020/lib/switch/src/switch-required-validator.mjs +27 -0
- package/esm2020/lib/switch/src/switch.component.mjs +216 -0
- package/esm2020/lib/switch/src/switch.module.mjs +19 -0
- package/esm2020/lib/table/public_api.mjs +15 -0
- package/esm2020/lib/table/src/table-body.directive.mjs +20 -0
- package/esm2020/lib/table/src/table-cell.directive.mjs +87 -0
- package/esm2020/lib/table/src/table-head-cell.component.mjs +57 -0
- package/esm2020/lib/table/src/table-head.directive.mjs +32 -0
- package/esm2020/lib/table/src/table-row.directive.mjs +48 -0
- package/esm2020/lib/table/src/table-sort-header.directive.mjs +44 -0
- package/esm2020/lib/table/src/table-sort.directive.mjs +26 -0
- package/esm2020/lib/table/src/table.component.mjs +82 -0
- package/esm2020/lib/table/src/table.directive.mjs +20 -0
- package/esm2020/lib/table/src/table.module.mjs +74 -0
- package/esm2020/lib/text-field/public_api.mjs +12 -0
- package/esm2020/lib/text-field/src/text-field-affix.directive.mjs +26 -0
- package/esm2020/lib/text-field/src/text-field-input.directive.mjs +51 -0
- package/esm2020/lib/text-field/src/text-field-postfix.directive.mjs +13 -0
- package/esm2020/lib/text-field/src/text-field-prefix.directive.mjs +13 -0
- package/esm2020/lib/text-field/src/text-field-textarea-autosize.directive.mjs +73 -0
- package/esm2020/lib/text-field/src/text-field.component.mjs +78 -0
- package/esm2020/lib/text-field/src/text-field.module.mjs +67 -0
- package/esm2020/lib/theme/public_api.mjs +34 -0
- package/esm2020/lib/theme/src/sapphire-overlay-container.service.mjs +38 -0
- package/esm2020/lib/theme/src/theme-base.directive.mjs +87 -0
- package/esm2020/lib/theme/src/theme-check.directive.mjs +29 -0
- package/esm2020/lib/theme/src/theme-root.directive.mjs +23 -0
- package/esm2020/lib/theme/src/theme.module.mjs +32 -0
- package/esm2020/lib/theme/src/themes.mjs +27 -0
- package/esm2020/lib/theme/src/top-level-theme-ref.service.mjs +21 -0
- package/esm2020/lib/tooltip/public_api.mjs +5 -0
- package/esm2020/lib/tooltip/src/tooltip.component.mjs +47 -0
- package/esm2020/lib/tooltip/src/tooltip.directive.mjs +223 -0
- package/esm2020/lib/tooltip/src/tooltip.module.mjs +30 -0
- package/esm2020/lib/tooltip/src/truncated-with-tooltip.directive.mjs +56 -0
- package/esm2020/lib/typography/public_api.mjs +8 -0
- package/esm2020/lib/typography/src/heading.component.mjs +44 -0
- package/esm2020/lib/typography/src/paragraph.component.mjs +24 -0
- package/esm2020/lib/typography/src/typography.module.mjs +19 -0
- package/esm2020/public-api.mjs +30 -0
- package/fesm2015/danske-sapphire-angular.mjs +8010 -0
- package/fesm2015/danske-sapphire-angular.mjs.map +1 -0
- package/fesm2020/danske-sapphire-angular.mjs +7890 -0
- package/fesm2020/danske-sapphire-angular.mjs.map +1 -0
- package/index.d.ts +5 -0
- package/lib/badge/public_api.d.ts +2 -0
- package/lib/badge/src/badge.component.d.ts +35 -0
- package/lib/badge/src/badge.module.d.ts +8 -0
- package/lib/button/public_api.d.ts +7 -0
- package/lib/button/src/button-group.component.d.ts +26 -0
- package/lib/button/src/button-icon.directive.d.ts +5 -0
- package/lib/button/src/button.component.d.ts +31 -0
- package/lib/button/src/button.module.d.ts +14 -0
- package/lib/button/src/icon-button.component.d.ts +26 -0
- package/lib/button/src/toggle-button-required-validator.d.ts +10 -0
- package/lib/button/src/toggle-button.component.d.ts +39 -0
- package/lib/checkbox/public_api.d.ts +3 -0
- package/lib/checkbox/src/checkbox-required-validator.d.ts +10 -0
- package/lib/checkbox/src/checkbox.component.d.ts +112 -0
- package/lib/checkbox/src/checkbox.module.d.ts +11 -0
- package/lib/common/auto-id.decorator.d.ts +7 -0
- package/lib/common/coerce-boolean.decorator.d.ts +7 -0
- package/lib/common/constructor.d.ts +2 -0
- package/lib/common/disabled.d.ts +9 -0
- package/lib/common/focused.directive.d.ts +30 -0
- package/lib/common/placement.d.ts +11 -0
- package/lib/common/platform.d.ts +9 -0
- package/lib/common/pressed.directive.d.ts +27 -0
- package/lib/common/public_api.d.ts +2 -0
- package/lib/common/sapphire-view-encapsulation.d.ts +37 -0
- package/lib/common/scroll-monitor.directive.d.ts +23 -0
- package/lib/common/tabindex.d.ts +12 -0
- package/lib/common/visually-hidden.directive.d.ts +11 -0
- package/lib/contextual-help/public_api.d.ts +2 -0
- package/lib/contextual-help/src/contextual-help.component.d.ts +50 -0
- package/lib/contextual-help/src/contextual-help.module.d.ts +7 -0
- package/lib/core.module.d.ts +30 -0
- package/lib/field/public_api.d.ts +7 -0
- package/lib/field/src/field-control.d.ts +16 -0
- package/lib/field/src/field-error.directive.d.ts +8 -0
- package/lib/field/src/field-label.directive.d.ts +8 -0
- package/lib/field/src/field-note-suffix.directive.d.ts +15 -0
- package/lib/field/src/field-note.directive.d.ts +8 -0
- package/lib/field/src/field.component.d.ts +77 -0
- package/lib/field/src/field.module.d.ts +14 -0
- package/lib/icon/public_api.d.ts +3 -0
- package/lib/icon/src/icon-size-provider.d.ts +23 -0
- package/lib/icon/src/icon.component.d.ts +18 -0
- package/lib/icon/src/icon.module.d.ts +8 -0
- package/lib/label/public_api.d.ts +5 -0
- package/lib/label/src/label.component.d.ts +43 -0
- package/lib/link/public_api.d.ts +2 -0
- package/lib/link/src/link.component.d.ts +13 -0
- package/lib/link/src/link.module.d.ts +7 -0
- package/lib/listbox/public_api.d.ts +9 -0
- package/lib/listbox/src/cdk-option-scroll-issue-patch.d.ts +17 -0
- package/lib/listbox/src/listbox-child.d.ts +6 -0
- package/lib/listbox/src/listbox-input.directive.d.ts +16 -0
- package/lib/listbox/src/listbox-item.component.d.ts +21 -0
- package/lib/listbox/src/listbox.component.d.ts +127 -0
- package/lib/listbox/src/listbox.module.d.ts +22 -0
- package/lib/listbox/src/option-icon.directive.d.ts +10 -0
- package/lib/listbox/src/option-primary-text.directive.d.ts +14 -0
- package/lib/listbox/src/option-secondary-text.directive.d.ts +10 -0
- package/lib/listbox/src/option.component.d.ts +21 -0
- package/lib/listbox/src/section.directive.d.ts +14 -0
- package/lib/menu/public_api.d.ts +5 -0
- package/lib/menu/src/menu-item.component.d.ts +12 -0
- package/lib/menu/src/menu-section.component.d.ts +12 -0
- package/lib/menu/src/menu-trigger.directive.d.ts +34 -0
- package/lib/menu/src/menu.component.d.ts +10 -0
- package/lib/menu/src/menu.module.d.ts +13 -0
- package/lib/modal/public_api.d.ts +18 -0
- package/lib/modal/src/dialog/confirmation-dialog.component.d.ts +43 -0
- package/lib/modal/src/dialog/danger-dialog.component.d.ts +44 -0
- package/lib/modal/src/dialog/dialog-close-button.directive.d.ts +7 -0
- package/lib/modal/src/dialog/dialog-content.directive.d.ts +13 -0
- package/lib/modal/src/dialog/dialog-footer.component.d.ts +20 -0
- package/lib/modal/src/dialog/dialog-header.component.d.ts +25 -0
- package/lib/modal/src/dialog/dialog-trigger.directive.d.ts +13 -0
- package/lib/modal/src/dialog/dialog.component.d.ts +33 -0
- package/lib/modal/src/modal-base.component.d.ts +33 -0
- package/lib/modal/src/modal-close-button.directive.d.ts +9 -0
- package/lib/modal/src/modal-trigger.directive.d.ts +24 -0
- package/lib/modal/src/modal.module.d.ts +30 -0
- package/lib/modal/src/modal.service.d.ts +26 -0
- package/lib/modal/src/panel/inline-panel.component.d.ts +6 -0
- package/lib/modal/src/panel/panel-back-button.directive.d.ts +5 -0
- package/lib/modal/src/panel/panel-close-button.directive.d.ts +6 -0
- package/lib/modal/src/panel/panel-content.component.d.ts +14 -0
- package/lib/modal/src/panel/panel-footer.component.d.ts +21 -0
- package/lib/modal/src/panel/panel-header.component.d.ts +26 -0
- package/lib/modal/src/panel/panel-trigger.directive.d.ts +13 -0
- package/lib/modal/src/panel/panel.component.d.ts +18 -0
- package/lib/notification-badge/public_api.d.ts +2 -0
- package/lib/notification-badge/src/notification-badge.component.d.ts +10 -0
- package/lib/notification-badge/src/notification-badge.module.d.ts +8 -0
- package/lib/pagination/public_api.d.ts +2 -0
- package/lib/pagination/src/pagination.component.d.ts +32 -0
- package/lib/pagination/src/pagination.module.d.ts +12 -0
- package/lib/popover/public_api.d.ts +5 -0
- package/lib/popover/src/popover-close-button.directive.d.ts +13 -0
- package/lib/popover/src/popover-title.directive.d.ts +11 -0
- package/lib/popover/src/popover-trigger.directive.d.ts +86 -0
- package/lib/popover/src/popover.component.d.ts +32 -0
- package/lib/popover/src/popover.module.d.ts +12 -0
- package/lib/radio/public_api.d.ts +3 -0
- package/lib/radio/src/radio-group.component.d.ts +65 -0
- package/lib/radio/src/radio.component.d.ts +99 -0
- package/lib/radio/src/radio.module.d.ts +12 -0
- package/lib/segmented-tabs/public_api.d.ts +5 -0
- package/lib/segmented-tabs/src/segmented-tab-content.directive.d.ts +6 -0
- package/lib/segmented-tabs/src/segmented-tab-label.directive.d.ts +6 -0
- package/lib/segmented-tabs/src/segmented-tab.component.d.ts +27 -0
- package/lib/segmented-tabs/src/segmented-tabs.component.d.ts +48 -0
- package/lib/segmented-tabs/src/segmented-tabs.module.d.ts +13 -0
- package/lib/select/public_api.d.ts +5 -0
- package/lib/select/src/basic-select/basic-select.component.d.ts +50 -0
- package/lib/select/src/common/hidden-select.component.d.ts +19 -0
- package/lib/select/src/common/select-component-base.d.ts +109 -0
- package/lib/select/src/common/select-value-holder.d.ts +54 -0
- package/lib/select/src/select/searchable-select.directive.d.ts +40 -0
- package/lib/select/src/select/select.component.d.ts +80 -0
- package/lib/select/src/select/select.module.d.ts +26 -0
- package/lib/select/src/select/selection-text.directive.d.ts +11 -0
- package/lib/skeleton/public_api.d.ts +5 -0
- package/lib/skeleton/src/skeleton-block.directive.d.ts +5 -0
- package/lib/skeleton/src/skeleton-circle.directive.d.ts +5 -0
- package/lib/skeleton/src/skeleton-text.directive.d.ts +13 -0
- package/lib/skeleton/src/skeleton.component.d.ts +13 -0
- package/lib/skeleton/src/skeleton.module.d.ts +11 -0
- package/lib/skeleton/src/util.d.ts +1 -0
- package/lib/surface/public_api.d.ts +2 -0
- package/lib/surface/src/surface.component.d.ts +11 -0
- package/lib/surface/src/surface.module.d.ts +7 -0
- package/lib/switch/public_api.d.ts +3 -0
- package/lib/switch/src/switch-required-validator.d.ts +10 -0
- package/lib/switch/src/switch.component.d.ts +91 -0
- package/lib/switch/src/switch.module.d.ts +9 -0
- package/lib/table/public_api.d.ts +10 -0
- package/lib/table/src/table-body.directive.d.ts +7 -0
- package/lib/table/src/table-cell.directive.d.ts +30 -0
- package/lib/table/src/table-head-cell.component.d.ts +19 -0
- package/lib/table/src/table-head.directive.d.ts +10 -0
- package/lib/table/src/table-row.directive.d.ts +16 -0
- package/lib/table/src/table-sort-header.directive.d.ts +11 -0
- package/lib/table/src/table-sort.directive.d.ts +13 -0
- package/lib/table/src/table.component.d.ts +37 -0
- package/lib/table/src/table.directive.d.ts +7 -0
- package/lib/table/src/table.module.d.ts +19 -0
- package/lib/text-field/public_api.d.ts +7 -0
- package/lib/text-field/src/text-field-affix.directive.d.ts +10 -0
- package/lib/text-field/src/text-field-input.directive.d.ts +16 -0
- package/lib/text-field/src/text-field-postfix.directive.d.ts +5 -0
- package/lib/text-field/src/text-field-prefix.directive.d.ts +5 -0
- package/lib/text-field/src/text-field-textarea-autosize.directive.d.ts +20 -0
- package/lib/text-field/src/text-field.component.d.ts +23 -0
- package/lib/text-field/src/text-field.module.d.ts +17 -0
- package/lib/theme/public_api.d.ts +24 -0
- package/lib/theme/src/sapphire-overlay-container.service.d.ts +14 -0
- package/lib/theme/src/theme-base.directive.d.ts +14 -0
- package/lib/theme/src/theme-check.directive.d.ts +14 -0
- package/lib/theme/src/theme-root.directive.d.ts +13 -0
- package/lib/theme/src/theme.module.d.ts +14 -0
- package/lib/theme/src/themes.d.ts +16 -0
- package/lib/theme/src/top-level-theme-ref.service.d.ts +13 -0
- package/lib/tooltip/public_api.d.ts +4 -0
- package/lib/tooltip/src/tooltip.component.d.ts +10 -0
- package/lib/tooltip/src/tooltip.directive.d.ts +39 -0
- package/lib/tooltip/src/tooltip.module.d.ts +11 -0
- package/lib/tooltip/src/truncated-with-tooltip.directive.d.ts +26 -0
- package/lib/typography/public_api.d.ts +3 -0
- package/lib/typography/src/heading.component.d.ts +17 -0
- package/lib/typography/src/paragraph.component.d.ts +8 -0
- package/lib/typography/src/typography.module.d.ts +9 -0
- package/package.json +45 -0
- package/public-api.d.ts +26 -0
- package/src/lib/listbox/src/README.md +67 -0
- package/theme-assets/default-dark.css +1 -0
- package/theme-assets/default.css +1 -0
|
@@ -0,0 +1,33 @@
|
|
|
1
|
+
import { Component } from '@angular/core';
|
|
2
|
+
import { isMobile } from '../../../common/platform';
|
|
3
|
+
import * as i0 from "@angular/core";
|
|
4
|
+
import * as i1 from "./select-component-base";
|
|
5
|
+
import * as i2 from "@angular/common";
|
|
6
|
+
import * as i3 from "@angular/forms";
|
|
7
|
+
import * as i4 from "../../../common/visually-hidden.directive";
|
|
8
|
+
import * as i5 from "../../../common/sapphire-view-encapsulation";
|
|
9
|
+
/**
|
|
10
|
+
* Hidden native select to be rendered in sapphire select components. It does two things:
|
|
11
|
+
* - Takes care of form accessibility and autofilling by rendering native select/input
|
|
12
|
+
* - Opens the native select dropdown instead of the custom dropdown, when possible.
|
|
13
|
+
*
|
|
14
|
+
* The latter proved to be a questionable, and we may rethink it at some point.
|
|
15
|
+
*
|
|
16
|
+
* @internal
|
|
17
|
+
*/
|
|
18
|
+
export class HiddenSelectComponent {
|
|
19
|
+
constructor(_select) {
|
|
20
|
+
this._select = _select;
|
|
21
|
+
this._isMobile = isMobile();
|
|
22
|
+
}
|
|
23
|
+
_onNativeSelectChange(e) {
|
|
24
|
+
this._select.listboxValue = [e.target.value];
|
|
25
|
+
}
|
|
26
|
+
}
|
|
27
|
+
HiddenSelectComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.0", ngImport: i0, type: HiddenSelectComponent, deps: [{ token: i1.SelectComponentBase }], target: i0.ɵɵFactoryTarget.Component });
|
|
28
|
+
HiddenSelectComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "15.2.0", type: HiddenSelectComponent, selector: "sp-hidden-select", ngImport: i0, template: "<ng-container *ngIf=\"_select._initialized\">\n <select\n *ngIf=\"_isMobile && !_select.multiple\"\n class=\"sapphire-select__native-select\"\n _spUseComponentStyles\n [disabled]=\"_select.disabled\"\n [attr.name]=\"_select.name\"\n (change)=\"_onNativeSelectChange($event)\"\n >\n <option\n *ngFor=\"let option of _select.options\"\n [value]=\"option.value\"\n [disabled]=\"option.disabled\"\n >\n {{ option.label }}\n </option>\n </select>\n <div\n *ngIf=\"!_isMobile || _select.multiple\"\n aria-hidden=\"true\"\n spVisuallyHidden\n >\n <input\n type=\"text\"\n style=\"font-size: 16px\"\n [disabled]=\"_select.disabled\"\n [tabIndex]=\"_select.isFocused() || _select.isOpen() ? -1 : 0\"\n (focus)=\"_select.focus('keyboard')\"\n />\n <label>\n {{ _select.getLabelText() }}\n <select\n [attr.size]=\"_select.multiple ? _select.options.length : null\"\n [disabled]=\"_select.disabled\"\n tabindex=\"-1\"\n [multiple]=\"_select.multiple\"\n (change)=\"_onNativeSelectChange($event)\"\n [attr.name]=\"_select.name\"\n >\n <option\n *ngFor=\"let option of _select.options\"\n [value]=\"option.value\"\n [selected]=\"_select.listboxValue.includes(option.value)\"\n [disabled]=\"option.disabled\"\n >\n {{ option.label }}\n </option>\n </select>\n </label>\n </div>\n</ng-container>\n", dependencies: [{ kind: "directive", type: i2.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i2.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i3.NgSelectOption, selector: "option", inputs: ["ngValue", "value"] }, { kind: "directive", type: i3.ɵNgSelectMultipleOption, selector: "option", inputs: ["ngValue", "value"] }, { kind: "directive", type: i4.VisuallyHiddenDirective, selector: "[spVisuallyHidden]", inputs: ["spVisuallyHidden"] }, { kind: "directive", type: i5.UseComponentStyles, selector: "[_spUseComponentStyles]" }] });
|
|
29
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.0", ngImport: i0, type: HiddenSelectComponent, decorators: [{
|
|
30
|
+
type: Component,
|
|
31
|
+
args: [{ selector: 'sp-hidden-select', template: "<ng-container *ngIf=\"_select._initialized\">\n <select\n *ngIf=\"_isMobile && !_select.multiple\"\n class=\"sapphire-select__native-select\"\n _spUseComponentStyles\n [disabled]=\"_select.disabled\"\n [attr.name]=\"_select.name\"\n (change)=\"_onNativeSelectChange($event)\"\n >\n <option\n *ngFor=\"let option of _select.options\"\n [value]=\"option.value\"\n [disabled]=\"option.disabled\"\n >\n {{ option.label }}\n </option>\n </select>\n <div\n *ngIf=\"!_isMobile || _select.multiple\"\n aria-hidden=\"true\"\n spVisuallyHidden\n >\n <input\n type=\"text\"\n style=\"font-size: 16px\"\n [disabled]=\"_select.disabled\"\n [tabIndex]=\"_select.isFocused() || _select.isOpen() ? -1 : 0\"\n (focus)=\"_select.focus('keyboard')\"\n />\n <label>\n {{ _select.getLabelText() }}\n <select\n [attr.size]=\"_select.multiple ? _select.options.length : null\"\n [disabled]=\"_select.disabled\"\n tabindex=\"-1\"\n [multiple]=\"_select.multiple\"\n (change)=\"_onNativeSelectChange($event)\"\n [attr.name]=\"_select.name\"\n >\n <option\n *ngFor=\"let option of _select.options\"\n [value]=\"option.value\"\n [selected]=\"_select.listboxValue.includes(option.value)\"\n [disabled]=\"option.disabled\"\n >\n {{ option.label }}\n </option>\n </select>\n </label>\n </div>\n</ng-container>\n" }]
|
|
32
|
+
}], ctorParameters: function () { return [{ type: i1.SelectComponentBase }]; } });
|
|
33
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaGlkZGVuLXNlbGVjdC5jb21wb25lbnQuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi9zcmMvbGliL3NlbGVjdC9zcmMvY29tbW9uL2hpZGRlbi1zZWxlY3QuY29tcG9uZW50LnRzIiwiLi4vLi4vLi4vLi4vLi4vLi4vc3JjL2xpYi9zZWxlY3Qvc3JjL2NvbW1vbi9oaWRkZW4tc2VsZWN0LmNvbXBvbmVudC5odG1sIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSxTQUFTLEVBQUUsTUFBTSxlQUFlLENBQUM7QUFDMUMsT0FBTyxFQUFFLFFBQVEsRUFBRSxNQUFNLDBCQUEwQixDQUFDOzs7Ozs7O0FBR3BEOzs7Ozs7OztHQVFHO0FBS0gsTUFBTSxPQUFPLHFCQUFxQjtJQUdoQyxZQUFtQixPQUE0QjtRQUE1QixZQUFPLEdBQVAsT0FBTyxDQUFxQjtRQUYvQyxjQUFTLEdBQUcsUUFBUSxFQUFFLENBQUM7SUFFMkIsQ0FBQztJQUVuRCxxQkFBcUIsQ0FBQyxDQUFNO1FBQzFCLElBQUksQ0FBQyxPQUFPLENBQUMsWUFBWSxHQUFHLENBQUMsQ0FBQyxDQUFDLE1BQU0sQ0FBQyxLQUFLLENBQUMsQ0FBQztJQUMvQyxDQUFDOztrSEFQVSxxQkFBcUI7c0dBQXJCLHFCQUFxQix3RENqQmxDLHMrQ0FtREE7MkZEbENhLHFCQUFxQjtrQkFKakMsU0FBUzsrQkFDRSxrQkFBa0IiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBDb21wb25lbnQgfSBmcm9tICdAYW5ndWxhci9jb3JlJztcbmltcG9ydCB7IGlzTW9iaWxlIH0gZnJvbSAnLi4vLi4vLi4vY29tbW9uL3BsYXRmb3JtJztcbmltcG9ydCB7IFNlbGVjdENvbXBvbmVudEJhc2UgfSBmcm9tICcuL3NlbGVjdC1jb21wb25lbnQtYmFzZSc7XG5cbi8qKlxuICogSGlkZGVuIG5hdGl2ZSBzZWxlY3QgdG8gYmUgcmVuZGVyZWQgaW4gc2FwcGhpcmUgc2VsZWN0IGNvbXBvbmVudHMuIEl0IGRvZXMgdHdvIHRoaW5nczpcbiAqIC0gVGFrZXMgY2FyZSBvZiBmb3JtIGFjY2Vzc2liaWxpdHkgYW5kIGF1dG9maWxsaW5nIGJ5IHJlbmRlcmluZyBuYXRpdmUgc2VsZWN0L2lucHV0XG4gKiAtIE9wZW5zIHRoZSBuYXRpdmUgc2VsZWN0IGRyb3Bkb3duIGluc3RlYWQgb2YgdGhlIGN1c3RvbSBkcm9wZG93biwgd2hlbiBwb3NzaWJsZS5cbiAqXG4gKiBUaGUgbGF0dGVyIHByb3ZlZCB0byBiZSBhIHF1ZXN0aW9uYWJsZSwgYW5kIHdlIG1heSByZXRoaW5rIGl0IGF0IHNvbWUgcG9pbnQuXG4gKlxuICogQGludGVybmFsXG4gKi9cbkBDb21wb25lbnQoe1xuICBzZWxlY3RvcjogJ3NwLWhpZGRlbi1zZWxlY3QnLFxuICB0ZW1wbGF0ZVVybDogJy4vaGlkZGVuLXNlbGVjdC5jb21wb25lbnQuaHRtbCcsXG59KVxuZXhwb3J0IGNsYXNzIEhpZGRlblNlbGVjdENvbXBvbmVudCB7XG4gIF9pc01vYmlsZSA9IGlzTW9iaWxlKCk7XG5cbiAgY29uc3RydWN0b3IocHVibGljIF9zZWxlY3Q6IFNlbGVjdENvbXBvbmVudEJhc2UpIHt9XG5cbiAgX29uTmF0aXZlU2VsZWN0Q2hhbmdlKGU6IGFueSkge1xuICAgIHRoaXMuX3NlbGVjdC5saXN0Ym94VmFsdWUgPSBbZS50YXJnZXQudmFsdWVdO1xuICB9XG59XG4iLCI8bmctY29udGFpbmVyICpuZ0lmPVwiX3NlbGVjdC5faW5pdGlhbGl6ZWRcIj5cbiAgPHNlbGVjdFxuICAgICpuZ0lmPVwiX2lzTW9iaWxlICYmICFfc2VsZWN0Lm11bHRpcGxlXCJcbiAgICBjbGFzcz1cInNhcHBoaXJlLXNlbGVjdF9fbmF0aXZlLXNlbGVjdFwiXG4gICAgX3NwVXNlQ29tcG9uZW50U3R5bGVzXG4gICAgW2Rpc2FibGVkXT1cIl9zZWxlY3QuZGlzYWJsZWRcIlxuICAgIFthdHRyLm5hbWVdPVwiX3NlbGVjdC5uYW1lXCJcbiAgICAoY2hhbmdlKT1cIl9vbk5hdGl2ZVNlbGVjdENoYW5nZSgkZXZlbnQpXCJcbiAgPlxuICAgIDxvcHRpb25cbiAgICAgICpuZ0Zvcj1cImxldCBvcHRpb24gb2YgX3NlbGVjdC5vcHRpb25zXCJcbiAgICAgIFt2YWx1ZV09XCJvcHRpb24udmFsdWVcIlxuICAgICAgW2Rpc2FibGVkXT1cIm9wdGlvbi5kaXNhYmxlZFwiXG4gICAgPlxuICAgICAge3sgb3B0aW9uLmxhYmVsIH19XG4gICAgPC9vcHRpb24+XG4gIDwvc2VsZWN0PlxuICA8ZGl2XG4gICAgKm5nSWY9XCIhX2lzTW9iaWxlIHx8IF9zZWxlY3QubXVsdGlwbGVcIlxuICAgIGFyaWEtaGlkZGVuPVwidHJ1ZVwiXG4gICAgc3BWaXN1YWxseUhpZGRlblxuICA+XG4gICAgPGlucHV0XG4gICAgICB0eXBlPVwidGV4dFwiXG4gICAgICBzdHlsZT1cImZvbnQtc2l6ZTogMTZweFwiXG4gICAgICBbZGlzYWJsZWRdPVwiX3NlbGVjdC5kaXNhYmxlZFwiXG4gICAgICBbdGFiSW5kZXhdPVwiX3NlbGVjdC5pc0ZvY3VzZWQoKSB8fCBfc2VsZWN0LmlzT3BlbigpID8gLTEgOiAwXCJcbiAgICAgIChmb2N1cyk9XCJfc2VsZWN0LmZvY3VzKCdrZXlib2FyZCcpXCJcbiAgICAvPlxuICAgIDxsYWJlbD5cbiAgICAgIHt7IF9zZWxlY3QuZ2V0TGFiZWxUZXh0KCkgfX1cbiAgICAgIDxzZWxlY3RcbiAgICAgICAgW2F0dHIuc2l6ZV09XCJfc2VsZWN0Lm11bHRpcGxlID8gX3NlbGVjdC5vcHRpb25zLmxlbmd0aCA6IG51bGxcIlxuICAgICAgICBbZGlzYWJsZWRdPVwiX3NlbGVjdC5kaXNhYmxlZFwiXG4gICAgICAgIHRhYmluZGV4PVwiLTFcIlxuICAgICAgICBbbXVsdGlwbGVdPVwiX3NlbGVjdC5tdWx0aXBsZVwiXG4gICAgICAgIChjaGFuZ2UpPVwiX29uTmF0aXZlU2VsZWN0Q2hhbmdlKCRldmVudClcIlxuICAgICAgICBbYXR0ci5uYW1lXT1cIl9zZWxlY3QubmFtZVwiXG4gICAgICA+XG4gICAgICAgIDxvcHRpb25cbiAgICAgICAgICAqbmdGb3I9XCJsZXQgb3B0aW9uIG9mIF9zZWxlY3Qub3B0aW9uc1wiXG4gICAgICAgICAgW3ZhbHVlXT1cIm9wdGlvbi52YWx1ZVwiXG4gICAgICAgICAgW3NlbGVjdGVkXT1cIl9zZWxlY3QubGlzdGJveFZhbHVlLmluY2x1ZGVzKG9wdGlvbi52YWx1ZSlcIlxuICAgICAgICAgIFtkaXNhYmxlZF09XCJvcHRpb24uZGlzYWJsZWRcIlxuICAgICAgICA+XG4gICAgICAgICAge3sgb3B0aW9uLmxhYmVsIH19XG4gICAgICAgIDwvb3B0aW9uPlxuICAgICAgPC9zZWxlY3Q+XG4gICAgPC9sYWJlbD5cbiAgPC9kaXY+XG48L25nLWNvbnRhaW5lcj5cbiJdfQ==
|
|
@@ -0,0 +1,267 @@
|
|
|
1
|
+
import { __decorate } from "tslib";
|
|
2
|
+
import { CoerceBoolean } from '../../../common/coerce-boolean.decorator';
|
|
3
|
+
import { ListboxChild, OptionComponent, } from '../../../listbox/public_api';
|
|
4
|
+
import { ChangeDetectorRef, ContentChild, ContentChildren, Directive, EventEmitter, inject, Input, Output, QueryList, } from '@angular/core';
|
|
5
|
+
import { AutoId } from '../../../common/auto-id.decorator';
|
|
6
|
+
import { ListKeyManager, } from '@angular/cdk/a11y';
|
|
7
|
+
import { MultiSelectDirectiveBase, } from './select-value-holder';
|
|
8
|
+
import { SelectionTextDirective } from '../select/selection-text.directive';
|
|
9
|
+
import * as i0 from "@angular/core";
|
|
10
|
+
import * as i1 from "./select-value-holder";
|
|
11
|
+
export class SelectComponentBase {
|
|
12
|
+
set listboxValue(value) {
|
|
13
|
+
this.valueHolder.setFromListboxValue(value);
|
|
14
|
+
this.controlValueAccessorChangeFn(this.valueHolder.value);
|
|
15
|
+
}
|
|
16
|
+
get listboxValue() {
|
|
17
|
+
return this.valueHolder.getListboxValue();
|
|
18
|
+
}
|
|
19
|
+
get selectedOptions() {
|
|
20
|
+
return this.options?.filter((option) => this.listboxValue.includes(option.value));
|
|
21
|
+
}
|
|
22
|
+
constructor(valueHolder) {
|
|
23
|
+
this.valueHolder = valueHolder;
|
|
24
|
+
/** A unique id for the select. If none is supplied, it will be auto-generated. */
|
|
25
|
+
this.id = '';
|
|
26
|
+
/**
|
|
27
|
+
* Emitted when the popover is opened
|
|
28
|
+
*/
|
|
29
|
+
this.opened = new EventEmitter();
|
|
30
|
+
/**
|
|
31
|
+
* Emitted when the popover is closed
|
|
32
|
+
*/
|
|
33
|
+
this.closed = new EventEmitter();
|
|
34
|
+
this.options = new QueryList();
|
|
35
|
+
this.listboxChildren = new QueryList();
|
|
36
|
+
/**
|
|
37
|
+
* Emits whenever the dropdown is closed as a result of an option (already
|
|
38
|
+
* selected or not) being selected.
|
|
39
|
+
*/
|
|
40
|
+
this.selected = new EventEmitter();
|
|
41
|
+
this.ariaLabel = null;
|
|
42
|
+
this.ariaLabelledby = null;
|
|
43
|
+
/**
|
|
44
|
+
* Used to work around ExpressionChangedAfterItHasBeenCheckedError error thrown from HiddenSelect,
|
|
45
|
+
* when the select component is rendered from a template (e.g. with *ngIf). It's set to true
|
|
46
|
+
* asynchronously after content is initialized, to delay rendering of things that depend on
|
|
47
|
+
* options. It's not clear why the issue happens, and it only happens when select is used with
|
|
48
|
+
* *ngIf, or generally in a template.
|
|
49
|
+
*/
|
|
50
|
+
this._initialized = false;
|
|
51
|
+
this.changeDetectorRef = inject(ChangeDetectorRef);
|
|
52
|
+
this.controlValueAccessorChangeFn = () => { };
|
|
53
|
+
this.onTouched = () => { };
|
|
54
|
+
}
|
|
55
|
+
handleKeyDown($event) {
|
|
56
|
+
if ($event.target !== this.triggerElementRef?.nativeElement) {
|
|
57
|
+
return;
|
|
58
|
+
}
|
|
59
|
+
switch ($event.key) {
|
|
60
|
+
case 'ArrowDown':
|
|
61
|
+
this.triggerRef?.open();
|
|
62
|
+
break;
|
|
63
|
+
case 'ArrowUp':
|
|
64
|
+
this.triggerRef?.open();
|
|
65
|
+
// we need to delay it a little, since the listbox will autofocus
|
|
66
|
+
// the first item.
|
|
67
|
+
setTimeout(() => {
|
|
68
|
+
if (this.options?.last && this.listboxValue.length === 0) {
|
|
69
|
+
this.listbox?.setActiveOption(this.options?.last.value);
|
|
70
|
+
}
|
|
71
|
+
});
|
|
72
|
+
break;
|
|
73
|
+
case 'ArrowRight':
|
|
74
|
+
case 'ArrowLeft':
|
|
75
|
+
if (this.multiple) {
|
|
76
|
+
this.triggerRef?.open();
|
|
77
|
+
}
|
|
78
|
+
else {
|
|
79
|
+
const option = this.getNextEnabledOptionValue($event.key === 'ArrowLeft' ? 'left' : 'right');
|
|
80
|
+
if (option) {
|
|
81
|
+
this.listboxValue = [option];
|
|
82
|
+
}
|
|
83
|
+
}
|
|
84
|
+
break;
|
|
85
|
+
default:
|
|
86
|
+
this.keyManager?.onKeydown($event);
|
|
87
|
+
}
|
|
88
|
+
}
|
|
89
|
+
getSelectionText() {
|
|
90
|
+
const selectedOptions = this.selectedOptions;
|
|
91
|
+
if (selectedOptions.length === 0) {
|
|
92
|
+
return '';
|
|
93
|
+
}
|
|
94
|
+
if (selectedOptions.length === 1) {
|
|
95
|
+
return selectedOptions[0]?.label;
|
|
96
|
+
}
|
|
97
|
+
return `${selectedOptions.length === this.options.length
|
|
98
|
+
? 'All'
|
|
99
|
+
: selectedOptions.length} selected`;
|
|
100
|
+
}
|
|
101
|
+
/**
|
|
102
|
+
* Whether multiple selection is allowed.
|
|
103
|
+
*/
|
|
104
|
+
get multiple() {
|
|
105
|
+
return this.valueHolder instanceof MultiSelectDirectiveBase;
|
|
106
|
+
}
|
|
107
|
+
ngAfterContentInit() {
|
|
108
|
+
this.initKeyManager();
|
|
109
|
+
this.options.changes.subscribe(() => {
|
|
110
|
+
this.initKeyManager();
|
|
111
|
+
});
|
|
112
|
+
Promise.resolve().then(() => {
|
|
113
|
+
this._initialized = true;
|
|
114
|
+
});
|
|
115
|
+
}
|
|
116
|
+
ngAfterViewInit() {
|
|
117
|
+
/**
|
|
118
|
+
* This is to avoid an NG0100 ExpressionChangedAfter... error when there is an initial value
|
|
119
|
+
* The issue is probably due to the fact that options are defined withing the **content**
|
|
120
|
+
* (sp-option elements), so the listbox value can change after the view of select is checked,
|
|
121
|
+
* if there is a default value. And that leads to the error, since we have bindings in the
|
|
122
|
+
* template, such as `getSelectionText()`, that depend on the listbox value.
|
|
123
|
+
*/
|
|
124
|
+
if (this.listboxValue.length > 0) {
|
|
125
|
+
this.changeDetectorRef.detectChanges();
|
|
126
|
+
}
|
|
127
|
+
}
|
|
128
|
+
ngOnDestroy() {
|
|
129
|
+
this.keyManager?.destroy();
|
|
130
|
+
}
|
|
131
|
+
/**
|
|
132
|
+
* Called when an item in the listbox is selected.
|
|
133
|
+
*/
|
|
134
|
+
_handleSelected() {
|
|
135
|
+
if (!this.multiple) {
|
|
136
|
+
this.triggerRef?.close();
|
|
137
|
+
}
|
|
138
|
+
this.selected.emit();
|
|
139
|
+
}
|
|
140
|
+
isOpen() {
|
|
141
|
+
return this.triggerRef?.isOpen() ?? false;
|
|
142
|
+
}
|
|
143
|
+
open() {
|
|
144
|
+
this.triggerRef?.open();
|
|
145
|
+
}
|
|
146
|
+
close() {
|
|
147
|
+
this.triggerRef?.close();
|
|
148
|
+
}
|
|
149
|
+
toggle() {
|
|
150
|
+
this.triggerRef?.open();
|
|
151
|
+
}
|
|
152
|
+
getLabelText() {
|
|
153
|
+
return (this.getFieldLabel?.() ?? this.ariaLabel)?.trim();
|
|
154
|
+
}
|
|
155
|
+
_onOpen() {
|
|
156
|
+
this.opened.emit();
|
|
157
|
+
}
|
|
158
|
+
_onClose() {
|
|
159
|
+
this.closed.emit();
|
|
160
|
+
this.onTouched();
|
|
161
|
+
}
|
|
162
|
+
/**
|
|
163
|
+
* Implemented as part of ControlValueAccessor.
|
|
164
|
+
*/
|
|
165
|
+
registerOnChange(fn) {
|
|
166
|
+
this.controlValueAccessorChangeFn = fn;
|
|
167
|
+
}
|
|
168
|
+
/**
|
|
169
|
+
* Implemented as part of ControlValueAccessor.
|
|
170
|
+
*/
|
|
171
|
+
registerOnTouched(fn) {
|
|
172
|
+
this.onTouched = fn;
|
|
173
|
+
}
|
|
174
|
+
/**
|
|
175
|
+
* Implemented as part of ControlValueAccessor.
|
|
176
|
+
*/
|
|
177
|
+
setDisabledState(isDisabled) {
|
|
178
|
+
this.disabled = isDisabled;
|
|
179
|
+
}
|
|
180
|
+
/**
|
|
181
|
+
* Implemented as part of ControlValueAccessor.
|
|
182
|
+
*/
|
|
183
|
+
writeValue(value) {
|
|
184
|
+
this.valueHolder.value = value;
|
|
185
|
+
}
|
|
186
|
+
getNextEnabledOptionValue(direction = 'right') {
|
|
187
|
+
const optionsArray = this.options?.toArray() || [];
|
|
188
|
+
let nextNonDisabledOption;
|
|
189
|
+
const selectedOption = this.selectedOptions[0];
|
|
190
|
+
const currentOptionIndex = (selectedOption && optionsArray.indexOf(selectedOption)) ?? -1;
|
|
191
|
+
const notDisabled = (o) => !o.disabled;
|
|
192
|
+
if (currentOptionIndex === -1) {
|
|
193
|
+
nextNonDisabledOption = optionsArray.find(notDisabled);
|
|
194
|
+
}
|
|
195
|
+
else if (direction === 'left') {
|
|
196
|
+
nextNonDisabledOption = optionsArray
|
|
197
|
+
.slice(0, currentOptionIndex)
|
|
198
|
+
.reverse()
|
|
199
|
+
.find(notDisabled);
|
|
200
|
+
}
|
|
201
|
+
else {
|
|
202
|
+
nextNonDisabledOption = optionsArray
|
|
203
|
+
.slice(currentOptionIndex + 1, optionsArray.length)
|
|
204
|
+
.find(notDisabled);
|
|
205
|
+
}
|
|
206
|
+
const nextNonDisabledOptionIndex = nextNonDisabledOption
|
|
207
|
+
? optionsArray.indexOf(nextNonDisabledOption)
|
|
208
|
+
: currentOptionIndex;
|
|
209
|
+
return this.options?.get(nextNonDisabledOptionIndex)?.value;
|
|
210
|
+
}
|
|
211
|
+
initKeyManager() {
|
|
212
|
+
// destroy any existing key manager
|
|
213
|
+
this.keyManager?.destroy();
|
|
214
|
+
this.keyManager = new ListKeyManager(this.options.map((option) => ({
|
|
215
|
+
value: option.value,
|
|
216
|
+
getLabel: () => option.label,
|
|
217
|
+
disabled: !!option.disabled,
|
|
218
|
+
}))).withTypeAhead();
|
|
219
|
+
this.keyManager.change.subscribe(() => {
|
|
220
|
+
const selectedOption = this.options?.find((option) => this.keyManager?.activeItem?.value === option.value);
|
|
221
|
+
if (selectedOption) {
|
|
222
|
+
this.listboxValue = [selectedOption.value];
|
|
223
|
+
}
|
|
224
|
+
});
|
|
225
|
+
}
|
|
226
|
+
}
|
|
227
|
+
SelectComponentBase.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.0", ngImport: i0, type: SelectComponentBase, deps: [{ token: i1.SelectValueHolder }], target: i0.ɵɵFactoryTarget.Directive });
|
|
228
|
+
SelectComponentBase.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "15.2.0", type: SelectComponentBase, inputs: { disabled: "disabled", id: "id", name: "name", placeholder: "placeholder", ariaLabel: ["aria-label", "ariaLabel"], ariaLabelledby: ["aria-labelledby", "ariaLabelledby"] }, outputs: { opened: "opened", closed: "closed", selected: "selected" }, queries: [{ propertyName: "selectionText", first: true, predicate: SelectionTextDirective, descendants: true }, { propertyName: "options", predicate: OptionComponent, descendants: true }, { propertyName: "listboxChildren", predicate: ListboxChild }], ngImport: i0 });
|
|
229
|
+
__decorate([
|
|
230
|
+
CoerceBoolean
|
|
231
|
+
], SelectComponentBase.prototype, "disabled", void 0);
|
|
232
|
+
__decorate([
|
|
233
|
+
AutoId()
|
|
234
|
+
], SelectComponentBase.prototype, "id", void 0);
|
|
235
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.0", ngImport: i0, type: SelectComponentBase, decorators: [{
|
|
236
|
+
type: Directive
|
|
237
|
+
}], ctorParameters: function () { return [{ type: i1.SelectValueHolder }]; }, propDecorators: { disabled: [{
|
|
238
|
+
type: Input
|
|
239
|
+
}], id: [{
|
|
240
|
+
type: Input
|
|
241
|
+
}], name: [{
|
|
242
|
+
type: Input
|
|
243
|
+
}], placeholder: [{
|
|
244
|
+
type: Input
|
|
245
|
+
}], opened: [{
|
|
246
|
+
type: Output
|
|
247
|
+
}], closed: [{
|
|
248
|
+
type: Output
|
|
249
|
+
}], options: [{
|
|
250
|
+
type: ContentChildren,
|
|
251
|
+
args: [OptionComponent, { descendants: true }]
|
|
252
|
+
}], listboxChildren: [{
|
|
253
|
+
type: ContentChildren,
|
|
254
|
+
args: [ListboxChild]
|
|
255
|
+
}], selected: [{
|
|
256
|
+
type: Output
|
|
257
|
+
}], ariaLabel: [{
|
|
258
|
+
type: Input,
|
|
259
|
+
args: ['aria-label']
|
|
260
|
+
}], ariaLabelledby: [{
|
|
261
|
+
type: Input,
|
|
262
|
+
args: ['aria-labelledby']
|
|
263
|
+
}], selectionText: [{
|
|
264
|
+
type: ContentChild,
|
|
265
|
+
args: [SelectionTextDirective]
|
|
266
|
+
}] } });
|
|
267
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"select-component-base.js","sourceRoot":"","sources":["../../../../../../src/lib/select/src/common/select-component-base.ts"],"names":[],"mappings":"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","sourcesContent":["import { CoerceBoolean } from '../../../common/coerce-boolean.decorator';\nimport { BooleanInput } from '@angular/cdk/coercion';\nimport {\n  ListboxChild,\n  ListboxComponent,\n  OptionComponent,\n} from '../../../listbox/public_api';\nimport { ControlValueAccessor } from '@angular/forms';\nimport {\n  AfterContentInit,\n  ChangeDetectorRef,\n  ContentChild,\n  ContentChildren,\n  Directive,\n  ElementRef,\n  EventEmitter,\n  inject,\n  Input,\n  OnDestroy,\n  Output,\n  QueryList,\n} from '@angular/core';\nimport { AutoId } from '../../../common/auto-id.decorator';\nimport {\n  FocusOrigin,\n  ListKeyManager,\n  ListKeyManagerOption,\n} from '@angular/cdk/a11y';\nimport { PopoverTriggerDirective } from '../../../popover/src/popover-trigger.directive';\nimport {\n  MultiSelectDirectiveBase,\n  SelectValueHolder,\n} from './select-value-holder';\nimport { SelectionTextDirective } from '../select/selection-text.directive';\n\n@Directive() // This is needed for decorators like Input to work\nexport abstract class SelectComponentBase\n  implements ControlValueAccessor, AfterContentInit, OnDestroy\n{\n  /**\n   * Whether the component is disabled.\n   */\n  @Input()\n  @CoerceBoolean\n  disabled: BooleanInput;\n\n  /** A unique id for the select. If none is supplied, it will be auto-generated. */\n  @Input()\n  @AutoId()\n  id = '';\n\n  /** The name of the input, used when submitting an HTML form. */\n  @Input()\n  name?: string;\n\n  /**\n   * Placeholder to be shown if no value has been selected.\n   */\n  @Input() placeholder?: string;\n\n  /**\n   * Emitted when the popover is opened\n   */\n  @Output()\n  readonly opened: EventEmitter<void> = new EventEmitter();\n\n  /**\n   * Emitted when the popover is closed\n   */\n  @Output()\n  readonly closed: EventEmitter<void> = new EventEmitter();\n\n  @ContentChildren(OptionComponent, { descendants: true })\n  public options: QueryList<OptionComponent> = new QueryList<OptionComponent>();\n\n  @ContentChildren(ListboxChild)\n  public listboxChildren: QueryList<ListboxChild> = new QueryList<ListboxChild>();\n\n  /**\n   * Manages keyboard events for options in the panel.\n   */\n  protected keyManager:\n    | ListKeyManager<{ value: string } & ListKeyManagerOption>\n    | undefined;\n\n  /**\n   * Emits whenever the dropdown is closed as a result of an option (already\n   * selected or not) being selected.\n   */\n  @Output()\n  selected = new EventEmitter();\n\n  @Input('aria-label')\n  ariaLabel: string | null = null;\n  @Input('aria-labelledby')\n  ariaLabelledby?: string | null = null;\n\n  @ContentChild(SelectionTextDirective)\n  selectionText?: SelectionTextDirective;\n\n  /**\n   * Used to work around ExpressionChangedAfterItHasBeenCheckedError error thrown from HiddenSelect,\n   * when the select component is rendered from a template (e.g. with *ngIf). It's set to true\n   * asynchronously after content is initialized, to delay rendering of things that depend on\n   * options. It's not clear why the issue happens, and it only happens when select is used with\n   * *ngIf, or generally in a template.\n   */\n  _initialized = false;\n\n  set listboxValue(value: readonly string[]) {\n    this.valueHolder.setFromListboxValue(value);\n    this.controlValueAccessorChangeFn(this.valueHolder.value);\n  }\n  get listboxValue() {\n    return this.valueHolder.getListboxValue();\n  }\n  get selectedOptions() {\n    return this.options?.filter((option) =>\n      this.listboxValue.includes(option.value)\n    );\n  }\n\n  private changeDetectorRef = inject(ChangeDetectorRef);\n\n  constructor(protected valueHolder: SelectValueHolder) {}\n\n  protected abstract triggerRef: PopoverTriggerDirective | undefined;\n  protected abstract triggerElementRef: ElementRef<HTMLElement> | undefined;\n  protected abstract listbox: ListboxComponent | undefined;\n  public abstract focus(focusOrigin: FocusOrigin): void;\n  public abstract isFocused(): boolean;\n\n  handleKeyDown($event: KeyboardEvent) {\n    if ($event.target !== this.triggerElementRef?.nativeElement) {\n      return;\n    }\n    switch ($event.key) {\n      case 'ArrowDown':\n        this.triggerRef?.open();\n        break;\n      case 'ArrowUp':\n        this.triggerRef?.open();\n        // we need to delay it a little, since the listbox will autofocus\n        // the first item.\n        setTimeout(() => {\n          if (this.options?.last && this.listboxValue.length === 0) {\n            this.listbox?.setActiveOption(this.options?.last.value);\n          }\n        });\n        break;\n      case 'ArrowRight':\n      case 'ArrowLeft':\n        if (this.multiple) {\n          this.triggerRef?.open();\n        } else {\n          const option = this.getNextEnabledOptionValue(\n            $event.key === 'ArrowLeft' ? 'left' : 'right'\n          );\n          if (option) {\n            this.listboxValue = [option];\n          }\n        }\n        break;\n      default:\n        this.keyManager?.onKeydown($event);\n    }\n  }\n\n  getSelectionText(): string {\n    const selectedOptions = this.selectedOptions;\n    if (selectedOptions.length === 0) {\n      return '';\n    }\n    if (selectedOptions.length === 1) {\n      return selectedOptions[0]?.label;\n    }\n    return `${\n      selectedOptions.length === this.options.length\n        ? 'All'\n        : selectedOptions.length\n    } selected`;\n  }\n\n  /**\n   * Whether multiple selection is allowed.\n   */\n  get multiple() {\n    return this.valueHolder instanceof MultiSelectDirectiveBase;\n  }\n\n  ngAfterContentInit(): void {\n    this.initKeyManager();\n\n    this.options.changes.subscribe(() => {\n      this.initKeyManager();\n    });\n    Promise.resolve().then(() => {\n      this._initialized = true;\n    });\n  }\n\n  ngAfterViewInit() {\n    /**\n     * This is to avoid an NG0100 ExpressionChangedAfter... error when there is an initial value\n     * The issue is probably due to the fact that options are defined withing the **content**\n     * (sp-option elements), so the listbox value can change after the view of select is checked,\n     * if there is a default value. And that leads to the error, since we have bindings in the\n     * template, such as `getSelectionText()`, that depend on the listbox value.\n     */\n    if (this.listboxValue.length > 0) {\n      this.changeDetectorRef.detectChanges();\n    }\n  }\n\n  ngOnDestroy(): void {\n    this.keyManager?.destroy();\n  }\n\n  /**\n   * Called when an item in the listbox is selected.\n   */\n  _handleSelected() {\n    if (!this.multiple) {\n      this.triggerRef?.close();\n    }\n    this.selected.emit();\n  }\n\n  isOpen(): boolean {\n    return this.triggerRef?.isOpen() ?? false;\n  }\n\n  open() {\n    this.triggerRef?.open();\n  }\n\n  close() {\n    this.triggerRef?.close();\n  }\n\n  toggle() {\n    this.triggerRef?.open();\n  }\n\n  protected getFieldLabel?(): string;\n\n  getLabelText(): string | undefined {\n    return (this.getFieldLabel?.() ?? this.ariaLabel)?.trim();\n  }\n\n  protected controlValueAccessorChangeFn: (value: any) => void = () => {};\n  public onTouched: () => any = () => {};\n\n  public _onOpen(): void {\n    this.opened.emit();\n  }\n\n  public _onClose(): void {\n    this.closed.emit();\n    this.onTouched();\n  }\n\n  /**\n   * Implemented as part of ControlValueAccessor.\n   */\n  registerOnChange(fn: any): void {\n    this.controlValueAccessorChangeFn = fn;\n  }\n\n  /**\n   * Implemented as part of ControlValueAccessor.\n   */\n  registerOnTouched(fn: any): void {\n    this.onTouched = fn;\n  }\n\n  /**\n   * Implemented as part of ControlValueAccessor.\n   */\n  setDisabledState(isDisabled: boolean): void {\n    this.disabled = isDisabled;\n  }\n\n  /**\n   * Implemented as part of ControlValueAccessor.\n   */\n  writeValue(value: unknown): void {\n    this.valueHolder.value = value;\n  }\n\n  protected getNextEnabledOptionValue(direction: 'left' | 'right' = 'right') {\n    const optionsArray = this.options?.toArray() || [];\n    let nextNonDisabledOption;\n    const selectedOption = this.selectedOptions[0];\n\n    const currentOptionIndex =\n      (selectedOption && optionsArray.indexOf(selectedOption)) ?? -1;\n    const notDisabled = (o: OptionComponent) => !o.disabled;\n\n    if (currentOptionIndex === -1) {\n      nextNonDisabledOption = optionsArray.find(notDisabled);\n    } else if (direction === 'left') {\n      nextNonDisabledOption = optionsArray\n        .slice(0, currentOptionIndex)\n        .reverse()\n        .find(notDisabled);\n    } else {\n      nextNonDisabledOption = optionsArray\n        .slice(currentOptionIndex + 1, optionsArray.length)\n        .find(notDisabled);\n    }\n\n    const nextNonDisabledOptionIndex = nextNonDisabledOption\n      ? optionsArray.indexOf(nextNonDisabledOption)\n      : currentOptionIndex;\n    return this.options?.get(nextNonDisabledOptionIndex)?.value;\n  }\n\n  protected initKeyManager(): void {\n    // destroy any existing key manager\n    this.keyManager?.destroy();\n\n    this.keyManager = new ListKeyManager<\n      { value: string } & ListKeyManagerOption\n    >(\n      this.options.map((option) => ({\n        value: option.value,\n        getLabel: () => option.label,\n        disabled: !!option.disabled,\n      }))\n    ).withTypeAhead();\n\n    this.keyManager.change.subscribe(() => {\n      const selectedOption = this.options?.find(\n        (option) => this.keyManager?.activeItem?.value === option.value\n      );\n\n      if (selectedOption) {\n        this.listboxValue = [selectedOption.value];\n      }\n    });\n  }\n}\n"]}
|
|
@@ -0,0 +1,86 @@
|
|
|
1
|
+
import { Directive, EventEmitter, Input, Output } from '@angular/core';
|
|
2
|
+
import * as i0 from "@angular/core";
|
|
3
|
+
/**
|
|
4
|
+
* A common base class to handle value for single and multiple selection.
|
|
5
|
+
* {@link MultiSelectDirectiveBase} and {@link SingleSelectDirectiveBase} implement and
|
|
6
|
+
* provide this interface. Note that the main purpose for such design is **not**
|
|
7
|
+
* to abstract out selection behavior. But to have accurate `value` and `valueChange`
|
|
8
|
+
* type for single and multiple selection mode. It would be possible to fully abstract out
|
|
9
|
+
* the selection behavior, and have Select component not know anything about selection being
|
|
10
|
+
* multiple or not, but it doesn't seem such a clear abstraction, and also not worth the additional
|
|
11
|
+
* complexity.
|
|
12
|
+
*
|
|
13
|
+
* At least one directive has to be defined for each select component to provide SelectValueHolder,
|
|
14
|
+
* extending either {@link MultiSelectDirectiveBase} or {@link SingleSelectDirectiveBase}
|
|
15
|
+
*/
|
|
16
|
+
export class SelectValueHolder {
|
|
17
|
+
}
|
|
18
|
+
/**
|
|
19
|
+
* Enables multiple selection. When present, `value` will be of type `string[]`, and `valueChange`
|
|
20
|
+
* will emit `string[]`.
|
|
21
|
+
*
|
|
22
|
+
* Each select component can define a directive with the right selector (usually based on
|
|
23
|
+
* the presence of `multiple` attribute) to enable multiple selection.
|
|
24
|
+
*/
|
|
25
|
+
export class MultiSelectDirectiveBase extends SelectValueHolder {
|
|
26
|
+
constructor() {
|
|
27
|
+
super(...arguments);
|
|
28
|
+
this.value = [];
|
|
29
|
+
this.valueChange = new EventEmitter();
|
|
30
|
+
}
|
|
31
|
+
setFromListboxValue(value) {
|
|
32
|
+
this.value = value;
|
|
33
|
+
this.valueChange.emit(this.value);
|
|
34
|
+
}
|
|
35
|
+
getListboxValue() {
|
|
36
|
+
return this.value || [];
|
|
37
|
+
}
|
|
38
|
+
}
|
|
39
|
+
MultiSelectDirectiveBase.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.0", ngImport: i0, type: MultiSelectDirectiveBase, deps: null, target: i0.ɵɵFactoryTarget.Directive });
|
|
40
|
+
MultiSelectDirectiveBase.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "15.2.0", type: MultiSelectDirectiveBase, inputs: { value: "value" }, outputs: { valueChange: "valueChange" }, usesInheritance: true, ngImport: i0 });
|
|
41
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.0", ngImport: i0, type: MultiSelectDirectiveBase, decorators: [{
|
|
42
|
+
type: Directive
|
|
43
|
+
}], propDecorators: { value: [{
|
|
44
|
+
type: Input
|
|
45
|
+
}], valueChange: [{
|
|
46
|
+
type: Output
|
|
47
|
+
}] } });
|
|
48
|
+
/**
|
|
49
|
+
* Enables single selection, the default behavior of select.
|
|
50
|
+
* When present, `value` will be of type `string`, and `valueChange` will emit `string`.
|
|
51
|
+
*
|
|
52
|
+
* Each select component has to define a directive with the right selector (usually based on
|
|
53
|
+
* the absence of `multiple` attribute) to enable single selection.
|
|
54
|
+
*/
|
|
55
|
+
export class SingleSelectDirectiveBase extends SelectValueHolder {
|
|
56
|
+
constructor() {
|
|
57
|
+
super(...arguments);
|
|
58
|
+
this._value = '';
|
|
59
|
+
this.values = [];
|
|
60
|
+
this.valueChange = new EventEmitter();
|
|
61
|
+
}
|
|
62
|
+
get value() {
|
|
63
|
+
return this._value;
|
|
64
|
+
}
|
|
65
|
+
set value(value) {
|
|
66
|
+
this._value = value;
|
|
67
|
+
this.values = value ? [value] : [];
|
|
68
|
+
}
|
|
69
|
+
setFromListboxValue(value) {
|
|
70
|
+
this.value = value[0];
|
|
71
|
+
this.valueChange.emit(this._value);
|
|
72
|
+
}
|
|
73
|
+
getListboxValue() {
|
|
74
|
+
return this.values;
|
|
75
|
+
}
|
|
76
|
+
}
|
|
77
|
+
SingleSelectDirectiveBase.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.0", ngImport: i0, type: SingleSelectDirectiveBase, deps: null, target: i0.ɵɵFactoryTarget.Directive });
|
|
78
|
+
SingleSelectDirectiveBase.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "15.2.0", type: SingleSelectDirectiveBase, inputs: { value: "value" }, outputs: { valueChange: "valueChange" }, usesInheritance: true, ngImport: i0 });
|
|
79
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.0", ngImport: i0, type: SingleSelectDirectiveBase, decorators: [{
|
|
80
|
+
type: Directive
|
|
81
|
+
}], propDecorators: { value: [{
|
|
82
|
+
type: Input
|
|
83
|
+
}], valueChange: [{
|
|
84
|
+
type: Output
|
|
85
|
+
}] } });
|
|
86
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic2VsZWN0LXZhbHVlLWhvbGRlci5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uL3NyYy9saWIvc2VsZWN0L3NyYy9jb21tb24vc2VsZWN0LXZhbHVlLWhvbGRlci50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsU0FBUyxFQUFFLFlBQVksRUFBRSxLQUFLLEVBQUUsTUFBTSxFQUFFLE1BQU0sZUFBZSxDQUFDOztBQUV2RTs7Ozs7Ozs7Ozs7O0dBWUc7QUFDSCxNQUFNLE9BQWdCLGlCQUFpQjtDQU90QztBQUVEOzs7Ozs7R0FNRztBQUVILE1BQU0sT0FBTyx3QkFBeUIsU0FBUSxpQkFBMkI7SUFEekU7O1FBR0UsVUFBSyxHQUFhLEVBQUUsQ0FBQztRQUdyQixnQkFBVyxHQUFHLElBQUksWUFBWSxFQUFZLENBQUM7S0FTNUM7SUFQQyxtQkFBbUIsQ0FBQyxLQUFlO1FBQ2pDLElBQUksQ0FBQyxLQUFLLEdBQUcsS0FBSyxDQUFDO1FBQ25CLElBQUksQ0FBQyxXQUFXLENBQUMsSUFBSSxDQUFDLElBQUksQ0FBQyxLQUFLLENBQUMsQ0FBQztJQUNwQyxDQUFDO0lBQ0QsZUFBZTtRQUNiLE9BQU8sSUFBSSxDQUFDLEtBQUssSUFBSSxFQUFFLENBQUM7SUFDMUIsQ0FBQzs7cUhBYlUsd0JBQXdCO3lHQUF4Qix3QkFBd0I7MkZBQXhCLHdCQUF3QjtrQkFEcEMsU0FBUzs4QkFHUixLQUFLO3NCQURKLEtBQUs7Z0JBSU4sV0FBVztzQkFEVixNQUFNOztBQVlUOzs7Ozs7R0FNRztBQUVILE1BQU0sT0FBTyx5QkFBMEIsU0FBUSxpQkFBeUI7SUFEeEU7O1FBVVUsV0FBTSxHQUFXLEVBQUUsQ0FBQztRQUVwQixXQUFNLEdBQWEsRUFBRSxDQUFDO1FBRzlCLGdCQUFXLEdBQUcsSUFBSSxZQUFZLEVBQVUsQ0FBQztLQVUxQztJQXZCQyxJQUNJLEtBQUs7UUFDUCxPQUFPLElBQUksQ0FBQyxNQUFNLENBQUM7SUFDckIsQ0FBQztJQUNELElBQUksS0FBSyxDQUFDLEtBQWE7UUFDckIsSUFBSSxDQUFDLE1BQU0sR0FBRyxLQUFLLENBQUM7UUFDcEIsSUFBSSxDQUFDLE1BQU0sR0FBRyxLQUFLLENBQUMsQ0FBQyxDQUFDLENBQUMsS0FBSyxDQUFDLENBQUMsQ0FBQyxDQUFDLEVBQUUsQ0FBQztJQUNyQyxDQUFDO0lBUUQsbUJBQW1CLENBQUMsS0FBZTtRQUNqQyxJQUFJLENBQUMsS0FBSyxHQUFHLEtBQUssQ0FBQyxDQUFDLENBQUMsQ0FBQztRQUN0QixJQUFJLENBQUMsV0FBVyxDQUFDLElBQUksQ0FBQyxJQUFJLENBQUMsTUFBTSxDQUFDLENBQUM7SUFDckMsQ0FBQztJQUVELGVBQWU7UUFDYixPQUFPLElBQUksQ0FBQyxNQUFNLENBQUM7SUFDckIsQ0FBQzs7c0hBdkJVLHlCQUF5QjswR0FBekIseUJBQXlCOzJGQUF6Qix5QkFBeUI7a0JBRHJDLFNBQVM7OEJBR0osS0FBSztzQkFEUixLQUFLO2dCQWFOLFdBQVc7c0JBRFYsTUFBTSIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IERpcmVjdGl2ZSwgRXZlbnRFbWl0dGVyLCBJbnB1dCwgT3V0cHV0IH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5cbi8qKlxuICogQSBjb21tb24gYmFzZSBjbGFzcyB0byBoYW5kbGUgdmFsdWUgZm9yIHNpbmdsZSBhbmQgbXVsdGlwbGUgc2VsZWN0aW9uLlxuICoge0BsaW5rIE11bHRpU2VsZWN0RGlyZWN0aXZlQmFzZX0gYW5kIHtAbGluayBTaW5nbGVTZWxlY3REaXJlY3RpdmVCYXNlfSBpbXBsZW1lbnQgYW5kXG4gKiBwcm92aWRlIHRoaXMgaW50ZXJmYWNlLiBOb3RlIHRoYXQgdGhlIG1haW4gcHVycG9zZSBmb3Igc3VjaCBkZXNpZ24gaXMgKipub3QqKlxuICogdG8gYWJzdHJhY3Qgb3V0IHNlbGVjdGlvbiBiZWhhdmlvci4gQnV0IHRvIGhhdmUgYWNjdXJhdGUgYHZhbHVlYCBhbmQgYHZhbHVlQ2hhbmdlYFxuICogdHlwZSBmb3Igc2luZ2xlIGFuZCBtdWx0aXBsZSBzZWxlY3Rpb24gbW9kZS4gSXQgd291bGQgYmUgcG9zc2libGUgdG8gZnVsbHkgYWJzdHJhY3Qgb3V0XG4gKiB0aGUgc2VsZWN0aW9uIGJlaGF2aW9yLCBhbmQgaGF2ZSBTZWxlY3QgY29tcG9uZW50IG5vdCBrbm93IGFueXRoaW5nIGFib3V0IHNlbGVjdGlvbiBiZWluZ1xuICogbXVsdGlwbGUgb3Igbm90LCBidXQgaXQgZG9lc24ndCBzZWVtIHN1Y2ggYSBjbGVhciBhYnN0cmFjdGlvbiwgYW5kIGFsc28gbm90IHdvcnRoIHRoZSBhZGRpdGlvbmFsXG4gKiBjb21wbGV4aXR5LlxuICpcbiAqIEF0IGxlYXN0IG9uZSBkaXJlY3RpdmUgaGFzIHRvIGJlIGRlZmluZWQgZm9yIGVhY2ggc2VsZWN0IGNvbXBvbmVudCB0byBwcm92aWRlIFNlbGVjdFZhbHVlSG9sZGVyLFxuICogZXh0ZW5kaW5nIGVpdGhlciB7QGxpbmsgTXVsdGlTZWxlY3REaXJlY3RpdmVCYXNlfSBvciB7QGxpbmsgU2luZ2xlU2VsZWN0RGlyZWN0aXZlQmFzZX1cbiAqL1xuZXhwb3J0IGFic3RyYWN0IGNsYXNzIFNlbGVjdFZhbHVlSG9sZGVyPFQgPSB1bmtub3duPiB7XG4gIGFic3RyYWN0IHZhbHVlOiBUO1xuICBhYnN0cmFjdCB2YWx1ZUNoYW5nZTogRXZlbnRFbWl0dGVyPFQ+O1xuXG4gIGFic3RyYWN0IHNldEZyb21MaXN0Ym94VmFsdWUodmFsdWU6IHJlYWRvbmx5IHN0cmluZ1tdKTogdm9pZDtcblxuICBhYnN0cmFjdCBnZXRMaXN0Ym94VmFsdWUoKTogcmVhZG9ubHkgc3RyaW5nW107XG59XG5cbi8qKlxuICogRW5hYmxlcyBtdWx0aXBsZSBzZWxlY3Rpb24uIFdoZW4gcHJlc2VudCwgYHZhbHVlYCB3aWxsIGJlIG9mIHR5cGUgYHN0cmluZ1tdYCwgYW5kIGB2YWx1ZUNoYW5nZWBcbiAqIHdpbGwgZW1pdCBgc3RyaW5nW11gLlxuICpcbiAqIEVhY2ggc2VsZWN0IGNvbXBvbmVudCBjYW4gZGVmaW5lIGEgZGlyZWN0aXZlIHdpdGggdGhlIHJpZ2h0IHNlbGVjdG9yICh1c3VhbGx5IGJhc2VkIG9uXG4gKiB0aGUgcHJlc2VuY2Ugb2YgYG11bHRpcGxlYCBhdHRyaWJ1dGUpIHRvIGVuYWJsZSBtdWx0aXBsZSBzZWxlY3Rpb24uXG4gKi9cbkBEaXJlY3RpdmUoKSAvLyBUaGlzIGlzIG5lZWRlZCBmb3IgZGVjb3JhdG9ycyBsaWtlIElucHV0IHRvIHdvcmtcbmV4cG9ydCBjbGFzcyBNdWx0aVNlbGVjdERpcmVjdGl2ZUJhc2UgZXh0ZW5kcyBTZWxlY3RWYWx1ZUhvbGRlcjxzdHJpbmdbXT4ge1xuICBASW5wdXQoKVxuICB2YWx1ZTogc3RyaW5nW10gPSBbXTtcblxuICBAT3V0cHV0KClcbiAgdmFsdWVDaGFuZ2UgPSBuZXcgRXZlbnRFbWl0dGVyPHN0cmluZ1tdPigpO1xuXG4gIHNldEZyb21MaXN0Ym94VmFsdWUodmFsdWU6IHN0cmluZ1tdKTogdm9pZCB7XG4gICAgdGhpcy52YWx1ZSA9IHZhbHVlO1xuICAgIHRoaXMudmFsdWVDaGFuZ2UuZW1pdCh0aGlzLnZhbHVlKTtcbiAgfVxuICBnZXRMaXN0Ym94VmFsdWUoKTogcmVhZG9ubHkgc3RyaW5nW10ge1xuICAgIHJldHVybiB0aGlzLnZhbHVlIHx8IFtdO1xuICB9XG59XG5cbi8qKlxuICogRW5hYmxlcyBzaW5nbGUgc2VsZWN0aW9uLCB0aGUgZGVmYXVsdCBiZWhhdmlvciBvZiBzZWxlY3QuXG4gKiBXaGVuIHByZXNlbnQsIGB2YWx1ZWAgd2lsbCBiZSBvZiB0eXBlIGBzdHJpbmdgLCBhbmQgYHZhbHVlQ2hhbmdlYCB3aWxsIGVtaXQgYHN0cmluZ2AuXG4gKlxuICogRWFjaCBzZWxlY3QgY29tcG9uZW50IGhhcyB0byBkZWZpbmUgYSBkaXJlY3RpdmUgd2l0aCB0aGUgcmlnaHQgc2VsZWN0b3IgKHVzdWFsbHkgYmFzZWQgb25cbiAqIHRoZSBhYnNlbmNlIG9mIGBtdWx0aXBsZWAgYXR0cmlidXRlKSB0byBlbmFibGUgc2luZ2xlIHNlbGVjdGlvbi5cbiAqL1xuQERpcmVjdGl2ZSgpIC8vIFRoaXMgaXMgbmVlZGVkIGZvciBkZWNvcmF0b3JzIGxpa2UgSW5wdXQgdG8gd29ya1xuZXhwb3J0IGNsYXNzIFNpbmdsZVNlbGVjdERpcmVjdGl2ZUJhc2UgZXh0ZW5kcyBTZWxlY3RWYWx1ZUhvbGRlcjxzdHJpbmc+IHtcbiAgQElucHV0KClcbiAgZ2V0IHZhbHVlKCk6IHN0cmluZyB7XG4gICAgcmV0dXJuIHRoaXMuX3ZhbHVlO1xuICB9XG4gIHNldCB2YWx1ZSh2YWx1ZTogc3RyaW5nKSB7XG4gICAgdGhpcy5fdmFsdWUgPSB2YWx1ZTtcbiAgICB0aGlzLnZhbHVlcyA9IHZhbHVlID8gW3ZhbHVlXSA6IFtdO1xuICB9XG4gIHByaXZhdGUgX3ZhbHVlOiBzdHJpbmcgPSAnJztcblxuICBwcml2YXRlIHZhbHVlczogc3RyaW5nW10gPSBbXTtcblxuICBAT3V0cHV0KClcbiAgdmFsdWVDaGFuZ2UgPSBuZXcgRXZlbnRFbWl0dGVyPHN0cmluZz4oKTtcblxuICBzZXRGcm9tTGlzdGJveFZhbHVlKHZhbHVlOiBzdHJpbmdbXSk6IHZvaWQge1xuICAgIHRoaXMudmFsdWUgPSB2YWx1ZVswXTtcbiAgICB0aGlzLnZhbHVlQ2hhbmdlLmVtaXQodGhpcy5fdmFsdWUpO1xuICB9XG5cbiAgZ2V0TGlzdGJveFZhbHVlKCk6IHJlYWRvbmx5IHN0cmluZ1tdIHtcbiAgICByZXR1cm4gdGhpcy52YWx1ZXM7XG4gIH1cbn1cbiJdfQ==
|
|
@@ -0,0 +1,90 @@
|
|
|
1
|
+
import { __decorate } from "tslib";
|
|
2
|
+
import { Directive, EventEmitter, Input, Output } from '@angular/core';
|
|
3
|
+
import { CoerceBoolean } from '../../../common/coerce-boolean.decorator';
|
|
4
|
+
import { OptionComponent } from '../../../listbox/src/option.component';
|
|
5
|
+
import { section, SectionDirective, } from '../../../listbox/src/section.directive';
|
|
6
|
+
import * as i0 from "@angular/core";
|
|
7
|
+
/**
|
|
8
|
+
* Marks sp-select as searchable. When used, a search input is rendered inside select trigger
|
|
9
|
+
* which allows filtering options, in a more convenient way than the default typeahead search.
|
|
10
|
+
* By default, rendered `sp-option`s are filtered based on the search value, but `searchValue`
|
|
11
|
+
* and `searchValueChange` can be used to control the search value and filter the `sp-options`
|
|
12
|
+
* upfront.
|
|
13
|
+
*/
|
|
14
|
+
export class SearchableSelectDirective {
|
|
15
|
+
constructor() {
|
|
16
|
+
/**
|
|
17
|
+
* Filtering function to be applied on each item. The default filter function
|
|
18
|
+
* does a case-insensitive search on optionComponent.label.
|
|
19
|
+
* The built-in filtering can be disabled by passing `null`. That's useful
|
|
20
|
+
* when filtering is fully controlled by filtering input sp-options based on
|
|
21
|
+
* {@property searchValue}.
|
|
22
|
+
*/
|
|
23
|
+
this.filter = (optionComponent, searchQuery) => optionComponent.label.toLowerCase().includes(searchQuery?.toLowerCase());
|
|
24
|
+
this.searchValueChange = new EventEmitter();
|
|
25
|
+
}
|
|
26
|
+
get searchValue() {
|
|
27
|
+
return this._searchValue ?? '';
|
|
28
|
+
}
|
|
29
|
+
set searchValue(value) {
|
|
30
|
+
this._searchValue = value;
|
|
31
|
+
}
|
|
32
|
+
filterListboxItems(items, filter) {
|
|
33
|
+
const filteredOutOptions = [];
|
|
34
|
+
const filteredChildren = Array.from(items)
|
|
35
|
+
?.map((item) => {
|
|
36
|
+
if (item instanceof OptionComponent) {
|
|
37
|
+
const isMatch = filter(item);
|
|
38
|
+
if (!isMatch) {
|
|
39
|
+
filteredOutOptions.push(item);
|
|
40
|
+
}
|
|
41
|
+
return isMatch ? item : null;
|
|
42
|
+
}
|
|
43
|
+
if (item instanceof SectionDirective) {
|
|
44
|
+
const { filteredChildren: sectionFilteredChildren, filteredOutOptions: sectionFilteredOutOptions, } = this.filterListboxItems(item.children, filter);
|
|
45
|
+
filteredOutOptions.push(...sectionFilteredOutOptions);
|
|
46
|
+
if (sectionFilteredChildren.length > 0) {
|
|
47
|
+
return section(sectionFilteredChildren.filter((item) => item instanceof OptionComponent), item.ariaLabel);
|
|
48
|
+
}
|
|
49
|
+
return null;
|
|
50
|
+
}
|
|
51
|
+
return null;
|
|
52
|
+
})
|
|
53
|
+
.filter(notNull);
|
|
54
|
+
return { filteredChildren, filteredOutOptions };
|
|
55
|
+
}
|
|
56
|
+
/**
|
|
57
|
+
* Used when search value is updated via UI, and searchValueChange output should be emitted.
|
|
58
|
+
* @internal
|
|
59
|
+
*/
|
|
60
|
+
setSearchValue(searchValue) {
|
|
61
|
+
this._searchValue = searchValue;
|
|
62
|
+
this.searchValueChange.emit(searchValue);
|
|
63
|
+
}
|
|
64
|
+
defaultFilter(optionComponent, searchQuery) {
|
|
65
|
+
return optionComponent.label
|
|
66
|
+
.toLowerCase()
|
|
67
|
+
.includes(searchQuery?.toLowerCase());
|
|
68
|
+
}
|
|
69
|
+
}
|
|
70
|
+
SearchableSelectDirective.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.0", ngImport: i0, type: SearchableSelectDirective, deps: [], target: i0.ɵɵFactoryTarget.Directive });
|
|
71
|
+
SearchableSelectDirective.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "15.2.0", type: SearchableSelectDirective, selector: "sp-select[searchable]", inputs: { searchValue: "searchValue", filter: "filter", selectTextOnOpen: "selectTextOnOpen" }, outputs: { searchValueChange: "searchValueChange" }, ngImport: i0 });
|
|
72
|
+
__decorate([
|
|
73
|
+
CoerceBoolean
|
|
74
|
+
], SearchableSelectDirective.prototype, "selectTextOnOpen", void 0);
|
|
75
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.0", ngImport: i0, type: SearchableSelectDirective, decorators: [{
|
|
76
|
+
type: Directive,
|
|
77
|
+
args: [{
|
|
78
|
+
selector: 'sp-select[searchable]',
|
|
79
|
+
}]
|
|
80
|
+
}], propDecorators: { searchValue: [{
|
|
81
|
+
type: Input
|
|
82
|
+
}], filter: [{
|
|
83
|
+
type: Input
|
|
84
|
+
}], selectTextOnOpen: [{
|
|
85
|
+
type: Input
|
|
86
|
+
}], searchValueChange: [{
|
|
87
|
+
type: Output
|
|
88
|
+
}] } });
|
|
89
|
+
export const notNull = (item) => item != null;
|
|
90
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"searchable-select.directive.js","sourceRoot":"","sources":["../../../../../../src/lib/select/src/select/searchable-select.directive.ts"],"names":[],"mappings":"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","sourcesContent":["import { Directive, EventEmitter, Input, Output } from '@angular/core';\nimport { CoerceBoolean } from '../../../common/coerce-boolean.decorator';\nimport { BooleanInput } from '@angular/cdk/coercion';\nimport { ListboxChild } from '../../../listbox/src/listbox-child';\nimport { OptionComponent } from '../../../listbox/src/option.component';\nimport {\n  section,\n  SectionDirective,\n} from '../../../listbox/src/section.directive';\n\n/**\n * Marks sp-select as searchable. When used, a search input is rendered inside select trigger\n * which allows filtering options, in a more convenient way than the default typeahead search.\n * By default, rendered `sp-option`s are filtered based on the search value, but `searchValue`\n * and `searchValueChange` can be used to control the search value and filter the `sp-options`\n * upfront.\n */\n@Directive({\n  selector: 'sp-select[searchable]',\n})\nexport class SearchableSelectDirective {\n  @Input()\n  get searchValue() {\n    return this._searchValue ?? '';\n  }\n  set searchValue(value: string | undefined) {\n    this._searchValue = value;\n  }\n  private _searchValue: string | undefined;\n\n  /**\n   * Filtering function to be applied on each item. The default filter function\n   * does a case-insensitive search on optionComponent.label.\n   * The built-in filtering can be disabled by passing `null`. That's useful\n   * when filtering is fully controlled by filtering input sp-options based on\n   * {@property searchValue}.\n   */\n  @Input()\n  filter?:\n    | null\n    | ((optionComponent: OptionComponent, searchQuery: string) => boolean) = (\n    optionComponent: OptionComponent,\n    searchQuery: string\n  ) => optionComponent.label.toLowerCase().includes(searchQuery?.toLowerCase());\n\n  @Input()\n  @CoerceBoolean\n  selectTextOnOpen?: BooleanInput;\n\n  @Output()\n  searchValueChange = new EventEmitter<string>();\n\n  filterListboxItems(\n    items: Iterable<ListboxChild>,\n    filter: (option: OptionComponent) => boolean\n  ): {\n    filteredChildren: ListboxChild[];\n    filteredOutOptions: OptionComponent[];\n  } {\n    const filteredOutOptions: OptionComponent[] = [];\n    const filteredChildren = Array.from(items)\n      ?.map((item): ListboxChild | null => {\n        if (item instanceof OptionComponent) {\n          const isMatch = filter(item);\n          if (!isMatch) {\n            filteredOutOptions.push(item);\n          }\n          return isMatch ? item : null;\n        }\n        if (item instanceof SectionDirective) {\n          const {\n            filteredChildren: sectionFilteredChildren,\n            filteredOutOptions: sectionFilteredOutOptions,\n          } = this.filterListboxItems(item.children, filter);\n          filteredOutOptions.push(...sectionFilteredOutOptions);\n          if (sectionFilteredChildren.length > 0) {\n            return section(\n              sectionFilteredChildren.filter(\n                (item): item is OptionComponent =>\n                  item instanceof OptionComponent\n              ),\n              item.ariaLabel\n            );\n          }\n          return null;\n        }\n        return null;\n      })\n      .filter(notNull);\n    return { filteredChildren, filteredOutOptions };\n  }\n\n  /**\n   * Used when search value is updated via UI, and searchValueChange output should be emitted.\n   * @internal\n   */\n  setSearchValue(searchValue: string) {\n    this._searchValue = searchValue;\n    this.searchValueChange.emit(searchValue);\n  }\n\n  defaultFilter(optionComponent: OptionComponent, searchQuery: string) {\n    return optionComponent.label\n      .toLowerCase()\n      .includes(searchQuery?.toLowerCase());\n  }\n}\n\nexport const notNull = <T>(\n  item: T | undefined | null\n): item is NonNullable<T> => item != null;\n"]}
|