mirror of
https://codeberg.org/PLG-Development/PLG-MuDiCS
synced 2026-07-05 16:37:09 +00:00
@@ -60,7 +60,7 @@
|
||||
file_size = liveQuery(() => get_size_recursively(f));
|
||||
});
|
||||
|
||||
let file_transfer_task: FileTransferTask | null = $derived(
|
||||
let file_transfer_task_list: FileTransferTask[] | null = $derived(
|
||||
Object.hasOwn($file_transfer_tasks, file_primary_key)
|
||||
? $file_transfer_tasks[file_primary_key]
|
||||
: null
|
||||
@@ -69,7 +69,7 @@
|
||||
let loading_finished = $state(false);
|
||||
let previous_loading_state = $state(false);
|
||||
$effect(() => {
|
||||
const ftt = file_transfer_task;
|
||||
const ftt = file_transfer_task_list;
|
||||
if (previous_loading_state && !ftt) {
|
||||
loading_finished = true;
|
||||
setTimeout(() => (loading_finished = false), 200);
|
||||
@@ -141,7 +141,7 @@
|
||||
|
||||
function get_grayed_out_text_color_strings(is_selected: boolean): string {
|
||||
if (not_interactable) return 'text-stone-400';
|
||||
if (file_transfer_task) return 'text-white/20';
|
||||
if (file_transfer_task_list) return 'text-white/20';
|
||||
const color = is_selected ? 'text-stone-600' : 'text-stone-400';
|
||||
const factor = is_selected ? -1 : 1;
|
||||
return `${color} group-hover:${get_shifted_color(color, factor * 100)} group-active:${get_shifted_color(color, factor * 150)}`;
|
||||
@@ -149,14 +149,14 @@
|
||||
|
||||
function get_grayed_out_border_color_strings(is_selected: boolean): string {
|
||||
if (not_interactable) return 'border-stone-550';
|
||||
if (file_transfer_task) return 'border-white/10';
|
||||
if (file_transfer_task_list) return 'border-white/10';
|
||||
const color = is_selected ? 'border-stone-450' : 'border-stone-550';
|
||||
const factor = is_selected ? 1 : 1;
|
||||
return `${color} group-hover:${get_shifted_color(color, factor * 100)} group-active:${get_shifted_color(color, factor * 150)}`;
|
||||
}
|
||||
|
||||
function onclick(e: Event) {
|
||||
if (not_interactable || file_transfer_task) return;
|
||||
if (not_interactable || file_transfer_task_list) return;
|
||||
select(selected_file_ids, file_primary_key, 'toggle');
|
||||
e.stopPropagation();
|
||||
}
|
||||
@@ -180,7 +180,7 @@
|
||||
|
||||
if (loading_finished) {
|
||||
out += 'bg-stone-500 text-white/30';
|
||||
} else if (file_transfer_task) {
|
||||
} else if (file_transfer_task_list) {
|
||||
out += 'bg-stone-700 text-white/30';
|
||||
} else {
|
||||
out += get_selectable_color_classes(
|
||||
@@ -196,7 +196,7 @@
|
||||
|
||||
if (not_interactable) {
|
||||
out += ' rounded-lg';
|
||||
} else if (file_transfer_task) {
|
||||
} else if (file_transfer_task_list) {
|
||||
out += ' rounded-r-lg';
|
||||
} else {
|
||||
out += ' rounded-r-lg cursor-pointer';
|
||||
@@ -205,7 +205,16 @@
|
||||
return out;
|
||||
}
|
||||
|
||||
function get_total_percentage(ftt: FileTransferTask): number {
|
||||
function get_total_percentage(ftt_list: FileTransferTask[]): number {
|
||||
let percentage_sum = 0;
|
||||
for (const ftt of ftt_list) {
|
||||
percentage_sum += get_percentage(ftt);
|
||||
}
|
||||
return Math.round(percentage_sum / ftt_list.length);
|
||||
}
|
||||
|
||||
|
||||
function get_percentage(ftt: FileTransferTask): number {
|
||||
let total_percentage: number;
|
||||
if (ftt.data.type === 'upload') {
|
||||
total_percentage = ftt.loading_data.percentage;
|
||||
@@ -312,10 +321,10 @@
|
||||
{onclick}
|
||||
class="{get_main_classes()} relative transition-colors duration-200 gap-4 flex flex-row justify-between group w-full min-w-0"
|
||||
>
|
||||
{#if !!file_transfer_task}
|
||||
{#if !!file_transfer_task_list}
|
||||
<div
|
||||
class="absolute pointer-events-none inset-y-0 left-0 transition-[width] duration-400 bg-stone-600 rounded-r-lg"
|
||||
style={`width: ${get_total_percentage(file_transfer_task)}%;`}
|
||||
style={`width: ${get_total_percentage(file_transfer_task_list)}%;`}
|
||||
></div>
|
||||
{/if}
|
||||
<div class="flex flex-row gap-2 min-w-0 w-full z-10">
|
||||
|
||||
@@ -28,12 +28,23 @@ const START_LOADING_DATA = {
|
||||
seconds_until_finish: -1
|
||||
};
|
||||
|
||||
export const file_transfer_tasks: Writable<Record<string, FileTransferTask>> = writable<
|
||||
Record<string, FileTransferTask>
|
||||
export const file_transfer_tasks: Writable<Record<string, FileTransferTask[]>> = writable<
|
||||
Record<string, FileTransferTask[]>
|
||||
>({});
|
||||
|
||||
let is_processing: boolean = false;
|
||||
|
||||
function add_file_transfer_task(file_primary_key: string, new_task: FileTransferTask) {
|
||||
file_transfer_tasks.update((tasks) => {
|
||||
if (Object.hasOwn(tasks, file_primary_key)) {
|
||||
tasks[file_primary_key].push(new_task);
|
||||
return tasks;
|
||||
} else {
|
||||
return { ...tasks, [file_primary_key]: [new_task] };
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
export async function add_upload(
|
||||
file_list: FileList,
|
||||
selected_display_ids: string[],
|
||||
@@ -89,10 +100,7 @@ export async function add_upload(
|
||||
loading_data: START_LOADING_DATA,
|
||||
bytes_total: file.size
|
||||
};
|
||||
file_transfer_tasks.update((tasks) => ({
|
||||
...tasks,
|
||||
[file_primary_key]: new_task
|
||||
}));
|
||||
add_file_transfer_task(file_primary_key, new_task);
|
||||
});
|
||||
|
||||
await Promise.all(upload_task_promises);
|
||||
@@ -146,10 +154,7 @@ export async function add_sync_recursively(
|
||||
bytes_total: file_data.file.size
|
||||
};
|
||||
|
||||
file_transfer_tasks.update((tasks) => ({
|
||||
...tasks,
|
||||
[selected_file_id]: new_task
|
||||
}));
|
||||
add_file_transfer_task(selected_file_id, new_task);
|
||||
|
||||
const display_ids_without_file = file_data.short_displays_without_file.map((d) => d.id);
|
||||
const new_fods: FileOnDisplay[] = display_ids_without_file.map((display_id) => ({
|
||||
@@ -230,15 +235,19 @@ function generate_valid_file_name(original_file_name: string, used_file_names: s
|
||||
return name;
|
||||
}
|
||||
|
||||
async function upload(file_primary_key: string, task: FileTransferTask): Promise<void> {
|
||||
async function upload(
|
||||
file_primary_key: string,
|
||||
task: FileTransferTask,
|
||||
list_index: number
|
||||
): Promise<void> {
|
||||
const task_data = task.data;
|
||||
if (task_data.type !== 'upload' || !task_data.file)
|
||||
return console.warn('Task cancelled: wrong task type:', task);
|
||||
|
||||
await upload_file_via_xhr(file_primary_key, task, task_data.file);
|
||||
await upload_file_via_xhr(file_primary_key, list_index, task, task_data.file);
|
||||
}
|
||||
|
||||
export async function sync(file_primary_key: string, task: FileTransferTask) {
|
||||
export async function sync(file_primary_key: string, task: FileTransferTask, list_index: number) {
|
||||
if (task.data.type !== 'sync') return console.warn('Task cancelled: wrong task type:', task);
|
||||
|
||||
const hasOPFS =
|
||||
@@ -273,18 +282,24 @@ export async function sync(file_primary_key: string, task: FileTransferTask) {
|
||||
if (done) break;
|
||||
if (!value) continue;
|
||||
|
||||
update_current_loading_data(file_primary_key, value.byteLength, start_time);
|
||||
update_current_loading_data(file_primary_key, list_index, value.byteLength, start_time);
|
||||
await writable.write(value);
|
||||
}
|
||||
await writable.close();
|
||||
|
||||
finish_loading_data(file_primary_key);
|
||||
finish_loading_data(file_primary_key, list_index);
|
||||
|
||||
// 02 - send downloaded file to every destination_display
|
||||
const temp_file = await file_handle.getFile();
|
||||
|
||||
for (const current_short_display of task.data.destination_display_data) {
|
||||
await upload_file_via_xhr(file_primary_key, task, temp_file, current_short_display.display);
|
||||
await upload_file_via_xhr(
|
||||
file_primary_key,
|
||||
list_index,
|
||||
task,
|
||||
temp_file,
|
||||
current_short_display.display
|
||||
);
|
||||
}
|
||||
|
||||
await dir.removeEntry(temp_name);
|
||||
@@ -292,7 +307,11 @@ export async function sync(file_primary_key: string, task: FileTransferTask) {
|
||||
// open file, if required
|
||||
if (task.data.open_file_afterwards_on_display_ids.length !== 0) {
|
||||
const path_to_file = task.path + task.file_name;
|
||||
await run_on_all_selected_displays((d) => open_file(d.ip, path_to_file), true, task.data.open_file_afterwards_on_display_ids);
|
||||
await run_on_all_selected_displays(
|
||||
(d) => open_file(d.ip, path_to_file),
|
||||
true,
|
||||
task.data.open_file_afterwards_on_display_ids
|
||||
);
|
||||
}
|
||||
} catch (e) {
|
||||
show_general_error(file_primary_key, task, String(e));
|
||||
@@ -336,24 +355,37 @@ async function start_task_loop() {
|
||||
while (Object.keys(get(file_transfer_tasks)).length > 0) {
|
||||
const tasks = get(file_transfer_tasks);
|
||||
const current_file_id = Object.keys(tasks)[0];
|
||||
const current_task = tasks[current_file_id];
|
||||
if (current_task.data.type === 'upload') {
|
||||
await upload(current_file_id, current_task);
|
||||
} else if (current_task.data.type === 'sync') {
|
||||
await sync(current_file_id, current_task);
|
||||
}
|
||||
const current_task_list = tasks[current_file_id];
|
||||
|
||||
file_transfer_tasks.update((all_tasks) => {
|
||||
const next = { ...all_tasks };
|
||||
delete next[current_file_id];
|
||||
return next;
|
||||
});
|
||||
for (const [list_index, current_task] of current_task_list.entries()) {
|
||||
if (current_task.data.type === 'upload') {
|
||||
await upload(current_file_id, current_task, list_index);
|
||||
} else if (current_task.data.type === 'sync') {
|
||||
await sync(current_file_id, current_task, list_index);
|
||||
}
|
||||
delete_current_task_if_needed(current_file_id);
|
||||
}
|
||||
}
|
||||
is_processing = false;
|
||||
}
|
||||
|
||||
function delete_current_task_if_needed(current_file_id: string) {
|
||||
file_transfer_tasks.update((all_tasks) => {
|
||||
const next = { ...all_tasks };
|
||||
const current_tasks = next[current_file_id];
|
||||
if (current_tasks.length !== 1) {
|
||||
if (current_tasks.find((t) => t.loading_data.percentage !== 100)) {
|
||||
return next; // not all tasks are finished -> do nothing
|
||||
}
|
||||
}
|
||||
delete next[current_file_id];
|
||||
return next;
|
||||
});
|
||||
}
|
||||
|
||||
async function upload_file_via_xhr(
|
||||
file_primary_key: string,
|
||||
list_index: number,
|
||||
task: FileTransferTask,
|
||||
current_file: File,
|
||||
destination_short_display: ShortDisplay | null = null
|
||||
@@ -373,6 +405,7 @@ async function upload_file_via_xhr(
|
||||
const apply = async () => {
|
||||
update_current_loading_data(
|
||||
file_primary_key,
|
||||
list_index,
|
||||
e.loaded,
|
||||
start_time,
|
||||
destination_short_display ? destination_short_display.id : null
|
||||
@@ -392,6 +425,7 @@ async function upload_file_via_xhr(
|
||||
// set loading_data to 100%
|
||||
finish_loading_data(
|
||||
file_primary_key,
|
||||
list_index,
|
||||
destination_short_display ? destination_short_display.id : null
|
||||
);
|
||||
// Generate Thumbnail if not done already
|
||||
@@ -416,38 +450,43 @@ async function upload_file_via_xhr(
|
||||
|
||||
function finish_loading_data(
|
||||
file_primary_key: string,
|
||||
list_index: number,
|
||||
destination_display_id: string | null = null
|
||||
) {
|
||||
file_transfer_tasks.update((tasks) => {
|
||||
const task = tasks[file_primary_key];
|
||||
const current_loading_data = {
|
||||
percentage: 100,
|
||||
bytes_per_second: 0,
|
||||
seconds_until_finish: 0
|
||||
};
|
||||
const new_task_list = tasks[file_primary_key].map((task, index) =>
|
||||
index === list_index
|
||||
? get_updated_task(current_loading_data, destination_display_id, task)
|
||||
: task
|
||||
);
|
||||
|
||||
return {
|
||||
...tasks,
|
||||
[file_primary_key]: get_updated_task(current_loading_data, destination_display_id, task)
|
||||
[file_primary_key]: new_task_list
|
||||
};
|
||||
});
|
||||
}
|
||||
|
||||
function update_current_loading_data(
|
||||
file_primary_key: string,
|
||||
list_index: number,
|
||||
current_bytes: number,
|
||||
start_time: Date,
|
||||
destination_display_id: string | null = null
|
||||
) {
|
||||
file_transfer_tasks.update((tasks) => {
|
||||
const task = tasks[file_primary_key];
|
||||
const task = tasks[file_primary_key][list_index];
|
||||
if (!task) return tasks;
|
||||
|
||||
const current_percentage = Math.min(
|
||||
task.bytes_total > 0 ? Math.round((current_bytes / task.bytes_total) * 100) : 1,
|
||||
99
|
||||
);
|
||||
|
||||
const prognosed_data = get_prognosed_data(start_time, current_bytes, task.bytes_total);
|
||||
|
||||
const current_loading_data: FileLoadingData = {
|
||||
@@ -456,9 +495,15 @@ function update_current_loading_data(
|
||||
seconds_until_finish: prognosed_data.seconds_until_finish
|
||||
};
|
||||
|
||||
const new_task_list = tasks[file_primary_key].map((task, index) =>
|
||||
index === list_index
|
||||
? get_updated_task(current_loading_data, destination_display_id, task)
|
||||
: task
|
||||
);
|
||||
|
||||
return {
|
||||
...tasks,
|
||||
[file_primary_key]: get_updated_task(current_loading_data, destination_display_id, task)
|
||||
[file_primary_key]: new_task_list
|
||||
};
|
||||
});
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user