import { createElement, Dropdown, Popup, showAlert, createIcon, DateSelector, showConfirm, Grid, OptionBase } from "../ui"; import { nullOrEmpty } from "../utility"; import AssetSelector from "./assetSelector"; const iconWorkOrder = ''; /** * @private * @param {HTMLElement} element * @param {boolean?} display * @returns {HTMLElement} */ function displayElement(element, display) { element.style.display = display === false ? 'none' : ''; return element; } /** * @private * @param {HTMLInputElement} element * @param {string} pattern * @returns {HTMLInputElement} */ function patternValidation(element, pattern) { element.pattern = pattern; return element; } /** * @private * @param {"mile" | "mi" | "m" | "kilometre" | "km" | "kilometres"} uom * @returns {"Mile" | "Kilometre"} */ function getOdometerUnit(uom) { if (nullOrEmpty(uom)) { return 'Mile'; } switch (uom.toLowerCase()) { case 'mile': case 'mi': case 'm': return 'Mile'; case 'kilometre': case 'km': case 'kilometres': return 'Kilometre'; default: return 'Mile'; } } export default class AddWorkOrder extends OptionBase { _var = { /** * @private * @type {HTMLElement} */ container: null, /** * @private * @type {Popup} */ assetSelectorPopup: null, /** * @private * @property {number} Id * @property {string} VIN * @property {string} DisplayName * @property {boolean} OnRoad * @property {boolean} Hide * @property {number} [EngineHours] * @property {number} [Odometer] * @property {string} [OdometerUOM] */ asset: null, /** * @private * @property {string} Id * @property {string} Name * @property {string} Code */ customer: null, contacts: [], followers: [], params: { Locations: [], Departments: [], Advisors: [], OrderTypes: [], Statuses: [], Jobsites: [], Codes: [] }, el: { /** * @private * @type {HTMLTextAreaElement} */ textComplaint: null, /** * @private * @type {Dropdown} */ dropWorkOrderType: null, /** * @private * @type {Dropdown} */ dropStatus: null, /** * @private * @type {DateSelector} */ dateCompleted: null, /** * @private * @type {HTMLInputElement} */ inputHours: null, /** * @private * @type {HTMLInputElement} */ inputOdometer: null, /** * @private * @type {Dropdown} */ dropOdometerUnit: null, /** * @private * @type {Dropdown} */ dropAssignedTo: null, /** * @private * @type {Dropdown} */ dropAdvisor: null, /** * @private * @type {Dropdown} */ dropLocation: null, /** * @private * @type {Dropdown} */ dropDepartment: null } }; onAssetAdded; onAssetSelected; onSave; static IconWorkOrder = iconWorkOrder; constructor(opt = { allowCustomer: false, assetFullcontrol: false, /** * @private * @type {(assetId: number) => Promise<{Id: number, VIN: string, DisplayName: string, OnRoad: boolean, Hide: boolean, EngineHours?: number, Odometer?: number, OdometerUOM?: string}>} */ requestAssetInfo: null, /** * @private * @type {() => Promise<{Statuses: any[], Jobsites: any[], Codes: any[], Advisors: any[], Locations: any[], Departments: any[], OrderTypes: any[], AssetId: null}>} */ requestWorkOrderParams: null, /** * @private * @type {(hidden: boolean, search?: string, groups?: string[], jobsites?: number[], codes?: string[]) => Promise} */ requestAssets: null, /** * @private * @type {() => Promise} */ requestAddAsset: null, /** * @private * @type {() => Promise<{Customer: any, Contacts: any[], Followers: any[]}>} */ requestCustomers: null, /** * @private * @type {(assetId: number, locationId: number, departmentId: number) => Promise} */ requestAssignedTo: null, /** * @private * @type {(assetId: number) => Promise} */ requestHideAsset: null, /** * @private * @type {(assetId: number) => Promise} */ requestOpenedWorkorder: null, zIndex: 999 }) { super(opt); } async getItem() { const title = this.r('P_WO_OPENWORKORDER', 'Open Work Order'); const el = this._var.el; let status = el.dropStatus.selected; if (status != null) { status = { Status: status.Id, StatusType: status.StatusType, StatusName: status.Name, StatusAutoText: status.AutoText, StatusMessage: status.Message }; } else { status = { Status: -1 }; } let machine = this._var.asset; if (machine == null) { showAlert(title, this.r('P_WO_ASSETNOTEMPTY', 'Asset cannot be empty.')).then(() => this._var.container.querySelector('.wo-asset>svg')?.focus()); return null; } const item = { Id: -1, Description: el.textComplaint.value, WorkOrderType: el.dropWorkOrderType.selected?.Key ?? '', CompleteDate: el.dateCompleted.element.value, // MeterType: HourMeter: el.inputHours.value || -1, Odometer: el.inputOdometer.value || -1, OdometerUnits: el.dropOdometerUnit.selected?.value ?? 'Mile', AssignedTo: el.dropAssignedTo.selected?.IID ?? '', AdvisorId: el.dropAdvisor.selected?.Key ?? '', LocationId: el.dropLocation.selected?.ID || -1, DepartmentId: el.dropDepartment.selected?.Id || -1, AssetID: machine.Id, VIN: machine.VIN, AssetName: machine.DisplayName, WorkOrderTotalCost: -1, HoursToComplete: -1, OtherCost: -1, PartsCost: -1, TravelTimeCost: -1, LaborCost: -1, HourlyRate: -1, InspectionTemplateId: -1, ...status }; item.CustomerId = this._var.customer?.Id ?? -1; item.Contacts = this._var.contacts ?? []; item.Followers = this._var.followers ?? []; if (nullOrEmpty(el.textComplaint?.value)) { showAlert(title, this.r('P_WO_COMPLAINTREQUIRED', 'Complaint is required.')).then(() => el.textComplaint.focus()); return null; } if (el.dropStatus.selected?.Completed) { if (!el.dateCompleted.element.validity.valid) { showAlert(title, this.r('P_WO_COMPLETEDDATECANNOTBEEMPTY', 'Completed Date cannot be empty.')).then(() => el.dateCompleted.element.focus()); return null; } if (machine.OnRoad) { item.MeterType = 'Odometer'; if (nullOrEmpty(item.Odometer) || isNaN(item.Odometer) || item.Odometer < 0) { showAlert(title, this.r('P_WO_ODOMETERFORMATERROR', 'Odometer format error.')).then(() => el.inputOdometer.focus()); return null; } } else { item.MeterType = 'HourMeter'; if (nullOrEmpty(item.HourMeter) || isNaN(item.HourMeter) || item.HourMeter < 0) { showAlert(title, this.r('P_WO_HOURMETERFORMATERROR', 'Hour Meter format error.')).then(() => el.inputHours.focus()); return null; } } } return item; } async show() { const option = this._option; const allowCustomer = option.allowCustomer === true; const title = this.r('P_WO_OPENWORKORDER', 'Open Work Order'); const tabIndex = Math.max.apply(null, [...document.querySelectorAll('[tabindex]')].map(e => e.tabIndex ?? 0)) + 3; const textComplaint = createElement('textarea', textarea => { textarea.tabIndex = tabIndex + 2; textarea.className = 'ui-text wo-complaint'; textarea.placeholder = this.r('P_WO_ENTERCOMPLAINT', 'Enter Complaint'); }); const baseOption = { search: true, textKey: 'Value', valueKey: 'Key' }; const dropWorkOrderType = new Dropdown({ tabIndex: tabIndex + 3, input: true, ...baseOption }); const dropStatus = new Dropdown({ tabIndex: tabIndex + 5, htmlTemplate: it => createElement('span', 'wo-color-line', createElement('em', em => em.style.backgroundColor = it.Color), createElement('label', label => label.innerText = it.Name) ), search: true, textKey: 'Name', valueKey: 'Id' }); const dateCompleted = new DateSelector({ tabIndex: tabIndex + 6, className: 'wo-status-closed' }); const inputHours = createElement('input', input => { input.type = 'text'; input.tabIndex = tabIndex + 7; input.className = 'ui-input wo-hours input-hours'; }); const inputOdometer = createElement('input', input => { input.type = 'text'; input.tabIndex = tabIndex + 8; input.className = 'ui-input wo-odometer input-odometer'; }); const dropOdometerUnit = new Dropdown({ tabIndex: tabIndex + 9, selected: 'Mile' }); dropOdometerUnit.source = [ { value: 'Mile', text: GetTextByKey('P_WO_MILE', 'Mile') }, { value: 'Kilometre', text: GetTextByKey('P_WO_KILOMETER', 'Kilometer') } ] const dropAssignedTo = new Dropdown({ tabIndex: tabIndex + 10, selected: '', search: true, valueKey: 'IID', textKey: 'DisplayName' }); dropAssignedTo.source = [{ IID: '', DisplayName: '' }]; const dropAdvisor = new Dropdown({ tabIndex: tabIndex + 11, selected: '', ...baseOption }); const dropLocation = new Dropdown({ tabIndex: tabIndex + 12, selected: -1, search: true, valueKey: 'ID', textKey: 'Name' }); const dropDepartment = new Dropdown({ tabIndex: tabIndex + 13, selected: -1, search: true, valueKey: 'Id', textKey: 'Name' }); // save variables this._var.el = { textComplaint, dropWorkOrderType, dropStatus, dateCompleted, inputHours, inputOdometer, dropOdometerUnit, dropAssignedTo, dropAdvisor, dropLocation, dropDepartment } const container = createElement('div', 'open-wo-container', createElement('div', 'open-wo-header', createElement('img', img => img.src = iconWorkOrder), createElement('h3', h3 => h3.innerText = title) ), createElement('div', 'open-wo-content', createElement('div', 'wo-combined wo-asset', createElement('span', span => { span.className = 'wo-title wo-title-required'; span.innerText = this.r('P_WO_ASSET_COLON', 'Asset:'); }), createIcon('fa-light', 'search', svg => { svg.tabIndex = tabIndex + 1; svg.addEventListener('click', async () => { let popup = this._var.assetSelectorPopup; if (popup == null) { const selector = new AssetSelector({ assetFullcontrol: option.assetFullcontrol, loading: flag => this._var.assetSelectorPopup.loading = flag, close: () => this._var.assetSelectorPopup.close(), requestAssets: option.requestAssets, requestAddAsset: option.requestAddAsset, dataSource: this._var.params }); selector.onSelected = async it => { const el = this._var.el; if (typeof option.requestAssetInfo === 'function') { const a = await option.requestAssetInfo(it.Id); if (a != null) { it = a; if (it.OnRoad) { el.inputHours.value = ''; el.inputOdometer.value = it.Odometer; el.dropOdometerUnit.select(getOdometerUnit(it.OdometerUOM)); } else { el.inputHours.value = it.EngineHours; el.inputOdometer.value = ''; el.dropOdometerUnit.select('Mile'); } } } this._var.asset = it; if (typeof this.onAssetSelected === 'function') { this.onAssetSelected(it); } this._var.container.querySelector('.wo-asset-name').innerText = it.DisplayName; // get assigned to if (typeof option.requestAssignedTo === 'function') { const data = await option.requestAssignedTo(it.Id, el.dropLocation.selected?.ID ?? -1, el.dropDepartment.selected?.Id ?? -1) el.dropAssignedTo.source = [{ IID: '', DisplayName: '' }, ...data]; el.dropStatus.onSelected(el.dropStatus.selected); } }; popup = new Popup({ title: this.r('P_MA_SELECTASSET', 'Select Asset'), content: selector.create(), persistent: true, buttons: [ { key: 'ok', text: this.r('P_GRID_OK', 'OK'), trigger: () => selector.select() }, { text: this.r('P_WO_CANCEL', 'Cancel') } ] }); this._var.assetSelectorPopup = popup; popup.create(); popup.rect = { width: 1220 }; const mask = await popup.show(); mask.style.zIndex = option.zIndex ?? 999; await selector.init(); } else { await popup.show(); } }); }) ), createElement('span', 'wo-asset-name'), createElement('div', 'wo-line wo-combined', createElement('span', span => { span.className = 'wo-title wo-title-required'; span.innerText = this.r('P_WO_COMPLAINTCOLON', 'Complaint:'); }) ), createElement('div', div => { div.className = 'wo-line wo-combined wo-sub-line'; div.style.marginBottom = '2px'; div.style.width = 'calc(100% - 10px)'; }, textComplaint ), createElement('span', span => { span.className = 'wo-title'; span.innerText = this.r('P_WO_WORKORDERTYPE_COLON', 'Work Order Type:'); }), dropWorkOrderType.create(), createElement('div', div => { div.className = 'wo-combined wo-customer-record'; if (!allowCustomer) { displayElement(div, false); } }, createElement('span', span => { span.className = 'wo-title'; span.innerText = this.r('P_WO_COMPANYNAME_COLON', 'Company Name:'); }), createIcon('fa-light', 'search', svg => { svg.tabIndex = tabIndex + 4; svg.addEventListener('click', async () => { if (typeof option.requestCustomers === 'function') { popup.loading = true; const data = await option.requestCustomers(); popup.loading = false; if (data != null) { this._var.customer = data.Customer; this._var.contacts = data.Contacts; this._var.followers = data.Followers; let name = data.Customer.Name; if (!nullOrEmpty(data.Customer.Code)) { name += ' / ' + data.Customer.Code; } this._var.container.querySelector('.wo-company-name').innerText = name; } } }); }) ), displayElement(createElement('span', 'wo-company-name'), allowCustomer), createElement('span', span => { span.className = 'wo-title'; span.innerText = this.r('P_WO_STATUS_COLON', 'Status:'); }), dropStatus.create(), createElement('span', span => { span.className = 'wo-title wo-title-required wo-sub-line wo-status-closed'; span.innerText = this.r('P_WO_COMPLETEDDATE_COLON', 'Completed Date:'); displayElement(span, false); }), displayElement(dateCompleted.create(), false), createElement('span', span => { span.className = 'wo-title wo-title-required wo-sub-line wo-hours'; span.innerText = this.r('P_WO_HOURS_COLON', 'Hours:'); displayElement(span, false); }), displayElement( patternValidation(inputHours, '\\d+\\.?\\d*'), false ), createElement('span', span => { span.className = 'wo-title wo-title-required wo-sub-line wo-odometer'; span.innerText = this.r('P_WO_ODOMETER_COLON', 'Odometer:'); displayElement(span, false); }), createElement('div', div => { div.className = 'wo-combined wo-odometer'; displayElement(div, false); }, patternValidation(inputOdometer, '\\d+\\.?\\d*'), dropOdometerUnit.create() ), createElement('span', span => { span.className = 'wo-title'; span.innerText = this.r('P_WO_ASSIGNEDTO_COLON', 'Assigned Tech:'); }), dropAssignedTo.create(), createElement('span', span => { span.className = 'wo-title wo-customer-record'; span.innerText = this.r('P_WO_ADVISOR_COLON', 'Advisor:'); if (!allowCustomer) { displayElement(span, false); } }), displayElement(dropAdvisor.create(), allowCustomer), createElement('span', span => { span.className = 'wo-title wo-customer-record'; span.innerText = this.r('P_WO_LOCATION_COLON', 'Location:'); if (!allowCustomer) { displayElement(span, false); } }), displayElement(dropLocation.create(), allowCustomer), createElement('span', span => { span.className = 'wo-title wo-customer-record'; span.innerText = this.r('P_WO_DEPARTMENT_COLON', 'Department:'); if (!allowCustomer) { displayElement(span, false); } }), displayElement(dropDepartment.create(), allowCustomer) ) ); dropStatus.onSelected = item => { if (item == null) { return; } container.querySelectorAll('.wo-status-closed').forEach(it => it.style.display = item.Completed ? '' : 'none'); if (item.Completed) { const onRoad = this._var.asset?.OnRoad; container.querySelectorAll('.wo-hours').forEach(it => it.style.display = onRoad ? 'none' : ''); container.querySelectorAll('.wo-odometer').forEach(it => it.style.display = onRoad ? '' : 'none'); } else { container.querySelectorAll('.wo-hours').forEach(it => it.style.display = 'none'); container.querySelectorAll('.wo-odometer').forEach(it => it.style.display = 'none'); } }; dropLocation.onSelected = async loc => { // get assigned to const asset = this._var.asset; if (asset != null && typeof option.requestAssignedTo === 'function') { const data = await option.requestAssignedTo(asset.Id, loc?.ID ?? -1, this._var.el.dropDepartment.selected?.Id ?? -1) this._var.el.dropAssignedTo.source = [{ IID: '', DisplayName: '' }, ...data]; } }; dropDepartment.onSelected = async dep => { // get assigned to const asset = this._var.asset; if (asset != null && typeof option.requestAssignedTo === 'function') { const data = await option.requestAssignedTo(asset.Id, this._var.el.dropLocation.selected?.ID ?? -1, dep?.Id ?? -1) this._var.el.dropAssignedTo.source = [{ IID: '', DisplayName: '' }, ...data]; } } dateCompleted.value = new Date(); this._var.container = container; const popup = new Popup({ title, content: container, buttons: [ { key: 'open', text: title, trigger: async () => { popup.loading = true; try { const item = await this.getItem(); if (item == null) { return false; } if (this._var.asset.Hide) { if (!option.assetFullcontrol) { await showAlert(title, this.r('P_WO_HIDDENCANNOTCREATE', 'The selected asset is hidden and a work order cannot be created.') + '\n\n' + this.r('P_WO_CONTACTTOUNHIDE', 'Please contact your Fleet Manager to Unhide the asset if you require a work order.')); return false; } const next = await showConfirm(title, this.r('P_WO_HIDDENCANNOTCREATE', 'The selected asset is hidden and a work order cannot be created.') + '\n\n' + this.r('P_WO_PROMPTUNHIDE', 'Do you wish to "Un-Hide" the asset?'), [ { key: 'unhide', text: this.r('P_WO_UNHIDE', 'Unhide') }, { key: 'cancel', text: this.r('P_WO_CANCELWO', 'Cancel Work Order') } ]); if (next.result !== 'unhide') { return false; } // hide if (typeof option.requestHideAsset === 'function') { await option.requestHideAsset(this._var.asset.Id); this._var.asset.Hide = false; } } if (typeof option.requestOpenedWorkorder === 'function') { const wos = await option.requestOpenedWorkorder(this._var.asset.Id); if (wos == null) { return false; } if (wos.length > 0) { const next = await new Promise(resolve => { const popWorkorders = new Popup({ title, content: createElement('div', 'wo-opened-workorder', createElement('header', header => header.innerText = this.r('P_WO_ASSETOPENEDWORKORDER', 'The selected asset has the following open work orders:')), createElement('div', 'wo-grid-opened') ), resolve, buttons: [ { key: 'create', text: this.r('P_WO_CREATEWO', 'Create Work Order'), trigger: () => resolve('create') }, { key: 'cancel', text: this.r('P_WO_CANCELWO', 'Cancel Work Order'), trigger: () => resolve('cancel') } ] }); popWorkorders.show().then(mask => { const grid = new Grid(mask.querySelector('.wo-grid-opened'), this.r); grid.columns = [ { key: 'WorkOrderNumber', caption: 'WO #', width: 100 }, { key: 'CreateDateStr', caption: this.r('P_WO_CREATEDDATE', 'Created Date'), width: 120 }, { key: 'Description', caption: this.r('P_WO_COMPLAINT', 'Complaint'), width: 360 } ]; grid.init(); grid.source = wos.map(w => ({ WorkOrderNumber: w.WorkOrderNumber, CreateDateStr: { DisplayValue: w.CreateDateStr, Value: w.CreateDate }, Description: w.Description })); // default focus const button = mask.querySelector('.ui-popup-container .ui-popup-footer .ui-popup-button:last-child'); button?.focus(); }); }); if (next !== 'create') { return false; } } } if (typeof this.onSave === 'function') { this.onSave(item, this._var.el.dropStatus.selected); } } finally { popup.loading = false; } } }, { text: this.r('P_WO_CANCEL', 'Cancel') } ] }); popup.create(); popup.rect = { width: 600 }; const mask = await popup.show(); mask.style.zIndex = option.zIndex ?? 999; if (typeof option.requestWorkOrderParams === 'function') { popup.loading = true; const data = await option.requestWorkOrderParams() if (!isNaN(data.AssetId) && data.AssetId > 0 && typeof option.requestAssetInfo === 'function') { const it = await option.requestAssetInfo(data.AssetId); if (it != null) { if (it.OnRoad) { inputOdometer.value = it.Odometer; dropOdometerUnit.select(getOdometerUnit(it.OdometerUOM)); } else { inputHours.value = it.EngineHours; } this._var.asset = it; // if (typeof this.onAssetSelected === 'function') { // this.onAssetSelected(it); // } this._var.container.querySelector('.wo-asset-name').innerText = it.DisplayName; // get assigned to if (typeof option.requestAssignedTo === 'function') { const data = await option.requestAssignedTo(it.Id, dropLocation.selected?.ID ?? -1, dropDepartment.selected?.Id ?? -1) dropAssignedTo.source = [{ IID: '', DisplayName: '' }, ...data]; // dropStatus.onSelected(dropStatus.selected); } } } popup.loading = false; dropWorkOrderType.source = data.OrderTypes; dropAdvisor.source = [{ Key: '', Value: '' }, ...data.Advisors]; dropLocation.source = [{ ID: -1, Name: '' }, ...data.Locations]; dropDepartment.source = [{ Id: -1, Name: '' }, ...data.Departments]; this._var.params = data; dropStatus.source = data.Statuses; const defaultStatus = data.Statuses.find(s => s.DefaultOnOpen); if (defaultStatus != null) { dropStatus.select(defaultStatus.Id); } } textComplaint.focus(); return mask; } }