diff --git a/api/courses.go b/api/courses.go
index 7502d87f5..463660e3e 100644
--- a/api/courses.go
+++ b/api/courses.go
@@ -70,6 +70,7 @@ func configGinCourseRouter(router *gin.Engine, daoWrapper dao.DaoWrapper) {
courses.POST("/deleteLectures", routes.deleteLectures)
courses.POST("/renameLecture/:streamID", routes.renameLecture)
courses.POST("/updateLectureSeries/:streamID", routes.updateLectureSeries)
+ courses.POST("/updateStartEnd/:streamID", routes.updateStartEnd)
courses.PUT("/updateDescription/:streamID", routes.updateDescription)
courses.DELETE("/deleteLectureSeries/:streamID", routes.deleteLectureSeries)
courses.POST("/submitCut", routes.submitCut)
@@ -991,6 +992,43 @@ func (r coursesRoutes) updateDescription(c *gin.Context) {
}
}
+type changeDateTime struct {
+ Start time.Time `json:"start" binding:"required"`
+ End time.Time `json:"end" binding:"required"`
+}
+
+func (r coursesRoutes) updateStartEnd(c *gin.Context) {
+ var req changeDateTime
+ if err := c.Bind(&req); err != nil {
+ _ = c.Error(tools.RequestError{
+ Status: http.StatusBadRequest,
+ CustomMessage: "invalid body",
+ Err: err,
+ })
+ return
+ }
+
+ stream, err := r.StreamsDao.GetStreamByID(context.Background(), c.Param("streamID"))
+ if err != nil {
+ _ = c.Error(tools.RequestError{
+ Status: http.StatusNotFound,
+ CustomMessage: "can not find stream",
+ Err: err,
+ })
+ return
+ }
+ stream.Start = req.Start
+ stream.End = req.End
+ if err = r.StreamsDao.UpdateStream(stream); err != nil {
+ _ = c.Error(tools.RequestError{
+ Status: http.StatusInternalServerError,
+ CustomMessage: "couldn't update lecture start/end time",
+ Err: err,
+ })
+ return
+ }
+}
+
func (r coursesRoutes) renameLecture(c *gin.Context) {
sIDInt, err := strconv.Atoi(c.Param("streamID"))
if err != nil {
diff --git a/web/assets/init-admin.js b/web/assets/init-admin.js
index f4e8be806..0b5fb6c6c 100644
--- a/web/assets/init-admin.js
+++ b/web/assets/init-admin.js
@@ -82,7 +82,7 @@ document.addEventListener("alpine:init", () => {
const changeSet = evaluate(expression);
const fieldName = value || el.name;
- if (el.type === "file") {
+ if (el.type.toLowerCase() === "file") {
const isSingle = modifiers.includes("single")
const changeHandler = (e) => {
@@ -103,7 +103,7 @@ document.addEventListener("alpine:init", () => {
changeSet.removeListener(onChangeSetUpdateHandler);
el.removeEventListener('change', changeHandler)
})
- } else if (el.type === "checkbox") {
+ } else if (el.type.toLowerCase() === "checkbox") {
const changeHandler = (e) => {
changeSet.patch(fieldName, e.target.checked);
};
@@ -121,7 +121,7 @@ document.addEventListener("alpine:init", () => {
changeSet.removeListener(onChangeSetUpdateHandler);
el.removeEventListener('change', changeHandler)
})
- } else if (el.tagName === "textarea" || textInputTypes.includes(el.type)) {
+ } else if (el.tagName.toLowerCase() === "textarea" || textInputTypes.includes(el.type.toLowerCase())) {
const keyupHandler = (e) => changeSet.patch(fieldName, convert(modifiers, e.target.value));
const changeHandler = (e) => changeSet.patch(fieldName, convert(modifiers, e.target.value));
@@ -172,23 +172,29 @@ document.addEventListener("alpine:init", () => {
*
* Modifiers:
* - "text": When provided, the directive will also update the element's innerText.
+ * - "value": When provided, the directive will also update the element's value.
*
* Custom Events:
* - "csupdate": Custom event triggered when the change set is updated.
* The detail property of the event object contains the new value of the specified field.
*/
Alpine.directive("change-set-listen", (el, { expression, modifiers }, { effect, evaluate, cleanup }) => {
- effect(() => {
- const [changeSetExpression, fieldName = null] = expression.split(".");
- const changeSet = evaluate(changeSetExpression);
+ const [changeSetExpression, fieldName = null] = expression.split(".");
+ let changeSet = evaluate(changeSetExpression);
- const onChangeSetUpdateHandler = (data) => {
- const value = fieldName != null ? data[fieldName] : data;
- if (modifiers.includes("text")) {
- el.innerText = `${value}`;
- }
- el.dispatchEvent(new CustomEvent(nativeEventName, { detail: { changeSet, value } }));
- };
+ const onChangeSetUpdateHandler = (data) => {
+ const value = fieldName != null ? data[fieldName] : data;
+ if (modifiers.includes("text")) {
+ el.innerText = `${value}`;
+ }
+ if (modifiers.includes("value")) {
+ el.value = value;
+ }
+ el.dispatchEvent(new CustomEvent(nativeEventName, { detail: { changeSet, value } }));
+ };
+
+ effect(() => {
+ changeSet = evaluate(changeSetExpression);
if (!changeSet) {
return;
@@ -197,11 +203,11 @@ document.addEventListener("alpine:init", () => {
changeSet.removeListener(onChangeSetUpdateHandler);
onChangeSetUpdateHandler(changeSet.get());
changeSet.listen(onChangeSetUpdateHandler);
-
- cleanup(() => {
- changeSet.removeListener(onChangeSetUpdateHandler);
- })
});
+
+ cleanup(() => {
+ changeSet.removeListener(onChangeSetUpdateHandler);
+ })
});
/**
diff --git a/web/template/partial/course/manage/lecture-management-card.gohtml b/web/template/partial/course/manage/lecture-management-card.gohtml
index 62ff54803..b84260384 100644
--- a/web/template/partial/course/manage/lecture-management-card.gohtml
+++ b/web/template/partial/course/manage/lecture-management-card.gohtml
@@ -255,6 +255,48 @@
+
+
+
+
+
+
+
+
+
@@ -395,7 +437,7 @@
diff --git a/web/ts/api/admin-lecture-list.ts b/web/ts/api/admin-lecture-list.ts
index fc65bc786..af8d51ce9 100644
--- a/web/ts/api/admin-lecture-list.ts
+++ b/web/ts/api/admin-lecture-list.ts
@@ -17,6 +17,11 @@ export interface UpdateLectureMetaRequest {
isChatEnabled?: boolean;
}
+export interface UpdateLectureStartEndRequest {
+ start: Date;
+ end: Date;
+}
+
export class LectureFile {
readonly id: number;
readonly fileType: number;
@@ -187,8 +192,9 @@ export interface Lecture {
startDateFormatted: string;
startTimeFormatted: string;
endDate: Date;
- endDateFormatted: string;
endTimeFormatted: string;
+ duration: number;
+ durationFormatted: string;
// Clientside pseudo fields
newCombinedVideo: File | null;
@@ -230,9 +236,9 @@ export const AdminLectureList = {
* @param request
*/
updateMetadata: async function (courseId: number, lectureId: number, request: UpdateLectureMetaRequest) {
- const promises = [];
+ const promises: (() => Promise)[] = [];
if (request.name !== undefined) {
- promises.push(
+ promises.push(() =>
post(`/api/course/${courseId}/renameLecture/${lectureId}`, {
name: request.name,
}),
@@ -240,7 +246,7 @@ export const AdminLectureList = {
}
if (request.description !== undefined) {
- promises.push(
+ promises.push(() =>
put(`/api/course/${courseId}/updateDescription/${lectureId}`, {
name: request.description,
}),
@@ -248,7 +254,7 @@ export const AdminLectureList = {
}
if (request.lectureHallId !== undefined) {
- promises.push(
+ promises.push(() =>
post("/api/setLectureHall", {
streamIds: [lectureId],
lectureHall: request.lectureHallId,
@@ -257,7 +263,7 @@ export const AdminLectureList = {
}
if (request.isChatEnabled !== undefined) {
- promises.push(
+ promises.push(() =>
patch(`/api/stream/${lectureId}/chat/enabled`, {
lectureId,
isChatEnabled: request.isChatEnabled,
@@ -265,8 +271,15 @@ export const AdminLectureList = {
);
}
- const errors = (await Promise.all(promises)).filter((res) => res.status !== StatusCodes.OK);
- if (errors.length > 0) {
+ let errors = 0;
+ for (const promise of promises) {
+ const res = await promise();
+ if (res.status !== StatusCodes.OK) {
+ errors++;
+ }
+ }
+
+ if (errors > 0) {
console.error(errors);
throw Error("Failed to update all data.");
}
@@ -281,6 +294,16 @@ export const AdminLectureList = {
await post(`/api/course/${courseId}/updateLectureSeries/${lectureId}`);
},
+ /**
+ * Updates date time of a lecture.
+ * @param courseId
+ * @param lectureId
+ * @param request
+ */
+ updateStartEnd: async function (courseId: number, lectureId: number, { start, end }: UpdateLectureStartEndRequest) {
+ await post(`/api/course/${courseId}/updateStartEnd/${lectureId}`, { start, end });
+ },
+
/**
* Add sections to a lecture
* @param lectureId
diff --git a/web/ts/change-set.ts b/web/ts/change-set.ts
index 72ed57ad4..7b5bf71c5 100644
--- a/web/ts/change-set.ts
+++ b/web/ts/change-set.ts
@@ -1,8 +1,27 @@
+import { throttle, ThrottleFunc } from "./throttle";
+
+export enum LogLevel {
+ none,
+ warn,
+ info,
+ debug,
+}
+
+const logLevelNames: string[] = ["None", "Warn", "Info", "Debug"];
+
export interface DirtyState {
isDirty: boolean;
dirtyKeys: string[];
}
+export interface ChangeSetOptions {
+ comparator?: (key: string, a: T, b: T) => boolean;
+ updateTransformer?: ComputedProperties;
+ onUpdate?: (changeState: T, dirtyState: DirtyState) => void;
+ updateThrottle?: number;
+ logLevel?: LogLevel;
+}
+
/**
* ## ChangeSet Class
*
@@ -57,16 +76,38 @@ export class ChangeSet {
private changeState: T;
private readonly comparator?: PropertyComparator;
private onUpdate: ((changeState: T, dirtyState: DirtyState) => void)[];
+ private readonly changeStateTransformer?: (changeState: T) => T;
+ private readonly stateTransformer?: (changeState: T) => T;
+
+ private readonly throttledDispatchUpdateNoStateChanged?: ThrottleFunc;
+ private readonly throttledDispatchUpdateStateChanged?: ThrottleFunc;
+
+ private readonly logLevel: LogLevel;
+ private lastLogTimestamp: number;
+ private static logIdCounter: number = 0;
+ private readonly logId: number;
constructor(
state: T,
- comparator?: (key: string, a: T, b: T) => boolean,
- onUpdate?: (changeState: T, dirtyState: DirtyState) => void,
+ {
+ comparator,
+ updateTransformer,
+ onUpdate,
+ updateThrottle = 0,
+ logLevel = LogLevel.none,
+ }: ChangeSetOptions = {},
) {
+ this.lastLogTimestamp = Date.now();
+ this.logLevel = logLevel;
+ this.logId = ChangeSet.logIdCounter++;
this.state = state;
this.onUpdate = onUpdate ? [onUpdate] : [];
+ this.changeStateTransformer = updateTransformer !== undefined ? updateTransformer.create() : undefined;
+ this.stateTransformer = updateTransformer !== undefined ? updateTransformer.create() : undefined;
this.comparator = comparator;
- this.reset();
+ this.throttledDispatchUpdateNoStateChanged = throttle(() => this._dispatchUpdate(false), updateThrottle, true);
+ this.throttledDispatchUpdateStateChanged = throttle(() => this._dispatchUpdate(true), updateThrottle, true);
+ this.init();
}
/**
@@ -74,6 +115,7 @@ export class ChangeSet {
* @param onUpdate
*/
listen(onUpdate: (changeState: T, dirtyState: DirtyState) => void) {
+ this._log(LogLevel.debug, "Add Update Listener", { onUpdate });
this.onUpdate.push(onUpdate);
}
@@ -82,6 +124,7 @@ export class ChangeSet {
* @param onUpdate
*/
removeListener(onUpdate: (changeState: T, dirtyState: DirtyState) => void) {
+ this._log(LogLevel.debug, "Remove Update Listener", { onUpdate });
this.onUpdate = this.onUpdate.filter((o) => o !== onUpdate);
}
@@ -90,7 +133,7 @@ export class ChangeSet {
* @param key key to return
* @param lastCommittedState if set to true, value of the last committed state is returned
*/
- getValue(key: string, { lastCommittedState = false }): T {
+ getValue(key: string, { lastCommittedState = false } = {}): T {
if (lastCommittedState) {
return this.state[key];
}
@@ -109,8 +152,9 @@ export class ChangeSet {
* @param val
*/
set(val: T) {
+ this._log(LogLevel.info, "Set ChangeState", { val });
this.changeState = { ...val };
- this.dispatchUpdate();
+ this.dispatchUpdateThrottled(false);
}
/**
@@ -121,11 +165,12 @@ export class ChangeSet {
*/
/* eslint-disable @typescript-eslint/no-explicit-any */
patch(key: string, val: any, { isCommitted = false }: { isCommitted?: boolean } = {}) {
+ this._log(LogLevel.info, "Patch State", { key, val, isCommitted });
this.changeState = { ...this.changeState, [key]: val };
if (isCommitted) {
this.state = { ...this.state, [key]: val };
}
- this.dispatchUpdate();
+ this.dispatchUpdateThrottled(isCommitted);
}
/**
@@ -133,6 +178,7 @@ export class ChangeSet {
* @param state
*/
updateState(state: T) {
+ this._log(LogLevel.info, "Update State", { state });
const changedKeys = this.changedKeys();
this.state = { ...state };
@@ -141,8 +187,7 @@ export class ChangeSet {
this.changeState[key] = this.state[key];
}
}
-
- this.dispatchUpdate();
+ this.dispatchUpdateThrottled(true);
}
/**
@@ -150,19 +195,30 @@ export class ChangeSet {
* @param discardKeys List of keys that should be discarded and not committed.
*/
commit({ discardKeys = [] }: { discardKeys?: string[] } = {}): void {
+ this._log(LogLevel.info, "Perform Commit", { discardKeys });
for (const key in discardKeys) {
this.changeState[key] = this.state[key];
}
this.state = { ...this.changeState };
- this.dispatchUpdate();
+ this.dispatchUpdateThrottled(true);
+ }
+
+ /**
+ * Init new state
+ */
+ init(): void {
+ this._log(LogLevel.info, "Perform Init");
+ this.changeState = { ...this.state };
+ this.dispatchUpdateThrottled(true);
}
/**
* Resets the change state to the state. Change state is the most current state afterwards.
*/
reset(): void {
+ this._log(LogLevel.info, "Perform Reset");
this.changeState = { ...this.state };
- this.dispatchUpdate();
+ this.dispatchUpdateThrottled(false);
}
/**
@@ -209,9 +265,23 @@ export class ChangeSet {
/**
* Executes all onUpdate listeners
+ * @param stateChanged if state changed, state computed values are recalculated
*/
- dispatchUpdate() {
+ _dispatchUpdate(stateChanged: boolean) {
+ this._log(LogLevel.info, "Dispatch Update", { stateChanged });
+
+ if (stateChanged && this.stateTransformer) {
+ this._log(LogLevel.info, "Reevaluate Computed on State");
+ this.state = this.stateTransformer(this.state);
+ }
+
+ if (this.changeStateTransformer) {
+ this._log(LogLevel.info, "Reevaluate Computed on ChangeState");
+ this.changeState = this.changeStateTransformer(this.changeState);
+ }
+
if (this.onUpdate.length > 0) {
+ this._log(LogLevel.info, `Trigger ${this.onUpdate.length} onUpdate handler`);
const dirtyKeys = this.changedKeys();
for (const onUpdate of this.onUpdate) {
onUpdate(this.changeState, {
@@ -221,6 +291,47 @@ export class ChangeSet {
}
}
}
+
+ /**
+ * Log Function that prints messages to console or discards them if the loglevel of changeSet is lower than the
+ * loglevel of the message
+ * @param level LogLevel of the message, if lower than changeSet's loglevel it will be discarded
+ * @param message The message to print to console
+ * @param payload Any payload object, "instance" is automatically attached if the changeSet loglevel is set to "debug"
+ */
+ _log(level: LogLevel, message: string, payload?: object) {
+ if (level <= this.logLevel) {
+ const ts = Date.now();
+ const tsDelta = ts - this.lastLogTimestamp;
+ this.lastLogTimestamp = ts;
+
+ if (this.logLevel === LogLevel.debug) {
+ payload = { ...(payload ?? {}), instance: this };
+ }
+
+ console.log(
+ `[CHANGESET | ${this.logId.toString().padStart(4, "0")} | ${logLevelNames[level].padEnd(
+ 5,
+ " ",
+ )}] ${message}`,
+ payload ?? "",
+ `[${tsDelta}ms]`,
+ );
+ }
+ }
+
+ /**
+ * Executes all onUpdate listeners
+ * @param stateChanged if state changed, state computed values are recalculated
+ */
+ dispatchUpdateThrottled(stateChanged: boolean) {
+ this._log(LogLevel.debug, "Throttled: Dispatch Update", { stateChanged });
+ if (stateChanged && this.stateTransformer) {
+ return this.throttledDispatchUpdateStateChanged();
+ } else {
+ return this.throttledDispatchUpdateNoStateChanged();
+ }
+ }
}
export type PropertyComparator = (key: string, a: T, b: T) => boolean | null;
@@ -244,6 +355,15 @@ export function singleProperty(key: string, comparator: SinglePropertyCompara
};
}
+export function multiProperty(keys: string[], comparator: PropertyComparator): PropertyComparator {
+ return (_key: string, a, b) => {
+ if (!keys.includes(_key)) {
+ return null;
+ }
+ return comparator(_key, a, b);
+ };
+}
+
export function comparatorPipeline(list: PropertyComparator[]): PropertyComparator {
return (key: string, a, b) => {
for (const comparator of list) {
@@ -257,3 +377,51 @@ export function comparatorPipeline(list: PropertyComparator[]): PropertyCo
return null;
};
}
+
+export type ComputedPropertyTransformer = (state: T) => T;
+export type ComputedPropertySubTransformer = {
+ transform: (state: T, oldState: T) => T;
+ key: string;
+};
+
+export class ComputedProperties {
+ private readonly computed: ComputedPropertySubTransformer[];
+
+ constructor(computed: ComputedPropertySubTransformer[]) {
+ this.computed = computed;
+ }
+
+ create(): ComputedPropertyTransformer {
+ let oldState: T | null = null;
+ return (state: T) => {
+ for (const transformer of this.computed) {
+ state = transformer.transform(state, oldState);
+ }
+ oldState = { ...state };
+ return state;
+ };
+ }
+
+ /**
+ * This is syntactic sugar to quickly ignore the computed keys in a changeset
+ */
+ ignore(): PropertyComparator {
+ return ignoreKeys(this.computed.map((transformer) => transformer.key));
+ }
+}
+
+export function computedProperty(
+ key: string,
+ updater: (changeState: T, old: T | null) => R,
+ deps: string[] = [],
+): ComputedPropertySubTransformer {
+ return {
+ transform: (state: T, oldState: T | null) => {
+ if (oldState == null || deps.length == 0 || deps.some((k) => oldState[k] !== state[k])) {
+ state[key] = updater(state, oldState);
+ }
+ return state;
+ },
+ key,
+ };
+}
diff --git a/web/ts/data-store/admin-lecture-list.ts b/web/ts/data-store/admin-lecture-list.ts
index 6633bf78e..c5295e95d 100644
--- a/web/ts/data-store/admin-lecture-list.ts
+++ b/web/ts/data-store/admin-lecture-list.ts
@@ -4,19 +4,20 @@ import {
Lecture,
LectureFile,
UpdateLectureMetaRequest,
+ UpdateLectureStartEndRequest,
VideoSection,
videoSectionSort,
} from "../api/admin-lecture-list";
import { FileType } from "../edit-course";
import { PostFormDataListener } from "../utilities/fetch-wrappers";
-const dateFormatOptions: Intl.DateTimeFormatOptions = {
+export const dateFormatOptions: Intl.DateTimeFormatOptions = {
weekday: "long",
year: "numeric",
month: "short",
day: "2-digit",
};
-const timeFormatOptions: Intl.DateTimeFormatOptions = {
+export const timeFormatOptions: Intl.DateTimeFormatOptions = {
hour: "2-digit",
minute: "2-digit",
};
@@ -36,14 +37,6 @@ export class AdminLectureListProvider extends StreamableMapProvider {
+ if (s.lectureId === lectureId) {
+ return {
+ ...s,
+ start: payload.start,
+ end: payload.end,
+ };
+ }
+ return s;
+ });
+ await this.triggerUpdate(courseId);
+ }
+
async uploadAttachmentFile(courseId: number, lectureId: number, file: File) {
const res = await AdminLectureList.uploadAttachmentFile(courseId, lectureId, file);
const newFile = new LectureFile({
diff --git a/web/ts/edit-course.ts b/web/ts/edit-course.ts
index c517000c6..f2ac6da78 100644
--- a/web/ts/edit-course.ts
+++ b/web/ts/edit-course.ts
@@ -15,9 +15,18 @@ import {
videoSectionSort,
videoSectionTimestamp,
} from "./api/admin-lecture-list";
-import { ChangeSet, comparatorPipeline, ignoreKeys, singleProperty } from "./change-set";
+import {
+ ChangeSet,
+ comparatorPipeline,
+ ComputedProperties,
+ computedProperty,
+ ignoreKeys,
+ multiProperty,
+ singleProperty,
+} from "./change-set";
import { AlpineComponent } from "./components/alpine-component";
import { uploadFile } from "./utilities/fetch-wrappers";
+import { dateFormatOptions, timeFormatOptions } from "./data-store/admin-lecture-list";
export enum UIEditMode {
none,
@@ -111,6 +120,7 @@ export function lectureEditor(lecture: Lecture): AlpineComponent {
uiEditMode: UIEditMode.none,
isDirty: false,
isSaving: false,
+ isInvalid: false,
// Lecture Data
changeSet: null as ChangeSet | null,
@@ -120,8 +130,65 @@ export function lectureEditor(lecture: Lecture): AlpineComponent {
* AlpineJS init function which is called automatically in addition to 'x-init'
*/
init() {
+ const computedFields = new ComputedProperties([
+ computedProperty(
+ "startDate",
+ (changeSet) => {
+ return new Date(changeSet.start);
+ },
+ ["start"],
+ ),
+ computedProperty(
+ "startDateFormatted",
+ (changeSet) => {
+ return changeSet.startDate.toLocaleDateString("en-US", dateFormatOptions);
+ },
+ ["start"],
+ ),
+ computedProperty(
+ "startTimeFormatted",
+ (changeSet) => {
+ return changeSet.startDate.toLocaleDateString("en-US", timeFormatOptions);
+ },
+ ["start"],
+ ),
+ computedProperty(
+ "endDate",
+ (changeSet) => {
+ return new Date(changeSet.end);
+ },
+ ["end"],
+ ),
+ computedProperty(
+ "endTimeFormatted",
+ (changeSet) => {
+ return changeSet.endDate.toLocaleTimeString("en-US", timeFormatOptions);
+ },
+ ["end"],
+ ),
+ computedProperty(
+ "duration",
+ (changeSet) => {
+ // To ignore day differences
+ const normalizedEndDate = new Date(changeSet.startDate.getTime());
+ normalizedEndDate.setHours(changeSet.endDate.getHours());
+ normalizedEndDate.setMinutes(changeSet.endDate.getMinutes());
+ return normalizedEndDate.getTime() - changeSet.startDate.getTime();
+ },
+ ["start", "end"],
+ ),
+ computedProperty(
+ "durationFormatted",
+ (changeSet) => {
+ return this.generateFormattedDuration(changeSet);
+ },
+ ["start", "end"],
+ ),
+ ]);
+
const customComparator = comparatorPipeline([
- ignoreKeys(["files"]),
+ computedFields.ignore(),
+ ignoreKeys(["files", "downloadableVods", "transcodingProgresses"]),
singleProperty("videoSections", (a: Lecture, b: Lecture): boolean | null => {
// List length differs, something was removed or added
if (a.videoSections.length !== b.videoSections.length) {
@@ -139,12 +206,22 @@ export function lectureEditor(lecture: Lecture): AlpineComponent {
// A section has edited and different information now
return a.videoSections.some((sA) => b.videoSections.some((sB) => videoSectionHasChanged(sA, sB)));
}),
+ multiProperty(["start", "end"], (key: string, a: Lecture, b: Lecture): boolean | null => {
+ const dateA = new Date(a[key]);
+ const dateB = new Date(b[key]);
+ return dateA.getTime() !== dateB.getTime();
+ }),
]);
// This tracks changes that are not saved yet
- this.changeSet = new ChangeSet(lecture, customComparator, (data, dirtyState) => {
- this.lectureData = data;
- this.isDirty = dirtyState.isDirty;
+ this.changeSet = new ChangeSet(lecture, {
+ comparator: customComparator,
+ updateTransformer: computedFields,
+ onUpdate: (data, dirtyState) => {
+ this.lectureData = data;
+ this.isDirty = dirtyState.isDirty;
+ this.isInvalid = data.duration <= 0;
+ },
});
// This updates the state live in background
@@ -218,6 +295,23 @@ export function lectureEditor(lecture: Lecture): AlpineComponent {
DataStore.adminLectureList.deleteAttachment(this.lectureData.courseId, this.lectureData.lectureId, id);
},
+ generateFormattedDuration(lecture: Lecture): string {
+ if (lecture.duration <= 0) {
+ return "invalid";
+ }
+ const duration = lecture.duration / 1000 / 60;
+ const hours = Math.floor(duration / 60);
+ const minutes = duration - hours * 60;
+ let res = "";
+ if (hours > 0) {
+ res += `${hours}h `;
+ }
+ if (minutes > 0) {
+ res += `${minutes}min`;
+ }
+ return res;
+ },
+
friendlySectionTimestamp(section: VideoSection): string {
return videoSectionFriendlyTimestamp(section);
},
@@ -317,7 +411,7 @@ export function lectureEditor(lecture: Lecture): AlpineComponent {
* Save changes send them to backend and commit change set.
*/
async saveEdit() {
- const { courseId, lectureId, name, description, lectureHallId, isChatEnabled, videoSections } =
+ const { courseId, lectureId, name, description, lectureHallId, isChatEnabled, videoSections, start, end } =
this.lectureData;
const changedKeys = this.changeSet.changedKeys();
@@ -335,6 +429,19 @@ export function lectureEditor(lecture: Lecture): AlpineComponent {
},
});
+ // Save new date and time
+ if (changedKeys.includes("start") || changedKeys.includes("end")) {
+ const startDate = new Date(start);
+ const endDate = new Date(end);
+ endDate.setFullYear(startDate.getFullYear());
+ endDate.setMonth(startDate.getMonth());
+ endDate.setDate(startDate.getDate());
+ await DataStore.adminLectureList.updateStartEnd(courseId, lectureId, {
+ start: startDate,
+ end: endDate,
+ });
+ }
+
// Saving VideoSections
if (changedKeys.includes("videoSections")) {
const oldVideoSections = this.changeSet.getValue("videoSections", { lastCommittedState: true });
diff --git a/web/ts/throttle.ts b/web/ts/throttle.ts
new file mode 100644
index 000000000..d3ebac9f6
--- /dev/null
+++ b/web/ts/throttle.ts
@@ -0,0 +1,22 @@
+// eslint-disable-next-line @typescript-eslint/no-explicit-any
+export type ThrottleFunc = (...args: any[]) => void;
+
+export function throttle(fun: ThrottleFunc, delay = 100, skipFirst: boolean = false): ThrottleFunc {
+ let lastInstance: NodeJS.Timeout | null = null;
+ let first: boolean = true;
+
+ return (...args): void => {
+ if (skipFirst && first) {
+ first = false;
+ fun(...args);
+ return;
+ }
+
+ if (lastInstance !== null) {
+ clearTimeout(lastInstance);
+ }
+ lastInstance = setTimeout(() => {
+ fun(...args);
+ }, delay);
+ };
+}