-import { Left, None, Option, Some } from "@sniptt/monads";
import classNames from "classnames";
import { Component, linkEvent } from "inferno";
import { Link } from "inferno-router";
BanFromCommunity,
BanPerson,
BlockPerson,
- CommentNode as CommentNodeI,
CommentReplyView,
CommentView,
CommunityModeratorView,
CreateCommentLike,
CreateCommentReport,
DeleteComment,
- EditComment,
+ DistinguishComment,
GetComments,
- ListingType,
+ Language,
MarkCommentReplyAsRead,
MarkPersonMentionAsRead,
PersonMentionView,
- PersonViewSafe,
+ PersonView,
PurgeComment,
PurgePerson,
RemoveComment,
SaveComment,
- toUndefined,
TransferCommunity,
} from "lemmy-js-client";
import moment from "moment";
import { i18n } from "../../i18next";
-import { BanType, CommentViewType, PurgeType } from "../../interfaces";
+import {
+ BanType,
+ CommentNodeI,
+ CommentViewType,
+ PurgeType,
+} from "../../interfaces";
import { UserService, WebSocketService } from "../../services";
import {
amCommunityCreator,
- auth,
canAdmin,
canMod,
colorList,
commentTreeMaxDepth,
futureDaysToUnixTime,
+ getCommentParentId,
isAdmin,
isBanned,
isMod,
mdToHtml,
+ mdToHtmlNoImages,
+ myAuth,
numToSI,
setupTippy,
showScores,
showReply: boolean;
showEdit: boolean;
showRemoveDialog: boolean;
- removeReason: Option<string>;
+ removeReason?: string;
showBanDialog: boolean;
removeData: boolean;
- banReason: Option<string>;
- banExpireDays: Option<number>;
+ banReason?: string;
+ banExpireDays?: number;
banType: BanType;
showPurgeDialog: boolean;
- purgeReason: Option<string>;
+ purgeReason?: string;
purgeType: PurgeType;
purgeLoading: boolean;
showConfirmTransferSite: boolean;
viewSource: boolean;
showAdvanced: boolean;
showReportDialog: boolean;
- reportReason: string;
- my_vote: Option<number>;
+ reportReason?: string;
+ my_vote?: number;
score: number;
upvotes: number;
downvotes: number;
interface CommentNodeProps {
node: CommentNodeI;
- moderators: Option<CommunityModeratorView[]>;
- admins: Option<PersonViewSafe[]>;
+ moderators?: CommunityModeratorView[];
+ admins?: PersonView[];
noBorder?: boolean;
noIndent?: boolean;
viewOnly?: boolean;
markable?: boolean;
showContext?: boolean;
showCommunity?: boolean;
- enableDownvotes: boolean;
+ enableDownvotes?: boolean;
viewType: CommentViewType;
+ allLanguages: Language[];
+ siteLanguages: number[];
+ hideImages?: boolean;
}
export class CommentNode extends Component<CommentNodeProps, CommentNodeState> {
- private emptyState: CommentNodeState = {
+ state: CommentNodeState = {
showReply: false,
showEdit: false,
showRemoveDialog: false,
- removeReason: None,
showBanDialog: false,
removeData: false,
- banReason: None,
- banExpireDays: None,
banType: BanType.Community,
showPurgeDialog: false,
purgeLoading: false,
- purgeReason: None,
purgeType: PurgeType.Person,
collapsed: false,
viewSource: false,
showConfirmAppointAsMod: false,
showConfirmAppointAsAdmin: false,
showReportDialog: false,
- reportReason: null,
my_vote: this.props.node.comment_view.my_vote,
score: this.props.node.comment_view.counts.score,
upvotes: this.props.node.comment_view.counts.upvotes,
constructor(props: any, context: any) {
super(props, context);
- this.state = this.emptyState;
this.handleReplyCancel = this.handleReplyCancel.bind(this);
this.handleCommentUpvote = this.handleCommentUpvote.bind(this);
this.handleCommentDownvote = this.handleCommentDownvote.bind(this);
// TODO see if there's a better way to do this, and all willReceiveProps
componentWillReceiveProps(nextProps: CommentNodeProps) {
let cv = nextProps.node.comment_view;
- this.state.my_vote = cv.my_vote;
- this.state.upvotes = cv.counts.upvotes;
- this.state.downvotes = cv.counts.downvotes;
- this.state.score = cv.counts.score;
- this.state.readLoading = false;
- this.state.saveLoading = false;
- this.setState(this.state);
+ this.setState({
+ my_vote: cv.my_vote,
+ upvotes: cv.counts.upvotes,
+ downvotes: cv.counts.downvotes,
+ score: cv.counts.score,
+ readLoading: false,
+ saveLoading: false,
+ });
}
render() {
let node = this.props.node;
let cv = this.props.node.comment_view;
- let purgeTypeText: string;
- if (this.state.purgeType == PurgeType.Comment) {
- purgeTypeText = i18n.t("purge_comment");
- } else if (this.state.purgeType == PurgeType.Person) {
- purgeTypeText = `${i18n.t("purge")} ${cv.creator.name}`;
- }
-
- let canMod_ = canMod(
- this.props.moderators,
- this.props.admins,
- cv.creator.id
- );
- let canModOnSelf = canMod(
- this.props.moderators,
- this.props.admins,
- cv.creator.id,
- UserService.Instance.myUserInfo,
- true
- );
- let canAdmin_ = canAdmin(this.props.admins, cv.creator.id);
- let canAdminOnSelf = canAdmin(
- this.props.admins,
- cv.creator.id,
- UserService.Instance.myUserInfo,
- true
- );
- let isMod_ = isMod(this.props.moderators, cv.creator.id);
- let isAdmin_ = isAdmin(this.props.admins, cv.creator.id);
+ let purgeTypeText =
+ this.state.purgeType == PurgeType.Comment
+ ? i18n.t("purge_comment")
+ : `${i18n.t("purge")} ${cv.creator.name}`;
+
+ let canMod_ =
+ canMod(cv.creator.id, this.props.moderators, this.props.admins) &&
+ cv.community.local;
+ let canModOnSelf =
+ canMod(
+ cv.creator.id,
+ this.props.moderators,
+ this.props.admins,
+ UserService.Instance.myUserInfo,
+ true
+ ) && cv.community.local;
+ let canAdmin_ =
+ canAdmin(cv.creator.id, this.props.admins) && cv.community.local;
+ let canAdminOnSelf =
+ canAdmin(
+ cv.creator.id,
+ this.props.admins,
+ UserService.Instance.myUserInfo,
+ true
+ ) && cv.community.local;
+ let isMod_ = isMod(cv.creator.id, this.props.moderators);
+ let isAdmin_ =
+ isAdmin(cv.creator.id, this.props.admins) && cv.community.local;
let amCommunityCreator_ = amCommunityCreator(
- this.props.moderators,
- cv.creator.id
+ cv.creator.id,
+ this.props.moderators
);
let borderColor = this.props.node.depth
this.props.node.comment_view.comment.distinguished,
})}
style={
- !this.props.noIndent &&
- this.props.node.depth &&
- `border-left: 2px ${borderColor} solid !important`
+ !this.props.noIndent && this.props.node.depth
+ ? `border-left: 2px ${borderColor} solid !important`
+ : ""
}
>
<div
- class={`${!this.props.noIndent && this.props.node.depth && "ml-2"}`}
+ className={classNames({
+ "ml-2": !this.props.noIndent && this.props.node.depth,
+ })}
>
- <div class="d-flex flex-wrap align-items-center text-muted small">
- <span class="mr-2">
+ <div className="d-flex flex-wrap align-items-center text-muted small">
+ <span className="mr-2">
<PersonListing person={cv.creator} />
</span>
{cv.comment.distinguished && (
{i18n.t("bot_account").toLowerCase()}
</div>
)}
- {(cv.creator_banned_from_community || isBanned(cv.creator)) && (
- <div className="badge badge-danger mr-2">
- {i18n.t("banned")}
- </div>
- )}
{this.props.showCommunity && (
<>
- <span class="mx-1">{i18n.t("to")}</span>
+ <span className="mx-1">{i18n.t("to")}</span>
<CommunityLink community={cv.community} />
- <span class="mx-2">•</span>
+ <span className="mx-2">•</span>
<Link className="mr-2" to={`/post/${cv.post.id}`}>
{cv.post.name}
</Link>
</>
)}
<button
- class="btn btn-sm text-muted"
+ className="btn btn-sm text-muted"
onClick={linkEvent(this, this.handleCommentCollapse)}
aria-label={this.expandText}
data-tippy-content={this.expandText}
)}
</button>
{this.linkBtn(true)}
+ {cv.comment.language_id != 0 && (
+ <span className="">
+ {
+ this.props.allLanguages.find(
+ lang => lang.id === cv.comment.language_id
+ )?.name
+ }
+ </span>
+ )}
{/* This is an expanding spacer for mobile */}
- <div className="mr-lg-5 flex-grow-1 flex-lg-grow-0 unselectable pointer mx-2"></div>
+ <div className="mr-lg-5 flex-grow-1 flex-lg-grow-0 unselectable pointer mx-2" />
{showScores() && (
<>
<a
data-tippy-content={this.pointsTippy}
>
<span
- class="mr-1 font-weight-bold"
+ className="mr-1 font-weight-bold"
aria-label={i18n.t("number_of_points", {
- count: this.state.score,
- formattedCount: this.state.score,
+ count: Number(this.state.score),
+ formattedCount: numToSI(this.state.score),
})}
>
{numToSI(this.state.score)}
{/* end of user row */}
{this.state.showEdit && (
<CommentForm
- node={Left(node)}
+ node={node}
edit
onReplyCancel={this.handleReplyCancel}
disabled={this.props.locked}
focus
+ allLanguages={this.props.allLanguages}
+ siteLanguages={this.props.siteLanguages}
/>
)}
{!this.state.showEdit && !this.state.collapsed && (
) : (
<div
className="md-div"
- dangerouslySetInnerHTML={mdToHtml(
- this.commentUnlessRemoved
- )}
+ dangerouslySetInnerHTML={
+ this.props.hideImages
+ ? mdToHtmlNoImages(this.commentUnlessRemoved)
+ : mdToHtml(this.commentUnlessRemoved)
+ }
/>
)}
- <div class="d-flex justify-content-between justify-content-lg-start flex-wrap text-muted font-weight-bold">
+ <div className="d-flex justify-content-between justify-content-lg-start flex-wrap text-muted font-weight-bold">
{this.props.showContext && this.linkBtn()}
{this.props.markable && (
<button
- class="btn btn-link btn-animate text-muted"
+ className="btn btn-link btn-animate text-muted"
onClick={linkEvent(this, this.handleMarkRead)}
data-tippy-content={
this.commentReplyOrMentionRead
)}
</button>
)}
- {UserService.Instance.myUserInfo.isSome() &&
- !this.props.viewOnly && (
- <>
+ {UserService.Instance.myUserInfo && !this.props.viewOnly && (
+ <>
+ <button
+ className={`btn btn-link btn-animate ${
+ this.state.my_vote == 1 ? "text-info" : "text-muted"
+ }`}
+ onClick={this.handleCommentUpvote}
+ data-tippy-content={i18n.t("upvote")}
+ aria-label={i18n.t("upvote")}
+ >
+ <Icon icon="arrow-up1" classes="icon-inline" />
+ {showScores() &&
+ this.state.upvotes !== this.state.score && (
+ <span className="ml-1">
+ {numToSI(this.state.upvotes)}
+ </span>
+ )}
+ </button>
+ {this.props.enableDownvotes && (
<button
className={`btn btn-link btn-animate ${
- this.state.my_vote.unwrapOr(0) == 1
- ? "text-info"
+ this.state.my_vote == -1
+ ? "text-danger"
: "text-muted"
}`}
- onClick={this.handleCommentUpvote}
- data-tippy-content={i18n.t("upvote")}
- aria-label={i18n.t("upvote")}
+ onClick={this.handleCommentDownvote}
+ data-tippy-content={i18n.t("downvote")}
+ aria-label={i18n.t("downvote")}
>
- <Icon icon="arrow-up1" classes="icon-inline" />
+ <Icon icon="arrow-down1" classes="icon-inline" />
{showScores() &&
this.state.upvotes !== this.state.score && (
- <span class="ml-1">
- {numToSI(this.state.upvotes)}
+ <span className="ml-1">
+ {numToSI(this.state.downvotes)}
</span>
)}
</button>
- {this.props.enableDownvotes && (
- <button
- className={`btn btn-link btn-animate ${
- this.state.my_vote.unwrapOr(0) == -1
- ? "text-danger"
- : "text-muted"
- }`}
- onClick={this.handleCommentDownvote}
- data-tippy-content={i18n.t("downvote")}
- aria-label={i18n.t("downvote")}
- >
- <Icon icon="arrow-down1" classes="icon-inline" />
- {showScores() &&
- this.state.upvotes !== this.state.score && (
- <span class="ml-1">
- {numToSI(this.state.downvotes)}
- </span>
- )}
- </button>
- )}
+ )}
+ <button
+ className="btn btn-link btn-animate text-muted"
+ onClick={linkEvent(this, this.handleReplyClick)}
+ data-tippy-content={i18n.t("reply")}
+ aria-label={i18n.t("reply")}
+ >
+ <Icon icon="reply1" classes="icon-inline" />
+ </button>
+ {!this.state.showAdvanced ? (
<button
- class="btn btn-link btn-animate text-muted"
- onClick={linkEvent(this, this.handleReplyClick)}
- data-tippy-content={i18n.t("reply")}
- aria-label={i18n.t("reply")}
+ className="btn btn-link btn-animate text-muted"
+ onClick={linkEvent(this, this.handleShowAdvanced)}
+ data-tippy-content={i18n.t("more")}
+ aria-label={i18n.t("more")}
>
- <Icon icon="reply1" classes="icon-inline" />
+ <Icon icon="more-vertical" classes="icon-inline" />
</button>
- {!this.state.showAdvanced ? (
+ ) : (
+ <>
+ {!this.myComment && (
+ <>
+ <button className="btn btn-link btn-animate">
+ <Link
+ className="text-muted"
+ to={`/create_private_message/${cv.creator.id}`}
+ title={i18n.t("message").toLowerCase()}
+ >
+ <Icon icon="mail" />
+ </Link>
+ </button>
+ <button
+ className="btn btn-link btn-animate text-muted"
+ onClick={linkEvent(
+ this,
+ this.handleShowReportDialog
+ )}
+ data-tippy-content={i18n.t(
+ "show_report_dialog"
+ )}
+ aria-label={i18n.t("show_report_dialog")}
+ >
+ <Icon icon="flag" />
+ </button>
+ <button
+ className="btn btn-link btn-animate text-muted"
+ onClick={linkEvent(
+ this,
+ this.handleBlockUserClick
+ )}
+ data-tippy-content={i18n.t("block_user")}
+ aria-label={i18n.t("block_user")}
+ >
+ <Icon icon="slash" />
+ </button>
+ </>
+ )}
+ <button
+ className="btn btn-link btn-animate text-muted"
+ onClick={linkEvent(
+ this,
+ this.handleSaveCommentClick
+ )}
+ data-tippy-content={
+ cv.saved ? i18n.t("unsave") : i18n.t("save")
+ }
+ aria-label={
+ cv.saved ? i18n.t("unsave") : i18n.t("save")
+ }
+ >
+ {this.state.saveLoading ? (
+ this.loadingIcon
+ ) : (
+ <Icon
+ icon="star"
+ classes={`icon-inline ${
+ cv.saved && "text-warning"
+ }`}
+ />
+ )}
+ </button>
<button
className="btn btn-link btn-animate text-muted"
- onClick={linkEvent(this, this.handleShowAdvanced)}
- data-tippy-content={i18n.t("more")}
- aria-label={i18n.t("more")}
+ onClick={linkEvent(this, this.handleViewSource)}
+ data-tippy-content={i18n.t("view_source")}
+ aria-label={i18n.t("view_source")}
>
- <Icon icon="more-vertical" classes="icon-inline" />
+ <Icon
+ icon="file-text"
+ classes={`icon-inline ${
+ this.state.viewSource && "text-success"
+ }`}
+ />
</button>
- ) : (
- <>
- {!this.myComment && (
- <>
- <button class="btn btn-link btn-animate">
- <Link
- className="text-muted"
- to={`/create_private_message/recipient/${cv.creator.id}`}
- title={i18n.t("message").toLowerCase()}
- >
- <Icon icon="mail" />
- </Link>
- </button>
- <button
- class="btn btn-link btn-animate text-muted"
- onClick={linkEvent(
- this,
- this.handleShowReportDialog
- )}
- data-tippy-content={i18n.t(
- "show_report_dialog"
- )}
- aria-label={i18n.t("show_report_dialog")}
- >
- <Icon icon="flag" />
- </button>
+ {this.myComment && (
+ <>
+ <button
+ className="btn btn-link btn-animate text-muted"
+ onClick={linkEvent(this, this.handleEditClick)}
+ data-tippy-content={i18n.t("edit")}
+ aria-label={i18n.t("edit")}
+ >
+ <Icon icon="edit" classes="icon-inline" />
+ </button>
+ <button
+ className="btn btn-link btn-animate text-muted"
+ onClick={linkEvent(
+ this,
+ this.handleDeleteClick
+ )}
+ data-tippy-content={
+ !cv.comment.deleted
+ ? i18n.t("delete")
+ : i18n.t("restore")
+ }
+ aria-label={
+ !cv.comment.deleted
+ ? i18n.t("delete")
+ : i18n.t("restore")
+ }
+ >
+ <Icon
+ icon="trash"
+ classes={`icon-inline ${
+ cv.comment.deleted && "text-danger"
+ }`}
+ />
+ </button>
+
+ {(canModOnSelf || canAdminOnSelf) && (
<button
- class="btn btn-link btn-animate text-muted"
+ className="btn btn-link btn-animate text-muted"
onClick={linkEvent(
this,
- this.handleBlockUserClick
+ this.handleDistinguishClick
)}
- data-tippy-content={i18n.t("block_user")}
- aria-label={i18n.t("block_user")}
+ data-tippy-content={
+ !cv.comment.distinguished
+ ? i18n.t("distinguish")
+ : i18n.t("undistinguish")
+ }
+ aria-label={
+ !cv.comment.distinguished
+ ? i18n.t("distinguish")
+ : i18n.t("undistinguish")
+ }
>
- <Icon icon="slash" />
+ <Icon
+ icon="shield"
+ classes={`icon-inline ${
+ cv.comment.distinguished && "text-danger"
+ }`}
+ />
</button>
- </>
- )}
- <button
- class="btn btn-link btn-animate text-muted"
- onClick={linkEvent(
- this,
- this.handleSaveCommentClick
- )}
- data-tippy-content={
- cv.saved ? i18n.t("unsave") : i18n.t("save")
- }
- aria-label={
- cv.saved ? i18n.t("unsave") : i18n.t("save")
- }
- >
- {this.state.saveLoading ? (
- this.loadingIcon
- ) : (
- <Icon
- icon="star"
- classes={`icon-inline ${
- cv.saved && "text-warning"
- }`}
- />
)}
- </button>
- <button
- className="btn btn-link btn-animate text-muted"
- onClick={linkEvent(this, this.handleViewSource)}
- data-tippy-content={i18n.t("view_source")}
- aria-label={i18n.t("view_source")}
- >
- <Icon
- icon="file-text"
- classes={`icon-inline ${
- this.state.viewSource && "text-success"
- }`}
- />
- </button>
- {this.myComment && (
- <>
+ </>
+ )}
+ {/* Admins and mods can remove comments */}
+ {(canMod_ || canAdmin_) && (
+ <>
+ {!cv.comment.removed ? (
<button
- class="btn btn-link btn-animate text-muted"
+ className="btn btn-link btn-animate text-muted"
onClick={linkEvent(
this,
- this.handleEditClick
+ this.handleModRemoveShow
)}
- data-tippy-content={i18n.t("edit")}
- aria-label={i18n.t("edit")}
+ aria-label={i18n.t("remove")}
>
- <Icon icon="edit" classes="icon-inline" />
+ {i18n.t("remove")}
</button>
+ ) : (
<button
- class="btn btn-link btn-animate text-muted"
+ className="btn btn-link btn-animate text-muted"
onClick={linkEvent(
this,
- this.handleDeleteClick
+ this.handleModRemoveSubmit
)}
- data-tippy-content={
- !cv.comment.deleted
- ? i18n.t("delete")
- : i18n.t("restore")
- }
- aria-label={
- !cv.comment.deleted
- ? i18n.t("delete")
- : i18n.t("restore")
- }
+ aria-label={i18n.t("restore")}
>
- <Icon
- icon="trash"
- classes={`icon-inline ${
- cv.comment.deleted && "text-danger"
- }`}
- />
+ {i18n.t("restore")}
</button>
-
- {(canModOnSelf || canAdminOnSelf) && (
+ )}
+ </>
+ )}
+ {/* Mods can ban from community, and appoint as mods to community */}
+ {canMod_ && (
+ <>
+ {!isMod_ &&
+ (!cv.creator_banned_from_community ? (
<button
- class="btn btn-link btn-animate text-muted"
+ className="btn btn-link btn-animate text-muted"
onClick={linkEvent(
this,
- this.handleDistinguishClick
+ this.handleModBanFromCommunityShow
)}
- data-tippy-content={
- !cv.comment.distinguished
- ? i18n.t("distinguish")
- : i18n.t("undistinguish")
- }
- aria-label={
- !cv.comment.distinguished
- ? i18n.t("distinguish")
- : i18n.t("undistinguish")
- }
+ aria-label={i18n.t("ban_from_community")}
>
- <Icon
- icon="shield"
- classes={`icon-inline ${
- cv.comment.distinguished &&
- "text-danger"
- }`}
- />
+ {i18n.t("ban_from_community")}
</button>
- )}
- </>
- )}
- {/* Admins and mods can remove comments */}
- {(canMod_ || canAdmin_) && (
- <>
- {!cv.comment.removed ? (
+ ) : (
<button
- class="btn btn-link btn-animate text-muted"
+ className="btn btn-link btn-animate text-muted"
onClick={linkEvent(
this,
- this.handleModRemoveShow
+ this.handleModBanFromCommunitySubmit
)}
- aria-label={i18n.t("remove")}
+ aria-label={i18n.t("unban")}
>
- {i18n.t("remove")}
+ {i18n.t("unban")}
</button>
- ) : (
+ ))}
+ {!cv.creator_banned_from_community &&
+ (!this.state.showConfirmAppointAsMod ? (
<button
- class="btn btn-link btn-animate text-muted"
+ className="btn btn-link btn-animate text-muted"
onClick={linkEvent(
this,
- this.handleModRemoveSubmit
+ this.handleShowConfirmAppointAsMod
)}
- aria-label={i18n.t("restore")}
+ aria-label={
+ isMod_
+ ? i18n.t("remove_as_mod")
+ : i18n.t("appoint_as_mod")
+ }
>
- {i18n.t("restore")}
+ {isMod_
+ ? i18n.t("remove_as_mod")
+ : i18n.t("appoint_as_mod")}
</button>
- )}
- </>
- )}
- {/* Mods can ban from community, and appoint as mods to community */}
- {canMod_ && (
- <>
- {!isMod_ &&
- (!cv.creator_banned_from_community ? (
+ ) : (
+ <>
<button
- class="btn btn-link btn-animate text-muted"
- onClick={linkEvent(
- this,
- this.handleModBanFromCommunityShow
- )}
- aria-label={i18n.t("ban")}
+ className="btn btn-link btn-animate text-muted"
+ aria-label={i18n.t("are_you_sure")}
>
- {i18n.t("ban")}
+ {i18n.t("are_you_sure")}
</button>
- ) : (
<button
- class="btn btn-link btn-animate text-muted"
+ className="btn btn-link btn-animate text-muted"
onClick={linkEvent(
this,
- this.handleModBanFromCommunitySubmit
+ this.handleAddModToCommunity
)}
- aria-label={i18n.t("unban")}
+ aria-label={i18n.t("yes")}
>
- {i18n.t("unban")}
+ {i18n.t("yes")}
</button>
- ))}
- {!cv.creator_banned_from_community &&
- (!this.state.showConfirmAppointAsMod ? (
<button
- class="btn btn-link btn-animate text-muted"
+ className="btn btn-link btn-animate text-muted"
onClick={linkEvent(
this,
- this.handleShowConfirmAppointAsMod
+ this.handleCancelConfirmAppointAsMod
)}
- aria-label={
- isMod_
- ? i18n.t("remove_as_mod")
- : i18n.t("appoint_as_mod")
- }
+ aria-label={i18n.t("no")}
>
- {isMod_
- ? i18n.t("remove_as_mod")
- : i18n.t("appoint_as_mod")}
+ {i18n.t("no")}
</button>
- ) : (
- <>
- <button
- class="btn btn-link btn-animate text-muted"
- aria-label={i18n.t("are_you_sure")}
- >
- {i18n.t("are_you_sure")}
- </button>
- <button
- class="btn btn-link btn-animate text-muted"
- onClick={linkEvent(
- this,
- this.handleAddModToCommunity
- )}
- aria-label={i18n.t("yes")}
- >
- {i18n.t("yes")}
- </button>
- <button
- class="btn btn-link btn-animate text-muted"
- onClick={linkEvent(
- this,
- this.handleCancelConfirmAppointAsMod
- )}
- aria-label={i18n.t("no")}
- >
- {i18n.t("no")}
- </button>
- </>
- ))}
- </>
- )}
- {/* Community creators and admins can transfer community to another mod */}
- {(amCommunityCreator_ || canAdmin_) &&
- isMod_ &&
- cv.creator.local &&
- (!this.state.showConfirmTransferCommunity ? (
+ </>
+ ))}
+ </>
+ )}
+ {/* Community creators and admins can transfer community to another mod */}
+ {(amCommunityCreator_ || canAdmin_) &&
+ isMod_ &&
+ cv.creator.local &&
+ (!this.state.showConfirmTransferCommunity ? (
+ <button
+ className="btn btn-link btn-animate text-muted"
+ onClick={linkEvent(
+ this,
+ this.handleShowConfirmTransferCommunity
+ )}
+ aria-label={i18n.t("transfer_community")}
+ >
+ {i18n.t("transfer_community")}
+ </button>
+ ) : (
+ <>
<button
- class="btn btn-link btn-animate text-muted"
+ className="btn btn-link btn-animate text-muted"
+ aria-label={i18n.t("are_you_sure")}
+ >
+ {i18n.t("are_you_sure")}
+ </button>
+ <button
+ className="btn btn-link btn-animate text-muted"
onClick={linkEvent(
this,
- this.handleShowConfirmTransferCommunity
+ this.handleTransferCommunity
)}
- aria-label={i18n.t("transfer_community")}
+ aria-label={i18n.t("yes")}
>
- {i18n.t("transfer_community")}
+ {i18n.t("yes")}
</button>
- ) : (
+ <button
+ className="btn btn-link btn-animate text-muted"
+ onClick={linkEvent(
+ this,
+ this
+ .handleCancelShowConfirmTransferCommunity
+ )}
+ aria-label={i18n.t("no")}
+ >
+ {i18n.t("no")}
+ </button>
+ </>
+ ))}
+ {/* Admins can ban from all, and appoint other admins */}
+ {canAdmin_ && (
+ <>
+ {!isAdmin_ && (
<>
<button
- class="btn btn-link btn-animate text-muted"
- aria-label={i18n.t("are_you_sure")}
- >
- {i18n.t("are_you_sure")}
- </button>
- <button
- class="btn btn-link btn-animate text-muted"
+ className="btn btn-link btn-animate text-muted"
onClick={linkEvent(
this,
- this.handleTransferCommunity
+ this.handlePurgePersonShow
)}
- aria-label={i18n.t("yes")}
+ aria-label={i18n.t("purge_user")}
>
- {i18n.t("yes")}
+ {i18n.t("purge_user")}
</button>
<button
- class="btn btn-link btn-animate text-muted"
+ className="btn btn-link btn-animate text-muted"
onClick={linkEvent(
this,
- this
- .handleCancelShowConfirmTransferCommunity
+ this.handlePurgeCommentShow
)}
- aria-label={i18n.t("no")}
+ aria-label={i18n.t("purge_comment")}
>
- {i18n.t("no")}
+ {i18n.t("purge_comment")}
</button>
- </>
- ))}
- {/* Admins can ban from all, and appoint other admins */}
- {canAdmin_ && (
- <>
- {!isAdmin_ && (
- <>
+
+ {!isBanned(cv.creator) ? (
<button
- class="btn btn-link btn-animate text-muted"
+ className="btn btn-link btn-animate text-muted"
onClick={linkEvent(
this,
- this.handlePurgePersonShow
+ this.handleModBanShow
)}
- aria-label={i18n.t("purge_user")}
+ aria-label={i18n.t("ban_from_site")}
>
- {i18n.t("purge_user")}
+ {i18n.t("ban_from_site")}
</button>
+ ) : (
<button
- class="btn btn-link btn-animate text-muted"
+ className="btn btn-link btn-animate text-muted"
onClick={linkEvent(
this,
- this.handlePurgeCommentShow
+ this.handleModBanSubmit
)}
- aria-label={i18n.t("purge_comment")}
+ aria-label={i18n.t("unban_from_site")}
>
- {i18n.t("purge_comment")}
+ {i18n.t("unban_from_site")}
</button>
-
- {!isBanned(cv.creator) ? (
- <button
- class="btn btn-link btn-animate text-muted"
- onClick={linkEvent(
- this,
- this.handleModBanShow
- )}
- aria-label={i18n.t("ban_from_site")}
- >
- {i18n.t("ban_from_site")}
- </button>
- ) : (
- <button
- class="btn btn-link btn-animate text-muted"
- onClick={linkEvent(
- this,
- this.handleModBanSubmit
- )}
- aria-label={i18n.t("unban_from_site")}
- >
- {i18n.t("unban_from_site")}
- </button>
+ )}
+ </>
+ )}
+ {!isBanned(cv.creator) &&
+ cv.creator.local &&
+ (!this.state.showConfirmAppointAsAdmin ? (
+ <button
+ className="btn btn-link btn-animate text-muted"
+ onClick={linkEvent(
+ this,
+ this.handleShowConfirmAppointAsAdmin
)}
- </>
- )}
- {!isBanned(cv.creator) &&
- cv.creator.local &&
- (!this.state.showConfirmAppointAsAdmin ? (
+ aria-label={
+ isAdmin_
+ ? i18n.t("remove_as_admin")
+ : i18n.t("appoint_as_admin")
+ }
+ >
+ {isAdmin_
+ ? i18n.t("remove_as_admin")
+ : i18n.t("appoint_as_admin")}
+ </button>
+ ) : (
+ <>
+ <button className="btn btn-link btn-animate text-muted">
+ {i18n.t("are_you_sure")}
+ </button>
<button
- class="btn btn-link btn-animate text-muted"
+ className="btn btn-link btn-animate text-muted"
onClick={linkEvent(
this,
- this.handleShowConfirmAppointAsAdmin
+ this.handleAddAdmin
)}
- aria-label={
- isAdmin_
- ? i18n.t("remove_as_admin")
- : i18n.t("appoint_as_admin")
- }
+ aria-label={i18n.t("yes")}
>
- {isAdmin_
- ? i18n.t("remove_as_admin")
- : i18n.t("appoint_as_admin")}
+ {i18n.t("yes")}
</button>
- ) : (
- <>
- <button class="btn btn-link btn-animate text-muted">
- {i18n.t("are_you_sure")}
- </button>
- <button
- class="btn btn-link btn-animate text-muted"
- onClick={linkEvent(
- this,
- this.handleAddAdmin
- )}
- aria-label={i18n.t("yes")}
- >
- {i18n.t("yes")}
- </button>
- <button
- class="btn btn-link btn-animate text-muted"
- onClick={linkEvent(
- this,
- this.handleCancelConfirmAppointAsAdmin
- )}
- aria-label={i18n.t("no")}
- >
- {i18n.t("no")}
- </button>
- </>
- ))}
- </>
- )}
- </>
- )}
- </>
- )}
+ <button
+ className="btn btn-link btn-animate text-muted"
+ onClick={linkEvent(
+ this,
+ this.handleCancelConfirmAppointAsAdmin
+ )}
+ aria-label={i18n.t("no")}
+ >
+ {i18n.t("no")}
+ </button>
+ </>
+ ))}
+ </>
+ )}
+ </>
+ )}
+ </>
+ )}
</div>
{/* end of button group */}
</div>
style={`border-left: 2px ${moreRepliesBorderColor} solid !important`}
>
<button
- class="btn btn-link text-muted"
+ className="btn btn-link text-muted"
onClick={linkEvent(this, this.handleFetchChildren)}
>
{i18n.t("x_more_replies", {
{/* end of details */}
{this.state.showRemoveDialog && (
<form
- class="form-inline"
+ className="form-inline"
onSubmit={linkEvent(this, this.handleModRemoveSubmit)}
>
<label
- class="sr-only"
+ className="sr-only"
htmlFor={`mod-remove-reason-${cv.comment.id}`}
>
{i18n.t("reason")}
<input
type="text"
id={`mod-remove-reason-${cv.comment.id}`}
- class="form-control mr-2"
+ className="form-control mr-2"
placeholder={i18n.t("reason")}
- value={toUndefined(this.state.removeReason)}
+ value={this.state.removeReason}
onInput={linkEvent(this, this.handleModRemoveReasonChange)}
/>
<button
type="submit"
- class="btn btn-secondary"
+ className="btn btn-secondary"
aria-label={i18n.t("remove_comment")}
>
{i18n.t("remove_comment")}
)}
{this.state.showReportDialog && (
<form
- class="form-inline"
+ className="form-inline"
onSubmit={linkEvent(this, this.handleReportSubmit)}
>
- <label class="sr-only" htmlFor={`report-reason-${cv.comment.id}`}>
+ <label
+ className="sr-only"
+ htmlFor={`report-reason-${cv.comment.id}`}
+ >
{i18n.t("reason")}
</label>
<input
type="text"
required
id={`report-reason-${cv.comment.id}`}
- class="form-control mr-2"
+ className="form-control mr-2"
placeholder={i18n.t("reason")}
value={this.state.reportReason}
onInput={linkEvent(this, this.handleReportReasonChange)}
/>
<button
type="submit"
- class="btn btn-secondary"
+ className="btn btn-secondary"
aria-label={i18n.t("create_report")}
>
{i18n.t("create_report")}
)}
{this.state.showBanDialog && (
<form onSubmit={linkEvent(this, this.handleModBanBothSubmit)}>
- <div class="form-group row col-12">
+ <div className="form-group row col-12">
<label
- class="col-form-label"
+ className="col-form-label"
htmlFor={`mod-ban-reason-${cv.comment.id}`}
>
{i18n.t("reason")}
<input
type="text"
id={`mod-ban-reason-${cv.comment.id}`}
- class="form-control mr-2"
+ className="form-control mr-2"
placeholder={i18n.t("reason")}
- value={toUndefined(this.state.banReason)}
+ value={this.state.banReason}
onInput={linkEvent(this, this.handleModBanReasonChange)}
/>
<label
- class="col-form-label"
+ className="col-form-label"
htmlFor={`mod-ban-expires-${cv.comment.id}`}
>
{i18n.t("expires")}
<input
type="number"
id={`mod-ban-expires-${cv.comment.id}`}
- class="form-control mr-2"
+ className="form-control mr-2"
placeholder={i18n.t("number_of_days")}
- value={toUndefined(this.state.banExpireDays)}
+ value={this.state.banExpireDays}
onInput={linkEvent(this, this.handleModBanExpireDaysChange)}
/>
- <div class="form-group">
- <div class="form-check">
+ <div className="form-group">
+ <div className="form-check">
<input
- class="form-check-input"
+ className="form-check-input"
id="mod-ban-remove-data"
type="checkbox"
checked={this.state.removeData}
onChange={linkEvent(this, this.handleModRemoveDataChange)}
/>
<label
- class="form-check-label"
+ className="form-check-label"
htmlFor="mod-ban-remove-data"
title={i18n.t("remove_content_more")}
>
{/* <label class="col-form-label">Expires</label> */}
{/* <input type="date" class="form-control mr-2" placeholder={i18n.t('expires')} value={this.state.banExpires} onInput={linkEvent(this, this.handleModBanExpiresChange)} /> */}
{/* </div> */}
- <div class="form-group row">
+ <div className="form-group row">
<button
type="submit"
- class="btn btn-secondary"
+ className="btn btn-secondary"
aria-label={i18n.t("ban")}
>
{i18n.t("ban")} {cv.creator.name}
{this.state.showPurgeDialog && (
<form onSubmit={linkEvent(this, this.handlePurgeSubmit)}>
<PurgeWarning />
- <label class="sr-only" htmlFor="purge-reason">
+ <label className="sr-only" htmlFor="purge-reason">
{i18n.t("reason")}
</label>
<input
type="text"
id="purge-reason"
- class="form-control my-3"
+ className="form-control my-3"
placeholder={i18n.t("reason")}
- value={toUndefined(this.state.purgeReason)}
+ value={this.state.purgeReason}
onInput={linkEvent(this, this.handlePurgeReasonChange)}
/>
- <div class="form-group row col-12">
+ <div className="form-group row col-12">
{this.state.purgeLoading ? (
<Spinner />
) : (
<button
type="submit"
- class="btn btn-secondary"
+ className="btn btn-secondary"
aria-label={purgeTypeText}
>
{purgeTypeText}
)}
{this.state.showReply && (
<CommentForm
- node={Left(node)}
+ node={node}
onReplyCancel={this.handleReplyCancel}
disabled={this.props.locked}
focus
+ allLanguages={this.props.allLanguages}
+ siteLanguages={this.props.siteLanguages}
/>
)}
{!this.state.collapsed && node.children.length > 0 && (
locked={this.props.locked}
moderators={this.props.moderators}
admins={this.props.admins}
- maxCommentsShown={None}
enableDownvotes={this.props.enableDownvotes}
viewType={this.props.viewType}
+ allLanguages={this.props.allLanguages}
+ siteLanguages={this.props.siteLanguages}
+ hideImages={this.props.hideImages}
/>
)}
{/* A collapsed clearfix */}
- {this.state.collapsed && <div class="row col-12"></div>}
+ {this.state.collapsed && <div className="row col-12"></div>}
</div>
);
}
? i18n.t("show_context")
: i18n.t("link");
+ // The context button should show the parent comment by default
+ const parentCommentId = getCommentParentId(cv.comment) ?? cv.comment.id;
+
return (
<>
<Link
className={classnames}
- to={`/comment/${cv.comment.id}`}
+ to={`/comment/${parentCommentId}`}
title={title}
>
<Icon icon="link" classes="icon-inline" />
}
get myComment(): boolean {
- return UserService.Instance.myUserInfo
- .map(
- m =>
- m.local_user_view.person.id == this.props.node.comment_view.creator.id
- )
- .unwrapOr(false);
+ return (
+ UserService.Instance.myUserInfo?.local_user_view.person.id ==
+ this.props.node.comment_view.creator.id
+ );
}
get isPostCreator(): boolean {
}
handleReplyClick(i: CommentNode) {
- i.state.showReply = true;
- i.setState(i.state);
+ i.setState({ showReply: true });
}
handleEditClick(i: CommentNode) {
- i.state.showEdit = true;
- i.setState(i.state);
+ i.setState({ showEdit: true });
}
handleBlockUserClick(i: CommentNode) {
- let blockUserForm = new BlockPerson({
- person_id: i.props.node.comment_view.creator.id,
- block: true,
- auth: auth().unwrap(),
- });
- WebSocketService.Instance.send(wsClient.blockPerson(blockUserForm));
+ let auth = myAuth();
+ if (auth) {
+ let blockUserForm: BlockPerson = {
+ person_id: i.props.node.comment_view.creator.id,
+ block: true,
+ auth,
+ };
+ WebSocketService.Instance.send(wsClient.blockPerson(blockUserForm));
+ }
}
handleDeleteClick(i: CommentNode) {
let comment = i.props.node.comment_view.comment;
- let deleteForm = new DeleteComment({
- comment_id: comment.id,
- deleted: !comment.deleted,
- auth: auth().unwrap(),
- });
- WebSocketService.Instance.send(wsClient.deleteComment(deleteForm));
+ let auth = myAuth();
+ if (auth) {
+ let deleteForm: DeleteComment = {
+ comment_id: comment.id,
+ deleted: !comment.deleted,
+ auth,
+ };
+ WebSocketService.Instance.send(wsClient.deleteComment(deleteForm));
+ }
}
handleSaveCommentClick(i: CommentNode) {
let cv = i.props.node.comment_view;
let save = cv.saved == undefined ? true : !cv.saved;
- let form = new SaveComment({
- comment_id: cv.comment.id,
- save,
- auth: auth().unwrap(),
- });
+ let auth = myAuth();
+ if (auth) {
+ let form: SaveComment = {
+ comment_id: cv.comment.id,
+ save,
+ auth,
+ };
- WebSocketService.Instance.send(wsClient.saveComment(form));
+ WebSocketService.Instance.send(wsClient.saveComment(form));
- i.state.saveLoading = true;
- i.setState(this.state);
+ i.setState({ saveLoading: true });
+ }
}
handleReplyCancel() {
- this.state.showReply = false;
- this.state.showEdit = false;
- this.setState(this.state);
+ this.setState({ showReply: false, showEdit: false });
}
handleCommentUpvote(event: any) {
event.preventDefault();
- let myVote = this.state.my_vote.unwrapOr(0);
+ let myVote = this.state.my_vote;
let newVote = myVote == 1 ? 0 : 1;
if (myVote == 1) {
- this.state.score--;
- this.state.upvotes--;
+ this.setState({
+ score: this.state.score - 1,
+ upvotes: this.state.upvotes - 1,
+ });
} else if (myVote == -1) {
- this.state.downvotes--;
- this.state.upvotes++;
- this.state.score += 2;
+ this.setState({
+ downvotes: this.state.downvotes - 1,
+ upvotes: this.state.upvotes + 1,
+ score: this.state.score + 2,
+ });
} else {
- this.state.upvotes++;
- this.state.score++;
+ this.setState({
+ score: this.state.score + 1,
+ upvotes: this.state.upvotes + 1,
+ });
}
- this.state.my_vote = Some(newVote);
-
- let form = new CreateCommentLike({
- comment_id: this.props.node.comment_view.comment.id,
- score: newVote,
- auth: auth().unwrap(),
- });
- WebSocketService.Instance.send(wsClient.likeComment(form));
- this.setState(this.state);
- setupTippy();
+ this.setState({ my_vote: newVote });
+
+ let auth = myAuth();
+ if (auth) {
+ let form: CreateCommentLike = {
+ comment_id: this.props.node.comment_view.comment.id,
+ score: newVote,
+ auth,
+ };
+ WebSocketService.Instance.send(wsClient.likeComment(form));
+ setupTippy();
+ }
}
handleCommentDownvote(event: any) {
event.preventDefault();
- let myVote = this.state.my_vote.unwrapOr(0);
+ let myVote = this.state.my_vote;
let newVote = myVote == -1 ? 0 : -1;
if (myVote == 1) {
- this.state.score -= 2;
- this.state.upvotes--;
- this.state.downvotes++;
+ this.setState({
+ downvotes: this.state.downvotes + 1,
+ upvotes: this.state.upvotes - 1,
+ score: this.state.score - 2,
+ });
} else if (myVote == -1) {
- this.state.downvotes--;
- this.state.score++;
+ this.setState({
+ downvotes: this.state.downvotes - 1,
+ score: this.state.score + 1,
+ });
} else {
- this.state.downvotes++;
- this.state.score--;
+ this.setState({
+ downvotes: this.state.downvotes + 1,
+ score: this.state.score - 1,
+ });
}
- this.state.my_vote = Some(newVote);
+ this.setState({ my_vote: newVote });
- let form = new CreateCommentLike({
- comment_id: this.props.node.comment_view.comment.id,
- score: newVote,
- auth: auth().unwrap(),
- });
+ let auth = myAuth();
+ if (auth) {
+ let form: CreateCommentLike = {
+ comment_id: this.props.node.comment_view.comment.id,
+ score: newVote,
+ auth,
+ };
- WebSocketService.Instance.send(wsClient.likeComment(form));
- this.setState(this.state);
- setupTippy();
+ WebSocketService.Instance.send(wsClient.likeComment(form));
+ setupTippy();
+ }
}
handleShowReportDialog(i: CommentNode) {
- i.state.showReportDialog = !i.state.showReportDialog;
- i.setState(i.state);
+ i.setState({ showReportDialog: !i.state.showReportDialog });
}
handleReportReasonChange(i: CommentNode, event: any) {
- i.state.reportReason = event.target.value;
- i.setState(i.state);
+ i.setState({ reportReason: event.target.value });
}
handleReportSubmit(i: CommentNode) {
let comment = i.props.node.comment_view.comment;
- let form = new CreateCommentReport({
- comment_id: comment.id,
- reason: i.state.reportReason,
- auth: auth().unwrap(),
- });
- WebSocketService.Instance.send(wsClient.createCommentReport(form));
-
- i.state.showReportDialog = false;
- i.setState(i.state);
+ let reason = i.state.reportReason;
+ let auth = myAuth();
+ if (reason && auth) {
+ let form: CreateCommentReport = {
+ comment_id: comment.id,
+ reason,
+ auth,
+ };
+ WebSocketService.Instance.send(wsClient.createCommentReport(form));
+ i.setState({ showReportDialog: false });
+ }
}
handleModRemoveShow(i: CommentNode) {
- i.state.showRemoveDialog = !i.state.showRemoveDialog;
- i.state.showBanDialog = false;
- i.setState(i.state);
+ i.setState({
+ showRemoveDialog: !i.state.showRemoveDialog,
+ showBanDialog: false,
+ });
}
handleModRemoveReasonChange(i: CommentNode, event: any) {
- i.state.removeReason = Some(event.target.value);
- i.setState(i.state);
+ i.setState({ removeReason: event.target.value });
}
handleModRemoveDataChange(i: CommentNode, event: any) {
- i.state.removeData = event.target.checked;
- i.setState(i.state);
+ i.setState({ removeData: event.target.checked });
}
handleModRemoveSubmit(i: CommentNode) {
let comment = i.props.node.comment_view.comment;
- let form = new RemoveComment({
- comment_id: comment.id,
- removed: !comment.removed,
- reason: i.state.removeReason,
- auth: auth().unwrap(),
- });
- WebSocketService.Instance.send(wsClient.removeComment(form));
-
- i.state.showRemoveDialog = false;
- i.setState(i.state);
+ let auth = myAuth();
+ if (auth) {
+ let form: RemoveComment = {
+ comment_id: comment.id,
+ removed: !comment.removed,
+ reason: i.state.removeReason,
+ auth,
+ };
+ WebSocketService.Instance.send(wsClient.removeComment(form));
+
+ i.setState({ showRemoveDialog: false });
+ }
}
handleDistinguishClick(i: CommentNode) {
let comment = i.props.node.comment_view.comment;
- let form = new EditComment({
- comment_id: comment.id,
- form_id: None, // TODO not sure about this
- content: None,
- distinguished: Some(!comment.distinguished),
- auth: auth().unwrap(),
- });
- WebSocketService.Instance.send(wsClient.editComment(form));
- i.setState(i.state);
+ let auth = myAuth();
+ if (auth) {
+ let form: DistinguishComment = {
+ comment_id: comment.id,
+ distinguished: !comment.distinguished,
+ auth,
+ };
+ WebSocketService.Instance.send(wsClient.editComment(form));
+ i.setState(i.state);
+ }
}
isPersonMentionType(
}
handleMarkRead(i: CommentNode) {
- if (i.isPersonMentionType(i.props.node.comment_view)) {
- let form = new MarkPersonMentionAsRead({
- person_mention_id: i.props.node.comment_view.person_mention.id,
- read: !i.props.node.comment_view.person_mention.read,
- auth: auth().unwrap(),
- });
- WebSocketService.Instance.send(wsClient.markPersonMentionAsRead(form));
- } else if (i.isCommentReplyType(i.props.node.comment_view)) {
- let form = new MarkCommentReplyAsRead({
- comment_reply_id: i.props.node.comment_view.comment_reply.id,
- read: !i.props.node.comment_view.comment_reply.read,
- auth: auth().unwrap(),
- });
- WebSocketService.Instance.send(wsClient.markCommentReplyAsRead(form));
- }
+ let auth = myAuth();
+ if (auth) {
+ if (i.isPersonMentionType(i.props.node.comment_view)) {
+ let form: MarkPersonMentionAsRead = {
+ person_mention_id: i.props.node.comment_view.person_mention.id,
+ read: !i.props.node.comment_view.person_mention.read,
+ auth,
+ };
+ WebSocketService.Instance.send(wsClient.markPersonMentionAsRead(form));
+ } else if (i.isCommentReplyType(i.props.node.comment_view)) {
+ let form: MarkCommentReplyAsRead = {
+ comment_reply_id: i.props.node.comment_view.comment_reply.id,
+ read: !i.props.node.comment_view.comment_reply.read,
+ auth,
+ };
+ WebSocketService.Instance.send(wsClient.markCommentReplyAsRead(form));
+ }
- i.state.readLoading = true;
- i.setState(this.state);
+ i.setState({ readLoading: true });
+ }
}
handleModBanFromCommunityShow(i: CommentNode) {
- i.state.showBanDialog = true;
- i.state.banType = BanType.Community;
- i.state.showRemoveDialog = false;
- i.setState(i.state);
+ i.setState({
+ showBanDialog: true,
+ banType: BanType.Community,
+ showRemoveDialog: false,
+ });
}
handleModBanShow(i: CommentNode) {
- i.state.showBanDialog = true;
- i.state.banType = BanType.Site;
- i.state.showRemoveDialog = false;
- i.setState(i.state);
+ i.setState({
+ showBanDialog: true,
+ banType: BanType.Site,
+ showRemoveDialog: false,
+ });
}
handleModBanReasonChange(i: CommentNode, event: any) {
- i.state.banReason = Some(event.target.value);
- i.setState(i.state);
+ i.setState({ banReason: event.target.value });
}
handleModBanExpireDaysChange(i: CommentNode, event: any) {
- i.state.banExpireDays = Some(event.target.value);
- i.setState(i.state);
+ i.setState({ banExpireDays: event.target.value });
}
handleModBanFromCommunitySubmit(i: CommentNode) {
- i.state.banType = BanType.Community;
- i.setState(i.state);
+ i.setState({ banType: BanType.Community });
i.handleModBanBothSubmit(i);
}
handleModBanSubmit(i: CommentNode) {
- i.state.banType = BanType.Site;
- i.setState(i.state);
+ i.setState({ banType: BanType.Site });
i.handleModBanBothSubmit(i);
}
handleModBanBothSubmit(i: CommentNode) {
let cv = i.props.node.comment_view;
-
- if (i.state.banType == BanType.Community) {
- // If its an unban, restore all their data
- let ban = !cv.creator_banned_from_community;
- if (ban == false) {
- i.state.removeData = false;
- }
- let form = new BanFromCommunity({
- person_id: cv.creator.id,
- community_id: cv.community.id,
- ban,
- remove_data: Some(i.state.removeData),
- reason: i.state.banReason,
- expires: i.state.banExpireDays.map(futureDaysToUnixTime),
- auth: auth().unwrap(),
- });
- WebSocketService.Instance.send(wsClient.banFromCommunity(form));
- } else {
- // If its an unban, restore all their data
- let ban = !cv.creator.banned;
- if (ban == false) {
- i.state.removeData = false;
+ let auth = myAuth();
+ if (auth) {
+ if (i.state.banType == BanType.Community) {
+ // If its an unban, restore all their data
+ let ban = !cv.creator_banned_from_community;
+ if (ban == false) {
+ i.setState({ removeData: false });
+ }
+ let form: BanFromCommunity = {
+ person_id: cv.creator.id,
+ community_id: cv.community.id,
+ ban,
+ remove_data: i.state.removeData,
+ reason: i.state.banReason,
+ expires: futureDaysToUnixTime(i.state.banExpireDays),
+ auth,
+ };
+ WebSocketService.Instance.send(wsClient.banFromCommunity(form));
+ } else {
+ // If its an unban, restore all their data
+ let ban = !cv.creator.banned;
+ if (ban == false) {
+ i.setState({ removeData: false });
+ }
+ let form: BanPerson = {
+ person_id: cv.creator.id,
+ ban,
+ remove_data: i.state.removeData,
+ reason: i.state.banReason,
+ expires: futureDaysToUnixTime(i.state.banExpireDays),
+ auth,
+ };
+ WebSocketService.Instance.send(wsClient.banPerson(form));
}
- let form = new BanPerson({
- person_id: cv.creator.id,
- ban,
- remove_data: Some(i.state.removeData),
- reason: i.state.banReason,
- expires: i.state.banExpireDays.map(futureDaysToUnixTime),
- auth: auth().unwrap(),
- });
- WebSocketService.Instance.send(wsClient.banPerson(form));
- }
- i.state.showBanDialog = false;
- i.setState(i.state);
+ i.setState({ showBanDialog: false });
+ }
}
handlePurgePersonShow(i: CommentNode) {
- i.state.showPurgeDialog = true;
- i.state.purgeType = PurgeType.Person;
- i.state.showRemoveDialog = false;
- i.setState(i.state);
+ i.setState({
+ showPurgeDialog: true,
+ purgeType: PurgeType.Person,
+ showRemoveDialog: false,
+ });
}
handlePurgeCommentShow(i: CommentNode) {
- i.state.showPurgeDialog = true;
- i.state.purgeType = PurgeType.Comment;
- i.state.showRemoveDialog = false;
- i.setState(i.state);
+ i.setState({
+ showPurgeDialog: true,
+ purgeType: PurgeType.Comment,
+ showRemoveDialog: false,
+ });
}
handlePurgeReasonChange(i: CommentNode, event: any) {
- i.state.purgeReason = Some(event.target.value);
- i.setState(i.state);
+ i.setState({ purgeReason: event.target.value });
}
handlePurgeSubmit(i: CommentNode, event: any) {
event.preventDefault();
+ let auth = myAuth();
+ if (auth) {
+ if (i.state.purgeType == PurgeType.Person) {
+ let form: PurgePerson = {
+ person_id: i.props.node.comment_view.creator.id,
+ reason: i.state.purgeReason,
+ auth,
+ };
+ WebSocketService.Instance.send(wsClient.purgePerson(form));
+ } else if (i.state.purgeType == PurgeType.Comment) {
+ let form: PurgeComment = {
+ comment_id: i.props.node.comment_view.comment.id,
+ reason: i.state.purgeReason,
+ auth,
+ };
+ WebSocketService.Instance.send(wsClient.purgeComment(form));
+ }
- if (i.state.purgeType == PurgeType.Person) {
- let form = new PurgePerson({
- person_id: i.props.node.comment_view.creator.id,
- reason: i.state.purgeReason,
- auth: auth().unwrap(),
- });
- WebSocketService.Instance.send(wsClient.purgePerson(form));
- } else if (i.state.purgeType == PurgeType.Comment) {
- let form = new PurgeComment({
- comment_id: i.props.node.comment_view.comment.id,
- reason: i.state.purgeReason,
- auth: auth().unwrap(),
- });
- WebSocketService.Instance.send(wsClient.purgeComment(form));
+ i.setState({ purgeLoading: true });
}
-
- i.state.purgeLoading = true;
- i.setState(i.state);
}
handleShowConfirmAppointAsMod(i: CommentNode) {
- i.state.showConfirmAppointAsMod = true;
- i.setState(i.state);
+ i.setState({ showConfirmAppointAsMod: true });
}
handleCancelConfirmAppointAsMod(i: CommentNode) {
- i.state.showConfirmAppointAsMod = false;
- i.setState(i.state);
+ i.setState({ showConfirmAppointAsMod: false });
}
handleAddModToCommunity(i: CommentNode) {
let cv = i.props.node.comment_view;
- let form = new AddModToCommunity({
- person_id: cv.creator.id,
- community_id: cv.community.id,
- added: !isMod(i.props.moderators, cv.creator.id),
- auth: auth().unwrap(),
- });
- WebSocketService.Instance.send(wsClient.addModToCommunity(form));
- i.state.showConfirmAppointAsMod = false;
- i.setState(i.state);
+ let auth = myAuth();
+ if (auth) {
+ let form: AddModToCommunity = {
+ person_id: cv.creator.id,
+ community_id: cv.community.id,
+ added: !isMod(cv.creator.id, i.props.moderators),
+ auth,
+ };
+ WebSocketService.Instance.send(wsClient.addModToCommunity(form));
+ i.setState({ showConfirmAppointAsMod: false });
+ }
}
handleShowConfirmAppointAsAdmin(i: CommentNode) {
- i.state.showConfirmAppointAsAdmin = true;
- i.setState(i.state);
+ i.setState({ showConfirmAppointAsAdmin: true });
}
handleCancelConfirmAppointAsAdmin(i: CommentNode) {
- i.state.showConfirmAppointAsAdmin = false;
- i.setState(i.state);
+ i.setState({ showConfirmAppointAsAdmin: false });
}
handleAddAdmin(i: CommentNode) {
- let creatorId = i.props.node.comment_view.creator.id;
- let form = new AddAdmin({
- person_id: creatorId,
- added: !isAdmin(i.props.admins, creatorId),
- auth: auth().unwrap(),
- });
- WebSocketService.Instance.send(wsClient.addAdmin(form));
- i.state.showConfirmAppointAsAdmin = false;
- i.setState(i.state);
+ let auth = myAuth();
+ if (auth) {
+ let creatorId = i.props.node.comment_view.creator.id;
+ let form: AddAdmin = {
+ person_id: creatorId,
+ added: !isAdmin(creatorId, i.props.admins),
+ auth,
+ };
+ WebSocketService.Instance.send(wsClient.addAdmin(form));
+ i.setState({ showConfirmAppointAsAdmin: false });
+ }
}
handleShowConfirmTransferCommunity(i: CommentNode) {
- i.state.showConfirmTransferCommunity = true;
- i.setState(i.state);
+ i.setState({ showConfirmTransferCommunity: true });
}
handleCancelShowConfirmTransferCommunity(i: CommentNode) {
- i.state.showConfirmTransferCommunity = false;
- i.setState(i.state);
+ i.setState({ showConfirmTransferCommunity: false });
}
handleTransferCommunity(i: CommentNode) {
let cv = i.props.node.comment_view;
- let form = new TransferCommunity({
- community_id: cv.community.id,
- person_id: cv.creator.id,
- auth: auth().unwrap(),
- });
- WebSocketService.Instance.send(wsClient.transferCommunity(form));
- i.state.showConfirmTransferCommunity = false;
- i.setState(i.state);
+ let auth = myAuth();
+ if (auth) {
+ let form: TransferCommunity = {
+ community_id: cv.community.id,
+ person_id: cv.creator.id,
+ auth,
+ };
+ WebSocketService.Instance.send(wsClient.transferCommunity(form));
+ i.setState({ showConfirmTransferCommunity: false });
+ }
}
handleShowConfirmTransferSite(i: CommentNode) {
- i.state.showConfirmTransferSite = true;
- i.setState(i.state);
+ i.setState({ showConfirmTransferSite: true });
}
handleCancelShowConfirmTransferSite(i: CommentNode) {
- i.state.showConfirmTransferSite = false;
- i.setState(i.state);
+ i.setState({ showConfirmTransferSite: false });
}
get isCommentNew(): boolean {
}
handleCommentCollapse(i: CommentNode) {
- i.state.collapsed = !i.state.collapsed;
- i.setState(i.state);
+ i.setState({ collapsed: !i.state.collapsed });
setupTippy();
}
handleViewSource(i: CommentNode) {
- i.state.viewSource = !i.state.viewSource;
- i.setState(i.state);
+ i.setState({ viewSource: !i.state.viewSource });
}
handleShowAdvanced(i: CommentNode) {
- i.state.showAdvanced = !i.state.showAdvanced;
- i.setState(i.state);
+ i.setState({ showAdvanced: !i.state.showAdvanced });
setupTippy();
}
handleFetchChildren(i: CommentNode) {
- let form = new GetComments({
- post_id: Some(i.props.node.comment_view.post.id),
- parent_id: Some(i.props.node.comment_view.comment.id),
- max_depth: Some(commentTreeMaxDepth),
- page: None,
- sort: None,
- limit: Some(999),
- type_: Some(ListingType.All),
- community_name: None,
- community_id: None,
- saved_only: Some(false),
- auth: auth(false).ok(),
- });
+ let form: GetComments = {
+ post_id: i.props.node.comment_view.post.id,
+ parent_id: i.props.node.comment_view.comment.id,
+ max_depth: commentTreeMaxDepth,
+ limit: 999, // TODO
+ type_: "All",
+ saved_only: false,
+ auth: myAuth(false),
+ };
WebSocketService.Instance.send(wsClient.getComments(form));
}
get scoreColor() {
- if (this.state.my_vote.unwrapOr(0) == 1) {
+ if (this.state.my_vote == 1) {
return "text-info";
- } else if (this.state.my_vote.unwrapOr(0) == -1) {
+ } else if (this.state.my_vote == -1) {
return "text-danger";
} else {
return "text-muted";
get pointsTippy(): string {
let points = i18n.t("number_of_points", {
- count: this.state.score,
- formattedCount: this.state.score,
+ count: Number(this.state.score),
+ formattedCount: numToSI(this.state.score),
});
let upvotes = i18n.t("number_of_upvotes", {
- count: this.state.upvotes,
- formattedCount: this.state.upvotes,
+ count: Number(this.state.upvotes),
+ formattedCount: numToSI(this.state.upvotes),
});
let downvotes = i18n.t("number_of_downvotes", {
- count: this.state.downvotes,
- formattedCount: this.state.downvotes,
+ count: Number(this.state.downvotes),
+ formattedCount: numToSI(this.state.downvotes),
});
return `${points} • ${upvotes} • ${downvotes}`;