Files
YouTube.js/src/parser/classes/comments/Comment.ts
Daniel Wykerd 87ed3960ff refactor!: replace unnecessary classes with pure functions (#468)
* deps: update linkedom

* refactor!: remove YTNodeGenerator in favour of namespaced pure functions

BREAKING CHANGES:
- Removes `YTNodeGenerator` from `import('youtubei.js').Generator` and exposes its functions directly in `import('youtubei.js').Generator`

* refactor!: replace Parser class with pure functions

- Remove Parser class in favour of pure functions
- Merge duplicate classes `AppendContinuationItemsAction` into a single class
- Move continuation parsers into a seperate file
- Add better custom logging support to parser methods as per issue #460

* refactor!: replace Proto class with pure functions

* chore: update package-lock.json

* refactor!: replace FormatUtils with pure functions and JSX components

- Replace linkedom DASH manifest generation with a dependency free JSX implementation
- Remove FormatUtils class in favour of pure functions
- Remove DOMParser requirement
- Remove duplicate types

* refactor: implement changes from #462

* chore: lint

* fix: deno support

* fix: render valid xml document

* fix: wrong function call in DashUtils

* fix: typo in parser

Co-authored-by: LuanRT <luan.lrt4@gmail.com>

* refactor!: move streaming info logic into seperate function

This allows users to access the same data available in the dash manifest while also simplifying the manifest generation

* chore: lint

* refactor: readability improvements & fixes

Remove redundant getAudioTrackGroups
General readability improvements in StreamingInfo.ts
Share response object between `getBitrate` and `getMimeType` as to not make duplicate requests

* build: remove unnecessary step in deno build

Co-authored-by: absidue <48293849+absidue@users.noreply.github.com>

* refactor: move types to `types` directory

* docs: add back comments lost during refactor

* chore: lint

---------

Co-authored-by: LuanRT <luan.lrt4@gmail.com>
Co-authored-by: absidue <48293849+absidue@users.noreply.github.com>
2023-08-18 06:49:58 -03:00

182 lines
6.2 KiB
TypeScript

import Parser from '../../index.js';
import Author from '../misc/Author.js';
import Text from '../misc/Text.js';
import Thumbnail from '../misc/Thumbnail.js';
import Menu from '../menus/Menu.js';
import AuthorCommentBadge from './AuthorCommentBadge.js';
import CommentActionButtons from './CommentActionButtons.js';
import CommentReplyDialog from './CommentReplyDialog.js';
import PdgCommentChip from './PdgCommentChip.js';
import SponsorCommentBadge from './SponsorCommentBadge.js';
import * as Proto from '../../../proto/index.js';
import { InnertubeError } from '../../../utils/Utils.js';
import { YTNode } from '../../helpers.js';
import type Actions from '../../../core/Actions.js';
import type { ApiResponse } from '../../../core/Actions.js';
import type { RawNode } from '../../index.js';
export default class Comment extends YTNode {
static type = 'Comment';
#actions?: Actions;
content: Text;
published: Text;
author_is_channel_owner: boolean;
current_user_reply_thumbnail: Thumbnail[];
sponsor_comment_badge: SponsorCommentBadge | null;
paid_comment_chip: PdgCommentChip | null;
author_badge: AuthorCommentBadge | null;
author: Author;
action_menu: Menu | null;
action_buttons: CommentActionButtons | null;
comment_id: string;
vote_status: string;
vote_count: string;
reply_count: number;
is_liked: boolean;
is_disliked: boolean;
is_hearted: boolean;
is_pinned: boolean;
is_member: boolean;
constructor(data: RawNode) {
super();
this.content = new Text(data.contentText);
this.published = new Text(data.publishedTimeText);
this.author_is_channel_owner = data.authorIsChannelOwner;
this.current_user_reply_thumbnail = Thumbnail.fromResponse(data.currentUserReplyThumbnail);
this.sponsor_comment_badge = Parser.parseItem(data.sponsorCommentBadge, SponsorCommentBadge);
this.paid_comment_chip = Parser.parseItem(data.paidCommentChipRenderer, PdgCommentChip);
this.author_badge = Parser.parseItem(data.authorCommentBadge, AuthorCommentBadge);
this.author = new Author({
...data.authorText,
navigationEndpoint: data.authorEndpoint
}, this.author_badge ? [ {
metadataBadgeRenderer: this.author_badge?.orig_badge
} ] : null, data.authorThumbnail);
this.action_menu = Parser.parseItem(data.actionMenu, Menu);
this.action_buttons = Parser.parseItem(data.actionButtons, CommentActionButtons);
this.comment_id = data.commentId;
this.vote_status = data.voteStatus;
this.vote_count = data.voteCount ? new Text(data.voteCount).toString() : '0';
this.reply_count = data.replyCount || 0;
this.is_liked = !!this.action_buttons?.like_button?.is_toggled;
this.is_disliked = !!this.action_buttons?.dislike_button?.is_toggled;
this.is_hearted = !!this.action_buttons?.creator_heart?.is_hearted;
this.is_pinned = !!data.pinnedCommentBadge;
this.is_member = !!data.sponsorCommentBadge;
}
/**
* Likes the comment.
*/
async like(): Promise<ApiResponse> {
if (!this.#actions)
throw new InnertubeError('An active caller must be provide to perform this operation.');
const button = this.action_buttons?.like_button;
if (!button)
throw new InnertubeError('Like button was not found.', { comment_id: this.comment_id });
if (button.is_toggled)
throw new InnertubeError('This comment is already liked', { comment_id: this.comment_id });
const response = await button.endpoint.call(this.#actions, { parse: false });
return response;
}
/**
* Dislikes the comment.
*/
async dislike(): Promise<ApiResponse> {
if (!this.#actions)
throw new InnertubeError('An active caller must be provide to perform this operation.');
const button = this.action_buttons?.dislike_button;
if (!button)
throw new InnertubeError('Dislike button was not found.', { comment_id: this.comment_id });
if (button.is_toggled)
throw new InnertubeError('This comment is already disliked', { comment_id: this.comment_id });
const response = await button.endpoint.call(this.#actions, { parse: false });
return response;
}
/**
* Creates a reply to the comment.
*/
async reply(text: string): Promise<ApiResponse> {
if (!this.#actions)
throw new InnertubeError('An active caller must be provide to perform this operation.');
if (!this.action_buttons?.reply_button)
throw new InnertubeError('Cannot reply to another reply. Try mentioning the user instead.', { comment_id: this.comment_id });
const button = this.action_buttons?.reply_button;
if (!button.endpoint?.dialog)
throw new InnertubeError('Reply button endpoint did not have a dialog.');
const dialog = button.endpoint.dialog.as(CommentReplyDialog);
const dialog_button = dialog.reply_button;
if (!dialog_button)
throw new InnertubeError('Reply button was not found in the dialog.', { comment_id: this.comment_id });
if (!dialog_button.endpoint)
throw new InnertubeError('Reply button endpoint was not found.', { comment_id: this.comment_id });
const response = await dialog_button.endpoint.call(this.#actions, { commentText: text });
return response;
}
/**
* Translates the comment to a given language.
* @param target_language - Ex; en, ja
*/
async translate(target_language: string): Promise<{
content: any;
success: boolean;
status_code: number;
data: any;
}> {
if (!this.#actions)
throw new InnertubeError('An active caller must be provide to perform this operation.');
// Emojis must be removed otherwise InnerTube throws a 400 status code at us.
const text = this.content.toString().replace(/[^\p{L}\p{N}\p{P}\p{Z}]/gu, '');
const payload = {
text,
target_language,
comment_id: this.comment_id
};
const action = Proto.encodeCommentActionParams(22, payload);
const response = await this.#actions.execute('comment/perform_comment_action', { action, client: 'ANDROID' });
// XXX: Should move this to Parser#parseResponse
const mutations = response.data.frameworkUpdates?.entityBatchUpdate?.mutations;
const content = mutations?.[0]?.payload?.commentEntityPayload?.translatedContent?.content;
return { ...response, content };
}
setActions(actions: Actions | undefined) {
this.#actions = actions;
}
}