Modrinth/pages/moderation.vue
Geometrically 673f7a82d1
New features (#592)
* New features

* Lots of bug fixes

* Fix respack creation

* Improve mobile nav with more project types

* Fix resolution sorting and remove icons

* Move cookie consent to top on small devices to get out of the way of navigation

* Move cookie consent + fix hydration

* Fix project editing + update search features

* Centralize hardcoding of loader/category names, fix cookie consent shadow, fix mobile navbar rounding

* Fix plugin platforms formatting

* Kitchen sink!

* Add support for display names

* LiteLoader formatting

* Fixed "show all loaders" toggle not resetting when changing pages

* Allow multiple loaders in version filter controls

* Fix clear filters button

* Revert "Add support for display names"

This reverts commit 370838763d86bcae51bf06c304248f7a1f8fc28f.

* Let's see how this goes. Upstream filters, attempt 1

* github? hello?

* No more "Server mod" on plugins

* Fix formatting of project types in project creation

* Move where project creation sets the resource pack loader

* Allow setting pixelated image-rendering

Allows to apply 'style' attribute to IMG tags with value
'image-rendering' set to 'pixelated', which can be useful for people who
use pixel art in their READMEs (to demonstrate items, for example).

* fix user page + hydration issue fix from Brawaru

* Rename to proxies

* Make categories use title case

* Always show project type on moderation page, improve project type display on project pages

* Remove invalid key

* Missed a check

* Fix browse menu animation

* Fix disabled button condition and minimum width for 2 lines

* Body -> Description in edit pages

* More casing consistency issues

* Fix duplicate version URLs

* Fix version creation

* Edit URLs, fix privacy page buttons

* Fix notifications popup overlaying

* Final merge fixes

Co-authored-by: Prospector <prospectordev@gmail.com>
Co-authored-by: Sasha Sorokin <10401817+Brawaru@users.noreply.github.com>
2022-08-14 12:42:58 -07:00

353 lines
9.1 KiB
Vue

<template>
<div class="page-container">
<Popup v-if="currentProject" :show-popup="true">
<div class="moderation-popup">
<h2>Moderation form</h2>
<p>
Both of these fields are optional, but can be used to communicate
problems with a project's team members. The body supports markdown
formatting!
</p>
<div class="status">
<span>New project status: </span>
<Badge
v-if="currentProject.newStatus === 'approved'"
color="green"
:type="currentProject.newStatus"
/>
<Badge
v-else-if="
currentProject.newStatus === 'processing' ||
currentProject.newStatus === 'unlisted' ||
currentProject.newStatus === 'archived'
"
color="yellow"
:type="currentProject.newStatus"
/>
<Badge
v-else-if="currentProject.newStatus === 'rejected'"
color="red"
:type="currentProject.newStatus"
/>
<Badge v-else color="gray" :type="currentProject.newStatus" />
</div>
<input
v-model="currentProject.moderation_message"
type="text"
placeholder="Enter the message..."
/>
<h3>Body</h3>
<ThisOrThat v-model="bodyViewMode" :items="['source', 'preview']" />
<div v-if="bodyViewMode === 'source'" class="textarea-wrapper">
<textarea
id="body"
v-model="currentProject.moderation_message_body"
/>
</div>
<div
v-if="bodyViewMode === 'preview'"
v-highlightjs
class="markdown-body"
v-html="$xss($md.render(currentProject.moderation_message_body))"
></div>
<div class="buttons">
<button class="iconified-button" @click="currentProject = null">
<CrossIcon />
Cancel
</button>
<button
class="iconified-button brand-button-colors"
@click="saveProject"
>
<CheckIcon />
Confirm
</button>
</div>
</div>
</Popup>
<div class="page-contents">
<div class="content">
<h1>Moderation</h1>
<ThisOrThat
v-model="selectedType"
class="card"
:items="moderationTypes"
/>
<div class="projects">
<ProjectCard
v-for="project in selectedType !== 'all'
? projects.filter((x) => x.project_type === selectedType)
: projects"
:id="project.slug || project.id"
:key="project.id"
:name="project.title"
:description="project.description"
:created-at="project.published"
:updated-at="project.updated"
:icon-url="project.icon_url"
:categories="project.categories"
:client-side="project.client_side"
:server-side="project.server_side"
:type="project.project_type"
:moderation="true"
>
<button
class="iconified-button"
@click="setProjectStatus(project, 'approved')"
>
<CheckIcon />
Approve
</button>
<button
class="iconified-button"
@click="setProjectStatus(project, 'unlisted')"
>
<UnlistIcon />
Unlist
</button>
<button
class="iconified-button"
@click="setProjectStatus(project, 'rejected')"
>
<CrossIcon />
Reject
</button>
</ProjectCard>
</div>
<div
v-if="selectedType === 'report' || selectedType === 'all'"
class="reports"
>
<div
v-for="(report, index) in reports"
:key="report.id"
class="report card"
>
<div class="header">
<h5 class="title">
Report for {{ report.item_type }}
<nuxt-link
:to="
'/' +
report.item_type +
'/' +
report.item_id.replace(/\W/g, '')
"
>{{ report.item_id }}
</nuxt-link>
</h5>
<p
v-tooltip="
$dayjs(report.created).format(
'[Created at] YYYY-MM-DD [at] HH:mm A'
)
"
class="date"
>
Created {{ $dayjs(report.created).fromNow() }}
</p>
<button
class="delete iconified-button"
@click="deleteReport(index)"
>
Delete
</button>
</div>
<div
v-highlightjs
class="markdown-body"
v-html="$xss($md.render(report.body))"
></div>
</div>
</div>
<div v-if="reports.length === 0 && projects.length === 0" class="error">
<Security class="icon"></Security>
<br />
<span class="text">You are up-to-date!</span>
</div>
</div>
</div>
</div>
</template>
<script>
import ThisOrThat from '~/components/ui/ThisOrThat'
import ProjectCard from '~/components/ui/ProjectCard'
import Popup from '~/components/ui/Popup'
import Badge from '~/components/ui/Badge'
import CheckIcon from '~/assets/images/utils/check.svg?inline'
import UnlistIcon from '~/assets/images/utils/eye-off.svg?inline'
import CrossIcon from '~/assets/images/utils/x.svg?inline'
import Security from '~/assets/images/illustrations/security.svg?inline'
export default {
name: 'Moderation',
components: {
ThisOrThat,
ProjectCard,
CheckIcon,
CrossIcon,
UnlistIcon,
Popup,
Badge,
Security,
},
async asyncData(data) {
const [projects, reports] = (
await Promise.all([
data.$axios.get(`moderation/projects`, data.$defaultHeaders()),
data.$axios.get(`report`, data.$defaultHeaders()),
])
).map((it) => it.data)
return {
projects,
reports,
selectedType: 'all',
}
},
data() {
return {
bodyViewMode: 'source',
currentProject: null,
}
},
head: {
title: 'Moderation - Modrinth',
},
computed: {
moderationTypes() {
const obj = { all: true }
for (const project of this.projects) {
obj[project.project_type] = true
}
if (this.reports.length > 0) {
obj.report = true
}
return Object.keys(obj)
},
},
methods: {
setProjectStatus(project, status) {
project.moderation_message = ''
project.moderation_message_body = ''
project.newStatus = status
this.currentProject = project
},
async saveProject() {
this.$nuxt.$loading.start()
try {
await this.$axios.patch(
`project/${this.currentProject.id}`,
{
moderation_message: this.currentProject.moderation_message
? this.currentProject.moderation_message
: null,
moderation_message_body: this.currentProject.moderation_message_body
? this.currentProject.moderation_message_body
: null,
status: this.currentProject.newStatus,
},
this.$defaultHeaders()
)
this.projects.splice(
this.projects.findIndex((x) => this.currentProject.id === x.id),
1
)
this.currentProject = null
} catch (err) {
this.$notify({
group: 'main',
title: 'An error occurred',
text: err.response.data.description,
type: 'error',
})
}
this.$nuxt.$loading.finish()
},
async deleteReport(index) {
this.$nuxt.$loading.start()
try {
await this.$axios.delete(
`report/${this.reports[index].id}`,
this.$defaultHeaders()
)
this.reports.splice(index, 1)
} catch (err) {
this.$notify({
group: 'main',
title: 'An error occurred',
text: err.response.data.description,
type: 'error',
})
}
this.$nuxt.$loading.finish()
},
},
}
</script>
<style lang="scss" scoped>
.moderation-popup {
width: auto;
padding: var(--spacing-card-md) var(--spacing-card-lg);
.status {
display: flex;
align-items: center;
margin-bottom: 0.5rem;
span {
margin-right: 0.5rem;
}
}
.buttons {
display: flex;
margin-top: 0.5rem;
:first-child {
margin-left: auto;
}
}
}
h1 {
color: var(--color-text-dark);
margin: 0 0 1rem 1.5rem;
}
.report {
.header {
display: flex;
align-items: center;
flex-direction: row;
justify-content: center;
.title {
font-size: var(--font-size-lg);
margin: 0 0.5rem 0 0;
}
.iconified-button {
margin-left: auto;
}
}
}
@media screen and (min-width: 1024px) {
.page-contents {
max-width: calc(1280px - 20rem) !important;
}
}
</style>