feat: add avatar editing

This commit is contained in:
Kentai Radiquum 2024-09-20 23:47:35 +05:00
parent 8164e7c8bd
commit 725b8a2c10
Signed by: Radiquum
GPG key ID: 858E8EE696525EED
2 changed files with 88 additions and 14 deletions

View file

@ -24,6 +24,7 @@ export const ENDPOINTS = {
change: `${API_PREFIX}/profile/preference/login/change`, // ?login=<url_encoded_string> change: `${API_PREFIX}/profile/preference/login/change`, // ?login=<url_encoded_string>
}, },
status: `${API_PREFIX}/profile/preference/status/edit`, status: `${API_PREFIX}/profile/preference/status/edit`,
avatar: `${API_PREFIX}/profile/preference/avatar/edit`,
privacy: { privacy: {
stats: `${API_PREFIX}/profile/preference/privacy/stats/edit`, stats: `${API_PREFIX}/profile/preference/privacy/stats/edit`,
counts: `${API_PREFIX}/profile/preference/privacy/counts/edit`, counts: `${API_PREFIX}/profile/preference/privacy/counts/edit`,

View file

@ -1,14 +1,17 @@
"use client"; "use client";
import { Modal } from "flowbite-react"; import { FileInput, Label, Modal } from "flowbite-react";
import { Spinner } from "../Spinner/Spinner"; import { Spinner } from "../Spinner/Spinner";
import useSWR from "swr"; import useSWR from "swr";
import { ENDPOINTS } from "#/api/config"; import { ENDPOINTS } from "#/api/config";
import { useEffect, useState } from "react"; import { useEffect, useState } from "react";
import { unixToDate } from "#/api/utils"; import { b64toBlob, unixToDate } from "#/api/utils";
import { ProfileEditPrivacyModal } from "./Profile.EditPrivacyModal"; import { ProfileEditPrivacyModal } from "./Profile.EditPrivacyModal";
import { ProfileEditStatusModal } from "./Profile.EditStatusModal"; import { ProfileEditStatusModal } from "./Profile.EditStatusModal";
import { ProfileEditSocialModal } from "./Profile.EditSocialModal"; import { ProfileEditSocialModal } from "./Profile.EditSocialModal";
import { CropModal } from "../CropModal/CropModal";
import { useSWRConfig } from "swr";
import { useUserStore } from "#/store/auth";
const fetcher = async (url: string) => { const fetcher = async (url: string) => {
const res = await fetch(url); const res = await fetch(url);
@ -33,6 +36,9 @@ export const ProfileEditModal = (props: {
const [privacyModalOpen, setPrivacyModalOpen] = useState(false); const [privacyModalOpen, setPrivacyModalOpen] = useState(false);
const [statusModalOpen, setStatusModalOpen] = useState(false); const [statusModalOpen, setStatusModalOpen] = useState(false);
const [socialModalOpen, setSocialModalOpen] = useState(false); const [socialModalOpen, setSocialModalOpen] = useState(false);
const [avatarModalOpen, setAvatarModalOpen] = useState(false);
const [avatarUri, setAvatarUri] = useState(null);
const [tempAvatarUri, setTempAvatarUri] = useState(null);
const [privacyModalSetting, setPrivacyModalSetting] = useState("none"); const [privacyModalSetting, setPrivacyModalSetting] = useState("none");
const [privacySettings, setPrivacySettings] = useState({ const [privacySettings, setPrivacySettings] = useState({
privacy_stats: 9, privacy_stats: 9,
@ -46,6 +52,8 @@ export const ProfileEditModal = (props: {
}); });
const [status, setStatus] = useState(""); const [status, setStatus] = useState("");
const [login, setLogin] = useState(""); const [login, setLogin] = useState("");
const { mutate } = useSWRConfig();
const userStore = useUserStore();
const privacy_stat_act_social_text = { const privacy_stat_act_social_text = {
0: "Все пользователи", 0: "Все пользователи",
@ -71,6 +79,19 @@ export const ProfileEditModal = (props: {
`${ENDPOINTS.user.settings.login.info}?token=${props.token}` `${ENDPOINTS.user.settings.login.info}?token=${props.token}`
); );
const handleFileRead = (e, fileReader) => {
const content = fileReader.result;
setTempAvatarUri(content);
};
const handleFilePreview = (file) => {
const fileReader = new FileReader();
fileReader.onloadend = (e) => {
handleFileRead(e, fileReader);
};
fileReader.readAsDataURL(file);
};
useEffect(() => { useEffect(() => {
if (prefData) { if (prefData) {
setPrivacySettings({ setPrivacySettings({
@ -93,6 +114,33 @@ export const ProfileEditModal = (props: {
} }
}, [loginData]); }, [loginData]);
useEffect(() => {
if (avatarUri) {
let block = avatarUri.split(";");
let contentType = block[0].split(":")[1];
let realData = block[1].split(",")[1];
const blob = b64toBlob(realData, contentType);
const formData = new FormData();
formData.append("image", blob, "cropped.jpg");
formData.append("name", "image");
const uploadRes = fetch(
`${ENDPOINTS.user.settings.avatar}?token=${props.token}`,
{
method: "POST",
body: formData,
}
).then((res) => {
if (res.ok) {
mutate(
`${ENDPOINTS.user.profile}/${props.profile_id}?token=${props.token}`
);
userStore.checkAuth();
}
});
}
}, [avatarUri]);
return ( return (
<> <>
<Modal <Modal
@ -120,15 +168,28 @@ export const ProfileEditModal = (props: {
className="p-2 text-left rounded-md hover:bg-gray-100" className="p-2 text-left rounded-md hover:bg-gray-100"
disabled={prefData.is_change_avatar_banned} disabled={prefData.is_change_avatar_banned}
> >
<p className="text-lg">Изменить фото профиля</p> <Label htmlFor="dropzone-file" className="cursor-pointer">
<p className="text-base text-gray-500"> <FileInput
{prefData.is_change_avatar_banned id="dropzone-file"
? `Заблокировано до ${unixToDate( className="hidden"
prefData.ban_change_avatar_expires, accept="image/jpg, image/jpeg, image/png"
"full" onChange={(e) => {
)}` handleFilePreview(e.target.files[0]);
: "Загрузить с устройства"} setAvatarModalOpen(true);
</p> }}
/>
<div>
<p className="text-lg">Изменить фото профиля</p>
<p className="text-base text-gray-500">
{prefData.is_change_avatar_banned
? `Заблокировано до ${unixToDate(
prefData.ban_change_avatar_expires,
"full"
)}`
: "Загрузить с устройства"}
</p>
</div>
</Label>
</button> </button>
<button <button
className="p-2 text-left rounded-md hover:bg-gray-100" className="p-2 text-left rounded-md hover:bg-gray-100"
@ -257,9 +318,7 @@ export const ProfileEditModal = (props: {
<div className="flex flex-col"> <div className="flex flex-col">
<div className="flex items-center gap-2"> <div className="flex items-center gap-2">
<span className="w-8 h-8 iconify mdi--link"></span> <span className="w-8 h-8 iconify mdi--link"></span>
<p className="text-xl font-bold"> <p className="text-xl font-bold">Привязка к сервисам</p>
Привязка к сервисам
</p>
</div> </div>
<p className="mx-1 text-base text-gray-500"> <p className="mx-1 text-base text-gray-500">
Недоступно для изменения в данном клиенте Недоступно для изменения в данном клиенте
@ -303,6 +362,20 @@ export const ProfileEditModal = (props: {
token={props.token} token={props.token}
profile_id={props.profile_id} profile_id={props.profile_id}
/> />
<CropModal
src={tempAvatarUri}
setSrc={setAvatarUri}
setTempSrc={setTempAvatarUri}
// setImageData={setImageData}
aspectRatio={1 / 1}
guides={true}
quality={100}
isOpen={avatarModalOpen}
setIsOpen={setAvatarModalOpen}
forceAspect={true}
width={600}
height={600}
/>
</> </>
); );
}; };