@angular/material 17.1.0-next.2 → 17.1.0-next.3

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (48) hide show
  1. package/button/_button-base.scss +14 -2
  2. package/button/_button-theme.scss +11 -11
  3. package/button/_fab-theme.scss +1 -8
  4. package/button/index.d.ts +31 -2
  5. package/core/index.d.ts +6 -2
  6. package/core/tokens/m2/mat/_fab.scss +3 -0
  7. package/core/tokens/m2/mat/_filled-button.scss +3 -0
  8. package/core/tokens/m2/mat/_icon-button.scss +3 -0
  9. package/core/tokens/m2/mat/_outlined-button.scss +3 -0
  10. package/core/tokens/m2/mat/_protected-button.scss +3 -0
  11. package/core/tokens/m2/mat/_tab-header.scss +6 -1
  12. package/core/tokens/m2/mat/_text-button.scss +3 -0
  13. package/core/tokens/m2/mdc/_filled-button.scss +7 -6
  14. package/core/tokens/m2/mdc/_outlined-button.scss +7 -7
  15. package/core/tokens/m2/mdc/_protected-button.scss +7 -6
  16. package/core/tokens/m2/mdc/_text-button.scss +7 -8
  17. package/esm2022/button/button-base.mjs +41 -14
  18. package/esm2022/button/button.mjs +5 -5
  19. package/esm2022/button/fab.mjs +9 -9
  20. package/esm2022/button/icon-button.mjs +5 -5
  21. package/esm2022/button/public-api.mjs +2 -1
  22. package/esm2022/button/testing/button-harness.mjs +4 -3
  23. package/esm2022/chips/chip-listbox.mjs +2 -3
  24. package/esm2022/chips/chip.mjs +2 -1
  25. package/esm2022/core/private/ripple-loader.mjs +27 -9
  26. package/esm2022/core/version.mjs +1 -1
  27. package/esm2022/tabs/tab-group.mjs +11 -2
  28. package/esm2022/tabs/tab-header.mjs +3 -3
  29. package/esm2022/tabs/tab-nav-bar/tab-nav-bar.mjs +3 -3
  30. package/fesm2022/button/testing.mjs +3 -2
  31. package/fesm2022/button/testing.mjs.map +1 -1
  32. package/fesm2022/button.mjs +57 -30
  33. package/fesm2022/button.mjs.map +1 -1
  34. package/fesm2022/chips.mjs +2 -2
  35. package/fesm2022/chips.mjs.map +1 -1
  36. package/fesm2022/core.mjs +27 -9
  37. package/fesm2022/core.mjs.map +1 -1
  38. package/fesm2022/tabs.mjs +14 -5
  39. package/fesm2022/tabs.mjs.map +1 -1
  40. package/package.json +2 -2
  41. package/prebuilt-themes/deeppurple-amber.css +1 -1
  42. package/prebuilt-themes/indigo-pink.css +1 -1
  43. package/prebuilt-themes/pink-bluegrey.css +1 -1
  44. package/prebuilt-themes/purple-green.css +1 -1
  45. package/schematics/ng-add/index.js +1 -1
  46. package/schematics/ng-add/index.mjs +1 -1
  47. package/tabs/_tabs-common.scss +20 -1
  48. package/tabs/index.d.ts +5 -1
@@ -68,12 +68,17 @@
68
68
  @include token-utils.create-token-slot(background-color, state-layer-color);
69
69
  }
70
70
 
71
+ &.mat-mdc-button-disabled .mat-mdc-button-persistent-ripple::before {
72
+ @include token-utils.create-token-slot(background-color, disabled-state-layer-color);
73
+ }
74
+
71
75
  &:hover .mat-mdc-button-persistent-ripple::before {
72
76
  @include token-utils.create-token-slot(opacity, hover-state-layer-opacity);
73
77
  }
74
78
 
75
79
  &.cdk-program-focused,
