DevExtreme v24.1 is now available.

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

Your search did not match any results.

Vue Charts - Bi-Directional Bar Chart

To create a bi-directional bar chart, follow the steps below.

Backend API
<template> <DxChart id="chart" :data-source="dataSource" :rotated="true" :bar-group-width="18" title="Population Pyramid For Norway 2016" > <DxCommonSeriesSettings argument-field="age" type="stackedbar" /> <DxSeries value-field="male" name="Male" color="#3F7FBF" /> <DxSeries value-field="female" name="Female" color="#F87CCC" /> <DxTooltip :enabled="true" :customize-tooltip="customizeTooltip" /> <DxValueAxis> <DxLabel :customize-text="customizeLabel"/> </DxValueAxis> <DxLegend :margin="{left: 50}" vertical-alignment="bottom" horizontal-alignment="center" /> </DxChart> </template> <script setup lang="ts"> import DxChart, { DxCommonSeriesSettings, DxValueAxis, DxLabel, DxLegend, DxSeries, DxTooltip, } from 'devextreme-vue/chart'; import { dataSource } from './data.ts'; const customizeTooltip = ({ valueText }) => ({ text: Math.abs(valueText) }); const customizeLabel = ({ value }) => `${Math.abs(value)}%`; </script> <style> #chart { height: 549px; width: 820px; margin: 0 auto; } </style>
window.exports = window.exports || {}; window.config = { transpiler: 'plugin-babel', meta: { '*.vue': { loader: 'vue-loader', }, '*.ts': { loader: 'demo-ts-loader', }, '*.svg': { loader: 'svg-loader', }, 'devextreme/time_zone_utils.js': { 'esModule': true, }, 'devextreme/localization.js': { 'esModule': true, }, 'devextreme/viz/palette.js': { 'esModule': true, }, }, paths: { 'root:': '../../../../', 'npm:': 'https://unpkg.com/', }, map: { 'vue': 'npm:vue@3.2.47/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@24.1.5/cjs', 'devextreme-vue': 'npm:devextreme-vue@24.1.5/cjs', 'jszip': 'npm:jszip@3.10.1/dist/jszip.min.js', 'devextreme-quill': 'npm:devextreme-quill@1.7.1/dist/dx-quill.min.js', 'devexpress-diagram': 'npm:devexpress-diagram@2.2.10/dist/dx-diagram.js', 'devexpress-gantt': 'npm:devexpress-gantt@4.1.56/dist/dx-gantt.js', '@devextreme/runtime': 'npm:@devextreme/runtime@3.0.13', '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.8/standalone.js', 'prettier/parser-html': 'npm:prettier@2.8.8/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.13/inferno/package.json', ], babelOptions: { sourceMaps: false, stage0: true, }, }; System.config(window.config);
export const dataSource = [ { age: '0-4', male: -3.1, female: 2.9, }, { age: '5-9', male: -3.1, female: 3.0, }, { age: '10-14', male: -3.0, female: 2.9, }, { age: '15-19', male: -3.2, female: 3.0, }, { age: '20-24', male: -3.5, female: 3.3, }, { age: '25-29', male: -3.5, female: 3.4, }, { age: '30-34', male: -3.5, female: 3.3, }, { age: '35-39', male: -3.3, female: 3.1, }, { age: '40-44', male: -3.7, female: 3.4, }, { age: '45-49', male: -3.8, female: 3.5, }, { age: '50-54', male: -3.4, female: 3.2, }, { age: '55-59', male: -3.1, female: 3.0, }, { age: '60-64', male: -2.7, female: 2.7, }, { age: '65-69', male: -2.9, female: 2.9, }, { age: '70-74', male: -2, female: 2.1, }, { age: '75-79', male: -1.2, female: 1.4, }, { age: '80-84', male: -0.8, female: 1.2, }, { age: '85-89', male: -0.5, female: 0.8, }, { age: '90-94', male: -0.2, female: 0.5, }, { age: '95+', male: 0, female: 0.1, }, ];
import { createApp } from 'vue'; import App from './App.vue'; createApp(App).mount('#app');
<!DOCTYPE html> <html lang="en"> <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=5.0" /> <link rel="stylesheet" type="text/css" href="https://cdn3.devexpress.com/jslib/24.1.5/css/dx.light.css" /> <script src="https://unpkg.com/typescript@4.9.5/lib/typescript.js"></script> <script type="module"> import * as vueCompilerSFC from "https://unpkg.com/@vue/compiler-sfc@3.4.16/dist/compiler-sfc.esm-browser.js"; window.vueCompilerSFC = vueCompilerSFC; </script> <script src="https://unpkg.com/core-js@2.6.12/client/shim.min.js"></script> <script src="https://unpkg.com/systemjs@0.21.3/dist/system.js"></script> <script type="text/javascript" src="config.js"></script> <script type="text/javascript"> System.import("./index.ts"); </script> </head> <body class="dx-viewport"> <div class="demo-container"> <div id="app"> </div> </div> </body> </html>
  1. Convert half of the data source values from positive to negative. In this demo, the male percentages are negative, while the female are positive.

  2. Implement a rotated stacked bar chart. Use the commonSeriesSettings object to specify the type and the argumentField property (age in this case). Set the rotated property to true.

  3. Declare two objects in the series array: one for negative values, the other for positive. Specify the color, name, and valueField for each series.

  4. Change the label text of the valueAxis so that the negative values appear as positive. To implement this technique, use the customizeText function.

To further customize your bi-directional bar chart, implement tooltips and change the legend's alignment.