DevExtreme v23.1 is now available.

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

Your search did not match any results.
Charts

Overview

DevExtreme Vue Charts is a collection of 30+ feature-rich UI components for data visualization. This collection includes bars, areas, lines, bubbles, pie, funnels, sparklines, treemaps, and many other chart types. You can use Vue syntax and techniques to instantiate and configure the charts or handle their events. In addition, the components support prop validation. Find out more about DevExtreme Vue components.

Backend API
Copy to CodeSandBox
Apply
Reset
<template> <DxChart id="chart" :data-source="dataSource" palette="Harmony Light" title="Pizza Shop Complaints" > <DxCommonSeriesSettings argument-field="complaint"/> <DxSeries name="Complaint frequency" value-field="count" axis="frequency" type="bar" color="#fac29a" /> <DxSeries name="Cumulative percentage" value-field="cumulativePercentage" axis="percentage" type="spline" color="#6b71c3" /> <DxArgumentAxis> <DxLabel overlapping-behavior="stagger"/> </DxArgumentAxis> <DxValueAxis :tick-interval="300" name="frequency" position="left" /> <DxValueAxis :tick-interval="20" :show-zero="true" :value-margins-enabled="false" name="percentage" position="right" > <DxLabel :customize-text="customizePercentageText"/> <DxConstantLine :value="80" :width="2" color="#fc3535" dash-style="dash" > <DxLabel :visible="false"/> </DxConstantLine> </DxValueAxis> <DxTooltip :enabled="true" :shared="true" :customize-tooltip="customizeTooltip" /> <DxLegend vertical-alignment="top" horizontal-alignment="center" /> </DxChart> </template> <script> import DxChart, { DxArgumentAxis, DxCommonSeriesSettings, DxLabel, DxLegend, DxSeries, DxTooltip, DxValueAxis, DxConstantLine, } from 'devextreme-vue/chart'; import { complaintsData } from './data.js'; export default { components: { DxChart, DxArgumentAxis, DxCommonSeriesSettings, DxLabel, DxLegend, DxSeries, DxTooltip, DxValueAxis, DxConstantLine, }, data() { const data = complaintsData.sort((a, b) => b.count - a.count); const totalCount = data.reduce((prevValue, item) => prevValue + item.count, 0); let cumulativeCount = 0; const dataSource = data.map((item) => { cumulativeCount += item.count; return { complaint: item.complaint, count: item.count, cumulativePercentage: Math.round((cumulativeCount * 100) / totalCount), }; }); return { dataSource, }; }, methods: { customizeTooltip(pointInfo) { return { html: `<div><div class='tooltip-header'>${ pointInfo.argumentText }</div><div class='tooltip-body'><div class='series-name'><span class='top-series-name'>${ pointInfo.points[0].seriesName }</span>: </div><div class='value-text'><span class='top-series-value'>${ pointInfo.points[0].valueText }</span></div><div class='series-name'><span class='bottom-series-name'>${ pointInfo.points[1].seriesName }</span>: </div><div class='value-text'><span class='bottom-series-value'>${ pointInfo.points[1].valueText }</span>% </div></div></div>`, }; }, customizePercentageText({ valueText }) { return `${valueText}%`; }, }, }; </script> <style> #chart { height: 440px; } .tooltip-header { margin-bottom: 5px; font-size: 16px; font-weight: 500; padding-bottom: 5px; border-bottom: 1px solid #c5c5c5; } .tooltip-body { width: 170px; } .tooltip-body .series-name { font-weight: normal; opacity: 0.6; display: inline-block; line-height: 1.5; padding-right: 10px; width: 126px; } .tooltip-body .value-text { display: inline-block; line-height: 1.5; width: 30px; } </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="https://cdn3.devexpress.com/jslib/23.1.5/css/dx.light.css" /> <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.js"); </script> </head> <body class="dx-viewport"> <div class="demo-container"> <div id="app"> </div> </div> </body> </html>
export const complaintsData = [ { complaint: 'Cold pizza', count: 780 }, { complaint: 'Not enough cheese', count: 120 }, { complaint: 'Underbaked or Overbaked', count: 52 }, { complaint: 'Delayed delivery', count: 1123 }, { complaint: 'Damaged pizza', count: 321 }, { complaint: 'Incorrect billing', count: 89 }, { complaint: 'Wrong size delivered', count: 222 }, ];
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:': 'https://unpkg.com/', }, 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);