76
- &.cdk-keyboard-focused {
80
+ &.cdk-keyboard-focused,
81
+ &.mat-mdc-button-disabled-interactive:focus {
77
82
  .mat-mdc-button-persistent-ripple::before {
78
83
  @include token-utils.create-token-slot(opacity, focus-state-layer-opacity);
79
84
  }
@@ -91,11 +96,18 @@
91
96
  // and note that having pointer-events may have unintended side-effects, e.g. allowing the user
92
97
  // to click the target underneath the button.
93
98
  @mixin mat-private-button-disabled() {
94
- &[disabled] {
99
+ // `[disabled]` shouldn't be necessary, but we keep it to maintain
100
+ // compatibility with apps setting it through host bindings.
101
+ &[disabled],
102
+ &.mat-mdc-button-disabled {
95
103
  cursor: default;
96
104
  pointer-events: none;
97
105
  @content;
98
106
  }
107
+
108
+ &.mat-mdc-button-disabled-interactive {
109
+ pointer-events: auto;
110
+ }
99
111
  }
100
112
 
101
113
  // Hides the touch target on lower densities.
@@ -1,4 +1,3 @@
1
- @use '@material/button/button' as mdc-button;
2
1
  @use '@material/button/button-text-theme' as mdc-button-text-theme;
3
2
  @use '@material/button/button-filled-theme' as mdc-button-filled-theme;
4
3
  @use '@material/button/button-protected-theme' as mdc-button-protected-theme;
@@ -19,7 +18,7 @@
19
18
  @use '../core/tokens/m2/mat/protected-button' as tokens-mat-protected-button;
20
19
  @use '../core/tokens/m2/mdc/text-button' as tokens-mdc-text-button;
21
20
  @use '../core/tokens/m2/mat/text-button' as tokens-mat-text-button;
22
-
21
+ @use '../core/style/sass-utils';
23
22
 
24
23
  @mixin _text-button-variant($theme, $palette) {
25
24
  $mdc-tokens: if($palette,
@@ -165,15 +164,16 @@
165
164
  }
166
165
 
167
166
  @mixin typography($theme) {
168
- @include mdc-helpers.using-mdc-typography($theme) {
169
- @include mdc-button.without-ripple($query: mdc-helpers.$mdc-typography-styles-query);
170
- }
171
-
172
- .mat-mdc-button,
173
- .mat-mdc-raised-button,
174
- .mat-mdc-outlined-button,
175
- .mat-mdc-unelevated-button {
176
- line-height: inherit;
167
+ @include sass-utils.current-selector-or-root() {
168
+ $text-typography-tokens: tokens-mdc-text-button.get-typography-tokens($theme);
169
+ $filled-typography-tokens: tokens-mdc-filled-button.get-typography-tokens($theme);
170
+ $outlined-typography-tokens: tokens-mdc-outlined-button.get-typography-tokens($theme);
171
+ $protected-typography-tokens: tokens-mdc-protected-button.get-typography-tokens($theme);
172
+
173
+ @include mdc-button-text-theme.theme($text-typography-tokens);
174
+ @include mdc-button-filled-theme.theme($filled-typography-tokens);
175
+ @include mdc-button-outlined-theme.theme($outlined-typography-tokens);
176
+ @include mdc-button-protected-theme.theme($protected-typography-tokens);
177
177
  }
178
178
  }
179
179
 
@@ -1,8 +1,6 @@
1
1
  @use 'sass:map';
2
- @use '@material/fab/fab' as mdc-fab;
3
2
  @use '@material/fab/fab-theme' as mdc-fab-theme;
4
3
  @use '@material/fab/extended-fab-theme' as mdc-extended-fab-theme;
5
- @use '../core/mdc-helpers/mdc-helpers';
6
4
  @use '../core/style/sass-utils';
7
5
  @use '../core/theming/theming';
8
6
  @use '../core/theming/inspection';
@@ -73,13 +71,8 @@
73
71
  @include _theme-from-tokens(inspection.get-theme-tokens($theme, typography));
74
72
  }
75
73
  @else {
76
- @include mdc-helpers.using-mdc-typography($theme) {
77
- @include mdc-fab.without-ripple($query: mdc-helpers.$mdc-typography-styles-query);
78
- }
79
-
80
- $typography-tokens: tokens-mdc-extended-fab.get-typography-tokens($theme);
81
74
  @include sass-utils.current-selector-or-root() {
82
- @include mdc-extended-fab-theme.theme($typography-tokens);
75
+ @include mdc-extended-fab-theme.theme(tokens-mdc-extended-fab.get-typography-tokens($theme));
83
76
  }
84
77
  }
85
78
  }
package/button/index.d.ts CHANGED
@@ -38,6 +38,9 @@ declare namespace i4 {
38
38
  }
39
39
  }
40
40
 
41
+ /** Injection token that can be used to provide the default options the button component. */
42
+ export declare const MAT_BUTTON_CONFIG: InjectionToken<MatButtonConfig>;
43
+
41
44
  /** Injection token to be used to override the default options for FAB. */
42
45
  export declare const MAT_FAB_DEFAULT_OPTIONS: InjectionToken<MatFabDefaultOptions>;
43
46
 
@@ -69,6 +72,7 @@ declare class MatAnchorBase extends MatButtonBase implements OnInit, OnDestroy {
69
72
  ngOnInit(): void;
70
73
  ngOnDestroy(): void;
71
74
  _haltDisabledEvents: (event: Event) => void;
75
+ protected _getAriaDisabled(): boolean;
72
76
  static ɵfac: i0.ɵɵFactoryDeclaration<MatAnchorBase, never>;
73
77
  static ɵdir: i0.ɵɵDirectiveDeclaration<MatAnchorBase, never, never, { "tabIndex": { "alias": "tabIndex"; "required": false; }; }, {}, never, never, false, never>;
74
78
  static ngAcceptInputType_tabIndex: unknown;
@@ -116,19 +120,44 @@ declare class MatButtonBase implements AfterViewInit, OnDestroy {
116
120
  get disableRipple(): boolean;
117
121
  set disableRipple(value: any);
118
122
  private _disableRipple;
123
+ /** Whether the button is disabled. */
119
124
  get disabled(): boolean;
120
125
  set disabled(value: any);
121
126
  private _disabled;
127
+ /** `aria-disabled` value of the button. */
128
+ ariaDisabled: boolean | undefined;
129
+ /**
130
+ * Natively disabled buttons prevent focus and any pointer events from reaching the button.
131
+ * In some scenarios this might not be desirable, because it can prevent users from finding out
132
+ * why the button is disabled (e.g. via tooltip).
133
+ *
134
+ * Enabling this input will change the button so that it is styled to be disabled and will be
135
+ * marked as `aria-disabled`, but it will allow the button to receive events and focus.
136
+ *
137
+ * Note that by enabling this, you need to set the `tabindex` yourself if the button isn't
138
+ * meant to be tabbable and you have to prevent the button action (e.g. form submissions).
139
+ */
140
+ disabledInteractive: boolean;
122
141
  constructor(_elementRef: ElementRef, _platform: Platform, _ngZone: NgZone, _animationMode?: string | undefined);
123
142
  ngAfterViewInit(): void;
124
143
  ngOnDestroy(): void;
125
144
  /** Focuses the button. */
126
- focus(_origin?: FocusOrigin, options?: FocusOptions): void;
145
+ focus(origin?: FocusOrigin, options?: FocusOptions): void;
146
+ protected _getAriaDisabled(): boolean | null;
147
+ protected _getDisabledAttribute(): true | null;
127
148
  private _updateRippleDisabled;
128
149
  static ɵfac: i0.ɵɵFactoryDeclaration<MatButtonBase, never>;
129
- static ɵdir: i0.ɵɵDirectiveDeclaration<MatButtonBase, never, never, { "color": { "alias": "color"; "required": false; }; "disableRipple": { "alias": "disableRipple"; "required": false; }; "disabled": { "alias": "disabled"; "required": false; }; }, {}, never, never, false, never>;
150
+ static ɵdir: i0.ɵɵDirectiveDeclaration<MatButtonBase, never, never, { "color": { "alias": "color"; "required": false; }; "disableRipple": { "alias": "disableRipple"; "required": false; }; "disabled": { "alias": "disabled"; "required": false; }; "ariaDisabled": { "alias": "aria-disabled"; "required": false; }; "disabledInteractive": { "alias": "disabledInteractive"; "required": false; }; }, {}, never, never, false, never>;
130
151
  static ngAcceptInputType_disableRipple: unknown;
131
152
  static ngAcceptInputType_disabled: unknown;
153
+ static ngAcceptInputType_ariaDisabled: unknown;
154
+ static ngAcceptInputType_disabledInteractive: unknown;
155
+ }
156
+
157
+ /** Object that can be used to configure the default options for the button component. */
158
+ export declare interface MatButtonConfig {
159
+ /** Whether disabled buttons should be interactive. */
160
+ disabledInteractive?: boolean;
132
161
  }
133
162
 
134
163
  export declare class MatButtonModule {
package/core/index.d.ts CHANGED
@@ -815,6 +815,8 @@ export declare class MatRipple implements OnInit, OnDestroy, RippleTarget {
815
815
  *
816
816
  * This service allows us to avoid eagerly creating & attaching MatRipples.
817
817
  * It works by creating & attaching a ripple only when a component is first interacted with.
818
+ *
819
+ * @docs-private
818
820
  */
819
821
  export declare class MatRippleLoader implements OnDestroy {
820
822
  private _document;
@@ -822,6 +824,7 @@ export declare class MatRippleLoader implements OnDestroy {
822
824
  private _globalRippleOptions;
823
825
  private _platform;
824
826
  private _ngZone;
827
+ private _hosts;
825
828
  constructor();
826
829
  ngOnDestroy(): void;
827
830
  /**
@@ -842,8 +845,9 @@ export declare class MatRippleLoader implements OnDestroy {
842
845
  /** Handles creating and attaching component internals when a component it is initially interacted with. */
843
846
  private _onInteraction;
844
847
  /** Creates a MatRipple and appends it to the given element. */
845
- createRipple(host: HTMLElement): MatRipple | undefined;
846
- attachRipple(host: Element, ripple: MatRipple): void;
848
+ private _createRipple;
849
+ attachRipple(host: HTMLElement, ripple: MatRipple): void;
850
+ destroyRipple(host: HTMLElement): void;
847
851
  static ɵfac: i0.ɵɵFactoryDeclaration<MatRippleLoader, never>;
848
852
  static ɵprov: i0.ɵɵInjectableDeclaration<MatRippleLoader>;
849
853
  }
@@ -30,6 +30,9 @@ $prefix: (mat, fab);
30
30
  // Color of the element that shows the hover, focus and pressed states.
31
31
  state-layer-color: $on-surface,
32
32
 
33
+ // Color of the element that shows the hover, focus and pressed states while disabled.
34
+ disabled-state-layer-color: $on-surface,
35
+
33
36
  // Color of the ripple element.
34
37
  ripple-color: rgba($on-surface, 0.1),
35
38
 
@@ -27,6 +27,9 @@ $prefix: (mat, filled-button);
27
27
  // Color of the element that shows the hover, focus and pressed states.
28
28
  state-layer-color: $on-surface,
29
29
 
30
+ // Color of the element that shows the hover, focus and pressed states while disabled.
31
+ disabled-state-layer-color: $on-surface,
32
+
30
33
  // Color of the ripple element.
31
34
  ripple-color: rgba($on-surface, 0.1),
32
35
 
@@ -27,6 +27,9 @@ $prefix: (mat, icon-button);
27
27
  // Color of the element that shows the hover, focus and pressed states.
28
28
  state-layer-color: $on-surface,
29
29
 
30
+ // Color of the element that shows the hover, focus and pressed states while disabled.
31
+ disabled-state-layer-color: $on-surface,
32
+
30
33
  // Color of the ripple element.
31
34
  ripple-color: rgba($on-surface, 0.1),
32
35
 
@@ -27,6 +27,9 @@ $prefix: (mat, outlined-button);
27
27
  // Color of the element that shows the hover, focus and pressed states.
28
28
  state-layer-color: $on-surface,
29
29
 
30
+ // Color of the element that shows the hover, focus and pressed states while disabled.
31
+ disabled-state-layer-color: $on-surface,
32
+
30
33
  // Color of the ripple element.
31
34
  ripple-color: rgba($on-surface, 0.1),
32
35
 
@@ -27,6 +27,9 @@ $prefix: (mat, protected-button);
27
27
  // Color of the element that shows the hover, focus and pressed states.
28
28
  state-layer-color: $on-surface,
29
29
 
30
+ // Color of the element that shows the hover, focus and pressed states while disabled.
31
+ disabled-state-layer-color: $on-surface,
32
+
30
33
  // Color of the ripple element.
31
34
  ripple-color: rgba($on-surface, 0.1),
32
35
 
@@ -8,7 +8,12 @@ $prefix: (mat, tab-header);
8
8
  // Tokens that can't be configured through Angular Material's current theming API,
9
9
  // but may be in a future version of the theming API.
10
10
  @function get-unthemable-tokens() {
11
- @return ();
11
+ @return (
12
+ // For some period of time, the MDC tabs removed the divider. This has been added back in
13
+ // and will be present in M3.
14
+ divider-color: transparent,
15
+ divider-height: 0,
16
+ );
12
17
  }
13
18
 
14
19
  // Tokens that can be configured through Angular Material's color theming API.
@@ -27,6 +27,9 @@ $prefix: (mat, text-button);
27
27
  // Color of the element that shows the hover, focus and pressed states.
28
28
  state-layer-color: $on-surface,
29
29
 
30
+ // Color of the element that shows the hover, focus and pressed states while disabled.
31
+ disabled-state-layer-color: $on-surface,
32
+
30
33
  // Color of the ripple element.
31
34
  ripple-color: rgba($on-surface, 0.1),
32
35
 
@@ -35,11 +35,6 @@ $prefix: (mdc, filled-button);
35
35
  container-shadow-color: null,
36
36
  focus-label-text-color: null,
37
37
  hover-label-text-color: null,
38
- label-text-font: null,
39
- label-text-size: null,
40
- label-text-tracking: null,
41
- label-text-transform: null,
42
- label-text-weight: null,
43
38
  pressed-label-text-color: null,
44
39
  with-icon-disabled-icon-color: null,
45
40
  with-icon-focus-icon-color: null,
@@ -81,7 +76,13 @@ $prefix: (mdc, filled-button);
81
76
 
82
77
  // Tokens that can be configured through Angular Material's typography theming API.
83
78
  @function get-typography-tokens($theme) {
84
- @return ();
79
+ @return (
80
+ label-text-font: inspection.get-theme-typography($theme, button, font-family),
81
+ label-text-size: inspection.get-theme-typography($theme, button, font-size),
82
+ label-text-tracking: inspection.get-theme-typography($theme, button, letter-spacing),
83
+ label-text-weight: inspection.get-theme-typography($theme, button, font-weight),
84
+ label-text-transform: none,
85
+ );
85
86
  }
86
87
 
87
88
  // Tokens that can be configured through Angular Material's density theming API.
@@ -49,12 +49,6 @@ $prefix: (mdc, outlined-button);
49
49
  with-icon-focus-icon-color: null,
50
50
  with-icon-pressed-icon-color: null,
51
51
  with-icon-disabled-icon-color: null,
52
-
53
- label-text-size: null,
54
- label-text-font: null,
55
- label-text-weight: null,
56
- label-text-tracking: null,
57
- label-text-transform: null
58
52
  );
59
53
  }
60
54
 
@@ -87,7 +81,13 @@ $prefix: (mdc, outlined-button);
87
81
 
88
82
  // Tokens that can be configured through Angular Material's typography theming API.
89
83
  @function get-typography-tokens($theme) {
90
- @return ();
84
+ @return (
85
+ label-text-font: inspection.get-theme-typography($theme, button, font-family),
86
+ label-text-size: inspection.get-theme-typography($theme, button, font-size),
87
+ label-text-tracking: inspection.get-theme-typography($theme, button, letter-spacing),
88
+ label-text-weight: inspection.get-theme-typography($theme, button, font-weight),
89
+ label-text-transform: none,
90
+ );
91
91
  }
92
92
 
93
93
  // Tokens that can be configured through Angular Material's density theming API.
@@ -33,11 +33,6 @@ $prefix: (mdc, protected-button);
33
33
  disabled-container-elevation: null,
34
34
  focus-container-elevation: null,
35
35
  pressed-container-elevation: null,
36
- label-text-font: null,
37
- label-text-size: null,
38
- label-text-tracking: null,
39
- label-text-transform: null,
40
- label-text-weight: null,
41
36
  with-icon-icon-size: null,
42
37
  focus-label-text-color: null,
43
38
  hover-label-text-color: null,
@@ -80,7 +75,13 @@ $prefix: (mdc, protected-button);
80
75
 
81
76
  // Tokens that can be configured through Angular Material's typography theming API.
82
77
  @function get-typography-tokens($theme) {
83
- @return ();
78
+ @return (
79
+ label-text-font: inspection.get-theme-typography($theme, button, font-family),
80
+ label-text-size: inspection.get-theme-typography($theme, button, font-size),
81
+ label-text-tracking: inspection.get-theme-typography($theme, button, letter-spacing),
82
+ label-text-weight: inspection.get-theme-typography($theme, button, font-weight),
83
+ label-text-transform: none,
84
+ );
84
85
  }
85
86
 
86
87
  // Tokens that can be configured through Angular Material's density theming API.
@@ -21,13 +21,6 @@ $prefix: (mdc, text-button);
21
21
  // prevent the buttons from collapsing if a density mixin isn't included.
22
22
  container-height: 36px,
23
23
 
24
- // TODO: handle these through the typography styles eventually.
25
- label-text-font: null,
26
- label-text-size: null,
27
- label-text-tracking: null,
28
- label-text-transform: null,
29
- label-text-weight: null,
30
-
31
24
  // =============================================================================================
32
25
  // = TOKENS NOT USED IN ANGULAR MATERIAL =
33
26
  // =============================================================================================
@@ -74,7 +67,13 @@ $prefix: (mdc, text-button);
74
67
 
75
68
  // Tokens that can be configured through Angular Material's typography theming API.
76
69
  @function get-typography-tokens($theme) {
77
- @return ();
70
+ @return (
71
+ label-text-font: inspection.get-theme-typography($theme, button, font-family),
72
+ label-text-size: inspection.get-theme-typography($theme, button, font-size),
73
+ label-text-tracking: inspection.get-theme-typography($theme, button, letter-spacing),
74
+ label-text-weight: inspection.get-theme-typography($theme, button, font-weight),
75
+ label-text-transform: none,
76
+ );
78
77
  }
79
78
 
80
79
  // Tokens that can be configured through Angular Material's density theming API.
@@ -7,13 +7,18 @@
7
7
  */
8
8
  import { FocusMonitor } from '@angular/cdk/a11y';
9
9
  import { Platform } from '@angular/cdk/platform';
10
- import { booleanAttribute, Directive, ElementRef, inject, Input, NgZone, numberAttribute, } from '@angular/core';
10
+ import { booleanAttribute, Directive, ElementRef, inject, InjectionToken, Input, NgZone, numberAttribute, } from '@angular/core';
11
11
  import { MatRippleLoader } from '@angular/material/core';
12
12
  import * as i0 from "@angular/core";
13
13
  import * as i1 from "@angular/cdk/platform";
14
+ /** Injection token that can be used to provide the default options the button component. */
15
+ export const MAT_BUTTON_CONFIG = new InjectionToken('MAT_BUTTON_CONFIG');
14
16
  /** Shared host configuration for all buttons */
15
17
  export const MAT_BUTTON_HOST = {
16
- '[attr.disabled]': 'disabled || null',
18
+ '[attr.disabled]': '_getDisabledAttribute()',
19
+ '[attr.aria-disabled]': '_getAriaDisabled()',
20
+ '[class.mat-mdc-button-disabled]': 'disabled',
21
+ '[class.mat-mdc-button-disabled-interactive]': 'disabledInteractive',
17
22
  '[class._mat-animation-noopable]': '_animationMode === "NoopAnimations"',
18
23
  // MDC automatically applies the primary theme color to the button, but we want to support
19
24
  // an unthemed version. If color is undefined, apply a CSS class that makes it easy to
@@ -76,6 +81,7 @@ export class MatButtonBase {
76
81
  this._disableRipple = value;
77
82
  this._updateRippleDisabled();
78
83
  }
84
+ /** Whether the button is disabled. */
79
85
  get disabled() {
80
86
  return this._disabled;
81
87
  }
@@ -98,11 +104,11 @@ export class MatButtonBase {
98
104
  this._isFab = false;
99
105
  this._disableRipple = false;
100
106
  this._disabled = false;
101
- this._rippleLoader?.configureRipple(this._elementRef.nativeElement, {
102
- className: 'mat-mdc-button-ripple',
103
- });
104
- const element = this._elementRef.nativeElement;
107
+ const config = inject(MAT_BUTTON_CONFIG, { optional: true });
108
+ const element = _elementRef.nativeElement;
105
109
  const classList = element.classList;
110
+ this.disabledInteractive = config?.disabledInteractive ?? false;
111
+ this._rippleLoader?.configureRipple(element, { className: 'mat-mdc-button-ripple' });
106
112
  // For each of the variant selectors that is present in the button's host
107
113
  // attributes, add the correct corresponding MDC classes.
108
114
  for (const { attribute, mdcClasses } of HOST_SELECTOR_MDC_CLASS_PAIR) {
@@ -116,21 +122,31 @@ export class MatButtonBase {
116
122
  }
117
123
  ngOnDestroy() {
118
124
  this._focusMonitor.stopMonitoring(this._elementRef);
125
+ this._rippleLoader?.destroyRipple(this._elementRef.nativeElement);
119
126
  }
120
127
  /** Focuses the button. */
121
- focus(_origin = 'program', options) {
122
- if (_origin) {
123
- this._focusMonitor.focusVia(this._elementRef.nativeElement, _origin, options);
128
+ focus(origin = 'program', options) {
129
+ if (origin) {
130
+ this._focusMonitor.focusVia(this._elementRef.nativeElement, origin, options);
124
131
  }
125
132
  else {
126
133
  this._elementRef.nativeElement.focus(options);
127
134
  }
128
135
  }
136
+ _getAriaDisabled() {
137
+ if (this.ariaDisabled != null) {
138
+ return this.ariaDisabled;
139
+ }
140
+ return this.disabled && this.disabledInteractive ? true : null;
141
+ }
142
+ _getDisabledAttribute() {
143
+ return this.disabledInteractive || !this.disabled ? null : true;
144
+ }
129
145
  _updateRippleDisabled() {
130
146
  this._rippleLoader?.setDisabled(this._elementRef.nativeElement, this.disableRipple || this.disabled);
131
147
  }
132
148
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.1.0-next.2", ngImport: i0, type: MatButtonBase, deps: "invalid", target: i0.ɵɵFactoryTarget.Directive }); }
133
- static { this.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "16.1.0", version: "17.1.0-next.2", type: MatButtonBase, inputs: { color: "color", disableRipple: ["disableRipple", "disableRipple", booleanAttribute], disabled: ["disabled", "disabled", booleanAttribute] }, ngImport: i0 }); }
149
+ static { this.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "16.1.0", version: "17.1.0-next.2", type: MatButtonBase, inputs: { color: "color", disableRipple: ["disableRipple", "disableRipple", booleanAttribute], disabled: ["disabled", "disabled", booleanAttribute], ariaDisabled: ["aria-disabled", "ariaDisabled", booleanAttribute], disabledInteractive: ["disabledInteractive", "disabledInteractive", booleanAttribute] }, ngImport: i0 }); }
134
150
  }
135
151
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.1.0-next.2", ngImport: i0, type: MatButtonBase, decorators: [{
136
152
  type: Directive
@@ -142,16 +158,24 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.1.0-next.2",
142
158
  }], disabled: [{
143
159
  type: Input,
144
160
  args: [{ transform: booleanAttribute }]
161
+ }], ariaDisabled: [{
162
+ type: Input,
163
+ args: [{ transform: booleanAttribute, alias: 'aria-disabled' }]
164
+ }], disabledInteractive: [{
165
+ type: Input,
166
+ args: [{ transform: booleanAttribute }]
145
167
  }] } });
