summaryrefslogtreecommitdiff
path: root/packages/frontend/src/pages/admin/moderation.vue
diff options
context:
space:
mode:
Diffstat (limited to 'packages/frontend/src/pages/admin/moderation.vue')
-rw-r--r--packages/frontend/src/pages/admin/moderation.vue259
1 files changed, 192 insertions, 67 deletions
diff --git a/packages/frontend/src/pages/admin/moderation.vue b/packages/frontend/src/pages/admin/moderation.vue
index 0a5b06a969..bbcf2a6f77 100644
--- a/packages/frontend/src/pages/admin/moderation.vue
+++ b/packages/frontend/src/pages/admin/moderation.vue
@@ -10,76 +10,130 @@ SPDX-License-Identifier: AGPL-3.0-only
<MkSpacer :contentMax="700" :marginMin="16" :marginMax="32">
<FormSuspense :p="init">
<div class="_gaps_m">
- <MkSwitch v-model="enableRegistration">
+ <MkSwitch v-model="enableRegistration" @change="onChange_enableRegistration">
<template #label>{{ i18n.ts.enableRegistration }}</template>
</MkSwitch>
- <MkSwitch v-model="emailRequiredForSignup">
+ <MkSwitch v-model="emailRequiredForSignup" @change="onChange_emailRequiredForSignup">
<template #label>{{ i18n.ts.emailRequiredForSignup }}</template>
</MkSwitch>
- <MkSwitch v-model="approvalRequiredForSignup">
+ <MkSwitch v-model="approvalRequiredForSignup" @change="onChange_approvalRequiredForSignup">
<template #label>{{ i18n.ts.approvalRequiredForSignup }}</template>
</MkSwitch>
<FormLink to="/admin/server-rules">{{ i18n.ts.serverRules }}</FormLink>
- <MkInput v-model="tosUrl" type="url">
- <template #prefix><i class="ti ti-link"></i></template>
- <template #label>{{ i18n.ts.tosUrl }}</template>
- </MkInput>
-
- <MkInput v-model="privacyPolicyUrl" type="url">
- <template #prefix><i class="ti ti-link"></i></template>
- <template #label>{{ i18n.ts.privacyPolicyUrl }}</template>
- </MkInput>
-
- <MkTextarea v-if="bubbleTimelineEnabled" v-model="bubbleTimeline">
+ <MkFolder v-if="bubbleTimelineEnabled">
+ <template #icon><i class="ph-drop ph-bold ph-lg"></i></template>
<template #label>Bubble timeline</template>
- <template #caption>Choose which instances should be displayed in the bubble.</template>
- </MkTextarea>
- <MkInput v-model="inquiryUrl" type="url">
+ <div class="_gaps">
+ <MkTextarea v-model="bubbleTimeline">
+ <template #caption>Choose which instances should be displayed in the bubble.</template>
+ </MkTextarea>
+ <MkButton primary @click="save_bubbleTimeline">{{ i18n.ts.save }}</MkButton>
+ </div>
+ </MkFolder>
+
+ <MkFolder>
<template #prefix><i class="ti ti-link"></i></template>
- <template #label>{{ i18n.ts._serverSettings.inquiryUrl }}</template>
- <template #caption>{{ i18n.ts._serverSettings.inquiryUrlDescription }}</template>
- </MkInput>
+ <template #label>{{ i18n.ts.trustedLinkUrlPatterns }}</template>
- <MkTextarea v-model="preservedUsernames">
+ <div class="_gaps">
+ <MkTextarea v-model="trustedLinkUrlPatterns">
+ <template #caption>{{ i18n.ts.trustedLinkUrlPatternsDescription }}</template>
+ </MkTextarea>
+ <MkButton primary @click="save_trustedLinkUrlPatterns">{{ i18n.ts.save }}</MkButton>
+ </div>
+ </MkFolder>
+
+ <MkFolder>
+ <template #icon><i class="ti ti-lock-star"></i></template>
<template #label>{{ i18n.ts.preservedUsernames }}</template>
- <template #caption>{{ i18n.ts.preservedUsernamesDescription }}</template>
- </MkTextarea>
- <MkTextarea v-model="trustedLinkUrlPatterns">
- <template #prefix><i class="ti ti-link"></i></template>
- <template #label>{{ i18n.ts.trustedLinkUrlPatterns }}</template>
- <template #caption>{{ i18n.ts.trustedLinkUrlPatternsDescription }}</template>
- </MkTextarea>
+ <div class="_gaps">
+ <MkTextarea v-model="preservedUsernames">
+ <template #caption>{{ i18n.ts.preservedUsernamesDescription }}</template>
+ </MkTextarea>
+ <MkButton primary @click="save_preservedUsernames">{{ i18n.ts.save }}</MkButton>
+ </div>
+ </MkFolder>
- <MkTextarea v-model="sensitiveWords">
+ <MkFolder>
+ <template #icon><i class="ti ti-message-exclamation"></i></template>
<template #label>{{ i18n.ts.sensitiveWords }}</template>
- <template #caption>{{ i18n.ts.sensitiveWordsDescription }}<br>{{ i18n.ts.sensitiveWordsDescription2 }}</template>
- </MkTextarea>
- <MkTextarea v-model="prohibitedWords">
+ <div class="_gaps">
+ <MkTextarea v-model="sensitiveWords">
+ <template #caption>{{ i18n.ts.sensitiveWordsDescription }}<br>{{ i18n.ts.sensitiveWordsDescription2 }}</template>
+ </MkTextarea>
+ <MkButton primary @click="save_sensitiveWords">{{ i18n.ts.save }}</MkButton>
+ </div>
+ </MkFolder>
+
+ <MkFolder>
+ <template #icon><i class="ti ti-message-x"></i></template>
<template #label>{{ i18n.ts.prohibitedWords }}</template>
- <template #caption>{{ i18n.ts.prohibitedWordsDescription }}<br>{{ i18n.ts.prohibitedWordsDescription2 }}</template>
- </MkTextarea>
- <MkTextarea v-model="hiddenTags">
+ <div class="_gaps">
+ <MkTextarea v-model="prohibitedWords">
+ <template #caption>{{ i18n.ts.prohibitedWordsDescription }}<br>{{ i18n.ts.prohibitedWordsDescription2 }}</template>
+ </MkTextarea>
+ <MkButton primary @click="save_prohibitedWords">{{ i18n.ts.save }}</MkButton>
+ </div>
+ </MkFolder>
+
+ <MkFolder>
+ <template #icon><i class="ti ti-eye-off"></i></template>
<template #label>{{ i18n.ts.hiddenTags }}</template>
- <template #caption>{{ i18n.ts.hiddenTagsDescription }}</template>
- </MkTextarea>
+
+ <div class="_gaps">
+ <MkTextarea v-model="hiddenTags">
+ <template #caption>{{ i18n.ts.hiddenTagsDescription }}</template>
+ </MkTextarea>
+ <MkButton primary @click="save_hiddenTags">{{ i18n.ts.save }}</MkButton>
+ </div>
+ </MkFolder>
+
+ <MkFolder>
+ <template #icon><i class="ti ti-eye-off"></i></template>
+ <template #label>{{ i18n.ts.silencedInstances }}</template>
+
+ <div class="_gaps">
+ <MkTextarea v-model="silencedHosts">
+ <template #caption>{{ i18n.ts.silencedInstancesDescription }}</template>
+ </MkTextarea>
+ <MkButton primary @click="save_silencedHosts">{{ i18n.ts.save }}</MkButton>
+ </div>
+ </MkFolder>
+
+ <MkFolder>
+ <template #icon><i class="ti ti-eye-off"></i></template>
+ <template #label>{{ i18n.ts.mediaSilencedInstances }}</template>
+
+ <div class="_gaps">
+ <MkTextarea v-model="mediaSilencedHosts">
+ <template #caption>{{ i18n.ts.mediaSilencedInstancesDescription }}</template>
+ </MkTextarea>
+ <MkButton primary @click="save_mediaSilencedHosts">{{ i18n.ts.save }}</MkButton>
+ </div>
+ </MkFolder>
+
+ <MkFolder>
+ <template #icon><i class="ti ti-ban"></i></template>
+ <template #label>{{ i18n.ts.blockedInstances }}</template>
+
+ <div class="_gaps">
+ <MkTextarea v-model="blockedHosts">
+ <template #caption>{{ i18n.ts.blockedInstancesDescription }}</template>
+ </MkTextarea>
+ <MkButton primary @click="save_blockedHosts">{{ i18n.ts.save }}</MkButton>
+ </div>
+ </MkFolder>
</div>
</FormSuspense>
</MkSpacer>
- <template #footer>
- <div :class="$style.footer">
- <MkSpacer :contentMax="700" :marginMin="16" :marginMax="16">
- <MkButton primary rounded @click="save"><i class="ti ti-check"></i> {{ i18n.ts.save }}</MkButton>
- </MkSpacer>
- </div>
- </template>
</MkStickyContainer>
</div>
</template>
@@ -98,6 +152,7 @@ import { i18n } from '@/i18n.js';
import { definePageMetadata } from '@/scripts/page-metadata.js';
import MkButton from '@/components/MkButton.vue';
import FormLink from '@/components/form/link.vue';
+import MkFolder from '@/components/MkFolder.vue';
const enableRegistration = ref<boolean>(false);
const emailRequiredForSignup = ref<boolean>(false);
@@ -108,10 +163,10 @@ const prohibitedWords = ref<string>('');
const hiddenTags = ref<string>('');
const preservedUsernames = ref<string>('');
const bubbleTimeline = ref<string>('');
-const tosUrl = ref<string | null>(null);
-const privacyPolicyUrl = ref<string | null>(null);
-const inquiryUrl = ref<string | null>(null);
const trustedLinkUrlPatterns = ref<string>('');
+const blockedHosts = ref<string>('');
+const silencedHosts = ref<string>('');
+const mediaSilencedHosts = ref<string>('');
async function init() {
const meta = await misskeyApi('admin/meta');
@@ -122,28 +177,105 @@ async function init() {
prohibitedWords.value = meta.prohibitedWords.join('\n');
hiddenTags.value = meta.hiddenTags.join('\n');
preservedUsernames.value = meta.preservedUsernames.join('\n');
- tosUrl.value = meta.tosUrl;
- privacyPolicyUrl.value = meta.privacyPolicyUrl;
bubbleTimeline.value = meta.bubbleInstances.join('\n');
bubbleTimelineEnabled.value = meta.policies.btlAvailable;
- inquiryUrl.value = meta.inquiryUrl;
trustedLinkUrlPatterns.value = meta.trustedLinkUrlPatterns.join('\n');
+ blockedHosts.value = meta.blockedHosts.join('\n');
+ silencedHosts.value = meta.silencedHosts.join('\n');
+ mediaSilencedHosts.value = meta.mediaSilencedHosts.join('\n');
+}
+
+function onChange_enableRegistration(value: boolean) {
+ os.apiWithDialog('admin/update-meta', {
+ disableRegistration: !value,
+ }).then(() => {
+ fetchInstance(true);
+ });
+}
+
+function onChange_emailRequiredForSignup(value: boolean) {
+ os.apiWithDialog('admin/update-meta', {
+ emailRequiredForSignup: value,
+ }).then(() => {
+ fetchInstance(true);
+ });
+}
+
+function onChange_approvalRequiredForSignup(value: boolean) {
+ os.apiWithDialog('admin/update-meta', {
+ approvalRequiredForSignup: value,
+ }).then(() => {
+ fetchInstance(true);
+ });
+}
+
+function save_bubbleTimeline() {
+ os.apiWithDialog('admin/update-meta', {
+ bubbleInstances: bubbleTimeline.value.split('\n'),
+ }).then(() => {
+ fetchInstance(true);
+ });
+}
+
+function save_trustedLinkUrlPatterns() {
+ os.apiWithDialog('admin/update-meta', {
+ trustedLinkUrlPatterns: trustedLinkUrlPatterns.value.split('\n'),
+ }).then(() => {
+ fetchInstance(true);
+ });
+}
+
+function save_preservedUsernames() {
+ os.apiWithDialog('admin/update-meta', {
+ preservedUsernames: preservedUsernames.value.split('\n'),
+ }).then(() => {
+ fetchInstance(true);
+ });
}
-function save() {
+function save_sensitiveWords() {
os.apiWithDialog('admin/update-meta', {
- disableRegistration: !enableRegistration.value,
- emailRequiredForSignup: emailRequiredForSignup.value,
- approvalRequiredForSignup: approvalRequiredForSignup.value,
- tosUrl: tosUrl.value,
- privacyPolicyUrl: privacyPolicyUrl.value,
- inquiryUrl: inquiryUrl.value,
sensitiveWords: sensitiveWords.value.split('\n'),
+ }).then(() => {
+ fetchInstance(true);
+ });
+}
+
+function save_prohibitedWords() {
+ os.apiWithDialog('admin/update-meta', {
prohibitedWords: prohibitedWords.value.split('\n'),
+ }).then(() => {
+ fetchInstance(true);
+ });
+}
+
+function save_hiddenTags() {
+ os.apiWithDialog('admin/update-meta', {
hiddenTags: hiddenTags.value.split('\n'),
- preservedUsernames: preservedUsernames.value.split('\n'),
- bubbleInstances: bubbleTimeline.value.split('\n'),
- trustedLinkUrlPatterns: trustedLinkUrlPatterns.value.split('\n'),
+ }).then(() => {
+ fetchInstance(true);
+ });
+}
+
+function save_blockedHosts() {
+ os.apiWithDialog('admin/update-meta', {
+ blockedHosts: blockedHosts.value.split('\n') || [],
+ }).then(() => {
+ fetchInstance(true);
+ });
+}
+
+function save_silencedHosts() {
+ os.apiWithDialog('admin/update-meta', {
+ silencedHosts: silencedHosts.value.split('\n') || [],
+ }).then(() => {
+ fetchInstance(true);
+ });
+}
+
+function save_mediaSilencedHosts() {
+ os.apiWithDialog('admin/update-meta', {
+ mediaSilencedHosts: mediaSilencedHosts.value.split('\n') || [],
}).then(() => {
fetchInstance(true);
});
@@ -156,10 +288,3 @@ definePageMetadata(() => ({
icon: 'ti ti-shield',
}));
</script>
-
-<style lang="scss" module>
-.footer {
- -webkit-backdrop-filter: var(--blur, blur(15px));
- backdrop-filter: var(--blur, blur(15px));
-}
-</style>