Files
blockly/core/toast.ts
Aaron Dodson c6e58c4f92 feat: Add support for displaying toast-style notifications. (#8896)
* feat: Allow resetting alert/prompt/confirm to defaults.

* chore: Add unit tests for Blockly.dialog.

* fix: Removed TEST_ONLY hack from Blockly.dialog.

* feat: Add a default toast notification implementation.

* feat: Add support for toasts to Blockly.dialog.

* chore: Add tests for default toast implementation.

* chore: Fix docstring.

* refactor: Use default arguments for dialog functions.

* refactor: Add 'close' to the list of messages.

* chore: Add new message in several other places.

* chore: clarify docstrings.

* feat: Make toast assertiveness configurable.
2025-04-21 15:32:45 -07:00

220 lines
5.7 KiB
TypeScript

/**
* @license
* Copyright 2025 Google LLC
* SPDX-License-Identifier: Apache-2.0
*/
import * as Css from './css.js';
import {Msg} from './msg.js';
import * as aria from './utils/aria.js';
import * as dom from './utils/dom.js';
import {Svg} from './utils/svg.js';
import type {WorkspaceSvg} from './workspace_svg.js';
const CLASS_NAME = 'blocklyToast';
const MESSAGE_CLASS_NAME = 'blocklyToastMessage';
const CLOSE_BUTTON_CLASS_NAME = 'blocklyToastCloseButton';
/**
* Display/configuration options for a toast notification.
*/
export interface ToastOptions {
/**
* Toast ID. If set along with `oncePerSession`, will cause subsequent toasts
* with this ID to not be shown.
*/
id?: string;
/**
* Flag to show the toast once per session only.
* Subsequent calls are ignored.
*/
oncePerSession?: boolean;
/**
* Text of the message to display on the toast.
*/
message: string;
/**
* Duration in seconds before the toast is removed. Defaults to 5.
*/
duration?: number;
/**
* How prominently/interrupting the readout of the toast should be for
* screenreaders. Corresponds to aria-live and defaults to polite.
*/
assertiveness?: Toast.Assertiveness;
}
/**
* Class that allows for showing and dismissing temporary notifications.
*/
export class Toast {
/** IDs of toasts that have previously been shown. */
private static shownIds = new Set<string>();
/**
* Shows a toast notification.
*
* @param workspace The workspace to show the toast on.
* @param options Configuration options for the toast message, duration, etc.
*/
static show(workspace: WorkspaceSvg, options: ToastOptions) {
if (options.oncePerSession && options.id) {
if (this.shownIds.has(options.id)) return;
this.shownIds.add(options.id);
}
// Clear any existing toasts.
this.hide(workspace);
const toast = this.createDom(workspace, options);
// Animate the toast into view.
requestAnimationFrame(() => {
toast.style.bottom = '2rem';
});
}
/**
* Creates the DOM representation of a toast.
*
* @param workspace The workspace to inject the toast notification onto.
* @param options Configuration options for the toast.
* @returns The root DOM element of the toast.
*/
protected static createDom(workspace: WorkspaceSvg, options: ToastOptions) {
const {
message,
duration = 5,
assertiveness = Toast.Assertiveness.POLITE,
} = options;
const toast = document.createElement('div');
workspace.getInjectionDiv().appendChild(toast);
toast.dataset.toastId = options.id;
toast.className = CLASS_NAME;
aria.setRole(toast, aria.Role.STATUS);
aria.setState(toast, aria.State.LIVE, assertiveness);
const messageElement = toast.appendChild(document.createElement('div'));
messageElement.className = MESSAGE_CLASS_NAME;
messageElement.innerText = message;
const closeButton = toast.appendChild(document.createElement('button'));
closeButton.className = CLOSE_BUTTON_CLASS_NAME;
aria.setState(closeButton, aria.State.LABEL, Msg['CLOSE']);
const closeIcon = dom.createSvgElement(
Svg.SVG,
{
width: 24,
height: 24,
viewBox: '0 0 24 24',
fill: 'none',
},
closeButton,
);
aria.setState(closeIcon, aria.State.HIDDEN, true);
dom.createSvgElement(
Svg.RECT,
{
x: 19.7782,
y: 2.80762,
width: 2,
height: 24,
transform: 'rotate(45, 19.7782, 2.80762)',
fill: 'black',
},
closeIcon,
);
dom.createSvgElement(
Svg.RECT,
{
x: 2.80762,
y: 4.22183,
width: 2,
height: 24,
transform: 'rotate(-45, 2.80762, 4.22183)',
fill: 'black',
},
closeIcon,
);
closeButton.addEventListener('click', () => {
toast.remove();
workspace.markFocused();
});
let timeout: ReturnType<typeof setTimeout>;
const setToastTimeout = () => {
timeout = setTimeout(() => toast.remove(), duration * 1000);
};
const clearToastTimeout = () => clearTimeout(timeout);
toast.addEventListener('focusin', clearToastTimeout);
toast.addEventListener('focusout', setToastTimeout);
toast.addEventListener('mouseenter', clearToastTimeout);
toast.addEventListener('mousemove', clearToastTimeout);
toast.addEventListener('mouseleave', setToastTimeout);
setToastTimeout();
return toast;
}
/**
* Dismiss a toast, e.g. in response to a user action.
*
* @param workspace The workspace to dismiss a toast in.
* @param id The toast ID, or undefined to clear any toast.
*/
static hide(workspace: WorkspaceSvg, id?: string) {
const toast = workspace.getInjectionDiv().querySelector(`.${CLASS_NAME}`);
if (toast instanceof HTMLElement && (!id || id === toast.dataset.toastId)) {
toast.remove();
}
}
}
/**
* Options for how aggressively toasts should be read out by screenreaders.
* Values correspond to those for aria-live.
*/
export namespace Toast {
export enum Assertiveness {
ASSERTIVE = 'assertive',
POLITE = 'polite',
}
}
Css.register(`
.${CLASS_NAME} {
font-size: 1.2rem;
position: absolute;
bottom: -10rem;
right: 2rem;
padding: 1rem;
color: black;
background-color: white;
border: 2px solid black;
border-radius: 0.4rem;
z-index: 999;
display: flex;
align-items: center;
gap: 0.8rem;
line-height: 1.5;
transition: bottom 0.3s ease-out;
}
.${CLASS_NAME} .${MESSAGE_CLASS_NAME} {
maxWidth: 18rem;
}
.${CLASS_NAME} .${CLOSE_BUTTON_CLASS_NAME} {
margin: 0;
padding: 0.2rem;
background-color: transparent;
color: black;
border: none;
cursor: pointer;
}
`);