146
168
  /** Shared host configuration for buttons using the `<a>` tag. */
147
169
  export const MAT_ANCHOR_HOST = {
148
- '[attr.disabled]': 'disabled || null',
170
+ '[attr.disabled]': '_getDisabledAttribute()',
171
+ '[class.mat-mdc-button-disabled]': 'disabled',
172
+ '[class.mat-mdc-button-disabled-interactive]': 'disabledInteractive',
149
173
  '[class._mat-animation-noopable]': '_animationMode === "NoopAnimations"',
150
174
  // Note that we ignore the user-specified tabindex when it's disabled for
151
175
  // consistency with the `mat-button` applied on native buttons where even
152
176
  // though they have an index, they're not tabbable.
153
- '[attr.tabindex]': 'disabled ? -1 : tabIndex',
154
- '[attr.aria-disabled]': 'disabled.toString()',
177
+ '[attr.tabindex]': 'disabled && !disabledInteractive ? -1 : tabIndex',
178
+ '[attr.aria-disabled]': '_getDisabledAttribute()',
155
179
  // MDC automatically applies the primary theme color to the button, but we want to support
156
180
  // an unthemed version. If color is undefined, apply a CSS class that makes it easy to
157
181
  // select and style this "theme".
@@ -184,6 +208,9 @@ export class MatAnchorBase extends MatButtonBase {
184
208
  super.ngOnDestroy();
185
209
  this._elementRef.nativeElement.removeEventListener('click', this._haltDisabledEvents);
186
210
  }
211
+ _getAriaDisabled() {
212
+ return this.ariaDisabled == null ? this.disabled : this.ariaDisabled;
213
+ }
187
214
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.1.0-next.2", ngImport: i0, type: MatAnchorBase, deps: "invalid", target: i0.ɵɵFactoryTarget.Directive }); }
188
215
  static { this.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "16.1.0", version: "17.1.0-next.2", type: MatAnchorBase, inputs: { tabIndex: ["tabIndex", "tabIndex", (value) => {
189
216
  return value == null ? undefined : numberAttribute(value);
@@ -199,4 +226,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.1.0-next.2",
199
226
  },
200
227
  }]
201
228
  }] } });
