DevExtreme v23.1 is now available.

Explore our newest features/capabilities and share your thoughts with us.

Your search did not match any results.



The DevExtreme Toast components can stack multiple notifications. Use the notify(message, stack) or notify(options, stack) method to display stacked messages.

These methods use a stack object that has the following structure: {position, direction}.

Specify Position

You can set the position field to a string (select 'predefined' in the radio group) or an object (select 'coordinates' in the radio group). Note that if you use coordinates for the position field, you need to specify one vertical and one horizontal coordinate only. For example, if you specify 'top', the demo disables the 'bottom' field, and vice versa.

Specify Direction

The direction field specifies two options: which way the notification stack grows and whether new notifications appear at the end or at the beginning of the line. For this reason, the field's pull-down menu choices show pairs of values such as 'up-push' and 'up-stack'.

  • 'up-push'
    New toasts push the previous toasts upwards.

  • 'up-stack'
    Toasts stack on top of each other.

Hide Toasts

To hide all toast messages, use the hideToasts method.

Backend API
Copy to CodeSandBox
<template> <div> <div class="options"> <div>Position</div> <DxRadioGroup layout="horizontal" :items="['predefined', 'coordinates']" value="predefined" @value-changed="radioGroupValueChanged($event)" /> <DxSelectBox :items="positions" :input-attr="{ 'aria-label': 'Position' }" v-model:value="predefinedPosition" :visible="isPredefined" /> <div class="section"> <DxNumberBox :visible="!isPredefined" placeholder="top" v-model:value="" width="48%" value-change-event="keyup" :disabled="!!coordinatePosition.bottom" :input-attr="{ 'aria-label': 'Position Top' }" /> <DxNumberBox :visible="!isPredefined" placeholder="bottom" v-model:value="coordinatePosition.bottom" width="48%" value-change-event="keyup" :disabled="!!" :input-attr="{ 'aria-label': 'Position Bottom' }" /> </div> <div class="section"> <DxNumberBox :visible="!isPredefined" placeholder="left" v-model:value="coordinatePosition.left" width="48%" value-change-event="keyup" :disabled="!!coordinatePosition.right" :input-attr="{ 'aria-label': 'Position Left' }" /> <DxNumberBox :visible="!isPredefined" placeholder="right" v-model:value="coordinatePosition.right" width="48%" value-change-event="keyup" :disabled="!!coordinatePosition.left" :input-attr="{ 'aria-label': 'Position Right' }" /> </div> <div>Direction</div> <DxSelectBox :items="directions" :input-attr="{ 'aria-label': 'Direction' }" v-model:value="direction" /> <div class="section"> <DxButton text="Show" width="48%" @click="show()" /> <DxButton text="Hide all" width="48%" @click="hideAll()" /> </div> </div> </div> </template> <script> import DxButton from 'devextreme-vue/button'; import DxRadioGroup from 'devextreme-vue/radio-group'; import DxSelectBox from 'devextreme-vue/select-box'; import DxNumberBox from 'devextreme-vue/number-box'; import notify from 'devextreme/ui/notify'; import hideToasts from 'devextreme/ui/toast/hide_toasts'; export default { components: { DxButton, DxRadioGroup, DxSelectBox, DxNumberBox, }, data() { return { types: ['error', 'info', 'success', 'warning'], positions: [ 'top left', 'top center', 'top right', 'bottom left', 'bottom center', 'bottom right', 'left center', 'center', 'right center', ], directions: [ 'down-push', 'down-stack', 'up-push', 'up-stack', 'left-push', 'left-stack', 'right-push', 'right-stack', ], id: 1, isPredefined: true, predefinedPosition: 'bottom center', coordinatePosition: { top: undefined, bottom: undefined, left: undefined, right: undefined, }, direction: 'up-push', }; }, methods: { show() { const position = this.isPredefined ? this.predefinedPosition : this.coordinatePosition; const direction = this.direction; notify({ message: `Toast ${}`, height: 45, width: 150, minWidth: 150, type: this.types[Math.floor(Math.random() * 4)], displayTime: 3500, animation: { show: { type: 'fade', duration: 400, from: 0, to: 1, }, hide: { type: 'fade', duration: 40, to: 0 }, }, }, { position, direction }); += 1; }, hideAll() { hideToasts(); }, radioGroupValueChanged({ value }) { this.isPredefined = value === 'predefined'; }, }, }; </script> <style> .options { padding: 20px; background-color: rgba(191, 191, 191, 0.15); position: absolute; right: 0; top: 0; width: 260px; display: flex; flex-direction: column; gap: 5px; } .section { width: 100%; display: flex; justify-content: space-between; } </style>
import { createApp } from 'vue'; import App from './App.vue'; createApp(App).mount('#app');
<!DOCTYPE html> <html> <head> <title>DevExtreme Demo</title> <meta http-equiv="X-UA-Compatible" content="IE=edge" /> <meta http-equiv="Content-Type" content="text/html; charset=utf-8" /> <meta name="viewport" content="width=device-width, initial-scale=1.0, maximum-scale=1.0" /> <link rel="stylesheet" type="text/css" href="" /> <link rel="stylesheet" type="text/css" href="styles.css" /> <script src=""></script> <script src=""></script> <script type="text/javascript" src="config.js"></script> <script type="text/javascript"> System.import("./index.js"); </script> </head> <body class="dx-viewport"> <div class="demo-container"> <div id="app"> </div> </div> </body> </html>
window.config = { transpiler: 'plugin-babel', meta: { '*.vue': { loader: 'vue-loader', }, '*.ts': { loader: 'demo-ts-loader', }, '*.svg': { loader: 'svg-loader', }, 'devextreme/localization.js': { 'esModule': true, }, }, paths: { 'root:': '../../../../../', 'npm:': '', }, map: { 'vue': 'npm:vue@3.3.4/dist/vue.esm-browser.js', 'vue-loader': 'npm:dx-systemjs-vue-browser@1.1.1/index.js', 'demo-ts-loader': 'root:utils/demo-ts-loader.js', 'svg-loader': 'root:utils/svg-loader.js', 'mitt': 'npm:mitt/dist/mitt.umd.js', 'rrule': 'npm:rrule@2.6.4/dist/es5/rrule.js', 'luxon': 'npm:luxon@1.28.1/build/global/luxon.min.js', 'es6-object-assign': 'npm:es6-object-assign@1.1.0', 'devextreme': 'npm:devextreme@23.1.5/cjs', 'devextreme-vue': 'npm:devextreme-vue@23.1.5', 'jszip': 'npm:jszip@3.7.1/dist/jszip.min.js', 'devextreme-quill': 'npm:devextreme-quill@1.6.2/dist/dx-quill.min.js', 'devexpress-diagram': 'npm:devexpress-diagram@2.2.1/dist/dx-diagram.js', 'devexpress-gantt': 'npm:devexpress-gantt@4.1.48/dist/dx-gantt.js', '@devextreme/runtime': 'npm:@devextreme/runtime@3.0.11', 'inferno': 'npm:inferno@7.4.11/dist/inferno.min.js', 'inferno-compat': 'npm:inferno-compat/dist/inferno-compat.min.js', 'inferno-create-element': 'npm:inferno-create-element@7.4.11/dist/inferno-create-element.min.js', 'inferno-dom': 'npm:inferno-dom/dist/inferno-dom.min.js', 'inferno-hydrate': 'npm:inferno-hydrate@7.4.11/dist/inferno-hydrate.min.js', 'inferno-clone-vnode': 'npm:inferno-clone-vnode/dist/inferno-clone-vnode.min.js', 'inferno-create-class': 'npm:inferno-create-class/dist/inferno-create-class.min.js', 'inferno-extras': 'npm:inferno-extras/dist/inferno-extras.min.js', 'plugin-babel': 'npm:systemjs-plugin-babel@0.0.25/plugin-babel.js', 'systemjs-babel-build': 'npm:systemjs-plugin-babel@0.0.25/systemjs-babel-browser.js', // Prettier 'prettier/standalone': 'npm:prettier@2.8.4/standalone.js', 'prettier/parser-html': 'npm:prettier@2.8.4/parser-html.js', }, packages: { 'devextreme-vue': { main: 'index.js', }, 'devextreme': { defaultExtension: 'js', }, 'devextreme/events/utils': { main: 'index', }, 'devextreme/events': { main: 'index', }, 'es6-object-assign': { main: './index.js', defaultExtension: 'js', }, }, packageConfigPaths: [ 'npm:@devextreme/*/package.json', 'npm:@devextreme/runtime@3.0.11/inferno/package.json', ], babelOptions: { sourceMaps: false, stage0: true, }, }; System.config(window.config);