Add Popular tab to channel page

- YouTube sort=p fetch: indexes top 100 most-viewed videos from a channel,
  storing view_count in the DB
- Popular tab on channel page shows videos sorted by view_count DESC
- Videos/Popular tab switcher with context-appropriate fetch buttons
- Expose view_count in VideoOut; add 'popular' sort to channel videos endpoint

Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
This commit is contained in:
2026-05-26 22:22:10 +02:00
parent aa91156bbc
commit d31fc1ef7f
3 changed files with 146 additions and 27 deletions

View File

@@ -62,6 +62,7 @@ class VideoOut(BaseModel):
is_downloaded: bool = False
is_watched: bool = False
queued: bool = False
view_count: Optional[int] = None
model_config = {"from_attributes": True}
@@ -611,7 +612,8 @@ def get_channel_videos(
"newest": "v.published_at DESC NULLS LAST",
"oldest": "v.published_at ASC NULLS LAST",
"title": "v.title ASC",
"unwatched": "COALESCE(uv.watched, 0) ASC, v.published_at DESC NULLS LAST",
"unwatched":"COALESCE(uv.watched, 0) ASC, v.published_at DESC NULLS LAST",
"popular": "v.view_count DESC NULLS LAST",
}.get(sort, "v.published_at DESC NULLS LAST")
params: dict = {"user_id": current_user.id, "channel_id": channel_id, "limit": limit, "offset": offset}
q_clause = ""
@@ -621,7 +623,7 @@ def get_channel_videos(
rows = db.execute(
text(f"""
SELECT v.id, v.youtube_video_id, v.title, v.thumbnail_url,
v.duration_seconds, v.published_at,
v.duration_seconds, v.published_at, v.view_count,
COALESCE(uv.downloaded, 0) AS is_downloaded,
COALESCE(uv.watched, 0) AS is_watched
FROM videos v
@@ -635,6 +637,77 @@ def get_channel_videos(
return [VideoOut(**dict(r)) for r in rows]
@router.post("/{channel_id}/fetch-popular", status_code=status.HTTP_202_ACCEPTED)
def fetch_popular_videos(
channel_id: int,
background_tasks: BackgroundTasks,
db: Session = Depends(get_db),
current_user: User = Depends(get_current_user),
):
"""Fetch the channel's most popular videos from YouTube and index them."""
channel = _get_channel_or_404(db, channel_id)
background_tasks.add_task(_fetch_popular_task, channel_id, channel.youtube_channel_id)
return {"detail": "Fetching popular videos"}
def _fetch_popular_task(channel_id: int, youtube_channel_id: str):
from ..database import SessionLocal
db = SessionLocal()
try:
if youtube_channel_id.startswith("@"):
url = f"https://www.youtube.com/{youtube_channel_id}/videos?sort=p"
else:
url = f"https://www.youtube.com/channel/{youtube_channel_id}/videos?sort=p"
stdout, _, code = ytdlp._run([
"yt-dlp", url,
"--dump-json", "--flat-playlist",
"--playlist-end", "100",
"--quiet",
*ytdlp._cookie_args(),
], timeout=120)
channel = db.query(Channel).filter_by(id=channel_id).first()
if not channel:
return
for line in stdout.splitlines():
line = line.strip()
if not line:
continue
try:
info = json.loads(line)
except json.JSONDecodeError:
continue
yt_id = info.get("id")
if not yt_id:
continue
existing = db.query(Video).filter_by(youtube_video_id=yt_id).first()
view_count = info.get("view_count")
published_at = ytdlp._parse_published(info)
if existing:
if view_count is not None:
existing.view_count = view_count
if published_at and not existing.published_at:
existing.published_at = published_at
else:
db.add(Video(
youtube_video_id=yt_id,
channel_id=channel.id,
title=info.get("title", ""),
thumbnail_url=ytdlp._stable_thumbnail(yt_id),
duration_seconds=info.get("duration"),
published_at=published_at,
tags=json.dumps(info.get("tags") or []),
view_count=view_count,
))
db.commit()
except Exception:
db.rollback()
finally:
db.close()
@router.post("/{channel_id}/search", status_code=status.HTTP_202_ACCEPTED)
def search_channel_youtube(
channel_id: int,

View File

@@ -49,6 +49,7 @@ export const unfollowChannel = (id) => api.delete(`/channels/${id}/follow`);
export const indexChannel = (id) => api.post(`/channels/${id}/index`);
export const indexChannelFull = (id) => api.post(`/channels/${id}/index-full`);
export const exploreChannelOlder = (id, page) => api.post(`/channels/${id}/explore`, null, { params: { page } });
export const fetchPopularVideos = (id) => api.post(`/channels/${id}/fetch-popular`);
export const followChannelByUrl = (data) => api.post("/channels/follow-by-url", data);
export const setChannelAutoDownload = (id, value) => api.patch(`/channels/${id}/auto-download`, { auto_download: value });
export const markChannelsSeen = () => api.post("/channels/mark-seen");

View File

@@ -3,12 +3,17 @@ import { useParams } from "react-router-dom";
import { useQuery, useMutation, useQueryClient, useInfiniteQuery } from "@tanstack/react-query";
import {
getChannel, getChannelVideos, searchChannelYoutube,
followChannel, unfollowChannel, indexChannel, indexChannelFull, exploreChannelOlder, downloadChannel,
followChannel, unfollowChannel, indexChannel, indexChannelFull, exploreChannelOlder, fetchPopularVideos, downloadChannel,
} from "../api";
import VideoCard from "../components/VideoCard";
const LIMIT = 60;
const TABS = [
{ value: "videos", label: "Videos" },
{ value: "popular", label: "Popular" },
];
const SORTS = [
{ value: "newest", label: "Newest" },
{ value: "oldest", label: "Oldest" },
@@ -26,6 +31,7 @@ function formatSubs(n) {
export default function ChannelPage() {
const { id } = useParams();
const qc = useQueryClient();
const [tab, setTab] = useState("videos");
const [sort, setSort] = useState("newest");
const [search, setSearch] = useState("");
const [activeQ, setActiveQ] = useState("");
@@ -38,6 +44,8 @@ export default function ChannelPage() {
queryFn: () => getChannel(id).then((r) => r.data),
});
const effectiveSort = tab === "popular" ? "popular" : sort;
const {
data: videosData,
isLoading: loadingVideos,
@@ -45,9 +53,9 @@ export default function ChannelPage() {
hasNextPage,
isFetchingNextPage,
} = useInfiniteQuery({
queryKey: ["channel-videos", id, sort, activeQ],
queryKey: ["channel-videos", id, effectiveSort, activeQ],
queryFn: ({ pageParam = 0 }) =>
getChannelVideos(id, sort, pageParam, LIMIT, activeQ).then((r) => r.data),
getChannelVideos(id, effectiveSort, pageParam, LIMIT, activeQ).then((r) => r.data),
getNextPageParam: (lastPage, pages) =>
lastPage.length === LIMIT ? pages.length * LIMIT : undefined,
enabled: !!id,
@@ -103,6 +111,11 @@ export default function ChannelPage() {
},
});
const popularMut = useMutation({
mutationFn: () => fetchPopularVideos(id),
onSuccess: () => scheduleRefetch(20000),
});
const deepSearchMut = useMutation({
mutationFn: () => searchChannelYoutube(id, activeQ || search),
onSuccess: () => scheduleRefetch(20000),
@@ -139,7 +152,7 @@ export default function ChannelPage() {
if (!channel) return <p className="text-zinc-500">Channel not found.</p>;
const isFollowed = channel.status === "followed";
const isPending = indexMut.isPending || fullIndexMut.isPending || deepSearchMut.isPending || exploreMut.isPending || indexing;
const isPending = indexMut.isPending || fullIndexMut.isPending || deepSearchMut.isPending || exploreMut.isPending || popularMut.isPending || indexing;
return (
<div className="flex flex-col gap-5">
@@ -240,15 +253,15 @@ export default function ChannelPage() {
)}
</form>
{/* Sort + index controls */}
<div className="flex items-center justify-between gap-3 -mt-1">
{/* Tabs + controls */}
<div className="flex items-center justify-between gap-3 -mt-1 border-b border-zinc-800/60 pb-3">
<div className="flex items-center gap-0.5">
{SORTS.map(s => (
<button key={s.value} onClick={() => setSort(s.value)}
className={`text-xs px-2.5 py-1 rounded-md transition-colors ${
sort === s.value ? "bg-zinc-700 text-zinc-100" : "text-zinc-500 hover:text-zinc-300"
{TABS.map(t => (
<button key={t.value} onClick={() => setTab(t.value)}
className={`text-sm px-3 py-1 rounded-md transition-colors font-medium ${
tab === t.value ? "text-zinc-100" : "text-zinc-500 hover:text-zinc-300"
}`}>
{s.label}
{t.label}
</button>
))}
</div>
@@ -260,9 +273,16 @@ export default function ChannelPage() {
<circle className="opacity-25" cx="12" cy="12" r="10" stroke="currentColor" strokeWidth="3" />
<path className="opacity-75" fill="currentColor" d="M4 12a8 8 0 018-8v4a4 4 0 00-4 4H4z" />
</svg>
Indexing
Fetching
</span>
)}
{tab === "popular" ? (
<button onClick={() => popularMut.mutate()} disabled={isPending}
className="text-xs text-zinc-400 hover:text-zinc-100 font-medium transition-colors disabled:opacity-40">
Fetch popular
</button>
) : (
<>
<button onClick={() => indexMut.mutate()} disabled={isPending}
className="text-xs text-zinc-500 hover:text-zinc-300 transition-colors disabled:opacity-40">
Fetch recent
@@ -271,9 +291,25 @@ export default function ChannelPage() {
className="text-xs text-zinc-400 hover:text-zinc-100 font-medium transition-colors disabled:opacity-40">
Fetch all
</button>
</>
)}
</div>
</div>
{/* Sort bar — videos tab only */}
{tab === "videos" && (
<div className="flex items-center gap-0.5 -mt-1">
{SORTS.map(s => (
<button key={s.value} onClick={() => setSort(s.value)}
className={`text-xs px-2.5 py-1 rounded-md transition-colors ${
sort === s.value ? "bg-zinc-700 text-zinc-100" : "text-zinc-500 hover:text-zinc-300"
}`}>
{s.label}
</button>
))}
</div>
)}
{/* Video list */}
{loadingVideos ? (
<div className="flex items-center justify-center py-8">
@@ -290,7 +326,7 @@ export default function ChannelPage() {
className="mt-4 self-center text-sm text-zinc-500 hover:text-zinc-300 transition-colors disabled:opacity-40 py-2 px-4">
{isFetchingNextPage ? "Loading…" : "Load more"}
</button>
) : !activeQ && (
) : !activeQ && tab === "videos" && (
<div className="mt-4 flex flex-col items-center gap-3 py-4 border-t border-zinc-800/50">
<p className="text-xs text-zinc-600">{videos.length} videos indexed</p>
<div className="flex items-center gap-4">
@@ -310,13 +346,22 @@ export default function ChannelPage() {
) : (
<div className="py-8 flex flex-col items-center gap-3">
<p className="text-zinc-500 text-sm">
{activeQ ? `No indexed videos match "${activeQ}"` : "No videos indexed yet."}
{activeQ
? `No indexed videos match "${activeQ}"`
: tab === "popular"
? "No popular videos fetched yet."
: "No videos indexed yet."}
</p>
{activeQ ? (
<button onClick={() => deepSearchMut.mutate()} disabled={isPending}
className="text-sm text-zinc-400 hover:text-zinc-100 transition-colors disabled:opacity-40 underline">
Search YouTube for "{activeQ}"
</button>
) : tab === "popular" ? (
<button onClick={() => popularMut.mutate()} disabled={isPending}
className="text-sm text-zinc-400 hover:text-zinc-100 transition-colors disabled:opacity-40 underline">
Fetch popular videos from YouTube
</button>
) : (
<button onClick={() => fullIndexMut.mutate()} disabled={isPending}
className="text-sm text-zinc-400 hover:text-zinc-100 transition-colors disabled:opacity-40 underline">