202
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"button-base.js","sourceRoot":"","sources":["../../../../../../src/material/button/button-base.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 {FocusMonitor, FocusOrigin} from '@angular/cdk/a11y';\nimport {Platform} from '@angular/cdk/platform';\nimport {\n  AfterViewInit,\n  booleanAttribute,\n  Directive,\n  ElementRef,\n  inject,\n  Input,\n  NgZone,\n  numberAttribute,\n  OnDestroy,\n  OnInit,\n} from '@angular/core';\nimport {MatRipple, MatRippleLoader} from '@angular/material/core';\n\n/** Shared host configuration for all buttons */\nexport const MAT_BUTTON_HOST = {\n  '[attr.disabled]': 'disabled || null',\n  '[class._mat-animation-noopable]': '_animationMode === \"NoopAnimations\"',\n  // MDC automatically applies the primary theme color to the button, but we want to support\n  // an unthemed version. If color is undefined, apply a CSS class that makes it easy to\n  // select and style this \"theme\".\n  '[class.mat-unthemed]': '!color',\n  // Add a class that applies to all buttons. This makes it easier to target if somebody\n  // wants to target all Material buttons.\n  '[class.mat-mdc-button-base]': 'true',\n  '[class]': 'color ? \"mat-\" + color : \"\"',\n};\n\n/** List of classes to add to buttons instances based on host attribute selector. */\nconst HOST_SELECTOR_MDC_CLASS_PAIR: {attribute: string; mdcClasses: string[]}[] = [\n  {\n    attribute: 'mat-button',\n    mdcClasses: ['mdc-button', 'mat-mdc-button'],\n  },\n  {\n    attribute: 'mat-flat-button',\n    mdcClasses: ['mdc-button', 'mdc-button--unelevated', 'mat-mdc-unelevated-button'],\n  },\n  {\n    attribute: 'mat-raised-button',\n    mdcClasses: ['mdc-button', 'mdc-button--raised', 'mat-mdc-raised-button'],\n  },\n  {\n    attribute: 'mat-stroked-button',\n    mdcClasses: ['mdc-button', 'mdc-button--outlined', 'mat-mdc-outlined-button'],\n  },\n  {\n    attribute: 'mat-fab',\n    mdcClasses: ['mdc-fab', 'mat-mdc-fab'],\n  },\n  {\n    attribute: 'mat-mini-fab',\n    mdcClasses: ['mdc-fab', 'mdc-fab--mini', 'mat-mdc-mini-fab'],\n  },\n  {\n    attribute: 'mat-icon-button',\n    mdcClasses: ['mdc-icon-button', 'mat-mdc-icon-button'],\n  },\n];\n\n/** Base class for all buttons.  */\n@Directive()\nexport class MatButtonBase implements AfterViewInit, OnDestroy {\n  private readonly _focusMonitor = inject(FocusMonitor);\n\n  /**\n   * Handles the lazy creation of the MatButton ripple.\n   * Used to improve initial load time of large applications.\n   */\n  _rippleLoader: MatRippleLoader = inject(MatRippleLoader);\n\n  /** Whether this button is a FAB. Used to apply the correct class on the ripple. */\n  _isFab = false;\n\n  /**\n   * Reference to the MatRipple instance of the button.\n   * @deprecated Considered an implementation detail. To be removed.\n   * @breaking-change 17.0.0\n   */\n  get ripple(): MatRipple {\n    return this._rippleLoader?.getRipple(this._elementRef.nativeElement)!;\n  }\n  set ripple(v: MatRipple) {\n    this._rippleLoader?.attachRipple(this._elementRef.nativeElement, v);\n  }\n\n  /** Theme color palette of the button */\n  @Input() color?: string | null;\n\n  /** Whether the ripple effect is disabled or not. */\n  @Input({transform: booleanAttribute})\n  get disableRipple(): boolean {\n    return this._disableRipple;\n  }\n  set disableRipple(value: any) {\n    this._disableRipple = value;\n    this._updateRippleDisabled();\n  }\n  private _disableRipple: boolean = false;\n\n  @Input({transform: booleanAttribute})\n  get disabled(): boolean {\n    return this._disabled;\n  }\n  set disabled(value: any) {\n    this._disabled = value;\n    this._updateRippleDisabled();\n  }\n  private _disabled: boolean = false;\n\n  constructor(\n    public _elementRef: ElementRef,\n    public _platform: Platform,\n    public _ngZone: NgZone,\n    public _animationMode?: string,\n  ) {\n    this._rippleLoader?.configureRipple(this._elementRef.nativeElement, {\n      className: 'mat-mdc-button-ripple',\n    });\n\n    const element = this._elementRef.nativeElement;\n    const classList = (element as HTMLElement).classList;\n\n    // For each of the variant selectors that is present in the button's host\n    // attributes, add the correct corresponding MDC classes.\n    for (const {attribute, mdcClasses} of HOST_SELECTOR_MDC_CLASS_PAIR) {\n      if (element.hasAttribute(attribute)) {\n        classList.add(...mdcClasses);\n      }\n    }\n  }\n\n  ngAfterViewInit() {\n    this._focusMonitor.monitor(this._elementRef, true);\n  }\n\n  ngOnDestroy() {\n    this._focusMonitor.stopMonitoring(this._elementRef);\n  }\n\n  /** Focuses the button. */\n  focus(_origin: FocusOrigin = 'program', options?: FocusOptions): void {\n    if (_origin) {\n      this._focusMonitor.focusVia(this._elementRef.nativeElement, _origin, options);\n    } else {\n      this._elementRef.nativeElement.focus(options);\n    }\n  }\n\n  private _updateRippleDisabled(): void {\n    this._rippleLoader?.setDisabled(\n      this._elementRef.nativeElement,\n      this.disableRipple || this.disabled,\n    );\n  }\n}\n\n/** Shared host configuration for buttons using the `<a>` tag. */\nexport const MAT_ANCHOR_HOST = {\n  '[attr.disabled]': 'disabled || null',\n  '[class._mat-animation-noopable]': '_animationMode === \"NoopAnimations\"',\n\n  // Note that we ignore the user-specified tabindex when it's disabled for\n  // consistency with the `mat-button` applied on native buttons where even\n  // though they have an index, they're not tabbable.\n  '[attr.tabindex]': 'disabled ? -1 : tabIndex',\n  '[attr.aria-disabled]': 'disabled.toString()',\n  // MDC automatically applies the primary theme color to the button, but we want to support\n  // an unthemed version. If color is undefined, apply a CSS class that makes it easy to\n  // select and style this \"theme\".\n  '[class.mat-unthemed]': '!color',\n  // Add a class that applies to all buttons. This makes it easier to target if somebody\n  // wants to target all Material buttons.\n  '[class.mat-mdc-button-base]': 'true',\n  '[class]': 'color ? \"mat-\" + color : \"\"',\n};\n\n/**\n * Anchor button base.\n */\n@Directive()\nexport class MatAnchorBase extends MatButtonBase implements OnInit, OnDestroy {\n  @Input({\n    transform: (value: unknown) => {\n      return value == null ? undefined : numberAttribute(value);\n    },\n  })\n  tabIndex: number;\n\n  constructor(elementRef: ElementRef, platform: Platform, ngZone: NgZone, animationMode?: string) {\n    super(elementRef, platform, ngZone, animationMode);\n  }\n\n  ngOnInit(): void {\n    this._ngZone.runOutsideAngular(() => {\n      this._elementRef.nativeElement.addEventListener('click', this._haltDisabledEvents);\n    });\n  }\n\n  override ngOnDestroy(): void {\n    super.ngOnDestroy();\n    this._elementRef.nativeElement.removeEventListener('click', this._haltDisabledEvents);\n  }\n\n  _haltDisabledEvents = (event: Event): void => {\n    // A disabled button shouldn't apply any actions\n    if (this.disabled) {\n      event.preventDefault();\n      event.stopImmediatePropagation();\n    }\n  };\n}\n"]}
229
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"button-base.js","sourceRoot":"","sources":["../../../../../../src/material/button/button-base.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 {FocusMonitor, FocusOrigin} from '@angular/cdk/a11y';\nimport {Platform} from '@angular/cdk/platform';\nimport {\n  AfterViewInit,\n  booleanAttribute,\n  Directive,\n  ElementRef,\n  inject,\n  InjectionToken,\n  Input,\n  NgZone,\n  numberAttribute,\n  OnDestroy,\n  OnInit,\n} from '@angular/core';\nimport {MatRipple, MatRippleLoader} from '@angular/material/core';\n\n/** Object that can be used to configure the default options for the button component. */\nexport interface MatButtonConfig {\n  /** Whether disabled buttons should be interactive. */\n  disabledInteractive?: boolean;\n}\n\n/** Injection token that can be used to provide the default options the button component. */\nexport const MAT_BUTTON_CONFIG = new InjectionToken<MatButtonConfig>('MAT_BUTTON_CONFIG');\n\n/** Shared host configuration for all buttons */\nexport const MAT_BUTTON_HOST = {\n  '[attr.disabled]': '_getDisabledAttribute()',\n  '[attr.aria-disabled]': '_getAriaDisabled()',\n  '[class.mat-mdc-button-disabled]': 'disabled',\n  '[class.mat-mdc-button-disabled-interactive]': 'disabledInteractive',\n  '[class._mat-animation-noopable]': '_animationMode === \"NoopAnimations\"',\n  // MDC automatically applies the primary theme color to the button, but we want to support\n  // an unthemed version. If color is undefined, apply a CSS class that makes it easy to\n  // select and style this \"theme\".\n  '[class.mat-unthemed]': '!color',\n  // Add a class that applies to all buttons. This makes it easier to target if somebody\n  // wants to target all Material buttons.\n  '[class.mat-mdc-button-base]': 'true',\n  '[class]': 'color ? \"mat-\" + color : \"\"',\n};\n\n/** List of classes to add to buttons instances based on host attribute selector. */\nconst HOST_SELECTOR_MDC_CLASS_PAIR: {attribute: string; mdcClasses: string[]}[] = [\n  {\n    attribute: 'mat-button',\n    mdcClasses: ['mdc-button', 'mat-mdc-button'],\n  },\n  {\n    attribute: 'mat-flat-button',\n    mdcClasses: ['mdc-button', 'mdc-button--unelevated', 'mat-mdc-unelevated-button'],\n  },\n  {\n    attribute: 'mat-raised-button',\n    mdcClasses: ['mdc-button', 'mdc-button--raised', 'mat-mdc-raised-button'],\n  },\n  {\n    attribute: 'mat-stroked-button',\n    mdcClasses: ['mdc-button', 'mdc-button--outlined', 'mat-mdc-outlined-button'],\n  },\n  {\n    attribute: 'mat-fab',\n    mdcClasses: ['mdc-fab', 'mat-mdc-fab'],\n  },\n  {\n    attribute: 'mat-mini-fab',\n    mdcClasses: ['mdc-fab', 'mdc-fab--mini', 'mat-mdc-mini-fab'],\n  },\n  {\n    attribute: 'mat-icon-button',\n    mdcClasses: ['mdc-icon-button', 'mat-mdc-icon-button'],\n  },\n];\n\n/** Base class for all buttons.  */\n@Directive()\nexport class MatButtonBase implements AfterViewInit, OnDestroy {\n  private readonly _focusMonitor = inject(FocusMonitor);\n\n  /**\n   * Handles the lazy creation of the MatButton ripple.\n   * Used to improve initial load time of large applications.\n   */\n  _rippleLoader: MatRippleLoader = inject(MatRippleLoader);\n\n  /** Whether this button is a FAB. Used to apply the correct class on the ripple. */\n  _isFab = false;\n\n  /**\n   * Reference to the MatRipple instance of the button.\n   * @deprecated Considered an implementation detail. To be removed.\n   * @breaking-change 17.0.0\n   */\n  get ripple(): MatRipple {\n    return this._rippleLoader?.getRipple(this._elementRef.nativeElement)!;\n  }\n  set ripple(v: MatRipple) {\n    this._rippleLoader?.attachRipple(this._elementRef.nativeElement, v);\n  }\n\n  /** Theme color palette of the button */\n  @Input() color?: string | null;\n\n  /** Whether the ripple effect is disabled or not. */\n  @Input({transform: booleanAttribute})\n  get disableRipple(): boolean {\n    return this._disableRipple;\n  }\n  set disableRipple(value: any) {\n    this._disableRipple = value;\n    this._updateRippleDisabled();\n  }\n  private _disableRipple: boolean = false;\n\n  /** Whether the button is disabled. */\n  @Input({transform: booleanAttribute})\n  get disabled(): boolean {\n    return this._disabled;\n  }\n  set disabled(value: any) {\n    this._disabled = value;\n    this._updateRippleDisabled();\n  }\n  private _disabled: boolean = false;\n\n  /** `aria-disabled` value of the button. */\n  @Input({transform: booleanAttribute, alias: 'aria-disabled'})\n  ariaDisabled: boolean | undefined;\n\n  /**\n   * Natively disabled buttons prevent focus and any pointer events from reaching the button.\n   * In some scenarios this might not be desirable, because it can prevent users from finding out\n   * why the button is disabled (e.g. via tooltip).\n   *\n   * Enabling this input will change the button so that it is styled to be disabled and will be\n   * marked as `aria-disabled`, but it will allow the button to receive events and focus.\n   *\n   * Note that by enabling this, you need to set the `tabindex` yourself if the button isn't\n   * meant to be tabbable and you have to prevent the button action (e.g. form submissions).\n   */\n  @Input({transform: booleanAttribute})\n  disabledInteractive: boolean;\n\n  constructor(\n    public _elementRef: ElementRef,\n    public _platform: Platform,\n    public _ngZone: NgZone,\n    public _animationMode?: string,\n  ) {\n    const config = inject(MAT_BUTTON_CONFIG, {optional: true});\n    const element = _elementRef.nativeElement;\n    const classList = (element as HTMLElement).classList;\n\n    this.disabledInteractive = config?.disabledInteractive ?? false;\n    this._rippleLoader?.configureRipple(element, {className: 'mat-mdc-button-ripple'});\n\n    // For each of the variant selectors that is present in the button's host\n    // attributes, add the correct corresponding MDC classes.\n    for (const {attribute, mdcClasses} of HOST_SELECTOR_MDC_CLASS_PAIR) {\n      if (element.hasAttribute(attribute)) {\n        classList.add(...mdcClasses);\n      }\n    }\n  }\n\n  ngAfterViewInit() {\n    this._focusMonitor.monitor(this._elementRef, true);\n  }\n\n  ngOnDestroy() {\n    this._focusMonitor.stopMonitoring(this._elementRef);\n    this._rippleLoader?.destroyRipple(this._elementRef.nativeElement);\n  }\n\n  /** Focuses the button. */\n  focus(origin: FocusOrigin = 'program', options?: FocusOptions): void {\n    if (origin) {\n      this._focusMonitor.focusVia(this._elementRef.nativeElement, origin, options);\n    } else {\n      this._elementRef.nativeElement.focus(options);\n    }\n  }\n\n  protected _getAriaDisabled() {\n    if (this.ariaDisabled != null) {\n      return this.ariaDisabled;\n    }\n\n    return this.disabled && this.disabledInteractive ? true : null;\n  }\n\n  protected _getDisabledAttribute() {\n    return this.disabledInteractive || !this.disabled ? null : true;\n  }\n\n  private _updateRippleDisabled(): void {\n    this._rippleLoader?.setDisabled(\n      this._elementRef.nativeElement,\n      this.disableRipple || this.disabled,\n    );\n  }\n}\n\n/** Shared host configuration for buttons using the `<a>` tag. */\nexport const MAT_ANCHOR_HOST = {\n  '[attr.disabled]': '_getDisabledAttribute()',\n  '[class.mat-mdc-button-disabled]': 'disabled',\n  '[class.mat-mdc-button-disabled-interactive]': 'disabledInteractive',\n  '[class._mat-animation-noopable]': '_animationMode === \"NoopAnimations\"',\n\n  // Note that we ignore the user-specified tabindex when it's disabled for\n  // consistency with the `mat-button` applied on native buttons where even\n  // though they have an index, they're not tabbable.\n  '[attr.tabindex]': 'disabled && !disabledInteractive ? -1 : tabIndex',\n  '[attr.aria-disabled]': '_getDisabledAttribute()',\n  // MDC automatically applies the primary theme color to the button, but we want to support\n  // an unthemed version. If color is undefined, apply a CSS class that makes it easy to\n  // select and style this \"theme\".\n  '[class.mat-unthemed]': '!color',\n  // Add a class that applies to all buttons. This makes it easier to target if somebody\n  // wants to target all Material buttons.\n  '[class.mat-mdc-button-base]': 'true',\n  '[class]': 'color ? \"mat-\" + color : \"\"',\n};\n\n/**\n * Anchor button base.\n */\n@Directive()\nexport class MatAnchorBase extends MatButtonBase implements OnInit, OnDestroy {\n  @Input({\n    transform: (value: unknown) => {\n      return value == null ? undefined : numberAttribute(value);\n    },\n  })\n  tabIndex: number;\n\n  constructor(elementRef: ElementRef, platform: Platform, ngZone: NgZone, animationMode?: string) {\n    super(elementRef, platform, ngZone, animationMode);\n  }\n\n  ngOnInit(): void {\n    this._ngZone.runOutsideAngular(() => {\n      this._elementRef.nativeElement.addEventListener('click', this._haltDisabledEvents);\n    });\n  }\n\n  override ngOnDestroy(): void {\n    super.ngOnDestroy();\n    this._elementRef.nativeElement.removeEventListener('click', this._haltDisabledEvents);\n  }\n\n  _haltDisabledEvents = (event: Event): void => {\n    // A disabled button shouldn't apply any actions\n    if (this.disabled) {\n      event.preventDefault();\n      event.stopImmediatePropagation();\n    }\n  };\n\n  protected override _getAriaDisabled() {\n    return this.ariaDisabled == null ? this.disabled : this.ariaDisabled;\n  }\n}\n"]}