add chat item component and edit delete message
This commit is contained in:
parent
952f93cab0
commit
5da047a274
242
components/chat/chat-item.tsx
Normal file
242
components/chat/chat-item.tsx
Normal file
@ -0,0 +1,242 @@
|
|||||||
|
"use client"
|
||||||
|
|
||||||
|
import * as z from "zod";
|
||||||
|
import axios from "axios";
|
||||||
|
import qs from "query-string";
|
||||||
|
import { useForm } from "react-hook-form";
|
||||||
|
import { zodResolver } from "@hookform/resolvers/zod";
|
||||||
|
|
||||||
|
import { Member, MemberRole, Profile } from "@prisma/client";
|
||||||
|
import { UserAvatar } from "@/components/user-avatar";
|
||||||
|
import { ActionTooltip } from "@/components/action-tooltip";
|
||||||
|
import { Edit, FileIcon, ShieldAlert, ShieldCheck, Trash } from "lucide-react";
|
||||||
|
import Image from "next/image";
|
||||||
|
import { useEffect, useState } from "react";
|
||||||
|
import { cn } from "@/lib/utils";
|
||||||
|
|
||||||
|
import {
|
||||||
|
Form,
|
||||||
|
FormControl,
|
||||||
|
FormField,
|
||||||
|
FormItem,
|
||||||
|
} from "@/components/ui/form"
|
||||||
|
import { Input } from "@/components/ui/input";
|
||||||
|
import { Button } from "@/components/ui/button";
|
||||||
|
import { useModal } from "@/hooks/use-modal-store";
|
||||||
|
|
||||||
|
interface ChatItemProps {
|
||||||
|
id: string;
|
||||||
|
content: string;
|
||||||
|
member: Member & {
|
||||||
|
profile: Profile;
|
||||||
|
};
|
||||||
|
timestamp: string
|
||||||
|
fileUrl: string | null;
|
||||||
|
deleted: boolean;
|
||||||
|
currentMember: Member;
|
||||||
|
isUpdated: boolean;
|
||||||
|
socketUrl: string;
|
||||||
|
socketQuery: Record<string, string>;
|
||||||
|
}
|
||||||
|
|
||||||
|
const roleIconMap = {
|
||||||
|
"GUEST": null,
|
||||||
|
"MODERATOR": <ShieldCheck className="h-4 w-4 ml-2 text-indigo-500"/>,
|
||||||
|
"ADMIN": <ShieldAlert className="h-4 w-4 ml-2 text-rose-500"/>,
|
||||||
|
}
|
||||||
|
|
||||||
|
const formSchema = z.object({
|
||||||
|
content: z.string().min(1),
|
||||||
|
})
|
||||||
|
|
||||||
|
export const ChatItem = ({
|
||||||
|
id,
|
||||||
|
content,
|
||||||
|
member,
|
||||||
|
timestamp,
|
||||||
|
fileUrl,
|
||||||
|
deleted,
|
||||||
|
currentMember,
|
||||||
|
isUpdated,
|
||||||
|
socketUrl,
|
||||||
|
socketQuery,
|
||||||
|
}: ChatItemProps) => {
|
||||||
|
const [isEditing, setIsEditing] = useState(false);
|
||||||
|
const { onOpen } = useModal();
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
const handleKeyDown = (event: KeyboardEvent) => {
|
||||||
|
if (event.key === "Escape" || event.keyCode === 27) {
|
||||||
|
setIsEditing(false);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
window.addEventListener("keydown", handleKeyDown);
|
||||||
|
|
||||||
|
return () => window.removeEventListener("keydown", handleKeyDown);
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
const form = useForm<z.infer<typeof formSchema>>({
|
||||||
|
resolver: zodResolver(formSchema),
|
||||||
|
defaultValues: {
|
||||||
|
content: content
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
const isLoading = form.formState.isSubmitting;
|
||||||
|
|
||||||
|
const onSubmit = async (values: z.infer<typeof formSchema>) => {
|
||||||
|
try {
|
||||||
|
const url = qs.stringifyUrl({
|
||||||
|
url: `${socketUrl}/${id}`,
|
||||||
|
query: socketQuery,
|
||||||
|
});
|
||||||
|
|
||||||
|
await axios.patch(url, values);
|
||||||
|
|
||||||
|
form.reset();
|
||||||
|
setIsEditing(false);
|
||||||
|
} catch (error) {
|
||||||
|
console.log(error);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
form.reset({
|
||||||
|
content: content,
|
||||||
|
})
|
||||||
|
}, [content]);
|
||||||
|
|
||||||
|
const filetype = fileUrl?.split(".").pop();
|
||||||
|
|
||||||
|
const isAdmin = currentMember.role === MemberRole.ADMIN;
|
||||||
|
const isModerator = currentMember.role === MemberRole.MODERATOR;
|
||||||
|
const isOwner = currentMember.id === member.id;
|
||||||
|
const canDeleteMessage = !deleted && (isAdmin || isModerator || isOwner);
|
||||||
|
const canEditMessage = !deleted && isOwner && !fileUrl;
|
||||||
|
const isPDF = filetype == "pdf" && fileUrl;
|
||||||
|
const isImage = !isPDF && fileUrl;
|
||||||
|
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className="relative group flex items-center hover:bg-black/5 p-4 transition w-full">
|
||||||
|
<div className="group flex gap-x-2 items-start w-full">
|
||||||
|
<div className="cursor-pointer hover:drop-shadow-md transition">
|
||||||
|
<UserAvatar src={member.profile.imageUrl}/>
|
||||||
|
</div>
|
||||||
|
<div className="flex flex-col w-full">
|
||||||
|
<div className="flex items-center gap-x-2">
|
||||||
|
<div className="flex items-center">
|
||||||
|
<p className="font-semibold text-sm hove:underline cursor-pointer">
|
||||||
|
{member.profile.name}
|
||||||
|
</p>
|
||||||
|
<ActionTooltip label={member.role}>
|
||||||
|
{roleIconMap[member.role]}
|
||||||
|
</ActionTooltip>
|
||||||
|
</div>
|
||||||
|
<span className="text-xs text-zinc-500 dark:text-zinc-400">
|
||||||
|
{timestamp}
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
{isImage && (
|
||||||
|
<a
|
||||||
|
href={fileUrl}
|
||||||
|
target="_blank"
|
||||||
|
rel="noopener noreferrer"
|
||||||
|
className="relative aspect-square rounded-md mt-2 overflow-hidden border flex items-center bg-secondary h-48 w-48"
|
||||||
|
>
|
||||||
|
<Image
|
||||||
|
src={fileUrl}
|
||||||
|
alt={content}
|
||||||
|
fill
|
||||||
|
className="object-cover"
|
||||||
|
/>
|
||||||
|
</a>
|
||||||
|
)}
|
||||||
|
{isPDF && (
|
||||||
|
<div className="relative flex items-center p-2 mt-2 rounded-md bg-background/10">
|
||||||
|
<FileIcon className="h-10 w-10 fill-indigo-200 stroke-indigo-400 "/>
|
||||||
|
<a
|
||||||
|
href={fileUrl}
|
||||||
|
target="_blank"
|
||||||
|
rel="noopener noreferrer"
|
||||||
|
className="ml-2 text-sm text-indigo-500 dark:text-indigo-400 hover:underline"
|
||||||
|
>
|
||||||
|
PDF File
|
||||||
|
</a>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
{!fileUrl && !isEditing && (
|
||||||
|
<p className={cn(
|
||||||
|
"text-sm text-zinc-600 dark:text-zinc-300",
|
||||||
|
deleted && "italic text-zinc-500 dark:text-zinc-400 text-xs mt-1"
|
||||||
|
)}>
|
||||||
|
{content}
|
||||||
|
{isUpdated && (
|
||||||
|
<span className="text-[10px] ml-2 text-zinc-500 dark:text-zinc-400">
|
||||||
|
(edited)
|
||||||
|
</span>
|
||||||
|
)}
|
||||||
|
</p>
|
||||||
|
)}
|
||||||
|
{!fileUrl && isEditing && (
|
||||||
|
<Form {...form}>
|
||||||
|
<form
|
||||||
|
className="flex items-center w-full gap-x-2 pt-2"
|
||||||
|
onSubmit={form.handleSubmit(onSubmit)}>
|
||||||
|
<FormField
|
||||||
|
control={form.control}
|
||||||
|
name="content"
|
||||||
|
render={({ field }) => (
|
||||||
|
<FormItem className="flex-1">
|
||||||
|
<FormControl>
|
||||||
|
<div className="relative w-full">
|
||||||
|
<Input
|
||||||
|
disabled={isLoading}
|
||||||
|
className="p-2 bg-zinc-200/90 dark:bg-zinc-700/75 border-0 focus-visible:ring-0 focus-visible:ring-offset-0 text-zinc-600 dark:text-zinc-200"
|
||||||
|
placeholder="Edited message"
|
||||||
|
{...field}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</FormControl>
|
||||||
|
|
||||||
|
</FormItem>
|
||||||
|
)}
|
||||||
|
/>
|
||||||
|
<Button disabled={isLoading} size="sm" variant="primary">
|
||||||
|
Save
|
||||||
|
</Button>
|
||||||
|
</form>
|
||||||
|
<span className="text-[10px] mt-1 text-zinc-400">
|
||||||
|
Press escape to cancel, enter to save
|
||||||
|
|
||||||
|
</span>
|
||||||
|
</Form>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
{canDeleteMessage && (
|
||||||
|
<div className="hidden group-hover:flex items-center gap-x-2 absolute p-1 -top-2 right-5 bg-white dark:bg-zinc-800 border rounded-sm">
|
||||||
|
{canEditMessage && (
|
||||||
|
<ActionTooltip label="Edit">
|
||||||
|
<Edit
|
||||||
|
onClick={() => setIsEditing(true)}
|
||||||
|
className="cursor-pointer ml-auto w-4 h-4 text-zinc-500 hover:text-zinc-600 dark:hove:text-zinc-300 transition"
|
||||||
|
/>
|
||||||
|
|
||||||
|
</ActionTooltip>
|
||||||
|
)}
|
||||||
|
<ActionTooltip label="Delete">
|
||||||
|
<Trash
|
||||||
|
onClick={() => onOpen("deleteMessage", {
|
||||||
|
apiUrl: `${socketUrl}/${id}`,
|
||||||
|
query: socketQuery,
|
||||||
|
})}
|
||||||
|
className="cursor-pointer ml-auto w-4 h-4 text-zinc-500 hover:text-zinc-600 dark:hove:text-zinc-300 transition"
|
||||||
|
/>
|
||||||
|
|
||||||
|
</ActionTooltip>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
}
|
@ -1,11 +1,15 @@
|
|||||||
"use client";
|
"use client";
|
||||||
|
|
||||||
import { Member, Message, Profile } from "@prisma/client";
|
import { Member, Message, Profile } from "@prisma/client";
|
||||||
|
import { format } from "date-fns"
|
||||||
|
|
||||||
import { ChatWelcome } from "./chat-welcome";
|
import { ChatWelcome } from "./chat-welcome";
|
||||||
import { useChatQuery } from "@/hooks/use-chat-query";
|
import { useChatQuery } from "@/hooks/use-chat-query";
|
||||||
import { Loader2, ServerCrash } from "lucide-react";
|
import { Loader2, ServerCrash } from "lucide-react";
|
||||||
import { Fragment } from "react";
|
import { Fragment } from "react";
|
||||||
|
import { ChatItem } from "./chat-item";
|
||||||
|
|
||||||
|
const DATE_FORMAT = "d MMM yyyy, HH:mm";
|
||||||
|
|
||||||
type MessageWithMemberWithProfile = Message & {
|
type MessageWithMemberWithProfile = Message & {
|
||||||
member: Member & {
|
member: Member & {
|
||||||
@ -84,9 +88,20 @@ export const ChatMessages = ({
|
|||||||
{data?.pages.map((group, i) => (
|
{data?.pages.map((group, i) => (
|
||||||
<Fragment key={i}>
|
<Fragment key={i}>
|
||||||
{group.items.map((message: MessageWithMemberWithProfile) => (
|
{group.items.map((message: MessageWithMemberWithProfile) => (
|
||||||
<div key={message.id}>
|
<ChatItem
|
||||||
{message.content}
|
key={message.id}
|
||||||
</div>
|
id={message.id}
|
||||||
|
currentMember={member}
|
||||||
|
member={message.member}
|
||||||
|
content={message.content}
|
||||||
|
fileUrl={message.fileUrl}
|
||||||
|
deleted={message.deleted}
|
||||||
|
timestamp={format(new Date(message.createdAt), DATE_FORMAT)}
|
||||||
|
isUpdated={message.updatedAt !== message.createdAt}
|
||||||
|
socketUrl={socketUrl}
|
||||||
|
socketQuery={socketQuery}
|
||||||
|
|
||||||
|
/>
|
||||||
))}
|
))}
|
||||||
</Fragment>
|
</Fragment>
|
||||||
))}
|
))}
|
||||||
|
79
components/modals/delete-message-modal.tsx
Normal file
79
components/modals/delete-message-modal.tsx
Normal file
@ -0,0 +1,79 @@
|
|||||||
|
"use client";
|
||||||
|
|
||||||
|
import qs from "query-string";
|
||||||
|
import axios from "axios";
|
||||||
|
import { useState } from "react";
|
||||||
|
|
||||||
|
import {
|
||||||
|
Dialog,
|
||||||
|
DialogContent,
|
||||||
|
DialogDescription,
|
||||||
|
DialogFooter,
|
||||||
|
DialogHeader,
|
||||||
|
DialogTitle,
|
||||||
|
} from "@/components/ui/dialog";
|
||||||
|
import { useModal } from "@/hooks/use-modal-store";
|
||||||
|
import { Button } from "@/components/ui/button";
|
||||||
|
|
||||||
|
|
||||||
|
export const DeleteMessageModal = () => {
|
||||||
|
const { isOpen, onClose, type, data } = useModal();
|
||||||
|
|
||||||
|
const isModalOpen = isOpen && type === "deleteMessage";
|
||||||
|
const { apiUrl, query } = data;
|
||||||
|
|
||||||
|
const [isLoading, setIsLoading] = useState(false);
|
||||||
|
|
||||||
|
const onClick = async () =>
|
||||||
|
{
|
||||||
|
try
|
||||||
|
{
|
||||||
|
setIsLoading(true);
|
||||||
|
|
||||||
|
const url = qs.stringifyUrl({
|
||||||
|
url: apiUrl || "",
|
||||||
|
query,
|
||||||
|
})
|
||||||
|
|
||||||
|
await axios.delete(url);
|
||||||
|
|
||||||
|
onClose();
|
||||||
|
}
|
||||||
|
catch (error)
|
||||||
|
{
|
||||||
|
console.log(error);
|
||||||
|
}
|
||||||
|
finally
|
||||||
|
{
|
||||||
|
setIsLoading(false);
|
||||||
|
onClose();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Dialog open={isModalOpen} onOpenChange={onClose}>
|
||||||
|
<DialogContent className="bg-white text-black p-0 overflow-hidden">
|
||||||
|
<DialogHeader className="pt-8 px-6">
|
||||||
|
<DialogTitle className="text-2xl text-center font-bold">
|
||||||
|
Delete Message
|
||||||
|
</DialogTitle>
|
||||||
|
<DialogDescription className="text-center text-zinc-500">
|
||||||
|
Are you sure you want to do this? <br/>
|
||||||
|
The message will be permenantly deleted.
|
||||||
|
</DialogDescription>
|
||||||
|
</DialogHeader>
|
||||||
|
<DialogFooter className="bg-gray-100 px-6 py-4">
|
||||||
|
<div className="flex items-center justify-between w-full">
|
||||||
|
<Button disabled={isLoading} onClick={onClose} variant="ghost">
|
||||||
|
Cancel
|
||||||
|
</Button>
|
||||||
|
<Button disabled={isLoading} onClick={onClick} variant="primary">
|
||||||
|
Confirm
|
||||||
|
</Button>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
</DialogFooter>
|
||||||
|
</DialogContent>
|
||||||
|
</Dialog>
|
||||||
|
)
|
||||||
|
}
|
@ -12,6 +12,7 @@ import { DeleteServerModal } from "@/components/modals/delete-server-modal";
|
|||||||
import { DeleteChannelModal } from "@/components/modals/delete-channel-modal";
|
import { DeleteChannelModal } from "@/components/modals/delete-channel-modal";
|
||||||
import { EditChannelModal } from "@/components/modals/edit-channel-modal";
|
import { EditChannelModal } from "@/components/modals/edit-channel-modal";
|
||||||
import { MessageFileModal } from "@/components/modals/message-file-modal.tsx";
|
import { MessageFileModal } from "@/components/modals/message-file-modal.tsx";
|
||||||
|
import { DeleteMessageModal } from "@/components/modals/delete-message-modal";
|
||||||
|
|
||||||
export const ModalProvider = () => {
|
export const ModalProvider = () => {
|
||||||
const [isMounted, setIsMounted] = useState(false);
|
const [isMounted, setIsMounted] = useState(false);
|
||||||
@ -36,6 +37,7 @@ export const ModalProvider = () => {
|
|||||||
<DeleteChannelModal/>
|
<DeleteChannelModal/>
|
||||||
<EditChannelModal/>
|
<EditChannelModal/>
|
||||||
<MessageFileModal/>
|
<MessageFileModal/>
|
||||||
|
<DeleteMessageModal/>
|
||||||
</>
|
</>
|
||||||
)
|
)
|
||||||
}
|
}
|
@ -2,7 +2,7 @@
|
|||||||
import { Channel, ChannelType, Server } from "@prisma/client";
|
import { Channel, ChannelType, Server } from "@prisma/client";
|
||||||
import { create } from "zustand";
|
import { create } from "zustand";
|
||||||
|
|
||||||
export type ModalType = "createServer" | "invite" | "editServer" | "members" | "createChannel" | "leaveServer" | "deleteServer" | "deleteChannel" | "editChannel" | "messageFile";
|
export type ModalType = "createServer" | "invite" | "editServer" | "members" | "createChannel" | "leaveServer" | "deleteServer" | "deleteChannel" | "editChannel" | "messageFile" | "deleteMessage";
|
||||||
|
|
||||||
interface ModalData {
|
interface ModalData {
|
||||||
server?: Server;
|
server?: Server;
|
||||||
|
143
pages/api/socket/messages/[messageId].ts
Normal file
143
pages/api/socket/messages/[messageId].ts
Normal file
@ -0,0 +1,143 @@
|
|||||||
|
import { currentProfilePages } from "@/lib/current-profile-pages";
|
||||||
|
import { db } from "@/lib/db";
|
||||||
|
import { NextApiResponseServerIo } from "@/types";
|
||||||
|
import { MemberRole } from "@prisma/client";
|
||||||
|
import { NextApiRequest } from "next";
|
||||||
|
|
||||||
|
export default async function handler(req: NextApiRequest, res: NextApiResponseServerIo) {
|
||||||
|
if (req.method !== "DELETE" && req.method !== "PATCH") {
|
||||||
|
return res.status(405).json({ error: "Method not allowed" });
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
const profile = await currentProfilePages(req);
|
||||||
|
const { serverId, channelId, messageId } = req.query;
|
||||||
|
const { content } = req.body;
|
||||||
|
|
||||||
|
if (!profile) {
|
||||||
|
return res.status(401).json({ error: "Unauthorized" });
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!serverId) {
|
||||||
|
return res.status(400).json({ error: "Server ID Missing" });
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!channelId) {
|
||||||
|
return res.status(400).json({ error: "Channel ID Missing" });
|
||||||
|
}
|
||||||
|
|
||||||
|
const server= await db.server.findFirst({
|
||||||
|
where: {
|
||||||
|
id: serverId as string,
|
||||||
|
members: {
|
||||||
|
some: {
|
||||||
|
profileId: profile.id
|
||||||
|
}
|
||||||
|
},
|
||||||
|
},
|
||||||
|
include: {
|
||||||
|
members: true
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
if (!server) {
|
||||||
|
return res.status(404).json({ error: "Server not found" });
|
||||||
|
}
|
||||||
|
|
||||||
|
const channel = await db.channel.findFirst({
|
||||||
|
where: {
|
||||||
|
id: channelId as string,
|
||||||
|
serverId: server.id,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
if (!channel) {
|
||||||
|
return res.status(404).json({ error: "Channel not found" });
|
||||||
|
}
|
||||||
|
|
||||||
|
const member = server.members.find((member) => member.profileId === profile.id);
|
||||||
|
|
||||||
|
if (!member) {
|
||||||
|
return res.status(404).json({ error: "Member not found" });
|
||||||
|
}
|
||||||
|
|
||||||
|
let message = await db.message.findFirst({
|
||||||
|
where: {
|
||||||
|
id: messageId as string,
|
||||||
|
channelId: channel.id,
|
||||||
|
},
|
||||||
|
include: {
|
||||||
|
member: {
|
||||||
|
include: {
|
||||||
|
profile: true,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
if (!message || message.deleted) {
|
||||||
|
return res.status(404).json({ error: "Message not found" });
|
||||||
|
}
|
||||||
|
|
||||||
|
const isMessageOwner = message.memberId === member.id;
|
||||||
|
const isAdmin = member.role === MemberRole.ADMIN;
|
||||||
|
const isModerator = member.role === MemberRole.MODERATOR;
|
||||||
|
const canModify = isMessageOwner || isAdmin || isModerator;
|
||||||
|
|
||||||
|
if (!canModify) {
|
||||||
|
return res.status(401).json({ error: "Unauthorized" });
|
||||||
|
}
|
||||||
|
|
||||||
|
if (req.method === "DELETE") {
|
||||||
|
message = await db.message.update({
|
||||||
|
where: {
|
||||||
|
id: messageId as string,
|
||||||
|
},
|
||||||
|
data: {
|
||||||
|
fileUrl: null,
|
||||||
|
content: "This message has been deleted.",
|
||||||
|
deleted: true,
|
||||||
|
},
|
||||||
|
include: {
|
||||||
|
member: {
|
||||||
|
include: {
|
||||||
|
profile: true,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
if (req.method === "PATCH") {
|
||||||
|
if (!isMessageOwner) {
|
||||||
|
return res.status(401).json({ error: "Unauthorized" });
|
||||||
|
}
|
||||||
|
|
||||||
|
message = await db.message.update({
|
||||||
|
where: {
|
||||||
|
id: messageId as string,
|
||||||
|
},
|
||||||
|
data: {
|
||||||
|
content,
|
||||||
|
},
|
||||||
|
include: {
|
||||||
|
member: {
|
||||||
|
include: {
|
||||||
|
profile: true,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
const updateKey = `chat:${channelId}:messages:update`;
|
||||||
|
|
||||||
|
res?.socket?.server?.io?.to(updateKey).emit(updateKey, message);
|
||||||
|
|
||||||
|
return res.status(200).json(message);
|
||||||
|
|
||||||
|
} catch (error) {
|
||||||
|
console.error(error);
|
||||||
|
return res.status(500).json({ error: "Internal Error" });
|
||||||
|
}
|
||||||
|
}
|
Loading…
x
Reference in New Issue
Block a user