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
import React, { useCallback, useState } from 'react';
import Scheduler, { Editing, SchedulerTypes } from 'devextreme-react/scheduler';
import SelectBox, { SelectBoxTypes } from 'devextreme-react/select-box';
import * as timeZoneUtils from 'devextreme/time_zone_utils';
import { data, locations } from './data.ts';
const timeZoneLabel = { 'aria-label': 'Time zone' };
const currentDate = new Date(2021, 3, 27);
const views: SchedulerTypes.ViewType[] = ['workWeek'];
const getTimeZones = (date: Date) => {
const timeZones = timeZoneUtils.getTimeZones(date);
return timeZones.filter((timeZone: { id: string; }) => locations.indexOf(timeZone.id) !== -1);
};
const defaultTimeZones = getTimeZones(currentDate);
const onAppointmentFormOpening = (e: SchedulerTypes.AppointmentFormOpeningEvent) => {
const { form } = e;
const startDateTimezoneEditor = form.getEditor('startDateTimeZone');
const endDateTimezoneEditor = form.getEditor('endDateTimeZone');
const startDateDataSource = startDateTimezoneEditor?.option('dataSource') as any;
const endDateDataSource = endDateTimezoneEditor?.option('dataSource') as any;
startDateDataSource.filter(['id', 'contains', 'Europe']);
endDateDataSource.filter(['id', 'contains', 'Europe']);
startDateDataSource.load();
endDateDataSource.load();
};
const App = () => {
const [currentTimeZone, setCurrentTimeZone] = useState(defaultTimeZones[0].id);
const [timeZones, setTimeZones] = useState(defaultTimeZones);
const onValueChanged = useCallback((e: SelectBoxTypes.ValueChangedEvent) => {
setCurrentTimeZone(e.value);
}, []);
const onOptionChanged = useCallback((e: SchedulerTypes.OptionChangedEvent) => {
if (e.name === 'currentDate') {
setTimeZones(getTimeZones(e.value));
}
}, []);
return (
<React.Fragment>
<div className="option">
<span>Office Time Zone</span>
<SelectBox
items={timeZones}
displayExpr="title"
valueExpr="id"
inputAttr={timeZoneLabel}
width={240}
value={currentTimeZone}
onValueChanged={onValueChanged}
/>
</div>
<Scheduler
dataSource={data}
views={views}
defaultCurrentView="workWeek"
startDayHour={8}
defaultCurrentDate={currentDate}
timeZone={currentTimeZone}
height={600}
onAppointmentFormOpening={onAppointmentFormOpening}
onOptionChanged={onOptionChanged}
>
<Editing allowTimeZoneEditing={true} />
</Scheduler>
</React.Fragment>
);
};
export default App;
import React, { useCallback, useState } from 'react';
import Scheduler, { Editing } from 'devextreme-react/scheduler';
import SelectBox from 'devextreme-react/select-box';
import * as timeZoneUtils from 'devextreme/time_zone_utils';
import { data, locations } from './data.js';
const timeZoneLabel = { 'aria-label': 'Time zone' };
const currentDate = new Date(2021, 3, 27);
const views = ['workWeek'];
const getTimeZones = (date) => {
const timeZones = timeZoneUtils.getTimeZones(date);
return timeZones.filter((timeZone) => locations.indexOf(timeZone.id) !== -1);
};
const defaultTimeZones = getTimeZones(currentDate);
const onAppointmentFormOpening = (e) => {
const { form } = e;
const startDateTimezoneEditor = form.getEditor('startDateTimeZone');
const endDateTimezoneEditor = form.getEditor('endDateTimeZone');
const startDateDataSource = startDateTimezoneEditor?.option('dataSource');
const endDateDataSource = endDateTimezoneEditor?.option('dataSource');
startDateDataSource.filter(['id', 'contains', 'Europe']);
endDateDataSource.filter(['id', 'contains', 'Europe']);
startDateDataSource.load();
endDateDataSource.load();
};
const App = () => {
const [currentTimeZone, setCurrentTimeZone] = useState(defaultTimeZones[0].id);
const [timeZones, setTimeZones] = useState(defaultTimeZones);
const onValueChanged = useCallback((e) => {
setCurrentTimeZone(e.value);
}, []);
const onOptionChanged = useCallback((e) => {
if (e.name === 'currentDate') {
setTimeZones(getTimeZones(e.value));
}
}, []);
return (
<React.Fragment>
<div className="option">
<span>Office Time Zone</span>
<SelectBox
items={timeZones}
displayExpr="title"
valueExpr="id"
inputAttr={timeZoneLabel}
width={240}
value={currentTimeZone}
onValueChanged={onValueChanged}
/>
</div>
<Scheduler
dataSource={data}
views={views}
defaultCurrentView="workWeek"
startDayHour={8}
defaultCurrentDate={currentDate}
timeZone={currentTimeZone}
height={600}
onAppointmentFormOpening={onAppointmentFormOpening}
onOptionChanged={onOptionChanged}
>
<Editing allowTimeZoneEditing={true} />
</Scheduler>
</React.Fragment>
);
};
export default App;
import React from 'react';
import ReactDOM from 'react-dom';
import App from './App.tsx';
ReactDOM.render(
<App />,
document.getElementById('app'),
);
import { SchedulerTypes } from 'devextreme-react/scheduler';
export const data: SchedulerTypes.Appointment[] = [
{
text: 'Stand-up meeting',
startDate: '2021-04-26T15:30:00.000Z',
endDate: '2021-04-26T15:45:00.000Z',
recurrenceRule: 'FREQ=DAILY',
}, {
text: 'Book Flights to San Fran for Sales Trip',
startDate: '2021-04-28T18:00:00.000Z',
endDate: '2021-04-28T19:00:00.000Z',
}, {
text: 'New Brochures',
startDate: '2021-04-30T18:30:00.000Z',
endDate: '2021-04-30T18:45:00.000Z',
}, {
text: 'Website Re-Design Plan',
startDate: '2021-04-27T12:30:00.000Z',
endDate: '2021-04-27T13:30:00.000Z',
}, {
text: 'Book Flights to San Fran for Sales Trip',
startDate: '2021-04-28T16:00:00.000Z',
endDate: '2021-04-28T15:00:00.000Z',
}, {
text: 'Prepare 2021 Marketing Plan',
startDate: '2021-04-26T07:00:00.000Z',
endDate: '2021-04-26T09:30:00.000Z',
}, {
text: 'Launch New Website',
startDate: '2021-04-28T08:00:00.000Z',
endDate: '2021-04-28T10:00:00.000Z',
}, {
text: 'Submit New Website Design',
startDate: '2021-04-29T09:30:00.000Z',
endDate: '2021-04-29T11:00:00.000Z',
}, {
text: 'Upgrade Server Hardware',
startDate: '2021-04-30T06:30:00.000Z',
endDate: '2021-04-30T08:00:00.000Z',
}, {
text: 'Approve New Online Marketing Strategy',
startDate: '2021-04-30T11:00:00.000Z',
endDate: '2021-04-30T12:30:00.000Z',
}, {
text: 'Final Budget Review',
startDate: '2021-04-27T09:00:00.000Z',
endDate: '2021-04-27T10:35:00.000Z',
},
];
export const locations = ['Europe/London', 'Europe/Berlin', 'Europe/Helsinki'];
window.exports = window.exports || {};
window.config = {
transpiler: 'ts',
typescriptOptions: {
module: 'system',
emitDecoratorMetadata: true,
experimentalDecorators: true,
jsx: 'react',
},
meta: {
'react': {
'esModule': true,
},
'typescript': {
'exports': 'ts',
},
'devextreme/time_zone_utils.js': {
'esModule': true,
},
'devextreme/localization.js': {
'esModule': true,
},
'devextreme/viz/palette.js': {
'esModule': true,
},
'openai': {
'esModule': true,
},
},
paths: {
'npm:': 'https://unpkg.com/',
'bundles:': '../../../../bundles/',
'externals:': '../../../../bundles/externals/',
},
defaultExtension: 'js',
map: {
'ts': 'npm:plugin-typescript@8.0.0/lib/plugin.js',
'typescript': 'npm:typescript@4.2.4/lib/typescript.js',
'jszip': 'npm:jszip@3.10.1/dist/jszip.min.js',
'react': 'npm:react@17.0.2/umd/react.development.js',
'react-dom': 'npm:react-dom@17.0.2/umd/react-dom.development.js',
'prop-types': 'npm:prop-types/prop-types.js',
'rrule': 'npm:rrule@2.6.4/dist/es5/rrule.js',
'luxon': 'npm:luxon@3.4.4/build/global/luxon.min.js',
'es6-object-assign': 'npm:es6-object-assign',
'devextreme': 'npm:devextreme@link:../../packages/devextreme/artifacts/npm/devextreme/cjs',
'devextreme-react': 'npm:devextreme-react@link:../../packages/devextreme-react/npm/cjs',
'devextreme-quill': 'npm:devextreme-quill@1.7.1/dist/dx-quill.min.js',
'devexpress-diagram': 'npm:devexpress-diagram@2.2.5/dist/dx-diagram.js',
'devexpress-gantt': 'npm:devexpress-gantt@4.1.54/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/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',
'devextreme-cldr-data': 'npm:devextreme-cldr-data@1.0.3',
// SystemJS plugins
'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': {
defaultExtension: 'js',
},
'devextreme-react': {
main: 'index.js',
},
'devextreme/events/utils': {
main: 'index',
},
'devextreme/localization/messages': {
format: 'json',
defaultExtension: 'json',
},
'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,
react: true,
},
};
System.config(window.config);
import React from 'react';
import ReactDOM from 'react-dom';
import App from './App.js';
ReactDOM.render(<App />, document.getElementById('app'));
export const data = [
{
text: 'Stand-up meeting',
startDate: '2021-04-26T15:30:00.000Z',
endDate: '2021-04-26T15:45:00.000Z',
recurrenceRule: 'FREQ=DAILY',
},
{
text: 'Book Flights to San Fran for Sales Trip',
startDate: '2021-04-28T18:00:00.000Z',
endDate: '2021-04-28T19:00:00.000Z',
},
{
text: 'New Brochures',
startDate: '2021-04-30T18:30:00.000Z',
endDate: '2021-04-30T18:45:00.000Z',
},
{
text: 'Website Re-Design Plan',
startDate: '2021-04-27T12:30:00.000Z',
endDate: '2021-04-27T13:30:00.000Z',
},
{
text: 'Book Flights to San Fran for Sales Trip',
startDate: '2021-04-28T16:00:00.000Z',
endDate: '2021-04-28T15:00:00.000Z',
},
{
text: 'Prepare 2021 Marketing Plan',
startDate: '2021-04-26T07:00:00.000Z',
endDate: '2021-04-26T09:30:00.000Z',
},
{
text: 'Launch New Website',
startDate: '2021-04-28T08:00:00.000Z',
endDate: '2021-04-28T10:00:00.000Z',
},
{
text: 'Submit New Website Design',
startDate: '2021-04-29T09:30:00.000Z',
endDate: '2021-04-29T11:00:00.000Z',
},
{
text: 'Upgrade Server Hardware',
startDate: '2021-04-30T06:30:00.000Z',
endDate: '2021-04-30T08:00:00.000Z',
},
{
text: 'Approve New Online Marketing Strategy',
startDate: '2021-04-30T11:00:00.000Z',
endDate: '2021-04-30T12:30:00.000Z',
},
{
text: 'Final Budget Review',
startDate: '2021-04-27T09:00:00.000Z',
endDate: '2021-04-27T10:35:00.000Z',
},
];
export const locations = ['Europe/London', 'Europe/Berlin', 'Europe/Helsinki'];
<!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.2.3/css/dx.light.css" />
<link rel="stylesheet" type="text/css" href="styles.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.tsx");
</script>
</head>
<body class="dx-viewport">
<div class="demo-container">
<div id="app"></div>
</div>
</body>
</html>
.option {
display: flex;
}
.option > span {
display: flex;
align-items: center;
margin-right: 10px;
}
.dx-scheduler {
margin-top: 20px;
}
Users can edit the time zones of individual appointments in the appointment details form. To enable this functionality, set the editing.allowTimeZoneEditing property to true. Information about individual time zones is saved in the startDateTimeZone and endDateTimeZone fields of the appointment data objects.