+import { myAuthRequired } from "@utils/app";
+import { getUnixTime, hostname } from "@utils/helpers";
+import { amAdmin, amMod, amTopMod } from "@utils/roles";
import { Component, InfernoNode, linkEvent } from "inferno";
import { T } from "inferno-i18next-dess";
import { Link } from "inferno-router";
PurgeCommunity,
RemoveCommunity,
} from "lemmy-js-client";
-import { i18n } from "../../i18next";
-import { UserService } from "../../services";
-import {
- amAdmin,
- amMod,
- amTopMod,
- getUnixTime,
- hostname,
- mdToHtml,
- myAuthRequired,
- numToSI,
-} from "../../utils";
+import { mdToHtml } from "../../markdown";
+import { I18NextService, UserService } from "../../services";
+import { Badges } from "../common/badges";
import { BannerIconHeader } from "../common/banner-icon-header";
import { Icon, PurgeWarning, Spinner } from "../common/icon";
import { CommunityForm } from "../community/community-form";
allLanguages: Language[];
siteLanguages: number[];
communityLanguages?: number[];
- online: number;
enableNsfw?: boolean;
showIcon?: boolean;
editable?: boolean;
render() {
return (
- <div>
+ <div className="community-sidebar">
{!this.state.showEdit ? (
this.sidebar()
) : (
const myUSerInfo = UserService.Instance.myUserInfo;
const { name, actor_id } = this.props.community_view.community;
return (
- <div>
- <div className="card border-secondary mb-3">
- <div className="card-body">
- {this.communityTitle()}
- {this.props.editable && this.adminButtons()}
- {myUSerInfo && this.subscribe()}
- {this.canPost && this.createPost()}
- {myUSerInfo && this.blockCommunity()}
- {!myUSerInfo && (
- <div className="alert alert-info" role="alert">
- <T
- i18nKey="community_not_logged_in_alert"
- interpolation={{
- community: name,
- instance: hostname(actor_id),
- }}
- >
- #<code className="user-select-all">#</code>#
- </T>
- </div>
- )}
- </div>
- </div>
- <div className="card border-secondary mb-3">
- <div className="card-body">
- {this.description()}
- {this.badges()}
- {this.mods()}
- </div>
+ <aside className="mb-3">
+ <div id="sidebarContainer">
+ <section id="sidebarMain" className="card border-secondary mb-3">
+ <div className="card-body">
+ {this.communityTitle()}
+ {this.props.editable && this.adminButtons()}
+ {myUSerInfo && this.subscribe()}
+ {this.canPost && this.createPost()}
+ {myUSerInfo && this.blockCommunity()}
+ {!myUSerInfo && (
+ <div className="alert alert-info" role="alert">
+ <T
+ i18nKey="community_not_logged_in_alert"
+ interpolation={{
+ community: name,
+ instance: hostname(actor_id),
+ }}
+ >
+ #<code className="user-select-all">#</code>#
+ </T>
+ </div>
+ )}
+ </div>
+ </section>
+ <section id="sidebarInfo" className="card border-secondary mb-3">
+ <div className="card-body">
+ {this.description()}
+ <Badges
+ communityId={this.props.community_view.community.id}
+ counts={this.props.community_view.counts}
+ />
+ {this.mods()}
+ </div>
+ </section>
</div>
- </div>
+ </aside>
);
}
communityTitle() {
const community = this.props.community_view.community;
- const subscribed = this.props.community_view.subscribed;
+
return (
<div>
- <h5 className="mb-0">
+ <h2 className="h5 mb-0">
{this.props.showIcon && !community.removed && (
<BannerIconHeader icon={community.icon} banner={community.banner} />
)}
- <span className="mr-2">
+ <span className="me-2">
<CommunityLink community={community} hideAvatar />
</span>
- {subscribed === "Subscribed" && (
- <button
- className="btn btn-secondary btn-sm mr-2"
- onClick={linkEvent(this, this.handleUnfollowCommunity)}
- >
- {this.state.followCommunityLoading ? (
- <Spinner />
- ) : (
- <>
- <Icon icon="check" classes="icon-inline text-success mr-1" />
- {i18n.t("joined")}
- </>
- )}
- </button>
- )}
- {subscribed === "Pending" && (
- <button
- className="btn btn-warning mr-2"
- onClick={linkEvent(this, this.handleUnfollowCommunity)}
- >
- {this.state.followCommunityLoading ? (
- <Spinner />
- ) : (
- i18n.t("subscribe_pending")
- )}
- </button>
- )}
{community.removed && (
- <small className="mr-2 text-muted font-italic">
- {i18n.t("removed")}
+ <small className="me-2 text-muted fst-italic">
+ {I18NextService.i18n.t("removed")}
</small>
)}
{community.deleted && (
- <small className="mr-2 text-muted font-italic">
- {i18n.t("deleted")}
+ <small className="me-2 text-muted fst-italic">
+ {I18NextService.i18n.t("deleted")}
</small>
)}
{community.nsfw && (
- <small className="mr-2 text-muted font-italic">
- {i18n.t("nsfw")}
+ <small className="me-2 text-muted fst-italic">
+ {I18NextService.i18n.t("nsfw")}
</small>
)}
- </h5>
+ </h2>
<CommunityLink
community={community}
realLink
);
}
- badges() {
- const community_view = this.props.community_view;
- const counts = community_view.counts;
- return (
- <ul className="my-1 list-inline">
- <li className="list-inline-item badge badge-secondary">
- {i18n.t("number_online", {
- count: this.props.online,
- formattedCount: numToSI(this.props.online),
- })}
- </li>
- <li
- className="list-inline-item badge badge-secondary pointer"
- data-tippy-content={i18n.t("active_users_in_the_last_day", {
- count: Number(counts.users_active_day),
- formattedCount: numToSI(counts.users_active_day),
- })}
- >
- {i18n.t("number_of_users", {
- count: Number(counts.users_active_day),
- formattedCount: numToSI(counts.users_active_day),
- })}{" "}
- / {i18n.t("day")}
- </li>
- <li
- className="list-inline-item badge badge-secondary pointer"
- data-tippy-content={i18n.t("active_users_in_the_last_week", {
- count: Number(counts.users_active_week),
- formattedCount: numToSI(counts.users_active_week),
- })}
- >
- {i18n.t("number_of_users", {
- count: Number(counts.users_active_week),
- formattedCount: numToSI(counts.users_active_week),
- })}{" "}
- / {i18n.t("week")}
- </li>
- <li
- className="list-inline-item badge badge-secondary pointer"
- data-tippy-content={i18n.t("active_users_in_the_last_month", {
- count: Number(counts.users_active_month),
- formattedCount: numToSI(counts.users_active_month),
- })}
- >
- {i18n.t("number_of_users", {
- count: Number(counts.users_active_month),
- formattedCount: numToSI(counts.users_active_month),
- })}{" "}
- / {i18n.t("month")}
- </li>
- <li
- className="list-inline-item badge badge-secondary pointer"
- data-tippy-content={i18n.t("active_users_in_the_last_six_months", {
- count: Number(counts.users_active_half_year),
- formattedCount: numToSI(counts.users_active_half_year),
- })}
- >
- {i18n.t("number_of_users", {
- count: Number(counts.users_active_half_year),
- formattedCount: numToSI(counts.users_active_half_year),
- })}{" "}
- / {i18n.t("number_of_months", { count: 6, formattedCount: 6 })}
- </li>
- <li className="list-inline-item badge badge-secondary">
- {i18n.t("number_of_subscribers", {
- count: Number(counts.subscribers),
- formattedCount: numToSI(counts.subscribers),
- })}
- </li>
- <li className="list-inline-item badge badge-secondary">
- {i18n.t("number_of_posts", {
- count: Number(counts.posts),
- formattedCount: numToSI(counts.posts),
- })}
- </li>
- <li className="list-inline-item badge badge-secondary">
- {i18n.t("number_of_comments", {
- count: Number(counts.comments),
- formattedCount: numToSI(counts.comments),
- })}
- </li>
- <li className="list-inline-item">
- <Link
- className="badge badge-primary"
- to={`/modlog/${this.props.community_view.community.id}`}
- >
- {i18n.t("modlog")}
- </Link>
- </li>
- </ul>
- );
- }
-
mods() {
return (
<ul className="list-inline small">
- <li className="list-inline-item">{i18n.t("mods")}: </li>
+ <li className="list-inline-item">{I18NextService.i18n.t("mods")}: </li>
{this.props.moderators.map(mod => (
<li key={mod.moderator.id} className="list-inline-item">
<PersonListing person={mod.moderator} />
const cv = this.props.community_view;
return (
<Link
- className={`btn btn-secondary btn-block mb-2 ${
+ className={`btn btn-secondary d-block mb-2 w-100 ${
cv.community.deleted || cv.community.removed ? "no-click" : ""
}`}
to={`/create_post?communityId=${cv.community.id}`}
>
- {i18n.t("create_a_post")}
+ {I18NextService.i18n.t("create_a_post")}
</Link>
);
}
subscribe() {
const community_view = this.props.community_view;
- return (
- <div className="mb-2">
- {community_view.subscribed == "NotSubscribed" && (
- <button
- className="btn btn-secondary btn-block"
- onClick={linkEvent(this, this.handleFollowCommunity)}
- >
- {this.state.followCommunityLoading ? (
- <Spinner />
- ) : (
- i18n.t("subscribe")
- )}
- </button>
- )}
- </div>
- );
+
+ if (community_view.subscribed === "NotSubscribed") {
+ return (
+ <button
+ className="btn btn-secondary d-block mb-2 w-100"
+ onClick={linkEvent(this, this.handleFollowCommunity)}
+ >
+ {this.state.followCommunityLoading ? (
+ <Spinner />
+ ) : (
+ I18NextService.i18n.t("subscribe")
+ )}
+ </button>
+ );
+ }
+
+ if (community_view.subscribed === "Subscribed") {
+ return (
+ <button
+ className="btn btn-secondary d-block mb-2 w-100"
+ onClick={linkEvent(this, this.handleUnfollowCommunity)}
+ >
+ {this.state.followCommunityLoading ? (
+ <Spinner />
+ ) : (
+ <>
+ <Icon icon="check" classes="icon-inline me-1" />
+ {I18NextService.i18n.t("joined")}
+ </>
+ )}
+ </button>
+ );
+ }
+
+ if (community_view.subscribed === "Pending") {
+ return (
+ <button
+ className="btn btn-warning d-block mb-2 w-100"
+ onClick={linkEvent(this, this.handleUnfollowCommunity)}
+ >
+ {this.state.followCommunityLoading ? (
+ <Spinner />
+ ) : (
+ I18NextService.i18n.t("subscribe_pending")
+ )}
+ </button>
+ );
+ }
}
blockCommunity() {
const { subscribed, blocked } = this.props.community_view;
return (
- <div className="mb-2">
- {subscribed == "NotSubscribed" && (
- <button
- className="btn btn-danger btn-block"
- onClick={linkEvent(this, this.handleBlockCommunity)}
- >
- {i18n.t(blocked ? "unblock_community" : "block_community")}
- </button>
- )}
- </div>
+ subscribed === "NotSubscribed" && (
+ <button
+ className="btn btn-danger d-block mb-2 w-100"
+ onClick={linkEvent(this, this.handleBlockCommunity)}
+ >
+ {I18NextService.i18n.t(
+ blocked ? "unblock_community" : "block_community"
+ )}
+ </button>
+ )
);
}
const community_view = this.props.community_view;
return (
<>
- <ul className="list-inline mb-1 text-muted font-weight-bold">
+ <ul className="list-inline mb-1 text-muted fw-bold">
{amMod(this.props.moderators) && (
<>
<li className="list-inline-item-action">
<button
className="btn btn-link text-muted d-inline-block"
onClick={linkEvent(this, this.handleEditClick)}
- data-tippy-content={i18n.t("edit")}
- aria-label={i18n.t("edit")}
+ data-tippy-content={I18NextService.i18n.t("edit")}
+ aria-label={I18NextService.i18n.t("edit")}
>
<Icon icon="edit" classes="icon-inline" />
</button>
this.handleShowConfirmLeaveModTeamClick
)}
>
- {i18n.t("leave_mod_team")}
+ {I18NextService.i18n.t("leave_mod_team")}
</button>
</li>
) : (
<>
<li className="list-inline-item-action">
- {i18n.t("are_you_sure")}
+ {I18NextService.i18n.t("are_you_sure")}
</li>
<li className="list-inline-item-action">
<button
className="btn btn-link text-muted d-inline-block"
onClick={linkEvent(this, this.handleLeaveModTeam)}
>
- {i18n.t("yes")}
+ {I18NextService.i18n.t("yes")}
</button>
</li>
<li className="list-inline-item-action">
this.handleCancelLeaveModTeamClick
)}
>
- {i18n.t("no")}
+ {I18NextService.i18n.t("no")}
</button>
</li>
</>
onClick={linkEvent(this, this.handleDeleteCommunity)}
data-tippy-content={
!community_view.community.deleted
- ? i18n.t("delete")
- : i18n.t("restore")
+ ? I18NextService.i18n.t("delete")
+ : I18NextService.i18n.t("restore")
}
aria-label={
!community_view.community.deleted
- ? i18n.t("delete")
- : i18n.t("restore")
+ ? I18NextService.i18n.t("delete")
+ : I18NextService.i18n.t("restore")
}
>
{this.state.deleteCommunityLoading ? (
className="btn btn-link text-muted d-inline-block"
onClick={linkEvent(this, this.handleModRemoveShow)}
>
- {i18n.t("remove")}
+ {I18NextService.i18n.t("remove")}
</button>
) : (
<button
{this.state.removeCommunityLoading ? (
<Spinner />
) : (
- i18n.t("restore")
+ I18NextService.i18n.t("restore")
)}
</button>
)}
<button
className="btn btn-link text-muted d-inline-block"
onClick={linkEvent(this, this.handlePurgeCommunityShow)}
- aria-label={i18n.t("purge_community")}
+ aria-label={I18NextService.i18n.t("purge_community")}
>
- {i18n.t("purge_community")}
+ {I18NextService.i18n.t("purge_community")}
</button>
</li>
)}
</ul>
{this.state.showRemoveDialog && (
<form onSubmit={linkEvent(this, this.handleRemoveCommunity)}>
- <div className="form-group">
+ <div className="input-group mb-3">
<label className="col-form-label" htmlFor="remove-reason">
- {i18n.t("reason")}
+ {I18NextService.i18n.t("reason")}
</label>
<input
type="text"
id="remove-reason"
- className="form-control mr-2"
- placeholder={i18n.t("optional")}
+ className="form-control me-2"
+ placeholder={I18NextService.i18n.t("optional")}
value={this.state.removeReason}
onInput={linkEvent(this, this.handleModRemoveReasonChange)}
/>
</div>
{/* TODO hold off on expires for now */}
- {/* <div class="form-group row"> */}
+ {/* <div class="mb-3 row"> */}
{/* <label class="col-form-label">Expires</label> */}
- {/* <input type="date" class="form-control mr-2" placeholder={i18n.t('expires')} value={this.state.removeExpires} onInput={linkEvent(this, this.handleModRemoveExpiresChange)} /> */}
+ {/* <input type="date" class="form-control me-2" placeholder={I18NextService.i18n.t('expires')} value={this.state.removeExpires} onInput={linkEvent(this, this.handleModRemoveExpiresChange)} /> */}
{/* </div> */}
- <div className="form-group">
+ <div className="input-group mb-3">
<button type="submit" className="btn btn-secondary">
{this.state.removeCommunityLoading ? (
<Spinner />
) : (
- i18n.t("remove_community")
+ I18NextService.i18n.t("remove_community")
)}
</button>
</div>
)}
{this.state.showPurgeDialog && (
<form onSubmit={linkEvent(this, this.handlePurgeCommunity)}>
- <div className="form-group">
+ <div className="input-group mb-3">
<PurgeWarning />
</div>
- <div className="form-group">
- <label className="sr-only" htmlFor="purge-reason">
- {i18n.t("reason")}
+ <div className="input-group mb-3">
+ <label className="visually-hidden" htmlFor="purge-reason">
+ {I18NextService.i18n.t("reason")}
</label>
<input
type="text"
id="purge-reason"
- className="form-control mr-2"
- placeholder={i18n.t("reason")}
+ className="form-control me-2"
+ placeholder={I18NextService.i18n.t("reason")}
value={this.state.purgeReason}
onInput={linkEvent(this, this.handlePurgeReasonChange)}
/>
</div>
- <div className="form-group">
+ <div className="input-group mb-3">
{this.state.purgeCommunityLoading ? (
<Spinner />
) : (
<button
type="submit"
className="btn btn-secondary"
- aria-label={i18n.t("purge_community")}
+ aria-label={I18NextService.i18n.t("purge_community")}
>
- {i18n.t("purge_community")}
+ {I18NextService.i18n.t("purge_community")}
</button>
)}
</div>