Skip to content

Commit 4f11851

Browse files
cdonovanabhiomkar
authored andcommitted
feat(snackbar): Add option for indefinite timeout (material-components#4998)
1 parent 46ce31e commit 4f11851

File tree

4 files changed

+33
-7
lines changed

4 files changed

+33
-7
lines changed

packages/mdc-snackbar/README.md

+1-1
Original file line numberDiff line numberDiff line change
@@ -166,7 +166,7 @@ Mixin | Description
166166
Property | Value Type | Description
167167
--- | --- | ---
168168
`isOpen` | `boolean` (read-only) | Gets whether the snackbar is currently open.
169-
`timeoutMs` | `number` | Gets/sets the automatic dismiss timeout in milliseconds. Value must be between `4000` and `10000` or an error will be thrown. Defaults to `5000` (5 seconds).
169+
`timeoutMs` | `number` | Gets/sets the automatic dismiss timeout in milliseconds. Value must be between `4000` and `10000` (or `-1` to disable the timeout completely) or an error will be thrown. Defaults to `5000` (5 seconds).
170170
`closeOnEscape` | `boolean` | Gets/sets whether the snackbar closes when it is focused and the user presses the <kbd>ESC</kbd> key. Defaults to `true`.
171171
`labelText` | `string` | Gets/sets the `textContent` of the label element.
172172
`actionButtonText` | `string` | Gets/sets the `textContent` of the action button element.

packages/mdc-snackbar/constants.ts

+1
Original file line numberDiff line numberDiff line change
@@ -43,6 +43,7 @@ const strings = {
4343

4444
const numbers = {
4545
DEFAULT_AUTO_DISMISS_TIMEOUT_MS: 5000,
46+
INDETERMINATE: -1,
4647
MAX_AUTO_DISMISS_TIMEOUT_MS: 10000,
4748
MIN_AUTO_DISMISS_TIMEOUT_MS: 4000,
4849

packages/mdc-snackbar/foundation.ts

+12-6
Original file line numberDiff line numberDiff line change
@@ -88,12 +88,15 @@ export class MDCSnackbarFoundation extends MDCFoundation<MDCSnackbarAdapter> {
8888
this.adapter_.addClass(OPEN);
8989

9090
this.animationTimer_ = setTimeout(() => {
91+
const timeoutMs = this.getTimeoutMs();
9192
this.handleAnimationTimerEnd_();
9293
this.adapter_.notifyOpened();
93-
this.autoDismissTimer_ = setTimeout(() => {
94-
this.close(REASON_DISMISS);
95-
}, this.getTimeoutMs());
96-
}, numbers.SNACKBAR_ANIMATION_OPEN_TIME_MS);
94+
if (timeoutMs !== numbers.INDETERMINATE) {
95+
this.autoDismissTimer_ = setTimeout(() => {
96+
this.close(REASON_DISMISS);
97+
}, timeoutMs);
98+
}
99+
}, numbers.SNACKBAR_ANIMATION_OPEN_TIME_MS);
97100
});
98101
}
99102

@@ -137,11 +140,14 @@ export class MDCSnackbarFoundation extends MDCFoundation<MDCSnackbarAdapter> {
137140
// Use shorter variable names to make the code more readable
138141
const minValue = numbers.MIN_AUTO_DISMISS_TIMEOUT_MS;
139142
const maxValue = numbers.MAX_AUTO_DISMISS_TIMEOUT_MS;
143+
const indeterminateValue = numbers.INDETERMINATE;
140144

141-
if (timeoutMs <= maxValue && timeoutMs >= minValue) {
145+
if (timeoutMs === numbers.INDETERMINATE || (timeoutMs <= maxValue && timeoutMs >= minValue)) {
142146
this.autoDismissTimeoutMs_ = timeoutMs;
143147
} else {
144-
throw new Error(`timeoutMs must be an integer in the range ${minValue}${maxValue}, but got '${timeoutMs}'`);
148+
throw new Error(`
149+
timeoutMs must be an integer in the range ${minValue}${maxValue}
150+
(or ${indeterminateValue} to disable), but got '${timeoutMs}'`);
145151
}
146152
}
147153

test/unit/mdc-snackbar/foundation.test.js

+19
Original file line numberDiff line numberDiff line change
@@ -234,6 +234,25 @@ test('#open automatically dismisses snackbar after timeout', () => {
234234
td.verify(foundation.close(strings.REASON_DISMISS), {times: 1});
235235
});
236236

237+
test('#snackbar remains open for indefinite timeout', () => {
238+
const {foundation} = setupTest();
239+
const clock = installClock();
240+
foundation.close = td.func('close');
241+
foundation.setTimeoutMs(-1);
242+
243+
foundation.open();
244+
245+
// Note: #open uses a combination of rAF and setTimeout due to Firefox behavior, so we need to wait 2 ticks
246+
clock.runToFrame();
247+
clock.runToFrame();
248+
249+
// Wait for max timeout and ensure that close has not been called
250+
clock.tick(numbers.SNACKBAR_ANIMATION_OPEN_TIME_MS);
251+
clock.tick(numbers.MAX_AUTO_DISMISS_TIMEOUT_MS);
252+
253+
td.verify(foundation.close(strings.REASON_DISMISS), {times: 0});
254+
});
255+
237256
test('#isOpen returns false when the snackbar has never been opened', () => {
238257
const {foundation} = setupTest();
239258
assert.isFalse(foundation.isOpen());

0 commit comments

Comments
 (0)