MapComplete/UI/OpeningHours/OpeningHoursRange.ts

103 lines
3.3 KiB
TypeScript
Raw Normal View History

/**
* A single opening hours range, shown on top of the OH-picker table
*/
import {UIEventSource} from "../../Logic/UIEventSource";
import {UIElement} from "../UIElement";
import {VariableUiElement} from "../Base/VariableUIElement";
import Svg from "../../Svg";
import {Utils} from "../../Utils";
import Combine from "../Base/Combine";
import {OH, OpeningHour} from "./OpeningHours";
2021-06-10 14:05:26 +02:00
import OpeningHoursPickerTable from "./OpeningHoursPickerTable";
import BaseUIElement from "../BaseUIElement";
2020-10-04 01:04:46 +02:00
export default class OpeningHoursRange extends UIElement {
private _oh: UIEventSource<OpeningHour>;
2021-06-10 14:05:26 +02:00
private readonly _startTime: BaseUIElement;
private readonly _endTime: BaseUIElement;
private readonly _deleteRange: BaseUIElement;
private readonly _tableId: OpeningHoursPickerTable;
2020-10-04 01:04:46 +02:00
2021-06-10 14:05:26 +02:00
constructor(oh: UIEventSource<OpeningHour>, tableId: OpeningHoursPickerTable) {
2020-10-04 01:04:46 +02:00
super(oh);
2020-10-08 19:03:00 +02:00
this._tableId = tableId;
2020-10-04 01:04:46 +02:00
const self = this;
this._oh = oh;
this.SetClass("oh-timerange");
2020-10-08 19:03:00 +02:00
oh.addCallbackAndRun(() => {
2020-10-04 01:04:46 +02:00
const el = document.getElementById(this.id) as HTMLElement;
self.InnerUpdate(el);
})
2020-11-06 03:17:27 +01:00
this._deleteRange =
Svg.delete_icon_ui()
2020-10-04 01:04:46 +02:00
.SetClass("oh-delete-range")
.onClick(() => {
oh.data.weekday = undefined;
oh.ping();
});
2020-10-08 19:03:00 +02:00
this._startTime = new VariableUiElement(oh.map(oh => {
return Utils.TwoDigits(oh.startHour) + ":" + Utils.TwoDigits(oh.startMinutes);
})).SetClass("oh-timerange-label")
2020-10-04 01:04:46 +02:00
2020-10-08 19:03:00 +02:00
this._endTime = new VariableUiElement(oh.map(oh => {
return Utils.TwoDigits(oh.endHour) + ":" + Utils.TwoDigits(oh.endMinutes);
})).SetClass("oh-timerange-label")
2020-10-04 01:04:46 +02:00
}
2020-10-04 01:04:46 +02:00
2021-06-10 14:05:26 +02:00
InnerRender(): BaseUIElement {
2020-10-04 01:04:46 +02:00
const oh = this._oh.data;
if (oh === undefined) {
2021-06-10 01:36:20 +02:00
return undefined;
2020-10-04 01:04:46 +02:00
}
const height = this.getHeight();
2020-10-08 19:03:00 +02:00
let content = [this._deleteRange]
if (height > 2) {
content = [this._startTime, this._deleteRange, this._endTime];
}
return new Combine(content)
.SetClass("oh-timerange-inner")
}
2020-10-04 01:04:46 +02:00
private getHeight(): number {
const oh = this._oh.data;
let endhour = oh.endHour;
if (oh.endHour == 0 && oh.endMinutes == 0) {
endhour = 24;
}
2021-06-10 14:05:26 +02:00
return (endhour - oh.startHour + ((oh.endMinutes - oh.startMinutes) / 60));
2020-10-04 01:04:46 +02:00
}
protected InnerUpdate(el: HTMLElement) {
if (el == null) {
return;
}
const oh = this._oh.data;
if (oh === undefined) {
return;
}
2020-10-08 19:03:00 +02:00
// The header cell containing monday, tuesday, ...
2021-06-10 14:05:26 +02:00
const table = this._tableId.ConstructElement() as HTMLTableElement;
2020-10-08 19:03:00 +02:00
const bodyRect = document.body.getBoundingClientRect();
const rangeStart = table.rows[1].cells[1].getBoundingClientRect().top - bodyRect.top;
const rangeEnd = table.rows[table.rows.length - 1].cells[1].getBoundingClientRect().bottom - bodyRect.top;
const pixelsPerHour = (rangeEnd - rangeStart) / 24;
el.style.top = (pixelsPerHour * OH.startTime(oh)) + "px";
el.style.height = (pixelsPerHour * (OH.endTime(oh) - OH.startTime(oh))) + "px";
2020-10-04 01:04:46 +02:00
}
}