mirror of
https://gitlab.com/foxixus/neomovies.git
synced 2025-10-27 17:38:50 +05:00
- /src/api.ts - /src/lib/utils.ts - /src/lib/neoApi.ts - /src/lib/mongodb.ts - /src/lib/favoritesApi.ts - /src/lib/models/Favorite.ts - /src/hooks/useTMDBMovies.ts - /src/hooks/useImageLoader.ts - /src/hooks/useMovies.ts - /src/types/movie.ts - /src/components/SearchResults.tsx - /src/components/SettingsContent.tsx - /src/components/MovieCard.tsx - /src/components/FavoriteButton.tsx - /src/components/admin/MovieSearch.tsx - /src/app/page.tsx - /src/app/movie/[id]/page.tsx - /src/app/movie/[id]/MovieContent.tsx - /src/app/api/movies/upcoming/route.ts - /src/app/api/movies/search/route.ts - /src/app/api/movies/top-rated/route.ts - /src/app/api/movies/[id]/route.ts - /src/app/api/movies/popular/route.ts - /src/app/api/favorites/route.ts - /src/app/api/favorites/check/[mediaId]/route.ts - /src/app/api/favorites/[mediaId]/route.ts - /src/app/tv/[id]/TVShowContent.tsx - /src/app/tv/[id]/TVShowPage.tsx - /src/app/tv/[id]/page.tsx - /src/app/favorites/page.tsx - /src/configs/auth.ts - /next.config.js - /package.json - /README.md - /package-lock.json
170 lines
4.1 KiB
TypeScript
170 lines
4.1 KiB
TypeScript
'use client';
|
|
|
|
import { useState } from 'react';
|
|
import { debounce } from 'lodash';
|
|
import { getImageUrl } from '@/lib/neoApi';
|
|
|
|
interface Movie {
|
|
id: number;
|
|
title: string;
|
|
overview: string;
|
|
release_date: string;
|
|
vote_average: number;
|
|
poster_path: string | null;
|
|
genre_ids: number[];
|
|
}
|
|
|
|
interface MovieCardProps {
|
|
children: React.ReactNode;
|
|
}
|
|
|
|
const MovieCard: React.FC<MovieCardProps> = ({ children }) => {
|
|
return (
|
|
<div className="bg-gray-800 rounded-lg overflow-hidden">
|
|
{children}
|
|
</div>
|
|
);
|
|
};
|
|
|
|
interface PosterContainerProps {
|
|
children: React.ReactNode;
|
|
}
|
|
|
|
const PosterContainer: React.FC<PosterContainerProps> = ({ children }) => {
|
|
return (
|
|
<div className="aspect-w-2 aspect-h-3">
|
|
{children}
|
|
</div>
|
|
);
|
|
};
|
|
|
|
interface ImageProps extends React.ImgHTMLAttributes<HTMLImageElement> {
|
|
src: string;
|
|
alt: string;
|
|
width: number;
|
|
height: number;
|
|
}
|
|
|
|
const Image: React.FC<ImageProps> = ({ src, alt, width, height, ...props }) => {
|
|
return (
|
|
<img
|
|
src={src}
|
|
alt={alt}
|
|
width={width}
|
|
height={height}
|
|
className="object-cover w-full h-full"
|
|
{...props}
|
|
/>
|
|
);
|
|
};
|
|
|
|
interface MovieInfoProps {
|
|
children: React.ReactNode;
|
|
}
|
|
|
|
const MovieInfo: React.FC<MovieInfoProps> = ({ children }) => {
|
|
return (
|
|
<div className="p-4">
|
|
{children}
|
|
</div>
|
|
);
|
|
};
|
|
|
|
interface TitleProps {
|
|
children: React.ReactNode;
|
|
}
|
|
|
|
const Title: React.FC<TitleProps> = ({ children }) => {
|
|
return (
|
|
<h3 className="font-semibold text-lg mb-2">{children}</h3>
|
|
);
|
|
};
|
|
|
|
interface YearProps {
|
|
children: React.ReactNode;
|
|
}
|
|
|
|
const Year: React.FC<YearProps> = ({ children }) => {
|
|
return (
|
|
<p className="text-sm text-gray-400 mb-4">{children}</p>
|
|
);
|
|
};
|
|
|
|
export default function MovieSearch() {
|
|
const [searchQuery, setSearchQuery] = useState('');
|
|
const [searchResults, setSearchResults] = useState<Movie[]>([]);
|
|
const [loading, setLoading] = useState(false);
|
|
|
|
const searchMovies = debounce(async (query: string) => {
|
|
if (!query.trim()) {
|
|
setSearchResults([]);
|
|
return;
|
|
}
|
|
|
|
try {
|
|
setLoading(true);
|
|
const response = await fetch(
|
|
`/api/movies/search?query=${encodeURIComponent(query)}`
|
|
);
|
|
const data = await response.json();
|
|
setSearchResults(data.results || []);
|
|
} catch (error) {
|
|
console.error('Error searching movies:', error);
|
|
} finally {
|
|
setLoading(false);
|
|
}
|
|
}, 500);
|
|
|
|
const handleSearch = (e: React.ChangeEvent<HTMLInputElement>) => {
|
|
const query = e.target.value;
|
|
setSearchQuery(query);
|
|
searchMovies(query);
|
|
};
|
|
|
|
return (
|
|
<div>
|
|
<div className="relative mb-4">
|
|
<input
|
|
type="text"
|
|
value={searchQuery}
|
|
onChange={handleSearch}
|
|
placeholder="Поиск фильмов..."
|
|
className="w-full px-4 py-2 bg-gray-800 rounded-lg focus:outline-none focus:ring-2 focus:ring-blue-500"
|
|
/>
|
|
{loading && (
|
|
<div className="absolute right-3 top-2">
|
|
<div className="animate-spin rounded-full h-6 w-6 border-b-2 border-white"></div>
|
|
</div>
|
|
)}
|
|
</div>
|
|
|
|
{searchResults.length > 0 && (
|
|
<div className="grid grid-cols-1 md:grid-cols-2 lg:grid-cols-3 gap-4">
|
|
{searchResults.map((movie) => (
|
|
<MovieCard key={movie.id}>
|
|
<PosterContainer>
|
|
<Image
|
|
src={movie.poster_path ? getImageUrl(movie.poster_path) : '/placeholder.jpg'}
|
|
alt={movie.title}
|
|
width={200}
|
|
height={300}
|
|
/>
|
|
</PosterContainer>
|
|
<MovieInfo>
|
|
<Title>{movie.title}</Title>
|
|
<Year>{new Date(movie.release_date).getFullYear()}</Year>
|
|
<p className="text-sm text-gray-400 mb-4">
|
|
{movie.vote_average.toFixed(1)} ⭐
|
|
</p>
|
|
<p className="text-sm text-gray-400 line-clamp-3 mb-4">
|
|
{movie.overview}
|
|
</p>
|
|
</MovieInfo>
|
|
</MovieCard>
|
|
))}
|
|
</div>
|
|
)}
|
|
</div>
|
|
);
|
|
}
|