1 import { Component, linkEvent } from "inferno";
2 import { Link } from "inferno-router";
9 GetPersonDetailsResponse,
14 } from "lemmy-js-client";
15 import moment from "moment";
16 import { Subscription } from "rxjs";
17 import { i18n } from "../../i18next";
18 import { InitialFetchRequest, PersonDetailsView } from "../../interfaces";
19 import { UserService, WebSocketService } from "../../services";
23 createPostLikeFindRes,
30 restoreScrollPosition,
44 import { BannerIconHeader } from "../common/banner-icon-header";
45 import { HtmlTags } from "../common/html-tags";
46 import { Icon, Spinner } from "../common/icon";
47 import { MomentTime } from "../common/moment-time";
48 import { SortSelect } from "../common/sort-select";
49 import { CommunityLink } from "../community/community-link";
50 import { PersonDetails } from "./person-details";
51 import { PersonListing } from "./person-listing";
53 interface ProfileState {
54 personRes: GetPersonDetailsResponse;
56 view: PersonDetailsView;
60 siteRes: GetSiteResponse;
63 interface ProfileProps {
64 view: PersonDetailsView;
67 person_id: number | null;
77 export class Profile extends Component<any, ProfileState> {
78 private isoData = setIsoData(this.context);
79 private subscription: Subscription;
80 private emptyState: ProfileState = {
82 userName: getUsernameFromProps(this.props),
84 view: Profile.getViewFromProps(this.props.match.view),
85 sort: Profile.getSortTypeFromProps(this.props.match.sort),
86 page: Profile.getPageFromProps(this.props.match.page),
87 siteRes: this.isoData.site_res,
90 constructor(props: any, context: any) {
91 super(props, context);
93 this.state = this.emptyState;
94 this.handleSortChange = this.handleSortChange.bind(this);
95 this.handlePageChange = this.handlePageChange.bind(this);
97 this.parseMessage = this.parseMessage.bind(this);
98 this.subscription = wsSubscribe(this.parseMessage);
100 // Only fetch the data if coming from another route
101 if (this.isoData.path == this.context.router.route.match.url) {
102 this.state.personRes = this.isoData.routeData[0];
103 this.state.loading = false;
105 this.fetchUserData();
112 let form: GetPersonDetails = {
113 username: this.state.userName,
114 sort: this.state.sort,
115 saved_only: this.state.view === PersonDetailsView.Saved,
116 page: this.state.page,
118 auth: authField(false),
120 WebSocketService.Instance.send(wsClient.getPersonDetails(form));
123 get isCurrentUser() {
125 UserService.Instance.myUserInfo?.local_user_view.person.id ==
126 this.state.personRes.person_view.person.id
130 static getViewFromProps(view: string): PersonDetailsView {
131 return view ? PersonDetailsView[view] : PersonDetailsView.Overview;
134 static getSortTypeFromProps(sort: string): SortType {
135 return sort ? routeSortTypeToEnum(sort) : SortType.New;
138 static getPageFromProps(page: number): number {
139 return page ? Number(page) : 1;
142 static fetchInitialData(req: InitialFetchRequest): Promise<any>[] {
143 let pathSplit = req.path.split("/");
144 let promises: Promise<any>[] = [];
146 // It can be /u/me, or /username/1
147 let idOrName = pathSplit[2];
148 let person_id: number;
149 let username: string;
150 if (isNaN(Number(idOrName))) {
153 person_id = Number(idOrName);
156 let view = this.getViewFromProps(pathSplit[4]);
157 let sort = this.getSortTypeFromProps(pathSplit[6]);
158 let page = this.getPageFromProps(Number(pathSplit[8]));
160 let form: GetPersonDetails = {
162 saved_only: view === PersonDetailsView.Saved,
166 setOptionalAuth(form, req.auth);
167 this.setIdOrName(form, person_id, username);
168 promises.push(req.client.getPersonDetails(form));
172 static setIdOrName(obj: any, id: number, name_: string) {
176 obj.username = name_;
180 componentWillUnmount() {
181 this.subscription.unsubscribe();
182 saveScrollPosition(this.context);
185 static getDerivedStateFromProps(props: any): ProfileProps {
187 view: this.getViewFromProps(props.match.params.view),
188 sort: this.getSortTypeFromProps(props.match.params.sort),
189 page: this.getPageFromProps(props.match.params.page),
190 person_id: Number(props.match.params.id) || null,
191 username: props.match.params.username,
195 componentDidUpdate(lastProps: any) {
196 // Necessary if you are on a post and you click another post (same route)
198 lastProps.location.pathname.split("/")[2] !==
199 lastProps.history.location.pathname.split("/")[2]
201 // Couldnt get a refresh working. This does for now.
206 get documentTitle(): string {
207 return `@${this.state.personRes.person_view.person.name} - ${this.state.siteRes.site_view.site.name}`;
210 get bioTag(): string {
211 return this.state.personRes.person_view.person.bio
212 ? previewLines(this.state.personRes.person_view.person.bio)
218 <div class="container">
219 {this.state.loading ? (
225 <div class="col-12 col-md-8">
228 title={this.documentTitle}
229 path={this.context.router.route.match.url}
230 description={this.bioTag}
231 image={this.state.personRes.person_view.person.avatar}
236 {!this.state.loading && this.selects()}
238 personRes={this.state.personRes}
239 admins={this.state.siteRes.admins}
240 sort={this.state.sort}
241 page={this.state.page}
244 this.state.siteRes.site_view.site.enable_downvotes
246 enableNsfw={this.state.siteRes.site_view.site.enable_nsfw}
247 view={this.state.view}
248 onPageChange={this.handlePageChange}
252 {!this.state.loading && (
253 <div class="col-12 col-md-4">
255 {this.isCurrentUser && this.follows()}
266 <div class="btn-group btn-group-toggle flex-wrap mb-2">
268 className={`btn btn-outline-secondary pointer
269 ${this.state.view == PersonDetailsView.Overview && "active"}
274 value={PersonDetailsView.Overview}
275 checked={this.state.view === PersonDetailsView.Overview}
276 onChange={linkEvent(this, this.handleViewChange)}
281 className={`btn btn-outline-secondary pointer
282 ${this.state.view == PersonDetailsView.Comments && "active"}
287 value={PersonDetailsView.Comments}
288 checked={this.state.view == PersonDetailsView.Comments}
289 onChange={linkEvent(this, this.handleViewChange)}
294 className={`btn btn-outline-secondary pointer
295 ${this.state.view == PersonDetailsView.Posts && "active"}
300 value={PersonDetailsView.Posts}
301 checked={this.state.view == PersonDetailsView.Posts}
302 onChange={linkEvent(this, this.handleViewChange)}
307 className={`btn btn-outline-secondary pointer
308 ${this.state.view == PersonDetailsView.Saved && "active"}
313 value={PersonDetailsView.Saved}
314 checked={this.state.view == PersonDetailsView.Saved}
315 onChange={linkEvent(this, this.handleViewChange)}
325 <div className="mb-2">
326 <span class="mr-3">{this.viewRadios()}</span>
328 sort={this.state.sort}
329 onChange={this.handleSortChange}
334 href={`/feeds/u/${this.state.userName}.xml?sort=${this.state.sort}`}
338 <Icon icon="rss" classes="text-muted small mx-2" />
345 let pv = this.state.personRes?.person_view;
349 <BannerIconHeader banner={pv.person.banner} icon={pv.person.avatar} />
352 <div class="mb-0 d-flex flex-wrap">
354 {pv.person.display_name && (
355 <h5 class="mb-0">{pv.person.display_name}</h5>
357 <ul class="list-inline mb-2">
358 <li className="list-inline-item">
367 {pv.person.banned && (
368 <li className="list-inline-item badge badge-danger">
374 <div className="flex-grow-1 unselectable pointer mx-2"></div>
375 {!this.isCurrentUser && (
378 className={`d-flex align-self-start btn btn-secondary mr-2 ${
379 !pv.person.matrix_user_id && "invisible"
382 href={`https://matrix.to/#/${pv.person.matrix_user_id}`}
384 {i18n.t("send_secure_message")}
387 className={"d-flex align-self-start btn btn-secondary"}
388 to={`/create_private_message/recipient/${pv.person.id}`}
390 {i18n.t("send_message")}
396 <div className="d-flex align-items-center mb-2">
399 dangerouslySetInnerHTML={mdToHtml(pv.person.bio)}
404 <ul class="list-inline mb-2">
405 <li className="list-inline-item badge badge-light">
406 {i18n.t("number_of_posts", { count: pv.counts.post_count })}
408 <li className="list-inline-item badge badge-light">
409 {i18n.t("number_of_comments", {
410 count: pv.counts.comment_count,
415 <div class="text-muted">
416 {i18n.t("joined")}{" "}
417 <MomentTime data={pv.person} showAgo ignoreUpdated />
419 <div className="d-flex align-items-center text-muted mb-2">
421 <span className="ml-2">
422 {i18n.t("cake_day_title")}{" "}
423 {moment.utc(pv.person.published).local().format("MMM DD, YYYY")}
435 {this.state.personRes.moderates.length > 0 && (
436 <div class="card border-secondary mb-3">
437 <div class="card-body">
438 <h5>{i18n.t("moderates")}</h5>
439 <ul class="list-unstyled mb-0">
440 {this.state.personRes.moderates.map(cmv => (
442 <CommunityLink community={cmv.community} />
454 let follows = UserService.Instance.myUserInfo.follows;
457 {follows.length > 0 && (
458 <div class="card border-secondary mb-3">
459 <div class="card-body">
460 <h5>{i18n.t("subscribed")}</h5>
461 <ul class="list-unstyled mb-0">
462 {follows.map(cfv => (
464 <CommunityLink community={cfv.community} />
475 updateUrl(paramUpdates: UrlParams) {
476 const page = paramUpdates.page || this.state.page;
477 const viewStr = paramUpdates.view || PersonDetailsView[this.state.view];
478 const sortStr = paramUpdates.sort || this.state.sort;
480 let typeView = `/u/${this.state.userName}`;
482 this.props.history.push(
483 `${typeView}/view/${viewStr}/sort/${sortStr}/page/${page}`
485 this.state.loading = true;
486 this.setState(this.state);
487 this.fetchUserData();
490 handlePageChange(page: number) {
491 this.updateUrl({ page });
494 handleSortChange(val: SortType) {
495 this.updateUrl({ sort: val, page: 1 });
498 handleViewChange(i: Profile, event: any) {
500 view: PersonDetailsView[Number(event.target.value)],
505 parseMessage(msg: any) {
506 let op = wsUserOp(msg);
509 toast(i18n.t(msg.error), "danger");
510 if (msg.error == "couldnt_find_that_username_or_email") {
511 this.context.router.history.push("/");
514 } else if (msg.reconnect) {
515 this.fetchUserData();
516 } else if (op == UserOperation.GetPersonDetails) {
517 // Since the PersonDetails contains posts/comments as well as some general user info we listen here as well
518 // and set the parent state if it is not set or differs
519 // TODO this might need to get abstracted
520 let data = wsJsonToRes<GetPersonDetailsResponse>(msg).data;
521 this.state.personRes = data;
523 this.state.loading = false;
524 this.setState(this.state);
525 restoreScrollPosition(this.context);
526 } else if (op == UserOperation.AddAdmin) {
527 let data = wsJsonToRes<AddAdminResponse>(msg).data;
528 this.state.siteRes.admins = data.admins;
529 this.setState(this.state);
530 } else if (op == UserOperation.CreateCommentLike) {
531 let data = wsJsonToRes<CommentResponse>(msg).data;
532 createCommentLikeRes(data.comment_view, this.state.personRes.comments);
533 this.setState(this.state);
535 op == UserOperation.EditComment ||
536 op == UserOperation.DeleteComment ||
537 op == UserOperation.RemoveComment
539 let data = wsJsonToRes<CommentResponse>(msg).data;
540 editCommentRes(data.comment_view, this.state.personRes.comments);
541 this.setState(this.state);
542 } else if (op == UserOperation.CreateComment) {
543 let data = wsJsonToRes<CommentResponse>(msg).data;
545 UserService.Instance.myUserInfo &&
546 data.comment_view.creator.id ==
547 UserService.Instance.myUserInfo.local_user_view.person.id
549 toast(i18n.t("reply_sent"));
551 } else if (op == UserOperation.SaveComment) {
552 let data = wsJsonToRes<CommentResponse>(msg).data;
553 saveCommentRes(data.comment_view, this.state.personRes.comments);
554 this.setState(this.state);
556 op == UserOperation.EditPost ||
557 op == UserOperation.DeletePost ||
558 op == UserOperation.RemovePost ||
559 op == UserOperation.LockPost ||
560 op == UserOperation.StickyPost ||
561 op == UserOperation.SavePost
563 let data = wsJsonToRes<PostResponse>(msg).data;
564 editPostFindRes(data.post_view, this.state.personRes.posts);
565 this.setState(this.state);
566 } else if (op == UserOperation.CreatePostLike) {
567 let data = wsJsonToRes<PostResponse>(msg).data;
568 createPostLikeFindRes(data.post_view, this.state.personRes.posts);
569 this.setState(this.state);
570 } else if (op == UserOperation.BanPerson) {
571 let data = wsJsonToRes<BanPersonResponse>(msg).data;
572 this.state.personRes.comments
573 .filter(c => c.creator.id == data.person_view.person.id)
574 .forEach(c => (c.creator.banned = data.banned));
575 this.state.personRes.posts
576 .filter(c => c.creator.id == data.person_view.person.id)
577 .forEach(c => (c.creator.banned = data.banned));
578 this.setState(this.state);
579 } else if (op == UserOperation.BlockPerson) {
580 let data = wsJsonToRes<BlockPersonResponse>(msg).data;
581 updatePersonBlock(data);