\r\n ${context.startClockHtml}\r\n ${context.endClockHtml}\r\n
${context.message}
\r\n
\r\n `\r\n };\r\n }\r\n\r\n static get pluginConfig() {\r\n return {\r\n chain: ['render']\r\n };\r\n } //endregion\r\n //region Init & destroy\r\n\r\n render() {\r\n var _me$resize, _me$resize$destroy;\r\n\r\n const me = this;\r\n (_me$resize = me.resize) === null || _me$resize === void 0 ? void 0 : (_me$resize$destroy = _me$resize.destroy) === null || _me$resize$destroy === void 0 ? void 0 : _me$resize$destroy.call(_me$resize);\r\n\r\n if (!me.disabled) {\r\n me.resize = me.createResizeHelper();\r\n }\r\n\r\n if (me.showTooltip) {\r\n me.clockTemplate = new ClockTemplate({\r\n scheduler: me.client\r\n });\r\n }\r\n }\r\n\r\n doDisable(disable) {\r\n if (!this.isConfiguring) {\r\n if (disable) {\r\n var _this$resize;\r\n\r\n (_this$resize = this.resize) === null || _this$resize === void 0 ? void 0 : _this$resize.destroy();\r\n this.resize = null;\r\n } else {\r\n this.resize = this.createResizeHelper();\r\n }\r\n }\r\n\r\n super.doDisable();\r\n }\r\n\r\n doDestroy() {\r\n var _me$tip, _me$tip$destroy, _me$clockTemplate, _me$resize2;\r\n\r\n const me = this;\r\n (_me$tip = me.tip) === null || _me$tip === void 0 ? void 0 : (_me$tip$destroy = _me$tip.destroy) === null || _me$tip$destroy === void 0 ? void 0 : _me$tip$destroy.call(_me$tip);\r\n (_me$clockTemplate = me.clockTemplate) === null || _me$clockTemplate === void 0 ? void 0 : _me$clockTemplate.destroy();\r\n (_me$resize2 = me.resize) === null || _me$resize2 === void 0 ? void 0 : _me$resize2.destroy();\r\n super.doDestroy();\r\n } //endregion\r\n //region Events\r\n\r\n isElementResizable(element, event) {\r\n const {\r\n client,\r\n resize\r\n } = this,\r\n timespanRecord = client.resolveTimeSpanRecord(element);\r\n\r\n if (client.readOnly) {\r\n return false;\r\n }\r\n\r\n let resizable = timespanRecord && timespanRecord.isResizable; // go up from \"handle\" to resizable element\r\n\r\n element = DomHelper.up(event.target, client.eventSelector); // Not resizable if the mousedown is on a resizing handle of\r\n // a percent bar.\r\n\r\n const handleHoldingElement = element ? element.firstElementChild : element,\r\n handleEl = event.target.closest('[class$=\"-handle\"]');\r\n\r\n if (!resizable || handleEl && handleEl !== handleHoldingElement) {\r\n return false;\r\n }\r\n\r\n const startsOutside = element.classList.contains('b-sch-event-startsoutside'),\r\n endsOutside = element.classList.contains('b-sch-event-endsoutside');\r\n\r\n if (resizable === true) {\r\n if (startsOutside && endsOutside) {\r\n return false;\r\n } else if (startsOutside) {\r\n resizable = 'end';\r\n } else if (endsOutside) {\r\n resizable = 'start';\r\n } else {\r\n return resize.overStartHandle(event, element) || resize.overEndHandle(event, element);\r\n }\r\n }\r\n\r\n if (startsOutside && resizable === 'start' || endsOutside && resizable === 'end') {\r\n return false;\r\n }\r\n\r\n if (resize.overStartHandle(event, element) && resizable === 'start' || resize.overEndHandle(event, element) && resizable === 'end') {\r\n return true;\r\n }\r\n\r\n return false;\r\n }\r\n\r\n onBeforeResizeStart({\r\n element,\r\n event\r\n }) {\r\n const {\r\n client\r\n } = this,\r\n name = client.scheduledEventName,\r\n timespanRecord = client.resolveTimeSpanRecord(element);\r\n\r\n if (this.disabled) {\r\n return false;\r\n } // trigger beforeEventResize or beforeTaskResize depending on product\r\n\r\n return client.trigger(`before${client.capitalizedEventName}Resize`, _objectSpread2({\r\n [`${name}Record`]: timespanRecord,\r\n event\r\n }, this.getBeforeResizeParams({\r\n element\r\n }))) !== false;\r\n }\r\n\r\n onResizeStart({\r\n context,\r\n event\r\n }) {\r\n var _client$resolveAssign;\r\n\r\n const me = this,\r\n {\r\n client\r\n } = me,\r\n {\r\n element\r\n } = context,\r\n timespanRecord = client.resolveTimeSpanRecord(element),\r\n name = client.scheduledEventName;\r\n client.element.classList.add('b-resizing-event'); // Let products to their specific stuff\r\n\r\n me.setupProductResizeContext(context, event);\r\n\r\n if (me.showTooltip) {\r\n if (me.tip instanceof Tooltip) {\r\n me.tip.rootElement = me.rootElement;\r\n me.tip.align = tipAlign[context.edge];\r\n me.tip.updateContentOnMouseMove = true; // Tip needs to be shown first for getTooltipTarget to be able to measure anchor size\r\n\r\n me.tip.show();\r\n me.tip.showBy(me.getTooltipTarget());\r\n } else {\r\n me.tip = Tooltip.new({\r\n id: `${client.id}-event-resize-tip`,\r\n autoShow: true,\r\n axisLock: true,\r\n trackMouse: false,\r\n updateContentOnMouseMove: true,\r\n getHtml: me.getTipHtml.bind(me),\r\n align: tipAlign[context.edge],\r\n rootElement: me.rootElement,\r\n hideDelay: 0,\r\n cls: 'b-resize-tooltip'\r\n }, me.tip);\r\n me.tip.on('innerhtmlupdate', me.updateDateIndicator, me);\r\n }\r\n } // flag to not allow release of element when scrolling\r\n\r\n timespanRecord.instanceMeta(client).retainElement = true; // Trigger eventResizeStart or taskResizeStart depending on product\r\n\r\n client.trigger(`${name}ResizeStart`, _objectSpread2({\r\n [`${name}Record`]: timespanRecord,\r\n event\r\n }, this.getResizeStartParams(context))); // Scheduler renders assignments, Gantt renders Tasks\r\n\r\n context.resizedRecord = ((_client$resolveAssign = client.resolveAssignmentRecord) === null || _client$resolveAssign === void 0 ? void 0 : _client$resolveAssign.call(client, context.element)) || timespanRecord;\r\n context.timespanRecord = timespanRecord;\r\n }\r\n\r\n updateDateIndicator() {\r\n const {\r\n edge,\r\n startDate,\r\n endDate\r\n } = this.resize.context,\r\n {\r\n element\r\n } = this.tip;\r\n\r\n if (startDate || endDate) {\r\n if (edge === 'right' || edge === 'bottom') {\r\n const el = element.querySelector('.b-sch-tooltip-enddate');\r\n\r\n if (el) {\r\n this.clockTemplate.updateDateIndicator(el, endDate);\r\n }\r\n } else {\r\n this.clockTemplate.updateDateIndicator(element, startDate);\r\n }\r\n }\r\n }\r\n\r\n getTooltipTarget() {\r\n const me = this,\r\n target = Rectangle.from(me.resize.context.element, null, true);\r\n\r\n if (me.resize.direction === 'horizontal') {\r\n // Align to the dragged edge of the proxy, and then bump right so that the anchor aligns perfectly.\r\n if (me.resize.context.edge === 'right') {\r\n target.x = target.right - 1;\r\n }\r\n\r\n target.width = me.tip.anchorSize[0] / 2;\r\n } else {\r\n // Align to the dragged edge of the proxy, and then bump bottom so that the anchor aligns perfectly.\r\n if (me.resize.context.edge === 'bottom') {\r\n target.y = target.bottom - 1;\r\n }\r\n\r\n target.height = me.tip.anchorSize[1] / 2;\r\n }\r\n\r\n return {\r\n target\r\n };\r\n }\r\n\r\n onResizing({\r\n context,\r\n event\r\n }) {\r\n const me = this,\r\n {\r\n client\r\n } = me,\r\n depFeature = client.features.dependencies,\r\n timespanRecord = me.getTimespanRecord(context),\r\n name = client.scheduledEventName,\r\n {\r\n element,\r\n edge\r\n } = context,\r\n xy = DomHelper.getTranslateXY(element);\r\n let start, end;\r\n\r\n if (edge === 'top' || edge === 'left') {\r\n end = timespanRecord.endDate;\r\n\r\n if (client.snapRelativeToEventStartDate) {\r\n start = client.getDateFromXY(xy, null, true);\r\n start = client.timeAxis.roundDate(start, timespanRecord.startDate);\r\n } else {\r\n start = client.getDateFromXY(xy, 'round', true);\r\n }\r\n } // bottom || right\r\n else {\r\n xy[0] += element.offsetWidth;\r\n xy[1] += element.offsetHeight;\r\n start = timespanRecord && timespanRecord.startDate;\r\n\r\n if (client.snapRelativeToEventStartDate) {\r\n end = client.getDateFromXY(xy, null, true);\r\n end = client.timeAxis.roundDate(end, timespanRecord.endDate);\r\n } else {\r\n end = client.getDateFromXY(xy, 'round', true);\r\n }\r\n }\r\n\r\n start = start || context.startDate;\r\n end = end || context.endDate;\r\n\r\n if (context.dateConstraints) {\r\n start = DateHelper.constrain(start, context.dateConstraints.start, context.dateConstraints.end);\r\n end = DateHelper.constrain(end, context.dateConstraints.start, context.dateConstraints.end);\r\n }\r\n\r\n if (me.showExactResizePosition || client.timeAxisViewModel.snap) {\r\n const exactSize = edge === 'top' || edge === 'left' ? client.timeAxisViewModel.getDistanceBetweenDates(start, timespanRecord.endDate) : client.timeAxisViewModel.getDistanceBetweenDates(timespanRecord.startDate, end);\r\n\r\n switch (edge) {\r\n case 'top':\r\n DomHelper.setTranslateY(element, context.elementStartY + context.elementWidth - exactSize);\r\n element.style.height = exactSize + 'px';\r\n break;\r\n\r\n case 'right':\r\n element.style.width = exactSize + 'px';\r\n break;\r\n\r\n case 'bottom':\r\n element.style.height = exactSize + 'px';\r\n break;\r\n\r\n case 'left':\r\n DomHelper.setTranslateX(element, context.elementStartX + context.elementWidth - exactSize);\r\n element.style.width = exactSize + 'px';\r\n break;\r\n }\r\n }\r\n\r\n const dateChanged = context.endDate - end !== 0 || context.startDate - start !== 0;\r\n context.endDate = end;\r\n context.startDate = start; // No need to query on every pixel of mouse move\r\n\r\n if (dateChanged) {\r\n context.valid = me.checkValidity(context, event);\r\n } // Trigger eventPartialResize or taskPartialResize depending on product\r\n\r\n client.trigger(`${name}PartialResize`, {\r\n [`${name}Record`]: timespanRecord,\r\n startDate: start,\r\n endDate: end,\r\n element,\r\n context\r\n });\r\n\r\n if (depFeature && !depFeature.isDisabled) {\r\n depFeature.updateDependenciesForTimeSpan(context.resizedRecord, element);\r\n }\r\n\r\n if (me.showTooltip) {\r\n me.tip.alignTo(me.getTooltipTarget());\r\n }\r\n }\r\n\r\n checkValidity(context, event) {\r\n let valid = context.startDate && context.endDate > context.startDate && this.validatorFn.call(this.validatorFnThisObj || this, context, event);\r\n\r\n if (valid && typeof valid !== 'boolean') {\r\n context.message = valid.message || '';\r\n valid = valid.valid;\r\n }\r\n\r\n return valid !== false;\r\n }\r\n\r\n onFinishResize({\r\n source: drag,\r\n context,\r\n event\r\n }) {\r\n const me = this,\r\n timespanRecord = me.getTimespanRecord(context),\r\n oldStart = timespanRecord.startDate,\r\n oldEnd = timespanRecord.endDate,\r\n start = context.startDate || oldStart,\r\n end = context.endDate || oldEnd,\r\n client = me.client;\r\n let modified = false;\r\n\r\n if (me.tip) {\r\n me.tip.hide();\r\n } // allow release of element again\r\n\r\n timespanRecord.instanceMeta(client).retainElement = false;\r\n context.oldFinalize = context.finalize;\r\n\r\n context.finalize = (...params) => {\r\n // We are overriding context of the resize helper. It is finalized automatically on sync resize. Which means,\r\n // we should only call finalize if context is async.\r\n context.async && me.finalize(...params);\r\n };\r\n\r\n context.valid = start && end && end - start > 0 && ( // Input sanity check\r\n start - oldStart !== 0 || end - oldEnd !== 0) && // Make sure start OR end changed\r\n context.valid !== false;\r\n\r\n if (context.valid) {\r\n // Seems to be a valid resize operation, ask outside world if anyone wants to take control over the finalizing,\r\n // to show a confirm dialog prior to applying the new values. Triggers beforeEventResizeFinalize or\r\n // beforeTaskResizeFinalize depending on product\r\n client.trigger(`before${client.scheduledEventName}ResizeFinalize`, {\r\n context,\r\n event\r\n });\r\n modified = true;\r\n }\r\n\r\n if (!context.async) {\r\n me.finalize(modified);\r\n }\r\n }\r\n\r\n onCancelResize({\r\n context\r\n }) {\r\n const timespanRecord = this.getTimespanRecord(context); // resizing may not have started at all (just clicking a resize handle)\r\n\r\n if (timespanRecord) {\r\n timespanRecord.instanceMeta(this.client).retainElement = false;\r\n }\r\n\r\n this.finalize(false);\r\n }\r\n\r\n async finalize(updateRecord) {\r\n const me = this,\r\n {\r\n client\r\n } = me,\r\n {\r\n context\r\n } = me.resize,\r\n timespanRecord = me.getTimespanRecord(context),\r\n name = client.scheduledEventName;\r\n let wasChanged = false;\r\n\r\n if (me.tip) {\r\n me.tip.hide();\r\n }\r\n\r\n if (context.started) {\r\n if (updateRecord) {\r\n // Scheduler and gantt updates the record differently\r\n wasChanged = await me.internalUpdateRecord(context, timespanRecord);\r\n }\r\n\r\n if (!updateRecord || !wasChanged) {\r\n var _dependencies$schedul;\r\n\r\n const dependencies = client.features.dependencies;\r\n me.resize.abortResize(null, true); // Dependencies are updated dynamically during resize, so ensure they are redrawn\r\n // if the event snaps back with no change.\r\n\r\n dependencies === null || dependencies === void 0 ? void 0 : (_dependencies$schedul = dependencies.scheduleDraw) === null || _dependencies$schedul === void 0 ? void 0 : _dependencies$schedul.call(dependencies, true);\r\n } else {\r\n context.oldFinalize();\r\n }\r\n }\r\n\r\n client.element.classList.remove('b-resizing-event'); // Triggers eventResizeEnd or taskResizeEnd depending on product\r\n\r\n client.trigger(`${name}ResizeEnd`, _objectSpread2({\r\n changed: wasChanged,\r\n [`${name}Record`]: timespanRecord\r\n }, me.getResizeEndParams(context)));\r\n } //endregion\r\n //region Tooltip\r\n\r\n getTipHtml({\r\n tip\r\n }) {\r\n const me = this;\r\n let {\r\n startDate,\r\n endDate,\r\n valid,\r\n message,\r\n timespanRecord\r\n } = me.resize.context; // Empty string hides the tip - we get called before the Resizer, so first call will be empty\r\n\r\n if (!startDate || !endDate) {\r\n return tip.html;\r\n }\r\n\r\n if (message === undefined) message = '';\r\n endDate = me.client.getDisplayEndDate(endDate, startDate);\r\n const startText = me.client.getFormattedDate(startDate),\r\n endText = me.client.getFormattedDate(endDate);\r\n return me.tooltipTemplate({\r\n record: timespanRecord,\r\n valid,\r\n startDate,\r\n endDate,\r\n startText,\r\n endText,\r\n message,\r\n startClockHtml: me.clockTemplate.template({\r\n date: startDate,\r\n text: startText,\r\n cls: 'b-sch-tooltip-startdate'\r\n }),\r\n endClockHtml: me.clockTemplate.template({\r\n date: endDate,\r\n text: endText,\r\n cls: 'b-sch-tooltip-enddate'\r\n })\r\n });\r\n } //endregion\r\n //region Product specific, implemented in subclasses\r\n\r\n getBeforeResizeParams(context) {\r\n // Can be overridden in subclass\r\n return {};\r\n }\r\n\r\n getResizeStartParams(context) {\r\n // Can be overridden in subclass\r\n return {};\r\n }\r\n\r\n getResizeEndParams(context) {\r\n // Can be overridden in subclass\r\n return {};\r\n }\r\n\r\n getRowRecord(context) {\r\n throw new Error('Implement in subclass');\r\n }\r\n\r\n getTimespanRecord(context) {\r\n throw new Error('Implement in subclass');\r\n }\r\n\r\n setupProductResizeContext(context, event) {\r\n throw new Error('Implement in subclass');\r\n } // Store containing the timespan record being resized\r\n\r\n get store() {\r\n throw new Error('Implement in subclass');\r\n } //endregion\r\n\r\n}\r\nResizeBase._$name = 'ResizeBase';\r\n\r\n/**\r\n * @module Scheduler/feature/base/ResourceTimeRangesBase\r\n */\r\n\r\n/**\r\n * Abstract base class for ResourceTimeRanges and ResourceNonWorkingTime features.\r\n * You should not use this class directly.\r\n *\r\n * @extends Core/mixin/InstancePlugin\r\n * @abstract\r\n */\r\n\r\nclass ResourceTimeRangesBase extends InstancePlugin.mixin(AttachToProjectMixin) {\r\n //region Config\r\n // Plugin configuration. This plugin chains some of the functions in Grid.\r\n static get pluginConfig() {\r\n return {\r\n chain: ['getEventsToRender', 'onEventDataGenerated', 'noFeatureElementsInAxis']\r\n };\r\n } // Let Scheduler know if we have ResourceTimeRanges in view or not\r\n\r\n noFeatureElementsInAxis() {\r\n const {\r\n timeAxis\r\n } = this.scheduler;\r\n return this.store && !this.store.storage.values.some(t => timeAxis.isTimeSpanInAxis(t));\r\n } //endregion\r\n //region Init\r\n\r\n construct(scheduler, config) {\r\n const me = this;\r\n me.scheduler = scheduler;\r\n super.construct(scheduler, config);\r\n }\r\n\r\n doDisable(disable) {\r\n if (this.client.isPainted) {\r\n this.client.refresh();\r\n }\r\n\r\n super.doDisable(disable);\r\n } //endregion\r\n\r\n getEventsToRender(resource, events) {\r\n throw new Error('Implement in subclass');\r\n } // Called for each event during render, allows manipulation of render data. Adjust any resource time ranges\r\n // (chained function from Scheduler)\r\n\r\n onEventDataGenerated(renderData) {\r\n const me = this,\r\n {\r\n eventRecord,\r\n iconCls\r\n } = renderData;\r\n\r\n if (me.shouldInclude(eventRecord)) {\r\n if (me.scheduler.isVertical) {\r\n renderData.width = me.scheduler.resourceColumnWidth;\r\n } else {\r\n renderData.top = 0;\r\n } // Flag that we should fill entire row/col\r\n\r\n renderData.fillSize = true; // Add our own cls\r\n\r\n renderData.wrapperCls[me.rangeCls] = 1;\r\n renderData.wrapperCls[`b-sch-color-${eventRecord.timeRangeColor}`] = eventRecord.timeRangeColor; // Add label\r\n\r\n renderData.children.push(eventRecord.name); // Add icon\r\n\r\n if (iconCls) {\r\n renderData.children.unshift({\r\n tag: 'i',\r\n className: iconCls\r\n });\r\n } // Event data for DOMSync comparison\r\n\r\n renderData.eventId = me.generateElementId(eventRecord);\r\n }\r\n }\r\n /**\r\n * Generates ID from the passed time range record\r\n * @param {Scheduler.model.TimeSpan} record\r\n * @returns {String} Generated ID for the DOM element\r\n * @internal\r\n */\r\n\r\n generateElementId(record) {\r\n return `${this.idPrefix}-${record.id}`;\r\n }\r\n\r\n shouldInclude(eventRecord) {\r\n throw new Error('Implement in subclass');\r\n } // Called when a ResourceTimeRangeModel is manipulated, relays to Scheduler#onInternalEventStoreChange which updates to UI\r\n\r\n onStoreChange(event) {\r\n this.scheduler.onInternalEventStoreChange(event);\r\n }\r\n\r\n} // No feature based styling needed, do not add a cls to Scheduler\r\n\r\nResourceTimeRangesBase.featureClass = '';\r\nResourceTimeRangesBase._$name = 'ResourceTimeRangesBase';\r\n\r\n/**\r\n * @module Scheduler/feature/base/TimeSpanMenuBase\r\n */\r\n\r\n/**\r\n * Abstract base class used by other context menu features which show the context menu for TimeAxis.\r\n * Using this class you can make sure the menu expects the target to disappear,\r\n * since it can be scroll out of the scheduling zone.\r\n *\r\n * Features that extend this class are:\r\n * * {@link Scheduler/feature/EventMenu EventMenu};\r\n * * {@link Scheduler/feature/ScheduleMenu ScheduleMenu};\r\n * * {@link Scheduler/feature/TimeAxisHeaderMenu TimeAxisHeaderMenu};\r\n *\r\n * @extends Grid/feature/base/ContextMenuBase\r\n * @abstract\r\n */\r\n\r\nclass TimeSpanMenuBase extends ContextMenuBase {\r\n changeMenu(menu, oldMenu) {\r\n if (menu) {\r\n menu = Objects.assign({}, {\r\n // It's a rectangle outside which the target of alignment may disappear.\r\n // It has to be like that because when scrolling in a scheduler, there are two scrolling elements.\r\n // The main Scheduler body for up/down and the subgrid for left and right\r\n clippedBy: [this.client.timeAxisSubGridElement, this.client.bodyContainer]\r\n }, menu);\r\n }\r\n\r\n return super.changeMenu(menu, oldMenu);\r\n }\r\n\r\n}\r\nTimeSpanMenuBase._$name = 'TimeSpanMenuBase';\r\n\r\n/**\r\n * @module Scheduler/feature/base/TimeSpanRecordContextMenuBase\r\n */\r\n// This is a version of what Containers do, except that we have to apply our namedItems\r\n// all the way down any configured menu hierarchy, and the resulting structure must\r\n// be available *before* menu instantiation for the processItems method to interrogate.\r\n\r\nfunction applyNamedItems(items, namedItems) {\r\n for (const ref in items) {\r\n let item = items[ref];\r\n\r\n if (item) {\r\n if (ref in namedItems) {\r\n item = items[ref] = typeof item === 'object' ? ObjectHelper.merge(ObjectHelper.clone(namedItems[ref]), item) : namedItems[ref];\r\n } // Our namedItems must apply all the way down any descendant menus.\r\n // Extract menu here because it may have been applied by a namedItem.\r\n\r\n const menu = item.menu;\r\n\r\n if (menu) {\r\n applyNamedItems('items' in menu ? menu.items : menu, namedItems);\r\n }\r\n }\r\n }\r\n}\r\n/**\r\n * DEPRECATED: This class is deprecated since v4.0.0.\r\n * It is **not** used by our features anymore and remains here only for backward compatibility.\r\n * If you already extended this class please re-implement it utilizing our new\r\n * {@link Scheduler.feature.base.TimeSpanMenuBase } abstract base class.\r\n *\r\n * @extends Core/mixin/InstancePlugin\r\n * @abstract\r\n * @deprecated 4.0.0\r\n */\r\n\r\nclass TimeSpanRecordContextMenuBase extends InstancePlugin {\r\n //region Config\r\n static get defaultConfig() {\r\n return {\r\n /**\r\n * This is a preconfigured set of {@link Core.widget.Container#config-namedItems} used to create the default\r\n * context menu.\r\n * @config {Object}\r\n */\r\n defaultItems: null,\r\n\r\n /**\r\n * An {@link Core.widget.Menu Menu} items object containing named child menu items\r\n * to apply to the feature's provided context menu, see {@link #config-defaultItems}.\r\n *\r\n * This may add extra items as below, but may also remove any of the {@link #config-defaultItems}\r\n * by configuring the name of the item as `false`\r\n *\r\n * ```javascript\r\n * features : {\r\n * taskMenu : { // use eventMenu in the Scheduler product\r\n * // This object is applied to the Feature's predefined defaultItems object\r\n * items : {\r\n * switchToDog : {\r\n * text : 'Dog',\r\n * icon : 'b-fa b-fa-fw b-fa-dog',\r\n * onItem({contextRecord}) {\r\n * contextRecord.dog = true;\r\n * contextRecord.cat = false;\r\n * },\r\n * weight : 500 // Make this second from end\r\n * },\r\n * switchToCat : {\r\n * text : 'Cat',\r\n * icon : 'b-fa b-fa-fw b-fa-cat',\r\n * onItem({contextRecord}) {\r\n * contextRecord.dog = false;\r\n * contextRecord.cat = true;\r\n * },\r\n * weight : 510 // Make this sink to end\r\n * },\r\n * add : false // We do not want the \"Add\" submenu to be available\r\n * }\r\n * }\r\n * }\r\n * ```\r\n *\r\n * @config {Object|Object[]}\r\n */\r\n items: null,\r\n\r\n /**\r\n * A function called before displaying the menu that allows manipulations of its items. Called with a\r\n * single parameter with format { contextRecord, eventElement, items }. Returning `false`\r\n * from this function prevents the menu from being shown.\r\n *\r\n * ```javascript\r\n * features : {\r\n * taskContextMenu : {\r\n * processItems({contextRecord, items}) {\r\n * // Add or remove items here as needed\r\n * if (contextRecord.type === 'Meeting') {\r\n * items.cancel = {\r\n * text : 'Cancel',\r\n * icon : 'b-fa b-fa-fw b-fa-ban',\r\n * weight : 200 // Move to end\r\n * };\r\n * }\r\n *\r\n * // Hide delete for parents\r\n * items.deleteTask.hidden = contextRecord.isParent;\r\n * }\r\n * }\r\n * }\r\n * ```\r\n *\r\n * @config {Function}\r\n */\r\n processItems: null,\r\n\r\n /**\r\n * Event which is used to show context menu.\r\n * Available options are: 'contextmenu', 'click', 'dblclick'.\r\n * Default value is used from {@link Grid/view/GridBase#config-contextMenuTriggerEvent}\r\n * @config {String}\r\n */\r\n triggerEvent: null\r\n };\r\n } // Plugin configuration. This plugin chains some of the functions in Grid.\r\n\r\n static get pluginConfig() {\r\n return {\r\n assign: ['showEventContextMenu'],\r\n chain: ['onElementContextMenu', 'onElementClick', 'onElementDblClick', 'onEventSpaceKey']\r\n };\r\n } //endregion\r\n //region Init\r\n\r\n construct(grid, config) {\r\n super.construct(grid, config);\r\n VersionHelper.deprecate('Scheduler', '5.0.0', '`TimeSpanRecordContextMenuBase` feature is deprecated, in favor of `TimeSpanMenuBase` feature. Please see https://bryntum.com/docs/scheduler/#Scheduler/guides/upgrades/3.1.0.md for more information.');\r\n }\r\n\r\n doDestroy() {\r\n if (this.menu) {\r\n this.menu.destroy();\r\n }\r\n\r\n super.doDestroy();\r\n } //endregion\r\n //region Events\r\n\r\n onElementContextMenu(event) {\r\n this.triggerEvent === 'contextmenu' && this.showEventContextMenu(event);\r\n }\r\n\r\n onElementClick(event) {\r\n this.triggerEvent === 'click' && this.showEventContextMenu(event);\r\n }\r\n\r\n onElementDblClick(event) {\r\n this.triggerEvent === 'dblclick' && this.showEventContextMenu(event);\r\n } // chained from EventNavigation\r\n\r\n onEventSpaceKey(keyEvent) {\r\n const targetPoint = Rectangle.from(keyEvent.target).center,\r\n contextmenuEvent = new MouseEvent('contextmenu', Object.assign({\r\n clientX: targetPoint.x,\r\n clientY: targetPoint.y\r\n }, keyEvent));\r\n Object.defineProperty(contextmenuEvent, 'target', {\r\n get() {\r\n return keyEvent.target;\r\n }\r\n\r\n });\r\n this.showEventContextMenu(contextmenuEvent);\r\n } //endregion\r\n //region Menu handlers\r\n\r\n /**\r\n * Show event context menu.\r\n * @param event\r\n * @internal\r\n * @on-owner\r\n */\r\n\r\n showEventContextMenu(event) {\r\n const me = this,\r\n client = me.client,\r\n target = event.target,\r\n eventElement = DomHelper.up(target, client.eventSelector),\r\n targetElement = eventElement || target;\r\n\r\n if (targetElement) {\r\n const record = me.resolveRecord(targetElement);\r\n\r\n if (record) {\r\n event.preventDefault();\r\n me.showContextMenuFor(record, {\r\n targetElement,\r\n eventElement,\r\n event\r\n });\r\n }\r\n }\r\n }\r\n /**\r\n * A method to retrieve record from its DOM element. Must be implemented in subclasses.\r\n * @abstract\r\n */\r\n\r\n resolveRecord() {}\r\n /**\r\n * Shows context menu for the provided record. If record is not rendered (outside of time span, or collapsed)\r\n * menu won't appear.\r\n * @param {Scheduler.model.TimeSpan} record\r\n * @param {Object} [options]\r\n * @param {HTMLElement} options.targetElement Element to align context menu to\r\n * @param {HTMLElement} options.eventElement Event element if target is an event, otherwise `null`\r\n * @param {Event} options.event Browser event. If provided menu will be aligned according to clientX/clientY coordinates.\r\n * If omitted, context menu will be centered to targetElement\r\n */\r\n\r\n showContextMenuFor(record, options) {} // Implement in subclasses to massage options or veto show.\r\n\r\n beforeContextMenuShow() {} //endregion\r\n //region Show/Hide\r\n\r\n /**\r\n * @param {Object} eventParams\r\n * @param {Object[]} items\r\n * @internal\r\n */\r\n\r\n showContextMenu(eventParams) {\r\n if (this.disabled) {\r\n return;\r\n }\r\n\r\n const me = this,\r\n event = eventParams.event,\r\n menuType = eventParams.menuType.toLowerCase(),\r\n eventType = StringHelper.uncapitalize(menuType),\r\n clientGetItemsMethod = `get${StringHelper.capitalize(menuType)}MenuItems`,\r\n {\r\n client,\r\n processItems,\r\n defaultItems,\r\n namedItems\r\n } = me,\r\n point = event ? [event.clientX + 1, event.clientY + 1] : Rectangle.from(eventParams.targetElement).center,\r\n items = eventParams.items = ObjectHelper.isEmpty(me.items) ? ObjectHelper.clone(defaultItems) : ObjectHelper.merge(ObjectHelper.clone(defaultItems), me.items);\r\n eventParams.namedItems = namedItems;\r\n eventParams.selection = client.selectedRecords; // Apply the named items prior to Container's item processing.\r\n // Our namedItems must cascade to all descendant Menu levels.\r\n // And they MUST have all been converted prior to the processItems call.\r\n\r\n applyNamedItems(items, namedItems); // Call the chainable method which other features use to add their own menu items.\r\n // For example getEventMenuItems\r\n\r\n if (client[clientGetItemsMethod]) {\r\n client[clientGetItemsMethod](eventParams, items);\r\n } // Allow user a chance at processing the items and preventing the menu from showing\r\n\r\n if ((!processItems || processItems(eventParams) !== false) && !ObjectHelper.isEmpty(eventParams.items)) {\r\n // beforeContextMenuShow is a lifecycle method which may be implemented in subclasses to\r\n // preprocess the event.\r\n if (me.beforeContextMenuShow(eventParams) !== false) {\r\n // Trigger event that allows preventing menu or manipulating its items\r\n if (client.trigger(`${eventType}ContextMenuBeforeShow`, eventParams) !== false) {\r\n me.menu = WidgetHelper.showContextMenu(point, {\r\n owner: client,\r\n scrollAction: 'hide',\r\n clippedBy: [client.timeAxisSubGridElement, client.bodyContainer],\r\n constrainTo: window,\r\n items,\r\n\r\n onDestroy() {\r\n me.menu = null;\r\n },\r\n\r\n // Load up the item event with the contextual info\r\n onBeforeItem: itemEvent => {\r\n Object.assign(itemEvent, eventParams);\r\n },\r\n onItem: itemEvent => client.trigger(`${eventType}ContextMenuItem`, itemEvent),\r\n listeners: {\r\n show({\r\n source: menu\r\n }) {\r\n eventParams.menu = menu;\r\n client.trigger(`${eventType}ContextMenuShow`, eventParams);\r\n }\r\n\r\n }\r\n });\r\n }\r\n }\r\n }\r\n }\r\n /**\r\n * Hides the context menu\r\n * @internal\r\n */\r\n\r\n hideContextMenu(animate) {\r\n this.menu && this.menu.hide(animate);\r\n } //endregion\r\n //region Getters/Setters\r\n\r\n /**\r\n * Provides the default configuration of the context menu.\r\n *\r\n * Concrete classes must all provide their own defaultItems value in their defaultConfig blocks\r\n * @private\r\n */\r\n\r\n set defaultItems(defaultItems) {\r\n this._defaultItems = defaultItems;\r\n }\r\n\r\n get defaultItems() {\r\n const result = ObjectHelper.clone(this._defaultItems); // Read-only client should have no default items enabled\r\n\r\n if (this.client.readOnly) {\r\n for (const item in result) {\r\n result[item] = false;\r\n }\r\n }\r\n\r\n return result;\r\n }\r\n\r\n get triggerEvent() {\r\n return this._triggerEvent || this.client.contextMenuTriggerEvent;\r\n }\r\n\r\n set triggerEvent(value) {\r\n this._triggerEvent = value;\r\n } //endregion\r\n\r\n}\r\nTimeSpanRecordContextMenuBase._$name = 'TimeSpanRecordContextMenuBase';\r\n\r\n/**\r\n * @module Scheduler/feature/base/TooltipBase\r\n */\r\n\r\n/**\r\n * Base class for `EventTooltip` (Scheduler) and `TaskTooltip` (Gantt) features. Contains shared code. Not to be used directly.\r\n *\r\n * @extends Core/mixin/InstancePlugin\r\n * @extendsconfigs Core/widget/Tooltip\r\n */\r\n\r\nclass TooltipBase extends InstancePlugin {\r\n //region Config\r\n static get defaultConfig() {\r\n return {\r\n /**\r\n * Specify true to have tooltip updated when mouse moves, if you for example want to display date at mouse\r\n * position.\r\n * @config {Boolean}\r\n * @default\r\n * @category Misc\r\n */\r\n autoUpdate: false,\r\n\r\n /**\r\n * The amount of time to hover before showing\r\n * @config {Number}\r\n * @default\r\n */\r\n hoverDelay: 250,\r\n\r\n /**\r\n * The time (in milliseconds) for which the Tooltip remains visible when the mouse leaves the target.\r\n *\r\n * May be configured as `false` to persist visible after the mouse exits the target element. Configure it\r\n * as 0 to always retrigger `hoverDelay` even when moving mouse inside `fromElement`\r\n * @config {Number}\r\n * @default\r\n */\r\n hideDelay: 100,\r\n template: null,\r\n cls: null,\r\n align: {\r\n align: 'b-t'\r\n },\r\n clockTemplate: null,\r\n // Set to true to update tooltip contents if record changes while tip is open\r\n monitorRecordUpdate: null,\r\n testConfig: {\r\n hoverDelay: 0\r\n }\r\n };\r\n } // Plugin configuration. This plugin chains some of the functions in Grid.\r\n\r\n static get pluginConfig() {\r\n return {\r\n chain: ['onPaint']\r\n };\r\n } //endregion\r\n //region Init\r\n\r\n construct(client, config) {\r\n const me = this; // process initial config into an actual config object\r\n\r\n config = me.processConfig(config);\r\n super.construct(client, config); // Default triggering selector is the client's inner element selector\r\n\r\n if (!me.forSelector) {\r\n me.forSelector = `${client.eventInnerSelector}:not(.b-dragproxy)`;\r\n }\r\n\r\n me.clockTemplate = new ClockTemplate({\r\n scheduler: client\r\n });\r\n client.on({\r\n [`before${client.scheduledEventName}drag`]: () => {\r\n var _me$tooltip;\r\n\r\n return (_me$tooltip = me.tooltip) === null || _me$tooltip === void 0 ? void 0 : _me$tooltip.hide();\r\n }\r\n });\r\n } // TooltipBase feature handles special config cases, where user can supply a function to use as template\r\n // instead of a normal config object\r\n\r\n processConfig(config) {\r\n if (typeof config === 'function') {\r\n return {\r\n template: config\r\n };\r\n }\r\n\r\n return config;\r\n } // override setConfig to process config before applying it (used mainly from ReactScheduler)\r\n\r\n setConfig(config) {\r\n super.setConfig(this.processConfig(config));\r\n }\r\n\r\n doDestroy() {\r\n var _this$clockTemplate, _this$tooltip;\r\n\r\n (_this$clockTemplate = this.clockTemplate) === null || _this$clockTemplate === void 0 ? void 0 : _this$clockTemplate.destroy();\r\n (_this$tooltip = this.tooltip) === null || _this$tooltip === void 0 ? void 0 : _this$tooltip.destroy();\r\n super.doDestroy();\r\n }\r\n\r\n doDisable(disable) {\r\n if (this.tooltip) {\r\n this.tooltip.disabled = disable;\r\n }\r\n\r\n super.doDisable(disable);\r\n } //endregion\r\n\r\n onPaint({\r\n firstPaint\r\n }) {\r\n if (firstPaint) {\r\n var _me$tooltip2;\r\n\r\n const me = this,\r\n {\r\n client\r\n } = me,\r\n ignoreSelector = ['.b-dragselecting', '.b-eventeditor-editing', '.b-taskeditor-editing', '.b-resizing-event', '.b-dragcreating', `.b-dragging-${client.scheduledEventName}`, '.b-creating-dependency', '.b-dragproxy'].map(cls => `:not(${cls})`).join('');\r\n (_me$tooltip2 = me.tooltip) === null || _me$tooltip2 === void 0 ? void 0 : _me$tooltip2.destroy();\r\n /**\r\n * A reference to the tooltip instance, which will have a special `eventRecord` property that\r\n * you can use to get data from the contextual event record to which this tooltip is related.\r\n * @member {Core.widget.Tooltip} tooltip\r\n * @readonly\r\n * @category Misc\r\n */\r\n\r\n me.tooltip = new Tooltip(_objectSpread2(_objectSpread2({\r\n axisLock: 'flexible',\r\n id: me.tipId || `${me.client.id}-event-tip`,\r\n cls: me.tipCls,\r\n forSelector: `.b-timelinebase${ignoreSelector} .b-grid-body-container:not(.b-scrolling) ${me.forSelector}`,\r\n scrollAction: 'realign',\r\n clippedBy: [client.timeAxisSubGridElement, client.bodyContainer],\r\n forElement: client.timeAxisSubGridElement,\r\n showOnHover: true,\r\n anchorToTarget: true,\r\n allowOver: Boolean(me.config.items || me.config.tools),\r\n getHtml: me.getTipHtml.bind(me),\r\n disabled: me.disabled\r\n }, me.config), {}, {\r\n listeners: me.configuredListeners\r\n }));\r\n me.tooltip.on({\r\n innerhtmlupdate: 'updateDateIndicator',\r\n overtarget: 'onOverNewTarget',\r\n show: 'onTipShow',\r\n hide: 'onTipHide',\r\n thisObj: me\r\n });\r\n }\r\n } // leave configuredListeners alone until render time at which they are used on the tooltip\r\n\r\n processConfiguredListeners() {}\r\n\r\n updateDateIndicator() {\r\n const me = this,\r\n tip = me.tooltip,\r\n endDateElement = tip.element.querySelector('.b-sch-tooltip-enddate');\r\n\r\n if (!me.record) {\r\n return;\r\n }\r\n\r\n me.clockTemplate.updateDateIndicator(tip.element, me.record.startDate);\r\n endDateElement && me.clockTemplate.updateDateIndicator(endDateElement, me.record.endDate);\r\n }\r\n\r\n resolveTimeSpanRecord(forElement) {\r\n return this.client.resolveTimeSpanRecord(forElement);\r\n }\r\n\r\n getTipHtml({\r\n tip,\r\n activeTarget\r\n }) {\r\n const me = this,\r\n {\r\n client\r\n } = me,\r\n recordProp = me.recordType || `${client.scheduledEventName}Record`,\r\n timeSpanRecord = me.resolveTimeSpanRecord(activeTarget); // If user has mouseovered a fading away element of a deleted event,\r\n // an event record will not be found. In this case the tip must hide.\r\n // Instance of check is to not display while propagating\r\n\r\n if ((timeSpanRecord === null || timeSpanRecord === void 0 ? void 0 : timeSpanRecord.startDate) instanceof Date) {\r\n const {\r\n startDate,\r\n endDate\r\n } = timeSpanRecord,\r\n startText = client.getFormattedDate(startDate),\r\n endDateValue = client.getDisplayEndDate(endDate, startDate),\r\n endText = client.getFormattedDate(endDateValue);\r\n tip.eventRecord = timeSpanRecord;\r\n return me.template({\r\n tip,\r\n // eventRecord for Scheduler, taskRecord for Gantt\r\n [`${recordProp}`]: timeSpanRecord,\r\n startDate,\r\n endDate,\r\n startText,\r\n endText,\r\n startClockHtml: me.clockTemplate.template({\r\n date: startDate,\r\n text: startText,\r\n cls: 'b-sch-tooltip-startdate'\r\n }),\r\n endClockHtml: timeSpanRecord.isMilestone ? '' : me.clockTemplate.template({\r\n date: endDateValue,\r\n text: endText,\r\n cls: 'b-sch-tooltip-enddate'\r\n })\r\n });\r\n } else {\r\n tip.hide();\r\n return '';\r\n }\r\n }\r\n\r\n get record() {\r\n return this.tooltip.eventRecord;\r\n }\r\n\r\n onTipShow() {\r\n const me = this;\r\n\r\n if (me.monitorRecordUpdate && !me.updateListener) {\r\n me.updateListener = me.client.eventStore.on({\r\n update: me.onRecordUpdate,\r\n thisObj: me\r\n });\r\n }\r\n }\r\n\r\n onTipHide() {\r\n var _this$updateListener;\r\n\r\n // To not retain full project when changing project\r\n this.tooltip.eventRecord = null;\r\n (_this$updateListener = this.updateListener) === null || _this$updateListener === void 0 ? void 0 : _this$updateListener.call(this);\r\n this.updateListener = null;\r\n }\r\n\r\n onOverNewTarget({\r\n newTarget\r\n }) {\r\n this.tooltip.eventRecord = this.resolveTimeSpanRecord(newTarget);\r\n }\r\n\r\n onRecordUpdate({\r\n record\r\n }) {\r\n // make sure the record we are showing the tip for is still relevant\r\n if (record === this.record) {\r\n // Stop aligning at this point\r\n this.tooltip.alignTo();\r\n this.tooltip.updateContent();\r\n }\r\n }\r\n\r\n}\r\nTooltipBase._$name = 'TooltipBase';\r\n\r\n/**\r\n * @module Scheduler/feature/AbstractTimeRanges\r\n */\r\n\r\n/**\r\n * Abstract base class, you should not use this class directly.\r\n * @abstract\r\n * @mixes Core/mixin/Delayable\r\n * @extends Core/mixin/InstancePlugin\r\n */\r\n\r\nclass AbstractTimeRanges extends InstancePlugin.mixin(Delayable) {\r\n //region Config\r\n static get defaultConfig() {\r\n return {\r\n // CSS class to apply to range elements\r\n rangeCls: 'b-sch-range',\r\n // CSS class to apply to line elements (0-duration time range)\r\n lineCls: 'b-sch-line',\r\n\r\n /**\r\n * Store that holds timeRanges (using the {@link Scheduler.model.TimeSpan} model or subclass thereof).\r\n * A store will be automatically created if none is specified\r\n * @config {Object|Core.data.Store}\r\n */\r\n store: {\r\n modelClass: TimeSpan\r\n },\r\n\r\n /**\r\n * Set to `true` to enable dragging and resizing of range elements in the header. Only relevant when {@link #config-showHeaderElements} is true.\r\n * @config {Boolean}\r\n * @defaultValue\r\n */\r\n enableResizing: false,\r\n\r\n /**\r\n * A Boolean specifying whether or not to show tooltip while resizing range elements, or a\r\n * {@link Core.widget.Tooltip} config object which is applied to the tooltip\r\n * @config {Boolean|Object}\r\n * @default\r\n */\r\n showTooltip: true,\r\n\r\n /**\r\n * `true` to render range elements into the time axis header\r\n * @config {Boolean}\r\n * @default\r\n */\r\n showHeaderElements: true,\r\n dragTipTemplate: data => `\r\n \r\n * ${resourceRecord.firstName} {resourceRecord.surname}\r\n *
\r\n * `\r\n * }\r\n * });\r\n * ```\r\n *\r\n * NOTE: When using `headerRenderer` no default internal markup is applied to the resource header cell,\r\n * `iconCls` and {@link Scheduler.model.mixin.ResourceModelMixin#field-imageUrl} or {@link Scheduler.model.mixin.ResourceModelMixin#field-image}\r\n * will have no effect unless you supply custom markup for them.\r\n *\r\n * @config {Function}\r\n * @param {Object} params Object containing the params below\r\n * @param {Scheduler.model.ResourceModel} resourceRecord Resource whose header is being rendered\r\n * @param {Object} elementConfig A {@link Core.helper.DomHelper#function-createElement-static} config object used to create the element for the resource\r\n */\r\n headerRenderer: null,\r\n\r\n /**\r\n * Automatically resize resource columns to **fill** available width. Set to `false` to always respect the\r\n * configured `columnWidth`\r\n * @config {Boolean}\r\n * @default\r\n */\r\n fillWidth: true,\r\n\r\n /**\r\n * Automatically resize resource columns to always **fit** available width\r\n * @config {Boolean}\r\n * @default\r\n */\r\n fitWidth: false,\r\n // Copied from Scheduler#resourceImagePath on creation in TimeAxisColumn.js\r\n imagePath: null,\r\n // Copied from Scheduler#resourceImageExtension on creation in TimeAxisColumn.js\r\n imageExtension: null,\r\n // Copied from Scheduler#defaultResourceImageName on creation in TimeAxisColumn.js\r\n defaultImageName: null\r\n };\r\n }\r\n\r\n static get configurable() {\r\n return {\r\n /**\r\n * Set to `false` to render just the resource name, `true` to render an avatar (or initials if no image exists)\r\n * @config {Boolean}\r\n * @default true\r\n */\r\n showAvatars: {\r\n value: true,\r\n $config: 'nullify'\r\n }\r\n };\r\n }\r\n\r\n static get properties() {\r\n return {\r\n /**\r\n * An index of the first visible resource in vertical mode\r\n * @property {Number}\r\n * @readonly\r\n * @private\r\n */\r\n firstResource: -1,\r\n\r\n /**\r\n * An index of the last visible resource in vertical mode\r\n * @property {Number}\r\n * @readonly\r\n * @private\r\n */\r\n lastResource: -1\r\n };\r\n } //endregion\r\n //region Init\r\n\r\n construct(config) {\r\n super.construct(config);\r\n const me = this;\r\n\r\n if (me.imagePath != null) {\r\n // Need to increase height a bit when displaying images\r\n me.element.classList.add('b-has-images');\r\n }\r\n\r\n EventHelper.on({\r\n element: me.element,\r\n delegate: '.b-resourceheader-cell',\r\n capture: true,\r\n click: 'onResourceMouseEvent',\r\n dblclick: 'onResourceMouseEvent',\r\n contextmenu: 'onResourceMouseEvent',\r\n thisObj: me\r\n });\r\n }\r\n\r\n changeShowAvatars(show) {\r\n var _this$avatarRendering;\r\n\r\n (_this$avatarRendering = this.avatarRendering) === null || _this$avatarRendering === void 0 ? void 0 : _this$avatarRendering.destroy();\r\n\r\n if (show) {\r\n this.avatarRendering = new AvatarRendering({\r\n element: this.element\r\n });\r\n }\r\n\r\n return show;\r\n }\r\n\r\n updateShowAvatars() {\r\n if (!this.isConfiguring) {\r\n this.refresh();\r\n }\r\n } //endregion\r\n //region ResourceStore\r\n\r\n set resourceStore(store) {\r\n const me = this;\r\n\r\n if (store !== me._resourceStore) {\r\n var _me$resourceStoreDeta;\r\n\r\n (_me$resourceStoreDeta = me.resourceStoreDetacher) === null || _me$resourceStoreDeta === void 0 ? void 0 : _me$resourceStoreDeta.call(me);\r\n me._resourceStore = store;\r\n me.resourceStoreDetacher = store.on({\r\n changePreCommit: 'onResourceStoreDataChange',\r\n thisObj: me\r\n }); // Already have data? Update width etc\r\n\r\n if (store.count) {\r\n me.onResourceStoreDataChange({});\r\n }\r\n }\r\n }\r\n\r\n get resourceStore() {\r\n return this._resourceStore;\r\n } // Redraw resource headers on any data change\r\n\r\n onResourceStoreDataChange({\r\n action\r\n }) {\r\n const me = this,\r\n width = me.resourceStore.count * me.columnWidth;\r\n\r\n if (width !== me.width) {\r\n me.element.style.width = width + 'px'; // During setup, silently set the width. It will then render correctly. After setup, let the world know...\r\n\r\n me.column.set('width', width, me.column.grid.isConfiguring);\r\n }\r\n\r\n if (action === 'removeall') {\r\n // Keep nothing\r\n me.element.innerHTML = '';\r\n }\r\n\r\n if (action === 'remove' || action === 'add' || action === 'filter') {\r\n me.refreshWidths();\r\n }\r\n } //endregion\r\n //region Properties\r\n\r\n get columnWidth() {\r\n return this._columnWidth;\r\n }\r\n\r\n set columnWidth(width) {\r\n const me = this;\r\n\r\n if (width !== me._columnWidth) {\r\n const oldWidth = me._columnWidth;\r\n me._columnWidth = width; // Flag set in refreshWidths, do not want to create a loop\r\n\r\n if (!me.refreshingWidths) {\r\n me._originalColumnWidth = width;\r\n me.refreshWidths();\r\n }\r\n\r\n if (!me.isConfiguring) {\r\n me.refresh(); // Cannot trigger with requested width, might have changed because of fit/fill\r\n\r\n me.trigger('columnWidthChange', {\r\n width: me._columnWidth,\r\n oldWidth\r\n });\r\n }\r\n }\r\n }\r\n /**\r\n * Assign to toggle resource columns **fill* mode. `true` means they will stretch (grow) to fill viewport, `false`\r\n * that they will respect their configured `columnWidth`.\r\n * @property {Boolean}\r\n */\r\n\r\n get fillWidth() {\r\n return this._fillWidth;\r\n }\r\n\r\n set fillWidth(fill) {\r\n this._fillWidth = fill;\r\n this.refreshWidths();\r\n }\r\n /**\r\n * Assign to toggle resource columns **fit* mode. `true` means they will grow or shrink to always fit viewport,\r\n * `false` that they will respect their configured `columnWidth`.\r\n * @property {Boolean}\r\n */\r\n\r\n get fitWidth() {\r\n return this._fitWidth;\r\n }\r\n\r\n set fitWidth(fit) {\r\n this._fitWidth = fit;\r\n this.refreshWidths();\r\n }\r\n\r\n getImageURL(imageName) {\r\n return StringHelper.joinPaths([this.imagePath || '', imageName || '']);\r\n }\r\n\r\n get imagePath() {\r\n return this._imagePath;\r\n }\r\n\r\n set imagePath(path) {\r\n this._imagePath = path;\r\n this.refresh();\r\n } //endregion\r\n //region Fit to width\r\n\r\n get availableWidth() {\r\n return this._availableWidth;\r\n }\r\n\r\n set availableWidth(width) {\r\n this._availableWidth = width;\r\n this.refreshWidths();\r\n } // Updates the column widths according to fill and fit settings\r\n\r\n refreshWidths() {\r\n const me = this,\r\n {\r\n availableWidth,\r\n _originalColumnWidth\r\n } = me,\r\n count = me.resourceStore && me.resourceStore.count; // Bail out if availableWidth not yet set or resource store not assigned/loaded\r\n\r\n if (!availableWidth || !count) {\r\n return;\r\n }\r\n\r\n me.refreshingWidths = true;\r\n const // Fit width if configured to do so or if configured to fill and used width is less than available width\r\n fit = me.fitWidth || me.fillWidth && _originalColumnWidth * count < availableWidth,\r\n useWidth = fit ? Math.floor(availableWidth / count) : _originalColumnWidth,\r\n shouldAnimate = me.column.grid.enableEventAnimations && Math.abs(me._columnWidth - useWidth) > 30;\r\n DomHelper.addTemporaryClass(me.element, 'b-animating', shouldAnimate ? 300 : 0, me);\r\n me.columnWidth = useWidth;\r\n me.refreshingWidths = false;\r\n } //endregion\r\n //region Rendering\r\n // Visual resource range, set by VerticalRendering + its buffer\r\n\r\n set visibleResources({\r\n firstResource,\r\n lastResource\r\n }) {\r\n this.firstResource = firstResource;\r\n this.lastResource = lastResource;\r\n this.refresh();\r\n }\r\n /**\r\n * Refreshes the visible headers\r\n */\r\n\r\n refresh() {\r\n const me = this,\r\n {\r\n firstResource,\r\n lastResource\r\n } = me,\r\n configs = [];\r\n\r\n if (!me.column.grid.isConfiguring && firstResource > -1 & lastResource > -1 && lastResource < me.resourceStore.count) {\r\n // Gather element configs for resource headers in view\r\n for (let i = firstResource; i <= lastResource; i++) {\r\n const resourceRecord = me.resourceStore.getAt(i),\r\n elementConfig = {\r\n // Might look like overkill to use DomClassList here, but can be used in headerRenderer\r\n className: new DomClassList({\r\n 'b-resourceheader-cell': 1\r\n }),\r\n dataset: {\r\n resourceId: resourceRecord.id\r\n },\r\n style: {\r\n left: i * me.columnWidth,\r\n width: me.columnWidth\r\n },\r\n children: []\r\n }; // Let a configured headerRenderer have a go at it before applying\r\n\r\n if (me.headerRenderer) {\r\n const value = me.headerRenderer({\r\n elementConfig,\r\n resourceRecord\r\n });\r\n\r\n if (value != null) {\r\n elementConfig.html = value;\r\n }\r\n } // No headerRenderer, apply default markup\r\n else {\r\n let imageUrl;\r\n\r\n if (resourceRecord.imageUrl) {\r\n imageUrl = resourceRecord.imageUrl;\r\n } else {\r\n if (me.imagePath != null) {\r\n if (resourceRecord.image !== false) {\r\n const imageName = resourceRecord.image || resourceRecord.name && resourceRecord.name.toLowerCase() + me.imageExtension;\r\n imageUrl = me.getImageURL(imageName);\r\n }\r\n }\r\n } // By default showing resource name and optionally avatar\r\n\r\n elementConfig.children.push(me.showAvatars && me.avatarRendering.getResourceAvatar({\r\n initials: resourceRecord.initials,\r\n color: resourceRecord.eventColor,\r\n iconCls: resourceRecord.iconCls,\r\n defaultImageUrl: me.defaultImageName && me.getImageURL(me.defaultImageName),\r\n imageUrl\r\n }), {\r\n tag: 'span',\r\n className: 'b-resource-name',\r\n html: StringHelper.encodeHtml(resourceRecord.name)\r\n });\r\n }\r\n\r\n configs.push(elementConfig);\r\n }\r\n } // Sync changes to the header\r\n\r\n DomSync.sync({\r\n domConfig: {\r\n onlyChildren: true,\r\n children: configs\r\n },\r\n targetElement: me.element,\r\n syncIdField: 'resourceId' // TODO: Add callback here to trigger events when rendering/derendering header cells. Sooner or later\r\n // someone is going to ask for a way to render JSX or what not to the header\r\n\r\n });\r\n } //endregion\r\n\r\n onResourceMouseEvent(event) {\r\n const resourceCell = event.target.closest('.b-resourceheader-cell'),\r\n resourceRecord = this.resourceStore.getById(resourceCell.dataset.resourceId);\r\n this.trigger('resourceHeader' + StringHelper.capitalize(event.type), {\r\n resourceRecord,\r\n event\r\n });\r\n }\r\n\r\n}\r\nResourceHeader._$name = 'ResourceHeader';\r\n\r\n/**\r\n * @module Scheduler/column/TimeAxisColumn\r\n */\r\n\r\n/**\r\n * A column containing the timeline \"viewport\", in which events, dependencies etc are drawn.\r\n * Normally you do not need to interact with or create this column, it is handled by Scheduler.\r\n *\r\n * If you wish to output custom contents inside the time axis row cells, you can provide your custom column configuration\r\n * using the {@link #config-renderer} like so:\r\n *\r\n * ```javascript\r\n * const scheduler = new Scheduler({\r\n * appendTo : document.body\r\n * columns : [\r\n * { text : 'Name', field : 'name', width : 130 },\r\n * {\r\n * type : 'timeAxis',\r\n * renderer({ record, cellElement }) {\r\n * return '