Modrinth/components/ui/ProjectCard.vue
Redblueflame a2266adb3f
Add initial support for the v2 of the API (Still WIP) (#250)
* Functionally implement modpacks

* Add LogoAnimated to logo license

* Fix eslint errors

* Add `z-index: 20` to user dropdown (#287)

* Fix pages not working, add changelog page, redesign versions page

* Update theme colors, add OLED theme, update some project creation text. (#292)

* Update theme colors, add OLED theme, update some project creation text.

* Make summary normal text color

* Update favicons, update logos to use dynamic colors, updated filters panel a bit

* Update wording from #250

* Version page rework

* Manually apply some commits from master, other minor v2 fixes (#296)

* Homepage styling improvements (#285)

* Add border radius to video + example code colors

* Change color + allow overflow scroll

* Minor v2 fixes

- Makes multiple loaders display correctly (used to be `Fabric,Forge` is now `Fabric, Forge`
- Fix oopses in #292
- Allow .jar and .zip in file prompt
- Apply 30cbd3a6c372940d1e86cc8134d0dfc7e8e5ee9c to pages/create/project.vue
- Display `fabric, forge` instead of broken icons on pages/create/project.vue

* Markdown styling fixes (#268)

* Add table color variables (+ prettier fixes)

* Add details and table styling to .markdown-body

* Add indexing meta value depending on the status of the mod. (#261)

* General UI Improvement (again) (#255)

* Add and fix some stuff

* Add warning when leaving to `mod/create`

* Fix mods/create not working

* Fix a bug & add improvements to a couple moderation aspects (#278)

This PR fixes reports on the moderation dashboard going to `/dashboard/mod/_id` instead of to `/mod/_id`.
It also allows the ability for moderators to unlist mods in the queue from the frontend instead of having to do it via the backend.
![image](https://i.imgur.com/x8shSVn.png)
Unlisted mods should have the ability to resubmit for approval, so I've also changed "Submit for Review" to "Submit for approval", allowing unlisted mods to do that as well.
![image](https://i.imgur.com/OC8Vyfo.png)

* Add project guidelines to Terms page (#275)

* Add project guidelines to Terms page

This adds the project guidelines as outlined [here](https://discord.com/channels/734077874708938864/734077874708938867/806556531491471368).
NOTE: I've made a few tweaks in wording to accommodate this format, so this is not an exact copy.

* Move rules to its own page

* Allow users to login from search page when it is rendered serverside (#272)

* Change `this.$route.fullPath` → `this.$route.path`

* Closes modrinth/knossos#256

* Wrap mod icon and title in link (#273)

* Wrap mod icon and title in link

* Fixes #218

* Editor's note

    Skipped #249 (search was rewritten), #266 (couldn't figure out how to apply it), #270 (didn't seem to apply properly), #252 (manually merged in with #292), #262 (superceded by #270), #282, #271, #277, #283, and #281 (those five didn't get wiped)

Co-authored-by: venashial <venashial.levo@aleeas.com>
Co-authored-by: Redblueflame <contact@redblueflame.com>
Co-authored-by: Johan Novak <wickedtree@wickedtree.codes>

* SSR descriptions, version edit page

* Working version editing + dependency management (besides files)

* Version create page, file functionality

* Fix some issues with the version page

* More versions page fixes

* Project gallery

* Box shadows, user profile page, WIP header

* Finish user dashboard

* Finish search and fix minor issues

* Moderator page + messages, notifications page

* Fix dropdown menu, fix XSS, fix team members page

* Change doc url on main page (#309)

* Re-Fix docs url (#313)

* Clean up. Part 1: Fix immediate problems (#316)

* Clean up tabs and cards CSS a little

* Fix project page; Remove bad styles from search

* Yeet and flatten lots of styles; fix font sizes

* Restyle search; fix moderation

* Fix profile page

* Remove injected SCSS entirely

* Fix a mobile layout overflowing

* Apiv2-support fixes (#320)

* Fix member user_id -> user.id

* Fix incorrect report redirect

* Change theme switcher from button to multiselect

* Fix remaining items

Co-authored-by: Jai A <jaiagr+gpg@pm.me>

* Fix bugs

* Full mobile support, update create project page, fix various bugs

* New Dark Mode brand colors (#325)

* Use "color-brand-hover" for auth-prompt when hover over

* New dark mode brand colors

* Fix new version featured bug

* Remove old home page, other fixes

* Fix error when merging

* Fix prettier error :(

Co-authored-by: Jai A <jaiagr+gpg@pm.me>
Co-authored-by: venashial <venashial.levo@aleeas.com>
Co-authored-by: Prospector <6166773+Prospector@users.noreply.github.com>
Co-authored-by: Emma <emmaffle@modrinth.com>
Co-authored-by: Johan Novak <wickedtree@wickedtree.codes>
Co-authored-by: Jai A <jaiagr@pm.me>
Co-authored-by: Mysterious_Dev <40738104+Mysterious-Dev@users.noreply.github.com>
Co-authored-by: Mikhail Oleynikov <contact@falseresync.ru>
Co-authored-by: Christian Popov <30723811+Xrey274@users.noreply.github.com>
2022-01-09 15:19:27 -07:00

389 lines
8.4 KiB
Vue

<template>
<article class="project-card card">
<div class="columns">
<div class="icon">
<nuxt-link :to="`/${type}/${id}`">
<img
:src="iconUrl || 'https://cdn.modrinth.com/placeholder.svg?inline'"
:alt="name"
loading="lazy"
/>
</nuxt-link>
<Categories :categories="categories" class="left-categories" />
</div>
<div class="card-content">
<div class="info">
<div class="top">
<h2 class="title">
<nuxt-link :to="`/${type}/${id}`">{{ name }}</nuxt-link>
</h2>
<p v-if="author" class="author">
by <nuxt-link :to="'/user/' + author">{{ author }}</nuxt-link>
</p>
</div>
<div class="side-type">
<div
v-if="clientSide === 'optional' && serverSide === 'optional'"
class="side-descriptor"
>
<InfoIcon />
Universal {{ type }}
</div>
<div
v-else-if="
(clientSide === 'optional' || clientSide === 'required') &&
(serverSide === 'optional' || serverSide === 'unsupported')
"
class="side-descriptor"
>
<InfoIcon />
Client {{ type }}
</div>
<div
v-else-if="
(serverSide === 'optional' || serverSide === 'required') &&
(clientSide === 'optional' || clientSide === 'unsupported')
"
class="side-descriptor"
>
<InfoIcon />
Server {{ type }}
</div>
</div>
<p class="description">
{{ description }}
</p>
<Categories :categories="categories" class="right-categories" />
<div class="dates">
<div class="date">
<CalendarIcon />
Created {{ $dayjs(createdAt).fromNow() }}
</div>
<div class="date">
<EditIcon />
Updated {{ $dayjs(updatedAt).fromNow() }}
</div>
</div>
</div>
<div class="right-side">
<div v-if="downloads" class="stat">
<DownloadIcon />
<p>
<strong>{{ formatNumber(downloads) }}</strong> downloads
</p>
</div>
<div v-if="follows" class="stat">
<HeartIcon />
<p>
<strong>{{ formatNumber(follows) }}</strong> followers
</p>
</div>
<div v-if="status" class="status">
<Badge
v-if="status === 'approved'"
color="green custom-circle"
:type="status"
/>
<Badge
v-else-if="status === 'processing' || status === 'archived'"
color="yellow custom-circle"
:type="status"
/>
<Badge
v-else-if="status === 'rejected'"
color="red custom-circle"
:type="status"
/>
<Badge v-else color="gray custom-circle" :type="status" />
</div>
<div class="buttons">
<slot />
</div>
</div>
</div>
</div>
</article>
</template>
<script>
import Categories from '~/components/ui/search/Categories'
import Badge from '~/components/ui/Badge'
import InfoIcon from '~/assets/images/utils/info.svg?inline'
import CalendarIcon from '~/assets/images/utils/calendar.svg?inline'
import EditIcon from '~/assets/images/utils/updated.svg?inline'
import DownloadIcon from '~/assets/images/utils/download-alt.svg?inline'
import HeartIcon from '~/assets/images/utils/heart.svg?inline'
export default {
name: 'ProjectCard',
components: {
Categories,
Badge,
InfoIcon,
CalendarIcon,
EditIcon,
DownloadIcon,
HeartIcon,
},
props: {
id: {
type: String,
default: 'modrinth-0',
},
type: {
type: String,
default: 'mod',
},
name: {
type: String,
default: 'Project Name',
},
author: {
type: String,
default: null,
},
description: {
type: String,
default: 'A _type description',
},
iconUrl: {
type: String,
default: '#',
required: false,
},
downloads: {
type: String,
default: null,
required: false,
},
follows: {
type: String,
default: null,
required: false,
},
createdAt: {
type: String,
default: '0000-00-00',
},
updatedAt: {
type: String,
default: null,
},
categories: {
type: Array,
default() {
return []
},
},
status: {
type: String,
default: null,
},
serverSide: {
type: String,
required: false,
default: '',
},
clientSide: {
type: String,
required: false,
default: '',
},
},
methods: {
formatNumber(x) {
return x.toString().replace(/\B(?=(\d{3})+(?!\d))/g, ',')
},
},
}
</script>
<style lang="scss" scoped>
.columns {
width: 100%;
}
.project-card {
display: flex;
flex-direction: row;
flex-direction: column;
padding: var(--spacing-card-bg);
width: calc(100% - 2 * var(--spacing-card-bg));
@media screen and (min-width: 1024px) {
flex-direction: row;
justify-content: space-between;
}
.icon {
img {
width: 6rem;
height: 6rem;
margin: 0 var(--spacing-card-md) var(--spacing-card-md) 0;
border-radius: var(--size-rounded-icon);
object-fit: contain;
}
}
.card-content {
display: flex;
justify-content: space-between;
flex-grow: 1;
.info {
display: flex;
flex-direction: column;
.top {
align-items: baseline;
display: flex;
flex-direction: row;
flex-wrap: wrap;
flex-shrink: 0;
margin-right: var(--spacing-card-md);
.title {
margin: 0 0.5rem 0 0;
color: var(--color-text-dark);
font-size: var(--font-size-xl);
}
.author {
margin: auto 0 0 0;
color: var(--color-text);
a {
text-decoration: underline;
}
}
}
.side-descriptor {
display: flex;
align-items: center;
font-weight: bolder;
font-size: var(--font-size-sm);
margin-top: 0.125rem;
margin-bottom: 0.5rem;
svg {
width: auto;
height: 1rem;
margin-right: 0.125rem;
}
}
.description {
margin: var(--spacing-card-sm) var(--spacing-card-md)
var(--spacing-card-sm) 0;
}
.right-categories {
margin-bottom: var(--spacing-card-sm);
}
.dates {
display: flex;
flex-wrap: wrap;
.date {
display: flex;
align-items: center;
margin-right: 2rem;
svg {
height: 1.25rem;
margin-right: 0.125rem;
}
}
}
}
.right-side {
min-width: 8.75rem;
text-align: right;
.stat {
display: flex;
align-items: center;
margin-bottom: 0.5rem;
svg {
width: auto;
height: 1.25rem;
margin-left: auto;
margin-right: 0.25rem;
}
p {
margin: 0;
strong {
font-weight: bolder;
font-size: var(--font-size-lg);
}
}
}
.status {
margin-bottom: 0.5rem;
}
.buttons {
display: flex;
flex-direction: column;
button,
a {
margin-right: 0;
margin-left: auto;
margin-bottom: 0.5rem;
}
}
}
}
.left-categories {
display: none;
}
@media screen and (max-width: 560px) {
.card-content {
flex-direction: column;
.right-side {
padding-top: var(--spacing-card-sm);
text-align: left;
.stat svg {
margin-left: 0;
}
.buttons {
flex-direction: row;
}
.buttons button,
a {
margin-left: unset;
margin-right: unset;
}
}
}
.left-categories {
display: flex;
margin: 0 0 0.75rem 0.75rem;
width: 7rem;
}
.right-categories {
display: none;
}
}
}
</style>