191 lines
6.2 KiB
TypeScript
191 lines
6.2 KiB
TypeScript
import { useOutletContext } from 'react-router-dom';
|
|
import loadChannelList from '../api/loader/loadChannelList';
|
|
import iconGridView from '/img/icon-gridview.svg';
|
|
import iconListView from '/img/icon-listview.svg';
|
|
import iconAdd from '/img/icon-add.svg';
|
|
import { useEffect, useState } from 'react';
|
|
import Pagination, { PaginationType } from '../components/Pagination';
|
|
import { ConfigType } from './Home';
|
|
import { OutletContextType } from './Base';
|
|
import ChannelList from '../components/ChannelList';
|
|
import ScrollToTopOnNavigate from '../components/ScrollToTop';
|
|
import Notifications from '../components/Notifications';
|
|
import Button from '../components/Button';
|
|
import updateChannelSubscription from '../api/actions/updateChannelSubscription';
|
|
import loadIsAdmin from '../functions/getIsAdmin';
|
|
import { useUserConfigStore } from '../stores/UserConfigStore';
|
|
|
|
type ChannelOverwritesType = {
|
|
download_format?: string;
|
|
autodelete_days?: number;
|
|
index_playlists?: boolean;
|
|
integrate_sponsorblock?: boolean;
|
|
subscriptions_channel_size?: number;
|
|
subscriptions_live_channel_size?: number;
|
|
subscriptions_shorts_channel_size?: number;
|
|
};
|
|
|
|
export type ChannelType = {
|
|
channel_active: boolean;
|
|
channel_banner_url: string;
|
|
channel_description: string;
|
|
channel_id: string;
|
|
channel_last_refresh: string;
|
|
channel_name: string;
|
|
channel_overwrites?: ChannelOverwritesType;
|
|
channel_subs: number;
|
|
channel_subscribed: boolean;
|
|
channel_tags: string[];
|
|
channel_thumb_url: string;
|
|
channel_tvart_url: string;
|
|
channel_views: number;
|
|
};
|
|
|
|
type ChannelsListResponse = {
|
|
data: ChannelType[];
|
|
paginate: PaginationType;
|
|
config?: ConfigType;
|
|
};
|
|
|
|
const Channels = () => {
|
|
const { userConfig, setPartialConfig } = useUserConfigStore();
|
|
const { currentPage, setCurrentPage } = useOutletContext() as OutletContextType;
|
|
const isAdmin = loadIsAdmin();
|
|
|
|
const [channelListResponse, setChannelListResponse] = useState<ChannelsListResponse>();
|
|
const [showAddForm, setShowAddForm] = useState(false);
|
|
const [refresh, setRefresh] = useState(false);
|
|
const [channelsToSubscribeTo, setChannelsToSubscribeTo] = useState('');
|
|
|
|
const channels = channelListResponse?.data;
|
|
const pagination = channelListResponse?.paginate;
|
|
const channelCount = pagination?.total_hits;
|
|
const hasChannels = channels?.length !== 0;
|
|
|
|
useEffect(() => {
|
|
(async () => {
|
|
const channelListResponse = await loadChannelList(
|
|
currentPage,
|
|
userConfig.config.show_subed_only,
|
|
);
|
|
setChannelListResponse(channelListResponse);
|
|
})();
|
|
}, [refresh, userConfig.config.show_subed_only, currentPage, pagination?.current_page]);
|
|
|
|
return (
|
|
<>
|
|
<title>TA | Channels</title>
|
|
<ScrollToTopOnNavigate />
|
|
<div className="boxed-content">
|
|
<div className="title-split">
|
|
<div className="title-bar">
|
|
<h1>Channels</h1>
|
|
</div>
|
|
{isAdmin && (
|
|
<div className="title-split-form">
|
|
<img
|
|
id="animate-icon"
|
|
onClick={() => {
|
|
setShowAddForm(!showAddForm);
|
|
}}
|
|
src={iconAdd}
|
|
alt="add-icon"
|
|
title="Subscribe to Channels"
|
|
/>
|
|
{showAddForm && (
|
|
<div className="show-form">
|
|
<div>
|
|
<label>Subscribe to channels:</label>
|
|
<textarea
|
|
value={channelsToSubscribeTo}
|
|
onChange={e => {
|
|
setChannelsToSubscribeTo(e.currentTarget.value);
|
|
}}
|
|
rows={3}
|
|
placeholder="Input channel ID, URL or Video of a channel"
|
|
/>
|
|
</div>
|
|
|
|
<Button
|
|
label="Subscribe"
|
|
type="submit"
|
|
onClick={async () => {
|
|
await updateChannelSubscription(channelsToSubscribeTo, true);
|
|
|
|
setRefresh(true);
|
|
}}
|
|
/>
|
|
</div>
|
|
)}
|
|
</div>
|
|
)}
|
|
</div>
|
|
|
|
<Notifications pageName="all" />
|
|
|
|
<div className="view-controls">
|
|
<div className="toggle">
|
|
<span>Show subscribed only:</span>
|
|
<div className="toggleBox">
|
|
<input
|
|
id="show_subed_only"
|
|
onChange={async () => {
|
|
setPartialConfig({ show_subed_only: !userConfig.config.show_subed_only });
|
|
setRefresh(true);
|
|
}}
|
|
type="checkbox"
|
|
checked={userConfig.config.show_subed_only}
|
|
/>
|
|
{!userConfig.config.show_subed_only && (
|
|
<label htmlFor="" className="ofbtn">
|
|
Off
|
|
</label>
|
|
)}
|
|
{userConfig.config.show_subed_only && (
|
|
<label htmlFor="" className="onbtn">
|
|
On
|
|
</label>
|
|
)}
|
|
</div>
|
|
</div>
|
|
<div className="view-icons">
|
|
<img
|
|
src={iconGridView}
|
|
onClick={() => {
|
|
setPartialConfig({ view_style_channel: 'grid' });
|
|
}}
|
|
data-origin="channel"
|
|
data-value="grid"
|
|
alt="grid view"
|
|
/>
|
|
<img
|
|
src={iconListView}
|
|
onClick={() => {
|
|
setPartialConfig({ view_style_channel: 'list' });
|
|
}}
|
|
data-origin="channel"
|
|
data-value="list"
|
|
alt="list view"
|
|
/>
|
|
</div>
|
|
</div>
|
|
{hasChannels && <h2>Total channels: {channelCount}</h2>}
|
|
|
|
<div className={`channel-list ${userConfig.config.view_style_channel}`}>
|
|
{!hasChannels && <h2>No channels found...</h2>}
|
|
|
|
{hasChannels && <ChannelList channelList={channels} refreshChannelList={setRefresh} />}
|
|
</div>
|
|
|
|
{pagination && (
|
|
<div className="boxed-content">
|
|
<Pagination pagination={pagination} setPage={setCurrentPage} />
|
|
</div>
|
|
)}
|
|
</div>
|
|
</>
|
|
);
|
|
};
|
|
|
|
export default Channels;
|