1 import { Component, linkEvent } from 'inferno';
2 import { Link } from 'inferno-router';
3 import { Subscription } from "rxjs";
4 import { retryWhen, delay, take } from 'rxjs/operators';
5 import { UserOperation, GetModlogForm, GetModlogResponse, ModRemovePost, ModLockPost, ModRemoveComment, ModRemoveCommunity, ModBanFromCommunity, ModBan, ModAddCommunity, ModAdd } from '../interfaces';
6 import { WebSocketService } from '../services';
7 import { msgOp, addTypeInfo, fetchLimit } from '../utils';
8 import { MomentTime } from './moment-time';
9 import * as moment from 'moment';
11 interface ModlogState {
12 combined: Array<{type_: string, data: ModRemovePost | ModLockPost | ModRemoveCommunity | ModAdd | ModBan}>,
14 communityName?: string,
19 export class Modlog extends Component<any, ModlogState> {
20 private subscription: Subscription;
21 private emptyState: ModlogState = {
27 constructor(props: any, context: any) {
28 super(props, context);
30 this.state = this.emptyState;
31 this.state.communityId = this.props.match.params.community_id ? Number(this.props.match.params.community_id) : undefined;
32 this.subscription = WebSocketService.Instance.subject
33 .pipe(retryWhen(errors => errors.pipe(delay(3000), take(10))))
35 (msg) => this.parseMessage(msg),
36 (err) => console.error(err),
37 () => console.log('complete')
43 componentWillUnmount() {
44 this.subscription.unsubscribe();
48 document.title = `Modlog - ${WebSocketService.Instance.site.name}`;
51 setCombined(res: GetModlogResponse) {
52 let removed_posts = addTypeInfo(res.removed_posts, "removed_posts");
53 let locked_posts = addTypeInfo(res.locked_posts, "locked_posts");
54 let removed_comments = addTypeInfo(res.removed_comments, "removed_comments");
55 let removed_communities = addTypeInfo(res.removed_communities, "removed_communities");
56 let banned_from_community = addTypeInfo(res.banned_from_community, "banned_from_community");
57 let added_to_community = addTypeInfo(res.added_to_community, "added_to_community");
58 let added = addTypeInfo(res.added, "added");
59 let banned = addTypeInfo(res.banned, "banned");
60 this.state.combined = [];
62 this.state.combined.push(...removed_posts);
63 this.state.combined.push(...locked_posts);
64 this.state.combined.push(...removed_comments);
65 this.state.combined.push(...removed_communities);
66 this.state.combined.push(...banned_from_community);
67 this.state.combined.push(...added_to_community);
68 this.state.combined.push(...added);
69 this.state.combined.push(...banned);
71 if (this.state.communityId && this.state.combined.length > 0) {
72 this.state.communityName = (this.state.combined[0].data as ModRemovePost).community_name;
76 this.state.combined.sort((a, b) => b.data.when_.localeCompare(a.data.when_));
78 this.setState(this.state);
84 {this.state.combined.map(i =>
86 <td><MomentTime data={i.data} /></td>
87 <td><Link to={`/u/${i.data.mod_user_name}`}>{i.data.mod_user_name}</Link></td>
89 {i.type_ == 'removed_posts' &&
91 {(i.data as ModRemovePost).removed? 'Removed' : 'Restored'}
92 <span> Post <Link to={`/post/${(i.data as ModRemovePost).post_id}`}>{(i.data as ModRemovePost).post_name}</Link></span>
93 <div>{(i.data as ModRemovePost).reason && ` reason: ${(i.data as ModRemovePost).reason}`}</div>
96 {i.type_ == 'locked_posts' &&
98 {(i.data as ModLockPost).locked? 'Locked' : 'Unlocked'}
99 <span> Post <Link to={`/post/${(i.data as ModLockPost).post_id}`}>{(i.data as ModLockPost).post_name}</Link></span>
102 {i.type_ == 'removed_comments' &&
104 {(i.data as ModRemoveComment).removed? 'Removed' : 'Restored'}
105 <span> Comment <Link to={`/post/${(i.data as ModRemoveComment).post_id}/comment/${(i.data as ModRemoveComment).comment_id}`}>{(i.data as ModRemoveComment).comment_content}</Link></span>
106 <span> by <Link to={`/u/${(i.data as ModRemoveComment).comment_user_name}`}>{(i.data as ModRemoveComment).comment_user_name}</Link></span>
107 <div>{(i.data as ModRemoveComment).reason && ` reason: ${(i.data as ModRemoveComment).reason}`}</div>
110 {i.type_ == 'removed_communities' &&
112 {(i.data as ModRemoveCommunity).removed ? 'Removed' : 'Restored'}
113 <span> Community <Link to={`/c/${(i.data as ModRemoveCommunity).community_name}`}>{(i.data as ModRemoveCommunity).community_name}</Link></span>
114 <div>{(i.data as ModRemoveCommunity).reason && ` reason: ${(i.data as ModRemoveCommunity).reason}`}</div>
115 <div>{(i.data as ModRemoveCommunity).expires && ` expires: ${moment.utc((i.data as ModRemoveCommunity).expires).fromNow()}`}</div>
118 {i.type_ == 'banned_from_community' &&
120 <span>{(i.data as ModBanFromCommunity).banned ? 'Banned ' : 'Unbanned '} </span>
121 <span><Link to={`/u/${(i.data as ModBanFromCommunity).other_user_name}`}>{(i.data as ModBanFromCommunity).other_user_name}</Link></span>
122 <span> from the community </span>
123 <span><Link to={`/c/${(i.data as ModBanFromCommunity).community_name}`}>{(i.data as ModBanFromCommunity).community_name}</Link></span>
124 <div>{(i.data as ModBanFromCommunity).reason && ` reason: ${(i.data as ModBanFromCommunity).reason}`}</div>
125 <div>{(i.data as ModBanFromCommunity).expires && ` expires: ${moment.utc((i.data as ModBanFromCommunity).expires).fromNow()}`}</div>
128 {i.type_ == 'added_to_community' &&
130 <span>{(i.data as ModAddCommunity).removed ? 'Removed ' : 'Appointed '} </span>
131 <span><Link to={`/u/${(i.data as ModAddCommunity).other_user_name}`}>{(i.data as ModAddCommunity).other_user_name}</Link></span>
132 <span> as a mod to the community </span>
133 <span><Link to={`/c/${(i.data as ModAddCommunity).community_name}`}>{(i.data as ModAddCommunity).community_name}</Link></span>
136 {i.type_ == 'banned' &&
138 <span>{(i.data as ModBan).banned ? 'Banned ' : 'Unbanned '} </span>
139 <span><Link to={`/u/${(i.data as ModBan).other_user_name}`}>{(i.data as ModBan).other_user_name}</Link></span>
140 <div>{(i.data as ModBan).reason && ` reason: ${(i.data as ModBan).reason}`}</div>
141 <div>{(i.data as ModBan).expires && ` expires: ${moment.utc((i.data as ModBan).expires).fromNow()}`}</div>
144 {i.type_ == 'added' &&
146 <span>{(i.data as ModAdd).removed ? 'Removed ' : 'Appointed '} </span>
147 <span><Link to={`/u/${(i.data as ModAdd).other_user_name}`}>{(i.data as ModAdd).other_user_name}</Link></span>
148 <span> as an admin </span>
163 <div class="container">
164 {this.state.loading ?
165 <h5 class=""><svg class="icon icon-spinner spin"><use xlinkHref="#icon-spinner"></use></svg></h5> :
168 {this.state.communityName && <Link className="text-white" to={`/c/${this.state.communityName}`}>/c/{this.state.communityName} </Link>}
171 <div class="table-responsive">
172 <table id="modlog_table" class="table table-sm table-hover">
173 <thead class="pointer">
193 {this.state.page > 1 &&
194 <button class="btn btn-sm btn-secondary mr-1" onClick={linkEvent(this, this.prevPage)}>Prev</button>
196 <button class="btn btn-sm btn-secondary" onClick={linkEvent(this, this.nextPage)}>Next</button>
201 nextPage(i: Modlog) {
207 prevPage(i: Modlog) {
214 let modlogForm: GetModlogForm = {
215 community_id: this.state.communityId,
216 page: this.state.page,
219 WebSocketService.Instance.getModlog(modlogForm);
222 parseMessage(msg: any) {
224 let op: UserOperation = msgOp(msg);
226 alert(i18n.t(msg.error));
228 } else if (op == UserOperation.GetModlog) {
229 let res: GetModlogResponse = msg;
230 this.state.loading = false;
231 window.scrollTo(0,0);
232 this.setCombined(res);