Compare commits
6 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| a6e78b29f1 | |||
| 2346b5fbe3 | |||
| d724a64451 | |||
| 781a131ade | |||
| 6fceeaf15b | |||
| 7886b98380 |
@@ -1,5 +1,35 @@
|
|||||||
# 의사결정 이력
|
# 의사결정 이력
|
||||||
|
|
||||||
|
## 2026-03-30 v1.2.10
|
||||||
|
- 목록 화면도 결국 같은 제품의 라이브러리 레이어이므로, 상단 통계 카드와 버튼의 높이·반경·배경을 공통 셸과 같은 문법으로 맞추는 편이 일관성이 높다고 정리했다.
|
||||||
|
- 홈 화면의 빠른 액션은 중복 의미 버튼보다 `즐겨찾기 / 내 리스트 / 커스텀 시작`처럼 실제 이동 동선이 분명한 버튼 구성이 더 적합하다고 판단했다.
|
||||||
|
- 카드 hover 반응은 화면마다 조금씩 다르게 두기보다, 모두 얕은 위로 이동과 배경 변화로 통일하는 편이 대시보드 감도를 유지하기 쉽다고 결정했다.
|
||||||
|
|
||||||
|
## 2026-03-30 v1.2.9
|
||||||
|
- 관리자 화면은 기능보다 먼저 정보 계층이 읽혀야 하므로, 현재 탭에 맞는 요약 통계를 헤더에서 먼저 보여주는 편이 운영 판단에 더 유리하다고 정리했다.
|
||||||
|
- 게임/아이템/티어표/회원 카드는 기능이 다른 대신 같은 제품 안에 있으므로, 배경층·반경·패딩은 하나의 대시보드 문법으로 맞춰 시안 톤을 더 강하게 유지하기로 결정했다.
|
||||||
|
- 우측 운영 패널은 단순 필터 모음보다 “현재 상태를 짧게 읽고 바로 액션하는 패널”에 가까워야 하므로, 입력과 통계 카드를 더 단단한 카드형 레이어로 정리하는 편이 맞다고 판단했다.
|
||||||
|
|
||||||
|
## 2026-03-30 v1.2.8
|
||||||
|
- 에디터는 “보드 편집”과 “옵션 편집”의 역할이 다르므로, 보드 옆에는 드래그용 아이템 풀을 두고 제목/설명/썸네일/저장 같은 설정은 최우측 사이드바에만 남기는 편이 맞다고 판단했다.
|
||||||
|
- 커스텀 아이템 이름 정리는 배치 중에 계속 보는 정보보다 저장 전 정리용 정보에 가까우므로, 아이템 풀 아래보다 우측 편집 패널 내부가 더 적합하다고 정리했다.
|
||||||
|
- 실제 SVG 에셋이 들어오기 시작한 만큼, 공통 셸은 새 아이콘을 우선 적용하고 나머지는 점진적으로 교체하는 방식이 안전하다고 판단했다.
|
||||||
|
|
||||||
|
## 2026-03-30 v1.2.7
|
||||||
|
- 피그마 감도는 개별 화면보다 공통 셸의 밀도와 아이콘 체계가 먼저 맞아야 하므로, 좌측/우측 레일을 먼저 아이콘형 카드 문법으로 정리하기로 했다.
|
||||||
|
- 실제 머티리얼 SVG 자산을 받기 전까지는 간단한 선형 SVG 아이콘으로 정보 구조를 먼저 맞추고, 이후 에셋 교체만으로 다듬을 수 있게 하는 편이 안전하다고 판단했다.
|
||||||
|
- 에디터는 기능은 이미 많은 상태이므로 구조를 더 바꾸기보다 보드, 툴바, 우측 편집 패널의 카드 톤을 공통 셸과 맞추는 방식으로 단계적으로 다듬기로 했다.
|
||||||
|
|
||||||
|
## 2026-03-30 v1.2.6
|
||||||
|
- 홈, 게임 허브, 내 티어표, 즐겨찾기처럼 카드 중심 화면은 한 번에 같은 카드 문법으로 맞춰야 전체 앱이 하나의 제품처럼 보이므로, 목록 화면을 우선 통일하기로 했다.
|
||||||
|
- 홈 화면은 단순 게임 버튼 모음보다 상태 카드와 CTA가 있는 라이브러리 대시보드 쪽이 피그마 톤에 더 가깝다고 판단했다.
|
||||||
|
- 게임 허브와 개인 목록도 썸네일/작성자/메타의 비중이 비슷하므로, 화면마다 다른 카드 구조를 유지하기보다 동일한 정보 계층을 반복하는 편이 더 읽기 쉽다고 정리했다.
|
||||||
|
|
||||||
|
## 2026-03-30 v1.2.5
|
||||||
|
- 관리자 화면도 에디터와 마찬가지로 공통 우측 패널보다 전용 로컬 운영 패널이 더 중요하므로, `/admin` 역시 화면 내부 `320px` 패널을 사용하는 포커스 화면으로 정리하기로 결정했다.
|
||||||
|
- 관리자 기능은 탭, 검색, 필터, 빠른 액션이 본문에 섞이면 밀도가 너무 높아지므로, 우측 패널에는 제어 요소를 모으고 중앙에는 실제 관리 대상 목록과 상세만 남기는 편이 낫다고 판단했다.
|
||||||
|
- 새 셸 단계에서는 기능을 줄이기보다 위치를 재배치하는 것이 안전하므로, 기존 게임/아이템/티어표/회원 관리 로직은 유지한 채 정보 구조만 피그마 방향으로 옮기기로 했다.
|
||||||
|
|
||||||
## 2026-03-30 v1.2.4
|
## 2026-03-30 v1.2.4
|
||||||
- 로그인 유도는 좌측 하단의 단일 버튼이면 충분하므로, 비로그인 상태에서 사이드 상단에 별도 안내 카드를 또 보여주는 구조는 제거하는 편이 더 깔끔하다고 판단했다.
|
- 로그인 유도는 좌측 하단의 단일 버튼이면 충분하므로, 비로그인 상태에서 사이드 상단에 별도 안내 카드를 또 보여주는 구조는 제거하는 편이 더 깔끔하다고 판단했다.
|
||||||
- 티어표 편집 화면은 공통 우측 패널의 generic 문맥 카드보다 실제 편집 필드가 우측에 있는 편이 훨씬 중요하므로, 이 화면은 전용 로컬 우측 패널을 두는 쪽으로 정리했다.
|
- 티어표 편집 화면은 공통 우측 패널의 generic 문맥 카드보다 실제 편집 필드가 우측에 있는 편이 훨씬 중요하므로, 이 화면은 전용 로컬 우측 패널을 두는 쪽으로 정리했다.
|
||||||
|
|||||||
14
docs/map.md
14
docs/map.md
@@ -2,17 +2,17 @@
|
|||||||
|
|
||||||
## `/`
|
## `/`
|
||||||
- 화면 파일: `frontend/src/views/HomeView.vue`
|
- 화면 파일: `frontend/src/views/HomeView.vue`
|
||||||
- 역할: 게임 목록 표시, 게임 카드 클릭 이동, `직접 티어표 만들기` 진입
|
- 역할: 상단 상태/CTA가 있는 라이브러리 대시보드, 게임 카드 클릭 이동, `직접 티어표 만들기` 진입
|
||||||
- 연동 API: `GET /api/games`
|
- 연동 API: `GET /api/games`
|
||||||
|
|
||||||
## `/games/:gameId`
|
## `/games/:gameId`
|
||||||
- 화면 파일: `frontend/src/views/GameHubView.vue`
|
- 화면 파일: `frontend/src/views/GameHubView.vue`
|
||||||
- 역할: 선택한 게임 정보 표시, 공개 티어표 목록 표시, 제목/작성자 검색, 티어표별 상단 썸네일/작성자 표시, 즐겨찾기 토글, 새 티어표 작성 진입
|
- 역할: 선택한 게임 정보 표시, 상단 통계/생성 CTA, 공개 티어표 목록 표시, 제목/작성자 검색, 티어표별 상단 썸네일/작성자 표시, 즐겨찾기 상태 표시, 새 티어표 작성 진입
|
||||||
- 연동 API: `GET /api/games/:gameId`, `GET /api/tierlists/public`, `POST /api/tierlists/:id/favorite`, `DELETE /api/tierlists/:id/favorite`
|
- 연동 API: `GET /api/games/:gameId`, `GET /api/tierlists/public`, `POST /api/tierlists/:id/favorite`, `DELETE /api/tierlists/:id/favorite`
|
||||||
|
|
||||||
## `/editor/:gameId/new`, `/editor/:gameId/:tierListId`
|
## `/editor/:gameId/new`, `/editor/:gameId/:tierListId`
|
||||||
- 화면 파일: `frontend/src/views/TierEditorView.vue`
|
- 화면 파일: `frontend/src/views/TierEditorView.vue`
|
||||||
- 역할: 티어 그룹 편집, 티어 행 추가/삭제, 관리자 아이템/커스텀 아이템 다중 드래그 앤 드롭 업로드, 우측 전용 편집 패널에서 티어표 제목/설명/대표 썸네일/공개 여부/저장 제어, 즐겨찾기 토글, PNG 다운로드
|
- 역할: 티어 그룹 편집, 티어 행 추가/삭제, 보드 옆 아이템 풀에서 관리자 아이템/커스텀 아이템 다중 드래그 앤 드롭 업로드, 우측 전용 편집 패널에서 티어표 제목/설명/대표 썸네일/공개 여부/저장 제어와 커스텀 아이템 이름 정리, 즐겨찾기 토글, PNG 다운로드
|
||||||
- 연동 API: `GET /api/games/:gameId`, `GET /api/tierlists/:id`, `POST /api/tierlists/:id/favorite`, `DELETE /api/tierlists/:id/favorite`, `POST /api/tierlists/thumbnail`, `POST /api/tierlists/custom-items`, `POST /api/tierlists`
|
- 연동 API: `GET /api/games/:gameId`, `GET /api/tierlists/:id`, `POST /api/tierlists/:id/favorite`, `DELETE /api/tierlists/:id/favorite`, `POST /api/tierlists/thumbnail`, `POST /api/tierlists/custom-items`, `POST /api/tierlists`
|
||||||
|
|
||||||
## `/login`
|
## `/login`
|
||||||
@@ -22,17 +22,17 @@
|
|||||||
|
|
||||||
## `/me`
|
## `/me`
|
||||||
- 화면 파일: `frontend/src/views/MyTierListsView.vue`
|
- 화면 파일: `frontend/src/views/MyTierListsView.vue`
|
||||||
- 역할: 내 티어표 목록 조회, 상단 썸네일 카드 표시, 편집 화면으로 이동, 작성자 본인 티어표 삭제
|
- 역할: 내 티어표 목록 조회, 라이브러리 카드형 썸네일 표시, 편집 화면으로 이동, 작성자 본인 티어표 삭제
|
||||||
- 연동 API: `GET /api/tierlists/me`, `DELETE /api/tierlists/:id`
|
- 연동 API: `GET /api/tierlists/me`, `DELETE /api/tierlists/:id`
|
||||||
|
|
||||||
## `/favorites`
|
## `/favorites`
|
||||||
- 화면 파일: `frontend/src/views/FavoriteTierListsView.vue`
|
- 화면 파일: `frontend/src/views/FavoriteTierListsView.vue`
|
||||||
- 역할: 즐겨찾기한 티어표 목록 조회, 검색/정렬, 편집 화면 이동, 즐겨찾기 해제
|
- 역할: 즐겨찾기한 티어표 목록 조회, 검색/정렬, 라이브러리 카드형 표시, 편집 화면 이동, 즐겨찾기 상태 확인
|
||||||
- 연동 API: `GET /api/tierlists/favorites/me`, `DELETE /api/tierlists/:id/favorite`
|
- 연동 API: `GET /api/tierlists/favorites/me`, `DELETE /api/tierlists/:id/favorite`
|
||||||
|
|
||||||
## `/admin`
|
## `/admin`
|
||||||
- 화면 파일: `frontend/src/views/AdminView.vue`
|
- 화면 파일: `frontend/src/views/AdminView.vue`
|
||||||
- 역할: `게임 관리 / 아이템 관리 / 티어표 관리 / 회원 관리` 탭 분리, 선택된 게임의 썸네일 관리, 기본 아이템 다중 드래그 앤 드롭 업로드, 기본 아이템 이름 수정, 사용자 커스텀 아이템 검색/페이지네이션/사용 횟수 확인/미사용 이미지 개별·일괄 삭제, 사용자 커스텀 아이템의 기본 템플릿 승격, 전체 티어표 검색/페이지네이션/공개 여부 확인/완성본 이동, 티어표의 추가 커스텀 아이템을 모달 기반으로 기존 템플릿 또는 새 템플릿에 가져오기, freeform 티어표의 게임 템플릿화, 사용자 템플릿 등록/업데이트 요청 승인·반려, 회원 비밀번호 초기화 포함 회원 관리, 파일 입력 초기화, 아이템 삭제, 게임 삭제
|
- 역할: 공통 우측 패널 대신 전용 `320px` 운영 패널을 사용해 `게임 관리 / 아이템 관리 / 티어표 관리 / 회원 관리` 탭과 검색·필터·빠른 작업을 우측에 배치하고, 중앙에는 선택된 게임 상세, 커스텀 아이템 목록, 템플릿 요청/전체 티어표, 회원 카드 등 실제 관리 대상만 표시, 기본 아이템 다중 드래그 앤 드롭 업로드, 기본 아이템 이름 수정, 사용자 커스텀 아이템 검색/페이지네이션/사용 횟수 확인/미사용 이미지 개별·일괄 삭제, 사용자 커스텀 아이템의 기본 템플릿 승격, 전체 티어표 검색/페이지네이션/공개 여부 확인/완성본 이동, 티어표의 추가 커스텀 아이템을 모달 기반으로 기존 템플릿 또는 새 템플릿에 가져오기, freeform 티어표의 게임 템플릿화, 사용자 템플릿 등록/업데이트 요청 승인·반려, 회원 비밀번호 초기화 포함 회원 관리, 파일 입력 초기화, 아이템 삭제, 게임 삭제
|
||||||
- 연동 API: `POST /api/admin/games`, `POST /api/admin/games/:gameId/thumbnail`, `POST /api/admin/games/:gameId/images`, `PATCH /api/admin/games/:gameId/items/:itemId`, `GET /api/admin/custom-items`, `POST /api/admin/custom-items/:itemId/promote`, `DELETE /api/admin/custom-items/:itemId`, `DELETE /api/admin/custom-items`, `GET /api/admin/tierlists`, `GET /api/admin/template-requests`, `POST /api/admin/template-requests/:requestId/approve`, `POST /api/admin/template-requests/:requestId/reject`, `POST /api/admin/tierlists/:tierListId/promote-items`, `POST /api/admin/tierlists/:tierListId/create-game-template`, `GET /api/admin/users`, `PATCH /api/admin/users/:userId`, `PATCH /api/admin/users/:userId/password`, `DELETE /api/admin/users/:userId`, `DELETE /api/admin/games/:gameId/items/:itemId`, `DELETE /api/admin/games/:gameId`
|
- 연동 API: `POST /api/admin/games`, `POST /api/admin/games/:gameId/thumbnail`, `POST /api/admin/games/:gameId/images`, `PATCH /api/admin/games/:gameId/items/:itemId`, `GET /api/admin/custom-items`, `POST /api/admin/custom-items/:itemId/promote`, `DELETE /api/admin/custom-items/:itemId`, `DELETE /api/admin/custom-items`, `GET /api/admin/tierlists`, `GET /api/admin/template-requests`, `POST /api/admin/template-requests/:requestId/approve`, `POST /api/admin/template-requests/:requestId/reject`, `POST /api/admin/tierlists/:tierListId/promote-items`, `POST /api/admin/tierlists/:tierListId/create-game-template`, `GET /api/admin/users`, `PATCH /api/admin/users/:userId`, `PATCH /api/admin/users/:userId/password`, `DELETE /api/admin/users/:userId`, `DELETE /api/admin/games/:gameId/items/:itemId`, `DELETE /api/admin/games/:gameId`
|
||||||
|
|
||||||
## `/profile`
|
## `/profile`
|
||||||
@@ -42,7 +42,7 @@
|
|||||||
|
|
||||||
## 공통 레이아웃
|
## 공통 레이아웃
|
||||||
- 앱 셸 파일: `frontend/src/App.vue`
|
- 앱 셸 파일: `frontend/src/App.vue`
|
||||||
- 역할: 좌측 내비게이션, 중앙 워크스페이스, 우측 컨텍스트 패널로 구성된 공통 앱 셸 렌더링, 로그인 상태 반영, 사용자 메뉴, 관리자 메뉴 노출 제어, 전역 우측 상단 토스트 렌더링
|
- 역할: 좌측 내비게이션, 중앙 워크스페이스, 우측 컨텍스트 패널로 구성된 공통 앱 셸 렌더링, 로그인 상태 반영, 사용자 메뉴, 관리자 메뉴 노출 제어, 일부 실제 SVG 에셋과 선형 SVG 아이콘이 혼합된 레일 UI, 전역 우측 상단 토스트 렌더링
|
||||||
- 세부: 좌측 패널은 `248px`, 우측 패널은 `320px` 기준 폭을 사용하며, 상단 토글 버튼으로 우측 패널을 접고 펼칠 수 있다.
|
- 세부: 좌측 패널은 `248px`, 우측 패널은 `320px` 기준 폭을 사용하며, 상단 토글 버튼으로 우측 패널을 접고 펼칠 수 있다.
|
||||||
|
|
||||||
## 백엔드 진입점
|
## 백엔드 진입점
|
||||||
|
|||||||
@@ -12,6 +12,7 @@
|
|||||||
- 프런트 앱 셸은 `좌측 내비게이션 / 중앙 워크스페이스 / 우측 컨텍스트 패널` 3단 구조로 재정의되었고, preview 모드에서는 이 셸을 숨기고 콘텐츠만 렌더링한다.
|
- 프런트 앱 셸은 `좌측 내비게이션 / 중앙 워크스페이스 / 우측 컨텍스트 패널` 3단 구조로 재정의되었고, preview 모드에서는 이 셸을 숨기고 콘텐츠만 렌더링한다.
|
||||||
- 좌측 패널은 `248px`, 우측 패널은 `320px` 기준 폭을 사용하며, 우측 패널은 상단 토글 버튼으로 접고 펼칠 수 있다.
|
- 좌측 패널은 `248px`, 우측 패널은 `320px` 기준 폭을 사용하며, 우측 패널은 상단 토글 버튼으로 접고 펼칠 수 있다.
|
||||||
- 비로그인 상태의 로그인 유도는 좌측 하단 버튼으로만 노출하고, 좌측 상단 사용자 카드 영역에는 별도 게스트 안내 카드를 렌더링하지 않는다.
|
- 비로그인 상태의 로그인 유도는 좌측 하단 버튼으로만 노출하고, 좌측 상단 사용자 카드 영역에는 별도 게스트 안내 카드를 렌더링하지 않는다.
|
||||||
|
- 공통 셸의 좌측 내비, 우측 패널, 빠른 점프 버튼은 간단한 선형 SVG 아이콘과 두꺼운 카드형 버튼 문법을 공유한다.
|
||||||
|
|
||||||
## 데이터 저장 구조
|
## 데이터 저장 구조
|
||||||
- 메인 데이터베이스: MariaDB `tier_cursor` (기본값)
|
- 메인 데이터베이스: MariaDB `tier_cursor` (기본값)
|
||||||
@@ -27,6 +28,8 @@
|
|||||||
- 사용자 요약, 빠른 검색 버튼, 주요 라우트 내비게이션, 즐겨찾기 성격의 빠른 링크, 관리자 진입 버튼을 배치한다.
|
- 사용자 요약, 빠른 검색 버튼, 주요 라우트 내비게이션, 즐겨찾기 성격의 빠른 링크, 관리자 진입 버튼을 배치한다.
|
||||||
- 중앙 워크스페이스
|
- 중앙 워크스페이스
|
||||||
- 현재 라우트의 핵심 콘텐츠를 렌더링하는 영역이며, 홈/목록 계열 화면은 카드형 대시보드 레이아웃을 우선 적용한다.
|
- 현재 라우트의 핵심 콘텐츠를 렌더링하는 영역이며, 홈/목록 계열 화면은 카드형 대시보드 레이아웃을 우선 적용한다.
|
||||||
|
- 홈, 게임 허브, 내 티어표, 즐겨찾기 화면은 같은 카드 문법(상단 16:9 썸네일, 제목, 작성자/보조 메타, 하단 상태 영역)을 공유하도록 정리한다.
|
||||||
|
- 목록 계열 화면의 상단 도구 영역은 통계 카드와 액션 버튼을 공통 높이/반경으로 맞춰, 같은 라이브러리 대시보드로 읽히도록 정리한다.
|
||||||
- 우측 패널
|
- 우측 패널
|
||||||
- 현재 화면 문맥에 맞는 설명, 빠른 액션, 계정 상태 같은 보조 정보를 배치한다.
|
- 현재 화면 문맥에 맞는 설명, 빠른 액션, 계정 상태 같은 보조 정보를 배치한다.
|
||||||
- 에디터/관리자 세부 옵션은 후속 단계에서 이 패널로 점진 이관한다.
|
- 에디터/관리자 세부 옵션은 후속 단계에서 이 패널로 점진 이관한다.
|
||||||
@@ -34,6 +37,11 @@
|
|||||||
- 티어표 편집 화면
|
- 티어표 편집 화면
|
||||||
- 공통 우측 패널 대신 전용 로컬 편집 패널을 사용한다.
|
- 공통 우측 패널 대신 전용 로컬 편집 패널을 사용한다.
|
||||||
- 제목, 설명, 대표 썸네일, 공개 여부, 저장/삭제/요청 액션을 우측 로컬 패널에 배치한다.
|
- 제목, 설명, 대표 썸네일, 공개 여부, 저장/삭제/요청 액션을 우측 로컬 패널에 배치한다.
|
||||||
|
- 보드 바로 옆에는 드래그용 아이템 풀을 별도 패널로 두고, 커스텀 아이템 이름 정리 목록은 우측 편집 패널 내부에서 관리한다.
|
||||||
|
- 관리자 화면
|
||||||
|
- 공통 우측 패널 대신 전용 로컬 운영 패널을 사용한다.
|
||||||
|
- 우측 로컬 패널에는 `게임/아이템/티어표/회원 관리` 탭, 검색, 필터, 새로고침, 빠른 작업 제어를 배치하고, 중앙 영역에는 실제 관리 대상 목록과 상세만 렌더링한다.
|
||||||
|
- 상단 헤더에는 현재 탭 기준 요약 통계 카드를 배치해 운영 상태를 먼저 읽고, 각 관리 카드는 공통 대시보드 카드 문법(두꺼운 반경, 얕은 레이어 배경, 강조된 액션 버튼)을 공유한다.
|
||||||
|
|
||||||
## DB 스키마
|
## DB 스키마
|
||||||
- `users`
|
- `users`
|
||||||
|
|||||||
@@ -2,8 +2,13 @@
|
|||||||
|
|
||||||
## 즉시 확인 필요
|
## 즉시 확인 필요
|
||||||
- 피그마 기반 리디자인은 현재 공통 셸과 카드 목록 화면, 포커스 화면 안정화까지만 반영된 상태이므로, 에디터/관리자 우측 옵션 패널을 시안 구조에 맞게 실제 기능 패널로 이관하는 작업이 남아 있다.
|
- 피그마 기반 리디자인은 현재 공통 셸과 카드 목록 화면, 포커스 화면 안정화까지만 반영된 상태이므로, 에디터/관리자 우측 옵션 패널을 시안 구조에 맞게 실제 기능 패널로 이관하는 작업이 남아 있다.
|
||||||
|
- 홈/게임 허브/내 티어표/즐겨찾기 카드 문법은 어느 정도 통일됐지만, 아직 실제 SVG 아이콘, 미세 간격, hover/selection 상태 같은 디테일은 더 다듬을 필요가 있다.
|
||||||
|
- 목록 화면 상단 도구 막대는 공통 카드 문법으로 거의 맞췄지만, 실제 피그마처럼 필터 토글/정렬 상태를 시각적으로 더 강하게 드러내는 디테일은 남아 있다.
|
||||||
|
- 현재 공통 셸에는 임시 선형 SVG 아이콘을 사용하므로, 최종 머티리얼 아이콘 에셋을 받으면 교체하고 아이콘 크기/정렬을 다시 미세 조정할 필요가 있다.
|
||||||
|
- 공통 셸과 에디터에는 일부 실제 SVG 아이콘을 연결했지만, 아직 즐겨찾기/설정/관리자 등 나머지 내비 아이콘은 임시 선형 SVG이므로 추가 에셋 교체가 남아 있다.
|
||||||
- 공통 우측 패널의 토글과 독립 컬럼 구조는 반영되었지만, 현재는 안내 카드 중심이므로 실제 화면별 기능 컨트롤을 이 패널로 옮기는 작업이 남아 있다.
|
- 공통 우측 패널의 토글과 독립 컬럼 구조는 반영되었지만, 현재는 안내 카드 중심이므로 실제 화면별 기능 컨트롤을 이 패널로 옮기는 작업이 남아 있다.
|
||||||
- 티어표 편집 화면은 기본 편집 필드를 우측 로컬 패널로 옮겼지만, 관리자 화면도 같은 방식으로 실제 운영 패널 중심 레이아웃으로 다시 정리할 필요가 있다.
|
- 티어표 편집 화면과 관리자 화면 모두 로컬 우측 패널 구조로 옮겼지만, 아직 세부 카드 밀도와 아이콘/모션 디테일은 피그마 시안 수준으로 더 다듬을 필요가 있다.
|
||||||
|
- 관리자 화면은 헤더 요약 통계와 카드 계층까지 정리됐지만, 아직 표준 SVG 아이콘 교체와 더 세밀한 상태 색상/선택 상태 표현은 남아 있다.
|
||||||
- 머티리얼 아이콘 SVG는 아직 임시 문자/배지 스타일로 대체된 부분이 있으므로, 최종 아이콘 에셋을 받아 반영하는 작업이 필요하다.
|
- 머티리얼 아이콘 SVG는 아직 임시 문자/배지 스타일로 대체된 부분이 있으므로, 최종 아이콘 에셋을 받아 반영하는 작업이 필요하다.
|
||||||
- 미사용 커스텀 이미지 일괄 삭제는 현재 "참조가 없는 항목" 기준이며, 보관 기간 정책 같은 운영 규칙은 아직 없다.
|
- 미사용 커스텀 이미지 일괄 삭제는 현재 "참조가 없는 항목" 기준이며, 보관 기간 정책 같은 운영 규칙은 아직 없다.
|
||||||
- 업로드 이미지는 현재 원본 파일을 그대로 저장하므로, 운영 부담이 커지면 서버 저장 전 리사이즈/압축(예: 긴 변 제한, WebP 변환) 도입이 필요하다.
|
- 업로드 이미지는 현재 원본 파일을 그대로 저장하므로, 운영 부담이 커지면 서버 저장 전 리사이즈/압축(예: 긴 변 제한, WebP 변환) 도입이 필요하다.
|
||||||
|
|||||||
@@ -1,5 +1,35 @@
|
|||||||
# 업데이트 로그
|
# 업데이트 로그
|
||||||
|
|
||||||
|
## 2026-03-30 v1.2.10
|
||||||
|
- **목록 화면 상단 툴바 밀도 통일**: 홈, 게임 허브, 내 티어표, 즐겨찾기 상단 영역의 통계 카드와 액션 버튼 높이/반경/배경을 맞춰 공통 셸과 같은 도구 막대 문법으로 정리
|
||||||
|
- **홈 빠른 진입 흐름 보정**: 홈 화면 툴바에서 중복되던 버튼 흐름을 `즐겨찾기 / 내 리스트 / 커스텀 티어표 만들기` 중심으로 재구성해 실제 사용 동선에 맞게 정리
|
||||||
|
- **목록 카드 인터랙션 보강**: 주요 카드 목록에 일관된 hover 이동과 배경 전환을 넣어, 대시보드 카드가 더 또렷하게 반응하도록 조정
|
||||||
|
|
||||||
|
## 2026-03-30 v1.2.9
|
||||||
|
- **관리자 대시보드 헤더 보강**: 관리자 화면 상단에 현재 탭 기준 요약 통계 카드를 추가해, 게임/아이템/티어표/회원 상태를 즉시 읽을 수 있게 정리
|
||||||
|
- **운영 패널 질감 정리**: 우측 `320px` 운영 패널의 탭, 입력, 통계 카드, 버튼 라운드/배경/호버 상태를 공통 셸 톤에 맞춰 더 두꺼운 대시보드 카드 문법으로 통일
|
||||||
|
- **관리 카드 밀도 개선**: 게임 상세, 커스텀 아이템, 템플릿 요청, 전체 티어표, 회원 카드의 배경층·패딩·반경을 함께 다듬어 시안에 가까운 평평한 관리용 레이아웃으로 보정
|
||||||
|
|
||||||
|
## 2026-03-30 v1.2.8
|
||||||
|
- **실제 SVG 아이콘 연결 시작**: 사용자가 추가한 `grid_view`, `lists`, `dock_to_left`, `dock_to_right` 아이콘을 공통 셸 내비와 우측 패널 토글에 연결해 문자 기반 아이콘을 일부 실제 에셋으로 교체
|
||||||
|
- **에디터 3열 구조 복구**: 티어표 편집 화면을 `보드 / 아이템 풀 / 우측 편집 사이드바` 구조로 재배치해, 아이템 풀은 보드 옆에서 바로 드래그 가능하고 편집 옵션은 최우측 패널에만 남도록 수정
|
||||||
|
- **커스텀 아이템 이름 정리 위치 조정**: 커스텀 아이템 이름 수정 목록은 드래그용 아이템 풀 아래가 아니라 우측 편집 사이드바 안으로 옮겨, 보드 배치 흐름과 옵션 정리 흐름을 분리
|
||||||
|
|
||||||
|
## 2026-03-30 v1.2.7
|
||||||
|
- **공통 셸 아이콘형 정리**: 좌측 내비와 우측 보조 패널의 임시 문자 배지를 간단한 SVG 아이콘형으로 바꾸고, 버튼/카드 라운드와 밀도를 통일
|
||||||
|
- **좌측 레일 정보 밀도 개선**: 사용자 카드, 빠른 검색, 내비 버튼, 하단 로그인/관리자 버튼을 더 두꺼운 카드 문법으로 맞춰 피그마 톤에 가까운 레일 형태로 재정리
|
||||||
|
- **에디터 패널 감도 보정**: 티어표 편집 화면의 보드, 보드 툴바, 우측 편집 패널, 아이템 풀/드롭존 카드의 배경·경계·라운드를 함께 정리해 공통 셸과 시각 언어를 맞춤
|
||||||
|
|
||||||
|
## 2026-03-30 v1.2.6
|
||||||
|
- **목록형 화면 카드 문법 통일**: 홈, 게임 허브, 내 티어표, 즐겨찾기 화면의 카드형 목록을 동일한 썸네일/제목/작성자/메타 구조로 정리해 대시보드 톤을 맞춤
|
||||||
|
- **홈 화면 대시보드 재정렬**: 메인 게임 라이브러리 화면에 상단 상태 카드와 CTA를 추가하고, 게임 카드는 `16:9` 썸네일 + ID 메타를 갖는 라이브러리 카드 형태로 재배치
|
||||||
|
- **게임 허브 헤더/검색 정리**: 게임 허브는 상단 통계와 생성 버튼, 보조 설명을 포함한 헤더로 재구성하고, 공개 티어표 카드도 같은 카드 밀도로 재정리
|
||||||
|
|
||||||
|
## 2026-03-30 v1.2.5
|
||||||
|
- **관리자 로컬 우측 패널 이관**: 관리자 화면도 공통 우측 패널 대신 화면 내부의 `320px` 전용 운영 패널을 사용하도록 정리하고, 탭·검색·필터·빠른 액션을 우측으로 이동
|
||||||
|
- **관리 화면 본문 집중도 개선**: 중앙 영역은 상단 고정 게임 순서, 선택된 게임 상세, 커스텀 아이템 카드, 템플릿 요청/전체 티어표, 회원 카드 같은 실제 관리 대상만 남기고 빈 상태 안내도 별도 패널로 정리
|
||||||
|
- **관리자 셸 예외 확장**: 공통 앱 셸에서 `/admin`도 전용 로컬 우측 패널을 사용하는 포커스 화면으로 분류해 generic 우측 문맥 카드가 중복 표시되지 않게 조정
|
||||||
|
|
||||||
## 2026-03-30 v1.2.4
|
## 2026-03-30 v1.2.4
|
||||||
- **비로그인 중복 안내 제거**: 좌측 사이드 상단의 별도 로그인 안내 카드를 제거하고, 비로그인 상태에서는 좌측 하단 버튼만 `로그인` 진입점으로 사용하도록 단순화
|
- **비로그인 중복 안내 제거**: 좌측 사이드 상단의 별도 로그인 안내 카드를 제거하고, 비로그인 상태에서는 좌측 하단 버튼만 `로그인` 진입점으로 사용하도록 단순화
|
||||||
- **에디터 우측 편집 패널 이관**: 티어표 편집 화면의 제목, 설명, 대표 썸네일, 공개 여부, 저장/삭제/요청 액션을 중앙 상단이 아니라 독립 우측 편집 패널로 이동
|
- **에디터 우측 편집 패널 이관**: 티어표 편집 화면의 제목, 설명, 대표 썸네일, 공개 여부, 저장/삭제/요청 액션을 중앙 상단이 아니라 독립 우측 편집 패널로 이동
|
||||||
|
|||||||
@@ -4,6 +4,10 @@ import { useRoute, useRouter } from 'vue-router'
|
|||||||
import { useAuthStore } from './stores/auth'
|
import { useAuthStore } from './stores/auth'
|
||||||
import { toApiUrl } from './lib/runtime'
|
import { toApiUrl } from './lib/runtime'
|
||||||
import { useToast } from './composables/useToast'
|
import { useToast } from './composables/useToast'
|
||||||
|
import iconDockToLeft from './assets/icons/dock_to_left.svg'
|
||||||
|
import iconDockToRight from './assets/icons/dock_to_right.svg'
|
||||||
|
import iconGridView from './assets/icons/grid_view.svg'
|
||||||
|
import iconLists from './assets/icons/lists.svg'
|
||||||
|
|
||||||
const route = useRoute()
|
const route = useRoute()
|
||||||
const router = useRouter()
|
const router = useRouter()
|
||||||
@@ -15,7 +19,7 @@ const rightRailOpen = ref(true)
|
|||||||
|
|
||||||
const isAdmin = computed(() => !!auth.user?.isAdmin)
|
const isAdmin = computed(() => !!auth.user?.isAdmin)
|
||||||
const isPreviewMode = computed(() => route.query.preview === '1')
|
const isPreviewMode = computed(() => route.query.preview === '1')
|
||||||
const usesLocalRightRail = computed(() => ['editEditor', 'newEditor'].includes(String(route.name || '')))
|
const usesLocalRightRail = computed(() => ['editEditor', 'newEditor', 'admin'].includes(String(route.name || '')))
|
||||||
const avatarUrl = computed(() => (auth.user?.avatarSrc ? toApiUrl(auth.user.avatarSrc) : ''))
|
const avatarUrl = computed(() => (auth.user?.avatarSrc ? toApiUrl(auth.user.avatarSrc) : ''))
|
||||||
const accountName = computed(() => {
|
const accountName = computed(() => {
|
||||||
const nickname = (auth.user?.nickname || '').trim()
|
const nickname = (auth.user?.nickname || '').trim()
|
||||||
@@ -27,21 +31,21 @@ const accountName = computed(() => {
|
|||||||
const accountEmail = computed(() => (auth.user?.email || '').trim() || '로그인 후 개인 메뉴를 사용할 수 있어요.')
|
const accountEmail = computed(() => (auth.user?.email || '').trim() || '로그인 후 개인 메뉴를 사용할 수 있어요.')
|
||||||
const leftNavItems = computed(() => {
|
const leftNavItems = computed(() => {
|
||||||
const items = [
|
const items = [
|
||||||
{ key: 'home', label: 'Games', path: '/', initials: 'GM' },
|
{ key: 'home', label: 'Games', path: '/', iconSrc: iconGridView },
|
||||||
{ key: 'me', label: '내 리스트', path: '/me', initials: 'ME', requiresAuth: true },
|
{ key: 'me', label: '내 리스트', path: '/me', iconSrc: iconLists, requiresAuth: true },
|
||||||
{ key: 'favorites', label: '즐겨찾기', path: '/favorites', initials: 'FV', requiresAuth: true },
|
{ key: 'favorites', label: '즐겨찾기', path: '/favorites', icon: 'M12 4.75l2.18 4.42 4.88.71-3.53 3.44.83 4.86L12 15.9 7.64 18.18l.83-4.86-3.53-3.44 4.88-.71z', requiresAuth: true },
|
||||||
{ key: 'profile', label: 'Settings', path: '/profile', initials: 'ST', requiresAuth: true },
|
{ key: 'profile', label: 'Settings', path: '/profile', icon: 'M12 4.75a2.2 2.2 0 0 1 2.08 1.5l.18.56.58.13a2.2 2.2 0 0 1 1.52 2.76l-.17.56.39.46a2.2 2.2 0 0 1 0 2.86l-.39.46.17.56a2.2 2.2 0 0 1-1.52 2.76l-.58.13-.18.56a2.2 2.2 0 0 1-4.16 0l-.18-.56-.58-.13a2.2 2.2 0 0 1-1.52-2.76l.17-.56-.39-.46a2.2 2.2 0 0 1 0-2.86l.39-.46-.17-.56a2.2 2.2 0 0 1 1.52-2.76l.58-.13.18-.56A2.2 2.2 0 0 1 12 4.75z M12 9.35a2.65 2.65 0 1 0 0 5.3 2.65 2.65 0 0 0 0-5.3z', requiresAuth: true },
|
||||||
]
|
]
|
||||||
if (isAdmin.value) {
|
if (isAdmin.value) {
|
||||||
items.push({ key: 'admin', label: 'Admin', path: '/admin', initials: 'AD' })
|
items.push({ key: 'admin', label: 'Admin', path: '/admin', iconSrc: iconLists })
|
||||||
}
|
}
|
||||||
return items.filter((item) => !item.requiresAuth || auth.user)
|
return items.filter((item) => !item.requiresAuth || auth.user)
|
||||||
})
|
})
|
||||||
const routeMeta = computed(() => {
|
const routeMeta = computed(() => {
|
||||||
if (route.name === 'home') {
|
if (route.name === 'home') {
|
||||||
return {
|
return {
|
||||||
title: 'Main Title',
|
title: 'Tier Maker',
|
||||||
subtitle: '게임 선택 및 커스텀 티어표 진입',
|
subtitle: '게임 템플릿 선택과 커스텀 보드 시작',
|
||||||
contextTitle: '빠른 시작',
|
contextTitle: '빠른 시작',
|
||||||
contextText: auth.user ? '커스텀 티어표를 만들거나 원하는 게임을 바로 선택할 수 있어요.' : '로그인하면 커스텀 티어표 생성과 개인 목록 관리가 열립니다.',
|
contextText: auth.user ? '커스텀 티어표를 만들거나 원하는 게임을 바로 선택할 수 있어요.' : '로그인하면 커스텀 티어표 생성과 개인 목록 관리가 열립니다.',
|
||||||
actionLabel: auth.user ? '커스텀 티어표 만들기' : '로그인하러 가기',
|
actionLabel: auth.user ? '커스텀 티어표 만들기' : '로그인하러 가기',
|
||||||
@@ -52,8 +56,8 @@ const routeMeta = computed(() => {
|
|||||||
}
|
}
|
||||||
if (route.name === 'gameHub') {
|
if (route.name === 'gameHub') {
|
||||||
return {
|
return {
|
||||||
title: 'Tier Lists',
|
title: 'Game Boards',
|
||||||
subtitle: '게임별 공개 티어표 목록',
|
subtitle: '게임별 공개 티어표 탐색',
|
||||||
contextTitle: '작성 작업',
|
contextTitle: '작성 작업',
|
||||||
contextText: auth.user ? '이 게임의 새 티어표를 만들거나 기존 공개 티어표를 확인할 수 있어요.' : '로그인 후 새 티어표를 만들 수 있어요.',
|
contextText: auth.user ? '이 게임의 새 티어표를 만들거나 기존 공개 티어표를 확인할 수 있어요.' : '로그인 후 새 티어표를 만들 수 있어요.',
|
||||||
actionLabel: auth.user ? '새 티어표 만들기' : '로그인하러 가기',
|
actionLabel: auth.user ? '새 티어표 만들기' : '로그인하러 가기',
|
||||||
@@ -128,6 +132,13 @@ const favoriteLinks = computed(() => [
|
|||||||
...(auth.user ? [{ label: 'My Lists', path: '/me' }] : []),
|
...(auth.user ? [{ label: 'My Lists', path: '/me' }] : []),
|
||||||
])
|
])
|
||||||
|
|
||||||
|
function railGlyph(type) {
|
||||||
|
if (type === 'menu') return 'M4 6.5h16M4 12h16M4 17.5h16'
|
||||||
|
if (type === 'search') return 'M10.2 6.2a4 4 0 1 1 0 8 4 4 0 0 1 0-8z M13.6 13.6l3.2 3.2'
|
||||||
|
if (type === 'link') return 'M8 12h8 M12 8l4 4-4 4'
|
||||||
|
return 'M4 12h16'
|
||||||
|
}
|
||||||
|
|
||||||
onMounted(async () => {
|
onMounted(async () => {
|
||||||
await auth.refresh()
|
await auth.refresh()
|
||||||
if (typeof window !== 'undefined') {
|
if (typeof window !== 'undefined') {
|
||||||
@@ -192,7 +203,9 @@ async function logout() {
|
|||||||
<template v-else>
|
<template v-else>
|
||||||
<aside class="leftRail">
|
<aside class="leftRail">
|
||||||
<div class="leftRail__top">
|
<div class="leftRail__top">
|
||||||
<button class="ghostIcon" type="button" aria-label="메뉴">▥</button>
|
<button class="ghostIcon ghostIcon--iconOnly" type="button" aria-label="메뉴">
|
||||||
|
<svg viewBox="0 0 24 24" aria-hidden="true"><path :d="railGlyph('menu')" /></svg>
|
||||||
|
</button>
|
||||||
<div class="brandBlock" @click="$router.push('/')">
|
<div class="brandBlock" @click="$router.push('/')">
|
||||||
<div class="brandBlock__title">Tier Maker</div>
|
<div class="brandBlock__title">Tier Maker</div>
|
||||||
<div class="brandBlock__sub">by zenn</div>
|
<div class="brandBlock__sub">by zenn</div>
|
||||||
@@ -215,8 +228,10 @@ async function logout() {
|
|||||||
</div>
|
</div>
|
||||||
|
|
||||||
<button class="searchStub" type="button" @click="$router.push('/favorites')">
|
<button class="searchStub" type="button" @click="$router.push('/favorites')">
|
||||||
<span class="searchStub__icon">⌕</span>
|
<span class="searchStub__icon">
|
||||||
<span>Search</span>
|
<svg viewBox="0 0 24 24" aria-hidden="true"><path :d="railGlyph('search')" /></svg>
|
||||||
|
</span>
|
||||||
|
<span>Quick Search</span>
|
||||||
</button>
|
</button>
|
||||||
|
|
||||||
<nav class="leftNav">
|
<nav class="leftNav">
|
||||||
@@ -227,7 +242,10 @@ async function logout() {
|
|||||||
class="leftNav__item"
|
class="leftNav__item"
|
||||||
:class="{ 'leftNav__item--active': isRouteActive(item.path) }"
|
:class="{ 'leftNav__item--active': isRouteActive(item.path) }"
|
||||||
>
|
>
|
||||||
<span class="leftNav__glyph">{{ item.initials }}</span>
|
<span class="leftNav__glyph">
|
||||||
|
<img v-if="item.iconSrc" :src="item.iconSrc" alt="" />
|
||||||
|
<svg v-else viewBox="0 0 24 24" aria-hidden="true"><path :d="item.icon" /></svg>
|
||||||
|
</span>
|
||||||
<span>{{ item.label }}</span>
|
<span>{{ item.label }}</span>
|
||||||
</RouterLink>
|
</RouterLink>
|
||||||
</nav>
|
</nav>
|
||||||
@@ -255,7 +273,8 @@ async function logout() {
|
|||||||
</div>
|
</div>
|
||||||
<div class="workspaceHead__actions">
|
<div class="workspaceHead__actions">
|
||||||
<button class="ghostIcon ghostIcon--workspace" type="button" :aria-pressed="rightRailOpen" @click="toggleRightRail">
|
<button class="ghostIcon ghostIcon--workspace" type="button" :aria-pressed="rightRailOpen" @click="toggleRightRail">
|
||||||
{{ rightRailOpen ? '우측 패널 숨기기' : '우측 패널 보기' }}
|
<img :src="rightRailOpen ? iconDockToRight : iconDockToLeft" alt="" />
|
||||||
|
<span>{{ rightRailOpen ? '패널 숨기기' : '패널 보기' }}</span>
|
||||||
</button>
|
</button>
|
||||||
</div>
|
</div>
|
||||||
</header>
|
</header>
|
||||||
@@ -272,7 +291,9 @@ async function logout() {
|
|||||||
:aria-hidden="!rightRailOpen"
|
:aria-hidden="!rightRailOpen"
|
||||||
>
|
>
|
||||||
<div class="rightRail__top">
|
<div class="rightRail__top">
|
||||||
<button class="ghostIcon" type="button" aria-label="상태">⌗</button>
|
<button class="ghostIcon ghostIcon--iconOnly" type="button" aria-label="상태">
|
||||||
|
<img :src="iconGridView" alt="" />
|
||||||
|
</button>
|
||||||
</div>
|
</div>
|
||||||
<section class="contextCard">
|
<section class="contextCard">
|
||||||
<div class="contextCard__label">Context</div>
|
<div class="contextCard__label">Context</div>
|
||||||
@@ -293,6 +314,17 @@ async function logout() {
|
|||||||
<span class="contextStat__value">{{ isAdmin ? 'Admin' : auth.user ? 'Member' : 'Guest' }}</span>
|
<span class="contextStat__value">{{ isAdmin ? 'Admin' : auth.user ? 'Member' : 'Guest' }}</span>
|
||||||
</div>
|
</div>
|
||||||
</section>
|
</section>
|
||||||
|
<section class="contextCard contextCard--links">
|
||||||
|
<div class="contextCard__label">Jump</div>
|
||||||
|
<button class="contextLink" type="button" @click="$router.push('/')">
|
||||||
|
<svg viewBox="0 0 24 24" aria-hidden="true"><path :d="railGlyph('link')" /></svg>
|
||||||
|
<span>게임 목록으로</span>
|
||||||
|
</button>
|
||||||
|
<button v-if="auth.user" class="contextLink" type="button" @click="$router.push('/me')">
|
||||||
|
<svg viewBox="0 0 24 24" aria-hidden="true"><path :d="railGlyph('link')" /></svg>
|
||||||
|
<span>내 티어표 열기</span>
|
||||||
|
</button>
|
||||||
|
</section>
|
||||||
</aside>
|
</aside>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
@@ -375,14 +407,46 @@ async function logout() {
|
|||||||
background: rgba(255, 255, 255, 0.03);
|
background: rgba(255, 255, 255, 0.03);
|
||||||
color: rgba(255, 255, 255, 0.72);
|
color: rgba(255, 255, 255, 0.72);
|
||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
|
display: inline-flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
gap: 8px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.ghostIcon svg,
|
||||||
|
.searchStub__icon svg,
|
||||||
|
.leftNav__glyph svg,
|
||||||
|
.contextLink svg {
|
||||||
|
width: 16px;
|
||||||
|
height: 16px;
|
||||||
|
stroke: currentColor;
|
||||||
|
stroke-width: 1.8;
|
||||||
|
fill: none;
|
||||||
|
stroke-linecap: round;
|
||||||
|
stroke-linejoin: round;
|
||||||
|
}
|
||||||
|
|
||||||
|
.ghostIcon img,
|
||||||
|
.leftNav__glyph img,
|
||||||
|
.ghostIcon--workspace img {
|
||||||
|
width: 16px;
|
||||||
|
height: 16px;
|
||||||
|
display: block;
|
||||||
|
filter: brightness(0) saturate(100%) invert(94%) sepia(6%) saturate(207%) hue-rotate(186deg) brightness(96%) contrast(92%);
|
||||||
|
}
|
||||||
|
|
||||||
|
.ghostIcon--iconOnly {
|
||||||
|
min-width: 32px;
|
||||||
|
width: 32px;
|
||||||
|
padding: 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
.ghostIcon--workspace {
|
.ghostIcon--workspace {
|
||||||
min-width: 118px;
|
min-width: 132px;
|
||||||
height: 36px;
|
height: 38px;
|
||||||
padding: 0 14px;
|
padding: 0 14px;
|
||||||
border-radius: 10px;
|
border-radius: 10px;
|
||||||
background: rgba(255, 255, 255, 0.05);
|
background: rgba(255, 255, 255, 0.06);
|
||||||
color: rgba(255, 255, 255, 0.88);
|
color: rgba(255, 255, 255, 0.88);
|
||||||
font-size: 12px;
|
font-size: 12px;
|
||||||
font-weight: 800;
|
font-weight: 800;
|
||||||
@@ -390,7 +454,7 @@ async function logout() {
|
|||||||
|
|
||||||
.brandBlock {
|
.brandBlock {
|
||||||
display: grid;
|
display: grid;
|
||||||
gap: 2px;
|
gap: 4px;
|
||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -418,9 +482,9 @@ async function logout() {
|
|||||||
gap: 12px;
|
gap: 12px;
|
||||||
align-items: center;
|
align-items: center;
|
||||||
padding: 10px;
|
padding: 10px;
|
||||||
border-radius: 14px;
|
border-radius: 16px;
|
||||||
border: 1px solid rgba(255, 255, 255, 0.08);
|
border: 1px solid rgba(255, 255, 255, 0.08);
|
||||||
background: rgba(255, 255, 255, 0.03);
|
background: rgba(255, 255, 255, 0.04);
|
||||||
color: inherit;
|
color: inherit;
|
||||||
text-align: left;
|
text-align: left;
|
||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
@@ -428,8 +492,8 @@ async function logout() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
.appUserCard__avatar {
|
.appUserCard__avatar {
|
||||||
width: 38px;
|
width: 42px;
|
||||||
height: 38px;
|
height: 42px;
|
||||||
border-radius: 12px;
|
border-radius: 12px;
|
||||||
object-fit: cover;
|
object-fit: cover;
|
||||||
flex: 0 0 auto;
|
flex: 0 0 auto;
|
||||||
@@ -490,17 +554,21 @@ async function logout() {
|
|||||||
display: flex;
|
display: flex;
|
||||||
align-items: center;
|
align-items: center;
|
||||||
gap: 10px;
|
gap: 10px;
|
||||||
padding: 10px 12px;
|
padding: 11px 12px;
|
||||||
border-radius: 12px;
|
border-radius: 14px;
|
||||||
border: 1px solid rgba(255, 255, 255, 0.08);
|
border: 1px solid rgba(255, 255, 255, 0.08);
|
||||||
background: rgba(255, 255, 255, 0.02);
|
background: rgba(255, 255, 255, 0.03);
|
||||||
color: rgba(255, 255, 255, 0.62);
|
color: rgba(255, 255, 255, 0.62);
|
||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
margin-bottom: 14px;
|
margin-bottom: 14px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.searchStub__icon {
|
.searchStub__icon {
|
||||||
font-size: 14px;
|
width: 18px;
|
||||||
|
height: 18px;
|
||||||
|
display: inline-flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
}
|
}
|
||||||
|
|
||||||
.leftNav {
|
.leftNav {
|
||||||
@@ -511,29 +579,27 @@ async function logout() {
|
|||||||
.leftNav__item {
|
.leftNav__item {
|
||||||
display: flex;
|
display: flex;
|
||||||
align-items: center;
|
align-items: center;
|
||||||
gap: 10px;
|
gap: 12px;
|
||||||
padding: 10px;
|
padding: 11px 12px;
|
||||||
border-radius: 12px;
|
border-radius: 14px;
|
||||||
color: rgba(255, 255, 255, 0.76);
|
color: rgba(255, 255, 255, 0.76);
|
||||||
text-decoration: none;
|
text-decoration: none;
|
||||||
|
transition: background 180ms ease, color 180ms ease, transform 180ms ease;
|
||||||
}
|
}
|
||||||
|
|
||||||
.leftNav__item--active,
|
.leftNav__item--active,
|
||||||
.leftNav__item.router-link-active {
|
.leftNav__item.router-link-active {
|
||||||
background: rgba(255, 255, 255, 0.08);
|
background: rgba(255, 255, 255, 0.1);
|
||||||
color: rgba(255, 255, 255, 0.96);
|
color: rgba(255, 255, 255, 0.96);
|
||||||
}
|
}
|
||||||
|
|
||||||
.leftNav__glyph {
|
.leftNav__glyph {
|
||||||
width: 24px;
|
width: 28px;
|
||||||
height: 24px;
|
height: 28px;
|
||||||
border-radius: 8px;
|
border-radius: 10px;
|
||||||
display: grid;
|
display: grid;
|
||||||
place-items: center;
|
place-items: center;
|
||||||
background: rgba(255, 255, 255, 0.06);
|
background: rgba(255, 255, 255, 0.06);
|
||||||
font-size: 10px;
|
|
||||||
font-weight: 900;
|
|
||||||
letter-spacing: 0.06em;
|
|
||||||
flex: 0 0 auto;
|
flex: 0 0 auto;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -557,6 +623,7 @@ async function logout() {
|
|||||||
color: rgba(255, 255, 255, 0.7);
|
color: rgba(255, 255, 255, 0.7);
|
||||||
text-decoration: none;
|
text-decoration: none;
|
||||||
font-size: 14px;
|
font-size: 14px;
|
||||||
|
padding: 4px 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
.favoriteLink__dot {
|
.favoriteLink__dot {
|
||||||
@@ -577,12 +644,13 @@ async function logout() {
|
|||||||
justify-content: center;
|
justify-content: center;
|
||||||
align-items: center;
|
align-items: center;
|
||||||
padding: 12px 14px;
|
padding: 12px 14px;
|
||||||
border-radius: 12px;
|
border-radius: 14px;
|
||||||
border: 1px solid rgba(255, 255, 255, 0.12);
|
border: 1px solid rgba(255, 255, 255, 0.12);
|
||||||
background: rgba(255, 255, 255, 0.04);
|
background: rgba(255, 255, 255, 0.05);
|
||||||
color: rgba(255, 255, 255, 0.92);
|
color: rgba(255, 255, 255, 0.92);
|
||||||
text-decoration: none;
|
text-decoration: none;
|
||||||
box-sizing: border-box;
|
box-sizing: border-box;
|
||||||
|
font-weight: 800;
|
||||||
}
|
}
|
||||||
|
|
||||||
.leftRail {
|
.leftRail {
|
||||||
@@ -620,7 +688,7 @@ async function logout() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
.workspaceHead__title {
|
.workspaceHead__title {
|
||||||
font-size: 28px;
|
font-size: 30px;
|
||||||
font-weight: 900;
|
font-weight: 900;
|
||||||
letter-spacing: -0.04em;
|
letter-spacing: -0.04em;
|
||||||
}
|
}
|
||||||
@@ -633,10 +701,10 @@ async function logout() {
|
|||||||
|
|
||||||
.workspaceBody {
|
.workspaceBody {
|
||||||
min-height: calc(100vh - 110px);
|
min-height: calc(100vh - 110px);
|
||||||
padding: 18px;
|
padding: 20px;
|
||||||
border-radius: 26px;
|
border-radius: 26px;
|
||||||
border: 1px solid rgba(255, 255, 255, 0.08);
|
border: 1px solid rgba(255, 255, 255, 0.08);
|
||||||
background: #2b2b2b;
|
background: linear-gradient(180deg, #2d2d2d 0%, #2a2a2a 100%);
|
||||||
box-shadow: inset 0 1px 0 rgba(255, 255, 255, 0.04);
|
box-shadow: inset 0 1px 0 rgba(255, 255, 255, 0.04);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -649,10 +717,10 @@ async function logout() {
|
|||||||
.contextCard {
|
.contextCard {
|
||||||
display: grid;
|
display: grid;
|
||||||
gap: 12px;
|
gap: 12px;
|
||||||
padding: 14px;
|
padding: 16px;
|
||||||
border-radius: 18px;
|
border-radius: 18px;
|
||||||
border: 1px solid rgba(255, 255, 255, 0.08);
|
border: 1px solid rgba(255, 255, 255, 0.08);
|
||||||
background: rgba(255, 255, 255, 0.02);
|
background: rgba(255, 255, 255, 0.03);
|
||||||
}
|
}
|
||||||
|
|
||||||
.contextCard__label {
|
.contextCard__label {
|
||||||
@@ -664,7 +732,7 @@ async function logout() {
|
|||||||
|
|
||||||
.contextCard__title {
|
.contextCard__title {
|
||||||
margin: 0;
|
margin: 0;
|
||||||
font-size: 20px;
|
font-size: 22px;
|
||||||
line-height: 1.2;
|
line-height: 1.2;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -686,6 +754,24 @@ async function logout() {
|
|||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.contextCard--links {
|
||||||
|
gap: 10px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.contextLink {
|
||||||
|
width: 100%;
|
||||||
|
display: inline-flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: space-between;
|
||||||
|
gap: 10px;
|
||||||
|
padding: 11px 12px;
|
||||||
|
border-radius: 12px;
|
||||||
|
border: 1px solid rgba(255, 255, 255, 0.08);
|
||||||
|
background: rgba(255, 255, 255, 0.03);
|
||||||
|
color: rgba(255, 255, 255, 0.86);
|
||||||
|
cursor: pointer;
|
||||||
|
}
|
||||||
|
|
||||||
.contextStat {
|
.contextStat {
|
||||||
display: flex;
|
display: flex;
|
||||||
justify-content: space-between;
|
justify-content: space-between;
|
||||||
|
|||||||
1
frontend/src/assets/icons/dock_to_left.svg
Normal file
1
frontend/src/assets/icons/dock_to_left.svg
Normal file
@@ -0,0 +1 @@
|
|||||||
|
<svg xmlns="http://www.w3.org/2000/svg" height="24px" viewBox="0 -960 960 960" width="24px" fill="ffffff"><path d="M200-120q-33 0-56.5-23.5T120-200v-560q0-33 23.5-56.5T200-840h560q33 0 56.5 23.5T840-760v560q0 33-23.5 56.5T760-120H200Zm440-80h120v-560H640v560Zm-80 0v-560H200v560h360Zm80 0h120-120Z"/></svg>
|
||||||
|
After Width: | Height: | Size: 306 B |
1
frontend/src/assets/icons/dock_to_right.svg
Normal file
1
frontend/src/assets/icons/dock_to_right.svg
Normal file
@@ -0,0 +1 @@
|
|||||||
|
<svg xmlns="http://www.w3.org/2000/svg" height="24px" viewBox="0 -960 960 960" width="24px" fill="ffffff"><path d="M200-120q-33 0-56.5-23.5T120-200v-560q0-33 23.5-56.5T200-840h560q33 0 56.5 23.5T840-760v560q0 33-23.5 56.5T760-120H200Zm120-80v-560H200v560h120Zm80 0h360v-560H400v560Zm-80 0H200h120Z"/></svg>
|
||||||
|
After Width: | Height: | Size: 306 B |
1
frontend/src/assets/icons/grid_view.svg
Normal file
1
frontend/src/assets/icons/grid_view.svg
Normal file
@@ -0,0 +1 @@
|
|||||||
|
<svg xmlns="http://www.w3.org/2000/svg" height="24px" viewBox="0 -960 960 960" width="24px" fill="ffffff"><path d="M120-520v-320h320v320H120Zm0 400v-320h320v320H120Zm400-400v-320h320v320H520Zm0 400v-320h320v320H520ZM200-600h160v-160H200v160Zm400 0h160v-160H600v160Zm0 400h160v-160H600v160Zm-400 0h160v-160H200v160Zm400-400Zm0 240Zm-240 0Zm0-240Z"/></svg>
|
||||||
|
After Width: | Height: | Size: 354 B |
1
frontend/src/assets/icons/lists.svg
Normal file
1
frontend/src/assets/icons/lists.svg
Normal file
@@ -0,0 +1 @@
|
|||||||
|
<svg xmlns="http://www.w3.org/2000/svg" height="24px" viewBox="0 -960 960 960" width="24px" fill="ffffff"><path d="M80-160v-160h160v160H80Zm240 0v-160h560v160H320ZM80-400v-160h160v160H80Zm240 0v-160h560v160H320ZM80-640v-160h160v160H80Zm240 0v-160h560v160H320Z"/></svg>
|
||||||
|
After Width: | Height: | Size: 268 B |
@@ -73,6 +73,67 @@ const featuredGames = computed(() =>
|
|||||||
)
|
)
|
||||||
const availableGamesForFeatured = computed(() => games.value.filter((game) => !featuredGameIds.value.includes(game.id)))
|
const availableGamesForFeatured = computed(() => games.value.filter((game) => !featuredGameIds.value.includes(game.id)))
|
||||||
const importModalItemCount = computed(() => importModalItems.value.length)
|
const importModalItemCount = computed(() => importModalItems.value.length)
|
||||||
|
const activeTabTitle = computed(() => {
|
||||||
|
if (activeTab.value === 'games') return '게임 관리'
|
||||||
|
if (activeTab.value === 'items') return '아이템 관리'
|
||||||
|
if (activeTab.value === 'tierlists') {
|
||||||
|
return tierlistsMode.value === 'requests' ? '템플릿 요청 관리' : '전체 티어표 관리'
|
||||||
|
}
|
||||||
|
return '회원 관리'
|
||||||
|
})
|
||||||
|
const activeTabDescription = computed(() => {
|
||||||
|
if (activeTab.value === 'games') {
|
||||||
|
return '홈 노출 순서, 게임 생성, 썸네일, 기본 아이템을 한 화면에서 정리합니다.'
|
||||||
|
}
|
||||||
|
if (activeTab.value === 'items') {
|
||||||
|
return '사용자 커스텀 이미지를 검색하고, 미사용 이미지를 정리하거나 템플릿으로 승격할 수 있어요.'
|
||||||
|
}
|
||||||
|
if (activeTab.value === 'tierlists') {
|
||||||
|
return tierlistsMode.value === 'requests'
|
||||||
|
? '사용자 요청 기반으로 새 템플릿 생성이나 템플릿 업데이트를 승인합니다.'
|
||||||
|
: '공개/비공개 포함 전체 티어표를 확인하고, 추가 아이템을 템플릿으로 가져올 수 있어요.'
|
||||||
|
}
|
||||||
|
return '계정 정보, 권한, 비밀번호와 최근 활동을 함께 확인합니다.'
|
||||||
|
})
|
||||||
|
const adminOverviewStats = computed(() => {
|
||||||
|
const publishedTierLists = adminTierLists.value.filter((tierList) => tierList.isPublic).length
|
||||||
|
const pendingRequests = templateRequests.value.length
|
||||||
|
const orphanItems = customItems.value.filter((item) => item.usageCount === 0).length
|
||||||
|
const adminCount = users.value.filter((user) => user.isAdmin || user.draftIsAdmin).length
|
||||||
|
|
||||||
|
if (activeTab.value === 'games') {
|
||||||
|
return [
|
||||||
|
{ label: '전체 게임', value: `${games.value.length}` },
|
||||||
|
{ label: '상단 고정', value: `${featuredGameIds.value.length}/50` },
|
||||||
|
{ label: '선택 상태', value: hasSelectedGame.value ? '활성' : '대기' },
|
||||||
|
]
|
||||||
|
}
|
||||||
|
if (activeTab.value === 'items') {
|
||||||
|
return [
|
||||||
|
{ label: '검색 결과', value: `${customItemTotal.value}` },
|
||||||
|
{ label: '미사용', value: `${orphanItems}` },
|
||||||
|
{ label: '대상 게임', value: customItemTargetGameId.value ? '선택됨' : '미선택' },
|
||||||
|
]
|
||||||
|
}
|
||||||
|
if (activeTab.value === 'tierlists') {
|
||||||
|
return tierlistsMode.value === 'requests'
|
||||||
|
? [
|
||||||
|
{ label: '대기 요청', value: `${pendingRequests}` },
|
||||||
|
{ label: '생성 요청', value: `${templateRequests.value.filter((request) => request.type === 'create').length}` },
|
||||||
|
{ label: '업데이트 요청', value: `${templateRequests.value.filter((request) => request.type === 'update').length}` },
|
||||||
|
]
|
||||||
|
: [
|
||||||
|
{ label: '검색 결과', value: `${adminTierListTotal.value}` },
|
||||||
|
{ label: '공개 티어표', value: `${publishedTierLists}` },
|
||||||
|
{ label: '현재 페이지', value: `${adminTierListPage.value}/${adminTierListPageCount.value}` },
|
||||||
|
]
|
||||||
|
}
|
||||||
|
return [
|
||||||
|
{ label: '가입 회원', value: `${users.value.length}` },
|
||||||
|
{ label: '관리자', value: `${adminCount}` },
|
||||||
|
{ label: '활동 계정', value: `${users.value.filter((user) => user.tierListCount > 0).length}` },
|
||||||
|
]
|
||||||
|
})
|
||||||
|
|
||||||
onMounted(async () => {
|
onMounted(async () => {
|
||||||
await auth.refresh()
|
await auth.refresh()
|
||||||
@@ -869,14 +930,21 @@ async function saveFeaturedOrder() {
|
|||||||
<div v-else-if="!isAdmin" class="warn">이 계정은 관리자 권한이 없어요.</div>
|
<div v-else-if="!isAdmin" class="warn">이 계정은 관리자 권한이 없어요.</div>
|
||||||
|
|
||||||
<template v-else>
|
<template v-else>
|
||||||
<div class="tabs">
|
<div class="adminWorkspace">
|
||||||
<button class="tab" :class="{ 'tab--active': activeTab === 'games' }" @click="setTab('games')">게임 관리</button>
|
<div class="adminMain">
|
||||||
<button class="tab" :class="{ 'tab--active': activeTab === 'items' }" @click="setTab('items')">아이템 관리</button>
|
<header class="adminHero">
|
||||||
<button class="tab" :class="{ 'tab--active': activeTab === 'tierlists' }" @click="setTab('tierlists')">티어표 관리</button>
|
<div class="adminHero__eyebrow">Admin Workspace</div>
|
||||||
<button class="tab" :class="{ 'tab--active': activeTab === 'users' }" @click="setTab('users')">회원 관리</button>
|
<h2 class="adminHero__title">{{ activeTabTitle }}</h2>
|
||||||
</div>
|
<p class="adminHero__desc">{{ activeTabDescription }}</p>
|
||||||
|
<div class="adminHero__stats">
|
||||||
|
<article v-for="stat in adminOverviewStats" :key="stat.label" class="adminHeroStat">
|
||||||
|
<span class="adminHeroStat__label">{{ stat.label }}</span>
|
||||||
|
<strong class="adminHeroStat__value">{{ stat.value }}</strong>
|
||||||
|
</article>
|
||||||
|
</div>
|
||||||
|
</header>
|
||||||
|
|
||||||
<template v-if="activeTab === 'games'">
|
<template v-if="activeTab === 'games'">
|
||||||
<div class="panel">
|
<div class="panel">
|
||||||
<div class="sectionHeader">
|
<div class="sectionHeader">
|
||||||
<div>
|
<div>
|
||||||
@@ -927,33 +995,6 @@ async function saveFeaturedOrder() {
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div class="modeTabs">
|
|
||||||
<button class="modeTab" :class="{ 'modeTab--active': gameMode === 'existing' }" @click="setGameMode('existing')">
|
|
||||||
등록된 게임 선택
|
|
||||||
</button>
|
|
||||||
<button class="modeTab" :class="{ 'modeTab--active': gameMode === 'new' }" @click="setGameMode('new')">
|
|
||||||
새 게임 추가
|
|
||||||
</button>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="panel panel--compact">
|
|
||||||
<template v-if="gameMode === 'existing'">
|
|
||||||
<!-- <div class="panel__title">등록된 게임 선택</div> -->
|
|
||||||
<select v-model="selectedGameId" class="select" @change="loadGame">
|
|
||||||
<option value="">게임을 선택해주세요</option>
|
|
||||||
<option v-for="game in games" :key="game.id" :value="game.id">{{ game.name }} ({{ game.id }})</option>
|
|
||||||
</select>
|
|
||||||
<!-- <div class="hint">이 영역은 게임 자체와 관리자 기본 아이템만 관리합니다. 여기서 아이템을 삭제해도 사용자 커스텀 이미지는 삭제되지 않아요.</div> -->
|
|
||||||
</template>
|
|
||||||
|
|
||||||
<template v-else>
|
|
||||||
<div class="panel__title">새 게임 정보 입력</div>
|
|
||||||
<input v-model="newGameId" class="input" placeholder="game id (영문/숫자)" />
|
|
||||||
<input v-model="newGameName" class="input" placeholder="게임 이름" />
|
|
||||||
<button class="btn btn--primary" @click="createGame">게임 생성</button>
|
|
||||||
</template>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div v-if="hasSelectedGame" class="panel">
|
<div v-if="hasSelectedGame" class="panel">
|
||||||
<div class="detailHead">
|
<div class="detailHead">
|
||||||
<div>
|
<div>
|
||||||
@@ -1039,41 +1080,18 @@ async function saveFeaturedOrder() {
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</template>
|
<div v-else class="panel panel--empty">
|
||||||
|
<div class="emptyState">
|
||||||
<template v-else-if="activeTab === 'items'">
|
<div class="emptyState__title">{{ gameMode === 'existing' ? '게임을 선택하면 상세 관리가 열려요.' : '새 게임 정보를 입력한 뒤 생성해 주세요.' }}</div>
|
||||||
<div class="panel">
|
<div class="emptyState__desc">
|
||||||
<div class="sectionHeader">
|
{{ gameMode === 'existing' ? '우측 패널에서 등록된 게임을 선택하면 썸네일과 기본 아이템 관리 영역이 활성화됩니다.' : '새 게임을 만들면 바로 선택 상태로 전환되어 썸네일과 기본 아이템 추가를 이어서 진행할 수 있어요.' }}
|
||||||
<div>
|
</div>
|
||||||
<div class="panel__title">사용자 커스텀 아이템 관리</div>
|
|
||||||
<div class="hint hint--tight">사용자가 업로드한 이미지를 파일명/라벨 기준으로 검색하고, 한 번에 50개 또는 200개씩 페이지 형태로 확인할 수 있어요.</div>
|
|
||||||
</div>
|
</div>
|
||||||
<button class="btn btn--ghost" @click="refreshCustomItems">새로고침</button>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="toolbar">
|
|
||||||
<input v-model="customItemQuery" class="input toolbar__search" placeholder="파일명, 라벨, 업로더 검색" @keydown.enter.prevent="submitCustomItemSearch" />
|
|
||||||
<button class="btn btn--ghost toolbar__button" @click="submitCustomItemSearch">검색</button>
|
|
||||||
<select :value="customItemLimit" class="select toolbar__select" @change="changeCustomItemLimit(Number($event.target.value))">
|
|
||||||
<option :value="50">50개씩 보기</option>
|
|
||||||
<option :value="200">200개씩 보기</option>
|
|
||||||
</select>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="toolbar toolbar--secondary">
|
|
||||||
<select v-model="customItemTargetGameId" class="select toolbar__select">
|
|
||||||
<option value="">가져올 게임 선택</option>
|
|
||||||
<option v-for="game in games" :key="game.id" :value="game.id">{{ game.name }}</option>
|
|
||||||
</select>
|
|
||||||
<label class="checkRow checkRow--toolbar">
|
|
||||||
<input v-model="customItemOrphanOnly" type="checkbox" @change="toggleCustomItemOrphanOnly" />
|
|
||||||
<span>미사용 커스텀 이미지만 보기</span>
|
|
||||||
</label>
|
|
||||||
<button class="btn btn--danger toolbar__button" :disabled="!customItems.length" @click="removeUnusedCustomItems">
|
|
||||||
미사용 이미지 일괄 삭제
|
|
||||||
</button>
|
|
||||||
</div>
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<template v-else-if="activeTab === 'items'">
|
||||||
|
<div class="panel">
|
||||||
<div v-if="!customItems.length" class="hint">조건에 맞는 커스텀 아이템이 없어요.</div>
|
<div v-if="!customItems.length" class="hint">조건에 맞는 커스텀 아이템이 없어요.</div>
|
||||||
<div v-else class="customItemGrid">
|
<div v-else class="customItemGrid">
|
||||||
<article v-for="item in customItems" :key="item.id" class="customItemCard">
|
<article v-for="item in customItems" :key="item.id" class="customItemCard">
|
||||||
@@ -1101,25 +1119,15 @@ async function saveFeaturedOrder() {
|
|||||||
<button class="btn btn--ghost" :disabled="customItemPage >= customItemPageCount" @click="moveCustomItemPage(1)">다음</button>
|
<button class="btn btn--ghost" :disabled="customItemPage >= customItemPageCount" @click="moveCustomItemPage(1)">다음</button>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
<template v-else-if="activeTab === 'tierlists'">
|
|
||||||
<div class="modeTabs modeTabs--admin">
|
|
||||||
<button class="modeTab" :class="{ 'modeTab--active': tierlistsMode === 'requests' }" @click="setTierlistsMode('requests')">
|
|
||||||
템플릿 요청 관리
|
|
||||||
</button>
|
|
||||||
<button class="modeTab" :class="{ 'modeTab--active': tierlistsMode === 'lists' }" @click="setTierlistsMode('lists')">
|
|
||||||
전체 티어표 관리
|
|
||||||
</button>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
|
<template v-else-if="activeTab === 'tierlists'">
|
||||||
<div v-if="tierlistsMode === 'requests'" class="panel">
|
<div v-if="tierlistsMode === 'requests'" class="panel">
|
||||||
<div class="sectionHeader">
|
<div class="sectionHeader">
|
||||||
<div>
|
<div>
|
||||||
<div class="panel__title">사용자 템플릿 요청</div>
|
<div class="panel__title">사용자 템플릿 요청</div>
|
||||||
<div class="hint hint--tight">freeform 템플릿 등록 요청과 기존 게임 템플릿 업데이트 요청을 여기서 승인하거나 반려할 수 있어요. 반려한 요청은 대기 목록에서 바로 제외됩니다.</div>
|
<div class="hint hint--tight">freeform 템플릿 등록 요청과 기존 게임 템플릿 업데이트 요청을 여기서 승인하거나 반려할 수 있어요. 반려한 요청은 대기 목록에서 바로 제외됩니다.</div>
|
||||||
</div>
|
</div>
|
||||||
<button class="btn btn--ghost" @click="refreshTemplateRequests">새로고침</button>
|
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div v-if="!templateRequests.length" class="hint">현재 처리 대기 중인 템플릿 요청이 없어요.</div>
|
<div v-if="!templateRequests.length" class="hint">현재 처리 대기 중인 템플릿 요청이 없어요.</div>
|
||||||
@@ -1166,21 +1174,6 @@ async function saveFeaturedOrder() {
|
|||||||
<div class="panel__title">전체 티어표 관리</div>
|
<div class="panel__title">전체 티어표 관리</div>
|
||||||
<div class="hint hint--tight">공개/비공개를 포함한 최근 티어표를 모두 확인하고, 추가 아이템을 기존 게임 템플릿으로 승격하거나 커스텀 티어표를 새 게임 템플릿으로 만들 수 있어요. 여기는 요청 목록과 별개로 전체 저장 티어표를 보는 영역입니다.</div>
|
<div class="hint hint--tight">공개/비공개를 포함한 최근 티어표를 모두 확인하고, 추가 아이템을 기존 게임 템플릿으로 승격하거나 커스텀 티어표를 새 게임 템플릿으로 만들 수 있어요. 여기는 요청 목록과 별개로 전체 저장 티어표를 보는 영역입니다.</div>
|
||||||
</div>
|
</div>
|
||||||
<button class="btn btn--ghost" @click="refreshAdminTierLists">새로고침</button>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="toolbar">
|
|
||||||
<input
|
|
||||||
v-model="adminTierListQuery"
|
|
||||||
class="input toolbar__search"
|
|
||||||
placeholder="제목, 작성자, 게임 이름 검색"
|
|
||||||
@keydown.enter.prevent="submitAdminTierListSearch"
|
|
||||||
/>
|
|
||||||
<button class="btn btn--ghost toolbar__button" @click="submitAdminTierListSearch">검색</button>
|
|
||||||
<select :value="adminTierListLimit" class="select toolbar__select" @change="changeAdminTierListLimit(Number($event.target.value))">
|
|
||||||
<option :value="50">50개씩 보기</option>
|
|
||||||
<option :value="200">200개씩 보기</option>
|
|
||||||
</select>
|
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div v-if="!adminTierLists.length" class="hint">조건에 맞는 티어표가 없어요.</div>
|
<div v-if="!adminTierLists.length" class="hint">조건에 맞는 티어표가 없어요.</div>
|
||||||
@@ -1233,68 +1226,15 @@ async function saveFeaturedOrder() {
|
|||||||
<button class="btn btn--ghost" :disabled="adminTierListPage >= adminTierListPageCount" @click="moveAdminTierListPage(1)">다음</button>
|
<button class="btn btn--ghost" :disabled="adminTierListPage >= adminTierListPageCount" @click="moveAdminTierListPage(1)">다음</button>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
<div v-if="importModalOpen" class="modalOverlay" @click.self="closeTierListImportModal">
|
<template v-else>
|
||||||
<div class="modalCard modalCard--import" role="dialog" aria-modal="true">
|
|
||||||
<div class="modalCard__title">티어표 아이템 가져오기</div>
|
|
||||||
<div class="modalCard__desc">
|
|
||||||
"{{ importModalTierList?.title }}"의 아이템 {{ importModalItemCount }}개를 어디로 가져올지 선택해주세요.
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="importModeTabs">
|
|
||||||
<button class="modeTab" :class="{ 'modeTab--active': importModalMode === 'existing' }" @click="importModalMode = 'existing'">
|
|
||||||
기존 템플릿에 추가
|
|
||||||
</button>
|
|
||||||
<button class="modeTab" :class="{ 'modeTab--active': importModalMode === 'new' }" @click="importModalMode = 'new'">
|
|
||||||
새 템플릿 만들기
|
|
||||||
</button>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div v-if="importModalMode === 'existing'" class="modalCard__form">
|
|
||||||
<select v-model="importModalTargetGameId" class="select">
|
|
||||||
<option value="">기존 게임 선택</option>
|
|
||||||
<option v-for="game in games" :key="game.id" :value="game.id">{{ game.name }}</option>
|
|
||||||
</select>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div v-else class="modalCard__form">
|
|
||||||
<input v-model="importModalNewGameId" class="input" placeholder="새 게임 ID" />
|
|
||||||
<input v-model="importModalNewGameName" class="input" placeholder="새 게임 이름" />
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="modalCard__actions">
|
|
||||||
<button class="btn btn--ghost" @click="closeTierListImportModal">취소</button>
|
|
||||||
<button class="btn btn--primary" @click="confirmTierListImport">
|
|
||||||
{{ importModalMode === 'existing' ? '여기로 가져오기' : '새 템플릿 생성' }}
|
|
||||||
</button>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div v-if="previewModalOpen" class="modalOverlay" @click.self="closePreviewModal">
|
|
||||||
<div class="modalCard modalCard--preview" role="dialog" aria-modal="true">
|
|
||||||
<div class="modalCard__titleRow">
|
|
||||||
<div class="modalCard__title">{{ previewTierList?.title || '티어표 미리보기' }}</div>
|
|
||||||
<button class="btn btn--ghost btn--small" @click="closePreviewModal">닫기</button>
|
|
||||||
</div>
|
|
||||||
<iframe
|
|
||||||
v-if="previewTierList"
|
|
||||||
class="previewFrame"
|
|
||||||
:src="previewTierListUrl(previewTierList)"
|
|
||||||
title="티어표 미리보기"
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</template>
|
|
||||||
|
|
||||||
<template v-else>
|
|
||||||
<div class="panel">
|
<div class="panel">
|
||||||
<div class="sectionHeader">
|
<div class="sectionHeader">
|
||||||
<div>
|
<div>
|
||||||
<div class="panel__title">회원 관리</div>
|
<div class="panel__title">회원 관리</div>
|
||||||
<div class="hint hint--tight">이메일, 닉네임, 관리자 권한을 수정하고 비밀번호도 직접 초기화할 수 있어요.</div>
|
<div class="hint hint--tight">이메일, 닉네임, 관리자 권한을 수정하고 비밀번호도 직접 초기화할 수 있어요.</div>
|
||||||
</div>
|
</div>
|
||||||
<button class="btn btn--ghost" @click="refreshUsers">새로고침</button>
|
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div v-if="!users.length" class="hint">아직 가입한 회원이 없어요.</div>
|
<div v-if="!users.length" class="hint">아직 가입한 회원이 없어요.</div>
|
||||||
@@ -1346,7 +1286,216 @@ async function saveFeaturedOrder() {
|
|||||||
</article>
|
</article>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
|
<div v-if="importModalOpen" class="modalOverlay" @click.self="closeTierListImportModal">
|
||||||
|
<div class="modalCard modalCard--import" role="dialog" aria-modal="true">
|
||||||
|
<div class="modalCard__title">티어표 아이템 가져오기</div>
|
||||||
|
<div class="modalCard__desc">
|
||||||
|
"{{ importModalTierList?.title }}"의 아이템 {{ importModalItemCount }}개를 어디로 가져올지 선택해주세요.
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="importModeTabs">
|
||||||
|
<button class="modeTab" :class="{ 'modeTab--active': importModalMode === 'existing' }" @click="importModalMode = 'existing'">
|
||||||
|
기존 템플릿에 추가
|
||||||
|
</button>
|
||||||
|
<button class="modeTab" :class="{ 'modeTab--active': importModalMode === 'new' }" @click="importModalMode = 'new'">
|
||||||
|
새 템플릿 만들기
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div v-if="importModalMode === 'existing'" class="modalCard__form">
|
||||||
|
<select v-model="importModalTargetGameId" class="select">
|
||||||
|
<option value="">기존 게임 선택</option>
|
||||||
|
<option v-for="game in games" :key="game.id" :value="game.id">{{ game.name }}</option>
|
||||||
|
</select>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div v-else class="modalCard__form">
|
||||||
|
<input v-model="importModalNewGameId" class="input" placeholder="새 게임 ID" />
|
||||||
|
<input v-model="importModalNewGameName" class="input" placeholder="새 게임 이름" />
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="modalCard__actions">
|
||||||
|
<button class="btn btn--ghost" @click="closeTierListImportModal">취소</button>
|
||||||
|
<button class="btn btn--primary" @click="confirmTierListImport">
|
||||||
|
{{ importModalMode === 'existing' ? '여기로 가져오기' : '새 템플릿 생성' }}
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div v-if="previewModalOpen" class="modalOverlay" @click.self="closePreviewModal">
|
||||||
|
<div class="modalCard modalCard--preview" role="dialog" aria-modal="true">
|
||||||
|
<div class="modalCard__titleRow">
|
||||||
|
<div class="modalCard__title">{{ previewTierList?.title || '티어표 미리보기' }}</div>
|
||||||
|
<button class="btn btn--ghost btn--small" @click="closePreviewModal">닫기</button>
|
||||||
|
</div>
|
||||||
|
<iframe
|
||||||
|
v-if="previewTierList"
|
||||||
|
class="previewFrame"
|
||||||
|
:src="previewTierListUrl(previewTierList)"
|
||||||
|
title="티어표 미리보기"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<aside class="adminSidebar">
|
||||||
|
<section class="adminSidebar__panel">
|
||||||
|
<div class="adminSidebar__label">Mode</div>
|
||||||
|
<div class="adminSidebar__tabs">
|
||||||
|
<button class="tab" :class="{ 'tab--active': activeTab === 'games' }" @click="setTab('games')">게임 관리</button>
|
||||||
|
<button class="tab" :class="{ 'tab--active': activeTab === 'items' }" @click="setTab('items')">아이템 관리</button>
|
||||||
|
<button class="tab" :class="{ 'tab--active': activeTab === 'tierlists' }" @click="setTab('tierlists')">티어표 관리</button>
|
||||||
|
<button class="tab" :class="{ 'tab--active': activeTab === 'users' }" @click="setTab('users')">회원 관리</button>
|
||||||
|
</div>
|
||||||
|
</section>
|
||||||
|
|
||||||
|
<section v-if="activeTab === 'games'" class="adminSidebar__panel">
|
||||||
|
<div class="adminSidebar__label">Game Flow</div>
|
||||||
|
<div class="modeTabs modeTabs--stack">
|
||||||
|
<button class="modeTab" :class="{ 'modeTab--active': gameMode === 'existing' }" @click="setGameMode('existing')">
|
||||||
|
등록된 게임 선택
|
||||||
|
</button>
|
||||||
|
<button class="modeTab" :class="{ 'modeTab--active': gameMode === 'new' }" @click="setGameMode('new')">
|
||||||
|
새 게임 추가
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div v-if="gameMode === 'existing'" class="adminSidebar__group">
|
||||||
|
<div class="adminSidebar__groupTitle">선택할 게임</div>
|
||||||
|
<select v-model="selectedGameId" class="select" @change="loadGame">
|
||||||
|
<option value="">게임을 선택해주세요</option>
|
||||||
|
<option v-for="game in games" :key="game.id" :value="game.id">{{ game.name }} ({{ game.id }})</option>
|
||||||
|
</select>
|
||||||
|
<button class="btn btn--ghost" @click="refreshGames">게임 목록 새로고침</button>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div v-else class="adminSidebar__group">
|
||||||
|
<div class="adminSidebar__groupTitle">새 게임 만들기</div>
|
||||||
|
<input v-model="newGameId" class="input" placeholder="game id (영문/숫자)" />
|
||||||
|
<input v-model="newGameName" class="input" placeholder="게임 이름" />
|
||||||
|
<button class="btn btn--primary" @click="createGame">게임 생성</button>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="adminSidebar__stats">
|
||||||
|
<div class="sidebarStat">
|
||||||
|
<span class="sidebarStat__label">전체 게임</span>
|
||||||
|
<strong class="sidebarStat__value">{{ games.length }}</strong>
|
||||||
|
</div>
|
||||||
|
<div class="sidebarStat">
|
||||||
|
<span class="sidebarStat__label">상단 고정</span>
|
||||||
|
<strong class="sidebarStat__value">{{ featuredGameIds.length }}/50</strong>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</section>
|
||||||
|
|
||||||
|
<section v-else-if="activeTab === 'items'" class="adminSidebar__panel">
|
||||||
|
<div class="adminSidebar__label">Filters</div>
|
||||||
|
<div class="adminSidebar__group">
|
||||||
|
<input v-model="customItemQuery" class="input" placeholder="파일명, 라벨, 업로더 검색" @keydown.enter.prevent="submitCustomItemSearch" />
|
||||||
|
<button class="btn btn--ghost" @click="submitCustomItemSearch">검색</button>
|
||||||
|
</div>
|
||||||
|
<div class="adminSidebar__group">
|
||||||
|
<select :value="customItemLimit" class="select" @change="changeCustomItemLimit(Number($event.target.value))">
|
||||||
|
<option :value="50">50개씩 보기</option>
|
||||||
|
<option :value="200">200개씩 보기</option>
|
||||||
|
</select>
|
||||||
|
<select v-model="customItemTargetGameId" class="select">
|
||||||
|
<option value="">가져올 게임 선택</option>
|
||||||
|
<option v-for="game in games" :key="game.id" :value="game.id">{{ game.name }}</option>
|
||||||
|
</select>
|
||||||
|
<label class="checkRow checkRow--compact">
|
||||||
|
<input v-model="customItemOrphanOnly" type="checkbox" @change="toggleCustomItemOrphanOnly" />
|
||||||
|
<span>미사용 커스텀 이미지만 보기</span>
|
||||||
|
</label>
|
||||||
|
</div>
|
||||||
|
<div class="adminSidebar__actions">
|
||||||
|
<button class="btn btn--ghost" @click="refreshCustomItems">새로고침</button>
|
||||||
|
<button class="btn btn--danger" :disabled="!customItems.length" @click="removeUnusedCustomItems">미사용 이미지 일괄 삭제</button>
|
||||||
|
</div>
|
||||||
|
<div class="adminSidebar__stats">
|
||||||
|
<div class="sidebarStat">
|
||||||
|
<span class="sidebarStat__label">현재 페이지</span>
|
||||||
|
<strong class="sidebarStat__value">{{ customItemPage }}/{{ customItemPageCount }}</strong>
|
||||||
|
</div>
|
||||||
|
<div class="sidebarStat">
|
||||||
|
<span class="sidebarStat__label">검색 결과</span>
|
||||||
|
<strong class="sidebarStat__value">{{ customItemTotal }}</strong>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</section>
|
||||||
|
|
||||||
|
<section v-else-if="activeTab === 'tierlists'" class="adminSidebar__panel">
|
||||||
|
<div class="adminSidebar__label">Tierlists</div>
|
||||||
|
<div class="modeTabs modeTabs--stack">
|
||||||
|
<button class="modeTab" :class="{ 'modeTab--active': tierlistsMode === 'requests' }" @click="setTierlistsMode('requests')">
|
||||||
|
템플릿 요청 관리
|
||||||
|
</button>
|
||||||
|
<button class="modeTab" :class="{ 'modeTab--active': tierlistsMode === 'lists' }" @click="setTierlistsMode('lists')">
|
||||||
|
전체 티어표 관리
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
<template v-if="tierlistsMode === 'requests'">
|
||||||
|
<div class="adminSidebar__actions">
|
||||||
|
<button class="btn btn--ghost" @click="refreshTemplateRequests">요청 새로고침</button>
|
||||||
|
</div>
|
||||||
|
<div class="adminSidebar__stats">
|
||||||
|
<div class="sidebarStat">
|
||||||
|
<span class="sidebarStat__label">대기 요청</span>
|
||||||
|
<strong class="sidebarStat__value">{{ templateRequests.length }}</strong>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
<template v-else>
|
||||||
|
<div class="adminSidebar__group">
|
||||||
|
<input
|
||||||
|
v-model="adminTierListQuery"
|
||||||
|
class="input"
|
||||||
|
placeholder="제목, 작성자, 게임 이름 검색"
|
||||||
|
@keydown.enter.prevent="submitAdminTierListSearch"
|
||||||
|
/>
|
||||||
|
<button class="btn btn--ghost" @click="submitAdminTierListSearch">검색</button>
|
||||||
|
<select :value="adminTierListLimit" class="select" @change="changeAdminTierListLimit(Number($event.target.value))">
|
||||||
|
<option :value="50">50개씩 보기</option>
|
||||||
|
<option :value="200">200개씩 보기</option>
|
||||||
|
</select>
|
||||||
|
</div>
|
||||||
|
<div class="adminSidebar__actions">
|
||||||
|
<button class="btn btn--ghost" @click="refreshAdminTierLists">새로고침</button>
|
||||||
|
</div>
|
||||||
|
<div class="adminSidebar__stats">
|
||||||
|
<div class="sidebarStat">
|
||||||
|
<span class="sidebarStat__label">현재 페이지</span>
|
||||||
|
<strong class="sidebarStat__value">{{ adminTierListPage }}/{{ adminTierListPageCount }}</strong>
|
||||||
|
</div>
|
||||||
|
<div class="sidebarStat">
|
||||||
|
<span class="sidebarStat__label">검색 결과</span>
|
||||||
|
<strong class="sidebarStat__value">{{ adminTierListTotal }}</strong>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
</section>
|
||||||
|
|
||||||
|
<section v-else class="adminSidebar__panel">
|
||||||
|
<div class="adminSidebar__label">Users</div>
|
||||||
|
<div class="adminSidebar__actions">
|
||||||
|
<button class="btn btn--ghost" @click="refreshUsers">회원 새로고침</button>
|
||||||
|
</div>
|
||||||
|
<div class="adminSidebar__stats">
|
||||||
|
<div class="sidebarStat">
|
||||||
|
<span class="sidebarStat__label">가입 회원</span>
|
||||||
|
<strong class="sidebarStat__value">{{ users.length }}</strong>
|
||||||
|
</div>
|
||||||
|
<div class="sidebarStat">
|
||||||
|
<span class="sidebarStat__label">관리자 수</span>
|
||||||
|
<strong class="sidebarStat__value">{{ users.filter((user) => user.isAdmin).length }}</strong>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</section>
|
||||||
|
</aside>
|
||||||
|
</div>
|
||||||
</template>
|
</template>
|
||||||
</div>
|
</div>
|
||||||
</section>
|
</section>
|
||||||
@@ -1357,6 +1506,124 @@ async function saveFeaturedOrder() {
|
|||||||
display: grid;
|
display: grid;
|
||||||
gap: 16px;
|
gap: 16px;
|
||||||
}
|
}
|
||||||
|
.adminWorkspace {
|
||||||
|
display: grid;
|
||||||
|
grid-template-columns: minmax(0, 1fr) 320px;
|
||||||
|
gap: 16px;
|
||||||
|
align-items: start;
|
||||||
|
}
|
||||||
|
.adminMain {
|
||||||
|
min-width: 0;
|
||||||
|
display: grid;
|
||||||
|
gap: 14px;
|
||||||
|
}
|
||||||
|
.adminHero {
|
||||||
|
display: grid;
|
||||||
|
gap: 10px;
|
||||||
|
padding: 22px 24px;
|
||||||
|
border-radius: 24px;
|
||||||
|
border: 1px solid rgba(255, 255, 255, 0.08);
|
||||||
|
background:
|
||||||
|
linear-gradient(180deg, rgba(255, 255, 255, 0.06), rgba(255, 255, 255, 0.02)),
|
||||||
|
rgba(255, 255, 255, 0.02);
|
||||||
|
box-shadow: 0 20px 50px rgba(0, 0, 0, 0.22);
|
||||||
|
}
|
||||||
|
.adminHero__eyebrow {
|
||||||
|
font-size: 11px;
|
||||||
|
letter-spacing: 0.12em;
|
||||||
|
text-transform: uppercase;
|
||||||
|
color: rgba(255, 255, 255, 0.42);
|
||||||
|
}
|
||||||
|
.adminHero__title {
|
||||||
|
margin: 0;
|
||||||
|
font-size: 28px;
|
||||||
|
line-height: 1.05;
|
||||||
|
font-weight: 900;
|
||||||
|
letter-spacing: -0.04em;
|
||||||
|
}
|
||||||
|
.adminHero__desc {
|
||||||
|
margin: 0;
|
||||||
|
color: rgba(255, 255, 255, 0.66);
|
||||||
|
line-height: 1.6;
|
||||||
|
}
|
||||||
|
.adminHero__stats {
|
||||||
|
display: grid;
|
||||||
|
grid-template-columns: repeat(3, minmax(0, 1fr));
|
||||||
|
gap: 12px;
|
||||||
|
margin-top: 6px;
|
||||||
|
}
|
||||||
|
.adminHeroStat {
|
||||||
|
display: grid;
|
||||||
|
gap: 6px;
|
||||||
|
padding: 14px 16px;
|
||||||
|
border-radius: 16px;
|
||||||
|
border: 1px solid rgba(255, 255, 255, 0.08);
|
||||||
|
background: rgba(7, 7, 7, 0.18);
|
||||||
|
}
|
||||||
|
.adminHeroStat__label {
|
||||||
|
font-size: 11px;
|
||||||
|
letter-spacing: 0.08em;
|
||||||
|
text-transform: uppercase;
|
||||||
|
color: rgba(255, 255, 255, 0.46);
|
||||||
|
}
|
||||||
|
.adminHeroStat__value {
|
||||||
|
font-size: 22px;
|
||||||
|
line-height: 1;
|
||||||
|
font-weight: 900;
|
||||||
|
letter-spacing: -0.04em;
|
||||||
|
}
|
||||||
|
.adminSidebar {
|
||||||
|
position: sticky;
|
||||||
|
top: 14px;
|
||||||
|
align-self: start;
|
||||||
|
display: grid;
|
||||||
|
gap: 12px;
|
||||||
|
}
|
||||||
|
.adminSidebar__panel {
|
||||||
|
display: grid;
|
||||||
|
gap: 12px;
|
||||||
|
padding: 16px;
|
||||||
|
border-radius: 22px;
|
||||||
|
border: 1px solid rgba(255, 255, 255, 0.08);
|
||||||
|
background:
|
||||||
|
linear-gradient(180deg, rgba(255, 255, 255, 0.04), rgba(255, 255, 255, 0.015)),
|
||||||
|
rgba(13, 13, 13, 0.94);
|
||||||
|
box-shadow: 0 18px 40px rgba(0, 0, 0, 0.18);
|
||||||
|
}
|
||||||
|
.adminSidebar__label {
|
||||||
|
font-size: 11px;
|
||||||
|
letter-spacing: 0.12em;
|
||||||
|
text-transform: uppercase;
|
||||||
|
color: rgba(255, 255, 255, 0.42);
|
||||||
|
}
|
||||||
|
.adminSidebar__tabs,
|
||||||
|
.adminSidebar__group,
|
||||||
|
.adminSidebar__actions,
|
||||||
|
.adminSidebar__stats {
|
||||||
|
display: grid;
|
||||||
|
gap: 10px;
|
||||||
|
}
|
||||||
|
.adminSidebar__groupTitle {
|
||||||
|
font-size: 13px;
|
||||||
|
font-weight: 800;
|
||||||
|
color: rgba(255, 255, 255, 0.84);
|
||||||
|
}
|
||||||
|
.sidebarStat {
|
||||||
|
display: grid;
|
||||||
|
gap: 4px;
|
||||||
|
padding: 12px 14px;
|
||||||
|
border-radius: 14px;
|
||||||
|
border: 1px solid rgba(255, 255, 255, 0.08);
|
||||||
|
background: rgba(255, 255, 255, 0.03);
|
||||||
|
}
|
||||||
|
.sidebarStat__label {
|
||||||
|
font-size: 12px;
|
||||||
|
color: rgba(255, 255, 255, 0.56);
|
||||||
|
}
|
||||||
|
.sidebarStat__value {
|
||||||
|
font-size: 14px;
|
||||||
|
font-weight: 900;
|
||||||
|
}
|
||||||
.card {
|
.card {
|
||||||
border: 0;
|
border: 0;
|
||||||
background: transparent;
|
background: transparent;
|
||||||
@@ -1393,31 +1660,72 @@ async function saveFeaturedOrder() {
|
|||||||
gap: 10px;
|
gap: 10px;
|
||||||
flex-wrap: wrap;
|
flex-wrap: wrap;
|
||||||
}
|
}
|
||||||
|
.modeTabs--stack {
|
||||||
|
display: grid;
|
||||||
|
gap: 8px;
|
||||||
|
}
|
||||||
.tab,
|
.tab,
|
||||||
.modeTab {
|
.modeTab {
|
||||||
padding: 10px 14px;
|
padding: 12px 14px;
|
||||||
border-radius: 10px;
|
border-radius: 14px;
|
||||||
border: 1px solid rgba(255, 255, 255, 0.1);
|
border: 1px solid rgba(255, 255, 255, 0.1);
|
||||||
background: rgba(255, 255, 255, 0.05);
|
background: rgba(255, 255, 255, 0.04);
|
||||||
color: rgba(255, 255, 255, 0.92);
|
color: rgba(255, 255, 255, 0.92);
|
||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
font-weight: 800;
|
font-weight: 800;
|
||||||
|
transition:
|
||||||
|
border-color 0.16s ease,
|
||||||
|
background 0.16s ease,
|
||||||
|
transform 0.16s ease;
|
||||||
|
}
|
||||||
|
.tab:hover,
|
||||||
|
.modeTab:hover {
|
||||||
|
border-color: rgba(255, 255, 255, 0.18);
|
||||||
|
background: rgba(255, 255, 255, 0.07);
|
||||||
|
transform: translateY(-1px);
|
||||||
}
|
}
|
||||||
.tab--active,
|
.tab--active,
|
||||||
.modeTab--active {
|
.modeTab--active {
|
||||||
background: rgba(255, 255, 255, 0.1);
|
background: rgba(96, 165, 250, 0.14);
|
||||||
border-color: rgba(255, 255, 255, 0.18);
|
border-color: rgba(96, 165, 250, 0.28);
|
||||||
|
color: rgba(239, 246, 255, 0.98);
|
||||||
|
}
|
||||||
|
.adminSidebar__tabs .tab,
|
||||||
|
.modeTabs--stack .modeTab {
|
||||||
|
width: 100%;
|
||||||
|
text-align: left;
|
||||||
}
|
}
|
||||||
.panel {
|
.panel {
|
||||||
margin-top: 14px;
|
|
||||||
border: 1px solid rgba(255, 255, 255, 0.1);
|
border: 1px solid rgba(255, 255, 255, 0.1);
|
||||||
background: rgba(48, 48, 48, 0.78);
|
background:
|
||||||
border-radius: 18px;
|
linear-gradient(180deg, rgba(255, 255, 255, 0.035), rgba(255, 255, 255, 0.018)),
|
||||||
padding: 16px;
|
rgba(34, 34, 34, 0.84);
|
||||||
|
border-radius: 24px;
|
||||||
|
padding: 18px;
|
||||||
|
box-shadow: 0 18px 44px rgba(0, 0, 0, 0.18);
|
||||||
|
}
|
||||||
|
.panel--empty {
|
||||||
|
min-height: 240px;
|
||||||
|
display: grid;
|
||||||
|
place-items: center;
|
||||||
}
|
}
|
||||||
.panel--compact {
|
.panel--compact {
|
||||||
max-width: 520px;
|
max-width: 520px;
|
||||||
}
|
}
|
||||||
|
.emptyState {
|
||||||
|
max-width: 520px;
|
||||||
|
display: grid;
|
||||||
|
gap: 8px;
|
||||||
|
text-align: center;
|
||||||
|
}
|
||||||
|
.emptyState__title {
|
||||||
|
font-size: 18px;
|
||||||
|
font-weight: 900;
|
||||||
|
}
|
||||||
|
.emptyState__desc {
|
||||||
|
color: rgba(255, 255, 255, 0.66);
|
||||||
|
line-height: 1.6;
|
||||||
|
}
|
||||||
.featuredOrderPanel {
|
.featuredOrderPanel {
|
||||||
margin-top: 14px;
|
margin-top: 14px;
|
||||||
display: grid;
|
display: grid;
|
||||||
@@ -1427,9 +1735,9 @@ async function saveFeaturedOrder() {
|
|||||||
.featuredOrderPanel__list,
|
.featuredOrderPanel__list,
|
||||||
.featuredOrderPanel__picker {
|
.featuredOrderPanel__picker {
|
||||||
border: 1px solid rgba(255, 255, 255, 0.1);
|
border: 1px solid rgba(255, 255, 255, 0.1);
|
||||||
background: rgba(255, 255, 255, 0.03);
|
background: rgba(255, 255, 255, 0.025);
|
||||||
border-radius: 16px;
|
border-radius: 18px;
|
||||||
padding: 14px;
|
padding: 16px;
|
||||||
}
|
}
|
||||||
.featuredList,
|
.featuredList,
|
||||||
.featuredPickerList {
|
.featuredPickerList {
|
||||||
@@ -1444,10 +1752,10 @@ async function saveFeaturedOrder() {
|
|||||||
gap: 12px;
|
gap: 12px;
|
||||||
justify-content: space-between;
|
justify-content: space-between;
|
||||||
align-items: center;
|
align-items: center;
|
||||||
padding: 12px;
|
padding: 14px;
|
||||||
border-radius: 14px;
|
border-radius: 16px;
|
||||||
border: 1px solid rgba(255, 255, 255, 0.1);
|
border: 1px solid rgba(255, 255, 255, 0.08);
|
||||||
background: rgba(0, 0, 0, 0.18);
|
background: rgba(0, 0, 0, 0.22);
|
||||||
}
|
}
|
||||||
.featuredCard__meta {
|
.featuredCard__meta {
|
||||||
display: flex;
|
display: flex;
|
||||||
@@ -1489,9 +1797,9 @@ async function saveFeaturedOrder() {
|
|||||||
gap: 10px;
|
gap: 10px;
|
||||||
justify-content: space-between;
|
justify-content: space-between;
|
||||||
align-items: center;
|
align-items: center;
|
||||||
padding: 12px;
|
padding: 14px;
|
||||||
border-radius: 14px;
|
border-radius: 16px;
|
||||||
border: 1px solid rgba(255, 255, 255, 0.1);
|
border: 1px solid rgba(255, 255, 255, 0.08);
|
||||||
background: rgba(0, 0, 0, 0.16);
|
background: rgba(0, 0, 0, 0.16);
|
||||||
color: rgba(255, 255, 255, 0.92);
|
color: rgba(255, 255, 255, 0.92);
|
||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
@@ -1521,9 +1829,9 @@ async function saveFeaturedOrder() {
|
|||||||
}
|
}
|
||||||
.adminCard {
|
.adminCard {
|
||||||
border: 1px solid rgba(255, 255, 255, 0.1);
|
border: 1px solid rgba(255, 255, 255, 0.1);
|
||||||
background: rgba(255, 255, 255, 0.03);
|
background: rgba(255, 255, 255, 0.025);
|
||||||
border-radius: 16px;
|
border-radius: 18px;
|
||||||
padding: 14px;
|
padding: 16px;
|
||||||
min-width: 0;
|
min-width: 0;
|
||||||
}
|
}
|
||||||
.sectionHeader {
|
.sectionHeader {
|
||||||
@@ -1566,8 +1874,8 @@ async function saveFeaturedOrder() {
|
|||||||
.input {
|
.input {
|
||||||
width: 100%;
|
width: 100%;
|
||||||
box-sizing: border-box;
|
box-sizing: border-box;
|
||||||
padding: 10px 12px;
|
padding: 11px 13px;
|
||||||
border-radius: 12px;
|
border-radius: 14px;
|
||||||
border: 1px solid rgba(255, 255, 255, 0.12);
|
border: 1px solid rgba(255, 255, 255, 0.12);
|
||||||
background: rgba(0, 0, 0, 0.18);
|
background: rgba(0, 0, 0, 0.18);
|
||||||
color: rgba(255, 255, 255, 0.92);
|
color: rgba(255, 255, 255, 0.92);
|
||||||
@@ -1609,8 +1917,8 @@ async function saveFeaturedOrder() {
|
|||||||
.btn {
|
.btn {
|
||||||
font-size: 12px;
|
font-size: 12px;
|
||||||
margin-top: 12px;
|
margin-top: 12px;
|
||||||
padding: 10px 12px;
|
padding: 11px 13px;
|
||||||
border-radius: 12px;
|
border-radius: 14px;
|
||||||
border: 1px solid rgba(255, 255, 255, 0.14);
|
border: 1px solid rgba(255, 255, 255, 0.14);
|
||||||
background: rgba(255, 255, 255, 0.06);
|
background: rgba(255, 255, 255, 0.06);
|
||||||
color: rgba(255, 255, 255, 0.92);
|
color: rgba(255, 255, 255, 0.92);
|
||||||
@@ -1618,6 +1926,15 @@ async function saveFeaturedOrder() {
|
|||||||
font-weight: 800;
|
font-weight: 800;
|
||||||
text-align: center;
|
text-align: center;
|
||||||
text-decoration: none;
|
text-decoration: none;
|
||||||
|
transition:
|
||||||
|
background 0.16s ease,
|
||||||
|
border-color 0.16s ease,
|
||||||
|
transform 0.16s ease;
|
||||||
|
}
|
||||||
|
.btn:hover:not(:disabled) {
|
||||||
|
transform: translateY(-1px);
|
||||||
|
border-color: rgba(255, 255, 255, 0.22);
|
||||||
|
background: rgba(255, 255, 255, 0.1);
|
||||||
}
|
}
|
||||||
.btn--small {
|
.btn--small {
|
||||||
margin-top: 0;
|
margin-top: 0;
|
||||||
@@ -1636,6 +1953,7 @@ async function saveFeaturedOrder() {
|
|||||||
}
|
}
|
||||||
.btn--primary {
|
.btn--primary {
|
||||||
background: rgba(96, 165, 250, 0.2);
|
background: rgba(96, 165, 250, 0.2);
|
||||||
|
border-color: rgba(96, 165, 250, 0.26);
|
||||||
}
|
}
|
||||||
.btn--danger {
|
.btn--danger {
|
||||||
background: rgba(239, 68, 68, 0.14);
|
background: rgba(239, 68, 68, 0.14);
|
||||||
@@ -1721,8 +2039,8 @@ async function saveFeaturedOrder() {
|
|||||||
flex-wrap: wrap;
|
flex-wrap: wrap;
|
||||||
}
|
}
|
||||||
.itemPreviewCard {
|
.itemPreviewCard {
|
||||||
padding: 10px;
|
padding: 12px;
|
||||||
border-radius: 14px;
|
border-radius: 16px;
|
||||||
border: 1px solid rgba(255, 255, 255, 0.12);
|
border: 1px solid rgba(255, 255, 255, 0.12);
|
||||||
background: rgba(255, 255, 255, 0.04);
|
background: rgba(255, 255, 255, 0.04);
|
||||||
}
|
}
|
||||||
@@ -1760,9 +2078,9 @@ async function saveFeaturedOrder() {
|
|||||||
}
|
}
|
||||||
.thumbCard {
|
.thumbCard {
|
||||||
border: 1px solid rgba(255, 255, 255, 0.12);
|
border: 1px solid rgba(255, 255, 255, 0.12);
|
||||||
border-radius: 14px;
|
border-radius: 16px;
|
||||||
background: rgba(255, 255, 255, 0.04);
|
background: rgba(255, 255, 255, 0.04);
|
||||||
padding: 10px;
|
padding: 12px;
|
||||||
min-width: 0;
|
min-width: 0;
|
||||||
}
|
}
|
||||||
.thumb {
|
.thumb {
|
||||||
@@ -1795,18 +2113,18 @@ async function saveFeaturedOrder() {
|
|||||||
.customItemGrid {
|
.customItemGrid {
|
||||||
margin-top: 14px;
|
margin-top: 14px;
|
||||||
display: grid;
|
display: grid;
|
||||||
grid-template-columns: repeat(2, minmax(0, 1fr));
|
grid-template-columns: repeat(auto-fit, minmax(280px, 1fr));
|
||||||
gap: 12px;
|
gap: 12px;
|
||||||
}
|
}
|
||||||
.customItemCard {
|
.customItemCard {
|
||||||
border: 1px solid rgba(255, 255, 255, 0.12);
|
border: 1px solid rgba(255, 255, 255, 0.12);
|
||||||
border-radius: 16px;
|
border-radius: 18px;
|
||||||
background: rgba(255, 255, 255, 0.04);
|
background: rgba(255, 255, 255, 0.04);
|
||||||
overflow: hidden;
|
overflow: hidden;
|
||||||
display: flex;
|
display: flex;
|
||||||
gap: 12px;
|
gap: 12px;
|
||||||
align-items: flex-start;
|
align-items: flex-start;
|
||||||
padding: 12px;
|
padding: 14px;
|
||||||
min-width: 0;
|
min-width: 0;
|
||||||
}
|
}
|
||||||
.customItemCard__image {
|
.customItemCard__image {
|
||||||
@@ -1864,9 +2182,9 @@ async function saveFeaturedOrder() {
|
|||||||
}
|
}
|
||||||
.userCard {
|
.userCard {
|
||||||
border: 1px solid rgba(255, 255, 255, 0.12);
|
border: 1px solid rgba(255, 255, 255, 0.12);
|
||||||
border-radius: 16px;
|
border-radius: 18px;
|
||||||
background: rgba(255, 255, 255, 0.04);
|
background: rgba(255, 255, 255, 0.04);
|
||||||
padding: 14px;
|
padding: 16px;
|
||||||
}
|
}
|
||||||
.userCard__head {
|
.userCard__head {
|
||||||
display: flex;
|
display: flex;
|
||||||
@@ -1917,8 +2235,8 @@ async function saveFeaturedOrder() {
|
|||||||
.userStat {
|
.userStat {
|
||||||
display: grid;
|
display: grid;
|
||||||
gap: 4px;
|
gap: 4px;
|
||||||
padding: 10px 12px;
|
padding: 12px 14px;
|
||||||
border-radius: 12px;
|
border-radius: 14px;
|
||||||
border: 1px solid rgba(255, 255, 255, 0.08);
|
border: 1px solid rgba(255, 255, 255, 0.08);
|
||||||
background: rgba(255, 255, 255, 0.03);
|
background: rgba(255, 255, 255, 0.03);
|
||||||
}
|
}
|
||||||
@@ -1959,8 +2277,8 @@ async function saveFeaturedOrder() {
|
|||||||
.templateRequestCard {
|
.templateRequestCard {
|
||||||
display: grid;
|
display: grid;
|
||||||
gap: 14px;
|
gap: 14px;
|
||||||
padding: 16px;
|
padding: 18px;
|
||||||
border-radius: 18px;
|
border-radius: 20px;
|
||||||
border: 1px solid rgba(255, 255, 255, 0.12);
|
border: 1px solid rgba(255, 255, 255, 0.12);
|
||||||
background: rgba(255, 255, 255, 0.03);
|
background: rgba(255, 255, 255, 0.03);
|
||||||
}
|
}
|
||||||
@@ -2025,9 +2343,9 @@ async function saveFeaturedOrder() {
|
|||||||
grid-template-columns: minmax(220px, 280px) minmax(0, 1fr);
|
grid-template-columns: minmax(220px, 280px) minmax(0, 1fr);
|
||||||
gap: 16px;
|
gap: 16px;
|
||||||
border: 1px solid rgba(255, 255, 255, 0.12);
|
border: 1px solid rgba(255, 255, 255, 0.12);
|
||||||
border-radius: 18px;
|
border-radius: 20px;
|
||||||
background: rgba(255, 255, 255, 0.03);
|
background: rgba(255, 255, 255, 0.03);
|
||||||
padding: 14px;
|
padding: 16px;
|
||||||
}
|
}
|
||||||
.tierAdminCard__preview {
|
.tierAdminCard__preview {
|
||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
@@ -2087,8 +2405,8 @@ async function saveFeaturedOrder() {
|
|||||||
.tierAdminSection {
|
.tierAdminSection {
|
||||||
display: grid;
|
display: grid;
|
||||||
gap: 10px;
|
gap: 10px;
|
||||||
padding: 12px;
|
padding: 14px;
|
||||||
border-radius: 14px;
|
border-radius: 16px;
|
||||||
border: 1px solid rgba(255, 255, 255, 0.08);
|
border: 1px solid rgba(255, 255, 255, 0.08);
|
||||||
background: rgba(0, 0, 0, 0.14);
|
background: rgba(0, 0, 0, 0.14);
|
||||||
}
|
}
|
||||||
@@ -2109,8 +2427,8 @@ async function saveFeaturedOrder() {
|
|||||||
display: grid;
|
display: grid;
|
||||||
gap: 8px;
|
gap: 8px;
|
||||||
justify-items: center;
|
justify-items: center;
|
||||||
padding: 10px;
|
padding: 12px 10px;
|
||||||
border-radius: 12px;
|
border-radius: 14px;
|
||||||
border: 1px solid rgba(255, 255, 255, 0.08);
|
border: 1px solid rgba(255, 255, 255, 0.08);
|
||||||
background: rgba(255, 255, 255, 0.03);
|
background: rgba(255, 255, 255, 0.03);
|
||||||
color: rgba(255, 255, 255, 0.92);
|
color: rgba(255, 255, 255, 0.92);
|
||||||
@@ -2146,8 +2464,8 @@ async function saveFeaturedOrder() {
|
|||||||
width: min(560px, 100%);
|
width: min(560px, 100%);
|
||||||
display: grid;
|
display: grid;
|
||||||
gap: 14px;
|
gap: 14px;
|
||||||
padding: 18px;
|
padding: 20px;
|
||||||
border-radius: 20px;
|
border-radius: 24px;
|
||||||
border: 1px solid rgba(255, 255, 255, 0.12);
|
border: 1px solid rgba(255, 255, 255, 0.12);
|
||||||
background: rgba(11, 18, 32, 0.96);
|
background: rgba(11, 18, 32, 0.96);
|
||||||
}
|
}
|
||||||
@@ -2198,10 +2516,23 @@ async function saveFeaturedOrder() {
|
|||||||
align-items: center;
|
align-items: center;
|
||||||
opacity: 0.88;
|
opacity: 0.88;
|
||||||
}
|
}
|
||||||
|
.checkRow--compact {
|
||||||
|
margin-top: 0;
|
||||||
|
}
|
||||||
.checkRow--toolbar {
|
.checkRow--toolbar {
|
||||||
margin-top: 0;
|
margin-top: 0;
|
||||||
}
|
}
|
||||||
@media (max-width: 980px) {
|
@media (max-width: 980px) {
|
||||||
|
.adminWorkspace {
|
||||||
|
grid-template-columns: 1fr;
|
||||||
|
}
|
||||||
|
.adminHero__stats {
|
||||||
|
grid-template-columns: 1fr;
|
||||||
|
}
|
||||||
|
.adminSidebar {
|
||||||
|
position: static;
|
||||||
|
order: -1;
|
||||||
|
}
|
||||||
.featuredOrderPanel,
|
.featuredOrderPanel,
|
||||||
.section--topGrid,
|
.section--topGrid,
|
||||||
.toolbar,
|
.toolbar,
|
||||||
@@ -2222,6 +2553,12 @@ async function saveFeaturedOrder() {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
@media (max-width: 640px) {
|
@media (max-width: 640px) {
|
||||||
|
.adminHero {
|
||||||
|
padding: 16px;
|
||||||
|
}
|
||||||
|
.adminHero__title {
|
||||||
|
font-size: 24px;
|
||||||
|
}
|
||||||
.thumbGrid,
|
.thumbGrid,
|
||||||
.customItemGrid,
|
.customItemGrid,
|
||||||
.userList {
|
.userList {
|
||||||
|
|||||||
@@ -62,6 +62,7 @@ onMounted(loadFavorites)
|
|||||||
<section class="wrap">
|
<section class="wrap">
|
||||||
<div class="head">
|
<div class="head">
|
||||||
<div>
|
<div>
|
||||||
|
<div class="head__eyebrow">Collection</div>
|
||||||
<h2 class="title">내 즐겨찾기</h2>
|
<h2 class="title">내 즐겨찾기</h2>
|
||||||
<div class="desc">마음에 드는 티어표를 모아보고, 원하는 기준으로 정렬할 수 있어요.</div>
|
<div class="desc">마음에 드는 티어표를 모아보고, 원하는 기준으로 정렬할 수 있어요.</div>
|
||||||
</div>
|
</div>
|
||||||
@@ -78,23 +79,23 @@ onMounted(loadFavorites)
|
|||||||
|
|
||||||
<div v-if="favorites.length === 0" class="empty">즐겨찾기한 티어표가 없어요.</div>
|
<div v-if="favorites.length === 0" class="empty">즐겨찾기한 티어표가 없어요.</div>
|
||||||
<div v-else class="list">
|
<div v-else class="list">
|
||||||
<article v-for="tierList in favorites" :key="tierList.id" class="row">
|
<article v-for="tierList in favorites" :key="tierList.id" class="boardCard">
|
||||||
<button class="row__body" @click="openTierList(tierList)">
|
<button class="boardCard__body" @click="openTierList(tierList)">
|
||||||
<div class="row__thumbWrap">
|
<div class="boardCard__thumbWrap">
|
||||||
<img v-if="tierListThumbnailUrl(tierList)" class="row__thumb" :src="tierListThumbnailUrl(tierList)" :alt="tierList.title" />
|
<img v-if="tierListThumbnailUrl(tierList)" class="boardCard__thumb" :src="tierListThumbnailUrl(tierList)" :alt="tierList.title" />
|
||||||
<div v-else class="row__thumbPlaceholder"></div>
|
<div v-else class="boardCard__thumbPlaceholder">대표 썸네일</div>
|
||||||
</div>
|
</div>
|
||||||
<div class="row__head">
|
<div class="boardCard__head">
|
||||||
<div class="row__title">{{ tierList.title }}</div>
|
<div class="boardCard__title">{{ tierList.title }}</div>
|
||||||
<div class="row__author">
|
<div class="boardCard__author">
|
||||||
<img v-if="avatarSrcOf(tierList)" class="row__avatar" :src="avatarSrcOf(tierList)" :alt="displayNameOf(tierList)" />
|
<img v-if="avatarSrcOf(tierList)" class="boardCard__avatar" :src="avatarSrcOf(tierList)" :alt="displayNameOf(tierList)" />
|
||||||
<div v-else class="row__avatar row__avatar--fallback">{{ avatarFallbackOf(tierList) }}</div>
|
<div v-else class="boardCard__avatar boardCard__avatar--fallback">{{ avatarFallbackOf(tierList) }}</div>
|
||||||
<span>by {{ displayNameOf(tierList) }}</span>
|
<span>by {{ displayNameOf(tierList) }}</span>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</button>
|
</button>
|
||||||
<div class="row__foot">
|
<div class="boardCard__foot">
|
||||||
<div class="row__meta">
|
<div class="boardCard__meta">
|
||||||
<div>{{ tierList.gameName || tierList.gameId }}</div>
|
<div>{{ tierList.gameName || tierList.gameId }}</div>
|
||||||
<div>{{ sortLabel }}: {{ fmt(sort === 'favorited' ? tierList.favoritedAt : tierList.updatedAt) }}</div>
|
<div>{{ sortLabel }}: {{ fmt(sort === 'favorited' ? tierList.favoritedAt : tierList.updatedAt) }}</div>
|
||||||
</div>
|
</div>
|
||||||
@@ -116,10 +117,17 @@ onMounted(loadFavorites)
|
|||||||
justify-content: space-between;
|
justify-content: space-between;
|
||||||
align-items: flex-end;
|
align-items: flex-end;
|
||||||
flex-wrap: wrap;
|
flex-wrap: wrap;
|
||||||
|
padding: 6px 2px 8px;
|
||||||
|
}
|
||||||
|
.head__eyebrow {
|
||||||
|
font-size: 11px;
|
||||||
|
letter-spacing: 0.12em;
|
||||||
|
text-transform: uppercase;
|
||||||
|
color: rgba(255, 255, 255, 0.42);
|
||||||
}
|
}
|
||||||
.title {
|
.title {
|
||||||
margin: 0;
|
margin: 4px 0 0;
|
||||||
font-size: 30px;
|
font-size: 32px;
|
||||||
color: rgba(255, 255, 255, 0.96);
|
color: rgba(255, 255, 255, 0.96);
|
||||||
letter-spacing: -0.04em;
|
letter-spacing: -0.04em;
|
||||||
}
|
}
|
||||||
@@ -134,15 +142,15 @@ onMounted(loadFavorites)
|
|||||||
}
|
}
|
||||||
.input,
|
.input,
|
||||||
.select {
|
.select {
|
||||||
padding: 10px 12px;
|
padding: 11px 13px;
|
||||||
border-radius: 10px;
|
border-radius: 14px;
|
||||||
border: 1px solid rgba(255, 255, 255, 0.08);
|
border: 1px solid rgba(255, 255, 255, 0.08);
|
||||||
background: rgba(255, 255, 255, 0.05);
|
background: rgba(255, 255, 255, 0.05);
|
||||||
color: rgba(255, 255, 255, 0.92);
|
color: rgba(255, 255, 255, 0.92);
|
||||||
}
|
}
|
||||||
.btn {
|
.btn {
|
||||||
padding: 10px 12px;
|
padding: 11px 13px;
|
||||||
border-radius: 10px;
|
border-radius: 14px;
|
||||||
border: 1px solid rgba(255, 255, 255, 0.08);
|
border: 1px solid rgba(255, 255, 255, 0.08);
|
||||||
background: rgba(255, 255, 255, 0.06);
|
background: rgba(255, 255, 255, 0.06);
|
||||||
color: rgba(255, 255, 255, 0.92);
|
color: rgba(255, 255, 255, 0.92);
|
||||||
@@ -157,15 +165,23 @@ onMounted(loadFavorites)
|
|||||||
grid-template-columns: repeat(3, minmax(0, 1fr));
|
grid-template-columns: repeat(3, minmax(0, 1fr));
|
||||||
gap: 18px;
|
gap: 18px;
|
||||||
}
|
}
|
||||||
.row {
|
.boardCard {
|
||||||
border-radius: 14px;
|
border-radius: 22px;
|
||||||
border: 1px solid rgba(255, 255, 255, 0.16);
|
border: 1px solid rgba(255, 255, 255, 0.16);
|
||||||
background: rgba(62, 62, 62, 0.82);
|
background: rgba(62, 62, 62, 0.82);
|
||||||
overflow: hidden;
|
overflow: hidden;
|
||||||
display: grid;
|
display: grid;
|
||||||
gap: 10px;
|
gap: 10px;
|
||||||
|
box-shadow: inset 0 1px 0 rgba(255, 255, 255, 0.04);
|
||||||
|
transition:
|
||||||
|
transform 0.16s ease,
|
||||||
|
background 0.16s ease;
|
||||||
}
|
}
|
||||||
.row__body {
|
.boardCard:hover {
|
||||||
|
transform: translateY(-2px);
|
||||||
|
background: rgba(70, 70, 70, 0.96);
|
||||||
|
}
|
||||||
|
.boardCard__body {
|
||||||
border: 0;
|
border: 0;
|
||||||
background: transparent;
|
background: transparent;
|
||||||
color: inherit;
|
color: inherit;
|
||||||
@@ -175,60 +191,67 @@ onMounted(loadFavorites)
|
|||||||
display: grid;
|
display: grid;
|
||||||
gap: 10px;
|
gap: 10px;
|
||||||
}
|
}
|
||||||
.row__thumbWrap {
|
.boardCard__thumbWrap {
|
||||||
width: 100%;
|
width: 100%;
|
||||||
aspect-ratio: 16 / 9;
|
aspect-ratio: 16 / 9;
|
||||||
background: #555;
|
background: #555;
|
||||||
|
display: grid;
|
||||||
|
place-items: center;
|
||||||
}
|
}
|
||||||
.row__thumb,
|
.boardCard__thumb,
|
||||||
.row__thumbPlaceholder {
|
.boardCard__thumbPlaceholder {
|
||||||
width: 100%;
|
width: 100%;
|
||||||
height: 100%;
|
height: 100%;
|
||||||
display: block;
|
display: block;
|
||||||
}
|
}
|
||||||
.row__thumb {
|
.boardCard__thumb {
|
||||||
object-fit: cover;
|
object-fit: cover;
|
||||||
}
|
}
|
||||||
.row__thumbPlaceholder {
|
.boardCard__thumbPlaceholder {
|
||||||
background: #555;
|
background: #555;
|
||||||
|
display: grid;
|
||||||
|
place-items: center;
|
||||||
|
color: rgba(255, 255, 255, 0.4);
|
||||||
|
font-size: 13px;
|
||||||
|
font-weight: 700;
|
||||||
}
|
}
|
||||||
.row__head {
|
.boardCard__head {
|
||||||
padding: 14px 14px 0;
|
padding: 14px 14px 0;
|
||||||
display: grid;
|
display: grid;
|
||||||
gap: 10px;
|
gap: 10px;
|
||||||
}
|
}
|
||||||
.row__title {
|
.boardCard__title {
|
||||||
font-weight: 800;
|
font-weight: 800;
|
||||||
font-size: 18px;
|
font-size: 18px;
|
||||||
}
|
}
|
||||||
.row__author {
|
.boardCard__author {
|
||||||
display: inline-flex;
|
display: inline-flex;
|
||||||
align-items: center;
|
align-items: center;
|
||||||
gap: 8px;
|
gap: 8px;
|
||||||
font-size: 13px;
|
font-size: 13px;
|
||||||
opacity: 0.86;
|
opacity: 0.86;
|
||||||
}
|
}
|
||||||
.row__avatar {
|
.boardCard__avatar {
|
||||||
width: 28px;
|
width: 28px;
|
||||||
height: 28px;
|
height: 28px;
|
||||||
border-radius: 999px;
|
border-radius: 999px;
|
||||||
object-fit: cover;
|
object-fit: cover;
|
||||||
background: rgba(255, 255, 255, 0.08);
|
background: rgba(255, 255, 255, 0.08);
|
||||||
}
|
}
|
||||||
.row__avatar--fallback {
|
.boardCard__avatar--fallback {
|
||||||
display: grid;
|
display: grid;
|
||||||
place-items: center;
|
place-items: center;
|
||||||
font-size: 12px;
|
font-size: 12px;
|
||||||
font-weight: 900;
|
font-weight: 900;
|
||||||
}
|
}
|
||||||
.row__foot {
|
.boardCard__foot {
|
||||||
padding: 0 14px 14px;
|
padding: 0 14px 14px;
|
||||||
display: flex;
|
display: flex;
|
||||||
gap: 12px;
|
gap: 12px;
|
||||||
align-items: center;
|
align-items: center;
|
||||||
justify-content: space-between;
|
justify-content: space-between;
|
||||||
}
|
}
|
||||||
.row__meta {
|
.boardCard__meta {
|
||||||
display: grid;
|
display: grid;
|
||||||
gap: 4px;
|
gap: 4px;
|
||||||
opacity: 0.78;
|
opacity: 0.78;
|
||||||
|
|||||||
@@ -77,13 +77,17 @@ function submitSearch() {
|
|||||||
</script>
|
</script>
|
||||||
|
|
||||||
<template>
|
<template>
|
||||||
<section class="head">
|
<section class="dashboardHero">
|
||||||
<div class="head__left">
|
<div class="dashboardHero__left">
|
||||||
<div class="kicker">Collection</div>
|
<div class="dashboardHero__eyebrow">Collection</div>
|
||||||
<h2 class="title">{{ gameName || gameId }}</h2>
|
<h2 class="dashboardHero__title">{{ gameName || gameId }}</h2>
|
||||||
<p class="desc">새 티어표를 만들거나, 다른 사람들이 올린 티어표를 카드형 목록으로 탐색할 수 있어요.</p>
|
<p class="dashboardHero__desc">이 게임의 공개 티어표를 탐색하고, 바로 새 보드를 만들어 같은 흐름으로 이어갈 수 있어요.</p>
|
||||||
</div>
|
</div>
|
||||||
<div class="head__right">
|
<div class="dashboardHero__right">
|
||||||
|
<div class="dashboardStat">
|
||||||
|
<span class="dashboardStat__label">Visible Lists</span>
|
||||||
|
<strong class="dashboardStat__value">{{ tierLists.length }}</strong>
|
||||||
|
</div>
|
||||||
<button class="primary" @click="createNew">{{ auth.user ? '새로운 티어표 만들기' : '로그인 후 새 티어표 만들기' }}</button>
|
<button class="primary" @click="createNew">{{ auth.user ? '새로운 티어표 만들기' : '로그인 후 새 티어표 만들기' }}</button>
|
||||||
</div>
|
</div>
|
||||||
</section>
|
</section>
|
||||||
@@ -91,7 +95,10 @@ function submitSearch() {
|
|||||||
<div v-if="error" class="error">{{ error }}</div>
|
<div v-if="error" class="error">{{ error }}</div>
|
||||||
<section class="panel">
|
<section class="panel">
|
||||||
<div class="panel__head">
|
<div class="panel__head">
|
||||||
<div class="panel__title">공개 티어표</div>
|
<div>
|
||||||
|
<div class="panel__title">공개 티어표</div>
|
||||||
|
<div class="panel__sub">제목이나 작성자로 빠르게 좁혀볼 수 있어요.</div>
|
||||||
|
</div>
|
||||||
<div class="searchBar">
|
<div class="searchBar">
|
||||||
<input v-model="query" class="searchBar__input" placeholder="제목 또는 작성자 검색" @keydown.enter.prevent="submitSearch" />
|
<input v-model="query" class="searchBar__input" placeholder="제목 또는 작성자 검색" @keydown.enter.prevent="submitSearch" />
|
||||||
<button class="searchBar__button" @click="submitSearch">검색</button>
|
<button class="searchBar__button" @click="submitSearch">검색</button>
|
||||||
@@ -99,23 +106,23 @@ function submitSearch() {
|
|||||||
</div>
|
</div>
|
||||||
<div v-if="tierLists.length === 0" class="empty">아직 공개 티어표가 없어요.</div>
|
<div v-if="tierLists.length === 0" class="empty">아직 공개 티어표가 없어요.</div>
|
||||||
<div v-else class="list">
|
<div v-else class="list">
|
||||||
<article v-for="t in tierLists" :key="t.id" class="row">
|
<article v-for="t in tierLists" :key="t.id" class="boardCard">
|
||||||
<button class="row__body" @click="openTierList(t.id)">
|
<button class="boardCard__body" @click="openTierList(t.id)">
|
||||||
<div class="row__thumbWrap">
|
<div class="boardCard__thumbWrap">
|
||||||
<img v-if="tierListThumbnailUrl(t)" class="row__thumb" :src="tierListThumbnailUrl(t)" :alt="t.title" />
|
<img v-if="tierListThumbnailUrl(t)" class="boardCard__thumb" :src="tierListThumbnailUrl(t)" :alt="t.title" />
|
||||||
<div v-else class="row__thumbPlaceholder"></div>
|
<div v-else class="boardCard__thumbPlaceholder">대표 썸네일</div>
|
||||||
</div>
|
</div>
|
||||||
<div class="row__head">
|
<div class="boardCard__head">
|
||||||
<div class="row__title">{{ t.title }}</div>
|
<div class="boardCard__title">{{ t.title }}</div>
|
||||||
<div class="row__author">
|
<div class="boardCard__author">
|
||||||
<img v-if="avatarSrcOf(t)" class="row__avatar" :src="avatarSrcOf(t)" :alt="displayNameOf(t)" />
|
<img v-if="avatarSrcOf(t)" class="boardCard__avatar" :src="avatarSrcOf(t)" :alt="displayNameOf(t)" />
|
||||||
<div v-else class="row__avatar row__avatar--fallback">{{ avatarFallbackOf(t) }}</div>
|
<div v-else class="boardCard__avatar boardCard__avatar--fallback">{{ avatarFallbackOf(t) }}</div>
|
||||||
<span>by {{ displayNameOf(t) }}</span>
|
<span>by {{ displayNameOf(t) }}</span>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</button>
|
</button>
|
||||||
<div class="row__foot">
|
<div class="boardCard__foot">
|
||||||
<div class="row__meta">{{ fmt(t.updatedAt) }}</div>
|
<div class="boardCard__meta">{{ fmt(t.updatedAt) }}</div>
|
||||||
<div class="favoriteStat" :title="t.isFavorited ? '이미 즐겨찾기한 티어표' : '즐겨찾기 수'">
|
<div class="favoriteStat" :title="t.isFavorited ? '이미 즐겨찾기한 티어표' : '즐겨찾기 수'">
|
||||||
{{ t.isFavorited ? '★' : '☆' }} {{ t.favoriteCount || 0 }}
|
{{ t.isFavorited ? '★' : '☆' }} {{ t.favoriteCount || 0 }}
|
||||||
</div>
|
</div>
|
||||||
@@ -126,44 +133,78 @@ function submitSearch() {
|
|||||||
</template>
|
</template>
|
||||||
|
|
||||||
<style scoped>
|
<style scoped>
|
||||||
.head {
|
.dashboardHero {
|
||||||
display: flex;
|
display: flex;
|
||||||
gap: 18px;
|
gap: 18px;
|
||||||
align-items: flex-end;
|
align-items: flex-start;
|
||||||
justify-content: space-between;
|
justify-content: space-between;
|
||||||
flex-wrap: wrap;
|
flex-wrap: wrap;
|
||||||
padding: 4px 2px 18px;
|
padding: 6px 2px 18px;
|
||||||
}
|
}
|
||||||
.kicker {
|
.dashboardHero__left {
|
||||||
|
display: grid;
|
||||||
|
gap: 8px;
|
||||||
|
}
|
||||||
|
.dashboardHero__right {
|
||||||
|
display: flex;
|
||||||
|
gap: 10px;
|
||||||
|
align-items: center;
|
||||||
|
flex-wrap: wrap;
|
||||||
|
}
|
||||||
|
.dashboardHero__eyebrow {
|
||||||
font-size: 12px;
|
font-size: 12px;
|
||||||
color: rgba(255, 255, 255, 0.42);
|
color: rgba(255, 255, 255, 0.42);
|
||||||
text-transform: uppercase;
|
text-transform: uppercase;
|
||||||
letter-spacing: 0.08em;
|
letter-spacing: 0.08em;
|
||||||
}
|
}
|
||||||
.title {
|
.dashboardHero__title {
|
||||||
margin: 4px 0 6px;
|
margin: 4px 0 6px;
|
||||||
font-size: 30px;
|
font-size: 32px;
|
||||||
letter-spacing: -0.04em;
|
letter-spacing: -0.04em;
|
||||||
color: rgba(255, 255, 255, 0.96);
|
color: rgba(255, 255, 255, 0.96);
|
||||||
}
|
}
|
||||||
.desc {
|
.dashboardHero__desc {
|
||||||
margin: 0;
|
margin: 0;
|
||||||
color: rgba(255, 255, 255, 0.58);
|
color: rgba(255, 255, 255, 0.58);
|
||||||
|
max-width: 720px;
|
||||||
|
}
|
||||||
|
.dashboardStat {
|
||||||
|
display: grid;
|
||||||
|
gap: 2px;
|
||||||
|
min-width: 112px;
|
||||||
|
padding: 12px 14px;
|
||||||
|
border-radius: 14px;
|
||||||
|
border: 1px solid rgba(255, 255, 255, 0.08);
|
||||||
|
background: rgba(255, 255, 255, 0.045);
|
||||||
|
}
|
||||||
|
.dashboardStat__label {
|
||||||
|
font-size: 11px;
|
||||||
|
color: rgba(255, 255, 255, 0.48);
|
||||||
|
text-transform: uppercase;
|
||||||
|
letter-spacing: 0.08em;
|
||||||
|
}
|
||||||
|
.dashboardStat__value {
|
||||||
|
font-size: 18px;
|
||||||
|
font-weight: 900;
|
||||||
}
|
}
|
||||||
.primary {
|
.primary {
|
||||||
padding: 10px 14px;
|
padding: 12px 16px;
|
||||||
border-radius: 10px;
|
border-radius: 14px;
|
||||||
border: 1px solid rgba(255, 255, 255, 0.08);
|
border: 1px solid rgba(77, 127, 233, 0.96);
|
||||||
background: rgba(255, 255, 255, 0.08);
|
background: rgba(77, 127, 233, 0.88);
|
||||||
color: rgba(255, 255, 255, 0.92);
|
color: #fff;
|
||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
font-weight: 700;
|
font-weight: 800;
|
||||||
|
transition:
|
||||||
|
transform 0.16s ease,
|
||||||
|
background 0.16s ease,
|
||||||
|
border-color 0.16s ease;
|
||||||
}
|
}
|
||||||
.primary:hover {
|
.primary:hover {
|
||||||
background: rgba(255, 255, 255, 0.12);
|
transform: translateY(-1px);
|
||||||
}
|
}
|
||||||
.panel {
|
.panel {
|
||||||
border: 1px solid rgba(255, 255, 255, 0.08);
|
/* border: 1px solid rgba(255, 255, 255, 0.08); */
|
||||||
background: transparent;
|
background: transparent;
|
||||||
border-radius: 0;
|
border-radius: 0;
|
||||||
padding: 0;
|
padding: 0;
|
||||||
@@ -177,6 +218,12 @@ function submitSearch() {
|
|||||||
}
|
}
|
||||||
.panel__title {
|
.panel__title {
|
||||||
font-weight: 800;
|
font-weight: 800;
|
||||||
|
font-size: 18px;
|
||||||
|
}
|
||||||
|
.panel__sub {
|
||||||
|
margin-top: 6px;
|
||||||
|
color: rgba(255, 255, 255, 0.56);
|
||||||
|
font-size: 13px;
|
||||||
}
|
}
|
||||||
.panel__head {
|
.panel__head {
|
||||||
display: flex;
|
display: flex;
|
||||||
@@ -194,15 +241,15 @@ function submitSearch() {
|
|||||||
}
|
}
|
||||||
.searchBar__input {
|
.searchBar__input {
|
||||||
min-width: 240px;
|
min-width: 240px;
|
||||||
padding: 10px 12px;
|
padding: 11px 13px;
|
||||||
border-radius: 10px;
|
border-radius: 14px;
|
||||||
border: 1px solid rgba(255, 255, 255, 0.08);
|
border: 1px solid rgba(255, 255, 255, 0.08);
|
||||||
background: rgba(255, 255, 255, 0.05);
|
background: rgba(255, 255, 255, 0.05);
|
||||||
color: rgba(255, 255, 255, 0.92);
|
color: rgba(255, 255, 255, 0.92);
|
||||||
}
|
}
|
||||||
.searchBar__button {
|
.searchBar__button {
|
||||||
padding: 10px 12px;
|
padding: 11px 14px;
|
||||||
border-radius: 10px;
|
border-radius: 14px;
|
||||||
border: 1px solid rgba(255, 255, 255, 0.08);
|
border: 1px solid rgba(255, 255, 255, 0.08);
|
||||||
background: rgba(255, 255, 255, 0.06);
|
background: rgba(255, 255, 255, 0.06);
|
||||||
color: rgba(255, 255, 255, 0.92);
|
color: rgba(255, 255, 255, 0.92);
|
||||||
@@ -217,8 +264,8 @@ function submitSearch() {
|
|||||||
grid-template-columns: repeat(3, minmax(0, 1fr));
|
grid-template-columns: repeat(3, minmax(0, 1fr));
|
||||||
gap: 18px;
|
gap: 18px;
|
||||||
}
|
}
|
||||||
.row {
|
.boardCard {
|
||||||
border-radius: 14px;
|
border-radius: 22px;
|
||||||
border: 1px solid rgba(255, 255, 255, 0.16);
|
border: 1px solid rgba(255, 255, 255, 0.16);
|
||||||
background: rgba(62, 62, 62, 0.82);
|
background: rgba(62, 62, 62, 0.82);
|
||||||
color: rgba(255, 255, 255, 0.92);
|
color: rgba(255, 255, 255, 0.92);
|
||||||
@@ -227,11 +274,16 @@ function submitSearch() {
|
|||||||
align-content: start;
|
align-content: start;
|
||||||
min-height: 168px;
|
min-height: 168px;
|
||||||
overflow: hidden;
|
overflow: hidden;
|
||||||
|
box-shadow: inset 0 1px 0 rgba(255, 255, 255, 0.04);
|
||||||
|
transition:
|
||||||
|
transform 0.16s ease,
|
||||||
|
background 0.16s ease;
|
||||||
}
|
}
|
||||||
.row:hover {
|
.boardCard:hover {
|
||||||
background: rgba(255, 255, 255, 0.05);
|
background: rgba(70, 70, 70, 0.96);
|
||||||
|
transform: translateY(-2px);
|
||||||
}
|
}
|
||||||
.row__body {
|
.boardCard__body {
|
||||||
text-align: left;
|
text-align: left;
|
||||||
padding: 0;
|
padding: 0;
|
||||||
border: 0;
|
border: 0;
|
||||||
@@ -242,35 +294,42 @@ function submitSearch() {
|
|||||||
display: grid;
|
display: grid;
|
||||||
gap: 10px;
|
gap: 10px;
|
||||||
}
|
}
|
||||||
.row__thumbWrap {
|
.boardCard__thumbWrap {
|
||||||
width: 100%;
|
width: 100%;
|
||||||
aspect-ratio: 16 / 9;
|
aspect-ratio: 16 / 9;
|
||||||
background: #555;
|
background: #555;
|
||||||
|
display: grid;
|
||||||
|
place-items: center;
|
||||||
}
|
}
|
||||||
.row__thumb {
|
.boardCard__thumb {
|
||||||
width: 100%;
|
width: 100%;
|
||||||
height: 100%;
|
height: 100%;
|
||||||
object-fit: cover;
|
object-fit: cover;
|
||||||
display: block;
|
display: block;
|
||||||
}
|
}
|
||||||
.row__thumbPlaceholder {
|
.boardCard__thumbPlaceholder {
|
||||||
width: 100%;
|
width: 100%;
|
||||||
height: 100%;
|
height: 100%;
|
||||||
background: #555;
|
background: #555;
|
||||||
|
display: grid;
|
||||||
|
place-items: center;
|
||||||
|
color: rgba(255, 255, 255, 0.4);
|
||||||
|
font-size: 13px;
|
||||||
|
font-weight: 700;
|
||||||
}
|
}
|
||||||
.row__title {
|
.boardCard__title {
|
||||||
font-weight: 800;
|
font-weight: 800;
|
||||||
min-width: 0;
|
min-width: 0;
|
||||||
font-size: 18px;
|
font-size: 18px;
|
||||||
line-height: 1.35;
|
line-height: 1.35;
|
||||||
}
|
}
|
||||||
.row__head {
|
.boardCard__head {
|
||||||
padding: 14px 14px 0;
|
padding: 14px 14px 0;
|
||||||
display: grid;
|
display: grid;
|
||||||
gap: 12px;
|
gap: 12px;
|
||||||
align-content: start;
|
align-content: start;
|
||||||
}
|
}
|
||||||
.row__author {
|
.boardCard__author {
|
||||||
display: inline-flex;
|
display: inline-flex;
|
||||||
gap: 8px;
|
gap: 8px;
|
||||||
align-items: center;
|
align-items: center;
|
||||||
@@ -278,7 +337,7 @@ function submitSearch() {
|
|||||||
opacity: 0.86;
|
opacity: 0.86;
|
||||||
flex: 0 0 auto;
|
flex: 0 0 auto;
|
||||||
}
|
}
|
||||||
.row__avatar {
|
.boardCard__avatar {
|
||||||
width: 28px;
|
width: 28px;
|
||||||
height: 28px;
|
height: 28px;
|
||||||
border-radius: 999px;
|
border-radius: 999px;
|
||||||
@@ -286,17 +345,17 @@ function submitSearch() {
|
|||||||
border: 1px solid rgba(255, 255, 255, 0.12);
|
border: 1px solid rgba(255, 255, 255, 0.12);
|
||||||
background: rgba(255, 255, 255, 0.08);
|
background: rgba(255, 255, 255, 0.08);
|
||||||
}
|
}
|
||||||
.row__avatar--fallback {
|
.boardCard__avatar--fallback {
|
||||||
display: grid;
|
display: grid;
|
||||||
place-items: center;
|
place-items: center;
|
||||||
font-size: 12px;
|
font-size: 12px;
|
||||||
font-weight: 900;
|
font-weight: 900;
|
||||||
}
|
}
|
||||||
.row__meta {
|
.boardCard__meta {
|
||||||
opacity: 0.78;
|
opacity: 0.78;
|
||||||
font-size: 13px;
|
font-size: 13px;
|
||||||
}
|
}
|
||||||
.row__foot {
|
.boardCard__foot {
|
||||||
padding: 0 14px 14px;
|
padding: 0 14px 14px;
|
||||||
display: flex;
|
display: flex;
|
||||||
gap: 12px;
|
gap: 12px;
|
||||||
@@ -323,6 +382,13 @@ function submitSearch() {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
@media (max-width: 720px) {
|
@media (max-width: 720px) {
|
||||||
|
.dashboardHero__right {
|
||||||
|
width: 100%;
|
||||||
|
}
|
||||||
|
.dashboardStat,
|
||||||
|
.primary {
|
||||||
|
width: 100%;
|
||||||
|
}
|
||||||
.list {
|
.list {
|
||||||
grid-template-columns: 1fr;
|
grid-template-columns: 1fr;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -33,6 +33,22 @@ function goFreeform() {
|
|||||||
router.push('/editor/freeform/new')
|
router.push('/editor/freeform/new')
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function goFavorites() {
|
||||||
|
if (!auth.user) {
|
||||||
|
router.push('/login?redirect=/favorites')
|
||||||
|
return
|
||||||
|
}
|
||||||
|
router.push('/favorites')
|
||||||
|
}
|
||||||
|
|
||||||
|
function goMyLists() {
|
||||||
|
if (!auth.user) {
|
||||||
|
router.push('/login?redirect=/me')
|
||||||
|
return
|
||||||
|
}
|
||||||
|
router.push('/me')
|
||||||
|
}
|
||||||
|
|
||||||
function thumbUrl(g) {
|
function thumbUrl(g) {
|
||||||
if (!g.thumbnailSrc) return ''
|
if (!g.thumbnailSrc) return ''
|
||||||
return toApiUrl(g.thumbnailSrc)
|
return toApiUrl(g.thumbnailSrc)
|
||||||
@@ -41,32 +57,47 @@ function thumbUrl(g) {
|
|||||||
</script>
|
</script>
|
||||||
|
|
||||||
<template>
|
<template>
|
||||||
<section class="topBar">
|
<section class="dashboardHero">
|
||||||
<div class="topBar__copy">
|
<div class="dashboardHero__copy">
|
||||||
<h1 class="topBar__title">Main Title</h1>
|
<div class="dashboardHero__eyebrow">Workspace</div>
|
||||||
<p class="topBar__desc">게임 선택과 커스텀 티어표 진입을 하나의 대시보드처럼 정리했습니다.</p>
|
<h1 class="dashboardHero__title">Game Library</h1>
|
||||||
|
<p class="dashboardHero__desc">자주 쓰는 게임 템플릿을 빠르게 고르고, 필요하면 바로 커스텀 티어표를 시작할 수 있어요.</p>
|
||||||
</div>
|
</div>
|
||||||
<div class="toolbar">
|
<div class="dashboardToolbar">
|
||||||
<button class="toolbar__ghost" @click="goFreeform">Toggle Filter</button>
|
<div class="dashboardToolbar__stat">
|
||||||
<button class="toolbar__select" @click="goFreeform">Select Filter</button>
|
<span class="dashboardToolbar__label">Visible Games</span>
|
||||||
|
<strong class="dashboardToolbar__value">{{ games.length }}</strong>
|
||||||
|
</div>
|
||||||
|
<div class="dashboardToolbar__stat">
|
||||||
|
<span class="dashboardToolbar__label">Account</span>
|
||||||
|
<strong class="dashboardToolbar__value">{{ auth.user ? 'Ready' : 'Guest' }}</strong>
|
||||||
|
</div>
|
||||||
|
<button class="dashboardToolbar__ghost" @click="goFavorites">{{ auth.user ? '즐겨찾기 보기' : '로그인 후 즐겨찾기' }}</button>
|
||||||
|
<button class="dashboardToolbar__ghost" @click="goMyLists">{{ auth.user ? '내 리스트 보기' : '로그인 후 내 리스트' }}</button>
|
||||||
<button class="customTierBtn" @click="goFreeform">{{ auth.user ? '+ 커스텀 티어표 만들기' : '+ 로그인 후 커스텀 티어표 만들기' }}</button>
|
<button class="customTierBtn" @click="goFreeform">{{ auth.user ? '+ 커스텀 티어표 만들기' : '+ 로그인 후 커스텀 티어표 만들기' }}</button>
|
||||||
</div>
|
</div>
|
||||||
</section>
|
</section>
|
||||||
|
|
||||||
<div v-if="error" class="error">{{ error }}</div>
|
<div v-if="error" class="error">{{ error }}</div>
|
||||||
<section class="grid">
|
<section class="libraryGrid">
|
||||||
<button v-for="g in games" :key="g.id" class="card" @click="goGame(g.id)">
|
<button v-for="g in games" :key="g.id" class="libraryCard" @click="goGame(g.id)">
|
||||||
<div class="thumbWrap">
|
<div class="libraryCard__thumbWrap">
|
||||||
<img v-if="thumbUrl(g)" class="thumb" :src="thumbUrl(g)" :alt="g.name" />
|
<img v-if="thumbUrl(g)" class="libraryCard__thumb" :src="thumbUrl(g)" :alt="g.name" />
|
||||||
<div v-else class="thumbFallback">{{ g.name[0] }}</div>
|
<div v-else class="libraryCard__thumbFallback">대표 썸네일</div>
|
||||||
|
</div>
|
||||||
|
<div class="libraryCard__body">
|
||||||
|
<div class="libraryCard__title">{{ g.name }}</div>
|
||||||
|
<div class="libraryCard__meta">
|
||||||
|
<span class="libraryCard__metaDot"></span>
|
||||||
|
<span>{{ g.id }}</span>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div class="card__title">{{ g.name }}</div>
|
|
||||||
</button>
|
</button>
|
||||||
</section>
|
</section>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
<style scoped>
|
<style scoped>
|
||||||
.topBar {
|
.dashboardHero {
|
||||||
display: flex;
|
display: flex;
|
||||||
gap: 18px;
|
gap: 18px;
|
||||||
justify-content: space-between;
|
justify-content: space-between;
|
||||||
@@ -74,43 +105,80 @@ function thumbUrl(g) {
|
|||||||
flex-wrap: wrap;
|
flex-wrap: wrap;
|
||||||
margin-top: 2px;
|
margin-top: 2px;
|
||||||
margin-bottom: 18px;
|
margin-bottom: 18px;
|
||||||
|
padding: 6px 2px 18px;
|
||||||
}
|
}
|
||||||
.topBar__copy {
|
.dashboardHero__copy {
|
||||||
display: grid;
|
display: grid;
|
||||||
gap: 8px;
|
gap: 8px;
|
||||||
|
max-width: 720px;
|
||||||
}
|
}
|
||||||
.topBar__title {
|
.dashboardHero__eyebrow {
|
||||||
|
font-size: 11px;
|
||||||
|
letter-spacing: 0.12em;
|
||||||
|
text-transform: uppercase;
|
||||||
|
color: rgba(255, 255, 255, 0.42);
|
||||||
|
}
|
||||||
|
.dashboardHero__title {
|
||||||
margin: 0;
|
margin: 0;
|
||||||
font-size: 32px;
|
font-size: 34px;
|
||||||
letter-spacing: -0.04em;
|
letter-spacing: -0.04em;
|
||||||
color: rgba(255, 255, 255, 0.96);
|
color: rgba(255, 255, 255, 0.96);
|
||||||
}
|
}
|
||||||
.topBar__desc {
|
.dashboardHero__desc {
|
||||||
margin: 0;
|
margin: 0;
|
||||||
color: rgba(255, 255, 255, 0.58);
|
color: rgba(255, 255, 255, 0.58);
|
||||||
line-height: 1.5;
|
line-height: 1.5;
|
||||||
|
max-width: 720px;
|
||||||
}
|
}
|
||||||
.toolbar {
|
.dashboardToolbar {
|
||||||
display: flex;
|
display: flex;
|
||||||
gap: 10px;
|
gap: 10px;
|
||||||
align-items: center;
|
align-items: center;
|
||||||
flex-wrap: wrap;
|
flex-wrap: wrap;
|
||||||
}
|
}
|
||||||
.toolbar__ghost,
|
.dashboardToolbar__stat {
|
||||||
.toolbar__select,
|
display: grid;
|
||||||
|
gap: 2px;
|
||||||
|
min-width: 112px;
|
||||||
|
padding: 12px 14px;
|
||||||
|
border-radius: 14px;
|
||||||
|
border: 1px solid rgba(255, 255, 255, 0.08);
|
||||||
|
background: rgba(255, 255, 255, 0.045);
|
||||||
|
}
|
||||||
|
.dashboardToolbar__label {
|
||||||
|
font-size: 11px;
|
||||||
|
color: rgba(255, 255, 255, 0.48);
|
||||||
|
text-transform: uppercase;
|
||||||
|
letter-spacing: 0.08em;
|
||||||
|
}
|
||||||
|
.dashboardToolbar__value {
|
||||||
|
font-size: 18px;
|
||||||
|
font-weight: 900;
|
||||||
|
}
|
||||||
|
.dashboardToolbar__ghost,
|
||||||
.customTierBtn {
|
.customTierBtn {
|
||||||
padding: 10px 14px;
|
padding: 12px 14px;
|
||||||
border-radius: 10px;
|
border-radius: 14px;
|
||||||
border: 1px solid rgba(255, 255, 255, 0.08);
|
border: 1px solid rgba(255, 255, 255, 0.08);
|
||||||
background: rgba(255, 255, 255, 0.06);
|
background: rgba(255, 255, 255, 0.06);
|
||||||
color: rgba(255, 255, 255, 0.84);
|
color: rgba(255, 255, 255, 0.84);
|
||||||
font-weight: 700;
|
font-weight: 700;
|
||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
|
transition:
|
||||||
|
transform 0.16s ease,
|
||||||
|
border-color 0.16s ease,
|
||||||
|
background 0.16s ease;
|
||||||
|
}
|
||||||
|
.dashboardToolbar__ghost:hover,
|
||||||
|
.customTierBtn:hover {
|
||||||
|
transform: translateY(-1px);
|
||||||
}
|
}
|
||||||
.customTierBtn {
|
.customTierBtn {
|
||||||
background: rgba(255, 255, 255, 0.08);
|
background: rgba(77, 127, 233, 0.88);
|
||||||
|
border-color: rgba(77, 127, 233, 0.96);
|
||||||
|
color: #fff;
|
||||||
}
|
}
|
||||||
.grid {
|
.libraryGrid {
|
||||||
display: grid;
|
display: grid;
|
||||||
grid-template-columns: repeat(3, minmax(0, 1fr));
|
grid-template-columns: repeat(3, minmax(0, 1fr));
|
||||||
gap: 18px;
|
gap: 18px;
|
||||||
@@ -123,69 +191,91 @@ function thumbUrl(g) {
|
|||||||
background: rgba(239, 68, 68, 0.1);
|
background: rgba(239, 68, 68, 0.1);
|
||||||
color: rgba(255, 255, 255, 0.92);
|
color: rgba(255, 255, 255, 0.92);
|
||||||
}
|
}
|
||||||
.card {
|
.libraryCard {
|
||||||
text-align: left;
|
text-align: left;
|
||||||
padding: 12px;
|
padding: 14px;
|
||||||
border-radius: 14px;
|
border-radius: 22px;
|
||||||
border: 1px solid rgba(255, 255, 255, 0.16);
|
border: 1px solid rgba(255, 255, 255, 0.16);
|
||||||
background: rgba(62, 62, 62, 0.82);
|
background: rgba(62, 62, 62, 0.82);
|
||||||
color: rgba(255, 255, 255, 0.92);
|
color: rgba(255, 255, 255, 0.92);
|
||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
display: grid;
|
display: grid;
|
||||||
gap: 12px;
|
gap: 12px;
|
||||||
box-shadow: 0 1px 0 rgba(255, 255, 255, 0.04);
|
box-shadow: inset 0 1px 0 rgba(255, 255, 255, 0.04);
|
||||||
|
transition:
|
||||||
|
transform 0.16s ease,
|
||||||
|
background 0.16s ease;
|
||||||
}
|
}
|
||||||
.card:hover {
|
.libraryCard:hover {
|
||||||
background: rgba(72, 72, 72, 0.92);
|
background: rgba(70, 70, 70, 0.96);
|
||||||
|
transform: translateY(-2px);
|
||||||
}
|
}
|
||||||
.thumbWrap {
|
.libraryCard__thumbWrap {
|
||||||
width: 100%;
|
width: 100%;
|
||||||
aspect-ratio: 16 / 9;
|
aspect-ratio: 16 / 9;
|
||||||
border-radius: 10px;
|
border-radius: 14px;
|
||||||
border: 1px solid rgba(255, 255, 255, 0.06);
|
border: 1px solid rgba(255, 255, 255, 0.06);
|
||||||
background: #555;
|
background: #555;
|
||||||
overflow: hidden;
|
overflow: hidden;
|
||||||
display: grid;
|
display: grid;
|
||||||
place-items: center;
|
place-items: center;
|
||||||
}
|
}
|
||||||
.thumb {
|
.libraryCard__thumb {
|
||||||
width: 100%;
|
width: 100%;
|
||||||
height: 100%;
|
height: 100%;
|
||||||
object-fit: cover;
|
object-fit: cover;
|
||||||
}
|
}
|
||||||
.thumbFallback {
|
.libraryCard__thumbFallback {
|
||||||
|
font-size: 13px;
|
||||||
font-weight: 700;
|
font-weight: 700;
|
||||||
font-size: 14px;
|
font-size: 14px;
|
||||||
color: rgba(255, 255, 255, 0.4);
|
color: rgba(255, 255, 255, 0.4);
|
||||||
}
|
}
|
||||||
.card__title {
|
.libraryCard__body {
|
||||||
|
display: grid;
|
||||||
|
gap: 8px;
|
||||||
|
}
|
||||||
|
.libraryCard__title {
|
||||||
font-weight: 800;
|
font-weight: 800;
|
||||||
letter-spacing: -0.02em;
|
letter-spacing: -0.02em;
|
||||||
font-size: 15px;
|
font-size: 16px;
|
||||||
|
}
|
||||||
|
.libraryCard__meta {
|
||||||
|
display: inline-flex;
|
||||||
|
gap: 8px;
|
||||||
|
align-items: center;
|
||||||
|
color: rgba(255, 255, 255, 0.6);
|
||||||
|
font-size: 13px;
|
||||||
|
}
|
||||||
|
.libraryCard__metaDot {
|
||||||
|
width: 10px;
|
||||||
|
height: 10px;
|
||||||
|
border-radius: 3px;
|
||||||
|
background: rgba(255, 255, 255, 0.9);
|
||||||
}
|
}
|
||||||
@media (max-width: 1200px) {
|
@media (max-width: 1200px) {
|
||||||
.grid {
|
.libraryGrid {
|
||||||
grid-template-columns: repeat(2, minmax(0, 1fr));
|
grid-template-columns: repeat(2, minmax(0, 1fr));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@media (max-width: 900px) {
|
@media (max-width: 900px) {
|
||||||
.grid {
|
.libraryGrid {
|
||||||
grid-template-columns: 1fr;
|
grid-template-columns: 1fr;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@media (max-width: 720px) {
|
@media (max-width: 720px) {
|
||||||
.topBar {
|
.dashboardHero {
|
||||||
align-items: stretch;
|
align-items: stretch;
|
||||||
}
|
}
|
||||||
.toolbar {
|
.dashboardToolbar {
|
||||||
width: 100%;
|
width: 100%;
|
||||||
}
|
}
|
||||||
.toolbar__ghost,
|
.dashboardToolbar__ghost,
|
||||||
.toolbar__select,
|
.dashboardToolbar__stat,
|
||||||
.customTierBtn {
|
.customTierBtn {
|
||||||
flex: 1 1 100%;
|
flex: 1 1 100%;
|
||||||
}
|
}
|
||||||
.grid {
|
.libraryGrid {
|
||||||
grid-template-columns: 1fr;
|
grid-template-columns: 1fr;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -72,25 +72,35 @@ async function removeList(t) {
|
|||||||
|
|
||||||
<template>
|
<template>
|
||||||
<section class="wrap">
|
<section class="wrap">
|
||||||
<h2 class="title">내 티어표</h2>
|
<header class="head">
|
||||||
|
<div>
|
||||||
|
<div class="head__eyebrow">Library</div>
|
||||||
|
<h2 class="title">내 티어표</h2>
|
||||||
|
<div class="desc">직접 저장한 티어표를 같은 카드 레이아웃으로 다시 열고 정리할 수 있어요.</div>
|
||||||
|
</div>
|
||||||
|
<div class="head__stat">
|
||||||
|
<span class="head__statLabel">Saved Lists</span>
|
||||||
|
<strong class="head__statValue">{{ myLists.length }}</strong>
|
||||||
|
</div>
|
||||||
|
</header>
|
||||||
<div class="card">
|
<div class="card">
|
||||||
<div v-if="myLists.length === 0" class="empty">아직 저장한 티어표가 없어요.</div>
|
<div v-if="myLists.length === 0" class="empty">아직 저장한 티어표가 없어요.</div>
|
||||||
<div v-else class="list">
|
<div v-else class="list">
|
||||||
<article v-for="t in myLists" :key="t.id" class="row">
|
<article v-for="t in myLists" :key="t.id" class="boardCard">
|
||||||
<button class="row__body" @click="openList(t)">
|
<button class="boardCard__body" @click="openList(t)">
|
||||||
<div class="row__thumbWrap">
|
<div class="boardCard__thumbWrap">
|
||||||
<img v-if="tierListThumbnailUrl(t)" class="row__thumb" :src="tierListThumbnailUrl(t)" :alt="t.title" />
|
<img v-if="tierListThumbnailUrl(t)" class="boardCard__thumb" :src="tierListThumbnailUrl(t)" :alt="t.title" />
|
||||||
<div v-else class="row__thumbPlaceholder"></div>
|
<div v-else class="boardCard__thumbPlaceholder">대표 썸네일</div>
|
||||||
</div>
|
</div>
|
||||||
<div class="row__head">
|
<div class="boardCard__head">
|
||||||
<div class="row__title">{{ t.title }}</div>
|
<div class="boardCard__title">{{ t.title }}</div>
|
||||||
<div class="row__author">
|
<div class="boardCard__author">
|
||||||
<img v-if="avatarSrcOf(t)" class="row__avatar" :src="avatarSrcOf(t)" :alt="displayNameOf(t)" />
|
<img v-if="avatarSrcOf(t)" class="boardCard__avatar" :src="avatarSrcOf(t)" :alt="displayNameOf(t)" />
|
||||||
<div v-else class="row__avatar row__avatar--fallback">{{ avatarFallbackOf(t) }}</div>
|
<div v-else class="boardCard__avatar boardCard__avatar--fallback">{{ avatarFallbackOf(t) }}</div>
|
||||||
<span>by {{ displayNameOf(t) }}</span>
|
<span>by {{ displayNameOf(t) }}</span>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div class="row__meta">{{ fmt(t.updatedAt) }}</div>
|
<div class="boardCard__meta">{{ fmt(t.updatedAt) }}</div>
|
||||||
</button>
|
</button>
|
||||||
<button class="link link--danger" @click="removeList(t)">삭제</button>
|
<button class="link link--danger" @click="removeList(t)">삭제</button>
|
||||||
</article>
|
</article>
|
||||||
@@ -103,12 +113,49 @@ async function removeList(t) {
|
|||||||
.wrap {
|
.wrap {
|
||||||
padding: 4px 2px;
|
padding: 4px 2px;
|
||||||
}
|
}
|
||||||
|
.head {
|
||||||
|
display: flex;
|
||||||
|
gap: 16px;
|
||||||
|
align-items: flex-start;
|
||||||
|
justify-content: space-between;
|
||||||
|
flex-wrap: wrap;
|
||||||
|
margin-bottom: 18px;
|
||||||
|
padding: 6px 2px 8px;
|
||||||
|
}
|
||||||
|
.head__eyebrow {
|
||||||
|
font-size: 11px;
|
||||||
|
letter-spacing: 0.12em;
|
||||||
|
text-transform: uppercase;
|
||||||
|
color: rgba(255, 255, 255, 0.42);
|
||||||
|
}
|
||||||
.title {
|
.title {
|
||||||
margin: 0 0 18px;
|
margin: 4px 0 6px;
|
||||||
font-size: 30px;
|
font-size: 32px;
|
||||||
letter-spacing: -0.04em;
|
letter-spacing: -0.04em;
|
||||||
color: rgba(255, 255, 255, 0.96);
|
color: rgba(255, 255, 255, 0.96);
|
||||||
}
|
}
|
||||||
|
.desc {
|
||||||
|
color: rgba(255, 255, 255, 0.58);
|
||||||
|
}
|
||||||
|
.head__stat {
|
||||||
|
display: grid;
|
||||||
|
gap: 2px;
|
||||||
|
min-width: 112px;
|
||||||
|
padding: 12px 14px;
|
||||||
|
border-radius: 14px;
|
||||||
|
border: 1px solid rgba(255, 255, 255, 0.08);
|
||||||
|
background: rgba(255, 255, 255, 0.045);
|
||||||
|
}
|
||||||
|
.head__statLabel {
|
||||||
|
font-size: 11px;
|
||||||
|
color: rgba(255, 255, 255, 0.48);
|
||||||
|
text-transform: uppercase;
|
||||||
|
letter-spacing: 0.08em;
|
||||||
|
}
|
||||||
|
.head__statValue {
|
||||||
|
font-size: 18px;
|
||||||
|
font-weight: 900;
|
||||||
|
}
|
||||||
.card {
|
.card {
|
||||||
border: 0;
|
border: 0;
|
||||||
background: transparent;
|
background: transparent;
|
||||||
@@ -116,8 +163,8 @@ async function removeList(t) {
|
|||||||
padding: 0;
|
padding: 0;
|
||||||
}
|
}
|
||||||
.link {
|
.link {
|
||||||
padding: 8px 10px;
|
padding: 10px 12px;
|
||||||
border-radius: 10px;
|
border-radius: 14px;
|
||||||
border: 1px solid rgba(255, 255, 255, 0.14);
|
border: 1px solid rgba(255, 255, 255, 0.14);
|
||||||
background: rgba(255, 255, 255, 0.06);
|
background: rgba(255, 255, 255, 0.06);
|
||||||
color: rgba(255, 255, 255, 0.92);
|
color: rgba(255, 255, 255, 0.92);
|
||||||
@@ -132,16 +179,24 @@ async function removeList(t) {
|
|||||||
grid-template-columns: repeat(3, minmax(0, 1fr));
|
grid-template-columns: repeat(3, minmax(0, 1fr));
|
||||||
gap: 18px;
|
gap: 18px;
|
||||||
}
|
}
|
||||||
.row {
|
.boardCard {
|
||||||
display: grid;
|
display: grid;
|
||||||
gap: 10px;
|
gap: 10px;
|
||||||
border-radius: 14px;
|
border-radius: 22px;
|
||||||
border: 1px solid rgba(255, 255, 255, 0.16);
|
border: 1px solid rgba(255, 255, 255, 0.16);
|
||||||
background: rgba(62, 62, 62, 0.82);
|
background: rgba(62, 62, 62, 0.82);
|
||||||
color: rgba(255, 255, 255, 0.92);
|
color: rgba(255, 255, 255, 0.92);
|
||||||
overflow: hidden;
|
overflow: hidden;
|
||||||
|
box-shadow: inset 0 1px 0 rgba(255, 255, 255, 0.04);
|
||||||
|
transition:
|
||||||
|
transform 0.16s ease,
|
||||||
|
background 0.16s ease;
|
||||||
}
|
}
|
||||||
.row__body {
|
.boardCard:hover {
|
||||||
|
transform: translateY(-2px);
|
||||||
|
background: rgba(70, 70, 70, 0.96);
|
||||||
|
}
|
||||||
|
.boardCard__body {
|
||||||
flex: 1 1 auto;
|
flex: 1 1 auto;
|
||||||
min-width: 0;
|
min-width: 0;
|
||||||
text-align: left;
|
text-align: left;
|
||||||
@@ -153,27 +208,34 @@ async function removeList(t) {
|
|||||||
display: grid;
|
display: grid;
|
||||||
gap: 10px;
|
gap: 10px;
|
||||||
}
|
}
|
||||||
.row__thumbWrap {
|
.boardCard__thumbWrap {
|
||||||
width: 100%;
|
width: 100%;
|
||||||
aspect-ratio: 16 / 9;
|
aspect-ratio: 16 / 9;
|
||||||
background: #555;
|
background: #555;
|
||||||
|
display: grid;
|
||||||
|
place-items: center;
|
||||||
}
|
}
|
||||||
.row__thumb {
|
.boardCard__thumb {
|
||||||
width: 100%;
|
width: 100%;
|
||||||
height: 100%;
|
height: 100%;
|
||||||
object-fit: cover;
|
object-fit: cover;
|
||||||
display: block;
|
display: block;
|
||||||
}
|
}
|
||||||
.row__thumbPlaceholder {
|
.boardCard__thumbPlaceholder {
|
||||||
width: 100%;
|
width: 100%;
|
||||||
height: 100%;
|
height: 100%;
|
||||||
background: #555;
|
background: #555;
|
||||||
|
display: grid;
|
||||||
|
place-items: center;
|
||||||
|
color: rgba(255, 255, 255, 0.4);
|
||||||
|
font-size: 13px;
|
||||||
|
font-weight: 700;
|
||||||
}
|
}
|
||||||
.row__title {
|
.boardCard__title {
|
||||||
font-weight: 900;
|
font-weight: 900;
|
||||||
min-width: 0;
|
min-width: 0;
|
||||||
}
|
}
|
||||||
.row__head {
|
.boardCard__head {
|
||||||
padding: 0 14px;
|
padding: 0 14px;
|
||||||
display: flex;
|
display: flex;
|
||||||
gap: 12px;
|
gap: 12px;
|
||||||
@@ -181,14 +243,14 @@ async function removeList(t) {
|
|||||||
justify-content: space-between;
|
justify-content: space-between;
|
||||||
flex-wrap: wrap;
|
flex-wrap: wrap;
|
||||||
}
|
}
|
||||||
.row__author {
|
.boardCard__author {
|
||||||
display: inline-flex;
|
display: inline-flex;
|
||||||
gap: 8px;
|
gap: 8px;
|
||||||
align-items: center;
|
align-items: center;
|
||||||
font-size: 13px;
|
font-size: 13px;
|
||||||
opacity: 0.84;
|
opacity: 0.84;
|
||||||
}
|
}
|
||||||
.row__avatar {
|
.boardCard__avatar {
|
||||||
width: 28px;
|
width: 28px;
|
||||||
height: 28px;
|
height: 28px;
|
||||||
border-radius: 999px;
|
border-radius: 999px;
|
||||||
@@ -196,13 +258,13 @@ async function removeList(t) {
|
|||||||
border: 1px solid rgba(255, 255, 255, 0.12);
|
border: 1px solid rgba(255, 255, 255, 0.12);
|
||||||
background: rgba(255, 255, 255, 0.08);
|
background: rgba(255, 255, 255, 0.08);
|
||||||
}
|
}
|
||||||
.row__avatar--fallback {
|
.boardCard__avatar--fallback {
|
||||||
display: grid;
|
display: grid;
|
||||||
place-items: center;
|
place-items: center;
|
||||||
font-size: 12px;
|
font-size: 12px;
|
||||||
font-weight: 900;
|
font-weight: 900;
|
||||||
}
|
}
|
||||||
.row__meta {
|
.boardCard__meta {
|
||||||
padding: 0 14px;
|
padding: 0 14px;
|
||||||
margin-top: 6px;
|
margin-top: 6px;
|
||||||
opacity: 0.76;
|
opacity: 0.76;
|
||||||
@@ -224,6 +286,9 @@ async function removeList(t) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
@media (max-width: 720px) {
|
@media (max-width: 720px) {
|
||||||
|
.head__stat {
|
||||||
|
width: 100%;
|
||||||
|
}
|
||||||
.list {
|
.list {
|
||||||
grid-template-columns: 1fr;
|
grid-template-columns: 1fr;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -685,114 +685,98 @@ onUnmounted(() => {
|
|||||||
</div>
|
</div>
|
||||||
</section>
|
</section>
|
||||||
|
|
||||||
<div ref="boardEl" class="board">
|
<div class="editorCanvas">
|
||||||
<div v-if="canEdit && !isExporting" class="boardTools">
|
<div ref="boardEl" class="board">
|
||||||
<div class="boardTools__left">
|
<div v-if="canEdit && !isExporting" class="boardTools">
|
||||||
<button class="btn btn--ghost" @click="addGroup">티어 추가</button>
|
<div class="boardTools__left">
|
||||||
</div>
|
<button class="btn btn--ghost" @click="addGroup">티어 추가</button>
|
||||||
<div class="boardTools__right">
|
|
||||||
<span class="boardTools__label">아이콘 크기</span>
|
|
||||||
<div class="sizePicker">
|
|
||||||
<button
|
|
||||||
v-for="size in iconSizeOptions"
|
|
||||||
:key="size"
|
|
||||||
class="sizePicker__button"
|
|
||||||
:class="{ 'sizePicker__button--active': iconSize === size }"
|
|
||||||
@click="setIconSize(size)"
|
|
||||||
>
|
|
||||||
{{ size }}
|
|
||||||
</button>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<div ref="exportBoardEl" class="exportBoard" :class="{ 'exportBoard--active': isExporting }">
|
|
||||||
<div v-if="isExporting" class="exportBoard__title">{{ effectiveTitle }}</div>
|
|
||||||
<div v-if="isExporting && description" class="exportBoard__description">{{ description }}</div>
|
|
||||||
<div ref="groupListEl" class="rows">
|
|
||||||
<div v-for="g in groups" :key="g.id" class="row">
|
|
||||||
<div class="row__label">
|
|
||||||
<template v-if="isExporting">
|
|
||||||
<div class="row__exportName">{{ g.name }}</div>
|
|
||||||
</template>
|
|
||||||
<template v-else>
|
|
||||||
<span class="grab" title="드래그로 순서 변경" data-group-handle>↕</span>
|
|
||||||
<input v-model="g.name" class="groupName" :readonly="!canEdit" />
|
|
||||||
<button v-if="canEdit" class="rowRemoveBtn" :disabled="groups.length <= 1" @click="removeGroup(g.id)">삭제</button>
|
|
||||||
</template>
|
|
||||||
</div>
|
</div>
|
||||||
<div
|
<div class="boardTools__right">
|
||||||
class="row__drop"
|
<span class="boardTools__label">아이콘 크기</span>
|
||||||
:data-list-type="'group'"
|
<div class="sizePicker">
|
||||||
:data-group-id="g.id"
|
|
||||||
:ref="(el) => setGroupDropEl(g.id, el)"
|
|
||||||
>
|
|
||||||
<div v-if="!isExporting" class="row__empty" v-show="g.itemIds.length === 0">여기로 드래그해서 배치</div>
|
|
||||||
<div v-for="id in g.itemIds" :key="id" class="cell" :data-item-id="id">
|
|
||||||
<img :src="resolveItemSrc(itemsById[id])" class="thumb" :alt="itemsById[id]?.label || id" />
|
|
||||||
<button
|
<button
|
||||||
v-if="canEdit && !isExporting"
|
v-for="size in iconSizeOptions"
|
||||||
class="cellRemoveBtn"
|
:key="size"
|
||||||
type="button"
|
class="sizePicker__button"
|
||||||
title="아이템 빼내기"
|
:class="{ 'sizePicker__button--active': iconSize === size }"
|
||||||
@pointerdown.stop
|
@click="setIconSize(size)"
|
||||||
@click.stop="removeItemFromGroup(g.id, id)"
|
|
||||||
>
|
>
|
||||||
×
|
{{ size }}
|
||||||
</button>
|
</button>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
<div ref="exportBoardEl" class="exportBoard" :class="{ 'exportBoard--active': isExporting }">
|
||||||
|
<div v-if="isExporting" class="exportBoard__title">{{ effectiveTitle }}</div>
|
||||||
|
<div v-if="isExporting && description" class="exportBoard__description">{{ description }}</div>
|
||||||
|
<div ref="groupListEl" class="rows">
|
||||||
|
<div v-for="g in groups" :key="g.id" class="row">
|
||||||
|
<div class="row__label">
|
||||||
|
<template v-if="isExporting">
|
||||||
|
<div class="row__exportName">{{ g.name }}</div>
|
||||||
|
</template>
|
||||||
|
<template v-else>
|
||||||
|
<span class="grab" title="드래그로 순서 변경" data-group-handle>↕</span>
|
||||||
|
<input v-model="g.name" class="groupName" :readonly="!canEdit" />
|
||||||
|
<button v-if="canEdit" class="rowRemoveBtn" :disabled="groups.length <= 1" @click="removeGroup(g.id)">삭제</button>
|
||||||
|
</template>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
class="row__drop"
|
||||||
|
:data-list-type="'group'"
|
||||||
|
:data-group-id="g.id"
|
||||||
|
:ref="(el) => setGroupDropEl(g.id, el)"
|
||||||
|
>
|
||||||
|
<div v-if="!isExporting" class="row__empty" v-show="g.itemIds.length === 0">여기로 드래그해서 배치</div>
|
||||||
|
<div v-for="id in g.itemIds" :key="id" class="cell" :data-item-id="id">
|
||||||
|
<img :src="resolveItemSrc(itemsById[id])" class="thumb" :alt="itemsById[id]?.label || id" />
|
||||||
|
<button
|
||||||
|
v-if="canEdit && !isExporting"
|
||||||
|
class="cellRemoveBtn"
|
||||||
|
type="button"
|
||||||
|
title="아이템 빼내기"
|
||||||
|
@pointerdown.stop
|
||||||
|
@click.stop="removeItemFromGroup(g.id, id)"
|
||||||
|
>
|
||||||
|
×
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div v-if="isExporting" class="exportBoard__footer">
|
||||||
|
<span>{{ effectiveAuthorName }}</span>
|
||||||
|
<span>{{ formatExportDate(fallbackTimestamp) }}</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div v-if="isExporting" class="exportBoard__footer">
|
|
||||||
<span>{{ effectiveAuthorName }}</span>
|
|
||||||
<span>{{ formatExportDate(fallbackTimestamp) }}</span>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="sidebar">
|
<div class="sidebar">
|
||||||
<div class="sidebar__title">아이템</div>
|
<div class="sidebar__title">아이템</div>
|
||||||
<div class="sidebar__hint">
|
<div class="sidebar__hint">
|
||||||
{{ canEdit ? '게임별 기본 이미지와 커스텀 업로드를 여기에 모읍니다.' : '공개 티어표는 보기 전용입니다.' }}
|
{{ canEdit ? '보드 바로 옆에서 드래그해 넣을 수 있도록 아이템 풀을 고정합니다.' : '공개 티어표는 보기 전용입니다.' }}
|
||||||
</div>
|
|
||||||
<div ref="poolEl" class="pool" data-list-type="pool">
|
|
||||||
<div v-for="id in pool" :key="id" class="poolItem" :data-item-id="id">
|
|
||||||
<img :src="resolveItemSrc(itemsById[id])" class="thumb" :alt="itemsById[id]?.label || id" />
|
|
||||||
<div class="poolItem__label">{{ itemsById[id]?.label || id }}</div>
|
|
||||||
</div>
|
</div>
|
||||||
</div>
|
<div ref="poolEl" class="pool" data-list-type="pool">
|
||||||
<div v-if="canEdit && customItems.length" class="customItemEditor">
|
<div v-for="id in pool" :key="id" class="poolItem" :data-item-id="id">
|
||||||
<div class="customItemEditor__title">추가한 커스텀 아이템 이름 정리</div>
|
<img :src="resolveItemSrc(itemsById[id])" class="thumb" :alt="itemsById[id]?.label || id" />
|
||||||
<div class="customItemEditor__desc">
|
<div class="poolItem__label">{{ itemsById[id]?.label || id }}</div>
|
||||||
템플릿 요청 전에 이름을 정리해두면 관리자가 그대로 기본 템플릿으로 반영할 수 있어요.
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div class="customItemEditor__list">
|
<div
|
||||||
<label v-for="item in customItems" :key="item.id" class="customItemEditor__row">
|
v-if="canEdit"
|
||||||
<img class="customItemEditor__thumb" :src="resolveItemSrc(item)" :alt="item.label" />
|
class="dropzone"
|
||||||
<input
|
:class="{ 'dropzone--active': isDragActive }"
|
||||||
class="customItemEditor__input"
|
@dragenter.prevent="onDragEnter"
|
||||||
:value="item.label"
|
@dragover.prevent="onDragEnter"
|
||||||
maxlength="60"
|
@dragleave="onDragLeave"
|
||||||
placeholder="아이템 이름"
|
@drop.prevent="onDropFiles"
|
||||||
@input="updateCustomItemLabel(item.id, $event.target.value)"
|
>
|
||||||
/>
|
<div class="dropzone__title">커스텀 이미지 추가</div>
|
||||||
</label>
|
<div class="dropzone__desc">여러 이미지를 한 번에 드래그하거나 파일 선택으로 추가할 수 있어요.</div>
|
||||||
</div>
|
</div>
|
||||||
|
<input ref="fileEl" type="file" accept="image/*" multiple class="hidden" @change="onFileChange" />
|
||||||
|
<button v-if="canEdit" class="btn btn--ghost" @click="openFile">파일 선택</button>
|
||||||
</div>
|
</div>
|
||||||
<div
|
|
||||||
v-if="canEdit"
|
|
||||||
class="dropzone"
|
|
||||||
:class="{ 'dropzone--active': isDragActive }"
|
|
||||||
@dragenter.prevent="onDragEnter"
|
|
||||||
@dragover.prevent="onDragEnter"
|
|
||||||
@dragleave="onDragLeave"
|
|
||||||
@drop.prevent="onDropFiles"
|
|
||||||
>
|
|
||||||
<div class="dropzone__title">커스텀 이미지 추가</div>
|
|
||||||
<div class="dropzone__desc">여러 이미지를 한 번에 드래그하거나 파일 선택으로 추가할 수 있어요.</div>
|
|
||||||
</div>
|
|
||||||
<input ref="fileEl" type="file" accept="image/*" multiple class="hidden" @change="onFileChange" />
|
|
||||||
<button v-if="canEdit" class="btn btn--ghost" @click="openFile">파일 선택</button>
|
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
@@ -831,6 +815,27 @@ onUnmounted(() => {
|
|||||||
</button>
|
</button>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
<div v-if="canEdit && customItems.length" class="editorSidebar__section">
|
||||||
|
<div class="editorSidebar__label">커스텀 이름 정리</div>
|
||||||
|
<div class="customItemEditor customItemEditor--sidebar">
|
||||||
|
<div class="customItemEditor__desc">
|
||||||
|
아래에서 이름만 정리해두면 관리자 요청 시 그대로 전달됩니다.
|
||||||
|
</div>
|
||||||
|
<div class="customItemEditor__list">
|
||||||
|
<label v-for="item in customItems" :key="item.id" class="customItemEditor__row">
|
||||||
|
<img class="customItemEditor__thumb" :src="resolveItemSrc(item)" :alt="item.label" />
|
||||||
|
<input
|
||||||
|
class="customItemEditor__input"
|
||||||
|
:value="item.label"
|
||||||
|
maxlength="60"
|
||||||
|
placeholder="아이템 이름"
|
||||||
|
@input="updateCustomItemLabel(item.id, $event.target.value)"
|
||||||
|
/>
|
||||||
|
</label>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
<div class="editorSidebar__section editorSidebar__section--footer">
|
<div class="editorSidebar__section editorSidebar__section--footer">
|
||||||
<label class="toggle" :class="{ 'toggle--disabled': !canEdit }">
|
<label class="toggle" :class="{ 'toggle--disabled': !canEdit }">
|
||||||
<input v-model="isPublic" type="checkbox" :disabled="!canEdit" />
|
<input v-model="isPublic" type="checkbox" :disabled="!canEdit" />
|
||||||
@@ -874,6 +879,12 @@ onUnmounted(() => {
|
|||||||
display: grid;
|
display: grid;
|
||||||
gap: 14px;
|
gap: 14px;
|
||||||
}
|
}
|
||||||
|
.editorCanvas {
|
||||||
|
display: grid;
|
||||||
|
grid-template-columns: minmax(0, 1fr) 280px;
|
||||||
|
gap: 16px;
|
||||||
|
align-items: start;
|
||||||
|
}
|
||||||
.editorMain__title {
|
.editorMain__title {
|
||||||
font-size: 28px;
|
font-size: 28px;
|
||||||
font-weight: 900;
|
font-weight: 900;
|
||||||
@@ -1038,10 +1049,11 @@ onUnmounted(() => {
|
|||||||
}
|
}
|
||||||
.board {
|
.board {
|
||||||
border: 1px solid rgba(255, 255, 255, 0.1);
|
border: 1px solid rgba(255, 255, 255, 0.1);
|
||||||
background: rgba(48, 48, 48, 0.78);
|
background: linear-gradient(180deg, rgba(55, 55, 55, 0.86), rgba(42, 42, 42, 0.82));
|
||||||
border-radius: 18px;
|
border-radius: 22px;
|
||||||
padding: 18px;
|
padding: 20px;
|
||||||
align-self: start;
|
align-self: start;
|
||||||
|
box-shadow: inset 0 1px 0 rgba(255, 255, 255, 0.04);
|
||||||
}
|
}
|
||||||
.modalOverlay {
|
.modalOverlay {
|
||||||
position: fixed;
|
position: fixed;
|
||||||
@@ -1124,8 +1136,12 @@ onUnmounted(() => {
|
|||||||
gap: 12px;
|
gap: 12px;
|
||||||
align-items: center;
|
align-items: center;
|
||||||
justify-content: flex-end;
|
justify-content: flex-end;
|
||||||
margin-bottom: 14px;
|
margin-bottom: 16px;
|
||||||
flex-wrap: wrap;
|
flex-wrap: wrap;
|
||||||
|
padding: 10px 12px;
|
||||||
|
border-radius: 16px;
|
||||||
|
border: 1px solid rgba(255, 255, 255, 0.08);
|
||||||
|
background: rgba(255, 255, 255, 0.03);
|
||||||
}
|
}
|
||||||
.boardTools__left,
|
.boardTools__left,
|
||||||
.boardTools__right {
|
.boardTools__right {
|
||||||
@@ -1150,8 +1166,8 @@ onUnmounted(() => {
|
|||||||
.sizePicker__button {
|
.sizePicker__button {
|
||||||
margin: 0;
|
margin: 0;
|
||||||
min-width: 48px;
|
min-width: 48px;
|
||||||
padding: 8px 10px;
|
padding: 9px 10px;
|
||||||
border-radius: 10px;
|
border-radius: 12px;
|
||||||
border: 1px solid rgba(255, 255, 255, 0.14);
|
border: 1px solid rgba(255, 255, 255, 0.14);
|
||||||
background: rgba(255, 255, 255, 0.05);
|
background: rgba(255, 255, 255, 0.05);
|
||||||
color: rgba(255, 255, 255, 0.92);
|
color: rgba(255, 255, 255, 0.92);
|
||||||
@@ -1208,8 +1224,8 @@ onUnmounted(() => {
|
|||||||
align-items: stretch;
|
align-items: stretch;
|
||||||
}
|
}
|
||||||
.row__label {
|
.row__label {
|
||||||
border-radius: 14px;
|
border-radius: 16px;
|
||||||
background: rgba(255, 255, 255, 0.06);
|
background: rgba(255, 255, 255, 0.08);
|
||||||
border: 1px solid rgba(255, 255, 255, 0.12);
|
border: 1px solid rgba(255, 255, 255, 0.12);
|
||||||
display: flex;
|
display: flex;
|
||||||
gap: 8px;
|
gap: 8px;
|
||||||
@@ -1234,7 +1250,7 @@ onUnmounted(() => {
|
|||||||
.groupName {
|
.groupName {
|
||||||
width: 100%;
|
width: 100%;
|
||||||
border: 1px solid rgba(255, 255, 255, 0.14);
|
border: 1px solid rgba(255, 255, 255, 0.14);
|
||||||
background: rgba(0, 0, 0, 0.12);
|
background: rgba(0, 0, 0, 0.18);
|
||||||
color: rgba(255, 255, 255, 0.92);
|
color: rgba(255, 255, 255, 0.92);
|
||||||
border-radius: 10px;
|
border-radius: 10px;
|
||||||
padding: 6px 8px;
|
padding: 6px 8px;
|
||||||
@@ -1264,7 +1280,7 @@ onUnmounted(() => {
|
|||||||
cursor: not-allowed;
|
cursor: not-allowed;
|
||||||
}
|
}
|
||||||
.row__drop {
|
.row__drop {
|
||||||
border-radius: 14px;
|
border-radius: 16px;
|
||||||
background: rgba(0, 0, 0, 0.18);
|
background: rgba(0, 0, 0, 0.18);
|
||||||
border: 1px solid rgba(255, 255, 255, 0.10);
|
border: 1px solid rgba(255, 255, 255, 0.10);
|
||||||
min-height: calc(var(--thumb-size, 80px) + 24px);
|
min-height: calc(var(--thumb-size, 80px) + 24px);
|
||||||
@@ -1322,36 +1338,46 @@ onUnmounted(() => {
|
|||||||
}
|
}
|
||||||
.sidebar {
|
.sidebar {
|
||||||
border: 1px solid rgba(255, 255, 255, 0.1);
|
border: 1px solid rgba(255, 255, 255, 0.1);
|
||||||
background: rgba(48, 48, 48, 0.78);
|
background: linear-gradient(180deg, rgba(52, 52, 52, 0.84), rgba(36, 36, 36, 0.8));
|
||||||
border-radius: 18px;
|
border-radius: 22px;
|
||||||
padding: 12px;
|
padding: 14px;
|
||||||
|
box-shadow: inset 0 1px 0 rgba(255, 255, 255, 0.04);
|
||||||
|
position: sticky;
|
||||||
|
top: 14px;
|
||||||
}
|
}
|
||||||
.editorSidebar {
|
.editorSidebar {
|
||||||
display: grid;
|
display: grid;
|
||||||
align-content: start;
|
align-content: start;
|
||||||
gap: 14px;
|
gap: 14px;
|
||||||
padding: 14px 12px;
|
padding: 14px 12px;
|
||||||
border-radius: 18px;
|
border-radius: 22px;
|
||||||
border: 1px solid rgba(255, 255, 255, 0.08);
|
border: 1px solid rgba(255, 255, 255, 0.08);
|
||||||
background: rgba(18, 18, 18, 0.96);
|
background: linear-gradient(180deg, rgba(17, 17, 17, 0.96), rgba(12, 12, 12, 0.96));
|
||||||
position: sticky;
|
position: sticky;
|
||||||
top: 14px;
|
top: 14px;
|
||||||
|
box-shadow: inset 0 1px 0 rgba(255, 255, 255, 0.04);
|
||||||
}
|
}
|
||||||
.editorSidebar__section {
|
.editorSidebar__section {
|
||||||
display: grid;
|
display: grid;
|
||||||
gap: 10px;
|
gap: 10px;
|
||||||
|
padding: 12px;
|
||||||
|
border-radius: 16px;
|
||||||
|
border: 1px solid rgba(255, 255, 255, 0.06);
|
||||||
|
background: rgba(255, 255, 255, 0.02);
|
||||||
}
|
}
|
||||||
.editorSidebar__label {
|
.editorSidebar__label {
|
||||||
font-size: 13px;
|
font-size: 11px;
|
||||||
font-weight: 800;
|
font-weight: 800;
|
||||||
color: rgba(255, 255, 255, 0.82);
|
color: rgba(255, 255, 255, 0.52);
|
||||||
|
text-transform: uppercase;
|
||||||
|
letter-spacing: 0.12em;
|
||||||
}
|
}
|
||||||
.editorSidebar__input,
|
.editorSidebar__input,
|
||||||
.editorSidebar__textarea {
|
.editorSidebar__textarea {
|
||||||
width: 100%;
|
width: 100%;
|
||||||
border-radius: 12px;
|
border-radius: 14px;
|
||||||
border: 1px solid rgba(255, 255, 255, 0.08);
|
border: 1px solid rgba(255, 255, 255, 0.08);
|
||||||
background: rgba(255, 255, 255, 0.03);
|
background: rgba(255, 255, 255, 0.04);
|
||||||
color: rgba(255, 255, 255, 0.92);
|
color: rgba(255, 255, 255, 0.92);
|
||||||
padding: 11px 12px;
|
padding: 11px 12px;
|
||||||
outline: none;
|
outline: none;
|
||||||
@@ -1371,7 +1397,7 @@ onUnmounted(() => {
|
|||||||
.editorSidebar__thumbFrame {
|
.editorSidebar__thumbFrame {
|
||||||
width: 100%;
|
width: 100%;
|
||||||
aspect-ratio: 16 / 9;
|
aspect-ratio: 16 / 9;
|
||||||
border-radius: 14px;
|
border-radius: 16px;
|
||||||
overflow: hidden;
|
overflow: hidden;
|
||||||
border: 1px solid rgba(255, 255, 255, 0.08);
|
border: 1px solid rgba(255, 255, 255, 0.08);
|
||||||
background: #4c4c4c;
|
background: #4c4c4c;
|
||||||
@@ -1404,17 +1430,16 @@ onUnmounted(() => {
|
|||||||
justify-content: space-between;
|
justify-content: space-between;
|
||||||
gap: 10px;
|
gap: 10px;
|
||||||
width: 100%;
|
width: 100%;
|
||||||
padding: 12px 0 0;
|
padding: 11px 12px;
|
||||||
border: 0;
|
border-radius: 14px;
|
||||||
border-top: 1px solid rgba(255, 255, 255, 0.08);
|
border: 1px solid rgba(255, 255, 255, 0.08);
|
||||||
background: transparent;
|
background: rgba(255, 255, 255, 0.03);
|
||||||
color: rgba(255, 255, 255, 0.9);
|
color: rgba(255, 255, 255, 0.9);
|
||||||
font-weight: 800;
|
font-weight: 800;
|
||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
}
|
}
|
||||||
.editorSidebar__section--footer {
|
.editorSidebar__section--footer {
|
||||||
padding-top: 12px;
|
padding-top: 12px;
|
||||||
border-top: 1px solid rgba(255, 255, 255, 0.08);
|
|
||||||
}
|
}
|
||||||
.editorSidebar__actionGrid {
|
.editorSidebar__actionGrid {
|
||||||
display: grid;
|
display: grid;
|
||||||
@@ -1423,19 +1448,26 @@ onUnmounted(() => {
|
|||||||
}
|
}
|
||||||
.sidebar__title {
|
.sidebar__title {
|
||||||
font-weight: 900;
|
font-weight: 900;
|
||||||
margin-bottom: 6px;
|
margin-bottom: 8px;
|
||||||
|
font-size: 18px;
|
||||||
|
letter-spacing: -0.02em;
|
||||||
}
|
}
|
||||||
.sidebar__hint {
|
.sidebar__hint {
|
||||||
opacity: 0.78;
|
opacity: 0.78;
|
||||||
font-size: 13px;
|
font-size: 13px;
|
||||||
margin-bottom: 10px;
|
margin-bottom: 12px;
|
||||||
|
line-height: 1.5;
|
||||||
}
|
}
|
||||||
.customItemEditor {
|
.customItemEditor {
|
||||||
margin-top: 12px;
|
margin-top: 0;
|
||||||
padding: 12px;
|
padding: 0;
|
||||||
border-radius: 16px;
|
border-radius: 0;
|
||||||
border: 1px solid rgba(255, 255, 255, 0.1);
|
border: 0;
|
||||||
background: rgba(255, 255, 255, 0.03);
|
background: transparent;
|
||||||
|
}
|
||||||
|
.customItemEditor--sidebar {
|
||||||
|
display: grid;
|
||||||
|
gap: 12px;
|
||||||
}
|
}
|
||||||
.customItemEditor__title {
|
.customItemEditor__title {
|
||||||
font-weight: 900;
|
font-weight: 900;
|
||||||
@@ -1480,7 +1512,7 @@ onUnmounted(() => {
|
|||||||
padding: 14px;
|
padding: 14px;
|
||||||
border-radius: 16px;
|
border-radius: 16px;
|
||||||
border: 1px dashed rgba(255, 255, 255, 0.18);
|
border: 1px dashed rgba(255, 255, 255, 0.18);
|
||||||
background: rgba(255, 255, 255, 0.03);
|
background: rgba(255, 255, 255, 0.04);
|
||||||
text-align: center;
|
text-align: center;
|
||||||
}
|
}
|
||||||
.dropzone--active {
|
.dropzone--active {
|
||||||
@@ -1506,9 +1538,9 @@ onUnmounted(() => {
|
|||||||
gap: 10px;
|
gap: 10px;
|
||||||
align-items: center;
|
align-items: center;
|
||||||
padding: 10px;
|
padding: 10px;
|
||||||
border-radius: 14px;
|
border-radius: 16px;
|
||||||
border: 1px solid rgba(255, 255, 255, 0.10);
|
border: 1px solid rgba(255, 255, 255, 0.10);
|
||||||
background: rgba(0, 0, 0, 0.16);
|
background: rgba(0, 0, 0, 0.18);
|
||||||
}
|
}
|
||||||
.poolItem__label {
|
.poolItem__label {
|
||||||
font-weight: 800;
|
font-weight: 800;
|
||||||
@@ -1534,9 +1566,13 @@ onUnmounted(() => {
|
|||||||
.layout {
|
.layout {
|
||||||
grid-template-columns: 1fr;
|
grid-template-columns: 1fr;
|
||||||
}
|
}
|
||||||
|
.editorCanvas {
|
||||||
|
grid-template-columns: 1fr;
|
||||||
|
}
|
||||||
.row {
|
.row {
|
||||||
grid-template-columns: 150px 1fr;
|
grid-template-columns: 150px 1fr;
|
||||||
}
|
}
|
||||||
|
.sidebar,
|
||||||
.editorSidebar {
|
.editorSidebar {
|
||||||
position: static;
|
position: static;
|
||||||
}
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user