1 import { Component, linkEvent } from 'inferno';
2 import { Helmet } from 'inferno-helmet';
3 import { Link } from 'inferno-router';
4 import { Subscription } from 'rxjs';
5 import { retryWhen, delay, take } from 'rxjs/operators';
9 GetFollowedCommunitiesResponse,
11 ListCommunitiesResponse,
27 WebSocketJsonResponse,
28 } from 'lemmy-js-client';
29 import { DataType } from '../interfaces';
30 import { WebSocketService, UserService } from '../services';
31 import { PostListings } from './post-listings';
32 import { CommentNodes } from './comment-nodes';
33 import { SortSelect } from './sort-select';
34 import { ListingTypeSelect } from './listing-type-select';
35 import { DataTypeSelect } from './data-type-select';
36 import { SiteForm } from './site-form';
37 import { UserListing } from './user-listing';
38 import { CommunityLink } from './community-link';
39 import { BannerIconHeader } from './banner-icon-header';
46 getListingTypeFromProps,
53 createPostLikeFindRes,
60 import { i18n } from '../i18next';
61 import { T } from 'inferno-i18next';
64 subscribedCommunities: Array<CommunityUser>;
65 trendingCommunities: Array<Community>;
66 siteRes: GetSiteResponse;
67 showEditSite: boolean;
70 comments: Array<Comment>;
71 listingType: ListingType;
78 listingType: ListingType;
85 listingType?: ListingType;
91 export class Main extends Component<any, MainState> {
92 private subscription: Subscription;
93 private emptyState: MainState = {
94 subscribedCommunities: [],
95 trendingCommunities: [],
103 number_of_users: null,
104 number_of_posts: null,
105 number_of_comments: null,
106 number_of_communities: null,
107 enable_downvotes: null,
108 open_registration: null,
112 creator_preferred_username: null,
118 federated_instances: null,
124 listingType: getListingTypeFromProps(this.props),
125 dataType: getDataTypeFromProps(this.props),
126 sort: getSortTypeFromProps(this.props),
127 page: getPageFromProps(this.props),
130 constructor(props: any, context: any) {
131 super(props, context);
133 this.state = this.emptyState;
134 this.handleEditCancel = this.handleEditCancel.bind(this);
135 this.handleSortChange = this.handleSortChange.bind(this);
136 this.handleListingTypeChange = this.handleListingTypeChange.bind(this);
137 this.handleDataTypeChange = this.handleDataTypeChange.bind(this);
139 this.subscription = WebSocketService.Instance.subject
140 .pipe(retryWhen(errors => errors.pipe(delay(3000), take(10))))
142 msg => this.parseMessage(msg),
143 err => console.error(err),
144 () => console.log('complete')
147 WebSocketService.Instance.getSite();
149 if (UserService.Instance.user) {
150 WebSocketService.Instance.getFollowedCommunities();
153 let listCommunitiesForm: ListCommunitiesForm = {
158 WebSocketService.Instance.listCommunities(listCommunitiesForm);
163 componentWillUnmount() {
164 this.subscription.unsubscribe();
167 static getDerivedStateFromProps(props: any): MainProps {
169 listingType: getListingTypeFromProps(props),
170 dataType: getDataTypeFromProps(props),
171 sort: getSortTypeFromProps(props),
172 page: getPageFromProps(props),
176 componentDidUpdate(_: any, lastState: MainState) {
178 lastState.listingType !== this.state.listingType ||
179 lastState.dataType !== this.state.dataType ||
180 lastState.sort !== this.state.sort ||
181 lastState.page !== this.state.page
183 this.setState({ loading: true });
188 get documentTitle(): string {
189 if (this.state.siteRes.site.name) {
190 return `${this.state.siteRes.site.name}`;
196 get favIcon(): string {
197 return this.state.siteRes.site.icon
198 ? this.state.siteRes.site.icon
204 <div class="container">
205 <Helmet title={this.documentTitle}>
214 <main role="main" class="col-12 col-md-8">
217 <aside class="col-12 col-md-4">{this.mySidebar()}</aside>
226 {!this.state.loading && (
228 <div class="card bg-transparent border-secondary mb-3">
229 <div class="card-header bg-transparent border-secondary">
232 {this.adminButtons()}
234 <BannerIconHeader banner={this.state.siteRes.site.banner} />
236 <div class="card-body">
237 {this.trendingCommunities()}
238 {this.createCommunityButton()}
240 {this.subscribedCommunities()}
245 <div class="card bg-transparent border-secondary mb-3">
246 <div class="card-body">{this.sidebar()}</div>
249 <div class="card bg-transparent border-secondary">
250 <div class="card-body">{this.landing()}</div>
258 createCommunityButton() {
260 <Link class="btn btn-secondary btn-block" to="/create_community">
261 {i18n.t('create_a_community')}
266 trendingCommunities() {
270 <T i18nKey="trending_communities">
272 <Link class="text-body" to="/communities">
277 <ul class="list-inline">
278 {this.state.trendingCommunities.map(community => (
279 <li class="list-inline-item">
280 <CommunityLink community={community} />
288 subscribedCommunities() {
290 UserService.Instance.user &&
291 this.state.subscribedCommunities.length > 0 && (
294 <T i18nKey="subscribed_to_communities">
296 <Link class="text-body" to="/communities">
301 <ul class="list-inline">
302 {this.state.subscribedCommunities.map(community => (
303 <li class="list-inline-item">
306 name: community.community_name,
307 id: community.community_id,
308 local: community.community_local,
309 actor_id: community.community_actor_id,
310 icon: community.community_icon,
324 {!this.state.showEditSite ? (
328 site={this.state.siteRes.site}
329 onCancel={this.handleEditCancel}
336 updateUrl(paramUpdates: UrlParams) {
337 const listingTypeStr = paramUpdates.listingType || this.state.listingType;
338 const dataTypeStr = paramUpdates.dataType || DataType[this.state.dataType];
339 const sortStr = paramUpdates.sort || this.state.sort;
340 const page = paramUpdates.page || this.state.page;
341 this.props.history.push(
342 `/home/data_type/${dataTypeStr}/listing_type/${listingTypeStr}/sort/${sortStr}/page/${page}`
349 {this.state.siteRes.site.description && this.siteDescription()}
357 return <h5 class="mb-0">{`${this.state.siteRes.site.name}`}</h5>;
362 <ul class="mt-1 list-inline small mb-0">
363 <li class="list-inline-item">{i18n.t('admins')}:</li>
364 {this.state.siteRes.admins.map(admin => (
365 <li class="list-inline-item">
369 preferred_username: admin.preferred_username,
370 avatar: admin.avatar,
372 actor_id: admin.actor_id,
384 <ul class="my-2 list-inline">
385 <li className="list-inline-item badge badge-light">
386 {i18n.t('number_online', { count: this.state.siteRes.online })}
388 <li className="list-inline-item badge badge-light">
389 {i18n.t('number_of_users', {
390 count: this.state.siteRes.site.number_of_users,
393 <li className="list-inline-item badge badge-light">
394 {i18n.t('number_of_communities', {
395 count: this.state.siteRes.site.number_of_communities,
398 <li className="list-inline-item badge badge-light">
399 {i18n.t('number_of_posts', {
400 count: this.state.siteRes.site.number_of_posts,
403 <li className="list-inline-item badge badge-light">
404 {i18n.t('number_of_comments', {
405 count: this.state.siteRes.site.number_of_comments,
408 <li className="list-inline-item">
409 <Link className="badge badge-light" to="/modlog">
420 <ul class="list-inline mb-1 text-muted font-weight-bold">
421 <li className="list-inline-item-action">
424 onClick={linkEvent(this, this.handleEditClick)}
425 data-tippy-content={i18n.t('edit')}
427 <svg class="icon icon-inline">
428 <use xlinkHref="#icon-edit"></use>
441 dangerouslySetInnerHTML={mdToHtml(this.state.siteRes.site.description)}
450 {i18n.t('powered_by')}
451 <svg class="icon mx-2">
452 <use xlinkHref="#icon-mouse">#</use>
459 <T i18nKey="landing_0">
461 <a href="https://en.wikipedia.org/wiki/Social_network_aggregation">
464 <a href="https://en.wikipedia.org/wiki/Fediverse">#</a>
465 <br class="big"></br>
469 <br class="big"></br>
470 <a href={repoUrl}>#</a>
471 <br class="big"></br>
472 <a href="https://www.rust-lang.org">#</a>
473 <a href="https://actix.rs/">#</a>
474 <a href="https://infernojs.org">#</a>
475 <a href="https://www.typescriptlang.org/">#</a>
476 <br class="big"></br>
477 <a href="https://github.com/LemmyNet/lemmy/graphs/contributors?type=a">
488 <div class="main-content-wrapper">
489 {this.state.loading ? (
491 <svg class="icon icon-spinner spin">
492 <use xlinkHref="#icon-spinner"></use>
507 return this.state.dataType == DataType.Post ? (
509 posts={this.state.posts}
512 sort={this.state.sort}
513 enableDownvotes={this.state.siteRes.site.enable_downvotes}
514 enableNsfw={this.state.siteRes.site.enable_nsfw}
518 nodes={commentsToFlatNodes(this.state.comments)}
521 sortType={this.state.sort}
523 enableDownvotes={this.state.siteRes.site.enable_downvotes}
530 <div className="mb-3">
533 type_={this.state.dataType}
534 onChange={this.handleDataTypeChange}
539 type_={this.state.listingType}
541 this.state.siteRes.federated_instances &&
542 this.state.siteRes.federated_instances.length > 0
544 onChange={this.handleListingTypeChange}
548 <SortSelect sort={this.state.sort} onChange={this.handleSortChange} />
550 {this.state.listingType == ListingType.All && (
552 href={`/feeds/all.xml?sort=${this.state.sort}`}
557 <svg class="icon text-muted small">
558 <use xlinkHref="#icon-rss">#</use>
562 {UserService.Instance.user &&
563 this.state.listingType == ListingType.Subscribed && (
565 href={`/feeds/front/${UserService.Instance.auth}.xml?sort=${this.state.sort}`}
570 <svg class="icon text-muted small">
571 <use xlinkHref="#icon-rss">#</use>
582 {this.state.page > 1 && (
584 class="btn btn-secondary mr-1"
585 onClick={linkEvent(this, this.prevPage)}
590 {this.state.posts.length > 0 && (
592 class="btn btn-secondary"
593 onClick={linkEvent(this, this.nextPage)}
602 get canAdmin(): boolean {
604 UserService.Instance.user &&
605 this.state.siteRes.admins
607 .includes(UserService.Instance.user.id)
611 handleEditClick(i: Main) {
612 i.state.showEditSite = true;
617 this.state.showEditSite = false;
618 this.setState(this.state);
622 i.updateUrl({ page: i.state.page + 1 });
623 window.scrollTo(0, 0);
627 i.updateUrl({ page: i.state.page - 1 });
628 window.scrollTo(0, 0);
631 handleSortChange(val: SortType) {
632 this.updateUrl({ sort: val, page: 1 });
633 window.scrollTo(0, 0);
636 handleListingTypeChange(val: ListingType) {
637 this.updateUrl({ listingType: val, page: 1 });
638 window.scrollTo(0, 0);
641 handleDataTypeChange(val: DataType) {
642 this.updateUrl({ dataType: DataType[val], page: 1 });
643 window.scrollTo(0, 0);
647 if (this.state.dataType == DataType.Post) {
648 let getPostsForm: GetPostsForm = {
649 page: this.state.page,
651 sort: this.state.sort,
652 type_: this.state.listingType,
654 WebSocketService.Instance.getPosts(getPostsForm);
656 let getCommentsForm: GetCommentsForm = {
657 page: this.state.page,
659 sort: this.state.sort,
660 type_: this.state.listingType,
662 WebSocketService.Instance.getComments(getCommentsForm);
666 parseMessage(msg: WebSocketJsonResponse) {
668 let res = wsJsonToRes(msg);
670 toast(i18n.t(msg.error), 'danger');
672 } else if (msg.reconnect) {
674 } else if (res.op == UserOperation.GetFollowedCommunities) {
675 let data = res.data as GetFollowedCommunitiesResponse;
676 this.state.subscribedCommunities = data.communities;
677 this.setState(this.state);
678 } else if (res.op == UserOperation.ListCommunities) {
679 let data = res.data as ListCommunitiesResponse;
680 this.state.trendingCommunities = data.communities;
681 this.setState(this.state);
682 } else if (res.op == UserOperation.GetSite) {
683 let data = res.data as GetSiteResponse;
685 // This means it hasn't been set up yet
687 this.context.router.history.push('/setup');
689 this.state.siteRes.admins = data.admins;
690 this.state.siteRes.site = data.site;
691 this.state.siteRes.banned = data.banned;
692 this.state.siteRes.online = data.online;
693 this.setState(this.state);
694 } else if (res.op == UserOperation.EditSite) {
695 let data = res.data as SiteResponse;
696 this.state.siteRes.site = data.site;
697 this.state.showEditSite = false;
698 this.setState(this.state);
699 toast(i18n.t('site_saved'));
700 } else if (res.op == UserOperation.GetPosts) {
701 let data = res.data as GetPostsResponse;
702 this.state.posts = data.posts;
703 this.state.loading = false;
704 this.setState(this.state);
706 } else if (res.op == UserOperation.CreatePost) {
707 let data = res.data as PostResponse;
709 // If you're on subscribed, only push it if you're subscribed.
710 if (this.state.listingType == ListingType.Subscribed) {
712 this.state.subscribedCommunities
713 .map(c => c.community_id)
714 .includes(data.post.community_id)
716 this.state.posts.unshift(data.post);
717 notifyPost(data.post, this.context.router);
721 let nsfw = data.post.nsfw || data.post.community_nsfw;
723 // Don't push the post if its nsfw, and don't have that setting on
727 UserService.Instance.user &&
728 UserService.Instance.user.show_nsfw)
730 this.state.posts.unshift(data.post);
731 notifyPost(data.post, this.context.router);
734 this.setState(this.state);
735 } else if (res.op == UserOperation.EditPost) {
736 let data = res.data as PostResponse;
737 editPostFindRes(data, this.state.posts);
738 this.setState(this.state);
739 } else if (res.op == UserOperation.CreatePostLike) {
740 let data = res.data as PostResponse;
741 createPostLikeFindRes(data, this.state.posts);
742 this.setState(this.state);
743 } else if (res.op == UserOperation.AddAdmin) {
744 let data = res.data as AddAdminResponse;
745 this.state.siteRes.admins = data.admins;
746 this.setState(this.state);
747 } else if (res.op == UserOperation.BanUser) {
748 let data = res.data as BanUserResponse;
749 let found = this.state.siteRes.banned.find(u => (u.id = data.user.id));
751 // Remove the banned if its found in the list, and the action is an unban
752 if (found && !data.banned) {
753 this.state.siteRes.banned = this.state.siteRes.banned.filter(
754 i => i.id !== data.user.id
757 this.state.siteRes.banned.push(data.user);
761 .filter(p => p.creator_id == data.user.id)
762 .forEach(p => (p.banned = data.banned));
764 this.setState(this.state);
765 } else if (res.op == UserOperation.GetComments) {
766 let data = res.data as GetCommentsResponse;
767 this.state.comments = data.comments;
768 this.state.loading = false;
769 this.setState(this.state);
771 res.op == UserOperation.EditComment ||
772 res.op == UserOperation.DeleteComment ||
773 res.op == UserOperation.RemoveComment
775 let data = res.data as CommentResponse;
776 editCommentRes(data, this.state.comments);
777 this.setState(this.state);
778 } else if (res.op == UserOperation.CreateComment) {
779 let data = res.data as CommentResponse;
781 // Necessary since it might be a user reply
782 if (data.recipient_ids.length == 0) {
783 // If you're on subscribed, only push it if you're subscribed.
784 if (this.state.listingType == ListingType.Subscribed) {
786 this.state.subscribedCommunities
787 .map(c => c.community_id)
788 .includes(data.comment.community_id)
790 this.state.comments.unshift(data.comment);
793 this.state.comments.unshift(data.comment);
795 this.setState(this.state);
797 } else if (res.op == UserOperation.SaveComment) {
798 let data = res.data as CommentResponse;
799 saveCommentRes(data, this.state.comments);
800 this.setState(this.state);
801 } else if (res.op == UserOperation.CreateCommentLike) {
802 let data = res.data as CommentResponse;
803 createCommentLikeRes(data, this.state.comments);
804 this.setState(this.state);