summaryrefslogtreecommitdiff
path: root/web/source/settings/views/user/posts/interaction-policy-settings/index.tsx
diff options
context:
space:
mode:
Diffstat (limited to 'web/source/settings/views/user/posts/interaction-policy-settings/index.tsx')
-rw-r--r--web/source/settings/views/user/posts/interaction-policy-settings/index.tsx553
1 files changed, 553 insertions, 0 deletions
diff --git a/web/source/settings/views/user/posts/interaction-policy-settings/index.tsx b/web/source/settings/views/user/posts/interaction-policy-settings/index.tsx
new file mode 100644
index 000000000..143cf0865
--- /dev/null
+++ b/web/source/settings/views/user/posts/interaction-policy-settings/index.tsx
@@ -0,0 +1,553 @@
+/*
+ GoToSocial
+ Copyright (C) GoToSocial Authors admin@gotosocial.org
+ SPDX-License-Identifier: AGPL-3.0-or-later
+
+ This program is free software: you can redistribute it and/or modify
+ it under the terms of the GNU Affero General Public License as published by
+ the Free Software Foundation, either version 3 of the License, or
+ (at your option) any later version.
+
+ This program is distributed in the hope that it will be useful,
+ but WITHOUT ANY WARRANTY; without even the implied warranty of
+ MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ GNU Affero General Public License for more details.
+
+ You should have received a copy of the GNU Affero General Public License
+ along with this program. If not, see <http://www.gnu.org/licenses/>.
+*/
+
+import React, { useCallback, useMemo } from "react";
+import {
+ useDefaultInteractionPoliciesQuery,
+ useResetDefaultInteractionPoliciesMutation,
+ useUpdateDefaultInteractionPoliciesMutation,
+} from "../../../../lib/query/user";
+import Loading from "../../../../components/loading";
+import { Error } from "../../../../components/error";
+import MutationButton from "../../../../components/form/mutation-button";
+import {
+ DefaultInteractionPolicies,
+ InteractionPolicy,
+ InteractionPolicyEntry,
+ InteractionPolicyValue,
+ PolicyValueAuthor,
+ PolicyValueFollowers,
+ PolicyValueFollowing,
+ PolicyValueMentioned,
+ PolicyValuePublic,
+} from "../../../../lib/types/interaction";
+import { useTextInput } from "../../../../lib/form";
+import { Select } from "../../../../components/form/inputs";
+import { TextFormInputHook } from "../../../../lib/form/types";
+import { useBasicFor } from "./basic";
+import { PolicyFormSomethingElse, useSomethingElseFor } from "./something-else";
+import { Action, PolicyFormSub, SomethingElseValue, Visibility } from "./types";
+
+export default function InteractionPolicySettings() {
+ const {
+ data: defaultPolicies,
+ isLoading,
+ isFetching,
+ isError,
+ error,
+ } = useDefaultInteractionPoliciesQuery();
+
+ if (isLoading || isFetching) {
+ return <Loading />;
+ }
+
+ if (isError) {
+ return <Error error={error} />;
+ }
+
+ if (!defaultPolicies) {
+ throw "default policies undefined";
+ }
+
+ return (
+ <InteractionPoliciesForm defaultPolicies={defaultPolicies} />
+ );
+}
+
+interface InteractionPoliciesFormProps {
+ defaultPolicies: DefaultInteractionPolicies;
+}
+
+function InteractionPoliciesForm({ defaultPolicies }: InteractionPoliciesFormProps) {
+ // Sub-form for visibility "public".
+ const formPublic = useFormForVis(defaultPolicies.public, "public");
+ const assemblePublic = useCallback(() => {
+ return {
+ can_favourite: assemblePolicyEntry("public", "favourite", formPublic),
+ can_reply: assemblePolicyEntry("public", "reply", formPublic),
+ can_reblog: assemblePolicyEntry("public", "reblog", formPublic),
+ };
+ }, [formPublic]);
+
+ // Sub-form for visibility "unlisted".
+ const formUnlisted = useFormForVis(defaultPolicies.unlisted, "unlisted");
+ const assembleUnlisted = useCallback(() => {
+ return {
+ can_favourite: assemblePolicyEntry("unlisted", "favourite", formUnlisted),
+ can_reply: assemblePolicyEntry("unlisted", "reply", formUnlisted),
+ can_reblog: assemblePolicyEntry("unlisted", "reblog", formUnlisted),
+ };
+ }, [formUnlisted]);
+
+ // Sub-form for visibility "private".
+ const formPrivate = useFormForVis(defaultPolicies.private, "private");
+ const assemblePrivate = useCallback(() => {
+ return {
+ can_favourite: assemblePolicyEntry("private", "favourite", formPrivate),
+ can_reply: assemblePolicyEntry("private", "reply", formPrivate),
+ can_reblog: assemblePolicyEntry("private", "reblog", formPrivate),
+ };
+ }, [formPrivate]);
+
+ const selectedVis = useTextInput("selectedVis", { defaultValue: "public" });
+
+ const [updatePolicies, updateResult] = useUpdateDefaultInteractionPoliciesMutation();
+ const [resetPolicies, resetResult] = useResetDefaultInteractionPoliciesMutation();
+
+ const onSubmit = (e) => {
+ e.preventDefault();
+ updatePolicies({
+ public: assemblePublic(),
+ unlisted: assembleUnlisted(),
+ private: assemblePrivate(),
+ // Always use the
+ // default for direct.
+ direct: null,
+ });
+ };
+
+ return (
+ <form className="interaction-default-settings" onSubmit={onSubmit}>
+ <div className="form-section-docs">
+ <h3>Default Interaction Policies</h3>
+ <p>
+ You can use this section to customize the default interaction
+ policy for posts created by you, per visibility setting.
+ <br/>
+ These settings apply only for new posts created by you <em>after</em> applying
+ these settings; they do not apply retroactively.
+ <br/>
+ The word "anyone" in the below options means <em>anyone with
+ permission to see the post</em>, taking account of blocks.
+ <br/>
+ Bear in mind that no matter what you set below, you will always
+ be able to like, reply-to, and boost your own posts.
+ </p>
+ <a
+ href="https://docs.gotosocial.org/en/latest/user_guide/settings#default-interaction-policies"
+ target="_blank"
+ className="docslink"
+ rel="noreferrer"
+ >
+ Learn more about these settings (opens in a new tab)
+ </a>
+ </div>
+ <div className="tabbable-sections">
+ <PolicyPanelsTablist selectedVis={selectedVis} />
+ <PolicyPanel
+ policyForm={formPublic}
+ forVis={"public"}
+ isActive={selectedVis.value === "public"}
+ />
+ <PolicyPanel
+ policyForm={formUnlisted}
+ forVis={"unlisted"}
+ isActive={selectedVis.value === "unlisted"}
+ />
+ <PolicyPanel
+ policyForm={formPrivate}
+ forVis={"private"}
+ isActive={selectedVis.value === "private"}
+ />
+ </div>
+
+ <div className="action-buttons row">
+ <MutationButton
+ disabled={false}
+ label="Save policies"
+ result={updateResult}
+ />
+
+ <MutationButton
+ disabled={false}
+ type="button"
+ onClick={() => resetPolicies()}
+ label="Reset to defaults"
+ result={resetResult}
+ className="button danger"
+ showError={false}
+ />
+ </div>
+
+ </form>
+ );
+}
+
+// A tablist of tab buttons, one for each visibility.
+function PolicyPanelsTablist({ selectedVis }: { selectedVis: TextFormInputHook}) {
+ return (
+ <div className="tab-buttons" role="tablist">
+ <Tab
+ thisVisibility="public"
+ label="Public"
+ selectedVis={selectedVis}
+ />
+ <Tab
+ thisVisibility="unlisted"
+ label="Unlisted"
+ selectedVis={selectedVis}
+ />
+ <Tab
+ thisVisibility="private"
+ label="Followers-only"
+ selectedVis={selectedVis}
+ />
+ </div>
+ );
+}
+
+interface TabProps {
+ thisVisibility: string;
+ label: string,
+ selectedVis: TextFormInputHook
+}
+
+// One tab in a tablist, corresponding to the given thisVisibility.
+function Tab({ thisVisibility, label, selectedVis }: TabProps) {
+ const selected = useMemo(() => {
+ return selectedVis.value === thisVisibility;
+ }, [selectedVis, thisVisibility]);
+
+ return (
+ <button
+ id={`tab-${thisVisibility}`}
+ title={label}
+ role="tab"
+ className={`tab-button ${selected && "active"}`}
+ onClick={(e) => {
+ e.preventDefault();
+ selectedVis.setter(thisVisibility);
+ }}
+ aria-selected={selected}
+ aria-controls={`panel-${thisVisibility}`}
+ tabIndex={selected ? 0 : -1}
+ >
+ {label}
+ </button>
+ );
+}
+
+interface PolicyPanelProps {
+ policyForm: PolicyForm;
+ forVis: Visibility;
+ isActive: boolean;
+}
+
+// Tab panel for one policy form of the given visibility.
+function PolicyPanel({ policyForm, forVis, isActive }: PolicyPanelProps) {
+ return (
+ <div
+ className={`interaction-policy-section ${isActive && "active"}`}
+ role="tabpanel"
+ hidden={!isActive}
+ >
+ <PolicyComponent
+ form={policyForm.favourite}
+ forAction="favourite"
+ />
+ <PolicyComponent
+ form={policyForm.reply}
+ forAction="reply"
+ />
+ { forVis !== "private" &&
+ <PolicyComponent
+ form={policyForm.reblog}
+ forAction="reblog"
+ />
+ }
+ </div>
+ );
+}
+
+interface PolicyComponentProps {
+ form: {
+ basic: PolicyFormSub;
+ somethingElse: PolicyFormSomethingElse;
+ };
+ forAction: Action;
+}
+
+// A component of one policy of the given
+// visibility, corresponding to the given action.
+function PolicyComponent({ form, forAction }: PolicyComponentProps) {
+ const legend = useLegend(forAction);
+ return (
+ <fieldset>
+ <legend>{legend}</legend>
+ { forAction === "reply" &&
+ <div className="info">
+ <i className="fa fa-fw fa-info-circle" aria-hidden="true"></i>
+ <b>Mentioned accounts can always reply.</b>
+ </div>
+ }
+ <Select
+ field={form.basic.field}
+ label={form.basic.label}
+ options={form.basic.options}
+ />
+ {/* Include advanced "something else" options if appropriate */}
+ { (form.basic.field.value === "something_else") &&
+ <>
+ <hr />
+ <div className="something-else">
+ <Select
+ field={form.somethingElse.followers.field}
+ label={form.somethingElse.followers.label}
+ options={form.somethingElse.followers.options}
+ />
+ <Select
+ field={form.somethingElse.following.field}
+ label={form.somethingElse.following.label}
+ options={form.somethingElse.following.options}
+ />
+ {/*
+ Skip mentioned accounts field for reply action,
+ since mentioned accounts can always reply.
+ */}
+ { forAction !== "reply" &&
+ <Select
+ field={form.somethingElse.mentioned.field}
+ label={form.somethingElse.mentioned.label}
+ options={form.somethingElse.mentioned.options}
+ />
+ }
+ <Select
+ field={form.somethingElse.everyoneElse.field}
+ label={form.somethingElse.everyoneElse.label}
+ options={form.somethingElse.everyoneElse.options}
+ />
+ </div>
+ </>
+ }
+ </fieldset>
+ );
+}
+
+/*
+ UTILITY FUNCTIONS
+*/
+
+// useLegend returns an appropriate
+// fieldset legend for the given action.
+function useLegend(action: Action) {
+ return useMemo(() => {
+ switch (action) {
+ case "favourite":
+ return (
+ <>
+ <i className="fa fa-fw fa-star" aria-hidden="true"></i>
+ <span>Like</span>
+ </>
+ );
+ case "reply":
+ return (
+ <>
+ <i className="fa fa-fw fa-reply-all" aria-hidden="true"></i>
+ <span>Reply</span>
+ </>
+ );
+ case "reblog":
+ return (
+ <>
+ <i className="fa fa-fw fa-retweet" aria-hidden="true"></i>
+ <span>Boost</span>
+ </>
+ );
+ }
+ }, [action]);
+}
+
+// Form encapsulating the different
+// actions for one visibility.
+interface PolicyForm {
+ favourite: {
+ basic: PolicyFormSub,
+ somethingElse: PolicyFormSomethingElse,
+ }
+ reply: {
+ basic: PolicyFormSub,
+ somethingElse: PolicyFormSomethingElse,
+ }
+ reblog: {
+ basic: PolicyFormSub,
+ somethingElse: PolicyFormSomethingElse,
+ }
+}
+
+// Return a PolicyForm for the given visibility,
+// set already to whatever the defaultPolicies value is.
+function useFormForVis(
+ currentPolicy: InteractionPolicy,
+ forVis: Visibility,
+): PolicyForm {
+ return {
+ favourite: {
+ basic: useBasicFor(
+ forVis,
+ "favourite",
+ currentPolicy.can_favourite.always,
+ currentPolicy.can_favourite.with_approval,
+ ),
+ somethingElse: useSomethingElseFor(
+ forVis,
+ "favourite",
+ currentPolicy.can_favourite.always,
+ currentPolicy.can_favourite.with_approval,
+ ),
+ },
+ reply: {
+ basic: useBasicFor(
+ forVis,
+ "reply",
+ currentPolicy.can_reply.always,
+ currentPolicy.can_reply.with_approval,
+ ),
+ somethingElse: useSomethingElseFor(
+ forVis,
+ "reply",
+ currentPolicy.can_reply.always,
+ currentPolicy.can_reply.with_approval,
+ ),
+ },
+ reblog: {
+ basic: useBasicFor(
+ forVis,
+ "reblog",
+ currentPolicy.can_reblog.always,
+ currentPolicy.can_reblog.with_approval,
+ ),
+ somethingElse: useSomethingElseFor(
+ forVis,
+ "reblog",
+ currentPolicy.can_reblog.always,
+ currentPolicy.can_reblog.with_approval,
+ ),
+ },
+ };
+}
+
+function assemblePolicyEntry(
+ forVis: Visibility,
+ forAction: Action,
+ policyForm: PolicyForm,
+): InteractionPolicyEntry {
+ const basic = policyForm[forAction].basic;
+
+ // If this is followers visibility then
+ // "anyone" only means followers, not public.
+ const anyone: InteractionPolicyValue =
+ (forVis === "private")
+ ? PolicyValueFollowers
+ : PolicyValuePublic;
+
+ // If this is a reply action then "just me"
+ // must include mentioned accounts as well,
+ // since they can always reply.
+ const justMe: InteractionPolicyValue[] =
+ (forAction === "reply")
+ ? [PolicyValueAuthor, PolicyValueMentioned]
+ : [PolicyValueAuthor];
+
+ switch (basic.field.value) {
+ case "anyone":
+ return {
+ // Anyone can do this.
+ always: [anyone],
+ with_approval: [],
+ };
+ case "anyone_with_approval":
+ return {
+ // Author and maybe mentioned can do
+ // this, everyone else needs approval.
+ always: justMe,
+ with_approval: [anyone],
+ };
+ case "just_me":
+ return {
+ // Only author and maybe
+ // mentioned can do this.
+ always: justMe,
+ with_approval: [],
+ };
+ }
+
+ // Something else!
+ const somethingElse = policyForm[forAction].somethingElse;
+
+ // Start with basic "always"
+ // and "with_approval" values.
+ let always: InteractionPolicyValue[] = justMe;
+ let withApproval: InteractionPolicyValue[] = [];
+
+ // Add PolicyValueFollowers depending on choices made.
+ switch (somethingElse.followers.field.value as SomethingElseValue) {
+ case "always":
+ always.push(PolicyValueFollowers);
+ break;
+ case "with_approval":
+ withApproval.push(PolicyValueFollowers);
+ break;
+ }
+
+ // Add PolicyValueFollowing depending on choices made.
+ switch (somethingElse.following.field.value as SomethingElseValue) {
+ case "always":
+ always.push(PolicyValueFollowing);
+ break;
+ case "with_approval":
+ withApproval.push(PolicyValueFollowing);
+ break;
+ }
+
+ // Add PolicyValueMentioned depending on choices made.
+ // Note: mentioned can always reply, and that's already
+ // included above, so only do this if action is not reply.
+ if (forAction !== "reply") {
+ switch (somethingElse.mentioned.field.value as SomethingElseValue) {
+ case "always":
+ always.push(PolicyValueMentioned);
+ break;
+ case "with_approval":
+ withApproval.push(PolicyValueMentioned);
+ break;
+ }
+ }
+
+ // Add anyone depending on choices made.
+ switch (somethingElse.everyoneElse.field.value as SomethingElseValue) {
+ case "with_approval":
+ withApproval.push(anyone);
+ break;
+ }
+
+ // Simplify a bit after
+ // all the parsing above.
+ if (always.includes(anyone)) {
+ always = [anyone];
+ }
+
+ if (withApproval.includes(anyone)) {
+ withApproval = [anyone];
+ }
+
+ return {
+ always: always,
+ with_approval: withApproval,
+ };
+}