릴리스: v0.1.46 에디터 아이템 제거와 회원 정보 보강

This commit is contained in:
2026-03-27 10:54:31 +09:00
parent 282e89b738
commit e0eeaa01cd
7 changed files with 158 additions and 9 deletions

View File

@@ -706,6 +706,18 @@ function fmt(ts) {
})
}
function userAvatarUrl(user) {
return user?.avatarSrc ? toApiUrl(user.avatarSrc) : ''
}
function userDisplayName(user) {
return user?.nickname || user?.email?.split('@')[0] || '알 수 없음'
}
function userAvatarFallback(user) {
return (user?.email?.trim()?.[0] || '?').toUpperCase()
}
function addFeaturedGame(gameId) {
resetMessages()
if (!gameId || featuredGameIds.value.includes(gameId)) return
@@ -1120,15 +1132,32 @@ async function saveFeaturedOrder() {
<div v-else class="userList">
<article v-for="user in users" :key="user.id" class="userCard">
<div class="userCard__head">
<div>
<div class="userCard__title">{{ user.nickname || '닉네임 없음' }}</div>
<div class="userCard__meta">{{ fmt(user.createdAt) }}</div>
<div class="userCard__identity">
<div class="userAvatar">
<img v-if="userAvatarUrl(user)" class="userAvatar__image" :src="userAvatarUrl(user)" :alt="userDisplayName(user)" />
<span v-else class="userAvatar__fallback">{{ userAvatarFallback(user) }}</span>
</div>
<div>
<div class="userCard__title">{{ userDisplayName(user) }}</div>
<div class="userCard__meta">가입일 {{ fmt(user.createdAt) }}</div>
</div>
</div>
<span class="roleBadge" :class="{ 'roleBadge--admin': user.draftIsAdmin }">
{{ user.draftIsAdmin ? '관리자' : '일반 회원' }}
</span>
</div>
<div class="userStats">
<div class="userStat">
<span class="userStat__label">작성 티어표</span>
<strong class="userStat__value">{{ user.tierListCount }}</strong>
</div>
<div class="userStat">
<span class="userStat__label">최근 활동</span>
<strong class="userStat__value">{{ fmt(user.recentActivityAt || user.createdAt) }}</strong>
</div>
</div>
<input v-model="user.draftEmail" class="input" placeholder="이메일" />
<input v-model="user.draftNickname" class="input" placeholder="닉네임" />
<label class="checkRow">
@@ -1679,6 +1708,12 @@ async function saveFeaturedOrder() {
justify-content: space-between;
align-items: flex-start;
}
.userCard__identity {
display: flex;
gap: 12px;
align-items: center;
min-width: 0;
}
.userCard__title {
font-weight: 900;
}
@@ -1687,6 +1722,48 @@ async function saveFeaturedOrder() {
opacity: 0.72;
font-size: 13px;
}
.userAvatar {
width: 48px;
height: 48px;
flex: 0 0 auto;
display: grid;
place-items: center;
border-radius: 999px;
overflow: hidden;
border: 1px solid rgba(255, 255, 255, 0.14);
background: rgba(96, 165, 250, 0.18);
}
.userAvatar__image {
width: 100%;
height: 100%;
object-fit: cover;
}
.userAvatar__fallback {
font-size: 18px;
font-weight: 900;
}
.userStats {
margin-top: 12px;
display: grid;
grid-template-columns: repeat(2, minmax(0, 1fr));
gap: 10px;
}
.userStat {
display: grid;
gap: 4px;
padding: 10px 12px;
border-radius: 12px;
border: 1px solid rgba(255, 255, 255, 0.08);
background: rgba(255, 255, 255, 0.03);
}
.userStat__label {
font-size: 12px;
opacity: 0.66;
}
.userStat__value {
font-size: 14px;
font-weight: 900;
}
.userCard__actions {
display: grid;
grid-template-columns: repeat(2, minmax(0, 1fr));
@@ -1882,7 +1959,8 @@ async function saveFeaturedOrder() {
.section--topGrid,
.toolbar,
.itemComposer,
.tierAdminCard {
.tierAdminCard,
.userStats {
grid-template-columns: 1fr;
}
.toolbar--secondary {
@@ -1891,6 +1969,9 @@ async function saveFeaturedOrder() {
.itemPreviewCard {
max-width: none;
}
.userCard__identity {
width: 100%;
}
}
@media (max-width: 640px) {
.thumbGrid,