mirror of
https://github.com/LuanRT/YouTube.js.git
synced 2026-06-19 20:41:17 +00:00
91 lines
2.8 KiB
TypeScript
91 lines
2.8 KiB
TypeScript
import type Actions from '../../core/Actions.js';
|
|
import { InnertubeError } from '../../utils/Utils.js';
|
|
|
|
import Feed from '../../core/mixins/Feed.js';
|
|
import History from './History.js';
|
|
import Playlist from './Playlist.js';
|
|
import Menu from '../classes/menus/Menu.js';
|
|
import Shelf from '../classes/Shelf.js';
|
|
import Button from '../classes/Button.js';
|
|
|
|
import ProfileColumnStats from '../classes/ProfileColumnStats.js';
|
|
import ProfileColumnUserInfo from '../classes/ProfileColumnUserInfo.js';
|
|
|
|
import type { IBrowseResponse } from '../types/ParsedResponse.js';
|
|
import type { ApiResponse } from '../../core/Actions.js';
|
|
|
|
class Library extends Feed<IBrowseResponse> {
|
|
profile: {
|
|
stats?: ProfileColumnStats;
|
|
user_info?: ProfileColumnUserInfo;
|
|
};
|
|
|
|
sections;
|
|
|
|
constructor(actions: Actions, data: ApiResponse | IBrowseResponse) {
|
|
super(actions, data);
|
|
|
|
if (!this.page.contents_memo)
|
|
throw new InnertubeError('Page contents not found');
|
|
|
|
const stats = this.page.contents_memo.getType(ProfileColumnStats).first();
|
|
const user_info = this.page.contents_memo.getType(ProfileColumnUserInfo).first();
|
|
|
|
this.profile = { stats, user_info };
|
|
|
|
const shelves = this.page.contents_memo.getType(Shelf);
|
|
|
|
this.sections = shelves.map((shelf: Shelf) => ({
|
|
type: shelf.icon_type,
|
|
title: shelf.title,
|
|
contents: shelf.content?.key('items').array() || [],
|
|
getAll: () => this.#getAll(shelf)
|
|
}));
|
|
}
|
|
|
|
async #getAll(shelf: Shelf): Promise<Playlist | History | Feed> {
|
|
if (!shelf.menu?.as(Menu).hasKey('top_level_buttons'))
|
|
throw new InnertubeError(`The ${shelf.title.text} shelf doesn't have more items`);
|
|
|
|
const button = shelf.menu.as(Menu).top_level_buttons.firstOfType(Button);
|
|
|
|
if (!button)
|
|
throw new InnertubeError('Did not find target button.');
|
|
|
|
const page = await button.as(Button).endpoint.call<IBrowseResponse>(this.actions, { parse: true });
|
|
|
|
switch (shelf.icon_type) {
|
|
case 'LIKE':
|
|
case 'WATCH_LATER':
|
|
return new Playlist(this.actions, page, true);
|
|
case 'WATCH_HISTORY':
|
|
return new History(this.actions, page, true);
|
|
case 'CONTENT_CUT':
|
|
return new Feed(this.actions, page, true);
|
|
default:
|
|
throw new InnertubeError('Target shelf not implemented.');
|
|
}
|
|
}
|
|
|
|
get history() {
|
|
return this.sections.find((section) => section.type === 'WATCH_HISTORY');
|
|
}
|
|
|
|
get watch_later() {
|
|
return this.sections.find((section) => section.type === 'WATCH_LATER');
|
|
}
|
|
|
|
get liked_videos() {
|
|
return this.sections.find((section) => section.type === 'LIKE');
|
|
}
|
|
|
|
get playlists_section() {
|
|
return this.sections.find((section) => section.type === 'PLAYLISTS');
|
|
}
|
|
|
|
get clips() {
|
|
return this.sections.find((section) => section.type === 'CONTENT_CUT');
|
|
}
|
|
}
|
|
|
|
export default Library; |