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';
22 WebSocketJsonResponse,
25 } from '../interfaces';
26 import { WebSocketService, UserService } from '../services';
31 capitalizeFirstLetter,
40 createPostLikeFindRes,
44 import { PostListing } from './post-listing';
45 import { UserListing } from './user-listing';
46 import { SortSelect } from './sort-select';
47 import { ListingTypeSelect } from './listing-type-select';
48 import { CommentNodes } from './comment-nodes';
49 import { MomentTime } from './moment-time';
50 import { i18n } from '../i18next';
51 import moment from 'moment';
64 follows: Array<CommunityUser>;
65 moderates: Array<CommunityUser>;
66 comments: Array<Comment>;
69 admins: Array<UserView>;
74 avatarLoading: boolean;
75 userSettingsForm: UserSettingsForm;
76 userSettingsLoading: boolean;
77 deleteAccountLoading: boolean;
78 deleteAccountShowConfirm: boolean;
79 deleteAccountForm: DeleteAccountForm;
83 export class User extends Component<any, UserState> {
84 private subscription: Subscription;
85 private emptyState: UserState = {
90 number_of_posts: null,
92 number_of_comments: null,
97 send_notifications_to_email: null,
109 avatarLoading: false,
110 view: this.getViewFromProps(this.props),
111 sort: this.getSortTypeFromProps(this.props),
112 page: this.getPageFromProps(this.props),
116 default_sort_type: null,
117 default_listing_type: null,
120 send_notifications_to_email: null,
123 userSettingsLoading: null,
124 deleteAccountLoading: null,
125 deleteAccountShowConfirm: false,
132 creator_id: undefined,
133 published: undefined,
134 creator_name: undefined,
135 number_of_users: undefined,
136 number_of_posts: undefined,
137 number_of_comments: undefined,
138 number_of_communities: undefined,
139 enable_downvotes: undefined,
140 open_registration: undefined,
141 enable_nsfw: undefined,
145 constructor(props: any, context: any) {
146 super(props, context);
148 this.state = this.emptyState;
149 this.handleSortChange = this.handleSortChange.bind(this);
150 this.handleUserSettingsSortTypeChange = this.handleUserSettingsSortTypeChange.bind(
153 this.handleUserSettingsListingTypeChange = this.handleUserSettingsListingTypeChange.bind(
157 this.state.user_id = Number(this.props.match.params.id);
158 this.state.username = this.props.match.params.username;
160 this.subscription = WebSocketService.Instance.subject
161 .pipe(retryWhen(errors => errors.pipe(delay(3000), take(10))))
163 msg => this.parseMessage(msg),
164 err => console.error(err),
165 () => console.log('complete')
169 WebSocketService.Instance.getSite();
172 get isCurrentUser() {
174 UserService.Instance.user &&
175 UserService.Instance.user.id == this.state.user.id
179 getViewFromProps(props: any): View {
180 return props.match.params.view
181 ? View[capitalizeFirstLetter(props.match.params.view)]
185 getSortTypeFromProps(props: any): SortType {
186 return props.match.params.sort
187 ? routeSortTypeToEnum(props.match.params.sort)
191 getPageFromProps(props: any): number {
192 return props.match.params.page ? Number(props.match.params.page) : 1;
195 componentWillUnmount() {
196 this.subscription.unsubscribe();
199 // Necessary for back button for some reason
200 componentWillReceiveProps(nextProps: any) {
202 nextProps.history.action == 'POP' ||
203 nextProps.history.action == 'PUSH'
205 this.state.view = this.getViewFromProps(nextProps);
206 this.state.sort = this.getSortTypeFromProps(nextProps);
207 this.state.page = this.getPageFromProps(nextProps);
208 this.setState(this.state);
213 componentDidUpdate(lastProps: any, _lastState: UserState, _snapshot: any) {
214 // Necessary if you are on a post and you click another post (same route)
216 lastProps.location.pathname.split('/')[2] !==
217 lastProps.history.location.pathname.split('/')[2]
219 // Couldnt get a refresh working. This does for now.
226 <div class="container">
227 {this.state.loading ? (
229 <svg class="icon icon-spinner spin">
230 <use xlinkHref="#icon-spinner"></use>
235 <div class="col-12 col-md-8">
237 {this.state.user.avatar && showAvatars() && (
241 src={this.state.user.avatar}
242 class="rounded-circle mr-2"
245 <span>/u/{this.state.user.name}</span>
248 {this.state.view == View.Overview && this.overview()}
249 {this.state.view == View.Comments && this.comments()}
250 {this.state.view == View.Posts && this.posts()}
251 {this.state.view == View.Saved && this.overview()}
254 <div class="col-12 col-md-4">
256 {this.isCurrentUser && this.userSettings()}
268 <div class="btn-group btn-group-toggle">
270 className={`btn btn-sm btn-secondary pointer btn-outline-light
271 ${this.state.view == View.Overview && 'active'}
276 value={View.Overview}
277 checked={this.state.view == View.Overview}
278 onChange={linkEvent(this, this.handleViewChange)}
283 className={`btn btn-sm btn-secondary pointer btn-outline-light
284 ${this.state.view == View.Comments && 'active'}
289 value={View.Comments}
290 checked={this.state.view == View.Comments}
291 onChange={linkEvent(this, this.handleViewChange)}
296 className={`btn btn-sm btn-secondary pointer btn-outline-light
297 ${this.state.view == View.Posts && 'active'}
303 checked={this.state.view == View.Posts}
304 onChange={linkEvent(this, this.handleViewChange)}
309 className={`btn btn-sm btn-secondary pointer btn-outline-light
310 ${this.state.view == View.Saved && 'active'}
316 checked={this.state.view == View.Saved}
317 onChange={linkEvent(this, this.handleViewChange)}
327 <div className="mb-2">
328 <span class="mr-3">{this.viewRadios()}</span>
330 sort={this.state.sort}
331 onChange={this.handleSortChange}
335 href={`/feeds/u/${this.state.username}.xml?sort=${
336 SortType[this.state.sort]
342 <svg class="icon mx-2 text-muted small">
343 <use xlinkHref="#icon-rss">#</use>
351 let combined: Array<{ type_: string; data: Comment | Post }> = [];
352 let comments = this.state.comments.map(e => {
353 return { type_: 'comments', data: e };
355 let posts = this.state.posts.map(e => {
356 return { type_: 'posts', data: e };
359 combined.push(...comments);
360 combined.push(...posts);
363 if (this.state.sort == SortType.New) {
364 combined.sort((a, b) => b.data.published.localeCompare(a.data.published));
366 combined.sort((a, b) => b.data.score - a.data.score);
373 {i.type_ == 'posts' ? (
375 post={i.data as Post}
376 admins={this.state.admins}
378 enableDownvotes={this.state.site.enable_downvotes}
379 enableNsfw={this.state.site.enable_nsfw}
383 nodes={[{ comment: i.data as Comment }]}
384 admins={this.state.admins}
388 enableDownvotes={this.state.site.enable_downvotes}
401 nodes={commentsToFlatNodes(this.state.comments)}
402 admins={this.state.admins}
406 enableDownvotes={this.state.site.enable_downvotes}
415 {this.state.posts.map(post => (
418 admins={this.state.admins}
420 enableDownvotes={this.state.site.enable_downvotes}
421 enableNsfw={this.state.site.enable_nsfw}
429 let user = this.state.user;
432 <div class="card border-secondary mb-3">
433 <div class="card-body">
435 <ul class="list-inline mb-0">
436 <li className="list-inline-item">
437 <UserListing user={user} realLink />
440 <li className="list-inline-item badge badge-danger">
446 <div className="d-flex align-items-center mb-2">
448 <use xlinkHref="#icon-cake"></use>
450 <span className="ml-2">
451 {i18n.t('cake_day_title')}{' '}
452 {moment.utc(user.published).local().format('MMM DD, YYYY')}
456 {i18n.t('joined')} <MomentTime data={user} showAgo />
458 <div class="table-responsive mt-1">
459 <table class="table table-bordered table-sm mt-2 mb-0">
462 <td class="text-center" colSpan={2}>
463 {i18n.t('number_of_points', {
464 count: user.post_score + user.comment_score,
472 {i18n.t('number_of_points', { count: user.post_score })}
476 {i18n.t('number_of_posts', { count: user.number_of_posts })}
482 {i18n.t('number_of_points', { count: user.comment_score })}
486 {i18n.t('number_of_comments', {
487 count: user.number_of_comments,
493 {this.isCurrentUser ? (
495 class="btn btn-block btn-secondary mt-3"
496 onClick={linkEvent(this, this.handleLogoutClick)}
503 className={`btn btn-block btn-secondary mt-3 ${
504 !this.state.user.matrix_user_id && 'disabled'
508 href={`https://matrix.to/#/${this.state.user.matrix_user_id}`}
510 {i18n.t('send_secure_message')}
513 class="btn btn-block btn-secondary mt-3"
514 to={`/create_private_message?recipient_id=${this.state.user.id}`}
516 {i18n.t('send_message')}
529 <div class="card border-secondary mb-3">
530 <div class="card-body">
531 <h5>{i18n.t('settings')}</h5>
532 <form onSubmit={linkEvent(this, this.handleUserSettingsSubmit)}>
533 <div class="form-group">
534 <label>{i18n.t('avatar')}</label>
535 <form class="d-inline">
537 htmlFor="file-upload"
538 class="pointer ml-4 text-muted small font-weight-bold"
540 {!this.checkSettingsAvatar ? (
541 <span class="btn btn-sm btn-secondary">
542 {i18n.t('upload_avatar')}
548 src={this.state.userSettingsForm.avatar}
549 class="rounded-circle"
556 accept="image/*,video/*"
559 disabled={!UserService.Instance.user}
560 onChange={linkEvent(this, this.handleImageUpload)}
564 {this.checkSettingsAvatar && (
565 <div class="form-group">
567 class="btn btn-secondary btn-block"
568 onClick={linkEvent(this, this.removeAvatar)}
570 {`${capitalizeFirstLetter(i18n.t('remove'))} ${i18n.t(
576 <div class="form-group">
577 <label>{i18n.t('language')}</label>
579 value={this.state.userSettingsForm.lang}
580 onChange={linkEvent(this, this.handleUserSettingsLangChange)}
581 class="ml-2 custom-select custom-select-sm w-auto"
583 <option disabled>{i18n.t('language')}</option>
584 <option value="browser">{i18n.t('browser_default')}</option>
585 <option disabled>──</option>
586 {languages.map(lang => (
587 <option value={lang.code}>{lang.name}</option>
591 <div class="form-group">
592 <label>{i18n.t('theme')}</label>
594 value={this.state.userSettingsForm.theme}
595 onChange={linkEvent(this, this.handleUserSettingsThemeChange)}
596 class="ml-2 custom-select custom-select-sm w-auto"
598 <option disabled>{i18n.t('theme')}</option>
599 {themes.map(theme => (
600 <option value={theme}>{theme}</option>
604 <form className="form-group">
606 <div class="mr-2">{i18n.t('sort_type')}</div>
609 type_={this.state.userSettingsForm.default_listing_type}
610 onChange={this.handleUserSettingsListingTypeChange}
613 <form className="form-group">
615 <div class="mr-2">{i18n.t('type')}</div>
618 sort={this.state.userSettingsForm.default_sort_type}
619 onChange={this.handleUserSettingsSortTypeChange}
622 <div class="form-group row">
623 <label class="col-lg-3 col-form-label" htmlFor="user-email">
626 <div class="col-lg-9">
631 placeholder={i18n.t('optional')}
632 value={this.state.userSettingsForm.email}
635 this.handleUserSettingsEmailChange
641 <div class="form-group row">
642 <label class="col-lg-5 col-form-label">
644 href="https://about.riot.im/"
648 {i18n.t('matrix_user_id')}
651 <div class="col-lg-7">
655 placeholder="@user:example.com"
656 value={this.state.userSettingsForm.matrix_user_id}
659 this.handleUserSettingsMatrixUserIdChange
665 <div class="form-group row">
666 <label class="col-lg-5 col-form-label" htmlFor="user-password">
667 {i18n.t('new_password')}
669 <div class="col-lg-7">
674 value={this.state.userSettingsForm.new_password}
675 autoComplete="new-password"
678 this.handleUserSettingsNewPasswordChange
683 <div class="form-group row">
685 class="col-lg-5 col-form-label"
686 htmlFor="user-verify-password"
688 {i18n.t('verify_password')}
690 <div class="col-lg-7">
693 id="user-verify-password"
695 value={this.state.userSettingsForm.new_password_verify}
696 autoComplete="new-password"
699 this.handleUserSettingsNewPasswordVerifyChange
704 <div class="form-group row">
706 class="col-lg-5 col-form-label"
707 htmlFor="user-old-password"
709 {i18n.t('old_password')}
711 <div class="col-lg-7">
714 id="user-old-password"
716 value={this.state.userSettingsForm.old_password}
717 autoComplete="new-password"
720 this.handleUserSettingsOldPasswordChange
725 {this.state.site.enable_nsfw && (
726 <div class="form-group">
727 <div class="form-check">
729 class="form-check-input"
732 checked={this.state.userSettingsForm.show_nsfw}
735 this.handleUserSettingsShowNsfwChange
738 <label class="form-check-label" htmlFor="user-show-nsfw">
739 {i18n.t('show_nsfw')}
744 <div class="form-group">
745 <div class="form-check">
747 class="form-check-input"
748 id="user-show-avatars"
750 checked={this.state.userSettingsForm.show_avatars}
753 this.handleUserSettingsShowAvatarsChange
756 <label class="form-check-label" htmlFor="user-show-avatars">
757 {i18n.t('show_avatars')}
761 <div class="form-group">
762 <div class="form-check">
764 class="form-check-input"
765 id="user-send-notifications-to-email"
767 disabled={!this.state.user.email}
769 this.state.userSettingsForm.send_notifications_to_email
773 this.handleUserSettingsSendNotificationsToEmailChange
777 class="form-check-label"
778 htmlFor="user-send-notifications-to-email"
780 {i18n.t('send_notifications_to_email')}
784 <div class="form-group">
785 <button type="submit" class="btn btn-block btn-secondary mr-4">
786 {this.state.userSettingsLoading ? (
787 <svg class="icon icon-spinner spin">
788 <use xlinkHref="#icon-spinner"></use>
791 capitalizeFirstLetter(i18n.t('save'))
796 <div class="form-group mb-0">
798 class="btn btn-block btn-danger"
801 this.handleDeleteAccountShowConfirmToggle
804 {i18n.t('delete_account')}
806 {this.state.deleteAccountShowConfirm && (
808 <div class="my-2 alert alert-danger" role="alert">
809 {i18n.t('delete_account_confirm')}
813 value={this.state.deleteAccountForm.password}
814 autoComplete="new-password"
817 this.handleDeleteAccountPasswordChange
819 class="form-control my-2"
822 class="btn btn-danger mr-4"
823 disabled={!this.state.deleteAccountForm.password}
824 onClick={linkEvent(this, this.handleDeleteAccount)}
826 {this.state.deleteAccountLoading ? (
827 <svg class="icon icon-spinner spin">
828 <use xlinkHref="#icon-spinner"></use>
831 capitalizeFirstLetter(i18n.t('delete'))
835 class="btn btn-secondary"
838 this.handleDeleteAccountShowConfirmToggle
856 {this.state.moderates.length > 0 && (
857 <div class="card border-secondary mb-3">
858 <div class="card-body">
859 <h5>{i18n.t('moderates')}</h5>
860 <ul class="list-unstyled mb-0">
861 {this.state.moderates.map(community => (
863 <Link to={`/c/${community.community_name}`}>
864 {community.community_name}
879 {this.state.follows.length > 0 && (
880 <div class="card border-secondary mb-3">
881 <div class="card-body">
882 <h5>{i18n.t('subscribed')}</h5>
883 <ul class="list-unstyled mb-0">
884 {this.state.follows.map(community => (
886 <Link to={`/c/${community.community_name}`}>
887 {community.community_name}
902 {this.state.page > 1 && (
904 class="btn btn-sm btn-secondary mr-1"
905 onClick={linkEvent(this, this.prevPage)}
910 {this.state.comments.length + this.state.posts.length > 0 && (
912 class="btn btn-sm btn-secondary"
913 onClick={linkEvent(this, this.nextPage)}
923 let viewStr = View[this.state.view].toLowerCase();
924 let sortStr = SortType[this.state.sort].toLowerCase();
925 this.props.history.push(
926 `/u/${this.state.user.name}/view/${viewStr}/sort/${sortStr}/page/${this.state.page}`
945 let form: GetUserDetailsForm = {
946 user_id: this.state.user_id,
947 username: this.state.username,
948 sort: SortType[this.state.sort],
949 saved_only: this.state.view == View.Saved,
950 page: this.state.page,
953 WebSocketService.Instance.getUserDetails(form);
956 handleSortChange(val: SortType) {
957 this.state.sort = val;
959 this.setState(this.state);
964 handleViewChange(i: User, event: any) {
965 i.state.view = Number(event.target.value);
972 handleUserSettingsShowNsfwChange(i: User, event: any) {
973 i.state.userSettingsForm.show_nsfw = event.target.checked;
977 handleUserSettingsShowAvatarsChange(i: User, event: any) {
978 i.state.userSettingsForm.show_avatars = event.target.checked;
979 UserService.Instance.user.show_avatars = event.target.checked; // Just for instant updates
983 handleUserSettingsSendNotificationsToEmailChange(i: User, event: any) {
984 i.state.userSettingsForm.send_notifications_to_email = event.target.checked;
988 handleUserSettingsThemeChange(i: User, event: any) {
989 i.state.userSettingsForm.theme = event.target.value;
990 setTheme(event.target.value, true);
994 handleUserSettingsLangChange(i: User, event: any) {
995 i.state.userSettingsForm.lang = event.target.value;
996 i18n.changeLanguage(i.state.userSettingsForm.lang);
1000 handleUserSettingsSortTypeChange(val: SortType) {
1001 this.state.userSettingsForm.default_sort_type = val;
1002 this.setState(this.state);
1005 handleUserSettingsListingTypeChange(val: ListingType) {
1006 this.state.userSettingsForm.default_listing_type = val;
1007 this.setState(this.state);
1010 handleUserSettingsEmailChange(i: User, event: any) {
1011 i.state.userSettingsForm.email = event.target.value;
1012 if (i.state.userSettingsForm.email == '' && !i.state.user.email) {
1013 i.state.userSettingsForm.email = undefined;
1015 i.setState(i.state);
1018 handleUserSettingsMatrixUserIdChange(i: User, event: any) {
1019 i.state.userSettingsForm.matrix_user_id = event.target.value;
1021 i.state.userSettingsForm.matrix_user_id == '' &&
1022 !i.state.user.matrix_user_id
1024 i.state.userSettingsForm.matrix_user_id = undefined;
1026 i.setState(i.state);
1029 handleUserSettingsNewPasswordChange(i: User, event: any) {
1030 i.state.userSettingsForm.new_password = event.target.value;
1031 if (i.state.userSettingsForm.new_password == '') {
1032 i.state.userSettingsForm.new_password = undefined;
1034 i.setState(i.state);
1037 handleUserSettingsNewPasswordVerifyChange(i: User, event: any) {
1038 i.state.userSettingsForm.new_password_verify = event.target.value;
1039 if (i.state.userSettingsForm.new_password_verify == '') {
1040 i.state.userSettingsForm.new_password_verify = undefined;
1042 i.setState(i.state);
1045 handleUserSettingsOldPasswordChange(i: User, event: any) {
1046 i.state.userSettingsForm.old_password = event.target.value;
1047 if (i.state.userSettingsForm.old_password == '') {
1048 i.state.userSettingsForm.old_password = undefined;
1050 i.setState(i.state);
1053 handleImageUpload(i: User, event: any) {
1054 event.preventDefault();
1055 let file = event.target.files[0];
1056 const imageUploadUrl = `/pictrs/image`;
1057 const formData = new FormData();
1058 formData.append('images[]', file);
1060 i.state.avatarLoading = true;
1061 i.setState(i.state);
1063 fetch(imageUploadUrl, {
1067 .then(res => res.json())
1069 console.log('pictrs upload:');
1071 if (res.msg == 'ok') {
1072 let hash = res.files[0].file;
1073 let url = `${window.location.origin}/pictrs/image/${hash}`;
1074 i.state.userSettingsForm.avatar = url;
1075 i.state.avatarLoading = false;
1076 i.setState(i.state);
1078 i.state.avatarLoading = false;
1079 i.setState(i.state);
1080 toast(JSON.stringify(res), 'danger');
1084 i.state.avatarLoading = false;
1085 i.setState(i.state);
1086 toast(error, 'danger');
1090 removeAvatar(i: User, event: any) {
1091 event.preventDefault();
1092 i.state.userSettingsLoading = true;
1093 i.state.userSettingsForm.avatar = '';
1094 i.setState(i.state);
1096 WebSocketService.Instance.saveUserSettings(i.state.userSettingsForm);
1099 get checkSettingsAvatar(): boolean {
1101 this.state.userSettingsForm.avatar &&
1102 this.state.userSettingsForm.avatar != ''
1106 handleUserSettingsSubmit(i: User, event: any) {
1107 event.preventDefault();
1108 i.state.userSettingsLoading = true;
1109 i.setState(i.state);
1111 WebSocketService.Instance.saveUserSettings(i.state.userSettingsForm);
1114 handleDeleteAccountShowConfirmToggle(i: User, event: any) {
1115 event.preventDefault();
1116 i.state.deleteAccountShowConfirm = !i.state.deleteAccountShowConfirm;
1117 i.setState(i.state);
1120 handleDeleteAccountPasswordChange(i: User, event: any) {
1121 i.state.deleteAccountForm.password = event.target.value;
1122 i.setState(i.state);
1125 handleLogoutClick(i: User) {
1126 UserService.Instance.logout();
1127 i.context.router.history.push('/');
1130 handleDeleteAccount(i: User, event: any) {
1131 event.preventDefault();
1132 i.state.deleteAccountLoading = true;
1133 i.setState(i.state);
1135 WebSocketService.Instance.deleteAccount(i.state.deleteAccountForm);
1138 parseMessage(msg: WebSocketJsonResponse) {
1140 let res = wsJsonToRes(msg);
1142 toast(i18n.t(msg.error), 'danger');
1143 this.state.deleteAccountLoading = false;
1144 this.state.avatarLoading = false;
1145 this.state.userSettingsLoading = false;
1146 if (msg.error == 'couldnt_find_that_username_or_email') {
1147 this.context.router.history.push('/');
1149 this.setState(this.state);
1151 } else if (msg.reconnect) {
1153 } else if (res.op == UserOperation.GetUserDetails) {
1154 let data = res.data as UserDetailsResponse;
1155 this.state.user = data.user;
1156 this.state.comments = data.comments;
1157 this.state.follows = data.follows;
1158 this.state.moderates = data.moderates;
1159 this.state.posts = data.posts;
1160 this.state.admins = data.admins;
1161 this.state.loading = false;
1162 if (this.isCurrentUser) {
1163 this.state.userSettingsForm.show_nsfw =
1164 UserService.Instance.user.show_nsfw;
1165 this.state.userSettingsForm.theme = UserService.Instance.user.theme
1166 ? UserService.Instance.user.theme
1168 this.state.userSettingsForm.default_sort_type =
1169 UserService.Instance.user.default_sort_type;
1170 this.state.userSettingsForm.default_listing_type =
1171 UserService.Instance.user.default_listing_type;
1172 this.state.userSettingsForm.lang = UserService.Instance.user.lang;
1173 this.state.userSettingsForm.avatar = UserService.Instance.user.avatar;
1174 this.state.userSettingsForm.email = this.state.user.email;
1175 this.state.userSettingsForm.send_notifications_to_email = this.state.user.send_notifications_to_email;
1176 this.state.userSettingsForm.show_avatars =
1177 UserService.Instance.user.show_avatars;
1178 this.state.userSettingsForm.matrix_user_id = this.state.user.matrix_user_id;
1180 document.title = `/u/${this.state.user.name} - ${this.state.site.name}`;
1181 window.scrollTo(0, 0);
1182 this.setState(this.state);
1184 } else if (res.op == UserOperation.EditComment) {
1185 let data = res.data as CommentResponse;
1186 editCommentRes(data, this.state.comments);
1187 this.setState(this.state);
1188 } else if (res.op == UserOperation.CreateComment) {
1189 let data = res.data as CommentResponse;
1191 UserService.Instance.user &&
1192 data.comment.creator_id == UserService.Instance.user.id
1194 toast(i18n.t('reply_sent'));
1196 } else if (res.op == UserOperation.SaveComment) {
1197 let data = res.data as CommentResponse;
1198 saveCommentRes(data, this.state.comments);
1199 this.setState(this.state);
1200 } else if (res.op == UserOperation.CreateCommentLike) {
1201 let data = res.data as CommentResponse;
1202 createCommentLikeRes(data, this.state.comments);
1203 this.setState(this.state);
1204 } else if (res.op == UserOperation.CreatePostLike) {
1205 let data = res.data as PostResponse;
1206 createPostLikeFindRes(data, this.state.posts);
1207 this.setState(this.state);
1208 } else if (res.op == UserOperation.BanUser) {
1209 let data = res.data as BanUserResponse;
1211 .filter(c => c.creator_id == data.user.id)
1212 .forEach(c => (c.banned = data.banned));
1214 .filter(c => c.creator_id == data.user.id)
1215 .forEach(c => (c.banned = data.banned));
1216 this.setState(this.state);
1217 } else if (res.op == UserOperation.AddAdmin) {
1218 let data = res.data as AddAdminResponse;
1219 this.state.admins = data.admins;
1220 this.setState(this.state);
1221 } else if (res.op == UserOperation.SaveUserSettings) {
1222 let data = res.data as LoginResponse;
1223 this.state.userSettingsLoading = false;
1224 this.setState(this.state);
1225 UserService.Instance.login(data);
1226 } else if (res.op == UserOperation.DeleteAccount) {
1227 this.state.deleteAccountLoading = false;
1228 this.state.deleteAccountShowConfirm = false;
1229 this.setState(this.state);
1230 this.context.router.history.push('/');
1231 } else if (res.op == UserOperation.GetSite) {
1232 let data = res.data as GetSiteResponse;
1233 this.state.site = data.site;
1234 this.setState(this.state);