mirror of
https://github.com/hydralauncher/hydra.git
synced 2026-01-18 08:43:57 +00:00
Compare commits
38 Commits
v3.7.3
...
fix/fixing
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
ac01930d68 | ||
|
|
37caeb8047 | ||
|
|
7d6eddb17e | ||
|
|
48775e57fc | ||
|
|
fdc3fecd6f | ||
|
|
f0dc7478cf | ||
|
|
2e8da53d1a | ||
|
|
8794fbc742 | ||
|
|
bf387aef3f | ||
|
|
c2a26b9750 | ||
|
|
3dc2a29114 | ||
|
|
6ebf7766aa | ||
|
|
f6c12c22b5 | ||
|
|
539010d817 | ||
|
|
ef52d710ed | ||
|
|
31d57a784e | ||
|
|
2fce12eba7 | ||
|
|
1427775c98 | ||
|
|
5c770bc7e7 | ||
|
|
b431ed479c | ||
|
|
9e09a5decb | ||
|
|
1e1a1c61c9 | ||
|
|
8de6c92d28 | ||
|
|
29e1713824 | ||
|
|
81a77411cc | ||
|
|
cc95deb709 | ||
|
|
daf9751cf6 | ||
|
|
d21ec52814 | ||
|
|
f539977431 | ||
|
|
3ff20417d5 | ||
|
|
65f83399f5 | ||
|
|
eb34f051e1 | ||
|
|
ab27f3295e | ||
|
|
3782f79100 | ||
|
|
86ab5b107b | ||
|
|
acf8f340dd | ||
|
|
035f6e8d24 | ||
|
|
362d6b634e |
@@ -75,6 +75,7 @@
|
|||||||
"react-dnd-html5-backend": "^16.0.1",
|
"react-dnd-html5-backend": "^16.0.1",
|
||||||
"react-hook-form": "^7.53.0",
|
"react-hook-form": "^7.53.0",
|
||||||
"react-i18next": "^14.1.0",
|
"react-i18next": "^14.1.0",
|
||||||
|
"react-infinite-scroll-component": "^6.1.0",
|
||||||
"react-loading-skeleton": "^3.4.0",
|
"react-loading-skeleton": "^3.4.0",
|
||||||
"react-redux": "^9.1.1",
|
"react-redux": "^9.1.1",
|
||||||
"react-router-dom": "^6.22.3",
|
"react-router-dom": "^6.22.3",
|
||||||
|
|||||||
@@ -223,6 +223,7 @@
|
|||||||
"show_more": "Show more",
|
"show_more": "Show more",
|
||||||
"show_less": "Show less",
|
"show_less": "Show less",
|
||||||
"reviews": "Reviews",
|
"reviews": "Reviews",
|
||||||
|
"review_played_for": "Played for",
|
||||||
"leave_a_review": "Leave a Review",
|
"leave_a_review": "Leave a Review",
|
||||||
"write_review_placeholder": "Share your thoughts about this game...",
|
"write_review_placeholder": "Share your thoughts about this game...",
|
||||||
"sort_newest": "Newest",
|
"sort_newest": "Newest",
|
||||||
@@ -692,7 +693,10 @@
|
|||||||
"game_added_to_pinned": "Game added to pinned",
|
"game_added_to_pinned": "Game added to pinned",
|
||||||
"karma": "Karma",
|
"karma": "Karma",
|
||||||
"karma_count": "karma",
|
"karma_count": "karma",
|
||||||
"karma_description": "Earned from positive likes on reviews"
|
"karma_description": "Earned from positive likes on reviews",
|
||||||
|
"user_reviews": "Reviews",
|
||||||
|
"delete_review": "Delete Review",
|
||||||
|
"loading_reviews": "Loading reviews..."
|
||||||
},
|
},
|
||||||
"achievement": {
|
"achievement": {
|
||||||
"achievement_unlocked": "Achievement unlocked",
|
"achievement_unlocked": "Achievement unlocked",
|
||||||
|
|||||||
@@ -325,6 +325,7 @@
|
|||||||
"maybe_later": "Tal vez después",
|
"maybe_later": "Tal vez después",
|
||||||
"no_repacks_found": "Sin fuentes encontradas para este juego",
|
"no_repacks_found": "Sin fuentes encontradas para este juego",
|
||||||
"no_reviews_yet": "Sin reseñas aún",
|
"no_reviews_yet": "Sin reseñas aún",
|
||||||
|
"review_played_for": "Jugado por",
|
||||||
"properties": "Propiedades",
|
"properties": "Propiedades",
|
||||||
"rating": "Calificación",
|
"rating": "Calificación",
|
||||||
"rating_count": "Calificación",
|
"rating_count": "Calificación",
|
||||||
@@ -681,7 +682,11 @@
|
|||||||
"karma_count": "karma",
|
"karma_count": "karma",
|
||||||
"karma_description": "Conseguido por me gustas positivos en reseñas",
|
"karma_description": "Conseguido por me gustas positivos en reseñas",
|
||||||
"sort_by": "Filtrar por:",
|
"sort_by": "Filtrar por:",
|
||||||
"game_added_to_pinned": "Juego añadido a fijados"
|
"game_added_to_pinned": "Juego añadido a fijados",
|
||||||
|
"user_reviews": "Reseñas",
|
||||||
|
"loading_reviews": "Cargando reseñas...",
|
||||||
|
"no_reviews": "Sin reseñas aún",
|
||||||
|
"delete_review": "Eliminar reseña"
|
||||||
},
|
},
|
||||||
"achievement": {
|
"achievement": {
|
||||||
"achievement_unlocked": "Logro desbloqueado",
|
"achievement_unlocked": "Logro desbloqueado",
|
||||||
|
|||||||
@@ -317,6 +317,7 @@
|
|||||||
"sort_lowest_score": "Menor Nota",
|
"sort_lowest_score": "Menor Nota",
|
||||||
"sort_most_voted": "Mais Votadas",
|
"sort_most_voted": "Mais Votadas",
|
||||||
"no_reviews_yet": "Ainda não há avaliações",
|
"no_reviews_yet": "Ainda não há avaliações",
|
||||||
|
"review_played_for": "Jogado por",
|
||||||
"be_first_to_review": "Seja o primeiro a compartilhar seus pensamentos sobre este jogo!",
|
"be_first_to_review": "Seja o primeiro a compartilhar seus pensamentos sobre este jogo!",
|
||||||
"rating": "Avaliação",
|
"rating": "Avaliação",
|
||||||
"rating_stats": "Avaliação",
|
"rating_stats": "Avaliação",
|
||||||
@@ -696,7 +697,11 @@
|
|||||||
"karma": "Karma",
|
"karma": "Karma",
|
||||||
"karma_count": "karma",
|
"karma_count": "karma",
|
||||||
"karma_description": "Ganho a partir de curtidas positivas em avaliações",
|
"karma_description": "Ganho a partir de curtidas positivas em avaliações",
|
||||||
"manual_playtime_tooltip": "Este tempo de jogo foi atualizado manualmente"
|
"manual_playtime_tooltip": "Este tempo de jogo foi atualizado manualmente",
|
||||||
|
"user_reviews": "Avaliações",
|
||||||
|
"loading_reviews": "Carregando avaliações...",
|
||||||
|
"no_reviews": "Ainda não há avaliações",
|
||||||
|
"delete_review": "Excluir avaliação"
|
||||||
},
|
},
|
||||||
"achievement": {
|
"achievement": {
|
||||||
"achievement_unlocked": "Conquista desbloqueada",
|
"achievement_unlocked": "Conquista desbloqueada",
|
||||||
|
|||||||
@@ -183,7 +183,8 @@
|
|||||||
"create_start_menu_shortcut": "Criar atalho no Menu Iniciar",
|
"create_start_menu_shortcut": "Criar atalho no Menu Iniciar",
|
||||||
"review_from_blocked_user": "Avaliação de utilizador bloqueado",
|
"review_from_blocked_user": "Avaliação de utilizador bloqueado",
|
||||||
"show": "Mostrar",
|
"show": "Mostrar",
|
||||||
"hide": "Ocultar"
|
"hide": "Ocultar",
|
||||||
|
"review_played_for": "Jogado por"
|
||||||
},
|
},
|
||||||
"activation": {
|
"activation": {
|
||||||
"title": "Ativação",
|
"title": "Ativação",
|
||||||
@@ -469,7 +470,11 @@
|
|||||||
"achievements_unlocked": "Conquistas desbloqueadas",
|
"achievements_unlocked": "Conquistas desbloqueadas",
|
||||||
"earned_points": "Pontos ganhos",
|
"earned_points": "Pontos ganhos",
|
||||||
"show_achievements_on_profile": "Mostre as suas conquistas no perfil",
|
"show_achievements_on_profile": "Mostre as suas conquistas no perfil",
|
||||||
"show_points_on_profile": "Mostre os seus pontos ganhos no perfil"
|
"show_points_on_profile": "Mostre os seus pontos ganhos no perfil",
|
||||||
|
"user_reviews": "Avaliações",
|
||||||
|
"loading_reviews": "A carregar avaliações...",
|
||||||
|
"no_reviews": "Ainda não há avaliações",
|
||||||
|
"delete_review": "Eliminar avaliação"
|
||||||
},
|
},
|
||||||
"achievement": {
|
"achievement": {
|
||||||
"achievement_unlocked": "Conquista desbloqueada",
|
"achievement_unlocked": "Conquista desbloqueada",
|
||||||
|
|||||||
@@ -227,6 +227,7 @@
|
|||||||
"write_review_placeholder": "Поделитесь своими мыслями об этой игре...",
|
"write_review_placeholder": "Поделитесь своими мыслями об этой игре...",
|
||||||
"sort_newest": "Сначала новые",
|
"sort_newest": "Сначала новые",
|
||||||
"no_reviews_yet": "Пока нет отзывов",
|
"no_reviews_yet": "Пока нет отзывов",
|
||||||
|
"review_played_for": "Играли",
|
||||||
"be_first_to_review": "Станьте первым, кто поделится своими мыслями об этой игре!",
|
"be_first_to_review": "Станьте первым, кто поделится своими мыслями об этой игре!",
|
||||||
"sort_oldest": "Сначала старые",
|
"sort_oldest": "Сначала старые",
|
||||||
"sort_highest_score": "Высший балл",
|
"sort_highest_score": "Высший балл",
|
||||||
@@ -692,7 +693,11 @@
|
|||||||
"game_added_to_pinned": "Игра добавлена в закрепленные",
|
"game_added_to_pinned": "Игра добавлена в закрепленные",
|
||||||
"karma": "Карма",
|
"karma": "Карма",
|
||||||
"karma_count": "карма",
|
"karma_count": "карма",
|
||||||
"karma_description": "Заработана положительными оценками отзывов"
|
"karma_description": "Заработана положительными оценками отзывов",
|
||||||
|
"user_reviews": "Отзывы",
|
||||||
|
"loading_reviews": "Загрузка отзывов...",
|
||||||
|
"no_reviews": "Пока нет отзывов",
|
||||||
|
"delete_review": "Удалить отзыв"
|
||||||
},
|
},
|
||||||
"achievement": {
|
"achievement": {
|
||||||
"achievement_unlocked": "Достижение разблокировано",
|
"achievement_unlocked": "Достижение разблокировано",
|
||||||
|
|||||||
@@ -1,6 +1,7 @@
|
|||||||
import axios, { AxiosResponse } from "axios";
|
import axios, { AxiosResponse } from "axios";
|
||||||
import { wrapper } from "axios-cookiejar-support";
|
import { wrapper } from "axios-cookiejar-support";
|
||||||
import { CookieJar } from "tough-cookie";
|
import { CookieJar } from "tough-cookie";
|
||||||
|
import { logger } from "@main/services";
|
||||||
|
|
||||||
export class DatanodesApi {
|
export class DatanodesApi {
|
||||||
private static readonly jar = new CookieJar();
|
private static readonly jar = new CookieJar();
|
||||||
@@ -20,51 +21,42 @@ export class DatanodesApi {
|
|||||||
|
|
||||||
await this.jar.setCookie("lang=english;", "https://datanodes.to");
|
await this.jar.setCookie("lang=english;", "https://datanodes.to");
|
||||||
|
|
||||||
const payload = new URLSearchParams({
|
const formData = new FormData();
|
||||||
op: "download2",
|
formData.append("op", "download2");
|
||||||
id: fileCode,
|
formData.append("id", fileCode);
|
||||||
method_free: "Free Download >>",
|
formData.append("rand", "");
|
||||||
dl: "1",
|
formData.append("referer", "https://datanodes.to/download");
|
||||||
});
|
formData.append("method_free", "Free Download >>");
|
||||||
|
formData.append("method_premium", "");
|
||||||
|
formData.append("__dl", "1");
|
||||||
|
|
||||||
const response: AxiosResponse = await this.session.post(
|
const response: AxiosResponse = await this.session.post(
|
||||||
"https://datanodes.to/download",
|
"https://datanodes.to/download",
|
||||||
payload,
|
formData,
|
||||||
{
|
{
|
||||||
headers: {
|
headers: {
|
||||||
"User-Agent":
|
accept: "*/*",
|
||||||
"Mozilla/5.0 (Windows NT 10.0; Win64; x64; rv:135.0) Gecko/20100101 Firefox/135.0",
|
"accept-language": "en-US,en;q=0.9",
|
||||||
|
priority: "u=1, i",
|
||||||
|
"sec-ch-ua":
|
||||||
|
'"Google Chrome";v="141", "Not?A_Brand";v="8", "Chromium";v="141"',
|
||||||
|
"sec-ch-ua-mobile": "?0",
|
||||||
|
"sec-ch-ua-platform": '"Windows"',
|
||||||
|
"sec-fetch-dest": "empty",
|
||||||
|
"sec-fetch-mode": "cors",
|
||||||
|
"sec-fetch-site": "same-origin",
|
||||||
Referer: "https://datanodes.to/download",
|
Referer: "https://datanodes.to/download",
|
||||||
Origin: "https://datanodes.to",
|
|
||||||
"Content-Type": "application/x-www-form-urlencoded",
|
|
||||||
},
|
},
|
||||||
maxRedirects: 0,
|
|
||||||
validateStatus: (status: number) => status === 302 || status < 400,
|
|
||||||
}
|
}
|
||||||
);
|
);
|
||||||
|
|
||||||
if (response.status === 302) {
|
|
||||||
return response.headers["location"];
|
|
||||||
}
|
|
||||||
|
|
||||||
if (typeof response.data === "object" && response.data.url) {
|
if (typeof response.data === "object" && response.data.url) {
|
||||||
return decodeURIComponent(response.data.url);
|
return decodeURIComponent(response.data.url);
|
||||||
}
|
}
|
||||||
|
|
||||||
const htmlContent = String(response.data);
|
|
||||||
if (!htmlContent) {
|
|
||||||
throw new Error("Empty response received");
|
|
||||||
}
|
|
||||||
|
|
||||||
const downloadLinkRegex = /href=["'](https:\/\/[^"']+)["']/;
|
|
||||||
const downloadLinkMatch = downloadLinkRegex.exec(htmlContent);
|
|
||||||
if (downloadLinkMatch) {
|
|
||||||
return downloadLinkMatch[1];
|
|
||||||
}
|
|
||||||
|
|
||||||
throw new Error("Failed to get the download link");
|
throw new Error("Failed to get the download link");
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
console.error("Error fetching download URL:", error);
|
logger.error("Error fetching download URL:", error);
|
||||||
throw error;
|
throw error;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -279,7 +279,11 @@ export function App() {
|
|||||||
<article className="container">
|
<article className="container">
|
||||||
<Header />
|
<Header />
|
||||||
|
|
||||||
<section ref={contentRef} className="container__content">
|
<section
|
||||||
|
ref={contentRef}
|
||||||
|
id="scrollableDiv"
|
||||||
|
className="container__content"
|
||||||
|
>
|
||||||
<Outlet />
|
<Outlet />
|
||||||
</section>
|
</section>
|
||||||
</article>
|
</article>
|
||||||
|
|||||||
@@ -14,12 +14,15 @@ export interface UserProfileContext {
|
|||||||
isMe: boolean;
|
isMe: boolean;
|
||||||
userStats: UserStats | null;
|
userStats: UserStats | null;
|
||||||
getUserProfile: () => Promise<void>;
|
getUserProfile: () => Promise<void>;
|
||||||
getUserLibraryGames: (sortBy?: string) => Promise<void>;
|
getUserLibraryGames: (sortBy?: string, reset?: boolean) => Promise<void>;
|
||||||
|
loadMoreLibraryGames: (sortBy?: string) => Promise<boolean>;
|
||||||
setSelectedBackgroundImage: React.Dispatch<React.SetStateAction<string>>;
|
setSelectedBackgroundImage: React.Dispatch<React.SetStateAction<string>>;
|
||||||
backgroundImage: string;
|
backgroundImage: string;
|
||||||
badges: Badge[];
|
badges: Badge[];
|
||||||
libraryGames: UserGame[];
|
libraryGames: UserGame[];
|
||||||
pinnedGames: UserGame[];
|
pinnedGames: UserGame[];
|
||||||
|
hasMoreLibraryGames: boolean;
|
||||||
|
isLoadingLibraryGames: boolean;
|
||||||
}
|
}
|
||||||
|
|
||||||
export const DEFAULT_USER_PROFILE_BACKGROUND = "#151515B3";
|
export const DEFAULT_USER_PROFILE_BACKGROUND = "#151515B3";
|
||||||
@@ -30,12 +33,15 @@ export const userProfileContext = createContext<UserProfileContext>({
|
|||||||
isMe: false,
|
isMe: false,
|
||||||
userStats: null,
|
userStats: null,
|
||||||
getUserProfile: async () => {},
|
getUserProfile: async () => {},
|
||||||
getUserLibraryGames: async (_sortBy?: string) => {},
|
getUserLibraryGames: async (_sortBy?: string, _reset?: boolean) => {},
|
||||||
|
loadMoreLibraryGames: async (_sortBy?: string) => false,
|
||||||
setSelectedBackgroundImage: () => {},
|
setSelectedBackgroundImage: () => {},
|
||||||
backgroundImage: "",
|
backgroundImage: "",
|
||||||
badges: [],
|
badges: [],
|
||||||
libraryGames: [],
|
libraryGames: [],
|
||||||
pinnedGames: [],
|
pinnedGames: [],
|
||||||
|
hasMoreLibraryGames: false,
|
||||||
|
isLoadingLibraryGames: false,
|
||||||
});
|
});
|
||||||
|
|
||||||
const { Provider } = userProfileContext;
|
const { Provider } = userProfileContext;
|
||||||
@@ -62,6 +68,9 @@ export function UserProfileContextProvider({
|
|||||||
DEFAULT_USER_PROFILE_BACKGROUND
|
DEFAULT_USER_PROFILE_BACKGROUND
|
||||||
);
|
);
|
||||||
const [selectedBackgroundImage, setSelectedBackgroundImage] = useState("");
|
const [selectedBackgroundImage, setSelectedBackgroundImage] = useState("");
|
||||||
|
const [libraryPage, setLibraryPage] = useState(0);
|
||||||
|
const [hasMoreLibraryGames, setHasMoreLibraryGames] = useState(true);
|
||||||
|
const [isLoadingLibraryGames, setIsLoadingLibraryGames] = useState(false);
|
||||||
|
|
||||||
const isMe = userDetails?.id === userProfile?.id;
|
const isMe = userDetails?.id === userProfile?.id;
|
||||||
|
|
||||||
@@ -93,7 +102,13 @@ export function UserProfileContextProvider({
|
|||||||
}, [userId]);
|
}, [userId]);
|
||||||
|
|
||||||
const getUserLibraryGames = useCallback(
|
const getUserLibraryGames = useCallback(
|
||||||
async (sortBy?: string) => {
|
async (sortBy?: string, reset = true) => {
|
||||||
|
if (reset) {
|
||||||
|
setLibraryPage(0);
|
||||||
|
setHasMoreLibraryGames(true);
|
||||||
|
setIsLoadingLibraryGames(true);
|
||||||
|
}
|
||||||
|
|
||||||
try {
|
try {
|
||||||
const params = new URLSearchParams();
|
const params = new URLSearchParams();
|
||||||
params.append("take", "12");
|
params.append("take", "12");
|
||||||
@@ -115,18 +130,74 @@ export function UserProfileContextProvider({
|
|||||||
if (response) {
|
if (response) {
|
||||||
setLibraryGames(response.library);
|
setLibraryGames(response.library);
|
||||||
setPinnedGames(response.pinnedGames);
|
setPinnedGames(response.pinnedGames);
|
||||||
|
setHasMoreLibraryGames(response.library.length === 12);
|
||||||
} else {
|
} else {
|
||||||
setLibraryGames([]);
|
setLibraryGames([]);
|
||||||
setPinnedGames([]);
|
setPinnedGames([]);
|
||||||
|
setHasMoreLibraryGames(false);
|
||||||
}
|
}
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
setLibraryGames([]);
|
setLibraryGames([]);
|
||||||
setPinnedGames([]);
|
setPinnedGames([]);
|
||||||
|
setHasMoreLibraryGames(false);
|
||||||
|
} finally {
|
||||||
|
setIsLoadingLibraryGames(false);
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
[userId]
|
[userId]
|
||||||
);
|
);
|
||||||
|
|
||||||
|
const loadMoreLibraryGames = useCallback(
|
||||||
|
async (sortBy?: string): Promise<boolean> => {
|
||||||
|
if (isLoadingLibraryGames || !hasMoreLibraryGames) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
setIsLoadingLibraryGames(true);
|
||||||
|
try {
|
||||||
|
const nextPage = libraryPage + 1;
|
||||||
|
const params = new URLSearchParams();
|
||||||
|
params.append("take", "12");
|
||||||
|
params.append("skip", String(nextPage * 12));
|
||||||
|
if (sortBy) {
|
||||||
|
params.append("sortBy", sortBy);
|
||||||
|
}
|
||||||
|
|
||||||
|
const queryString = params.toString();
|
||||||
|
const url = queryString
|
||||||
|
? `/users/${userId}/library?${queryString}`
|
||||||
|
: `/users/${userId}/library`;
|
||||||
|
|
||||||
|
const response = await window.electron.hydraApi.get<{
|
||||||
|
library: UserGame[];
|
||||||
|
pinnedGames: UserGame[];
|
||||||
|
}>(url);
|
||||||
|
|
||||||
|
if (response && response.library.length > 0) {
|
||||||
|
setLibraryGames((prev) => {
|
||||||
|
const existingIds = new Set(prev.map((game) => game.objectId));
|
||||||
|
const newGames = response.library.filter(
|
||||||
|
(game) => !existingIds.has(game.objectId)
|
||||||
|
);
|
||||||
|
return [...prev, ...newGames];
|
||||||
|
});
|
||||||
|
setLibraryPage(nextPage);
|
||||||
|
setHasMoreLibraryGames(response.library.length === 12);
|
||||||
|
return true;
|
||||||
|
} else {
|
||||||
|
setHasMoreLibraryGames(false);
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
setHasMoreLibraryGames(false);
|
||||||
|
return false;
|
||||||
|
} finally {
|
||||||
|
setIsLoadingLibraryGames(false);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
[userId, libraryPage, hasMoreLibraryGames, isLoadingLibraryGames]
|
||||||
|
);
|
||||||
|
|
||||||
const getUserProfile = useCallback(async () => {
|
const getUserProfile = useCallback(async () => {
|
||||||
getUserStats();
|
getUserStats();
|
||||||
getUserLibraryGames();
|
getUserLibraryGames();
|
||||||
@@ -164,6 +235,8 @@ export function UserProfileContextProvider({
|
|||||||
setLibraryGames([]);
|
setLibraryGames([]);
|
||||||
setPinnedGames([]);
|
setPinnedGames([]);
|
||||||
setHeroBackground(DEFAULT_USER_PROFILE_BACKGROUND);
|
setHeroBackground(DEFAULT_USER_PROFILE_BACKGROUND);
|
||||||
|
setLibraryPage(0);
|
||||||
|
setHasMoreLibraryGames(true);
|
||||||
|
|
||||||
getUserProfile();
|
getUserProfile();
|
||||||
getBadges();
|
getBadges();
|
||||||
@@ -177,12 +250,15 @@ export function UserProfileContextProvider({
|
|||||||
isMe,
|
isMe,
|
||||||
getUserProfile,
|
getUserProfile,
|
||||||
getUserLibraryGames,
|
getUserLibraryGames,
|
||||||
|
loadMoreLibraryGames,
|
||||||
setSelectedBackgroundImage,
|
setSelectedBackgroundImage,
|
||||||
backgroundImage: getBackgroundImageUrl(),
|
backgroundImage: getBackgroundImageUrl(),
|
||||||
userStats,
|
userStats,
|
||||||
badges,
|
badges,
|
||||||
libraryGames,
|
libraryGames,
|
||||||
pinnedGames,
|
pinnedGames,
|
||||||
|
hasMoreLibraryGames,
|
||||||
|
isLoadingLibraryGames,
|
||||||
}}
|
}}
|
||||||
>
|
>
|
||||||
{children}
|
{children}
|
||||||
|
|||||||
@@ -3,12 +3,14 @@ import { useState, useCallback } from "react";
|
|||||||
interface SectionCollapseState {
|
interface SectionCollapseState {
|
||||||
pinned: boolean;
|
pinned: boolean;
|
||||||
library: boolean;
|
library: boolean;
|
||||||
|
reviews: boolean;
|
||||||
}
|
}
|
||||||
|
|
||||||
export function useSectionCollapse() {
|
export function useSectionCollapse() {
|
||||||
const [collapseState, setCollapseState] = useState<SectionCollapseState>({
|
const [collapseState, setCollapseState] = useState<SectionCollapseState>({
|
||||||
pinned: false,
|
pinned: false,
|
||||||
library: false,
|
library: false,
|
||||||
|
reviews: false,
|
||||||
});
|
});
|
||||||
|
|
||||||
const toggleSection = useCallback((section: keyof SectionCollapseState) => {
|
const toggleSection = useCallback((section: keyof SectionCollapseState) => {
|
||||||
@@ -23,5 +25,6 @@ export function useSectionCollapse() {
|
|||||||
toggleSection,
|
toggleSection,
|
||||||
isPinnedCollapsed: collapseState.pinned,
|
isPinnedCollapsed: collapseState.pinned,
|
||||||
isLibraryCollapsed: collapseState.library,
|
isLibraryCollapsed: collapseState.library,
|
||||||
|
isReviewsCollapsed: collapseState.reviews,
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -8,11 +8,23 @@
|
|||||||
|
|
||||||
&__review-header {
|
&__review-header {
|
||||||
display: flex;
|
display: flex;
|
||||||
justify-content: space-between;
|
flex-direction: column;
|
||||||
align-items: center;
|
gap: calc(globals.$spacing-unit * 1);
|
||||||
margin-bottom: calc(globals.$spacing-unit * 1.5);
|
margin-bottom: calc(globals.$spacing-unit * 1.5);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
&__review-header-top {
|
||||||
|
display: flex;
|
||||||
|
justify-content: space-between;
|
||||||
|
align-items: flex-start;
|
||||||
|
}
|
||||||
|
|
||||||
|
&__review-header-bottom {
|
||||||
|
display: flex;
|
||||||
|
justify-content: flex-start;
|
||||||
|
align-items: center;
|
||||||
|
}
|
||||||
|
|
||||||
&__review-user {
|
&__review-user {
|
||||||
display: flex;
|
display: flex;
|
||||||
align-items: center;
|
align-items: center;
|
||||||
@@ -22,7 +34,13 @@
|
|||||||
&__review-user-info {
|
&__review-user-info {
|
||||||
display: flex;
|
display: flex;
|
||||||
flex-direction: column;
|
flex-direction: column;
|
||||||
gap: calc(globals.$spacing-unit * 0.25);
|
gap: calc(globals.$spacing-unit * 0.45);
|
||||||
|
}
|
||||||
|
|
||||||
|
&__review-meta-row {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
gap: calc(globals.$spacing-unit * 0.75);
|
||||||
}
|
}
|
||||||
|
|
||||||
&__review-display-name {
|
&__review-display-name {
|
||||||
@@ -157,28 +175,28 @@
|
|||||||
&__review-score-stars {
|
&__review-score-stars {
|
||||||
display: flex;
|
display: flex;
|
||||||
align-items: center;
|
align-items: center;
|
||||||
gap: 2px;
|
gap: 4px;
|
||||||
|
background: rgba(255, 255, 255, 0.05);
|
||||||
|
border-radius: 8px;
|
||||||
|
padding: 2px 6px;
|
||||||
|
border: 1px solid rgba(255, 255, 255, 0.1);
|
||||||
|
font-size: 11px;
|
||||||
|
font-weight: 500;
|
||||||
|
}
|
||||||
|
|
||||||
|
&__review-right {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
align-items: flex-end;
|
||||||
}
|
}
|
||||||
|
|
||||||
&__review-star {
|
&__review-star {
|
||||||
color: #666666;
|
color: rgba(255, 255, 255, 0.7);
|
||||||
transition: color 0.2s ease;
|
transition: color 0.2s ease;
|
||||||
cursor: default;
|
cursor: default;
|
||||||
|
|
||||||
&--filled {
|
&--filled {
|
||||||
color: #ffffff;
|
color: rgba(255, 255, 255, 0.7);
|
||||||
|
|
||||||
&.game-details__review-score--red {
|
|
||||||
color: #fca5a5;
|
|
||||||
}
|
|
||||||
|
|
||||||
&.game-details__review-score--yellow {
|
|
||||||
color: #fcd34d;
|
|
||||||
}
|
|
||||||
|
|
||||||
&.game-details__review-score--green {
|
|
||||||
color: #86efac;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
&--empty {
|
&--empty {
|
||||||
@@ -198,6 +216,24 @@
|
|||||||
font-size: globals.$small-font-size;
|
font-size: globals.$small-font-size;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
&__review-playtime {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
gap: 4px;
|
||||||
|
color: rgba(255, 255, 255, 0.7);
|
||||||
|
font-size: 11px;
|
||||||
|
font-weight: 500;
|
||||||
|
background: rgba(255, 255, 255, 0.05);
|
||||||
|
border-radius: 8px;
|
||||||
|
padding: 2px 6px;
|
||||||
|
border: 1px solid rgba(255, 255, 255, 0.1);
|
||||||
|
margin-top: 0;
|
||||||
|
|
||||||
|
svg {
|
||||||
|
color: rgba(255, 255, 255, 0.6);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
&__review-content {
|
&__review-content {
|
||||||
color: globals.$body-color;
|
color: globals.$body-color;
|
||||||
line-height: 1.5;
|
line-height: 1.5;
|
||||||
|
|||||||
@@ -7,9 +7,10 @@ import { useState } from "react";
|
|||||||
import type { GameReview } from "@types";
|
import type { GameReview } from "@types";
|
||||||
|
|
||||||
import { sanitizeHtml } from "@shared";
|
import { sanitizeHtml } from "@shared";
|
||||||
import { useDate } from "@renderer/hooks";
|
import { useDate, useFormat } from "@renderer/hooks";
|
||||||
import { formatNumber } from "@renderer/helpers";
|
import { formatNumber } from "@renderer/helpers";
|
||||||
import { Avatar } from "@renderer/components";
|
import { Avatar } from "@renderer/components";
|
||||||
|
import { MAX_MINUTES_TO_SHOW_IN_PLAYTIME } from "@renderer/constants";
|
||||||
|
|
||||||
import "./review-item.scss";
|
import "./review-item.scss";
|
||||||
|
|
||||||
@@ -29,13 +30,6 @@ interface ReviewItemProps {
|
|||||||
) => void;
|
) => void;
|
||||||
}
|
}
|
||||||
|
|
||||||
const getScoreColorClass = (score: number): string => {
|
|
||||||
if (score >= 1 && score <= 2) return "game-details__review-score--red";
|
|
||||||
if (score >= 3 && score <= 3) return "game-details__review-score--yellow";
|
|
||||||
if (score >= 4 && score <= 5) return "game-details__review-score--green";
|
|
||||||
return "";
|
|
||||||
};
|
|
||||||
|
|
||||||
const getRatingText = (score: number, t: (key: string) => string): string => {
|
const getRatingText = (score: number, t: (key: string) => string): string => {
|
||||||
switch (score) {
|
switch (score) {
|
||||||
case 1:
|
case 1:
|
||||||
@@ -68,6 +62,7 @@ export function ReviewItem({
|
|||||||
const navigate = useNavigate();
|
const navigate = useNavigate();
|
||||||
const { t, i18n } = useTranslation("game_details");
|
const { t, i18n } = useTranslation("game_details");
|
||||||
const { formatDistance } = useDate();
|
const { formatDistance } = useDate();
|
||||||
|
const { numberFormatter } = useFormat();
|
||||||
|
|
||||||
const [showOriginal, setShowOriginal] = useState(false);
|
const [showOriginal, setShowOriginal] = useState(false);
|
||||||
|
|
||||||
@@ -93,6 +88,21 @@ export function ReviewItem({
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
// Format playtime similar to hero panel
|
||||||
|
const formatPlayTime = (playTimeInSeconds: number) => {
|
||||||
|
const minutes = playTimeInSeconds / 60;
|
||||||
|
|
||||||
|
if (minutes < MAX_MINUTES_TO_SHOW_IN_PLAYTIME) {
|
||||||
|
return t("amount_minutes", {
|
||||||
|
amount: minutes.toFixed(0),
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
const hours = minutes / 60;
|
||||||
|
return t("amount_hours", { amount: numberFormatter.format(hours) });
|
||||||
|
};
|
||||||
|
|
||||||
|
// Determine which content to show - always show original for own reviews
|
||||||
const displayContent = needsTranslation
|
const displayContent = needsTranslation
|
||||||
? review.translations[i18n.language]
|
? review.translations[i18n.language]
|
||||||
: review.reviewHtml;
|
: review.reviewHtml;
|
||||||
@@ -116,54 +126,61 @@ export function ReviewItem({
|
|||||||
return (
|
return (
|
||||||
<div className="game-details__review-item">
|
<div className="game-details__review-item">
|
||||||
<div className="game-details__review-header">
|
<div className="game-details__review-header">
|
||||||
<div className="game-details__review-user">
|
<div className="game-details__review-header-top">
|
||||||
<button
|
<div className="game-details__review-user">
|
||||||
onClick={() => navigate(`/profile/${review.user.id}`)}
|
|
||||||
title={review.user.displayName}
|
|
||||||
>
|
|
||||||
<Avatar
|
|
||||||
src={review.user.profileImageUrl}
|
|
||||||
alt={review.user.displayName || "User"}
|
|
||||||
size={40}
|
|
||||||
/>
|
|
||||||
</button>
|
|
||||||
<div className="game-details__review-user-info">
|
|
||||||
<button
|
<button
|
||||||
className="game-details__review-display-name game-details__review-display-name--clickable"
|
onClick={() => navigate(`/profile/${review.user.id}`)}
|
||||||
onClick={() =>
|
title={review.user.displayName}
|
||||||
review.user.id && navigate(`/profile/${review.user.id}`)
|
|
||||||
}
|
|
||||||
>
|
>
|
||||||
{review.user.displayName || "Anonymous"}
|
<Avatar
|
||||||
|
src={review.user.profileImageUrl}
|
||||||
|
alt={review.user.displayName || "User"}
|
||||||
|
size={40}
|
||||||
|
/>
|
||||||
</button>
|
</button>
|
||||||
<div className="game-details__review-date">
|
<div className="game-details__review-user-info">
|
||||||
<ClockIcon size={12} />
|
<button
|
||||||
{formatDistance(new Date(review.createdAt), new Date(), {
|
className="game-details__review-display-name game-details__review-display-name--clickable"
|
||||||
addSuffix: true,
|
onClick={() =>
|
||||||
})}
|
review.user.id && navigate(`/profile/${review.user.id}`)
|
||||||
|
}
|
||||||
|
>
|
||||||
|
{review.user.displayName || "Anonymous"}
|
||||||
|
</button>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
<div className="game-details__review-date">
|
||||||
|
{formatDistance(new Date(review.createdAt), new Date(), {
|
||||||
|
addSuffix: true,
|
||||||
|
})}
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div
|
<div className="game-details__review-header-bottom">
|
||||||
className="game-details__review-score-stars"
|
<div className="game-details__review-meta-row">
|
||||||
title={getRatingText(review.score, t)}
|
<div
|
||||||
>
|
className="game-details__review-score-stars"
|
||||||
{[1, 2, 3, 4, 5].map((starValue) => (
|
title={getRatingText(review.score, t)}
|
||||||
<Star
|
>
|
||||||
key={starValue}
|
<Star
|
||||||
size={20}
|
size={12}
|
||||||
fill={starValue <= review.score ? "currentColor" : "none"}
|
className="game-details__review-star game-details__review-star--filled"
|
||||||
className={`game-details__review-star ${
|
/>
|
||||||
starValue <= review.score
|
<span className="game-details__review-score-text">
|
||||||
? "game-details__review-star--filled"
|
{review.score}/5
|
||||||
: "game-details__review-star--empty"
|
</span>
|
||||||
} ${
|
</div>
|
||||||
starValue <= review.score
|
{Boolean(
|
||||||
? getScoreColorClass(review.score)
|
review.playTimeInSeconds && review.playTimeInSeconds > 0
|
||||||
: ""
|
) && (
|
||||||
}`}
|
<div className="game-details__review-playtime">
|
||||||
/>
|
<ClockIcon size={12} />
|
||||||
))}
|
<span>
|
||||||
|
{t("review_played_for")}{" "}
|
||||||
|
{formatPlayTime(review.playTimeInSeconds || 0)}
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div>
|
<div>
|
||||||
|
|||||||
178
src/renderer/src/pages/profile/profile-content/library-tab.tsx
Normal file
178
src/renderer/src/pages/profile/profile-content/library-tab.tsx
Normal file
@@ -0,0 +1,178 @@
|
|||||||
|
import { motion } from "framer-motion";
|
||||||
|
import { useTranslation } from "react-i18next";
|
||||||
|
import { TelescopeIcon } from "@primer/octicons-react";
|
||||||
|
import InfiniteScroll from "react-infinite-scroll-component";
|
||||||
|
import { useFormat } from "@renderer/hooks";
|
||||||
|
import type { UserGame } from "@types";
|
||||||
|
import { SortOptions } from "./sort-options";
|
||||||
|
import { UserLibraryGameCard } from "./user-library-game-card";
|
||||||
|
import "./profile-content.scss";
|
||||||
|
|
||||||
|
type SortOption = "playtime" | "achievementCount" | "playedRecently";
|
||||||
|
|
||||||
|
interface LibraryTabProps {
|
||||||
|
sortBy: SortOption;
|
||||||
|
onSortChange: (sortBy: SortOption) => void;
|
||||||
|
pinnedGames: UserGame[];
|
||||||
|
libraryGames: UserGame[];
|
||||||
|
hasMoreLibraryGames: boolean;
|
||||||
|
isLoadingLibraryGames: boolean;
|
||||||
|
statsIndex: number;
|
||||||
|
userStats: { libraryCount: number } | null;
|
||||||
|
animatedGameIdsRef: React.MutableRefObject<Set<string>>;
|
||||||
|
onLoadMore: () => void;
|
||||||
|
onMouseEnter: () => void;
|
||||||
|
onMouseLeave: () => void;
|
||||||
|
isMe: boolean;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function LibraryTab({
|
||||||
|
sortBy,
|
||||||
|
onSortChange,
|
||||||
|
pinnedGames,
|
||||||
|
libraryGames,
|
||||||
|
hasMoreLibraryGames,
|
||||||
|
isLoadingLibraryGames,
|
||||||
|
statsIndex,
|
||||||
|
userStats,
|
||||||
|
animatedGameIdsRef,
|
||||||
|
onLoadMore,
|
||||||
|
onMouseEnter,
|
||||||
|
onMouseLeave,
|
||||||
|
isMe,
|
||||||
|
}: Readonly<LibraryTabProps>) {
|
||||||
|
const { t } = useTranslation("user_profile");
|
||||||
|
const { numberFormatter } = useFormat();
|
||||||
|
|
||||||
|
const hasGames = libraryGames.length > 0;
|
||||||
|
const hasPinnedGames = pinnedGames.length > 0;
|
||||||
|
const hasAnyGames = hasGames || hasPinnedGames;
|
||||||
|
|
||||||
|
return (
|
||||||
|
<motion.div
|
||||||
|
key="library"
|
||||||
|
className="profile-content__tab-panel"
|
||||||
|
initial={{ opacity: 0, x: -10 }}
|
||||||
|
animate={{ opacity: 1, x: 0 }}
|
||||||
|
exit={{ opacity: 0, x: 10 }}
|
||||||
|
transition={{ duration: 0.2 }}
|
||||||
|
aria-hidden={false}
|
||||||
|
>
|
||||||
|
{hasAnyGames && (
|
||||||
|
<SortOptions sortBy={sortBy} onSortChange={onSortChange} />
|
||||||
|
)}
|
||||||
|
|
||||||
|
{!hasAnyGames && (
|
||||||
|
<div className="profile-content__no-games">
|
||||||
|
<div className="profile-content__telescope-icon">
|
||||||
|
<TelescopeIcon size={24} />
|
||||||
|
</div>
|
||||||
|
<h2>{t("no_recent_activity_title")}</h2>
|
||||||
|
{isMe && <p>{t("no_recent_activity_description")}</p>}
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
|
||||||
|
{hasAnyGames && (
|
||||||
|
<div>
|
||||||
|
{hasPinnedGames && (
|
||||||
|
<div style={{ marginBottom: "2rem" }}>
|
||||||
|
<div className="profile-content__section-header">
|
||||||
|
<div className="profile-content__section-title-group">
|
||||||
|
<h2>{t("pinned")}</h2>
|
||||||
|
<span className="profile-content__section-badge">
|
||||||
|
{pinnedGames.length}
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<ul className="profile-content__games-grid">
|
||||||
|
{pinnedGames?.map((game) => (
|
||||||
|
<li key={game.objectId} style={{ listStyle: "none" }}>
|
||||||
|
<UserLibraryGameCard
|
||||||
|
game={game}
|
||||||
|
statIndex={statsIndex}
|
||||||
|
onMouseEnter={onMouseEnter}
|
||||||
|
onMouseLeave={onMouseLeave}
|
||||||
|
sortBy={sortBy}
|
||||||
|
/>
|
||||||
|
</li>
|
||||||
|
))}
|
||||||
|
</ul>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
|
||||||
|
{hasGames && (
|
||||||
|
<div>
|
||||||
|
<div className="profile-content__section-header">
|
||||||
|
<div className="profile-content__section-title-group">
|
||||||
|
<h2>{t("library")}</h2>
|
||||||
|
{userStats && (
|
||||||
|
<span className="profile-content__section-badge">
|
||||||
|
{numberFormatter.format(userStats.libraryCount)}
|
||||||
|
</span>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<InfiniteScroll
|
||||||
|
dataLength={libraryGames.length}
|
||||||
|
next={onLoadMore}
|
||||||
|
hasMore={hasMoreLibraryGames}
|
||||||
|
loader={null}
|
||||||
|
scrollThreshold={0.9}
|
||||||
|
style={{ overflow: "visible" }}
|
||||||
|
scrollableTarget="scrollableDiv"
|
||||||
|
>
|
||||||
|
<ul className="profile-content__games-grid">
|
||||||
|
{libraryGames?.map((game, index) => {
|
||||||
|
const hasAnimated = animatedGameIdsRef.current.has(
|
||||||
|
game.objectId
|
||||||
|
);
|
||||||
|
const isNewGame = !hasAnimated && !isLoadingLibraryGames;
|
||||||
|
|
||||||
|
return (
|
||||||
|
<motion.li
|
||||||
|
key={`${sortBy}-${game.objectId}`}
|
||||||
|
style={{ listStyle: "none" }}
|
||||||
|
initial={
|
||||||
|
isNewGame
|
||||||
|
? { opacity: 0.5, y: 15, scale: 0.96 }
|
||||||
|
: false
|
||||||
|
}
|
||||||
|
animate={
|
||||||
|
isNewGame ? { opacity: 1, y: 0, scale: 1 } : false
|
||||||
|
}
|
||||||
|
transition={
|
||||||
|
isNewGame
|
||||||
|
? {
|
||||||
|
duration: 0.15,
|
||||||
|
ease: "easeOut",
|
||||||
|
delay: index * 0.01,
|
||||||
|
}
|
||||||
|
: undefined
|
||||||
|
}
|
||||||
|
onAnimationComplete={() => {
|
||||||
|
if (isNewGame) {
|
||||||
|
animatedGameIdsRef.current.add(game.objectId);
|
||||||
|
}
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<UserLibraryGameCard
|
||||||
|
game={game}
|
||||||
|
statIndex={statsIndex}
|
||||||
|
onMouseEnter={onMouseEnter}
|
||||||
|
onMouseLeave={onMouseLeave}
|
||||||
|
sortBy={sortBy}
|
||||||
|
/>
|
||||||
|
</motion.li>
|
||||||
|
);
|
||||||
|
})}
|
||||||
|
</ul>
|
||||||
|
</InfiniteScroll>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
</motion.div>
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -101,6 +101,11 @@
|
|||||||
gap: calc(globals.$spacing-unit);
|
gap: calc(globals.$spacing-unit);
|
||||||
margin-bottom: calc(globals.$spacing-unit * 2);
|
margin-bottom: calc(globals.$spacing-unit * 2);
|
||||||
border-bottom: 1px solid rgba(255, 255, 255, 0.1);
|
border-bottom: 1px solid rgba(255, 255, 255, 0.1);
|
||||||
|
position: relative;
|
||||||
|
}
|
||||||
|
|
||||||
|
&__tab-wrapper {
|
||||||
|
position: relative;
|
||||||
}
|
}
|
||||||
|
|
||||||
&__tab {
|
&__tab {
|
||||||
@@ -111,19 +116,40 @@
|
|||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
font-size: 14px;
|
font-size: 14px;
|
||||||
font-weight: 500;
|
font-weight: 500;
|
||||||
border-bottom: 2px solid transparent;
|
transition: color ease 0.2s;
|
||||||
transition: all ease 0.2s;
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
&:hover {
|
gap: calc(globals.$spacing-unit * 0.5);
|
||||||
color: rgba(255, 255, 255, 0.8);
|
|
||||||
}
|
|
||||||
|
|
||||||
&--active {
|
&--active {
|
||||||
color: white;
|
color: white;
|
||||||
border-bottom-color: #c9aa71;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
&__tab-badge {
|
||||||
|
display: inline-flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
min-width: 18px;
|
||||||
|
height: 18px;
|
||||||
|
padding: 0 6px;
|
||||||
|
background-color: rgba(255, 255, 255, 0.15);
|
||||||
|
border-radius: 9px;
|
||||||
|
font-size: 11px;
|
||||||
|
font-weight: 600;
|
||||||
|
color: rgba(255, 255, 255, 0.9);
|
||||||
|
line-height: 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
&__tab-underline {
|
||||||
|
position: absolute;
|
||||||
|
bottom: -1px;
|
||||||
|
left: 0;
|
||||||
|
right: 0;
|
||||||
|
height: 2px;
|
||||||
|
background: white;
|
||||||
|
}
|
||||||
|
|
||||||
&__games-grid {
|
&__games-grid {
|
||||||
list-style: none;
|
list-style: none;
|
||||||
margin: 0;
|
margin: 0;
|
||||||
@@ -175,5 +201,245 @@
|
|||||||
backdrop-filter: blur(10px);
|
backdrop-filter: blur(10px);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
&__tab-panels {
|
||||||
|
display: block;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Reviews minimal styles
|
||||||
|
.user-reviews__loading {
|
||||||
|
padding: calc(globals.$spacing-unit * 2);
|
||||||
|
color: rgba(255, 255, 255, 0.8);
|
||||||
|
text-align: center;
|
||||||
|
display: flex;
|
||||||
|
justify-content: center;
|
||||||
|
align-items: center;
|
||||||
|
}
|
||||||
|
|
||||||
|
.user-reviews__empty {
|
||||||
|
text-align: center;
|
||||||
|
padding: calc(globals.$spacing-unit * 4) calc(globals.$spacing-unit * 2);
|
||||||
|
color: rgba(255, 255, 255, 0.6);
|
||||||
|
}
|
||||||
|
|
||||||
|
.user-reviews__list {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
gap: calc(globals.$spacing-unit * 4);
|
||||||
|
}
|
||||||
|
|
||||||
|
.user-reviews__review-item {
|
||||||
|
border-radius: 8px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.user-reviews__review-header {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
gap: calc(globals.$spacing-unit * 1);
|
||||||
|
margin-bottom: calc(globals.$spacing-unit * 1.5);
|
||||||
|
}
|
||||||
|
|
||||||
|
.user-reviews__review-header-top {
|
||||||
|
display: flex;
|
||||||
|
justify-content: space-between;
|
||||||
|
align-items: flex-start;
|
||||||
|
}
|
||||||
|
|
||||||
|
.user-reviews__review-header-bottom {
|
||||||
|
display: flex;
|
||||||
|
justify-content: space-between;
|
||||||
|
align-items: center;
|
||||||
|
}
|
||||||
|
|
||||||
|
.user-reviews__review-meta-row {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
gap: calc(globals.$spacing-unit * 0.75);
|
||||||
|
}
|
||||||
|
|
||||||
|
.user-reviews__review-footer {
|
||||||
|
display: flex;
|
||||||
|
justify-content: space-between;
|
||||||
|
align-items: center;
|
||||||
|
margin-top: calc(globals.$spacing-unit * 1.5);
|
||||||
|
margin-bottom: calc(globals.$spacing-unit * 1.5);
|
||||||
|
}
|
||||||
|
|
||||||
|
.user-reviews__review-game {
|
||||||
|
display: flex;
|
||||||
|
gap: calc(globals.$spacing-unit);
|
||||||
|
}
|
||||||
|
|
||||||
|
.user-reviews__game-icon {
|
||||||
|
width: 24px;
|
||||||
|
height: 24px;
|
||||||
|
object-fit: cover;
|
||||||
|
}
|
||||||
|
|
||||||
|
.user-reviews__game-info {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
}
|
||||||
|
|
||||||
|
.user-reviews__game-details {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
gap: calc(globals.$spacing-unit * 0.75);
|
||||||
|
}
|
||||||
|
|
||||||
|
.user-reviews__game-title {
|
||||||
|
background: none;
|
||||||
|
border: none;
|
||||||
|
color: rgba(255, 255, 255, 0.9);
|
||||||
|
font-weight: 600;
|
||||||
|
cursor: pointer;
|
||||||
|
text-align: left;
|
||||||
|
|
||||||
|
&--clickable:hover {
|
||||||
|
text-decoration: underline;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.user-reviews__review-date {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
gap: 4px;
|
||||||
|
color: rgba(255, 255, 255, 0.6);
|
||||||
|
font-size: globals.$small-font-size;
|
||||||
|
}
|
||||||
|
|
||||||
|
.user-reviews__review-score-stars {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
gap: 4px;
|
||||||
|
background: rgba(255, 255, 255, 0.05);
|
||||||
|
border-radius: 8px;
|
||||||
|
padding: 2px 6px;
|
||||||
|
border: 1px solid rgba(255, 255, 255, 0.1);
|
||||||
|
font-size: 11px;
|
||||||
|
font-weight: 500;
|
||||||
|
}
|
||||||
|
|
||||||
|
.user-reviews__review-star {
|
||||||
|
color: rgba(255, 255, 255, 0.7);
|
||||||
|
transition: color 0.2s ease;
|
||||||
|
|
||||||
|
&--filled {
|
||||||
|
color: rgba(255, 255, 255, 0.7);
|
||||||
|
|
||||||
|
svg {
|
||||||
|
fill: currentColor;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.user-reviews__review-score-text {
|
||||||
|
font-weight: 500;
|
||||||
|
}
|
||||||
|
|
||||||
|
.user-reviews__review-playtime {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
gap: 4px;
|
||||||
|
color: rgba(255, 255, 255, 0.7);
|
||||||
|
font-size: 11px;
|
||||||
|
font-weight: 500;
|
||||||
|
background: rgba(255, 255, 255, 0.05);
|
||||||
|
border-radius: 8px;
|
||||||
|
padding: 2px 6px;
|
||||||
|
border: 1px solid rgba(255, 255, 255, 0.1);
|
||||||
|
}
|
||||||
|
|
||||||
|
.user-reviews__review-content {
|
||||||
|
color: rgba(255, 255, 255, 0.85);
|
||||||
|
line-height: 1.5;
|
||||||
|
word-wrap: break-word;
|
||||||
|
word-break: break-word;
|
||||||
|
overflow-wrap: break-word;
|
||||||
|
white-space: pre-wrap;
|
||||||
|
max-width: 100%;
|
||||||
|
}
|
||||||
|
|
||||||
|
.user-reviews__review-translation-toggle {
|
||||||
|
display: inline-flex;
|
||||||
|
align-items: center;
|
||||||
|
gap: calc(globals.$spacing-unit * 1);
|
||||||
|
margin-top: calc(globals.$spacing-unit * 1.5);
|
||||||
|
padding: 0;
|
||||||
|
background: none;
|
||||||
|
border: none;
|
||||||
|
color: rgba(255, 255, 255, 0.6);
|
||||||
|
font-size: 0.875rem;
|
||||||
|
cursor: pointer;
|
||||||
|
text-decoration: none;
|
||||||
|
transition: all 0.2s ease;
|
||||||
|
|
||||||
|
&:hover {
|
||||||
|
text-decoration: underline;
|
||||||
|
color: rgba(255, 255, 255, 0.9);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.user-reviews__review-actions {
|
||||||
|
margin-top: calc(globals.$spacing-unit * 2);
|
||||||
|
padding-top: calc(globals.$spacing-unit);
|
||||||
|
border-top: 1px solid rgba(255, 255, 255, 0.1);
|
||||||
|
display: flex;
|
||||||
|
justify-content: space-between;
|
||||||
|
align-items: center;
|
||||||
|
}
|
||||||
|
|
||||||
|
.user-reviews__review-votes {
|
||||||
|
display: flex;
|
||||||
|
gap: calc(globals.$spacing-unit);
|
||||||
|
}
|
||||||
|
|
||||||
|
.user-reviews__vote-button {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
gap: 6px;
|
||||||
|
background: rgba(255, 255, 255, 0.05);
|
||||||
|
border: 1px solid rgba(255, 255, 255, 0.1);
|
||||||
|
border-radius: 6px;
|
||||||
|
padding: 6px 12px;
|
||||||
|
color: #ccc;
|
||||||
|
font-size: 14px;
|
||||||
|
cursor: pointer;
|
||||||
|
transition: all 0.2s ease;
|
||||||
|
|
||||||
|
&:hover {
|
||||||
|
background: rgba(255, 255, 255, 0.1);
|
||||||
|
border-color: rgba(255, 255, 255, 0.2);
|
||||||
|
color: #ffffff;
|
||||||
|
}
|
||||||
|
|
||||||
|
&--active {
|
||||||
|
color: #ffffff;
|
||||||
|
border-color: rgba(255, 255, 255, 0.3);
|
||||||
|
|
||||||
|
svg {
|
||||||
|
fill: white;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.user-reviews__delete-review-button {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
gap: 6px;
|
||||||
|
background: rgba(244, 67, 54, 0.1);
|
||||||
|
border: 1px solid rgba(244, 67, 54, 0.3);
|
||||||
|
border-radius: 6px;
|
||||||
|
padding: 6px 10px;
|
||||||
|
color: #f44336;
|
||||||
|
cursor: pointer;
|
||||||
|
transition: all 0.2s ease;
|
||||||
|
|
||||||
|
&:hover {
|
||||||
|
background: rgba(244, 67, 54, 0.2);
|
||||||
|
border-color: rgba(244, 67, 54, 0.4);
|
||||||
|
color: #ff7961;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,29 +1,82 @@
|
|||||||
import { userProfileContext } from "@renderer/context";
|
import { userProfileContext } from "@renderer/context";
|
||||||
import { useContext, useEffect, useMemo, useRef, useState } from "react";
|
import {
|
||||||
|
useCallback,
|
||||||
|
useContext,
|
||||||
|
useEffect,
|
||||||
|
useMemo,
|
||||||
|
useRef,
|
||||||
|
useState,
|
||||||
|
} from "react";
|
||||||
import { ProfileHero } from "../profile-hero/profile-hero";
|
import { ProfileHero } from "../profile-hero/profile-hero";
|
||||||
import { useAppDispatch, useFormat } from "@renderer/hooks";
|
import { useAppDispatch, useFormat, useUserDetails } from "@renderer/hooks";
|
||||||
import { setHeaderTitle } from "@renderer/features";
|
import { setHeaderTitle } from "@renderer/features";
|
||||||
import { TelescopeIcon, ChevronRightIcon } from "@primer/octicons-react";
|
|
||||||
import { useTranslation } from "react-i18next";
|
import { useTranslation } from "react-i18next";
|
||||||
|
import type { GameShop } from "@types";
|
||||||
import { LockedProfile } from "./locked-profile";
|
import { LockedProfile } from "./locked-profile";
|
||||||
import { ReportProfile } from "../report-profile/report-profile";
|
import { ReportProfile } from "../report-profile/report-profile";
|
||||||
import { FriendsBox } from "./friends-box";
|
import { FriendsBox } from "./friends-box";
|
||||||
import { RecentGamesBox } from "./recent-games-box";
|
import { RecentGamesBox } from "./recent-games-box";
|
||||||
import { UserStatsBox } from "./user-stats-box";
|
import { UserStatsBox } from "./user-stats-box";
|
||||||
import { UserKarmaBox } from "./user-karma-box";
|
import { UserKarmaBox } from "./user-karma-box";
|
||||||
import { UserLibraryGameCard } from "./user-library-game-card";
|
import { DeleteReviewModal } from "@renderer/pages/game-details/modals/delete-review-modal";
|
||||||
import { SortOptions } from "./sort-options";
|
import { GAME_STATS_ANIMATION_DURATION_IN_MS } from "./profile-animations";
|
||||||
import { useSectionCollapse } from "@renderer/hooks/use-section-collapse";
|
import { MAX_MINUTES_TO_SHOW_IN_PLAYTIME } from "@renderer/constants";
|
||||||
import { motion, AnimatePresence } from "framer-motion";
|
import { ProfileTabs } from "./profile-tabs";
|
||||||
import {
|
import { LibraryTab } from "./library-tab";
|
||||||
sectionVariants,
|
import { ReviewsTab } from "./reviews-tab";
|
||||||
chevronVariants,
|
import { AnimatePresence } from "framer-motion";
|
||||||
GAME_STATS_ANIMATION_DURATION_IN_MS,
|
|
||||||
} from "./profile-animations";
|
|
||||||
import "./profile-content.scss";
|
import "./profile-content.scss";
|
||||||
|
|
||||||
type SortOption = "playtime" | "achievementCount" | "playedRecently";
|
type SortOption = "playtime" | "achievementCount" | "playedRecently";
|
||||||
|
|
||||||
|
interface UserReview {
|
||||||
|
id: string;
|
||||||
|
reviewHtml: string;
|
||||||
|
score: number;
|
||||||
|
playTimeInSeconds?: number;
|
||||||
|
upvotes: number;
|
||||||
|
downvotes: number;
|
||||||
|
hasUpvoted: boolean;
|
||||||
|
hasDownvoted: boolean;
|
||||||
|
createdAt: string;
|
||||||
|
updatedAt: string;
|
||||||
|
user: {
|
||||||
|
id: string;
|
||||||
|
};
|
||||||
|
game: {
|
||||||
|
title: string;
|
||||||
|
iconUrl: string;
|
||||||
|
objectId: string;
|
||||||
|
shop: GameShop;
|
||||||
|
};
|
||||||
|
translations: {
|
||||||
|
[key: string]: string;
|
||||||
|
};
|
||||||
|
detectedLanguage: string | null;
|
||||||
|
}
|
||||||
|
|
||||||
|
interface UserReviewsResponse {
|
||||||
|
totalCount: number;
|
||||||
|
reviews: UserReview[];
|
||||||
|
}
|
||||||
|
|
||||||
|
const getRatingText = (score: number, t: (key: string) => string): string => {
|
||||||
|
switch (score) {
|
||||||
|
case 1:
|
||||||
|
return t("rating_very_negative");
|
||||||
|
case 2:
|
||||||
|
return t("rating_negative");
|
||||||
|
case 3:
|
||||||
|
return t("rating_neutral");
|
||||||
|
case 4:
|
||||||
|
return t("rating_positive");
|
||||||
|
case 5:
|
||||||
|
return t("rating_very_positive");
|
||||||
|
default:
|
||||||
|
return "";
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
export function ProfileContent() {
|
export function ProfileContent() {
|
||||||
const {
|
const {
|
||||||
userProfile,
|
userProfile,
|
||||||
@@ -32,16 +85,43 @@ export function ProfileContent() {
|
|||||||
libraryGames,
|
libraryGames,
|
||||||
pinnedGames,
|
pinnedGames,
|
||||||
getUserLibraryGames,
|
getUserLibraryGames,
|
||||||
|
loadMoreLibraryGames,
|
||||||
|
hasMoreLibraryGames,
|
||||||
|
isLoadingLibraryGames,
|
||||||
} = useContext(userProfileContext);
|
} = useContext(userProfileContext);
|
||||||
|
const { userDetails } = useUserDetails();
|
||||||
const [statsIndex, setStatsIndex] = useState(0);
|
const [statsIndex, setStatsIndex] = useState(0);
|
||||||
const [isAnimationRunning, setIsAnimationRunning] = useState(true);
|
const [isAnimationRunning, setIsAnimationRunning] = useState(true);
|
||||||
const [sortBy, setSortBy] = useState<SortOption>("playedRecently");
|
const [sortBy, setSortBy] = useState<SortOption>("playedRecently");
|
||||||
const statsAnimation = useRef(-1);
|
const statsAnimation = useRef(-1);
|
||||||
const { toggleSection, isPinnedCollapsed } = useSectionCollapse();
|
|
||||||
|
const [activeTab, setActiveTab] = useState<"library" | "reviews">("library");
|
||||||
|
|
||||||
|
// User reviews state
|
||||||
|
const [reviews, setReviews] = useState<UserReview[]>([]);
|
||||||
|
const [reviewsTotalCount, setReviewsTotalCount] = useState(0);
|
||||||
|
const [isLoadingReviews, setIsLoadingReviews] = useState(false);
|
||||||
|
const [votingReviews, setVotingReviews] = useState<Set<string>>(new Set());
|
||||||
|
const [deleteModalVisible, setDeleteModalVisible] = useState(false);
|
||||||
|
const [reviewToDelete, setReviewToDelete] = useState<string | null>(null);
|
||||||
|
|
||||||
const dispatch = useAppDispatch();
|
const dispatch = useAppDispatch();
|
||||||
|
|
||||||
const { t } = useTranslation("user_profile");
|
const { t } = useTranslation("user_profile");
|
||||||
|
const { numberFormatter } = useFormat();
|
||||||
|
|
||||||
|
const formatPlayTime = (playTimeInSeconds: number) => {
|
||||||
|
const minutes = playTimeInSeconds / 60;
|
||||||
|
|
||||||
|
if (minutes < MAX_MINUTES_TO_SHOW_IN_PLAYTIME) {
|
||||||
|
return t("amount_minutes", {
|
||||||
|
amount: minutes.toFixed(0),
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
const hours = minutes / 60;
|
||||||
|
return t("amount_hours", { amount: numberFormatter.format(hours) });
|
||||||
|
};
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
dispatch(setHeaderTitle(""));
|
dispatch(setHeaderTitle(""));
|
||||||
@@ -53,10 +133,201 @@ export function ProfileContent() {
|
|||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
if (userProfile) {
|
if (userProfile) {
|
||||||
getUserLibraryGames(sortBy);
|
// When sortBy changes, clear animated games so all games animate in
|
||||||
|
if (currentSortByRef.current !== sortBy) {
|
||||||
|
animatedGameIdsRef.current.clear();
|
||||||
|
currentSortByRef.current = sortBy;
|
||||||
|
}
|
||||||
|
getUserLibraryGames(sortBy, true);
|
||||||
}
|
}
|
||||||
}, [sortBy, getUserLibraryGames, userProfile]);
|
}, [sortBy, getUserLibraryGames, userProfile]);
|
||||||
|
|
||||||
|
const animatedGameIdsRef = useRef<Set<string>>(new Set());
|
||||||
|
const currentSortByRef = useRef<SortOption>(sortBy);
|
||||||
|
|
||||||
|
const handleLoadMore = useCallback(() => {
|
||||||
|
if (
|
||||||
|
activeTab === "library" &&
|
||||||
|
hasMoreLibraryGames &&
|
||||||
|
!isLoadingLibraryGames
|
||||||
|
) {
|
||||||
|
loadMoreLibraryGames(sortBy);
|
||||||
|
}
|
||||||
|
}, [
|
||||||
|
activeTab,
|
||||||
|
hasMoreLibraryGames,
|
||||||
|
isLoadingLibraryGames,
|
||||||
|
loadMoreLibraryGames,
|
||||||
|
sortBy,
|
||||||
|
]);
|
||||||
|
|
||||||
|
// Clear reviews state and reset tab when switching users
|
||||||
|
useEffect(() => {
|
||||||
|
setReviews([]);
|
||||||
|
setReviewsTotalCount(0);
|
||||||
|
setIsLoadingReviews(false);
|
||||||
|
setActiveTab("library");
|
||||||
|
}, [userProfile?.id]);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
if (userProfile?.id) {
|
||||||
|
fetchUserReviews();
|
||||||
|
}
|
||||||
|
}, [userProfile?.id]);
|
||||||
|
|
||||||
|
const fetchUserReviews = async () => {
|
||||||
|
if (!userProfile?.id) return;
|
||||||
|
|
||||||
|
setIsLoadingReviews(true);
|
||||||
|
try {
|
||||||
|
const response = await window.electron.hydraApi.get<UserReviewsResponse>(
|
||||||
|
`/users/${userProfile.id}/reviews`,
|
||||||
|
{ needsAuth: true }
|
||||||
|
);
|
||||||
|
setReviews(response.reviews);
|
||||||
|
setReviewsTotalCount(response.totalCount);
|
||||||
|
} catch (error) {
|
||||||
|
// Error handling for fetching reviews
|
||||||
|
} finally {
|
||||||
|
setIsLoadingReviews(false);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const handleDeleteReview = async (reviewId: string) => {
|
||||||
|
try {
|
||||||
|
const reviewToDeleteObj = reviews.find(
|
||||||
|
(review) => review.id === reviewId
|
||||||
|
);
|
||||||
|
if (!reviewToDeleteObj) return;
|
||||||
|
|
||||||
|
await window.electron.hydraApi.delete(
|
||||||
|
`/games/${reviewToDeleteObj.game.shop}/${reviewToDeleteObj.game.objectId}/reviews/${reviewId}`
|
||||||
|
);
|
||||||
|
// Remove the review from the local state
|
||||||
|
setReviews((prev) => prev.filter((review) => review.id !== reviewId));
|
||||||
|
setReviewsTotalCount((prev) => prev - 1);
|
||||||
|
} catch (error) {
|
||||||
|
console.error("Failed to delete review:", error);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const handleDeleteClick = (reviewId: string) => {
|
||||||
|
setReviewToDelete(reviewId);
|
||||||
|
setDeleteModalVisible(true);
|
||||||
|
};
|
||||||
|
|
||||||
|
const handleDeleteConfirm = () => {
|
||||||
|
if (reviewToDelete) {
|
||||||
|
handleDeleteReview(reviewToDelete);
|
||||||
|
setReviewToDelete(null);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const handleDeleteCancel = () => {
|
||||||
|
setDeleteModalVisible(false);
|
||||||
|
setReviewToDelete(null);
|
||||||
|
};
|
||||||
|
|
||||||
|
const handleVoteReview = async (reviewId: string, isUpvote: boolean) => {
|
||||||
|
if (votingReviews.has(reviewId)) return;
|
||||||
|
|
||||||
|
setVotingReviews((prev) => new Set(prev).add(reviewId));
|
||||||
|
|
||||||
|
const review = reviews.find((r) => r.id === reviewId);
|
||||||
|
if (!review) {
|
||||||
|
setVotingReviews((prev) => {
|
||||||
|
const next = new Set(prev);
|
||||||
|
next.delete(reviewId);
|
||||||
|
return next;
|
||||||
|
});
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const wasUpvoted = review.hasUpvoted;
|
||||||
|
const wasDownvoted = review.hasDownvoted;
|
||||||
|
|
||||||
|
// Optimistic update
|
||||||
|
setReviews((prev) =>
|
||||||
|
prev.map((r) => {
|
||||||
|
if (r.id !== reviewId) return r;
|
||||||
|
|
||||||
|
let newUpvotes = r.upvotes;
|
||||||
|
let newDownvotes = r.downvotes;
|
||||||
|
let newHasUpvoted = r.hasUpvoted;
|
||||||
|
let newHasDownvoted = r.hasDownvoted;
|
||||||
|
|
||||||
|
if (isUpvote) {
|
||||||
|
if (wasUpvoted) {
|
||||||
|
// Remove upvote
|
||||||
|
newUpvotes--;
|
||||||
|
newHasUpvoted = false;
|
||||||
|
} else {
|
||||||
|
// Add upvote
|
||||||
|
newUpvotes++;
|
||||||
|
newHasUpvoted = true;
|
||||||
|
if (wasDownvoted) {
|
||||||
|
// Remove downvote if it was downvoted
|
||||||
|
newDownvotes--;
|
||||||
|
newHasDownvoted = false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else if (wasDownvoted) {
|
||||||
|
// Remove downvote
|
||||||
|
newDownvotes--;
|
||||||
|
newHasDownvoted = false;
|
||||||
|
} else {
|
||||||
|
// Add downvote
|
||||||
|
newDownvotes++;
|
||||||
|
newHasDownvoted = true;
|
||||||
|
if (wasUpvoted) {
|
||||||
|
// Remove upvote if it was upvoted
|
||||||
|
newUpvotes--;
|
||||||
|
newHasUpvoted = false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return {
|
||||||
|
...r,
|
||||||
|
upvotes: newUpvotes,
|
||||||
|
downvotes: newDownvotes,
|
||||||
|
hasUpvoted: newHasUpvoted,
|
||||||
|
hasDownvoted: newHasDownvoted,
|
||||||
|
};
|
||||||
|
})
|
||||||
|
);
|
||||||
|
|
||||||
|
try {
|
||||||
|
const endpoint = isUpvote ? "upvote" : "downvote";
|
||||||
|
await window.electron.hydraApi.put(
|
||||||
|
`/games/${review.game.shop}/${review.game.objectId}/reviews/${reviewId}/${endpoint}`
|
||||||
|
);
|
||||||
|
} catch (error) {
|
||||||
|
console.error("Failed to vote on review:", error);
|
||||||
|
|
||||||
|
// Rollback optimistic update on error
|
||||||
|
setReviews((prev) =>
|
||||||
|
prev.map((r) => {
|
||||||
|
if (r.id !== reviewId) return r;
|
||||||
|
return {
|
||||||
|
...r,
|
||||||
|
upvotes: review.upvotes,
|
||||||
|
downvotes: review.downvotes,
|
||||||
|
hasUpvoted: review.hasUpvoted,
|
||||||
|
hasDownvoted: review.hasDownvoted,
|
||||||
|
};
|
||||||
|
})
|
||||||
|
);
|
||||||
|
} finally {
|
||||||
|
setTimeout(() => {
|
||||||
|
setVotingReviews((prev) => {
|
||||||
|
const newSet = new Set(prev);
|
||||||
|
newSet.delete(reviewId);
|
||||||
|
return newSet;
|
||||||
|
});
|
||||||
|
}, 500);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
const handleOnMouseEnterGameCard = () => {
|
const handleOnMouseEnterGameCard = () => {
|
||||||
setIsAnimationRunning(false);
|
setIsAnimationRunning(false);
|
||||||
};
|
};
|
||||||
@@ -86,8 +357,6 @@ export function ProfileContent() {
|
|||||||
};
|
};
|
||||||
}, [setStatsIndex, isAnimationRunning]);
|
}, [setStatsIndex, isAnimationRunning]);
|
||||||
|
|
||||||
const { numberFormatter } = useFormat();
|
|
||||||
|
|
||||||
const usersAreFriends = useMemo(() => {
|
const usersAreFriends = useMemo(() => {
|
||||||
return userProfile?.relation?.status === "ACCEPTED";
|
return userProfile?.relation?.status === "ACCEPTED";
|
||||||
}, [userProfile]);
|
}, [userProfile]);
|
||||||
@@ -113,112 +382,46 @@ export function ProfileContent() {
|
|||||||
return (
|
return (
|
||||||
<section className="profile-content__section">
|
<section className="profile-content__section">
|
||||||
<div className="profile-content__main">
|
<div className="profile-content__main">
|
||||||
{hasAnyGames && (
|
<ProfileTabs
|
||||||
<SortOptions sortBy={sortBy} onSortChange={setSortBy} />
|
activeTab={activeTab}
|
||||||
)}
|
reviewsTotalCount={reviewsTotalCount}
|
||||||
|
onTabChange={setActiveTab}
|
||||||
|
/>
|
||||||
|
|
||||||
{!hasAnyGames && (
|
<div className="profile-content__tab-panels">
|
||||||
<div className="profile-content__no-games">
|
<AnimatePresence mode="wait">
|
||||||
<div className="profile-content__telescope-icon">
|
{activeTab === "library" && (
|
||||||
<TelescopeIcon size={24} />
|
<LibraryTab
|
||||||
</div>
|
sortBy={sortBy}
|
||||||
<h2>{t("no_recent_activity_title")}</h2>
|
onSortChange={setSortBy}
|
||||||
{isMe && <p>{t("no_recent_activity_description")}</p>}
|
pinnedGames={pinnedGames}
|
||||||
</div>
|
libraryGames={libraryGames}
|
||||||
)}
|
hasMoreLibraryGames={hasMoreLibraryGames}
|
||||||
|
isLoadingLibraryGames={isLoadingLibraryGames}
|
||||||
{hasAnyGames && (
|
statsIndex={statsIndex}
|
||||||
<div>
|
userStats={userStats}
|
||||||
{hasPinnedGames && (
|
animatedGameIdsRef={animatedGameIdsRef}
|
||||||
<div style={{ marginBottom: "2rem" }}>
|
onLoadMore={handleLoadMore}
|
||||||
<div className="profile-content__section-header">
|
onMouseEnter={handleOnMouseEnterGameCard}
|
||||||
<div className="profile-content__section-title-group">
|
onMouseLeave={handleOnMouseLeaveGameCard}
|
||||||
<button
|
isMe={isMe}
|
||||||
type="button"
|
/>
|
||||||
className="profile-content__collapse-button"
|
|
||||||
onClick={() => toggleSection("pinned")}
|
|
||||||
aria-label={
|
|
||||||
isPinnedCollapsed
|
|
||||||
? "Expand pinned section"
|
|
||||||
: "Collapse pinned section"
|
|
||||||
}
|
|
||||||
>
|
|
||||||
<motion.div
|
|
||||||
variants={chevronVariants}
|
|
||||||
animate={isPinnedCollapsed ? "collapsed" : "expanded"}
|
|
||||||
>
|
|
||||||
<ChevronRightIcon size={16} />
|
|
||||||
</motion.div>
|
|
||||||
</button>
|
|
||||||
<h2>{t("pinned")}</h2>
|
|
||||||
<span className="profile-content__section-badge">
|
|
||||||
{pinnedGames.length}
|
|
||||||
</span>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<AnimatePresence initial={true} mode="wait">
|
|
||||||
{!isPinnedCollapsed && (
|
|
||||||
<motion.div
|
|
||||||
key="pinned-content"
|
|
||||||
variants={sectionVariants}
|
|
||||||
initial="collapsed"
|
|
||||||
animate="expanded"
|
|
||||||
exit="collapsed"
|
|
||||||
layout
|
|
||||||
>
|
|
||||||
<ul className="profile-content__games-grid">
|
|
||||||
{pinnedGames?.map((game) => (
|
|
||||||
<li
|
|
||||||
key={game.objectId}
|
|
||||||
style={{ listStyle: "none" }}
|
|
||||||
>
|
|
||||||
<UserLibraryGameCard
|
|
||||||
game={game}
|
|
||||||
statIndex={statsIndex}
|
|
||||||
onMouseEnter={handleOnMouseEnterGameCard}
|
|
||||||
onMouseLeave={handleOnMouseLeaveGameCard}
|
|
||||||
sortBy={sortBy}
|
|
||||||
/>
|
|
||||||
</li>
|
|
||||||
))}
|
|
||||||
</ul>
|
|
||||||
</motion.div>
|
|
||||||
)}
|
|
||||||
</AnimatePresence>
|
|
||||||
</div>
|
|
||||||
)}
|
)}
|
||||||
|
|
||||||
{hasGames && (
|
{activeTab === "reviews" && (
|
||||||
<div>
|
<ReviewsTab
|
||||||
<div className="profile-content__section-header">
|
reviews={reviews}
|
||||||
<div className="profile-content__section-title-group">
|
isLoadingReviews={isLoadingReviews}
|
||||||
<h2>{t("library")}</h2>
|
votingReviews={votingReviews}
|
||||||
{userStats && (
|
userDetailsId={userDetails?.id}
|
||||||
<span className="profile-content__section-badge">
|
formatPlayTime={formatPlayTime}
|
||||||
{numberFormatter.format(userStats.libraryCount)}
|
getRatingText={getRatingText}
|
||||||
</span>
|
onVote={handleVoteReview}
|
||||||
)}
|
onDelete={handleDeleteClick}
|
||||||
</div>
|
/>
|
||||||
</div>
|
|
||||||
|
|
||||||
<ul className="profile-content__games-grid">
|
|
||||||
{libraryGames?.map((game) => (
|
|
||||||
<li key={game.objectId} style={{ listStyle: "none" }}>
|
|
||||||
<UserLibraryGameCard
|
|
||||||
game={game}
|
|
||||||
statIndex={statsIndex}
|
|
||||||
onMouseEnter={handleOnMouseEnterGameCard}
|
|
||||||
onMouseLeave={handleOnMouseLeaveGameCard}
|
|
||||||
sortBy={sortBy}
|
|
||||||
/>
|
|
||||||
</li>
|
|
||||||
))}
|
|
||||||
</ul>
|
|
||||||
</div>
|
|
||||||
)}
|
)}
|
||||||
</div>
|
</AnimatePresence>
|
||||||
)}
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
{shouldShowRightContent && (
|
{shouldShowRightContent && (
|
||||||
@@ -230,6 +433,12 @@ export function ProfileContent() {
|
|||||||
<ReportProfile />
|
<ReportProfile />
|
||||||
</div>
|
</div>
|
||||||
)}
|
)}
|
||||||
|
|
||||||
|
<DeleteReviewModal
|
||||||
|
visible={deleteModalVisible}
|
||||||
|
onClose={handleDeleteCancel}
|
||||||
|
onConfirm={handleDeleteConfirm}
|
||||||
|
/>
|
||||||
</section>
|
</section>
|
||||||
);
|
);
|
||||||
}, [
|
}, [
|
||||||
@@ -242,9 +451,15 @@ export function ProfileContent() {
|
|||||||
statsIndex,
|
statsIndex,
|
||||||
libraryGames,
|
libraryGames,
|
||||||
pinnedGames,
|
pinnedGames,
|
||||||
isPinnedCollapsed,
|
|
||||||
toggleSection,
|
|
||||||
sortBy,
|
sortBy,
|
||||||
|
activeTab,
|
||||||
|
// ensure reviews UI updates correctly
|
||||||
|
reviews,
|
||||||
|
reviewsTotalCount,
|
||||||
|
isLoadingReviews,
|
||||||
|
votingReviews,
|
||||||
|
deleteModalVisible,
|
||||||
]);
|
]);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
|
|||||||
@@ -0,0 +1,252 @@
|
|||||||
|
import { motion, AnimatePresence } from "framer-motion";
|
||||||
|
import { useNavigate } from "react-router-dom";
|
||||||
|
import { ClockIcon } from "@primer/octicons-react";
|
||||||
|
import { Star, ThumbsUp, ThumbsDown, TrashIcon, Languages } from "lucide-react";
|
||||||
|
import { useTranslation } from "react-i18next";
|
||||||
|
import { useState } from "react";
|
||||||
|
import type { GameShop } from "@types";
|
||||||
|
import { sanitizeHtml } from "@shared";
|
||||||
|
import { useDate } from "@renderer/hooks";
|
||||||
|
import { buildGameDetailsPath } from "@renderer/helpers";
|
||||||
|
import "./profile-content.scss";
|
||||||
|
|
||||||
|
interface UserReview {
|
||||||
|
id: string;
|
||||||
|
reviewHtml: string;
|
||||||
|
score: number;
|
||||||
|
playTimeInSeconds?: number;
|
||||||
|
upvotes: number;
|
||||||
|
downvotes: number;
|
||||||
|
hasUpvoted: boolean;
|
||||||
|
hasDownvoted: boolean;
|
||||||
|
createdAt: string;
|
||||||
|
updatedAt: string;
|
||||||
|
user: {
|
||||||
|
id: string;
|
||||||
|
};
|
||||||
|
game: {
|
||||||
|
title: string;
|
||||||
|
iconUrl: string;
|
||||||
|
objectId: string;
|
||||||
|
shop: GameShop;
|
||||||
|
};
|
||||||
|
translations: {
|
||||||
|
[key: string]: string;
|
||||||
|
};
|
||||||
|
detectedLanguage: string | null;
|
||||||
|
}
|
||||||
|
|
||||||
|
interface ProfileReviewItemProps {
|
||||||
|
review: UserReview;
|
||||||
|
isOwnReview: boolean;
|
||||||
|
isVoting: boolean;
|
||||||
|
formatPlayTime: (playTimeInSeconds: number) => string;
|
||||||
|
getRatingText: (score: number, t: (key: string) => string) => string;
|
||||||
|
onVote: (reviewId: string, isUpvote: boolean) => void;
|
||||||
|
onDelete: (reviewId: string) => void;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function ProfileReviewItem({
|
||||||
|
review,
|
||||||
|
isOwnReview,
|
||||||
|
isVoting,
|
||||||
|
formatPlayTime,
|
||||||
|
getRatingText,
|
||||||
|
onVote,
|
||||||
|
onDelete,
|
||||||
|
}: Readonly<ProfileReviewItemProps>) {
|
||||||
|
const navigate = useNavigate();
|
||||||
|
const { formatDistance } = useDate();
|
||||||
|
const { t } = useTranslation("user_profile");
|
||||||
|
const { t: tGameDetails, i18n } = useTranslation("game_details");
|
||||||
|
const [showOriginal, setShowOriginal] = useState(false);
|
||||||
|
|
||||||
|
const getBaseLanguage = (lang: string | null) => lang?.split("-")[0] || "";
|
||||||
|
|
||||||
|
const isDifferentLanguage =
|
||||||
|
getBaseLanguage(review.detectedLanguage) !== getBaseLanguage(i18n.language);
|
||||||
|
|
||||||
|
const needsTranslation =
|
||||||
|
!isOwnReview && isDifferentLanguage && review.translations[i18n.language];
|
||||||
|
|
||||||
|
const getLanguageName = (languageCode: string | null) => {
|
||||||
|
if (!languageCode) return "";
|
||||||
|
try {
|
||||||
|
const displayNames = new Intl.DisplayNames([i18n.language], {
|
||||||
|
type: "language",
|
||||||
|
});
|
||||||
|
return displayNames.of(languageCode) || languageCode.toUpperCase();
|
||||||
|
} catch {
|
||||||
|
return languageCode.toUpperCase();
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const displayContent = needsTranslation
|
||||||
|
? review.translations[i18n.language]
|
||||||
|
: review.reviewHtml;
|
||||||
|
|
||||||
|
return (
|
||||||
|
<motion.div
|
||||||
|
key={review.id}
|
||||||
|
className="user-reviews__review-item"
|
||||||
|
initial={{ opacity: 0, y: 20 }}
|
||||||
|
animate={{ opacity: 1, y: 0 }}
|
||||||
|
transition={{ duration: 0.3 }}
|
||||||
|
>
|
||||||
|
<div className="user-reviews__review-header">
|
||||||
|
<div className="user-reviews__review-header-top">
|
||||||
|
<div className="user-reviews__review-game">
|
||||||
|
<div className="user-reviews__game-info">
|
||||||
|
<div className="user-reviews__game-details">
|
||||||
|
<img
|
||||||
|
src={review.game.iconUrl}
|
||||||
|
alt={review.game.title}
|
||||||
|
className="user-reviews__game-icon"
|
||||||
|
/>
|
||||||
|
<button
|
||||||
|
className="user-reviews__game-title user-reviews__game-title--clickable"
|
||||||
|
onClick={() => navigate(buildGameDetailsPath(review.game))}
|
||||||
|
>
|
||||||
|
{review.game.title}
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div className="user-reviews__review-date">
|
||||||
|
{formatDistance(new Date(review.createdAt), new Date(), {
|
||||||
|
addSuffix: true,
|
||||||
|
})}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div className="user-reviews__review-header-bottom">
|
||||||
|
<div className="user-reviews__review-meta-row">
|
||||||
|
<div
|
||||||
|
className="user-reviews__review-score-stars"
|
||||||
|
title={getRatingText(review.score, tGameDetails)}
|
||||||
|
>
|
||||||
|
<Star
|
||||||
|
size={12}
|
||||||
|
className="user-reviews__review-star user-reviews__review-star--filled"
|
||||||
|
/>
|
||||||
|
<span className="user-reviews__review-score-text">
|
||||||
|
{review.score}/5
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
{Boolean(
|
||||||
|
review.playTimeInSeconds && review.playTimeInSeconds > 0
|
||||||
|
) && (
|
||||||
|
<div className="user-reviews__review-playtime">
|
||||||
|
<ClockIcon size={12} />
|
||||||
|
<span>
|
||||||
|
{tGameDetails("review_played_for")}{" "}
|
||||||
|
{formatPlayTime(review.playTimeInSeconds || 0)}
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div>
|
||||||
|
<div
|
||||||
|
className="user-reviews__review-content"
|
||||||
|
dangerouslySetInnerHTML={{
|
||||||
|
__html: sanitizeHtml(displayContent),
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
{needsTranslation && (
|
||||||
|
<>
|
||||||
|
<button
|
||||||
|
className="user-reviews__review-translation-toggle"
|
||||||
|
onClick={() => setShowOriginal(!showOriginal)}
|
||||||
|
>
|
||||||
|
<Languages size={13} />
|
||||||
|
{showOriginal
|
||||||
|
? tGameDetails("hide_original")
|
||||||
|
: tGameDetails("show_original_translated_from", {
|
||||||
|
language: getLanguageName(review.detectedLanguage),
|
||||||
|
})}
|
||||||
|
</button>
|
||||||
|
{showOriginal && (
|
||||||
|
<div
|
||||||
|
className="user-reviews__review-content"
|
||||||
|
style={{
|
||||||
|
opacity: 0.6,
|
||||||
|
marginTop: "12px",
|
||||||
|
}}
|
||||||
|
dangerouslySetInnerHTML={{
|
||||||
|
__html: sanitizeHtml(review.reviewHtml),
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
</>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div className="user-reviews__review-actions">
|
||||||
|
<div className="user-reviews__review-votes">
|
||||||
|
<motion.button
|
||||||
|
className={`user-reviews__vote-button ${review.hasUpvoted ? "user-reviews__vote-button--active" : ""}`}
|
||||||
|
onClick={() => onVote(review.id, true)}
|
||||||
|
disabled={isVoting}
|
||||||
|
style={{
|
||||||
|
opacity: isVoting ? 0.5 : 1,
|
||||||
|
cursor: isVoting ? "not-allowed" : "pointer",
|
||||||
|
}}
|
||||||
|
whileHover={{ scale: 1.05 }}
|
||||||
|
whileTap={{ scale: 0.95 }}
|
||||||
|
>
|
||||||
|
<ThumbsUp size={14} />
|
||||||
|
<AnimatePresence mode="wait">
|
||||||
|
<motion.span
|
||||||
|
key={review.upvotes}
|
||||||
|
initial={{ opacity: 0, y: -10 }}
|
||||||
|
animate={{ opacity: 1, y: 0 }}
|
||||||
|
exit={{ opacity: 0, y: 10 }}
|
||||||
|
transition={{ duration: 0.2 }}
|
||||||
|
>
|
||||||
|
{review.upvotes}
|
||||||
|
</motion.span>
|
||||||
|
</AnimatePresence>
|
||||||
|
</motion.button>
|
||||||
|
|
||||||
|
<motion.button
|
||||||
|
className={`user-reviews__vote-button ${review.hasDownvoted ? "user-reviews__vote-button--active" : ""}`}
|
||||||
|
onClick={() => onVote(review.id, false)}
|
||||||
|
disabled={isVoting}
|
||||||
|
style={{
|
||||||
|
opacity: isVoting ? 0.5 : 1,
|
||||||
|
cursor: isVoting ? "not-allowed" : "pointer",
|
||||||
|
}}
|
||||||
|
whileHover={{ scale: 1.05 }}
|
||||||
|
whileTap={{ scale: 0.95 }}
|
||||||
|
>
|
||||||
|
<ThumbsDown size={14} />
|
||||||
|
<AnimatePresence mode="wait">
|
||||||
|
<motion.span
|
||||||
|
key={review.downvotes}
|
||||||
|
initial={{ opacity: 0, y: -10 }}
|
||||||
|
animate={{ opacity: 1, y: 0 }}
|
||||||
|
exit={{ opacity: 0, y: 10 }}
|
||||||
|
transition={{ duration: 0.2 }}
|
||||||
|
>
|
||||||
|
{review.downvotes}
|
||||||
|
</motion.span>
|
||||||
|
</AnimatePresence>
|
||||||
|
</motion.button>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{isOwnReview && (
|
||||||
|
<button
|
||||||
|
className="user-reviews__delete-review-button"
|
||||||
|
onClick={() => onDelete(review.id)}
|
||||||
|
title={t("delete_review")}
|
||||||
|
>
|
||||||
|
<TrashIcon size={14} />
|
||||||
|
<span>{t("delete_review")}</span>
|
||||||
|
</button>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
</motion.div>
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -0,0 +1,67 @@
|
|||||||
|
import { motion } from "framer-motion";
|
||||||
|
import { useTranslation } from "react-i18next";
|
||||||
|
import "./profile-content.scss";
|
||||||
|
|
||||||
|
interface ProfileTabsProps {
|
||||||
|
activeTab: "library" | "reviews";
|
||||||
|
reviewsTotalCount: number;
|
||||||
|
onTabChange: (tab: "library" | "reviews") => void;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function ProfileTabs({
|
||||||
|
activeTab,
|
||||||
|
reviewsTotalCount,
|
||||||
|
onTabChange,
|
||||||
|
}: Readonly<ProfileTabsProps>) {
|
||||||
|
const { t } = useTranslation("user_profile");
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className="profile-content__tabs">
|
||||||
|
<div className="profile-content__tab-wrapper">
|
||||||
|
<button
|
||||||
|
type="button"
|
||||||
|
className={`profile-content__tab ${activeTab === "library" ? "profile-content__tab--active" : ""}`}
|
||||||
|
onClick={() => onTabChange("library")}
|
||||||
|
>
|
||||||
|
{t("library")}
|
||||||
|
</button>
|
||||||
|
{activeTab === "library" && (
|
||||||
|
<motion.div
|
||||||
|
className="profile-content__tab-underline"
|
||||||
|
layoutId="tab-underline"
|
||||||
|
transition={{
|
||||||
|
type: "spring",
|
||||||
|
stiffness: 300,
|
||||||
|
damping: 30,
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
<div className="profile-content__tab-wrapper">
|
||||||
|
<button
|
||||||
|
type="button"
|
||||||
|
className={`profile-content__tab ${activeTab === "reviews" ? "profile-content__tab--active" : ""}`}
|
||||||
|
onClick={() => onTabChange("reviews")}
|
||||||
|
>
|
||||||
|
{t("user_reviews")}
|
||||||
|
{reviewsTotalCount > 0 && (
|
||||||
|
<span className="profile-content__tab-badge">
|
||||||
|
{reviewsTotalCount}
|
||||||
|
</span>
|
||||||
|
)}
|
||||||
|
</button>
|
||||||
|
{activeTab === "reviews" && (
|
||||||
|
<motion.div
|
||||||
|
className="profile-content__tab-underline"
|
||||||
|
layoutId="tab-underline"
|
||||||
|
transition={{
|
||||||
|
type: "spring",
|
||||||
|
stiffness: 300,
|
||||||
|
damping: 30,
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -0,0 +1,96 @@
|
|||||||
|
import { motion } from "framer-motion";
|
||||||
|
import { useTranslation } from "react-i18next";
|
||||||
|
import type { GameShop } from "@types";
|
||||||
|
import { ProfileReviewItem } from "./profile-review-item";
|
||||||
|
import "./profile-content.scss";
|
||||||
|
|
||||||
|
interface UserReview {
|
||||||
|
id: string;
|
||||||
|
reviewHtml: string;
|
||||||
|
score: number;
|
||||||
|
playTimeInSeconds?: number;
|
||||||
|
upvotes: number;
|
||||||
|
downvotes: number;
|
||||||
|
hasUpvoted: boolean;
|
||||||
|
hasDownvoted: boolean;
|
||||||
|
createdAt: string;
|
||||||
|
updatedAt: string;
|
||||||
|
user: {
|
||||||
|
id: string;
|
||||||
|
};
|
||||||
|
game: {
|
||||||
|
title: string;
|
||||||
|
iconUrl: string;
|
||||||
|
objectId: string;
|
||||||
|
shop: GameShop;
|
||||||
|
};
|
||||||
|
translations: {
|
||||||
|
[key: string]: string;
|
||||||
|
};
|
||||||
|
detectedLanguage: string | null;
|
||||||
|
}
|
||||||
|
|
||||||
|
interface ReviewsTabProps {
|
||||||
|
reviews: UserReview[];
|
||||||
|
isLoadingReviews: boolean;
|
||||||
|
votingReviews: Set<string>;
|
||||||
|
userDetailsId?: string;
|
||||||
|
formatPlayTime: (playTimeInSeconds: number) => string;
|
||||||
|
getRatingText: (score: number, t: (key: string) => string) => string;
|
||||||
|
onVote: (reviewId: string, isUpvote: boolean) => void;
|
||||||
|
onDelete: (reviewId: string) => void;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function ReviewsTab({
|
||||||
|
reviews,
|
||||||
|
isLoadingReviews,
|
||||||
|
votingReviews,
|
||||||
|
userDetailsId,
|
||||||
|
formatPlayTime,
|
||||||
|
getRatingText,
|
||||||
|
onVote,
|
||||||
|
onDelete,
|
||||||
|
}: Readonly<ReviewsTabProps>) {
|
||||||
|
const { t } = useTranslation("user_profile");
|
||||||
|
|
||||||
|
return (
|
||||||
|
<motion.div
|
||||||
|
key="reviews"
|
||||||
|
className="profile-content__tab-panel"
|
||||||
|
initial={{ opacity: 0, x: -10 }}
|
||||||
|
animate={{ opacity: 1, x: 0 }}
|
||||||
|
exit={{ opacity: 0, x: 10 }}
|
||||||
|
transition={{ duration: 0.2 }}
|
||||||
|
aria-hidden={false}
|
||||||
|
>
|
||||||
|
{isLoadingReviews && (
|
||||||
|
<div className="user-reviews__loading">{t("loading_reviews")}</div>
|
||||||
|
)}
|
||||||
|
{!isLoadingReviews && reviews.length === 0 && (
|
||||||
|
<div className="user-reviews__empty">
|
||||||
|
<p>{t("no_reviews", "No reviews yet")}</p>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
{!isLoadingReviews && reviews.length > 0 && (
|
||||||
|
<div className="user-reviews__list">
|
||||||
|
{reviews.map((review) => {
|
||||||
|
const isOwnReview = userDetailsId === review.user.id;
|
||||||
|
|
||||||
|
return (
|
||||||
|
<ProfileReviewItem
|
||||||
|
key={review.id}
|
||||||
|
review={review}
|
||||||
|
isOwnReview={isOwnReview}
|
||||||
|
isVoting={votingReviews.has(review.id)}
|
||||||
|
formatPlayTime={formatPlayTime}
|
||||||
|
getRatingText={getRatingText}
|
||||||
|
onVote={onVote}
|
||||||
|
onDelete={onDelete}
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
})}
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
</motion.div>
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -36,6 +36,7 @@
|
|||||||
box-shadow: 0 8px 10px -2px rgba(0, 0, 0, 0.5);
|
box-shadow: 0 8px 10px -2px rgba(0, 0, 0, 0.5);
|
||||||
width: 100%;
|
width: 100%;
|
||||||
position: relative;
|
position: relative;
|
||||||
|
overflow: hidden;
|
||||||
|
|
||||||
&:before {
|
&:before {
|
||||||
content: "";
|
content: "";
|
||||||
@@ -193,8 +194,28 @@
|
|||||||
border-radius: 4px;
|
border-radius: 4px;
|
||||||
width: 100%;
|
width: 100%;
|
||||||
height: 100%;
|
height: 100%;
|
||||||
min-width: 100%;
|
display: block;
|
||||||
min-height: 100%;
|
}
|
||||||
|
|
||||||
|
&__cover-placeholder {
|
||||||
|
position: relative;
|
||||||
|
width: 100%;
|
||||||
|
padding-bottom: 150%;
|
||||||
|
background: linear-gradient(
|
||||||
|
90deg,
|
||||||
|
rgba(255, 255, 255, 0.08) 0%,
|
||||||
|
rgba(255, 255, 255, 0.04) 50%,
|
||||||
|
rgba(255, 255, 255, 0.08) 100%
|
||||||
|
);
|
||||||
|
border-radius: 4px;
|
||||||
|
color: rgba(255, 255, 255, 0.3);
|
||||||
|
|
||||||
|
svg {
|
||||||
|
position: absolute;
|
||||||
|
top: 50%;
|
||||||
|
left: 50%;
|
||||||
|
transform: translate(-50%, -50%);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
&__achievements-progress {
|
&__achievements-progress {
|
||||||
|
|||||||
@@ -2,7 +2,7 @@ import { UserGame } from "@types";
|
|||||||
import HydraIcon from "@renderer/assets/icons/hydra.svg?react";
|
import HydraIcon from "@renderer/assets/icons/hydra.svg?react";
|
||||||
import { useFormat, useToast } from "@renderer/hooks";
|
import { useFormat, useToast } from "@renderer/hooks";
|
||||||
import { useNavigate } from "react-router-dom";
|
import { useNavigate } from "react-router-dom";
|
||||||
import { useCallback, useContext, useState } from "react";
|
import { useCallback, useContext, useEffect, useState } from "react";
|
||||||
import {
|
import {
|
||||||
buildGameAchievementPath,
|
buildGameAchievementPath,
|
||||||
buildGameDetailsPath,
|
buildGameDetailsPath,
|
||||||
@@ -15,6 +15,7 @@ import {
|
|||||||
AlertFillIcon,
|
AlertFillIcon,
|
||||||
PinIcon,
|
PinIcon,
|
||||||
PinSlashIcon,
|
PinSlashIcon,
|
||||||
|
ImageIcon,
|
||||||
} from "@primer/octicons-react";
|
} from "@primer/octicons-react";
|
||||||
import { MAX_MINUTES_TO_SHOW_IN_PLAYTIME } from "@renderer/constants";
|
import { MAX_MINUTES_TO_SHOW_IN_PLAYTIME } from "@renderer/constants";
|
||||||
import { Tooltip } from "react-tooltip";
|
import { Tooltip } from "react-tooltip";
|
||||||
@@ -44,6 +45,11 @@ export function UserLibraryGameCard({
|
|||||||
const navigate = useNavigate();
|
const navigate = useNavigate();
|
||||||
const [isTooltipHovered, setIsTooltipHovered] = useState(false);
|
const [isTooltipHovered, setIsTooltipHovered] = useState(false);
|
||||||
const [isPinning, setIsPinning] = useState(false);
|
const [isPinning, setIsPinning] = useState(false);
|
||||||
|
const [imageError, setImageError] = useState(false);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
setImageError(false);
|
||||||
|
}, [game.coverImageUrl]);
|
||||||
|
|
||||||
const getStatsItemCount = useCallback(() => {
|
const getStatsItemCount = useCallback(() => {
|
||||||
let statsCount = 1;
|
let statsCount = 1;
|
||||||
@@ -233,11 +239,18 @@ export function UserLibraryGameCard({
|
|||||||
)}
|
)}
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<img
|
{imageError || !game.coverImageUrl ? (
|
||||||
src={game.coverImageUrl ?? undefined}
|
<div className="user-library-game__cover-placeholder">
|
||||||
alt={game.title}
|
<ImageIcon size={48} />
|
||||||
className="user-library-game__game-image"
|
</div>
|
||||||
/>
|
) : (
|
||||||
|
<img
|
||||||
|
src={game.coverImageUrl}
|
||||||
|
alt={game.title}
|
||||||
|
className="user-library-game__game-image"
|
||||||
|
onError={() => setImageError(true)}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
</button>
|
</button>
|
||||||
</li>
|
</li>
|
||||||
<Tooltip
|
<Tooltip
|
||||||
|
|||||||
@@ -244,6 +244,7 @@ export interface GameReview {
|
|||||||
isBlocked: boolean;
|
isBlocked: boolean;
|
||||||
hasUpvoted: boolean;
|
hasUpvoted: boolean;
|
||||||
hasDownvoted: boolean;
|
hasDownvoted: boolean;
|
||||||
|
playTimeInSeconds?: number;
|
||||||
user: {
|
user: {
|
||||||
id: string;
|
id: string;
|
||||||
displayName: string;
|
displayName: string;
|
||||||
|
|||||||
12
yarn.lock
12
yarn.lock
@@ -7538,6 +7538,13 @@ react-i18next@^14.1.0:
|
|||||||
"@babel/runtime" "^7.23.9"
|
"@babel/runtime" "^7.23.9"
|
||||||
html-parse-stringify "^3.0.1"
|
html-parse-stringify "^3.0.1"
|
||||||
|
|
||||||
|
react-infinite-scroll-component@^6.1.0:
|
||||||
|
version "6.1.0"
|
||||||
|
resolved "https://registry.yarnpkg.com/react-infinite-scroll-component/-/react-infinite-scroll-component-6.1.0.tgz#7e511e7aa0f728ac3e51f64a38a6079ac522407f"
|
||||||
|
integrity sha512-SQu5nCqy8DxQWpnUVLx7V7b7LcA37aM7tvoWjTLZp1dk6EJibM5/4EJKzOnl07/BsM1Y40sKLuqjCwwH/xV0TQ==
|
||||||
|
dependencies:
|
||||||
|
throttle-debounce "^2.1.0"
|
||||||
|
|
||||||
react-is@^16.13.1, react-is@^16.7.0:
|
react-is@^16.13.1, react-is@^16.7.0:
|
||||||
version "16.13.1"
|
version "16.13.1"
|
||||||
resolved "https://registry.yarnpkg.com/react-is/-/react-is-16.13.1.tgz#789729a4dc36de2999dc156dd6c1d9c18cea56a4"
|
resolved "https://registry.yarnpkg.com/react-is/-/react-is-16.13.1.tgz#789729a4dc36de2999dc156dd6c1d9c18cea56a4"
|
||||||
@@ -8540,6 +8547,11 @@ text-table@^0.2.0:
|
|||||||
resolved "https://registry.yarnpkg.com/text-table/-/text-table-0.2.0.tgz#7f5ee823ae805207c00af2df4a84ec3fcfa570b4"
|
resolved "https://registry.yarnpkg.com/text-table/-/text-table-0.2.0.tgz#7f5ee823ae805207c00af2df4a84ec3fcfa570b4"
|
||||||
integrity sha512-N+8UisAXDGk8PFXP4HAzVR9nbfmVJ3zYLAWiTIoqC5v5isinhr+r5uaO8+7r3BMfuNIufIsA7RdpVgacC2cSpw==
|
integrity sha512-N+8UisAXDGk8PFXP4HAzVR9nbfmVJ3zYLAWiTIoqC5v5isinhr+r5uaO8+7r3BMfuNIufIsA7RdpVgacC2cSpw==
|
||||||
|
|
||||||
|
throttle-debounce@^2.1.0:
|
||||||
|
version "2.3.0"
|
||||||
|
resolved "https://registry.yarnpkg.com/throttle-debounce/-/throttle-debounce-2.3.0.tgz#fd31865e66502071e411817e241465b3e9c372e2"
|
||||||
|
integrity sha512-H7oLPV0P7+jgvrk+6mwwwBDmxTaxnu9HMXmloNLXwnNO0ZxZ31Orah2n8lU1eMPvsaowP2CX+USCgyovXfdOFQ==
|
||||||
|
|
||||||
"through@>=2.2.7 <3":
|
"through@>=2.2.7 <3":
|
||||||
version "2.3.8"
|
version "2.3.8"
|
||||||
resolved "https://registry.yarnpkg.com/through/-/through-2.3.8.tgz#0dd4c9ffaabc357960b1b724115d7e0e86a2e1f5"
|
resolved "https://registry.yarnpkg.com/through/-/through-2.3.8.tgz#0dd4c9ffaabc357960b1b724115d7e0e86a2e1f5"
|
||||||
|
|||||||
Reference in New Issue
Block a user