Use Range Selection for Calculation
Documentation
The RangeSelector component allows you to handle changes of the currently selected range. For this purpose, the callback function is specified in the onValueChanged property. Select the function’s call mode using the «Handle Range Changes» drop-down menu.
Feel free to share demo-related thoughts here.
If you have technical questions, please create a support ticket in the DevExpress Support Center.
Thank you for the feedback!
If you have technical questions, please create a support ticket in the DevExpress Support Center.
Backend API
<template>
<div>
<div id="range-selector-demo">
<DxRangeSelector
id="range-selector"
v-model:value="range"
title="Calculate the Working Days Count in a Date Period"
>
<DxMargin :top="50"/>
<DxScale
:start-value="startValue"
:end-value="endValue"
tick-interval="month"
minor-tick-interval="day"
>
<DxMinorTick :visible="false"/>
<DxMarker :visible="false"/>
<DxLabel format="MMM"/>
</DxScale>
<DxBehavior :value-change-mode="currentBehaviorMode"/>
<DxSliderMarker format="dd EEEE"/>
</DxRangeSelector>
<h2>Working days count: {{ workingDaysCount }}</h2>
<div class="options">
<div class="caption">Options</div>
<div class="option">
<span>Handle Range Changes </span>
<DxSelectBox
:data-source="behaviorModes"
:input-attr="{ 'aria-label': 'Value Change Mode' }"
:width="210"
v-model:value="currentBehaviorMode"
/>
</div>
</div>
</div>
</div>
</template>
<script>
import {
DxRangeSelector,
DxMargin,
DxScale,
DxMinorTick,
DxMarker,
DxLabel,
DxBehavior,
DxSliderMarker,
} from 'devextreme-vue/range-selector';
import { DxSelectBox } from 'devextreme-vue/select-box';
export default {
components: {
DxRangeSelector,
DxMargin,
DxScale,
DxMinorTick,
DxMarker,
DxLabel,
DxBehavior,
DxSliderMarker,
DxSelectBox,
},
data() {
const range = [new Date(2011, 0, 1), new Date(2011, 11, 31)];
const behaviorModes = ['onHandleMove', 'onHandleRelease'];
return {
startValue: range[0],
endValue: range[1],
range,
behaviorModes,
currentBehaviorMode: behaviorModes[0],
};
},
computed: {
workingDaysCount() {
const currentDate = new Date(this.range[0]);
let workingDaysCount = 0;
while (currentDate <= this.range[1]) {
if (currentDate.getDay() > 0 && currentDate.getDay() < 6) {
workingDaysCount += 1;
}
currentDate.setDate(currentDate.getDate() + 1);
}
return workingDaysCount;
},
},
};
</script>
<style scoped>
#range-selector {
height: 200px;
}
h2 {
text-align: center;
}
.options {
padding: 20px;
background-color: rgba(191, 191, 191, 0.15);
position: absolute;
bottom: 0;
left: 0;
right: 0;
}
.option {
margin-top: 10px;
}
.caption {
font-size: 18px;
font-weight: 500;
}
.option > span {
margin-right: 10px;
}
.option > .dx-widget {
display: inline-block;
vertical-align: middle;
}
</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="dx-theme" data-theme="generic.light" 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>
window.exports = window.exports || {};
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.6/cjs',
'devextreme-vue': 'npm:devextreme-vue@23.1.6',
'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.2/dist/dx-diagram.js',
'devexpress-gantt': 'npm:devexpress-gantt@4.1.49/dist/dx-gantt.js',
'@devextreme/runtime': 'npm:@devextreme/runtime@3.0.12',
'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.12/inferno/package.json',
],
babelOptions: {
sourceMaps: false,
stage0: true,
},
};
System.config(window.config);