2023-10-15 20:25:14 +02:00
|
|
|
import { create } from "zustand";
|
|
|
|
import { persist } from "zustand/middleware";
|
|
|
|
import { immer } from "zustand/middleware/immer";
|
|
|
|
|
|
|
|
import { PlayerMeta } from "@/stores/player/slices/source";
|
|
|
|
|
|
|
|
export interface ProgressItem {
|
|
|
|
watched: number;
|
|
|
|
duration: number;
|
|
|
|
}
|
|
|
|
|
|
|
|
export interface ProgressSeasonItem {
|
|
|
|
title: string;
|
|
|
|
number: number;
|
|
|
|
id: string;
|
|
|
|
}
|
|
|
|
|
|
|
|
export interface ProgressEpisodeItem {
|
|
|
|
title: string;
|
|
|
|
number: number;
|
|
|
|
id: string;
|
|
|
|
seasonId: string;
|
2023-10-30 21:29:02 +01:00
|
|
|
updatedAt: number;
|
2023-10-15 20:25:14 +02:00
|
|
|
progress: ProgressItem;
|
|
|
|
}
|
|
|
|
|
|
|
|
export interface ProgressMediaItem {
|
|
|
|
title: string;
|
2023-10-30 21:29:02 +01:00
|
|
|
year?: number;
|
2023-10-17 16:01:26 +02:00
|
|
|
poster?: string;
|
2023-10-15 20:25:14 +02:00
|
|
|
type: "show" | "movie";
|
|
|
|
progress?: ProgressItem;
|
2023-10-17 17:04:03 +02:00
|
|
|
updatedAt: number;
|
2023-10-15 20:25:14 +02:00
|
|
|
seasons: Record<string, ProgressSeasonItem>;
|
|
|
|
episodes: Record<string, ProgressEpisodeItem>;
|
|
|
|
}
|
|
|
|
|
2023-11-19 20:03:35 +01:00
|
|
|
export interface ProgressUpdateItem {
|
|
|
|
title?: string;
|
|
|
|
year?: number;
|
|
|
|
poster?: string;
|
|
|
|
type?: "show" | "movie";
|
|
|
|
progress?: ProgressItem;
|
|
|
|
tmdbId: string;
|
|
|
|
id: string;
|
|
|
|
episodeId?: string;
|
|
|
|
seasonId?: string;
|
2023-11-19 20:47:20 +01:00
|
|
|
episodeNumber?: number;
|
|
|
|
seasonNumber?: number;
|
2023-11-19 20:03:35 +01:00
|
|
|
action: "upsert" | "delete";
|
|
|
|
}
|
|
|
|
|
2023-10-15 20:25:14 +02:00
|
|
|
export interface UpdateItemOptions {
|
|
|
|
meta: PlayerMeta;
|
|
|
|
progress: ProgressItem;
|
|
|
|
}
|
|
|
|
|
|
|
|
export interface ProgressStore {
|
|
|
|
items: Record<string, ProgressMediaItem>;
|
2023-11-19 20:03:35 +01:00
|
|
|
updateQueue: ProgressUpdateItem[];
|
2023-10-15 20:25:14 +02:00
|
|
|
updateItem(ops: UpdateItemOptions): void;
|
2023-10-17 16:01:26 +02:00
|
|
|
removeItem(id: string): void;
|
2023-10-30 21:29:02 +01:00
|
|
|
replaceItems(items: Record<string, ProgressMediaItem>): void;
|
2023-11-05 17:56:56 +01:00
|
|
|
clear(): void;
|
2023-11-19 20:47:20 +01:00
|
|
|
clearUpdateQueue(): void;
|
|
|
|
removeUpdateItem(id: string): void;
|
2023-10-15 20:25:14 +02:00
|
|
|
}
|
|
|
|
|
2023-11-19 20:03:35 +01:00
|
|
|
let updateId = 0;
|
|
|
|
|
2023-10-15 20:25:14 +02:00
|
|
|
export const useProgressStore = create(
|
|
|
|
persist(
|
|
|
|
immer<ProgressStore>((set) => ({
|
|
|
|
items: {},
|
2023-11-19 20:03:35 +01:00
|
|
|
updateQueue: [],
|
2023-10-17 16:01:26 +02:00
|
|
|
removeItem(id) {
|
|
|
|
set((s) => {
|
2023-11-19 20:03:35 +01:00
|
|
|
updateId += 1;
|
|
|
|
s.updateQueue.push({
|
|
|
|
id: updateId.toString(),
|
|
|
|
action: "delete",
|
|
|
|
tmdbId: id,
|
|
|
|
});
|
|
|
|
|
2023-10-17 16:01:26 +02:00
|
|
|
delete s.items[id];
|
|
|
|
});
|
|
|
|
},
|
2023-10-30 21:29:02 +01:00
|
|
|
replaceItems(items: Record<string, ProgressMediaItem>) {
|
|
|
|
set((s) => {
|
|
|
|
s.items = items;
|
|
|
|
});
|
|
|
|
},
|
2023-10-15 20:25:14 +02:00
|
|
|
updateItem({ meta, progress }) {
|
|
|
|
set((s) => {
|
2023-11-19 20:03:35 +01:00
|
|
|
// add to updateQueue
|
|
|
|
updateId += 1;
|
|
|
|
s.updateQueue.push({
|
|
|
|
tmdbId: meta.tmdbId,
|
|
|
|
title: meta.title,
|
|
|
|
year: meta.releaseYear,
|
|
|
|
poster: meta.poster,
|
|
|
|
type: meta.type,
|
|
|
|
progress: { ...progress },
|
|
|
|
id: updateId.toString(),
|
|
|
|
episodeId: meta.episode?.tmdbId,
|
|
|
|
seasonId: meta.season?.tmdbId,
|
2023-11-19 20:47:20 +01:00
|
|
|
seasonNumber: meta.season?.number,
|
|
|
|
episodeNumber: meta.episode?.number,
|
2023-11-19 20:03:35 +01:00
|
|
|
action: "upsert",
|
|
|
|
});
|
|
|
|
|
|
|
|
// add to progress store
|
2023-10-15 20:25:14 +02:00
|
|
|
if (!s.items[meta.tmdbId])
|
|
|
|
s.items[meta.tmdbId] = {
|
|
|
|
type: meta.type,
|
|
|
|
episodes: {},
|
|
|
|
seasons: {},
|
2023-10-17 17:04:03 +02:00
|
|
|
updatedAt: 0,
|
2023-10-15 20:25:14 +02:00
|
|
|
title: meta.title,
|
|
|
|
year: meta.releaseYear,
|
2023-10-17 16:01:26 +02:00
|
|
|
poster: meta.poster,
|
2023-10-15 20:25:14 +02:00
|
|
|
};
|
|
|
|
const item = s.items[meta.tmdbId];
|
2023-10-17 17:04:03 +02:00
|
|
|
item.updatedAt = Date.now();
|
|
|
|
|
2023-10-15 20:25:14 +02:00
|
|
|
if (meta.type === "movie") {
|
|
|
|
if (!item.progress)
|
|
|
|
item.progress = {
|
|
|
|
duration: 0,
|
|
|
|
watched: 0,
|
|
|
|
};
|
|
|
|
item.progress = { ...progress };
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!meta.episode || !meta.season) return;
|
|
|
|
|
|
|
|
if (!item.seasons[meta.season.tmdbId])
|
|
|
|
item.seasons[meta.season.tmdbId] = {
|
|
|
|
id: meta.season.tmdbId,
|
|
|
|
number: meta.season.number,
|
|
|
|
title: meta.season.title,
|
|
|
|
};
|
|
|
|
|
|
|
|
if (!item.episodes[meta.episode.tmdbId])
|
|
|
|
item.episodes[meta.episode.tmdbId] = {
|
|
|
|
id: meta.episode.tmdbId,
|
|
|
|
number: meta.episode.number,
|
|
|
|
title: meta.episode.title,
|
|
|
|
seasonId: meta.season.tmdbId,
|
2023-10-30 21:29:02 +01:00
|
|
|
updatedAt: Date.now(),
|
2023-10-15 20:25:14 +02:00
|
|
|
progress: {
|
|
|
|
duration: 0,
|
|
|
|
watched: 0,
|
|
|
|
},
|
|
|
|
};
|
|
|
|
|
|
|
|
item.episodes[meta.episode.tmdbId].progress = { ...progress };
|
|
|
|
});
|
|
|
|
},
|
2023-11-05 17:56:56 +01:00
|
|
|
clear() {
|
2023-11-24 17:11:00 +01:00
|
|
|
set((s) => {
|
|
|
|
s.items = {};
|
|
|
|
});
|
2023-11-05 17:56:56 +01:00
|
|
|
},
|
2023-11-19 20:47:20 +01:00
|
|
|
clearUpdateQueue() {
|
|
|
|
set((s) => {
|
|
|
|
s.updateQueue = [];
|
|
|
|
});
|
|
|
|
},
|
|
|
|
removeUpdateItem(id: string) {
|
|
|
|
set((s) => {
|
|
|
|
s.updateQueue = [...s.updateQueue.filter((v) => v.id !== id)];
|
|
|
|
});
|
|
|
|
},
|
2023-10-15 20:25:14 +02:00
|
|
|
})),
|
|
|
|
{
|
|
|
|
name: "__MW::progress",
|
2023-12-23 06:24:43 +01:00
|
|
|
},
|
|
|
|
),
|
2023-10-15 20:25:14 +02:00
|
|
|
);
|