Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Improvement/refactor course sections #1156

Merged
merged 71 commits into from
Sep 30, 2023
Merged
Show file tree
Hide file tree
Changes from 1 commit
Commits
Show all changes
71 commits
Select commit Hold shift + click to select a range
84d18ee
started wit hrefactor of course management
mono424 Aug 23, 2023
eb2fcd7
Merge branch 'dev' into improvement/refactor-course-mgmt
mono424 Aug 28, 2023
429ad67
add dao & route
mono424 Aug 28, 2023
d1ea050
update route
mono424 Aug 28, 2023
8aa4d69
started wit hfetching
mono424 Aug 28, 2023
db3f3bb
Add changeset abstraction
mono424 Aug 28, 2023
0225a59
cleanup file
mono424 Aug 28, 2023
29c0977
remove get all streams again, as there is AdminJson method
mono424 Aug 28, 2023
0f32725
Its rendering
mono424 Aug 28, 2023
797d00d
further dev :S
mono424 Aug 28, 2023
17f460a
added some directive
mono424 Aug 28, 2023
ca9c567
little fixes
mono424 Aug 28, 2023
246fa48
...
mono424 Aug 28, 2023
1e66fc9
... no idea...
mono424 Aug 28, 2023
e36890d
right path i guess
mono424 Aug 29, 2023
fcf15c2
changeset working except video sections
mono424 Aug 29, 2023
e576712
fix discard files
mono424 Aug 29, 2023
c25956a
save from laptop
mono424 Sep 3, 2023
5102141
Add changeset doku
mono424 Sep 3, 2023
7518b5d
make private works like charm
mono424 Sep 3, 2023
8416d26
changing works pretty well
mono424 Sep 3, 2023
8afb2b9
add video upload
mono424 Sep 3, 2023
f4a005f
add series update
mono424 Sep 3, 2023
a258dc5
ported transcoding
mono424 Sep 3, 2023
184b0aa
add readme and lecture hall select
mono424 Sep 5, 2023
81e783d
more description
mono424 Sep 5, 2023
cffbf3e
simplified lecture hall set
mono424 Sep 5, 2023
5548883
Merge branch 'dev' into improvement/refactor-course-mgmt
mono424 Sep 5, 2023
2db8643
linter
mono424 Sep 5, 2023
098b2d5
more fixes
mono424 Sep 5, 2023
91f80cf
:)
mono424 Sep 5, 2023
b224222
add attachments
mono424 Sep 10, 2023
0a65347
:)
mono424 Sep 10, 2023
1b97d1a
Merge branch 'dev' into improvement/refactor-course-mgmt
mono424 Sep 11, 2023
74b180d
add video sections to admin streams
mono424 Sep 11, 2023
85a17f5
main functionality
mono424 Sep 11, 2023
f21f0db
using fetch wrtappers
mono424 Sep 14, 2023
0b939c6
moved uploadFile and postFormData
mono424 Sep 14, 2023
09105a4
Merge branch 'dev' into improvement/refactor-course-sections
mono424 Sep 14, 2023
df179a7
fix linteer
mono424 Sep 14, 2023
4408da4
Impl. Feedback
mono424 Sep 14, 2023
394218e
add video sections to admin streams
mono424 Sep 11, 2023
61b36b5
main functionality
mono424 Sep 11, 2023
f78c548
using fetch wrtappers
mono424 Sep 14, 2023
74cd080
moved uploadFile and postFormData
mono424 Sep 14, 2023
a6cf16e
version workers with tag (#1157)
joschahenningsen Sep 13, 2023
f1ad4a0
fix linteer
mono424 Sep 14, 2023
bf11272
Merge branch 'improvement/refactor-course-sections' of github.com:jos…
mono424 Sep 14, 2023
859555c
fix delete lecture
mono424 Sep 14, 2023
d375cf7
reenable page reload on create
mono424 Sep 14, 2023
8755076
Merge branch 'improvement/refactor-course-mgmt' into improvement/refa…
mono424 Sep 15, 2023
1b1b1ab
some linter fixes
mono424 Sep 15, 2023
b996d07
fix merge complications
mono424 Sep 15, 2023
7bb8870
fix changeset; adding onchange listener
mono424 Sep 15, 2023
8c132ec
fixed adding sections
mono424 Sep 15, 2023
6c57fa0
added directives
mono424 Sep 15, 2023
8090778
fix typo
mono424 Sep 15, 2023
c810b4c
improved directives
mono424 Sep 15, 2023
96ba056
design improvement
mono424 Sep 15, 2023
44d2031
fixed creating and deleting
mono424 Sep 15, 2023
2ebd58a
add sections fixed
mono424 Sep 15, 2023
738087a
cleanup
mono424 Sep 15, 2023
f11c393
:)
mono424 Sep 15, 2023
0beec3a
fix alpine js error issues
mono424 Sep 15, 2023
7e63051
remoev console log
mono424 Sep 15, 2023
a50db12
nested :)
mono424 Sep 15, 2023
0da5f03
remove console log
mono424 Sep 15, 2023
6a181d2
lint-fix
mono424 Sep 15, 2023
cc9f2d0
Merge branch 'dev' into improvement/refactor-course-sections
mono424 Sep 21, 2023
3616440
Merge branch 'dev' into improvement/refactor-course-sections
joschahenningsen Sep 29, 2023
4958d64
fix admin.LectureList initialisation in settings tab
joschahenningsen Sep 30, 2023
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Prev Previous commit
Next Next commit
add attachments
  • Loading branch information
