@angular/material 10.0.0-rc.3 → 10.1.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/_theming.scss +17 -17
- package/autocomplete/index.metadata.json +1 -1
- package/autocomplete/testing/autocomplete-harness.d.ts +2 -0
- package/bundles/material-autocomplete-testing.umd.js +11 -0
- package/bundles/material-autocomplete-testing.umd.js.map +1 -1
- package/bundles/material-autocomplete-testing.umd.min.js +2 -2
- package/bundles/material-autocomplete-testing.umd.min.js.map +1 -1
- package/bundles/material-autocomplete.umd.js +3 -1
- package/bundles/material-autocomplete.umd.js.map +1 -1
- package/bundles/material-autocomplete.umd.min.js +5 -5
- package/bundles/material-autocomplete.umd.min.js.map +1 -1
- package/bundles/material-badge-testing.umd.min.js +1 -1
- package/bundles/material-badge-testing.umd.min.js.map +1 -1
- package/bundles/material-bottom-sheet-testing.umd.min.js +1 -1
- package/bundles/material-bottom-sheet-testing.umd.min.js.map +1 -1
- package/bundles/material-button-testing.umd.js +11 -0
- package/bundles/material-button-testing.umd.js.map +1 -1
- package/bundles/material-button-testing.umd.min.js +2 -2
- package/bundles/material-button-testing.umd.min.js.map +1 -1
- package/bundles/material-button-toggle-testing.umd.js +11 -0
- package/bundles/material-button-toggle-testing.umd.js.map +1 -1
- package/bundles/material-button-toggle-testing.umd.min.js +3 -3
- package/bundles/material-button-toggle-testing.umd.min.js.map +1 -1
- package/bundles/material-button-toggle.umd.js +12 -3
- package/bundles/material-button-toggle.umd.js.map +1 -1
- package/bundles/material-button-toggle.umd.min.js +2 -2
- package/bundles/material-button-toggle.umd.min.js.map +1 -1
- package/bundles/material-button.umd.js +7 -2
- package/bundles/material-button.umd.js.map +1 -1
- package/bundles/material-button.umd.min.js +4 -4
- package/bundles/material-button.umd.min.js.map +1 -1
- package/bundles/material-card-testing.umd.js +335 -0
- package/bundles/material-card-testing.umd.js.map +1 -0
- package/bundles/material-card-testing.umd.min.js +44 -0
- package/bundles/material-card-testing.umd.min.js.map +1 -0
- package/bundles/material-checkbox-testing.umd.js +11 -0
- package/bundles/material-checkbox-testing.umd.js.map +1 -1
- package/bundles/material-checkbox-testing.umd.min.js +3 -3
- package/bundles/material-checkbox-testing.umd.min.js.map +1 -1
- package/bundles/material-checkbox.umd.js.map +1 -1
- package/bundles/material-chips.umd.js +36 -12
- package/bundles/material-chips.umd.js.map +1 -1
- package/bundles/material-chips.umd.min.js +4 -4
- package/bundles/material-chips.umd.min.js.map +1 -1
- package/bundles/material-core-testing.umd.min.js +1 -1
- package/bundles/material-core-testing.umd.min.js.map +1 -1
- package/bundles/material-core.umd.js +11 -3
- package/bundles/material-core.umd.js.map +1 -1
- package/bundles/material-core.umd.min.js +11 -11
- package/bundles/material-core.umd.min.js.map +1 -1
- package/bundles/material-datepicker.umd.js +20 -11
- package/bundles/material-datepicker.umd.js.map +1 -1
- package/bundles/material-datepicker.umd.min.js +5 -5
- package/bundles/material-datepicker.umd.min.js.map +1 -1
- package/bundles/material-dialog-testing.umd.min.js +1 -1
- package/bundles/material-dialog-testing.umd.min.js.map +1 -1
- package/bundles/material-dialog.umd.js +45 -8
- package/bundles/material-dialog.umd.js.map +1 -1
- package/bundles/material-dialog.umd.min.js +14 -7
- package/bundles/material-dialog.umd.min.js.map +1 -1
- package/bundles/material-divider-testing.umd.min.js +1 -1
- package/bundles/material-divider-testing.umd.min.js.map +1 -1
- package/bundles/material-expansion-testing.umd.js +11 -0
- package/bundles/material-expansion-testing.umd.js.map +1 -1
- package/bundles/material-expansion-testing.umd.min.js +3 -3
- package/bundles/material-expansion-testing.umd.min.js.map +1 -1
- package/bundles/material-form-field-testing.umd.js +32 -0
- package/bundles/material-form-field-testing.umd.js.map +1 -1
- package/bundles/material-form-field-testing.umd.min.js +4 -4
- package/bundles/material-form-field-testing.umd.min.js.map +1 -1
- package/bundles/material-form-field.umd.js +47 -21
- package/bundles/material-form-field.umd.js.map +1 -1
- package/bundles/material-form-field.umd.min.js +5 -5
- package/bundles/material-form-field.umd.min.js.map +1 -1
- package/bundles/material-grid-list-testing.umd.min.js +1 -1
- package/bundles/material-grid-list-testing.umd.min.js.map +1 -1
- package/bundles/material-grid-list.umd.js +2 -2
- package/bundles/material-grid-list.umd.js.map +1 -1
- package/bundles/material-grid-list.umd.min.js +2 -2
- package/bundles/material-grid-list.umd.min.js.map +1 -1
- package/bundles/material-input-testing.umd.js +23 -6
- package/bundles/material-input-testing.umd.js.map +1 -1
- package/bundles/material-input-testing.umd.min.js +3 -3
- package/bundles/material-input-testing.umd.min.js.map +1 -1
- package/bundles/material-input.umd.js +28 -3
- package/bundles/material-input.umd.js.map +1 -1
- package/bundles/material-input.umd.min.js +3 -3
- package/bundles/material-input.umd.min.js.map +1 -1
- package/bundles/material-list-testing.umd.js +33 -0
- package/bundles/material-list-testing.umd.js.map +1 -1
- package/bundles/material-list-testing.umd.min.js +2 -2
- package/bundles/material-list-testing.umd.min.js.map +1 -1
- package/bundles/material-list.umd.js +3 -3
- package/bundles/material-list.umd.min.js +1 -1
- package/bundles/material-list.umd.min.js.map +1 -1
- package/bundles/material-menu-testing.umd.js +22 -0
- package/bundles/material-menu-testing.umd.js.map +1 -1
- package/bundles/material-menu-testing.umd.min.js +2 -2
- package/bundles/material-menu-testing.umd.min.js.map +1 -1
- package/bundles/material-menu.umd.js +13 -2
- package/bundles/material-menu.umd.js.map +1 -1
- package/bundles/material-menu.umd.min.js +4 -4
- package/bundles/material-menu.umd.min.js.map +1 -1
- package/bundles/material-paginator-testing.umd.min.js +1 -1
- package/bundles/material-paginator-testing.umd.min.js.map +1 -1
- package/bundles/material-progress-bar-testing.umd.min.js +1 -1
- package/bundles/material-progress-bar-testing.umd.min.js.map +1 -1
- package/bundles/material-progress-spinner-testing.umd.js +1 -1
- package/bundles/material-progress-spinner-testing.umd.js.map +1 -1
- package/bundles/material-progress-spinner-testing.umd.min.js +2 -2
- package/bundles/material-progress-spinner-testing.umd.min.js.map +1 -1
- package/bundles/material-progress-spinner.umd.js +2 -2
- package/bundles/material-progress-spinner.umd.min.js +3 -3
- package/bundles/material-progress-spinner.umd.min.js.map +1 -1
- package/bundles/material-radio-testing.umd.js +11 -0
- package/bundles/material-radio-testing.umd.js.map +1 -1
- package/bundles/material-radio-testing.umd.min.js +3 -3
- package/bundles/material-radio-testing.umd.min.js.map +1 -1
- package/bundles/material-radio.umd.js +15 -5
- package/bundles/material-radio.umd.js.map +1 -1
- package/bundles/material-radio.umd.min.js +2 -2
- package/bundles/material-radio.umd.min.js.map +1 -1
- package/bundles/material-select-testing.umd.js +11 -0
- package/bundles/material-select-testing.umd.js.map +1 -1
- package/bundles/material-select-testing.umd.min.js +2 -2
- package/bundles/material-select-testing.umd.min.js.map +1 -1
- package/bundles/material-select.umd.js +11 -3
- package/bundles/material-select.umd.js.map +1 -1
- package/bundles/material-select.umd.min.js +4 -4
- package/bundles/material-select.umd.min.js.map +1 -1
- package/bundles/material-sidenav-testing.umd.min.js +1 -1
- package/bundles/material-sidenav-testing.umd.min.js.map +1 -1
- package/bundles/material-sidenav.umd.js +41 -17
- package/bundles/material-sidenav.umd.js.map +1 -1
- package/bundles/material-sidenav.umd.min.js +2 -2
- package/bundles/material-sidenav.umd.min.js.map +1 -1
- package/bundles/material-slide-toggle-testing.umd.js +11 -0
- package/bundles/material-slide-toggle-testing.umd.js.map +1 -1
- package/bundles/material-slide-toggle-testing.umd.min.js +2 -2
- package/bundles/material-slide-toggle-testing.umd.min.js.map +1 -1
- package/bundles/material-slide-toggle.umd.js.map +1 -1
- package/bundles/material-slider-testing.umd.js +11 -0
- package/bundles/material-slider-testing.umd.js.map +1 -1
- package/bundles/material-slider-testing.umd.min.js +2 -2
- package/bundles/material-slider-testing.umd.min.js.map +1 -1
- package/bundles/material-snack-bar-testing.umd.min.js +1 -1
- package/bundles/material-snack-bar-testing.umd.min.js.map +1 -1
- package/bundles/material-snack-bar.umd.js +47 -39
- package/bundles/material-snack-bar.umd.js.map +1 -1
- package/bundles/material-snack-bar.umd.min.js +2 -2
- package/bundles/material-snack-bar.umd.min.js.map +1 -1
- package/bundles/material-sort-testing.umd.js +10 -10
- package/bundles/material-sort-testing.umd.js.map +1 -1
- package/bundles/material-sort-testing.umd.min.js +3 -3
- package/bundles/material-sort-testing.umd.min.js.map +1 -1
- package/bundles/material-sort.umd.js +25 -12
- package/bundles/material-sort.umd.js.map +1 -1
- package/bundles/material-sort.umd.min.js +5 -5
- package/bundles/material-sort.umd.min.js.map +1 -1
- package/bundles/material-table-testing.umd.min.js +1 -1
- package/bundles/material-table-testing.umd.min.js.map +1 -1
- package/bundles/material-table.umd.min.js +2 -2
- package/bundles/material-table.umd.min.js.map +1 -1
- package/bundles/material-tabs-testing.umd.min.js +1 -1
- package/bundles/material-tabs-testing.umd.min.js.map +1 -1
- package/bundles/material-tabs.umd.js +23 -4
- package/bundles/material-tabs.umd.js.map +1 -1
- package/bundles/material-tabs.umd.min.js +5 -12
- package/bundles/material-tabs.umd.min.js.map +1 -1
- package/bundles/material-toolbar-testing.umd.js +330 -0
- package/bundles/material-toolbar-testing.umd.js.map +1 -0
- package/bundles/material-toolbar-testing.umd.min.js +44 -0
- package/bundles/material-toolbar-testing.umd.min.js.map +1 -0
- package/bundles/material-tooltip-testing.umd.min.js +1 -1
- package/bundles/material-tooltip-testing.umd.min.js.map +1 -1
- package/bundles/material-tooltip.umd.js +14 -1
- package/bundles/material-tooltip.umd.js.map +1 -1
- package/bundles/material-tooltip.umd.min.js +3 -3
- package/bundles/material-tooltip.umd.min.js.map +1 -1
- package/bundles/material-tree.umd.js.map +1 -1
- package/button/_button-base.scss +1 -1
- package/button/_button-theme.scss +8 -8
- package/button/index.metadata.json +1 -1
- package/button/testing/button-harness.d.ts +2 -0
- package/button-toggle/button-toggle.d.ts +10 -3
- package/button-toggle/index.metadata.json +1 -1
- package/button-toggle/testing/button-toggle-harness.d.ts +2 -0
- package/card/testing/card-harness-filters.d.ts +17 -0
- package/card/testing/card-harness.d.ts +36 -0
- package/card/testing/index.d.ts +8 -0
- package/card/testing/package.json +9 -0
- package/card/testing/public-api.d.ts +9 -0
- package/checkbox/index.metadata.json +1 -1
- package/checkbox/testing/checkbox-harness.d.ts +2 -0
- package/chips/chip.d.ts +19 -1
- package/chips/index.metadata.json +1 -1
- package/core/focus-indicators/_focus-indicators.scss +6 -6
- package/core/index.metadata.json +1 -1
- package/core/option/optgroup.d.ts +7 -0
- package/core/ripple/ripple-ref.d.ts +21 -2
- package/core/ripple/ripple-renderer.d.ts +1 -19
- package/core/ripple/ripple.d.ts +2 -2
- package/core/style/_list-common.scss +1 -1
- package/datepicker/date-range-input-parts.d.ts +2 -4
- package/datepicker/date-range-input.d.ts +4 -4
- package/datepicker/date-range-picker.d.ts +10 -3
- package/datepicker/datepicker.d.ts +2 -3
- package/datepicker/index.metadata.json +1 -1
- package/datepicker/public-api.d.ts +1 -1
- package/dialog/dialog-container.d.ts +9 -2
- package/dialog/dialog-content-directives.d.ts +1 -0
- package/dialog/dialog-ref.d.ts +7 -0
- package/dialog/index.metadata.json +1 -1
- package/esm2015/autocomplete/autocomplete-module.js +18 -22
- package/esm2015/autocomplete/autocomplete-origin.js +16 -20
- package/esm2015/autocomplete/autocomplete-trigger.js +519 -521
- package/esm2015/autocomplete/autocomplete.js +126 -130
- package/esm2015/autocomplete/testing/autocomplete-harness.js +99 -97
- package/esm2015/badge/badge-module.js +13 -17
- package/esm2015/badge/badge.js +184 -188
- package/esm2015/badge/testing/badge-harness.js +74 -78
- package/esm2015/bottom-sheet/bottom-sheet-container.js +161 -165
- package/esm2015/bottom-sheet/bottom-sheet-module.js +15 -19
- package/esm2015/bottom-sheet/bottom-sheet.js +124 -128
- package/esm2015/bottom-sheet/testing/bottom-sheet-harness.js +29 -33
- package/esm2015/button/button-module.js +20 -24
- package/esm2015/button/button.js +111 -114
- package/esm2015/button/testing/button-harness.js +60 -58
- package/esm2015/button-toggle/button-toggle-module.js +10 -14
- package/esm2015/button-toggle/button-toggle.js +368 -368
- package/esm2015/button-toggle/testing/button-toggle-group-harness.js +42 -46
- package/esm2015/button-toggle/testing/button-toggle-harness.js +113 -111
- package/esm2015/card/card-module.js +30 -34
- package/esm2015/card/card.js +147 -203
- package/esm2015/card/testing/card-harness-filters.js +8 -0
- package/esm2015/card/testing/card-harness.js +52 -0
- package/esm2015/card/testing/index.js +9 -0
- package/esm2015/card/testing/public-api.js +10 -0
- package/esm2015/card/testing/testing.externs.js +0 -0
- package/esm2015/checkbox/checkbox-module.js +21 -29
- package/esm2015/checkbox/checkbox-required-validator.js +9 -13
- package/esm2015/checkbox/checkbox.js +315 -319
- package/esm2015/checkbox/testing/checkbox-harness.js +150 -148
- package/esm2015/chips/chip-input.js +119 -123
- package/esm2015/chips/chip-list.js +572 -576
- package/esm2015/chips/chip.js +333 -328
- package/esm2015/chips/chips-module.js +16 -20
- package/esm2015/core/animation/animation.js +12 -20
- package/esm2015/core/common-behaviors/common-module.js +91 -95
- package/esm2015/core/datetime/index.js +19 -27
- package/esm2015/core/datetime/native-date-adapter.js +202 -206
- package/esm2015/core/error/error-options.js +16 -24
- package/esm2015/core/line/line.js +18 -26
- package/esm2015/core/option/index.js +10 -14
- package/esm2015/core/option/optgroup.js +36 -33
- package/esm2015/core/option/option.js +181 -185
- package/esm2015/core/ripple/index.js +10 -14
- package/esm2015/core/ripple/ripple-ref.js +1 -1
- package/esm2015/core/ripple/ripple-renderer.js +1 -1
- package/esm2015/core/ripple/ripple.js +102 -106
- package/esm2015/core/selection/index.js +9 -13
- package/esm2015/core/selection/pseudo-checkbox/pseudo-checkbox.js +32 -36
- package/esm2015/core/testing/optgroup-harness.js +39 -43
- package/esm2015/core/testing/option-harness.js +51 -55
- package/esm2015/core/version.js +1 -1
- package/esm2015/datepicker/calendar-body.js +214 -218
- package/esm2015/datepicker/calendar.js +295 -303
- package/esm2015/datepicker/date-range-input-parts.js +214 -229
- package/esm2015/datepicker/date-range-input.js +244 -242
- package/esm2015/datepicker/date-range-picker.js +19 -23
- package/esm2015/datepicker/date-range-selection-strategy.js +31 -35
- package/esm2015/datepicker/date-selection-model.js +110 -122
- package/esm2015/datepicker/datepicker-base.js +418 -426
- package/esm2015/datepicker/datepicker-input-base.js +227 -224
- package/esm2015/datepicker/datepicker-input.js +111 -115
- package/esm2015/datepicker/datepicker-intl.js +37 -41
- package/esm2015/datepicker/datepicker-module.js +62 -66
- package/esm2015/datepicker/datepicker-toggle.js +81 -89
- package/esm2015/datepicker/datepicker.js +13 -17
- package/esm2015/datepicker/month-view.js +283 -287
- package/esm2015/datepicker/multi-year-view.js +198 -202
- package/esm2015/datepicker/public-api.js +2 -2
- package/esm2015/datepicker/year-view.js +220 -224
- package/esm2015/dialog/dialog-container.js +175 -164
- package/esm2015/dialog/dialog-content-directives.js +105 -114
- package/esm2015/dialog/dialog-module.js +32 -36
- package/esm2015/dialog/dialog-ref.js +17 -3
- package/esm2015/dialog/dialog.js +236 -240
- package/esm2015/dialog/testing/dialog-harness.js +58 -62
- package/esm2015/divider/divider-module.js +10 -14
- package/esm2015/divider/divider.js +33 -37
- package/esm2015/divider/testing/divider-harness.js +16 -20
- package/esm2015/expansion/accordion.js +73 -77
- package/esm2015/expansion/expansion-module.js +26 -30
- package/esm2015/expansion/expansion-panel-content.js +13 -17
- package/esm2015/expansion/expansion-panel-header.js +168 -180
- package/esm2015/expansion/expansion-panel.js +144 -152
- package/esm2015/expansion/testing/accordion-harness.js +24 -28
- package/esm2015/expansion/testing/expansion-harness.js +136 -134
- package/esm2015/form-field/error.js +26 -23
- package/esm2015/form-field/form-field-control.js +6 -10
- package/esm2015/form-field/form-field-module.js +31 -35
- package/esm2015/form-field/form-field.js +407 -409
- package/esm2015/form-field/hint.js +35 -29
- package/esm2015/form-field/label.js +8 -12
- package/esm2015/form-field/placeholder.js +8 -12
- package/esm2015/form-field/prefix.js +16 -13
- package/esm2015/form-field/suffix.js +16 -13
- package/esm2015/form-field/testing/form-field-harness.js +220 -206
- package/esm2015/grid-list/grid-list-module.js +26 -30
- package/esm2015/grid-list/grid-list.js +108 -112
- package/esm2015/grid-list/grid-tile.js +91 -111
- package/esm2015/grid-list/testing/grid-list-harness.js +62 -66
- package/esm2015/grid-list/testing/grid-tile-harness.js +69 -73
- package/esm2015/grid-list/tile-styler.js +1 -1
- package/esm2015/icon/icon-module.js +10 -14
- package/esm2015/icon/icon-registry.js +406 -410
- package/esm2015/icon/icon.js +228 -232
- package/esm2015/icon/testing/fake-icon-registry.js +66 -74
- package/esm2015/input/autosize.js +30 -34
- package/esm2015/input/input-module.js +21 -25
- package/esm2015/input/input.js +305 -284
- package/esm2015/input/testing/input-harness.js +129 -123
- package/esm2015/list/list-module.js +32 -36
- package/esm2015/list/list.js +165 -189
- package/esm2015/list/selection-list.js +503 -511
- package/esm2015/list/testing/action-list-harness.js +55 -57
- package/esm2015/list/testing/list-harness.js +31 -39
- package/esm2015/list/testing/list-item-harness-base.js +13 -17
- package/esm2015/list/testing/nav-list-harness.js +62 -64
- package/esm2015/list/testing/selection-list-harness.js +136 -138
- package/esm2015/menu/menu-content.js +74 -71
- package/esm2015/menu/menu-item.js +119 -123
- package/esm2015/menu/menu-module.js +29 -37
- package/esm2015/menu/menu-panel.js +1 -1
- package/esm2015/menu/menu-trigger.js +402 -405
- package/esm2015/menu/menu.js +329 -339
- package/esm2015/menu/testing/menu-harness.js +193 -189
- package/esm2015/paginator/paginator-intl.js +36 -40
- package/esm2015/paginator/paginator-module.js +16 -20
- package/esm2015/paginator/paginator.js +205 -209
- package/esm2015/paginator/testing/paginator-harness.js +91 -95
- package/esm2015/progress-bar/progress-bar-module.js +10 -14
- package/esm2015/progress-bar/progress-bar.js +114 -118
- package/esm2015/progress-bar/testing/progress-bar-harness.js +27 -31
- package/esm2015/progress-spinner/progress-spinner-module.js +17 -21
- package/esm2015/progress-spinner/progress-spinner.js +181 -189
- package/esm2015/progress-spinner/testing/progress-spinner-harness.js +28 -32
- package/esm2015/radio/radio-module.js +10 -14
- package/esm2015/radio/radio.js +437 -444
- package/esm2015/radio/testing/radio-harness.js +241 -243
- package/esm2015/select/select-module.js +23 -27
- package/esm2015/select/select.js +917 -918
- package/esm2015/select/testing/select-harness.js +138 -136
- package/esm2015/sidenav/drawer.js +632 -620
- package/esm2015/sidenav/sidenav-module.js +31 -35
- package/esm2015/sidenav/sidenav.js +104 -116
- package/esm2015/sidenav/testing/drawer-harness.js +40 -44
- package/esm2015/sidenav/testing/sidenav-harness.js +20 -24
- package/esm2015/slide-toggle/slide-toggle-module.js +27 -35
- package/esm2015/slide-toggle/slide-toggle-required-validator.js +9 -13
- package/esm2015/slide-toggle/slide-toggle.js +184 -188
- package/esm2015/slide-toggle/testing/slide-toggle-harness.js +123 -121
- package/esm2015/slider/slider-module.js +10 -14
- package/esm2015/slider/slider.js +640 -644
- package/esm2015/slider/testing/slider-harness.js +129 -127
- package/esm2015/snack-bar/simple-snack-bar.js +32 -36
- package/esm2015/snack-bar/snack-bar-container.js +144 -148
- package/esm2015/snack-bar/snack-bar-module.js +17 -21
- package/esm2015/snack-bar/snack-bar-ref.js +1 -1
- package/esm2015/snack-bar/snack-bar.js +208 -205
- package/esm2015/snack-bar/testing/snack-bar-harness.js +112 -116
- package/esm2015/sort/sort-header-intl.js +21 -21
- package/esm2015/sort/sort-header.js +200 -194
- package/esm2015/sort/sort-module.js +11 -15
- package/esm2015/sort/sort.js +92 -96
- package/esm2015/sort/testing/sort-harness.js +28 -32
- package/esm2015/sort/testing/sort-header-harness.js +66 -67
- package/esm2015/stepper/step-header.js +78 -82
- package/esm2015/stepper/step-label.js +8 -12
- package/esm2015/stepper/stepper-button.js +23 -31
- package/esm2015/stepper/stepper-icon.js +16 -20
- package/esm2015/stepper/stepper-intl.js +15 -19
- package/esm2015/stepper/stepper-module.js +40 -44
- package/esm2015/stepper/stepper.js +139 -155
- package/esm2015/table/cell.js +94 -122
- package/esm2015/table/row.js +90 -118
- package/esm2015/table/table-module.js +13 -17
- package/esm2015/table/table.js +26 -30
- package/esm2015/table/testing/cell-harness.js +56 -68
- package/esm2015/table/testing/row-harness.js +90 -102
- package/esm2015/table/testing/table-harness.js +65 -69
- package/esm2015/table/text-column.js +17 -21
- package/esm2015/tabs/index.js +4 -2
- package/esm2015/tabs/ink-bar.js +55 -59
- package/esm2015/tabs/paginated-tab-header.js +415 -419
- package/esm2015/tabs/tab-body.js +179 -191
- package/esm2015/tabs/tab-content.js +21 -16
- package/esm2015/tabs/tab-group.js +263 -271
- package/esm2015/tabs/tab-header.js +69 -77
- package/esm2015/tabs/tab-label-wrapper.js +29 -33
- package/esm2015/tabs/tab-label.js +16 -13
- package/esm2015/tabs/tab-nav-bar/tab-nav-bar.js +199 -215
- package/esm2015/tabs/tab.js +80 -83
- package/esm2015/tabs/tabs-module.js +38 -42
- package/esm2015/tabs/testing/tab-group-harness.js +52 -56
- package/esm2015/tabs/testing/tab-harness.js +78 -82
- package/esm2015/toolbar/testing/index.js +9 -0
- package/esm2015/toolbar/testing/public-api.js +10 -0
- package/esm2015/toolbar/testing/testing.externs.js +0 -0
- package/esm2015/toolbar/testing/toolbar-harness-filters.js +8 -0
- package/esm2015/toolbar/testing/toolbar-harness.js +47 -0
- package/esm2015/toolbar/toolbar-module.js +10 -14
- package/esm2015/toolbar/toolbar.js +61 -69
- package/esm2015/tooltip/testing/tooltip-harness.js +43 -47
- package/esm2015/tooltip/tooltip-module.js +17 -21
- package/esm2015/tooltip/tooltip.js +502 -497
- package/esm2015/tree/data-source/flat-data-source.js +1 -1
- package/esm2015/tree/node.js +99 -111
- package/esm2015/tree/outlet.js +19 -23
- package/esm2015/tree/padding.js +13 -17
- package/esm2015/tree/toggle.js +15 -19
- package/esm2015/tree/tree-module.js +10 -14
- package/esm2015/tree/tree.js +23 -27
- package/expansion/testing/expansion-harness.d.ts +2 -0
- package/fesm2015/autocomplete/testing.js +98 -95
- package/fesm2015/autocomplete/testing.js.map +1 -1
- package/fesm2015/autocomplete.js +677 -687
- package/fesm2015/autocomplete.js.map +1 -1
- package/fesm2015/badge/testing.js +73 -76
- package/fesm2015/badge/testing.js.map +1 -1
- package/fesm2015/badge.js +195 -201
- package/fesm2015/badge.js.map +1 -1
- package/fesm2015/bottom-sheet/testing.js +28 -31
- package/fesm2015/bottom-sheet/testing.js.map +1 -1
- package/fesm2015/bottom-sheet.js +297 -306
- package/fesm2015/bottom-sheet.js.map +1 -1
- package/fesm2015/button/testing.js +59 -56
- package/fesm2015/button/testing.js.map +1 -1
- package/fesm2015/button-toggle/testing.js +155 -155
- package/fesm2015/button-toggle/testing.js.map +1 -1
- package/fesm2015/button-toggle.js +377 -378
- package/fesm2015/button-toggle.js.map +1 -1
- package/fesm2015/button.js +129 -133
- package/fesm2015/button.js.map +1 -1
- package/fesm2015/card/testing.js +79 -0
- package/fesm2015/card/testing.js.map +1 -0
- package/fesm2015/card.js +175 -220
- package/fesm2015/card.js.map +1 -1
- package/fesm2015/checkbox/testing.js +149 -146
- package/fesm2015/checkbox/testing.js.map +1 -1
- package/fesm2015/checkbox.js +342 -354
- package/fesm2015/checkbox.js.map +1 -1
- package/fesm2015/chips.js +1039 -1039
- package/fesm2015/chips.js.map +1 -1
- package/fesm2015/core/testing.js +88 -94
- package/fesm2015/core/testing.js.map +1 -1
- package/fesm2015/core.js +729 -773
- package/fesm2015/core.js.map +1 -1
- package/fesm2015/datepicker.js +2821 -2884
- package/fesm2015/datepicker.js.map +1 -1
- package/fesm2015/dialog/testing.js +57 -60
- package/fesm2015/dialog/testing.js.map +1 -1
- package/fesm2015/dialog.js +564 -549
- package/fesm2015/dialog.js.map +1 -1
- package/fesm2015/divider/testing.js +15 -18
- package/fesm2015/divider/testing.js.map +1 -1
- package/fesm2015/divider.js +41 -47
- package/fesm2015/divider.js.map +1 -1
- package/fesm2015/expansion/testing.js +159 -159
- package/fesm2015/expansion/testing.js.map +1 -1
- package/fesm2015/expansion.js +422 -446
- package/fesm2015/expansion.js.map +1 -1
- package/fesm2015/form-field/testing.js +219 -204
- package/fesm2015/form-field/testing.js.map +1 -1
- package/fesm2015/form-field.js +538 -532
- package/fesm2015/form-field.js.map +1 -1
- package/fesm2015/grid-list/testing.js +129 -135
- package/fesm2015/grid-list/testing.js.map +1 -1
- package/fesm2015/grid-list.js +221 -242
- package/fesm2015/grid-list.js.map +1 -1
- package/fesm2015/icon/testing.js +65 -71
- package/fesm2015/icon/testing.js.map +1 -1
- package/fesm2015/icon.js +660 -669
- package/fesm2015/icon.js.map +1 -1
- package/fesm2015/input/testing.js +128 -121
- package/fesm2015/input/testing.js.map +1 -1
- package/fesm2015/input.js +353 -337
- package/fesm2015/input.js.map +1 -1
- package/fesm2015/list/testing.js +298 -307
- package/fesm2015/list/testing.js.map +1 -1
- package/fesm2015/list.js +698 -725
- package/fesm2015/list.js.map +1 -1
- package/fesm2015/menu/testing.js +192 -186
- package/fesm2015/menu/testing.js.map +1 -1
- package/fesm2015/menu.js +948 -962
- package/fesm2015/menu.js.map +1 -1
- package/fesm2015/paginator/testing.js +90 -93
- package/fesm2015/paginator/testing.js.map +1 -1
- package/fesm2015/paginator.js +247 -256
- package/fesm2015/paginator.js.map +1 -1
- package/fesm2015/progress-bar/testing.js +26 -29
- package/fesm2015/progress-bar/testing.js.map +1 -1
- package/fesm2015/progress-bar.js +122 -128
- package/fesm2015/progress-bar.js.map +1 -1
- package/fesm2015/progress-spinner/testing.js +27 -30
- package/fesm2015/progress-spinner/testing.js.map +1 -1
- package/fesm2015/progress-spinner.js +196 -205
- package/fesm2015/progress-spinner.js.map +1 -1
- package/fesm2015/radio/testing.js +240 -240
- package/fesm2015/radio/testing.js.map +1 -1
- package/fesm2015/radio.js +447 -453
- package/fesm2015/radio.js.map +1 -1
- package/fesm2015/select/testing.js +137 -134
- package/fesm2015/select/testing.js.map +1 -1
- package/fesm2015/select.js +939 -941
- package/fesm2015/select.js.map +1 -1
- package/fesm2015/sidenav/testing.js +58 -64
- package/fesm2015/sidenav/testing.js.map +1 -1
- package/fesm2015/sidenav.js +765 -762
- package/fesm2015/sidenav.js.map +1 -1
- package/fesm2015/slide-toggle/testing.js +122 -119
- package/fesm2015/slide-toggle/testing.js.map +1 -1
- package/fesm2015/slide-toggle.js +217 -229
- package/fesm2015/slide-toggle.js.map +1 -1
- package/fesm2015/slider/testing.js +128 -125
- package/fesm2015/slider/testing.js.map +1 -1
- package/fesm2015/slider.js +648 -654
- package/fesm2015/slider.js.map +1 -1
- package/fesm2015/snack-bar/testing.js +111 -114
- package/fesm2015/snack-bar/testing.js.map +1 -1
- package/fesm2015/snack-bar.js +426 -431
- package/fesm2015/snack-bar.js.map +1 -1
- package/fesm2015/sort/testing.js +92 -95
- package/fesm2015/sort/testing.js.map +1 -1
- package/fesm2015/sort.js +320 -318
- package/fesm2015/sort.js.map +1 -1
- package/fesm2015/stepper.js +312 -345
- package/fesm2015/stepper.js.map +1 -1
- package/fesm2015/table/testing.js +208 -229
- package/fesm2015/table/testing.js.map +1 -1
- package/fesm2015/table.js +235 -286
- package/fesm2015/table.js.map +1 -1
- package/fesm2015/tabs/testing.js +128 -134
- package/fesm2015/tabs/testing.js.map +1 -1
- package/fesm2015/tabs.js +1362 -1399
- package/fesm2015/tabs.js.map +1 -1
- package/fesm2015/toolbar/testing.js +74 -0
- package/fesm2015/toolbar/testing.js.map +1 -0
- package/fesm2015/toolbar.js +69 -78
- package/fesm2015/toolbar.js.map +1 -1
- package/fesm2015/tooltip/testing.js +42 -45
- package/fesm2015/tooltip/testing.js.map +1 -1
- package/fesm2015/tooltip.js +517 -513
- package/fesm2015/tooltip.js.map +1 -1
- package/fesm2015/tree.js +173 -197
- package/fesm2015/tree.js.map +1 -1
- package/form-field/error.d.ts +7 -0
- package/form-field/hint.d.ts +10 -0
- package/form-field/index.metadata.json +1 -1
- package/form-field/prefix.d.ts +7 -0
- package/form-field/suffix.d.ts +7 -0
- package/form-field/testing/form-field-harness.d.ts +8 -0
- package/grid-list/grid-list.d.ts +2 -1
- package/grid-list/index.metadata.json +1 -1
- package/grid-list/tile-styler.d.ts +10 -5
- package/input/_input-theme.scss +2 -2
- package/input/index.metadata.json +1 -1
- package/input/input.d.ts +6 -2
- package/input/testing/input-harness.d.ts +2 -0
- package/list/index.metadata.json +1 -1
- package/list/testing/action-list-harness.d.ts +2 -0
- package/list/testing/nav-list-harness.d.ts +2 -0
- package/list/testing/selection-list-harness.d.ts +2 -0
- package/menu/index.metadata.json +1 -1
- package/menu/menu-content.d.ts +7 -1
- package/menu/menu-panel.d.ts +1 -0
- package/menu/menu.d.ts +4 -0
- package/menu/testing/menu-harness.d.ts +4 -0
- package/package.json +6 -6
- package/prebuilt-themes/deeppurple-amber.css +1 -1
- package/prebuilt-themes/indigo-pink.css +1 -1
- package/prebuilt-themes/pink-bluegrey.css +1 -1
- package/prebuilt-themes/purple-green.css +1 -1
- package/progress-spinner/index.metadata.json +1 -1
- package/radio/index.metadata.json +1 -1
- package/radio/radio.d.ts +6 -0
- package/radio/testing/radio-harness.d.ts +2 -0
- package/schematics/migration.json +5 -0
- package/schematics/ng-add/index.js +2 -2
- package/schematics/ng-generate/navigation/files/__path__/__name@dasherize@if-flat__/__name@dasherize__.component.html.template +3 -3
- package/schematics/ng-generate/navigation/schema.json +5 -0
- package/schematics/ng-update/data/index.js +1 -1
- package/schematics/ng-update/index.d.ts +2 -0
- package/schematics/ng-update/index.js +7 -2
- package/schematics/ng-update/migrations/hammer-gestures-v9/hammer-gestures-migration.js +639 -639
- package/schematics/ng-update/migrations/misc-ripples-v7/ripple-speed-factor-migration.js +1 -1
- package/select/index.metadata.json +1 -1
- package/select/select.d.ts +6 -0
- package/select/testing/select-harness.d.ts +2 -0
- package/sidenav/drawer.d.ts +15 -3
- package/sidenav/index.metadata.json +1 -1
- package/slide-toggle/index.metadata.json +1 -1
- package/slide-toggle/testing/slide-toggle-harness.d.ts +2 -0
- package/slider/testing/slider-harness.d.ts +2 -0
- package/snack-bar/index.metadata.json +1 -1
- package/snack-bar/simple-snack-bar.d.ts +13 -1
- package/snack-bar/snack-bar-container.d.ts +16 -3
- package/snack-bar/snack-bar-ref.d.ts +3 -3
- package/snack-bar/snack-bar.d.ts +10 -3
- package/sort/index.metadata.json +1 -1
- package/sort/sort-header-intl.d.ts +5 -1
- package/sort/sort-header.d.ts +3 -1
- package/sort/testing/sort-header-harness.d.ts +6 -2
- package/tabs/index.d.ts +3 -1
- package/tabs/index.metadata.json +1 -1
- package/tabs/tab-content.d.ts +7 -1
- package/tabs/tab-label.d.ts +7 -0
- package/toolbar/testing/index.d.ts +8 -0
- package/toolbar/testing/package.json +9 -0
- package/toolbar/testing/public-api.d.ts +9 -0
- package/toolbar/testing/toolbar-harness-filters.d.ts +13 -0
- package/toolbar/testing/toolbar-harness.d.ts +31 -0
- package/tooltip/index.metadata.json +1 -1
- package/tooltip/tooltip.d.ts +1 -0
- package/tree/data-source/flat-data-source.d.ts +4 -4
- package/tree/index.metadata.json +1 -1
package/esm2015/select/select.js
CHANGED
|
@@ -14,7 +14,7 @@ import { CdkConnectedOverlay, Overlay, } from '@angular/cdk/overlay';
|
|
|
14
14
|
import { ViewportRuler } from '@angular/cdk/scrolling';
|
|
15
15
|
import { Attribute, ChangeDetectionStrategy, ChangeDetectorRef, Component, ContentChild, ContentChildren, Directive, ElementRef, EventEmitter, Inject, InjectionToken, Input, isDevMode, NgZone, Optional, Output, QueryList, Self, ViewChild, ViewEncapsulation, } from '@angular/core';
|
|
16
16
|
import { FormGroupDirective, NgControl, NgForm } from '@angular/forms';
|
|
17
|
-
import { _countGroupLabelsBeforeOption, _getOptionScrollPosition, ErrorStateMatcher,
|
|
17
|
+
import { _countGroupLabelsBeforeOption, _getOptionScrollPosition, ErrorStateMatcher, MAT_OPTGROUP, MAT_OPTION_PARENT_COMPONENT, MatOption, mixinDisabled, mixinDisableRipple, mixinErrorState, mixinTabIndex, } from '@angular/material/core';
|
|
18
18
|
import { MAT_FORM_FIELD, MatFormField, MatFormFieldControl } from '@angular/material/form-field';
|
|
19
19
|
import { defer, merge, Observable, Subject } from 'rxjs';
|
|
20
20
|
import { distinctUntilChanged, filter, map, startWith, switchMap, take, takeUntil, } from 'rxjs/operators';
|
|
@@ -87,1005 +87,1004 @@ class MatSelectBase {
|
|
|
87
87
|
}
|
|
88
88
|
}
|
|
89
89
|
const _MatSelectMixinBase = mixinDisableRipple(mixinTabIndex(mixinDisabled(mixinErrorState(MatSelectBase))));
|
|
90
|
+
/**
|
|
91
|
+
* Injection token that can be used to reference instances of `MatSelectTrigger`. It serves as
|
|
92
|
+
* alternative token to the actual `MatSelectTrigger` class which could cause unnecessary
|
|
93
|
+
* retention of the class and its directive metadata.
|
|
94
|
+
*/
|
|
95
|
+
export const MAT_SELECT_TRIGGER = new InjectionToken('MatSelectTrigger');
|
|
90
96
|
/**
|
|
91
97
|
* Allows the user to customize the trigger that is displayed when the select has a value.
|
|
92
98
|
*/
|
|
93
|
-
|
|
94
|
-
|
|
95
|
-
|
|
96
|
-
|
|
97
|
-
|
|
98
|
-
|
|
99
|
-
|
|
100
|
-
|
|
101
|
-
|
|
102
|
-
|
|
103
|
-
|
|
104
|
-
|
|
105
|
-
|
|
106
|
-
|
|
107
|
-
|
|
108
|
-
|
|
109
|
-
|
|
110
|
-
|
|
111
|
-
|
|
112
|
-
|
|
113
|
-
|
|
114
|
-
|
|
115
|
-
|
|
116
|
-
|
|
117
|
-
|
|
118
|
-
|
|
119
|
-
|
|
120
|
-
|
|
121
|
-
|
|
122
|
-
|
|
123
|
-
|
|
124
|
-
|
|
125
|
-
|
|
126
|
-
|
|
127
|
-
|
|
128
|
-
|
|
129
|
-
|
|
130
|
-
|
|
131
|
-
|
|
132
|
-
|
|
133
|
-
|
|
134
|
-
|
|
135
|
-
|
|
136
|
-
|
|
137
|
-
/** The value of the select panel's transform-origin property. */
|
|
138
|
-
this._transformOrigin = 'top';
|
|
139
|
-
/** Emits when the panel element is finished transforming in. */
|
|
140
|
-
this._panelDoneAnimatingStream = new Subject();
|
|
141
|
-
/**
|
|
142
|
-
* The y-offset of the overlay panel in relation to the trigger's top start corner.
|
|
143
|
-
* This must be adjusted to align the selected option text over the trigger text.
|
|
144
|
-
* when the panel opens. Will change based on the y-position of the selected option.
|
|
145
|
-
*/
|
|
146
|
-
this._offsetY = 0;
|
|
147
|
-
/**
|
|
148
|
-
* This position config ensures that the top "start" corner of the overlay
|
|
149
|
-
* is aligned with with the top "start" of the origin by default (overlapping
|
|
150
|
-
* the trigger completely). If the panel cannot fit below the trigger, it
|
|
151
|
-
* will fall back to a position above the trigger.
|
|
152
|
-
*/
|
|
153
|
-
this._positions = [
|
|
154
|
-
{
|
|
155
|
-
originX: 'start',
|
|
156
|
-
originY: 'top',
|
|
157
|
-
overlayX: 'start',
|
|
158
|
-
overlayY: 'top',
|
|
159
|
-
},
|
|
160
|
-
{
|
|
161
|
-
originX: 'start',
|
|
162
|
-
originY: 'bottom',
|
|
163
|
-
overlayX: 'start',
|
|
164
|
-
overlayY: 'bottom',
|
|
165
|
-
},
|
|
166
|
-
];
|
|
167
|
-
/** Whether the component is disabling centering of the active option over the trigger. */
|
|
168
|
-
this._disableOptionCentering = false;
|
|
169
|
-
this._focused = false;
|
|
170
|
-
/** A name for this control that can be used by `mat-form-field`. */
|
|
171
|
-
this.controlType = 'mat-select';
|
|
172
|
-
/** Aria label of the select. If not specified, the placeholder will be used as label. */
|
|
173
|
-
this.ariaLabel = '';
|
|
174
|
-
/** Combined stream of all of the child options' change events. */
|
|
175
|
-
this.optionSelectionChanges = defer(() => {
|
|
176
|
-
const options = this.options;
|
|
177
|
-
if (options) {
|
|
178
|
-
return options.changes.pipe(startWith(options), switchMap(() => merge(...options.map(option => option.onSelectionChange))));
|
|
179
|
-
}
|
|
180
|
-
return this._ngZone.onStable
|
|
181
|
-
.asObservable()
|
|
182
|
-
.pipe(take(1), switchMap(() => this.optionSelectionChanges));
|
|
183
|
-
});
|
|
184
|
-
/** Event emitted when the select panel has been toggled. */
|
|
185
|
-
this.openedChange = new EventEmitter();
|
|
186
|
-
/** Event emitted when the select has been opened. */
|
|
187
|
-
this._openedStream = this.openedChange.pipe(filter(o => o), map(() => { }));
|
|
188
|
-
/** Event emitted when the select has been closed. */
|
|
189
|
-
this._closedStream = this.openedChange.pipe(filter(o => !o), map(() => { }));
|
|
190
|
-
/** Event emitted when the selected value has been changed by the user. */
|
|
191
|
-
this.selectionChange = new EventEmitter();
|
|
192
|
-
/**
|
|
193
|
-
* Event that emits whenever the raw value of the select changes. This is here primarily
|
|
194
|
-
* to facilitate the two-way binding for the `value` input.
|
|
195
|
-
* @docs-private
|
|
196
|
-
*/
|
|
197
|
-
this.valueChange = new EventEmitter();
|
|
198
|
-
if (this.ngControl) {
|
|
199
|
-
// Note: we provide the value accessor through here, instead of
|
|
200
|
-
// the `providers` to avoid running into a circular import.
|
|
201
|
-
this.ngControl.valueAccessor = this;
|
|
202
|
-
}
|
|
203
|
-
this._scrollStrategyFactory = scrollStrategyFactory;
|
|
204
|
-
this._scrollStrategy = this._scrollStrategyFactory();
|
|
205
|
-
this.tabIndex = parseInt(tabIndex) || 0;
|
|
206
|
-
// Force setter to be called in case id was not specified.
|
|
207
|
-
this.id = this.id;
|
|
208
|
-
if (defaults) {
|
|
209
|
-
if (defaults.disableOptionCentering != null) {
|
|
210
|
-
this.disableOptionCentering = defaults.disableOptionCentering;
|
|
211
|
-
}
|
|
212
|
-
if (defaults.typeaheadDebounceInterval != null) {
|
|
213
|
-
this.typeaheadDebounceInterval = defaults.typeaheadDebounceInterval;
|
|
214
|
-
}
|
|
215
|
-
}
|
|
216
|
-
}
|
|
217
|
-
/** Whether the select is focused. */
|
|
218
|
-
get focused() {
|
|
219
|
-
return this._focused || this._panelOpen;
|
|
220
|
-
}
|
|
221
|
-
/** Placeholder to be shown if no value has been selected. */
|
|
222
|
-
get placeholder() { return this._placeholder; }
|
|
223
|
-
set placeholder(value) {
|
|
224
|
-
this._placeholder = value;
|
|
225
|
-
this.stateChanges.next();
|
|
226
|
-
}
|
|
227
|
-
/** Whether the component is required. */
|
|
228
|
-
get required() { return this._required; }
|
|
229
|
-
set required(value) {
|
|
230
|
-
this._required = coerceBooleanProperty(value);
|
|
231
|
-
this.stateChanges.next();
|
|
232
|
-
}
|
|
233
|
-
/** Whether the user should be allowed to select multiple options. */
|
|
234
|
-
get multiple() { return this._multiple; }
|
|
235
|
-
set multiple(value) {
|
|
236
|
-
if (this._selectionModel) {
|
|
237
|
-
throw getMatSelectDynamicMultipleError();
|
|
238
|
-
}
|
|
239
|
-
this._multiple = coerceBooleanProperty(value);
|
|
240
|
-
}
|
|
241
|
-
/** Whether to center the active option over the trigger. */
|
|
242
|
-
get disableOptionCentering() { return this._disableOptionCentering; }
|
|
243
|
-
set disableOptionCentering(value) {
|
|
244
|
-
this._disableOptionCentering = coerceBooleanProperty(value);
|
|
245
|
-
}
|
|
99
|
+
export class MatSelectTrigger {
|
|
100
|
+
}
|
|
101
|
+
MatSelectTrigger.decorators = [
|
|
102
|
+
{ type: Directive, args: [{
|
|
103
|
+
selector: 'mat-select-trigger',
|
|
104
|
+
providers: [{ provide: MAT_SELECT_TRIGGER, useExisting: MatSelectTrigger }],
|
|
105
|
+
},] }
|
|
106
|
+
];
|
|
107
|
+
export class MatSelect extends _MatSelectMixinBase {
|
|
108
|
+
constructor(_viewportRuler, _changeDetectorRef, _ngZone, _defaultErrorStateMatcher, elementRef, _dir, _parentForm, _parentFormGroup, _parentFormField, ngControl, tabIndex, scrollStrategyFactory, _liveAnnouncer, defaults) {
|
|
109
|
+
super(elementRef, _defaultErrorStateMatcher, _parentForm, _parentFormGroup, ngControl);
|
|
110
|
+
this._viewportRuler = _viewportRuler;
|
|
111
|
+
this._changeDetectorRef = _changeDetectorRef;
|
|
112
|
+
this._ngZone = _ngZone;
|
|
113
|
+
this._dir = _dir;
|
|
114
|
+
this._parentFormField = _parentFormField;
|
|
115
|
+
this.ngControl = ngControl;
|
|
116
|
+
this._liveAnnouncer = _liveAnnouncer;
|
|
117
|
+
/** Whether or not the overlay panel is open. */
|
|
118
|
+
this._panelOpen = false;
|
|
119
|
+
/** Whether filling out the select is required in the form. */
|
|
120
|
+
this._required = false;
|
|
121
|
+
/** The scroll position of the overlay panel, calculated to center the selected option. */
|
|
122
|
+
this._scrollTop = 0;
|
|
123
|
+
/** Whether the component is in multiple selection mode. */
|
|
124
|
+
this._multiple = false;
|
|
125
|
+
/** Comparison function to specify which option is displayed. Defaults to object equality. */
|
|
126
|
+
this._compareWith = (o1, o2) => o1 === o2;
|
|
127
|
+
/** Unique id for this input. */
|
|
128
|
+
this._uid = `mat-select-${nextUniqueId++}`;
|
|
129
|
+
/** Emits whenever the component is destroyed. */
|
|
130
|
+
this._destroy = new Subject();
|
|
131
|
+
/** The cached font-size of the trigger element. */
|
|
132
|
+
this._triggerFontSize = 0;
|
|
133
|
+
/** `View -> model callback called when value changes` */
|
|
134
|
+
this._onChange = () => { };
|
|
135
|
+
/** `View -> model callback called when select has been touched` */
|
|
136
|
+
this._onTouched = () => { };
|
|
137
|
+
/** The IDs of child options to be passed to the aria-owns attribute. */
|
|
138
|
+
this._optionIds = '';
|
|
139
|
+
/** The value of the select panel's transform-origin property. */
|
|
140
|
+
this._transformOrigin = 'top';
|
|
141
|
+
/** Emits when the panel element is finished transforming in. */
|
|
142
|
+
this._panelDoneAnimatingStream = new Subject();
|
|
246
143
|
/**
|
|
247
|
-
*
|
|
248
|
-
*
|
|
249
|
-
*
|
|
144
|
+
* The y-offset of the overlay panel in relation to the trigger's top start corner.
|
|
145
|
+
* This must be adjusted to align the selected option text over the trigger text.
|
|
146
|
+
* when the panel opens. Will change based on the y-position of the selected option.
|
|
250
147
|
*/
|
|
251
|
-
|
|
252
|
-
|
|
253
|
-
|
|
254
|
-
|
|
255
|
-
|
|
256
|
-
|
|
257
|
-
|
|
258
|
-
|
|
259
|
-
|
|
148
|
+
this._offsetY = 0;
|
|
149
|
+
/**
|
|
150
|
+
* This position config ensures that the top "start" corner of the overlay
|
|
151
|
+
* is aligned with with the top "start" of the origin by default (overlapping
|
|
152
|
+
* the trigger completely). If the panel cannot fit below the trigger, it
|
|
153
|
+
* will fall back to a position above the trigger.
|
|
154
|
+
*/
|
|
155
|
+
this._positions = [
|
|
156
|
+
{
|
|
157
|
+
originX: 'start',
|
|
158
|
+
originY: 'top',
|
|
159
|
+
overlayX: 'start',
|
|
160
|
+
overlayY: 'top',
|
|
161
|
+
},
|
|
162
|
+
{
|
|
163
|
+
originX: 'start',
|
|
164
|
+
originY: 'bottom',
|
|
165
|
+
overlayX: 'start',
|
|
166
|
+
overlayY: 'bottom',
|
|
167
|
+
},
|
|
168
|
+
];
|
|
169
|
+
/** Whether the component is disabling centering of the active option over the trigger. */
|
|
170
|
+
this._disableOptionCentering = false;
|
|
171
|
+
this._focused = false;
|
|
172
|
+
/** A name for this control that can be used by `mat-form-field`. */
|
|
173
|
+
this.controlType = 'mat-select';
|
|
174
|
+
/** Aria label of the select. If not specified, the placeholder will be used as label. */
|
|
175
|
+
this.ariaLabel = '';
|
|
176
|
+
/** Combined stream of all of the child options' change events. */
|
|
177
|
+
this.optionSelectionChanges = defer(() => {
|
|
178
|
+
const options = this.options;
|
|
179
|
+
if (options) {
|
|
180
|
+
return options.changes.pipe(startWith(options), switchMap(() => merge(...options.map(option => option.onSelectionChange))));
|
|
260
181
|
}
|
|
182
|
+
return this._ngZone.onStable
|
|
183
|
+
.asObservable()
|
|
184
|
+
.pipe(take(1), switchMap(() => this.optionSelectionChanges));
|
|
185
|
+
});
|
|
186
|
+
/** Event emitted when the select panel has been toggled. */
|
|
187
|
+
this.openedChange = new EventEmitter();
|
|
188
|
+
/** Event emitted when the select has been opened. */
|
|
189
|
+
this._openedStream = this.openedChange.pipe(filter(o => o), map(() => { }));
|
|
190
|
+
/** Event emitted when the select has been closed. */
|
|
191
|
+
this._closedStream = this.openedChange.pipe(filter(o => !o), map(() => { }));
|
|
192
|
+
/** Event emitted when the selected value has been changed by the user. */
|
|
193
|
+
this.selectionChange = new EventEmitter();
|
|
194
|
+
/**
|
|
195
|
+
* Event that emits whenever the raw value of the select changes. This is here primarily
|
|
196
|
+
* to facilitate the two-way binding for the `value` input.
|
|
197
|
+
* @docs-private
|
|
198
|
+
*/
|
|
199
|
+
this.valueChange = new EventEmitter();
|
|
200
|
+
if (this.ngControl) {
|
|
201
|
+
// Note: we provide the value accessor through here, instead of
|
|
202
|
+
// the `providers` to avoid running into a circular import.
|
|
203
|
+
this.ngControl.valueAccessor = this;
|
|
261
204
|
}
|
|
262
|
-
|
|
263
|
-
|
|
264
|
-
|
|
265
|
-
|
|
266
|
-
|
|
267
|
-
|
|
205
|
+
this._scrollStrategyFactory = scrollStrategyFactory;
|
|
206
|
+
this._scrollStrategy = this._scrollStrategyFactory();
|
|
207
|
+
this.tabIndex = parseInt(tabIndex) || 0;
|
|
208
|
+
// Force setter to be called in case id was not specified.
|
|
209
|
+
this.id = this.id;
|
|
210
|
+
if (defaults) {
|
|
211
|
+
if (defaults.disableOptionCentering != null) {
|
|
212
|
+
this.disableOptionCentering = defaults.disableOptionCentering;
|
|
213
|
+
}
|
|
214
|
+
if (defaults.typeaheadDebounceInterval != null) {
|
|
215
|
+
this.typeaheadDebounceInterval = defaults.typeaheadDebounceInterval;
|
|
268
216
|
}
|
|
269
217
|
}
|
|
270
|
-
|
|
271
|
-
|
|
272
|
-
|
|
273
|
-
|
|
274
|
-
|
|
275
|
-
|
|
276
|
-
|
|
277
|
-
|
|
278
|
-
|
|
279
|
-
|
|
218
|
+
}
|
|
219
|
+
/** Whether the select is focused. */
|
|
220
|
+
get focused() {
|
|
221
|
+
return this._focused || this._panelOpen;
|
|
222
|
+
}
|
|
223
|
+
/** Placeholder to be shown if no value has been selected. */
|
|
224
|
+
get placeholder() { return this._placeholder; }
|
|
225
|
+
set placeholder(value) {
|
|
226
|
+
this._placeholder = value;
|
|
227
|
+
this.stateChanges.next();
|
|
228
|
+
}
|
|
229
|
+
/** Whether the component is required. */
|
|
230
|
+
get required() { return this._required; }
|
|
231
|
+
set required(value) {
|
|
232
|
+
this._required = coerceBooleanProperty(value);
|
|
233
|
+
this.stateChanges.next();
|
|
234
|
+
}
|
|
235
|
+
/** Whether the user should be allowed to select multiple options. */
|
|
236
|
+
get multiple() { return this._multiple; }
|
|
237
|
+
set multiple(value) {
|
|
238
|
+
if (this._selectionModel) {
|
|
239
|
+
throw getMatSelectDynamicMultipleError();
|
|
280
240
|
}
|
|
281
|
-
|
|
282
|
-
|
|
283
|
-
|
|
284
|
-
|
|
285
|
-
|
|
286
|
-
|
|
287
|
-
|
|
288
|
-
|
|
289
|
-
|
|
290
|
-
|
|
291
|
-
|
|
292
|
-
|
|
293
|
-
|
|
294
|
-
|
|
295
|
-
|
|
296
|
-
|
|
297
|
-
this._changeDetectorRef.markForCheck();
|
|
298
|
-
}
|
|
299
|
-
});
|
|
300
|
-
this._viewportRuler.change()
|
|
301
|
-
.pipe(takeUntil(this._destroy))
|
|
302
|
-
.subscribe(() => {
|
|
303
|
-
if (this._panelOpen) {
|
|
304
|
-
this._triggerRect = this.trigger.nativeElement.getBoundingClientRect();
|
|
305
|
-
this._changeDetectorRef.markForCheck();
|
|
306
|
-
}
|
|
307
|
-
});
|
|
241
|
+
this._multiple = coerceBooleanProperty(value);
|
|
242
|
+
}
|
|
243
|
+
/** Whether to center the active option over the trigger. */
|
|
244
|
+
get disableOptionCentering() { return this._disableOptionCentering; }
|
|
245
|
+
set disableOptionCentering(value) {
|
|
246
|
+
this._disableOptionCentering = coerceBooleanProperty(value);
|
|
247
|
+
}
|
|
248
|
+
/**
|
|
249
|
+
* Function to compare the option values with the selected values. The first argument
|
|
250
|
+
* is a value from an option. The second is a value from the selection. A boolean
|
|
251
|
+
* should be returned.
|
|
252
|
+
*/
|
|
253
|
+
get compareWith() { return this._compareWith; }
|
|
254
|
+
set compareWith(fn) {
|
|
255
|
+
if (typeof fn !== 'function') {
|
|
256
|
+
throw getMatSelectNonFunctionValueError();
|
|
308
257
|
}
|
|
309
|
-
|
|
310
|
-
|
|
311
|
-
|
|
312
|
-
|
|
313
|
-
event.removed.forEach(option => option.deselect());
|
|
314
|
-
});
|
|
315
|
-
this.options.changes.pipe(startWith(null), takeUntil(this._destroy)).subscribe(() => {
|
|
316
|
-
this._resetOptions();
|
|
317
|
-
this._initializeSelection();
|
|
318
|
-
});
|
|
258
|
+
this._compareWith = fn;
|
|
259
|
+
if (this._selectionModel) {
|
|
260
|
+
// A different comparator means the selection could change.
|
|
261
|
+
this._initializeSelection();
|
|
319
262
|
}
|
|
320
|
-
|
|
321
|
-
|
|
322
|
-
|
|
323
|
-
|
|
263
|
+
}
|
|
264
|
+
/** Value of the select control. */
|
|
265
|
+
get value() { return this._value; }
|
|
266
|
+
set value(newValue) {
|
|
267
|
+
if (newValue !== this._value) {
|
|
268
|
+
this.writeValue(newValue);
|
|
269
|
+
this._value = newValue;
|
|
324
270
|
}
|
|
325
|
-
|
|
326
|
-
|
|
327
|
-
|
|
328
|
-
|
|
329
|
-
|
|
330
|
-
|
|
331
|
-
|
|
332
|
-
|
|
271
|
+
}
|
|
272
|
+
/** Time to wait in milliseconds after the last keystroke before moving focus to an item. */
|
|
273
|
+
get typeaheadDebounceInterval() { return this._typeaheadDebounceInterval; }
|
|
274
|
+
set typeaheadDebounceInterval(value) {
|
|
275
|
+
this._typeaheadDebounceInterval = coerceNumberProperty(value);
|
|
276
|
+
}
|
|
277
|
+
/** Unique id of the element. */
|
|
278
|
+
get id() { return this._id; }
|
|
279
|
+
set id(value) {
|
|
280
|
+
this._id = value || this._uid;
|
|
281
|
+
this.stateChanges.next();
|
|
282
|
+
}
|
|
283
|
+
ngOnInit() {
|
|
284
|
+
this._selectionModel = new SelectionModel(this.multiple);
|
|
285
|
+
this.stateChanges.next();
|
|
286
|
+
// We need `distinctUntilChanged` here, because some browsers will
|
|
287
|
+
// fire the animation end event twice for the same animation. See:
|
|
288
|
+
// https://github.com/angular/angular/issues/24084
|
|
289
|
+
this._panelDoneAnimatingStream
|
|
290
|
+
.pipe(distinctUntilChanged(), takeUntil(this._destroy))
|
|
291
|
+
.subscribe(() => {
|
|
292
|
+
if (this.panelOpen) {
|
|
293
|
+
this._scrollTop = 0;
|
|
294
|
+
this.openedChange.emit(true);
|
|
333
295
|
}
|
|
334
|
-
|
|
335
|
-
|
|
336
|
-
|
|
337
|
-
|
|
338
|
-
this.stateChanges.complete();
|
|
339
|
-
}
|
|
340
|
-
/** Toggles the overlay panel open or closed. */
|
|
341
|
-
toggle() {
|
|
342
|
-
this.panelOpen ? this.close() : this.open();
|
|
343
|
-
}
|
|
344
|
-
/** Opens the overlay panel. */
|
|
345
|
-
open() {
|
|
346
|
-
if (this.disabled || !this.options || !this.options.length || this._panelOpen) {
|
|
347
|
-
return;
|
|
296
|
+
else {
|
|
297
|
+
this.openedChange.emit(false);
|
|
298
|
+
this.overlayDir.offsetX = 0;
|
|
299
|
+
this._changeDetectorRef.markForCheck();
|
|
348
300
|
}
|
|
349
|
-
|
|
350
|
-
|
|
351
|
-
|
|
352
|
-
|
|
353
|
-
this._panelOpen = true;
|
|
354
|
-
this._keyManager.withHorizontalOrientation(null);
|
|
355
|
-
this._calculateOverlayPosition();
|
|
356
|
-
this._highlightCorrectOption();
|
|
357
|
-
this._changeDetectorRef.markForCheck();
|
|
358
|
-
// Set the font size on the panel element once it exists.
|
|
359
|
-
this._ngZone.onStable.asObservable().pipe(take(1)).subscribe(() => {
|
|
360
|
-
if (this._triggerFontSize && this.overlayDir.overlayRef &&
|
|
361
|
-
this.overlayDir.overlayRef.overlayElement) {
|
|
362
|
-
this.overlayDir.overlayRef.overlayElement.style.fontSize = `${this._triggerFontSize}px`;
|
|
363
|
-
}
|
|
364
|
-
});
|
|
365
|
-
}
|
|
366
|
-
/** Closes the overlay panel and focuses the host element. */
|
|
367
|
-
close() {
|
|
301
|
+
});
|
|
302
|
+
this._viewportRuler.change()
|
|
303
|
+
.pipe(takeUntil(this._destroy))
|
|
304
|
+
.subscribe(() => {
|
|
368
305
|
if (this._panelOpen) {
|
|
369
|
-
this.
|
|
370
|
-
this._keyManager.withHorizontalOrientation(this._isRtl() ? 'rtl' : 'ltr');
|
|
306
|
+
this._triggerRect = this.trigger.nativeElement.getBoundingClientRect();
|
|
371
307
|
this._changeDetectorRef.markForCheck();
|
|
372
|
-
this._onTouched();
|
|
373
308
|
}
|
|
309
|
+
});
|
|
310
|
+
}
|
|
311
|
+
ngAfterContentInit() {
|
|
312
|
+
this._initKeyManager();
|
|
313
|
+
this._selectionModel.changed.pipe(takeUntil(this._destroy)).subscribe(event => {
|
|
314
|
+
event.added.forEach(option => option.select());
|
|
315
|
+
event.removed.forEach(option => option.deselect());
|
|
316
|
+
});
|
|
317
|
+
this.options.changes.pipe(startWith(null), takeUntil(this._destroy)).subscribe(() => {
|
|
318
|
+
this._resetOptions();
|
|
319
|
+
this._initializeSelection();
|
|
320
|
+
});
|
|
321
|
+
}
|
|
322
|
+
ngDoCheck() {
|
|
323
|
+
if (this.ngControl) {
|
|
324
|
+
this.updateErrorState();
|
|
374
325
|
}
|
|
375
|
-
|
|
376
|
-
|
|
377
|
-
|
|
378
|
-
|
|
379
|
-
|
|
380
|
-
|
|
381
|
-
writeValue(value) {
|
|
382
|
-
if (this.options) {
|
|
383
|
-
this._setSelectionByValue(value);
|
|
384
|
-
}
|
|
326
|
+
}
|
|
327
|
+
ngOnChanges(changes) {
|
|
328
|
+
// Updating the disabled state is handled by `mixinDisabled`, but we need to additionally let
|
|
329
|
+
// the parent form field know to run change detection when the disabled state changes.
|
|
330
|
+
if (changes['disabled']) {
|
|
331
|
+
this.stateChanges.next();
|
|
385
332
|
}
|
|
386
|
-
|
|
387
|
-
|
|
388
|
-
* changes from user input. Part of the ControlValueAccessor interface
|
|
389
|
-
* required to integrate with Angular's core forms API.
|
|
390
|
-
*
|
|
391
|
-
* @param fn Callback to be triggered when the value changes.
|
|
392
|
-
*/
|
|
393
|
-
registerOnChange(fn) {
|
|
394
|
-
this._onChange = fn;
|
|
333
|
+
if (changes['typeaheadDebounceInterval'] && this._keyManager) {
|
|
334
|
+
this._keyManager.withTypeAhead(this._typeaheadDebounceInterval);
|
|
395
335
|
}
|
|
396
|
-
|
|
397
|
-
|
|
398
|
-
|
|
399
|
-
|
|
400
|
-
|
|
401
|
-
|
|
402
|
-
|
|
403
|
-
|
|
404
|
-
|
|
336
|
+
}
|
|
337
|
+
ngOnDestroy() {
|
|
338
|
+
this._destroy.next();
|
|
339
|
+
this._destroy.complete();
|
|
340
|
+
this.stateChanges.complete();
|
|
341
|
+
}
|
|
342
|
+
/** Toggles the overlay panel open or closed. */
|
|
343
|
+
toggle() {
|
|
344
|
+
this.panelOpen ? this.close() : this.open();
|
|
345
|
+
}
|
|
346
|
+
/** Opens the overlay panel. */
|
|
347
|
+
open() {
|
|
348
|
+
if (this.disabled || !this.options || !this.options.length || this._panelOpen) {
|
|
349
|
+
return;
|
|
405
350
|
}
|
|
406
|
-
|
|
407
|
-
|
|
408
|
-
|
|
409
|
-
|
|
410
|
-
|
|
411
|
-
|
|
412
|
-
|
|
413
|
-
|
|
351
|
+
this._triggerRect = this.trigger.nativeElement.getBoundingClientRect();
|
|
352
|
+
// Note: The computed font-size will be a string pixel value (e.g. "16px").
|
|
353
|
+
// `parseInt` ignores the trailing 'px' and converts this to a number.
|
|
354
|
+
this._triggerFontSize = parseInt(getComputedStyle(this.trigger.nativeElement).fontSize || '0');
|
|
355
|
+
this._panelOpen = true;
|
|
356
|
+
this._keyManager.withHorizontalOrientation(null);
|
|
357
|
+
this._calculateOverlayPosition();
|
|
358
|
+
this._highlightCorrectOption();
|
|
359
|
+
this._changeDetectorRef.markForCheck();
|
|
360
|
+
// Set the font size on the panel element once it exists.
|
|
361
|
+
this._ngZone.onStable.asObservable().pipe(take(1)).subscribe(() => {
|
|
362
|
+
if (this._triggerFontSize && this.overlayDir.overlayRef &&
|
|
363
|
+
this.overlayDir.overlayRef.overlayElement) {
|
|
364
|
+
this.overlayDir.overlayRef.overlayElement.style.fontSize = `${this._triggerFontSize}px`;
|
|
365
|
+
}
|
|
366
|
+
});
|
|
367
|
+
}
|
|
368
|
+
/** Closes the overlay panel and focuses the host element. */
|
|
369
|
+
close() {
|
|
370
|
+
if (this._panelOpen) {
|
|
371
|
+
this._panelOpen = false;
|
|
372
|
+
this._keyManager.withHorizontalOrientation(this._isRtl() ? 'rtl' : 'ltr');
|
|
414
373
|
this._changeDetectorRef.markForCheck();
|
|
415
|
-
this.
|
|
374
|
+
this._onTouched();
|
|
416
375
|
}
|
|
417
|
-
|
|
418
|
-
|
|
419
|
-
|
|
376
|
+
}
|
|
377
|
+
/**
|
|
378
|
+
* Sets the select's value. Part of the ControlValueAccessor interface
|
|
379
|
+
* required to integrate with Angular's core forms API.
|
|
380
|
+
*
|
|
381
|
+
* @param value New value to be written to the model.
|
|
382
|
+
*/
|
|
383
|
+
writeValue(value) {
|
|
384
|
+
if (this.options) {
|
|
385
|
+
this._setSelectionByValue(value);
|
|
420
386
|
}
|
|
421
|
-
|
|
422
|
-
|
|
423
|
-
|
|
387
|
+
}
|
|
388
|
+
/**
|
|
389
|
+
* Saves a callback function to be invoked when the select's value
|
|
390
|
+
* changes from user input. Part of the ControlValueAccessor interface
|
|
391
|
+
* required to integrate with Angular's core forms API.
|
|
392
|
+
*
|
|
393
|
+
* @param fn Callback to be triggered when the value changes.
|
|
394
|
+
*/
|
|
395
|
+
registerOnChange(fn) {
|
|
396
|
+
this._onChange = fn;
|
|
397
|
+
}
|
|
398
|
+
/**
|
|
399
|
+
* Saves a callback function to be invoked when the select is blurred
|
|
400
|
+
* by the user. Part of the ControlValueAccessor interface required
|
|
401
|
+
* to integrate with Angular's core forms API.
|
|
402
|
+
*
|
|
403
|
+
* @param fn Callback to be triggered when the component has been touched.
|
|
404
|
+
*/
|
|
405
|
+
registerOnTouched(fn) {
|
|
406
|
+
this._onTouched = fn;
|
|
407
|
+
}
|
|
408
|
+
/**
|
|
409
|
+
* Disables the select. Part of the ControlValueAccessor interface required
|
|
410
|
+
* to integrate with Angular's core forms API.
|
|
411
|
+
*
|
|
412
|
+
* @param isDisabled Sets whether the component is disabled.
|
|
413
|
+
*/
|
|
414
|
+
setDisabledState(isDisabled) {
|
|
415
|
+
this.disabled = isDisabled;
|
|
416
|
+
this._changeDetectorRef.markForCheck();
|
|
417
|
+
this.stateChanges.next();
|
|
418
|
+
}
|
|
419
|
+
/** Whether or not the overlay panel is open. */
|
|
420
|
+
get panelOpen() {
|
|
421
|
+
return this._panelOpen;
|
|
422
|
+
}
|
|
423
|
+
/** The currently selected option. */
|
|
424
|
+
get selected() {
|
|
425
|
+
return this.multiple ? this._selectionModel.selected : this._selectionModel.selected[0];
|
|
426
|
+
}
|
|
427
|
+
/** The value displayed in the trigger. */
|
|
428
|
+
get triggerValue() {
|
|
429
|
+
if (this.empty) {
|
|
430
|
+
return '';
|
|
424
431
|
}
|
|
425
|
-
|
|
426
|
-
|
|
427
|
-
if (this.
|
|
428
|
-
|
|
432
|
+
if (this._multiple) {
|
|
433
|
+
const selectedOptions = this._selectionModel.selected.map(option => option.viewValue);
|
|
434
|
+
if (this._isRtl()) {
|
|
435
|
+
selectedOptions.reverse();
|
|
429
436
|
}
|
|
430
|
-
|
|
431
|
-
|
|
432
|
-
if (this._isRtl()) {
|
|
433
|
-
selectedOptions.reverse();
|
|
434
|
-
}
|
|
435
|
-
// TODO(crisbeto): delimiter should be configurable for proper localization.
|
|
436
|
-
return selectedOptions.join(', ');
|
|
437
|
-
}
|
|
438
|
-
return this._selectionModel.selected[0].viewValue;
|
|
437
|
+
// TODO(crisbeto): delimiter should be configurable for proper localization.
|
|
438
|
+
return selectedOptions.join(', ');
|
|
439
439
|
}
|
|
440
|
-
|
|
441
|
-
|
|
442
|
-
|
|
443
|
-
|
|
444
|
-
|
|
445
|
-
|
|
446
|
-
|
|
447
|
-
|
|
448
|
-
|
|
440
|
+
return this._selectionModel.selected[0].viewValue;
|
|
441
|
+
}
|
|
442
|
+
/** Whether the element is in RTL mode. */
|
|
443
|
+
_isRtl() {
|
|
444
|
+
return this._dir ? this._dir.value === 'rtl' : false;
|
|
445
|
+
}
|
|
446
|
+
/** Handles all keydown events on the select. */
|
|
447
|
+
_handleKeydown(event) {
|
|
448
|
+
if (!this.disabled) {
|
|
449
|
+
this.panelOpen ? this._handleOpenKeydown(event) : this._handleClosedKeydown(event);
|
|
449
450
|
}
|
|
450
|
-
|
|
451
|
-
|
|
452
|
-
|
|
453
|
-
|
|
454
|
-
|
|
455
|
-
|
|
456
|
-
|
|
457
|
-
|
|
458
|
-
|
|
459
|
-
|
|
460
|
-
|
|
461
|
-
|
|
462
|
-
|
|
463
|
-
else if (!this.multiple) {
|
|
464
|
-
const previouslySelectedOption = this.selected;
|
|
465
|
-
if (keyCode === HOME || keyCode === END) {
|
|
466
|
-
keyCode === HOME ? manager.setFirstItemActive() : manager.setLastItemActive();
|
|
467
|
-
event.preventDefault();
|
|
468
|
-
}
|
|
469
|
-
else {
|
|
470
|
-
manager.onKeydown(event);
|
|
471
|
-
}
|
|
472
|
-
const selectedOption = this.selected;
|
|
473
|
-
// Since the value has changed, we need to announce it ourselves.
|
|
474
|
-
if (selectedOption && previouslySelectedOption !== selectedOption) {
|
|
475
|
-
// We set a duration on the live announcement, because we want the live element to be
|
|
476
|
-
// cleared after a while so that users can't navigate to it using the arrow keys.
|
|
477
|
-
this._liveAnnouncer.announce(selectedOption.viewValue, 10000);
|
|
478
|
-
}
|
|
479
|
-
}
|
|
451
|
+
}
|
|
452
|
+
/** Handles keyboard events while the select is closed. */
|
|
453
|
+
_handleClosedKeydown(event) {
|
|
454
|
+
const keyCode = event.keyCode;
|
|
455
|
+
const isArrowKey = keyCode === DOWN_ARROW || keyCode === UP_ARROW ||
|
|
456
|
+
keyCode === LEFT_ARROW || keyCode === RIGHT_ARROW;
|
|
457
|
+
const isOpenKey = keyCode === ENTER || keyCode === SPACE;
|
|
458
|
+
const manager = this._keyManager;
|
|
459
|
+
// Open the select on ALT + arrow key to match the native <select>
|
|
460
|
+
if (!manager.isTyping() && (isOpenKey && !hasModifierKey(event)) ||
|
|
461
|
+
((this.multiple || event.altKey) && isArrowKey)) {
|
|
462
|
+
event.preventDefault(); // prevents the page from scrolling down when pressing space
|
|
463
|
+
this.open();
|
|
480
464
|
}
|
|
481
|
-
|
|
482
|
-
|
|
483
|
-
const manager = this._keyManager;
|
|
484
|
-
const keyCode = event.keyCode;
|
|
485
|
-
const isArrowKey = keyCode === DOWN_ARROW || keyCode === UP_ARROW;
|
|
486
|
-
const isTyping = manager.isTyping();
|
|
465
|
+
else if (!this.multiple) {
|
|
466
|
+
const previouslySelectedOption = this.selected;
|
|
487
467
|
if (keyCode === HOME || keyCode === END) {
|
|
488
|
-
event.preventDefault();
|
|
489
468
|
keyCode === HOME ? manager.setFirstItemActive() : manager.setLastItemActive();
|
|
490
|
-
}
|
|
491
|
-
else if (isArrowKey && event.altKey) {
|
|
492
|
-
// Close the select on ALT + arrow key to match the native <select>
|
|
493
|
-
event.preventDefault();
|
|
494
|
-
this.close();
|
|
495
|
-
// Don't do anything in this case if the user is typing,
|
|
496
|
-
// because the typing sequence can include the space key.
|
|
497
|
-
}
|
|
498
|
-
else if (!isTyping && (keyCode === ENTER || keyCode === SPACE) && manager.activeItem &&
|
|
499
|
-
!hasModifierKey(event)) {
|
|
500
|
-
event.preventDefault();
|
|
501
|
-
manager.activeItem._selectViaInteraction();
|
|
502
|
-
}
|
|
503
|
-
else if (!isTyping && this._multiple && keyCode === A && event.ctrlKey) {
|
|
504
469
|
event.preventDefault();
|
|
505
|
-
const hasDeselectedOptions = this.options.some(opt => !opt.disabled && !opt.selected);
|
|
506
|
-
this.options.forEach(option => {
|
|
507
|
-
if (!option.disabled) {
|
|
508
|
-
hasDeselectedOptions ? option.select() : option.deselect();
|
|
509
|
-
}
|
|
510
|
-
});
|
|
511
470
|
}
|
|
512
471
|
else {
|
|
513
|
-
const previouslyFocusedIndex = manager.activeItemIndex;
|
|
514
472
|
manager.onKeydown(event);
|
|
515
|
-
if (this._multiple && isArrowKey && event.shiftKey && manager.activeItem &&
|
|
516
|
-
manager.activeItemIndex !== previouslyFocusedIndex) {
|
|
517
|
-
manager.activeItem._selectViaInteraction();
|
|
518
|
-
}
|
|
519
|
-
}
|
|
520
|
-
}
|
|
521
|
-
_onFocus() {
|
|
522
|
-
if (!this.disabled) {
|
|
523
|
-
this._focused = true;
|
|
524
|
-
this.stateChanges.next();
|
|
525
473
|
}
|
|
526
|
-
|
|
527
|
-
|
|
528
|
-
|
|
529
|
-
|
|
530
|
-
|
|
531
|
-
|
|
532
|
-
this._focused = false;
|
|
533
|
-
if (!this.disabled && !this.panelOpen) {
|
|
534
|
-
this._onTouched();
|
|
535
|
-
this._changeDetectorRef.markForCheck();
|
|
536
|
-
this.stateChanges.next();
|
|
474
|
+
const selectedOption = this.selected;
|
|
475
|
+
// Since the value has changed, we need to announce it ourselves.
|
|
476
|
+
if (selectedOption && previouslySelectedOption !== selectedOption) {
|
|
477
|
+
// We set a duration on the live announcement, because we want the live element to be
|
|
478
|
+
// cleared after a while so that users can't navigate to it using the arrow keys.
|
|
479
|
+
this._liveAnnouncer.announce(selectedOption.viewValue, 10000);
|
|
537
480
|
}
|
|
538
481
|
}
|
|
539
|
-
|
|
540
|
-
|
|
541
|
-
|
|
542
|
-
|
|
543
|
-
|
|
544
|
-
|
|
545
|
-
|
|
546
|
-
|
|
547
|
-
|
|
482
|
+
}
|
|
483
|
+
/** Handles keyboard events when the selected is open. */
|
|
484
|
+
_handleOpenKeydown(event) {
|
|
485
|
+
const manager = this._keyManager;
|
|
486
|
+
const keyCode = event.keyCode;
|
|
487
|
+
const isArrowKey = keyCode === DOWN_ARROW || keyCode === UP_ARROW;
|
|
488
|
+
const isTyping = manager.isTyping();
|
|
489
|
+
if (keyCode === HOME || keyCode === END) {
|
|
490
|
+
event.preventDefault();
|
|
491
|
+
keyCode === HOME ? manager.setFirstItemActive() : manager.setLastItemActive();
|
|
548
492
|
}
|
|
549
|
-
|
|
550
|
-
|
|
551
|
-
|
|
493
|
+
else if (isArrowKey && event.altKey) {
|
|
494
|
+
// Close the select on ALT + arrow key to match the native <select>
|
|
495
|
+
event.preventDefault();
|
|
496
|
+
this.close();
|
|
497
|
+
// Don't do anything in this case if the user is typing,
|
|
498
|
+
// because the typing sequence can include the space key.
|
|
552
499
|
}
|
|
553
|
-
|
|
554
|
-
|
|
555
|
-
|
|
500
|
+
else if (!isTyping && (keyCode === ENTER || keyCode === SPACE) && manager.activeItem &&
|
|
501
|
+
!hasModifierKey(event)) {
|
|
502
|
+
event.preventDefault();
|
|
503
|
+
manager.activeItem._selectViaInteraction();
|
|
556
504
|
}
|
|
557
|
-
|
|
558
|
-
|
|
559
|
-
|
|
560
|
-
|
|
561
|
-
|
|
562
|
-
|
|
505
|
+
else if (!isTyping && this._multiple && keyCode === A && event.ctrlKey) {
|
|
506
|
+
event.preventDefault();
|
|
507
|
+
const hasDeselectedOptions = this.options.some(opt => !opt.disabled && !opt.selected);
|
|
508
|
+
this.options.forEach(option => {
|
|
509
|
+
if (!option.disabled) {
|
|
510
|
+
hasDeselectedOptions ? option.select() : option.deselect();
|
|
511
|
+
}
|
|
563
512
|
});
|
|
564
513
|
}
|
|
565
|
-
|
|
566
|
-
|
|
567
|
-
|
|
568
|
-
|
|
569
|
-
|
|
570
|
-
|
|
571
|
-
if (!Array.isArray(value)) {
|
|
572
|
-
throw getMatSelectNonArrayValueError();
|
|
573
|
-
}
|
|
574
|
-
this._selectionModel.clear();
|
|
575
|
-
value.forEach((currentValue) => this._selectValue(currentValue));
|
|
576
|
-
this._sortValues();
|
|
577
|
-
}
|
|
578
|
-
else {
|
|
579
|
-
this._selectionModel.clear();
|
|
580
|
-
const correspondingOption = this._selectValue(value);
|
|
581
|
-
// Shift focus to the active item. Note that we shouldn't do this in multiple
|
|
582
|
-
// mode, because we don't know what option the user interacted with last.
|
|
583
|
-
if (correspondingOption) {
|
|
584
|
-
this._keyManager.setActiveItem(correspondingOption);
|
|
585
|
-
}
|
|
586
|
-
else if (!this.panelOpen) {
|
|
587
|
-
// Otherwise reset the highlighted option. Note that we only want to do this while
|
|
588
|
-
// closed, because doing it while open can shift the user's focus unnecessarily.
|
|
589
|
-
this._keyManager.setActiveItem(-1);
|
|
590
|
-
}
|
|
514
|
+
else {
|
|
515
|
+
const previouslyFocusedIndex = manager.activeItemIndex;
|
|
516
|
+
manager.onKeydown(event);
|
|
517
|
+
if (this._multiple && isArrowKey && event.shiftKey && manager.activeItem &&
|
|
518
|
+
manager.activeItemIndex !== previouslyFocusedIndex) {
|
|
519
|
+
manager.activeItem._selectViaInteraction();
|
|
591
520
|
}
|
|
521
|
+
}
|
|
522
|
+
}
|
|
523
|
+
_onFocus() {
|
|
524
|
+
if (!this.disabled) {
|
|
525
|
+
this._focused = true;
|
|
526
|
+
this.stateChanges.next();
|
|
527
|
+
}
|
|
528
|
+
}
|
|
529
|
+
/**
|
|
530
|
+
* Calls the touched callback only if the panel is closed. Otherwise, the trigger will
|
|
531
|
+
* "blur" to the panel when it opens, causing a false positive.
|
|
532
|
+
*/
|
|
533
|
+
_onBlur() {
|
|
534
|
+
this._focused = false;
|
|
535
|
+
if (!this.disabled && !this.panelOpen) {
|
|
536
|
+
this._onTouched();
|
|
592
537
|
this._changeDetectorRef.markForCheck();
|
|
538
|
+
this.stateChanges.next();
|
|
593
539
|
}
|
|
594
|
-
|
|
595
|
-
|
|
596
|
-
|
|
597
|
-
|
|
598
|
-
|
|
599
|
-
|
|
600
|
-
|
|
601
|
-
|
|
602
|
-
|
|
603
|
-
|
|
604
|
-
|
|
605
|
-
|
|
606
|
-
|
|
607
|
-
|
|
608
|
-
|
|
609
|
-
|
|
610
|
-
|
|
611
|
-
|
|
540
|
+
}
|
|
541
|
+
/**
|
|
542
|
+
* Callback that is invoked when the overlay panel has been attached.
|
|
543
|
+
*/
|
|
544
|
+
_onAttached() {
|
|
545
|
+
this.overlayDir.positionChange.pipe(take(1)).subscribe(() => {
|
|
546
|
+
this._changeDetectorRef.detectChanges();
|
|
547
|
+
this._calculateOverlayOffsetX();
|
|
548
|
+
this.panel.nativeElement.scrollTop = this._scrollTop;
|
|
549
|
+
});
|
|
550
|
+
}
|
|
551
|
+
/** Returns the theme to be used on the panel. */
|
|
552
|
+
_getPanelTheme() {
|
|
553
|
+
return this._parentFormField ? `mat-${this._parentFormField.color}` : '';
|
|
554
|
+
}
|
|
555
|
+
/** Whether the select has a value. */
|
|
556
|
+
get empty() {
|
|
557
|
+
return !this._selectionModel || this._selectionModel.isEmpty();
|
|
558
|
+
}
|
|
559
|
+
_initializeSelection() {
|
|
560
|
+
// Defer setting the value in order to avoid the "Expression
|
|
561
|
+
// has changed after it was checked" errors from Angular.
|
|
562
|
+
Promise.resolve().then(() => {
|
|
563
|
+
this._setSelectionByValue(this.ngControl ? this.ngControl.value : this._value);
|
|
564
|
+
this.stateChanges.next();
|
|
565
|
+
});
|
|
566
|
+
}
|
|
567
|
+
/**
|
|
568
|
+
* Sets the selected option based on a value. If no option can be
|
|
569
|
+
* found with the designated value, the select trigger is cleared.
|
|
570
|
+
*/
|
|
571
|
+
_setSelectionByValue(value) {
|
|
572
|
+
if (this.multiple && value) {
|
|
573
|
+
if (!Array.isArray(value)) {
|
|
574
|
+
throw getMatSelectNonArrayValueError();
|
|
575
|
+
}
|
|
576
|
+
this._selectionModel.clear();
|
|
577
|
+
value.forEach((currentValue) => this._selectValue(currentValue));
|
|
578
|
+
this._sortValues();
|
|
579
|
+
}
|
|
580
|
+
else {
|
|
581
|
+
this._selectionModel.clear();
|
|
582
|
+
const correspondingOption = this._selectValue(value);
|
|
583
|
+
// Shift focus to the active item. Note that we shouldn't do this in multiple
|
|
584
|
+
// mode, because we don't know what option the user interacted with last.
|
|
612
585
|
if (correspondingOption) {
|
|
613
|
-
this.
|
|
586
|
+
this._keyManager.setActiveItem(correspondingOption);
|
|
587
|
+
}
|
|
588
|
+
else if (!this.panelOpen) {
|
|
589
|
+
// Otherwise reset the highlighted option. Note that we only want to do this while
|
|
590
|
+
// closed, because doing it while open can shift the user's focus unnecessarily.
|
|
591
|
+
this._keyManager.setActiveItem(-1);
|
|
614
592
|
}
|
|
615
|
-
return correspondingOption;
|
|
616
593
|
}
|
|
617
|
-
|
|
618
|
-
|
|
619
|
-
|
|
620
|
-
|
|
621
|
-
|
|
622
|
-
|
|
623
|
-
|
|
624
|
-
|
|
625
|
-
|
|
626
|
-
|
|
627
|
-
|
|
628
|
-
|
|
629
|
-
|
|
630
|
-
|
|
631
|
-
//
|
|
632
|
-
|
|
633
|
-
this.focus();
|
|
634
|
-
this.close();
|
|
635
|
-
}
|
|
636
|
-
});
|
|
637
|
-
this._keyManager.change.pipe(takeUntil(this._destroy)).subscribe(() => {
|
|
638
|
-
if (this._panelOpen && this.panel) {
|
|
639
|
-
this._scrollActiveOptionIntoView();
|
|
640
|
-
}
|
|
641
|
-
else if (!this._panelOpen && !this.multiple && this._keyManager.activeItem) {
|
|
642
|
-
this._keyManager.activeItem._selectViaInteraction();
|
|
594
|
+
this._changeDetectorRef.markForCheck();
|
|
595
|
+
}
|
|
596
|
+
/**
|
|
597
|
+
* Finds and selects and option based on its value.
|
|
598
|
+
* @returns Option that has the corresponding value.
|
|
599
|
+
*/
|
|
600
|
+
_selectValue(value) {
|
|
601
|
+
const correspondingOption = this.options.find((option) => {
|
|
602
|
+
try {
|
|
603
|
+
// Treat null as a special reset value.
|
|
604
|
+
return option.value != null && this._compareWith(option.value, value);
|
|
605
|
+
}
|
|
606
|
+
catch (error) {
|
|
607
|
+
if (isDevMode()) {
|
|
608
|
+
// Notify developers of errors in their comparator.
|
|
609
|
+
console.warn(error);
|
|
643
610
|
}
|
|
644
|
-
|
|
611
|
+
return false;
|
|
612
|
+
}
|
|
613
|
+
});
|
|
614
|
+
if (correspondingOption) {
|
|
615
|
+
this._selectionModel.select(correspondingOption);
|
|
645
616
|
}
|
|
646
|
-
|
|
647
|
-
|
|
648
|
-
|
|
649
|
-
|
|
650
|
-
|
|
651
|
-
|
|
652
|
-
|
|
653
|
-
|
|
617
|
+
return correspondingOption;
|
|
618
|
+
}
|
|
619
|
+
/** Sets up a key manager to listen to keyboard events on the overlay panel. */
|
|
620
|
+
_initKeyManager() {
|
|
621
|
+
this._keyManager = new ActiveDescendantKeyManager(this.options)
|
|
622
|
+
.withTypeAhead(this._typeaheadDebounceInterval)
|
|
623
|
+
.withVerticalOrientation()
|
|
624
|
+
.withHorizontalOrientation(this._isRtl() ? 'rtl' : 'ltr')
|
|
625
|
+
.withAllowedModifierKeys(['shiftKey']);
|
|
626
|
+
this._keyManager.tabOut.pipe(takeUntil(this._destroy)).subscribe(() => {
|
|
627
|
+
if (this.panelOpen) {
|
|
628
|
+
// Select the active item when tabbing away. This is consistent with how the native
|
|
629
|
+
// select behaves. Note that we only want to do this in single selection mode.
|
|
630
|
+
if (!this.multiple && this._keyManager.activeItem) {
|
|
631
|
+
this._keyManager.activeItem._selectViaInteraction();
|
|
654
632
|
}
|
|
655
|
-
|
|
656
|
-
|
|
657
|
-
|
|
658
|
-
|
|
659
|
-
.pipe(takeUntil(changedOrDestroyed))
|
|
660
|
-
.subscribe(() => {
|
|
661
|
-
this._changeDetectorRef.markForCheck();
|
|
662
|
-
this.stateChanges.next();
|
|
663
|
-
});
|
|
664
|
-
this._setOptionIds();
|
|
665
|
-
}
|
|
666
|
-
/** Invoked when an option is clicked. */
|
|
667
|
-
_onSelect(option, isUserInput) {
|
|
668
|
-
const wasSelected = this._selectionModel.isSelected(option);
|
|
669
|
-
if (option.value == null && !this._multiple) {
|
|
670
|
-
option.deselect();
|
|
671
|
-
this._selectionModel.clear();
|
|
672
|
-
this._propagateChanges(option.value);
|
|
633
|
+
// Restore focus to the trigger before closing. Ensures that the focus
|
|
634
|
+
// position won't be lost if the user got focus into the overlay.
|
|
635
|
+
this.focus();
|
|
636
|
+
this.close();
|
|
673
637
|
}
|
|
674
|
-
|
|
675
|
-
|
|
676
|
-
|
|
677
|
-
|
|
678
|
-
}
|
|
679
|
-
if (isUserInput) {
|
|
680
|
-
this._keyManager.setActiveItem(option);
|
|
681
|
-
}
|
|
682
|
-
if (this.multiple) {
|
|
683
|
-
this._sortValues();
|
|
684
|
-
if (isUserInput) {
|
|
685
|
-
// In case the user selected the option with their mouse, we
|
|
686
|
-
// want to restore focus back to the trigger, in order to
|
|
687
|
-
// prevent the select keyboard controls from clashing with
|
|
688
|
-
// the ones from `mat-option`.
|
|
689
|
-
this.focus();
|
|
690
|
-
}
|
|
691
|
-
}
|
|
638
|
+
});
|
|
639
|
+
this._keyManager.change.pipe(takeUntil(this._destroy)).subscribe(() => {
|
|
640
|
+
if (this._panelOpen && this.panel) {
|
|
641
|
+
this._scrollActiveOptionIntoView();
|
|
692
642
|
}
|
|
693
|
-
if (
|
|
694
|
-
this.
|
|
643
|
+
else if (!this._panelOpen && !this.multiple && this._keyManager.activeItem) {
|
|
644
|
+
this._keyManager.activeItem._selectViaInteraction();
|
|
695
645
|
}
|
|
696
|
-
|
|
697
|
-
|
|
698
|
-
|
|
699
|
-
|
|
700
|
-
|
|
701
|
-
|
|
702
|
-
|
|
703
|
-
|
|
704
|
-
|
|
705
|
-
|
|
706
|
-
this.stateChanges.next();
|
|
646
|
+
});
|
|
647
|
+
}
|
|
648
|
+
/** Drops current option subscriptions and IDs and resets from scratch. */
|
|
649
|
+
_resetOptions() {
|
|
650
|
+
const changedOrDestroyed = merge(this.options.changes, this._destroy);
|
|
651
|
+
this.optionSelectionChanges.pipe(takeUntil(changedOrDestroyed)).subscribe(event => {
|
|
652
|
+
this._onSelect(event.source, event.isUserInput);
|
|
653
|
+
if (event.isUserInput && !this.multiple && this._panelOpen) {
|
|
654
|
+
this.close();
|
|
655
|
+
this.focus();
|
|
707
656
|
}
|
|
657
|
+
});
|
|
658
|
+
// Listen to changes in the internal state of the options and react accordingly.
|
|
659
|
+
// Handles cases like the labels of the selected options changing.
|
|
660
|
+
merge(...this.options.map(option => option._stateChanges))
|
|
661
|
+
.pipe(takeUntil(changedOrDestroyed))
|
|
662
|
+
.subscribe(() => {
|
|
663
|
+
this._changeDetectorRef.markForCheck();
|
|
664
|
+
this.stateChanges.next();
|
|
665
|
+
});
|
|
666
|
+
this._setOptionIds();
|
|
667
|
+
}
|
|
668
|
+
/** Invoked when an option is clicked. */
|
|
669
|
+
_onSelect(option, isUserInput) {
|
|
670
|
+
const wasSelected = this._selectionModel.isSelected(option);
|
|
671
|
+
if (option.value == null && !this._multiple) {
|
|
672
|
+
option.deselect();
|
|
673
|
+
this._selectionModel.clear();
|
|
674
|
+
this._propagateChanges(option.value);
|
|
708
675
|
}
|
|
709
|
-
|
|
710
|
-
|
|
711
|
-
|
|
712
|
-
|
|
713
|
-
valueToEmit = this.selected.map(option => option.value);
|
|
676
|
+
else {
|
|
677
|
+
if (wasSelected !== option.selected) {
|
|
678
|
+
option.selected ? this._selectionModel.select(option) :
|
|
679
|
+
this._selectionModel.deselect(option);
|
|
714
680
|
}
|
|
715
|
-
|
|
716
|
-
|
|
681
|
+
if (isUserInput) {
|
|
682
|
+
this._keyManager.setActiveItem(option);
|
|
717
683
|
}
|
|
718
|
-
this.
|
|
719
|
-
|
|
720
|
-
|
|
721
|
-
|
|
722
|
-
|
|
723
|
-
|
|
724
|
-
|
|
725
|
-
|
|
726
|
-
this._optionIds = this.options.map(option => option.id).join(' ');
|
|
727
|
-
}
|
|
728
|
-
/**
|
|
729
|
-
* Highlights the selected item. If no option is selected, it will highlight
|
|
730
|
-
* the first item instead.
|
|
731
|
-
*/
|
|
732
|
-
_highlightCorrectOption() {
|
|
733
|
-
if (this._keyManager) {
|
|
734
|
-
if (this.empty) {
|
|
735
|
-
this._keyManager.setFirstItemActive();
|
|
736
|
-
}
|
|
737
|
-
else {
|
|
738
|
-
this._keyManager.setActiveItem(this._selectionModel.selected[0]);
|
|
684
|
+
if (this.multiple) {
|
|
685
|
+
this._sortValues();
|
|
686
|
+
if (isUserInput) {
|
|
687
|
+
// In case the user selected the option with their mouse, we
|
|
688
|
+
// want to restore focus back to the trigger, in order to
|
|
689
|
+
// prevent the select keyboard controls from clashing with
|
|
690
|
+
// the ones from `mat-option`.
|
|
691
|
+
this.focus();
|
|
739
692
|
}
|
|
740
693
|
}
|
|
741
694
|
}
|
|
742
|
-
|
|
743
|
-
|
|
744
|
-
const activeOptionIndex = this._keyManager.activeItemIndex || 0;
|
|
745
|
-
const labelCount = _countGroupLabelsBeforeOption(activeOptionIndex, this.options, this.optionGroups);
|
|
746
|
-
this.panel.nativeElement.scrollTop = _getOptionScrollPosition(activeOptionIndex + labelCount, this._getItemHeight(), this.panel.nativeElement.scrollTop, SELECT_PANEL_MAX_HEIGHT);
|
|
695
|
+
if (wasSelected !== this._selectionModel.isSelected(option)) {
|
|
696
|
+
this._propagateChanges();
|
|
747
697
|
}
|
|
748
|
-
|
|
749
|
-
|
|
750
|
-
|
|
751
|
-
|
|
752
|
-
|
|
753
|
-
|
|
754
|
-
|
|
755
|
-
|
|
756
|
-
|
|
757
|
-
|
|
758
|
-
|
|
759
|
-
}, undefined);
|
|
760
|
-
}
|
|
761
|
-
/** Calculates the scroll position and x- and y-offsets of the overlay panel. */
|
|
762
|
-
_calculateOverlayPosition() {
|
|
763
|
-
const itemHeight = this._getItemHeight();
|
|
764
|
-
const items = this._getItemCount();
|
|
765
|
-
const panelHeight = Math.min(items * itemHeight, SELECT_PANEL_MAX_HEIGHT);
|
|
766
|
-
const scrollContainerHeight = items * itemHeight;
|
|
767
|
-
// The farthest the panel can be scrolled before it hits the bottom
|
|
768
|
-
const maxScroll = scrollContainerHeight - panelHeight;
|
|
769
|
-
// If no value is selected we open the popup to the first item.
|
|
770
|
-
let selectedOptionOffset = this.empty ? 0 : this._getOptionIndex(this._selectionModel.selected[0]);
|
|
771
|
-
selectedOptionOffset += _countGroupLabelsBeforeOption(selectedOptionOffset, this.options, this.optionGroups);
|
|
772
|
-
// We must maintain a scroll buffer so the selected option will be scrolled to the
|
|
773
|
-
// center of the overlay panel rather than the top.
|
|
774
|
-
const scrollBuffer = panelHeight / 2;
|
|
775
|
-
this._scrollTop = this._calculateOverlayScroll(selectedOptionOffset, scrollBuffer, maxScroll);
|
|
776
|
-
this._offsetY = this._calculateOverlayOffsetY(selectedOptionOffset, scrollBuffer, maxScroll);
|
|
777
|
-
this._checkOverlayWithinViewport(maxScroll);
|
|
698
|
+
this.stateChanges.next();
|
|
699
|
+
}
|
|
700
|
+
/** Sorts the selected values in the selected based on their order in the panel. */
|
|
701
|
+
_sortValues() {
|
|
702
|
+
if (this.multiple) {
|
|
703
|
+
const options = this.options.toArray();
|
|
704
|
+
this._selectionModel.sort((a, b) => {
|
|
705
|
+
return this.sortComparator ? this.sortComparator(a, b, options) :
|
|
706
|
+
options.indexOf(a) - options.indexOf(b);
|
|
707
|
+
});
|
|
708
|
+
this.stateChanges.next();
|
|
778
709
|
}
|
|
779
|
-
|
|
780
|
-
|
|
781
|
-
|
|
782
|
-
|
|
783
|
-
|
|
784
|
-
|
|
785
|
-
*/
|
|
786
|
-
_calculateOverlayScroll(selectedIndex, scrollBuffer, maxScroll) {
|
|
787
|
-
const itemHeight = this._getItemHeight();
|
|
788
|
-
const optionOffsetFromScrollTop = itemHeight * selectedIndex;
|
|
789
|
-
const halfOptionHeight = itemHeight / 2;
|
|
790
|
-
// Starts at the optionOffsetFromScrollTop, which scrolls the option to the top of the
|
|
791
|
-
// scroll container, then subtracts the scroll buffer to scroll the option down to
|
|
792
|
-
// the center of the overlay panel. Half the option height must be re-added to the
|
|
793
|
-
// scrollTop so the option is centered based on its middle, not its top edge.
|
|
794
|
-
const optimalScrollPosition = optionOffsetFromScrollTop - scrollBuffer + halfOptionHeight;
|
|
795
|
-
return Math.min(Math.max(0, optimalScrollPosition), maxScroll);
|
|
710
|
+
}
|
|
711
|
+
/** Emits change event to set the model value. */
|
|
712
|
+
_propagateChanges(fallbackValue) {
|
|
713
|
+
let valueToEmit = null;
|
|
714
|
+
if (this.multiple) {
|
|
715
|
+
valueToEmit = this.selected.map(option => option.value);
|
|
796
716
|
}
|
|
797
|
-
|
|
798
|
-
|
|
799
|
-
// If an ariaLabelledby value has been set by the consumer, the select should not overwrite the
|
|
800
|
-
// `aria-labelledby` value by setting the ariaLabel to the placeholder.
|
|
801
|
-
return this.ariaLabelledby ? null : this.ariaLabel || this.placeholder;
|
|
717
|
+
else {
|
|
718
|
+
valueToEmit = this.selected ? this.selected.value : fallbackValue;
|
|
802
719
|
}
|
|
803
|
-
|
|
804
|
-
|
|
805
|
-
|
|
806
|
-
|
|
720
|
+
this._value = valueToEmit;
|
|
721
|
+
this.valueChange.emit(valueToEmit);
|
|
722
|
+
this._onChange(valueToEmit);
|
|
723
|
+
this.selectionChange.emit(new MatSelectChange(this, valueToEmit));
|
|
724
|
+
this._changeDetectorRef.markForCheck();
|
|
725
|
+
}
|
|
726
|
+
/** Records option IDs to pass to the aria-owns property. */
|
|
727
|
+
_setOptionIds() {
|
|
728
|
+
this._optionIds = this.options.map(option => option.id).join(' ');
|
|
729
|
+
}
|
|
730
|
+
/**
|
|
731
|
+
* Highlights the selected item. If no option is selected, it will highlight
|
|
732
|
+
* the first item instead.
|
|
733
|
+
*/
|
|
734
|
+
_highlightCorrectOption() {
|
|
735
|
+
if (this._keyManager) {
|
|
736
|
+
if (this.empty) {
|
|
737
|
+
this._keyManager.setFirstItemActive();
|
|
807
738
|
}
|
|
808
|
-
|
|
809
|
-
|
|
810
|
-
if (!this._parentFormField || !this._parentFormField._hasFloatingLabel() ||
|
|
811
|
-
this._getAriaLabel()) {
|
|
812
|
-
return null;
|
|
739
|
+
else {
|
|
740
|
+
this._keyManager.setActiveItem(this._selectionModel.selected[0]);
|
|
813
741
|
}
|
|
814
|
-
return this._parentFormField._labelId || null;
|
|
815
742
|
}
|
|
816
|
-
|
|
817
|
-
|
|
818
|
-
|
|
819
|
-
|
|
743
|
+
}
|
|
744
|
+
/** Scrolls the active option into view. */
|
|
745
|
+
_scrollActiveOptionIntoView() {
|
|
746
|
+
const activeOptionIndex = this._keyManager.activeItemIndex || 0;
|
|
747
|
+
const labelCount = _countGroupLabelsBeforeOption(activeOptionIndex, this.options, this.optionGroups);
|
|
748
|
+
this.panel.nativeElement.scrollTop = _getOptionScrollPosition(activeOptionIndex + labelCount, this._getItemHeight(), this.panel.nativeElement.scrollTop, SELECT_PANEL_MAX_HEIGHT);
|
|
749
|
+
}
|
|
750
|
+
/** Focuses the select element. */
|
|
751
|
+
focus(options) {
|
|
752
|
+
this._elementRef.nativeElement.focus(options);
|
|
753
|
+
}
|
|
754
|
+
/** Gets the index of the provided option in the option list. */
|
|
755
|
+
_getOptionIndex(option) {
|
|
756
|
+
return this.options.reduce((result, current, index) => {
|
|
757
|
+
if (result !== undefined) {
|
|
758
|
+
return result;
|
|
820
759
|
}
|
|
760
|
+
return option === current ? index : undefined;
|
|
761
|
+
}, undefined);
|
|
762
|
+
}
|
|
763
|
+
/** Calculates the scroll position and x- and y-offsets of the overlay panel. */
|
|
764
|
+
_calculateOverlayPosition() {
|
|
765
|
+
const itemHeight = this._getItemHeight();
|
|
766
|
+
const items = this._getItemCount();
|
|
767
|
+
const panelHeight = Math.min(items * itemHeight, SELECT_PANEL_MAX_HEIGHT);
|
|
768
|
+
const scrollContainerHeight = items * itemHeight;
|
|
769
|
+
// The farthest the panel can be scrolled before it hits the bottom
|
|
770
|
+
const maxScroll = scrollContainerHeight - panelHeight;
|
|
771
|
+
// If no value is selected we open the popup to the first item.
|
|
772
|
+
let selectedOptionOffset = this.empty ? 0 : this._getOptionIndex(this._selectionModel.selected[0]);
|
|
773
|
+
selectedOptionOffset += _countGroupLabelsBeforeOption(selectedOptionOffset, this.options, this.optionGroups);
|
|
774
|
+
// We must maintain a scroll buffer so the selected option will be scrolled to the
|
|
775
|
+
// center of the overlay panel rather than the top.
|
|
776
|
+
const scrollBuffer = panelHeight / 2;
|
|
777
|
+
this._scrollTop = this._calculateOverlayScroll(selectedOptionOffset, scrollBuffer, maxScroll);
|
|
778
|
+
this._offsetY = this._calculateOverlayOffsetY(selectedOptionOffset, scrollBuffer, maxScroll);
|
|
779
|
+
this._checkOverlayWithinViewport(maxScroll);
|
|
780
|
+
}
|
|
781
|
+
/**
|
|
782
|
+
* Calculates the scroll position of the select's overlay panel.
|
|
783
|
+
*
|
|
784
|
+
* Attempts to center the selected option in the panel. If the option is
|
|
785
|
+
* too high or too low in the panel to be scrolled to the center, it clamps the
|
|
786
|
+
* scroll position to the min or max scroll positions respectively.
|
|
787
|
+
*/
|
|
788
|
+
_calculateOverlayScroll(selectedIndex, scrollBuffer, maxScroll) {
|
|
789
|
+
const itemHeight = this._getItemHeight();
|
|
790
|
+
const optionOffsetFromScrollTop = itemHeight * selectedIndex;
|
|
791
|
+
const halfOptionHeight = itemHeight / 2;
|
|
792
|
+
// Starts at the optionOffsetFromScrollTop, which scrolls the option to the top of the
|
|
793
|
+
// scroll container, then subtracts the scroll buffer to scroll the option down to
|
|
794
|
+
// the center of the overlay panel. Half the option height must be re-added to the
|
|
795
|
+
// scrollTop so the option is centered based on its middle, not its top edge.
|
|
796
|
+
const optimalScrollPosition = optionOffsetFromScrollTop - scrollBuffer + halfOptionHeight;
|
|
797
|
+
return Math.min(Math.max(0, optimalScrollPosition), maxScroll);
|
|
798
|
+
}
|
|
799
|
+
/** Returns the aria-label of the select component. */
|
|
800
|
+
_getAriaLabel() {
|
|
801
|
+
// If an ariaLabelledby value has been set by the consumer, the select should not overwrite the
|
|
802
|
+
// `aria-labelledby` value by setting the ariaLabel to the placeholder.
|
|
803
|
+
return this.ariaLabelledby ? null : this.ariaLabel || this.placeholder;
|
|
804
|
+
}
|
|
805
|
+
/** Returns the aria-labelledby of the select component. */
|
|
806
|
+
_getAriaLabelledby() {
|
|
807
|
+
if (this.ariaLabelledby) {
|
|
808
|
+
return this.ariaLabelledby;
|
|
809
|
+
}
|
|
810
|
+
// Note: we use `_getAriaLabel` here, because we want to check whether there's a
|
|
811
|
+
// computed label. `this.ariaLabel` is only the user-specified label.
|
|
812
|
+
if (!this._parentFormField || !this._parentFormField._hasFloatingLabel() ||
|
|
813
|
+
this._getAriaLabel()) {
|
|
821
814
|
return null;
|
|
822
815
|
}
|
|
823
|
-
|
|
824
|
-
|
|
825
|
-
|
|
826
|
-
|
|
827
|
-
|
|
828
|
-
|
|
829
|
-
*/
|
|
830
|
-
_calculateOverlayOffsetX() {
|
|
831
|
-
const overlayRect = this.overlayDir.overlayRef.overlayElement.getBoundingClientRect();
|
|
832
|
-
const viewportSize = this._viewportRuler.getViewportSize();
|
|
833
|
-
const isRtl = this._isRtl();
|
|
834
|
-
const paddingWidth = this.multiple ? SELECT_MULTIPLE_PANEL_PADDING_X + SELECT_PANEL_PADDING_X :
|
|
835
|
-
SELECT_PANEL_PADDING_X * 2;
|
|
836
|
-
let offsetX;
|
|
837
|
-
// Adjust the offset, depending on the option padding.
|
|
838
|
-
if (this.multiple) {
|
|
839
|
-
offsetX = SELECT_MULTIPLE_PANEL_PADDING_X;
|
|
840
|
-
}
|
|
841
|
-
else {
|
|
842
|
-
let selected = this._selectionModel.selected[0] || this.options.first;
|
|
843
|
-
offsetX = selected && selected.group ? SELECT_PANEL_INDENT_PADDING_X : SELECT_PANEL_PADDING_X;
|
|
844
|
-
}
|
|
845
|
-
// Invert the offset in LTR.
|
|
846
|
-
if (!isRtl) {
|
|
847
|
-
offsetX *= -1;
|
|
848
|
-
}
|
|
849
|
-
// Determine how much the select overflows on each side.
|
|
850
|
-
const leftOverflow = 0 - (overlayRect.left + offsetX - (isRtl ? paddingWidth : 0));
|
|
851
|
-
const rightOverflow = overlayRect.right + offsetX - viewportSize.width
|
|
852
|
-
+ (isRtl ? 0 : paddingWidth);
|
|
853
|
-
// If the element overflows on either side, reduce the offset to allow it to fit.
|
|
854
|
-
if (leftOverflow > 0) {
|
|
855
|
-
offsetX += leftOverflow + SELECT_PANEL_VIEWPORT_PADDING;
|
|
856
|
-
}
|
|
857
|
-
else if (rightOverflow > 0) {
|
|
858
|
-
offsetX -= rightOverflow + SELECT_PANEL_VIEWPORT_PADDING;
|
|
859
|
-
}
|
|
860
|
-
// Set the offset directly in order to avoid having to go through change detection and
|
|
861
|
-
// potentially triggering "changed after it was checked" errors. Round the value to avoid
|
|
862
|
-
// blurry content in some browsers.
|
|
863
|
-
this.overlayDir.offsetX = Math.round(offsetX);
|
|
864
|
-
this.overlayDir.overlayRef.updatePosition();
|
|
816
|
+
return this._parentFormField._labelId || null;
|
|
817
|
+
}
|
|
818
|
+
/** Determines the `aria-activedescendant` to be set on the host. */
|
|
819
|
+
_getAriaActiveDescendant() {
|
|
820
|
+
if (this.panelOpen && this._keyManager && this._keyManager.activeItem) {
|
|
821
|
+
return this._keyManager.activeItem.id;
|
|
865
822
|
}
|
|
866
|
-
|
|
867
|
-
|
|
868
|
-
|
|
869
|
-
|
|
870
|
-
|
|
871
|
-
|
|
872
|
-
|
|
873
|
-
|
|
874
|
-
|
|
875
|
-
|
|
876
|
-
|
|
877
|
-
|
|
878
|
-
|
|
879
|
-
|
|
880
|
-
|
|
881
|
-
|
|
882
|
-
|
|
883
|
-
|
|
884
|
-
|
|
885
|
-
const selectedDisplayIndex = selectedIndex - firstDisplayedIndex;
|
|
886
|
-
// The first item is partially out of the viewport. Therefore we need to calculate what
|
|
887
|
-
// portion of it is shown in the viewport and account for it in our offset.
|
|
888
|
-
let partialItemHeight = itemHeight - (this._getItemCount() * itemHeight - SELECT_PANEL_MAX_HEIGHT) % itemHeight;
|
|
889
|
-
// Because the panel height is longer than the height of the options alone,
|
|
890
|
-
// there is always extra padding at the top or bottom of the panel. When
|
|
891
|
-
// scrolled to the very bottom, this padding is at the top of the panel and
|
|
892
|
-
// must be added to the offset.
|
|
893
|
-
optionOffsetFromPanelTop = selectedDisplayIndex * itemHeight + partialItemHeight;
|
|
894
|
-
}
|
|
895
|
-
else {
|
|
896
|
-
// If the option was scrolled to the middle of the panel using a scroll buffer,
|
|
897
|
-
// its offset will be the scroll buffer minus the half height that was added to
|
|
898
|
-
// center it.
|
|
899
|
-
optionOffsetFromPanelTop = scrollBuffer - itemHeight / 2;
|
|
900
|
-
}
|
|
901
|
-
// The final offset is the option's offset from the top, adjusted for the height difference,
|
|
902
|
-
// multiplied by -1 to ensure that the overlay moves in the correct direction up the page.
|
|
903
|
-
// The value is rounded to prevent some browsers from blurring the content.
|
|
904
|
-
return Math.round(optionOffsetFromPanelTop * -1 - optionHeightAdjustment);
|
|
823
|
+
return null;
|
|
824
|
+
}
|
|
825
|
+
/**
|
|
826
|
+
* Sets the x-offset of the overlay panel in relation to the trigger's top start corner.
|
|
827
|
+
* This must be adjusted to align the selected option text over the trigger text when
|
|
828
|
+
* the panel opens. Will change based on LTR or RTL text direction. Note that the offset
|
|
829
|
+
* can't be calculated until the panel has been attached, because we need to know the
|
|
830
|
+
* content width in order to constrain the panel within the viewport.
|
|
831
|
+
*/
|
|
832
|
+
_calculateOverlayOffsetX() {
|
|
833
|
+
const overlayRect = this.overlayDir.overlayRef.overlayElement.getBoundingClientRect();
|
|
834
|
+
const viewportSize = this._viewportRuler.getViewportSize();
|
|
835
|
+
const isRtl = this._isRtl();
|
|
836
|
+
const paddingWidth = this.multiple ? SELECT_MULTIPLE_PANEL_PADDING_X + SELECT_PANEL_PADDING_X :
|
|
837
|
+
SELECT_PANEL_PADDING_X * 2;
|
|
838
|
+
let offsetX;
|
|
839
|
+
// Adjust the offset, depending on the option padding.
|
|
840
|
+
if (this.multiple) {
|
|
841
|
+
offsetX = SELECT_MULTIPLE_PANEL_PADDING_X;
|
|
905
842
|
}
|
|
906
|
-
|
|
907
|
-
|
|
908
|
-
|
|
909
|
-
* y-offset so the panel can open fully on-screen. If it still won't fit,
|
|
910
|
-
* sets the offset back to 0 to allow the fallback position to take over.
|
|
911
|
-
*/
|
|
912
|
-
_checkOverlayWithinViewport(maxScroll) {
|
|
913
|
-
const itemHeight = this._getItemHeight();
|
|
914
|
-
const viewportSize = this._viewportRuler.getViewportSize();
|
|
915
|
-
const topSpaceAvailable = this._triggerRect.top - SELECT_PANEL_VIEWPORT_PADDING;
|
|
916
|
-
const bottomSpaceAvailable = viewportSize.height - this._triggerRect.bottom - SELECT_PANEL_VIEWPORT_PADDING;
|
|
917
|
-
const panelHeightTop = Math.abs(this._offsetY);
|
|
918
|
-
const totalPanelHeight = Math.min(this._getItemCount() * itemHeight, SELECT_PANEL_MAX_HEIGHT);
|
|
919
|
-
const panelHeightBottom = totalPanelHeight - panelHeightTop - this._triggerRect.height;
|
|
920
|
-
if (panelHeightBottom > bottomSpaceAvailable) {
|
|
921
|
-
this._adjustPanelUp(panelHeightBottom, bottomSpaceAvailable);
|
|
922
|
-
}
|
|
923
|
-
else if (panelHeightTop > topSpaceAvailable) {
|
|
924
|
-
this._adjustPanelDown(panelHeightTop, topSpaceAvailable, maxScroll);
|
|
925
|
-
}
|
|
926
|
-
else {
|
|
927
|
-
this._transformOrigin = this._getOriginBasedOnOption();
|
|
928
|
-
}
|
|
843
|
+
else {
|
|
844
|
+
let selected = this._selectionModel.selected[0] || this.options.first;
|
|
845
|
+
offsetX = selected && selected.group ? SELECT_PANEL_INDENT_PADDING_X : SELECT_PANEL_PADDING_X;
|
|
929
846
|
}
|
|
930
|
-
|
|
931
|
-
|
|
932
|
-
|
|
933
|
-
const distanceBelowViewport = Math.round(panelHeightBottom - bottomSpaceAvailable);
|
|
934
|
-
// Scrolls the panel up by the distance it was extending past the boundary, then
|
|
935
|
-
// adjusts the offset by that amount to move the panel up into the viewport.
|
|
936
|
-
this._scrollTop -= distanceBelowViewport;
|
|
937
|
-
this._offsetY -= distanceBelowViewport;
|
|
938
|
-
this._transformOrigin = this._getOriginBasedOnOption();
|
|
939
|
-
// If the panel is scrolled to the very top, it won't be able to fit the panel
|
|
940
|
-
// by scrolling, so set the offset to 0 to allow the fallback position to take
|
|
941
|
-
// effect.
|
|
942
|
-
if (this._scrollTop <= 0) {
|
|
943
|
-
this._scrollTop = 0;
|
|
944
|
-
this._offsetY = 0;
|
|
945
|
-
this._transformOrigin = `50% bottom 0px`;
|
|
946
|
-
}
|
|
847
|
+
// Invert the offset in LTR.
|
|
848
|
+
if (!isRtl) {
|
|
849
|
+
offsetX *= -1;
|
|
947
850
|
}
|
|
948
|
-
|
|
949
|
-
|
|
950
|
-
|
|
951
|
-
|
|
952
|
-
|
|
953
|
-
|
|
954
|
-
|
|
955
|
-
this._offsetY += distanceAboveViewport;
|
|
956
|
-
this._transformOrigin = this._getOriginBasedOnOption();
|
|
957
|
-
// If the panel is scrolled to the very bottom, it won't be able to fit the
|
|
958
|
-
// panel by scrolling, so set the offset to 0 to allow the fallback position
|
|
959
|
-
// to take effect.
|
|
960
|
-
if (this._scrollTop >= maxScroll) {
|
|
961
|
-
this._scrollTop = maxScroll;
|
|
962
|
-
this._offsetY = 0;
|
|
963
|
-
this._transformOrigin = `50% top 0px`;
|
|
964
|
-
return;
|
|
965
|
-
}
|
|
851
|
+
// Determine how much the select overflows on each side.
|
|
852
|
+
const leftOverflow = 0 - (overlayRect.left + offsetX - (isRtl ? paddingWidth : 0));
|
|
853
|
+
const rightOverflow = overlayRect.right + offsetX - viewportSize.width
|
|
854
|
+
+ (isRtl ? 0 : paddingWidth);
|
|
855
|
+
// If the element overflows on either side, reduce the offset to allow it to fit.
|
|
856
|
+
if (leftOverflow > 0) {
|
|
857
|
+
offsetX += leftOverflow + SELECT_PANEL_VIEWPORT_PADDING;
|
|
966
858
|
}
|
|
967
|
-
|
|
968
|
-
|
|
969
|
-
const itemHeight = this._getItemHeight();
|
|
970
|
-
const optionHeightAdjustment = (itemHeight - this._triggerRect.height) / 2;
|
|
971
|
-
const originY = Math.abs(this._offsetY) - optionHeightAdjustment + itemHeight / 2;
|
|
972
|
-
return `50% ${originY}px 0px`;
|
|
859
|
+
else if (rightOverflow > 0) {
|
|
860
|
+
offsetX -= rightOverflow + SELECT_PANEL_VIEWPORT_PADDING;
|
|
973
861
|
}
|
|
974
|
-
|
|
975
|
-
|
|
976
|
-
|
|
862
|
+
// Set the offset directly in order to avoid having to go through change detection and
|
|
863
|
+
// potentially triggering "changed after it was checked" errors. Round the value to avoid
|
|
864
|
+
// blurry content in some browsers.
|
|
865
|
+
this.overlayDir.offsetX = Math.round(offsetX);
|
|
866
|
+
this.overlayDir.overlayRef.updatePosition();
|
|
867
|
+
}
|
|
868
|
+
/**
|
|
869
|
+
* Calculates the y-offset of the select's overlay panel in relation to the
|
|
870
|
+
* top start corner of the trigger. It has to be adjusted in order for the
|
|
871
|
+
* selected option to be aligned over the trigger when the panel opens.
|
|
872
|
+
*/
|
|
873
|
+
_calculateOverlayOffsetY(selectedIndex, scrollBuffer, maxScroll) {
|
|
874
|
+
const itemHeight = this._getItemHeight();
|
|
875
|
+
const optionHeightAdjustment = (itemHeight - this._triggerRect.height) / 2;
|
|
876
|
+
const maxOptionsDisplayed = Math.floor(SELECT_PANEL_MAX_HEIGHT / itemHeight);
|
|
877
|
+
let optionOffsetFromPanelTop;
|
|
878
|
+
// Disable offset if requested by user by returning 0 as value to offset
|
|
879
|
+
if (this._disableOptionCentering) {
|
|
880
|
+
return 0;
|
|
977
881
|
}
|
|
978
|
-
|
|
979
|
-
|
|
980
|
-
return this._triggerFontSize * SELECT_ITEM_HEIGHT_EM;
|
|
882
|
+
if (this._scrollTop === 0) {
|
|
883
|
+
optionOffsetFromPanelTop = selectedIndex * itemHeight;
|
|
981
884
|
}
|
|
982
|
-
|
|
983
|
-
|
|
984
|
-
|
|
985
|
-
|
|
986
|
-
|
|
987
|
-
|
|
885
|
+
else if (this._scrollTop === maxScroll) {
|
|
886
|
+
const firstDisplayedIndex = this._getItemCount() - maxOptionsDisplayed;
|
|
887
|
+
const selectedDisplayIndex = selectedIndex - firstDisplayedIndex;
|
|
888
|
+
// The first item is partially out of the viewport. Therefore we need to calculate what
|
|
889
|
+
// portion of it is shown in the viewport and account for it in our offset.
|
|
890
|
+
let partialItemHeight = itemHeight - (this._getItemCount() * itemHeight - SELECT_PANEL_MAX_HEIGHT) % itemHeight;
|
|
891
|
+
// Because the panel height is longer than the height of the options alone,
|
|
892
|
+
// there is always extra padding at the top or bottom of the panel. When
|
|
893
|
+
// scrolled to the very bottom, this padding is at the top of the panel and
|
|
894
|
+
// must be added to the offset.
|
|
895
|
+
optionOffsetFromPanelTop = selectedDisplayIndex * itemHeight + partialItemHeight;
|
|
988
896
|
}
|
|
989
|
-
|
|
990
|
-
|
|
991
|
-
|
|
992
|
-
|
|
993
|
-
|
|
994
|
-
this.focus();
|
|
995
|
-
this.open();
|
|
897
|
+
else {
|
|
898
|
+
// If the option was scrolled to the middle of the panel using a scroll buffer,
|
|
899
|
+
// its offset will be the scroll buffer minus the half height that was added to
|
|
900
|
+
// center it.
|
|
901
|
+
optionOffsetFromPanelTop = scrollBuffer - itemHeight / 2;
|
|
996
902
|
}
|
|
997
|
-
|
|
998
|
-
|
|
999
|
-
|
|
1000
|
-
|
|
1001
|
-
|
|
1002
|
-
|
|
903
|
+
// The final offset is the option's offset from the top, adjusted for the height difference,
|
|
904
|
+
// multiplied by -1 to ensure that the overlay moves in the correct direction up the page.
|
|
905
|
+
// The value is rounded to prevent some browsers from blurring the content.
|
|
906
|
+
return Math.round(optionOffsetFromPanelTop * -1 - optionHeightAdjustment);
|
|
907
|
+
}
|
|
908
|
+
/**
|
|
909
|
+
* Checks that the attempted overlay position will fit within the viewport.
|
|
910
|
+
* If it will not fit, tries to adjust the scroll position and the associated
|
|
911
|
+
* y-offset so the panel can open fully on-screen. If it still won't fit,
|
|
912
|
+
* sets the offset back to 0 to allow the fallback position to take over.
|
|
913
|
+
*/
|
|
914
|
+
_checkOverlayWithinViewport(maxScroll) {
|
|
915
|
+
const itemHeight = this._getItemHeight();
|
|
916
|
+
const viewportSize = this._viewportRuler.getViewportSize();
|
|
917
|
+
const topSpaceAvailable = this._triggerRect.top - SELECT_PANEL_VIEWPORT_PADDING;
|
|
918
|
+
const bottomSpaceAvailable = viewportSize.height - this._triggerRect.bottom - SELECT_PANEL_VIEWPORT_PADDING;
|
|
919
|
+
const panelHeightTop = Math.abs(this._offsetY);
|
|
920
|
+
const totalPanelHeight = Math.min(this._getItemCount() * itemHeight, SELECT_PANEL_MAX_HEIGHT);
|
|
921
|
+
const panelHeightBottom = totalPanelHeight - panelHeightTop - this._triggerRect.height;
|
|
922
|
+
if (panelHeightBottom > bottomSpaceAvailable) {
|
|
923
|
+
this._adjustPanelUp(panelHeightBottom, bottomSpaceAvailable);
|
|
924
|
+
}
|
|
925
|
+
else if (panelHeightTop > topSpaceAvailable) {
|
|
926
|
+
this._adjustPanelDown(panelHeightTop, topSpaceAvailable, maxScroll);
|
|
927
|
+
}
|
|
928
|
+
else {
|
|
929
|
+
this._transformOrigin = this._getOriginBasedOnOption();
|
|
930
|
+
}
|
|
931
|
+
}
|
|
932
|
+
/** Adjusts the overlay panel up to fit in the viewport. */
|
|
933
|
+
_adjustPanelUp(panelHeightBottom, bottomSpaceAvailable) {
|
|
934
|
+
// Browsers ignore fractional scroll offsets, so we need to round.
|
|
935
|
+
const distanceBelowViewport = Math.round(panelHeightBottom - bottomSpaceAvailable);
|
|
936
|
+
// Scrolls the panel up by the distance it was extending past the boundary, then
|
|
937
|
+
// adjusts the offset by that amount to move the panel up into the viewport.
|
|
938
|
+
this._scrollTop -= distanceBelowViewport;
|
|
939
|
+
this._offsetY -= distanceBelowViewport;
|
|
940
|
+
this._transformOrigin = this._getOriginBasedOnOption();
|
|
941
|
+
// If the panel is scrolled to the very top, it won't be able to fit the panel
|
|
942
|
+
// by scrolling, so set the offset to 0 to allow the fallback position to take
|
|
943
|
+
// effect.
|
|
944
|
+
if (this._scrollTop <= 0) {
|
|
945
|
+
this._scrollTop = 0;
|
|
946
|
+
this._offsetY = 0;
|
|
947
|
+
this._transformOrigin = `50% bottom 0px`;
|
|
1003
948
|
}
|
|
1004
949
|
}
|
|
1005
|
-
|
|
1006
|
-
|
|
1007
|
-
|
|
1008
|
-
|
|
1009
|
-
|
|
1010
|
-
|
|
1011
|
-
|
|
1012
|
-
|
|
1013
|
-
|
|
1014
|
-
|
|
1015
|
-
|
|
1016
|
-
|
|
1017
|
-
|
|
1018
|
-
|
|
1019
|
-
|
|
1020
|
-
|
|
1021
|
-
|
|
1022
|
-
|
|
1023
|
-
|
|
1024
|
-
|
|
1025
|
-
|
|
1026
|
-
|
|
1027
|
-
|
|
1028
|
-
|
|
1029
|
-
|
|
1030
|
-
|
|
1031
|
-
|
|
1032
|
-
|
|
1033
|
-
|
|
1034
|
-
|
|
1035
|
-
|
|
1036
|
-
|
|
1037
|
-
|
|
1038
|
-
|
|
1039
|
-
|
|
1040
|
-
|
|
1041
|
-
|
|
1042
|
-
|
|
1043
|
-
|
|
1044
|
-
|
|
1045
|
-
|
|
1046
|
-
|
|
1047
|
-
|
|
1048
|
-
|
|
1049
|
-
|
|
1050
|
-
|
|
1051
|
-
|
|
1052
|
-
|
|
1053
|
-
|
|
1054
|
-
|
|
1055
|
-
|
|
1056
|
-
|
|
1057
|
-
|
|
1058
|
-
|
|
1059
|
-
|
|
1060
|
-
|
|
1061
|
-
|
|
1062
|
-
|
|
1063
|
-
|
|
1064
|
-
|
|
1065
|
-
|
|
1066
|
-
|
|
1067
|
-
|
|
1068
|
-
|
|
1069
|
-
|
|
1070
|
-
|
|
1071
|
-
|
|
1072
|
-
|
|
1073
|
-
|
|
1074
|
-
|
|
1075
|
-
|
|
1076
|
-
|
|
1077
|
-
|
|
1078
|
-
|
|
1079
|
-
|
|
1080
|
-
|
|
1081
|
-
|
|
1082
|
-
|
|
1083
|
-
|
|
1084
|
-
|
|
1085
|
-
|
|
1086
|
-
|
|
1087
|
-
|
|
1088
|
-
|
|
1089
|
-
|
|
1090
|
-
|
|
1091
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"select.js","sourceRoot":"","sources":["../../../../../../src/material/select/select.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * @license\n * Copyright Google LLC All Rights Reserved.\n *\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://angular.io/license\n */\n\nimport {ActiveDescendantKeyManager, LiveAnnouncer} from '@angular/cdk/a11y';\nimport {Directionality} from '@angular/cdk/bidi';\nimport {\n  BooleanInput,\n  coerceBooleanProperty,\n  coerceNumberProperty,\n  NumberInput\n} from '@angular/cdk/coercion';\nimport {SelectionModel} from '@angular/cdk/collections';\nimport {\n  A,\n  DOWN_ARROW,\n  END,\n  ENTER,\n  hasModifierKey,\n  HOME,\n  LEFT_ARROW,\n  RIGHT_ARROW,\n  SPACE,\n  UP_ARROW,\n} from '@angular/cdk/keycodes';\nimport {\n  CdkConnectedOverlay,\n  ConnectedPosition,\n  Overlay,\n  ScrollStrategy,\n} from '@angular/cdk/overlay';\nimport {ViewportRuler} from '@angular/cdk/scrolling';\nimport {\n  AfterContentInit,\n  Attribute,\n  ChangeDetectionStrategy,\n  ChangeDetectorRef,\n  Component,\n  ContentChild,\n  ContentChildren,\n  Directive,\n  DoCheck,\n  ElementRef,\n  EventEmitter,\n  Inject,\n  InjectionToken,\n  Input,\n  isDevMode,\n  NgZone,\n  OnChanges,\n  OnDestroy,\n  OnInit,\n  Optional,\n  Output,\n  QueryList,\n  Self,\n  SimpleChanges,\n  ViewChild,\n  ViewEncapsulation,\n} from '@angular/core';\nimport {ControlValueAccessor, FormGroupDirective, NgControl, NgForm} from '@angular/forms';\nimport {\n  _countGroupLabelsBeforeOption,\n  _getOptionScrollPosition,\n  CanDisable,\n  CanDisableCtor,\n  CanDisableRipple,\n  CanDisableRippleCtor,\n  CanUpdateErrorState,\n  CanUpdateErrorStateCtor,\n  ErrorStateMatcher,\n  HasTabIndex,\n  HasTabIndexCtor,\n  MAT_OPTION_PARENT_COMPONENT,\n  MatOptgroup,\n  MatOption,\n  MatOptionSelectionChange,\n  mixinDisabled,\n  mixinDisableRipple,\n  mixinErrorState,\n  mixinTabIndex,\n} from '@angular/material/core';\nimport {MAT_FORM_FIELD, MatFormField, MatFormFieldControl} from '@angular/material/form-field';\nimport {defer, merge, Observable, Subject} from 'rxjs';\nimport {\n  distinctUntilChanged,\n  filter,\n  map,\n  startWith,\n  switchMap,\n  take,\n  takeUntil,\n} from 'rxjs/operators';\nimport {matSelectAnimations} from './select-animations';\nimport {\n  getMatSelectDynamicMultipleError,\n  getMatSelectNonArrayValueError,\n  getMatSelectNonFunctionValueError,\n} from './select-errors';\n\n\nlet nextUniqueId = 0;\n\n/**\n * The following style constants are necessary to save here in order\n * to properly calculate the alignment of the selected option over\n * the trigger element.\n */\n\n/** The max height of the select's overlay panel */\nexport const SELECT_PANEL_MAX_HEIGHT = 256;\n\n/** The panel's padding on the x-axis */\nexport const SELECT_PANEL_PADDING_X = 16;\n\n/** The panel's x axis padding if it is indented (e.g. there is an option group). */\nexport const SELECT_PANEL_INDENT_PADDING_X = SELECT_PANEL_PADDING_X * 2;\n\n/** The height of the select items in `em` units. */\nexport const SELECT_ITEM_HEIGHT_EM = 3;\n\n// TODO(josephperrott): Revert to a constant after 2018 spec updates are fully merged.\n/**\n * Distance between the panel edge and the option text in\n * multi-selection mode.\n *\n * Calculated as:\n * (SELECT_PANEL_PADDING_X * 1.5) + 16 = 40\n * The padding is multiplied by 1.5 because the checkbox's margin is half the padding.\n * The checkbox width is 16px.\n */\nexport const SELECT_MULTIPLE_PANEL_PADDING_X = SELECT_PANEL_PADDING_X * 1.5 + 16;\n\n/**\n * The select panel will only \"fit\" inside the viewport if it is positioned at\n * this value or more away from the viewport boundary.\n */\nexport const SELECT_PANEL_VIEWPORT_PADDING = 8;\n\n/** Injection token that determines the scroll handling while a select is open. */\nexport const MAT_SELECT_SCROLL_STRATEGY =\n    new InjectionToken<() => ScrollStrategy>('mat-select-scroll-strategy');\n\n/** @docs-private */\nexport function MAT_SELECT_SCROLL_STRATEGY_PROVIDER_FACTORY(overlay: Overlay):\n    () => ScrollStrategy {\n  return () => overlay.scrollStrategies.reposition();\n}\n\n/** Object that can be used to configure the default options for the select module. */\nexport interface MatSelectConfig {\n  /** Whether option centering should be disabled. */\n  disableOptionCentering?: boolean;\n\n  /** Time to wait in milliseconds after the last keystroke before moving focus to an item. */\n  typeaheadDebounceInterval?: number;\n}\n\n/** Injection token that can be used to provide the default options the select module. */\nexport const MAT_SELECT_CONFIG = new InjectionToken<MatSelectConfig>('MAT_SELECT_CONFIG');\n\n/** @docs-private */\nexport const MAT_SELECT_SCROLL_STRATEGY_PROVIDER = {\n  provide: MAT_SELECT_SCROLL_STRATEGY,\n  deps: [Overlay],\n  useFactory: MAT_SELECT_SCROLL_STRATEGY_PROVIDER_FACTORY,\n};\n\n/** Change event object that is emitted when the select value has changed. */\nexport class MatSelectChange {\n  constructor(\n    /** Reference to the select that emitted the change event. */\n    public source: MatSelect,\n    /** Current value of the select that emitted the event. */\n    public value: any) { }\n}\n\n// Boilerplate for applying mixins to MatSelect.\n/** @docs-private */\nclass MatSelectBase {\n  constructor(public _elementRef: ElementRef,\n              public _defaultErrorStateMatcher: ErrorStateMatcher,\n              public _parentForm: NgForm,\n              public _parentFormGroup: FormGroupDirective,\n              public ngControl: NgControl) {}\n}\nconst _MatSelectMixinBase:\n    CanDisableCtor &\n    HasTabIndexCtor &\n    CanDisableRippleCtor &\n    CanUpdateErrorStateCtor &\n    typeof MatSelectBase =\n        mixinDisableRipple(mixinTabIndex(mixinDisabled(mixinErrorState(MatSelectBase))));\n\n\n/**\n * Allows the user to customize the trigger that is displayed when the select has a value.\n */\n@Directive({\n  selector: 'mat-select-trigger'\n})\nexport class MatSelectTrigger {}\n\n\n@Component({\n  selector: 'mat-select',\n  exportAs: 'matSelect',\n  templateUrl: 'select.html',\n  styleUrls: ['select.css'],\n  inputs: ['disabled', 'disableRipple', 'tabIndex'],\n  encapsulation: ViewEncapsulation.None,\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  host: {\n    'role': 'listbox',\n    '[attr.id]': 'id',\n    '[attr.tabindex]': 'tabIndex',\n    '[attr.aria-label]': '_getAriaLabel()',\n    '[attr.aria-labelledby]': '_getAriaLabelledby()',\n    '[attr.aria-required]': 'required.toString()',\n    '[attr.aria-disabled]': 'disabled.toString()',\n    '[attr.aria-invalid]': 'errorState',\n    '[attr.aria-owns]': 'panelOpen ? _optionIds : null',\n    '[attr.aria-multiselectable]': 'multiple',\n    '[attr.aria-describedby]': '_ariaDescribedby || null',\n    '[attr.aria-activedescendant]': '_getAriaActiveDescendant()',\n    '[class.mat-select-disabled]': 'disabled',\n    '[class.mat-select-invalid]': 'errorState',\n    '[class.mat-select-required]': 'required',\n    '[class.mat-select-empty]': 'empty',\n    'class': 'mat-select',\n    '(keydown)': '_handleKeydown($event)',\n    '(focus)': '_onFocus()',\n    '(blur)': '_onBlur()',\n  },\n  animations: [\n    matSelectAnimations.transformPanelWrap,\n    matSelectAnimations.transformPanel\n  ],\n  providers: [\n    {provide: MatFormFieldControl, useExisting: MatSelect},\n    {provide: MAT_OPTION_PARENT_COMPONENT, useExisting: MatSelect}\n  ],\n})\nexport class MatSelect extends _MatSelectMixinBase implements AfterContentInit, OnChanges,\n    OnDestroy, OnInit, DoCheck, ControlValueAccessor, CanDisable, HasTabIndex,\n    MatFormFieldControl<any>, CanUpdateErrorState, CanDisableRipple {\n  private _scrollStrategyFactory: () => ScrollStrategy;\n\n  /** Whether or not the overlay panel is open. */\n  private _panelOpen = false;\n\n  /** Whether filling out the select is required in the form. */\n  private _required: boolean = false;\n\n  /** The scroll position of the overlay panel, calculated to center the selected option. */\n  private _scrollTop = 0;\n\n  /** The placeholder displayed in the trigger of the select. */\n  private _placeholder: string;\n\n  /** Whether the component is in multiple selection mode. */\n  private _multiple: boolean = false;\n\n  /** Comparison function to specify which option is displayed. Defaults to object equality. */\n  private _compareWith = (o1: any, o2: any) => o1 === o2;\n\n  /** Unique id for this input. */\n  private _uid = `mat-select-${nextUniqueId++}`;\n\n  /** Emits whenever the component is destroyed. */\n  private readonly _destroy = new Subject<void>();\n\n  /** The last measured value for the trigger's client bounding rect. */\n  _triggerRect: ClientRect;\n\n  /** The aria-describedby attribute on the select for improved a11y. */\n  _ariaDescribedby: string;\n\n  /** The cached font-size of the trigger element. */\n  _triggerFontSize = 0;\n\n  /** Deals with the selection logic. */\n  _selectionModel: SelectionModel<MatOption>;\n\n  /** Manages keyboard events for options in the panel. */\n  _keyManager: ActiveDescendantKeyManager<MatOption>;\n\n  /** `View -> model callback called when value changes` */\n  _onChange: (value: any) => void = () => {};\n\n  /** `View -> model callback called when select has been touched` */\n  _onTouched = () => {};\n\n  /** The IDs of child options to be passed to the aria-owns attribute. */\n  _optionIds: string = '';\n\n  /** The value of the select panel's transform-origin property. */\n  _transformOrigin: string = 'top';\n\n  /** Emits when the panel element is finished transforming in. */\n  _panelDoneAnimatingStream = new Subject<string>();\n\n  /** Strategy that will be used to handle scrolling while the select panel is open. */\n  _scrollStrategy: ScrollStrategy;\n\n  /**\n   * The y-offset of the overlay panel in relation to the trigger's top start corner.\n   * This must be adjusted to align the selected option text over the trigger text.\n   * when the panel opens. Will change based on the y-position of the selected option.\n   */\n  _offsetY = 0;\n\n  /**\n   * This position config ensures that the top \"start\" corner of the overlay\n   * is aligned with with the top \"start\" of the origin by default (overlapping\n   * the trigger completely). If the panel cannot fit below the trigger, it\n   * will fall back to a position above the trigger.\n   */\n  _positions: ConnectedPosition[] = [\n    {\n      originX: 'start',\n      originY: 'top',\n      overlayX: 'start',\n      overlayY: 'top',\n    },\n    {\n      originX: 'start',\n      originY: 'bottom',\n      overlayX: 'start',\n      overlayY: 'bottom',\n    },\n  ];\n\n  /** Whether the component is disabling centering of the active option over the trigger. */\n  private _disableOptionCentering: boolean = false;\n\n  /** Whether the select is focused. */\n  get focused(): boolean {\n    return this._focused || this._panelOpen;\n  }\n  private _focused = false;\n\n  /** A name for this control that can be used by `mat-form-field`. */\n  controlType = 'mat-select';\n\n  /** Trigger that opens the select. */\n  @ViewChild('trigger') trigger: ElementRef;\n\n  /** Panel containing the select options. */\n  @ViewChild('panel') panel: ElementRef;\n\n  /**\n   * Overlay pane containing the options.\n   * @deprecated To be turned into a private API.\n   * @breaking-change 10.0.0\n   * @docs-private\n   */\n  @ViewChild(CdkConnectedOverlay) overlayDir: CdkConnectedOverlay;\n\n  /** All of the defined select options. */\n  @ContentChildren(MatOption, {descendants: true}) options: QueryList<MatOption>;\n\n  /** All of the defined groups of options. */\n  @ContentChildren(MatOptgroup, {descendants: true}) optionGroups: QueryList<MatOptgroup>;\n\n  /** Classes to be passed to the select panel. Supports the same syntax as `ngClass`. */\n  @Input() panelClass: string|string[]|Set<string>|{[key: string]: any};\n\n  /** User-supplied override of the trigger element. */\n  @ContentChild(MatSelectTrigger) customTrigger: MatSelectTrigger;\n\n  /** Placeholder to be shown if no value has been selected. */\n  @Input()\n  get placeholder(): string { return this._placeholder; }\n  set placeholder(value: string) {\n    this._placeholder = value;\n    this.stateChanges.next();\n  }\n\n  /** Whether the component is required. */\n  @Input()\n  get required(): boolean { return this._required; }\n  set required(value: boolean) {\n    this._required = coerceBooleanProperty(value);\n    this.stateChanges.next();\n  }\n\n  /** Whether the user should be allowed to select multiple options. */\n  @Input()\n  get multiple(): boolean { return this._multiple; }\n  set multiple(value: boolean) {\n    if (this._selectionModel) {\n      throw getMatSelectDynamicMultipleError();\n    }\n\n    this._multiple = coerceBooleanProperty(value);\n  }\n\n  /** Whether to center the active option over the trigger. */\n  @Input()\n  get disableOptionCentering(): boolean { return this._disableOptionCentering; }\n  set disableOptionCentering(value: boolean) {\n    this._disableOptionCentering = coerceBooleanProperty(value);\n  }\n\n  /**\n   * Function to compare the option values with the selected values. The first argument\n   * is a value from an option. The second is a value from the selection. A boolean\n   * should be returned.\n   */\n  @Input()\n  get compareWith() { return this._compareWith; }\n  set compareWith(fn: (o1: any, o2: any) => boolean) {\n    if (typeof fn !== 'function') {\n      throw getMatSelectNonFunctionValueError();\n    }\n    this._compareWith = fn;\n    if (this._selectionModel) {\n      // A different comparator means the selection could change.\n      this._initializeSelection();\n    }\n  }\n\n  /** Value of the select control. */\n  @Input()\n  get value(): any { return this._value; }\n  set value(newValue: any) {\n    if (newValue !== this._value) {\n      this.writeValue(newValue);\n      this._value = newValue;\n    }\n  }\n  private _value: any;\n\n  /** Aria label of the select. If not specified, the placeholder will be used as label. */\n  @Input('aria-label') ariaLabel: string = '';\n\n  /** Input that can be used to specify the `aria-labelledby` attribute. */\n  @Input('aria-labelledby') ariaLabelledby: string;\n\n  /** Object used to control when error messages are shown. */\n  @Input() errorStateMatcher: ErrorStateMatcher;\n\n  /** Time to wait in milliseconds after the last keystroke before moving focus to an item. */\n  @Input()\n  get typeaheadDebounceInterval(): number { return this._typeaheadDebounceInterval; }\n  set typeaheadDebounceInterval(value: number) {\n    this._typeaheadDebounceInterval = coerceNumberProperty(value);\n  }\n  private _typeaheadDebounceInterval: number;\n\n  /**\n   * Function used to sort the values in a select in multiple mode.\n   * Follows the same logic as `Array.prototype.sort`.\n   */\n  @Input() sortComparator: (a: MatOption, b: MatOption, options: MatOption[]) => number;\n\n  /** Unique id of the element. */\n  @Input()\n  get id(): string { return this._id; }\n  set id(value: string) {\n    this._id = value || this._uid;\n    this.stateChanges.next();\n  }\n  private _id: string;\n\n  /** Combined stream of all of the child options' change events. */\n  readonly optionSelectionChanges: Observable<MatOptionSelectionChange> = defer(() => {\n    const options = this.options;\n\n    if (options) {\n      return options.changes.pipe(\n        startWith(options),\n        switchMap(() => merge(...options.map(option => option.onSelectionChange)))\n      );\n    }\n\n    return this._ngZone.onStable\n      .asObservable()\n      .pipe(take(1), switchMap(() => this.optionSelectionChanges));\n  }) as Observable<MatOptionSelectionChange>;\n\n  /** Event emitted when the select panel has been toggled. */\n  @Output() readonly openedChange: EventEmitter<boolean> = new EventEmitter<boolean>();\n\n  /** Event emitted when the select has been opened. */\n  @Output('opened') readonly _openedStream: Observable<void> =\n      this.openedChange.pipe(filter(o => o), map(() => {}));\n\n  /** Event emitted when the select has been closed. */\n  @Output('closed') readonly _closedStream: Observable<void> =\n      this.openedChange.pipe(filter(o => !o), map(() => {}));\n\n   /** Event emitted when the selected value has been changed by the user. */\n  @Output() readonly selectionChange: EventEmitter<MatSelectChange> =\n      new EventEmitter<MatSelectChange>();\n\n  /**\n   * Event that emits whenever the raw value of the select changes. This is here primarily\n   * to facilitate the two-way binding for the `value` input.\n   * @docs-private\n   */\n  @Output() readonly valueChange: EventEmitter<any> = new EventEmitter<any>();\n\n  constructor(\n    private _viewportRuler: ViewportRuler,\n    private _changeDetectorRef: ChangeDetectorRef,\n    private _ngZone: NgZone,\n    _defaultErrorStateMatcher: ErrorStateMatcher,\n    elementRef: ElementRef,\n    @Optional() private _dir: Directionality,\n    @Optional() _parentForm: NgForm,\n    @Optional() _parentFormGroup: FormGroupDirective,\n    @Optional() @Inject(MAT_FORM_FIELD) private _parentFormField: MatFormField,\n    @Self() @Optional() public ngControl: NgControl,\n    @Attribute('tabindex') tabIndex: string,\n    @Inject(MAT_SELECT_SCROLL_STRATEGY) scrollStrategyFactory: any,\n    private _liveAnnouncer: LiveAnnouncer,\n    @Optional() @Inject(MAT_SELECT_CONFIG) defaults?: MatSelectConfig) {\n    super(elementRef, _defaultErrorStateMatcher, _parentForm,\n          _parentFormGroup, ngControl);\n\n    if (this.ngControl) {\n      // Note: we provide the value accessor through here, instead of\n      // the `providers` to avoid running into a circular import.\n      this.ngControl.valueAccessor = this;\n    }\n\n    this._scrollStrategyFactory = scrollStrategyFactory;\n    this._scrollStrategy = this._scrollStrategyFactory();\n    this.tabIndex = parseInt(tabIndex) || 0;\n\n    // Force setter to be called in case id was not specified.\n    this.id = this.id;\n\n    if (defaults) {\n      if (defaults.disableOptionCentering != null) {\n        this.disableOptionCentering = defaults.disableOptionCentering;\n      }\n\n      if (defaults.typeaheadDebounceInterval != null) {\n        this.typeaheadDebounceInterval = defaults.typeaheadDebounceInterval;\n      }\n    }\n  }\n\n  ngOnInit() {\n    this._selectionModel = new SelectionModel<MatOption>(this.multiple);\n    this.stateChanges.next();\n\n    // We need `distinctUntilChanged` here, because some browsers will\n    // fire the animation end event twice for the same animation. See:\n    // https://github.com/angular/angular/issues/24084\n    this._panelDoneAnimatingStream\n      .pipe(distinctUntilChanged(), takeUntil(this._destroy))\n      .subscribe(() => {\n        if (this.panelOpen) {\n          this._scrollTop = 0;\n          this.openedChange.emit(true);\n        } else {\n          this.openedChange.emit(false);\n          this.overlayDir.offsetX = 0;\n          this._changeDetectorRef.markForCheck();\n        }\n      });\n\n    this._viewportRuler.change()\n      .pipe(takeUntil(this._destroy))\n      .subscribe(() => {\n        if (this._panelOpen) {\n          this._triggerRect = this.trigger.nativeElement.getBoundingClientRect();\n          this._changeDetectorRef.markForCheck();\n        }\n      });\n  }\n\n  ngAfterContentInit() {\n    this._initKeyManager();\n\n    this._selectionModel.changed.pipe(takeUntil(this._destroy)).subscribe(event => {\n      event.added.forEach(option => option.select());\n      event.removed.forEach(option => option.deselect());\n    });\n\n    this.options.changes.pipe(startWith(null), takeUntil(this._destroy)).subscribe(() => {\n      this._resetOptions();\n      this._initializeSelection();\n    });\n  }\n\n  ngDoCheck() {\n    if (this.ngControl) {\n      this.updateErrorState();\n    }\n  }\n\n  ngOnChanges(changes: SimpleChanges) {\n    // Updating the disabled state is handled by `mixinDisabled`, but we need to additionally let\n    // the parent form field know to run change detection when the disabled state changes.\n    if (changes['disabled']) {\n      this.stateChanges.next();\n    }\n\n    if (changes['typeaheadDebounceInterval'] && this._keyManager) {\n      this._keyManager.withTypeAhead(this._typeaheadDebounceInterval);\n    }\n  }\n\n  ngOnDestroy() {\n    this._destroy.next();\n    this._destroy.complete();\n    this.stateChanges.complete();\n  }\n\n  /** Toggles the overlay panel open or closed. */\n  toggle(): void {\n    this.panelOpen ? this.close() : this.open();\n  }\n\n  /** Opens the overlay panel. */\n  open(): void {\n    if (this.disabled || !this.options || !this.options.length || this._panelOpen) {\n      return;\n    }\n\n    this._triggerRect = this.trigger.nativeElement.getBoundingClientRect();\n    // Note: The computed font-size will be a string pixel value (e.g. \"16px\").\n    // `parseInt` ignores the trailing 'px' and converts this to a number.\n    this._triggerFontSize = parseInt(getComputedStyle(this.trigger.nativeElement).fontSize || '0');\n\n    this._panelOpen = true;\n    this._keyManager.withHorizontalOrientation(null);\n    this._calculateOverlayPosition();\n    this._highlightCorrectOption();\n    this._changeDetectorRef.markForCheck();\n\n    // Set the font size on the panel element once it exists.\n    this._ngZone.onStable.asObservable().pipe(take(1)).subscribe(() => {\n      if (this._triggerFontSize && this.overlayDir.overlayRef &&\n          this.overlayDir.overlayRef.overlayElement) {\n        this.overlayDir.overlayRef.overlayElement.style.fontSize = `${this._triggerFontSize}px`;\n      }\n    });\n  }\n\n  /** Closes the overlay panel and focuses the host element. */\n  close(): void {\n    if (this._panelOpen) {\n      this._panelOpen = false;\n      this._keyManager.withHorizontalOrientation(this._isRtl() ? 'rtl' : 'ltr');\n      this._changeDetectorRef.markForCheck();\n      this._onTouched();\n    }\n  }\n\n  /**\n   * Sets the select's value. Part of the ControlValueAccessor interface\n   * required to integrate with Angular's core forms API.\n   *\n   * @param value New value to be written to the model.\n   */\n  writeValue(value: any): void {\n    if (this.options) {\n      this._setSelectionByValue(value);\n    }\n  }\n\n  /**\n   * Saves a callback function to be invoked when the select's value\n   * changes from user input. Part of the ControlValueAccessor interface\n   * required to integrate with Angular's core forms API.\n   *\n   * @param fn Callback to be triggered when the value changes.\n   */\n  registerOnChange(fn: (value: any) => void): void {\n    this._onChange = fn;\n  }\n\n  /**\n   * Saves a callback function to be invoked when the select is blurred\n   * by the user. Part of the ControlValueAccessor interface required\n   * to integrate with Angular's core forms API.\n   *\n   * @param fn Callback to be triggered when the component has been touched.\n   */\n  registerOnTouched(fn: () => {}): void {\n    this._onTouched = fn;\n  }\n\n  /**\n   * Disables the select. Part of the ControlValueAccessor interface required\n   * to integrate with Angular's core forms API.\n   *\n   * @param isDisabled Sets whether the component is disabled.\n   */\n  setDisabledState(isDisabled: boolean): void {\n    this.disabled = isDisabled;\n    this._changeDetectorRef.markForCheck();\n    this.stateChanges.next();\n  }\n\n  /** Whether or not the overlay panel is open. */\n  get panelOpen(): boolean {\n    return this._panelOpen;\n  }\n\n  /** The currently selected option. */\n  get selected(): MatOption | MatOption[] {\n    return this.multiple ? this._selectionModel.selected : this._selectionModel.selected[0];\n  }\n\n  /** The value displayed in the trigger. */\n  get triggerValue(): string {\n    if (this.empty) {\n      return '';\n    }\n\n    if (this._multiple) {\n      const selectedOptions = this._selectionModel.selected.map(option => option.viewValue);\n\n      if (this._isRtl()) {\n        selectedOptions.reverse();\n      }\n\n      // TODO(crisbeto): delimiter should be configurable for proper localization.\n      return selectedOptions.join(', ');\n    }\n\n    return this._selectionModel.selected[0].viewValue;\n  }\n\n  /** Whether the element is in RTL mode. */\n  _isRtl(): boolean {\n    return this._dir ? this._dir.value === 'rtl' : false;\n  }\n\n  /** Handles all keydown events on the select. */\n  _handleKeydown(event: KeyboardEvent): void {\n    if (!this.disabled) {\n      this.panelOpen ? this._handleOpenKeydown(event) : this._handleClosedKeydown(event);\n    }\n  }\n\n  /** Handles keyboard events while the select is closed. */\n  private _handleClosedKeydown(event: KeyboardEvent): void {\n    const keyCode = event.keyCode;\n    const isArrowKey = keyCode === DOWN_ARROW || keyCode === UP_ARROW ||\n                       keyCode === LEFT_ARROW || keyCode === RIGHT_ARROW;\n    const isOpenKey = keyCode === ENTER || keyCode === SPACE;\n    const manager = this._keyManager;\n\n    // Open the select on ALT + arrow key to match the native <select>\n    if (!manager.isTyping() && (isOpenKey && !hasModifierKey(event)) ||\n      ((this.multiple || event.altKey) && isArrowKey)) {\n      event.preventDefault(); // prevents the page from scrolling down when pressing space\n      this.open();\n    } else if (!this.multiple) {\n      const previouslySelectedOption = this.selected;\n\n      if (keyCode === HOME || keyCode === END) {\n        keyCode === HOME ? manager.setFirstItemActive() : manager.setLastItemActive();\n        event.preventDefault();\n      } else {\n        manager.onKeydown(event);\n      }\n\n      const selectedOption = this.selected;\n\n      // Since the value has changed, we need to announce it ourselves.\n      if (selectedOption && previouslySelectedOption !== selectedOption) {\n        // We set a duration on the live announcement, because we want the live element to be\n        // cleared after a while so that users can't navigate to it using the arrow keys.\n        this._liveAnnouncer.announce((selectedOption as MatOption).viewValue, 10000);\n      }\n    }\n  }\n\n  /** Handles keyboard events when the selected is open. */\n  private _handleOpenKeydown(event: KeyboardEvent): void {\n    const manager = this._keyManager;\n    const keyCode = event.keyCode;\n    const isArrowKey = keyCode === DOWN_ARROW || keyCode === UP_ARROW;\n    const isTyping = manager.isTyping();\n\n    if (keyCode === HOME || keyCode === END) {\n      event.preventDefault();\n      keyCode === HOME ? manager.setFirstItemActive() : manager.setLastItemActive();\n    } else if (isArrowKey && event.altKey) {\n      // Close the select on ALT + arrow key to match the native <select>\n      event.preventDefault();\n      this.close();\n      // Don't do anything in this case if the user is typing,\n      // because the typing sequence can include the space key.\n    } else if (!isTyping && (keyCode === ENTER || keyCode === SPACE) && manager.activeItem &&\n      !hasModifierKey(event)) {\n      event.preventDefault();\n      manager.activeItem._selectViaInteraction();\n    } else if (!isTyping && this._multiple && keyCode === A && event.ctrlKey) {\n      event.preventDefault();\n      const hasDeselectedOptions = this.options.some(opt => !opt.disabled && !opt.selected);\n\n      this.options.forEach(option => {\n        if (!option.disabled) {\n          hasDeselectedOptions ? option.select() : option.deselect();\n        }\n      });\n    } else {\n      const previouslyFocusedIndex = manager.activeItemIndex;\n\n      manager.onKeydown(event);\n\n      if (this._multiple && isArrowKey && event.shiftKey && manager.activeItem &&\n          manager.activeItemIndex !== previouslyFocusedIndex) {\n        manager.activeItem._selectViaInteraction();\n      }\n    }\n  }\n\n  _onFocus() {\n    if (!this.disabled) {\n      this._focused = true;\n      this.stateChanges.next();\n    }\n  }\n\n  /**\n   * Calls the touched callback only if the panel is closed. Otherwise, the trigger will\n   * \"blur\" to the panel when it opens, causing a false positive.\n   */\n  _onBlur() {\n    this._focused = false;\n\n    if (!this.disabled && !this.panelOpen) {\n      this._onTouched();\n      this._changeDetectorRef.markForCheck();\n      this.stateChanges.next();\n    }\n  }\n\n  /**\n   * Callback that is invoked when the overlay panel has been attached.\n   */\n  _onAttached(): void {\n    this.overlayDir.positionChange.pipe(take(1)).subscribe(() => {\n      this._changeDetectorRef.detectChanges();\n      this._calculateOverlayOffsetX();\n      this.panel.nativeElement.scrollTop = this._scrollTop;\n    });\n  }\n\n  /** Returns the theme to be used on the panel. */\n  _getPanelTheme(): string {\n    return this._parentFormField ? `mat-${this._parentFormField.color}` : '';\n  }\n\n  /** Whether the select has a value. */\n  get empty(): boolean {\n    return !this._selectionModel || this._selectionModel.isEmpty();\n  }\n\n  private _initializeSelection(): void {\n    // Defer setting the value in order to avoid the \"Expression\n    // has changed after it was checked\" errors from Angular.\n    Promise.resolve().then(() => {\n      this._setSelectionByValue(this.ngControl ? this.ngControl.value : this._value);\n      this.stateChanges.next();\n    });\n  }\n\n  /**\n   * Sets the selected option based on a value. If no option can be\n   * found with the designated value, the select trigger is cleared.\n   */\n  private _setSelectionByValue(value: any | any[]): void {\n    if (this.multiple && value) {\n      if (!Array.isArray(value)) {\n        throw getMatSelectNonArrayValueError();\n      }\n\n      this._selectionModel.clear();\n      value.forEach((currentValue: any) => this._selectValue(currentValue));\n      this._sortValues();\n    } else {\n      this._selectionModel.clear();\n      const correspondingOption = this._selectValue(value);\n\n      // Shift focus to the active item. Note that we shouldn't do this in multiple\n      // mode, because we don't know what option the user interacted with last.\n      if (correspondingOption) {\n        this._keyManager.setActiveItem(correspondingOption);\n      } else if (!this.panelOpen) {\n        // Otherwise reset the highlighted option. Note that we only want to do this while\n        // closed, because doing it while open can shift the user's focus unnecessarily.\n        this._keyManager.setActiveItem(-1);\n      }\n    }\n\n    this._changeDetectorRef.markForCheck();\n  }\n\n  /**\n   * Finds and selects and option based on its value.\n   * @returns Option that has the corresponding value.\n   */\n  private _selectValue(value: any): MatOption | undefined {\n    const correspondingOption = this.options.find((option: MatOption) => {\n      try {\n        // Treat null as a special reset value.\n        return option.value != null && this._compareWith(option.value,  value);\n      } catch (error) {\n        if (isDevMode()) {\n          // Notify developers of errors in their comparator.\n          console.warn(error);\n        }\n        return false;\n      }\n    });\n\n    if (correspondingOption) {\n      this._selectionModel.select(correspondingOption);\n    }\n\n    return correspondingOption;\n  }\n\n  /** Sets up a key manager to listen to keyboard events on the overlay panel. */\n  private _initKeyManager() {\n    this._keyManager = new ActiveDescendantKeyManager<MatOption>(this.options)\n      .withTypeAhead(this._typeaheadDebounceInterval)\n      .withVerticalOrientation()\n      .withHorizontalOrientation(this._isRtl() ? 'rtl' : 'ltr')\n      .withAllowedModifierKeys(['shiftKey']);\n\n    this._keyManager.tabOut.pipe(takeUntil(this._destroy)).subscribe(() => {\n      if (this.panelOpen) {\n        // Select the active item when tabbing away. This is consistent with how the native\n        // select behaves. Note that we only want to do this in single selection mode.\n        if (!this.multiple && this._keyManager.activeItem) {\n          this._keyManager.activeItem._selectViaInteraction();\n        }\n\n        // Restore focus to the trigger before closing. Ensures that the focus\n        // position won't be lost if the user got focus into the overlay.\n        this.focus();\n        this.close();\n      }\n    });\n\n    this._keyManager.change.pipe(takeUntil(this._destroy)).subscribe(() => {\n      if (this._panelOpen && this.panel) {\n        this._scrollActiveOptionIntoView();\n      } else if (!this._panelOpen && !this.multiple && this._keyManager.activeItem) {\n        this._keyManager.activeItem._selectViaInteraction();\n      }\n    });\n  }\n\n  /** Drops current option subscriptions and IDs and resets from scratch. */\n  private _resetOptions(): void {\n    const changedOrDestroyed = merge(this.options.changes, this._destroy);\n\n    this.optionSelectionChanges.pipe(takeUntil(changedOrDestroyed)).subscribe(event => {\n      this._onSelect(event.source, event.isUserInput);\n\n      if (event.isUserInput && !this.multiple && this._panelOpen) {\n        this.close();\n        this.focus();\n      }\n    });\n\n    // Listen to changes in the internal state of the options and react accordingly.\n    // Handles cases like the labels of the selected options changing.\n    merge(...this.options.map(option => option._stateChanges))\n      .pipe(takeUntil(changedOrDestroyed))\n      .subscribe(() => {\n        this._changeDetectorRef.markForCheck();\n        this.stateChanges.next();\n      });\n\n    this._setOptionIds();\n  }\n\n  /** Invoked when an option is clicked. */\n  private _onSelect(option: MatOption, isUserInput: boolean): void {\n    const wasSelected = this._selectionModel.isSelected(option);\n\n    if (option.value == null && !this._multiple) {\n      option.deselect();\n      this._selectionModel.clear();\n      this._propagateChanges(option.value);\n    } else {\n      if (wasSelected !== option.selected) {\n        option.selected ? this._selectionModel.select(option) :\n                          this._selectionModel.deselect(option);\n      }\n\n      if (isUserInput) {\n        this._keyManager.setActiveItem(option);\n      }\n\n      if (this.multiple) {\n        this._sortValues();\n\n        if (isUserInput) {\n          // In case the user selected the option with their mouse, we\n          // want to restore focus back to the trigger, in order to\n          // prevent the select keyboard controls from clashing with\n          // the ones from `mat-option`.\n          this.focus();\n        }\n      }\n    }\n\n    if (wasSelected !== this._selectionModel.isSelected(option)) {\n      this._propagateChanges();\n    }\n\n    this.stateChanges.next();\n  }\n\n  /** Sorts the selected values in the selected based on their order in the panel. */\n  private _sortValues() {\n    if (this.multiple) {\n      const options = this.options.toArray();\n\n      this._selectionModel.sort((a, b) => {\n        return this.sortComparator ? this.sortComparator(a, b, options) :\n                                     options.indexOf(a) - options.indexOf(b);\n      });\n      this.stateChanges.next();\n    }\n  }\n\n  /** Emits change event to set the model value. */\n  private _propagateChanges(fallbackValue?: any): void {\n    let valueToEmit: any = null;\n\n    if (this.multiple) {\n      valueToEmit = (this.selected as MatOption[]).map(option => option.value);\n    } else {\n      valueToEmit = this.selected ? (this.selected as MatOption).value : fallbackValue;\n    }\n\n    this._value = valueToEmit;\n    this.valueChange.emit(valueToEmit);\n    this._onChange(valueToEmit);\n    this.selectionChange.emit(new MatSelectChange(this, valueToEmit));\n    this._changeDetectorRef.markForCheck();\n  }\n\n  /** Records option IDs to pass to the aria-owns property. */\n  private _setOptionIds() {\n    this._optionIds = this.options.map(option => option.id).join(' ');\n  }\n\n  /**\n   * Highlights the selected item. If no option is selected, it will highlight\n   * the first item instead.\n   */\n  private _highlightCorrectOption(): void {\n    if (this._keyManager) {\n      if (this.empty) {\n        this._keyManager.setFirstItemActive();\n      } else {\n        this._keyManager.setActiveItem(this._selectionModel.selected[0]);\n      }\n    }\n  }\n\n  /** Scrolls the active option into view. */\n  private _scrollActiveOptionIntoView(): void {\n    const activeOptionIndex = this._keyManager.activeItemIndex || 0;\n    const labelCount = _countGroupLabelsBeforeOption(activeOptionIndex, this.options,\n        this.optionGroups);\n\n    this.panel.nativeElement.scrollTop = _getOptionScrollPosition(\n      activeOptionIndex + labelCount,\n      this._getItemHeight(),\n      this.panel.nativeElement.scrollTop,\n      SELECT_PANEL_MAX_HEIGHT\n    );\n  }\n\n  /** Focuses the select element. */\n  focus(options?: FocusOptions): void {\n    this._elementRef.nativeElement.focus(options);\n  }\n\n  /** Gets the index of the provided option in the option list. */\n  private _getOptionIndex(option: MatOption): number | undefined {\n    return this.options.reduce((result: number | undefined, current: MatOption, index: number) => {\n      if (result !== undefined) {\n        return result;\n      }\n\n      return option === current ? index : undefined;\n    }, undefined);\n  }\n\n  /** Calculates the scroll position and x- and y-offsets of the overlay panel. */\n  private _calculateOverlayPosition(): void {\n    const itemHeight = this._getItemHeight();\n    const items = this._getItemCount();\n    const panelHeight = Math.min(items * itemHeight, SELECT_PANEL_MAX_HEIGHT);\n    const scrollContainerHeight = items * itemHeight;\n\n    // The farthest the panel can be scrolled before it hits the bottom\n    const maxScroll = scrollContainerHeight - panelHeight;\n\n    // If no value is selected we open the popup to the first item.\n    let selectedOptionOffset =\n        this.empty ? 0 : this._getOptionIndex(this._selectionModel.selected[0])!;\n\n    selectedOptionOffset += _countGroupLabelsBeforeOption(selectedOptionOffset, this.options,\n        this.optionGroups);\n\n    // We must maintain a scroll buffer so the selected option will be scrolled to the\n    // center of the overlay panel rather than the top.\n    const scrollBuffer = panelHeight / 2;\n    this._scrollTop = this._calculateOverlayScroll(selectedOptionOffset, scrollBuffer, maxScroll);\n    this._offsetY = this._calculateOverlayOffsetY(selectedOptionOffset, scrollBuffer, maxScroll);\n\n    this._checkOverlayWithinViewport(maxScroll);\n  }\n\n  /**\n   * Calculates the scroll position of the select's overlay panel.\n   *\n   * Attempts to center the selected option in the panel. If the option is\n   * too high or too low in the panel to be scrolled to the center, it clamps the\n   * scroll position to the min or max scroll positions respectively.\n   */\n  _calculateOverlayScroll(selectedIndex: number, scrollBuffer: number,\n                          maxScroll: number): number {\n    const itemHeight = this._getItemHeight();\n    const optionOffsetFromScrollTop = itemHeight * selectedIndex;\n    const halfOptionHeight = itemHeight / 2;\n\n    // Starts at the optionOffsetFromScrollTop, which scrolls the option to the top of the\n    // scroll container, then subtracts the scroll buffer to scroll the option down to\n    // the center of the overlay panel. Half the option height must be re-added to the\n    // scrollTop so the option is centered based on its middle, not its top edge.\n    const optimalScrollPosition = optionOffsetFromScrollTop - scrollBuffer + halfOptionHeight;\n    return Math.min(Math.max(0, optimalScrollPosition), maxScroll);\n  }\n\n  /** Returns the aria-label of the select component. */\n  _getAriaLabel(): string | null {\n    // If an ariaLabelledby value has been set by the consumer, the select should not overwrite the\n    // `aria-labelledby` value by setting the ariaLabel to the placeholder.\n    return this.ariaLabelledby ? null : this.ariaLabel || this.placeholder;\n  }\n\n  /** Returns the aria-labelledby of the select component. */\n  _getAriaLabelledby(): string | null {\n    if (this.ariaLabelledby) {\n      return this.ariaLabelledby;\n    }\n\n    // Note: we use `_getAriaLabel` here, because we want to check whether there's a\n    // computed label. `this.ariaLabel` is only the user-specified label.\n    if (!this._parentFormField || !this._parentFormField._hasFloatingLabel() ||\n      this._getAriaLabel()) {\n      return null;\n    }\n\n    return this._parentFormField._labelId || null;\n  }\n\n  /** Determines the `aria-activedescendant` to be set on the host. */\n  _getAriaActiveDescendant(): string | null {\n    if (this.panelOpen && this._keyManager && this._keyManager.activeItem) {\n      return this._keyManager.activeItem.id;\n    }\n\n    return null;\n  }\n\n  /**\n   * Sets the x-offset of the overlay panel in relation to the trigger's top start corner.\n   * This must be adjusted to align the selected option text over the trigger text when\n   * the panel opens. Will change based on LTR or RTL text direction. Note that the offset\n   * can't be calculated until the panel has been attached, because we need to know the\n   * content width in order to constrain the panel within the viewport.\n   */\n  private _calculateOverlayOffsetX(): void {\n    const overlayRect = this.overlayDir.overlayRef.overlayElement.getBoundingClientRect();\n    const viewportSize = this._viewportRuler.getViewportSize();\n    const isRtl = this._isRtl();\n    const paddingWidth = this.multiple ? SELECT_MULTIPLE_PANEL_PADDING_X + SELECT_PANEL_PADDING_X :\n                                         SELECT_PANEL_PADDING_X * 2;\n    let offsetX: number;\n\n    // Adjust the offset, depending on the option padding.\n    if (this.multiple) {\n      offsetX = SELECT_MULTIPLE_PANEL_PADDING_X;\n    } else {\n      let selected = this._selectionModel.selected[0] || this.options.first;\n      offsetX = selected && selected.group ? SELECT_PANEL_INDENT_PADDING_X : SELECT_PANEL_PADDING_X;\n    }\n\n    // Invert the offset in LTR.\n    if (!isRtl) {\n      offsetX *= -1;\n    }\n\n    // Determine how much the select overflows on each side.\n    const leftOverflow = 0 - (overlayRect.left + offsetX - (isRtl ? paddingWidth : 0));\n    const rightOverflow = overlayRect.right + offsetX - viewportSize.width\n                          + (isRtl ? 0 : paddingWidth);\n\n    // If the element overflows on either side, reduce the offset to allow it to fit.\n    if (leftOverflow > 0) {\n      offsetX += leftOverflow + SELECT_PANEL_VIEWPORT_PADDING;\n    } else if (rightOverflow > 0) {\n      offsetX -= rightOverflow + SELECT_PANEL_VIEWPORT_PADDING;\n    }\n\n    // Set the offset directly in order to avoid having to go through change detection and\n    // potentially triggering \"changed after it was checked\" errors. Round the value to avoid\n    // blurry content in some browsers.\n    this.overlayDir.offsetX = Math.round(offsetX);\n    this.overlayDir.overlayRef.updatePosition();\n  }\n\n  /**\n   * Calculates the y-offset of the select's overlay panel in relation to the\n   * top start corner of the trigger. It has to be adjusted in order for the\n   * selected option to be aligned over the trigger when the panel opens.\n   */\n  private _calculateOverlayOffsetY(selectedIndex: number, scrollBuffer: number,\n                                  maxScroll: number): number {\n    const itemHeight = this._getItemHeight();\n    const optionHeightAdjustment = (itemHeight - this._triggerRect.height) / 2;\n    const maxOptionsDisplayed = Math.floor(SELECT_PANEL_MAX_HEIGHT / itemHeight);\n    let optionOffsetFromPanelTop: number;\n\n    // Disable offset if requested by user by returning 0 as value to offset\n    if (this._disableOptionCentering) {\n      return 0;\n    }\n\n    if (this._scrollTop === 0) {\n      optionOffsetFromPanelTop = selectedIndex * itemHeight;\n    } else if (this._scrollTop === maxScroll) {\n      const firstDisplayedIndex = this._getItemCount() - maxOptionsDisplayed;\n      const selectedDisplayIndex = selectedIndex - firstDisplayedIndex;\n\n      // The first item is partially out of the viewport. Therefore we need to calculate what\n      // portion of it is shown in the viewport and account for it in our offset.\n      let partialItemHeight =\n          itemHeight - (this._getItemCount() * itemHeight - SELECT_PANEL_MAX_HEIGHT) % itemHeight;\n\n      // Because the panel height is longer than the height of the options alone,\n      // there is always extra padding at the top or bottom of the panel. When\n      // scrolled to the very bottom, this padding is at the top of the panel and\n      // must be added to the offset.\n      optionOffsetFromPanelTop = selectedDisplayIndex * itemHeight + partialItemHeight;\n    } else {\n      // If the option was scrolled to the middle of the panel using a scroll buffer,\n      // its offset will be the scroll buffer minus the half height that was added to\n      // center it.\n      optionOffsetFromPanelTop = scrollBuffer - itemHeight / 2;\n    }\n\n    // The final offset is the option's offset from the top, adjusted for the height difference,\n    // multiplied by -1 to ensure that the overlay moves in the correct direction up the page.\n    // The value is rounded to prevent some browsers from blurring the content.\n    return Math.round(optionOffsetFromPanelTop * -1 - optionHeightAdjustment);\n  }\n\n  /**\n   * Checks that the attempted overlay position will fit within the viewport.\n   * If it will not fit, tries to adjust the scroll position and the associated\n   * y-offset so the panel can open fully on-screen. If it still won't fit,\n   * sets the offset back to 0 to allow the fallback position to take over.\n   */\n  private _checkOverlayWithinViewport(maxScroll: number): void {\n    const itemHeight = this._getItemHeight();\n    const viewportSize = this._viewportRuler.getViewportSize();\n\n    const topSpaceAvailable = this._triggerRect.top - SELECT_PANEL_VIEWPORT_PADDING;\n    const bottomSpaceAvailable =\n        viewportSize.height - this._triggerRect.bottom - SELECT_PANEL_VIEWPORT_PADDING;\n\n    const panelHeightTop = Math.abs(this._offsetY);\n    const totalPanelHeight =\n        Math.min(this._getItemCount() * itemHeight, SELECT_PANEL_MAX_HEIGHT);\n    const panelHeightBottom = totalPanelHeight - panelHeightTop - this._triggerRect.height;\n\n    if (panelHeightBottom > bottomSpaceAvailable) {\n      this._adjustPanelUp(panelHeightBottom, bottomSpaceAvailable);\n    } else if (panelHeightTop > topSpaceAvailable) {\n     this._adjustPanelDown(panelHeightTop, topSpaceAvailable, maxScroll);\n    } else {\n      this._transformOrigin = this._getOriginBasedOnOption();\n    }\n  }\n\n  /** Adjusts the overlay panel up to fit in the viewport. */\n  private _adjustPanelUp(panelHeightBottom: number, bottomSpaceAvailable: number) {\n    // Browsers ignore fractional scroll offsets, so we need to round.\n    const distanceBelowViewport = Math.round(panelHeightBottom - bottomSpaceAvailable);\n\n    // Scrolls the panel up by the distance it was extending past the boundary, then\n    // adjusts the offset by that amount to move the panel up into the viewport.\n    this._scrollTop -= distanceBelowViewport;\n    this._offsetY -= distanceBelowViewport;\n    this._transformOrigin = this._getOriginBasedOnOption();\n\n    // If the panel is scrolled to the very top, it won't be able to fit the panel\n    // by scrolling, so set the offset to 0 to allow the fallback position to take\n    // effect.\n    if (this._scrollTop <= 0) {\n      this._scrollTop = 0;\n      this._offsetY = 0;\n      this._transformOrigin = `50% bottom 0px`;\n    }\n  }\n\n  /** Adjusts the overlay panel down to fit in the viewport. */\n  private _adjustPanelDown(panelHeightTop: number, topSpaceAvailable: number,\n                           maxScroll: number) {\n    // Browsers ignore fractional scroll offsets, so we need to round.\n    const distanceAboveViewport = Math.round(panelHeightTop - topSpaceAvailable);\n\n    // Scrolls the panel down by the distance it was extending past the boundary, then\n    // adjusts the offset by that amount to move the panel down into the viewport.\n    this._scrollTop += distanceAboveViewport;\n    this._offsetY += distanceAboveViewport;\n    this._transformOrigin = this._getOriginBasedOnOption();\n\n    // If the panel is scrolled to the very bottom, it won't be able to fit the\n    // panel by scrolling, so set the offset to 0 to allow the fallback position\n    // to take effect.\n    if (this._scrollTop >= maxScroll) {\n      this._scrollTop = maxScroll;\n      this._offsetY = 0;\n      this._transformOrigin = `50% top 0px`;\n      return;\n    }\n  }\n\n  /** Sets the transform origin point based on the selected option. */\n  private _getOriginBasedOnOption(): string {\n    const itemHeight = this._getItemHeight();\n    const optionHeightAdjustment = (itemHeight - this._triggerRect.height) / 2;\n    const originY = Math.abs(this._offsetY) - optionHeightAdjustment + itemHeight / 2;\n    return `50% ${originY}px 0px`;\n  }\n\n  /** Calculates the amount of items in the select. This includes options and group labels. */\n  private _getItemCount(): number {\n    return this.options.length + this.optionGroups.length;\n  }\n\n  /** Calculates the height of the select's options. */\n  private _getItemHeight(): number {\n    return this._triggerFontSize * SELECT_ITEM_HEIGHT_EM;\n  }\n\n  /**\n   * Implemented as part of MatFormFieldControl.\n   * @docs-private\n   */\n  setDescribedByIds(ids: string[]) {\n    this._ariaDescribedby = ids.join(' ');\n  }\n\n  /**\n   * Implemented as part of MatFormFieldControl.\n   * @docs-private\n   */\n  onContainerClick() {\n    this.focus();\n    this.open();\n  }\n\n  /**\n   * Implemented as part of MatFormFieldControl.\n   * @docs-private\n   */\n  get shouldLabelFloat(): boolean {\n    return this._panelOpen || !this.empty;\n  }\n\n  static ngAcceptInputType_required: BooleanInput;\n  static ngAcceptInputType_multiple: BooleanInput;\n  static ngAcceptInputType_disableOptionCentering: BooleanInput;\n  static ngAcceptInputType_typeaheadDebounceInterval: NumberInput;\n  static ngAcceptInputType_disabled: BooleanInput;\n  static ngAcceptInputType_disableRipple: BooleanInput;\n}\n"]}
|
|
950
|
+
/** Adjusts the overlay panel down to fit in the viewport. */
|
|
951
|
+
_adjustPanelDown(panelHeightTop, topSpaceAvailable, maxScroll) {
|
|
952
|
+
// Browsers ignore fractional scroll offsets, so we need to round.
|
|
953
|
+
const distanceAboveViewport = Math.round(panelHeightTop - topSpaceAvailable);
|
|
954
|
+
// Scrolls the panel down by the distance it was extending past the boundary, then
|
|
955
|
+
// adjusts the offset by that amount to move the panel down into the viewport.
|
|
956
|
+
this._scrollTop += distanceAboveViewport;
|
|
957
|
+
this._offsetY += distanceAboveViewport;
|
|
958
|
+
this._transformOrigin = this._getOriginBasedOnOption();
|
|
959
|
+
// If the panel is scrolled to the very bottom, it won't be able to fit the
|
|
960
|
+
// panel by scrolling, so set the offset to 0 to allow the fallback position
|
|
961
|
+
// to take effect.
|
|
962
|
+
if (this._scrollTop >= maxScroll) {
|
|
963
|
+
this._scrollTop = maxScroll;
|
|
964
|
+
this._offsetY = 0;
|
|
965
|
+
this._transformOrigin = `50% top 0px`;
|
|
966
|
+
return;
|
|
967
|
+
}
|
|
968
|
+
}
|
|
969
|
+
/** Sets the transform origin point based on the selected option. */
|
|
970
|
+
_getOriginBasedOnOption() {
|
|
971
|
+
const itemHeight = this._getItemHeight();
|
|
972
|
+
const optionHeightAdjustment = (itemHeight - this._triggerRect.height) / 2;
|
|
973
|
+
const originY = Math.abs(this._offsetY) - optionHeightAdjustment + itemHeight / 2;
|
|
974
|
+
return `50% ${originY}px 0px`;
|
|
975
|
+
}
|
|
976
|
+
/** Calculates the amount of items in the select. This includes options and group labels. */
|
|
977
|
+
_getItemCount() {
|
|
978
|
+
return this.options.length + this.optionGroups.length;
|
|
979
|
+
}
|
|
980
|
+
/** Calculates the height of the select's options. */
|
|
981
|
+
_getItemHeight() {
|
|
982
|
+
return this._triggerFontSize * SELECT_ITEM_HEIGHT_EM;
|
|
983
|
+
}
|
|
984
|
+
/**
|
|
985
|
+
* Implemented as part of MatFormFieldControl.
|
|
986
|
+
* @docs-private
|
|
987
|
+
*/
|
|
988
|
+
setDescribedByIds(ids) {
|
|
989
|
+
this._ariaDescribedby = ids.join(' ');
|
|
990
|
+
}
|
|
991
|
+
/**
|
|
992
|
+
* Implemented as part of MatFormFieldControl.
|
|
993
|
+
* @docs-private
|
|
994
|
+
*/
|
|
995
|
+
onContainerClick() {
|
|
996
|
+
this.focus();
|
|
997
|
+
this.open();
|
|
998
|
+
}
|
|
999
|
+
/**
|
|
1000
|
+
* Implemented as part of MatFormFieldControl.
|
|
1001
|
+
* @docs-private
|
|
1002
|
+
*/
|
|
1003
|
+
get shouldLabelFloat() {
|
|
1004
|
+
return this._panelOpen || !this.empty;
|
|
1005
|
+
}
|
|
1006
|
+
}
|
|
1007
|
+
MatSelect.decorators = [
|
|
1008
|
+
{ type: Component, args: [{
|
|
1009
|
+
selector: 'mat-select',
|
|
1010
|
+
exportAs: 'matSelect',
|
|
1011
|
+
template: "<div cdk-overlay-origin\n class=\"mat-select-trigger\"\n aria-hidden=\"true\"\n (click)=\"toggle()\"\n #origin=\"cdkOverlayOrigin\"\n #trigger>\n <div class=\"mat-select-value\" [ngSwitch]=\"empty\">\n <span class=\"mat-select-placeholder\" *ngSwitchCase=\"true\">{{placeholder || '\\u00A0'}}</span>\n <span class=\"mat-select-value-text\" *ngSwitchCase=\"false\" [ngSwitch]=\"!!customTrigger\">\n <span *ngSwitchDefault>{{triggerValue || '\\u00A0'}}</span>\n <ng-content select=\"mat-select-trigger\" *ngSwitchCase=\"true\"></ng-content>\n </span>\n </div>\n\n <div class=\"mat-select-arrow-wrapper\"><div class=\"mat-select-arrow\"></div></div>\n</div>\n\n<ng-template\n cdk-connected-overlay\n cdkConnectedOverlayLockPosition\n cdkConnectedOverlayHasBackdrop\n cdkConnectedOverlayBackdropClass=\"cdk-overlay-transparent-backdrop\"\n [cdkConnectedOverlayScrollStrategy]=\"_scrollStrategy\"\n [cdkConnectedOverlayOrigin]=\"origin\"\n [cdkConnectedOverlayOpen]=\"panelOpen\"\n [cdkConnectedOverlayPositions]=\"_positions\"\n [cdkConnectedOverlayMinWidth]=\"_triggerRect?.width\"\n [cdkConnectedOverlayOffsetY]=\"_offsetY\"\n (backdropClick)=\"close()\"\n (attach)=\"_onAttached()\"\n (detach)=\"close()\">\n <div class=\"mat-select-panel-wrap\" [@transformPanelWrap]>\n <div\n #panel\n [attr.id]=\"id + '-panel'\"\n class=\"mat-select-panel {{ _getPanelTheme() }}\"\n [ngClass]=\"panelClass\"\n [@transformPanel]=\"multiple ? 'showing-multiple' : 'showing'\"\n (@transformPanel.done)=\"_panelDoneAnimatingStream.next($event.toState)\"\n [style.transformOrigin]=\"_transformOrigin\"\n [style.font-size.px]=\"_triggerFontSize\"\n (keydown)=\"_handleKeydown($event)\">\n <ng-content></ng-content>\n </div>\n </div>\n</ng-template>\n",
|
|
1012
|
+
inputs: ['disabled', 'disableRipple', 'tabIndex'],
|
|
1013
|
+
encapsulation: ViewEncapsulation.None,
|
|
1014
|
+
changeDetection: ChangeDetectionStrategy.OnPush,
|
|
1015
|
+
host: {
|
|
1016
|
+
'role': 'listbox',
|
|
1017
|
+
'[attr.id]': 'id',
|
|
1018
|
+
'[attr.tabindex]': 'tabIndex',
|
|
1019
|
+
'[attr.aria-label]': '_getAriaLabel()',
|
|
1020
|
+
'[attr.aria-labelledby]': '_getAriaLabelledby()',
|
|
1021
|
+
'[attr.aria-required]': 'required.toString()',
|
|
1022
|
+
'[attr.aria-disabled]': 'disabled.toString()',
|
|
1023
|
+
'[attr.aria-invalid]': 'errorState',
|
|
1024
|
+
'[attr.aria-owns]': 'panelOpen ? _optionIds : null',
|
|
1025
|
+
'[attr.aria-multiselectable]': 'multiple',
|
|
1026
|
+
'[attr.aria-describedby]': '_ariaDescribedby || null',
|
|
1027
|
+
'[attr.aria-activedescendant]': '_getAriaActiveDescendant()',
|
|
1028
|
+
'[class.mat-select-disabled]': 'disabled',
|
|
1029
|
+
'[class.mat-select-invalid]': 'errorState',
|
|
1030
|
+
'[class.mat-select-required]': 'required',
|
|
1031
|
+
'[class.mat-select-empty]': 'empty',
|
|
1032
|
+
'class': 'mat-select',
|
|
1033
|
+
'(keydown)': '_handleKeydown($event)',
|
|
1034
|
+
'(focus)': '_onFocus()',
|
|
1035
|
+
'(blur)': '_onBlur()',
|
|
1036
|
+
},
|
|
1037
|
+
animations: [
|
|
1038
|
+
matSelectAnimations.transformPanelWrap,
|
|
1039
|
+
matSelectAnimations.transformPanel
|
|
1040
|
+
],
|
|
1041
|
+
providers: [
|
|
1042
|
+
{ provide: MatFormFieldControl, useExisting: MatSelect },
|
|
1043
|
+
{ provide: MAT_OPTION_PARENT_COMPONENT, useExisting: MatSelect }
|
|
1044
|
+
],
|
|
1045
|
+
styles: [".mat-select{display:inline-block;width:100%;outline:none}.mat-select-trigger{display:inline-table;cursor:pointer;position:relative;box-sizing:border-box}.mat-select-disabled .mat-select-trigger{-webkit-user-select:none;-moz-user-select:none;-ms-user-select:none;user-select:none;cursor:default}.mat-select-value{display:table-cell;max-width:0;width:100%;overflow:hidden;text-overflow:ellipsis;white-space:nowrap}.mat-select-value-text{white-space:nowrap;overflow:hidden;text-overflow:ellipsis}.mat-select-arrow-wrapper{display:table-cell;vertical-align:middle}.mat-form-field-appearance-fill .mat-select-arrow-wrapper{transform:translateY(-50%)}.mat-form-field-appearance-outline .mat-select-arrow-wrapper{transform:translateY(-25%)}.mat-form-field-appearance-standard.mat-form-field-has-label .mat-select:not(.mat-select-empty) .mat-select-arrow-wrapper{transform:translateY(-50%)}.mat-form-field-appearance-standard .mat-select.mat-select-empty .mat-select-arrow-wrapper{transition:transform 400ms cubic-bezier(0.25, 0.8, 0.25, 1)}._mat-animation-noopable.mat-form-field-appearance-standard .mat-select.mat-select-empty .mat-select-arrow-wrapper{transition:none}.mat-select-arrow{width:0;height:0;border-left:5px solid transparent;border-right:5px solid transparent;border-top:5px solid;margin:0 4px}.mat-select-panel-wrap{flex-basis:100%}.mat-select-panel{min-width:112px;max-width:280px;overflow:auto;-webkit-overflow-scrolling:touch;padding-top:0;padding-bottom:0;max-height:256px;min-width:100%;border-radius:4px}.cdk-high-contrast-active .mat-select-panel{outline:solid 1px}.mat-select-panel .mat-optgroup-label,.mat-select-panel .mat-option{font-size:inherit;line-height:3em;height:3em}.mat-form-field-type-mat-select:not(.mat-form-field-disabled) .mat-form-field-flex{cursor:pointer}.mat-form-field-type-mat-select .mat-form-field-label{width:calc(100% - 18px)}.mat-select-placeholder{transition:color 400ms 133.3333333333ms cubic-bezier(0.25, 0.8, 0.25, 1)}._mat-animation-noopable .mat-select-placeholder{transition:none}.mat-form-field-hide-placeholder .mat-select-placeholder{color:transparent;-webkit-text-fill-color:transparent;transition:none;display:block}\n"]
|
|
1046
|
+
},] }
|
|
1047
|
+
];
|
|
1048
|
+
MatSelect.ctorParameters = () => [
|
|
1049
|
+
{ type: ViewportRuler },
|
|
1050
|
+
{ type: ChangeDetectorRef },
|
|
1051
|
+
{ type: NgZone },
|
|
1052
|
+
{ type: ErrorStateMatcher },
|
|
1053
|
+
{ type: ElementRef },
|
|
1054
|
+
{ type: Directionality, decorators: [{ type: Optional }] },
|
|
1055
|
+
{ type: NgForm, decorators: [{ type: Optional }] },
|
|
1056
|
+
{ type: FormGroupDirective, decorators: [{ type: Optional }] },
|
|
1057
|
+
{ type: MatFormField, decorators: [{ type: Optional }, { type: Inject, args: [MAT_FORM_FIELD,] }] },
|
|
1058
|
+
{ type: NgControl, decorators: [{ type: Self }, { type: Optional }] },
|
|
1059
|
+
{ type: String, decorators: [{ type: Attribute, args: ['tabindex',] }] },
|
|
1060
|
+
{ type: undefined, decorators: [{ type: Inject, args: [MAT_SELECT_SCROLL_STRATEGY,] }] },
|
|
1061
|
+
{ type: LiveAnnouncer },
|
|
1062
|
+
{ type: undefined, decorators: [{ type: Optional }, { type: Inject, args: [MAT_SELECT_CONFIG,] }] }
|
|
1063
|
+
];
|
|
1064
|
+
MatSelect.propDecorators = {
|
|
1065
|
+
trigger: [{ type: ViewChild, args: ['trigger',] }],
|
|
1066
|
+
panel: [{ type: ViewChild, args: ['panel',] }],
|
|
1067
|
+
overlayDir: [{ type: ViewChild, args: [CdkConnectedOverlay,] }],
|
|
1068
|
+
options: [{ type: ContentChildren, args: [MatOption, { descendants: true },] }],
|
|
1069
|
+
optionGroups: [{ type: ContentChildren, args: [MAT_OPTGROUP, { descendants: true },] }],
|
|
1070
|
+
panelClass: [{ type: Input }],
|
|
1071
|
+
customTrigger: [{ type: ContentChild, args: [MAT_SELECT_TRIGGER,] }],
|
|
1072
|
+
placeholder: [{ type: Input }],
|
|
1073
|
+
required: [{ type: Input }],
|
|
1074
|
+
multiple: [{ type: Input }],
|
|
1075
|
+
disableOptionCentering: [{ type: Input }],
|
|
1076
|
+
compareWith: [{ type: Input }],
|
|
1077
|
+
value: [{ type: Input }],
|
|
1078
|
+
ariaLabel: [{ type: Input, args: ['aria-label',] }],
|
|
1079
|
+
ariaLabelledby: [{ type: Input, args: ['aria-labelledby',] }],
|
|
1080
|
+
errorStateMatcher: [{ type: Input }],
|
|
1081
|
+
typeaheadDebounceInterval: [{ type: Input }],
|
|
1082
|
+
sortComparator: [{ type: Input }],
|
|
1083
|
+
id: [{ type: Input }],
|
|
1084
|
+
openedChange: [{ type: Output }],
|
|
1085
|
+
_openedStream: [{ type: Output, args: ['opened',] }],
|
|
1086
|
+
_closedStream: [{ type: Output, args: ['closed',] }],
|
|
1087
|
+
selectionChange: [{ type: Output }],
|
|
1088
|
+
valueChange: [{ type: Output }]
|
|
1089
|
+
};
|
|
1090
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"select.js","sourceRoot":"","sources":["../../../../../../src/material/select/select.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * @license\n * Copyright Google LLC All Rights Reserved.\n *\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://angular.io/license\n */\n\nimport {ActiveDescendantKeyManager, LiveAnnouncer} from '@angular/cdk/a11y';\nimport {Directionality} from '@angular/cdk/bidi';\nimport {\n  BooleanInput,\n  coerceBooleanProperty,\n  coerceNumberProperty,\n  NumberInput\n} from '@angular/cdk/coercion';\nimport {SelectionModel} from '@angular/cdk/collections';\nimport {\n  A,\n  DOWN_ARROW,\n  END,\n  ENTER,\n  hasModifierKey,\n  HOME,\n  LEFT_ARROW,\n  RIGHT_ARROW,\n  SPACE,\n  UP_ARROW,\n} from '@angular/cdk/keycodes';\nimport {\n  CdkConnectedOverlay,\n  ConnectedPosition,\n  Overlay,\n  ScrollStrategy,\n} from '@angular/cdk/overlay';\nimport {ViewportRuler} from '@angular/cdk/scrolling';\nimport {\n  AfterContentInit,\n  Attribute,\n  ChangeDetectionStrategy,\n  ChangeDetectorRef,\n  Component,\n  ContentChild,\n  ContentChildren,\n  Directive,\n  DoCheck,\n  ElementRef,\n  EventEmitter,\n  Inject,\n  InjectionToken,\n  Input,\n  isDevMode,\n  NgZone,\n  OnChanges,\n  OnDestroy,\n  OnInit,\n  Optional,\n  Output,\n  QueryList,\n  Self,\n  SimpleChanges,\n  ViewChild,\n  ViewEncapsulation,\n} from '@angular/core';\nimport {ControlValueAccessor, FormGroupDirective, NgControl, NgForm} from '@angular/forms';\nimport {\n  _countGroupLabelsBeforeOption,\n  _getOptionScrollPosition,\n  CanDisable,\n  CanDisableCtor,\n  CanDisableRipple,\n  CanDisableRippleCtor,\n  CanUpdateErrorState,\n  CanUpdateErrorStateCtor,\n  ErrorStateMatcher,\n  HasTabIndex,\n  HasTabIndexCtor,\n  MAT_OPTGROUP,\n  MAT_OPTION_PARENT_COMPONENT,\n  MatOptgroup,\n  MatOption,\n  MatOptionSelectionChange,\n  mixinDisabled,\n  mixinDisableRipple,\n  mixinErrorState,\n  mixinTabIndex,\n} from '@angular/material/core';\nimport {MAT_FORM_FIELD, MatFormField, MatFormFieldControl} from '@angular/material/form-field';\nimport {defer, merge, Observable, Subject} from 'rxjs';\nimport {\n  distinctUntilChanged,\n  filter,\n  map,\n  startWith,\n  switchMap,\n  take,\n  takeUntil,\n} from 'rxjs/operators';\nimport {matSelectAnimations} from './select-animations';\nimport {\n  getMatSelectDynamicMultipleError,\n  getMatSelectNonArrayValueError,\n  getMatSelectNonFunctionValueError,\n} from './select-errors';\n\n\nlet nextUniqueId = 0;\n\n/**\n * The following style constants are necessary to save here in order\n * to properly calculate the alignment of the selected option over\n * the trigger element.\n */\n\n/** The max height of the select's overlay panel */\nexport const SELECT_PANEL_MAX_HEIGHT = 256;\n\n/** The panel's padding on the x-axis */\nexport const SELECT_PANEL_PADDING_X = 16;\n\n/** The panel's x axis padding if it is indented (e.g. there is an option group). */\nexport const SELECT_PANEL_INDENT_PADDING_X = SELECT_PANEL_PADDING_X * 2;\n\n/** The height of the select items in `em` units. */\nexport const SELECT_ITEM_HEIGHT_EM = 3;\n\n// TODO(josephperrott): Revert to a constant after 2018 spec updates are fully merged.\n/**\n * Distance between the panel edge and the option text in\n * multi-selection mode.\n *\n * Calculated as:\n * (SELECT_PANEL_PADDING_X * 1.5) + 16 = 40\n * The padding is multiplied by 1.5 because the checkbox's margin is half the padding.\n * The checkbox width is 16px.\n */\nexport const SELECT_MULTIPLE_PANEL_PADDING_X = SELECT_PANEL_PADDING_X * 1.5 + 16;\n\n/**\n * The select panel will only \"fit\" inside the viewport if it is positioned at\n * this value or more away from the viewport boundary.\n */\nexport const SELECT_PANEL_VIEWPORT_PADDING = 8;\n\n/** Injection token that determines the scroll handling while a select is open. */\nexport const MAT_SELECT_SCROLL_STRATEGY =\n    new InjectionToken<() => ScrollStrategy>('mat-select-scroll-strategy');\n\n/** @docs-private */\nexport function MAT_SELECT_SCROLL_STRATEGY_PROVIDER_FACTORY(overlay: Overlay):\n    () => ScrollStrategy {\n  return () => overlay.scrollStrategies.reposition();\n}\n\n/** Object that can be used to configure the default options for the select module. */\nexport interface MatSelectConfig {\n  /** Whether option centering should be disabled. */\n  disableOptionCentering?: boolean;\n\n  /** Time to wait in milliseconds after the last keystroke before moving focus to an item. */\n  typeaheadDebounceInterval?: number;\n}\n\n/** Injection token that can be used to provide the default options the select module. */\nexport const MAT_SELECT_CONFIG = new InjectionToken<MatSelectConfig>('MAT_SELECT_CONFIG');\n\n/** @docs-private */\nexport const MAT_SELECT_SCROLL_STRATEGY_PROVIDER = {\n  provide: MAT_SELECT_SCROLL_STRATEGY,\n  deps: [Overlay],\n  useFactory: MAT_SELECT_SCROLL_STRATEGY_PROVIDER_FACTORY,\n};\n\n/** Change event object that is emitted when the select value has changed. */\nexport class MatSelectChange {\n  constructor(\n    /** Reference to the select that emitted the change event. */\n    public source: MatSelect,\n    /** Current value of the select that emitted the event. */\n    public value: any) { }\n}\n\n// Boilerplate for applying mixins to MatSelect.\n/** @docs-private */\nclass MatSelectBase {\n  constructor(public _elementRef: ElementRef,\n              public _defaultErrorStateMatcher: ErrorStateMatcher,\n              public _parentForm: NgForm,\n              public _parentFormGroup: FormGroupDirective,\n              public ngControl: NgControl) {}\n}\nconst _MatSelectMixinBase:\n    CanDisableCtor &\n    HasTabIndexCtor &\n    CanDisableRippleCtor &\n    CanUpdateErrorStateCtor &\n    typeof MatSelectBase =\n        mixinDisableRipple(mixinTabIndex(mixinDisabled(mixinErrorState(MatSelectBase))));\n\n\n\n/**\n * Injection token that can be used to reference instances of `MatSelectTrigger`. It serves as\n * alternative token to the actual `MatSelectTrigger` class which could cause unnecessary\n * retention of the class and its directive metadata.\n */\nexport const MAT_SELECT_TRIGGER = new InjectionToken<MatSelectTrigger>('MatSelectTrigger');\n\n/**\n * Allows the user to customize the trigger that is displayed when the select has a value.\n */\n@Directive({\n  selector: 'mat-select-trigger',\n  providers: [{provide: MAT_SELECT_TRIGGER, useExisting: MatSelectTrigger}],\n})\nexport class MatSelectTrigger {}\n\n\n@Component({\n  selector: 'mat-select',\n  exportAs: 'matSelect',\n  templateUrl: 'select.html',\n  styleUrls: ['select.css'],\n  inputs: ['disabled', 'disableRipple', 'tabIndex'],\n  encapsulation: ViewEncapsulation.None,\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  host: {\n    'role': 'listbox',\n    '[attr.id]': 'id',\n    '[attr.tabindex]': 'tabIndex',\n    '[attr.aria-label]': '_getAriaLabel()',\n    '[attr.aria-labelledby]': '_getAriaLabelledby()',\n    '[attr.aria-required]': 'required.toString()',\n    '[attr.aria-disabled]': 'disabled.toString()',\n    '[attr.aria-invalid]': 'errorState',\n    '[attr.aria-owns]': 'panelOpen ? _optionIds : null',\n    '[attr.aria-multiselectable]': 'multiple',\n    '[attr.aria-describedby]': '_ariaDescribedby || null',\n    '[attr.aria-activedescendant]': '_getAriaActiveDescendant()',\n    '[class.mat-select-disabled]': 'disabled',\n    '[class.mat-select-invalid]': 'errorState',\n    '[class.mat-select-required]': 'required',\n    '[class.mat-select-empty]': 'empty',\n    'class': 'mat-select',\n    '(keydown)': '_handleKeydown($event)',\n    '(focus)': '_onFocus()',\n    '(blur)': '_onBlur()',\n  },\n  animations: [\n    matSelectAnimations.transformPanelWrap,\n    matSelectAnimations.transformPanel\n  ],\n  providers: [\n    {provide: MatFormFieldControl, useExisting: MatSelect},\n    {provide: MAT_OPTION_PARENT_COMPONENT, useExisting: MatSelect}\n  ],\n})\nexport class MatSelect extends _MatSelectMixinBase implements AfterContentInit, OnChanges,\n    OnDestroy, OnInit, DoCheck, ControlValueAccessor, CanDisable, HasTabIndex,\n    MatFormFieldControl<any>, CanUpdateErrorState, CanDisableRipple {\n  private _scrollStrategyFactory: () => ScrollStrategy;\n\n  /** Whether or not the overlay panel is open. */\n  private _panelOpen = false;\n\n  /** Whether filling out the select is required in the form. */\n  private _required: boolean = false;\n\n  /** The scroll position of the overlay panel, calculated to center the selected option. */\n  private _scrollTop = 0;\n\n  /** The placeholder displayed in the trigger of the select. */\n  private _placeholder: string;\n\n  /** Whether the component is in multiple selection mode. */\n  private _multiple: boolean = false;\n\n  /** Comparison function to specify which option is displayed. Defaults to object equality. */\n  private _compareWith = (o1: any, o2: any) => o1 === o2;\n\n  /** Unique id for this input. */\n  private _uid = `mat-select-${nextUniqueId++}`;\n\n  /** Emits whenever the component is destroyed. */\n  private readonly _destroy = new Subject<void>();\n\n  /** The last measured value for the trigger's client bounding rect. */\n  _triggerRect: ClientRect;\n\n  /** The aria-describedby attribute on the select for improved a11y. */\n  _ariaDescribedby: string;\n\n  /** The cached font-size of the trigger element. */\n  _triggerFontSize = 0;\n\n  /** Deals with the selection logic. */\n  _selectionModel: SelectionModel<MatOption>;\n\n  /** Manages keyboard events for options in the panel. */\n  _keyManager: ActiveDescendantKeyManager<MatOption>;\n\n  /** `View -> model callback called when value changes` */\n  _onChange: (value: any) => void = () => {};\n\n  /** `View -> model callback called when select has been touched` */\n  _onTouched = () => {};\n\n  /** The IDs of child options to be passed to the aria-owns attribute. */\n  _optionIds: string = '';\n\n  /** The value of the select panel's transform-origin property. */\n  _transformOrigin: string = 'top';\n\n  /** Emits when the panel element is finished transforming in. */\n  _panelDoneAnimatingStream = new Subject<string>();\n\n  /** Strategy that will be used to handle scrolling while the select panel is open. */\n  _scrollStrategy: ScrollStrategy;\n\n  /**\n   * The y-offset of the overlay panel in relation to the trigger's top start corner.\n   * This must be adjusted to align the selected option text over the trigger text.\n   * when the panel opens. Will change based on the y-position of the selected option.\n   */\n  _offsetY = 0;\n\n  /**\n   * This position config ensures that the top \"start\" corner of the overlay\n   * is aligned with with the top \"start\" of the origin by default (overlapping\n   * the trigger completely). If the panel cannot fit below the trigger, it\n   * will fall back to a position above the trigger.\n   */\n  _positions: ConnectedPosition[] = [\n    {\n      originX: 'start',\n      originY: 'top',\n      overlayX: 'start',\n      overlayY: 'top',\n    },\n    {\n      originX: 'start',\n      originY: 'bottom',\n      overlayX: 'start',\n      overlayY: 'bottom',\n    },\n  ];\n\n  /** Whether the component is disabling centering of the active option over the trigger. */\n  private _disableOptionCentering: boolean = false;\n\n  /** Whether the select is focused. */\n  get focused(): boolean {\n    return this._focused || this._panelOpen;\n  }\n  private _focused = false;\n\n  /** A name for this control that can be used by `mat-form-field`. */\n  controlType = 'mat-select';\n\n  /** Trigger that opens the select. */\n  @ViewChild('trigger') trigger: ElementRef;\n\n  /** Panel containing the select options. */\n  @ViewChild('panel') panel: ElementRef;\n\n  /**\n   * Overlay pane containing the options.\n   * @deprecated To be turned into a private API.\n   * @breaking-change 10.0.0\n   * @docs-private\n   */\n  @ViewChild(CdkConnectedOverlay) overlayDir: CdkConnectedOverlay;\n\n  /** All of the defined select options. */\n  @ContentChildren(MatOption, {descendants: true}) options: QueryList<MatOption>;\n\n  // TODO: Remove cast once https://github.com/angular/angular/pull/37506 is available.\n  /** All of the defined groups of options. */\n  @ContentChildren(MAT_OPTGROUP as any, {descendants: true}) optionGroups: QueryList<MatOptgroup>;\n\n  /** Classes to be passed to the select panel. Supports the same syntax as `ngClass`. */\n  @Input() panelClass: string|string[]|Set<string>|{[key: string]: any};\n\n  // TODO: Remove cast once https://github.com/angular/angular/pull/37506 is available.\n  /** User-supplied override of the trigger element. */\n  @ContentChild(MAT_SELECT_TRIGGER as any) customTrigger: MatSelectTrigger;\n\n  /** Placeholder to be shown if no value has been selected. */\n  @Input()\n  get placeholder(): string { return this._placeholder; }\n  set placeholder(value: string) {\n    this._placeholder = value;\n    this.stateChanges.next();\n  }\n\n  /** Whether the component is required. */\n  @Input()\n  get required(): boolean { return this._required; }\n  set required(value: boolean) {\n    this._required = coerceBooleanProperty(value);\n    this.stateChanges.next();\n  }\n\n  /** Whether the user should be allowed to select multiple options. */\n  @Input()\n  get multiple(): boolean { return this._multiple; }\n  set multiple(value: boolean) {\n    if (this._selectionModel) {\n      throw getMatSelectDynamicMultipleError();\n    }\n\n    this._multiple = coerceBooleanProperty(value);\n  }\n\n  /** Whether to center the active option over the trigger. */\n  @Input()\n  get disableOptionCentering(): boolean { return this._disableOptionCentering; }\n  set disableOptionCentering(value: boolean) {\n    this._disableOptionCentering = coerceBooleanProperty(value);\n  }\n\n  /**\n   * Function to compare the option values with the selected values. The first argument\n   * is a value from an option. The second is a value from the selection. A boolean\n   * should be returned.\n   */\n  @Input()\n  get compareWith() { return this._compareWith; }\n  set compareWith(fn: (o1: any, o2: any) => boolean) {\n    if (typeof fn !== 'function') {\n      throw getMatSelectNonFunctionValueError();\n    }\n    this._compareWith = fn;\n    if (this._selectionModel) {\n      // A different comparator means the selection could change.\n      this._initializeSelection();\n    }\n  }\n\n  /** Value of the select control. */\n  @Input()\n  get value(): any { return this._value; }\n  set value(newValue: any) {\n    if (newValue !== this._value) {\n      this.writeValue(newValue);\n      this._value = newValue;\n    }\n  }\n  private _value: any;\n\n  /** Aria label of the select. If not specified, the placeholder will be used as label. */\n  @Input('aria-label') ariaLabel: string = '';\n\n  /** Input that can be used to specify the `aria-labelledby` attribute. */\n  @Input('aria-labelledby') ariaLabelledby: string;\n\n  /** Object used to control when error messages are shown. */\n  @Input() errorStateMatcher: ErrorStateMatcher;\n\n  /** Time to wait in milliseconds after the last keystroke before moving focus to an item. */\n  @Input()\n  get typeaheadDebounceInterval(): number { return this._typeaheadDebounceInterval; }\n  set typeaheadDebounceInterval(value: number) {\n    this._typeaheadDebounceInterval = coerceNumberProperty(value);\n  }\n  private _typeaheadDebounceInterval: number;\n\n  /**\n   * Function used to sort the values in a select in multiple mode.\n   * Follows the same logic as `Array.prototype.sort`.\n   */\n  @Input() sortComparator: (a: MatOption, b: MatOption, options: MatOption[]) => number;\n\n  /** Unique id of the element. */\n  @Input()\n  get id(): string { return this._id; }\n  set id(value: string) {\n    this._id = value || this._uid;\n    this.stateChanges.next();\n  }\n  private _id: string;\n\n  /** Combined stream of all of the child options' change events. */\n  readonly optionSelectionChanges: Observable<MatOptionSelectionChange> = defer(() => {\n    const options = this.options;\n\n    if (options) {\n      return options.changes.pipe(\n        startWith(options),\n        switchMap(() => merge(...options.map(option => option.onSelectionChange)))\n      );\n    }\n\n    return this._ngZone.onStable\n      .asObservable()\n      .pipe(take(1), switchMap(() => this.optionSelectionChanges));\n  }) as Observable<MatOptionSelectionChange>;\n\n  /** Event emitted when the select panel has been toggled. */\n  @Output() readonly openedChange: EventEmitter<boolean> = new EventEmitter<boolean>();\n\n  /** Event emitted when the select has been opened. */\n  @Output('opened') readonly _openedStream: Observable<void> =\n      this.openedChange.pipe(filter(o => o), map(() => {}));\n\n  /** Event emitted when the select has been closed. */\n  @Output('closed') readonly _closedStream: Observable<void> =\n      this.openedChange.pipe(filter(o => !o), map(() => {}));\n\n   /** Event emitted when the selected value has been changed by the user. */\n  @Output() readonly selectionChange: EventEmitter<MatSelectChange> =\n      new EventEmitter<MatSelectChange>();\n\n  /**\n   * Event that emits whenever the raw value of the select changes. This is here primarily\n   * to facilitate the two-way binding for the `value` input.\n   * @docs-private\n   */\n  @Output() readonly valueChange: EventEmitter<any> = new EventEmitter<any>();\n\n  constructor(\n    private _viewportRuler: ViewportRuler,\n    private _changeDetectorRef: ChangeDetectorRef,\n    private _ngZone: NgZone,\n    _defaultErrorStateMatcher: ErrorStateMatcher,\n    elementRef: ElementRef,\n    @Optional() private _dir: Directionality,\n    @Optional() _parentForm: NgForm,\n    @Optional() _parentFormGroup: FormGroupDirective,\n    @Optional() @Inject(MAT_FORM_FIELD) private _parentFormField: MatFormField,\n    @Self() @Optional() public ngControl: NgControl,\n    @Attribute('tabindex') tabIndex: string,\n    @Inject(MAT_SELECT_SCROLL_STRATEGY) scrollStrategyFactory: any,\n    private _liveAnnouncer: LiveAnnouncer,\n    @Optional() @Inject(MAT_SELECT_CONFIG) defaults?: MatSelectConfig) {\n    super(elementRef, _defaultErrorStateMatcher, _parentForm,\n          _parentFormGroup, ngControl);\n\n    if (this.ngControl) {\n      // Note: we provide the value accessor through here, instead of\n      // the `providers` to avoid running into a circular import.\n      this.ngControl.valueAccessor = this;\n    }\n\n    this._scrollStrategyFactory = scrollStrategyFactory;\n    this._scrollStrategy = this._scrollStrategyFactory();\n    this.tabIndex = parseInt(tabIndex) || 0;\n\n    // Force setter to be called in case id was not specified.\n    this.id = this.id;\n\n    if (defaults) {\n      if (defaults.disableOptionCentering != null) {\n        this.disableOptionCentering = defaults.disableOptionCentering;\n      }\n\n      if (defaults.typeaheadDebounceInterval != null) {\n        this.typeaheadDebounceInterval = defaults.typeaheadDebounceInterval;\n      }\n    }\n  }\n\n  ngOnInit() {\n    this._selectionModel = new SelectionModel<MatOption>(this.multiple);\n    this.stateChanges.next();\n\n    // We need `distinctUntilChanged` here, because some browsers will\n    // fire the animation end event twice for the same animation. See:\n    // https://github.com/angular/angular/issues/24084\n    this._panelDoneAnimatingStream\n      .pipe(distinctUntilChanged(), takeUntil(this._destroy))\n      .subscribe(() => {\n        if (this.panelOpen) {\n          this._scrollTop = 0;\n          this.openedChange.emit(true);\n        } else {\n          this.openedChange.emit(false);\n          this.overlayDir.offsetX = 0;\n          this._changeDetectorRef.markForCheck();\n        }\n      });\n\n    this._viewportRuler.change()\n      .pipe(takeUntil(this._destroy))\n      .subscribe(() => {\n        if (this._panelOpen) {\n          this._triggerRect = this.trigger.nativeElement.getBoundingClientRect();\n          this._changeDetectorRef.markForCheck();\n        }\n      });\n  }\n\n  ngAfterContentInit() {\n    this._initKeyManager();\n\n    this._selectionModel.changed.pipe(takeUntil(this._destroy)).subscribe(event => {\n      event.added.forEach(option => option.select());\n      event.removed.forEach(option => option.deselect());\n    });\n\n    this.options.changes.pipe(startWith(null), takeUntil(this._destroy)).subscribe(() => {\n      this._resetOptions();\n      this._initializeSelection();\n    });\n  }\n\n  ngDoCheck() {\n    if (this.ngControl) {\n      this.updateErrorState();\n    }\n  }\n\n  ngOnChanges(changes: SimpleChanges) {\n    // Updating the disabled state is handled by `mixinDisabled`, but we need to additionally let\n    // the parent form field know to run change detection when the disabled state changes.\n    if (changes['disabled']) {\n      this.stateChanges.next();\n    }\n\n    if (changes['typeaheadDebounceInterval'] && this._keyManager) {\n      this._keyManager.withTypeAhead(this._typeaheadDebounceInterval);\n    }\n  }\n\n  ngOnDestroy() {\n    this._destroy.next();\n    this._destroy.complete();\n    this.stateChanges.complete();\n  }\n\n  /** Toggles the overlay panel open or closed. */\n  toggle(): void {\n    this.panelOpen ? this.close() : this.open();\n  }\n\n  /** Opens the overlay panel. */\n  open(): void {\n    if (this.disabled || !this.options || !this.options.length || this._panelOpen) {\n      return;\n    }\n\n    this._triggerRect = this.trigger.nativeElement.getBoundingClientRect();\n    // Note: The computed font-size will be a string pixel value (e.g. \"16px\").\n    // `parseInt` ignores the trailing 'px' and converts this to a number.\n    this._triggerFontSize = parseInt(getComputedStyle(this.trigger.nativeElement).fontSize || '0');\n\n    this._panelOpen = true;\n    this._keyManager.withHorizontalOrientation(null);\n    this._calculateOverlayPosition();\n    this._highlightCorrectOption();\n    this._changeDetectorRef.markForCheck();\n\n    // Set the font size on the panel element once it exists.\n    this._ngZone.onStable.asObservable().pipe(take(1)).subscribe(() => {\n      if (this._triggerFontSize && this.overlayDir.overlayRef &&\n          this.overlayDir.overlayRef.overlayElement) {\n        this.overlayDir.overlayRef.overlayElement.style.fontSize = `${this._triggerFontSize}px`;\n      }\n    });\n  }\n\n  /** Closes the overlay panel and focuses the host element. */\n  close(): void {\n    if (this._panelOpen) {\n      this._panelOpen = false;\n      this._keyManager.withHorizontalOrientation(this._isRtl() ? 'rtl' : 'ltr');\n      this._changeDetectorRef.markForCheck();\n      this._onTouched();\n    }\n  }\n\n  /**\n   * Sets the select's value. Part of the ControlValueAccessor interface\n   * required to integrate with Angular's core forms API.\n   *\n   * @param value New value to be written to the model.\n   */\n  writeValue(value: any): void {\n    if (this.options) {\n      this._setSelectionByValue(value);\n    }\n  }\n\n  /**\n   * Saves a callback function to be invoked when the select's value\n   * changes from user input. Part of the ControlValueAccessor interface\n   * required to integrate with Angular's core forms API.\n   *\n   * @param fn Callback to be triggered when the value changes.\n   */\n  registerOnChange(fn: (value: any) => void): void {\n    this._onChange = fn;\n  }\n\n  /**\n   * Saves a callback function to be invoked when the select is blurred\n   * by the user. Part of the ControlValueAccessor interface required\n   * to integrate with Angular's core forms API.\n   *\n   * @param fn Callback to be triggered when the component has been touched.\n   */\n  registerOnTouched(fn: () => {}): void {\n    this._onTouched = fn;\n  }\n\n  /**\n   * Disables the select. Part of the ControlValueAccessor interface required\n   * to integrate with Angular's core forms API.\n   *\n   * @param isDisabled Sets whether the component is disabled.\n   */\n  setDisabledState(isDisabled: boolean): void {\n    this.disabled = isDisabled;\n    this._changeDetectorRef.markForCheck();\n    this.stateChanges.next();\n  }\n\n  /** Whether or not the overlay panel is open. */\n  get panelOpen(): boolean {\n    return this._panelOpen;\n  }\n\n  /** The currently selected option. */\n  get selected(): MatOption | MatOption[] {\n    return this.multiple ? this._selectionModel.selected : this._selectionModel.selected[0];\n  }\n\n  /** The value displayed in the trigger. */\n  get triggerValue(): string {\n    if (this.empty) {\n      return '';\n    }\n\n    if (this._multiple) {\n      const selectedOptions = this._selectionModel.selected.map(option => option.viewValue);\n\n      if (this._isRtl()) {\n        selectedOptions.reverse();\n      }\n\n      // TODO(crisbeto): delimiter should be configurable for proper localization.\n      return selectedOptions.join(', ');\n    }\n\n    return this._selectionModel.selected[0].viewValue;\n  }\n\n  /** Whether the element is in RTL mode. */\n  _isRtl(): boolean {\n    return this._dir ? this._dir.value === 'rtl' : false;\n  }\n\n  /** Handles all keydown events on the select. */\n  _handleKeydown(event: KeyboardEvent): void {\n    if (!this.disabled) {\n      this.panelOpen ? this._handleOpenKeydown(event) : this._handleClosedKeydown(event);\n    }\n  }\n\n  /** Handles keyboard events while the select is closed. */\n  private _handleClosedKeydown(event: KeyboardEvent): void {\n    const keyCode = event.keyCode;\n    const isArrowKey = keyCode === DOWN_ARROW || keyCode === UP_ARROW ||\n                       keyCode === LEFT_ARROW || keyCode === RIGHT_ARROW;\n    const isOpenKey = keyCode === ENTER || keyCode === SPACE;\n    const manager = this._keyManager;\n\n    // Open the select on ALT + arrow key to match the native <select>\n    if (!manager.isTyping() && (isOpenKey && !hasModifierKey(event)) ||\n      ((this.multiple || event.altKey) && isArrowKey)) {\n      event.preventDefault(); // prevents the page from scrolling down when pressing space\n      this.open();\n    } else if (!this.multiple) {\n      const previouslySelectedOption = this.selected;\n\n      if (keyCode === HOME || keyCode === END) {\n        keyCode === HOME ? manager.setFirstItemActive() : manager.setLastItemActive();\n        event.preventDefault();\n      } else {\n        manager.onKeydown(event);\n      }\n\n      const selectedOption = this.selected;\n\n      // Since the value has changed, we need to announce it ourselves.\n      if (selectedOption && previouslySelectedOption !== selectedOption) {\n        // We set a duration on the live announcement, because we want the live element to be\n        // cleared after a while so that users can't navigate to it using the arrow keys.\n        this._liveAnnouncer.announce((selectedOption as MatOption).viewValue, 10000);\n      }\n    }\n  }\n\n  /** Handles keyboard events when the selected is open. */\n  private _handleOpenKeydown(event: KeyboardEvent): void {\n    const manager = this._keyManager;\n    const keyCode = event.keyCode;\n    const isArrowKey = keyCode === DOWN_ARROW || keyCode === UP_ARROW;\n    const isTyping = manager.isTyping();\n\n    if (keyCode === HOME || keyCode === END) {\n      event.preventDefault();\n      keyCode === HOME ? manager.setFirstItemActive() : manager.setLastItemActive();\n    } else if (isArrowKey && event.altKey) {\n      // Close the select on ALT + arrow key to match the native <select>\n      event.preventDefault();\n      this.close();\n      // Don't do anything in this case if the user is typing,\n      // because the typing sequence can include the space key.\n    } else if (!isTyping && (keyCode === ENTER || keyCode === SPACE) && manager.activeItem &&\n      !hasModifierKey(event)) {\n      event.preventDefault();\n      manager.activeItem._selectViaInteraction();\n    } else if (!isTyping && this._multiple && keyCode === A && event.ctrlKey) {\n      event.preventDefault();\n      const hasDeselectedOptions = this.options.some(opt => !opt.disabled && !opt.selected);\n\n      this.options.forEach(option => {\n        if (!option.disabled) {\n          hasDeselectedOptions ? option.select() : option.deselect();\n        }\n      });\n    } else {\n      const previouslyFocusedIndex = manager.activeItemIndex;\n\n      manager.onKeydown(event);\n\n      if (this._multiple && isArrowKey && event.shiftKey && manager.activeItem &&\n          manager.activeItemIndex !== previouslyFocusedIndex) {\n        manager.activeItem._selectViaInteraction();\n      }\n    }\n  }\n\n  _onFocus() {\n    if (!this.disabled) {\n      this._focused = true;\n      this.stateChanges.next();\n    }\n  }\n\n  /**\n   * Calls the touched callback only if the panel is closed. Otherwise, the trigger will\n   * \"blur\" to the panel when it opens, causing a false positive.\n   */\n  _onBlur() {\n    this._focused = false;\n\n    if (!this.disabled && !this.panelOpen) {\n      this._onTouched();\n      this._changeDetectorRef.markForCheck();\n      this.stateChanges.next();\n    }\n  }\n\n  /**\n   * Callback that is invoked when the overlay panel has been attached.\n   */\n  _onAttached(): void {\n    this.overlayDir.positionChange.pipe(take(1)).subscribe(() => {\n      this._changeDetectorRef.detectChanges();\n      this._calculateOverlayOffsetX();\n      this.panel.nativeElement.scrollTop = this._scrollTop;\n    });\n  }\n\n  /** Returns the theme to be used on the panel. */\n  _getPanelTheme(): string {\n    return this._parentFormField ? `mat-${this._parentFormField.color}` : '';\n  }\n\n  /** Whether the select has a value. */\n  get empty(): boolean {\n    return !this._selectionModel || this._selectionModel.isEmpty();\n  }\n\n  private _initializeSelection(): void {\n    // Defer setting the value in order to avoid the \"Expression\n    // has changed after it was checked\" errors from Angular.\n    Promise.resolve().then(() => {\n      this._setSelectionByValue(this.ngControl ? this.ngControl.value : this._value);\n      this.stateChanges.next();\n    });\n  }\n\n  /**\n   * Sets the selected option based on a value. If no option can be\n   * found with the designated value, the select trigger is cleared.\n   */\n  private _setSelectionByValue(value: any | any[]): void {\n    if (this.multiple && value) {\n      if (!Array.isArray(value)) {\n        throw getMatSelectNonArrayValueError();\n      }\n\n      this._selectionModel.clear();\n      value.forEach((currentValue: any) => this._selectValue(currentValue));\n      this._sortValues();\n    } else {\n      this._selectionModel.clear();\n      const correspondingOption = this._selectValue(value);\n\n      // Shift focus to the active item. Note that we shouldn't do this in multiple\n      // mode, because we don't know what option the user interacted with last.\n      if (correspondingOption) {\n        this._keyManager.setActiveItem(correspondingOption);\n      } else if (!this.panelOpen) {\n        // Otherwise reset the highlighted option. Note that we only want to do this while\n        // closed, because doing it while open can shift the user's focus unnecessarily.\n        this._keyManager.setActiveItem(-1);\n      }\n    }\n\n    this._changeDetectorRef.markForCheck();\n  }\n\n  /**\n   * Finds and selects and option based on its value.\n   * @returns Option that has the corresponding value.\n   */\n  private _selectValue(value: any): MatOption | undefined {\n    const correspondingOption = this.options.find((option: MatOption) => {\n      try {\n        // Treat null as a special reset value.\n        return option.value != null && this._compareWith(option.value,  value);\n      } catch (error) {\n        if (isDevMode()) {\n          // Notify developers of errors in their comparator.\n          console.warn(error);\n        }\n        return false;\n      }\n    });\n\n    if (correspondingOption) {\n      this._selectionModel.select(correspondingOption);\n    }\n\n    return correspondingOption;\n  }\n\n  /** Sets up a key manager to listen to keyboard events on the overlay panel. */\n  private _initKeyManager() {\n    this._keyManager = new ActiveDescendantKeyManager<MatOption>(this.options)\n      .withTypeAhead(this._typeaheadDebounceInterval)\n      .withVerticalOrientation()\n      .withHorizontalOrientation(this._isRtl() ? 'rtl' : 'ltr')\n      .withAllowedModifierKeys(['shiftKey']);\n\n    this._keyManager.tabOut.pipe(takeUntil(this._destroy)).subscribe(() => {\n      if (this.panelOpen) {\n        // Select the active item when tabbing away. This is consistent with how the native\n        // select behaves. Note that we only want to do this in single selection mode.\n        if (!this.multiple && this._keyManager.activeItem) {\n          this._keyManager.activeItem._selectViaInteraction();\n        }\n\n        // Restore focus to the trigger before closing. Ensures that the focus\n        // position won't be lost if the user got focus into the overlay.\n        this.focus();\n        this.close();\n      }\n    });\n\n    this._keyManager.change.pipe(takeUntil(this._destroy)).subscribe(() => {\n      if (this._panelOpen && this.panel) {\n        this._scrollActiveOptionIntoView();\n      } else if (!this._panelOpen && !this.multiple && this._keyManager.activeItem) {\n        this._keyManager.activeItem._selectViaInteraction();\n      }\n    });\n  }\n\n  /** Drops current option subscriptions and IDs and resets from scratch. */\n  private _resetOptions(): void {\n    const changedOrDestroyed = merge(this.options.changes, this._destroy);\n\n    this.optionSelectionChanges.pipe(takeUntil(changedOrDestroyed)).subscribe(event => {\n      this._onSelect(event.source, event.isUserInput);\n\n      if (event.isUserInput && !this.multiple && this._panelOpen) {\n        this.close();\n        this.focus();\n      }\n    });\n\n    // Listen to changes in the internal state of the options and react accordingly.\n    // Handles cases like the labels of the selected options changing.\n    merge(...this.options.map(option => option._stateChanges))\n      .pipe(takeUntil(changedOrDestroyed))\n      .subscribe(() => {\n        this._changeDetectorRef.markForCheck();\n        this.stateChanges.next();\n      });\n\n    this._setOptionIds();\n  }\n\n  /** Invoked when an option is clicked. */\n  private _onSelect(option: MatOption, isUserInput: boolean): void {\n    const wasSelected = this._selectionModel.isSelected(option);\n\n    if (option.value == null && !this._multiple) {\n      option.deselect();\n      this._selectionModel.clear();\n      this._propagateChanges(option.value);\n    } else {\n      if (wasSelected !== option.selected) {\n        option.selected ? this._selectionModel.select(option) :\n                          this._selectionModel.deselect(option);\n      }\n\n      if (isUserInput) {\n        this._keyManager.setActiveItem(option);\n      }\n\n      if (this.multiple) {\n        this._sortValues();\n\n        if (isUserInput) {\n          // In case the user selected the option with their mouse, we\n          // want to restore focus back to the trigger, in order to\n          // prevent the select keyboard controls from clashing with\n          // the ones from `mat-option`.\n          this.focus();\n        }\n      }\n    }\n\n    if (wasSelected !== this._selectionModel.isSelected(option)) {\n      this._propagateChanges();\n    }\n\n    this.stateChanges.next();\n  }\n\n  /** Sorts the selected values in the selected based on their order in the panel. */\n  private _sortValues() {\n    if (this.multiple) {\n      const options = this.options.toArray();\n\n      this._selectionModel.sort((a, b) => {\n        return this.sortComparator ? this.sortComparator(a, b, options) :\n                                     options.indexOf(a) - options.indexOf(b);\n      });\n      this.stateChanges.next();\n    }\n  }\n\n  /** Emits change event to set the model value. */\n  private _propagateChanges(fallbackValue?: any): void {\n    let valueToEmit: any = null;\n\n    if (this.multiple) {\n      valueToEmit = (this.selected as MatOption[]).map(option => option.value);\n    } else {\n      valueToEmit = this.selected ? (this.selected as MatOption).value : fallbackValue;\n    }\n\n    this._value = valueToEmit;\n    this.valueChange.emit(valueToEmit);\n    this._onChange(valueToEmit);\n    this.selectionChange.emit(new MatSelectChange(this, valueToEmit));\n    this._changeDetectorRef.markForCheck();\n  }\n\n  /** Records option IDs to pass to the aria-owns property. */\n  private _setOptionIds() {\n    this._optionIds = this.options.map(option => option.id).join(' ');\n  }\n\n  /**\n   * Highlights the selected item. If no option is selected, it will highlight\n   * the first item instead.\n   */\n  private _highlightCorrectOption(): void {\n    if (this._keyManager) {\n      if (this.empty) {\n        this._keyManager.setFirstItemActive();\n      } else {\n        this._keyManager.setActiveItem(this._selectionModel.selected[0]);\n      }\n    }\n  }\n\n  /** Scrolls the active option into view. */\n  private _scrollActiveOptionIntoView(): void {\n    const activeOptionIndex = this._keyManager.activeItemIndex || 0;\n    const labelCount = _countGroupLabelsBeforeOption(activeOptionIndex, this.options,\n        this.optionGroups);\n\n    this.panel.nativeElement.scrollTop = _getOptionScrollPosition(\n      activeOptionIndex + labelCount,\n      this._getItemHeight(),\n      this.panel.nativeElement.scrollTop,\n      SELECT_PANEL_MAX_HEIGHT\n    );\n  }\n\n  /** Focuses the select element. */\n  focus(options?: FocusOptions): void {\n    this._elementRef.nativeElement.focus(options);\n  }\n\n  /** Gets the index of the provided option in the option list. */\n  private _getOptionIndex(option: MatOption): number | undefined {\n    return this.options.reduce((result: number | undefined, current: MatOption, index: number) => {\n      if (result !== undefined) {\n        return result;\n      }\n\n      return option === current ? index : undefined;\n    }, undefined);\n  }\n\n  /** Calculates the scroll position and x- and y-offsets of the overlay panel. */\n  private _calculateOverlayPosition(): void {\n    const itemHeight = this._getItemHeight();\n    const items = this._getItemCount();\n    const panelHeight = Math.min(items * itemHeight, SELECT_PANEL_MAX_HEIGHT);\n    const scrollContainerHeight = items * itemHeight;\n\n    // The farthest the panel can be scrolled before it hits the bottom\n    const maxScroll = scrollContainerHeight - panelHeight;\n\n    // If no value is selected we open the popup to the first item.\n    let selectedOptionOffset =\n        this.empty ? 0 : this._getOptionIndex(this._selectionModel.selected[0])!;\n\n    selectedOptionOffset += _countGroupLabelsBeforeOption(selectedOptionOffset, this.options,\n        this.optionGroups);\n\n    // We must maintain a scroll buffer so the selected option will be scrolled to the\n    // center of the overlay panel rather than the top.\n    const scrollBuffer = panelHeight / 2;\n    this._scrollTop = this._calculateOverlayScroll(selectedOptionOffset, scrollBuffer, maxScroll);\n    this._offsetY = this._calculateOverlayOffsetY(selectedOptionOffset, scrollBuffer, maxScroll);\n\n    this._checkOverlayWithinViewport(maxScroll);\n  }\n\n  /**\n   * Calculates the scroll position of the select's overlay panel.\n   *\n   * Attempts to center the selected option in the panel. If the option is\n   * too high or too low in the panel to be scrolled to the center, it clamps the\n   * scroll position to the min or max scroll positions respectively.\n   */\n  _calculateOverlayScroll(selectedIndex: number, scrollBuffer: number,\n                          maxScroll: number): number {\n    const itemHeight = this._getItemHeight();\n    const optionOffsetFromScrollTop = itemHeight * selectedIndex;\n    const halfOptionHeight = itemHeight / 2;\n\n    // Starts at the optionOffsetFromScrollTop, which scrolls the option to the top of the\n    // scroll container, then subtracts the scroll buffer to scroll the option down to\n    // the center of the overlay panel. Half the option height must be re-added to the\n    // scrollTop so the option is centered based on its middle, not its top edge.\n    const optimalScrollPosition = optionOffsetFromScrollTop - scrollBuffer + halfOptionHeight;\n    return Math.min(Math.max(0, optimalScrollPosition), maxScroll);\n  }\n\n  /** Returns the aria-label of the select component. */\n  _getAriaLabel(): string | null {\n    // If an ariaLabelledby value has been set by the consumer, the select should not overwrite the\n    // `aria-labelledby` value by setting the ariaLabel to the placeholder.\n    return this.ariaLabelledby ? null : this.ariaLabel || this.placeholder;\n  }\n\n  /** Returns the aria-labelledby of the select component. */\n  _getAriaLabelledby(): string | null {\n    if (this.ariaLabelledby) {\n      return this.ariaLabelledby;\n    }\n\n    // Note: we use `_getAriaLabel` here, because we want to check whether there's a\n    // computed label. `this.ariaLabel` is only the user-specified label.\n    if (!this._parentFormField || !this._parentFormField._hasFloatingLabel() ||\n      this._getAriaLabel()) {\n      return null;\n    }\n\n    return this._parentFormField._labelId || null;\n  }\n\n  /** Determines the `aria-activedescendant` to be set on the host. */\n  _getAriaActiveDescendant(): string | null {\n    if (this.panelOpen && this._keyManager && this._keyManager.activeItem) {\n      return this._keyManager.activeItem.id;\n    }\n\n    return null;\n  }\n\n  /**\n   * Sets the x-offset of the overlay panel in relation to the trigger's top start corner.\n   * This must be adjusted to align the selected option text over the trigger text when\n   * the panel opens. Will change based on LTR or RTL text direction. Note that the offset\n   * can't be calculated until the panel has been attached, because we need to know the\n   * content width in order to constrain the panel within the viewport.\n   */\n  private _calculateOverlayOffsetX(): void {\n    const overlayRect = this.overlayDir.overlayRef.overlayElement.getBoundingClientRect();\n    const viewportSize = this._viewportRuler.getViewportSize();\n    const isRtl = this._isRtl();\n    const paddingWidth = this.multiple ? SELECT_MULTIPLE_PANEL_PADDING_X + SELECT_PANEL_PADDING_X :\n                                         SELECT_PANEL_PADDING_X * 2;\n    let offsetX: number;\n\n    // Adjust the offset, depending on the option padding.\n    if (this.multiple) {\n      offsetX = SELECT_MULTIPLE_PANEL_PADDING_X;\n    } else {\n      let selected = this._selectionModel.selected[0] || this.options.first;\n      offsetX = selected && selected.group ? SELECT_PANEL_INDENT_PADDING_X : SELECT_PANEL_PADDING_X;\n    }\n\n    // Invert the offset in LTR.\n    if (!isRtl) {\n      offsetX *= -1;\n    }\n\n    // Determine how much the select overflows on each side.\n    const leftOverflow = 0 - (overlayRect.left + offsetX - (isRtl ? paddingWidth : 0));\n    const rightOverflow = overlayRect.right + offsetX - viewportSize.width\n                          + (isRtl ? 0 : paddingWidth);\n\n    // If the element overflows on either side, reduce the offset to allow it to fit.\n    if (leftOverflow > 0) {\n      offsetX += leftOverflow + SELECT_PANEL_VIEWPORT_PADDING;\n    } else if (rightOverflow > 0) {\n      offsetX -= rightOverflow + SELECT_PANEL_VIEWPORT_PADDING;\n    }\n\n    // Set the offset directly in order to avoid having to go through change detection and\n    // potentially triggering \"changed after it was checked\" errors. Round the value to avoid\n    // blurry content in some browsers.\n    this.overlayDir.offsetX = Math.round(offsetX);\n    this.overlayDir.overlayRef.updatePosition();\n  }\n\n  /**\n   * Calculates the y-offset of the select's overlay panel in relation to the\n   * top start corner of the trigger. It has to be adjusted in order for the\n   * selected option to be aligned over the trigger when the panel opens.\n   */\n  private _calculateOverlayOffsetY(selectedIndex: number, scrollBuffer: number,\n                                  maxScroll: number): number {\n    const itemHeight = this._getItemHeight();\n    const optionHeightAdjustment = (itemHeight - this._triggerRect.height) / 2;\n    const maxOptionsDisplayed = Math.floor(SELECT_PANEL_MAX_HEIGHT / itemHeight);\n    let optionOffsetFromPanelTop: number;\n\n    // Disable offset if requested by user by returning 0 as value to offset\n    if (this._disableOptionCentering) {\n      return 0;\n    }\n\n    if (this._scrollTop === 0) {\n      optionOffsetFromPanelTop = selectedIndex * itemHeight;\n    } else if (this._scrollTop === maxScroll) {\n      const firstDisplayedIndex = this._getItemCount() - maxOptionsDisplayed;\n      const selectedDisplayIndex = selectedIndex - firstDisplayedIndex;\n\n      // The first item is partially out of the viewport. Therefore we need to calculate what\n      // portion of it is shown in the viewport and account for it in our offset.\n      let partialItemHeight =\n          itemHeight - (this._getItemCount() * itemHeight - SELECT_PANEL_MAX_HEIGHT) % itemHeight;\n\n      // Because the panel height is longer than the height of the options alone,\n      // there is always extra padding at the top or bottom of the panel. When\n      // scrolled to the very bottom, this padding is at the top of the panel and\n      // must be added to the offset.\n      optionOffsetFromPanelTop = selectedDisplayIndex * itemHeight + partialItemHeight;\n    } else {\n      // If the option was scrolled to the middle of the panel using a scroll buffer,\n      // its offset will be the scroll buffer minus the half height that was added to\n      // center it.\n      optionOffsetFromPanelTop = scrollBuffer - itemHeight / 2;\n    }\n\n    // The final offset is the option's offset from the top, adjusted for the height difference,\n    // multiplied by -1 to ensure that the overlay moves in the correct direction up the page.\n    // The value is rounded to prevent some browsers from blurring the content.\n    return Math.round(optionOffsetFromPanelTop * -1 - optionHeightAdjustment);\n  }\n\n  /**\n   * Checks that the attempted overlay position will fit within the viewport.\n   * If it will not fit, tries to adjust the scroll position and the associated\n   * y-offset so the panel can open fully on-screen. If it still won't fit,\n   * sets the offset back to 0 to allow the fallback position to take over.\n   */\n  private _checkOverlayWithinViewport(maxScroll: number): void {\n    const itemHeight = this._getItemHeight();\n    const viewportSize = this._viewportRuler.getViewportSize();\n\n    const topSpaceAvailable = this._triggerRect.top - SELECT_PANEL_VIEWPORT_PADDING;\n    const bottomSpaceAvailable =\n        viewportSize.height - this._triggerRect.bottom - SELECT_PANEL_VIEWPORT_PADDING;\n\n    const panelHeightTop = Math.abs(this._offsetY);\n    const totalPanelHeight =\n        Math.min(this._getItemCount() * itemHeight, SELECT_PANEL_MAX_HEIGHT);\n    const panelHeightBottom = totalPanelHeight - panelHeightTop - this._triggerRect.height;\n\n    if (panelHeightBottom > bottomSpaceAvailable) {\n      this._adjustPanelUp(panelHeightBottom, bottomSpaceAvailable);\n    } else if (panelHeightTop > topSpaceAvailable) {\n     this._adjustPanelDown(panelHeightTop, topSpaceAvailable, maxScroll);\n    } else {\n      this._transformOrigin = this._getOriginBasedOnOption();\n    }\n  }\n\n  /** Adjusts the overlay panel up to fit in the viewport. */\n  private _adjustPanelUp(panelHeightBottom: number, bottomSpaceAvailable: number) {\n    // Browsers ignore fractional scroll offsets, so we need to round.\n    const distanceBelowViewport = Math.round(panelHeightBottom - bottomSpaceAvailable);\n\n    // Scrolls the panel up by the distance it was extending past the boundary, then\n    // adjusts the offset by that amount to move the panel up into the viewport.\n    this._scrollTop -= distanceBelowViewport;\n    this._offsetY -= distanceBelowViewport;\n    this._transformOrigin = this._getOriginBasedOnOption();\n\n    // If the panel is scrolled to the very top, it won't be able to fit the panel\n    // by scrolling, so set the offset to 0 to allow the fallback position to take\n    // effect.\n    if (this._scrollTop <= 0) {\n      this._scrollTop = 0;\n      this._offsetY = 0;\n      this._transformOrigin = `50% bottom 0px`;\n    }\n  }\n\n  /** Adjusts the overlay panel down to fit in the viewport. */\n  private _adjustPanelDown(panelHeightTop: number, topSpaceAvailable: number,\n                           maxScroll: number) {\n    // Browsers ignore fractional scroll offsets, so we need to round.\n    const distanceAboveViewport = Math.round(panelHeightTop - topSpaceAvailable);\n\n    // Scrolls the panel down by the distance it was extending past the boundary, then\n    // adjusts the offset by that amount to move the panel down into the viewport.\n    this._scrollTop += distanceAboveViewport;\n    this._offsetY += distanceAboveViewport;\n    this._transformOrigin = this._getOriginBasedOnOption();\n\n    // If the panel is scrolled to the very bottom, it won't be able to fit the\n    // panel by scrolling, so set the offset to 0 to allow the fallback position\n    // to take effect.\n    if (this._scrollTop >= maxScroll) {\n      this._scrollTop = maxScroll;\n      this._offsetY = 0;\n      this._transformOrigin = `50% top 0px`;\n      return;\n    }\n  }\n\n  /** Sets the transform origin point based on the selected option. */\n  private _getOriginBasedOnOption(): string {\n    const itemHeight = this._getItemHeight();\n    const optionHeightAdjustment = (itemHeight - this._triggerRect.height) / 2;\n    const originY = Math.abs(this._offsetY) - optionHeightAdjustment + itemHeight / 2;\n    return `50% ${originY}px 0px`;\n  }\n\n  /** Calculates the amount of items in the select. This includes options and group labels. */\n  private _getItemCount(): number {\n    return this.options.length + this.optionGroups.length;\n  }\n\n  /** Calculates the height of the select's options. */\n  private _getItemHeight(): number {\n    return this._triggerFontSize * SELECT_ITEM_HEIGHT_EM;\n  }\n\n  /**\n   * Implemented as part of MatFormFieldControl.\n   * @docs-private\n   */\n  setDescribedByIds(ids: string[]) {\n    this._ariaDescribedby = ids.join(' ');\n  }\n\n  /**\n   * Implemented as part of MatFormFieldControl.\n   * @docs-private\n   */\n  onContainerClick() {\n    this.focus();\n    this.open();\n  }\n\n  /**\n   * Implemented as part of MatFormFieldControl.\n   * @docs-private\n   */\n  get shouldLabelFloat(): boolean {\n    return this._panelOpen || !this.empty;\n  }\n\n  static ngAcceptInputType_required: BooleanInput;\n  static ngAcceptInputType_multiple: BooleanInput;\n  static ngAcceptInputType_disableOptionCentering: BooleanInput;\n  static ngAcceptInputType_typeaheadDebounceInterval: NumberInput;\n  static ngAcceptInputType_disabled: BooleanInput;\n  static ngAcceptInputType_disableRipple: BooleanInput;\n}\n"]}
|