Skip to content

Commit 855ed49

Browse files
committedAug 25, 2024
fix(material/core): avoid having to manually load ripple styles
Makes it so the ripple loads the necessary styles itself, instead of requiring the user to do it. BREAKING CHANGE: * The ripples styles are now loaded slightly later than before which can change their specificity. You may have to update any ripple style overrides.
1 parent fcb76d3 commit 855ed49

File tree

10 files changed

+51
-8
lines changed

10 files changed

+51
-8
lines changed
 

‎src/material/core/BUILD.bazel

+8
Original file line numberDiff line numberDiff line change
@@ -24,6 +24,7 @@ ng_module(
2424
":option/option.css",
2525
":option/optgroup.css",
2626
":internal-form-field/internal-form-field.css",
27+
":ripple/ripple-structure.css",
2728
] + glob(["**/*.html"]),
2829
deps = [
2930
"//src:dev_mode_types",
@@ -33,6 +34,7 @@ ng_module(
3334
"//src/cdk/coercion",
3435
"//src/cdk/keycodes",
3536
"//src/cdk/platform",
37+
"//src/cdk/private",
3638
"@npm//@angular/animations",
3739
"@npm//@angular/core",
3840
"@npm//@angular/forms",
@@ -94,6 +96,12 @@ sass_binary(
9496
deps = [":core_scss_lib"],
9597
)
9698

99+
sass_binary(
100+
name = "ripple_structure_scss",
101+
src = "ripple/ripple-structure.scss",
102+
deps = [":core_scss_lib"],
103+
)
104+
97105
# M3 themes
98106
sass_binary(
99107
name = "azure_blue_prebuilt",

‎src/material/core/_core.scss

-2
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,6 @@
11
@use '@angular/cdk';
22
@use './tokens/m2/mat/app' as tokens-mat-app;
33
@use './tokens/token-utils';
4-
@use './ripple/ripple';
54
@use './style/elevation';
65
@use './focus-indicators/private';
76

@@ -15,7 +14,6 @@
1514
--mat-app-on-surface: initial;
1615
}
1716

18-
@include ripple.ripple();
1917
@include cdk.a11y-visually-hidden();
2018
@include cdk.overlay();
2119
@include cdk.text-field-autosize();

‎src/material/core/private/ripple-loader.ts

+1
Original file line numberDiff line numberDiff line change
@@ -15,6 +15,7 @@ import {
1515
defaultRippleAnimationConfig,
1616
} from '../ripple';
1717
import {Platform, _getEventTarget} from '@angular/cdk/platform';
18+
import {_CdkPrivateStyleLoader} from '@angular/cdk/private';
1819

1920
/** The options for the MatRippleLoader's event listeners. */
2021
const eventListenerOptions = {capture: true};

‎src/material/core/ripple/index.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -12,7 +12,7 @@ import {MatRipple} from './ripple';
1212

1313
export * from './ripple';
1414
export * from './ripple-ref';
15-
export * from './ripple-renderer';
15+
export {RippleRenderer, RippleTarget, defaultRippleAnimationConfig} from './ripple-renderer';
1616

1717
@NgModule({
1818
imports: [MatCommonModule, MatRipple],

‎src/material/core/ripple/ripple-renderer.ts

+24-1
Original file line numberDiff line numberDiff line change
@@ -5,10 +5,18 @@
55
* Use of this source code is governed by an MIT-style license that can be
66
* found in the LICENSE file at https://angular.io/license
77
*/
8-
import {ElementRef, NgZone} from '@angular/core';
8+
import {
9+
ElementRef,
10+
NgZone,
11+
Component,
12+
ChangeDetectionStrategy,
13+
ViewEncapsulation,
14+
Injector,
15+
} from '@angular/core';
916
import {Platform, normalizePassiveListenerOptions, _getEventTarget} from '@angular/cdk/platform';
1017
import {isFakeMousedownFromScreenReader, isFakeTouchstartFromScreenReader} from '@angular/cdk/a11y';
1118
import {coerceElement} from '@angular/cdk/coercion';
19+
import {_CdkPrivateStyleLoader} from '@angular/cdk/private';
1220
import {RippleRef, RippleState, RippleConfig} from './ripple-ref';
1321
import {RippleEventManager} from './ripple-event-manager';
1422

@@ -58,6 +66,16 @@ const pointerDownEvents = ['mousedown', 'touchstart'];
5866
/** Events that signal that the pointer is up. */
5967
const pointerUpEvents = ['mouseup', 'mouseleave', 'touchend', 'touchcancel'];
6068

69+
@Component({
70+
template: '',
71+
changeDetection: ChangeDetectionStrategy.OnPush,
72+
encapsulation: ViewEncapsulation.None,
73+
standalone: true,
74+
styleUrl: 'ripple-structure.css',
75+
host: {'mat-ripple-style-loader': ''},
76+
})
77+
export class _MatRippleStylesLoader {}
78+
6179
/**
6280
* Helper service that performs DOM manipulations. Not intended to be used outside this module.
6381
* The constructor takes a reference to the ripple directive's host element and a map of DOM
@@ -105,11 +123,16 @@ export class RippleRenderer implements EventListenerObject {
105123
private _ngZone: NgZone,
106124
elementOrElementRef: HTMLElement | ElementRef<HTMLElement>,
107125
private _platform: Platform,
126+
injector?: Injector,
108127
) {
109128
// Only do anything if we're on the browser.
110129
if (_platform.isBrowser) {
111130
this._containerElement = coerceElement(elementOrElementRef);
112131
}
132+
133+
if (injector) {
134+
injector.get(_CdkPrivateStyleLoader).load(_MatRippleStylesLoader);
135+
}
113136
}
114137

115138
/**
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,3 @@
1+
@use './ripple';
2+
3+
@include ripple.ripple;

‎src/material/core/ripple/ripple.ts

+6-1
Original file line numberDiff line numberDiff line change
@@ -18,7 +18,9 @@ import {
1818
OnInit,
1919
Optional,
2020
ANIMATION_MODULE_TYPE,
21+
Injector,
2122
} from '@angular/core';
23+
import {_CdkPrivateStyleLoader} from '@angular/cdk/private';
2224
import {RippleAnimationConfig, RippleConfig, RippleRef} from './ripple-ref';
2325
import {RippleRenderer, RippleTarget} from './ripple-renderer';
2426

@@ -136,9 +138,12 @@ export class MatRipple implements OnInit, OnDestroy, RippleTarget {
136138
platform: Platform,
137139
@Optional() @Inject(MAT_RIPPLE_GLOBAL_OPTIONS) globalOptions?: RippleGlobalOptions,
138140
@Optional() @Inject(ANIMATION_MODULE_TYPE) private _animationMode?: string,
141+
injector?: Injector,
139142
) {
143+
// Note: cannot use `inject()` here, because this class
144+
// gets instantiated manually in the ripple loader.
140145
this._globalOptions = globalOptions || {};
141-
this._rippleRenderer = new RippleRenderer(this, ngZone, _elementRef, platform);
146+
this._rippleRenderer = new RippleRenderer(this, ngZone, _elementRef, platform, injector);
142147
}
143148

144149
ngOnInit() {

‎src/material/list/BUILD.bazel

+1
Original file line numberDiff line numberDiff line change
@@ -27,6 +27,7 @@ ng_module(
2727
"//src/cdk/coercion",
2828
"//src/cdk/collections",
2929
"//src/cdk/observers",
30+
"//src/cdk/private",
3031
"//src/material/core",
3132
"//src/material/divider",
3233
"@npm//@angular/core",

‎src/material/list/list-base.ts

+3
Original file line numberDiff line numberDiff line change
@@ -21,6 +21,7 @@ import {
2121
Optional,
2222
QueryList,
2323
ANIMATION_MODULE_TYPE,
24+
Injector,
2425
} from '@angular/core';
2526
import {
2627
MAT_RIPPLE_GLOBAL_OPTIONS,
@@ -29,6 +30,7 @@ import {
2930
RippleRenderer,
3031
RippleTarget,
3132
} from '@angular/material/core';
33+
import {_CdkPrivateStyleLoader} from '@angular/cdk/private';
3234
import {Subscription, merge} from 'rxjs';
3335
import {
3436
MatListItemLine,
@@ -223,6 +225,7 @@ export abstract class MatListItemBase implements AfterViewInit, OnDestroy, Rippl
223225
this._ngZone,
224226
this._hostElement,
225227
this._platform,
228+
inject(Injector),
226229
);
227230
this._rippleRenderer.setupTriggerEvents(this._hostElement);
228231
}

‎tools/public_api_guard/material/core.md

+4-3
Original file line numberDiff line numberDiff line change
@@ -16,6 +16,7 @@ import { HighContrastModeDetector } from '@angular/cdk/a11y';
1616
import * as i0 from '@angular/core';
1717
import * as i1 from '@angular/cdk/bidi';
1818
import { InjectionToken } from '@angular/core';
19+
import { Injector } from '@angular/core';
1920
import { NgControl } from '@angular/forms';
2021
import { NgForm } from '@angular/forms';
2122
import { NgZone } from '@angular/core';
@@ -372,7 +373,7 @@ export type MatPseudoCheckboxState = 'unchecked' | 'checked' | 'indeterminate';
372373

373374
// @public (undocumented)
374375
export class MatRipple implements OnInit, OnDestroy, RippleTarget {
375-
constructor(_elementRef: ElementRef<HTMLElement>, ngZone: NgZone, platform: Platform, globalOptions?: RippleGlobalOptions, _animationMode?: string | undefined);
376+
constructor(_elementRef: ElementRef<HTMLElement>, ngZone: NgZone, platform: Platform, globalOptions?: RippleGlobalOptions, _animationMode?: string | undefined, injector?: Injector);
376377
animation: RippleAnimationConfig;
377378
centered: boolean;
378379
color: string;
@@ -396,7 +397,7 @@ export class MatRipple implements OnInit, OnDestroy, RippleTarget {
396397
// (undocumented)
397398
static ɵdir: i0.ɵɵDirectiveDeclaration<MatRipple, "[mat-ripple], [matRipple]", ["matRipple"], { "color": { "alias": "matRippleColor"; "required": false; }; "unbounded": { "alias": "matRippleUnbounded"; "required": false; }; "centered": { "alias": "matRippleCentered"; "required": false; }; "radius": { "alias": "matRippleRadius"; "required": false; }; "animation": { "alias": "matRippleAnimation"; "required": false; }; "disabled": { "alias": "matRippleDisabled"; "required": false; }; "trigger": { "alias": "matRippleTrigger"; "required": false; }; }, {}, never, never, true, never>;
398399
// (undocumented)
399-
static ɵfac: i0.ɵɵFactoryDeclaration<MatRipple, [null, null, null, { optional: true; }, { optional: true; }]>;
400+
static ɵfac: i0.ɵɵFactoryDeclaration<MatRipple, [null, null, null, { optional: true; }, { optional: true; }, null]>;
400401
}
401402

402403
// @public
@@ -557,7 +558,7 @@ export class RippleRef {
557558

558559
// @public
559560
export class RippleRenderer implements EventListenerObject {
560-
constructor(_target: RippleTarget, _ngZone: NgZone, elementOrElementRef: HTMLElement | ElementRef<HTMLElement>, _platform: Platform);
561+
constructor(_target: RippleTarget, _ngZone: NgZone, elementOrElementRef: HTMLElement | ElementRef<HTMLElement>, _platform: Platform, injector?: Injector);
561562
fadeInRipple(x: number, y: number, config?: RippleConfig): RippleRef;
562563
fadeOutAll(): void;
563564
fadeOutAllNonPersistent(): void;

0 commit comments

Comments
 (0)
Please sign in to comment.