devextreme
Version:
JavaScript/TypeScript Component Suite for Responsive Web Development
1,048 lines (1,047 loc) • 84.5 kB
JavaScript
/**
* DevExtreme (cjs/__internal/scheduler/m_scheduler.js)
* Version: 25.2.7
* Build date: Tue May 05 2026
*
* Copyright (c) 2012 - 2026 Developer Express Inc. ALL RIGHTS RESERVED
* Read about DevExtreme licensing here: https://js.devexpress.com/Licensing/
*/
"use strict";
Object.defineProperty(exports, "__esModule", {
value: true
});
exports.default = exports.POPUP_DIALOG_CLASS = void 0;
var _visibility_change = require("../../common/core/events/visibility_change");
var _date = _interopRequireDefault(require("../../common/core/localization/date"));
var _message = _interopRequireDefault(require("../../common/core/localization/message"));
var _component_registrator = _interopRequireDefault(require("../../core/component_registrator"));
var _config = _interopRequireDefault(require("../../core/config"));
var _element = require("../../core/element");
var _renderer = _interopRequireDefault(require("../../core/renderer"));
var _bindable_template = require("../../core/templates/bindable_template");
var _empty_template = require("../../core/templates/empty_template");
var _callbacks = _interopRequireDefault(require("../../core/utils/callbacks"));
var _common = require("../../core/utils/common");
var _data = require("../../core/utils/data");
var _date2 = _interopRequireDefault(require("../../core/utils/date"));
var _date_serialization = _interopRequireDefault(require("../../core/utils/date_serialization"));
var _deferred = require("../../core/utils/deferred");
var _extend = require("../../core/utils/extend");
var _position = require("../../core/utils/position");
var _type = require("../../core/utils/type");
var _window = require("../../core/utils/window");
var _data_helper = _interopRequireDefault(require("../../data_helper"));
var _dialog = require("../../ui/dialog");
var _ui = _interopRequireDefault(require("../../ui/widget/ui.errors"));
var _date3 = require("../core/utils/date");
var _a11y_status_render = require("./a11y_status/a11y_status_render");
var _a11y_status_text = require("./a11y_status/a11y_status_text");
var _m_form = require("./appointment_popup/m_form");
var _m_popup = require("./appointment_popup/m_popup");
var _m_appointment_collection = _interopRequireDefault(require("./appointments/m_appointment_collection"));
var _appointments = require("./appointments_new/appointments");
var _m_widget_notify_scheduler = _interopRequireDefault(require("./base/m_widget_notify_scheduler"));
var _m_header = require("./header/m_header");
var _m_compact_appointments_helper = require("./m_compact_appointments_helper");
var _m_loading = require("./m_loading");
var _m_subscribes = _interopRequireDefault(require("./m_subscribes"));
var _m_utils = require("./m_utils");
var _m_utils_time_zone = _interopRequireDefault(require("./m_utils_time_zone"));
var _remote = require("./r1/filterting/remote");
var _index = require("./r1/timezone_calculator/index");
var _index2 = require("./r1/utils/index");
var _validate_rule = require("./recurrence/validate_rule");
var _scheduler_options_base_widget = require("./scheduler_options_base_widget");
var _m_desktop_tooltip_strategy = require("./tooltip_strategies/m_desktop_tooltip_strategy");
var _m_mobile_tooltip_strategy = require("./tooltip_strategies/m_mobile_tooltip_strategy");
var _appointment_adapter = require("./utils/appointment_adapter/appointment_adapter");
var _appointment_data_accessor = require("./utils/data_accessor/appointment_data_accessor");
var _get_targeted_appointment = require("./utils/get_targeted_appointment");
var _index3 = require("./utils/index");
var _is_agenda_workpace_component = require("./utils/is_agenda_workpace_component");
var _constants_view = require("./utils/options/constants_view");
var _appointment_groups_utils = require("./utils/resource_manager/appointment_groups_utils");
var _resource_manager = require("./utils/resource_manager/resource_manager");
var _appointments_layout_manager = _interopRequireDefault(require("./view_model/appointments_layout_manager"));
var _m_appointment_data_source = require("./view_model/m_appointment_data_source");
var _m_agenda = _interopRequireDefault(require("./workspaces/m_agenda"));
var _m_timeline_day = _interopRequireDefault(require("./workspaces/m_timeline_day"));
var _m_timeline_month = _interopRequireDefault(require("./workspaces/m_timeline_month"));
var _m_timeline_week = _interopRequireDefault(require("./workspaces/m_timeline_week"));
var _m_timeline_work_week = _interopRequireDefault(require("./workspaces/m_timeline_work_week"));
var _m_work_space_day = _interopRequireDefault(require("./workspaces/m_work_space_day"));
var _m_work_space_month = _interopRequireDefault(require("./workspaces/m_work_space_month"));
var _m_work_space_week = _interopRequireDefault(require("./workspaces/m_work_space_week"));
var _m_work_space_work_week = _interopRequireDefault(require("./workspaces/m_work_space_work_week"));
function _interopRequireDefault(e) {
return e && e.__esModule ? e : {
default: e
}
}
const toMs = _date2.default.dateToMilliseconds;
const WIDGET_CLASS = "dx-scheduler";
const WIDGET_SMALL_CLASS = `${WIDGET_CLASS}-small`;
const WIDGET_ADAPTIVE_CLASS = `${WIDGET_CLASS}-adaptive`;
const WIDGET_READONLY_CLASS = `${WIDGET_CLASS}-readonly`;
const POPUP_DIALOG_CLASS = exports.POPUP_DIALOG_CLASS = "dx-dialog";
const WIDGET_SMALL_WIDTH = 400;
const FULL_DATE_FORMAT = "yyyyMMddTHHmmss";
const UTC_FULL_DATE_FORMAT = `${FULL_DATE_FORMAT}Z`;
const VIEWS_CONFIG = {
day: {
workSpace: _m_work_space_day.default,
renderingStrategy: "vertical"
},
week: {
workSpace: _m_work_space_week.default,
renderingStrategy: "vertical"
},
workWeek: {
workSpace: _m_work_space_work_week.default,
renderingStrategy: "vertical"
},
month: {
workSpace: _m_work_space_month.default,
renderingStrategy: "horizontalMonth"
},
timelineDay: {
workSpace: _m_timeline_day.default,
renderingStrategy: "horizontal"
},
timelineWeek: {
workSpace: _m_timeline_week.default,
renderingStrategy: "horizontal"
},
timelineWorkWeek: {
workSpace: _m_timeline_work_week.default,
renderingStrategy: "horizontal"
},
timelineMonth: {
workSpace: _m_timeline_month.default,
renderingStrategy: "horizontalMonthLine"
},
agenda: {
workSpace: _m_agenda.default,
renderingStrategy: "agenda"
}
};
const StoreEventNames = {
ADDING: "onAppointmentAdding",
ADDED: "onAppointmentAdded",
DELETING: "onAppointmentDeleting",
DELETED: "onAppointmentDeleted",
UPDATING: "onAppointmentUpdating",
UPDATED: "onAppointmentUpdated"
};
const RECURRENCE_EDITING_MODE = {
SERIES: "editSeries",
OCCURRENCE: "editOccurrence",
CANCEL: "cancel"
};
class Scheduler extends _scheduler_options_base_widget.SchedulerOptionsBaseWidget {
constructor() {
super(...arguments);
this.updatingAppointments = new Set
}
get timeZoneCalculator() {
if (!this.timeZoneCalculatorInstance) {
this.timeZoneCalculatorInstance = (0, _index.createTimeZoneCalculator)(this.option("timeZone"))
}
return this.timeZoneCalculatorInstance
}
postponeDataSourceLoading(promise) {
this.postponedOperations.add("_reloadDataSource", this.reloadDataSource.bind(this), promise)
}
postponeResourceLoading() {
let forceReload = arguments.length > 0 && void 0 !== arguments[0] ? arguments[0] : false;
const whenLoaded = this.postponedOperations.add("loadResources", () => {
const groups = this.getViewOption("groups");
return (0, _deferred.fromPromise)(this.resourceManager.loadGroupResources(groups, forceReload))
});
const resolveCallbacks = new _deferred.Deferred;
whenLoaded.done(() => {
resolveCallbacks.resolve()
});
this.postponeDataSourceLoading(whenLoaded);
return resolveCallbacks.promise()
}
_optionChanged(args) {
var _this$resourceManager;
this.schedulerOptionChanged(args);
const {
value: value,
name: name
} = args;
switch (args.name) {
case "customizeDateNavigatorText":
this.updateOption("header", name, value);
break;
case "firstDayOfWeek":
this.updateOption("workSpace", name, value);
this.updateOption("header", name, value);
this.cleanPopup();
break;
case "currentDate": {
const dateValue = this.getViewOption(name);
this.option("selectedCellData", []);
this.updateOption("workSpace", name, dateValue);
this.updateOption("header", name, dateValue);
this.updateOption("header", "startViewDate", this.getStartViewDate());
this._appointments.option("items", []);
this.setRemoteFilterIfNeeded();
this.postponeDataSourceLoading();
break
}
case "dataSource":
this._initDataSource();
this.postponeResourceLoading().done(() => {
this.appointmentDataSource.setDataSource(this._dataSource);
this.setRemoteFilterIfNeeded();
this.updateOption("workSpace", "showAllDayPanel", this.option("showAllDayPanel"))
});
break;
case "min":
case "max": {
const value = this.getViewOption(name);
this.updateOption("header", name, value);
this.updateOption("workSpace", name, value);
break
}
case "views":
if (this.currentView) {
this.repaint()
} else {
this.updateOption("header", "views", this.views)
}
break;
case "useDropDownViewSwitcher":
this.updateOption("header", name, value);
break;
case "currentView":
if (this.option("_newAppointments")) {
this._appointments.option({
currentView: value,
viewModel: [],
appointmentTemplate: this.getViewOption("appointmentTemplate"),
appointmentCollectorTemplate: this.getViewOption("appointmentCollectorTemplate")
})
} else {
this._appointments.option({
items: [],
allowDrag: this.allowDragging(),
allowResize: this.allowResizing(),
itemTemplate: this.getAppointmentTemplate("appointmentTemplate")
})
}
this.postponeResourceLoading().done(() => {
var _this$header;
this.refreshWorkSpace();
null === (_this$header = this.header) || void 0 === _this$header || _this$header.option(this.headerConfig());
this.setRemoteFilterIfNeeded();
if (!this.option("_newAppointments")) {
this._appointments.option("allowAllDayResize", "day" !== value)
}
});
this.postponedOperations.callPostponedOperations();
break;
case "appointmentTemplate":
if (this.option("_newAppointments")) {
this._appointments.option("appointmentTemplate", this.getViewOption("appointmentTemplate"))
} else {
this._appointments.option("itemTemplate", value)
}
break;
case "dateCellTemplate":
case "resourceCellTemplate":
case "dataCellTemplate":
case "timeCellTemplate":
this.repaint();
break;
case "groups":
this.postponeResourceLoading().done(() => {
this.refreshWorkSpace();
this.setRemoteFilterIfNeeded()
});
break;
case "resources":
null === (_this$resourceManager = this.resourceManager) || void 0 === _this$resourceManager || _this$resourceManager.dispose();
this.resourceManager = new _resource_manager.ResourceManager(this.option("resources"));
this.updateAppointmentDataSource();
this.postponeResourceLoading().done(() => {
this._appointments.option("items", []);
this.refreshWorkSpace();
this.setRemoteFilterIfNeeded();
this.createAppointmentPopupForm()
});
break;
case "startDayHour":
case "endDayHour":
this.updateAppointmentDataSource();
this._appointments.option("items", []);
this.updateOption("workSpace", name, value);
if (!this.option("_newAppointments")) {
this._appointments.repaint()
}
this.setRemoteFilterIfNeeded();
this.postponeDataSourceLoading();
break;
case "offset":
this.updateAppointmentDataSource();
this._appointments.option("items", []);
this.updateOption("workSpace", "viewOffset", this.normalizeViewOffsetValue(value));
if (!this.option("_newAppointments")) {
this._appointments.repaint()
}
this.setRemoteFilterIfNeeded();
this.postponeDataSourceLoading();
break;
case StoreEventNames.ADDING:
case StoreEventNames.ADDED:
case StoreEventNames.UPDATING:
case StoreEventNames.UPDATED:
case StoreEventNames.DELETING:
case StoreEventNames.DELETED:
case "onAppointmentFormOpening":
case "onAppointmentTooltipShowing":
this.actions[name] = this._createActionByOption(name);
break;
case "onAppointmentRendered":
if (this.option("_newAppointments")) {
this.createAppointmentRenderedAction()
} else {
this._appointments.option("onItemRendered", this.getAppointmentRenderedAction())
}
break;
case "onAppointmentClick":
this._appointments.option("onItemClick", this._createActionByOption(name));
break;
case "onAppointmentDblClick":
this._appointments.option(name, this._createActionByOption(name));
break;
case "onAppointmentContextMenu":
this._appointments.option("onItemContextMenu", this._createActionByOption(name));
this.appointmentTooltip._options.onItemContextMenu = this._createActionByOption(name);
break;
case "noDataText":
case "allowMultipleCellSelection":
case "selectedCellData":
case "accessKey":
case "onCellClick":
case "onCellContextMenu":
this.updateOption("workSpace", name, value);
break;
case "crossScrollingEnabled":
this.postponeResourceLoading().done(() => {
this._appointments.option("items", []);
this.refreshWorkSpace();
if (this.readyToRenderAppointments) {
this._appointments.option("items", this._layoutManager.generateViewModel())
}
});
break;
case "cellDuration":
this.updateOption("workSpace", name, value);
this._appointments.option("items", []);
if (this.readyToRenderAppointments) {
this.updateOption("workSpace", "hoursInterval", value / 60);
this._appointments.option("items", this._layoutManager.generateViewModel())
}
break;
case "tabIndex":
case "focusStateEnabled":
this.updateOption("header", name, value);
this.updateOption("workSpace", name, value);
this._appointments.option(name, value);
super._optionChanged(args);
break;
case "width":
this.updateOption("header", name, value);
if (this.option("crossScrollingEnabled")) {
this.updateOption("workSpace", "width", value)
}
this.updateOption("workSpace", "schedulerWidth", value);
super._optionChanged(args);
this._dimensionChanged(null, true);
break;
case "height":
super._optionChanged(args);
this._dimensionChanged(null, true);
this.updateOption("workSpace", "schedulerHeight", value);
break;
case "editing": {
this.initEditing();
const {
editing: editing
} = this;
this.bringEditingModeToAppointments(editing);
this.hideAppointmentTooltip();
this.cleanPopup();
break
}
case "showAllDayPanel":
this.updateAppointmentDataSource();
this.repaint();
break;
case "showCurrentTimeIndicator":
case "indicatorUpdateInterval":
case "shadeUntilCurrentTime":
case "groupByDate":
this.updateOption("workSpace", name, value);
this.repaint();
break;
case "indicatorTime":
this.updateOption("workSpace", name, value);
this.updateOption("header", name, value);
this.repaint();
break;
case "appointmentDragging":
case "appointmentTooltipTemplate":
case "appointmentPopupTemplate":
case "recurrenceEditMode":
case "remoteFiltering":
case "timeZone":
this.updateAppointmentDataSource();
this.repaint();
break;
case "appointmentCollectorTemplate":
if (this.option("_newAppointments")) {
this._appointments.option("appointmentCollectorTemplate", this.getViewOption("appointmentCollectorTemplate"))
} else {
this.repaint()
}
break;
case "_appointmentTooltipOffset":
this.repaint();
break;
case "dateSerializationFormat":
break;
case "maxAppointmentsPerCell":
this.repaint();
break;
case "startDateExpr":
case "endDateExpr":
case "startDateTimeZoneExpr":
case "endDateTimeZoneExpr":
case "textExpr":
case "descriptionExpr":
case "allDayExpr":
case "recurrenceRuleExpr":
case "recurrenceExceptionExpr":
case "disabledExpr":
case "visibleExpr":
this.updateExpression(name, value);
this.initAppointmentTemplate();
this.repaint();
break;
case "adaptivityEnabled":
this.toggleAdaptiveClass();
this.repaint();
break;
case "scrolling":
this.option("crossScrollingEnabled", this.isHorizontalVirtualScrolling() || this.option("crossScrollingEnabled"));
this.updateOption("workSpace", args.fullName, value);
break;
case "allDayPanelMode":
this.updateAppointmentDataSource();
this.updateOption("workSpace", args.fullName, value);
break;
case "renovateRender":
this.updateOption("workSpace", name, value);
break;
case "_draggingMode":
this.updateOption("workSpace", "draggingMode", value);
break;
case "toolbar":
this.header ? this.header.onToolbarOptionChanged(args.fullName, value) : this.repaint();
break;
default:
super._optionChanged(args)
}
}
bringEditingModeToAppointments(editing) {
const editingConfig = {
allowDelete: editing.allowUpdating && editing.allowDeleting
};
if (!this.isAgenda()) {
editingConfig.allowDrag = editing.allowDragging;
editingConfig.allowResize = editing.allowResizing;
editingConfig.allowAllDayResize = editing.allowResizing && this.supportAllDayResizing()
}
this._appointments.option(editingConfig);
this.repaint()
}
isAgenda() {
return "agenda" === this.currentView.type
}
allowDragging() {
return this.editing.allowDragging && !this.isAgenda()
}
allowResizing() {
return this.editing.allowResizing && !this.isAgenda()
}
allowAllDayResizing() {
return this.editing.allowResizing && this.supportAllDayResizing()
}
supportAllDayResizing() {
return "day" !== this.currentView.type || this.currentView.intervalCount > 1
}
isAllDayExpanded() {
return this.option("showAllDayPanel") && this._layoutManager.hasAllDayAppointments()
}
setRemoteFilterIfNeeded() {
const dataSource = this._dataSource;
const remoteFiltering = this.option("remoteFiltering");
if (!this._workSpace || !remoteFiltering || !dataSource) {
return
}
const dateRange = this._workSpace.getDateRange();
const startDate = this.timeZoneCalculator.createDate(dateRange[0], "fromGrid");
const endDate = this.timeZoneCalculator.createDate(dateRange[1], "fromGrid");
const dateSerializationFormat = this.option("dateSerializationFormat");
const dataSourceFilter = dataSource.filter();
const filter = (0, _remote.combineRemoteFilter)({
dataSourceFilter: dataSourceFilter,
dataAccessors: this._dataAccessors,
min: startDate,
max: endDate,
dateSerializationFormat: dateSerializationFormat,
forceIsoDateParsing: (0, _config.default)().forceIsoDateParsing
});
dataSource.filter(filter)
}
reloadDataSource() {
const result = new _deferred.Deferred;
if (this._dataSource) {
this._dataSource.load().done(() => {
(0, _m_loading.hide)();
this._fireContentReadyAction(result)
}).fail(() => {
(0, _m_loading.hide)();
result.reject()
});
this._dataSource.isLoading() && (0, _m_loading.show)({
container: this.$element(),
position: {
of: this.$element()
}
})
} else {
this._fireContentReadyAction(result)
}
return result.promise()
}
_fireContentReadyAction(result) {
const contentReadyBase = super._fireContentReadyAction.bind(this);
const fireContentReady = () => {
contentReadyBase();
null === result || void 0 === result || result.resolve()
};
if (this.workSpaceRecalculation) {
var _this$workSpaceRecalc;
null === (_this$workSpaceRecalc = this.workSpaceRecalculation) || void 0 === _this$workSpaceRecalc || _this$workSpaceRecalc.done(() => {
fireContentReady()
})
} else {
fireContentReady()
}
}
_dimensionChanged(value) {
let isForce = arguments.length > 1 && void 0 !== arguments[1] ? arguments[1] : false;
const isFixedHeight = "number" === typeof this.option("height");
const isFixedWidth = "number" === typeof this.option("width");
if (!this._isVisible()) {
return
}
this.toggleSmallClass();
const workspace = this.getWorkSpace();
if (!this.isAgenda() && this._layoutManager && workspace && !(0, _is_agenda_workpace_component.isAgendaWorkspaceComponent)(workspace)) {
if (isForce || !isFixedHeight || !isFixedWidth) {
workspace.option("allDayExpanded", this.isAllDayExpanded());
workspace._dimensionChanged();
const appointments = this._layoutManager.generateViewModel();
this._appointments.option("items", appointments)
}
}
this.hideAppointmentTooltip();
this.appointmentPopup.triggerResize();
this.appointmentPopup.updatePopupFullScreenMode()
}
_clean() {
this.cleanPopup();
super._clean()
}
toggleSmallClass() {
const {
width: width
} = (0, _position.getBoundingRect)(this.$element().get(0));
this.$element().toggleClass(WIDGET_SMALL_CLASS, width < 400)
}
toggleAdaptiveClass() {
this.$element().toggleClass(WIDGET_ADAPTIVE_CLASS, this.option("adaptivityEnabled"))
}
_visibilityChanged(visible) {
visible && this._dimensionChanged(null, true)
}
_dataSourceOptions() {
return {
paginate: false
}
}
initAllDayPanel() {
if ("hidden" === this.option("allDayPanelMode")) {
this.option("showAllDayPanel", false)
}
}
_init() {
this.timeZonesPromise = _m_utils_time_zone.default.cacheTimeZones();
this.initExpressions({
startDateExpr: this.option("startDateExpr"),
endDateExpr: this.option("endDateExpr"),
startDateTimeZoneExpr: this.option("startDateTimeZoneExpr"),
endDateTimeZoneExpr: this.option("endDateTimeZoneExpr"),
allDayExpr: this.option("allDayExpr"),
textExpr: this.option("textExpr"),
descriptionExpr: this.option("descriptionExpr"),
recurrenceRuleExpr: this.option("recurrenceRuleExpr"),
recurrenceExceptionExpr: this.option("recurrenceExceptionExpr"),
disabledExpr: this.option("disabledExpr"),
visibleExpr: this.option("visibleExpr")
});
super._init();
this.initAllDayPanel();
this._initDataSource();
this.customizeDataSourceLoadOptions();
this.$element().addClass(WIDGET_CLASS);
this.initEditing();
this.updateAppointmentDataSource();
this.initActions();
this._compactAppointmentsHelper = new _m_compact_appointments_helper.CompactAppointmentsHelper(this);
this.asyncTemplatesTimers = [];
this.dataSourceLoadedCallback = (0, _callbacks.default)();
this.subscribes = _m_subscribes.default;
this.resourceManager = new _resource_manager.ResourceManager(this.option("resources"));
this.notifyScheduler = new _m_widget_notify_scheduler.default({
scheduler: this
});
this.createAppointmentRenderedAction()
}
createAppointmentRenderedAction() {
this.appointmentRenderedAction = this._createActionByOption("onAppointmentRendered")
}
createAppointmentDataSource() {
var _this$appointmentData;
null === (_this$appointmentData = this.appointmentDataSource) || void 0 === _this$appointmentData || _this$appointmentData.destroy();
this.appointmentDataSource = new _m_appointment_data_source.AppointmentDataSource(this._dataSource)
}
updateAppointmentDataSource() {
this.timeZoneCalculatorInstance = null;
if (this.getWorkSpace()) {
this.createAppointmentDataSource()
}
}
customizeDataSourceLoadOptions() {
var _this$_dataSource;
null === (_this$_dataSource = this._dataSource) || void 0 === _this$_dataSource || _this$_dataSource.on("customizeStoreLoadOptions", _ref => {
let {
storeLoadOptions: storeLoadOptions
} = _ref;
storeLoadOptions.startDate = this.getStartViewDate();
storeLoadOptions.endDate = this.getEndViewDate()
})
}
_initTemplates() {
this.initAppointmentTemplate();
this._templateManager.addDefaultTemplates({
appointmentTooltip: new _empty_template.EmptyTemplate,
dropDownAppointment: new _empty_template.EmptyTemplate
});
super._initTemplates()
}
initAppointmentTemplate() {
const {
expr: expr
} = this._dataAccessors;
const createGetter = property => (0, _data.compileGetter)(`appointmentData.${property}`);
const getDate = getter => data => {
const value = getter(data);
if (value instanceof Date) {
return value.valueOf()
}
return value
};
this._templateManager.addDefaultTemplates({
item: new _bindable_template.BindableTemplate(($container, data, model) => this.getAppointmentsInstance()._renderAppointmentTemplate($container, data, model), ["html", "text", "startDate", "endDate", "allDay", "description", "recurrenceRule", "recurrenceException", "startDateTimeZone", "endDateTimeZone"], this.option("integrationOptions.watchMethod"), {
text: createGetter(expr.textExpr),
startDate: getDate(createGetter(expr.startDateExpr)),
endDate: getDate(createGetter(expr.endDateExpr)),
startDateTimeZone: createGetter(expr.startDateTimeZoneExpr),
endDateTimeZone: createGetter(expr.endDateTimeZoneExpr),
allDay: createGetter(expr.allDayExpr),
recurrenceRule: createGetter(expr.recurrenceRuleExpr)
})
})
}
_renderContent() {
this._renderContentImpl()
}
_dataSourceChangedHandler(result) {
if (this.readyToRenderAppointments) {
this.workSpaceRecalculation.done(() => {
this._layoutManager.prepareAppointments(result);
this.renderAppointments();
this.updateA11yStatus()
})
}
}
isVirtualScrolling() {
const workspace = this.getWorkSpace();
if (workspace) {
return workspace.isVirtualScrolling()
}
const scrolling = this.getViewOption("scrolling");
return "virtual" === (null === scrolling || void 0 === scrolling ? void 0 : scrolling.mode)
}
renderAppointments() {
const workspace = this.getWorkSpace();
this._layoutManager.filterAppointments();
workspace.option("allDayExpanded", this.isAllDayExpanded());
const viewModel = this._isVisible() ? this._layoutManager.generateViewModel() : [];
this._appointments.option("items", viewModel);
this.appointmentDataSource.cleanState();
if (this.isAgenda()) {
this._workSpace.renderAgendaLayout(viewModel)
}
}
initExpressions(fields) {
this._dataAccessors = new _appointment_data_accessor.AppointmentDataAccessor(fields, Boolean((0, _config.default)().forceIsoDateParsing), this.option("dateSerializationFormat"))
}
updateExpression(name, value) {
this._dataAccessors.updateExpression(name, value)
}
initEditing() {
const editing = this.option("editing");
this.editing = {
allowAdding: Boolean(editing),
allowUpdating: Boolean(editing),
allowDeleting: Boolean(editing),
allowResizing: Boolean(editing),
allowDragging: Boolean(editing)
};
if ((0, _type.isObject)(editing)) {
this.editing = (0, _extend.extend)(this.editing, editing)
}
this.editing.allowDragging = this.editing.allowDragging && this.editing.allowUpdating;
this.editing.allowResizing = this.editing.allowResizing && this.editing.allowUpdating;
const isReadOnly = Object.values(Object.assign({}, this.editing, {
form: void 0,
popup: void 0
})).every(value => !value);
this.$element().toggleClass(WIDGET_READONLY_CLASS, isReadOnly)
}
_dispose() {
var _this$resourceManager2, _this$appointmentTool, _this$recurrenceDialo;
null === (_this$resourceManager2 = this.resourceManager) || void 0 === _this$resourceManager2 || _this$resourceManager2.dispose();
null === (_this$appointmentTool = this.appointmentTooltip) || void 0 === _this$appointmentTool || _this$appointmentTool.dispose();
null === (_this$recurrenceDialo = this.recurrenceDialog) || void 0 === _this$recurrenceDialo || _this$recurrenceDialo.hide(RECURRENCE_EDITING_MODE.CANCEL);
this.hideAppointmentPopup();
this.hideAppointmentTooltip();
this.asyncTemplatesTimers.forEach(clearTimeout);
this.asyncTemplatesTimers = [];
_index3.macroTaskArray.dispose();
super._dispose()
}
initActions() {
this.actions = {
onAppointmentAdding: this._createActionByOption(StoreEventNames.ADDING),
onAppointmentAdded: this._createActionByOption(StoreEventNames.ADDED),
onAppointmentUpdating: this._createActionByOption(StoreEventNames.UPDATING),
onAppointmentUpdated: this._createActionByOption(StoreEventNames.UPDATED),
onAppointmentDeleting: this._createActionByOption(StoreEventNames.DELETING),
onAppointmentDeleted: this._createActionByOption(StoreEventNames.DELETED),
onAppointmentFormOpening: this._createActionByOption("onAppointmentFormOpening"),
onAppointmentTooltipShowing: this._createActionByOption("onAppointmentTooltipShowing")
}
}
getAppointmentRenderedAction() {
return this._createActionByOption("onAppointmentRendered", {
excludeValidators: ["disabled", "readOnly"]
})
}
_renderFocusTarget() {
return (0, _common.noop)()
}
updateA11yStatus() {
const dateRange = this._workSpace.getDateRange();
const indicatorTime = this.option("showCurrentTimeIndicator") ? (0, _index2.getToday)(this.option("indicatorTime"), this.timeZoneCalculator) : void 0;
const label = (0, _a11y_status_text.getA11yStatusText)(this.currentView, dateRange[0], dateRange[1], this._layoutManager.filteredItems.length, indicatorTime);
this.setAria({
label: label
});
this.a11yStatus.text(label)
}
renderA11yStatus() {
this.a11yStatus = (0, _a11y_status_render.createA11yStatusContainer)();
this.a11yStatus.prependTo(this.$element());
this.setAria({
role: "group"
})
}
initMarkupOnResourceLoaded() {
if (!this._disposed) {
this.initMarkupCore();
this.reloadDataSource()
}
}
_initMarkup() {
super._initMarkup();
this.renderA11yStatus();
this.renderMainContainer();
this.renderHeader();
this.toggleAdaptiveClass();
this._layoutManager = new _appointments_layout_manager.default(this);
if (this.option("_newAppointments")) {
const appointmentsConfig = {
tabIndex: this.option("tabIndex"),
currentView: this.option("currentView"),
appointmentTemplate: this.getViewOption("appointmentTemplate"),
appointmentCollectorTemplate: this.getViewOption("appointmentCollectorTemplate"),
onAppointmentRendered: e => {
this.appointmentRenderedAction({
appointmentElement: e.element,
appointmentData: e.appointmentData,
targetedAppointmentData: e.targetedAppointmentData
})
},
getResourceManager: () => this.resourceManager,
getAppointmentDataSource: () => this.appointmentDataSource,
getDataAccessor: () => this._dataAccessors,
getStartViewDate: () => this.getStartViewDate(),
getSortedAppointments: () => this._layoutManager.sortedItems,
isVirtualScrolling: () => this.isVirtualScrolling(),
scrollTo: this.scrollTo.bind(this)
};
this._appointments = this._createComponent("<div>", _appointments.Appointments, appointmentsConfig)
} else {
this._appointments = this._createComponent("<div>", _m_appointment_collection.default, this.appointmentsConfig());
this._appointments.option("itemTemplate", this.getAppointmentTemplate("appointmentTemplate"))
}
this.appointmentTooltip = new(this.option("adaptivityEnabled") ? _m_mobile_tooltip_strategy.MobileTooltipStrategy : _m_desktop_tooltip_strategy.DesktopTooltipStrategy)(this.getAppointmentTooltipOptions());
this.createAppointmentPopupForm();
if (this.isDataSourceLoaded() || this._isDataSourceLoading()) {
this.initMarkupCore();
this._dataSourceChangedHandler(this._dataSource.items());
this._fireContentReadyAction()
} else {
const groups = this.getViewOption("groups");
if (null !== groups && void 0 !== groups && groups.length) {
this.resourceManager.loadGroupResources(groups, true).then(() => this.initMarkupOnResourceLoaded())
} else {
this.initMarkupOnResourceLoaded()
}
}
}
createAppointmentPopupForm() {
var _this$appointmentPopu;
if (this.appointmentForm) {
this.appointmentForm.dispose()
}
this.appointmentForm = this.createAppointmentForm();
null === (_this$appointmentPopu = this.appointmentPopup) || void 0 === _this$appointmentPopu || _this$appointmentPopu.dispose();
this.appointmentPopup = this.createAppointmentPopup(this.appointmentForm)
}
renderMainContainer() {
this.mainContainer = (0, _renderer.default)("<div>").addClass("dx-scheduler-container");
this.$element().append(this.mainContainer)
}
createAppointmentForm() {
const scheduler = {
getResourceById: () => this.resourceManager.resourceById,
getDataAccessors: () => this._dataAccessors,
createComponent: (element, component, options) => this._createComponent(element, component, options),
getEditingConfig: () => this.editing,
getResourceManager: () => this.resourceManager,
getFirstDayOfWeek: () => this.option("firstDayOfWeek"),
getStartDayHour: () => this.option("startDayHour"),
getCalculatedEndDate: startDateWithStartHour => this._workSpace.calculateEndDate(startDateWithStartHour),
getTimeZoneCalculator: () => this.timeZoneCalculator
};
return new _m_form.AppointmentForm(scheduler)
}
createAppointmentPopup(form) {
const scheduler = {
getElement: () => this.$element(),
createComponent: (element, component, options) => this._createComponent(element, component, options),
focus: () => this.focus(),
getResourceManager: () => this.resourceManager,
getEditingConfig: () => this.editing,
getTimeZoneCalculator: () => this.timeZoneCalculator,
getDataAccessors: () => this._dataAccessors,
getAppointmentFormOpening: () => this.actions.onAppointmentFormOpening,
processActionResult: (arg, canceled) => this.processActionResult(arg, canceled),
addAppointment: appointment => this.addAppointment(appointment),
updateAppointment: (sourceAppointment, updatedAppointment) => this.updateAppointment(sourceAppointment, updatedAppointment)
};
return new _m_popup.AppointmentPopup(scheduler, form)
}
scrollToAppointment(appointment) {
const adapter = new _appointment_adapter.AppointmentAdapter(appointment, this._dataAccessors);
const {
startDate: startDate,
endDate: endDate,
allDay: allDay
} = adapter;
if (!startDate) {
return
}
const startTime = startDate.getTime();
const endTime = endDate ? endDate.getTime() : startTime;
const dayInMs = toMs("day");
const inAllDayRow = allDay || endTime - startTime >= dayInMs;
const appointmentGroupValues = (0, _appointment_groups_utils.getAppointmentGroupValues)(appointment, this.resourceManager.resources);
this._workSpace.updateScrollPosition(startDate, appointmentGroupValues, inAllDayRow)
}
getAppointmentTooltipOptions() {
return {
createComponent: this._createComponent.bind(this),
container: this.$element(),
getScrollableContainer: this.getWorkSpaceScrollableContainer.bind(this),
addDefaultTemplates: this._templateManager.addDefaultTemplates.bind(this._templateManager),
getAppointmentTemplate: this.getAppointmentTemplate.bind(this),
showAppointmentPopup: this.showAppointmentPopup.bind(this),
checkAndDeleteAppointment: this.checkAndDeleteAppointment.bind(this),
isAppointmentInAllDayPanel: this.isAppointmentInAllDayPanel.bind(this),
createFormattedDateText: (appointment, targetedAppointment, format) => this.fire("createFormattedDateText", appointment, targetedAppointment, format),
getAppointmentDisabled: appointment => this._dataAccessors.get("disabled", appointment),
onItemContextMenu: this._createActionByOption("onAppointmentContextMenu"),
createEventArgs: this._createEventArgs.bind(this)
}
}
_createEventArgs(e) {
const config = {
itemData: e.itemData.appointment,
itemElement: e.itemElement,
targetedAppointment: e.itemData.targetedAppointment
};
return (0, _extend.extend)({}, this.fire("mapAppointmentFields", config), {
component: e.component,
element: e.element,
event: e.event,
model: e.model
})
}
checkAndDeleteAppointment(appointment, targetedAppointment) {
const targetedAdapter = new _appointment_adapter.AppointmentAdapter(targetedAppointment, this._dataAccessors);
const deletingOptions = this.fireOnAppointmentDeleting(appointment, targetedAdapter);
this.checkRecurringAppointment(appointment, targetedAppointment, targetedAdapter.startDate, () => {
this.processDeleteAppointment(appointment, deletingOptions)
}, true)
}
getExtraAppointmentTooltipOptions() {
return {
rtlEnabled: this.option("rtlEnabled"),
focusStateEnabled: this.option("focusStateEnabled"),
editing: this.option("editing"),
offset: this.option("_appointmentTooltipOffset")
}
}
isAppointmentInAllDayPanel(appointmentData) {
const workSpace = this._workSpace;
const itTakesAllDay = this.appointmentTakesAllDay(appointmentData);
return itTakesAllDay && workSpace.supportAllDayRow() && workSpace.option("showAllDayPanel")
}
initMarkupCore() {
this.readyToRenderAppointments = (0, _window.hasWindow)();
this._workSpace && this.cleanWorkSpace();
this.renderWorkSpace();
if (this.option("_newAppointments")) {
this._appointments.option("$allDayContainer", this._workSpace.getAllDayContainer())
} else {
this._appointments.option({
fixedContainer: this._workSpace.getFixedContainer(),
allDayContainer: this._workSpace.getAllDayContainer()
})
}
this.waitAsyncTemplate(() => {
var _this$workSpaceRecalc2;
return null === (_this$workSpaceRecalc2 = this.workSpaceRecalculation) || void 0 === _this$workSpaceRecalc2 ? void 0 : _this$workSpaceRecalc2.resolve()
});
this.createAppointmentDataSource();
this.setRemoteFilterIfNeeded();
this.validateKeyFieldIfAgendaExist();
this.updateA11yStatus()
}
isDataSourceLoaded() {
var _this$_dataSource2;
return null === (_this$_dataSource2 = this._dataSource) || void 0 === _this$_dataSource2 ? void 0 : _this$_dataSource2.isLoaded()
}
_render() {
var _this$getWorkSpace;
null === (_this$getWorkSpace = this.getWorkSpace()) || void 0 === _this$getWorkSpace || _this$getWorkSpace.updateHeaderEmptyCellWidth();
super._render()
}
renderHeader() {
var _toolbarOptions$items;
const toolbarOptions = this.option("toolbar");
const isHeaderShown = Boolean(toolbarOptions.visible ?? (null === (_toolbarOptions$items = toolbarOptions.items) || void 0 === _toolbarOptions$items ? void 0 : _toolbarOptions$items.length));
if (isHeaderShown) {
const $header = (0, _renderer.default)("<div>").appendTo(this.mainContainer);
const headerOptions = this.headerConfig();
this.header = this._createComponent($header, _m_header.SchedulerHeader, headerOptions)
}
}
headerConfig() {
return {
currentView: this.currentView,
views: this.views,
currentDate: this.getViewOption("currentDate"),
min: this.getViewOption("min"),
max: this.getViewOption("max"),
indicatorTime: this.option("indicatorTime"),
startViewDate: this.getStartViewDate(),
tabIndex: this.option("tabIndex"),
focusStateEnabled: this.option("focusStateEnabled"),
useDropDownViewSwitcher: this.option("useDropDownViewSwitcher"),
firstDayOfWeek: this.getFirstDayOfWeek(),
toolbar: this.option("toolbar"),
customizeDateNavigatorText: this.option("customizeDateNavigatorText"),
onCurrentViewChange: name => {
this.option("currentView", name)
},
onCurrentDateChange: date => {
this.option("currentDate", date)
}
}
}
appointmentsConfig() {
const config = {
getResourceManager: () => this.resourceManager,
getAppointmentDataSource: () => this.appointmentDataSource,
getSortedAppointments: () => this._layoutManager.sortedItems,
scrollTo: this.scrollTo.bind(this),
dataAccessors: this._dataAccessors,
notifyScheduler: this.notifyScheduler,
onItemRendered: this.getAppointmentRenderedAction(),
onItemClick: this._createActionByOption("onAppointmentClick"),
onItemContextMenu: this._createActionByOption("onAppointmentContextMenu"),
onAppointmentDblClick: this._createActionByOption("onAppointmentDblClick"),
tabIndex: this.option("tabIndex"),
focusStateEnabled: this.option("focusStateEnabled"),
allowDrag: this.allowDragging(),
allowDelete: this.editing.allowUpdating && this.editing.allowDeleting,
allowResize: this.allowResizing(),
allowAllDayResize: this.allowAllDayResizing(),
rtlEnabled: this.option("rtlEnabled"),
groups: this.getViewOption("groups"),
groupByDate: this.getViewOption("groupByDate"),
timeZoneCalculator: this.timeZoneCalculator,
getResizableStep: () => this._workSpace ? this._workSpace.positionHelper.getResizableStep() : 0,
getDOMElementsMetaData: () => {
var _this$_workSpace;
return null === (_this$_workSpace = this._workSpace) || void 0 === _this$_workSpace ? void 0 : _this$_workSpace.getDOMElementsMetaData()
},
getViewDataProvider: () => {
var _this$_workSpace2;
return null === (_this$_workSpace2 = this._workSpace) || void 0 === _this$_workSpace2 ? void 0 : _this$_workSpace2.viewDataProvider
},
isVerticalGroupedWorkSpace: () => this._workSpace.isVerticalGroupedWorkSpace(),
isDateAndTimeView: () => (0, _index2.isDateAndTimeView)(this._wor