@blueprintjs/core
Version:
Core styles & components
137 lines (135 loc) • 20.7 kB
JavaScript
/*
* Copyright 2016 Palantir Technologies, Inc. All rights reserved.
* Licensed under the BSD-3 License as modified (the “License”); you may obtain a copy
* of the license at https://github.com/palantir/blueprint/blob/master/LICENSE
* and https://github.com/palantir/blueprint/blob/master/PATENTS
*/
;
var __extends = (this && this.__extends) || function (d, b) {
for (var p in b) if (b.hasOwnProperty(p)) d[p] = b[p];
function __() { this.constructor = d; }
d.prototype = b === null ? Object.create(b) : (__.prototype = b.prototype, new __());
};
var __assign = (this && this.__assign) || Object.assign || function(t) {
for (var s, i = 1, n = arguments.length; i < n; i++) {
s = arguments[i];
for (var p in s) if (Object.prototype.hasOwnProperty.call(s, p))
t[p] = s[p];
}
return t;
};
var __decorate = (this && this.__decorate) || function (decorators, target, key, desc) {
var c = arguments.length, r = c < 3 ? target : desc === null ? desc = Object.getOwnPropertyDescriptor(target, key) : desc, d;
if (typeof Reflect === "object" && typeof Reflect.decorate === "function") r = Reflect.decorate(decorators, target, key, desc);
else for (var i = decorators.length - 1; i >= 0; i--) if (d = decorators[i]) r = (c < 3 ? d(r) : c > 3 ? d(target, key, r) : d(target, key)) || r;
return c > 3 && r && Object.defineProperty(target, key, r), r;
};
var classNames = require("classnames");
var PureRender = require("pure-render-decorator");
var React = require("react");
var ReactDOM = require("react-dom");
var abstractComponent_1 = require("../../common/abstractComponent");
var Classes = require("../../common/classes");
var keys_1 = require("../../common/keys");
var position_1 = require("../../common/position");
var utils_1 = require("../../common/utils");
var overlay_1 = require("../overlay/overlay");
var toast_1 = require("./toast");
var Toaster = (function (_super) {
__extends(Toaster, _super);
function Toaster() {
var _this = this;
_super.apply(this, arguments);
this.state = {
toasts: [],
};
// auto-incrementing identifier for un-keyed toasts
this.toastId = 0;
this.getDismissHandler = function (toast) { return function (timeoutExpired) {
_this.dismiss(toast.key, timeoutExpired);
}; };
this.handleClose = function (e) {
// NOTE that `e` isn't always a KeyboardEvent but that's the only type we care about
if (e.which === keys_1.ESCAPE) {
_this.clear();
}
};
}
/**
* Create a new `Toaster` instance that can be shared around your application.
* The `Toaster` will be rendered into a new element appended to the given container.
*/
Toaster.create = function (props, container) {
if (container === void 0) { container = document.body; }
var containerElement = document.createElement("div");
container.appendChild(containerElement);
return ReactDOM.render(React.createElement(Toaster, __assign({}, props, {inline: true})), containerElement);
};
Toaster.prototype.show = function (props) {
var options = props;
options.key = "toast-" + this.toastId++;
this.setState(function (prevState) { return ({
toasts: [options].concat(prevState.toasts),
}); });
return options.key;
};
Toaster.prototype.update = function (key, props) {
var options = props;
options.key = key;
this.setState(function (prevState) { return ({
toasts: prevState.toasts.map(function (t) { return t.key === key ? options : t; }),
}); });
};
Toaster.prototype.dismiss = function (key, timeoutExpired) {
if (timeoutExpired === void 0) { timeoutExpired = false; }
this.setState(function (_a) {
var toasts = _a.toasts;
return ({
toasts: toasts.filter(function (t) {
var matchesKey = t.key === key;
if (matchesKey) {
utils_1.safeInvoke(t.onDismiss, timeoutExpired);
}
return !matchesKey;
}),
});
});
};
Toaster.prototype.clear = function () {
this.state.toasts.map(function (t) { return utils_1.safeInvoke(t.onDismiss, false); });
this.setState({ toasts: [] });
};
Toaster.prototype.getToasts = function () {
return this.state.toasts;
};
Toaster.prototype.render = function () {
// $pt-transition-duration * 3 + $pt-transition-duration / 2
var classes = classNames(Classes.TOAST_CONTAINER, this.getPositionClasses(), this.props.className);
return (React.createElement(overlay_1.Overlay, {autoFocus: this.props.autoFocus, canEscapeKeyClose: this.props.canEscapeKeyClear, canOutsideClickClose: false, className: classes, enforceFocus: false, hasBackdrop: false, inline: this.props.inline, isOpen: this.state.toasts.length > 0, onClose: this.handleClose, transitionDuration: 350, transitionName: "pt-toast"}, this.state.toasts.map(this.renderToast, this)));
};
Toaster.prototype.validateProps = function (props) {
if (props.position === position_1.Position.LEFT || props.position === position_1.Position.RIGHT) {
throw new Error("Toaster does not support LEFT or RIGHT positions.");
}
};
Toaster.prototype.renderToast = function (toast) {
return React.createElement(toast_1.Toast, __assign({}, toast, {onDismiss: this.getDismissHandler(toast)}));
};
Toaster.prototype.getPositionClasses = function () {
var positions = position_1.Position[this.props.position].split("_");
// NOTE that there is no -center class because that's the default style
return positions.map(function (p) { return (Classes.TOAST_CONTAINER + "-" + p.toLowerCase()); });
};
Toaster.defaultProps = {
autoFocus: false,
canEscapeKeyClear: true,
inline: false,
position: position_1.Position.TOP,
};
Toaster = __decorate([
PureRender
], Toaster);
return Toaster;
}(abstractComponent_1.AbstractComponent));
exports.Toaster = Toaster;
//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../src/components/toast/toaster.tsx"],"names":[],"mappings":"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","file":"components/toast/toaster.js","sourcesContent":["/*\n * Copyright 2016 Palantir Technologies, Inc. All rights reserved.\n * Licensed under the BSD-3 License as modified (the “License”); you may obtain a copy\n * of the license at https://github.com/palantir/blueprint/blob/master/LICENSE\n * and https://github.com/palantir/blueprint/blob/master/PATENTS\n */\n\nimport * as classNames from \"classnames\";\nimport * as PureRender from \"pure-render-decorator\";\nimport * as React from \"react\";\nimport * as ReactDOM from \"react-dom\";\n\nimport { AbstractComponent } from \"../../common/abstractComponent\";\nimport * as Classes from \"../../common/classes\";\nimport { ESCAPE } from \"../../common/keys\";\nimport { Position } from \"../../common/position\";\nimport { IProps } from \"../../common/props\";\nimport { safeInvoke } from \"../../common/utils\";\nimport { Overlay } from \"../overlay/overlay\";\nimport { IToastProps, Toast } from \"./toast\";\n\nexport type IToastOptions = IToastProps & {key?: string};\n\nexport interface IToaster {\n    /** Show a new toast to the user. Returns the unique key of the new toast. */\n    show(props: IToastProps): string;\n\n    /**\n     * Updates the toast with the given key to use the new props.\n     * Updating a key that does not exist is effectively a no-op.\n     */\n    update(key: string, props: IToastProps): void;\n\n    /** Dismiss the given toast instantly. */\n    dismiss(key: string): void;\n\n    /** Dismiss all toasts instantly. */\n    clear(): void;\n\n    /** Returns the props for all current toasts. */\n    getToasts(): IToastOptions[];\n}\n\nexport interface IToasterProps extends IProps {\n    /**\n     * Whether a toast should acquire application focus when it first opens.\n     * This is disabled by default so that toasts do not interrupt the user's flow.\n     * Note that `enforceFocus` is always disabled for `Toaster`s.\n     * @default false\n     */\n    autoFocus?: boolean;\n\n    /**\n     * Whether pressing the `esc` key should clear all active toasts.\n     * @default true\n     */\n    canEscapeKeyClear?: boolean;\n\n    /**\n     * Whether the toaster should be rendered inline or into a new element on `document.body`.\n     * If `true`, then positioning will be relative to the parent element.\n     * @default false\n     */\n    inline?: boolean;\n\n    /**\n     * Position of `Toaster` within its container. Note that `LEFT` and `RIGHT` are disallowed\n     * because Toaster only supports the top and bottom edges.\n     * @default Position.TOP\n     */\n    position?: Position;\n}\n\nexport interface IToasterState {\n    toasts: IToastOptions[];\n}\n\n@PureRender\nexport class Toaster extends AbstractComponent<IToasterProps, IToasterState> implements IToaster {\n    public static defaultProps: IToasterProps = {\n        autoFocus: false,\n        canEscapeKeyClear: true,\n        inline: false,\n        position: Position.TOP,\n    };\n\n    /**\n     * Create a new `Toaster` instance that can be shared around your application.\n     * The `Toaster` will be rendered into a new element appended to the given container.\n     */\n    public static create(props?: IToasterProps, container = document.body): IToaster {\n        const containerElement = document.createElement(\"div\");\n        container.appendChild(containerElement);\n        return ReactDOM.render(<Toaster {...props} inline /> , containerElement) as Toaster;\n    }\n\n    public state = {\n        toasts: [] as IToastOptions[],\n    };\n\n    // auto-incrementing identifier for un-keyed toasts\n    private toastId = 0;\n\n    public show(props: IToastProps) {\n        const options = props as IToastOptions;\n        options.key = `toast-${this.toastId++}`;\n        this.setState((prevState) => ({\n            toasts: [options, ...prevState.toasts],\n        }));\n        return options.key;\n    }\n\n    public update(key: string, props: IToastProps) {\n        const options = props as IToastOptions;\n        options.key = key;\n        this.setState((prevState) => ({\n            toasts: prevState.toasts.map((t) => t.key === key ? options : t),\n        }));\n    }\n\n    public dismiss(key: string, timeoutExpired = false) {\n        this.setState(({ toasts }) => ({\n            toasts: toasts.filter((t) => {\n                const matchesKey = t.key === key;\n                if (matchesKey) {\n                    safeInvoke(t.onDismiss, timeoutExpired);\n                }\n                return !matchesKey;\n            }),\n        }));\n    }\n\n    public clear() {\n        this.state.toasts.map((t) => safeInvoke(t.onDismiss, false));\n        this.setState({ toasts: [] });\n    }\n\n    public getToasts() {\n        return this.state.toasts;\n    }\n\n    public render() {\n        // $pt-transition-duration * 3 + $pt-transition-duration / 2\n        const classes = classNames(Classes.TOAST_CONTAINER, this.getPositionClasses(), this.props.className);\n        return (\n            <Overlay\n                autoFocus={this.props.autoFocus}\n                canEscapeKeyClose={this.props.canEscapeKeyClear}\n                canOutsideClickClose={false}\n                className={classes}\n                enforceFocus={false}\n                hasBackdrop={false}\n                inline={this.props.inline}\n                isOpen={this.state.toasts.length > 0}\n                onClose={this.handleClose}\n                transitionDuration={350}\n                transitionName=\"pt-toast\"\n            >\n                {this.state.toasts.map(this.renderToast, this)}\n            </Overlay>\n        );\n    }\n\n    protected validateProps(props: IToasterProps) {\n        if (props.position === Position.LEFT || props.position === Position.RIGHT) {\n            throw new Error(\"Toaster does not support LEFT or RIGHT positions.\");\n        }\n    }\n\n    private renderToast(toast: IToastOptions) {\n        return <Toast {...toast} onDismiss={this.getDismissHandler(toast)} />;\n    }\n\n    private getPositionClasses() {\n        const positions = Position[this.props.position].split(\"_\");\n        // NOTE that there is no -center class because that's the default style\n        return positions.map((p) => `${Classes.TOAST_CONTAINER}-${p.toLowerCase()}`);\n    }\n\n    private getDismissHandler = (toast: IToastOptions) => (timeoutExpired: boolean) => {\n        this.dismiss(toast.key, timeoutExpired);\n    }\n\n    private handleClose = (e: React.KeyboardEvent<HTMLElement>) => {\n        // NOTE that `e` isn't always a KeyboardEvent but that's the only type we care about\n        if (e.which === ESCAPE) {\n            this.clear();\n        }\n    }\n}\n"],"sourceRoot":"/source/"}