13 } from "lemmy-js-client";
14 import { i18n } from "../../i18next";
16 capitalizeFirstLetter,
20 import { Icon, Spinner } from "../common/icon";
21 import { ImageUploadForm } from "../common/image-upload-form";
22 import { LanguageSelect } from "../common/language-select";
23 import { ListingTypeSelect } from "../common/listing-type-select";
24 import { MarkdownTextArea } from "../common/markdown-textarea";
25 import NavigationPrompt from "../common/navigation-prompt";
27 interface SiteFormProps {
28 blockedInstances?: Instance[];
29 allowedInstances?: Instance[];
32 onSaveSite(form: EditSite): void;
33 siteRes: GetSiteResponse;
37 interface SiteFormState {
40 allowed_instances: string;
41 blocked_instances: string;
46 type InstanceKey = "allowed_instances" | "blocked_instances";
48 export class SiteForm extends Component<SiteFormProps, SiteFormState> {
49 state: SiteFormState = {
50 siteForm: this.initSiteForm(),
52 allowed_instances: "",
53 blocked_instances: "",
58 initSiteForm(): EditSite {
59 const site = this.props.siteRes.site_view.site;
60 const ls = this.props.siteRes.site_view.local_site;
63 sidebar: site.sidebar,
64 description: site.description,
65 enable_downvotes: ls.enable_downvotes,
66 registration_mode: ls.registration_mode,
67 enable_nsfw: ls.enable_nsfw,
68 community_creation_admin_only: ls.community_creation_admin_only,
71 require_email_verification: ls.require_email_verification,
72 application_question: ls.application_question,
73 private_instance: ls.private_instance,
74 default_theme: ls.default_theme,
75 default_post_listing_type: ls.default_post_listing_type,
76 legal_information: ls.legal_information,
77 application_email_admins: ls.application_email_admins,
78 reports_email_admins: ls.reports_email_admins,
79 hide_modlog_mod_names: ls.hide_modlog_mod_names,
80 discussion_languages: this.props.siteRes.discussion_languages,
81 slur_filter_regex: ls.slur_filter_regex,
82 actor_name_max_length: ls.actor_name_max_length,
83 federation_enabled: ls.federation_enabled,
84 federation_worker_count: ls.federation_worker_count,
85 captcha_enabled: ls.captcha_enabled,
86 captcha_difficulty: ls.captcha_difficulty,
87 allowed_instances: this.props.allowedInstances?.map(i => i.domain),
88 blocked_instances: this.props.blockedInstances?.map(i => i.domain),
93 constructor(props: any, context: any) {
94 super(props, context);
96 this.handleSiteSidebarChange = this.handleSiteSidebarChange.bind(this);
97 this.handleSiteLegalInfoChange = this.handleSiteLegalInfoChange.bind(this);
98 this.handleSiteApplicationQuestionChange =
99 this.handleSiteApplicationQuestionChange.bind(this);
101 this.handleIconUpload = this.handleIconUpload.bind(this);
102 this.handleIconRemove = this.handleIconRemove.bind(this);
104 this.handleBannerUpload = this.handleBannerUpload.bind(this);
105 this.handleBannerRemove = this.handleBannerRemove.bind(this);
107 this.handleDefaultPostListingTypeChange =
108 this.handleDefaultPostListingTypeChange.bind(this);
110 this.handleDiscussionLanguageChange =
111 this.handleDiscussionLanguageChange.bind(this);
113 this.handleAddInstance = this.handleAddInstance.bind(this);
114 this.handleRemoveInstance = this.handleRemoveInstance.bind(this);
116 this.handleInstanceEnterPress = this.handleInstanceEnterPress.bind(this);
117 this.handleInstanceTextChange = this.handleInstanceTextChange.bind(this);
121 const siteSetup = this.props.siteRes.site_view.local_site.site_setup;
123 <form onSubmit={linkEvent(this, this.handleSaveSiteSubmit)}>
126 !this.props.loading &&
129 this.state.siteForm.name ||
130 this.state.siteForm.sidebar ||
131 this.state.siteForm.application_question ||
132 this.state.siteForm.description
134 !this.state.submitted
139 ? capitalizeFirstLetter(i18n.t("edit"))
140 : capitalizeFirstLetter(i18n.t("setup"))
141 } ${i18n.t("your_site")}`}</h5>
142 <div className="form-group row">
143 <label className="col-12 col-form-label" htmlFor="create-site-name">
146 <div className="col-12">
149 id="create-site-name"
150 className="form-control"
151 value={this.state.siteForm.name}
152 onInput={linkEvent(this, this.handleSiteNameChange)}
159 <div className="form-group">
160 <label className="mr-2">{i18n.t("icon")}</label>
162 uploadTitle={i18n.t("upload_icon")}
163 imageSrc={this.state.siteForm.icon}
164 onUpload={this.handleIconUpload}
165 onRemove={this.handleIconRemove}
169 <div className="form-group">
170 <label className="mr-2">{i18n.t("banner")}</label>
172 uploadTitle={i18n.t("upload_banner")}
173 imageSrc={this.state.siteForm.banner}
174 onUpload={this.handleBannerUpload}
175 onRemove={this.handleBannerRemove}
178 <div className="form-group row">
179 <label className="col-12 col-form-label" htmlFor="site-desc">
180 {i18n.t("description")}
182 <div className="col-12">
185 className="form-control"
187 value={this.state.siteForm.description}
188 onInput={linkEvent(this, this.handleSiteDescChange)}
193 <div className="form-group row">
194 <label className="col-12 col-form-label">{i18n.t("sidebar")}</label>
195 <div className="col-12">
197 initialContent={this.state.siteForm.sidebar}
198 onContentChange={this.handleSiteSidebarChange}
199 hideNavigationWarnings
205 <div className="form-group row">
206 <label className="col-12 col-form-label">
207 {i18n.t("legal_information")}
209 <div className="col-12">
211 initialContent={this.state.siteForm.legal_information}
212 onContentChange={this.handleSiteLegalInfoChange}
213 hideNavigationWarnings
219 <div className="form-group row">
220 <div className="col-12">
221 <div className="form-check">
223 className="form-check-input"
224 id="create-site-downvotes"
226 checked={this.state.siteForm.enable_downvotes}
227 onChange={linkEvent(this, this.handleSiteEnableDownvotesChange)}
230 className="form-check-label"
231 htmlFor="create-site-downvotes"
233 {i18n.t("enable_downvotes")}
238 <div className="form-group row">
239 <div className="col-12">
240 <div className="form-check">
242 className="form-check-input"
243 id="create-site-enable-nsfw"
245 checked={this.state.siteForm.enable_nsfw}
246 onChange={linkEvent(this, this.handleSiteEnableNsfwChange)}
249 className="form-check-label"
250 htmlFor="create-site-enable-nsfw"
252 {i18n.t("enable_nsfw")}
257 <div className="form-group row">
258 <div className="col-12">
260 className="form-check-label mr-2"
261 htmlFor="create-site-registration-mode"
263 {i18n.t("registration_mode")}
266 id="create-site-registration-mode"
267 value={this.state.siteForm.registration_mode}
268 onChange={linkEvent(this, this.handleSiteRegistrationModeChange)}
269 className="custom-select w-auto"
271 <option value={"RequireApplication"}>
272 {i18n.t("require_registration_application")}
274 <option value={"Open"}>{i18n.t("open_registration")}</option>
275 <option value={"Closed"}>{i18n.t("close_registration")}</option>
279 {this.state.siteForm.registration_mode == "RequireApplication" && (
280 <div className="form-group row">
281 <label className="col-12 col-form-label">
282 {i18n.t("application_questionnaire")}
284 <div className="col-12">
286 initialContent={this.state.siteForm.application_question}
287 onContentChange={this.handleSiteApplicationQuestionChange}
288 hideNavigationWarnings
295 <div className="form-group row">
296 <div className="col-12">
297 <div className="form-check">
299 className="form-check-input"
300 id="create-site-community-creation-admin-only"
302 checked={this.state.siteForm.community_creation_admin_only}
305 this.handleSiteCommunityCreationAdminOnly
309 className="form-check-label"
310 htmlFor="create-site-community-creation-admin-only"
312 {i18n.t("community_creation_admin_only")}
317 <div className="form-group row">
318 <div className="col-12">
319 <div className="form-check">
321 className="form-check-input"
322 id="create-site-require-email-verification"
324 checked={this.state.siteForm.require_email_verification}
327 this.handleSiteRequireEmailVerification
331 className="form-check-label"
332 htmlFor="create-site-require-email-verification"
334 {i18n.t("require_email_verification")}
339 <div className="form-group row">
340 <div className="col-12">
341 <div className="form-check">
343 className="form-check-input"
344 id="create-site-application-email-admins"
346 checked={this.state.siteForm.application_email_admins}
349 this.handleSiteApplicationEmailAdmins
353 className="form-check-label"
354 htmlFor="create-site-email-admins"
356 {i18n.t("application_email_admins")}
361 <div className="form-group row">
362 <div className="col-12">
363 <div className="form-check">
365 className="form-check-input"
366 id="create-site-reports-email-admins"
368 checked={this.state.siteForm.reports_email_admins}
369 onChange={linkEvent(this, this.handleSiteReportsEmailAdmins)}
372 className="form-check-label"
373 htmlFor="create-site-reports-email-admins"
375 {i18n.t("reports_email_admins")}
380 <div className="form-group row">
381 <div className="col-12">
383 className="form-check-label mr-2"
384 htmlFor="create-site-default-theme"
389 id="create-site-default-theme"
390 value={this.state.siteForm.default_theme}
391 onChange={linkEvent(this, this.handleSiteDefaultTheme)}
392 className="custom-select w-auto"
394 <option value="browser">{i18n.t("browser_default")}</option>
395 {this.props.themeList?.map(theme => (
396 <option key={theme} value={theme}>
403 {this.props.showLocal && (
404 <form className="form-group row">
405 <label className="col-sm-3">{i18n.t("listing_type")}</label>
406 <div className="col-sm-9">
408 type_={this.state.siteForm.default_post_listing_type ?? "Local"}
410 showSubscribed={false}
411 onChange={this.handleDefaultPostListingTypeChange}
416 <div className="form-group row">
417 <div className="col-12">
418 <div className="form-check">
420 className="form-check-input"
421 id="create-site-private-instance"
423 checked={this.state.siteForm.private_instance}
424 onChange={linkEvent(this, this.handleSitePrivateInstance)}
427 className="form-check-label"
428 htmlFor="create-site-private-instance"
430 {i18n.t("private_instance")}
435 <div className="form-group row">
436 <div className="col-12">
437 <div className="form-check">
439 className="form-check-input"
440 id="create-site-hide-modlog-mod-names"
442 checked={this.state.siteForm.hide_modlog_mod_names}
443 onChange={linkEvent(this, this.handleSiteHideModlogModNames)}
446 className="form-check-label"
447 htmlFor="create-site-hide-modlog-mod-names"
449 {i18n.t("hide_modlog_mod_names")}
454 <div className="form-group row">
456 className="col-12 col-form-label"
457 htmlFor="create-site-slur-filter-regex"
459 {i18n.t("slur_filter_regex")}
461 <div className="col-12">
464 id="create-site-slur-filter-regex"
465 placeholder="(word1|word2)"
466 className="form-control"
467 value={this.state.siteForm.slur_filter_regex}
468 onInput={linkEvent(this, this.handleSiteSlurFilterRegex)}
474 allLanguages={this.props.siteRes.all_languages}
475 siteLanguages={this.props.siteRes.discussion_languages}
476 selectedLanguageIds={this.state.siteForm.discussion_languages}
478 onChange={this.handleDiscussionLanguageChange}
481 <div className="form-group row">
483 className="col-12 col-form-label"
484 htmlFor="create-site-actor-name"
486 {i18n.t("actor_name_max_length")}
488 <div className="col-12">
491 id="create-site-actor-name"
492 className="form-control"
494 value={this.state.siteForm.actor_name_max_length}
495 onInput={linkEvent(this, this.handleSiteActorNameMaxLength)}
499 <div className="form-group row">
500 <div className="col-12">
501 <div className="form-check">
503 className="form-check-input"
504 id="create-site-federation-enabled"
506 checked={this.state.siteForm.federation_enabled}
507 onChange={linkEvent(this, this.handleSiteFederationEnabled)}
510 className="form-check-label"
511 htmlFor="create-site-federation-enabled"
513 {i18n.t("federation_enabled")}
518 {this.state.siteForm.federation_enabled && (
520 <div className="form-group row">
521 {this.federatedInstanceSelect("allowed_instances")}
522 {this.federatedInstanceSelect("blocked_instances")}
524 <div className="form-group row">
525 <div className="col-12">
526 <div className="form-check">
528 className="form-check-input"
529 id="create-site-federation-debug"
531 checked={this.state.siteForm.federation_debug}
532 onChange={linkEvent(this, this.handleSiteFederationDebug)}
535 className="form-check-label"
536 htmlFor="create-site-federation-debug"
538 {i18n.t("federation_debug")}
543 <div className="form-group row">
545 className="col-12 col-form-label"
546 htmlFor="create-site-federation-worker-count"
548 {i18n.t("federation_worker_count")}
550 <div className="col-12">
553 id="create-site-federation-worker-count"
554 className="form-control"
556 value={this.state.siteForm.federation_worker_count}
559 this.handleSiteFederationWorkerCount
566 <div className="form-group row">
567 <div className="col-12">
568 <div className="form-check">
570 className="form-check-input"
571 id="create-site-captcha-enabled"
573 checked={this.state.siteForm.captcha_enabled}
574 onChange={linkEvent(this, this.handleSiteCaptchaEnabled)}
577 className="form-check-label"
578 htmlFor="create-site-captcha-enabled"
580 {i18n.t("captcha_enabled")}
585 {this.state.siteForm.captcha_enabled && (
586 <div className="form-group row">
587 <div className="col-12">
589 className="form-check-label mr-2"
590 htmlFor="create-site-captcha-difficulty"
592 {i18n.t("captcha_difficulty")}
595 id="create-site-captcha-difficulty"
596 value={this.state.siteForm.captcha_difficulty}
597 onChange={linkEvent(this, this.handleSiteCaptchaDifficulty)}
598 className="custom-select w-auto"
600 <option value="easy">{i18n.t("easy")}</option>
601 <option value="medium">{i18n.t("medium")}</option>
602 <option value="hard">{i18n.t("hard")}</option>
607 <div className="form-group row">
608 <div className="col-12">
611 className="btn btn-secondary mr-2"
612 disabled={this.props.loading}
614 {this.props.loading ? (
617 capitalizeFirstLetter(i18n.t("save"))
619 capitalizeFirstLetter(i18n.t("create"))
628 federatedInstanceSelect(key: InstanceKey) {
629 const id = `create_site_${key}`;
630 const value = this.state.instance_select[key];
631 const selectedInstances = this.state.siteForm[key];
633 <div className="col-12 col-md-6">
634 <label className="col-form-label" htmlFor={id}>
637 <div className="d-flex justify-content-between align-items-center">
640 placeholder="instance.tld"
642 className="form-control"
644 onInput={linkEvent(key, this.handleInstanceTextChange)}
645 onKeyUp={linkEvent(key, this.handleInstanceEnterPress)}
649 className="btn btn-sm bg-success ml-2"
650 onClick={linkEvent(key, this.handleAddInstance)}
651 style={"width: 2rem; height: 2rem;"}
653 -1 /* Making this untabble because handling enter key in text input makes keyboard support for this button redundant */
658 classes="icon-inline text-light m-auto d-block position-static"
662 {selectedInstances && selectedInstances.length > 0 && (
663 <ul className="mt-3 list-unstyled w-100 d-flex flex-column justify-content-around align-items-center">
664 {selectedInstances.map(instance => (
667 className="my-1 w-100 w-md-75 d-flex align-items-center justify-content-between"
669 <label className="d-block m-0 w-100 " htmlFor={instance}>
670 <strong>{instance}</strong>
675 style={"width: 2rem; height: 2rem;"}
676 className="btn btn-sm bg-danger"
679 this.handleRemoveInstance
684 classes="icon-inline text-light m-auto d-block position-static"
695 handleInstanceTextChange(type: InstanceKey, event: any) {
696 this.setState(s => ({
699 ...s.instance_select,
700 [type]: event.target.value,
705 handleInstanceEnterPress(
707 event: InfernoKeyboardEvent<HTMLInputElement>
709 if (event.code.toLowerCase() === "enter") {
710 event.preventDefault();
712 this.handleAddInstance(key);
716 handleSaveSiteSubmit(i: SiteForm, event: any) {
717 event.preventDefault();
718 const auth = myAuthRequired();
719 i.setState(s => ((s.siteForm.auth = auth), s));
720 i.setState({ submitted: true });
722 const stateSiteForm = i.state.siteForm;
724 let form: EditSite | CreateSite;
726 if (i.props.siteRes.site_view.local_site.site_setup) {
727 form = stateSiteForm;
730 name: stateSiteForm.name ?? "My site",
731 sidebar: stateSiteForm.sidebar,
732 description: stateSiteForm.description,
733 icon: stateSiteForm.icon,
734 banner: stateSiteForm.banner,
735 community_creation_admin_only:
736 stateSiteForm.community_creation_admin_only,
737 enable_nsfw: stateSiteForm.enable_nsfw,
738 enable_downvotes: stateSiteForm.enable_downvotes,
739 application_question: stateSiteForm.application_question,
740 registration_mode: stateSiteForm.registration_mode,
741 require_email_verification: stateSiteForm.require_email_verification,
742 private_instance: stateSiteForm.private_instance,
743 default_theme: stateSiteForm.default_theme,
744 default_post_listing_type: stateSiteForm.default_post_listing_type,
745 application_email_admins: stateSiteForm.application_email_admins,
746 hide_modlog_mod_names: stateSiteForm.hide_modlog_mod_names,
747 legal_information: stateSiteForm.legal_information,
748 slur_filter_regex: stateSiteForm.slur_filter_regex,
749 actor_name_max_length: stateSiteForm.actor_name_max_length,
750 rate_limit_message: stateSiteForm.rate_limit_message,
751 rate_limit_message_per_second:
752 stateSiteForm.rate_limit_message_per_second,
753 rate_limit_comment: stateSiteForm.rate_limit_comment,
754 rate_limit_comment_per_second:
755 stateSiteForm.rate_limit_comment_per_second,
756 rate_limit_image: stateSiteForm.rate_limit_image,
757 rate_limit_image_per_second: stateSiteForm.rate_limit_image_per_second,
758 rate_limit_post: stateSiteForm.rate_limit_post,
759 rate_limit_post_per_second: stateSiteForm.rate_limit_post_per_second,
760 rate_limit_register: stateSiteForm.rate_limit_register,
761 rate_limit_register_per_second:
762 stateSiteForm.rate_limit_register_per_second,
763 rate_limit_search: stateSiteForm.rate_limit_search,
764 rate_limit_search_per_second:
765 stateSiteForm.rate_limit_search_per_second,
766 federation_enabled: stateSiteForm.federation_enabled,
767 federation_debug: stateSiteForm.federation_debug,
768 federation_worker_count: stateSiteForm.federation_worker_count,
769 captcha_enabled: stateSiteForm.captcha_enabled,
770 captcha_difficulty: stateSiteForm.captcha_difficulty,
771 allowed_instances: stateSiteForm.allowed_instances,
772 blocked_instances: stateSiteForm.blocked_instances,
773 discussion_languages: stateSiteForm.discussion_languages,
778 i.props.onSaveSite(form);
781 handleAddInstance(key: InstanceKey) {
782 const instance = this.state.instance_select[key].trim();
784 if (!validInstanceTLD(instance)) {
788 if (!this.state.siteForm[key]?.includes(instance)) {
789 this.setState(s => ({
793 [key]: [...(s.siteForm[key] ?? []), instance],
796 ...s.instance_select,
801 const oppositeKey: InstanceKey =
802 key === "allowed_instances" ? "blocked_instances" : "allowed_instances";
803 if (this.state.siteForm[oppositeKey]?.includes(instance)) {
804 this.handleRemoveInstance({ key: oppositeKey, instance });
809 handleRemoveInstance({
816 this.setState(s => ({
820 [key]: s.siteForm[key]?.filter(i => i !== instance),
825 handleSiteNameChange(i: SiteForm, event: any) {
826 i.state.siteForm.name = event.target.value;
830 handleSiteSidebarChange(val: string) {
831 this.setState(s => ((s.siteForm.sidebar = val), s));
834 handleSiteLegalInfoChange(val: string) {
835 this.setState(s => ((s.siteForm.legal_information = val), s));
838 handleTaglineChange(i: SiteForm, index: number, val: string) {
839 const taglines = i.state.siteForm.taglines;
841 taglines[index] = val;
846 handleDeleteTaglineClick(
849 event: InfernoMouseEvent<HTMLButtonElement>
851 event.preventDefault();
852 const taglines = i.state.siteForm.taglines;
854 taglines.splice(index, 1);
855 i.state.siteForm.taglines = undefined;
857 i.state.siteForm.taglines = taglines;
862 handleAddTaglineClick(
864 event: InfernoMouseEvent<HTMLButtonElement>
866 event.preventDefault();
867 if (!i.state.siteForm.taglines) {
868 i.state.siteForm.taglines = [];
870 i.state.siteForm.taglines.push("");
874 handleSiteApplicationQuestionChange(val: string) {
875 this.setState(s => ((s.siteForm.application_question = val), s));
878 handleSiteDescChange(i: SiteForm, event: any) {
879 i.state.siteForm.description = event.target.value;
883 handleSiteEnableNsfwChange(i: SiteForm, event: any) {
884 i.state.siteForm.enable_nsfw = event.target.checked;
888 handleSiteRegistrationModeChange(i: SiteForm, event: any) {
889 i.state.siteForm.registration_mode = event.target.value;
893 handleSiteCommunityCreationAdminOnly(i: SiteForm, event: any) {
894 i.state.siteForm.community_creation_admin_only = event.target.checked;
898 handleSiteEnableDownvotesChange(i: SiteForm, event: any) {
899 i.state.siteForm.enable_downvotes = event.target.checked;
903 handleSiteRequireEmailVerification(i: SiteForm, event: any) {
904 i.state.siteForm.require_email_verification = event.target.checked;
908 handleSiteApplicationEmailAdmins(i: SiteForm, event: any) {
909 i.state.siteForm.application_email_admins = event.target.checked;
913 handleSiteReportsEmailAdmins(i: SiteForm, event: any) {
914 i.state.siteForm.reports_email_admins = event.target.checked;
918 handleSitePrivateInstance(i: SiteForm, event: any) {
919 i.state.siteForm.private_instance = event.target.checked;
923 handleSiteHideModlogModNames(i: SiteForm, event: any) {
924 i.state.siteForm.hide_modlog_mod_names = event.target.checked;
928 handleSiteDefaultTheme(i: SiteForm, event: any) {
929 i.state.siteForm.default_theme = event.target.value;
933 handleIconUpload(url: string) {
934 this.setState(s => ((s.siteForm.icon = url), s));
938 this.setState(s => ((s.siteForm.icon = ""), s));
941 handleBannerUpload(url: string) {
942 this.setState(s => ((s.siteForm.banner = url), s));
945 handleBannerRemove() {
946 this.setState(s => ((s.siteForm.banner = ""), s));
949 handleSiteSlurFilterRegex(i: SiteForm, event: any) {
950 i.setState(s => ((s.siteForm.slur_filter_regex = event.target.value), s));
953 handleSiteActorNameMaxLength(i: SiteForm, event: any) {
955 s => ((s.siteForm.actor_name_max_length = Number(event.target.value)), s)
959 handleSiteFederationEnabled(i: SiteForm, event: any) {
960 i.state.siteForm.federation_enabled = event.target.checked;
964 handleSiteFederationDebug(i: SiteForm, event: any) {
965 i.state.siteForm.federation_debug = event.target.checked;
969 handleSiteFederationWorkerCount(i: SiteForm, event: any) {
972 (s.siteForm.federation_worker_count = Number(event.target.value)), s
977 handleSiteCaptchaEnabled(i: SiteForm, event: any) {
978 i.state.siteForm.captcha_enabled = event.target.checked;
982 handleSiteCaptchaDifficulty(i: SiteForm, event: any) {
983 i.setState(s => ((s.siteForm.captcha_difficulty = event.target.value), s));
986 handleDiscussionLanguageChange(val: number[]) {
987 this.setState(s => ((s.siteForm.discussion_languages = val), s));
990 handleDefaultPostListingTypeChange(val: ListingType) {
991 this.setState(s => ((s.siteForm.default_post_listing_type = val), s));