This commit is contained in:
tkrmagid-desktop
2026-04-11 14:22:48 +09:00
parent b329ccc03b
commit 4037afeb68
12 changed files with 200 additions and 86 deletions

View File

@@ -1,9 +1,10 @@
"use client";
import { useEffect, useState } from "react";
import { useSession } from "next-auth/react";
// 🌟 [수정됨] 상단 아이콘에 쓸 ListPlus 추가
import { Play, ChevronLeft, Server, Music, Loader2, SearchX, ListPlus } from "lucide-react";
import { Play, ChevronLeft, Server, Music, Loader2, SearchX, MonitorPlay, Disc } from "lucide-react";
import { ViewMode } from "@/app/page";
// 🌟 [추가됨] 전역 토스트 훅 불러오기
import { useToast } from "@/components/ToastProvider";
interface MainContentProps {
viewMode: ViewMode;
@@ -15,6 +16,12 @@ interface MainContentProps {
onSelectServer: (server: any) => void;
}
interface SearchResultsType {
spotify: any[];
youtubeMusic: any[];
youtubeVideo: any[];
}
export default function MainContent({
viewMode,
setViewMode,
@@ -25,19 +32,27 @@ export default function MainContent({
onSelectServer,
}: MainContentProps) {
const { data: session, status } = useSession();
// 🌟 [추가됨] 훅을 실행해서 showToast 함수 꺼내기
const { showToast } = useToast();
const [servers, setServers] = useState<any[]>([]);
const [isFetching, setIsFetching] = useState(false);
const [searchResults, setSearchResults] = useState<any[]>([]);
const [searchResults, setSearchResults] = useState<SearchResultsType>({
spotify: [],
youtubeMusic: [],
youtubeVideo: []
});
const [isSearching, setIsSearching] = useState(false);
// 🌟 [추가됨] 봇에게 재생 및 대기열 추가 명령을 내리는 함수
const handleMusicAction = async (actionType: 'player_play' | 'player_playlist', track?: any, playlistUrl?: string) => {
if (!selectedServer) {
alert("명령을 내릴 디스코드 서버가 선택되지 않았습니다.");
// 🌟 alert -> showToast 교체
showToast("명령을 내릴 디스코드 서버가 선택되지 않았습니다.", "error");
return;
}
// 🌟 [추가된 부분] next-auth 세션에서 유저 식별자 가져오기
const userId = (session?.user as any)?.id;
let endpoint = "";
@@ -60,17 +75,19 @@ export default function MainContent({
const data = await res.json();
if (res.ok && data.success) {
// alert(data.message || "요청이 성공했습니다.");
// 🌟 성공 시 초록색 알림
showToast("재생목록에 곡을 추가했습니다!", "success");
} else {
alert(data.message || data.error || "요청 처리 중 문제가 발생했습니다.");
// 🌟 에러 시 빨간색 알림
showToast(data.message || data.error || "요청 처리 중 문제가 발생했습니다.", "error");
}
} catch (error) {
console.error("Music Action Error:", error);
alert('서버와 통신할 수 없습니다.');
// 🌟 네트워크 에러 시 알림
showToast('서버와 통신할 수 없습니다.', "error");
}
};
// 권한 계산 함수
const getPermissionLabel = (server: any) => {
if (!server) return "알 수 없음";
if (server.owner) return "👑 서버 주인";
@@ -84,7 +101,6 @@ export default function MainContent({
}
};
// 1. 서버 목록 불러오기
useEffect(() => {
if (status === "authenticated") {
setIsFetching(true);
@@ -106,17 +122,20 @@ export default function MainContent({
}
}, [status]);
// 2. 검색 결과 불러오기
useEffect(() => {
if (viewMode === "SEARCH_RESULT" && searchQuery) {
setIsSearching(true);
setSearchResults([]);
setSearchResults({ spotify: [], youtubeMusic: [], youtubeVideo: [] });
fetch(`/api/search?q=${encodeURIComponent(searchQuery)}`)
.then(async (res) => {
const data = await res.json();
if (res.ok && Array.isArray(data)) {
setSearchResults(data);
if (res.ok && data) {
setSearchResults({
spotify: data.spotify || [],
youtubeMusic: data.youtubeMusic || [],
youtubeVideo: data.youtubeVideo || []
});
} else {
console.error("Search Error:", data);
}
@@ -126,7 +145,6 @@ export default function MainContent({
}
}, [viewMode, searchQuery]);
// 로그인 안 된 상태
if (status === "unauthenticated") {
return (
<main className="flex-1 flex flex-col items-center justify-center bg-neutral-900">
@@ -137,9 +155,30 @@ export default function MainContent({
);
}
// 세션 로딩 중
if (status === "loading") return <main className="flex-1 bg-neutral-900" />;
const hasAnyResults = searchResults.spotify.length > 0 || searchResults.youtubeMusic.length > 0 || searchResults.youtubeVideo.length > 0;
const renderTrackCard = (track: any) => (
<div key={track.videoId || track.id} className="bg-neutral-800/40 p-3 rounded-xl hover:bg-neutral-800 transition-all group border border-transparent hover:border-neutral-700 shadow-md w-full">
<div className="aspect-square bg-neutral-700 rounded-md mb-2 relative overflow-hidden shadow-lg">
{track.thumbnail && <img src={track.thumbnail} className="w-full h-full object-cover" alt={track.title} />}
<button
onClick={(e) => {
e.stopPropagation();
handleMusicAction('player_play', track);
}}
className="absolute right-2 bottom-2 cursor-pointer bg-green-500 hover:bg-green-400 rounded-full p-2 opacity-0 group-hover:opacity-100 transition-all translate-y-2 group-hover:translate-y-0 shadow-xl"
title="바로 재생"
>
<Play fill="black" stroke="black" size={16} />
</button>
</div>
<h3 className="font-medium text-white text-sm truncate">{track.title}</h3>
<p className="text-xs text-neutral-400 truncate">{track.artist}</p>
</div>
);
return (
<main className="flex-1 flex flex-col bg-gradient-to-b from-neutral-800/50 to-neutral-900 overflow-y-auto p-8">
@@ -153,19 +192,21 @@ export default function MainContent({
<p> ...</p>
</div>
) : (
<div className="grid grid-cols-2 md:grid-cols-3 lg:grid-cols-4 gap-6">
<div className="grid grid-cols-2 sm:grid-cols-3 md:grid-cols-4 xl:grid-cols-5 gap-4">
{servers.map((server) => (
<div
key={server.id}
onClick={() => onSelectServer(server)}
className="bg-neutral-800/40 p-4 rounded-xl hover:bg-neutral-800 transition-all cursor-pointer group border border-transparent hover:border-neutral-700 shadow-md"
className="bg-neutral-800/40 p-3 rounded-xl hover:bg-neutral-800 transition-all cursor-pointer group border border-transparent hover:border-neutral-700 shadow-md w-full"
>
<div className="aspect-square bg-neutral-700 rounded-full mb-4 overflow-hidden shadow-lg flex items-center justify-center">
<div className="aspect-square bg-neutral-700 rounded-full mb-2 overflow-hidden shadow-lg flex items-center justify-center">
{server.icon ? (
<img src={`https://cdn.discordapp.com/icons/${server.id}/${server.icon}.png`} className="w-full h-full object-cover" alt="" />
) : <Server size={32} className="text-neutral-500" />}
) : (
<Server size={24} className="text-neutral-500" />
)}
</div>
<h3 className="font-semibold text-center truncate">{server.name}</h3>
<h3 className="font-medium text-center text-sm truncate">{server.name}</h3>
</div>
))}
</div>
@@ -214,60 +255,61 @@ export default function MainContent({
</div>
)}
{/* 화면 3: 검색 결과 목록 */}
{/* 화면 3: 검색 결과 목록 (3가지 카테고리로 분할) */}
{viewMode === "SEARCH_RESULT" && (
<div>
<h2 className="text-2xl font-bold mb-2">"{searchQuery}" </h2>
<p className="text-neutral-400 mb-8 text-sm"> .</p>
<p className="text-neutral-400 mb-8 text-sm"> .</p>
{isSearching ? (
<div className="flex flex-col items-center py-20">
<Loader2 className="animate-spin text-green-500 mb-4" size={40} />
<p> ...</p>
</div>
) : searchResults.length === 0 ? (
) : !hasAnyResults ? (
<div className="flex flex-col items-center py-20 text-neutral-500">
<SearchX size={48} className="mb-4" />
<p> .</p>
</div>
) : (
<div className="grid grid-cols-2 md:grid-cols-3 lg:grid-cols-4 gap-6">
{searchResults.map((track) => (
<div key={track.videoId} className="bg-neutral-800/40 p-4 rounded-xl hover:bg-neutral-800 transition-all group border border-transparent hover:border-neutral-700 shadow-md">
<div className="flex flex-col gap-10 pb-10">
{/* 🌟 썸네일 이미지 영역 (여기 안에 버튼 2개가 떠다닙니다) */}
<div className="aspect-square bg-neutral-700 rounded-md mb-4 relative overflow-hidden shadow-lg">
{track.thumbnail && <img src={track.thumbnail} className="w-full h-full object-cover" alt={track.title} />}
{/* 🌟 [주석으로 제거] 우측 상단: 재생목록 추가 버튼 (위에서 아래로 스르륵) */}
{/* <button
onClick={(e) => {
e.stopPropagation(); // 부모 div 클릭 이벤트 방지
handleMusicAction('queue_add', track);
}}
className="absolute right-2 top-2 cursor-pointer bg-black/60 hover:bg-black/80 rounded-full p-2 opacity-0 group-hover:opacity-100 transition-all -translate-y-2 group-hover:translate-y-0 shadow-xl text-white backdrop-blur-sm"
title="재생목록에 추가"
>
<ListPlus fill="black" size={25} />
</button> */}
{/* 🌟 [수정됨] 우측 하단: 바로 재생 버튼 (기존 코드 유지 및 onClick 연결) */}
<button
onClick={(e) => {
e.stopPropagation();
handleMusicAction('player_play', track);
}}
className="absolute right-2 bottom-2 cursor-pointer bg-green-500 hover:bg-green-400 rounded-full p-3 opacity-0 group-hover:opacity-100 transition-all translate-y-2 group-hover:translate-y-0 shadow-xl"
title="바로 재생"
>
<Play fill="black" stroke="black" size={20} />
</button>
{/* 카테고리 1: Spotify */}
{searchResults.spotify.length > 0 && (
<section>
<h3 className="text-xl font-bold mb-4 flex items-center gap-2 text-green-400">
<Disc size={24} /> Spotify
</h3>
<div className="grid grid-cols-2 sm:grid-cols-3 md:grid-cols-4 xl:grid-cols-5 gap-4">
{searchResults.spotify.map(renderTrackCard)}
</div>
</section>
)}
{/* 카테고리 2: YouTube Music */}
{searchResults.youtubeMusic.length > 0 && (
<section>
<h3 className="text-xl font-bold mb-4 flex items-center gap-2 text-red-500">
<Music size={24} /> YouTube Music
</h3>
<div className="grid grid-cols-2 sm:grid-cols-3 md:grid-cols-4 xl:grid-cols-5 gap-4">
{searchResults.youtubeMusic.map(renderTrackCard)}
</div>
</section>
)}
{/* 카테고리 3: YouTube Video */}
{searchResults.youtubeVideo.length > 0 && (
<section>
<h3 className="text-xl font-bold mb-4 flex items-center gap-2 text-white">
<MonitorPlay size={24} className="text-red-600" /> YouTube
</h3>
<div className="grid grid-cols-2 sm:grid-cols-3 md:grid-cols-4 xl:grid-cols-5 gap-4">
{searchResults.youtubeVideo.map(renderTrackCard)}
</div>
</section>
)}
<h3 className="font-semibold text-white truncate">{track.title}</h3>
<p className="text-sm text-neutral-400 truncate">{track.artist}</p>
</div>
))}
</div>
)}
</div>

View File

@@ -2,6 +2,7 @@
import { useState, useRef, useEffect, useCallback } from "react";
import { Trash2, GripVertical, Music } from "lucide-react";
import { useSession } from "next-auth/react";
import { useToast } from "@/components/ToastProvider";
interface QueueSidebarProps {
selectedServer: any;
@@ -10,6 +11,9 @@ interface QueueSidebarProps {
export default function QueueSidebar({ selectedServer }: QueueSidebarProps) {
const { data: session, status } = useSession();
// 👇 [추가] 토스트 사용 준비 완료!
const { showToast } = useToast();
const [queue, setQueue] = useState<any[]>([]);
const [isLoading, setIsLoading] = useState(false);
@@ -123,12 +127,12 @@ export default function QueueSidebar({ selectedServer }: QueueSidebarProps) {
});
const data = await res.json();
if (!res.ok || !data.success) {
alert(data.message || data.error || "요청 처리 중 문제가 발생했습니다.");
showToast(data.message || data.error || "요청 처리 중 문제가 발생했습니다.", "error");
fetchQueue(); // 실패 시 원래 큐로 복구
}
} catch (error) {
console.error("순서 동기화 실패:", error);
alert("순서 변경을 서버에 적용하지 못했습니다.");
showToast("순서 변경을 서버에 적용하지 못했습니다.", "error");
fetchQueue();
}
};
@@ -152,12 +156,12 @@ export default function QueueSidebar({ selectedServer }: QueueSidebarProps) {
});
const data = await res.json();
if (!res.ok || !data.success) {
alert(data.message || data.error || "요청 처리 중 문제가 발생했습니다.");
showToast(data.message || data.error || "요청 처리 중 문제가 발생했습니다.", "error");
fetchQueue();
}
} catch (error) {
console.error("삭제 실패:", error);
alert("노래를 삭제하지 못했습니다.");
showToast("노래를 삭제하지 못했습니다.", "error");
fetchQueue();
}
};