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}
387 enableDownvotes={this.state.site.enable_downvotes}
400 nodes={commentsToFlatNodes(this.state.comments)}
401 admins={this.state.admins}
404 enableDownvotes={this.state.site.enable_downvotes}
413 {this.state.posts.map(post => (
416 admins={this.state.admins}
418 enableDownvotes={this.state.site.enable_downvotes}
419 enableNsfw={this.state.site.enable_nsfw}
427 let user = this.state.user;
430 <div class="card border-secondary mb-3">
431 <div class="card-body">
433 <ul class="list-inline mb-0">
434 <li className="list-inline-item">
435 <UserListing user={user} realLink />
438 <li className="list-inline-item badge badge-danger">
444 <div className="d-flex align-items-center mb-2">
446 <use xlinkHref="#icon-cake"></use>
448 <span className="ml-2">
449 {i18n.t('cake_day_title')}{' '}
450 {moment.utc(user.published).local().format('MMM DD, YYYY')}
454 {i18n.t('joined')} <MomentTime data={user} showAgo />
456 <div class="table-responsive mt-1">
457 <table class="table table-bordered table-sm mt-2 mb-0">
460 <td class="text-center" colSpan={2}>
461 {i18n.t('number_of_points', {
462 count: user.post_score + user.comment_score,
470 {i18n.t('number_of_points', { count: user.post_score })}
474 {i18n.t('number_of_posts', { count: user.number_of_posts })}
480 {i18n.t('number_of_points', { count: user.comment_score })}
484 {i18n.t('number_of_comments', {
485 count: user.number_of_comments,
491 {this.isCurrentUser ? (
493 class="btn btn-block btn-secondary mt-3"
494 onClick={linkEvent(this, this.handleLogoutClick)}
501 className={`btn btn-block btn-secondary mt-3 ${
502 !this.state.user.matrix_user_id && 'disabled'
506 href={`https://matrix.to/#/${this.state.user.matrix_user_id}`}
508 {i18n.t('send_secure_message')}
511 class="btn btn-block btn-secondary mt-3"
512 to={`/create_private_message?recipient_id=${this.state.user.id}`}
514 {i18n.t('send_message')}
527 <div class="card border-secondary mb-3">
528 <div class="card-body">
529 <h5>{i18n.t('settings')}</h5>
530 <form onSubmit={linkEvent(this, this.handleUserSettingsSubmit)}>
531 <div class="form-group">
532 <label>{i18n.t('avatar')}</label>
533 <form class="d-inline">
535 htmlFor="file-upload"
536 class="pointer ml-4 text-muted small font-weight-bold"
538 {!this.checkSettingsAvatar ? (
539 <span class="btn btn-sm btn-secondary">
540 {i18n.t('upload_avatar')}
546 src={this.state.userSettingsForm.avatar}
547 class="rounded-circle"
554 accept="image/*,video/*"
557 disabled={!UserService.Instance.user}
558 onChange={linkEvent(this, this.handleImageUpload)}
562 {this.checkSettingsAvatar && (
563 <div class="form-group">
565 class="btn btn-secondary btn-block"
566 onClick={linkEvent(this, this.removeAvatar)}
568 {`${capitalizeFirstLetter(i18n.t('remove'))} ${i18n.t(
574 <div class="form-group">
575 <label>{i18n.t('language')}</label>
577 value={this.state.userSettingsForm.lang}
578 onChange={linkEvent(this, this.handleUserSettingsLangChange)}
579 class="ml-2 custom-select custom-select-sm w-auto"
581 <option disabled>{i18n.t('language')}</option>
582 <option value="browser">{i18n.t('browser_default')}</option>
583 <option disabled>──</option>
584 {languages.map(lang => (
585 <option value={lang.code}>{lang.name}</option>
589 <div class="form-group">
590 <label>{i18n.t('theme')}</label>
592 value={this.state.userSettingsForm.theme}
593 onChange={linkEvent(this, this.handleUserSettingsThemeChange)}
594 class="ml-2 custom-select custom-select-sm w-auto"
596 <option disabled>{i18n.t('theme')}</option>
597 {themes.map(theme => (
598 <option value={theme}>{theme}</option>
602 <form className="form-group">
604 <div class="mr-2">{i18n.t('sort_type')}</div>
607 type_={this.state.userSettingsForm.default_listing_type}
608 onChange={this.handleUserSettingsListingTypeChange}
611 <form className="form-group">
613 <div class="mr-2">{i18n.t('type')}</div>
616 sort={this.state.userSettingsForm.default_sort_type}
617 onChange={this.handleUserSettingsSortTypeChange}
620 <div class="form-group row">
621 <label class="col-lg-3 col-form-label" htmlFor="user-email">
624 <div class="col-lg-9">
629 placeholder={i18n.t('optional')}
630 value={this.state.userSettingsForm.email}
633 this.handleUserSettingsEmailChange
639 <div class="form-group row">
640 <label class="col-lg-5 col-form-label">
642 href="https://about.riot.im/"
646 {i18n.t('matrix_user_id')}
649 <div class="col-lg-7">
653 placeholder="@user:example.com"
654 value={this.state.userSettingsForm.matrix_user_id}
657 this.handleUserSettingsMatrixUserIdChange
663 <div class="form-group row">
664 <label class="col-lg-5 col-form-label" htmlFor="user-password">
665 {i18n.t('new_password')}
667 <div class="col-lg-7">
672 value={this.state.userSettingsForm.new_password}
673 autoComplete="new-password"
676 this.handleUserSettingsNewPasswordChange
681 <div class="form-group row">
683 class="col-lg-5 col-form-label"
684 htmlFor="user-verify-password"
686 {i18n.t('verify_password')}
688 <div class="col-lg-7">
691 id="user-verify-password"
693 value={this.state.userSettingsForm.new_password_verify}
694 autoComplete="new-password"
697 this.handleUserSettingsNewPasswordVerifyChange
702 <div class="form-group row">
704 class="col-lg-5 col-form-label"
705 htmlFor="user-old-password"
707 {i18n.t('old_password')}
709 <div class="col-lg-7">
712 id="user-old-password"
714 value={this.state.userSettingsForm.old_password}
715 autoComplete="new-password"
718 this.handleUserSettingsOldPasswordChange
723 {this.state.site.enable_nsfw && (
724 <div class="form-group">
725 <div class="form-check">
727 class="form-check-input"
730 checked={this.state.userSettingsForm.show_nsfw}
733 this.handleUserSettingsShowNsfwChange
736 <label class="form-check-label" htmlFor="user-show-nsfw">
737 {i18n.t('show_nsfw')}
742 <div class="form-group">
743 <div class="form-check">
745 class="form-check-input"
746 id="user-show-avatars"
748 checked={this.state.userSettingsForm.show_avatars}
751 this.handleUserSettingsShowAvatarsChange
754 <label class="form-check-label" htmlFor="user-show-avatars">
755 {i18n.t('show_avatars')}
759 <div class="form-group">
760 <div class="form-check">
762 class="form-check-input"
763 id="user-send-notifications-to-email"
765 disabled={!this.state.user.email}
767 this.state.userSettingsForm.send_notifications_to_email
771 this.handleUserSettingsSendNotificationsToEmailChange
775 class="form-check-label"
776 htmlFor="user-send-notifications-to-email"
778 {i18n.t('send_notifications_to_email')}
782 <div class="form-group">
783 <button type="submit" class="btn btn-block btn-secondary mr-4">
784 {this.state.userSettingsLoading ? (
785 <svg class="icon icon-spinner spin">
786 <use xlinkHref="#icon-spinner"></use>
789 capitalizeFirstLetter(i18n.t('save'))
794 <div class="form-group mb-0">
796 class="btn btn-block btn-danger"
799 this.handleDeleteAccountShowConfirmToggle
802 {i18n.t('delete_account')}
804 {this.state.deleteAccountShowConfirm && (
806 <div class="my-2 alert alert-danger" role="alert">
807 {i18n.t('delete_account_confirm')}
811 value={this.state.deleteAccountForm.password}
812 autoComplete="new-password"
815 this.handleDeleteAccountPasswordChange
817 class="form-control my-2"
820 class="btn btn-danger mr-4"
821 disabled={!this.state.deleteAccountForm.password}
822 onClick={linkEvent(this, this.handleDeleteAccount)}
824 {this.state.deleteAccountLoading ? (
825 <svg class="icon icon-spinner spin">
826 <use xlinkHref="#icon-spinner"></use>
829 capitalizeFirstLetter(i18n.t('delete'))
833 class="btn btn-secondary"
836 this.handleDeleteAccountShowConfirmToggle
854 {this.state.moderates.length > 0 && (
855 <div class="card border-secondary mb-3">
856 <div class="card-body">
857 <h5>{i18n.t('moderates')}</h5>
858 <ul class="list-unstyled mb-0">
859 {this.state.moderates.map(community => (
861 <Link to={`/c/${community.community_name}`}>
862 {community.community_name}
877 {this.state.follows.length > 0 && (
878 <div class="card border-secondary mb-3">
879 <div class="card-body">
880 <h5>{i18n.t('subscribed')}</h5>
881 <ul class="list-unstyled mb-0">
882 {this.state.follows.map(community => (
884 <Link to={`/c/${community.community_name}`}>
885 {community.community_name}
900 {this.state.page > 1 && (
902 class="btn btn-sm btn-secondary mr-1"
903 onClick={linkEvent(this, this.prevPage)}
908 {this.state.comments.length + this.state.posts.length > 0 && (
910 class="btn btn-sm btn-secondary"
911 onClick={linkEvent(this, this.nextPage)}
921 let viewStr = View[this.state.view].toLowerCase();
922 let sortStr = SortType[this.state.sort].toLowerCase();
923 this.props.history.push(
924 `/u/${this.state.user.name}/view/${viewStr}/sort/${sortStr}/page/${this.state.page}`
943 let form: GetUserDetailsForm = {
944 user_id: this.state.user_id,
945 username: this.state.username,
946 sort: SortType[this.state.sort],
947 saved_only: this.state.view == View.Saved,
948 page: this.state.page,
951 WebSocketService.Instance.getUserDetails(form);
954 handleSortChange(val: SortType) {
955 this.state.sort = val;
957 this.setState(this.state);
962 handleViewChange(i: User, event: any) {
963 i.state.view = Number(event.target.value);
970 handleUserSettingsShowNsfwChange(i: User, event: any) {
971 i.state.userSettingsForm.show_nsfw = event.target.checked;
975 handleUserSettingsShowAvatarsChange(i: User, event: any) {
976 i.state.userSettingsForm.show_avatars = event.target.checked;
977 UserService.Instance.user.show_avatars = event.target.checked; // Just for instant updates
981 handleUserSettingsSendNotificationsToEmailChange(i: User, event: any) {
982 i.state.userSettingsForm.send_notifications_to_email = event.target.checked;
986 handleUserSettingsThemeChange(i: User, event: any) {
987 i.state.userSettingsForm.theme = event.target.value;
988 setTheme(event.target.value, true);
992 handleUserSettingsLangChange(i: User, event: any) {
993 i.state.userSettingsForm.lang = event.target.value;
994 i18n.changeLanguage(i.state.userSettingsForm.lang);
998 handleUserSettingsSortTypeChange(val: SortType) {
999 this.state.userSettingsForm.default_sort_type = val;
1000 this.setState(this.state);
1003 handleUserSettingsListingTypeChange(val: ListingType) {
1004 this.state.userSettingsForm.default_listing_type = val;
1005 this.setState(this.state);
1008 handleUserSettingsEmailChange(i: User, event: any) {
1009 i.state.userSettingsForm.email = event.target.value;
1010 if (i.state.userSettingsForm.email == '' && !i.state.user.email) {
1011 i.state.userSettingsForm.email = undefined;
1013 i.setState(i.state);
1016 handleUserSettingsMatrixUserIdChange(i: User, event: any) {
1017 i.state.userSettingsForm.matrix_user_id = event.target.value;
1019 i.state.userSettingsForm.matrix_user_id == '' &&
1020 !i.state.user.matrix_user_id
1022 i.state.userSettingsForm.matrix_user_id = undefined;
1024 i.setState(i.state);
1027 handleUserSettingsNewPasswordChange(i: User, event: any) {
1028 i.state.userSettingsForm.new_password = event.target.value;
1029 if (i.state.userSettingsForm.new_password == '') {
1030 i.state.userSettingsForm.new_password = undefined;
1032 i.setState(i.state);
1035 handleUserSettingsNewPasswordVerifyChange(i: User, event: any) {
1036 i.state.userSettingsForm.new_password_verify = event.target.value;
1037 if (i.state.userSettingsForm.new_password_verify == '') {
1038 i.state.userSettingsForm.new_password_verify = undefined;
1040 i.setState(i.state);
1043 handleUserSettingsOldPasswordChange(i: User, event: any) {
1044 i.state.userSettingsForm.old_password = event.target.value;
1045 if (i.state.userSettingsForm.old_password == '') {
1046 i.state.userSettingsForm.old_password = undefined;
1048 i.setState(i.state);
1051 handleImageUpload(i: User, event: any) {
1052 event.preventDefault();
1053 let file = event.target.files[0];
1054 const imageUploadUrl = `/pictrs/image`;
1055 const formData = new FormData();
1056 formData.append('images[]', file);
1058 i.state.avatarLoading = true;
1059 i.setState(i.state);
1061 fetch(imageUploadUrl, {
1065 .then(res => res.json())
1067 console.log('pictrs upload:');
1069 if (res.msg == 'ok') {
1070 let hash = res.files[0].file;
1071 let url = `${window.location.origin}/pictrs/image/${hash}`;
1072 i.state.userSettingsForm.avatar = url;
1073 i.state.avatarLoading = false;
1074 i.setState(i.state);
1076 i.state.avatarLoading = false;
1077 i.setState(i.state);
1078 toast(JSON.stringify(res), 'danger');
1082 i.state.avatarLoading = false;
1083 i.setState(i.state);
1084 toast(error, 'danger');
1088 removeAvatar(i: User, event: any) {
1089 event.preventDefault();
1090 i.state.userSettingsLoading = true;
1091 i.state.userSettingsForm.avatar = '';
1092 i.setState(i.state);
1094 WebSocketService.Instance.saveUserSettings(i.state.userSettingsForm);
1097 get checkSettingsAvatar(): boolean {
1099 this.state.userSettingsForm.avatar &&
1100 this.state.userSettingsForm.avatar != ''
1104 handleUserSettingsSubmit(i: User, event: any) {
1105 event.preventDefault();
1106 i.state.userSettingsLoading = true;
1107 i.setState(i.state);
1109 WebSocketService.Instance.saveUserSettings(i.state.userSettingsForm);
1112 handleDeleteAccountShowConfirmToggle(i: User, event: any) {
1113 event.preventDefault();
1114 i.state.deleteAccountShowConfirm = !i.state.deleteAccountShowConfirm;
1115 i.setState(i.state);
1118 handleDeleteAccountPasswordChange(i: User, event: any) {
1119 i.state.deleteAccountForm.password = event.target.value;
1120 i.setState(i.state);
1123 handleLogoutClick(i: User) {
1124 UserService.Instance.logout();
1125 i.context.router.history.push('/');
1128 handleDeleteAccount(i: User, event: any) {
1129 event.preventDefault();
1130 i.state.deleteAccountLoading = true;
1131 i.setState(i.state);
1133 WebSocketService.Instance.deleteAccount(i.state.deleteAccountForm);
1136 parseMessage(msg: WebSocketJsonResponse) {
1138 let res = wsJsonToRes(msg);
1140 toast(i18n.t(msg.error), 'danger');
1141 this.state.deleteAccountLoading = false;
1142 this.state.avatarLoading = false;
1143 this.state.userSettingsLoading = false;
1144 if (msg.error == 'couldnt_find_that_username_or_email') {
1145 this.context.router.history.push('/');
1147 this.setState(this.state);
1149 } else if (msg.reconnect) {
1151 } else if (res.op == UserOperation.GetUserDetails) {
1152 let data = res.data as UserDetailsResponse;
1153 this.state.user = data.user;
1154 this.state.comments = data.comments;
1155 this.state.follows = data.follows;
1156 this.state.moderates = data.moderates;
1157 this.state.posts = data.posts;
1158 this.state.admins = data.admins;
1159 this.state.loading = false;
1160 if (this.isCurrentUser) {
1161 this.state.userSettingsForm.show_nsfw =
1162 UserService.Instance.user.show_nsfw;
1163 this.state.userSettingsForm.theme = UserService.Instance.user.theme
1164 ? UserService.Instance.user.theme
1166 this.state.userSettingsForm.default_sort_type =
1167 UserService.Instance.user.default_sort_type;
1168 this.state.userSettingsForm.default_listing_type =
1169 UserService.Instance.user.default_listing_type;
1170 this.state.userSettingsForm.lang = UserService.Instance.user.lang;
1171 this.state.userSettingsForm.avatar = UserService.Instance.user.avatar;
1172 this.state.userSettingsForm.email = this.state.user.email;
1173 this.state.userSettingsForm.send_notifications_to_email = this.state.user.send_notifications_to_email;
1174 this.state.userSettingsForm.show_avatars =
1175 UserService.Instance.user.show_avatars;
1176 this.state.userSettingsForm.matrix_user_id = this.state.user.matrix_user_id;
1178 document.title = `/u/${this.state.user.name} - ${this.state.site.name}`;
1179 window.scrollTo(0, 0);
1180 this.setState(this.state);
1182 } else if (res.op == UserOperation.EditComment) {
1183 let data = res.data as CommentResponse;
1184 editCommentRes(data, this.state.comments);
1185 this.setState(this.state);
1186 } else if (res.op == UserOperation.CreateComment) {
1187 let data = res.data as CommentResponse;
1189 UserService.Instance.user &&
1190 data.comment.creator_id == UserService.Instance.user.id
1192 toast(i18n.t('reply_sent'));
1194 } else if (res.op == UserOperation.SaveComment) {
1195 let data = res.data as CommentResponse;
1196 saveCommentRes(data, this.state.comments);
1197 this.setState(this.state);
1198 } else if (res.op == UserOperation.CreateCommentLike) {
1199 let data = res.data as CommentResponse;
1200 createCommentLikeRes(data, this.state.comments);
1201 this.setState(this.state);
1202 } else if (res.op == UserOperation.CreatePostLike) {
1203 let data = res.data as PostResponse;
1204 createPostLikeFindRes(data, this.state.posts);
1205 this.setState(this.state);
1206 } else if (res.op == UserOperation.BanUser) {
1207 let data = res.data as BanUserResponse;
1209 .filter(c => c.creator_id == data.user.id)
1210 .forEach(c => (c.banned = data.banned));
1212 .filter(c => c.creator_id == data.user.id)
1213 .forEach(c => (c.banned = data.banned));
1214 this.setState(this.state);
1215 } else if (res.op == UserOperation.AddAdmin) {
1216 let data = res.data as AddAdminResponse;
1217 this.state.admins = data.admins;
1218 this.setState(this.state);
1219 } else if (res.op == UserOperation.SaveUserSettings) {
1220 let data = res.data as LoginResponse;
1221 this.state.userSettingsLoading = false;
1222 this.setState(this.state);
1223 UserService.Instance.login(data);
1224 } else if (res.op == UserOperation.DeleteAccount) {
1225 this.state.deleteAccountLoading = false;
1226 this.state.deleteAccountShowConfirm = false;
1227 this.setState(this.state);
1228 this.context.router.history.push('/');
1229 } else if (res.op == UserOperation.GetSite) {
1230 let data = res.data as GetSiteResponse;
1231 this.state.site = data.site;
1232 this.setState(this.state);