mono424 committed Sep 10, 2023
commit b22422261f2ea8588b61d4c07a5f3f8063d95cac
Original file line number Diff line number Diff line change
Expand Up @@ -272,7 +272,7 @@
class="w-full">
<label :for="id" class="hidden">Lecture description</label>
<textarea :id="id"
@drop.prevent="(e) => lecture.onFileDrop(e)"
@drop.prevent="(e) => onAttachmentFileDrop(e)"
@dragover.prevent=""
class="tl-textarea grow"
placeholder="Add a nice description, links, and more. You can use Markdown. Drop files here."
Expand Down Expand Up @@ -349,7 +349,7 @@
<span class="text-xs font-semibold text-3 my-auto"
x-text="file.friendlyName"></span>
<button class="px-3"
@click="lecture.deleteFile(file.id)">
@click="deleteAttachment(file.id)">
<i class="fa fa-xmark"></i>
</button>
</section>
Expand Down
68 changes: 66 additions & 2 deletions web/ts/api/admin-lecture-list.ts
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
import { del, get, post, put } from "../utilities/fetch-wrappers";
import { StatusCodes } from "http-status-codes";
import { patchData, postData, putData, uploadFile, UploadFileListener } from "../global";
import {Delete, patchData, postData, postFormData, putData, uploadFile, UploadFileListener} from "../global";

export interface UpdateLectureMetaRequest {
name?: string;
Expand All @@ -9,6 +9,18 @@ export interface UpdateLectureMetaRequest {
isChatEnabled?: boolean;
}

export class LectureFile {
readonly id: number;
readonly fileType: number;
readonly friendlyName: string;

constructor({ id, fileType, friendlyName }) {
this.id = id;
this.fileType = fileType;
this.friendlyName = friendlyName;
}
}

export interface CreateNewLectureRequest {
title: "";
lectureHallId: 0;
Expand Down Expand Up @@ -67,7 +79,7 @@ export interface Lecture {
description: string;
downloadableVods: DownloadableVOD[];
end: string;
files: null;
files: LectureFile[];
hasStats: boolean;
isChatEnabled: boolean;
isConverting: boolean;
Expand Down Expand Up @@ -223,6 +235,58 @@ export const AdminLectureList = {
);
},

/**
* Upload a file as attachment for a lecture
* @param courseId
* @param lectureId
* @param file
* @param listener
*/
uploadAttachmentFile: async (
courseId: number,
lectureId: number,
file: File,
listener: UploadFileListener = {},
) => {
return await uploadFile(
`/api/stream/${lectureId}/files?type=file`,
file,
listener,
);
},

/**
* Upload a url as attachment for a lecture
* @param courseId
* @param lectureId
* @param url
* @param listener
*/
uploadAttachmentUrl: async (
courseId: number,
lectureId: number,
url: string,
listener: UploadFileListener = {},
) => {
const vodUploadFormData = new FormData();
vodUploadFormData.append("file_url", url);
return postFormData(`/api/stream/${lectureId}/files?type=url`, vodUploadFormData, listener);
},

deleteAttachment: async (
courseId: number,
lectureId: number,
attachmentId: number,
) => {
return Delete(`/api/stream/${lectureId}/files/${attachmentId}`);
},

/**
* Get transcoding progress
* @param courseId
* @param lectureId
* @param version
*/
getTranscodingProgress: async (courseId: number, lectureId: number, version: number): Promise<number> => {
return (await fetch(`/api/course/${courseId}/stream/${lectureId}/transcodingProgress?v=${version}`)).json();
},
Expand Down
13 changes: 12 additions & 1 deletion web/ts/change-set.ts
Original file line number Diff line number Diff line change
Expand Up @@ -55,7 +55,7 @@ export interface DirtyState {
export class ChangeSet<T> {
private state: T;
private changeState: T;
private readonly comparator?: (key: string, a: T, b: T) => boolean | null;
private readonly comparator?: LectureComparator<T>;
private onUpdate: ((changeState: T, dirtyState: DirtyState) => void)[];

constructor(
Expand Down Expand Up @@ -210,3 +210,14 @@ export class ChangeSet<T> {
}
}
}

export type LectureComparator<T> = (key: string, a: T, b: T) => boolean | null;

export function ignoreKeys<T>(list: string[]): LectureComparator<T> {
return (key: string, a, b) => {
if (list.includes(key)) {
return false;
}
return null;
}
}
57 changes: 56 additions & 1 deletion web/ts/data-store/admin-lecture-list.ts
Original file line number Diff line number Diff line change
@@ -1,5 +1,5 @@
import { StreamableMapProvider } from "./provider";
import { AdminLectureList, Lecture, UpdateLectureMetaRequest } from "../api/admin-lecture-list";
import {AdminLectureList, Lecture, LectureFile, UpdateLectureMetaRequest} from "../api/admin-lecture-list";
import { FileType } from "../edit-course";
import { UploadFileListener } from "../global";

Expand Down Expand Up @@ -101,6 +101,61 @@ export class AdminLectureListProvider extends StreamableMapProvider<number, Lect
await this.triggerUpdate(courseId);
}

async uploadAttachmentFile(courseId: number, lectureId: number, file: File) {
const res = await AdminLectureList.uploadAttachmentFile(courseId, lectureId, file);
const newFile = new LectureFile({
id: JSON.parse(res.responseText),
fileType: 2,
friendlyName: file.name
});

this.data[courseId] = (await this.getData(courseId)).map((s) => {
if (s.lectureId === lectureId) {
return {
...s,
files: [...s.files, newFile]
};
}
return s;
});
await this.triggerUpdate(courseId);
}

async uploadAttachmentUrl(courseId: number, lectureId: number, url: String) {
const res = await AdminLectureList.uploadAttachmentUrl(courseId, lectureId, url);
const newFile = new LectureFile({
id: JSON.parse(res.responseText),
fileType: 2,
friendlyName: url.substring(url.lastIndexOf("/") + 1) }
);

this.data[courseId] = (await this.getData(courseId)).map((s) => {
if (s.lectureId === lectureId) {
return {
...s,
files: [...s.files, newFile]
};
}
return s;
});
await this.triggerUpdate(courseId);
}

async deleteAttachment(courseId: number, lectureId: number, attachmentId: number) {
await AdminLectureList.deleteAttachment(courseId, lectureId, attachmentId);

this.data[courseId] = (await this.getData(courseId)).map((s) => {
if (s.lectureId === lectureId) {
return {
...s,
files: [...s.files.filter(((a)=> a.id !== attachmentId))],
};
}
return s;
});
await this.triggerUpdate(courseId);
}

async uploadVideo(
courseId: number,
lectureId: number,
Expand Down
60 changes: 34 additions & 26 deletions web/ts/edit-course.ts
Original file line number Diff line number Diff line change
Expand Up @@ -10,7 +10,7 @@ import {
LectureVideoTypePres,
LectureVideoTypes,
} from "./api/admin-lecture-list";
import { ChangeSet } from "./change-set";
import {ChangeSet, ignoreKeys} from "./change-set";
import { AlpineComponent } from "./components/alpine-component";

export enum UIEditMode {
Expand Down Expand Up @@ -81,18 +81,6 @@ export class LectureList {
}
}

class LectureFile {
readonly id: number;
readonly fileType: number;
readonly friendlyName: string;

constructor({ id, fileType, friendlyName }) {
this.id = id;
this.fileType = fileType;
this.friendlyName = friendlyName;
}
}

class DownloadableVod {
downloadURL: string;
friendlyName: string;
Expand Down Expand Up @@ -127,7 +115,7 @@ export function lectureEditor(lecture: Lecture): AlpineComponent {
*/
init() {
// This tracks changes that are not saved yet
this.changeSet = new ChangeSet<Lecture>(lecture, this.lectureComparator, (data, dirtyState) => {
this.changeSet = new ChangeSet<Lecture>(lecture, ignoreKeys(["files"]), (data, dirtyState) => {
this.lectureData = data;
this.isDirty = dirtyState.isDirty;
});
Expand All @@ -141,18 +129,6 @@ export function lectureEditor(lecture: Lecture): AlpineComponent {
});
},

/**
* A custom comparator to setup special comparison strategies for specific keys
* @param key key in Lecture
* @param a
* @param b
* @return return 0 to use naive default comparator
*/
lectureComparator(key: string, a: Lecture, b: Lecture): boolean | null {
// here we can set some custom comparisons
return null;
},

toggleVisibility() {
DataStore.adminLectureList.setPrivate(
this.lectureData.courseId,
Expand Down Expand Up @@ -187,6 +163,38 @@ export function lectureEditor(lecture: Lecture): AlpineComponent {
return this.lectureData[key];
},

onAttachmentFileDrop(e) {
if (e.dataTransfer.items) {
const item = e.dataTransfer.items[0];
const { kind } = item;
switch (kind) {
case "file": {
DataStore.adminLectureList.uploadAttachmentFile(
this.lectureData.courseId,
this.lectureData.lectureId,
item.getAsFile(),
);
break;
}
case "string": {
DataStore.adminLectureList.uploadAttachmentUrl(
this.lectureData.courseId,
this.lectureData.lectureId,
e.dataTransfer.getData("URL"),
);
}
}
}
},

deleteAttachment(id: number) {
DataStore.adminLectureList.deleteAttachment(
this.lectureData.courseId,
this.lectureData.lectureId,
id,
);
},

/**
* Opens the series lecture editor UI
*/
Expand Down
8 changes: 6 additions & 2 deletions web/ts/global.ts
Original file line number Diff line number Diff line change
Expand Up @@ -50,9 +50,13 @@ export interface UploadFileListener {
}

export function uploadFile(url: string, file: File, listener: UploadFileListener = {}): Promise<XMLHttpRequest> {
const xhr = new XMLHttpRequest();
const vodUploadFormData = new FormData();
vodUploadFormData.append("file", file);
return postFormData(url, vodUploadFormData, listener);
}

export function postFormData(url: string, data: FormData, listener: UploadFileListener = {}): Promise<XMLHttpRequest> {
const xhr = new XMLHttpRequest();
return new Promise((resolve, reject) => {
xhr.onloadend = () => {
if (xhr.status === 200) {
Expand All @@ -70,7 +74,7 @@ export function uploadFile(url: string, file: File, listener: UploadFileListener
}
};
xhr.open("POST", url);
xhr.send(vodUploadFormData);
xhr.send(data);
});
}

Expand Down